2020-04-21 17:54:14 +08:00
|
|
|
// RUN: mlir-opt -allow-unregistered-dialect %s -pass-pipeline="func(sccp)" -split-input-file | FileCheck %s
|
|
|
|
|
|
|
|
/// Check that a constant is properly propagated when only one edge is taken.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @simple(
|
|
|
|
func @simple(%arg0 : i32) -> i32 {
|
|
|
|
// CHECK: %[[CST:.*]] = constant 1 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
// CHECK-NOT: scf.if
|
2020-04-21 17:54:14 +08:00
|
|
|
// CHECK: return %[[CST]] : i32
|
|
|
|
|
|
|
|
%cond = constant true
|
2020-05-13 18:12:30 +08:00
|
|
|
%res = scf.if %cond -> (i32) {
|
2020-04-21 17:54:14 +08:00
|
|
|
%1 = constant 1 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.yield %1 : i32
|
2020-04-21 17:54:14 +08:00
|
|
|
} else {
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.yield %arg0 : i32
|
2020-04-21 17:54:14 +08:00
|
|
|
}
|
|
|
|
return %res : i32
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Check that a constant is properly propagated when both edges produce the
|
|
|
|
/// same value.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @simple_both_same(
|
|
|
|
func @simple_both_same(%cond : i1) -> i32 {
|
|
|
|
// CHECK: %[[CST:.*]] = constant 1 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
// CHECK-NOT: scf.if
|
2020-04-21 17:54:14 +08:00
|
|
|
// CHECK: return %[[CST]] : i32
|
|
|
|
|
2020-05-13 18:12:30 +08:00
|
|
|
%res = scf.if %cond -> (i32) {
|
2020-04-21 17:54:14 +08:00
|
|
|
%1 = constant 1 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.yield %1 : i32
|
2020-04-21 17:54:14 +08:00
|
|
|
} else {
|
|
|
|
%2 = constant 1 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.yield %2 : i32
|
2020-04-21 17:54:14 +08:00
|
|
|
}
|
|
|
|
return %res : i32
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Check that the arguments go to overdefined if the branch cannot detect when
|
|
|
|
/// a specific successor is taken.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @overdefined_unknown_condition(
|
|
|
|
func @overdefined_unknown_condition(%cond : i1, %arg0 : i32) -> i32 {
|
2020-05-13 18:12:30 +08:00
|
|
|
// CHECK: %[[RES:.*]] = scf.if
|
2020-04-21 17:54:14 +08:00
|
|
|
// CHECK: return %[[RES]] : i32
|
|
|
|
|
2020-05-13 18:12:30 +08:00
|
|
|
%res = scf.if %cond -> (i32) {
|
2020-04-21 17:54:14 +08:00
|
|
|
%1 = constant 1 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.yield %1 : i32
|
2020-04-21 17:54:14 +08:00
|
|
|
} else {
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.yield %arg0 : i32
|
2020-04-21 17:54:14 +08:00
|
|
|
}
|
|
|
|
return %res : i32
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Check that the arguments go to overdefined if there are conflicting
|
|
|
|
/// constants.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @overdefined_different_constants(
|
|
|
|
func @overdefined_different_constants(%cond : i1) -> i32 {
|
2020-05-13 18:12:30 +08:00
|
|
|
// CHECK: %[[RES:.*]] = scf.if
|
2020-04-21 17:54:14 +08:00
|
|
|
// CHECK: return %[[RES]] : i32
|
|
|
|
|
2020-05-13 18:12:30 +08:00
|
|
|
%res = scf.if %cond -> (i32) {
|
2020-04-21 17:54:14 +08:00
|
|
|
%1 = constant 1 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.yield %1 : i32
|
2020-04-21 17:54:14 +08:00
|
|
|
} else {
|
|
|
|
%2 = constant 2 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.yield %2 : i32
|
2020-04-21 17:54:14 +08:00
|
|
|
}
|
|
|
|
return %res : i32
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Check that arguments are properly merged across loop-like control flow.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @simple_loop(
|
|
|
|
func @simple_loop(%arg0 : index, %arg1 : index, %arg2 : index) -> i32 {
|
|
|
|
// CHECK: %[[CST:.*]] = constant 0 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
// CHECK-NOT: scf.for
|
2020-04-21 17:54:14 +08:00
|
|
|
// CHECK: return %[[CST]] : i32
|
|
|
|
|
|
|
|
%s0 = constant 0 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
%result = scf.for %i0 = %arg0 to %arg1 step %arg2 iter_args(%si = %s0) -> (i32) {
|
2020-04-21 17:54:14 +08:00
|
|
|
%sn = addi %si, %si : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.yield %sn : i32
|
2020-04-21 17:54:14 +08:00
|
|
|
}
|
|
|
|
return %result : i32
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Check that arguments go to overdefined when loop backedges produce a
|
|
|
|
/// conflicting value.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @loop_overdefined(
|
|
|
|
func @loop_overdefined(%arg0 : index, %arg1 : index, %arg2 : index) -> i32 {
|
2020-05-13 18:12:30 +08:00
|
|
|
// CHECK: %[[RES:.*]] = scf.for
|
2020-04-21 17:54:14 +08:00
|
|
|
// CHECK: return %[[RES]] : i32
|
|
|
|
|
|
|
|
%s0 = constant 1 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
%result = scf.for %i0 = %arg0 to %arg1 step %arg2 iter_args(%si = %s0) -> (i32) {
|
2020-04-21 17:54:14 +08:00
|
|
|
%sn = addi %si, %si : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.yield %sn : i32
|
2020-04-21 17:54:14 +08:00
|
|
|
}
|
|
|
|
return %result : i32
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Test that we can properly propagate within inner control, and in situations
|
|
|
|
/// where the executable edges within the CFG are sensitive to the current state
|
|
|
|
/// of the analysis.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @loop_inner_control_flow(
|
|
|
|
func @loop_inner_control_flow(%arg0 : index, %arg1 : index, %arg2 : index) -> i32 {
|
|
|
|
// CHECK: %[[CST:.*]] = constant 1 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
// CHECK-NOT: scf.for
|
|
|
|
// CHECK-NOT: scf.if
|
2020-04-21 17:54:14 +08:00
|
|
|
// CHECK: return %[[CST]] : i32
|
|
|
|
|
|
|
|
%cst_1 = constant 1 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
%result = scf.for %i0 = %arg0 to %arg1 step %arg2 iter_args(%si = %cst_1) -> (i32) {
|
2020-04-21 17:54:14 +08:00
|
|
|
%cst_20 = constant 20 : i32
|
2021-01-15 03:35:15 +08:00
|
|
|
%cond = cmpi ult, %si, %cst_20 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
%inner_res = scf.if %cond -> (i32) {
|
2020-04-21 17:54:14 +08:00
|
|
|
%1 = constant 1 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.yield %1 : i32
|
2020-04-21 17:54:14 +08:00
|
|
|
} else {
|
|
|
|
%si_inc = addi %si, %cst_1 : i32
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.yield %si_inc : i32
|
2020-04-21 17:54:14 +08:00
|
|
|
}
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.yield %inner_res : i32
|
2020-04-21 17:54:14 +08:00
|
|
|
}
|
|
|
|
return %result : i32
|
|
|
|
}
|
2021-05-05 05:50:08 +08:00
|
|
|
|
|
|
|
/// Test that we can properly visit region successors when the terminator is not
|
|
|
|
/// return-like.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @overdefined_non_returnlike(
|
|
|
|
func @overdefined_non_returnlike(%arg1 : i32) {
|
|
|
|
// CHECK: scf.while (%[[ARG:.*]] = %[[INPUT:.*]])
|
|
|
|
// CHECK-NEXT: %[[COND:.*]] = cmpi slt, %[[ARG]], %{{.*}} : i32
|
|
|
|
// CHECK-NEXT: scf.condition(%[[COND]]) %[[ARG]] : i32
|
|
|
|
|
|
|
|
%c2_i32 = constant 2 : i32
|
|
|
|
%0 = scf.while (%arg2 = %c2_i32) : (i32) -> (i32) {
|
|
|
|
%1 = cmpi slt, %arg2, %arg1 : i32
|
|
|
|
scf.condition(%1) %arg2 : i32
|
|
|
|
} do {
|
|
|
|
^bb0(%arg2: i32):
|
|
|
|
scf.yield %arg2 : i32
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|