diff --git a/llvm/test/Transforms/ConstraintElimination/dom.ll b/llvm/test/Transforms/ConstraintElimination/dom.ll new file mode 100644 index 00000000000000..a6b8629bed78ae --- /dev/null +++ b/llvm/test/Transforms/ConstraintElimination/dom.ll @@ -0,0 +1,136 @@ +; NOTE: Assertions have been autogenerated by utils/update_test_checks.py +; RUN: opt -S %s | FileCheck %s + +; Test cases where both the true and false successors reach the same block, +; dominated by one of them. + +declare void @use(i1) + +define i32 @test1(i32 %x) { +; CHECK-LABEL: @test1( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp ule i32 [[X:%.*]], 10 +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[BB2:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[C_2:%.*]] = icmp ule i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[C_2]]) +; CHECK-NEXT: br label [[BB2]] +; CHECK: bb2: +; CHECK-NEXT: [[C_3:%.*]] = icmp ugt i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[C_3]]) +; CHECK-NEXT: ret i32 20 +; +entry: + %c.1 = icmp ule i32 %x, 10 + br i1 %c.1, label %bb1, label %bb2 + +bb1: + %c.2 = icmp ule i32 %x, 10 + call void @use(i1 %c.2) + br label %bb2 + +bb2: + %c.3 = icmp ugt i32 %x, 10 + call void @use(i1 %c.3) + ret i32 20 +} + + +define i32 @test2(i32 %x) { +; CHECK-LABEL: @test2( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp ule i32 [[X:%.*]], 10 +; CHECK-NEXT: br i1 [[C_1]], label [[BB2:%.*]], label [[BB1:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[C_2:%.*]] = icmp ugt i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[C_2]]) +; CHECK-NEXT: ret i32 20 +; CHECK: bb2: +; CHECK-NEXT: [[C_3:%.*]] = icmp ule i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[C_3]]) +; CHECK-NEXT: br label [[BB1]] +; +entry: + %c.1 = icmp ule i32 %x, 10 + br i1 %c.1, label %bb2, label %bb1 + +bb1: + %c.2 = icmp ugt i32 %x, 10 + call void @use(i1 %c.2) + ret i32 20 + +bb2: + %c.3 = icmp ule i32 %x, 10 + call void @use(i1 %c.3) + br label %bb1 +} + + +; Test cases where the true/false successors are not domianted by the conditional branching block. +define i32 @test3(i32 %x, i1 %c) { +; CHECK-LABEL: @test3( +; CHECK-NEXT: entry: +; CHECK-NEXT: br i1 [[C:%.*]], label [[BB_COND:%.*]], label [[BB1:%.*]] +; CHECK: bb.cond: +; CHECK-NEXT: [[C_1:%.*]] = icmp ule i32 [[X:%.*]], 10 +; CHECK-NEXT: br i1 [[C_1]], label [[BB1]], label [[BB2:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[C_2:%.*]] = icmp ule i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[C_2]]) +; CHECK-NEXT: ret i32 10 +; CHECK: bb2: +; CHECK-NEXT: [[C_3:%.*]] = icmp ugt i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[C_3]]) +; CHECK-NEXT: ret i32 20 +; +entry: + br i1 %c, label %bb.cond, label %bb1 + +bb.cond: + %c.1 = icmp ule i32 %x, 10 + br i1 %c.1, label %bb1, label %bb2 + +bb1: + %c.2 = icmp ule i32 %x, 10 + call void @use(i1 %c.2) + ret i32 10 + +bb2: + %c.3 = icmp ugt i32 %x, 10 + call void @use(i1 %c.3) + ret i32 20 +} + +define i32 @test4(i32 %x, i1 %c) { +; CHECK-LABEL: @test4( +; CHECK-NEXT: entry: +; CHECK-NEXT: br i1 [[C:%.*]], label [[BB_COND:%.*]], label [[BB2:%.*]] +; CHECK: bb.cond: +; CHECK-NEXT: [[C_1:%.*]] = icmp ule i32 [[X:%.*]], 10 +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[BB2]] +; CHECK: bb1: +; CHECK-NEXT: [[C_2:%.*]] = icmp ule i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[C_2]]) +; CHECK-NEXT: ret i32 10 +; CHECK: bb2: +; CHECK-NEXT: [[C_3:%.*]] = icmp ugt i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[C_3]]) +; CHECK-NEXT: ret i32 20 +; +entry: + br i1 %c, label %bb.cond, label %bb2 + +bb.cond: + %c.1 = icmp ule i32 %x, 10 + br i1 %c.1, label %bb1, label %bb2 + +bb1: + %c.2 = icmp ule i32 %x, 10 + call void @use(i1 %c.2) + ret i32 10 + +bb2: + %c.3 = icmp ugt i32 %x, 10 + call void @use(i1 %c.3) + ret i32 20 +} diff --git a/llvm/test/Transforms/ConstraintElimination/geps.2d.ll b/llvm/test/Transforms/ConstraintElimination/geps.2d.ll new file mode 100644 index 00000000000000..bb245144044140 --- /dev/null +++ b/llvm/test/Transforms/ConstraintElimination/geps.2d.ll @@ -0,0 +1,134 @@ +; NOTE: Assertions have been autogenerated by utils/update_test_checks.py +; RUN: opt -S %s | FileCheck %s + +define void @test.not.uge.ult([10 x i8]* %start, i8* %low, i8* %high) { +; CHECK-LABEL: @test.not.uge.ult( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[ADD_PTR_I:%.*]] = getelementptr inbounds [10 x i8], [10 x i8]* [[START:%.*]], i64 1, i64 3 +; CHECK-NEXT: [[C_1:%.*]] = icmp uge i8* [[ADD_PTR_I]], [[HIGH:%.*]] +; CHECK-NEXT: br i1 [[C_1]], label [[IF_THEN:%.*]], label [[IF_END:%.*]] +; CHECK: if.then: +; CHECK-NEXT: ret void +; CHECK: if.end: +; CHECK-NEXT: [[START_0:%.*]] = getelementptr inbounds [10 x i8], [10 x i8]* [[START]], i64 10, i64 0 +; CHECK-NEXT: [[C_0:%.*]] = icmp ult i8* [[START_0]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[C_0]]) +; CHECK-NEXT: ret void +; +entry: + %add.ptr.i = getelementptr inbounds [10 x i8], [10 x i8]* %start, i64 1, i64 3 + %c.1 = icmp uge i8* %add.ptr.i, %high + br i1 %c.1, label %if.then, label %if.end + +if.then: ; preds = %entry + ret void + +if.end: ; preds = %entry + %start.0 = getelementptr inbounds [10 x i8], [10 x i8]* %start, i64 10, i64 0 + %c.0 = icmp ult i8* %start.0, %high + call void @use(i1 %c.0) + ret void +} + +define void @test.not.uge.ule([10 x i8]* %start, i8* %low, i8* %high) { +; CHECK-LABEL: @test.not.uge.ule( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[ADD_PTR_I:%.*]] = getelementptr inbounds [10 x i8], [10 x i8]* [[START:%.*]], i64 1, i64 3 +; CHECK-NEXT: [[C:%.*]] = icmp uge i8* [[ADD_PTR_I]], [[HIGH:%.*]] +; CHECK-NEXT: br i1 [[C]], label [[IF_THEN:%.*]], label [[IF_END:%.*]] +; CHECK: if.then: +; CHECK-NEXT: ret void +; CHECK: if.end: +; CHECK-NEXT: [[START_0:%.*]] = getelementptr inbounds [10 x i8], [10 x i8]* [[START]], i64 10, i64 0 +; CHECK-NEXT: [[C_0:%.*]] = icmp ule i8* [[START_0]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[C_0]]) +; CHECK-NEXT: [[START_1:%.*]] = getelementptr inbounds [10 x i8], [10 x i8]* [[START]], i64 2, i64 1 +; CHECK-NEXT: [[C_1:%.*]] = icmp ule i8* [[START_1]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[C_1]]) +; CHECK-NEXT: ret void +; +entry: + %add.ptr.i = getelementptr inbounds [10 x i8], [10 x i8]* %start, i64 1, i64 3 + %c = icmp uge i8* %add.ptr.i, %high + br i1 %c, label %if.then, label %if.end + +if.then: ; preds = %entry + ret void + +if.end: ; preds = %entry + %start.0 = getelementptr inbounds [10 x i8], [10 x i8]* %start, i64 10, i64 0 + %c.0 = icmp ule i8* %start.0, %high + call void @use(i1 %c.0) + %start.1 = getelementptr inbounds [10 x i8], [10 x i8]* %start, i64 2, i64 1 + %c.1 = icmp ule i8* %start.1, %high + call void @use(i1 %c.1) + ret void +} + +define void @test.not.uge.ugt([10 x i8]* %start, i8* %low, i8* %high) { +; CHECK-LABEL: @test.not.uge.ugt( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[ADD_PTR_I:%.*]] = getelementptr inbounds [10 x i8], [10 x i8]* [[START:%.*]], i64 1, i64 3 +; CHECK-NEXT: [[C:%.*]] = icmp uge i8* [[ADD_PTR_I]], [[HIGH:%.*]] +; CHECK-NEXT: br i1 [[C]], label [[IF_THEN:%.*]], label [[IF_END:%.*]] +; CHECK: if.then: +; CHECK-NEXT: ret void +; CHECK: if.end: +; CHECK-NEXT: [[START_0:%.*]] = getelementptr inbounds [10 x i8], [10 x i8]* [[START]], i64 3, i64 0 +; CHECK-NEXT: [[C_0:%.*]] = icmp ugt i8* [[START_0]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[C_0]]) +; CHECK-NEXT: [[START_1:%.*]] = getelementptr inbounds [10 x i8], [10 x i8]* [[START]], i64 3, i64 1 +; CHECK-NEXT: [[C_1:%.*]] = icmp ugt i8* [[START_1]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[C_1]]) +; CHECK-NEXT: ret void +; +entry: + %add.ptr.i = getelementptr inbounds [10 x i8], [10 x i8]* %start, i64 1, i64 3 + %c = icmp uge i8* %add.ptr.i, %high + br i1 %c, label %if.then, label %if.end + +if.then: ; preds = %entry + ret void + +if.end: ; preds = %entry + %start.0 = getelementptr inbounds [10 x i8], [10 x i8]* %start, i64 3, i64 0 + %c.0 = icmp ugt i8* %start.0, %high + call void @use(i1 %c.0) + + %start.1 = getelementptr inbounds [10 x i8], [10 x i8]* %start, i64 3, i64 1 + %c.1 = icmp ugt i8* %start.1, %high + call void @use(i1 %c.1) + ret void +} + +define void @test.not.uge.uge([10 x i8]* %start, i8* %low, i8* %high) { +; CHECK-LABEL: @test.not.uge.uge( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[ADD_PTR_I:%.*]] = getelementptr inbounds [10 x i8], [10 x i8]* [[START:%.*]], i64 1, i64 3 +; CHECK-NEXT: [[C_1:%.*]] = icmp uge i8* [[ADD_PTR_I]], [[HIGH:%.*]] +; CHECK-NEXT: br i1 [[C_1]], label [[IF_THEN:%.*]], label [[IF_END:%.*]] +; CHECK: if.then: +; CHECK-NEXT: ret void +; CHECK: if.end: +; CHECK-NEXT: [[START_0:%.*]] = getelementptr inbounds [10 x i8], [10 x i8]* [[START]], i64 3, i64 0 +; CHECK-NEXT: [[C_0:%.*]] = icmp uge i8* [[START_0]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[C_0]]) +; CHECK-NEXT: ret void +; +entry: + %add.ptr.i = getelementptr inbounds [10 x i8], [10 x i8]* %start, i64 1, i64 3 + %c.1 = icmp uge i8* %add.ptr.i, %high + br i1 %c.1, label %if.then, label %if.end + +if.then: ; preds = %entry + ret void + +if.end: ; preds = %entry + %start.0 = getelementptr inbounds [10 x i8], [10 x i8]* %start, i64 3, i64 0 + %c.0 = icmp uge i8* %start.0, %high + call void @use(i1 %c.0) + + ret void +} + +declare void @use(i1) diff --git a/llvm/test/Transforms/ConstraintElimination/geps.ll b/llvm/test/Transforms/ConstraintElimination/geps.ll new file mode 100644 index 00000000000000..0e36ebf07f0f43 --- /dev/null +++ b/llvm/test/Transforms/ConstraintElimination/geps.ll @@ -0,0 +1,332 @@ +; NOTE: Assertions have been autogenerated by utils/update_test_checks.py +; RUN: opt -S %s | FileCheck %s + +define i32 @test.ult(i32* readonly %src, i32* readnone %min, i32* readnone %max) { +; CHECK-LABEL: @test.ult( +; CHECK-NEXT: check.0.min: +; CHECK-NEXT: [[C_MIN_0:%.*]] = icmp ult i32* [[SRC:%.*]], [[MIN:%.*]] +; CHECK-NEXT: br i1 [[C_MIN_0]], label [[TRAP:%.*]], label [[CHECK_0_MAX:%.*]] +; CHECK: trap: +; CHECK-NEXT: ret i32 10 +; CHECK: check.0.max: +; CHECK-NEXT: [[C_MAX_0:%.*]] = icmp ult i32* [[SRC]], [[MAX:%.*]] +; CHECK-NEXT: br i1 [[C_MAX_0]], label [[CHECK_3_MIN:%.*]], label [[TRAP]] +; CHECK: check.3.min: +; CHECK-NEXT: [[L0:%.*]] = load i32, i32* [[SRC]], align 4 +; CHECK-NEXT: [[ADD_PTR_I36:%.*]] = getelementptr inbounds i32, i32* [[SRC]], i64 3 +; CHECK-NEXT: [[C_3_MIN:%.*]] = icmp ult i32* [[ADD_PTR_I36]], [[MIN]] +; CHECK-NEXT: br i1 [[C_3_MIN]], label [[TRAP]], label [[CHECK_3_MAX:%.*]] +; CHECK: check.3.max: +; CHECK-NEXT: [[C_3_MAX:%.*]] = icmp ult i32* [[ADD_PTR_I36]], [[MAX]] +; CHECK-NEXT: br i1 [[C_3_MAX]], label [[CHECK_1_MIN:%.*]], label [[TRAP]] +; CHECK: check.1.min: +; CHECK-NEXT: [[L1:%.*]] = load i32, i32* [[ADD_PTR_I36]], align 4 +; CHECK-NEXT: [[ADD_PTR_I29:%.*]] = getelementptr inbounds i32, i32* [[SRC]], i64 1 +; CHECK-NEXT: [[C_1_MIN:%.*]] = icmp ult i32* [[ADD_PTR_I29]], [[MIN]] +; CHECK-NEXT: br i1 [[C_1_MIN]], label [[TRAP]], label [[CHECK_1_MAX:%.*]] +; CHECK: check.1.max: +; CHECK-NEXT: [[C_1_MAX:%.*]] = icmp ult i32* [[ADD_PTR_I29]], [[MAX]] +; CHECK-NEXT: br i1 [[C_1_MAX]], label [[CHECK_2_MIN:%.*]], label [[TRAP]] +; CHECK: check.2.min: +; CHECK-NEXT: [[L2:%.*]] = load i32, i32* [[ADD_PTR_I29]], align 4 +; CHECK-NEXT: [[ADD_PTR_I:%.*]] = getelementptr inbounds i32, i32* [[SRC]], i64 2 +; CHECK-NEXT: [[C_2_MIN:%.*]] = icmp ult i32* [[ADD_PTR_I]], [[MIN]] +; CHECK-NEXT: br i1 [[C_2_MIN]], label [[TRAP]], label [[CHECK_2_MAX:%.*]] +; CHECK: check.2.max: +; CHECK-NEXT: [[C_2_MAX:%.*]] = icmp ult i32* [[ADD_PTR_I]], [[MAX]] +; CHECK-NEXT: br i1 [[C_2_MAX]], label [[EXIT:%.*]], label [[TRAP]] +; CHECK: exit: +; CHECK-NEXT: [[L3:%.*]] = load i32, i32* [[ADD_PTR_I]], align 4 +; CHECK-NEXT: [[ADD:%.*]] = add nsw i32 [[L1]], [[L0]] +; CHECK-NEXT: [[ADD8:%.*]] = add nsw i32 [[ADD]], [[L2]] +; CHECK-NEXT: [[ADD9:%.*]] = add nsw i32 [[ADD8]], [[L3]] +; CHECK-NEXT: ret i32 [[ADD9]] +; +check.0.min: + %c.min.0 = icmp ult i32* %src, %min + br i1 %c.min.0, label %trap, label %check.0.max + +trap: ; preds = %check.2.max, %check.2.min, %check.1.max, %check.1.min, %check.3.max, %check.3.min, %check.0.max, %check.0.min + ret i32 10 + +check.0.max: ; preds = %check.0.min + %c.max.0 = icmp ult i32* %src, %max + br i1 %c.max.0, label %check.3.min, label %trap + +check.3.min: ; preds = %check.0.max + %l0 = load i32, i32* %src, align 4 + %add.ptr.i36 = getelementptr inbounds i32, i32* %src, i64 3 + %c.3.min = icmp ult i32* %add.ptr.i36, %min + br i1 %c.3.min, label %trap, label %check.3.max + +check.3.max: ; preds = %check.3.min + %c.3.max = icmp ult i32* %add.ptr.i36, %max + br i1 %c.3.max, label %check.1.min, label %trap + +check.1.min: ; preds = %check.3.max + %l1 = load i32, i32* %add.ptr.i36, align 4 + %add.ptr.i29 = getelementptr inbounds i32, i32* %src, i64 1 + %c.1.min = icmp ult i32* %add.ptr.i29, %min + br i1 %c.1.min, label %trap, label %check.1.max + +check.1.max: ; preds = %check.1.min + %c.1.max = icmp ult i32* %add.ptr.i29, %max + br i1 %c.1.max, label %check.2.min, label %trap + +check.2.min: ; preds = %check.1.max + %l2 = load i32, i32* %add.ptr.i29, align 4 + %add.ptr.i = getelementptr inbounds i32, i32* %src, i64 2 + %c.2.min = icmp ult i32* %add.ptr.i, %min + br i1 %c.2.min, label %trap, label %check.2.max + +check.2.max: ; preds = %check.2.min + %c.2.max = icmp ult i32* %add.ptr.i, %max + br i1 %c.2.max, label %exit, label %trap + +exit: ; preds = %check.2.max + %l3 = load i32, i32* %add.ptr.i, align 4 + %add = add nsw i32 %l1, %l0 + %add8 = add nsw i32 %add, %l2 + %add9 = add nsw i32 %add8, %l3 + ret i32 %add9 +} + +define void @test.not.uge.ult(i8* %start, i8* %low, i8* %high) { +; CHECK-LABEL: @test.not.uge.ult( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[ADD_PTR_I:%.*]] = getelementptr inbounds i8, i8* [[START:%.*]], i64 3 +; CHECK-NEXT: [[C_1:%.*]] = icmp uge i8* [[ADD_PTR_I]], [[HIGH:%.*]] +; CHECK-NEXT: br i1 [[C_1]], label [[IF_THEN:%.*]], label [[IF_END:%.*]] +; CHECK: if.then: +; CHECK-NEXT: ret void +; CHECK: if.end: +; CHECK-NEXT: [[T_0:%.*]] = icmp ult i8* [[START]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[T_0]]) +; CHECK-NEXT: [[START_1:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 1 +; CHECK-NEXT: [[T_1:%.*]] = icmp ult i8* [[START_1]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[T_1]]) +; CHECK-NEXT: [[START_2:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 2 +; CHECK-NEXT: [[T_2:%.*]] = icmp ult i8* [[START_2]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[T_2]]) +; CHECK-NEXT: [[START_3:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 3 +; CHECK-NEXT: [[T_3:%.*]] = icmp ult i8* [[START_3]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[T_3]]) +; CHECK-NEXT: [[START_4:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 4 +; CHECK-NEXT: [[C_4:%.*]] = icmp ult i8* [[START_4]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[C_4]]) +; CHECK-NEXT: ret void +; +entry: + %add.ptr.i = getelementptr inbounds i8, i8* %start, i64 3 + %c.1 = icmp uge i8* %add.ptr.i, %high + br i1 %c.1, label %if.then, label %if.end + +if.then: ; preds = %entry + ret void + +if.end: ; preds = %entry + %t.0 = icmp ult i8* %start, %high + call void @use(i1 %t.0) + %start.1 = getelementptr inbounds i8, i8* %start, i64 1 + %t.1 = icmp ult i8* %start.1, %high + call void @use(i1 %t.1) + %start.2 = getelementptr inbounds i8, i8* %start, i64 2 + %t.2 = icmp ult i8* %start.2, %high + call void @use(i1 %t.2) + %start.3 = getelementptr inbounds i8, i8* %start, i64 3 + %t.3 = icmp ult i8* %start.3, %high + call void @use(i1 %t.3) + %start.4 = getelementptr inbounds i8, i8* %start, i64 4 + %c.4 = icmp ult i8* %start.4, %high + call void @use(i1 %c.4) + ret void +} + +define void @test.not.uge.ule(i8* %start, i8* %low, i8* %high) { +; CHECK-LABEL: @test.not.uge.ule( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[ADD_PTR_I:%.*]] = getelementptr inbounds i8, i8* [[START:%.*]], i64 3 +; CHECK-NEXT: [[C_1:%.*]] = icmp uge i8* [[ADD_PTR_I]], [[HIGH:%.*]] +; CHECK-NEXT: br i1 [[C_1]], label [[IF_THEN:%.*]], label [[IF_END:%.*]] +; CHECK: if.then: +; CHECK-NEXT: ret void +; CHECK: if.end: +; CHECK-NEXT: [[T_0:%.*]] = icmp ule i8* [[START]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[T_0]]) +; CHECK-NEXT: [[START_1:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 1 +; CHECK-NEXT: [[T_1:%.*]] = icmp ule i8* [[START_1]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[T_1]]) +; CHECK-NEXT: [[START_2:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 2 +; CHECK-NEXT: [[T_2:%.*]] = icmp ule i8* [[START_2]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[T_2]]) +; CHECK-NEXT: [[START_3:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 3 +; CHECK-NEXT: [[T_3:%.*]] = icmp ule i8* [[START_3]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[T_3]]) +; CHECK-NEXT: [[START_4:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 4 +; CHECK-NEXT: [[T_4:%.*]] = icmp ule i8* [[START_4]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[T_4]]) +; CHECK-NEXT: [[START_5:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 5 +; CHECK-NEXT: [[C_5:%.*]] = icmp ule i8* [[START_5]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[C_5]]) +; CHECK-NEXT: ret void +; +entry: + %add.ptr.i = getelementptr inbounds i8, i8* %start, i64 3 + %c.1 = icmp uge i8* %add.ptr.i, %high + br i1 %c.1, label %if.then, label %if.end + +if.then: ; preds = %entry + ret void + +if.end: ; preds = %entry + %t.0 = icmp ule i8* %start, %high + call void @use(i1 %t.0) + %start.1 = getelementptr inbounds i8, i8* %start, i64 1 + %t.1 = icmp ule i8* %start.1, %high + call void @use(i1 %t.1) + %start.2 = getelementptr inbounds i8, i8* %start, i64 2 + %t.2 = icmp ule i8* %start.2, %high + call void @use(i1 %t.2) + %start.3 = getelementptr inbounds i8, i8* %start, i64 3 + %t.3 = icmp ule i8* %start.3, %high + call void @use(i1 %t.3) + %start.4 = getelementptr inbounds i8, i8* %start, i64 4 + %t.4 = icmp ule i8* %start.4, %high + call void @use(i1 %t.4) + + %start.5 = getelementptr inbounds i8, i8* %start, i64 5 + %c.5 = icmp ule i8* %start.5, %high + call void @use(i1 %c.5) + + ret void +} + +define void @test.not.uge.ugt(i8* %start, i8* %low, i8* %high) { +; CHECK-LABEL: @test.not.uge.ugt( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[ADD_PTR_I:%.*]] = getelementptr inbounds i8, i8* [[START:%.*]], i64 3 +; CHECK-NEXT: [[C_1:%.*]] = icmp uge i8* [[ADD_PTR_I]], [[HIGH:%.*]] +; CHECK-NEXT: br i1 [[C_1]], label [[IF_THEN:%.*]], label [[IF_END:%.*]] +; CHECK: if.then: +; CHECK-NEXT: ret void +; CHECK: if.end: +; CHECK-NEXT: [[F_0:%.*]] = icmp ugt i8* [[START]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[F_0]]) +; CHECK-NEXT: [[START_1:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 1 +; CHECK-NEXT: [[F_1:%.*]] = icmp ugt i8* [[START_1]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[F_1]]) +; CHECK-NEXT: [[START_2:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 2 +; CHECK-NEXT: [[F_2:%.*]] = icmp ugt i8* [[START_2]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[F_2]]) +; CHECK-NEXT: [[START_3:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 3 +; CHECK-NEXT: [[F_3:%.*]] = icmp ugt i8* [[START_3]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[F_3]]) +; CHECK-NEXT: [[START_4:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 4 +; CHECK-NEXT: [[F_4:%.*]] = icmp ugt i8* [[START_4]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[F_4]]) +; CHECK-NEXT: [[START_5:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 5 +; CHECK-NEXT: [[C_5:%.*]] = icmp ugt i8* [[START_5]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[C_5]]) +; CHECK-NEXT: ret void +; +entry: + %add.ptr.i = getelementptr inbounds i8, i8* %start, i64 3 + %c.1 = icmp uge i8* %add.ptr.i, %high + br i1 %c.1, label %if.then, label %if.end + +if.then: ; preds = %entry + ret void + +if.end: ; preds = %entry + %f.0 = icmp ugt i8* %start, %high + call void @use(i1 %f.0) + + %start.1 = getelementptr inbounds i8, i8* %start, i64 1 + %f.1 = icmp ugt i8* %start.1, %high + call void @use(i1 %f.1) + + %start.2 = getelementptr inbounds i8, i8* %start, i64 2 + %f.2 = icmp ugt i8* %start.2, %high + call void @use(i1 %f.2) + + %start.3 = getelementptr inbounds i8, i8* %start, i64 3 + %f.3 = icmp ugt i8* %start.3, %high + call void @use(i1 %f.3) + + %start.4 = getelementptr inbounds i8, i8* %start, i64 4 + %f.4 = icmp ugt i8* %start.4, %high + call void @use(i1 %f.4) + + %start.5 = getelementptr inbounds i8, i8* %start, i64 5 + %c.5 = icmp ugt i8* %start.5, %high + call void @use(i1 %c.5) + + ret void +} + +define void @test.not.uge.uge(i8* %start, i8* %low, i8* %high) { +; CHECK-LABEL: @test.not.uge.uge( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[ADD_PTR_I:%.*]] = getelementptr inbounds i8, i8* [[START:%.*]], i64 3 +; CHECK-NEXT: [[C_1:%.*]] = icmp uge i8* [[ADD_PTR_I]], [[HIGH:%.*]] +; CHECK-NEXT: br i1 [[C_1]], label [[IF_THEN:%.*]], label [[IF_END:%.*]] +; CHECK: if.then: +; CHECK-NEXT: ret void +; CHECK: if.end: +; CHECK-NEXT: [[F_0:%.*]] = icmp ugt i8* [[START]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[F_0]]) +; CHECK-NEXT: [[START_1:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 1 +; CHECK-NEXT: [[F_1:%.*]] = icmp uge i8* [[START_1]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[F_1]]) +; CHECK-NEXT: [[START_2:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 2 +; CHECK-NEXT: [[F_2:%.*]] = icmp uge i8* [[START_2]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[F_2]]) +; CHECK-NEXT: [[START_3:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 3 +; CHECK-NEXT: [[F_3:%.*]] = icmp uge i8* [[START_3]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[F_3]]) +; CHECK-NEXT: [[START_4:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 4 +; CHECK-NEXT: [[C_4:%.*]] = icmp uge i8* [[START_4]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[C_4]]) +; CHECK-NEXT: [[START_5:%.*]] = getelementptr inbounds i8, i8* [[START]], i64 5 +; CHECK-NEXT: [[C_5:%.*]] = icmp uge i8* [[START_5]], [[HIGH]] +; CHECK-NEXT: call void @use(i1 [[C_5]]) +; CHECK-NEXT: ret void +; +entry: + %add.ptr.i = getelementptr inbounds i8, i8* %start, i64 3 + %c.1 = icmp uge i8* %add.ptr.i, %high + br i1 %c.1, label %if.then, label %if.end + +if.then: ; preds = %entry + ret void + +if.end: ; preds = %entry + %f.0 = icmp ugt i8* %start, %high + call void @use(i1 %f.0) + + %start.1 = getelementptr inbounds i8, i8* %start, i64 1 + %f.1 = icmp uge i8* %start.1, %high + call void @use(i1 %f.1) + + %start.2 = getelementptr inbounds i8, i8* %start, i64 2 + %f.2 = icmp uge i8* %start.2, %high + call void @use(i1 %f.2) + + %start.3 = getelementptr inbounds i8, i8* %start, i64 3 + %f.3 = icmp uge i8* %start.3, %high + call void @use(i1 %f.3) + + %start.4 = getelementptr inbounds i8, i8* %start, i64 4 + %c.4 = icmp uge i8* %start.4, %high + call void @use(i1 %c.4) + + %start.5 = getelementptr inbounds i8, i8* %start, i64 5 + %c.5 = icmp uge i8* %start.5, %high + call void @use(i1 %c.5) + + ret void +} + + +declare void @use(i1) +declare void @llvm.trap() diff --git a/llvm/test/Transforms/ConstraintElimination/i128.ll b/llvm/test/Transforms/ConstraintElimination/i128.ll new file mode 100644 index 00000000000000..6a10ea770dd585 --- /dev/null +++ b/llvm/test/Transforms/ConstraintElimination/i128.ll @@ -0,0 +1,37 @@ +; NOTE: Assertions have been autogenerated by utils/update_test_checks.py +; RUN: opt -S %s | FileCheck %s + +declare void @use(i1) + +define void @test_unsigned_too_large(i128 %x) { +; CHECK-LABEL: @test_unsigned_too_large( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp ule i128 [[X:%.*]], 12345678901234123123123 +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[BB2:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[C_2:%.*]] = icmp ult i128 [[X]], -12345678901234123123123 +; CHECK-NEXT: call void @use(i1 [[C_2]]) +; CHECK-NEXT: [[C_3:%.*]] = icmp uge i128 [[X]], -12345678901234123123123 +; CHECK-NEXT: call void @use(i1 [[C_3]]) +; CHECK-NEXT: [[C_4:%.*]] = icmp uge i128 [[X]], -12345678901234123123123 +; CHECK-NEXT: call void @use(i1 [[C_4]]) +; CHECK-NEXT: ret void +; CHECK: bb2: +; CHECK-NEXT: ret void +; +entry: + %c.1 = icmp ule i128 %x, 12345678901234123123123 + br i1 %c.1, label %bb1, label %bb2 + +bb1: + %c.2 = icmp ult i128 %x, -12345678901234123123123 + call void @use(i1 %c.2) + %c.3 = icmp uge i128 %x, -12345678901234123123123 + call void @use(i1 %c.3) + %c.4 = icmp uge i128 %x, -12345678901234123123123 + call void @use(i1 %c.4) + ret void + +bb2: + ret void +} diff --git a/llvm/test/Transforms/ConstraintElimination/loops.ll b/llvm/test/Transforms/ConstraintElimination/loops.ll new file mode 100644 index 00000000000000..be25308c46dfe4 --- /dev/null +++ b/llvm/test/Transforms/ConstraintElimination/loops.ll @@ -0,0 +1,47 @@ +; NOTE: Assertions have been autogenerated by utils/update_test_checks.py +; RUN: opt -S %s | FileCheck %s + +; Make sure conditions in loops are not used to simplify themselves. + +define void @loop1(float* %T, float* %x, i32 %points, i32 %trigint) { +; CHECK-LABEL: @loop1( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[IDX_EXT:%.*]] = sext i32 [[POINTS:%.*]] to i64 +; CHECK-NEXT: [[ADD_PTR:%.*]] = getelementptr inbounds float, float* [[X:%.*]], i64 [[IDX_EXT]] +; CHECK-NEXT: [[ADD_PTR1:%.*]] = getelementptr inbounds float, float* [[ADD_PTR]], i64 -8 +; CHECK-NEXT: [[SHR:%.*]] = ashr i32 [[POINTS]], 1 +; CHECK-NEXT: [[IDX_EXT2:%.*]] = sext i32 [[SHR]] to i64 +; CHECK-NEXT: [[ADD_PTR3:%.*]] = getelementptr inbounds float, float* [[X]], i64 [[IDX_EXT2]] +; CHECK-NEXT: [[ADD_PTR4:%.*]] = getelementptr inbounds float, float* [[ADD_PTR3]], i64 -8 +; CHECK-NEXT: br label [[DO_BODY:%.*]] +; CHECK: do.body: +; CHECK-NEXT: [[X2_0:%.*]] = phi float* [ [[ADD_PTR4]], [[ENTRY:%.*]] ], [ [[ADD_PTR106:%.*]], [[DO_BODY]] ] +; CHECK-NEXT: [[X1_0:%.*]] = phi float* [ [[ADD_PTR1]], [[ENTRY]] ], [ [[ADD_PTR105:%.*]], [[DO_BODY]] ] +; CHECK-NEXT: [[ADD_PTR105]] = getelementptr inbounds float, float* [[X1_0]], i64 -8 +; CHECK-NEXT: [[ADD_PTR106]] = getelementptr inbounds float, float* [[X2_0]], i64 -8 +; CHECK-NEXT: [[CMP:%.*]] = icmp uge float* [[ADD_PTR106]], [[X]] +; CHECK-NEXT: br i1 [[CMP]], label [[DO_BODY]], label [[DO_END:%.*]] +; CHECK: do.end: +; CHECK-NEXT: ret void +; +entry: + %idx.ext = sext i32 %points to i64 + %add.ptr = getelementptr inbounds float, float* %x, i64 %idx.ext + %add.ptr1 = getelementptr inbounds float, float* %add.ptr, i64 -8 + %shr = ashr i32 %points, 1 + %idx.ext2 = sext i32 %shr to i64 + %add.ptr3 = getelementptr inbounds float, float* %x, i64 %idx.ext2 + %add.ptr4 = getelementptr inbounds float, float* %add.ptr3, i64 -8 + br label %do.body + +do.body: ; preds = %do.body, %entry + %x2.0 = phi float* [ %add.ptr4, %entry ], [ %add.ptr106, %do.body ] + %x1.0 = phi float* [ %add.ptr1, %entry ], [ %add.ptr105, %do.body ] + %add.ptr105 = getelementptr inbounds float, float* %x1.0, i64 -8 + %add.ptr106 = getelementptr inbounds float, float* %x2.0, i64 -8 + %cmp = icmp uge float* %add.ptr106, %x + br i1 %cmp, label %do.body, label %do.end + +do.end: ; preds = %do.body + ret void +} diff --git a/llvm/test/Transforms/ConstraintElimination/mixed.ll b/llvm/test/Transforms/ConstraintElimination/mixed.ll new file mode 100644 index 00000000000000..e4a264a8f0a0f1 --- /dev/null +++ b/llvm/test/Transforms/ConstraintElimination/mixed.ll @@ -0,0 +1,40 @@ +; NOTE: Assertions have been autogenerated by utils/update_test_checks.py +; RUN: opt -S %s | FileCheck %s + +; Make sure we do not incorrectly add variables to the system. + +define i1 @test(i32* %p1, i32* %p2, i32 %num_rows, i32 %start_row, i1 %c) { +; CHECK-LABEL: @test( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[ADD:%.*]] = add i32 [[NUM_ROWS:%.*]], [[START_ROW:%.*]] +; CHECK-NEXT: [[L3:%.*]] = load i32, i32* [[P1:%.*]], align 4 +; CHECK-NEXT: [[CMP6:%.*]] = icmp ugt i32 [[L3]], [[START_ROW]] +; CHECK-NEXT: br i1 [[C:%.*]], label [[IF_END36:%.*]], label [[IF_END36]] +; CHECK: if.end36: +; CHECK-NEXT: [[L1:%.*]] = load i32, i32* [[P2:%.*]], align 4 +; CHECK-NEXT: [[CMP37:%.*]] = icmp ult i32 [[L1]], [[ADD]] +; CHECK-NEXT: br i1 [[CMP37]], label [[IF_THEN39:%.*]], label [[EXIT:%.*]] +; CHECK: if.then39: +; CHECK-NEXT: [[CMP41:%.*]] = icmp ult i32 [[L1]], [[START_ROW]] +; CHECK-NEXT: ret i1 [[CMP41]] +; CHECK: exit: +; CHECK-NEXT: ret i1 false +; +entry: + %add = add i32 %num_rows, %start_row + %l3 = load i32, i32* %p1, align 4 + %cmp6 = icmp ugt i32 %l3, %start_row + br i1 %c, label %if.end36, label %if.end36 + +if.end36: ; preds = %if.then11 + %l1 = load i32, i32* %p2, align 4 + %cmp37 = icmp ult i32 %l1, %add + br i1 %cmp37, label %if.then39, label %exit + +if.then39: ; preds = %if.end36 + %cmp41 = icmp ult i32 %l1, %start_row + ret i1 %cmp41 + +exit: ; preds = %if.end36 + ret i1 false +} diff --git a/llvm/test/Transforms/ConstraintElimination/uge.ll b/llvm/test/Transforms/ConstraintElimination/uge.ll new file mode 100644 index 00000000000000..ca91733d2af98f --- /dev/null +++ b/llvm/test/Transforms/ConstraintElimination/uge.ll @@ -0,0 +1,255 @@ +; NOTE: Assertions have been autogenerated by utils/update_test_checks.py +; RUN: opt -S %s | FileCheck %s + +declare void @use(i1) + +define void @test_1_variable_constraint(i32 %x, i32 %y, i32 %z) { +; CHECK-LABEL: @test_1_variable_constraint( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp uge i32 [[X:%.*]], [[Y:%.*]] +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[BB2:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[T_1:%.*]] = icmp uge i32 [[X]], [[Y]] +; CHECK-NEXT: call void @use(i1 [[T_1]]) +; CHECK-NEXT: [[C_2:%.*]] = icmp uge i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[C_2]]) +; CHECK-NEXT: [[C_3:%.*]] = icmp uge i32 [[Y]], [[X]] +; CHECK-NEXT: call void @use(i1 [[C_3]]) +; CHECK-NEXT: [[C_4:%.*]] = icmp uge i32 10, [[X]] +; CHECK-NEXT: call void @use(i1 [[C_4]]) +; CHECK-NEXT: ret void +; CHECK: bb2: +; CHECK-NEXT: [[T_2:%.*]] = icmp uge i32 [[Y]], [[X]] +; CHECK-NEXT: call void @use(i1 [[T_2]]) +; CHECK-NEXT: [[F_1:%.*]] = icmp uge i32 [[X]], [[Y]] +; CHECK-NEXT: call void @use(i1 [[F_1]]) +; CHECK-NEXT: [[C_5:%.*]] = icmp uge i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[C_5]]) +; CHECK-NEXT: [[C_6:%.*]] = icmp uge i32 10, [[X]] +; CHECK-NEXT: call void @use(i1 [[C_6]]) +; CHECK-NEXT: ret void +; +entry: + %c.1 = icmp uge i32 %x, %y + br i1 %c.1, label %bb1, label %bb2 + +bb1: + %t.1 = icmp uge i32 %x, %y + call void @use(i1 %t.1) + %c.2 = icmp uge i32 %x, 10 + call void @use(i1 %c.2) + %c.3 = icmp uge i32 %y, %x + call void @use(i1 %c.3) + %c.4 = icmp uge i32 10, %x + call void @use(i1 %c.4) + ret void + +bb2: + %t.2 = icmp uge i32 %y, %x + call void @use(i1 %t.2) + %f.1 = icmp uge i32 %x, %y + call void @use(i1 %f.1) + %c.5 = icmp uge i32 %x, 10 + call void @use(i1 %c.5) + %c.6 = icmp uge i32 10, %x + call void @use(i1 %c.6) + ret void +} + +define void @test_1_constant_constraint(i32 %x) { +; CHECK-LABEL: @test_1_constant_constraint( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp uge i32 [[X:%.*]], 10 +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[BB2:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[T_1:%.*]] = icmp uge i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[T_1]]) +; CHECK-NEXT: [[T_2:%.*]] = icmp uge i32 [[X]], 9 +; CHECK-NEXT: call void @use(i1 [[T_2]]) +; CHECK-NEXT: [[C_2:%.*]] = icmp uge i32 [[X]], 11 +; CHECK-NEXT: call void @use(i1 [[C_2]]) +; CHECK-NEXT: [[C_4:%.*]] = icmp uge i32 10, [[X]] +; CHECK-NEXT: call void @use(i1 [[C_4]]) +; CHECK-NEXT: ret void +; CHECK: bb2: +; CHECK-NEXT: [[T_3:%.*]] = icmp uge i32 11, [[X]] +; CHECK-NEXT: call void @use(i1 [[T_3]]) +; CHECK-NEXT: [[F_1:%.*]] = icmp uge i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[F_1]]) +; CHECK-NEXT: [[F_1_1:%.*]] = icmp uge i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[F_1_1]]) +; CHECK-NEXT: [[C_5:%.*]] = icmp uge i32 [[X]], 9 +; CHECK-NEXT: call void @use(i1 [[C_5]]) +; CHECK-NEXT: [[C_6:%.*]] = icmp uge i32 1, [[X]] +; CHECK-NEXT: call void @use(i1 [[C_6]]) +; CHECK-NEXT: ret void +; +entry: + %c.1 = icmp uge i32 %x, 10 + br i1 %c.1, label %bb1, label %bb2 + +bb1: + %t.1 = icmp uge i32 %x, 10 + call void @use(i1 %t.1) + %t.2 = icmp uge i32 %x, 9 + call void @use(i1 %t.2) + %c.2 = icmp uge i32 %x, 11 + call void @use(i1 %c.2) + %c.4 = icmp uge i32 10, %x + call void @use(i1 %c.4) + ret void + +bb2: + %t.3 = icmp uge i32 11, %x + call void @use(i1 %t.3) + %f.1 = icmp uge i32 %x, 10 + call void @use(i1 %f.1) + + + %f.1.1 = icmp uge i32 %x, 10 + call void @use(i1 %f.1.1) + %c.5 = icmp uge i32 %x, 9 + call void @use(i1 %c.5) + %c.6 = icmp uge i32 1, %x + call void @use(i1 %c.6) + ret void +} + +define i32 @test1(i32 %x, i32 %y, i32 %z) { +; CHECK-LABEL: @test1( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp uge i32 [[X:%.*]], [[Y:%.*]] +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[EXIT:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[C_2:%.*]] = icmp uge i32 [[Y]], [[Z:%.*]] +; CHECK-NEXT: br i1 [[C_2]], label [[BB2:%.*]], label [[EXIT]] +; CHECK: bb2: +; CHECK-NEXT: [[C_3:%.*]] = icmp uge i32 [[X]], [[Z]] +; CHECK-NEXT: br i1 [[C_3]], label [[BB3:%.*]], label [[EXIT]] +; CHECK: bb3: +; CHECK-NEXT: ret i32 10 +; CHECK: exit: +; CHECK-NEXT: ret i32 20 +; +entry: + %c.1 = icmp uge i32 %x, %y + br i1 %c.1, label %bb1, label %exit + +bb1: + %c.2 = icmp uge i32 %y, %z + br i1 %c.2, label %bb2, label %exit + +bb2: + %c.3 = icmp uge i32 %x, %z + br i1 %c.3, label %bb3, label %exit + +bb3: + ret i32 10 + +exit: + ret i32 20 +} + + +define i32 @test2(i32 %x, i32 %y, i32 %z, i32 %a) { +; CHECK-LABEL: @test2( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp uge i32 [[X:%.*]], [[Y:%.*]] +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[EXIT:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[C_2:%.*]] = icmp uge i32 [[Y]], [[Z:%.*]] +; CHECK-NEXT: br i1 [[C_2]], label [[BB2:%.*]], label [[EXIT]] +; CHECK: bb2: +; CHECK-NEXT: [[C_3:%.*]] = icmp uge i32 [[X]], [[A:%.*]] +; CHECK-NEXT: br i1 [[C_3]], label [[BB3:%.*]], label [[EXIT]] +; CHECK: bb3: +; CHECK-NEXT: ret i32 10 +; CHECK: exit: +; CHECK-NEXT: ret i32 20 +; +entry: + %c.1 = icmp uge i32 %x, %y + br i1 %c.1, label %bb1, label %exit + +bb1: + %c.2 = icmp uge i32 %y, %z + br i1 %c.2, label %bb2, label %exit + +bb2: + %c.3 = icmp uge i32 %x, %a + br i1 %c.3, label %bb3, label %exit + +bb3: + ret i32 10 + +exit: + ret i32 20 +} + + +define i32 @test3(i32 %x, i32 %y) { +; CHECK-LABEL: @test3( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp uge i32 [[X:%.*]], 10 +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[EXIT:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[C_2:%.*]] = icmp uge i32 [[Y:%.*]], 20 +; CHECK-NEXT: br i1 [[C_2]], label [[BB2:%.*]], label [[EXIT]] +; CHECK: bb2: +; CHECK-NEXT: ret i32 10 +; CHECK: exit: +; CHECK-NEXT: ret i32 20 +; +entry: + %c.1 = icmp uge i32 %x, 10 + br i1 %c.1, label %bb1, label %exit + +bb1: + %c.2 = icmp uge i32 %y, 20 + br i1 %c.2, label %bb2, label %exit + +bb2: + ret i32 10 + +exit: + ret i32 20 +} + +define i32 @test4(i32 %x, i32 %y, i32 %z) { +; CHECK-LABEL: @test4( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp uge i32 [[X:%.*]], [[Y:%.*]] +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[EXIT:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[C_2:%.*]] = icmp uge i32 [[Y]], [[Z:%.*]] +; CHECK-NEXT: br i1 [[C_2]], label [[BB2:%.*]], label [[EXIT]] +; CHECK: bb2: +; CHECK-NEXT: [[T_1:%.*]] = icmp uge i32 [[X]], [[Z]] +; CHECK-NEXT: call void @use(i1 [[T_1]]) +; CHECK-NEXT: [[U_1:%.*]] = icmp eq i32 [[X]], [[Z]] +; CHECK-NEXT: call void @use(i1 [[U_1]]) +; CHECK-NEXT: ret i32 10 +; CHECK: exit: +; CHECK-NEXT: ret i32 20 +; +entry: + %c.1 = icmp uge i32 %x, %y + br i1 %c.1, label %bb1, label %exit + +bb1: + %c.2 = icmp uge i32 %y, %z + br i1 %c.2, label %bb2, label %exit + +bb2: + %t.1 = icmp uge i32 %x, %z + call void @use(i1 %t.1) + %u.1 = icmp eq i32 %x, %z + call void @use(i1 %u.1) + ret i32 10 + + +exit: + ret i32 20 +} + + diff --git a/llvm/test/Transforms/ConstraintElimination/ugt-ule.ll b/llvm/test/Transforms/ConstraintElimination/ugt-ule.ll new file mode 100644 index 00000000000000..c49ce7360cd683 --- /dev/null +++ b/llvm/test/Transforms/ConstraintElimination/ugt-ule.ll @@ -0,0 +1,38 @@ +; NOTE: Assertions have been autogenerated by utils/update_test_checks.py +; RUN: opt -S %s | FileCheck %s + +declare void @use(i1) + +define void @test(i8* %m, i8* %ptr) { +; CHECK-LABEL: @test( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[CMP_1:%.*]] = icmp ult i8* [[M:%.*]], [[PTR:%.*]] +; CHECK-NEXT: br i1 [[CMP_1]], label [[BB_1:%.*]], label [[BB_2:%.*]] +; CHECK: bb.1: +; CHECK-NEXT: [[CMP_2:%.*]] = icmp uge i8* [[M]], [[PTR]] +; CHECK-NEXT: call void @use(i1 [[CMP_2]]) +; CHECK-NEXT: ret void +; CHECK: bb.2: +; CHECK-NEXT: br label [[BB_2_NEXT:%.*]] +; CHECK: bb.2.next: +; CHECK-NEXT: [[CMP_3:%.*]] = icmp uge i8* [[M]], [[PTR]] +; CHECK-NEXT: call void @use(i1 [[CMP_3]]) +; CHECK-NEXT: ret void +; +entry: + %cmp.1 = icmp ult i8* %m, %ptr + br i1 %cmp.1, label %bb.1, label %bb.2 + +bb.1: + %cmp.2 = icmp uge i8* %m, %ptr + call void @use(i1 %cmp.2) + ret void + +bb.2: + br label %bb.2.next + +bb.2.next: + %cmp.3 = icmp uge i8* %m, %ptr + call void @use(i1 %cmp.3) + ret void +} diff --git a/llvm/test/Transforms/ConstraintElimination/ule.ll b/llvm/test/Transforms/ConstraintElimination/ule.ll new file mode 100644 index 00000000000000..2cb3750fad243d --- /dev/null +++ b/llvm/test/Transforms/ConstraintElimination/ule.ll @@ -0,0 +1,254 @@ +; NOTE: Assertions have been autogenerated by utils/update_test_checks.py +; RUN: opt -S %s | FileCheck %s + +declare void @use(i1) + +define void @test_1_variable_constraint(i32 %x, i32 %y, i32 %z) { +; CHECK-LABEL: @test_1_variable_constraint( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp ule i32 [[X:%.*]], [[Y:%.*]] +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[BB2:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[T_1:%.*]] = icmp ule i32 [[X]], [[Y]] +; CHECK-NEXT: call void @use(i1 [[T_1]]) +; CHECK-NEXT: [[C_2:%.*]] = icmp ule i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[C_2]]) +; CHECK-NEXT: [[C_3:%.*]] = icmp ule i32 [[Y]], [[X]] +; CHECK-NEXT: call void @use(i1 [[C_3]]) +; CHECK-NEXT: [[C_4:%.*]] = icmp ule i32 10, [[X]] +; CHECK-NEXT: call void @use(i1 [[C_4]]) +; CHECK-NEXT: ret void +; CHECK: bb2: +; CHECK-NEXT: [[T_2:%.*]] = icmp ule i32 [[Y]], [[X]] +; CHECK-NEXT: call void @use(i1 [[T_2]]) +; CHECK-NEXT: [[F_1:%.*]] = icmp ule i32 [[X]], [[Y]] +; CHECK-NEXT: call void @use(i1 [[F_1]]) +; CHECK-NEXT: [[C_5:%.*]] = icmp ule i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[C_5]]) +; CHECK-NEXT: [[C_6:%.*]] = icmp ule i32 10, [[X]] +; CHECK-NEXT: call void @use(i1 [[C_6]]) +; CHECK-NEXT: ret void +; +entry: + %c.1 = icmp ule i32 %x, %y + br i1 %c.1, label %bb1, label %bb2 + +bb1: + %t.1 = icmp ule i32 %x, %y + call void @use(i1 %t.1) + %c.2 = icmp ule i32 %x, 10 + call void @use(i1 %c.2) + %c.3 = icmp ule i32 %y, %x + call void @use(i1 %c.3) + %c.4 = icmp ule i32 10, %x + call void @use(i1 %c.4) + ret void + +bb2: + %t.2 = icmp ule i32 %y, %x + call void @use(i1 %t.2) + %f.1 = icmp ule i32 %x, %y + call void @use(i1 %f.1) + %c.5 = icmp ule i32 %x, 10 + call void @use(i1 %c.5) + %c.6 = icmp ule i32 10, %x + call void @use(i1 %c.6) + ret void +} + +define void @test_1_constant_constraint(i32 %x) { +; CHECK-LABEL: @test_1_constant_constraint( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp ule i32 [[X:%.*]], 10 +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[BB2:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[T_1:%.*]] = icmp ule i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[T_1]]) +; CHECK-NEXT: [[T_2:%.*]] = icmp ule i32 [[X]], 11 +; CHECK-NEXT: call void @use(i1 [[T_2]]) +; CHECK-NEXT: [[C_2:%.*]] = icmp ule i32 [[X]], 9 +; CHECK-NEXT: call void @use(i1 [[C_2]]) +; CHECK-NEXT: [[C_4:%.*]] = icmp ule i32 10, [[X]] +; CHECK-NEXT: call void @use(i1 [[C_4]]) +; CHECK-NEXT: ret void +; CHECK: bb2: +; CHECK-NEXT: [[T_3:%.*]] = icmp ule i32 10, [[X]] +; CHECK-NEXT: call void @use(i1 [[T_3]]) +; CHECK-NEXT: [[F_1:%.*]] = icmp ule i32 [[X]], 9 +; CHECK-NEXT: call void @use(i1 [[F_1]]) +; CHECK-NEXT: [[F_1_1:%.*]] = icmp ule i32 [[X]], 10 +; CHECK-NEXT: call void @use(i1 [[F_1_1]]) +; CHECK-NEXT: [[C_5:%.*]] = icmp ule i32 [[X]], 11 +; CHECK-NEXT: call void @use(i1 [[C_5]]) +; CHECK-NEXT: [[C_6:%.*]] = icmp ule i32 10, [[X]] +; CHECK-NEXT: call void @use(i1 [[C_6]]) +; CHECK-NEXT: ret void +; +entry: + %c.1 = icmp ule i32 %x, 10 + br i1 %c.1, label %bb1, label %bb2 + +bb1: + %t.1 = icmp ule i32 %x, 10 + call void @use(i1 %t.1) + %t.2 = icmp ule i32 %x, 11 + call void @use(i1 %t.2) + %c.2 = icmp ule i32 %x, 9 + call void @use(i1 %c.2) + %c.4 = icmp ule i32 10, %x + call void @use(i1 %c.4) + ret void + +bb2: + %t.3 = icmp ule i32 10, %x + call void @use(i1 %t.3) + %f.1 = icmp ule i32 %x, 9 + call void @use(i1 %f.1) + + + %f.1.1 = icmp ule i32 %x, 10 + call void @use(i1 %f.1.1) + %c.5 = icmp ule i32 %x, 11 + call void @use(i1 %c.5) + %c.6 = icmp ule i32 10, %x + call void @use(i1 %c.6) + ret void +} + + +define i32 @test1(i32 %x, i32 %y, i32 %z) { +; CHECK-LABEL: @test1( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp ule i32 [[X:%.*]], [[Y:%.*]] +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[EXIT:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[C_2:%.*]] = icmp ule i32 [[Y]], [[Z:%.*]] +; CHECK-NEXT: br i1 [[C_2]], label [[BB2:%.*]], label [[EXIT]] +; CHECK: bb2: +; CHECK-NEXT: [[C_3:%.*]] = icmp ule i32 [[X]], [[Z]] +; CHECK-NEXT: br i1 [[C_3]], label [[BB3:%.*]], label [[EXIT]] +; CHECK: bb3: +; CHECK-NEXT: ret i32 10 +; CHECK: exit: +; CHECK-NEXT: ret i32 20 +; +entry: + %c.1 = icmp ule i32 %x, %y + br i1 %c.1, label %bb1, label %exit + +bb1: + %c.2 = icmp ule i32 %y, %z + br i1 %c.2, label %bb2, label %exit + +bb2: + %c.3 = icmp ule i32 %x, %z + br i1 %c.3, label %bb3, label %exit + +bb3: + ret i32 10 + +exit: + ret i32 20 +} + + +define i32 @test2(i32 %x, i32 %y, i32 %z, i32 %a) { +; CHECK-LABEL: @test2( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp ule i32 [[X:%.*]], [[Y:%.*]] +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[EXIT:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[C_2:%.*]] = icmp ule i32 [[Y]], [[Z:%.*]] +; CHECK-NEXT: br i1 [[C_2]], label [[BB2:%.*]], label [[EXIT]] +; CHECK: bb2: +; CHECK-NEXT: [[C_3:%.*]] = icmp ule i32 [[X]], [[A:%.*]] +; CHECK-NEXT: br i1 [[C_3]], label [[BB3:%.*]], label [[EXIT]] +; CHECK: bb3: +; CHECK-NEXT: ret i32 10 +; CHECK: exit: +; CHECK-NEXT: ret i32 20 +; +entry: + %c.1 = icmp ule i32 %x, %y + br i1 %c.1, label %bb1, label %exit + +bb1: + %c.2 = icmp ule i32 %y, %z + br i1 %c.2, label %bb2, label %exit + +bb2: + %c.3 = icmp ule i32 %x, %a + br i1 %c.3, label %bb3, label %exit + +bb3: + ret i32 10 + +exit: + ret i32 20 +} + + +define i32 @test3(i32 %x, i32 %y) { +; CHECK-LABEL: @test3( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp ule i32 [[X:%.*]], 10 +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[EXIT:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[C_2:%.*]] = icmp ule i32 [[Y:%.*]], 20 +; CHECK-NEXT: br i1 [[C_2]], label [[BB2:%.*]], label [[EXIT]] +; CHECK: bb2: +; CHECK-NEXT: ret i32 10 +; CHECK: exit: +; CHECK-NEXT: ret i32 20 +; +entry: + %c.1 = icmp ule i32 %x, 10 + br i1 %c.1, label %bb1, label %exit + +bb1: + %c.2 = icmp ule i32 %y, 20 + br i1 %c.2, label %bb2, label %exit + +bb2: + ret i32 10 + +exit: + ret i32 20 +} + +define i32 @test4(i32 %x, i32 %y, i32 %z) { +; CHECK-LABEL: @test4( +; CHECK-NEXT: entry: +; CHECK-NEXT: [[C_1:%.*]] = icmp ule i32 [[X:%.*]], [[Y:%.*]] +; CHECK-NEXT: br i1 [[C_1]], label [[BB1:%.*]], label [[EXIT:%.*]] +; CHECK: bb1: +; CHECK-NEXT: [[C_2:%.*]] = icmp ule i32 [[Y]], [[Z:%.*]] +; CHECK-NEXT: br i1 [[C_2]], label [[BB2:%.*]], label [[EXIT]] +; CHECK: bb2: +; CHECK-NEXT: [[T_1:%.*]] = icmp ule i32 [[X]], [[Z]] +; CHECK-NEXT: call void @use(i1 [[T_1]]) +; CHECK-NEXT: [[U_1:%.*]] = icmp eq i32 [[X]], [[Z]] +; CHECK-NEXT: call void @use(i1 [[U_1]]) +; CHECK-NEXT: ret i32 10 +; CHECK: exit: +; CHECK-NEXT: ret i32 20 +; +entry: + %c.1 = icmp ule i32 %x, %y + br i1 %c.1, label %bb1, label %exit + +bb1: + %c.2 = icmp ule i32 %y, %z + br i1 %c.2, label %bb2, label %exit + +bb2: + %t.1 = icmp ule i32 %x, %z + call void @use(i1 %t.1) + %u.1 = icmp eq i32 %x, %z + call void @use(i1 %u.1) + ret i32 10 + + +exit: + ret i32 20 +}