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 isWeakExternal() const {
214 return StorageClass == COFF::IMAGE_SYM_CLASS_WEAK_EXTERNAL ||
215 (StorageClass == COFF::IMAGE_SYM_CLASS_EXTERNAL &&
216 SectionNumber == COFF::IMAGE_SYM_UNDEFINED && Value == 0);
219 bool isFileRecord() const {
220 return StorageClass == COFF::IMAGE_SYM_CLASS_FILE;
223 bool isSectionDefinition() const {
224 // C++/CLI creates external ABS symbols for non-const appdomain globals.
225 // These are also followed by an auxiliary section definition.
226 bool isAppdomainGlobal = StorageClass == COFF::IMAGE_SYM_CLASS_EXTERNAL &&
227 SectionNumber == COFF::IMAGE_SYM_ABSOLUTE;
228 bool isOrdinarySection =
229 StorageClass == COFF::IMAGE_SYM_CLASS_STATIC && Value == 0;
230 return isAppdomainGlobal || isOrdinarySection;
233 bool isCLRToken() const {
234 return StorageClass == COFF::IMAGE_SYM_CLASS_CLR_TOKEN;
238 struct coff_section {
240 support::ulittle32_t VirtualSize;
241 support::ulittle32_t VirtualAddress;
242 support::ulittle32_t SizeOfRawData;
243 support::ulittle32_t PointerToRawData;
244 support::ulittle32_t PointerToRelocations;
245 support::ulittle32_t PointerToLinenumbers;
246 support::ulittle16_t NumberOfRelocations;
247 support::ulittle16_t NumberOfLinenumbers;
248 support::ulittle32_t Characteristics;
251 struct coff_relocation {
252 support::ulittle32_t VirtualAddress;
253 support::ulittle32_t SymbolTableIndex;
254 support::ulittle16_t Type;
257 struct coff_aux_function_definition {
258 support::ulittle32_t TagIndex;
259 support::ulittle32_t TotalSize;
260 support::ulittle32_t PointerToLinenumber;
261 support::ulittle32_t PointerToNextFunction;
265 struct coff_aux_bf_and_ef_symbol {
267 support::ulittle16_t Linenumber;
269 support::ulittle32_t PointerToNextFunction;
273 struct coff_aux_weak_external {
274 support::ulittle32_t TagIndex;
275 support::ulittle32_t Characteristics;
279 struct coff_aux_section_definition {
280 support::ulittle32_t Length;
281 support::ulittle16_t NumberOfRelocations;
282 support::ulittle16_t NumberOfLinenumbers;
283 support::ulittle32_t CheckSum;
284 support::ulittle16_t Number;
285 support::ulittle8_t Selection;
289 struct coff_aux_clr_token {
290 support::ulittle8_t AuxType;
291 support::ulittle8_t Reserved;
292 support::ulittle32_t SymbolTableIndex;
296 struct coff_load_configuration32 {
297 support::ulittle32_t Characteristics;
298 support::ulittle32_t TimeDateStamp;
299 support::ulittle16_t MajorVersion;
300 support::ulittle16_t MinorVersion;
301 support::ulittle32_t GlobalFlagsClear;
302 support::ulittle32_t GlobalFlagsSet;
303 support::ulittle32_t CriticalSectionDefaultTimeout;
304 support::ulittle32_t DeCommitFreeBlockThreshold;
305 support::ulittle32_t DeCommitTotalFreeThreshold;
306 support::ulittle32_t LockPrefixTable;
307 support::ulittle32_t MaximumAllocationSize;
308 support::ulittle32_t VirtualMemoryThreshold;
309 support::ulittle32_t ProcessAffinityMask;
310 support::ulittle32_t ProcessHeapFlags;
311 support::ulittle16_t CSDVersion;
313 support::ulittle32_t EditList;
314 support::ulittle32_t SecurityCookie;
315 support::ulittle32_t SEHandlerTable;
316 support::ulittle32_t SEHandlerCount;
319 struct coff_runtime_function_x64 {
320 support::ulittle32_t BeginAddress;
321 support::ulittle32_t EndAddress;
322 support::ulittle32_t UnwindInformation;
325 class COFFObjectFile : public ObjectFile {
327 friend class ImportDirectoryEntryRef;
328 friend class ExportDirectoryEntryRef;
329 const coff_file_header *COFFHeader;
330 const pe32_header *PE32Header;
331 const pe32plus_header *PE32PlusHeader;
332 const data_directory *DataDirectory;
333 const coff_section *SectionTable;
334 const coff_symbol *SymbolTable;
335 const char *StringTable;
336 uint32_t StringTableSize;
337 const import_directory_table_entry *ImportDirectory;
338 uint32_t NumberOfImportDirectory;
339 const export_directory_table_entry *ExportDirectory;
341 error_code getString(uint32_t offset, StringRef &Res) const;
343 const coff_symbol *toSymb(DataRefImpl Symb) const;
344 const coff_section *toSec(DataRefImpl Sec) const;
345 const coff_relocation *toRel(DataRefImpl Rel) const;
347 error_code initSymbolTablePtr();
348 error_code initImportTablePtr();
349 error_code initExportTablePtr();
352 void moveSymbolNext(DataRefImpl &Symb) const override;
353 error_code getSymbolName(DataRefImpl Symb, StringRef &Res) const override;
354 error_code getSymbolAddress(DataRefImpl Symb, uint64_t &Res) const override;
355 error_code getSymbolFileOffset(DataRefImpl Symb,
356 uint64_t &Res) const override;
357 error_code getSymbolSize(DataRefImpl Symb, uint64_t &Res) const override;
358 uint32_t getSymbolFlags(DataRefImpl Symb) const override;
359 error_code getSymbolType(DataRefImpl Symb,
360 SymbolRef::Type &Res) const override;
361 error_code getSymbolSection(DataRefImpl Symb,
362 section_iterator &Res) const override;
363 error_code getSymbolValue(DataRefImpl Symb, uint64_t &Val) const override;
364 void moveSectionNext(DataRefImpl &Sec) const override;
365 error_code getSectionName(DataRefImpl Sec, StringRef &Res) const override;
366 error_code getSectionAddress(DataRefImpl Sec, uint64_t &Res) const override;
367 error_code getSectionSize(DataRefImpl Sec, uint64_t &Res) const override;
368 error_code getSectionContents(DataRefImpl Sec, StringRef &Res) const override;
369 error_code getSectionAlignment(DataRefImpl Sec, uint64_t &Res) const override;
370 error_code isSectionText(DataRefImpl Sec, bool &Res) const override;
371 error_code isSectionData(DataRefImpl Sec, bool &Res) const override;
372 error_code isSectionBSS(DataRefImpl Sec, bool &Res) const override;
373 error_code isSectionVirtual(DataRefImpl Sec, bool &Res) const override;
374 error_code isSectionZeroInit(DataRefImpl Sec, bool &Res) const override;
375 error_code isSectionReadOnlyData(DataRefImpl Sec, bool &Res) const override;
376 error_code isSectionRequiredForExecution(DataRefImpl Sec,
377 bool &Res) const override;
378 error_code sectionContainsSymbol(DataRefImpl Sec, DataRefImpl Symb,
379 bool &Result) const override;
380 relocation_iterator section_rel_begin(DataRefImpl Sec) const override;
381 relocation_iterator section_rel_end(DataRefImpl Sec) const override;
383 void moveRelocationNext(DataRefImpl &Rel) const override;
384 error_code getRelocationAddress(DataRefImpl Rel,
385 uint64_t &Res) const override;
386 error_code getRelocationOffset(DataRefImpl Rel, uint64_t &Res) const override;
387 symbol_iterator getRelocationSymbol(DataRefImpl Rel) const override;
388 error_code getRelocationType(DataRefImpl Rel, uint64_t &Res) const override;
390 getRelocationTypeName(DataRefImpl Rel,
391 SmallVectorImpl<char> &Result) const override;
393 getRelocationValueString(DataRefImpl Rel,
394 SmallVectorImpl<char> &Result) const override;
396 error_code getLibraryNext(DataRefImpl LibData,
397 LibraryRef &Result) const override;
398 error_code getLibraryPath(DataRefImpl LibData,
399 StringRef &Result) const override;
402 COFFObjectFile(MemoryBuffer *Object, error_code &EC, bool BufferOwned = true);
403 basic_symbol_iterator symbol_begin_impl() const override;
404 basic_symbol_iterator symbol_end_impl() const override;
405 library_iterator needed_library_begin() const override;
406 library_iterator needed_library_end() const override;
407 section_iterator section_begin() const override;
408 section_iterator section_end() const override;
410 const coff_section *getCOFFSection(const SectionRef &Section) const;
411 const coff_symbol *getCOFFSymbol(const SymbolRef &Symbol) const;
412 const coff_relocation *getCOFFRelocation(const RelocationRef &Reloc) const;
414 uint8_t getBytesInAddress() const override;
415 StringRef getFileFormatName() const override;
416 unsigned getArch() const override;
417 StringRef getLoadName() const override;
419 import_directory_iterator import_directory_begin() const;
420 import_directory_iterator import_directory_end() const;
421 export_directory_iterator export_directory_begin() const;
422 export_directory_iterator export_directory_end() const;
424 error_code getHeader(const coff_file_header *&Res) const;
425 error_code getCOFFHeader(const coff_file_header *&Res) const;
426 error_code getPE32Header(const pe32_header *&Res) const;
427 error_code getPE32PlusHeader(const pe32plus_header *&Res) const;
428 error_code getDataDirectory(uint32_t index, const data_directory *&Res) const;
429 error_code getSection(int32_t index, const coff_section *&Res) const;
430 error_code getSymbol(uint32_t index, const coff_symbol *&Res) const;
431 template <typename T>
432 error_code getAuxSymbol(uint32_t index, const T *&Res) const {
433 const coff_symbol *s;
434 error_code ec = getSymbol(index, s);
435 Res = reinterpret_cast<const T *>(s);
438 error_code getSymbolName(const coff_symbol *symbol, StringRef &Res) const;
439 ArrayRef<uint8_t> getSymbolAuxData(const coff_symbol *symbol) const;
441 error_code getSectionName(const coff_section *Sec, StringRef &Res) const;
442 error_code getSectionContents(const coff_section *Sec,
443 ArrayRef<uint8_t> &Res) const;
445 error_code getVaPtr(uint64_t VA, uintptr_t &Res) const;
446 error_code getRvaPtr(uint32_t Rva, uintptr_t &Res) const;
447 error_code getHintName(uint32_t Rva, uint16_t &Hint, StringRef &Name) const;
449 static inline bool classof(const Binary *v) { return v->isCOFF(); }
452 // The iterator for the import directory table.
453 class ImportDirectoryEntryRef {
455 ImportDirectoryEntryRef() : OwningObject(0) {}
456 ImportDirectoryEntryRef(const import_directory_table_entry *Table, uint32_t I,
457 const COFFObjectFile *Owner)
458 : ImportTable(Table), Index(I), OwningObject(Owner) {}
460 bool operator==(const ImportDirectoryEntryRef &Other) const;
462 error_code getName(StringRef &Result) const;
465 getImportTableEntry(const import_directory_table_entry *&Result) const;
468 getImportLookupEntry(const import_lookup_table_entry32 *&Result) const;
471 const import_directory_table_entry *ImportTable;
473 const COFFObjectFile *OwningObject;
476 // The iterator for the export directory table entry.
477 class ExportDirectoryEntryRef {
479 ExportDirectoryEntryRef() : OwningObject(0) {}
480 ExportDirectoryEntryRef(const export_directory_table_entry *Table, uint32_t I,
481 const COFFObjectFile *Owner)
482 : ExportTable(Table), Index(I), OwningObject(Owner) {}
484 bool operator==(const ExportDirectoryEntryRef &Other) const;
487 error_code getDllName(StringRef &Result) const;
488 error_code getOrdinalBase(uint32_t &Result) const;
489 error_code getOrdinal(uint32_t &Result) const;
490 error_code getExportRVA(uint32_t &Result) const;
491 error_code getSymbolName(StringRef &Result) const;
494 const export_directory_table_entry *ExportTable;
496 const COFFObjectFile *OwningObject;
498 } // end namespace object
499 } // end namespace llvm