Skip to content
Snippets Groups Projects
diu_coolskin.F90 6.43 KiB
Newer Older
Guillaume Samson's avatar
Guillaume Samson committed
MODULE diu_coolskin
   !!======================================================================
   !!                    ***  MODULE  diu_coolskin  ***
   !!     Cool skin thickness and delta T correction using Artele et al. (2002)
   !!     [see also Tu and Tsuang (2005)]
   !!
   !!=====================================================================
   !! History :        !  2012-01  (P. Sykes)  Original code
   !!----------------------------------------------------------------------

   !!----------------------------------------------------------------------
   !!   diurnal_sst_coolskin_init : initialisation of the cool skin
   !!   diurnal_sst_coolskin_step : time-stepping  of the cool skin corrections
   !!----------------------------------------------------------------------
   USE par_kind
   USE phycst
   USE dom_oce
   USE in_out_manager
   USE sbc_oce
   USE lib_mpp
   USE lbclnk          ! ocean lateral boundary conditions (or mpp link)

   IMPLICIT NONE
   PRIVATE

   ! Namelist parameters

   ! Parameters
   REAL(wp), PRIVATE, PARAMETER :: pp_k = 0.596_wp          ! Thermal conductivity of seawater
   REAL(wp), PRIVATE, PARAMETER :: pp_v = 1.05e-6_wp        ! Kinematic viscosity of seawater
   REAL(wp), PRIVATE, PARAMETER :: pp_C = 86400             ! seconds [see Tu and Tsuang (2005)]
   REAL(wp), PRIVATE, PARAMETER :: pp_cw = 3993._wp         ! specific heat capacity of seawater
   REAL(wp), PRIVATE, PARAMETER :: pp_h = 10._wp            ! reference depth [using 10m from Artale et al. (2002)]
   REAL(wp), PRIVATE, PARAMETER :: pp_rhoa = 1.20421_wp     ! density of air (at 20C)
   REAL(wp), PRIVATE, PARAMETER :: pp_cda = 1.45e-3_wp      ! assumed air-sea drag coefficient for calculating wind speed

   ! Key variables
   REAL(wp), PUBLIC, ALLOCATABLE, DIMENSION(:,:) :: x_csdsst    ! Cool skin delta SST
   REAL(wp), PUBLIC, ALLOCATABLE, DIMENSION(:,:) :: x_csthick   ! Cool skin thickness
   PUBLIC diurnal_sst_coolskin_step, diurnal_sst_coolskin_init

      !! * Substitutions
#  include "do_loop_substitute.h90"
   !!----------------------------------------------------------------------
   !! NEMO/OCE 4.0 , NEMO Consortium (2018)
   !! $Id: diu_coolskin.F90 14072 2020-12-04 07:48:38Z laurent $
   !! Software governed by the CeCILL license (see ./LICENSE)
   !!----------------------------------------------------------------------
   CONTAINS

   SUBROUTINE diurnal_sst_coolskin_init
      !!----------------------------------------------------------------------
      !! *** ROUTINE diurnal_sst_coolskin_init ***
      !!
      !! ** Purpose :   initialise the cool skin model
      !!
      !! ** Method :
      !!
      !! ** Reference :
      !!
      !!----------------------------------------------------------------------
      ALLOCATE( x_csdsst(A2D(0)), x_csthick(A2D(0)) )
Guillaume Samson's avatar
Guillaume Samson committed
      x_csdsst = 0.
      x_csthick = 0.
      !
   END SUBROUTINE diurnal_sst_coolskin_init


   SUBROUTINE diurnal_sst_coolskin_step(psqflux, pstauflux, psrho, pDt)
      !!----------------------------------------------------------------------
      !! *** ROUTINE diurnal_sst_takaya_step ***
      !!
      !! ** Purpose :   Time-step the Artale cool skin model
      !!
      !! ** Method :
      !!
      !! ** Reference :
      !!----------------------------------------------------------------------
      ! Dummy variables
      REAL(wp), INTENT(IN), DIMENSION(A2D(0))  :: psqflux     ! Heat (non-solar)(Watts)
      REAL(wp), INTENT(IN), DIMENSION(A2D(0))  :: pstauflux   ! Wind stress (kg/ m s^2)
