26 #ifndef CASADI_GENERIC_SHARED_INTERNAL_HPP
27 #define CASADI_GENERIC_SHARED_INTERNAL_HPP
29 #include "generic_shared.hpp"
30 #ifdef CASADI_WITH_THREADSAFE_SYMBOLICS
34 #ifdef CASADI_WITH_THREAD
42 template<
typename Shared,
typename Internal>
43 class GenericSharedInternal {
44 friend class GenericShared<Shared, Internal>;
48 GenericSharedInternal();
51 GenericSharedInternal(
const GenericSharedInternal& node);
54 GenericSharedInternal& operator=(
const GenericSharedInternal& node);
57 virtual ~GenericSharedInternal() = 0;
60 casadi_int getCount()
const;
62 std::string debug_repr(
const Internal*)
const;
67 GenericWeakRef<Shared, Internal>* weak();
71 void initSingleton() {
72 casadi_assert_dev(
static_cast<Internal*
>(
this)->count==0);
73 static_cast<Internal*
>(
this)->count++;
77 void destroySingleton() {
78 static_cast<Internal*
>(
this)->count--;
83 B shared_from_this() {
84 casadi_assert_dev(B::test_cast(
static_cast<Internal*
>(
this)));
86 ret.own(
static_cast<Internal*
>(
this));
92 const B shared_from_this()
const {
93 casadi_assert_dev(B::test_cast(
static_cast<const Internal*
>(
this)));
95 ret.own(
const_cast<Internal*
>(
static_cast<const Internal*
>(
this)));
101 GenericWeakRef<Shared, Internal>* weak_ref_;
104 template<
typename Shared,
typename Internal>
105 class GenericWeakRefInternal :
public Internal {
108 GenericWeakRefInternal(Internal* raw);
111 ~GenericWeakRefInternal()
override;
116 #ifdef CASADI_WITH_THREADSAFE_SYMBOLICS
117 mutable std::shared_ptr<std::mutex> mutex_;
123 A getcopy(
const A& a,
124 std::map<
typename A::base_type*,
125 typename A::internal_base_type> & already_copied) {
129 already_copied.find(
const_cast<typename A::base_type*
>(a.get()));
130 if (it!=already_copied.end()) {
131 ret.own(it->second.get());
140 template<
typename Shared,
typename Internal>
141 GenericSharedInternal<Shared, Internal>::
142 GenericSharedInternal(
const GenericSharedInternal& node) {
143 static_cast<Internal*
>(
this)->count = 0;
147 template<
typename Shared,
typename Internal>
148 GenericSharedInternal<Shared, Internal>&
149 GenericSharedInternal<Shared, Internal>::
150 operator=(
const GenericSharedInternal<Shared, Internal>& node) {
155 template<
typename Shared,
typename Internal>
156 GenericSharedInternal<Shared, Internal>::GenericSharedInternal() {
157 static_cast<Internal*
>(
this)->count = 0;
161 template<
typename Shared,
typename Internal>
162 std::string GenericSharedInternal<Shared, Internal>::debug_repr(
const Internal* i)
const {
164 return str( (casadi_int)(i)) +
"/" +
static_cast<const Internal*
>(
this)->class_name();
167 template<
typename Shared,
typename Internal>
168 GenericSharedInternal<Shared, Internal>::~GenericSharedInternal() {
169 #ifdef WITH_REFCOUNT_WARNINGS
170 if (
static_cast<Internal*
>(
this)->count!=0) {
172 std::cerr <<
"Reference counting failure." <<
173 "Possible cause: Circular dependency in user code." << std::endl;
176 if (weak_ref_!=
nullptr) {
185 template<
typename Shared,
typename Internal>
186 casadi_int GenericSharedInternal<Shared, Internal>::getCount()
const {
187 return static_cast<const Internal*
>(
this)->count;
190 template<
typename Shared,
typename Internal>
191 GenericWeakRef<Shared, Internal>* GenericSharedInternal<Shared, Internal>::weak() {
192 if (weak_ref_==
nullptr) {
193 weak_ref_ =
new GenericWeakRef<Shared, Internal>(
static_cast<Internal*
>(
this));
198 template<
typename Shared,
typename Internal>
199 GenericWeakRefInternal<Shared, Internal>::GenericWeakRefInternal(Internal* raw) :
201 #ifdef CASADI_WITH_THREADSAFE_SYMBOLICS
202 , mutex_(std::make_shared<std::mutex>())
207 template<
typename Shared,
typename Internal>
208 GenericWeakRefInternal<Shared, Internal>::~GenericWeakRefInternal() {