1fe5ee802SMichael J. Spencer //===- ObjectFile.cpp - File format independent object file -----*- C++ -*-===// 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 14*2c6f75ddSDerek Schuff #include "llvm/Object/ObjectFile.h" 15437b0d58SRafael Espindola #include "llvm/Object/COFF.h" 16437b0d58SRafael Espindola #include "llvm/Object/MachO.h" 17*2c6f75ddSDerek Schuff #include "llvm/Object/Wasm.h" 18fe5ee802SMichael J. Spencer #include "llvm/Support/ErrorHandling.h" 19cd81b909SRafael Espindola #include "llvm/Support/FileSystem.h" 20fe5ee802SMichael J. Spencer #include "llvm/Support/MemoryBuffer.h" 21b1f8a5a5SAaron Ballman #include "llvm/Support/raw_ostream.h" 22a6e9c3e4SRafael Espindola #include <system_error> 23fe5ee802SMichael J. Spencer 24f177d16fSMichael J. Spencer using namespace llvm; 25f177d16fSMichael J. Spencer using namespace object; 26fe5ee802SMichael J. Spencer 27a379b181SDavid Blaikie void ObjectFile::anchor() { } 28a379b181SDavid Blaikie 2948af1c2aSRafael Espindola ObjectFile::ObjectFile(unsigned int Type, MemoryBufferRef Source) 3048af1c2aSRafael Espindola : SymbolicFile(Type, Source) {} 31f12b8282SRafael Espindola 32e9c58c74SRafael Espindola bool SectionRef::containsSymbol(SymbolRef S) const { 337bd8d994SKevin Enderby Expected<section_iterator> SymSec = S.getSection(); 347bd8d994SKevin Enderby if (!SymSec) { 357bd8d994SKevin Enderby // TODO: Actually report errors helpfully. 367bd8d994SKevin Enderby consumeError(SymSec.takeError()); 37e9c58c74SRafael Espindola return false; 387bd8d994SKevin Enderby } 398bab889bSRafael Espindola return *this == **SymSec; 40e9c58c74SRafael Espindola } 41e9c58c74SRafael Espindola 42be8b0ea8SRafael Espindola uint64_t ObjectFile::getSymbolValue(DataRefImpl Ref) const { 43be8b0ea8SRafael Espindola uint32_t Flags = getSymbolFlags(Ref); 44be8b0ea8SRafael Espindola if (Flags & SymbolRef::SF_Undefined) 45be8b0ea8SRafael Espindola return 0; 46be8b0ea8SRafael Espindola if (Flags & SymbolRef::SF_Common) 47be8b0ea8SRafael Espindola return getCommonSymbolSize(Ref); 48be8b0ea8SRafael Espindola return getSymbolValueImpl(Ref); 49be8b0ea8SRafael Espindola } 50be8b0ea8SRafael Espindola 51db4ed0bdSRafael Espindola std::error_code ObjectFile::printSymbolName(raw_ostream &OS, 52f12b8282SRafael Espindola DataRefImpl Symb) const { 5381e8b7d9SKevin Enderby Expected<StringRef> Name = getSymbolName(Symb); 5481e8b7d9SKevin Enderby if (!Name) 5581e8b7d9SKevin Enderby return errorToErrorCode(Name.takeError()); 565d0c2ffaSRafael Espindola OS << *Name; 577d099195SRui Ueyama return std::error_code(); 58f12b8282SRafael Espindola } 59fe5ee802SMichael J. Spencer 60a4d22472SRafael Espindola uint32_t ObjectFile::getSymbolAlignment(DataRefImpl DRI) const { return 0; } 61e4dd2e01SRafael Espindola 62f2fe0141SSteven Wu bool ObjectFile::isSectionBitcode(DataRefImpl Sec) const { 63f2fe0141SSteven Wu StringRef SectName; 64f2fe0141SSteven Wu if (!getSectionName(Sec, SectName)) 65f2fe0141SSteven Wu return SectName == ".llvmbc"; 66f2fe0141SSteven Wu return false; 67f2fe0141SSteven Wu } 68f2fe0141SSteven Wu 694f60a38fSRafael Espindola section_iterator ObjectFile::getRelocatedSection(DataRefImpl Sec) const { 704f60a38fSRafael Espindola return section_iterator(SectionRef(Sec, this)); 714f60a38fSRafael Espindola } 724f60a38fSRafael Espindola 733fcdf6aeSKevin Enderby Expected<std::unique_ptr<ObjectFile>> 7448af1c2aSRafael Espindola ObjectFile::createObjectFile(MemoryBufferRef Object, sys::fs::file_magic Type) { 7548af1c2aSRafael Espindola StringRef Data = Object.getBuffer(); 76ec46f318SRafael Espindola if (Type == sys::fs::file_magic::unknown) 7748af1c2aSRafael Espindola Type = sys::fs::identify_magic(Data); 7851cc3602SRafael Espindola 79cd81b909SRafael Espindola switch (Type) { 80cd81b909SRafael Espindola case sys::fs::file_magic::unknown: 81447d2d12SRafael Espindola case sys::fs::file_magic::bitcode: 822d02166bSRui Ueyama case sys::fs::file_magic::coff_cl_gl_object: 83447d2d12SRafael Espindola case sys::fs::file_magic::archive: 84e6388e62SAlexey Samsonov case sys::fs::file_magic::macho_universal_binary: 85fc149a69SRui Ueyama case sys::fs::file_magic::windows_resource: 863fcdf6aeSKevin Enderby return errorCodeToError(object_error::invalid_file_type); 87bbd875b6SMichael J. Spencer case sys::fs::file_magic::elf: 88cd81b909SRafael Espindola case sys::fs::file_magic::elf_relocatable: 89cd81b909SRafael Espindola case sys::fs::file_magic::elf_executable: 90cd81b909SRafael Espindola case sys::fs::file_magic::elf_shared_object: 91cd81b909SRafael Espindola case sys::fs::file_magic::elf_core: 923fcdf6aeSKevin Enderby return errorOrToExpected(createELFObjectFile(Object)); 93cd81b909SRafael Espindola case sys::fs::file_magic::macho_object: 94cd81b909SRafael Espindola case sys::fs::file_magic::macho_executable: 95cd81b909SRafael Espindola case sys::fs::file_magic::macho_fixed_virtual_memory_shared_lib: 96cd81b909SRafael Espindola case sys::fs::file_magic::macho_core: 97cd81b909SRafael Espindola case sys::fs::file_magic::macho_preload_executable: 98cd81b909SRafael Espindola case sys::fs::file_magic::macho_dynamically_linked_shared_lib: 99cd81b909SRafael Espindola case sys::fs::file_magic::macho_dynamic_linker: 100cd81b909SRafael Espindola case sys::fs::file_magic::macho_bundle: 101cd81b909SRafael Espindola case sys::fs::file_magic::macho_dynamically_linked_shared_lib_stub: 102cd81b909SRafael Espindola case sys::fs::file_magic::macho_dsym_companion: 103a7ad4b3fSJustin Bogner case sys::fs::file_magic::macho_kext_bundle: 1043fcdf6aeSKevin Enderby return createMachOObjectFile(Object); 105cd81b909SRafael Espindola case sys::fs::file_magic::coff_object: 106e448f9e4SRui Ueyama case sys::fs::file_magic::coff_import_library: 107cd81b909SRafael Espindola case sys::fs::file_magic::pecoff_executable: 1083fcdf6aeSKevin Enderby return errorOrToExpected(createCOFFObjectFile(Object)); 109*2c6f75ddSDerek Schuff case sys::fs::file_magic::wasm_object: 110*2c6f75ddSDerek Schuff return createWasmObjectFile(Object); 111fe5ee802SMichael J. Spencer } 112cd81b909SRafael Espindola llvm_unreachable("Unexpected Object File Type"); 113fe5ee802SMichael J. Spencer } 114fe5ee802SMichael J. Spencer 1153fcdf6aeSKevin Enderby Expected<OwningBinary<ObjectFile>> 116437b0d58SRafael Espindola ObjectFile::createObjectFile(StringRef ObjectPath) { 117adf21f2aSRafael Espindola ErrorOr<std::unique_ptr<MemoryBuffer>> FileOrErr = 118adf21f2aSRafael Espindola MemoryBuffer::getFile(ObjectPath); 119adf21f2aSRafael Espindola if (std::error_code EC = FileOrErr.getError()) 1203fcdf6aeSKevin Enderby return errorCodeToError(EC); 12148af1c2aSRafael Espindola std::unique_ptr<MemoryBuffer> Buffer = std::move(FileOrErr.get()); 12248af1c2aSRafael Espindola 1233fcdf6aeSKevin Enderby Expected<std::unique_ptr<ObjectFile>> ObjOrErr = 12448af1c2aSRafael Espindola createObjectFile(Buffer->getMemBufferRef()); 125b7c9deb5SJustin Bogner if (Error Err = ObjOrErr.takeError()) 126b7c9deb5SJustin Bogner return std::move(Err); 12748af1c2aSRafael Espindola std::unique_ptr<ObjectFile> Obj = std::move(ObjOrErr.get()); 12848af1c2aSRafael Espindola 12948af1c2aSRafael Espindola return OwningBinary<ObjectFile>(std::move(Obj), std::move(Buffer)); 130fe5ee802SMichael J. Spencer } 131