genesis-3d_engine/Engine/foundation/util/blob.h

414 lines
9.4 KiB
C
Raw Permalink Normal View History

#pragma once
/****************************************************************************
Copyright (c) 2006, Radon Labs GmbH
Copyright (c) 2011-2013,WebJet Business Division,CYOU
http://www.genesis-3d.com.cn
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
****************************************************************************/
#include "core/types.h"
#include "memory/heap.h"
#include "memory/poolarrayallocator.h"
//------------------------------------------------------------------------------
namespace Util
{
class Blob
{
public:
/// static Setup method, called by SysFunc::Setup()
static void Setup();
/// static Shutdown method called by SysFunc::Exit
static void Shutdown();
/// override new operator
void* operator new(size_t s);
/// override delete operator
void operator delete(void* ptr);
/// default constructor
Blob();
/// constructor
Blob(const void* ptr, SizeT size);
/// reserve N bytes
Blob(SizeT size);
/// copy constructor
Blob(const Blob& rhs);
/// destructor
~Blob();
/// assignment operator
void operator=(const Blob& rhs);
/// equality operator
bool operator==(const Blob& rhs) const;
/// inequality operator
bool operator!=(const Blob& rhs) const;
/// greater operator
bool operator>(const Blob& rhs) const;
/// less operator
bool operator<(const Blob& rhs) const;
/// greater-equal operator
bool operator>=(const Blob& rhs) const;
/// less-eqial operator
bool operator<=(const Blob& rhs) const;
/// return true if the blob contains data
bool IsValid() const;
/// reserve N bytes
void Reserve(SizeT size);
/// trim the size member (without re-allocating!)
void Trim(SizeT size);
/// set blob contents
void Set(const void* ptr, SizeT size);
/// get blob ptr
void* GetPtr() const;
/// get blob size
SizeT Size() const;
/// get a hash code (compatible with Util::HashTable)
IndexT HashCode() const;
private:
/// delete content
void Delete();
/// allocate internal buffer
void Allocate(SizeT size);
/// copy content
void Copy(const void* ptr, SizeT size);
/// do a binary comparison between this and other blob
int BinaryCompare(const Blob& rhs) const;
static Memory::Heap* DataHeap;
void* ptr;
SizeT size;
SizeT allocSize;
};
//------------------------------------------------------------------------------
/**
*/
__forceinline void*
Blob::operator new(size_t size)
{
#if NEBULA3_DEBUG
n_assert(size == sizeof(Blob));
#endif
#if NEBULA3_OBJECTS_USE_MEMORYPOOL
return Memory::ObjectPoolAllocator->Alloc(size);
#else
return Memory::Alloc(Memory::ObjectHeap, size);
#endif
}
//------------------------------------------------------------------------------
/**
*/
__forceinline void
Blob::operator delete(void* ptr)
{
#if NEBULA3_OBJECTS_USE_MEMORYPOOL
return Memory::ObjectPoolAllocator->Free(ptr, sizeof(Blob));
#else
return Memory::Free(Memory::ObjectHeap, ptr);
#endif
}
//------------------------------------------------------------------------------
/**
*/
inline void
Blob::Setup()
{
n_assert(0 == DataHeap);
DataHeap = n_new(Memory::Heap("Util.Blob.DataHeap"));
}
//------------------------------------------------------------------------------
/**
*/
inline void
Blob::Shutdown()
{
n_assert(0 != DataHeap);
n_delete(DataHeap);
DataHeap = 0;
}
//------------------------------------------------------------------------------
/**
*/
inline
Blob::Blob() :
ptr(0),
size(0),
allocSize(0)
{
// empty
}
//------------------------------------------------------------------------------
/**
*/
inline bool
Blob::IsValid() const
{
return (0 != this->ptr);
}
//------------------------------------------------------------------------------
/**
*/
inline void
Blob::Delete()
{
if (this->IsValid())
{
n_assert(0 != DataHeap);
DataHeap->Free((void*)this->ptr);
this->ptr = 0;
this->size = 0;
this->allocSize = 0;
}
}
//------------------------------------------------------------------------------
/**
*/
inline
Blob::~Blob()
{
this->Delete();
}
//------------------------------------------------------------------------------
/**
*/
inline void
Blob::Allocate(SizeT s)
{
n_assert(!this->IsValid());
n_assert(0 != DataHeap);
this->ptr = DataHeap->Alloc(s);
this->allocSize = s;
this->size = s;
}
//------------------------------------------------------------------------------
/**
*/
inline void
Blob::Copy(const void* fromPtr, SizeT fromSize)
{
n_assert((0 != fromPtr) && (fromSize > 0));
// only re-allocate if not enough space
if ((0 == this->ptr) || (this->allocSize < fromSize))
{
this->Delete();
this->Allocate(fromSize);
}
this->size = fromSize;
Memory::Copy(fromPtr, (void*) this->ptr, fromSize);
}
//------------------------------------------------------------------------------
/**
*/
inline
Blob::Blob(const void* fromPtr, SizeT fromSize) :
ptr(0),
size(0),
allocSize(0)
{
this->Copy(fromPtr, fromSize);
}
//------------------------------------------------------------------------------
/**
*/
inline
Blob::Blob(const Blob& rhs) :
ptr(0),
size(0),
allocSize(0)
{
if (rhs.IsValid())
{
this->Copy(rhs.ptr, rhs.size);
}
}
//------------------------------------------------------------------------------
/**
*/
inline
Blob::Blob(SizeT s) :
ptr(0),
size(0),
allocSize(0)
{
this->Allocate(s);
}
//------------------------------------------------------------------------------
/**
*/
inline void
Blob::operator=(const Blob& rhs)
{
if (rhs.IsValid())
{
this->Copy(rhs.ptr, rhs.size);
}
}
//------------------------------------------------------------------------------
/**
*/
inline bool
Blob::operator==(const Blob& rhs) const
{
return (this->BinaryCompare(rhs) == 0);
}
//------------------------------------------------------------------------------
/**
*/
inline bool
Blob::operator!=(const Blob& rhs) const
{
return (this->BinaryCompare(rhs) != 0);
}
//------------------------------------------------------------------------------
/**
*/
inline bool
Blob::operator>(const Blob& rhs) const
{
return (this->BinaryCompare(rhs) > 0);
}
//------------------------------------------------------------------------------
/**
*/
inline bool
Blob::operator<(const Blob& rhs) const
{
return (this->BinaryCompare(rhs) < 0);
}
//------------------------------------------------------------------------------
/**
*/
inline bool
Blob::operator>=(const Blob& rhs) const
{
return (this->BinaryCompare(rhs) >= 0);
}
//------------------------------------------------------------------------------
/**
*/
inline bool
Blob::operator<=(const Blob& rhs) const
{
return (this->BinaryCompare(rhs) <= 0);
}
//------------------------------------------------------------------------------
/**
*/
inline void
Blob::Reserve(SizeT s)
{
if (this->allocSize < s)
{
this->Delete();
this->Allocate(s);
}
this->size = s;
}
//------------------------------------------------------------------------------
/**
*/
inline void
Blob::Trim(SizeT trimSize)
{
n_assert(trimSize <= this->size);
this->size = trimSize;
}
//------------------------------------------------------------------------------
/**
*/
inline void
Blob::Set(const void* fromPtr, SizeT fromSize)
{
this->Copy(fromPtr, fromSize);
}
//------------------------------------------------------------------------------
/**
*/
inline void*
Blob::GetPtr() const
{
n_assert(this->IsValid());
return this->ptr;
}
//------------------------------------------------------------------------------
/**
*/
inline SizeT
Blob::Size() const
{
n_assert(this->IsValid());
return this->size;
}
//------------------------------------------------------------------------------
/**
*/
inline IndexT
Blob::HashCode() const
{
IndexT hash = 0;
const char* charPtr = (const char*) this->ptr;
IndexT i;
for (i = 0; i < this->size; i++)
{
hash += charPtr[i];
hash += hash << 10;
hash ^= hash >> 6;
}
hash += hash << 3;
hash ^= hash >> 11;
hash += hash << 15;
hash &= ~(1<<31); // don't return a negative number (in case IndexT is defined signed)
return hash;
}
} // namespace Util
//------------------------------------------------------------------------------