SUBROUTINE DRCOEF(CDM,CDH,RIB,CFLUX,QG,QA,ZOMIN,ZOHIN, 5
     1                  CRIB,TVIRTG,TVIRTA,VA,ZOM,ZOH,FI,ITER,
     2                  ILG,IL1,IL2)
C
C     * AUG 19/04 - Y.DELAGE.   REGROUP COMMON BLOCKS
C     *                         MAKE DECLARATIONS EXPLICIT
C     * SEP 10/02 - K.ABDELLA.  BUGFIX IN CALCULATION OF "OLS" (2 PLACES).
C     * MAY 22/02 - N.MCFARLANE.USE THE ASYMPTOTIC VALUE FOR CDH 
C     *                         GENERALLY TO LIMIT FLUXES AND PREVENT
C     *                         OVERSHOOT. (COMMMENTED OUT FOR
C     *                         OFF-LINE RUNS.)
C     * APR 11/01 - M.LAZARE.   SHORTENED "CLASS2" COMMON BLOCK.
C     * OCT 26/99 - E. CHAN.    COMMENT OUT ARTIFICIAL DAMPING OF 
C     *                         TURBULENT FLUXES FOR STAND-ALONE TESTING.
C     * JUL 18/97 - M. LAZARE,  CLASS 2.7. PASS IN ADDITIONAL WORK ARRAYS
C     *             D.VERSEGHY. ZOMIN AND ZOHIN TO USE INTERNALLY,
C     *                         SO THAT INPUT ZOMIN AND ZOHIN DO NOT
C     *                         CHANGE WHEN PASSED BACK TO THE
C     *                         ITERATION (BUG FROM PREVIOUS
C     *                         VERSIONS). PREVIOUS ZOM AND ZOH
C     *                         BECOME WORK ARRAYS. 
C     * MAR 10/97 - M. LAZARE.  PASS IN QG AND QA AND USE TO ONLY
C     *                         DAMP TURBULENT FLUXES USING CDHMOD
C     *                         UNDER STABLE CONDITIONS IF MOISTURE
C     *                         FLUX IS UPWARDS. ALSO, BETTER 
C     *                         DEFINITION OF SURFACE LAYER TOP
C     *                         FROM K.ABDELLA.
C     * MAY 21/96 - K. ABDELLA. MODIFICATION FOR FREE-CONVECTIVE
C     *                         LIMIT ON UNSTABLE SIDE ADDED.
C     * JAN 10/96 - K. ABDELLA. CORRECT ERROR IN AU1 (UNSTABLE
C     *                         SIDE) AND PUT IN PRANDTL NUMBER
C     *                         RANGE (0.74->1)/
C     *                         "CDHMOD" USED ON BOTH STABLE AND
C     *                         UNSTABLE SIDE, TO LIMIT FLUXES
C     *                         OVER LONG TIMESTEP.     
C     * M. LAZARE - FEB 14/95.  USE VARIABLES "ZOLN" AND "ZMLN"
C     *                         ON UNSTABLE SIDE, FOR OPTIMIZATION.
C     *                         THIS IS PREVIOUS VERSION "DRCOEFX".
C     * K. ABDELLA/M. LAZARE. - NOV 30/94.

C     * CALCULATES DRAG COEFFICIENTS AND RELATED VARIABLES FOR CLASS.

C     * OUTPUT FIELDS ARE:
C     *    CDM    : STABILITY-DEPENDENT DRAG COEFFICIENT FOR MOMENTUM.
C     *    CDH    : STABILITY-DEPENDENT DRAG COEFFICIENT FOR HEAT.
C     *    RIB    : BULK RICHARDSON NUMBER.
C     *    CFLUX  : CD * MOD(V), BOUNDED BY FREE-CONVECTIVE LIMIT.

C     * INPUT FIELDS ARE:
C     *    ZOMIN/: ROUGHNESS HEIGHTS FOR MOMENTUM/HEAT NORMALIZED BY
C     *    ZOHIN   REFERENCE HEIGHT.
C     *    CRIB   : -RGAS*SLTHKEF/(VA**2), WHERE
C     *             SLTHKEF=-LOG(MAX(SGJ(ILEV),SHJ(ILEV)))
C     *    TVIRTG : "SURFACE" VIRTUAL TEMPERATURE.
C     *    TVIRTA : LOWEST LEVEL VIRTUAL TEMPERATURE.
C     *    VA     : AMPLITUDE OF LOWEST LEVEL WIND.
C     *    FI     : FRACTION OF SURFACE TYPE BEING STUDIED.
C     *    QG     : SATURATION SPECIFIC HUMIDITY AT GROUND TEMPERATURE.
C     *    QA     : LOWEST LEVEL SPECIFIC HUMIDITY.
C     *    ITER   : INDEX ARRAY INDICATING IF POINT IS UNDERGOING
C     *             FURTHER ITERATION OR NOT. 

