2012-08-09 16:31:53 +05:30
!--------------------------------------------------------------------------------------------------
2013-09-14 16:29:35 +05:30
!> @author Franz Roters, Max-Planck-Institut für Eisenforschung GmbH
!> @author Philip Eisenlohr, Max-Planck-Institut für Eisenforschung GmbH
!> @author Christoph Kords, Max-Planck-Institut für Eisenforschung GmbH
!> @author Martin Diehl, Max-Planck-Institut für Eisenforschung GmbH
2014-05-08 20:25:19 +05:30
!> @author Luv Sharma, Max-Planck-Institut für Eisenforschung GmbH
2016-03-21 03:50:58 +05:30
!> @brief setting precision for real and int type
2012-08-09 16:31:53 +05:30
!--------------------------------------------------------------------------------------------------
2012-03-06 20:22:48 +05:30
module prec
2019-05-11 01:18:43 +05:30
use , intrinsic :: IEEE_arithmetic
2019-03-07 23:59:48 +05:30
2019-03-06 19:54:42 +05:30
implicit none
2019-09-23 12:23:56 +05:30
public
2020-02-29 19:30:47 +05:30
2019-03-06 19:54:42 +05:30
! https://software.intel.com/en-us/blogs/2017/03/27/doctor-fortran-in-it-takes-all-kinds
2020-02-29 19:30:47 +05:30
integer , parameter :: pReal = IEEE_selected_real_kind ( 15 , 307 ) !< number with 15 significant digits, up to 1e+-307 (typically 64 bit)
2019-03-06 20:22:52 +05:30
#if(INT==8)
2020-02-29 19:30:47 +05:30
integer , parameter :: pInt = selected_int_kind ( 18 ) !< number with at least up to +-1e18 (typically 64 bit)
2012-08-31 01:56:28 +05:30
#else
2020-02-29 19:30:47 +05:30
integer , parameter :: pInt = selected_int_kind ( 9 ) !< number with at least up to +-1e9 (typically 32 bit)
2012-08-28 21:38:17 +05:30
#endif
2020-02-29 19:30:47 +05:30
integer , parameter :: pLongInt = selected_int_kind ( 18 ) !< number with at least up to +-1e18 (typically 64 bit)
integer , parameter :: pStringLen = 256 !< default string length
integer , parameter :: pPathLen = 4096 !< maximum length of a path name on linux
2019-03-06 19:54:42 +05:30
2020-02-29 19:30:47 +05:30
real ( pReal ) , parameter :: tol_math_check = 1.0e-8_pReal !< tolerance for internal math self-checks (rotation)
2019-03-06 19:54:42 +05:30
2020-02-29 19:30:47 +05:30
type :: group_float !< variable length datatype used for storage of state
2019-03-06 19:54:42 +05:30
real ( pReal ) , dimension ( : ) , pointer :: p
end type group_float
! http://stackoverflow.com/questions/3948210/can-i-have-a-pointer-to-an-item-in-an-allocatable-array
2020-02-29 19:30:47 +05:30
type :: tState
2019-03-07 23:59:48 +05:30
integer :: &
sizeState = 0 , & !< size of state
sizeDotState = 0 , & !< size of dot state, i.e. state(1:sizeDot) follows time evolution by dotState rates
offsetDeltaState = 0 , & !< index offset of delta state
2019-12-11 04:40:02 +05:30
sizeDeltaState = 0 !< size of delta state, i.e. state(offset+1:offset+sizeDelta) follows time evolution by deltaState increments
2019-03-06 19:54:42 +05:30
real ( pReal ) , pointer , dimension ( : ) , contiguous :: &
2020-03-15 14:21:40 +05:30
atol
2019-03-07 23:59:48 +05:30
real ( pReal ) , pointer , dimension ( : , : ) , contiguous :: & ! a pointer is needed here because we might point to state/doState. However, they will never point to something, but are rather allocated and, hence, contiguous
2019-03-06 19:54:42 +05:30
state0 , &
state , & !< state
dotState , & !< rate of state change
deltaState !< increment of state change
real ( pReal ) , allocatable , dimension ( : , : ) :: &
partionedState0 , &
2020-04-01 01:32:53 +05:30
subState0
2019-03-06 19:54:42 +05:30
end type
2020-02-29 19:30:47 +05:30
type , extends ( tState ) :: tPlasticState
2019-03-07 23:59:48 +05:30
logical :: &
2019-03-06 19:54:42 +05:30
nonlocal = . false .
real ( pReal ) , pointer , dimension ( : , : ) :: &
2020-04-01 13:30:12 +05:30
slipRate !< slip rate
2019-03-06 19:54:42 +05:30
end type
2020-02-29 19:30:47 +05:30
type :: tSourceState
2019-03-06 19:54:42 +05:30
type ( tState ) , dimension ( : ) , allocatable :: p !< tState for each active source mechanism in a phase
end type
2019-03-07 23:59:48 +05:30
2020-02-29 19:30:47 +05:30
type :: tHomogMapping
2019-03-07 23:59:48 +05:30
integer , pointer , dimension ( : , : ) :: p
end type
2019-03-06 19:54:42 +05:30
2019-03-07 23:59:48 +05:30
real ( pReal ) , private , parameter :: PREAL_EPSILON = epsilon ( 0.0_pReal ) !< minimum positive number such that 1.0 + EPSILON /= 1.0.
2019-03-07 15:32:27 +05:30
real ( pReal ) , private , parameter :: PREAL_MIN = tiny ( 0.0_pReal ) !< smallest normalized floating point number
2019-03-06 19:54:42 +05:30
2020-02-29 19:30:47 +05:30
integer , dimension ( 0 ) , parameter :: &
2020-01-02 23:14:51 +05:30
emptyIntArray = [ integer :: ]
2020-02-29 19:30:47 +05:30
real ( pReal ) , dimension ( 0 ) , parameter :: &
2020-01-02 23:14:51 +05:30
emptyRealArray = [ real ( pReal ) :: ]
2020-02-29 19:30:47 +05:30
character ( len = pStringLen ) , dimension ( 0 ) , parameter :: &
2020-01-02 23:14:51 +05:30
emptyStringArray = [ character ( len = pStringLen ) :: ]
2019-12-21 15:49:33 +05:30
2019-09-23 12:23:56 +05:30
private :: &
2020-05-16 20:35:03 +05:30
selfTest
2019-03-07 23:59:48 +05:30
2012-03-06 20:22:48 +05:30
contains
2012-08-28 21:38:17 +05:30
2015-04-13 15:32:52 +05:30
2012-08-09 16:31:53 +05:30
!--------------------------------------------------------------------------------------------------
2017-05-04 04:02:44 +05:30
!> @brief reporting precision
2012-08-09 16:31:53 +05:30
!--------------------------------------------------------------------------------------------------
2012-03-06 20:22:48 +05:30
subroutine prec_init
2015-04-13 15:32:52 +05:30
2019-03-06 19:54:42 +05:30
write ( 6 , '(/,a)' ) ' <<<+- prec init -+>>>'
2014-10-10 18:38:34 +05:30
2019-03-07 23:59:48 +05:30
write ( 6 , '(a,i3)' ) ' Size of integer in bit: ' , bit_size ( 0 )
write ( 6 , '(a,i19)' ) ' Maximum value: ' , huge ( 0 )
2019-03-06 19:54:42 +05:30
write ( 6 , '(/,a,i3)' ) ' Size of float in bit: ' , storage_size ( 0.0_pReal )
write ( 6 , '(a,e10.3)' ) ' Maximum value: ' , huge ( 0.0_pReal )
write ( 6 , '(a,e10.3)' ) ' Minimum value: ' , tiny ( 0.0_pReal )
write ( 6 , '(a,i3)' ) ' Decimal precision: ' , precision ( 0.0_pReal )
2014-04-15 14:50:38 +05:30
2020-05-16 20:35:03 +05:30
call selfTest
2019-03-07 23:59:48 +05:30
2012-03-06 20:22:48 +05:30
end subroutine prec_init
2009-08-31 20:39:15 +05:30
2015-04-13 15:32:52 +05:30
2016-03-06 02:07:22 +05:30
!--------------------------------------------------------------------------------------------------
2016-05-29 14:15:03 +05:30
!> @brief equality comparison for float with double precision
2016-03-21 03:50:58 +05:30
! replaces "==" but for certain (relative) tolerance. Counterpart to dNeq
2016-10-18 23:20:05 +05:30
! https://randomascii.wordpress.com/2012/02/25/comparing-floating-point-numbers-2012-edition/
2017-11-21 13:48:03 +05:30
! AlmostEqualRelative
2016-03-06 02:07:22 +05:30
!--------------------------------------------------------------------------------------------------
logical elemental pure function dEq ( a , b , tol )
2016-05-25 11:22:56 +05:30
2019-03-06 20:27:42 +05:30
real ( pReal ) , intent ( in ) :: a , b
real ( pReal ) , intent ( in ) , optional :: tol
real ( pReal ) :: eps
2019-03-07 23:59:48 +05:30
2019-03-06 20:27:42 +05:30
if ( present ( tol ) ) then
eps = tol
else
2019-03-07 15:32:27 +05:30
eps = PREAL_EPSILON * maxval ( abs ( [ a , b ] ) )
2019-03-06 20:27:42 +05:30
endif
2019-09-23 21:59:38 +05:30
dEq = merge ( . True . , . False . , abs ( a - b ) < = eps )
2016-05-25 11:22:56 +05:30
2016-03-06 02:07:22 +05:30
end function dEq
!--------------------------------------------------------------------------------------------------
2016-05-29 14:15:03 +05:30
!> @brief inequality comparison for float with double precision
2016-03-21 03:50:58 +05:30
! replaces "!=" but for certain (relative) tolerance. Counterpart to dEq
2016-10-18 23:20:05 +05:30
! https://randomascii.wordpress.com/2012/02/25/comparing-floating-point-numbers-2012-edition/
2017-11-21 13:48:03 +05:30
! AlmostEqualRelative NOT
2016-03-06 02:07:22 +05:30
!--------------------------------------------------------------------------------------------------
logical elemental pure function dNeq ( a , b , tol )
2016-05-25 11:22:56 +05:30
2019-03-06 20:27:42 +05:30
real ( pReal ) , intent ( in ) :: a , b
real ( pReal ) , intent ( in ) , optional :: tol
2019-03-07 23:59:48 +05:30
2019-03-06 20:27:42 +05:30
if ( present ( tol ) ) then
2019-09-23 12:23:56 +05:30
dNeq = . not . dEq ( a , b , tol )
2019-03-06 20:27:42 +05:30
else
2019-09-23 12:23:56 +05:30
dNeq = . not . dEq ( a , b )
2019-03-06 20:27:42 +05:30
endif
2016-03-06 02:07:22 +05:30
end function dNeq
2016-06-30 14:00:40 +05:30
!--------------------------------------------------------------------------------------------------
2016-10-18 23:20:05 +05:30
!> @brief equality to 0 comparison for float with double precision
2017-11-21 13:48:03 +05:30
! replaces "==0" but everything not representable as a normal number is treated as 0. Counterpart to dNeq0
! https://de.mathworks.com/help/matlab/ref/realmin.html
! https://docs.oracle.com/cd/E19957-01/806-3568/ncg_math.html
2016-06-30 14:00:40 +05:30
!--------------------------------------------------------------------------------------------------
logical elemental pure function dEq0 ( a , tol )
2019-03-06 20:27:42 +05:30
real ( pReal ) , intent ( in ) :: a
real ( pReal ) , intent ( in ) , optional :: tol
real ( pReal ) :: eps
2019-03-07 23:59:48 +05:30
2019-03-06 20:27:42 +05:30
if ( present ( tol ) ) then
eps = tol
else
2019-03-07 15:32:27 +05:30
eps = PREAL_MIN * 1 0.0_pReal
2019-03-06 20:27:42 +05:30
endif
2019-09-23 21:59:38 +05:30
dEq0 = merge ( . True . , . False . , abs ( a ) < = eps )
2016-06-30 14:00:40 +05:30
end function dEq0
!--------------------------------------------------------------------------------------------------
2016-10-18 23:20:05 +05:30
!> @brief inequality to 0 comparison for float with double precision
2017-11-21 13:48:03 +05:30
! replaces "!=0" but everything not representable as a normal number is treated as 0. Counterpart to dEq0
! https://de.mathworks.com/help/matlab/ref/realmin.html
! https://docs.oracle.com/cd/E19957-01/806-3568/ncg_math.html
2016-06-30 14:00:40 +05:30
!--------------------------------------------------------------------------------------------------
logical elemental pure function dNeq0 ( a , tol )
2019-03-06 20:27:42 +05:30
real ( pReal ) , intent ( in ) :: a
real ( pReal ) , intent ( in ) , optional :: tol
2019-03-07 23:59:48 +05:30
2019-03-06 20:27:42 +05:30
if ( present ( tol ) ) then
2019-09-23 12:23:56 +05:30
dNeq0 = . not . dEq0 ( a , tol )
2019-03-06 20:27:42 +05:30
else
2019-09-23 12:23:56 +05:30
dNeq0 = . not . dEq0 ( a )
2019-03-06 20:27:42 +05:30
endif
2016-06-30 14:00:40 +05:30
end function dNeq0
2016-05-29 14:15:03 +05:30
!--------------------------------------------------------------------------------------------------
!> @brief equality comparison for complex with double precision
! replaces "==" but for certain (relative) tolerance. Counterpart to cNeq
2016-10-18 23:20:05 +05:30
! https://randomascii.wordpress.com/2012/02/25/comparing-floating-point-numbers-2012-edition/
2016-05-29 14:15:03 +05:30
! probably a component wise comparison would be more accurate than the comparsion of the absolute
! value
!--------------------------------------------------------------------------------------------------
logical elemental pure function cEq ( a , b , tol )
2019-03-06 20:27:42 +05:30
complex ( pReal ) , intent ( in ) :: a , b
real ( pReal ) , intent ( in ) , optional :: tol
real ( pReal ) :: eps
2019-03-07 23:59:48 +05:30
2019-03-06 20:27:42 +05:30
if ( present ( tol ) ) then
eps = tol
else
2019-03-07 15:32:27 +05:30
eps = PREAL_EPSILON * maxval ( abs ( [ a , b ] ) )
2019-03-06 20:27:42 +05:30
endif
2019-09-23 21:59:38 +05:30
cEq = merge ( . True . , . False . , abs ( a - b ) < = eps )
2016-05-29 14:15:03 +05:30
end function cEq
!--------------------------------------------------------------------------------------------------
!> @brief inequality comparison for complex with double precision
! replaces "!=" but for certain (relative) tolerance. Counterpart to cEq
2016-10-18 23:20:05 +05:30
! https://randomascii.wordpress.com/2012/02/25/comparing-floating-point-numbers-2012-edition/
2016-05-29 14:15:03 +05:30
! probably a component wise comparison would be more accurate than the comparsion of the absolute
! value
!--------------------------------------------------------------------------------------------------
logical elemental pure function cNeq ( a , b , tol )
2019-03-06 20:27:42 +05:30
complex ( pReal ) , intent ( in ) :: a , b
real ( pReal ) , intent ( in ) , optional :: tol
2019-03-07 23:59:48 +05:30
2019-03-06 20:27:42 +05:30
if ( present ( tol ) ) then
2019-09-23 12:23:56 +05:30
cNeq = . not . cEq ( a , b , tol )
2019-03-06 20:27:42 +05:30
else
2019-09-23 12:23:56 +05:30
cNeq = . not . cEq ( a , b )
2019-03-06 20:27:42 +05:30
endif
2016-05-29 14:15:03 +05:30
end function cNeq
2019-09-23 12:23:56 +05:30
!--------------------------------------------------------------------------------------------------
2020-03-14 21:59:08 +05:30
!> @brief check correctness of some prec functions
2019-09-23 12:23:56 +05:30
!--------------------------------------------------------------------------------------------------
2020-05-16 20:35:03 +05:30
subroutine selfTest
2019-09-23 12:23:56 +05:30
integer , allocatable , dimension ( : ) :: realloc_lhs_test
real ( pReal ) , dimension ( 2 ) :: r
external :: &
quit
2020-03-27 02:16:28 +05:30
2019-09-23 12:23:56 +05:30
call random_number ( r )
r = r / minval ( r )
if ( . not . all ( dEq ( r , r + PREAL_EPSILON ) ) ) call quit ( 9000 )
if ( dEq ( r ( 1 ) , r ( 2 ) ) . and . dNeq ( r ( 1 ) , r ( 2 ) ) ) call quit ( 9000 )
2019-09-23 22:01:14 +05:30
if ( . not . all ( dEq0 ( r - ( r + PREAL_MIN ) ) ) ) call quit ( 9000 )
2020-03-27 02:16:28 +05:30
2019-09-23 12:23:56 +05:30
realloc_lhs_test = [ 1 , 2 ]
if ( any ( realloc_lhs_test / = [ 1 , 2 ] ) ) call quit ( 9000 )
2020-03-27 02:16:28 +05:30
2020-05-16 20:35:03 +05:30
end subroutine selfTest
2019-09-23 12:23:56 +05:30
2012-03-06 20:22:48 +05:30
end module prec