diff --git a/MY_RUN/KTEST/003_KW78/006_diag/DIAG1.nam b/MY_RUN/KTEST/003_KW78/006_diag/DIAG1.nam new file mode 100644 index 0000000000000000000000000000000000000000..7ffdfcd844fb88088cc47592dc83db278e78054c --- /dev/null +++ b/MY_RUN/KTEST/003_KW78/006_diag/DIAG1.nam @@ -0,0 +1,12 @@ +&NAM_CONFIO LCDF4=T, LLFIOUT=F, LLFIREAD=F / +&NAM_CONFZ + ! NZ_VERB=5 , NZ_PROC=0 , NB_PROCIO_R=8 , NB_PROCIO_W=1 +/ +&NAM_DIAG + LCLSTR=T, + LBOTUP=T, + CFIELD='W', + XTHRES=10. / +&NAM_DIAG_FILE YINIFILE(1) = "KWRAI.1.SEG01.004" , + YSUFFIX='dia' / + diff --git a/MY_RUN/KTEST/003_KW78/006_diag/clean_diag_xyz b/MY_RUN/KTEST/003_KW78/006_diag/clean_diag_xyz new file mode 100755 index 0000000000000000000000000000000000000000..6daf8d6753c4b000775465519d27c06ad7490999 --- /dev/null +++ b/MY_RUN/KTEST/003_KW78/006_diag/clean_diag_xyz @@ -0,0 +1,4 @@ +set -x +rm -f KWRAI* OUTPUT_LISTING* OUTPUT_TRANSFER pipe* *.tex +rm -f file_for_xtransfer + diff --git a/MY_RUN/KTEST/003_KW78/006_diag/diag.nam b/MY_RUN/KTEST/003_KW78/006_diag/diag.nam new file mode 100644 index 0000000000000000000000000000000000000000..a1252a72b1b6ef1bf8d6ad58ebe2ffd962e53fc6 --- /dev/null +++ b/MY_RUN/KTEST/003_KW78/006_diag/diag.nam @@ -0,0 +1,12 @@ + +exemple 1, identification des ascendances > 10 m/s + +&NAM_DIAG LCLSTR = T, LBOTUP = T, CFIELD = "W", XTHRES = 10. / + + + +exemple 2, identification des overshoots + +&NAM_DIAG LCLSTR = T, LBOTUP = F, CFIELD = "CLOUD", XTHRES = 0.00001 / + +NB: par défaut CLOUD prend en compte Rc+Ri+Rs+Rg i.e. cloud water and ice, graupel and snow diff --git a/MY_RUN/KTEST/003_KW78/006_diag/run_diag_xyz b/MY_RUN/KTEST/003_KW78/006_diag/run_diag_xyz new file mode 100755 index 0000000000000000000000000000000000000000..9b504a69dc91c64f27f55d335270d79dc659e6bc --- /dev/null +++ b/MY_RUN/KTEST/003_KW78/006_diag/run_diag_xyz @@ -0,0 +1,15 @@ +#!/bin/sh +#MNH_LIC Copyright 1994-2014 CNRS, Meteo-France and Universite Paul Sabatier +#MNH_LIC This is part of the Meso-NH software governed by the CeCILL-C licence +#MNH_LIC version 1. See LICENSE, CeCILL-C_V1-en.txt and CeCILL-C_V1-fr.txt +#MNH_LIC for details. version 1. +set -x +set -e +rm -f KWRAI* OUT* +# +# +ln -sf ../002_mesonh/KWRAI.1.SEG01.004.des . +ln -sf ../002_mesonh/KWRAI.1.SEG01.004.nc4 . + +time ${MPIRUN} DIAG${XYZ} + diff --git a/MY_RUN/KTEST/003_KW78/007_ncl_diag/clean_diag_xyz b/MY_RUN/KTEST/003_KW78/007_ncl_diag/clean_diag_xyz new file mode 100755 index 0000000000000000000000000000000000000000..6daf8d6753c4b000775465519d27c06ad7490999 --- /dev/null +++ b/MY_RUN/KTEST/003_KW78/007_ncl_diag/clean_diag_xyz @@ -0,0 +1,4 @@ +set -x +rm -f KWRAI* OUTPUT_LISTING* OUTPUT_TRANSFER pipe* *.tex +rm -f file_for_xtransfer + diff --git a/MY_RUN/KTEST/003_KW78/007_ncl_diag/clean_ncl b/MY_RUN/KTEST/003_KW78/007_ncl_diag/clean_ncl new file mode 100755 index 0000000000000000000000000000000000000000..9c04c2a804db6e7421b6de10a9e937f55d859447 --- /dev/null +++ b/MY_RUN/KTEST/003_KW78/007_ncl_diag/clean_ncl @@ -0,0 +1,2 @@ +set -x +rm -f KWRAI.1.SEG01.004* visu_KW78_diag.png \ No newline at end of file diff --git a/MY_RUN/KTEST/003_KW78/007_ncl_diag/plot_KW78_diag.ncl b/MY_RUN/KTEST/003_KW78/007_ncl_diag/plot_KW78_diag.ncl new file mode 100755 index 0000000000000000000000000000000000000000..20285ca41a241925a074bee150683a773c9b7dbd --- /dev/null +++ b/MY_RUN/KTEST/003_KW78/007_ncl_diag/plot_KW78_diag.ncl @@ -0,0 +1,127 @@ +;================================================; + load "$NCARG_ROOT/lib/ncarg/nclscripts/csm/gsn_code.ncl" + load "$NCARG_ROOT/lib/ncarg/nclscripts/csm/gsn_csm.ncl" + load "$NCARG_ROOT/lib/ncarg/nclscripts/csm/contributed.ncl" + load "$NCARG_ROOT/lib/ncarg/nclscripts/wrf/WRFUserARW.ncl" +; ================================================; +;================================================; + begin +;=================================================; +; open file and read in data +;=================================================; + fichier1 = addfile("KWRAI.1.SEG01.004.nc4", "r") + fichier2 = addfile("KWRAI.1.SEG01.004dia.nc4", "r") +;==================================================; +; Open the workstation +;==================================================; + type = "png" + wks = gsn_open_wks(type,"visu_KW78_diag") +;=================================================; +; Get informations on variable sizes +; dims are dims-2 to remove non-physical values +;=================================================; + jphext = fichier2->JPHEXT + mdims = getfilevardimsizes(fichier2,"CLUSTERID") + nd = dimsizes(mdims) + imax=mdims(nd-1)-2*jphext + jmax=mdims(nd-2)-2*jphext + kmax=mdims(nd-3)-2 + + +;=================================================; +; lecture des différents champs +;=================================================; + zs = fichier1->ZS(jphext:jmax+jphext-1,jphext:imax+jphext-1) ; ZS + zhat = fichier1->ZHAT(1:kmax+1) ; ZHAT + yhat = fichier1->YHAT(1:jmax+jphext-1+1) ; YHAT + xhat = fichier1->XHAT(1:imax+jphext-1+1) ; XHAT + + wt= fichier1->WT(1:kmax+1,jphext:jmax+jphext-1,jphext:imax+jphext-1) ; WT + vt= fichier1->VT(1:kmax,jphext:jmax+jphext-1+1,jphext:imax+jphext-1) ; VT + ut= fichier1->UT(1:kmax,jphext:jmax+jphext-1,jphext:imax+jphext-1+1) ; UT + + idt1= fichier2->CLUSTERID(1:kmax,jphext:jmax+jphext-1,jphext:imax+jphext-1) ; CLUSTERID + idt1@long_name="Cluster Identity Number" + idt1@units="" + + YYYYDDMM1= fichier1->DTCUR__TDATE + SS1=fichier1->DTCUR__TIME + +;=================================================; +; Récupération de la date +;=================================================; + year1=YYYYDDMM1(0) + day1=YYYYDDMM1(1) + min1=YYYYDDMM1(2) + time1=SS1 +;=================================================; +; On mets toutes les variables sur la grille 1 +;=================================================; + + ut1 = wrf_user_unstagger(ut,"X") + ut1@long_name="Zonal wind" + ut1@units="m/s" + vt1 = wrf_user_unstagger(vt,"Y") + vt1@long_name="Meridian wind" + vt1@units="m/s" + wt1 = wrf_user_unstagger(wt,"Z") + wt1@long_name="Vertical wind" + wt1@units="m/s" + +;=================================================; +; Set some other basic resources +;=================================================; + resmap = True + resmap@gsnFrame = False + resmap@gsnDraw = False + resmap@gsnMaximize = True + resmap@gsnPaperOrientation = "portrait" + resmap@gsnSpreadColors= True + resmap@tiYAxisString =" " + resmap@cnFillOn= True + resmap@cnLinesOn= False + resmap@lbLabelStride = 12. ; to reduce the number of labels on labelbar +;=================================================; +; Altitude des niveaux modèles +;=================================================; +; Unstagger zhat (from grid 4 to 1) + nzh=new(kmax,double) + do k=0,kmax-1 + nzh(k)=(zhat(k)+zhat(k+1))/2. + end do + +; Create alt (altitude des niveaux modèle) + alt=new(dimsizes(ut1),double) + zcoef=new((/jmax,imax/),double) + zcoef=1.-zs/nzh(kmax-1) + do i=0,imax-1 + do j=0,jmax-1 + alt(:,j,i) = nzh*zcoef(j,i)+zs(j,i) + end do + end do + + +;======================================= +; coupe verticale clustering +;======================================= + + gsn_define_colormap(wks,"BlRe") + + opt_cv=resmap + opt_cv@gsnScalarContour=True + opt_cv@vcMinDistanceF = 0.03 ; espacement des fleches + opt_cv@vcRefLengthF=0.1 ; taille des flèches + opt_cv@cnLevelSpacingF = 0.0005 + opt_cv@cnMinLevelValF = 0 + opt_cv@cnLevelSelectionMode = "ExplicitLevels" + opt_cv@cnLevels = (/1/) + opt_cv@cnFillColors = (/0,84/) + opt_cv@tiMainString = "Identified Updraft as cluster where w > 10 m/s" + + plot_idt=gsn_csm_vector_scalar(wks,ut1(:,13,:),wt1(:,13,:),idt1(:,13,:),opt_cv) + + draw(plot_idt) + frame(wks) + + +end diff --git a/MY_RUN/KTEST/003_KW78/007_ncl_diag/run_ncl b/MY_RUN/KTEST/003_KW78/007_ncl_diag/run_ncl new file mode 100755 index 0000000000000000000000000000000000000000..eb0f0d91743bfc5dd88b59216c5c657897d2b9ca --- /dev/null +++ b/MY_RUN/KTEST/003_KW78/007_ncl_diag/run_ncl @@ -0,0 +1,13 @@ +#MNH_LIC Copyright 1994-2014 CNRS, Meteo-France and Universite Paul Sabatier +#MNH_LIC This is part of the Meso-NH software governed by the CeCILL-C licence +#MNH_LIC version 1. See LICENSE, CeCILL-C_V1-en.txt and CeCILL-C_V1-fr.txt +#MNH_LIC for details. version 1. +set -x + +ln -sf ../002_mesonh/KWRAI.1.SEG01.004.nc4 . +ln -sf ../006_diag/KWRAI.1.SEG01.004dia.nc4 . + +ncl plot_KW78_diag.ncl +display visu_KW78_diag.png + +exit 0 diff --git a/MY_RUN/KTEST/003_KW78/007_ncl_diag/visu_KW78_diag.png b/MY_RUN/KTEST/003_KW78/007_ncl_diag/visu_KW78_diag.png new file mode 100644 index 0000000000000000000000000000000000000000..52376e3401b394d812df4f65ff6341216c741e5e Binary files /dev/null and b/MY_RUN/KTEST/003_KW78/007_ncl_diag/visu_KW78_diag.png differ diff --git a/MY_RUN/KTEST/003_KW78/Makefile b/MY_RUN/KTEST/003_KW78/Makefile index e50e3b333d17d9505c803b9ba51ca34fcdcaafea..43f486a681a58240ba48a5fa53a950b27e684939 100755 --- a/MY_RUN/KTEST/003_KW78/Makefile +++ b/MY_RUN/KTEST/003_KW78/Makefile @@ -4,6 +4,8 @@ all: cd 003_convdia && run_conv2dia cd 004_diaprog && run_diaprog cd 005_ncl && run_ncl + cd 006_diag && run_diag_xyz + cd 007_ncl_diag && run_ncl clean: cd 001_prep_ideal_case && clean_prep_ideal_case_xyz @@ -11,3 +13,5 @@ clean: cd 003_convdia && clean_conv2dia cd 004_diaprog && clean_diaprog cd 005_ncl && clean_ncl + cd 006_diag && clean_diag_xyz + cd 007_ncl_diag && clean_ncl diff --git a/src/MNH/clustering.f90 b/src/MNH/clustering.f90 new file mode 100644 index 0000000000000000000000000000000000000000..a42562bbb7640e36ba3f9d2f2cf0702691bbba1d --- /dev/null +++ b/src/MNH/clustering.f90 @@ -0,0 +1,730 @@ +! ################ + MODULE MODI_CLUSTERING +! ################ +! +INTERFACE +! + SUBROUTINE CLUSTERING(OBOTTOMUP,OCLOUD,PFIELD,KCLUSTERIDT,KCLUSTERLVL,PCLUSTERSEC) +! +LOGICAL, INTENT(IN) :: OBOTTOMUP +LOGICAL, DIMENSION(:,:,:), INTENT(IN) :: OCLOUD +REAL, DIMENSION(:,:,:), INTENT(IN) :: PFIELD +INTEGER, DIMENSION(:,:,:), INTENT(OUT) :: KCLUSTERIDT +INTEGER, DIMENSION(:,:,:), INTENT(OUT) :: KCLUSTERLVL +REAL, DIMENSION(:,:,:), INTENT(OUT) :: PCLUSTERSEC +! +END SUBROUTINE CLUSTERING +! +END INTERFACE +! +END MODULE MODI_CLUSTERING +! +! ####################### + SUBROUTINE CLUSTERING(OBOTTOMUP,OCLOUD,PFIELD,KCLUSTERIDT,KCLUSTERLVL,PCLUSTERSEC) +! ####################### + ! +!! PURPOSE +!! ------- +!! +!! Identify structures as 3D objects made of connected points. +!! +!!** METHOD +!! ------ +!! +!! Uses a 3D mask as input. Outputs 3D fields containing: +!! IDT the identitdy number of each structure +!! LVL the level from which each structure is identified +!! SEC the section of each structure at current level +!! +!! Both IDT and LVL are necessary to identify unequivocally the structures. +!! +!! The identification is first led level by level. +!! The connection between the levels is done after. +!! +!! +!! AUTHOR +!! ------ +!! T. Dauhut * LA * +!! J.-P. Chaboureau * LA * +!! +!! MODIFICATIONS +!! ------------- +!! Original 05/06/13 +!! Modified 05/08/14 T. Dauhut v75 +!! Modified 10/10/14 T. Dauhut toward 3D version +!! Modified 13/11/14 T. Dauhut adding property field analyse +!! Modified 13/06/17 T. Dauhut to start volume scan from top +!! Modified 04/10/17 T. Dauhut to be added to next MNH versions +!! +!------------------------------------------------------------------------------- +! +!* 0. DECLARATIONS +! ------------ + USE MODD_MPIF , ONLY : MPI_INTEGER +USE MODE_ll +USE MODD_ARGSLIST_ll, ONLY : LIST_ll +USE MODD_VAR_ll, ONLY : MPI_PRECISION, NPROC, IP, NMNH_COMM_WORLD +USE MODD_DYN_n, ONLY : XDXHATM, XDYHATM +!USE& +! MPI +! +IMPLICIT NONE +! +!* 0.1 declarations of arguments +! +LOGICAL, INTENT(IN) :: OBOTTOMUP +LOGICAL, DIMENSION(:,:,:), INTENT(IN) :: OCLOUD +REAL , DIMENSION(:,:,:), INTENT(IN) :: PFIELD +INTEGER, DIMENSION(:,:,:), INTENT(OUT) :: KCLUSTERIDT +INTEGER, DIMENSION(:,:,:), INTENT(OUT) :: KCLUSTERLVL +REAL, DIMENSION(:,:,:), INTENT(OUT) :: PCLUSTERSEC +! +!* 0.2 declarations of local variables +! +INTEGER, PARAMETER :: JPUNDEF = 999999 ! NaN value outside clusters +INTEGER :: IKBEG,IKEND,IKINC ! for the loop on the levels +INTEGER :: JI,JJ,JC,JK,JH,JDI,JDJ ! loop counters +INTEGER :: IMINI, IMIX, IMUX ! local cluster index +INTEGER :: IIB,IJB,IIE,IJE,IKE,IXOR,IYOR ! physical bounds of local domain +INTEGER :: IIU, IJU ! dimensions of local domain+halo +INTEGER :: NIMAX_ll, NJMAX_ll ! dimensions of global domain +INTEGER :: ICLUSMAX ! maximum nbr of clusters in local domain +INTEGER :: ICPT, ICHANGES, ICPR ! local counter +INTEGER :: ITEMPIDT,ITEMPLVL,ITEMPAFL ! temporary cluster ID, baselevel, affluent section +REAL :: ZMESHAREA +INTEGER, ALLOCATABLE, DIMENSION(:) :: ICLUSIND ! 1D table for local indices +INTEGER, ALLOCATABLE, DIMENSION(:) :: ICLUSIZE ! 1D table for local cluster size in pixels +INTEGER, ALLOCATABLE, DIMENSION(:,:) :: IMAPIND ! local 2D field pointing toward local indices +INTEGER, ALLOCATABLE, DIMENSION(:) :: ICLUSTERIDT ! 1D table for cluster ID (need LVL to be unequivocal) +INTEGER, ALLOCATABLE, DIMENSION(:) :: ICLUSTERLVL ! 1D table for cluster base level +INTEGER, ALLOCATABLE, DIMENSION(:) :: IAFFLUSECTN ! 1D table for cluster affluent section +INTEGER, DIMENSION(SIZE(KCLUSTERIDT,1),SIZE(KCLUSTERIDT,2)) :: IMAPIDT_ll,IMAPLVL_ll,IMAPAFL_ll +REAL, DIMENSION(SIZE(KCLUSTERIDT,1),SIZE(KCLUSTERIDT,2)) :: ZMAPIDT_ll,ZMAPLVL_ll,ZMAPAFL_ll +REAL, ALLOCATABLE, DIMENSION(:) :: ZCLUSSUMFLD ! 1D table for cluster local sum of field values +! +!* 0.3 declarations of variables for MPI exchanges +! +INTEGER :: ITOTNBR ! counter to create final tables (.GLBLIST..) +INTEGER :: IRANK, IROOT, INFO ! for MPI subroutines +INTEGER :: IINFO_ll ! (MNH-MPI) return code of parallel routine +TYPE(LIST_ll), POINTER :: TZFIELDS_ll ! (MNH-MPI) list of fields to exchange +INTEGER, ALLOCATABLE, DIMENSION(:) :: ICLUSNBR ! 1D table for nbr of clusters in local domains +INTEGER, ALLOCATABLE, DIMENSION(:) :: IPROCDPL ! 1D table for procs starting indices in global tables +INTEGER, ALLOCATABLE, DIMENSION(:) :: ILOCIND ! old indices point toward new ones after optimization +INTEGER, ALLOCATABLE, DIMENSION(:) :: ILOCLISTIDT,ILOCLISTIDT2 ! non-redundant list of local cluster IDs +INTEGER, ALLOCATABLE, DIMENSION(:) :: ILOCLISTLVL,ILOCLISTLVL2 ! their corresponding base level +INTEGER, ALLOCATABLE, DIMENSION(:) :: ILOCLISTSEC,ILOCLISTSEC2 ! their corresponding sections +REAL , ALLOCATABLE, DIMENSION(:) :: ZLOCLISTFLD,ZLOCLISTFLD2 ! their corresponding field average +INTEGER, ALLOCATABLE, DIMENSION(:) :: IGLBLISTIDT ! concatenation of local domain cluster ID lists +INTEGER, ALLOCATABLE, DIMENSION(:) :: IGLBLISTLVL ! their corresponding base level +INTEGER, ALLOCATABLE, DIMENSION(:) :: IGLBLISTSEC ! their corresponding sections +REAL , ALLOCATABLE, DIMENSION(:) :: ZGLBLISTFLD ! their corresponding field average +! +!------------------------------------------------------------------------------- +! +!* 1. INITIALISATION +! -------------- +! +INFO = -1 +IROOT = 0 +CALL MPI_COMM_RANK(NMNH_COMM_WORLD, IRANK, INFO) ! get the rank of the current processor +! +IKE = SIZE(OCLOUD,3) +! +IF (OBOTTOMUP) THEN + IKBEG=1 ! scans the volume from the surface + IKEND=IKE ! up to the top + IKINC=1 ! increment to next level +ELSE + IKBEG=IKE ! scans the volume from the top + IKEND=1 ! down to the surface + IKINC=-1 ! increment to next level +END IF +! +DO JK=IKBEG,IKEND,IKINC +!PRINT *,IRANK,'initialisation, level =',JK +! +NULLIFY(TZFIELDS_ll) +! +CALL GET_GLOBALDIMS_ll (NIMAX_ll,NJMAX_ll) +CALL GET_DIM_EXT_ll ('B',IIU,IJU) +CALL GET_INDICE_ll (IIB,IJB,IIE,IJE) +CALL GET_OR_ll('B',IXOR,IYOR) +! +!------------------------------------------------------------------------------- +! +!* 2. IDENTIFY CLUSTER PIXELS THANKS TO OCLOUD MASK +! --------------------------------------------- +!PRINT *,IRANK,'identification' +! +!*--------------------------------------------------------------------*C +!* 2.1 identify clusters by uninterrupted zonal segments +!* 1st label in IMAPIND -- & -- 1st measure of size in ICLUSIZE +!*--------------------------------------------------------------------*C +! +ALLOCATE(IMAPIND(IIU,IJU)) +IMAPIND(:,:) = JPUNDEF ! The pixels first point toward no local indices. +! +ICLUSMAX=IIU*IJU +ALLOCATE(ICLUSIZE(ICLUSMAX)) +ICLUSIZE(:) = 0 ! The local cluster size is initiated to 0. +! +ICPT = 0 +! +DO JJ=IJB,IJE + IF (OCLOUD(IIB,JJ,JK)) THEN + ICPT = ICPT + 1 ! at begining of a row, pixel points toward a new local index + IMAPIND(IIB,JJ) = ICPT + ICLUSIZE(ICPT) = ICLUSIZE(ICPT) + 1 + END IF + DO JI=IIB+1,IIE + IF (OCLOUD(JI,JJ,JK)) THEN + IF (.NOT.OCLOUD(JI-1,JJ,JK)) THEN + ICPT = ICPT + 1 ! keep previous local index only if previous pixel mask was also true + END IF + IMAPIND(JI,JJ) = ICPT + ICLUSIZE(ICPT) = ICLUSIZE(ICPT) + 1 + END IF + END DO +END DO +!PRINT *,'IP=',IP,' ICPT=',ICPT,' COUNT(OCLOUD)=',COUNT(OCLOUD) +!IF (ICPT.GT.ICLUSMAX) PRINT *,"ICLUSMAX=",ICLUSMAX," LT ICPT=",ICPT +! +! IMAPIND(JI,JJ) = JC means that (JI,JJ) pixel points toward JC as local index +! IMAPIND is JPUNDEF where OCLOUD is FALSE, it is in [1,ICLUSMAX] elsewhere +! IMAPIND have the same value along uninterrupted zonal rows +! IMAPIND will not change before step 4. +! +! ICLUSIZE(JC) is equal to the number of pixels pointing toward JC as local index +! ICLUSIZE will not change before step 3. +! sum(ICLUSIZE) is a constant and it is equal to num(OCLOUD) in the local domain +! +!*--------------------------------------------------------------------*C +!* 2.2 GATHER CONNECTED CLUSTERS +!* create ICLUSIND table which will contain Local Cluster Indices +!* those Indices will further point toward Global Cluster ID in ICLUSTER... +!*--------------------------------------------------------------------*C +! +ALLOCATE(ICLUSIND(ICLUSMAX)) +ICLUSIND = (/ (JC, JC=1,ICLUSMAX) /) +! +DO JJ=IJB,IJE + DO JI=IIB,IIE +! DO JI=IIB,IIE-1 + IF (OCLOUD(JI,JJ,JK)) THEN +!*--------------------------------------------------------------------*C +!* MINIMUM OF THE REGIONS THAT BORDER THIS ONE + IMINI = IMAPIND(JI,JJ) + IMINI = MIN(IMINI,MIN(IMAPIND(JI-1,JJ ),IMAPIND(JI ,JJ-1))) +! IMINI = MIN(IMINI,MIN(IMAPIND(JI-1,JJ-1),IMAPIND(JI+1,JJ-1))) +!*--------------------------------------------------------------------*C +!* SEE WHETHER THERE ARE TWO CHAINS +!*--------------------------------------------------------------------*C + IF (IMINI.GT.ICLUSIND(IMAPIND(JI,JJ))) THEN +!*--------------------------------------------------------------------*C +!* LOOK FOR THE MINIMUM OF THE CHAIN STARTING WITH KCLUSTER(I,J) +!*--------------------------------------------------------------------*C + IMIX = ICLUSIND(IMAPIND(JI,JJ)) + DO + IF ( ICLUSIND(IMIX) .NE. IMIX ) THEN + IMIX = ICLUSIND(IMIX) + ELSE + EXIT + END IF + END DO +!*--------------------------------------------------------------------*C +!* LOOK FOR THE MINIMUM OF THE CHAIN STARTING WITH MINI +!*--------------------------------------------------------------------*C + IMUX = ICLUSIND(IMINI) + DO + IF ( ICLUSIND(IMUX) .NE. IMUX ) THEN + IMUX = ICLUSIND(IMUX) + ELSE + EXIT + END IF + END DO +!*--------------------------------------------------------------------*C +!* USE THE ABSOLUTE MINIMUM OF THE TWO CHAINS TO CONNECT THEM +!*--------------------------------------------------------------------*C + IF (IMUX.GT.IMIX) THEN + ICLUSIND(IMUX) = IMIX + ELSEIF (IMIX.GT.IMUX) THEN + ICLUSIND(IMIX) = IMUX + END IF +!*--------------------------------------------------------------------*C +!* SINGLE CHAIN +!*--------------------------------------------------------------------*C + ELSE + ICLUSIND(IMAPIND(JI,JJ)) = IMINI + END IF + END IF + END DO +END DO +! +! since now, each Local Index JC that belongs to a pixel in the mask +! points (via ICLUSIND) toward itself -OR- toward a lower Local Index +! that belongs to a connected pixel in the mask +! +! +!--------------------------------------------------------------------------------- +! +!* 3. OPTIMISE ICLUSIND TABLE -&- GATHER LOCAL CLUSTER SIZES IN ICLUSIZE +! ------------------------------------------------------------------ +!PRINT *,IRANK,'optimisation' +! +DO JC=1,ICLUSMAX + ICLUSIND(JC) = ICLUSIND(ICLUSIND(JC)) + IF (ICLUSIND(JC).LT.JC) THEN + ICLUSIZE(ICLUSIND(JC)) = ICLUSIZE(ICLUSIND(JC)) + ICLUSIZE(JC) + ICLUSIZE(JC) = 0 + END IF +END DO +! +! since now, each Local Index JC that belong to a pixel in the mask +! points (via ICLUSIND) toward the LOWEST Local Index +! that belongs to a *connected* pixel in the mask +! +! ICLUSIZE(JC) gives now the number of pixels +! that point (via IMAPIND and ICLUSIND) toward JC as Local Index +! sum(ICLUSIZE) is still equal to num(OCLOUD) in the local domain +! +!---------------------------------------------------------------------------------- +! +!* 4. GIVE TO CLUSTER PIXEL LOWEST LOCAL INDEX AND THEN GLOBAL CLUSTER ID +! ------------------------------------------------------------------- +!PRINT *,IRANK,'naming' +! +!*--------------------------------------------------------------------*C +!* 2nd label in IMAPIND -- & -- 1st label in in ICLUSTER... and IMAP..._ll +!*--------------------------------------------------------------------*C +! +ALLOCATE(ICLUSTERIDT(ICLUSMAX)) +ALLOCATE(ICLUSTERLVL(ICLUSMAX)) +ALLOCATE(IAFFLUSECTN(ICLUSMAX)) +ALLOCATE(ZCLUSSUMFLD(ICLUSMAX)) +ICLUSTERIDT(:) = 0 ! local indices first indicate no global cluster ID +ICLUSTERLVL(:) = IKE+1 ! and cluster base level is setup above highest level +IAFFLUSECTN(:) = 0 ! no affluent from under so affluent section = 0 +ZCLUSSUMFLD(:) = 0. ! sum of field values inside the clusters is set up to 0. +! +! 4.1 CHOOSING CLUSTER FUTURE NAMES & WRITING IT AND PROPERTIES IN ICLUSTER... +! (UPDATE 1D TABLES) +! +DO JJ=IJB,IJE + DO JI=IIB,IIE + IF (OCLOUD(JI,JJ,JK)) THEN + ! using optimised ICLUSIND to connect pixel to lowest local index pixel + ! belonging to the same cluster in this horizontal level + IMAPIND(JI,JJ) = ICLUSIND(IMAPIND(JI,JJ)) ! lowest local index + ! adding field value of current pixel to its cluster sum + ZCLUSSUMFLD(IMAPIND(JI,JJ)) = ZCLUSSUMFLD(IMAPIND(JI,JJ))+PFIELD(JI,JJ,JK) + ! in case that pixel belongs to an already named cluster: + ITEMPIDT = ICLUSTERIDT(IMAPIND(JI,JJ)) ! potential IDnumber + ITEMPLVL = ICLUSTERLVL(IMAPIND(JI,JJ)) ! potential baselvl + ITEMPAFL = IAFFLUSECTN(IMAPIND(JI,JJ)) ! potential affluent section + ! checking whether there is an identified cluster at same position in previous level, an "affluent" + ! (previous level is JK-IKINC) + ! in this case, we settup future name in consequence + IF (JK.NE.IKBEG) THEN + ! + DO JDI=-1,1 ! cross-shape sieve of conseidered underlying pixels + DO JDJ=-1,1 ! JI +/- 1 and JJ +/- 1 and (JI,JJ) + IF ((ABS(JDI)+ABS(JDJ)).LT.2) THEN + ! + IF (OCLOUD(JI+JDI,JJ+JDJ,JK-IKINC)) THEN + ! 1st priority given to biggest affluent in section + IF (PCLUSTERSEC(JI+JDI,JJ+JDJ,JK-IKINC).GT.ITEMPAFL) THEN + ITEMPIDT = KCLUSTERIDT(JI+JDI,JJ+JDJ,JK-IKINC) + ITEMPLVL = KCLUSTERLVL(JI+JDI,JJ+JDJ,JK-IKINC) + ITEMPAFL = PCLUSTERSEC(JI+JDI,JJ+JDJ,JK-IKINC) + ELSEIF (PCLUSTERSEC(JI+JDI,JJ+JDJ,JK-IKINC).EQ.ITEMPAFL) THEN + ! 2nd priority given to affluent with lowest base + IF (KCLUSTERLVL(JI+JDI,JJ+JDJ,JK-IKINC).LT.ITEMPLVL) THEN + ITEMPIDT = KCLUSTERIDT(JI+JDI,JJ+JDJ,JK-IKINC) + ITEMPLVL = KCLUSTERLVL(JI+JDI,JJ+JDJ,JK-IKINC) + ELSEIF (KCLUSTERLVL(JI+JDI,JJ+JDJ,JK-IKINC).EQ.ITEMPLVL) THEN + ! 3rd priority given to affluent with lowest IDnumber + IF (ITEMPIDT.EQ.0) THEN + ITEMPIDT = KCLUSTERIDT(JI+JDI,JJ+JDJ,JK-IKINC) + ELSE + ITEMPIDT = MIN(KCLUSTERIDT(JI+JDI,JJ+JDJ,JK-IKINC),ITEMPIDT) + END IF + END IF + END IF + END IF + ! + END IF + END DO + END DO + ! + END IF + ! + IF (ITEMPIDT.EQ.0) THEN + ICLUSTERIDT(IMAPIND(JI,JJ)) = NIMAX_ll*(IYOR+JJ-1) + (IXOR+JI) + ICLUSTERLVL(IMAPIND(JI,JJ)) = JK + IAFFLUSECTN(IMAPIND(JI,JJ)) = 0 + ELSE + ICLUSTERIDT(IMAPIND(JI,JJ)) = ITEMPIDT + ICLUSTERLVL(IMAPIND(JI,JJ)) = ITEMPLVL + IAFFLUSECTN(IMAPIND(JI,JJ)) = ITEMPAFL + END IF + ! + END IF ! OCLOUD(JI,JJ) is TRUE + END DO +END DO +! +DEALLOCATE(ICLUSIND) +! +! now, ICLUSTER... has a name for each cluster in the current level +! the same as the cluster under if there is one, a new name otherwise +! in case of several possibilities, we linked current level cluster prioritarly with: +! 1. cluster under with the bigest section +! 2. cluster under with the lowest base level +! 3. cluster under with the lowest ID number +! +!* 4.2 GIVING ID, BASE LEVEL and AFFLUENT SECTION in IMAP..._ll +! (COLORATION OF 2D TABLES) +! +IMAPIDT_ll(:,:) = 0 ! pixels fist have no global cluster ID +IMAPLVL_ll(:,:) = IKE+1 ! and cluster base level is setup above highest level +IMAPAFL_ll(:,:) = 0 ! no affluent from under so affluent section = 0 +! +DO JJ=IJB,IJE + DO JI=IIB,IIE + IF (OCLOUD(JI,JJ,JK)) THEN + IMAPIDT_ll(JI,JJ) = ICLUSTERIDT(IMAPIND(JI,JJ)) ! global cluster ID + IMAPLVL_ll(JI,JJ) = ICLUSTERLVL(IMAPIND(JI,JJ)) ! cluster baselevel + IMAPAFL_ll(JI,JJ) = IAFFLUSECTN(IMAPIND(JI,JJ)) ! affluent section + END IF + END DO +END DO +! +!------------------------------------------------------------------------------- +! +!* 5. UPDATE HALO AND UNIFY ADJACENT CLUSTERS GIVING COMMON ID* +! -------------------------------------------------------- +! *ID is defined unequivocally by (IDT number, base LVL) +! +! (MPI COMMUNICATIONS FOR 2D FIELDS: +! IDT, LVL, AFL useful for clustering) +! +!PRINT *,IRANK,'halo-update' +! +DO JH=1,9999 ! update halo until there is no more changes anywhere +!PRINT *,'compteur premiere boucle infinie =',JH +! +ZMAPIDT_ll=IMAPIDT_ll +ZMAPLVL_ll=IMAPLVL_ll +ZMAPAFL_ll=IMAPAFL_ll +CALL ADD2DFIELD_ll(TZFIELDS_ll,ZMAPIDT_ll) +CALL ADD2DFIELD_ll(TZFIELDS_ll,ZMAPLVL_ll) +CALL ADD2DFIELD_ll(TZFIELDS_ll,ZMAPAFL_ll) +CALL UPDATE_HALO_ll(TZFIELDS_ll,IINFO_ll) +IMAPIDT_ll=NINT(ZMAPIDT_ll) +IMAPLVL_ll=NINT(ZMAPLVL_ll) +IMAPAFL_ll=NINT(ZMAPAFL_ll) +ICHANGES = 0 ! counter of changes +! +DO JJ=IJB,IJE ! check Western and Eastern borders of local domain +! +IF ( (IMAPIDT_ll(IIB,JJ).NE.0).AND.(IMAPIDT_ll(1,JJ).NE.0) ) THEN + IF ( IMAPAFL_ll(1,JJ).GT.IAFFLUSECTN(IMAPIND(IIB,JJ)) ) THEN + ! if halo pixel has a bigger affluent than current pixel + ICLUSTERIDT(IMAPIND(IIB,JJ)) = IMAPIDT_ll(1,JJ) + ICLUSTERLVL(IMAPIND(IIB,JJ)) = IMAPLVL_ll(1,JJ) + IAFFLUSECTN(IMAPIND(IIB,JJ)) = IMAPAFL_ll(1,JJ) + ICHANGES = ICHANGES + 1 + ELSEIF (IMAPAFL_ll(1,JJ).EQ.IAFFLUSECTN(IMAPIND(IIB,JJ))) THEN + IF (IMAPLVL_ll(1,JJ).LT.ICLUSTERLVL(IMAPIND(IIB,JJ))) THEN + ! if halo pixel belongs to a cluster with a lower base + ICLUSTERIDT(IMAPIND(IIB,JJ)) = IMAPIDT_ll(1,JJ) + ICLUSTERLVL(IMAPIND(IIB,JJ)) = IMAPLVL_ll(1,JJ) + ICHANGES = ICHANGES + 1 + ELSEIF (IMAPLVL_ll(1,JJ).EQ.ICLUSTERLVL(IMAPIND(IIB,JJ))) THEN + IF (IMAPIDT_ll(1,JJ).LT.ICLUSTERIDT(IMAPIND(IIB,JJ))) THEN + ! if halo pixel belongs to a cluster with a lower ID number + ICLUSTERIDT(IMAPIND(IIB,JJ)) = IMAPIDT_ll(1,JJ) + ICHANGES = ICHANGES + 1 + END IF + END IF + END IF +END IF +! +IF ( (IMAPIDT_ll(IIE,JJ).NE.0).AND.(IMAPIDT_ll(IIU,JJ).NE.0) ) THEN + IF ( IMAPAFL_ll(IIU,JJ).GT.IAFFLUSECTN(IMAPIND(IIE,JJ)) ) THEN + ! if halo pixel has a bigger affluent than current pixel + ICLUSTERIDT(IMAPIND(IIE,JJ)) = IMAPIDT_ll(IIU,JJ) + ICLUSTERLVL(IMAPIND(IIE,JJ)) = IMAPLVL_ll(IIU,JJ) + IAFFLUSECTN(IMAPIND(IIE,JJ)) = IMAPAFL_ll(IIU,JJ) + ICHANGES = ICHANGES + 1 + ELSEIF (IMAPAFL_ll(IIU,JJ).EQ.IAFFLUSECTN(IMAPIND(IIE,JJ))) THEN + IF (IMAPLVL_ll(IIU,JJ).LT.ICLUSTERLVL(IMAPIND(IIE,JJ))) THEN + ! if halo pixel belongs to a cluster with a lower base + ICLUSTERIDT(IMAPIND(IIE,JJ)) = IMAPIDT_ll(IIU,JJ) + ICLUSTERLVL(IMAPIND(IIE,JJ)) = IMAPLVL_ll(IIU,JJ) + ICHANGES = ICHANGES + 1 + ELSEIF (IMAPLVL_ll(IIU,JJ).EQ.ICLUSTERLVL(IMAPIND(IIE,JJ))) THEN + IF (IMAPIDT_ll(IIU,JJ).LT.ICLUSTERIDT(IMAPIND(IIE,JJ))) THEN + ! if halo pixel belongs to a cluster with a lower ID number + ICLUSTERIDT(IMAPIND(IIE,JJ)) = IMAPIDT_ll(IIU,JJ) + ICHANGES = ICHANGES + 1 + END IF + END IF + END IF +END IF +! +END DO ! two borders checked +! +DO JI=IIB,IIE ! check Southern and Northern borders of local domain +! +IF ( (IMAPIDT_ll(JI,IJB).NE.0).AND.(IMAPIDT_ll(JI,1).NE.0) ) THEN + IF ( IMAPAFL_ll(JI,1).GT.IAFFLUSECTN(IMAPIND(JI,IJB)) ) THEN + ! if halo pixel has a bigger affluent than current pixel + ICLUSTERIDT(IMAPIND(JI,IJB)) = IMAPIDT_ll(JI,1) + ICLUSTERLVL(IMAPIND(JI,IJB)) = IMAPLVL_ll(JI,1) + IAFFLUSECTN(IMAPIND(JI,IJB)) = IMAPAFL_ll(JI,1) + ICHANGES = ICHANGES + 1 + ELSEIF (IMAPAFL_ll(JI,1).EQ.IAFFLUSECTN(IMAPIND(JI,IJB))) THEN + IF (IMAPLVL_ll(JI,1).LT.ICLUSTERLVL(IMAPIND(JI,IJB))) THEN + ! if halo pixel belongs to a cluster with a lower base + ICLUSTERIDT(IMAPIND(JI,IJB)) = IMAPIDT_ll(JI,1) + ICLUSTERLVL(IMAPIND(JI,IJB)) = IMAPLVL_ll(JI,1) + ICHANGES = ICHANGES + 1 + ELSEIF (IMAPLVL_ll(JI,1).EQ.ICLUSTERLVL(IMAPIND(JI,IJB))) THEN + IF (IMAPIDT_ll(JI,1).LT.ICLUSTERIDT(IMAPIND(JI,IJB))) THEN + ! if halo pixel belongs to a cluster with a lower ID number + ICLUSTERIDT(IMAPIND(JI,IJB)) = IMAPIDT_ll(JI,1) + ICHANGES = ICHANGES + 1 + END IF + END IF + END IF +END IF +! +IF ( (IMAPIDT_ll(JI,IJE).NE.0).AND.(IMAPIDT_ll(JI,IJU).NE.0) ) THEN + IF ( IMAPAFL_ll(JI,IJU).GT.IAFFLUSECTN(IMAPIND(JI,IJE)) ) THEN + ! if halo pixel has a bigger affluent than current pixel + ICLUSTERIDT(IMAPIND(JI,IJE)) = IMAPIDT_ll(JI,IJU) + ICLUSTERLVL(IMAPIND(JI,IJE)) = IMAPLVL_ll(JI,IJU) + IAFFLUSECTN(IMAPIND(JI,IJE)) = IMAPAFL_ll(JI,IJU) + ICHANGES = ICHANGES + 1 + ELSEIF (IMAPAFL_ll(JI,IJU).EQ.IAFFLUSECTN(IMAPIND(JI,IJE))) THEN + IF (IMAPLVL_ll(JI,IJU).LT.ICLUSTERLVL(IMAPIND(JI,IJE))) THEN + ! if halo pixel belongs to a cluster with a lower base + ICLUSTERIDT(IMAPIND(JI,IJE)) = IMAPIDT_ll(JI,IJU) + ICLUSTERLVL(IMAPIND(JI,IJE)) = IMAPLVL_ll(JI,IJU) + ICHANGES = ICHANGES + 1 + ELSEIF (IMAPLVL_ll(JI,IJU).EQ.ICLUSTERLVL(IMAPIND(JI,IJE))) THEN + IF (IMAPIDT_ll(JI,IJU).LT.ICLUSTERIDT(IMAPIND(JI,IJE))) THEN + ! if halo pixel belongs to a cluster with a lower ID number + ICLUSTERIDT(IMAPIND(JI,IJE)) = IMAPIDT_ll(JI,IJU) + ICHANGES = ICHANGES + 1 + END IF + END IF + END IF +END IF +! +END DO ! all 4 borders checked +! +IF (ICHANGES.GT.0) THEN ! in case of changes due to halo, rename clusters + DO JJ=IJB,IJE + DO JI=IIB,IIE + IF (OCLOUD(JI,JJ,JK)) THEN + IMAPIDT_ll(JI,JJ) = ICLUSTERIDT(IMAPIND(JI,JJ)) + IMAPLVL_ll(JI,JJ) = ICLUSTERLVL(IMAPIND(JI,JJ)) + IMAPAFL_ll(JI,JJ) = IAFFLUSECTN(IMAPIND(JI,JJ)) + ELSE + IMAPIDT_ll(JI,JJ) = 0 + IMAPLVL_ll(JI,JJ) = IKE+1 + IMAPAFL_ll(JI,JJ) = 0 + END IF + END DO + END DO +END IF +! +CALL REDUCESUM_ll(ICHANGES,IINFO_ll) +IF (ICHANGES.EQ.0) EXIT +! +END DO ! infinite loop until there is no more changes anywhere +! +KCLUSTERIDT(:,:,JK) = IMAPIDT_ll +KCLUSTERLVL(:,:,JK) = IMAPLVL_ll +! +! +CALL CLEANLIST_ll(TZFIELDS_ll) +DEALLOCATE(IAFFLUSECTN) +! +!------------------------------------------------------------------------------- +! +!* 6. COMPUTE GLOBAL CLUSTER SECTIONS AND FIELD AVERAGES +! -------------------------------------------------- +! +! (MPI COMMUNICATIONS FOR 1D TABLES +! useful for computing properties distributions +! within cluster population) +! +!PRINT *,IRANK,'measuring' +! +!* 6.1 SUM UP LOCAL CLUSTER INFO IN 1D TABLES +!PRINT *,IRANK,'build non-redundant 1D table' +! +ICPR = COUNT(ICLUSTERIDT.NE.0) ! number of cluster parts in the local domain, may be redundant +! +ALLOCATE(ILOCLISTIDT(ICPR)) ! list of cluster IDs in local domain +ALLOCATE(ILOCLISTLVL(ICPR)) ! list of cluster baselevel in local domain +ALLOCATE(ILOCLISTSEC(ICPR)) ! list of cluster sections in local domain +ALLOCATE(ZLOCLISTFLD(ICPR)) ! list of cluster field sum in local domain +ALLOCATE(ILOCIND(ICLUSMAX)) ! pointer from old indices (1..iclusmax) toward new non-redundant tables +ILOCLISTIDT = 0 +ILOCLISTLVL = 0 +ILOCLISTSEC = 0 +ZLOCLISTFLD = 0. +ILOCIND = 0 +! +! create ILOCLIST... and ILOCLISTSEC from ICLUSTER... and ICLUSIZE to get rid of all 0 elements +! and to sum the sizes of clusters appearing in different parts inside the local domain +! the sum of field values inside each cluster is continuously updated +! +ICPT=0 ! counter to know last index of ILOCLIST... that has been filled +IF (ICPR.GT.0) THEN +DO JC=1,ICLUSMAX + IF (ICLUSTERIDT(JC).NE.0) THEN + JJ=1 ! to check whether ILOCLIST... already contains ICLUSTER...(JC) + DO + IF (ILOCLISTIDT(JJ).EQ.ICLUSTERIDT(JC)) THEN + IF (ILOCLISTLVL(JJ).EQ.ICLUSTERLVL(JC)) THEN + ZLOCLISTFLD(JJ) = ZLOCLISTFLD(JJ) + ZCLUSSUMFLD(JC) + ILOCLISTSEC(JJ) = ILOCLISTSEC(JJ) + ICLUSIZE(JC) + ILOCIND(JC) = JJ + EXIT + END IF + END IF + IF (JJ.EQ.ICPT+1) THEN + ILOCLISTIDT(JJ) = ICLUSTERIDT(JC) + ILOCLISTLVL(JJ) = ICLUSTERLVL(JC) + ZLOCLISTFLD(JJ) = ZCLUSSUMFLD(JC) + ILOCLISTSEC(JJ) = ICLUSIZE(JC) + ILOCIND(JC) = JJ + ICPT = ICPT+1 + EXIT + END IF + JJ = JJ+1 + END DO + END IF +END DO +END IF +! +DEALLOCATE(ZCLUSSUMFLD) +DEALLOCATE(ICLUSIZE) +DEALLOCATE(ICLUSTERIDT) +DEALLOCATE(ICLUSTERLVL) +! +! normally, ICPT = COUNT(ILOCLISTID.NE.0) +! number of cluster IDs in local domain, *non-redundant* +! +ALLOCATE(ILOCLISTIDT2(ICPT)) +ALLOCATE(ILOCLISTLVL2(ICPT)) +ALLOCATE(ILOCLISTSEC2(ICPT)) +ALLOCATE(ZLOCLISTFLD2(ICPT)) +ILOCLISTIDT2 = ILOCLISTIDT(1:ICPT) ! JUST TO REMOVE ALL 0 AT THE END +ILOCLISTLVL2 = ILOCLISTLVL(1:ICPT) +ILOCLISTSEC2 = ILOCLISTSEC(1:ICPT) +ZLOCLISTFLD2 = ZLOCLISTFLD(1:ICPT) ! /ILOCLISTSEC(1:ICPT) ! from sum compute average +DEALLOCATE(ILOCLISTIDT) +DEALLOCATE(ILOCLISTLVL) +DEALLOCATE(ILOCLISTSEC) +DEALLOCATE(ZLOCLISTFLD) +! +!* 6.2 CONCATENATE ALL LOCAL INFORMATION +!PRINT *,IRANK,'concatenate non-redundant 1D table' +! +ALLOCATE(ICLUSNBR(NPROC)) ! number of clusters in each proc's local domain +CALL MPI_ALLGATHER(ICPT, 1, MPI_INTEGER, ICLUSNBR, 1, MPI_INTEGER, NMNH_COMM_WORLD, INFO) +! each processor knows now how many clusters appear in all other processor domains +! +!PRINT *,IRANK,'build IPROCDPL' +ALLOCATE(IPROCDPL(NPROC)) +IPROCDPL(1)=0 +IF (NPROC.GT.1) THEN +DO JC=2,NPROC + IPROCDPL(JC)=IPROCDPL(JC-1)+ICLUSNBR(JC-1) +END DO +END IF +! IPROCDPL(JC) = nbr of clusters contained by processors whose rank < JC (i.e. cumulative) +! +ITOTNBR = SUM(ICLUSNBR) ! number of clusters ! with redundance from one local domain to the other ! +! +! +ALLOCATE(IGLBLISTIDT(ITOTNBR)) ! concatenated list of cluster IDs of all local domains +ALLOCATE(IGLBLISTLVL(ITOTNBR)) ! concatenated list of cluster baselevels of all local domains +ALLOCATE(IGLBLISTSEC(ITOTNBR)) ! their corresponding LOCAL sections !! +ALLOCATE(ZGLBLISTFLD(ITOTNBR)) ! their corresponding LOCAL field sum +IGLBLISTIDT=0 +IGLBLISTLVL=0 +IGLBLISTSEC=0 +ZGLBLISTFLD=0. +! +!PRINT *,IRANK,'call all-gatherv' +CALL MPI_ALLGATHERV(ILOCLISTIDT2, ICPT, MPI_INTEGER, IGLBLISTIDT, ICLUSNBR, IPROCDPL, MPI_INTEGER, & + NMNH_COMM_WORLD, INFO) +CALL MPI_ALLGATHERV(ILOCLISTLVL2, ICPT, MPI_INTEGER, IGLBLISTLVL, ICLUSNBR, IPROCDPL, MPI_INTEGER, & + NMNH_COMM_WORLD, INFO) +CALL MPI_ALLGATHERV(ILOCLISTSEC2, ICPT, MPI_INTEGER, IGLBLISTSEC, ICLUSNBR, IPROCDPL, MPI_INTEGER, & + NMNH_COMM_WORLD, INFO) +CALL MPI_ALLGATHERV(ZLOCLISTFLD2, ICPT, MPI_PRECISION, ZGLBLISTFLD, ICLUSNBR, IPROCDPL, MPI_PRECISION, & + NMNH_COMM_WORLD, INFO) +! +!* 6.3 EACH PROC COMPUTES GLOBAL SECTIONS AND FIELD AVERAGE OF ITS CLUSTERS +!PRINT *,IRANK,'global sizes of its own clusters' +! +! update cluster sections in ILOCLISTSEC2 in order to have GLOBAL sections +DO JC=1,ICPT + DO JJ=1,ITOTNBR + ! TO BE SURE NOT TO SUM ITS OWN SIZE TO THE OTHER DOMAIN PART SIZES + IF ( (JJ.LE.IPROCDPL(IRANK+1)) .OR. (JJ.GT.(IPROCDPL(IRANK+1)+ICLUSNBR(IRANK+1))) ) THEN + IF ( (IGLBLISTIDT(JJ).EQ.ILOCLISTIDT2(JC)) .AND. (IGLBLISTLVL(JJ).EQ.ILOCLISTLVL2(JC)) ) THEN + ! if IGLBLIST...(JJ) correspond to ILOCLIST...2(JC) then sum sections and field avg + ILOCLISTSEC2(JC)=ILOCLISTSEC2(JC)+IGLBLISTSEC(JJ) + ZLOCLISTFLD2(JC)=ZLOCLISTFLD2(JC)+ZGLBLISTFLD(JJ) + END IF + END IF + END DO +END DO +! +DO JC=1,ICPT + ZLOCLISTFLD2(JC)=ZLOCLISTFLD2(JC)/ILOCLISTSEC2(JC) ! from sum toward average +END DO +! +!PRINT *,'ILOCLISTID2 = ',ILOCLISTID2 +!PRINT *,'ILOCLISTSZ2 = ',ILOCLISTSZ2 +! +! since now, ILOCLISTSEC2 contains global sizes of local clusters +! and, ZLOCLISTFLD2 contains global field avg of local clusters +! +!* 6.4 FILL PCLUSTERSEC TABLE WITH GLOBAL SIZES +!PRINT *,IRANK,'filling PCLUSTERSEC' + ! + PCLUSTERSEC(:,:,JK) = 0. ! pixels first belong to no clusters, so clustersize is zero + ZMESHAREA = XDXHATM*XDYHATM/10**6 ! in km2 + DO JJ=IJB,IJE + DO JI=IIB,IIE + IF (OCLOUD(JI,JJ,JK)) THEN + PCLUSTERSEC(JI,JJ,JK) = ILOCLISTSEC2(ILOCIND(IMAPIND(JI,JJ)))*ZMESHAREA + END IF + END DO + END DO +! +!* DEALLOCATIONS +! +DEALLOCATE(ILOCLISTIDT2) +DEALLOCATE(ILOCLISTLVL2) +DEALLOCATE(ILOCLISTSEC2) +DEALLOCATE(ZLOCLISTFLD2) +! +DEALLOCATE(ILOCIND,IMAPIND) +DEALLOCATE(ICLUSNBR,IPROCDPL) +DEALLOCATE(IGLBLISTIDT,IGLBLISTLVL,IGLBLISTSEC,ZGLBLISTFLD) +! +END DO ! JK +! +! +END SUBROUTINE CLUSTERING diff --git a/src/MNH/diag.f90 b/src/MNH/diag.f90 index 55b9599aa4d43cbcb0ae9aca5467527a8370383e..7f386161f88e17a1cc72833f504571c63513c9fc 100644 --- a/src/MNH/diag.f90 +++ b/src/MNH/diag.f90 @@ -79,6 +79,7 @@ !! 10/2016 (C.LAC) add LVISI !! 10/2016 (F Brosse) Add prod/loss terms computation for chemistry !! 10/2017 (G.Delautier) New boundary layer height : replace LBLTOP by CBLTOP +!! 10/2017 (T Dauhut) Add parallel 3D clustering !------------------------------------------------------------------------------- ! !* 0. DECLARATIONS @@ -228,7 +229,8 @@ NAMELIST/NAM_DIAG/ CISO, LVAR_RS, LVAR_LS, & XGRID,NBELEV,XELEV,NBRAD,LQUAD,LFALL,LWBSCS,LWREFL,& XREFLMIN,XREFLVDOPMIN,LSNRT,XSNRMIN,& LLIDAR,CVIEW_LIDAR,XALT_LIDAR,XWVL_LIDAR,& - LISOPR,XISOPR,LISOTH,XISOTH, LHU_FLX,LVISI, LLIMA_DIAG + LISOPR,XISOPR,LISOTH,XISOTH, LHU_FLX,LVISI, LLIMA_DIAG,& + LCLSTR,LBOTUP,CFIELD,XTHRES ! NAMELIST/NAM_DIAG_FILE/ YINIFILE,YINIFILEPGD, YSUFFIX NAMELIST/NAM_STO_FILE/ CFILES, NSTART_SUPP @@ -316,6 +318,11 @@ XLAT_GPS(:)=XUNDEF XLON_GPS(:)=XUNDEF XZS_GPS(:)=-999.0 XDIFFORO=150.0 +! +LCLSTR=.FALSE. +LBOTUP=.TRUE. +CFIELD='CLOUD' +XTHRES=0.00001 !! initialization of radar parameters NVERSION_RAD=1 diff --git a/src/MNH/modd_diag_flag.f90 b/src/MNH/modd_diag_flag.f90 index fda235d7a4d8fdeb005dde66660cd6a9c66ce8ff..9b7fdc8a1285d9cf25d2fa3a62e28a05f002c54b 100644 --- a/src/MNH/modd_diag_flag.f90 +++ b/src/MNH/modd_diag_flag.f90 @@ -40,6 +40,7 @@ !! C.Lac 10/2016 Add visibility diagnostic !! 10/2016 (F Brosse) Add prod/loss terms computation for chemistry !! 10/2017 (G.Delautier) New boundary layer height : replace LBLTOP by CBLTOP +!! T. Dauhut 10/2017 add parallel 3D clustering !! !------------------------------------------------------------------------------- ! @@ -133,5 +134,10 @@ LOGICAL :: LISOPR ! flag to write on isobaric level REAL,DIMENSION(10):: XISOPR ! list of level for isobaric interpolation LOGICAL :: LISOTH ! flag to write on isentropic level REAL,DIMENSION(10):: XISOTH ! list of level for isentropic interpolation +! +LOGICAL :: LCLSTR ! flag for 3D clustering +LOGICAL :: LBOTUP ! to propagate clustering from bottom to top, otherwise top to bottom +CHARACTER(LEN=8) :: CFIELD ! field on which clustering is applied, could be 'W' or 'CLOUD' +REAL :: XTHRES ! threshold value to detect the 3D structures END MODULE MODD_DIAG_FLAG diff --git a/src/MNH/write_lfifm1_for_diag.f90 b/src/MNH/write_lfifm1_for_diag.f90 index 284def011b7da7cf07d1c95894fe6a5d5dddf4b8..2acd1d1c609ca6bd5115ebff7f30a77318b8678d 100644 --- a/src/MNH/write_lfifm1_for_diag.f90 +++ b/src/MNH/write_lfifm1_for_diag.f90 @@ -142,6 +142,7 @@ END MODULE MODI_WRITE_LFIFM1_FOR_DIAG !! Modification 01/2016 (JP Pinty) Add LIMA !! C.Lac 04/2016 : add visibility and droplet deposition !! 10/2017 (G.Delautier) New boundary layer height : replace LBLTOP by CBLTOP +!! T.Dauhut 10/2017 : add parallel 3D clustering !------------------------------------------------------------------------------- ! !* 0. DECLARATIONS @@ -234,6 +235,7 @@ USE MODE_IO_ll USE MODE_THERMO USE MODE_MODELN_HANDLER USE MODI_LIDAR +USE MODI_CLUSTERING ! USE MODD_MPIF USE MODD_VAR_ll @@ -300,6 +302,12 @@ REAL,DIMENSION(SIZE(XSVT,1),SIZE(XSVT,2),SIZE(XSVT,3),NMODE_SLT*2):: ZSSLTDEP REAL,DIMENSION(:,:,:,:), ALLOCATABLE :: ZSIG_DST, ZRG_DST, ZN0_DST REAL,DIMENSION(:,:,:,:), ALLOCATABLE :: ZSIG_SLT, ZRG_SLT, ZN0_SLT REAL,DIMENSION(:,:,:), ALLOCATABLE :: ZRHOT, ZTMP ! work array +! +! GBOTUP = True does clustering from bottom up to top, False top down to surface +LOGICAL :: GBOTUP ! clustering propagation +LOGICAL,DIMENSION(SIZE(XTHT,1),SIZE(XTHT,2),SIZE(XTHT,3)) :: GCLOUD ! mask +INTEGER,DIMENSION(SIZE(XTHT,1),SIZE(XTHT,2),SIZE(XTHT,3)) :: ICLUSTERID, ICLUSTERLV +REAL, DIMENSION(SIZE(XTHT,1),SIZE(XTHT,2),SIZE(XTHT,3)) :: ZCLUSTERID, ZCLUSTERLV, ZCLDSIZE !ECRITURE DANS UN FICHIER ASCII DE RESULTATS !INITIALISATION DU NOM DE FICHIER CREE EN PARALLELE AVEC CELUI LFI @@ -313,6 +321,7 @@ CHARACTER(LEN=3) :: YGRID_SIZE INTEGER :: IEL,IIELV CHARACTER(LEN=5) :: YVIEW ! Upward or Downward integration INTEGER :: IACCMODE +! !------------------------------------------------------------------------------- INTEGER :: IAUX ! work variable REAL,DIMENSION(SIZE(XTHT,1),SIZE(XTHT,2),SIZE(XTHT,3)) :: ZWORK35,ZWORK36 @@ -2521,6 +2530,45 @@ ENDIF ! !------------------------------------------------------------------------------- ! +!* Clustering +! +IF (LCLSTR) THEN + GCLOUD(:,:,:)=.FALSE. + GBOTUP=LBOTUP + IF (CFIELD=='W') THEN + WHERE(XWT(:,:,:).GT.XTHRES) GCLOUD(:,:,:)=.TRUE. + END IF + IF (CFIELD=='CLOUD') THEN + WHERE((XRT(:,:,:,2)+XRT(:,:,:,4)+XRT(:,:,:,5)+XRT(:,:,:,6)).GT.XTHRES) GCLOUD(:,:,:)=.TRUE. + END IF + PRINT *,'CALL CLUSTERING COUNT(GCLOUD)=',COUNT(GCLOUD) + CALL CLUSTERING(GBOTUP,GCLOUD,XWT,ICLUSTERID,ICLUSTERLV,ZCLDSIZE) + PRINT *,'GOT OUT OF CLUSTERING' + ! + ZCLUSTERID=ICLUSTERID + ZCLUSTERLV=ICLUSTERLV + ! + YRECFM='CLUSTERID' + YCOMMENT='X_Y_Z_CLUSTER (ID NUMBER)' + IGRID=1 + ILENCH=LEN(YCOMMENT) + CALL FMWRIT(HFMFILE,YRECFM,CLUOUT,'XY',ZCLUSTERID,IGRID,ILENCH,YCOMMENT,IRESP) + ! + YRECFM='CLUSTERLV' + YCOMMENT='X_Y_Z_CLUSTER (BASE OR TOP LEVEL)' + IGRID=1 + ILENCH=LEN(YCOMMENT) + CALL FMWRIT(HFMFILE,YRECFM,CLUOUT,'XY',ZCLUSTERLV,IGRID,ILENCH,YCOMMENT,IRESP) + ! + YRECFM='CLDSIZE' + YCOMMENT='X_Y_Z_CLDSIZE (HOR. SECTION)' + IGRID=1 + ILENCH=LEN(YCOMMENT) + CALL FMWRIT(HFMFILE,YRECFM,CLUOUT,'XY',ZCLDSIZE,IGRID,ILENCH,YCOMMENT,IRESP) +END IF +! +!------------------------------------------------------------------------------- +! !* Geostrophic and Ageostrophic wind (m/s) ! IF (LGEO .OR. LAGEO) THEN