mirror of
https://github.com/rui314/mold.git
synced 2025-01-01 05:02:36 +03:00
176 lines
5.2 KiB
C++
176 lines
5.2 KiB
C++
// This file implements a mark-sweep garbage collector for -gc-sections.
|
|
// In this algorithm, vertices are sections and edges are relocations.
|
|
// Any section that is reachable from a root section is considered alive.
|
|
|
|
#include "mold.h"
|
|
|
|
#include <tbb/concurrent_vector.h>
|
|
#include <tbb/parallel_for_each.h>
|
|
|
|
typedef tbb::parallel_do_feeder<InputSection *> Feeder;
|
|
|
|
static bool is_init_fini(const InputSection &isec) {
|
|
return isec.shdr.sh_type == SHT_INIT_ARRAY ||
|
|
isec.shdr.sh_type == SHT_FINI_ARRAY ||
|
|
isec.shdr.sh_type == SHT_PREINIT_ARRAY ||
|
|
isec.name.starts_with(".ctors") ||
|
|
isec.name.starts_with(".dtors") ||
|
|
isec.name.starts_with(".init") ||
|
|
isec.name.starts_with(".fini");
|
|
}
|
|
|
|
static bool mark_section(InputSection *isec) {
|
|
return isec && isec->is_alive && !isec->is_visited.exchange(true);
|
|
}
|
|
|
|
static void visit(InputSection *isec, Feeder &feeder, i64 depth) {
|
|
assert(isec->is_visited);
|
|
|
|
// A relocation can refer either a section fragment (i.e. a piece of
|
|
// string in a mergeable string section) or a symbol. Mark all
|
|
// section fragments as alive.
|
|
for (SectionFragmentRef &ref : isec->rel_fragments)
|
|
ref.frag->is_alive = true;
|
|
|
|
// If this is a text section, .eh_frame may contain records
|
|
// describing how to handle exceptions for that function.
|
|
// We want to keep associated .eh_frame records.
|
|
for (FdeRecord &fde : isec->fdes)
|
|
for (EhReloc &rel : std::span(fde.rels).subspan(1))
|
|
if (InputSection *isec = rel.sym.input_section)
|
|
if (mark_section(isec))
|
|
feeder.add(isec);
|
|
|
|
for (ElfRela &rel : isec->rels) {
|
|
Symbol &sym = *isec->file.symbols[rel.r_sym];
|
|
|
|
// Symbol can refer either a section fragment or an input section.
|
|
// Mark a fragment as alive.
|
|
if (sym.frag) {
|
|
sym.frag->is_alive = true;
|
|
continue;
|
|
}
|
|
|
|
if (!mark_section(sym.input_section))
|
|
continue;
|
|
|
|
// Mark a section alive. For better performacne, we don't call
|
|
// `feeder.add` too often.
|
|
if (depth < 3)
|
|
visit(sym.input_section, feeder, depth + 1);
|
|
else
|
|
feeder.add(sym.input_section);
|
|
}
|
|
}
|
|
|
|
static tbb::concurrent_vector<InputSection *> collect_root_set() {
|
|
Timer t("collect_root_set");
|
|
tbb::concurrent_vector<InputSection *> roots;
|
|
|
|
auto enqueue_section = [&](InputSection *isec) {
|
|
if (mark_section(isec))
|
|
roots.push_back(isec);
|
|
};
|
|
|
|
auto enqueue_symbol = [&](Symbol *sym) {
|
|
if (sym) {
|
|
if (sym->frag)
|
|
sym->frag->is_alive = true;
|
|
else
|
|
enqueue_section(sym->input_section);
|
|
}
|
|
};
|
|
|
|
// Add sections that are not subject to garbage collection.
|
|
tbb::parallel_for_each(out::objs, [&](ObjectFile *file) {
|
|
for (InputSection *isec : file->sections) {
|
|
if (!isec)
|
|
continue;
|
|
|
|
// -gc-sections discards only SHF_ALLOC sections. If you want to
|
|
// reduce the amount of non-memory-mapped segments, you should
|
|
// use `strip` command, compile without debug info or use
|
|
// -strip-all linker option.
|
|
if (!(isec->shdr.sh_flags & SHF_ALLOC))
|
|
isec->is_visited = true;
|
|
|
|
if (is_init_fini(*isec) || isec->shdr.sh_type == SHT_NOTE)
|
|
enqueue_section(isec);
|
|
}
|
|
});
|
|
|
|
// Add sections containing exported symbols
|
|
tbb::parallel_for_each(out::objs, [&](ObjectFile *file) {
|
|
for (Symbol *sym : file->symbols)
|
|
if (sym->file == file && sym->is_exported)
|
|
enqueue_symbol(sym);
|
|
});
|
|
|
|
// Add sections referenced by root symbols.
|
|
enqueue_symbol(Symbol::intern(config.entry));
|
|
|
|
for (std::string_view name : config.undefined)
|
|
enqueue_symbol(Symbol::intern(name));
|
|
|
|
// .eh_frame consists of variable-length records called CIE and FDE
|
|
// records, and they are a unit of inclusion or exclusion.
|
|
// We just keep all CIEs and everything that are referenced by them.
|
|
tbb::parallel_for_each(out::objs, [&](ObjectFile *file) {
|
|
for (CieRecord &cie : file->cies)
|
|
for (EhReloc &rel : cie.rels)
|
|
enqueue_section(rel.sym.input_section);
|
|
});
|
|
|
|
return roots;
|
|
}
|
|
|
|
// Mark all reachable sections
|
|
static void mark(tbb::concurrent_vector<InputSection *> &roots) {
|
|
Timer t("mark");
|
|
|
|
tbb::parallel_do(roots, [&](InputSection *isec, Feeder &feeder) {
|
|
visit(isec, feeder, 0);
|
|
});
|
|
}
|
|
|
|
// Remove unreachable sections
|
|
static void sweep() {
|
|
Timer t("sweep");
|
|
static Counter counter("garbage_sections");
|
|
|
|
tbb::parallel_for_each(out::objs, [&](ObjectFile *file) {
|
|
for (i64 i = 0; i < file->sections.size(); i++) {
|
|
InputSection *isec = file->sections[i];
|
|
|
|
if (isec && isec->is_alive && !isec->is_visited) {
|
|
if (config.print_gc_sections)
|
|
SyncOut() << "removing unused section " << *isec;
|
|
isec->kill();
|
|
counter++;
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
// Non-alloc section fragments are not subject of garbage collection.
|
|
// This function marks such fragments.
|
|
static void mark_nonalloc_fragments() {
|
|
Timer t("mark_nonalloc_fragments");
|
|
|
|
tbb::parallel_for_each(out::objs, [](ObjectFile *file) {
|
|
for (SectionFragment *frag : file->fragments)
|
|
if (!(frag->output_section.shdr.sh_flags & SHF_ALLOC))
|
|
frag->is_alive = true;
|
|
});
|
|
}
|
|
|
|
void gc_sections() {
|
|
Timer t("gc");
|
|
|
|
mark_nonalloc_fragments();
|
|
|
|
tbb::concurrent_vector<InputSection *> roots = collect_root_set();
|
|
mark(roots);
|
|
sweep();
|
|
}
|