; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --include-generated-funcs --version 5 ; RUN: opt -passes="ipsccp" -funcspec-min-function-size=1 \ ; RUN: -funcspec-for-literal-constant=true \ ; RUN: -funcspec-min-codesize-savings=50 \ ; RUN: -funcspec-min-latency-savings=0 \ ; RUN: -S < %s | FileCheck %s ; Verify that we are able to estimate the codesize savings arising from a branch ; based on a comparison with a value found to have a constant range by IPSCCP. define i32 @main() { %notspec = call i32 @test_use_on_lhs(i32 8) %spec1 = call i32 @test_use_on_lhs(i32 0) %spec2 = call i32 @test_use_on_rhs(i32 1) %sum1 = add i32 %notspec, %spec1 %sum2 = add i32 %sum1, %spec2 ret i32 %sum2 } define i32 @test_use_on_lhs(i32 %x) { entry: %range = call i32 @foo(), !range !{ i32 1, i32 0 } %bound = shl nsw nuw i32 %range, 3 %cmp = icmp uge i32 %x, %bound br i1 %cmp, label %if.then, label %if.end if.then: call void @do_something() call void @do_something() call void @do_something() call void @do_something() br label %if.end if.end: %res = phi i32 [ 0, %entry ], [ 1, %if.then] ret i32 %res } define i32 @test_use_on_rhs(i32 %x) { entry: %range = call i32 @foo(), !range !{ i32 1, i32 0 } %bound = shl nsw nuw i32 %range, 3 %x.sub = sub nsw nuw i32 %x, 1 %cmp = icmp ult i32 %bound, %x.sub br i1 %cmp, label %if.then, label %if.end if.then: call void @do_something() call void @do_something() call void @do_something() call void @do_something() br label %if.end if.end: %res = phi i32 [ 0, %entry ], [ 1, %if.then] ret i32 %res } declare i32 @foo() declare void @do_something() ; CHECK-LABEL: define range(i32 0, 2) i32 @main() { ; CHECK-NEXT: [[NOTSPEC:%.*]] = call i32 @test_use_on_lhs(i32 8) ; CHECK-NEXT: [[SPEC1:%.*]] = call i32 @test_use_on_lhs.specialized.1(i32 0) ; CHECK-NEXT: [[SPEC2:%.*]] = call i32 @test_use_on_rhs.specialized.2(i32 1) ; CHECK-NEXT: [[SUM:%.*]] = add nuw nsw i32 [[NOTSPEC]], 0 ; CHECK-NEXT: [[RES:%.*]] = add nuw nsw i32 [[SUM]], 0 ; CHECK-NEXT: ret i32 [[RES]] ; ; ; CHECK-LABEL: define range(i32 0, 2) i32 @test_use_on_lhs( ; CHECK-SAME: i32 [[X:%.*]]) { ; CHECK-NEXT: [[ENTRY:.*]]: ; CHECK-NEXT: [[RANGE:%.*]] = call i32 @foo(), !range [[RNG0:![0-9]+]] ; CHECK-NEXT: [[BOUND:%.*]] = shl nuw nsw i32 [[RANGE]], 3 ; CHECK-NEXT: [[CMP:%.*]] = icmp uge i32 [[X]], [[BOUND]] ; CHECK-NEXT: br i1 [[CMP]], label %[[IF_THEN:.*]], label %[[IF_END:.*]] ; CHECK: [[IF_THEN]]: ; CHECK-NEXT: call void @do_something() ; CHECK-NEXT: call void @do_something() ; CHECK-NEXT: call void @do_something() ; CHECK-NEXT: call void @do_something() ; CHECK-NEXT: br label %[[IF_END]] ; CHECK: [[IF_END]]: ; CHECK-NEXT: [[RES:%.*]] = phi i32 [ 0, %[[ENTRY]] ], [ 1, %[[IF_THEN]] ] ; CHECK-NEXT: ret i32 [[RES]] ; ; ; CHECK-LABEL: define range(i32 0, 2) i32 @test_use_on_rhs( ; CHECK-SAME: i32 [[X:%.*]]) { ; CHECK-NEXT: [[ENTRY:.*]]: ; CHECK-NEXT: [[RANGE:%.*]] = call i32 @foo(), !range [[RNG0]] ; CHECK-NEXT: [[BOUND:%.*]] = shl nuw nsw i32 [[RANGE]], 3 ; CHECK-NEXT: [[X_SUB:%.*]] = sub nuw nsw i32 [[X]], 1 ; CHECK-NEXT: [[CMP:%.*]] = icmp ult i32 [[BOUND]], [[X_SUB]] ; CHECK-NEXT: br i1 [[CMP]], label %[[IF_THEN:.*]], label %[[IF_END:.*]] ; CHECK: [[IF_THEN]]: ; CHECK-NEXT: call void @do_something() ; CHECK-NEXT: call void @do_something() ; CHECK-NEXT: call void @do_something() ; CHECK-NEXT: call void @do_something() ; CHECK-NEXT: br label %[[IF_END]] ; CHECK: [[IF_END]]: ; CHECK-NEXT: [[RES:%.*]] = phi i32 [ 0, %[[ENTRY]] ], [ 1, %[[IF_THEN]] ] ; CHECK-NEXT: ret i32 [[RES]] ; ; ; CHECK-LABEL: define internal i32 @test_use_on_lhs.specialized.1( ; CHECK-SAME: i32 [[X:%.*]]) { ; CHECK-NEXT: [[ENTRY:.*:]] ; CHECK-NEXT: [[RANGE:%.*]] = call i32 @foo(), !range [[RNG0]] ; CHECK-NEXT: [[BOUND:%.*]] = shl nuw nsw i32 [[RANGE]], 3 ; CHECK-NEXT: br label %[[IF_END:.*]] ; CHECK: [[IF_END]]: ; CHECK-NEXT: ret i32 poison ; ; ; CHECK-LABEL: define internal i32 @test_use_on_rhs.specialized.2( ; CHECK-SAME: i32 [[X:%.*]]) { ; CHECK-NEXT: [[ENTRY:.*:]] ; CHECK-NEXT: [[RANGE:%.*]] = call i32 @foo(), !range [[RNG0]] ; CHECK-NEXT: [[BOUND:%.*]] = shl nuw nsw i32 [[RANGE]], 3 ; CHECK-NEXT: br label %[[IF_END:.*]] ; CHECK: [[IF_END]]: ; CHECK-NEXT: ret i32 poison ; ;. ; CHECK: [[RNG0]] = !{i32 1, i32 0} ;.