source: LMDZ5/trunk/libf/dyn3d_common/fyhyp_m.F90 @ 2343

Last change on this file since 2343 was 2228, checked in by lguez, 9 years ago

Correcting a problem from revision 2218. The type double precision
with option "-fdefault-real-8" of gfortran is promoted to 16-byte
precision and there is no specific procedure in arth with this
precision. Could not add a specific procedure in arth with double
precision because, with ifort, the option "-real-size 64" does not
promote the double precision, so that would make two identical
specific procedures in arth.

In module nrtype, replaced double precision by a parameterized real
kind so that the effective precision does not depend on a compiler
option.

In coefpoly, fxhyp, fyhyp and invert_zoom_x, use the parameterized
real kind defined in nrtype, instead of double precision.

Also, in module nrtype, removed unused derived types sprs2_sp and
sprs2_dp.

  • 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:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 8.7 KB
RevLine 
[2218]1module fyhyp_m
[524]2
[2218]3  IMPLICIT NONE
[524]4
[2218]5contains
[524]6
[2218]7  SUBROUTINE fyhyp(rlatu, yyprimu, rlatv, rlatu2, yprimu2, rlatu1, yprimu1)
[524]8
[2218]9    ! From LMDZ4/libf/dyn3d/fyhyp.F, version 1.2, 2005/06/03 09:11:32
[524]10
[2218]11    ! Author: P. Le Van, from analysis by R. Sadourny
[524]12
[2218]13    ! Calcule les latitudes et dérivées dans la grille du GCM pour une
14    ! fonction f(y) à dérivée tangente hyperbolique.
[524]15
[2218]16    ! Il vaut mieux avoir : grossismy * dzoom < pi / 2
[524]17
[2218]18    use coefpoly_m, only: coefpoly
[2228]19    use nrtype, only: k8
[524]20
[2218]21    include "dimensions.h"
22    ! for jjm
[524]23
[2218]24    include "serre.h"
25    ! for clat, grossismy, dzoomy, tauy
[524]26
[2218]27    REAL, intent(out):: rlatu(jjm + 1), yyprimu(jjm + 1)
28    REAL, intent(out):: rlatv(jjm)
29    real, intent(out):: rlatu2(jjm), yprimu2(jjm), rlatu1(jjm), yprimu1(jjm)
[524]30
[2218]31    ! Local:
[524]32
[2228]33    REAL(K8) champmin, champmax
[2218]34    INTEGER, PARAMETER:: nmax=30000, nmax2=2*nmax
35    REAL dzoom ! distance totale de la zone du zoom (en radians)
[2228]36    REAL(K8) ylat(jjm + 1), yprim(jjm + 1)
37    REAL(K8) yuv
38    REAL(K8), save:: yt(0:nmax2)
39    REAL(K8) fhyp(0:nmax2), beta
40    REAL(K8), save:: ytprim(0:nmax2)
41    REAL(K8) fxm(0:nmax2)
42    REAL(K8), save:: yf(0:nmax2)
43    REAL(K8) yypr(0:nmax2)
44    REAL(K8) yvrai(jjm + 1), yprimm(jjm + 1), ylatt(jjm + 1)
45    REAL(K8) pi, pis2, epsilon, y0, pisjm
46    REAL(K8) yo1, yi, ylon2, ymoy, yprimin
47    REAL(K8) yfi, yf1, ffdy
48    REAL(K8) ypn, deply, y00
[2218]49    SAVE y00, deply
[524]50
[2218]51    INTEGER i, j, it, ik, iter, jlat
52    INTEGER jpn, jjpn
53    SAVE jpn
[2228]54    REAL(K8) a0, a1, a2, a3, yi2, heavyy0, heavyy0m
55    REAL(K8) fa(0:nmax2), fb(0:nmax2)
[2218]56    REAL y0min, y0max
[524]57
[2228]58    REAL(K8) heavyside
[524]59
[2218]60    !-------------------------------------------------------------------
[524]61
[2218]62    print *, "Call sequence information: fyhyp"
[524]63
[2218]64    pi = 2.*asin(1.)
65    pis2 = pi/2.
66    pisjm = pi/real(jjm)
67    epsilon = 1e-3
68    y0 = clat*pi/180.
69    dzoom = dzoomy*pi
70    print *, 'yzoom(rad), grossismy, tauy, dzoom (rad):'
71    print *, y0, grossismy, tauy, dzoom
[524]72
[2218]73    DO i = 0, nmax2
74       yt(i) = -pis2 + real(i)*pi/nmax2
75    END DO
[524]76
[2218]77    heavyy0m = heavyside(-y0)
78    heavyy0 = heavyside(y0)
79    y0min = 2.*y0*heavyy0m - pis2
80    y0max = 2.*y0*heavyy0 + pis2
[524]81
[2218]82    fa = 999.999
83    fb = 999.999
[524]84
[2218]85    DO i = 0, nmax2
86       IF (yt(i)<y0) THEN
87          fa(i) = tauy*(yt(i)-y0 + dzoom/2.)
88          fb(i) = (yt(i)-2.*y0*heavyy0m + pis2)*(y0-yt(i))
89       ELSE IF (yt(i)>y0) THEN
90          fa(i) = tauy*(y0-yt(i) + dzoom/2.)
91          fb(i) = (2.*y0*heavyy0-yt(i) + pis2)*(yt(i)-y0)
92       END IF
[524]93
[2218]94       IF (200.*fb(i)<-fa(i)) THEN
95          fhyp(i) = -1.
96       ELSE IF (200.*fb(i)<fa(i)) THEN
97          fhyp(i) = 1.
98       ELSE
99          fhyp(i) = tanh(fa(i)/fb(i))
100       END IF
[524]101
[2218]102       IF (yt(i)==y0) fhyp(i) = 1.
103       IF (yt(i)==y0min .OR. yt(i)==y0max) fhyp(i) = -1.
104    END DO
[524]105
[2218]106    ! Calcul de beta
[524]107
[2218]108    ffdy = 0.
[524]109
[2218]110    DO i = 1, nmax2
111       ymoy = 0.5*(yt(i-1) + yt(i))
112       IF (ymoy<y0) THEN
113          fa(i) = tauy*(ymoy-y0 + dzoom/2.)
114          fb(i) = (ymoy-2.*y0*heavyy0m + pis2)*(y0-ymoy)
115       ELSE IF (ymoy>y0) THEN
116          fa(i) = tauy*(y0-ymoy + dzoom/2.)
117          fb(i) = (2.*y0*heavyy0-ymoy + pis2)*(ymoy-y0)
118       END IF
[524]119
[2218]120       IF (200.*fb(i)<-fa(i)) THEN
121          fxm(i) = -1.
122       ELSE IF (200.*fb(i)<fa(i)) THEN
123          fxm(i) = 1.
124       ELSE
125          fxm(i) = tanh(fa(i)/fb(i))
126       END IF
127       IF (ymoy==y0) fxm(i) = 1.
128       IF (ymoy==y0min .OR. yt(i)==y0max) fxm(i) = -1.
129       ffdy = ffdy + fxm(i)*(yt(i)-yt(i-1))
130    END DO
[524]131
[2218]132    beta = (grossismy*ffdy-pi)/(ffdy-pi)
[524]133
[2218]134    IF (2. * beta - grossismy <= 0.) THEN
135       print *, 'Attention ! La valeur beta calculee dans la routine fyhyp ' &
136            // 'est mauvaise. Modifier les valeurs de grossismy, tauy ou ' &
137            // 'dzoomy et relancer.'
138       STOP 1
139    END IF
[524]140
[2218]141    ! calcul de Ytprim
[524]142
[2218]143    DO i = 0, nmax2
144       ytprim(i) = beta + (grossismy-beta)*fhyp(i)
145    END DO
[524]146
[2218]147    ! Calcul de Yf
[524]148
[2218]149    yf(0) = -pis2
150    DO i = 1, nmax2
151       yypr(i) = beta + (grossismy-beta)*fxm(i)
152    END DO
[524]153
[2218]154    DO i = 1, nmax2
155       yf(i) = yf(i-1) + yypr(i)*(yt(i)-yt(i-1))
156    END DO
[524]157
[2218]158    ! yuv = 0. si calcul des latitudes aux pts. U
159    ! yuv = 0.5 si calcul des latitudes aux pts. V
[524]160
[2218]161    loop_ik: DO ik = 1, 4
162       IF (ik==1) THEN
163          yuv = 0.
164          jlat = jjm + 1
165       ELSE IF (ik==2) THEN
166          yuv = 0.5
167          jlat = jjm
168       ELSE IF (ik==3) THEN
169          yuv = 0.25
170          jlat = jjm
171       ELSE IF (ik==4) THEN
172          yuv = 0.75
173          jlat = jjm
174       END IF
[524]175
[2218]176       yo1 = 0.
177       DO j = 1, jlat
178          yo1 = 0.
179          ylon2 = -pis2 + pisjm*(real(j) + yuv-1.)
180          yfi = ylon2
[524]181
[2218]182          it = nmax2
183          DO while (it >= 1 .and. yfi < yf(it))
184             it = it - 1
185          END DO
[524]186
[2218]187          yi = yt(it)
188          IF (it==nmax2) THEN
189             it = nmax2 - 1
190             yf(it + 1) = pis2
191          END IF
[524]192
[2218]193          ! Interpolation entre yi(it) et yi(it + 1) pour avoir Y(yi)
194          ! et Y'(yi)
[524]195
[2218]196          CALL coefpoly(yf(it), yf(it + 1), ytprim(it), ytprim(it + 1), &
197               yt(it), yt(it + 1), a0, a1, a2, a3)
198
199          yf1 = yf(it)
200          yprimin = a1 + 2.*a2*yi + 3.*a3*yi*yi
201
202          iter = 1
203          DO
204             yi = yi - (yf1-yfi)/yprimin
205             IF (abs(yi-yo1)<=epsilon .or. iter == 300) exit
206             yo1 = yi
207             yi2 = yi*yi
208             yf1 = a0 + a1*yi + a2*yi2 + a3*yi2*yi
209             yprimin = a1 + 2.*a2*yi + 3.*a3*yi2
210          END DO
211          if (abs(yi-yo1) > epsilon) then
212             print *, 'Pas de solution.', j, ylon2
213             STOP 1
214          end if
215
216          yprimin = a1 + 2.*a2*yi + 3.*a3*yi*yi
217          yprim(j) = pi/(jjm*yprimin)
218          yvrai(j) = yi
219       END DO
220
221       DO j = 1, jlat - 1
222          IF (yvrai(j + 1)<yvrai(j)) THEN
223             print *, 'Problème avec rlat(', j + 1, ') plus petit que rlat(', &
224                  j, ')'
225             STOP 1
226          END IF
227       END DO
228
229       print *, 'Reorganisation des latitudes pour avoir entre - pi/2 et pi/2'
230
231       IF (ik==1) THEN
232          ypn = pis2
233          DO j = jjm + 1, 1, -1
234             IF (yvrai(j)<=ypn) exit
235          END DO
236
237          jpn = j
238          y00 = yvrai(jpn)
239          deply = pis2 - y00
240       END IF
241
242       DO j = 1, jjm + 1 - jpn
243          ylatt(j) = -pis2 - y00 + yvrai(jpn + j-1)
244          yprimm(j) = yprim(jpn + j-1)
245       END DO
246
247       jjpn = jpn
248       IF (jlat==jjm) jjpn = jpn - 1
249
250       DO j = 1, jjpn
251          ylatt(j + jjm + 1-jpn) = yvrai(j) + deply
252          yprimm(j + jjm + 1-jpn) = yprim(j)
253       END DO
254
255       ! Fin de la reorganisation
256
[524]257       DO j = 1, jlat
[2218]258          ylat(j) = ylatt(jlat + 1-j)
259          yprim(j) = yprimm(jlat + 1-j)
260       END DO
[524]261
[2218]262       DO j = 1, jlat
263          yvrai(j) = ylat(j)*180./pi
264       END DO
[524]265
[2218]266       IF (ik==1) THEN
267          DO j = 1, jjm + 1
268             rlatu(j) = ylat(j)
269             yyprimu(j) = yprim(j)
270          END DO
271       ELSE IF (ik==2) THEN
272          DO j = 1, jjm
273             rlatv(j) = ylat(j)
274          END DO
275       ELSE IF (ik==3) THEN
276          DO j = 1, jjm
277             rlatu2(j) = ylat(j)
278             yprimu2(j) = yprim(j)
279          END DO
280       ELSE IF (ik==4) THEN
281          DO j = 1, jjm
282             rlatu1(j) = ylat(j)
283             yprimu1(j) = yprim(j)
284          END DO
285       END IF
286    END DO loop_ik
[524]287
[2218]288    DO j = 1, jjm
289       ylat(j) = rlatu(j) - rlatu(j + 1)
290    END DO
291    champmin = 1e12
292    champmax = -1e12
293    DO j = 1, jjm
294       champmin = min(champmin, ylat(j))
295       champmax = max(champmax, ylat(j))
296    END DO
297    champmin = champmin*180./pi
298    champmax = champmax*180./pi
[524]299
[2218]300    DO j = 1, jjm
301       IF (rlatu1(j) <= rlatu2(j)) THEN
302          print *, 'Attention ! rlatu1 < rlatu2 ', rlatu1(j), rlatu2(j), j
303          STOP 13
304       ENDIF
[524]305
[2218]306       IF (rlatu2(j) <= rlatu(j+1)) THEN
307          print *, 'Attention ! rlatu2 < rlatup1 ', rlatu2(j), rlatu(j+1), j
308          STOP 14
309       ENDIF
[524]310
[2218]311       IF (rlatu(j) <= rlatu1(j)) THEN
312          print *, ' Attention ! rlatu < rlatu1 ', rlatu(j), rlatu1(j), j
313          STOP 15
314       ENDIF
[524]315
[2218]316       IF (rlatv(j) <= rlatu2(j)) THEN
317          print *, ' Attention ! rlatv < rlatu2 ', rlatv(j), rlatu2(j), j
318          STOP 16
319       ENDIF
[524]320
[2218]321       IF (rlatv(j) >= rlatu1(j)) THEN
322          print *, ' Attention ! rlatv > rlatu1 ', rlatv(j), rlatu1(j), j
323          STOP 17
324       ENDIF
[524]325
[2218]326       IF (rlatv(j) >= rlatu(j)) THEN
327          print *, ' Attention ! rlatv > rlatu ', rlatv(j), rlatu(j), j
328          STOP 18
329       ENDIF
330    ENDDO
[524]331
[2218]332    print *, 'Latitudes'
333    print 3, champmin, champmax
[524]334
[2218]3353   Format(1x, ' Au centre du zoom, la longueur de la maille est', &
336         ' d environ ', f0.2, ' degres ', /, &
337         ' alors que la maille en dehors de la zone du zoom est ', &
338         "d'environ ", f0.2, ' degres ')
[524]339
[2218]340  END SUBROUTINE fyhyp
[524]341
[2218]342end module fyhyp_m
Note: See TracBrowser for help on using the repository browser.