shared_ptr now autodetects counted_bases; minor test updates; intrusive_ptr no longer calls addref/release for NULL pointers.

[SVN r13602]
This commit is contained in:
Peter Dimov
2002-05-01 11:22:22 +00:00
parent 11eacab70e
commit 4653c3673b
4 changed files with 52 additions and 12 deletions

View File

@ -169,12 +169,12 @@ private:
inline void intrusive_ptr_add_ref(counted_base * p)
{
if(p != 0) p->add_ref();
p->add_ref();
}
inline void intrusive_ptr_release(counted_base * p)
{
if(p != 0) p->release();
p->release();
}
namespace detail
@ -215,6 +215,8 @@ private:
friend class weak_count;
template<class P, class D> shared_count(P, D, counted_base const *);
public:
shared_count(): pi_(new counted_base(1, 1))
@ -226,7 +228,7 @@ public:
pi_->add_ref();
}
template<class P, class D> shared_count(P p, D d): pi_(0)
template<class P, class D> shared_count(P p, D d, void const * = 0): pi_(0)
{
try
{
@ -239,6 +241,11 @@ public:
}
}
template<class P, class D> shared_count(P, D, counted_base * pi): pi_(pi)
{
pi_->add_ref();
}
#ifndef BOOST_NO_AUTO_PTR
// auto_ptr<Y> is special cased to provide the strong guarantee

View File

@ -33,7 +33,8 @@ namespace boost
//
// void intrusive_ptr_add_ref(T * p);
// void intrusive_ptr_release(T * p);
// (note: p may be 0!)
//
// (p != 0)
//
// The object is responsible for destroying itself.
//
@ -52,26 +53,26 @@ public:
intrusive_ptr(T * p): p_(p)
{
intrusive_ptr_add_ref(p_);
if(p_ != 0) intrusive_ptr_add_ref(p_);
}
~intrusive_ptr()
{
intrusive_ptr_release(p_);
if(p_ != 0) intrusive_ptr_release(p_);
}
#ifdef BOOST_MSVC6_MEMBER_TEMPLATES
template<class U> intrusive_ptr(intrusive_ptr<U> const & rhs): p_(rhs.get())
{
intrusive_ptr_add_ref(p_);
if(p_ != 0) intrusive_ptr_add_ref(p_);
}
#endif
intrusive_ptr(intrusive_ptr const & rhs): p_(rhs.p_)
{
intrusive_ptr_add_ref(p_);
if(p_ != 0) intrusive_ptr_add_ref(p_);
}
#ifdef BOOST_MSVC6_MEMBER_TEMPLATES

View File

@ -87,7 +87,7 @@ public:
}
template<typename Y>
explicit shared_ptr(Y * p): px(p), pn(p, checked_deleter<Y>()) // Y must be complete
explicit shared_ptr(Y * p): px(p), pn(p, checked_deleter<Y>(), p) // Y must be complete
{
}

View File

@ -40,7 +40,7 @@ struct X
std::cout << "X(" << this << ")::X()\n";
}
virtual ~X()
~X() // virtual destructor deliberately omitted
{
--cnt;
std::cout << "X(" << this << ")::~X()\n";
@ -96,6 +96,28 @@ void release_object(int * p)
std::cout << "release_object()\n";
}
class Z: public virtual boost::counted_base
{
public:
Z()
{
++cnt;
std::cout << "Z(" << this << ")::Z()\n";
}
~Z()
{
--cnt;
std::cout << "Z(" << this << ")::~Z()\n";
}
private:
Z(Z const &);
Z & operator= (Z const &);
};
template<class T> void test_is_X(boost::shared_ptr<T> const & p)
{
BOOST_TEST(p->id() == 1);
@ -302,7 +324,17 @@ int test_main(int, char * [])
BOOST_TEST(b1 == (wp1 < wp5));
BOOST_TEST(b2 == (wp5 < wp1));
shared_ptr<int> p6(get_object(), release_object);
{
// note that both get_object and release_object deal with int*
shared_ptr<void> p6(get_object(), release_object);
}
{
// test intrusive counting
boost::shared_ptr<void> pv(new Z);
boost::shared_ptr<Z> pz = boost::shared_static_cast<Z>(pv);
BOOST_TEST(pz.use_count() == pz->use_count());
}
}
BOOST_TEST(cnt == 0);