Skip to content

Commit 5912de9

Browse files
committed
Combination of MoveInsertable fixes (#2170)
The master hashes of the commits cherry-picked and squashed to create this commit: PR #2720 783091b PR #2731 b67b3e4 0a1dbe4 73d58f2 3a61395 a6f14fe 00cd55d 1310eac A similar "combined" commit should also be cherry-picked onto the 1.4.x and 1.5.x series and new patch-level releases should be made for them, since this is an issue which prevents certain compilers from being able to compile the library. Refs #2710
1 parent 8d1614a commit 5912de9

File tree

4 files changed

+164
-104
lines changed

4 files changed

+164
-104
lines changed

include/numerics/parsed_fem_function.h

+70-33
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,7 @@
3838
#include <sstream>
3939
#include <string>
4040
#include <vector>
41+
#include <memory>
4142

4243
namespace libMesh
4344
{
@@ -66,21 +67,16 @@ class ParsedFEMFunction : public FEMFunctionBase<Output>
6667
const std::vector<Output> * initial_vals=nullptr);
6768

6869
/**
69-
* This class contains a const reference so it can't be copy or move-assigned.
70+
* Constructors
71+
* - This class contains a const reference so it can't be default
72+
* copy or move-assigned. We manually implement the former
73+
* - This class contains unique_ptrs so it can't be default copy
74+
* constructed or assigned.
75+
* - It can still be default moved and deleted.
7076
*/
71-
ParsedFEMFunction & operator= (const ParsedFEMFunction &) = delete;
77+
ParsedFEMFunction & operator= (const ParsedFEMFunction &);
7278
ParsedFEMFunction & operator= (ParsedFEMFunction &&) = delete;
73-
74-
/**
75-
* The remaining 5 special functions can be safely defaulted.
76-
*
77-
* \note The underlying FunctionParserBase class has a copy
78-
* constructor, so this class should be default-constructible. And,
79-
* although FunctionParserBase's move constructor is deleted, _this_
80-
* class should still be move-constructible because
81-
* FunctionParserBase only appears in a vector.
82-
*/
83-
ParsedFEMFunction (const ParsedFEMFunction &) = default;
79+
ParsedFEMFunction (const ParsedFEMFunction &);
8480
ParsedFEMFunction (ParsedFEMFunction &&) = default;
8581
virtual ~ParsedFEMFunction () = default;
8682

@@ -166,9 +162,9 @@ class ParsedFEMFunction : public FEMFunctionBase<Output>
166162
_n_requested_hess_components;
167163
bool _requested_normals;
168164
#ifdef LIBMESH_HAVE_FPARSER
169-
std::vector<FunctionParserBase<Output>> parsers;
165+
std::vector<std::unique_ptr<FunctionParserBase<Output>>> parsers;
170166
#else
171-
std::vector<char> parsers;
167+
std::vector<char*> parsers;
172168
#endif
173169
std::vector<Output> _spacetime;
174170

@@ -220,6 +216,47 @@ ParsedFEMFunction<Output>::ParsedFEMFunction (const System & sys,
220216
}
221217

222218

219+
template <typename Output>
220+
inline
221+
ParsedFEMFunction<Output>::ParsedFEMFunction (const ParsedFEMFunction<Output> & other) :
222+
FEMFunctionBase<Output>(other),
223+
_sys(other._sys),
224+
_expression(other._expression),
225+
_subexpressions(other._subexpressions),
226+
_n_vars(other._n_vars),
227+
_n_requested_vars(other._n_requested_vars),
228+
_n_requested_grad_components(other._n_requested_grad_components),
229+
_n_requested_hess_components(other._n_requested_hess_components),
230+
_requested_normals(other._requested_normals),
231+
_spacetime(other._spacetime),
232+
_need_var(other._need_var),
233+
_need_var_grad(other._need_var_grad),
234+
#ifdef LIBMESH_ENABLE_SECOND_DERIVATIVES
235+
_need_var_hess(other._need_var_hess),
236+
#endif // LIBMESH_ENABLE_SECOND_DERIVATIVES
237+
variables(other.variables),
238+
_additional_vars(other._additional_vars),
239+
_initial_vals(other._initial_vals)
240+
{
241+
this->reparse(_expression);
242+
}
243+
244+
245+
template <typename Output>
246+
inline
247+
ParsedFEMFunction<Output> &
248+
ParsedFEMFunction<Output>::operator= (const ParsedFEMFunction<Output> & other)
249+
{
250+
// We can only be assigned another ParsedFEMFunction defined on the same System
251+
libmesh_assert(&_sys == &other._sys);
252+
253+
// Use copy-and-swap idiom
254+
ParsedFEMFunction<Output> tmp(other);
255+
std::swap(tmp, *this);
256+
return *this;
257+
}
258+
259+
223260
template <typename Output>
224261
inline
225262
void
@@ -399,7 +436,7 @@ ParsedFEMFunction<Output>::operator() (const FEMContext & c,
399436
{
400437
eval_args(c, p, time);
401438

402-
return eval(parsers[0], "f", 0);
439+
return eval(*parsers[0], "f", 0);
403440
}
404441

405442

@@ -419,7 +456,7 @@ ParsedFEMFunction<Output>::operator() (const FEMContext & c,
419456
libmesh_assert_equal_to (size, parsers.size());
420457

421458
for (unsigned int i=0; i != size; ++i)
422-
output(i) = eval(parsers[i], "f", i);
459+
output(i) = eval(*parsers[i], "f", i);
423460
}
424461

425462

@@ -434,7 +471,7 @@ ParsedFEMFunction<Output>::component (const FEMContext & c,
434471
eval_args(c, p, time);
435472

436473
libmesh_assert_less (i, parsers.size());
437-
return eval(parsers[i], "f", i);
474+
return eval(*parsers[i], "f", i);
438475
}
439476

440477
template <typename Output>
@@ -478,16 +515,16 @@ ParsedFEMFunction<Output>::get_inline_value(const std::string & inline_var_name)
478515
#ifdef LIBMESH_HAVE_FPARSER
479516
// Parse and evaluate the new subexpression.
480517
// Add the same constants as we used originally.
481-
FunctionParserBase<Output> fp;
482-
fp.AddConstant("NaN", std::numeric_limits<Real>::quiet_NaN());
483-
fp.AddConstant("pi", std::acos(Real(-1)));
484-
fp.AddConstant("e", std::exp(Real(1)));
485-
if (fp.Parse(new_subexpression, variables) != -1) // -1 for success
518+
auto fp = libmesh_make_unique<FunctionParserBase<Output>>();
519+
fp->AddConstant("NaN", std::numeric_limits<Real>::quiet_NaN());
520+
fp->AddConstant("pi", std::acos(Real(-1)));
521+
fp->AddConstant("e", std::exp(Real(1)));
522+
if (fp->Parse(new_subexpression, variables) != -1) // -1 for success
486523
libmesh_error_msg
487524
("ERROR: FunctionParser is unable to parse modified expression: "
488-
<< new_subexpression << '\n' << fp.ErrorMsg());
525+
<< new_subexpression << '\n' << fp->ErrorMsg());
489526

490-
Output new_var_value = this->eval(fp, new_subexpression, 0);
527+
Output new_var_value = this->eval(*fp, new_subexpression, 0);
491528
#ifdef NDEBUG
492529
return new_var_value;
493530
#else
@@ -620,16 +657,16 @@ ParsedFEMFunction<Output>::partial_reparse (const std::string & expression)
620657
#ifdef LIBMESH_HAVE_FPARSER
621658
// Parse (and optimize if possible) the subexpression.
622659
// Add some basic constants, to Real precision.
623-
FunctionParserBase<Output> fp;
624-
fp.AddConstant("NaN", std::numeric_limits<Real>::quiet_NaN());
625-
fp.AddConstant("pi", std::acos(Real(-1)));
626-
fp.AddConstant("e", std::exp(Real(1)));
627-
if (fp.Parse(_subexpressions.back(), variables) != -1) // -1 for success
660+
auto fp = libmesh_make_unique<FunctionParserBase<Output>>();
661+
fp->AddConstant("NaN", std::numeric_limits<Real>::quiet_NaN());
662+
fp->AddConstant("pi", std::acos(Real(-1)));
663+
fp->AddConstant("e", std::exp(Real(1)));
664+
if (fp->Parse(_subexpressions.back(), variables) != -1) // -1 for success
628665
libmesh_error_msg
629666
("ERROR: FunctionParser is unable to parse expression: "
630-
<< _subexpressions.back() << '\n' << fp.ErrorMsg());
631-
fp.Optimize();
632-
parsers.push_back(fp);
667+
<< _subexpressions.back() << '\n' << fp->ErrorMsg());
668+
fp->Optimize();
669+
parsers.push_back(std::move(fp));
633670
#else
634671
libmesh_error_msg("ERROR: This functionality requires fparser!");
635672
#endif

