123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433 |
- /*
- Copyright (c) 2013 Khaled Mammou - Advanced Micro Devices, Inc.
- 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.
- */
- #pragma once
- #ifndef O3DGC_BINARY_STREAM_H
- #define O3DGC_BINARY_STREAM_H
- #include "o3dgcCommon.h"
- #include "o3dgcVector.h"
- namespace o3dgc
- {
- const unsigned long O3DGC_BINARY_STREAM_DEFAULT_SIZE = 4096;
- const unsigned long O3DGC_BINARY_STREAM_BITS_PER_SYMBOL0 = 7;
- const unsigned long O3DGC_BINARY_STREAM_MAX_SYMBOL0 = (1 << O3DGC_BINARY_STREAM_BITS_PER_SYMBOL0) - 1;
- const unsigned long O3DGC_BINARY_STREAM_BITS_PER_SYMBOL1 = 6;
- const unsigned long O3DGC_BINARY_STREAM_MAX_SYMBOL1 = (1 << O3DGC_BINARY_STREAM_BITS_PER_SYMBOL1) - 1;
- const unsigned long O3DGC_BINARY_STREAM_NUM_SYMBOLS_UINT32 = (32+O3DGC_BINARY_STREAM_BITS_PER_SYMBOL0-1) /
- O3DGC_BINARY_STREAM_BITS_PER_SYMBOL0;
- //!
- class BinaryStream
- {
- public:
- //! Constructor.
- BinaryStream(unsigned long size = O3DGC_BINARY_STREAM_DEFAULT_SIZE)
- {
- m_endianness = SystemEndianness();
- m_stream.Allocate(size);
- };
- //! Destructor.
- ~BinaryStream(void){};
- void WriteFloat32(float value, O3DGCStreamType streamType)
- {
- if (streamType == O3DGC_STREAM_TYPE_ASCII)
- {
- WriteFloat32ASCII(value);
- }
- else
- {
- WriteFloat32Bin(value);
- }
- }
- void WriteUInt32(unsigned long position, unsigned long value, O3DGCStreamType streamType)
- {
- if (streamType == O3DGC_STREAM_TYPE_ASCII)
- {
- WriteUInt32ASCII(position, value);
- }
- else
- {
- WriteUInt32Bin(position, value);
- }
- }
- void WriteUInt32(unsigned long value, O3DGCStreamType streamType)
- {
- if (streamType == O3DGC_STREAM_TYPE_ASCII)
- {
- WriteUInt32ASCII(value);
- }
- else
- {
- WriteUInt32Bin(value);
- }
- }
- void WriteUChar(unsigned int position, unsigned char value, O3DGCStreamType streamType)
- {
- if (streamType == O3DGC_STREAM_TYPE_ASCII)
- {
- WriteUInt32ASCII(position, value);
- }
- else
- {
- WriteUInt32Bin(position, value);
- }
- }
- void WriteUChar(unsigned char value, O3DGCStreamType streamType)
- {
- if (streamType == O3DGC_STREAM_TYPE_ASCII)
- {
- WriteUCharASCII(value);
- }
- else
- {
- WriteUChar8Bin(value);
- }
- }
- float ReadFloat32(unsigned long & position, O3DGCStreamType streamType) const
- {
- float value;
- if (streamType == O3DGC_STREAM_TYPE_ASCII)
- {
- value = ReadFloat32ASCII(position);
- }
- else
- {
- value = ReadFloat32Bin(position);
- }
- return value;
- }
- unsigned long ReadUInt32(unsigned long & position, O3DGCStreamType streamType) const
- {
- unsigned long value;
- if (streamType == O3DGC_STREAM_TYPE_ASCII)
- {
- value = ReadUInt32ASCII(position);
- }
- else
- {
- value = ReadUInt32Bin(position);
- }
- return value;
- }
- unsigned char ReadUChar(unsigned long & position, O3DGCStreamType streamType) const
- {
- unsigned char value;
- if (streamType == O3DGC_STREAM_TYPE_ASCII)
- {
- value = ReadUCharASCII(position);
- }
- else
- {
- value = ReadUChar8Bin(position);
- }
- return value;
- }
- void WriteFloat32Bin(unsigned long position, float value)
- {
- assert(position < m_stream.GetSize() - 4);
- unsigned char * ptr = (unsigned char *) (&value);
- if (m_endianness == O3DGC_BIG_ENDIAN)
- {
- m_stream[position++] = ptr[3];
- m_stream[position++] = ptr[2];
- m_stream[position++] = ptr[1];
- m_stream[position ] = ptr[0];
- }
- else
- {
- m_stream[position++] = ptr[0];
- m_stream[position++] = ptr[1];
- m_stream[position++] = ptr[2];
- m_stream[position ] = ptr[3];
- }
- }
- void WriteFloat32Bin(float value)
- {
- unsigned char * ptr = (unsigned char *) (&value);
- if (m_endianness == O3DGC_BIG_ENDIAN)
- {
- m_stream.PushBack(ptr[3]);
- m_stream.PushBack(ptr[2]);
- m_stream.PushBack(ptr[1]);
- m_stream.PushBack(ptr[0]);
- }
- else
- {
- m_stream.PushBack(ptr[0]);
- m_stream.PushBack(ptr[1]);
- m_stream.PushBack(ptr[2]);
- m_stream.PushBack(ptr[3]);
- }
- }
- void WriteUInt32Bin(unsigned long position, unsigned long value)
- {
- assert(position < m_stream.GetSize() - 4);
- unsigned char * ptr = (unsigned char *) (&value);
- if (m_endianness == O3DGC_BIG_ENDIAN)
- {
- m_stream[position++] = ptr[3];
- m_stream[position++] = ptr[2];
- m_stream[position++] = ptr[1];
- m_stream[position ] = ptr[0];
- }
- else
- {
- m_stream[position++] = ptr[0];
- m_stream[position++] = ptr[1];
- m_stream[position++] = ptr[2];
- m_stream[position ] = ptr[3];
- }
- }
- void WriteUInt32Bin(unsigned long value)
- {
- unsigned char * ptr = (unsigned char *) (&value);
- if (m_endianness == O3DGC_BIG_ENDIAN)
- {
- m_stream.PushBack(ptr[3]);
- m_stream.PushBack(ptr[2]);
- m_stream.PushBack(ptr[1]);
- m_stream.PushBack(ptr[0]);
- }
- else
- {
- m_stream.PushBack(ptr[0]);
- m_stream.PushBack(ptr[1]);
- m_stream.PushBack(ptr[2]);
- m_stream.PushBack(ptr[3]);
- }
- }
- void WriteUChar8Bin(unsigned int position, unsigned char value)
- {
- m_stream[position] = value;
- }
- void WriteUChar8Bin(unsigned char value)
- {
- m_stream.PushBack(value);
- }
- float ReadFloat32Bin(unsigned long & position) const
- {
- unsigned long value = ReadUInt32Bin(position);
- float fvalue;
- memcpy(&fvalue, &value, 4);
- return fvalue;
- }
- unsigned long ReadUInt32Bin(unsigned long & position) const
- {
- assert(position < m_stream.GetSize() - 4);
- unsigned long value = 0;
- if (m_endianness == O3DGC_BIG_ENDIAN)
- {
- value += (m_stream[position++]<<24);
- value += (m_stream[position++]<<16);
- value += (m_stream[position++]<<8);
- value += (m_stream[position++]);
- }
- else
- {
- value += (m_stream[position++]);
- value += (m_stream[position++]<<8);
- value += (m_stream[position++]<<16);
- value += (m_stream[position++]<<24);
- }
- return value;
- }
- unsigned char ReadUChar8Bin(unsigned long & position) const
- {
- return m_stream[position++];
- }
- void WriteFloat32ASCII(float value)
- {
- unsigned long uiValue;
- memcpy(&uiValue, &value, 4);
- WriteUInt32ASCII(uiValue);
- }
- void WriteUInt32ASCII(unsigned long position, unsigned long value)
- {
- assert(position < m_stream.GetSize() - O3DGC_BINARY_STREAM_NUM_SYMBOLS_UINT32);
- unsigned long value0 = value;
- for(unsigned long i = 0; i < O3DGC_BINARY_STREAM_NUM_SYMBOLS_UINT32; ++i)
- {
- m_stream[position++] = (value0 & O3DGC_BINARY_STREAM_MAX_SYMBOL0);
- value0 >>= O3DGC_BINARY_STREAM_BITS_PER_SYMBOL0;
- }
- }
- void WriteUInt32ASCII(unsigned long value)
- {
- for(unsigned long i = 0; i < O3DGC_BINARY_STREAM_NUM_SYMBOLS_UINT32; ++i)
- {
- m_stream.PushBack(value & O3DGC_BINARY_STREAM_MAX_SYMBOL0);
- value >>= O3DGC_BINARY_STREAM_BITS_PER_SYMBOL0;
- }
- }
- void WriteIntASCII(long value)
- {
- WriteUIntASCII(IntToUInt(value));
- }
- void WriteUIntASCII(unsigned long value)
- {
- if (value >= O3DGC_BINARY_STREAM_MAX_SYMBOL0)
- {
- m_stream.PushBack(O3DGC_BINARY_STREAM_MAX_SYMBOL0);
- value -= O3DGC_BINARY_STREAM_MAX_SYMBOL0;
- unsigned char a, b;
- do
- {
- a = ((value & O3DGC_BINARY_STREAM_MAX_SYMBOL1) << 1);
- b = ( (value >>= O3DGC_BINARY_STREAM_BITS_PER_SYMBOL1) > 0);
- a += b;
- m_stream.PushBack(a);
- } while (b);
- }
- else
- {
- m_stream.PushBack((unsigned char) value);
- }
- }
- void WriteUCharASCII(unsigned char value)
- {
- assert(value <= O3DGC_BINARY_STREAM_MAX_SYMBOL0);
- m_stream.PushBack(value);
- }
- float ReadFloat32ASCII(unsigned long & position) const
- {
- unsigned long value = ReadUInt32ASCII(position);
- float fvalue;
- memcpy(&fvalue, &value, 4);
- return fvalue;
- }
- unsigned long ReadUInt32ASCII(unsigned long & position) const
- {
- assert(position < m_stream.GetSize() - O3DGC_BINARY_STREAM_NUM_SYMBOLS_UINT32);
- unsigned long value = 0;
- unsigned long shift = 0;
- for(unsigned long i = 0; i < O3DGC_BINARY_STREAM_NUM_SYMBOLS_UINT32; ++i)
- {
- value += (m_stream[position++] << shift);
- shift += O3DGC_BINARY_STREAM_BITS_PER_SYMBOL0;
- }
- return value;
- }
- long ReadIntASCII(unsigned long & position) const
- {
- return UIntToInt(ReadUIntASCII(position));
- }
- unsigned long ReadUIntASCII(unsigned long & position) const
- {
- unsigned long value = m_stream[position++];
- if (value == O3DGC_BINARY_STREAM_MAX_SYMBOL0)
- {
- long x;
- unsigned long i = 0;
- do
- {
- x = m_stream[position++];
- value += ( (x>>1) << i);
- i += O3DGC_BINARY_STREAM_BITS_PER_SYMBOL1;
- } while (x & 1);
- }
- return value;
- }
- unsigned char ReadUCharASCII(unsigned long & position) const
- {
- return m_stream[position++];
- }
- O3DGCErrorCode Save(const char * const fileName)
- {
- FILE * fout = fopen(fileName, "wb");
- if (!fout)
- {
- return O3DGC_ERROR_CREATE_FILE;
- }
- fwrite(m_stream.GetBuffer(), 1, m_stream.GetSize(), fout);
- fclose(fout);
- return O3DGC_OK;
- }
- O3DGCErrorCode Load(const char * const fileName)
- {
- FILE * fin = fopen(fileName, "rb");
- if (!fin)
- {
- return O3DGC_ERROR_OPEN_FILE;
- }
- fseek(fin, 0, SEEK_END);
- unsigned long size = ftell(fin);
- m_stream.Allocate(size);
- rewind(fin);
- unsigned int nread = (unsigned int) fread((void *) m_stream.GetBuffer(), 1, size, fin);
- m_stream.SetSize(size);
- if (nread != size)
- {
- return O3DGC_ERROR_READ_FILE;
- }
- fclose(fin);
- return O3DGC_OK;
- }
- O3DGCErrorCode LoadFromBuffer(unsigned char * buffer, unsigned long bufferSize)
- {
- m_stream.Allocate(bufferSize);
- memcpy(m_stream.GetBuffer(), buffer, bufferSize);
- m_stream.SetSize(bufferSize);
- return O3DGC_OK;
- }
- unsigned long GetSize() const
- {
- return m_stream.GetSize();
- }
- const unsigned char * GetBuffer(unsigned long position) const
- {
- return m_stream.GetBuffer() + position;
- }
- unsigned char * GetBuffer(unsigned long position)
- {
- return (m_stream.GetBuffer() + position);
- }
- unsigned char * GetBuffer()
- {
- return m_stream.GetBuffer();
- }
- void GetBuffer(unsigned long position, unsigned char * & buffer) const
- {
- buffer = (unsigned char *) (m_stream.GetBuffer() + position); // fix me: ugly!
- }
- void SetSize(unsigned long size)
- {
- m_stream.SetSize(size);
- };
- void Allocate(unsigned long size)
- {
- m_stream.Allocate(size);
- }
- private:
- Vector<unsigned char> m_stream;
- O3DGCEndianness m_endianness;
- };
- }
- #endif // O3DGC_BINARY_STREAM_H
|