blob: 8abaa6a2074be89dae622ccc153c7c549420839f [file] [edit]
; RUN: pnacl-llc -O2 -mtriple=x86_64-none-nacl -filetype=obj < %s | \
; RUN: llvm-objdump -d -r - | FileCheck %s
; RUN: pnacl-llc -O2 -mtriple=x86_64-none-nacl -filetype=obj < %s | \
; RUN: llvm-objdump -d -r - | FileCheck %s --check-prefix=NOCALLRET
; RUN: pnacl-llc -O2 -mtriple=x86_64-none-nacl -filetype=obj \
; RUN: -relocation-model=pic < %s | \
; RUN: llvm-objdump -d -r - | FileCheck %s --check-prefix=PIC
; RUN: pnacl-llc -O2 -mtriple=x86_64-none-nacl -filetype=asm < %s | \
; RUN: FileCheck %s --check-prefix=ASM
; ModuleID = 'pnacl-hides-sandbox-x86-64.c'
target datalayout = "e-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f32:32:32-f64:64:64-p:32:32:32-v128:32:32"
target triple = "le32-unknown-nacl"
@IndirectCallTarget = external global void ()*
@.str = private unnamed_addr constant [8 x i8] c"Prime 1\00", align 1
@.str1 = private unnamed_addr constant [8 x i8] c"Prime 2\00", align 1
@.str2 = private unnamed_addr constant [8 x i8] c"Prime 3\00", align 1
@.str3 = private unnamed_addr constant [8 x i8] c"Prime 4\00", align 1
@.str4 = private unnamed_addr constant [8 x i8] c"Prime 5\00", align 1
; Function Attrs: nounwind
define void @TestDirectCall() #0 {
entry:
call void @DirectCallTarget()
ret void
}
; CHECK-LABEL: TestDirectCall:
; Push only the bottom 32-bits of the frame pointer
; CHECK: movl %ebp, %eax
; CHECK-NEXT: pushq %rax
; Push the immediate return address
; CHECK: pushq $0
; CHECK-NEXT: .text
; Immediate jump to the target
; CHECK: jmp 0
; CHECK-NEXT: DirectCallTarget
; The return sequence should use %r11
; CHECK: popq %r11
; CHECK: andl $-32, %r11d
; CHECK-NEXT: addq %r15, %r11
; CHECK-NEXT: jmpq *%r11
; PIC-LABEL: TestDirectCall
; PIC: leal 19(%rip), %r10d
; PIC-NEXT: pushq %r10
; PIC-NEXT: jmp 0
; PIC-NEXT: DirectCallTarget
; ASM-LABEL: TestDirectCall:
; Push only the bottom 32-bits of the frame pointer
; ASM: movl %ebp, %eax
; ASM-NEXT: pushq %rax
; In asm, direct calls are just 'call'
; ASM: callq DirectCallTarget
; The return sequence should use %r11
; ASM: popq %r11
; ASM: nacljmp %r11d, %r15
declare hidden void @DirectCallTarget() #1
; Function Attrs: nounwind
define void @TestIndirectCall() #0 {
entry:
%0 = load void ()*, void ()** @IndirectCallTarget, align 4
call void %0()
ret void
}
; CHECK-LABEL: TestIndirectCall:
; Push the immediate return address
; CHECK: pushq $0
; CHECK-NEXT: .text
; Fixed sequence for indirect jump
; CHECK: andl $-32, %r11d
; CHECK-NEXT: addq %r15, %r11
; CHECK-NEXT: jmpq *%r11
; PIC-LABEL: TestIndirectCall:
; Ensure that the mov of the call target happens before the return address
; calculation
; PIC: movl {{.*}}, %r11d
; Calculate and push the return address
; PIC-NEXT: leal {{[0-9]+}}(%rip), %r10d
; PIC-NEXT: pushq %r10
; Fixed sequence for indirect jump
; PIC: andl $-32, %r11d
; PIC-NEXT: addq %r15, %r11
; PIC-NEXT: jmpq *%r11
; Function Attrs: nounwind
define void @TestMaskedFramePointer(i32 %Arg) #0 {
entry:
%Arg.addr = alloca i32, align 4
%Tmp = alloca i8*, align 4
store i32 %Arg, i32* %Arg.addr, align 4
%0 = load i32, i32* %Arg.addr, align 4
%1 = alloca i8, i32 %0
store i8* %1, i8** %Tmp, align 4
%2 = load i8*, i8** %Tmp, align 4
call void @Consume(i8* %2)
ret void
}
; Verify that the old frame pointer isn't leaked when saved
; CHECK: TestMaskedFramePointer:
; CHECK: movl %ebp, %eax
; CHECK: pushq %rax
; CHECK: movq %rsp, %rbp
declare void @Consume(i8*) #1
; Function Attrs: nounwind
define void @TestMaskedFramePointerVarargs(i32 %Arg, ...) #0 {
entry:
%Arg.addr = alloca i32, align 4
%Tmp = alloca i8*, align 4
store i32 %Arg, i32* %Arg.addr, align 4
%0 = load i32, i32* %Arg.addr, align 4
%1 = alloca i8, i32 %0
store i8* %1, i8** %Tmp, align 4
%2 = load i8*, i8** %Tmp, align 4
call void @Consume(i8* %2)
ret void
}
; Verify use of r10 instead of rax in the presence of varargs,
; when saving the old rbp.
; CHECK: TestMaskedFramePointerVarargs:
; CHECK: movl %ebp, %r10d
; CHECK: pushq %r10
; CHECK: movq %rsp, %rbp
; Function Attrs: nounwind
define void @TestIndirectJump(i32 %Arg) #0 {
entry:
%Arg.addr = alloca i32, align 4
store i32 %Arg, i32* %Arg.addr, align 4
%0 = load i32, i32* %Arg.addr, align 4
switch i32 %0, label %sw.epilog [
i32 2, label %sw.bb
i32 3, label %sw.bb1
i32 5, label %sw.bb3
i32 7, label %sw.bb5
i32 11, label %sw.bb7
]
sw.bb: ; preds = %entry
%call = call i32 @puts(i8* getelementptr inbounds ([8 x i8], [8 x i8]* @.str, i32 0, i32 0))
br label %sw.epilog
sw.bb1: ; preds = %entry
%call2 = call i32 @puts(i8* getelementptr inbounds ([8 x i8], [8 x i8]* @.str1, i32 0, i32 0))
br label %sw.epilog
sw.bb3: ; preds = %entry
%call4 = call i32 @puts(i8* getelementptr inbounds ([8 x i8], [8 x i8]* @.str2, i32 0, i32 0))
br label %sw.epilog
sw.bb5: ; preds = %entry
%call6 = call i32 @puts(i8* getelementptr inbounds ([8 x i8], [8 x i8]* @.str3, i32 0, i32 0))
br label %sw.epilog
sw.bb7: ; preds = %entry
%call8 = call i32 @puts(i8* getelementptr inbounds ([8 x i8], [8 x i8]* @.str4, i32 0, i32 0))
br label %sw.epilog
sw.epilog: ; preds = %entry, %sw.bb7, %sw.bb5, %sw.bb3, %sw.bb1, %sw.bb
ret void
}
; Test the indirect jump sequence derived from a "switch" statement.
; CHECK: TestIndirectJump:
; CHECK: andl $-32, %r11d
; CHECK-NEXT: addq %r15, %r11
; CHECK-NEXT: jmpq *%r11
; At least 4 "jmp"s due to 5 switch cases
; CHECK: jmp
; CHECK: jmp
; CHECK: jmp
; CHECK: jmp
; At least 1 direct call to puts()
; CHECK: pushq $0
; CHECK-NEXT: .text
; CHECK: jmp 0
; CHECK-NEXT: puts
declare i32 @puts(i8*) #1
; Function Attrs: nounwind
define void @TestReturn() #0 {
entry:
ret void
}
; Return sequence is just the indirect jump sequence
; CHECK: TestReturn:
; CHECK: andl $-32, %r11d
; CHECK-NEXT: addq %r15, %r11
; CHECK-NEXT: jmpq *%r11
attributes #0 = { nounwind "less-precise-fpmad"="false" "no-frame-pointer-elim"="true" "no-frame-pointer-elim-non-leaf"="true" "no-infs-fp-math"="false" "no-nans-fp-math"="false" "unsafe-fp-math"="false" "use-soft-float"="false" }
attributes #1 = { "less-precise-fpmad"="false" "no-frame-pointer-elim"="true" "no-frame-pointer-elim-non-leaf"="true" "no-infs-fp-math"="false" "no-nans-fp-math"="false" "unsafe-fp-math"="false" "use-soft-float"="false" }
; Special test that no "call" or "ret" instructions are generated.
; NOCALLRET-NOT: call
; NOCALLRET-NOT: ret