forked from OSchip/llvm-project
148 lines
5.4 KiB
Fortran
148 lines
5.4 KiB
Fortran
! RUN: bbc -emit-fir -outline-intrinsics %s -o - | FileCheck %s
|
|
|
|
! Test statement function lowering
|
|
|
|
! Simple case
|
|
! CHECK-LABEL: func @_QPtest_stmt_0(
|
|
! CHECK-SAME: %{{.*}}: !fir.ref<f32>{{.*}}) -> f32
|
|
real function test_stmt_0(x)
|
|
real :: x, func, arg
|
|
func(arg) = arg + 0.123456
|
|
|
|
! CHECK-DAG: %[[x:.*]] = fir.load %arg0
|
|
! CHECK-DAG: %[[cst:.*]] = arith.constant 1.234560e-01
|
|
! CHECK: %[[eval:.*]] = arith.addf %[[x]], %[[cst]]
|
|
! CHECK: fir.store %[[eval]] to %[[resmem:.*]] : !fir.ref<f32>
|
|
test_stmt_0 = func(x)
|
|
|
|
! CHECK: %[[res:.*]] = fir.load %[[resmem]]
|
|
! CHECK: return %[[res]]
|
|
end function
|
|
|
|
! Check this is not lowered as a simple macro: e.g. argument is only
|
|
! evaluated once even if it appears in several placed inside the
|
|
! statement function expression
|
|
! CHECK-LABEL: func @_QPtest_stmt_only_eval_arg_once() -> f32
|
|
real(4) function test_stmt_only_eval_arg_once()
|
|
real(4) :: only_once, x1
|
|
func(x1) = x1 + x1
|
|
! CHECK: %[[x2:.*]] = fir.alloca f32 {adapt.valuebyref}
|
|
! CHECK: %[[x1:.*]] = fir.call @_QPonly_once()
|
|
! Note: using -emit-fir, so the faked pass-by-reference is exposed
|
|
! CHECK: fir.store %[[x1]] to %[[x2]]
|
|
! CHECK: addf %{{.*}}, %{{.*}}
|
|
test_stmt_only_eval_arg_once = func(only_once())
|
|
end function
|
|
|
|
! Test nested statement function (note that they cannot be recursively
|
|
! nested as per F2018 C1577).
|
|
real function test_stmt_1(x, a)
|
|
real :: y, a, b, foo
|
|
real :: func1, arg1, func2, arg2
|
|
real :: res1, res2
|
|
func1(arg1) = a + foo(arg1)
|
|
func2(arg2) = func1(arg2) + b
|
|
! CHECK-DAG: %[[bmem:.*]] = fir.alloca f32 {{{.*}}uniq_name = "{{.*}}Eb"}
|
|
! CHECK-DAG: %[[res1:.*]] = fir.alloca f32 {{{.*}}uniq_name = "{{.*}}Eres1"}
|
|
! CHECK-DAG: %[[res2:.*]] = fir.alloca f32 {{{.*}}uniq_name = "{{.*}}Eres2"}
|
|
|
|
b = 5
|
|
|
|
! CHECK-DAG: %[[cst_8:.*]] = arith.constant 8.000000e+00
|
|
! CHECK-DAG: fir.store %[[cst_8]] to %[[tmp1:.*]] : !fir.ref<f32>
|
|
! CHECK-DAG: %[[foocall1:.*]] = fir.call @_QPfoo(%[[tmp1]])
|
|
! CHECK-DAG: %[[aload1:.*]] = fir.load %arg1
|
|
! CHECK: %[[add1:.*]] = arith.addf %[[aload1]], %[[foocall1]]
|
|
! CHECK: fir.store %[[add1]] to %[[res1]]
|
|
res1 = func1(8.)
|
|
|
|
! CHECK-DAG: %[[a2:.*]] = fir.load %arg1
|
|
! CHECK-DAG: %[[foocall2:.*]] = fir.call @_QPfoo(%arg0)
|
|
! CHECK-DAG: %[[add2:.*]] = arith.addf %[[a2]], %[[foocall2]]
|
|
! CHECK-DAG: %[[b:.*]] = fir.load %[[bmem]]
|
|
! CHECK: %[[add3:.*]] = arith.addf %[[add2]], %[[b]]
|
|
! CHECK: fir.store %[[add3]] to %[[res2]]
|
|
res2 = func2(x)
|
|
|
|
! CHECK-DAG: %[[res12:.*]] = fir.load %[[res1]]
|
|
! CHECK-DAG: %[[res22:.*]] = fir.load %[[res2]]
|
|
! CHECK: = arith.addf %[[res12]], %[[res22]] : f32
|
|
test_stmt_1 = res1 + res2
|
|
! CHECK: return %{{.*}} : f32
|
|
end function
|
|
|
|
|
|
! Test statement functions with no argument.
|
|
! Test that they are not pre-evaluated.
|
|
! CHECK-LABEL: func @_QPtest_stmt_no_args
|
|
real function test_stmt_no_args(x, y)
|
|
func() = x + y
|
|
! CHECK: addf
|
|
a = func()
|
|
! CHECK: fir.call @_QPfoo_may_modify_xy
|
|
call foo_may_modify_xy(x, y)
|
|
! CHECK: addf
|
|
! CHECK: addf
|
|
test_stmt_no_args = func() + a
|
|
end function
|
|
|
|
! Test statement function with character arguments
|
|
! CHECK-LABEL: @_QPtest_stmt_character
|
|
integer function test_stmt_character(c, j)
|
|
integer :: i, j, func, argj
|
|
character(10) :: c, argc
|
|
! CHECK-DAG: %[[unboxed:.*]]:2 = fir.unboxchar %arg0 :
|
|
! CHECK-DAG: %[[c10:.*]] = arith.constant 10 :
|
|
! CHECK: %[[c10_cast:.*]] = fir.convert %[[c10]] : (i32) -> index
|
|
! CHECK: %[[c:.*]] = fir.emboxchar %[[unboxed]]#0, %[[c10_cast]]
|
|
|
|
func(argc, argj) = len_trim(argc, 4) + argj
|
|
! CHECK: addi %{{.*}}, %{{.*}} : i
|
|
test_stmt_character = func(c, j)
|
|
end function
|
|
|
|
! Test statement function with a character actual argument whose
|
|
! length may be different than the dummy length (the dummy length
|
|
! must be used inside the statement function).
|
|
! CHECK-LABEL: @_QPtest_stmt_character_with_different_length(
|
|
! CHECK-SAME: %[[arg0:.*]]: !fir.boxchar<1>
|
|
integer function test_stmt_character_with_different_length(c)
|
|
integer :: func, ifoo
|
|
character(10) :: argc
|
|
character(*) :: c
|
|
! CHECK-DAG: %[[unboxed:.*]]:2 = fir.unboxchar %[[arg0]] :
|
|
! CHECK-DAG: %[[c10:.*]] = arith.constant 10 :
|
|
! CHECK: %[[c10_cast:.*]] = fir.convert %[[c10]] : (i32) -> index
|
|
! CHECK: %[[argc:.*]] = fir.emboxchar %[[unboxed]]#0, %[[c10_cast]]
|
|
! CHECK: fir.call @_QPifoo(%[[argc]]) : (!fir.boxchar<1>) -> i32
|
|
func(argc) = ifoo(argc)
|
|
test_stmt_character = func(c)
|
|
end function
|
|
|
|
! CHECK-LABEL: @_QPtest_stmt_character_with_different_length_2(
|
|
! CHECK-SAME: %[[arg0:.*]]: !fir.boxchar<1>{{.*}}, %[[arg1:.*]]: !fir.ref<i32>
|
|
integer function test_stmt_character_with_different_length_2(c, n)
|
|
integer :: func, ifoo
|
|
character(n) :: argc
|
|
character(*) :: c
|
|
! CHECK: %[[unboxed:.*]]:2 = fir.unboxchar %[[arg0]] :
|
|
! CHECK: fir.load %[[arg1]] : !fir.ref<i32>
|
|
! CHECK: %[[n:.*]] = fir.load %[[arg1]] : !fir.ref<i32>
|
|
! CHECK: %[[n_is_positive:.*]] = arith.cmpi sgt, %[[n]], %c0{{.*}} : i32
|
|
! CHECK: %[[len:.*]] = arith.select %[[n_is_positive]], %[[n]], %c0{{.*}} : i32
|
|
! CHECK: %[[lenCast:.*]] = fir.convert %[[len]] : (i32) -> index
|
|
! CHECK: %[[argc:.*]] = fir.emboxchar %[[unboxed]]#0, %[[lenCast]] : (!fir.ref<!fir.char<1,?>>, index) -> !fir.boxchar<1>
|
|
! CHECK: fir.call @_QPifoo(%[[argc]]) : (!fir.boxchar<1>) -> i32
|
|
func(argc) = ifoo(argc)
|
|
test_stmt_character = func(c)
|
|
end function
|
|
|
|
! issue #247
|
|
! CHECK-LABEL: @_QPbug247
|
|
subroutine bug247(r)
|
|
I(R) = R
|
|
! CHECK: fir.call {{.*}}OutputInteger
|
|
PRINT *, I(2.5)
|
|
! CHECK: fir.call {{.*}}EndIo
|
|
END subroutine bug247
|