Extension: Json and NetCDF utilities
#include "dg/file/file.h"
Loading...
Searching...
No Matches
probes.h
Go to the documentation of this file.
1#pragma once
2
3
4#include "dg/topology/interpolation.h"
5#ifdef MPI_VERSION
6#include "dg/topology/mpi_projection.h"
7#include "nc_mpi_file.h"
8#endif //MPI_VERSION
9
10#include "probes_params.h"
11#include "nc_file.h"
12#include "records.h"
13
14namespace dg
15{
16namespace file
17{
48template<class NcFile, class Topology>
49struct Probes
50{
51 Probes() = default;
63 NcFile& file,
64 const Topology& grid,
65 const ProbesParams& params // do nothing if probes is false
66 ) : m_file(&file)
67 {
68 m_probes = params.probes;
69 if( !params.probes) return;
70
71 if ( params.coords.size() != grid.ndim())
72 throw std::runtime_error( "Need "+std::to_string(grid.ndim())+" values in coords!");
73 unsigned num_pins = params.get_coords_sizes();
74 // params.coords is empty on ranks other than master
75
76// TODO We could think about distributing the coords among ranks using grid.contains ...
77 m_probe_interpolate = dg::create::interpolation( params.coords, grid,
78 grid.get_bc(), "dg");
79 // Create helper storage probe variables
80 m_simple_probes = create_probes_vec( params.coords[0], grid);
81 m_resultH = dg::evaluate( dg::zero, grid);
82
83 file.def_grp( "probes");
84 m_grp = file.get_current_path() / "probes";
85 file.set_grp( m_grp);
86 file.put_att( {"format", params.format});
87 file.def_dim( "pdim", num_pins);
88 file.template def_dimvar_as<double>( "ptime", NC_UNLIMITED, {{"axis" , "T"}});
89 for( unsigned i=0; i<params.coords.size(); i++)
90 {
91 auto probes_vec = create_probes_vec( params.coords[0], grid);
92 file.defput_var( params.coords_names[i], {"pdim"},
93 {{"long_name" , "Coordinate variable for probe position"}},
94 {probes_vec}, probes_vec);
95 }
96 file.set_grp("..");
97 }
98
121 template<class ListClass, class ...Params>
122 void static_write( const ListClass& records, Params&& ... ps)
123 {
124 if(!m_probes) return;
126 auto result = dg::construct<Result>( m_resultH);
127 static_assert( dg::is_vector_v<Result>, "Result must be a vector type");
128 m_file->set_grp( m_grp);
129
130 for ( auto& record : records)
131 {
132 record.function( result, std::forward<Params>(ps)...);
133 dg::assign( result, m_resultH);
134 dg::blas2::symv( m_probe_interpolate, m_resultH, m_simple_probes);
135
136 m_file->defput_var( record.name, {"pdim"}, record.atts,
137 {m_simple_probes}, m_simple_probes);
138 }
139 m_file->set_grp("..");
140 }
141
153 template<class ListClass, class ...Params>
154 void buffer( double time, const ListClass& probe_list, Params&& ... ps)
155 {
156 if(!m_probes) return;
158 auto result = dg::construct<Result>( m_resultH);
159 m_time_intern.push_back(time);
160 if( m_simple_probes_intern.empty())
161 init<dg::get_value_type<Result>>( probe_list);
162
163 for( auto& record : probe_list)
164 {
165 record.function( result, std::forward<Params>(ps)...);
166 dg::assign( result, m_resultH);
167 dg::blas2::symv( m_probe_interpolate, m_resultH, m_simple_probes);
168 m_simple_probes_intern.at(record.name).push_back(m_simple_probes);
169 }
170 }
171
179 void flush()
180 {
181 if(!m_probes) return;
182 // else we write the internal buffer to file
183 m_file->set_grp( m_grp);
184 for( unsigned j=0; j<m_time_intern.size(); j++)
185 {
186 m_file->put_var( "ptime", {m_probe_start}, m_time_intern[j]);
187 for( auto& field : m_simple_probes_intern)
188 m_file->put_var( field.first, {m_probe_start, field.second[j]},
189 field.second[j]);
190 m_probe_start ++;
191 }
192 // clear internal buffers
193 m_time_intern.clear();
194 for( auto& field : m_simple_probes_intern)
195 field.second.clear();
196 m_file->set_grp( "..");
197 }
198
206 template<class ListClass, class ...Params>
207 void write( double time, const ListClass& probe_list, Params&& ... ps)
208 {
209 buffer( time, probe_list, std::forward<Params>(ps)...);
210 flush();
211 }
212
213 private:
214 template<class T=Topology, std::enable_if_t<dg::is_vector_v<typename
215 T::host_vector, dg::SharedVectorTag>, bool> = true>
216 auto create_probes_vec( const dg::HVec& coord, const T& grid)
217 {
218 return typename Topology::host_vector(coord);
219 }
220 template<class T=Topology, std::enable_if_t<dg::is_vector_v<typename
221 T::host_vector, dg::MPIVectorTag>, bool> = true>
222 auto create_probes_vec( const dg::HVec& coord, const T& grid)
223 {
224 return typename Topology::host_vector(coord, grid.communicator());
225 }
226 template<class value_type, class ListClass>
227 void init( const ListClass& probe_list)
228 {
229 m_file->set_grp( m_grp);
230 for( auto& record : probe_list)
231 {
232 m_simple_probes_intern[record.name] = {}; // empty vectors
233 m_file->template def_var_as<value_type>( record.name, {"ptime", "pdim"});
234 m_file->put_atts( record.atts, record.name);
235 }
236 m_file->set_grp( "..");
237 }
238 bool m_probes = false;
239 NcFile* m_file;
240 std::filesystem::path m_grp;
241 int m_probe_grp_id;
242 typename Topology::host_vector m_simple_probes;
243#ifdef MPI_VERSION
244 std::conditional_t<
245 dg::is_vector_v<typename Topology::host_vector, dg::SharedVectorTag>,
246 dg::IHMatrix_t< typename Topology::value_type>,
247 dg::MIHMatrix_t<typename Topology::value_type> > m_probe_interpolate;
248#else
249 dg::IHMatrix_t< typename Topology::value_type> m_probe_interpolate;
250#endif
251 std::map<std::string, std::vector<typename Topology::host_vector>> m_simple_probes_intern;
252 std::vector<double> m_time_intern;
253 //Container m_resultD;
254 typename Topology::host_vector m_resultH;
255 size_t m_probe_start = 0; // always point to where we currently can write
256};
257
258} //namespace file
259}//namespace dg
MPINcFile NcFile
Definition nc_mpi_file.h:686
std::decay_t< typename detail::get_first_argument_type< Signature >::type > get_first_argument_type_t
Definition records.h:27
Definition easy_atts.h:15
MPI NetCDF-4 file based on serial NetCDF.
Definition nc_mpi_file.h:46
void set_grp(std::filesystem::path path="")
Change group to path.
Definition nc_mpi_file.h:172
std::filesystem::path get_current_path() const
Get the absolute path of the current group.
Definition nc_mpi_file.h:191
void put_att(const std::pair< std::string, nc_att_t > &att, std::string id="")
Put an individual attribute.
Definition nc_mpi_file.h:246
void def_dim(std::string name, size_t size)
Define a dimension named name of size size.
Definition nc_mpi_file.h:209
void def_grp(std::string name)
Definition nc_mpi_file.h:157
Facilitate output at selected points.
Definition probes.h:50
void static_write(const ListClass &records, Params &&... ps)
Directly write results of a list of callback functions to file.
Definition probes.h:122
Probes()=default
void flush()
Flush the buffer to file.
Definition probes.h:179
Probes(NcFile &file, const Topology &grid, const ProbesParams &params)
Construct from parameter struct.
Definition probes.h:62
void write(double time, const ListClass &probe_list, Params &&... ps)
Same as buffer followed by flush.
Definition probes.h:207
void buffer(double time, const ListClass &probe_list, Params &&... ps)
Write (time-dependent) results of a list of callback functions to internal buffer.
Definition probes.h:154
Parameter struct for probe values.
Definition probes_params.h:15
std::vector< dg::HVec > coords
Coordinates list.
Definition probes_params.h:31
unsigned get_coords_sizes() const
Check that coords all have same size and return that size.
Definition probes_params.h:57
bool probes
Indicate existence of probes.
Definition probes_params.h:49