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>
89 memcpy(uuid,
UUID,
sizeof(uuid));
94 memcpy(uuid,
UUID.data(),
sizeof(uuid));
98template <MachO::LoadCommandType LCType>
106 MachO::dylib{24, Timestamp, CurrentVersion, CompatibilityVersion}),
108 this->cmdsize += (this->Name.size() + 1 + 3) & ~0x3;
112 bool SwapStruct)
override {
115 return Offset + ((
Name.size() + 1 + 3) & ~0x3);
144 cmdsize += (this->Path.size() + 1 + 3) & ~0x3;
148 bool SwapStruct)
override {
151 return Offset + ((
Path.size() + 1 + 3) & ~0x3);
160 struct SymbolContainer {
161 size_t SymbolIndexBase = 0;
162 std::vector<typename MachOTraits::NList> Symbols;
165 struct StringTableEntry {
170 using StringTable = std::vector<StringTableEntry>;
172 static bool swapStruct() {
192 return SC->SymbolIndexBase + Idx;
197 return S->SectionNumber;
200 typename MachOTraits::NList &
nlist() {
202 return SC->Symbols[Idx];
219 assert(
Type < 16 &&
"Relocation type out of range");
247 memset(&
rawStruct(), 0,
sizeof(
typename MachOTraits::Section));
248 assert(SecName.
size() <= 16 &&
"SecName too long");
249 assert(SegName.
size() <= 16 &&
"SegName too long");
250 memcpy(this->sectname, SecName.
data(), SecName.
size());
251 memcpy(this->segname, SegName.
data(), SegName.
size());
257 typename MachOTraits::NList Sym;
263 SC.Symbols.push_back(Sym);
264 return {
SC,
SC.Symbols.size() - 1};
273 return static_cast<typename MachOTraits::Section &
>(*this);
283 assert(SegName.
size() <= 16 &&
"SegName too long");
284 memcpy(this->segname, SegName.
data(), SegName.
size());
287 this->initprot = this->maxprot;
291 Sections.push_back(std::make_unique<Section>(
Builder, SecName, SegName));
296 bool SwapStruct)
override {
307 Header.magic = MachOTraits::Magic;
312 static_assert(LCType != MachOTraits::SegmentCmd,
313 "Use addSegment to add segment load command");
314 auto LC = std::make_unique<MachOBuilderLoadCommand<LCType>>(
315 std::forward<ArgTs>(Args)...);
317 LoadCommands.push_back(std::move(LC));
322 if (Strings.empty() && !Str.empty())
324 return Strings.insert(std::make_pair(Str, Strings.size())).first->second;
328 Segments.push_back(
Segment(*
this, SegName));
329 return Segments.back();
335 typename MachOTraits::NList Sym;
341 SC.Symbols.push_back(Sym);
342 return {SC, SC.Symbols.size() - 1};
359 for (
auto &Seg : Segments) {
361 Seg.Sections.size() *
sizeof(
typename MachOTraits::Section);
362 Seg.nsects = Seg.Sections.size();
365 for (
auto &LC : LoadCommands)
371 size_t SegVMAddr = 0;
372 for (
auto &Seg : Segments) {
373 Seg.vmaddr = SegVMAddr;
375 for (
auto &Sec : Seg.Sections) {
377 if (Sec->Content.Size)
379 Sec->size = Sec->Content.Size;
380 Sec->addr = SegVMAddr + Sec->offset - Seg.fileoff;
381 Offset += Sec->Content.Size;
383 size_t SegContentSize =
Offset - Seg.fileoff;
384 Seg.filesize = SegContentSize;
387 :
alignTo(SegContentSize, PageSize);
388 SegVMAddr += Seg.vmsize;
392 for (
auto &Sym : SC.Symbols)
393 Sym.n_strx = StrTab[Sym.n_strx].Offset;
397 size_t NumSymbols = SC.Symbols.size();
398 size_t SectionNumber = 0;
399 for (
auto &Seg : Segments) {
400 for (
auto &Sec : Seg.Sections) {
402 Sec->SectionNumber = SectionNumber;
403 Sec->SC.SymbolIndexBase = NumSymbols;
404 NumSymbols += Sec->SC.Symbols.size();
405 for (
auto &Sym : Sec->SC.Symbols) {
406 Sym.n_sect = SectionNumber;
407 Sym.n_strx = StrTab[Sym.n_strx].Offset;
408 Sym.n_value += Sec->addr;
414 bool OffsetAlignedForRelocs =
false;
415 for (
auto &Seg : Segments) {
416 for (
auto &Sec : Seg.Sections) {
417 if (!Sec->Relocs.empty()) {
418 if (!OffsetAlignedForRelocs) {
420 OffsetAlignedForRelocs =
true;
423 Sec->nreloc = Sec->Relocs.size();
425 for (
auto &R : Sec->Relocs)
426 R.r_symbolnum = R.Target.isSymbol() ? R.Target.getSymbolNum()
427 : R.Target.getSectionId();
433 if (NumSymbols > 0) {
435 SymTabLC->symoff =
Offset;
436 SymTabLC->nsyms = NumSymbols;
439 if (!StrTab.empty()) {
440 Offset += NumSymbols *
sizeof(
typename MachOTraits::NList);
441 size_t StringTableSize =
442 StrTab.back().Offset + StrTab.back().S.size() + 1;
444 SymTabLC->stroff =
Offset;
445 SymTabLC->strsize = StringTableSize;
446 Offset += StringTableSize;
467 void makeStringTable() {
471 StrTab.resize(Strings.size());
472 for (
auto &[Str, Idx] : Strings)
473 StrTab[Idx] = {Str, 0};
475 for (
auto &Elem : StrTab) {
477 Offset += Elem.S.size() + 1;
482 Header.ncmds = Segments.size() + LoadCommands.size();
487 for (
auto &Seg : Segments)
492 size_t writeLoadCommands(MutableArrayRef<char> Buf,
size_t Offset) {
493 for (
auto &LC : LoadCommands)
498 size_t writeSectionContent(MutableArrayRef<char> Buf,
size_t Offset) {
499 for (
auto &Seg : Segments) {
500 for (
auto &Sec : Seg.Sections) {
501 if (!Sec->Content.Data) {
502 assert(Sec->Relocs.empty() &&
503 "Cant' have relocs for zero-fill segment");
506 while (
Offset != Sec->offset)
509 assert(
Offset + Sec->Content.Size <= Buf.size() &&
"Buffer overflow");
510 memcpy(&Buf[
Offset], Sec->Content.Data, Sec->Content.Size);
511 Offset += Sec->Content.Size;
517 size_t writeRelocations(MutableArrayRef<char> Buf,
size_t Offset) {
518 for (
auto &Seg : Segments) {
519 for (
auto &Sec : Seg.Sections) {
520 if (!Sec->Relocs.empty()) {
521 while (
Offset %
sizeof(MachO::relocation_info))
524 for (
auto &R : Sec->Relocs) {
525 assert(
Offset +
sizeof(MachO::relocation_info) <= Buf.size() &&
527 memcpy(&Buf[
Offset],
reinterpret_cast<const char *
>(&
R.rawStruct()),
528 sizeof(MachO::relocation_info));
529 Offset +=
sizeof(MachO::relocation_info);
536 size_t writeSymbols(MutableArrayRef<char> Buf,
size_t Offset) {
539 size_t NumSymbols = SC.Symbols.size();
540 for (
auto &Seg : Segments)
541 for (
auto &Sec : Seg.Sections)
542 NumSymbols += Sec->SC.Symbols.size();
549 while (
Offset %
sizeof(
typename MachOTraits::NList))
553 for (
auto &Sym : SC.Symbols)
557 for (
auto &Seg : Segments) {
558 for (
auto &Sec : Seg.Sections) {
559 for (
auto &Sym : Sec->SC.Symbols) {
567 size_t writeStrings(MutableArrayRef<char> Buf,
size_t Offset) {
568 for (
auto &Elem : StrTab) {
569 assert(
Offset + Elem.S.size() + 1 <= Buf.size() &&
"Buffer overflow");
570 memcpy(&Buf[
Offset], Elem.S.data(), Elem.S.size());
578 std::list<Segment> Segments;
579 std::vector<std::unique_ptr<MachOBuilderLoadCommandBase>> LoadCommands;
583 std::map<StringRef, size_t> Strings;
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
std::pair< llvm::MachO::Target, std::string > UUID
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.
constexpr 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.
static constexpr uint32_t Magic
MachO::relocation_info Relocation
static constexpr llvm::endianness Endianness
static constexpr MachO::LoadCommandType SymTabCmd
static constexpr MachO::LoadCommandType SegmentCmd
MachO::section_64 Section
MachO::mach_header_64 Header
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 ~MachOBuilderLoadCommandBase()=default
virtual size_t write(MutableArrayRef< char > Buf, size_t Offset, bool SwapStruct)=0
MachOBuilder load command wrapper type.
MachOBuilderDylibLoadCommand(std::string Name, uint32_t Timestamp, uint32_t CurrentVersion, uint32_t CompatibilityVersion)
MachOBuilderDylibLoadCommand(std::string Name, uint32_t Timestamp, uint32_t CurrentVersion, uint32_t CompatibilityVersion)
MachOBuilderDylibLoadCommand(std::string Name, uint32_t Timestamp, uint32_t CurrentVersion, uint32_t CompatibilityVersion)
MachOBuilderLoadCommand(std::string Path)
size_t write(MutableArrayRef< char > Buf, size_t Offset, bool SwapStruct) override
MachOBuilderLoadCommand(const uint8_t(&UUID)[16])
MachOBuilderLoadCommand(const std::array< uint8_t, 16 > &UUID)
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