1 /*
2  * Copyright 2011, The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *     http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #ifndef ELF_SYMBOL_HXX
18 #define ELF_SYMBOL_HXX
19 
20 #include "ELFSectionHeaderTable.h"
21 #include "ELFSection.h"
22 #include "ELFSectionStrTab.h"
23 
24 #include "ELFObject.h"
25 #include "ELFSectionHeaderTable.h"
26 #include "ELFSectionProgBits.h"
27 #include "ELFSectionNoBits.h"
28 
29 #include "utils/rsl_assert.h"
30 #include "ELF.h"
31 
32 template <unsigned Bitwidth>
getName() const33 inline char const *ELFSymbol_CRTP<Bitwidth>::getName() const {
34   ELFSectionHeaderTableTy const &shtab = *owner->getSectionHeaderTable();
35   size_t const index = shtab.getByName(std::string(".strtab"))->getIndex();
36   ELFSectionTy const *section = owner->getSectionByIndex(index);
37   ELFSectionStrTabTy const &strtab =
38     *static_cast<ELFSectionStrTabTy const *>(section);
39   return strtab[getNameIndex()];
40 }
41 
42 template <unsigned Bitwidth>
43 template <typename Archiver>
44 inline ELFSymbol<Bitwidth> *
read(Archiver & AR,ELFObjectTy const * owner,size_t index)45 ELFSymbol_CRTP<Bitwidth>::read(Archiver &AR,
46                                ELFObjectTy const *owner,
47                                size_t index) {
48   if (!AR) {
49     // Archiver is in bad state before calling read function.
50     // Return NULL and do nothing.
51     return 0;
52   }
53 
54   std::unique_ptr<ELFSymbolTy> sh(new ELFSymbolTy());
55 
56   if (!sh->serialize(AR)) {
57     // Unable to read the structure.  Return NULL.
58     return 0;
59   }
60 
61   if (!sh->isValid()) {
62     // SymTabEntry read from archiver is not valid.  Return NULL.
63     return 0;
64   }
65 
66   // Set the section header index
67   sh->index = index;
68 
69   // Set the owner elf object
70   sh->owner = owner;
71 
72   return sh.release();
73 }
74 
75 template <unsigned Bitwidth>
print(bool shouldPrintHeader) const76 inline void ELFSymbol_CRTP<Bitwidth>::print(bool shouldPrintHeader) const {
77   using namespace llvm;
78 
79   if (shouldPrintHeader) {
80     out() << '\n' << fillformat('=', 79) << '\n';
81     out().changeColor(raw_ostream::WHITE, true);
82     out() << "ELF Symbol Table Entry "
83           << this->getIndex() << '\n';
84     out().resetColor();
85     out() << fillformat('-', 79) << '\n';
86   } else {
87     out() << fillformat('-', 79) << '\n';
88     out().changeColor(raw_ostream::YELLOW, true);
89     out() << "ELF Symbol Table Entry "
90           << this->getIndex() << " : " << '\n';
91     out().resetColor();
92   }
93 
94 #define PRINT_LINT(title, value) \
95   out() << format("  %-11s : ", (char const *)(title)) << (value) << '\n'
96   PRINT_LINT("Name",        getName()                                    );
97   PRINT_LINT("Type",        getTypeStr(getType())                        );
98   PRINT_LINT("Bind",        getBindingAttributeStr(getBindingAttribute()));
99   PRINT_LINT("Visibility",  getVisibilityStr(getVisibility())            );
100   PRINT_LINT("Shtab Index", getSectionIndex()                            );
101   PRINT_LINT("Value",       getValue()                                   );
102   PRINT_LINT("Size",        getSize()                                    );
103 #undef PRINT_LINT
104 
105 // TODO: Horizontal type or vertical type can use option to decide.
106 #if 0
107   using namespace term::color;
108   using namespace std;
109 
110   cout << setw(20) << getName() <<
111           setw(10) << getTypeStr(getType()) <<
112           setw(10) << getBindingAttributeStr(getBindingAttribute()) <<
113           setw(15) << getVisibilityStr(getVisibility()) <<
114           setw(10) << getSectionIndex() <<
115           setw(7) << getValue() <<
116           setw(7) << getSize() <<
117           endl;
118 #endif
119 }
120 
121 template <unsigned Bitwidth>
getAddress(int machine,bool autoAlloc) const122 void *ELFSymbol_CRTP<Bitwidth>::getAddress(int machine, bool autoAlloc) const {
123   if (my_addr != 0) {
124     return my_addr;
125   }
126   size_t idx = (size_t)getSectionIndex();
127   switch (getType()) {
128     default:
129       break;
130 
131     case STT_OBJECT:
132       switch (idx) {
133         default:
134           {
135             ELFSectionHeaderTableTy const *header =
136               owner->getSectionHeaderTable();
137 
138             unsigned section_type = (*header)[idx]->getType();
139 
140             rsl_assert((section_type == SHT_PROGBITS ||
141                         section_type == SHT_NOBITS) &&
142                        "STT_OBJECT with not BITS section.");
143 
144             if (section_type == SHT_NOBITS) {
145               // FIXME(logan): This is a workaround for .lcomm directives
146               // bug of LLVM ARM MC code generator.  Remove this when the
147               // LLVM bug is fixed.
148 
149               size_t align = 16;
150 
151               my_addr = const_cast<ELFObjectTy *>(owner)->
152                 allocateSHNCommonData((size_t)getSize(), align);
153 
154               if (!my_addr) {
155                 rsl_assert(0 && "Unable to allocate memory for SHN_COMMON.");
156                 abort();
157               }
158             } else {
159               ELFSectionTy const *sec = owner->getSectionByIndex(idx);
160               rsl_assert(sec != 0 && "STT_OBJECT with null section.");
161 
162               ELFSectionBitsTy const &st =
163                 static_cast<ELFSectionBitsTy const &>(*sec);
164               my_addr =const_cast<unsigned char *>(&st[0] + (off_t)getValue());
165             }
166           }
167           break;
168 
169         case SHN_COMMON:
170           {
171             if (!autoAlloc) {
172               return NULL;
173             }
174 #if 0
175 #if _POSIX_C_SOURCE >= 200112L || _XOPEN_SOURCE >= 600
176             if (posix_memalign(&my_addr,
177                                std::max((size_t)getValue(), sizeof(void*)),
178                                (size_t)getSize()) != 0) {
179               rsl_assert(0 && "posix_memalign failed.");
180             }
181 #else
182             my_addr = memalign(std::max((size_t)getValue(), sizeof(void *)),
183                                (size_t)getSize());
184 
185             rsl_assert(my_addr != NULL && "memalign failed.");
186 #endif
187             if (my_addr) {
188               memset(my_addr, '\0', getSize());
189             }
190 #else
191             size_t align = (size_t)getValue();
192             my_addr = const_cast<ELFObjectTy *>(owner)->
193                           allocateSHNCommonData((size_t)getSize(), align);
194             if (!my_addr) {
195               rsl_assert(0 && "Unable to allocate memory for SHN_COMMON.");
196               abort();
197             }
198 #endif
199           }
200           break;
201 
202         case SHN_UNDEF:
203           if (machine == EM_MIPS && strcmp(getName(), "_gp_disp") == 0) // OK for MIPS
204             break;
205 
206         case SHN_ABS:
207         case SHN_XINDEX:
208           rsl_assert(0 && "STT_OBJECT with special st_shndx.");
209           break;
210       }
211       break;
212 
213 
214     case STT_FUNC:
215       switch (idx) {
216         default:
217           {
218 #ifndef NDEBUG
219             ELFSectionHeaderTableTy const *header =
220               owner->getSectionHeaderTable();
221             rsl_assert((*header)[idx]->getType() == SHT_PROGBITS &&
222                    "STT_FUNC with not PROGBITS section.");
223 #endif
224             ELFSectionTy const *sec = owner->getSectionByIndex(idx);
225             rsl_assert(sec != 0 && "STT_FUNC with null section.");
226 
227             ELFSectionProgBitsTy const &st =
228               static_cast<ELFSectionProgBitsTy const &>(*sec);
229             my_addr = const_cast<unsigned char *>(&st[0] + (off_t)getValue());
230           }
231           break;
232 
233         case SHN_ABS:
234         case SHN_COMMON:
235         case SHN_UNDEF:
236         case SHN_XINDEX:
237           rsl_assert(0 && "STT_FUNC with special st_shndx.");
238           break;
239       }
240       break;
241 
242 
243     case STT_SECTION:
244       switch (idx) {
245         default:
246           {
247 #ifndef NDEBUG
248             ELFSectionHeaderTableTy const *header =
249               owner->getSectionHeaderTable();
250             rsl_assert(((*header)[idx]->getType() == SHT_PROGBITS ||
251                     (*header)[idx]->getType() == SHT_NOBITS) &&
252                    "STT_SECTION with not BITS section.");
253 #endif
254             ELFSectionTy const *sec = owner->getSectionByIndex(idx);
255             rsl_assert(sec != 0 && "STT_SECTION with null section.");
256 
257             ELFSectionBitsTy const &st =
258               static_cast<ELFSectionBitsTy const &>(*sec);
259             my_addr = const_cast<unsigned char *>(&st[0] + (off_t)getValue());
260           }
261           break;
262 
263         case SHN_ABS:
264         case SHN_COMMON:
265         case SHN_UNDEF:
266         case SHN_XINDEX:
267           rsl_assert(0 && "STT_SECTION with special st_shndx.");
268           break;
269       }
270       break;
271 
272     case STT_NOTYPE:
273       switch (idx) {
274         default:
275           {
276 #ifndef NDEBUG
277             ELFSectionHeaderTableTy const *header =
278               owner->getSectionHeaderTable();
279             rsl_assert(((*header)[idx]->getType() == SHT_PROGBITS ||
280                     (*header)[idx]->getType() == SHT_NOBITS) &&
281                    "STT_SECTION with not BITS section.");
282 #endif
283             ELFSectionTy const *sec = owner->getSectionByIndex(idx);
284             rsl_assert(sec != 0 && "STT_SECTION with null section.");
285 
286             ELFSectionBitsTy const &st =
287               static_cast<ELFSectionBitsTy const &>(*sec);
288             my_addr = const_cast<unsigned char *>(&st[0] + (off_t)getValue());
289           }
290           break;
291 
292         case SHN_ABS:
293         case SHN_COMMON:
294         case SHN_XINDEX:
295           rsl_assert(0 && "STT_SECTION with special st_shndx.");
296           break;
297         case SHN_UNDEF:
298           return 0;
299       }
300       break;
301       return 0;
302 
303     case STT_COMMON:
304     case STT_FILE:
305     case STT_TLS:
306     case STT_LOOS:
307     case STT_HIOS:
308     case STT_LOPROC:
309     case STT_HIPROC:
310       rsl_assert(0 && "Not implement.");
311       return 0;
312   }
313   return my_addr;
314 }
315 
316 #endif // ELF_SYMBOL_HXX
317