forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			348 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			LLVM
		
	
	
	
			
		
		
	
	
			348 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			LLVM
		
	
	
	
| ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
 | |
| ; RUN: opt < %s -simplifycfg -switch-to-lookup -S | FileCheck %s
 | |
| ; RUN: opt < %s -passes='simplify-cfg<switch-to-lookup>' -S | FileCheck %s
 | |
| 
 | |
| target datalayout = "e-n32"
 | |
| 
 | |
| define i32 @test1(i32 %a) {
 | |
| ; CHECK-LABEL: @test1(
 | |
| ; CHECK-NEXT:    [[TMP1:%.*]] = sub i32 [[A:%.*]], 97
 | |
| ; CHECK-NEXT:    [[TMP2:%.*]] = lshr i32 [[TMP1]], 2
 | |
| ; CHECK-NEXT:    [[TMP3:%.*]] = shl i32 [[TMP1]], 30
 | |
| ; CHECK-NEXT:    [[TMP4:%.*]] = or i32 [[TMP2]], [[TMP3]]
 | |
| ; CHECK-NEXT:    switch i32 [[TMP4]], label [[DEF:%.*]] [
 | |
| ; CHECK-NEXT:    i32 0, label [[ONE:%.*]]
 | |
| ; CHECK-NEXT:    i32 1, label [[TWO:%.*]]
 | |
| ; CHECK-NEXT:    i32 2, label [[THREE:%.*]]
 | |
| ; CHECK-NEXT:    i32 3, label [[THREE]]
 | |
| ; CHECK-NEXT:    ]
 | |
| ; CHECK:       def:
 | |
| ; CHECK-NEXT:    [[MERGE:%.*]] = phi i32 [ 8867, [[TMP0:%.*]] ], [ 11984, [[ONE]] ], [ 1143, [[TWO]] ], [ 99783, [[THREE]] ]
 | |
| ; CHECK-NEXT:    ret i32 [[MERGE]]
 | |
| ; CHECK:       one:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       two:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       three:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ;
 | |
|   switch i32 %a, label %def [
 | |
|   i32 97, label %one
 | |
|   i32 101, label %two
 | |
|   i32 105, label %three
 | |
|   i32 109, label %three
 | |
|   ]
 | |
| 
 | |
| def:
 | |
|   ret i32 8867
 | |
| 
 | |
| one:
 | |
|   ret i32 11984
 | |
| two:
 | |
|   ret i32 1143
 | |
| three:
 | |
|   ret i32 99783
 | |
| }
 | |
| 
 | |
| ; Optimization shouldn't trigger; bitwidth > 64
 | |
| define i128 @test2(i128 %a) {
 | |
| ; CHECK-LABEL: @test2(
 | |
| ; CHECK-NEXT:    switch i128 [[A:%.*]], label [[DEF:%.*]] [
 | |
| ; CHECK-NEXT:    i128 97, label [[ONE:%.*]]
 | |
| ; CHECK-NEXT:    i128 101, label [[TWO:%.*]]
 | |
| ; CHECK-NEXT:    i128 105, label [[THREE:%.*]]
 | |
| ; CHECK-NEXT:    i128 109, label [[THREE]]
 | |
| ; CHECK-NEXT:    ]
 | |
| ; CHECK:       def:
 | |
| ; CHECK-NEXT:    [[MERGE:%.*]] = phi i128 [ 8867, [[TMP0:%.*]] ], [ 11984, [[ONE]] ], [ 1143, [[TWO]] ], [ 99783, [[THREE]] ]
 | |
| ; CHECK-NEXT:    ret i128 [[MERGE]]
 | |
| ; CHECK:       one:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       two:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       three:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ;
 | |
|   switch i128 %a, label %def [
 | |
|   i128 97, label %one
 | |
|   i128 101, label %two
 | |
|   i128 105, label %three
 | |
|   i128 109, label %three
 | |
|   ]
 | |
| 
 | |
| def:
 | |
|   ret i128 8867
 | |
| 
 | |
| one:
 | |
|   ret i128 11984
 | |
| two:
 | |
|   ret i128 1143
 | |
| three:
 | |
|   ret i128 99783
 | |
| }
 | |
| 
 | |
| ; Optimization shouldn't trigger; no holes present
 | |
| define i32 @test3(i32 %a) {
 | |
| ; CHECK-LABEL: @test3(
 | |
| ; CHECK-NEXT:    switch i32 [[A:%.*]], label [[DEF:%.*]] [
 | |
| ; CHECK-NEXT:    i32 97, label [[ONE:%.*]]
 | |
| ; CHECK-NEXT:    i32 98, label [[TWO:%.*]]
 | |
| ; CHECK-NEXT:    i32 99, label [[THREE:%.*]]
 | |
| ; CHECK-NEXT:    ]
 | |
| ; CHECK:       def:
 | |
| ; CHECK-NEXT:    [[MERGE:%.*]] = phi i32 [ 8867, [[TMP0:%.*]] ], [ 11984, [[ONE]] ], [ 1143, [[TWO]] ], [ 99783, [[THREE]] ]
 | |
| ; CHECK-NEXT:    ret i32 [[MERGE]]
 | |
| ; CHECK:       one:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       two:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       three:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ;
 | |
|   switch i32 %a, label %def [
 | |
|   i32 97, label %one
 | |
|   i32 98, label %two
 | |
|   i32 99, label %three
 | |
|   ]
 | |
| 
 | |
| def:
 | |
|   ret i32 8867
 | |
| 
 | |
| one:
 | |
|   ret i32 11984
 | |
| two:
 | |
|   ret i32 1143
 | |
| three:
 | |
|   ret i32 99783
 | |
| }
 | |
| 
 | |
| ; Optimization shouldn't trigger; not an arithmetic progression
 | |
| define i32 @test4(i32 %a) {
 | |
| ; CHECK-LABEL: @test4(
 | |
| ; CHECK-NEXT:    switch i32 [[A:%.*]], label [[DEF:%.*]] [
 | |
| ; CHECK-NEXT:    i32 97, label [[ONE:%.*]]
 | |
| ; CHECK-NEXT:    i32 102, label [[TWO:%.*]]
 | |
| ; CHECK-NEXT:    i32 105, label [[THREE:%.*]]
 | |
| ; CHECK-NEXT:    i32 109, label [[THREE]]
 | |
| ; CHECK-NEXT:    ]
 | |
| ; CHECK:       def:
 | |
| ; CHECK-NEXT:    [[MERGE:%.*]] = phi i32 [ 8867, [[TMP0:%.*]] ], [ 11984, [[ONE]] ], [ 1143, [[TWO]] ], [ 99783, [[THREE]] ]
 | |
| ; CHECK-NEXT:    ret i32 [[MERGE]]
 | |
| ; CHECK:       one:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       two:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       three:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ;
 | |
|   switch i32 %a, label %def [
 | |
|   i32 97, label %one
 | |
|   i32 102, label %two
 | |
|   i32 105, label %three
 | |
|   i32 109, label %three
 | |
|   ]
 | |
| 
 | |
| def:
 | |
|   ret i32 8867
 | |
| 
 | |
| one:
 | |
|   ret i32 11984
 | |
| two:
 | |
|   ret i32 1143
 | |
| three:
 | |
|   ret i32 99783
 | |
| }
 | |
| 
 | |
| ; Optimization shouldn't trigger; not a power of two
 | |
| define i32 @test5(i32 %a) {
 | |
| ; CHECK-LABEL: @test5(
 | |
| ; CHECK-NEXT:    switch i32 [[A:%.*]], label [[DEF:%.*]] [
 | |
| ; CHECK-NEXT:    i32 97, label [[ONE:%.*]]
 | |
| ; CHECK-NEXT:    i32 102, label [[TWO:%.*]]
 | |
| ; CHECK-NEXT:    i32 107, label [[THREE:%.*]]
 | |
| ; CHECK-NEXT:    i32 112, label [[THREE]]
 | |
| ; CHECK-NEXT:    ]
 | |
| ; CHECK:       def:
 | |
| ; CHECK-NEXT:    [[MERGE:%.*]] = phi i32 [ 8867, [[TMP0:%.*]] ], [ 11984, [[ONE]] ], [ 1143, [[TWO]] ], [ 99783, [[THREE]] ]
 | |
| ; CHECK-NEXT:    ret i32 [[MERGE]]
 | |
| ; CHECK:       one:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       two:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       three:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ;
 | |
|   switch i32 %a, label %def [
 | |
|   i32 97, label %one
 | |
|   i32 102, label %two
 | |
|   i32 107, label %three
 | |
|   i32 112, label %three
 | |
|   ]
 | |
| 
 | |
| def:
 | |
|   ret i32 8867
 | |
| 
 | |
| one:
 | |
|   ret i32 11984
 | |
| two:
 | |
|   ret i32 1143
 | |
| three:
 | |
|   ret i32 99783
 | |
| }
 | |
| 
 | |
| define i32 @test6(i32 %a) optsize {
 | |
| ; CHECK-LABEL: @test6(
 | |
| ; CHECK-NEXT:    [[TMP1:%.*]] = sub i32 [[A:%.*]], -109
 | |
| ; CHECK-NEXT:    [[TMP2:%.*]] = lshr i32 [[TMP1]], 2
 | |
| ; CHECK-NEXT:    [[TMP3:%.*]] = shl i32 [[TMP1]], 30
 | |
| ; CHECK-NEXT:    [[TMP4:%.*]] = or i32 [[TMP2]], [[TMP3]]
 | |
| ; CHECK-NEXT:    switch i32 [[TMP4]], label [[DEF:%.*]] [
 | |
| ; CHECK-NEXT:    i32 3, label [[ONE:%.*]]
 | |
| ; CHECK-NEXT:    i32 2, label [[TWO:%.*]]
 | |
| ; CHECK-NEXT:    i32 1, label [[THREE:%.*]]
 | |
| ; CHECK-NEXT:    i32 0, label [[THREE]]
 | |
| ; CHECK-NEXT:    ]
 | |
| ; CHECK:       def:
 | |
| ; CHECK-NEXT:    [[MERGE:%.*]] = phi i32 [ 8867, [[TMP0:%.*]] ], [ 11984, [[ONE]] ], [ 1143, [[TWO]] ], [ 99783, [[THREE]] ]
 | |
| ; CHECK-NEXT:    ret i32 [[MERGE]]
 | |
| ; CHECK:       one:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       two:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       three:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ;
 | |
|   switch i32 %a, label %def [
 | |
|   i32 -97, label %one
 | |
|   i32 -101, label %two
 | |
|   i32 -105, label %three
 | |
|   i32 -109, label %three
 | |
|   ]
 | |
| 
 | |
| def:
 | |
|   ret i32 8867
 | |
| 
 | |
| one:
 | |
|   ret i32 11984
 | |
| two:
 | |
|   ret i32 1143
 | |
| three:
 | |
|   ret i32 99783
 | |
| }
 | |
| 
 | |
| define i8 @test7(i8 %a) optsize {
 | |
| ; CHECK-LABEL: @test7(
 | |
| ; CHECK-NEXT:    [[TMP1:%.*]] = sub i8 [[A:%.*]], -36
 | |
| ; CHECK-NEXT:    [[TMP2:%.*]] = lshr i8 [[TMP1]], 2
 | |
| ; CHECK-NEXT:    [[TMP3:%.*]] = shl i8 [[TMP1]], 6
 | |
| ; CHECK-NEXT:    [[TMP4:%.*]] = or i8 [[TMP2]], [[TMP3]]
 | |
| ; CHECK-NEXT:    [[TMP5:%.*]] = icmp ult i8 [[TMP4]], 4
 | |
| ; CHECK-NEXT:    br i1 [[TMP5]], label [[SWITCH_LOOKUP:%.*]], label [[DEF:%.*]]
 | |
| ; CHECK:       switch.lookup:
 | |
| ; CHECK-NEXT:    [[SWITCH_CAST:%.*]] = zext i8 [[TMP4]] to i32
 | |
| ; CHECK-NEXT:    [[SWITCH_SHIFTAMT:%.*]] = mul i32 [[SWITCH_CAST]], 8
 | |
| ; CHECK-NEXT:    [[SWITCH_DOWNSHIFT:%.*]] = lshr i32 -943228976, [[SWITCH_SHIFTAMT]]
 | |
| ; CHECK-NEXT:    [[SWITCH_MASKED:%.*]] = trunc i32 [[SWITCH_DOWNSHIFT]] to i8
 | |
| ; CHECK-NEXT:    ret i8 [[SWITCH_MASKED]]
 | |
| ; CHECK:       def:
 | |
| ; CHECK-NEXT:    ret i8 -93
 | |
| ;
 | |
|   switch i8 %a, label %def [
 | |
|   i8 220, label %one
 | |
|   i8 224, label %two
 | |
|   i8 228, label %three
 | |
|   i8 232, label %three
 | |
|   ]
 | |
| 
 | |
| def:
 | |
|   ret i8 8867
 | |
| 
 | |
| one:
 | |
|   ret i8 11984
 | |
| two:
 | |
|   ret i8 1143
 | |
| three:
 | |
|   ret i8 99783
 | |
| }
 | |
| 
 | |
| define i32 @test8(i32 %a) optsize {
 | |
| ; CHECK-LABEL: @test8(
 | |
| ; CHECK-NEXT:    [[TMP1:%.*]] = sub i32 [[A:%.*]], 97
 | |
| ; CHECK-NEXT:    [[TMP2:%.*]] = lshr i32 [[TMP1]], 2
 | |
| ; CHECK-NEXT:    [[TMP3:%.*]] = shl i32 [[TMP1]], 30
 | |
| ; CHECK-NEXT:    [[TMP4:%.*]] = or i32 [[TMP2]], [[TMP3]]
 | |
| ; CHECK-NEXT:    switch i32 [[TMP4]], label [[DEF:%.*]] [
 | |
| ; CHECK-NEXT:    i32 0, label [[ONE:%.*]]
 | |
| ; CHECK-NEXT:    i32 1, label [[TWO:%.*]]
 | |
| ; CHECK-NEXT:    i32 2, label [[THREE:%.*]]
 | |
| ; CHECK-NEXT:    i32 4, label [[THREE]]
 | |
| ; CHECK-NEXT:    ]
 | |
| ; CHECK:       def:
 | |
| ; CHECK-NEXT:    [[MERGE:%.*]] = phi i32 [ 8867, [[TMP0:%.*]] ], [ 11984, [[ONE]] ], [ 1143, [[TWO]] ], [ 99783, [[THREE]] ]
 | |
| ; CHECK-NEXT:    ret i32 [[MERGE]]
 | |
| ; CHECK:       one:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       two:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       three:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ;
 | |
|   switch i32 %a, label %def [
 | |
|   i32 97, label %one
 | |
|   i32 101, label %two
 | |
|   i32 105, label %three
 | |
|   i32 113, label %three
 | |
|   ]
 | |
| 
 | |
| def:
 | |
|   ret i32 8867
 | |
| 
 | |
| one:
 | |
|   ret i32 11984
 | |
| two:
 | |
|   ret i32 1143
 | |
| three:
 | |
|   ret i32 99783
 | |
| }
 | |
| 
 | |
| define i32 @test9(i32 %a) {
 | |
| ; CHECK-LABEL: @test9(
 | |
| ; CHECK-NEXT:    [[TMP1:%.*]] = sub i32 [[A:%.*]], 6
 | |
| ; CHECK-NEXT:    [[TMP2:%.*]] = lshr i32 [[TMP1]], 1
 | |
| ; CHECK-NEXT:    [[TMP3:%.*]] = shl i32 [[TMP1]], 31
 | |
| ; CHECK-NEXT:    [[TMP4:%.*]] = or i32 [[TMP2]], [[TMP3]]
 | |
| ; CHECK-NEXT:    switch i32 [[TMP4]], label [[DEF:%.*]] [
 | |
| ; CHECK-NEXT:    i32 6, label [[ONE:%.*]]
 | |
| ; CHECK-NEXT:    i32 7, label [[TWO:%.*]]
 | |
| ; CHECK-NEXT:    i32 0, label [[THREE:%.*]]
 | |
| ; CHECK-NEXT:    i32 2, label [[THREE]]
 | |
| ; CHECK-NEXT:    ]
 | |
| ; CHECK:       def:
 | |
| ; CHECK-NEXT:    [[MERGE:%.*]] = phi i32 [ 8867, [[TMP0:%.*]] ], [ 11984, [[ONE]] ], [ 1143, [[TWO]] ], [ 99783, [[THREE]] ]
 | |
| ; CHECK-NEXT:    ret i32 [[MERGE]]
 | |
| ; CHECK:       one:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       two:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ; CHECK:       three:
 | |
| ; CHECK-NEXT:    br label [[DEF]]
 | |
| ;
 | |
|   switch i32 %a, label %def [
 | |
|   i32 18, label %one
 | |
|   i32 20, label %two
 | |
|   i32 6, label %three
 | |
|   i32 10, label %three
 | |
|   ]
 | |
| 
 | |
| def:
 | |
|   ret i32 8867
 | |
| 
 | |
| one:
 | |
|   ret i32 11984
 | |
| two:
 | |
|   ret i32 1143
 | |
| three:
 | |
|   ret i32 99783
 | |
| }
 | |
| 
 |