New URL for NEMO forge!   http://forge.nemo-ocean.eu

Since March 2022 along with NEMO 4.2 release, the code development moved to a self-hosted GitLab.
This present forge is now archived and remained online for history.
dynldf.F90 on Ticket #371 – Attachment – NEMO

Ticket #371: dynldf.F90

File dynldf.F90, 11.6 KB (added by ed.blockley, 15 years ago)

DYN/dynldf.F90

Line 
1MODULE dynldf
2   !!======================================================================
3   !!                       ***  MODULE  dynldf  ***
4   !! Ocean physics:  lateral diffusivity trends
5   !!=====================================================================
6   !! History :  9.0  !  05-11  (G. Madec)  Original code (new step architecture)
7   !!----------------------------------------------------------------------
8
9   !!----------------------------------------------------------------------
10   !!   dyn_ldf     : update the dynamics trend with the lateral diffusion
11   !!   dyn_ldf_ctl : initialization, namelist read, and parameters control
12   !!----------------------------------------------------------------------
13   USE oce            ! ocean dynamics and tracers
14   USE dom_oce        ! ocean space and time domain
15   USE phycst         ! physical constants
16   USE ldfdyn_oce     ! ocean dynamics lateral physics
17   USE ldfslp         ! lateral mixing: slopes of mixing orientation
18   USE dynldf_bilapg  ! lateral mixing            (dyn_ldf_bilapg routine)
19   USE dynldf_bilap   ! lateral mixing            (dyn_ldf_bilap  routine)
20   USE dynldf_iso     ! lateral mixing            (dyn_ldf_iso    routine)
21   USE dynldf_lap     ! lateral mixing            (dyn_ldf_lap    routine)
22   USE trdmod         ! ocean dynamics and tracer trends
23   USE trdmod_oce     ! ocean variables trends
24   USE prtctl         ! Print control
25   USE in_out_manager ! I/O manager
26   USE lib_mpp        ! distribued memory computing library
27   USE lbclnk         ! ocean lateral boundary conditions (or mpp link)
28
29   IMPLICIT NONE
30   PRIVATE
31
32   PUBLIC   dyn_ldf   ! called by step module
33
34   INTEGER ::   nldf = 0   ! type of lateral diffusion used defined from ln_dynldf_... namlist logicals)
35
36   !! * Substitutions
37#  include "domzgr_substitute.h90"
38#  include "vectopt_loop_substitute.h90"
39   !!---------------------------------------------------------------------------------
40   !!   OPA 9.0 , LOCEAN-IPSL (2005)
41   !! $Id: dynldf.F90 1152 2008-06-26 14:11:13Z rblod $
42   !! Software governed by the CeCILL licence (modipsl/doc/NEMO_CeCILL.txt)
43   !!----------------------------------------------------------------------
44
45CONTAINS
46
47   SUBROUTINE dyn_ldf( kt )
48      !!----------------------------------------------------------------------
49      !!                  ***  ROUTINE dyn_ldf  ***
50      !!
51      !! ** Purpose :   compute the lateral ocean dynamics physics.
52      !!----------------------------------------------------------------------
53      INTEGER, INTENT(in) ::   kt   ! ocean time-step index
54      !!
55      REAL(wp), DIMENSION(jpi,jpj,jpk) ::   ztrdu, ztrdv   ! 3D workspace
56      !!----------------------------------------------------------------------
57
58      IF( kt == nit000 )   CALL dyn_ldf_ctl      ! initialisation & control of options
59
60      IF( l_trddyn )   THEN                      ! temporary save of ta and sa trends
61         ztrdu(:,:,:) = ua(:,:,:) 
62         ztrdv(:,:,:) = va(:,:,:) 
63      ENDIF
64
65      SELECT CASE ( nldf )                       ! compute lateral mixing trend and add it to the general trend
66      !
67      CASE ( 0 )    ;   CALL dyn_ldf_lap    ( kt )      ! iso-level laplacian
68      CASE ( 1 )    ;   CALL dyn_ldf_iso    ( kt )      ! rotated laplacian (except dk[ dk[.] ] part)
69      CASE ( 2 )    ;   CALL dyn_ldf_bilap  ( kt )      ! iso-level bilaplacian
70      CASE ( 3 )    ;   CALL dyn_ldf_bilapg ( kt )      ! s-coord. horizontal bilaplacian
71      !
72      CASE ( -1 )                                       ! esopa: test all possibility with control print
73                        CALL dyn_ldf_lap    ( kt )
74                        CALL prt_ctl( tab3d_1=ua, clinfo1=' ldf0 - Ua: ', mask1=umask,   &
75            &                         tab3d_2=va, clinfo2=       ' Va: ', mask2=vmask, clinfo3='dyn' )
76                        CALL dyn_ldf_iso    ( kt )
77                        CALL prt_ctl( tab3d_1=ua, clinfo1=' ldf1 - Ua: ', mask1=umask,   &
78            &                         tab3d_2=va, clinfo2=       ' Va: ', mask2=vmask, clinfo3='dyn' )
79                        CALL dyn_ldf_bilap  ( kt )
80                        CALL prt_ctl( tab3d_1=ua, clinfo1=' ldf2 - Ua: ', mask1=umask,   &
81            &                         tab3d_2=va, clinfo2=       ' Va: ', mask2=vmask, clinfo3='dyn' )
82                        CALL dyn_ldf_bilapg ( kt )
83                        CALL prt_ctl( tab3d_1=ua, clinfo1=' ldf3 - Ua: ', mask1=umask,   &
84            &                         tab3d_2=va, clinfo2=       ' Va: ', mask2=vmask, clinfo3='dyn' )
85      CASE ( 4 )                                       ! iso-level laplacian + bilaplacian
86         CALL dyn_ldf_lap    ( kt )
87         CALL dyn_ldf_bilap  ( kt )
88      CASE ( 5 )                                       ! rotated laplacian + bilaplacian (s-coord)
89         CALL dyn_ldf_iso    ( kt )
90         CALL dyn_ldf_bilapg ( kt )
91      END SELECT
92
93      IF( l_trddyn ) THEN                        ! save the horizontal diffusive trends for further diagnostics
94         ztrdu(:,:,:) = ua(:,:,:) - ztrdu(:,:,:)
95         ztrdv(:,:,:) = va(:,:,:) - ztrdv(:,:,:)
96         CALL trd_mod( ztrdu, ztrdv, jpdyn_trd_ldf, 'DYN', kt )
97      ENDIF
98      !                                          ! print sum trends (used for debugging)
99      IF(ln_ctl)   CALL prt_ctl( tab3d_1=ua, clinfo1=' ldf  - Ua: ', mask1=umask,   &
100         &                       tab3d_2=va, clinfo2=       ' Va: ', mask2=vmask, clinfo3='dyn' )
101      !
102   END SUBROUTINE dyn_ldf
103
104
105   SUBROUTINE dyn_ldf_ctl
106      !!----------------------------------------------------------------------
107      !!                  ***  ROUTINE dyn_ldf_ctl  ***
108      !!
109      !! ** Purpose :   initializations of the horizontal ocean dynamics physics
110      !!----------------------------------------------------------------------
111      INTEGER ::   ioptio, ierr         ! temporary integers
112      !!----------------------------------------------------------------------
113   
114      !                                   ! Namelist nam_dynldf: already read in ldfdyn module
115
116      IF(lwp) THEN                        ! Namelist print
117         WRITE(numout,*)
118         WRITE(numout,*) 'dyn_ldf_ctl : Choice of the lateral diffusive operator on dynamics'
119         WRITE(numout,*) '~~~~~~~~~~~'
120         WRITE(numout,*) '       Namelist nam_dynldf : set lateral mixing parameters (type, direction, coefficients)'
121         WRITE(numout,*) '          laplacian operator          ln_dynldf_lap   = ', ln_dynldf_lap
122         WRITE(numout,*) '          bilaplacian operator        ln_dynldf_bilap = ', ln_dynldf_bilap
123         WRITE(numout,*) '          iso-level                   ln_dynldf_level = ', ln_dynldf_level
124         WRITE(numout,*) '          horizontal (geopotential)   ln_dynldf_hor   = ', ln_dynldf_hor
125         WRITE(numout,*) '          iso-neutral                 ln_dynldf_iso   = ', ln_dynldf_iso
126      ENDIF
127
128      !                                   ! control the consistency
129      ioptio = 0
130      IF( ln_dynldf_lap   )   ioptio = ioptio + 1
131      IF( ln_dynldf_bilap )   ioptio = ioptio + 1
132      IF( ioptio <  1 ) CALL ctl_stop( '          use at least ONE of the 2 lap/bilap operator type on dynamics' )
133      ioptio = 0
134      IF( ln_dynldf_level )   ioptio = ioptio + 1
135      IF( ln_dynldf_hor   )   ioptio = ioptio + 1
136      IF( ln_dynldf_iso   )   ioptio = ioptio + 1
137      IF( ioptio /= 1 ) CALL ctl_stop( '          use only ONE direction (level/hor/iso)' )
138
139      !                                   ! Set nldf, the type of lateral diffusion, from ln_dynldf_... logicals
140      ierr = 0
141      IF ( ln_dynldf_lap ) THEN      ! laplacian operator
142         IF ( ln_zco ) THEN                ! z-coordinate
143            IF ( ln_dynldf_level )   nldf = 0      ! iso-level  (no rotation)
144            IF ( ln_dynldf_hor   )   nldf = 0      ! horizontal (no rotation)
145            IF ( ln_dynldf_iso   )   nldf = 1      ! isoneutral (   rotation)
146         ENDIF
147         IF ( ln_zps ) THEN             ! z-coordinate
148            IF ( ln_dynldf_level )   ierr = 1      ! iso-level not allowed
149            IF ( ln_dynldf_hor   )   nldf = 0      ! horizontal (no rotation)
150            IF ( ln_dynldf_iso   )   nldf = 1      ! isoneutral (   rotation)
151         ENDIF
152         IF ( ln_sco ) THEN             ! s-coordinate
153            IF ( ln_dynldf_level )   nldf = 0      ! iso-level  (no rotation)
154            IF ( ln_dynldf_hor   )   nldf = 1      ! horizontal (   rotation)
155            IF ( ln_dynldf_iso   )   nldf = 1      ! isoneutral (   rotation)
156         ENDIF
157      ENDIF
158
159      IF( ln_dynldf_bilap ) THEN      ! bilaplacian operator
160         IF ( ln_zco ) THEN                ! z-coordinate
161            IF ( ln_dynldf_level )   nldf = 2      ! iso-level  (no rotation)
162            IF ( ln_dynldf_hor   )   nldf = 2      ! horizontal (no rotation)
163            IF ( ln_dynldf_iso   )   ierr = 2      ! isoneutral (   rotation)
164         ENDIF
165         IF ( ln_zps ) THEN             ! z-coordinate
166            IF ( ln_dynldf_level )   ierr = 1      ! iso-level not allowed
167            IF ( ln_dynldf_hor   )   nldf = 2      ! horizontal (no rotation)
168            IF ( ln_dynldf_iso   )   ierr = 2      ! isoneutral (   rotation)
169         ENDIF
170         IF ( ln_sco ) THEN             ! s-coordinate
171            IF ( ln_dynldf_level )   nldf = 2      ! iso-level  (no rotation)
172            IF ( ln_dynldf_hor   )   nldf = 3      ! horizontal (   rotation)
173            IF ( ln_dynldf_iso   )   ierr = 2      ! isoneutral (   rotation)
174         ENDIF
175      ENDIF
176     
177      IF( ln_dynldf_lap .AND. ln_dynldf_bilap ) THEN  ! mixed laplacian and bilaplacian operators
178         IF ( ln_zco ) THEN                ! z-coordinate
179            IF ( ln_dynldf_level )   nldf = 4      ! iso-level  (no rotation)
180            IF ( ln_dynldf_hor   )   nldf = 4      ! horizontal (no rotation)
181            IF ( ln_dynldf_iso   )   ierr = 2      ! isoneutral (   rotation)
182         ENDIF
183         IF ( ln_zps ) THEN             ! z-coordinate
184            IF ( ln_dynldf_level )   ierr = 1      ! iso-level not allowed
185            IF ( ln_dynldf_hor   )   nldf = 4      ! horizontal (no rotation)
186            IF ( ln_dynldf_iso   )   ierr = 2      ! isoneutral (   rotation)
187         ENDIF
188         IF ( ln_sco ) THEN             ! s-coordinate
189            IF ( ln_dynldf_level )   nldf = 4      ! iso-level  (no rotation)
190            IF ( ln_dynldf_hor   )   nldf = 5      ! horizontal (   rotation)
191            IF ( ln_dynldf_iso   )   ierr = 2      ! isoneutral (   rotation)
192         ENDIF
193      ENDIF
194
195      IF( lk_esopa )                 nldf = -1     ! esopa test
196
197      IF( ierr == 1 )   CALL ctl_stop( 'iso-level in z-coordinate - partial step, not allowed' )
198      IF( ierr == 2 )   CALL ctl_stop( 'isoneutral bilaplacian operator does not exist' )
199      IF( nldf == 1 .OR. nldf == 3 ) THEN      ! rotation
200         IF( .NOT.lk_ldfslp )   CALL ctl_stop( 'the rotation of the diffusive tensor require key_ldfslp' )
201      ENDIF
202
203      IF(lwp) THEN
204         WRITE(numout,*)
205         IF( nldf == -1 )   WRITE(numout,*) '              ESOPA test All scheme used'
206         IF( nldf ==  0 )   WRITE(numout,*) '              laplacian operator'
207         IF( nldf ==  1 )   WRITE(numout,*) '              Rotated laplacian operator'
208         IF( nldf ==  2 )   WRITE(numout,*) '              bilaplacian operator'
209         IF( nldf ==  3 )   WRITE(numout,*) '              Rotated bilaplacian'
210         IF( nldf ==  4 )   WRITE(numout,*) '              laplacian and bilaplacian operators'
211         IF( nldf ==  5 )   WRITE(numout,*) '              Rotated laplacian and bilaplacian operators'
212      ENDIF
213      !
214   END SUBROUTINE dyn_ldf_ctl
215
216   !!======================================================================
217END MODULE dynldf