ERF
Energy Research and Forecasting: An Atmospheric Modeling Code
ERF_MOSTAverage.H
Go to the documentation of this file.
1 #ifndef ERF_MOSTAverage_H
2 #define ERF_MOSTAverage_H
3 
4 #include "AMReX_Gpu.H"
5 #include "AMReX_FArrayBox.H"
6 #include "AMReX_MultiFab.H"
7 #include "AMReX_iMultiFab.H"
8 #include "AMReX_ParmParse.H"
9 
10 #include "ERF_IndexDefines.H"
11 #include "ERF_TerrainMetrics.H"
12 #include "ERF_DataStruct.H"
13 
14 class MOSTAverage {
15 public:
16  explicit MOSTAverage (amrex::Vector<amrex::Geometry> geom,
17  const bool& has_zphys,
18  std::string a_pp_prefix,
19  const MeshType& m_mesh_type,
20  const TerrainType& m_terrain_type);
21 
22  // MOSTAverage() = default;
24  {}
25 
26  // Declare a default move constructor so we ensure the destructor is
27  // not called when we return an object of this class by value
28  MOSTAverage (MOSTAverage&&) noexcept = default;
29 
30  // Delete the move assignment operator
31  MOSTAverage& operator=(MOSTAverage&& other) noexcept = delete;
32 
33  // Delete the copy constructor
34  MOSTAverage (const MOSTAverage& other) = delete;
35 
36  // Delete the copy assignment operator
37  MOSTAverage& operator=(const MOSTAverage& other) = delete;
38 
39  // Make data structures at level
40  void make_MOSTAverage_at_level (const int& lev,
41  const amrex::Vector<amrex::MultiFab*>& vars_old,
42  std::unique_ptr<amrex::MultiFab>& Theta_prim,
43  std::unique_ptr<amrex::MultiFab>& Qv_prim,
44  std::unique_ptr<amrex::MultiFab>& Qr_prim,
45  std::unique_ptr<amrex::MultiFab>& z_phys_nd);
46 
47  // Reset the pointers to field MFs
48  void update_field_ptrs (const int& lev,
49  amrex::Vector<amrex::Vector<amrex::MultiFab>>& vars_old,
50  amrex::Vector<std::unique_ptr<amrex::MultiFab>>& Theta_prim,
51  amrex::Vector<std::unique_ptr<amrex::MultiFab>>& Qv_prim,
52  amrex::Vector<std::unique_ptr<amrex::MultiFab>>& Qr_prim);
53 
54  // Update the rotated fields
55  void set_rotated_fields (const int& lev);
56 
57  // Compute ncells per plane
58  void set_plane_normalization (const int& lev);
59 
60  // Compute ncells per plane
61  void set_region_normalization (const int& /*lev*/)
62  {m_ncell_region = (2 * m_radius + 1) * (2 * m_radius + 1) * (2 * m_radius + 1);}
63 
64  // Populate a 2D iMF k_indx (w/o terrain)
65  void set_k_indices_N (const int& lev);
66 
67  // Populate a 2D iMF k_indx (for EB)
68  void set_k_indices_EB (const int& lev);
69 
70  // Populate a 2D iMF k_indx (w/ terrain)
71  void set_k_indices_T (const int& lev);
72 
73  // Populate all 2D iMFs ijk_indx (w/ terrain)
74  void set_norm_indices_T (const int& lev);
75 
76  // Populate positions (w/ terrain & norm vector & interpolation)
77  void set_z_positions_T (const int& lev);
78 
79  // Populate positions (w/ terrain & norm vector & interpolation)
80  void set_norm_positions_T (const int& lev);
81 
82  // Driver for the different average policies
83  void compute_averages (const int& lev);
84 
85  // Fill averages for policy::plane
86  void compute_plane_averages (const int& lev);
87 
88  // Fill averages for policy::point
89  void compute_region_averages (const int& lev);
90 
91  // Write k indices
92  void write_k_indices (const int& lev);
93 
94  // Write ijk indices
95  void write_norm_indices (const int& lev);
96 
97  // Write XZ planar positions
98  void write_xz_positions (const int& lev,
99  const int& j);
100 
101  // Write averages on 2D mf
102  void write_averages (const int& lev);
103 
104  // Get pointer to the 2D mf of averages
105  [[nodiscard]] const amrex::MultiFab* get_average (const int& lev, const int& comp) const { return m_averages[lev][comp].get(); }
106 
107  // Get z_ref (may be computed from specified k_indx)
108  [[nodiscard]] amrex::MultiFab* get_zref (const int& lev) const { return m_zref[lev].get(); }
109 
110  // Get pointer to the k-index mf
111  [[nodiscard]] const amrex::iMultiFab* get_k_indices (const int& lev) const { return m_k_indx[lev].get(); }
112 
113  /**
114  * Function to compute trilinear interpolation with terrain.
115  *
116  * @param[in] xp X-position
117  * @param[in] yp Y-position
118  * @param[out] interp_vals Values interpolated
119  * @param[in] interp_array Array to interpolate on
120  * @param[in] z_arr Physical heights
121  * @param[in] plo Problem lower bounds
122  * @param[in] dxi Inverse cell size array
123  * @param[in] interp_comp Number of components to interpolate
124  */
125  AMREX_GPU_HOST_DEVICE AMREX_INLINE
126  static void trilinear_interp_T (const amrex::Real& xp,
127  const amrex::Real& yp,
128  const amrex::Real& zp,
129  amrex::Real* interp_vals,
130  amrex::Array4<amrex::Real const> const& interp_array,
131  amrex::Array4<amrex::Real const> const& z_arr,
132  const amrex::GpuArray<amrex::Real, AMREX_SPACEDIM>& plo,
133  const amrex::GpuArray<amrex::Real, AMREX_SPACEDIM>& dxi,
134  const int interp_comp)
135  {
136  // Search to get z/k
137  bool found = false;
138  int kmax = ubound(z_arr).z;
139  amrex::Real zval = zero;
140  amrex::Real z_target = zp;
141 
142  // Map position to i,j (must be same mapping in cpp file)
143  amrex::Real ireal = (xp - plo[0]) * dxi[0];
144  amrex::Real jreal = (yp - plo[1]) * dxi[1];
145  int i_new = (int) (ireal - myhalf);
146  int j_new = (int) (jreal - myhalf);
147 
148  for (int lk(0); lk<kmax; ++lk) {
149  amrex::Real z_lo = fourth * ( z_arr(i_new,j_new ,lk ) + z_arr(i_new+1,j_new ,lk )
150  + z_arr(i_new,j_new+1,lk ) + z_arr(i_new+1,j_new+1,lk ) );
151  amrex::Real z_hi = fourth * ( z_arr(i_new,j_new ,lk+1) + z_arr(i_new+1,j_new ,lk+1)
152  + z_arr(i_new,j_new+1,lk+1) + z_arr(i_new+1,j_new+1,lk+1) );
153  if (z_target > z_lo && z_target < z_hi){
154  found = true;
155  zval = (amrex::Real) lk + ((z_target - z_lo) / (z_hi - z_lo)) + myhalf;
156  break;
157  }
158  }
159 
160  amrex::ignore_unused(found);
161  AMREX_ALWAYS_ASSERT_WITH_MESSAGE(found, "MOSTAverage: Height above terrain not found, try increasing z_ref!");
162 
163  // NOTE: This is the point ahead of the current i,j (e.g. i/j_new + 1)
164  const amrex::RealVect lx(ireal + myhalf, jreal + myhalf, zval);
165 
166  const amrex::IntVect ijk = lx.floor();
167 
168  int i = ijk[0]; int j = ijk[1]; int k = ijk[2];
169 
170  // Convert ijk (IntVect) to a RealVect explicitly
171  amrex::RealVect ijk_r(static_cast<amrex::Real>(ijk[0]),
172  static_cast<amrex::Real>(ijk[1]),
173  static_cast<amrex::Real>(ijk[2]));
174 
175  // Weights
176  const amrex::RealVect sx_hi = lx - ijk_r;
177  const amrex::RealVect sx_lo = one - sx_hi;
178 
179  for (int n = 0; n < interp_comp; n++) {
180  interp_vals[n] = sx_lo[0]*sx_lo[1]*sx_lo[2]*interp_array(i-1, j-1, k-1,n) +
181  sx_lo[0]*sx_lo[1]*sx_hi[2]*interp_array(i-1, j-1, k ,n) +
182  sx_lo[0]*sx_hi[1]*sx_lo[2]*interp_array(i-1, j , k-1,n) +
183  sx_lo[0]*sx_hi[1]*sx_hi[2]*interp_array(i-1, j , k ,n) +
184  sx_hi[0]*sx_lo[1]*sx_lo[2]*interp_array(i , j-1, k-1,n) +
185  sx_hi[0]*sx_lo[1]*sx_hi[2]*interp_array(i , j-1, k ,n) +
186  sx_hi[0]*sx_hi[1]*sx_lo[2]*interp_array(i , j , k-1,n) +
187  sx_hi[0]*sx_hi[1]*sx_hi[2]*interp_array(i , j , k ,n);
188  }
189  }
190 
191 protected:
192 
193  // Passed through constructor
194  //--------------------------------------------
195  const amrex::Vector<amrex::Geometry> m_geom; // Geometry at each level
196  amrex::Vector<amrex::Vector<amrex::MultiFab*>> m_fields; // Ptr to fields to be averaged
197  amrex::Vector<amrex::MultiFab*> m_z_phys_nd; // Ptr to terrain height coords
198  std::string m_pp_prefix; // ParmParse prefix
199  MeshType m_mesh_type; // Mesh type
200  TerrainType m_terrain_type; // Terrain type
201 
202  // General vars for multiple or all policies
203  //--------------------------------------------
204  int m_nvar{6}; // 6 fields for U/V/T/Qv/Qr/W
205  int m_navg{6}; // 6 averages for U/V/T/Qv/Tv/Umag
206  int m_maxlev{0}; // Total number of levels
207  int m_policy{0}; // Policy for type of averaging
208  bool m_rotate{false}; // Do vector rotations for terrain?
209  amrex::Vector<std::unique_ptr<amrex::MultiFab>> m_zref; // Height above surface for MOST BC
210  amrex::Vector<std::unique_ptr<amrex::MultiFab>> m_x_pos; // Ptr to 2D mf to hold x position (maxlev)
211  amrex::Vector<std::unique_ptr<amrex::MultiFab>> m_y_pos; // Ptr to 2D mf to hold y position (maxlev)
212  amrex::Vector<std::unique_ptr<amrex::MultiFab>> m_z_pos; // Ptr to 2D mf to hold z position (maxlev)
213  amrex::Vector<std::unique_ptr<amrex::iMultiFab>> m_i_indx; // Ptr to 2D imf to hold i indices (maxlev)
214  amrex::Vector<std::unique_ptr<amrex::iMultiFab>> m_j_indx; // Ptr to 2D imf to hold j indices (maxlev)
215  amrex::Vector<std::unique_ptr<amrex::iMultiFab>> m_k_indx; // Ptr to 2D imf to hold k indices (maxlev)
216  amrex::Vector<amrex::Vector<std::unique_ptr<amrex::MultiFab>>> m_averages; // Ptr to 2D mf to hold averages (maxlev,navg)
217  amrex::Vector<amrex::Vector<std::unique_ptr<amrex::MultiFab>>> m_rot_fields; // Rotated field data
218 
219  // Vars for planar average policy
220  //--------------------------------------------
221  amrex::Vector<amrex::Vector<int>> m_ncell_plane; // Number of cells in plane (maxlev,navg)
222  amrex::Vector<amrex::Vector<amrex::Real>> m_plane_average; // Plane avgs (maxlev,navg)
223 
224  // Vars for point/region average policy
225  //--------------------------------------------
226  int m_radius{0}; // Radius around k_index
227  int m_ncell_region{1}; // Number of cells in local region
228  amrex::Vector<int> m_k_in; // Specified k_index for region avg (maxlev)
229 
230  // Vars for normal vector policy
231  //--------------------------------------------
232  bool m_interp{false}; // Do interpolation on destination?
233  bool m_norm_vec{false}; // Use normal vector to find IJK?
234 
235  // Time average w/ exponential filter fun
236  //--------------------------------------------
237  bool m_t_avg{false}; // Flag to do moving average in time
238  amrex::Vector<int> m_t_init; // Flag to specify if averages are initialized
239  amrex::Real m_time_window{amrex::Real(1.0e-16)}; // Width of the exp filter function
240  amrex::Real m_fact_new, m_fact_old; // Time average factors for new and old means
241 
242  // Surface velocity correction
243  //--------------------------------------------
244  bool include_subgrid_vel = false;
245  amrex::Vector<amrex::Real> m_Vsg; // Subgrid velocity scale (Mahrt & Sun 1995 MWR)
246 
247  // Default values
248  //--------------------------------------------
250 };
251 #endif
constexpr amrex::Real one
Definition: ERF_Constants.H:7
constexpr amrex::Real fourth
Definition: ERF_Constants.H:12
constexpr amrex::Real zero
Definition: ERF_Constants.H:6
constexpr amrex::Real myhalf
Definition: ERF_Constants.H:11
amrex::Real Real
Definition: ERF_ShocInterface.H:19
Definition: ERF_MOSTAverage.H:14
int m_navg
Definition: ERF_MOSTAverage.H:205
bool m_t_avg
Definition: ERF_MOSTAverage.H:237
void write_xz_positions(const int &lev, const int &j)
Definition: ERF_MOSTAverage.cpp:1643
void compute_plane_averages(const int &lev)
Definition: ERF_MOSTAverage.cpp:862
int m_policy
Definition: ERF_MOSTAverage.H:207
amrex::Vector< amrex::Vector< std::unique_ptr< amrex::MultiFab > > > m_averages
Definition: ERF_MOSTAverage.H:216
amrex::Vector< std::unique_ptr< amrex::MultiFab > > m_y_pos
Definition: ERF_MOSTAverage.H:211
int m_radius
Definition: ERF_MOSTAverage.H:226
void write_averages(const int &lev)
Definition: ERF_MOSTAverage.cpp:1685
amrex::Vector< std::unique_ptr< amrex::iMultiFab > > m_i_indx
Definition: ERF_MOSTAverage.H:213
void compute_region_averages(const int &lev)
Definition: ERF_MOSTAverage.cpp:1161
amrex::Vector< amrex::MultiFab * > m_z_phys_nd
Definition: ERF_MOSTAverage.H:197
amrex::Vector< int > m_t_init
Definition: ERF_MOSTAverage.H:238
void compute_averages(const int &lev)
Definition: ERF_MOSTAverage.cpp:836
void set_rotated_fields(const int &lev)
Definition: ERF_MOSTAverage.cpp:288
void set_region_normalization(const int &)
Definition: ERF_MOSTAverage.H:61
void set_z_positions_T(const int &lev)
Definition: ERF_MOSTAverage.cpp:684
amrex::Vector< std::unique_ptr< amrex::MultiFab > > m_x_pos
Definition: ERF_MOSTAverage.H:210
void set_norm_positions_T(const int &lev)
Definition: ERF_MOSTAverage.cpp:747
amrex::Vector< amrex::Real > m_Vsg
Definition: ERF_MOSTAverage.H:245
void set_k_indices_T(const int &lev)
Definition: ERF_MOSTAverage.cpp:516
amrex::Vector< amrex::Vector< std::unique_ptr< amrex::MultiFab > > > m_rot_fields
Definition: ERF_MOSTAverage.H:217
bool m_rotate
Definition: ERF_MOSTAverage.H:208
amrex::MultiFab * get_zref(const int &lev) const
Definition: ERF_MOSTAverage.H:108
amrex::Vector< amrex::Vector< amrex::Real > > m_plane_average
Definition: ERF_MOSTAverage.H:222
amrex::Vector< std::unique_ptr< amrex::MultiFab > > m_z_pos
Definition: ERF_MOSTAverage.H:212
MOSTAverage(amrex::Vector< amrex::Geometry > geom, const bool &has_zphys, std::string a_pp_prefix, const MeshType &m_mesh_type, const TerrainType &m_terrain_type)
amrex::Vector< amrex::Vector< int > > m_ncell_plane
Definition: ERF_MOSTAverage.H:221
void set_plane_normalization(const int &lev)
Definition: ERF_MOSTAverage.cpp:347
void write_k_indices(const int &lev)
Definition: ERF_MOSTAverage.cpp:1539
TerrainType m_terrain_type
Definition: ERF_MOSTAverage.H:200
std::string m_pp_prefix
Definition: ERF_MOSTAverage.H:198
bool m_norm_vec
Definition: ERF_MOSTAverage.H:233
int m_nvar
Definition: ERF_MOSTAverage.H:204
amrex::Real m_fact_new
Definition: ERF_MOSTAverage.H:240
void set_k_indices_EB(const int &lev)
Definition: ERF_MOSTAverage.cpp:461
const amrex::iMultiFab * get_k_indices(const int &lev) const
Definition: ERF_MOSTAverage.H:111
int m_ncell_region
Definition: ERF_MOSTAverage.H:227
void set_norm_indices_T(const int &lev)
Definition: ERF_MOSTAverage.cpp:596
amrex::Vector< std::unique_ptr< amrex::MultiFab > > m_zref
Definition: ERF_MOSTAverage.H:209
int m_maxlev
Definition: ERF_MOSTAverage.H:206
const amrex::MultiFab * get_average(const int &lev, const int &comp) const
Definition: ERF_MOSTAverage.H:105
bool include_subgrid_vel
Definition: ERF_MOSTAverage.H:244
void update_field_ptrs(const int &lev, amrex::Vector< amrex::Vector< amrex::MultiFab >> &vars_old, amrex::Vector< std::unique_ptr< amrex::MultiFab >> &Theta_prim, amrex::Vector< std::unique_ptr< amrex::MultiFab >> &Qv_prim, amrex::Vector< std::unique_ptr< amrex::MultiFab >> &Qr_prim)
Definition: ERF_MOSTAverage.cpp:269
~MOSTAverage()
Definition: ERF_MOSTAverage.H:23
amrex::Real m_time_window
Definition: ERF_MOSTAverage.H:239
MeshType m_mesh_type
Definition: ERF_MOSTAverage.H:199
amrex::Vector< std::unique_ptr< amrex::iMultiFab > > m_j_indx
Definition: ERF_MOSTAverage.H:214
void set_k_indices_N(const int &lev)
Definition: ERF_MOSTAverage.cpp:405
void make_MOSTAverage_at_level(const int &lev, const amrex::Vector< amrex::MultiFab * > &vars_old, std::unique_ptr< amrex::MultiFab > &Theta_prim, std::unique_ptr< amrex::MultiFab > &Qv_prim, std::unique_ptr< amrex::MultiFab > &Qr_prim, std::unique_ptr< amrex::MultiFab > &z_phys_nd)
Definition: ERF_MOSTAverage.cpp:85
MOSTAverage(MOSTAverage &&) noexcept=default
void write_norm_indices(const int &lev)
Definition: ERF_MOSTAverage.cpp:1586
amrex::Vector< amrex::Vector< amrex::MultiFab * > > m_fields
Definition: ERF_MOSTAverage.H:196
amrex::Vector< int > m_k_in
Definition: ERF_MOSTAverage.H:228
const amrex::Real zref_default
Definition: ERF_MOSTAverage.H:249
amrex::Real m_fact_old
Definition: ERF_MOSTAverage.H:240
bool m_interp
Definition: ERF_MOSTAverage.H:232
const amrex::Vector< amrex::Geometry > m_geom
Definition: ERF_MOSTAverage.H:195
AMREX_GPU_HOST_DEVICE static AMREX_INLINE void trilinear_interp_T(const amrex::Real &xp, const amrex::Real &yp, const amrex::Real &zp, amrex::Real *interp_vals, amrex::Array4< amrex::Real const > const &interp_array, amrex::Array4< amrex::Real const > const &z_arr, const amrex::GpuArray< amrex::Real, AMREX_SPACEDIM > &plo, const amrex::GpuArray< amrex::Real, AMREX_SPACEDIM > &dxi, const int interp_comp)
Definition: ERF_MOSTAverage.H:126
amrex::Vector< std::unique_ptr< amrex::iMultiFab > > m_k_indx
Definition: ERF_MOSTAverage.H:215
Definition: ERF_ConsoleIO.cpp:12