1 /** 2 * Compiler implementation of the 3 * $(LINK2 http://www.dlang.org, D programming language). 4 * 5 * Copyright: Copyright (c) 1999-2016 by Digital Mars, All Rights Reserved 6 * Authors: $(LINK2 http://www.digitalmars.com, Walter Bright) 7 * License: $(LINK2 http://www.boost.org/LICENSE_1_0.txt, Boost License 1.0) 8 * Source: $(DMDSRC _scanelf.d) 9 */ 10 11 module ddmd.scanelf; 12 13 version (linux) 14 import core.sys.linux.elf; 15 else version (FreeBSD) 16 import core.sys.freebsd.sys.elf; 17 else version (Solaris) 18 import core.sys.solaris.elf; 19 20 import core.stdc..string; 21 import core.checkedint; 22 23 import ddmd.globals; 24 import ddmd.errors; 25 26 enum LOG = false; 27 28 /***************************************** 29 * Reads an object module from base[] and passes the names 30 * of any exported symbols to (*pAddSymbol)(). 31 * Params: 32 * pAddSymbol = function to pass the names to 33 * base = array of contents of object module 34 * module_name = name of the object module (used for error messages) 35 * loc = location to use for error printing 36 */ 37 void scanElfObjModule(void delegate(const(char)[] name, int pickAny) pAddSymbol, 38 const(ubyte)[] base, const(char)* module_name, Loc loc) 39 { 40 static if (LOG) 41 { 42 printf("scanElfObjModule(%s)\n", module_name); 43 } 44 45 void corrupt(int reason) 46 { 47 error(loc, "corrupt ELF object module %s %d", module_name, reason); 48 } 49 50 if (base.length < Elf32_Ehdr.sizeof) 51 return corrupt(__LINE__); // must be at least large enough for ELF32 52 static immutable ubyte[4] elf = [0x7F, 'E', 'L', 'F']; // ELF file signature 53 if (base[0 .. elf.length] != elf[]) 54 return corrupt(__LINE__); 55 56 if (base[EI_VERSION] != EV_CURRENT) 57 { 58 return error(loc, "ELF object module %s has EI_VERSION = %d, should be %d", 59 module_name, base[EI_VERSION], EV_CURRENT); 60 } 61 if (base[EI_DATA] != ELFDATA2LSB) 62 { 63 return error(loc, "ELF object module %s is byte swapped and unsupported", module_name); 64 } 65 if (base[EI_CLASS] != ELFCLASS32 && base[EI_CLASS] != ELFCLASS64) 66 { 67 return error(loc, "ELF object module %s is unrecognized class %d", module_name, base[EI_CLASS]); 68 } 69 70 void scanELF(uint model)() 71 { 72 static if (model == 32) 73 { 74 alias ElfXX_Ehdr = Elf32_Ehdr; 75 alias ElfXX_Shdr = Elf32_Shdr; 76 alias ElfXX_Sym = Elf32_Sym; 77 } 78 else 79 { 80 static assert(model == 64); 81 alias ElfXX_Ehdr = Elf64_Ehdr; 82 alias ElfXX_Shdr = Elf64_Shdr; 83 alias ElfXX_Sym = Elf64_Sym; 84 } 85 86 if (base.length < ElfXX_Ehdr.sizeof) 87 return corrupt(__LINE__); 88 89 const eh = cast(const(ElfXX_Ehdr)*) base.ptr; 90 if (eh.e_type != ET_REL) 91 return error(loc, "ELF object module %s is not relocatable", module_name); 92 if (eh.e_version != EV_CURRENT) 93 return corrupt(__LINE__); 94 95 bool overflow; 96 const end = addu(eh.e_shoff, mulu(eh.e_shentsize, eh.e_shnum, overflow), overflow); 97 if (overflow || end > base.length) 98 return corrupt(__LINE__); 99 100 /* For each Section 101 */ 102 const sections = (cast(const(ElfXX_Shdr)*)(base.ptr + eh.e_shoff))[0 .. eh.e_shnum]; 103 foreach (ref const section; sections) 104 { 105 if (section.sh_type != SHT_SYMTAB) 106 continue; 107 108 bool checkShdrXX(const ref ElfXX_Shdr shdr) 109 { 110 bool overflow; 111 return addu(shdr.sh_offset, shdr.sh_size, overflow) > base.length || overflow; 112 } 113 114 if (checkShdrXX(section)) 115 return corrupt(__LINE__); 116 117 /* sh_link gives the particular string table section 118 * used for the symbol names. 119 */ 120 if (section.sh_link >= eh.e_shnum) 121 return corrupt(__LINE__); 122 123 const string_section = §ions[section.sh_link]; 124 if (string_section.sh_type != SHT_STRTAB) 125 return corrupt(__LINE__); 126 127 if (checkShdrXX(*string_section)) 128 return corrupt(__LINE__); 129 130 const string_tab = (cast(const(char)[])base) 131 [cast(size_t)string_section.sh_offset .. 132 cast(size_t)(string_section.sh_offset + string_section.sh_size)]; 133 134 /* Get the array of symbols this section refers to 135 */ 136 const symbols = (cast(ElfXX_Sym*)(base.ptr + cast(size_t)section.sh_offset)) 137 [0 .. cast(size_t)(section.sh_size / ElfXX_Sym.sizeof)]; 138 139 foreach (ref const sym; symbols) 140 { 141 const stb = sym.st_info >> 4; 142 if (stb != STB_GLOBAL && stb != STB_WEAK || sym.st_shndx == SHN_UNDEF) 143 continue; // it's extern 144 145 if (sym.st_name >= string_tab.length) 146 return corrupt(__LINE__); 147 148 const name = &string_tab[sym.st_name]; 149 //printf("sym st_name = x%x\n", sym.st_name); 150 const pend = memchr(name, 0, string_tab.length - sym.st_name); 151 if (!pend) // if didn't find terminating 0 inside the string section 152 return corrupt(__LINE__); 153 pAddSymbol(name[0 .. pend - name], 1); 154 } 155 } 156 } 157 158 if (base[EI_CLASS] == ELFCLASS32) 159 { 160 scanELF!32; 161 } 162 else 163 { 164 assert(base[EI_CLASS] == ELFCLASS64); 165 scanELF!64; 166 } 167 }