[NVPTX] Fix handling of indirect calls

Using a special machine node is cleaner than an InlineAsm node, and fixes an assertion failure in InstrEmitter

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@194810 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Justin Holewinski 2013-11-15 12:30:04 +00:00
parent 0dd0d1af2b
commit 4d748eb0e4
7 changed files with 56 additions and 12 deletions

View File

@ -18,6 +18,7 @@
#include "llvm/MC/MCExpr.h" #include "llvm/MC/MCExpr.h"
#include "llvm/MC/MCInst.h" #include "llvm/MC/MCInst.h"
#include "llvm/MC/MCInstrInfo.h" #include "llvm/MC/MCInstrInfo.h"
#include "llvm/MC/MCSymbol.h"
#include "llvm/MC/MCSubtargetInfo.h" #include "llvm/MC/MCSubtargetInfo.h"
#include "llvm/Support/ErrorHandling.h" #include "llvm/Support/ErrorHandling.h"
#include "llvm/Support/FormattedStream.h" #include "llvm/Support/FormattedStream.h"
@ -277,3 +278,12 @@ void NVPTXInstPrinter::printMemOperand(const MCInst *MI, int OpNum,
printOperand(MI, OpNum + 1, O); printOperand(MI, OpNum + 1, O);
} }
} }
void NVPTXInstPrinter::printProtoIdent(const MCInst *MI, int OpNum,
raw_ostream &O, const char *Modifier) {
const MCOperand &Op = MI->getOperand(OpNum);
assert(Op.isExpr() && "Call prototype is not an MCExpr?");
const MCExpr *Expr = Op.getExpr();
const MCSymbol &Sym = cast<MCSymbolRefExpr>(Expr)->getSymbol();
O << Sym.getName();
}

View File

@ -44,7 +44,8 @@ public:
raw_ostream &O, const char *Modifier = 0); raw_ostream &O, const char *Modifier = 0);
void printMemOperand(const MCInst *MI, int OpNum, void printMemOperand(const MCInst *MI, int OpNum,
raw_ostream &O, const char *Modifier = 0); raw_ostream &O, const char *Modifier = 0);
void printProtoIdent(const MCInst *MI, int OpNum,
raw_ostream &O, const char *Modifier = 0);
}; };
} }

View File

