summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorTorok Edwin <edwintorok@gmail.com>2009-05-22 06:41:43 +0000
committerTorok Edwin <edwintorok@gmail.com>2009-05-22 06:41:43 +0000
commita39d69684ac7b0cbb0a2cc67e59665f75fc8f458 (patch)
tree6187599d488826d19f8beb894d69d2fe210dbf22
parent0cf811df8245b88a010ce9bf052dba0bb5e915ed (diff)
downloadbcm5719-llvm-a39d69684ac7b0cbb0a2cc67e59665f75fc8f458.tar.gz
bcm5719-llvm-a39d69684ac7b0cbb0a2cc67e59665f75fc8f458.zip
Verify that calling conventions match function prototype.
This only rejects mismatches between target specific calling convention and C/LLVM specific calling convention. There are too many fastcc/C, coldcc/cc42 mismatches in the testsuite, these are not reject by the verifier. llvm-svn: 72248
-rw-r--r--llvm/lib/VMCore/Verifier.cpp25
-rw-r--r--llvm/test/Assembler/2009-05-22-CC.ll9
-rw-r--r--llvm/test/Verifier/2009-05-22-CC.ll11
3 files changed, 45 insertions, 0 deletions
diff --git a/llvm/lib/VMCore/Verifier.cpp b/llvm/lib/VMCore/Verifier.cpp
index fc4cfcfe45d..65d0a8236b9 100644
--- a/llvm/lib/VMCore/Verifier.cpp
+++ b/llvm/lib/VMCore/Verifier.cpp
@@ -1024,6 +1024,31 @@ void Verifier::VerifyCallSite(CallSite CS) {
"Call parameter type does not match function signature!",
CS.getArgument(i), FTy->getParamType(i), I);
+ Assert2(CS.getType() == FTy->getReturnType(),
+ "Call return type does not match function signature!",
+ CS.getInstruction(), FTy->getReturnType());
+
+ // Verify calling convention for direct calls
+ Value *CalledF = CS.getCalledValue()->stripPointerCasts();
+ if (Function *F = dyn_cast<Function>(CalledF)) {
+ unsigned CC1 = CS.getCallingConv();
+ unsigned CC2 = F->getCallingConv();
+ if(CC1 != CC2) {
+ // tolerate some mismatch among C prototype and LLVM-specific calling conv
+ if (CC2 >= CallingConv::FirstTargetCC ||
+ CC1 >= CallingConv::FirstTargetCC) {
+ Instruction *I = CS.getInstruction()->clone();
+ if (CallInst *CI = dyn_cast<CallInst>(I)) {
+ CI->setCallingConv(F->getCallingConv());
+ } else
+ cast<InvokeInst>(I)->setCallingConv(F->getCallingConv());
+ Assert2(0,"Calling convention does not match function signature!",
+ CS.getInstruction(), I);
+ delete I;
+ }
+ }
+ }
+
if (CS.getCalledValue()->getNameLen() < 5 ||
strncmp(CS.getCalledValue()->getNameStart(), "llvm.", 5) != 0) {
// Verify that none of the arguments are metadata...
diff --git a/llvm/test/Assembler/2009-05-22-CC.ll b/llvm/test/Assembler/2009-05-22-CC.ll
new file mode 100644
index 00000000000..ba80d64ac28
--- /dev/null
+++ b/llvm/test/Assembler/2009-05-22-CC.ll
@@ -0,0 +1,9 @@
+; RUN: llvm-as < %s
+
+; Verify that calls with correct calling conv are accepted
+declare x86_stdcallcc i32 @re_string_construct(i8* inreg %pstr, i8* inreg %str, i32 inreg %len, i8* %trans, i32 %icase, i8* %dfa);
+define void @main() {
+entry:
+ %0 = call x86_stdcallcc i32 (...)* bitcast (i32 (i8*, i8*, i32, i8*, i32, i8*)* @re_string_construct to i32 (...)*)(i32 inreg 0, i32 inreg 0, i32 inreg 0, i32 0, i32 0, i8* inttoptr (i32 673194176 to i8*));
+ ret void
+}
diff --git a/llvm/test/Verifier/2009-05-22-CC.ll b/llvm/test/Verifier/2009-05-22-CC.ll
new file mode 100644
index 00000000000..7775dede03a
--- /dev/null
+++ b/llvm/test/Verifier/2009-05-22-CC.ll
@@ -0,0 +1,11 @@
+; RUN: not llvm-as < %s |& grep {Calling convention does not match function signature}
+; PR 4239
+
+; Verify that the calling convention on the call instruction matches the
+; declared calling convention
+declare x86_stdcallcc i32 @re_string_construct(i8* inreg %pstr, i8* inreg %str, i32 inreg %len, i8* %trans, i32 %icase, i8* %dfa);
+define void @main() {
+entry:
+ %0 = call i32 (...)* bitcast (i32 (i8*, i8*, i32, i8*, i32, i8*)* @re_string_construct to i32 (...)*)(i32 inreg 0, i32 inreg 0, i32 inreg 0, i32 0, i32 0, i8* inttoptr (i32 673194176 to i8*))
+ ret void
+}
OpenPOWER on IntegriCloud