// RUN: fir-opt --split-input-file --constant-argument-globalisation-opt < %s | FileCheck %s
module {
// Test for "two conditional writes to the same alloca doesn't get replaced."
func.func @func(%arg0: i32, %arg1: i1) {
%c2_i32 = arith.constant 2 : i32
%addr = fir.alloca i32 {adapt.valuebyref}
fir.if %arg1 {
fir.store %c2_i32 to %addr : !fir.ref<i32>
} else {
fir.store %arg0 to %addr : !fir.ref<i32>
}
fir.call @sub2(%addr) : (!fir.ref<i32>) -> ()
return
}
func.func private @sub2(!fir.ref<i32>)
// CHECK-LABEL: func.func @func
// CHECK-SAME: [[ARG0:%.*]]: i32
// CHECK-SAME: [[ARG1:%.*]]: i1)
// CHECK: [[CONST:%.*]] = arith.constant
// CHECK: [[ADDR:%.*]] = fir.alloca i32
// CHECK: fir.if [[ARG1]]
// CHECK: fir.store [[CONST]] to [[ADDR]]
// CHECK: } else {
// CHECK: fir.store [[ARG0]] to [[ADDR]]
// CHECK: fir.call @sub2([[ADDR]])
// CHECK: return
}
// -----
module {
// Test for "two writes to the same alloca doesn't get replaced."
func.func @func() {
%c1_i32 = arith.constant 1 : i32
%c2_i32 = arith.constant 2 : i32
%addr = fir.alloca i32 {adapt.valuebyref}
fir.store %c1_i32 to %addr : !fir.ref<i32>
fir.store %c2_i32 to %addr : !fir.ref<i32>
fir.call @sub2(%addr) : (!fir.ref<i32>) -> ()
return
}
func.func private @sub2(!fir.ref<i32>)
// CHECK-LABEL: func.func @func
// CHECK: [[CONST1:%.*]] = arith.constant
// CHECK: [[CONST2:%.*]] = arith.constant
// CHECK: [[ADDR:%.*]] = fir.alloca i32
// CHECK: fir.store [[CONST1]] to [[ADDR]]
// CHECK: fir.store [[CONST2]] to [[ADDR]]
// CHECK: fir.call @sub2([[ADDR]])
// CHECK: return
}
// -----
module {
// Test for "one write to the the alloca gets replaced."
func.func @func() {
%c1_i32 = arith.constant 1 : i32
%addr = fir.alloca i32 {adapt.valuebyref}
fir.store %c1_i32 to %addr : !fir.ref<i32>
fir.call @sub2(%addr) : (!fir.ref<i32>) -> ()
return
}
func.func private @sub2(!fir.ref<i32>)
// CHECK-LABEL: func.func @func
// CHECK: [[ADDR:%.*]] = fir.address_of([[EXTR:@.*]]) : !fir.ref<i32>
// CHECK: fir.call @sub2([[ADDR]])
// CHECK: return
// CHECK: fir.global internal [[EXTR]] constant : i32 {
// CHECK: %{{.*}} = arith.constant 1 : i32
// CHECK: fir.has_value %{{.*}} : i32
// CHECK: }
}
// -----
// Check that same argument used twice is converted.
module {
func.func @func(%arg0: !fir.ref<i32>, %arg1: i1) {
%c2_i32 = arith.constant 2 : i32
%addr1 = fir.alloca i32 {adapt.valuebyref}
fir.store %c2_i32 to %addr1 : !fir.ref<i32>
fir.call @sub1(%addr1, %addr1) : (!fir.ref<i32>, !fir.ref<i32>) -> ()
return
}
}
// CHECK-LABEL: func.func @func
// CHECK-NEXT: %[[ARG1:.*]] = fir.address_of([[CONST1:@.*]]) : !fir.ref<i32>
// CHECK-NEXT: %[[ARG2:.*]] = fir.address_of([[CONST2:@.*]]) : !fir.ref<i32>
// CHECK-NEXT: fir.call @sub1(%[[ARG1]], %[[ARG2]])
// CHECK-NEXT: return