Changeset 1569 for trunk/LMDZ.MARS/libf


Ignore:
Timestamp:
Jun 29, 2016, 6:08:29 PM (9 years ago)
Author:
flefevre
Message:

Nouvelle version du solveur chimique ASIS:

  • subroutine separee de calcul du pas de temps variable
  • calcul du first-guess de la solution en semi-implicite
  • balayage de toutes les especes pour le calcul de la courbure de la solution
  • introduction des tolerances absolue et relative sur l'erreur de la solution (reglages provisoires)
File:
1 edited

Legend:

Unmodified
Added
Removed
  • trunk/LMDZ.MARS/libf/aeronomars/photochemistry_asis.F90

    r1503 r1569  
    8585integer,parameter :: i_n2   = 17
    8686
    87 real :: stimestep           ! standard timestep for the chemistry (s)
    88 real :: ctimestep           ! real timestep for the chemistry (s)
     87real :: ctimestep           ! standard timestep for the chemistry (s)
    8988real :: dt_guess            ! first-guess timestep (s)
    9089real :: dt_corrected        ! corrected timestep (s)
    91 real :: dt_min = 1.         ! minimum allowed timestep (s)
    92 real :: dtg                 ! correction factor for the timestep (s)
    9390real :: j(nlayer,nd)        ! interpolated photolysis rates (s-1)
    9491real :: time                ! internal time (between 0 and ptimestep, in s)
     
    115112
    116113real (kind = 8), dimension(nesp) :: prod, loss
    117 
    118 ! curvatures
    119 
    120 real :: ratio, curv, e, e1, e2, e3
    121114
    122115!===================================================================
     
    169162
    170163!===================================================================
    171 !     stimestep : standard chemical timestep (s)                           
    172 !     ctimestep : real chemical timestep (s),
    173 !                 taking into account the physical timestep                           
    174 !===================================================================
    175 
    176 stimestep = 600. ! standard value : 10 mn
    177 
    178 phychemrat = nint(ptimestep/stimestep)
     164!     ctimestep : standard chemical timestep (s), defined as
     165!                 the fraction phychemrat of the physical timestep                           
     166!===================================================================
     167
    179168phychemrat = 1
    180169
    181170ctimestep = ptimestep/real(phychemrat)
    182171
    183 !print*, "stimestep  = ", stimestep
    184172!print*, "ptimestep  = ", ptimestep
    185173!print*, "phychemrat = ", phychemrat
     
    210198   call fill_matrix(ilev, mat1, prod, loss, c, nesp, nlayer, v_phot, v_3, v_4)
    211199
    212 ! first guess : form the matrix identity + mat*dt_guess
    213 
    214    mat(:,:) = mat1(:,:)*dt_guess
     200!  adaptative evaluation of the sub time step
     201
     202   call define_dt(nesp, dt_corrected, dt_guess, ctimestep, cold(:), c(ilev,:),  &
     203                  mat1, prod, loss)
     204
     205   if (time + dt_corrected > ptimestep) then
     206      dt_corrected = ptimestep - time
     207   end if
     208
     209!  if (dt_corrected /= dt_guess) then  ! the timestep has been modified
     210
     211!  form the matrix identity + mat*dt_corrected
     212
     213   mat(:,:) = mat1(:,:)*dt_corrected
    215214   do iesp = 1,nesp
    216215      mat(iesp,iesp) = 1. + mat(iesp,iesp)
    217216   end do
    218217
    219 first-guess: solve linear system
     218solve the linear system  M*Cn+1 = Cn (RHS in cnew, then replaced by solution)
    220219     
    221220   cnew(:) = c(ilev,:)
     
    227226   stop
    228227#endif
     228
     229!  end if
     230
    229231!  eliminate small values
    230232
     
    233235   end where
    234236
    235 !  curvature of the solution and weighted error norm for O
    236 !
    237 !  cold : concentrations at time t - dt
    238 !  c    : concentrations at time t
    239 !  cnew : concentrations guessed at time t + dt
    240 
    241 !  ratio = dtold/dtnew
    242 !  here ratio = 1 since the timestep for the guess is equal to the previous timestep
    243 
    244    ratio = 1.
    245 
    246    curv = 2.*(ratio*cnew(i_o) - (1. + ratio)*c(ilev,i_o) + cold(i_o)) &
    247           /(ratio*(1. + ratio))
    248    e1 = (curv/(cnew(i_o) + cnew(i_o3)))*100.
    249    e1 = abs(e1)
    250 
    251 !  curvature of the solution and weighted error norm for HO2
    252 
    253    curv = 2.*(ratio*cnew(i_ho2) - (1. + ratio)*c(ilev,i_ho2) + cold(i_ho2)) &
    254           /(ratio*(1. + ratio))
    255    e2 = (curv/(cnew(i_h) + cnew(i_oh) + cnew(i_ho2) + 2.*cnew(i_h2o2)))*100.
    256    e2 = abs(e2)
    257 
    258 !  curvature of the solution and weighted error norm for NO2
    259 
    260 !  curv = 2.*(ratio*cnew(i_no2) - (1. + ratio)*c(ilev,i_no2) + cold(i_no2)) &
    261 !         /(ratio*(1. + ratio))
    262 !  e3 = (curv/(cnew(i_no) + cnew(i_no2)))*100.
    263 !  e3 = abs(e3)
    264    e3 = 0.
    265 
    266 !  timestep correction
    267 
    268    e = max(e1, e2)
    269    e = max(e,  e3)
    270    e = max(e, 0.1)
    271    dtg = max(0.001, min(2.5,0.8/sqrt(e)))
    272 !  dtg = 1.                                     ! uncomment this line to turn off the variable timestep scheme
    273    dt_corrected = max(dt_min,dtg*dt_guess)      ! minimal timestep: dt_min
    274    dt_corrected = min(dt_corrected,ctimestep)   ! maximal timestep: ctimestep
    275 
    276    if (time + dt_corrected > ptimestep) then
    277       dt_corrected = ptimestep - time
    278    end if
    279 
    280    if (dt_corrected /= dt_guess) then  ! the timestep has been modified
    281 
    282 !  form the matrix identity + mat*dt_corrected
    283 
    284       mat(:,:) = mat1(:,:)*dt_corrected
    285       do iesp = 1,nesp
    286          mat(iesp,iesp) = 1. + mat(iesp,iesp)
    287       end do
    288 
    289 !  solve linear system
    290      
    291       cnew(:) = c(ilev,:)
     237!  update concentrations
     238
     239   cold(:)   = c(ilev,:)
     240   c(ilev,:) = cnew(:)
     241   cnew(:)   = 0.
     242
     243!  increment internal time
     244
     245   time = time + dt_corrected
     246   dt_guess = dt_corrected     ! first-guess timestep for next iteration
     247
     248   end do ! while (time < ptimestep)
     249
     250end do ! ilev
     251
     252!===================================================================
     253!     save chemical species for the gcm       
     254!===================================================================
     255
     256call chimtogcm(nlayer, nq, zycol, lswitch, nesp,              &
     257               i_co2, i_co, i_o, i_o1d, i_o2, i_o3, i_h,      &
     258               i_h2, i_oh, i_ho2, i_h2o2, i_h2o,              &
     259               i_n, i_n2d, i_no, i_no2, i_n2, dens, c)
     260
     261contains
     262
     263!================================================================
     264
     265 subroutine define_dt(nesp, dtnew, dtold, ctimestep, cold, ccur, mat1, prod, loss)
     266
     267!================================================================
     268! iterative evaluation of the appropriate time step dtnew
     269! according to curvature criterion based on
     270! e = 2 Rtol [r Cn+1 -(1-r) Cn + Cn-1 ]/[(1+r) Cn]
     271! with r = (tn - tn-1)/(tn+1 - tn)
     272!================================================================
     273
     274implicit none
     275
     276! input
     277
     278integer :: nesp  ! number of species in the chemistry
     279
     280real :: dtold, ctimestep
     281real (kind = 8), dimension(nesp)      :: cold, ccur
     282real (kind = 8), dimension(nesp,nesp) :: mat1
     283real (kind = 8), dimension(nesp)      :: prod, loss
     284
     285! output
     286
     287real :: dtnew
     288
     289! local
     290
     291real (kind = 8), dimension(nesp)      :: cnew
     292real (kind = 8), dimension(nesp,nesp) :: mat
     293real (kind = 8) :: ratio, e, es, coef
     294
     295integer                  :: code, iesp, iter
     296integer, dimension(nesp) :: indx
     297
     298real :: dttest
     299
     300! parameters
     301
     302real (kind = 8), parameter :: dtmin   = 10.      ! minimum time step (s)
     303real (kind = 8), parameter :: atol    = 1.e4     ! recommended value : 1.e4
     304real (kind = 8), parameter :: rtol    = 1./0.1   ! 1/rtol recommended value : 0.1-0.02
     305integer,         parameter :: niter   = 3        ! number of iterations
     306real (kind = 8), parameter :: coefmax = 2.
     307real (kind = 8), parameter :: coefmin = 0.1
     308logical                    :: fast_guess = .true.
     309
     310dttest = dtold   ! dttest = dtold = dt_guess
     311
     312do iter = 1,niter
     313
     314if (fast_guess) then
     315
     316! first guess : fast implicit method
     317
     318   do iesp = 1, nesp
     319      cnew(iesp) = (ccur(iesp) + prod(iesp)*dttest)/(1. + loss(iesp)*dttest)
     320   end do
     321
     322else
     323
     324! first guess : form the matrix identity + mat*dt_guess
     325
     326   mat(:,:) = mat1(:,:)*dttest
     327   do iesp = 1,nesp
     328      mat(iesp,iesp) = 1. + mat(iesp,iesp)
     329   end do
     330
     331! form right-hand side (RHS) of the system
     332
     333   cnew(:) = ccur(:)
     334
     335! solve the linear system  M*Cn+1 = Cn (RHS in cnew, then replaced by solution)
    292336
    293337#ifdef LAPACK
     
    297341   stop
    298342#endif
     343
     344end if
     345
     346! ratio old/new subtimestep
     347
     348ratio = dtold/dttest
     349
     350! e : local error indicatocitr
     351
     352e = 0.
     353
     354do iesp = 1,nesp
     355   es = 2.*abs((ratio*cnew(iesp) - (1. + ratio)*ccur(iesp) + cold(iesp))   &
     356         /(1. + ratio)/max(ccur(iesp),atol))
     357
     358   if (es > e) then
     359      e = es
    299360   end if
    300 
    301 !  eliminate small values
    302 
    303    where (cnew(:)/dens(ilev) < 1.e-30)
    304       cnew(:) = 0.
    305    end where
    306 
    307    cold(:)   = c(ilev,:)
    308    c(ilev,:) = cnew(:)
    309    cnew(:)   = 0.
    310 
    311 !  increment internal time
    312 
    313    time = time + dt_corrected
    314    dt_guess = dt_corrected     ! first-guess timestep for next iteration
    315 
    316    end do ! while (time < ptimestep)
    317 
    318 end do ! ilev
    319 
    320 !===================================================================
    321 !     save chemical species for the gcm       
    322 !===================================================================
    323 
    324 call chimtogcm(nlayer, nq, zycol, lswitch, nesp,              &
    325                i_co2, i_co, i_o, i_o1d, i_o2, i_o3, i_h,      &
    326                i_h2, i_oh, i_ho2, i_h2o2, i_h2o,              &
    327                i_n, i_n2d, i_no, i_no2, i_n2, dens, c)
    328 
    329 contains
     361end do
     362e = rtol*e
     363
     364! timestep correction
     365
     366coef = max(coefmin, min(coefmax,0.8/sqrt(e)))
     367
     368dttest = max(dtmin,dttest*coef)
     369dttest = min(ctimestep,dttest)
     370
     371end do ! iter
     372
     373! new timestep
     374
     375dtnew = dttest
     376
     377end subroutine define_dt
    330378
    331379!======================================================================
     
    9731021integer :: iphot,i3,i4
    9741022
    975 real(kind = jprb) :: eps, eps_4  ! implicit/explicit coefficient
     1023real(kind = 8) :: eps, eps_4  ! implicit/explicit coefficient
    9761024
    9771025! initialisations
     
    10311079
    10321080  eps_4 = abs(c(ilev,ind_4_2))/(abs(c(ilev,ind_4_2)) + abs(c(ilev,ind_4_4)) + eps)
    1033   eps_4 = min(eps_4,1.0_jprb)
     1081  eps_4 = min(eps_4,1.0)
    10341082
    10351083  mat(ind_4_2,ind_4_2) = mat(ind_4_2,ind_4_2) + indice_4(i4)%z1*v_4(ilev,i4)*(1. - eps_4)*c(ilev,ind_4_4)
Note: See TracChangeset for help on using the changeset viewer.