
[RFC on discourse](https://discourse.llvm.org/t/rfc-volatile-representation-in-flang/85404/1) Flang currently lacks support for volatile variables. For some cases, the compiler produces TODO error messages and others are ignored. Some of our tests are like the example from _C.4 Clause 8 notes: The VOLATILE attribute (8.5.20)_ and require volatile variables. Prior commits: ``` c9ec1bc753b0 [flang] Handle volatility in lowering and codegen (#135311) e42f8609858f [flang][nfc] Support volatility in Fir ops (#134858) b2711e1526f9 [flang][nfc] Support volatile on ref, box, and class types (#134386) ```
114 lines
7.3 KiB
Plaintext
114 lines
7.3 KiB
Plaintext
// RUN: fir-opt --cse -split-input-file %s | FileCheck %s
|
|
|
|
// Check that the redundant fir.load is removed.
|
|
func.func @fun(%arg0: !fir.ref<i64>) -> i64 {
|
|
%0 = fir.load %arg0 : !fir.ref<i64>
|
|
%1 = fir.load %arg0 : !fir.ref<i64>
|
|
%2 = arith.addi %0, %1 : i64
|
|
return %2 : i64
|
|
}
|
|
|
|
// CHECK-LABEL: func @fun
|
|
// CHECK-NEXT: %[[LOAD:.*]] = fir.load %{{.*}} : !fir.ref<i64>
|
|
// CHECK-NEXT: %{{.*}} = arith.addi %[[LOAD]], %[[LOAD]] : i64
|
|
|
|
// -----
|
|
|
|
// CHECK-LABEL: func @fun(
|
|
// CHECK-SAME: %[[A:.*]]: !fir.ref<i64>
|
|
func.func @fun(%a : !fir.ref<i64>) -> i64 {
|
|
// CHECK: %[[LOAD:.*]] = fir.load %[[A]] : !fir.ref<i64>
|
|
%1 = fir.load %a : !fir.ref<i64>
|
|
%2 = fir.load %a : !fir.ref<i64>
|
|
// CHECK-NEXT: %{{.*}} = arith.addi %[[LOAD]], %[[LOAD]] : i64
|
|
%3 = arith.addi %1, %2 : i64
|
|
%4 = fir.load %a : !fir.ref<i64>
|
|
// CHECK-NEXT: %{{.*}} = arith.addi
|
|
%5 = arith.addi %3, %4 : i64
|
|
%6 = fir.load %a : !fir.ref<i64>
|
|
// CHECK-NEXT: %{{.*}} = arith.addi
|
|
%7 = arith.addi %5, %6 : i64
|
|
%8 = fir.load %a : !fir.ref<i64>
|
|
// CHECK-NEXT: %{{.*}} = arith.addi
|
|
%9 = arith.addi %7, %8 : i64
|
|
%10 = fir.load %a : !fir.ref<i64>
|
|
// CHECK-NEXT: %{{.*}} = arith.addi
|
|
%11 = arith.addi %10, %9 : i64
|
|
%12 = fir.load %a : !fir.ref<i64>
|
|
// CHECK-NEXT: %{{.*}} = arith.addi
|
|
%13 = arith.addi %11, %12 : i64
|
|
// CHECK-NEXT: return %{{.*}} : i64
|
|
return %13 : i64
|
|
}
|
|
|
|
// -----
|
|
|
|
func.func @fun(%a : !fir.ref<i64>) -> i64 {
|
|
cf.br ^bb1
|
|
^bb1:
|
|
%1 = fir.load %a : !fir.ref<i64>
|
|
%2 = fir.load %a : !fir.ref<i64>
|
|
%3 = arith.addi %1, %2 : i64
|
|
cf.br ^bb2
|
|
^bb2:
|
|
%4 = fir.load %a : !fir.ref<i64>
|
|
%5 = arith.subi %4, %4 : i64
|
|
return %5 : i64
|
|
}
|
|
|
|
// -----
|
|
|
|
// Check that the redundant ops on volatile operands are PRESERVED.
|
|
func.func @fun(%arg0: !fir.ref<i64, volatile>) -> i64 {
|
|
%0 = fir.load %arg0 : !fir.ref<i64, volatile>
|
|
%1 = fir.load %arg0 : !fir.ref<i64, volatile>
|
|
%2 = arith.addi %0, %1 : i64
|
|
fir.store %2 to %arg0 : !fir.ref<i64, volatile>
|
|
fir.store %2 to %arg0 : !fir.ref<i64, volatile>
|
|
return %2 : i64
|
|
}
|
|
// CHECK-LABEL: func.func @fun(%arg0: !fir.ref<i64, volatile>) -> i64 {
|
|
// CHECK: %[[VAL_1:.*]] = fir.load %arg0 : !fir.ref<i64, volatile>
|
|
// CHECK: %[[VAL_2:.*]] = fir.load %arg0 : !fir.ref<i64, volatile>
|
|
// CHECK: %[[VAL_3:.*]] = arith.addi %[[VAL_1]], %[[VAL_2]] : i64
|
|
// CHECK: fir.store %[[VAL_3]] to %arg0 : !fir.ref<i64, volatile>
|
|
// CHECK: fir.store %[[VAL_3]] to %arg0 : !fir.ref<i64, volatile>
|
|
// CHECK: return %[[VAL_3]] : i64
|
|
// CHECK: }
|
|
|
|
// -----
|
|
|
|
// Check that volatile hlfir assignments are PRESERVED.
|
|
func.func @_QPdot_product2(%arg0: !fir.box<!fir.array<?x!fir.logical<4>>> {fir.bindc_name = "lhs"}, %arg1: !fir.box<!fir.array<?x!fir.logical<4>>> {fir.bindc_name = "rhs"}, %arg2: !fir.ref<!fir.logical<4>> {fir.bindc_name = "res"}) {
|
|
%0 = fir.dummy_scope : !fir.dscope
|
|
%1 = fir.volatile_cast %arg0 : (!fir.box<!fir.array<?x!fir.logical<4>>>) -> !fir.box<!fir.array<?x!fir.logical<4>>, volatile>
|
|
%2:2 = hlfir.declare %1 dummy_scope %0 {fortran_attrs = #fir.var_attrs<volatile>, uniq_name = "_QFdot_product2Elhs"} : (!fir.box<!fir.array<?x!fir.logical<4>>, volatile>, !fir.dscope) -> (!fir.box<!fir.array<?x!fir.logical<4>>, volatile>, !fir.box<!fir.array<?x!fir.logical<4>>, volatile>)
|
|
%3 = fir.volatile_cast %arg2 : (!fir.ref<!fir.logical<4>>) -> !fir.ref<!fir.logical<4>, volatile>
|
|
%4:2 = hlfir.declare %3 dummy_scope %0 {fortran_attrs = #fir.var_attrs<volatile>, uniq_name = "_QFdot_product2Eres"} : (!fir.ref<!fir.logical<4>, volatile>, !fir.dscope) -> (!fir.ref<!fir.logical<4>, volatile>, !fir.ref<!fir.logical<4>, volatile>)
|
|
%5 = fir.volatile_cast %arg1 : (!fir.box<!fir.array<?x!fir.logical<4>>>) -> !fir.box<!fir.array<?x!fir.logical<4>>, volatile>
|
|
%6:2 = hlfir.declare %5 dummy_scope %0 {fortran_attrs = #fir.var_attrs<volatile>, uniq_name = "_QFdot_product2Erhs"} : (!fir.box<!fir.array<?x!fir.logical<4>>, volatile>, !fir.dscope) -> (!fir.box<!fir.array<?x!fir.logical<4>>, volatile>, !fir.box<!fir.array<?x!fir.logical<4>>, volatile>)
|
|
%7 = hlfir.dot_product %2#0 %6#0 {fastmath = #arith.fastmath<contract>} : (!fir.box<!fir.array<?x!fir.logical<4>>, volatile>, !fir.box<!fir.array<?x!fir.logical<4>>, volatile>) -> !fir.logical<4>
|
|
hlfir.assign %7 to %4#0 : !fir.logical<4>, !fir.ref<!fir.logical<4>, volatile>
|
|
%8 = hlfir.dot_product %2#0 %6#0 {fastmath = #arith.fastmath<contract>} : (!fir.box<!fir.array<?x!fir.logical<4>>, volatile>, !fir.box<!fir.array<?x!fir.logical<4>>, volatile>) -> !fir.logical<4>
|
|
hlfir.assign %8 to %4#0 : !fir.logical<4>, !fir.ref<!fir.logical<4>, volatile>
|
|
return
|
|
}
|
|
|
|
// CHECK-LABEL: func.func @_QPdot_product2(
|
|
// CHECK-SAME: %[[VAL_0:[0-9]+|[a-zA-Z$._-][a-zA-Z0-9$._-]*]]: !fir.box<!fir.array<?x!fir.logical<4>>> {fir.bindc_name = "lhs"},
|
|
// CHECK-SAME: %[[VAL_1:[0-9]+|[a-zA-Z$._-][a-zA-Z0-9$._-]*]]: !fir.box<!fir.array<?x!fir.logical<4>>> {fir.bindc_name = "rhs"},
|
|
// CHECK-SAME: %[[VAL_2:[0-9]+|[a-zA-Z$._-][a-zA-Z0-9$._-]*]]: !fir.ref<!fir.logical<4>> {fir.bindc_name = "res"}) {
|
|
// CHECK: %[[VAL_3:.*]] = fir.dummy_scope : !fir.dscope
|
|
// CHECK: %[[VAL_4:.*]] = fir.volatile_cast %[[VAL_0]] : (!fir.box<!fir.array<?x!fir.logical<4>>>) -> !fir.box<!fir.array<?x!fir.logical<4>>, volatile>
|
|
// CHECK: %[[VAL_5:.*]]:2 = hlfir.declare %[[VAL_4]] dummy_scope %[[VAL_3]] {fortran_attrs = #fir.var_attrs<volatile>, uniq_name = "_QFdot_product2Elhs"} : (!fir.box<!fir.array<?x!fir.logical<4>>, volatile>, !fir.dscope) -> (!fir.box<!fir.array<?x!fir.logical<4>>, volatile>, !fir.box<!fir.array<?x!fir.logical<4>>, volatile>)
|
|
// CHECK: %[[VAL_6:.*]] = fir.volatile_cast %[[VAL_2]] : (!fir.ref<!fir.logical<4>>) -> !fir.ref<!fir.logical<4>, volatile>
|
|
// CHECK: %[[VAL_7:.*]]:2 = hlfir.declare %[[VAL_6]] dummy_scope %[[VAL_3]] {fortran_attrs = #fir.var_attrs<volatile>, uniq_name = "_QFdot_product2Eres"} : (!fir.ref<!fir.logical<4>, volatile>, !fir.dscope) -> (!fir.ref<!fir.logical<4>, volatile>, !fir.ref<!fir.logical<4>, volatile>)
|
|
// CHECK: %[[VAL_8:.*]] = fir.volatile_cast %[[VAL_1]] : (!fir.box<!fir.array<?x!fir.logical<4>>>) -> !fir.box<!fir.array<?x!fir.logical<4>>, volatile>
|
|
// CHECK: %[[VAL_9:.*]]:2 = hlfir.declare %[[VAL_8]] dummy_scope %[[VAL_3]] {fortran_attrs = #fir.var_attrs<volatile>, uniq_name = "_QFdot_product2Erhs"} : (!fir.box<!fir.array<?x!fir.logical<4>>, volatile>, !fir.dscope) -> (!fir.box<!fir.array<?x!fir.logical<4>>, volatile>, !fir.box<!fir.array<?x!fir.logical<4>>, volatile>)
|
|
// CHECK: %[[VAL_10:.*]] = hlfir.dot_product %[[VAL_5]]#0 %[[VAL_9]]#0 {fastmath = #arith.fastmath<contract>} : (!fir.box<!fir.array<?x!fir.logical<4>>, volatile>, !fir.box<!fir.array<?x!fir.logical<4>>, volatile>) -> !fir.logical<4>
|
|
// CHECK: hlfir.assign %[[VAL_10]] to %[[VAL_7]]#0 : !fir.logical<4>, !fir.ref<!fir.logical<4>, volatile>
|
|
// CHECK: %[[VAL_11:.*]] = hlfir.dot_product %[[VAL_5]]#0 %[[VAL_9]]#0 {fastmath = #arith.fastmath<contract>} : (!fir.box<!fir.array<?x!fir.logical<4>>, volatile>, !fir.box<!fir.array<?x!fir.logical<4>>, volatile>) -> !fir.logical<4>
|
|
// CHECK: hlfir.assign %[[VAL_11]] to %[[VAL_7]]#0 : !fir.logical<4>, !fir.ref<!fir.logical<4>, volatile>
|
|
// CHECK: return
|
|
// CHECK: }
|