Index: test/NaCl/Bitcode/call-elide.ll |
diff --git a/test/NaCl/Bitcode/call-elide.ll b/test/NaCl/Bitcode/call-elide.ll |
new file mode 100644 |
index 0000000000000000000000000000000000000000..f53e9bee38eb23db06aed32b228fd08fa4cb7332 |
--- /dev/null |
+++ b/test/NaCl/Bitcode/call-elide.ll |
@@ -0,0 +1,219 @@ |
+; Test how we handle eliding pointers in call instructions. |
+ |
+; RUN: llvm-as < %s | pnacl-freeze \ |
+; RUN: | pnacl-bcanalyzer -dump-records \ |
+; RUN: | FileCheck %s -check-prefix=PF2 |
+ |
+; RUN: llvm-as < %s | pnacl-freeze -allow-local-symbol-tables \ |
+; RUN: | pnacl-thaw -allow-local-symbol-tables \ |
+; RUN: | llvm-dis - | FileCheck %s -check-prefix=TD2 |
+ |
+; ------------------------------------------------------ |
+; Define some global functions/variables to be used in testing. |
+ |
+ |
+@bytes = internal global [4 x i8] c"abcd" |
+declare void @foo(i32 %i) |
+declare i32 @llvm.nacl.setjmp(i8* %i) |
+ |
+; ------------------------------------------------------ |
+; Test how we handle a direct call. |
+ |
+define void @DirectCall() { |
+ call void @foo(i32 0) |
+ ret void |
+} |
+ |
+; TD2: define void @DirectCall() { |
+; TD2-NEXT: call void @foo(i32 0) |
+; TD2-NEXT: ret void |
+; TD2-NEXT: } |
+ |
+; PF2: <FUNCTION_BLOCK> |
+; PF2: </CONSTANTS_BLOCK> |
+; PF2-NEXT: <INST_CALL op0=0 op1=14 op2=1/> |
+; PF2-NEXT: <INST_RET/> |
+; PF2-NEXT: </FUNCTION_BLOCK> |
+ |
+; ------------------------------------------------------ |
+; Test how we handle a direct call with a normalized inttoptr argument. |
+; Pointer arguments are only allowed for intrinsic calls. |
+ |
+define void @DirectCallIntToPtrArg(i32 %i) { |
+ %1 = inttoptr i32 %i to i8* |
+ %2 = call i32 @llvm.nacl.setjmp(i8* %1) |
+ ret void |
+} |
+ |
+; TD2: define void @DirectCallIntToPtrArg(i32 %i) { |
+; TD2-NEXT: %1 = inttoptr i32 %i to i8* |
+; TD2-NEXT: %2 = call i32 @llvm.nacl.setjmp(i8* %1) |
+; TD2-NEXT: ret void |
+; TD2-NEXT: } |
+ |
+; PF2: <FUNCTION_BLOCK> |
+; PF2-NEXT: <DECLAREBLOCKS op0=1/> |
+; PF2-NEXT: <INST_CALL op0=0 op1=13 op2=1/> |
+; PF2-NEXT: <INST_RET/> |
+; PF2-NEXT: </FUNCTION_BLOCK> |
+ |
+; ------------------------------------------------------ |
+; Test how we handle a direct call with a normalized ptroint argument. |
+; Pointer arguments are only allowed for intrinsic calls. |
+ |
+define void @DirectCallPtrToIntArg() { |
+ %1 = alloca i8, i32 4, align 8 |
+ %2 = ptrtoint i8* %1 to i32 |
+ call void @foo(i32 %2) |
+ ret void |
+} |
+ |
+; TD2: define void @DirectCallPtrToIntArg() { |
+; TD2-NEXT: %1 = alloca i8, i32 4, align 8 |
+; TD2-NEXT: %2 = ptrtoint i8* %1 to i32 |
+; TD2-NEXT: call void @foo(i32 %2) |
+; TD2-NEXT: ret void |
+; TD2-NEXT: } |
+ |
+; PF2: <FUNCTION_BLOCK> |
+; PF2: </CONSTANTS_BLOCK> |
+; PF2-NEXT: <INST_ALLOCA op0=1 op1=4/> |
+; PF2-NEXT: <INST_CALL op0=0 op1=15 op2=1/> |
+; PF2-NEXT: <INST_RET/> |
+; PF2-NEXT: </FUNCTION_BLOCK> |
+ |
+; ------------------------------------------------------ |
+; Test how we handle a direct call with a normalized bitcast argument. |
+ |
+define void @DirectCallBitcastArg(i32 %i) { |
+ %1 = bitcast [4 x i8]* @bytes to i8* |
+ %2 = call i32 @llvm.nacl.setjmp(i8* %1) |
+ ret void |
+} |
+ |
+; TD2: define void @DirectCallBitcastArg(i32 %i) { |
+; TD2-NEXT: %1 = bitcast [4 x i8]* @bytes to i8* |
+; TD2-NEXT: %2 = call i32 @llvm.nacl.setjmp(i8* %1) |
+; TD2-NEXT: ret void |
+; TD2-NEXT: } |
+ |
+; PF2: <FUNCTION_BLOCK> |
+; PF2-NEXT: <DECLAREBLOCKS op0=1/> |
+; PF2-NEXT: <INST_CALL op0=0 op1=13 op2=2/> |
+; PF2-NEXT: <INST_RET/> |
+; PF2-NEXT: </FUNCTION_BLOCK> |
+ |
+; ------------------------------------------------------ |
+; Test how we handle a direct call with a pointer to scalar conversion. |
+ |
+define void @DirectCallScalarArg() { |
+ %1 = ptrtoint [4 x i8]* @bytes to i32 |
+ call void @foo(i32 %1) |
+ ret void |
+} |
+ |
+; TD2: define void @DirectCallScalarArg() { |
+; TD2-NEXT: %1 = ptrtoint [4 x i8]* @bytes to i32 |
+; TD2-NEXT: call void @foo(i32 %1) |
+; TD2-NEXT: ret void |
+; TD2-NEXT: } |
+ |
+; PF2: <FUNCTION_BLOCK> |
+; PF2-NEXT: <DECLAREBLOCKS op0=1/> |
+; PF2-NEXT: <INST_CALL op0=0 op1=13 op2=1/> |
+; PF2-NEXT: <INST_RET/> |
+; PF2-NEXT: </FUNCTION_BLOCK> |
+ |
+; ------------------------------------------------------ |
+; Test how we handle an indirect call. |
+ |
+define void @IndirectCall(i32 %i) { |
+ %1 = inttoptr i32 %i to void (i32)* |
+ call void %1(i32 %i) |
+ ret void |
+} |
+ |
+; TD2: define void @IndirectCall(i32 %i) { |
+; TD2-NEXT: %1 = inttoptr i32 %i to void (i32)* |
+; TD2-NEXT: call void %1(i32 %i) |
+; TD2-NEXT: ret void |
+; TD2-NEXT: } |
+ |
+; PF2: <FUNCTION_BLOCK> |
+; PF2-NEXT: <DECLAREBLOCKS op0=1/> |
+; PF2-NEXT: <INST_CALL_INDIRECT op0=0 op1=1 op2=1 op3=1/> |
+; PF2-NEXT: <INST_RET/> |
+; PF2-NEXT: </FUNCTION_BLOCK> |
+ |
+; ------------------------------------------------------ |
+; Test how we handle an indirect call with a normalized ptrtoint argument. |
+ |
+define void @IndirectCallPtrToIntArg(i32 %i) { |
+ %1 = alloca i8, i32 4, align 8 |
+ %2 = inttoptr i32 %i to void (i32)* |
+ %3 = ptrtoint i8* %1 to i32 |
+ call void %2(i32 %3) |
+ ret void |
+} |
+ |
+; TD2: define void @IndirectCallPtrToIntArg(i32 %i) { |
+; TD2-NEXT: %1 = alloca i8, i32 4, align 8 |
+; TD2-NEXT: %2 = ptrtoint i8* %1 to i32 |
+; TD2-NEXT: %3 = inttoptr i32 %i to void (i32)* |
+; TD2-NEXT: call void %3(i32 %2) |
+; TD2-NEXT: ret void |
+; TD2-NEXT: } |
+ |
+; PF2: <FUNCTION_BLOCK> |
+; PF2: </CONSTANTS_BLOCK> |
+; PF2-NEXT: <INST_ALLOCA op0=1 op1=4/> |
+; PF2-NEXT: <INST_CALL_INDIRECT op0=0 op1=3 op2=1 op3=1/> |
+; PF2-NEXT: <INST_RET/> |
+; PF2-NEXT: </FUNCTION_BLOCK> |
+ |
+; ------------------------------------------------------ |
+; Test how we handle an indirect call with a pointer to scalar conversion. |
+ |
+define void @IndirectCallScalarArg(i32 %i) { |
+ %1 = inttoptr i32 %i to void (i32)* |
+ %2 = ptrtoint [4 x i8]* @bytes to i32 |
+ call void %1(i32 %2) |
+ ret void |
+} |
+ |
+; TD2: define void @IndirectCallScalarArg(i32 %i) { |
+; TD2-NEXT: %1 = ptrtoint [4 x i8]* @bytes to i32 |
+; TD2-NEXT: %2 = inttoptr i32 %i to void (i32)* |
+; TD2-NEXT: call void %2(i32 %1) |
+; TD2-NEXT: ret void |
+; TD2-NEXT: } |
+ |
+; PF2: <FUNCTION_BLOCK> |
+; PF2-NEXT: <DECLAREBLOCKS op0=1/> |
+; PF2-NEXT: <INST_CALL_INDIRECT op0=0 op1=1 op2=1 op3=2/> |
+; PF2-NEXT: <INST_RET/> |
+; PF2-NEXT: </FUNCTION_BLOCK> |
+ |
+; ------------------------------------------------------ |
+; Test how we handle intrinsics that can return (inherent) pointers, and |
+; return statements that expect scalar values. |
+ |
+declare i8* @llvm.nacl.read.tp() |
+ |
+define i32 @ReturnPtrIntrinsic() { |
+ %1 = call i8* @llvm.nacl.read.tp() |
+ %2 = ptrtoint i8* %1 to i32 |
+ ret i32 %2 |
+} |
+ |
+; TD2: define i32 @ReturnPtrIntrinsic() { |
+; TD2-NEXT: %1 = call i8* @llvm.nacl.read.tp() |
+; TD2-NEXT: %2 = ptrtoint i8* %1 to i32 |
+; TD2-NEXT: ret i32 %2 |
+; TD2-NEXT: } |
+ |
+; PF2: <FUNCTION_BLOCK> |
+; PF2-NEXT: <DECLAREBLOCKS op0=1/> |
+; PF2-NEXT: <INST_CALL op0=0 op1=3/> |
+; PF2-NEXT: <INST_RET op0=1/> |
+; PF2-NEXT: </FUNCTION_BLOCK> |