123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291 |
- /*
- * Copyright (c) 2021, the SerenityOS developers.
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #include <LibC/elf.h>
- #include <LibCore/File.h>
- #include <LibTest/TestCase.h>
- #include <fcntl.h>
- #include <stdio.h>
- #include <string.h>
- #include <unistd.h>
- TEST_CASE(test_interp_header_tiny_p_filesz)
- {
- char buffer[0x2000];
- auto& header = *(Elf32_Ehdr*)buffer;
- header.e_ident[EI_MAG0] = ELFMAG0;
- header.e_ident[EI_MAG1] = ELFMAG1;
- header.e_ident[EI_MAG2] = ELFMAG2;
- header.e_ident[EI_MAG3] = ELFMAG3;
- header.e_ident[EI_CLASS] = ELFCLASS32;
- header.e_ident[EI_DATA] = ELFDATA2LSB;
- header.e_ident[EI_VERSION] = EV_CURRENT;
- header.e_ident[EI_OSABI] = ELFOSABI_SYSV;
- header.e_ident[EI_ABIVERSION] = 0;
- header.e_type = ET_REL;
- header.e_version = EV_CURRENT;
- header.e_ehsize = sizeof(Elf32_Ehdr);
- header.e_machine = EM_386;
- header.e_shentsize = sizeof(Elf32_Shdr);
- header.e_phnum = 1;
- header.e_phoff = 52; // inaccurate
- header.e_phentsize = sizeof(Elf32_Phdr); // inaccurate
- header.e_shnum = 3; // inaccurate
- header.e_shoff = 1024; // inaccurate
- header.e_shstrndx = 2; // inaccurate
- header.e_entry = 1024; // inaccurate
- auto* ph = (Elf32_Phdr*)(&buffer[header.e_phoff]);
- ph[0].p_flags = PF_R | PF_X;
- ph[0].p_vaddr = 0x00d4;
- ph[0].p_align = PAGE_SIZE;
- ph[0].p_type = PT_INTERP;
- ph[0].p_memsz = 0xffff0000;
- ph[0].p_offset = 0x100;
- // p_filesz (1 or less) to trigger crash
- ph[0].p_filesz = 1;
- char path[] = "/tmp/test-elf.XXXXXX";
- auto fd = mkstemp(path);
- EXPECT_NE(fd, -1);
- EXPECT_EQ(fchmod(fd, 0700), 0);
- int nwritten = write(fd, buffer, sizeof(buffer));
- EXPECT(nwritten);
- auto elf_path = Core::File::read_link(String::formatted("/proc/{}/fd/{}", getpid(), fd));
- EXPECT(elf_path.characters());
- int rc = execl(elf_path.characters(), "test-elf", nullptr);
- EXPECT_EQ(rc, -1);
- EXPECT_EQ(errno, 8);
- EXPECT_EQ(unlink(path), 0);
- }
- TEST_CASE(test_interp_header_p_filesz_larger_than_p_memsz)
- {
- char buffer[0x2000];
- auto& header = *(Elf32_Ehdr*)buffer;
- header.e_ident[EI_MAG0] = ELFMAG0;
- header.e_ident[EI_MAG1] = ELFMAG1;
- header.e_ident[EI_MAG2] = ELFMAG2;
- header.e_ident[EI_MAG3] = ELFMAG3;
- header.e_ident[EI_CLASS] = ELFCLASS32;
- header.e_ident[EI_DATA] = ELFDATA2LSB;
- header.e_ident[EI_VERSION] = EV_CURRENT;
- header.e_ident[EI_OSABI] = ELFOSABI_SYSV;
- header.e_ident[EI_ABIVERSION] = 0;
- header.e_type = ET_REL;
- header.e_version = EV_CURRENT;
- header.e_ehsize = sizeof(Elf32_Ehdr);
- header.e_machine = EM_386;
- header.e_shentsize = sizeof(Elf32_Shdr);
- header.e_phnum = 1;
- header.e_phoff = 52; // inaccurate
- header.e_phentsize = sizeof(Elf32_Phdr); // inaccurate
- header.e_shnum = 3; // inaccurate
- header.e_shoff = 1024; // inaccurate
- header.e_shstrndx = 2; // inaccurate
- header.e_entry = 1024; // inaccurate
- auto* ph = (Elf32_Phdr*)(&buffer[header.e_phoff]);
- ph[0].p_flags = PF_R | PF_X;
- ph[0].p_vaddr = 0x00d4;
- ph[0].p_align = PAGE_SIZE;
- ph[0].p_type = PT_INTERP;
- ph[0].p_memsz = 0xffff0000;
- ph[0].p_offset = 0x1000;
- ph[0].p_filesz = 0x1000;
- char path[] = "/tmp/test-elf.XXXXXX";
- auto fd = mkstemp(path);
- EXPECT_NE(fd, -1);
- EXPECT_EQ(fchmod(fd, 0700), 0);
- int nwritten = write(fd, buffer, sizeof(buffer));
- EXPECT(nwritten);
- auto elf_path = Core::File::read_link(String::formatted("/proc/{}/fd/{}", getpid(), fd));
- EXPECT(elf_path.characters());
- int rc = execl(elf_path.characters(), "test-elf", nullptr);
- EXPECT_EQ(rc, -1);
- EXPECT_EQ(errno, 8);
- EXPECT_EQ(unlink(path), 0);
- }
- TEST_CASE(test_interp_header_p_filesz_plus_p_offset_overflow_p_memsz)
- {
- char buffer[0x2000];
- auto& header = *(Elf32_Ehdr*)buffer;
- header.e_ident[EI_MAG0] = ELFMAG0;
- header.e_ident[EI_MAG1] = ELFMAG1;
- header.e_ident[EI_MAG2] = ELFMAG2;
- header.e_ident[EI_MAG3] = ELFMAG3;
- header.e_ident[EI_CLASS] = ELFCLASS32;
- header.e_ident[EI_DATA] = ELFDATA2LSB;
- header.e_ident[EI_VERSION] = EV_CURRENT;
- header.e_ident[EI_OSABI] = ELFOSABI_SYSV;
- header.e_ident[EI_ABIVERSION] = 0;
- header.e_type = ET_REL;
- header.e_version = EV_CURRENT;
- header.e_ehsize = sizeof(Elf32_Ehdr);
- header.e_machine = EM_386;
- header.e_shentsize = sizeof(Elf32_Shdr);
- header.e_phoff = 52; // inaccurate
- header.e_phentsize = sizeof(Elf32_Phdr); // inaccurate
- header.e_shnum = 3; // inaccurate
- header.e_shoff = 1024; // inaccurate
- header.e_shstrndx = 2; // inaccurate
- header.e_entry = 1024; // inaccurate
- auto* ph = (Elf32_Phdr*)(&buffer[header.e_phoff]);
- ph[0].p_flags = PF_R | PF_X;
- ph[0].p_vaddr = 0x00d4;
- ph[0].p_align = PAGE_SIZE;
- ph[0].p_type = PT_INTERP;
- // p_memsz must be of sufficient size to hold maxint - 0x1000
- ph[0].p_memsz = 0xfffff000;
- // p_offset + p_filesz must not exceed buffer size in order to pass buffer size check in ELF::validate_program_headers().
- // p_memsz + p_offset must be sufficiently large to overflow maxint.
- ph[0].p_offset = 0x1234;
- ph[0].p_filesz = -0x1000;
- char path[] = "/tmp/test-elf.XXXXXX";
- auto fd = mkstemp(path);
- EXPECT_NE(fd, -1);
- EXPECT_EQ(fchmod(fd, 0700), 0);
- int nwritten = write(fd, buffer, sizeof(buffer));
- EXPECT(nwritten);
- auto elf_path = Core::File::read_link(String::formatted("/proc/{}/fd/{}", getpid(), fd));
- EXPECT(elf_path.characters());
- int rc = execl(elf_path.characters(), "test-elf", nullptr);
- EXPECT_EQ(rc, -1);
- EXPECT_EQ(errno, 8);
- EXPECT_EQ(unlink(path), 0);
- }
- TEST_CASE(test_load_header_p_memsz_zero)
- {
- char buffer[0x2000];
- auto& header = *(Elf32_Ehdr*)buffer;
- header.e_ident[EI_MAG0] = ELFMAG0;
- header.e_ident[EI_MAG1] = ELFMAG1;
- header.e_ident[EI_MAG2] = ELFMAG2;
- header.e_ident[EI_MAG3] = ELFMAG3;
- header.e_ident[EI_CLASS] = ELFCLASS32;
- header.e_ident[EI_DATA] = ELFDATA2LSB;
- header.e_ident[EI_VERSION] = EV_CURRENT;
- header.e_ident[EI_OSABI] = ELFOSABI_SYSV;
- header.e_ident[EI_ABIVERSION] = 0;
- header.e_type = ET_REL;
- header.e_version = EV_CURRENT;
- header.e_ehsize = sizeof(Elf32_Ehdr);
- header.e_machine = EM_386;
- header.e_shentsize = sizeof(Elf32_Shdr);
- header.e_phoff = 52; // inaccurate
- header.e_phentsize = sizeof(Elf32_Phdr); // inaccurate
- header.e_shnum = 3; // inaccurate
- header.e_shoff = 1024; // inaccurate
- header.e_shstrndx = 2; // inaccurate
- header.e_entry = 1024; // inaccurate
- auto* ph = (Elf32_Phdr*)(&buffer[header.e_phoff]);
- ph[0].p_flags = PF_R | PF_X;
- ph[0].p_vaddr = 0x00d4;
- ph[0].p_align = PAGE_SIZE;
- ph[0].p_type = PT_LOAD;
- ph[0].p_offset = 0;
- ph[0].p_filesz = 0;
- // p_memsz zero to trigger crash
- ph[0].p_memsz = 0;
- char path[] = "/tmp/test-elf.XXXXXX";
- auto fd = mkstemp(path);
- EXPECT_NE(fd, -1);
- EXPECT_EQ(fchmod(fd, 0700), 0);
- int nwritten = write(fd, buffer, sizeof(buffer));
- EXPECT(nwritten);
- auto elf_path = Core::File::read_link(String::formatted("/proc/{}/fd/{}", getpid(), fd));
- EXPECT(elf_path.characters());
- int rc = execl(elf_path.characters(), "test-elf", nullptr);
- EXPECT_EQ(rc, -1);
- EXPECT_EQ(errno, 8);
- EXPECT_EQ(unlink(path), 0);
- }
- TEST_CASE(test_load_header_p_memsz_not_equal_to_p_align)
- {
- char buffer[0x2000];
- auto& header = *(Elf32_Ehdr*)buffer;
- header.e_ident[EI_MAG0] = ELFMAG0;
- header.e_ident[EI_MAG1] = ELFMAG1;
- header.e_ident[EI_MAG2] = ELFMAG2;
- header.e_ident[EI_MAG3] = ELFMAG3;
- header.e_ident[EI_CLASS] = ELFCLASS32;
- header.e_ident[EI_DATA] = ELFDATA2LSB;
- header.e_ident[EI_VERSION] = EV_CURRENT;
- header.e_ident[EI_OSABI] = ELFOSABI_SYSV;
- header.e_ident[EI_ABIVERSION] = 0;
- header.e_type = ET_REL;
- header.e_version = EV_CURRENT;
- header.e_ehsize = sizeof(Elf32_Ehdr);
- header.e_machine = EM_386;
- header.e_shentsize = sizeof(Elf32_Shdr);
- header.e_phoff = 52; // inaccurate
- header.e_phentsize = sizeof(Elf32_Phdr); // inaccurate
- header.e_shnum = 3; // inaccurate
- header.e_shoff = 1024; // inaccurate
- header.e_shstrndx = 2; // inaccurate
- header.e_entry = 1024; // inaccurate
- auto* ph = (Elf32_Phdr*)(&buffer[header.e_phoff]);
- ph[0].p_flags = PF_R | PF_X;
- ph[0].p_vaddr = 0x00d4;
- ph[0].p_type = PT_LOAD;
- ph[0].p_memsz = 0xffff0000;
- ph[0].p_offset = 0x1000;
- ph[0].p_filesz = 0x1000;
- // p_align not equal to PAGE_SIZE to trigger crash
- ph[0].p_align = PAGE_SIZE / 2;
- char path[] = "/tmp/test-elf.XXXXXX";
- auto fd = mkstemp(path);
- EXPECT_NE(fd, -1);
- EXPECT_EQ(fchmod(fd, 0700), 0);
- int nwritten = write(fd, buffer, sizeof(buffer));
- EXPECT(nwritten);
- auto elf_path = Core::File::read_link(String::formatted("/proc/{}/fd/{}", getpid(), fd));
- EXPECT(elf_path.characters());
- int rc = execl(elf_path.characters(), "test-elf", nullptr);
- EXPECT_EQ(rc, -1);
- EXPECT_EQ(errno, 8);
- EXPECT_EQ(unlink(path), 0);
- }
|