#include "COFFLinkerContext.h"
#include "Chunks.h"
#include "Symbols.h"
#include "lld/Common/Timer.h"
#include "llvm/ADT/STLExtras.h"
#include "llvm/Support/TimeProfiler.h"
#include <vector>
namespace lld::coff {
void markLive(COFFLinkerContext &ctx) {
llvm::TimeTraceScope timeScope("Mark live");
ScopedTimer t(ctx.gcTimer);
SmallVector<SectionChunk *, 256> worklist;
for (Chunk *c : ctx.symtab.getChunks())
if (auto *sc = dyn_cast<SectionChunk>(c))
if (sc->live && !sc->isDWARF())
worklist.push_back(sc);
auto enqueue = [&](SectionChunk *c) {
if (c->live)
return;
c->live = true;
worklist.push_back(c);
};
auto addSym = [&](Symbol *b) {
if (auto *sym = dyn_cast<DefinedRegular>(b))
enqueue(sym->getChunk());
else if (auto *sym = dyn_cast<DefinedImportData>(b))
sym->file->live = true;
else if (auto *sym = dyn_cast<DefinedImportThunk>(b))
sym->wrappedSym->file->live = sym->wrappedSym->file->thunkLive = true;
};
for (Symbol *b : ctx.config.gcroot)
addSym(b);
while (!worklist.empty()) {
SectionChunk *sc = worklist.pop_back_val();
assert(sc->live && "We mark as live when pushing onto the worklist!");
for (Symbol *b : sc->symbols())
if (b)
addSym(b);
for (SectionChunk &c : sc->children())
enqueue(&c);
if (Defined *entryThunk = sc->getEntryThunk())
addSym(entryThunk);
}
}
}