sll_m_bsplines_uniform.F90 Source File


This file depends on

sourcefile~~sll_m_bsplines_uniform.f90~~EfferentGraph sourcefile~sll_m_bsplines_uniform.f90 sll_m_bsplines_uniform.F90 sourcefile~sll_m_assert.f90 sll_m_assert.F90 sourcefile~sll_m_bsplines_uniform.f90->sourcefile~sll_m_assert.f90 sourcefile~sll_m_bsplines_base.f90 sll_m_bsplines_base.F90 sourcefile~sll_m_bsplines_uniform.f90->sourcefile~sll_m_bsplines_base.f90 sourcefile~sll_m_errors.f90 sll_m_errors.F90 sourcefile~sll_m_bsplines_uniform.f90->sourcefile~sll_m_errors.f90 sourcefile~sll_m_working_precision.f90 sll_m_working_precision.F90 sourcefile~sll_m_bsplines_uniform.f90->sourcefile~sll_m_working_precision.f90 sourcefile~globals.f90 globals.F90 sourcefile~sll_m_assert.f90->sourcefile~globals.f90 sourcefile~sll_m_bsplines_base.f90->sourcefile~sll_m_assert.f90 sourcefile~sll_m_bsplines_base.f90->sourcefile~sll_m_working_precision.f90

Files dependent on this one

