mirror of
https://git.lyx.org/repos/lyx.git
synced 2025-01-07 01:46:27 +00:00
b9a9f9fca1
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@2080 a592a061-630c-0410-9148-cb99ea01b6c8
431 lines
16 KiB
C++
431 lines
16 KiB
C++
// Baruch Even baruch@ev-en.org 2001-02-20
|
|
// This version is a modified version for use in LyX
|
|
// The modifications are done in order to use it where exceptions are disabled
|
|
// Currently it has no "no memory" checks in place, asserts are probably the
|
|
// only real way.
|
|
// all changed are #ifded'ed by LYX_NO_EXCEPTIONS
|
|
#define LYX_NO_EXCEPTIONS
|
|
|
|
// Boost smart_ptr.hpp header file -----------------------------------------//
|
|
|
|
// (C) Copyright Greg Colvin and Beman Dawes 1998, 1999. Permission to copy,
|
|
// use, modify, sell and distribute this software is granted provided this
|
|
// copyright notice appears in all copies. This software is provided "as is"
|
|
// without express or implied warranty, and with no claim as to its
|
|
// suitability for any purpose.
|
|
|
|
// See http://www.boost.org for most recent version including documentation.
|
|
|
|
// Revision History
|
|
// 21 May 01 Require complete type where incomplete type is unsafe.
|
|
// (suggested by Vladimir Prus)
|
|
// 21 May 01 operator= fails if operand transitively owned by *this, as in a
|
|
// linked list (report by Ken Johnson, fix by Beman Dawes)
|
|
// 21 Jan 01 Suppress some useless warnings with MSVC (David Abrahams)
|
|
// 19 Oct 00 Make shared_ptr ctor from auto_ptr explicit. (Robert Vugts)
|
|
// 24 Jul 00 Change throw() to // never throws. See lib guidelines
|
|
// Exception-specification rationale. (Beman Dawes)
|
|
// 22 Jun 00 Remove #if continuations to fix GCC 2.95.2 problem (Beman Dawes)
|
|
// 1 Feb 00 Additional shared_ptr BOOST_NO_MEMBER_TEMPLATES workarounds
|
|
// (Dave Abrahams)
|
|
// 31 Dec 99 Condition tightened for no member template friend workaround
|
|
// (Dave Abrahams)
|
|
// 30 Dec 99 Moved BOOST_NMEMBER_TEMPLATES compatibility code to config.hpp
|
|
// (Dave Abrahams)
|
|
// 30 Nov 99 added operator ==, operator !=, and std::swap and std::less
|
|
// specializations for shared types (Darin Adler)
|
|
// 11 Oct 99 replaced op[](int) with op[](std::size_t) (Ed Brey, Valentin
|
|
// Bonnard), added shared_ptr workaround for no member template
|
|
// friends (Matthew Langston)
|
|
// 25 Sep 99 added shared_ptr::swap and shared_array::swap (Luis Coelho).
|
|
// 20 Jul 99 changed name to smart_ptr.hpp, #include <boost/config.hpp>,
|
|
// #include <boost/utility.hpp> and use boost::noncopyable
|
|
// 17 May 99 remove scoped_array and shared_array operator*() as
|
|
// unnecessary (Beman Dawes)
|
|
// 14 May 99 reorder code so no effects when bad_alloc thrown (Abrahams/Dawes)
|
|
// 13 May 99 remove certain throw() specifiers to avoid generated try/catch
|
|
// code cost (Beman Dawes)
|
|
// 11 May 99 get() added, conversion to T* placed in macro guard (Valentin
|
|
// Bonnard, Dave Abrahams, and others argued for elimination
|
|
// of the automatic conversion)
|
|
// 28 Apr 99 #include <memory> fix (Valentin Bonnard)
|
|
// 28 Apr 99 rename transfer() to share() for clarity (Dave Abrahams)
|
|
// 28 Apr 99 remove unsafe shared_array template conversions(Valentin Bonnard)
|
|
// 28 Apr 99 p(r) changed to p(r.px) for clarity (Dave Abrahams)
|
|
// 21 Apr 99 reset() self assignment fix (Valentin Bonnard)
|
|
// 21 Apr 99 dispose() provided to improve clarity (Valentin Bonnard)
|
|
// 27 Apr 99 leak when new throws fixes (Dave Abrahams)
|
|
// 21 Oct 98 initial Version (Greg Colvin/Beman Dawes)
|
|
|
|
#ifndef BOOST_SMART_PTR_HPP
|
|
#define BOOST_SMART_PTR_HPP
|
|
|
|
#include <boost/config.hpp> // for broken compiler workarounds
|
|
#include <cstddef> // for std::size_t
|
|
#include <memory> // for std::auto_ptr
|
|
#include <algorithm> // for std::swap
|
|
#include <boost/utility.hpp> // for boost::noncopyable, checked_delete, checked_array_delete
|
|
#include <functional> // for std::less
|
|
#include <boost/static_assert.hpp> // for BOOST_STATIC_ASSERT
|
|
|
|
#ifdef LYX_NO_EXCEPTIONS
|
|
#include <assert.h>
|
|
#endif
|
|
|
|
#ifdef BOOST_MSVC // moved here to work around VC++ compiler crash
|
|
# pragma warning(push)
|
|
# pragma warning(disable:4284) // return type for 'identifier::operator->' is not a UDT or reference to a UDT. Will produce errors if applied using infix notation
|
|
#endif
|
|
|
|
namespace boost {
|
|
|
|
// scoped_ptr --------------------------------------------------------------//
|
|
|
|
// scoped_ptr mimics a built-in pointer except that it guarantees deletion
|
|
// of the object pointed to, either on destruction of the scoped_ptr or via
|
|
// an explicit reset(). scoped_ptr is a simple solution for simple needs;
|
|
// see shared_ptr (below) or std::auto_ptr if your needs are more complex.
|
|
|
|
template<typename T> class scoped_ptr : noncopyable {
|
|
|
|
T* ptr;
|
|
|
|
public:
|
|
typedef T element_type;
|
|
|
|
explicit scoped_ptr( T* p=0 ) : ptr(p) {} // never throws
|
|
~scoped_ptr() { checked_delete(ptr); }
|
|
void reset( T* p=0 ) { if ( ptr != p ) { checked_delete(ptr); ptr = p; } }
|
|
T& operator*() const { return *ptr; } // never throws
|
|
T* operator->() const { return ptr; } // never throws
|
|
T* get() const { return ptr; } // never throws
|
|
#ifdef BOOST_SMART_PTR_CONVERSION
|
|
// get() is safer! Define BOOST_SMART_PTR_CONVERSION at your own risk!
|
|
operator T*() const { return ptr; } // never throws
|
|
#endif
|
|
}; // scoped_ptr
|
|
|
|
// scoped_array ------------------------------------------------------------//
|
|
|
|
// scoped_array extends scoped_ptr to arrays. Deletion of the array pointed to
|
|
// is guaranteed, either on destruction of the scoped_array or via an explicit
|
|
// reset(). See shared_array or std::vector if your needs are more complex.
|
|
|
|
template<typename T> class scoped_array : noncopyable {
|
|
|
|
T* ptr;
|
|
|
|
public:
|
|
typedef T element_type;
|
|
|
|
explicit scoped_array( T* p=0 ) : ptr(p) {} // never throws
|
|
~scoped_array() { checked_array_delete(ptr); }
|
|
|
|
void reset( T* p=0 ) { if ( ptr != p )
|
|
{checked_array_delete(ptr); ptr=p;} }
|
|
|
|
T* get() const { return ptr; } // never throws
|
|
#ifdef BOOST_SMART_PTR_CONVERSION
|
|
// get() is safer! Define BOOST_SMART_PTR_CONVERSION at your own risk!
|
|
operator T*() const { return ptr; } // never throws
|
|
#else
|
|
T& operator[](std::size_t i) const { return ptr[i]; } // never throws
|
|
#endif
|
|
}; // scoped_array
|
|
|
|
// shared_ptr --------------------------------------------------------------//
|
|
|
|
// An enhanced relative of scoped_ptr with reference counted copy semantics.
|
|
// The object pointed to is deleted when the last shared_ptr pointing to it
|
|
// is destroyed or reset.
|
|
|
|
template<typename T> class shared_ptr {
|
|
public:
|
|
typedef T element_type;
|
|
|
|
explicit shared_ptr(T* p =0) : px(p) {
|
|
#ifndef LYX_NO_EXCEPTIONS
|
|
try { pn = new long(1); } // fix: prevent leak if new throws
|
|
catch (...) { checked_delete(p); throw; }
|
|
#else
|
|
pn = new long(1);
|
|
assert(pn != 0);
|
|
#endif
|
|
}
|
|
|
|
shared_ptr(const shared_ptr& r) : px(r.px) { ++*(pn = r.pn); } // never throws
|
|
|
|
~shared_ptr() { dispose(); }
|
|
|
|
shared_ptr& operator=(const shared_ptr& r) {
|
|
share(r.px,r.pn);
|
|
return *this;
|
|
}
|
|
|
|
#if !defined( BOOST_NO_MEMBER_TEMPLATES )
|
|
template<typename Y>
|
|
shared_ptr(const shared_ptr<Y>& r) : px(r.px) { // never throws
|
|
++*(pn = r.pn);
|
|
}
|
|
#ifndef BOOST_NO_AUTO_PTR
|
|
template<typename Y>
|
|
explicit shared_ptr(std::auto_ptr<Y>& r) {
|
|
pn = new long(1); // may throw
|
|
px = r.release(); // fix: moved here to stop leak if new throws
|
|
}
|
|
#endif
|
|
|
|
template<typename Y>
|
|
shared_ptr& operator=(const shared_ptr<Y>& r) {
|
|
share(r.px,r.pn);
|
|
return *this;
|
|
}
|
|
|
|
#ifndef BOOST_NO_AUTO_PTR
|
|
template<typename Y>
|
|
shared_ptr& operator=(std::auto_ptr<Y>& r) {
|
|
// code choice driven by guarantee of "no effect if new throws"
|
|
if (*pn == 1) { checked_delete(px); }
|
|
else { // allocate new reference counter
|
|
long * tmp = new long(1); // may throw
|
|
--*pn; // only decrement once danger of new throwing is past
|
|
pn = tmp;
|
|
} // allocate new reference counter
|
|
px = r.release(); // fix: moved here so doesn't leak if new throws
|
|
return *this;
|
|
}
|
|
#endif
|
|
#else
|
|
#ifndef BOOST_NO_AUTO_PTR
|
|
explicit shared_ptr(std::auto_ptr<T>& r) {
|
|
pn = new long(1); // may throw
|
|
px = r.release(); // fix: moved here to stop leak if new throws
|
|
}
|
|
|
|
shared_ptr& operator=(std::auto_ptr<T>& r) {
|
|
// code choice driven by guarantee of "no effect if new throws"
|
|
if (*pn == 1) { checked_delete(px); }
|
|
else { // allocate new reference counter
|
|
long * tmp = new long(1); // may throw
|
|
--*pn; // only decrement once danger of new throwing is past
|
|
pn = tmp;
|
|
} // allocate new reference counter
|
|
px = r.release(); // fix: moved here so doesn't leak if new throws
|
|
return *this;
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
void reset(T* p=0) {
|
|
if ( px == p ) return; // fix: self-assignment safe
|
|
if (--*pn == 0) { checked_delete(px); }
|
|
else { // allocate new reference counter
|
|
#ifndef LYX_NO_EXCEPTIONS
|
|
try { pn = new long; } // fix: prevent leak if new throws
|
|
catch (...) {
|
|
++*pn; // undo effect of --*pn above to meet effects guarantee
|
|
checked_delete(p);
|
|
throw;
|
|
} // catch
|
|
#else
|
|
pn = new long;
|
|
assert(pn != 0);
|
|
#endif
|
|
} // allocate new reference counter
|
|
*pn = 1;
|
|
px = p;
|
|
} // reset
|
|
|
|
T& operator*() const { return *px; } // never throws
|
|
T* operator->() const { return px; } // never throws
|
|
T* get() const { return px; } // never throws
|
|
#ifdef BOOST_SMART_PTR_CONVERSION
|
|
// get() is safer! Define BOOST_SMART_PTR_CONVERSION at your own risk!
|
|
operator T*() const { return px; } // never throws
|
|
#endif
|
|
|
|
long use_count() const { return *pn; } // never throws
|
|
bool unique() const { return *pn == 1; } // never throws
|
|
|
|
void swap(shared_ptr<T>& other) // never throws
|
|
{ std::swap(px,other.px); std::swap(pn,other.pn); }
|
|
|
|
// Tasteless as this may seem, making all members public allows member templates
|
|
// to work in the absence of member template friends. (Matthew Langston)
|
|
// Don't split this line into two; that causes problems for some GCC 2.95.2 builds
|
|
#if defined(BOOST_NO_MEMBER_TEMPLATES) || !defined( BOOST_NO_MEMBER_TEMPLATE_FRIENDS )
|
|
private:
|
|
#endif
|
|
|
|
T* px; // contained pointer
|
|
long* pn; // ptr to reference counter
|
|
|
|
// Don't split this line into two; that causes problems for some GCC 2.95.2 builds
|
|
#if !defined( BOOST_NO_MEMBER_TEMPLATES ) && !defined( BOOST_NO_MEMBER_TEMPLATE_FRIENDS )
|
|
template<typename Y> friend class shared_ptr;
|
|
#endif
|
|
|
|
void dispose() { if (--*pn == 0) { checked_delete(px); delete pn; } }
|
|
|
|
void share(T* rpx, long* rpn) {
|
|
if (pn != rpn) { // Q: why not px != rpx? A: fails when both == 0
|
|
++*rpn; // done before dispose() in case rpn transitively
|
|
// dependent on *this (bug reported by Ken Johnson)
|
|
dispose();
|
|
px = rpx;
|
|
pn = rpn;
|
|
}
|
|
} // share
|
|
}; // shared_ptr
|
|
|
|
template<typename T, typename U>
|
|
inline bool operator==(const shared_ptr<T>& a, const shared_ptr<U>& b)
|
|
{ return a.get() == b.get(); }
|
|
|
|
template<typename T, typename U>
|
|
inline bool operator!=(const shared_ptr<T>& a, const shared_ptr<U>& b)
|
|
{ return a.get() != b.get(); }
|
|
|
|
// shared_array ------------------------------------------------------------//
|
|
|
|
// shared_array extends shared_ptr to arrays.
|
|
// The array pointed to is deleted when the last shared_array pointing to it
|
|
// is destroyed or reset.
|
|
|
|
template<typename T> class shared_array {
|
|
public:
|
|
typedef T element_type;
|
|
|
|
explicit shared_array(T* p =0) : px(p) {
|
|
#ifndef LYX_NO_EXCEPTIONS
|
|
try { pn = new long(1); } // fix: prevent leak if new throws
|
|
catch (...) { checked_array_delete(p); throw; }
|
|
#else
|
|
pn = new long(1);
|
|
assert(pn != 0);
|
|
#endif
|
|
}
|
|
|
|
shared_array(const shared_array& r) : px(r.px) // never throws
|
|
{ ++*(pn = r.pn); }
|
|
|
|
~shared_array() { dispose(); }
|
|
|
|
shared_array& operator=(const shared_array& r) {
|
|
if (pn != r.pn) { // Q: why not px != r.px? A: fails when both px == 0
|
|
++*r.pn; // done before dispose() in case r.pn transitively
|
|
// dependent on *this (bug reported by Ken Johnson)
|
|
dispose();
|
|
px = r.px;
|
|
pn = r.pn;
|
|
}
|
|
return *this;
|
|
} // operator=
|
|
|
|
void reset(T* p=0) {
|
|
if ( px == p ) return; // fix: self-assignment safe
|
|
if (--*pn == 0) { checked_array_delete(px); }
|
|
else { // allocate new reference counter
|
|
#ifndef LYX_NO_EXCEPTIONS
|
|
try { pn = new long; } // fix: prevent leak if new throws
|
|
catch (...) {
|
|
++*pn; // undo effect of --*pn above to meet effects guarantee
|
|
checked_array_delete(p);
|
|
throw;
|
|
} // catch
|
|
#else
|
|
pn = new long;
|
|
assert(pn != 0);
|
|
#endif
|
|
} // allocate new reference counter
|
|
*pn = 1;
|
|
px = p;
|
|
} // reset
|
|
|
|
T* get() const { return px; } // never throws
|
|
#ifdef BOOST_SMART_PTR_CONVERSION
|
|
// get() is safer! Define BOOST_SMART_PTR_CONVERSION at your own risk!
|
|
operator T*() const { return px; } // never throws
|
|
#else
|
|
T& operator[](std::size_t i) const { return px[i]; } // never throws
|
|
#endif
|
|
|
|
long use_count() const { return *pn; } // never throws
|
|
bool unique() const { return *pn == 1; } // never throws
|
|
|
|
void swap(shared_array<T>& other) // never throws
|
|
{ std::swap(px,other.px); std::swap(pn,other.pn); }
|
|
|
|
private:
|
|
|
|
T* px; // contained pointer
|
|
long* pn; // ptr to reference counter
|
|
|
|
void dispose() { if (--*pn == 0) { checked_array_delete(px); delete pn; } }
|
|
|
|
}; // shared_array
|
|
|
|
template<typename T>
|
|
inline bool operator==(const shared_array<T>& a, const shared_array<T>& b)
|
|
{ return a.get() == b.get(); }
|
|
|
|
template<typename T>
|
|
inline bool operator!=(const shared_array<T>& a, const shared_array<T>& b)
|
|
{ return a.get() != b.get(); }
|
|
|
|
} // namespace boost
|
|
|
|
// specializations for things in namespace std -----------------------------//
|
|
|
|
#ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION
|
|
|
|
namespace std {
|
|
|
|
// Specialize std::swap to use the fast, non-throwing swap that's provided
|
|
// as a member function instead of using the default algorithm which creates
|
|
// a temporary and uses assignment.
|
|
|
|
template<typename T>
|
|
inline void swap(boost::shared_ptr<T>& a, boost::shared_ptr<T>& b)
|
|
{ a.swap(b); }
|
|
|
|
template<typename T>
|
|
inline void swap(boost::shared_array<T>& a, boost::shared_array<T>& b)
|
|
{ a.swap(b); }
|
|
|
|
// Specialize std::less so we can use shared pointers and arrays as keys in
|
|
// associative collections.
|
|
|
|
// It's still a controversial question whether this is better than supplying
|
|
// a full range of comparison operators (<, >, <=, >=).
|
|
|
|
template<typename T>
|
|
struct less< boost::shared_ptr<T> >
|
|
: binary_function<boost::shared_ptr<T>, boost::shared_ptr<T>, bool>
|
|
{
|
|
bool operator()(const boost::shared_ptr<T>& a,
|
|
const boost::shared_ptr<T>& b) const
|
|
{ return less<T*>()(a.get(),b.get()); }
|
|
};
|
|
|
|
template<typename T>
|
|
struct less< boost::shared_array<T> >
|
|
: binary_function<boost::shared_array<T>, boost::shared_array<T>, bool>
|
|
{
|
|
bool operator()(const boost::shared_array<T>& a,
|
|
const boost::shared_array<T>& b) const
|
|
{ return less<T*>()(a.get(),b.get()); }
|
|
};
|
|
|
|
} // namespace std
|
|
|
|
#endif // ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION
|
|
|
|
#ifdef BOOST_MSVC
|
|
#pragma warning(pop)
|
|
#endif
|
|
|
|
#endif // BOOST_SMART_PTR_HPP
|
|
|
|
|