include/numerics/parsed_function.h

+71-52
Original file line numberDiff line numberDiff line change
@@ -65,27 +65,13 @@ class ParsedFunction : public FunctionBase<Output>
6565
const std::vector<Output> * initial_vals=nullptr);
6666

6767
/**
68-
* This class cannot be (default) copy assigned because the
69-
* underlying FunctionParserADBase class does not define a custom
70-
* copy assignment operator, and manually manages memory.
68+
* Constructors
69+
* - This class contains unique_ptrs so it can't be default copy
70+
* constructed or assigned, only default moved and deleted.
7171
*/
72-
ParsedFunction & operator= (const ParsedFunction &) = delete;
72+
ParsedFunction (const ParsedFunction &);
73+
ParsedFunction & operator= (const ParsedFunction &);
7374

74-
/**
75-
* The remaining special functions can be defaulted for this class.
76-
*
77-
* \note Despite the fact that the underlying FunctionParserADBase
78-
* class is not move-assignable or move-constructible, it is still
79-
* possible for _this_ class to be move-assigned and
80-
* move-constructed, because FunctionParserADBase objects only
81-
* appear within std::vectors in this class, and std::vectors can
82-
* generally still be move-assigned and move-constructed even when
83-
* their contents cannot. There are some allocator-specific
84-
* exceptions to this, but it should be guaranteed to work for
85-
* std::allocator in C++14 and beyond. See also:
86-
* https://stackoverflow.com/q/42051917/659433
87-
*/
88-
ParsedFunction (const ParsedFunction &) = default;
8975
ParsedFunction (ParsedFunction &&) = default;
9076
ParsedFunction & operator= (ParsedFunction &&) = default;
9177
virtual ~ParsedFunction () = default;
@@ -182,18 +168,18 @@ class ParsedFunction : public FunctionBase<Output>
182168

