; Test how we handle eliding pointers in call instructions.
; RUN: llvm-as < %s | pnacl-freeze --pnacl-version=1 \
; RUN: | pnacl-bcanalyzer -dump-records \
; RUN: | FileCheck %s -check-prefix=PF1
; RUN: llvm-as < %s | pnacl-freeze --pnacl-version=1 | pnacl-thaw \
; RUN: | llvm-dis - | FileCheck %s -check-prefix=TD1
; RUN: llvm-as < %s | pnacl-freeze --pnacl-version=2 \
; RUN: | pnacl-bcanalyzer -dump-records \
; RUN: | FileCheck %s -check-prefix=PF2
; RUN: llvm-as < %s | pnacl-freeze --pnacl-version=2 | pnacl-thaw \
; 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
}
; TD1: define void @DirectCall() {
; TD1-NEXT: call void @foo(i32 0)
; TD1-NEXT: ret void
; TD1-NEXT: }
; PF1:
; PF1:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; TD2: define void @DirectCall() {
; TD2-NEXT: call void @foo(i32 0)
; TD2-NEXT: ret void
; TD2-NEXT: }
; PF2:
; PF2:
; PF2-NEXT:
; PF2-NEXT:
; PF2-NEXT:
; ------------------------------------------------------
; 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
}
; TD1: define void @DirectCallIntToPtrArg(i32 %i) {
; TD1-NEXT: %1 = inttoptr i32 %i to i8*
; TD1-NEXT: %2 = call i32 @llvm.nacl.setjmp(i8* %1)
; TD1-NEXT: ret void
; TD1-NEXT: }
; PF1:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1:
; 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:
; PF2-NEXT:
; PF2-NEXT:
; PF2-NEXT:
; PF2:
; ------------------------------------------------------
; 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
}
; TD1: define void @DirectCallPtrToIntArg() {
; TD1-NEXT: %1 = alloca i8, i32 4, align 8
; TD1-NEXT: %2 = ptrtoint i8* %1 to i32
; TD1-NEXT: call void @foo(i32 %2)
; TD1-NEXT: ret void
; TD1-NEXT: }
; PF1:
; PF1:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; 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:
; PF2:
; PF2-NEXT:
; PF2-NEXT:
; PF2-NEXT:
; PF2-NEXT:
; ------------------------------------------------------
; 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
}
; TD1: define void @DirectCallBitcastArg(i32 %i) {
; TD1-NEXT: %1 = bitcast [4 x i8]* @bytes to i8*
; TD1-NEXT: %2 = call i32 @llvm.nacl.setjmp(i8* %1)
; TD1-NEXT: ret void
; TD1-NEXT: }
; PF1:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1:
; 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:
; PF2-NEXT:
; PF2-NEXT:
; PF2-NEXT:
; PF2:
; ------------------------------------------------------
; 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
}
; TD1: define void @DirectCallScalarArg() {
; TD1-NEXT: %1 = ptrtoint [4 x i8]* @bytes to i32
; TD1-NEXT: call void @foo(i32 %1)
; TD1-NEXT: ret void
; TD1-NEXT: }
; PF1:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1:
; 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:
; PF2-NEXT:
; PF2-NEXT:
; PF2-NEXT:
; PF2:
; ------------------------------------------------------
; 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
}
; TD1: define void @IndirectCall(i32 %i) {
; TD1-NEXT: %1 = inttoptr i32 %i to void (i32)*
; TD1-NEXT: call void %1(i32 %i)
; TD1-NEXT: ret void
; TD1-NEXT: }
; PF1:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1:
; 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:
; PF2-NEXT:
; PF2-NEXT:
; PF2-NEXT:
; PF2:
; ------------------------------------------------------
; 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
}
; TD1: define void @IndirectCallPtrToIntArg(i32 %i) {
; TD1-NEXT: %1 = alloca i8, i32 4, align 8
; TD1-NEXT: %2 = inttoptr i32 %i to void (i32)*
; TD1-NEXT: %3 = ptrtoint i8* %1 to i32
; TD1-NEXT: call void %2(i32 %3)
; TD1-NEXT: ret void
; TD1-NEXT: }
; PF1:
; PF1:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1:
; 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:
; PF2:
; PF2-NEXT:
; PF2-NEXT:
; PF2-NEXT:
; PF2:
; ------------------------------------------------------
; 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
}
; TD1: define void @IndirectCallScalarArg(i32 %i) {
; TD1-NEXT: %1 = inttoptr i32 %i to void (i32)*
; TD1-NEXT: %2 = ptrtoint [4 x i8]* @bytes to i32
; TD1-NEXT: call void %1(i32 %2)
; TD1-NEXT: ret void
; TD1-NEXT: }
; PF1:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1:
; 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:
; PF2-NEXT:
; PF2-NEXT:
; PF2-NEXT:
; PF2:
; ------------------------------------------------------
; 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
}
; TD1: define i32 @ReturnPtrIntrinsic() {
; TD1-NEXT: %1 = call i8* @llvm.nacl.read.tp()
; TD1-NEXT: %2 = ptrtoint i8* %1 to i32
; TD1-NEXT: ret i32 %2
; TD1-NEXT: }
; PF1:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; PF1-NEXT:
; 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:
; PF2-NEXT:
; PF2-NEXT:
; PF2-NEXT:
; PF2-NEXT: