mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-13 20:32:21 +00:00
3490d23337
In the current implementation, GCStrategy is a part of the ownership structure for the gc metadata which describes a Module. It also contains a reference to the module in question. As a result, GCStrategy instances are essentially Module specific. I plan to transition away from this design. Instead, a GCStrategy will be owned by the LLVMContext. It will be a lightweight policy object which contains no information about the Modules or Functions involved, but can be easily reached given a Function. The first step in this transition is to remove the direct Module reference from GCStrategy. This also requires removing the single user of this reference, the GCMetadataPrinter hierarchy. In theory, this will allow the lifetime of the printers to be scoped to the LLVMContext as well, but in practice, I'm not actually changing that. (Yet?) An alternate design would have been to move the direct Module reference into the GCMetadataPrinter and change the keying of the owning maps to explicitly key off both GCStrategy and Module. I'm open to doing it that way instead, but didn't see much value in preserving the per Module association for GCMetadataPrinters. The next change in this sequence will be to start unwinding the intertwined ownership between GCStrategy, GCModuleInfo, and GCFunctionInfo. Differential Revision: http://reviews.llvm.org/D6566 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@223859 91177308-0d34-0410-b5e6-96231b3b80d8
169 lines
5.6 KiB
C++
169 lines
5.6 KiB
C++
//===-- OcamlGCPrinter.cpp - Ocaml frametable emitter ---------------------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file implements printing the assembly code for an Ocaml frametable.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/CodeGen/GCs.h"
|
|
#include "llvm/ADT/SmallString.h"
|
|
#include "llvm/CodeGen/AsmPrinter.h"
|
|
#include "llvm/CodeGen/GCMetadataPrinter.h"
|
|
#include "llvm/IR/DataLayout.h"
|
|
#include "llvm/IR/Mangler.h"
|
|
#include "llvm/IR/Module.h"
|
|
#include "llvm/MC/MCAsmInfo.h"
|
|
#include "llvm/MC/MCContext.h"
|
|
#include "llvm/MC/MCStreamer.h"
|
|
#include "llvm/MC/MCSymbol.h"
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
#include "llvm/Support/FormattedStream.h"
|
|
#include "llvm/Target/TargetLoweringObjectFile.h"
|
|
#include "llvm/Target/TargetMachine.h"
|
|
#include "llvm/Target/TargetSubtargetInfo.h"
|
|
#include <cctype>
|
|
using namespace llvm;
|
|
|
|
namespace {
|
|
|
|
class OcamlGCMetadataPrinter : public GCMetadataPrinter {
|
|
public:
|
|
void beginAssembly(Module &M, AsmPrinter &AP) override;
|
|
void finishAssembly(Module &M, AsmPrinter &AP) override;
|
|
};
|
|
|
|
}
|
|
|
|
static GCMetadataPrinterRegistry::Add<OcamlGCMetadataPrinter>
|
|
Y("ocaml", "ocaml 3.10-compatible collector");
|
|
|
|
void llvm::linkOcamlGCPrinter() { }
|
|
|
|
static void EmitCamlGlobal(const Module &M, AsmPrinter &AP, const char *Id) {
|
|
const std::string &MId = M.getModuleIdentifier();
|
|
|
|
std::string SymName;
|
|
SymName += "caml";
|
|
size_t Letter = SymName.size();
|
|
SymName.append(MId.begin(), std::find(MId.begin(), MId.end(), '.'));
|
|
SymName += "__";
|
|
SymName += Id;
|
|
|
|
// Capitalize the first letter of the module name.
|
|
SymName[Letter] = toupper(SymName[Letter]);
|
|
|
|
SmallString<128> TmpStr;
|
|
AP.Mang->getNameWithPrefix(TmpStr, SymName);
|
|
|
|
MCSymbol *Sym = AP.OutContext.GetOrCreateSymbol(TmpStr);
|
|
|
|
AP.OutStreamer.EmitSymbolAttribute(Sym, MCSA_Global);
|
|
AP.OutStreamer.EmitLabel(Sym);
|
|
}
|
|
|
|
void OcamlGCMetadataPrinter::beginAssembly(Module &M, AsmPrinter &AP) {
|
|
AP.OutStreamer.SwitchSection(AP.getObjFileLowering().getTextSection());
|
|
EmitCamlGlobal(M, AP, "code_begin");
|
|
|
|
AP.OutStreamer.SwitchSection(AP.getObjFileLowering().getDataSection());
|
|
EmitCamlGlobal(M, AP, "data_begin");
|
|
}
|
|
|
|
/// emitAssembly - Print the frametable. The ocaml frametable format is thus:
|
|
///
|
|
/// extern "C" struct align(sizeof(intptr_t)) {
|
|
/// uint16_t NumDescriptors;
|
|
/// struct align(sizeof(intptr_t)) {
|
|
/// void *ReturnAddress;
|
|
/// uint16_t FrameSize;
|
|
/// uint16_t NumLiveOffsets;
|
|
/// uint16_t LiveOffsets[NumLiveOffsets];
|
|
/// } Descriptors[NumDescriptors];
|
|
/// } caml${module}__frametable;
|
|
///
|
|
/// Note that this precludes programs from stack frames larger than 64K
|
|
/// (FrameSize and LiveOffsets would overflow). FrameTablePrinter will abort if
|
|
/// either condition is detected in a function which uses the GC.
|
|
///
|
|
void OcamlGCMetadataPrinter::finishAssembly(Module &M, AsmPrinter &AP) {
|
|
unsigned IntPtrSize =
|
|
AP.TM.getSubtargetImpl()->getDataLayout()->getPointerSize();
|
|
|
|
AP.OutStreamer.SwitchSection(AP.getObjFileLowering().getTextSection());
|
|
EmitCamlGlobal(M, AP, "code_end");
|
|
|
|
AP.OutStreamer.SwitchSection(AP.getObjFileLowering().getDataSection());
|
|
EmitCamlGlobal(M, AP, "data_end");
|
|
|
|
// FIXME: Why does ocaml emit this??
|
|
AP.OutStreamer.EmitIntValue(0, IntPtrSize);
|
|
|
|
AP.OutStreamer.SwitchSection(AP.getObjFileLowering().getDataSection());
|
|
EmitCamlGlobal(M, AP, "frametable");
|
|
|
|
int NumDescriptors = 0;
|
|
for (iterator I = begin(), IE = end(); I != IE; ++I) {
|
|
GCFunctionInfo &FI = **I;
|
|
for (GCFunctionInfo::iterator J = FI.begin(), JE = FI.end(); J != JE; ++J) {
|
|
NumDescriptors++;
|
|
}
|
|
}
|
|
|
|
if (NumDescriptors >= 1<<16) {
|
|
// Very rude!
|
|
report_fatal_error(" Too much descriptor for ocaml GC");
|
|
}
|
|
AP.EmitInt16(NumDescriptors);
|
|
AP.EmitAlignment(IntPtrSize == 4 ? 2 : 3);
|
|
|
|
for (iterator I = begin(), IE = end(); I != IE; ++I) {
|
|
GCFunctionInfo &FI = **I;
|
|
|
|
uint64_t FrameSize = FI.getFrameSize();
|
|
if (FrameSize >= 1<<16) {
|
|
// Very rude!
|
|
report_fatal_error("Function '" + FI.getFunction().getName() +
|
|
"' is too large for the ocaml GC! "
|
|
"Frame size " + Twine(FrameSize) + ">= 65536.\n"
|
|
"(" + Twine(uintptr_t(&FI)) + ")");
|
|
}
|
|
|
|
AP.OutStreamer.AddComment("live roots for " +
|
|
Twine(FI.getFunction().getName()));
|
|
AP.OutStreamer.AddBlankLine();
|
|
|
|
for (GCFunctionInfo::iterator J = FI.begin(), JE = FI.end(); J != JE; ++J) {
|
|
size_t LiveCount = FI.live_size(J);
|
|
if (LiveCount >= 1<<16) {
|
|
// Very rude!
|
|
report_fatal_error("Function '" + FI.getFunction().getName() +
|
|
"' is too large for the ocaml GC! "
|
|
"Live root count "+Twine(LiveCount)+" >= 65536.");
|
|
}
|
|
|
|
AP.OutStreamer.EmitSymbolValue(J->Label, IntPtrSize);
|
|
AP.EmitInt16(FrameSize);
|
|
AP.EmitInt16(LiveCount);
|
|
|
|
for (GCFunctionInfo::live_iterator K = FI.live_begin(J),
|
|
KE = FI.live_end(J); K != KE; ++K) {
|
|
if (K->StackOffset >= 1<<16) {
|
|
// Very rude!
|
|
report_fatal_error(
|
|
"GC root stack offset is outside of fixed stack frame and out "
|
|
"of range for ocaml GC!");
|
|
}
|
|
AP.EmitInt16(K->StackOffset);
|
|
}
|
|
|
|
AP.EmitAlignment(IntPtrSize == 4 ? 2 : 3);
|
|
}
|
|
}
|
|
}
|