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;
 | 
						|
 |