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;
254 // Returns true if the actual number of relocations is stored in
255 // VirtualAddress field of the first relocation table entry.
256 bool hasExtendedRelocations() const {
257 return Characteristics & COFF::IMAGE_SCN_LNK_NRELOC_OVFL &&
258 NumberOfRelocations == UINT16_MAX;
262 struct coff_relocation {
263 support::ulittle32_t VirtualAddress;
264 support::ulittle32_t SymbolTableIndex;
265 support::ulittle16_t Type;
268 struct coff_aux_function_definition {
269 support::ulittle32_t TagIndex;
270 support::ulittle32_t TotalSize;
271 support::ulittle32_t PointerToLinenumber;
272 support::ulittle32_t PointerToNextFunction;
276 struct coff_aux_bf_and_ef_symbol {
278 support::ulittle16_t Linenumber;
280 support::ulittle32_t PointerToNextFunction;
284 struct coff_aux_weak_external {
285 support::ulittle32_t TagIndex;
286 support::ulittle32_t Characteristics;
290 struct coff_aux_section_definition {
291 support::ulittle32_t Length;
292 support::ulittle16_t NumberOfRelocations;
293 support::ulittle16_t NumberOfLinenumbers;
294 support::ulittle32_t CheckSum;
295 support::ulittle16_t Number;
296 support::ulittle8_t Selection;
300 struct coff_aux_clr_token {
301 support::ulittle8_t AuxType;
302 support::ulittle8_t Reserved;
303 support::ulittle32_t SymbolTableIndex;
307 struct coff_load_configuration32 {
308 support::ulittle32_t Characteristics;
309 support::ulittle32_t TimeDateStamp;
310 support::ulittle16_t MajorVersion;
311 support::ulittle16_t MinorVersion;
312 support::ulittle32_t GlobalFlagsClear;
313 support::ulittle32_t GlobalFlagsSet;
314 support::ulittle32_t CriticalSectionDefaultTimeout;
315 support::ulittle32_t DeCommitFreeBlockThreshold;
316 support::ulittle32_t DeCommitTotalFreeThreshold;
317 support::ulittle32_t LockPrefixTable;
318 support::ulittle32_t MaximumAllocationSize;
319 support::ulittle32_t VirtualMemoryThreshold;
320 support::ulittle32_t ProcessAffinityMask;
321 support::ulittle32_t ProcessHeapFlags;
322 support::ulittle16_t CSDVersion;
324 support::ulittle32_t EditList;
325 support::ulittle32_t SecurityCookie;
326 support::ulittle32_t SEHandlerTable;
327 support::ulittle32_t SEHandlerCount;
330 struct coff_runtime_function_x64 {
331 support::ulittle32_t BeginAddress;
332 support::ulittle32_t EndAddress;
333 support::ulittle32_t UnwindInformation;
336 class COFFObjectFile : public ObjectFile {
338 friend class ImportDirectoryEntryRef;
339 friend class ExportDirectoryEntryRef;
340 const coff_file_header *COFFHeader;
341 const pe32_header *PE32Header;
342 const pe32plus_header *PE32PlusHeader;
343 const data_directory *DataDirectory;
344 const coff_section *SectionTable;
345 const coff_symbol *SymbolTable;
346 const char *StringTable;
347 uint32_t StringTableSize;
348 const import_directory_table_entry *ImportDirectory;
349 uint32_t NumberOfImportDirectory;
350 const export_directory_table_entry *ExportDirectory;
352 error_code getString(uint32_t offset, StringRef &Res) const;
354 const coff_symbol *toSymb(DataRefImpl Symb) const;
355 const coff_section *toSec(DataRefImpl Sec) const;
356 const coff_relocation *toRel(DataRefImpl Rel) const;
358 error_code initSymbolTablePtr();
359 error_code initImportTablePtr();
360 error_code initExportTablePtr();
363 void moveSymbolNext(DataRefImpl &Symb) const override;
364 error_code getSymbolName(DataRefImpl Symb, StringRef &Res) const override;
365 error_code getSymbolAddress(DataRefImpl Symb, uint64_t &Res) const override;
366 error_code getSymbolFileOffset(DataRefImpl Symb,
367 uint64_t &Res) const override;
368 error_code getSymbolSize(DataRefImpl Symb, uint64_t &Res) const override;
369 uint32_t getSymbolFlags(DataRefImpl Symb) const override;
370 error_code getSymbolType(DataRefImpl Symb,
371 SymbolRef::Type &Res) const override;
372 error_code getSymbolSection(DataRefImpl Symb,
373 section_iterator &Res) const override;
374 error_code getSymbolValue(DataRefImpl Symb, uint64_t &Val) const override;
375 void moveSectionNext(DataRefImpl &Sec) const override;
376 error_code getSectionName(DataRefImpl Sec, StringRef &Res) const override;
377 error_code getSectionAddress(DataRefImpl Sec, uint64_t &Res) const override;
378 error_code getSectionSize(DataRefImpl Sec, uint64_t &Res) const override;
379 error_code getSectionContents(DataRefImpl Sec, StringRef &Res) const override;
380 error_code getSectionAlignment(DataRefImpl Sec, uint64_t &Res) const override;
381 error_code isSectionText(DataRefImpl Sec, bool &Res) const override;
382 error_code isSectionData(DataRefImpl Sec, bool &Res) const override;
383 error_code isSectionBSS(DataRefImpl Sec, bool &Res) const override;
384 error_code isSectionVirtual(DataRefImpl Sec, bool &Res) const override;
385 error_code isSectionZeroInit(DataRefImpl Sec, bool &Res) const override;
386 error_code isSectionReadOnlyData(DataRefImpl Sec, bool &Res) const override;
387 error_code isSectionRequiredForExecution(DataRefImpl Sec,
388 bool &Res) const override;
389 error_code sectionContainsSymbol(DataRefImpl Sec, DataRefImpl Symb,
390 bool &Result) const override;
391 relocation_iterator section_rel_begin(DataRefImpl Sec) const override;
392 relocation_iterator section_rel_end(DataRefImpl Sec) const override;
394 void moveRelocationNext(DataRefImpl &Rel) const override;
395 error_code getRelocationAddress(DataRefImpl Rel,
396 uint64_t &Res) const override;
397 error_code getRelocationOffset(DataRefImpl Rel, uint64_t &Res) const override;
398 symbol_iterator getRelocationSymbol(DataRefImpl Rel) const override;
399 error_code getRelocationType(DataRefImpl Rel, uint64_t &Res) const override;
401 getRelocationTypeName(DataRefImpl Rel,
402 SmallVectorImpl<char> &Result) const override;
404 getRelocationValueString(DataRefImpl Rel,
405 SmallVectorImpl<char> &Result) const override;
407 error_code getLibraryNext(DataRefImpl LibData,
408 LibraryRef &Result) const override;
409 error_code getLibraryPath(DataRefImpl LibData,
410 StringRef &Result) const override;
413 COFFObjectFile(MemoryBuffer *Object, error_code &EC, bool BufferOwned = true);
414 basic_symbol_iterator symbol_begin_impl() const override;
415 basic_symbol_iterator symbol_end_impl() const override;
416 library_iterator needed_library_begin() const override;
417 library_iterator needed_library_end() const override;
418 section_iterator section_begin() const override;
419 section_iterator section_end() const override;
421 const coff_section *getCOFFSection(const SectionRef &Section) const;
422 const coff_symbol *getCOFFSymbol(const SymbolRef &Symbol) const;
423 const coff_relocation *getCOFFRelocation(const RelocationRef &Reloc) const;
425 uint8_t getBytesInAddress() const override;
426 StringRef getFileFormatName() const override;
427 unsigned getArch() const override;
428 StringRef getLoadName() const override;
430 import_directory_iterator import_directory_begin() const;
431 import_directory_iterator import_directory_end() const;
432 export_directory_iterator export_directory_begin() const;
433 export_directory_iterator export_directory_end() const;
435 error_code getHeader(const coff_file_header *&Res) const;
436 error_code getCOFFHeader(const coff_file_header *&Res) const;
437 error_code getPE32Header(const pe32_header *&Res) const;
438 error_code getPE32PlusHeader(const pe32plus_header *&Res) const;
439 error_code getDataDirectory(uint32_t index, const data_directory *&Res) const;
440 error_code getSection(int32_t index, const coff_section *&Res) const;
441 error_code getSymbol(uint32_t index, const coff_symbol *&Res) const;
442 template <typename T>
443 error_code getAuxSymbol(uint32_t index, const T *&Res) const {
444 const coff_symbol *s;
445 error_code ec = getSymbol(index, s);
446 Res = reinterpret_cast<const T *>(s);
449 error_code getSymbolName(const coff_symbol *symbol, StringRef &Res) const;
450 ArrayRef<uint8_t> getSymbolAuxData(const coff_symbol *symbol) const;
452 error_code getSectionName(const coff_section *Sec, StringRef &Res) const;
453 error_code getSectionContents(const coff_section *Sec,
454 ArrayRef<uint8_t> &Res) const;
456 error_code getVaPtr(uint64_t VA, uintptr_t &Res) const;
457 error_code getRvaPtr(uint32_t Rva, uintptr_t &Res) const;
458 error_code getHintName(uint32_t Rva, uint16_t &Hint, StringRef &Name) const;
460 static inline bool classof(const Binary *v) { return v->isCOFF(); }
463 // The iterator for the import directory table.
464 class ImportDirectoryEntryRef {
466 ImportDirectoryEntryRef() : OwningObject(0) {}
467 ImportDirectoryEntryRef(const import_directory_table_entry *Table, uint32_t I,
468 const COFFObjectFile *Owner)
469 : ImportTable(Table), Index(I), OwningObject(Owner) {}
471 bool operator==(const ImportDirectoryEntryRef &Other) const;
473 error_code getName(StringRef &Result) const;
476 getImportTableEntry(const import_directory_table_entry *&Result) const;
479 getImportLookupEntry(const import_lookup_table_entry32 *&Result) const;
482 const import_directory_table_entry *ImportTable;
484 const COFFObjectFile *OwningObject;
487 // The iterator for the export directory table entry.
488 class ExportDirectoryEntryRef {
490 ExportDirectoryEntryRef() : OwningObject(0) {}
491 ExportDirectoryEntryRef(const export_directory_table_entry *Table, uint32_t I,
492 const COFFObjectFile *Owner)
493 : ExportTable(Table), Index(I), OwningObject(Owner) {}
495 bool operator==(const ExportDirectoryEntryRef &Other) const;
498 error_code getDllName(StringRef &Result) const;
499 error_code getOrdinalBase(uint32_t &Result) const;
500 error_code getOrdinal(uint32_t &Result) const;
501 error_code getExportRVA(uint32_t &Result) const;
502 error_code getSymbolName(StringRef &Result) const;
505 const export_directory_table_entry *ExportTable;
507 const COFFObjectFile *OwningObject;
509 } // end namespace object
510 } // end namespace llvm