1ab2eb2bfSHubert Tong //===--- XCOFFObjectFile.cpp - XCOFF object file implementation -----------===// 2ab2eb2bfSHubert Tong // 3ab2eb2bfSHubert Tong // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4ab2eb2bfSHubert Tong // See https://llvm.org/LICENSE.txt for license information. 5ab2eb2bfSHubert Tong // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6ab2eb2bfSHubert Tong // 7ab2eb2bfSHubert Tong //===----------------------------------------------------------------------===// 8ab2eb2bfSHubert Tong // 9ab2eb2bfSHubert Tong // This file defines the XCOFFObjectFile class. 10ab2eb2bfSHubert Tong // 11ab2eb2bfSHubert Tong //===----------------------------------------------------------------------===// 12ab2eb2bfSHubert Tong 13ab2eb2bfSHubert Tong #include "llvm/Object/XCOFFObjectFile.h" 14d33b016aSChen Zheng #include "llvm/ADT/StringSwitch.h" 150865d414SSimon Pilgrim #include "llvm/MC/SubtargetFeature.h" 162f0d755dSdiggerlin #include "llvm/Support/DataExtractor.h" 17ab2eb2bfSHubert Tong #include <cstddef> 18ab2eb2bfSHubert Tong #include <cstring> 19ab2eb2bfSHubert Tong 20ab2eb2bfSHubert Tong namespace llvm { 212f0d755dSdiggerlin 222f0d755dSdiggerlin using namespace XCOFF; 232f0d755dSdiggerlin 24ab2eb2bfSHubert Tong namespace object { 25ab2eb2bfSHubert Tong 26d60d7d69Sjasonliu static const uint8_t FunctionSym = 0x20; 27d60d7d69Sjasonliu static const uint16_t NoRelMask = 0x0001; 288e84311aSjasonliu static const size_t SymbolAuxTypeOffset = 17; 297c72e82bSJason Liu 30837ae69fSSean Fertile // Checks that [Ptr, Ptr + Size) bytes fall inside the memory buffer 31837ae69fSSean Fertile // 'M'. Returns a pointer to the underlying object on success. 32ab2eb2bfSHubert Tong template <typename T> 33837ae69fSSean Fertile static Expected<const T *> getObject(MemoryBufferRef M, const void *Ptr, 34ab2eb2bfSHubert Tong const uint64_t Size = sizeof(T)) { 3522cf54a7SArthur O'Dwyer uintptr_t Addr = reinterpret_cast<uintptr_t>(Ptr); 36e03a135bSReid Kleckner if (Error E = Binary::checkOffset(M, Addr, Size)) 37e03a135bSReid Kleckner return std::move(E); 38837ae69fSSean Fertile return reinterpret_cast<const T *>(Addr); 39837ae69fSSean Fertile } 40837ae69fSSean Fertile 41837ae69fSSean Fertile static uintptr_t getWithOffset(uintptr_t Base, ptrdiff_t Offset) { 42837ae69fSSean Fertile return reinterpret_cast<uintptr_t>(reinterpret_cast<const char *>(Base) + 43837ae69fSSean Fertile Offset); 44ab2eb2bfSHubert Tong } 45ab2eb2bfSHubert Tong 46a93a33cbSSean Fertile template <typename T> static const T *viewAs(uintptr_t in) { 47a93a33cbSSean Fertile return reinterpret_cast<const T *>(in); 48a93a33cbSSean Fertile } 49a93a33cbSSean Fertile 507c72e82bSJason Liu static StringRef generateXCOFFFixedNameStringRef(const char *Name) { 5129141da7SSean Fertile auto NulCharPtr = 5229141da7SSean Fertile static_cast<const char *>(memchr(Name, '\0', XCOFF::NameSize)); 539212206dSJason Liu return NulCharPtr ? StringRef(Name, NulCharPtr - Name) 5429141da7SSean Fertile : StringRef(Name, XCOFF::NameSize); 559212206dSJason Liu } 569212206dSJason Liu 57c63c1a72Sdiggerlin template <typename T> StringRef XCOFFSectionHeader<T>::getName() const { 58c63c1a72Sdiggerlin const T &DerivedXCOFFSectionHeader = static_cast<const T &>(*this); 59c63c1a72Sdiggerlin return generateXCOFFFixedNameStringRef(DerivedXCOFFSectionHeader.Name); 60c63c1a72Sdiggerlin } 61c63c1a72Sdiggerlin 62c63c1a72Sdiggerlin template <typename T> uint16_t XCOFFSectionHeader<T>::getSectionType() const { 63c63c1a72Sdiggerlin const T &DerivedXCOFFSectionHeader = static_cast<const T &>(*this); 64c63c1a72Sdiggerlin return DerivedXCOFFSectionHeader.Flags & SectionFlagsTypeMask; 65c63c1a72Sdiggerlin } 66c63c1a72Sdiggerlin 67c63c1a72Sdiggerlin template <typename T> 68c63c1a72Sdiggerlin bool XCOFFSectionHeader<T>::isReservedSectionType() const { 69c63c1a72Sdiggerlin return getSectionType() & SectionFlagsReservedMask; 70c63c1a72Sdiggerlin } 71c63c1a72Sdiggerlin 7234d4bff3SDigger Lin bool XCOFFRelocation32::isRelocationSigned() const { 7334d4bff3SDigger Lin return Info & XR_SIGN_INDICATOR_MASK; 7434d4bff3SDigger Lin } 7534d4bff3SDigger Lin 7634d4bff3SDigger Lin bool XCOFFRelocation32::isFixupIndicated() const { 7734d4bff3SDigger Lin return Info & XR_FIXUP_INDICATOR_MASK; 7834d4bff3SDigger Lin } 7934d4bff3SDigger Lin 8034d4bff3SDigger Lin uint8_t XCOFFRelocation32::getRelocatedLength() const { 8134d4bff3SDigger Lin // The relocation encodes the bit length being relocated minus 1. Add back 8234d4bff3SDigger Lin // the 1 to get the actual length being relocated. 8334d4bff3SDigger Lin return (Info & XR_BIASED_LENGTH_MASK) + 1; 8434d4bff3SDigger Lin } 8534d4bff3SDigger Lin 868e84311aSjasonliu uintptr_t 878e84311aSjasonliu XCOFFObjectFile::getAdvancedSymbolEntryAddress(uintptr_t CurrentAddress, 888e84311aSjasonliu uint32_t Distance) { 898e84311aSjasonliu return getWithOffset(CurrentAddress, Distance * XCOFF::SymbolTableEntrySize); 908e84311aSjasonliu } 918e84311aSjasonliu 928e84311aSjasonliu const XCOFF::SymbolAuxType * 938e84311aSjasonliu XCOFFObjectFile::getSymbolAuxType(uintptr_t AuxEntryAddress) const { 948e84311aSjasonliu assert(is64Bit() && "64-bit interface called on a 32-bit object file."); 958e84311aSjasonliu return viewAs<XCOFF::SymbolAuxType>( 968e84311aSjasonliu getWithOffset(AuxEntryAddress, SymbolAuxTypeOffset)); 978e84311aSjasonliu } 988e84311aSjasonliu 99837ae69fSSean Fertile void XCOFFObjectFile::checkSectionAddress(uintptr_t Addr, 100837ae69fSSean Fertile uintptr_t TableAddress) const { 101837ae69fSSean Fertile if (Addr < TableAddress) 102a93a33cbSSean Fertile report_fatal_error("Section header outside of section header table."); 103a93a33cbSSean Fertile 104837ae69fSSean Fertile uintptr_t Offset = Addr - TableAddress; 105837ae69fSSean Fertile if (Offset >= getSectionHeaderSize() * getNumberOfSections()) 106837ae69fSSean Fertile report_fatal_error("Section header outside of section header table."); 107837ae69fSSean Fertile 108a93a33cbSSean Fertile if (Offset % getSectionHeaderSize() != 0) 109a93a33cbSSean Fertile report_fatal_error( 110a93a33cbSSean Fertile "Section header pointer does not point to a valid section header."); 111837ae69fSSean Fertile } 112837ae69fSSean Fertile 113837ae69fSSean Fertile const XCOFFSectionHeader32 * 114837ae69fSSean Fertile XCOFFObjectFile::toSection32(DataRefImpl Ref) const { 115837ae69fSSean Fertile assert(!is64Bit() && "32-bit interface called on 64-bit object file."); 116837ae69fSSean Fertile #ifndef NDEBUG 117837ae69fSSean Fertile checkSectionAddress(Ref.p, getSectionHeaderTableAddress()); 118a93a33cbSSean Fertile #endif 119837ae69fSSean Fertile return viewAs<XCOFFSectionHeader32>(Ref.p); 120837ae69fSSean Fertile } 121837ae69fSSean Fertile 122837ae69fSSean Fertile const XCOFFSectionHeader64 * 123837ae69fSSean Fertile XCOFFObjectFile::toSection64(DataRefImpl Ref) const { 124837ae69fSSean Fertile assert(is64Bit() && "64-bit interface called on a 32-bit object file."); 125837ae69fSSean Fertile #ifndef NDEBUG 126837ae69fSSean Fertile checkSectionAddress(Ref.p, getSectionHeaderTableAddress()); 127837ae69fSSean Fertile #endif 128837ae69fSSean Fertile return viewAs<XCOFFSectionHeader64>(Ref.p); 129a93a33cbSSean Fertile } 130a93a33cbSSean Fertile 1318e84311aSjasonliu XCOFFSymbolRef XCOFFObjectFile::toSymbolRef(DataRefImpl Ref) const { 1329212206dSJason Liu assert(Ref.p != 0 && "Symbol table pointer can not be nullptr!"); 1337c72e82bSJason Liu #ifndef NDEBUG 1347c72e82bSJason Liu checkSymbolEntryPointer(Ref.p); 1357c72e82bSJason Liu #endif 1368e84311aSjasonliu return XCOFFSymbolRef(Ref, this); 1379212206dSJason Liu } 1389212206dSJason Liu 139837ae69fSSean Fertile const XCOFFFileHeader32 *XCOFFObjectFile::fileHeader32() const { 140837ae69fSSean Fertile assert(!is64Bit() && "32-bit interface called on 64-bit object file."); 141837ae69fSSean Fertile return static_cast<const XCOFFFileHeader32 *>(FileHeader); 142a93a33cbSSean Fertile } 143a93a33cbSSean Fertile 144837ae69fSSean Fertile const XCOFFFileHeader64 *XCOFFObjectFile::fileHeader64() const { 145837ae69fSSean Fertile assert(is64Bit() && "64-bit interface called on a 32-bit object file."); 146837ae69fSSean Fertile return static_cast<const XCOFFFileHeader64 *>(FileHeader); 147a93a33cbSSean Fertile } 148a93a33cbSSean Fertile 149837ae69fSSean Fertile const XCOFFSectionHeader32 * 150837ae69fSSean Fertile XCOFFObjectFile::sectionHeaderTable32() const { 151837ae69fSSean Fertile assert(!is64Bit() && "32-bit interface called on 64-bit object file."); 152837ae69fSSean Fertile return static_cast<const XCOFFSectionHeader32 *>(SectionHeaderTable); 153837ae69fSSean Fertile } 154837ae69fSSean Fertile 155837ae69fSSean Fertile const XCOFFSectionHeader64 * 156837ae69fSSean Fertile XCOFFObjectFile::sectionHeaderTable64() const { 157837ae69fSSean Fertile assert(is64Bit() && "64-bit interface called on a 32-bit object file."); 158837ae69fSSean Fertile return static_cast<const XCOFFSectionHeader64 *>(SectionHeaderTable); 159837ae69fSSean Fertile } 1609212206dSJason Liu 161ab2eb2bfSHubert Tong void XCOFFObjectFile::moveSymbolNext(DataRefImpl &Symb) const { 1628e84311aSjasonliu uintptr_t NextSymbolAddr = getAdvancedSymbolEntryAddress( 1638e84311aSjasonliu Symb.p, toSymbolRef(Symb).getNumberOfAuxEntries() + 1); 1647c72e82bSJason Liu #ifndef NDEBUG 1657c72e82bSJason Liu // This function is used by basic_symbol_iterator, which allows to 1667c72e82bSJason Liu // point to the end-of-symbol-table address. 1678e84311aSjasonliu if (NextSymbolAddr != getEndOfSymbolTableAddress()) 1688e84311aSjasonliu checkSymbolEntryPointer(NextSymbolAddr); 1697c72e82bSJason Liu #endif 1708e84311aSjasonliu Symb.p = NextSymbolAddr; 171ab2eb2bfSHubert Tong } 172ab2eb2bfSHubert Tong 1737c72e82bSJason Liu Expected<StringRef> 1747c72e82bSJason Liu XCOFFObjectFile::getStringTableEntry(uint32_t Offset) const { 1757c72e82bSJason Liu // The byte offset is relative to the start of the string table. 1767c72e82bSJason Liu // A byte offset value of 0 is a null or zero-length symbol 1779212206dSJason Liu // name. A byte offset in the range 1 to 3 (inclusive) points into the length 1789212206dSJason Liu // field; as a soft-error recovery mechanism, we treat such cases as having an 1799212206dSJason Liu // offset of 0. 1809212206dSJason Liu if (Offset < 4) 1819212206dSJason Liu return StringRef(nullptr, 0); 1829212206dSJason Liu 1839212206dSJason Liu if (StringTable.Data != nullptr && StringTable.Size > Offset) 1849212206dSJason Liu return (StringTable.Data + Offset); 1859212206dSJason Liu 1867c72e82bSJason Liu return make_error<GenericBinaryError>("Bad offset for string table entry", 1879212206dSJason Liu object_error::parse_failed); 188ab2eb2bfSHubert Tong } 189ab2eb2bfSHubert Tong 1907c72e82bSJason Liu Expected<StringRef> 1917c72e82bSJason Liu XCOFFObjectFile::getCFileName(const XCOFFFileAuxEnt *CFileEntPtr) const { 1928e84311aSjasonliu if (CFileEntPtr->NameInStrTbl.Magic != XCOFFSymbolRef::NAME_IN_STR_TBL_MAGIC) 1937c72e82bSJason Liu return generateXCOFFFixedNameStringRef(CFileEntPtr->Name); 1947c72e82bSJason Liu return getStringTableEntry(CFileEntPtr->NameInStrTbl.Offset); 1957c72e82bSJason Liu } 1967c72e82bSJason Liu 1977c72e82bSJason Liu Expected<StringRef> XCOFFObjectFile::getSymbolName(DataRefImpl Symb) const { 1988e84311aSjasonliu return toSymbolRef(Symb).getName(); 1997c72e82bSJason Liu } 2007c72e82bSJason Liu 201ab2eb2bfSHubert Tong Expected<uint64_t> XCOFFObjectFile::getSymbolAddress(DataRefImpl Symb) const { 2028e84311aSjasonliu return toSymbolRef(Symb).getValue(); 203ab2eb2bfSHubert Tong } 204ab2eb2bfSHubert Tong 205ab2eb2bfSHubert Tong uint64_t XCOFFObjectFile::getSymbolValueImpl(DataRefImpl Symb) const { 2068e84311aSjasonliu return toSymbolRef(Symb).getValue(); 207ab2eb2bfSHubert Tong } 208ab2eb2bfSHubert Tong 209ab2eb2bfSHubert Tong uint64_t XCOFFObjectFile::getCommonSymbolSizeImpl(DataRefImpl Symb) const { 210ab2eb2bfSHubert Tong uint64_t Result = 0; 211ab2eb2bfSHubert Tong llvm_unreachable("Not yet implemented!"); 212ab2eb2bfSHubert Tong return Result; 213ab2eb2bfSHubert Tong } 214ab2eb2bfSHubert Tong 215ab2eb2bfSHubert Tong Expected<SymbolRef::Type> 216ab2eb2bfSHubert Tong XCOFFObjectFile::getSymbolType(DataRefImpl Symb) const { 217*8a23f74eSEsme-Yi // TODO: Return the correct symbol type. 218ab2eb2bfSHubert Tong return SymbolRef::ST_Other; 219ab2eb2bfSHubert Tong } 220ab2eb2bfSHubert Tong 221ab2eb2bfSHubert Tong Expected<section_iterator> 222ab2eb2bfSHubert Tong XCOFFObjectFile::getSymbolSection(DataRefImpl Symb) const { 2238e84311aSjasonliu const int16_t SectNum = toSymbolRef(Symb).getSectionNumber(); 2249212206dSJason Liu 2259212206dSJason Liu if (isReservedSectionNumber(SectNum)) 2269212206dSJason Liu return section_end(); 2279212206dSJason Liu 228837ae69fSSean Fertile Expected<DataRefImpl> ExpSec = getSectionByNum(SectNum); 229837ae69fSSean Fertile if (!ExpSec) 230837ae69fSSean Fertile return ExpSec.takeError(); 2319212206dSJason Liu 232837ae69fSSean Fertile return section_iterator(SectionRef(ExpSec.get(), this)); 233ab2eb2bfSHubert Tong } 234ab2eb2bfSHubert Tong 235ab2eb2bfSHubert Tong void XCOFFObjectFile::moveSectionNext(DataRefImpl &Sec) const { 236a93a33cbSSean Fertile const char *Ptr = reinterpret_cast<const char *>(Sec.p); 237a93a33cbSSean Fertile Sec.p = reinterpret_cast<uintptr_t>(Ptr + getSectionHeaderSize()); 238ab2eb2bfSHubert Tong } 239ab2eb2bfSHubert Tong 2408be28cdcSFangrui Song Expected<StringRef> XCOFFObjectFile::getSectionName(DataRefImpl Sec) const { 2417c72e82bSJason Liu return generateXCOFFFixedNameStringRef(getSectionNameInternal(Sec)); 242ab2eb2bfSHubert Tong } 243ab2eb2bfSHubert Tong 244ab2eb2bfSHubert Tong uint64_t XCOFFObjectFile::getSectionAddress(DataRefImpl Sec) const { 245210314aeSSean Fertile // Avoid ternary due to failure to convert the ubig32_t value to a unit64_t 246210314aeSSean Fertile // with MSVC. 247210314aeSSean Fertile if (is64Bit()) 248210314aeSSean Fertile return toSection64(Sec)->VirtualAddress; 249210314aeSSean Fertile 250210314aeSSean Fertile return toSection32(Sec)->VirtualAddress; 251ab2eb2bfSHubert Tong } 252ab2eb2bfSHubert Tong 253ab2eb2bfSHubert Tong uint64_t XCOFFObjectFile::getSectionIndex(DataRefImpl Sec) const { 254a93a33cbSSean Fertile // Section numbers in XCOFF are numbered beginning at 1. A section number of 255a93a33cbSSean Fertile // zero is used to indicate that a symbol is being imported or is undefined. 256837ae69fSSean Fertile if (is64Bit()) 257837ae69fSSean Fertile return toSection64(Sec) - sectionHeaderTable64() + 1; 258837ae69fSSean Fertile else 259837ae69fSSean Fertile return toSection32(Sec) - sectionHeaderTable32() + 1; 260ab2eb2bfSHubert Tong } 261ab2eb2bfSHubert Tong 262ab2eb2bfSHubert Tong uint64_t XCOFFObjectFile::getSectionSize(DataRefImpl Sec) const { 263210314aeSSean Fertile // Avoid ternary due to failure to convert the ubig32_t value to a unit64_t 264210314aeSSean Fertile // with MSVC. 265210314aeSSean Fertile if (is64Bit()) 266210314aeSSean Fertile return toSection64(Sec)->SectionSize; 267210314aeSSean Fertile 268210314aeSSean Fertile return toSection32(Sec)->SectionSize; 269ab2eb2bfSHubert Tong } 270ab2eb2bfSHubert Tong 271e1cb2c0fSFangrui Song Expected<ArrayRef<uint8_t>> 272e1cb2c0fSFangrui Song XCOFFObjectFile::getSectionContents(DataRefImpl Sec) const { 273b91f798fSdiggerlin if (isSectionVirtual(Sec)) 274b91f798fSdiggerlin return ArrayRef<uint8_t>(); 275b91f798fSdiggerlin 276ea13683fSdiggerlin uint64_t OffsetToRaw; 277ea13683fSdiggerlin if (is64Bit()) 278ea13683fSdiggerlin OffsetToRaw = toSection64(Sec)->FileOffsetToRawData; 279ea13683fSdiggerlin else 280ea13683fSdiggerlin OffsetToRaw = toSection32(Sec)->FileOffsetToRawData; 281b91f798fSdiggerlin 282b91f798fSdiggerlin const uint8_t * ContentStart = base() + OffsetToRaw; 283b91f798fSdiggerlin uint64_t SectionSize = getSectionSize(Sec); 28422cf54a7SArthur O'Dwyer if (checkOffset(Data, reinterpret_cast<uintptr_t>(ContentStart), SectionSize)) 285b91f798fSdiggerlin return make_error<BinaryError>(); 286b91f798fSdiggerlin 287b91f798fSdiggerlin return makeArrayRef(ContentStart,SectionSize); 288ab2eb2bfSHubert Tong } 289ab2eb2bfSHubert Tong 290ab2eb2bfSHubert Tong uint64_t XCOFFObjectFile::getSectionAlignment(DataRefImpl Sec) const { 291ab2eb2bfSHubert Tong uint64_t Result = 0; 292ab2eb2bfSHubert Tong llvm_unreachable("Not yet implemented!"); 293ab2eb2bfSHubert Tong return Result; 294ab2eb2bfSHubert Tong } 295ab2eb2bfSHubert Tong 296ab2eb2bfSHubert Tong bool XCOFFObjectFile::isSectionCompressed(DataRefImpl Sec) const { 297d33b016aSChen Zheng return false; 298ab2eb2bfSHubert Tong } 299ab2eb2bfSHubert Tong 300ab2eb2bfSHubert Tong bool XCOFFObjectFile::isSectionText(DataRefImpl Sec) const { 301837ae69fSSean Fertile return getSectionFlags(Sec) & XCOFF::STYP_TEXT; 302ab2eb2bfSHubert Tong } 303ab2eb2bfSHubert Tong 304ab2eb2bfSHubert Tong bool XCOFFObjectFile::isSectionData(DataRefImpl Sec) const { 305837ae69fSSean Fertile uint32_t Flags = getSectionFlags(Sec); 306a93a33cbSSean Fertile return Flags & (XCOFF::STYP_DATA | XCOFF::STYP_TDATA); 307ab2eb2bfSHubert Tong } 308ab2eb2bfSHubert Tong 309ab2eb2bfSHubert Tong bool XCOFFObjectFile::isSectionBSS(DataRefImpl Sec) const { 310837ae69fSSean Fertile uint32_t Flags = getSectionFlags(Sec); 311a93a33cbSSean Fertile return Flags & (XCOFF::STYP_BSS | XCOFF::STYP_TBSS); 312ab2eb2bfSHubert Tong } 313ab2eb2bfSHubert Tong 314ab2eb2bfSHubert Tong bool XCOFFObjectFile::isSectionVirtual(DataRefImpl Sec) const { 315b91f798fSdiggerlin return is64Bit() ? toSection64(Sec)->FileOffsetToRawData == 0 316b91f798fSdiggerlin : toSection32(Sec)->FileOffsetToRawData == 0; 317ab2eb2bfSHubert Tong } 318ab2eb2bfSHubert Tong 319ab2eb2bfSHubert Tong relocation_iterator XCOFFObjectFile::section_rel_begin(DataRefImpl Sec) const { 320d60d7d69Sjasonliu if (is64Bit()) 321d60d7d69Sjasonliu report_fatal_error("64-bit support not implemented yet"); 322d60d7d69Sjasonliu const XCOFFSectionHeader32 *SectionEntPtr = toSection32(Sec); 323d60d7d69Sjasonliu auto RelocationsOrErr = relocations(*SectionEntPtr); 324d60d7d69Sjasonliu if (Error E = RelocationsOrErr.takeError()) 325ab2eb2bfSHubert Tong return relocation_iterator(RelocationRef()); 326d60d7d69Sjasonliu DataRefImpl Ret; 327d60d7d69Sjasonliu Ret.p = reinterpret_cast<uintptr_t>(&*RelocationsOrErr.get().begin()); 328d60d7d69Sjasonliu return relocation_iterator(RelocationRef(Ret, this)); 329ab2eb2bfSHubert Tong } 330ab2eb2bfSHubert Tong 331ab2eb2bfSHubert Tong relocation_iterator XCOFFObjectFile::section_rel_end(DataRefImpl Sec) const { 332d60d7d69Sjasonliu if (is64Bit()) 333d60d7d69Sjasonliu report_fatal_error("64-bit support not implemented yet"); 334d60d7d69Sjasonliu const XCOFFSectionHeader32 *SectionEntPtr = toSection32(Sec); 335d60d7d69Sjasonliu auto RelocationsOrErr = relocations(*SectionEntPtr); 336d60d7d69Sjasonliu if (Error E = RelocationsOrErr.takeError()) 337ab2eb2bfSHubert Tong return relocation_iterator(RelocationRef()); 338d60d7d69Sjasonliu DataRefImpl Ret; 339d60d7d69Sjasonliu Ret.p = reinterpret_cast<uintptr_t>(&*RelocationsOrErr.get().end()); 340d60d7d69Sjasonliu return relocation_iterator(RelocationRef(Ret, this)); 341ab2eb2bfSHubert Tong } 342ab2eb2bfSHubert Tong 343ab2eb2bfSHubert Tong void XCOFFObjectFile::moveRelocationNext(DataRefImpl &Rel) const { 344d60d7d69Sjasonliu Rel.p = reinterpret_cast<uintptr_t>(viewAs<XCOFFRelocation32>(Rel.p) + 1); 345ab2eb2bfSHubert Tong } 346ab2eb2bfSHubert Tong 347ab2eb2bfSHubert Tong uint64_t XCOFFObjectFile::getRelocationOffset(DataRefImpl Rel) const { 348d60d7d69Sjasonliu if (is64Bit()) 349d60d7d69Sjasonliu report_fatal_error("64-bit support not implemented yet"); 350d60d7d69Sjasonliu const XCOFFRelocation32 *Reloc = viewAs<XCOFFRelocation32>(Rel.p); 351d60d7d69Sjasonliu const XCOFFSectionHeader32 *Sec32 = sectionHeaderTable32(); 352d60d7d69Sjasonliu const uint32_t RelocAddress = Reloc->VirtualAddress; 353d60d7d69Sjasonliu const uint16_t NumberOfSections = getNumberOfSections(); 354d60d7d69Sjasonliu for (uint16_t i = 0; i < NumberOfSections; ++i) { 355d60d7d69Sjasonliu // Find which section this relocation is belonging to, and get the 356d60d7d69Sjasonliu // relocation offset relative to the start of the section. 357d60d7d69Sjasonliu if (Sec32->VirtualAddress <= RelocAddress && 358d60d7d69Sjasonliu RelocAddress < Sec32->VirtualAddress + Sec32->SectionSize) { 359d60d7d69Sjasonliu return RelocAddress - Sec32->VirtualAddress; 360d60d7d69Sjasonliu } 361d60d7d69Sjasonliu ++Sec32; 362d60d7d69Sjasonliu } 363d60d7d69Sjasonliu return InvalidRelocOffset; 364ab2eb2bfSHubert Tong } 365ab2eb2bfSHubert Tong 366ab2eb2bfSHubert Tong symbol_iterator XCOFFObjectFile::getRelocationSymbol(DataRefImpl Rel) const { 367d60d7d69Sjasonliu if (is64Bit()) 368d60d7d69Sjasonliu report_fatal_error("64-bit support not implemented yet"); 369d60d7d69Sjasonliu const XCOFFRelocation32 *Reloc = viewAs<XCOFFRelocation32>(Rel.p); 370d60d7d69Sjasonliu const uint32_t Index = Reloc->SymbolIndex; 371d60d7d69Sjasonliu 372d60d7d69Sjasonliu if (Index >= getLogicalNumberOfSymbolTableEntries32()) 373d60d7d69Sjasonliu return symbol_end(); 374d60d7d69Sjasonliu 375d60d7d69Sjasonliu DataRefImpl SymDRI; 3768e84311aSjasonliu SymDRI.p = getSymbolEntryAddressByIndex(Index); 377d60d7d69Sjasonliu return symbol_iterator(SymbolRef(SymDRI, this)); 378ab2eb2bfSHubert Tong } 379ab2eb2bfSHubert Tong 380ab2eb2bfSHubert Tong uint64_t XCOFFObjectFile::getRelocationType(DataRefImpl Rel) const { 381d60d7d69Sjasonliu if (is64Bit()) 382d60d7d69Sjasonliu report_fatal_error("64-bit support not implemented yet"); 383d60d7d69Sjasonliu return viewAs<XCOFFRelocation32>(Rel.p)->Type; 384ab2eb2bfSHubert Tong } 385ab2eb2bfSHubert Tong 386ab2eb2bfSHubert Tong void XCOFFObjectFile::getRelocationTypeName( 387ab2eb2bfSHubert Tong DataRefImpl Rel, SmallVectorImpl<char> &Result) const { 388d60d7d69Sjasonliu if (is64Bit()) 389d60d7d69Sjasonliu report_fatal_error("64-bit support not implemented yet"); 390d60d7d69Sjasonliu const XCOFFRelocation32 *Reloc = viewAs<XCOFFRelocation32>(Rel.p); 391d60d7d69Sjasonliu StringRef Res = XCOFF::getRelocationTypeString(Reloc->Type); 392d60d7d69Sjasonliu Result.append(Res.begin(), Res.end()); 393ab2eb2bfSHubert Tong } 394ab2eb2bfSHubert Tong 395ac00376aSvgxbj Expected<uint32_t> XCOFFObjectFile::getSymbolFlags(DataRefImpl Symb) const { 396ab2eb2bfSHubert Tong uint32_t Result = 0; 397*8a23f74eSEsme-Yi // TODO: Return correct symbol flags. 398ab2eb2bfSHubert Tong return Result; 399ab2eb2bfSHubert Tong } 400ab2eb2bfSHubert Tong 401ab2eb2bfSHubert Tong basic_symbol_iterator XCOFFObjectFile::symbol_begin() const { 4029212206dSJason Liu DataRefImpl SymDRI; 4039212206dSJason Liu SymDRI.p = reinterpret_cast<uintptr_t>(SymbolTblPtr); 4049212206dSJason Liu return basic_symbol_iterator(SymbolRef(SymDRI, this)); 405ab2eb2bfSHubert Tong } 406ab2eb2bfSHubert Tong 407ab2eb2bfSHubert Tong basic_symbol_iterator XCOFFObjectFile::symbol_end() const { 4089212206dSJason Liu DataRefImpl SymDRI; 4098e84311aSjasonliu const uint32_t NumberOfSymbolTableEntries = getNumberOfSymbolTableEntries(); 4108e84311aSjasonliu SymDRI.p = getSymbolEntryAddressByIndex(NumberOfSymbolTableEntries); 4119212206dSJason Liu return basic_symbol_iterator(SymbolRef(SymDRI, this)); 412ab2eb2bfSHubert Tong } 413ab2eb2bfSHubert Tong 414ab2eb2bfSHubert Tong section_iterator XCOFFObjectFile::section_begin() const { 415a93a33cbSSean Fertile DataRefImpl DRI; 416837ae69fSSean Fertile DRI.p = getSectionHeaderTableAddress(); 417a93a33cbSSean Fertile return section_iterator(SectionRef(DRI, this)); 418ab2eb2bfSHubert Tong } 419ab2eb2bfSHubert Tong 420ab2eb2bfSHubert Tong section_iterator XCOFFObjectFile::section_end() const { 421a93a33cbSSean Fertile DataRefImpl DRI; 422837ae69fSSean Fertile DRI.p = getWithOffset(getSectionHeaderTableAddress(), 423837ae69fSSean Fertile getNumberOfSections() * getSectionHeaderSize()); 424a93a33cbSSean Fertile return section_iterator(SectionRef(DRI, this)); 425ab2eb2bfSHubert Tong } 426ab2eb2bfSHubert Tong 427837ae69fSSean Fertile uint8_t XCOFFObjectFile::getBytesInAddress() const { return is64Bit() ? 8 : 4; } 428ab2eb2bfSHubert Tong 429ab2eb2bfSHubert Tong StringRef XCOFFObjectFile::getFileFormatName() const { 430837ae69fSSean Fertile return is64Bit() ? "aix5coff64-rs6000" : "aixcoff-rs6000"; 431ab2eb2bfSHubert Tong } 432ab2eb2bfSHubert Tong 433ab2eb2bfSHubert Tong Triple::ArchType XCOFFObjectFile::getArch() const { 434837ae69fSSean Fertile return is64Bit() ? Triple::ppc64 : Triple::ppc; 435ab2eb2bfSHubert Tong } 436ab2eb2bfSHubert Tong 437ab2eb2bfSHubert Tong SubtargetFeatures XCOFFObjectFile::getFeatures() const { 438ab2eb2bfSHubert Tong return SubtargetFeatures(); 439ab2eb2bfSHubert Tong } 440ab2eb2bfSHubert Tong 441ab2eb2bfSHubert Tong bool XCOFFObjectFile::isRelocatableObject() const { 442d60d7d69Sjasonliu if (is64Bit()) 4438ec73e96SAmir Ayupov return !(fileHeader64()->Flags & NoRelMask); 444d60d7d69Sjasonliu return !(fileHeader32()->Flags & NoRelMask); 445ab2eb2bfSHubert Tong } 446ab2eb2bfSHubert Tong 447a93a33cbSSean Fertile Expected<uint64_t> XCOFFObjectFile::getStartAddress() const { 448a93a33cbSSean Fertile // TODO FIXME Should get from auxiliary_header->o_entry when support for the 449a93a33cbSSean Fertile // auxiliary_header is added. 450a93a33cbSSean Fertile return 0; 451a93a33cbSSean Fertile } 452a93a33cbSSean Fertile 453d33b016aSChen Zheng StringRef XCOFFObjectFile::mapDebugSectionName(StringRef Name) const { 454d33b016aSChen Zheng return StringSwitch<StringRef>(Name) 455d33b016aSChen Zheng .Case("dwinfo", "debug_info") 456d33b016aSChen Zheng .Case("dwline", "debug_line") 457d33b016aSChen Zheng .Case("dwpbnms", "debug_pubnames") 458d33b016aSChen Zheng .Case("dwpbtyp", "debug_pubtypes") 459d33b016aSChen Zheng .Case("dwarnge", "debug_aranges") 460d33b016aSChen Zheng .Case("dwabrev", "debug_abbrev") 461d33b016aSChen Zheng .Case("dwstr", "debug_str") 462d33b016aSChen Zheng .Case("dwrnges", "debug_ranges") 463d33b016aSChen Zheng .Case("dwloc", "debug_loc") 464d33b016aSChen Zheng .Case("dwframe", "debug_frame") 465d33b016aSChen Zheng .Case("dwmac", "debug_macinfo") 466d33b016aSChen Zheng .Default(Name); 467d33b016aSChen Zheng } 468d33b016aSChen Zheng 469837ae69fSSean Fertile size_t XCOFFObjectFile::getFileHeaderSize() const { 470837ae69fSSean Fertile return is64Bit() ? sizeof(XCOFFFileHeader64) : sizeof(XCOFFFileHeader32); 4719212206dSJason Liu } 4729212206dSJason Liu 473837ae69fSSean Fertile size_t XCOFFObjectFile::getSectionHeaderSize() const { 474837ae69fSSean Fertile return is64Bit() ? sizeof(XCOFFSectionHeader64) : 475837ae69fSSean Fertile sizeof(XCOFFSectionHeader32); 476837ae69fSSean Fertile } 477837ae69fSSean Fertile 478837ae69fSSean Fertile bool XCOFFObjectFile::is64Bit() const { 479837ae69fSSean Fertile return Binary::ID_XCOFF64 == getType(); 480837ae69fSSean Fertile } 481837ae69fSSean Fertile 482837ae69fSSean Fertile uint16_t XCOFFObjectFile::getMagic() const { 483837ae69fSSean Fertile return is64Bit() ? fileHeader64()->Magic : fileHeader32()->Magic; 484837ae69fSSean Fertile } 485837ae69fSSean Fertile 486837ae69fSSean Fertile Expected<DataRefImpl> XCOFFObjectFile::getSectionByNum(int16_t Num) const { 487837ae69fSSean Fertile if (Num <= 0 || Num > getNumberOfSections()) 488837ae69fSSean Fertile return errorCodeToError(object_error::invalid_section_index); 489837ae69fSSean Fertile 490837ae69fSSean Fertile DataRefImpl DRI; 491837ae69fSSean Fertile DRI.p = getWithOffset(getSectionHeaderTableAddress(), 492837ae69fSSean Fertile getSectionHeaderSize() * (Num - 1)); 493837ae69fSSean Fertile return DRI; 4949212206dSJason Liu } 4959212206dSJason Liu 4969212206dSJason Liu Expected<StringRef> 4978e84311aSjasonliu XCOFFObjectFile::getSymbolSectionName(XCOFFSymbolRef SymEntPtr) const { 4988e84311aSjasonliu const int16_t SectionNum = SymEntPtr.getSectionNumber(); 4999212206dSJason Liu 5009212206dSJason Liu switch (SectionNum) { 5019212206dSJason Liu case XCOFF::N_DEBUG: 5029212206dSJason Liu return "N_DEBUG"; 5039212206dSJason Liu case XCOFF::N_ABS: 5049212206dSJason Liu return "N_ABS"; 5059212206dSJason Liu case XCOFF::N_UNDEF: 5069212206dSJason Liu return "N_UNDEF"; 507837ae69fSSean Fertile default: 508837ae69fSSean Fertile Expected<DataRefImpl> SecRef = getSectionByNum(SectionNum); 509837ae69fSSean Fertile if (SecRef) 5107c72e82bSJason Liu return generateXCOFFFixedNameStringRef( 5117c72e82bSJason Liu getSectionNameInternal(SecRef.get())); 512837ae69fSSean Fertile return SecRef.takeError(); 5139212206dSJason Liu } 5149212206dSJason Liu } 5159212206dSJason Liu 516*8a23f74eSEsme-Yi unsigned XCOFFObjectFile::getSymbolSectionID(SymbolRef Sym) const { 517*8a23f74eSEsme-Yi XCOFFSymbolRef XCOFFSymRef(Sym.getRawDataRefImpl(), this); 518*8a23f74eSEsme-Yi return XCOFFSymRef.getSectionNumber(); 519*8a23f74eSEsme-Yi } 520*8a23f74eSEsme-Yi 5219212206dSJason Liu bool XCOFFObjectFile::isReservedSectionNumber(int16_t SectionNumber) { 5229212206dSJason Liu return (SectionNumber <= 0 && SectionNumber >= -2); 5239212206dSJason Liu } 5249212206dSJason Liu 5259212206dSJason Liu uint16_t XCOFFObjectFile::getNumberOfSections() const { 526837ae69fSSean Fertile return is64Bit() ? fileHeader64()->NumberOfSections 527837ae69fSSean Fertile : fileHeader32()->NumberOfSections; 5289212206dSJason Liu } 5299212206dSJason Liu 530837ae69fSSean Fertile int32_t XCOFFObjectFile::getTimeStamp() const { 531837ae69fSSean Fertile return is64Bit() ? fileHeader64()->TimeStamp : fileHeader32()->TimeStamp; 5329212206dSJason Liu } 5339212206dSJason Liu 5349212206dSJason Liu uint16_t XCOFFObjectFile::getOptionalHeaderSize() const { 535837ae69fSSean Fertile return is64Bit() ? fileHeader64()->AuxHeaderSize 536837ae69fSSean Fertile : fileHeader32()->AuxHeaderSize; 5379212206dSJason Liu } 5389212206dSJason Liu 539837ae69fSSean Fertile uint32_t XCOFFObjectFile::getSymbolTableOffset32() const { 540837ae69fSSean Fertile return fileHeader32()->SymbolTableOffset; 541837ae69fSSean Fertile } 542ab2eb2bfSHubert Tong 543837ae69fSSean Fertile int32_t XCOFFObjectFile::getRawNumberOfSymbolTableEntries32() const { 544837ae69fSSean Fertile // As far as symbol table size is concerned, if this field is negative it is 545837ae69fSSean Fertile // to be treated as a 0. However since this field is also used for printing we 546837ae69fSSean Fertile // don't want to truncate any negative values. 547837ae69fSSean Fertile return fileHeader32()->NumberOfSymTableEntries; 548837ae69fSSean Fertile } 549ab2eb2bfSHubert Tong 550837ae69fSSean Fertile uint32_t XCOFFObjectFile::getLogicalNumberOfSymbolTableEntries32() const { 551837ae69fSSean Fertile return (fileHeader32()->NumberOfSymTableEntries >= 0 552837ae69fSSean Fertile ? fileHeader32()->NumberOfSymTableEntries 553837ae69fSSean Fertile : 0); 554837ae69fSSean Fertile } 555a93a33cbSSean Fertile 556837ae69fSSean Fertile uint64_t XCOFFObjectFile::getSymbolTableOffset64() const { 557837ae69fSSean Fertile return fileHeader64()->SymbolTableOffset; 558837ae69fSSean Fertile } 559837ae69fSSean Fertile 560837ae69fSSean Fertile uint32_t XCOFFObjectFile::getNumberOfSymbolTableEntries64() const { 561837ae69fSSean Fertile return fileHeader64()->NumberOfSymTableEntries; 562837ae69fSSean Fertile } 563837ae69fSSean Fertile 5648e84311aSjasonliu uint32_t XCOFFObjectFile::getNumberOfSymbolTableEntries() const { 5658e84311aSjasonliu return is64Bit() ? getNumberOfSymbolTableEntries64() 5667c72e82bSJason Liu : getLogicalNumberOfSymbolTableEntries32(); 5678e84311aSjasonliu } 5688e84311aSjasonliu 5698e84311aSjasonliu uintptr_t XCOFFObjectFile::getEndOfSymbolTableAddress() const { 5708e84311aSjasonliu const uint32_t NumberOfSymTableEntries = getNumberOfSymbolTableEntries(); 5717c72e82bSJason Liu return getWithOffset(reinterpret_cast<uintptr_t>(SymbolTblPtr), 5727c72e82bSJason Liu XCOFF::SymbolTableEntrySize * NumberOfSymTableEntries); 5737c72e82bSJason Liu } 5747c72e82bSJason Liu 5757c72e82bSJason Liu void XCOFFObjectFile::checkSymbolEntryPointer(uintptr_t SymbolEntPtr) const { 5767c72e82bSJason Liu if (SymbolEntPtr < reinterpret_cast<uintptr_t>(SymbolTblPtr)) 5777c72e82bSJason Liu report_fatal_error("Symbol table entry is outside of symbol table."); 5787c72e82bSJason Liu 5797c72e82bSJason Liu if (SymbolEntPtr >= getEndOfSymbolTableAddress()) 5807c72e82bSJason Liu report_fatal_error("Symbol table entry is outside of symbol table."); 5817c72e82bSJason Liu 5827c72e82bSJason Liu ptrdiff_t Offset = reinterpret_cast<const char *>(SymbolEntPtr) - 5837c72e82bSJason Liu reinterpret_cast<const char *>(SymbolTblPtr); 5847c72e82bSJason Liu 5857c72e82bSJason Liu if (Offset % XCOFF::SymbolTableEntrySize != 0) 5867c72e82bSJason Liu report_fatal_error( 5877c72e82bSJason Liu "Symbol table entry position is not valid inside of symbol table."); 5887c72e82bSJason Liu } 5897c72e82bSJason Liu 5907c72e82bSJason Liu uint32_t XCOFFObjectFile::getSymbolIndex(uintptr_t SymbolEntPtr) const { 5917c72e82bSJason Liu return (reinterpret_cast<const char *>(SymbolEntPtr) - 5927c72e82bSJason Liu reinterpret_cast<const char *>(SymbolTblPtr)) / 5937c72e82bSJason Liu XCOFF::SymbolTableEntrySize; 5947c72e82bSJason Liu } 5957c72e82bSJason Liu 5968e84311aSjasonliu uintptr_t XCOFFObjectFile::getSymbolEntryAddressByIndex(uint32_t Index) const { 5978e84311aSjasonliu return getAdvancedSymbolEntryAddress( 5988e84311aSjasonliu reinterpret_cast<uintptr_t>(getPointerToSymbolTable()), Index); 5998e84311aSjasonliu } 6008e84311aSjasonliu 60134d4bff3SDigger Lin Expected<StringRef> 60234d4bff3SDigger Lin XCOFFObjectFile::getSymbolNameByIndex(uint32_t Index) const { 6038e84311aSjasonliu const uint32_t NumberOfSymTableEntries = getNumberOfSymbolTableEntries(); 60434d4bff3SDigger Lin 6058e84311aSjasonliu if (Index >= NumberOfSymTableEntries) 60634d4bff3SDigger Lin return errorCodeToError(object_error::invalid_symbol_index); 60734d4bff3SDigger Lin 60834d4bff3SDigger Lin DataRefImpl SymDRI; 6098e84311aSjasonliu SymDRI.p = getSymbolEntryAddressByIndex(Index); 61034d4bff3SDigger Lin return getSymbolName(SymDRI); 61134d4bff3SDigger Lin } 61234d4bff3SDigger Lin 613837ae69fSSean Fertile uint16_t XCOFFObjectFile::getFlags() const { 614837ae69fSSean Fertile return is64Bit() ? fileHeader64()->Flags : fileHeader32()->Flags; 615837ae69fSSean Fertile } 616837ae69fSSean Fertile 617837ae69fSSean Fertile const char *XCOFFObjectFile::getSectionNameInternal(DataRefImpl Sec) const { 618837ae69fSSean Fertile return is64Bit() ? toSection64(Sec)->Name : toSection32(Sec)->Name; 619837ae69fSSean Fertile } 620837ae69fSSean Fertile 621837ae69fSSean Fertile uintptr_t XCOFFObjectFile::getSectionHeaderTableAddress() const { 622837ae69fSSean Fertile return reinterpret_cast<uintptr_t>(SectionHeaderTable); 623837ae69fSSean Fertile } 624837ae69fSSean Fertile 625837ae69fSSean Fertile int32_t XCOFFObjectFile::getSectionFlags(DataRefImpl Sec) const { 626837ae69fSSean Fertile return is64Bit() ? toSection64(Sec)->Flags : toSection32(Sec)->Flags; 627837ae69fSSean Fertile } 628837ae69fSSean Fertile 629837ae69fSSean Fertile XCOFFObjectFile::XCOFFObjectFile(unsigned int Type, MemoryBufferRef Object) 630837ae69fSSean Fertile : ObjectFile(Type, Object) { 631837ae69fSSean Fertile assert(Type == Binary::ID_XCOFF32 || Type == Binary::ID_XCOFF64); 632837ae69fSSean Fertile } 633837ae69fSSean Fertile 634837ae69fSSean Fertile ArrayRef<XCOFFSectionHeader64> XCOFFObjectFile::sections64() const { 635837ae69fSSean Fertile assert(is64Bit() && "64-bit interface called for non 64-bit file."); 636837ae69fSSean Fertile const XCOFFSectionHeader64 *TablePtr = sectionHeaderTable64(); 637837ae69fSSean Fertile return ArrayRef<XCOFFSectionHeader64>(TablePtr, 638837ae69fSSean Fertile TablePtr + getNumberOfSections()); 639837ae69fSSean Fertile } 640837ae69fSSean Fertile 641837ae69fSSean Fertile ArrayRef<XCOFFSectionHeader32> XCOFFObjectFile::sections32() const { 642837ae69fSSean Fertile assert(!is64Bit() && "32-bit interface called for non 32-bit file."); 643837ae69fSSean Fertile const XCOFFSectionHeader32 *TablePtr = sectionHeaderTable32(); 644837ae69fSSean Fertile return ArrayRef<XCOFFSectionHeader32>(TablePtr, 645837ae69fSSean Fertile TablePtr + getNumberOfSections()); 646837ae69fSSean Fertile } 647837ae69fSSean Fertile 64834d4bff3SDigger Lin // In an XCOFF32 file, when the field value is 65535, then an STYP_OVRFLO 64934d4bff3SDigger Lin // section header contains the actual count of relocation entries in the s_paddr 65034d4bff3SDigger Lin // field. STYP_OVRFLO headers contain the section index of their corresponding 65134d4bff3SDigger Lin // sections as their raw "NumberOfRelocations" field value. 65234d4bff3SDigger Lin Expected<uint32_t> XCOFFObjectFile::getLogicalNumberOfRelocationEntries( 65334d4bff3SDigger Lin const XCOFFSectionHeader32 &Sec) const { 65434d4bff3SDigger Lin 65534d4bff3SDigger Lin uint16_t SectionIndex = &Sec - sectionHeaderTable32() + 1; 65634d4bff3SDigger Lin 657775ef445Sjasonliu if (Sec.NumberOfRelocations < XCOFF::RelocOverflow) 65834d4bff3SDigger Lin return Sec.NumberOfRelocations; 65934d4bff3SDigger Lin for (const auto &Sec : sections32()) { 66034d4bff3SDigger Lin if (Sec.Flags == XCOFF::STYP_OVRFLO && 66134d4bff3SDigger Lin Sec.NumberOfRelocations == SectionIndex) 66234d4bff3SDigger Lin return Sec.PhysicalAddress; 66334d4bff3SDigger Lin } 66434d4bff3SDigger Lin return errorCodeToError(object_error::parse_failed); 66534d4bff3SDigger Lin } 66634d4bff3SDigger Lin 66734d4bff3SDigger Lin Expected<ArrayRef<XCOFFRelocation32>> 66834d4bff3SDigger Lin XCOFFObjectFile::relocations(const XCOFFSectionHeader32 &Sec) const { 66934d4bff3SDigger Lin uintptr_t RelocAddr = getWithOffset(reinterpret_cast<uintptr_t>(FileHeader), 67034d4bff3SDigger Lin Sec.FileOffsetToRelocationInfo); 67134d4bff3SDigger Lin auto NumRelocEntriesOrErr = getLogicalNumberOfRelocationEntries(Sec); 67234d4bff3SDigger Lin if (Error E = NumRelocEntriesOrErr.takeError()) 673c55cf4afSBill Wendling return std::move(E); 67434d4bff3SDigger Lin 67534d4bff3SDigger Lin uint32_t NumRelocEntries = NumRelocEntriesOrErr.get(); 67634d4bff3SDigger Lin 67749231c1fSKazu Hirata static_assert( 67849231c1fSKazu Hirata sizeof(XCOFFRelocation32) == XCOFF::RelocationSerializationSize32, ""); 67934d4bff3SDigger Lin auto RelocationOrErr = 68034d4bff3SDigger Lin getObject<XCOFFRelocation32>(Data, reinterpret_cast<void *>(RelocAddr), 68134d4bff3SDigger Lin NumRelocEntries * sizeof(XCOFFRelocation32)); 68234d4bff3SDigger Lin if (Error E = RelocationOrErr.takeError()) 683c55cf4afSBill Wendling return std::move(E); 68434d4bff3SDigger Lin 68534d4bff3SDigger Lin const XCOFFRelocation32 *StartReloc = RelocationOrErr.get(); 68634d4bff3SDigger Lin 68734d4bff3SDigger Lin return ArrayRef<XCOFFRelocation32>(StartReloc, StartReloc + NumRelocEntries); 68834d4bff3SDigger Lin } 68934d4bff3SDigger Lin 690837ae69fSSean Fertile Expected<XCOFFStringTable> 691837ae69fSSean Fertile XCOFFObjectFile::parseStringTable(const XCOFFObjectFile *Obj, uint64_t Offset) { 692837ae69fSSean Fertile // If there is a string table, then the buffer must contain at least 4 bytes 693837ae69fSSean Fertile // for the string table's size. Not having a string table is not an error. 694e03a135bSReid Kleckner if (Error E = Binary::checkOffset( 695e03a135bSReid Kleckner Obj->Data, reinterpret_cast<uintptr_t>(Obj->base() + Offset), 4)) { 696e03a135bSReid Kleckner consumeError(std::move(E)); 697837ae69fSSean Fertile return XCOFFStringTable{0, nullptr}; 698e03a135bSReid Kleckner } 699837ae69fSSean Fertile 700837ae69fSSean Fertile // Read the size out of the buffer. 701837ae69fSSean Fertile uint32_t Size = support::endian::read32be(Obj->base() + Offset); 702837ae69fSSean Fertile 703837ae69fSSean Fertile // If the size is less then 4, then the string table is just a size and no 704837ae69fSSean Fertile // string data. 705837ae69fSSean Fertile if (Size <= 4) 706837ae69fSSean Fertile return XCOFFStringTable{4, nullptr}; 707837ae69fSSean Fertile 708837ae69fSSean Fertile auto StringTableOrErr = 709837ae69fSSean Fertile getObject<char>(Obj->Data, Obj->base() + Offset, Size); 710837ae69fSSean Fertile if (Error E = StringTableOrErr.takeError()) 711c55cf4afSBill Wendling return std::move(E); 712837ae69fSSean Fertile 713837ae69fSSean Fertile const char *StringTablePtr = StringTableOrErr.get(); 714837ae69fSSean Fertile if (StringTablePtr[Size - 1] != '\0') 715837ae69fSSean Fertile return errorCodeToError(object_error::string_table_non_null_end); 716837ae69fSSean Fertile 717837ae69fSSean Fertile return XCOFFStringTable{Size, StringTablePtr}; 718837ae69fSSean Fertile } 719837ae69fSSean Fertile 720837ae69fSSean Fertile Expected<std::unique_ptr<XCOFFObjectFile>> 721837ae69fSSean Fertile XCOFFObjectFile::create(unsigned Type, MemoryBufferRef MBR) { 7220eaee545SJonas Devlieghere // Can't use std::make_unique because of the private constructor. 723837ae69fSSean Fertile std::unique_ptr<XCOFFObjectFile> Obj; 724837ae69fSSean Fertile Obj.reset(new XCOFFObjectFile(Type, MBR)); 725837ae69fSSean Fertile 726837ae69fSSean Fertile uint64_t CurOffset = 0; 727837ae69fSSean Fertile const auto *Base = Obj->base(); 728837ae69fSSean Fertile MemoryBufferRef Data = Obj->Data; 729837ae69fSSean Fertile 730837ae69fSSean Fertile // Parse file header. 731837ae69fSSean Fertile auto FileHeaderOrErr = 732837ae69fSSean Fertile getObject<void>(Data, Base + CurOffset, Obj->getFileHeaderSize()); 733837ae69fSSean Fertile if (Error E = FileHeaderOrErr.takeError()) 734c55cf4afSBill Wendling return std::move(E); 735837ae69fSSean Fertile Obj->FileHeader = FileHeaderOrErr.get(); 736837ae69fSSean Fertile 737837ae69fSSean Fertile CurOffset += Obj->getFileHeaderSize(); 738a93a33cbSSean Fertile // TODO FIXME we don't have support for an optional header yet, so just skip 739a93a33cbSSean Fertile // past it. 740837ae69fSSean Fertile CurOffset += Obj->getOptionalHeaderSize(); 741a93a33cbSSean Fertile 742837ae69fSSean Fertile // Parse the section header table if it is present. 743837ae69fSSean Fertile if (Obj->getNumberOfSections()) { 744837ae69fSSean Fertile auto SecHeadersOrErr = getObject<void>(Data, Base + CurOffset, 745837ae69fSSean Fertile Obj->getNumberOfSections() * 746837ae69fSSean Fertile Obj->getSectionHeaderSize()); 747837ae69fSSean Fertile if (Error E = SecHeadersOrErr.takeError()) 748c55cf4afSBill Wendling return std::move(E); 749837ae69fSSean Fertile Obj->SectionHeaderTable = SecHeadersOrErr.get(); 750a93a33cbSSean Fertile } 751ab2eb2bfSHubert Tong 7528e84311aSjasonliu const uint32_t NumberOfSymbolTableEntries = 7538e84311aSjasonliu Obj->getNumberOfSymbolTableEntries(); 754fd75ee91SSean Fertile 755837ae69fSSean Fertile // If there is no symbol table we are done parsing the memory buffer. 7568e84311aSjasonliu if (NumberOfSymbolTableEntries == 0) 757c55cf4afSBill Wendling return std::move(Obj); 758837ae69fSSean Fertile 759837ae69fSSean Fertile // Parse symbol table. 7608e84311aSjasonliu CurOffset = Obj->is64Bit() ? Obj->getSymbolTableOffset64() 7618e84311aSjasonliu : Obj->getSymbolTableOffset32(); 7628e84311aSjasonliu const uint64_t SymbolTableSize = 7638e84311aSjasonliu static_cast<uint64_t>(XCOFF::SymbolTableEntrySize) * 7648e84311aSjasonliu NumberOfSymbolTableEntries; 765837ae69fSSean Fertile auto SymTableOrErr = 7668e84311aSjasonliu getObject<void *>(Data, Base + CurOffset, SymbolTableSize); 767837ae69fSSean Fertile if (Error E = SymTableOrErr.takeError()) 768c55cf4afSBill Wendling return std::move(E); 769837ae69fSSean Fertile Obj->SymbolTblPtr = SymTableOrErr.get(); 770837ae69fSSean Fertile CurOffset += SymbolTableSize; 771fd75ee91SSean Fertile 772837ae69fSSean Fertile // Parse String table. 773837ae69fSSean Fertile Expected<XCOFFStringTable> StringTableOrErr = 774837ae69fSSean Fertile parseStringTable(Obj.get(), CurOffset); 775837ae69fSSean Fertile if (Error E = StringTableOrErr.takeError()) 776c55cf4afSBill Wendling return std::move(E); 777837ae69fSSean Fertile Obj->StringTable = StringTableOrErr.get(); 778fd75ee91SSean Fertile 779c55cf4afSBill Wendling return std::move(Obj); 780fd75ee91SSean Fertile } 781fd75ee91SSean Fertile 782ab2eb2bfSHubert Tong Expected<std::unique_ptr<ObjectFile>> 783837ae69fSSean Fertile ObjectFile::createXCOFFObjectFile(MemoryBufferRef MemBufRef, 784837ae69fSSean Fertile unsigned FileType) { 785837ae69fSSean Fertile return XCOFFObjectFile::create(FileType, MemBufRef); 786ab2eb2bfSHubert Tong } 787ab2eb2bfSHubert Tong 7888e84311aSjasonliu bool XCOFFSymbolRef::isFunction() const { 7898e84311aSjasonliu if (!isCsectSymbol()) 7908e84311aSjasonliu return false; 7918e84311aSjasonliu 7928e84311aSjasonliu if (getSymbolType() & FunctionSym) 7938e84311aSjasonliu return true; 7948e84311aSjasonliu 7958e84311aSjasonliu Expected<XCOFFCsectAuxRef> ExpCsectAuxEnt = getXCOFFCsectAuxRef(); 7968e84311aSjasonliu if (!ExpCsectAuxEnt) 7978e84311aSjasonliu return false; 7988e84311aSjasonliu 7998e84311aSjasonliu const XCOFFCsectAuxRef CsectAuxRef = ExpCsectAuxEnt.get(); 8008e84311aSjasonliu 8018e84311aSjasonliu // A function definition should be a label definition. 8028e84311aSjasonliu // FIXME: This is not necessarily the case when -ffunction-sections is 8038e84311aSjasonliu // enabled. 8048e84311aSjasonliu if (!CsectAuxRef.isLabel()) 8058e84311aSjasonliu return false; 8068e84311aSjasonliu 8078e84311aSjasonliu if (CsectAuxRef.getStorageMappingClass() != XCOFF::XMC_PR) 8088e84311aSjasonliu return false; 8098e84311aSjasonliu 8108e84311aSjasonliu const int16_t SectNum = getSectionNumber(); 8118e84311aSjasonliu Expected<DataRefImpl> SI = OwningObjectPtr->getSectionByNum(SectNum); 8128e84311aSjasonliu if (!SI) { 8138e84311aSjasonliu // If we could not get the section, then this symbol should not be 8148e84311aSjasonliu // a function. So consume the error and return `false` to move on. 8158e84311aSjasonliu consumeError(SI.takeError()); 8168e84311aSjasonliu return false; 8177c72e82bSJason Liu } 8187c72e82bSJason Liu 8198e84311aSjasonliu return (OwningObjectPtr->getSectionFlags(SI.get()) & XCOFF::STYP_TEXT); 8207c72e82bSJason Liu } 8217c72e82bSJason Liu 8228e84311aSjasonliu bool XCOFFSymbolRef::isCsectSymbol() const { 8237c72e82bSJason Liu XCOFF::StorageClass SC = getStorageClass(); 8247c72e82bSJason Liu return (SC == XCOFF::C_EXT || SC == XCOFF::C_WEAKEXT || 8257c72e82bSJason Liu SC == XCOFF::C_HIDEXT); 8267c72e82bSJason Liu } 8277c72e82bSJason Liu 8288e84311aSjasonliu Expected<XCOFFCsectAuxRef> XCOFFSymbolRef::getXCOFFCsectAuxRef() const { 8298e84311aSjasonliu assert(isCsectSymbol() && 8308e84311aSjasonliu "Calling csect symbol interface with a non-csect symbol."); 8317c72e82bSJason Liu 8328e84311aSjasonliu uint8_t NumberOfAuxEntries = getNumberOfAuxEntries(); 8337c72e82bSJason Liu 8348e84311aSjasonliu Expected<StringRef> NameOrErr = getName(); 8358e84311aSjasonliu if (auto Err = NameOrErr.takeError()) 8368e84311aSjasonliu return std::move(Err); 8377c72e82bSJason Liu 8388e84311aSjasonliu if (!NumberOfAuxEntries) { 8398e84311aSjasonliu return createStringError(object_error::parse_failed, 8408e84311aSjasonliu "csect symbol \"" + *NameOrErr + 8418e84311aSjasonliu "\" contains no auxiliary entry"); 8428e84311aSjasonliu } 8437c72e82bSJason Liu 8448e84311aSjasonliu if (!OwningObjectPtr->is64Bit()) { 8458e84311aSjasonliu // In XCOFF32, the csect auxilliary entry is always the last auxiliary 8468e84311aSjasonliu // entry for the symbol. 8478e84311aSjasonliu uintptr_t AuxAddr = XCOFFObjectFile::getAdvancedSymbolEntryAddress( 8488e84311aSjasonliu getEntryAddress(), NumberOfAuxEntries); 8498e84311aSjasonliu return XCOFFCsectAuxRef(viewAs<XCOFFCsectAuxEnt32>(AuxAddr)); 8508e84311aSjasonliu } 8517c72e82bSJason Liu 8528e84311aSjasonliu // XCOFF64 uses SymbolAuxType to identify the auxiliary entry type. 8538e84311aSjasonliu // We need to iterate through all the auxiliary entries to find it. 8548e84311aSjasonliu for (uint8_t Index = NumberOfAuxEntries; Index > 0; --Index) { 8558e84311aSjasonliu uintptr_t AuxAddr = XCOFFObjectFile::getAdvancedSymbolEntryAddress( 8568e84311aSjasonliu getEntryAddress(), Index); 8578e84311aSjasonliu if (*OwningObjectPtr->getSymbolAuxType(AuxAddr) == 8588e84311aSjasonliu XCOFF::SymbolAuxType::AUX_CSECT) { 8598e84311aSjasonliu #ifndef NDEBUG 8608e84311aSjasonliu OwningObjectPtr->checkSymbolEntryPointer(AuxAddr); 8618e84311aSjasonliu #endif 8628e84311aSjasonliu return XCOFFCsectAuxRef(viewAs<XCOFFCsectAuxEnt64>(AuxAddr)); 8638e84311aSjasonliu } 8648e84311aSjasonliu } 8657c72e82bSJason Liu 8668e84311aSjasonliu return createStringError( 8678e84311aSjasonliu object_error::parse_failed, 8688e84311aSjasonliu "a csect auxiliary entry is not found for symbol \"" + *NameOrErr + "\""); 8698e84311aSjasonliu } 8707c72e82bSJason Liu 8718e84311aSjasonliu Expected<StringRef> XCOFFSymbolRef::getName() const { 8728e84311aSjasonliu // A storage class value with the high-order bit on indicates that the name is 8738e84311aSjasonliu // a symbolic debugger stabstring. 8748e84311aSjasonliu if (getStorageClass() & 0x80) 8758e84311aSjasonliu return StringRef("Unimplemented Debug Name"); 8768e84311aSjasonliu 8778e84311aSjasonliu if (Entry32) { 8788e84311aSjasonliu if (Entry32->NameInStrTbl.Magic != XCOFFSymbolRef::NAME_IN_STR_TBL_MAGIC) 8798e84311aSjasonliu return generateXCOFFFixedNameStringRef(Entry32->SymbolName); 8808e84311aSjasonliu 8818e84311aSjasonliu return OwningObjectPtr->getStringTableEntry(Entry32->NameInStrTbl.Offset); 8828e84311aSjasonliu } 8838e84311aSjasonliu 8848e84311aSjasonliu return OwningObjectPtr->getStringTableEntry(Entry64->Offset); 885ab2eb2bfSHubert Tong } 886ab2eb2bfSHubert Tong 887f8622543SFangrui Song // Explictly instantiate template classes. 888f8622543SFangrui Song template struct XCOFFSectionHeader<XCOFFSectionHeader32>; 889f8622543SFangrui Song template struct XCOFFSectionHeader<XCOFFSectionHeader64>; 890f8622543SFangrui Song 8912f0d755dSdiggerlin bool doesXCOFFTracebackTableBegin(ArrayRef<uint8_t> Bytes) { 8922f0d755dSdiggerlin if (Bytes.size() < 4) 8932f0d755dSdiggerlin return false; 8942f0d755dSdiggerlin 8952f0d755dSdiggerlin return support::endian::read32be(Bytes.data()) == 0; 8962f0d755dSdiggerlin } 8972f0d755dSdiggerlin 898ab77fa51Sdiggerlin TBVectorExt::TBVectorExt(StringRef TBvectorStrRef) { 899ab77fa51Sdiggerlin const uint8_t *Ptr = reinterpret_cast<const uint8_t *>(TBvectorStrRef.data()); 900ab77fa51Sdiggerlin Data = support::endian::read16be(Ptr); 901ab77fa51Sdiggerlin VecParmsInfo = support::endian::read32be(Ptr + 2); 902ab77fa51Sdiggerlin } 903ab77fa51Sdiggerlin 904ab77fa51Sdiggerlin #define GETVALUEWITHMASK(X) (Data & (TracebackTable::X)) 905ab77fa51Sdiggerlin #define GETVALUEWITHMASKSHIFT(X, S) \ 906ab77fa51Sdiggerlin ((Data & (TracebackTable::X)) >> (TracebackTable::S)) 90785177965Sdiggerlin uint8_t TBVectorExt::getNumberOfVRSaved() const { 908ab77fa51Sdiggerlin return GETVALUEWITHMASKSHIFT(NumberOfVRSavedMask, NumberOfVRSavedShift); 909ab77fa51Sdiggerlin } 910ab77fa51Sdiggerlin 911ab77fa51Sdiggerlin bool TBVectorExt::isVRSavedOnStack() const { 912ab77fa51Sdiggerlin return GETVALUEWITHMASK(IsVRSavedOnStackMask); 913ab77fa51Sdiggerlin } 914ab77fa51Sdiggerlin 915ab77fa51Sdiggerlin bool TBVectorExt::hasVarArgs() const { 916ab77fa51Sdiggerlin return GETVALUEWITHMASK(HasVarArgsMask); 917ab77fa51Sdiggerlin } 918ab77fa51Sdiggerlin uint8_t TBVectorExt::getNumberOfVectorParms() const { 919ab77fa51Sdiggerlin return GETVALUEWITHMASKSHIFT(NumberOfVectorParmsMask, 920ab77fa51Sdiggerlin NumberOfVectorParmsShift); 921ab77fa51Sdiggerlin } 922ab77fa51Sdiggerlin 923ab77fa51Sdiggerlin bool TBVectorExt::hasVMXInstruction() const { 924ab77fa51Sdiggerlin return GETVALUEWITHMASK(HasVMXInstructionMask); 925ab77fa51Sdiggerlin } 926ab77fa51Sdiggerlin #undef GETVALUEWITHMASK 927ab77fa51Sdiggerlin #undef GETVALUEWITHMASKSHIFT 928ab77fa51Sdiggerlin 929ab77fa51Sdiggerlin SmallString<32> TBVectorExt::getVectorParmsInfoString() const { 930ab77fa51Sdiggerlin SmallString<32> ParmsType; 931ab77fa51Sdiggerlin uint32_t Value = VecParmsInfo; 932ab77fa51Sdiggerlin for (uint8_t I = 0; I < getNumberOfVectorParms(); ++I) { 933ab77fa51Sdiggerlin if (I != 0) 934ab77fa51Sdiggerlin ParmsType += ", "; 935ab77fa51Sdiggerlin switch (Value & TracebackTable::ParmTypeMask) { 936ab77fa51Sdiggerlin case TracebackTable::ParmTypeIsVectorCharBit: 937ab77fa51Sdiggerlin ParmsType += "vc"; 938ab77fa51Sdiggerlin break; 939ab77fa51Sdiggerlin 940ab77fa51Sdiggerlin case TracebackTable::ParmTypeIsVectorShortBit: 941ab77fa51Sdiggerlin ParmsType += "vs"; 942ab77fa51Sdiggerlin break; 943ab77fa51Sdiggerlin 944ab77fa51Sdiggerlin case TracebackTable::ParmTypeIsVectorIntBit: 945ab77fa51Sdiggerlin ParmsType += "vi"; 946ab77fa51Sdiggerlin break; 947ab77fa51Sdiggerlin 948ab77fa51Sdiggerlin case TracebackTable::ParmTypeIsVectorFloatBit: 949ab77fa51Sdiggerlin ParmsType += "vf"; 950ab77fa51Sdiggerlin break; 951ab77fa51Sdiggerlin } 952ab77fa51Sdiggerlin Value <<= 2; 953ab77fa51Sdiggerlin } 954ab77fa51Sdiggerlin return ParmsType; 955ab77fa51Sdiggerlin } 956ab77fa51Sdiggerlin 957ab77fa51Sdiggerlin static SmallString<32> parseParmsTypeWithVecInfo(uint32_t Value, 958ab77fa51Sdiggerlin unsigned int ParmsNum) { 959ab77fa51Sdiggerlin SmallString<32> ParmsType; 960ab77fa51Sdiggerlin unsigned I = 0; 961ab77fa51Sdiggerlin bool Begin = false; 962ab77fa51Sdiggerlin while (I < ParmsNum || Value) { 963ab77fa51Sdiggerlin if (Begin) 964ab77fa51Sdiggerlin ParmsType += ", "; 965ab77fa51Sdiggerlin else 966ab77fa51Sdiggerlin Begin = true; 967ab77fa51Sdiggerlin 968ab77fa51Sdiggerlin switch (Value & TracebackTable::ParmTypeMask) { 969ab77fa51Sdiggerlin case TracebackTable::ParmTypeIsFixedBits: 970ab77fa51Sdiggerlin ParmsType += "i"; 971ab77fa51Sdiggerlin ++I; 972ab77fa51Sdiggerlin break; 973ab77fa51Sdiggerlin case TracebackTable::ParmTypeIsVectorBits: 974ab77fa51Sdiggerlin ParmsType += "v"; 975ab77fa51Sdiggerlin break; 976ab77fa51Sdiggerlin case TracebackTable::ParmTypeIsFloatingBits: 977ab77fa51Sdiggerlin ParmsType += "f"; 978ab77fa51Sdiggerlin ++I; 979ab77fa51Sdiggerlin break; 980ab77fa51Sdiggerlin case TracebackTable::ParmTypeIsDoubleBits: 981ab77fa51Sdiggerlin ParmsType += "d"; 982ab77fa51Sdiggerlin ++I; 983ab77fa51Sdiggerlin break; 984ab77fa51Sdiggerlin default: 985ab77fa51Sdiggerlin assert(false && "Unrecognized bits in ParmsType."); 986ab77fa51Sdiggerlin } 987ab77fa51Sdiggerlin Value <<= 2; 988ab77fa51Sdiggerlin } 989ab77fa51Sdiggerlin assert(I == ParmsNum && 990ab77fa51Sdiggerlin "The total parameters number of fixed-point or floating-point " 991ab77fa51Sdiggerlin "parameters not equal to the number in the parameter type!"); 992ab77fa51Sdiggerlin return ParmsType; 993ab77fa51Sdiggerlin } 994ab77fa51Sdiggerlin 9952f0d755dSdiggerlin Expected<XCOFFTracebackTable> XCOFFTracebackTable::create(const uint8_t *Ptr, 9962f0d755dSdiggerlin uint64_t &Size) { 9972f0d755dSdiggerlin Error Err = Error::success(); 9982f0d755dSdiggerlin XCOFFTracebackTable TBT(Ptr, Size, Err); 9992f0d755dSdiggerlin if (Err) 10002f0d755dSdiggerlin return std::move(Err); 10012f0d755dSdiggerlin return TBT; 10022f0d755dSdiggerlin } 10032f0d755dSdiggerlin 10042f0d755dSdiggerlin XCOFFTracebackTable::XCOFFTracebackTable(const uint8_t *Ptr, uint64_t &Size, 10052f0d755dSdiggerlin Error &Err) 10062f0d755dSdiggerlin : TBPtr(Ptr) { 10072f0d755dSdiggerlin ErrorAsOutParameter EAO(&Err); 10082f0d755dSdiggerlin DataExtractor DE(ArrayRef<uint8_t>(Ptr, Size), /*IsLittleEndian=*/false, 10092f0d755dSdiggerlin /*AddressSize=*/0); 10102f0d755dSdiggerlin DataExtractor::Cursor Cur(/*Offset=*/0); 10112f0d755dSdiggerlin 10122f0d755dSdiggerlin // Skip 8 bytes of mandatory fields. 10132f0d755dSdiggerlin DE.getU64(Cur); 10142f0d755dSdiggerlin 10152f0d755dSdiggerlin // Begin to parse optional fields. 10162f0d755dSdiggerlin if (Cur) { 10172f0d755dSdiggerlin unsigned ParmNum = getNumberOfFixedParms() + getNumberOfFPParms(); 10182f0d755dSdiggerlin 10192f0d755dSdiggerlin // As long as there are no "fixed-point" or floating-point parameters, this 10202f0d755dSdiggerlin // field remains not present even when hasVectorInfo gives true and 10212f0d755dSdiggerlin // indicates the presence of vector parameters. 10222f0d755dSdiggerlin if (ParmNum > 0) { 10232f0d755dSdiggerlin uint32_t ParamsTypeValue = DE.getU32(Cur); 1024ab77fa51Sdiggerlin if (Cur) 1025ab77fa51Sdiggerlin ParmsType = hasVectorInfo() 1026ab77fa51Sdiggerlin ? parseParmsTypeWithVecInfo(ParamsTypeValue, ParmNum) 1027ab77fa51Sdiggerlin : parseParmsType(ParamsTypeValue, ParmNum); 10282f0d755dSdiggerlin } 10292f0d755dSdiggerlin } 10302f0d755dSdiggerlin 10312f0d755dSdiggerlin if (Cur && hasTraceBackTableOffset()) 10322f0d755dSdiggerlin TraceBackTableOffset = DE.getU32(Cur); 10332f0d755dSdiggerlin 10342f0d755dSdiggerlin if (Cur && isInterruptHandler()) 10352f0d755dSdiggerlin HandlerMask = DE.getU32(Cur); 10362f0d755dSdiggerlin 10372f0d755dSdiggerlin if (Cur && hasControlledStorage()) { 10382f0d755dSdiggerlin NumOfCtlAnchors = DE.getU32(Cur); 10392f0d755dSdiggerlin if (Cur && NumOfCtlAnchors) { 10402f0d755dSdiggerlin SmallVector<uint32_t, 8> Disp; 10412f0d755dSdiggerlin Disp.reserve(NumOfCtlAnchors.getValue()); 10422f0d755dSdiggerlin for (uint32_t I = 0; I < NumOfCtlAnchors && Cur; ++I) 10432f0d755dSdiggerlin Disp.push_back(DE.getU32(Cur)); 10442f0d755dSdiggerlin if (Cur) 10452f0d755dSdiggerlin ControlledStorageInfoDisp = std::move(Disp); 10462f0d755dSdiggerlin } 10472f0d755dSdiggerlin } 10482f0d755dSdiggerlin 10492f0d755dSdiggerlin if (Cur && isFuncNamePresent()) { 10502f0d755dSdiggerlin uint16_t FunctionNameLen = DE.getU16(Cur); 10512f0d755dSdiggerlin if (Cur) 10522f0d755dSdiggerlin FunctionName = DE.getBytes(Cur, FunctionNameLen); 10532f0d755dSdiggerlin } 10542f0d755dSdiggerlin 10552f0d755dSdiggerlin if (Cur && isAllocaUsed()) 10562f0d755dSdiggerlin AllocaRegister = DE.getU8(Cur); 10572f0d755dSdiggerlin 1058ab77fa51Sdiggerlin if (Cur && hasVectorInfo()) { 1059ab77fa51Sdiggerlin StringRef VectorExtRef = DE.getBytes(Cur, 6); 1060ab77fa51Sdiggerlin if (Cur) 1061ab77fa51Sdiggerlin VecExt = TBVectorExt(VectorExtRef); 1062ab77fa51Sdiggerlin } 1063ab77fa51Sdiggerlin 1064ab77fa51Sdiggerlin if (Cur && hasExtensionTable()) 1065ab77fa51Sdiggerlin ExtensionTable = DE.getU8(Cur); 10662f0d755dSdiggerlin 10672f0d755dSdiggerlin if (!Cur) 10682f0d755dSdiggerlin Err = Cur.takeError(); 10692f0d755dSdiggerlin Size = Cur.tell(); 10702f0d755dSdiggerlin } 10712f0d755dSdiggerlin 10722f0d755dSdiggerlin #define GETBITWITHMASK(P, X) \ 10732f0d755dSdiggerlin (support::endian::read32be(TBPtr + (P)) & (TracebackTable::X)) 10742f0d755dSdiggerlin #define GETBITWITHMASKSHIFT(P, X, S) \ 10752f0d755dSdiggerlin ((support::endian::read32be(TBPtr + (P)) & (TracebackTable::X)) >> \ 10762f0d755dSdiggerlin (TracebackTable::S)) 10772f0d755dSdiggerlin 10782f0d755dSdiggerlin uint8_t XCOFFTracebackTable::getVersion() const { 10792f0d755dSdiggerlin return GETBITWITHMASKSHIFT(0, VersionMask, VersionShift); 10802f0d755dSdiggerlin } 10812f0d755dSdiggerlin 10822f0d755dSdiggerlin uint8_t XCOFFTracebackTable::getLanguageID() const { 10832f0d755dSdiggerlin return GETBITWITHMASKSHIFT(0, LanguageIdMask, LanguageIdShift); 10842f0d755dSdiggerlin } 10852f0d755dSdiggerlin 10862f0d755dSdiggerlin bool XCOFFTracebackTable::isGlobalLinkage() const { 10872f0d755dSdiggerlin return GETBITWITHMASK(0, IsGlobaLinkageMask); 10882f0d755dSdiggerlin } 10892f0d755dSdiggerlin 10902f0d755dSdiggerlin bool XCOFFTracebackTable::isOutOfLineEpilogOrPrologue() const { 10912f0d755dSdiggerlin return GETBITWITHMASK(0, IsOutOfLineEpilogOrPrologueMask); 10922f0d755dSdiggerlin } 10932f0d755dSdiggerlin 10942f0d755dSdiggerlin bool XCOFFTracebackTable::hasTraceBackTableOffset() const { 10952f0d755dSdiggerlin return GETBITWITHMASK(0, HasTraceBackTableOffsetMask); 10962f0d755dSdiggerlin } 10972f0d755dSdiggerlin 10982f0d755dSdiggerlin bool XCOFFTracebackTable::isInternalProcedure() const { 10992f0d755dSdiggerlin return GETBITWITHMASK(0, IsInternalProcedureMask); 11002f0d755dSdiggerlin } 11012f0d755dSdiggerlin 11022f0d755dSdiggerlin bool XCOFFTracebackTable::hasControlledStorage() const { 11032f0d755dSdiggerlin return GETBITWITHMASK(0, HasControlledStorageMask); 11042f0d755dSdiggerlin } 11052f0d755dSdiggerlin 11062f0d755dSdiggerlin bool XCOFFTracebackTable::isTOCless() const { 11072f0d755dSdiggerlin return GETBITWITHMASK(0, IsTOClessMask); 11082f0d755dSdiggerlin } 11092f0d755dSdiggerlin 11102f0d755dSdiggerlin bool XCOFFTracebackTable::isFloatingPointPresent() const { 11112f0d755dSdiggerlin return GETBITWITHMASK(0, IsFloatingPointPresentMask); 11122f0d755dSdiggerlin } 11132f0d755dSdiggerlin 11142f0d755dSdiggerlin bool XCOFFTracebackTable::isFloatingPointOperationLogOrAbortEnabled() const { 11152f0d755dSdiggerlin return GETBITWITHMASK(0, IsFloatingPointOperationLogOrAbortEnabledMask); 11162f0d755dSdiggerlin } 11172f0d755dSdiggerlin 11182f0d755dSdiggerlin bool XCOFFTracebackTable::isInterruptHandler() const { 11192f0d755dSdiggerlin return GETBITWITHMASK(0, IsInterruptHandlerMask); 11202f0d755dSdiggerlin } 11212f0d755dSdiggerlin 11222f0d755dSdiggerlin bool XCOFFTracebackTable::isFuncNamePresent() const { 11232f0d755dSdiggerlin return GETBITWITHMASK(0, IsFunctionNamePresentMask); 11242f0d755dSdiggerlin } 11252f0d755dSdiggerlin 11262f0d755dSdiggerlin bool XCOFFTracebackTable::isAllocaUsed() const { 11272f0d755dSdiggerlin return GETBITWITHMASK(0, IsAllocaUsedMask); 11282f0d755dSdiggerlin } 11292f0d755dSdiggerlin 11302f0d755dSdiggerlin uint8_t XCOFFTracebackTable::getOnConditionDirective() const { 11312f0d755dSdiggerlin return GETBITWITHMASKSHIFT(0, OnConditionDirectiveMask, 11322f0d755dSdiggerlin OnConditionDirectiveShift); 11332f0d755dSdiggerlin } 11342f0d755dSdiggerlin 11352f0d755dSdiggerlin bool XCOFFTracebackTable::isCRSaved() const { 11362f0d755dSdiggerlin return GETBITWITHMASK(0, IsCRSavedMask); 11372f0d755dSdiggerlin } 11382f0d755dSdiggerlin 11392f0d755dSdiggerlin bool XCOFFTracebackTable::isLRSaved() const { 11402f0d755dSdiggerlin return GETBITWITHMASK(0, IsLRSavedMask); 11412f0d755dSdiggerlin } 11422f0d755dSdiggerlin 11432f0d755dSdiggerlin bool XCOFFTracebackTable::isBackChainStored() const { 11442f0d755dSdiggerlin return GETBITWITHMASK(4, IsBackChainStoredMask); 11452f0d755dSdiggerlin } 11462f0d755dSdiggerlin 11472f0d755dSdiggerlin bool XCOFFTracebackTable::isFixup() const { 11482f0d755dSdiggerlin return GETBITWITHMASK(4, IsFixupMask); 11492f0d755dSdiggerlin } 11502f0d755dSdiggerlin 11512f0d755dSdiggerlin uint8_t XCOFFTracebackTable::getNumOfFPRsSaved() const { 11522f0d755dSdiggerlin return GETBITWITHMASKSHIFT(4, FPRSavedMask, FPRSavedShift); 11532f0d755dSdiggerlin } 11542f0d755dSdiggerlin 11552f0d755dSdiggerlin bool XCOFFTracebackTable::hasExtensionTable() const { 11562f0d755dSdiggerlin return GETBITWITHMASK(4, HasExtensionTableMask); 11572f0d755dSdiggerlin } 11582f0d755dSdiggerlin 11592f0d755dSdiggerlin bool XCOFFTracebackTable::hasVectorInfo() const { 11602f0d755dSdiggerlin return GETBITWITHMASK(4, HasVectorInfoMask); 11612f0d755dSdiggerlin } 11622f0d755dSdiggerlin 1163c80fbdf2Sdiggerlin uint8_t XCOFFTracebackTable::getNumOfGPRsSaved() const { 11642f0d755dSdiggerlin return GETBITWITHMASKSHIFT(4, GPRSavedMask, GPRSavedShift); 11652f0d755dSdiggerlin } 11662f0d755dSdiggerlin 11672f0d755dSdiggerlin uint8_t XCOFFTracebackTable::getNumberOfFixedParms() const { 11682f0d755dSdiggerlin return GETBITWITHMASKSHIFT(4, NumberOfFixedParmsMask, 11692f0d755dSdiggerlin NumberOfFixedParmsShift); 11702f0d755dSdiggerlin } 11712f0d755dSdiggerlin 11722f0d755dSdiggerlin uint8_t XCOFFTracebackTable::getNumberOfFPParms() const { 11732f0d755dSdiggerlin return GETBITWITHMASKSHIFT(4, NumberOfFloatingPointParmsMask, 11742f0d755dSdiggerlin NumberOfFloatingPointParmsShift); 11752f0d755dSdiggerlin } 11762f0d755dSdiggerlin 11772f0d755dSdiggerlin bool XCOFFTracebackTable::hasParmsOnStack() const { 11782f0d755dSdiggerlin return GETBITWITHMASK(4, HasParmsOnStackMask); 11792f0d755dSdiggerlin } 11802f0d755dSdiggerlin 11812f0d755dSdiggerlin #undef GETBITWITHMASK 11822f0d755dSdiggerlin #undef GETBITWITHMASKSHIFT 1183ab2eb2bfSHubert Tong } // namespace object 1184ab2eb2bfSHubert Tong } // namespace llvm 1185