Flow123d  JS_before_hm-1008-g3dab983
field_algo_base.impl.hh
Go to the documentation of this file.
1 /*!
2  *
3  * Copyright (C) 2015 Technical University of Liberec. All rights reserved.
4  *
5  * This program is free software; you can redistribute it and/or modify it under
6  * the terms of the GNU General Public License version 3 as published by the
7  * Free Software Foundation. (http://www.gnu.org/licenses/gpl-3.0.en.html)
8  *
9  * This program is distributed in the hope that it will be useful, but WITHOUT
10  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
11  * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
12  *
13  *
14  * @file field_algo_base.impl.hh
15  * @brief
16  */
17 
18 #ifndef field_algo_base_IMPL_HH_
19 #define field_algo_base_IMPL_HH_
20 
21 #include <string>
22 #include <limits>
23 #include <memory>
24 using namespace std;
25 
27 #include "fields/field_python.hh"
28 #include "fields/field_constant.hh"
29 #include "fields/field_formula.hh"
30 
31 #include "fields/field_values.hh"
32 
33 #include "tools/unit_converter.hh"
34 
35 #include "tools/time_governor.hh"
36 #include "input/factory.hh"
37 #include "input/accessors.hh"
39 
40 namespace it = Input::Type;
41 
42 
43 
44 
45 /******************************************************************************************
46  * Implementation of FieldBase<...>
47  */
48 
49 template <int spacedim, class Value>
51 : value_(r_value_),
52  field_result_(result_other),
53  component_idx_(std::numeric_limits<unsigned int>::max()),
54  unit_conversion_coefficient_(1.0)
55 {
56  value_.set_n_comp(n_comp);
57 }
58 
59 
60 
61 template <int spacedim, class Value>
63  return fmt::format("R{:d}_to_{}", spacedim, Value::type_name() );
64 }
65 
66 
67 
68 template <int spacedim, class Value>
70  stringstream ss;
71  ss << "[" << Value::NRows_ << ", " << Value::NCols_ << "]";
72  return it::Abstract("Field_"+template_name(), "Abstract for all time-space functions.")
73  .allow_auto_conversion("FieldConstant")
76  .close();
77 }
78 
79 
80 template <int spacedim, class Value>
83  if (is_enum_valued) {
84  ASSERT( !(value_selection==Input::Type::Selection()) ).error("Not defined 'value_selection' for enum element type.\n");
85  param_vec.push_back( std::make_pair("element_input_type", std::make_shared<it::Selection>(value_selection)) );
86  } else {
87  param_vec.push_back( std::make_pair("element_input_type", std::make_shared<typename Value::ElementInputType>()) );
88  }
89 
90  return it::Instance(get_input_type(), param_vec).close();
91 }
92 
93 template <int spacedim, class Value>
95  auto unit_record = it::Record("Unit",
96  "Specify unit of an input value. "
97  "Evaluation of the unit formula results into a coeficient and a "
98  "unit in terms of powers of base SI units. The unit must match "
99  "expected SI unit of the value, while the value provided on the input "
100  "is multiplied by the coefficient before further processing. "
101  "The unit formula have a form:\n"
102  "```\n"
103  "<UnitExpr>;<Variable>=<Number>*<UnitExpr>;...,\n"
104  "```\n"
105  "where ```<Variable>``` is a variable name and ```<UnitExpr>``` is a units expression "
106  "which consists of products and divisions of terms.\n\n"
107  "A term has a form: "
108  "```<Base>^<N>```, where ```<N>``` is an integer exponent and ```<Base>``` "
109  "is either a base SI unit, a derived unit, or a variable defined in the same unit formula. "
110  "Example, unit for the pressure head:\n\n"
111  "```MPa/rho/g_; rho = 990*kg*m^-3; g_ = 9.8*m*s^-2```"
112  )
113  .allow_auto_conversion("unit_formula")
114  .declare_key("unit_formula", it::String(), it::Default::obligatory(),
115  "Definition of unit." )
116  .close();
117 
118  return it::Record("FieldAlgorithmBase_common_aux", "")
119  .declare_key("unit", unit_record, it::Default::optional(),
120  "Unit of the field values provided in the main input file, in the external file, or "
121  "by a function (FieldPython).")
122  .close();
123 }
124 
125 template <int spacedim, class Value>
126 shared_ptr< FieldAlgorithmBase<spacedim, Value> >
128 {
129  shared_ptr< FieldAlgorithmBase<spacedim, Value> > func;
130  func = rec.factory< FieldAlgorithmBase<spacedim, Value> >(init_data.n_comp_);
131  func->init_from_input(rec, init_data);
132  return func;
133 }
134 
135 
136 
137 template <int spacedim, class Value>
139  xprintf(PrgErr, "The field '%s' do not support initialization from input.\n",
140  typeid(this).name());
141 }
142 
143 
144 
145 template <int spacedim, class Value>
147  time_ = time;
148  return false; // no change
149 }
150 
151 
152 
153 template <int spacedim, class Value>
155 }
156 
157 
158 
159 template<int spacedim, class Value>
161  return (Value::NRows_ ? 0 : value_.n_rows());
162 }
163 
164 
165 template<int spacedim, class Value>
168  FMT_UNUSED ElementCacheMap &cache_map,
169  FMT_UNUSED unsigned int region_idx)
170 {
171  //ASSERT(false).error("Must be implemented in descendants!\n");
172 }
173 
174 
175 template<int spacedim, class Value>
177  const Armor::array &point_list,
178  const ElementAccessor<spacedim> &elm,
180 {
181  ASSERT_EQ( point_list.size(), value_list.size() ).error();
182  ASSERT_DBG(point_list.n_rows() == spacedim && point_list.n_cols() == 1).error("Invalid point size.\n");
183  for(unsigned int i=0; i< point_list.size(); i++) {
184  ASSERT( Value(value_list[i]).n_rows()==this->value_.n_rows() )(i)(Value(value_list[i]).n_rows())(this->value_.n_rows())
185  .error("value_list has wrong number of rows");
186  value_list[i]=this->value(point_list.vec<spacedim>(i), elm);
187  }
188 
189 }
190 
191 template<int spacedim, class Value>
193  const struct FieldAlgoBaseInitData& init_data)
194 {
195  Input::Record unit_record;
196  if ( rec.opt_val("unit", unit_record) ) {
197  if (!Value::is_scalable()) {
198  WarningOut().fmt("Setting unit conversion coefficient of non-floating point field at address {}\nCoefficient will be skipped.\n",
199  rec.address_string());
200  }
201  std::string unit_str = unit_record.val<std::string>("unit_formula");
202  try {
203  this->unit_conversion_coefficient_ = init_data.unit_si_.convert_unit_from(unit_str);
204  } catch (ExcInvalidUnit &e) {
205  e << rec.ei_address();
206  throw;
207  } catch (ExcNoncorrespondingUnit &e) {
208  e << rec.ei_address();
209  throw;
210  }
211  }
212 }
213 
214 
215 
216 
217 #endif //FUNCTION_BASE_IMPL_HH_
TimeStep time_
Actual time level; initial value is -infinity.
void init_unit_conversion_coefficient(const Input::Record &rec, const struct FieldAlgoBaseInitData &init_data)
Init value of unit_conversion_coefficient_ from input.
Abstract & allow_auto_conversion(const string &type_default)
Allows shorter input of the Abstract providing the default value to the "TYPE" key.
unsigned int size() const
Definition: armor.hh:718
virtual void cache_update(FieldValueCache< typename Value::element_type > &data_cache, ElementCacheMap &cache_map, unsigned int region_idx)
static const Input::Type::Instance & get_input_type_instance(Input::Type::Selection value_selection=Input::Type::Selection())
virtual bool set_time(const TimeStep &time)
double convert_unit_from(std::string actual_unit) const
Convert and check user-defined unit.
Definition: unit_si.cc:217
const Instance & close() const
Used for set Instance to TypeRepository.
std::string format(CStringRef format_str, ArgList args)
Definition: format.h:3141
static Default obligatory()
The factory function to make an empty default value which is obligatory.
Definition: type_record.hh:110
Directing class of FieldValueCache.
Definition: mesh.h:78
Helper class that stores data of generic types.
Definition: type_generic.hh:89
Helper struct stores data for initizalize descentants of FieldAlgorithmBase.
uint n_cols() const
Definition: armor.hh:710
#define ASSERT(expr)
Allow use shorter versions of macro names if these names is not used with external library...
Definition: asserts.hh:347
virtual void value_list(const Armor::array &point_list, const ElementAccessor< spacedim > &elm, std::vector< typename Value::return_type > &value_list)=0
ArmaVec< Type, nr > vec(uint mat_index) const
Definition: armor.hh:807
virtual Value::return_type const & value(const Point &p, const ElementAccessor< spacedim > &elm)=0
Abstract & add_attribute(std::string key, TypeBase::json_string value)
Frontend to TypeBase::add_attribute_.
Record & close() const
Close the Record for further declarations of keys.
Definition: type_record.cc:304
Basic time management class.
static constexpr bool is_enum_valued
double unit_conversion_coefficient_
Coeficient of conversion of user-defined unit.
EI_Address ei_address() const
Definition: accessors.cc:178
static Default optional()
The factory function to make an empty default value which is optional.
Definition: type_record.hh:124
bool opt_val(const string &key, Ret &value) const
virtual Record & allow_auto_conversion(const string &from_key)
Allows shorter input of the Record providing only value of the from_key given as the parameter...
Definition: type_record.cc:133
#define FMT_UNUSED
Definition: posix.h:75
virtual void init_from_input(const Input::Record &rec, const struct FieldAlgoBaseInitData &init_data)
Accessor to the data with type Type::Record.
Definition: accessors.hh:291
const Ret val(const string &key) const
#define xprintf(...)
Definition: system.hh:93
static string field_value_shape()
Record & declare_key(const string &key, std::shared_ptr< TypeBase > type, const Default &default_value, const string &description, TypeBase::attribute_map key_attributes=TypeBase::attribute_map())
Declares a new key of the Record.
Definition: type_record.cc:503
uint n_rows() const
Definition: armor.hh:705
Class for declaration of polymorphic Record.
Accessor to the polymorphic input data of a type given by an AbstracRecord object.
Definition: accessors.hh:458
Abstract & root_of_generic_subtree()
static std::shared_ptr< FieldAlgorithmBase< spacedim, Value > > function_factory(const Input::AbstractRecord &rec, const struct FieldAlgoBaseInitData &init_data)
const UnitSI & unit_si_
FieldAlgorithmBase(unsigned int n_comp=0)
Value value_
Last value, prevents passing large values (vectors) by value.
Definition: system.hh:65
static Input::Type::Abstract & get_input_type()
#define ASSERT_DBG(expr)
virtual void set_mesh(const Mesh *mesh, bool boundary_domain)
#define WarningOut()
Macro defining &#39;warning&#39; record of log.
Definition: logger.hh:270
Record type proxy class.
Definition: type_record.hh:182
const std::shared_ptr< Type > factory(Arguments...arguments) const
static const Input::Type::Record & get_field_algo_common_keys()
static std::string template_name()
Class for declaration of the input data that are in string format.
Definition: type_base.hh:582
Representation of one time step..
Template for classes storing finite set of named values.
#define ASSERT_EQ(a, b)
Definition of comparative assert macro (EQual)
Definition: asserts.hh:328
unsigned int n_comp() const
string address_string() const
Definition: accessors.cc:184