1d341c932SEugene Zelenko //===- ObjectFile.cpp - File format independent object file ---------------===// 2fe5ee802SMichael J. Spencer // 3fe5ee802SMichael J. Spencer // The LLVM Compiler Infrastructure 4fe5ee802SMichael J. Spencer // 5fe5ee802SMichael J. Spencer // This file is distributed under the University of Illinois Open Source 6fe5ee802SMichael J. Spencer // License. See LICENSE.TXT for details. 7fe5ee802SMichael J. Spencer // 8fe5ee802SMichael J. Spencer //===----------------------------------------------------------------------===// 9fe5ee802SMichael J. Spencer // 10fe5ee802SMichael J. Spencer // This file defines a file format independent ObjectFile class. 11fe5ee802SMichael J. Spencer // 12fe5ee802SMichael J. Spencer //===----------------------------------------------------------------------===// 13fe5ee802SMichael J. Spencer 146bda14b3SChandler Carruth #include "llvm/Object/ObjectFile.h" 15d341c932SEugene Zelenko #include "llvm/ADT/StringRef.h" 16*264b5d9eSZachary 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" 25cd81b909SRafael Espindola #include "llvm/Support/FileSystem.h" 26fe5ee802SMichael J. Spencer #include "llvm/Support/MemoryBuffer.h" 27b1f8a5a5SAaron Ballman #include "llvm/Support/raw_ostream.h" 28d341c932SEugene Zelenko #include <algorithm> 29d341c932SEugene Zelenko #include <cstdint> 30d341c932SEugene Zelenko #include <memory> 31a6e9c3e4SRafael Espindola #include <system_error> 32fe5ee802SMichael J. Spencer 33f177d16fSMichael J. Spencer using namespace llvm; 34f177d16fSMichael J. Spencer using namespace object; 35fe5ee802SMichael J. Spencer 36a379b181SDavid Blaikie void ObjectFile::anchor() {} 37a379b181SDavid Blaikie 3848af1c2aSRafael Espindola ObjectFile::ObjectFile(unsigned int Type, MemoryBufferRef Source) 3948af1c2aSRafael Espindola : SymbolicFile(Type, Source) {} 40f12b8282SRafael Espindola 41e9c58c74SRafael Espindola bool SectionRef::containsSymbol(SymbolRef S) const { 427bd8d994SKevin Enderby Expected<section_iterator> SymSec = S.getSection(); 437bd8d994SKevin Enderby if (!SymSec) { 447bd8d994SKevin Enderby // TODO: Actually report errors helpfully. 457bd8d994SKevin Enderby consumeError(SymSec.takeError()); 46e9c58c74SRafael Espindola return false; 477bd8d994SKevin Enderby } 488bab889bSRafael Espindola return *this == **SymSec; 49e9c58c74SRafael Espindola } 50e9c58c74SRafael Espindola 51be8b0ea8SRafael Espindola uint64_t ObjectFile::getSymbolValue(DataRefImpl Ref) const { 52be8b0ea8SRafael Espindola uint32_t Flags = getSymbolFlags(Ref); 53be8b0ea8SRafael Espindola if (Flags & SymbolRef::SF_Undefined) 54be8b0ea8SRafael Espindola return 0; 55be8b0ea8SRafael Espindola if (Flags & SymbolRef::SF_Common) 56be8b0ea8SRafael Espindola return getCommonSymbolSize(Ref); 57be8b0ea8SRafael Espindola return getSymbolValueImpl(Ref); 58be8b0ea8SRafael Espindola } 59be8b0ea8SRafael Espindola 60db4ed0bdSRafael Espindola std::error_code ObjectFile::printSymbolName(raw_ostream &OS, 61f12b8282SRafael Espindola DataRefImpl Symb) const { 6281e8b7d9SKevin Enderby Expected<StringRef> Name = getSymbolName(Symb); 6381e8b7d9SKevin Enderby if (!Name) 6481e8b7d9SKevin Enderby return errorToErrorCode(Name.takeError()); 655d0c2ffaSRafael Espindola OS << *Name; 667d099195SRui Ueyama return std::error_code(); 67f12b8282SRafael Espindola } 68fe5ee802SMichael J. Spencer 69a4d22472SRafael Espindola uint32_t ObjectFile::getSymbolAlignment(DataRefImpl DRI) const { return 0; } 70e4dd2e01SRafael Espindola 71f2fe0141SSteven Wu bool ObjectFile::isSectionBitcode(DataRefImpl Sec) const { 72f2fe0141SSteven Wu StringRef SectName; 73f2fe0141SSteven Wu if (!getSectionName(Sec, SectName)) 74f2fe0141SSteven Wu return SectName == ".llvmbc"; 75f2fe0141SSteven Wu return false; 76f2fe0141SSteven Wu } 77f2fe0141SSteven Wu 784f60a38fSRafael Espindola section_iterator ObjectFile::getRelocatedSection(DataRefImpl Sec) const { 794f60a38fSRafael Espindola return section_iterator(SectionRef(Sec, this)); 804f60a38fSRafael Espindola } 814f60a38fSRafael Espindola 823fcdf6aeSKevin Enderby Expected<std::unique_ptr<ObjectFile>> 83*264b5d9eSZachary Turner ObjectFile::createObjectFile(MemoryBufferRef Object, file_magic Type) { 8448af1c2aSRafael Espindola StringRef Data = Object.getBuffer(); 85*264b5d9eSZachary Turner if (Type == file_magic::unknown) 86*264b5d9eSZachary Turner Type = identify_magic(Data); 8751cc3602SRafael Espindola 88cd81b909SRafael Espindola switch (Type) { 89*264b5d9eSZachary Turner case file_magic::unknown: 90*264b5d9eSZachary Turner case file_magic::bitcode: 91*264b5d9eSZachary Turner case file_magic::coff_cl_gl_object: 92*264b5d9eSZachary Turner case file_magic::archive: 93*264b5d9eSZachary Turner case file_magic::macho_universal_binary: 94*264b5d9eSZachary Turner case file_magic::windows_resource: 953fcdf6aeSKevin Enderby return errorCodeToError(object_error::invalid_file_type); 96*264b5d9eSZachary Turner case file_magic::elf: 97*264b5d9eSZachary Turner case file_magic::elf_relocatable: 98*264b5d9eSZachary Turner case file_magic::elf_executable: 99*264b5d9eSZachary Turner case file_magic::elf_shared_object: 100*264b5d9eSZachary Turner case file_magic::elf_core: 1013fcdf6aeSKevin Enderby return errorOrToExpected(createELFObjectFile(Object)); 102*264b5d9eSZachary Turner case file_magic::macho_object: 103*264b5d9eSZachary Turner case file_magic::macho_executable: 104*264b5d9eSZachary Turner case file_magic::macho_fixed_virtual_memory_shared_lib: 105*264b5d9eSZachary Turner case file_magic::macho_core: 106*264b5d9eSZachary Turner case file_magic::macho_preload_executable: 107*264b5d9eSZachary Turner case file_magic::macho_dynamically_linked_shared_lib: 108*264b5d9eSZachary Turner case file_magic::macho_dynamic_linker: 109*264b5d9eSZachary Turner case file_magic::macho_bundle: 110*264b5d9eSZachary Turner case file_magic::macho_dynamically_linked_shared_lib_stub: 111*264b5d9eSZachary Turner case file_magic::macho_dsym_companion: 112*264b5d9eSZachary Turner case file_magic::macho_kext_bundle: 1133fcdf6aeSKevin Enderby return createMachOObjectFile(Object); 114*264b5d9eSZachary Turner case file_magic::coff_object: 115*264b5d9eSZachary Turner case file_magic::coff_import_library: 116*264b5d9eSZachary Turner case file_magic::pecoff_executable: 1173fcdf6aeSKevin Enderby return errorOrToExpected(createCOFFObjectFile(Object)); 118*264b5d9eSZachary Turner case file_magic::wasm_object: 1192c6f75ddSDerek Schuff return createWasmObjectFile(Object); 120fe5ee802SMichael J. Spencer } 121cd81b909SRafael Espindola llvm_unreachable("Unexpected Object File Type"); 122fe5ee802SMichael J. Spencer } 123fe5ee802SMichael J. Spencer 1243fcdf6aeSKevin Enderby Expected<OwningBinary<ObjectFile>> 125437b0d58SRafael Espindola ObjectFile::createObjectFile(StringRef ObjectPath) { 126adf21f2aSRafael Espindola ErrorOr<std::unique_ptr<MemoryBuffer>> FileOrErr = 127adf21f2aSRafael Espindola MemoryBuffer::getFile(ObjectPath); 128adf21f2aSRafael Espindola if (std::error_code EC = FileOrErr.getError()) 1293fcdf6aeSKevin Enderby return errorCodeToError(EC); 13048af1c2aSRafael Espindola std::unique_ptr<MemoryBuffer> Buffer = std::move(FileOrErr.get()); 13148af1c2aSRafael Espindola 1323fcdf6aeSKevin Enderby Expected<std::unique_ptr<ObjectFile>> ObjOrErr = 13348af1c2aSRafael Espindola createObjectFile(Buffer->getMemBufferRef()); 134b7c9deb5SJustin Bogner if (Error Err = ObjOrErr.takeError()) 135b7c9deb5SJustin Bogner return std::move(Err); 13648af1c2aSRafael Espindola std::unique_ptr<ObjectFile> Obj = std::move(ObjOrErr.get()); 13748af1c2aSRafael Espindola 13848af1c2aSRafael Espindola return OwningBinary<ObjectFile>(std::move(Obj), std::move(Buffer)); 139fe5ee802SMichael J. Spencer } 140