mod_eigenfunctions.f08 Source File


Contents


Source Code

module mod_eigenfunctions
  use mod_global_variables, only: dp
  use mod_settings, only: settings_t
  use mod_background, only: background_t
  use mod_grid, only: grid_t
  use mod_base_efs, only: base_ef_t
  use mod_derived_efs, only: derived_ef_t, deallocate_derived_ef_module_variables
  use mod_derived_ef_names, only: create_and_set_derived_state_vector
  implicit none

  private

  type, public :: eigenfunctions_t
    type(settings_t), pointer, private :: settings
    type(grid_t), pointer, private :: grid
    type(background_t), pointer, private :: background
    type(base_ef_t), allocatable :: base_efs(:)
    type(derived_ef_t), allocatable :: derived_efs(:)
    logical, allocatable :: ef_written_flags(:)
    integer, allocatable :: ef_written_idxs(:)

  contains

    procedure, public :: initialise
    procedure, public :: assemble
    procedure, public :: delete

    procedure, private :: select_eigenfunctions_to_save
  end type eigenfunctions_t

  public :: new_eigenfunctions

contains

  function new_eigenfunctions(settings, grid, background) result(eigenfunctions)
    type(settings_t), target, intent(inout) :: settings
    type(grid_t), target, intent(in) :: grid
    type(background_t), target, intent(in) :: background
    type(eigenfunctions_t) :: eigenfunctions
    eigenfunctions%settings => settings
    eigenfunctions%grid => grid
    eigenfunctions%background => background
  end function new_eigenfunctions


  subroutine initialise(this, omega)
    class(eigenfunctions_t), intent(inout) :: this
    complex(dp), intent(in) :: omega(:)
    character(len=:), allocatable :: state_vector(:)
    character(len=:), allocatable :: derived_state_vector(:)
    integer :: i, nb_efs

    call this%select_eigenfunctions_to_save(omega)
    state_vector = this%settings%get_state_vector()
    nb_efs = size(this%ef_written_idxs)

    allocate(this%base_efs(size(state_vector)))
    do i = 1, size(this%base_efs)
      call this%base_efs(i)%initialise( &
        name=state_vector(i), ef_grid_size=size(this%grid%ef_grid), nb_efs=nb_efs &
      )
    end do
    deallocate(state_vector)

    if (.not. this%settings%io%write_derived_eigenfunctions) return

    derived_state_vector = create_and_set_derived_state_vector( &
      this%settings, this%background &
    )
    allocate(this%derived_efs(size(derived_state_vector)))
    do i = 1, size(this%derived_efs)
      call this%derived_efs(i)%initialise( &
        name=derived_state_vector(i), &
        ef_grid_size=size(this%grid%ef_grid), &
        nb_efs=nb_efs &
      )
    end do
    deallocate(derived_state_vector)
  end subroutine initialise


  subroutine assemble(this, right_eigenvectors)
    class(eigenfunctions_t), intent(inout) :: this
    complex(dp), intent(in) :: right_eigenvectors(:, :)
    integer :: i
    do i = 1, size(this%base_efs)
      call this%base_efs(i)%assemble( &
        settings=this%settings, &
        grid=this%grid, &
        idxs_to_assemble=this%ef_written_idxs, &
        right_eigenvectors=right_eigenvectors &
      )
    end do

    if (.not. this%settings%io%write_derived_eigenfunctions) return
    do i = 1, size(this%derived_efs)
      call this%derived_efs(i)%assemble( &
        settings=this%settings, &
        grid=this%grid, &
        background=this%background, &
        idxs_to_assemble=this%ef_written_idxs, &
        right_eigenvectors=right_eigenvectors &
      )
    end do
    call deallocate_derived_ef_module_variables()
  end subroutine assemble


  pure subroutine delete(this)
    class(eigenfunctions_t), intent(inout) :: this
    integer :: i
    if (allocated(this%base_efs)) then
      do i = 1, size(this%base_efs)
        call this%base_efs(i)%delete()
      end do
      deallocate(this%base_efs)
    end if
    if (allocated(this%derived_efs)) then
      do i = 1, size(this%derived_efs)
        call this%derived_efs(i)%delete()
      end do
      deallocate(this%derived_efs)
    end if
    if (allocated(this%ef_written_flags)) deallocate(this%ef_written_flags)
    if (allocated(this%ef_written_idxs)) deallocate(this%ef_written_idxs)
    nullify(this%settings)
    nullify(this%background)
    nullify(this%grid)
  end subroutine delete


  pure subroutine select_eigenfunctions_to_save(this, omega)
    class(eigenfunctions_t), intent(inout) :: this
    complex(dp), intent(in) :: omega(:)
    integer :: i

    allocate(this%ef_written_flags(size(omega)))
    if (this%settings%io%write_ef_subset) then
      this%ef_written_flags = eigenvalue_is_inside_subset_radius( &
        eigenvalue=omega, &
        radius=this%settings%io%ef_subset_radius, &
        center=this%settings%io%ef_subset_center &
      )
    else
      this%ef_written_flags = .true.
    end if
    ! extract indices of those eigenvalues that have their eigenfunctions written
    allocate(this%ef_written_idxs(count(this%ef_written_flags)))
    this%ef_written_idxs = pack([(i, i=1, size(omega))], this%ef_written_flags)
  end subroutine select_eigenfunctions_to_save


  elemental logical function eigenvalue_is_inside_subset_radius( &
    eigenvalue, radius, center &
  )
    complex(dp), intent(in) :: eigenvalue
    real(dp), intent(in) :: radius
    complex(dp), intent(in) :: center
    real(dp) :: distance_from_subset_center

    distance_from_subset_center = sqrt( &
      (aimag(eigenvalue) - aimag(center)) ** 2 &
      + (real(eigenvalue) - real(center)) ** 2 &
    )
    eigenvalue_is_inside_subset_radius = ( &
      distance_from_subset_center <= radius &
    )
  end function eigenvalue_is_inside_subset_radius

end module mod_eigenfunctions