26 #ifndef CASADI_EINSTEIN_CPP
27 #define CASADI_EINSTEIN_CPP
29 #include "einstein.hpp"
30 #include "casadi_misc.hpp"
31 #include "function_internal.hpp"
32 #include "runtime/shared.hpp"
37 const std::vector<casadi_int>& dim_c,
const std::vector<casadi_int>& dim_a,
38 const std::vector<casadi_int>& dim_b,
39 const std::vector<casadi_int>& c,
const std::vector<casadi_int>& a,
40 const std::vector<casadi_int>& b):
41 dim_c_(dim_c), dim_a_(dim_a), dim_b_(dim_b), c_(c), a_(a), b_(b) {
52 return "einstein(" + arg.at(0) +
"," + arg.at(1) +
"," + arg.at(2) +
")";
55 int Einstein::eval(
const double** arg,
double** res, casadi_int* iw,
double* w)
const {
56 return eval_gen<double>(arg, res, iw, w);
60 return eval_gen<SXElem>(arg, res, iw, w);
65 if (arg[0]!=res[0]) std::copy(arg[0], arg[0]+
dep(0).
nnz(), res[0]);
72 std::vector<std::vector<MX> >& fsens)
const {
73 for (casadi_int d=0; d<fsens.size(); ++d) {
74 fsens[d][0] = fseed[d][0]
81 std::vector<std::vector<MX> >& asens)
const {
82 for (casadi_int d=0; d<aseed.size(); ++d) {
85 asens[d][0] += aseed[d][0];
92 return eval_gen<bvec_t>(arg, res, iw, w);
100 for (casadi_int i=0;i<
n_iter_;++i) {
109 for (casadi_int j=0;j<
iter_dims_.size();++j) {
118 Contraction<bvec_t>(*c, 0, *a);
119 Contraction<bvec_t>(0, *c, *b);
130 const std::vector<casadi_int>& arg,
131 const std::vector<casadi_int>& res,
132 const std::vector<bool>& arg_is_ref,
133 std::vector<bool>& res_is_ref)
const {
136 if (arg[0]!=res[0] || arg_is_ref[0]) {
141 g.
local(
"i",
"casadi_int");
142 g <<
"for (i=0; i<" <<
n_iter_ <<
"; ++i) {\n";
145 g.
local(
"cr",
"const casadi_real",
"*");
146 g.
local(
"cs",
"const casadi_real",
"*");
147 g.
local(
"rr",
"casadi_real",
"*");
153 for (casadi_int j=0; j<
iter_dims_.size(); ++j) {
155 g.
local(
"k",
"casadi_int");
157 g.
local(
"j",
"casadi_int");
167 g <<
"*rr += *cr**cs;\n";
Helper class for C code generation.
std::string work(casadi_int n, casadi_int sz, bool is_ref) const
std::string copy(const std::string &arg, std::size_t n, const std::string &res)
Create a copy operation.
void local(const std::string &name, const std::string &type, const std::string &ref="")
Declare a local variable.
void generate(CodeGenerator &g, const std::vector< casadi_int > &arg, const std::vector< casadi_int > &res, const std::vector< bool > &arg_is_ref, std::vector< bool > &res_is_ref) const override
Generate code for the operation.
int eval(const double **arg, double **res, casadi_int *iw, double *w) const override
Evaluate the function numerically.
void ad_reverse(const std::vector< std::vector< MX > > &aseed, std::vector< std::vector< MX > > &asens) const override
Calculate reverse mode directional derivatives.
int sp_reverse(bvec_t **arg, bvec_t **res, casadi_int *iw, bvec_t *w) const override
Propagate sparsity backwards.
std::vector< casadi_int > dim_c_
Dimensions of tensors A B C.
int sp_forward(const bvec_t **arg, bvec_t **res, casadi_int *iw, bvec_t *w) const override
Propagate sparsity forward.
std::vector< casadi_int > dim_a_
std::string disp(const std::vector< std::string > &arg) const override
Print expression.
std::vector< casadi_int > dim_b_
void ad_forward(const std::vector< std::vector< MX > > &fseed, std::vector< std::vector< MX > > &fsens) const override
Calculate forward mode directional derivatives.
std::vector< casadi_int > strides_a_
Einstein(const MX &C, const MX &A, const MX &B, const std::vector< casadi_int > &dim_c, const std::vector< casadi_int > &dim_a, const std::vector< casadi_int > &dim_b, const std::vector< casadi_int > &c, const std::vector< casadi_int > &a, const std::vector< casadi_int > &b)
Constructor.
std::vector< casadi_int > iter_dims_
std::vector< casadi_int > b_
std::vector< casadi_int > c_
Einstein indices.
std::vector< casadi_int > strides_c_
int eval_sx(const SXElem **arg, SXElem **res, casadi_int *iw, SXElem *w) const override
Evaluate the function symbolically (SX)
int eval_gen(const T **arg, T **res, casadi_int *iw, T *w) const
Evaluate the function (template)
std::vector< casadi_int > a_
std::vector< casadi_int > strides_b_
void eval_mx(const std::vector< MX > &arg, std::vector< MX > &res) const override
Evaluate symbolically (MX)
virtual casadi_int ind() const
static void copy_rev(bvec_t *arg, bvec_t *res, casadi_int len)
Propagate sparsities backwards through a copy operation.
casadi_int nnz(casadi_int i=0) const
const MX & dep(casadi_int ind=0) const
dependencies - functions that have to be evaluated before this one
void set_sparsity(const Sparsity &sparsity)
Set the sparsity.
void set_dep(const MX &dep)
Set unary dependency.
static MX einstein(const MX &A, const MX &B, const MX &C, const std::vector< casadi_int > &dim_a, const std::vector< casadi_int > &dim_b, const std::vector< casadi_int > &dim_c, const std::vector< casadi_int > &a, const std::vector< casadi_int > &b, const std::vector< casadi_int > &c)
Computes an einstein dense tensor contraction.
The basic scalar symbolic class of CasADi.
void einstein_eval(casadi_int n_iter, const std::vector< casadi_int > &iter_dims, const std::vector< casadi_int > &strides_a, const std::vector< casadi_int > &strides_b, const std::vector< casadi_int > &strides_c, const T *a_in, const T *b_in, T *c_in)
casadi_int einstein_process(const T &A, const T &B, const T &C, const std::vector< casadi_int > &dim_a, const std::vector< casadi_int > &dim_b, const std::vector< casadi_int > &dim_c, const std::vector< casadi_int > &a, const std::vector< casadi_int > &b, const std::vector< casadi_int > &c, std::vector< casadi_int > &iter_dims, std::vector< casadi_int > &strides_a, std::vector< casadi_int > &strides_b, std::vector< casadi_int > &strides_c)
unsigned long long bvec_t