160 lines
4.0 KiB
LLVM
160 lines
4.0 KiB
LLVM
|
; RUN: llc -mtriple=x86_64-pc-windows-msvc < %s | FileCheck %s
|
||
|
|
||
|
declare i32 @__CxxFrameHandler3(...)
|
||
|
|
||
|
declare void @throw()
|
||
|
declare i16 @f()
|
||
|
|
||
|
define i16 @test1(i16 %a, i8* %b) personality i8* bitcast (i32 (...)* @__CxxFrameHandler3 to i8*) {
|
||
|
entry:
|
||
|
%cmp = icmp eq i16 %a, 10
|
||
|
br i1 %cmp, label %if.then, label %if.else
|
||
|
|
||
|
if.then:
|
||
|
%call1 = invoke i16 @f()
|
||
|
to label %cleanup unwind label %catch.dispatch
|
||
|
|
||
|
if.else:
|
||
|
%call2 = invoke i16 @f()
|
||
|
to label %cleanup unwind label %catch.dispatch
|
||
|
|
||
|
catch.dispatch:
|
||
|
%cs = catchswitch within none [ label %catch, label %catch.2 ] unwind to caller
|
||
|
|
||
|
catch:
|
||
|
catchpad within %cs [i8* null, i32 8, i8* null]
|
||
|
call void @throw() noreturn
|
||
|
br label %unreachable
|
||
|
|
||
|
catch.2:
|
||
|
catchpad within %cs [i8* null, i32 64, i8* null]
|
||
|
store i8 1, i8* %b
|
||
|
call void @throw() noreturn
|
||
|
br label %unreachable
|
||
|
|
||
|
cleanup:
|
||
|
%retval = phi i16 [ %call1, %if.then ], [ %call2, %if.else ]
|
||
|
ret i16 %retval
|
||
|
|
||
|
unreachable:
|
||
|
unreachable
|
||
|
}
|
||
|
|
||
|
; This test verifies the case where two funclet blocks meet the old criteria
|
||
|
; to be placed at the end. The order of the blocks is not important for the
|
||
|
; purposes of this test. The failure mode is an infinite loop during
|
||
|
; compilation.
|
||
|
;
|
||
|
; CHECK-LABEL: .def test1;
|
||
|
|
||
|
define i16 @test2(i16 %a, i8* %b) personality i8* bitcast (i32 (...)* @__CxxFrameHandler3 to i8*) {
|
||
|
entry:
|
||
|
%cmp = icmp eq i16 %a, 10
|
||
|
br i1 %cmp, label %if.then, label %if.else
|
||
|
|
||
|
if.then:
|
||
|
%call1 = invoke i16 @f()
|
||
|
to label %cleanup unwind label %catch.dispatch
|
||
|
|
||
|
if.else:
|
||
|
%call2 = invoke i16 @f()
|
||
|
to label %cleanup unwind label %catch.dispatch
|
||
|
|
||
|
catch.dispatch:
|
||
|
%cs = catchswitch within none [ label %catch, label %catch.2, label %catch.3 ] unwind to caller
|
||
|
|
||
|
catch:
|
||
|
catchpad within %cs [i8* null, i32 8, i8* null]
|
||
|
call void @throw() noreturn
|
||
|
br label %unreachable
|
||
|
|
||
|
catch.2:
|
||
|
%c2 = catchpad within %cs [i8* null, i32 32, i8* null]
|
||
|
store i8 1, i8* %b
|
||
|
catchret from %c2 to label %cleanup
|
||
|
|
||
|
catch.3:
|
||
|
%c3 = catchpad within %cs [i8* null, i32 64, i8* null]
|
||
|
store i8 2, i8* %b
|
||
|
catchret from %c3 to label %cleanup
|
||
|
|
||
|
cleanup:
|
||
|
%retval = phi i16 [ %call1, %if.then ], [ %call2, %if.else ], [ -1, %catch.2 ], [ -1, %catch.3 ]
|
||
|
ret i16 %retval
|
||
|
|
||
|
unreachable:
|
||
|
unreachable
|
||
|
}
|
||
|
|
||
|
; This test verifies the case where three funclet blocks all meet the old
|
||
|
; criteria to be placed at the end. The order of the blocks is not important
|
||
|
; for the purposes of this test. The failure mode is an infinite loop during
|
||
|
; compilation.
|
||
|
;
|
||
|
; CHECK-LABEL: .def test2;
|
||
|
|
||
|
declare void @g()
|
||
|
|
||
|
define void @test3() optsize personality i8* bitcast (i32 (...)* @__CxxFrameHandler3 to i8*) {
|
||
|
entry:
|
||
|
switch i32 undef, label %if.end57 [
|
||
|
i32 64, label %sw.bb
|
||
|
i32 128, label %sw.epilog
|
||
|
i32 256, label %if.then56
|
||
|
i32 1024, label %sw.bb
|
||
|
i32 4096, label %sw.bb33
|
||
|
i32 16, label %sw.epilog
|
||
|
i32 8, label %sw.epilog
|
||
|
i32 32, label %sw.bb44
|
||
|
]
|
||
|
|
||
|
sw.bb:
|
||
|
unreachable
|
||
|
|
||
|
sw.bb33:
|
||
|
br i1 undef, label %if.end57, label %while.cond.i163.preheader
|
||
|
|
||
|
while.cond.i163.preheader:
|
||
|
unreachable
|
||
|
|
||
|
sw.bb44:
|
||
|
%temp0 = load void ()*, void ()** undef
|
||
|
invoke void %temp0()
|
||
|
to label %if.end57 unwind label %catch.dispatch
|
||
|
|
||
|
sw.epilog:
|
||
|
%temp1 = load i8*, i8** undef
|
||
|
br label %if.end57
|
||
|
|
||
|
catch.dispatch:
|
||
|
%cs = catchswitch within none [label %catch1, label %catch2, label %catch3] unwind to caller
|
||
|
|
||
|
catch1:
|
||
|
%c1 = catchpad within %cs [i8* null, i32 8, i8* null]
|
||
|
unreachable
|
||
|
|
||
|
catch2:
|
||
|
%c2 = catchpad within %cs [i8* null, i32 32, i8* null]
|
||
|
unreachable
|
||
|
|
||
|
catch3:
|
||
|
%c3 = catchpad within %cs [i8* null, i32 64, i8* null]
|
||
|
unreachable
|
||
|
|
||
|
if.then56:
|
||
|
call void @g()
|
||
|
br label %if.end57
|
||
|
|
||
|
if.end57:
|
||
|
ret void
|
||
|
}
|
||
|
|
||
|
; This test exercises a complex case that produced an infinite loop during
|
||
|
; compilation when the two cases above did not. The multiple targets from the
|
||
|
; entry switch are not actually fundamental to the failure, but they are
|
||
|
; necessary to suppress various control flow optimizations that would prevent
|
||
|
; the conditions that lead to the failure.
|
||
|
;
|
||
|
; CHECK-LABEL: .def test3;
|
||
|
|