ldc/gen/irstate.h

247 lines
7.1 KiB
C++
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

//===-- gen/irstate.h - Global codegen state --------------------*- C++ -*-===//
//
// LDC the LLVM D compiler
//
// This file is distributed under the BSD-style LDC license. See the LICENSE
// file for details.
//
//===----------------------------------------------------------------------===//
//
// This file contains the global state used and modified when generating the
// code (i.e. LLVM IR) for a given D module.
//
//===----------------------------------------------------------------------===//
#ifndef LDC_GEN_IRSTATE_H
#define LDC_GEN_IRSTATE_H
#include "aggregate.h"
#include "root.h"
#include "ir/irfunction.h"
#include "ir/iraggr.h"
#include "ir/irvar.h"
#include "gen/dibuilder.h"
#include <deque>
#include <list>
#include <set>
#include <sstream>
#include <vector>
#if LDC_LLVM_VER >= 305
#include "llvm/IR/CallSite.h"
#else
#include <map>
#include "llvm/Support/CallSite.h"
#endif
namespace llvm {
class LLVMContext;
class TargetMachine;
}
// global ir state for current module
struct IRState;
struct TargetABI;
extern IRState* gIR;
extern llvm::TargetMachine* gTargetMachine;
#if LDC_LLVM_VER >= 302
extern const llvm::DataLayout* gDataLayout;
#else
extern const llvm::TargetData* gDataLayout;
#endif
extern TargetABI* gABI;
class TypeFunction;
class TypeStruct;
class ClassDeclaration;
class FuncDeclaration;
class Module;
class TypeStruct;
struct BaseClass;
class AnonDeclaration;
struct IrModule;
// represents a scope
struct IRScope
{
llvm::BasicBlock* begin;
llvm::BasicBlock* end;
IRBuilder<> builder;
IRScope();
IRScope(llvm::BasicBlock* b, llvm::BasicBlock* e);
const IRScope& operator=(const IRScope& rhs);
};
struct IRBuilderHelper
{
IRState* state;
IRBuilder<>* operator->();
};
struct IRAsmStmt
{
IRAsmStmt()
: isBranchToLabel(NULL) {}
std::string code;
std::string out_c;
std::string in_c;
std::vector<LLValue*> out;
std::vector<LLValue*> in;
// if this is nonzero, it contains the target label
LabelDsymbol* isBranchToLabel;
};
struct IRAsmBlock
{
std::deque<IRAsmStmt*> s;
std::set<std::string> clobs;
size_t outputcount;
// stores the labels within the asm block
std::vector<Identifier*> internalLabels;
CompoundAsmStatement* asmBlock;
LLType* retty;
unsigned retn;
bool retemu; // emulate abi ret with a temporary
LLValue* (*retfixup)(IRBuilderHelper b, LLValue* orig); // Modifies retval
IRAsmBlock(CompoundAsmStatement* b)
: outputcount(0), asmBlock(b), retty(NULL), retn(0), retemu(false),
retfixup(NULL)
{}
};
// represents the module
struct IRState
{
IRState(const char *name, llvm::LLVMContext &context);
llvm::Module module;
llvm::LLVMContext& context() const { return module.getContext(); }
Module *dmodule;
LLStructType* mutexType;
LLStructType* moduleRefType;
// functions
typedef std::vector<IrFunction*> FunctionVector;
FunctionVector functions;
IrFunction* func();
llvm::Function* topfunc();
llvm::Instruction* topallocapoint();
// The function containing the D main() body, if any (not the actual main()
// implicitly emitted).
llvm::Function* mainFunc;
// basic block scopes
std::vector<IRScope> scopes;
IRScope& scope();
llvm::BasicBlock* scopebb();
llvm::BasicBlock* scopeend();
bool scopereturned();
// create a call or invoke, depending on the landing pad info
// the template function is defined further down in this file
template <typename T>
llvm::CallSite CreateCallOrInvoke(LLValue* Callee, const T& args, const char* Name="");
llvm::CallSite CreateCallOrInvoke(LLValue* Callee, const char* Name="");
llvm::CallSite CreateCallOrInvoke(LLValue* Callee, LLValue* Arg1, const char* Name="");
llvm::CallSite CreateCallOrInvoke2(LLValue* Callee, LLValue* Arg1, LLValue* Arg2, const char* Name="");
llvm::CallSite CreateCallOrInvoke3(LLValue* Callee, LLValue* Arg1, LLValue* Arg2, LLValue* Arg3, const char* Name="");
llvm::CallSite CreateCallOrInvoke4(LLValue* Callee, LLValue* Arg1, LLValue* Arg2, LLValue* Arg3, LLValue* Arg4, const char* Name="");
// this holds the array being indexed or sliced so $ will work
// might be a better way but it works. problem is I only get a
// VarDeclaration for __dollar, but I can't see how to get the
// array pointer from this :(
std::vector<DValue*> arrays;
// builder helper
IRBuilderHelper ir;
// debug info helper
ldc::DIBuilder DBuilder;
// for inline asm
IRAsmBlock* asmBlock;
std::ostringstream nakedAsm;
// Globals to pin in the llvm.used array to make sure they are not
// eliminated.
std::vector<LLConstant*> usedArray;
/// Whether to emit array bounds checking in the current function.
bool emitArrayBoundsChecks();
// Global variables bound to string literals. Once created such a
// variable is reused whenever the same string literal is
// referenced in the module. Caching them per module prevents the
// duplication of identical literals.
#if LDC_LLVM_VER >= 305
llvm::StringMap<llvm::GlobalVariable*> stringLiteral1ByteCache;
llvm::StringMap<llvm::GlobalVariable*> stringLiteral2ByteCache;
llvm::StringMap<llvm::GlobalVariable*> stringLiteral4ByteCache;
#else
std::map<llvm::StringRef, llvm::GlobalVariable*> stringLiteral1ByteCache;
std::map<llvm::StringRef, llvm::GlobalVariable*> stringLiteral2ByteCache;
std::map<llvm::StringRef, llvm::GlobalVariable*> stringLiteral4ByteCache;
#endif
#if LDC_LLVM_VER >= 303
/// Vector of options passed to the linker as metadata in object file.
#if LDC_LLVM_VER >= 306
llvm::SmallVector<llvm::Metadata *, 5> LinkerMetadataArgs;
#else
llvm::SmallVector<llvm::Value *, 5> LinkerMetadataArgs;
#endif
#endif
};
template <typename T>
llvm::CallSite IRState::CreateCallOrInvoke(LLValue* Callee, const T &args, const char* Name)
{
FuncGen& funcGen = *func()->gen;
LLFunction* fn = llvm::dyn_cast<LLFunction>(Callee);
const bool hasTemporaries = funcGen.hasTemporariesToDestruct();
// intrinsics don't support invoking and 'nounwind' functions don't need it.
const bool doesNotThrow = (fn && (fn->isIntrinsic() || fn->doesNotThrow()));
if (doesNotThrow || (!hasTemporaries && funcGen.landingPad == NULL))
{
llvm::CallInst* call = ir->CreateCall(Callee, args, Name);
if (fn)
call->setAttributes(fn->getAttributes());
return call;
}
if (hasTemporaries)
funcGen.prepareToDestructAllTemporariesOnThrow(this);
llvm::BasicBlock* landingPad = funcGen.landingPad;
llvm::BasicBlock* postinvoke = llvm::BasicBlock::Create(context(), "postinvoke", topfunc(), landingPad);
llvm::InvokeInst* invoke = ir->CreateInvoke(Callee, postinvoke, landingPad, args, Name);
if (fn)
invoke->setAttributes(fn->getAttributes());
if (hasTemporaries)
funcGen.landingPadInfo.pop();
scope() = IRScope(postinvoke, landingPad);
return invoke;
}
void codegenFunction(Statement *s, IRState *irs);
void Statement_toIR(Statement *s, IRState *irs);
#endif // LDC_GEN_IRSTATE_H