255 lines
11 KiB
C
255 lines
11 KiB
C
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
/**
|
||
|
* Contains a simple container class.
|
||
|
* \file IceContainer.h
|
||
|
* \author Pierre Terdiman
|
||
|
* \date February, 5, 2000
|
||
|
*/
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
// Include Guard
|
||
|
#ifndef ICECONTAINER_H
|
||
|
#define ICECONTAINER_H
|
||
|
|
||
|
// #define CONTAINER_STATS // #### doesn't work with micro-threads!
|
||
|
|
||
|
class LinkedList;
|
||
|
|
||
|
enum FindMode
|
||
|
{
|
||
|
FIND_CLAMP,
|
||
|
FIND_WRAP,
|
||
|
|
||
|
FIND_FORCE_DWORD = 0x7fffffff
|
||
|
};
|
||
|
|
||
|
class ICECORE_API Container : public Allocateable
|
||
|
#ifdef CONTAINER_STATS
|
||
|
, public ListElem
|
||
|
#endif
|
||
|
{
|
||
|
public:
|
||
|
// Constructor / Destructor
|
||
|
Container();
|
||
|
Container(const Container& object);
|
||
|
Container(udword size, float growth_factor);
|
||
|
~Container();
|
||
|
// Management
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
/**
|
||
|
* Initializes the container so that it uses an external memory buffer. The container doesn't own the memory, resizing is disabled.
|
||
|
* \param max_entries [in] max number of entries in the container
|
||
|
* \param entries [in] external memory buffer
|
||
|
*/
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
void InitSharedBuffers(udword max_entries, udword* entries);
|
||
|
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
/**
|
||
|
* A O(1) method to add a value in the container. The container is automatically resized if needed.
|
||
|
* The method is inline, not the resize. The call overhead happens on resizes only, which is not a problem since the resizing operation
|
||
|
* costs a lot more than the call overhead...
|
||
|
*
|
||
|
* \param entry [in] a udword to store in the container
|
||
|
* \see Add(float entry)
|
||
|
* \see Empty()
|
||
|
* \see Contains(udword entry)
|
||
|
* \return Self-Reference
|
||
|
*/
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
inline_ Container& Add(udword entry)
|
||
|
{
|
||
|
// Resize if needed
|
||
|
if(mCurNbEntries==mMaxNbEntries) Resize();
|
||
|
|
||
|
// Add new entry
|
||
|
mEntries[mCurNbEntries++] = entry;
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
inline_ Container& Add(const udword* entries, udword nb)
|
||
|
{
|
||
|
if(entries && nb)
|
||
|
{
|
||
|
// Resize if needed
|
||
|
if(mCurNbEntries+nb>mMaxNbEntries) Resize(nb);
|
||
|
|
||
|
// Add new entry
|
||
|
CopyMemory(&mEntries[mCurNbEntries], entries, nb*sizeof(udword));
|
||
|
mCurNbEntries += nb;
|
||
|
}
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
inline_ Container& Add(const Container& container)
|
||
|
{
|
||
|
return Add(container.GetEntries(), container.GetNbEntries());
|
||
|
}
|
||
|
|
||
|
inline_ udword* Reserve(udword nb)
|
||
|
{
|
||
|
// Resize if needed
|
||
|
if(mCurNbEntries+nb>mMaxNbEntries) Resize(nb);
|
||
|
|
||
|
// We expect the user to fill reserved memory with 'nb' udwords
|
||
|
udword* Reserved = &mEntries[mCurNbEntries];
|
||
|
|
||
|
// Meanwhile, we do as if it had been filled
|
||
|
mCurNbEntries += nb;
|
||
|
|
||
|
// This is mainly used to avoid the copy when possible
|
||
|
return Reserved;
|
||
|
}
|
||
|
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
/**
|
||
|
* A O(1) method to add a value in the container. The container is automatically resized if needed.
|
||
|
* The method is inline, not the resize. The call overhead happens on resizes only, which is not a problem since the resizing operation
|
||
|
* costs a lot more than the call overhead...
|
||
|
*
|
||
|
* \param entry [in] a float to store in the container
|
||
|
* \see Add(udword entry)
|
||
|
* \see Empty()
|
||
|
* \see Contains(udword entry)
|
||
|
* \return Self-Reference
|
||
|
*/
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
inline_ Container& Add(float entry)
|
||
|
{
|
||
|
// Resize if needed
|
||
|
if(mCurNbEntries==mMaxNbEntries) Resize();
|
||
|
|
||
|
// Add new entry
|
||
|
mEntries[mCurNbEntries++] = IR(entry);
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
inline_ Container& Add(const float* entries, udword nb)
|
||
|
{
|
||
|
// Resize if needed
|
||
|
if(mCurNbEntries+nb>mMaxNbEntries) Resize(nb);
|
||
|
|
||
|
// Add new entry
|
||
|
CopyMemory(&mEntries[mCurNbEntries], entries, nb*sizeof(float));
|
||
|
mCurNbEntries+=nb;
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
//! Add unique [slow]
|
||
|
inline_ Container& AddUnique(udword entry)
|
||
|
{
|
||
|
if(!Contains(entry)) Add(entry);
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
/**
|
||
|
* Clears the container. All stored values are deleted, and it frees used ram.
|
||
|
* \see Reset()
|
||
|
* \return Self-Reference
|
||
|
*/
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
Container& Empty();
|
||
|
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
/**
|
||
|
* Resets the container. Stored values are discarded but the buffer is kept so that further calls don't need resizing again.
|
||
|
* That's a kind of temporal coherence.
|
||
|
* \see Empty()
|
||
|
*/
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
inline_ void Reset()
|
||
|
{
|
||
|
// Avoid the write if possible
|
||
|
// ### CMOV
|
||
|
if(mCurNbEntries) mCurNbEntries = 0;
|
||
|
}
|
||
|
|
||
|
// HANDLE WITH CARE - I hope you know what you're doing
|
||
|
inline_ void ForceSize(udword size)
|
||
|
{
|
||
|
mCurNbEntries = size;
|
||
|
}
|
||
|
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
/**
|
||
|
* Sets the initial size of the container. If it already contains something, it's discarded.
|
||
|
* \param nb [in] Number of entries
|
||
|
* \return true if success
|
||
|
*/
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
bool SetSize(udword nb);
|
||
|
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
/**
|
||
|
* Refits the container and get rid of unused bytes.
|
||
|
* \return true if success
|
||
|
*/
|
||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
bool Refit();
|
||
|
|
||
|
bool Shrink();
|
||
|
|
||
|
// Checks whether the container already contains a given value.
|
||
|
bool Contains(udword entry, udword* location=null) const;
|
||
|
// Deletes an entry - doesn't preserve insertion order.
|
||
|
bool Delete(udword entry);
|
||
|
// Deletes an entry - does preserve insertion order.
|
||
|
bool DeleteKeepingOrder(udword entry);
|
||
|
//! Deletes the very last entry.
|
||
|
inline_ void DeleteLastEntry() { if(mCurNbEntries) mCurNbEntries--; }
|
||
|
//! Deletes the entry whose index is given
|
||
|
inline_ void DeleteIndex(udword index) { mEntries[index] = mEntries[--mCurNbEntries]; }
|
||
|
|
||
|
// Helpers
|
||
|
Container& FindNext(udword& entry, FindMode find_mode=FIND_CLAMP);
|
||
|
Container& FindPrev(udword& entry, FindMode find_mode=FIND_CLAMP);
|
||
|
// Data access.
|
||
|
inline_ udword GetNbEntries() const { return mCurNbEntries; } //!< Returns the current number of entries.
|
||
|
inline_ udword GetMaxNbEntries() const { return mMaxNbEntries; } //!< Returns max number of entries before resizing.
|
||
|
inline_ udword GetEntry(udword i) const { return mEntries[i]; } //!< Returns ith entry.
|
||
|
inline_ udword* GetEntries() const { return mEntries; } //!< Returns the list of entries.
|
||
|
|
||
|
inline_ udword GetFirst() const { return mEntries[0]; }
|
||
|
inline_ udword GetLast() const { return mEntries[mCurNbEntries-1]; }
|
||
|
|
||
|
// Growth control
|
||
|
float GetGrowthFactor() const; //!< Returns the growth factor
|
||
|
void SetGrowthFactor(float growth); //!< Sets the growth factor
|
||
|
inline_ bool IsFull() const { return mCurNbEntries==mMaxNbEntries; } //!< Checks the container is full
|
||
|
inline_ BOOL IsNotEmpty() const { return mCurNbEntries; } //!< Checks the container is empty
|
||
|
|
||
|
//! Read-access as an array
|
||
|
inline_ udword operator[](udword i) const { ASSERT(i>=0 && i<mCurNbEntries); return mEntries[i]; }
|
||
|
//! Write-access as an array
|
||
|
inline_ udword& operator[](udword i) { ASSERT(i>=0 && i<mCurNbEntries); return mEntries[i]; }
|
||
|
|
||
|
// Stats
|
||
|
udword GetUsedRam() const;
|
||
|
|
||
|
//! Operator for "Container A = Container B"
|
||
|
void operator = (const Container& object);
|
||
|
|
||
|
#ifdef CONTAINER_STATS
|
||
|
static udword GetNbContainers() { return mNbContainers; }
|
||
|
static udword GetTotalBytes() { return mUsedRam; }
|
||
|
static LinkedList& GetContainers() { return mContainers; }
|
||
|
private:
|
||
|
|
||
|
static udword mNbContainers; //!< Number of containers around
|
||
|
static udword mUsedRam; //!< Amount of bytes used by containers in the system
|
||
|
static LinkedList mContainers;
|
||
|
#endif
|
||
|
private:
|
||
|
// Resizing
|
||
|
bool Resize(udword needed=1);
|
||
|
// Data
|
||
|
udword mMaxNbEntries; //!< Maximum possible number of entries
|
||
|
udword mCurNbEntries; //!< Current number of entries
|
||
|
udword* mEntries; //!< List of entries
|
||
|
float mGrowthFactor; //!< Resize: new number of entries = old number * mGrowthFactor
|
||
|
};
|
||
|
|
||
|
#endif // ICECONTAINER_H
|