76 lines
2.9 KiB
LLVM
76 lines
2.9 KiB
LLVM
|
; RUN: llc -regalloc=greedy -mtriple=x86_64-pc-windows-msvc < %s -o - | FileCheck %s
|
||
|
|
||
|
; This test checks for proper handling of a condition where the greedy register
|
||
|
; allocator encounters a very short interval that contains no uses but does
|
||
|
; contain an EH pad unwind edge, which requires spilling. Previously the
|
||
|
; register allocator marked a interval like this as unspillable, resulting in
|
||
|
; a compilation failure.
|
||
|
|
||
|
|
||
|
; The following checks that the value %p is reloaded within the catch handler.
|
||
|
; CHECK-LABEL: "?catch$8@?0?test@4HA":
|
||
|
; CHECK: .seh_endprologue
|
||
|
; CHECK: movq -16(%rbp), %rax
|
||
|
; CHECK: movb $0, (%rax)
|
||
|
|
||
|
define i32* @test(i32* %a) personality i8* bitcast (i32 (...)* @__CxxFrameHandler3 to i8*) {
|
||
|
entry:
|
||
|
%call = call i32 @f()
|
||
|
%p = bitcast i32* %a to i8*
|
||
|
br i1 undef, label %if.end, label %if.else
|
||
|
|
||
|
if.else: ; preds = %entry
|
||
|
br i1 undef, label %cond.false.i, label %if.else.else
|
||
|
|
||
|
if.else.else: ; preds = %if.else
|
||
|
br i1 undef, label %cond.true.i, label %cond.false.i
|
||
|
|
||
|
cond.true.i: ; preds = %if.else.else
|
||
|
br label %invoke.cont
|
||
|
|
||
|
cond.false.i: ; preds = %if.else.else, %if.else
|
||
|
%call.i = invoke i32 @f()
|
||
|
to label %invoke.cont unwind label %catch.dispatch
|
||
|
|
||
|
catch.dispatch: ; preds = %cond.false.i
|
||
|
%tmp0 = catchswitch within none [label %catch] unwind label %ehcleanup
|
||
|
|
||
|
catch: ; preds = %catch.dispatch
|
||
|
%tmp1 = catchpad within %tmp0 [i8* null, i32 64, i8* null]
|
||
|
%p.0 = getelementptr inbounds i8, i8* %p, i64 0
|
||
|
store i8 0, i8* %p.0, align 8
|
||
|
invoke void @_CxxThrowException(i8* null, %eh.ThrowInfo* null) [ "funclet"(token %tmp1) ]
|
||
|
to label %noexc unwind label %ehcleanup
|
||
|
|
||
|
noexc: ; preds = %catch
|
||
|
unreachable
|
||
|
|
||
|
invoke.cont: ; preds = %cond.false.i, %cond.true.i
|
||
|
%cond.i = phi i32 [ %call, %cond.true.i ], [ %call.i, %cond.false.i ]
|
||
|
%cmp = icmp eq i32 %cond.i, -1
|
||
|
%tmp3 = select i1 %cmp, i32 4, i32 0
|
||
|
br label %if.end
|
||
|
|
||
|
if.end: ; preds = %invoke.cont, %entry
|
||
|
%state.0 = phi i32 [ %tmp3, %invoke.cont ], [ 4, %entry ]
|
||
|
%p.1 = getelementptr inbounds i8, i8* %p, i64 0
|
||
|
invoke void @g(i8* %p.1, i32 %state.0)
|
||
|
to label %invoke.cont.1 unwind label %ehcleanup
|
||
|
|
||
|
invoke.cont.1: ; preds = %if.end
|
||
|
ret i32* %a
|
||
|
|
||
|
ehcleanup: ; preds = %if.end, %catch, %catch.dispatch
|
||
|
%tmp4 = cleanuppad within none []
|
||
|
cleanupret from %tmp4 unwind to caller
|
||
|
}
|
||
|
|
||
|
%eh.ThrowInfo = type { i32, i32, i32, i32 }
|
||
|
|
||
|
declare i32 @__CxxFrameHandler3(...)
|
||
|
|
||
|
declare void @_CxxThrowException(i8*, %eh.ThrowInfo*)
|
||
|
|
||
|
declare i32 @f()
|
||
|
declare void @g(i8*, i32)
|