16#ifndef dealii_differentiation_sd_symengine_optimizer_h
17#define dealii_differentiation_sd_symengine_optimizer_h
21#ifdef DEAL_II_WITH_SYMENGINE
24# include <symengine/basic.h>
25# include <symengine/dict.h>
26# include <symengine/symengine_exception.h>
27# include <symengine/symengine_rcp.h>
30# include <symengine/lambda_double.h>
31# include <symengine/visitor.h>
32# ifdef HAVE_SYMENGINE_LLVM
33# include <symengine/llvm_double.h>
46# include <boost/serialization/split_member.hpp>
47# include <boost/type_traits.hpp>
52# include <type_traits>
74 "SymEngine has not been built with LLVM support.");
81 "The SymEngine LLVM optimizer does not (yet) support the "
82 "selected return type.");
88 template <
typename ReturnType>
119 template <
class StreamType>
181 static_cast<unsigned int>(
f2));
213 static_cast<unsigned int>(
f2));
271 template <
class StreamType>
275 s <<
" OptimizationFlags|";
298 template <
typename ReturnType,
typename T =
void>
311 template <
typename ReturnType,
typename T =
void>
315# ifdef HAVE_SYMENGINE_LLVM
325 template <
typename ReturnType,
typename T =
void>
345 template <
typename ReturnType,
typename Optimizer,
typename T =
void>
357 template <
typename ReturnType_,
typename T =
void>
360 static const bool is_supported =
false;
362 using ReturnType = void;
368 template <
typename ReturnType_>
373 static const bool is_supported =
true;
376 typename std::conditional<std::is_same<ReturnType_, float>::value,
384 template <
typename ReturnType_>
388 boost::is_complex<ReturnType_>::value &&
389 std::is_arithmetic<typename ReturnType_::value_type>::value>>
391 static const bool is_supported =
true;
393 using ReturnType =
typename std::conditional<
394 std::is_same<ReturnType_, std::complex<float>>::value,
396 std::complex<double>>::type;
401 template <
typename ReturnType_>
404 ReturnType_>::is_supported>>
409 internal::DictionarySubstitutionVisitor<ReturnType, SD::Expression>;
422 const SymEngine::vec_basic & independent_symbols,
423 const SymEngine::vec_basic & dependent_functions,
427 optimizer.init(independent_symbols,
438 template <
class Archive>
441 const unsigned int version,
444 optimizer.save(
archive, version);
453 template <
class Archive>
456 const unsigned int version,
458 const SymEngine::vec_basic & ,
459 const SymEngine::vec_basic & ,
462 optimizer.load(
archive, version);
482 template <
typename Stream>
490 optimizer.print(stream,
499 template <
typename ReturnType_>
502 ReturnType_>::is_supported>>
505 typename std::conditional<!boost::is_complex<ReturnType_>::value,
507 std::complex<double>>::type;
509 !boost::is_complex<ReturnType_>::value,
510 SymEngine::LambdaRealDoubleVisitor,
511 SymEngine::LambdaComplexDoubleVisitor>::type;
524 const SymEngine::vec_basic & independent_symbols,
525 const SymEngine::vec_basic & dependent_functions,
529 optimizer.init(independent_symbols,
540 template <
class Archive>
552 template <
class Archive>
557 const SymEngine::vec_basic & independent_symbols,
558 const SymEngine::vec_basic & dependent_functions,
561 initialize(optimizer,
584 template <
typename StreamType>
598# ifdef HAVE_SYMENGINE_LLVM
599 template <
typename ReturnType_>
605 typename std::conditional<std::is_same<ReturnType_, float>::value,
609 typename std::conditional<std::is_same<ReturnType_, float>::value,
610 SymEngine::LLVMFloatVisitor,
611 SymEngine::LLVMDoubleVisitor>::type;
629 initialize(OptimizerType & optimizer,
630 const SymEngine::vec_basic & independent_symbols,
631 const SymEngine::vec_basic & dependent_functions,
632 const enum OptimizationFlags &optimization_flags)
636 optimizer.init(independent_symbols,
648 template <
class Archive>
652 OptimizerType &optimizer)
664 template <
class Archive>
668 OptimizerType &optimizer,
669 const SymEngine::vec_basic & ,
670 const SymEngine::vec_basic & ,
671 const enum OptimizationFlags & )
695 template <
typename StreamType>
698 const OptimizerType & ,
713 template <
typename ReturnType_>
717 boost::is_complex<ReturnType_>::value &&
718 std::is_arithmetic<typename ReturnType_::value_type>::value>>
722 using ReturnType =
typename LambdaOptimizer<ReturnType_>::ReturnType;
724 typename LambdaOptimizer<ReturnType_>::OptimizerType;
742 initialize(OptimizerType & ,
743 const SymEngine::vec_basic & ,
744 const SymEngine::vec_basic & ,
745 const enum OptimizationFlags & )
756 template <
class Archive>
771 template <
class Archive>
776 const SymEngine::vec_basic & ,
777 const SymEngine::vec_basic & ,
778 const enum OptimizationFlags & )
800 template <
typename StreamType>
803 const OptimizerType & ,
817 template <
typename ReturnType,
typename Optimizer>
818 struct OptimizerHelper<
822 std::is_same<ReturnType, typename Optimizer::ReturnType>::value>>
833 initialize(
typename Optimizer::OptimizerType *optimizer,
834 const SymEngine::vec_basic & independent_symbols,
835 const SymEngine::vec_basic & dependent_functions,
843 Optimizer::initialize(*optimizer,
865 substitute(
typename Optimizer::OptimizerType *optimizer,
879 template <
class Archive>
882 const unsigned int version,
883 typename Optimizer::OptimizerType *optimizer)
890 Optimizer::save(
archive, version, *optimizer);
899 template <
class Archive>
902 const unsigned int version,
903 typename Optimizer::OptimizerType *optimizer,
904 const SymEngine::vec_basic & independent_symbols,
905 const SymEngine::vec_basic & dependent_functions,
938 template <
typename Stream>
941 typename Optimizer::OptimizerType *optimizer,
951 Optimizer::print(stream,
959 template <
typename ReturnType,
typename Optimizer>
960 struct OptimizerHelper<
964 !std::is_same<ReturnType, typename Optimizer::ReturnType>::value>>
975 initialize(
typename Optimizer::OptimizerType *optimizer,
976 const SymEngine::vec_basic & independent_symbols,
977 const SymEngine::vec_basic & dependent_functions,
983 optimizer->init(independent_symbols,
1004 substitute(
typename Optimizer::OptimizerType *optimizer,
1012 std::vector<typename Optimizer::ReturnType>
int_outputs(
1014 std::vector<typename Optimizer::ReturnType>
int_inputs(
1032 template <
class Archive>
1035 const unsigned int version,
1036 typename Optimizer::OptimizerType *optimizer)
1039 Optimizer::save(
archive, version, *optimizer);
1048 template <
class Archive>
1051 const unsigned int version,
1052 typename Optimizer::OptimizerType *optimizer,
1053 const SymEngine::vec_basic & independent_symbols,
1054 const SymEngine::vec_basic & dependent_functions,
1065 independent_symbols,
1066 dependent_functions,
1067 optimization_flags);
1087 template <
typename Stream>
1090 typename Optimizer::OptimizerType *optimizer,
1097 optimizer->print(stream,
1131 template <
typename NumberType,
1134 template <
int,
int,
typename>
1143 for (
unsigned int i = 0; i < out.n_independent_components; ++i)
1146 out.unrolled_to_component_indices(i));
1176 template <
typename NumberType,
int dim>
1184 for (
unsigned int i = 0;
1187 for (
unsigned int j = 0;
1217 template <
typename NumberType,
typename T>
1222 optimizer.register_function(function);
1243 template <
typename NumberType,
typename T>
1246 const std::vector<T> & functions)
1248 for (
const auto &function : functions)
1272 template <
typename NumberType,
typename T,
typename...
Args>
1296 template <
int,
int,
typename>
1304 for (
unsigned int i = 0; i <
symbol_tensor.n_independent_components;
1331 for (
unsigned int i = 0;
1334 for (
unsigned int j = 0;
1439 template <
typename ReturnType>
1571 template <
class Archive>
1660 template <
int rank,
int dim>
1668 template <
int rank,
int dim>
1696 template <
typename T>
1713 template <
typename T,
typename...
Args>
1832 substitute(
const SymEngine::map_basic_basic &substitution_map)
const;
1846 const std::vector<ReturnType> &values)
const;
1860 const std::vector<ReturnType> &values)
const;
1899 const std::vector<ReturnType> &
1920 std::vector<ReturnType>
1931 template <
int rank,
int dim>
1944 template <
int rank,
int dim>
1968 std::vector<ReturnType>
1980 template <
int rank,
int dim>
1993 template <
int rank,
int dim>
2044 const SymEngine::RCP<const SymEngine::Basic> &function)
const;
2152 template <
typename ReturnType>
2153 template <
typename Stream>
2159 stream <<
"Method? " << optimization_method() <<
'\n';
2160 stream <<
"Flags: " << optimization_flags() <<
'\n';
2161 stream <<
"Optimized? " << (optimized() ?
"Yes" :
"No") <<
'\n';
2162 stream <<
"Values substituted? " << values_substituted() <<
"\n\n";
2165 stream <<
"Symbols (" << n_independent_variables()
2166 <<
" independent variables):" <<
'\n';
2168 for (SD::types::substitution_map::const_iterator
it =
2169 independent_variables_symbols.
begin();
2170 it != independent_variables_symbols.
end();
2173 stream <<
cntr <<
": " <<
it->first <<
'\n';
2175 stream <<
'\n' << std::flush;
2178 stream <<
"Functions (" << n_dependent_variables()
2179 <<
" dependent variables):" <<
'\n';
2181 for (
typename SD::types::symbol_vector::const_iterator
it =
2182 dependent_variables_functions.
begin();
2183 it != dependent_variables_functions.
end();
2186 stream <<
cntr <<
": " << (*it) <<
'\n';
2188 stream <<
'\n' << std::flush;
2191 if (optimized() ==
true && use_symbolic_CSE() ==
true)
2202 ExcMessage(
"Cannot cast optimizer to Dictionary type."));
2204 internal::OptimizerHelper<
2206 internal::DictionaryOptimizer<ReturnType>>
::
2208 dynamic_cast<typename internal::DictionaryOptimizer<
2214 stream <<
'\n' << std::flush;
2220 ExcMessage(
"Cannot cast optimizer to Lambda type."));
2222 internal::OptimizerHelper<ReturnType,
2223 internal::LambdaOptimizer<ReturnType>>
::
2225 dynamic_cast<typename internal::LambdaOptimizer<
2231# ifdef HAVE_SYMENGINE_LLVM
2236 ExcMessage(
"Cannot cast optimizer to LLVM type."));
2238 internal::OptimizerHelper<ReturnType,
2239 internal::LLVMOptimizer<ReturnType>>
::
2241 dynamic_cast<typename internal::LLVMOptimizer<
2254 if (values_substituted())
2256 stream <<
"Evaluated functions:" <<
'\n';
2257 stream << std::flush;
2259 for (
typename std::vector<ReturnType>::const_iterator
it =
2260 dependent_variables_output.
begin();
2261 it != dependent_variables_output.
end();
2264 stream <<
cntr <<
": " << (*it) <<
'\n';
2266 stream <<
'\n' << std::flush;
2272 template <
typename ReturnType>
2273 template <
class Archive>
2276 const unsigned int version)
const
2281 static_cast<typename std::underlying_type<OptimizerType>::type
>(
2287 static_cast<typename std::underlying_type<OptimizationFlags>::type
>(
2294 ar &independent_variables_symbols;
2295 ar &dependent_variables_functions;
2297 ar &dependent_variables_output;
2298 ar &map_dep_expr_vec_entry;
2299 ar &ready_for_value_extraction;
2302 has_been_serialized =
true;
2303 ar &has_been_serialized;
2311 if (
typename internal::DictionaryOptimizer<ReturnType>::OptimizerType
2312 *
opt =
dynamic_cast<typename internal::DictionaryOptimizer<
2317 internal::OptimizerHelper<
2319 internal::DictionaryOptimizer<ReturnType>>::save(
ar, version,
opt);
2321 else if (
typename internal::LambdaOptimizer<ReturnType>::OptimizerType
2322 *
opt =
dynamic_cast<typename internal::LambdaOptimizer<
2327 internal::OptimizerHelper<
2329 internal::LambdaOptimizer<ReturnType>>::save(
ar, version,
opt);
2331# ifdef HAVE_SYMENGINE_LLVM
2332 else if (
typename internal::LLVMOptimizer<ReturnType>::OptimizerType
2333 *
opt =
dynamic_cast<typename internal::LLVMOptimizer<
2338 internal::OptimizerHelper<
2340 internal::LLVMOptimizer<ReturnType>>::save(
ar, version,
opt);
2351 template <
typename ReturnType>
2352 template <
class Archive>
2364 typename std::underlying_type<OptimizerType>::type m;
2369 typename std::underlying_type<OptimizationFlags>::type f;
2376 ar &independent_variables_symbols;
2377 ar &dependent_variables_functions;
2379 ar &dependent_variables_output;
2380 ar &map_dep_expr_vec_entry;
2381 ar &ready_for_value_extraction;
2383 ar &has_been_serialized;
2390 create_optimizer(optimizer);
2399 if (
typename internal::DictionaryOptimizer<ReturnType>::OptimizerType
2400 *
opt =
dynamic_cast<typename internal::DictionaryOptimizer<
2405 internal::OptimizerHelper<ReturnType,
2406 internal::DictionaryOptimizer<ReturnType>>
::
2413 dependent_variables_functions),
2414 optimization_flags());
2416 else if (
typename internal::LambdaOptimizer<ReturnType>::OptimizerType
2417 *
opt =
dynamic_cast<typename internal::LambdaOptimizer<
2422 internal::OptimizerHelper<ReturnType,
2423 internal::LambdaOptimizer<ReturnType>>
::
2430 dependent_variables_functions),
2431 optimization_flags());
2433# ifdef HAVE_SYMENGINE_LLVM
2434 else if (
typename internal::LLVMOptimizer<ReturnType>::OptimizerType
2435 *
opt =
dynamic_cast<typename internal::LLVMOptimizer<
2440 internal::OptimizerHelper<ReturnType,
2441 internal::LLVMOptimizer<ReturnType>>
::
2448 dependent_variables_functions),
2449 optimization_flags());
2460 template <
typename ReturnType>
2461 template <
int rank,
int dim>
2466 Assert(optimized() ==
false,
2468 "Cannot register functions once the optimizer is finalised."));
2470 register_vector_functions(
2476 template <
typename ReturnType>
2477 template <
int rank,
int dim>
2482 Assert(optimized() ==
false,
2484 "Cannot register functions once the optimizer is finalised."));
2486 register_vector_functions(
2492 template <
typename ReturnType>
2493 template <
typename T,
typename...
Args>
2505 template <
typename ReturnType>
2506 template <
typename T>
2509 const std::vector<T> &functions)
2516 template <
typename ReturnType>
2517 template <
int rank,
int dim>
2530 template <
typename ReturnType>
2531 template <
int rank,
int dim>
2537 values_substituted() ==
true,
2539 "The optimizer is not configured to perform evaluation. "
2540 "This action can only performed after substitute() has been called."));
2547 template <
typename ReturnType>
2548 template <
int rank,
int dim>
2561 template <
typename ReturnType>
2562 template <
int rank,
int dim>
2568 values_substituted() ==
true,
2570 "The optimizer is not configured to perform evaluation. "
2571 "This action can only performed after substitute() has been called."));
value_type * data() const noexcept
SymmetricTensor< rank, dim, ReturnType > extract(const SymmetricTensor< rank, dim, Expression > &funcs, const std::vector< ReturnType > &cached_evaluation) const
bool use_symbolic_CSE() const
types::substitution_map independent_variables_symbols
types::symbol_vector dependent_variables_functions
void substitute(const types::substitution_map &substitution_map) const
enum OptimizationFlags flags
void register_functions(const T &functions, const Args &...other_functions)
void register_scalar_function(const SD::Expression &function)
const types::symbol_vector & get_dependent_functions() const
void create_optimizer(std::unique_ptr< SymEngine::Visitor > &optimizer)
bool ready_for_value_extraction
void print(Stream &stream, const bool print_cse=false) const
enum OptimizerType optimization_method() const
void copy_from(const BatchOptimizer &other)
void register_function(const Tensor< rank, dim, Expression > &function_tensor)
void set_optimization_method(const enum OptimizerType &optimization_method, const enum OptimizationFlags &optimization_flags=OptimizationFlags::optimize_all)
SymmetricTensor< rank, dim, ReturnType > evaluate(const SymmetricTensor< rank, dim, Expression > &funcs) const
void save(Archive &archive, const unsigned int version) const
enum OptimizationFlags optimization_flags() const
void register_functions(const types::symbol_vector &functions)
std::vector< ReturnType > dependent_variables_output
enum OptimizerType method
Tensor< rank, dim, ReturnType > extract(const Tensor< rank, dim, Expression > &funcs, const std::vector< ReturnType > &cached_evaluation) const
std::size_t n_dependent_variables() const
void serialize(Archive &archive, const unsigned int version)
void register_symbols(const types::substitution_map &substitution_map)
const std::vector< ReturnType > & evaluate() const
std::map< SD::Expression, std::size_t, SD::types::internal::ExpressionKeyLess > map_dependent_expression_to_vector_entry_t
void register_function(const Expression &function)
Tensor< rank, dim, ReturnType > evaluate(const Tensor< rank, dim, Expression > &funcs) const
std::unique_ptr< SymEngine::Visitor > optimizer
ReturnType extract(const Expression &func, const std::vector< ReturnType > &cached_evaluation) const
BatchOptimizer(BatchOptimizer &&) noexcept=default
void load(Archive &archive, const unsigned int version)
void register_functions(const std::vector< T > &functions)
bool is_valid_nonunique_dependent_variable(const SD::Expression &function) const
void register_vector_functions(const types::symbol_vector &functions)
std::size_t n_independent_variables() const
types::symbol_vector get_independent_symbols() const
void register_function(const SymmetricTensor< rank, dim, Expression > &function_tensor)
map_dependent_expression_to_vector_entry_t map_dep_expr_vec_entry
bool values_substituted() const
#define DEAL_II_NAMESPACE_OPEN
#define DEAL_II_NAMESPACE_CLOSE
static ::ExceptionBase & ExcNotImplemented()
#define Assert(cond, exc)
static ::ExceptionBase & ExcSymEngineLLVMReturnTypeNotSupported()
#define DeclExceptionMsg(Exception, defaulttext)
static ::ExceptionBase & ExcSymEngineLLVMNotAvailable()
static ::ExceptionBase & ExcInternalError()
static ::ExceptionBase & ExcNotInitialized()
static ::ExceptionBase & ExcMessage(std::string arg1)
#define AssertThrow(cond, exc)
SD::types::symbol_vector extract_symbols(const SD::types::substitution_map &substitution_values)
SymEngine::vec_basic convert_expression_vector_to_basic_vector(const SD::types::symbol_vector &symbol_vector)
TensorType< rank, dim, NumberType > tensor_evaluate_optimized(const TensorType< rank, dim, Expression > &symbol_tensor, const std::vector< NumberType > &cached_evaluation, const BatchOptimizer< NumberType > &optimizer)
bool use_symbolic_CSE(const enum OptimizationFlags &flags)
types::symbol_vector unroll_to_expression_vector(const TensorType< rank, dim, Expression > &symbol_tensor)
int get_LLVM_optimization_level(const enum OptimizationFlags &flags)
void register_functions(BatchOptimizer< NumberType > &optimizer, const T &function)
std::vector< SD::Expression > symbol_vector
std::map< SD::Expression, SD::Expression, internal::ExpressionKeyLess > substitution_map
OptimizationFlags & operator|=(OptimizationFlags &f1, const OptimizationFlags f2)
Expression operator|(const Expression &lhs, const Expression &rhs)
Expression operator&(const Expression &lhs, const Expression &rhs)
Expression substitute(const Expression &expression, const types::substitution_map &substitution_map)
std::ostream & operator<<(std::ostream &stream, const Expression &expression)
OptimizationFlags & operator&=(OptimizationFlags &f1, const OptimizationFlags f2)
constexpr ReturnType< rank, T >::value_type & extract(T &t, const ArrayType &indices)