limine/src/lib/elf.c

313 lines
7.5 KiB
C
Raw Normal View History

2020-03-25 03:04:18 +03:00
#include <stdint.h>
#include <stddef.h>
#include <lib/blib.h>
#include <lib/libc.h>
#include <lib/elf.h>
2020-05-10 01:38:27 +03:00
#include <lib/print.h>
2020-04-14 06:20:55 +03:00
#include <fs/file.h>
2020-03-25 03:04:18 +03:00
#define PT_LOAD 0x00000001
#define PT_INTERP 0x00000003
#define PT_PHDR 0x00000006
2020-04-18 19:01:29 +03:00
#define ABI_SYSV 0x00
2020-03-25 03:04:18 +03:00
#define ARCH_X86_64 0x3e
2020-04-18 19:01:29 +03:00
#define ARCH_X86_32 0x03
#define BITS_LE 0x01
2020-03-25 03:04:18 +03:00
/* Indices into identification array */
#define EI_CLASS 4
#define EI_DATA 5
#define EI_VERSION 6
#define EI_OSABI 7
2020-03-25 03:04:18 +03:00
2020-04-18 19:01:29 +03:00
struct elf64_hdr {
uint8_t ident[16];
2020-03-25 03:04:18 +03:00
uint16_t type;
uint16_t machine;
uint32_t version;
2020-03-25 03:31:59 +03:00
uint64_t entry;
uint64_t phoff;
uint64_t shoff;
2020-03-25 03:04:18 +03:00
uint32_t flags;
uint16_t hdr_size;
uint16_t phdr_size;
uint16_t ph_num;
uint16_t shdr_size;
uint16_t sh_num;
uint16_t shstrndx;
};
2020-04-18 19:01:29 +03:00
struct elf32_hdr {
uint8_t ident[16];
uint16_t type;
uint16_t machine;
uint32_t version;
uint32_t entry;
uint32_t phoff;
uint32_t shoff;
uint32_t flags;
uint16_t hdr_size;
uint16_t phdr_size;
uint16_t ph_num;
uint16_t shdr_size;
uint16_t sh_num;
uint16_t shstrndx;
};
struct elf64_phdr {
2020-03-25 03:04:18 +03:00
uint32_t p_type;
uint32_t p_flags;
2020-03-25 03:31:59 +03:00
uint64_t p_offset;
uint64_t p_vaddr;
uint64_t p_paddr;
uint64_t p_filesz;
uint64_t p_memsz;
uint64_t p_align;
2020-03-25 03:04:18 +03:00
};
2020-04-18 19:01:29 +03:00
struct elf32_phdr {
uint32_t p_type;
uint32_t p_offset;
uint32_t p_vaddr;
uint32_t p_paddr;
uint32_t p_filesz;
uint32_t p_memsz;
uint32_t p_flags;
uint32_t p_align;
};
struct elf64_shdr {
uint32_t sh_name;
uint32_t sh_type;
uint64_t sh_flags;
uint64_t sh_addr;
uint64_t sh_offset;
uint64_t sh_size;
uint32_t sh_link;
uint32_t sh_info;
uint64_t sh_addralign;
uint64_t sh_entsize;
};
struct elf32_shdr {
uint32_t sh_name;
uint32_t sh_type;
uint32_t sh_flags;
uint32_t sh_addr;
uint32_t sh_offset;
uint32_t sh_size;
uint32_t sh_link;
uint32_t sh_info;
uint32_t sh_addralign;
uint32_t sh_entsize;
};
2020-04-18 19:01:29 +03:00
int elf_bits(struct file_handle *fd) {
struct elf64_hdr hdr;
fread(fd, &hdr, 0, 20);
if (strncmp((char *)hdr.ident, "\177ELF", 4)) {
print("elf: Not a valid ELF file.\n");
return -1;
}
switch (hdr.machine) {
case ARCH_X86_64:
return 64;
case ARCH_X86_32:
return 32;
default:
return -1;
}
}
int elf64_load_section(struct file_handle *fd, void *buffer, const char *name, size_t limit) {
struct elf64_hdr hdr;
fread(fd, &hdr, 0, sizeof(struct elf64_hdr));
2020-03-26 02:46:35 +03:00
if (strncmp((char *)hdr.ident, "\177ELF", 4)) {
print("elf: Not a valid ELF file.\n");
return 1;
}
if (hdr.ident[EI_DATA] != BITS_LE) {
print("elf: Not a Little-endian ELF file.\n");
return 1;
}
if (hdr.machine != ARCH_X86_64) {
print("elf: Not an x86_64 ELF file.\n");
return 1;
}
2020-04-18 19:01:29 +03:00
struct elf64_shdr shstrtab;
fread(fd, &shstrtab, hdr.shoff + hdr.shstrndx * sizeof(struct elf64_shdr),
sizeof(struct elf64_shdr));
2020-03-26 01:57:10 +03:00
char names[shstrtab.sh_size];
2020-04-14 06:20:55 +03:00
fread(fd, names, shstrtab.sh_offset, shstrtab.sh_size);
for (uint16_t i = 0; i < hdr.sh_num; i++) {
2020-04-18 19:01:29 +03:00
struct elf64_shdr section;
fread(fd, &section, hdr.shoff + i * sizeof(struct elf64_shdr),
sizeof(struct elf64_shdr));
2020-03-26 01:57:10 +03:00
if (!strcmp(&names[section.sh_name], name)) {
2020-03-26 02:46:35 +03:00
if (section.sh_size > limit)
return 3;
2020-04-14 06:20:55 +03:00
fread(fd, buffer, section.sh_offset, section.sh_size);
return 0;
}
}
2020-03-26 02:46:35 +03:00
return 2;
}
2020-04-18 19:01:29 +03:00
int elf32_load_section(struct file_handle *fd, void *buffer, const char *name, size_t limit) {
struct elf32_hdr hdr;
fread(fd, &hdr, 0, sizeof(struct elf32_hdr));
if (strncmp((char *)hdr.ident, "\177ELF", 4)) {
print("elf: Not a valid ELF file.\n");
return 1;
}
if (hdr.ident[EI_DATA] != BITS_LE) {
print("elf: Not a Little-endian ELF file.\n");
return 1;
}
if (hdr.machine != ARCH_X86_32) {
print("elf: Not an x86_32 ELF file.\n");
return 1;
}
struct elf32_shdr shstrtab;
fread(fd, &shstrtab, hdr.shoff + hdr.shstrndx * sizeof(struct elf32_shdr),
sizeof(struct elf32_shdr));
char names[shstrtab.sh_size];
fread(fd, names, shstrtab.sh_offset, shstrtab.sh_size);
2020-03-25 03:04:18 +03:00
2020-04-18 19:01:29 +03:00
for (uint16_t i = 0; i < hdr.sh_num; i++) {
struct elf32_shdr section;
fread(fd, &section, hdr.shoff + i * sizeof(struct elf32_shdr),
sizeof(struct elf32_shdr));
if (!strcmp(&names[section.sh_name], name)) {
if (section.sh_size > limit)
return 3;
fread(fd, buffer, section.sh_offset, section.sh_size);
return 0;
}
}
return 2;
}
#define FIXED_HIGHER_HALF_OFFSET_64 ((uint64_t)0xffffffff80000000)
int elf64_load(struct file_handle *fd, uint64_t *entry_point, uint64_t *top) {
struct elf64_hdr hdr;
fread(fd, &hdr, 0, sizeof(struct elf64_hdr));
2020-03-25 03:04:18 +03:00
if (strncmp((char *)hdr.ident, "\177ELF", 4)) {
print("Not a valid ELF file.\n");
2020-03-30 23:27:15 +03:00
return -1;
2020-03-25 03:04:18 +03:00
}
if (hdr.ident[EI_DATA] != BITS_LE) {
print("Not a Little-endian ELF file.\n");
return -1;
}
2020-03-25 03:31:59 +03:00
if (hdr.machine != ARCH_X86_64) {
print("Not an x86_64 ELF file.\n");
return -1;
}
2020-03-30 23:27:15 +03:00
*top = 0;
2020-03-25 03:04:18 +03:00
for (uint16_t i = 0; i < hdr.ph_num; i++) {
2020-04-18 19:01:29 +03:00
struct elf64_phdr phdr;
fread(fd, &phdr, hdr.phoff + i * sizeof(struct elf64_phdr),
sizeof(struct elf64_phdr));
2020-03-25 03:04:18 +03:00
if (phdr.p_type != PT_LOAD)
continue;
2020-04-18 19:01:29 +03:00
if (phdr.p_vaddr & ((uint64_t)1 << 63))
phdr.p_vaddr -= FIXED_HIGHER_HALF_OFFSET_64;
2020-03-30 23:27:15 +03:00
uint64_t this_top = phdr.p_vaddr + phdr.p_memsz;
if (this_top > *top)
*top = this_top;
2020-05-03 00:38:57 +03:00
is_valid_memory_range((size_t)phdr.p_vaddr, (size_t)phdr.p_memsz);
fread(fd, (void *)(uint32_t)phdr.p_vaddr, phdr.p_offset, phdr.p_filesz);
2020-03-25 03:04:18 +03:00
size_t to_zero = (size_t)(phdr.p_memsz - phdr.p_filesz);
if (to_zero) {
2020-03-25 03:31:59 +03:00
void *ptr = (void *)(uint32_t)(phdr.p_vaddr + phdr.p_filesz);
2020-03-25 03:04:18 +03:00
memset(ptr, 0, to_zero);
}
}
*entry_point = hdr.entry;
2020-03-25 07:05:06 +03:00
return 0;
2020-03-25 03:04:18 +03:00
}
2020-04-18 19:01:29 +03:00
int elf32_load(struct file_handle *fd, uint32_t *entry_point, uint32_t *top) {
struct elf32_hdr hdr;
fread(fd, &hdr, 0, sizeof(struct elf32_hdr));
if (strncmp((char *)hdr.ident, "\177ELF", 4)) {
print("Not a valid ELF file.\n");
return -1;
}
if (hdr.ident[EI_DATA] != BITS_LE) {
print("Not a Little-endian ELF file.\n");
return -1;
}
if (hdr.machine != ARCH_X86_32) {
print("Not an x86_32 ELF file.\n");
return -1;
}
*top = 0;
for (uint16_t i = 0; i < hdr.ph_num; i++) {
struct elf32_phdr phdr;
fread(fd, &phdr, hdr.phoff + i * sizeof(struct elf32_phdr),
sizeof(struct elf32_phdr));
if (phdr.p_type != PT_LOAD)
continue;
uint32_t this_top = phdr.p_vaddr + phdr.p_memsz;
if (this_top > *top)
*top = this_top;
2020-05-03 00:38:57 +03:00
is_valid_memory_range((size_t)phdr.p_paddr, (size_t)phdr.p_memsz);
2020-04-18 19:01:29 +03:00
fread(fd, (void *)phdr.p_paddr, phdr.p_offset, phdr.p_filesz);
size_t to_zero = (size_t)(phdr.p_memsz - phdr.p_filesz);
if (to_zero) {
void *ptr = (void *)(phdr.p_vaddr + phdr.p_filesz);
memset(ptr, 0, to_zero);
}
}
*entry_point = hdr.entry;
return 0;
}