new impl for SetLongevity which looks more like c++ and circumvents the usage of the so called (ms) 'deprecated' function copy_backward

git-svn-id: svn://svn.code.sf.net/p/loki-lib/code/trunk@355 7ec92016-0320-0410-acc4-a06ded1c099a
This commit is contained in:
syntheticpp 2005-11-12 15:14:08 +00:00
parent 1c8bce4d43
commit d45e0a1b08
2 changed files with 86 additions and 11 deletions

View file

@ -23,6 +23,8 @@
#include <cstdlib>
#include <new>
#include <vector>
#include <list>
#ifdef _MSC_VER
#define LOKI_C_CALLING_CONVENTION_QUALIFIER __cdecl
@ -43,6 +45,24 @@ namespace Loki
namespace Private
{
void LOKI_C_CALLING_CONVENTION_QUALIFIER AtExitFn(); // declaration needed below
class LifetimeTracker;
#define LOKI_ENABLE_NEW_SETLONGLIVITY_HELPER_DATA_IMPL
#ifdef LOKI_ENABLE_NEW_SETLONGLIVITY_HELPER_DATA_IMPL
// Helper data
// std::list because of the inserts
typedef std::list<LifetimeTracker*> TrackerArray;
extern TrackerArray* pTrackerArray;
#else
// Helper data
typedef LifetimeTracker** TrackerArray;
extern TrackerArray pTrackerArray;
extern unsigned int elements;
#endif
////////////////////////////////////////////////////////////////////////////////
// class LifetimeTracker
// Helper class for SetLongevity
@ -68,11 +88,6 @@ namespace Loki
// Definition required
inline LifetimeTracker::~LifetimeTracker() {}
// Helper data
typedef LifetimeTracker** TrackerArray;
extern TrackerArray pTrackerArray;
extern unsigned int elements;
// Helper destroyer function
template <typename T>
@ -102,8 +117,6 @@ namespace Loki
Destroyer destroyer_;
};
void LOKI_C_CALLING_CONVENTION_QUALIFIER AtExitFn(); // declaration needed below
} // namespace Private
////////////////////////////////////////////////////////////////////////////////
@ -112,6 +125,38 @@ namespace Loki
/// Assigns an object a longevity; ensures ordered destructions of objects
/// registered thusly during the exit sequence of the application
////////////////////////////////////////////////////////////////////////////////
#ifdef LOKI_ENABLE_NEW_SETLONGLIVITY_HELPER_DATA_IMPL
template <typename T, typename Destroyer>
void SetLongevity(T* pDynObject, unsigned int longevity,
Destroyer d)
{
using namespace Private;
// manage lifetime of stack manually
if(pTrackerArray==0)
pTrackerArray = new TrackerArray;
LifetimeTracker* p = new ConcreteLifetimeTracker<T, Destroyer>(
pDynObject, longevity, d);
// Find correct position
TrackerArray::iterator pos = std::upper_bound(
pTrackerArray->begin(),
pTrackerArray->end(),
p,
LifetimeTracker::Compare);
// Insert a pointer to the object into the queue
pTrackerArray->insert(pos, p);
// Register a call to AtExitFn
std::atexit(Private::AtExitFn);
}
#else
template <typename T, typename Destroyer>
void SetLongevity(T* pDynObject, unsigned int longevity,
Destroyer d)
@ -128,7 +173,7 @@ namespace Loki
LifetimeTracker* p = new ConcreteLifetimeTracker<T, Destroyer>(
pDynObject, longevity, d);
// Insert a pointer to the object into the queue
TrackerArray pos = std::upper_bound(
pTrackerArray,
@ -146,6 +191,8 @@ namespace Loki
std::atexit(Private::AtExitFn);
}
#endif
template <typename T>
void SetLongevity(T* pDynObject, unsigned int longevity,
typename Private::Deleter<T>::Type d = Private::Deleter<T>::Delete)

View file

@ -13,20 +13,46 @@
// without express or implied warranty.
////////////////////////////////////////////////////////////////////////////////
// Last update: June 20, 2001
#include "../include/loki/Singleton.h"
using namespace Loki::Private;
#ifdef LOKI_ENABLE_NEW_SETLONGLIVITY_HELPER_DATA_IMPL
Loki::Private::TrackerArray* Loki::Private::pTrackerArray = 0;
#else
Loki::Private::TrackerArray Loki::Private::pTrackerArray = 0;
unsigned int Loki::Private::elements = 0;
#endif
////////////////////////////////////////////////////////////////////////////////
// function AtExitFn
// Ensures proper destruction of objects with longevity
////////////////////////////////////////////////////////////////////////////////
#ifdef LOKI_ENABLE_NEW_SETLONGLIVITY_HELPER_DATA_IMPL
void LOKI_C_CALLING_CONVENTION_QUALIFIER Loki::Private::AtExitFn()
{
assert(pTrackerArray!=0 && !pTrackerArray->empty());
// Pick the element at the top of the stack
LifetimeTracker* pTop = pTrackerArray->back();
// Remove that object off the stack _before_ deleting pTop
pTrackerArray->pop_back();
// Destroy the element
delete pTop;
// Destroy stack when it's empty _after_ deleting pTop
if(pTrackerArray->empty())
{
delete pTrackerArray;
pTrackerArray = 0;
}
}
#else
void LOKI_C_CALLING_CONVENTION_QUALIFIER Loki::Private::AtExitFn()
{
assert(elements > 0 && pTrackerArray != 0);
@ -41,6 +67,8 @@ void LOKI_C_CALLING_CONVENTION_QUALIFIER Loki::Private::AtExitFn()
delete pTop;
}
#endif
////////////////////////////////////////////////////////////////////////////////
// Change log:
// June 20, 2001: ported by Nick Thurn to gcc 2.95.3. Kudos, Nick!!!