! Test lowering of MAXVAL intrinsic to HLFIR ! RUN: bbc -emit-hlfir -o - %s 2>&1 | FileCheck %s ! simple 1 argument MAXVAL subroutine maxval1(a, s) integer :: a(:), s s = MAXVAL(a) end subroutine ! CHECK-LABEL: func.func @_QPmaxval1( ! CHECK: %[[ARG0:.*]]: !fir.box> {fir.bindc_name = "a"}, %[[ARG1:.*]]: !fir.ref ! CHECK-DAG: %[[ARRAY:.*]]:2 = hlfir.declare %[[ARG0]] ! CHECK-DAG: %[[OUT:.*]]:2 = hlfir.declare %[[ARG1]] ! CHECK-NEXT: %[[EXPR:.*]] = hlfir.maxval %[[ARRAY]]#0 {fastmath = #arith.fastmath} : (!fir.box>) -> i32 ! CHECK-NEXT: hlfir.assign %[[EXPR]] to %[[OUT]]#0 : i32, !fir.ref ! CHECK-NEXT: return ! CHECK-NEXT: } ! maxval with by-ref DIM argument subroutine maxval2(a, s, d) integer :: a(:,:), s(:), d s = MAXVAL(a, d) end subroutine ! CHECK-LABEL: func.func @_QPmaxval2( ! CHECK: %[[ARG0:.*]]: !fir.box> {fir.bindc_name = "a"}, %[[ARG1:.*]]: !fir.box> {fir.bindc_name = "s"}, %[[ARG2:.*]]: !fir.ref ! CHECK-DAG: %[[ARRAY:.*]]:2 = hlfir.declare %[[ARG0]] ! CHECK-DAG: %[[OUT:.*]]:2 = hlfir.declare %[[ARG1]] ! CHECK-DAG: %[[DIM_REF:.*]]:2 = hlfir.declare %[[ARG2]] ! CHECK-NEXT: %[[DIM:.*]] = fir.load %[[DIM_REF]]#0 : !fir.ref ! CHECK-NEXT: %[[EXPR:.*]] = hlfir.maxval %[[ARRAY]]#0 dim %[[DIM]] {fastmath = #arith.fastmath} : (!fir.box>, i32) -> !hlfir.expr ! CHECK-NEXT: hlfir.assign %[[EXPR]] to %[[OUT]]#0 : !hlfir.expr, !fir.box> ! CHECK-NEXT: hlfir.destroy %[[EXPR]] ! CHECK-NEXT: return ! CHECK-NEXT: } ! maxval with scalar mask argument subroutine maxval3(a, s, m) integer :: a(:), s logical :: m s = MAXVAL(a, m) end subroutine ! CHECK-LABEL: func.func @_QPmaxval3( ! CHECK: %[[ARG0:.*]]: !fir.box> {fir.bindc_name = "a"}, %[[ARG1:.*]]: !fir.ref {fir.bindc_name = "s"}, %[[ARG2:.*]]: !fir.ref> ! CHECK-DAG: %[[ARRAY:.*]]:2 = hlfir.declare %[[ARG0]] ! CHECK-DAG: %[[OUT:.*]]:2 = hlfir.declare %[[ARG1]] ! CHECK-DAG: %[[MASK:.*]]:2 = hlfir.declare %[[ARG2]] ! CHECK-NEXT: %[[EXPR:.*]] = hlfir.maxval %[[ARRAY]]#0 mask %[[MASK]]#0 {fastmath = #arith.fastmath} : (!fir.box>, !fir.ref>) -> i32 ! CHECK-NEXT: hlfir.assign %[[EXPR]] to %[[OUT]]#0 : i32, !fir.ref ! CHECK-NEXT: return ! CHECK-NEXT: } ! maxval with array mask argument subroutine maxval4(a, s, m) integer :: a(:), s logical :: m(:) s = MAXVAL(a, m) end subroutine ! CHECK-LABEL: func.func @_QPmaxval4( ! CHECK: %[[ARG0:.*]]: !fir.box> {fir.bindc_name = "a"}, %[[ARG1:.*]]: !fir.ref {fir.bindc_name = "s"}, %[[ARG2:.*]]: !fir.box>> ! CHECK-DAG: %[[ARRAY:.*]]:2 = hlfir.declare %[[ARG0]] ! CHECK-DAG: %[[OUT:.*]]:2 = hlfir.declare %[[ARG1]] ! CHECK-DAG: %[[MASK:.*]]:2 = hlfir.declare %[[ARG2]] ! CHECK-NEXT: %[[EXPR:.*]] = hlfir.maxval %[[ARRAY]]#0 mask %[[MASK]]#0 {fastmath = #arith.fastmath} : (!fir.box>, !fir.box>>) -> i32 ! CHECK-NEXT: hlfir.assign %[[EXPR]] to %[[OUT]]#0 : i32, !fir.ref ! CHECK-NEXT: return ! CHECK-NEXT: } ! maxval with all 3 arguments, dim is by-val, array isn't boxed subroutine maxval5(s) integer :: s(2) integer :: a(2,2) = reshape((/1, 2, 3, 4/), [2,2]) s = maxval(a, 1, .true.) end subroutine ! CHECK-LABEL: func.func @_QPmaxval5 ! CHECK: %[[ARG0:.*]]: !fir.ref> ! CHECK-DAG: %[[ADDR:.*]] = fir.address_of({{.*}}) : !fir.ref> ! CHECK-DAG: %[[ARRAY_SHAPE:.*]] = fir.shape {{.*}} -> !fir.shape<2> ! CHECK-DAG: %[[ARRAY:.*]]:2 = hlfir.declare %[[ADDR]](%[[ARRAY_SHAPE]]) ! CHECK-DAG: %[[OUT_SHAPE:.*]] = fir.shape {{.*}} -> !fir.shape<1> ! CHECK-DAG: %[[OUT:.*]]:2 = hlfir.declare %[[ARG0]](%[[OUT_SHAPE]]) ! CHECK-DAG: %[[TRUE:.*]] = arith.constant true ! CHECK-DAG: %[[C1:.*]] = arith.constant 1 : i32 ! CHECK-NEXT: %[[EXPR:.*]] = hlfir.maxval %[[ARRAY]]#0 dim %[[C1]] mask %[[TRUE]] {fastmath = #arith.fastmath} : (!fir.ref>, i32, i1) -> !hlfir.expr<2xi32> ! CHECK-NEXT: hlfir.assign %[[EXPR]] to %[[OUT]]#0 : !hlfir.expr<2xi32>, !fir.ref> ! CHECK-NEXT: hlfir.destroy %[[EXPR]] : !hlfir.expr<2xi32> ! CHECK-NEXT: return ! CHECK-nEXT: } subroutine maxval6(a, s, d) integer, pointer :: d real :: a(:,:), s(:) s = maxval(a, (d)) end subroutine ! CHECK-LABEL: func.func @_QPmaxval6( ! CHECK: %[[ARG0:.*]]: !fir.box> ! CHECK: %[[ARG1:.*]]: !fir.box> ! CHECK: %[[ARG2:.*]]: !fir.ref>> ! CHECK-DAG: %[[ARRAY:.*]]:2 = hlfir.declare %[[ARG0]] ! CHECK-DAG: %[[OUT:.*]]:2 = hlfir.declare %[[ARG1]] ! CHECK-DAG: %[[DIM_VAR:.*]]:2 = hlfir.declare %[[ARG2]] ! CHECK-NEXT: %[[DIM_BOX:.*]] = fir.load %[[DIM_VAR]]#0 : !fir.ref>> ! CHECK-NEXT: %[[DIM_ADDR:.*]] = fir.box_addr %[[DIM_BOX]] : (!fir.box>) -> !fir.ptr ! CHECK-NEXT: %[[DIM0:.*]] = fir.load %[[DIM_ADDR]] : !fir.ptr ! CHECK-NEXT: %[[DIM1:.*]] = hlfir.no_reassoc %[[DIM0]] : i32 ! CHECK-NEXT: %[[EXPR:.*]] = hlfir.maxval %[[ARRAY]]#0 dim %[[DIM1]] {fastmath = #arith.fastmath} : (!fir.box>, i32) -> !hlfir.expr ! CHECK-NEXT: hlfir.assign %[[EXPR]] to %[[OUT]]#0 : !hlfir.expr, !fir.box> ! CHECK-NEXT: hlfir.destroy %[[EXPR]] ! CHECK-NEXT: return ! CHECK-NEXT: } ! simple 1 argument MAXVAL for character subroutine maxval7(a, s) character(*) :: a(:), s s = MAXVAL(a) end subroutine ! CHECK-LABEL: func.func @_QPmaxval7( ! CHECK: %[[ARG0:.*]]: !fir.box>> {fir.bindc_name = "a"}, %[[ARG1:.*]]: !fir.boxchar<1> ! CHECK-DAG: %[[ARRAY:.*]]:2 = hlfir.declare %[[ARG0]] ! CHECK-DAG: %[[UNBOXED:.*]]:2 = fir.unboxchar %[[ARG1]] ! CHECK-DAG: %[[OUT:.*]]:2 = hlfir.declare %[[UNBOXED]]#0 typeparams %[[UNBOXED]]#1 ! CHECK-NEXT: %[[EXPR:.*]] = hlfir.maxval %[[ARRAY]]#0 {fastmath = #arith.fastmath} : (!fir.box>>) -> !hlfir.expr> ! CHECK-NEXT: hlfir.assign %[[EXPR]] to %[[OUT]]#0 : !hlfir.expr>, !fir.boxchar<1> ! CHECK-NEXT: hlfir.destroy %[[EXPR]] ! CHECK-NEXT: return ! CHECK-NEXT: } ! maxval for character with by-ref DIM argument subroutine maxval8(a, s, d) character(*) :: a(:,:), s(:) integer :: d s = MAXVAL(a, d) end subroutine ! CHECK-LABEL: func.func @_QPmaxval8( ! CHECK: %[[ARG0:.*]]: !fir.box>> {fir.bindc_name = "a"}, %[[ARG1:.*]]: !fir.box>> {fir.bindc_name = "s"}, %[[ARG2:.*]]: !fir.ref ! CHECK-DAG: %[[ARRAY:.*]]:2 = hlfir.declare %[[ARG0]] ! CHECK-DAG: %[[OUT:.*]]:2 = hlfir.declare %[[ARG1]] ! CHECK-DAG: %[[DIM_REF:.*]]:2 = hlfir.declare %[[ARG2]] ! CHECK-NEXT: %[[DIM:.*]] = fir.load %[[DIM_REF]]#0 : !fir.ref ! CHECK-NEXT: %[[EXPR:.*]] = hlfir.maxval %[[ARRAY]]#0 dim %[[DIM]] {fastmath = #arith.fastmath} : (!fir.box>>, i32) -> !hlfir.expr> ! CHECK-NEXT: hlfir.assign %[[EXPR]] to %[[OUT]]#0 : !hlfir.expr>, !fir.box>> ! CHECK-NEXT: hlfir.destroy %[[EXPR]] ! CHECK-NEXT: return ! CHECK-NEXT: } ! maxval for character with scalar mask argument subroutine maxval9(a, s, m) character(*) :: a(:), s logical :: m s = MAXVAL(a, m) end subroutine ! CHECK-LABEL: func.func @_QPmaxval9( ! CHECK: %[[ARG0:.*]]: !fir.box>> {fir.bindc_name = "a"}, %[[ARG1:.*]]: !fir.boxchar<1> {fir.bindc_name = "s"}, %[[ARG2:.*]]: !fir.ref> ! CHECK-DAG: %[[ARRAY:.*]]:2 = hlfir.declare %[[ARG0]] ! CHECK-DAG: %[[UNBOXED:.*]]:2 = fir.unboxchar %[[ARG1]] ! CHECK-DAG: %[[MASK:.*]]:2 = hlfir.declare %[[ARG2]] ! CHECK-DAG: %[[OUT:.*]]:2 = hlfir.declare %[[UNBOXED]]#0 typeparams %[[UNBOXED]]#1 ! CHECK-NEXT: %[[EXPR:.*]] = hlfir.maxval %[[ARRAY]]#0 mask %[[MASK]]#0 {fastmath = #arith.fastmath} : (!fir.box>>, !fir.ref>) -> !hlfir.expr> ! CHECK-NEXT: hlfir.assign %[[EXPR]] to %[[OUT]]#0 : !hlfir.expr>, !fir.boxchar<1> ! CHECK-NEXT: hlfir.destroy %[[EXPR]] ! CHECK-NEXT: return ! CHECK-NEXT: } subroutine testDynamicallyOptionalMask(array, mask, res) integer :: array(:), res logical, allocatable :: mask(:) res = MAXVAL(array, mask=mask) end subroutine ! CHECK-LABEL: func.func @_QPtestdynamicallyoptionalmask( ! CHECK-SAME: %[[ARG0:.*]]: !fir.box> ! CHECK-SAME: %[[ARG1:.*]]: !fir.ref>>>> ! CHECK-SAME: %[[ARG2:.*]]: !fir.ref ! CHECK-DAG: %[[ARRAY:.*]]:2 = hlfir.declare %[[ARG0]] ! CHECK-DAG: %[[MASK:.*]]:2 = hlfir.declare %[[ARG1]] ! CHECK-DAG: %[[RES:.*]]:2 = hlfir.declare %[[ARG2]] ! CHECK-NEXT: %[[MASK_LOAD:.*]] = fir.load %[[MASK]]#1 ! CHECK-NEXT: %[[MASK_ADDR:.*]] = fir.box_addr %[[MASK_LOAD]] ! CHECK-NEXT: %[[MASK_ADDR_INT:.*]] = fir.convert %[[MASK_ADDR]] ! CHECK-NEXT: %[[C0:.*]] = arith.constant 0 : i64 ! CHECK-NEXT: %[[CMP:.*]] = arith.cmpi ne, %[[MASK_ADDR_INT]], %[[C0]] : i64 ! it is a shame there is a second load here. The first is generated for ! PreparedActualArgument::isPresent, the second is for optional handling ! CHECK-NEXT: %[[MASK_LOAD2:.*]] = fir.load %[[MASK]]#1 ! CHECK-NEXT: %[[ABSENT:.*]] = fir.absent !fir.box>>> ! CHECK-NEXT: %[[SELECT:.*]] = arith.select %[[CMP]], %[[MASK_LOAD2]], %[[ABSENT]] ! CHECK-NEXT: %[[MAXVAL:.*]] = hlfir.maxval %[[ARRAY]]#0 mask %[[SELECT]] ! CHECK-NEXT: hlfir.assign %[[MAXVAL]] to %[[RES]]#0 ! CHECK-NEXT: return ! CHECK-NEXT: } subroutine testAllocatableArray(array, mask, res) integer, allocatable :: array(:) integer :: res logical :: mask(:) res = MAXVAL(array, mask=mask) end subroutine ! CHECK-LABEL: func.func @_QPtestallocatablearray( ! CHECK-SAME: %[[ARG0:.*]]: !fir.ref>>> ! CHECK-SAME: %[[ARG1:.*]]: !fir.box>> ! CHECK-SAME: %[[ARG2:.*]]: !fir.ref ! CHECK-DAG: %[[ARRAY:.*]]:2 = hlfir.declare %[[ARG0]] ! CHECK-DAG: %[[MASK:.*]]:2 = hlfir.declare %[[ARG1]] ! CHECK-DAG: %[[RES:.*]]:2 = hlfir.declare %[[ARG2]] ! CHECK-NEXT: %[[LOADED_ARRAY:.*]] = fir.load %[[ARRAY]]#0 ! CHECK-NEXT: %[[MAXVAL:.*]] = hlfir.maxval %[[LOADED_ARRAY]] mask %[[MASK]]#0 ! CHECK-NEXT: hlfir.assign %[[MAXVAL]] to %[[RES]]#0 ! CHECK-NEXT: return ! CHECK-NEXT: } function testOptionalScalar(array, mask) integer :: array(:) logical, optional :: mask integer :: testOptionalScalar testOptionalScalar = maxval(array, mask) end function ! CHECK-LABEL: func.func @_QPtestoptionalscalar( ! CHECK-SAME: %[[ARRAY_ARG:.*]]: !fir.box> {fir.bindc_name = "array"}, ! CHECK-SAME: %[[MASK_ARG:.*]]: !fir.ref> {fir.bindc_name = "mask", fir.optional}) -> i32 ! CHECK: %[[ARRAY_VAR:.*]]:2 = hlfir.declare %[[ARRAY_ARG]] ! CHECK: %[[MASK_VAR:.*]]:2 = hlfir.declare %[[MASK_ARG]] ! CHECK: %[[RET_ALLOC:.*]] = fir.alloca i32 {bindc_name = "testoptionalscalar", uniq_name = "_QFtestoptionalscalarEtestoptionalscalar"} ! CHECK: %[[RET_VAR:.*]]:2 = hlfir.declare %[[RET_ALLOC]] ! CHECK: %[[MASK_IS_PRESENT:.*]] = fir.is_present %[[MASK_VAR]]#0 : (!fir.ref>) -> i1 ! CHECK: %[[MASK_BOX:.*]] = fir.embox %[[MASK_VAR]]#1 ! CHECK: %[[ABSENT:.*]] = fir.absent !fir.box> ! CHECK: %[[MASK_SELECT:.*]] = arith.select %[[MASK_IS_PRESENT]], %[[MASK_BOX]], %[[ABSENT]] ! CHECK: %[[RES:.*]] = hlfir.maxval %[[ARRAY_VAR]]#0 mask %[[MASK_SELECT]] {{.*}}: (!fir.box>, !fir.box>) -> i32 ! CHECK: hlfir.assign %[[RES]] to %[[RET_VAR]]#0 ! CHECK: %[[RET:.*]] = fir.load %[[RET_VAR]]#1 : !fir.ref ! CHECK: return %[[RET]] : i32 ! CHECK: } ! Test that hlfir.maxval lowering inherits constant ! character length from the argument, when the length ! is unknown from the Fortran::evaluate expression type. subroutine test_unknown_char_len_result character(len=3) :: array(3,3), res res = maxval(array(:,:)(:)) end subroutine test_unknown_char_len_result ! CHECK-LABEL: func.func @_QPtest_unknown_char_len_result() { ! CHECK-DAG: %[[C3:.*]] = arith.constant 3 : index ! CHECK-DAG: %[[C3_0:.*]] = arith.constant 3 : index ! CHECK-DAG: %[[C3_1:.*]] = arith.constant 3 : index ! CHECK-DAG: %[[ARRAY_ALLOC:.*]] = fir.alloca !fir.array<3x3x!fir.char<1,3>> ! CHECK-DAG: %[[ARRAY_SHAPE:.*]] = fir.shape %[[C3_0]], %[[C3_1]] : (index, index) -> !fir.shape<2> ! CHECK-DAG: %[[ARRAY:.*]]:2 = hlfir.declare %[[ARRAY_ALLOC]](%[[ARRAY_SHAPE]]) typeparams %[[C3]] ! CHECK-DAG: %[[C3_2:.*]] = arith.constant 3 : index ! CHECK-DAG: %[[RES_ALLOC:.*]] = fir.alloca !fir.char<1,3> ! CHECK-DAG: %[[RES:.*]]:2 = hlfir.declare %[[RES_ALLOC]] typeparams %[[C3_2]] ! CHECK-DAG: %[[C1:.*]] = arith.constant 1 : index ! CHECK-DAG: %[[C1_3:.*]] = arith.constant 1 : index ! CHECK-DAG: %[[C3_4:.*]] = arith.constant 3 : index ! CHECK-DAG: %[[C1_5:.*]] = arith.constant 1 : index ! CHECK-DAG: %[[C3_6:.*]] = arith.constant 3 : index ! CHECK-DAG: %[[SHAPE:.*]] = fir.shape %[[C3_4]], %[[C3_6]] : (index, index) -> !fir.shape<2> ! CHECK-DAG: %[[C1_7:.*]] = arith.constant 1 : index ! CHECK-DAG: %[[C3_8:.*]] = arith.constant 3 : index ! CHECK-DAG: %[[C3_9:.*]] = arith.constant 3 : index ! CHECK-DAG: %[[ARRAY_BOX:.*]] = hlfir.designate %[[ARRAY]]#0 (%[[C1]]:%[[C3_0]]:%[[C1_3]], %[[C1]]:%[[C3_1]]:%[[C1_5]]) substr %[[C1_7]], %[[C3_8]] shape %[[SHAPE]] typeparams %[[C3_9]] ! CHECK: %[[EXPR:.*]] = hlfir.maxval %[[ARRAY_BOX]] {fastmath = #arith.fastmath} : (!fir.box>>) -> !hlfir.expr> ! CHECK-NEXT: hlfir.assign %[[EXPR]] to %[[RES]]#0 : !hlfir.expr>, !fir.ref> ! CHECK-NEXT: hlfir.destroy %[[EXPR]] ! CHECK-NEXT: return ! CHECK-NEXT: }