Constants actual arguments in function/subroutine calls are currently lowered as allocas + store. This can sometimes inhibit LTO and the constant will not be propagated to the called function. Particularly in cases where the function/subroutine call happens inside a condition. This patch changes the lowering of these constant actual arguments to a global constant + fir.address_of_op. This lowering makes it easier for LTO to propagate the constant. The optimization must be enabled explicitly to run. Use -mmlir --enable-constant-argument-globalisation to enable. --------- Co-authored-by: Dmitriy Smirnov <dmitriy.smirnov@arm.com>
99 lines
2.8 KiB
Plaintext
99 lines
2.8 KiB
Plaintext
// 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
|