sourcefile~~sll_m_bsplines_uniform.f90~~AfferentGraph sourcefile~sll_m_bsplines_uniform.f90 sll_m_bsplines_uniform.F90 sourcefile~sll_m_bsplines.f90 sll_m_bsplines.F90 sourcefile~sll_m_bsplines.f90->sourcefile~sll_m_bsplines_uniform.f90 sourcefile~cubic_spline.f90 cubic_spline.F90 sourcefile~cubic_spline.f90->sourcefile~sll_m_bsplines.f90 sourcefile~rprofile_bspline.f90 rprofile_bspline.F90 sourcefile~rprofile_bspline.f90->sourcefile~sll_m_bsplines.f90 sourcefile~sbase.f90 sbase.F90 sourcefile~sbase.f90->sourcefile~sll_m_bsplines.f90 sourcefile~analyze.f90 analyze.F90 sourcefile~analyze.f90->sourcefile~cubic_spline.f90 sourcefile~vmec.f90 vmec.F90 sourcefile~analyze.f90->sourcefile~vmec.f90 sourcefile~vmec_readin.f90 vmec_readin.F90 sourcefile~analyze.f90->sourcefile~vmec_readin.f90 sourcefile~vmec_vars.f90 vmec_vars.F90 sourcefile~analyze.f90->sourcefile~vmec_vars.f90 sourcefile~mhd3d_vars.f90 mhd3d_vars.F90 sourcefile~analyze.f90->sourcefile~mhd3d_vars.f90 sourcefile~base.f90 base.F90 sourcefile~base.f90->sourcefile~sbase.f90 sourcefile~mhd3d.f90 mhd3d.F90 sourcefile~mhd3d.f90->sourcefile~cubic_spline.f90 sourcefile~mhd3d.f90->sourcefile~rprofile_bspline.f90 sourcefile~mhd3d.f90->sourcefile~analyze.f90 sourcefile~mhd3d.f90->sourcefile~base.f90 sourcefile~mhd3d.f90->sourcefile~vmec.f90 sourcefile~mhd3d.f90->sourcefile~vmec_readin.f90 sourcefile~mhd3d.f90->sourcefile~vmec_vars.f90 sourcefile~lambda_solve.f90 lambda_solve.F90 sourcefile~mhd3d.f90->sourcefile~lambda_solve.f90 sourcefile~mhd3d_evalfunc.f90 mhd3d_evalfunc.F90 sourcefile~mhd3d.f90->sourcefile~mhd3d_evalfunc.f90 sourcefile~mhd3d_minimize.f90 mhd3d_minimize.F90 sourcefile~mhd3d.f90->sourcefile~mhd3d_minimize.f90 sourcefile~mhd3d.f90->sourcefile~mhd3d_vars.f90 sourcefile~restart.f90 restart.F90 sourcefile~mhd3d.f90->sourcefile~restart.f90 sourcefile~readstate.f90 readstate.F90 sourcefile~readstate.f90->sourcefile~sbase.f90 sourcefile~readstate.f90->sourcefile~base.f90 sourcefile~readstate_vars.f90 readstate_vars.F90 sourcefile~readstate.f90->sourcefile~readstate_vars.f90 sourcefile~readstate_vars.f90->sourcefile~sbase.f90 sourcefile~readstate_vars.f90->sourcefile~base.f90 sourcefile~vmec.f90->sourcefile~cubic_spline.f90 sourcefile~vmec.f90->sourcefile~rprofile_bspline.f90 sourcefile~vmec.f90->sourcefile~vmec_readin.f90 sourcefile~vmec.f90->sourcefile~vmec_vars.f90 sourcefile~vmec_readin.f90->sourcefile~cubic_spline.f90 sourcefile~vmec_vars.f90->sourcefile~cubic_spline.f90 sourcefile~gvec_post.f90 gvec_post.F90 sourcefile~gvec_post.f90->sourcefile~analyze.f90 sourcefile~gvec_post.f90->sourcefile~mhd3d.f90 sourcefile~gvec_post.f90->sourcefile~readstate_vars.f90 sourcefile~gvec_post.f90->sourcefile~mhd3d_evalfunc.f90 sourcefile~gvec_post.f90->sourcefile~restart.f90 sourcefile~gvec_to_jorek.f90 gvec_to_jorek.F90 sourcefile~gvec_to_jorek.f90->sourcefile~base.f90 sourcefile~gvec_to_jorek.f90->sourcefile~readstate.f90 sourcefile~gvec_to_jorek.f90->sourcefile~readstate_vars.f90 sourcefile~gvec_to_jorek_vars.f90 gvec_to_jorek_vars.F90 sourcefile~gvec_to_jorek.f90->sourcefile~gvec_to_jorek_vars.f90 sourcefile~gvec_to_jorek_vars.f90->sourcefile~base.f90 sourcefile~lambda_solve.f90->sourcefile~base.f90 sourcefile~mhd3d_evalfunc.f90->sourcefile~base.f90 sourcefile~mhd3d_evalfunc.f90->sourcefile~mhd3d_vars.f90 sourcefile~mhd3d_minimize.f90->sourcefile~analyze.f90 sourcefile~mhd3d_minimize.f90->sourcefile~mhd3d_evalfunc.f90 sourcefile~mhd3d_minimize.f90->sourcefile~restart.f90 sourcefile~mhd3d_vars.f90->sourcefile~base.f90 sourcefile~restart.f90->sourcefile~base.f90 sourcefile~restart.f90->sourcefile~readstate.f90 sourcefile~restart.f90->sourcefile~readstate_vars.f90 sourcefile~restart.f90->sourcefile~mhd3d_evalfunc.f90 sourcefile~restart.f90->sourcefile~mhd3d_vars.f90 sourcefile~run.f90 run.F90 sourcefile~run.f90->sourcefile~analyze.f90 sourcefile~run.f90->sourcefile~restart.f90 sourcefile~rungvec.f90 rungvec.F90 sourcefile~run.f90->sourcefile~rungvec.f90 sourcefile~rungvec.f90->sourcefile~analyze.f90 sourcefile~rungvec.f90->sourcefile~mhd3d.f90 sourcefile~rungvec.f90->sourcefile~restart.f90 sourcefile~sfl_boozer.f90 sfl_boozer.F90 sourcefile~sfl_boozer.f90->sourcefile~base.f90 sourcefile~sfl_boozer.f90->sourcefile~lambda_solve.f90 sourcefile~state.f90 state.F90 sourcefile~state.f90->sourcefile~analyze.f90 sourcefile~state.f90->sourcefile~base.f90 sourcefile~state.f90->sourcefile~mhd3d.f90 sourcefile~state.f90->sourcefile~readstate_vars.f90 sourcefile~state.f90->sourcefile~mhd3d_vars.f90 sourcefile~state.f90->sourcefile~restart.f90 sourcefile~state.f90->sourcefile~sfl_boozer.f90 sourcefile~transform_sfl.f90 transform_sfl.F90 sourcefile~state.f90->sourcefile~transform_sfl.f90 sourcefile~transform_sfl.f90->sourcefile~base.f90 sourcefile~transform_sfl.f90->sourcefile~sfl_boozer.f90 sourcefile~convert_gvec_to_jorek.f90 convert_gvec_to_jorek.F90 sourcefile~convert_gvec_to_jorek.f90->sourcefile~gvec_to_jorek.f90 sourcefile~gvec.f90 gvec.F90 sourcefile~gvec.f90->sourcefile~rungvec.f90 sourcefile~gvec_to_castor3d_vars.f90 gvec_to_castor3d_vars.F90 sourcefile~gvec_to_castor3d_vars.f90->sourcefile~transform_sfl.f90 sourcefile~gvec_to_gene_vars.f90 gvec_to_gene_vars.F90 sourcefile~gvec_to_gene_vars.f90->sourcefile~transform_sfl.f90 sourcefile~gvec_to_hopr_vars.f90 gvec_to_hopr_vars.F90 sourcefile~gvec_to_hopr_vars.f90->sourcefile~transform_sfl.f90