C     *    ZOM/  : WORK ARRAYS USED FOR SCALING ZOMIN/ZOHIN
C     *    ZOH     ON STABLE SIDE, AS PART OF CALCULATION.

      IMPLICIT NONE
      INTEGER ILG,IL1,IL2,I
      REAL ZLEV,ZS,ZOLN,ZMLN,CPR,ZI,OLFACT,OLSF,ZL,ZMOL,
     1     XM,XH,BH1,BH2,BH,WB,WSTAR,RIB0,WSPEED,AU1,OLS,PSIM1,PSIM0,
     2     PSIH1,PSIH0,USTAR,TSTAR,WTS,AS1,AS2,AS3,ZTIK,CDFAC,
     3     CLIMIT,AA,AA1,BET,PR,ZHOL
C
C     * OUTPUT ARRAYS.

      REAL CDM    (ILG),   CDH    (ILG),   RIB    (ILG),   CFLUX  (ILG) 

C     * INPUT ARRAYS.

      REAL ZOMIN  (ILG),   ZOHIN  (ILG),   CRIB   (ILG),   TVIRTG (ILG)
      REAL TVIRTA (ILG),   VA     (ILG),   FI     (ILG)
      REAL QG     (ILG),   QA     (ILG)

      INTEGER ITER(ILG)

C     * WORK ARRAYS.

      REAL ZOM    (ILG),   ZOH    (ILG)

#include "class_com.cdk"
C-------------------------------------------------------------
      AA=9.5285714
      AA1=14.285714
      BET=1.2
      PR = 1.
C      
      DO 100 I=IL1,IL2
        IF(FI(I).GT.0. .AND. ITER(I).EQ.1)                         THEN
          RIB(I)=CRIB(I)*(TVIRTG(I)-TVIRTA(I))
          IF(RIB(I).GE.0.0) THEN
            ZLEV=-CRIB(I)*TVIRTA(I)*(VA(I)**2)/G  
            ZS=MAX(10.,5.*MAX(ZOMIN(I)*ZLEV, ZOHIN(I)*ZLEV))
            ZS=ZLEV*(1.+RIB(I))/(1.+(ZLEV/ZS)*RIB(I))
            ZOM(I)=ZOMIN(I)*ZLEV/ZS
            ZOH(I)=ZOHIN(I)*ZLEV/ZS
            RIB(I)=RIB(I)*ZS/ZLEV
          ELSE
            ZOM(I)=ZOMIN(I)
            ZOH(I)=ZOHIN(I)
          ENDIF
          ZOLN=LOG(ZOH(I))
          ZMLN=LOG(ZOM(I))
          IF(RIB(I).LT.0.0) THEN
            CPR=MAX(ZOLN/ZMLN,0.74)
            CPR=MIN(CPR,1.0)
            ZI=1000.0
            OLSF=BET**3*ZI*VKC**2/ZMLN**3
            OLFACT=1.7*(LOG(1.+ZOM(I)/ZOH(I)))**0.5+0.9
            OLSF=OLSF*OLFACT
            ZL = -CRIB(I)*TVIRTA(I)*(VA(I)**2)/G
            ZMOL=ZOM(I)*ZL/OLSF
            ZHOL=ZOH(I)*ZL/OLSF
            XM=(1.00-15.0*ZMOL)**(0.250)
            XH=(1.00-9.0*ZHOL)**0.25
            BH1=-LOG(-2.41*ZMOL)+LOG(((1.+XM)/2.)**2*(1.+XM**2)/2.)
            BH1=BH1-2.*ATAN(XM)+ATAN(1.)*2.
            BH1=BH1**1.5
            BH2=-LOG(-0.25*ZHOL)+2.*LOG(((1.00+XH**2)/2.00))
            BH=VKC**3.*BET**1.5/(BH1*(BH2)**1.5)
            WB=SQRT(G*(TVIRTG(I)-TVIRTA(I))*ZI/TVIRTG(I))
            WSTAR=BH**(0.333333)*WB
            RIB0=RIB(I)

            WSPEED=SQRT(VA(I)**2+(BET*WSTAR)**2)
            RIB(I)=RIB0*VA(I)**2/WSPEED**2
            AU1=1.+5.0*(ZOLN-ZMLN)*RIB(I)*(ZOH(I)/ZOM(I))**0.25
            OLS=-RIB(I)*ZMLN**2/(CPR*ZOLN)*(1.0+AU1/ 
     1           (1.0-RIB(I)/(ZOM(I)*ZOH(I))**0.25))
            PSIM1=LOG(((1.00+(1.00-15.0*OLS)**0.250)/2.00)**2*
     1            (1.0+(1.00-15.0*OLS)**0.5)/2.0)-2.0*ATAN(
     2            (1.00-15.0*OLS)**0.250)+ATAN(1.00)*2.00
            PSIM0=LOG(((1.00+(1.00-15.0*OLS*ZOM(I))**0.250)/2.00)**2
     1            *(1.0+(1.00-15.0*OLS*ZOM(I))**0.5)/2.0)-2.0*
     2            ATAN((1.00-15.0*OLS*ZOM(I))**0.250)+ATAN(1.00)*2.0
            PSIH1=LOG(((1.00+(1.00-9.0*OLS)**0.50)/2.00)**2)
            PSIH0=LOG(((1.00+(1.00-9.0*OLS*ZOH(I))**0.50)/2.00)**2)

            USTAR=VKC/(-ZMLN-PSIM1+PSIM0)
            TSTAR=VKC/(-ZOLN-PSIH1+PSIH0)
            CDH(I)=USTAR*TSTAR/PR
            WTS=CDH(I)*WSPEED*(TVIRTG(I)-TVIRTA(I))
            WSTAR=(G*ZI/TVIRTG(I)*WTS)**(0.333333)

            WSPEED=SQRT(VA(I)**2+(BET*WSTAR)**2)
            RIB(I)=RIB0*VA(I)**2/WSPEED**2
            AU1=1.+5.0*(ZOLN-ZMLN)*RIB(I)*(ZOH(I)/ZOM(I))**0.25
            OLS=-RIB(I)*ZMLN**2/(CPR*ZOLN)*(1.0+AU1/ 
     1           (1.0-RIB(I)/(ZOM(I)*ZOH(I))**0.25))
            PSIM1=LOG(((1.00+(1.00-15.0*OLS)**0.250)/2.00)**2*
     1            (1.0+(1.00-15.0*OLS)**0.5)/2.0)-2.0*ATAN(
     2            (1.00-15.0*OLS)**0.250)+ATAN(1.00)*2.00
            PSIM0=LOG(((1.00+(1.00-15.0*OLS*ZOM(I))**0.250)/2.00)**2
     1            *(1.0+(1.00-15.0*OLS*ZOM(I))**0.5)/2.0)-2.0*
     2            ATAN((1.00-15.0*OLS*ZOM(I))**0.250)+ATAN(1.00)*2.0
            PSIH1=LOG(((1.00+(1.00-9.0*OLS)**0.50)/2.00)**2)
            PSIH0=LOG(((1.00+(1.00-9.0*OLS*ZOH(I))**0.50)/2.00)**2)
 
          ELSE

            WSPEED=VA(I)
            AS1=10.0*ZMLN*(ZOM(I)-1.0)
            AS2=5.00/(2.0-8.53*RIB(I)*EXP(-3.35*RIB(I))+0.05*RIB(I)**2)
