mirror of
https://github.com/DarkflameUniverse/DarkflameServer.git
synced 2025-01-15 01:07:05 +00:00
0545adfac3
Have fun!
1697 lines
60 KiB
C++
1697 lines
60 KiB
C++
/// \file
|
|
/// \brief This class allows you to write and read native types as a string of bits. BitStream is used extensively throughout RakNet and is designed to be used by users as well.
|
|
///
|
|
/// This file is part of RakNet Copyright 2003 Kevin Jenkins.
|
|
///
|
|
/// Usage of RakNet is subject to the appropriate license agreement.
|
|
/// Creative Commons Licensees are subject to the
|
|
/// license found at
|
|
/// http://creativecommons.org/licenses/by-nc/2.5/
|
|
/// Single application licensees are subject to the license found at
|
|
/// http://www.jenkinssoftware.com/SingleApplicationLicense.html
|
|
/// Custom license users are subject to the terms therein.
|
|
/// GPL license users are subject to the GNU General Public
|
|
/// License as published by the Free
|
|
/// Software Foundation; either version 2 of the License, or (at your
|
|
/// option) any later version.
|
|
|
|
|
|
#if defined(_MSC_VER) && _MSC_VER < 1299 // VC6 doesn't support template specialization
|
|
#include "BitStream_NoTemplate.h"
|
|
#else
|
|
|
|
#ifndef __BITSTREAM_H
|
|
#define __BITSTREAM_H
|
|
|
|
#include "RakMemoryOverride.h"
|
|
#include "RakNetDefines.h"
|
|
#include "Export.h"
|
|
#include "RakNetTypes.h"
|
|
#include "RakString.h"
|
|
#include <assert.h>
|
|
#include <math.h>
|
|
#include <float.h>
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma warning( push )
|
|
#endif
|
|
|
|
/// The namespace RakNet is not consistently used. It's only purpose is to avoid compiler errors for classes whose names are very common.
|
|
/// For the most part I've tried to avoid this simply by using names very likely to be unique for my classes.
|
|
namespace RakNet
|
|
{
|
|
/// This class allows you to write and read native types as a string of bits. BitStream is used extensively throughout RakNet and is designed to be used by users as well.
|
|
/// \sa BitStreamSample.txt
|
|
class RAK_DLL_EXPORT BitStream : public RakNet::RakMemoryOverride
|
|
{
|
|
|
|
public:
|
|
/// Default Constructor
|
|
BitStream();
|
|
|
|
/// Create the bitstream, with some number of bytes to immediately allocate.
|
|
/// There is no benefit to calling this, unless you know exactly how many bytes you need and it is greater than BITSTREAM_STACK_ALLOCATION_SIZE.
|
|
/// In that case all it does is save you one or more realloc calls.
|
|
/// \param[in] initialBytesToAllocate the number of bytes to pre-allocate.
|
|
BitStream( const unsigned int initialBytesToAllocate );
|
|
|
|
/// Initialize the BitStream, immediately setting the data it contains to a predefined pointer.
|
|
/// Set \a _copyData to true if you want to make an internal copy of the data you are passing. Set it to false to just save a pointer to the data.
|
|
/// You shouldn't call Write functions with \a _copyData as false, as this will write to unallocated memory
|
|
/// 99% of the time you will use this function to cast Packet::data to a bitstream for reading, in which case you should write something as follows:
|
|
/// \code
|
|
/// RakNet::BitStream bs(packet->data, packet->length, false);
|
|
/// \endcode
|
|
/// \param[in] _data An array of bytes.
|
|
/// \param[in] lengthInBytes Size of the \a _data.
|
|
/// \param[in] _copyData true or false to make a copy of \a _data or not.
|
|
BitStream( unsigned char* _data, const unsigned int lengthInBytes, bool _copyData );
|
|
|
|
/// Destructor
|
|
~BitStream();
|
|
|
|
/// Resets the bitstream for reuse.
|
|
void Reset( void );
|
|
|
|
/// Bidirectional serialize/deserialize any integral type to/from a bitstream. Undefine __BITSTREAM_NATIVE_END if you need endian swapping.
|
|
/// \param[in] writeToBitstream true to write from your data to this bitstream. False to read from this bitstream and write to your data
|
|
/// \param[in] var The value to write
|
|
/// \return true if \a writeToBitstream is true. true if \a writeToBitstream is false and the read was successful. false if \a writeToBitstream is false and the read was not successful.
|
|
template <class templateType>
|
|
bool Serialize(bool writeToBitstream, templateType &var);
|
|
|
|
/// Bidirectional serialize/deserialize any integral type to/from a bitstream. If the current value is different from the last value
|
|
/// the current value will be written. Otherwise, a single bit will be written
|
|
/// \param[in] writeToBitstream true to write from your data to this bitstream. False to read from this bitstream and write to your data
|
|
/// \param[in] currentValue The current value to write
|
|
/// \param[in] lastValue The last value to compare against. Only used if \a writeToBitstream is true.
|
|
/// \return true if \a writeToBitstream is true. true if \a writeToBitstream is false and the read was successful. false if \a writeToBitstream is false and the read was not successful.
|
|
template <class templateType>
|
|
bool SerializeDelta(bool writeToBitstream, templateType ¤tValue, templateType lastValue);
|
|
|
|
/// Bidirectional version of SerializeDelta when you don't know what the last value is, or there is no last value.
|
|
/// \param[in] writeToBitstream true to write from your data to this bitstream. False to read from this bitstream and write to your data
|
|
/// \param[in] currentValue The current value to write
|
|
/// \return true if \a writeToBitstream is true. true if \a writeToBitstream is false and the read was successful. false if \a writeToBitstream is false and the read was not successful.
|
|
template <class templateType>
|
|
bool SerializeDelta(bool writeToBitstream, templateType ¤tValue);
|
|
|
|
/// Bidirectional serialize/deserialize any integral type to/from a bitstream. Undefine __BITSTREAM_NATIVE_END if you need endian swapping.
|
|
/// If you are not using __BITSTREAM_NATIVE_END the opposite is true for types larger than 1 byte
|
|
/// For floating point, this is lossy, using 2 bytes for a float and 4 for a double. The range must be between -1 and +1.
|
|
/// For non-floating point, this is lossless, but only has benefit if you use less than half the range of the type
|
|
/// \param[in] writeToBitstream true to write from your data to this bitstream. False to read from this bitstream and write to your data
|
|
/// \param[in] var The value to write
|
|
/// \return true if \a writeToBitstream is true. true if \a writeToBitstream is false and the read was successful. false if \a writeToBitstream is false and the read was not successful.
|
|
template <class templateType>
|
|
bool SerializeCompressed(bool writeToBitstream, templateType &var);
|
|
|
|
/// Bidirectional serialize/deserialize any integral type to/from a bitstream. If the current value is different from the last value
|
|
/// the current value will be written. Otherwise, a single bit will be written
|
|
/// For floating point, this is lossy, using 2 bytes for a float and 4 for a double. The range must be between -1 and +1.
|
|
/// For non-floating point, this is lossless, but only has benefit if you use less than half the range of the type
|
|
/// If you are not using __BITSTREAM_NATIVE_END the opposite is true for types larger than 1 byte
|
|
/// \param[in] writeToBitstream true to write from your data to this bitstream. False to read from this bitstream and write to your data
|
|
/// \param[in] currentValue The current value to write
|
|
/// \param[in] lastValue The last value to compare against. Only used if \a writeToBitstream is true.
|
|
/// \return true if \a writeToBitstream is true. true if \a writeToBitstream is false and the read was successful. false if \a writeToBitstream is false and the read was not successful.
|
|
template <class templateType>
|
|
bool SerializeCompressedDelta(bool writeToBitstream, templateType ¤tValue, templateType lastValue);
|
|
|
|
/// Save as SerializeCompressedDelta(templateType ¤tValue, templateType lastValue) when we have an unknown second parameter
|
|
template <class templateType>
|
|
bool SerializeCompressedDelta(bool writeToBitstream, templateType ¤tValue);
|
|
|
|
/// Bidirectional serialize/deserialize an array or casted stream or raw data. This does NOT do endian swapping.
|
|
/// \param[in] writeToBitstream true to write from your data to this bitstream. False to read from this bitstream and write to your data
|
|
/// \param[in] input a byte buffer
|
|
/// \param[in] numberOfBytes the size of \a input in bytes
|
|
/// \return true if \a writeToBitstream is true. true if \a writeToBitstream is false and the read was successful. false if \a writeToBitstream is false and the read was not successful.
|
|
bool Serialize(bool writeToBitstream, char* input, const unsigned int numberOfBytes );
|
|
|
|
/// Bidirectional serialize/deserialize a normalized 3D vector, using (at most) 4 bytes + 3 bits instead of 12-24 bytes. Will further compress y or z axis aligned vectors.
|
|
/// Accurate to 1/32767.5.
|
|
/// \param[in] writeToBitstream true to write from your data to this bitstream. False to read from this bitstream and write to your data
|
|
/// \param[in] x x
|
|
/// \param[in] y y
|
|
/// \param[in] z z
|
|
/// \return true if \a writeToBitstream is true. true if \a writeToBitstream is false and the read was successful. false if \a writeToBitstream is false and the read was not successful.
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
bool SerializeNormVector(bool writeToBitstream, templateType &x, templateType &y, templateType &z );
|
|
|
|
/// Bidirectional serialize/deserialize a vector, using 10 bytes instead of 12.
|
|
/// Loses accuracy to about 3/10ths and only saves 2 bytes, so only use if accuracy is not important.
|
|
/// \param[in] writeToBitstream true to write from your data to this bitstream. False to read from this bitstream and write to your data
|
|
/// \param[in] x x
|
|
/// \param[in] y y
|
|
/// \param[in] z z
|
|
/// \return true if \a writeToBitstream is true. true if \a writeToBitstream is false and the read was successful. false if \a writeToBitstream is false and the read was not successful.
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
bool SerializeVector(bool writeToBitstream, templateType &x, templateType &y, templateType &z );
|
|
|
|
/// Bidirectional serialize/deserialize a normalized quaternion in 6 bytes + 4 bits instead of 16 bytes. Slightly lossy.
|
|
/// \param[in] writeToBitstream true to write from your data to this bitstream. False to read from this bitstream and write to your data
|
|
/// \param[in] w w
|
|
/// \param[in] x x
|
|
/// \param[in] y y
|
|
/// \param[in] z z
|
|
/// \return true if \a writeToBitstream is true. true if \a writeToBitstream is false and the read was successful. false if \a writeToBitstream is false and the read was not successful.
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
bool SerializeNormQuat(bool writeToBitstream, templateType &w, templateType &x, templateType &y, templateType &z);
|
|
|
|
/// Bidirectional serialize/deserialize an orthogonal matrix by creating a quaternion, and writing 3 components of the quaternion in 2 bytes each
|
|
/// for 6 bytes instead of 36
|
|
/// Lossy, although the result is renormalized
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
bool SerializeOrthMatrix(
|
|
bool writeToBitstream,
|
|
templateType &m00, templateType &m01, templateType &m02,
|
|
templateType &m10, templateType &m11, templateType &m12,
|
|
templateType &m20, templateType &m21, templateType &m22 );
|
|
|
|
/// Bidirectional serialize/deserialize numberToSerialize bits to/from the input. Right aligned
|
|
/// data means in the case of a partial byte, the bits are aligned
|
|
/// from the right (bit 0) rather than the left (as in the normal
|
|
/// internal representation) You would set this to true when
|
|
/// writing user data, and false when copying bitstream data, such
|
|
/// as writing one bitstream to another
|
|
/// \param[in] writeToBitstream true to write from your data to this bitstream. False to read from this bitstream and write to your data
|
|
/// \param[in] input The data
|
|
/// \param[in] numberOfBitsToSerialize The number of bits to write
|
|
/// \param[in] rightAlignedBits if true data will be right aligned
|
|
/// \return true if \a writeToBitstream is true. true if \a writeToBitstream is false and the read was successful. false if \a writeToBitstream is false and the read was not successful.
|
|
bool SerializeBits(bool writeToBitstream, unsigned char* input, const BitSize_t numberOfBitsToSerialize, const bool rightAlignedBits = true );
|
|
|
|
/// Write any integral type to a bitstream. Undefine __BITSTREAM_NATIVE_END if you need endian swapping.
|
|
/// \param[in] var The value to write
|
|
template <class templateType>
|
|
void Write(templateType var);
|
|
|
|
/// Write the dereferenced pointer to any integral type to a bitstream. Undefine __BITSTREAM_NATIVE_END if you need endian swapping.
|
|
/// \param[in] var The value to write
|
|
template <class templateType>
|
|
void WritePtr(templateType *var);
|
|
|
|
/// Write any integral type to a bitstream. If the current value is different from the last value
|
|
/// the current value will be written. Otherwise, a single bit will be written
|
|
/// \param[in] currentValue The current value to write
|
|
/// \param[in] lastValue The last value to compare against
|
|
template <class templateType>
|
|
void WriteDelta(templateType currentValue, templateType lastValue);
|
|
|
|
/// WriteDelta when you don't know what the last value is, or there is no last value.
|
|
/// \param[in] currentValue The current value to write
|
|
template <class templateType>
|
|
void WriteDelta(templateType currentValue);
|
|
|
|
/// Write any integral type to a bitstream. Undefine __BITSTREAM_NATIVE_END if you need endian swapping.
|
|
/// If you are not using __BITSTREAM_NATIVE_END the opposite is true for types larger than 1 byte
|
|
/// For floating point, this is lossy, using 2 bytes for a float and 4 for a double. The range must be between -1 and +1.
|
|
/// For non-floating point, this is lossless, but only has benefit if you use less than half the range of the type
|
|
/// \param[in] var The value to write
|
|
template <class templateType>
|
|
void WriteCompressed(templateType var);
|
|
|
|
/// Write any integral type to a bitstream. If the current value is different from the last value
|
|
/// the current value will be written. Otherwise, a single bit will be written
|
|
/// For floating point, this is lossy, using 2 bytes for a float and 4 for a double. The range must be between -1 and +1.
|
|
/// For non-floating point, this is lossless, but only has benefit if you use less than half the range of the type
|
|
/// If you are not using __BITSTREAM_NATIVE_END the opposite is true for types larger than 1 byte
|
|
/// \param[in] currentValue The current value to write
|
|
/// \param[in] lastValue The last value to compare against
|
|
template <class templateType>
|
|
void WriteCompressedDelta(templateType currentValue, templateType lastValue);
|
|
|
|
/// Save as WriteCompressedDelta(templateType currentValue, templateType lastValue) when we have an unknown second parameter
|
|
template <class templateType>
|
|
void WriteCompressedDelta(templateType currentValue);
|
|
|
|
/// Read any integral type from a bitstream. Define __BITSTREAM_NATIVE_END if you need endian swapping.
|
|
/// \param[in] var The value to read
|
|
template <class templateType>
|
|
bool Read(templateType &var);
|
|
|
|
/// Read into a pointer to any integral type from a bitstream. Define __BITSTREAM_NATIVE_END if you need endian swapping.
|
|
/// \param[in] var The value to read
|
|
template <class templateType>
|
|
bool ReadPtr(templateType *var);
|
|
|
|
/// Read any integral type from a bitstream. If the written value differed from the value compared against in the write function,
|
|
/// var will be updated. Otherwise it will retain the current value.
|
|
/// ReadDelta is only valid from a previous call to WriteDelta
|
|
/// \param[in] var The value to read
|
|
template <class templateType>
|
|
bool ReadDelta(templateType &var);
|
|
|
|
/// Read any integral type from a bitstream. Undefine __BITSTREAM_NATIVE_END if you need endian swapping.
|
|
/// For floating point, this is lossy, using 2 bytes for a float and 4 for a double. The range must be between -1 and +1.
|
|
/// For non-floating point, this is lossless, but only has benefit if you use less than half the range of the type
|
|
/// If you are not using __BITSTREAM_NATIVE_END the opposite is true for types larger than 1 byte
|
|
/// \param[in] var The value to read
|
|
template <class templateType>
|
|
bool ReadCompressed(templateType &var);
|
|
|
|
/// Read any integral type from a bitstream. If the written value differed from the value compared against in the write function,
|
|
/// var will be updated. Otherwise it will retain the current value.
|
|
/// the current value will be updated.
|
|
/// For floating point, this is lossy, using 2 bytes for a float and 4 for a double. The range must be between -1 and +1.
|
|
/// For non-floating point, this is lossless, but only has benefit if you use less than half the range of the type
|
|
/// If you are not using __BITSTREAM_NATIVE_END the opposite is true for types larger than 1 byte
|
|
/// ReadCompressedDelta is only valid from a previous call to WriteDelta
|
|
/// \param[in] var The value to read
|
|
template <class templateType>
|
|
bool ReadCompressedDelta(templateType &var);
|
|
|
|
/// Read one bitstream to another
|
|
/// \param[in] numberOfBits bits to read
|
|
/// \param bitStream the bitstream to read into from
|
|
bool Read( BitStream *bitStream, BitSize_t numberOfBits );
|
|
bool Read( BitStream *bitStream );
|
|
bool Read( BitStream &bitStream, BitSize_t numberOfBits );
|
|
bool Read( BitStream &bitStream );
|
|
|
|
/// Write an array or casted stream or raw data. This does NOT do endian swapping.
|
|
/// \param[in] input a byte buffer
|
|
/// \param[in] numberOfBytes the size of \a input in bytes
|
|
void Write( const char* input, const unsigned int numberOfBytes );
|
|
|
|
/// Write one bitstream to another
|
|
/// \param[in] numberOfBits bits to write
|
|
/// \param bitStream the bitstream to copy from
|
|
void Write( BitStream *bitStream, BitSize_t numberOfBits );
|
|
void Write( BitStream *bitStream );
|
|
void Write( BitStream &bitStream, BitSize_t numberOfBits );
|
|
void Write( BitStream &bitStream );
|
|
|
|
/// Read a normalized 3D vector, using (at most) 4 bytes + 3 bits instead of 12-24 bytes. Will further compress y or z axis aligned vectors.
|
|
/// Accurate to 1/32767.5.
|
|
/// \param[in] x x
|
|
/// \param[in] y y
|
|
/// \param[in] z z
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
void WriteNormVector( templateType x, templateType y, templateType z );
|
|
|
|
/// Write a vector, using 10 bytes instead of 12.
|
|
/// Loses accuracy to about 3/10ths and only saves 2 bytes, so only use if accuracy is not important.
|
|
/// \param[in] x x
|
|
/// \param[in] y y
|
|
/// \param[in] z z
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
void WriteVector( templateType x, templateType y, templateType z );
|
|
|
|
/// Write a normalized quaternion in 6 bytes + 4 bits instead of 16 bytes. Slightly lossy.
|
|
/// \param[in] w w
|
|
/// \param[in] x x
|
|
/// \param[in] y y
|
|
/// \param[in] z z
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
void WriteNormQuat( templateType w, templateType x, templateType y, templateType z);
|
|
|
|
/// Write an orthogonal matrix by creating a quaternion, and writing 3 components of the quaternion in 2 bytes each
|
|
/// for 6 bytes instead of 36
|
|
/// Lossy, although the result is renormalized
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
void WriteOrthMatrix(
|
|
templateType m00, templateType m01, templateType m02,
|
|
templateType m10, templateType m11, templateType m12,
|
|
templateType m20, templateType m21, templateType m22 );
|
|
|
|
/// Read an array or casted stream of byte. The array
|
|
/// is raw data. There is no automatic endian conversion with this function
|
|
/// \param[in] output The result byte array. It should be larger than @em numberOfBytes.
|
|
/// \param[in] numberOfBytes The number of byte to read
|
|
/// \return true on success false if there is some missing bytes.
|
|
bool Read( char* output, const unsigned int numberOfBytes );
|
|
|
|
/// Read a normalized 3D vector, using (at most) 4 bytes + 3 bits instead of 12-24 bytes. Will further compress y or z axis aligned vectors.
|
|
/// Accurate to 1/32767.5.
|
|
/// \param[in] x x
|
|
/// \param[in] y y
|
|
/// \param[in] z z
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
bool ReadNormVector( templateType &x, templateType &y, templateType &z );
|
|
|
|
/// Read 3 floats or doubles, using 10 bytes, where those float or doubles comprise a vector
|
|
/// Loses accuracy to about 3/10ths and only saves 2 bytes, so only use if accuracy is not important.
|
|
/// \param[in] x x
|
|
/// \param[in] y y
|
|
/// \param[in] z z
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
bool ReadVector( templateType &x, templateType &y, templateType &z );
|
|
|
|
/// Read a normalized quaternion in 6 bytes + 4 bits instead of 16 bytes.
|
|
/// \param[in] w w
|
|
/// \param[in] x x
|
|
/// \param[in] y y
|
|
/// \param[in] z z
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
bool ReadNormQuat( templateType &w, templateType &x, templateType &y, templateType &z);
|
|
|
|
/// Read an orthogonal matrix from a quaternion, reading 3 components of the quaternion in 2 bytes each and extrapolatig the 4th.
|
|
/// for 6 bytes instead of 36
|
|
/// Lossy, although the result is renormalized
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
bool ReadOrthMatrix(
|
|
templateType &m00, templateType &m01, templateType &m02,
|
|
templateType &m10, templateType &m11, templateType &m12,
|
|
templateType &m20, templateType &m21, templateType &m22 );
|
|
|
|
///Sets the read pointer back to the beginning of your data.
|
|
void ResetReadPointer( void );
|
|
|
|
/// Sets the write pointer back to the beginning of your data.
|
|
void ResetWritePointer( void );
|
|
|
|
///This is good to call when you are done with the stream to make
|
|
/// sure you didn't leave any data left over void
|
|
void AssertStreamEmpty( void );
|
|
|
|
/// printf the bits in the stream. Great for debugging.
|
|
void PrintBits( void ) const;
|
|
|
|
/// Ignore data we don't intend to read
|
|
/// \param[in] numberOfBits The number of bits to ignore
|
|
void IgnoreBits( const BitSize_t numberOfBits );
|
|
|
|
/// Ignore data we don't intend to read
|
|
/// \param[in] numberOfBits The number of bytes to ignore
|
|
void IgnoreBytes( const unsigned int numberOfBytes );
|
|
|
|
///Move the write pointer to a position on the array.
|
|
/// \param[in] offset the offset from the start of the array.
|
|
/// \attention
|
|
/// Dangerous if you don't know what you are doing!
|
|
/// For efficiency reasons you can only write mid-stream if your data is byte aligned.
|
|
void SetWriteOffset( const BitSize_t offset );
|
|
|
|
/// Returns the length in bits of the stream
|
|
inline BitSize_t GetNumberOfBitsUsed( void ) const {return GetWriteOffset();}
|
|
inline BitSize_t GetWriteOffset( void ) const {return numberOfBitsUsed;}
|
|
|
|
///Returns the length in bytes of the stream
|
|
inline BitSize_t GetNumberOfBytesUsed( void ) const {return BITS_TO_BYTES( numberOfBitsUsed );}
|
|
|
|
///Returns the number of bits into the stream that we have read
|
|
inline BitSize_t GetReadOffset( void ) const {return readOffset;}
|
|
|
|
// Sets the read bit index
|
|
void SetReadOffset( const BitSize_t newReadOffset ) {readOffset=newReadOffset;}
|
|
|
|
///Returns the number of bits left in the stream that haven't been read
|
|
inline BitSize_t GetNumberOfUnreadBits( void ) const {return numberOfBitsUsed - readOffset;}
|
|
|
|
/// Makes a copy of the internal data for you \a _data will point to
|
|
/// the stream. Returns the length in bits of the stream. Partial
|
|
/// bytes are left aligned
|
|
/// \param[out] _data The allocated copy of GetData()
|
|
BitSize_t CopyData( unsigned char** _data ) const;
|
|
|
|
/// Set the stream to some initial data.
|
|
/// \internal
|
|
void SetData( unsigned char *input );
|
|
|
|
/// Gets the data that BitStream is writing to / reading from
|
|
/// Partial bytes are left aligned.
|
|
/// \return A pointer to the internal state
|
|
inline unsigned char* GetData( void ) const {return data;}
|
|
|
|
/// Write numberToWrite bits from the input source Right aligned
|
|
/// data means in the case of a partial byte, the bits are aligned
|
|
/// from the right (bit 0) rather than the left (as in the normal
|
|
/// internal representation) You would set this to true when
|
|
/// writing user data, and false when copying bitstream data, such
|
|
/// as writing one bitstream to another
|
|
/// \param[in] input The data
|
|
/// \param[in] numberOfBitsToWrite The number of bits to write
|
|
/// \param[in] rightAlignedBits if true data will be right aligned
|
|
void WriteBits( const unsigned char* input, BitSize_t numberOfBitsToWrite, const bool rightAlignedBits = true );
|
|
|
|
/// Align the bitstream to the byte boundary and then write the
|
|
/// specified number of bits. This is faster than WriteBits but
|
|
/// wastes the bits to do the alignment and requires you to call
|
|
/// ReadAlignedBits at the corresponding read position.
|
|
/// \param[in] input The data
|
|
/// \param[in] numberOfBytesToWrite The size of input.
|
|
void WriteAlignedBytes( const unsigned char *input, const unsigned int numberOfBytesToWrite );
|
|
|
|
/// Aligns the bitstream, writes inputLength, and writes input. Won't write beyond maxBytesToWrite
|
|
/// \param[in] input The data
|
|
/// \param[in] inputLength The size of input.
|
|
/// \param[in] maxBytesToWrite Max bytes to write
|
|
void WriteAlignedBytesSafe( const char *input, const unsigned int inputLength, const unsigned int maxBytesToWrite );
|
|
|
|
/// Read bits, starting at the next aligned bits. Note that the
|
|
/// modulus 8 starting offset of the sequence must be the same as
|
|
/// was used with WriteBits. This will be a problem with packet
|
|
/// coalescence unless you byte align the coalesced packets.
|
|
/// \param[in] output The byte array larger than @em numberOfBytesToRead
|
|
/// \param[in] numberOfBytesToRead The number of byte to read from the internal state
|
|
/// \return true if there is enough byte.
|
|
bool ReadAlignedBytes( unsigned char *output, const unsigned int numberOfBytesToRead );
|
|
|
|
/// Reads what was written by WriteAlignedBytesSafe
|
|
/// \param[in] input The data
|
|
/// \param[in] maxBytesToRead Maximum number of bytes to read
|
|
bool ReadAlignedBytesSafe( char *input, int &inputLength, const int maxBytesToRead );
|
|
bool ReadAlignedBytesSafe( char *input, unsigned int &inputLength, const unsigned int maxBytesToRead );
|
|
|
|
/// Same as ReadAlignedBytesSafe() but allocates the memory for you using new, rather than assuming it is safe to write to
|
|
/// \param[in] input input will be deleted if it is not a pointer to 0
|
|
bool ReadAlignedBytesSafeAlloc( char **input, int &inputLength, const unsigned int maxBytesToRead );
|
|
bool ReadAlignedBytesSafeAlloc( char **input, unsigned int &inputLength, const unsigned int maxBytesToRead );
|
|
|
|
/// Align the next write and/or read to a byte boundary. This can
|
|
/// be used to 'waste' bits to byte align for efficiency reasons It
|
|
/// can also be used to force coalesced bitstreams to start on byte
|
|
/// boundaries so so WriteAlignedBits and ReadAlignedBits both
|
|
/// calculate the same offset when aligning.
|
|
void AlignWriteToByteBoundary( void );
|
|
|
|
/// Align the next write and/or read to a byte boundary. This can
|
|
/// be used to 'waste' bits to byte align for efficiency reasons It
|
|
/// can also be used to force coalesced bitstreams to start on byte
|
|
/// boundaries so so WriteAlignedBits and ReadAlignedBits both
|
|
/// calculate the same offset when aligning.
|
|
void AlignReadToByteBoundary( void );
|
|
|
|
/// Read \a numberOfBitsToRead bits to the output source
|
|
/// alignBitsToRight should be set to true to convert internal
|
|
/// bitstream data to userdata. It should be false if you used
|
|
/// WriteBits with rightAlignedBits false
|
|
/// \param[in] output The resulting bits array
|
|
/// \param[in] numberOfBitsToRead The number of bits to read
|
|
/// \param[in] alignBitsToRight if true bits will be right aligned.
|
|
/// \return true if there is enough bits to read
|
|
bool ReadBits( unsigned char *output, BitSize_t numberOfBitsToRead, const bool alignBitsToRight = true );
|
|
|
|
/// Write a 0
|
|
void Write0( void );
|
|
|
|
/// Write a 1
|
|
void Write1( void );
|
|
|
|
/// Reads 1 bit and returns true if that bit is 1 and false if it is 0
|
|
bool ReadBit( void );
|
|
|
|
/// If we used the constructor version with copy data off, this
|
|
/// *makes sure it is set to on and the data pointed to is copied.
|
|
void AssertCopyData( void );
|
|
|
|
/// Use this if you pass a pointer copy to the constructor
|
|
/// *(_copyData==false) and want to overallocate to prevent
|
|
/// *reallocation
|
|
void SetNumberOfBitsAllocated( const BitSize_t lengthInBits );
|
|
|
|
/// Reallocates (if necessary) in preparation of writing numberOfBitsToWrite
|
|
void AddBitsAndReallocate( const BitSize_t numberOfBitsToWrite );
|
|
|
|
/// \internal
|
|
/// \return How many bits have been allocated internally
|
|
BitSize_t GetNumberOfBitsAllocated(void) const;
|
|
|
|
|
|
// Read strings, non reference
|
|
bool Read(char *var);
|
|
bool Read(unsigned char *var);
|
|
|
|
|
|
/// ---- Member function template specialization declarations ----
|
|
// Used for VC7
|
|
#if defined(_MSC_VER) && _MSC_VER == 1300
|
|
/// Write a bool to a bitstream
|
|
/// \param[in] var The value to write
|
|
template <>
|
|
void Write(bool var);
|
|
|
|
/// Write a systemAddress to a bitstream
|
|
/// \param[in] var The value to write
|
|
template <>
|
|
void Write(SystemAddress var);
|
|
|
|
/// Write an networkID to a bitstream
|
|
/// \param[in] var The value to write
|
|
template <>
|
|
void Write(NetworkID var);
|
|
|
|
/// Write a string to a bitstream
|
|
/// \param[in] var The value to write
|
|
template <>
|
|
void Write(const char* var);
|
|
template <>
|
|
void Write(const unsigned char* var);
|
|
template <>
|
|
void Write(char* var);
|
|
template <>
|
|
void Write(unsigned char* var);
|
|
template <>
|
|
void Write(RakString var);
|
|
|
|
/// Write a systemAddress. If the current value is different from the last value
|
|
/// the current value will be written. Otherwise, a single bit will be written
|
|
/// \param[in] currentValue The current value to write
|
|
/// \param[in] lastValue The last value to compare against
|
|
template <>
|
|
void WriteDelta(SystemAddress currentValue, SystemAddress lastValue);
|
|
|
|
/// Write an networkID. If the current value is different from the last value
|
|
/// the current value will be written. Otherwise, a single bit will be written
|
|
/// \param[in] currentValue The current value to write
|
|
/// \param[in] lastValue The last value to compare against
|
|
template <>
|
|
void WriteDelta(NetworkID currentValue, NetworkID lastValue);
|
|
|
|
/// Write a bool delta. Same thing as just calling Write
|
|
/// \param[in] currentValue The current value to write
|
|
/// \param[in] lastValue The last value to compare against
|
|
template <>
|
|
void WriteDelta(bool currentValue, bool lastValue);
|
|
|
|
template <>
|
|
void WriteCompressed(SystemAddress var);
|
|
|
|
template <>
|
|
void WriteCompressed(NetworkID var);
|
|
|
|
template <>
|
|
void WriteCompressed(bool var);
|
|
|
|
/// For values between -1 and 1
|
|
template <>
|
|
void WriteCompressed(float var);
|
|
|
|
/// For values between -1 and 1
|
|
template <>
|
|
void WriteCompressed(double var);
|
|
|
|
/// Compressed string
|
|
template <>
|
|
void WriteCompressed(const char* var);
|
|
template <>
|
|
void WriteCompressed(const unsigned char* var);
|
|
template <>
|
|
void WriteCompressed(char* var);
|
|
template <>
|
|
void WriteCompressed(unsigned char* var);
|
|
template <>
|
|
void WriteCompressed(RakString var);
|
|
|
|
/// Write a bool delta. Same thing as just calling Write
|
|
/// \param[in] currentValue The current value to write
|
|
/// \param[in] lastValue The last value to compare against
|
|
template <>
|
|
void WriteCompressedDelta(bool currentValue, bool lastValue);
|
|
|
|
/// Save as WriteCompressedDelta(bool currentValue, templateType lastValue) when we have an unknown second bool
|
|
template <>
|
|
void WriteCompressedDelta(bool currentValue);
|
|
|
|
/// Read a bool from a bitstream
|
|
/// \param[in] var The value to read
|
|
template <>
|
|
bool Read(bool &var);
|
|
|
|
/// Read a systemAddress from a bitstream
|
|
/// \param[in] var The value to read
|
|
template <>
|
|
bool Read(SystemAddress &var);
|
|
|
|
/// Read an NetworkID from a bitstream
|
|
/// \param[in] var The value to read
|
|
template <>
|
|
bool Read(NetworkID &var);
|
|
|
|
/// Read a String from a bitstream
|
|
/// \param[in] var The value to read
|
|
template <>
|
|
bool Read(char *&var);
|
|
template <>
|
|
bool Read(unsigned char *&var);
|
|
template <>
|
|
bool Read(RakString &var);
|
|
|
|
/// Read a bool from a bitstream
|
|
/// \param[in] var The value to read
|
|
template <>
|
|
bool ReadDelta(bool &var);
|
|
|
|
template <>
|
|
bool ReadCompressed(SystemAddress &var);
|
|
|
|
template <>
|
|
bool ReadCompressed(NetworkID &var);
|
|
|
|
template <>
|
|
bool ReadCompressed(bool &var);
|
|
|
|
template <>
|
|
bool ReadCompressed(float &var);
|
|
|
|
/// For values between -1 and 1
|
|
template <>
|
|
bool ReadCompressed(double &var);
|
|
|
|
template <>
|
|
bool ReadCompressed(char* &var);
|
|
template <>
|
|
bool ReadCompressed(unsigned char *&var);
|
|
template <>
|
|
bool ReadCompressed(RakString &var);
|
|
|
|
/// Read a bool from a bitstream
|
|
/// \param[in] var The value to read
|
|
template <>
|
|
bool ReadCompressedDelta(bool &var);
|
|
#endif
|
|
|
|
static bool DoEndianSwap(void);
|
|
static bool IsBigEndian(void);
|
|
static bool IsNetworkOrder(void);
|
|
static void ReverseBytes(unsigned char *input, unsigned char *output, const unsigned int length);
|
|
static void ReverseBytesInPlace(unsigned char *data,const unsigned int length);
|
|
|
|
private:
|
|
|
|
BitStream( const BitStream &invalid) {
|
|
(void) invalid;
|
|
assert(0);
|
|
}
|
|
|
|
/// Assume the input source points to a native type, compress and write it.
|
|
void WriteCompressed( const unsigned char* input, const unsigned int size, const bool unsignedData );
|
|
|
|
/// Assume the input source points to a compressed native type. Decompress and read it.
|
|
bool ReadCompressed( unsigned char* output, const unsigned int size, const bool unsignedData );
|
|
|
|
|
|
BitSize_t numberOfBitsUsed;
|
|
|
|
BitSize_t numberOfBitsAllocated;
|
|
|
|
BitSize_t readOffset;
|
|
|
|
unsigned char *data;
|
|
|
|
/// true if the internal buffer is copy of the data passed to the constructor
|
|
bool copyData;
|
|
|
|
/// BitStreams that use less than BITSTREAM_STACK_ALLOCATION_SIZE use the stack, rather than the heap to store data. It switches over if BITSTREAM_STACK_ALLOCATION_SIZE is exceeded
|
|
unsigned char stackData[BITSTREAM_STACK_ALLOCATION_SIZE];
|
|
};
|
|
|
|
template <class templateType>
|
|
inline bool BitStream::Serialize(bool writeToBitstream, templateType &var)
|
|
{
|
|
if (writeToBitstream)
|
|
Write(var);
|
|
else
|
|
return Read(var);
|
|
return true;
|
|
}
|
|
|
|
template <class templateType>
|
|
inline bool BitStream::SerializeDelta(bool writeToBitstream, templateType ¤tValue, templateType lastValue)
|
|
{
|
|
if (writeToBitstream)
|
|
WriteDelta(currentValue, lastValue);
|
|
else
|
|
return ReadDelta(currentValue);
|
|
return true;
|
|
}
|
|
|
|
template <class templateType>
|
|
inline bool BitStream::SerializeDelta(bool writeToBitstream, templateType ¤tValue)
|
|
{
|
|
if (writeToBitstream)
|
|
WriteDelta(currentValue);
|
|
else
|
|
return ReadDelta(currentValue);
|
|
return true;
|
|
}
|
|
|
|
template <class templateType>
|
|
inline bool BitStream::SerializeCompressed(bool writeToBitstream, templateType &var)
|
|
{
|
|
if (writeToBitstream)
|
|
WriteCompressed(var);
|
|
else
|
|
return ReadCompressed(var);
|
|
return true;
|
|
}
|
|
|
|
template <class templateType>
|
|
inline bool BitStream::SerializeCompressedDelta(bool writeToBitstream, templateType ¤tValue, templateType lastValue)
|
|
{
|
|
if (writeToBitstream)
|
|
WriteCompressedDelta(currentValue,lastValue);
|
|
else
|
|
return ReadCompressedDelta(currentValue);
|
|
return true;
|
|
}
|
|
|
|
template <class templateType>
|
|
inline bool BitStream::SerializeCompressedDelta(bool writeToBitstream, templateType ¤tValue)
|
|
{
|
|
if (writeToBitstream)
|
|
WriteCompressedDelta(currentValue);
|
|
else
|
|
return ReadCompressedDelta(currentValue);
|
|
return true;
|
|
}
|
|
|
|
inline bool BitStream::Serialize(bool writeToBitstream, char* input, const unsigned int numberOfBytes )
|
|
{
|
|
if (writeToBitstream)
|
|
Write(input, numberOfBytes);
|
|
else
|
|
return Read(input, numberOfBytes);
|
|
return true;
|
|
}
|
|
|
|
template <class templateType>
|
|
inline bool BitStream::SerializeNormVector(bool writeToBitstream, templateType &x, templateType &y, templateType &z )
|
|
{
|
|
if (writeToBitstream)
|
|
WriteNormVector(x,y,z);
|
|
else
|
|
return ReadNormVector(x,y,z);
|
|
return true;
|
|
}
|
|
|
|
template <class templateType>
|
|
inline bool BitStream::SerializeVector(bool writeToBitstream, templateType &x, templateType &y, templateType &z )
|
|
{
|
|
if (writeToBitstream)
|
|
WriteVector(x,y,z);
|
|
else
|
|
return ReadVector(x,y,z);
|
|
return true;
|
|
}
|
|
|
|
template <class templateType>
|
|
inline bool BitStream::SerializeNormQuat(bool writeToBitstream, templateType &w, templateType &x, templateType &y, templateType &z)
|
|
{
|
|
if (writeToBitstream)
|
|
WriteNormQuat(w,x,y,z);
|
|
else
|
|
return ReadNormQuat(w,x,y,z);
|
|
return true;
|
|
}
|
|
|
|
template <class templateType>
|
|
inline bool BitStream::SerializeOrthMatrix(
|
|
bool writeToBitstream,
|
|
templateType &m00, templateType &m01, templateType &m02,
|
|
templateType &m10, templateType &m11, templateType &m12,
|
|
templateType &m20, templateType &m21, templateType &m22 )
|
|
{
|
|
if (writeToBitstream)
|
|
WriteOrthMatrix(m00,m01,m02,m10,m11,m12,m20,m21,m22);
|
|
else
|
|
return ReadOrthMatrix(m00,m01,m02,m10,m11,m12,m20,m21,m22);
|
|
return true;
|
|
}
|
|
|
|
inline bool BitStream::SerializeBits(bool writeToBitstream, unsigned char* input, const BitSize_t numberOfBitsToSerialize, const bool rightAlignedBits )
|
|
{
|
|
if (writeToBitstream)
|
|
WriteBits(input,numberOfBitsToSerialize,rightAlignedBits);
|
|
else
|
|
return ReadBits(input,numberOfBitsToSerialize,rightAlignedBits);
|
|
return true;
|
|
}
|
|
|
|
template <class templateType>
|
|
inline void BitStream::Write(templateType var)
|
|
{
|
|
#ifdef _MSC_VER
|
|
#pragma warning(disable:4127) // conditional expression is constant
|
|
#endif
|
|
if (sizeof(var)==1)
|
|
WriteBits( ( unsigned char* ) & var, sizeof( templateType ) * 8, true );
|
|
else
|
|
{
|
|
#ifndef __BITSTREAM_NATIVE_END
|
|
if (DoEndianSwap())
|
|
{
|
|
unsigned char output[sizeof(templateType)];
|
|
ReverseBytes((unsigned char*)&var, output, sizeof(templateType));
|
|
WriteBits( ( unsigned char* ) output, sizeof(templateType) * 8, true );
|
|
}
|
|
else
|
|
#endif
|
|
WriteBits( ( unsigned char* ) & var, sizeof(templateType) * 8, true );
|
|
}
|
|
}
|
|
|
|
template <class templateType>
|
|
inline void BitStream::WritePtr(templateType *var)
|
|
{
|
|
#ifdef _MSC_VER
|
|
#pragma warning(disable:4127) // conditional expression is constant
|
|
#endif
|
|
if (sizeof(var)==1)
|
|
WriteBits( ( unsigned char* ) var, sizeof( templateType ) * 8, true );
|
|
else
|
|
{
|
|
#ifndef __BITSTREAM_NATIVE_END
|
|
if (DoEndianSwap())
|
|
{
|
|
unsigned char output[sizeof(templateType)];
|
|
ReverseBytes((unsigned char*) var, output, sizeof(templateType));
|
|
WriteBits( ( unsigned char* ) output, sizeof(templateType) * 8, true );
|
|
}
|
|
else
|
|
#endif
|
|
WriteBits( ( unsigned char* ) var, sizeof(templateType) * 8, true );
|
|
}
|
|
}
|
|
|
|
/// Write a bool to a bitstream
|
|
/// \param[in] var The value to write
|
|
template <>
|
|
inline void BitStream::Write(bool var)
|
|
{
|
|
if ( var )
|
|
Write1();
|
|
else
|
|
Write0();
|
|
}
|
|
|
|
/// Write a systemAddress to a bitstream
|
|
/// \param[in] var The value to write
|
|
template <>
|
|
inline void BitStream::Write(SystemAddress var)
|
|
{
|
|
// Write(var.binaryAddress);
|
|
WriteBits( ( unsigned char* ) & var.binaryAddress, sizeof(var.binaryAddress) * 8, true );
|
|
Write(var.port);
|
|
}
|
|
|
|
/// Write an networkID to a bitstream
|
|
/// \param[in] var The value to write
|
|
template <>
|
|
inline void BitStream::Write(NetworkID var)
|
|
{
|
|
if (NetworkID::IsPeerToPeerMode()) // Use the function rather than directly access the member or DLL users will get an undefined external error
|
|
Write(var.systemAddress);
|
|
Write(var.localSystemAddress);
|
|
}
|
|
|
|
/// Write a string to a bitstream
|
|
/// \param[in] var The value to write
|
|
template <>
|
|
inline void BitStream::Write(RakString var)
|
|
{
|
|
var.Serialize(this);
|
|
}
|
|
template <>
|
|
inline void BitStream::Write(const char * var)
|
|
{
|
|
RakString::Serialize(var, this);
|
|
}
|
|
template <>
|
|
inline void BitStream::Write(const unsigned char * var)
|
|
{
|
|
Write((const char*)var);
|
|
}
|
|
template <>
|
|
inline void BitStream::Write(char * var)
|
|
{
|
|
Write((const char*)var);
|
|
}
|
|
template <>
|
|
inline void BitStream::Write(unsigned char * var)
|
|
{
|
|
Write((const char*)var);
|
|
}
|
|
|
|
/// Write any integral type to a bitstream. If the current value is different from the last value
|
|
/// the current value will be written. Otherwise, a single bit will be written
|
|
/// \param[in] currentValue The current value to write
|
|
/// \param[in] lastValue The last value to compare against
|
|
template <class templateType>
|
|
inline void BitStream::WriteDelta(templateType currentValue, templateType lastValue)
|
|
{
|
|
if (currentValue==lastValue)
|
|
{
|
|
Write(false);
|
|
}
|
|
else
|
|
{
|
|
Write(true);
|
|
Write(currentValue);
|
|
}
|
|
}
|
|
|
|
/// Write a systemAddress. If the current value is different from the last value
|
|
/// the current value will be written. Otherwise, a single bit will be written
|
|
/// \param[in] currentValue The current value to write
|
|
/// \param[in] lastValue The last value to compare against
|
|
template <>
|
|
inline void BitStream::WriteDelta(SystemAddress currentValue, SystemAddress lastValue)
|
|
{
|
|
if (currentValue==lastValue)
|
|
{
|
|
Write(false);
|
|
}
|
|
else
|
|
{
|
|
Write(true);
|
|
Write(currentValue.binaryAddress);
|
|
Write(currentValue.port);
|
|
}
|
|
}
|
|
|
|
/// Write a systemAddress. If the current value is different from the last value
|
|
/// the current value will be written. Otherwise, a single bit will be written
|
|
/// \param[in] currentValue The current value to write
|
|
/// \param[in] lastValue The last value to compare against
|
|
template <>
|
|
inline void BitStream::WriteDelta(NetworkID currentValue, NetworkID lastValue)
|
|
{
|
|
if (currentValue==lastValue)
|
|
{
|
|
Write(false);
|
|
}
|
|
else
|
|
{
|
|
Write(true);
|
|
if (NetworkID::IsPeerToPeerMode()) // Use the function rather than directly access the member or DLL users will get an undefined external error
|
|
Write(currentValue.systemAddress);
|
|
Write(currentValue.localSystemAddress);
|
|
}
|
|
}
|
|
|
|
/// Write a bool delta. Same thing as just calling Write
|
|
/// \param[in] currentValue The current value to write
|
|
/// \param[in] lastValue The last value to compare against
|
|
template <>
|
|
inline void BitStream::WriteDelta(bool currentValue, bool lastValue)
|
|
{
|
|
(void) lastValue;
|
|
|
|
Write(currentValue);
|
|
}
|
|
|
|
/// WriteDelta when you don't know what the last value is, or there is no last value.
|
|
/// \param[in] currentValue The current value to write
|
|
template <class templateType>
|
|
inline void BitStream::WriteDelta(templateType currentValue)
|
|
{
|
|
Write(true);
|
|
Write(currentValue);
|
|
}
|
|
|
|
/// Write any integral type to a bitstream. Undefine __BITSTREAM_NATIVE_END if you need endian swapping.
|
|
/// For floating point, this is lossy, using 2 bytes for a float and 4 for a double. The range must be between -1 and +1.
|
|
/// For non-floating point, this is lossless, but only has benefit if you use less than half the range of the type
|
|
/// If you are not using __BITSTREAM_NATIVE_END the opposite is true for types larger than 1 byte
|
|
/// \param[in] var The value to write
|
|
template <class templateType>
|
|
inline void BitStream::WriteCompressed(templateType var)
|
|
{
|
|
#ifdef _MSC_VER
|
|
#pragma warning(disable:4127) // conditional expression is constant
|
|
#endif
|
|
if (sizeof(var)==1)
|
|
WriteCompressed( ( unsigned char* ) & var, sizeof( templateType ) * 8, true );
|
|
else
|
|
{
|
|
#ifndef __BITSTREAM_NATIVE_END
|
|
#ifdef _MSC_VER
|
|
#pragma warning(disable:4244) // '=' : conversion from 'unsigned long' to 'unsigned short', possible loss of data
|
|
#endif
|
|
|
|
if (DoEndianSwap())
|
|
{
|
|
unsigned char output[sizeof(templateType)];
|
|
ReverseBytes((unsigned char*)&var, output, sizeof(templateType));
|
|
WriteCompressed( ( unsigned char* ) output, sizeof(templateType) * 8, true );
|
|
}
|
|
else
|
|
#endif
|
|
WriteCompressed( ( unsigned char* ) & var, sizeof(templateType) * 8, true );
|
|
}
|
|
}
|
|
|
|
template <>
|
|
inline void BitStream::WriteCompressed(SystemAddress var)
|
|
{
|
|
Write(var);
|
|
}
|
|
|
|
template <>
|
|
inline void BitStream::WriteCompressed(NetworkID var)
|
|
{
|
|
Write(var);
|
|
}
|
|
|
|
template <>
|
|
inline void BitStream::WriteCompressed(bool var)
|
|
{
|
|
Write(var);
|
|
}
|
|
|
|
/// For values between -1 and 1
|
|
template <>
|
|
inline void BitStream::WriteCompressed(float var)
|
|
{
|
|
assert(var > -1.01f && var < 1.01f);
|
|
if (var < -1.0f)
|
|
var=-1.0f;
|
|
if (var > 1.0f)
|
|
var=1.0f;
|
|
Write((unsigned short)((var+1.0f)*32767.5f));
|
|
}
|
|
|
|
/// For values between -1 and 1
|
|
template <>
|
|
inline void BitStream::WriteCompressed(double var)
|
|
{
|
|
assert(var > -1.01 && var < 1.01);
|
|
if (var < -1.0f)
|
|
var=-1.0f;
|
|
if (var > 1.0f)
|
|
var=1.0f;
|
|
#ifdef _DEBUG
|
|
assert(sizeof(unsigned long)==4);
|
|
#endif
|
|
Write((unsigned long)((var+1.0)*2147483648.0));
|
|
}
|
|
|
|
/// Compress the string
|
|
template <>
|
|
inline void BitStream::WriteCompressed(RakString var)
|
|
{
|
|
var.SerializeCompressed(this,0,false);
|
|
}
|
|
template <>
|
|
inline void BitStream::WriteCompressed(const char * var)
|
|
{
|
|
RakString::SerializeCompressed(var,this,0,false);
|
|
}
|
|
template <>
|
|
inline void BitStream::WriteCompressed(const unsigned char * var)
|
|
{
|
|
WriteCompressed((const char*) var);
|
|
}
|
|
template <>
|
|
inline void BitStream::WriteCompressed(char * var)
|
|
{
|
|
WriteCompressed((const char*) var);
|
|
}
|
|
template <>
|
|
inline void BitStream::WriteCompressed(unsigned char * var)
|
|
{
|
|
WriteCompressed((const char*) var);
|
|
}
|
|
|
|
/// Write any integral type to a bitstream. If the current value is different from the last value
|
|
/// the current value will be written. Otherwise, a single bit will be written
|
|
/// For floating point, this is lossy, using 2 bytes for a float and 4 for a double. The range must be between -1 and +1.
|
|
/// For non-floating point, this is lossless, but only has benefit if you use less than half the range of the type
|
|
/// If you are not using __BITSTREAM_NATIVE_END the opposite is true for types larger than 1 byte
|
|
/// \param[in] currentValue The current value to write
|
|
/// \param[in] lastValue The last value to compare against
|
|
template <class templateType>
|
|
inline void BitStream::WriteCompressedDelta(templateType currentValue, templateType lastValue)
|
|
{
|
|
if (currentValue==lastValue)
|
|
{
|
|
Write(false);
|
|
}
|
|
else
|
|
{
|
|
Write(true);
|
|
WriteCompressed(currentValue);
|
|
}
|
|
}
|
|
|
|
/// Write a bool delta. Same thing as just calling Write
|
|
/// \param[in] currentValue The current value to write
|
|
/// \param[in] lastValue The last value to compare against
|
|
template <>
|
|
inline void BitStream::WriteCompressedDelta(bool currentValue, bool lastValue)
|
|
{
|
|
(void) lastValue;
|
|
|
|
Write(currentValue);
|
|
}
|
|
|
|
/// Save as WriteCompressedDelta(templateType currentValue, templateType lastValue) when we have an unknown second parameter
|
|
template <class templateType>
|
|
inline void BitStream::WriteCompressedDelta(templateType currentValue)
|
|
{
|
|
Write(true);
|
|
WriteCompressed(currentValue);
|
|
}
|
|
|
|
/// Save as WriteCompressedDelta(bool currentValue, templateType lastValue) when we have an unknown second bool
|
|
template <>
|
|
inline void BitStream::WriteCompressedDelta(bool currentValue)
|
|
{
|
|
Write(currentValue);
|
|
}
|
|
|
|
/// Read any integral type from a bitstream. Define __BITSTREAM_NATIVE_END if you need endian swapping.
|
|
/// \param[in] var The value to read
|
|
template <class templateType>
|
|
inline bool BitStream::Read(templateType &var)
|
|
{
|
|
#ifdef _MSC_VER
|
|
#pragma warning(disable:4127) // conditional expression is constant
|
|
#endif
|
|
if (sizeof(var)==1)
|
|
return ReadBits( ( unsigned char* ) &var, sizeof(templateType) * 8, true );
|
|
else
|
|
{
|
|
#ifndef __BITSTREAM_NATIVE_END
|
|
#ifdef _MSC_VER
|
|
#pragma warning(disable:4244) // '=' : conversion from 'unsigned long' to 'unsigned short', possible loss of data
|
|
#endif
|
|
if (DoEndianSwap())
|
|
{
|
|
unsigned char output[sizeof(templateType)];
|
|
if (ReadBits( ( unsigned char* ) output, sizeof(templateType) * 8, true ))
|
|
{
|
|
ReverseBytes(output, (unsigned char*)&var, sizeof(templateType));
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
else
|
|
#endif
|
|
return ReadBits( ( unsigned char* ) & var, sizeof(templateType) * 8, true );
|
|
}
|
|
}
|
|
|
|
template <class templateType>
|
|
inline bool BitStream::ReadPtr(templateType *var)
|
|
{
|
|
#ifdef _MSC_VER
|
|
#pragma warning(disable:4127) // conditional expression is constant
|
|
#endif
|
|
if (sizeof(var)==1)
|
|
return ReadBits( ( unsigned char* ) var, sizeof(templateType) * 8, true );
|
|
else
|
|
{
|
|
#ifndef __BITSTREAM_NATIVE_END
|
|
#ifdef _MSC_VER
|
|
#pragma warning(disable:4244) // '=' : conversion from 'unsigned long' to 'unsigned short', possible loss of data
|
|
#endif
|
|
if (DoEndianSwap())
|
|
{
|
|
unsigned char output[sizeof(templateType)];
|
|
if (ReadBits( ( unsigned char* ) output, sizeof(templateType) * 8, true ))
|
|
{
|
|
ReverseBytes(output, (unsigned char*)var, sizeof(templateType));
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
else
|
|
#endif
|
|
return ReadBits( ( unsigned char* ) var, sizeof(templateType) * 8, true );
|
|
}
|
|
}
|
|
|
|
/// Read a bool from a bitstream
|
|
/// \param[in] var The value to read
|
|
template <>
|
|
inline bool BitStream::Read(bool &var)
|
|
{
|
|
if ( readOffset + 1 > numberOfBitsUsed )
|
|
return false;
|
|
|
|
if ( data[ readOffset >> 3 ] & ( 0x80 >> ( readOffset & 7 ) ) ) // Is it faster to just write it out here?
|
|
var = true;
|
|
else
|
|
var = false;
|
|
|
|
// Has to be on a different line for Mac
|
|
readOffset++;
|
|
|
|
return true;
|
|
}
|
|
|
|
/// Read a systemAddress from a bitstream
|
|
/// \param[in] var The value to read
|
|
template <>
|
|
inline bool BitStream::Read(SystemAddress &var)
|
|
{
|
|
// Read(var.binaryAddress);
|
|
ReadBits( ( unsigned char* ) & var.binaryAddress, sizeof(var.binaryAddress) * 8, true );
|
|
return Read(var.port);
|
|
}
|
|
|
|
/// Read an networkID from a bitstream
|
|
/// \param[in] var The value to read
|
|
template <>
|
|
inline bool BitStream::Read(NetworkID &var)
|
|
{
|
|
if (NetworkID::IsPeerToPeerMode()) // Use the function rather than directly access the member or DLL users will get an undefined external error
|
|
Read(var.systemAddress);
|
|
return Read(var.localSystemAddress);
|
|
}
|
|
|
|
/// Read an networkID from a bitstream
|
|
/// \param[in] var The value to read
|
|
template <>
|
|
inline bool BitStream::Read(RakString &var)
|
|
{
|
|
return var.Deserialize(this);
|
|
}
|
|
template <>
|
|
inline bool BitStream::Read(char *&var)
|
|
{
|
|
return RakString::Deserialize(var,this);
|
|
}
|
|
template <>
|
|
inline bool BitStream::Read(unsigned char *&var)
|
|
{
|
|
return RakString::Deserialize((char*) var,this);
|
|
}
|
|
|
|
// asdf
|
|
|
|
/// Read any integral type from a bitstream. If the written value differed from the value compared against in the write function,
|
|
/// var will be updated. Otherwise it will retain the current value.
|
|
/// ReadDelta is only valid from a previous call to WriteDelta
|
|
/// \param[in] var The value to read
|
|
template <class templateType>
|
|
inline bool BitStream::ReadDelta(templateType &var)
|
|
{
|
|
bool dataWritten;
|
|
bool success;
|
|
success=Read(dataWritten);
|
|
if (dataWritten)
|
|
success=Read(var);
|
|
return success;
|
|
}
|
|
|
|
/// Read a bool from a bitstream
|
|
/// \param[in] var The value to read
|
|
template <>
|
|
inline bool BitStream::ReadDelta(bool &var)
|
|
{
|
|
return Read(var);
|
|
}
|
|
|
|
/// Read any integral type from a bitstream. Undefine __BITSTREAM_NATIVE_END if you need endian swapping.
|
|
/// For floating point, this is lossy, using 2 bytes for a float and 4 for a double. The range must be between -1 and +1.
|
|
/// For non-floating point, this is lossless, but only has benefit if you use less than half the range of the type
|
|
/// If you are not using __BITSTREAM_NATIVE_END the opposite is true for types larger than 1 byte
|
|
/// \param[in] var The value to read
|
|
template <class templateType>
|
|
inline bool BitStream::ReadCompressed(templateType &var)
|
|
{
|
|
#ifdef _MSC_VER
|
|
#pragma warning(disable:4127) // conditional expression is constant
|
|
#endif
|
|
if (sizeof(var)==1)
|
|
return ReadCompressed( ( unsigned char* ) &var, sizeof(templateType) * 8, true );
|
|
else
|
|
{
|
|
#ifndef __BITSTREAM_NATIVE_END
|
|
if (DoEndianSwap())
|
|
{
|
|
unsigned char output[sizeof(templateType)];
|
|
if (ReadCompressed( ( unsigned char* ) output, sizeof(templateType) * 8, true ))
|
|
{
|
|
ReverseBytes(output, (unsigned char*)&var, sizeof(templateType));
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
else
|
|
#endif
|
|
return ReadCompressed( ( unsigned char* ) & var, sizeof(templateType) * 8, true );
|
|
}
|
|
}
|
|
|
|
template <>
|
|
inline bool BitStream::ReadCompressed(SystemAddress &var)
|
|
{
|
|
return Read(var);
|
|
}
|
|
|
|
template <>
|
|
inline bool BitStream::ReadCompressed(NetworkID &var)
|
|
{
|
|
return Read(var);
|
|
}
|
|
|
|
template <>
|
|
inline bool BitStream::ReadCompressed(bool &var)
|
|
{
|
|
return Read(var);
|
|
}
|
|
|
|
/// For values between -1 and 1
|
|
template <>
|
|
inline bool BitStream::ReadCompressed(float &var)
|
|
{
|
|
unsigned short compressedFloat;
|
|
if (Read(compressedFloat))
|
|
{
|
|
var = ((float)compressedFloat / 32767.5f - 1.0f);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/// For values between -1 and 1
|
|
template <>
|
|
inline bool BitStream::ReadCompressed(double &var)
|
|
{
|
|
unsigned long compressedFloat;
|
|
if (Read(compressedFloat))
|
|
{
|
|
var = ((double)compressedFloat / 2147483648.0 - 1.0);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/// For strings
|
|
template <>
|
|
inline bool BitStream::ReadCompressed(RakString &var)
|
|
{
|
|
return var.DeserializeCompressed(this,false);
|
|
}
|
|
template <>
|
|
inline bool BitStream::ReadCompressed(char *&var)
|
|
{
|
|
return RakString::DeserializeCompressed(var,this,false);
|
|
}
|
|
template <>
|
|
inline bool BitStream::ReadCompressed(unsigned char *&var)
|
|
{
|
|
return RakString::DeserializeCompressed((char*) var,this,false);
|
|
}
|
|
|
|
/// Read any integral type from a bitstream. If the written value differed from the value compared against in the write function,
|
|
/// var will be updated. Otherwise it will retain the current value.
|
|
/// the current value will be updated.
|
|
/// For floating point, this is lossy, using 2 bytes for a float and 4 for a double. The range must be between -1 and +1.
|
|
/// For non-floating point, this is lossless, but only has benefit if you use less than half the range of the type
|
|
/// If you are not using __BITSTREAM_NATIVE_END the opposite is true for types larger than 1 byte
|
|
/// ReadCompressedDelta is only valid from a previous call to WriteDelta
|
|
/// \param[in] var The value to read
|
|
template <class templateType>
|
|
inline bool BitStream::ReadCompressedDelta(templateType &var)
|
|
{
|
|
bool dataWritten;
|
|
bool success;
|
|
success=Read(dataWritten);
|
|
if (dataWritten)
|
|
success=ReadCompressed(var);
|
|
return success;
|
|
}
|
|
|
|
/// Read a bool from a bitstream
|
|
/// \param[in] var The value to read
|
|
template <>
|
|
inline bool BitStream::ReadCompressedDelta(bool &var)
|
|
{
|
|
return Read(var);
|
|
}
|
|
|
|
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
void BitStream::WriteNormVector( templateType x, templateType y, templateType z )
|
|
{
|
|
#ifdef _DEBUG
|
|
assert(x <= 1.01 && y <= 1.01 && z <= 1.01 && x >= -1.01 && y >= -1.01 && z >= -1.01);
|
|
#endif
|
|
if (x>1.0)
|
|
x=1.0;
|
|
if (y>1.0)
|
|
y=1.0;
|
|
if (z>1.0)
|
|
z=1.0;
|
|
if (x<-1.0)
|
|
x=-1.0;
|
|
if (y<-1.0)
|
|
y=-1.0;
|
|
if (z<-1.0)
|
|
z=-1.0;
|
|
|
|
Write((bool) (x < 0.0));
|
|
if (y==0.0)
|
|
Write(true);
|
|
else
|
|
{
|
|
Write(false);
|
|
WriteCompressed((float)y);
|
|
//Write((unsigned short)((y+1.0f)*32767.5f));
|
|
}
|
|
if (z==0.0)
|
|
Write(true);
|
|
else
|
|
{
|
|
Write(false);
|
|
WriteCompressed((float)z);
|
|
//Write((unsigned short)((z+1.0f)*32767.5f));
|
|
}
|
|
}
|
|
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
void BitStream::WriteVector( templateType x, templateType y, templateType z )
|
|
{
|
|
templateType magnitude = sqrt(x * x + y * y + z * z);
|
|
Write((float)magnitude);
|
|
if (magnitude > 0.0)
|
|
{
|
|
WriteCompressed((float)(x/magnitude));
|
|
WriteCompressed((float)(y/magnitude));
|
|
WriteCompressed((float)(z/magnitude));
|
|
// Write((unsigned short)((x/magnitude+1.0f)*32767.5f));
|
|
// Write((unsigned short)((y/magnitude+1.0f)*32767.5f));
|
|
// Write((unsigned short)((z/magnitude+1.0f)*32767.5f));
|
|
}
|
|
}
|
|
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
void BitStream::WriteNormQuat( templateType w, templateType x, templateType y, templateType z)
|
|
{
|
|
Write((bool)(w<0.0));
|
|
Write((bool)(x<0.0));
|
|
Write((bool)(y<0.0));
|
|
Write((bool)(z<0.0));
|
|
Write((unsigned short)(fabs(x)*65535.0));
|
|
Write((unsigned short)(fabs(y)*65535.0));
|
|
Write((unsigned short)(fabs(z)*65535.0));
|
|
// Leave out w and calculate it on the target
|
|
}
|
|
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
void BitStream::WriteOrthMatrix(
|
|
templateType m00, templateType m01, templateType m02,
|
|
templateType m10, templateType m11, templateType m12,
|
|
templateType m20, templateType m21, templateType m22 )
|
|
{
|
|
|
|
double qw;
|
|
double qx;
|
|
double qy;
|
|
double qz;
|
|
|
|
// Convert matrix to quat
|
|
// http://www.euclideanspace.com/maths/geometry/rotations/conversions/matrixToQuaternion/
|
|
float sum;
|
|
sum = 1 + m00 + m11 + m22;
|
|
if (sum < 0.0f) sum=0.0f;
|
|
qw = sqrt( sum ) / 2;
|
|
sum = 1 + m00 - m11 - m22;
|
|
if (sum < 0.0f) sum=0.0f;
|
|
qx = sqrt( sum ) / 2;
|
|
sum = 1 - m00 + m11 - m22;
|
|
if (sum < 0.0f) sum=0.0f;
|
|
qy = sqrt( sum ) / 2;
|
|
sum = 1 - m00 - m11 + m22;
|
|
if (sum < 0.0f) sum=0.0f;
|
|
qz = sqrt( sum ) / 2;
|
|
if (qw < 0.0) qw=0.0;
|
|
if (qx < 0.0) qx=0.0;
|
|
if (qy < 0.0) qy=0.0;
|
|
if (qz < 0.0) qz=0.0;
|
|
qx = copysign( qx, m21 - m12 );
|
|
qy = copysign( qy, m02 - m20 );
|
|
qz = copysign( qz, m10 - m01 );
|
|
|
|
WriteNormQuat(qw,qx,qy,qz);
|
|
}
|
|
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
bool BitStream::ReadNormVector( templateType &x, templateType &y, templateType &z )
|
|
{
|
|
// unsigned short sy, sz;
|
|
bool yZero, zZero;
|
|
bool xNeg;
|
|
float cy,cz;
|
|
|
|
Read(xNeg);
|
|
|
|
Read(yZero);
|
|
if (yZero)
|
|
y=0.0;
|
|
else
|
|
{
|
|
ReadCompressed(cy);
|
|
y=cy;
|
|
//Read(sy);
|
|
//y=((float)sy / 32767.5f - 1.0f);
|
|
}
|
|
|
|
if (!Read(zZero))
|
|
return false;
|
|
|
|
if (zZero)
|
|
z=0.0;
|
|
else
|
|
{
|
|
// if (!Read(sz))
|
|
// return false;
|
|
|
|
// z=((float)sz / 32767.5f - 1.0f);
|
|
if (!ReadCompressed(cz))
|
|
return false;
|
|
z=cz;
|
|
}
|
|
|
|
x = (templateType) (sqrtf((templateType)1.0 - y*y - z*z));
|
|
if (xNeg)
|
|
x=-x;
|
|
return true;
|
|
}
|
|
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
bool BitStream::ReadVector( templateType &x, templateType &y, templateType &z )
|
|
{
|
|
float magnitude;
|
|
//unsigned short sx,sy,sz;
|
|
if (!Read(magnitude))
|
|
return false;
|
|
if (magnitude!=0.0)
|
|
{
|
|
// Read(sx);
|
|
// Read(sy);
|
|
// if (!Read(sz))
|
|
// return false;
|
|
// x=((float)sx / 32767.5f - 1.0f) * magnitude;
|
|
// y=((float)sy / 32767.5f - 1.0f) * magnitude;
|
|
// z=((float)sz / 32767.5f - 1.0f) * magnitude;
|
|
float cx,cy,cz;
|
|
ReadCompressed(cx);
|
|
ReadCompressed(cy);
|
|
if (!ReadCompressed(cz))
|
|
return false;
|
|
x=cx;
|
|
y=cy;
|
|
z=cz;
|
|
x*=magnitude;
|
|
y*=magnitude;
|
|
z*=magnitude;
|
|
}
|
|
else
|
|
{
|
|
x=0.0;
|
|
y=0.0;
|
|
z=0.0;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
bool BitStream::ReadNormQuat( templateType &w, templateType &x, templateType &y, templateType &z)
|
|
{
|
|
bool cwNeg, cxNeg, cyNeg, czNeg;
|
|
unsigned short cx,cy,cz;
|
|
Read(cwNeg);
|
|
Read(cxNeg);
|
|
Read(cyNeg);
|
|
Read(czNeg);
|
|
Read(cx);
|
|
Read(cy);
|
|
if (!Read(cz))
|
|
return false;
|
|
|
|
// Calculate w from x,y,z
|
|
x=(templateType)(cx/65535.0);
|
|
y=(templateType)(cy/65535.0);
|
|
z=(templateType)(cz/65535.0);
|
|
if (cxNeg) x=-x;
|
|
if (cyNeg) y=-y;
|
|
if (czNeg) z=-z;
|
|
float difference = 1.0 - x*x - y*y - z*z;
|
|
if (difference < 0.0f)
|
|
difference=0.0f;
|
|
w = (templateType)(sqrt(difference));
|
|
if (cwNeg)
|
|
w=-w;
|
|
|
|
return true;
|
|
}
|
|
|
|
template <class templateType> // templateType for this function must be a float or double
|
|
bool BitStream::ReadOrthMatrix(
|
|
templateType &m00, templateType &m01, templateType &m02,
|
|
templateType &m10, templateType &m11, templateType &m12,
|
|
templateType &m20, templateType &m21, templateType &m22 )
|
|
{
|
|
float qw,qx,qy,qz;
|
|
if (!ReadNormQuat(qw,qx,qy,qz))
|
|
return false;
|
|
|
|
// Quat to orthogonal rotation matrix
|
|
// http://www.euclideanspace.com/maths/geometry/rotations/conversions/quaternionToMatrix/index.htm
|
|
double sqw = (double)qw*(double)qw;
|
|
double sqx = (double)qx*(double)qx;
|
|
double sqy = (double)qy*(double)qy;
|
|
double sqz = (double)qz*(double)qz;
|
|
m00 = (templateType)(sqx - sqy - sqz + sqw); // since sqw + sqx + sqy + sqz =1
|
|
m11 = (templateType)(-sqx + sqy - sqz + sqw);
|
|
m22 = (templateType)(-sqx - sqy + sqz + sqw);
|
|
|
|
double tmp1 = (double)qx*(double)qy;
|
|
double tmp2 = (double)qz*(double)qw;
|
|
m10 = (templateType)(2.0 * (tmp1 + tmp2));
|
|
m01 = (templateType)(2.0 * (tmp1 - tmp2));
|
|
|
|
tmp1 = (double)qx*(double)qz;
|
|
tmp2 = (double)qy*(double)qw;
|
|
m20 =(templateType)(2.0 * (tmp1 - tmp2));
|
|
m02 = (templateType)(2.0 * (tmp1 + tmp2));
|
|
tmp1 = (double)qy*(double)qz;
|
|
tmp2 = (double)qx*(double)qw;
|
|
m21 = (templateType)(2.0 * (tmp1 + tmp2));
|
|
m12 = (templateType)(2.0 * (tmp1 - tmp2));
|
|
|
|
return true;
|
|
}
|
|
|
|
template <class templateType>
|
|
BitStream& operator<<(BitStream& out, templateType& c)
|
|
{
|
|
out.Write(c);
|
|
return out;
|
|
}
|
|
template <class templateType>
|
|
BitStream& operator>>(BitStream& in, templateType& c)
|
|
{
|
|
bool success = in.Read(c);
|
|
assert(success);
|
|
return in;
|
|
}
|
|
|
|
}
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma warning( pop )
|
|
#endif
|
|
|
|
#endif
|
|
|
|
#endif // VC6
|