mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-05-02 04:39:35 +00:00
core changes for varargs
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@22254 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
f542821385
commit
558bc88a00
@ -119,7 +119,6 @@ namespace {
|
|||||||
void visitInstruction(Instruction &I);
|
void visitInstruction(Instruction &I);
|
||||||
|
|
||||||
void visitCallSite(CallSite CS);
|
void visitCallSite(CallSite CS);
|
||||||
void visitVANextInst(VANextInst &I);
|
|
||||||
void visitVAArgInst(VAArgInst &I);
|
void visitVAArgInst(VAArgInst &I);
|
||||||
|
|
||||||
void MergeConstantInitIntoNode(DSNodeHandle &NH, Constant *C);
|
void MergeConstantInitIntoNode(DSNodeHandle &NH, Constant *C);
|
||||||
@ -475,11 +474,8 @@ void GraphBuilder::visitReturnInst(ReturnInst &RI) {
|
|||||||
RetNode->mergeWith(getValueDest(*RI.getOperand(0)));
|
RetNode->mergeWith(getValueDest(*RI.getOperand(0)));
|
||||||
}
|
}
|
||||||
|
|
||||||
void GraphBuilder::visitVANextInst(VANextInst &I) {
|
|
||||||
getValueDest(*I.getOperand(0)).mergeWith(getValueDest(I));
|
|
||||||
}
|
|
||||||
|
|
||||||
void GraphBuilder::visitVAArgInst(VAArgInst &I) {
|
void GraphBuilder::visitVAArgInst(VAArgInst &I) {
|
||||||
|
//FIXME: also updates the argument
|
||||||
DSNodeHandle Ptr = getValueDest(*I.getOperand(0));
|
DSNodeHandle Ptr = getValueDest(*I.getOperand(0));
|
||||||
if (Ptr.isNull()) return;
|
if (Ptr.isNull()) return;
|
||||||
|
|
||||||
|
@ -330,7 +330,6 @@ namespace {
|
|||||||
void visitCastInst(CastInst &CI);
|
void visitCastInst(CastInst &CI);
|
||||||
void visitSetCondInst(SetCondInst &SCI) {} // NOOP!
|
void visitSetCondInst(SetCondInst &SCI) {} // NOOP!
|
||||||
void visitSelectInst(SelectInst &SI);
|
void visitSelectInst(SelectInst &SI);
|
||||||
void visitVANext(VANextInst &I);
|
|
||||||
void visitVAArg(VAArgInst &I);
|
void visitVAArg(VAArgInst &I);
|
||||||
void visitInstruction(Instruction &I);
|
void visitInstruction(Instruction &I);
|
||||||
};
|
};
|
||||||
@ -867,10 +866,6 @@ void Andersens::visitSelectInst(SelectInst &SI) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Andersens::visitVANext(VANextInst &I) {
|
|
||||||
// FIXME: Implement
|
|
||||||
assert(0 && "vanext not handled yet!");
|
|
||||||
}
|
|
||||||
void Andersens::visitVAArg(VAArgInst &I) {
|
void Andersens::visitVAArg(VAArgInst &I) {
|
||||||
assert(0 && "vaarg not handled yet!");
|
assert(0 && "vaarg not handled yet!");
|
||||||
}
|
}
|
||||||
|
@ -253,9 +253,9 @@ cast { RET_TOK(OtherOpVal, Cast, CAST); }
|
|||||||
select { RET_TOK(OtherOpVal, Select, SELECT); }
|
select { RET_TOK(OtherOpVal, Select, SELECT); }
|
||||||
shl { RET_TOK(OtherOpVal, Shl, SHL); }
|
shl { RET_TOK(OtherOpVal, Shl, SHL); }
|
||||||
shr { RET_TOK(OtherOpVal, Shr, SHR); }
|
shr { RET_TOK(OtherOpVal, Shr, SHR); }
|
||||||
vanext { RET_TOK(OtherOpVal, VANext, VANEXT); }
|
vanext { return VANEXT_old; }
|
||||||
vaarg { RET_TOK(OtherOpVal, VAArg , VAARG); }
|
vaarg { return VAARG_old; }
|
||||||
|
va_arg { RET_TOK(OtherOpVal, VAArg , VAARG); }
|
||||||
ret { RET_TOK(TermOpVal, Ret, RET); }
|
ret { RET_TOK(TermOpVal, Ret, RET); }
|
||||||
br { RET_TOK(TermOpVal, Br, BR); }
|
br { RET_TOK(TermOpVal, Br, BR); }
|
||||||
switch { RET_TOK(TermOpVal, Switch, SWITCH); }
|
switch { RET_TOK(TermOpVal, Switch, SWITCH); }
|
||||||
|
@ -47,6 +47,10 @@ static Module *ParserResult;
|
|||||||
|
|
||||||
#define YYERROR_VERBOSE 1
|
#define YYERROR_VERBOSE 1
|
||||||
|
|
||||||
|
static bool ObsoleteVarArgs;
|
||||||
|
static BasicBlock* CurBB;
|
||||||
|
|
||||||
|
|
||||||
// This contains info used when building the body of a function. It is
|
// This contains info used when building the body of a function. It is
|
||||||
// destroyed when the function is completed.
|
// destroyed when the function is completed.
|
||||||
//
|
//
|
||||||
@ -723,6 +727,7 @@ static PATypeHolder HandleUpRefs(const Type *ty) {
|
|||||||
static Module * RunParser(Module * M) {
|
static Module * RunParser(Module * M) {
|
||||||
|
|
||||||
llvmAsmlineno = 1; // Reset the current line number...
|
llvmAsmlineno = 1; // Reset the current line number...
|
||||||
|
ObsoleteVarArgs = false;
|
||||||
|
|
||||||
CurModule.CurrentModule = M;
|
CurModule.CurrentModule = M;
|
||||||
yyparse(); // Parse the file, potentially throwing exception
|
yyparse(); // Parse the file, potentially throwing exception
|
||||||
@ -730,6 +735,80 @@ static PATypeHolder HandleUpRefs(const Type *ty) {
|
|||||||
Module *Result = ParserResult;
|
Module *Result = ParserResult;
|
||||||
ParserResult = 0;
|
ParserResult = 0;
|
||||||
|
|
||||||
|
if(ObsoleteVarArgs) {
|
||||||
|
if(Function* F = Result->getNamedFunction("llvm.va_start")) {
|
||||||
|
assert(F->arg_size() == 0 && "Obsolete va_start takes 0 argument!");
|
||||||
|
|
||||||
|
//foo = va_start()
|
||||||
|
// ->
|
||||||
|
//bar = alloca typeof(foo)
|
||||||
|
//va_start(bar)
|
||||||
|
//foo = load bar
|
||||||
|
|
||||||
|
const Type* RetTy = Type::getPrimitiveType(Type::VoidTyID);
|
||||||
|
const Type* ArgTy = F->getFunctionType()->getReturnType();
|
||||||
|
const Type* ArgTyPtr = PointerType::get(ArgTy);
|
||||||
|
Function* NF = Result->getOrInsertFunction("llvm.va_start",
|
||||||
|
RetTy, ArgTyPtr, 0);
|
||||||
|
|
||||||
|
while (!F->use_empty()) {
|
||||||
|
CallInst* CI = cast<CallInst>(F->use_back());
|
||||||
|
AllocaInst* bar = new AllocaInst(ArgTy, 0, "vastart.fix.1", CI);
|
||||||
|
new CallInst(NF, bar, "", CI);
|
||||||
|
Value* foo = new LoadInst(bar, "vastart.fix.2", CI);
|
||||||
|
CI->replaceAllUsesWith(foo);
|
||||||
|
CI->getParent()->getInstList().erase(CI);
|
||||||
|
}
|
||||||
|
Result->getFunctionList().erase(F);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(Function* F = Result->getNamedFunction("llvm.va_end")) {
|
||||||
|
assert(F->arg_size() == 1 && "Obsolete va_end takes 1 argument!");
|
||||||
|
//vaend foo
|
||||||
|
// ->
|
||||||
|
//bar = alloca 1 of typeof(foo)
|
||||||
|
//vaend bar
|
||||||
|
const Type* RetTy = Type::getPrimitiveType(Type::VoidTyID);
|
||||||
|
const Type* ArgTy = F->getFunctionType()->getParamType(0);
|
||||||
|
const Type* ArgTyPtr = PointerType::get(ArgTy);
|
||||||
|
Function* NF = Result->getOrInsertFunction("llvm.va_end",
|
||||||
|
RetTy, ArgTyPtr, 0);
|
||||||
|
|
||||||
|
while (!F->use_empty()) {
|
||||||
|
CallInst* CI = cast<CallInst>(F->use_back());
|
||||||
|
AllocaInst* bar = new AllocaInst(ArgTy, 0, "vaend.fix.1", CI);
|
||||||
|
new CallInst(NF, bar, "", CI);
|
||||||
|
CI->getParent()->getInstList().erase(CI);
|
||||||
|
}
|
||||||
|
Result->getFunctionList().erase(F);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(Function* F = Result->getNamedFunction("llvm.va_copy")) {
|
||||||
|
assert(F->arg_size() == 1 && "Obsolete va_copy takes 1 argument!");
|
||||||
|
//foo = vacopy(bar)
|
||||||
|
// ->
|
||||||
|
//a = alloca 1 of typeof(foo)
|
||||||
|
//vacopy(a, bar)
|
||||||
|
//foo = load a
|
||||||
|
|
||||||
|
const Type* RetTy = Type::getPrimitiveType(Type::VoidTyID);
|
||||||
|
const Type* ArgTy = F->getFunctionType()->getReturnType();
|
||||||
|
const Type* ArgTyPtr = PointerType::get(ArgTy);
|
||||||
|
Function* NF = Result->getOrInsertFunction("llvm.va_copy",
|
||||||
|
RetTy, ArgTyPtr, ArgTy, 0);
|
||||||
|
|
||||||
|
while (!F->use_empty()) {
|
||||||
|
CallInst* CI = cast<CallInst>(F->use_back());
|
||||||
|
AllocaInst* a = new AllocaInst(ArgTy, 0, "vacopy.fix.1", CI);
|
||||||
|
new CallInst(NF, a, CI->getOperand(1), "", CI);
|
||||||
|
Value* foo = new LoadInst(a, "vacopy.fix.2", CI);
|
||||||
|
CI->replaceAllUsesWith(foo);
|
||||||
|
CI->getParent()->getInstList().erase(CI);
|
||||||
|
}
|
||||||
|
Result->getFunctionList().erase(F);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return Result;
|
return Result;
|
||||||
|
|
||||||
}
|
}
|
||||||
@ -866,7 +945,8 @@ Module *llvm::RunVMAsmParser(const char * AsmString, Module * M) {
|
|||||||
|
|
||||||
// Other Operators
|
// Other Operators
|
||||||
%type <OtherOpVal> ShiftOps
|
%type <OtherOpVal> ShiftOps
|
||||||
%token <OtherOpVal> PHI_TOK CAST SELECT SHL SHR VAARG VANEXT
|
%token <OtherOpVal> PHI_TOK CAST SELECT SHL SHR VAARG
|
||||||
|
%token VAARG_old VANEXT_old //OBSOLETE
|
||||||
|
|
||||||
|
|
||||||
%start Module
|
%start Module
|
||||||
@ -1727,7 +1807,7 @@ InstructionList : InstructionList Inst {
|
|||||||
$$ = $1;
|
$$ = $1;
|
||||||
}
|
}
|
||||||
| /* empty */ {
|
| /* empty */ {
|
||||||
$$ = getBBVal(ValID::create((int)CurFun.NextBBNum++), true);
|
$$ = CurBB = getBBVal(ValID::create((int)CurFun.NextBBNum++), true);
|
||||||
|
|
||||||
// Make sure to move the basic block to the correct location in the
|
// Make sure to move the basic block to the correct location in the
|
||||||
// function, instead of leaving it inserted wherever it was first
|
// function, instead of leaving it inserted wherever it was first
|
||||||
@ -1737,7 +1817,7 @@ InstructionList : InstructionList Inst {
|
|||||||
BBL.splice(BBL.end(), BBL, $$);
|
BBL.splice(BBL.end(), BBL, $$);
|
||||||
}
|
}
|
||||||
| LABELSTR {
|
| LABELSTR {
|
||||||
$$ = getBBVal(ValID::create($1), true);
|
$$ = CurBB = getBBVal(ValID::create($1), true);
|
||||||
|
|
||||||
// Make sure to move the basic block to the correct location in the
|
// Make sure to move the basic block to the correct location in the
|
||||||
// function, instead of leaving it inserted wherever it was first
|
// function, instead of leaving it inserted wherever it was first
|
||||||
@ -1964,8 +2044,45 @@ InstVal : ArithmeticOps Types ValueRef ',' ValueRef {
|
|||||||
$$ = new VAArgInst($2, *$4);
|
$$ = new VAArgInst($2, *$4);
|
||||||
delete $4;
|
delete $4;
|
||||||
}
|
}
|
||||||
| VANEXT ResolvedVal ',' Types {
|
| VAARG_old ResolvedVal ',' Types {
|
||||||
$$ = new VANextInst($2, *$4);
|
ObsoleteVarArgs = true;
|
||||||
|
const Type* ArgTy = $2->getType();
|
||||||
|
Function* NF = CurModule.CurrentModule->
|
||||||
|
getOrInsertFunction("llvm.va_copy", ArgTy, ArgTy, 0);
|
||||||
|
|
||||||
|
//b = vaarg a, t ->
|
||||||
|
//foo = alloca 1 of t
|
||||||
|
//bar = vacopy a
|
||||||
|
//store bar -> foo
|
||||||
|
//b = vaarg foo, t
|
||||||
|
AllocaInst* foo = new AllocaInst(ArgTy, 0, "vaarg.fix");
|
||||||
|
CurBB->getInstList().push_back(foo);
|
||||||
|
CallInst* bar = new CallInst(NF, $2);
|
||||||
|
CurBB->getInstList().push_back(bar);
|
||||||
|
CurBB->getInstList().push_back(new StoreInst(bar, foo));
|
||||||
|
$$ = new VAArgInst(foo, *$4);
|
||||||
|
delete $4;
|
||||||
|
}
|
||||||
|
| VANEXT_old ResolvedVal ',' Types {
|
||||||
|
ObsoleteVarArgs = true;
|
||||||
|
const Type* ArgTy = $2->getType();
|
||||||
|
Function* NF = CurModule.CurrentModule->
|
||||||
|
getOrInsertFunction("llvm.va_copy", ArgTy, ArgTy, 0);
|
||||||
|
|
||||||
|
//b = vanext a, t ->
|
||||||
|
//foo = alloca 1 of t
|
||||||
|
//bar = vacopy a
|
||||||
|
//store bar -> foo
|
||||||
|
//tmp = vaarg foo, t
|
||||||
|
//b = load foo
|
||||||
|
AllocaInst* foo = new AllocaInst(ArgTy, 0, "vanext.fix");
|
||||||
|
CurBB->getInstList().push_back(foo);
|
||||||
|
CallInst* bar = new CallInst(NF, $2);
|
||||||
|
CurBB->getInstList().push_back(bar);
|
||||||
|
CurBB->getInstList().push_back(new StoreInst(bar, foo));
|
||||||
|
Instruction* tmp = new VAArgInst(foo, *$4);
|
||||||
|
CurBB->getInstList().push_back(tmp);
|
||||||
|
$$ = new LoadInst(foo);
|
||||||
delete $4;
|
delete $4;
|
||||||
}
|
}
|
||||||
| PHI_TOK PHIList {
|
| PHI_TOK PHIList {
|
||||||
|
@ -658,10 +658,43 @@ void BytecodeReader::ParseInstruction(std::vector<unsigned> &Oprnds,
|
|||||||
Result = new VAArgInst(getValue(iType, Oprnds[0]),
|
Result = new VAArgInst(getValue(iType, Oprnds[0]),
|
||||||
getSanitizedType(Oprnds[1]));
|
getSanitizedType(Oprnds[1]));
|
||||||
break;
|
break;
|
||||||
case Instruction::VANext:
|
case 32: { //VANext_old
|
||||||
Result = new VANextInst(getValue(iType, Oprnds[0]),
|
const Type* ArgTy = getValue(iType, Oprnds[0])->getType();
|
||||||
getSanitizedType(Oprnds[1]));
|
Function* NF = TheModule->getOrInsertFunction("llvm.va_copy", ArgTy, ArgTy, 0);
|
||||||
|
|
||||||
|
//b = vanext a, t ->
|
||||||
|
//foo = alloca 1 of t
|
||||||
|
//bar = vacopy a
|
||||||
|
//store bar -> foo
|
||||||
|
//tmp = vaarg foo, t
|
||||||
|
//b = load foo
|
||||||
|
AllocaInst* foo = new AllocaInst(ArgTy, 0, "vanext.fix");
|
||||||
|
BB->getInstList().push_back(foo);
|
||||||
|
CallInst* bar = new CallInst(NF, getValue(iType, Oprnds[0]));
|
||||||
|
BB->getInstList().push_back(bar);
|
||||||
|
BB->getInstList().push_back(new StoreInst(bar, foo));
|
||||||
|
Instruction* tmp = new VAArgInst(foo, getSanitizedType(Oprnds[1]));
|
||||||
|
BB->getInstList().push_back(tmp);
|
||||||
|
Result = new LoadInst(foo);
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
case 33: { //VAArg_old
|
||||||
|
const Type* ArgTy = getValue(iType, Oprnds[0])->getType();
|
||||||
|
Function* NF = TheModule->getOrInsertFunction("llvm.va_copy", ArgTy, ArgTy, 0);
|
||||||
|
|
||||||
|
//b = vaarg a, t ->
|
||||||
|
//foo = alloca 1 of t
|
||||||
|
//bar = vacopy a
|
||||||
|
//store bar -> foo
|
||||||
|
//b = vaarg foo, t
|
||||||
|
AllocaInst* foo = new AllocaInst(ArgTy, 0, "vaarg.fix");
|
||||||
|
BB->getInstList().push_back(foo);
|
||||||
|
CallInst* bar = new CallInst(NF, getValue(iType, Oprnds[0]));
|
||||||
|
BB->getInstList().push_back(bar);
|
||||||
|
BB->getInstList().push_back(new StoreInst(bar, foo));
|
||||||
|
Result = new VAArgInst(foo, getSanitizedType(Oprnds[1]));
|
||||||
|
break;
|
||||||
|
}
|
||||||
case Instruction::Cast:
|
case Instruction::Cast:
|
||||||
Result = new CastInst(getValue(iType, Oprnds[0]),
|
Result = new CastInst(getValue(iType, Oprnds[0]),
|
||||||
getSanitizedType(Oprnds[1]));
|
getSanitizedType(Oprnds[1]));
|
||||||
|
@ -148,6 +148,100 @@ BytecodeStdinReader::BytecodeStdinReader( BytecodeHandler* H )
|
|||||||
ParseBytecode(FileBuf, FileData.size(), "<stdin>");
|
ParseBytecode(FileBuf, FileData.size(), "<stdin>");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//===----------------------------------------------------------------------===//
|
||||||
|
// Varargs transmogrification code...
|
||||||
|
//
|
||||||
|
|
||||||
|
// CheckVarargs - This is used to automatically translate old-style varargs to
|
||||||
|
// new style varargs for backwards compatibility.
|
||||||
|
static ModuleProvider* CheckVarargs(ModuleProvider* MP) {
|
||||||
|
Module* M = MP->getModule();
|
||||||
|
|
||||||
|
// check to see if va_start takes arguements...
|
||||||
|
Function* F = M->getNamedFunction("llvm.va_start");
|
||||||
|
if(F == 0) return MP; //No varargs use, just return.
|
||||||
|
|
||||||
|
if (F->getFunctionType()->getNumParams() == 1)
|
||||||
|
return MP; // Modern varargs processing, just return.
|
||||||
|
|
||||||
|
// If we get to this point, we know that we have an old-style module.
|
||||||
|
// Materialize the whole thing to perform the rewriting.
|
||||||
|
MP->materializeModule();
|
||||||
|
|
||||||
|
if(Function* F = M->getNamedFunction("llvm.va_start")) {
|
||||||
|
assert(F->arg_size() == 0 && "Obsolete va_start takes 0 argument!");
|
||||||
|
|
||||||
|
//foo = va_start()
|
||||||
|
// ->
|
||||||
|
//bar = alloca typeof(foo)
|
||||||
|
//va_start(bar)
|
||||||
|
//foo = load bar
|
||||||
|
|
||||||
|
const Type* RetTy = Type::getPrimitiveType(Type::VoidTyID);
|
||||||
|
const Type* ArgTy = F->getFunctionType()->getReturnType();
|
||||||
|
const Type* ArgTyPtr = PointerType::get(ArgTy);
|
||||||
|
Function* NF = M->getOrInsertFunction("llvm.va_start",
|
||||||
|
RetTy, ArgTyPtr, 0);
|
||||||
|
|
||||||
|
for(Value::use_iterator I = F->use_begin(), E = F->use_end(); I != E;)
|
||||||
|
if (CallInst* CI = dyn_cast<CallInst>(*I++)) {
|
||||||
|
AllocaInst* bar = new AllocaInst(ArgTy, 0, "vastart.fix.1", CI);
|
||||||
|
new CallInst(NF, bar, "", CI);
|
||||||
|
Value* foo = new LoadInst(bar, "vastart.fix.2", CI);
|
||||||
|
CI->replaceAllUsesWith(foo);
|
||||||
|
CI->getParent()->getInstList().erase(CI);
|
||||||
|
}
|
||||||
|
F->setName("");
|
||||||
|
}
|
||||||
|
|
||||||
|
if(Function* F = M->getNamedFunction("llvm.va_end")) {
|
||||||
|
assert(F->arg_size() == 1 && "Obsolete va_end takes 1 argument!");
|
||||||
|
//vaend foo
|
||||||
|
// ->
|
||||||
|
//bar = alloca 1 of typeof(foo)
|
||||||
|
//vaend bar
|
||||||
|
const Type* RetTy = Type::getPrimitiveType(Type::VoidTyID);
|
||||||
|
const Type* ArgTy = F->getFunctionType()->getParamType(0);
|
||||||
|
const Type* ArgTyPtr = PointerType::get(ArgTy);
|
||||||
|
Function* NF = M->getOrInsertFunction("llvm.va_end",
|
||||||
|
RetTy, ArgTyPtr, 0);
|
||||||
|
|
||||||
|
for(Value::use_iterator I = F->use_begin(), E = F->use_end(); I != E;)
|
||||||
|
if (CallInst* CI = dyn_cast<CallInst>(*I++)) {
|
||||||
|
AllocaInst* bar = new AllocaInst(ArgTy, 0, "vaend.fix.1", CI);
|
||||||
|
new CallInst(NF, bar, "", CI);
|
||||||
|
CI->getParent()->getInstList().erase(CI);
|
||||||
|
}
|
||||||
|
F->setName("");
|
||||||
|
}
|
||||||
|
|
||||||
|
if(Function* F = M->getNamedFunction("llvm.va_copy")) {
|
||||||
|
assert(F->arg_size() == 1 && "Obsolete va_copy takes 1 argument!");
|
||||||
|
//foo = vacopy(bar)
|
||||||
|
// ->
|
||||||
|
//a = alloca 1 of typeof(foo)
|
||||||
|
//vacopy(a, bar)
|
||||||
|
//foo = load a
|
||||||
|
|
||||||
|
const Type* RetTy = Type::getPrimitiveType(Type::VoidTyID);
|
||||||
|
const Type* ArgTy = F->getFunctionType()->getReturnType();
|
||||||
|
const Type* ArgTyPtr = PointerType::get(ArgTy);
|
||||||
|
Function* NF = M->getOrInsertFunction("llvm.va_copy",
|
||||||
|
RetTy, ArgTyPtr, ArgTy, 0);
|
||||||
|
|
||||||
|
for(Value::use_iterator I = F->use_begin(), E = F->use_end(); I != E;)
|
||||||
|
if (CallInst* CI = dyn_cast<CallInst>(*I++)) {
|
||||||
|
AllocaInst* a = new AllocaInst(ArgTy, 0, "vacopy.fix.1", CI);
|
||||||
|
new CallInst(NF, a, CI->getOperand(1), "", CI);
|
||||||
|
Value* foo = new LoadInst(a, "vacopy.fix.2", CI);
|
||||||
|
CI->replaceAllUsesWith(foo);
|
||||||
|
CI->getParent()->getInstList().erase(CI);
|
||||||
|
}
|
||||||
|
F->setName("");
|
||||||
|
}
|
||||||
|
return MP;
|
||||||
|
}
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
// Wrapper functions
|
// Wrapper functions
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
@ -159,7 +253,8 @@ llvm::getBytecodeBufferModuleProvider(const unsigned char *Buffer,
|
|||||||
unsigned Length,
|
unsigned Length,
|
||||||
const std::string &ModuleID,
|
const std::string &ModuleID,
|
||||||
BytecodeHandler* H ) {
|
BytecodeHandler* H ) {
|
||||||
return new BytecodeBufferReader(Buffer, Length, ModuleID, H);
|
return CheckVarargs(
|
||||||
|
new BytecodeBufferReader(Buffer, Length, ModuleID, H));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// ParseBytecodeBuffer - Parse a given bytecode buffer
|
/// ParseBytecodeBuffer - Parse a given bytecode buffer
|
||||||
@ -182,9 +277,9 @@ Module *llvm::ParseBytecodeBuffer(const unsigned char *Buffer, unsigned Length,
|
|||||||
ModuleProvider *llvm::getBytecodeModuleProvider(const std::string &Filename,
|
ModuleProvider *llvm::getBytecodeModuleProvider(const std::string &Filename,
|
||||||
BytecodeHandler* H) {
|
BytecodeHandler* H) {
|
||||||
if (Filename != std::string("-")) // Read from a file...
|
if (Filename != std::string("-")) // Read from a file...
|
||||||
return new BytecodeFileReader(Filename,H);
|
return CheckVarargs(new BytecodeFileReader(Filename,H));
|
||||||
else // Read from stdin
|
else // Read from stdin
|
||||||
return new BytecodeStdinReader(H);
|
return CheckVarargs(new BytecodeStdinReader(H));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// ParseBytecodeFile - Parse the given bytecode file
|
/// ParseBytecodeFile - Parse the given bytecode file
|
||||||
|
@ -187,8 +187,6 @@ void SlotCalculator::processModule() {
|
|||||||
!isa<GlobalValue>(I->getOperand(op)))
|
!isa<GlobalValue>(I->getOperand(op)))
|
||||||
getOrCreateSlot(I->getOperand(op));
|
getOrCreateSlot(I->getOperand(op));
|
||||||
getOrCreateSlot(I->getType());
|
getOrCreateSlot(I->getType());
|
||||||
if (const VANextInst *VAN = dyn_cast<VANextInst>(&*I))
|
|
||||||
getOrCreateSlot(VAN->getArgType());
|
|
||||||
}
|
}
|
||||||
processSymbolTableConstants(&F->getSymbolTable());
|
processSymbolTableConstants(&F->getSymbolTable());
|
||||||
}
|
}
|
||||||
@ -320,8 +318,6 @@ void SlotCalculator::incorporateFunction(const Function *F) {
|
|||||||
getOrCreateSlot(BB);
|
getOrCreateSlot(BB);
|
||||||
for (BasicBlock::const_iterator I = BB->begin(), E = BB->end(); I!=E; ++I) {
|
for (BasicBlock::const_iterator I = BB->begin(), E = BB->end(); I!=E; ++I) {
|
||||||
getOrCreateSlot(I);
|
getOrCreateSlot(I);
|
||||||
if (const VANextInst *VAN = dyn_cast<VANextInst>(I))
|
|
||||||
getOrCreateSlot(VAN->getArgType());
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -472,8 +468,6 @@ void SlotCalculator::buildCompactionTable(const Function *F) {
|
|||||||
for (unsigned op = 0, e = I->getNumOperands(); op != e; ++op)
|
for (unsigned op = 0, e = I->getNumOperands(); op != e; ++op)
|
||||||
if (isa<Constant>(I->getOperand(op)))
|
if (isa<Constant>(I->getOperand(op)))
|
||||||
getOrCreateCompactionTableSlot(I->getOperand(op));
|
getOrCreateCompactionTableSlot(I->getOperand(op));
|
||||||
if (const VANextInst *VAN = dyn_cast<VANextInst>(&*I))
|
|
||||||
getOrCreateCompactionTableSlot(VAN->getArgType());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Do the types in the symbol table
|
// Do the types in the symbol table
|
||||||
|
@ -439,7 +439,7 @@ void BytecodeWriter::outputInstructionFormat0(const Instruction *I,
|
|||||||
output_typeid(Type); // Result type
|
output_typeid(Type); // Result type
|
||||||
|
|
||||||
unsigned NumArgs = I->getNumOperands();
|
unsigned NumArgs = I->getNumOperands();
|
||||||
output_vbr(NumArgs + (isa<CastInst>(I) || isa<VANextInst>(I) ||
|
output_vbr(NumArgs + (isa<CastInst>(I) ||
|
||||||
isa<VAArgInst>(I) || Opcode == 56 || Opcode == 58));
|
isa<VAArgInst>(I) || Opcode == 56 || Opcode == 58));
|
||||||
|
|
||||||
if (!isa<GetElementPtrInst>(&I)) {
|
if (!isa<GetElementPtrInst>(&I)) {
|
||||||
@ -453,10 +453,6 @@ void BytecodeWriter::outputInstructionFormat0(const Instruction *I,
|
|||||||
int Slot = Table.getSlot(I->getType());
|
int Slot = Table.getSlot(I->getType());
|
||||||
assert(Slot != -1 && "Cast return type unknown?");
|
assert(Slot != -1 && "Cast return type unknown?");
|
||||||
output_typeid((unsigned)Slot);
|
output_typeid((unsigned)Slot);
|
||||||
} else if (const VANextInst *VAI = dyn_cast<VANextInst>(I)) {
|
|
||||||
int Slot = Table.getSlot(VAI->getArgType());
|
|
||||||
assert(Slot != -1 && "VarArg argument type unknown?");
|
|
||||||
output_typeid((unsigned)Slot);
|
|
||||||
} else if (Opcode == 56) { // Invoke escape sequence
|
} else if (Opcode == 56) { // Invoke escape sequence
|
||||||
output_vbr(cast<InvokeInst>(I)->getCallingConv());
|
output_vbr(cast<InvokeInst>(I)->getCallingConv());
|
||||||
} else if (Opcode == 58) { // Call escape sequence
|
} else if (Opcode == 58) { // Call escape sequence
|
||||||
@ -704,11 +700,6 @@ void BytecodeWriter::outputInstruction(const Instruction &I) {
|
|||||||
assert(Slots[1] != ~0U && "Cast return type unknown?");
|
assert(Slots[1] != ~0U && "Cast return type unknown?");
|
||||||
if (Slots[1] > MaxOpSlot) MaxOpSlot = Slots[1];
|
if (Slots[1] > MaxOpSlot) MaxOpSlot = Slots[1];
|
||||||
NumOperands++;
|
NumOperands++;
|
||||||
} else if (const VANextInst *VANI = dyn_cast<VANextInst>(&I)) {
|
|
||||||
Slots[1] = Table.getSlot(VANI->getArgType());
|
|
||||||
assert(Slots[1] != ~0U && "va_next return type unknown?");
|
|
||||||
if (Slots[1] > MaxOpSlot) MaxOpSlot = Slots[1];
|
|
||||||
NumOperands++;
|
|
||||||
} else if (const GetElementPtrInst *GEP = dyn_cast<GetElementPtrInst>(&I)) {
|
} else if (const GetElementPtrInst *GEP = dyn_cast<GetElementPtrInst>(&I)) {
|
||||||
// We need to encode the type of sequential type indices into their slot #
|
// We need to encode the type of sequential type indices into their slot #
|
||||||
unsigned Idx = 1;
|
unsigned Idx = 1;
|
||||||
|
@ -356,7 +356,6 @@ public:
|
|||||||
void visitCall(CallInst &I);
|
void visitCall(CallInst &I);
|
||||||
|
|
||||||
void visitVAStart(CallInst &I);
|
void visitVAStart(CallInst &I);
|
||||||
void visitVANext(VANextInst &I);
|
|
||||||
void visitVAArg(VAArgInst &I);
|
void visitVAArg(VAArgInst &I);
|
||||||
void visitVAEnd(CallInst &I);
|
void visitVAEnd(CallInst &I);
|
||||||
void visitVACopy(CallInst &I);
|
void visitVACopy(CallInst &I);
|
||||||
@ -839,7 +838,7 @@ void SelectionDAGLowering::visitFree(FreeInst &I) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
std::pair<SDOperand, SDOperand>
|
std::pair<SDOperand, SDOperand>
|
||||||
TargetLowering::LowerVAStart(SDOperand Chain, SelectionDAG &DAG) {
|
TargetLowering::LowerVAStart(SDOperand Chain, SelectionDAG &DAG, SDOperand Dest) {
|
||||||
// We have no sane default behavior, just emit a useful error message and bail
|
// We have no sane default behavior, just emit a useful error message and bail
|
||||||
// out.
|
// out.
|
||||||
std::cerr << "Variable arguments handling not implemented on this target!\n";
|
std::cerr << "Variable arguments handling not implemented on this target!\n";
|
||||||
@ -854,13 +853,16 @@ SDOperand TargetLowering::LowerVAEnd(SDOperand Chain, SDOperand L,
|
|||||||
}
|
}
|
||||||
|
|
||||||
std::pair<SDOperand,SDOperand>
|
std::pair<SDOperand,SDOperand>
|
||||||
TargetLowering::LowerVACopy(SDOperand Chain, SDOperand L, SelectionDAG &DAG) {
|
TargetLowering::LowerVACopy(SDOperand Chain, SDOperand Src, SDOperand Dest,
|
||||||
// Default to returning the input list.
|
SelectionDAG &DAG) {
|
||||||
return std::make_pair(L, Chain);
|
// We have no sane default behavior, just emit a useful error message and bail
|
||||||
|
// out.
|
||||||
|
std::cerr << "Variable arguments handling not implemented on this target!\n";
|
||||||
|
abort();
|
||||||
}
|
}
|
||||||
|
|
||||||
std::pair<SDOperand,SDOperand>
|
std::pair<SDOperand,SDOperand>
|
||||||
TargetLowering::LowerVAArgNext(bool isVANext, SDOperand Chain, SDOperand VAList,
|
TargetLowering::LowerVAArgNext(SDOperand Chain, SDOperand VAList,
|
||||||
const Type *ArgTy, SelectionDAG &DAG) {
|
const Type *ArgTy, SelectionDAG &DAG) {
|
||||||
// We have no sane default behavior, just emit a useful error message and bail
|
// We have no sane default behavior, just emit a useful error message and bail
|
||||||
// out.
|
// out.
|
||||||
@ -871,34 +873,26 @@ TargetLowering::LowerVAArgNext(bool isVANext, SDOperand Chain, SDOperand VAList,
|
|||||||
|
|
||||||
|
|
||||||
void SelectionDAGLowering::visitVAStart(CallInst &I) {
|
void SelectionDAGLowering::visitVAStart(CallInst &I) {
|
||||||
std::pair<SDOperand,SDOperand> Result = TLI.LowerVAStart(getRoot(), DAG);
|
std::pair<SDOperand,SDOperand> Result = TLI.LowerVAStart(getRoot(), DAG, getValue(I.getOperand(1)));
|
||||||
setValue(&I, Result.first);
|
setValue(&I, Result.first);
|
||||||
DAG.setRoot(Result.second);
|
DAG.setRoot(Result.second);
|
||||||
}
|
}
|
||||||
|
|
||||||
void SelectionDAGLowering::visitVAArg(VAArgInst &I) {
|
void SelectionDAGLowering::visitVAArg(VAArgInst &I) {
|
||||||
std::pair<SDOperand,SDOperand> Result =
|
std::pair<SDOperand,SDOperand> Result =
|
||||||
TLI.LowerVAArgNext(false, getRoot(), getValue(I.getOperand(0)),
|
TLI.LowerVAArgNext(getRoot(), getValue(I.getOperand(0)),
|
||||||
I.getType(), DAG);
|
I.getType(), DAG);
|
||||||
setValue(&I, Result.first);
|
setValue(&I, Result.first);
|
||||||
DAG.setRoot(Result.second);
|
DAG.setRoot(Result.second);
|
||||||
}
|
}
|
||||||
|
|
||||||
void SelectionDAGLowering::visitVANext(VANextInst &I) {
|
|
||||||
std::pair<SDOperand,SDOperand> Result =
|
|
||||||
TLI.LowerVAArgNext(true, getRoot(), getValue(I.getOperand(0)),
|
|
||||||
I.getArgType(), DAG);
|
|
||||||
setValue(&I, Result.first);
|
|
||||||
DAG.setRoot(Result.second);
|
|
||||||
}
|
|
||||||
|
|
||||||
void SelectionDAGLowering::visitVAEnd(CallInst &I) {
|
void SelectionDAGLowering::visitVAEnd(CallInst &I) {
|
||||||
DAG.setRoot(TLI.LowerVAEnd(getRoot(), getValue(I.getOperand(1)), DAG));
|
DAG.setRoot(TLI.LowerVAEnd(getRoot(), getValue(I.getOperand(1)), DAG));
|
||||||
}
|
}
|
||||||
|
|
||||||
void SelectionDAGLowering::visitVACopy(CallInst &I) {
|
void SelectionDAGLowering::visitVACopy(CallInst &I) {
|
||||||
std::pair<SDOperand,SDOperand> Result =
|
std::pair<SDOperand,SDOperand> Result =
|
||||||
TLI.LowerVACopy(getRoot(), getValue(I.getOperand(1)), DAG);
|
TLI.LowerVACopy(getRoot(), getValue(I.getOperand(2)), getValue(I.getOperand(1)), DAG);
|
||||||
setValue(&I, Result.first);
|
setValue(&I, Result.first);
|
||||||
DAG.setRoot(Result.second);
|
DAG.setRoot(Result.second);
|
||||||
}
|
}
|
||||||
|
@ -988,18 +988,6 @@ void Interpreter::visitCastInst(CastInst &I) {
|
|||||||
SetValue(&I, executeCastOperation(I.getOperand(0), I.getType(), SF), SF);
|
SetValue(&I, executeCastOperation(I.getOperand(0), I.getType(), SF), SF);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Interpreter::visitVANextInst(VANextInst &I) {
|
|
||||||
ExecutionContext &SF = ECStack.back();
|
|
||||||
|
|
||||||
// Get the incoming valist parameter. LLI treats the valist as a
|
|
||||||
// (ec-stack-depth var-arg-index) pair.
|
|
||||||
GenericValue VAList = getOperandValue(I.getOperand(0), SF);
|
|
||||||
|
|
||||||
// Move the pointer to the next vararg.
|
|
||||||
++VAList.UIntPairVal.second;
|
|
||||||
SetValue(&I, VAList, SF);
|
|
||||||
}
|
|
||||||
|
|
||||||
#define IMPLEMENT_VAARG(TY) \
|
#define IMPLEMENT_VAARG(TY) \
|
||||||
case Type::TY##TyID: Dest.TY##Val = Src.TY##Val; break
|
case Type::TY##TyID: Dest.TY##Val = Src.TY##Val; break
|
||||||
|
|
||||||
@ -1033,6 +1021,9 @@ void Interpreter::visitVAArgInst(VAArgInst &I) {
|
|||||||
|
|
||||||
// Set the Value of this Instruction.
|
// Set the Value of this Instruction.
|
||||||
SetValue(&I, Dest, SF);
|
SetValue(&I, Dest, SF);
|
||||||
|
|
||||||
|
// Move the pointer to the next vararg.
|
||||||
|
++VAList.UIntPairVal.second;
|
||||||
}
|
}
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
@ -153,7 +153,6 @@ public:
|
|||||||
|
|
||||||
void visitShl(ShiftInst &I);
|
void visitShl(ShiftInst &I);
|
||||||
void visitShr(ShiftInst &I);
|
void visitShr(ShiftInst &I);
|
||||||
void visitVANextInst(VANextInst &I);
|
|
||||||
void visitVAArgInst(VAArgInst &I);
|
void visitVAArgInst(VAArgInst &I);
|
||||||
void visitInstruction(Instruction &I) {
|
void visitInstruction(Instruction &I) {
|
||||||
std::cerr << I;
|
std::cerr << I;
|
||||||
|
@ -72,7 +72,8 @@ namespace {
|
|||||||
// AlphaTargetLowering - Alpha Implementation of the TargetLowering interface
|
// AlphaTargetLowering - Alpha Implementation of the TargetLowering interface
|
||||||
namespace {
|
namespace {
|
||||||
class AlphaTargetLowering : public TargetLowering {
|
class AlphaTargetLowering : public TargetLowering {
|
||||||
int VarArgsFrameIndex; // FrameIndex for start of varargs area.
|
int VarArgsOffset; // What is the offset to the first vaarg
|
||||||
|
int VarArgsBase; // What is the base FrameIndex
|
||||||
unsigned GP; //GOT vreg
|
unsigned GP; //GOT vreg
|
||||||
public:
|
public:
|
||||||
AlphaTargetLowering(TargetMachine &TM) : TargetLowering(TM) {
|
AlphaTargetLowering(TargetMachine &TM) : TargetLowering(TM) {
|
||||||
@ -151,10 +152,10 @@ namespace {
|
|||||||
SelectionDAG &DAG);
|
SelectionDAG &DAG);
|
||||||
|
|
||||||
virtual std::pair<SDOperand, SDOperand>
|
virtual std::pair<SDOperand, SDOperand>
|
||||||
LowerVAStart(SDOperand Chain, SelectionDAG &DAG);
|
LowerVAStart(SDOperand Chain, SelectionDAG &DAG, SDOperand Dest);
|
||||||
|
|
||||||
virtual std::pair<SDOperand,SDOperand>
|
virtual std::pair<SDOperand,SDOperand>
|
||||||
LowerVAArgNext(bool isVANext, SDOperand Chain, SDOperand VAList,
|
LowerVAArgNext(SDOperand Chain, SDOperand VAList,
|
||||||
const Type *ArgTy, SelectionDAG &DAG);
|
const Type *ArgTy, SelectionDAG &DAG);
|
||||||
|
|
||||||
virtual std::pair<SDOperand, SDOperand>
|
virtual std::pair<SDOperand, SDOperand>
|
||||||
@ -300,12 +301,14 @@ AlphaTargetLowering::LowerArguments(Function &F, SelectionDAG &DAG)
|
|||||||
|
|
||||||
// If the functions takes variable number of arguments, copy all regs to stack
|
// If the functions takes variable number of arguments, copy all regs to stack
|
||||||
if (F.isVarArg()) {
|
if (F.isVarArg()) {
|
||||||
|
VarArgsOffset = count * 8;
|
||||||
std::vector<SDOperand> LS;
|
std::vector<SDOperand> LS;
|
||||||
for (int i = 0; i < 6; ++i) {
|
for (int i = 0; i < 6; ++i) {
|
||||||
if (args_int[i] < 1024)
|
if (args_int[i] < 1024)
|
||||||
args_int[i] = AddLiveIn(MF,args_int[i], getRegClassFor(MVT::i64));
|
args_int[i] = AddLiveIn(MF,args_int[i], getRegClassFor(MVT::i64));
|
||||||
SDOperand argt = DAG.getCopyFromReg(args_int[i], MVT::i64, DAG.getRoot());
|
SDOperand argt = DAG.getCopyFromReg(args_int[i], MVT::i64, DAG.getRoot());
|
||||||
int FI = MFI->CreateFixedObject(8, -8 * (6 - i));
|
int FI = MFI->CreateFixedObject(8, -8 * (6 - i));
|
||||||
|
if (i == 0) VarArgsBase = FI;
|
||||||
SDOperand SDFI = DAG.getFrameIndex(FI, MVT::i64);
|
SDOperand SDFI = DAG.getFrameIndex(FI, MVT::i64);
|
||||||
LS.push_back(DAG.getNode(ISD::STORE, MVT::Other, DAG.getRoot(), argt, SDFI, DAG.getSrcValue(NULL)));
|
LS.push_back(DAG.getNode(ISD::STORE, MVT::Other, DAG.getRoot(), argt, SDFI, DAG.getSrcValue(NULL)));
|
||||||
|
|
||||||
@ -393,15 +396,34 @@ AlphaTargetLowering::LowerCallTo(SDOperand Chain,
|
|||||||
}
|
}
|
||||||
|
|
||||||
std::pair<SDOperand, SDOperand>
|
std::pair<SDOperand, SDOperand>
|
||||||
AlphaTargetLowering::LowerVAStart(SDOperand Chain, SelectionDAG &DAG) {
|
AlphaTargetLowering::LowerVAStart(SDOperand Chain, SelectionDAG &DAG, SDOperand Dest) {
|
||||||
//vastart just returns the address of the VarArgsFrameIndex slot.
|
// vastart just stores the address of the VarArgsBase and VarArgsOffset
|
||||||
return std::make_pair(DAG.getFrameIndex(VarArgsFrameIndex, MVT::i64), Chain);
|
SDOperand FR = DAG.getFrameIndex(VarArgsBase, MVT::i32);
|
||||||
|
SDOperand S1 = DAG.getNode(ISD::STORE, MVT::Other, Chain, FR, Dest, DAG.getSrcValue(NULL));
|
||||||
|
SDOperand SA2 = DAG.getNode(ISD::ADD, MVT::i64, Dest, DAG.getConstant(8, MVT::i64));
|
||||||
|
SDOperand S2 = DAG.getNode(ISD::STORE, MVT::Other, S1,
|
||||||
|
DAG.getConstant(VarArgsOffset, MVT::i64), SA2,
|
||||||
|
DAG.getSrcValue(NULL));
|
||||||
|
|
||||||
|
return std::make_pair(S2, S2);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::pair<SDOperand,SDOperand> AlphaTargetLowering::
|
std::pair<SDOperand,SDOperand> AlphaTargetLowering::
|
||||||
LowerVAArgNext(bool isVANext, SDOperand Chain, SDOperand VAList,
|
LowerVAArgNext(SDOperand Chain, SDOperand VAList,
|
||||||
const Type *ArgTy, SelectionDAG &DAG) {
|
const Type *ArgTy, SelectionDAG &DAG) {
|
||||||
abort();
|
//FIXME: For now, ignore FP
|
||||||
|
SDOperand Base = DAG.getLoad(MVT::i64, Chain, VAList, DAG.getSrcValue(NULL));
|
||||||
|
SDOperand Tmp = DAG.getNode(ISD::ADD, MVT::i64, VAList,
|
||||||
|
DAG.getConstant(8, MVT::i64));
|
||||||
|
SDOperand Offset = DAG.getLoad(MVT::i64, Chain, Tmp, DAG.getSrcValue(NULL));
|
||||||
|
SDOperand DataPtr = DAG.getNode(ISD::ADD, MVT::i64, Base, Offset);
|
||||||
|
SDOperand Result = DAG.getLoad(MVT::i64, Chain, DataPtr,
|
||||||
|
DAG.getSrcValue(NULL));
|
||||||
|
SDOperand NewOffset = DAG.getNode(ISD::ADD, MVT::i64, Offset,
|
||||||
|
DAG.getConstant(8, MVT::i64));
|
||||||
|
SDOperand Update = DAG.getNode(ISD::STORE, MVT::Other, Result, NewOffset,
|
||||||
|
Tmp, DAG.getSrcValue(NULL));
|
||||||
|
return std::make_pair(Result, Update);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -145,7 +145,7 @@ namespace {
|
|||||||
// emit it inline where it would go.
|
// emit it inline where it would go.
|
||||||
if (I.getType() == Type::VoidTy || !I.hasOneUse() ||
|
if (I.getType() == Type::VoidTy || !I.hasOneUse() ||
|
||||||
isa<TerminatorInst>(I) || isa<CallInst>(I) || isa<PHINode>(I) ||
|
isa<TerminatorInst>(I) || isa<CallInst>(I) || isa<PHINode>(I) ||
|
||||||
isa<LoadInst>(I) || isa<VAArgInst>(I) || isa<VANextInst>(I))
|
isa<LoadInst>(I) || isa<VAArgInst>(I))
|
||||||
// Don't inline a load across a store or other bad things!
|
// Don't inline a load across a store or other bad things!
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
@ -196,7 +196,6 @@ namespace {
|
|||||||
void visitLoadInst (LoadInst &I);
|
void visitLoadInst (LoadInst &I);
|
||||||
void visitStoreInst (StoreInst &I);
|
void visitStoreInst (StoreInst &I);
|
||||||
void visitGetElementPtrInst(GetElementPtrInst &I);
|
void visitGetElementPtrInst(GetElementPtrInst &I);
|
||||||
void visitVANextInst(VANextInst &I);
|
|
||||||
void visitVAArgInst (VAArgInst &I);
|
void visitVAArgInst (VAArgInst &I);
|
||||||
|
|
||||||
void visitInstruction(Instruction &I) {
|
void visitInstruction(Instruction &I) {
|
||||||
@ -1469,7 +1468,10 @@ void CWriter::visitCallInst(CallInst &I) {
|
|||||||
case Intrinsic::vastart:
|
case Intrinsic::vastart:
|
||||||
Out << "0; ";
|
Out << "0; ";
|
||||||
|
|
||||||
Out << "va_start(*(va_list*)&" << Mang->getValueName(&I) << ", ";
|
// Out << "va_start(*(va_list*)&" << Mang->getValueName(&I) << ", ";
|
||||||
|
Out << "va_start(*(va_list*)";
|
||||||
|
writeOperand(I.getOperand(1));
|
||||||
|
Out << ", ";
|
||||||
// Output the last argument to the enclosing function...
|
// Output the last argument to the enclosing function...
|
||||||
if (I.getParent()->getParent()->arg_empty()) {
|
if (I.getParent()->getParent()->arg_empty()) {
|
||||||
std::cerr << "The C backend does not currently support zero "
|
std::cerr << "The C backend does not currently support zero "
|
||||||
@ -1482,7 +1484,7 @@ void CWriter::visitCallInst(CallInst &I) {
|
|||||||
return;
|
return;
|
||||||
case Intrinsic::vaend:
|
case Intrinsic::vaend:
|
||||||
if (!isa<ConstantPointerNull>(I.getOperand(1))) {
|
if (!isa<ConstantPointerNull>(I.getOperand(1))) {
|
||||||
Out << "va_end(*(va_list*)&";
|
Out << "0; va_end(*(va_list*)";
|
||||||
writeOperand(I.getOperand(1));
|
writeOperand(I.getOperand(1));
|
||||||
Out << ')';
|
Out << ')';
|
||||||
} else {
|
} else {
|
||||||
@ -1491,9 +1493,10 @@ void CWriter::visitCallInst(CallInst &I) {
|
|||||||
return;
|
return;
|
||||||
case Intrinsic::vacopy:
|
case Intrinsic::vacopy:
|
||||||
Out << "0; ";
|
Out << "0; ";
|
||||||
Out << "va_copy(*(va_list*)&" << Mang->getValueName(&I) << ", ";
|
Out << "va_copy(*(va_list*)";
|
||||||
Out << "*(va_list*)&";
|
|
||||||
writeOperand(I.getOperand(1));
|
writeOperand(I.getOperand(1));
|
||||||
|
Out << ", *(va_list*)&";
|
||||||
|
writeOperand(I.getOperand(2));
|
||||||
Out << ')';
|
Out << ')';
|
||||||
return;
|
return;
|
||||||
case Intrinsic::returnaddress:
|
case Intrinsic::returnaddress:
|
||||||
@ -1710,20 +1713,12 @@ void CWriter::visitGetElementPtrInst(GetElementPtrInst &I) {
|
|||||||
gep_type_end(I));
|
gep_type_end(I));
|
||||||
}
|
}
|
||||||
|
|
||||||
void CWriter::visitVANextInst(VANextInst &I) {
|
|
||||||
Out << Mang->getValueName(I.getOperand(0));
|
|
||||||
Out << "; va_arg(*(va_list*)&" << Mang->getValueName(&I) << ", ";
|
|
||||||
printType(Out, I.getArgType());
|
|
||||||
Out << ')';
|
|
||||||
}
|
|
||||||
|
|
||||||
void CWriter::visitVAArgInst(VAArgInst &I) {
|
void CWriter::visitVAArgInst(VAArgInst &I) {
|
||||||
Out << "0;\n";
|
Out << "va_arg(*(va_list*)";
|
||||||
Out << "{ va_list Tmp; va_copy(Tmp, *(va_list*)&";
|
|
||||||
writeOperand(I.getOperand(0));
|
writeOperand(I.getOperand(0));
|
||||||
Out << ");\n " << Mang->getValueName(&I) << " = va_arg(Tmp, ";
|
Out << ", ";
|
||||||
printType(Out, I.getType());
|
printType(Out, I.getType());
|
||||||
Out << ");\n va_end(Tmp); }";
|
Out << ");\n ";
|
||||||
}
|
}
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
@ -145,7 +145,7 @@ namespace {
|
|||||||
// emit it inline where it would go.
|
// emit it inline where it would go.
|
||||||
if (I.getType() == Type::VoidTy || !I.hasOneUse() ||
|
if (I.getType() == Type::VoidTy || !I.hasOneUse() ||
|
||||||
isa<TerminatorInst>(I) || isa<CallInst>(I) || isa<PHINode>(I) ||
|
isa<TerminatorInst>(I) || isa<CallInst>(I) || isa<PHINode>(I) ||
|
||||||
isa<LoadInst>(I) || isa<VAArgInst>(I) || isa<VANextInst>(I))
|
isa<LoadInst>(I) || isa<VAArgInst>(I))
|
||||||
// Don't inline a load across a store or other bad things!
|
// Don't inline a load across a store or other bad things!
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
@ -196,7 +196,6 @@ namespace {
|
|||||||
void visitLoadInst (LoadInst &I);
|
void visitLoadInst (LoadInst &I);
|
||||||
void visitStoreInst (StoreInst &I);
|
void visitStoreInst (StoreInst &I);
|
||||||
void visitGetElementPtrInst(GetElementPtrInst &I);
|
void visitGetElementPtrInst(GetElementPtrInst &I);
|
||||||
void visitVANextInst(VANextInst &I);
|
|
||||||
void visitVAArgInst (VAArgInst &I);
|
void visitVAArgInst (VAArgInst &I);
|
||||||
|
|
||||||
void visitInstruction(Instruction &I) {
|
void visitInstruction(Instruction &I) {
|
||||||
@ -1469,7 +1468,10 @@ void CWriter::visitCallInst(CallInst &I) {
|
|||||||
case Intrinsic::vastart:
|
case Intrinsic::vastart:
|
||||||
Out << "0; ";
|
Out << "0; ";
|
||||||
|
|
||||||
Out << "va_start(*(va_list*)&" << Mang->getValueName(&I) << ", ";
|
// Out << "va_start(*(va_list*)&" << Mang->getValueName(&I) << ", ";
|
||||||
|
Out << "va_start(*(va_list*)";
|
||||||
|
writeOperand(I.getOperand(1));
|
||||||
|
Out << ", ";
|
||||||
// Output the last argument to the enclosing function...
|
// Output the last argument to the enclosing function...
|
||||||
if (I.getParent()->getParent()->arg_empty()) {
|
if (I.getParent()->getParent()->arg_empty()) {
|
||||||
std::cerr << "The C backend does not currently support zero "
|
std::cerr << "The C backend does not currently support zero "
|
||||||
@ -1482,7 +1484,7 @@ void CWriter::visitCallInst(CallInst &I) {
|
|||||||
return;
|
return;
|
||||||
case Intrinsic::vaend:
|
case Intrinsic::vaend:
|
||||||
if (!isa<ConstantPointerNull>(I.getOperand(1))) {
|
if (!isa<ConstantPointerNull>(I.getOperand(1))) {
|
||||||
Out << "va_end(*(va_list*)&";
|
Out << "0; va_end(*(va_list*)";
|
||||||
writeOperand(I.getOperand(1));
|
writeOperand(I.getOperand(1));
|
||||||
Out << ')';
|
Out << ')';
|
||||||
} else {
|
} else {
|
||||||
@ -1491,9 +1493,10 @@ void CWriter::visitCallInst(CallInst &I) {
|
|||||||
return;
|
return;
|
||||||
case Intrinsic::vacopy:
|
case Intrinsic::vacopy:
|
||||||
Out << "0; ";
|
Out << "0; ";
|
||||||
Out << "va_copy(*(va_list*)&" << Mang->getValueName(&I) << ", ";
|
Out << "va_copy(*(va_list*)";
|
||||||
Out << "*(va_list*)&";
|
|
||||||
writeOperand(I.getOperand(1));
|
writeOperand(I.getOperand(1));
|
||||||
|
Out << ", *(va_list*)&";
|
||||||
|
writeOperand(I.getOperand(2));
|
||||||
Out << ')';
|
Out << ')';
|
||||||
return;
|
return;
|
||||||
case Intrinsic::returnaddress:
|
case Intrinsic::returnaddress:
|
||||||
@ -1710,20 +1713,12 @@ void CWriter::visitGetElementPtrInst(GetElementPtrInst &I) {
|
|||||||
gep_type_end(I));
|
gep_type_end(I));
|
||||||
}
|
}
|
||||||
|
|
||||||
void CWriter::visitVANextInst(VANextInst &I) {
|
|
||||||
Out << Mang->getValueName(I.getOperand(0));
|
|
||||||
Out << "; va_arg(*(va_list*)&" << Mang->getValueName(&I) << ", ";
|
|
||||||
printType(Out, I.getArgType());
|
|
||||||
Out << ')';
|
|
||||||
}
|
|
||||||
|
|
||||||
void CWriter::visitVAArgInst(VAArgInst &I) {
|
void CWriter::visitVAArgInst(VAArgInst &I) {
|
||||||
Out << "0;\n";
|
Out << "va_arg(*(va_list*)";
|
||||||
Out << "{ va_list Tmp; va_copy(Tmp, *(va_list*)&";
|
|
||||||
writeOperand(I.getOperand(0));
|
writeOperand(I.getOperand(0));
|
||||||
Out << ");\n " << Mang->getValueName(&I) << " = va_arg(Tmp, ";
|
Out << ", ";
|
||||||
printType(Out, I.getType());
|
printType(Out, I.getType());
|
||||||
Out << ");\n va_end(Tmp); }";
|
Out << ");\n ";
|
||||||
}
|
}
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
@ -114,12 +114,16 @@ namespace {
|
|||||||
SelectionDAG &DAG);
|
SelectionDAG &DAG);
|
||||||
|
|
||||||
virtual std::pair<SDOperand, SDOperand>
|
virtual std::pair<SDOperand, SDOperand>
|
||||||
LowerVAStart(SDOperand Chain, SelectionDAG &DAG);
|
LowerVAStart(SDOperand Chain, SelectionDAG &DAG, SDOperand Dest);
|
||||||
|
|
||||||
virtual std::pair<SDOperand,SDOperand>
|
virtual std::pair<SDOperand,SDOperand>
|
||||||
LowerVAArgNext(bool isVANext, SDOperand Chain, SDOperand VAList,
|
LowerVAArgNext(SDOperand Chain, SDOperand VAList,
|
||||||
const Type *ArgTy, SelectionDAG &DAG);
|
const Type *ArgTy, SelectionDAG &DAG);
|
||||||
|
|
||||||
|
virtual std::pair<SDOperand,SDOperand>
|
||||||
|
LowerVACopy(SDOperand Chain, SDOperand Src, SDOperand Dest,
|
||||||
|
SelectionDAG &DAG);
|
||||||
|
|
||||||
virtual std::pair<SDOperand, SDOperand>
|
virtual std::pair<SDOperand, SDOperand>
|
||||||
LowerFrameReturnAddress(bool isFrameAddr, SDOperand Chain, unsigned Depth,
|
LowerFrameReturnAddress(bool isFrameAddr, SDOperand Chain, unsigned Depth,
|
||||||
SelectionDAG &DAG);
|
SelectionDAG &DAG);
|
||||||
@ -380,20 +384,20 @@ IA64TargetLowering::LowerCallTo(SDOperand Chain,
|
|||||||
}
|
}
|
||||||
|
|
||||||
std::pair<SDOperand, SDOperand>
|
std::pair<SDOperand, SDOperand>
|
||||||
IA64TargetLowering::LowerVAStart(SDOperand Chain, SelectionDAG &DAG) {
|
IA64TargetLowering::LowerVAStart(SDOperand Chain, SelectionDAG &DAG, SDOperand Dest) {
|
||||||
// vastart just returns the address of the VarArgsFrameIndex slot.
|
// vastart just stores the address of the VarArgsFrameIndex slot.
|
||||||
return std::make_pair(DAG.getFrameIndex(VarArgsFrameIndex, MVT::i64), Chain);
|
SDOperand FR = DAG.getFrameIndex(VarArgsFrameIndex, MVT::i64);
|
||||||
|
SDOperand Result = DAG.getNode(ISD::STORE, MVT::Other, Chain, FR, Dest, DAG.getSrcValue(NULL));
|
||||||
|
return std::make_pair(Result, Result);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::pair<SDOperand,SDOperand> IA64TargetLowering::
|
std::pair<SDOperand,SDOperand> IA64TargetLowering::
|
||||||
LowerVAArgNext(bool isVANext, SDOperand Chain, SDOperand VAList,
|
LowerVAArgNext(SDOperand Chain, SDOperand VAList,
|
||||||
const Type *ArgTy, SelectionDAG &DAG) {
|
const Type *ArgTy, SelectionDAG &DAG) {
|
||||||
|
|
||||||
MVT::ValueType ArgVT = getValueType(ArgTy);
|
MVT::ValueType ArgVT = getValueType(ArgTy);
|
||||||
SDOperand Result;
|
SDOperand Val = DAG.getLoad(MVT::i64, Chain, VAList, DAG.getSrcValue(NULL));
|
||||||
if (!isVANext) {
|
SDOperand Result = DAG.getLoad(ArgVT, DAG.getEntryNode(), Val, DAG.getSrcValue(NULL));
|
||||||
Result = DAG.getLoad(ArgVT, DAG.getEntryNode(), VAList, DAG.getSrcValue(NULL));
|
|
||||||
} else {
|
|
||||||
unsigned Amt;
|
unsigned Amt;
|
||||||
if (ArgVT == MVT::i32 || ArgVT == MVT::f32)
|
if (ArgVT == MVT::i32 || ArgVT == MVT::f32)
|
||||||
Amt = 8;
|
Amt = 8;
|
||||||
@ -402,12 +406,22 @@ LowerVAArgNext(bool isVANext, SDOperand Chain, SDOperand VAList,
|
|||||||
"Other types should have been promoted for varargs!");
|
"Other types should have been promoted for varargs!");
|
||||||
Amt = 8;
|
Amt = 8;
|
||||||
}
|
}
|
||||||
Result = DAG.getNode(ISD::ADD, VAList.getValueType(), VAList,
|
Val = DAG.getNode(ISD::ADD, Val.getValueType(), Val,
|
||||||
DAG.getConstant(Amt, VAList.getValueType()));
|
DAG.getConstant(Amt, Val.getValueType()));
|
||||||
}
|
Chain = DAG.getNode(ISD::STORE, MVT::Other, Chain,
|
||||||
|
Val, VAList, DAG.getSrcValue(NULL));
|
||||||
return std::make_pair(Result, Chain);
|
return std::make_pair(Result, Chain);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::pair<SDOperand,SDOperand>
|
||||||
|
IA64TargetLowering::LowerVACopy(SDOperand Chain, SDOperand Src,
|
||||||
|
SDOperand Dest, SelectionDAG &DAG)
|
||||||
|
{
|
||||||
|
SDOperand Result = DAG.getNode(ISD::STORE, MVT::Other, Chain,
|
||||||
|
Src, Dest, DAG.getSrcValue(NULL));
|
||||||
|
return std::make_pair(Result, Result);
|
||||||
|
}
|
||||||
|
|
||||||
std::pair<SDOperand, SDOperand> IA64TargetLowering::
|
std::pair<SDOperand, SDOperand> IA64TargetLowering::
|
||||||
LowerFrameReturnAddress(bool isFrameAddress, SDOperand Chain, unsigned Depth,
|
LowerFrameReturnAddress(bool isFrameAddress, SDOperand Chain, unsigned Depth,
|
||||||
SelectionDAG &DAG) {
|
SelectionDAG &DAG) {
|
||||||
|
@ -310,7 +310,6 @@ namespace {
|
|||||||
void visitShiftInst(ShiftInst &I);
|
void visitShiftInst(ShiftInst &I);
|
||||||
void visitPHINode(PHINode &I) {} // PHI nodes handled by second pass
|
void visitPHINode(PHINode &I) {} // PHI nodes handled by second pass
|
||||||
void visitCastInst(CastInst &I);
|
void visitCastInst(CastInst &I);
|
||||||
void visitVANextInst(VANextInst &I);
|
|
||||||
void visitVAArgInst(VAArgInst &I);
|
void visitVAArgInst(VAArgInst &I);
|
||||||
|
|
||||||
void visitInstruction(Instruction &I) {
|
void visitInstruction(Instruction &I) {
|
||||||
@ -1978,6 +1977,7 @@ void PPC32ISel::visitIntrinsicCall(Intrinsic::ID ID, CallInst &CI) {
|
|||||||
unsigned TmpReg1, TmpReg2, TmpReg3;
|
unsigned TmpReg1, TmpReg2, TmpReg3;
|
||||||
switch (ID) {
|
switch (ID) {
|
||||||
case Intrinsic::vastart:
|
case Intrinsic::vastart:
|
||||||
|
//FIXME: need to store, not return a value
|
||||||
// Get the address of the first vararg value...
|
// Get the address of the first vararg value...
|
||||||
TmpReg1 = getReg(CI);
|
TmpReg1 = getReg(CI);
|
||||||
addFrameReference(BuildMI(BB, PPC::ADDI, 2, TmpReg1), VarArgsFrameIndex,
|
addFrameReference(BuildMI(BB, PPC::ADDI, 2, TmpReg1), VarArgsFrameIndex,
|
||||||
@ -1985,6 +1985,7 @@ void PPC32ISel::visitIntrinsicCall(Intrinsic::ID ID, CallInst &CI) {
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
case Intrinsic::vacopy:
|
case Intrinsic::vacopy:
|
||||||
|
//FIXME: need to store into first arg the value of the second
|
||||||
TmpReg1 = getReg(CI);
|
TmpReg1 = getReg(CI);
|
||||||
TmpReg2 = getReg(CI.getOperand(1));
|
TmpReg2 = getReg(CI.getOperand(1));
|
||||||
BuildMI(BB, PPC::OR, 2, TmpReg1).addReg(TmpReg2).addReg(TmpReg2);
|
BuildMI(BB, PPC::OR, 2, TmpReg1).addReg(TmpReg2).addReg(TmpReg2);
|
||||||
@ -3679,37 +3680,12 @@ void PPC32ISel::emitCastOperation(MachineBasicBlock *MBB,
|
|||||||
abort();
|
abort();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// visitVANextInst - Implement the va_next instruction...
|
|
||||||
///
|
|
||||||
void PPC32ISel::visitVANextInst(VANextInst &I) {
|
|
||||||
unsigned VAList = getReg(I.getOperand(0));
|
|
||||||
unsigned DestReg = getReg(I);
|
|
||||||
|
|
||||||
unsigned Size;
|
|
||||||
switch (I.getArgType()->getTypeID()) {
|
|
||||||
default:
|
|
||||||
std::cerr << I;
|
|
||||||
assert(0 && "Error: bad type for va_next instruction!");
|
|
||||||
return;
|
|
||||||
case Type::PointerTyID:
|
|
||||||
case Type::UIntTyID:
|
|
||||||
case Type::IntTyID:
|
|
||||||
Size = 4;
|
|
||||||
break;
|
|
||||||
case Type::ULongTyID:
|
|
||||||
case Type::LongTyID:
|
|
||||||
case Type::DoubleTyID:
|
|
||||||
Size = 8;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Increment the VAList pointer...
|
|
||||||
BuildMI(BB, PPC::ADDI, 2, DestReg).addReg(VAList).addSImm(Size);
|
|
||||||
}
|
|
||||||
|
|
||||||
void PPC32ISel::visitVAArgInst(VAArgInst &I) {
|
void PPC32ISel::visitVAArgInst(VAArgInst &I) {
|
||||||
unsigned VAList = getReg(I.getOperand(0));
|
unsigned VAListPtr = getReg(I.getOperand(0));
|
||||||
unsigned DestReg = getReg(I);
|
unsigned DestReg = getReg(I);
|
||||||
|
unsigned VAList = makeAnotherReg(Type::IntTy);
|
||||||
|
BuildMI(BB, PPC::LWZ, 2, VAList).addSImm(0).addReg(VAListPtr);
|
||||||
|
int Size;
|
||||||
|
|
||||||
switch (I.getType()->getTypeID()) {
|
switch (I.getType()->getTypeID()) {
|
||||||
default:
|
default:
|
||||||
@ -3719,20 +3695,28 @@ void PPC32ISel::visitVAArgInst(VAArgInst &I) {
|
|||||||
case Type::PointerTyID:
|
case Type::PointerTyID:
|
||||||
case Type::UIntTyID:
|
case Type::UIntTyID:
|
||||||
case Type::IntTyID:
|
case Type::IntTyID:
|
||||||
|
Size = 4;
|
||||||
BuildMI(BB, PPC::LWZ, 2, DestReg).addSImm(0).addReg(VAList);
|
BuildMI(BB, PPC::LWZ, 2, DestReg).addSImm(0).addReg(VAList);
|
||||||
break;
|
break;
|
||||||
case Type::ULongTyID:
|
case Type::ULongTyID:
|
||||||
case Type::LongTyID:
|
case Type::LongTyID:
|
||||||
|
Size = 8;
|
||||||
BuildMI(BB, PPC::LWZ, 2, DestReg).addSImm(0).addReg(VAList);
|
BuildMI(BB, PPC::LWZ, 2, DestReg).addSImm(0).addReg(VAList);
|
||||||
BuildMI(BB, PPC::LWZ, 2, DestReg+1).addSImm(4).addReg(VAList);
|
BuildMI(BB, PPC::LWZ, 2, DestReg+1).addSImm(4).addReg(VAList);
|
||||||
break;
|
break;
|
||||||
case Type::FloatTyID:
|
case Type::FloatTyID:
|
||||||
|
Size = 4; //?? Bad value?
|
||||||
BuildMI(BB, PPC::LFS, 2, DestReg).addSImm(0).addReg(VAList);
|
BuildMI(BB, PPC::LFS, 2, DestReg).addSImm(0).addReg(VAList);
|
||||||
break;
|
break;
|
||||||
case Type::DoubleTyID:
|
case Type::DoubleTyID:
|
||||||
|
Size = 8;
|
||||||
BuildMI(BB, PPC::LFD, 2, DestReg).addSImm(0).addReg(VAList);
|
BuildMI(BB, PPC::LFD, 2, DestReg).addSImm(0).addReg(VAList);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
// Increment the VAList pointer...
|
||||||
|
unsigned NP = makeAnotherReg(Type::IntTy);
|
||||||
|
BuildMI(BB, PPC::ADDI, 2, NP).addReg(VAList).addSImm(Size);
|
||||||
|
BuildMI(BB, PPC::STW, 3).addReg(NP).addSImm(0).addReg(VAListPtr);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// visitGetElementPtrInst - instruction-select GEP instructions
|
/// visitGetElementPtrInst - instruction-select GEP instructions
|
||||||
|
@ -98,12 +98,16 @@ namespace {
|
|||||||
SelectionDAG &DAG);
|
SelectionDAG &DAG);
|
||||||
|
|
||||||
virtual std::pair<SDOperand, SDOperand>
|
virtual std::pair<SDOperand, SDOperand>
|
||||||
LowerVAStart(SDOperand Chain, SelectionDAG &DAG);
|
LowerVAStart(SDOperand Chain, SelectionDAG &DAG, SDOperand Dest);
|
||||||
|
|
||||||
virtual std::pair<SDOperand,SDOperand>
|
virtual std::pair<SDOperand,SDOperand>
|
||||||
LowerVAArgNext(bool isVANext, SDOperand Chain, SDOperand VAList,
|
LowerVAArgNext(SDOperand Chain, SDOperand VAList,
|
||||||
const Type *ArgTy, SelectionDAG &DAG);
|
const Type *ArgTy, SelectionDAG &DAG);
|
||||||
|
|
||||||
|
virtual std::pair<SDOperand,SDOperand>
|
||||||
|
LowerVACopy(SDOperand Chain, SDOperand Src, SDOperand Dest,
|
||||||
|
SelectionDAG &DAG);
|
||||||
|
|
||||||
virtual std::pair<SDOperand, SDOperand>
|
virtual std::pair<SDOperand, SDOperand>
|
||||||
LowerFrameReturnAddress(bool isFrameAddr, SDOperand Chain, unsigned Depth,
|
LowerFrameReturnAddress(bool isFrameAddr, SDOperand Chain, unsigned Depth,
|
||||||
SelectionDAG &DAG);
|
SelectionDAG &DAG);
|
||||||
@ -365,26 +369,36 @@ PPC64TargetLowering::LowerCallTo(SDOperand Chain,
|
|||||||
}
|
}
|
||||||
|
|
||||||
std::pair<SDOperand, SDOperand>
|
std::pair<SDOperand, SDOperand>
|
||||||
PPC64TargetLowering::LowerVAStart(SDOperand Chain, SelectionDAG &DAG) {
|
PPC64TargetLowering::LowerVAStart(SDOperand Chain, SelectionDAG &DAG, SDOperand Dest) {
|
||||||
//vastart just returns the address of the VarArgsFrameIndex slot.
|
// vastart just stores the address of the VarArgsFrameIndex slot.
|
||||||
return std::make_pair(DAG.getFrameIndex(VarArgsFrameIndex, MVT::i64), Chain);
|
SDOperand FR = DAG.getFrameIndex(VarArgsFrameIndex, MVT::i64);
|
||||||
|
SDOperand Result = DAG.getNode(ISD::STORE, MVT::Other, Chain, FR, Dest, DAG.getSrcValue(NULL));
|
||||||
|
return std::make_pair(Result, Result);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::pair<SDOperand,SDOperand> PPC64TargetLowering::
|
std::pair<SDOperand,SDOperand> PPC64TargetLowering::
|
||||||
LowerVAArgNext(bool isVANext, SDOperand Chain, SDOperand VAList,
|
LowerVAArgNext(SDOperand Chain, SDOperand VAList,
|
||||||
const Type *ArgTy, SelectionDAG &DAG) {
|
const Type *ArgTy, SelectionDAG &DAG) {
|
||||||
MVT::ValueType ArgVT = getValueType(ArgTy);
|
MVT::ValueType ArgVT = getValueType(ArgTy);
|
||||||
SDOperand Result;
|
SDOperand Result;
|
||||||
if (!isVANext) {
|
SDOperand Val = DAG.getLoad(MVT::i64, Chain, VAList, DAG.getSrcValue(NULL));
|
||||||
Result = DAG.getLoad(ArgVT, DAG.getEntryNode(), VAList,
|
Result = DAG.getLoad(ArgVT, Val.getValue(1), Val, DAG.getSrcValue(NULL));
|
||||||
DAG.getSrcValue(NULL));
|
Val = DAG.getNode(ISD::ADD, VAList.getValueType(), Val,
|
||||||
} else {
|
|
||||||
Result = DAG.getNode(ISD::ADD, VAList.getValueType(), VAList,
|
|
||||||
DAG.getConstant(8, VAList.getValueType()));
|
DAG.getConstant(8, VAList.getValueType()));
|
||||||
}
|
Chain = DAG.getNode(ISD::STORE, MVT::Other, Chain,
|
||||||
|
Val, VAList, DAG.getSrcValue(NULL));
|
||||||
return std::make_pair(Result, Chain);
|
return std::make_pair(Result, Chain);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::pair<SDOperand,SDOperand>
|
||||||
|
PPC64TargetLowering::LowerVACopy(SDOperand Chain, SDOperand Src,
|
||||||
|
SDOperand Dest, SelectionDAG &DAG)
|
||||||
|
{
|
||||||
|
SDOperand Result = DAG.getNode(ISD::STORE, MVT::Other, Chain,
|
||||||
|
Src, Dest, DAG.getSrcValue(NULL));
|
||||||
|
return std::make_pair(Result, Result);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
std::pair<SDOperand, SDOperand> PPC64TargetLowering::
|
std::pair<SDOperand, SDOperand> PPC64TargetLowering::
|
||||||
LowerFrameReturnAddress(bool isFrameAddress, SDOperand Chain, unsigned Depth,
|
LowerFrameReturnAddress(bool isFrameAddress, SDOperand Chain, unsigned Depth,
|
||||||
|
@ -102,7 +102,6 @@ namespace {
|
|||||||
void visitBranchInst(BranchInst &I);
|
void visitBranchInst(BranchInst &I);
|
||||||
void visitUnreachableInst(UnreachableInst &I) {}
|
void visitUnreachableInst(UnreachableInst &I) {}
|
||||||
void visitCastInst(CastInst &I);
|
void visitCastInst(CastInst &I);
|
||||||
void visitVANextInst(VANextInst &I);
|
|
||||||
void visitVAArgInst(VAArgInst &I);
|
void visitVAArgInst(VAArgInst &I);
|
||||||
void visitLoadInst(LoadInst &I);
|
void visitLoadInst(LoadInst &I);
|
||||||
void visitStoreInst(StoreInst &I);
|
void visitStoreInst(StoreInst &I);
|
||||||
@ -1754,8 +1753,10 @@ void V8ISel::visitIntrinsicCall(Intrinsic::ID ID, CallInst &CI) {
|
|||||||
|
|
||||||
case Intrinsic::vastart: {
|
case Intrinsic::vastart: {
|
||||||
// Add the VarArgsOffset to the frame pointer, and copy it to the result.
|
// Add the VarArgsOffset to the frame pointer, and copy it to the result.
|
||||||
unsigned DestReg = getReg (CI);
|
unsigned DestReg = getReg (CI.getOperand(1));
|
||||||
BuildMI (BB, V8::ADDri, 2, DestReg).addReg (V8::FP).addSImm (VarArgsOffset);
|
unsigned Tmp = makeAnotherReg(Type::IntTy);
|
||||||
|
BuildMI (BB, V8::ADDri, 2, Tmp).addReg (V8::FP).addSImm (VarArgsOffset);
|
||||||
|
BuildMI(BB, V8::ST, 3).addReg(DestReg).addSImm(0).addReg(Tmp);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1765,39 +1766,37 @@ void V8ISel::visitIntrinsicCall(Intrinsic::ID ID, CallInst &CI) {
|
|||||||
|
|
||||||
case Intrinsic::vacopy: {
|
case Intrinsic::vacopy: {
|
||||||
// Copy the va_list ptr (arg1) to the result.
|
// Copy the va_list ptr (arg1) to the result.
|
||||||
unsigned DestReg = getReg (CI), SrcReg = getReg (CI.getOperand (1));
|
unsigned DestReg = getReg (CI.getOperand(1)), SrcReg = getReg (CI.getOperand (2));
|
||||||
BuildMI (BB, V8::ORrr, 2, DestReg).addReg (V8::G0).addReg (SrcReg);
|
BuildMI(BB, V8::ST, 3).addReg(DestReg).addSImm(0).addReg(SrcReg);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void V8ISel::visitVANextInst (VANextInst &I) {
|
|
||||||
// Add the type size to the vararg pointer (arg0).
|
|
||||||
unsigned DestReg = getReg (I);
|
|
||||||
unsigned SrcReg = getReg (I.getOperand (0));
|
|
||||||
unsigned TySize = TM.getTargetData ().getTypeSize (I.getArgType ());
|
|
||||||
BuildMI (BB, V8::ADDri, 2, DestReg).addReg (SrcReg).addSImm (TySize);
|
|
||||||
}
|
|
||||||
|
|
||||||
void V8ISel::visitVAArgInst (VAArgInst &I) {
|
void V8ISel::visitVAArgInst (VAArgInst &I) {
|
||||||
unsigned VAList = getReg (I.getOperand (0));
|
unsigned VAListPtr = getReg (I.getOperand (0));
|
||||||
unsigned DestReg = getReg (I);
|
unsigned DestReg = getReg (I);
|
||||||
|
unsigned Size;
|
||||||
|
unsigned VAList = makeAnotherReg(Type::IntTy);
|
||||||
|
BuildMI(BB, V8::LD, 2, VAList).addReg(VAListPtr).addSImm(0);
|
||||||
|
|
||||||
switch (I.getType ()->getTypeID ()) {
|
switch (I.getType ()->getTypeID ()) {
|
||||||
case Type::PointerTyID:
|
case Type::PointerTyID:
|
||||||
case Type::UIntTyID:
|
case Type::UIntTyID:
|
||||||
case Type::IntTyID:
|
case Type::IntTyID:
|
||||||
|
Size = 4;
|
||||||
BuildMI (BB, V8::LD, 2, DestReg).addReg (VAList).addSImm (0);
|
BuildMI (BB, V8::LD, 2, DestReg).addReg (VAList).addSImm (0);
|
||||||
return;
|
break;
|
||||||
|
|
||||||
case Type::ULongTyID:
|
case Type::ULongTyID:
|
||||||
case Type::LongTyID:
|
case Type::LongTyID:
|
||||||
|
Size = 8;
|
||||||
BuildMI (BB, V8::LD, 2, DestReg).addReg (VAList).addSImm (0);
|
BuildMI (BB, V8::LD, 2, DestReg).addReg (VAList).addSImm (0);
|
||||||
BuildMI (BB, V8::LD, 2, DestReg+1).addReg (VAList).addSImm (4);
|
BuildMI (BB, V8::LD, 2, DestReg+1).addReg (VAList).addSImm (4);
|
||||||
return;
|
break;
|
||||||
|
|
||||||
case Type::DoubleTyID: {
|
case Type::DoubleTyID: {
|
||||||
|
Size = 8;
|
||||||
unsigned DblAlign = TM.getTargetData().getDoubleAlignment();
|
unsigned DblAlign = TM.getTargetData().getDoubleAlignment();
|
||||||
unsigned TempReg = makeAnotherReg (Type::IntTy);
|
unsigned TempReg = makeAnotherReg (Type::IntTy);
|
||||||
unsigned TempReg2 = makeAnotherReg (Type::IntTy);
|
unsigned TempReg2 = makeAnotherReg (Type::IntTy);
|
||||||
@ -1807,7 +1806,7 @@ void V8ISel::visitVAArgInst (VAArgInst &I) {
|
|||||||
BuildMI (BB, V8::ST, 3).addFrameIndex (FI).addSImm (0).addReg (TempReg);
|
BuildMI (BB, V8::ST, 3).addFrameIndex (FI).addSImm (0).addReg (TempReg);
|
||||||
BuildMI (BB, V8::ST, 3).addFrameIndex (FI).addSImm (4).addReg (TempReg2);
|
BuildMI (BB, V8::ST, 3).addFrameIndex (FI).addSImm (4).addReg (TempReg2);
|
||||||
BuildMI (BB, V8::LDDFri, 2, DestReg).addFrameIndex (FI).addSImm (0);
|
BuildMI (BB, V8::LDDFri, 2, DestReg).addFrameIndex (FI).addSImm (0);
|
||||||
return;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
@ -1816,4 +1815,8 @@ void V8ISel::visitVAArgInst (VAArgInst &I) {
|
|||||||
abort ();
|
abort ();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
unsigned tmp = makeAnotherReg(Type::IntTy);
|
||||||
|
BuildMI (BB, V8::ADDri, 2, tmp).addReg(VAList).addSImm(Size);
|
||||||
|
BuildMI(BB, V8::ST, 3).addReg(VAListPtr).addSImm(0).addReg(VAList);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
@ -102,7 +102,6 @@ namespace {
|
|||||||
void visitBranchInst(BranchInst &I);
|
void visitBranchInst(BranchInst &I);
|
||||||
void visitUnreachableInst(UnreachableInst &I) {}
|
void visitUnreachableInst(UnreachableInst &I) {}
|
||||||
void visitCastInst(CastInst &I);
|
void visitCastInst(CastInst &I);
|
||||||
void visitVANextInst(VANextInst &I);
|
|
||||||
void visitVAArgInst(VAArgInst &I);
|
void visitVAArgInst(VAArgInst &I);
|
||||||
void visitLoadInst(LoadInst &I);
|
void visitLoadInst(LoadInst &I);
|
||||||
void visitStoreInst(StoreInst &I);
|
void visitStoreInst(StoreInst &I);
|
||||||
@ -1754,8 +1753,10 @@ void V8ISel::visitIntrinsicCall(Intrinsic::ID ID, CallInst &CI) {
|
|||||||
|
|
||||||
case Intrinsic::vastart: {
|
case Intrinsic::vastart: {
|
||||||
// Add the VarArgsOffset to the frame pointer, and copy it to the result.
|
// Add the VarArgsOffset to the frame pointer, and copy it to the result.
|
||||||
unsigned DestReg = getReg (CI);
|
unsigned DestReg = getReg (CI.getOperand(1));
|
||||||
BuildMI (BB, V8::ADDri, 2, DestReg).addReg (V8::FP).addSImm (VarArgsOffset);
|
unsigned Tmp = makeAnotherReg(Type::IntTy);
|
||||||
|
BuildMI (BB, V8::ADDri, 2, Tmp).addReg (V8::FP).addSImm (VarArgsOffset);
|
||||||
|
BuildMI(BB, V8::ST, 3).addReg(DestReg).addSImm(0).addReg(Tmp);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1765,39 +1766,37 @@ void V8ISel::visitIntrinsicCall(Intrinsic::ID ID, CallInst &CI) {
|
|||||||
|
|
||||||
case Intrinsic::vacopy: {
|
case Intrinsic::vacopy: {
|
||||||
// Copy the va_list ptr (arg1) to the result.
|
// Copy the va_list ptr (arg1) to the result.
|
||||||
unsigned DestReg = getReg (CI), SrcReg = getReg (CI.getOperand (1));
|
unsigned DestReg = getReg (CI.getOperand(1)), SrcReg = getReg (CI.getOperand (2));
|
||||||
BuildMI (BB, V8::ORrr, 2, DestReg).addReg (V8::G0).addReg (SrcReg);
|
BuildMI(BB, V8::ST, 3).addReg(DestReg).addSImm(0).addReg(SrcReg);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void V8ISel::visitVANextInst (VANextInst &I) {
|
|
||||||
// Add the type size to the vararg pointer (arg0).
|
|
||||||
unsigned DestReg = getReg (I);
|
|
||||||
unsigned SrcReg = getReg (I.getOperand (0));
|
|
||||||
unsigned TySize = TM.getTargetData ().getTypeSize (I.getArgType ());
|
|
||||||
BuildMI (BB, V8::ADDri, 2, DestReg).addReg (SrcReg).addSImm (TySize);
|
|
||||||
}
|
|
||||||
|
|
||||||
void V8ISel::visitVAArgInst (VAArgInst &I) {
|
void V8ISel::visitVAArgInst (VAArgInst &I) {
|
||||||
unsigned VAList = getReg (I.getOperand (0));
|
unsigned VAListPtr = getReg (I.getOperand (0));
|
||||||
unsigned DestReg = getReg (I);
|
unsigned DestReg = getReg (I);
|
||||||
|
unsigned Size;
|
||||||
|
unsigned VAList = makeAnotherReg(Type::IntTy);
|
||||||
|
BuildMI(BB, V8::LD, 2, VAList).addReg(VAListPtr).addSImm(0);
|
||||||
|
|
||||||
switch (I.getType ()->getTypeID ()) {
|
switch (I.getType ()->getTypeID ()) {
|
||||||
case Type::PointerTyID:
|
case Type::PointerTyID:
|
||||||
case Type::UIntTyID:
|
case Type::UIntTyID:
|
||||||
case Type::IntTyID:
|
case Type::IntTyID:
|
||||||
|
Size = 4;
|
||||||
BuildMI (BB, V8::LD, 2, DestReg).addReg (VAList).addSImm (0);
|
BuildMI (BB, V8::LD, 2, DestReg).addReg (VAList).addSImm (0);
|
||||||
return;
|
break;
|
||||||
|
|
||||||
case Type::ULongTyID:
|
case Type::ULongTyID:
|
||||||
case Type::LongTyID:
|
case Type::LongTyID:
|
||||||
|
Size = 8;
|
||||||
BuildMI (BB, V8::LD, 2, DestReg).addReg (VAList).addSImm (0);
|
BuildMI (BB, V8::LD, 2, DestReg).addReg (VAList).addSImm (0);
|
||||||
BuildMI (BB, V8::LD, 2, DestReg+1).addReg (VAList).addSImm (4);
|
BuildMI (BB, V8::LD, 2, DestReg+1).addReg (VAList).addSImm (4);
|
||||||
return;
|
break;
|
||||||
|
|
||||||
case Type::DoubleTyID: {
|
case Type::DoubleTyID: {
|
||||||
|
Size = 8;
|
||||||
unsigned DblAlign = TM.getTargetData().getDoubleAlignment();
|
unsigned DblAlign = TM.getTargetData().getDoubleAlignment();
|
||||||
unsigned TempReg = makeAnotherReg (Type::IntTy);
|
unsigned TempReg = makeAnotherReg (Type::IntTy);
|
||||||
unsigned TempReg2 = makeAnotherReg (Type::IntTy);
|
unsigned TempReg2 = makeAnotherReg (Type::IntTy);
|
||||||
@ -1807,7 +1806,7 @@ void V8ISel::visitVAArgInst (VAArgInst &I) {
|
|||||||
BuildMI (BB, V8::ST, 3).addFrameIndex (FI).addSImm (0).addReg (TempReg);
|
BuildMI (BB, V8::ST, 3).addFrameIndex (FI).addSImm (0).addReg (TempReg);
|
||||||
BuildMI (BB, V8::ST, 3).addFrameIndex (FI).addSImm (4).addReg (TempReg2);
|
BuildMI (BB, V8::ST, 3).addFrameIndex (FI).addSImm (4).addReg (TempReg2);
|
||||||
BuildMI (BB, V8::LDDFri, 2, DestReg).addFrameIndex (FI).addSImm (0);
|
BuildMI (BB, V8::LDDFri, 2, DestReg).addFrameIndex (FI).addSImm (0);
|
||||||
return;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
@ -1816,4 +1815,8 @@ void V8ISel::visitVAArgInst (VAArgInst &I) {
|
|||||||
abort ();
|
abort ();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
unsigned tmp = makeAnotherReg(Type::IntTy);
|
||||||
|
BuildMI (BB, V8::ADDri, 2, tmp).addReg(VAList).addSImm(Size);
|
||||||
|
BuildMI(BB, V8::ST, 3).addReg(VAListPtr).addSImm(0).addReg(VAList);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
@ -176,12 +176,16 @@ namespace {
|
|||||||
SelectionDAG &DAG);
|
SelectionDAG &DAG);
|
||||||
|
|
||||||
virtual std::pair<SDOperand, SDOperand>
|
virtual std::pair<SDOperand, SDOperand>
|
||||||
LowerVAStart(SDOperand Chain, SelectionDAG &DAG);
|
LowerVAStart(SDOperand Chain, SelectionDAG &DAG, SDOperand Dest);
|
||||||
|
|
||||||
virtual std::pair<SDOperand,SDOperand>
|
virtual std::pair<SDOperand,SDOperand>
|
||||||
LowerVAArgNext(bool isVANext, SDOperand Chain, SDOperand VAList,
|
LowerVAArgNext(SDOperand Chain, SDOperand VAList,
|
||||||
const Type *ArgTy, SelectionDAG &DAG);
|
const Type *ArgTy, SelectionDAG &DAG);
|
||||||
|
|
||||||
|
virtual std::pair<SDOperand,SDOperand>
|
||||||
|
LowerVACopy(SDOperand Chain, SDOperand Src, SDOperand Dest,
|
||||||
|
SelectionDAG &DAG);
|
||||||
|
|
||||||
virtual std::pair<SDOperand, SDOperand>
|
virtual std::pair<SDOperand, SDOperand>
|
||||||
LowerFrameReturnAddress(bool isFrameAddr, SDOperand Chain, unsigned Depth,
|
LowerFrameReturnAddress(bool isFrameAddr, SDOperand Chain, unsigned Depth,
|
||||||
SelectionDAG &DAG);
|
SelectionDAG &DAG);
|
||||||
@ -443,20 +447,19 @@ X86TargetLowering::LowerCCCCallTo(SDOperand Chain, const Type *RetTy,
|
|||||||
}
|
}
|
||||||
|
|
||||||
std::pair<SDOperand,SDOperand>
|
std::pair<SDOperand,SDOperand>
|
||||||
X86TargetLowering::LowerVAStart(SDOperand Chain, SelectionDAG &DAG) {
|
X86TargetLowering::LowerVAStart(SDOperand Chain, SelectionDAG &DAG, SDOperand Dest) {
|
||||||
// vastart just returns the address of the VarArgsFrameIndex slot.
|
// vastart just stores the address of the VarArgsFrameIndex slot.
|
||||||
return std::make_pair(DAG.getFrameIndex(VarArgsFrameIndex, MVT::i32), Chain);
|
SDOperand FR = DAG.getFrameIndex(VarArgsFrameIndex, MVT::i32);
|
||||||
|
SDOperand Result = DAG.getNode(ISD::STORE, MVT::Other, Chain, FR, Dest, DAG.getSrcValue(NULL));
|
||||||
|
return std::make_pair(Result, Result);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::pair<SDOperand,SDOperand> X86TargetLowering::
|
std::pair<SDOperand,SDOperand>
|
||||||
LowerVAArgNext(bool isVANext, SDOperand Chain, SDOperand VAList,
|
X86TargetLowering::LowerVAArgNext(SDOperand Chain, SDOperand VAList,
|
||||||
const Type *ArgTy, SelectionDAG &DAG) {
|
const Type *ArgTy, SelectionDAG &DAG) {
|
||||||
MVT::ValueType ArgVT = getValueType(ArgTy);
|
MVT::ValueType ArgVT = getValueType(ArgTy);
|
||||||
SDOperand Result;
|
SDOperand Val = DAG.getLoad(MVT::i32, Chain, VAList, DAG.getSrcValue(NULL));
|
||||||
if (!isVANext) {
|
SDOperand Result = DAG.getLoad(ArgVT, Val.getValue(1), Val, DAG.getSrcValue(NULL));
|
||||||
Result = DAG.getLoad(ArgVT, DAG.getEntryNode(), VAList,
|
|
||||||
DAG.getSrcValue(NULL));
|
|
||||||
} else {
|
|
||||||
unsigned Amt;
|
unsigned Amt;
|
||||||
if (ArgVT == MVT::i32)
|
if (ArgVT == MVT::i32)
|
||||||
Amt = 4;
|
Amt = 4;
|
||||||
@ -465,12 +468,22 @@ LowerVAArgNext(bool isVANext, SDOperand Chain, SDOperand VAList,
|
|||||||
"Other types should have been promoted for varargs!");
|
"Other types should have been promoted for varargs!");
|
||||||
Amt = 8;
|
Amt = 8;
|
||||||
}
|
}
|
||||||
Result = DAG.getNode(ISD::ADD, VAList.getValueType(), VAList,
|
Val = DAG.getNode(ISD::ADD, Val.getValueType(), Val,
|
||||||
DAG.getConstant(Amt, VAList.getValueType()));
|
DAG.getConstant(Amt, Val.getValueType()));
|
||||||
}
|
Chain = DAG.getNode(ISD::STORE, MVT::Other, Chain,
|
||||||
|
Val, VAList, DAG.getSrcValue(NULL));
|
||||||
return std::make_pair(Result, Chain);
|
return std::make_pair(Result, Chain);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::pair<SDOperand,SDOperand>
|
||||||
|
X86TargetLowering::LowerVACopy(SDOperand Chain, SDOperand Src,
|
||||||
|
SDOperand Dest, SelectionDAG &DAG)
|
||||||
|
{
|
||||||
|
SDOperand Result = DAG.getNode(ISD::STORE, MVT::Other, Chain,
|
||||||
|
Src, Dest, DAG.getSrcValue(NULL));
|
||||||
|
return std::make_pair(Result, Result);
|
||||||
|
}
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
// Fast Calling Convention implementation
|
// Fast Calling Convention implementation
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
@ -232,7 +232,6 @@ namespace {
|
|||||||
void visitShiftInst(ShiftInst &I);
|
void visitShiftInst(ShiftInst &I);
|
||||||
void visitPHINode(PHINode &I) {} // PHI nodes handled by second pass
|
void visitPHINode(PHINode &I) {} // PHI nodes handled by second pass
|
||||||
void visitCastInst(CastInst &I);
|
void visitCastInst(CastInst &I);
|
||||||
void visitVANextInst(VANextInst &I);
|
|
||||||
void visitVAArgInst(VAArgInst &I);
|
void visitVAArgInst(VAArgInst &I);
|
||||||
|
|
||||||
void visitInstruction(Instruction &I) {
|
void visitInstruction(Instruction &I) {
|
||||||
@ -1838,12 +1837,14 @@ void X86ISel::visitIntrinsicCall(Intrinsic::ID ID, CallInst &CI) {
|
|||||||
unsigned TmpReg1, TmpReg2;
|
unsigned TmpReg1, TmpReg2;
|
||||||
switch (ID) {
|
switch (ID) {
|
||||||
case Intrinsic::vastart:
|
case Intrinsic::vastart:
|
||||||
|
//FIXME: store to first arg, don't return
|
||||||
// Get the address of the first vararg value...
|
// Get the address of the first vararg value...
|
||||||
TmpReg1 = getReg(CI);
|
TmpReg1 = getReg(CI);
|
||||||
addFrameReference(BuildMI(BB, X86::LEA32r, 5, TmpReg1), VarArgsFrameIndex);
|
addFrameReference(BuildMI(BB, X86::LEA32r, 5, TmpReg1), VarArgsFrameIndex);
|
||||||
return;
|
return;
|
||||||
|
|
||||||
case Intrinsic::vacopy:
|
case Intrinsic::vacopy:
|
||||||
|
//FIXME: copy val of second into first (which is a ptr)
|
||||||
TmpReg1 = getReg(CI);
|
TmpReg1 = getReg(CI);
|
||||||
TmpReg2 = getReg(CI.getOperand(1));
|
TmpReg2 = getReg(CI.getOperand(1));
|
||||||
BuildMI(BB, X86::MOV32rr, 1, TmpReg1).addReg(TmpReg2);
|
BuildMI(BB, X86::MOV32rr, 1, TmpReg1).addReg(TmpReg2);
|
||||||
@ -3745,38 +3746,12 @@ void X86ISel::emitCastOperation(MachineBasicBlock *BB,
|
|||||||
abort();
|
abort();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// visitVANextInst - Implement the va_next instruction...
|
|
||||||
///
|
|
||||||
void X86ISel::visitVANextInst(VANextInst &I) {
|
|
||||||
unsigned VAList = getReg(I.getOperand(0));
|
|
||||||
unsigned DestReg = getReg(I);
|
|
||||||
|
|
||||||
unsigned Size;
|
|
||||||
switch (I.getArgType()->getTypeID()) {
|
|
||||||
default:
|
|
||||||
std::cerr << I;
|
|
||||||
assert(0 && "Error: bad type for va_next instruction!");
|
|
||||||
return;
|
|
||||||
case Type::PointerTyID:
|
|
||||||
case Type::UIntTyID:
|
|
||||||
case Type::IntTyID:
|
|
||||||
Size = 4;
|
|
||||||
break;
|
|
||||||
case Type::ULongTyID:
|
|
||||||
case Type::LongTyID:
|
|
||||||
case Type::DoubleTyID:
|
|
||||||
Size = 8;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Increment the VAList pointer...
|
|
||||||
BuildMI(BB, X86::ADD32ri, 2, DestReg).addReg(VAList).addImm(Size);
|
|
||||||
}
|
|
||||||
|
|
||||||
void X86ISel::visitVAArgInst(VAArgInst &I) {
|
void X86ISel::visitVAArgInst(VAArgInst &I) {
|
||||||
unsigned VAList = getReg(I.getOperand(0));
|
unsigned VAListPtr = getReg(I.getOperand(0));
|
||||||
unsigned DestReg = getReg(I);
|
unsigned DestReg = getReg(I);
|
||||||
|
unsigned VAList = makeAnotherReg(Type::IntTy);
|
||||||
|
addDirectMem(BuildMI(BB, X86::MOV32rm, 4, VAList), VAListPtr);
|
||||||
|
unsigned Size;
|
||||||
switch (I.getType()->getTypeID()) {
|
switch (I.getType()->getTypeID()) {
|
||||||
default:
|
default:
|
||||||
std::cerr << I;
|
std::cerr << I;
|
||||||
@ -3785,17 +3760,24 @@ void X86ISel::visitVAArgInst(VAArgInst &I) {
|
|||||||
case Type::PointerTyID:
|
case Type::PointerTyID:
|
||||||
case Type::UIntTyID:
|
case Type::UIntTyID:
|
||||||
case Type::IntTyID:
|
case Type::IntTyID:
|
||||||
|
Size = 4;
|
||||||
addDirectMem(BuildMI(BB, X86::MOV32rm, 4, DestReg), VAList);
|
addDirectMem(BuildMI(BB, X86::MOV32rm, 4, DestReg), VAList);
|
||||||
break;
|
break;
|
||||||
case Type::ULongTyID:
|
case Type::ULongTyID:
|
||||||
case Type::LongTyID:
|
case Type::LongTyID:
|
||||||
|
Size = 8;
|
||||||
addDirectMem(BuildMI(BB, X86::MOV32rm, 4, DestReg), VAList);
|
addDirectMem(BuildMI(BB, X86::MOV32rm, 4, DestReg), VAList);
|
||||||
addRegOffset(BuildMI(BB, X86::MOV32rm, 4, DestReg+1), VAList, 4);
|
addRegOffset(BuildMI(BB, X86::MOV32rm, 4, DestReg+1), VAList, 4);
|
||||||
break;
|
break;
|
||||||
case Type::DoubleTyID:
|
case Type::DoubleTyID:
|
||||||
|
Size = 8;
|
||||||
addDirectMem(BuildMI(BB, X86::FLD64m, 4, DestReg), VAList);
|
addDirectMem(BuildMI(BB, X86::FLD64m, 4, DestReg), VAList);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
// Increment the VAList pointer...
|
||||||
|
unsigned NP = makeAnotherReg(Type::IntTy);
|
||||||
|
BuildMI(BB, X86::ADD32ri, 2, NP).addReg(VAList).addSImm(Size);
|
||||||
|
addDirectMem(BuildMI(BB, X86::MOV32rm, 5), VAListPtr).addReg(VAList);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// visitGetElementPtrInst - instruction-select GEP instructions
|
/// visitGetElementPtrInst - instruction-select GEP instructions
|
||||||
|
@ -386,7 +386,7 @@ bool LICM::canSinkOrHoistInst(Instruction &I) {
|
|||||||
|
|
||||||
return isa<BinaryOperator>(I) || isa<ShiftInst>(I) || isa<CastInst>(I) ||
|
return isa<BinaryOperator>(I) || isa<ShiftInst>(I) || isa<CastInst>(I) ||
|
||||||
isa<SelectInst>(I) ||
|
isa<SelectInst>(I) ||
|
||||||
isa<GetElementPtrInst>(I) || isa<VANextInst>(I) || isa<VAArgInst>(I);
|
isa<GetElementPtrInst>(I) || isa<VAArgInst>(I);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// isNotUsedInLoop - Return true if the only users of this instruction are
|
/// isNotUsedInLoop - Return true if the only users of this instruction are
|
||||||
|
@ -1188,10 +1188,6 @@ void AssemblyWriter::printInstruction(const Instruction &I) {
|
|||||||
if (Operand) writeOperand(Operand, true); // Work with broken code
|
if (Operand) writeOperand(Operand, true); // Work with broken code
|
||||||
Out << ", ";
|
Out << ", ";
|
||||||
printType(I.getType());
|
printType(I.getType());
|
||||||
} else if (const VANextInst *VAN = dyn_cast<VANextInst>(&I)) {
|
|
||||||
if (Operand) writeOperand(Operand, true); // Work with broken code
|
|
||||||
Out << ", ";
|
|
||||||
printType(VAN->getArgType());
|
|
||||||
} else if (Operand) { // Print the normal way...
|
} else if (Operand) { // Print the normal way...
|
||||||
|
|
||||||
// PrintAllTypes - Instructions who have operands of all the same type
|
// PrintAllTypes - Instructions who have operands of all the same type
|
||||||
|
@ -110,8 +110,7 @@ const char *Instruction::getOpcodeName(unsigned OpCode) {
|
|||||||
case Call: return "call";
|
case Call: return "call";
|
||||||
case Shl: return "shl";
|
case Shl: return "shl";
|
||||||
case Shr: return "shr";
|
case Shr: return "shr";
|
||||||
case VANext: return "vanext";
|
case VAArg: return "va_arg";
|
||||||
case VAArg: return "vaarg";
|
|
||||||
|
|
||||||
default: return "<Invalid operator> ";
|
default: return "<Invalid operator> ";
|
||||||
}
|
}
|
||||||
@ -139,8 +138,6 @@ bool Instruction::isIdenticalTo(Instruction *I) const {
|
|||||||
return LI->isVolatile() == cast<LoadInst>(I)->isVolatile();
|
return LI->isVolatile() == cast<LoadInst>(I)->isVolatile();
|
||||||
if (const StoreInst *SI = dyn_cast<StoreInst>(this))
|
if (const StoreInst *SI = dyn_cast<StoreInst>(this))
|
||||||
return SI->isVolatile() == cast<StoreInst>(I)->isVolatile();
|
return SI->isVolatile() == cast<StoreInst>(I)->isVolatile();
|
||||||
if (const VANextInst *VAN = dyn_cast<VANextInst>(this))
|
|
||||||
return VAN->getArgType() == cast<VANextInst>(I)->getArgType();
|
|
||||||
if (const CallInst *CI = dyn_cast<CallInst>(this))
|
if (const CallInst *CI = dyn_cast<CallInst>(this))
|
||||||
return CI->isTailCall() == cast<CallInst>(I)->isTailCall();
|
return CI->isTailCall() == cast<CallInst>(I)->isTailCall();
|
||||||
return true;
|
return true;
|
||||||
|
@ -1096,7 +1096,6 @@ CastInst *CastInst::clone() const { return new CastInst(*this); }
|
|||||||
CallInst *CallInst::clone() const { return new CallInst(*this); }
|
CallInst *CallInst::clone() const { return new CallInst(*this); }
|
||||||
ShiftInst *ShiftInst::clone() const { return new ShiftInst(*this); }
|
ShiftInst *ShiftInst::clone() const { return new ShiftInst(*this); }
|
||||||
SelectInst *SelectInst::clone() const { return new SelectInst(*this); }
|
SelectInst *SelectInst::clone() const { return new SelectInst(*this); }
|
||||||
VANextInst *VANextInst::clone() const { return new VANextInst(*this); }
|
|
||||||
VAArgInst *VAArgInst::clone() const { return new VAArgInst(*this); }
|
VAArgInst *VAArgInst::clone() const { return new VAArgInst(*this); }
|
||||||
PHINode *PHINode::clone() const { return new PHINode(*this); }
|
PHINode *PHINode::clone() const { return new PHINode(*this); }
|
||||||
ReturnInst *ReturnInst::clone() const { return new ReturnInst(*this); }
|
ReturnInst *ReturnInst::clone() const { return new ReturnInst(*this); }
|
||||||
|
@ -178,7 +178,6 @@ namespace { // Anonymous namespace for class
|
|||||||
void visitPHINode(PHINode &PN);
|
void visitPHINode(PHINode &PN);
|
||||||
void visitBinaryOperator(BinaryOperator &B);
|
void visitBinaryOperator(BinaryOperator &B);
|
||||||
void visitShiftInst(ShiftInst &SI);
|
void visitShiftInst(ShiftInst &SI);
|
||||||
void visitVANextInst(VANextInst &VAN) { visitInstruction(VAN); }
|
|
||||||
void visitVAArgInst(VAArgInst &VAA) { visitInstruction(VAA); }
|
void visitVAArgInst(VAArgInst &VAA) { visitInstruction(VAA); }
|
||||||
void visitCallInst(CallInst &CI);
|
void visitCallInst(CallInst &CI);
|
||||||
void visitGetElementPtrInst(GetElementPtrInst &GEP);
|
void visitGetElementPtrInst(GetElementPtrInst &GEP);
|
||||||
@ -655,10 +654,10 @@ void Verifier::visitIntrinsicFunctionCall(Intrinsic::ID ID, CallInst &CI) {
|
|||||||
Assert1(CI.getParent()->getParent()->getFunctionType()->isVarArg(),
|
Assert1(CI.getParent()->getParent()->getFunctionType()->isVarArg(),
|
||||||
"llvm.va_start intrinsic may only occur in function with variable"
|
"llvm.va_start intrinsic may only occur in function with variable"
|
||||||
" args!", &CI);
|
" args!", &CI);
|
||||||
NumArgs = 0;
|
NumArgs = 1;
|
||||||
break;
|
break;
|
||||||
case Intrinsic::vaend: NumArgs = 1; break;
|
case Intrinsic::vaend: NumArgs = 1; break;
|
||||||
case Intrinsic::vacopy: NumArgs = 1; break;
|
case Intrinsic::vacopy: NumArgs = 2; break;
|
||||||
|
|
||||||
case Intrinsic::returnaddress:
|
case Intrinsic::returnaddress:
|
||||||
case Intrinsic::frameaddress:
|
case Intrinsic::frameaddress:
|
||||||
|
Loading…
x
Reference in New Issue
Block a user