147 lines
4.1 KiB
LLVM
147 lines
4.1 KiB
LLVM
|
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
|
||
|
; RUN: llc < %s -mtriple=x86_64-apple-darwin | FileCheck %s
|
||
|
; RUN: opt -S -codegenprepare %s -mtriple=x86_64-apple-darwin -o - | FileCheck %s --check-prefix OPT
|
||
|
|
||
|
; Teach CGP to dup returns to enable tail call optimization.
|
||
|
; rdar://9147433
|
||
|
|
||
|
define i32 @foo(i32 %x) nounwind ssp {
|
||
|
; CHECK-LABEL: foo:
|
||
|
entry:
|
||
|
switch i32 %x, label %return [
|
||
|
i32 1, label %sw.bb
|
||
|
i32 2, label %sw.bb1
|
||
|
i32 3, label %sw.bb3
|
||
|
i32 4, label %sw.bb5
|
||
|
i32 5, label %sw.bb7
|
||
|
i32 6, label %sw.bb9
|
||
|
]
|
||
|
|
||
|
sw.bb: ; preds = %entry
|
||
|
; CHECK: jmp _f1
|
||
|
%call = tail call i32 @f1() nounwind
|
||
|
br label %return
|
||
|
|
||
|
sw.bb1: ; preds = %entry
|
||
|
; CHECK: jmp _f2
|
||
|
%call2 = tail call i32 @f2() nounwind
|
||
|
br label %return
|
||
|
|
||
|
sw.bb3: ; preds = %entry
|
||
|
; CHECK: jmp _f3
|
||
|
%call4 = tail call i32 @f3() nounwind
|
||
|
br label %return
|
||
|
|
||
|
sw.bb5: ; preds = %entry
|
||
|
; CHECK: jmp _f4
|
||
|
%call6 = tail call i32 @f4() nounwind
|
||
|
br label %return
|
||
|
|
||
|
sw.bb7: ; preds = %entry
|
||
|
; CHECK: jmp _f5
|
||
|
%call8 = tail call i32 @f5() nounwind
|
||
|
br label %return
|
||
|
|
||
|
sw.bb9: ; preds = %entry
|
||
|
; CHECK: jmp _f6
|
||
|
%call10 = tail call i32 @f6() nounwind
|
||
|
br label %return
|
||
|
|
||
|
return: ; preds = %entry, %sw.bb9, %sw.bb7, %sw.bb5, %sw.bb3, %sw.bb1, %sw.bb
|
||
|
%retval.0 = phi i32 [ %call10, %sw.bb9 ], [ %call8, %sw.bb7 ], [ %call6, %sw.bb5 ], [ %call4, %sw.bb3 ], [ %call2, %sw.bb1 ], [ %call, %sw.bb ], [ 0, %entry ]
|
||
|
ret i32 %retval.0
|
||
|
}
|
||
|
|
||
|
declare i32 @f1()
|
||
|
|
||
|
declare i32 @f2()
|
||
|
|
||
|
declare i32 @f3()
|
||
|
|
||
|
declare i32 @f4()
|
||
|
|
||
|
declare i32 @f5()
|
||
|
|
||
|
declare i32 @f6()
|
||
|
|
||
|
; rdar://11958338
|
||
|
%0 = type opaque
|
||
|
|
||
|
declare i8* @bar(i8*) uwtable optsize noinline ssp
|
||
|
|
||
|
define hidden %0* @thingWithValue(i8* %self) uwtable ssp {
|
||
|
entry:
|
||
|
; CHECK-LABEL: thingWithValue:
|
||
|
; CHECK: jmp _bar
|
||
|
br i1 undef, label %if.then.i, label %if.else.i
|
||
|
|
||
|
if.then.i: ; preds = %entry
|
||
|
br label %someThingWithValue.exit
|
||
|
|
||
|
if.else.i: ; preds = %entry
|
||
|
%call4.i = tail call i8* @bar(i8* undef) optsize
|
||
|
br label %someThingWithValue.exit
|
||
|
|
||
|
someThingWithValue.exit: ; preds = %if.else.i, %if.then.i
|
||
|
%retval.0.in.i = phi i8* [ undef, %if.then.i ], [ %call4.i, %if.else.i ]
|
||
|
%retval.0.i = bitcast i8* %retval.0.in.i to %0*
|
||
|
ret %0* %retval.0.i
|
||
|
}
|
||
|
|
||
|
|
||
|
; Correctly handle zext returns.
|
||
|
declare zeroext i1 @foo_i1()
|
||
|
|
||
|
; CHECK-LABEL: zext_i1
|
||
|
; CHECK: jmp _foo_i1
|
||
|
define zeroext i1 @zext_i1(i1 %k) {
|
||
|
entry:
|
||
|
br i1 %k, label %land.end, label %land.rhs
|
||
|
|
||
|
land.rhs: ; preds = %entry
|
||
|
%call1 = tail call zeroext i1 @foo_i1()
|
||
|
br label %land.end
|
||
|
|
||
|
land.end: ; preds = %entry, %land.rhs
|
||
|
%0 = phi i1 [ false, %entry ], [ %call1, %land.rhs ]
|
||
|
ret i1 %0
|
||
|
}
|
||
|
|
||
|
; We need to look through bitcasts when looking for tail calls in phi incoming
|
||
|
; values.
|
||
|
declare i32* @g_ret32()
|
||
|
define i8* @f_ret8(i8* %obj) nounwind {
|
||
|
; OPT-LABEL: @f_ret8(
|
||
|
; OPT-NEXT: entry:
|
||
|
; OPT-NEXT: [[CMP:%.*]] = icmp eq i8* [[OBJ:%.*]], null
|
||
|
; OPT-NEXT: br i1 [[CMP]], label [[RETURN:%.*]], label [[IF_THEN:%.*]]
|
||
|
; OPT: if.then:
|
||
|
; OPT-NEXT: [[PTR:%.*]] = tail call i32* @g_ret32()
|
||
|
; OPT-NEXT: [[CASTED:%.*]] = bitcast i32* [[PTR]] to i8*
|
||
|
; OPT-NEXT: ret i8* [[CASTED]]
|
||
|
; OPT: return:
|
||
|
; OPT-NEXT: ret i8* [[OBJ]]
|
||
|
;
|
||
|
; CHECK-LABEL: f_ret8:
|
||
|
; CHECK: ## %bb.0: ## %entry
|
||
|
; CHECK-NEXT: testq %rdi, %rdi
|
||
|
; CHECK-NEXT: je LBB3_1
|
||
|
; CHECK-NEXT: ## %bb.2: ## %if.then
|
||
|
; CHECK-NEXT: jmp _g_ret32 ## TAILCALL
|
||
|
; CHECK-NEXT: LBB3_1: ## %return
|
||
|
; CHECK-NEXT: movq %rdi, %rax
|
||
|
; CHECK-NEXT: retq
|
||
|
entry:
|
||
|
%cmp = icmp eq i8* %obj, null
|
||
|
br i1 %cmp, label %return, label %if.then
|
||
|
|
||
|
if.then:
|
||
|
%ptr = tail call i32* @g_ret32()
|
||
|
%casted = bitcast i32* %ptr to i8*
|
||
|
br label %return
|
||
|
|
||
|
return:
|
||
|
%retval = phi i8* [ %casted, %if.then ], [ %obj, %entry ]
|
||
|
ret i8* %retval
|
||
|
}
|