|  | !===-- module/__fortran_builtins.f90 ---------------------------------------===! | 
|  | ! | 
|  | ! Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|  | ! See https://llvm.org/LICENSE.txt for license information. | 
|  | ! SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|  | ! | 
|  | !===------------------------------------------------------------------------===! | 
|  |  | 
|  | #include '../include/flang/Runtime/magic-numbers.h' | 
|  |  | 
|  | ! These naming shenanigans prevent names from Fortran intrinsic modules | 
|  | ! from being usable on INTRINSIC statements, and force the program | 
|  | ! to USE the standard intrinsic modules in order to access the | 
|  | ! standard names of the procedures. | 
|  | module __fortran_builtins | 
|  | implicit none | 
|  |  | 
|  | ! Set PRIVATE by default to explicitly only export what is meant | 
|  | ! to be exported by this MODULE. | 
|  | private | 
|  |  | 
|  | intrinsic :: __builtin_c_loc | 
|  | public :: __builtin_c_loc | 
|  |  | 
|  | intrinsic :: __builtin_c_devloc | 
|  | public :: __builtin_c_devloc | 
|  |  | 
|  | intrinsic :: __builtin_c_f_pointer | 
|  | public :: __builtin_c_f_pointer | 
|  |  | 
|  | intrinsic :: sizeof ! extension | 
|  | public :: sizeof | 
|  |  | 
|  | intrinsic :: selected_int_kind | 
|  | integer, parameter :: int64 = selected_int_kind(18) | 
|  |  | 
|  | type, bind(c), public :: __builtin_c_ptr | 
|  | integer(kind=int64), private :: __address | 
|  | end type | 
|  |  | 
|  | type, bind(c), public :: __builtin_c_funptr | 
|  | integer(kind=int64), private :: __address | 
|  | end type | 
|  |  | 
|  | type, public :: __builtin_event_type | 
|  | integer(kind=int64), private :: __count = -1 | 
|  | end type | 
|  |  | 
|  | type, public :: __builtin_notify_type | 
|  | integer(kind=int64), private :: __count = -1 | 
|  | end type | 
|  |  | 
|  | type, public :: __builtin_lock_type | 
|  | integer(kind=int64), private :: __count = -1 | 
|  | end type | 
|  |  | 
|  | type, public :: __builtin_ieee_flag_type | 
|  | integer(kind=1), private :: flag = 0 | 
|  | end type | 
|  |  | 
|  | type(__builtin_ieee_flag_type), parameter, public :: & | 
|  | __builtin_ieee_invalid = & | 
|  | __builtin_ieee_flag_type(_FORTRAN_RUNTIME_IEEE_INVALID), & | 
|  | __builtin_ieee_overflow = & | 
|  | __builtin_ieee_flag_type(_FORTRAN_RUNTIME_IEEE_OVERFLOW), & | 
|  | __builtin_ieee_divide_by_zero = & | 
|  | __builtin_ieee_flag_type(_FORTRAN_RUNTIME_IEEE_DIVIDE_BY_ZERO), & | 
|  | __builtin_ieee_underflow = & | 
|  | __builtin_ieee_flag_type(_FORTRAN_RUNTIME_IEEE_UNDERFLOW), & | 
|  | __builtin_ieee_inexact = & | 
|  | __builtin_ieee_flag_type(_FORTRAN_RUNTIME_IEEE_INEXACT), & | 
|  | __builtin_ieee_denorm = & ! extension | 
|  | __builtin_ieee_flag_type(_FORTRAN_RUNTIME_IEEE_DENORM) | 
|  |  | 
|  | type, public :: __builtin_ieee_round_type | 
|  | integer(kind=1), private :: mode = 0 | 
|  | end type | 
|  |  | 
|  | type(__builtin_ieee_round_type), parameter, public :: & | 
|  | __builtin_ieee_to_zero = & | 
|  | __builtin_ieee_round_type(_FORTRAN_RUNTIME_IEEE_TO_ZERO), & | 
|  | __builtin_ieee_nearest = & | 
|  | __builtin_ieee_round_type(_FORTRAN_RUNTIME_IEEE_NEAREST), & | 
|  | __builtin_ieee_up = & | 
|  | __builtin_ieee_round_type(_FORTRAN_RUNTIME_IEEE_UP), & | 
|  | __builtin_ieee_down = & | 
|  | __builtin_ieee_round_type(_FORTRAN_RUNTIME_IEEE_DOWN), & | 
|  | __builtin_ieee_away = & | 
|  | __builtin_ieee_round_type(_FORTRAN_RUNTIME_IEEE_AWAY), & | 
|  | __builtin_ieee_other = & | 
|  | __builtin_ieee_round_type(_FORTRAN_RUNTIME_IEEE_OTHER) | 
|  |  | 
|  | type, public :: __builtin_team_type | 
|  | integer(kind=int64), private :: __id = -1 | 
|  | end type | 
|  |  | 
|  | integer, parameter, public :: __builtin_atomic_int_kind = selected_int_kind(18) | 
|  | integer, parameter, public :: & | 
|  | __builtin_atomic_logical_kind = __builtin_atomic_int_kind | 
|  |  | 
|  | type, public :: __builtin_dim3 | 
|  | integer :: x=1, y=1, z=1 | 
|  | end type | 
|  | type(__builtin_dim3), public :: & | 
|  | __builtin_threadIdx, __builtin_blockDim, __builtin_blockIdx, & | 
|  | __builtin_gridDim | 
|  | integer, parameter, public :: __builtin_warpsize = 32 | 
|  |  | 
|  | type, public, bind(c) :: __builtin_c_devptr | 
|  | type(__builtin_c_ptr) :: cptr | 
|  | end type | 
|  |  | 
|  | intrinsic :: __builtin_fma | 
|  | intrinsic :: __builtin_ieee_int | 
|  | intrinsic :: __builtin_ieee_is_nan, __builtin_ieee_is_negative, & | 
|  | __builtin_ieee_is_normal | 
|  | intrinsic :: __builtin_ieee_next_after, __builtin_ieee_next_down, & | 
|  | __builtin_ieee_next_up | 
|  | intrinsic :: scale ! for ieee_scalb | 
|  | intrinsic :: __builtin_ieee_real | 
|  | intrinsic :: __builtin_ieee_selected_real_kind | 
|  | intrinsic :: __builtin_ieee_support_datatype, & | 
|  | __builtin_ieee_support_denormal, __builtin_ieee_support_divide, & | 
|  | __builtin_ieee_support_flag, __builtin_ieee_support_halting, & | 
|  | __builtin_ieee_support_inf, __builtin_ieee_support_io, & | 
|  | __builtin_ieee_support_nan, __builtin_ieee_support_rounding, & | 
|  | __builtin_ieee_support_sqrt, & | 
|  | __builtin_ieee_support_standard, __builtin_ieee_support_subnormal, & | 
|  | __builtin_ieee_support_underflow_control | 
|  | public :: __builtin_fma | 
|  | public :: __builtin_ieee_int | 
|  | public :: __builtin_ieee_is_nan, __builtin_ieee_is_negative, & | 
|  | __builtin_ieee_is_normal | 
|  | public :: __builtin_ieee_next_after, __builtin_ieee_next_down, & | 
|  | __builtin_ieee_next_up | 
|  | public :: __builtin_ieee_real | 
|  | public :: scale ! for ieee_scalb | 
|  | public :: __builtin_ieee_selected_real_kind | 
|  | public :: __builtin_ieee_support_datatype, & | 
|  | __builtin_ieee_support_denormal, __builtin_ieee_support_divide, & | 
|  | __builtin_ieee_support_flag, __builtin_ieee_support_halting, & | 
|  | __builtin_ieee_support_inf, __builtin_ieee_support_io, & | 
|  | __builtin_ieee_support_nan, __builtin_ieee_support_rounding, & | 
|  | __builtin_ieee_support_sqrt, & | 
|  | __builtin_ieee_support_standard, __builtin_ieee_support_subnormal, & | 
|  | __builtin_ieee_support_underflow_control | 
|  |  | 
|  | type :: __force_derived_type_instantiations | 
|  | type(__builtin_c_ptr) :: c_ptr | 
|  | type(__builtin_c_devptr) :: c_devptr | 
|  | type(__builtin_c_funptr) :: c_funptr | 
|  | type(__builtin_event_type) :: event_type | 
|  | type(__builtin_lock_type) :: lock_type | 
|  | type(__builtin_team_type) :: team_type | 
|  | end type | 
|  |  | 
|  | intrinsic :: __builtin_compiler_options, __builtin_compiler_version | 
|  | public :: __builtin_compiler_options, __builtin_compiler_version | 
|  |  | 
|  | interface operator(==) | 
|  | module procedure __builtin_c_ptr_eq | 
|  | end interface | 
|  | public :: operator(==) | 
|  |  | 
|  | interface operator(/=) | 
|  | module procedure __builtin_c_ptr_ne | 
|  | end interface | 
|  | public :: operator(/=) | 
|  |  | 
|  | interface __builtin_c_associated | 
|  | module procedure c_associated_c_ptr | 
|  | module procedure c_associated_c_funptr | 
|  | end interface | 
|  | public :: __builtin_c_associated | 
|  | !  private :: c_associated_c_ptr, c_associated_c_funptr | 
|  |  | 
|  | type(__builtin_c_ptr), parameter, public :: __builtin_c_null_ptr = __builtin_c_ptr(0) | 
|  | type(__builtin_c_funptr), parameter, public :: & | 
|  | __builtin_c_null_funptr = __builtin_c_funptr(0) | 
|  |  | 
|  | public :: __builtin_c_ptr_eq | 
|  | public :: __builtin_c_ptr_ne | 
|  | public :: __builtin_c_funloc | 
|  |  | 
|  | contains | 
|  |  | 
|  | elemental logical function __builtin_c_ptr_eq(x, y) | 
|  | type(__builtin_c_ptr), intent(in) :: x, y | 
|  | __builtin_c_ptr_eq = x%__address == y%__address | 
|  | end function | 
|  |  | 
|  | elemental logical function __builtin_c_ptr_ne(x, y) | 
|  | type(__builtin_c_ptr), intent(in) :: x, y | 
|  | __builtin_c_ptr_ne = x%__address /= y%__address | 
|  | end function | 
|  |  | 
|  | ! Semantics has some special-case code that allows c_funloc() | 
|  | ! to appear in a specification expression and exempts it | 
|  | ! from the requirement that "x" be a pure dummy procedure. | 
|  | pure function __builtin_c_funloc(x) | 
|  | type(__builtin_c_funptr) :: __builtin_c_funloc | 
|  | external :: x | 
|  | __builtin_c_funloc = __builtin_c_funptr(loc(x)) | 
|  | end function | 
|  |  | 
|  | pure logical function c_associated_c_ptr(c_ptr_1, c_ptr_2) | 
|  | type(__builtin_c_ptr), intent(in) :: c_ptr_1 | 
|  | type(__builtin_c_ptr), intent(in), optional :: c_ptr_2 | 
|  | if (c_ptr_1%__address == __builtin_c_null_ptr%__address) then | 
|  | c_associated_c_ptr = .false. | 
|  | else if (present(c_ptr_2)) then | 
|  | c_associated_c_ptr = c_ptr_1%__address == c_ptr_2%__address | 
|  | else | 
|  | c_associated_c_ptr = .true. | 
|  | end if | 
|  | end function c_associated_c_ptr | 
|  |  | 
|  | pure logical function c_associated_c_funptr(c_ptr_1, c_ptr_2) | 
|  | type(__builtin_c_funptr), intent(in) :: c_ptr_1 | 
|  | type(__builtin_c_funptr), intent(in), optional :: c_ptr_2 | 
|  | if (c_ptr_1%__address == __builtin_c_null_ptr%__address) then | 
|  | c_associated_c_funptr = .false. | 
|  | else if (present(c_ptr_2)) then | 
|  | c_associated_c_funptr = c_ptr_1%__address == c_ptr_2%__address | 
|  | else | 
|  | c_associated_c_funptr = .true. | 
|  | end if | 
|  | end function c_associated_c_funptr | 
|  |  | 
|  | end module |