#include "ft_re.h" void display_64_section_header(t_elf64 *nm, int i) { ft_printf( " [Nr] Name Type Addr Off Size " " " "ES Flg Lk Inf Al\n"); ft_printf(" [%2d] ", i); ft_printf("%-18s", nm->string_table + nm->shdr[i].sh_name); ft_printf("%-16s", get_section_type(nm->shdr[i].sh_type)); ft_printf("%016lx ", nm->shdr[i].sh_addr); ft_printf("%06lx ", nm->shdr[i].sh_offset); ft_printf("%06lx ", nm->shdr[i].sh_size); ft_printf("%02lx ", nm->shdr[i].sh_entsize); ft_printf("%3s ", get_section_flags(nm->shdr[i].sh_flags)); ft_printf("%02x ", nm->shdr[i].sh_link); ft_printf("%02x ", nm->shdr[i].sh_info); ft_printf("%02lx\n", nm->shdr[i].sh_addralign); } void display_64_header(t_elf64 *nm) { ft_printf("ELF Header:\n"); ft_printf(" Magic: "); for (int i = 0; i < EI_NIDENT; i++)ft_printf("%02x ", nm->ehdr->e_ident[i]); ft_printf("\n Class: ELF64\n"); ft_printf( " Data: 2's complement, little endian\n"); ft_printf(" Version: 1 (current)\n"); ft_printf(" OS/ABI: UNIX - System V\n"); ft_printf(" ABI Version: 0\n"); ft_printf(" Version: 0x%016x\n", nm->ehdr->e_version); ft_printf(" Entry point address: 0x%016lx\n", nm->ehdr->e_entry); ft_printf(" Start of program headers: %ld (bytes into file)\n", nm->ehdr->e_phoff); ft_printf(" Start of section headers: %ld (bytes into file)\n", nm->ehdr->e_shoff); ft_printf(" Flags: 0x%016x\n", nm->ehdr->e_flags); ft_printf(" Size of this header: %d (bytes)\n", nm->ehdr->e_ehsize); ft_printf(" Size of program headers: %d (bytes)\n", nm->ehdr->e_phentsize); ft_printf(" Number of program headers: %d\n", nm->ehdr->e_phnum); ft_printf(" Size of section headers: %d (bytes)\n", nm->ehdr->e_shentsize); ft_printf(" Number of section headers: %d\n", nm->ehdr->e_shnum); ft_printf(" Section header string table index: %d\n", nm->ehdr->e_shstrndx); } int print_64_symtab(t_elf64 *nm) { int sym_num = nm->symtab_end - nm->symtab; ft_printf("Symbol table '%s' contains %d entries:\n", nm->file, sym_num); ft_printf("Contents of the (%s) section: \n", nm->string_table + nm->shdr[nm->ehdr->e_shstrndx].sh_name); ft_printf( "| Num: |Value |Size |Type |Bind |Vis " " |Ndx0x |Ndx " " |shndxty |shndxfl|nmChar |Name\n"); for (int i = 0; i < sym_num; i++) { int id = nm->opts.p == 0 ? nm->ordered_sym_ids[i] : i; ft_printf("| %4d: ", id); ft_printf("|%16lx ", nm->symtab[id].st_value); ft_printf("|%16lx ", nm->symtab[id].st_size); ft_printf("|%-6s ", get_sym_type(ELF64_ST_TYPE(nm->symtab[id].st_info))); ft_printf("|%-6s ", get_sym_bind(ELF64_ST_BIND(nm->symtab[id].st_info))); ft_printf("|%-7s ", get_sym_vis(ELF64_ST_VISIBILITY(nm->symtab[id].st_other))); ft_printf("|%5d", nm->symtab[id].st_shndx); ft_printf("|%7s ", get_sym_shndx(nm->symtab[id].st_shndx)); if (nm->symtab[id].st_shndx < nm->ehdr->e_shnum) { ft_printf("|%8s", get_sh_type(nm->shdr[nm->symtab[id].st_shndx].sh_type)); ft_printf("|%8s", get_sh_flag(nm->shdr[nm->symtab[id].st_shndx].sh_flags)); } else { ft_printf("|%8c", ' '); ft_printf("|%8c", ' '); } ft_printf("|%6c ", nm->sym_types[id]); ft_printf("|%s\n", nm->sym_names[id]); } return 1; } nt parse_64_symtab(t_elf64 *nm) { Elf64_Shdr *shdr = &nm->shdr[nm->ehdr->e_shstrndx]; nm->string_table = (char *)(nm->ptr + shdr->sh_offset); for (int i = 0; i < nm->ehdr->e_shnum; i++) { display_64_section_header(nm, i); if (nm->shdr[i].sh_type == SHT_SYMTAB || (nm->opts.D && nm->shdr[i].sh_type == SHT_DYNSYM)) { nm->symtab = (Elf64_Sym *)(nm->ptr + nm->shdr[i].sh_offset); nm->symtab_end = nm->symtab + (nm->shdr[i].sh_size / sizeof(Elf64_Sym)); int symbol_num = nm->shdr[i].sh_size / nm->shdr[i].sh_entsize; nm->linked_symnames = (char *)(nm->ptr + nm->shdr[nm->shdr[i].sh_link].sh_offset); nm->sym_names = malloc(sizeof(char *) * symbol_num); nm->ordered_sym_ids = malloc(sizeof(int) * symbol_num); nm->sym_types = malloc(sizeof(char) * symbol_num); for (int j = 0; j < symbol_num; j++) { nm->ordered_sym_ids[j] = j; if (ELF64_ST_TYPE(nm->symtab[j].st_info) == STT_SECTION ) nm->sym_names[j] = nm->string_table + nm->shdr[nm->symtab[j].st_shndx].sh_name; else nm->sym_names[j] = nm->linked_symnames + nm->symtab[j].st_name; } print_64_symtab(nm); free(nm->sym_names); free(nm->sym_types); free(nm->ordered_sym_ids); } nm->symtab = NULL; nm->symtab_end = NULL; nm->sym_names = NULL; nm->ordered_sym_ids = NULL; } return (0); } int ft_re64(char *filename, int filesize, char *file, t_opts opts) { t_elf64 nm; nm.file = filename; nm.size = filesize; nm.ptr = file; nm.opts = opts; nm.ehdr = (Elf64_Ehdr *)nm.ptr; nm.shdr = (Elf64_Shdr *)(nm.ptr + nm.ehdr->e_shoff); nm.name = malloc(nm.ehdr->e_shnum * sizeof(char *)); // check if file is complete if (nm.ehdr->e_shoff + nm.ehdr->e_shnum * nm.ehdr->e_shentsize > nm.size) { ft_fprintf(stderr, "bfd plugin: %s: file too short\n", nm.file); return -1; } for (int i = 0; i < nm.ehdr->e_shnum; i++) nm.name[i] = &nm.string_table[nm.shdr[i].sh_name]; display_64_header(&nm); parse_64_symtab(&nm); return 0; }