Guillaume Samson's avatar
Guillaume Samson committed
      REAL(wp), INTENT(IN), DIMENSION(jpi,jpj) :: psrho       ! Water density (kg/m^3)
      REAL(wp), INTENT(IN) :: pDt                             ! Time-step

      ! Local variables
      REAL(wp), DIMENSION(A2D(0)) :: z_fv                     ! Friction velocity
      REAL(wp), DIMENSION(A2D(0)) :: z_gamma                  ! Dimensionless function of wind speed
      REAL(wp), DIMENSION(A2D(0)) :: z_lamda                  ! Sauders (dimensionless) proportionality constant
      REAL(wp), DIMENSION(A2D(0)) :: z_wspd                   ! Wind speed (m/s)
Guillaume Samson's avatar
Guillaume Samson committed
      REAL(wp) :: z_ztx                                       ! Temporary u wind stress
      REAL(wp) :: z_zty                                       ! Temporary v wind stress
      REAL(wp) :: z_zmod                                      ! Temporary total wind stress

      INTEGER :: ji,jj
      !!----------------------------------------------------------------------
      !
      IF( .NOT. (ln_blk .OR. ln_abl) )   CALL ctl_stop("diu_coolskin.f90: diurnal flux processing only implemented for bulk forcing")
      !
      DO_2D( 0, 0, 0, 0 )
Guillaume Samson's avatar
Guillaume Samson committed
         !
         ! Calcualte wind speed from wind stress and friction velocity
         IF( smask0(ji,jj) == 1. .AND. pstauflux(ji,jj) /= 0 .AND. psrho(ji,jj) /=0 ) THEN
Guillaume Samson's avatar
Guillaume Samson committed
            z_fv(ji,jj) = SQRT( pstauflux(ji,jj) / psrho(ji,jj) )
            z_wspd(ji,jj) = SQRT( pstauflux(ji,jj) / ( pp_cda * pp_rhoa ) )
         ELSE
            z_fv(ji,jj) = 0.
            z_wspd(ji,jj) = 0.
         ENDIF
         !
         ! Calculate gamma function which is dependent upon wind speed
         IF( smask0(ji,jj) == 1. ) THEN
Guillaume Samson's avatar
Guillaume Samson committed
            IF( ( z_wspd(ji,jj) <= 7.5 ) ) z_gamma(ji,jj) = ( 0.2 * z_wspd(ji,jj) ) + 0.5
            IF( ( z_wspd(ji,jj) > 7.5 ) .AND. ( z_wspd(ji,jj) < 10. ) ) z_gamma(ji,jj) = ( 1.6 * z_wspd(ji,jj) ) - 10.
            IF( ( z_wspd(ji,jj) >= 10. ) ) z_gamma(ji,jj) = 6.
         ENDIF
         !
         ! Calculate lamda function
         IF( smask0(ji,jj) == 1. .AND. z_fv(ji,jj) /= 0 ) THEN
Guillaume Samson's avatar
Guillaume Samson committed
            z_lamda(ji,jj) = ( z_fv(ji,jj) * pp_k * pp_C ) / ( z_gamma(ji,jj) * psrho(ji,jj) * pp_cw * pp_h * pp_v )
         ELSE
            z_lamda(ji,jj) = 0.
         ENDIF
         !
         ! Calculate the cool skin thickness - only when heat flux is out of the ocean
         IF( smask0(ji,jj) == 1. .AND. z_fv(ji,jj) /= 0 .AND. psqflux(ji,jj) < 0 ) THEN
Guillaume Samson's avatar
Guillaume Samson committed
            x_csthick(ji,jj) = ( z_lamda(ji,jj) * pp_v ) / z_fv(ji,jj)
         ELSE
            x_csthick(ji,jj) = 0.
         ENDIF
         !
         ! Calculate the cool skin correction - only when the heat flux is out of the ocean
         IF( smask0(ji,jj) == 1. .AND. x_csthick(ji,jj) /= 0. .AND. psqflux(ji,jj) < 0. ) THEN
Guillaume Samson's avatar
Guillaume Samson committed
            x_csdsst(ji,jj) = ( psqflux(ji,jj) * x_csthick(ji,jj) ) / pp_k
          ELSE
            x_csdsst(ji,jj) = 0.
         ENDIF
         !
      END_2D
      !
   END SUBROUTINE diurnal_sst_coolskin_step

   !!======================================================================
END MODULE diu_coolskin