2008-08-17 18:44:35 +00:00
|
|
|
//===-- OcamlGCPrinter.cpp - Ocaml frametable emitter ---------------------===//
|
2008-01-07 02:31:11 +00:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
2008-08-17 18:44:35 +00:00
|
|
|
// This file implements printing the assembly code for an Ocaml frametable.
|
2008-01-07 02:31:11 +00:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
2009-01-16 06:53:46 +00:00
|
|
|
|
2008-08-17 12:56:54 +00:00
|
|
|
#include "llvm/CodeGen/GCs.h"
|
2012-12-03 16:50:05 +00:00
|
|
|
#include "llvm/ADT/SmallString.h"
|
2008-01-07 02:31:11 +00:00
|
|
|
#include "llvm/CodeGen/AsmPrinter.h"
|
2008-08-17 18:44:35 +00:00
|
|
|
#include "llvm/CodeGen/GCMetadataPrinter.h"
|
2013-01-02 11:36:10 +00:00
|
|
|
#include "llvm/IR/DataLayout.h"
|
2014-01-07 21:19:40 +00:00
|
|
|
#include "llvm/IR/Mangler.h"
|
2013-01-02 11:36:10 +00:00
|
|
|
#include "llvm/IR/Module.h"
|
2009-08-22 20:48:53 +00:00
|
|
|
#include "llvm/MC/MCAsmInfo.h"
|
2010-04-04 07:39:04 +00:00
|
|
|
#include "llvm/MC/MCContext.h"
|
2010-03-14 07:36:49 +00:00
|
|
|
#include "llvm/MC/MCStreamer.h"
|
2012-12-03 16:50:05 +00:00
|
|
|
#include "llvm/MC/MCSymbol.h"
|
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
|
|
#include "llvm/Support/FormattedStream.h"
|
2009-07-28 03:13:23 +00:00
|
|
|
#include "llvm/Target/TargetLoweringObjectFile.h"
|
2008-01-07 02:31:11 +00:00
|
|
|
#include "llvm/Target/TargetMachine.h"
|
2010-12-19 20:43:38 +00:00
|
|
|
#include <cctype>
|
2008-01-07 02:31:11 +00:00
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2009-10-25 06:33:48 +00:00
|
|
|
class OcamlGCMetadataPrinter : public GCMetadataPrinter {
|
2008-08-17 12:08:44 +00:00
|
|
|
public:
|
2014-03-08 06:31:39 +00:00
|
|
|
void beginAssembly(AsmPrinter &AP) override;
|
|
|
|
void finishAssembly(AsmPrinter &AP) override;
|
2008-01-07 02:31:11 +00:00
|
|
|
};
|
2009-01-16 06:53:46 +00:00
|
|
|
|
2008-01-07 02:31:11 +00:00
|
|
|
}
|
|
|
|
|
2008-08-17 12:08:44 +00:00
|
|
|
static GCMetadataPrinterRegistry::Add<OcamlGCMetadataPrinter>
|
|
|
|
Y("ocaml", "ocaml 3.10-compatible collector");
|
|
|
|
|
2008-08-17 18:44:35 +00:00
|
|
|
void llvm::linkOcamlGCPrinter() { }
|
2008-01-07 02:31:11 +00:00
|
|
|
|
2010-04-04 07:39:04 +00:00
|
|
|
static void EmitCamlGlobal(const Module &M, AsmPrinter &AP, const char *Id) {
|
2008-01-07 02:31:11 +00:00
|
|
|
const std::string &MId = M.getModuleIdentifier();
|
2009-01-16 06:53:46 +00:00
|
|
|
|
2010-04-04 07:39:04 +00:00
|
|
|
std::string SymName;
|
|
|
|
SymName += "caml";
|
|
|
|
size_t Letter = SymName.size();
|
|
|
|
SymName.append(MId.begin(), std::find(MId.begin(), MId.end(), '.'));
|
|
|
|
SymName += "__";
|
|
|
|
SymName += Id;
|
2010-07-01 01:00:22 +00:00
|
|
|
|
2008-01-07 02:31:11 +00:00
|
|
|
// Capitalize the first letter of the module name.
|
2010-04-04 07:39:04 +00:00
|
|
|
SymName[Letter] = toupper(SymName[Letter]);
|
2010-07-01 01:00:22 +00:00
|
|
|
|
2010-04-04 07:39:04 +00:00
|
|
|
SmallString<128> TmpStr;
|
|
|
|
AP.Mang->getNameWithPrefix(TmpStr, SymName);
|
2010-07-01 01:00:22 +00:00
|
|
|
|
2010-04-04 07:39:04 +00:00
|
|
|
MCSymbol *Sym = AP.OutContext.GetOrCreateSymbol(TmpStr);
|
|
|
|
|
|
|
|
AP.OutStreamer.EmitSymbolAttribute(Sym, MCSA_Global);
|
|
|
|
AP.OutStreamer.EmitLabel(Sym);
|
2008-01-07 02:31:11 +00:00
|
|
|
}
|
|
|
|
|
2010-04-04 07:39:04 +00:00
|
|
|
void OcamlGCMetadataPrinter::beginAssembly(AsmPrinter &AP) {
|
2009-08-19 05:49:37 +00:00
|
|
|
AP.OutStreamer.SwitchSection(AP.getObjFileLowering().getTextSection());
|
2010-04-04 07:39:04 +00:00
|
|
|
EmitCamlGlobal(getModule(), AP, "code_begin");
|
2009-01-16 06:53:46 +00:00
|
|
|
|
2009-08-19 05:49:37 +00:00
|
|
|
AP.OutStreamer.SwitchSection(AP.getObjFileLowering().getDataSection());
|
2010-04-04 07:39:04 +00:00
|
|
|
EmitCamlGlobal(getModule(), AP, "data_begin");
|
2008-01-07 02:31:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// emitAssembly - Print the frametable. The ocaml frametable format is thus:
|
2009-01-16 06:53:46 +00:00
|
|
|
///
|
2008-01-07 02:31:11 +00:00
|
|
|
/// 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;
|
2009-01-16 06:53:46 +00:00
|
|
|
///
|
2008-01-07 02:31:11 +00:00
|
|
|
/// Note that this precludes programs from stack frames larger than 64K
|
|
|
|
/// (FrameSize and LiveOffsets would overflow). FrameTablePrinter will abort if
|
2008-08-17 18:44:35 +00:00
|
|
|
/// either condition is detected in a function which uses the GC.
|
2009-01-16 06:53:46 +00:00
|
|
|
///
|
2010-04-04 07:39:04 +00:00
|
|
|
void OcamlGCMetadataPrinter::finishAssembly(AsmPrinter &AP) {
|
Revert the majority of the next patch in the address space series:
r165941: Resubmit the changes to llvm core to update the functions to
support different pointer sizes on a per address space basis.
Despite this commit log, this change primarily changed stuff outside of
VMCore, and those changes do not carry any tests for correctness (or
even plausibility), and we have consistently found questionable or flat
out incorrect cases in these changes. Most of them are probably correct,
but we need to devise a system that makes it more clear when we have
handled the address space concerns correctly, and ideally each pass that
gets updated would receive an accompanying test case that exercises that
pass specificaly w.r.t. alternate address spaces.
However, from this commit, I have retained the new C API entry points.
Those were an orthogonal change that probably should have been split
apart, but they seem entirely good.
In several places the changes were very obvious cleanups with no actual
multiple address space code added; these I have not reverted when
I spotted them.
In a few other places there were merge conflicts due to a cleaner
solution being implemented later, often not using address spaces at all.
In those cases, I've preserved the new code which isn't address space
dependent.
This is part of my ongoing effort to clean out the partial address space
code which carries high risk and low test coverage, and not likely to be
finished before the 3.2 release looms closer. Duncan and I would both
like to see the above issues addressed before we return to these
changes.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@167222 91177308-0d34-0410-b5e6-96231b3b80d8
2012-11-01 09:14:31 +00:00
|
|
|
unsigned IntPtrSize = AP.TM.getDataLayout()->getPointerSize();
|
2008-01-07 02:31:11 +00:00
|
|
|
|
2009-08-19 05:49:37 +00:00
|
|
|
AP.OutStreamer.SwitchSection(AP.getObjFileLowering().getTextSection());
|
2010-04-04 07:39:04 +00:00
|
|
|
EmitCamlGlobal(getModule(), AP, "code_end");
|
2009-01-16 06:53:46 +00:00
|
|
|
|
2009-08-19 05:49:37 +00:00
|
|
|
AP.OutStreamer.SwitchSection(AP.getObjFileLowering().getDataSection());
|
2010-04-04 07:39:04 +00:00
|
|
|
EmitCamlGlobal(getModule(), AP, "data_end");
|
2009-01-16 06:53:46 +00:00
|
|
|
|
2010-04-04 07:39:04 +00:00
|
|
|
// FIXME: Why does ocaml emit this??
|
2013-01-09 01:57:54 +00:00
|
|
|
AP.OutStreamer.EmitIntValue(0, IntPtrSize);
|
2009-01-16 06:53:46 +00:00
|
|
|
|
2009-08-19 05:49:37 +00:00
|
|
|
AP.OutStreamer.SwitchSection(AP.getObjFileLowering().getDataSection());
|
2010-04-04 07:39:04 +00:00
|
|
|
EmitCamlGlobal(getModule(), AP, "frametable");
|
2009-01-16 06:53:46 +00:00
|
|
|
|
2010-05-24 12:24:11 +00:00
|
|
|
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);
|
|
|
|
|
2008-08-17 18:44:35 +00:00
|
|
|
for (iterator I = begin(), IE = end(); I != IE; ++I) {
|
|
|
|
GCFunctionInfo &FI = **I;
|
2009-01-16 06:53:46 +00:00
|
|
|
|
2008-08-17 18:44:35 +00:00
|
|
|
uint64_t FrameSize = FI.getFrameSize();
|
|
|
|
if (FrameSize >= 1<<16) {
|
2010-04-08 10:44:28 +00:00
|
|
|
// 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)) + ")");
|
2008-08-17 18:44:35 +00:00
|
|
|
}
|
2009-01-16 06:53:46 +00:00
|
|
|
|
2010-04-04 07:39:04 +00:00
|
|
|
AP.OutStreamer.AddComment("live roots for " +
|
|
|
|
Twine(FI.getFunction().getName()));
|
|
|
|
AP.OutStreamer.AddBlankLine();
|
2009-01-16 06:53:46 +00:00
|
|
|
|
2008-08-17 18:44:35 +00:00
|
|
|
for (GCFunctionInfo::iterator J = FI.begin(), JE = FI.end(); J != JE; ++J) {
|
|
|
|
size_t LiveCount = FI.live_size(J);
|
2008-08-09 03:48:46 +00:00
|
|
|
if (LiveCount >= 1<<16) {
|
2010-04-08 10:44:28 +00:00
|
|
|
// Very rude!
|
|
|
|
report_fatal_error("Function '" + FI.getFunction().getName() +
|
|
|
|
"' is too large for the ocaml GC! "
|
|
|
|
"Live root count "+Twine(LiveCount)+" >= 65536.");
|
2008-01-07 02:31:11 +00:00
|
|
|
}
|
2009-01-16 06:53:46 +00:00
|
|
|
|
2013-01-09 03:52:05 +00:00
|
|
|
AP.OutStreamer.EmitSymbolValue(J->Label, IntPtrSize);
|
2008-01-07 02:31:11 +00:00
|
|
|
AP.EmitInt16(FrameSize);
|
|
|
|
AP.EmitInt16(LiveCount);
|
2009-01-16 06:53:46 +00:00
|
|
|
|
2008-08-17 18:44:35 +00:00
|
|
|
for (GCFunctionInfo::live_iterator K = FI.live_begin(J),
|
|
|
|
KE = FI.live_end(J); K != KE; ++K) {
|
2010-05-24 12:24:11 +00:00
|
|
|
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);
|
2008-01-07 02:31:11 +00:00
|
|
|
}
|
2009-01-16 06:53:46 +00:00
|
|
|
|
2010-04-04 07:39:04 +00:00
|
|
|
AP.EmitAlignment(IntPtrSize == 4 ? 2 : 3);
|
2008-01-07 02:31:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|