18e90adafSMichael J. Spencer //===- COFFObjectFile.cpp - COFF object file implementation -----*- C++ -*-===// 28e90adafSMichael J. Spencer // 38e90adafSMichael J. Spencer // The LLVM Compiler Infrastructure 48e90adafSMichael J. Spencer // 58e90adafSMichael J. Spencer // This file is distributed under the University of Illinois Open Source 68e90adafSMichael J. Spencer // License. See LICENSE.TXT for details. 78e90adafSMichael J. Spencer // 88e90adafSMichael J. Spencer //===----------------------------------------------------------------------===// 98e90adafSMichael J. Spencer // 108e90adafSMichael J. Spencer // This file declares the COFFObjectFile class. 118e90adafSMichael J. Spencer // 128e90adafSMichael J. Spencer //===----------------------------------------------------------------------===// 138e90adafSMichael J. Spencer 14ec29b121SMichael J. Spencer #include "llvm/Object/COFF.h" 159da9e693SMichael J. Spencer #include "llvm/ADT/ArrayRef.h" 168e90adafSMichael J. Spencer #include "llvm/ADT/StringSwitch.h" 178e90adafSMichael J. Spencer #include "llvm/ADT/Triple.h" 186a75acb1SRui Ueyama #include "llvm/ADT/iterator_range.h" 19f078eff3SRui Ueyama #include "llvm/Support/COFF.h" 20c2bed429SRui Ueyama #include "llvm/Support/Debug.h" 21c2bed429SRui Ueyama #include "llvm/Support/raw_ostream.h" 22981af002SWill Dietz #include <cctype> 239d2c15efSNico Rieck #include <limits> 248e90adafSMichael J. Spencer 258e90adafSMichael J. Spencer using namespace llvm; 268e90adafSMichael J. Spencer using namespace object; 278e90adafSMichael J. Spencer 288e90adafSMichael J. Spencer using support::ulittle16_t; 298e90adafSMichael J. Spencer using support::ulittle32_t; 30861021f9SRui Ueyama using support::ulittle64_t; 318e90adafSMichael J. Spencer using support::little16_t; 328e90adafSMichael J. Spencer 331d6167fdSMichael J. Spencer // Returns false if size is greater than the buffer size. And sets ec. 3448af1c2aSRafael Espindola static bool checkSize(MemoryBufferRef M, std::error_code &EC, uint64_t Size) { 35c3f9b5a5SRafael Espindola if (M.getBufferSize() < Size) { 368ff24d25SRui Ueyama EC = object_error::unexpected_eof; 371d6167fdSMichael J. Spencer return false; 381d6167fdSMichael J. Spencer } 391d6167fdSMichael J. Spencer return true; 408e90adafSMichael J. Spencer } 418e90adafSMichael J. Spencer 42e830c60dSDavid Majnemer static std::error_code checkOffset(MemoryBufferRef M, uintptr_t Addr, 4394751be7SDavid Majnemer const uint64_t Size) { 44e830c60dSDavid Majnemer if (Addr + Size < Addr || Addr + Size < Size || 45e830c60dSDavid Majnemer Addr + Size > uintptr_t(M.getBufferEnd()) || 46e830c60dSDavid Majnemer Addr < uintptr_t(M.getBufferStart())) { 47e830c60dSDavid Majnemer return object_error::unexpected_eof; 48e830c60dSDavid Majnemer } 497d099195SRui Ueyama return std::error_code(); 50e830c60dSDavid Majnemer } 51e830c60dSDavid Majnemer 52ed64342bSRui Ueyama // Sets Obj unless any bytes in [addr, addr + size) fall outsize of m. 53ed64342bSRui Ueyama // Returns unexpected_eof if error. 54ed64342bSRui Ueyama template <typename T> 5548af1c2aSRafael Espindola static std::error_code getObject(const T *&Obj, MemoryBufferRef M, 5658323a97SDavid Majnemer const void *Ptr, 57236b0ca7SDavid Majnemer const uint64_t Size = sizeof(T)) { 58ed64342bSRui Ueyama uintptr_t Addr = uintptr_t(Ptr); 59e830c60dSDavid Majnemer if (std::error_code EC = checkOffset(M, Addr, Size)) 60e830c60dSDavid Majnemer return EC; 61ed64342bSRui Ueyama Obj = reinterpret_cast<const T *>(Addr); 627d099195SRui Ueyama return std::error_code(); 631d6167fdSMichael J. Spencer } 641d6167fdSMichael J. Spencer 659d2c15efSNico Rieck // Decode a string table entry in base 64 (//AAAAAA). Expects \arg Str without 669d2c15efSNico Rieck // prefixed slashes. 679d2c15efSNico Rieck static bool decodeBase64StringEntry(StringRef Str, uint32_t &Result) { 689d2c15efSNico Rieck assert(Str.size() <= 6 && "String too long, possible overflow."); 699d2c15efSNico Rieck if (Str.size() > 6) 709d2c15efSNico Rieck return true; 719d2c15efSNico Rieck 729d2c15efSNico Rieck uint64_t Value = 0; 739d2c15efSNico Rieck while (!Str.empty()) { 749d2c15efSNico Rieck unsigned CharVal; 759d2c15efSNico Rieck if (Str[0] >= 'A' && Str[0] <= 'Z') // 0..25 769d2c15efSNico Rieck CharVal = Str[0] - 'A'; 779d2c15efSNico Rieck else if (Str[0] >= 'a' && Str[0] <= 'z') // 26..51 789d2c15efSNico Rieck CharVal = Str[0] - 'a' + 26; 799d2c15efSNico Rieck else if (Str[0] >= '0' && Str[0] <= '9') // 52..61 809d2c15efSNico Rieck CharVal = Str[0] - '0' + 52; 819d2c15efSNico Rieck else if (Str[0] == '+') // 62 825500b07cSRui Ueyama CharVal = 62; 839d2c15efSNico Rieck else if (Str[0] == '/') // 63 845500b07cSRui Ueyama CharVal = 63; 859d2c15efSNico Rieck else 869d2c15efSNico Rieck return true; 879d2c15efSNico Rieck 889d2c15efSNico Rieck Value = (Value * 64) + CharVal; 899d2c15efSNico Rieck Str = Str.substr(1); 909d2c15efSNico Rieck } 919d2c15efSNico Rieck 929d2c15efSNico Rieck if (Value > std::numeric_limits<uint32_t>::max()) 939d2c15efSNico Rieck return true; 949d2c15efSNico Rieck 959d2c15efSNico Rieck Result = static_cast<uint32_t>(Value); 969d2c15efSNico Rieck return false; 979d2c15efSNico Rieck } 989d2c15efSNico Rieck 9944f51e51SDavid Majnemer template <typename coff_symbol_type> 10044f51e51SDavid Majnemer const coff_symbol_type *COFFObjectFile::toSymb(DataRefImpl Ref) const { 10144f51e51SDavid Majnemer const coff_symbol_type *Addr = 10244f51e51SDavid Majnemer reinterpret_cast<const coff_symbol_type *>(Ref.p); 1031d6167fdSMichael J. Spencer 104236b0ca7SDavid Majnemer assert(!checkOffset(Data, uintptr_t(Addr), sizeof(*Addr))); 1051d6167fdSMichael J. Spencer #ifndef NDEBUG 1061d6167fdSMichael J. Spencer // Verify that the symbol points to a valid entry in the symbol table. 1078ff24d25SRui Ueyama uintptr_t Offset = uintptr_t(Addr) - uintptr_t(base()); 1081d6167fdSMichael J. Spencer 10944f51e51SDavid Majnemer assert((Offset - getPointerToSymbolTable()) % sizeof(coff_symbol_type) == 0 && 11044f51e51SDavid Majnemer "Symbol did not point to the beginning of a symbol"); 1111d6167fdSMichael J. Spencer #endif 1121d6167fdSMichael J. Spencer 1138ff24d25SRui Ueyama return Addr; 1141d6167fdSMichael J. Spencer } 1151d6167fdSMichael J. Spencer 1168ff24d25SRui Ueyama const coff_section *COFFObjectFile::toSec(DataRefImpl Ref) const { 1178ff24d25SRui Ueyama const coff_section *Addr = reinterpret_cast<const coff_section*>(Ref.p); 1181d6167fdSMichael J. Spencer 1191d6167fdSMichael J. Spencer # ifndef NDEBUG 1201d6167fdSMichael J. Spencer // Verify that the section points to a valid entry in the section table. 12144f51e51SDavid Majnemer if (Addr < SectionTable || Addr >= (SectionTable + getNumberOfSections())) 1221d6167fdSMichael J. Spencer report_fatal_error("Section was outside of section table."); 1231d6167fdSMichael J. Spencer 1248ff24d25SRui Ueyama uintptr_t Offset = uintptr_t(Addr) - uintptr_t(SectionTable); 1258ff24d25SRui Ueyama assert(Offset % sizeof(coff_section) == 0 && 1261d6167fdSMichael J. Spencer "Section did not point to the beginning of a section"); 1271d6167fdSMichael J. Spencer # endif 1281d6167fdSMichael J. Spencer 1298ff24d25SRui Ueyama return Addr; 1301d6167fdSMichael J. Spencer } 1311d6167fdSMichael J. Spencer 1325e812afaSRafael Espindola void COFFObjectFile::moveSymbolNext(DataRefImpl &Ref) const { 133236b0ca7SDavid Majnemer auto End = reinterpret_cast<uintptr_t>(StringTable); 13444f51e51SDavid Majnemer if (SymbolTable16) { 13544f51e51SDavid Majnemer const coff_symbol16 *Symb = toSymb<coff_symbol16>(Ref); 1368ff24d25SRui Ueyama Symb += 1 + Symb->NumberOfAuxSymbols; 137236b0ca7SDavid Majnemer Ref.p = std::min(reinterpret_cast<uintptr_t>(Symb), End); 13844f51e51SDavid Majnemer } else if (SymbolTable32) { 13944f51e51SDavid Majnemer const coff_symbol32 *Symb = toSymb<coff_symbol32>(Ref); 14044f51e51SDavid Majnemer Symb += 1 + Symb->NumberOfAuxSymbols; 141236b0ca7SDavid Majnemer Ref.p = std::min(reinterpret_cast<uintptr_t>(Symb), End); 14244f51e51SDavid Majnemer } else { 14344f51e51SDavid Majnemer llvm_unreachable("no symbol table pointer!"); 14444f51e51SDavid Majnemer } 1451d6167fdSMichael J. Spencer } 1461d6167fdSMichael J. Spencer 14781e8b7d9SKevin Enderby Expected<StringRef> COFFObjectFile::getSymbolName(DataRefImpl Ref) const { 14844f51e51SDavid Majnemer COFFSymbolRef Symb = getCOFFSymbol(Ref); 1495d0c2ffaSRafael Espindola StringRef Result; 1505d0c2ffaSRafael Espindola std::error_code EC = getSymbolName(Symb, Result); 1515d0c2ffaSRafael Espindola if (EC) 15281e8b7d9SKevin Enderby return errorCodeToError(EC); 1535d0c2ffaSRafael Espindola return Result; 1548e90adafSMichael J. Spencer } 1558e90adafSMichael J. Spencer 156be8b0ea8SRafael Espindola uint64_t COFFObjectFile::getSymbolValueImpl(DataRefImpl Ref) const { 157be8b0ea8SRafael Espindola return getCOFFSymbol(Ref).getValue(); 158991af666SRafael Espindola } 159991af666SRafael Espindola 160*931cb65dSKevin Enderby Expected<uint64_t> COFFObjectFile::getSymbolAddress(DataRefImpl Ref) const { 161ed067c45SRafael Espindola uint64_t Result = getSymbolValue(Ref); 16244f51e51SDavid Majnemer COFFSymbolRef Symb = getCOFFSymbol(Ref); 163c7d7c6fbSDavid Majnemer int32_t SectionNumber = Symb.getSectionNumber(); 164991af666SRafael Espindola 165991af666SRafael Espindola if (Symb.isAnyUndefined() || Symb.isCommon() || 166991af666SRafael Espindola COFF::isReservedSectionNumber(SectionNumber)) 167ed067c45SRafael Espindola return Result; 16854c9f3daSRafael Espindola 1692617dcceSCraig Topper const coff_section *Section = nullptr; 170c7d7c6fbSDavid Majnemer if (std::error_code EC = getSection(SectionNumber, Section)) 171*931cb65dSKevin Enderby return errorCodeToError(EC); 172991af666SRafael Espindola Result += Section->VirtualAddress; 17347ea9eceSReid Kleckner 17447ea9eceSReid Kleckner // The section VirtualAddress does not include ImageBase, and we want to 17547ea9eceSReid Kleckner // return virtual addresses. 17621427adaSReid Kleckner Result += getImageBase(); 17747ea9eceSReid Kleckner 178ed067c45SRafael Espindola return Result; 179c7d7c6fbSDavid Majnemer } 180c7d7c6fbSDavid Majnemer 1817bd8d994SKevin Enderby Expected<SymbolRef::Type> COFFObjectFile::getSymbolType(DataRefImpl Ref) const { 18244f51e51SDavid Majnemer COFFSymbolRef Symb = getCOFFSymbol(Ref); 183c7d7c6fbSDavid Majnemer int32_t SectionNumber = Symb.getSectionNumber(); 18444f51e51SDavid Majnemer 185e834f420SPeter Collingbourne if (Symb.getComplexType() == COFF::IMAGE_SYM_DTYPE_FUNCTION) 186e834f420SPeter Collingbourne return SymbolRef::ST_Function; 1872fa80cc5SRafael Espindola if (Symb.isAnyUndefined()) 1882fa80cc5SRafael Espindola return SymbolRef::ST_Unknown; 1892fa80cc5SRafael Espindola if (Symb.isCommon()) 1902fa80cc5SRafael Espindola return SymbolRef::ST_Data; 1912fa80cc5SRafael Espindola if (Symb.isFileRecord()) 1922fa80cc5SRafael Espindola return SymbolRef::ST_File; 1932fa80cc5SRafael Espindola 1941a666e0fSDavid Majnemer // TODO: perhaps we need a new symbol type ST_Section. 1952fa80cc5SRafael Espindola if (SectionNumber == COFF::IMAGE_SYM_DEBUG || Symb.isSectionDefinition()) 1962fa80cc5SRafael Espindola return SymbolRef::ST_Debug; 1972fa80cc5SRafael Espindola 1982fa80cc5SRafael Espindola if (!COFF::isReservedSectionNumber(SectionNumber)) 1992fa80cc5SRafael Espindola return SymbolRef::ST_Data; 2002fa80cc5SRafael Espindola 2012fa80cc5SRafael Espindola return SymbolRef::ST_Other; 20275d1cf33SBenjamin Kramer } 20375d1cf33SBenjamin Kramer 20420122a43SRafael Espindola uint32_t COFFObjectFile::getSymbolFlags(DataRefImpl Ref) const { 20544f51e51SDavid Majnemer COFFSymbolRef Symb = getCOFFSymbol(Ref); 20620122a43SRafael Espindola uint32_t Result = SymbolRef::SF_None; 20775d1cf33SBenjamin Kramer 208c7d7c6fbSDavid Majnemer if (Symb.isExternal() || Symb.isWeakExternal()) 2099dc0eb42SLang Hames Result |= SymbolRef::SF_Global; 2101df4b84dSDavid Meyer 211c7d7c6fbSDavid Majnemer if (Symb.isWeakExternal()) 2121df4b84dSDavid Meyer Result |= SymbolRef::SF_Weak; 2131df4b84dSDavid Meyer 21444f51e51SDavid Majnemer if (Symb.getSectionNumber() == COFF::IMAGE_SYM_ABSOLUTE) 2151df4b84dSDavid Meyer Result |= SymbolRef::SF_Absolute; 2161df4b84dSDavid Meyer 217c7d7c6fbSDavid Majnemer if (Symb.isFileRecord()) 218c7d7c6fbSDavid Majnemer Result |= SymbolRef::SF_FormatSpecific; 219c7d7c6fbSDavid Majnemer 220c7d7c6fbSDavid Majnemer if (Symb.isSectionDefinition()) 221c7d7c6fbSDavid Majnemer Result |= SymbolRef::SF_FormatSpecific; 222c7d7c6fbSDavid Majnemer 223c7d7c6fbSDavid Majnemer if (Symb.isCommon()) 224c7d7c6fbSDavid Majnemer Result |= SymbolRef::SF_Common; 225c7d7c6fbSDavid Majnemer 226c7d7c6fbSDavid Majnemer if (Symb.isAnyUndefined()) 227c7d7c6fbSDavid Majnemer Result |= SymbolRef::SF_Undefined; 228c7d7c6fbSDavid Majnemer 22920122a43SRafael Espindola return Result; 23001759754SMichael J. Spencer } 23101759754SMichael J. Spencer 232d7a32ea4SRafael Espindola uint64_t COFFObjectFile::getCommonSymbolSizeImpl(DataRefImpl Ref) const { 233c7d7c6fbSDavid Majnemer COFFSymbolRef Symb = getCOFFSymbol(Ref); 2345eb02e45SRafael Espindola return Symb.getValue(); 2358e90adafSMichael J. Spencer } 2368e90adafSMichael J. Spencer 2377bd8d994SKevin Enderby Expected<section_iterator> 2388bab889bSRafael Espindola COFFObjectFile::getSymbolSection(DataRefImpl Ref) const { 23944f51e51SDavid Majnemer COFFSymbolRef Symb = getCOFFSymbol(Ref); 2408bab889bSRafael Espindola if (COFF::isReservedSectionNumber(Symb.getSectionNumber())) 2418bab889bSRafael Espindola return section_end(); 2422617dcceSCraig Topper const coff_section *Sec = nullptr; 24344f51e51SDavid Majnemer if (std::error_code EC = getSection(Symb.getSectionNumber(), Sec)) 2447bd8d994SKevin Enderby return errorCodeToError(EC); 2458bab889bSRafael Espindola DataRefImpl Ret; 2468bab889bSRafael Espindola Ret.p = reinterpret_cast<uintptr_t>(Sec); 2478bab889bSRafael Espindola return section_iterator(SectionRef(Ret, this)); 24832173153SMichael J. Spencer } 24932173153SMichael J. Spencer 2506bf32210SRafael Espindola unsigned COFFObjectFile::getSymbolSectionID(SymbolRef Sym) const { 2516bf32210SRafael Espindola COFFSymbolRef Symb = getCOFFSymbol(Sym.getRawDataRefImpl()); 2526bf32210SRafael Espindola return Symb.getSectionNumber(); 2536bf32210SRafael Espindola } 2546bf32210SRafael Espindola 2555e812afaSRafael Espindola void COFFObjectFile::moveSectionNext(DataRefImpl &Ref) const { 2568ff24d25SRui Ueyama const coff_section *Sec = toSec(Ref); 2578ff24d25SRui Ueyama Sec += 1; 2588ff24d25SRui Ueyama Ref.p = reinterpret_cast<uintptr_t>(Sec); 2598e90adafSMichael J. Spencer } 2608e90adafSMichael J. Spencer 261db4ed0bdSRafael Espindola std::error_code COFFObjectFile::getSectionName(DataRefImpl Ref, 2621d6167fdSMichael J. Spencer StringRef &Result) const { 2638ff24d25SRui Ueyama const coff_section *Sec = toSec(Ref); 2648ff24d25SRui Ueyama return getSectionName(Sec, Result); 2658e90adafSMichael J. Spencer } 2668e90adafSMichael J. Spencer 26780291274SRafael Espindola uint64_t COFFObjectFile::getSectionAddress(DataRefImpl Ref) const { 2688ff24d25SRui Ueyama const coff_section *Sec = toSec(Ref); 2697c6a071bSDavid Majnemer uint64_t Result = Sec->VirtualAddress; 2707c6a071bSDavid Majnemer 2717c6a071bSDavid Majnemer // The section VirtualAddress does not include ImageBase, and we want to 2727c6a071bSDavid Majnemer // return virtual addresses. 27321427adaSReid Kleckner Result += getImageBase(); 2747c6a071bSDavid Majnemer return Result; 2758e90adafSMichael J. Spencer } 2768e90adafSMichael J. Spencer 27780291274SRafael Espindola uint64_t COFFObjectFile::getSectionSize(DataRefImpl Ref) const { 278a9ee5c06SDavid Majnemer return getSectionSize(toSec(Ref)); 2798e90adafSMichael J. Spencer } 2808e90adafSMichael J. Spencer 281db4ed0bdSRafael Espindola std::error_code COFFObjectFile::getSectionContents(DataRefImpl Ref, 2821d6167fdSMichael J. Spencer StringRef &Result) const { 2838ff24d25SRui Ueyama const coff_section *Sec = toSec(Ref); 2849da9e693SMichael J. Spencer ArrayRef<uint8_t> Res; 285db4ed0bdSRafael Espindola std::error_code EC = getSectionContents(Sec, Res); 2869da9e693SMichael J. Spencer Result = StringRef(reinterpret_cast<const char*>(Res.data()), Res.size()); 2879da9e693SMichael J. Spencer return EC; 2888e90adafSMichael J. Spencer } 2898e90adafSMichael J. Spencer 29080291274SRafael Espindola uint64_t COFFObjectFile::getSectionAlignment(DataRefImpl Ref) const { 2918ff24d25SRui Ueyama const coff_section *Sec = toSec(Ref); 292511391feSDavid Majnemer return Sec->getAlignment(); 2937989460aSMichael J. Spencer } 2947989460aSMichael J. Spencer 295401e4e57SGeorge Rimar bool COFFObjectFile::isSectionCompressed(DataRefImpl Sec) const { 296401e4e57SGeorge Rimar return false; 297401e4e57SGeorge Rimar } 298401e4e57SGeorge Rimar 29980291274SRafael Espindola bool COFFObjectFile::isSectionText(DataRefImpl Ref) const { 3008ff24d25SRui Ueyama const coff_section *Sec = toSec(Ref); 30180291274SRafael Espindola return Sec->Characteristics & COFF::IMAGE_SCN_CNT_CODE; 3028e90adafSMichael J. Spencer } 3038e90adafSMichael J. Spencer 30480291274SRafael Espindola bool COFFObjectFile::isSectionData(DataRefImpl Ref) const { 3058ff24d25SRui Ueyama const coff_section *Sec = toSec(Ref); 30680291274SRafael Espindola return Sec->Characteristics & COFF::IMAGE_SCN_CNT_INITIALIZED_DATA; 307800619f2SMichael J. Spencer } 308800619f2SMichael J. Spencer 30980291274SRafael Espindola bool COFFObjectFile::isSectionBSS(DataRefImpl Ref) const { 3108ff24d25SRui Ueyama const coff_section *Sec = toSec(Ref); 3111a666e0fSDavid Majnemer const uint32_t BssFlags = COFF::IMAGE_SCN_CNT_UNINITIALIZED_DATA | 3121a666e0fSDavid Majnemer COFF::IMAGE_SCN_MEM_READ | 3131a666e0fSDavid Majnemer COFF::IMAGE_SCN_MEM_WRITE; 3141a666e0fSDavid Majnemer return (Sec->Characteristics & BssFlags) == BssFlags; 315800619f2SMichael J. Spencer } 316800619f2SMichael J. Spencer 3176bf32210SRafael Espindola unsigned COFFObjectFile::getSectionID(SectionRef Sec) const { 3186bf32210SRafael Espindola uintptr_t Offset = 3196bf32210SRafael Espindola uintptr_t(Sec.getRawDataRefImpl().p) - uintptr_t(SectionTable); 3206bf32210SRafael Espindola assert((Offset % sizeof(coff_section)) == 0); 3216bf32210SRafael Espindola return (Offset / sizeof(coff_section)) + 1; 3226bf32210SRafael Espindola } 3236bf32210SRafael Espindola 32480291274SRafael Espindola bool COFFObjectFile::isSectionVirtual(DataRefImpl Ref) const { 3258ff24d25SRui Ueyama const coff_section *Sec = toSec(Ref); 3261a666e0fSDavid Majnemer // In COFF, a virtual section won't have any in-file 3271a666e0fSDavid Majnemer // content, so the file pointer to the content will be zero. 3281a666e0fSDavid Majnemer return Sec->PointerToRawData == 0; 3292138ef6dSPreston Gurd } 3302138ef6dSPreston Gurd 331e830c60dSDavid Majnemer static uint32_t getNumberOfRelocations(const coff_section *Sec, 332e830c60dSDavid Majnemer MemoryBufferRef M, const uint8_t *base) { 333e830c60dSDavid Majnemer // The field for the number of relocations in COFF section table is only 334e830c60dSDavid Majnemer // 16-bit wide. If a section has more than 65535 relocations, 0xFFFF is set to 335e830c60dSDavid Majnemer // NumberOfRelocations field, and the actual relocation count is stored in the 336e830c60dSDavid Majnemer // VirtualAddress field in the first relocation entry. 337e830c60dSDavid Majnemer if (Sec->hasExtendedRelocations()) { 338e830c60dSDavid Majnemer const coff_relocation *FirstReloc; 339e830c60dSDavid Majnemer if (getObject(FirstReloc, M, reinterpret_cast<const coff_relocation*>( 340e830c60dSDavid Majnemer base + Sec->PointerToRelocations))) 341e830c60dSDavid Majnemer return 0; 34298fe58a3SRui Ueyama // -1 to exclude this first relocation entry. 34398fe58a3SRui Ueyama return FirstReloc->VirtualAddress - 1; 344e830c60dSDavid Majnemer } 345e830c60dSDavid Majnemer return Sec->NumberOfRelocations; 346e830c60dSDavid Majnemer } 347e830c60dSDavid Majnemer 34894751be7SDavid Majnemer static const coff_relocation * 34994751be7SDavid Majnemer getFirstReloc(const coff_section *Sec, MemoryBufferRef M, const uint8_t *Base) { 35094751be7SDavid Majnemer uint64_t NumRelocs = getNumberOfRelocations(Sec, M, Base); 35194751be7SDavid Majnemer if (!NumRelocs) 35294751be7SDavid Majnemer return nullptr; 353827c8a2bSRui Ueyama auto begin = reinterpret_cast<const coff_relocation *>( 35494751be7SDavid Majnemer Base + Sec->PointerToRelocations); 355827c8a2bSRui Ueyama if (Sec->hasExtendedRelocations()) { 356827c8a2bSRui Ueyama // Skip the first relocation entry repurposed to store the number of 357827c8a2bSRui Ueyama // relocations. 358827c8a2bSRui Ueyama begin++; 359827c8a2bSRui Ueyama } 36094751be7SDavid Majnemer if (checkOffset(M, uintptr_t(begin), sizeof(coff_relocation) * NumRelocs)) 36194751be7SDavid Majnemer return nullptr; 36294751be7SDavid Majnemer return begin; 363827c8a2bSRui Ueyama } 36494751be7SDavid Majnemer 36594751be7SDavid Majnemer relocation_iterator COFFObjectFile::section_rel_begin(DataRefImpl Ref) const { 36694751be7SDavid Majnemer const coff_section *Sec = toSec(Ref); 36794751be7SDavid Majnemer const coff_relocation *begin = getFirstReloc(Sec, Data, base()); 36876d650e8SRafael Espindola if (begin && Sec->VirtualAddress != 0) 36976d650e8SRafael Espindola report_fatal_error("Sections with relocations should have an address of 0"); 37094751be7SDavid Majnemer DataRefImpl Ret; 37194751be7SDavid Majnemer Ret.p = reinterpret_cast<uintptr_t>(begin); 3728ff24d25SRui Ueyama return relocation_iterator(RelocationRef(Ret, this)); 373e5fd0047SMichael J. Spencer } 374e5fd0047SMichael J. Spencer 3758ff24d25SRui Ueyama relocation_iterator COFFObjectFile::section_rel_end(DataRefImpl Ref) const { 3768ff24d25SRui Ueyama const coff_section *Sec = toSec(Ref); 37794751be7SDavid Majnemer const coff_relocation *I = getFirstReloc(Sec, Data, base()); 37894751be7SDavid Majnemer if (I) 37994751be7SDavid Majnemer I += getNumberOfRelocations(Sec, Data, base()); 3808ff24d25SRui Ueyama DataRefImpl Ret; 38194751be7SDavid Majnemer Ret.p = reinterpret_cast<uintptr_t>(I); 3828ff24d25SRui Ueyama return relocation_iterator(RelocationRef(Ret, this)); 383e5fd0047SMichael J. Spencer } 384e5fd0047SMichael J. Spencer 385c2bed429SRui Ueyama // Initialize the pointer to the symbol table. 386db4ed0bdSRafael Espindola std::error_code COFFObjectFile::initSymbolTablePtr() { 38744f51e51SDavid Majnemer if (COFFHeader) 388236b0ca7SDavid Majnemer if (std::error_code EC = getObject( 389236b0ca7SDavid Majnemer SymbolTable16, Data, base() + getPointerToSymbolTable(), 390236b0ca7SDavid Majnemer (uint64_t)getNumberOfSymbols() * getSymbolTableEntrySize())) 39144f51e51SDavid Majnemer return EC; 39244f51e51SDavid Majnemer 39344f51e51SDavid Majnemer if (COFFBigObjHeader) 394236b0ca7SDavid Majnemer if (std::error_code EC = getObject( 395236b0ca7SDavid Majnemer SymbolTable32, Data, base() + getPointerToSymbolTable(), 396236b0ca7SDavid Majnemer (uint64_t)getNumberOfSymbols() * getSymbolTableEntrySize())) 3978ff24d25SRui Ueyama return EC; 398c2bed429SRui Ueyama 399c2bed429SRui Ueyama // Find string table. The first four byte of the string table contains the 400c2bed429SRui Ueyama // total size of the string table, including the size field itself. If the 401c2bed429SRui Ueyama // string table is empty, the value of the first four byte would be 4. 402f69b0585SDavid Majnemer uint32_t StringTableOffset = getPointerToSymbolTable() + 40344f51e51SDavid Majnemer getNumberOfSymbols() * getSymbolTableEntrySize(); 404f69b0585SDavid Majnemer const uint8_t *StringTableAddr = base() + StringTableOffset; 405c2bed429SRui Ueyama const ulittle32_t *StringTableSizePtr; 40648af1c2aSRafael Espindola if (std::error_code EC = getObject(StringTableSizePtr, Data, StringTableAddr)) 4078ff24d25SRui Ueyama return EC; 408c2bed429SRui Ueyama StringTableSize = *StringTableSizePtr; 409db4ed0bdSRafael Espindola if (std::error_code EC = 41048af1c2aSRafael Espindola getObject(StringTable, Data, StringTableAddr, StringTableSize)) 4118ff24d25SRui Ueyama return EC; 412c2bed429SRui Ueyama 413773a5795SNico Rieck // Treat table sizes < 4 as empty because contrary to the PECOFF spec, some 414773a5795SNico Rieck // tools like cvtres write a size of 0 for an empty table instead of 4. 415773a5795SNico Rieck if (StringTableSize < 4) 416773a5795SNico Rieck StringTableSize = 4; 417773a5795SNico Rieck 418c2bed429SRui Ueyama // Check that the string table is null terminated if has any in it. 419773a5795SNico Rieck if (StringTableSize > 4 && StringTable[StringTableSize - 1] != 0) 420c2bed429SRui Ueyama return object_error::parse_failed; 4217d099195SRui Ueyama return std::error_code(); 422c2bed429SRui Ueyama } 423c2bed429SRui Ueyama 42421427adaSReid Kleckner uint64_t COFFObjectFile::getImageBase() const { 425e94fef7bSReid Kleckner if (PE32Header) 42621427adaSReid Kleckner return PE32Header->ImageBase; 427e94fef7bSReid Kleckner else if (PE32PlusHeader) 42821427adaSReid Kleckner return PE32PlusHeader->ImageBase; 42921427adaSReid Kleckner // This actually comes up in practice. 43021427adaSReid Kleckner return 0; 431e94fef7bSReid Kleckner } 432e94fef7bSReid Kleckner 433215a586cSRui Ueyama // Returns the file offset for the given VA. 434db4ed0bdSRafael Espindola std::error_code COFFObjectFile::getVaPtr(uint64_t Addr, uintptr_t &Res) const { 43521427adaSReid Kleckner uint64_t ImageBase = getImageBase(); 436b7a40081SRui Ueyama uint64_t Rva = Addr - ImageBase; 437b7a40081SRui Ueyama assert(Rva <= UINT32_MAX); 438b7a40081SRui Ueyama return getRvaPtr((uint32_t)Rva, Res); 439215a586cSRui Ueyama } 440215a586cSRui Ueyama 441c2bed429SRui Ueyama // Returns the file offset for the given RVA. 442db4ed0bdSRafael Espindola std::error_code COFFObjectFile::getRvaPtr(uint32_t Addr, uintptr_t &Res) const { 44327dc8394SAlexey Samsonov for (const SectionRef &S : sections()) { 44427dc8394SAlexey Samsonov const coff_section *Section = getCOFFSection(S); 445c2bed429SRui Ueyama uint32_t SectionStart = Section->VirtualAddress; 446c2bed429SRui Ueyama uint32_t SectionEnd = Section->VirtualAddress + Section->VirtualSize; 447215a586cSRui Ueyama if (SectionStart <= Addr && Addr < SectionEnd) { 448215a586cSRui Ueyama uint32_t Offset = Addr - SectionStart; 449c2bed429SRui Ueyama Res = uintptr_t(base()) + Section->PointerToRawData + Offset; 4507d099195SRui Ueyama return std::error_code(); 451c2bed429SRui Ueyama } 452c2bed429SRui Ueyama } 453c2bed429SRui Ueyama return object_error::parse_failed; 454c2bed429SRui Ueyama } 455c2bed429SRui Ueyama 4562da433eaSReid Kleckner std::error_code 4572da433eaSReid Kleckner COFFObjectFile::getRvaAndSizeAsBytes(uint32_t RVA, uint32_t Size, 4582da433eaSReid Kleckner ArrayRef<uint8_t> &Contents) const { 4592da433eaSReid Kleckner for (const SectionRef &S : sections()) { 4602da433eaSReid Kleckner const coff_section *Section = getCOFFSection(S); 4612da433eaSReid Kleckner uint32_t SectionStart = Section->VirtualAddress; 4622da433eaSReid Kleckner // Check if this RVA is within the section bounds. Be careful about integer 4632da433eaSReid Kleckner // overflow. 4642da433eaSReid Kleckner uint32_t OffsetIntoSection = RVA - SectionStart; 4652da433eaSReid Kleckner if (SectionStart <= RVA && OffsetIntoSection < Section->VirtualSize && 4662da433eaSReid Kleckner Size <= Section->VirtualSize - OffsetIntoSection) { 4672da433eaSReid Kleckner uintptr_t Begin = 4682da433eaSReid Kleckner uintptr_t(base()) + Section->PointerToRawData + OffsetIntoSection; 4692da433eaSReid Kleckner Contents = 4702da433eaSReid Kleckner ArrayRef<uint8_t>(reinterpret_cast<const uint8_t *>(Begin), Size); 4712da433eaSReid Kleckner return std::error_code(); 4722da433eaSReid Kleckner } 4732da433eaSReid Kleckner } 4742da433eaSReid Kleckner return object_error::parse_failed; 4752da433eaSReid Kleckner } 4762da433eaSReid Kleckner 477c2bed429SRui Ueyama // Returns hint and name fields, assuming \p Rva is pointing to a Hint/Name 478c2bed429SRui Ueyama // table entry. 479db4ed0bdSRafael Espindola std::error_code COFFObjectFile::getHintName(uint32_t Rva, uint16_t &Hint, 480db4ed0bdSRafael Espindola StringRef &Name) const { 481c2bed429SRui Ueyama uintptr_t IntPtr = 0; 482db4ed0bdSRafael Espindola if (std::error_code EC = getRvaPtr(Rva, IntPtr)) 4838ff24d25SRui Ueyama return EC; 484c2bed429SRui Ueyama const uint8_t *Ptr = reinterpret_cast<const uint8_t *>(IntPtr); 485c2bed429SRui Ueyama Hint = *reinterpret_cast<const ulittle16_t *>(Ptr); 486c2bed429SRui Ueyama Name = StringRef(reinterpret_cast<const char *>(Ptr + 2)); 4877d099195SRui Ueyama return std::error_code(); 488c2bed429SRui Ueyama } 489c2bed429SRui Ueyama 4902da433eaSReid Kleckner std::error_code COFFObjectFile::getDebugPDBInfo(const debug_directory *DebugDir, 4912da433eaSReid Kleckner const debug_pdb_info *&PDBInfo, 4922da433eaSReid Kleckner StringRef &PDBFileName) const { 4932da433eaSReid Kleckner ArrayRef<uint8_t> InfoBytes; 4942da433eaSReid Kleckner if (std::error_code EC = getRvaAndSizeAsBytes( 4952da433eaSReid Kleckner DebugDir->AddressOfRawData, DebugDir->SizeOfData, InfoBytes)) 4962da433eaSReid Kleckner return EC; 4972da433eaSReid Kleckner if (InfoBytes.size() < sizeof(debug_pdb_info) + 1) 4982da433eaSReid Kleckner return object_error::parse_failed; 4992da433eaSReid Kleckner PDBInfo = reinterpret_cast<const debug_pdb_info *>(InfoBytes.data()); 5002da433eaSReid Kleckner InfoBytes = InfoBytes.drop_front(sizeof(debug_pdb_info)); 5012da433eaSReid Kleckner PDBFileName = StringRef(reinterpret_cast<const char *>(InfoBytes.data()), 5022da433eaSReid Kleckner InfoBytes.size()); 5032da433eaSReid Kleckner // Truncate the name at the first null byte. Ignore any padding. 5042da433eaSReid Kleckner PDBFileName = PDBFileName.split('\0').first; 5052da433eaSReid Kleckner return std::error_code(); 5062da433eaSReid Kleckner } 5072da433eaSReid Kleckner 508f27f3f84SReid Kleckner std::error_code COFFObjectFile::getDebugPDBInfo(const debug_pdb_info *&PDBInfo, 509f27f3f84SReid Kleckner StringRef &PDBFileName) const { 510f27f3f84SReid Kleckner for (const debug_directory &D : debug_directories()) 511f27f3f84SReid Kleckner if (D.Type == COFF::IMAGE_DEBUG_TYPE_CODEVIEW) 512f27f3f84SReid Kleckner return getDebugPDBInfo(&D, PDBInfo, PDBFileName); 513f27f3f84SReid Kleckner // If we get here, there is no PDB info to return. 514f27f3f84SReid Kleckner PDBInfo = nullptr; 515f27f3f84SReid Kleckner PDBFileName = StringRef(); 516f27f3f84SReid Kleckner return std::error_code(); 517f27f3f84SReid Kleckner } 518f27f3f84SReid Kleckner 519c2bed429SRui Ueyama // Find the import table. 520db4ed0bdSRafael Espindola std::error_code COFFObjectFile::initImportTablePtr() { 521c2bed429SRui Ueyama // First, we get the RVA of the import table. If the file lacks a pointer to 522c2bed429SRui Ueyama // the import table, do nothing. 523c2bed429SRui Ueyama const data_directory *DataEntry; 524c2bed429SRui Ueyama if (getDataDirectory(COFF::IMPORT_TABLE, DataEntry)) 5257d099195SRui Ueyama return std::error_code(); 526c2bed429SRui Ueyama 527c2bed429SRui Ueyama // Do nothing if the pointer to import table is NULL. 528c2bed429SRui Ueyama if (DataEntry->RelativeVirtualAddress == 0) 5297d099195SRui Ueyama return std::error_code(); 530c2bed429SRui Ueyama 531c2bed429SRui Ueyama uint32_t ImportTableRva = DataEntry->RelativeVirtualAddress; 5321e152d5eSRui Ueyama // -1 because the last entry is the null entry. 533c2bed429SRui Ueyama NumberOfImportDirectory = DataEntry->Size / 5341e152d5eSRui Ueyama sizeof(import_directory_table_entry) - 1; 535c2bed429SRui Ueyama 536c2bed429SRui Ueyama // Find the section that contains the RVA. This is needed because the RVA is 537c2bed429SRui Ueyama // the import table's memory address which is different from its file offset. 538c2bed429SRui Ueyama uintptr_t IntPtr = 0; 539db4ed0bdSRafael Espindola if (std::error_code EC = getRvaPtr(ImportTableRva, IntPtr)) 5408ff24d25SRui Ueyama return EC; 541c2bed429SRui Ueyama ImportDirectory = reinterpret_cast< 542c2bed429SRui Ueyama const import_directory_table_entry *>(IntPtr); 5437d099195SRui Ueyama return std::error_code(); 544ad882ba8SRui Ueyama } 545c2bed429SRui Ueyama 54615d99359SRui Ueyama // Initializes DelayImportDirectory and NumberOfDelayImportDirectory. 54715d99359SRui Ueyama std::error_code COFFObjectFile::initDelayImportTablePtr() { 54815d99359SRui Ueyama const data_directory *DataEntry; 54915d99359SRui Ueyama if (getDataDirectory(COFF::DELAY_IMPORT_DESCRIPTOR, DataEntry)) 5507d099195SRui Ueyama return std::error_code(); 55115d99359SRui Ueyama if (DataEntry->RelativeVirtualAddress == 0) 5527d099195SRui Ueyama return std::error_code(); 55315d99359SRui Ueyama 55415d99359SRui Ueyama uint32_t RVA = DataEntry->RelativeVirtualAddress; 55515d99359SRui Ueyama NumberOfDelayImportDirectory = DataEntry->Size / 55615d99359SRui Ueyama sizeof(delay_import_directory_table_entry) - 1; 55715d99359SRui Ueyama 55815d99359SRui Ueyama uintptr_t IntPtr = 0; 55915d99359SRui Ueyama if (std::error_code EC = getRvaPtr(RVA, IntPtr)) 56015d99359SRui Ueyama return EC; 56115d99359SRui Ueyama DelayImportDirectory = reinterpret_cast< 56215d99359SRui Ueyama const delay_import_directory_table_entry *>(IntPtr); 5637d099195SRui Ueyama return std::error_code(); 56415d99359SRui Ueyama } 56515d99359SRui Ueyama 566ad882ba8SRui Ueyama // Find the export table. 567db4ed0bdSRafael Espindola std::error_code COFFObjectFile::initExportTablePtr() { 568ad882ba8SRui Ueyama // First, we get the RVA of the export table. If the file lacks a pointer to 569ad882ba8SRui Ueyama // the export table, do nothing. 570ad882ba8SRui Ueyama const data_directory *DataEntry; 571ad882ba8SRui Ueyama if (getDataDirectory(COFF::EXPORT_TABLE, DataEntry)) 5727d099195SRui Ueyama return std::error_code(); 573ad882ba8SRui Ueyama 574ad882ba8SRui Ueyama // Do nothing if the pointer to export table is NULL. 575ad882ba8SRui Ueyama if (DataEntry->RelativeVirtualAddress == 0) 5767d099195SRui Ueyama return std::error_code(); 577ad882ba8SRui Ueyama 578ad882ba8SRui Ueyama uint32_t ExportTableRva = DataEntry->RelativeVirtualAddress; 579ad882ba8SRui Ueyama uintptr_t IntPtr = 0; 580db4ed0bdSRafael Espindola if (std::error_code EC = getRvaPtr(ExportTableRva, IntPtr)) 581ad882ba8SRui Ueyama return EC; 58224fc2d64SRui Ueyama ExportDirectory = 58324fc2d64SRui Ueyama reinterpret_cast<const export_directory_table_entry *>(IntPtr); 5847d099195SRui Ueyama return std::error_code(); 585c2bed429SRui Ueyama } 586c2bed429SRui Ueyama 58774e85130SRui Ueyama std::error_code COFFObjectFile::initBaseRelocPtr() { 58874e85130SRui Ueyama const data_directory *DataEntry; 58974e85130SRui Ueyama if (getDataDirectory(COFF::BASE_RELOCATION_TABLE, DataEntry)) 5907d099195SRui Ueyama return std::error_code(); 59174e85130SRui Ueyama if (DataEntry->RelativeVirtualAddress == 0) 5927d099195SRui Ueyama return std::error_code(); 59374e85130SRui Ueyama 59474e85130SRui Ueyama uintptr_t IntPtr = 0; 59574e85130SRui Ueyama if (std::error_code EC = getRvaPtr(DataEntry->RelativeVirtualAddress, IntPtr)) 59674e85130SRui Ueyama return EC; 59774e85130SRui Ueyama BaseRelocHeader = reinterpret_cast<const coff_base_reloc_block_header *>( 59874e85130SRui Ueyama IntPtr); 59974e85130SRui Ueyama BaseRelocEnd = reinterpret_cast<coff_base_reloc_block_header *>( 60074e85130SRui Ueyama IntPtr + DataEntry->Size); 6017d099195SRui Ueyama return std::error_code(); 60274e85130SRui Ueyama } 60374e85130SRui Ueyama 6042da433eaSReid Kleckner std::error_code COFFObjectFile::initDebugDirectoryPtr() { 6052da433eaSReid Kleckner // Get the RVA of the debug directory. Do nothing if it does not exist. 6062da433eaSReid Kleckner const data_directory *DataEntry; 6072da433eaSReid Kleckner if (getDataDirectory(COFF::DEBUG_DIRECTORY, DataEntry)) 6082da433eaSReid Kleckner return std::error_code(); 6092da433eaSReid Kleckner 6102da433eaSReid Kleckner // Do nothing if the RVA is NULL. 6112da433eaSReid Kleckner if (DataEntry->RelativeVirtualAddress == 0) 6122da433eaSReid Kleckner return std::error_code(); 6132da433eaSReid Kleckner 6142da433eaSReid Kleckner // Check that the size is a multiple of the entry size. 6152da433eaSReid Kleckner if (DataEntry->Size % sizeof(debug_directory) != 0) 6162da433eaSReid Kleckner return object_error::parse_failed; 6172da433eaSReid Kleckner 6182da433eaSReid Kleckner uintptr_t IntPtr = 0; 6192da433eaSReid Kleckner if (std::error_code EC = getRvaPtr(DataEntry->RelativeVirtualAddress, IntPtr)) 6202da433eaSReid Kleckner return EC; 6212da433eaSReid Kleckner DebugDirectoryBegin = reinterpret_cast<const debug_directory *>(IntPtr); 6222da433eaSReid Kleckner if (std::error_code EC = getRvaPtr( 6232da433eaSReid Kleckner DataEntry->RelativeVirtualAddress + DataEntry->Size, IntPtr)) 6242da433eaSReid Kleckner return EC; 6252da433eaSReid Kleckner DebugDirectoryEnd = reinterpret_cast<const debug_directory *>(IntPtr); 6262da433eaSReid Kleckner return std::error_code(); 6272da433eaSReid Kleckner } 6282da433eaSReid Kleckner 62948af1c2aSRafael Espindola COFFObjectFile::COFFObjectFile(MemoryBufferRef Object, std::error_code &EC) 63048af1c2aSRafael Espindola : ObjectFile(Binary::ID_COFF, Object), COFFHeader(nullptr), 63144f51e51SDavid Majnemer COFFBigObjHeader(nullptr), PE32Header(nullptr), PE32PlusHeader(nullptr), 63244f51e51SDavid Majnemer DataDirectory(nullptr), SectionTable(nullptr), SymbolTable16(nullptr), 63344f51e51SDavid Majnemer SymbolTable32(nullptr), StringTable(nullptr), StringTableSize(0), 63444f51e51SDavid Majnemer ImportDirectory(nullptr), NumberOfImportDirectory(0), 63515d99359SRui Ueyama DelayImportDirectory(nullptr), NumberOfDelayImportDirectory(0), 6362da433eaSReid Kleckner ExportDirectory(nullptr), BaseRelocHeader(nullptr), BaseRelocEnd(nullptr), 6372da433eaSReid Kleckner DebugDirectoryBegin(nullptr), DebugDirectoryEnd(nullptr) { 6381d6167fdSMichael J. Spencer // Check that we at least have enough room for a header. 63948af1c2aSRafael Espindola if (!checkSize(Data, EC, sizeof(coff_file_header))) 640c3f9b5a5SRafael Espindola return; 641ee066fc4SEric Christopher 64282ebd8e3SRui Ueyama // The current location in the file where we are looking at. 64382ebd8e3SRui Ueyama uint64_t CurPtr = 0; 64482ebd8e3SRui Ueyama 64582ebd8e3SRui Ueyama // PE header is optional and is present only in executables. If it exists, 64682ebd8e3SRui Ueyama // it is placed right after COFF header. 6478ff24d25SRui Ueyama bool HasPEHeader = false; 648ee066fc4SEric Christopher 6491d6167fdSMichael J. Spencer // Check if this is a PE/COFF file. 65050267222SDavid Majnemer if (checkSize(Data, EC, sizeof(dos_header) + sizeof(COFF::PEMagic))) { 651ee066fc4SEric Christopher // PE/COFF, seek through MS-DOS compatibility stub and 4-byte 652ee066fc4SEric Christopher // PE signature to find 'normal' COFF header. 65350267222SDavid Majnemer const auto *DH = reinterpret_cast<const dos_header *>(base()); 65450267222SDavid Majnemer if (DH->Magic[0] == 'M' && DH->Magic[1] == 'Z') { 65550267222SDavid Majnemer CurPtr = DH->AddressOfNewExeHeader; 65682ebd8e3SRui Ueyama // Check the PE magic bytes. ("PE\0\0") 65750267222SDavid Majnemer if (memcmp(base() + CurPtr, COFF::PEMagic, sizeof(COFF::PEMagic)) != 0) { 6588ff24d25SRui Ueyama EC = object_error::parse_failed; 6591d6167fdSMichael J. Spencer return; 6601d6167fdSMichael J. Spencer } 66144f51e51SDavid Majnemer CurPtr += sizeof(COFF::PEMagic); // Skip the PE magic bytes. 6628ff24d25SRui Ueyama HasPEHeader = true; 663ee066fc4SEric Christopher } 66450267222SDavid Majnemer } 665ee066fc4SEric Christopher 66648af1c2aSRafael Espindola if ((EC = getObject(COFFHeader, Data, base() + CurPtr))) 6671d6167fdSMichael J. Spencer return; 66844f51e51SDavid Majnemer 66944f51e51SDavid Majnemer // It might be a bigobj file, let's check. Note that COFF bigobj and COFF 67044f51e51SDavid Majnemer // import libraries share a common prefix but bigobj is more restrictive. 67144f51e51SDavid Majnemer if (!HasPEHeader && COFFHeader->Machine == COFF::IMAGE_FILE_MACHINE_UNKNOWN && 67244f51e51SDavid Majnemer COFFHeader->NumberOfSections == uint16_t(0xffff) && 67344f51e51SDavid Majnemer checkSize(Data, EC, sizeof(coff_bigobj_file_header))) { 67444f51e51SDavid Majnemer if ((EC = getObject(COFFBigObjHeader, Data, base() + CurPtr))) 67544f51e51SDavid Majnemer return; 67644f51e51SDavid Majnemer 67744f51e51SDavid Majnemer // Verify that we are dealing with bigobj. 67844f51e51SDavid Majnemer if (COFFBigObjHeader->Version >= COFF::BigObjHeader::MinBigObjectVersion && 67944f51e51SDavid Majnemer std::memcmp(COFFBigObjHeader->UUID, COFF::BigObjMagic, 68044f51e51SDavid Majnemer sizeof(COFF::BigObjMagic)) == 0) { 68144f51e51SDavid Majnemer COFFHeader = nullptr; 68244f51e51SDavid Majnemer CurPtr += sizeof(coff_bigobj_file_header); 68344f51e51SDavid Majnemer } else { 68444f51e51SDavid Majnemer // It's not a bigobj. 68544f51e51SDavid Majnemer COFFBigObjHeader = nullptr; 68644f51e51SDavid Majnemer } 68744f51e51SDavid Majnemer } 68844f51e51SDavid Majnemer if (COFFHeader) { 68944f51e51SDavid Majnemer // The prior checkSize call may have failed. This isn't a hard error 69044f51e51SDavid Majnemer // because we were just trying to sniff out bigobj. 6917d099195SRui Ueyama EC = std::error_code(); 69282ebd8e3SRui Ueyama CurPtr += sizeof(coff_file_header); 69382ebd8e3SRui Ueyama 69444f51e51SDavid Majnemer if (COFFHeader->isImportLibrary()) 69544f51e51SDavid Majnemer return; 69644f51e51SDavid Majnemer } 69744f51e51SDavid Majnemer 6988ff24d25SRui Ueyama if (HasPEHeader) { 69910ed9ddcSRui Ueyama const pe32_header *Header; 70048af1c2aSRafael Espindola if ((EC = getObject(Header, Data, base() + CurPtr))) 70182ebd8e3SRui Ueyama return; 70210ed9ddcSRui Ueyama 70310ed9ddcSRui Ueyama const uint8_t *DataDirAddr; 70410ed9ddcSRui Ueyama uint64_t DataDirSize; 70550267222SDavid Majnemer if (Header->Magic == COFF::PE32Header::PE32) { 70610ed9ddcSRui Ueyama PE32Header = Header; 70710ed9ddcSRui Ueyama DataDirAddr = base() + CurPtr + sizeof(pe32_header); 70810ed9ddcSRui Ueyama DataDirSize = sizeof(data_directory) * PE32Header->NumberOfRvaAndSize; 70950267222SDavid Majnemer } else if (Header->Magic == COFF::PE32Header::PE32_PLUS) { 71010ed9ddcSRui Ueyama PE32PlusHeader = reinterpret_cast<const pe32plus_header *>(Header); 71110ed9ddcSRui Ueyama DataDirAddr = base() + CurPtr + sizeof(pe32plus_header); 71210ed9ddcSRui Ueyama DataDirSize = sizeof(data_directory) * PE32PlusHeader->NumberOfRvaAndSize; 71310ed9ddcSRui Ueyama } else { 71410ed9ddcSRui Ueyama // It's neither PE32 nor PE32+. 71510ed9ddcSRui Ueyama EC = object_error::parse_failed; 716ed64342bSRui Ueyama return; 717ed64342bSRui Ueyama } 71848af1c2aSRafael Espindola if ((EC = getObject(DataDirectory, Data, DataDirAddr, DataDirSize))) 71910ed9ddcSRui Ueyama return; 72082ebd8e3SRui Ueyama CurPtr += COFFHeader->SizeOfOptionalHeader; 72182ebd8e3SRui Ueyama } 7221d6167fdSMichael J. Spencer 72348af1c2aSRafael Espindola if ((EC = getObject(SectionTable, Data, base() + CurPtr, 724236b0ca7SDavid Majnemer (uint64_t)getNumberOfSections() * sizeof(coff_section)))) 7251d6167fdSMichael J. Spencer return; 7261d6167fdSMichael J. Spencer 727c2bed429SRui Ueyama // Initialize the pointer to the symbol table. 728236b0ca7SDavid Majnemer if (getPointerToSymbolTable() != 0) { 7298ff24d25SRui Ueyama if ((EC = initSymbolTablePtr())) 7301d6167fdSMichael J. Spencer return; 731236b0ca7SDavid Majnemer } else { 732236b0ca7SDavid Majnemer // We had better not have any symbols if we don't have a symbol table. 733236b0ca7SDavid Majnemer if (getNumberOfSymbols() != 0) { 734236b0ca7SDavid Majnemer EC = object_error::parse_failed; 735236b0ca7SDavid Majnemer return; 736236b0ca7SDavid Majnemer } 737236b0ca7SDavid Majnemer } 7388e90adafSMichael J. Spencer 739c2bed429SRui Ueyama // Initialize the pointer to the beginning of the import table. 7408ff24d25SRui Ueyama if ((EC = initImportTablePtr())) 741ed64342bSRui Ueyama return; 74215d99359SRui Ueyama if ((EC = initDelayImportTablePtr())) 74315d99359SRui Ueyama return; 7441d6167fdSMichael J. Spencer 745ad882ba8SRui Ueyama // Initialize the pointer to the export table. 7468ff24d25SRui Ueyama if ((EC = initExportTablePtr())) 747ad882ba8SRui Ueyama return; 748ad882ba8SRui Ueyama 74974e85130SRui Ueyama // Initialize the pointer to the base relocation table. 75074e85130SRui Ueyama if ((EC = initBaseRelocPtr())) 75174e85130SRui Ueyama return; 75274e85130SRui Ueyama 7532da433eaSReid Kleckner // Initialize the pointer to the export table. 7542da433eaSReid Kleckner if ((EC = initDebugDirectoryPtr())) 7552da433eaSReid Kleckner return; 7562da433eaSReid Kleckner 7577d099195SRui Ueyama EC = std::error_code(); 7588e90adafSMichael J. Spencer } 7598e90adafSMichael J. Spencer 760f12b8282SRafael Espindola basic_symbol_iterator COFFObjectFile::symbol_begin_impl() const { 7618ff24d25SRui Ueyama DataRefImpl Ret; 76244f51e51SDavid Majnemer Ret.p = getSymbolTable(); 763f12b8282SRafael Espindola return basic_symbol_iterator(SymbolRef(Ret, this)); 7648e90adafSMichael J. Spencer } 7658e90adafSMichael J. Spencer 766f12b8282SRafael Espindola basic_symbol_iterator COFFObjectFile::symbol_end_impl() const { 7678e90adafSMichael J. Spencer // The symbol table ends where the string table begins. 7688ff24d25SRui Ueyama DataRefImpl Ret; 7698ff24d25SRui Ueyama Ret.p = reinterpret_cast<uintptr_t>(StringTable); 770f12b8282SRafael Espindola return basic_symbol_iterator(SymbolRef(Ret, this)); 7718e90adafSMichael J. Spencer } 7728e90adafSMichael J. Spencer 773bc654b18SRui Ueyama import_directory_iterator COFFObjectFile::import_directory_begin() const { 774a045b73aSRui Ueyama return import_directory_iterator( 775a045b73aSRui Ueyama ImportDirectoryEntryRef(ImportDirectory, 0, this)); 776c2bed429SRui Ueyama } 777c2bed429SRui Ueyama 778bc654b18SRui Ueyama import_directory_iterator COFFObjectFile::import_directory_end() const { 779a045b73aSRui Ueyama return import_directory_iterator( 780a045b73aSRui Ueyama ImportDirectoryEntryRef(ImportDirectory, NumberOfImportDirectory, this)); 781c2bed429SRui Ueyama } 782c429b80dSDavid Meyer 78315d99359SRui Ueyama delay_import_directory_iterator 78415d99359SRui Ueyama COFFObjectFile::delay_import_directory_begin() const { 78515d99359SRui Ueyama return delay_import_directory_iterator( 78615d99359SRui Ueyama DelayImportDirectoryEntryRef(DelayImportDirectory, 0, this)); 78715d99359SRui Ueyama } 78815d99359SRui Ueyama 78915d99359SRui Ueyama delay_import_directory_iterator 79015d99359SRui Ueyama COFFObjectFile::delay_import_directory_end() const { 79115d99359SRui Ueyama return delay_import_directory_iterator( 79215d99359SRui Ueyama DelayImportDirectoryEntryRef( 79315d99359SRui Ueyama DelayImportDirectory, NumberOfDelayImportDirectory, this)); 79415d99359SRui Ueyama } 79515d99359SRui Ueyama 796ad882ba8SRui Ueyama export_directory_iterator COFFObjectFile::export_directory_begin() const { 797ad882ba8SRui Ueyama return export_directory_iterator( 798ad882ba8SRui Ueyama ExportDirectoryEntryRef(ExportDirectory, 0, this)); 799ad882ba8SRui Ueyama } 800ad882ba8SRui Ueyama 801ad882ba8SRui Ueyama export_directory_iterator COFFObjectFile::export_directory_end() const { 8022617dcceSCraig Topper if (!ExportDirectory) 8032617dcceSCraig Topper return export_directory_iterator(ExportDirectoryEntryRef(nullptr, 0, this)); 8048ff24d25SRui Ueyama ExportDirectoryEntryRef Ref(ExportDirectory, 805ad882ba8SRui Ueyama ExportDirectory->AddressTableEntries, this); 8068ff24d25SRui Ueyama return export_directory_iterator(Ref); 807ad882ba8SRui Ueyama } 808ad882ba8SRui Ueyama 809b5155a57SRafael Espindola section_iterator COFFObjectFile::section_begin() const { 8108ff24d25SRui Ueyama DataRefImpl Ret; 8118ff24d25SRui Ueyama Ret.p = reinterpret_cast<uintptr_t>(SectionTable); 8128ff24d25SRui Ueyama return section_iterator(SectionRef(Ret, this)); 8138e90adafSMichael J. Spencer } 8148e90adafSMichael J. Spencer 815b5155a57SRafael Espindola section_iterator COFFObjectFile::section_end() const { 8168ff24d25SRui Ueyama DataRefImpl Ret; 81744f51e51SDavid Majnemer int NumSections = 81844f51e51SDavid Majnemer COFFHeader && COFFHeader->isImportLibrary() ? 0 : getNumberOfSections(); 8198ff24d25SRui Ueyama Ret.p = reinterpret_cast<uintptr_t>(SectionTable + NumSections); 8208ff24d25SRui Ueyama return section_iterator(SectionRef(Ret, this)); 8218e90adafSMichael J. Spencer } 8228e90adafSMichael J. Spencer 82374e85130SRui Ueyama base_reloc_iterator COFFObjectFile::base_reloc_begin() const { 82474e85130SRui Ueyama return base_reloc_iterator(BaseRelocRef(BaseRelocHeader, this)); 82574e85130SRui Ueyama } 82674e85130SRui Ueyama 82774e85130SRui Ueyama base_reloc_iterator COFFObjectFile::base_reloc_end() const { 82874e85130SRui Ueyama return base_reloc_iterator(BaseRelocRef(BaseRelocEnd, this)); 82974e85130SRui Ueyama } 83074e85130SRui Ueyama 8318e90adafSMichael J. Spencer uint8_t COFFObjectFile::getBytesInAddress() const { 8320324b672SMichael J. Spencer return getArch() == Triple::x86_64 ? 8 : 4; 8338e90adafSMichael J. Spencer } 8348e90adafSMichael J. Spencer 8358e90adafSMichael J. Spencer StringRef COFFObjectFile::getFileFormatName() const { 83644f51e51SDavid Majnemer switch(getMachine()) { 8378e90adafSMichael J. Spencer case COFF::IMAGE_FILE_MACHINE_I386: 8388e90adafSMichael J. Spencer return "COFF-i386"; 8398e90adafSMichael J. Spencer case COFF::IMAGE_FILE_MACHINE_AMD64: 8408e90adafSMichael J. Spencer return "COFF-x86-64"; 8419b7c0af2SSaleem Abdulrasool case COFF::IMAGE_FILE_MACHINE_ARMNT: 8429b7c0af2SSaleem Abdulrasool return "COFF-ARM"; 8431eff5c9cSMartell Malone case COFF::IMAGE_FILE_MACHINE_ARM64: 8441eff5c9cSMartell Malone return "COFF-ARM64"; 8458e90adafSMichael J. Spencer default: 8468e90adafSMichael J. Spencer return "COFF-<unknown arch>"; 8478e90adafSMichael J. Spencer } 8488e90adafSMichael J. Spencer } 8498e90adafSMichael J. Spencer 8508e90adafSMichael J. Spencer unsigned COFFObjectFile::getArch() const { 85144f51e51SDavid Majnemer switch (getMachine()) { 8528e90adafSMichael J. Spencer case COFF::IMAGE_FILE_MACHINE_I386: 8538e90adafSMichael J. Spencer return Triple::x86; 8548e90adafSMichael J. Spencer case COFF::IMAGE_FILE_MACHINE_AMD64: 8558e90adafSMichael J. Spencer return Triple::x86_64; 8569b7c0af2SSaleem Abdulrasool case COFF::IMAGE_FILE_MACHINE_ARMNT: 8579b7c0af2SSaleem Abdulrasool return Triple::thumb; 8581eff5c9cSMartell Malone case COFF::IMAGE_FILE_MACHINE_ARM64: 8591eff5c9cSMartell Malone return Triple::aarch64; 8608e90adafSMichael J. Spencer default: 8618e90adafSMichael J. Spencer return Triple::UnknownArch; 8628e90adafSMichael J. Spencer } 8638e90adafSMichael J. Spencer } 8648e90adafSMichael J. Spencer 865979fb40bSRui Ueyama iterator_range<import_directory_iterator> 866979fb40bSRui Ueyama COFFObjectFile::import_directories() const { 867979fb40bSRui Ueyama return make_range(import_directory_begin(), import_directory_end()); 868979fb40bSRui Ueyama } 869979fb40bSRui Ueyama 870979fb40bSRui Ueyama iterator_range<delay_import_directory_iterator> 871979fb40bSRui Ueyama COFFObjectFile::delay_import_directories() const { 872979fb40bSRui Ueyama return make_range(delay_import_directory_begin(), 873979fb40bSRui Ueyama delay_import_directory_end()); 874979fb40bSRui Ueyama } 875979fb40bSRui Ueyama 876979fb40bSRui Ueyama iterator_range<export_directory_iterator> 877979fb40bSRui Ueyama COFFObjectFile::export_directories() const { 878979fb40bSRui Ueyama return make_range(export_directory_begin(), export_directory_end()); 879979fb40bSRui Ueyama } 880979fb40bSRui Ueyama 88174e85130SRui Ueyama iterator_range<base_reloc_iterator> COFFObjectFile::base_relocs() const { 88274e85130SRui Ueyama return make_range(base_reloc_begin(), base_reloc_end()); 88374e85130SRui Ueyama } 88474e85130SRui Ueyama 885db4ed0bdSRafael Espindola std::error_code COFFObjectFile::getPE32Header(const pe32_header *&Res) const { 88682ebd8e3SRui Ueyama Res = PE32Header; 8877d099195SRui Ueyama return std::error_code(); 88889a7a5eaSMichael J. Spencer } 88989a7a5eaSMichael J. Spencer 890db4ed0bdSRafael Espindola std::error_code 89110ed9ddcSRui Ueyama COFFObjectFile::getPE32PlusHeader(const pe32plus_header *&Res) const { 89210ed9ddcSRui Ueyama Res = PE32PlusHeader; 8937d099195SRui Ueyama return std::error_code(); 89410ed9ddcSRui Ueyama } 89510ed9ddcSRui Ueyama 896db4ed0bdSRafael Espindola std::error_code 897db4ed0bdSRafael Espindola COFFObjectFile::getDataDirectory(uint32_t Index, 898ed64342bSRui Ueyama const data_directory *&Res) const { 899ed64342bSRui Ueyama // Error if if there's no data directory or the index is out of range. 900f69b0585SDavid Majnemer if (!DataDirectory) { 901f69b0585SDavid Majnemer Res = nullptr; 90210ed9ddcSRui Ueyama return object_error::parse_failed; 903f69b0585SDavid Majnemer } 90410ed9ddcSRui Ueyama assert(PE32Header || PE32PlusHeader); 90510ed9ddcSRui Ueyama uint32_t NumEnt = PE32Header ? PE32Header->NumberOfRvaAndSize 90610ed9ddcSRui Ueyama : PE32PlusHeader->NumberOfRvaAndSize; 907f69b0585SDavid Majnemer if (Index >= NumEnt) { 908f69b0585SDavid Majnemer Res = nullptr; 909ed64342bSRui Ueyama return object_error::parse_failed; 910f69b0585SDavid Majnemer } 9118ff24d25SRui Ueyama Res = &DataDirectory[Index]; 9127d099195SRui Ueyama return std::error_code(); 913ed64342bSRui Ueyama } 914ed64342bSRui Ueyama 915db4ed0bdSRafael Espindola std::error_code COFFObjectFile::getSection(int32_t Index, 9161d6167fdSMichael J. Spencer const coff_section *&Result) const { 9172617dcceSCraig Topper Result = nullptr; 918236b0ca7SDavid Majnemer if (COFF::isReservedSectionNumber(Index)) 9197d099195SRui Ueyama return std::error_code(); 920236b0ca7SDavid Majnemer if (static_cast<uint32_t>(Index) <= getNumberOfSections()) { 9211d6167fdSMichael J. Spencer // We already verified the section table data, so no need to check again. 9228ff24d25SRui Ueyama Result = SectionTable + (Index - 1); 9237d099195SRui Ueyama return std::error_code(); 9248e90adafSMichael J. Spencer } 925236b0ca7SDavid Majnemer return object_error::parse_failed; 926236b0ca7SDavid Majnemer } 9278e90adafSMichael J. Spencer 928db4ed0bdSRafael Espindola std::error_code COFFObjectFile::getString(uint32_t Offset, 9291d6167fdSMichael J. Spencer StringRef &Result) const { 9301d6167fdSMichael J. Spencer if (StringTableSize <= 4) 9311d6167fdSMichael J. Spencer // Tried to get a string from an empty string table. 9321d6167fdSMichael J. Spencer return object_error::parse_failed; 9338ff24d25SRui Ueyama if (Offset >= StringTableSize) 9341d6167fdSMichael J. Spencer return object_error::unexpected_eof; 9358ff24d25SRui Ueyama Result = StringRef(StringTable + Offset); 9367d099195SRui Ueyama return std::error_code(); 9378e90adafSMichael J. Spencer } 938022ecdf2SBenjamin Kramer 93944f51e51SDavid Majnemer std::error_code COFFObjectFile::getSymbolName(COFFSymbolRef Symbol, 94089a7a5eaSMichael J. Spencer StringRef &Res) const { 941e40d30f3SRui Ueyama return getSymbolName(Symbol.getGeneric(), Res); 942e40d30f3SRui Ueyama } 943e40d30f3SRui Ueyama 944e40d30f3SRui Ueyama std::error_code COFFObjectFile::getSymbolName(const coff_symbol_generic *Symbol, 945e40d30f3SRui Ueyama StringRef &Res) const { 94689a7a5eaSMichael J. Spencer // Check for string table entry. First 4 bytes are 0. 947e40d30f3SRui Ueyama if (Symbol->Name.Offset.Zeroes == 0) { 948e40d30f3SRui Ueyama if (std::error_code EC = getString(Symbol->Name.Offset.Offset, Res)) 9498ff24d25SRui Ueyama return EC; 9507d099195SRui Ueyama return std::error_code(); 95189a7a5eaSMichael J. Spencer } 95289a7a5eaSMichael J. Spencer 953e40d30f3SRui Ueyama if (Symbol->Name.ShortName[COFF::NameSize - 1] == 0) 95489a7a5eaSMichael J. Spencer // Null terminated, let ::strlen figure out the length. 955e40d30f3SRui Ueyama Res = StringRef(Symbol->Name.ShortName); 95689a7a5eaSMichael J. Spencer else 95789a7a5eaSMichael J. Spencer // Not null terminated, use all 8 bytes. 958e40d30f3SRui Ueyama Res = StringRef(Symbol->Name.ShortName, COFF::NameSize); 9597d099195SRui Ueyama return std::error_code(); 96089a7a5eaSMichael J. Spencer } 96189a7a5eaSMichael J. Spencer 96244f51e51SDavid Majnemer ArrayRef<uint8_t> 96344f51e51SDavid Majnemer COFFObjectFile::getSymbolAuxData(COFFSymbolRef Symbol) const { 9642617dcceSCraig Topper const uint8_t *Aux = nullptr; 96571757ef3SMarshall Clow 96644f51e51SDavid Majnemer size_t SymbolSize = getSymbolTableEntrySize(); 96744f51e51SDavid Majnemer if (Symbol.getNumberOfAuxSymbols() > 0) { 96871757ef3SMarshall Clow // AUX data comes immediately after the symbol in COFF 96944f51e51SDavid Majnemer Aux = reinterpret_cast<const uint8_t *>(Symbol.getRawPtr()) + SymbolSize; 97071757ef3SMarshall Clow # ifndef NDEBUG 9718ff24d25SRui Ueyama // Verify that the Aux symbol points to a valid entry in the symbol table. 9728ff24d25SRui Ueyama uintptr_t Offset = uintptr_t(Aux) - uintptr_t(base()); 97344f51e51SDavid Majnemer if (Offset < getPointerToSymbolTable() || 97444f51e51SDavid Majnemer Offset >= 97544f51e51SDavid Majnemer getPointerToSymbolTable() + (getNumberOfSymbols() * SymbolSize)) 97671757ef3SMarshall Clow report_fatal_error("Aux Symbol data was outside of symbol table."); 97771757ef3SMarshall Clow 97844f51e51SDavid Majnemer assert((Offset - getPointerToSymbolTable()) % SymbolSize == 0 && 97944f51e51SDavid Majnemer "Aux Symbol data did not point to the beginning of a symbol"); 98071757ef3SMarshall Clow # endif 981bfb85e67SMarshall Clow } 98244f51e51SDavid Majnemer return makeArrayRef(Aux, Symbol.getNumberOfAuxSymbols() * SymbolSize); 98371757ef3SMarshall Clow } 98471757ef3SMarshall Clow 985db4ed0bdSRafael Espindola std::error_code COFFObjectFile::getSectionName(const coff_section *Sec, 98653c2d547SMichael J. Spencer StringRef &Res) const { 98753c2d547SMichael J. Spencer StringRef Name; 98844f51e51SDavid Majnemer if (Sec->Name[COFF::NameSize - 1] == 0) 98953c2d547SMichael J. Spencer // Null terminated, let ::strlen figure out the length. 99053c2d547SMichael J. Spencer Name = Sec->Name; 99153c2d547SMichael J. Spencer else 99253c2d547SMichael J. Spencer // Not null terminated, use all 8 bytes. 99344f51e51SDavid Majnemer Name = StringRef(Sec->Name, COFF::NameSize); 99453c2d547SMichael J. Spencer 99553c2d547SMichael J. Spencer // Check for string table entry. First byte is '/'. 9962314b3deSDavid Majnemer if (Name.startswith("/")) { 99753c2d547SMichael J. Spencer uint32_t Offset; 9982314b3deSDavid Majnemer if (Name.startswith("//")) { 9999d2c15efSNico Rieck if (decodeBase64StringEntry(Name.substr(2), Offset)) 10009d2c15efSNico Rieck return object_error::parse_failed; 10019d2c15efSNico Rieck } else { 100253c2d547SMichael J. Spencer if (Name.substr(1).getAsInteger(10, Offset)) 100353c2d547SMichael J. Spencer return object_error::parse_failed; 10049d2c15efSNico Rieck } 1005db4ed0bdSRafael Espindola if (std::error_code EC = getString(Offset, Name)) 10068ff24d25SRui Ueyama return EC; 100753c2d547SMichael J. Spencer } 100853c2d547SMichael J. Spencer 100953c2d547SMichael J. Spencer Res = Name; 10107d099195SRui Ueyama return std::error_code(); 101153c2d547SMichael J. Spencer } 101253c2d547SMichael J. Spencer 1013a9ee5c06SDavid Majnemer uint64_t COFFObjectFile::getSectionSize(const coff_section *Sec) const { 1014a9ee5c06SDavid Majnemer // SizeOfRawData and VirtualSize change what they represent depending on 1015a9ee5c06SDavid Majnemer // whether or not we have an executable image. 1016a9ee5c06SDavid Majnemer // 1017a9ee5c06SDavid Majnemer // For object files, SizeOfRawData contains the size of section's data; 1018d5297ee7SRui Ueyama // VirtualSize should be zero but isn't due to buggy COFF writers. 1019a9ee5c06SDavid Majnemer // 1020a9ee5c06SDavid Majnemer // For executables, SizeOfRawData *must* be a multiple of FileAlignment; the 1021a9ee5c06SDavid Majnemer // actual section size is in VirtualSize. It is possible for VirtualSize to 1022a9ee5c06SDavid Majnemer // be greater than SizeOfRawData; the contents past that point should be 1023a9ee5c06SDavid Majnemer // considered to be zero. 1024d5297ee7SRui Ueyama if (getDOSHeader()) 1025d5297ee7SRui Ueyama return std::min(Sec->VirtualSize, Sec->SizeOfRawData); 1026d5297ee7SRui Ueyama return Sec->SizeOfRawData; 1027a9ee5c06SDavid Majnemer } 1028a9ee5c06SDavid Majnemer 1029db4ed0bdSRafael Espindola std::error_code 1030db4ed0bdSRafael Espindola COFFObjectFile::getSectionContents(const coff_section *Sec, 10319da9e693SMichael J. Spencer ArrayRef<uint8_t> &Res) const { 1032e2129662SDavid Majnemer // In COFF, a virtual section won't have any in-file 1033e2129662SDavid Majnemer // content, so the file pointer to the content will be zero. 1034e2129662SDavid Majnemer if (Sec->PointerToRawData == 0) 1035e2129662SDavid Majnemer return object_error::parse_failed; 10369da9e693SMichael J. Spencer // The only thing that we need to verify is that the contents is contained 10379da9e693SMichael J. Spencer // within the file bounds. We don't need to make sure it doesn't cover other 10389da9e693SMichael J. Spencer // data, as there's nothing that says that is not allowed. 10399da9e693SMichael J. Spencer uintptr_t ConStart = uintptr_t(base()) + Sec->PointerToRawData; 1040a9ee5c06SDavid Majnemer uint32_t SectionSize = getSectionSize(Sec); 1041e830c60dSDavid Majnemer if (checkOffset(Data, ConStart, SectionSize)) 10429da9e693SMichael J. Spencer return object_error::parse_failed; 1043a9ee5c06SDavid Majnemer Res = makeArrayRef(reinterpret_cast<const uint8_t *>(ConStart), SectionSize); 10447d099195SRui Ueyama return std::error_code(); 10459da9e693SMichael J. Spencer } 10469da9e693SMichael J. Spencer 1047022ecdf2SBenjamin Kramer const coff_relocation *COFFObjectFile::toRel(DataRefImpl Rel) const { 1048e5fd0047SMichael J. Spencer return reinterpret_cast<const coff_relocation*>(Rel.p); 1049022ecdf2SBenjamin Kramer } 10508ff24d25SRui Ueyama 10515e812afaSRafael Espindola void COFFObjectFile::moveRelocationNext(DataRefImpl &Rel) const { 1052e5fd0047SMichael J. Spencer Rel.p = reinterpret_cast<uintptr_t>( 1053e5fd0047SMichael J. Spencer reinterpret_cast<const coff_relocation*>(Rel.p) + 1); 1054022ecdf2SBenjamin Kramer } 10558ff24d25SRui Ueyama 105696d071cdSRafael Espindola uint64_t COFFObjectFile::getRelocationOffset(DataRefImpl Rel) const { 105758323a97SDavid Majnemer const coff_relocation *R = toRel(Rel); 105896d071cdSRafael Espindola return R->VirtualAddress; 1059cbe72fc9SDanil Malyshev } 10608ff24d25SRui Ueyama 1061806f0064SRafael Espindola symbol_iterator COFFObjectFile::getRelocationSymbol(DataRefImpl Rel) const { 1062022ecdf2SBenjamin Kramer const coff_relocation *R = toRel(Rel); 10638ff24d25SRui Ueyama DataRefImpl Ref; 1064236b0ca7SDavid Majnemer if (R->SymbolTableIndex >= getNumberOfSymbols()) 1065236b0ca7SDavid Majnemer return symbol_end(); 106644f51e51SDavid Majnemer if (SymbolTable16) 106744f51e51SDavid Majnemer Ref.p = reinterpret_cast<uintptr_t>(SymbolTable16 + R->SymbolTableIndex); 106844f51e51SDavid Majnemer else if (SymbolTable32) 106944f51e51SDavid Majnemer Ref.p = reinterpret_cast<uintptr_t>(SymbolTable32 + R->SymbolTableIndex); 107044f51e51SDavid Majnemer else 1071c7353b58SDavid Majnemer llvm_unreachable("no symbol table pointer!"); 10728ff24d25SRui Ueyama return symbol_iterator(SymbolRef(Ref, this)); 1073022ecdf2SBenjamin Kramer } 10748ff24d25SRui Ueyama 107599c041b7SRafael Espindola uint64_t COFFObjectFile::getRelocationType(DataRefImpl Rel) const { 1076022ecdf2SBenjamin Kramer const coff_relocation* R = toRel(Rel); 107799c041b7SRafael Espindola return R->Type; 1078022ecdf2SBenjamin Kramer } 1079e5fd0047SMichael J. Spencer 108027dc8394SAlexey Samsonov const coff_section * 108127dc8394SAlexey Samsonov COFFObjectFile::getCOFFSection(const SectionRef &Section) const { 108227dc8394SAlexey Samsonov return toSec(Section.getRawDataRefImpl()); 108371757ef3SMarshall Clow } 108471757ef3SMarshall Clow 108544f51e51SDavid Majnemer COFFSymbolRef COFFObjectFile::getCOFFSymbol(const DataRefImpl &Ref) const { 108644f51e51SDavid Majnemer if (SymbolTable16) 108744f51e51SDavid Majnemer return toSymb<coff_symbol16>(Ref); 108844f51e51SDavid Majnemer if (SymbolTable32) 108944f51e51SDavid Majnemer return toSymb<coff_symbol32>(Ref); 109044f51e51SDavid Majnemer llvm_unreachable("no symbol table pointer!"); 109144f51e51SDavid Majnemer } 109244f51e51SDavid Majnemer 109344f51e51SDavid Majnemer COFFSymbolRef COFFObjectFile::getCOFFSymbol(const SymbolRef &Symbol) const { 109444f51e51SDavid Majnemer return getCOFFSymbol(Symbol.getRawDataRefImpl()); 109571757ef3SMarshall Clow } 109671757ef3SMarshall Clow 1097f12b8282SRafael Espindola const coff_relocation * 109827dc8394SAlexey Samsonov COFFObjectFile::getCOFFRelocation(const RelocationRef &Reloc) const { 109927dc8394SAlexey Samsonov return toRel(Reloc.getRawDataRefImpl()); 1100d3e2a76cSMarshall Clow } 1101d3e2a76cSMarshall Clow 11026a75acb1SRui Ueyama iterator_range<const coff_relocation *> 11036a75acb1SRui Ueyama COFFObjectFile::getRelocations(const coff_section *Sec) const { 11046a75acb1SRui Ueyama const coff_relocation *I = getFirstReloc(Sec, Data, base()); 11056a75acb1SRui Ueyama const coff_relocation *E = I; 11066a75acb1SRui Ueyama if (I) 11076a75acb1SRui Ueyama E += getNumberOfRelocations(Sec, Data, base()); 11086a75acb1SRui Ueyama return make_range(I, E); 11096a75acb1SRui Ueyama } 11106a75acb1SRui Ueyama 111127dc8394SAlexey Samsonov #define LLVM_COFF_SWITCH_RELOC_TYPE_NAME(reloc_type) \ 111227dc8394SAlexey Samsonov case COFF::reloc_type: \ 111327dc8394SAlexey Samsonov Res = #reloc_type; \ 111427dc8394SAlexey Samsonov break; 1115e5fd0047SMichael J. Spencer 111641bb4325SRafael Espindola void COFFObjectFile::getRelocationTypeName( 111741bb4325SRafael Espindola DataRefImpl Rel, SmallVectorImpl<char> &Result) const { 11188ff24d25SRui Ueyama const coff_relocation *Reloc = toRel(Rel); 11198ff24d25SRui Ueyama StringRef Res; 112044f51e51SDavid Majnemer switch (getMachine()) { 1121e5fd0047SMichael J. Spencer case COFF::IMAGE_FILE_MACHINE_AMD64: 11228ff24d25SRui Ueyama switch (Reloc->Type) { 1123e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_ABSOLUTE); 1124e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_ADDR64); 1125e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_ADDR32); 1126e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_ADDR32NB); 1127e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_REL32); 1128e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_REL32_1); 1129e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_REL32_2); 1130e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_REL32_3); 1131e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_REL32_4); 1132e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_REL32_5); 1133e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_SECTION); 1134e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_SECREL); 1135e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_SECREL7); 1136e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_TOKEN); 1137e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_SREL32); 1138e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_PAIR); 1139e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_AMD64_SSPAN32); 1140e5fd0047SMichael J. Spencer default: 11418ff24d25SRui Ueyama Res = "Unknown"; 1142e5fd0047SMichael J. Spencer } 1143e5fd0047SMichael J. Spencer break; 11445c503bf4SSaleem Abdulrasool case COFF::IMAGE_FILE_MACHINE_ARMNT: 11455c503bf4SSaleem Abdulrasool switch (Reloc->Type) { 11465c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_ABSOLUTE); 11475c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_ADDR32); 11485c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_ADDR32NB); 11495c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_BRANCH24); 11505c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_BRANCH11); 11515c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_TOKEN); 11525c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_BLX24); 11535c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_BLX11); 11545c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_SECTION); 11555c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_SECREL); 11565c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_MOV32A); 11575c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_MOV32T); 11585c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_BRANCH20T); 11595c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_BRANCH24T); 11605c503bf4SSaleem Abdulrasool LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_ARM_BLX23T); 11615c503bf4SSaleem Abdulrasool default: 11625c503bf4SSaleem Abdulrasool Res = "Unknown"; 11635c503bf4SSaleem Abdulrasool } 11645c503bf4SSaleem Abdulrasool break; 1165e5fd0047SMichael J. Spencer case COFF::IMAGE_FILE_MACHINE_I386: 11668ff24d25SRui Ueyama switch (Reloc->Type) { 1167e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_I386_ABSOLUTE); 1168e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_I386_DIR16); 1169e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_I386_REL16); 1170e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_I386_DIR32); 1171e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_I386_DIR32NB); 1172e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_I386_SEG12); 1173e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_I386_SECTION); 1174e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_I386_SECREL); 1175e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_I386_TOKEN); 1176e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_I386_SECREL7); 1177e5fd0047SMichael J. Spencer LLVM_COFF_SWITCH_RELOC_TYPE_NAME(IMAGE_REL_I386_REL32); 1178e5fd0047SMichael J. Spencer default: 11798ff24d25SRui Ueyama Res = "Unknown"; 1180e5fd0047SMichael J. Spencer } 1181e5fd0047SMichael J. Spencer break; 1182e5fd0047SMichael J. Spencer default: 11838ff24d25SRui Ueyama Res = "Unknown"; 1184e5fd0047SMichael J. Spencer } 11858ff24d25SRui Ueyama Result.append(Res.begin(), Res.end()); 1186e5fd0047SMichael J. Spencer } 1187e5fd0047SMichael J. Spencer 1188e5fd0047SMichael J. Spencer #undef LLVM_COFF_SWITCH_RELOC_TYPE_NAME 1189e5fd0047SMichael J. Spencer 1190c66d761bSRafael Espindola bool COFFObjectFile::isRelocatableObject() const { 1191c66d761bSRafael Espindola return !DataDirectory; 1192c66d761bSRafael Espindola } 1193c66d761bSRafael Espindola 1194c2bed429SRui Ueyama bool ImportDirectoryEntryRef:: 1195c2bed429SRui Ueyama operator==(const ImportDirectoryEntryRef &Other) const { 1196a045b73aSRui Ueyama return ImportTable == Other.ImportTable && Index == Other.Index; 1197c2bed429SRui Ueyama } 1198c2bed429SRui Ueyama 11995e812afaSRafael Espindola void ImportDirectoryEntryRef::moveNext() { 12005e812afaSRafael Espindola ++Index; 1201c2bed429SRui Ueyama } 1202c2bed429SRui Ueyama 1203db4ed0bdSRafael Espindola std::error_code ImportDirectoryEntryRef::getImportTableEntry( 1204db4ed0bdSRafael Espindola const import_directory_table_entry *&Result) const { 12051e152d5eSRui Ueyama Result = ImportTable + Index; 12067d099195SRui Ueyama return std::error_code(); 1207c2bed429SRui Ueyama } 1208c2bed429SRui Ueyama 1209861021f9SRui Ueyama static imported_symbol_iterator 121015d99359SRui Ueyama makeImportedSymbolIterator(const COFFObjectFile *Object, 1211861021f9SRui Ueyama uintptr_t Ptr, int Index) { 121215d99359SRui Ueyama if (Object->getBytesInAddress() == 4) { 1213861021f9SRui Ueyama auto *P = reinterpret_cast<const import_lookup_table_entry32 *>(Ptr); 121415d99359SRui Ueyama return imported_symbol_iterator(ImportedSymbolRef(P, Index, Object)); 1215861021f9SRui Ueyama } 1216861021f9SRui Ueyama auto *P = reinterpret_cast<const import_lookup_table_entry64 *>(Ptr); 121715d99359SRui Ueyama return imported_symbol_iterator(ImportedSymbolRef(P, Index, Object)); 1218861021f9SRui Ueyama } 1219861021f9SRui Ueyama 122015d99359SRui Ueyama static imported_symbol_iterator 122115d99359SRui Ueyama importedSymbolBegin(uint32_t RVA, const COFFObjectFile *Object) { 1222861021f9SRui Ueyama uintptr_t IntPtr = 0; 122315d99359SRui Ueyama Object->getRvaPtr(RVA, IntPtr); 122415d99359SRui Ueyama return makeImportedSymbolIterator(Object, IntPtr, 0); 1225861021f9SRui Ueyama } 1226861021f9SRui Ueyama 122715d99359SRui Ueyama static imported_symbol_iterator 122815d99359SRui Ueyama importedSymbolEnd(uint32_t RVA, const COFFObjectFile *Object) { 1229861021f9SRui Ueyama uintptr_t IntPtr = 0; 123015d99359SRui Ueyama Object->getRvaPtr(RVA, IntPtr); 1231861021f9SRui Ueyama // Forward the pointer to the last entry which is null. 1232861021f9SRui Ueyama int Index = 0; 123315d99359SRui Ueyama if (Object->getBytesInAddress() == 4) { 1234861021f9SRui Ueyama auto *Entry = reinterpret_cast<ulittle32_t *>(IntPtr); 1235861021f9SRui Ueyama while (*Entry++) 1236861021f9SRui Ueyama ++Index; 1237861021f9SRui Ueyama } else { 1238861021f9SRui Ueyama auto *Entry = reinterpret_cast<ulittle64_t *>(IntPtr); 1239861021f9SRui Ueyama while (*Entry++) 1240861021f9SRui Ueyama ++Index; 1241861021f9SRui Ueyama } 124215d99359SRui Ueyama return makeImportedSymbolIterator(Object, IntPtr, Index); 124315d99359SRui Ueyama } 124415d99359SRui Ueyama 124515d99359SRui Ueyama imported_symbol_iterator 124615d99359SRui Ueyama ImportDirectoryEntryRef::imported_symbol_begin() const { 124715d99359SRui Ueyama return importedSymbolBegin(ImportTable[Index].ImportLookupTableRVA, 124815d99359SRui Ueyama OwningObject); 124915d99359SRui Ueyama } 125015d99359SRui Ueyama 125115d99359SRui Ueyama imported_symbol_iterator 125215d99359SRui Ueyama ImportDirectoryEntryRef::imported_symbol_end() const { 125315d99359SRui Ueyama return importedSymbolEnd(ImportTable[Index].ImportLookupTableRVA, 125415d99359SRui Ueyama OwningObject); 1255861021f9SRui Ueyama } 1256861021f9SRui Ueyama 1257979fb40bSRui Ueyama iterator_range<imported_symbol_iterator> 1258979fb40bSRui Ueyama ImportDirectoryEntryRef::imported_symbols() const { 1259979fb40bSRui Ueyama return make_range(imported_symbol_begin(), imported_symbol_end()); 1260979fb40bSRui Ueyama } 1261979fb40bSRui Ueyama 1262db4ed0bdSRafael Espindola std::error_code ImportDirectoryEntryRef::getName(StringRef &Result) const { 1263c2bed429SRui Ueyama uintptr_t IntPtr = 0; 1264db4ed0bdSRafael Espindola if (std::error_code EC = 12651e152d5eSRui Ueyama OwningObject->getRvaPtr(ImportTable[Index].NameRVA, IntPtr)) 1266a045b73aSRui Ueyama return EC; 1267a045b73aSRui Ueyama Result = StringRef(reinterpret_cast<const char *>(IntPtr)); 12687d099195SRui Ueyama return std::error_code(); 1269c2bed429SRui Ueyama } 1270c2bed429SRui Ueyama 12711e152d5eSRui Ueyama std::error_code 12721e152d5eSRui Ueyama ImportDirectoryEntryRef::getImportLookupTableRVA(uint32_t &Result) const { 12731e152d5eSRui Ueyama Result = ImportTable[Index].ImportLookupTableRVA; 12747d099195SRui Ueyama return std::error_code(); 12751e152d5eSRui Ueyama } 12761e152d5eSRui Ueyama 12771e152d5eSRui Ueyama std::error_code 12781e152d5eSRui Ueyama ImportDirectoryEntryRef::getImportAddressTableRVA(uint32_t &Result) const { 12791e152d5eSRui Ueyama Result = ImportTable[Index].ImportAddressTableRVA; 12807d099195SRui Ueyama return std::error_code(); 12811e152d5eSRui Ueyama } 12821e152d5eSRui Ueyama 1283db4ed0bdSRafael Espindola std::error_code ImportDirectoryEntryRef::getImportLookupEntry( 1284c2bed429SRui Ueyama const import_lookup_table_entry32 *&Result) const { 1285c2bed429SRui Ueyama uintptr_t IntPtr = 0; 12861e152d5eSRui Ueyama uint32_t RVA = ImportTable[Index].ImportLookupTableRVA; 12871e152d5eSRui Ueyama if (std::error_code EC = OwningObject->getRvaPtr(RVA, IntPtr)) 1288a045b73aSRui Ueyama return EC; 1289c2bed429SRui Ueyama Result = reinterpret_cast<const import_lookup_table_entry32 *>(IntPtr); 12907d099195SRui Ueyama return std::error_code(); 1291c2bed429SRui Ueyama } 1292c2bed429SRui Ueyama 129315d99359SRui Ueyama bool DelayImportDirectoryEntryRef:: 129415d99359SRui Ueyama operator==(const DelayImportDirectoryEntryRef &Other) const { 129515d99359SRui Ueyama return Table == Other.Table && Index == Other.Index; 129615d99359SRui Ueyama } 129715d99359SRui Ueyama 129815d99359SRui Ueyama void DelayImportDirectoryEntryRef::moveNext() { 129915d99359SRui Ueyama ++Index; 130015d99359SRui Ueyama } 130115d99359SRui Ueyama 130215d99359SRui Ueyama imported_symbol_iterator 130315d99359SRui Ueyama DelayImportDirectoryEntryRef::imported_symbol_begin() const { 130415d99359SRui Ueyama return importedSymbolBegin(Table[Index].DelayImportNameTable, 130515d99359SRui Ueyama OwningObject); 130615d99359SRui Ueyama } 130715d99359SRui Ueyama 130815d99359SRui Ueyama imported_symbol_iterator 130915d99359SRui Ueyama DelayImportDirectoryEntryRef::imported_symbol_end() const { 131015d99359SRui Ueyama return importedSymbolEnd(Table[Index].DelayImportNameTable, 131115d99359SRui Ueyama OwningObject); 131215d99359SRui Ueyama } 131315d99359SRui Ueyama 1314979fb40bSRui Ueyama iterator_range<imported_symbol_iterator> 1315979fb40bSRui Ueyama DelayImportDirectoryEntryRef::imported_symbols() const { 1316979fb40bSRui Ueyama return make_range(imported_symbol_begin(), imported_symbol_end()); 1317979fb40bSRui Ueyama } 1318979fb40bSRui Ueyama 131915d99359SRui Ueyama std::error_code DelayImportDirectoryEntryRef::getName(StringRef &Result) const { 132015d99359SRui Ueyama uintptr_t IntPtr = 0; 132115d99359SRui Ueyama if (std::error_code EC = OwningObject->getRvaPtr(Table[Index].Name, IntPtr)) 132215d99359SRui Ueyama return EC; 132315d99359SRui Ueyama Result = StringRef(reinterpret_cast<const char *>(IntPtr)); 13247d099195SRui Ueyama return std::error_code(); 132515d99359SRui Ueyama } 132615d99359SRui Ueyama 13271af08658SRui Ueyama std::error_code DelayImportDirectoryEntryRef:: 13281af08658SRui Ueyama getDelayImportTable(const delay_import_directory_table_entry *&Result) const { 13291af08658SRui Ueyama Result = Table; 13307d099195SRui Ueyama return std::error_code(); 13311af08658SRui Ueyama } 13321af08658SRui Ueyama 1333ffa4cebeSRui Ueyama std::error_code DelayImportDirectoryEntryRef:: 1334ffa4cebeSRui Ueyama getImportAddress(int AddrIndex, uint64_t &Result) const { 1335ffa4cebeSRui Ueyama uint32_t RVA = Table[Index].DelayImportAddressTable + 1336ffa4cebeSRui Ueyama AddrIndex * (OwningObject->is64() ? 8 : 4); 1337ffa4cebeSRui Ueyama uintptr_t IntPtr = 0; 1338ffa4cebeSRui Ueyama if (std::error_code EC = OwningObject->getRvaPtr(RVA, IntPtr)) 1339ffa4cebeSRui Ueyama return EC; 1340ffa4cebeSRui Ueyama if (OwningObject->is64()) 13415dcf11d1SRui Ueyama Result = *reinterpret_cast<const ulittle64_t *>(IntPtr); 1342ffa4cebeSRui Ueyama else 13435dcf11d1SRui Ueyama Result = *reinterpret_cast<const ulittle32_t *>(IntPtr); 13447d099195SRui Ueyama return std::error_code(); 1345ffa4cebeSRui Ueyama } 1346ffa4cebeSRui Ueyama 1347ad882ba8SRui Ueyama bool ExportDirectoryEntryRef:: 1348ad882ba8SRui Ueyama operator==(const ExportDirectoryEntryRef &Other) const { 1349ad882ba8SRui Ueyama return ExportTable == Other.ExportTable && Index == Other.Index; 1350ad882ba8SRui Ueyama } 1351ad882ba8SRui Ueyama 13525e812afaSRafael Espindola void ExportDirectoryEntryRef::moveNext() { 13535e812afaSRafael Espindola ++Index; 1354ad882ba8SRui Ueyama } 1355ad882ba8SRui Ueyama 1356da49d0d4SRui Ueyama // Returns the name of the current export symbol. If the symbol is exported only 1357da49d0d4SRui Ueyama // by ordinal, the empty string is set as a result. 1358db4ed0bdSRafael Espindola std::error_code ExportDirectoryEntryRef::getDllName(StringRef &Result) const { 1359da49d0d4SRui Ueyama uintptr_t IntPtr = 0; 1360db4ed0bdSRafael Espindola if (std::error_code EC = 1361db4ed0bdSRafael Espindola OwningObject->getRvaPtr(ExportTable->NameRVA, IntPtr)) 1362da49d0d4SRui Ueyama return EC; 1363da49d0d4SRui Ueyama Result = StringRef(reinterpret_cast<const char *>(IntPtr)); 13647d099195SRui Ueyama return std::error_code(); 1365da49d0d4SRui Ueyama } 1366da49d0d4SRui Ueyama 1367e5df6095SRui Ueyama // Returns the starting ordinal number. 1368db4ed0bdSRafael Espindola std::error_code 1369db4ed0bdSRafael Espindola ExportDirectoryEntryRef::getOrdinalBase(uint32_t &Result) const { 1370e5df6095SRui Ueyama Result = ExportTable->OrdinalBase; 13717d099195SRui Ueyama return std::error_code(); 1372e5df6095SRui Ueyama } 1373e5df6095SRui Ueyama 1374ad882ba8SRui Ueyama // Returns the export ordinal of the current export symbol. 1375db4ed0bdSRafael Espindola std::error_code ExportDirectoryEntryRef::getOrdinal(uint32_t &Result) const { 1376ad882ba8SRui Ueyama Result = ExportTable->OrdinalBase + Index; 13777d099195SRui Ueyama return std::error_code(); 1378ad882ba8SRui Ueyama } 1379ad882ba8SRui Ueyama 1380ad882ba8SRui Ueyama // Returns the address of the current export symbol. 1381db4ed0bdSRafael Espindola std::error_code ExportDirectoryEntryRef::getExportRVA(uint32_t &Result) const { 1382ad882ba8SRui Ueyama uintptr_t IntPtr = 0; 1383db4ed0bdSRafael Espindola if (std::error_code EC = 1384db4ed0bdSRafael Espindola OwningObject->getRvaPtr(ExportTable->ExportAddressTableRVA, IntPtr)) 1385ad882ba8SRui Ueyama return EC; 138624fc2d64SRui Ueyama const export_address_table_entry *entry = 138724fc2d64SRui Ueyama reinterpret_cast<const export_address_table_entry *>(IntPtr); 1388ad882ba8SRui Ueyama Result = entry[Index].ExportRVA; 13897d099195SRui Ueyama return std::error_code(); 1390ad882ba8SRui Ueyama } 1391ad882ba8SRui Ueyama 1392ad882ba8SRui Ueyama // Returns the name of the current export symbol. If the symbol is exported only 1393ad882ba8SRui Ueyama // by ordinal, the empty string is set as a result. 1394db4ed0bdSRafael Espindola std::error_code 1395db4ed0bdSRafael Espindola ExportDirectoryEntryRef::getSymbolName(StringRef &Result) const { 1396ad882ba8SRui Ueyama uintptr_t IntPtr = 0; 1397db4ed0bdSRafael Espindola if (std::error_code EC = 1398db4ed0bdSRafael Espindola OwningObject->getRvaPtr(ExportTable->OrdinalTableRVA, IntPtr)) 1399ad882ba8SRui Ueyama return EC; 1400ad882ba8SRui Ueyama const ulittle16_t *Start = reinterpret_cast<const ulittle16_t *>(IntPtr); 1401ad882ba8SRui Ueyama 1402ad882ba8SRui Ueyama uint32_t NumEntries = ExportTable->NumberOfNamePointers; 1403ad882ba8SRui Ueyama int Offset = 0; 1404ad882ba8SRui Ueyama for (const ulittle16_t *I = Start, *E = Start + NumEntries; 1405ad882ba8SRui Ueyama I < E; ++I, ++Offset) { 1406ad882ba8SRui Ueyama if (*I != Index) 1407ad882ba8SRui Ueyama continue; 1408db4ed0bdSRafael Espindola if (std::error_code EC = 1409db4ed0bdSRafael Espindola OwningObject->getRvaPtr(ExportTable->NamePointerRVA, IntPtr)) 1410ad882ba8SRui Ueyama return EC; 1411ad882ba8SRui Ueyama const ulittle32_t *NamePtr = reinterpret_cast<const ulittle32_t *>(IntPtr); 1412db4ed0bdSRafael Espindola if (std::error_code EC = OwningObject->getRvaPtr(NamePtr[Offset], IntPtr)) 1413ad882ba8SRui Ueyama return EC; 1414ad882ba8SRui Ueyama Result = StringRef(reinterpret_cast<const char *>(IntPtr)); 14157d099195SRui Ueyama return std::error_code(); 1416ad882ba8SRui Ueyama } 1417ad882ba8SRui Ueyama Result = ""; 14187d099195SRui Ueyama return std::error_code(); 1419ad882ba8SRui Ueyama } 1420ad882ba8SRui Ueyama 14216161b38dSRui Ueyama std::error_code ExportDirectoryEntryRef::isForwarder(bool &Result) const { 14226161b38dSRui Ueyama const data_directory *DataEntry; 14236161b38dSRui Ueyama if (auto EC = OwningObject->getDataDirectory(COFF::EXPORT_TABLE, DataEntry)) 14246161b38dSRui Ueyama return EC; 14256161b38dSRui Ueyama uint32_t RVA; 14266161b38dSRui Ueyama if (auto EC = getExportRVA(RVA)) 14276161b38dSRui Ueyama return EC; 14286161b38dSRui Ueyama uint32_t Begin = DataEntry->RelativeVirtualAddress; 14296161b38dSRui Ueyama uint32_t End = DataEntry->RelativeVirtualAddress + DataEntry->Size; 14306161b38dSRui Ueyama Result = (Begin <= RVA && RVA < End); 14316161b38dSRui Ueyama return std::error_code(); 14326161b38dSRui Ueyama } 14336161b38dSRui Ueyama 14346161b38dSRui Ueyama std::error_code ExportDirectoryEntryRef::getForwardTo(StringRef &Result) const { 14356161b38dSRui Ueyama uint32_t RVA; 14366161b38dSRui Ueyama if (auto EC = getExportRVA(RVA)) 14376161b38dSRui Ueyama return EC; 14386161b38dSRui Ueyama uintptr_t IntPtr = 0; 14396161b38dSRui Ueyama if (auto EC = OwningObject->getRvaPtr(RVA, IntPtr)) 14406161b38dSRui Ueyama return EC; 14416161b38dSRui Ueyama Result = StringRef(reinterpret_cast<const char *>(IntPtr)); 14426161b38dSRui Ueyama return std::error_code(); 14436161b38dSRui Ueyama } 14446161b38dSRui Ueyama 1445861021f9SRui Ueyama bool ImportedSymbolRef:: 1446861021f9SRui Ueyama operator==(const ImportedSymbolRef &Other) const { 1447861021f9SRui Ueyama return Entry32 == Other.Entry32 && Entry64 == Other.Entry64 1448861021f9SRui Ueyama && Index == Other.Index; 1449861021f9SRui Ueyama } 1450861021f9SRui Ueyama 1451861021f9SRui Ueyama void ImportedSymbolRef::moveNext() { 1452861021f9SRui Ueyama ++Index; 1453861021f9SRui Ueyama } 1454861021f9SRui Ueyama 1455861021f9SRui Ueyama std::error_code 1456861021f9SRui Ueyama ImportedSymbolRef::getSymbolName(StringRef &Result) const { 1457861021f9SRui Ueyama uint32_t RVA; 1458861021f9SRui Ueyama if (Entry32) { 1459861021f9SRui Ueyama // If a symbol is imported only by ordinal, it has no name. 1460861021f9SRui Ueyama if (Entry32[Index].isOrdinal()) 14617d099195SRui Ueyama return std::error_code(); 1462861021f9SRui Ueyama RVA = Entry32[Index].getHintNameRVA(); 1463861021f9SRui Ueyama } else { 1464861021f9SRui Ueyama if (Entry64[Index].isOrdinal()) 14657d099195SRui Ueyama return std::error_code(); 1466861021f9SRui Ueyama RVA = Entry64[Index].getHintNameRVA(); 1467861021f9SRui Ueyama } 1468861021f9SRui Ueyama uintptr_t IntPtr = 0; 1469861021f9SRui Ueyama if (std::error_code EC = OwningObject->getRvaPtr(RVA, IntPtr)) 1470861021f9SRui Ueyama return EC; 1471861021f9SRui Ueyama // +2 because the first two bytes is hint. 1472861021f9SRui Ueyama Result = StringRef(reinterpret_cast<const char *>(IntPtr + 2)); 14737d099195SRui Ueyama return std::error_code(); 1474861021f9SRui Ueyama } 1475861021f9SRui Ueyama 1476861021f9SRui Ueyama std::error_code ImportedSymbolRef::getOrdinal(uint16_t &Result) const { 1477861021f9SRui Ueyama uint32_t RVA; 1478861021f9SRui Ueyama if (Entry32) { 1479861021f9SRui Ueyama if (Entry32[Index].isOrdinal()) { 1480861021f9SRui Ueyama Result = Entry32[Index].getOrdinal(); 14817d099195SRui Ueyama return std::error_code(); 1482861021f9SRui Ueyama } 1483861021f9SRui Ueyama RVA = Entry32[Index].getHintNameRVA(); 1484861021f9SRui Ueyama } else { 1485861021f9SRui Ueyama if (Entry64[Index].isOrdinal()) { 1486861021f9SRui Ueyama Result = Entry64[Index].getOrdinal(); 14877d099195SRui Ueyama return std::error_code(); 1488861021f9SRui Ueyama } 1489861021f9SRui Ueyama RVA = Entry64[Index].getHintNameRVA(); 1490861021f9SRui Ueyama } 1491861021f9SRui Ueyama uintptr_t IntPtr = 0; 1492861021f9SRui Ueyama if (std::error_code EC = OwningObject->getRvaPtr(RVA, IntPtr)) 1493861021f9SRui Ueyama return EC; 1494861021f9SRui Ueyama Result = *reinterpret_cast<const ulittle16_t *>(IntPtr); 14957d099195SRui Ueyama return std::error_code(); 1496861021f9SRui Ueyama } 1497861021f9SRui Ueyama 1498437b0d58SRafael Espindola ErrorOr<std::unique_ptr<COFFObjectFile>> 149948af1c2aSRafael Espindola ObjectFile::createCOFFObjectFile(MemoryBufferRef Object) { 1500db4ed0bdSRafael Espindola std::error_code EC; 150148af1c2aSRafael Espindola std::unique_ptr<COFFObjectFile> Ret(new COFFObjectFile(Object, EC)); 1502692410efSRafael Espindola if (EC) 1503692410efSRafael Espindola return EC; 1504437b0d58SRafael Espindola return std::move(Ret); 1505686738e2SRui Ueyama } 150674e85130SRui Ueyama 150774e85130SRui Ueyama bool BaseRelocRef::operator==(const BaseRelocRef &Other) const { 150874e85130SRui Ueyama return Header == Other.Header && Index == Other.Index; 150974e85130SRui Ueyama } 151074e85130SRui Ueyama 151174e85130SRui Ueyama void BaseRelocRef::moveNext() { 151274e85130SRui Ueyama // Header->BlockSize is the size of the current block, including the 151374e85130SRui Ueyama // size of the header itself. 151474e85130SRui Ueyama uint32_t Size = sizeof(*Header) + 1515970dda29SRui Ueyama sizeof(coff_base_reloc_block_entry) * (Index + 1); 151674e85130SRui Ueyama if (Size == Header->BlockSize) { 151774e85130SRui Ueyama // .reloc contains a list of base relocation blocks. Each block 151874e85130SRui Ueyama // consists of the header followed by entries. The header contains 151974e85130SRui Ueyama // how many entories will follow. When we reach the end of the 152074e85130SRui Ueyama // current block, proceed to the next block. 152174e85130SRui Ueyama Header = reinterpret_cast<const coff_base_reloc_block_header *>( 152274e85130SRui Ueyama reinterpret_cast<const uint8_t *>(Header) + Size); 152374e85130SRui Ueyama Index = 0; 152474e85130SRui Ueyama } else { 152574e85130SRui Ueyama ++Index; 152674e85130SRui Ueyama } 152774e85130SRui Ueyama } 152874e85130SRui Ueyama 152974e85130SRui Ueyama std::error_code BaseRelocRef::getType(uint8_t &Type) const { 153074e85130SRui Ueyama auto *Entry = reinterpret_cast<const coff_base_reloc_block_entry *>(Header + 1); 153174e85130SRui Ueyama Type = Entry[Index].getType(); 15327d099195SRui Ueyama return std::error_code(); 153374e85130SRui Ueyama } 153474e85130SRui Ueyama 153574e85130SRui Ueyama std::error_code BaseRelocRef::getRVA(uint32_t &Result) const { 153674e85130SRui Ueyama auto *Entry = reinterpret_cast<const coff_base_reloc_block_entry *>(Header + 1); 153774e85130SRui Ueyama Result = Header->PageRVA + Entry[Index].getOffset(); 15387d099195SRui Ueyama return std::error_code(); 153974e85130SRui Ueyama } 1540