Changeset 2132 for trunk/LMDZ.GENERIC


Ignore:
Timestamp:
Apr 29, 2019, 3:13:04 PM (6 years ago)
Author:
aboissinot
Message:

Alimentation (more precisely lalim) is removed from thermcell_flux as well as corrections which need it.
Consequently, iflag_thermals_optflux flag is removed from thermcell_mod.

Location:
trunk/LMDZ.GENERIC/libf/phystd
Files:
4 edited

Legend:

Unmodified
Added
Removed
  • trunk/LMDZ.GENERIC/libf/phystd/physiq_mod.F90

    r2129 r2132  
    255255      INTEGER lmin(ngrid)                 ! Plume base level
    256256      INTEGER lmix(ngrid)                 ! Plume maximal velocity level
    257       INTEGER lalim(ngrid)                ! Plume alimentation top level
    258257      INTEGER lmax(ngrid)                 ! Maximal level reached by the plume
    259258     
     
    11861185                             f0, fm0, entr0, detr0, zw2, fraca,                  &
    11871186                             zqta, zqla, ztv, ztva, ztla, zthl, zqsatth,         &
    1188                              lmin, lmix, lalim, lmax)
     1187                             lmin, lmix, lmax)
    11891188         
    11901189         DO ig=1,ngrid
  • trunk/LMDZ.GENERIC/libf/phystd/thermcell_flux.F90

    r2127 r2132  
    33!
    44SUBROUTINE thermcell_flux(ngrid,nlay,ptimestep,masse,                         &
    5                           lalim,lmin,lmax,entr_star,detr_star,                &
     5                          lmin,lmax,entr_star,detr_star,                      &
    66                          f,rhobarz,zlev,zw2,fm,entr,detr,zqla)
    77     
    88     
    99!===============================================================================
    10 !  Purpose: deduction des flux
     10!  Purpose: fluxes deduction
    1111
    1212!  Modif 2019/04 (AB alexandre.boissinot@lmd.jussieu.fr)
     
    1515     
    1616      USE print_control_mod, ONLY: prt_level
    17       USE thermcell_mod
     17      USE thermcell_mod, ONLY: fomass_max, alpha_max
    1818     
    1919      IMPLICIT NONE
     
    3030      INTEGER lmin(ngrid)
    3131      INTEGER lmax(ngrid)
    32       INTEGER lalim(ngrid)
    33      
    34 ! AB : I remove alimentation, which is in fact included in entr_star. EN COURS
    35 !      REAL alim_star(ngrid,nlay)
     32     
    3633      REAL entr_star(ngrid,nlay)
    3734      REAL detr_star(ngrid,nlay)
     
    5855      INTEGER igout, lout                 ! Error grid point number and level
    5956     
    60       REAL zfm                            ! Mass flux such as updraft fraction is equal to its maximal authorized value alphamax
     57      REAL zfm                            ! Mass flux such as updraft fraction is equal to its maximal authorized value alpha_max
    6158      REAL f_old                          ! Save initial value of mass flux
    6259      REAL eee0                           ! Save initial value of entrainment
     
    6562      REAL ddd                            ! ddd0 - eee
    6663      REAL zzz                            ! Temporary variable set to mass flux
    67       REAL fact
    68       REAL test
    69      
    70       INTEGER ncorecfm1
    71       INTEGER ncorecfm2
    72       INTEGER ncorecfm3
    73       INTEGER ncorecfm4
    74       INTEGER ncorecfm5
    75       INTEGER ncorecfm6
    76       INTEGER ncorecalpha
    77      
    78       LOGICAL check_debug
     64      REAL fact                           ! Factor between old norm and new norm
     65     
     66      INTEGER ncorecdetr                  ! detr > fm-   counter
     67      INTEGER ncorecentr                  ! entr > e_max counter
     68      INTEGER ncorecalpha                 ! fm > zfm     counter
     69     
    7970      LOGICAL labort_physic
    8071     
     
    8677!===============================================================================
    8778     
    88       ncorecfm1 = 0
    89       ncorecfm2 = 0
    90       ncorecfm3 = 0
    91       ncorecfm4 = 0
    92       ncorecfm5 = 0
    93       ncorecfm6 = 0
    94      
     79      ncorecdetr  = 0
     80      ncorecentr  = 0
    9581      ncorecalpha = 0
    9682     
     
    10086     
    10187!===============================================================================
    102 ! Calcul de l'entrainement, du detrainement et du flux de masse
     88! Mass flux, entrainment and detrainment
    10389!===============================================================================
    10490     
     
    10894     
    10995      DO l=1,nlay
    110 ! AB : I remove alimentation, which is in fact included in entr_star. EN COURS
    111 !         entr(:,l) = f(:) * (entr_star(:,l) + alim_star(:,l))
    11296         entr(:,l) = f(:) * entr_star(:,l)
    11397         detr(:,l) = f(:) * detr_star(:,l)
    11498      ENDDO
    11599     
    116 ! AB : temporary test
    117       DO l=1,nlay
    118          DO ig=1,ngrid
    119             IF (detr(ig,l)<0.) THEN
    120                print *, 'ERROR: detr is negative in thermcell_flux!'
    121                print *, 'ig,l', ig, l
    122                print *, 'detr_star', detr_star(ig,l)
    123                print *, 'detr', detr(ig,l)
    124             ENDIF
    125          ENDDO
    126       ENDDO
    127      
    128 !-------------------------------------------------------------------------------
    129 ! Calcul du flux de masse
     100!-------------------------------------------------------------------------------
     101! Mass flux and boundary conditions
    130102!-------------------------------------------------------------------------------
    131103     
     
    146118     
    147119!===============================================================================
    148 ! Tests de validite
     120! Validity tests and corrections
    149121!===============================================================================
    150122     
     
    152124         
    153125!-------------------------------------------------------------------------------
    154 ! Verification de la positivite du flux de masse entrant
     126! Is incoming mass flux positive ?
    155127!-------------------------------------------------------------------------------
    156128         
     
    189161         
    190162!-------------------------------------------------------------------------------
    191 ! Test entrainment positif
     163! Is entrainment positive ?
    192164!-------------------------------------------------------------------------------
    193165         
     
    216188               print *, 'entr,detr', entr(igout,k), detr(igout,k)
    217189            ENDDO
    218             print *, 'fm-', fm(igout,1)
     190            print *, 'fm ', fm(igout,1)
    219191            print *, '-------------------------------'
    220192            abort_message = 'Negative entrainment in thermcell_flux'
     
    223195         
    224196!-------------------------------------------------------------------------------
    225 ! Test detrainment positif
     197! Is detrainment positive ?
    226198!-------------------------------------------------------------------------------
    227199         
     
    243215               print *, 'entr,detr', entr(igout,k), detr(igout,k)
    244216            ENDDO
    245             print *, 'fm-', fm(igout,1)
     217            print *, 'fm ', fm(igout,1)
    246218            print *, '-------------------------------'
    247219            abort_message = 'Negative detrainment in thermcell_flux'
     
    250222         
    251223!-------------------------------------------------------------------------------
    252 ! Test sur fraca constante ou croissante au-dessus de lalim
    253 !-------------------------------------------------------------------------------
    254          
    255 ! AB : Do we have to decree that? If so, set iflag_thermals_optflux to 0
    256          IF (iflag_thermals_optflux==0) THEN
    257             DO ig=1,ngrid
    258                IF (l.ge.lalim(ig).and.l.le.lmax(ig).and.(zw2(ig,l+1).gt.1.e-10).and.(zw2(ig,l).gt.1.e-10) ) THEN
    259                   zzz = fm(ig,l) * rhobarz(ig,l+1) * zw2(ig,l+1)              &
    260                   &   / (rhobarz(ig,l) * zw2(ig,l))
    261                  
    262                   IF (fm(ig,l+1).gt.zzz) THEN
    263                      detr(ig,l) = detr(ig,l) + fm(ig,l+1) - zzz
    264                      fm(ig,l+1) = zzz
    265                      ncorecfm4  = ncorecfm4 + 1
    266                   ENDIF
    267                ENDIF
    268             ENDDO
    269          ENDIF
    270          
    271 !-------------------------------------------------------------------------------
    272 ! Test sur flux de masse constant ou decroissant au-dessus de lalim
    273 !-------------------------------------------------------------------------------
    274          
    275 ! AB : Do we have to decree that? If so, set iflag_thermals_optflux to 0
    276          IF (iflag_thermals_optflux==0) THEN
    277             DO ig=1,ngrid
    278                IF ((fm(ig,l+1).gt.fm(ig,l)).and.(l.gt.lalim(ig))) then
    279                   f_old = fm(ig,l+1)
    280                   fm(ig,l+1) = fm(ig,l)
    281                   detr(ig,l) = detr(ig,l) + f_old - fm(ig,l+1)
    282                   ncorecfm5  = ncorecfm5 + 1
    283                ENDIF
    284             ENDDO
    285          ENDIF
    286          
    287 !-------------------------------------------------------------------------------
    288 ! Test detrainment inferieur au flux de masse
     224! Is detrainment lesser than incoming mass flux ?
    289225!-------------------------------------------------------------------------------
    290226         
     
    309245               
    310246               IF (prt_level.ge.100) THEN
     247                  print *, 'fm+', fm(ig,l+1)
     248                  print *, 'entr,detr', entr(ig,l), detr(ig,l)
    311249                  print *, 'fm-', fm(ig,l)
    312                   print *, 'entr,detr', entr(ig,l), detr(ig,l)
    313                   print *, 'fm+', fm(ig,l+1)
    314                ENDIF
    315                
    316                ncorecfm6  = ncorecfm6 + 1
    317                
    318 !~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    319 ! Dans le cas ou on est au dessus de la couche d'alimentation et que le
    320 ! detrainement est plus fort que le flux de masse, on stope le thermique.
    321 ! test : on commente
    322 !
    323 ! AB : Do we have to stop the plume here?
    324 ! AB : Attention, if lmax is modified, be sure that fm, zw2, entr, detr are set
    325 !      to zero above this level!
    326 !~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    327 !               IF (l.gt.lalim(ig)) THEN
    328 !                  lmax(ig)=l
    329 !                  fm(ig,l+1)=0.
    330 !                  entr(ig,l)=0.
    331 !               ELSE
    332 !                  ncorecfm2=ncorecfm2+1
    333 !               ENDIF
    334             ENDIF
    335            
    336 !            IF (l.gt.lmax(ig)) THEN
    337 !               detr(ig,l) = 0.
    338 !               fm(ig,l+1) = 0.
    339 !               entr(ig,l) = 0.
    340 !            ENDIF
    341          ENDDO
    342          
    343 !         labort_physic=.false.
    344 !         
    345 !         DO ig=1,ngrid
    346 !            IF (entr(ig,l).lt.0.) THEN
    347 !               labort_physic=.true.
    348 !               igout=ig
    349 !            ENDIF
    350 !         ENDDO
    351 !         
    352 !         IF (labort_physic) THEN
    353 !            print *, 'ERROR: detrainment is greater than mass flux and'
    354 !            print *, '       entrainment cannot compensate it!'
    355 !            print *, 'ig,l,lmax(ig)', igout, l, lmax(igout)
    356 !            print *, '--------------------'
    357 !            print *, 'fm+', fm(igout,lout+1)
    358 !            print *, 'entr,detr', entr(igout,lout), detr(igout,lout)
    359 !            print *, 'fm-', fm(igout,lout)
    360 !            abort_message = 'problem in thermcell_flux'
    361 !            CALL abort_physic (modname,abort_message,1)
    362 !         ENDIF
    363          
    364 !-------------------------------------------------------------------------------
    365 ! Test fraction masse entrainee inferieure a fomass_max
     250               ENDIF
     251               
     252               ncorecdetr  = ncorecdetr + 1
     253               
     254            ENDIF
     255         ENDDO
     256         
     257!-------------------------------------------------------------------------------
     258! Is entrainment mass fraction lesser than fomass_max ?
    366259!-------------------------------------------------------------------------------
    367260         
     
    388281            IF (eee.gt.0.) THEN
    389282               entr(ig,l) = entr(ig,l) - eee
    390                ncorecfm3  = ncorecfm3 + 1
     283               ncorecentr  = ncorecentr + 1
    391284               
    392285               IF (prt_level.ge.10) THEN
     
    395288               ENDIF
    396289               
    397                IF (ddd.gt.0.) THEN                          ! on reduit le detr dans la couche
     290               IF (ddd.gt.0.) THEN                          ! detr in the current layer is reduced
    398291                  detr(ig,l) = ddd
    399                ELSEIF (l.eq.lmax(ig)) THEN                  ! on ajuste le detr dans la couche
     292               ELSEIF (l.eq.lmax(ig)) THEN                  ! detr in the last layer is adjusted
    400293                  detr(ig,l) = fm(ig,l) + entr(ig,l)
    401                ELSEIF (entr(ig,l+1).gt.ABS(ddd)) THEN       ! on reduit le detr dans la couche et l'entr au-dessus
     294               ELSEIF (entr(ig,l+1).gt.ABS(ddd)) THEN       ! detr in the current layer is set to 0 and entr in the layer above is reduced
    402295                  detr(ig,l) = 0.
    403296                  fm(ig,l+1) = fm(ig,l) + entr(ig,l)
     
    423316         ENDDO
    424317         
    425          IF (labort_physic) THEN
    426             print *, 'ERROR: Entrainment is greater than its maximal authorized value!'
    427             print *, '       Nor detrainment neither entrainment can compensate it!'
    428             print *, 'ig,l,entr', igout, lout, entr(igout,lout)
    429             print *, 'lmin,lmax', lmin(igout), lmax(igout)
    430             print *, '--------------------'
    431             print *, 'e_max :', masse(igout,lout)*fomass_max/ptimestep
    432             print *, '   fomass_max :', fomass_max
    433             print *, '   masse      :', masse(igout,lout)
    434             print *, '   ptimestep  :', ptimestep
    435             print *, 'norm  :', f(igout)
    436 !            print *, 'alim* :', alim_star(igout,lout)
    437             print *, 'entr* :', entr_star(igout,lout)
    438             print *, '--------------------'
    439             print *, 'fm l+1   ', fm(igout,lout+2)
    440             print *, 'entr <-- ', entr(igout,lout+1)
    441             print *, 'detr  -->', detr(igout,lout+1)
    442             print *, 'fm l     ', fm(igout,lout+1)
    443             print *, 'entr <-- ', entr(igout,lout)
    444             print *, 'detr  -->', detr(igout,lout)
    445             print *, 'fm l-1   ', fm(igout,lout)
    446             call flush
    447             abort_message = 'problem in thermcell_flux'
    448             CALL abort_physic (modname,abort_message,1)
    449          ENDIF
    450          
    451 !-------------------------------------------------------------------------------
    452 ! Test fraction couverte inferieure a alphamax
     318!         IF (labort_physic) THEN
     319!            print *, 'ERROR: Entrainment is greater than its maximal authorized value!'
     320!            print *, '       Nor detrainment neither entrainment can compensate it!'
     321!            print *, 'ig,l,entr', igout, lout, entr(igout,lout)
     322!            print *, 'lmin,lmax', lmin(igout), lmax(igout)
     323!            print *, '--------------------'
     324!            print *, 'e_max :', masse(igout,lout)*fomass_max/ptimestep
     325!            print *, '   fomass_max :', fomass_max
     326!            print *, '   masse      :', masse(igout,lout)
     327!            print *, '   ptimestep  :', ptimestep
     328!            print *, 'norm  :', f(igout)
     329!            print *, 'entr* :', entr_star(igout,lout)
     330!            print *, '--------------------'
     331!            DO k=nlay,1,-1
     332!               print *, 'fm ', fm(igout,k+1)
     333!               print *, 'entr,detr', entr(igout,k), detr(igout,k)
     334!            ENDDO
     335!            print *, 'fm ', fm(igout,1)
     336!            print *, '-------------------------------'
     337!            abort_message = 'Entrainment is too large in thermcell_flux'
     338!            CALL abort_physic (modname,abort_message,1)
     339!         ENDIF
     340         
     341!-------------------------------------------------------------------------------
     342! Is updraft fraction lesser than alpha_max ?
    453343!-------------------------------------------------------------------------------
    454344         
     
    466356         
    467357         DO ig=1,ngrid
    468             zfm = rhobarz(ig,l+1) * zw2(ig,l+1) * alphamax
     358            zfm = rhobarz(ig,l+1) * zw2(ig,l+1) * alpha_max
    469359           
    470360            IF (fm(ig,l+1) .gt. zfm) THEN
     
    495385               
    496386               ncorecalpha = ncorecalpha + 1
    497                
    498387            ENDIF
    499388         ENDDO
     
    502391     
    503392!-------------------------------------------------------------------------------
    504 ! We check if fm, entr and detr vanish correctly in layer lmax
     393! Boundary conditions
    505394!-------------------------------------------------------------------------------
    506395     
     
    518407     
    519408!-------------------------------------------------------------------------------
    520 ! Impression des bidouilles utilisees pour corriger les problemes
    521 !-------------------------------------------------------------------------------
    522      
    523       IF (prt_level.ge.1) THEN
    524          
    525          IF (ncorecfm4.ge.1) THEN
    526             print *, 'WARNING: Deacreasing updraft fraction above lalim!'
    527             print *, 'in', ncorecfm4, 'point(s)'
    528          ENDIF
    529          
    530          IF (ncorecfm5.ge.1) THEN
    531             print *, 'WARNING: Increasing mass flux above lalim!'
    532             print *, 'in', ncorecfm5, 'point(s)'
    533          ENDIF
    534          
    535          IF (ncorecfm6.ge.1) THEN
     409! Corrections display
     410!-------------------------------------------------------------------------------
     411     
     412      IF (prt_level.GE.1) THEN
     413                 
     414         IF (ncorecdetr.GE.1) THEN
    536415            print *, 'WARNING: Detrainment is greater than mass flux!'
    537             print *, 'in', ncorecfm6, 'point(s)'
    538          ENDIF
    539          
    540 ! AB : those outputs are commented because corresponding test is also commented.
    541 !         IF (ncorecfm2.ge.1) THEN
    542 !            print *, 'WARNING: Detrainment is greater than mass flux!'
    543 !            print *, 'in', ncorecfm2, 'point(s)'
    544 !         ENDIF
    545          
    546          IF (ncorecfm3.ge.1) THEN
     416            print *, '         In', ncorecdetr, 'point(s)'
     417         ENDIF
     418         
     419         IF (ncorecentr.GE.1) THEN
    547420            print *, 'WARNING: Entrained mass is greater than maximal authorized value!'
    548             print *, 'in', ncorecfm3, 'point(s)'
    549          ENDIF
    550          
    551          IF (ncorecalpha.ge.1) THEN
     421            print *, '         In', ncorecentr, 'point(s)'
     422         ENDIF
     423         
     424         IF (ncorecalpha.GE.1) THEN
    552425            print *, 'WARNING: Updraft fraction is greater than maximal authorized value!'
    553             print *, 'in', ncorecalpha, 'point(s)'
     426            print *, '         In', ncorecalpha, 'point(s)'
    554427         ENDIF
    555428         
    556429      ENDIF
    557430     
    558 !~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    559431! AB : temporary test added to check the validity of eq. df/dz = e - d
    560 !~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    561432!      DO l=1,nlay
    562433!         DO ig=1,ngrid
  • trunk/LMDZ.GENERIC/libf/phystd/thermcell_main.F90

    r2127 r2132  
    88                          f0,fm0,entr0,detr0,zw2,fraca,                       &
    99                          zqta,zqla,ztv,ztva,zhla,zhl,zqsa,                   &
    10                           lmin,lmix,lalim,lmax)
     10                          lmin,lmix,lmax)
    1111     
    1212     
     
    126126      REAL linter(ngrid)                        ! Level (continuous) of maximal vertical speed
    127127      REAL zmix(ngrid)                          ! Altitude of maximal vertical speed
    128       REAL zmax(ngrid)                          ! Maximal altitude reached by the plume
     128      REAL zmax(ngrid)                          ! Plume height
    129129      REAL wmax(ngrid)                          ! Maximal vertical speed
    130130      REAL zw2(ngrid,nlay+1)                    ! Plume vertical speed
     
    149149      CHARACTER (len=80) :: abort_message
    150150     
    151 ! AB: I remove alimentation, which is in fact included in entr_star. EN COURS
    152       INTEGER lalim(ngrid)                      ! Highest alimentation level
    153       REAL alim_star(ngrid,nlay)                ! Normalized alimentation
    154       REAL alim_star_tot(ngrid)                 ! Integrated alimentation
    155       REAL alim_star_clos(ngrid,nlay)           ! Closure alimentation
    156      
    157151!===============================================================================
    158152! Initialization
     
    183177      pdtadj(:,:) = 0.0
    184178      pdqadj(:,:,:) = 0.0
    185      
    186 ! AB: I remove alimentation, which is in fact included in entr_star. EN COURS
    187       alim_star(:,:) = 0.
    188       alim_star_tot(:) = 0.
    189       alim_star_clos(:,:) = 0.
    190       lalim(:) = 1
    191179     
    192180! AB: Careful, hard-coded value from Earth tuned version of the thermal plume model!
     
    333321      &                    ztva,zhla,zqla,zqta,zta,                           &
    334322      &                    zw2,zqsa,lmix,lmin)
    335      
     323           
    336324!-------------------------------------------------------------------------------
    337325! Thermal plumes characteristics: zmax, zmix, wmax
     
    374362     
    375363      CALL thermcell_flux(ngrid,nlay,ptimestep,masse,                         &
    376 ! AB: I remove alimentation, which is in fact included in entr_star. EN COURS
    377 !     That is not already done for thermcell_flux.
    378       &                   lalim,lmin,lmax,entr_star,detr_star,                &
     364      &                   lmin,lmax,entr_star,detr_star,                      &
    379365      &                   f,rhobarz,zlev,zw2,fm,entr,detr,zqla)
    380366     
  • trunk/LMDZ.GENERIC/libf/phystd/thermcell_mod.F90

    r2127 r2132  
    66! Flags for computations
    77                                                                  !  default
    8       INTEGER,PARAMETER :: iflag_thermals_optflux     = 1         !  0        -
    98      INTEGER,PARAMETER :: dqimpl                     = 1         !  1        flag for thermcell_dq version (1 : implicit scheme || 0 : explicit scheme)
    109     
     
    1312      REAL,PARAMETER :: r_aspect_thermals             = 1.0       !           Aspect ratio of the thermals (width / height)
    1413      REAL,PARAMETER :: tau_thermals                  = 0.        !  0.       Relaxation time
    15       REAL,PARAMETER :: betalpha                      = 1.0       !  0.9      - between 0 (e=d) and 1 (rho*fraca=cst)
    16       REAL,PARAMETER :: afact                         = 1.        !  2./3.    - buoyancy contribution, between 0 and 1
    17       REAL,PARAMETER :: fact_epsilon                  = 1.e-4     !  2.e-3    - friction
     14      REAL,PARAMETER :: betalpha                      = 1.0       !  0.9       - factor between 0 (e=d) and 1 (rho*fraca=cst)
     15      REAL,PARAMETER :: afact                         = 1.        !  2./3.    Buoyancy contribution - factor between 0 and 1
     16      REAL,PARAMETER :: fact_epsilon                  = 1.e-4     !  2.e-3    Friction at plume borders - exponential decrease
    1817      REAL,PARAMETER :: nu                            = 0.000     !           Geometrical contributions to entrainment and detrainment
    19       REAL,PARAMETER :: alphamax                      = 0.7       !  0.7      Maximal permitted updraft fraction
     18      REAL,PARAMETER :: alpha_max                     = 0.7       !  0.7      Maximal permitted updraft fraction
    2019      REAL,PARAMETER :: fomass_max                    = 0.5       !  0.5      Maximal permitted outgoing layer mass fraction
    21       REAL,PARAMETER :: pres_limit                    = 2.e5      !  1.e5     -
     20      REAL,PARAMETER :: pres_limit                    = 2.e5      !  1.e5     Minimal permitted pressure to trigger a thermal plume
    2221     
    2322!~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Note: See TracChangeset for help on using the changeset viewer.