26 #ifndef CASADI_FUNCTION_INTERNAL_HPP
27 #define CASADI_FUNCTION_INTERNAL_HPP
29 #include "function.hpp"
32 #include "code_generator.hpp"
33 #include "importer.hpp"
34 #include "options.hpp"
35 #include "shared_object.hpp"
37 #ifdef CASADI_WITH_THREAD
38 #ifdef CASADI_WITH_THREAD_MINGW
39 #include <mingw.mutex.h>
46 #define INPUTSCHEME(name)
49 #define OUTPUTSCHEME(name)
55 std::vector<std::pair<std::string, T>> zip(
const std::vector<std::string>&
id,
56 const std::vector<T>& mat) {
57 casadi_assert_dev(
id.size()==mat.size());
58 std::vector<std::pair<std::string, T>> r(
id.size());
59 for (casadi_uint i=0; i<r.size(); ++i) r[i] = std::make_pair(
id[i], mat[i]);
66 struct CASADI_EXPORT ProtoFunctionMemory {
68 std::map<std::string, FStats> fstats;
74 void add_stat(
const std::string& s) {
75 bool added = fstats.insert(std::make_pair(s, FStats())).second;
76 casadi_assert(added,
"Duplicate stat: '" + s +
"'");
83 struct CASADI_EXPORT FunctionMemory :
public ProtoFunctionMemory {
85 FunctionMemory() : stats_available(false) {}
94 class CASADI_EXPORT ProtoFunction :
public SharedObjectInternal {
99 ProtoFunction(
const std::string& name);
104 ~ProtoFunction()
override = 0;
111 void construct(
const Dict& opts);
117 static const Options options_;
118 virtual const Options& get_options()
const {
return options_;}
122 virtual Dict generate_options(
const std::string& target)
const;
127 void print_options(std::ostream &stream)
const;
132 void print_option(
const std::string &name, std::ostream &stream)
const;
137 bool has_option(
const std::string &option_name)
const;
142 virtual void change_option(
const std::string& option_name,
const GenericType& option_value);
152 virtual void init(
const Dict& opts);
160 virtual void finalize();
163 int checkout()
const;
166 void release(
int mem)
const;
169 void* memory(
int ind)
const;
172 bool has_memory(
int ind)
const;
179 virtual void check_mem_count(casadi_int n)
const { }
184 virtual void* alloc_mem()
const {
return new ProtoFunctionMemory(); }
189 virtual int init_mem(
void* mem)
const;
194 virtual void free_mem(
void *mem)
const {
delete static_cast<ProtoFunctionMemory*
>(mem); }
197 virtual Dict get_stats(
void* mem)
const;
207 void print(
const char* fmt, ...)
const;
212 void sprint(
char* buf,
size_t buf_sz,
const char* fmt, ...)
const;
217 void format_time(
char* buffer,
double time)
const;
222 void print_time(
const std::map<std::string, FStats>& fstats)
const;
227 void serialize(SerializingStream &s)
const;
232 virtual void serialize_body(SerializingStream &s)
const;
236 virtual void serialize_type(SerializingStream &s)
const {}
241 virtual std::string serialize_base_function()
const {
258 bool regularity_check_;
267 explicit ProtoFunction(DeserializingStream& s);
269 #ifdef CASADI_WITH_THREAD
271 mutable std::mutex mtx_;
276 mutable std::vector<void*> mem_;
279 mutable std::stack<int> unused_;
288 class CASADI_EXPORT FunctionInternal :
public ProtoFunction {
289 friend class Function;
294 FunctionInternal(
const std::string& name);
299 ~FunctionInternal()
override = 0;
304 virtual std::string getAdaptorSolverName()
const {
return ""; }
310 static const Options options_;
311 const Options& get_options()
const override {
return options_;}
315 Dict generate_options(
const std::string& target)
const override;
320 void change_option(
const std::string& option_name,
const GenericType& option_value)
override;
325 void init(
const Dict& opts)
override;
330 void finalize()
override;
335 void* alloc_mem()
const override {
return new FunctionMemory(); }
340 void free_mem(
void *mem)
const override {
delete static_cast<FunctionMemory*
>(mem); }
343 Dict get_stats(
void* mem)
const override;
348 Function
self()
const {
return shared_from_this<Function>();}
351 virtual Function factory(
const std::string& name,
352 const std::vector<std::string>& s_in,
353 const std::vector<std::string>& s_out,
355 const Dict& opts)
const;
358 virtual std::vector<std::string> get_function()
const;
361 virtual const Function& get_function(
const std::string &name)
const;
364 virtual bool has_function(
const std::string& fname)
const {
return false;}
367 void add_embedded(std::map<FunctionInternal*, Function>& all_fun,
368 const Function& dep, casadi_int max_depth)
const;
371 virtual void find(std::map<FunctionInternal*, Function>& all_fun, casadi_int max_depth)
const {}
381 virtual std::vector<bool> which_depends(
const std::string& s_in,
382 const std::vector<std::string>& s_out,
383 casadi_int order,
bool tr=
false)
const;
389 virtual bool has_spfwd()
const {
return false;}
390 virtual bool has_sprev()
const {
return false;}
397 int eval_gen(
const double** arg,
double** res, casadi_int* iw,
double* w,
void* mem,
398 bool always_inline,
bool never_inline)
const;
399 virtual int eval(
const double** arg,
double** res, casadi_int* iw,
double* w,
void* mem)
const;
405 virtual int eval_sx(
const SXElem** arg, SXElem** res,
406 casadi_int* iw, SXElem* w,
void* mem,
bool always_inline,
bool never_inline)
const;
412 bool always_inline,
bool never_inline)
const;
418 virtual std::vector<DM> eval_dm(
const std::vector<DM>& arg)
const;
419 virtual bool has_eval_dm()
const {
return false;}
426 int eval_gen(
const SXElem** arg, SXElem** res, casadi_int* iw, SXElem* w,
void* mem,
427 bool always_inline,
bool never_inline)
const {
428 return eval_sx(arg, res, iw, w, mem, always_inline, never_inline);
430 int eval_gen(
const bvec_t** arg, bvec_t** res, casadi_int* iw, bvec_t* w,
void* mem,
431 bool always_inline,
bool never_inline)
const {
432 return sp_forward(arg, res, iw, w, mem);
441 bool always_inline,
bool never_inline)
const;
444 void call_gen(
const std::vector<Matrix<D> >& arg, std::vector<Matrix<D> >& res,
445 casadi_int npar,
bool always_inline,
bool never_inline)
const;
452 void call(
const std::vector<M>& arg, std::vector<M>& res,
453 bool always_inline,
bool never_inline)
const;
461 static bool check_mat(
const Sparsity& arg,
const Sparsity& inp, casadi_int& npar);
474 void check_arg(
const std::vector<M>& arg, casadi_int& npar)
const;
487 void check_res(
const std::vector<M>& res, casadi_int& npar)
const;
498 template<
typename M>
bool
499 matching_arg(
const std::vector<M>& arg, casadi_int& npar)
const;
509 template<
typename M>
bool
510 matching_res(
const std::vector<M>& arg, casadi_int& npar)
const;
515 template<
typename M> std::vector<M>
516 replace_arg(
const std::vector<M>& arg, casadi_int npar)
const;
521 template<
typename M> std::vector<M>
522 project_arg(
const std::vector<M>& arg, casadi_int npar)
const;
527 template<
typename M> std::vector<M>
528 project_res(
const std::vector<M>& arg, casadi_int npar)
const;
533 template<
typename M> std::vector<M>
534 replace_res(
const std::vector<M>& res, casadi_int npar)
const;
539 template<
typename M> std::vector<std::vector<M>>
540 replace_fseed(
const std::vector<std::vector<M>>& fseed, casadi_int npar)
const;
545 template<
typename M> std::vector<std::vector<M>>
546 replace_aseed(
const std::vector<std::vector<M>>& aseed, casadi_int npar)
const;
553 std::map<std::string, M> convert_arg(
const std::vector<M>& arg)
const;
555 std::vector<M> convert_arg(
const std::map<std::string, M>& arg)
const;
557 std::map<std::string, M> convert_res(
const std::vector<M>& res)
const;
559 std::vector<M> convert_res(
const std::map<std::string, M>& res)
const;
566 std::vector<double> nz_in(
const std::vector<DM>& arg)
const;
567 std::vector<double> nz_out(
const std::vector<DM>& res)
const;
568 std::vector<DM> nz_in(
const std::vector<double>& arg)
const;
569 std::vector<DM> nz_out(
const std::vector<double>& res)
const;
576 virtual void call_forward(
const std::vector<MX>& arg,
const std::vector<MX>& res,
577 const std::vector<std::vector<MX> >& fseed,
578 std::vector<std::vector<MX> >& fsens,
579 bool always_inline,
bool never_inline)
const;
580 virtual void call_forward(
const std::vector<SX>& arg,
const std::vector<SX>& res,
581 const std::vector<std::vector<SX> >& fseed,
582 std::vector<std::vector<SX> >& fsens,
583 bool always_inline,
bool never_inline)
const;
590 virtual void call_reverse(
const std::vector<MX>& arg,
const std::vector<MX>& res,
591 const std::vector<std::vector<MX> >& aseed,
592 std::vector<std::vector<MX> >& asens,
593 bool always_inline,
bool never_inline)
const;
594 virtual void call_reverse(
const std::vector<SX>& arg,
const std::vector<SX>& res,
595 const std::vector<std::vector<SX> >& aseed,
596 std::vector<std::vector<SX> >& asens,
597 bool always_inline,
bool never_inline)
const;
603 std::vector<MX> mapsum_mx(
const std::vector<MX > &arg,
const std::string& parallelization);
608 virtual bool uses_output()
const {
return false;}
614 Function jacobian()
const;
615 virtual bool has_jacobian()
const {
return false;}
616 virtual Function get_jacobian(
const std::string& name,
617 const std::vector<std::string>& inames,
618 const std::vector<std::string>& onames,
619 const Dict& opts)
const;
627 Sparsity& jac_sparsity(casadi_int oind, casadi_int iind,
bool compact,
bool symmetric)
const;
628 virtual bool has_jac_sparsity(casadi_int oind, casadi_int iind)
const {
return false;}
629 virtual Sparsity get_jac_sparsity(casadi_int oind, casadi_int iind,
bool symmetric)
const;
633 static std::string forward_name(
const std::string& fcn, casadi_int nfwd) {
634 return "fwd" + str(nfwd) +
"_" + fcn;
638 std::string diff_prefix(
const std::string& prefix)
const;
648 Function forward(casadi_int nfwd)
const;
649 virtual bool has_forward(casadi_int nfwd)
const {
return false;}
650 virtual Function get_forward(casadi_int nfwd,
const std::string& name,
651 const std::vector<std::string>& inames,
652 const std::vector<std::string>& onames,
653 const Dict& opts)
const;
657 static std::string reverse_name(
const std::string& fcn, casadi_int nadj) {
658 return "adj" + str(nadj) +
"_" + fcn;
669 Function reverse(casadi_int nadj)
const;
670 virtual bool has_reverse(casadi_int nadj)
const {
return false;}
671 virtual Function get_reverse(casadi_int nadj,
const std::string& name,
672 const std::vector<std::string>& inames,
673 const std::vector<std::string>& onames,
674 const Dict& opts)
const;
680 template<
typename MatType>
681 static MatType ensure_stacked(
const MatType& v,
const Sparsity& sp, casadi_int n);
686 virtual Function slice(
const std::string& name,
const std::vector<casadi_int>& order_in,
687 const std::vector<casadi_int>& order_out,
const Dict& opts)
const;
692 virtual const Function& oracle()
const;
697 bool has_derivative()
const;
702 virtual double ad_weight()
const;
709 virtual double sp_weight()
const;
715 virtual const SX sx_in(casadi_int ind)
const;
716 virtual const SX sx_out(casadi_int ind)
const;
717 virtual const std::vector<SX> sx_in()
const;
718 virtual const std::vector<SX> sx_out()
const;
719 virtual const MX mx_in(casadi_int ind)
const;
720 virtual const MX mx_out(casadi_int ind)
const;
721 virtual const std::vector<MX> mx_in()
const;
722 virtual const std::vector<MX> mx_out()
const;
723 const DM dm_in(casadi_int ind)
const;
724 const DM dm_out(casadi_int ind)
const;
725 const std::vector<DM> dm_in()
const;
726 const std::vector<DM> dm_out()
const;
730 virtual std::vector<MX> free_mx()
const;
733 virtual std::vector<SX> free_sx()
const;
738 virtual bool has_free()
const {
return false;}
743 virtual void generate_lifted(Function& vdef_fcn, Function& vinit_fcn)
const;
748 virtual casadi_int n_instructions()
const;
753 virtual casadi_int instruction_id(casadi_int k)
const;
758 virtual std::vector<casadi_int> instruction_input(casadi_int k)
const;
763 virtual double instruction_constant(casadi_int k)
const;
768 virtual std::vector<casadi_int> instruction_output(casadi_int k)
const;
773 virtual casadi_int n_nodes()
const;
778 virtual MX instruction_MX(casadi_int k)
const;
783 virtual SX instructions_sx()
const;
788 Function wrap()
const;
793 Function wrap_as_needed(
const Dict& opts)
const;
803 bool incache(
const std::string& fname, Function& f,
const std::string& suffix=
"")
const;
808 void tocache(
const Function& f,
const std::string& suffix=
"")
const;
814 void tocache_if_missing(Function& f,
const std::string& suffix=
"")
const;
819 void codegen(CodeGenerator& g,
const std::string& fname)
const;
824 void codegen_meta(CodeGenerator& g)
const;
829 void codegen_sparsities(CodeGenerator& g)
const;
834 virtual std::string codegen_name(
const CodeGenerator& g,
bool ns=
true)
const;
839 std::string codegen_mem(CodeGenerator& g,
const std::string& index=
"mem")
const;
844 virtual void codegen_incref(CodeGenerator& g)
const {}
849 virtual void codegen_decref(CodeGenerator& g)
const {}
854 virtual void codegen_alloc_mem(CodeGenerator& g)
const;
859 virtual void codegen_init_mem(CodeGenerator& g)
const;
864 virtual void codegen_free_mem(CodeGenerator& g)
const {}
869 virtual void codegen_checkout(CodeGenerator& g)
const;
874 virtual void codegen_release(CodeGenerator& g)
const;
879 std::string signature(
const std::string& fname)
const;
884 std::string signature_unrolled(
const std::string& fname)
const;
889 virtual void codegen_declarations(CodeGenerator& g)
const;
894 virtual void codegen_body(CodeGenerator& g)
const;
899 virtual std::string codegen_mem_type()
const {
return ""; }
904 virtual std::string generate_dependencies(
const std::string& fname,
const Dict& opts)
const;
909 virtual bool has_codegen()
const {
return false;}
914 virtual void jit_dependencies(
const std::string& fname) {}
919 virtual void export_code(
const std::string& lang,
920 std::ostream &stream,
const Dict& options)
const;
925 void serialize_type(SerializingStream &s)
const override;
930 void serialize_body(SerializingStream &s)
const override;
935 void disp(std::ostream& stream,
bool more)
const override;
940 virtual void disp_more(std::ostream& stream)
const {}
945 std::string definition()
const;
950 void print_dimensions(std::ostream &stream)
const;
955 virtual std::vector<std::string> get_free()
const;
960 void get_partition(casadi_int iind, casadi_int oind, Sparsity& D1, Sparsity& D2,
961 bool compact,
bool symmetric,
962 bool allow_forward,
bool allow_reverse)
const;
968 casadi_int nnz_in()
const;
969 casadi_int nnz_in(casadi_int ind)
const {
return sparsity_in(ind).nnz(); }
970 casadi_int nnz_out()
const;
971 casadi_int nnz_out(casadi_int ind)
const {
return sparsity_out(ind).nnz(); }
978 casadi_int numel_in()
const;
979 casadi_int numel_in(casadi_int ind)
const {
return sparsity_in(ind).numel(); }
980 casadi_int numel_out(casadi_int ind)
const {
return sparsity_out(ind).numel(); }
981 casadi_int numel_out()
const;
988 casadi_int size1_in(casadi_int ind)
const {
return sparsity_in(ind).size1(); }
989 casadi_int size2_in(casadi_int ind)
const {
return sparsity_in(ind).size2(); }
990 casadi_int size1_out(casadi_int ind)
const {
return sparsity_out(ind).size1(); }
991 casadi_int size2_out(casadi_int ind)
const {
return sparsity_out(ind).size2(); }
992 std::pair<casadi_int, casadi_int> size_in(casadi_int ind)
const {
993 return sparsity_in(ind).size();
995 std::pair<casadi_int, casadi_int> size_out(casadi_int ind)
const {
996 return sparsity_out(ind).size();
1004 const Sparsity& sparsity_in(casadi_int ind)
const {
return sparsity_in_.at(ind); }
1005 const Sparsity& sparsity_out(casadi_int ind)
const {
return sparsity_out_.at(ind); }
1012 bool all_scalar()
const;
1015 virtual bool jac_is_symm(casadi_int oind, casadi_int iind)
const;
1018 Sparsity to_compact(casadi_int oind, casadi_int iind,
const Sparsity& sp)
const;
1021 Sparsity from_compact(casadi_int oind, casadi_int iind,
const Sparsity& sp)
const;
1025 Sparsity get_jac_sparsity_gen(casadi_int oind, casadi_int iind)
const;
1028 Sparsity get_jac_sparsity_hierarchical(casadi_int oind, casadi_int iind)
const;
1033 Sparsity get_jac_sparsity_hierarchical_symm(casadi_int oind, casadi_int iind)
const;
1036 virtual std::vector<MX> symbolic_output(
const std::vector<MX>& arg)
const;
1042 virtual size_t get_n_in();
1043 virtual size_t get_n_out();
1051 virtual std::string get_name_in(casadi_int i);
1052 virtual std::string get_name_out(casadi_int i);
1058 virtual double get_default_in(casadi_int ind)
const {
1065 virtual double get_max_in(casadi_int ind)
const {
1072 virtual double get_min_in(casadi_int ind)
const {
1076 virtual std::vector<double> get_nominal_in(casadi_int ind)
const {
1077 return std::vector<double>(nnz_in(ind), 1.);
1080 virtual std::vector<double> get_nominal_out(casadi_int ind)
const {
1081 return std::vector<double>(nnz_out(ind), 1.);
1087 virtual double get_reltol()
const {
1094 virtual double get_abstol()
const {
1101 virtual Sparsity get_sparsity_in(casadi_int i);
1106 virtual Sparsity get_sparsity_out(casadi_int i);
1111 virtual bool get_diff_in(casadi_int i) {
return true; }
1116 virtual bool get_diff_out(casadi_int i) {
return true; }
1121 casadi_int index_in(
const std::string &name)
const {
1122 for (casadi_int i=0; i<name_in_.size(); ++i) {
1123 if (name_in_[i]==name)
return i;
1125 casadi_error(
"FunctionInternal::index_in: could not find entry \""
1126 + name +
"\". Available names are: " + str(name_in_) +
".");
1133 casadi_int index_out(
const std::string &name)
const {
1134 for (casadi_int i=0; i<name_out_.size(); ++i) {
1135 if (name_out_[i]==name)
return i;
1137 casadi_error(
"FunctionInternal::index_out: could not find entry \""
1138 + name +
"\". Available names are: " + str(name_out_) +
".");
1145 virtual int sp_forward(
const bvec_t** arg, bvec_t** res,
1146 casadi_int* iw, bvec_t* w,
void* mem)
const;
1151 virtual int sp_forward_block(
const bvec_t** arg, bvec_t** res,
1152 casadi_int* iw, bvec_t* w,
void* mem, casadi_int oind, casadi_int iind)
const;
1157 virtual int sp_reverse(bvec_t** arg, bvec_t** res, casadi_int* iw, bvec_t* w,
void* mem)
const;
1162 void sz_work(
size_t& sz_arg,
size_t& sz_res,
size_t& sz_iw,
size_t& sz_w)
const;
1167 size_t sz_arg()
const {
return sz_arg_per_ + sz_arg_tmp_;}
1172 size_t sz_res()
const {
return sz_res_per_ + sz_res_tmp_;}
1177 size_t sz_iw()
const {
return sz_iw_per_ + sz_iw_tmp_;}
1182 size_t sz_w()
const {
return sz_w_per_ + sz_w_tmp_;}
1188 virtual size_t codegen_sz_arg(
const CodeGenerator& g)
const;
1189 virtual size_t codegen_sz_res(
const CodeGenerator& g)
const;
1190 virtual size_t codegen_sz_iw(
const CodeGenerator& g)
const;
1191 virtual size_t codegen_sz_w(
const CodeGenerator& g)
const;
1197 void alloc_arg(
size_t sz_arg,
bool persistent=
false);
1202 void alloc_res(
size_t sz_res,
bool persistent=
false);
1207 void alloc_iw(
size_t sz_iw,
bool persistent=
false);
1212 void alloc_w(
size_t sz_w,
bool persistent=
false);
1217 void alloc(
const Function& f,
bool persistent=
false,
int num_threads=1);
1222 virtual void set_work(
void* mem,
const double**& arg,
double**& res,
1223 casadi_int*& iw,
double*& w)
const {}
1228 virtual void set_temp(
void* mem,
const double** arg,
double** res,
1229 casadi_int* iw,
double* w)
const {}
1234 void setup(
void* mem,
const double** arg,
double** res, casadi_int* iw,
double* w)
const;
1240 virtual bool fwdViaJac(casadi_int nfwd)
const;
1241 virtual bool adjViaJac(casadi_int nadj)
const;
1245 virtual Dict info()
const;
1250 Function map(casadi_int n,
const std::string& parallelization)
const;
1255 void generate_in(
const std::string& fname,
const double** arg)
const;
1256 void generate_out(
const std::string& fname,
double** res)
const;
1258 bool always_inline_, never_inline_;
1261 size_t n_in_, n_out_;
1264 std::vector<bool> is_diff_in_, is_diff_out_;
1267 std::vector<Sparsity> sparsity_in_, sparsity_out_;
1270 std::vector<std::string> name_in_, name_out_;
1285 std::string jit_serialize_;
1290 std::string jit_name_;
1292 std::string jit_base_name_;
1297 bool jit_temp_suffix_;
1307 casadi_checkout_t checkout_;
1312 casadi_release_t release_;
1330 mutable WeakCache<std::string, Function> cache_;
1333 mutable std::vector<Sparsity> jac_sparsity_[2];
1335 #ifdef CASADI_WITH_THREADSAFE_SYMBOLICS
1337 mutable std::mutex jac_sparsity_mtx_;
1341 Function derivative_of_;
1347 std::string compiler_plugin_;
1352 double jac_penalty_;
1355 bool enable_forward_, enable_reverse_, enable_jacobian_, enable_fd_;
1356 bool enable_forward_op_, enable_reverse_op_, enable_jacobian_op_, enable_fd_op_;
1359 double ad_weight_, ad_weight_sp_;
1362 casadi_int max_num_dir_;
1374 std::string fd_method_;
1384 bool dump_in_, dump_out_, dump_;
1387 std::string dump_dir_;
1390 std::string dump_format_;
1393 Dict forward_options_, reverse_options_, jacobian_options_, der_options_;
1396 Function custom_jacobian_;
1399 #ifdef CASADI_WITH_THREAD
1400 mutable std::atomic<casadi_int> dump_count_;
1402 mutable casadi_int dump_count_;
1408 virtual bool is_a(
const std::string& type,
bool recursive)
const;
1413 virtual void merge(
const std::vector<MX>& arg,
1414 std::vector<MX>& subs_from, std::vector<MX>& subs_to)
const;
1419 template<
typename MatType>
1420 static bool purgable(
const std::vector<MatType>& seed);
1425 template<
typename MatType>
1426 std::vector<std::vector<MatType> >
1427 fwd_seed(casadi_int nfwd)
const;
1432 template<
typename MatType>
1433 std::vector<std::vector<MatType> >
1434 symbolicAdjSeed(casadi_int nadj,
const std::vector<MatType>& v)
const;
1441 explicit FunctionInternal(DeserializingStream& e);
1446 static Function deserialize(DeserializingStream& s);
1447 static std::map<std::string, ProtoFunction* (*)(DeserializingStream&)> deserialize_map;
1452 void print_in(std::ostream &stream,
const double** arg,
bool truncate)
const;
1457 void print_out(std::ostream &stream,
double** res,
bool truncate)
const;
1463 void set_jac_sparsity(casadi_int oind, casadi_int iind,
const Sparsity& sp);
1468 casadi_int get_dump_id()
const;
1469 void dump_in(casadi_int
id,
const double** arg)
const;
1470 void dump_out(casadi_int
id,
double** res)
const;
1477 size_t sz_arg_per_, sz_res_per_, sz_iw_per_, sz_w_per_;
1482 size_t sz_arg_tmp_, sz_res_tmp_, sz_iw_tmp_, sz_w_tmp_;
1486 template<
typename MatType>
1487 bool FunctionInternal::purgable(
const std::vector<MatType>& v) {
1488 for (
auto i=v.begin(); i!=v.end(); ++i) {
1489 if (!i->is_zero())
return false;
1494 template<
typename MatType>
1495 std::vector<std::vector<MatType> >
1497 fwd_seed(casadi_int nfwd)
const {
1498 std::vector<std::vector<MatType>> fseed(nfwd);
1499 for (casadi_int dir=0; dir<nfwd; ++dir) {
1500 fseed[dir].resize(n_in_);
1501 for (casadi_int iind=0; iind<n_in_; ++iind) {
1502 std::string n =
"f" + str(dir) +
"_" + name_in_[iind];
1503 Sparsity sp = is_diff_in_[iind] ? sparsity_in(iind) : Sparsity(size_in(iind));
1504 fseed[dir][iind] = MatType::sym(n, sp);
1510 template<
typename MatType>
1511 std::vector<std::vector<MatType> >
1513 symbolicAdjSeed(casadi_int nadj,
const std::vector<MatType>& v)
const {
1514 std::vector<std::vector<MatType> > aseed(nadj, v);
1515 for (casadi_int dir=0; dir<nadj; ++dir) {
1518 for (
typename std::vector<MatType>::iterator i=aseed[dir].begin();
1519 i!=aseed[dir].end();
1522 std::stringstream ss;
1524 if (nadj>1) ss << dir <<
"_";
1528 *i = MatType::sym(ss.str(), is_diff_out_[oind] ? i->sparsity() : Sparsity(i->size()));
1535 template<
typename M>
1536 void FunctionInternal::call(
const std::vector<M>& arg, std::vector<M>& res,
1537 bool always_inline,
bool never_inline)
const {
1541 bool matrix_call =
false;
1542 std::pair<casadi_int, casadi_int> sz;
1543 for (
auto&& a : arg) {
1544 if (!a.is_scalar() && !a.is_empty()) {
1549 }
else if (a.size()!=sz) {
1551 matrix_call =
false;
1562 for (
auto&& a : res) a = z;
1564 std::vector<M> arg1 = arg, res1;
1565 for (casadi_int c=0; c<sz.second; ++c) {
1566 for (casadi_int r=0; r<sz.first; ++r) {
1568 for (casadi_int i=0; i<arg.size(); ++i) {
1569 if (arg[i].size()==sz) arg1[i] = arg[i](r, c);
1572 call(arg1, res1, always_inline, never_inline);
1574 casadi_assert_dev(res.size() == res1.size());
1575 for (casadi_int i=0; i<res.size(); ++i) res[i](r, c) = res1[i];
1584 casadi_int npar = 1;
1585 if (!matching_arg(arg, npar)) {
1586 return call(replace_arg(arg, npar), res, always_inline, never_inline);
1590 call_gen(arg, res, npar, always_inline, never_inline);
1593 template<
typename M>
1594 std::vector<M> FunctionInternal::
1595 project_arg(
const std::vector<M>& arg, casadi_int npar)
const {
1596 casadi_assert_dev(arg.size()==n_in_);
1599 std::vector<bool> mapped(n_in_);
1600 for (casadi_int i=0; i<n_in_; ++i) {
1601 mapped[i] = arg[i].size2()!=size2_in(i);
1605 std::vector<bool> matching(n_in_);
1606 bool any_mismatch =
false;
1607 for (casadi_int i=0; i<n_in_; ++i) {
1609 matching[i] = arg[i].sparsity().is_stacked(sparsity_in(i), npar);
1611 matching[i] = arg[i].sparsity()==sparsity_in(i);
1613 any_mismatch = any_mismatch || !matching[i];
1618 std::vector<M> arg2(arg);
1619 for (casadi_int i=0; i<n_in_; ++i) {
1622 arg2[i] = project(arg2[i], repmat(sparsity_in(i), 1, npar));
1624 arg2[i] = project(arg2[i], sparsity_in(i));
1633 template<
typename M>
1634 std::vector<M> FunctionInternal::
1635 project_res(
const std::vector<M>& arg, casadi_int npar)
const {
1639 template<
typename D>
1640 void FunctionInternal::
1641 call_gen(
const std::vector<Matrix<D> >& arg, std::vector<Matrix<D> >& res,
1642 casadi_int npar,
bool always_inline,
bool never_inline)
const {
1643 std::vector< Matrix<D> > arg2 = project_arg(arg, npar);
1646 std::vector<bool> mapped(n_in_);
1647 for (casadi_int i=0; i<n_in_; ++i) {
1648 mapped[i] = arg[i].size2()!=size2_in(i);
1653 for (casadi_int i=0; i<n_out_; ++i) {
1654 if (!res[i].sparsity().is_stacked(sparsity_out(i), npar)) {
1660 std::vector<casadi_int> iw_tmp(sz_iw());
1661 std::vector<D> w_tmp(sz_w());
1664 std::vector<const D*> argp(sz_arg());
1665 for (casadi_int i=0; i<n_in_; ++i) argp[i]=get_ptr(arg2[i]);
1668 std::vector<D*> resp(sz_res());
1669 for (casadi_int i=0; i<n_out_; ++i) resp[i]=get_ptr(res[i]);
1672 for (casadi_int p=0; p<npar; ++p) {
1674 if (eval_gen(get_ptr(argp), get_ptr(resp),
1675 get_ptr(iw_tmp), get_ptr(w_tmp), memory(0),
1676 always_inline, never_inline)) {
1677 if (error_on_fail_) casadi_error(
"Evaluation failed");
1680 if (p==npar-1)
break;
1681 for (casadi_int i=0; i<n_in_; ++i)
if (mapped[i]) argp[i] += nnz_in(i);
1682 for (casadi_int i=0; i<n_out_; ++i) resp[i] += nnz_out(i);
1686 template<
typename M>
1687 void FunctionInternal::check_arg(
const std::vector<M>& arg, casadi_int& npar)
const {
1688 casadi_assert(arg.size()==n_in_,
"Incorrect number of inputs: Expected "
1689 + str(n_in_) +
", got " + str(arg.size()));
1690 for (casadi_int i=0; i<n_in_; ++i) {
1691 if (!check_mat(arg[i].sparsity(), sparsity_in(i), npar)) {
1693 std::string d_arg = str(arg[i].size1()) +
"-by-" + str(arg[i].size2());
1694 std::string d_in = str(size1_in(i)) +
"-by-" + str(size2_in(i));
1695 std::string e =
"Input " + str(i) +
" (" + name_in_[i] +
") has mismatching shape. "
1696 "Got " + d_arg +
". Allowed dimensions, in general, are:\n"
1697 " - The input dimension N-by-M (here " + d_in +
")\n"
1698 " - A scalar, i.e. 1-by-1\n"
1699 " - M-by-N if N=1 or M=1 (i.e. a transposed vector)\n"
1700 " - N-by-M1 if K*M1=M for some K (argument repeated horizontally)\n";
1702 e +=
" - N-by-P*M, indicating evaluation with multiple arguments (P must be a "
1703 "multiple of " + str(npar) +
" for consistency with previous inputs)";
1710 template<
typename M>
1711 void FunctionInternal::check_res(
const std::vector<M>& res, casadi_int& npar)
const {
1712 casadi_assert(res.size()==n_out_,
"Incorrect number of outputs: Expected "
1713 + str(n_out_) +
", got " + str(res.size()));
1714 for (casadi_int i=0; i<n_out_; ++i) {
1715 casadi_assert(check_mat(res[i].sparsity(), sparsity_out(i), npar),
1716 "Output " + str(i) +
" (" + name_out_[i] +
") has mismatching shape. "
1717 "Expected " + str(size_out(i)) +
", got " + str(res[i].size()));
1721 template<
typename M>
1722 bool FunctionInternal::matching_arg(
const std::vector<M>& arg, casadi_int& npar)
const {
1723 check_arg(arg, npar);
1724 for (casadi_int i=0; i<n_in_; ++i) {
1725 if (arg.at(i).size1()!=size1_in(i))
return false;
1726 if (arg.at(i).size2()!=size2_in(i) && arg.at(i).size2()!=npar*size2_in(i))
return false;
1731 template<
typename M>
1732 bool FunctionInternal::matching_res(
const std::vector<M>& res, casadi_int& npar)
const {
1733 check_res(res, npar);
1734 for (casadi_int i=0; i<n_out_; ++i) {
1735 if (res.at(i).size1()!=size1_out(i))
return false;
1736 if (res.at(i).size2()!=size2_out(i) && res.at(i).size2()!=npar*size2_out(i))
return false;
1741 template<
typename M>
1742 M replace_mat(
const M& arg,
const Sparsity& inp, casadi_int npar) {
1743 if (arg.size()==inp.size()) {
1746 }
else if (arg.is_empty()) {
1748 return M(inp.size());
1749 }
else if (arg.is_scalar()) {
1752 }
else if (arg.is_vector() && inp.size()==std::make_pair(arg.size2(), arg.size1())) {
1755 }
else if (arg.size1()==inp.size1() && arg.size2()>0 && inp.size2()>0
1756 && inp.size2()%arg.size2()==0) {
1758 return repmat(arg, 1, inp.size2()/arg.size2());
1760 casadi_assert_dev(npar!=-1);
1762 return repmat(arg, 1, (npar*inp.size2())/arg.size2());
1766 template<
typename M>
1767 std::vector<M> FunctionInternal::
1768 replace_arg(
const std::vector<M>& arg, casadi_int npar)
const {
1769 std::vector<M> r(arg.size());
1770 for (casadi_int i=0; i<r.size(); ++i) r[i] = replace_mat(arg[i], sparsity_in(i), npar);
1774 template<
typename M>
1775 std::vector<M> FunctionInternal::
1776 replace_res(
const std::vector<M>& res, casadi_int npar)
const {
1777 std::vector<M> r(res.size());
1778 for (casadi_int i=0; i<r.size(); ++i) r[i] = replace_mat(res[i], sparsity_out(i), npar);
1782 template<
typename M>
1783 std::vector<std::vector<M> > FunctionInternal::
1784 replace_fseed(
const std::vector<std::vector<M> >& fseed, casadi_int npar)
const {
1785 std::vector<std::vector<M> > r(fseed.size());
1786 for (casadi_int d=0; d<r.size(); ++d) r[d] = replace_arg(fseed[d], npar);
1790 template<
typename M>
1791 std::vector<std::vector<M> > FunctionInternal::
1792 replace_aseed(
const std::vector<std::vector<M> >& aseed, casadi_int npar)
const {
1793 std::vector<std::vector<M> > r(aseed.size());
1794 for (casadi_int d=0; d<r.size(); ++d) r[d] = replace_res(aseed[d], npar);
1798 template<
typename M>
1799 std::map<std::string, M> FunctionInternal::
1800 convert_arg(
const std::vector<M>& arg)
const {
1801 casadi_assert(arg.size()==n_in_,
"Incorrect number of inputs: Expected "
1802 + str(n_in_) +
", got " + str(arg.size()));
1803 std::map<std::string, M> ret;
1804 for (casadi_int i=0;i<n_in_;++i) {
1805 ret[name_in_[i]] = arg[i];
1810 template<
typename M>
1811 std::vector<M> FunctionInternal::
1812 convert_arg(
const std::map<std::string, M>& arg)
const {
1814 std::vector<M> arg_v(n_in_);
1815 for (casadi_int i=0; i<arg_v.size(); ++i) {
1816 arg_v[i] = get_default_in(i);
1820 for (
auto&& e : arg) {
1821 arg_v.at(index_in(e.first)) = e.second;
1827 template<
typename M>
1828 std::map<std::string, M> FunctionInternal::
1829 convert_res(
const std::vector<M>& res)
const {
1830 casadi_assert(res.size()==n_out_,
"Incorrect number of outputs: Expected "
1831 + str(n_out_) +
", got " + str(res.size()));
1832 std::map<std::string, M> ret;
1833 for (casadi_int i=0;i<n_out_;++i) {
1834 ret[name_out_[i]] = res[i];
1839 template<
typename M>
1840 std::vector<M> FunctionInternal::
1841 convert_res(
const std::map<std::string, M>& res)
const {
1843 std::vector<M> res_v(n_out_);
1844 for (casadi_int i=0; i<res_v.size(); ++i) {
1845 res_v[i] = std::numeric_limits<double>::quiet_NaN();
1849 for (
auto&& e : res) {
1851 res_v.at(index_out(e.first)) = a;
1856 template<
typename MatType>
1857 MatType FunctionInternal::ensure_stacked(
const MatType& v,
const Sparsity& sp, casadi_int n) {
1859 if (v.size1() == sp.size1() && v.size2() == n * sp.size2()) {
1861 if (v.nnz() != 0 && !v.sparsity().is_stacked(sp, n)) {
1862 return project(v, repmat(sp, 1, n));
1866 casadi_assert_dev(v.is_empty());
1867 return MatType(sp.size1(), sp.size2() * n);
std::map< std::string, std::vector< std::string > > AuxOut
static Matrix< Scalar > zeros(casadi_int nrow=1, casadi_int ncol=1)
Create a dense matrix or a matrix with specified sparsity with all entries zero.
std::vector< MX > MXVector
GenericType::Dict Dict
C++ equivalent of Python's dict or MATLAB's struct.