@@ -65,27 +65,13 @@ class ParsedFunction : public FunctionBase<Output>
65
65
const std::vector<Output> * initial_vals=nullptr );
66
66
67
67
/* *
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 .
71
71
*/
72
- ParsedFunction & operator = (const ParsedFunction &) = delete ;
72
+ ParsedFunction (const ParsedFunction &);
73
+ ParsedFunction & operator = (const ParsedFunction &);
73
74
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 ;
89
75
ParsedFunction (ParsedFunction &&) = default ;
90
76
ParsedFunction & operator = (ParsedFunction &&) = default ;
91
77
virtual ~ParsedFunction () = default ;
@@ -182,18 +168,18 @@ class ParsedFunction : public FunctionBase<Output>
182
168
183
169
std::string _expression;
184
170
std::vector<std::string> _subexpressions;
185
- std::vector<FunctionParserADBase<Output>> parsers;
171
+ std::vector<std::unique_ptr< FunctionParserADBase<Output> >> parsers;
186
172
std::vector<Output> _spacetime;
187
173
188
174
// derivative functions
189
- std::vector<FunctionParserADBase<Output>> dx_parsers;
175
+ std::vector<std::unique_ptr< FunctionParserADBase<Output> >> dx_parsers;
190
176
#if LIBMESH_DIM > 1
191
- std::vector<FunctionParserADBase<Output>> dy_parsers;
177
+ std::vector<std::unique_ptr< FunctionParserADBase<Output> >> dy_parsers;
192
178
#endif
193
179
#if LIBMESH_DIM > 2
194
- std::vector<FunctionParserADBase<Output>> dz_parsers;
180
+ std::vector<std::unique_ptr< FunctionParserADBase<Output> >> dz_parsers;
195
181
#endif
196
- std::vector<FunctionParserADBase<Output>> dt_parsers;
182
+ std::vector<std::unique_ptr< FunctionParserADBase<Output> >> dt_parsers;
197
183
bool _valid_derivatives;
198
184
199
185
// Variables/values that can be parsed and handled by the function parser
@@ -224,6 +210,37 @@ ParsedFunction<Output,OutputGradient>::ParsedFunction (const std::string & expre
224
210
}
225
211
226
212
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
+
227
244
template <typename Output, typename OutputGradient>
228
245
inline
229
246
void
@@ -261,7 +278,7 @@ Output
261
278
ParsedFunction<Output,OutputGradient>::operator () (const Point & p, const Real time)
262
279
{
263
280
set_spacetime (p, time );
264
- return eval (parsers[0 ], " f" , 0 );
281
+ return eval (* parsers[0 ], " f" , 0 );
265
282
}
266
283
267
284
template <typename Output, typename OutputGradient>
@@ -270,7 +287,7 @@ Output
270
287
ParsedFunction<Output,OutputGradient>::dot (const Point & p, const Real time)
271
288
{
272
289
set_spacetime (p, time );
273
- return eval (dt_parsers[0 ], " df/dt" , 0 );
290
+ return eval (* dt_parsers[0 ], " df/dt" , 0 );
274
291
}
275
292
276
293
template <typename Output, typename OutputGradient>
@@ -281,12 +298,12 @@ ParsedFunction<Output,OutputGradient>::gradient (const Point & p, const Real tim
281
298
OutputGradient grad;
282
299
set_spacetime (p, time );
283
300
284
- grad (0 ) = eval (dx_parsers[0 ], " df/dx" , 0 );
301
+ grad (0 ) = eval (* dx_parsers[0 ], " df/dx" , 0 );
285
302
#if LIBMESH_DIM > 1
286
- grad (1 ) = eval (dy_parsers[0 ], " df/dy" , 0 );
303
+ grad (1 ) = eval (* dy_parsers[0 ], " df/dy" , 0 );
287
304
#endif
288
305
#if LIBMESH_DIM > 2
289
- grad (2 ) = eval (dz_parsers[0 ], " df/dz" , 0 );
306
+ grad (2 ) = eval (* dz_parsers[0 ], " df/dz" , 0 );
290
307
#endif
291
308
292
309
return grad;
@@ -309,7 +326,7 @@ ParsedFunction<Output,OutputGradient>::operator()
309
326
// The remaining locations in _spacetime are currently fixed at construction
310
327
// but could potentially be made dynamic
311
328
for (unsigned int i=0 ; i != size; ++i)
312
- output (i) = eval (parsers[i], " f" , i);
329
+ output (i) = eval (* parsers[i], " f" , i);
313
330
}
314
331
315
332
/* *
@@ -329,7 +346,7 @@ ParsedFunction<Output,OutputGradient>::component (unsigned int i,
329
346
// The remaining locations in _spacetime are currently fixed at construction
330
347
// but could potentially be made dynamic
331
348
libmesh_assert_less (i, parsers.size ());
332
- return eval (parsers[i], " f" , i);
349
+ return eval (* parsers[i], " f" , i);
333
350
}
334
351
335
352
/* *
@@ -540,51 +557,53 @@ ParsedFunction<Output,OutputGradient>::partial_reparse (const std::string & expr
540
557
541
558
// Parse (and optimize if possible) the subexpression.
542
559
// 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
548
565
libmesh_error_msg
549
566
(" ERROR: FunctionParser is unable to parse expression: "
550
- << _subexpressions.back () << ' \n ' << fp. ErrorMsg ());
567
+ << _subexpressions.back () << ' \n ' << fp-> ErrorMsg ());
551
568
552
569
// use of derivatives is optional. suppress error output on the console
553
570
// use the has_derivatives() method to check if AutoDiff was successful.
554
571
// also enable immediate optimization
555
- fp. SetADFlags (FunctionParserADBase<Output>::ADSilenceErrors |
572
+ fp-> SetADFlags (FunctionParserADBase<Output>::ADSilenceErrors |
556
573
FunctionParserADBase<Output>::ADAutoOptimize);
557
574
558
575
// optimize original function
559
- fp.Optimize ();
560
- parsers.push_back (fp);
576
+ fp->Optimize ();
561
577
562
578
// 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
565
581
_valid_derivatives = false ;
566
- dx_parsers.push_back (dx_fp);
582
+ dx_parsers.push_back (std::move ( dx_fp) );
567
583
#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
570
586
_valid_derivatives = false ;
571
- dy_parsers.push_back (dy_fp);
587
+ dy_parsers.push_back (std::move ( dy_fp) );
572
588
#endif
573
589
#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
576
592
_valid_derivatives = false ;
577
- dz_parsers.push_back (dz_fp);
593
+ dz_parsers.push_back (std::move ( dz_fp) );
578
594
#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
581
597
_valid_derivatives = false ;
582
- dt_parsers.push_back (dt_fp);
598
+ dt_parsers.push_back (std::move ( dt_fp) );
583
599
584
600
// If at end, use nextstart=maxSize. Else start at next
585
601
// character.
586
602
nextstart = (end == std::string::npos) ?
587
603
std::string::npos : end + 1 ;
604
+
605
+ // Store fp for later use
606
+ parsers.push_back (std::move (fp));
588
607
}
589
608
}
590
609
0 commit comments