1d341c932SEugene Zelenko //===- ObjectFile.cpp - File format independent object file ---------------===// 2fe5ee802SMichael J. Spencer // 32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information. 52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6fe5ee802SMichael J. Spencer // 7fe5ee802SMichael J. Spencer //===----------------------------------------------------------------------===// 8fe5ee802SMichael J. Spencer // 9fe5ee802SMichael J. Spencer // This file defines a file format independent ObjectFile class. 10fe5ee802SMichael J. Spencer // 11fe5ee802SMichael J. Spencer //===----------------------------------------------------------------------===// 12fe5ee802SMichael J. Spencer 136bda14b3SChandler Carruth #include "llvm/Object/ObjectFile.h" 14*e72c195fSserge-sans-paille #include "llvm/ADT/DenseMapInfo.h" 15d341c932SEugene Zelenko #include "llvm/ADT/StringRef.h" 16264b5d9eSZachary Turner #include "llvm/BinaryFormat/Magic.h" 17d341c932SEugene Zelenko #include "llvm/Object/Binary.h" 18437b0d58SRafael Espindola #include "llvm/Object/COFF.h" 19d341c932SEugene Zelenko #include "llvm/Object/Error.h" 20437b0d58SRafael Espindola #include "llvm/Object/MachO.h" 212c6f75ddSDerek Schuff #include "llvm/Object/Wasm.h" 22d341c932SEugene Zelenko #include "llvm/Support/Error.h" 23fe5ee802SMichael J. Spencer #include "llvm/Support/ErrorHandling.h" 24d341c932SEugene Zelenko #include "llvm/Support/ErrorOr.h" 25*e72c195fSserge-sans-paille #include "llvm/Support/Format.h" 26fe5ee802SMichael J. Spencer #include "llvm/Support/MemoryBuffer.h" 27b1f8a5a5SAaron Ballman #include "llvm/Support/raw_ostream.h" 28d341c932SEugene Zelenko #include <cstdint> 29d341c932SEugene Zelenko #include <memory> 30a6e9c3e4SRafael Espindola #include <system_error> 31fe5ee802SMichael J. Spencer 32f177d16fSMichael J. Spencer using namespace llvm; 33f177d16fSMichael J. Spencer using namespace object; 34fe5ee802SMichael J. Spencer 35c0fc29c4SPavel Labath raw_ostream &object::operator<<(raw_ostream &OS, const SectionedAddress &Addr) { 36c0fc29c4SPavel Labath OS << "SectionedAddress{" << format_hex(Addr.Address, 10); 37c0fc29c4SPavel Labath if (Addr.SectionIndex != SectionedAddress::UndefSection) 38c0fc29c4SPavel Labath OS << ", " << Addr.SectionIndex; 39c0fc29c4SPavel Labath return OS << "}"; 40c0fc29c4SPavel Labath } 41c0fc29c4SPavel Labath 42a379b181SDavid Blaikie void ObjectFile::anchor() {} 43a379b181SDavid Blaikie 4448af1c2aSRafael Espindola ObjectFile::ObjectFile(unsigned int Type, MemoryBufferRef Source) 4548af1c2aSRafael Espindola : SymbolicFile(Type, Source) {} 46f12b8282SRafael Espindola 47e9c58c74SRafael Espindola bool SectionRef::containsSymbol(SymbolRef S) const { 487bd8d994SKevin Enderby Expected<section_iterator> SymSec = S.getSection(); 497bd8d994SKevin Enderby if (!SymSec) { 507bd8d994SKevin Enderby // TODO: Actually report errors helpfully. 517bd8d994SKevin Enderby consumeError(SymSec.takeError()); 52e9c58c74SRafael Espindola return false; 537bd8d994SKevin Enderby } 548bab889bSRafael Espindola return *this == **SymSec; 55e9c58c74SRafael Espindola } 56e9c58c74SRafael Espindola 57ff6a0b6aSXing GUO Expected<uint64_t> ObjectFile::getSymbolValue(DataRefImpl Ref) const { 582410fb46SDuncan P. N. Exon Smith uint32_t Flags; 592410fb46SDuncan P. N. Exon Smith if (Error E = getSymbolFlags(Ref).moveInto(Flags)) 60ff6a0b6aSXing GUO // TODO: Test this error. 612410fb46SDuncan P. N. Exon Smith return std::move(E); 622410fb46SDuncan P. N. Exon Smith 632410fb46SDuncan P. N. Exon Smith if (Flags & SymbolRef::SF_Undefined) 642410fb46SDuncan P. N. Exon Smith return 0; 652410fb46SDuncan P. N. Exon Smith if (Flags & SymbolRef::SF_Common) 662410fb46SDuncan P. N. Exon Smith return getCommonSymbolSize(Ref); 67be8b0ea8SRafael Espindola return getSymbolValueImpl(Ref); 68be8b0ea8SRafael Espindola } 69be8b0ea8SRafael Espindola 70e357ca82SFangrui Song Error ObjectFile::printSymbolName(raw_ostream &OS, DataRefImpl Symb) const { 7181e8b7d9SKevin Enderby Expected<StringRef> Name = getSymbolName(Symb); 7281e8b7d9SKevin Enderby if (!Name) 73e357ca82SFangrui Song return Name.takeError(); 745d0c2ffaSRafael Espindola OS << *Name; 75e357ca82SFangrui Song return Error::success(); 76f12b8282SRafael Espindola } 77fe5ee802SMichael J. Spencer 78a4d22472SRafael Espindola uint32_t ObjectFile::getSymbolAlignment(DataRefImpl DRI) const { return 0; } 79e4dd2e01SRafael Espindola 80f2fe0141SSteven Wu bool ObjectFile::isSectionBitcode(DataRefImpl Sec) const { 8178e8011aSGeorge Rimar Expected<StringRef> NameOrErr = getSectionName(Sec); 8278e8011aSGeorge Rimar if (NameOrErr) 838be28cdcSFangrui Song return *NameOrErr == ".llvmbc"; 8478e8011aSGeorge Rimar consumeError(NameOrErr.takeError()); 85f2fe0141SSteven Wu return false; 86f2fe0141SSteven Wu } 87f2fe0141SSteven Wu 888af2387bSJonas Devlieghere bool ObjectFile::isSectionStripped(DataRefImpl Sec) const { return false; } 898af2387bSJonas Devlieghere 904888c4abSJordan Rupprecht bool ObjectFile::isBerkeleyText(DataRefImpl Sec) const { 914888c4abSJordan Rupprecht return isSectionText(Sec); 924888c4abSJordan Rupprecht } 934888c4abSJordan Rupprecht 944888c4abSJordan Rupprecht bool ObjectFile::isBerkeleyData(DataRefImpl Sec) const { 954888c4abSJordan Rupprecht return isSectionData(Sec); 964888c4abSJordan Rupprecht } 974888c4abSJordan Rupprecht 98bf809cd1SEsme-Yi bool ObjectFile::isDebugSection(DataRefImpl Sec) const { return false; } 9929d253c4SDjordje Todorovic 1002bf01dcbSGeorge Rimar Expected<section_iterator> 1012bf01dcbSGeorge Rimar ObjectFile::getRelocatedSection(DataRefImpl Sec) const { 1024f60a38fSRafael Espindola return section_iterator(SectionRef(Sec, this)); 1034f60a38fSRafael Espindola } 1044f60a38fSRafael Espindola 105de62046dSVlad Tsyrklevich Triple ObjectFile::makeTriple() const { 106de62046dSVlad Tsyrklevich Triple TheTriple; 107de62046dSVlad Tsyrklevich auto Arch = getArch(); 108de62046dSVlad Tsyrklevich TheTriple.setArch(Triple::ArchType(Arch)); 109de62046dSVlad Tsyrklevich 110de62046dSVlad Tsyrklevich // For ARM targets, try to use the build attributes to build determine 111de62046dSVlad Tsyrklevich // the build target. Target features are also added, but later during 112de62046dSVlad Tsyrklevich // disassembly. 113de62046dSVlad Tsyrklevich if (Arch == Triple::arm || Arch == Triple::armeb) 114de62046dSVlad Tsyrklevich setARMSubArch(TheTriple); 115de62046dSVlad Tsyrklevich 116de62046dSVlad Tsyrklevich // TheTriple defaults to ELF, and COFF doesn't have an environment: 117478af447SHubert Tong // something we can do here is indicate that it is mach-o. 118478af447SHubert Tong if (isMachO()) { 119de62046dSVlad Tsyrklevich TheTriple.setObjectFormat(Triple::MachO); 120478af447SHubert Tong } else if (isCOFF()) { 121bc9a1b37SSimon Pilgrim const auto COFFObj = cast<COFFObjectFile>(this); 122de62046dSVlad Tsyrklevich if (COFFObj->getArch() == Triple::thumb) 123de62046dSVlad Tsyrklevich TheTriple.setTriple("thumbv7-windows"); 124478af447SHubert Tong } else if (isXCOFF()) { 125478af447SHubert Tong // XCOFF implies AIX. 126478af447SHubert Tong TheTriple.setOS(Triple::AIX); 127478af447SHubert Tong TheTriple.setObjectFormat(Triple::XCOFF); 128de62046dSVlad Tsyrklevich } 129de62046dSVlad Tsyrklevich 130de62046dSVlad Tsyrklevich return TheTriple; 131de62046dSVlad Tsyrklevich } 132de62046dSVlad Tsyrklevich 1333fcdf6aeSKevin Enderby Expected<std::unique_ptr<ObjectFile>> 13499a6401aSGeorgii Rymar ObjectFile::createObjectFile(MemoryBufferRef Object, file_magic Type, 13599a6401aSGeorgii Rymar bool InitContent) { 13648af1c2aSRafael Espindola StringRef Data = Object.getBuffer(); 137264b5d9eSZachary Turner if (Type == file_magic::unknown) 138264b5d9eSZachary Turner Type = identify_magic(Data); 13951cc3602SRafael Espindola 140cd81b909SRafael Espindola switch (Type) { 141264b5d9eSZachary Turner case file_magic::unknown: 142264b5d9eSZachary Turner case file_magic::bitcode: 143264b5d9eSZachary Turner case file_magic::coff_cl_gl_object: 144264b5d9eSZachary Turner case file_magic::archive: 145264b5d9eSZachary Turner case file_magic::macho_universal_binary: 146264b5d9eSZachary Turner case file_magic::windows_resource: 147aac28f31SZachary Turner case file_magic::pdb: 148581d79a4SPavel Labath case file_magic::minidump: 1490977f31cSAnirudh Prasad case file_magic::goff_object: 1503fcdf6aeSKevin Enderby return errorCodeToError(object_error::invalid_file_type); 151359840a6SCyndy Ishida case file_magic::tapi_file: 152359840a6SCyndy Ishida return errorCodeToError(object_error::invalid_file_type); 153264b5d9eSZachary Turner case file_magic::elf: 154264b5d9eSZachary Turner case file_magic::elf_relocatable: 155264b5d9eSZachary Turner case file_magic::elf_executable: 156264b5d9eSZachary Turner case file_magic::elf_shared_object: 157264b5d9eSZachary Turner case file_magic::elf_core: 15899a6401aSGeorgii Rymar return createELFObjectFile(Object, InitContent); 159264b5d9eSZachary Turner case file_magic::macho_object: 160264b5d9eSZachary Turner case file_magic::macho_executable: 161264b5d9eSZachary Turner case file_magic::macho_fixed_virtual_memory_shared_lib: 162264b5d9eSZachary Turner case file_magic::macho_core: 163264b5d9eSZachary Turner case file_magic::macho_preload_executable: 164264b5d9eSZachary Turner case file_magic::macho_dynamically_linked_shared_lib: 165264b5d9eSZachary Turner case file_magic::macho_dynamic_linker: 166264b5d9eSZachary Turner case file_magic::macho_bundle: 167264b5d9eSZachary Turner case file_magic::macho_dynamically_linked_shared_lib_stub: 168264b5d9eSZachary Turner case file_magic::macho_dsym_companion: 169264b5d9eSZachary Turner case file_magic::macho_kext_bundle: 1703fcdf6aeSKevin Enderby return createMachOObjectFile(Object); 171264b5d9eSZachary Turner case file_magic::coff_object: 172264b5d9eSZachary Turner case file_magic::coff_import_library: 173264b5d9eSZachary Turner case file_magic::pecoff_executable: 17412db383eSRafael Espindola return createCOFFObjectFile(Object); 175ab2eb2bfSHubert Tong case file_magic::xcoff_object_32: 176837ae69fSSean Fertile return createXCOFFObjectFile(Object, Binary::ID_XCOFF32); 177837ae69fSSean Fertile case file_magic::xcoff_object_64: 178837ae69fSSean Fertile return createXCOFFObjectFile(Object, Binary::ID_XCOFF64); 179264b5d9eSZachary Turner case file_magic::wasm_object: 1802c6f75ddSDerek Schuff return createWasmObjectFile(Object); 181fe5ee802SMichael J. Spencer } 182cd81b909SRafael Espindola llvm_unreachable("Unexpected Object File Type"); 183fe5ee802SMichael J. Spencer } 184fe5ee802SMichael J. Spencer 1853fcdf6aeSKevin Enderby Expected<OwningBinary<ObjectFile>> 186437b0d58SRafael Espindola ObjectFile::createObjectFile(StringRef ObjectPath) { 187adf21f2aSRafael Espindola ErrorOr<std::unique_ptr<MemoryBuffer>> FileOrErr = 188adf21f2aSRafael Espindola MemoryBuffer::getFile(ObjectPath); 189adf21f2aSRafael Espindola if (std::error_code EC = FileOrErr.getError()) 1903fcdf6aeSKevin Enderby return errorCodeToError(EC); 19148af1c2aSRafael Espindola std::unique_ptr<MemoryBuffer> Buffer = std::move(FileOrErr.get()); 19248af1c2aSRafael Espindola 1933fcdf6aeSKevin Enderby Expected<std::unique_ptr<ObjectFile>> ObjOrErr = 19448af1c2aSRafael Espindola createObjectFile(Buffer->getMemBufferRef()); 195b7c9deb5SJustin Bogner if (Error Err = ObjOrErr.takeError()) 196c55cf4afSBill Wendling return std::move(Err); 19748af1c2aSRafael Espindola std::unique_ptr<ObjectFile> Obj = std::move(ObjOrErr.get()); 19848af1c2aSRafael Espindola 19948af1c2aSRafael Espindola return OwningBinary<ObjectFile>(std::move(Obj), std::move(Buffer)); 200fe5ee802SMichael J. Spencer } 201