Files
clang-p2996/mlir/test/Conversion/ControlFlowToLLVM/switch.mlir
Matthias Springer eb6c4197d5 [mlir][CF] Split cf-to-llvm from func-to-llvm (#120580)
Do not run `cf-to-llvm` as part of `func-to-llvm`. This commit fixes
https://github.com/llvm/llvm-project/issues/70982.

This commit changes the way how `func.func` ops are lowered to LLVM.
Previously, the signature of the entire region (i.e., entry block and
all other blocks in the `func.func` op) was converted as part of the
`func.func` lowering pattern.

Now, only the entry block is converted. The remaining block signatures
are converted together with `cf.br` and `cf.cond_br` as part of
`cf-to-llvm`. All unstructured control flow is not converted as part of
a single pass (`cf-to-llvm`). `func-to-llvm` no longer deals with
unstructured control flow.

Also add more test cases for control flow dialect ops.

Note: This PR is in preparation of #120431, which adds an additional
GPU-specific lowering for `cf.assert`. This was a problem because
`cf.assert` used to be converted as part of `func-to-llvm`.

Note for LLVM integration: If you see failures, add
`-convert-cf-to-llvm` to your pass pipeline.
2024-12-20 13:46:45 +01:00

67 lines
2.1 KiB
MLIR

// RUN: mlir-opt %s -convert-cf-to-llvm -split-input-file | FileCheck %s
// Unstructured control flow is converted, but the enclosing op is not
// converted.
// CHECK-LABEL: func.func @single_case(
// CHECK-SAME: %[[val:.*]]: i32, %[[idx:.*]]: index) -> index {
// CHECK: %[[cast0:.*]] = builtin.unrealized_conversion_cast %[[idx]] : index to i64
// CHECK: llvm.switch %[[val]] : i32, ^[[bb1:.*]](%[[cast0]] : i64) [
// CHECK: ]
// CHECK: ^[[bb1]](%[[arg0:.*]]: i64):
// CHECK: %[[cast1:.*]] = builtin.unrealized_conversion_cast %[[arg0]] : i64 to index
// CHECK: return %[[cast1]] : index
// CHECK: }
func.func @single_case(%val: i32, %idx: index) -> index {
cf.switch %val : i32, [
default: ^bb1(%idx : index)
]
^bb1(%arg0: index):
return %arg0 : index
}
// -----
// func.func and func.return types match. No unrealized_conversion_cast is
// needed.
// CHECK-LABEL: func.func @single_case_type_match(
// CHECK-SAME: %[[val:.*]]: i32, %[[i:.*]]: i64) -> i64 {
// CHECK: llvm.switch %[[val]] : i32, ^[[bb1:.*]](%[[i]] : i64) [
// CHECK: ]
// CHECK: ^[[bb1]](%[[arg0:.*]]: i64):
// CHECK: return %[[arg0]] : i64
// CHECK: }
func.func @single_case_type_match(%val: i32, %i: i64) -> i64 {
cf.switch %val : i32, [
default: ^bb1(%i : i64)
]
^bb1(%arg0: i64):
return %arg0 : i64
}
// -----
// CHECK-LABEL: func.func @multi_case
// CHECK-COUNT-2: unrealized_conversion_cast {{.*}} : index to i64
// CHECK: llvm.switch %{{.*}} : i32, ^{{.*}}(%{{.*}} : i64) [
// CHECK: 12: ^{{.*}}(%{{.*}} : i64),
// CHECK: 13: ^{{.*}}(%{{.*}} : i64),
// CHECK: 14: ^{{.*}}(%{{.*}} : i64)
// CHECK: ]
func.func @multi_case(%val: i32, %idx1: index, %idx2: index, %i: i64) -> index {
cf.switch %val : i32, [
default: ^bb1(%idx1 : index),
12: ^bb2(%idx2 : index),
13: ^bb1(%idx1 : index),
14: ^bb3(%i : i64)
]
^bb1(%arg0: index):
return %arg0 : index
^bb2(%arg1: index):
return %arg1 : index
^bb3(%arg2: i64):
%cast = arith.index_cast %arg2 : i64 to index
return %cast : index
}