// RUN: mlir-opt %s --pass-pipeline='builtin.module(func.func(mem2reg{region-simplify=false}))' --split-input-file | FileCheck %s // CHECK-LABEL: func.func @basic func.func @basic() -> i32 { // CHECK-NOT: = memref.alloca // CHECK: %[[RES:.*]] = arith.constant 5 : i32 // CHECK-NOT: = memref.alloca %0 = arith.constant 5 : i32 %1 = memref.alloca() : memref memref.store %0, %1[] : memref %2 = memref.load %1[] : memref // CHECK: return %[[RES]] : i32 return %2 : i32 } // ----- // CHECK-LABEL: func.func @basic_default func.func @basic_default() -> i32 { // CHECK-NOT: = memref.alloca // CHECK: %[[RES:.*]] = arith.constant 0 : i32 // CHECK-NOT: = memref.alloca %0 = arith.constant 5 : i32 %1 = memref.alloca() : memref %2 = memref.load %1[] : memref // CHECK-NOT: memref.store memref.store %0, %1[] : memref // CHECK: return %[[RES]] : i32 return %2 : i32 } // ----- // CHECK-LABEL: func.func @basic_float func.func @basic_float() -> f32 { // CHECK-NOT: = memref.alloca // CHECK: %[[RES:.*]] = arith.constant {{.*}} : f32 %0 = arith.constant 5.2 : f32 // CHECK-NOT: = memref.alloca %1 = memref.alloca() : memref memref.store %0, %1[] : memref %2 = memref.load %1[] : memref // CHECK: return %[[RES]] : f32 return %2 : f32 } // ----- // CHECK-LABEL: func.func @basic_ranked func.func @basic_ranked() -> i32 { // CHECK-NOT: = memref.alloca // CHECK: %[[RES:.*]] = arith.constant 5 : i32 // CHECK-NOT: = memref.alloca %0 = arith.constant 0 : index %1 = arith.constant 5 : i32 %2 = memref.alloca() : memref<1x1xi32> memref.store %1, %2[%0, %0] : memref<1x1xi32> %3 = memref.load %2[%0, %0] : memref<1x1xi32> // CHECK: return %[[RES]] : i32 return %3 : i32 } // ----- // CHECK-LABEL: func.func @reject_multiple_elements func.func @reject_multiple_elements() -> i32 { // CHECK: %[[INDEX:.*]] = arith.constant 0 : index %0 = arith.constant 0 : index // CHECK: %[[STORED:.*]] = arith.constant 5 : i32 %1 = arith.constant 5 : i32 // CHECK: %[[ALLOCA:.*]] = memref.alloca() %2 = memref.alloca() : memref<1x2xi32> // CHECK: memref.store %[[STORED]], %[[ALLOCA]][%[[INDEX]], %[[INDEX]]] memref.store %1, %2[%0, %0] : memref<1x2xi32> // CHECK: %[[RES:.*]] = memref.load %[[ALLOCA]][%[[INDEX]], %[[INDEX]]] %3 = memref.load %2[%0, %0] : memref<1x2xi32> // CHECK: return %[[RES]] : i32 return %3 : i32 } // ----- // CHECK-LABEL: func.func @cycle // CHECK-SAME: (%[[ARG0:.*]]: i64, %[[ARG1:.*]]: i1, %[[ARG2:.*]]: i64) func.func @cycle(%arg0: i64, %arg1: i1, %arg2: i64) { // CHECK-NOT: = memref.alloca %alloca = memref.alloca() : memref memref.store %arg2, %alloca[] : memref // CHECK: cf.cond_br %[[ARG1:.*]], ^[[BB1:.*]](%[[ARG2]] : i64), ^[[BB2:.*]](%[[ARG2]] : i64) cf.cond_br %arg1, ^bb1, ^bb2 // CHECK: ^[[BB1]](%[[USE:.*]]: i64): ^bb1: %use = memref.load %alloca[] : memref // CHECK: call @use(%[[USE]]) func.call @use(%use) : (i64) -> () memref.store %arg0, %alloca[] : memref // CHECK: cf.br ^[[BB2]](%[[ARG0]] : i64) cf.br ^bb2 // CHECK: ^[[BB2]](%[[FWD:.*]]: i64): ^bb2: // CHECK: cf.br ^[[BB1]](%[[FWD]] : i64) cf.br ^bb1 } func.func @use(%arg: i64) { return } // ----- // CHECK-LABEL: func.func @recursive // CHECK-SAME: (%[[ARG:.*]]: i64) func.func @recursive(%arg: i64) -> i64 { // CHECK-NOT: = memref.alloca() %alloca0 = memref.alloca() : memref>> %alloca1 = memref.alloca() : memref> %alloca2 = memref.alloca() : memref memref.store %arg, %alloca2[] : memref memref.store %alloca2, %alloca1[] : memref> memref.store %alloca1, %alloca0[] : memref>> %load0 = memref.load %alloca0[] : memref>> %load1 = memref.load %load0[] : memref> %load2 = memref.load %load1[] : memref // CHECK: return %[[ARG]] : i64 return %load2 : i64 } // ----- // CHECK-LABEL: func.func @deny_store_of_alloca // CHECK-SAME: (%[[ARG:.*]]: memref>) func.func @deny_store_of_alloca(%arg: memref>) -> i32 { // CHECK: %[[VALUE:.*]] = arith.constant 5 : i32 %0 = arith.constant 5 : i32 // CHECK: %[[ALLOCA:.*]] = memref.alloca %1 = memref.alloca() : memref // Storing into the memref is allowed. // CHECK: memref.store %[[VALUE]], %[[ALLOCA]][] memref.store %0, %1[] : memref // Storing the memref itself is NOT allowed. // CHECK: memref.store %[[ALLOCA]], %[[ARG]][] memref.store %1, %arg[] : memref> // CHECK: %[[RES:.*]] = memref.load %[[ALLOCA]][] %2 = memref.load %1[] : memref // CHECK: return %[[RES]] : i32 return %2 : i32 } // ----- // CHECK-LABEL: func.func @promotable_nonpromotable_intertwined func.func @promotable_nonpromotable_intertwined() -> i32 { // CHECK: %[[NON_PROMOTED:.*]] = memref.alloca() : memref %0 = memref.alloca() : memref // CHECK-NOT: = memref.alloca() : memref> %1 = memref.alloca() : memref> memref.store %0, %1[] : memref> %2 = memref.load %1[] : memref> // CHECK: call @use(%[[NON_PROMOTED]]) call @use(%0) : (memref) -> () // CHECK: %[[RES:.*]] = memref.load %[[NON_PROMOTED]][] %3 = memref.load %0[] : memref // CHECK: return %[[RES]] : i32 return %3 : i32 } func.func @use(%arg: memref) { return }