source: LMDZ5/branches/LMDZ6_rc0/libf/dyn3d/guide_mod.F90 @ 5441

Last change on this file since 5441 was 2160, checked in by Laurent Fairhead, 10 years ago

Merged trunk changes -r2070:2158 into testing branch. Compilation problems introduced by revision r2155 have been corrected by hand

  • Property copyright set to
    Name of program: LMDZ
    Creation date: 1984
    Version: LMDZ5
    License: CeCILL version 2
    Holder: Laboratoire de m\'et\'eorologie dynamique, CNRS, UMR 8539
    See the license file in the root directory
  • Property svn:executable set to *
File size: 58.4 KB
Line 
1!
2! $Id$
3!
4MODULE guide_mod
5
6!=======================================================================
7!   Auteur:  F.Hourdin
8!            F. Codron 01/09
9!=======================================================================
10
11  USE getparam
12  USE Write_Field
13  use netcdf, only: nf90_nowrite, nf90_open, nf90_inq_varid, nf90_close
14  use pres2lev_mod
15
16  IMPLICIT NONE
17
18! ---------------------------------------------
19! Declarations des cles logiques et parametres
20! ---------------------------------------------
21  INTEGER, PRIVATE, SAVE  :: iguide_read,iguide_int,iguide_sav
22  INTEGER, PRIVATE, SAVE  :: nlevnc
23  LOGICAL, PRIVATE, SAVE  :: guide_u,guide_v,guide_T,guide_Q,guide_P
24  LOGICAL, PRIVATE, SAVE  :: guide_hr,guide_teta 
25  LOGICAL, PRIVATE, SAVE  :: guide_BL,guide_reg,guide_add,gamma4,guide_zon
26  LOGICAL, PRIVATE, SAVE  :: guide_modele,invert_p,invert_y,ini_anal
27  LOGICAL, PRIVATE, SAVE  :: guide_2D,guide_sav
28 
29  REAL, PRIVATE, SAVE     :: tau_min_u,tau_max_u
30  REAL, PRIVATE, SAVE     :: tau_min_v,tau_max_v
31  REAL, PRIVATE, SAVE     :: tau_min_T,tau_max_T
32  REAL, PRIVATE, SAVE     :: tau_min_Q,tau_max_Q
33  REAL, PRIVATE, SAVE     :: tau_min_P,tau_max_P
34
35  REAL, PRIVATE, SAVE     :: lat_min_g,lat_max_g
36  REAL, PRIVATE, SAVE     :: lon_min_g,lon_max_g
37  REAL, PRIVATE, SAVE     :: tau_lon,tau_lat
38
39  REAL, ALLOCATABLE, DIMENSION(:), PRIVATE, SAVE     :: alpha_u,alpha_v
40  REAL, ALLOCATABLE, DIMENSION(:), PRIVATE, SAVE     :: alpha_T,alpha_Q
41  REAL, ALLOCATABLE, DIMENSION(:), PRIVATE, SAVE     :: alpha_P,alpha_pcor
42 
43! ---------------------------------------------
44! Variables de guidage
45! ---------------------------------------------
46! Variables des fichiers de guidage
47  REAL, ALLOCATABLE, DIMENSION(:,:,:), PRIVATE, SAVE   :: unat1,unat2
48  REAL, ALLOCATABLE, DIMENSION(:,:,:), PRIVATE, SAVE   :: vnat1,vnat2
49  REAL, ALLOCATABLE, DIMENSION(:,:,:), PRIVATE, SAVE   :: tnat1,tnat2
50  REAL, ALLOCATABLE, DIMENSION(:,:,:), PRIVATE, SAVE   :: qnat1,qnat2
51  REAL, ALLOCATABLE, DIMENSION(:,:),   PRIVATE, SAVE   :: psnat1,psnat2
52  REAL, ALLOCATABLE, DIMENSION(:),     PRIVATE, SAVE   :: apnc,bpnc
53! Variables aux dimensions du modele
54  REAL, ALLOCATABLE, DIMENSION(:,:),   PRIVATE, SAVE   :: ugui1,ugui2
55  REAL, ALLOCATABLE, DIMENSION(:,:),   PRIVATE, SAVE   :: vgui1,vgui2
56  REAL, ALLOCATABLE, DIMENSION(:,:),   PRIVATE, SAVE   :: tgui1,tgui2
57  REAL, ALLOCATABLE, DIMENSION(:,:),   PRIVATE, SAVE   :: qgui1,qgui2
58  REAL, ALLOCATABLE, DIMENSION(:),   PRIVATE, SAVE   :: psgui1,psgui2
59
60CONTAINS
61!=======================================================================
62
63  SUBROUTINE guide_init
64
65    USE control_mod
66
67    IMPLICIT NONE
68 
69    INCLUDE "dimensions.h"
70    INCLUDE "paramet.h"
71    INCLUDE "netcdf.inc"
72
73    ! For grossismx:
74    include "serre.h"
75
76    INTEGER                :: error,ncidpl,rid,rcod
77    CHARACTER (len = 80)   :: abort_message
78    CHARACTER (len = 20)   :: modname = 'guide_init'
79
80! ---------------------------------------------
81! Lecture des parametres: 
82! ---------------------------------------------
83! Variables guidees
84    CALL getpar('guide_u',.true.,guide_u,'guidage de u')
85    CALL getpar('guide_v',.true.,guide_v,'guidage de v')
86    CALL getpar('guide_T',.true.,guide_T,'guidage de T')
87    CALL getpar('guide_P',.true.,guide_P,'guidage de P')
88    CALL getpar('guide_Q',.true.,guide_Q,'guidage de Q')
89    CALL getpar('guide_hr',.true.,guide_hr,'guidage de Q par H.R')
90    CALL getpar('guide_teta',.false.,guide_teta,'guidage de T par Teta')
91
92    CALL getpar('guide_add',.false.,guide_add,'for�age constant?')
93    CALL getpar('guide_zon',.false.,guide_zon,'guidage moy zonale')
94    if (guide_zon .and. abs(grossismx - 1.) > 0.01) &
95         call abort_gcm("guide_init", &
96         "zonal nudging requires grid regular in longitude", 1)
97
98!   Constantes de rappel. Unite : fraction de jour
99    CALL getpar('tau_min_u',0.02,tau_min_u,'Cste de rappel min, u')
100    CALL getpar('tau_max_u', 10.,tau_max_u,'Cste de rappel max, u')
101    CALL getpar('tau_min_v',0.02,tau_min_v,'Cste de rappel min, v')
102    CALL getpar('tau_max_v', 10.,tau_max_v,'Cste de rappel max, v')
103    CALL getpar('tau_min_T',0.02,tau_min_T,'Cste de rappel min, T')
104    CALL getpar('tau_max_T', 10.,tau_max_T,'Cste de rappel max, T')
105    CALL getpar('tau_min_Q',0.02,tau_min_Q,'Cste de rappel min, Q')
106    CALL getpar('tau_max_Q', 10.,tau_max_Q,'Cste de rappel max, Q')
107    CALL getpar('tau_min_P',0.02,tau_min_P,'Cste de rappel min, P')
108    CALL getpar('tau_max_P', 10.,tau_max_P,'Cste de rappel max, P')
109    CALL getpar('gamma4',.false.,gamma4,'Zone sans rappel elargie')
110    CALL getpar('guide_BL',.true.,guide_BL,'guidage dans C.Lim')
111   
112! Sauvegarde du for�age
113    CALL getpar('guide_sav',.false.,guide_sav,'sauvegarde guidage')
114    CALL getpar('iguide_sav',4,iguide_sav,'freq. sauvegarde guidage')
115    ! frequences f>0: fx/jour; f<0: tous les f jours; f=0: 1 seule fois.
116    IF (iguide_sav.GT.0) THEN
117       iguide_sav=day_step/iguide_sav
118    ELSE if (iguide_sav == 0) then
119       iguide_sav = huge(0)
120    ELSE
121       iguide_sav=day_step*iguide_sav
122    ENDIF
123
124! Guidage regional seulement (sinon constant ou suivant le zoom)
125    CALL getpar('guide_reg',.false.,guide_reg,'guidage regional')
126    CALL getpar('lat_min_g',-90.,lat_min_g,'Latitude mini guidage ')
127    CALL getpar('lat_max_g', 90.,lat_max_g,'Latitude maxi guidage ')
128    CALL getpar('lon_min_g',-180.,lon_min_g,'longitude mini guidage ')
129    CALL getpar('lon_max_g', 180.,lon_max_g,'longitude maxi guidage ')
130    CALL getpar('tau_lat', 5.,tau_lat,'raideur lat guide regional ')
131    CALL getpar('tau_lon', 5.,tau_lon,'raideur lon guide regional ')
132
133! Parametres pour lecture des fichiers
134    CALL getpar('iguide_read',4,iguide_read,'freq. lecture guidage')
135    CALL getpar('iguide_int',4,iguide_int,'freq. interpolation vert')
136    IF (iguide_int.EQ.0) THEN
137        iguide_int=1
138    ELSEIF (iguide_int.GT.0) THEN
139        iguide_int=day_step/iguide_int
140    ELSE
141        iguide_int=day_step*iguide_int
142    ENDIF
143    CALL getpar('guide_modele',.false.,guide_modele,'guidage niveaux modele')
144    CALL getpar('ini_anal',.false.,ini_anal,'Etat initial = analyse')
145    CALL getpar('guide_invertp',.true.,invert_p,'niveaux p inverses')
146    CALL getpar('guide_inverty',.true.,invert_y,'inversion N-S')
147    CALL getpar('guide_2D',.false.,guide_2D,'fichier guidage lat-P')
148
149! ---------------------------------------------
150! Determination du nombre de niveaux verticaux
151! des fichiers guidage
152! ---------------------------------------------
153    ncidpl=-99
154    if (guide_modele) then
155       if (ncidpl.eq.-99) then
156          rcod=nf90_open('apbp.nc',Nf90_NOWRITe, ncidpl)
157          if (rcod.NE.NF_NOERR) THEN
158             print *,'Guide: probleme -> pas de fichier apbp.nc'
159             CALL abort_gcm(modname,abort_message,1)
160          endif
161       endif
162    else
163         if (guide_u) then
164           if (ncidpl.eq.-99) then
165               rcod=nf90_open('u.nc',Nf90_NOWRITe,ncidpl)
166               if (rcod.NE.NF_NOERR) THEN
167                  print *,'Guide: probleme -> pas de fichier u.nc'
168                  CALL abort_gcm(modname,abort_message,1)
169               endif
170           endif
171         elseif (guide_v) then
172           if (ncidpl.eq.-99) then
173               rcod=nf90_open('v.nc',nf90_nowrite,ncidpl)
174               if (rcod.NE.NF_NOERR) THEN
175                  print *,'Guide: probleme -> pas de fichier v.nc'
176                  CALL abort_gcm(modname,abort_message,1)
177               endif
178           endif
179         elseif (guide_T) then
180           if (ncidpl.eq.-99) then
181               rcod=nf90_open('T.nc',nf90_nowrite,ncidpl)
182               if (rcod.NE.NF_NOERR) THEN
183                  print *,'Guide: probleme -> pas de fichier T.nc'
184                  CALL abort_gcm(modname,abort_message,1)
185               endif
186           endif
187         elseif (guide_Q) then
188           if (ncidpl.eq.-99) then
189               rcod=nf90_open('hur.nc',nf90_nowrite, ncidpl)
190               if (rcod.NE.NF_NOERR) THEN
191                  print *,'Guide: probleme -> pas de fichier hur.nc'
192                  CALL abort_gcm(modname,abort_message,1)
193               endif
194           endif
195         endif
196    endif
197    error=NF_INQ_DIMID(ncidpl,'LEVEL',rid)
198    IF (error.NE.NF_NOERR) error=NF_INQ_DIMID(ncidpl,'PRESSURE',rid)
199    IF (error.NE.NF_NOERR) THEN
200        print *,'Guide: probleme lecture niveaux pression'
201        CALL abort_gcm(modname,abort_message,1)
202    ENDIF
203    error=NF_INQ_DIMLEN(ncidpl,rid,nlevnc)
204    print *,'Guide: nombre niveaux vert. nlevnc', nlevnc
205    rcod = nf90_close(ncidpl)
206
207! ---------------------------------------------
208! Allocation des variables
209! ---------------------------------------------
210    abort_message='pb in allocation guide'
211
212    ALLOCATE(apnc(nlevnc), stat = error)
213    IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
214    ALLOCATE(bpnc(nlevnc), stat = error)
215    IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
216    apnc=0.;bpnc=0.
217
218    ALLOCATE(alpha_pcor(llm), stat = error)
219    IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
220    ALLOCATE(alpha_u(ip1jmp1), stat = error)
221    IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
222    ALLOCATE(alpha_v(ip1jm), stat = error)
223    IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
224    ALLOCATE(alpha_T(ip1jmp1), stat = error)
225    IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
226    ALLOCATE(alpha_Q(ip1jmp1), stat = error)
227    IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
228    ALLOCATE(alpha_P(ip1jmp1), stat = error)
229    IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
230    alpha_u=0.;alpha_v=0;alpha_T=0;alpha_Q=0;alpha_P=0
231   
232    IF (guide_u) THEN
233        ALLOCATE(unat1(iip1,jjp1,nlevnc), stat = error)
234        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
235        ALLOCATE(ugui1(ip1jmp1,llm), stat = error)
236        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
237        ALLOCATE(unat2(iip1,jjp1,nlevnc), stat = error)
238        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
239        ALLOCATE(ugui2(ip1jmp1,llm), stat = error)
240        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
241        unat1=0.;unat2=0.;ugui1=0.;ugui2=0.
242    ENDIF
243
244    IF (guide_T) THEN
245        ALLOCATE(tnat1(iip1,jjp1,nlevnc), stat = error)
246        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
247        ALLOCATE(tgui1(ip1jmp1,llm), stat = error)
248        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
249        ALLOCATE(tnat2(iip1,jjp1,nlevnc), stat = error)
250        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
251        ALLOCATE(tgui2(ip1jmp1,llm), stat = error)
252        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
253        tnat1=0.;tnat2=0.;tgui1=0.;tgui2=0.
254    ENDIF
255     
256    IF (guide_Q) THEN
257        ALLOCATE(qnat1(iip1,jjp1,nlevnc), stat = error)
258        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
259        ALLOCATE(qgui1(ip1jmp1,llm), stat = error)
260        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
261        ALLOCATE(qnat2(iip1,jjp1,nlevnc), stat = error)
262        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
263        ALLOCATE(qgui2(ip1jmp1,llm), stat = error)
264        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
265        qnat1=0.;qnat2=0.;qgui1=0.;qgui2=0.
266    ENDIF
267
268    IF (guide_v) THEN
269        ALLOCATE(vnat1(iip1,jjm,nlevnc), stat = error)
270        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
271        ALLOCATE(vgui1(ip1jm,llm), stat = error)
272        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
273        ALLOCATE(vnat2(iip1,jjm,nlevnc), stat = error)
274        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
275        ALLOCATE(vgui2(ip1jm,llm), stat = error)
276        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
277        vnat1=0.;vnat2=0.;vgui1=0.;vgui2=0.
278    ENDIF
279
280    IF (guide_P.OR.guide_modele) THEN
281        ALLOCATE(psnat1(iip1,jjp1), stat = error)
282        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
283        ALLOCATE(psnat2(iip1,jjp1), stat = error)
284        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
285        psnat1=0.;psnat2=0.;
286    ENDIF
287    IF (guide_P) THEN
288        ALLOCATE(psgui2(ip1jmp1), stat = error)
289        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
290        ALLOCATE(psgui1(ip1jmp1), stat = error)
291        IF (error /= 0) CALL abort_gcm(modname,abort_message,1)
292        psgui1=0.;psgui2=0.
293    ENDIF
294
295! ---------------------------------------------
296!   Lecture du premier etat de guidage.
297! ---------------------------------------------
298    IF (guide_2D) THEN
299        CALL guide_read2D(1)
300    ELSE
301        CALL guide_read(1)
302    ENDIF
303    IF (guide_v) vnat1=vnat2
304    IF (guide_u) unat1=unat2
305    IF (guide_T) tnat1=tnat2
306    IF (guide_Q) qnat1=qnat2
307    IF (guide_P.OR.guide_modele) psnat1=psnat2
308
309  END SUBROUTINE guide_init
310
311!=======================================================================
312  SUBROUTINE guide_main(itau,ucov,vcov,teta,q,masse,ps)
313
314    USE control_mod
315 
316    IMPLICIT NONE
317 
318    INCLUDE "dimensions.h"
319    INCLUDE "paramet.h"
320    INCLUDE "comconst.h"
321    INCLUDE "comvert.h"
322
323    ! Variables entree
324    INTEGER,                       INTENT(IN)    :: itau !pas de temps
325    REAL, DIMENSION (ip1jmp1,llm), INTENT(INOUT) :: ucov,teta,q,masse
326    REAL, DIMENSION (ip1jm,llm),   INTENT(INOUT) :: vcov
327    REAL, DIMENSION (ip1jmp1),     INTENT(INOUT) :: ps
328
329    ! Variables locales
330    LOGICAL, SAVE :: first=.TRUE.
331    LOGICAL       :: f_out ! sortie guidage
332    REAL, DIMENSION (ip1jmp1,llm) :: f_add ! var aux: champ de guidage
333    REAL, DIMENSION (ip1jmp1,llm) :: p ! besoin si guide_P
334    ! Compteurs temps:
335    INTEGER, SAVE :: step_rea,count_no_rea,itau_test ! lecture guidage
336    REAL          :: ditau, dday_step
337    REAL          :: tau,reste ! position entre 2 etats de guidage
338    REAL, SAVE    :: factt ! pas de temps en fraction de jour
339   
340    INTEGER       :: l
341
342!-----------------------------------------------------------------------
343! Initialisations au premier passage
344!-----------------------------------------------------------------------
345    IF (first) THEN
346        first=.FALSE.
347        CALL guide_init
348        itau_test=1001
349        step_rea=1
350        count_no_rea=0
351! Calcul des constantes de rappel
352        factt=dtvr*iperiod/daysec
353        call tau2alpha(3,iip1,jjm ,factt,tau_min_v,tau_max_v,alpha_v)
354        call tau2alpha(2,iip1,jjp1,factt,tau_min_u,tau_max_u,alpha_u)
355        call tau2alpha(1,iip1,jjp1,factt,tau_min_T,tau_max_T,alpha_T)
356        call tau2alpha(1,iip1,jjp1,factt,tau_min_P,tau_max_P,alpha_P)
357        call tau2alpha(1,iip1,jjp1,factt,tau_min_Q,tau_max_Q,alpha_Q)
358! correction de rappel dans couche limite
359        if (guide_BL) then
360             alpha_pcor(:)=1.
361        else
362            do l=1,llm
363                alpha_pcor(l)=(1.+tanh((0.85-presnivs(l)/preff)/0.05))/2.
364            enddo
365        endif
366! ini_anal: etat initial egal au guidage       
367        IF (ini_anal) THEN
368            CALL guide_interp(ps,teta)
369            IF (guide_u) ucov=ugui2
370            IF (guide_v) vcov=ugui2
371            IF (guide_T) teta=tgui2
372            IF (guide_Q) q=qgui2
373            IF (guide_P) THEN
374                ps=psgui2
375                CALL pression(ip1jmp1,ap,bp,ps,p)
376                CALL massdair(p,masse)
377            ENDIF
378            RETURN
379        ENDIF
380! Verification structure guidage
381        IF (guide_u) THEN
382            CALL writefield('unat',unat1)
383            CALL writefield('ucov',RESHAPE(ucov,(/iip1,jjp1,llm/)))
384        ENDIF
385        IF (guide_T) THEN
386            CALL writefield('tnat',tnat1)
387            CALL writefield('teta',RESHAPE(teta,(/iip1,jjp1,llm/)))
388        ENDIF
389
390    ENDIF !first
391
392!-----------------------------------------------------------------------
393! Lecture des fichiers de guidage ?
394!-----------------------------------------------------------------------
395    IF (iguide_read.NE.0) THEN
396      ditau=real(itau)
397      dday_step=real(day_step)
398      IF (iguide_read.LT.0) THEN
399          tau=ditau/dday_step/REAL(iguide_read)
400      ELSE
401          tau=REAL(iguide_read)*ditau/dday_step
402      ENDIF
403      reste=tau-AINT(tau)
404      IF (reste.EQ.0.) THEN
405          IF (itau_test.EQ.itau) THEN
406              write(*,*)'deuxieme passage de advreel a itau=',itau
407              stop
408          ELSE
409              IF (guide_v) vnat1=vnat2
410              IF (guide_u) unat1=unat2
411              IF (guide_T) tnat1=tnat2
412              IF (guide_Q) qnat1=qnat2
413              IF (guide_P.OR.guide_modele) psnat1=psnat2
414              step_rea=step_rea+1
415              itau_test=itau
416              print*,'Lecture fichiers guidage, pas ',step_rea, &
417                    'apres ',count_no_rea,' non lectures'
418              IF (guide_2D) THEN
419                  CALL guide_read2D(step_rea)
420              ELSE
421                  CALL guide_read(step_rea)
422              ENDIF
423              count_no_rea=0
424          ENDIF
425      ELSE
426        count_no_rea=count_no_rea+1
427
428      ENDIF
429    ENDIF !iguide_read=0
430
431!-----------------------------------------------------------------------
432! Interpolation et conversion des champs de guidage
433!-----------------------------------------------------------------------
434    IF (MOD(itau,iguide_int).EQ.0) THEN
435        CALL guide_interp(ps,teta)
436    ENDIF
437! Repartition entre 2 etats de guidage
438    IF (iguide_read.NE.0) THEN
439        tau=reste
440    ELSE
441        tau=1.
442    ENDIF
443
444!-----------------------------------------------------------------------
445!   Ajout des champs de guidage
446!-----------------------------------------------------------------------
447! Sauvegarde du guidage?
448    f_out=((MOD(itau,iguide_sav).EQ.0).AND.guide_sav) 
449    IF (f_out) CALL guide_out("SP",jjp1,1,ps)
450   
451    if (guide_u) then
452        if (guide_add) then
453           f_add=(1.-tau)*ugui1+tau*ugui2
454        else
455           f_add=(1.-tau)*ugui1+tau*ugui2-ucov
456        endif
457        if (guide_zon) CALL guide_zonave(1,jjp1,llm,f_add)
458        CALL guide_addfield(ip1jmp1,llm,f_add,alpha_u)
459        IF (f_out) CALL guide_out("ua",jjp1,llm,(1.-tau)*ugui1+tau*ugui2)
460        IF (f_out) CALL guide_out("u",jjp1,llm,ucov)
461        IF (f_out) CALL guide_out("ucov",jjp1,llm,f_add/factt)
462        ucov=ucov+f_add
463    endif
464
465    if (guide_T) then
466        if (guide_add) then
467           f_add=(1.-tau)*tgui1+tau*tgui2
468        else
469           f_add=(1.-tau)*tgui1+tau*tgui2-teta
470        endif
471        if (guide_zon) CALL guide_zonave(2,jjp1,llm,f_add)
472        CALL guide_addfield(ip1jmp1,llm,f_add,alpha_T)
473        IF (f_out) CALL guide_out("teta",jjp1,llm,f_add/factt)
474        teta=teta+f_add
475    endif
476
477    if (guide_P) then
478        if (guide_add) then
479           f_add(1:ip1jmp1,1)=(1.-tau)*psgui1+tau*psgui2
480        else
481           f_add(1:ip1jmp1,1)=(1.-tau)*psgui1+tau*psgui2-ps
482        endif
483        if (guide_zon) CALL guide_zonave(2,jjp1,1,f_add(1:ip1jmp1,1))
484        CALL guide_addfield(ip1jmp1,1,f_add(1:ip1jmp1,1),alpha_P)
485        IF (f_out) CALL guide_out("ps",jjp1,1,f_add(1:ip1jmp1,1)/factt)
486        ps=ps+f_add(1:ip1jmp1,1)
487        CALL pression(ip1jmp1,ap,bp,ps,p)
488        CALL massdair(p,masse)
489    endif
490
491    if (guide_Q) then
492        if (guide_add) then
493           f_add=(1.-tau)*qgui1+tau*qgui2
494        else
495           f_add=(1.-tau)*qgui1+tau*qgui2-q
496        endif
497        if (guide_zon) CALL guide_zonave(2,jjp1,llm,f_add)
498        CALL guide_addfield(ip1jmp1,llm,f_add,alpha_Q)
499        IF (f_out) CALL guide_out("q",jjp1,llm,f_add/factt)
500        q=q+f_add
501    endif
502
503    if (guide_v) then
504        if (guide_add) then
505           f_add(1:ip1jm,:)=(1.-tau)*vgui1+tau*vgui2
506        else
507           f_add(1:ip1jm,:)=(1.-tau)*vgui1+tau*vgui2-vcov
508        endif
509        if (guide_zon) CALL guide_zonave(2,jjm,llm,f_add(1:ip1jm,:))
510        CALL guide_addfield(ip1jm,llm,f_add(1:ip1jm,:),alpha_v)
511        IF (f_out) CALL guide_out("v",jjm,llm,vcov)
512        IF (f_out) CALL guide_out("va",jjm,llm,(1.-tau)*vgui1+tau*vgui2)
513        IF (f_out) CALL guide_out("vcov",jjm,llm,f_add(1:ip1jm,:)/factt)
514        vcov=vcov+f_add(1:ip1jm,:)
515    endif
516  END SUBROUTINE guide_main
517
518!=======================================================================
519  SUBROUTINE guide_addfield(hsize,vsize,field,alpha)
520! field1=a*field1+alpha*field2
521
522    IMPLICIT NONE
523
524    ! input variables
525    INTEGER,                      INTENT(IN)    :: hsize
526    INTEGER,                      INTENT(IN)    :: vsize
527    REAL, DIMENSION(hsize),       INTENT(IN)    :: alpha
528    REAL, DIMENSION(hsize,vsize), INTENT(INOUT) :: field
529
530    ! Local variables
531    INTEGER :: l
532
533    do l=1,vsize
534        field(:,l)=alpha*field(:,l)*alpha_pcor(l)
535    enddo
536
537  END SUBROUTINE guide_addfield
538
539!=======================================================================
540  SUBROUTINE guide_zonave(typ,hsize,vsize,field)
541
542    IMPLICIT NONE
543
544    INCLUDE "dimensions.h"
545    INCLUDE "paramet.h"
546    INCLUDE "comgeom.h"
547    INCLUDE "comconst.h"
548   
549    ! input/output variables
550    INTEGER,                           INTENT(IN)    :: typ
551    INTEGER,                           INTENT(IN)    :: vsize
552    INTEGER,                           INTENT(IN)    :: hsize
553    REAL, DIMENSION(hsize*iip1,vsize), INTENT(INOUT) :: field
554
555    ! Local variables
556    LOGICAL, SAVE                :: first=.TRUE.
557    INTEGER, DIMENSION (2), SAVE :: imin, imax ! averaging domain
558    INTEGER                      :: i,j,l,ij
559    REAL, DIMENSION (iip1)       :: lond       ! longitude in Deg.
560    REAL, DIMENSION (hsize,vsize):: fieldm     ! zon-averaged field
561
562    IF (first) THEN
563        first=.FALSE.
564!Compute domain for averaging
565        lond=rlonu*180./pi
566        imin(1)=1;imax(1)=iip1;
567        imin(2)=1;imax(2)=iip1;
568        IF (guide_reg) THEN
569            DO i=1,iim
570                IF (lond(i).LT.lon_min_g) imin(1)=i
571                IF (lond(i).LE.lon_max_g) imax(1)=i
572            ENDDO
573            lond=rlonv*180./pi
574            DO i=1,iim
575                IF (lond(i).LT.lon_min_g) imin(2)=i
576                IF (lond(i).LE.lon_max_g) imax(2)=i
577            ENDDO
578        ENDIF
579    ENDIF
580
581    fieldm=0.
582    DO l=1,vsize
583    ! Compute zonal average
584        DO j=1,hsize
585            DO i=imin(typ),imax(typ)
586                ij=(j-1)*iip1+i
587                fieldm(j,l)=fieldm(j,l)+field(ij,l)
588            ENDDO
589        ENDDO
590        fieldm(:,l)=fieldm(:,l)/REAL(imax(typ)-imin(typ)+1)
591    ! Compute forcing
592        DO j=1,hsize
593            DO i=1,iip1
594                ij=(j-1)*iip1+i
595                field(ij,l)=fieldm(j,l)
596            ENDDO
597        ENDDO
598    ENDDO
599
600  END SUBROUTINE guide_zonave
601
602!=======================================================================
603  SUBROUTINE guide_interp(psi,teta)
604 
605  use exner_hyb_m, only: exner_hyb
606  use exner_milieu_m, only: exner_milieu
607  IMPLICIT NONE
608
609  include "dimensions.h"
610  include "paramet.h"
611  include "comvert.h"
612  include "comgeom2.h"
613  include "comconst.h"
614
615  REAL, DIMENSION (iip1,jjp1),     INTENT(IN) :: psi ! Psol gcm
616  REAL, DIMENSION (iip1,jjp1,llm), INTENT(IN) :: teta ! Temp. Pot. gcm
617
618  LOGICAL, SAVE                      :: first=.TRUE.
619  ! Variables pour niveaux pression:
620  REAL, DIMENSION (iip1,jjp1,nlevnc) :: plnc1,plnc2 !niveaux pression guidage
621  REAL, DIMENSION (iip1,jjp1,llm)    :: plunc,plsnc !niveaux pression modele
622  REAL, DIMENSION (iip1,jjm,llm)     :: plvnc       !niveaux pression modele
623  REAL, DIMENSION (iip1,jjp1,llmp1)  :: p           ! pression intercouches
624  REAL, DIMENSION (iip1,jjp1,llm)    :: pls, pext   ! var intermediaire
625  REAL, DIMENSION (iip1,jjp1,llm)    :: pbarx
626  REAL, DIMENSION (iip1,jjm,llm)     :: pbary
627  ! Variables pour fonction Exner (P milieu couche)
628  REAL, DIMENSION (iip1,jjp1,llm)    :: pk
629  REAL, DIMENSION (iip1,jjp1)        :: pks   
630  REAL                               :: prefkap,unskap
631  ! Pression de vapeur saturante
632  REAL, DIMENSION (ip1jmp1,llm)      :: qsat
633  !Variables intermediaires interpolation
634  REAL, DIMENSION (iip1,jjp1,llm)    :: zu1,zu2
635  REAL, DIMENSION (iip1,jjm,llm)     :: zv1,zv2
636 
637  INTEGER                            :: i,j,l,ij
638 
639    print *,'Guide: conversion variables guidage'
640! -----------------------------------------------------------------
641! Calcul des niveaux de pression champs guidage
642! -----------------------------------------------------------------
643if (guide_modele) then
644    do i=1,iip1
645        do j=1,jjp1
646            do l=1,nlevnc
647                plnc2(i,j,l)=apnc(l)+bpnc(l)*psnat2(i,j)
648                plnc1(i,j,l)=apnc(l)+bpnc(l)*psnat1(i,j)
649            enddo
650        enddo
651    enddo
652else
653    do i=1,iip1
654        do j=1,jjp1
655            do l=1,nlevnc
656                plnc2(i,j,l)=apnc(l)
657                plnc1(i,j,l)=apnc(l)
658           enddo
659        enddo
660    enddo
661
662endif
663    if (first) then
664        first=.FALSE.
665        print*,'Guide: verification ordre niveaux verticaux'
666        print*,'LMDZ :'
667        do l=1,llm
668            print*,'PL(',l,')=',(ap(l)+ap(l+1))/2. &
669                  +psi(1,jjp1)*(bp(l)+bp(l+1))/2.
670        enddo
671        print*,'Fichiers guidage'
672        do l=1,nlevnc
673             print*,'PL(',l,')=',plnc2(1,1,l)
674        enddo
675        print *,'inversion de l''ordre: invert_p=',invert_p
676        if (guide_u) then
677            do l=1,nlevnc
678                print*,'U(',l,')=',unat2(1,1,l)
679            enddo
680        endif
681        if (guide_T) then
682            do l=1,nlevnc
683                print*,'T(',l,')=',tnat2(1,1,l)
684            enddo
685        endif
686    endif
687   
688! -----------------------------------------------------------------
689! Calcul niveaux pression modele
690! -----------------------------------------------------------------
691    CALL pression( ip1jmp1, ap, bp, psi, p )
692    if (pressure_exner) then
693      CALL exner_hyb(ip1jmp1,psi,p,pks,pk)
694    else
695      CALL exner_milieu(ip1jmp1,psi,p,pks,pk)
696    endif
697!    ....  Calcul de pls , pression au milieu des couches ,en Pascals
698    unskap=1./kappa
699    prefkap =  preff  ** kappa
700    DO l = 1, llm
701        DO j=1,jjp1
702            DO i =1, iip1
703                pls(i,j,l) = preff * ( pk(i,j,l)/cpp) ** unskap
704            ENDDO
705        ENDDO
706    ENDDO
707
708!   calcul des pressions pour les grilles u et v
709    do l=1,llm
710        do j=1,jjp1
711            do i=1,iip1
712                pext(i,j,l)=pls(i,j,l)*aire(i,j)
713            enddo
714        enddo
715    enddo
716    call massbar(pext, pbarx, pbary )
717    do l=1,llm
718        do j=1,jjp1
719            do i=1,iip1
720                plunc(i,j,l)=pbarx(i,j,l)/aireu(i,j)
721                plsnc(i,j,l)=pls(i,j,l)
722            enddo
723        enddo
724    enddo
725    do l=1,llm
726        do j=1,jjm
727            do i=1,iip1
728                plvnc(i,j,l)=pbary(i,j,l)/airev(i,j)
729            enddo
730        enddo
731    enddo
732
733! -----------------------------------------------------------------
734! Interpolation champs guidage sur niveaux modele (+inversion N/S)
735! Conversion en variables gcm (ucov, vcov...)
736! -----------------------------------------------------------------
737    if (guide_P) then
738        do j=1,jjp1
739            do i=1,iim
740                ij=(j-1)*iip1+i
741                psgui1(ij)=psnat1(i,j)
742                psgui2(ij)=psnat2(i,j)
743            enddo
744            psgui1(iip1*j)=psnat1(1,j)
745            psgui2(iip1*j)=psnat2(1,j)
746        enddo
747    endif
748
749    IF (guide_u) THEN
750        CALL pres2lev(unat1,zu1,nlevnc,llm,plnc1,plunc,iip1,jjp1,invert_p)
751        CALL pres2lev(unat2,zu2,nlevnc,llm,plnc2,plunc,iip1,jjp1,invert_p)
752        do l=1,llm
753            do j=1,jjp1
754                do i=1,iim
755                    ij=(j-1)*iip1+i
756                    ugui1(ij,l)=zu1(i,j,l)*cu(i,j)
757                    ugui2(ij,l)=zu2(i,j,l)*cu(i,j)
758                enddo
759                ugui1(j*iip1,l)=ugui1((j-1)*iip1+1,l)   
760                ugui2(j*iip1,l)=ugui2((j-1)*iip1+1,l)   
761            enddo
762            do i=1,iip1
763                ugui1(i,l)=0.
764                ugui1(ip1jm+i,l)=0.
765                ugui2(i,l)=0.
766                ugui2(ip1jm+i,l)=0.
767            enddo
768        enddo
769    ENDIF
770   
771    IF (guide_T) THEN
772        CALL pres2lev(tnat1,zu1,nlevnc,llm,plnc1,plsnc,iip1,jjp1,invert_p)
773        CALL pres2lev(tnat2,zu2,nlevnc,llm,plnc2,plsnc,iip1,jjp1,invert_p)
774        do l=1,llm
775            do j=1,jjp1
776                IF (guide_teta) THEN
777                    do i=1,iim
778                        ij=(j-1)*iip1+i
779                        tgui1(ij,l)=zu1(i,j,l)
780                        tgui2(ij,l)=zu2(i,j,l)
781                    enddo
782                ELSE
783                    do i=1,iim
784                        ij=(j-1)*iip1+i
785                        tgui1(ij,l)=zu1(i,j,l)*cpp/pk(i,j,l)
786                        tgui2(ij,l)=zu2(i,j,l)*cpp/pk(i,j,l)
787                    enddo
788                ENDIF
789                tgui1(j*iip1,l)=tgui1((j-1)*iip1+1,l)   
790                tgui2(j*iip1,l)=tgui2((j-1)*iip1+1,l)   
791            enddo
792            do i=1,iip1
793                tgui1(i,l)=tgui1(1,l)
794                tgui1(ip1jm+i,l)=tgui1(ip1jm+1,l)
795                tgui2(i,l)=tgui2(1,l)
796                tgui2(ip1jm+i,l)=tgui2(ip1jm+1,l)
797            enddo
798        enddo
799    ENDIF
800
801    IF (guide_v) THEN
802
803        CALL pres2lev(vnat1,zv1,nlevnc,llm,plnc1(:,:jjm,:),plvnc,iip1,jjm,invert_p)
804        CALL pres2lev(vnat2,zv2,nlevnc,llm,plnc2(:,:jjm,:),plvnc,iip1,jjm,invert_p)
805
806        do l=1,llm
807            do j=1,jjm
808                do i=1,iim
809                    ij=(j-1)*iip1+i
810                    vgui1(ij,l)=zv1(i,j,l)*cv(i,j)
811                    vgui2(ij,l)=zv2(i,j,l)*cv(i,j)
812                enddo
813                vgui1(j*iip1,l)=vgui1((j-1)*iip1+1,l)   
814                vgui2(j*iip1,l)=vgui2((j-1)*iip1+1,l)   
815            enddo
816        enddo
817    ENDIF
818   
819    IF (guide_Q) THEN
820        ! On suppose qu'on a la bonne variable dans le fichier de guidage:
821        ! Hum.Rel si guide_hr, Hum.Spec. sinon.
822        CALL pres2lev(qnat1,zu1,nlevnc,llm,plnc1,plsnc,iip1,jjp1,invert_p)
823        CALL pres2lev(qnat2,zu2,nlevnc,llm,plnc2,plsnc,iip1,jjp1,invert_p)
824        do l=1,llm
825            do j=1,jjp1
826                do i=1,iim
827                    ij=(j-1)*iip1+i
828                    qgui1(ij,l)=zu1(i,j,l)
829                    qgui2(ij,l)=zu2(i,j,l)
830                enddo
831                qgui1(j*iip1,l)=qgui1((j-1)*iip1+1,l)   
832                qgui2(j*iip1,l)=qgui2((j-1)*iip1+1,l)   
833            enddo
834            do i=1,iip1
835                qgui1(i,l)=qgui1(1,l)
836                qgui1(ip1jm+i,l)=qgui1(ip1jm+1,l)
837                qgui2(i,l)=qgui2(1,l)
838                qgui2(ip1jm+i,l)=qgui2(ip1jm+1,l)
839            enddo
840        enddo
841        IF (guide_hr) THEN
842            CALL q_sat(iip1*jjp1*llm,teta*pk/cpp,plsnc,qsat)
843            qgui1=qgui1*qsat*0.01 !hum. rel. en %
844            qgui2=qgui2*qsat*0.01
845        ENDIF
846    ENDIF
847
848  END SUBROUTINE guide_interp
849
850!=======================================================================
851  SUBROUTINE tau2alpha(typ,pim,pjm,factt,taumin,taumax,alpha)
852
853! Calcul des constantes de rappel alpha (=1/tau)
854
855    implicit none
856
857    include "dimensions.h"
858    include "paramet.h"
859    include "comconst.h"
860    include "comgeom2.h"
861    include "serre.h"
862
863! input arguments :
864    INTEGER, INTENT(IN) :: typ    ! u(2),v(3), ou scalaire(1)
865    INTEGER, INTENT(IN) :: pim,pjm ! dimensions en lat, lon
866    REAL, INTENT(IN)    :: factt   ! pas de temps en fraction de jour
867    REAL, INTENT(IN)    :: taumin,taumax
868! output arguments:
869    REAL, DIMENSION(pim,pjm), INTENT(OUT) :: alpha
870 
871!  local variables:
872    LOGICAL, SAVE               :: first=.TRUE.
873    REAL, SAVE                  :: gamma,dxdy_min,dxdy_max
874    REAL, DIMENSION (iip1,jjp1) :: zdx,zdy
875    REAL, DIMENSION (iip1,jjp1) :: dxdys,dxdyu
876    REAL, DIMENSION (iip1,jjm)  :: dxdyv
877    real dxdy_
878    real zlat,zlon
879    real alphamin,alphamax,xi
880    integer i,j,ilon,ilat
881
882
883    alphamin=factt/taumax
884    alphamax=factt/taumin
885    IF (guide_reg.OR.guide_add) THEN
886        alpha=alphamax
887!-----------------------------------------------------------------------
888! guide_reg: alpha=alpha_min dans region, 0. sinon.
889!-----------------------------------------------------------------------
890        IF (guide_reg) THEN
891            do j=1,pjm
892                do i=1,pim
893                    if (typ.eq.2) then
894                       zlat=rlatu(j)*180./pi
895                       zlon=rlonu(i)*180./pi
896                    elseif (typ.eq.1) then
897                       zlat=rlatu(j)*180./pi
898                       zlon=rlonv(i)*180./pi
899                    elseif (typ.eq.3) then
900                       zlat=rlatv(j)*180./pi
901                       zlon=rlonv(i)*180./pi
902                    endif
903                    alpha(i,j)=alphamax/16.* &
904                              (1.+tanh((zlat-lat_min_g)/tau_lat))* &
905                              (1.+tanh((lat_max_g-zlat)/tau_lat))* &
906                              (1.+tanh((zlon-lon_min_g)/tau_lon))* &
907                              (1.+tanh((lon_max_g-zlon)/tau_lon))
908                enddo
909            enddo
910        ENDIF
911    ELSE
912!-----------------------------------------------------------------------
913! Sinon, alpha varie entre alpha_min et alpha_max suivant le zoom.
914!-----------------------------------------------------------------------
915!Calcul de l'aire des mailles
916        do j=2,jjm
917            do i=2,iip1
918               zdx(i,j)=0.5*(cu(i-1,j)+cu(i,j))/cos(rlatu(j))
919            enddo
920            zdx(1,j)=zdx(iip1,j)
921        enddo
922        do j=2,jjm
923            do i=1,iip1
924               zdy(i,j)=0.5*(cv(i,j-1)+cv(i,j))
925            enddo
926        enddo
927        do i=1,iip1
928            zdx(i,1)=zdx(i,2)
929            zdx(i,jjp1)=zdx(i,jjm)
930            zdy(i,1)=zdy(i,2)
931            zdy(i,jjp1)=zdy(i,jjm)
932        enddo
933        do j=1,jjp1
934            do i=1,iip1
935               dxdys(i,j)=sqrt(zdx(i,j)*zdx(i,j)+zdy(i,j)*zdy(i,j))
936            enddo
937        enddo
938        IF (typ.EQ.2) THEN
939            do j=1,jjp1
940                do i=1,iim
941                   dxdyu(i,j)=0.5*(dxdys(i,j)+dxdys(i+1,j))
942                enddo
943                dxdyu(iip1,j)=dxdyu(1,j)
944            enddo
945        ENDIF
946        IF (typ.EQ.3) THEN
947            do j=1,jjm
948                do i=1,iip1
949                   dxdyv(i,j)=0.5*(dxdys(i,j)+dxdys(i,j+1))
950                enddo
951            enddo
952        ENDIF
953! Premier appel: calcul des aires min et max et de gamma.
954        IF (first) THEN
955            first=.FALSE.
956            ! coordonnees du centre du zoom
957            CALL coordij(clon,clat,ilon,ilat)
958            ! aire de la maille au centre du zoom
959            dxdy_min=dxdys(ilon,ilat)
960            ! dxdy maximale de la maille
961            dxdy_max=0.
962            do j=1,jjp1
963                do i=1,iip1
964                     dxdy_max=max(dxdy_max,dxdys(i,j))
965                enddo
966            enddo
967            ! Calcul de gamma
968            if (abs(grossismx-1.).lt.0.1.or.abs(grossismy-1.).lt.0.1) then
969                 print*,'ATTENTION modele peu zoome'
970                 print*,'ATTENTION on prend une constante de guidage cste'
971                 gamma=0.
972            else
973                gamma=(dxdy_max-2.*dxdy_min)/(dxdy_max-dxdy_min)
974                print*,'gamma=',gamma
975                if (gamma.lt.1.e-5) then
976                  print*,'gamma =',gamma,'<1e-5'
977                  stop
978                endif
979                gamma=log(0.5)/log(gamma)
980                if (gamma4) then
981                  gamma=min(gamma,4.)
982                endif
983                print*,'gamma=',gamma
984            endif
985        ENDIF !first
986
987        do j=1,pjm
988            do i=1,pim
989                if (typ.eq.1) then
990                   dxdy_=dxdys(i,j)
991                   zlat=rlatu(j)*180./pi
992                elseif (typ.eq.2) then
993                   dxdy_=dxdyu(i,j)
994                   zlat=rlatu(j)*180./pi
995                elseif (typ.eq.3) then
996                   dxdy_=dxdyv(i,j)
997                   zlat=rlatv(j)*180./pi
998                endif
999                if (abs(grossismx-1.).lt.0.1.or.abs(grossismy-1.).lt.0.1) then
1000                ! pour une grille reguliere, xi=xxx**0=1 -> alpha=alphamin
1001                    alpha(i,j)=alphamin
1002                else
1003                    xi=((dxdy_max-dxdy_)/(dxdy_max-dxdy_min))**gamma
1004                    xi=min(xi,1.)
1005                    if(lat_min_g.le.zlat .and. zlat.le.lat_max_g) then
1006                        alpha(i,j)=xi*alphamin+(1.-xi)*alphamax
1007                    else
1008                        alpha(i,j)=0.
1009                    endif
1010                endif
1011            enddo
1012        enddo
1013    ENDIF ! guide_reg
1014
1015    if (.not. guide_add) alpha = 1. - exp(- alpha)
1016
1017  END SUBROUTINE tau2alpha
1018
1019!=======================================================================
1020  SUBROUTINE guide_read(timestep)
1021
1022    IMPLICIT NONE
1023
1024#include "netcdf.inc"
1025#include "dimensions.h"
1026#include "paramet.h"
1027
1028    INTEGER, INTENT(IN)   :: timestep
1029
1030    LOGICAL, SAVE         :: first=.TRUE.
1031! Identification fichiers et variables NetCDF:
1032    INTEGER, SAVE         :: ncidu,varidu,ncidv,varidv,ncidQ
1033    INTEGER, SAVE         :: varidQ,ncidt,varidt,ncidps,varidps
1034    INTEGER               :: ncidpl,varidpl,varidap,varidbp
1035! Variables auxiliaires NetCDF:
1036    INTEGER, DIMENSION(4) :: start,count
1037    INTEGER               :: status,rcode
1038
1039    CHARACTER (len = 80)   :: abort_message
1040    CHARACTER (len = 20)   :: modname = 'guide_read'
1041! -----------------------------------------------------------------
1042! Premier appel: initialisation de la lecture des fichiers
1043! -----------------------------------------------------------------
1044    if (first) then
1045         ncidpl=-99
1046         print*,'Guide: ouverture des fichiers guidage '
1047! Niveaux de pression si non constants
1048         if (guide_modele) then
1049             print *,'Lecture du guidage sur niveaux modele'
1050             rcode = nf90_open('apbp.nc', nf90_nowrite, ncidpl)
1051             IF (rcode.NE.NF_NOERR) THEN
1052              print *,'Guide: probleme -> pas de fichier apbp.nc'
1053              CALL abort_gcm(modname,abort_message,1)
1054             ENDIF
1055             rcode = nf90_inq_varid(ncidpl, 'AP', varidap)
1056             IF (rcode.NE.NF_NOERR) THEN
1057              print *,'Guide: probleme -> pas de variable AP, fichier apbp.nc'
1058              CALL abort_gcm(modname,abort_message,1)
1059             ENDIF
1060             rcode = nf90_inq_varid(ncidpl, 'BP', varidbp)
1061             IF (rcode.NE.NF_NOERR) THEN
1062              print *,'Guide: probleme -> pas de variable BP, fichier apbp.nc'
1063              CALL abort_gcm(modname,abort_message,1)
1064             ENDIF
1065             print*,'ncidpl,varidap',ncidpl,varidap
1066         endif
1067! Vent zonal
1068         if (guide_u) then
1069             rcode = nf90_open('u.nc', nf90_nowrite, ncidu)
1070             IF (rcode.NE.NF_NOERR) THEN
1071              print *,'Guide: probleme -> pas de fichier u.nc'
1072              CALL abort_gcm(modname,abort_message,1)
1073             ENDIF
1074             rcode = nf90_inq_varid(ncidu, 'UWND', varidu)
1075             IF (rcode.NE.NF_NOERR) THEN
1076              print *,'Guide: probleme -> pas de variable UWND, fichier u.nc'
1077              CALL abort_gcm(modname,abort_message,1)
1078             ENDIF
1079             print*,'ncidu,varidu',ncidu,varidu
1080             if (ncidpl.eq.-99) ncidpl=ncidu
1081         endif
1082! Vent meridien
1083         if (guide_v) then
1084             rcode = nf90_open('v.nc', nf90_nowrite, ncidv)
1085             IF (rcode.NE.NF_NOERR) THEN
1086              print *,'Guide: probleme -> pas de fichier v.nc'
1087              CALL abort_gcm(modname,abort_message,1)
1088             ENDIF
1089             rcode = nf90_inq_varid(ncidv, 'VWND', varidv)
1090             IF (rcode.NE.NF_NOERR) THEN
1091              print *,'Guide: probleme -> pas de variable VWND, fichier v.nc'
1092              CALL abort_gcm(modname,abort_message,1)
1093             ENDIF
1094             print*,'ncidv,varidv',ncidv,varidv
1095             if (ncidpl.eq.-99) ncidpl=ncidv
1096         endif
1097! Temperature
1098         if (guide_T) then
1099             rcode = nf90_open('T.nc', nf90_nowrite, ncidt)
1100             IF (rcode.NE.NF_NOERR) THEN
1101              print *,'Guide: probleme -> pas de fichier T.nc'
1102              CALL abort_gcm(modname,abort_message,1)
1103             ENDIF
1104             rcode = nf90_inq_varid(ncidt, 'AIR', varidt)
1105             IF (rcode.NE.NF_NOERR) THEN
1106              print *,'Guide: probleme -> pas de variable AIR, fichier T.nc'
1107              CALL abort_gcm(modname,abort_message,1)
1108             ENDIF
1109             print*,'ncidT,varidT',ncidt,varidt
1110             if (ncidpl.eq.-99) ncidpl=ncidt
1111         endif
1112! Humidite
1113         if (guide_Q) then
1114             rcode = nf90_open('hur.nc', nf90_nowrite, ncidQ)
1115             IF (rcode.NE.NF_NOERR) THEN
1116              print *,'Guide: probleme -> pas de fichier hur.nc'
1117              CALL abort_gcm(modname,abort_message,1)
1118             ENDIF
1119             rcode = nf90_inq_varid(ncidQ, 'RH', varidQ)
1120             IF (rcode.NE.NF_NOERR) THEN
1121              print *,'Guide: probleme -> pas de variable RH, fichier hur.nc'
1122              CALL abort_gcm(modname,abort_message,1)
1123             ENDIF
1124             print*,'ncidQ,varidQ',ncidQ,varidQ
1125             if (ncidpl.eq.-99) ncidpl=ncidQ
1126         endif
1127! Pression de surface
1128         if ((guide_P).OR.(guide_modele)) then
1129             rcode = nf90_open('ps.nc', nf90_nowrite, ncidps)
1130             IF (rcode.NE.NF_NOERR) THEN
1131              print *,'Guide: probleme -> pas de fichier ps.nc'
1132              CALL abort_gcm(modname,abort_message,1)
1133             ENDIF
1134             rcode = nf90_inq_varid(ncidps, 'SP', varidps)
1135             IF (rcode.NE.NF_NOERR) THEN
1136              print *,'Guide: probleme -> pas de variable SP, fichier ps.nc'
1137              CALL abort_gcm(modname,abort_message,1)
1138             ENDIF
1139             print*,'ncidps,varidps',ncidps,varidps
1140         endif
1141! Coordonnee verticale
1142         if (.not.guide_modele) then
1143              rcode = nf90_inq_varid(ncidpl, 'LEVEL', varidpl)
1144              IF (rcode.NE.0) rcode = nf90_inq_varid(ncidpl, 'PRESSURE', varidpl)
1145              print*,'ncidpl,varidpl',ncidpl,varidpl
1146         endif
1147! Coefs ap, bp pour calcul de la pression aux differents niveaux
1148         if (guide_modele) then
1149#ifdef NC_DOUBLE
1150             status=NF_GET_VARA_DOUBLE(ncidpl,varidap,1,nlevnc,apnc)
1151             status=NF_GET_VARA_DOUBLE(ncidpl,varidbp,1,nlevnc,bpnc)
1152#else
1153             status=NF_GET_VARA_REAL(ncidpl,varidap,1,nlevnc,apnc)
1154             status=NF_GET_VARA_REAL(ncidpl,varidbp,1,nlevnc,bpnc)
1155#endif
1156         else
1157#ifdef NC_DOUBLE
1158             status=NF_GET_VARA_DOUBLE(ncidpl,varidpl,1,nlevnc,apnc)
1159#else
1160             status=NF_GET_VARA_REAL(ncidpl,varidpl,1,nlevnc,apnc)
1161#endif
1162             apnc=apnc*100.! conversion en Pascals
1163             bpnc(:)=0.
1164         endif
1165         first=.FALSE.
1166     endif ! (first)
1167
1168! -----------------------------------------------------------------
1169!   lecture des champs u, v, T, Q, ps
1170! -----------------------------------------------------------------
1171
1172!  dimensions pour les champs scalaires et le vent zonal
1173     start(1)=1
1174     start(2)=1
1175     start(3)=1
1176     start(4)=timestep
1177
1178     count(1)=iip1
1179     count(2)=jjp1
1180     count(3)=nlevnc
1181     count(4)=1
1182
1183!  Vent zonal
1184     if (guide_u) then
1185#ifdef NC_DOUBLE
1186         status=NF_GET_VARA_DOUBLE(ncidu,varidu,start,count,unat2)
1187#else
1188         status=NF_GET_VARA_REAL(ncidu,varidu,start,count,unat2)
1189#endif
1190         IF (invert_y) THEN
1191           CALL invert_lat(iip1,jjp1,nlevnc,unat2)
1192         ENDIF
1193     endif
1194
1195!  Temperature
1196     if (guide_T) then
1197#ifdef NC_DOUBLE
1198         status=NF_GET_VARA_DOUBLE(ncidt,varidt,start,count,tnat2)
1199#else
1200         status=NF_GET_VARA_REAL(ncidt,varidt,start,count,tnat2)
1201#endif
1202         IF (invert_y) THEN
1203           CALL invert_lat(iip1,jjp1,nlevnc,tnat2)
1204         ENDIF
1205     endif
1206
1207!  Humidite
1208     if (guide_Q) then
1209#ifdef NC_DOUBLE
1210         status=NF_GET_VARA_DOUBLE(ncidQ,varidQ,start,count,qnat2)
1211#else
1212         status=NF_GET_VARA_REAL(ncidQ,varidQ,start,count,qnat2)
1213#endif
1214         IF (invert_y) THEN
1215           CALL invert_lat(iip1,jjp1,nlevnc,qnat2)
1216         ENDIF
1217         
1218     endif
1219
1220!  Vent meridien
1221     if (guide_v) then
1222         count(2)=jjm
1223#ifdef NC_DOUBLE
1224         status=NF_GET_VARA_DOUBLE(ncidv,varidv,start,count,vnat2)
1225#else
1226         status=NF_GET_VARA_REAL(ncidv,varidv,start,count,vnat2)
1227#endif
1228         IF (invert_y) THEN
1229           CALL invert_lat(iip1,jjm,nlevnc,vnat2)
1230         ENDIF
1231     endif
1232
1233!  Pression de surface
1234     if ((guide_P).OR.(guide_modele))  then
1235         start(3)=timestep
1236         start(4)=0
1237         count(2)=jjp1
1238         count(3)=1
1239         count(4)=0
1240#ifdef NC_DOUBLE
1241         status=NF_GET_VARA_DOUBLE(ncidps,varidps,start,count,psnat2)
1242#else
1243         status=NF_GET_VARA_REAL(ncidps,varidps,start,count,psnat2)
1244#endif
1245         IF (invert_y) THEN
1246           CALL invert_lat(iip1,jjp1,1,psnat2)
1247         ENDIF
1248     endif
1249
1250  END SUBROUTINE guide_read
1251
1252!=======================================================================
1253  SUBROUTINE guide_read2D(timestep)
1254
1255    IMPLICIT NONE
1256
1257#include "netcdf.inc"
1258#include "dimensions.h"
1259#include "paramet.h"
1260
1261    INTEGER, INTENT(IN)   :: timestep
1262
1263    LOGICAL, SAVE         :: first=.TRUE.
1264! Identification fichiers et variables NetCDF:
1265    INTEGER, SAVE         :: ncidu,varidu,ncidv,varidv,ncidQ
1266    INTEGER, SAVE         :: varidQ,ncidt,varidt,ncidps,varidps
1267    INTEGER               :: ncidpl,varidpl,varidap,varidbp
1268! Variables auxiliaires NetCDF:
1269    INTEGER, DIMENSION(4) :: start,count
1270    INTEGER               :: status,rcode
1271! Variables for 3D extension:
1272    REAL, DIMENSION (jjp1,llm) :: zu
1273    REAL, DIMENSION (jjm,llm)  :: zv
1274    INTEGER               :: i
1275
1276    CHARACTER (len = 80)   :: abort_message
1277    CHARACTER (len = 20)   :: modname = 'guide_read2D'
1278! -----------------------------------------------------------------
1279! Premier appel: initialisation de la lecture des fichiers
1280! -----------------------------------------------------------------
1281    if (first) then
1282         ncidpl=-99
1283         print*,'Guide: ouverture des fichiers guidage '
1284! Niveaux de pression si non constants
1285         if (guide_modele) then
1286             print *,'Lecture du guidage sur niveaux modele'
1287             rcode = nf90_open('apbp.nc', nf90_nowrite, ncidpl)
1288             IF (rcode.NE.NF_NOERR) THEN
1289              print *,'Guide: probleme -> pas de fichier apbp.nc'
1290              CALL abort_gcm(modname,abort_message,1)
1291             ENDIF
1292             rcode = nf90_inq_varid(ncidpl, 'AP', varidap)
1293             IF (rcode.NE.NF_NOERR) THEN
1294              print *,'Guide: probleme -> pas de variable AP, fichier apbp.nc'
1295              CALL abort_gcm(modname,abort_message,1)
1296             ENDIF
1297             rcode = nf90_inq_varid(ncidpl, 'BP', varidbp)
1298             IF (rcode.NE.NF_NOERR) THEN
1299              print *,'Guide: probleme -> pas de variable BP, fichier apbp.nc'
1300              CALL abort_gcm(modname,abort_message,1)
1301             ENDIF
1302             print*,'ncidpl,varidap',ncidpl,varidap
1303         endif
1304! Vent zonal
1305         if (guide_u) then
1306             rcode = nf90_open('u.nc', nf90_nowrite, ncidu)
1307             IF (rcode.NE.NF_NOERR) THEN
1308              print *,'Guide: probleme -> pas de fichier u.nc'
1309              CALL abort_gcm(modname,abort_message,1)
1310             ENDIF
1311             rcode = nf90_inq_varid(ncidu, 'UWND', varidu)
1312             IF (rcode.NE.NF_NOERR) THEN
1313              print *,'Guide: probleme -> pas de variable UWND, fichier u.nc'
1314              CALL abort_gcm(modname,abort_message,1)
1315             ENDIF
1316             print*,'ncidu,varidu',ncidu,varidu
1317             if (ncidpl.eq.-99) ncidpl=ncidu
1318         endif
1319! Vent meridien
1320         if (guide_v) then
1321             rcode = nf90_open('v.nc', nf90_nowrite, ncidv)
1322             IF (rcode.NE.NF_NOERR) THEN
1323              print *,'Guide: probleme -> pas de fichier v.nc'
1324              CALL abort_gcm(modname,abort_message,1)
1325             ENDIF
1326             rcode = nf90_inq_varid(ncidv, 'VWND', varidv)
1327             IF (rcode.NE.NF_NOERR) THEN
1328              print *,'Guide: probleme -> pas de variable VWND, fichier v.nc'
1329              CALL abort_gcm(modname,abort_message,1)
1330             ENDIF
1331             print*,'ncidv,varidv',ncidv,varidv
1332             if (ncidpl.eq.-99) ncidpl=ncidv
1333         endif
1334! Temperature
1335         if (guide_T) then
1336             rcode = nf90_open('T.nc', nf90_nowrite, ncidt)
1337             IF (rcode.NE.NF_NOERR) THEN
1338              print *,'Guide: probleme -> pas de fichier T.nc'
1339              CALL abort_gcm(modname,abort_message,1)
1340             ENDIF
1341             rcode = nf90_inq_varid(ncidt, 'AIR', varidt)
1342             IF (rcode.NE.NF_NOERR) THEN
1343              print *,'Guide: probleme -> pas de variable AIR, fichier T.nc'
1344              CALL abort_gcm(modname,abort_message,1)
1345             ENDIF
1346             print*,'ncidT,varidT',ncidt,varidt
1347             if (ncidpl.eq.-99) ncidpl=ncidt
1348         endif
1349! Humidite
1350         if (guide_Q) then
1351             rcode = nf90_open('hur.nc', nf90_nowrite, ncidQ)
1352             IF (rcode.NE.NF_NOERR) THEN
1353              print *,'Guide: probleme -> pas de fichier hur.nc'
1354              CALL abort_gcm(modname,abort_message,1)
1355             ENDIF
1356             rcode = nf90_inq_varid(ncidQ, 'RH', varidQ)
1357             IF (rcode.NE.NF_NOERR) THEN
1358              print *,'Guide: probleme -> pas de variable RH, fichier hur.nc'
1359              CALL abort_gcm(modname,abort_message,1)
1360             ENDIF
1361             print*,'ncidQ,varidQ',ncidQ,varidQ
1362             if (ncidpl.eq.-99) ncidpl=ncidQ
1363         endif
1364! Pression de surface
1365         if ((guide_P).OR.(guide_modele)) then
1366             rcode = nf90_open('ps.nc', nf90_nowrite, ncidps)
1367             IF (rcode.NE.NF_NOERR) THEN
1368              print *,'Guide: probleme -> pas de fichier ps.nc'
1369              CALL abort_gcm(modname,abort_message,1)
1370             ENDIF
1371             rcode = nf90_inq_varid(ncidps, 'SP', varidps)
1372             IF (rcode.NE.NF_NOERR) THEN
1373              print *,'Guide: probleme -> pas de variable SP, fichier ps.nc'
1374              CALL abort_gcm(modname,abort_message,1)
1375             ENDIF
1376             print*,'ncidps,varidps',ncidps,varidps
1377         endif
1378! Coordonnee verticale
1379         if (.not.guide_modele) then
1380              rcode = nf90_inq_varid(ncidpl, 'LEVEL', varidpl)
1381              IF (rcode.NE.0) rcode = nf90_inq_varid(ncidpl, 'PRESSURE', varidpl)
1382              print*,'ncidpl,varidpl',ncidpl,varidpl
1383         endif
1384! Coefs ap, bp pour calcul de la pression aux differents niveaux
1385         if (guide_modele) then
1386#ifdef NC_DOUBLE
1387             status=NF_GET_VARA_DOUBLE(ncidpl,varidap,1,nlevnc,apnc)
1388             status=NF_GET_VARA_DOUBLE(ncidpl,varidbp,1,nlevnc,bpnc)
1389#else
1390             status=NF_GET_VARA_REAL(ncidpl,varidap,1,nlevnc,apnc)
1391             status=NF_GET_VARA_REAL(ncidpl,varidbp,1,nlevnc,bpnc)
1392#endif
1393         else
1394#ifdef NC_DOUBLE
1395             status=NF_GET_VARA_DOUBLE(ncidpl,varidpl,1,nlevnc,apnc)
1396#else
1397             status=NF_GET_VARA_REAL(ncidpl,varidpl,1,nlevnc,apnc)
1398#endif
1399             apnc=apnc*100.! conversion en Pascals
1400             bpnc(:)=0.
1401         endif
1402         first=.FALSE.
1403     endif ! (first)
1404
1405! -----------------------------------------------------------------
1406!   lecture des champs u, v, T, Q, ps
1407! -----------------------------------------------------------------
1408
1409!  dimensions pour les champs scalaires et le vent zonal
1410     start(1)=1
1411     start(2)=1
1412     start(3)=1
1413     start(4)=timestep
1414
1415     count(1)=1
1416     count(2)=jjp1
1417     count(3)=nlevnc
1418     count(4)=1
1419
1420!  Vent zonal
1421     if (guide_u) then
1422#ifdef NC_DOUBLE
1423         status=NF_GET_VARA_DOUBLE(ncidu,varidu,start,count,zu)
1424#else
1425         status=NF_GET_VARA_REAL(ncidu,varidu,start,count,zu)
1426#endif
1427         DO i=1,iip1
1428             unat2(i,:,:)=zu(:,:)
1429         ENDDO
1430
1431         IF (invert_y) THEN
1432           CALL invert_lat(iip1,jjp1,nlevnc,unat2)
1433         ENDIF
1434
1435     endif
1436
1437!  Temperature
1438     if (guide_T) then
1439#ifdef NC_DOUBLE
1440         status=NF_GET_VARA_DOUBLE(ncidt,varidt,start,count,zu)
1441#else
1442         status=NF_GET_VARA_REAL(ncidt,varidt,start,count,zu)
1443#endif
1444         DO i=1,iip1
1445             tnat2(i,:,:)=zu(:,:)
1446         ENDDO
1447
1448         IF (invert_y) THEN
1449           CALL invert_lat(iip1,jjp1,nlevnc,tnat2)
1450         ENDIF
1451
1452     endif
1453
1454!  Humidite
1455     if (guide_Q) then
1456#ifdef NC_DOUBLE
1457         status=NF_GET_VARA_DOUBLE(ncidQ,varidQ,start,count,zu)
1458#else
1459         status=NF_GET_VARA_REAL(ncidQ,varidQ,start,count,zu)
1460#endif
1461         DO i=1,iip1
1462             qnat2(i,:,:)=zu(:,:)
1463         ENDDO
1464
1465         IF (invert_y) THEN
1466           CALL invert_lat(iip1,jjp1,nlevnc,qnat2)
1467         ENDIF
1468
1469     endif
1470
1471!  Vent meridien
1472     if (guide_v) then
1473         count(2)=jjm
1474#ifdef NC_DOUBLE
1475         status=NF_GET_VARA_DOUBLE(ncidv,varidv,start,count,zv)
1476#else
1477         status=NF_GET_VARA_REAL(ncidv,varidv,start,count,zv)
1478#endif
1479         DO i=1,iip1
1480             vnat2(i,:,:)=zv(:,:)
1481         ENDDO
1482
1483         IF (invert_y) THEN
1484           CALL invert_lat(iip1,jjm,nlevnc,vnat2)
1485         ENDIF
1486
1487     endif
1488
1489!  Pression de surface
1490     if ((guide_P).OR.(guide_modele))  then
1491         start(3)=timestep
1492         start(4)=0
1493         count(2)=jjp1
1494         count(3)=1
1495         count(4)=0
1496#ifdef NC_DOUBLE
1497         status=NF_GET_VARA_DOUBLE(ncidps,varidps,start,count,zu(:,1))
1498#else
1499         status=NF_GET_VARA_REAL(ncidps,varidps,start,count,zu(:,1))
1500#endif
1501         DO i=1,iip1
1502             psnat2(i,:)=zu(:,1)
1503         ENDDO
1504
1505         IF (invert_y) THEN
1506           CALL invert_lat(iip1,jjp1,1,psnat2)
1507         ENDIF
1508
1509     endif
1510
1511  END SUBROUTINE guide_read2D
1512 
1513!=======================================================================
1514  SUBROUTINE guide_out(varname,hsize,vsize,field)
1515
1516    IMPLICIT NONE
1517
1518    INCLUDE "dimensions.h"
1519    INCLUDE "paramet.h"
1520    INCLUDE "netcdf.inc"
1521    INCLUDE "comgeom2.h"
1522    INCLUDE "comconst.h"
1523    INCLUDE "comvert.h"
1524   
1525    ! Variables entree
1526    CHARACTER*(*), INTENT(IN)                          :: varname
1527    INTEGER,   INTENT (IN)                         :: hsize,vsize
1528    REAL, DIMENSION (iip1,hsize,vsize), INTENT(IN) :: field
1529
1530    ! Variables locales
1531    INTEGER, SAVE :: timestep=0
1532    ! Identites fichier netcdf
1533    INTEGER       :: nid, id_lonu, id_lonv, id_latu, id_latv, id_tim, id_lev
1534    INTEGER       :: vid_lonu,vid_lonv,vid_latu,vid_latv,vid_cu,vid_cv,vid_lev
1535    INTEGER       :: vid_au,vid_av
1536    INTEGER, DIMENSION (3) :: dim3
1537    INTEGER, DIMENSION (4) :: dim4,count,start
1538    INTEGER                :: ierr, varid,l
1539    REAL, DIMENSION (iip1,hsize,vsize) :: field2
1540
1541    print *,'Guide: output timestep',timestep,'var ',varname
1542    IF (timestep.EQ.0) THEN
1543! ----------------------------------------------
1544! initialisation fichier de sortie
1545! ----------------------------------------------
1546! Ouverture du fichier
1547        ierr=NF_CREATE("guide_ins.nc",NF_CLOBBER,nid)
1548! Definition des dimensions
1549        ierr=NF_DEF_DIM(nid,"LONU",iip1,id_lonu)
1550        ierr=NF_DEF_DIM(nid,"LONV",iip1,id_lonv)
1551        ierr=NF_DEF_DIM(nid,"LATU",jjp1,id_latu)
1552        ierr=NF_DEF_DIM(nid,"LATV",jjm,id_latv)
1553        ierr=NF_DEF_DIM(nid,"LEVEL",llm,id_lev)
1554        ierr=NF_DEF_DIM(nid,"TIME",NF_UNLIMITED,id_tim)
1555
1556! Creation des variables dimensions
1557        ierr=NF_DEF_VAR(nid,"LONU",NF_FLOAT,1,id_lonu,vid_lonu)
1558        ierr=NF_DEF_VAR(nid,"LONV",NF_FLOAT,1,id_lonv,vid_lonv)
1559        ierr=NF_DEF_VAR(nid,"LATU",NF_FLOAT,1,id_latu,vid_latu)
1560        ierr=NF_DEF_VAR(nid,"LATV",NF_FLOAT,1,id_latv,vid_latv)
1561        ierr=NF_DEF_VAR(nid,"LEVEL",NF_FLOAT,1,id_lev,vid_lev)
1562        ierr=NF_DEF_VAR(nid,"cu",NF_FLOAT,2,(/id_lonu,id_latu/),vid_cu)
1563        ierr=NF_DEF_VAR(nid,"au",NF_FLOAT,2,(/id_lonu,id_latu/),vid_au)
1564        ierr=NF_DEF_VAR(nid,"cv",NF_FLOAT,2,(/id_lonv,id_latv/),vid_cv)
1565        ierr=NF_DEF_VAR(nid,"av",NF_FLOAT,2,(/id_lonv,id_latv/),vid_av)
1566       
1567        ierr=NF_ENDDEF(nid)
1568
1569! Enregistrement des variables dimensions
1570#ifdef NC_DOUBLE
1571        ierr = NF_PUT_VAR_DOUBLE(nid,vid_lonu,rlonu*180./pi)
1572        ierr = NF_PUT_VAR_DOUBLE(nid,vid_lonv,rlonv*180./pi)
1573        ierr = NF_PUT_VAR_DOUBLE(nid,vid_latu,rlatu*180./pi)
1574        ierr = NF_PUT_VAR_DOUBLE(nid,vid_latv,rlatv*180./pi)
1575        ierr = NF_PUT_VAR_DOUBLE(nid,vid_lev,presnivs)
1576        ierr = NF_PUT_VAR_DOUBLE(nid,vid_cu,cu)
1577        ierr = NF_PUT_VAR_DOUBLE(nid,vid_cv,cv)
1578        ierr = NF_PUT_VAR_DOUBLE(nid,vid_au,alpha_u)
1579        ierr = NF_PUT_VAR_DOUBLE(nid,vid_av,alpha_v)
1580#else
1581        ierr = NF_PUT_VAR_REAL(nid,vid_lonu,rlonu*180./pi)
1582        ierr = NF_PUT_VAR_REAL(nid,vid_lonv,rlonv*180./pi)
1583        ierr = NF_PUT_VAR_REAL(nid,vid_latu,rlatu*180./pi)
1584        ierr = NF_PUT_VAR_REAL(nid,vid_latv,rlatv*180./pi)
1585        ierr = NF_PUT_VAR_REAL(nid,vid_lev,presnivs)
1586        ierr = NF_PUT_VAR_REAL(nid,vid_cu,cu)
1587        ierr = NF_PUT_VAR_REAL(nid,vid_cv,cv)
1588        ierr = NF_PUT_VAR_REAL(nid,vid_au,alpha_u)
1589        ierr = NF_PUT_VAR_REAL(nid,vid_av,alpha_v)
1590#endif
1591! --------------------------------------------------------------------
1592! Cr�ation des variables sauvegard�es
1593! --------------------------------------------------------------------
1594        ierr = NF_REDEF(nid)
1595! Surface pressure (GCM)
1596        dim3=(/id_lonv,id_latu,id_tim/)
1597        ierr = NF_DEF_VAR(nid,"SP",NF_FLOAT,3,dim3,varid)
1598! Surface pressure (guidage)
1599        IF (guide_P) THEN
1600            dim3=(/id_lonv,id_latu,id_tim/)
1601            ierr = NF_DEF_VAR(nid,"ps",NF_FLOAT,3,dim3,varid)
1602        ENDIF
1603! Zonal wind
1604        IF (guide_u) THEN
1605            dim4=(/id_lonu,id_latu,id_lev,id_tim/)
1606            ierr = NF_DEF_VAR(nid,"u",NF_FLOAT,4,dim4,varid)
1607            ierr = NF_DEF_VAR(nid,"ua",NF_FLOAT,4,dim4,varid)
1608            ierr = NF_DEF_VAR(nid,"ucov",NF_FLOAT,4,dim4,varid)
1609        ENDIF
1610! Merid. wind
1611        IF (guide_v) THEN
1612            dim4=(/id_lonv,id_latv,id_lev,id_tim/)
1613            ierr = NF_DEF_VAR(nid,"v",NF_FLOAT,4,dim4,varid)
1614            ierr = NF_DEF_VAR(nid,"va",NF_FLOAT,4,dim4,varid)
1615            ierr = NF_DEF_VAR(nid,"vcov",NF_FLOAT,4,dim4,varid)
1616        ENDIF
1617! Pot. Temperature
1618        IF (guide_T) THEN
1619            dim4=(/id_lonv,id_latu,id_lev,id_tim/)
1620            ierr = NF_DEF_VAR(nid,"teta",NF_FLOAT,4,dim4,varid)
1621        ENDIF
1622! Specific Humidity
1623        IF (guide_Q) THEN
1624            dim4=(/id_lonv,id_latu,id_lev,id_tim/)
1625            ierr = NF_DEF_VAR(nid,"q",NF_FLOAT,4,dim4,varid)
1626        ENDIF
1627       
1628        ierr = NF_ENDDEF(nid)
1629        ierr = NF_CLOSE(nid)
1630    ENDIF ! timestep=0
1631
1632! --------------------------------------------------------------------
1633! Enregistrement du champ
1634! --------------------------------------------------------------------
1635    ierr=NF_OPEN("guide_ins.nc",NF_WRITE,nid)
1636
1637    IF (varname=="SP") timestep=timestep+1
1638
1639    ierr = NF_INQ_VARID(nid,varname,varid)
1640    SELECT CASE (varname)
1641    CASE ("SP","ps")
1642        start=(/1,1,timestep,0/)
1643        count=(/iip1,jjp1,1,0/)
1644    CASE ("v","va","vcov")
1645        start=(/1,1,1,timestep/)
1646        count=(/iip1,jjm,llm,1/)
1647    CASE DEFAULT
1648        start=(/1,1,1,timestep/)
1649        count=(/iip1,jjp1,llm,1/)
1650    END SELECT
1651
1652    SELECT CASE (varname)
1653    CASE("u","ua")
1654        DO l=1,llm ; field2(:,2:jjm,l)=field(:,2:jjm,l)/cu(:,2:jjm) ; ENDDO
1655        field2(:,1,:)=0. ; field2(:,jjp1,:)=0.
1656    CASE("v","va")
1657        DO l=1,llm ; field2(:,:,l)=field(:,:,l)/cv(:,:) ; ENDDO
1658    CASE DEFAULT
1659        field2=field
1660    END SELECT
1661
1662
1663#ifdef NC_DOUBLE
1664    ierr = NF_PUT_VARA_DOUBLE(nid,varid,start,count,field2)
1665#else
1666    ierr = NF_PUT_VARA_REAL(nid,varid,start,count,field2)
1667#endif
1668
1669    ierr = NF_CLOSE(nid)
1670
1671  END SUBROUTINE guide_out
1672   
1673 
1674!===========================================================================
1675  subroutine correctbid(iim,nl,x)
1676    integer iim,nl
1677    real x(iim+1,nl)
1678    integer i,l
1679    real zz
1680
1681    do l=1,nl
1682        do i=2,iim-1
1683            if(abs(x(i,l)).gt.1.e10) then
1684               zz=0.5*(x(i-1,l)+x(i+1,l))
1685              print*,'correction ',i,l,x(i,l),zz
1686               x(i,l)=zz
1687            endif
1688         enddo
1689     enddo
1690     return
1691  end subroutine correctbid
1692
1693!===========================================================================
1694END MODULE guide_mod
Note: See TracBrowser for help on using the repository browser.