ERF
Energy Research and Forecasting: An Atmospheric Modeling Code
ERF_SAM.H
Go to the documentation of this file.
1 /*
2  * Implementation 1-moment microphysics model
3  * NOTE: this model is based on the SAM code, and the Klemp's paper
4  * 1): Joseph, Klemp, the simulation of three-dimensional convective storm dynamics,
5  * Journal of the atmospheric sciences, vol35, p1070
6  * 2): Marat Khairoutdinov and David Randall, cloud resolving modeling of the ARM summer 1997 IOP:
7  * model formulation, results, uncertainties, and sensitivities, Journal of the atmospheric sciences, vol60, p607
8  */
9 #ifndef ERF_SAM_H
10 #define ERF_SAM_H
11 
12 #include <string>
13 #include <vector>
14 #include <memory>
15 
16 #include <AMReX_FArrayBox.H>
17 #include <AMReX_Geometry.H>
18 #include <AMReX_TableData.H>
19 #include <AMReX_MultiFabUtil.H>
20 
21 #include "ERF_Constants.H"
22 #include "ERF_MicrophysicsUtils.H"
23 #include "ERF_IndexDefines.H"
24 #include "ERF_DataStruct.H"
25 #include "ERF_NullMoist.H"
26 #include "ERF_SAMUtils.H"
27 
28 namespace MicVar {
29  enum {
30  // independent variables
31  rho=0, // density
32  theta, // liquid/ice water potential temperature
33  tabs, // temperature
34  pres, // pressure
35  // non-precipitating vars
36  qt, // total cloud
37  qn, // cloud condensate (liquid + ice)
38  qv, // cloud vapor
39  qcl, // cloud water
40  qci, // cloud ice
41  // precipitating vars
42  qp, // total precip
43  qpr, // precip rain
44  qps, // precip ice
45  qpg, // graupel
46  // derived vars
50  NumVars
51  };
52 }
53 
54 class SAM : public NullMoist {
55 
56  using FabPtr = std::shared_ptr<amrex::MultiFab>;
57 
58 public:
59  // constructor
60  SAM () {}
61 
62  // destructor
63  virtual ~SAM () = default;
64 
65  // cloud physics
66  void Cloud (const SolverChoice& sc);
67 
68  // ice physics
69  void IceFall (const SolverChoice& sc);
70 
71  // precip
72  void Precip (const SolverChoice& sc);
73 
74  // precip fall
75  void PrecipFall (const SolverChoice& sc);
76 
77  // Set up for first time
78  void
79  Define (SolverChoice& sc) override
80  {
81  m_fac_cond = lcond / sc.c_p;
82  m_fac_fus = lfus / sc.c_p;
83  m_fac_sub = lsub / sc.c_p;
84  m_axis = sc.ave_plane;
85  m_rdOcp = sc.rdOcp;
86  m_do_cond = (!sc.use_shoc);
87  }
88 
89  // init
90  void
91  Init (const amrex::MultiFab& cons_in,
92  const amrex::BoxArray& grids,
93  const amrex::Geometry& geom,
94  const amrex::Real& dt_advance,
95  std::unique_ptr<amrex::MultiFab>& z_phys_nd,
96  std::unique_ptr<amrex::MultiFab>& detJ_cc) override;
97 
98  // Import minimum dz at this level
99  void
100  Set_dzmin (const amrex::Real dz_min) override
101  {
102  m_dzmin = dz_min;
103  }
104 
105  // Copy state into micro vars
106  void
107  Copy_State_to_Micro (const amrex::MultiFab& cons_in) override;
108 
109  // Copy state into micro vars
110  void
111  Copy_Micro_to_State (amrex::MultiFab& cons_in) override;
112 
113  void
114  Update_Micro_Vars (amrex::MultiFab& cons_in) override
115  {
116  this->Copy_State_to_Micro(cons_in);
117  this->Compute_Coefficients();
118  }
119 
120  void
121  Update_State_Vars (amrex::MultiFab& cons_in,
122  const amrex::MultiFab& /*z_phys_nd*/) override
123  {
124  this->Copy_Micro_to_State(cons_in);
125  }
126 
127  // wrapper to do all the updating
128  void
129  Advance (const amrex::Real& dt_advance,
130  const SolverChoice& sc) override
131  {
132  dt = dt_advance;
133 
134  this->Cloud(sc);
135  this->IceFall(sc);
136  this->Precip(sc);
137  this->PrecipFall(sc);
138  }
139 
140  amrex::MultiFab*
141  Qmoist_Ptr (const int& varIdx) override
142  {
144  return mic_fab_vars[MicVarMap[varIdx]].get();
145  }
146 
147  const amrex::MultiFab*
148  Qmoist_Ptr (const int& varIdx) const
149  {
151  return mic_fab_vars[MicVarMap[varIdx]].get();
152  }
153 
154  void
156 
158  CoefficientRowAt (int k) const;
159 
160  int
161  Qmoist_Size () override { return SAM::m_qmoist_size; }
162 
163  int
165 
166  int
168 
169  void
171  std::vector<int>& a_idx,
172  std::vector<std::string>& a_names) const override
173  {
174  a_idx.clear();
175  a_names.clear();
176 
177  // NOTE: These are the indices to access into qmoist (not mic_fab_vars)
178  a_idx.push_back(0); a_names.push_back("RainAccum");
179  a_idx.push_back(1); a_names.push_back("SnowAccum");
180  a_idx.push_back(2); a_names.push_back("GraupAccum");
181  }
182 
183  AMREX_GPU_HOST_DEVICE
184  AMREX_FORCE_INLINE
185  static amrex::Real
186  NewtonIterSat (int& i, int& j, int& k,
187  const int& SAM_moisture_type,
188  const amrex::Real& fac_cond,
189  const amrex::Real& /*fac_fus*/,
190  const amrex::Real& fac_sub,
191  const amrex::Real& an,
192  const amrex::Real& bn,
193  const amrex::Array4<amrex::Real>& tabs_array,
194  const amrex::Array4<amrex::Real>& pres_array,
195  const amrex::Array4<amrex::Real>& qv_array,
196  const amrex::Array4<amrex::Real>& qc_array,
197  const amrex::Array4<amrex::Real>& qi_array,
198  const amrex::Array4<amrex::Real>& qn_array,
199  const amrex::Array4<amrex::Real>& qt_array)
200  {
201  // Solution tolerance
202  amrex::Real tol = amrex::Real(1.0e-4);
203 
204  // Saturation moisture fractions
205  amrex::Real omn, domn;
206  amrex::Real qsat, dqsat;
207  amrex::Real qsatw, dqsatw;
208  amrex::Real qsati, dqsati;
209 
210  // Newton iteration vars
211  int niter;
212  amrex::Real fff, dfff, dtabs;
213  amrex::Real lstar, dlstar;
214  amrex::Real lstarw, lstari;
215  amrex::Real delta_qc, delta_qi;
216 
217  // Initial guess for temperature & pressure
218  amrex::Real tabs = tabs_array(i,j,k);
219  amrex::Real pres = pres_array(i,j,k);
220 
221  niter = 0;
222  dtabs = 1;
223  //==================================================
224  // Newton iteration to qv=qsat (cloud phase only)
225  //==================================================
226  do {
227  // Latent heats and their derivatives wrt to T
228  lstarw = fac_cond;
229  lstari = fac_sub;
230  domn = zero;
231 
232  // Saturation moisture fractions
235  erf_dtqsatw(tabs, pres, dqsatw);
236  erf_dtqsati(tabs, pres, dqsati);
237 
238  omn = sam_cloud_liquid_fraction(SAM_moisture_type, tabs, an, bn);
239 
240  if (SAM_moisture_type == 1) {
241  // Cloud ice not permitted (condensation)
242  if(tabs >= tbgmax) {
243  }
244  // Cloud water not permitted (sublimation)
245  else if(tabs <= tbgmin) {
246  }
247  // Mixed cloud phase (condensation & deposition)
248  else {
249  lstari = fac_sub;
250  domn = an;
251  }
252  } else if (SAM_moisture_type == 2) {
253  domn = zero;
254  }
255 
256  // Linear combination of each component
257  qsat = sam_mixed_qsat(omn, qsatw, qsati);
258  dqsat = sam_mixed_dqsat_dT(omn, domn, qsatw, qsati, dqsatw, dqsati);
259  lstar = omn * lstarw + (one-omn) * lstari;
260  dlstar = domn * lstarw - domn * lstari;
261 
262  // Function for root finding:
263  // 0 = -T_new + T_old + L_eff/C_p * (qv - qsat)
264  fff = sam_newton_residual(tabs, tabs_array(i,j,k), lstar, qv_array(i,j,k), qsat);
265 
266  // Derivative of function (T_new iterated on)
267  dfff = sam_newton_residual_derivative(lstar, dlstar, qv_array(i,j,k), qsat, dqsat);
268 
269  // Update the temperature
270  dtabs = -fff/dfff;
271  tabs += dtabs;
272 
273  // Update iteration
274  niter = niter+1;
275  } while(std::abs(dtabs) > tol && niter < 20);
276 
277  // Update qsat from last iteration (dq = dq/dt * dt)
278  qsat += dqsat*dtabs;
279 
280  // Changes in each component
281  const SAMCloudPhaseChange phase_change =
282  sam_apply_condensate_limiter(qv_array(i,j,k), qsat,
283  qc_array(i,j,k), qi_array(i,j,k), omn);
284  delta_qc = phase_change.delta_qc;
285  delta_qi = phase_change.delta_qi;
286 
287  // Partition the change in non-precipitating q
288  qv_array(i,j,k) = phase_change.qv;
289  qc_array(i,j,k) += delta_qc;
290  qi_array(i,j,k) += delta_qi;
291  qn_array(i,j,k) = qc_array(i,j,k) + qi_array(i,j,k);
292  qt_array(i,j,k) = qv_array(i,j,k) + qn_array(i,j,k);
293 
294  // Return to temperature
295  return tabs;
296  }
297 
298 private:
299  // Number of qmoist variables (rain_accum, snow_accum, graup_accum)
300  int m_qmoist_size = 3;
301 
302  // Number of qstate variables
304 
305  // Number of qstate variables that are number concentrations
307 
308  // CFL MAX for vertical advection
309  static constexpr amrex::Real CFL_MAX = myhalf;
310 
311  // MicVar map (Qmoist indices -> MicVar enum)
312  amrex::Vector<int> MicVarMap;
313 
314  // geometry
315  amrex::Geometry m_geom;
316 
317  // timestep
319 
320  // number of vertical levels
321  int nlev, zlo, zhi;
322 
323  // plane average axis
324  int m_axis;
325 
326  // constants
331  bool m_do_cond;
332 
333  // Minimum dz at this level
335 
336  // Pointer to terrain data
337  amrex::MultiFab* m_z_phys_nd;
338  amrex::MultiFab* m_detJ_cc;
339 
340  // independent variables
341  amrex::Array<FabPtr, MicVar::NumVars> mic_fab_vars;
342 
343  // microphysics parameters/coefficients
344  amrex::TableData<amrex::Real, 1> accrrc;
345  amrex::TableData<amrex::Real, 1> accrsi;
346  amrex::TableData<amrex::Real, 1> accrsc;
347  amrex::TableData<amrex::Real, 1> coefice;
348  amrex::TableData<amrex::Real, 1> evaps1;
349  amrex::TableData<amrex::Real, 1> evaps2;
350  amrex::TableData<amrex::Real, 1> accrgi;
351  amrex::TableData<amrex::Real, 1> accrgc;
352  amrex::TableData<amrex::Real, 1> evapg1;
353  amrex::TableData<amrex::Real, 1> evapg2;
354  amrex::TableData<amrex::Real, 1> evapr1;
355  amrex::TableData<amrex::Real, 1> evapr2;
356 
357  // vertical plane average data
358  amrex::TableData<amrex::Real, 1> rho1d;
359  amrex::TableData<amrex::Real, 1> pres1d;
360  amrex::TableData<amrex::Real, 1> tabs1d;
361  amrex::TableData<amrex::Real, 1> qt1d;
362  amrex::TableData<amrex::Real, 1> qv1d;
363  amrex::TableData<amrex::Real, 1> qn1d;
364 };
365 #endif
constexpr amrex::Real lfus
Definition: ERF_Constants.H:86
constexpr amrex::Real tbgmax
Definition: ERF_Constants.H:51
constexpr amrex::Real lsub
Definition: ERF_Constants.H:87
constexpr amrex::Real one
Definition: ERF_Constants.H:7
constexpr amrex::Real zero
Definition: ERF_Constants.H:6
constexpr amrex::Real lcond
Definition: ERF_Constants.H:85
constexpr amrex::Real myhalf
Definition: ERF_Constants.H:11
constexpr amrex::Real tbgmin
Definition: ERF_Constants.H:50
AMREX_ALWAYS_ASSERT(bx.length()[2]==khi+1)
AMREX_GPU_HOST_DEVICE AMREX_FORCE_INLINE void erf_dtqsatw(amrex::Real t, amrex::Real p, amrex::Real &dtqsatw)
Definition: ERF_MicrophysicsUtils.H:244
AMREX_GPU_HOST_DEVICE AMREX_FORCE_INLINE void erf_dtqsati(amrex::Real t, amrex::Real p, amrex::Real &dtqsati)
Definition: ERF_MicrophysicsUtils.H:235
AMREX_GPU_HOST_DEVICE AMREX_FORCE_INLINE void erf_qsatw(amrex::Real t, amrex::Real p, amrex::Real &qsatw)
Definition: ERF_MicrophysicsUtils.H:228
AMREX_GPU_HOST_DEVICE AMREX_FORCE_INLINE void erf_qsati(amrex::Real t, amrex::Real p, amrex::Real &qsati)
Definition: ERF_MicrophysicsUtils.H:218
AMREX_GPU_HOST_DEVICE AMREX_FORCE_INLINE amrex::Real sam_newton_residual_derivative(const amrex::Real lstar, const amrex::Real dlstar, const amrex::Real qv, const amrex::Real qsat, const amrex::Real dqsat) noexcept
Definition: ERF_SAMUtils.H:376
AMREX_GPU_HOST_DEVICE AMREX_FORCE_INLINE amrex::Real sam_cloud_liquid_fraction(const int SAM_moisture_type, const amrex::Real tabs, const amrex::Real an, const amrex::Real bn) noexcept
Definition: ERF_SAMUtils.H:245
AMREX_GPU_HOST_DEVICE AMREX_FORCE_INLINE amrex::Real sam_mixed_dqsat_dT(const amrex::Real omn, const amrex::Real domn, const amrex::Real qsatw, const amrex::Real qsati, const amrex::Real dqsatw, const amrex::Real dqsati) noexcept
Definition: ERF_SAMUtils.H:352
AMREX_GPU_HOST_DEVICE AMREX_FORCE_INLINE SAMCloudPhaseChange sam_apply_condensate_limiter(const amrex::Real qv, const amrex::Real qsat, const amrex::Real qc, const amrex::Real qi, const amrex::Real omn) noexcept
Definition: ERF_SAMUtils.H:387
AMREX_GPU_HOST_DEVICE AMREX_FORCE_INLINE amrex::Real sam_newton_residual(const amrex::Real tabs_new, const amrex::Real tabs_old, const amrex::Real lstar, const amrex::Real qv, const amrex::Real qsat) noexcept
Definition: ERF_SAMUtils.H:365
AMREX_GPU_HOST_DEVICE AMREX_FORCE_INLINE amrex::Real sam_mixed_qsat(const amrex::Real omn, const amrex::Real qsatw, const amrex::Real qsati) noexcept
Definition: ERF_SAMUtils.H:343
amrex::Real Real
Definition: ERF_ShocInterface.H:19
Definition: ERF_NullMoist.H:8
Definition: ERF_SAM.H:54
void Copy_Micro_to_State(amrex::MultiFab &cons_in) override
Definition: ERF_UpdateSAM.cpp:15
void Advance(const amrex::Real &dt_advance, const SolverChoice &sc) override
Definition: ERF_SAM.H:129
int zlo
Definition: ERF_SAM.H:321
void Precip(const SolverChoice &sc)
Definition: ERF_Precip.cpp:11
amrex::TableData< amrex::Real, 1 > rho1d
Definition: ERF_SAM.H:358
amrex::TableData< amrex::Real, 1 > evapr2
Definition: ERF_SAM.H:355
SAM()
Definition: ERF_SAM.H:60
amrex::MultiFab * m_detJ_cc
Definition: ERF_SAM.H:338
void Copy_State_to_Micro(const amrex::MultiFab &cons_in) override
Definition: ERF_InitSAM.cpp:94
int n_qstate_moist_numconc_size
Definition: ERF_SAM.H:306
void Update_State_Vars(amrex::MultiFab &cons_in, const amrex::MultiFab &) override
Definition: ERF_SAM.H:121
amrex::TableData< amrex::Real, 1 > evapg1
Definition: ERF_SAM.H:352
void Update_Micro_Vars(amrex::MultiFab &cons_in) override
Definition: ERF_SAM.H:114
amrex::TableData< amrex::Real, 1 > accrgi
Definition: ERF_SAM.H:350
std::shared_ptr< amrex::MultiFab > FabPtr
Definition: ERF_SAM.H:56
virtual ~SAM()=default
amrex::TableData< amrex::Real, 1 > evapg2
Definition: ERF_SAM.H:353
void IceFall(const SolverChoice &sc)
Definition: ERF_IceFall.cpp:13
amrex::Array< FabPtr, MicVar::NumVars > mic_fab_vars
Definition: ERF_SAM.H:341
int m_axis
Definition: ERF_SAM.H:324
amrex::Real m_rdOcp
Definition: ERF_SAM.H:330
void Cloud(const SolverChoice &sc)
Definition: ERF_CloudSAM.cpp:13
int m_qmoist_size
Definition: ERF_SAM.H:300
amrex::MultiFab * m_z_phys_nd
Definition: ERF_SAM.H:337
amrex::Real m_fac_fus
Definition: ERF_SAM.H:328
amrex::TableData< amrex::Real, 1 > evaps1
Definition: ERF_SAM.H:348
amrex::Vector< int > MicVarMap
Definition: ERF_SAM.H:312
bool m_do_cond
Definition: ERF_SAM.H:331
amrex::TableData< amrex::Real, 1 > qv1d
Definition: ERF_SAM.H:362
amrex::Real m_fac_cond
Definition: ERF_SAM.H:327
const amrex::MultiFab * Qmoist_Ptr(const int &varIdx) const
Definition: ERF_SAM.H:148
amrex::TableData< amrex::Real, 1 > accrsi
Definition: ERF_SAM.H:345
amrex::TableData< amrex::Real, 1 > accrrc
Definition: ERF_SAM.H:344
void Compute_Coefficients()
Definition: ERF_InitSAM.cpp:150
amrex::TableData< amrex::Real, 1 > pres1d
Definition: ERF_SAM.H:359
int nlev
Definition: ERF_SAM.H:321
int Qstate_Moist_Size() override
Definition: ERF_SAM.H:164
amrex::Geometry m_geom
Definition: ERF_SAM.H:315
amrex::TableData< amrex::Real, 1 > evapr1
Definition: ERF_SAM.H:354
amrex::Real m_fac_sub
Definition: ERF_SAM.H:329
amrex::TableData< amrex::Real, 1 > evaps2
Definition: ERF_SAM.H:349
void Init(const amrex::MultiFab &cons_in, const amrex::BoxArray &grids, const amrex::Geometry &geom, const amrex::Real &dt_advance, std::unique_ptr< amrex::MultiFab > &z_phys_nd, std::unique_ptr< amrex::MultiFab > &detJ_cc) override
Definition: ERF_InitSAM.cpp:37
void Set_dzmin(const amrex::Real dz_min) override
Definition: ERF_SAM.H:100
amrex::TableData< amrex::Real, 1 > qn1d
Definition: ERF_SAM.H:363
amrex::TableData< amrex::Real, 1 > accrgc
Definition: ERF_SAM.H:351
amrex::TableData< amrex::Real, 1 > accrsc
Definition: ERF_SAM.H:346
int n_qstate_moist_size
Definition: ERF_SAM.H:303
amrex::TableData< amrex::Real, 1 > qt1d
Definition: ERF_SAM.H:361
amrex::TableData< amrex::Real, 1 > coefice
Definition: ERF_SAM.H:347
int zhi
Definition: ERF_SAM.H:321
static constexpr amrex::Real CFL_MAX
Definition: ERF_SAM.H:309
void Define(SolverChoice &sc) override
Definition: ERF_SAM.H:79
void Qmoist_Restart_Vars(const SolverChoice &, std::vector< int > &a_idx, std::vector< std::string > &a_names) const override
Definition: ERF_SAM.H:170
AMREX_GPU_HOST_DEVICE static AMREX_FORCE_INLINE amrex::Real NewtonIterSat(int &i, int &j, int &k, const int &SAM_moisture_type, const amrex::Real &fac_cond, const amrex::Real &, const amrex::Real &fac_sub, const amrex::Real &an, const amrex::Real &bn, const amrex::Array4< amrex::Real > &tabs_array, const amrex::Array4< amrex::Real > &pres_array, const amrex::Array4< amrex::Real > &qv_array, const amrex::Array4< amrex::Real > &qc_array, const amrex::Array4< amrex::Real > &qi_array, const amrex::Array4< amrex::Real > &qn_array, const amrex::Array4< amrex::Real > &qt_array)
Definition: ERF_SAM.H:186
int Qmoist_Size() override
Definition: ERF_SAM.H:161
SAMCoefficientRow CoefficientRowAt(int k) const
Definition: ERF_InitSAM.cpp:244
void PrecipFall(const SolverChoice &sc)
Definition: ERF_PrecipFall.cpp:17
amrex::TableData< amrex::Real, 1 > tabs1d
Definition: ERF_SAM.H:360
amrex::MultiFab * Qmoist_Ptr(const int &varIdx) override
Definition: ERF_SAM.H:141
int Qstate_Moist_NumConc_Size() override
Definition: ERF_SAM.H:167
amrex::Real dt
Definition: ERF_SAM.H:318
amrex::Real m_dzmin
Definition: ERF_SAM.H:334
Definition: ERF_SAM.H:28
@ NumVars
Definition: ERF_SAM.H:50
@ pres
Definition: ERF_SAM.H:34
@ qpr
Definition: ERF_SAM.H:43
@ qci
Definition: ERF_SAM.H:40
@ qv
Definition: ERF_SAM.H:38
@ rho
Definition: ERF_SAM.H:31
@ snow_accum
Definition: ERF_SAM.H:48
@ qpg
Definition: ERF_SAM.H:45
@ qt
Definition: ERF_SAM.H:36
@ qn
Definition: ERF_SAM.H:37
@ rain_accum
Definition: ERF_SAM.H:47
@ theta
Definition: ERF_SAM.H:32
@ qcl
Definition: ERF_SAM.H:39
@ graup_accum
Definition: ERF_SAM.H:49
@ tabs
Definition: ERF_SAM.H:33
@ qps
Definition: ERF_SAM.H:44
@ qp
Definition: ERF_SAM.H:42
@ qsatw
Definition: ERF_WSM6.H:236
@ qsati
Definition: ERF_WSM6.H:236
Definition: ERF_SAMUtils.H:22
amrex::Real delta_qi
Definition: ERF_SAMUtils.H:26
amrex::Real qv
Definition: ERF_SAMUtils.H:27
amrex::Real delta_qc
Definition: ERF_SAMUtils.H:25
Definition: ERF_SAMUtils.H:98
Definition: ERF_DataStruct.H:141
amrex::Real rdOcp
Definition: ERF_DataStruct.H:1222
amrex::Real c_p
Definition: ERF_DataStruct.H:1221
bool use_shoc
Definition: ERF_DataStruct.H:1253
int ave_plane
Definition: ERF_DataStruct.H:1313