183169
std::string _expression;
184170
std::vector<std::string> _subexpressions;
185-
std::vector<FunctionParserADBase<Output>> parsers;
171+
std::vector<std::unique_ptr<FunctionParserADBase<Output>>> parsers;
186172
std::vector<Output> _spacetime;
187173

188174
// derivative functions
189-
std::vector<FunctionParserADBase<Output>> dx_parsers;
175+
std::vector<std::unique_ptr<FunctionParserADBase<Output>>> dx_parsers;
190176
#if LIBMESH_DIM > 1
191-
std::vector<FunctionParserADBase<Output>> dy_parsers;
177+
std::vector<std::unique_ptr<FunctionParserADBase<Output>>> dy_parsers;
192178
#endif
193179
#if LIBMESH_DIM > 2
194-
std::vector<FunctionParserADBase<Output>> dz_parsers;
180+
std::vector<std::unique_ptr<FunctionParserADBase<Output>>> dz_parsers;
195181
#endif
196-
std::vector<FunctionParserADBase<Output>> dt_parsers;
182+
std::vector<std::unique_ptr<FunctionParserADBase<Output>>> dt_parsers;
197183
bool _valid_derivatives;
198184

199185
// Variables/values that can be parsed and handled by the function parser
@@ -224,6 +210,37 @@ ParsedFunction<Output,OutputGradient>::ParsedFunction (const std::string & expre
224210
}
225211