C<<<
            AS2=AS2*PR*SQRT(-ZMLN)/2.
            AS3=27./(8.*PR*PR)
C>>>
            OLS=RIB(I)*(ZMLN**2+AS3*AS1*(RIB(I)**2+AS2*RIB(I)))
     1          /(AS1*RIB(I)-PR*ZOLN)
            PSIM1=-0.667*(OLS-AA1)*EXP(-0.35*OLS)-AA-OLS
            PSIM0=-0.667*(OLS*ZOM(I)-AA1)*EXP(-0.35*OLS*ZOM(I))
     1            -AA-OLS*ZOM(I)
            PSIH1=-(1.0+2.0*OLS/3.0)**1.5-0.667*(OLS-AA1)
     1            *EXP(-0.35*OLS)-AA+1.0
            PSIH0=-(1.0+2.0*OLS*ZOH(I)/3.0)**1.5-0.667*(OLS*ZOH(I)-AA1)
     1            *EXP(-0.35*OLS*ZOH(I))-AA+1.0

          ENDIF  
      
          USTAR=VKC/(-ZMLN-PSIM1+PSIM0)
          TSTAR=VKC/(-ZOLN-PSIH1+PSIH0)

          CDM(I)=USTAR**2.0
          CDH(I)=USTAR*TSTAR/PR    
          
          ZTIK=-CRIB(I)*TVIRTA(I)*(WSPEED**2)/G
          CDFAC=2.0*DELT*ABS(WSPEED/ZTIK)
C
C         * USE THE ASYMPTOTIC VALUE FOR CDH TO LIMIT FLUXES AND PREVENT
C         * OVERSHOOT.
C
C          CDH(I)=MIN(CDH(I),1./CDFAC)
C
C         * CALCULATE CD*MOD(V) UNDER FREE-CONVECTIVE LIMIT.
C
          IF(TVIRTG(I).GT.TVIRTA(I))    THEN
            CLIMIT=1.9E-3*(TVIRTG(I)-TVIRTA(I))**0.333333 
          ELSE
            CLIMIT=0.          
          ENDIF             
          CFLUX(I)=MAX(CDH(I)*WSPEED,CLIMIT)
        ENDIF
 100  CONTINUE  
C-------------------------------------------------------------
      RETURN
      END