Source Code

! Copyright (c) INRIA
! License: CECILL-B
!
!> @ingroup splines
!> @brief   Derived type for uniform B-splines of arbitrary degree
!> @author  Yaman Güçlü  - IPP Garching
!> @author  Edoardo Zoni - IPP Garching

module sll_m_bsplines_uniform
!+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
#include "sll_assert.h"
#include "sll_errors.h"

  use sll_m_working_precision, only: f64
  use sll_m_bsplines_base    , only: sll_c_bsplines

  implicit none

  public :: &
    sll_t_bsplines_uniform

  private
!+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

  !> Working precision
  integer, parameter :: wp = f64

  !> Abstract type, B-splines
  type, extends(sll_c_bsplines) :: sll_t_bsplines_uniform

    ! Private components (public ones defined in base class)
    real(wp), private :: inv_dx

  contains
    procedure :: init                    => s_bsplines_uniform__init
    procedure :: free                    => s_bsplines_uniform__free
    procedure :: find_cell               => f_bsplines_uniform__find_cell
    procedure :: eval_basis              => s_bsplines_uniform__eval_basis
    procedure :: eval_deriv              => s_bsplines_uniform__eval_deriv
    procedure :: eval_basis_and_n_derivs => s_bsplines_uniform__eval_basis_and_n_derivs

  end type sll_t_bsplines_uniform

  ! Inverse of integers for later use (max spline degree = 32)
  integer             :: index
  real(wp), parameter :: inv(1:32) = [(1.0_wp/real(index,wp), index=1,32)]

