264 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			LLVM
		
	
	
	
			
		
		
	
	
			264 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			LLVM
		
	
	
	
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
 | 
						|
; Replace a 'select' with 'or' in 'select - cmp [eq|ne] - br' sequence
 | 
						|
; RUN: opt -instcombine -S < %s | FileCheck %s
 | 
						|
 | 
						|
%struct.S = type { i64*, i32, i32 }
 | 
						|
%C = type <{ %struct.S }>
 | 
						|
 | 
						|
declare void @bar(%struct.S*)
 | 
						|
declare void @foobar()
 | 
						|
 | 
						|
define void @test1(%C* %arg) {
 | 
						|
; CHECK-LABEL: @test1(
 | 
						|
; CHECK-NEXT:  entry:
 | 
						|
; CHECK-NEXT:    [[TMP:%.*]] = getelementptr inbounds [[C:%.*]], %C* [[ARG:%.*]], i64 0, i32 0, i32 0
 | 
						|
; CHECK-NEXT:    [[M:%.*]] = load i64*, i64** [[TMP]], align 8
 | 
						|
; CHECK-NEXT:    [[TMP1:%.*]] = getelementptr inbounds [[C]], %C* [[ARG]], i64 1, i32 0, i32 0
 | 
						|
; CHECK-NEXT:    [[N:%.*]] = load i64*, i64** [[TMP1]], align 8
 | 
						|
; CHECK-NEXT:    [[TMP5:%.*]] = icmp ne i64* [[M]], [[N]]
 | 
						|
; CHECK-NEXT:    [[TMP71:%.*]] = icmp eq %C* [[ARG]], null
 | 
						|
; CHECK-NEXT:    [[TMP7:%.*]] = or i1 [[TMP5]], [[TMP71]]
 | 
						|
; CHECK-NEXT:    br i1 [[TMP7]], label [[BB10:%.*]], label [[BB8:%.*]]
 | 
						|
; CHECK:       bb:
 | 
						|
; CHECK-NEXT:    ret void
 | 
						|
; CHECK:       bb8:
 | 
						|
; CHECK-NEXT:    [[TMP9:%.*]] = getelementptr inbounds [[C]], %C* [[ARG]], i64 0, i32 0
 | 
						|
; CHECK-NEXT:    tail call void @bar(%struct.S* [[TMP9]])
 | 
						|
; CHECK-NEXT:    br label [[BB:%.*]]
 | 
						|
; CHECK:       bb10:
 | 
						|
; CHECK-NEXT:    [[TMP2:%.*]] = getelementptr inbounds i64, i64* [[M]], i64 9
 | 
						|
; CHECK-NEXT:    [[TMP3:%.*]] = bitcast i64* [[TMP2]] to i64 (%C*)**
 | 
						|
; CHECK-NEXT:    [[TMP4:%.*]] = load i64 (%C*)*, i64 (%C*)** [[TMP3]], align 8
 | 
						|
; CHECK-NEXT:    [[TMP11:%.*]] = tail call i64 [[TMP4]](%C* [[ARG]])
 | 
						|
; CHECK-NEXT:    br label [[BB]]
 | 
						|
;
 | 
						|
entry:
 | 
						|
  %tmp = getelementptr inbounds %C, %C* %arg, i64 0, i32 0, i32 0
 | 
						|
  %m = load i64*, i64** %tmp, align 8
 | 
						|
  %tmp1 = getelementptr inbounds %C, %C* %arg, i64 1, i32 0, i32 0
 | 
						|
  %n = load i64*, i64** %tmp1, align 8
 | 
						|
  %tmp2 = getelementptr inbounds i64, i64* %m, i64 9
 | 
						|
  %tmp3 = bitcast i64* %tmp2 to i64 (%C*)**
 | 
						|
  %tmp4 = load i64 (%C*)*, i64 (%C*)** %tmp3, align 8
 | 
						|
  %tmp5 = icmp eq i64* %m, %n
 | 
						|
  %tmp6 = select i1 %tmp5, %C* %arg, %C* null
 | 
						|
  %tmp7 = icmp eq %C* %tmp6, null
 | 
						|
  br i1 %tmp7, label %bb10, label %bb8
 | 
						|
 | 
						|
bb:                                               ; preds = %bb10, %bb8
 | 
						|
  ret void
 | 
						|
 | 
						|
bb8:                                              ; preds = %entry
 | 
						|
  %tmp9 = getelementptr inbounds %C, %C* %tmp6, i64 0, i32 0
 | 
						|
  tail call void @bar(%struct.S* %tmp9)
 | 
						|
  br label %bb
 | 
						|
 | 
						|
bb10:                                             ; preds = %entry
 | 
						|
  %tmp11 = tail call i64 %tmp4(%C* %arg)
 | 
						|
  br label %bb
 | 
						|
}
 | 
						|
 | 
						|
define void @test2(%C* %arg) {
 | 
						|
; CHECK-LABEL: @test2(
 | 
						|
; CHECK-NEXT:  entry:
 | 
						|
; CHECK-NEXT:    [[TMP:%.*]] = getelementptr inbounds [[C:%.*]], %C* [[ARG:%.*]], i64 0, i32 0, i32 0
 | 
						|
; CHECK-NEXT:    [[M:%.*]] = load i64*, i64** [[TMP]], align 8
 | 
						|
; CHECK-NEXT:    [[TMP1:%.*]] = getelementptr inbounds [[C]], %C* [[ARG]], i64 1, i32 0, i32 0
 | 
						|
; CHECK-NEXT:    [[N:%.*]] = load i64*, i64** [[TMP1]], align 8
 | 
						|
; CHECK-NEXT:    [[TMP5:%.*]] = icmp eq i64* [[M]], [[N]]
 | 
						|
; CHECK-NEXT:    [[TMP71:%.*]] = icmp eq %C* [[ARG]], null
 | 
						|
; CHECK-NEXT:    [[TMP7:%.*]] = or i1 [[TMP5]], [[TMP71]]
 | 
						|
; CHECK-NEXT:    br i1 [[TMP7]], label [[BB10:%.*]], label [[BB8:%.*]]
 | 
						|
; CHECK:       bb:
 | 
						|
; CHECK-NEXT:    ret void
 | 
						|
; CHECK:       bb8:
 | 
						|
; CHECK-NEXT:    [[TMP9:%.*]] = getelementptr inbounds [[C]], %C* [[ARG]], i64 0, i32 0
 | 
						|
; CHECK-NEXT:    tail call void @bar(%struct.S* [[TMP9]])
 | 
						|
; CHECK-NEXT:    br label [[BB:%.*]]
 | 
						|
; CHECK:       bb10:
 | 
						|
; CHECK-NEXT:    [[TMP2:%.*]] = getelementptr inbounds i64, i64* [[M]], i64 9
 | 
						|
; CHECK-NEXT:    [[TMP3:%.*]] = bitcast i64* [[TMP2]] to i64 (%C*)**
 | 
						|
; CHECK-NEXT:    [[TMP4:%.*]] = load i64 (%C*)*, i64 (%C*)** [[TMP3]], align 8
 | 
						|
; CHECK-NEXT:    [[TMP11:%.*]] = tail call i64 [[TMP4]](%C* [[ARG]])
 | 
						|
; CHECK-NEXT:    br label [[BB]]
 | 
						|
;
 | 
						|
entry:
 | 
						|
  %tmp = getelementptr inbounds %C, %C* %arg, i64 0, i32 0, i32 0
 | 
						|
  %m = load i64*, i64** %tmp, align 8
 | 
						|
  %tmp1 = getelementptr inbounds %C, %C* %arg, i64 1, i32 0, i32 0
 | 
						|
  %n = load i64*, i64** %tmp1, align 8
 | 
						|
  %tmp2 = getelementptr inbounds i64, i64* %m, i64 9
 | 
						|
  %tmp3 = bitcast i64* %tmp2 to i64 (%C*)**
 | 
						|
  %tmp4 = load i64 (%C*)*, i64 (%C*)** %tmp3, align 8
 | 
						|
  %tmp5 = icmp eq i64* %m, %n
 | 
						|
  %tmp6 = select i1 %tmp5, %C* null, %C* %arg
 | 
						|
  %tmp7 = icmp eq %C* %tmp6, null
 | 
						|
  br i1 %tmp7, label %bb10, label %bb8
 | 
						|
 | 
						|
bb:                                               ; preds = %bb10, %bb8
 | 
						|
  ret void
 | 
						|
 | 
						|
bb8:                                              ; preds = %entry
 | 
						|
  %tmp9 = getelementptr inbounds %C, %C* %tmp6, i64 0, i32 0
 | 
						|
  tail call void @bar(%struct.S* %tmp9)
 | 
						|
  br label %bb
 | 
						|
 | 
						|
bb10:                                             ; preds = %entry
 | 
						|
  %tmp11 = tail call i64 %tmp4(%C* %arg)
 | 
						|
  br label %bb
 | 
						|
}
 | 
						|
 | 
						|
define void @test3(%C* %arg) {
 | 
						|
; CHECK-LABEL: @test3(
 | 
						|
; CHECK-NEXT:  entry:
 | 
						|
; CHECK-NEXT:    [[TMP:%.*]] = getelementptr inbounds [[C:%.*]], %C* [[ARG:%.*]], i64 0, i32 0, i32 0
 | 
						|
; CHECK-NEXT:    [[M:%.*]] = load i64*, i64** [[TMP]], align 8
 | 
						|
; CHECK-NEXT:    [[TMP1:%.*]] = getelementptr inbounds [[C]], %C* [[ARG]], i64 1, i32 0, i32 0
 | 
						|
; CHECK-NEXT:    [[N:%.*]] = load i64*, i64** [[TMP1]], align 8
 | 
						|
; CHECK-NEXT:    [[TMP5:%.*]] = icmp ne i64* [[M]], [[N]]
 | 
						|
; CHECK-NEXT:    [[TMP71:%.*]] = icmp eq %C* [[ARG]], null
 | 
						|
; CHECK-NEXT:    [[TMP7:%.*]] = or i1 [[TMP5]], [[TMP71]]
 | 
						|
; CHECK-NEXT:    br i1 [[TMP7]], label [[BB10:%.*]], label [[BB8:%.*]]
 | 
						|
; CHECK:       bb:
 | 
						|
; CHECK-NEXT:    ret void
 | 
						|
; CHECK:       bb8:
 | 
						|
; CHECK-NEXT:    [[TMP9:%.*]] = getelementptr inbounds [[C]], %C* [[ARG]], i64 0, i32 0
 | 
						|
; CHECK-NEXT:    tail call void @bar(%struct.S* [[TMP9]])
 | 
						|
; CHECK-NEXT:    br label [[BB:%.*]]
 | 
						|
; CHECK:       bb10:
 | 
						|
; CHECK-NEXT:    [[TMP2:%.*]] = getelementptr inbounds i64, i64* [[M]], i64 9
 | 
						|
; CHECK-NEXT:    [[TMP3:%.*]] = bitcast i64* [[TMP2]] to i64 (%C*)**
 | 
						|
; CHECK-NEXT:    [[TMP4:%.*]] = load i64 (%C*)*, i64 (%C*)** [[TMP3]], align 8
 | 
						|
; CHECK-NEXT:    [[TMP11:%.*]] = tail call i64 [[TMP4]](%C* [[ARG]])
 | 
						|
; CHECK-NEXT:    br label [[BB]]
 | 
						|
;
 | 
						|
entry:
 | 
						|
  %tmp = getelementptr inbounds %C, %C* %arg, i64 0, i32 0, i32 0
 | 
						|
  %m = load i64*, i64** %tmp, align 8
 | 
						|
  %tmp1 = getelementptr inbounds %C, %C* %arg, i64 1, i32 0, i32 0
 | 
						|
  %n = load i64*, i64** %tmp1, align 8
 | 
						|
  %tmp2 = getelementptr inbounds i64, i64* %m, i64 9
 | 
						|
  %tmp3 = bitcast i64* %tmp2 to i64 (%C*)**
 | 
						|
  %tmp4 = load i64 (%C*)*, i64 (%C*)** %tmp3, align 8
 | 
						|
  %tmp5 = icmp eq i64* %m, %n
 | 
						|
  %tmp6 = select i1 %tmp5, %C* %arg, %C* null
 | 
						|
  %tmp7 = icmp ne %C* %tmp6, null
 | 
						|
  br i1 %tmp7, label %bb8, label %bb10
 | 
						|
 | 
						|
bb:                                               ; preds = %bb10, %bb8
 | 
						|
  ret void
 | 
						|
 | 
						|
bb8:                                              ; preds = %entry
 | 
						|
  %tmp9 = getelementptr inbounds %C, %C* %tmp6, i64 0, i32 0
 | 
						|
  tail call void @bar(%struct.S* %tmp9)
 | 
						|
  br label %bb
 | 
						|
 | 
						|
bb10:                                             ; preds = %entry
 | 
						|
  %tmp11 = tail call i64 %tmp4(%C* %arg)
 | 
						|
  br label %bb
 | 
						|
}
 | 
						|
 | 
						|
define void @test4(%C* %arg) {
 | 
						|
; CHECK-LABEL: @test4(
 | 
						|
; CHECK-NEXT:  entry:
 | 
						|
; CHECK-NEXT:    [[TMP:%.*]] = getelementptr inbounds [[C:%.*]], %C* [[ARG:%.*]], i64 0, i32 0, i32 0
 | 
						|
; CHECK-NEXT:    [[M:%.*]] = load i64*, i64** [[TMP]], align 8
 | 
						|
; CHECK-NEXT:    [[TMP1:%.*]] = getelementptr inbounds [[C]], %C* [[ARG]], i64 1, i32 0, i32 0
 | 
						|
; CHECK-NEXT:    [[N:%.*]] = load i64*, i64** [[TMP1]], align 8
 | 
						|
; CHECK-NEXT:    [[TMP5:%.*]] = icmp eq i64* [[M]], [[N]]
 | 
						|
; CHECK-NEXT:    [[TMP71:%.*]] = icmp eq %C* [[ARG]], null
 | 
						|
; CHECK-NEXT:    [[TMP7:%.*]] = or i1 [[TMP5]], [[TMP71]]
 | 
						|
; CHECK-NEXT:    br i1 [[TMP7]], label [[BB10:%.*]], label [[BB8:%.*]]
 | 
						|
; CHECK:       bb:
 | 
						|
; CHECK-NEXT:    ret void
 | 
						|
; CHECK:       bb8:
 | 
						|
; CHECK-NEXT:    [[TMP9:%.*]] = getelementptr inbounds [[C]], %C* [[ARG]], i64 0, i32 0
 | 
						|
; CHECK-NEXT:    tail call void @bar(%struct.S* [[TMP9]])
 | 
						|
; CHECK-NEXT:    br label [[BB:%.*]]
 | 
						|
; CHECK:       bb10:
 | 
						|
; CHECK-NEXT:    [[TMP2:%.*]] = getelementptr inbounds i64, i64* [[M]], i64 9
 | 
						|
; CHECK-NEXT:    [[TMP3:%.*]] = bitcast i64* [[TMP2]] to i64 (%C*)**
 | 
						|
; CHECK-NEXT:    [[TMP4:%.*]] = load i64 (%C*)*, i64 (%C*)** [[TMP3]], align 8
 | 
						|
; CHECK-NEXT:    [[TMP11:%.*]] = tail call i64 [[TMP4]](%C* [[ARG]])
 | 
						|
; CHECK-NEXT:    br label [[BB]]
 | 
						|
;
 | 
						|
entry:
 | 
						|
  %tmp = getelementptr inbounds %C, %C* %arg, i64 0, i32 0, i32 0
 | 
						|
  %m = load i64*, i64** %tmp, align 8
 | 
						|
  %tmp1 = getelementptr inbounds %C, %C* %arg, i64 1, i32 0, i32 0
 | 
						|
  %n = load i64*, i64** %tmp1, align 8
 | 
						|
  %tmp2 = getelementptr inbounds i64, i64* %m, i64 9
 | 
						|
  %tmp3 = bitcast i64* %tmp2 to i64 (%C*)**
 | 
						|
  %tmp4 = load i64 (%C*)*, i64 (%C*)** %tmp3, align 8
 | 
						|
  %tmp5 = icmp eq i64* %m, %n
 | 
						|
  %tmp6 = select i1 %tmp5, %C* null, %C* %arg
 | 
						|
  %tmp7 = icmp ne %C* %tmp6, null
 | 
						|
  br i1 %tmp7, label %bb8, label %bb10
 | 
						|
 | 
						|
bb:                                               ; preds = %bb10, %bb8
 | 
						|
  ret void
 | 
						|
 | 
						|
bb8:                                              ; preds = %entry
 | 
						|
  %tmp9 = getelementptr inbounds %C, %C* %tmp6, i64 0, i32 0
 | 
						|
  tail call void @bar(%struct.S* %tmp9)
 | 
						|
  br label %bb
 | 
						|
 | 
						|
bb10:                                             ; preds = %entry
 | 
						|
  %tmp11 = tail call i64 %tmp4(%C* %arg)
 | 
						|
  br label %bb
 | 
						|
}
 | 
						|
 | 
						|
define void @test5(%C* %arg, i1 %arg1) {
 | 
						|
; CHECK-LABEL: @test5(
 | 
						|
; CHECK-NEXT:  entry:
 | 
						|
; CHECK-NEXT:    [[TMP21:%.*]] = icmp eq %C* [[ARG:%.*]], null
 | 
						|
; CHECK-NEXT:    [[TMP2:%.*]] = or i1 [[TMP21]], [[ARG1:%.*]]
 | 
						|
; CHECK-NEXT:    br i1 [[TMP2]], label [[BB5:%.*]], label [[BB3:%.*]]
 | 
						|
; CHECK:       bb:
 | 
						|
; CHECK-NEXT:    ret void
 | 
						|
; CHECK:       bb3:
 | 
						|
; CHECK-NEXT:    [[TMP4:%.*]] = getelementptr inbounds [[C:%.*]], %C* [[ARG]], i64 0, i32 0
 | 
						|
; CHECK-NEXT:    tail call void @bar(%struct.S* [[TMP4]])
 | 
						|
; CHECK-NEXT:    br label [[BB:%.*]]
 | 
						|
; CHECK:       bb5:
 | 
						|
; CHECK-NEXT:    tail call void @foobar()
 | 
						|
; CHECK-NEXT:    br label [[BB]]
 | 
						|
;
 | 
						|
entry:
 | 
						|
  %tmp = select i1 %arg1, %C* null, %C* %arg
 | 
						|
  %tmp2 = icmp ne %C* %tmp, null
 | 
						|
  br i1 %tmp2, label %bb3, label %bb5
 | 
						|
 | 
						|
bb:                                               ; preds = %bb5, %bb3
 | 
						|
  ret void
 | 
						|
 | 
						|
bb3:                                              ; preds = %entry
 | 
						|
  %tmp4 = getelementptr inbounds %C, %C* %tmp, i64 0, i32 0
 | 
						|
  tail call void @bar(%struct.S* %tmp4)
 | 
						|
  br label %bb
 | 
						|
 | 
						|
bb5:                                              ; preds = %entry
 | 
						|
  tail call void @foobar()
 | 
						|
  br label %bb
 | 
						|
}
 | 
						|
 | 
						|
; Negative test. Must not trigger the select-cmp-br combine because the result
 | 
						|
; of the select is used in both flows following the br (the special case where
 | 
						|
; the conditional branch has the same target for both flows).
 | 
						|
define i32 @test6(i32 %arg, i1 %arg1) {
 | 
						|
; CHECK-LABEL: @test6(
 | 
						|
; CHECK-NEXT:  entry:
 | 
						|
; CHECK-NEXT:    br i1 false, label [[BB:%.*]], label [[BB]]
 | 
						|
; CHECK:       bb:
 | 
						|
; CHECK-NEXT:    [[TMP:%.*]] = select i1 [[ARG1:%.*]], i32 [[ARG:%.*]], i32 0
 | 
						|
; CHECK-NEXT:    ret i32 [[TMP]]
 | 
						|
;
 | 
						|
entry:
 | 
						|
  %tmp = select i1 %arg1, i32 %arg, i32 0
 | 
						|
  %tmp2 = icmp eq i32 %tmp, 0
 | 
						|
  br i1 %tmp2, label %bb, label %bb
 | 
						|
 | 
						|
bb:                                               ; preds = %entry, %entry
 | 
						|
  ret i32 %tmp
 | 
						|
}
 |