2020-08-11 14:43:53 +02:00
//*************************************************************************************************
//*************************************************************************************************
//
// BUFFER
//
//*************************************************************************************************
//*************************************************************************************************
# if !defined(__XBUFFER_H__)
# define __XBUFFER_H__
# include <XToolsConf.h>
# include "XToolsCommon.h"
# include "XRBuffer.h"
# include "XString.h"
2020-08-12 17:15:47 +02:00
# define XBuffer_Super XRBuffer<T>
template < typename T >
2020-08-11 14:43:53 +02:00
class XBuffer : public XBuffer_Super
{
2021-03-15 10:02:34 +01:00
protected :
2020-08-12 17:15:47 +02:00
T * _WData ; // same as RData (see XRBuffer)
2020-08-11 14:43:53 +02:00
size_t m_allocatedSize ;
2020-08-30 21:53:40 +02:00
void Initialize ( const T * p , size_t count , size_t index ) ;
2020-08-11 14:43:53 +02:00
public :
2020-08-12 17:15:47 +02:00
XBuffer ( ) : _WData ( NULL ) , m_allocatedSize ( 0 ) { Initialize ( NULL , 0 , 0 ) ; } // ": _WData(NULL), m_allocatedSize(0)" to avoid effc++ warning
XBuffer ( const XBuffer < T > & aBuffer ) : _WData ( NULL ) , m_allocatedSize ( 0 ) { Initialize ( aBuffer . data ( ) , aBuffer . size ( ) , aBuffer . index ( ) ) ; }
XBuffer ( XRBuffer < T > & aBuffer , size_t pos = 0 , size_t count = MAX_XSIZE ) ;
// XBuffer(XBuffer &aBuffer, size_t pos = 0, size_t count = MAX_XSIZE);
2020-08-11 14:43:53 +02:00
XBuffer ( void * p , size_t count ) ;
2020-08-12 17:15:47 +02:00
const XBuffer & operator = ( const XRBuffer < T > & aBuffer ) ;
2020-08-11 14:43:53 +02:00
const XBuffer & operator = ( const XBuffer & aBuffer ) ;
2020-08-12 17:15:47 +02:00
template < typename IntegralType , enable_if ( is_integral ( IntegralType ) ) >
2021-03-25 15:32:56 +01:00
void stealValueFrom ( T * p , IntegralType count ) {
2020-08-12 17:15:47 +02:00
if ( count < 0 ) {
2021-02-21 14:49:09 +01:00
# ifdef DEBUG
2020-08-12 17:15:47 +02:00
panic ( " XBuffer::stealValueFrom : count < 0. System halted \n " ) ;
2021-02-21 14:49:09 +01:00
# else
return ;
# endif
2020-08-12 17:15:47 +02:00
}
if ( _WData ) free ( _WData ) ;
2021-03-25 15:32:56 +01:00
m_allocatedSize = count ;
XRBuffer < T > : : _RData = _WData = p ;
XRBuffer < T > : : m_size = count ;
XRBuffer < T > : : _Index = 0 ;
}
template < typename IntegralType , enable_if ( is_integral ( IntegralType ) ) >
void stealValueFrom ( T * p , IntegralType count , IntegralType allocatedSize ) {
if ( count < 0 ) {
panic ( " XBuffer::stealValueFrom : count < 0. System halted \n " ) ;
}
if ( allocatedSize < count ) {
panic ( " XBuffer::stealValueFrom : allocatedSize < count. System halted \n " ) ;
}
if ( _WData ) free ( _WData ) ;
m_allocatedSize = allocatedSize ;
XRBuffer < T > : : _RData = _WData = p ;
XRBuffer < T > : : m_size = count ;
XRBuffer < T > : : _Index = 0 ;
2020-08-12 17:15:47 +02:00
}
2020-08-11 14:43:53 +02:00
~ XBuffer ( ) ;
2021-03-15 10:02:34 +01:00
public :
static XBuffer < T > NullXBuffer ;
2020-08-11 14:43:53 +02:00
void CheckSize ( size_t nNewSize , size_t nGrowBy = XBufferGrowByDefault ) ;
2020-09-08 11:33:35 +02:00
void * vdata ( ) const { return ( void * ) XBuffer_Super : : data ( ) ; }
2020-08-30 21:53:40 +02:00
const T * data ( ) const { return _WData ; }
T * data ( ) { return _WData ; }
2020-08-12 17:15:47 +02:00
template < typename IntegralType , enable_if ( is_integral ( IntegralType ) ) >
const T * data ( IntegralType i ) const { return XBuffer_Super : : data ( i ) ; }
2020-08-13 14:07:40 +02:00
template < typename IntegralType , enable_if ( is_integral ( IntegralType ) ) >
T * data ( IntegralType i ) { return _WData + i ; }
2020-08-12 17:15:47 +02:00
template < typename IntegralType , enable_if ( is_integral ( IntegralType ) ) >
T * dataSized ( IntegralType size ) {
if ( size < 0 ) {
2021-02-21 14:49:09 +01:00
# ifdef DEBUG
2020-08-12 17:15:47 +02:00
panic ( " XBuffer::dataSized : size < 0. System halted \n " ) ;
2021-02-21 14:49:09 +01:00
# else
return NULL ;
# endif
2020-08-12 17:15:47 +02:00
}
CheckSize ( size , 0 ) ;
return data ( ) ;
}
// void *Data(size_t ui=0) { return _WData+ui; }
// void *DataWithSizeMin(size_t ui, size_t size, size_t nGrowBy=XBufferGrowByDefault) { CheckSize(size, nGrowBy); return Data(ui); }
//
// char *CData(size_t ui=0) { return (char *)(_WData+ui); }
// char *CDataWithSizeMin(size_t ui, size_t size, size_t nGrowBy=XBufferGrowByDefault) { CheckSize(size, nGrowBy); return CData(ui); }
//
// unsigned char *UCData(size_t ui=0) { return _WData+ui; }
// void *UCDataWithSizeMin(size_t ui, unsigned int size, size_t nGrowBy=XBufferGrowByDefault) { CheckSize(size, nGrowBy); return UCData(ui); }
size_t size ( ) const { return XRBuffer < T > : : size ( ) ; }
2020-08-11 14:43:53 +02:00
2020-08-12 17:15:47 +02:00
template < typename IntegralType , enable_if ( is_integral ( IntegralType ) ) >
2020-10-12 16:49:43 +02:00
void setSize ( IntegralType size ) {
2021-02-21 14:49:09 +01:00
# ifdef DEBUG
2020-10-12 16:49:43 +02:00
if ( size < 0 ) panic ( " XBuffer::setSize() -> i < 0 " ) ;
if ( ( unsigned_type ( IntegralType ) ) size > MAX_XSIZE ) panic ( " XBuffer::setSize() -> i > MAX_XSIZE " ) ;
2021-02-21 14:49:09 +01:00
# else
if ( size < 0 ) return ;
if ( ( unsigned_type ( IntegralType ) ) size > MAX_XSIZE ) return ;
# endif
2020-10-12 16:49:43 +02:00
CheckSize ( ( unsigned_type ( IntegralType ) ) size ) ;
XBuffer_Super : : m_size = ( unsigned_type ( IntegralType ) ) size ;
} ;
2020-08-13 14:07:40 +02:00
2020-08-12 17:15:47 +02:00
void setEmpty ( ) { setSize ( 0 ) ; } ;
2020-08-11 14:43:53 +02:00
2020-08-30 21:53:40 +02:00
bool operator = = ( const XBuffer < T > & other ) const {
if ( size ( ) ! = other . size ( ) ) return false ;
if ( size ( ) = = 0 ) return true ;
if ( memcmp ( _WData , other . _WData , size ( ) ) ! = 0 ) return false ;
return true ;
}
bool operator ! = ( const XBuffer < T > & other ) const { return ! ( * this = = other ) ; }
2020-08-12 17:15:47 +02:00
/* [] */
template < typename IntegralType , enable_if ( is_integral ( IntegralType ) ) >
T & operator [ ] ( IntegralType i )
{
2021-03-25 15:32:56 +01:00
//#ifdef DEBUG
2020-08-12 17:15:47 +02:00
if ( i < 0 ) panic ( " XBuffer::[] : i < 0. System halted \n " ) ;
if ( ( unsigned_type ( IntegralType ) ) i > = size ( ) ) panic ( " XBuffer::[] : i > _Len. System halted \n " ) ;
2021-03-25 15:32:56 +01:00
//#else
// // Cannot return 0, return value type is T, unknown at this stage.
// if (i < 0) return 0;
// if ( (unsigned_type(IntegralType))i >= size() ) return 0;
//#endif
2020-08-12 17:15:47 +02:00
return _WData [ ( unsigned_type ( IntegralType ) ) i ] ;
}
2020-08-30 21:53:40 +02:00
template < typename IntegralType , enable_if ( is_integral ( IntegralType ) ) >
const T & operator [ ] ( IntegralType i ) const
{
2021-03-25 15:32:56 +01:00
//#ifdef DEBUG
2020-08-30 21:53:40 +02:00
if ( i < 0 ) panic ( " XBuffer::[] : i < 0. System halted \n " ) ;
if ( ( unsigned_type ( IntegralType ) ) i > = size ( ) ) panic ( " XBuffer::[] : i > _Len. System halted \n " ) ;
2021-03-25 15:32:56 +01:00
//#else
// // Cannot return 0, return value type is T, unknown at this stage.
// if (i < 0) return 0;
// if ( (unsigned_type(IntegralType))i >= size() ) return 0;
//#endif
2020-08-30 21:53:40 +02:00
return _WData [ ( unsigned_type ( IntegralType ) ) i ] ;
}
2020-08-11 14:43:53 +02:00
2020-08-12 17:15:47 +02:00
// unsigned char& operator [](int i) { return UCData()[i]; } // underflow ? overflow ?
// unsigned char& operator [](size_t i) { return UCData()[i]; }
2020-08-11 14:43:53 +02:00
2020-08-12 17:15:47 +02:00
void memset ( unsigned char c ) {
memset ( _WData , c , size ( ) ) ;
}
2020-08-11 14:43:53 +02:00
2020-08-12 17:15:47 +02:00
template < typename IntegralType , enable_if ( is_integral ( IntegralType ) ) >
void memset ( unsigned char c , IntegralType count ) {
2021-02-21 14:49:09 +01:00
# ifdef DEBUG
2020-08-12 17:15:47 +02:00
if ( count < 0 ) panic ( " XBuffer::memset : count < 0. System halted \n " ) ;
2021-02-21 14:49:09 +01:00
# else
if ( count < 0 ) return ;
# endif
2020-08-12 17:15:47 +02:00
if ( ( unsigned_type ( IntegralType ) ) count > = size ( ) ) setSize ( count ) ;
: : memset ( _WData , c , count ) ;
}
2020-08-11 14:43:53 +02:00
2020-08-12 17:15:47 +02:00
template < typename IntegralType1 , typename IntegralType2 , enable_if ( is_integral ( IntegralType1 ) & & is_integral ( IntegralType2 ) ) >
void memsetAtPos ( IntegralType1 pos , unsigned char c , IntegralType2 count ) {
2021-02-21 14:49:09 +01:00
# ifdef DEBUG
2020-08-12 17:15:47 +02:00
if ( pos < 0 ) panic ( " XBuffer::memset : pos < 0. System halted \n " ) ;
if ( count < 0 ) panic ( " XBuffer::memset : count < 0. System halted \n " ) ;
2021-02-21 14:49:09 +01:00
# else
if ( pos < 0 ) return ;
if ( count < 0 ) return ;
# endif
2020-08-12 17:15:47 +02:00
if ( ( size_t ) pos + ( unsigned_type ( IntegralType2 ) ) count > = size ( ) ) setSize ( ( size_t ) pos + ( unsigned_type ( IntegralType2 ) ) count ) ;
2021-02-21 14:49:09 +01:00
2020-08-12 17:15:47 +02:00
: : memset ( _WData , c , count ) ;
}
void ncpy ( const void * buf , size_t len ) ;
2020-08-11 14:43:53 +02:00
// Cat
2020-08-13 14:07:40 +02:00
void ncat ( const void * buf , size_t len ) ;
void cat ( bool b ) { ncat ( & b , sizeof ( b ) ) ; } ;
2020-08-11 14:43:53 +02:00
2020-08-13 14:07:40 +02:00
void cat ( char c ) { ncat ( & c , sizeof ( c ) ) ; } ;
void cat ( unsigned char c ) { ncat ( & c , sizeof ( c ) ) ; } ;
void cat ( signed char c ) { ncat ( & c , sizeof ( c ) ) ; } ;
void cat ( signed short s ) { ncat ( & s , sizeof ( s ) ) ; } ;
void cat ( unsigned short us ) { ncat ( & us , sizeof ( us ) ) ; } ;
void cat ( signed int i ) { ncat ( & i , sizeof ( i ) ) ; } ;
void cat ( unsigned int ui ) { ncat ( & ui , sizeof ( ui ) ) ; } ;
void cat ( signed long l ) { ncat ( & l , sizeof ( l ) ) ; } ;
void cat ( unsigned long ul ) { ncat ( & ul , sizeof ( ul ) ) ; } ;
void cat ( signed long long ull ) { ncat ( & ull , sizeof ( ull ) ) ; } ;
void cat ( unsigned long long ull ) { ncat ( & ull , sizeof ( ull ) ) ; } ;
2020-08-11 14:43:53 +02:00
2020-08-13 14:07:40 +02:00
void cat ( float f ) { ncat ( & f , sizeof ( f ) ) ; } ;
void cat ( double d ) { ncat ( & d , sizeof ( d ) ) ; } ;
2020-08-11 14:43:53 +02:00
2020-08-13 14:07:40 +02:00
void cat ( void * p ) { ncat ( & p , sizeof ( p ) ) ; } ;
2020-08-11 14:43:53 +02:00
2020-08-15 15:47:56 +02:00
// void cat(const XString8 &aXString8);
2020-08-13 14:07:40 +02:00
void cat ( const XBuffer & unXBuffer ) { ncat ( unXBuffer . Length ( ) ) ; ncat ( unXBuffer . Data ( ) , unXBuffer . Length ( ) ) ; }
void deleteAtPos ( unsigned int pos , unsigned int count = 1 ) ;
2020-08-11 14:43:53 +02:00
2020-08-12 17:15:47 +02:00
const XBuffer & operator + = ( const XRBuffer < T > & aBuffer ) ;
2020-08-11 14:43:53 +02:00
size_t Sizeof ( ) const ;
// bool ReadFileName(const char* FileName, unsigned int AddZeros=0);
bool ReadFromBuf ( const char * buf , size_t * idx , size_t count ) ;
2020-08-12 17:15:47 +02:00
bool ReadFromXRBuffer ( XRBuffer < T > & unXBuffer ) ;
} ;
2021-03-15 10:02:34 +01:00
template < typename T >
XBuffer < T > XBuffer < T > : : NullXBuffer = XBuffer < T > ( ) ;
2020-08-12 17:15:47 +02:00
//*************************************************************************************************
template < typename T >
2020-08-30 21:53:40 +02:00
void XBuffer < T > : : Initialize ( const T * p , size_t count , size_t index )
2020-08-12 17:15:47 +02:00
{
if ( p ! = NULL & & count > 0 )
{
m_allocatedSize = count ;
_WData = ( unsigned char * ) malloc ( m_allocatedSize ) ;
if ( ! _WData ) {
2021-02-21 14:49:09 +01:00
# ifdef DEBUG
2020-08-12 17:15:47 +02:00
panic ( " XBuffer<T>::Initialize(%zu) : malloc returned NULL. System halted \n " , count ) ;
2021-02-21 14:49:09 +01:00
# else
return ;
# endif
2020-08-12 17:15:47 +02:00
}
memcpy ( _WData , p , count ) ;
XRBuffer < T > : : _RData = _WData ;
XRBuffer < T > : : m_size = count ;
XRBuffer < T > : : _Index = index ;
}
else {
m_allocatedSize = 0 ;
_WData = NULL ;
XRBuffer < T > : : _RData = NULL ;
XRBuffer < T > : : m_size = 0 ;
XRBuffer < T > : : _Index = 0 ;
}
}
//-------------------------------------------------------------------------------------------------
// CheckSize
//-------------------------------------------------------------------------------------------------
template < typename T >
void XBuffer < T > : : CheckSize ( size_t nNewSize , size_t nGrowBy )
{
if ( m_allocatedSize < nNewSize )
{
nNewSize + = nGrowBy ;
_WData = ( unsigned char * ) Xrealloc ( _WData , nNewSize , m_allocatedSize ) ;
if ( ! _WData ) {
2021-02-21 14:49:09 +01:00
# ifdef DEBUG
2020-08-12 17:15:47 +02:00
panic ( " XBuffer<T>::CheckSize(%zu, %zu) : Xrealloc(% " PRIuPTR " %zu, %zu) returned NULL. System halted \n " , nNewSize , nGrowBy , uintptr_t ( _WData ) , nNewSize , m_allocatedSize ) ;
2021-02-21 14:49:09 +01:00
# else
m_allocatedSize = 0 ;
return ;
# endif
2020-08-12 17:15:47 +02:00
}
XRBuffer < T > : : _RData = _WData ;
m_allocatedSize = nNewSize ;
}
}
//-------------------------------------------------------------------------------------------------
// ctor
//-------------------------------------------------------------------------------------------------
template < typename T >
XBuffer < T > : : XBuffer ( XRBuffer < T > & aXRBuffer , size_t pos , size_t count ) : _WData ( NULL ) , m_allocatedSize ( 0 ) // initialisation to avoid effc++ warning
{
if ( pos < aXRBuffer . Length ( ) ) {
Initialize ( NULL , 0 , 0 ) ;
} else {
if ( count > aXRBuffer . Length ( ) - pos ) count = aXRBuffer . Length ( ) - pos ;
Initialize ( aXRBuffer . UCData ( pos ) , count , aXRBuffer . Index ( ) ) ;
}
}
//
//template <typename T>
//XBuffer<T>::XBuffer(XBuffer &aXBuffer, size_t pos, size_t count) : _WData(NULL), m_allocatedSize(0) // initialisation to avoid effc++ warning
//{
// if ( pos >= aXBuffer.Length() ) {
// Initialize(NULL, 0, 0);
// }else{
// if ( count > aXBuffer.Length()-pos ) count = aXBuffer.Length()-pos;
// Initialize(aXBuffer.UCData(pos), count, aXBuffer.Index());
// }
//}
template < typename T >
XBuffer < T > : : XBuffer ( void * p , size_t count ) : _WData ( NULL ) , m_allocatedSize ( 0 ) // initialisation to avoid effc++ warning
{
Initialize ( ( const unsigned char * ) p , count , 0 ) ;
}
template < typename T >
XBuffer < T > : : ~ XBuffer ( )
{
free ( _WData ) ;
}
//-------------------------------------------------------------------------------------------------
// operator =, +=
//-------------------------------------------------------------------------------------------------
template < typename T >
const XBuffer < T > & XBuffer < T > : : operator = ( const XRBuffer < T > & aBuffer )
{
//TRACE("Operator =const XBuffer&\n");
Cpy ( aBuffer . Data ( ) , aBuffer . Length ( ) ) ;
SetIndex ( aBuffer . Index ( ) ) ;
return * this ;
}
template < typename T >
const XBuffer < T > & XBuffer < T > : : operator = ( const XBuffer < T > & aBuffer )
{
//TRACE("Operator =const XBuffer&\n");
ncpy ( aBuffer . data ( ) , aBuffer . size ( ) ) ;
XBuffer_Super : : setIndex ( aBuffer . index ( ) ) ;
return * this ;
}
template < typename T >
const XBuffer < T > & XBuffer < T > : : operator + = ( const XRBuffer < T > & aBuffer )
{
//TRACE("Operator +=const XBuffer&\n");
Cat ( aBuffer . Data ( ) , aBuffer . Length ( ) ) ;
return * this ;
}
//-------------------------------------------------------------------------------------------------
// Cpy, Cat, Delete
//-------------------------------------------------------------------------------------------------
template < typename T >
void XBuffer < T > : : ncpy ( const void * buf , size_t len )
{
if ( buf & & len > 0 ) {
setSize ( len ) ;
memcpy ( data ( ) , buf , len ) ;
}
}
template < typename T >
2020-08-13 14:07:40 +02:00
void XBuffer < T > : : ncat ( const void * buf , size_t len )
2020-08-12 17:15:47 +02:00
{
if ( buf & & len > 0 ) {
CheckSize ( size ( ) + len ) ;
2020-08-13 14:07:40 +02:00
memcpy ( data ( size ( ) ) , buf , len ) ;
setSize ( size ( ) + len ) ;
2020-08-12 17:15:47 +02:00
}
}
2020-08-15 15:47:56 +02:00
//
//template <typename T>
//void XBuffer<T>::cat(const XString8 &aXString8)
//{
// cat(aXString8.sizeInBytes());
// cat(aXString8.s(), aXString8.sizeInBytes());
//};
2020-08-11 14:43:53 +02:00
2020-08-12 17:15:47 +02:00
template < typename T >
2020-08-13 14:07:40 +02:00
void XBuffer < T > : : deleteAtPos ( unsigned int pos , unsigned int count )
2020-08-12 17:15:47 +02:00
{
if ( pos < size ( ) ) {
if ( pos + count < size ( ) ) {
memmove ( _WData + pos , _WData + pos + count , size ( ) - pos - count ) ;
setSize ( size ( ) - count ) ;
} else {
setSize ( pos ) ;
}
}
}
//-------------------------------------------------------------------------------------------------
// ReadFrom
//-------------------------------------------------------------------------------------------------
//bool XBuffer<T>::ReadFileName(const char* FileName, unsigned int AddZeros)
//{
// FILE *fp;
// long int ulen;
// unsigned int len;
//
// fp = fopen(FileName, "rb");
// if ( fp == NULL ) {
// SetLastErrorMsgf("Impossible d'ouvrir le fichier en lecture '%s'", FileName);
// goto finally;
// }
//
// fseek(fp, 0, 2); // Seek to end of file
// ulen = ftell(fp);
// if ( ulen <= 0 ) {
// SetLastErrorMsgf("Impossible d'avoir la longueur du fichier '%s'", FileName);
// goto finally;
// }
//
// len = (unsigned int)ulen;
// fseek(fp, 0, 0);
//
// if ( fread(CDataWithSizeMin(0, len+AddZeros), 1, len, fp) != len ) {
// SetLastErrorMsgf("Impossible de lire %d octets du fichier '%s'", len, FileName);
// goto finally;
// }
// SetLength(len);
// SetIndex(0);
// if ( fclose(fp) != 0 ) {
// SetLastErrorMsgf("Impossible de fermer les fichier '%s'", FileName);
// goto finally;
// }
// {for ( unsigned int ui=len ; ui<len+AddZeros ; ui+=1 ) *CData(ui) = 0;}
// return YES;
// finally:
// if ( fp != NULL ) fclose(fp);
// SetLength(0);
// return false;
//
//}
//bool XBuffer<T>::ReadFromFILE(FILE *fp)
//{
// unsigned int longueur;
//
// if ( fread(&longueur, sizeof(longueur), 1, fp) != 1 ) goto fin;
// if ( longueur > 0 && fread(DataWithSizeMin(0, longueur, 0), longueur, 1, fp) != 1 ) goto fin;
// SetLength(longueur);
// SetIndex(0);
// return OUI;
// fin:
// SetNull();
// return NON;
//}
template < typename T >
bool XBuffer < T > : : ReadFromXRBuffer ( XRBuffer < T > & unXBuffer )
{
size_t longueur ;
if ( ! unXBuffer . GetSize_t ( & longueur ) ) return false ;
if ( ! unXBuffer . Get ( dataSized ( longueur ) , longueur ) ) return false ; // GrowBy (param 3 of DataWithSizeMin is 0 to avoid memory waste
setSize ( longueur ) ;
XBuffer_Super : : setIndex ( 0 ) ;
return true ;
}
template < typename T >
bool XBuffer < T > : : ReadFromBuf ( const char * buf , size_t * idx , size_t count )
{
size_t longueur ;
if ( count - * idx > = sizeof ( longueur ) ) {
longueur = * ( ( size_t * ) ( buf + * idx ) ) ;
* idx + = sizeof ( longueur ) ;
if ( longueur > 0 & & count - * idx > = longueur ) memcpy ( dataSized ( longueur ) , buf + * idx , longueur ) ;
* idx + = longueur ;
setSize ( longueur ) ;
XBuffer_Super : : setIndex ( 0 ) ;
return true ;
} else {
setEmpty ( ) ;
return false ;
}
}
2020-08-11 14:43:53 +02:00
# endif