1
1
mirror of https://github.com/rui314/mold.git synced 2024-11-12 23:48:51 +03:00
mold/macho/mold.h
2022-06-28 19:53:28 +08:00

1017 lines
25 KiB
C++

#pragma once
#include "macho.h"
#include "lto.h"
#include "../mold.h"
#include <map>
#include <memory>
#include <optional>
#include <set>
#include <span>
#include <tbb/concurrent_hash_map.h>
#include <tbb/spin_mutex.h>
#include <tbb/task_group.h>
#include <unordered_map>
#include <unordered_set>
#include <variant>
#if MOLD_DEBUG_X86_64_ONLY
# define INSTANTIATE_ALL INSTANTIATE(X86_64)
#elif MOLD_DEBUG_ARM64_ONLY
# define INSTANTIATE_ALL INSTANTIATE(ARM64)
#else
# define INSTANTIATE_ALL \
INSTANTIATE(X86_64); \
INSTANTIATE(ARM64);
#endif
namespace mold::macho {
static constexpr i64 COMMON_PAGE_SIZE = 0x4000;
static constexpr i64 SHA256_SIZE = 32;
template <typename E> class Chunk;
template <typename E> class InputSection;
template <typename E> class OutputSection;
template <typename E> class Subsection;
template <typename E> struct Context;
template <typename E> struct Symbol;
//
// input-files.cc
//
template <typename E>
struct Relocation {
u32 offset = 0;
u8 type = -1;
u8 p2size = 0;
bool is_pcrel : 1 = false;
bool needs_dynrel : 1 = false;
i64 addend = 0;
Symbol<E> *sym = nullptr;
Subsection<E> *subsec = nullptr;
// For range extension thunks
i32 thunk_idx = -1;
i32 thunk_sym_idx = -1;
};
template <typename E>
std::ostream &operator<<(std::ostream &out, const Relocation<E> &rel) {
out << rel_to_string<E>(rel.type);
return out;
}
template <typename E>
struct UnwindRecord {
UnwindRecord(u32 len, u32 enc) : code_len(len), encoding(enc) {}
u64 get_func_addr(Context<E> &ctx) const {
return subsec->get_addr(ctx) + offset;
}
Subsection<E> *subsec = nullptr;
Symbol<E> *personality = nullptr;
Subsection<E> *lsda = nullptr;
u32 offset = 0;
u32 code_len = 0;
u32 encoding = 0;
u32 lsda_offset = 0;
};
template <typename E>
class InputFile {
public:
virtual ~InputFile() = default;
virtual void resolve_symbols(Context<E> &ctx) = 0;
void clear_symbols();
MappedFile<Context<E>> *mf = nullptr;
std::string_view filename;
std::vector<Symbol<E> *> syms;
i64 priority = 0;
std::atomic_bool is_alive = false;
bool is_dylib = false;
bool is_hidden = false;
bool is_weak = false;
std::string archive_name;
protected:
InputFile(MappedFile<Context<E>> *mf) : mf(mf), filename(mf->name) {}
InputFile() : filename("<internal>") {}
};
template <typename E>
class ObjectFile : public InputFile<E> {
public:
ObjectFile() = default;
ObjectFile(MappedFile<Context<E>> *mf) : InputFile<E>(mf) {}
static ObjectFile *create(Context<E> &ctx, MappedFile<Context<E>> *mf,
std::string archive_name);
void parse(Context<E> &ctx);
Subsection<E> *find_subsection(Context<E> &ctx, u32 addr);
Symbol<E> *find_symbol(Context<E> &ctx, u32 addr);
std::vector<std::string> get_linker_options(Context<E> &ctx);
void parse_compact_unwind(Context<E> &ctx, MachSection &hdr);
void resolve_symbols(Context<E> &ctx) override;
bool is_objc_object(Context<E> &ctx);
void mark_live_objects(Context<E> &ctx,
std::function<void(ObjectFile<E> *)> feeder);
void convert_common_symbols(Context<E> &ctx);
void check_duplicate_symbols(Context<E> &ctx);
Relocation<E> read_reloc(Context<E> &ctx, const MachSection &hdr, MachRel r);
std::vector<std::unique_ptr<InputSection<E>>> sections;
std::vector<Subsection<E> *> subsections;
std::vector<Subsection<E> *> sym_to_subsec;
std::span<MachSym> mach_syms;
std::vector<Symbol<E>> local_syms;
std::vector<UnwindRecord<E>> unwind_records;
std::span<DataInCodeEntry> data_in_code_entries;
LTOModule *lto_module = nullptr;
// For the internal file and LTO object files
std::vector<MachSym> mach_syms2;
private:
void parse_sections(Context<E> &ctx);
void parse_symbols(Context<E> &ctx);
void split_subsections_via_symbols(Context<E> &ctx);
void init_subsections(Context<E> &ctx);
void fix_subsec_members(Context<E> &ctx);
void parse_data_in_code(Context<E> &ctx);
LoadCommand *find_load_command(Context<E> &ctx, u32 type);
InputSection<E> *get_common_sec(Context<E> &ctx);
void parse_lto_symbols(Context<E> &ctx);
MachSection *unwind_sec = nullptr;
std::unique_ptr<MachSection> common_hdr;
InputSection<E> *common_sec = nullptr;
std::vector<Subsection<E> *> subsec_pool;
};
template <typename E>
class DylibFile : public InputFile<E> {
public:
static DylibFile *create(Context<E> &ctx, MappedFile<Context<E>> *mf);
void parse(Context<E> &ctx);
void resolve_symbols(Context<E> &ctx) override;
std::string_view install_name;
i64 dylib_idx = 0;
bool is_reexported = false;
std::vector<std::string_view> reexported_libs;
std::set<std::string_view> exports;
std::set<std::string_view> weak_exports;
private:
DylibFile(Context<E> &ctx, MappedFile<Context<E>> *mf);
void parse_tapi(Context<E> &ctx);
void parse_dylib(Context<E> &ctx);
void read_trie(Context<E> &ctx, u8 *start, i64 offset = 0,
const std::string &prefix = "");
std::vector<bool> is_weak_symbol;
};
template <typename E>
std::ostream &operator<<(std::ostream &out, const InputFile<E> &file);
//
// input-sections.cc
//
template <typename E>
class InputSection {
public:
InputSection(Context<E> &ctx, ObjectFile<E> &file, const MachSection &hdr);
void parse_relocations(Context<E> &ctx);
ObjectFile<E> &file;
const MachSection &hdr;
OutputSection<E> &osec;
std::string_view contents;
std::vector<Symbol<E> *> syms;
std::vector<Relocation<E>> rels;
};
template <typename E>
std::ostream &operator<<(std::ostream &out, const InputSection<E> &sec);
template <typename E>
class Subsection {
public:
inline u64 get_addr(Context<E> &ctx) const;
std::string_view get_contents() {
assert(isec.hdr.type != S_ZEROFILL);
return isec.contents.substr(input_offset, input_size);
}
std::span<UnwindRecord<E>> get_unwind_records() {
return std::span<UnwindRecord<E>>(isec.file.unwind_records)
.subspan(unwind_offset, nunwind);
}
std::span<Relocation<E>> get_rels() const {
return std::span<Relocation<E>>(isec.rels).subspan(rel_offset, nrels);
}
void scan_relocations(Context<E> &ctx);
void apply_reloc(Context<E> &ctx, u8 *buf);
InputSection<E> &isec;
u32 input_offset = 0;
u32 input_size = 0;
u32 input_addr = 0;
u32 output_offset = -1;
u32 rel_offset = 0;
u32 nrels = 0;
u32 unwind_offset = 0;
u32 nunwind = 0;
Subsection<E> *replacer; // Used if is_coalesced is true
std::atomic_uint8_t p2align = 0;
std::atomic_bool is_alive = true;
bool is_coalesced : 1 = false;
bool added_to_osec : 1 = false;
};
template <typename E>
std::vector<Relocation<E>>
read_relocations(Context<E> &ctx, ObjectFile<E> &file, const MachSection &hdr);
//
// Symbol
//
enum {
NEEDS_GOT = 1 << 0,
NEEDS_STUB = 1 << 1,
NEEDS_THREAD_PTR = 1 << 2,
NEEDS_RANGE_EXTN_THUNK = 1 << 3,
};
enum {
SCOPE_LOCAL, // input file scope
SCOPE_PRIVATE_EXTERN, // output file scope (non-exported symbol)
SCOPE_EXTERN, // global scope (exported symbol)
};
template <typename E>
struct Symbol {
Symbol() = default;
Symbol(std::string_view name) : name(name) {}
Symbol(const Symbol<E> &other) : name(other.name) {}
std::string_view name;
InputFile<E> *file = nullptr;
Subsection<E> *subsec = nullptr;
u64 value = 0;
i32 stub_idx = -1;
i32 got_idx = -1;
i32 tlv_idx = -1;
tbb::spin_mutex mu;
std::atomic_uint8_t flags = 0;
u8 scope : 2 = SCOPE_LOCAL;
bool is_common : 1 = false;
bool is_weak : 1 = false;
bool is_imported : 1 = false;
bool referenced_dynamically : 1 = false;
// For range extension thunks
i32 thunk_idx = -1;
i32 thunk_sym_idx = -1;
inline u64 get_addr(Context<E> &ctx) const;
inline u64 get_got_addr(Context<E> &ctx) const;
inline u64 get_tlv_addr(Context<E> &ctx) const;
};
template <typename E>
std::ostream &operator<<(std::ostream &out, const Symbol<E> &sym);
// This operator defines a total order over symbols. This is used to
// make the output deterministic.
template <typename E>
inline bool operator<(const Symbol<E> &a, const Symbol<E> &b) {
return std::tuple{a.file->priority, a.value} <
std::tuple{b.file->priority, b.value};
}
//
// output-chunks.cc
//
template <typename E>
class OutputSegment {
public:
static OutputSegment<E> *
get_instance(Context<E> &ctx, std::string_view name);
void set_offset(Context<E> &ctx, i64 fileoff, u64 vmaddr);
SegmentCommand cmd = {};
i32 seg_idx = -1;
std::vector<Chunk<E> *> chunks;
private:
void set_offset_regular(Context<E> &ctx, i64 fileoff, u64 vmaddr);
void set_offset_linkedit(Context<E> &ctx, i64 fileoff, u64 vmaddr);
OutputSegment(std::string_view name);
};
template <typename E>
class Chunk {
public:
Chunk(Context<E> &ctx, std::string_view segname, std::string_view sectname) {
ctx.chunks.push_back(this);
hdr.set_segname(segname);
hdr.set_sectname(sectname);
}
virtual ~Chunk() = default;
virtual void compute_size(Context<E> &ctx) {}
virtual void copy_buf(Context<E> &ctx) {}
MachSection hdr = {};
u32 sect_idx = 0;
bool is_hidden = false;
bool is_output_section = false;
};
template <typename E>
std::ostream &operator<<(std::ostream &out, const Chunk<E> &chunk);
template <typename E>
class OutputMachHeader : public Chunk<E> {
public:
OutputMachHeader(Context<E> &ctx)
: Chunk<E>(ctx, "__TEXT", "__mach_header") {
this->is_hidden = true;
}
void compute_size(Context<E> &ctx) override;
void copy_buf(Context<E> &ctx) override;
};
template <typename E>
class RangeExtensionThunk {};
template <>
class RangeExtensionThunk<ARM64> {
public:
RangeExtensionThunk(OutputSection<ARM64> &osec)
: output_section(osec) {}
i64 size() const { return symbols.size() * ENTRY_SIZE; }
u64 get_addr(i64 idx) const;
void copy_buf(Context<ARM64> &ctx);
static constexpr i64 ENTRY_SIZE = 12;
OutputSection<ARM64> &output_section;
i32 thunk_idx = -1;
i64 offset = -1;
std::mutex mu;
std::vector<Symbol<ARM64> *> symbols;
};
template <typename E>
class OutputSection : public Chunk<E> {
public:
static OutputSection<E> *
get_instance(Context<E> &ctx, std::string_view segname,
std::string_view sectname);
OutputSection(Context<E> &ctx, std::string_view segname,
std::string_view sectname)
: Chunk<E>(ctx, segname, sectname) {
this->is_output_section = true;
}
void compute_size(Context<E> &ctx) override;
void copy_buf(Context<E> &ctx) override;
void add_subsec(Subsection<E> *subsec) {
members.push_back(subsec);
this->hdr.p2align = std::max<u32>(this->hdr.p2align, subsec->p2align);
this->hdr.attr |= subsec->isec.hdr.attr;
this->hdr.type = subsec->isec.hdr.type;
assert(!subsec->added_to_osec);
subsec->added_to_osec = true;
}
std::vector<Subsection<E> *> members;
std::vector<std::unique_ptr<RangeExtensionThunk<E>>> thunks;
};
class RebaseEncoder {
public:
RebaseEncoder();
void add(i64 seg_idx, i64 offset);
void flush();
void finish();
std::vector<u8> buf;
private:
i64 cur_seg = -1;
i64 cur_off = 0;
i64 times = 0;
};
template <typename E>
class RebaseSection : public Chunk<E> {
public:
RebaseSection(Context<E> &ctx)
: Chunk<E>(ctx, "__LINKEDIT", "__rebase") {
this->is_hidden = true;
}
void compute_size(Context<E> &ctx) override;
void copy_buf(Context<E> &ctx) override;
std::vector<u8> contents;
};
class BindEncoder {
public:
BindEncoder();
template <typename E> void add(Symbol<E> &sym, i64 seg_idx, i64 offset);
void finish();
std::vector<u8> buf;
private:
std::string_view last_name;
i64 last_flags = -1;
i64 last_dylib = -1;
i64 last_seg = -1;
i64 last_off = -1;
};
template <typename E>
class BindSection : public Chunk<E> {
public:
BindSection(Context<E> &ctx)
: Chunk<E>(ctx, "__LINKEDIT", "__binding") {
this->is_hidden = true;
}
void compute_size(Context<E> &ctx) override;
void copy_buf(Context<E> &ctx) override;
std::vector<u8> contents;
};
template <typename E>
class LazyBindSection : public Chunk<E> {
public:
LazyBindSection(Context<E> &ctx)
: Chunk<E>(ctx, "__LINKEDIT", "__lazy_binding") {
this->is_hidden = true;
this->hdr.p2align = 3;
}
void add(Context<E> &ctx, Symbol<E> &sym);
void compute_size(Context<E> &ctx) override;
void copy_buf(Context<E> &ctx) override;
std::vector<u8> contents;
};
class ExportEncoder {
public:
i64 finish();
struct Entry {
std::string_view name;
u32 flags;
u64 addr;
};
struct TrieNode {
std::string_view prefix;
std::vector<std::unique_ptr<TrieNode>> children;
u64 addr = 0;
u32 flags = 0;
u32 offset = -1;
bool is_leaf = false;
};
void construct_trie(TrieNode &node, std::span<Entry> entries, i64 len,
tbb::task_group *tg, i64 grain_size, bool divide);
static i64 set_offset(TrieNode &node, i64 offset);
void write_trie(u8 *buf, TrieNode &node);
TrieNode root;
std::vector<Entry> entries;
};
template <typename E>
class ExportSection : public Chunk<E> {
public:
ExportSection(Context<E> &ctx)
: Chunk<E>(ctx, "__LINKEDIT", "__export") {
this->is_hidden = true;
}
void compute_size(Context<E> &ctx) override;
void copy_buf(Context<E> &ctx) override;
private:
ExportEncoder enc;
};
template <typename E>
class FunctionStartsSection : public Chunk<E> {
public:
FunctionStartsSection(Context<E> &ctx)
: Chunk<E>(ctx, "__LINKEDIT", "__func_starts") {
this->is_hidden = true;
}
void compute_size(Context<E> &ctx) override;
void copy_buf(Context<E> &ctx) override;
std::vector<u8> contents;
};
template <typename E>
class SymtabSection : public Chunk<E> {
public:
SymtabSection(Context<E> &ctx)
: Chunk<E>(ctx, "__LINKEDIT", "__symbol_table") {
this->is_hidden = true;
this->hdr.p2align = 3;
}
void compute_size(Context<E> &ctx) override;
void copy_buf(Context<E> &ctx) override;
std::vector<i64> symtab_offsets;
std::vector<i64> strtab_offsets;
i64 num_locals = 0;
i64 num_globals = 0;
i64 num_undefs = 0;
};
template <typename E>
class StrtabSection : public Chunk<E> {
public:
StrtabSection(Context<E> &ctx)
: Chunk<E>(ctx, "__LINKEDIT", "__string_table") {
this->is_hidden = true;
this->hdr.p2align = 3;
this->hdr.size = 1;
}
};
template <typename E>
class CodeSignatureSection : public Chunk<E> {
public:
CodeSignatureSection(Context<E> &ctx)
: Chunk<E>(ctx, "__LINKEDIT", "__code_signature") {
this->is_hidden = true;
this->hdr.p2align = std::countr_zero(16U);
}
void compute_size(Context<E> &ctx) override;
void write_signature(Context<E> &ctx);
static constexpr i64 BLOCK_SIZE = 4096;
};
template <typename E>
class DataInCodeSection : public Chunk<E> {
public:
DataInCodeSection(Context<E> &ctx)
: Chunk<E>(ctx, "__LINKEDIT", "__data_in_code") {
this->is_hidden = true;
this->hdr.p2align = std::countr_zero(alignof(DataInCodeEntry));
}
void compute_size(Context<E> &ctx) override;
void copy_buf(Context<E> &ctx) override;
std::vector<DataInCodeEntry> contents;
};
template <typename E>
class StubsSection : public Chunk<E> {
public:
StubsSection(Context<E> &ctx) : Chunk<E>(ctx, "__TEXT", "__stubs") {
this->hdr.p2align = std::countr_zero(2U);
this->hdr.type = S_SYMBOL_STUBS;
this->hdr.attr = S_ATTR_SOME_INSTRUCTIONS | S_ATTR_PURE_INSTRUCTIONS;
this->hdr.reserved1 = 0;
this->hdr.reserved2 = E::stub_size;
}
void add(Context<E> &ctx, Symbol<E> *sym);
void copy_buf(Context<E> &ctx) override;
std::vector<Symbol<E> *> syms;
std::vector<u32> bind_offsets;
};
template <typename E>
class StubHelperSection : public Chunk<E> {
public:
StubHelperSection(Context<E> &ctx)
: Chunk<E>(ctx, "__TEXT", "__stub_helper") {
this->hdr.p2align = std::countr_zero(4U);
this->hdr.attr = S_ATTR_SOME_INSTRUCTIONS | S_ATTR_PURE_INSTRUCTIONS;
}
void copy_buf(Context<E> &ctx) override;
};
template <typename E>
class UnwindInfoSection : public Chunk<E> {
public:
UnwindInfoSection(Context<E> &ctx)
: Chunk<E>(ctx, "__TEXT", "__unwind_info") {
this->hdr.p2align = std::countr_zero(4U);
}
void compute_size(Context<E> &ctx) override;
void copy_buf(Context<E> &ctx) override;
std::vector<u8> contents;
};
template <typename E>
class GotSection : public Chunk<E> {
public:
GotSection(Context<E> &ctx) : Chunk<E>(ctx, "__DATA_CONST", "__got") {
this->hdr.p2align = 3;
this->hdr.type = S_NON_LAZY_SYMBOL_POINTERS;
}
void add(Context<E> &ctx, Symbol<E> *sym);
void copy_buf(Context<E> &ctx) override;
std::vector<Symbol<E> *> syms;
std::vector<Subsection<E> *> subsections;
};
template <typename E>
class LazySymbolPtrSection : public Chunk<E> {
public:
LazySymbolPtrSection(Context<E> &ctx)
: Chunk<E>(ctx, "__DATA", "__la_symbol_ptr") {
this->hdr.p2align = 3;
this->hdr.type = S_LAZY_SYMBOL_POINTERS;
}
void copy_buf(Context<E> &ctx) override;
};
template <typename E>
class ThreadPtrsSection : public Chunk<E> {
public:
ThreadPtrsSection(Context<E> &ctx)
: Chunk<E>(ctx, "__DATA", "__thread_ptrs") {
this->hdr.p2align = 3;
this->hdr.type = S_THREAD_LOCAL_VARIABLE_POINTERS;
}
void add(Context<E> &ctx, Symbol<E> *sym);
void copy_buf(Context<E> &ctx) override;
std::vector<Symbol<E> *> syms;
};
template <typename E>
class SectCreateSection : public Chunk<E> {
public:
SectCreateSection(Context<E> &ctx, std::string_view seg, std::string_view sect,
std::string_view contents);
void copy_buf(Context<E> &ctx) override;
std::string_view contents;
};
//
// mapfile.cc
//
template <typename E>
void print_map(Context<E> &ctx);
//
// yaml.cc
//
struct YamlNode {
std::variant<std::string_view,
std::vector<YamlNode>,
std::map<std::string_view, YamlNode>> data;
};
struct YamlError {
std::string msg;
i64 pos;
};
std::variant<std::vector<YamlNode>, YamlError>
parse_yaml(std::string_view str);
//
// tapi.cc
//
struct TextDylib {
std::string_view install_name;
std::vector<std::string_view> reexported_libs;
std::set<std::string_view> exports;
std::set<std::string_view> weak_exports;
};
template <typename E>
TextDylib parse_tbd(Context<E> &ctx, MappedFile<Context<E>> *mf);
//
// cmdline.cc
//
template <typename E>
i64 parse_version(Context<E> &ctx, std::string_view arg);
template <typename E>
std::vector<std::string> parse_nonpositional_args(Context<E> &ctx);
//
// dead-strip.cc
//
template <typename E>
void dead_strip(Context<E> &ctx);
//
// lto.cc
//
template <typename E>
void load_lto_plugin(Context<E> &ctx);
template <typename E>
void do_lto(Context<E> &ctx);
//
// arch-arm64.cc
//
void create_range_extension_thunks(Context<ARM64> &ctx, OutputSection<ARM64> &osec);
//
// main.cc
//
enum UuidKind { UUID_NONE, UUID_HASH, UUID_RANDOM };
struct AddEmptySectionOption {
std::string_view segname;
std::string_view sectname;
};
struct SectCreateOption {
std::string_view segname;
std::string_view sectname;
std::string_view filename;
};
template <typename E>
struct Context {
Context() {
text_seg = OutputSegment<E>::get_instance(*this, "__TEXT");
data_const_seg = OutputSegment<E>::get_instance(*this, "__DATA_CONST");
data_seg = OutputSegment<E>::get_instance(*this, "__DATA");
linkedit_seg = OutputSegment<E>::get_instance(*this, "__LINKEDIT");
text = OutputSection<E>::get_instance(*this, "__TEXT", "__text");
data = OutputSection<E>::get_instance(*this, "__DATA", "__data");
bss = OutputSection<E>::get_instance(*this, "__DATA", "__bss");
cstring = OutputSection<E>::get_instance(*this, "__TEXT", "__cstring");
common = OutputSection<E>::get_instance(*this, "__DATA", "__common");
bss->hdr.type = S_ZEROFILL;
cstring->hdr.type = S_CSTRING_LITERALS;
common->hdr.type = S_ZEROFILL;
}
Context(const Context<E> &) = delete;
void checkpoint() {
if (has_error) {
cleanup();
_exit(1);
}
}
// Command-line arguments
struct {
MultiGlob exported_symbols_list;
MultiGlob unexported_symbols_list;
Symbol<E> *entry = nullptr;
UuidKind uuid = UUID_HASH;
bool ObjC = false;
bool adhoc_codesign = std::is_same_v<E, ARM64>;
bool application_extension = false;
bool color_diagnostics = false;
bool dead_strip = false;
bool dead_strip_dylibs = false;
bool deduplicate = true;
bool demangle = true;
bool dylib = false;
bool dynamic = true;
bool export_dynamic = false;
bool fatal_warnings = false;
bool noinhibit_exec = false;
bool perf = false;
bool quick_exit = true;
bool search_paths_first = true;
bool stats = false;
bool trace = false;
i64 arch = CPU_TYPE_ARM64;
i64 compatibility_version = 0;
i64 current_version = 0;
i64 filler = 0;
i64 headerpad = 256;
i64 pagezero_size = 0;
i64 platform = PLATFORM_MACOS;
i64 platform_min_version = 0;
i64 platform_sdk_version = 0;
i64 stack_size = 0;
i64 thread_count = 0;
std::string chroot;
std::string final_output;
std::string install_name;
std::string lto_library;
std::string map;
std::string object_path_lto;
std::string output = "a.out";
std::vector<AddEmptySectionOption> add_empty_section;
std::vector<SectCreateOption> sectcreate;
std::vector<std::string> U;
std::vector<std::string> framework_paths;
std::vector<std::string> library_paths;
std::vector<std::string> mllvm;
std::vector<std::string> order_file;
std::vector<std::string> rpath;
std::vector<std::string> syslibroot;
std::vector<std::string> u;
} arg;
std::vector<std::string_view> cmdline_args;
u32 output_type = MH_EXECUTE;
i64 file_priority = 10000;
bool all_load = false;
bool needed_l = false;
bool hidden_l = false;
bool weak_l = false;
bool reexport_l = false;
std::unordered_set<std::string_view> loaded_archives;
u8 uuid[16] = {};
bool has_error = false;
u64 tls_begin = 0;
LTOPlugin lto = {};
std::once_flag lto_plugin_loaded;
tbb::concurrent_hash_map<std::string_view, Symbol<E>, HashCmp> symbol_map;
std::unique_ptr<OutputFile<Context<E>>> output_file;
u8 *buf;
bool overwrite_output_file = false;
tbb::concurrent_vector<std::unique_ptr<ObjectFile<E>>> obj_pool;
tbb::concurrent_vector<std::unique_ptr<DylibFile<E>>> dylib_pool;
tbb::concurrent_vector<std::unique_ptr<u8[]>> string_pool;
tbb::concurrent_vector<std::unique_ptr<MappedFile<Context<E>>>> mf_pool;
std::vector<std::unique_ptr<Chunk<E>>> chunk_pool;
tbb::concurrent_vector<std::unique_ptr<TimerRecord>> timer_records;
std::vector<ObjectFile<E> *> objs;
std::vector<DylibFile<E> *> dylibs;
OutputSegment<E> *text_seg = nullptr;
OutputSegment<E> *data_const_seg = nullptr;
OutputSegment<E> *data_seg = nullptr;
OutputSegment<E> *linkedit_seg = nullptr;
std::vector<std::unique_ptr<OutputSegment<E>>> segments;
std::vector<Chunk<E> *> chunks;
OutputMachHeader<E> mach_hdr{*this};
StubsSection<E> stubs{*this};
StubHelperSection<E> stub_helper{*this};
UnwindInfoSection<E> unwind_info{*this};
GotSection<E> got{*this};
LazySymbolPtrSection<E> lazy_symbol_ptr{*this};
DataInCodeSection<E> data_in_code{*this};
ThreadPtrsSection<E> thread_ptrs{*this};
RebaseSection<E> rebase{*this};
BindSection<E> bind{*this};
LazyBindSection<E> lazy_bind{*this};
ExportSection<E> export_{*this};
FunctionStartsSection<E> function_starts{*this};
SymtabSection<E> symtab{*this};
StrtabSection<E> strtab{*this};
std::unique_ptr<CodeSignatureSection<E>> code_sig;
OutputSection<E> *text = nullptr;
OutputSection<E> *data = nullptr;
OutputSection<E> *bss = nullptr;
OutputSection<E> *cstring = nullptr;
OutputSection<E> *common = nullptr;
};
int main(int argc, char **argv);
//
// Inline functions
//
template <typename E>
std::ostream &operator<<(std::ostream &out, const InputSection<E> &sec) {
out << sec.file << "(" << sec.hdr.get_segname() << ","
<< sec.hdr.get_sectname() << ")";
return out;
}
template <typename E>
u64 Subsection<E>::get_addr(Context<E> &ctx) const {
return isec.osec.hdr.addr + output_offset;
}
template <typename E>
u64 Symbol<E>::get_addr(Context<E> &ctx) const {
if (subsec) {
assert(subsec->is_alive);
return subsec->get_addr(ctx) + value;
}
if (stub_idx != -1)
return ctx.stubs.hdr.addr + stub_idx * E::stub_size;
return value;
}
template <typename E>
u64 Symbol<E>::get_got_addr(Context<E> &ctx) const {
assert(got_idx != -1);
return ctx.got.hdr.addr + got_idx * E::word_size;
}
template <typename E>
u64 Symbol<E>::get_tlv_addr(Context<E> &ctx) const {
assert(tlv_idx != -1);
return ctx.thread_ptrs.hdr.addr + tlv_idx * E::word_size;
}
template <typename E>
inline Symbol<E> *get_symbol(Context<E> &ctx, std::string_view name) {
typename decltype(ctx.symbol_map)::const_accessor acc;
ctx.symbol_map.insert(acc, {name, Symbol<E>(name)});
return (Symbol<E> *)(&acc->second);
}
template <typename E>
inline std::ostream &operator<<(std::ostream &out, const Symbol<E> &sym) {
if (opt_demangle && sym.name.starts_with("__Z"))
out << demangle(sym.name.substr(1));
else
out << sym.name;
return out;
}
inline u64 RangeExtensionThunk<ARM64>::get_addr(i64 idx) const {
return output_section.hdr.addr + offset + idx * ENTRY_SIZE;
}
} // namespace mold::macho