Ice Sheet System Model  4.18
Code documentation
PddSurfaceMassBalance.cpp
Go to the documentation of this file.
1 /* file: PddSurfaceMassBlance.cpp
2  Calculating the surface mass balance using the positive degree day method.
3  Updating the precipitation and temperature to the new elevation
4  */
5 
6 #include "../io/io.h"
7 #include "./elements.h"
8 #include "../Numerics/numerics.h"
9 #include <cmath>
10 
11 IssmDouble PddSurfaceMassBalance(IssmDouble* monthlytemperatures, IssmDouble* monthlyprec,
12  IssmDouble* pdds, IssmDouble* pds, IssmDouble* melt, IssmDouble* accu,
13  IssmDouble signorm, IssmDouble yts, IssmDouble h, IssmDouble s, IssmDouble desfac,
14  IssmDouble s0t,IssmDouble s0p, IssmDouble rlaps,IssmDouble rlapslgm,
15  IssmDouble TdiffTime,IssmDouble sealevTime, IssmDouble pddsnowfac,IssmDouble pddicefac,
16  IssmDouble rho_water,IssmDouble rho_ice){
17 
18  // output:
19  IssmDouble B; // surface mass balance, melt+accumulation
20  int iqj,imonth;
21 
22  IssmDouble saccu; // yearly surface accumulation
23  IssmDouble smelt; // yearly melt
24  IssmDouble precrunoff; // yearly runoff
25  IssmDouble prect; // total precipitation during 1 year taking into account des. ef.
26  IssmDouble water; //water=rain + snowmelt
27  IssmDouble runoff; //meltwater only, does not include rain
28  IssmDouble sconv; //rhow_rain/rhoi / 12 months
29 
30  //IssmDouble sealev=0.; // degrees per meter. 7.5 lev's 99 paper, 9 Marshall 99 paper
31  //IssmDouble Pfac=0.5,Tdiff=0.5;
32  IssmDouble rtlaps;
33  // IssmDouble lapser=6.5 // lapse rate
34  // IssmDouble desfac = 0.3; // desert elevation factor
35  // IssmDouble s0p=0.; // should be set to elevation from precip source
36  // IssmDouble s0t=0.; // should be set to elevation from temperature source
37  IssmDouble st; // elevation between altitude of the temp record and current altitude
38  IssmDouble sp; // elevation between altitude of the prec record and current altitude
39  IssmDouble deselcut=1.0;
40 
41  // PDD and PD constants and variables
42  IssmDouble siglim; // sigma limit for the integration which is equal to 2.5 sigmanorm
43  IssmDouble signormc = signorm - 0.5; // sigma of the temperature distribution for cloudy day
44  IssmDouble siglimc, siglim0, siglim0c;
45  IssmDouble PDup, pddsig, PDCUT = 2.0; // PDcut: rain/snow cutoff temperature (C)
46  IssmDouble DT = 0.02;
47  IssmDouble pddt, pd; // pd: snow/precip fraction, precipitation falling as snow
48 
49  IssmDouble q, qmpt; // q is desert/elev. fact, hnpfac is huybrect fact, and pd is normal dist.
50  IssmDouble qm = 0.; // snow part of the precipitation
51  IssmDouble qmt = 0.; // precipitation without desertification effect adjustment
52  IssmDouble qmp = 0.; // desertification taken into account
53  IssmDouble pdd = 0.;
54  IssmDouble frzndd = 0.;
55 
56  IssmDouble tstar; // monthly mean surface temp
57  IssmDouble Tsum= 0.; // average summer (JJA) temperature
58  IssmDouble Tsurf = 0.; // average annual temperature
59 
60  IssmDouble deltm=1./12.;
61  int ismon[12]={11,0,1,2,3,4,5,6,7,8,9,10};
62 
63  IssmDouble snwm; // snow that could have been melted in a year.
64  IssmDouble snwmf; // ablation factor for snow per positive degree day.
65  IssmDouble smf; // ablation factor for ice per pdd (Braithwaite 1995 from tarasov 2002).
66 
67  IssmDouble dfrz=1.5, CovrLm=2009./3.35e+5, dCovrLm=dfrz*CovrLm; //m*J kg^-1 C^-1 /(J kg^-1)=m/C yr
68  IssmDouble supice,supcap,diffndd;
69  IssmDouble fsupT=0.5, fsupndd=0.5; // Tsurf mode factors for supice
70  IssmDouble pddtj, hmx2;
71  IssmDouble pddsnowfac0=4.3, pddicefac0=8.3;
72  IssmDouble snowfac, icefac;
73 
74  sconv=(rho_water/rho_ice)/12.; //rhow_rain/rhoi / 12 months
75 
76  /*PDD constant*/
77  siglim = 2.5*signorm;
78  siglimc = 2.5*signormc;
79  siglim0 = siglim/DT + 0.5;
80  siglim0c = siglimc/DT + 0.5;
81  PDup = siglimc+PDCUT;
82 
83  // seasonal loop
84  for (iqj = 0; iqj < 12; iqj++){
85  imonth = ismon[iqj];
86 
87  /*********compute lapse rate ****************/
88  st=(s-s0t)/1000.;
89  rtlaps=TdiffTime*rlapslgm + (1.-TdiffTime)*rlaps; // lapse rate
90 
91  /*********compute Surface temperature *******/
92  monthlytemperatures[imonth]=monthlytemperatures[imonth] - rtlaps *max(st,sealevTime*0.001);
93  tstar = monthlytemperatures[imonth];
94  Tsurf = tstar*deltm+Tsurf;
95 
96  /*********compute PD ****************/
97  if (tstar < PDup){
98  pd = 1.;
99  if (tstar >= -siglimc){ pd = pds[reCast<int,IssmDouble>(tstar/DT + siglim0c)];}}
100  else {
101  pd = 0.;}
102 
103  /******exp des/elev precip reduction*******/
104  sp=(s-s0p)/1000.-deselcut; // deselev effect is wrt chng in topo
105  if (sp>0.0){q = exp(-desfac*sp);}
106  else {q = 1.0;}
107 
108  qmt= qmt + monthlyprec[imonth]*sconv; //*sconv to convert in m of ice equivalent per month
109  qmpt= q*monthlyprec[imonth]*sconv;
110  qmp= qmp + qmpt;
111  qm= qm + qmpt*pd;
112 
113  /*********compute PDD************/
114  // ndd(month)=-(tstar-pdd(month)) since ndd+pdd gives expectation of
115  // gaussian=T_m, so ndd=-(Tsurf-pdd)
116  if (iqj>5 && iqj<9){ Tsum=Tsum+tstar;}
117 
118  if (tstar >= siglim) {pdd = pdd + tstar*deltm;}
119  else if (tstar> -siglim){
120  pddsig=pdds[reCast<int,IssmDouble>(tstar/DT + siglim0)];
121  pdd = pdd + pddsig*deltm;
122  frzndd = frzndd - (tstar-pddsig)*deltm;}
123  else{frzndd = frzndd - tstar*deltm; }
124 
125  /*Assign output pointer*/
126  *(monthlytemperatures+imonth) = monthlytemperatures[imonth];
127  *(monthlyprec+imonth) = monthlyprec[imonth];
128  } // end of seasonal loop
129  //******************************************************************
130 
131  saccu = qm;
132  prect = qmp; // total precipitation during 1 year taking into account des. ef.
133  Tsum=Tsum/3;
134 
135  snowfac=pddsnowfac0;
136  icefac=pddicefac0;
137  if (pddsnowfac>0) {
138  if (pddsnowfac<1.65) {
139  _printf0_("WARNING: Pdd snow factor input, " << pddsnowfac << ", results in a negative value. It will be ignored. \n");
140  }
141  else{
142  snowfac=pddsnowfac;
143  }
144  }
145  if (pddicefac>0) {
146  if (pddicefac>17.22) {
147  _printf0_("WARNING: Pdd ice factor input, " << pddicefac << ", results in a negative value. It will be ignored. \n");
148  }
149  else{
150  icefac=pddicefac;
151  }
152  }
153 
154  /***** determine PDD factors *****/
155  if(Tsum< -1.) {
156  snwmf=(2.65+snowfac-pddsnowfac0)*0.001; // ablation factor for snow per positive degree day.*0.001 to go from mm to m/ppd
157  smf=17.22*0.001; // ablation factor for ice per pdd (Braithwaite 1995 from tarasov 2002)
158  }
159  else if(Tsum< 10){
160  snwmf = (0.15*(Tsum+1) + (2.65+snowfac-pddsnowfac0))*0.001;
161  smf = (((17.22-icefac)/(pow(11,3)))*pow((10.-Tsum),3) + pddicefac0)*0.001;
162  //JC,smf = (((icefac-pddicefac0)/(Tsum+1))*pow((10.-Tsum),3) + pddicefac0)*0.001;
163  }
164  else{
165  snwmf=snowfac*0.001;
166  smf=icefac*0.001;
167  }
168  snwmf=0.95*snwmf;
169  smf=0.95*smf;
170 
171  /***** compute PDD ablation and refreezing *****/
172  pddt = pdd *365;
173  snwm = snwmf*pddt; // snow that could have been melted in a year
174  hmx2 = min(h,dfrz); // refreeze active layer max depth: dfrz
175 
176  if(snwm < saccu) {
177  water=prect-saccu + snwm; //water=rain + snowmelt
178  // l 2.2= capillary factor
179  // Should refreezing be controlled by frzndd or by mean annual Tsurf?
180  // dCovrLm concept is of warming of active layer (thickness =d@=1-
181  // >2m)
182  // problem with water seepage into ice: should be sealed after
183  // refreezing
184  // so everything needs to be predicated on 1 year scale, except for
185  // thermal
186  // conductivity through ice
187  // also, need to account that melt season has low accum, so what's
188  // going to
189  // hold the meltwater around for refreezing? And melt-time will have
190  // low seasonal frzndd
191 
192  // Superimposed ice : Pfeffer et al. 1991, Tarasov 2002
193 
194  supice= min(hmx2*CovrLm*frzndd+2.2*(saccu-snwm), water); // superimposed ice
195  supcap=min(2.2*(saccu-snwm),water);
196  runoff=snwm - supice; //meltwater only, does not include rain
197  }
198  else { //all snow melted
199  supice= min(hmx2*CovrLm*frzndd, prect );
200  runoff= saccu + smf*(pddt-saccu/snwmf) - supice;
201  supcap=0;
202  }
203  // pdd melting doesn't cool Tsurf, so ndd refreeze shouldn't warm it
204  // except pdd melt heat source is atmosphere, while refreeze is
205  // ground/ice stored interim
206  // assume pdd=ndd, then melt should equal refreeze and Tsurf should=0
207  // assume ndd=2*pdd, then all supice is refrozen, but Tsurf should be
208  // <0
209  // assume ndd>pdd, little melt => little supice
210  // bottom line: compare for Tsurf<0 : supice and no supice case,
211  // expect Tsurf difference
212  // except some of cooling flux comes from atmosphere//
213  // 1 dm supice should not raise Tsurf by 1/dCovrLm = 16.675C
214  // does supice make sense when H< 0.1m? then d=thermoactive ice layer ////
215  // < 0.1
216 
217  // make more sense to just use residual pdd-ndd except that pdd
218  // residual not clear yet
219  // frzndd should not be used up by refreezing in snow, so stick in
220  // supcap.
221  diffndd=0;
222  if (frzndd>0) {
223  diffndd=fsupndd*min((supice-supcap)/dCovrLm ,frzndd);
224  frzndd=frzndd-diffndd;
225  }
226  if(runoff<0){
227  saccu= saccu -runoff;
228  smelt = 0;
229  precrunoff=prect-saccu;
230  //here assume pdd residual is 0, =>
231  Tsurf= max(Tsurf,-frzndd);
232  }
233  else {
234  smelt = runoff;
235  precrunoff=prect-max(0.,supice)-saccu;}
236  //here really need pdd balance, try 0.5 fudge factor?
237  //at least runoff>0 => it's fairly warm, so Tsurf is !<<0,
238  //yet from site plots, can be ice free with Tsurf=-5.5C
239  if(Tsurf<0) {
240  Tsurf= min(Tsurf+fsupT*diffndd , 0.);}
241 
242  melt[0]=smelt/yts;
243  accu[0]=saccu/yts;
244  B = saccu - smelt;
245  B = B/yts;
246  pddtj=pddt;
247 
248  return B;
249 }
melt
void melt(IssmDouble *pM, IssmDouble *pMs, IssmDouble *pR, IssmDouble *pmAdd, IssmDouble *pdz_add, IssmDouble **pT, IssmDouble **pd, IssmDouble **pdz, IssmDouble **pW, IssmDouble **pa, IssmDouble **pre, IssmDouble **pgdn, IssmDouble **pgsp, int *pn, IssmDouble dzMin, IssmDouble zMax, IssmDouble zMin, IssmDouble zTop, IssmDouble zY, IssmDouble dIce, int sid)
Definition: Gembx.cpp:1374
IssmDouble
double IssmDouble
Definition: types.h:37
_printf0_
#define _printf0_(StreamArgs)
Definition: Print.h:29
elements.h
prototypes for elements.h
min
IssmDouble min(IssmDouble a, IssmDouble b)
Definition: extrema.cpp:14
PddSurfaceMassBalance
IssmDouble PddSurfaceMassBalance(IssmDouble *monthlytemperatures, IssmDouble *monthlyprec, IssmDouble *pdds, IssmDouble *pds, IssmDouble *melt, IssmDouble *accu, IssmDouble signorm, IssmDouble yts, IssmDouble h, IssmDouble s, IssmDouble desfac, IssmDouble s0t, IssmDouble s0p, IssmDouble rlaps, IssmDouble rlapslgm, IssmDouble TdiffTime, IssmDouble sealevTime, IssmDouble pddsnowfac, IssmDouble pddicefac, IssmDouble rho_water, IssmDouble rho_ice)
Definition: PddSurfaceMassBalance.cpp:11
max
IssmDouble max(IssmDouble a, IssmDouble b)
Definition: extrema.cpp:24