192 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			LLVM
		
	
	
	
			
		
		
	
	
			192 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			LLVM
		
	
	
	
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
 | 
						|
; RUN: opt %s -simplifycfg -simplifycfg-require-and-preserve-domtree=1 -switch-range-to-icmp -S | FileCheck %s
 | 
						|
 | 
						|
declare i32 @f(i32)
 | 
						|
 | 
						|
define i32 @basic(i32 %x) {
 | 
						|
; CHECK-LABEL: @basic(
 | 
						|
; CHECK-NEXT:  entry:
 | 
						|
; CHECK-NEXT:    [[X_OFF:%.*]] = add i32 [[X:%.*]], -5
 | 
						|
; CHECK-NEXT:    [[SWITCH:%.*]] = icmp ult i32 [[X_OFF]], 3
 | 
						|
; CHECK-NEXT:    br i1 [[SWITCH]], label [[A:%.*]], label [[DEFAULT:%.*]]
 | 
						|
; CHECK:       common.ret:
 | 
						|
; CHECK-NEXT:    [[COMMON_RET_OP:%.*]] = phi i32 [ [[TMP0:%.*]], [[DEFAULT]] ], [ [[TMP1:%.*]], [[A]] ]
 | 
						|
; CHECK-NEXT:    ret i32 [[COMMON_RET_OP]]
 | 
						|
; CHECK:       default:
 | 
						|
; CHECK-NEXT:    [[TMP0]] = call i32 @f(i32 0)
 | 
						|
; CHECK-NEXT:    br label [[COMMON_RET:%.*]]
 | 
						|
; CHECK:       a:
 | 
						|
; CHECK-NEXT:    [[TMP1]] = call i32 @f(i32 1)
 | 
						|
; CHECK-NEXT:    br label [[COMMON_RET]]
 | 
						|
;
 | 
						|
 | 
						|
entry:
 | 
						|
  switch i32 %x, label %default [
 | 
						|
  i32 5, label %a
 | 
						|
  i32 6, label %a
 | 
						|
  i32 7, label %a
 | 
						|
  ]
 | 
						|
default:
 | 
						|
  %0 = call i32 @f(i32 0)
 | 
						|
  ret i32 %0
 | 
						|
a:
 | 
						|
  %1 = call i32 @f(i32 1)
 | 
						|
  ret i32 %1
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
define i32 @unreachable(i32 %x) {
 | 
						|
; CHECK-LABEL: @unreachable(
 | 
						|
; CHECK-NEXT:  entry:
 | 
						|
; CHECK-NEXT:    [[X_OFF:%.*]] = add i32 [[X:%.*]], -5
 | 
						|
; CHECK-NEXT:    [[SWITCH:%.*]] = icmp ult i32 [[X_OFF]], 3
 | 
						|
; CHECK-NEXT:    br i1 [[SWITCH]], label [[A:%.*]], label [[B:%.*]]
 | 
						|
; CHECK:       common.ret:
 | 
						|
; CHECK-NEXT:    [[COMMON_RET_OP:%.*]] = phi i32 [ [[TMP0:%.*]], [[A]] ], [ [[TMP1:%.*]], [[B]] ]
 | 
						|
; CHECK-NEXT:    ret i32 [[COMMON_RET_OP]]
 | 
						|
; CHECK:       a:
 | 
						|
; CHECK-NEXT:    [[TMP0]] = call i32 @f(i32 0)
 | 
						|
; CHECK-NEXT:    br label [[COMMON_RET:%.*]]
 | 
						|
; CHECK:       b:
 | 
						|
; CHECK-NEXT:    [[TMP1]] = call i32 @f(i32 1)
 | 
						|
; CHECK-NEXT:    br label [[COMMON_RET]]
 | 
						|
;
 | 
						|
 | 
						|
entry:
 | 
						|
  switch i32 %x, label %unreachable [
 | 
						|
  i32 5, label %a
 | 
						|
  i32 6, label %a
 | 
						|
  i32 7, label %a
 | 
						|
  i32 10, label %b
 | 
						|
  i32 20, label %b
 | 
						|
  i32 30, label %b
 | 
						|
  i32 40, label %b
 | 
						|
  ]
 | 
						|
unreachable:
 | 
						|
  unreachable
 | 
						|
a:
 | 
						|
  %0 = call i32 @f(i32 0)
 | 
						|
  ret i32 %0
 | 
						|
b:
 | 
						|
  %1 = call i32 @f(i32 1)
 | 
						|
  ret i32 %1
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
define i32 @unreachable2(i32 %x) {
 | 
						|
; CHECK-LABEL: @unreachable2(
 | 
						|
; CHECK-NEXT:  entry:
 | 
						|
; CHECK-NEXT:    [[X_OFF:%.*]] = add i32 [[X:%.*]], -5
 | 
						|
; CHECK-NEXT:    [[SWITCH:%.*]] = icmp ult i32 [[X_OFF]], 3
 | 
						|
; CHECK-NEXT:    br i1 [[SWITCH]], label [[A:%.*]], label [[B:%.*]]
 | 
						|
; CHECK:       common.ret:
 | 
						|
; CHECK-NEXT:    [[COMMON_RET_OP:%.*]] = phi i32 [ [[TMP0:%.*]], [[A]] ], [ [[TMP1:%.*]], [[B]] ]
 | 
						|
; CHECK-NEXT:    ret i32 [[COMMON_RET_OP]]
 | 
						|
; CHECK:       a:
 | 
						|
; CHECK-NEXT:    [[TMP0]] = call i32 @f(i32 0)
 | 
						|
; CHECK-NEXT:    br label [[COMMON_RET:%.*]]
 | 
						|
; CHECK:       b:
 | 
						|
; CHECK-NEXT:    [[TMP1]] = call i32 @f(i32 1)
 | 
						|
; CHECK-NEXT:    br label [[COMMON_RET]]
 | 
						|
;
 | 
						|
 | 
						|
entry:
 | 
						|
  ; Note: folding the most popular case destination into the default
 | 
						|
  ; would prevent switch-to-icmp here.
 | 
						|
  switch i32 %x, label %unreachable [
 | 
						|
  i32 5, label %a
 | 
						|
  i32 6, label %a
 | 
						|
  i32 7, label %a
 | 
						|
  i32 10, label %b
 | 
						|
  i32 20, label %b
 | 
						|
  ]
 | 
						|
unreachable:
 | 
						|
  unreachable
 | 
						|
a:
 | 
						|
  %0 = call i32 @f(i32 0)
 | 
						|
  ret i32 %0
 | 
						|
b:
 | 
						|
  %1 = call i32 @f(i32 1)
 | 
						|
  ret i32 %1
 | 
						|
}
 | 
						|
 | 
						|
; This would crash because we did not clean up the
 | 
						|
; default block of the switch before removing the switch.
 | 
						|
 | 
						|
define void @PR42737(i32* %a, i1 %c) {
 | 
						|
; CHECK-LABEL: @PR42737(
 | 
						|
; CHECK-NEXT:  entry:
 | 
						|
; CHECK-NEXT:    unreachable
 | 
						|
;
 | 
						|
entry:
 | 
						|
  br i1 %c, label %switch, label %else
 | 
						|
 | 
						|
else:
 | 
						|
  store i32 2, i32* %a
 | 
						|
  br label %switch
 | 
						|
 | 
						|
switch:
 | 
						|
  %cleanup.dest1 = phi i32 [ 0, %else ], [ 3, %entry ]
 | 
						|
  switch i32 %cleanup.dest1, label %unreach1 [
 | 
						|
  i32 0, label %cleanup1
 | 
						|
  i32 3, label %cleanup2
 | 
						|
  ]
 | 
						|
 | 
						|
cleanup1:
 | 
						|
  br label %unreach2
 | 
						|
 | 
						|
cleanup2:
 | 
						|
  br label %unreach2
 | 
						|
 | 
						|
unreach1:
 | 
						|
  %phi2 = phi i32 [ %cleanup.dest1, %switch ]
 | 
						|
  unreachable
 | 
						|
 | 
						|
unreach2:
 | 
						|
  unreachable
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
define void @pr53208_single_reachable_dest(i8 %sw, i32* %p0) {
 | 
						|
; CHECK-LABEL: @pr53208_single_reachable_dest(
 | 
						|
; CHECK-NEXT:  group2:
 | 
						|
; CHECK-NEXT:    call void @bar(i32* [[P0:%.*]])
 | 
						|
; CHECK-NEXT:    ret void
 | 
						|
;
 | 
						|
  switch i8 %sw, label %group3 [
 | 
						|
  i8 0, label %group1
 | 
						|
  i8 1, label %group1
 | 
						|
  i8 2, label %group1
 | 
						|
  i8 3, label %group1
 | 
						|
  i8 11, label %group1
 | 
						|
  i8 12, label %group1
 | 
						|
  i8 13, label %group1
 | 
						|
  i8 7, label %group1
 | 
						|
  i8 17, label %group1
 | 
						|
  i8 14, label %group1
 | 
						|
  i8 15, label %group1
 | 
						|
  i8 4, label %group2
 | 
						|
  i8 5, label %group2
 | 
						|
  i8 6, label %group2
 | 
						|
  i8 8, label %group2
 | 
						|
  i8 9, label %group2
 | 
						|
  i8 10, label %group2
 | 
						|
  ]
 | 
						|
 | 
						|
group1:
 | 
						|
  br label %exit
 | 
						|
 | 
						|
group2:
 | 
						|
  br label %exit
 | 
						|
 | 
						|
group3:
 | 
						|
  br label %exit
 | 
						|
 | 
						|
exit:
 | 
						|
  %phi = phi i32* [ null, %group3 ], [ %p0, %group2 ], [ null, %group1 ]
 | 
						|
  call void @bar(i32* %phi)
 | 
						|
  ret void
 | 
						|
}
 | 
						|
 | 
						|
declare void @bar(i32* nonnull dereferenceable(4))
 |