|
| 1 | +// RUN: fir-opt %s --hlfir-expression-simplification | FileCheck %s |
| 2 | + |
| 3 | +// Test removal of trim() calls. |
| 4 | + |
| 5 | +// logical function test_char_cmp(x, y) result(cmp) |
| 6 | +// character(*) :: x, y |
| 7 | +// cmp = trim(x) == trim(y) |
| 8 | +// end function |
| 9 | + |
| 10 | +func.func @_QPtest_char_cmp(%arg0: !fir.boxchar<1> {fir.bindc_name = "x"}, |
| 11 | + %arg1: !fir.boxchar<1> {fir.bindc_name = "y"}) -> !fir.logical<4> { |
| 12 | + %0 = fir.dummy_scope : !fir.dscope |
| 13 | + %1 = fir.alloca !fir.logical<4> {bindc_name = "cmp", uniq_name = "_QFtest_char_cmpEcmp"} |
| 14 | + %2:2 = hlfir.declare %1 {uniq_name = "_QFtest_char_cmpEcmp"} : (!fir.ref<!fir.logical<4>>) -> (!fir.ref<!fir.logical<4>>, !fir.ref<!fir.logical<4>>) |
| 15 | + %3:2 = fir.unboxchar %arg0 : (!fir.boxchar<1>) -> (!fir.ref<!fir.char<1,?>>, index) |
| 16 | + %4:2 = hlfir.declare %3#0 typeparams %3#1 dummy_scope %0 {uniq_name = "_QFtest_char_cmpEx"} : (!fir.ref<!fir.char<1,?>>, index, !fir.dscope) -> (!fir.boxchar<1>, !fir.ref<!fir.char<1,?>>) |
| 17 | + %5:2 = fir.unboxchar %arg1 : (!fir.boxchar<1>) -> (!fir.ref<!fir.char<1,?>>, index) |
| 18 | + %6:2 = hlfir.declare %5#0 typeparams %5#1 dummy_scope %0 {uniq_name = "_QFtest_char_cmpEy"} : (!fir.ref<!fir.char<1,?>>, index, !fir.dscope) -> (!fir.boxchar<1>, !fir.ref<!fir.char<1,?>>) |
| 19 | + %7 = hlfir.char_trim %4#0 : (!fir.boxchar<1>) -> !hlfir.expr<!fir.char<1,?>> |
| 20 | + %8 = hlfir.char_trim %6#0 : (!fir.boxchar<1>) -> !hlfir.expr<!fir.char<1,?>> |
| 21 | + %9 = hlfir.cmpchar eq %7 %8 : (!hlfir.expr<!fir.char<1,?>>, !hlfir.expr<!fir.char<1,?>>) -> i1 |
| 22 | + %10 = fir.convert %9 : (i1) -> !fir.logical<4> |
| 23 | + hlfir.assign %10 to %2#0 : !fir.logical<4>, !fir.ref<!fir.logical<4>> |
| 24 | + hlfir.destroy %8 : !hlfir.expr<!fir.char<1,?>> |
| 25 | + hlfir.destroy %7 : !hlfir.expr<!fir.char<1,?>> |
| 26 | + %11 = fir.load %2#0 : !fir.ref<!fir.logical<4>> |
| 27 | + return %11 : !fir.logical<4> |
| 28 | +} |
| 29 | + |
| 30 | +// CHECK-LABEL: func.func @_QPtest_char_cmp( |
| 31 | +// CHECK-SAME: %[[ARG_0:.*]]: !fir.boxchar<1> {fir.bindc_name = "x"}, |
| 32 | +// CHECK-SAME: %[[ARG_1:.*]]: !fir.boxchar<1> {fir.bindc_name = "y"}) -> !fir.logical<4> { |
| 33 | +// CHECK: %[[VAL_0:.*]] = fir.dummy_scope : !fir.dscope |
| 34 | +// CHECK: %[[VAL_1:.*]] = fir.alloca !fir.logical<4> {bindc_name = "cmp", uniq_name = "_QFtest_char_cmpEcmp"} |
| 35 | +// CHECK: %[[VAL_2:.*]]:2 = hlfir.declare %[[VAL_1]] {uniq_name = "_QFtest_char_cmpEcmp"} : (!fir.ref<!fir.logical<4>>) -> (!fir.ref<!fir.logical<4>>, !fir.ref<!fir.logical<4>>) |
| 36 | +// CHECK: %[[VAL_3:.*]]:2 = fir.unboxchar %[[ARG_0]] : (!fir.boxchar<1>) -> (!fir.ref<!fir.char<1,?>>, index) |
| 37 | +// CHECK: %[[VAL_5:.*]]:2 = hlfir.declare %[[VAL_3]]#0 typeparams %[[VAL_3]]#1 dummy_scope %[[VAL_0]] {uniq_name = "_QFtest_char_cmpEx"} : (!fir.ref<!fir.char<1,?>>, index, !fir.dscope) -> (!fir.boxchar<1>, !fir.ref<!fir.char<1,?>>) |
| 38 | +// CHECK: %[[VAL_6:.*]]:2 = fir.unboxchar %[[ARG_1]] : (!fir.boxchar<1>) -> (!fir.ref<!fir.char<1,?>>, index) |
| 39 | +// CHECK: %[[VAL_8:.*]]:2 = hlfir.declare %[[VAL_6]]#0 typeparams %[[VAL_6]]#1 dummy_scope %[[VAL_0]] {uniq_name = "_QFtest_char_cmpEy"} : (!fir.ref<!fir.char<1,?>>, index, !fir.dscope) -> (!fir.boxchar<1>, !fir.ref<!fir.char<1,?>>) |
| 40 | +// CHECK: %[[VAL_9:.*]] = hlfir.cmpchar eq %[[VAL_5]]#0 %[[VAL_8]]#0 : (!fir.boxchar<1>, !fir.boxchar<1>) -> i1 |
| 41 | +// CHECK: %[[VAL_10:.*]] = fir.convert %[[VAL_9]] : (i1) -> !fir.logical<4> |
| 42 | +// CHECK: hlfir.assign %[[VAL_10]] to %[[VAL_2]]#0 : !fir.logical<4>, !fir.ref<!fir.logical<4>> |
| 43 | +// CHECK: %[[VAL_11:.*]] = fir.load %[[VAL_2]]#0 : !fir.ref<!fir.logical<4>> |
| 44 | +// CHECK: return %[[VAL_11]] : !fir.logical<4> |
| 45 | +// CHECK: } |
| 46 | + |
| 47 | +// Check that trim() is not removed when its result is stored. |
| 48 | + |
| 49 | +// logical function test_char_cmp2(x, y) result(res) |
| 50 | +// character(*) :: x, y |
| 51 | +// character(:), allocatable :: tx |
| 52 | +// |
| 53 | +// tx = trim(x) |
| 54 | +// res = tx == y |
| 55 | +// end function |
| 56 | + |
| 57 | +func.func @_QPtest_char_cmp2(%arg0: !fir.boxchar<1> {fir.bindc_name = "x"}, %arg1: !fir.boxchar<1> {fir.bindc_name = "y"}) -> !fir.logical<4> { |
| 58 | + %0 = fir.dummy_scope : !fir.dscope |
| 59 | + %1 = fir.alloca !fir.logical<4> {bindc_name = "res", uniq_name = "_QFtest_char_cmp2Eres"} |
| 60 | + %2:2 = hlfir.declare %1 {uniq_name = "_QFtest_char_cmp2Eres"} : (!fir.ref<!fir.logical<4>>) -> (!fir.ref<!fir.logical<4>>, !fir.ref<!fir.logical<4>>) |
| 61 | + %3 = fir.alloca !fir.box<!fir.heap<!fir.char<1,?>>> {bindc_name = "tx", uniq_name = "_QFtest_char_cmp2Etx"} |
| 62 | + %4 = fir.zero_bits !fir.heap<!fir.char<1,?>> |
| 63 | + %c0 = arith.constant 0 : index |
| 64 | + %5 = fir.embox %4 typeparams %c0 : (!fir.heap<!fir.char<1,?>>, index) -> !fir.box<!fir.heap<!fir.char<1,?>>> |
| 65 | + fir.store %5 to %3 : !fir.ref<!fir.box<!fir.heap<!fir.char<1,?>>>> |
| 66 | + %6:2 = hlfir.declare %3 {fortran_attrs = #fir.var_attrs<allocatable>, uniq_name = "_QFtest_char_cmp2Etx"} : (!fir.ref<!fir.box<!fir.heap<!fir.char<1,?>>>>) -> (!fir.ref<!fir.box<!fir.heap<!fir.char<1,?>>>>, !fir.ref<!fir.box<!fir.heap<!fir.char<1,?>>>>) |
| 67 | + %7:2 = fir.unboxchar %arg0 : (!fir.boxchar<1>) -> (!fir.ref<!fir.char<1,?>>, index) |
| 68 | + %8:2 = hlfir.declare %7#0 typeparams %7#1 dummy_scope %0 {uniq_name = "_QFtest_char_cmp2Ex"} : (!fir.ref<!fir.char<1,?>>, index, !fir.dscope) -> (!fir.boxchar<1>, !fir.ref<!fir.char<1,?>>) |
| 69 | + %9:2 = fir.unboxchar %arg1 : (!fir.boxchar<1>) -> (!fir.ref<!fir.char<1,?>>, index) |
| 70 | + %10:2 = hlfir.declare %9#0 typeparams %9#1 dummy_scope %0 {uniq_name = "_QFtest_char_cmp2Ey"} : (!fir.ref<!fir.char<1,?>>, index, !fir.dscope) -> (!fir.boxchar<1>, !fir.ref<!fir.char<1,?>>) |
| 71 | + %11 = hlfir.char_trim %8#0 : (!fir.boxchar<1>) -> !hlfir.expr<!fir.char<1,?>> |
| 72 | + hlfir.assign %11 to %6#0 realloc : !hlfir.expr<!fir.char<1,?>>, !fir.ref<!fir.box<!fir.heap<!fir.char<1,?>>>> |
| 73 | + hlfir.destroy %11 : !hlfir.expr<!fir.char<1,?>> |
| 74 | + %12 = fir.load %6#0 : !fir.ref<!fir.box<!fir.heap<!fir.char<1,?>>>> |
| 75 | + %13 = fir.box_addr %12 : (!fir.box<!fir.heap<!fir.char<1,?>>>) -> !fir.heap<!fir.char<1,?>> |
| 76 | + %14 = fir.load %6#0 : !fir.ref<!fir.box<!fir.heap<!fir.char<1,?>>>> |
| 77 | + %15 = fir.box_elesize %14 : (!fir.box<!fir.heap<!fir.char<1,?>>>) -> index |
| 78 | + %16 = fir.emboxchar %13, %15 : (!fir.heap<!fir.char<1,?>>, index) -> !fir.boxchar<1> |
| 79 | + %17 = hlfir.cmpchar eq %16 %10#0 : (!fir.boxchar<1>, !fir.boxchar<1>) -> i1 |
| 80 | + %18 = fir.convert %17 : (i1) -> !fir.logical<4> |
| 81 | + hlfir.assign %18 to %2#0 : !fir.logical<4>, !fir.ref<!fir.logical<4>> |
| 82 | + %19 = fir.load %2#0 : !fir.ref<!fir.logical<4>> |
| 83 | + %20 = fir.load %6#0 : !fir.ref<!fir.box<!fir.heap<!fir.char<1,?>>>> |
| 84 | + %21 = fir.box_addr %20 : (!fir.box<!fir.heap<!fir.char<1,?>>>) -> !fir.heap<!fir.char<1,?>> |
| 85 | + %22 = fir.convert %21 : (!fir.heap<!fir.char<1,?>>) -> i64 |
| 86 | + %c0_i64 = arith.constant 0 : i64 |
| 87 | + %23 = arith.cmpi ne, %22, %c0_i64 : i64 |
| 88 | + fir.if %23 { |
| 89 | + %24 = fir.load %6#0 : !fir.ref<!fir.box<!fir.heap<!fir.char<1,?>>>> |
| 90 | + %25 = fir.box_addr %24 : (!fir.box<!fir.heap<!fir.char<1,?>>>) -> !fir.heap<!fir.char<1,?>> |
| 91 | + fir.freemem %25 : !fir.heap<!fir.char<1,?>> |
| 92 | + %26 = fir.zero_bits !fir.heap<!fir.char<1,?>> |
| 93 | + %c0_0 = arith.constant 0 : index |
| 94 | + %27 = fir.embox %26 typeparams %c0_0 : (!fir.heap<!fir.char<1,?>>, index) -> !fir.box<!fir.heap<!fir.char<1,?>>> |
| 95 | + fir.store %27 to %6#0 : !fir.ref<!fir.box<!fir.heap<!fir.char<1,?>>>> |
| 96 | + } |
| 97 | + return %19 : !fir.logical<4> |
| 98 | +} |
| 99 | + |
| 100 | +// CHECK-LABEL: func.func @_QPtest_char_cmp2( |
| 101 | +// CHECK: hlfir.char_trim |
0 commit comments