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 = &sections[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 }