12#include <unordered_set>
18 : Segname(SegName), Sectname(SectName),
19 CanonicalName((
Twine(SegName) +
Twine(
',') + SectName).str()) {}
22 : Segname(SegName), Sectname(SectName),
23 CanonicalName((
Twine(SegName) +
Twine(
',') + SectName).str()),
41 auto ExternalBegin = std::stable_partition(
43 [](
const auto &
Sym) {
return Sym->isLocalSymbol(); });
44 std::stable_partition(ExternalBegin, std::end(
Symbols), [](
const auto &
Sym) {
45 return !
Sym->isUndefinedSymbol();
55 static constexpr char TextSegmentName[] =
"__TEXT";
60 case MachO::LC_CODE_SIGNATURE:
63 case MachO::LC_SEGMENT:
68 case MachO::LC_SEGMENT_64:
73 case MachO::LC_SYMTAB:
76 case MachO::LC_DYSYMTAB:
79 case MachO::LC_DYLD_INFO:
80 case MachO::LC_DYLD_INFO_ONLY:
83 case MachO::LC_DATA_IN_CODE:
86 case MachO::LC_LINKER_OPTIMIZATION_HINT:
89 case MachO::LC_FUNCTION_STARTS:
92 case MachO::LC_DYLIB_CODE_SIGN_DRS:
95 case MachO::LC_DYLD_CHAINED_FIXUPS:
98 case MachO::LC_DYLD_EXPORTS_TRIE:
107 auto It = std::stable_partition(
109 [&](
const LoadCommand &LC) { return !ToRemove(LC); });
121 auto It = std::stable_partition(
122 std::begin(LC.Sections), std::end(LC.Sections),
123 [&](
const std::unique_ptr<Section> &Sec) { return !ToRemove(Sec); });
124 for (
auto I = LC.Sections.begin(),
End = It;
I !=
End; ++
I) {
125 OldIndexToSection[(*I)->Index] =
I->get();
126 (*I)->Index = NextSectionIndex++;
128 LC.Sections.erase(It, LC.Sections.end());
131 auto IsDead = [&](
const std::unique_ptr<SymbolEntry> &S) ->
bool {
132 std::optional<uint32_t>
Section = S->section();
142 for (
const std::unique_ptr<Section> &Sec : LC.Sections)
144 if (R.Symbol && *R.Symbol && DeadSymbols.
count(*R.Symbol))
146 "symbol '%s' defined in section with index "
147 "'%u' cannot be removed because it is "
148 "referenced by a relocation in section '%s'",
149 (*R.Symbol)->Name.c_str(),
150 *((*R.Symbol)->section()),
151 Sec->CanonicalName.c_str());
155 S->n_sect = OldIndexToSection[S->n_sect]->Index;
165 switch (MLC.load_command_data.cmd) {
166 case MachO::LC_SEGMENT:
168 static_cast<uint64_t>(MLC.segment_command_data.vmaddr) +
169 MLC.segment_command_data.vmsize);
171 case MachO::LC_SEGMENT_64:
172 Addr = std::max(
Addr, MLC.segment_command_64_data.vmaddr +
173 MLC.segment_command_64_data.vmsize);
182template <
typename SegmentType>
186 assert(SegName.
size() <=
sizeof(Seg.segname) &&
"too long segment name");
187 memset(&Seg, 0,
sizeof(SegmentType));
189 strncpy(Seg.segname, SegName.
data(), SegName.
size());
194 Seg.vmaddr = SegVMAddr;
195 Seg.vmsize = SegVMSize;
203 MachO::LC_SEGMENT_64, SegName, SegVMAddr, SegVMSize);
206 MachO::LC_SEGMENT, SegName, SegVMAddr, SegVMSize);
220 switch (MLC.load_command_data.cmd) {
221 case MachO::LC_SEGMENT:
223 case MachO::LC_SEGMENT_64:
232 switch (MLC.load_command_data.cmd) {
233 case MachO::LC_SEGMENT:
234 return MLC.segment_command_data.vmaddr;
235 case MachO::LC_SEGMENT_64:
236 return MLC.segment_command_64_data.vmaddr;
ReachingDefAnalysis InstSet & ToRemove
static void constructSegment(SegmentType &Seg, llvm::MachO::LoadCommandType CmdType, StringRef SegName, uint64_t SegVMAddr, uint64_t SegVMSize)
static StringRef extractSegmentName(const char *SegName)
Extracts a segment name from a string which is possibly non-null-terminated.
assert(ImpDefSCC.getReg()==AMDGPU::SCC &&ImpDefSCC.isDef())
This file defines the SmallPtrSet class.
size_type count(const_arg_type_t< KeyT > Val) const
Return 1 if the specified key is in the map, 0 otherwise.
Lightweight error class with error context and mandatory checking.
static ErrorSuccess success()
Create a success value.
size_type count(ConstPtrType Ptr) const
count - Return 1 if the specified pointer is in the set, 0 otherwise.
std::pair< iterator, bool > insert(PtrType Ptr)
Inserts Ptr if and only if there is no element in the container equal to Ptr.
SmallPtrSet - This class implements a set which is optimized for holding SmallSize or less elements.
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).
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
An efficient, type-erasing, non-owning reference to a callable.
This is an optimization pass for GlobalISel generic memory operations.
Error createStringError(std::error_code EC, char const *Fmt, const Ts &... Vals)
Create formatted StringError object.
void erase_if(Container &C, UnaryPredicate P)
Provide a container algorithm similar to C++ Library Fundamentals v2's erase_if which is equivalent t...
MachO::macho_load_command MachOLoadCommand
std::optional< StringRef > getSegmentName() const
std::optional< uint64_t > getSegmentVMAddr() const
std::optional< size_t > FunctionStartsCommandIndex
The index LC_FUNCTION_STARTS load command if present.
void updateLoadCommandIndexes()
Error removeLoadCommands(function_ref< bool(const LoadCommand &)> ToRemove)
std::optional< size_t > ChainedFixupsCommandIndex
The index LC_DYLD_CHAINED_FIXUPS load command if present.
std::optional< size_t > ExportsTrieCommandIndex
The index LC_DYLD_EXPORTS_TRIE load command if present.
Error removeSections(function_ref< bool(const std::unique_ptr< Section > &)> ToRemove)
std::optional< size_t > DylibCodeSignDRsIndex
The index of LC_DYLIB_CODE_SIGN_DRS load command if present.
std::optional< size_t > SymTabCommandIndex
The index of LC_SYMTAB load command if present.
std::optional< size_t > DyLdInfoCommandIndex
The index of LC_DYLD_INFO or LC_DYLD_INFO_ONLY load command if present.
std::vector< LoadCommand > LoadCommands
std::optional< size_t > DataInCodeCommandIndex
The index LC_DATA_IN_CODE load command if present.
std::optional< size_t > DySymTabCommandIndex
The index LC_DYSYMTAB load command if present.
std::optional< size_t > TextSegmentCommandIndex
The index of the LC_SEGMENT or LC_SEGMENT_64 load command corresponding to the __TEXT segment.
uint64_t nextAvailableSegmentAddress() const
std::optional< size_t > CodeSignatureCommandIndex
The index of LC_CODE_SIGNATURE load command if present.
LoadCommand & addSegment(StringRef SegName, uint64_t SegVMSize)
Creates a new segment load command in the object and returns a reference to the newly created load co...
std::optional< size_t > LinkerOptimizationHintCommandIndex
The index of LC_LINKER_OPTIMIZATIN_HINT load command if present.
Section(StringRef SegName, StringRef SectName)
The location of the symbol table inside the binary is described by LC_SYMTAB load command.
void updateSymbols(function_ref< void(SymbolEntry &)> Callable)
const SymbolEntry * getSymbolByIndex(uint32_t Index) const
std::vector< std::unique_ptr< SymbolEntry > > Symbols
void removeSymbols(function_ref< bool(const std::unique_ptr< SymbolEntry > &)> ToRemove)