! Test creation of temporary from polymorphic enities ! RUN: bbc -polymorphic-type -emit-fir -hlfir=false %s -o - | FileCheck %s module poly_tmp type p1 integer :: a end type type, extends(p1) :: p2 integer :: b end type contains subroutine pass_unlimited_poly_1d(x) class(*), intent(in) :: x(:) end subroutine subroutine test_temp_from_intrinsic_spread() class(*), pointer :: p class(*), pointer :: pa(:) allocate(p2::p) allocate(p2::pa(10)) call pass_unlimited_poly_1d(spread(p, dim=1, ncopies=2)) call pass_unlimited_poly_1d(spread(pa(1), dim=1, ncopies=2)) end subroutine ! CHECK-LABEL: func.func @_QMpoly_tmpPtest_temp_from_intrinsic_spread() { ! CHECK: %[[TEMP_RES1:.*]] = fir.alloca !fir.class>> ! CHECK: %[[TEMP_RES0:.*]] = fir.alloca !fir.class>> ! CHECK: %[[P:.*]] = fir.alloca !fir.class> {bindc_name = "p", uniq_name = "_QMpoly_tmpFtest_temp_from_intrinsic_spreadEp"} ! CHECK: %[[PA:.*]] = fir.alloca !fir.class>> {bindc_name = "pa", uniq_name = "_QMpoly_tmpFtest_temp_from_intrinsic_spreadEpa"} ! CHECK: fir.call @_FortranAPointerNullifyDerived ! CHECK: fir.call @_FortranAPointerAllocate ! CHECK: %[[LOAD_P:.*]] = fir.load %[[P]] : !fir.ref>> ! CHECK: %[[C1:.*]] = arith.constant 1 : i32 ! CHECK: %[[C2:.*]] = arith.constant 2 : i32 ! CHECK: %[[ZERO:.*]] = fir.zero_bits !fir.heap> ! CHECK: %[[C0:.*]] = arith.constant 0 : index ! CHECK: %[[SHAPE:.*]] = fir.shape %[[C0]] : (index) -> !fir.shape<1> ! Make sure the fir.embox contains the source_box pointing to the polymoprhic entity ! CHECK: %[[BOX_RES:.*]] = fir.embox %[[ZERO]](%[[SHAPE]]) source_box %[[LOAD_P]] : (!fir.heap>, !fir.shape<1>, !fir.class>) -> !fir.class>> ! CHECK: fir.store %[[BOX_RES]] to %[[TEMP_RES0]] : !fir.ref>>> ! CHECK: %[[RES_BOX_NONE:.*]] = fir.convert %[[TEMP_RES0]] : (!fir.ref>>>) -> !fir.ref> ! CHECK: %[[P_BOX_NONE:.*]] = fir.convert %[[LOAD_P]] : (!fir.class>) -> !fir.box ! CHECK: %[[C2_I64:.*]] = fir.convert %[[C2]] : (i32) -> i64 ! CHECK: %{{.*}} = fir.call @_FortranASpread(%[[RES_BOX_NONE]], %[[P_BOX_NONE]], %[[C1]], %[[C2_I64]], %{{.*}}, %{{.*}}) fastmath : (!fir.ref>, !fir.box, i32, i64, !fir.ref, i32) -> none ! CHECK: %[[LOAD_RES:.*]] = fir.load %[[TEMP_RES0]] : !fir.ref>>> ! CHECK: %[[RES_ADDR:.*]] = fir.box_addr %[[LOAD_RES]] : (!fir.class>>) -> !fir.heap> ! CHECK: %[[REBOX:.*]] = fir.rebox %[[LOAD_RES]] : (!fir.class>>) -> !fir.class> ! CHECK: fir.call @_QMpoly_tmpPpass_unlimited_poly_1d(%[[REBOX]]) {{.*}} : (!fir.class>) -> () ! CHECK: fir.freemem %[[RES_ADDR]] : !fir.heap> ! CHECK: %[[LOAD_PA:.*]] = fir.load %[[PA]] : !fir.ref>>> ! CHECK: %[[COORD_PA_1:.*]] = fir.coordinate_of %[[LOAD_PA]], %{{.*}} : (!fir.class>>, i64) -> !fir.ref ! CHECK: %[[EMBOX_PA_1:.*]] = fir.embox %[[COORD_PA_1]] source_box %[[LOAD_PA]] : (!fir.ref, !fir.class>>) -> !fir.class ! CHECK: %[[RES1_BOX_NONE:.*]] = fir.convert %[[TEMP_RES1]] : (!fir.ref>>>) -> !fir.ref> ! CHECK: %[[PA1_BOX_NONE:.*]] = fir.convert %[[EMBOX_PA_1]] : (!fir.class) -> !fir.box ! CHECK: %{{.*}} = fir.call @_FortranASpread(%[[RES1_BOX_NONE]], %[[PA1_BOX_NONE]], %{{.*}}, %{{.*}}, %{{.*}}, %{{.*}}) {{.*}} : (!fir.ref>, !fir.box, i32, i64, !fir.ref, i32) -> none subroutine test_temp_from_intrinsic_reshape(i) class(*), allocatable :: a(:,:) class(*), intent(in) :: i(20,20) allocate(a(10,10), source=reshape(i,[10,10])) end subroutine ! CHECK-LABEL: func.func @_QMpoly_tmpPtest_temp_from_intrinsic_reshape( ! CHECK-SAME: %[[I:.*]]: !fir.class> {fir.bindc_name = "i"}) { ! CHECK: %[[TMP_RES:.*]] = fir.alloca !fir.class>> ! CHECK: %[[A:.*]] = fir.alloca !fir.class>> {bindc_name = "a", uniq_name = "_QMpoly_tmpFtest_temp_from_intrinsic_reshapeEa"} ! CHECK: %[[EMBOX_WITH_SOURCE:.*]] = fir.embox %{{.*}}(%{{.*}}) source_box %[[I]] : (!fir.heap>, !fir.shape<2>, !fir.class>) -> !fir.class>> ! CHECK: fir.store %[[EMBOX_WITH_SOURCE]] to %[[TMP_RES]] : !fir.ref>>> ! CHECK: %[[RES_BOX_NONE:.*]] = fir.convert %[[TMP_RES]] : (!fir.ref>>>) -> !fir.ref> ! CHECK: %[[I_BOX_NONE:.*]] = fir.convert %[[I]] : (!fir.class>) -> !fir.box ! CHECK: %{{.*}} = fir.call @_FortranAReshape(%[[RES_BOX_NONE]], %[[I_BOX_NONE]], %{{.*}}, %{{.*}}, %{{.*}}, %{{.*}}, %{{.*}}) {{.*}} : (!fir.ref>, !fir.box, !fir.box, !fir.box, !fir.box, !fir.ref, i32) -> none ! CHECK: %[[LOAD_RES:.*]] = fir.load %[[TMP_RES]] : !fir.ref>>> ! CHECK: %[[RANK:.*]] = arith.constant 2 : i32 ! CHECK: %[[A_BOX_NONE:.*]] = fir.convert %[[A]] : (!fir.ref>>>) -> !fir.ref> ! CHECK: %[[RES_BOX_NONE:.*]] = fir.convert %[[LOAD_RES]] : (!fir.class>>) -> !fir.box ! CHECK: %{{.*}} = fir.call @_FortranAAllocatableApplyMold(%[[A_BOX_NONE]], %[[RES_BOX_NONE]], %[[RANK]]) {{.*}} : (!fir.ref>, !fir.box, i32) -> none subroutine check(r) class(p1) :: r(:) end subroutine subroutine test_temp_from_intrinsic_pack(i, mask) class(p1), intent(in) :: i(20, 20) logical, intent(in) :: mask(20, 20) call check(pack(i, mask)) end subroutine ! CHECK-LABEL: func.func @_QMpoly_tmpPtest_temp_from_intrinsic_pack( ! CHECK-SAME: %[[I:.*]]: !fir.class>> {fir.bindc_name = "i"}, %[[MASK:.*]]: !fir.ref>> {fir.bindc_name = "mask"}) { ! CHECK: %[[TMP_RES:.*]] = fir.alloca !fir.class>>> ! CHECK: %[[EMBOXED_MASK:.*]] = fir.embox %[[MASK]](%{{.*}}) : (!fir.ref>>, !fir.shape<2>) -> !fir.box>> ! CHECK: %[[ZERO:.*]] = fir.zero_bits !fir.heap>> ! CHECK: %[[EMBOX_RES:.*]] = fir.embox %[[ZERO]](%{{.*}}) source_box %[[I]] : (!fir.heap>>, !fir.shape<1>, !fir.class>>) -> !fir.class>>> ! CHECK: fir.store %[[EMBOX_RES]] to %[[TMP_RES]] : !fir.ref>>>> ! CHECK: %[[RES_BOX_NONE:.*]] = fir.convert %[[TMP_RES]] : (!fir.ref>>>>) -> !fir.ref> ! CHECK: %[[I_BOX_NONE:.*]] = fir.convert %[[I]] : (!fir.class>>) -> !fir.box ! CHECK: %[[MASK_BOX_NONE:.*]] = fir.convert %[[EMBOXED_MASK]] : (!fir.box>>) -> !fir.box ! CHECK: %{{.*}} = fir.call @_FortranAPack(%[[RES_BOX_NONE]], %[[I_BOX_NONE]], %[[MASK_BOX_NONE]], %{{.*}}, %{{.*}}, %{{.*}}) {{.*}} : (!fir.ref>, !fir.box, !fir.box, !fir.box, !fir.ref, i32) -> none subroutine check_rank2(r) class(p1), intent(in) :: r(:,:) end subroutine subroutine test_temp_from_unpack(v, m, f) class(p1), intent(in) :: v(:), f(:,:) logical, intent(in) :: m(:,:) call check_rank2(unpack(v,m,f)) end subroutine ! CHECK-LABEL: func.func @_QMpoly_tmpPtest_temp_from_unpack( ! CHECK-SAME: %[[V:.*]]: !fir.class>> {fir.bindc_name = "v"}, %[[M:.*]]: !fir.box>> {fir.bindc_name = "m"}, %[[F:.*]]: !fir.class>> {fir.bindc_name = "f"}) { ! CHECK: %[[TMP_RES:.*]] = fir.alloca !fir.class>>> ! CHECK: %[[TMP_BOX_NONE:.*]] = fir.convert %[[TMP_RES]] : (!fir.ref>>>>) -> !fir.ref> ! CHECK: %[[V_BOX_NONE:.*]] = fir.convert %[[V]] : (!fir.class>>) -> !fir.box ! CHECK: %[[M_BOX_NONE:.*]] = fir.convert %[[M]] : (!fir.box>>) -> !fir.box ! CHECK: %[[F_BOX_NONE:.*]] = fir.convert %[[F]] : (!fir.class>>) -> !fir.box ! CHECK: %{{.*}} = fir.call @_FortranAUnpack(%[[TMP_BOX_NONE]], %[[V_BOX_NONE]], %[[M_BOX_NONE]], %[[F_BOX_NONE]], %{{.*}}, %{{.*}}) {{.*}} : (!fir.ref>, !fir.box, !fir.box, !fir.box, !fir.ref, i32) -> none subroutine check_cshift(r) class(p1) :: r(:) end subroutine subroutine test_temp_from_intrinsic_cshift(a, shift) class(p1), intent(in) :: a(20) integer :: shift call check(cshift(a, shift)) end subroutine ! CHECK-LABEL: func.func @_QMpoly_tmpPtest_temp_from_intrinsic_cshift( ! CHECK-SAME: %[[ARRAY:.*]]: !fir.class>> {fir.bindc_name = "a"}, %[[SHIFT:.*]]: !fir.ref {fir.bindc_name = "shift"}) { ! CHECK: %[[TMP_RES:.*]] = fir.alloca !fir.class>>> ! CHECK: %[[LOAD_SHIFT:.*]] = fir.load %[[SHIFT]] : !fir.ref ! CHECK: %[[RES_BOX_NONE:.*]] = fir.convert %[[TMP_RES]] : (!fir.ref>>>>) -> !fir.ref> ! CHECK: %[[ARRAY_NONE:.*]] = fir.convert %[[ARRAY]] : (!fir.class>>) -> !fir.box ! CHECK: %[[SHIFT_I64:.*]] = fir.convert %[[LOAD_SHIFT]] : (i32) -> i64 ! CHECK: %{{.*}} = fir.call @_FortranACshiftVector(%[[RES_BOX_NONE]], %[[ARRAY_NONE]], %[[SHIFT_I64]], %{{.*}}, %{{.*}}) {{.*}} : (!fir.ref>, !fir.box, i64, !fir.ref, i32) -> none subroutine test_temp_from_intrinsic_eoshift(a, shift, b) class(p1), intent(in) :: a(20) class(p1), intent(in) :: b integer :: shift call check(eoshift(a, shift, b)) end subroutine ! CHECK-LABEL: func.func @_QMpoly_tmpPtest_temp_from_intrinsic_eoshift( ! CHECK-SAME: %[[ARRAY:.*]]: !fir.class>> {fir.bindc_name = "a"}, %[[SHIFT:.*]]: !fir.ref {fir.bindc_name = "shift"}, %[[BOUNDARY:.*]]: !fir.class> {fir.bindc_name = "b"}) { ! CHECK: %[[TMP_RES:.*]] = fir.alloca !fir.class>>> ! CHECK: %[[LOAD_SHIFT:.*]] = fir.load %[[SHIFT]] : !fir.ref ! CHECK: %[[RES_BOX_NONE:.*]] = fir.convert %[[TMP_RES]] : (!fir.ref>>>>) -> !fir.ref> ! CHECK: %[[ARRAY_NONE:.*]] = fir.convert %[[ARRAY]] : (!fir.class>>) -> !fir.box ! CHECK: %[[SHIFT_I64:.*]] = fir.convert %[[LOAD_SHIFT]] : (i32) -> i64 ! CHECK: %[[BOUNDARY_NONE:.*]] = fir.convert %[[BOUNDARY]] : (!fir.class>) -> !fir.box ! CHECK: %{{.*}} = fir.call @_FortranAEoshiftVector(%[[RES_BOX_NONE]], %[[ARRAY_NONE]], %[[SHIFT_I64]], %[[BOUNDARY_NONE]], %{{.*}}, %{{.*}}) {{.*}} : (!fir.ref>, !fir.box, i64, !fir.box, !fir.ref, i32) -> none subroutine test_temp_from_intrinsic_transfer(source, mold) class(p1), intent(in) :: source(:) class(p1), intent(in) :: mold(:) call check(transfer(source, mold)) end subroutine ! CHECK-LABEL: func.func @_QMpoly_tmpPtest_temp_from_intrinsic_transfer( ! CHECK-SAME: %[[SOURCE:.*]]: !fir.class>> {fir.bindc_name = "source"}, %[[MOLD:.*]]: !fir.class>> {fir.bindc_name = "mold"}) { ! CHECK: %[[TMP_RES:.*]] = fir.alloca !fir.class>>> ! CHECK: %[[RES_BOX_NONE:.*]] = fir.convert %[[TMP_RES]] : (!fir.ref>>>>) -> !fir.ref> ! CHECK: %[[SOURCE_NONE:.*]] = fir.convert %[[SOURCE]] : (!fir.class>>) -> !fir.box ! CHECK: %[[MOLD_NONE:.*]] = fir.convert %[[MOLD]] : (!fir.class>>) -> !fir.box ! CHECK: %{{.*}} = fir.call @_FortranATransfer(%[[RES_BOX_NONE]], %[[SOURCE_NONE]], %[[MOLD_NONE]], %{{.*}}, %{{.*}}) {{.*}} : (!fir.ref>, !fir.box, !fir.box, !fir.ref, i32) -> none subroutine test_temp_from_intrinsic_transpose(matrix) class(p1), intent(in) :: matrix(:,:) call check_rank2(transpose(matrix)) end subroutine ! CHECK-LABEL: func.func @_QMpoly_tmpPtest_temp_from_intrinsic_transpose( ! CHECK-SAME: %[[MATRIX:.*]]: !fir.class>> {fir.bindc_name = "matrix"}) { ! CHECK: %[[TMP_RES:.*]] = fir.alloca !fir.class>>> ! CHECK: %[[RES_BOX_NONE:.*]] = fir.convert %[[TMP_RES]] : (!fir.ref>>>>) -> !fir.ref> ! CHECK: %[[MATRIX_NONE:.*]] = fir.convert %[[MATRIX]] : (!fir.class>>) -> !fir.box ! CHECK: %{{.*}} = fir.call @_FortranATranspose(%[[RES_BOX_NONE]], %[[MATRIX_NONE]], %{{.*}}, %{{.*}}) {{.*}} : (!fir.ref>, !fir.box, !fir.ref, i32) -> none subroutine check_scalar(a) class(p1), intent(in) :: a end subroutine subroutine test_merge_intrinsic(a, b) class(p1), intent(in) :: a, b call check_scalar(merge(a, b, a%a > b%a)) end subroutine ! CHECK-LABEL: func.func @_QMpoly_tmpPtest_merge_intrinsic( ! CHECK-SAME: %[[ARG0:.*]]: !fir.class> {fir.bindc_name = "a"}, %[[ARG1:.*]]: !fir.class> {fir.bindc_name = "b"}) { ! CHECK: %[[FIELD_A:.*]] = fir.field_index a, !fir.type<_QMpoly_tmpTp1{a:i32}> ! CHECK: %[[COORD_A:.*]] = fir.coordinate_of %[[ARG0]], %[[FIELD_A]] : (!fir.class>, !fir.field) -> !fir.ref ! CHECK: %[[LOAD_A1:.*]] = fir.load %[[COORD_A]] : !fir.ref ! CHECK: %[[FIELD_A:.*]] = fir.field_index a, !fir.type<_QMpoly_tmpTp1{a:i32}> ! CHECK: %[[COORD_A:.*]] = fir.coordinate_of %[[ARG1]], %[[FIELD_A]] : (!fir.class>, !fir.field) -> !fir.ref ! CHECK: %[[LOAD_A2:.*]] = fir.load %[[COORD_A]] : !fir.ref ! CHECK: %[[CMPI:.*]] = arith.cmpi sgt, %[[LOAD_A1]], %[[LOAD_A2]] : i32 ! CHECK: %[[SELECT:.*]] = arith.select %[[CMPI]], %[[ARG0]], %[[ARG1]] : !fir.class> ! CHECK: fir.call @_QMpoly_tmpPcheck_scalar(%[[SELECT]]) {{.*}} : (!fir.class>) -> () subroutine test_merge_intrinsic2(a, b, i) class(p1), allocatable, intent(in) :: a type(p1), allocatable :: b integer, intent(in) :: i call check_scalar(merge(a, b, i==1)) end subroutine ! CHECK-LABEL: func.func @_QMpoly_tmpPtest_merge_intrinsic2( ! CHECK-SAME: %[[A:.*]]: !fir.ref>>> {fir.bindc_name = "a"}, %[[B:.*]]: !fir.ref>>> {fir.bindc_name = "b"}, %[[I:.*]]: !fir.ref {fir.bindc_name = "i"}) { ! CHECK: %[[LOAD_A:.*]] = fir.load %[[A]] : !fir.ref>>> ! CHECK: %[[LOAD_B:.*]] = fir.load %[[B]] : !fir.ref>>> ! CHECK: %[[LOAD_I:.*]] = fir.load %[[I]] : !fir.ref ! CHECK: %[[C1:.*]] = arith.constant 1 : i32 ! CHECK: %[[CMPI:.*]] = arith.cmpi eq, %[[LOAD_I]], %[[C1]] : i32 ! CHECK: %[[A_REBOX:.*]] = fir.rebox %[[LOAD_A]] : (!fir.class>>) -> !fir.box>> ! CHECK: %{{.*}} = arith.select %[[CMPI]], %[[A_REBOX]], %[[LOAD_B]] : !fir.box>> end module