@ -314,6 +314,14 @@ void NVPTXAsmPrinter::EmitInstruction(const MachineInstr *MI) {
void NVPTXAsmPrinter::lowerToMCInst(const MachineInstr *MI, MCInst &OutMI) { void NVPTXAsmPrinter::lowerToMCInst(const MachineInstr *MI, MCInst &OutMI) {
OutMI.setOpcode(MI->getOpcode()); OutMI.setOpcode(MI->getOpcode());
// Special: Do not mangle symbol operand of CALL_PROTOTYPE
if (MI->getOpcode() == NVPTX::CALL_PROTOTYPE) {
const MachineOperand &MO = MI->getOperand(0);
OutMI.addOperand(GetSymbolRef(MO,
OutContext.GetOrCreateSymbol(Twine(MO.getSymbolName()))));
return;
}
for (unsigned i = 0, e = MI->getNumOperands(); i != e; ++i) { for (unsigned i = 0, e = MI->getNumOperands(); i != e; ++i) {
const MachineOperand &MO = MI->getOperand(i); const MachineOperand &MO = MI->getOperand(i);

View File

@ -310,6 +310,8 @@ const char *NVPTXTargetLowering::getTargetNodeName(unsigned Opcode) const {
return "NVPTXISD::CallSeqBegin"; return "NVPTXISD::CallSeqBegin";
case NVPTXISD::CallSeqEnd: case NVPTXISD::CallSeqEnd:
return "NVPTXISD::CallSeqEnd"; return "NVPTXISD::CallSeqEnd";
case NVPTXISD::CallPrototype:
return "NVPTXISD::CallPrototype";
case NVPTXISD::LoadV2: case NVPTXISD::LoadV2:
return "NVPTXISD::LoadV2"; return "NVPTXISD::LoadV2";
case NVPTXISD::LoadV4: case NVPTXISD::LoadV4:
@ -885,18 +887,16 @@ SDValue NVPTXTargetLowering::LowerCall(TargetLowering::CallLoweringInfo &CLI,
// proto_0 : .callprototype(.param .b32 _) _ (.param .b32 _); // proto_0 : .callprototype(.param .b32 _) _ (.param .b32 _);
// to be emitted, and the label has to used as the last arg of call // to be emitted, and the label has to used as the last arg of call
// instruction. // instruction.
// The prototype is embedded in a string and put as the operand for an // The prototype is embedded in a string and put as the operand for a
// INLINEASM SDNode. // CallPrototype SDNode which will print out to the value of the string.
SDVTList InlineAsmVTs = DAG.getVTList(MVT::Other, MVT::Glue); SDVTList ProtoVTs = DAG.getVTList(MVT::Other, MVT::Glue);
std::string proto_string = std::string Proto = getPrototype(retTy, Args, Outs, retAlignment, CS);
getPrototype(retTy, Args, Outs, retAlignment, CS); const char *ProtoStr =
const char *asmstr = nvTM->getManagedStrPool() nvTM->getManagedStrPool()->getManagedString(Proto.c_str())->c_str();
->getManagedString(proto_string.c_str())->c_str(); SDValue ProtoOps[] = {
SDValue InlineAsmOps[] = { Chain, DAG.getTargetExternalSymbol(ProtoStr, MVT::i32), InFlag,
Chain, DAG.getTargetExternalSymbol(asmstr, getPointerTy()),
DAG.getMDNode(0), DAG.getTargetConstant(0, MVT::i32), InFlag
}; };
Chain = DAG.getNode(ISD::INLINEASM, dl, InlineAsmVTs, InlineAsmOps, 5); Chain = DAG.getNode(NVPTXISD::CallPrototype, dl, ProtoVTs, &ProtoOps[0], 3);
InFlag = Chain.getValue(1); InFlag = Chain.getValue(1);
} }
// Op to just print "call" // Op to just print "call"

View File

@ -49,6 +49,7 @@ enum NodeType {
RETURN, RETURN,
CallSeqBegin, CallSeqBegin,
CallSeqEnd, CallSeqEnd,
CallPrototype,
Dummy, Dummy,
LoadV2 = ISD::FIRST_TARGET_MEMORY_OPCODE, LoadV2 = ISD::FIRST_TARGET_MEMORY_OPCODE,

View File

@ -2607,6 +2607,20 @@ def trapinst : NVPTXInst<(outs), (ins),
"trap;", "trap;",
[(trap)]>; [(trap)]>;
// Call prototype wrapper
def SDTCallPrototype : SDTypeProfile<0, 1, [SDTCisInt<0>]>;
def CallPrototype
: SDNode<"NVPTXISD::CallPrototype", SDTCallPrototype,
[SDNPHasChain, SDNPOutGlue, SDNPInGlue, SDNPSideEffect]>;
def ProtoIdent : Operand<i32> {
let PrintMethod = "printProtoIdent";
}
def CALL_PROTOTYPE
: NVPTXInst<(outs), (ins ProtoIdent:$ident),
"$ident", [(CallPrototype (i32 texternalsym:$ident))]>;
include "NVPTXIntrinsics.td" include "NVPTXIntrinsics.td"

View File

@ -0,0 +1,10 @@
; RUN: llc < %s -march=nvptx -mcpu=sm_20 | FileCheck %s
target triple = "nvptx"
define void @foo(i8* %ptr) {
%fnptr = bitcast i8* %ptr to void ()*
; CHECK: prototype_0 : .callprototype ()_ ()
tail call void %fnptr()
ret void
}