blob: 8e89288a4eb2f5781c20697cdb3ea9f8e18fc2e7 [file] [log] [blame]
//===-- go-llvm-materialize.cpp - Llvm_backend materalize* methods -------===//
//
// Copyright 2018 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
//
//===----------------------------------------------------------------------===//
//
// Llvm_backend methods relating to materialization of llvm values.
//
//===----------------------------------------------------------------------===//
#include "go-llvm.h"
#include "go-llvm-builtins.h"
#include "go-c.h"
#include "go-system.h"
#include "go-llvm-cabi-oracle.h"
#include "go-llvm-irbuilders.h"
#include "gogo.h"
#include "llvm/IR/Constants.h"
#include "llvm/IR/DataLayout.h"
#include "llvm/IR/DerivedTypes.h"
#include "llvm/IR/DIBuilder.h"
#include "llvm/IR/GlobalValue.h"
#include "llvm/IR/Instructions.h"
#include "llvm/IR/Intrinsics.h"
#include "llvm/IR/IRBuilder.h"
#include "llvm/IR/Module.h"
#include "llvm/IR/Type.h"
#include "llvm/IR/Value.h"
#include "llvm/Support/CommandLine.h"
#include "llvm/IR/InlineAsm.h"
static llvm::cl::opt<bool> DisableInlineGetg("disable-inline-getg",
llvm::cl::desc("Disable inlining getg"),
llvm::cl::init(false),
llvm::cl::Hidden);
Bexpression *Llvm_backend::materializeIndirect(Bexpression *indExpr, bool isLHS)
{
Location location = indExpr->location();
Btype *btype = indExpr->btype();
std::vector<Bexpression *> iexprs =
nbuilder_.extractChildenAndDestroy(indExpr);
assert(iexprs.size() == 1);
Bexpression *expr = iexprs[0];
// Handle cases such as
//
// *(*sometype)(unsafe.Pointer(uintptr(<constant>))) = ...
//
// where we have a LHS expression intended to cause a crash or fault.
if (isLHS && !expr->varExprPending()) {
Bexpression *rval = nbuilder_.mkDeref(btype, expr->value(), expr,
location);
return rval;
}
const VarContext *vc = nullptr;
if (expr->varExprPending()) {
vc = &expr->varContext();
// handle *&x
if (vc->addrLevel() != 0) {
Bexpression *rval = nbuilder_.mkDeref(btype, expr->value(), expr,
location);
rval->setVarExprPending(vc->lvalue(), vc->addrLevel() - 1);
return rval;
}
}
std::string tag(expr->tag().size() == 0 ? "deref" : expr->tag());
Bexpression *rval = genLoad(expr, btype, location, tag);
if (vc) {
if (rval->varExprPending())
rval->resetVarExprContext();
rval->setVarExprPending(expr->varContext());
}
return rval;
}
Bexpression *Llvm_backend::materializeAddress(Bexpression *addrExpr)
{
Location location = addrExpr->location();
std::vector<Bexpression *> aexprs =
nbuilder_.extractChildenAndDestroy(addrExpr);
Bexpression *bexpr = aexprs[0];
assert(aexprs.size() == 1);
assert(bexpr->value());
// Gofrontend tends to take the address of things that are already
// pointer-like to begin with (for example, C strings and and
// arrays). This presents wrinkles here, since an array type
// in LLVM is already effectively a pointer (you can feed it
// directly into a GEP as opposed to having to take the address of
// it first). Bypass the effects of the address operator if
// this is the case. This is hacky, maybe I can come up with a
// better solution for this issue(?).
if (llvm::isa<llvm::ConstantArray>(bexpr->value()))
return bexpr;
if (bexpr->value()->getType() == stringType()->type() &&
bexpr->isConstant())
return bexpr;
// If the value we're trying to take the address of is a composite
// constant, we have to spill it to memory here in order for us to
// take its address.
llvm::Value *val = bexpr->value();
if (bexpr->isConstant()) {
llvm::Constant *cval = llvm::cast<llvm::Constant>(val);
Bvariable *cv = genVarForConstant(cval, bexpr->btype());
val = cv->value();
}
// When using non-integral pointers, the Go pointer types (Btype)
// are in address space 1. Local variables (allocas) are always
// in address space 0. We need to insert a cast if we are taking
// the address of a local variable.
llvm::Type *valtyp = val->getType();
assert(valtyp->isPointerTy());
if (valtyp->getPointerAddressSpace() != addressSpace_) {
llvm::Type *typ =
llvm::PointerType::get(valtyp->getPointerElementType(),
addressSpace_);
val = new llvm::AddrSpaceCastInst(val, typ, "ascast");
}
// Create new expression with proper type.
Btype *pt = pointer_type(bexpr->btype());
Bexpression *rval = nbuilder_.mkAddress(pt, val, bexpr, location);
std::string adtag(bexpr->tag());
adtag += ".ad";
rval->setTag(adtag);
const VarContext &vc = bexpr->varContext();
rval->setVarExprPending(vc.lvalue(), vc.addrLevel() + 1);
return rval;
}
Bexpression *Llvm_backend::materializeConversion(Bexpression *convExpr)
{
Location location = convExpr->location();
Btype *type = convExpr->btype();
std::vector<Bexpression *> iexprs =
nbuilder_.extractChildenAndDestroy(convExpr);
assert(iexprs.size() == 1);
Bexpression *expr = iexprs[0];
// For composite-init-pending values, materialize a variable now.
if (expr->compositeInitPending()) {
assert(!expr->varExprPending());
expr = resolveCompositeInit(expr, nullptr);
}
llvm::Value *val = expr->value();
assert(val);
llvm::Type *valType = val->getType();
llvm::Type *toType = type->type();
// In the varexpr pending case, decide what to do depending on whether
// the var is in an lvalue or rvalue context. For something like
//
// var y int32
// z = int64(y)
//
// we want to force the load of "y" before converting to int64. For
// an lvalue context, the conversion will be applied to the pointed-to-type
// as well as the value type.
if (expr->varExprPending()) {
bool lvalue = expr->varContext().lvalue();
if (!lvalue) {
expr = resolveVarContext(expr);
val = expr->value();
valType = val->getType();
}
if (lvalue || useCopyForLoadStore(type->type())) {
llvm::Type *et = expr->btype()->type();
if (valType->isPointerTy() &&
valType->getPointerElementType() == et)
toType = llvm::PointerType::get(toType, addressSpace_);
}
}
// If we're converting between two different Btypes that have the
// same underlying LLVM type, then we can create a new Bexpression
// for the conversion but not do anything else.
if (toType == valType) {
Bexpression *rval =
nbuilder_.mkConversion(type, expr->value(), expr, location);
if (expr->varExprPending())
rval->setVarExprPending(expr->varContext());
return rval;
}
// If we're applying a conversion to an aggregate constant, call a helper to
// see if we can create a new (but equivalent) constant value using the target
// type. If this works, we're effectively done. If the conversion doesn't
// succeed, materialize a variable containing the constant and apply the
// conversion to the variable's type (which will be a pointer to the type of
// the constant), then flag the result as "load pending".
bool pending = false;
if (expr->isConstant() && val->getType()->isAggregateType()) {
llvm::Constant *cval = llvm::cast<llvm::Constant>(val);
assert(valType == cval->getType());
llvm::Value *convertedValue = genConvertedConstant(cval, toType);
if (convertedValue != nullptr) {
// We have a new value of the correct type. Wrap a conversion
// expr around it and return.
return nbuilder_.mkConversion(type, convertedValue, expr, location);
}
// materialize constant into variable.
Bvariable *cv = genVarForConstant(cval, expr->btype());
val = cv->value();
valType = val->getType();
toType = llvm::PointerType::get(toType, addressSpace_);
pending = true;
}
Bexpression *rval = nullptr;
LIRBuilder builder(context_, llvm::ConstantFolder());
// Pointer type to pointer-sized-integer type. Comes up when
// converting function pointer to function descriptor (during
// creation of function descriptor vals) or constant array to
// uintptr (as part of GC symbol initializer creation), and in other
// places in FE-generated code (ex: array index checks).
if (valType->isPointerTy() && toType == llvmIntegerType()) {
if (val->getType()->getPointerAddressSpace() != 0) {
// We are using non-integral pointer. Cast to address space 0
// before casting to int.
llvm::Type *et = val->getType()->getPointerElementType();
llvm::Type *pt = llvm::PointerType::get(et, 0);
std::string tname(namegen("ascast"));
val = builder.CreateAddrSpaceCast(val, pt, tname);
expr = nbuilder_.mkConversion(type, val, expr, location);
}
std::string tname(namegen("pticast"));
llvm::Value *pticast = builder.CreatePtrToInt(val, toType, tname);
rval = nbuilder_.mkConversion(type, pticast, expr, location);
}
// Pointer-sized-integer type to pointer type. This comes up
// in type hash/compare functions.
if (toType->isPointerTy() && valType == llvmIntegerType()) {
llvm::Type *pt = toType;
if (toType->getPointerAddressSpace() != 0) {
// We are using non-integral pointer. Cast to address space 0
// first.
llvm::Type *et = toType->getPointerElementType();
pt = llvm::PointerType::get(et, 0);
}
std::string tname(namegen("itpcast"));
llvm::Value *itpcast = builder.CreateIntToPtr(val, pt, tname);
rval = nbuilder_.mkConversion(type, itpcast, expr, location);
if (pt != toType) {
std::string tname(namegen("ascast"));
llvm::Value *ascast = builder.CreateAddrSpaceCast(itpcast, toType, tname);
rval = nbuilder_.mkConversion(type, ascast, rval, location);
}
}
// For pointer conversions (ex: *int32 => *int64) create an
// appropriate bitcast.
if (valType->isPointerTy() && toType->isPointerTy()) {
std::string tag(namegen("cast"));
llvm::Value *bitcast = builder.CreatePointerBitCastOrAddrSpaceCast(val, toType, tag);
rval = nbuilder_.mkConversion(type, bitcast, expr, location);
}
// Integer-to-integer conversions
// FIXME: the type of the the operand could be AuxT, which we use
// to wrap an LLVM type for intrinsics. Assume it is signed for now.
// If this turns to be an issue, we should define the correct Btype
// for intrinsics.
if (valType->isIntegerTy() && toType->isIntegerTy()) {
llvm::IntegerType *valIntTyp =
llvm::cast<llvm::IntegerType>(valType);
llvm::IntegerType *toIntTyp =
llvm::cast<llvm::IntegerType>(toType);
unsigned valbits = valIntTyp->getBitWidth();
unsigned tobits = toIntTyp->getBitWidth();
llvm::Value *conv = nullptr;
if (tobits > valbits) {
if (expr->btype()->type() == llvmBoolType() ||
(expr->btype()->castToBIntegerType() &&
expr->btype()->castToBIntegerType()->isUnsigned()))
conv = builder.CreateZExt(val, toType, namegen("zext"));
else
conv = builder.CreateSExt(val, toType, namegen("sext"));
} else {
conv = builder.CreateTrunc(val, toType, namegen("trunc"));
}
rval = nbuilder_.mkConversion(type, conv, expr, location);
}
// Float -> float conversions
if (toType->isFloatingPointTy() && valType->isFloatingPointTy()) {
llvm::Value *conv = nullptr;
if (toType == llvmFloatType() && valType == llvmDoubleType())
conv = builder.CreateFPTrunc(val, toType, namegen("fptrunc"));
else if (toType == llvmDoubleType() && valType == llvmFloatType())
conv = builder.CreateFPExt(val, toType, namegen("fpext"));
else
assert(0 && "unexpected float type");
rval = nbuilder_.mkConversion(type, conv, expr, location);
}
// Float -> integer conversions
if (toType->isIntegerTy() && valType->isFloatingPointTy()) {
llvm::Value *conv = nullptr;
if (type->castToBIntegerType()->isUnsigned())
conv = builder.CreateFPToUI(val, toType, namegen("ftoui"));
else
conv = builder.CreateFPToSI(val, toType, namegen("ftosi"));
rval = nbuilder_.mkConversion(type, conv, expr, location);
}
// Integer -> float conversions
if (toType->isFloatingPointTy() && valType->isIntegerTy()) {
llvm::Value *conv = nullptr;
if (expr->btype()->castToBIntegerType() &&
expr->btype()->castToBIntegerType()->isUnsigned())
conv = builder.CreateUIToFP(val, toType, namegen("uitof"));
else
conv = builder.CreateSIToFP(val, toType, namegen("sitof"));
rval = nbuilder_.mkConversion(type, conv, expr, location);
}
if (!rval)
// This case not handled.
assert(false && "this flavor of conversion not handled");
// Propagate pending var context if we didn't resolve it here.
// This may happen for composite values.
if (expr->varExprPending())
rval->setVarExprPending(expr->varContext());
else if (pending)
rval->setVarExprPending(false, 0);
return rval;
}
llvm::Value *Llvm_backend::makePointerOffsetGEP(llvm::PointerType *llpt,
llvm::Value *idxval,
llvm::Value *sptr)
{
LIRBuilder builder(context_, llvm::ConstantFolder());
llvm::SmallVector<llvm::Value *, 1> elems(1);
elems[0] = idxval;
llvm::Value *val = builder.CreateGEP(llpt->getElementType(), sptr, elems, namegen("ptroff"));
return val;
}
llvm::Value *Llvm_backend::makeArrayIndexGEP(llvm::ArrayType *llat,
llvm::Value *idxval,
llvm::Value *sptr)
{
LIRBuilder builder(context_, llvm::ConstantFolder());
llvm::SmallVector<llvm::Value *, 2> elems(2);
elems[0] = llvm::ConstantInt::get(llvmInt32Type(), 0);
elems[1] = idxval;
llvm::Value *val = builder.CreateGEP(llat, sptr, elems, namegen("index"));
return val;
}
llvm::Value *Llvm_backend::makeFieldGEP(unsigned fieldIndex,
llvm::Value *sptr)
{
assert(sptr->getType()->isPointerTy());
llvm::PointerType *srcTyp = llvm::cast<llvm::PointerType>(sptr->getType());
llvm::StructType *llst = llvm::cast<llvm::StructType>(srcTyp->getElementType());
LIRBuilder builder(context_, llvm::ConstantFolder());
assert(fieldIndex < llst->getNumElements());
std::string tag(namegen("field"));
llvm::Value *val =
builder.CreateConstInBoundsGEP2_32(llst, sptr, 0, fieldIndex, tag);
return val;
}
Bexpression *Llvm_backend::materializeStructField(Bexpression *fieldExpr)
{
Location location = fieldExpr->location();
const std::string ftag(fieldExpr->tag());
unsigned index = fieldExpr->fieldIndex();
std::vector<Bexpression *> fexprs =
nbuilder_.extractChildenAndDestroy(fieldExpr);
assert(fexprs.size() == 1);
Bexpression *bstruct = fexprs[0];
if (bstruct->compositeInitPending())
bstruct = resolveCompositeInit(bstruct, nullptr);
// Construct an appropriate GEP
llvm::Type *llt = bstruct->btype()->type();
assert(llt->isStructTy());
llvm::Value *sval = bstruct->value();
llvm::Value *fval;
if (bstruct->isConstant())
fval = llvm::cast<llvm::Constant>(sval)->getAggregateElement(index);
else
fval = makeFieldGEP(index, sval);
Btype *bft = elementTypeByIndex(bstruct->btype(), index);
// Wrap result in a Bexpression
Bexpression *rval = nbuilder_.mkStructField(bft, fval, bstruct,
index, location);
if (bstruct->varExprPending())
rval->setVarExprPending(bstruct->varContext());
std::string tag(bstruct->tag());
tag += (ftag.empty() ? ".field" : ftag);
rval->setTag(tag);
// We're done
return rval;
}
Bexpression *Llvm_backend::materializeCompound(Bexpression *comExpr)
{
Location location = comExpr->location();
std::vector<Bnode *> kids = nbuilder_.extractChildNodesAndDestroy(comExpr);
assert(kids.size() == 2);
Bstatement *bstat = kids[0]->castToBstatement();
assert(bstat);
Bexpression *bexpr = kids[1]->castToBexpression();
assert(bexpr);
bexpr = materialize(bexpr);
// Compound expressions can be used to produce lvalues, so we don't
// want to call resolve() on bexpr here.
// But we do want to resolve composite init.
if (bexpr->compositeInitPending())
bexpr = resolveCompositeInit(bexpr, nullptr);
Bexpression *rval = nbuilder_.mkCompound(bstat, bexpr, bexpr->value(),
location);
if (bexpr->varExprPending())
rval->setVarExprPending(bexpr->varContext());
return rval;
}
Bexpression *Llvm_backend::materializeConditional(Bexpression *condExpr)
{
Bfunction *function = condExpr->getFunction();
Location location = condExpr->location();
Btype *btype = condExpr->btype();
std::vector<Bexpression *> cexprs =
nbuilder_.extractChildenAndDestroy(condExpr);
assert(cexprs.size() == 2 || cexprs.size() == 3);
Bexpression *condition = cexprs[0];
Bexpression *then_expr = cexprs[1];
Bexpression *else_expr = (cexprs.size() == 3 ? cexprs[2] : nullptr);
condition = resolveVarContext(condition);
then_expr = resolve(then_expr);
if (else_expr)
else_expr = resolve(else_expr);
std::vector<Bvariable *> novars;
Bblock *thenBlock = nbuilder_.mkBlock(function, novars, location);
Bblock *elseBlock = nullptr;
Bvariable *tempv = nullptr;
// FIXME: add lifetime intrinsics for temp var below.
Bstatement *thenStmt = nullptr;
if (!btype || then_expr->btype() == void_type() || btype == void_type())
thenStmt = expression_statement(function, then_expr);
else
tempv = temporary_variable(function, nullptr,
btype, then_expr, false,
location, &thenStmt);
nbuilder_.addStatementToBlock(thenBlock, thenStmt);
if (else_expr) {
Bstatement *elseStmt = nullptr;
elseBlock = nbuilder_.mkBlock(function, novars, location);
if (!btype || btype == void_type() || else_expr->btype() == void_type()) {
elseStmt = expression_statement(function, else_expr);
} else {
// Capture "else_expr" into temporary. Type needs to agree with
// then_expr if then_expr had non-void type.
if (!tempv) {
tempv = temporary_variable(function, nullptr,
btype, else_expr, false,
location, &elseStmt);
} else {
// Ideally it would be nice to assert that the types are
// identical for if_expr and else_expr, but particularly for
// pointer types we need to allow for some disagreement (ex:
// nil_pointer_expression, which is untyped/polymorphic).
// Assume that the type checking in the call to
// assignment_statement will catch any problems.
Bexpression *varExpr = var_expression(tempv, location);
elseStmt = assignment_statement(function, varExpr, else_expr, location);
}
}
nbuilder_.addStatementToBlock(elseBlock, elseStmt);
}
// Wrap up and return the result
Bstatement *ifStmt = if_statement(function, condition,
thenBlock, elseBlock, location);
Bexpression *rval = (tempv ?
var_expression(tempv, location) :
nbuilder_.mkVoidValue(void_type()));
Bexpression *result =
materialize(compound_expression(ifStmt, rval, location));
return result;
}
Bexpression *Llvm_backend::materializeUnary(Bexpression *unExpr)
{
Operator op = unExpr->op();
Location location = unExpr->location();
std::vector<Bexpression *> uexprs =
nbuilder_.extractChildenAndDestroy(unExpr);
assert(uexprs.size() == 1);
Bexpression *expr = uexprs[0];
expr = resolveVarContext(expr);
Btype *bt = expr->btype();
switch (op) {
case OPERATOR_MINUS: {
assert(false && "should have been expanded away");
break;
}
case OPERATOR_NOT: {
LIRBuilder builder(context_, llvm::ConstantFolder());
assert(isBooleanType(bt));
// FIXME: is this additional compare-to-zero needed? Or can we be certain
// that the value in question has a single bit set?
Bexpression *bzero = zero_expression(bt);
llvm::Value *cmp =
builder.CreateICmpNE(expr->value(), bzero->value(), namegen("icmp"));
Btype *lbt = makeAuxType(llvmBoolType());
Bexpression *cmpex =
nbuilder_.mkBinaryOp(OPERATOR_EQEQ, lbt, cmp, bzero, expr, location);
llvm::Constant *one = llvm::ConstantInt::get(llvmBoolType(), 1);
llvm::Value *xorex = builder.CreateXor(cmp, one, namegen("xor"));
Bexpression *notex = nbuilder_.mkUnaryOp(op, lbt, xorex, cmpex, location);
Bexpression *tobool = lateConvert(bool_type(), notex, location);
return tobool;
}
case OPERATOR_XOR: {
// ^x bitwise complement is m ^ x with m = "all bits set to 1"
// for unsigned x and m = -1 for signed x
assert(bt->type()->isIntegerTy());
LIRBuilder builder(context_, llvm::ConstantFolder());
llvm::Value *onesval = llvm::Constant::getAllOnesValue(bt->type());
llvm::Value *xorExpr = builder.CreateXor(expr->value(), onesval,
namegen("xor"));
Bexpression *rval = nbuilder_.mkUnaryOp(op, bt, xorExpr, expr, location);
return rval;
break;
}
default:
assert(false && "unexpected unary opcode");
}
return nullptr;
}
static llvm::CmpInst::Predicate compare_op_to_pred(Operator op,
llvm::Type *typ,
bool isUnsigned)
{
bool isFloat = typ->isFloatingPointTy();
if (isFloat) {
switch (op) {
case OPERATOR_EQEQ:
return llvm::CmpInst::Predicate::FCMP_OEQ;
case OPERATOR_NOTEQ:
return llvm::CmpInst::Predicate::FCMP_UNE;
case OPERATOR_LT:
return llvm::CmpInst::Predicate::FCMP_OLT;
case OPERATOR_LE:
return llvm::CmpInst::Predicate::FCMP_OLE;
case OPERATOR_GT:
return llvm::CmpInst::Predicate::FCMP_OGT;
case OPERATOR_GE:
return llvm::CmpInst::Predicate::FCMP_OGE;
default:
break;
}
} else {
switch (op) {
case OPERATOR_EQEQ:
return llvm::CmpInst::Predicate::ICMP_EQ;
case OPERATOR_NOTEQ:
return llvm::CmpInst::Predicate::ICMP_NE;
case OPERATOR_LT:
return (isUnsigned ? llvm::CmpInst::Predicate::ICMP_ULT
: llvm::CmpInst::Predicate::ICMP_SLT);
case OPERATOR_LE:
return (isUnsigned ? llvm::CmpInst::Predicate::ICMP_ULE
: llvm::CmpInst::Predicate::ICMP_SLE);
case OPERATOR_GT:
return (isUnsigned ? llvm::CmpInst::Predicate::ICMP_UGT
: llvm::CmpInst::Predicate::ICMP_SGT);
case OPERATOR_GE:
return (isUnsigned ? llvm::CmpInst::Predicate::ICMP_UGE
: llvm::CmpInst::Predicate::ICMP_SGE);
default:
break;
}
}
assert(false);
return llvm::CmpInst::BAD_ICMP_PREDICATE;
}
std::pair<llvm::Value *, llvm::Value *>
Llvm_backend::convertForBinary(Operator op,
Bexpression *left,
Bexpression *right)
{
llvm::Value *leftVal = left->value();
llvm::Value *rightVal = right->value();
std::pair<llvm::Value *, llvm::Value *> rval =
std::make_pair(leftVal, rightVal);
llvm::Type *leftType = leftVal->getType();
llvm::Type *rightType = rightVal->getType();
if (leftType == rightType)
return rval;
// Case 1: nil op X
if (llvm::isa<llvm::ConstantPointerNull>(leftVal) &&
rightType->isPointerTy()) {
BexprLIRBuilder builder(context_, left);
std::string tag(namegen("cast"));
llvm::Value *bitcast = builder.CreateBitCast(leftVal, rightType, tag);
rval.first = bitcast;
return rval;
}
// Case 2: X op nil
if (llvm::isa<llvm::ConstantPointerNull>(rightVal) &&
leftType->isPointerTy()) {
BexprLIRBuilder builder(context_, right);
std::string tag(namegen("cast"));
llvm::Value *bitcast = builder.CreateBitCast(rightVal, leftType, tag);
rval.second = bitcast;
return rval;
}
// Case 3: shift with different sized operands (ex: int64(v) << uint8(3)).
// Promote or demote shift amount operand to match width of left operand.
if ((op == OPERATOR_LSHIFT || op == OPERATOR_RSHIFT) &&
leftType != rightType) {
BexprLIRBuilder builder(context_, right);
llvm::IntegerType *leftITyp = llvm::cast<llvm::IntegerType>(leftType);
llvm::IntegerType *rightITyp = llvm::cast<llvm::IntegerType>(rightType);
llvm::Value *conv = nullptr;
if (leftITyp->getBitWidth() > rightITyp->getBitWidth())
conv = builder.CreateZExt(rightVal, leftType, namegen("zext"));
else
conv = builder.CreateTrunc(rightVal, leftType, namegen("trunc"));
rval.second = conv;
return rval;
}
// Case 4: pointer type comparison
// We check that if both are pointer types and pointing to the same type,
// insert a cast (it doesn't matter we cast which one). This mostly needed
// for circular pointer types (ex: type T *T; var p, q T; p == &q), where
// the two sides have semantically identical types but with different
// representations (in this case, T vs. *T).
if (leftType->isPointerTy() && rightType->isPointerTy()) {
BPointerType *lbpt = left->btype()->castToBPointerType();
BPointerType *rbpt = right->btype()->castToBPointerType();
if (lbpt->toType()->type() == rbpt->toType()->type()) {
BexprLIRBuilder builder(context_, right);
std::string tag(namegen("cast"));
llvm::Value *bitcast = builder.CreateBitCast(rightVal, leftType, tag);
rval.second = bitcast;
return rval;
}
}
return rval;
}
Bexpression *Llvm_backend::materializeBinary(Bexpression *binExpr)
{
Operator op = binExpr->op();
Location location = binExpr->location();
std::vector<Bexpression *> bexprs =
nbuilder_.extractChildenAndDestroy(binExpr);
assert(bexprs.size() == 2);
Bexpression *left = bexprs[0];
Bexpression *right = bexprs[1];
Btype *bltype = left->btype();
Btype *brtype = right->btype();
left = resolveVarContext(left);
right = resolveVarContext(right);
assert(left->value() && right->value());
std::pair<llvm::Value *, llvm::Value *> converted =
convertForBinary(op, left, right);
llvm::Value *leftVal = converted.first;
llvm::Value *rightVal = converted.second;
llvm::Type *ltype = leftVal->getType();
llvm::Type *rtype = rightVal->getType();
assert(ltype == rtype);
BIntegerType *blitype = bltype->castToBIntegerType();
BIntegerType *britype = brtype->castToBIntegerType();
assert((blitype == nullptr) == (britype == nullptr));
bool isUnsigned = false;
if (blitype) {
// As of Go 1.13, shift amount is allowed to be a signed integer.
// Note that the front end emits tests to guard against negative
// shift amounts.
assert(op == OPERATOR_LSHIFT || op == OPERATOR_RSHIFT ||
blitype->isUnsigned() == britype->isUnsigned());
isUnsigned = blitype->isUnsigned();
}
LIRBuilder builder(context_, llvm::ConstantFolder());
llvm::Value *val = nullptr;
switch (op) {
case OPERATOR_EQEQ:
case OPERATOR_NOTEQ:
case OPERATOR_LT:
case OPERATOR_LE:
case OPERATOR_GT:
case OPERATOR_GE: {
llvm::CmpInst::Predicate pred = compare_op_to_pred(op, ltype, isUnsigned);
if (ltype->isFloatingPointTy())
val = builder.CreateFCmp(pred, leftVal, rightVal, namegen("fcmp"));
else
val = builder.CreateICmp(pred, leftVal, rightVal, namegen("icmp"));
Btype *bcmpt = makeAuxType(llvmBoolType());
// gen compare...
Bexpression *cmpex =
nbuilder_.mkBinaryOp(op, bcmpt, val, left, right, location);
// ... widen to go boolean type
return lateConvert(bool_type(), cmpex, location);
}
case OPERATOR_MINUS: {
if (ltype->isFloatingPointTy())
val = builder.CreateFSub(leftVal, rightVal, namegen("fsub"));
else
val = builder.CreateSub(leftVal, rightVal, namegen("sub"));
break;
}
case OPERATOR_PLUS: {
if (ltype->isFloatingPointTy())
val = builder.CreateFAdd(leftVal, rightVal, namegen("fadd"));
else
val = builder.CreateAdd(leftVal, rightVal, namegen("add"));
break;
}
case OPERATOR_MULT: {
if (ltype->isFloatingPointTy())
val = builder.CreateFMul(leftVal, rightVal, namegen("fmul"));
else
val = builder.CreateMul(leftVal, rightVal, namegen("mul"));
break;
}
case OPERATOR_MOD: {
assert(! ltype->isFloatingPointTy());
if (isUnsigned)
val = builder.CreateURem(leftVal, rightVal, namegen("mod"));
else
val = builder.CreateSRem(leftVal, rightVal, namegen("mod"));
break;
}
case OPERATOR_DIV: {
if (ltype->isFloatingPointTy())
val = builder.CreateFDiv(leftVal, rightVal, namegen("fdiv"));
else if (isUnsigned)
val = builder.CreateUDiv(leftVal, rightVal, namegen("div"));
else
val = builder.CreateSDiv(leftVal, rightVal, namegen("div"));
break;
}
case OPERATOR_OROR:
// Note that the FE will have already expanded out || in a control
// flow context (short circuiting)
// fall through...
case OPERATOR_OR: {
assert(!ltype->isFloatingPointTy());
val = builder.CreateOr(leftVal, rightVal, namegen("ior"));
break;
}
case OPERATOR_BITCLEAR:
// Note that the FE already inserted a complement op to RHS. So
// this is effectively an AND expression.
// fall through...
case OPERATOR_ANDAND:
// Note that the FE will have already expanded out && in a control
// flow context (short circuiting).
// fall through...
case OPERATOR_AND: {
assert(!ltype->isFloatingPointTy());
val = builder.CreateAnd(leftVal, rightVal, namegen("iand"));
break;
}
case OPERATOR_XOR: {
assert(!ltype->isFloatingPointTy() && !rtype->isFloatingPointTy());
val = builder.CreateXor(leftVal, rightVal, namegen("xor"));
break;
}
case OPERATOR_LSHIFT: {
// Note that the FE already inserted conditionals for checking
// large shift amounts. So this can simply lower to a shift
// instruction.
assert(!ltype->isFloatingPointTy() && !rtype->isFloatingPointTy());
val = builder.CreateShl(leftVal, rightVal, namegen("shl"));
break;
}
case OPERATOR_RSHIFT: {
// Note that the FE already inserted conditionals for checking
// large shift amounts. So this can simply lower to a shift
// instruction.
assert(!ltype->isFloatingPointTy() && !rtype->isFloatingPointTy());
if (isUnsigned)
val = builder.CreateLShr(leftVal, rightVal, namegen("shr"));
else
val = builder.CreateAShr(leftVal, rightVal, namegen("shr"));
break;
}
default:
std::cerr << "Op " << op << " unhandled\n";
assert(false);
}
return nbuilder_.mkBinaryOp(op, bltype, val, left, right, location);
}
Bexpression *Llvm_backend::materializeComposite(Bexpression *comExpr)
{
Location location = comExpr->location();
Btype *btype = comExpr->btype();
const std::vector<unsigned long> *indexes = nbuilder_.getIndices(comExpr);
std::vector<Bexpression *> vals =
nbuilder_.extractChildenAndDestroy(comExpr);
llvm::Type *llt = btype->type();
unsigned numElements = 0;
assert(llt->isStructTy() || llt->isArrayTy());
llvm::Type *llct = nullptr;
if (llt->isStructTy()) {
llvm::StructType *llst = llvm::cast<llvm::StructType>(llt);
numElements = llst->getNumElements();
assert(vals.size() == numElements);
llct = llst;
} else {
llvm::ArrayType *llat = llvm::cast<llvm::ArrayType>(llt);
numElements = llat->getNumElements();
llct = llat;
}
// Constant values?
bool isConstant = valuesAreConstant(vals);
if (isConstant)
return makeConstCompositeExpr(btype, llct, numElements,
indexes, vals, location);
else
return makeDelayedCompositeExpr(btype, llct, numElements,
indexes, vals, location);
}
Bexpression *
Llvm_backend::makeDelayedCompositeExpr(Btype *btype,
llvm::Type *llct,
unsigned numElements,
const std::vector<unsigned long> *indexes,
const std::vector<Bexpression *> &vals,
Location location)
{
std::vector<Bexpression *> init_vals(numElements);
if (indexes) {
unsigned long nvals = vals.size();
unsigned long nindxs = indexes->size();
std::set<unsigned long> touched;
for (unsigned ii = 0; ii < nindxs; ++ii) {
auto idx = (*indexes)[ii];
if (numElements != nvals)
touched.insert(idx);
init_vals[idx] = vals[ii];
}
if (numElements != nvals) {
for (unsigned long ii = 0; ii < numElements; ++ii) {
Btype *bElemTyp = elementTypeByIndex(btype, ii);
if (touched.find(ii) == touched.end())
init_vals[ii] = zero_expression(bElemTyp);
}
}
} else {
init_vals = vals;
}
// Here the NULL value signals that we want to delay full instantiation
// of this constant expression until we can identify the storage for it.
llvm::Value *nilval = nullptr;
Binstructions noInstructions;
Bexpression *ccon = nbuilder_.mkComposite(btype, nilval, init_vals,
noInstructions, location);
return ccon;
}
Bexpression *
Llvm_backend::makeConstCompositeExpr(Btype *btype,
llvm::Type *llct,
unsigned numElements,
const std::vector<unsigned long> *indexes,
const std::vector<Bexpression *> &vals,
Location location)
{
llvm::Value *scon;
// If all elements are zero, just create a zero value for the
// aggregate type. No need to create LLVM Value for each element.
bool allZero = true;
for (auto v : vals) {
llvm::Constant *con = llvm::cast<llvm::Constant>(v->value());
if (!con->isNullValue()) {
allZero = false;
break;
}
}
if (allZero)
scon = llvm::ConstantAggregateZero::get(llct);
else {
llvm::SmallVector<llvm::Constant *, 64> llvals(numElements);
unsigned long nvals = vals.size();
if (indexes) {
std::set<unsigned long> touched;
unsigned long nindxs = indexes->size();
for (unsigned ii = 0; ii < nindxs; ++ii) {
auto idx = (*indexes)[ii];
if (numElements != nvals)
touched.insert(idx);
Bexpression *bex = vals[ii];
llvm::Constant *con = llvm::cast<llvm::Constant>(bex->value());
llvm::Type *elt = TypeManager::getLlvmTypeAtIndex(llct, ii);
if (elt != con->getType()) {
con = genConvertedConstant(con, elt);
assert(con != nullptr);
}
llvals[idx] = con;
}
if (numElements != nvals) {
for (unsigned long ii = 0; ii < numElements; ++ii) {
if (touched.find(ii) == touched.end()) {
llvm::Type *elt = TypeManager::getLlvmTypeAtIndex(llct, ii);
llvals[ii] = llvm::Constant::getNullValue(elt);
}
}
}
} else {
for (unsigned long ii = 0; ii < numElements; ++ii) {
llvm::Constant *con = llvm::cast<llvm::Constant>(vals[ii]->value());
llvm::Type *elt = TypeManager::getLlvmTypeAtIndex(llct, ii);
if (elt != con->getType()) {
con = genConvertedConstant(con, elt);
assert(con != nullptr);
}
llvals[ii] = con;
}
}
if (llct->isStructTy()) {
llvm::StructType *llst = llvm::cast<llvm::StructType>(llct);
scon = llvm::ConstantStruct::get(llst, llvals);
} else {
llvm::ArrayType *llat = llvm::cast<llvm::ArrayType>(llct);
scon = llvm::ConstantArray::get(llat, llvals);
}
}
Binstructions noInstructions;
Bexpression *bcon = nbuilder_.mkComposite(btype, scon, vals,
noInstructions, location);
return makeGlobalExpression(bcon, scon, btype, location);
}
Bexpression *Llvm_backend::materializePointerOffset(Bexpression *ptroffExpr)
{
Location location = ptroffExpr->location();
std::vector<Bexpression *> cexprs =
nbuilder_.extractChildenAndDestroy(ptroffExpr);
assert(cexprs.size() == 2);
Bexpression *base = cexprs[0];
Bexpression *index = cexprs[1];
// Resolve index expression
index = resolveVarContext(index);
// When a pointer offset expression appears in a left-hand-side (assignment)
// context, the expected semantics are that location to be written is the
// one pointer to by the result of the pointer offset, meaning that we want
// to propagate any "lvalue-ness" found in 'base' up into the result
// expression (as opposed to delaying a load from 'base' itself).
//
// To achieve this effect, the code below essentially hides away the
// lvalue context on 'base' and then re-establishes it on 'rval' after the
// GEP. This is a painful hack, it would be nice to have a clean way
// to do this.
VarContext vc;
bool setLHS = false;
if (base->varExprPending() && base->varContext().lvalue()) {
setLHS = true;
base->resetVarExprContext();
base->setVarExprPending(false, 0);
}
base = resolveVarContext(base);
// Construct an appropriate GEP
llvm::PointerType *llpt =
llvm::cast<llvm::PointerType>(base->btype()->type());
llvm::Value *gep =
makePointerOffsetGEP(llpt, index->value(), base->value());
// Wrap in a Bexpression
Bexpression *rval = nbuilder_.mkPointerOffset(base->btype(), gep, base,
index, location);
// Re-establish lvalue context (as described above)
if (setLHS)
rval->setVarExprPending(true, 1);
std::string tag(base->tag());
tag += ".ptroff";
rval->setTag(tag);
// We're done
return rval;
}
Bexpression *Llvm_backend::materializeArrayIndex(Bexpression *arindExpr)
{
Location location = arindExpr->location();
std::vector<Bexpression *> cexprs =
nbuilder_.extractChildenAndDestroy(arindExpr);
assert(cexprs.size() == 2);
Bexpression *barray = cexprs[0];
Bexpression *index = cexprs[1];
if (barray->compositeInitPending())
barray = resolveCompositeInit(barray, nullptr);
index = resolveVarContext(index);
// Construct an appropriate GEP
llvm::ArrayType *llat =
llvm::cast<llvm::ArrayType>(barray->btype()->type());
llvm::Value *aval = barray->value();
llvm::Value *ival = index->value();
llvm::Value *eval = nullptr;
bool pending = false;
if (barray->isConstant()) {
if (index->isConstant())
eval = llvm::cast<llvm::Constant>(aval)->getAggregateElement(llvm::cast<llvm::Constant>(ival));
else {
// Constant array with non-constant index. Put the array
// into a temp var and load from there.
llvm::Constant *cval = llvm::cast<llvm::Constant>(aval);
Bvariable *cv = genVarForConstant(cval, barray->btype());
aval = cv->value();
pending = true;
}
}
if (!eval)
eval = makeArrayIndexGEP(llat, ival, aval);
Btype *bet = elementTypeByIndex(barray->btype(), 0);
// Wrap in a Bexpression
Bexpression *rval = nbuilder_.mkArrayIndex(bet, eval, barray, index, location);
if (pending)
rval->setVarExprPending(false, 0);
if (barray->varExprPending())
rval->setVarExprPending(barray->varContext());
std::string tag(barray->tag());
tag += ".index";
rval->setTag(tag);
// We're done
return rval;
}
struct GenCallState {
CABIOracle oracle;
BlockLIRBuilder builder;
std::vector<Bexpression *> resolvedArgs;
llvm::SmallVector<llvm::Value *, 16> llargs;
llvm::Value *chainVal;
llvm::Value *sretTemp;
BFunctionType *calleeFcnType;
Bfunction *callerFcn;
GenCallState(llvm::LLVMContext &context,
Bfunction *callerFunc,
BFunctionType *calleeFcnTyp,
TypeManager *tm,
NameGen *namegen,
llvm::Function *dummyFcn)
: oracle(calleeFcnTyp, tm),
builder(dummyFcn, namegen),
chainVal(nullptr),
sretTemp(nullptr),
calleeFcnType(calleeFcnTyp),
callerFcn(callerFunc) { }
};
static bool needSretTemp(const CABIParamInfo &returnInfo,
BFunctionType *calleeFcnTyp)
{
if (returnInfo.disp() == ParmIgnore)
return false;
if (returnInfo.disp() == ParmIndirect)
return true;
if (returnInfo.abiType()->isAggregateType())
return true;
if (calleeFcnTyp->resultType()->type()->isAggregateType())
return true;
return false;
}
void Llvm_backend::genCallProlog(GenCallState &state)
{
const CABIParamInfo &returnInfo = state.oracle.returnInfo();
if (needSretTemp(returnInfo, state.calleeFcnType)) {
assert(state.sretTemp == nullptr);
std::string tname(namegen("sret.actual"));
Btype *resTyp = state.calleeFcnType->resultType();
assert(state.callerFcn);
state.sretTemp = state.callerFcn->createTemporary(resTyp, tname);
if (returnInfo.disp() == ParmIndirect)
state.llargs.push_back(state.sretTemp);
}
// Chain param if needed
const CABIParamInfo &chainInfo = state.oracle.chainInfo();
if (chainInfo.disp() != ParmIgnore) {
assert(chainInfo.disp() == ParmDirect);
llvm::Value *cval = state.chainVal;
if (cval == nullptr)
cval = llvm::UndefValue::get(llvmPtrType());
state.llargs.push_back(cval);
}
}
void
Llvm_backend::genCallMarshallArgs(const std::vector<Bexpression *> &fn_args,
GenCallState &state)
{
for (unsigned idx = 0; idx < fn_args.size(); ++idx) {
const CABIParamInfo &paramInfo = state.oracle.paramInfo(idx);
if (paramInfo.attr() == AttrNest)
continue;
BlockLIRBuilder &builder = state.builder;
// For arguments not passed by value, no call to resolveVarContext
// (we want var address, not var value).
if (paramInfo.disp() == ParmIndirect) {
Bexpression *fnarg = fn_args[idx];
if (fnarg->compositeInitPending())
fnarg = resolveCompositeInit(fnarg, nullptr);
state.resolvedArgs.push_back(fnarg);
llvm::Value *val = fnarg->value();
assert(val);
// spill a constant arg to memory if needed
if (fnarg->isConstant()) {
llvm::Constant *cval = llvm::cast<llvm::Constant>(val);
Bvariable *cv = genVarForConstant(cval, fn_args[idx]->btype());
val = cv->value();
}
llvm::Type *vt = val->getType();
assert(vt->isPointerTy());
if (paramInfo.attr() == AttrByVal && vt->getPointerAddressSpace() != 0) {
// We pass a stack address, which is always in address space 0.
std::string castname(namegen("ascast"));
llvm::Type *pt = llvm::PointerType::get(vt->getPointerElementType(), 0);
val = builder.CreateAddrSpaceCast(val, pt, castname);
}
// For some architectures, such as arm64, the indirect parameter needs to
// be copied to the space allocated by the caller on the stack, and pass
// the address of the copied version to the callee.
if (paramInfo.attr() == AttrDoCopy) {
TypeManager *tm = state.oracle.tm();
Btype *bty = fnarg->btype();
uint64_t sz = tm->typeSize(bty);
uint64_t algn = tm->typeAlignment(bty);
llvm::MaybeAlign malgn(algn);
std::string tname(namegen("doCopy.addr"));
llvm::Value *tmpV = state.callerFcn->createTemporary(bty, tname);
builder.CreateMemCpy(tmpV, malgn, val, malgn, sz);
val = tmpV;
}
state.llargs.push_back(val);
continue;
}
// Resolve argument
Varexpr_context ctx = varContextDisp(fn_args[idx]);
if (paramInfo.abiTypes().size() == 2)
ctx = VE_lvalue;
Bexpression *resarg = resolve(fn_args[idx], ctx);
state.resolvedArgs.push_back(resarg);
if (paramInfo.disp() == ParmIgnore)
continue;
// At this point we're passing an argument directly,
// as opposed to in memory.
assert(paramInfo.disp() == ParmDirect);
llvm::Value *val = resarg->value();
if (paramInfo.abiTypes().size() == 1) {
if (ctx == VE_lvalue) {
// Passing single-eightbyte struct or array directly.
if (resarg->isConstant()) {
// If the value we're passing is a composite constant, we have to
// spill it to memory here in order for the casts below to work.
llvm::Constant *cval = llvm::cast<llvm::Constant>(val);
Bvariable *cv = genVarForConstant(cval, resarg->btype());
val = cv->value();
}
std::string castname(namegen("cast"));
// We are going to do a load, so the address space does not matter.
// It seems we may get here with either address space, so we just
// do an address-space-preserving cast.
llvm::Type *ptv =
llvm::PointerType::get(paramInfo.abiType(),
val->getType()->getPointerAddressSpace());
llvm::Value *bitcast = builder.CreateBitCast(val, ptv, castname);
std::string ltag(namegen("ld"));
llvm::Value *ld = builder.CreateLoad(paramInfo.abiType(), bitcast, ltag);
state.llargs.push_back(ld);
continue;
}
// Passing a single 8-byte-or-less argument.
// Apply any necessary pointer type conversions.
if (val->getType()->isPointerTy() && ctx == VE_rvalue) {
llvm::FunctionType *llft =
llvm::cast<llvm::FunctionType>(state.calleeFcnType->type());
llvm::Type *paramTyp = llft->getParamType(paramInfo.sigOffset());
val = convertForAssignment(resarg, paramTyp);
}
// Apply any necessary sign-extensions or zero-extensions.
if (paramInfo.abiType()->isIntegerTy()) {
if (paramInfo.attr() == AttrZext)
val = builder.CreateZExt(val, paramInfo.abiType(), namegen("zext"));
else if (paramInfo.attr() == AttrSext)
val = builder.CreateZExt(val, paramInfo.abiType(), namegen("sext"));
}
state.llargs.push_back(val);
continue;
}
// This now corresponds to the case of passing the contents of
// a small structure via no more than CABIParamInfo::ABI_TYPES_MAX_SIZE
// pieces / params.
assert(paramInfo.abiTypes().size() <= CABIParamInfo::ABI_TYPES_MAX_SIZE);
assert(paramInfo.attr() == AttrNone);
assert(ctx == VE_lvalue);
// Create a struct type of the appropriate shape
llvm::Type *llst = paramInfo.computeABIStructType(typeManager());
llvm::Type *ptst = makeLLVMPointerType(llst);
// If the value we're passing is a composite constant, we have to
// spill it to memory here in order for the casts below to work.
// Note that the spill is not needed if the value corresponds to a
// delated load of a composite variable (in which case it will
// already be an address). For example, if resarg corresponds to
// an anonymous constant value like [2]float64{10.0,10.0} then we
// need to spill, whereas if we're dealing with a reference to a
// named global constant, there is no need to spill.
if (resarg->isConstant() && val->getType() == resarg->btype()->type()) {
llvm::Constant *cval = llvm::cast<llvm::Constant>(val);
Bvariable *cv = genVarForConstant(cval, resarg->btype());
val = cv->value();
}
// Cast the value to the struct type
std::string tag(namegen("cast"));
llvm::Value *bitcast =
builder.CreatePointerBitCastOrAddrSpaceCast(val, ptst, tag);
// Load up each field
for ( unsigned i = 0; i < paramInfo.abiTypes().size(); ++i) {
std::string ftag(namegen("field"+std::to_string(i)));
llvm::Value *fieldgep =
builder.CreateConstInBoundsGEP2_32(llst, bitcast, 0, i, ftag);
std::string ltag(namegen("ld"));
llvm::Value *ld = builder.CreateLoad(paramInfo.abiTypes()[i], fieldgep, ltag);
state.llargs.push_back(ld);
}
}
}
void Llvm_backend::genCallAttributes(GenCallState &state, llvm::CallInst *call)
{
const llvm::AttributeList &callAttrList = call->getAttributes();
llvm::AttrBuilder retAttrs(context_, callAttrList.getRetAttrs());
const std::vector<Btype *> &paramTypes = state.calleeFcnType->paramTypes();
size_t na = state.oracle.getFunctionTypeForABI()->getNumParams();
llvm::SmallVector<llvm::AttributeSet, 4> argAttrs(na);
// Sret attribute if needed
const CABIParamInfo &returnInfo = state.oracle.returnInfo();
if (returnInfo.disp() == ParmIndirect) {
llvm::AttrBuilder ab(context_);
ab.addStructRetAttr(state.calleeFcnType->resultType()->type());
ab.addAttribute(llvm::Attribute::get(call->getContext(), "go_sret"));
argAttrs[0] = llvm::AttributeSet::get(context_, ab);
}
// Nest attribute if needed
const CABIParamInfo &chainInfo = state.oracle.chainInfo();
if (chainInfo.disp() != ParmIgnore) {
llvm::AttrBuilder ab(context_);
ab.addAttribute(llvm::Attribute::Nest);
argAttrs[chainInfo.sigOffset()] =
llvm::AttributeSet::get(context_, ab);
}
// Remainder of param attributes
for (unsigned idx = 0; idx < paramTypes.size(); ++idx) {
const CABIParamInfo &paramInfo = state.oracle.paramInfo(idx);
if (paramInfo.disp() == ParmIgnore)
continue;
assert(paramInfo.attr() != AttrNest);
assert(paramInfo.attr() != AttrStructReturn);
if (paramInfo.attr() != AttrNone) {
unsigned off = paramInfo.sigOffset();
llvm::AttrBuilder ab(context_);
if (paramInfo.attr() == AttrByVal) {
ab.addByValAttr(paramTypes[idx]->type());
} else if (paramInfo.attr() == AttrZext) {
ab.addAttribute(llvm::Attribute::ZExt);
} else if (paramInfo.attr() == AttrSext) {
ab.addAttribute(llvm::Attribute::SExt);
}
argAttrs[off] = llvm::AttributeSet::get(context_, ab);
}
}
call->setAttributes(
llvm::AttributeList::get(context_,
callAttrList.getFnAttrs(),
llvm::AttributeSet::get(context_, retAttrs),
argAttrs));
}
void Llvm_backend::genCallEpilog(GenCallState &state,
llvm::Instruction *callInst,
Bexpression *callExpr)
{
const CABIParamInfo &returnInfo = state.oracle.returnInfo();
if (needSretTemp(returnInfo, state.calleeFcnType)) {
assert(state.sretTemp);
assert(callExpr->value() == state.sretTemp);
callExpr->setVarExprPending(VE_rvalue, 0);
if (returnInfo.disp() == ParmDirect) {
// The call is returning something by value that doesn't match
// the expected abstract result type of the function. Cast the
// sret storage location to a pointer to the abi type and store
// the ABI return value into it.
llvm::Type *rt = (returnInfo.abiTypes().size() == 1 ?
returnInfo.abiType() :
returnInfo.computeABIStructType(typeManager()));
llvm::Type *ptrt = llvm::PointerType::get(rt, 0);
std::string castname(namegen("cast"));
llvm::Value *bitcast =
state.builder.CreateBitCast(state.sretTemp,
ptrt, castname);
std::string stname(namegen("st"));
state.builder.CreateStore(callInst, bitcast);
callExpr->appendInstructions(state.builder.instructions());
}
}
}
// makeGetgArm64 uses inline asm to implement the function of
// runtime.getg used in Go files on linux arm64.
static llvm::Value *makeGetgArm64(Btype *resType,
BlockLIRBuilder *builder,
Llvm_backend *be)
{
std::string asmStr;
std::string constr;
if (be->module().getPICLevel() > llvm::PICLevel::Level::NotPIC ||
be->module().getPIELevel() > llvm::PIELevel::Level::Default ) {
// Dynamic link.
asmStr += "adrp x0, :tlsdesc:runtime.g\n";
asmStr += "ldr $0, [x0, :tlsdesc_lo12:runtime.g]\n";
asmStr += "add x0, x0, :tlsdesc_lo12:runtime.g\n";
asmStr += ".tlsdesccall runtime.g\n";
asmStr += "blr $0\n";
asmStr += "mrs $0, TPIDR_EL0\n";
asmStr += "ldr $0, [$0, x0]\n";
// We need to clobber x0 because we have to use it to pass parameters.
// We also only need to clobber x0, because the TLS descriptor helper
// function only modifies x0
constr += "=r,~{x0}";
llvm::FunctionType *fnType =
llvm::FunctionType::get(resType->type(), llvm::ArrayRef<llvm::Type*>{}, false);
llvm::Value *callee = llvm::InlineAsm::get(fnType, llvm::StringRef(asmStr),
llvm::StringRef(constr), true);
std::string callname(be->namegen("asmcall"));
return builder->CreateCall(fnType, callee, {}, callname);
} else {
// Static link.
asmStr += "adrp $0, :gottprel:runtime.g\n";
asmStr += "ldr $0, [$0, #:gottprel_lo12:runtime.g]\n";
asmStr += "mrs $1, tpidr_el0\n";
asmStr += "ldr $0, [$1, $0]\n";
// In order not to clobber registers, we declare a temporary variable
// as the second output and return the first output.
constr += "=r,=r";
llvm::Type *tempRegType = llvm::IntegerType::get(builder->getContext(), 64);
llvm::Type *fnResType = llvm::StructType::create(
builder->getContext(), {resType->type(), tempRegType});
llvm::FunctionType *fnType =
llvm::FunctionType::get(fnResType, llvm::ArrayRef<llvm::Type*>{}, false);
llvm::Value *callee = llvm::InlineAsm::get(fnType, llvm::StringRef(asmStr),
llvm::StringRef(constr), true);
std::string callname(be->namegen("asmcall"));
llvm::Instruction *calI = builder->CreateCall(fnType, callee, {}, callname);
return builder->CreateExtractValue(calI, {0});
}
}
// Inline runtime.getg, generate a load of g.
// This is not done as a builtin because, unlike other builtins,
// we need the FE to tell us the result type.
static llvm::Value *makeGetg(Btype *resType,
BlockLIRBuilder *builder,
Llvm_backend *be)
{
llvm::GlobalValue* g = be->module().getGlobalVariable("runtime.g");
if (!g) {
unsigned int flags = Backend::variable_is_external;
Location location; // dummy
Bvariable* bv = be->global_variable("runtime.g", "runtime.g", resType,
flags, location);
g = llvm::cast<llvm::GlobalValue>(bv->value());
g->setThreadLocal(true);
}
if (be->triple().getArch() == llvm::Triple::aarch64)
return makeGetgArm64(resType, builder, be);
else
return builder->CreateLoad(resType->type(), g);
}
Bexpression *Llvm_backend::materializeCall(Bexpression *callExpr)
{
Location location = callExpr->location();
Bfunction *caller = callExpr->getFunction();
std::vector<Bexpression *> cexprs =
nbuilder_.extractChildenAndDestroy(callExpr);
Bexpression *fn_expr = cexprs[0];
Bexpression *chain_expr = cexprs[1];
std::vector<Bexpression *> fn_args;
for (unsigned idx = 2; idx < cexprs.size(); ++idx) {
fn_args.push_back(cexprs[idx]);
}
// Resolve fcn. Expect pointer-to-function type here.
fn_expr = resolveVarContext(fn_expr);
assert(fn_expr->btype()->type()->isPointerTy());
BFunctionType *calleeFcnTyp = unpackFunctionType(fn_expr->btype());
Btype *rbtype = calleeFcnTyp->resultType();
llvm::Value *fnval = fn_expr->value();
// Some intrinsic functions need additional args. Add them.
// TODO: currently this is specific to llvm.cttz, llvm.memmove, and
// llvm.memcpy; if the list expands too much more it might make
// sense to incorporate a description of the extra args into the
// builtin table entry.
if (llvm::isa<llvm::Function>(fnval)) {
llvm::Function *fcn = llvm::cast<llvm::Function>(fnval);
switch (fcn->getIntrinsicID()) {
case llvm::Intrinsic::cttz:
case llvm::Intrinsic::ctlz: {
// @llvm.cttz.i32 (i32 <src>, i1 <is_zero_undef>)
// Add the <is_zero_undef> arg.
// GCC's __builtin_ctz results undefined for 0 input.
llvm::Value *con = llvm::ConstantInt::getTrue(context_);
Btype *bt = makeAuxType(llvmBoolType());
Bexpression *conexpr = nbuilder_.mkConst(bt, con);
fn_args.push_back(conexpr);
break;
}
case llvm::Intrinsic::memmove:
case llvm::Intrinsic::memcpy: {
// memmove/memcpy take additional volatile arg
// volatile => false
llvm::Value *fcon = llvm::ConstantInt::getFalse(context_);
Btype *bt = makeAuxType(llvmBoolType());
Bexpression *volexpr = nbuilder_.mkConst(bt, fcon);
fn_args.push_back(volexpr);
break;
}
case llvm::Intrinsic::prefetch: {
// prefetch takes an additional arg for cache type
// (0: instruction, 1: data).
Btype *buint32t = integerType(true, 32);
llvm::Constant *c1 = llvm::ConstantInt::get(llvmInt32Type(), 1);
Bexpression *conexpr = nbuilder_.mkConst(buint32t, c1);
fn_args.push_back(conexpr);
break;
}
case llvm::Intrinsic::eh_dwarf_cfa: {
// llvm.eh.dwarf.cfa takes an additional arg 0.
Btype *buint32t = integerType(true, 32);
llvm::Constant *c1 = llvm::ConstantInt::get(llvmInt32Type(), 0);
Bexpression *conexpr = nbuilder_.mkConst(buint32t, c1);
fn_args.push_back(conexpr);
break;
}
default: {
// at the moment no other instrinsics need special handling
}
}
}
// State object to help with marshalling of call arguments, etc.
llvm::Function *dummyFcn = errorFunction_->function();
GenCallState state(context_, caller, calleeFcnTyp, typeManager(),
nameTags(), dummyFcn);
// Static chain expression if applicable
if (chain_expr->btype() != void_type()) {
chain_expr = resolveVarContext(chain_expr);
assert(chain_expr->btype()->type()->isPointerTy());
Btype *bpt = makeAuxType(llvmPtrType());
chain_expr = lateConvert(bpt, chain_expr, location);
assert(chain_expr->value() != nullptr);
state.chainVal = chain_expr->value();
}
// Set up for call (including creation of return tmp if needed)
genCallProlog(state);
// Unpack / resolve / marshall arguments
genCallMarshallArgs(fn_args, state);
// Create the actual call instruction
llvm::CallInst *call = nullptr;
llvm::Value *callValue = nullptr;
if (llvm::isa<llvm::Function>(fnval)) {
llvm::Function *fcn = llvm::cast<llvm::Function>(fnval);
BuiltinEntry *be = builtinTable_->lookup(fcn->getName().str());
if (be) {
BuiltinExprMaker makerfn = be->exprMaker();
if (makerfn)
callValue = makerfn(state.llargs, &state.builder, this);
} else if (fcn->getName() == "runtime.getg" && !DisableInlineGetg)
callValue = makeGetg(rbtype, &state.builder, this);
}
if (!callValue) {
llvm::FunctionType *llft =
llvm::cast<llvm::FunctionType>(calleeFcnTyp->type());
bool isvoid = llft->getReturnType()->isVoidTy();
std::string callname(isvoid ? "" : namegen("call"));
call = state.builder.CreateCall(llft, fnval,
state.llargs, callname);
genCallAttributes(state, call);
callValue = (state.sretTemp ? state.sretTemp : call);
}
Binstructions callInstructions;
std::vector<llvm::Instruction *> binstructions = state.builder.instructions();
for (auto i : binstructions)
callInstructions.appendInstruction(i);
Bexpression *rval =
nbuilder_.mkCall(rbtype, callValue, caller, fn_expr, chain_expr,
state.resolvedArgs, callInstructions, location);
if (call)
genCallEpilog(state, call, rval);
return rval;
}
llvm::Value *
Llvm_backend::convertForAssignment(Bexpression *src,
llvm::Type *dstToType)
{
if (src->value()->getType() == dstToType)
return src->value();
llvm::Function *dummyFcn = errorFunction_->function();
BlockLIRBuilder builder(dummyFcn, this);
llvm::Value *val = convertForAssignment(src->btype(), src->value(),
dstToType, &builder);
src->appendInstructions(builder.instructions());
return val;
}
llvm::Value *
Llvm_backend::convertForAssignment(Btype *srcBType,
llvm::Value *srcVal,
llvm::Type *dstToType,
BlockLIRBuilder *builder)
{
llvm::Type *srcType = srcVal->getType();
if (dstToType == srcType)
return srcVal;
// Case 1: handle discrepancies between representations of function
// descriptors. All front end function descriptor types are structs
// with a single field, however this field can sometimes be a pointer
// to function, and sometimes it can be of uintptr type.
bool srcPtrToFD = isPtrToFuncDescriptorType(srcType);
bool dstPtrToFD = isPtrToFuncDescriptorType(dstToType);
if (srcPtrToFD && dstPtrToFD) {
std::string tag(namegen("cast"));
llvm::Value *bitcast = builder->CreateBitCast(srcVal, dstToType, tag);
return bitcast;
}
// Case 2: handle circular function types.
bool dstCircFunc = isCircularFunctionType(dstToType);
bool srcCircFunc = isCircularFunctionType(srcType);
bool srcFuncPtr = isPtrToFuncType(srcType);
if (((srcPtrToFD || srcFuncPtr || srcCircFunc) && dstCircFunc) ||
(srcCircFunc && dstPtrToFD)) {
std::string tag(namegen("cast"));
llvm::Value *bitcast = builder->CreateBitCast(srcVal, dstToType, tag);
return bitcast;
}
// Case 3: handle raw function pointer assignment (frontend will
// sometimes take a function pointer and assign it to "void *" without
// an explicit conversion).
bool dstPtrToVoid = isPtrToVoidType(dstToType);
if (dstPtrToVoid && srcFuncPtr) {
std::string tag(namegen("cast"));
llvm::Value *bitcast = builder->CreateBitCast(srcVal, dstToType, tag);
return bitcast;
}
// Case 4: in some cases when calling a function (for example, __gogo)
// the front end will pass a raw function vale in place of a function
// descriptor. Allow this case.
if (dstPtrToFD && srcFuncPtr) {
std::string tag(namegen("cast"));
llvm::Value *bitcast =
builder->CreatePointerBitCastOrAddrSpaceCast(srcVal, dstToType, tag);
return bitcast;
}
// Case 5: handle polymorphic nil pointer expressions-- these are
// generated without a type initially, so we need to convert them
// to the appropriate type if they appear in an assignment context.
if (srcVal == nil_pointer_expression()->value()) {
std::string tag(namegen("cast"));
llvm::Value *bitcast = builder->CreateBitCast(srcVal, dstToType, tag);
return bitcast;
}
// Case 6: the code that creates interface values stores pointers of
// various flavors into i8*. Ideally it would be better to insert
// forced type conversions in the FE, but for now we allow this case.
bool srcPtrToIface = isPtrToIfaceStructType(srcType);
if (dstPtrToVoid && srcPtrToIface) {
std::string tag(namegen("cast"));
llvm::Value *bitcast = builder->CreateBitCast(srcVal, dstToType, tag);
return bitcast;
}
// Case 7: when creating slice values it's common for the frontend
// to mix pointers and arrays, e.g. assign "[3 x i64]*" to "i64**".
// Allow this sort of conversion.
llvm::Type *elt =
(dstToType->isPointerTy() ?
llvm::cast<llvm::PointerType>(dstToType)->getElementType() : nullptr);
if (elt && isPtrToArrayOf(srcType, elt)) {
std::string tag(namegen("cast"));
llvm::Value *bitcast = builder->CreateBitCast(srcVal, dstToType, tag);
return bitcast;
}
// Case 8: also when creating slice values it's common for the
// frontend to assign pointer-to-X to unsafe.Pointer (and vice versa)
// without an explicit cast. Allow this for now.
if ((dstToType == llvmPtrType() && llvm::isa<llvm::PointerType>(srcType)) ||
(srcType == llvmPtrType() && llvm::isa<llvm::PointerType>(dstToType))) {
std::string tag(namegen("cast"));
llvm::Value *bitcast = builder->CreatePointerBitCastOrAddrSpaceCast(srcVal, dstToType, tag);
return bitcast;
}
// Case 9: related to case 1 above, interface value expressions can
// contain C function pointers stored as "void *" instead of
// concrete pointer-to-function values. For example, consider a
// value V1 of the form
//
// { { T1*, void* }, O* }
//
// where T1 is a type descriptor and O is an object; this value can
// sometimes be assigned to a location of type
//
// { { T1*, F* }, O* }
//
// where F is a concrete C pointer-to-function (as opposed to "void*").
// Allow conversions of this sort for now.
std::set<llvm::Type *> visited;
if (fcnPointerCompatible(dstToType, srcType, visited)) {
std::string tag(namegen("cast"));
llvm::Value *bitcast =
builder->CreatePointerBitCastOrAddrSpaceCast(srcVal, dstToType, tag);
return bitcast;
}
// Case 10: circular pointer type (ex: type T *T; var p T; p = &p)
BPointerType *srcbpt = srcBType->castToBPointerType();
if (srcbpt) {
Btype *ctypconv = circularTypeAddrConversion(srcbpt->toType());
if (ctypconv != nullptr && ctypconv->type() == dstToType) {
std::string tag(namegen("cast"));
llvm::Value *bitcast = builder->CreateBitCast(srcVal, dstToType, tag);
return bitcast;
}
}
return srcVal;
}
// Walk the specified expression and invoke setVarExprPending on
// each var expression, with correct lvalue/rvalue tag depending on
// context.
class VarContextVisitor {
public:
VarContextVisitor(Bexpression *top,
Varexpr_context lvalueContext,
bool dumpDiffs)
: dumpDiffs_(dumpDiffs)
{
if (lvalueContext == VE_lvalue && isMem(top))
setLvalue(top);
}
std::pair< std::pair<VisitDisp, VisitChildDisp>, Bnode *>
visitChildPre(Bnode *parent, Bnode *child) {
Bexpression *eparent = parent->castToBexpression();
assert(eparent != nullptr);
// Don't descend into stmts or non-var nodes with values.
Bexpression *echild = child->castToBexpression();
if (child->isStmt() || (echild->value() != nullptr &&
echild->flavor() != N_Var))
return std::make_pair(std::make_pair(ContinueWalk, SkipChild), child);
// Propagate lvalue property down from parent to child through a memory
// operation such as a fieldref or arrayindex if applicable. For example,
// for the tree build from go code "x.f1[y]" such as
//
// array_index
// / \.
// field var(y)
// /
// var(x)
//
// If the top node (array_index) is in a left-hand-side position,
// then we want to flag "field" and "var(x)" nodes as being in an
// LHS context also, but not the "var(y)" node (it is not being
// assigned).
assert(echild != nullptr);
if (isLvalue(eparent) && isMem(echild)) {
Bexpression *cmem = memArg(eparent);
if (cmem == echild)
setLvalue(echild);
}
return std::make_pair(std::make_pair(ContinueWalk, VisitChild), child);
}
// Apply "var pending" tags to var exprs bottom up, once downward propagation
// of lvalue context is complete.
std::pair<VisitDisp, Bnode *> visitNodePost(Bnode *node)
{
Bexpression *expr = node->castToBexpression();
if (expr == nullptr || (expr->value() && expr->flavor() != N_Var))
return std::make_pair(ContinueWalk, expr);
if (expr->flavor() == N_Var)
setVarExprPending(expr, isLvalue(expr), 0);
// Debugging only. This is intended to provide a way to compare the
// tags applied by the frontend vs the tags this visitor generates.
if (dumpDiffs_)
dumpDiff(expr);
auto it = varcontext_.find(expr);
if (it != varcontext_.end()) {
VarContext vc(it->second);
bool lvalue = vc.lvalue();
if (expr->varExprPending()) {
assert(vc.equal(expr->varContext()));
} else {
expr->setVarExprPending(lvalue, 0);
}
}
return std::make_pair(ContinueWalk, expr);
}
// boilerplate
std::pair<VisitDisp, Bnode *> visitNodePre(Bnode *node) {
return std::make_pair(ContinueWalk, node);
}
// boilerplate
std::pair<VisitDisp, Bnode *> visitChildPost(Bnode *parent, Bnode *child) {
return std::make_pair(ContinueWalk, child);
}
private:
// Debugging only.
void dumpDiff(Bexpression *expr)
{
if (expr->value() && expr->flavor() != N_Var)
return;
VarContext oldvc;
if (expr->varExprPending())
oldvc = expr->varContext();
VarContext newvc;
auto nit = varcontext_.find(expr);
if (nit != varcontext_.end())
newvc = nit->second;
if (newvc.equal(oldvc))
return;
std::cerr << "Expr " << expr->id()
<< " " << expr->flavstr() << " ";
if (oldvc.pending()) {
std::cerr << "old VC(" << (oldvc.pending() ? "true" : "false")
<< "," << (oldvc.lvalue() ? "lval" : "rval")
<< "," << oldvc.addrLevel() << ") ";
}
if (newvc.pending()) {
std::cerr << "new VC(" << (newvc.pending() ? "true" : "false")
<< "," << (newvc.lvalue() ? "lval" : "rval")
<< "," << newvc.addrLevel() << ") ";
}
std::cerr << "\n";
expr->dump();
std::cerr << "\n";
}
Bexpression *memArg(Bexpression *expr) {
const std::vector<Bnode *> &kids = expr->children();
switch(expr->flavor()) {
case N_StructField:
case N_ArrayIndex:
case N_Address:
case N_Conversion:
case N_PointerOffset:
return kids[0]->castToBexpression();
default:
return nullptr;
}
}
bool isMem(Bexpression *expr) {
switch(expr->flavor()) {
case N_Var:
case N_StructField:
case N_Address:
case N_Deref:
case N_Conversion:
case N_ArrayIndex:
case N_PointerOffset:
return true;
default:
return false;
}
}
void setLvalue(Bexpression *expr) {
assert(lvalue_.find(expr) == lvalue_.end());
lvalue_.insert(expr);
}
bool isLvalue(Bexpression *expr) {
return lvalue_.find(expr) != lvalue_.end();
}
void setVarExprPending(Bexpression *expr, bool lvalue, unsigned addrLevel) {
assert(expr);
VarContext vc(lvalue, addrLevel);
varcontext_[expr] = vc;
}
private:
std::set<Bnode *> lvalue_;
std::map<Bexpression *, VarContext> varcontext_;
bool dumpDiffs_;
};
// Helper visitor class for expression folding; removes redundant
// nodes in preparation for materialization.
class FoldVisitor {
public:
FoldVisitor(Llvm_backend *be) : be_(be) { }
std::pair<VisitDisp, Bnode *> visitNodePost(Bnode *node) {
Bexpression *expr = node->castToBexpression();
if (!expr)
return std::make_pair(ContinueWalk, node);
if (expr && expr->value())
return std::make_pair(ContinueWalk, node);
// Fold addr(deref(X)) and deref(addr(X)) => X
if (node->flavor() == N_Address) {
std::vector<Bexpression *> akids = expr->getChildExprs();
if (akids[0]->flavor() == N_Deref) {
Bexpression *deref = akids[0];
// Extract children and delete first the addr node, then the
// deref node. Order is important; if we delete the deref first
// then the integrity visitor will wind up trying to access the
// deleted deref.
be_->nodeBuilder().extractChildenAndDestroy(expr);
std::vector<Bexpression *> dkids =
be_->nodeBuilder().extractChildenAndDestroy(deref);
// Return result
expr = dkids[0];
}
} else if (node->flavor() == N_Deref) {
std::vector<Bexpression *> dkids = expr->getChildExprs();
if (dkids[0]->flavor() == N_Address) {
Bexpression *address = dkids[0];
// Extract children and delete first the deref node, then the
// addr node. Order is important; if we delete the addr first
// then the integrity visitor will wind up trying to access the
// deleted addr.
be_->nodeBuilder().extractChildenAndDestroy(expr);
std::vector<Bexpression *> akids =
be_->nodeBuilder().extractChildenAndDestroy(address);
// Return result
expr = akids[0];
}
}
return std::make_pair(ContinueWalk, expr);
}
// If child is a statement (ex: compound expr) or if child
// already has an LLVM value, then prune walk at this node.
std::pair< std::pair<VisitDisp, VisitChildDisp>, Bnode *>
visitChildPre(Bnode *parent, Bnode *child) {
Bexpression *echild = child->castToBexpression();
if (child->isStmt() || echild->value() != nullptr)
return std::make_pair(std::make_pair(ContinueWalk, SkipChild), child);
return std::make_pair(std::make_pair(ContinueWalk, VisitChild), child);
}
// Boilerplate
std::pair<VisitDisp, Bnode *> visitNodePre(Bnode *node) {
return std::make_pair(ContinueWalk, node);
}
std::pair<VisitDisp, Bnode *> visitChildPost(Bnode *parent, Bnode *child) {
return std::make_pair(ContinueWalk, child);
}
private:
Llvm_backend *be_;
};
class MaterializeVisitor {
public:
MaterializeVisitor(Llvm_backend *be,
Bexpression *topNode,
Varexpr_context ctx)
: be_(be), topNode_(topNode), ctx_(ctx) { }
std::pair<VisitDisp, Bnode *> visitNodePre(Bnode *node) {
return std::make_pair(ContinueWalk, node);
}
std::pair<VisitDisp, Bnode *> visitNodePost(Bnode *node) {
Bexpression *expr = node->castToBexpression();
if (expr && expr->value())
return std::make_pair(ContinueWalk, node);
switch(node->flavor()) {
case N_EmptyStmt:
case N_LabelStmt:
case N_GotoStmt:
case N_ExprStmt:
case N_ReturnStmt:
case N_DeferStmt:
case N_IfStmt:
case N_ExcepStmt:
case N_BlockStmt:
case N_SwitchStmt: {
return std::make_pair(ContinueWalk, node);
}
case N_Error:
case N_Const:
case N_Var:
case N_FcnAddress:
case N_LabelAddress: {
break;
}
case N_Conversion: {
expr = be_->materializeConversion(expr);
break;
}
case N_Deref: {
bool isLHS = (expr == topNode_ && ctx_ == VE_lvalue);
expr = be_->materializeIndirect(expr, isLHS);
break;
}
case N_Address: {
expr = be_->materializeAddress(expr);
break;
}
case N_UnaryOp: {
expr = be_->materializeUnary(expr);
break;
}
case N_StructField: {
expr = be_->materializeStructField(expr);
break;
}
case N_BinaryOp: {
expr = be_->materializeBinary(expr);
break;
}
case N_Compound: {
expr = be_->materializeCompound(expr);
break;
}
case N_ArrayIndex: {
expr = be_->materializeArrayIndex(expr);
break;
}
case N_PointerOffset: {
expr = be_->materializePointerOffset(expr);
break;
}
case N_Composite: {
expr = be_->materializeComposite(expr);
break;
}
case N_Call: {
expr = be_->materializeCall(expr);
break;
}
case N_Conditional: {
expr = be_->materializeConditional(expr);
break;
}
}
assert(expr->value() != nullptr ||
expr->flavor() == N_Composite ||
expr->btype() == be_->void_type());
return std::make_pair(ContinueWalk, expr);
}
// If child is a statement (ex: compound expr), then no need to
// visit subtree (should already be materialized). Similarly if
// child already has an LLVM value, then prune walk at this node
std::pair< std::pair<VisitDisp, VisitChildDisp>, Bnode *>
visitChildPre(Bnode *parent, Bnode *child) {
Bexpression *echild = child->castToBexpression();
if (child->isStmt() || echild->value() != nullptr)
return std::make_pair(std::make_pair(ContinueWalk, SkipChild), child);
return std::make_pair(std::make_pair(ContinueWalk, VisitChild), child);
}
std::pair<VisitDisp, Bnode *> visitChildPost(Bnode *parent, Bnode *child) {
return std::make_pair(ContinueWalk, child);
}
private:
Llvm_backend *be_;
Bexpression *topNode_;
Varexpr_context ctx_;
};
Bexpression *Llvm_backend::materialize(Bexpression *expr,
Varexpr_context lvalueContext)
{
// Repair any node sharing at this point-- the materializer
// assumes that any node it visits can be destroyed/replaced
// without impacting some other portion of the tree.
enforceTreeIntegrity(expr);
// TODO:
// - don't really need a treewalk in the non-LHS case to apply
// can context tags; it would be simpler to only do the walk
// in the LHS case (and just apply var context tags when
// var expression is initially created)
// - an extension of the above: pass in the LHS/RHS context
// and propagate recursively during the materialize() walk,
// instead of having a separate tree-walk here.
// Locate and tag var expressions within the tree, selecting LHS or
// RHS context as appropriate. Needs to be done after the call above
// so as to insure that there are no share var expressions.
VarContextVisitor vcvis(expr, lvalueContext, false);
update_walk_nodes(expr, vcvis);
// Perform some basic folding operations. This is easier to do
// here so as not to worry about sharing.
FoldVisitor fvis(this);
Bnode *folded = update_walk_nodes(expr, fvis);
expr = folded->castToBexpression();
// Walk to materialize llvm values.
MaterializeVisitor mvis(this, expr, lvalueContext);
Bnode *materialized = update_walk_nodes(expr, mvis);
return materialized->castToBexpression();
}
Bexpression *Llvm_backend::lateConvert(Btype *type,
Bexpression *expr,
Location loc)
{
return materialize(convert_expression(type, expr, loc));
}