2022-03-08 11:16:03 +08:00
|
|
|
// RUN: mlir-opt -allow-unregistered-dialect %s -pass-pipeline='func.func(canonicalize)' -split-input-file | FileCheck %s
|
2020-05-05 10:54:36 +08:00
|
|
|
|
|
|
|
// Check the simple case of single operation blocks with a return.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @return_blocks(
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @return_blocks() {
|
2020-05-05 10:54:36 +08:00
|
|
|
// CHECK: "foo.cond_br"()[^bb1, ^bb1]
|
|
|
|
// CHECK: ^bb1:
|
|
|
|
// CHECK-NEXT: return
|
|
|
|
// CHECK-NOT: ^bb2
|
|
|
|
|
|
|
|
"foo.cond_br"() [^bb1, ^bb2] : () -> ()
|
|
|
|
|
|
|
|
^bb1:
|
|
|
|
return
|
|
|
|
^bb2:
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the case of identical blocks with matching arguments.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @matching_arguments(
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @matching_arguments() -> i32 {
|
2020-05-05 10:54:36 +08:00
|
|
|
// CHECK: "foo.cond_br"()[^bb1, ^bb1]
|
|
|
|
// CHECK: ^bb1(%{{.*}}: i32):
|
|
|
|
// CHECK-NEXT: return
|
|
|
|
// CHECK-NOT: ^bb2
|
|
|
|
|
|
|
|
"foo.cond_br"() [^bb1, ^bb2] : () -> ()
|
|
|
|
|
|
|
|
^bb1(%arg0 : i32):
|
|
|
|
return %arg0 : i32
|
|
|
|
^bb2(%arg1 : i32):
|
|
|
|
return %arg1 : i32
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that no merging occurs if there is an operand mismatch and we can't
|
|
|
|
// update th predecessor.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @mismatch_unknown_terminator
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @mismatch_unknown_terminator(%arg0 : i32, %arg1 : i32) -> i32 {
|
2020-05-05 10:54:36 +08:00
|
|
|
// CHECK: "foo.cond_br"()[^bb1, ^bb2]
|
|
|
|
|
|
|
|
"foo.cond_br"() [^bb1, ^bb2] : () -> ()
|
|
|
|
|
|
|
|
^bb1:
|
|
|
|
return %arg0 : i32
|
|
|
|
^bb2:
|
|
|
|
return %arg1 : i32
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that merging does occurs if there is an operand mismatch and we can
|
|
|
|
// update th predecessor.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @mismatch_operands
|
|
|
|
// CHECK-SAME: %[[COND:.*]]: i1, %[[ARG0:.*]]: i32, %[[ARG1:.*]]: i32
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @mismatch_operands(%cond : i1, %arg0 : i32, %arg1 : i32) -> i32 {
|
2022-02-01 04:44:35 +08:00
|
|
|
// CHECK: %[[RES:.*]] = arith.select %[[COND]], %[[ARG0]], %[[ARG1]]
|
2020-05-05 10:54:36 +08:00
|
|
|
// CHECK: return %[[RES]]
|
|
|
|
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.cond_br %cond, ^bb1, ^bb2
|
2020-05-05 10:54:36 +08:00
|
|
|
|
|
|
|
^bb1:
|
|
|
|
return %arg0 : i32
|
|
|
|
^bb2:
|
|
|
|
return %arg1 : i32
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the same as above, but with pre-existing arguments.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @mismatch_operands_matching_arguments(
|
|
|
|
// CHECK-SAME: %[[COND:.*]]: i1, %[[ARG0:.*]]: i32, %[[ARG1:.*]]: i32
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @mismatch_operands_matching_arguments(%cond : i1, %arg0 : i32, %arg1 : i32) -> (i32, i32) {
|
2022-02-01 04:44:35 +08:00
|
|
|
// CHECK: %[[RES0:.*]] = arith.select %[[COND]], %[[ARG1]], %[[ARG0]]
|
|
|
|
// CHECK: %[[RES1:.*]] = arith.select %[[COND]], %[[ARG0]], %[[ARG1]]
|
2020-05-05 10:54:36 +08:00
|
|
|
// CHECK: return %[[RES1]], %[[RES0]]
|
|
|
|
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.cond_br %cond, ^bb1(%arg1 : i32), ^bb2(%arg0 : i32)
|
2020-05-05 10:54:36 +08:00
|
|
|
|
|
|
|
^bb1(%arg2 : i32):
|
|
|
|
return %arg0, %arg2 : i32, i32
|
|
|
|
^bb2(%arg3 : i32):
|
|
|
|
return %arg1, %arg3 : i32, i32
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that merging does not occur if the uses of the arguments differ.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @mismatch_argument_uses(
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @mismatch_argument_uses(%cond : i1, %arg0 : i32, %arg1 : i32) -> (i32, i32) {
|
2022-02-04 12:59:43 +08:00
|
|
|
// CHECK: cf.cond_br %{{.*}}, ^bb1(%{{.*}}), ^bb2
|
2020-05-05 10:54:36 +08:00
|
|
|
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.cond_br %cond, ^bb1(%arg1 : i32), ^bb2(%arg0 : i32)
|
2020-05-05 10:54:36 +08:00
|
|
|
|
|
|
|
^bb1(%arg2 : i32):
|
|
|
|
return %arg0, %arg2 : i32, i32
|
|
|
|
^bb2(%arg3 : i32):
|
|
|
|
return %arg3, %arg1 : i32, i32
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that merging does not occur if the types of the arguments differ.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @mismatch_argument_types(
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @mismatch_argument_types(%cond : i1, %arg0 : i32, %arg1 : i16) {
|
2022-02-04 12:59:43 +08:00
|
|
|
// CHECK: cf.cond_br %{{.*}}, ^bb1(%{{.*}}), ^bb2
|
2020-05-05 10:54:36 +08:00
|
|
|
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.cond_br %cond, ^bb1(%arg0 : i32), ^bb2(%arg1 : i16)
|
2020-05-05 10:54:36 +08:00
|
|
|
|
|
|
|
^bb1(%arg2 : i32):
|
|
|
|
"foo.return"(%arg2) : (i32) -> ()
|
|
|
|
^bb2(%arg3 : i16):
|
|
|
|
"foo.return"(%arg3) : (i16) -> ()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that merging does not occur if the number of the arguments differ.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @mismatch_argument_count(
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @mismatch_argument_count(%cond : i1, %arg0 : i32) {
|
2022-02-04 12:59:43 +08:00
|
|
|
// CHECK: cf.cond_br %{{.*}}, ^bb1(%{{.*}}), ^bb2
|
2020-05-05 10:54:36 +08:00
|
|
|
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.cond_br %cond, ^bb1(%arg0 : i32), ^bb2
|
2020-05-05 10:54:36 +08:00
|
|
|
|
|
|
|
^bb1(%arg2 : i32):
|
|
|
|
"foo.return"(%arg2) : (i32) -> ()
|
|
|
|
^bb2:
|
|
|
|
"foo.return"() : () -> ()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that merging does not occur if the operations differ.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @mismatch_operations(
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @mismatch_operations(%cond : i1) {
|
2022-02-04 12:59:43 +08:00
|
|
|
// CHECK: cf.cond_br %{{.*}}, ^bb1, ^bb2
|
2020-05-05 10:54:36 +08:00
|
|
|
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.cond_br %cond, ^bb1, ^bb2
|
2020-05-05 10:54:36 +08:00
|
|
|
|
|
|
|
^bb1:
|
|
|
|
"foo.return"() : () -> ()
|
|
|
|
^bb2:
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that merging does not occur if the number of operations differ.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @mismatch_operation_count(
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @mismatch_operation_count(%cond : i1) {
|
2022-02-04 12:59:43 +08:00
|
|
|
// CHECK: cf.cond_br %{{.*}}, ^bb1, ^bb2
|
2020-05-05 10:54:36 +08:00
|
|
|
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.cond_br %cond, ^bb1, ^bb2
|
2020-05-05 10:54:36 +08:00
|
|
|
|
|
|
|
^bb1:
|
|
|
|
"foo.op"() : () -> ()
|
|
|
|
return
|
|
|
|
^bb2:
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that merging does not occur if the blocks contain regions.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @contains_regions(
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @contains_regions(%cond : i1) {
|
2022-02-04 12:59:43 +08:00
|
|
|
// CHECK: cf.cond_br %{{.*}}, ^bb1, ^bb2
|
2020-05-05 10:54:36 +08:00
|
|
|
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.cond_br %cond, ^bb1, ^bb2
|
2020-05-05 10:54:36 +08:00
|
|
|
|
|
|
|
^bb1:
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.if %cond {
|
2020-05-05 10:54:36 +08:00
|
|
|
"foo.op"() : () -> ()
|
|
|
|
}
|
|
|
|
return
|
|
|
|
^bb2:
|
2020-05-13 18:12:30 +08:00
|
|
|
scf.if %cond {
|
2020-05-05 10:54:36 +08:00
|
|
|
"foo.op"() : () -> ()
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-11-21 02:05:09 +08:00
|
|
|
// Check that properly handles back edges.
|
2020-05-05 10:54:36 +08:00
|
|
|
|
|
|
|
// CHECK-LABEL: func @mismatch_loop(
|
2020-10-19 04:43:09 +08:00
|
|
|
// CHECK-SAME: %[[ARG:.*]]: i1, %[[ARG2:.*]]: i1
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @mismatch_loop(%cond : i1, %cond2 : i1) {
|
2020-11-21 02:05:09 +08:00
|
|
|
// CHECK-NEXT: %[[LOOP_CARRY:.*]] = "foo.op"
|
2022-02-04 12:59:43 +08:00
|
|
|
// CHECK: cf.cond_br %{{.*}}, ^bb1(%[[ARG2]] : i1), ^bb2
|
2020-05-05 10:54:36 +08:00
|
|
|
|
2020-11-21 02:05:09 +08:00
|
|
|
%cond3 = "foo.op"() : () -> (i1)
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.cond_br %cond, ^bb2, ^bb3
|
2020-05-05 10:54:36 +08:00
|
|
|
|
|
|
|
^bb1:
|
2020-10-19 04:43:09 +08:00
|
|
|
// CHECK: ^bb1(%[[ARG3:.*]]: i1):
|
2022-02-04 12:59:43 +08:00
|
|
|
// CHECK-NEXT: cf.cond_br %[[ARG3]], ^bb1(%[[LOOP_CARRY]] : i1), ^bb2
|
2020-05-05 10:54:36 +08:00
|
|
|
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.cond_br %cond3, ^bb1, ^bb3
|
2020-05-05 10:54:36 +08:00
|
|
|
|
|
|
|
^bb2:
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.cond_br %cond2, ^bb1, ^bb3
|
2020-05-05 10:54:36 +08:00
|
|
|
|
|
|
|
^bb3:
|
|
|
|
// CHECK: ^bb2:
|
|
|
|
// CHECK-NEXT: return
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
2020-08-26 16:13:01 +08:00
|
|
|
|
|
|
|
// Check that blocks are not merged if the types of the operands differ.
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @mismatch_operand_types(
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @mismatch_operand_types(%arg0 : i1, %arg1 : memref<i32>, %arg2 : memref<i1>) {
|
2021-10-13 07:14:57 +08:00
|
|
|
%c0_i32 = arith.constant 0 : i32
|
|
|
|
%true = arith.constant true
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.br ^bb1
|
2020-08-26 16:13:01 +08:00
|
|
|
|
|
|
|
^bb1:
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.cond_br %arg0, ^bb2, ^bb3
|
2020-08-26 16:13:01 +08:00
|
|
|
|
|
|
|
^bb2:
|
2021-02-10 20:53:11 +08:00
|
|
|
// CHECK: memref.store %{{.*}}, %{{.*}} : memref<i32>
|
|
|
|
memref.store %c0_i32, %arg1[] : memref<i32>
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.br ^bb1
|
2020-08-26 16:13:01 +08:00
|
|
|
|
|
|
|
^bb3:
|
2021-02-10 20:53:11 +08:00
|
|
|
// CHECK: memref.store %{{.*}}, %{{.*}} : memref<i1>
|
|
|
|
memref.store %true, %arg2[] : memref<i1>
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.br ^bb1
|
2020-08-26 16:13:01 +08:00
|
|
|
}
|
2020-11-21 02:05:09 +08:00
|
|
|
|
|
|
|
// Check that it is illegal to merge blocks containing an operand
|
|
|
|
// with an external user. Incorrectly performing the optimization
|
|
|
|
// anyways will result in print(merged, merged) rather than
|
|
|
|
// distinct operands.
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func private @print(%arg0: i32, %arg1: i32)
|
2020-11-21 02:05:09 +08:00
|
|
|
// CHECK-LABEL: @nomerge
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @nomerge(%arg0: i32, %i: i32) {
|
2021-10-13 07:14:57 +08:00
|
|
|
%c1_i32 = arith.constant 1 : i32
|
|
|
|
%icmp = arith.cmpi slt, %i, %arg0 : i32
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.cond_br %icmp, ^bb2, ^bb3
|
2020-11-21 02:05:09 +08:00
|
|
|
|
|
|
|
^bb2: // pred: ^bb1
|
2021-10-13 07:14:57 +08:00
|
|
|
%ip1 = arith.addi %i, %c1_i32 : i32
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.br ^bb4(%ip1 : i32)
|
2020-11-21 02:05:09 +08:00
|
|
|
|
|
|
|
^bb7: // pred: ^bb5
|
2021-10-13 07:14:57 +08:00
|
|
|
%jp1 = arith.addi %j, %c1_i32 : i32
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.br ^bb4(%jp1 : i32)
|
2020-11-21 02:05:09 +08:00
|
|
|
|
|
|
|
^bb4(%j: i32): // 2 preds: ^bb2, ^bb7
|
2021-10-13 07:14:57 +08:00
|
|
|
%jcmp = arith.cmpi slt, %j, %arg0 : i32
|
2020-11-21 02:05:09 +08:00
|
|
|
// CHECK-NOT: call @print(%[[arg1:.+]], %[[arg1]])
|
|
|
|
call @print(%j, %ip1) : (i32, i32) -> ()
|
2022-02-04 12:59:43 +08:00
|
|
|
cf.cond_br %jcmp, ^bb7, ^bb3
|
2020-11-21 02:05:09 +08:00
|
|
|
|
|
|
|
^bb3: // pred: ^bb1
|
|
|
|
return
|
|
|
|
}
|
2022-03-21 20:26:00 +08:00
|
|
|
|
|
|
|
|
|
|
|
// CHECK-LABEL: func @mismatch_dominance(
|
2022-04-21 07:22:21 +08:00
|
|
|
func.func @mismatch_dominance() -> i32 {
|
2022-03-21 20:26:00 +08:00
|
|
|
// CHECK: %[[RES:.*]] = "test.producing_br"()
|
|
|
|
%0 = "test.producing_br"()[^bb1, ^bb2] {
|
|
|
|
operand_segment_sizes = dense<0> : vector<2 x i32>
|
|
|
|
} : () -> i32
|
|
|
|
|
|
|
|
^bb1:
|
|
|
|
// CHECK: "test.br"(%[[RES]])[^[[MERGE_BLOCK:.*]]]
|
|
|
|
"test.br"(%0)[^bb4] : (i32) -> ()
|
|
|
|
|
|
|
|
^bb2:
|
|
|
|
%1 = "foo.def"() : () -> i32
|
|
|
|
"test.br"()[^bb3] : () -> ()
|
|
|
|
|
|
|
|
^bb3:
|
|
|
|
// CHECK: "test.br"(%{{.*}})[^[[MERGE_BLOCK]]]
|
|
|
|
"test.br"(%1)[^bb4] : (i32) -> ()
|
|
|
|
|
|
|
|
^bb4(%3: i32):
|
|
|
|
return %3 : i32
|
|
|
|
}
|