226212

213+
template <typename Output, typename OutputGradient>
214+
inline
215+
ParsedFunction<Output,OutputGradient>::ParsedFunction (const ParsedFunction<Output,OutputGradient> & other) :
216+
FunctionBase<Output>(other),
217+
_expression(other._expression),
218+
_subexpressions(other._subexpressions),
219+
_spacetime(other._spacetime),
220+
_valid_derivatives(other._valid_derivatives),
221+
variables(other.variables),
222+
_additional_vars(other._additional_vars),
223+
_initial_vals(other._initial_vals)
224+
{
225+
// parsers can be generated from scratch by reparsing expression
226+
this->reparse(this->_expression);
227+
this->_initialized = true;
228+
}
229+
230+
231+
232+
template <typename Output, typename OutputGradient>
233+
inline
234+
ParsedFunction<Output,OutputGradient> &
235+
ParsedFunction<Output,OutputGradient>::operator= (const ParsedFunction<Output,OutputGradient> & other)
236+
{
237+
// Use copy-and-swap idiom
238+
ParsedFunction<Output,OutputGradient> tmp(other);
239+
std::swap(tmp, *this);
240+
return *this;
241+
}
242+
243+
227244
template <typename Output, typename OutputGradient>
228245
inline
229246
void
@@ -261,7 +278,7 @@ Output
261278
ParsedFunction<Output,OutputGradient>::operator() (const Point & p, const Real time)
262279
{
263280
set_spacetime(p, time);
264-
return eval(parsers[0], "f", 0);
281+
return eval(*parsers[0], "f", 0);
265282
}
266283

267284
template <typename Output, typename OutputGradient>
@@ -270,7 +287,7 @@ Output
270287
ParsedFunction<Output,OutputGradient>::dot (const Point & p, const Real time)
271288
{
272289
set_spacetime(p, time);
273-
return eval(dt_parsers[0], "df/dt", 0);
290+
return eval(*dt_parsers[0], "df/dt", 0);
274291
}
275292

276293
template <typename Output, typename OutputGradient>
@@ -281,12 +298,12 @@ ParsedFunction<Output,OutputGradient>::gradient (const Point & p, const Real tim
281298
OutputGradient grad;
282299
set_spacetime(p, time);
283300

284-
grad(0) = eval(dx_parsers[0], "df/dx", 0);
301+
grad(0) = eval(*dx_parsers[0], "df/dx", 0);
285302
#if LIBMESH_DIM > 1
286-
grad(1) = eval(dy_parsers[0], "df/dy", 0);
303+
grad(1) = eval(*dy_parsers[0], "df/dy", 0);
287304
#endif
288305
#if LIBMESH_DIM > 2
289-
grad(2) = eval(dz_parsers[0], "df/dz", 0);
306+
grad(2) = eval(*dz_parsers[0], "df/dz", 0);
290307
#endif
291308

292309
return grad;
@@ -309,7 +326,7 @@ ParsedFunction<Output,OutputGradient>::operator()
309326
// The remaining locations in _spacetime are currently fixed at construction
310327
// but could potentially be made dynamic
311328
for (unsigned int i=0; i != size; ++i)
312-
output(i) = eval(parsers[i], "f", i);
329+
output(i) = eval(*parsers[i], "f", i);
313330
}
314331