!+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
contains
!+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

  ! Module subroutine, private
  SLL_PURE subroutine s_bsplines_uniform__get_icell_and_offset( self, x, icell, x_offset )
    class(sll_t_bsplines_uniform), intent(in   ) :: self
    real(wp)                     , intent(in   ) :: x
    integer                      , intent(  out) :: icell
    real(wp)                     , intent(  out) :: x_offset

    SLL_ASSERT( x >= self % xmin )
    SLL_ASSERT( x <= self % xmax )

    if      (x == self%xmin) then; icell = 1          ; x_offset = 0.0_wp
    else if (x == self%xmax) then; icell = self%ncells; x_offset = 1.0_wp
    else
      x_offset = (x-self%xmin) * self%inv_dx  ! 0 <= x_offset <= num_cells
      icell    = int( x_offset )
      x_offset = x_offset - real( icell, wp ) ! 0 <= x_offset < 1
      icell    = icell + 1
    end if

  end subroutine s_bsplines_uniform__get_icell_and_offset

  !-----------------------------------------------------------------------------
  !> @brief      Initialize uniform B-splines object
  !> @param[out] self      uniform B-splines
  !> @param[in]  degree    spline degree
  !> @param[in]  periodic  .true. if domain is periodic, .false. otherwise
  !> @param[in]  xmin      left  boundary of x domain
  !> @param[in]  xmax      right boundary of x domain
  !> @param[in]  ncells    number of grid cells in domain
  !-----------------------------------------------------------------------------
  subroutine s_bsplines_uniform__init( self, degree, periodic, xmin, xmax, ncells )
    class(sll_t_bsplines_uniform), intent(  out) :: self
    integer                      , intent(in   ) :: degree
    logical                      , intent(in   ) :: periodic
    real(wp)                     , intent(in   ) :: xmin
    real(wp)                     , intent(in   ) :: xmax
    integer                      , intent(in   ) :: ncells

    ! Public attributes
    self % degree   = degree
    self % periodic = periodic
    self % uniform  = .true.
    self % ncells   = ncells
    self % nbasis   = merge( ncells  , ncells+degree, periodic )
    self % offset   = merge( degree/2, 0            , periodic )
    self % xmin     = xmin
    self % xmax     = xmax

    ! Private attributes
    self % inv_dx   = real(ncells,wp) / (xmax-xmin)

  end subroutine s_bsplines_uniform__init

  !-----------------------------------------------------------------------------
  !> @brief        Free storage
  !> @param[inout] self  uniform B-splines
  !-----------------------------------------------------------------------------
  subroutine s_bsplines_uniform__free( self )
    class(sll_t_bsplines_uniform), intent(inout) :: self
  end subroutine s_bsplines_uniform__free

  !-----------------------------------------------------------------------------
  !> @brief     Find which grid cell contains the given point
  !> @param[in] self  uniform B-splines
  !> @param[in] x     point of interest
  !> results          cell index
  !-----------------------------------------------------------------------------
  SLL_PURE function f_bsplines_uniform__find_cell( self, x ) result( icell )
    class(sll_t_bsplines_uniform), intent(in) :: self
    real(wp)                     , intent(in) :: x
    integer :: icell

    icell = -1
    !TODO: handle error within pure procedure
    !SLL_ERROR("sll_t_bsplines_uniform % find_cell","procedure not implemented")

  end function f_bsplines_uniform__find_cell

  !-----------------------------------------------------------------------------
  !> Evaluate value at x of all basis functions with support in local cell
  !> values[j] = B_j(x) for jmin <= j <= jmin+degree
  !>
  !> @param[in]  self    uniform B-splines
  !> @param[in]  x       evaluation point
  !> @param[out] values  array of B-splines' values
  !> @param[out] jmin    index of first non-zero B-spline
  !-----------------------------------------------------------------------------
  SLL_PURE subroutine s_bsplines_uniform__eval_basis( self, x, values, jmin )
    class(sll_t_bsplines_uniform), intent(in   ) :: self
    real(wp)                     , intent(in   ) :: x
    real(wp)                     , intent(  out) :: values(0:)
    integer                      , intent(  out) :: jmin

    integer  :: icell
    real(wp) :: x_offset

    integer  :: j, r
    real(wp) :: j_real
    real(wp) :: xx
    real(wp) :: temp
    real(wp) :: saved

    ! Check on inputs
    SLL_ASSERT( size(values) == 1+self%degree )

    ! 1. Compute cell index 'icell' and x_offset
    call s_bsplines_uniform__get_icell_and_offset( self, x, icell, x_offset )

    ! 2. Compute index range of B-splines with support over cell 'icell'
    jmin = icell - self%offset

    ! 3. Compute values of aforementioned B-splines
    associate( bspl => values, spline_degree => self%degree )

      bspl(0) = 1.0_wp
      do j = 1, spline_degree
         j_real = real(j,wp)
         xx     = -x_offset
         saved  = 0.0_wp
         do r = 0, j-1
            xx      = xx + 1.0_wp
            temp    = bspl(r) * inv(j)
            bspl(r) = saved + xx * temp
            saved   = (j_real - xx) * temp
         end do
         bspl(j) = saved
      end do

    end associate  ! bspl, spline_degree

  end subroutine s_bsplines_uniform__eval_basis

  !-----------------------------------------------------------------------------
  !> Evaluate derivative at x of all basis functions with support in local cell
  !> derivs[j] = B_j'(x) for jmin <= j <= jmin+degree
  !>
  !> @param[in]  self    uniform B-splines
  !> @param[in]  x       evaluation point
  !> @param[out] derivs  array of B-splines' derivatives
  !> @param[out] jmin    index of first non-zero B-spline
  !-----------------------------------------------------------------------------
  SLL_PURE subroutine s_bsplines_uniform__eval_deriv( self, x, derivs, jmin )
    class(sll_t_bsplines_uniform), intent(in   ) :: self
    real(wp)                     , intent(in   ) :: x
    real(wp)                     , intent(  out) :: derivs(0:)
    integer                      , intent(  out) :: jmin

    integer  :: icell
    real(wp) :: x_offset

    integer  :: j, r
    real(wp) :: j_real
    real(wp) :: xx
    real(wp) :: temp
    real(wp) :: saved

    real(wp) :: bj, bjm1

    ! Check on inputs
    SLL_ASSERT( size(derivs) == 1+self%degree )

    ! 1. Compute cell index 'icell' and x_offset
    call s_bsplines_uniform__get_icell_and_offset( self, x, icell, x_offset )

    ! 2. Compute index range of B-splines with support over cell 'icell'
    jmin = icell - self%offset

    ! 3. Compute derivatives of aforementioned B-splines
    !    Derivatives are normalized, hence they should be divided by dx
    associate( bspl => derivs, spline_degree => self%degree, start => self%inv_dx )

      ! only need splines of lower degree to compute derivatives
      bspl(0) = start
      do j = 1, spline_degree - 1
         j_real = real(j,wp)
         xx     = -x_offset
         saved  = 0.0_wp
         do r = 0, j-1
            xx      = xx + 1.0_wp
            temp    = bspl(r) * inv(j)
            bspl(r) = saved + xx * temp
            saved   = (j_real - xx) * temp
         end do
         bspl(j) = saved
      end do

      ! compute derivatives
      bjm1 = bspl(0)
      bj = bjm1
      bspl(0) = -bjm1
      do j=1, spline_degree - 1
         bj = bspl(j)
         bspl(j) = bjm1 - bj
         bjm1 = bj
      end do
      bspl(spline_degree) = bj

    end associate  ! bspl, spline_degree

  end subroutine s_bsplines_uniform__eval_deriv

  !-----------------------------------------------------------------------------
  !> Evaluate value and n derivatives at x of all basis functions with support in local cell
  !> derivs[i,j] = (d/dx)^i B_j(x) for 0 <= i <= n and jmin <= j <= jmin+degree
  !>
  !> @param[in]  self    uniform B-splines
  !> @param[in]  x       evaluation point
  !> @param[in]  n       number of required derivatives
  !> @param[out] derivs  array of B-splines' (multiple) derivatives
  !> @param[out] jmin    index of first non-zero B-spline
  !-----------------------------------------------------------------------------
  SLL_PURE subroutine s_bsplines_uniform__eval_basis_and_n_derivs( self, x, n, derivs, jmin )
    class(sll_t_bsplines_uniform), intent(in   ) :: self
    real(wp)                     , intent(in   ) :: x
    integer                      , intent(in   ) :: n
    real(wp)                     , intent(  out) :: derivs(0:,0:)
    integer                      , intent(  out) :: jmin

    integer  :: icell
    real(wp) :: x_offset

    integer  :: j, r
    real(wp) :: j_real
    real(wp) :: xx
    real(wp) :: temp
    real(wp) :: saved

    integer  :: k, s1, s2, rk, pk, j1, j2
    real(wp) :: d

    ! GFortran: to allocate on stack use -fstack-arrays
    real(wp) :: ndu (0:self%degree, 0:self%degree)
    real(wp) :: a   (0:1          , 0:self%degree)

    ! Check on inputs
    SLL_ASSERT( n >= 0 )
    SLL_ASSERT( size(derivs,1) == 1+n           )
    SLL_ASSERT( size(derivs,2) == 1+self%degree )

    ! 1. Compute cell index 'icell' and x_offset
    call s_bsplines_uniform__get_icell_and_offset( self, x, icell, x_offset )

    ! 2. Compute index range of B-splines with support over cell 'icell'
    jmin = icell - self%offset

    ! 3. Recursively evaluate B-splines (see "sll_s_uniform_bsplines_eval_basis")
    !    up to self%degree, and store them all in the upper-right triangle of ndu
    associate( spline_degree => self%degree, bspl => derivs )

      ndu(0,0) = 1.0_wp
      do j = 1, spline_degree
         j_real = real(j,wp)
         xx     = -x_offset
         saved  = 0.0_wp
         do r = 0, j-1
            xx       = xx + 1.0_wp
            temp     = ndu(r,j-1) * inv(j)
            ndu(r,j) = saved + xx * temp
            saved    = (j_real - xx) * temp
         end do
         ndu(j,j) = saved
      end do
      bspl(0,:) = ndu(:,spline_degree)

    end associate  ! spline_degree, bspl

    ! 4. Use equation 2.10 in "The NURBS Book" to compute n derivatives
    associate( deg => self%degree, bsdx => derivs )

      do r = 0, deg
         s1 = 0
         s2 = 1
         a(0,0) = 1.0_wp
         do k = 1, n
            d  = 0.0_wp
            rk = r-k
            pk = deg-k
            if (r >= k) then
               a(s2,0) = a(s1,0) * inv(pk+1)
               d = a(s2,0) * ndu(rk,pk)
            end if
            if (rk > -1) then
               j1 = 1
            else
               j1 = -rk
            end if
            if (r-1 <= pk) then
               j2 = k-1
            else
               j2 = deg-r
            end if
            do j = j1, j2
               a(s2,j) = (a(s1,j) - a(s1,j-1)) * inv(pk+1)
               d = d + a(s2,j) * ndu(rk+j,pk)
            end do
            if (r <= pk) then
               a(s2,k) = - a(s1,k-1) * inv(pk+1)
               d = d + a(s2,k) * ndu(r,pk)
            end if
            bsdx(k,r) = d
            j  = s1
            s1 = s2
            s2 = j
         end do
      end do

      ! Multiply result by correct factors:
      ! deg!/(deg-n)! = deg*(deg-1)*...*(deg-n+1)
      ! k-th derivatives are normalized, hence they should be divided by dx^k
      d = real(deg,wp) * self%inv_dx
      do k = 1, n
         bsdx(k,:) = bsdx(k,:) * d
         d = d * real(deg-k,wp) * self%inv_dx
      end do

    end associate  ! deg, bsdx

  end subroutine s_bsplines_uniform__eval_basis_and_n_derivs

end module sll_m_bsplines_uniform