1 //===- COFF.h - COFF object file implementation -----------------*- C++ -*-===//
3 // The LLVM Compiler Infrastructure
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
8 //===----------------------------------------------------------------------===//
10 // This file declares the COFFObjectFile class.
12 //===----------------------------------------------------------------------===//
14 #ifndef LLVM_OBJECT_COFF_H
15 #define LLVM_OBJECT_COFF_H
17 #include "llvm/Object/ObjectFile.h"
18 #include "llvm/Support/COFF.h"
19 #include "llvm/Support/Endian.h"
22 template <typename T> class ArrayRef;
25 class ImportDirectoryEntryRef;
26 class ExportDirectoryEntryRef;
27 typedef content_iterator<ImportDirectoryEntryRef> import_directory_iterator;
28 typedef content_iterator<ExportDirectoryEntryRef> export_directory_iterator;
30 /// The DOS compatible header at the front of all PE/COFF executables.
32 support::ulittle16_t Magic;
33 support::ulittle16_t UsedBytesInTheLastPage;
34 support::ulittle16_t FileSizeInPages;
35 support::ulittle16_t NumberOfRelocationItems;
36 support::ulittle16_t HeaderSizeInParagraphs;
37 support::ulittle16_t MinimumExtraParagraphs;
38 support::ulittle16_t MaximumExtraParagraphs;
39 support::ulittle16_t InitialRelativeSS;
40 support::ulittle16_t InitialSP;
41 support::ulittle16_t Checksum;
42 support::ulittle16_t InitialIP;
43 support::ulittle16_t InitialRelativeCS;
44 support::ulittle16_t AddressOfRelocationTable;
45 support::ulittle16_t OverlayNumber;
46 support::ulittle16_t Reserved[4];
47 support::ulittle16_t OEMid;
48 support::ulittle16_t OEMinfo;
49 support::ulittle16_t Reserved2[10];
50 support::ulittle32_t AddressOfNewExeHeader;
53 struct coff_file_header {
54 support::ulittle16_t Machine;
55 support::ulittle16_t NumberOfSections;
56 support::ulittle32_t TimeDateStamp;
57 support::ulittle32_t PointerToSymbolTable;
58 support::ulittle32_t NumberOfSymbols;
59 support::ulittle16_t SizeOfOptionalHeader;
60 support::ulittle16_t Characteristics;
62 bool isImportLibrary() const { return NumberOfSections == 0xffff; }
65 /// The 32-bit PE header that follows the COFF header.
67 support::ulittle16_t Magic;
68 uint8_t MajorLinkerVersion;
69 uint8_t MinorLinkerVersion;
70 support::ulittle32_t SizeOfCode;
71 support::ulittle32_t SizeOfInitializedData;
72 support::ulittle32_t SizeOfUninitializedData;
73 support::ulittle32_t AddressOfEntryPoint;
74 support::ulittle32_t BaseOfCode;
75 support::ulittle32_t BaseOfData;
76 support::ulittle32_t ImageBase;
77 support::ulittle32_t SectionAlignment;
78 support::ulittle32_t FileAlignment;
79 support::ulittle16_t MajorOperatingSystemVersion;
80 support::ulittle16_t MinorOperatingSystemVersion;
81 support::ulittle16_t MajorImageVersion;
82 support::ulittle16_t MinorImageVersion;
83 support::ulittle16_t MajorSubsystemVersion;
84 support::ulittle16_t MinorSubsystemVersion;
85 support::ulittle32_t Win32VersionValue;
86 support::ulittle32_t SizeOfImage;
87 support::ulittle32_t SizeOfHeaders;
88 support::ulittle32_t CheckSum;
89 support::ulittle16_t Subsystem;
90 support::ulittle16_t DLLCharacteristics;
91 support::ulittle32_t SizeOfStackReserve;
92 support::ulittle32_t SizeOfStackCommit;
93 support::ulittle32_t SizeOfHeapReserve;
94 support::ulittle32_t SizeOfHeapCommit;
95 support::ulittle32_t LoaderFlags;
96 support::ulittle32_t NumberOfRvaAndSize;
99 /// The 64-bit PE header that follows the COFF header.
100 struct pe32plus_header {
101 support::ulittle16_t Magic;
102 uint8_t MajorLinkerVersion;
103 uint8_t MinorLinkerVersion;
104 support::ulittle32_t SizeOfCode;
105 support::ulittle32_t SizeOfInitializedData;
106 support::ulittle32_t SizeOfUninitializedData;
107 support::ulittle32_t AddressOfEntryPoint;
108 support::ulittle32_t BaseOfCode;
109 support::ulittle64_t ImageBase;
110 support::ulittle32_t SectionAlignment;
111 support::ulittle32_t FileAlignment;
112 support::ulittle16_t MajorOperatingSystemVersion;
113 support::ulittle16_t MinorOperatingSystemVersion;
114 support::ulittle16_t MajorImageVersion;
115 support::ulittle16_t MinorImageVersion;
116 support::ulittle16_t MajorSubsystemVersion;
117 support::ulittle16_t MinorSubsystemVersion;
118 support::ulittle32_t Win32VersionValue;
119 support::ulittle32_t SizeOfImage;
120 support::ulittle32_t SizeOfHeaders;
121 support::ulittle32_t CheckSum;
122 support::ulittle16_t Subsystem;
123 support::ulittle16_t DLLCharacteristics;
124 support::ulittle64_t SizeOfStackReserve;
125 support::ulittle64_t SizeOfStackCommit;
126 support::ulittle64_t SizeOfHeapReserve;
127 support::ulittle64_t SizeOfHeapCommit;
128 support::ulittle32_t LoaderFlags;
129 support::ulittle32_t NumberOfRvaAndSize;
132 struct data_directory {
133 support::ulittle32_t RelativeVirtualAddress;
134 support::ulittle32_t Size;
137 struct import_directory_table_entry {
138 support::ulittle32_t ImportLookupTableRVA;
139 support::ulittle32_t TimeDateStamp;
140 support::ulittle32_t ForwarderChain;
141 support::ulittle32_t NameRVA;
142 support::ulittle32_t ImportAddressTableRVA;
145 struct import_lookup_table_entry32 {
146 support::ulittle32_t data;
148 bool isOrdinal() const { return data & 0x80000000; }
150 uint16_t getOrdinal() const {
151 assert(isOrdinal() && "ILT entry is not an ordinal!");
152 return data & 0xFFFF;
155 uint32_t getHintNameRVA() const {
156 assert(!isOrdinal() && "ILT entry is not a Hint/Name RVA!");
161 struct export_directory_table_entry {
162 support::ulittle32_t ExportFlags;
163 support::ulittle32_t TimeDateStamp;
164 support::ulittle16_t MajorVersion;
165 support::ulittle16_t MinorVersion;
166 support::ulittle32_t NameRVA;
167 support::ulittle32_t OrdinalBase;
168 support::ulittle32_t AddressTableEntries;
169 support::ulittle32_t NumberOfNamePointers;
170 support::ulittle32_t ExportAddressTableRVA;
171 support::ulittle32_t NamePointerRVA;
172 support::ulittle32_t OrdinalTableRVA;
175 union export_address_table_entry {
176 support::ulittle32_t ExportRVA;
177 support::ulittle32_t ForwarderRVA;
180 typedef support::ulittle32_t export_name_pointer_table_entry;
181 typedef support::ulittle16_t export_ordinal_table_entry;
184 struct StringTableOffset {
185 support::ulittle32_t Zeroes;
186 support::ulittle32_t Offset;
191 StringTableOffset Offset;
194 support::ulittle32_t Value;
195 support::ulittle16_t SectionNumber;
197 support::ulittle16_t Type;
199 support::ulittle8_t StorageClass;
200 support::ulittle8_t NumberOfAuxSymbols;
202 uint8_t getBaseType() const { return Type & 0x0F; }
204 uint8_t getComplexType() const { return (Type & 0xF0) >> 4; }
206 bool isFunctionDefinition() const {
207 return StorageClass == COFF::IMAGE_SYM_CLASS_EXTERNAL &&
208 getBaseType() == COFF::IMAGE_SYM_TYPE_NULL &&
209 getComplexType() == COFF::IMAGE_SYM_DTYPE_FUNCTION &&
210 !COFF::isReservedSectionNumber(SectionNumber);
213 bool isFunctionLineInfo() const {
214 return StorageClass == COFF::IMAGE_SYM_CLASS_FUNCTION;
217 bool isWeakExternal() const {
218 return StorageClass == COFF::IMAGE_SYM_CLASS_WEAK_EXTERNAL ||
219 (StorageClass == COFF::IMAGE_SYM_CLASS_EXTERNAL &&
220 SectionNumber == COFF::IMAGE_SYM_UNDEFINED && Value == 0);
223 bool isFileRecord() const {
224 return StorageClass == COFF::IMAGE_SYM_CLASS_FILE;
227 bool isSectionDefinition() const {
228 // C++/CLI creates external ABS symbols for non-const appdomain globals.
229 // These are also followed by an auxiliary section definition.
230 bool isAppdomainGlobal = StorageClass == COFF::IMAGE_SYM_CLASS_EXTERNAL &&
231 SectionNumber == COFF::IMAGE_SYM_ABSOLUTE;
232 bool isOrdinarySection =
233 StorageClass == COFF::IMAGE_SYM_CLASS_STATIC && Value == 0;
234 return isAppdomainGlobal || isOrdinarySection;
237 bool isCLRToken() const {
238 return StorageClass == COFF::IMAGE_SYM_CLASS_CLR_TOKEN;
242 struct coff_section {
244 support::ulittle32_t VirtualSize;
245 support::ulittle32_t VirtualAddress;
246 support::ulittle32_t SizeOfRawData;
247 support::ulittle32_t PointerToRawData;
248 support::ulittle32_t PointerToRelocations;
249 support::ulittle32_t PointerToLinenumbers;
250 support::ulittle16_t NumberOfRelocations;
251 support::ulittle16_t NumberOfLinenumbers;
252 support::ulittle32_t Characteristics;
255 struct coff_relocation {
256 support::ulittle32_t VirtualAddress;
257 support::ulittle32_t SymbolTableIndex;
258 support::ulittle16_t Type;
261 struct coff_aux_function_definition {
262 support::ulittle32_t TagIndex;
263 support::ulittle32_t TotalSize;
264 support::ulittle32_t PointerToLinenumber;
265 support::ulittle32_t PointerToNextFunction;
269 struct coff_aux_bf_and_ef_symbol {
271 support::ulittle16_t Linenumber;
273 support::ulittle32_t PointerToNextFunction;
277 struct coff_aux_weak_external {
278 support::ulittle32_t TagIndex;
279 support::ulittle32_t Characteristics;
283 struct coff_aux_section_definition {
284 support::ulittle32_t Length;
285 support::ulittle16_t NumberOfRelocations;
286 support::ulittle16_t NumberOfLinenumbers;
287 support::ulittle32_t CheckSum;
288 support::ulittle16_t Number;
289 support::ulittle8_t Selection;
293 struct coff_aux_clr_token {
294 support::ulittle8_t AuxType;
295 support::ulittle8_t Reserved;
296 support::ulittle32_t SymbolTableIndex;
300 struct coff_load_configuration32 {
301 support::ulittle32_t Characteristics;
302 support::ulittle32_t TimeDateStamp;
303 support::ulittle16_t MajorVersion;
304 support::ulittle16_t MinorVersion;
305 support::ulittle32_t GlobalFlagsClear;
306 support::ulittle32_t GlobalFlagsSet;
307 support::ulittle32_t CriticalSectionDefaultTimeout;
308 support::ulittle32_t DeCommitFreeBlockThreshold;
309 support::ulittle32_t DeCommitTotalFreeThreshold;
310 support::ulittle32_t LockPrefixTable;
311 support::ulittle32_t MaximumAllocationSize;
312 support::ulittle32_t VirtualMemoryThreshold;
313 support::ulittle32_t ProcessAffinityMask;
314 support::ulittle32_t ProcessHeapFlags;
315 support::ulittle16_t CSDVersion;
317 support::ulittle32_t EditList;
318 support::ulittle32_t SecurityCookie;
319 support::ulittle32_t SEHandlerTable;
320 support::ulittle32_t SEHandlerCount;
323 struct coff_runtime_function_x64 {
324 support::ulittle32_t BeginAddress;
325 support::ulittle32_t EndAddress;
326 support::ulittle32_t UnwindInformation;
329 class COFFObjectFile : public ObjectFile {
331 friend class ImportDirectoryEntryRef;
332 friend class ExportDirectoryEntryRef;
333 const coff_file_header *COFFHeader;
334 const pe32_header *PE32Header;
335 const pe32plus_header *PE32PlusHeader;
336 const data_directory *DataDirectory;
337 const coff_section *SectionTable;
338 const coff_symbol *SymbolTable;
339 const char *StringTable;
340 uint32_t StringTableSize;
341 const import_directory_table_entry *ImportDirectory;
342 uint32_t NumberOfImportDirectory;
343 const export_directory_table_entry *ExportDirectory;
345 error_code getString(uint32_t offset, StringRef &Res) const;
347 const coff_symbol *toSymb(DataRefImpl Symb) const;
348 const coff_section *toSec(DataRefImpl Sec) const;
349 const coff_relocation *toRel(DataRefImpl Rel) const;
351 error_code initSymbolTablePtr();
352 error_code initImportTablePtr();
353 error_code initExportTablePtr();
356 void moveSymbolNext(DataRefImpl &Symb) const override;
357 error_code getSymbolName(DataRefImpl Symb, StringRef &Res) const override;
358 error_code getSymbolAddress(DataRefImpl Symb, uint64_t &Res) const override;
359 error_code getSymbolFileOffset(DataRefImpl Symb,
360 uint64_t &Res) const override;
361 error_code getSymbolSize(DataRefImpl Symb, uint64_t &Res) const override;
362 uint32_t getSymbolFlags(DataRefImpl Symb) const override;
363 error_code getSymbolType(DataRefImpl Symb,
364 SymbolRef::Type &Res) const override;
365 error_code getSymbolSection(DataRefImpl Symb,
366 section_iterator &Res) const override;
367 error_code getSymbolValue(DataRefImpl Symb, uint64_t &Val) const override;
368 void moveSectionNext(DataRefImpl &Sec) const override;
369 error_code getSectionName(DataRefImpl Sec, StringRef &Res) const override;
370 error_code getSectionAddress(DataRefImpl Sec, uint64_t &Res) const override;
371 error_code getSectionSize(DataRefImpl Sec, uint64_t &Res) const override;
372 error_code getSectionContents(DataRefImpl Sec, StringRef &Res) const override;
373 error_code getSectionAlignment(DataRefImpl Sec, uint64_t &Res) const override;
374 error_code isSectionText(DataRefImpl Sec, bool &Res) const override;
375 error_code isSectionData(DataRefImpl Sec, bool &Res) const override;
376 error_code isSectionBSS(DataRefImpl Sec, bool &Res) const override;
377 error_code isSectionVirtual(DataRefImpl Sec, bool &Res) const override;
378 error_code isSectionZeroInit(DataRefImpl Sec, bool &Res) const override;
379 error_code isSectionReadOnlyData(DataRefImpl Sec, bool &Res) const override;
380 error_code isSectionRequiredForExecution(DataRefImpl Sec,
381 bool &Res) const override;
382 error_code sectionContainsSymbol(DataRefImpl Sec, DataRefImpl Symb,
383 bool &Result) const override;
384 relocation_iterator section_rel_begin(DataRefImpl Sec) const override;
385 relocation_iterator section_rel_end(DataRefImpl Sec) const override;
387 void moveRelocationNext(DataRefImpl &Rel) const override;
388 error_code getRelocationAddress(DataRefImpl Rel,
389 uint64_t &Res) const override;
390 error_code getRelocationOffset(DataRefImpl Rel, uint64_t &Res) const override;
391 symbol_iterator getRelocationSymbol(DataRefImpl Rel) const override;
392 error_code getRelocationType(DataRefImpl Rel, uint64_t &Res) const override;
394 getRelocationTypeName(DataRefImpl Rel,
395 SmallVectorImpl<char> &Result) const override;
397 getRelocationValueString(DataRefImpl Rel,
398 SmallVectorImpl<char> &Result) const override;
400 error_code getLibraryNext(DataRefImpl LibData,
401 LibraryRef &Result) const override;
402 error_code getLibraryPath(DataRefImpl LibData,
403 StringRef &Result) const override;
406 COFFObjectFile(MemoryBuffer *Object, error_code &EC, bool BufferOwned = true);
407 basic_symbol_iterator symbol_begin_impl() const override;
408 basic_symbol_iterator symbol_end_impl() const override;
409 library_iterator needed_library_begin() const override;
410 library_iterator needed_library_end() const override;
411 section_iterator section_begin() const override;
412 section_iterator section_end() const override;
414 const coff_section *getCOFFSection(const SectionRef &Section) const;
415 const coff_symbol *getCOFFSymbol(const SymbolRef &Symbol) const;
416 const coff_relocation *getCOFFRelocation(const RelocationRef &Reloc) const;
418 uint8_t getBytesInAddress() const override;
419 StringRef getFileFormatName() const override;
420 unsigned getArch() const override;
421 StringRef getLoadName() const override;
423 import_directory_iterator import_directory_begin() const;
424 import_directory_iterator import_directory_end() const;
425 export_directory_iterator export_directory_begin() const;
426 export_directory_iterator export_directory_end() const;
428 error_code getHeader(const coff_file_header *&Res) const;
429 error_code getCOFFHeader(const coff_file_header *&Res) const;
430 error_code getPE32Header(const pe32_header *&Res) const;
431 error_code getPE32PlusHeader(const pe32plus_header *&Res) const;
432 error_code getDataDirectory(uint32_t index, const data_directory *&Res) const;
433 error_code getSection(int32_t index, const coff_section *&Res) const;
434 error_code getSymbol(uint32_t index, const coff_symbol *&Res) const;
435 template <typename T>
436 error_code getAuxSymbol(uint32_t index, const T *&Res) const {
437 const coff_symbol *s;
438 error_code ec = getSymbol(index, s);
439 Res = reinterpret_cast<const T *>(s);
442 error_code getSymbolName(const coff_symbol *symbol, StringRef &Res) const;
443 ArrayRef<uint8_t> getSymbolAuxData(const coff_symbol *symbol) const;
445 error_code getSectionName(const coff_section *Sec, StringRef &Res) const;
446 error_code getSectionContents(const coff_section *Sec,
447 ArrayRef<uint8_t> &Res) const;
449 error_code getVaPtr(uint64_t VA, uintptr_t &Res) const;
450 error_code getRvaPtr(uint32_t Rva, uintptr_t &Res) const;
451 error_code getHintName(uint32_t Rva, uint16_t &Hint, StringRef &Name) const;
453 static inline bool classof(const Binary *v) { return v->isCOFF(); }
456 // The iterator for the import directory table.
457 class ImportDirectoryEntryRef {
459 ImportDirectoryEntryRef() : OwningObject(0) {}
460 ImportDirectoryEntryRef(const import_directory_table_entry *Table, uint32_t I,
461 const COFFObjectFile *Owner)
462 : ImportTable(Table), Index(I), OwningObject(Owner) {}
464 bool operator==(const ImportDirectoryEntryRef &Other) const;
466 error_code getName(StringRef &Result) const;
469 getImportTableEntry(const import_directory_table_entry *&Result) const;
472 getImportLookupEntry(const import_lookup_table_entry32 *&Result) const;
475 const import_directory_table_entry *ImportTable;
477 const COFFObjectFile *OwningObject;
480 // The iterator for the export directory table entry.
481 class ExportDirectoryEntryRef {
483 ExportDirectoryEntryRef() : OwningObject(0) {}
484 ExportDirectoryEntryRef(const export_directory_table_entry *Table, uint32_t I,
485 const COFFObjectFile *Owner)
486 : ExportTable(Table), Index(I), OwningObject(Owner) {}
488 bool operator==(const ExportDirectoryEntryRef &Other) const;
491 error_code getDllName(StringRef &Result) const;
492 error_code getOrdinalBase(uint32_t &Result) const;
493 error_code getOrdinal(uint32_t &Result) const;
494 error_code getExportRVA(uint32_t &Result) const;
495 error_code getSymbolName(StringRef &Result) const;
498 const export_directory_table_entry *ExportTable;
500 const COFFObjectFile *OwningObject;
502 } // end namespace object
503 } // end namespace llvm