// Copyright (C) 2007 Davis E. King (davis@dlib.net) // License: Boost Software License See LICENSE.txt for the full license. #ifndef DLIB_SHARED_PTr_ #define DLIB_SHARED_PTr_ #include #include #include #include // for the exceptions #include "../algs.h" #include "shared_ptr_abstract.h" namespace dlib { // ---------------------------------------------------------------------------------------- class bad_weak_ptr: public std::exception {}; // ---------------------------------------------------------------------------------------- template class weak_ptr; // ---------------------------------------------------------------------------------------- struct shared_ptr_deleter { virtual void del(const void* p) = 0; virtual ~shared_ptr_deleter() {} virtual void* get_deleter_void(const std::type_info& t) const = 0; /*! ensures - if (the deleter in this object has typeid() == t) then - returns a pointer to the deleter - else - return 0 !*/ }; struct shared_ptr_node; struct weak_ptr_node { weak_ptr_node ( shared_ptr_node* sn ) : ref_count(1), shared_node(sn) { DLIB_ASSERT(sn != 0,""); } long ref_count; shared_ptr_node* shared_node; }; struct shared_ptr_node { shared_ptr_node( ) : ref_count(1), del(0), weak_node(0) {} long ref_count; shared_ptr_deleter* del; weak_ptr_node* weak_node; }; struct shared_ptr_static_cast {}; struct shared_ptr_const_cast {}; struct shared_ptr_dynamic_cast {}; // ---------------------------------------------------------------------------------------- template class shared_ptr { /*! CONVENTION - get() == data - unique() == (shared_node != 0) && (shared_node->ref_count == 1) - if (shared_node != 0) then - use_count() == shared_node->ref_count - get() == a valid pointer - if (we are supposed to use the deleter) then - shared_node->del == the deleter to use - else - shared_node->del == 0 - else - use_count() == 0 - get() == 0 - if (there are any weak_ptrs that reference this->data) then - shared_node->weak_node->ref_count == the number of referencing weak_ptrs - else - shared_node->weak_node == 0 !*/ template struct deleter_template : public shared_ptr_deleter { deleter_template(const D& d_) : d(d_) {} void del(const void* p) { d((T*)p); } D d; void* get_deleter_void(const std::type_info& t) const { if (typeid(D) == t) return (void*)&d; else return 0; } }; struct default_deleter : public shared_ptr_deleter { void del(const void* p) { delete ((T*)p); } void* get_deleter_void(const std::type_info&) const { return 0; } }; public: typedef T element_type; shared_ptr( ) : data(0), shared_node(0) {} template explicit shared_ptr( Y* p ) : data(p) { DLIB_ASSERT(p != 0, "\tshared_ptr::shared_ptr(p)" << "\n\tp can't be null" << "\n\tthis: " << this ); try { shared_node = new shared_ptr_node; shared_node->del = new default_deleter; } catch (...) { delete p; throw; } } template shared_ptr( Y* p, const D& d ) : data(p) { DLIB_ASSERT(p != 0, "\tshared_ptr::shared_ptr(p,d)" << "\n\tp can't be null" << "\n\tthis: " << this ); try { shared_node = 0; shared_node = new shared_ptr_node; shared_node->del = new deleter_template(d); } catch (...) { if (shared_node) delete shared_node; d(p); throw; } } ~shared_ptr() { if ( shared_node != 0) { if (shared_node->ref_count == 1) { // delete the data in the appropriate way shared_node->del->del(data); delete shared_node->del; // notify any weak_ptrs that the data has now expired if (shared_node->weak_node) shared_node->weak_node->shared_node = 0; // finally delete the shared_node delete shared_node; } else { shared_node->ref_count -= 1; } } } shared_ptr( const shared_ptr& r ) { data = r.data; shared_node = r.shared_node; if (shared_node) shared_node->ref_count += 1; } template shared_ptr( const shared_ptr& r, const shared_ptr_static_cast& ) { data = static_cast(r.data); if (data != 0) { shared_node = r.shared_node; shared_node->ref_count += 1; } else { shared_node = 0; } } template shared_ptr( const shared_ptr& r, const shared_ptr_const_cast& ) { data = const_cast(r.data); if (data != 0) { shared_node = r.shared_node; shared_node->ref_count += 1; } else { shared_node = 0; } } template shared_ptr( const shared_ptr& r, const shared_ptr_dynamic_cast& ) { data = dynamic_cast(r.data); if (data != 0) { shared_node = r.shared_node; shared_node->ref_count += 1; } else { shared_node = 0; } } template shared_ptr( const shared_ptr& r ) { data = r.data; shared_node = r.shared_node; if (shared_node) shared_node->ref_count += 1; } template explicit shared_ptr( const weak_ptr& r ) { if (r.expired()) throw bad_weak_ptr(); data = r.data; shared_node = r.weak_node->shared_node; shared_node->ref_count += 1; } template explicit shared_ptr( std::auto_ptr& r ) { DLIB_ASSERT(r.get() != 0, "\tshared_ptr::shared_ptr(auto_ptr r)" << "\n\tr.get() can't be null" << "\n\tthis: " << this ); shared_node = new shared_ptr_node; shared_node->del = new default_deleter; data = r.release(); } shared_ptr& operator= ( const shared_ptr& r ) { shared_ptr(r).swap(*this); return *this; } template shared_ptr& operator= ( const shared_ptr& r ) { shared_ptr(r).swap(*this); return *this; } template shared_ptr& operator= ( std::auto_ptr& r ) { DLIB_ASSERT(r.get() != 0, "\tshared_ptr::operator=(auto_ptr r)" << "\n\tr.get() can't be null" << "\n\tthis: " << this ); reset(); shared_node = new shared_ptr_node; shared_node->del = new default_deleter; data = r.release(); return *this; } void reset() { shared_ptr().swap(*this); } template void reset(Y* p) { DLIB_ASSERT(p != 0, "\tshared_ptr::reset(p)" << "\n\tp can't be null" << "\n\tthis: " << this ); shared_ptr(p).swap(*this); } template void reset( Y* p, const D& d ) { DLIB_ASSERT(p != 0, "\tshared_ptr::reset(p,d)" << "\n\tp can't be null" << "\n\tthis: " << this ); shared_ptr(p,d).swap(*this); } T& operator*( ) const { DLIB_ASSERT(get() != 0, "\tshared_ptr::operator*()" << "\n\tget() can't be null if you are going to dereference it" << "\n\tthis: " << this ); return *data; } T* operator->( ) const { DLIB_ASSERT(get() != 0, "\tshared_ptr::operator->()" << "\n\tget() can't be null" << "\n\tthis: " << this ); return data; } T* get() const { return data; } bool unique() const { return use_count() == 1; } long use_count() const { if (shared_node != 0) return shared_node->ref_count; else return 0; } operator bool( ) const { return get() != 0; } void swap(shared_ptr& b) { std::swap(data, b.data); std::swap(shared_node, b.shared_node); } template D* _get_deleter( ) const { if (shared_node && shared_node->del) return static_cast(shared_node->del->get_deleter_void(typeid(D))); else return 0; } template bool _private_less ( const shared_ptr& rhs ) const { return shared_node < rhs.shared_node; } private: template friend class shared_ptr; template friend class weak_ptr; T* data; shared_ptr_node* shared_node; }; // ---------------------------------------------------------------------------------------- template bool operator== ( const shared_ptr& a, const shared_ptr& b ) { return a.get() == b.get(); } template bool operator!= ( const shared_ptr& a, const shared_ptr& b ) { return a.get() != b.get(); } template bool operator< ( const shared_ptr& a, const shared_ptr& b ) { return a._private_less(b); } template void swap( shared_ptr& a, shared_ptr& b ) { a.swap(b); } template shared_ptr static_pointer_cast( const shared_ptr& r ) { return shared_ptr(r, shared_ptr_static_cast()); } template shared_ptr const_pointer_cast( shared_ptr const & r ) { return shared_ptr(r, shared_ptr_const_cast()); } template shared_ptr dynamic_pointer_cast( const shared_ptr& r ) { return shared_ptr(r, shared_ptr_dynamic_cast()); } template std::basic_ostream & operator<< (std::basic_ostream & os, shared_ptr const & p) { os << p.get(); return os; } template D* get_deleter(const shared_ptr& p) { return p.template _get_deleter(); } // ---------------------------------------------------------------------------------------- } #endif // DLIB_SHARED_PTr_