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" 14d341c932SEugene Zelenko #include "llvm/ADT/StringRef.h" 15264b5d9eSZachary Turner #include "llvm/BinaryFormat/Magic.h" 16d341c932SEugene Zelenko #include "llvm/Object/Binary.h" 17437b0d58SRafael Espindola #include "llvm/Object/COFF.h" 18d341c932SEugene Zelenko #include "llvm/Object/Error.h" 19437b0d58SRafael Espindola #include "llvm/Object/MachO.h" 202c6f75ddSDerek Schuff #include "llvm/Object/Wasm.h" 21d341c932SEugene Zelenko #include "llvm/Support/Error.h" 22fe5ee802SMichael J. Spencer #include "llvm/Support/ErrorHandling.h" 23d341c932SEugene Zelenko #include "llvm/Support/ErrorOr.h" 24cd81b909SRafael Espindola #include "llvm/Support/FileSystem.h" 25fe5ee802SMichael J. Spencer #include "llvm/Support/MemoryBuffer.h" 26b1f8a5a5SAaron Ballman #include "llvm/Support/raw_ostream.h" 27d341c932SEugene Zelenko #include <algorithm> 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 35a379b181SDavid Blaikie void ObjectFile::anchor() {} 36a379b181SDavid Blaikie 3748af1c2aSRafael Espindola ObjectFile::ObjectFile(unsigned int Type, MemoryBufferRef Source) 3848af1c2aSRafael Espindola : SymbolicFile(Type, Source) {} 39f12b8282SRafael Espindola 40e9c58c74SRafael Espindola bool SectionRef::containsSymbol(SymbolRef S) const { 417bd8d994SKevin Enderby Expected<section_iterator> SymSec = S.getSection(); 427bd8d994SKevin Enderby if (!SymSec) { 437bd8d994SKevin Enderby // TODO: Actually report errors helpfully. 447bd8d994SKevin Enderby consumeError(SymSec.takeError()); 45e9c58c74SRafael Espindola return false; 467bd8d994SKevin Enderby } 478bab889bSRafael Espindola return *this == **SymSec; 48e9c58c74SRafael Espindola } 49e9c58c74SRafael Espindola 50be8b0ea8SRafael Espindola uint64_t ObjectFile::getSymbolValue(DataRefImpl Ref) const { 51be8b0ea8SRafael Espindola uint32_t Flags = getSymbolFlags(Ref); 52be8b0ea8SRafael Espindola if (Flags & SymbolRef::SF_Undefined) 53be8b0ea8SRafael Espindola return 0; 54be8b0ea8SRafael Espindola if (Flags & SymbolRef::SF_Common) 55be8b0ea8SRafael Espindola return getCommonSymbolSize(Ref); 56be8b0ea8SRafael Espindola return getSymbolValueImpl(Ref); 57be8b0ea8SRafael Espindola } 58be8b0ea8SRafael Espindola 59e357ca82SFangrui Song Error ObjectFile::printSymbolName(raw_ostream &OS, DataRefImpl Symb) const { 6081e8b7d9SKevin Enderby Expected<StringRef> Name = getSymbolName(Symb); 6181e8b7d9SKevin Enderby if (!Name) 62e357ca82SFangrui Song return Name.takeError(); 635d0c2ffaSRafael Espindola OS << *Name; 64e357ca82SFangrui Song return Error::success(); 65f12b8282SRafael Espindola } 66fe5ee802SMichael J. Spencer 67a4d22472SRafael Espindola uint32_t ObjectFile::getSymbolAlignment(DataRefImpl DRI) const { return 0; } 68e4dd2e01SRafael Espindola 69f2fe0141SSteven Wu bool ObjectFile::isSectionBitcode(DataRefImpl Sec) const { 708be28cdcSFangrui Song if (Expected<StringRef> NameOrErr = getSectionName(Sec)) 718be28cdcSFangrui Song return *NameOrErr == ".llvmbc"; 72f2fe0141SSteven Wu return false; 73f2fe0141SSteven Wu } 74f2fe0141SSteven Wu 758af2387bSJonas Devlieghere bool ObjectFile::isSectionStripped(DataRefImpl Sec) const { return false; } 768af2387bSJonas Devlieghere 774888c4abSJordan Rupprecht bool ObjectFile::isBerkeleyText(DataRefImpl Sec) const { 784888c4abSJordan Rupprecht return isSectionText(Sec); 794888c4abSJordan Rupprecht } 804888c4abSJordan Rupprecht 814888c4abSJordan Rupprecht bool ObjectFile::isBerkeleyData(DataRefImpl Sec) const { 824888c4abSJordan Rupprecht return isSectionData(Sec); 834888c4abSJordan Rupprecht } 844888c4abSJordan Rupprecht 854f60a38fSRafael Espindola section_iterator ObjectFile::getRelocatedSection(DataRefImpl Sec) const { 864f60a38fSRafael Espindola return section_iterator(SectionRef(Sec, this)); 874f60a38fSRafael Espindola } 884f60a38fSRafael Espindola 89de62046dSVlad Tsyrklevich Triple ObjectFile::makeTriple() const { 90de62046dSVlad Tsyrklevich Triple TheTriple; 91de62046dSVlad Tsyrklevich auto Arch = getArch(); 92de62046dSVlad Tsyrklevich TheTriple.setArch(Triple::ArchType(Arch)); 93de62046dSVlad Tsyrklevich 94de62046dSVlad Tsyrklevich // For ARM targets, try to use the build attributes to build determine 95de62046dSVlad Tsyrklevich // the build target. Target features are also added, but later during 96de62046dSVlad Tsyrklevich // disassembly. 97de62046dSVlad Tsyrklevich if (Arch == Triple::arm || Arch == Triple::armeb) 98de62046dSVlad Tsyrklevich setARMSubArch(TheTriple); 99de62046dSVlad Tsyrklevich 100de62046dSVlad Tsyrklevich // TheTriple defaults to ELF, and COFF doesn't have an environment: 101de62046dSVlad Tsyrklevich // the best we can do here is indicate that it is mach-o. 102de62046dSVlad Tsyrklevich if (isMachO()) 103de62046dSVlad Tsyrklevich TheTriple.setObjectFormat(Triple::MachO); 104de62046dSVlad Tsyrklevich 105de62046dSVlad Tsyrklevich if (isCOFF()) { 106de62046dSVlad Tsyrklevich const auto COFFObj = dyn_cast<COFFObjectFile>(this); 107de62046dSVlad Tsyrklevich if (COFFObj->getArch() == Triple::thumb) 108de62046dSVlad Tsyrklevich TheTriple.setTriple("thumbv7-windows"); 109de62046dSVlad Tsyrklevich } 110de62046dSVlad Tsyrklevich 111de62046dSVlad Tsyrklevich return TheTriple; 112de62046dSVlad Tsyrklevich } 113de62046dSVlad Tsyrklevich 1143fcdf6aeSKevin Enderby Expected<std::unique_ptr<ObjectFile>> 115264b5d9eSZachary Turner ObjectFile::createObjectFile(MemoryBufferRef Object, file_magic Type) { 11648af1c2aSRafael Espindola StringRef Data = Object.getBuffer(); 117264b5d9eSZachary Turner if (Type == file_magic::unknown) 118264b5d9eSZachary Turner Type = identify_magic(Data); 11951cc3602SRafael Espindola 120cd81b909SRafael Espindola switch (Type) { 121264b5d9eSZachary Turner case file_magic::unknown: 122264b5d9eSZachary Turner case file_magic::bitcode: 123264b5d9eSZachary Turner case file_magic::coff_cl_gl_object: 124264b5d9eSZachary Turner case file_magic::archive: 125264b5d9eSZachary Turner case file_magic::macho_universal_binary: 126264b5d9eSZachary Turner case file_magic::windows_resource: 127aac28f31SZachary Turner case file_magic::pdb: 128581d79a4SPavel Labath case file_magic::minidump: 1293fcdf6aeSKevin Enderby return errorCodeToError(object_error::invalid_file_type); 130264b5d9eSZachary Turner case file_magic::elf: 131264b5d9eSZachary Turner case file_magic::elf_relocatable: 132264b5d9eSZachary Turner case file_magic::elf_executable: 133264b5d9eSZachary Turner case file_magic::elf_shared_object: 134264b5d9eSZachary Turner case file_magic::elf_core: 13512db383eSRafael Espindola return createELFObjectFile(Object); 136264b5d9eSZachary Turner case file_magic::macho_object: 137264b5d9eSZachary Turner case file_magic::macho_executable: 138264b5d9eSZachary Turner case file_magic::macho_fixed_virtual_memory_shared_lib: 139264b5d9eSZachary Turner case file_magic::macho_core: 140264b5d9eSZachary Turner case file_magic::macho_preload_executable: 141264b5d9eSZachary Turner case file_magic::macho_dynamically_linked_shared_lib: 142264b5d9eSZachary Turner case file_magic::macho_dynamic_linker: 143264b5d9eSZachary Turner case file_magic::macho_bundle: 144264b5d9eSZachary Turner case file_magic::macho_dynamically_linked_shared_lib_stub: 145264b5d9eSZachary Turner case file_magic::macho_dsym_companion: 146264b5d9eSZachary Turner case file_magic::macho_kext_bundle: 1473fcdf6aeSKevin Enderby return createMachOObjectFile(Object); 148264b5d9eSZachary Turner case file_magic::coff_object: 149264b5d9eSZachary Turner case file_magic::coff_import_library: 150264b5d9eSZachary Turner case file_magic::pecoff_executable: 15112db383eSRafael Espindola return createCOFFObjectFile(Object); 152ab2eb2bfSHubert Tong case file_magic::xcoff_object_32: 153*837ae69fSSean Fertile return createXCOFFObjectFile(Object, Binary::ID_XCOFF32); 154*837ae69fSSean Fertile case file_magic::xcoff_object_64: 155*837ae69fSSean Fertile return createXCOFFObjectFile(Object, Binary::ID_XCOFF64); 156264b5d9eSZachary Turner case file_magic::wasm_object: 1572c6f75ddSDerek Schuff return createWasmObjectFile(Object); 158fe5ee802SMichael J. Spencer } 159cd81b909SRafael Espindola llvm_unreachable("Unexpected Object File Type"); 160fe5ee802SMichael J. Spencer } 161fe5ee802SMichael J. Spencer 1623fcdf6aeSKevin Enderby Expected<OwningBinary<ObjectFile>> 163437b0d58SRafael Espindola ObjectFile::createObjectFile(StringRef ObjectPath) { 164adf21f2aSRafael Espindola ErrorOr<std::unique_ptr<MemoryBuffer>> FileOrErr = 165adf21f2aSRafael Espindola MemoryBuffer::getFile(ObjectPath); 166adf21f2aSRafael Espindola if (std::error_code EC = FileOrErr.getError()) 1673fcdf6aeSKevin Enderby return errorCodeToError(EC); 16848af1c2aSRafael Espindola std::unique_ptr<MemoryBuffer> Buffer = std::move(FileOrErr.get()); 16948af1c2aSRafael Espindola 1703fcdf6aeSKevin Enderby Expected<std::unique_ptr<ObjectFile>> ObjOrErr = 17148af1c2aSRafael Espindola createObjectFile(Buffer->getMemBufferRef()); 172b7c9deb5SJustin Bogner if (Error Err = ObjOrErr.takeError()) 173b7c9deb5SJustin Bogner return std::move(Err); 17448af1c2aSRafael Espindola std::unique_ptr<ObjectFile> Obj = std::move(ObjOrErr.get()); 17548af1c2aSRafael Espindola 17648af1c2aSRafael Espindola return OwningBinary<ObjectFile>(std::move(Obj), std::move(Buffer)); 177fe5ee802SMichael J. Spencer } 178