2020-09-30 16:53:55 +02:00

151 lines
3.9 KiB
C++

/***************************************************************************\
*
* File: AllocPool.inl
*
* History:
* 1/28/2000: JStall: Created
*
* Copyright (C) 2000 by Microsoft Corporation. All rights reserved.
*
\***************************************************************************/
#if !defined(BASE__AllocPool_inl__INCLUDED)
#define BASE__AllocPool_inl__INCLUDED
#pragma once
/***************************************************************************\
*****************************************************************************
*
* class AllocListNL
*
*****************************************************************************
\***************************************************************************/
//------------------------------------------------------------------------------
template <class T, int cbBlock, class heap>
inline
AllocPoolNL<T, cbBlock, heap>::AllocPoolNL()
{
m_nTop = 0;
}
//------------------------------------------------------------------------------
template <class T, int cbBlock, class heap>
inline
AllocPoolNL<T, cbBlock, heap>::~AllocPoolNL()
{
Destroy();
}
//------------------------------------------------------------------------------
template <class T, int cbBlock, class heap>
inline void
AllocPoolNL<T, cbBlock, heap>::Destroy()
{
if (m_nTop > 0) {
ContextMultiFree(heap::GetHeap(), m_nTop, (void **) m_rgItems, sizeof(T));
m_nTop = 0;
}
}
//------------------------------------------------------------------------------
template <class T, int cbBlock, class heap>
inline T *
AllocPoolNL<T, cbBlock, heap>::New()
{
T * ptNew;
if (m_nTop <= 0) {
//
// Not enough items in the pool to hand any new one out, so we need to
// allocate more. These will NOT be zero-initialized by the memory
// allocator.
//
ContextMultiAlloc(heap::GetHeap(), &m_nTop, (void **) m_rgItems, cbBlock, sizeof(T));
if (m_nTop == 0) {
ptNew = NULL;
goto exit;
}
}
//
// There is an item in the pool, but we need to "scrub" it before handing
// it out.
//
ptNew = m_rgItems[--m_nTop];
ZeroMemory(ptNew, sizeof(T));
placement_new(ptNew, T);
exit:
return ptNew;
}
//------------------------------------------------------------------------------
template <class T, int cbBlock, class heap>
inline void
AllocPoolNL<T, cbBlock, heap>::Delete(T * pvMem)
{
if (pvMem == NULL) {
return;
}
placement_delete(pvMem, T);
m_rgItems[m_nTop++] = pvMem;
if (m_nTop >= cbBlock * 2) {
ContextMultiFree(heap::GetHeap(), cbBlock, (void **) &m_rgItems[cbBlock], sizeof(T));
m_nTop -= cbBlock;
}
}
//------------------------------------------------------------------------------
template <class T, int cbBlock, class heap>
inline BOOL
AllocPoolNL<T, cbBlock, heap>::IsEmpty() const
{
return m_nTop == 0;
}
/***************************************************************************\
*****************************************************************************
*
* class AllocList
*
*****************************************************************************
\***************************************************************************/
//------------------------------------------------------------------------------
template <class T, int cbBlock>
inline T *
AllocPool<T, cbBlock>::New()
{
m_lock.Enter();
T * pNew = AllocPoolNL<T, cbBlock>::New();
m_lock.Leave();
return pNew;
}
//------------------------------------------------------------------------------
template <class T, int cbBlock>
inline void
AllocPool<T, cbBlock>::Delete(T * pvMem)
{
m_lock.Enter();
AllocPoolNL<T, cbBlock>::Delete(pvMem);
m_lock.Leave();
}
#endif // BASE__AllocPool_inl__INCLUDED