315332
/**
@@ -329,7 +346,7 @@ ParsedFunction<Output,OutputGradient>::component (unsigned int i,
329346
// The remaining locations in _spacetime are currently fixed at construction
330347
// but could potentially be made dynamic
331348
libmesh_assert_less(i, parsers.size());
332-
return eval(parsers[i], "f", i);
349+
return eval(*parsers[i], "f", i);
333350
}
334351

335352
/**
@@ -540,51 +557,53 @@ ParsedFunction<Output,OutputGradient>::partial_reparse (const std::string & expr
540557

541558
// Parse (and optimize if possible) the subexpression.
542559
// Add some basic constants, to Real precision.
543-
FunctionParserADBase<Output> fp;
544-
fp.AddConstant("NaN", std::numeric_limits<Real>::quiet_NaN());
545-
fp.AddConstant("pi", std::acos(Real(-1)));
546-
fp.AddConstant("e", std::exp(Real(1)));
547-
if (fp.Parse(_subexpressions.back(), variables) != -1) // -1 for success
560+
auto fp = libmesh_make_unique<FunctionParserADBase<Output>>();
561+
fp->AddConstant("NaN", std::numeric_limits<Real>::quiet_NaN());
562+
fp->AddConstant("pi", std::acos(Real(-1)));
563+
fp->AddConstant("e", std::exp(Real(1)));
564+
if (fp->Parse(_subexpressions.back(), variables) != -1) // -1 for success
548565
libmesh_error_msg
549566
("ERROR: FunctionParser is unable to parse expression: "
550-
<< _subexpressions.back() << '\n' << fp.ErrorMsg());
567+
<< _subexpressions.back() << '\n' << fp->ErrorMsg());
551568

552569
// use of derivatives is optional. suppress error output on the console
553570
// use the has_derivatives() method to check if AutoDiff was successful.
554571
// also enable immediate optimization
555-
fp.SetADFlags(FunctionParserADBase<Output>::ADSilenceErrors |
572+
fp->SetADFlags(FunctionParserADBase<Output>::ADSilenceErrors |
556573
FunctionParserADBase<Output>::ADAutoOptimize);
557574

558575
// optimize original function
559-
fp.Optimize();
560-
parsers.push_back(fp);
576+
fp->Optimize();
561577

562578
// generate derivatives through automatic differentiation
563-
FunctionParserADBase<Output> dx_fp(fp);
564-
if (dx_fp.AutoDiff("x") != -1) // -1 for success
579+
auto dx_fp = libmesh_make_unique<FunctionParserADBase<Output>>(*fp);
580+
if (dx_fp->AutoDiff("x") != -1) // -1 for success
565581
_valid_derivatives = false;
566-
dx_parsers.push_back(dx_fp);
582+
dx_parsers.push_back(std::move(dx_fp));
567583
#if LIBMESH_DIM > 1
568-
FunctionParserADBase<Output> dy_fp(fp);
569-
if (dy_fp.AutoDiff("y") != -1) // -1 for success
584+
auto dy_fp = libmesh_make_unique<FunctionParserADBase<Output>>(*fp);
585+
if (dy_fp->AutoDiff("y") != -1) // -1 for success
570586
_valid_derivatives = false;
571-
dy_parsers.push_back(dy_fp);
587+
dy_parsers.push_back(std::move(dy_fp));
572588
#endif
573589
#if LIBMESH_DIM > 2
574-
FunctionParserADBase<Output> dz_fp(fp);
575-
if (dz_fp.AutoDiff("z") != -1) // -1 for success
590+
auto dz_fp = libmesh_make_unique<FunctionParserADBase<Output>>(*fp);
591+
if (dz_fp->AutoDiff("z") != -1) // -1 for success
576592
_valid_derivatives = false;
577-
dz_parsers.push_back(dz_fp);
593+
dz_parsers.push_back(std::move(dz_fp));
578594
#endif
579-
FunctionParserADBase<Output> dt_fp(fp);
580-
if (dt_fp.AutoDiff("t") != -1) // -1 for success
595+
auto dt_fp = libmesh_make_unique<FunctionParserADBase<Output>>(*fp);
596+
if (dt_fp->AutoDiff("t") != -1) // -1 for success
581597
_valid_derivatives = false;
582-
dt_parsers.push_back(dt_fp);
598+
dt_parsers.push_back(std::move(dt_fp));
583599

584600
// If at end, use nextstart=maxSize. Else start at next
585601
// character.
586602
nextstart = (end == std::string::npos) ?
587603
std::string::npos : end + 1;
604+
605+
// Store fp for later use
606+
parsers.push_back(std::move(fp));
588607
}
589608
}
590609

0 commit comments

Comments
 (0)