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 (w/ terrain)
68  void set_k_indices_T (const int& lev);
69 
70  // Populate all 2D iMFs ijk_indx (w/ terrain)
71  void set_norm_indices_T (const int& lev);
72 
73  // Populate positions (w/ terrain & norm vector & interpolation)
74  void set_z_positions_T (const int& lev);
75 
76  // Populate positions (w/ terrain & norm vector & interpolation)
77  void set_norm_positions_T (const int& lev);
78 
79  // Driver for the different average policies
80  void compute_averages (const int& lev);
81 
82  // Fill averages for policy::plane
83  void compute_plane_averages (const int& lev);
84 
85  // Fill averages for policy::point
86  void compute_region_averages (const int& lev);
87 
88  // Write k indices
89  void write_k_indices (const int& lev);
90 
91  // Write ijk indices
92  void write_norm_indices (const int& lev);
93 
94  // Write XZ planar positions
95  void write_xz_positions (const int& lev,
96  const int& j);
97 
98  // Write averages on 2D mf
99  void write_averages (const int& lev);
100 
101  // Get pointer to the 2D mf of averages
102  [[nodiscard]] const amrex::MultiFab* get_average (const int& lev, const int& comp) const { return m_averages[lev][comp].get(); }
103 
104  // Get z_ref (may be computed from specified k_indx)
105  [[nodiscard]] amrex::Real get_zref () const { return m_zref; }
106 
107  /**
108  * Function to compute trilinear interpolation with terrain.
109  *
110  * @param[in] xp X-position
111  * @param[in] yp Y-position
112  * @param[out] interp_vals Values interpolated
113  * @param[in] interp_array Array to interpolate on
114  * @param[in] z_arr Physical heights
115  * @param[in] plo Problem lower bounds
116  * @param[in] dxi Inverse cell size array
117  * @param[in] interp_comp Number of components to interpolate
118  */
119  AMREX_GPU_HOST_DEVICE AMREX_INLINE
120  static void trilinear_interp_T (const amrex::Real& xp,
121  const amrex::Real& yp,
122  const amrex::Real& zp,
123  amrex::Real* interp_vals,
124  amrex::Array4<amrex::Real const> const& interp_array,
125  amrex::Array4<amrex::Real const> const& z_arr,
126  const amrex::GpuArray<amrex::Real, AMREX_SPACEDIM>& plo,
127  const amrex::GpuArray<amrex::Real, AMREX_SPACEDIM>& dxi,
128  const int interp_comp)
129  {
130  // Search to get z/k
131  bool found = false;
132  int kmax = ubound(z_arr).z;
133  amrex::Real zval = 0.0;
134  amrex::Real z_target = zp;
135 
136  // Map position to i,j (must be same mapping in cpp file)
137  amrex::Real ireal = (xp - plo[0]) * dxi[0];
138  amrex::Real jreal = (yp - plo[1]) * dxi[1];
139  int i_new = (int) (ireal - 0.5);
140  int j_new = (int) (jreal - 0.5);
141 
142  for (int lk(0); lk<kmax; ++lk) {
143  amrex::Real z_lo = 0.25 * ( z_arr(i_new,j_new ,lk ) + z_arr(i_new+1,j_new ,lk )
144  + z_arr(i_new,j_new+1,lk ) + z_arr(i_new+1,j_new+1,lk ) );
145  amrex::Real z_hi = 0.25 * ( z_arr(i_new,j_new ,lk+1) + z_arr(i_new+1,j_new ,lk+1)
146  + z_arr(i_new,j_new+1,lk+1) + z_arr(i_new+1,j_new+1,lk+1) );
147  if (z_target > z_lo && z_target < z_hi){
148  found = true;
149  zval = (amrex::Real) lk + ((z_target - z_lo) / (z_hi - z_lo)) + 0.5;
150  break;
151  }
152  }
153 
154  amrex::ignore_unused(found);
155  AMREX_ALWAYS_ASSERT_WITH_MESSAGE(found, "MOSTAverage: Height above terrain not found, try increasing z_ref!");
156 
157  // NOTE: This is the point ahead of the current i,j (e.g. i/j_new + 1)
158  const amrex::RealVect lx(ireal + 0.5, jreal + 0.5, zval);
159 
160  const amrex::IntVect ijk = lx.floor();
161 
162  int i = ijk[0]; int j = ijk[1]; int k = ijk[2];
163 
164  // Convert ijk (IntVect) to a RealVect explicitly
165  amrex::RealVect ijk_r(static_cast<amrex::Real>(ijk[0]),
166  static_cast<amrex::Real>(ijk[1]),
167  static_cast<amrex::Real>(ijk[2]));
168 
169  // Weights
170  const amrex::RealVect sx_hi = lx - ijk_r;
171  const amrex::RealVect sx_lo = 1.0 - sx_hi;
172 
173  for (int n = 0; n < interp_comp; n++) {
174  interp_vals[n] = sx_lo[0]*sx_lo[1]*sx_lo[2]*interp_array(i-1, j-1, k-1,n) +
175  sx_lo[0]*sx_lo[1]*sx_hi[2]*interp_array(i-1, j-1, k ,n) +
176  sx_lo[0]*sx_hi[1]*sx_lo[2]*interp_array(i-1, j , k-1,n) +
177  sx_lo[0]*sx_hi[1]*sx_hi[2]*interp_array(i-1, j , k ,n) +
178  sx_hi[0]*sx_lo[1]*sx_lo[2]*interp_array(i , j-1, k-1,n) +
179  sx_hi[0]*sx_lo[1]*sx_hi[2]*interp_array(i , j-1, k ,n) +
180  sx_hi[0]*sx_hi[1]*sx_lo[2]*interp_array(i , j , k-1,n) +
181  sx_hi[0]*sx_hi[1]*sx_hi[2]*interp_array(i , j , k ,n);
182  }
183  }
184 
185 protected:
186 
187  // Passed through constructor
188  //--------------------------------------------
189  const amrex::Vector<amrex::Geometry> m_geom; // Geometry at each level
190  amrex::Vector<amrex::Vector<amrex::MultiFab*>> m_fields; // Ptr to fields to be averaged
191  amrex::Vector<amrex::MultiFab*> m_z_phys_nd; // Ptr to terrain height coords
192  std::string m_pp_prefix; // ParmParse prefix
193  MeshType m_mesh_type; // Mesh type
194  TerrainType m_terrain_type; // Terrain type
195 
196  // General vars for multiple or all policies
197  //--------------------------------------------
198  int m_nvar{6}; // 6 fields for U/V/T/Qv/Qr/W
199  int m_navg{6}; // 6 averages for U/V/T/Qv/Tv/Umag
200  int m_maxlev{0}; // Total number of levels
201  int m_policy{0}; // Policy for type of averaging
202  bool m_rotate{false}; // Do vector rotations for terrain?
203  amrex::Real m_zref{10.0}; // Height above surface for MOST BC
204  amrex::Vector<std::unique_ptr<amrex::MultiFab>> m_x_pos; // Ptr to 2D mf to hold x position (maxlev)
205  amrex::Vector<std::unique_ptr<amrex::MultiFab>> m_y_pos; // Ptr to 2D mf to hold y position (maxlev)
206  amrex::Vector<std::unique_ptr<amrex::MultiFab>> m_z_pos; // Ptr to 2D mf to hold z position (maxlev)
207  amrex::Vector<std::unique_ptr<amrex::iMultiFab>> m_i_indx; // Ptr to 2D imf to hold i indices (maxlev)
208  amrex::Vector<std::unique_ptr<amrex::iMultiFab>> m_j_indx; // Ptr to 2D imf to hold j indices (maxlev)
209  amrex::Vector<std::unique_ptr<amrex::iMultiFab>> m_k_indx; // Ptr to 2D imf to hold k indices (maxlev)
210  amrex::Vector<amrex::Vector<std::unique_ptr<amrex::MultiFab>>> m_averages; // Ptr to 2D mf to hold averages (maxlev,navg)
211  amrex::Vector<amrex::Vector<std::unique_ptr<amrex::MultiFab>>> m_rot_fields; // Rotated field data
212 
213  // Vars for planar average policy
214  //--------------------------------------------
215  amrex::Vector<amrex::Vector<int>> m_ncell_plane; // Number of cells in plane (maxlev,navg)
216  amrex::Vector<amrex::Vector<amrex::Real>> m_plane_average; // Plane avgs (maxlev,navg)
217 
218  // Vars for point/region average policy
219  //--------------------------------------------
220  int m_radius{0}; // Radius around k_index
221  int m_ncell_region{1}; // Number of cells in local region
222  amrex::Vector<int> m_k_in; // Specified k_index for region avg (maxlev)
223 
224  // Vars for normal vector policy
225  //--------------------------------------------
226  bool m_interp{false}; // Do interpolation on destination?
227  bool m_norm_vec{false}; // Use normal vector to find IJK?
228 
229  // Time average w/ exponential filter fun
230  //--------------------------------------------
231  bool m_t_avg{false}; // Flag to do moving average in time
232  amrex::Vector<int> m_t_init; // Flag to specify if averages are initialized
233  amrex::Real m_time_window{1.0e-16}; // Width of the exp filter function
234  amrex::Real m_fact_new, m_fact_old; // Time average factors for new and old means
235 
236  // Surface velocity correction
237  //--------------------------------------------
238  bool include_subgrid_vel = false;
239  amrex::Vector<amrex::Real> m_Vsg; // Subgrid velocity scale (Mahrt & Sun 1995 MWR)
240 };
241 #endif
amrex::Real Real
Definition: ERF_ShocInterface.H:19
Definition: ERF_MOSTAverage.H:14
int m_navg
Definition: ERF_MOSTAverage.H:199
bool m_t_avg
Definition: ERF_MOSTAverage.H:231
void write_xz_positions(const int &lev, const int &j)
Definition: ERF_MOSTAverage.cpp:1464
void compute_plane_averages(const int &lev)
Definition: ERF_MOSTAverage.cpp:734
int m_policy
Definition: ERF_MOSTAverage.H:201
amrex::Vector< amrex::Vector< std::unique_ptr< amrex::MultiFab > > > m_averages
Definition: ERF_MOSTAverage.H:210
amrex::Vector< std::unique_ptr< amrex::MultiFab > > m_y_pos
Definition: ERF_MOSTAverage.H:205
int m_radius
Definition: ERF_MOSTAverage.H:220
void write_averages(const int &lev)
Definition: ERF_MOSTAverage.cpp:1495
amrex::Vector< std::unique_ptr< amrex::iMultiFab > > m_i_indx
Definition: ERF_MOSTAverage.H:207
void compute_region_averages(const int &lev)
Definition: ERF_MOSTAverage.cpp:1019
amrex::Vector< amrex::MultiFab * > m_z_phys_nd
Definition: ERF_MOSTAverage.H:191
amrex::Vector< int > m_t_init
Definition: ERF_MOSTAverage.H:232
void compute_averages(const int &lev)
Definition: ERF_MOSTAverage.cpp:708
void set_rotated_fields(const int &lev)
Definition: ERF_MOSTAverage.cpp:279
void set_region_normalization(const int &)
Definition: ERF_MOSTAverage.H:61
void set_z_positions_T(const int &lev)
Definition: ERF_MOSTAverage.cpp:586
amrex::Vector< std::unique_ptr< amrex::MultiFab > > m_x_pos
Definition: ERF_MOSTAverage.H:204
amrex::Real get_zref() const
Definition: ERF_MOSTAverage.H:105
void set_norm_positions_T(const int &lev)
Definition: ERF_MOSTAverage.cpp:634
amrex::Vector< amrex::Real > m_Vsg
Definition: ERF_MOSTAverage.H:239
void set_k_indices_T(const int &lev)
Definition: ERF_MOSTAverage.cpp:451
amrex::Vector< amrex::Vector< std::unique_ptr< amrex::MultiFab > > > m_rot_fields
Definition: ERF_MOSTAverage.H:211
bool m_rotate
Definition: ERF_MOSTAverage.H:202
amrex::Vector< amrex::Vector< amrex::Real > > m_plane_average
Definition: ERF_MOSTAverage.H:216
amrex::Vector< std::unique_ptr< amrex::MultiFab > > m_z_pos
Definition: ERF_MOSTAverage.H:206
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:215
void set_plane_normalization(const int &lev)
Definition: ERF_MOSTAverage.cpp:338
void write_k_indices(const int &lev)
Definition: ERF_MOSTAverage.cpp:1376
TerrainType m_terrain_type
Definition: ERF_MOSTAverage.H:194
std::string m_pp_prefix
Definition: ERF_MOSTAverage.H:192
bool m_norm_vec
Definition: ERF_MOSTAverage.H:227
int m_nvar
Definition: ERF_MOSTAverage.H:198
amrex::Real m_fact_new
Definition: ERF_MOSTAverage.H:234
int m_ncell_region
Definition: ERF_MOSTAverage.H:221
void set_norm_indices_T(const int &lev)
Definition: ERF_MOSTAverage.cpp:512
int m_maxlev
Definition: ERF_MOSTAverage.H:200
const amrex::MultiFab * get_average(const int &lev, const int &comp) const
Definition: ERF_MOSTAverage.H:102
bool include_subgrid_vel
Definition: ERF_MOSTAverage.H:238
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:260
~MOSTAverage()
Definition: ERF_MOSTAverage.H:23
amrex::Real m_time_window
Definition: ERF_MOSTAverage.H:233
MeshType m_mesh_type
Definition: ERF_MOSTAverage.H:193
amrex::Vector< std::unique_ptr< amrex::iMultiFab > > m_j_indx
Definition: ERF_MOSTAverage.H:208
void set_k_indices_N(const int &lev)
Definition: ERF_MOSTAverage.cpp:396
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:84
MOSTAverage(MOSTAverage &&) noexcept=default
void write_norm_indices(const int &lev)
Definition: ERF_MOSTAverage.cpp:1415
amrex::Vector< amrex::Vector< amrex::MultiFab * > > m_fields
Definition: ERF_MOSTAverage.H:190
amrex::Vector< int > m_k_in
Definition: ERF_MOSTAverage.H:222
amrex::Real m_fact_old
Definition: ERF_MOSTAverage.H:234
bool m_interp
Definition: ERF_MOSTAverage.H:226
const amrex::Vector< amrex::Geometry > m_geom
Definition: ERF_MOSTAverage.H:189
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:120
amrex::Real m_zref
Definition: ERF_MOSTAverage.H:203
amrex::Vector< std::unique_ptr< amrex::iMultiFab > > m_k_indx
Definition: ERF_MOSTAverage.H:209
Definition: ERF_ConsoleIO.cpp:12