tapenade_ge_api.f90 Source File


Source Code

module yaeos__tapenade_ge_api
   !! Module that wraps tapenade generated routines to calculate !
   !! Ge and derivatives.
   use yaeos__constants, only: pr, R
   use yaeos__models_ge, only: GeModel
   implicit none

   private

   public :: GeModelTapenade
   type, abstract, extends(GeModel) :: GeModelTapenade
   contains
      procedure(tapenade_ge), deferred  :: ge
      procedure(tapenade_ge_d), deferred :: ge_d
      procedure(tapenade_ge_b), deferred :: ge_b
      procedure(tapenade_ge_d_b), deferred :: ge_d_b
      procedure(tapenade_ge_d_d), deferred :: ge_d_d
      procedure :: excess_gibbs => excess_gibbs
   end type

   abstract interface
      subroutine tapenade_ge(model, n, t, ge)
         import GeModelTapenade, pr
         class(GeModelTapenade) :: model
         real(pr), intent(in) :: n(:), t
         real(pr), intent(out) :: ge
      end subroutine

      subroutine tapenade_ge_d(model, n, nd, t, td, ge, ged)
         import pr, GeModelTapenade
         class(GeModelTapenade) :: model
         real(pr), intent(in) :: n(:), t
         real(pr), intent(in) :: nd(:), td
         real(pr), intent(out) :: ge, ged
      end subroutine

      subroutine tapenade_ge_b(model, n, nb, t, tb, ge, geb)
         import pr, GeModelTapenade
         class(GeModelTapenade) :: model
         real(pr), intent(in) :: n(:), t
         real(pr) :: geb
         real(pr) :: nb(:), tb
         real(pr) :: ge
      end subroutine

      subroutine tapenade_ge_d_b(model, &
         n, nb, nd, ndb, t, tb, td, tdb, &
         ge, geb, ged, gedb)
         import pr, GeModelTapenade
         class(GeModelTapenade) :: model
         real(pr), intent(in) :: n(:), t
         real(pr) :: ge

         real(pr), intent(in) :: nd(:), td
         real(pr) :: ged

         real(pr) :: nb(:), tb
         real(pr) :: geb

         real(pr) :: ndb(:), tdb
         real(pr) :: gedb
      end subroutine

      subroutine tapenade_ge_d_d(model, n, nd, t, td0, td, ge, ged0, ged, gedd)
         import pr, GeModelTapenade
         class(GeModelTapenade) :: model
         real(pr), intent(in) :: n(:), t
         real(pr), intent(in) :: td0
         real(pr), intent(in) :: nd(:), td
         real(pr), intent(out) :: ge, ged0, ged, gedd
      end subroutine
   end interface

contains

   subroutine excess_gibbs(&
      self, n, t, Ge, GeT, GeT2, Gen, GeTn, Gen2 &
      )
      !! Excess Gibbs model generic interface
      class(GeModelTapenade), intent(in) :: self
      real(pr), intent(in) :: n(:)
      real(pr), intent(in) :: t
      real(pr), optional, intent(out) :: Ge, GeT, GeT2
      real(pr), optional, dimension(size(n)), intent(out) :: Gen, GeTn
      real(pr), optional, intent(out) :: Gen2(size(n), size(n))

      real(pr) :: nb(size(n)), nd(size(n)), ndb(size(n))
      real(pr) :: tb, td, tdb, td0
      real(pr) :: geb, ged, gedb, ged0, gedd

      integer :: i, nc

      nc = size(n)

      if (present(Gen2)) then
         do i=1, nc
            call reset_vars
            gedb = 1
            if (i <= nc) then
               nd(i) = 1
            end if

            call self%ge_d_b(&
               n, nb, nd, ndb, &
               t, tb, td, tdb, &
               ge, geb, ged, gedb &
            )

            Gen2(i, :) = nb
         end do

         if(present(Gen)) Gen = ndb
         if(present(GeT)) GeT = tdb
      else if (present(Gen)) then
            call reset_vars
            geb = 1
            call self%ge_b(n, nb, t, tb, ge, geb)
            Gen = nb
            if (present(GeT)) GeT = tb
      else
         if (present(GeT)) GeT = get_dGedT()
      end if

      if (present(GeTn)) GeTn = get_GenT()
      if (present(GeT2)) GeT2 = get_dGedT2()

      call reset_vars
      if (present(Ge)) call self%ge(n, t, ge)

   contains
      subroutine reset_vars
         nb=0
         nd=0
         ndb=0

         tb=0
         td=0
         td0 = 0
         tdb=0

         ge = 0
         geb = 0
         ged = 0
         ged0 = 0
         gedb = 0
      end subroutine

      function get_dGedT2()
         real(pr) :: get_dGedT2

         call reset_vars

         td = 1
         td0 = 1

         call self%ge_d_d(&
            n, nd, t, td0, td, &
            ge, ged0, ged, gedd &
         )
         get_dGedT2 = gedd
      end function

      function get_dGedT()
         real(pr) :: get_dGedT
         call reset_vars

         td = 1

         call self%ge_d(&
            n, nd, t, td, ge, ged &
            )
         get_dGedT = ged
      end function

      function get_GenT()
         real(pr) :: get_GenT(size(n))
         call reset_vars
         
         gedb = 1
         td = 1

         call self%ge_d_b(&
            n, nb, nd, ndb, &
            t, tb, td, tdb, &
            ge, geb, ged, gedb &
            )
         get_GenT = nb
      end function
   end subroutine
end module