Files
tlib/oversampling/WDL/rpool.h
2024-05-24 13:28:31 +02:00

192 lines
4.7 KiB
C++

/*
WDL - rpool.h
Copyright (C) 2006 and later, Cockos Incorporated
This software is provided 'as-is', without any express or implied
warranty. In no event will the authors be held liable for any damages
arising from the use of this software.
Permission is granted to anyone to use this software for any purpose,
including commercial applications, and to alter it and redistribute it
freely, subject to the following restrictions:
1. The origin of this software must not be misrepresented; you must not
claim that you wrote the original software. If you use this software
in a product, an acknowledgment in the product documentation would be
appreciated but is not required.
2. Altered source versions must be plainly marked as such, and must not be
misrepresented as being the original software.
3. This notice may not be removed or altered from any source distribution.
This file defines a template for a class that stores a list of objects, and allows the caller
to periodically get an object, do something with it, and add it back into the pool.
When the caller does this, it can set ownership of the object, and an expiration for that ownership.
The PTYPE1 and PTYPE2entries for the template are there to store additional information (for use with poollist.h)
This is pretty esoteric. But we use it for some things.
*/
#ifndef _WDL_RPOOL_H_
#define _WDL_RPOOL_H_
// resource pool (time based)
#include "ptrlist.h"
#include "mutex.h"
class WDL_ResourcePool_ResInfo // include in class RTYPE as WDL_ResourcePool_ResInfo m_rpoolinfo;
{
public:
WDL_ResourcePool_ResInfo(){ m_owneduntil=0; m_ownerptr=0; next=0; }
~WDL_ResourcePool_ResInfo() {}
unsigned int m_owneduntil;
void *m_ownerptr;
void *next;
} WDL_FIXALIGN;
template<class RTYPE, class EXTRAINFOTYPE> class WDL_ResourcePool
{
public:
WDL_ResourcePool(char *identstr)
{
WDL_POOLLIST_refcnt=0;
WDL_POOLLIST_identstr=identstr;
m_rlist=NULL;
extraInfo=0;
m_hadres=false;
}
~WDL_ResourcePool()
{
while (m_rlist)
{
RTYPE *tp=m_rlist;
m_rlist=(RTYPE *)m_rlist->m_rpoolinfo.next;
delete tp;
}
delete extraInfo;
}
void Clear()
{
m_mutex.Enter();
while (m_rlist)
{
RTYPE *tp=m_rlist;
m_rlist=(RTYPE *)m_rlist->m_rpoolinfo.next;
delete tp;
}
m_hadres=false;
m_mutex.Leave();
}
bool HasResources()
{
return m_hadres;
}
void AddResource(RTYPE *item, void *own, unsigned int until)
{
item->m_rpoolinfo.m_ownerptr = own;
item->m_rpoolinfo.m_owneduntil = until;
m_mutex.Enter();
item->m_rpoolinfo.next = m_rlist;
m_rlist = item;
m_hadres=true;
m_mutex.Leave();
}
void ReleaseResources(void *own)
{
m_mutex.Enter();
RTYPE *ent=m_rlist;
while (ent)
{
if (ent->m_rpoolinfo.m_ownerptr == own)
{
ent->m_rpoolinfo.m_ownerptr = 0;
ent->m_rpoolinfo.m_owneduntil=0;
}
ent=(RTYPE *)ent->m_rpoolinfo.next;
}
m_mutex.Leave();
}
RTYPE *GetResource(void *own, unsigned int now, int flags=0) // flags&1 to only match owner (not look at owneduntil). flags&2 to ignore owneduntil and always take from others (greedy)
{
m_mutex.Enter();
RTYPE *ent=m_rlist, *lastent=NULL, *bestent=NULL, *bestlastent=NULL;
bool bestnoown=false;
while (ent)
{
if (ent->m_rpoolinfo.m_ownerptr == own)
{
if (lastent) lastent->m_rpoolinfo.next = ent->m_rpoolinfo.next;
else m_rlist = (RTYPE *)ent->m_rpoolinfo.next;
m_mutex.Leave();
return ent;
}
if (!bestnoown)
{
if (!ent->m_rpoolinfo.m_ownerptr ||
(flags & 2) ||
(!(flags & 1) && (now - (ent->m_rpoolinfo.m_owneduntil+1)) <= 0x7FFFFFFF))
{
bestent=ent;
bestlastent=lastent;
if (!ent->m_rpoolinfo.m_ownerptr || !ent->m_rpoolinfo.m_owneduntil) bestnoown=true;
}
}
lastent=ent;
ent=(RTYPE *)ent->m_rpoolinfo.next;
}
if (bestent)
{
if (bestlastent) bestlastent->m_rpoolinfo.next = bestent->m_rpoolinfo.next;
else m_rlist = (RTYPE *)bestent->m_rpoolinfo.next;
}
m_mutex.Leave();
return bestent;
}
int WDL_POOLLIST_refcnt;
char *WDL_POOLLIST_identstr;
bool m_hadres;
EXTRAINFOTYPE *extraInfo;
RTYPE *PeekList()
{
return m_rlist;
}
void LockList()
{
m_mutex.Enter();
}
void UnlockList()
{
m_mutex.Leave();
}
private:
WDL_Mutex m_mutex;
RTYPE *m_rlist;
} WDL_FIXALIGN;
#endif