//===- lib/ReaderWriter/MachO/MachOLinkingContext.cpp ---------------------===// // // The LLVM Linker // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "lld/ReaderWriter/MachOLinkingContext.h" #include "lld/ReaderWriter/MachOFormat.hpp" #include "GOTPass.hpp" #include "StubsPass.hpp" #include "ReferenceKinds.h" #include "lld/Core/PassManager.h" #include "lld/ReaderWriter/Reader.h" #include "lld/ReaderWriter/Writer.h" #include "lld/Passes/LayoutPass.h" #include "llvm/ADT/StringExtras.h" #include "llvm/ADT/Triple.h" using lld::mach_o::KindHandler; namespace lld { bool MachOLinkingContext::parsePackedVersion(StringRef str, uint32_t &result) { result = 0; if (str.empty()) return false; SmallVector parts; llvm::SplitString(str, parts, "."); unsigned long long num; if (llvm::getAsUnsignedInteger(parts[0], 10, num)) return true; if (num > 65535) return true; result = num << 16; if (parts.size() > 1) { if (llvm::getAsUnsignedInteger(parts[1], 10, num)) return true; if (num > 255) return true; result |= (num << 8); } if (parts.size() > 2) { if (llvm::getAsUnsignedInteger(parts[2], 10, num)) return true; if (num > 255) return true; result |= num; } return false; } struct ArchInfo { StringRef archName; MachOLinkingContext::Arch arch; uint32_t cputype; uint32_t cpusubtype; }; static ArchInfo archInfos[] = { { "x86_64", MachOLinkingContext::arch_x86_64, mach_o::CPU_TYPE_X86_64, mach_o::CPU_SUBTYPE_X86_64_ALL }, { "i386", MachOLinkingContext::arch_x86, mach_o::CPU_TYPE_I386, mach_o::CPU_SUBTYPE_X86_ALL }, { "armv6", MachOLinkingContext::arch_armv6, mach_o::CPU_TYPE_ARM, mach_o::CPU_SUBTYPE_ARM_V6 }, { "armv7", MachOLinkingContext::arch_armv7, mach_o::CPU_TYPE_ARM, mach_o::CPU_SUBTYPE_ARM_V7 }, { "armv7s", MachOLinkingContext::arch_armv7s, mach_o::CPU_TYPE_ARM, mach_o::CPU_SUBTYPE_ARM_V7S }, { StringRef(), MachOLinkingContext::arch_unknown, 0, 0 } }; MachOLinkingContext::Arch MachOLinkingContext::archFromCpuType(uint32_t cputype, uint32_t cpusubtype) { for (ArchInfo *info = archInfos; !info->archName.empty(); ++info) { if ((info->cputype == cputype) && (info->cpusubtype == cpusubtype)) { return info->arch; } } return arch_unknown; } MachOLinkingContext::Arch MachOLinkingContext::archFromName(StringRef archName) { for (ArchInfo *info = archInfos; !info->archName.empty(); ++info) { if (info->archName.equals(archName)) { return info->arch; } } return arch_unknown; } uint32_t MachOLinkingContext::cpuTypeFromArch(Arch arch) { assert(arch != arch_unknown); for (ArchInfo *info = archInfos; !info->archName.empty(); ++info) { if (info->arch == arch) { return info->cputype; } } llvm_unreachable("Unknown arch type"); } uint32_t MachOLinkingContext::cpuSubtypeFromArch(Arch arch) { assert(arch != arch_unknown); for (ArchInfo *info = archInfos; !info->archName.empty(); ++info) { if (info->arch == arch) { return info->cpusubtype; } } llvm_unreachable("Unknown arch type"); } MachOLinkingContext::MachOLinkingContext() : _outputFileType(mach_o::MH_EXECUTE), _outputFileTypeStatic(false), _doNothing(false), _arch(arch_unknown), _os(OS::macOSX), _osMinVersion(0), _pageZeroSize(0x1000), _compatibilityVersion(0), _currentVersion(0), _deadStrippableDylib(false), _kindHandler(nullptr) {} MachOLinkingContext::~MachOLinkingContext() {} uint32_t MachOLinkingContext::getCPUType() const { return cpuTypeFromArch(_arch); } uint32_t MachOLinkingContext::getCPUSubType() const { return cpuSubtypeFromArch(_arch); } bool MachOLinkingContext::outputTypeHasEntry() const { switch (_outputFileType) { case mach_o::MH_EXECUTE: case mach_o::MH_DYLINKER: case mach_o::MH_PRELOAD: return true; default: return false; } } bool MachOLinkingContext::minOS(StringRef mac, StringRef iOS) const { switch (_os) { case OS::macOSX: { uint32_t parsedVersion; if (parsePackedVersion(mac, parsedVersion)) return false; return _osMinVersion >= parsedVersion; } case OS::iOS: case OS::iOS_simulator: { uint32_t parsedVersion; if (parsePackedVersion(iOS, parsedVersion)) return false; return _osMinVersion >= parsedVersion; } } llvm_unreachable("target not configured for iOS or MacOSX"); } bool MachOLinkingContext::addEntryPointLoadCommand() const { if ((_outputFileType == mach_o::MH_EXECUTE) && !_outputFileTypeStatic) { return minOS("10.8", "6.0"); } return false; } bool MachOLinkingContext::addUnixThreadLoadCommand() const { switch (_outputFileType) { case mach_o::MH_EXECUTE: if (_outputFileTypeStatic) return true; else return !minOS("10.8", "6.0"); break; case mach_o::MH_DYLINKER: case mach_o::MH_PRELOAD: return true; default: return false; } } bool MachOLinkingContext::validateImpl(raw_ostream &diagnostics) { if ((_outputFileType == mach_o::MH_EXECUTE) && _entrySymbolName.empty()) { if (_outputFileTypeStatic) { _entrySymbolName = "start"; } else { // If targeting newer OS, use _main if (addEntryPointLoadCommand()) _entrySymbolName = "_main"; // If targeting older OS, use start (in crt1.o) if (addUnixThreadLoadCommand()) _entrySymbolName = "start"; } } if (_currentVersion && _outputFileType != mach_o::MH_DYLIB) { diagnostics << "error: -current_version can only be used with dylibs\n"; return true; } if (_compatibilityVersion && _outputFileType != mach_o::MH_DYLIB) { diagnostics << "error: -compatibility_version can only be used with dylibs\n"; return true; } if (_deadStrippableDylib && _outputFileType != mach_o::MH_DYLIB) { diagnostics << "error: -mark_dead_strippable_dylib can only be used with dylibs.\n"; return true; } if (!_bundleLoader.empty() && outputFileType() != mach_o::MH_BUNDLE) { diagnostics << "error: -bundle_loader can only be used with Mach-O bundles\n"; return true; } return false; } bool MachOLinkingContext::setOS(OS os, StringRef minOSVersion) { _os = os; return parsePackedVersion(minOSVersion, _osMinVersion); } void MachOLinkingContext::addPasses(PassManager &pm) const { pm.add(std::unique_ptr(new mach_o::GOTPass)); pm.add(std::unique_ptr(new mach_o::StubsPass(*this))); pm.add(std::unique_ptr(new LayoutPass())); } error_code MachOLinkingContext::parseFile( LinkerInput &input, std::vector> &result) const { // if (!_machoReader) // _machoReader = createReaderMachO(*this); // error_code ec = _machoReader->parseFile(input,result); // if (ec) { return _yamlReader->parseFile(input, result); // } return error_code::success(); } Writer &MachOLinkingContext::writer() const { if (!_writer) { _writer = createWriterMachO(*this); } return *_writer; } KindHandler &MachOLinkingContext::kindHandler() const { if (!_kindHandler) _kindHandler = KindHandler::create(_arch); return *_kindHandler; } ErrorOr MachOLinkingContext::relocKindFromString(StringRef str) const { return kindHandler().stringToKind(str); } ErrorOr MachOLinkingContext::stringFromRelocKind(Reference::Kind kind) const { return std::string(kindHandler().kindToString(kind)); } } // end namespace lld