13#ifndef LLVM_EXECUTIONENGINE_ORC_MACHOBUILDER_H
14#define LLVM_EXECUTIONENGINE_ORC_MACHOBUILDER_H
27template <
typename MachOStruct>
33 memcpy(&Buf[
Offset],
reinterpret_cast<const char *
>(&S),
sizeof(MachOStruct));
34 return Offset +
sizeof(MachOStruct);
40 virtual size_t size()
const = 0;
48#define HANDLE_LOAD_COMMAND(Name, Value, LCStruct) \
50 struct MachOBuilderLoadCommandImplBase<MachO::Name> \
51 : public MachO::LCStruct, public MachOBuilderLoadCommandBase { \
52 using CmdStruct = LCStruct; \
53 MachOBuilderLoadCommandImplBase() { \
54 memset(&rawStruct(), 0, sizeof(CmdStruct)); \
56 cmdsize = sizeof(CmdStruct); \
58 template <typename... ArgTs> \
59 MachOBuilderLoadCommandImplBase(ArgTs &&...Args) \
60 : CmdStruct{Value, sizeof(CmdStruct), std::forward<ArgTs>(Args)...} {} \
61 CmdStruct &rawStruct() { return static_cast<CmdStruct &>(*this); } \
62 size_t size() const override { return cmdsize; } \
63 size_t write(MutableArrayRef<char> Buf, size_t Offset, \
64 bool SwapStruct) override { \
65 return writeMachOStruct(Buf, Offset, rawStruct(), SwapStruct); \
69#include "llvm/BinaryFormat/MachO.def"
71#undef HANDLE_LOAD_COMMAND
73template <MachO::LoadCommandType LCType>
79 template <
typename... ArgTs>
84template <MachO::LoadCommandType LCType>
92 MachO::dylib{24,
Timestamp, CurrentVersion, CompatibilityVersion}),
94 this->cmdsize += (this->Name.size() + 1 + 3) & ~0x3;
98 bool SwapStruct)
override {
101 return Offset + ((
Name.size() + 1 + 3) & ~0x3);
130 cmdsize += (this->Path.size() + 1 + 3) & ~0x3;
134 bool SwapStruct)
override {
137 return Offset + ((Path.size() + 1 + 3) & ~0x3);
146 struct SymbolContainer {
147 size_t SymbolIndexBase = 0;
148 std::vector<typename MachOTraits::NList> Symbols;
151 struct StringTableEntry {
156 using StringTable = std::vector<StringTableEntry>;
158 static bool swapStruct() {
178 return SC->SymbolIndexBase + Idx;
186 typename MachOTraits::NList &
nlist() {
188 return SC->Symbols[Idx];
205 assert(
Type < 16 &&
"Relocation type out of range");
233 memset(&
rawStruct(), 0,
sizeof(
typename MachOTraits::Section));
234 assert(SecName.
size() <= 16 &&
"SecName too long");
235 assert(SegName.
size() <= 16 &&
"SegName too long");
236 memcpy(this->sectname, SecName.
data(), SecName.
size());
237 memcpy(this->segname, SegName.
data(), SegName.
size());
243 typename MachOTraits::NList
Sym;
249 SC.Symbols.push_back(
Sym);
250 return {
SC,
SC.Symbols.size() - 1};
259 return static_cast<typename MachOTraits::Section &
>(*this);
269 assert(SegName.
size() <= 16 &&
"SegName too long");
270 memcpy(this->segname, SegName.
data(), SegName.
size());
273 this->initprot = this->maxprot;
277 Sections.push_back(std::make_unique<Section>(
Builder, SecName, SegName));
282 bool SwapStruct)
override {
293 Header.magic = MachOTraits::Magic;
298 static_assert(LCType != MachOTraits::SegmentCmd,
299 "Use addSegment to add segment load command");
300 auto LC = std::make_unique<MachOBuilderLoadCommand<LCType>>(
301 std::forward<ArgTs>(Args)...);
303 LoadCommands.push_back(std::move(LC));
308 if (Strings.empty() && !Str.empty())
310 return Strings.insert(std::make_pair(Str, Strings.size())).first->second;
314 Segments.push_back(
Segment(*
this, SegName));
315 return Segments.back();
321 typename MachOTraits::NList
Sym;
327 SC.Symbols.push_back(
Sym);
328 return {SC, SC.Symbols.size() - 1};
341 SymTabLC = &addLoadCommand<MachOTraits::SymTabCmd>();
345 for (
auto &Seg : Segments) {
347 Seg.Sections.size() *
sizeof(
typename MachOTraits::Section);
348 Seg.nsects = Seg.Sections.size();
351 for (
auto &LC : LoadCommands)
357 size_t SegVMAddr = 0;
358 for (
auto &Seg : Segments) {
359 Seg.vmaddr = SegVMAddr;
361 for (
auto &Sec : Seg.Sections) {
363 if (Sec->Content.Size)
365 Sec->size = Sec->Content.Size;
366 Sec->addr = SegVMAddr + Sec->offset - Seg.fileoff;
367 Offset += Sec->Content.Size;
369 size_t SegContentSize =
Offset - Seg.fileoff;
370 Seg.filesize = SegContentSize;
373 :
alignTo(SegContentSize, PageSize);
374 SegVMAddr += Seg.vmsize;
378 for (
auto &
Sym : SC.Symbols)
379 Sym.n_strx = StrTab[
Sym.n_strx].Offset;
383 size_t NumSymbols = SC.Symbols.size();
384 size_t SectionNumber = 0;
385 for (
auto &Seg : Segments) {
386 for (
auto &Sec : Seg.Sections) {
388 Sec->SectionNumber = SectionNumber;
389 Sec->SC.SymbolIndexBase = NumSymbols;
390 NumSymbols += Sec->SC.Symbols.size();
391 for (
auto &
Sym : Sec->SC.Symbols) {
392 Sym.n_sect = SectionNumber;
393 Sym.n_strx = StrTab[
Sym.n_strx].Offset;
394 Sym.n_value += Sec->addr;
400 bool OffsetAlignedForRelocs =
false;
401 for (
auto &Seg : Segments) {
402 for (
auto &Sec : Seg.Sections) {
403 if (!Sec->Relocs.empty()) {
404 if (!OffsetAlignedForRelocs) {
406 OffsetAlignedForRelocs =
true;
409 Sec->nreloc = Sec->Relocs.size();
411 for (
auto &R : Sec->Relocs)
412 R.r_symbolnum = R.Target.isSymbol() ? R.Target.getSymbolNum()
413 : R.Target.getSectionId();
419 if (NumSymbols > 0) {
421 SymTabLC->symoff =
Offset;
422 SymTabLC->nsyms = NumSymbols;
425 if (!StrTab.empty()) {
426 Offset += NumSymbols *
sizeof(
typename MachOTraits::NList);
427 size_t StringTableSize =
428 StrTab.back().Offset + StrTab.back().S.size() + 1;
430 SymTabLC->stroff =
Offset;
431 SymTabLC->strsize = StringTableSize;
432 Offset += StringTableSize;
453 void makeStringTable() {
457 StrTab.resize(Strings.size());
458 for (
auto &[Str,
Idx] : Strings)
459 StrTab[
Idx] = {Str, 0};
461 for (
auto &Elem : StrTab) {
463 Offset += Elem.S.size() + 1;
468 Header.ncmds = Segments.size() + LoadCommands.size();
473 for (
auto &Seg : Segments)
478 size_t writeLoadCommands(MutableArrayRef<char> Buf,
size_t Offset) {
479 for (
auto &LC : LoadCommands)
484 size_t writeSectionContent(MutableArrayRef<char> Buf,
size_t Offset) {
485 for (
auto &Seg : Segments) {
486 for (
auto &Sec : Seg.Sections) {
487 if (!Sec->Content.Data) {
488 assert(Sec->Relocs.empty() &&
489 "Cant' have relocs for zero-fill segment");
492 while (
Offset != Sec->offset)
495 assert(
Offset + Sec->Content.Size <= Buf.size() &&
"Buffer overflow");
496 memcpy(&Buf[
Offset], Sec->Content.Data, Sec->Content.Size);
497 Offset += Sec->Content.Size;
503 size_t writeRelocations(MutableArrayRef<char> Buf,
size_t Offset) {
504 for (
auto &Seg : Segments) {
505 for (
auto &Sec : Seg.Sections) {
506 if (!Sec->Relocs.empty()) {
507 while (
Offset %
sizeof(MachO::relocation_info))
510 for (
auto &R : Sec->Relocs) {
511 assert(
Offset +
sizeof(MachO::relocation_info) <= Buf.size() &&
513 memcpy(&Buf[
Offset],
reinterpret_cast<const char *
>(&
R.rawStruct()),
514 sizeof(MachO::relocation_info));
515 Offset +=
sizeof(MachO::relocation_info);
522 size_t writeSymbols(MutableArrayRef<char> Buf,
size_t Offset) {
525 size_t NumSymbols = SC.Symbols.size();
526 for (
auto &Seg : Segments)
527 for (
auto &Sec : Seg.Sections)
528 NumSymbols += Sec->SC.Symbols.size();
535 while (
Offset %
sizeof(
typename MachOTraits::NList))
539 for (
auto &
Sym : SC.Symbols)
543 for (
auto &Seg : Segments) {
544 for (
auto &Sec : Seg.Sections) {
545 for (
auto &
Sym : Sec->SC.Symbols) {
553 size_t writeStrings(MutableArrayRef<char> Buf,
size_t Offset) {
554 for (
auto &Elem : StrTab) {
555 assert(
Offset + Elem.S.size() + 1 <= Buf.size() &&
"Buffer overflow");
556 memcpy(&Buf[
Offset], Elem.S.data(), Elem.S.size());
565 std::vector<std::unique_ptr<MachOBuilderLoadCommandBase>> LoadCommands;
569 std::map<StringRef, size_t> Strings;
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
Returns the sub type a function will return at a given Idx Should correspond to the result type of an ExtractValue instruction executed with just that one unsigned Idx
size_t size() const
size - Get the array size.
MutableArrayRef - Represent a mutable reference to an array (0 or more elements consecutively in memo...
StringRef - Represent a constant reference to a string, i.e.
constexpr size_t size() const
size - Get the string size.
constexpr const char * data() const
data - Get a pointer to the start of the string (which may not be null terminated).
Target - Wrapper for Target specific information.
The instances of the Type class are immutable: once they are created, they are never changed.
LLVM Value Representation.
RelocTarget(SymbolContainer &SC, size_t Idx)
MachOTraits::NList & nlist()
RelocTarget(const Section &S)
MachOBuilderLoadCommand< LCType > & addLoadCommand(ArgTs &&...Args)
void write(MutableArrayRef< char > Buffer)
RelocTarget addSymbol(StringRef Name, uint8_t Type, uint8_t Sect, uint16_t Desc, typename MachOTraits::UIntPtr Value)
StringId addString(StringRef Str)
MachOBuilder(size_t PageSize)
Segment & addSegment(StringRef SegName)
MachOTraits::Header Header
void swapStruct(fat_header &mh)
size_t writeMachOStruct(MutableArrayRef< char > Buf, size_t Offset, MachOStruct S, bool SwapStruct)
This is an optimization pass for GlobalISel generic memory operations.
uint64_t alignTo(uint64_t Size, Align A)
Returns a multiple of A needed to store Size bytes.
OutputIt move(R &&Range, OutputIt Out)
Provide wrappers to std::move which take ranges instead of having to pass begin/end explicitly.
Implement std::hash so that hash_code can be used in STL containers.
Description of the encoding of one expression Op.
static constexpr uint32_t Magic
static constexpr llvm::endianness Endianness
static constexpr MachO::LoadCommandType SymTabCmd
static constexpr MachO::LoadCommandType SegmentCmd
MachOBuilderDylibLoadCommand(std::string Name, uint32_t Timestamp, uint32_t CurrentVersion, uint32_t CompatibilityVersion)
size_t write(MutableArrayRef< char > Buf, size_t Offset, bool SwapStruct) override
Base type for MachOBuilder load command wrappers.
virtual size_t size() const =0
virtual size_t write(MutableArrayRef< char > Buf, size_t Offset, bool SwapStruct)=0
virtual ~MachOBuilderLoadCommandBase()
MachOBuilder load command wrapper type.
MachOBuilderLoadCommand(std::string Path)
size_t write(MutableArrayRef< char > Buf, size_t Offset, bool SwapStruct) override
MachOBuilderLoadCommand(ArgTs &&...Args)
MachOBuilderLoadCommand()=default
Reloc(int32_t Offset, RelocTarget Target, bool PCRel, unsigned Length, unsigned Type)
MachO::relocation_info & rawStruct()
void addReloc(int32_t Offset, RelocTarget Target, bool PCRel, unsigned Length, unsigned Type)
std::vector< Reloc > Relocs
Section(MachOBuilder &Builder, StringRef SecName, StringRef SegName)
RelocTarget addSymbol(int32_t Offset, StringRef Name, uint8_t Type, uint16_t Desc)
Segment(MachOBuilder &Builder, StringRef SegName)
Section & addSection(StringRef SecName, StringRef SegName)
std::vector< std::unique_ptr< Section > > Sections
size_t write(MutableArrayRef< char > Buf, size_t Offset, bool SwapStruct) override