|
| 1 | +// RUN: fir-opt --fir-add-alias-tags --split-input-file %s | FileCheck %s |
| 2 | + |
| 3 | +// This test demonstrates the need for fir.dummy_scope even |
| 4 | +// when a function does not have dummy arguments. |
| 5 | +// |
| 6 | +// The original source is: |
| 7 | +// module m |
| 8 | +// integer :: glob |
| 9 | +// end module m |
| 10 | +// subroutine test1 |
| 11 | +// use m |
| 12 | +// call inner(glob) |
| 13 | +// glob = 2 |
| 14 | +// contains |
| 15 | +// subroutine inner(x) |
| 16 | +// integer :: x |
| 17 | +// integer :: y |
| 18 | +// y = 1 |
| 19 | +// x = y |
| 20 | +// end subroutine inner |
| 21 | +// end subroutine test1 |
| 22 | +// |
| 23 | +// 'inner' function is manually inlined in FIR. |
| 24 | +// When fir.dummy_scope is missing, TBAA tags for glob and x |
| 25 | +// are placed into the same TBAA root. Since glob is a global |
| 26 | +// and x is a dummy argument, TBAA ends up reporting no-alias |
| 27 | +// for them, which is incorrect. |
| 28 | +func.func @_QPtest1() attributes {noinline} { |
| 29 | + %c1_i32 = arith.constant 1 : i32 |
| 30 | + %c2_i32 = arith.constant 2 : i32 |
| 31 | + %0 = fir.alloca i32 {bindc_name = "y", uniq_name = "_QFtest1FinnerEy"} |
| 32 | + %1 = fir.address_of(@_QMmEglob) : !fir.ref<i32> |
| 33 | + %2 = fir.declare %1 {uniq_name = "_QMmEglob"} : (!fir.ref<i32>) -> !fir.ref<i32> |
| 34 | + %3 = fir.dummy_scope : !fir.dscope |
| 35 | + %4 = fir.declare %2 dummy_scope %3 {uniq_name = "_QFtest1FinnerEx"} : (!fir.ref<i32>, !fir.dscope) -> !fir.ref<i32> |
| 36 | + %5 = fir.declare %0 {uniq_name = "_QFtest1FinnerEy"} : (!fir.ref<i32>) -> !fir.ref<i32> |
| 37 | + fir.store %c1_i32 to %5 : !fir.ref<i32> |
| 38 | + %6 = fir.load %5 : !fir.ref<i32> |
| 39 | + fir.store %6 to %4 : !fir.ref<i32> |
| 40 | + fir.store %c2_i32 to %2 : !fir.ref<i32> |
| 41 | + return |
| 42 | +} |
| 43 | +// CHECK: #[[$ATTR_0:.+]] = #llvm.tbaa_root<id = "Flang function root _QPtest1"> |
| 44 | +// CHECK: #[[$ATTR_1:.+]] = #llvm.tbaa_type_desc<id = "any access", members = {<#[[$ATTR_0]], 0>}> |
| 45 | +// CHECK: #[[$ATTR_2:.+]] = #llvm.tbaa_type_desc<id = "any data access", members = {<#[[$ATTR_1]], 0>}> |
| 46 | +// CHECK: #[[$ATTR_3:.+]] = #llvm.tbaa_type_desc<id = "dummy arg data", members = {<#[[$ATTR_2]], 0>}> |
| 47 | +// CHECK: #[[$ATTR_4:.+]] = #llvm.tbaa_type_desc<id = "global data", members = {<#[[$ATTR_2]], 0>}> |
| 48 | +// CHECK: #[[$ATTR_5:.+]] = #llvm.tbaa_type_desc<id = "dummy arg data/_QFtest1FinnerEx", members = {<#[[$ATTR_3]], 0>}> |
| 49 | +// CHECK: #[[$ATTR_6:.+]] = #llvm.tbaa_type_desc<id = "global data/_QMmEglob", members = {<#[[$ATTR_4]], 0>}> |
| 50 | +// CHECK: #[[$ATTR_7:.+]] = #llvm.tbaa_tag<base_type = #[[$ATTR_5]], access_type = #[[$ATTR_5]], offset = 0> |
| 51 | +// CHECK: #[[$ATTR_8:.+]] = #llvm.tbaa_tag<base_type = #[[$ATTR_6]], access_type = #[[$ATTR_6]], offset = 0> |
| 52 | +// CHECK-LABEL: func.func @_QPtest1() attributes {noinline} { |
| 53 | +// CHECK: %[[VAL_2:.*]] = fir.alloca i32 {bindc_name = "y", uniq_name = "_QFtest1FinnerEy"} |
| 54 | +// CHECK: %[[VAL_3:.*]] = fir.address_of(@_QMmEglob) : !fir.ref<i32> |
| 55 | +// CHECK: %[[VAL_4:.*]] = fir.declare %[[VAL_3]] {uniq_name = "_QMmEglob"} : (!fir.ref<i32>) -> !fir.ref<i32> |
| 56 | +// CHECK: %[[VAL_5:.*]] = fir.dummy_scope : !fir.dscope |
| 57 | +// CHECK: %[[VAL_6:.*]] = fir.declare %[[VAL_4]] dummy_scope %[[VAL_5]] {uniq_name = "_QFtest1FinnerEx"} : (!fir.ref<i32>, !fir.dscope) -> !fir.ref<i32> |
| 58 | +// CHECK: %[[VAL_7:.*]] = fir.declare %[[VAL_2]] {uniq_name = "_QFtest1FinnerEy"} : (!fir.ref<i32>) -> !fir.ref<i32> |
| 59 | +// CHECK: fir.store %{{.*}} to %[[VAL_7]] : !fir.ref<i32> |
| 60 | +// CHECK: %[[VAL_8:.*]] = fir.load %[[VAL_7]] : !fir.ref<i32> |
| 61 | +// CHECK: fir.store %[[VAL_8]] to %[[VAL_6]] {tbaa = [#[[$ATTR_7]]]} : !fir.ref<i32> |
| 62 | +// CHECK: fir.store %{{.*}} to %[[VAL_4]] {tbaa = [#[[$ATTR_8]]]} : !fir.ref<i32> |
| 63 | + |
| 64 | +// ----- |
| 65 | + |
| 66 | +// This test has fir.dummy_scope in place, and TBAA is correct. |
| 67 | +func.func @_QPtest2() attributes {noinline} { |
| 68 | + %c1_i32 = arith.constant 1 : i32 |
| 69 | + %c2_i32 = arith.constant 2 : i32 |
| 70 | + %0 = fir.alloca i32 {bindc_name = "y", uniq_name = "_QFtest2FinnerEy"} |
| 71 | + %test_dummy_scope = fir.dummy_scope : !fir.dscope |
| 72 | + %1 = fir.address_of(@_QMmEglob) : !fir.ref<i32> |
| 73 | + %2 = fir.declare %1 {uniq_name = "_QMmEglob"} : (!fir.ref<i32>) -> !fir.ref<i32> |
| 74 | + %3 = fir.dummy_scope : !fir.dscope |
| 75 | + %4 = fir.declare %2 dummy_scope %3 {uniq_name = "_QFtest2FinnerEx"} : (!fir.ref<i32>, !fir.dscope) -> !fir.ref<i32> |
| 76 | + %5 = fir.declare %0 {uniq_name = "_QFtest2FinnerEy"} : (!fir.ref<i32>) -> !fir.ref<i32> |
| 77 | + fir.store %c1_i32 to %5 : !fir.ref<i32> |
| 78 | + %6 = fir.load %5 : !fir.ref<i32> |
| 79 | + fir.store %6 to %4 : !fir.ref<i32> |
| 80 | + fir.store %c2_i32 to %2 : !fir.ref<i32> |
| 81 | + return |
| 82 | +} |
| 83 | +// CHECK: #[[$ATTR_0:.+]] = #llvm.tbaa_root<id = "Flang function root _QPtest2 - Scope 1"> |
| 84 | +// CHECK: #[[$ATTR_1:.+]] = #llvm.tbaa_root<id = "Flang function root _QPtest2"> |
| 85 | +// CHECK: #[[$ATTR_2:.+]] = #llvm.tbaa_type_desc<id = "any access", members = {<#[[$ATTR_0]], 0>}> |
| 86 | +// CHECK: #[[$ATTR_3:.+]] = #llvm.tbaa_type_desc<id = "any access", members = {<#[[$ATTR_1]], 0>}> |
| 87 | +// CHECK: #[[$ATTR_4:.+]] = #llvm.tbaa_type_desc<id = "any data access", members = {<#[[$ATTR_2]], 0>}> |
| 88 | +// CHECK: #[[$ATTR_5:.+]] = #llvm.tbaa_type_desc<id = "any data access", members = {<#[[$ATTR_3]], 0>}> |
| 89 | +// CHECK: #[[$ATTR_6:.+]] = #llvm.tbaa_type_desc<id = "dummy arg data", members = {<#[[$ATTR_4]], 0>}> |
| 90 | +// CHECK: #[[$ATTR_7:.+]] = #llvm.tbaa_type_desc<id = "global data", members = {<#[[$ATTR_5]], 0>}> |
| 91 | +// CHECK: #[[$ATTR_8:.+]] = #llvm.tbaa_type_desc<id = "dummy arg data/_QFtest2FinnerEx", members = {<#[[$ATTR_6]], 0>}> |
| 92 | +// CHECK: #[[$ATTR_9:.+]] = #llvm.tbaa_type_desc<id = "global data/_QMmEglob", members = {<#[[$ATTR_7]], 0>}> |
| 93 | +// CHECK: #[[$ATTR_10:.+]] = #llvm.tbaa_tag<base_type = #[[$ATTR_8]], access_type = #[[$ATTR_8]], offset = 0> |
| 94 | +// CHECK: #[[$ATTR_11:.+]] = #llvm.tbaa_tag<base_type = #[[$ATTR_9]], access_type = #[[$ATTR_9]], offset = 0> |
| 95 | +// CHECK-LABEL: func.func @_QPtest2() attributes {noinline} { |
| 96 | +// CHECK: %[[VAL_2:.*]] = fir.alloca i32 {bindc_name = "y", uniq_name = "_QFtest2FinnerEy"} |
| 97 | +// CHECK: %[[VAL_3:.*]] = fir.dummy_scope : !fir.dscope |
| 98 | +// CHECK: %[[VAL_4:.*]] = fir.address_of(@_QMmEglob) : !fir.ref<i32> |
| 99 | +// CHECK: %[[VAL_5:.*]] = fir.declare %[[VAL_4]] {uniq_name = "_QMmEglob"} : (!fir.ref<i32>) -> !fir.ref<i32> |
| 100 | +// CHECK: %[[VAL_6:.*]] = fir.dummy_scope : !fir.dscope |
| 101 | +// CHECK: %[[VAL_7:.*]] = fir.declare %[[VAL_5]] dummy_scope %[[VAL_6]] {uniq_name = "_QFtest2FinnerEx"} : (!fir.ref<i32>, !fir.dscope) -> !fir.ref<i32> |
| 102 | +// CHECK: %[[VAL_8:.*]] = fir.declare %[[VAL_2]] {uniq_name = "_QFtest2FinnerEy"} : (!fir.ref<i32>) -> !fir.ref<i32> |
| 103 | +// CHECK: fir.store %{{.*}} to %[[VAL_8]] : !fir.ref<i32> |
| 104 | +// CHECK: %[[VAL_9:.*]] = fir.load %[[VAL_8]] : !fir.ref<i32> |
| 105 | +// CHECK: fir.store %[[VAL_9]] to %[[VAL_7]] {tbaa = [#[[$ATTR_10]]]} : !fir.ref<i32> |
| 106 | +// CHECK: fir.store %{{.*}} to %[[VAL_5]] {tbaa = [#[[$ATTR_11]]]} : !fir.ref<i32> |
0 commit comments