module classes_pair_potential use, intrinsic :: iso_fortran_env, only: DP => REAL64 use procedures_checks, only: check_potential_domain use classes_potential_expression, only: Abstract_Potential_Expression use types_potential_domain, only: Concrete_Potential_Domain use types_potential_domain_selector, only: Concrete_Potential_Domain_Selector implicit none private type, abstract, public :: Abstract_Pair_Potential private class(Abstract_Potential_Expression), allocatable :: expression type(Concrete_Potential_Domain) :: domain contains procedure(Abstract_construct), deferred :: construct procedure(Abstract_destroy), deferred :: destroy procedure :: get_min_distance => Abstract_get_min_distance procedure :: get_max_distance => Abstract_get_max_distance procedure(Abstract_meet), deferred :: meet end type Abstract_Pair_Potential abstract interface subroutine Abstract_construct(this, domain, expression) import :: Concrete_Potential_Domain, Abstract_Potential_Expression, Abstract_Pair_Potential class(Abstract_Pair_Potential), intent(out) :: this type(Concrete_Potential_Domain), intent(in) :: domain class(Abstract_Potential_Expression), intent(in) :: expression end subroutine Abstract_construct subroutine Abstract_destroy(this) import :: Abstract_Pair_Potential class(Abstract_Pair_Potential), intent(inout) :: this end subroutine Abstract_destroy pure subroutine Abstract_meet(this, overlap, energy, distance) import :: DP, Abstract_Pair_Potential class(Abstract_Pair_Potential), intent(in) :: this logical, intent(out) :: overlap real(DP), intent(out) :: energy real(DP), intent(in) :: distance end subroutine Abstract_meet end interface type, extends(Abstract_Pair_Potential), public :: Tabulated_Pair_Potential private real(DP), allocatable :: tabulation(:) contains procedure :: construct => Tabulated_construct procedure, private :: set_domain => Tabulated_set_domain procedure, private :: set_tabulation => Tabulated_set_tabulation procedure :: destroy => Tabulated_destroy procedure :: meet => Tabulated_meet end type Tabulated_Pair_Potential type, extends(Abstract_Pair_Potential), public :: Raw_Pair_Potential private real(DP) :: energy_domain_max = 0._DP contains procedure :: construct => Raw_construct procedure, private :: set_domain => Raw_set_domain procedure :: destroy => Raw_destroy procedure :: meet => Raw_meet end type Raw_Pair_Potential type, extends(Abstract_Pair_Potential), public :: Null_Pair_Potential contains procedure :: construct => Null_construct procedure :: destroy => Null_destroy procedure :: get_min_distance => Null_get_distance procedure :: get_max_distance => Null_get_distance procedure :: meet => Null_meet end type Null_Pair_Potential type, public :: Pair_Potential_Wrapper class(Abstract_Pair_Potential), allocatable :: potential end type Pair_Potential_Wrapper type, public :: Pair_Potential_Line type(Pair_Potential_Wrapper), allocatable :: line(:) end type Pair_Potential_Line contains !implementation Abstract_Pair_Potential pure real(DP) function Abstract_get_min_distance(this) result(min_distance) class(Abstract_Pair_Potential), intent(in) :: this min_distance = this%domain%min end function Abstract_get_min_distance pure real(DP) function Abstract_get_max_distance(this) result(max_distance) class(Abstract_Pair_Potential), intent(in) :: this max_distance = this%domain%max end function Abstract_get_max_distance !end implementation Abstract_Pair_Potential !implementation Tabulated_Pair_Potential subroutine Tabulated_construct(this, domain, expression) class(Tabulated_Pair_Potential), intent(out) :: this type(Concrete_Potential_Domain), intent(in) :: domain class(Abstract_Potential_Expression), intent(in) :: expression allocate(this%expression, source = expression) call this%set_domain(domain) call this%set_tabulation() end subroutine Tabulated_construct subroutine Tabulated_set_domain(this, domain) class(Tabulated_Pair_Potential), intent(inout) :: this type(Concrete_Potential_Domain), intent(in) :: domain type(Concrete_Potential_Domain_Selector) :: selector selector%check_max = .true. selector%check_max_over_box_edge = .false. selector%check_delta = .true. call check_potential_domain("Tabulated_Pair_Potential: set_domain", domain, selector) this%domain%min = domain%min this%domain%max = domain%max this%domain%delta = domain%delta end subroutine Tabulated_set_domain subroutine Tabulated_set_tabulation(this) class(Tabulated_Pair_Potential), intent(inout) :: this real(DP) :: distance_i integer :: i_min, i_max, i_distance i_min = int(this%domain%min/this%domain%delta) i_max = int(this%domain%max/this%domain%delta) + 1 allocate(this%tabulation(i_min:i_max)) do i_distance = i_min, i_max distance_i = real(i_distance, DP) * this%domain%delta this%tabulation(i_distance) = this%expression%get(distance_i) end do this%tabulation = this%tabulation - this%tabulation(i_max) end subroutine Tabulated_set_tabulation subroutine Tabulated_destroy(this) class(Tabulated_Pair_Potential), intent(inout) :: this if (allocated(this%tabulation)) deallocate(this%tabulation) if (allocated(this%expression)) deallocate(this%expression) end subroutine Tabulated_destroy pure subroutine Tabulated_meet(this, overlap, energy, distance) class(Tabulated_Pair_Potential), intent(in) :: this logical, intent(out) :: overlap real(DP), intent(out) :: energy real(DP), intent(in) :: distance real(DP) :: distance_i integer :: i_distance overlap = .false. energy = 0._DP if (distance <= this%domain%min) then overlap = .true. else if (distance < this%domain%max) then i_distance = int(distance/this%domain%delta) distance_i = real(i_distance, DP) * this%domain%delta energy = this%tabulation(i_distance) + & (distance - distance_i) * & (this%tabulation(i_distance + 1) - this%tabulation(i_distance)) / & this%domain%delta end if end subroutine Tabulated_meet !end implementation Tabulated_Pair_Potential !implementation Raw_Pair_Potential subroutine Raw_construct(this, domain, expression) class(Raw_Pair_Potential), intent(out) :: this type(Concrete_Potential_Domain), intent(in) :: domain class(Abstract_Potential_Expression), intent(in) :: expression allocate(this%expression, source = expression) call this%set_domain(domain) this%energy_domain_max = this%expression%get(this%domain%max) end subroutine Raw_construct subroutine Raw_set_domain(this, domain) class(Raw_Pair_Potential), intent(inout) :: this type(Concrete_Potential_Domain), intent(in) :: domain type(Concrete_Potential_Domain_Selector) :: selector selector%check_max = .true. selector%check_max_over_box_edge = .false. selector%check_delta = .false. call check_potential_domain("Raw_Pair_Potential: set_domain", domain, selector) this%domain%min = domain%min this%domain%max = domain%max this%domain%delta = 0._DP end subroutine Raw_set_domain subroutine Raw_destroy(this) class(Raw_Pair_Potential), intent(inout) :: this if (allocated(this%expression)) deallocate(this%expression) end subroutine Raw_destroy pure subroutine Raw_meet(this, overlap, energy, distance) class(Raw_Pair_Potential), intent(in) :: this logical, intent(out) :: overlap real(DP), intent(out) :: energy real(DP), intent(in) :: distance overlap = .false. energy = 0._DP if (distance <= this%domain%min) then overlap = .true. else if (distance < this%domain%max) then energy = this%expression%get(distance) - this%energy_domain_max end if end subroutine Raw_meet !end implementation Raw_Pair_Potential !implementation Null_Pair_Potential subroutine Null_construct(this, domain, expression) class(Null_Pair_Potential), intent(out) :: this type(Concrete_Potential_Domain), intent(in) :: domain class(Abstract_Potential_Expression), intent(in) :: expression end subroutine Null_construct subroutine Null_destroy(this) class(Null_Pair_Potential), intent(inout) :: this end subroutine Null_destroy pure real(DP) function Null_get_distance(this) result(distance) class(Null_Pair_Potential), intent(in) :: this distance = 0._DP end function Null_get_distance pure subroutine Null_meet(this, overlap, energy, distance) class(Null_Pair_Potential), intent(in) :: this logical, intent(out) :: overlap real(DP), intent(out) :: energy real(DP), intent(in) :: distance overlap = .false. energy = 0._DP end subroutine Null_meet !end implementation Null_Pair_Potential end module classes_pair_potential