mirror of
https://github.com/bkaradzic/bx.git
synced 2026-02-19 05:23:00 +01:00
Cleanup.
This commit is contained in:
@@ -24,6 +24,7 @@ BX_ERROR_RESULT(BX_ERROR_READERWRITER_ALREADY_OPEN, BX_MAKEFOURCC('R', 'W', 0, 5
|
||||
|
||||
namespace bx
|
||||
{
|
||||
///
|
||||
struct Whence
|
||||
{
|
||||
enum Enum
|
||||
@@ -34,350 +35,111 @@ namespace bx
|
||||
};
|
||||
};
|
||||
|
||||
///
|
||||
struct BX_NO_VTABLE ReaderI
|
||||
{
|
||||
virtual ~ReaderI() = 0;
|
||||
virtual int32_t read(void* _data, int32_t _size, Error* _err) = 0;
|
||||
};
|
||||
|
||||
inline ReaderI::~ReaderI()
|
||||
{
|
||||
}
|
||||
|
||||
///
|
||||
struct BX_NO_VTABLE WriterI
|
||||
{
|
||||
virtual ~WriterI() = 0;
|
||||
virtual int32_t write(const void* _data, int32_t _size, Error* _err) = 0;
|
||||
};
|
||||
|
||||
inline WriterI::~WriterI()
|
||||
{
|
||||
}
|
||||
|
||||
///
|
||||
struct BX_NO_VTABLE SeekerI
|
||||
{
|
||||
virtual ~SeekerI() = 0;
|
||||
virtual int64_t seek(int64_t _offset = 0, Whence::Enum _whence = Whence::Current) = 0;
|
||||
};
|
||||
|
||||
inline SeekerI::~SeekerI()
|
||||
{
|
||||
}
|
||||
|
||||
/// Read data.
|
||||
inline int32_t read(ReaderI* _reader, void* _data, int32_t _size, Error* _err = NULL)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
return _reader->read(_data, _size, _err);
|
||||
}
|
||||
|
||||
/// Read value.
|
||||
template<typename Ty>
|
||||
inline int32_t read(ReaderI* _reader, Ty& _value, Error* _err = NULL)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
BX_STATIC_ASSERT(BX_TYPE_IS_POD(Ty) );
|
||||
return _reader->read(&_value, sizeof(Ty), _err);
|
||||
}
|
||||
|
||||
/// Read value and converts it to host endianess. _fromLittleEndian specifies
|
||||
/// underlying stream endianess.
|
||||
template<typename Ty>
|
||||
inline int32_t readHE(ReaderI* _reader, Ty& _value, bool _fromLittleEndian, Error* _err = NULL)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
BX_STATIC_ASSERT(BX_TYPE_IS_POD(Ty) );
|
||||
Ty value;
|
||||
int32_t result = _reader->read(&value, sizeof(Ty), _err);
|
||||
_value = toHostEndian(value, _fromLittleEndian);
|
||||
return result;
|
||||
}
|
||||
|
||||
/// Write data.
|
||||
inline int32_t write(WriterI* _writer, const void* _data, int32_t _size, Error* _err = NULL)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
return _writer->write(_data, _size, _err);
|
||||
}
|
||||
|
||||
/// Write repeat the same value.
|
||||
inline int32_t writeRep(WriterI* _writer, uint8_t _byte, int32_t _size, Error* _err = NULL)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
|
||||
const uint32_t tmp0 = uint32_sels(64 - _size, 64, _size);
|
||||
const uint32_t tmp1 = uint32_sels(256 - _size, 256, tmp0);
|
||||
const uint32_t blockSize = uint32_sels(1024 - _size, 1024, tmp1);
|
||||
uint8_t* temp = (uint8_t*)alloca(blockSize);
|
||||
memset(temp, _byte, blockSize);
|
||||
|
||||
int32_t size = 0;
|
||||
while (0 < _size)
|
||||
{
|
||||
int32_t bytes = write(_writer, temp, uint32_min(blockSize, _size), _err);
|
||||
size += bytes;
|
||||
_size -= bytes;
|
||||
}
|
||||
|
||||
return size;
|
||||
}
|
||||
|
||||
/// Write value.
|
||||
template<typename Ty>
|
||||
inline int32_t write(WriterI* _writer, const Ty& _value, Error* _err = NULL)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
BX_STATIC_ASSERT(BX_TYPE_IS_POD(Ty) );
|
||||
return _writer->write(&_value, sizeof(Ty), _err);
|
||||
}
|
||||
|
||||
/// Write value as little endian.
|
||||
template<typename Ty>
|
||||
inline int32_t writeLE(WriterI* _writer, const Ty& _value, Error* _err = NULL)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
BX_STATIC_ASSERT(BX_TYPE_IS_POD(Ty) );
|
||||
Ty value = toLittleEndian(_value);
|
||||
int32_t result = _writer->write(&value, sizeof(Ty), _err);
|
||||
return result;
|
||||
}
|
||||
|
||||
/// Write value as big endian.
|
||||
template<typename Ty>
|
||||
inline int32_t writeBE(WriterI* _writer, const Ty& _value, Error* _err = NULL)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
BX_STATIC_ASSERT(BX_TYPE_IS_POD(Ty) );
|
||||
Ty value = toBigEndian(_value);
|
||||
int32_t result = _writer->write(&value, sizeof(Ty), _err);
|
||||
return result;
|
||||
}
|
||||
|
||||
/// Write formated string.
|
||||
inline int32_t writePrintf(WriterI* _writer, const char* _format, ...)
|
||||
{
|
||||
va_list argList;
|
||||
va_start(argList, _format);
|
||||
|
||||
char temp[2048];
|
||||
char* out = temp;
|
||||
int32_t max = sizeof(temp);
|
||||
int32_t len = vsnprintf(out, max, _format, argList);
|
||||
if (len > max)
|
||||
{
|
||||
out = (char*)alloca(len);
|
||||
len = vsnprintf(out, len, _format, argList);
|
||||
}
|
||||
|
||||
int32_t size = write(_writer, out, len);
|
||||
|
||||
va_end(argList);
|
||||
|
||||
return size;
|
||||
}
|
||||
|
||||
/// Skip _offset bytes forward.
|
||||
inline int64_t skip(SeekerI* _seeker, int64_t _offset)
|
||||
{
|
||||
return _seeker->seek(_offset, Whence::Current);
|
||||
}
|
||||
|
||||
/// Seek to any position in file.
|
||||
inline int64_t seek(SeekerI* _seeker, int64_t _offset = 0, Whence::Enum _whence = Whence::Current)
|
||||
{
|
||||
return _seeker->seek(_offset, _whence);
|
||||
}
|
||||
|
||||
/// Returns size of file.
|
||||
inline int64_t getSize(SeekerI* _seeker)
|
||||
{
|
||||
int64_t offset = _seeker->seek();
|
||||
int64_t size = _seeker->seek(0, Whence::End);
|
||||
_seeker->seek(offset, Whence::Begin);
|
||||
return size;
|
||||
}
|
||||
|
||||
///
|
||||
struct BX_NO_VTABLE ReaderSeekerI : public ReaderI, public SeekerI
|
||||
{
|
||||
};
|
||||
|
||||
/// Peek data.
|
||||
inline int32_t peek(ReaderSeekerI* _reader, void* _data, int32_t _size, Error* _err = NULL)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
int64_t offset = bx::seek(_reader);
|
||||
int32_t size = _reader->read(_data, _size, _err);
|
||||
bx::seek(_reader, offset, bx::Whence::Begin);
|
||||
return size;
|
||||
}
|
||||
|
||||
/// Peek value.
|
||||
template<typename Ty>
|
||||
inline int32_t peek(ReaderSeekerI* _reader, Ty& _value, Error* _err = NULL)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
BX_STATIC_ASSERT(BX_TYPE_IS_POD(Ty) );
|
||||
return peek(_reader, &_value, sizeof(Ty), _err);
|
||||
}
|
||||
|
||||
///
|
||||
struct BX_NO_VTABLE WriterSeekerI : public WriterI, public SeekerI
|
||||
{
|
||||
};
|
||||
|
||||
/// Align reader stream.
|
||||
inline int32_t align(ReaderSeekerI* _reader, uint32_t _alignment, Error* _err = NULL)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
const int64_t current = bx::seek(_reader);
|
||||
const int64_t aligned = ( (current + _alignment-1)/_alignment) * _alignment;
|
||||
const int32_t size = int32_t(aligned - current);
|
||||
if (0 != size)
|
||||
{
|
||||
const int64_t offset = bx::seek(_reader, size);
|
||||
if (offset != aligned)
|
||||
{
|
||||
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_WRITE, "Align: read truncated.");
|
||||
}
|
||||
return int32_t(offset - current);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/// Align writer stream (pads stream with zeros).
|
||||
inline int32_t align(WriterSeekerI* _writer, uint32_t _alignment, Error* _err = NULL)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
const int64_t current = bx::seek(_writer);
|
||||
const int64_t aligned = ( (current + _alignment-1)/_alignment) * _alignment;
|
||||
const int32_t size = int32_t(aligned - current);
|
||||
if (0 != size)
|
||||
{
|
||||
return writeRep(_writer, 0, size, _err);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
///
|
||||
struct BX_NO_VTABLE ReaderOpenI
|
||||
{
|
||||
virtual ~ReaderOpenI() = 0;
|
||||
virtual bool open(const char* _filePath, Error* _err) = 0;
|
||||
};
|
||||
|
||||
inline ReaderOpenI::~ReaderOpenI()
|
||||
{
|
||||
}
|
||||
|
||||
///
|
||||
struct BX_NO_VTABLE WriterOpenI
|
||||
{
|
||||
virtual ~WriterOpenI() = 0;
|
||||
virtual bool open(const char* _filePath, bool _append, Error* _err) = 0;
|
||||
};
|
||||
|
||||
inline WriterOpenI::~WriterOpenI()
|
||||
{
|
||||
}
|
||||
|
||||
///
|
||||
struct BX_NO_VTABLE CloserI
|
||||
{
|
||||
virtual ~CloserI() = 0;
|
||||
virtual void close() = 0;
|
||||
};
|
||||
|
||||
inline CloserI::~CloserI()
|
||||
{
|
||||
}
|
||||
|
||||
///
|
||||
struct BX_NO_VTABLE FileReaderI : public ReaderOpenI, public CloserI, public ReaderSeekerI
|
||||
{
|
||||
};
|
||||
|
||||
///
|
||||
struct BX_NO_VTABLE FileWriterI : public WriterOpenI, public CloserI, public WriterSeekerI
|
||||
{
|
||||
};
|
||||
|
||||
inline bool open(ReaderOpenI* _reader, const char* _filePath, Error* _err = NULL)
|
||||
{
|
||||
BX_ERROR_USE_TEMP_WHEN_NULL(_err);
|
||||
return _reader->open(_filePath, _err);
|
||||
}
|
||||
|
||||
inline bool open(WriterOpenI* _writer, const char* _filePath, bool _append = false, Error* _err = NULL)
|
||||
{
|
||||
BX_ERROR_USE_TEMP_WHEN_NULL(_err);
|
||||
return _writer->open(_filePath, _append, _err);
|
||||
}
|
||||
|
||||
inline void close(CloserI* _reader)
|
||||
{
|
||||
_reader->close();
|
||||
}
|
||||
|
||||
///
|
||||
struct BX_NO_VTABLE MemoryBlockI
|
||||
{
|
||||
virtual void* more(uint32_t _size = 0) = 0;
|
||||
virtual uint32_t getSize() = 0;
|
||||
};
|
||||
|
||||
///
|
||||
class StaticMemoryBlock : public MemoryBlockI
|
||||
{
|
||||
public:
|
||||
StaticMemoryBlock(void* _data, uint32_t _size)
|
||||
: m_data(_data)
|
||||
, m_size(_size)
|
||||
{
|
||||
}
|
||||
///
|
||||
StaticMemoryBlock(void* _data, uint32_t _size);
|
||||
|
||||
virtual ~StaticMemoryBlock()
|
||||
{
|
||||
}
|
||||
///
|
||||
virtual ~StaticMemoryBlock();
|
||||
|
||||
virtual void* more(uint32_t /*_size*/ = 0) BX_OVERRIDE
|
||||
{
|
||||
return m_data;
|
||||
}
|
||||
///
|
||||
virtual void* more(uint32_t _size = 0);
|
||||
|
||||
virtual uint32_t getSize() BX_OVERRIDE
|
||||
{
|
||||
return m_size;
|
||||
}
|
||||
///
|
||||
virtual uint32_t getSize() BX_OVERRIDE;
|
||||
|
||||
private:
|
||||
void* m_data;
|
||||
uint32_t m_size;
|
||||
};
|
||||
|
||||
///
|
||||
class MemoryBlock : public MemoryBlockI
|
||||
{
|
||||
public:
|
||||
MemoryBlock(AllocatorI* _allocator)
|
||||
: m_allocator(_allocator)
|
||||
, m_data(NULL)
|
||||
, m_size(0)
|
||||
{
|
||||
}
|
||||
///
|
||||
MemoryBlock(AllocatorI* _allocator);
|
||||
|
||||
virtual ~MemoryBlock()
|
||||
{
|
||||
BX_FREE(m_allocator, m_data);
|
||||
}
|
||||
///
|
||||
virtual ~MemoryBlock();
|
||||
|
||||
virtual void* more(uint32_t _size = 0) BX_OVERRIDE
|
||||
{
|
||||
if (0 < _size)
|
||||
{
|
||||
m_size += _size;
|
||||
m_data = BX_REALLOC(m_allocator, m_data, m_size);
|
||||
}
|
||||
///
|
||||
virtual void* more(uint32_t _size = 0) BX_OVERRIDE;
|
||||
|
||||
return m_data;
|
||||
}
|
||||
|
||||
virtual uint32_t getSize() BX_OVERRIDE
|
||||
{
|
||||
return m_size;
|
||||
}
|
||||
///
|
||||
virtual uint32_t getSize() BX_OVERRIDE;
|
||||
|
||||
private:
|
||||
AllocatorI* m_allocator;
|
||||
@@ -385,128 +147,51 @@ namespace bx
|
||||
uint32_t m_size;
|
||||
};
|
||||
|
||||
///
|
||||
class SizerWriter : public WriterSeekerI
|
||||
{
|
||||
public:
|
||||
SizerWriter()
|
||||
: m_pos(0)
|
||||
, m_top(0)
|
||||
{
|
||||
}
|
||||
///
|
||||
SizerWriter();
|
||||
|
||||
virtual ~SizerWriter()
|
||||
{
|
||||
}
|
||||
///
|
||||
virtual ~SizerWriter();
|
||||
|
||||
virtual int64_t seek(int64_t _offset = 0, Whence::Enum _whence = Whence::Current) BX_OVERRIDE
|
||||
{
|
||||
switch (_whence)
|
||||
{
|
||||
case Whence::Begin:
|
||||
m_pos = int64_clamp(_offset, 0, m_top);
|
||||
break;
|
||||
///
|
||||
virtual int64_t seek(int64_t _offset = 0, Whence::Enum _whence = Whence::Current) BX_OVERRIDE;
|
||||
|
||||
case Whence::Current:
|
||||
m_pos = int64_clamp(m_pos + _offset, 0, m_top);
|
||||
break;
|
||||
|
||||
case Whence::End:
|
||||
m_pos = int64_clamp(m_top - _offset, 0, m_top);
|
||||
break;
|
||||
}
|
||||
|
||||
return m_pos;
|
||||
}
|
||||
|
||||
virtual int32_t write(const void* /*_data*/, int32_t _size, Error* _err) BX_OVERRIDE
|
||||
{
|
||||
BX_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
||||
|
||||
int32_t morecore = int32_t(m_pos - m_top) + _size;
|
||||
|
||||
if (0 < morecore)
|
||||
{
|
||||
m_top += morecore;
|
||||
}
|
||||
|
||||
int64_t remainder = m_top-m_pos;
|
||||
int32_t size = uint32_min(_size, uint32_t(int64_min(remainder, INT32_MAX) ) );
|
||||
m_pos += size;
|
||||
if (size != _size)
|
||||
{
|
||||
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_WRITE, "SizerWriter: write truncated.");
|
||||
}
|
||||
return size;
|
||||
}
|
||||
///
|
||||
virtual int32_t write(const void* /*_data*/, int32_t _size, Error* _err) BX_OVERRIDE;
|
||||
|
||||
private:
|
||||
int64_t m_pos;
|
||||
int64_t m_top;
|
||||
};
|
||||
|
||||
///
|
||||
class MemoryReader : public ReaderSeekerI
|
||||
{
|
||||
public:
|
||||
MemoryReader(const void* _data, uint32_t _size)
|
||||
: m_data( (const uint8_t*)_data)
|
||||
, m_pos(0)
|
||||
, m_top(_size)
|
||||
{
|
||||
}
|
||||
///
|
||||
MemoryReader(const void* _data, uint32_t _size);
|
||||
|
||||
virtual ~MemoryReader()
|
||||
{
|
||||
}
|
||||
///
|
||||
virtual ~MemoryReader();
|
||||
|
||||
virtual int64_t seek(int64_t _offset, Whence::Enum _whence) BX_OVERRIDE
|
||||
{
|
||||
switch (_whence)
|
||||
{
|
||||
case Whence::Begin:
|
||||
m_pos = int64_clamp(_offset, 0, m_top);
|
||||
break;
|
||||
///
|
||||
virtual int64_t seek(int64_t _offset, Whence::Enum _whence) BX_OVERRIDE;
|
||||
|
||||
case Whence::Current:
|
||||
m_pos = int64_clamp(m_pos + _offset, 0, m_top);
|
||||
break;
|
||||
///
|
||||
virtual int32_t read(void* _data, int32_t _size, Error* _err) BX_OVERRIDE;
|
||||
|
||||
case Whence::End:
|
||||
m_pos = int64_clamp(m_top - _offset, 0, m_top);
|
||||
break;
|
||||
}
|
||||
///
|
||||
const uint8_t* getDataPtr() const;
|
||||
|
||||
return m_pos;
|
||||
}
|
||||
///
|
||||
int64_t getPos() const;
|
||||
|
||||
virtual int32_t read(void* _data, int32_t _size, Error* _err) BX_OVERRIDE
|
||||
{
|
||||
BX_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
||||
|
||||
int64_t remainder = m_top-m_pos;
|
||||
int32_t size = uint32_min(_size, uint32_t(int64_min(remainder, INT32_MAX) ) );
|
||||
memcpy(_data, &m_data[m_pos], size);
|
||||
m_pos += size;
|
||||
if (size != _size)
|
||||
{
|
||||
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_READ, "MemoryReader: read truncated.");
|
||||
}
|
||||
return size;
|
||||
}
|
||||
|
||||
const uint8_t* getDataPtr() const
|
||||
{
|
||||
return &m_data[m_pos];
|
||||
}
|
||||
|
||||
int64_t getPos() const
|
||||
{
|
||||
return m_pos;
|
||||
}
|
||||
|
||||
int64_t remaining() const
|
||||
{
|
||||
return m_top-m_pos;
|
||||
}
|
||||
///
|
||||
int64_t remaining() const;
|
||||
|
||||
private:
|
||||
const uint8_t* m_data;
|
||||
@@ -514,66 +199,21 @@ namespace bx
|
||||
int64_t m_top;
|
||||
};
|
||||
|
||||
///
|
||||
class MemoryWriter : public WriterSeekerI
|
||||
{
|
||||
public:
|
||||
MemoryWriter(MemoryBlockI* _memBlock)
|
||||
: m_memBlock(_memBlock)
|
||||
, m_data(NULL)
|
||||
, m_pos(0)
|
||||
, m_top(0)
|
||||
, m_size(0)
|
||||
{
|
||||
}
|
||||
///
|
||||
MemoryWriter(MemoryBlockI* _memBlock);
|
||||
|
||||
virtual ~MemoryWriter()
|
||||
{
|
||||
}
|
||||
///
|
||||
virtual ~MemoryWriter();
|
||||
|
||||
virtual int64_t seek(int64_t _offset = 0, Whence::Enum _whence = Whence::Current) BX_OVERRIDE
|
||||
{
|
||||
switch (_whence)
|
||||
{
|
||||
case Whence::Begin:
|
||||
m_pos = int64_clamp(_offset, 0, m_top);
|
||||
break;
|
||||
///
|
||||
virtual int64_t seek(int64_t _offset = 0, Whence::Enum _whence = Whence::Current) BX_OVERRIDE;
|
||||
|
||||
case Whence::Current:
|
||||
m_pos = int64_clamp(m_pos + _offset, 0, m_top);
|
||||
break;
|
||||
|
||||
case Whence::End:
|
||||
m_pos = int64_clamp(m_top - _offset, 0, m_top);
|
||||
break;
|
||||
}
|
||||
|
||||
return m_pos;
|
||||
}
|
||||
|
||||
virtual int32_t write(const void* _data, int32_t _size, Error* _err) BX_OVERRIDE
|
||||
{
|
||||
BX_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
||||
|
||||
int32_t morecore = int32_t(m_pos - m_size) + _size;
|
||||
|
||||
if (0 < morecore)
|
||||
{
|
||||
morecore = BX_ALIGN_MASK(morecore, 0xfff);
|
||||
m_data = (uint8_t*)m_memBlock->more(morecore);
|
||||
m_size = m_memBlock->getSize();
|
||||
}
|
||||
|
||||
int64_t remainder = m_size-m_pos;
|
||||
int32_t size = uint32_min(_size, uint32_t(int64_min(remainder, INT32_MAX) ) );
|
||||
memcpy(&m_data[m_pos], _data, size);
|
||||
m_pos += size;
|
||||
m_top = int64_max(m_top, m_pos);
|
||||
if (size != _size)
|
||||
{
|
||||
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_WRITE, "MemoryWriter: write truncated.");
|
||||
}
|
||||
return size;
|
||||
}
|
||||
///
|
||||
virtual int32_t write(const void* _data, int32_t _size, Error* _err) BX_OVERRIDE;
|
||||
|
||||
private:
|
||||
MemoryBlockI* m_memBlock;
|
||||
@@ -583,23 +223,86 @@ namespace bx
|
||||
int64_t m_size;
|
||||
};
|
||||
|
||||
///
|
||||
class StaticMemoryBlockWriter : public MemoryWriter
|
||||
{
|
||||
public:
|
||||
StaticMemoryBlockWriter(void* _data, uint32_t _size)
|
||||
: MemoryWriter(&m_smb)
|
||||
, m_smb(_data, _size)
|
||||
{
|
||||
}
|
||||
///
|
||||
StaticMemoryBlockWriter(void* _data, uint32_t _size);
|
||||
|
||||
~StaticMemoryBlockWriter()
|
||||
{
|
||||
}
|
||||
///
|
||||
virtual ~StaticMemoryBlockWriter();
|
||||
|
||||
private:
|
||||
StaticMemoryBlock m_smb;
|
||||
};
|
||||
|
||||
/// Read data.
|
||||
int32_t read(ReaderI* _reader, void* _data, int32_t _size, Error* _err = NULL);
|
||||
|
||||
/// Read value.
|
||||
template<typename Ty>
|
||||
int32_t read(ReaderI* _reader, Ty& _value, Error* _err = NULL);
|
||||
|
||||
/// Read value and converts it to host endianess. _fromLittleEndian specifies
|
||||
/// underlying stream endianess.
|
||||
template<typename Ty>
|
||||
int32_t readHE(ReaderI* _reader, Ty& _value, bool _fromLittleEndian, Error* _err = NULL);
|
||||
|
||||
/// Write data.
|
||||
int32_t write(WriterI* _writer, const void* _data, int32_t _size, Error* _err = NULL);
|
||||
|
||||
/// Write repeat the same value.
|
||||
int32_t writeRep(WriterI* _writer, uint8_t _byte, int32_t _size, Error* _err = NULL);
|
||||
|
||||
/// Write value.
|
||||
template<typename Ty>
|
||||
int32_t write(WriterI* _writer, const Ty& _value, Error* _err = NULL);
|
||||
|
||||
/// Write value as little endian.
|
||||
template<typename Ty>
|
||||
int32_t writeLE(WriterI* _writer, const Ty& _value, Error* _err = NULL);
|
||||
|
||||
/// Write value as big endian.
|
||||
template<typename Ty>
|
||||
int32_t writeBE(WriterI* _writer, const Ty& _value, Error* _err = NULL);
|
||||
|
||||
/// Write formated string.
|
||||
int32_t writePrintf(WriterI* _writer, const char* _format, ...);
|
||||
|
||||
/// Skip _offset bytes forward.
|
||||
int64_t skip(SeekerI* _seeker, int64_t _offset);
|
||||
|
||||
/// Seek to any position in file.
|
||||
int64_t seek(SeekerI* _seeker, int64_t _offset = 0, Whence::Enum _whence = Whence::Current);
|
||||
|
||||
/// Returns size of file.
|
||||
int64_t getSize(SeekerI* _seeker);
|
||||
|
||||
/// Peek data.
|
||||
int32_t peek(ReaderSeekerI* _reader, void* _data, int32_t _size, Error* _err = NULL);
|
||||
|
||||
/// Peek value.
|
||||
template<typename Ty>
|
||||
int32_t peek(ReaderSeekerI* _reader, Ty& _value, Error* _err = NULL);
|
||||
|
||||
/// Align reader stream.
|
||||
int32_t align(ReaderSeekerI* _reader, uint32_t _alignment, Error* _err = NULL);
|
||||
|
||||
/// Align writer stream (pads stream with zeros).
|
||||
int32_t align(WriterSeekerI* _writer, uint32_t _alignment, Error* _err = NULL);
|
||||
|
||||
///
|
||||
bool open(ReaderOpenI* _reader, const char* _filePath, Error* _err = NULL);
|
||||
|
||||
///
|
||||
bool open(WriterOpenI* _writer, const char* _filePath, bool _append = false, Error* _err = NULL);
|
||||
|
||||
///
|
||||
void close(CloserI* _reader);
|
||||
|
||||
} // namespace bx
|
||||
|
||||
#include "readerwriter.inl"
|
||||
|
||||
#endif // BX_READERWRITER_H_HEADER_GUARD
|
||||
|
||||
452
include/bx/readerwriter.inl
Normal file
452
include/bx/readerwriter.inl
Normal file
@@ -0,0 +1,452 @@
|
||||
/*
|
||||
* Copyright 2010-2017 Branimir Karadzic. All rights reserved.
|
||||
* License: https://github.com/bkaradzic/bx#license-bsd-2-clause
|
||||
*/
|
||||
|
||||
#ifndef BX_READERWRITER_H_HEADER_GUARD
|
||||
# error "Must be included from bx/readerwriter!"
|
||||
#endif // BX_READERWRITER_H_HEADER_GUARD
|
||||
|
||||
namespace bx
|
||||
{
|
||||
inline ReaderI::~ReaderI()
|
||||
{
|
||||
}
|
||||
|
||||
inline WriterI::~WriterI()
|
||||
{
|
||||
}
|
||||
|
||||
inline SeekerI::~SeekerI()
|
||||
{
|
||||
}
|
||||
|
||||
inline ReaderOpenI::~ReaderOpenI()
|
||||
{
|
||||
}
|
||||
|
||||
inline WriterOpenI::~WriterOpenI()
|
||||
{
|
||||
}
|
||||
|
||||
inline CloserI::~CloserI()
|
||||
{
|
||||
}
|
||||
|
||||
inline StaticMemoryBlock::StaticMemoryBlock(void* _data, uint32_t _size)
|
||||
: m_data(_data)
|
||||
, m_size(_size)
|
||||
{
|
||||
}
|
||||
|
||||
inline StaticMemoryBlock::~StaticMemoryBlock()
|
||||
{
|
||||
}
|
||||
|
||||
inline void* StaticMemoryBlock::more(uint32_t _size) BX_OVERRIDE
|
||||
{
|
||||
BX_UNUSED(_size);
|
||||
return m_data;
|
||||
}
|
||||
|
||||
inline uint32_t StaticMemoryBlock::getSize() BX_OVERRIDE
|
||||
{
|
||||
return m_size;
|
||||
}
|
||||
|
||||
inline MemoryBlock::MemoryBlock(AllocatorI* _allocator)
|
||||
: m_allocator(_allocator)
|
||||
, m_data(NULL)
|
||||
, m_size(0)
|
||||
{
|
||||
}
|
||||
|
||||
inline MemoryBlock::~MemoryBlock()
|
||||
{
|
||||
BX_FREE(m_allocator, m_data);
|
||||
}
|
||||
|
||||
inline void* MemoryBlock::more(uint32_t _size) BX_OVERRIDE
|
||||
{
|
||||
if (0 < _size)
|
||||
{
|
||||
m_size += _size;
|
||||
m_data = BX_REALLOC(m_allocator, m_data, m_size);
|
||||
}
|
||||
|
||||
return m_data;
|
||||
}
|
||||
|
||||
inline uint32_t MemoryBlock::getSize() BX_OVERRIDE
|
||||
{
|
||||
return m_size;
|
||||
}
|
||||
|
||||
inline SizerWriter::SizerWriter()
|
||||
: m_pos(0)
|
||||
, m_top(0)
|
||||
{
|
||||
}
|
||||
|
||||
inline SizerWriter::~SizerWriter()
|
||||
{
|
||||
}
|
||||
|
||||
inline int64_t SizerWriter::seek(int64_t _offset, Whence::Enum _whence) BX_OVERRIDE
|
||||
{
|
||||
switch (_whence)
|
||||
{
|
||||
case Whence::Begin:
|
||||
m_pos = int64_clamp(_offset, 0, m_top);
|
||||
break;
|
||||
|
||||
case Whence::Current:
|
||||
m_pos = int64_clamp(m_pos + _offset, 0, m_top);
|
||||
break;
|
||||
|
||||
case Whence::End:
|
||||
m_pos = int64_clamp(m_top - _offset, 0, m_top);
|
||||
break;
|
||||
}
|
||||
|
||||
return m_pos;
|
||||
}
|
||||
|
||||
inline int32_t SizerWriter::write(const void* /*_data*/, int32_t _size, Error* _err) BX_OVERRIDE
|
||||
{
|
||||
BX_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
||||
|
||||
int32_t morecore = int32_t(m_pos - m_top) + _size;
|
||||
|
||||
if (0 < morecore)
|
||||
{
|
||||
m_top += morecore;
|
||||
}
|
||||
|
||||
int64_t remainder = m_top-m_pos;
|
||||
int32_t size = uint32_min(_size, uint32_t(int64_min(remainder, INT32_MAX) ) );
|
||||
m_pos += size;
|
||||
if (size != _size)
|
||||
{
|
||||
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_WRITE, "SizerWriter: write truncated.");
|
||||
}
|
||||
return size;
|
||||
}
|
||||
|
||||
inline MemoryReader::MemoryReader(const void* _data, uint32_t _size)
|
||||
: m_data( (const uint8_t*)_data)
|
||||
, m_pos(0)
|
||||
, m_top(_size)
|
||||
{
|
||||
}
|
||||
|
||||
inline MemoryReader::~MemoryReader()
|
||||
{
|
||||
}
|
||||
|
||||
inline int64_t MemoryReader::seek(int64_t _offset, Whence::Enum _whence) BX_OVERRIDE
|
||||
{
|
||||
switch (_whence)
|
||||
{
|
||||
case Whence::Begin:
|
||||
m_pos = int64_clamp(_offset, 0, m_top);
|
||||
break;
|
||||
|
||||
case Whence::Current:
|
||||
m_pos = int64_clamp(m_pos + _offset, 0, m_top);
|
||||
break;
|
||||
|
||||
case Whence::End:
|
||||
m_pos = int64_clamp(m_top - _offset, 0, m_top);
|
||||
break;
|
||||
}
|
||||
|
||||
return m_pos;
|
||||
}
|
||||
|
||||
inline int32_t MemoryReader::read(void* _data, int32_t _size, Error* _err) BX_OVERRIDE
|
||||
{
|
||||
BX_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
||||
|
||||
int64_t remainder = m_top-m_pos;
|
||||
int32_t size = uint32_min(_size, uint32_t(int64_min(remainder, INT32_MAX) ) );
|
||||
memcpy(_data, &m_data[m_pos], size);
|
||||
m_pos += size;
|
||||
if (size != _size)
|
||||
{
|
||||
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_READ, "MemoryReader: read truncated.");
|
||||
}
|
||||
return size;
|
||||
}
|
||||
|
||||
inline const uint8_t* MemoryReader::getDataPtr() const
|
||||
{
|
||||
return &m_data[m_pos];
|
||||
}
|
||||
|
||||
inline int64_t MemoryReader::getPos() const
|
||||
{
|
||||
return m_pos;
|
||||
}
|
||||
|
||||
inline int64_t MemoryReader::remaining() const
|
||||
{
|
||||
return m_top-m_pos;
|
||||
}
|
||||
|
||||
inline MemoryWriter::MemoryWriter(MemoryBlockI* _memBlock)
|
||||
: m_memBlock(_memBlock)
|
||||
, m_data(NULL)
|
||||
, m_pos(0)
|
||||
, m_top(0)
|
||||
, m_size(0)
|
||||
{
|
||||
}
|
||||
|
||||
inline MemoryWriter::~MemoryWriter()
|
||||
{
|
||||
}
|
||||
|
||||
inline int64_t MemoryWriter::seek(int64_t _offset, Whence::Enum _whence) BX_OVERRIDE
|
||||
{
|
||||
switch (_whence)
|
||||
{
|
||||
case Whence::Begin:
|
||||
m_pos = int64_clamp(_offset, 0, m_top);
|
||||
break;
|
||||
|
||||
case Whence::Current:
|
||||
m_pos = int64_clamp(m_pos + _offset, 0, m_top);
|
||||
break;
|
||||
|
||||
case Whence::End:
|
||||
m_pos = int64_clamp(m_top - _offset, 0, m_top);
|
||||
break;
|
||||
}
|
||||
|
||||
return m_pos;
|
||||
}
|
||||
|
||||
inline int32_t MemoryWriter::write(const void* _data, int32_t _size, Error* _err) BX_OVERRIDE
|
||||
{
|
||||
BX_CHECK(NULL != _err, "Reader/Writer interface calling functions must handle errors.");
|
||||
|
||||
int32_t morecore = int32_t(m_pos - m_size) + _size;
|
||||
|
||||
if (0 < morecore)
|
||||
{
|
||||
morecore = BX_ALIGN_MASK(morecore, 0xfff);
|
||||
m_data = (uint8_t*)m_memBlock->more(morecore);
|
||||
m_size = m_memBlock->getSize();
|
||||
}
|
||||
|
||||
int64_t remainder = m_size-m_pos;
|
||||
int32_t size = uint32_min(_size, uint32_t(int64_min(remainder, INT32_MAX) ) );
|
||||
memcpy(&m_data[m_pos], _data, size);
|
||||
m_pos += size;
|
||||
m_top = int64_max(m_top, m_pos);
|
||||
if (size != _size)
|
||||
{
|
||||
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_WRITE, "MemoryWriter: write truncated.");
|
||||
}
|
||||
return size;
|
||||
}
|
||||
|
||||
inline StaticMemoryBlockWriter::StaticMemoryBlockWriter(void* _data, uint32_t _size)
|
||||
: MemoryWriter(&m_smb)
|
||||
, m_smb(_data, _size)
|
||||
{
|
||||
}
|
||||
|
||||
inline StaticMemoryBlockWriter::~StaticMemoryBlockWriter()
|
||||
{
|
||||
}
|
||||
|
||||
inline int32_t read(ReaderI* _reader, void* _data, int32_t _size, Error* _err)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
return _reader->read(_data, _size, _err);
|
||||
}
|
||||
|
||||
template<typename Ty>
|
||||
int32_t read(ReaderI* _reader, Ty& _value, Error* _err)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
BX_STATIC_ASSERT(BX_TYPE_IS_POD(Ty) );
|
||||
return _reader->read(&_value, sizeof(Ty), _err);
|
||||
}
|
||||
|
||||
template<typename Ty>
|
||||
int32_t readHE(ReaderI* _reader, Ty& _value, bool _fromLittleEndian, Error* _err)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
BX_STATIC_ASSERT(BX_TYPE_IS_POD(Ty) );
|
||||
Ty value;
|
||||
int32_t result = _reader->read(&value, sizeof(Ty), _err);
|
||||
_value = toHostEndian(value, _fromLittleEndian);
|
||||
return result;
|
||||
}
|
||||
|
||||
inline int32_t write(WriterI* _writer, const void* _data, int32_t _size, Error* _err)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
return _writer->write(_data, _size, _err);
|
||||
}
|
||||
|
||||
inline int32_t writeRep(WriterI* _writer, uint8_t _byte, int32_t _size, Error* _err)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
|
||||
const uint32_t tmp0 = uint32_sels(64 - _size, 64, _size);
|
||||
const uint32_t tmp1 = uint32_sels(256 - _size, 256, tmp0);
|
||||
const uint32_t blockSize = uint32_sels(1024 - _size, 1024, tmp1);
|
||||
uint8_t* temp = (uint8_t*)alloca(blockSize);
|
||||
memset(temp, _byte, blockSize);
|
||||
|
||||
int32_t size = 0;
|
||||
while (0 < _size)
|
||||
{
|
||||
int32_t bytes = write(_writer, temp, uint32_min(blockSize, _size), _err);
|
||||
size += bytes;
|
||||
_size -= bytes;
|
||||
}
|
||||
|
||||
return size;
|
||||
}
|
||||
|
||||
template<typename Ty>
|
||||
int32_t write(WriterI* _writer, const Ty& _value, Error* _err)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
BX_STATIC_ASSERT(BX_TYPE_IS_POD(Ty) );
|
||||
return _writer->write(&_value, sizeof(Ty), _err);
|
||||
}
|
||||
|
||||
template<typename Ty>
|
||||
int32_t writeLE(WriterI* _writer, const Ty& _value, Error* _err)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
BX_STATIC_ASSERT(BX_TYPE_IS_POD(Ty) );
|
||||
Ty value = toLittleEndian(_value);
|
||||
int32_t result = _writer->write(&value, sizeof(Ty), _err);
|
||||
return result;
|
||||
}
|
||||
|
||||
template<typename Ty>
|
||||
int32_t writeBE(WriterI* _writer, const Ty& _value, Error* _err)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
BX_STATIC_ASSERT(BX_TYPE_IS_POD(Ty) );
|
||||
Ty value = toBigEndian(_value);
|
||||
int32_t result = _writer->write(&value, sizeof(Ty), _err);
|
||||
return result;
|
||||
}
|
||||
|
||||
inline int32_t writePrintf(WriterI* _writer, const char* _format, ...)
|
||||
{
|
||||
va_list argList;
|
||||
va_start(argList, _format);
|
||||
|
||||
char temp[2048];
|
||||
char* out = temp;
|
||||
int32_t max = sizeof(temp);
|
||||
int32_t len = vsnprintf(out, max, _format, argList);
|
||||
if (len > max)
|
||||
{
|
||||
out = (char*)alloca(len);
|
||||
len = vsnprintf(out, len, _format, argList);
|
||||
}
|
||||
|
||||
int32_t size = write(_writer, out, len);
|
||||
|
||||
va_end(argList);
|
||||
|
||||
return size;
|
||||
}
|
||||
|
||||
inline int64_t skip(SeekerI* _seeker, int64_t _offset)
|
||||
{
|
||||
return _seeker->seek(_offset, Whence::Current);
|
||||
}
|
||||
|
||||
inline int64_t seek(SeekerI* _seeker, int64_t _offset, Whence::Enum _whence)
|
||||
{
|
||||
return _seeker->seek(_offset, _whence);
|
||||
}
|
||||
|
||||
inline int64_t getSize(SeekerI* _seeker)
|
||||
{
|
||||
int64_t offset = _seeker->seek();
|
||||
int64_t size = _seeker->seek(0, Whence::End);
|
||||
_seeker->seek(offset, Whence::Begin);
|
||||
return size;
|
||||
}
|
||||
|
||||
inline int32_t peek(ReaderSeekerI* _reader, void* _data, int32_t _size, Error* _err)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
int64_t offset = bx::seek(_reader);
|
||||
int32_t size = _reader->read(_data, _size, _err);
|
||||
bx::seek(_reader, offset, bx::Whence::Begin);
|
||||
return size;
|
||||
}
|
||||
|
||||
template<typename Ty>
|
||||
int32_t peek(ReaderSeekerI* _reader, Ty& _value, Error* _err)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
BX_STATIC_ASSERT(BX_TYPE_IS_POD(Ty) );
|
||||
return peek(_reader, &_value, sizeof(Ty), _err);
|
||||
}
|
||||
|
||||
inline int32_t align(ReaderSeekerI* _reader, uint32_t _alignment, Error* _err)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
const int64_t current = bx::seek(_reader);
|
||||
const int64_t aligned = ( (current + _alignment-1)/_alignment) * _alignment;
|
||||
const int32_t size = int32_t(aligned - current);
|
||||
if (0 != size)
|
||||
{
|
||||
const int64_t offset = bx::seek(_reader, size);
|
||||
if (offset != aligned)
|
||||
{
|
||||
BX_ERROR_SET(_err, BX_ERROR_READERWRITER_WRITE, "Align: read truncated.");
|
||||
}
|
||||
return int32_t(offset - current);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
inline int32_t align(WriterSeekerI* _writer, uint32_t _alignment, Error* _err)
|
||||
{
|
||||
BX_ERROR_SCOPE(_err);
|
||||
const int64_t current = bx::seek(_writer);
|
||||
const int64_t aligned = ( (current + _alignment-1)/_alignment) * _alignment;
|
||||
const int32_t size = int32_t(aligned - current);
|
||||
if (0 != size)
|
||||
{
|
||||
return writeRep(_writer, 0, size, _err);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
inline bool open(ReaderOpenI* _reader, const char* _filePath, Error* _err)
|
||||
{
|
||||
BX_ERROR_USE_TEMP_WHEN_NULL(_err);
|
||||
return _reader->open(_filePath, _err);
|
||||
}
|
||||
|
||||
inline bool open(WriterOpenI* _writer, const char* _filePath, bool _append, Error* _err)
|
||||
{
|
||||
BX_ERROR_USE_TEMP_WHEN_NULL(_err);
|
||||
return _writer->open(_filePath, _append, _err);
|
||||
}
|
||||
|
||||
inline void close(CloserI* _reader)
|
||||
{
|
||||
_reader->close();
|
||||
}
|
||||
|
||||
} // namespace bx
|
||||
Reference in New Issue
Block a user