2013-07-29 07:13:03 -04:00
|
|
|
|
|
|
|
// ByteBuffer.cpp
|
|
|
|
|
|
|
|
// Implements the cByteBuffer class representing a ringbuffer of bytes
|
|
|
|
|
|
|
|
#include "Globals.h"
|
|
|
|
|
|
|
|
#include "ByteBuffer.h"
|
|
|
|
#include "Endianness.h"
|
|
|
|
#include "OSSupport/IsThread.h"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-12-24 03:13:58 -05:00
|
|
|
/** When defined, each access to a cByteBuffer object is checked whether it's done in the same thread.
|
|
|
|
cByteBuffer assumes that it is not used by multiple threads at once, this macro adds a runtime check for that.
|
|
|
|
Unfortunately it is very slow, so it is disabled even for regular DEBUG builds. */
|
|
|
|
// #define DEBUG_SINGLE_THREAD_ACCESS
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-11-24 08:35:35 -05:00
|
|
|
// Try to determine endianness:
|
|
|
|
#if ( \
|
|
|
|
defined(__i386__) || defined(__alpha__) || \
|
|
|
|
defined(__ia64) || defined(__ia64__) || \
|
|
|
|
defined(_M_IX86) || defined(_M_IA64) || \
|
|
|
|
defined(_M_ALPHA) || defined(__amd64) || \
|
|
|
|
defined(__amd64__) || defined(_M_AMD64) || \
|
|
|
|
defined(__x86_64) || defined(__x86_64__) || \
|
|
|
|
defined(_M_X64) || defined(__bfin__) || \
|
2016-07-20 11:34:22 -04:00
|
|
|
defined(__ARMEL__) || defined(__AARCH64EL__) || \
|
2013-11-24 08:35:35 -05:00
|
|
|
(defined(_WIN32) && defined(__ARM__) && defined(_MSC_VER)) \
|
|
|
|
)
|
|
|
|
#define IS_LITTLE_ENDIAN
|
2013-12-04 13:57:23 -05:00
|
|
|
#elif ( \
|
2014-08-23 11:06:34 -04:00
|
|
|
defined (__ARMEB__) || defined(__sparc) || defined(__powerpc__) || defined(__POWERPC__) \
|
2013-12-04 13:57:23 -05:00
|
|
|
)
|
2013-11-24 08:35:35 -05:00
|
|
|
#define IS_BIG_ENDIAN
|
|
|
|
#else
|
|
|
|
#error Cannot determine endianness of this platform
|
|
|
|
#endif
|
|
|
|
|
2013-10-28 15:40:42 -04:00
|
|
|
// If a string sent over the protocol is larger than this, a warning is emitted to the console
|
|
|
|
#define MAX_STRING_SIZE (512 KiB)
|
|
|
|
|
|
|
|
#define NEEDBYTES(Num) if (!CanReadBytes(Num)) return false; // Check if at least Num bytes can be read from the buffer, return false if not
|
|
|
|
#define PUTBYTES(Num) if (!CanWriteBytes(Num)) return false; // Check if at least Num bytes can be written to the buffer, return false if not
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-12-24 03:13:58 -05:00
|
|
|
#ifdef DEBUG_SINGLE_THREAD_ACCESS
|
2014-12-03 12:04:08 -05:00
|
|
|
|
|
|
|
/** Simple RAII class that is used for checking that no two threads are using an object simultanously.
|
|
|
|
It requires the monitored object to provide the storage for a thread ID.
|
|
|
|
It uses that storage to check if the thread ID of consecutive calls is the same all the time. */
|
|
|
|
class cSingleThreadAccessChecker
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
cSingleThreadAccessChecker(std::thread::id * a_ThreadID) :
|
|
|
|
m_ThreadID(a_ThreadID)
|
|
|
|
{
|
|
|
|
ASSERT(
|
|
|
|
(*a_ThreadID == std::this_thread::get_id()) || // Either the object is used by current thread...
|
2015-01-03 16:20:30 -05:00
|
|
|
(*a_ThreadID == m_EmptyThreadID) // ... or by no thread at all
|
2014-12-03 12:04:08 -05:00
|
|
|
);
|
|
|
|
|
|
|
|
// Mark as being used by this thread:
|
|
|
|
*m_ThreadID = std::this_thread::get_id();
|
|
|
|
}
|
|
|
|
|
|
|
|
~cSingleThreadAccessChecker()
|
|
|
|
{
|
|
|
|
// Mark as not being used by any thread:
|
|
|
|
*m_ThreadID = std::thread::id();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
/** Points to the storage used for ID of the thread using the object. */
|
|
|
|
std::thread::id * m_ThreadID;
|
2015-01-03 16:20:30 -05:00
|
|
|
|
|
|
|
/** The value of an unassigned thread ID, used to speed up checking. */
|
|
|
|
static std::thread::id m_EmptyThreadID;
|
2014-12-03 12:04:08 -05:00
|
|
|
};
|
|
|
|
|
2015-01-03 16:20:30 -05:00
|
|
|
std::thread::id cSingleThreadAccessChecker::m_EmptyThreadID;
|
|
|
|
|
2014-12-03 12:24:28 -05:00
|
|
|
#define CHECK_THREAD cSingleThreadAccessChecker Checker(&m_ThreadID);
|
2014-12-03 12:04:08 -05:00
|
|
|
|
|
|
|
#else
|
2014-12-03 12:24:28 -05:00
|
|
|
#define CHECK_THREAD
|
2014-12-03 12:04:08 -05:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-07-17 16:15:34 -04:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
2013-07-29 07:13:03 -04:00
|
|
|
// cByteBuffer:
|
|
|
|
|
2014-04-24 15:29:59 -04:00
|
|
|
cByteBuffer::cByteBuffer(size_t a_BufferSize) :
|
2013-07-29 07:13:03 -04:00
|
|
|
m_Buffer(new char[a_BufferSize + 1]),
|
|
|
|
m_BufferSize(a_BufferSize + 1),
|
|
|
|
m_DataStart(0),
|
|
|
|
m_WritePos(0),
|
|
|
|
m_ReadPos(0)
|
|
|
|
{
|
|
|
|
// Allocating one byte more than the buffer size requested, so that we can distinguish between
|
|
|
|
// completely-full and completely-empty states
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
cByteBuffer::~cByteBuffer()
|
|
|
|
{
|
|
|
|
CheckValid();
|
|
|
|
delete[] m_Buffer;
|
2014-10-20 16:55:07 -04:00
|
|
|
m_Buffer = nullptr;
|
2013-07-29 07:13:03 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-04-24 16:42:33 -04:00
|
|
|
bool cByteBuffer::Write(const void * a_Bytes, size_t a_Count)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
|
|
|
|
// Store the current free space for a check after writing:
|
2014-03-08 11:33:38 -05:00
|
|
|
size_t CurFreeSpace = GetFreeSpace();
|
2016-08-02 07:12:34 -04:00
|
|
|
#ifdef _DEBUG
|
|
|
|
size_t CurReadableSpace = GetReadableSpace();
|
|
|
|
#endif
|
2014-03-08 11:33:38 -05:00
|
|
|
size_t WrittenBytes = 0;
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2014-01-26 11:48:09 -05:00
|
|
|
if (CurFreeSpace < a_Count)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2014-03-08 14:18:51 -05:00
|
|
|
ASSERT(m_BufferSize >= m_WritePos);
|
2014-03-08 11:33:38 -05:00
|
|
|
size_t TillEnd = m_BufferSize - m_WritePos;
|
2015-07-29 11:04:03 -04:00
|
|
|
const char * Bytes = reinterpret_cast<const char *>(a_Bytes);
|
2013-07-29 07:13:03 -04:00
|
|
|
if (TillEnd <= a_Count)
|
|
|
|
{
|
|
|
|
// Need to wrap around the ringbuffer end
|
|
|
|
if (TillEnd > 0)
|
|
|
|
{
|
2014-04-24 16:42:33 -04:00
|
|
|
memcpy(m_Buffer + m_WritePos, Bytes, TillEnd);
|
|
|
|
Bytes += TillEnd;
|
2013-07-29 07:13:03 -04:00
|
|
|
a_Count -= TillEnd;
|
|
|
|
WrittenBytes = TillEnd;
|
|
|
|
}
|
|
|
|
m_WritePos = 0;
|
|
|
|
}
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2013-07-29 07:13:03 -04:00
|
|
|
// We're guaranteed that we'll fit in a single write op
|
|
|
|
if (a_Count > 0)
|
|
|
|
{
|
2014-04-24 16:42:33 -04:00
|
|
|
memcpy(m_Buffer + m_WritePos, Bytes, a_Count);
|
2013-07-29 07:13:03 -04:00
|
|
|
m_WritePos += a_Count;
|
|
|
|
WrittenBytes += a_Count;
|
|
|
|
}
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2013-07-29 07:13:03 -04:00
|
|
|
ASSERT(GetFreeSpace() == CurFreeSpace - WrittenBytes);
|
|
|
|
ASSERT(GetReadableSpace() == CurReadableSpace + WrittenBytes);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-03-08 11:33:38 -05:00
|
|
|
size_t cByteBuffer::GetFreeSpace(void) const
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
if (m_WritePos >= m_DataStart)
|
|
|
|
{
|
|
|
|
// Wrap around the buffer end:
|
2014-03-08 14:18:51 -05:00
|
|
|
ASSERT(m_BufferSize >= m_WritePos);
|
|
|
|
ASSERT((m_BufferSize - m_WritePos + m_DataStart) >= 1);
|
2013-07-29 07:13:03 -04:00
|
|
|
return m_BufferSize - m_WritePos + m_DataStart - 1;
|
|
|
|
}
|
|
|
|
// Single free space partition:
|
2014-03-08 14:18:51 -05:00
|
|
|
ASSERT(m_BufferSize >= m_WritePos);
|
|
|
|
ASSERT(m_BufferSize - m_WritePos >= 1);
|
2013-07-29 07:13:03 -04:00
|
|
|
return m_DataStart - m_WritePos - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-03-08 11:33:38 -05:00
|
|
|
size_t cByteBuffer::GetUsedSpace(void) const
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
2014-03-08 14:18:51 -05:00
|
|
|
ASSERT(m_BufferSize >= GetFreeSpace());
|
|
|
|
ASSERT((m_BufferSize - GetFreeSpace()) >= 1);
|
2013-11-03 05:58:11 -05:00
|
|
|
return m_BufferSize - GetFreeSpace() - 1;
|
2013-07-29 07:13:03 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-03-08 11:33:38 -05:00
|
|
|
size_t cByteBuffer::GetReadableSpace(void) const
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
if (m_ReadPos > m_WritePos)
|
|
|
|
{
|
|
|
|
// Wrap around the buffer end:
|
2014-03-08 14:18:51 -05:00
|
|
|
ASSERT(m_BufferSize >= m_ReadPos);
|
2013-07-29 07:13:03 -04:00
|
|
|
return m_BufferSize - m_ReadPos + m_WritePos;
|
|
|
|
}
|
|
|
|
// Single readable space partition:
|
2014-03-08 14:18:51 -05:00
|
|
|
ASSERT(m_WritePos >= m_ReadPos);
|
2014-09-11 20:42:04 -04:00
|
|
|
return m_WritePos - m_ReadPos;
|
2013-07-29 07:13:03 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-03-07 14:04:25 -05:00
|
|
|
bool cByteBuffer::CanReadBytes(size_t a_Count) const
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
return (a_Count <= GetReadableSpace());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-03-07 14:04:25 -05:00
|
|
|
bool cByteBuffer::CanWriteBytes(size_t a_Count) const
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
return (a_Count <= GetFreeSpace());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-21 08:00:20 -04:00
|
|
|
bool cByteBuffer::ReadBEInt8(Int8 & a_Value)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
NEEDBYTES(1);
|
|
|
|
ReadBuf(&a_Value, 1);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-21 08:00:20 -04:00
|
|
|
bool cByteBuffer::ReadBEUInt8(UInt8 & a_Value)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
NEEDBYTES(1);
|
|
|
|
ReadBuf(&a_Value, 1);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-21 08:00:20 -04:00
|
|
|
bool cByteBuffer::ReadBEInt16(Int16 & a_Value)
|
2015-01-21 06:12:22 -05:00
|
|
|
{
|
|
|
|
CHECK_THREAD
|
|
|
|
CheckValid();
|
|
|
|
NEEDBYTES(2);
|
2015-03-21 08:00:20 -04:00
|
|
|
UInt16 val;
|
2015-01-21 06:12:22 -05:00
|
|
|
ReadBuf(&val, 2);
|
|
|
|
val = ntohs(val);
|
2015-03-21 08:00:20 -04:00
|
|
|
memcpy(&a_Value, &val, 2);
|
2015-01-21 06:12:22 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool cByteBuffer::ReadBEUInt16(UInt16 & a_Value)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
NEEDBYTES(2);
|
|
|
|
ReadBuf(&a_Value, 2);
|
2015-01-21 06:12:22 -05:00
|
|
|
a_Value = ntohs(a_Value);
|
2013-07-29 07:13:03 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-21 08:00:20 -04:00
|
|
|
bool cByteBuffer::ReadBEInt32(Int32 & a_Value)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
NEEDBYTES(4);
|
2015-03-21 08:00:20 -04:00
|
|
|
UInt32 val;
|
|
|
|
ReadBuf(&val, 4);
|
|
|
|
val = ntohl(val);
|
|
|
|
memcpy(&a_Value, &val, 4);
|
2013-07-29 07:13:03 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-01-21 06:12:22 -05:00
|
|
|
bool cByteBuffer::ReadBEUInt32(UInt32 & a_Value)
|
|
|
|
{
|
|
|
|
CHECK_THREAD
|
|
|
|
CheckValid();
|
|
|
|
NEEDBYTES(4);
|
|
|
|
ReadBuf(&a_Value, 4);
|
|
|
|
a_Value = ntohl(a_Value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-07-29 07:13:03 -04:00
|
|
|
bool cByteBuffer::ReadBEInt64(Int64 & a_Value)
|
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
NEEDBYTES(8);
|
|
|
|
ReadBuf(&a_Value, 8);
|
|
|
|
a_Value = NetworkToHostLong8(&a_Value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-21 08:00:20 -04:00
|
|
|
bool cByteBuffer::ReadBEUInt64(UInt64 & a_Value)
|
|
|
|
{
|
|
|
|
CHECK_THREAD
|
|
|
|
CheckValid();
|
|
|
|
NEEDBYTES(8);
|
|
|
|
ReadBuf(&a_Value, 8);
|
|
|
|
a_Value = NetworkToHostULong8(&a_Value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-07-29 07:13:03 -04:00
|
|
|
bool cByteBuffer::ReadBEFloat(float & a_Value)
|
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
NEEDBYTES(4);
|
|
|
|
ReadBuf(&a_Value, 4);
|
|
|
|
a_Value = NetworkToHostFloat4(&a_Value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool cByteBuffer::ReadBEDouble(double & a_Value)
|
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
NEEDBYTES(8);
|
|
|
|
ReadBuf(&a_Value, 8);
|
|
|
|
a_Value = NetworkToHostDouble8(&a_Value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool cByteBuffer::ReadBool(bool & a_Value)
|
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
NEEDBYTES(1);
|
2015-03-21 08:00:20 -04:00
|
|
|
UInt8 Value = 0;
|
2013-07-29 07:13:03 -04:00
|
|
|
ReadBuf(&Value, 1);
|
|
|
|
a_Value = (Value != 0);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-22 18:09:23 -04:00
|
|
|
bool cByteBuffer::ReadVarInt32(UInt32 & a_Value)
|
2013-10-28 15:40:42 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-10-28 15:40:42 -04:00
|
|
|
CheckValid();
|
2013-10-28 15:57:03 -04:00
|
|
|
UInt32 Value = 0;
|
2013-10-28 15:40:42 -04:00
|
|
|
int Shift = 0;
|
|
|
|
unsigned char b = 0;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
NEEDBYTES(1);
|
|
|
|
ReadBuf(&b, 1);
|
2015-03-22 18:09:23 -04:00
|
|
|
Value = Value | ((static_cast<UInt32>(b & 0x7f)) << Shift);
|
|
|
|
Shift += 7;
|
|
|
|
} while ((b & 0x80) != 0);
|
|
|
|
a_Value = Value;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool cByteBuffer::ReadVarInt64(UInt64 & a_Value)
|
|
|
|
{
|
|
|
|
CHECK_THREAD
|
|
|
|
CheckValid();
|
|
|
|
UInt64 Value = 0;
|
|
|
|
int Shift = 0;
|
|
|
|
unsigned char b = 0;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
NEEDBYTES(1);
|
|
|
|
ReadBuf(&b, 1);
|
|
|
|
Value = Value | ((static_cast<UInt64>(b & 0x7f)) << Shift);
|
2013-10-28 15:40:42 -04:00
|
|
|
Shift += 7;
|
|
|
|
} while ((b & 0x80) != 0);
|
|
|
|
a_Value = Value;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool cByteBuffer::ReadVarUTF8String(AString & a_Value)
|
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-10-28 15:40:42 -04:00
|
|
|
CheckValid();
|
2013-10-28 15:57:03 -04:00
|
|
|
UInt32 Size = 0;
|
2013-10-28 15:40:42 -04:00
|
|
|
if (!ReadVarInt(Size))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (Size > MAX_STRING_SIZE)
|
|
|
|
{
|
2014-03-11 17:43:14 -04:00
|
|
|
LOGWARNING("%s: String too large: %u (%u KiB)", __FUNCTION__, Size, Size / 1024);
|
2013-10-28 15:40:42 -04:00
|
|
|
}
|
2015-07-29 11:04:03 -04:00
|
|
|
return ReadString(a_Value, static_cast<size_t>(Size));
|
2013-10-28 15:40:42 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-11-24 08:35:35 -05:00
|
|
|
bool cByteBuffer::ReadLEInt(int & a_Value)
|
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-11-24 08:35:35 -05:00
|
|
|
CheckValid();
|
|
|
|
NEEDBYTES(4);
|
|
|
|
ReadBuf(&a_Value, 4);
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2013-11-24 08:35:35 -05:00
|
|
|
#ifdef IS_BIG_ENDIAN
|
|
|
|
// Convert:
|
|
|
|
a_Value = ((a_Value >> 24) & 0xff) | ((a_Value >> 16) & 0xff00) | ((a_Value >> 8) & 0xff0000) | (a_Value & 0xff000000);
|
|
|
|
#endif
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2013-11-24 08:35:35 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-22 18:09:23 -04:00
|
|
|
bool cByteBuffer::ReadPosition64(int & a_BlockX, int & a_BlockY, int & a_BlockZ)
|
2014-09-08 11:02:54 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2014-09-08 11:02:54 -04:00
|
|
|
Int64 Value;
|
|
|
|
if (!ReadBEInt64(Value))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:06:21 -04:00
|
|
|
// Convert the 64 received bits into 3 coords:
|
|
|
|
UInt32 BlockXRaw = (Value >> 38) & 0x03ffffff; // Top 26 bits
|
|
|
|
UInt32 BlockYRaw = (Value >> 26) & 0x0fff; // Middle 12 bits
|
|
|
|
UInt32 BlockZRaw = (Value & 0x03ffffff); // Bottom 26 bits
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2014-09-25 17:06:21 -04:00
|
|
|
// If the highest bit in the number's range is set, convert the number into negative:
|
2015-07-29 11:04:03 -04:00
|
|
|
a_BlockX = ((BlockXRaw & 0x02000000) == 0) ? static_cast<int>(BlockXRaw) : -(0x04000000 - static_cast<int>(BlockXRaw));
|
|
|
|
a_BlockY = ((BlockYRaw & 0x0800) == 0) ? static_cast<int>(BlockYRaw) : -(0x0800 - static_cast<int>(BlockYRaw));
|
|
|
|
a_BlockZ = ((BlockZRaw & 0x02000000) == 0) ? static_cast<int>(BlockZRaw) : -(0x04000000 - static_cast<int>(BlockZRaw));
|
2014-09-08 11:02:54 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-21 08:00:20 -04:00
|
|
|
bool cByteBuffer::WriteBEInt8(Int8 a_Value)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
PUTBYTES(1);
|
|
|
|
return WriteBuf(&a_Value, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-21 08:00:20 -04:00
|
|
|
bool cByteBuffer::WriteBEUInt8(UInt8 a_Value)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
PUTBYTES(1);
|
|
|
|
return WriteBuf(&a_Value, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-21 08:00:20 -04:00
|
|
|
bool cByteBuffer::WriteBEInt16(Int16 a_Value)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
PUTBYTES(2);
|
2015-03-21 08:00:20 -04:00
|
|
|
UInt16 val;
|
|
|
|
memcpy(&val, &a_Value, 2);
|
|
|
|
val = htons(val);
|
|
|
|
return WriteBuf(&val, 2);
|
2013-07-29 07:13:03 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-21 08:00:20 -04:00
|
|
|
bool cByteBuffer::WriteBEUInt16(UInt16 a_Value)
|
2014-09-25 14:46:50 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2014-09-25 14:46:50 -04:00
|
|
|
CheckValid();
|
|
|
|
PUTBYTES(2);
|
2015-03-21 08:00:20 -04:00
|
|
|
a_Value = htons(a_Value);
|
|
|
|
return WriteBuf(&a_Value, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool cByteBuffer::WriteBEInt32(Int32 a_Value)
|
|
|
|
{
|
|
|
|
CHECK_THREAD
|
|
|
|
CheckValid();
|
|
|
|
PUTBYTES(4);
|
|
|
|
UInt32 Converted = HostToNetwork4(&a_Value);
|
|
|
|
return WriteBuf(&Converted, 4);
|
2014-09-25 14:46:50 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-21 08:00:20 -04:00
|
|
|
bool cByteBuffer::WriteBEUInt32(UInt32 a_Value)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
PUTBYTES(4);
|
2014-05-01 16:54:22 -04:00
|
|
|
UInt32 Converted = HostToNetwork4(&a_Value);
|
2013-07-29 07:13:03 -04:00
|
|
|
return WriteBuf(&Converted, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool cByteBuffer::WriteBEInt64(Int64 a_Value)
|
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
PUTBYTES(8);
|
2014-05-01 16:54:22 -04:00
|
|
|
UInt64 Converted = HostToNetwork8(&a_Value);
|
2013-07-29 07:13:03 -04:00
|
|
|
return WriteBuf(&Converted, 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-21 08:00:20 -04:00
|
|
|
bool cByteBuffer::WriteBEUInt64(UInt64 a_Value)
|
|
|
|
{
|
|
|
|
CHECK_THREAD
|
|
|
|
CheckValid();
|
|
|
|
PUTBYTES(8);
|
|
|
|
UInt64 Converted = HostToNetwork8(&a_Value);
|
|
|
|
return WriteBuf(&Converted, 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-07-29 07:13:03 -04:00
|
|
|
bool cByteBuffer::WriteBEFloat(float a_Value)
|
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
PUTBYTES(4);
|
2014-05-01 16:54:22 -04:00
|
|
|
UInt32 Converted = HostToNetwork4(&a_Value);
|
2013-07-29 07:13:03 -04:00
|
|
|
return WriteBuf(&Converted, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool cByteBuffer::WriteBEDouble(double a_Value)
|
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
PUTBYTES(8);
|
2014-05-01 16:54:22 -04:00
|
|
|
UInt64 Converted = HostToNetwork8(&a_Value);
|
2013-07-29 07:13:03 -04:00
|
|
|
return WriteBuf(&Converted, 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool cByteBuffer::WriteBool(bool a_Value)
|
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
2015-03-21 08:00:20 -04:00
|
|
|
UInt8 val = a_Value ? 1 : 0;
|
|
|
|
return Write(&val, 1);
|
2013-07-29 07:13:03 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-22 18:09:23 -04:00
|
|
|
bool cByteBuffer::WriteVarInt32(UInt32 a_Value)
|
2013-10-28 15:40:42 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-10-28 15:40:42 -04:00
|
|
|
CheckValid();
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2013-10-28 15:57:03 -04:00
|
|
|
// A 32-bit integer can be encoded by at most 5 bytes:
|
|
|
|
unsigned char b[5];
|
2014-05-01 16:54:22 -04:00
|
|
|
size_t idx = 0;
|
2013-10-28 15:40:42 -04:00
|
|
|
do
|
|
|
|
{
|
|
|
|
b[idx] = (a_Value & 0x7f) | ((a_Value > 0x7f) ? 0x80 : 0x00);
|
|
|
|
a_Value = a_Value >> 7;
|
|
|
|
idx++;
|
|
|
|
} while (a_Value > 0);
|
|
|
|
|
|
|
|
return WriteBuf(b, idx);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-22 18:09:23 -04:00
|
|
|
|
|
|
|
bool cByteBuffer::WriteVarInt64(UInt64 a_Value)
|
|
|
|
{
|
|
|
|
CHECK_THREAD
|
|
|
|
CheckValid();
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2015-03-22 18:09:23 -04:00
|
|
|
// A 64-bit integer can be encoded by at most 10 bytes:
|
|
|
|
unsigned char b[10];
|
|
|
|
size_t idx = 0;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
b[idx] = (a_Value & 0x7f) | ((a_Value > 0x7f) ? 0x80 : 0x00);
|
|
|
|
a_Value = a_Value >> 7;
|
|
|
|
idx++;
|
|
|
|
} while (a_Value > 0);
|
|
|
|
|
|
|
|
return WriteBuf(b, idx);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-10-31 18:47:22 -04:00
|
|
|
bool cByteBuffer::WriteVarUTF8String(const AString & a_Value)
|
2013-10-28 15:40:42 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-10-28 15:40:42 -04:00
|
|
|
CheckValid();
|
|
|
|
PUTBYTES(a_Value.size() + 1); // This is a lower-bound on the bytes that will be actually written. Fail early.
|
2015-03-22 18:09:23 -04:00
|
|
|
bool res = WriteVarInt32(static_cast<UInt32>(a_Value.size()));
|
2013-10-28 15:40:42 -04:00
|
|
|
if (!res)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return WriteBuf(a_Value.data(), a_Value.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-21 08:00:20 -04:00
|
|
|
bool cByteBuffer::WriteLEInt32(Int32 a_Value)
|
2013-11-24 08:35:35 -05:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-11-24 08:35:35 -05:00
|
|
|
CheckValid();
|
|
|
|
#ifdef IS_LITTLE_ENDIAN
|
2015-03-22 14:46:08 -04:00
|
|
|
return WriteBuf(reinterpret_cast<const char *>(&a_Value), 4);
|
2013-11-24 08:35:35 -05:00
|
|
|
#else
|
|
|
|
int Value = ((a_Value >> 24) & 0xff) | ((a_Value >> 16) & 0xff00) | ((a_Value >> 8) & 0xff0000) | (a_Value & 0xff000000);
|
2015-03-22 14:46:08 -04:00
|
|
|
return WriteBuf(reinterpret_cast<const char *>(&Value), 4);
|
2013-11-24 08:35:35 -05:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-03-22 14:46:08 -04:00
|
|
|
bool cByteBuffer::WritePosition64(Int32 a_BlockX, Int32 a_BlockY, Int32 a_BlockZ)
|
2014-09-08 11:02:54 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2015-03-21 08:00:20 -04:00
|
|
|
CheckValid();
|
2015-03-22 14:46:08 -04:00
|
|
|
return WriteBEInt64(
|
|
|
|
(static_cast<Int64>(a_BlockX & 0x3FFFFFF) << 38) |
|
|
|
|
(static_cast<Int64>(a_BlockY & 0xFFF) << 26) |
|
|
|
|
(static_cast<Int64>(a_BlockZ & 0x3FFFFFF))
|
|
|
|
);
|
2014-09-08 11:02:54 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-03-08 10:36:52 -05:00
|
|
|
bool cByteBuffer::ReadBuf(void * a_Buffer, size_t a_Count)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
NEEDBYTES(a_Count);
|
2015-07-29 11:04:03 -04:00
|
|
|
char * Dst = reinterpret_cast<char *>(a_Buffer); // So that we can do byte math
|
2014-03-08 14:18:51 -05:00
|
|
|
ASSERT(m_BufferSize >= m_ReadPos);
|
2014-03-08 11:33:38 -05:00
|
|
|
size_t BytesToEndOfBuffer = m_BufferSize - m_ReadPos;
|
2013-07-29 07:13:03 -04:00
|
|
|
if (BytesToEndOfBuffer <= a_Count)
|
|
|
|
{
|
|
|
|
// Reading across the ringbuffer end, read the first part and adjust parameters:
|
|
|
|
if (BytesToEndOfBuffer > 0)
|
|
|
|
{
|
|
|
|
memcpy(Dst, m_Buffer + m_ReadPos, BytesToEndOfBuffer);
|
|
|
|
Dst += BytesToEndOfBuffer;
|
|
|
|
a_Count -= BytesToEndOfBuffer;
|
|
|
|
}
|
|
|
|
m_ReadPos = 0;
|
|
|
|
}
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2013-07-29 07:13:03 -04:00
|
|
|
// Read the rest of the bytes in a single read (guaranteed to fit):
|
|
|
|
if (a_Count > 0)
|
|
|
|
{
|
|
|
|
memcpy(Dst, m_Buffer + m_ReadPos, a_Count);
|
|
|
|
m_ReadPos += a_Count;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-03-08 10:36:52 -05:00
|
|
|
bool cByteBuffer::WriteBuf(const void * a_Buffer, size_t a_Count)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
PUTBYTES(a_Count);
|
2015-07-29 11:04:03 -04:00
|
|
|
char * Src = reinterpret_cast<char *>(const_cast<void*>(a_Buffer)); // So that we can do byte math
|
2014-03-08 14:18:51 -05:00
|
|
|
ASSERT(m_BufferSize >= m_ReadPos);
|
2014-03-08 11:33:38 -05:00
|
|
|
size_t BytesToEndOfBuffer = m_BufferSize - m_WritePos;
|
2013-07-29 07:13:03 -04:00
|
|
|
if (BytesToEndOfBuffer <= a_Count)
|
|
|
|
{
|
|
|
|
// Reading across the ringbuffer end, read the first part and adjust parameters:
|
|
|
|
memcpy(m_Buffer + m_WritePos, Src, BytesToEndOfBuffer);
|
|
|
|
Src += BytesToEndOfBuffer;
|
|
|
|
a_Count -= BytesToEndOfBuffer;
|
|
|
|
m_WritePos = 0;
|
|
|
|
}
|
2016-02-05 16:45:45 -05:00
|
|
|
|
2013-07-29 07:13:03 -04:00
|
|
|
// Read the rest of the bytes in a single read (guaranteed to fit):
|
|
|
|
if (a_Count > 0)
|
|
|
|
{
|
|
|
|
memcpy(m_Buffer + m_WritePos, Src, a_Count);
|
|
|
|
m_WritePos += a_Count;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-03-08 10:36:52 -05:00
|
|
|
bool cByteBuffer::ReadString(AString & a_String, size_t a_Count)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
NEEDBYTES(a_Count);
|
|
|
|
a_String.clear();
|
|
|
|
a_String.reserve(a_Count);
|
2014-03-08 14:18:51 -05:00
|
|
|
ASSERT(m_BufferSize >= m_ReadPos);
|
2014-03-08 11:33:38 -05:00
|
|
|
size_t BytesToEndOfBuffer = m_BufferSize - m_ReadPos;
|
2013-07-29 07:13:03 -04:00
|
|
|
if (BytesToEndOfBuffer <= a_Count)
|
|
|
|
{
|
|
|
|
// Reading across the ringbuffer end, read the first part and adjust parameters:
|
|
|
|
if (BytesToEndOfBuffer > 0)
|
|
|
|
{
|
|
|
|
a_String.assign(m_Buffer + m_ReadPos, BytesToEndOfBuffer);
|
2014-03-08 14:18:51 -05:00
|
|
|
ASSERT(a_Count >= BytesToEndOfBuffer);
|
2013-07-29 07:13:03 -04:00
|
|
|
a_Count -= BytesToEndOfBuffer;
|
|
|
|
}
|
|
|
|
m_ReadPos = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read the rest of the bytes in a single read (guaranteed to fit):
|
|
|
|
if (a_Count > 0)
|
|
|
|
{
|
|
|
|
a_String.append(m_Buffer + m_ReadPos, a_Count);
|
|
|
|
m_ReadPos += a_Count;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-03-07 14:04:25 -05:00
|
|
|
bool cByteBuffer::SkipRead(size_t a_Count)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
if (!CanReadBytes(a_Count))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
AdvanceReadPos(a_Count);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void cByteBuffer::ReadAll(AString & a_Data)
|
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
ReadString(a_Data, GetReadableSpace());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-12-22 09:19:29 -05:00
|
|
|
bool cByteBuffer::ReadToByteBuffer(cByteBuffer & a_Dst, size_t a_NumBytes)
|
2013-12-13 11:53:00 -05:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-12-13 11:53:00 -05:00
|
|
|
if (!a_Dst.CanWriteBytes(a_NumBytes) || !CanReadBytes(a_NumBytes))
|
|
|
|
{
|
|
|
|
// There's not enough source bytes or space in the dest BB
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
char buf[1024];
|
2013-12-22 12:02:36 -05:00
|
|
|
// > 0 without generating warnings about unsigned comparisons where size_t is unsigned
|
|
|
|
while (a_NumBytes != 0)
|
2013-12-13 11:53:00 -05:00
|
|
|
{
|
2013-12-22 09:19:29 -05:00
|
|
|
size_t num = (a_NumBytes > sizeof(buf)) ? sizeof(buf) : a_NumBytes;
|
2013-12-13 11:53:00 -05:00
|
|
|
VERIFY(ReadBuf(buf, num));
|
|
|
|
VERIFY(a_Dst.Write(buf, num));
|
2014-03-08 14:18:51 -05:00
|
|
|
ASSERT(a_NumBytes >= num);
|
2013-12-13 11:53:00 -05:00
|
|
|
a_NumBytes -= num;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-07-29 07:13:03 -04:00
|
|
|
void cByteBuffer::CommitRead(void)
|
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
m_DataStart = m_ReadPos;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void cByteBuffer::ResetRead(void)
|
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
m_ReadPos = m_DataStart;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void cByteBuffer::ReadAgain(AString & a_Out)
|
|
|
|
{
|
|
|
|
// Return the data between m_DataStart and m_ReadPos (the data that has been read but not committed)
|
|
|
|
// Used by ProtoProxy to repeat communication twice, once for parsing and the other time for the remote party
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
2014-03-08 14:18:51 -05:00
|
|
|
size_t DataStart = m_DataStart;
|
2013-07-29 07:13:03 -04:00
|
|
|
if (m_ReadPos < m_DataStart)
|
|
|
|
{
|
|
|
|
// Across the ringbuffer end, read the first part and adjust next part's start:
|
2014-03-08 14:18:51 -05:00
|
|
|
ASSERT(m_BufferSize >= m_DataStart);
|
2013-07-29 07:13:03 -04:00
|
|
|
a_Out.append(m_Buffer + m_DataStart, m_BufferSize - m_DataStart);
|
|
|
|
DataStart = 0;
|
|
|
|
}
|
2014-03-08 14:18:51 -05:00
|
|
|
ASSERT(m_ReadPos >= DataStart);
|
2013-07-29 07:13:03 -04:00
|
|
|
a_Out.append(m_Buffer + DataStart, m_ReadPos - DataStart);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-03-07 14:04:25 -05:00
|
|
|
void cByteBuffer::AdvanceReadPos(size_t a_Count)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-12-03 12:04:08 -05:00
|
|
|
CHECK_THREAD
|
2013-07-29 07:13:03 -04:00
|
|
|
CheckValid();
|
|
|
|
m_ReadPos += a_Count;
|
2013-11-29 12:39:40 -05:00
|
|
|
if (m_ReadPos >= m_BufferSize)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
m_ReadPos -= m_BufferSize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void cByteBuffer::CheckValid(void) const
|
|
|
|
{
|
|
|
|
ASSERT(m_ReadPos < m_BufferSize);
|
|
|
|
ASSERT(m_WritePos < m_BufferSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-01-26 11:48:09 -05:00
|
|
|
|
1.9 / 1.9.2 / 1.9.3 / 1.9.4 protocol support (#3135)
* Semistable update to 15w31a
I'm going through snapshots in a sequential order since it should make things easier, and since protocol version history is written.
* Update to 15w34b protocol
Also, fix an issue with the Entity Equipment packet from the past version. Clients are able to connect and do stuff!
* Partially update to 15w35e
Chunk data doesn't work, but the client joins. I'm waiting to do chunk data because chunk data has an incomplete format until 15w36d.
* Add '/blk' debug command
This command lets one see what block they are looking at, and makes figuring out what's supposed to be where in a highly broken chunk possible.
* Fix CRLF normalization in CheckBasicStyle.lua
Normally, this doesn't cause an issue, but when running from cygwin, it detects the CR as whitespace and creates thousands of violations for every single line. Lua, when run on windows, will normalize automatically, but when run via cygwin, it won't.
The bug was simply that gsub was returning a replaced version, but not changing the parameter, so the replaced version was ignored.
* Update to 15w40b
This includes chunk serialization. Fully functional chunk serialization for 1.9.
I'm not completely happy with the chunk serialization as-is (correct use of palettes would be great), but cuberite also doesn't skip sending empty chunks so this performance optimization should probably come later. The creation of a full buffer is suboptimal, but it's the easiest way to implement this code.
* Write long-by-long rather than creating a buffer
This is a bit faster and should be equivalent. However, the code still doesn't look too good.
* Update to 15w41a protocol
This includes the new set passengers packet, which works off of the ridden entity, not the rider. That means, among other things, that information about the previously ridden vehicle is needed when detaching. So a new method with that info was added.
* Update to 15w45a
* 15w51b protocol
* Update to 1.9.0 protocol
Closes #3067. There are still a few things that need to be worked out (picking up items, effects, particles, and most importantly inventory), but in general this should work. I'll make a few more changes tomorrow to get the rest of the protocol set up, along with 1.9.1/1.9.2 (which did make a few changes). Chunks, however, _are_ working, along with most other parts of the game (placing/breaking blocks).
* Fix item pickup packet not working
That was a silly mistake, but at least it was an easy one.
* 1.9.2 protocol support
* Fix version info found in server list ping
Thus, the client reports that it can connect rather than saying that the server is out of date. This required creating separate classes for 1.9.1 and 1.9.2, unfortunately.
* Fix build errors generated by clang
These didn't happen in MSVC.
* Add protocol19x.cpp and protocol19x.h to CMakeLists
* Ignore warnings in protocol19x that are ignored in protocol18x
* Document BLOCK_FACE and DIG_STATUS constants
* Fix BLOCK_FACE links and add separate section for DIG_STATUS
* Fix bat animation and object spawning
The causes of both of these are explained in #3135, but the gist is that both were typos.
* Implement Use Item packet
This means that buckets, bows, fishing rods, and several other similar items now work when not looking at a block.
* Handle DIG_STATUS_SWAP_ITEM_IN_HAND
* Add support for spawn eggs and potions
The items are transformed from the 1.9 version to the 1.8 version when reading and transformed back when sending.
* Remove spammy potion debug logging
* Fix wolf collar color metadata
The wrong type was being used, causing several clientside issues (including the screen going black).
* Fix 1.9 chunk sending in the nether
The nether and the end don't send skylight.
* Fix clang build errors
* Fix water bottles becoming mundane potions
This happened because the can become splash potion bit got set incorrectly. Water bottles and mundane potions are only differentiated by the fact that water bottles have a metadata of 0, so setting that bit made it a mundane potion.
Also add missing break statements to the read item NBT switch, which would otherwise break items with custom names and also cause incorrect "Unimplemented NBT data when parsing!" logging.
* Copy Protocol18x as Protocol19x
Aditionally, method and class names have been swapped to clean up other diffs. This commit is only added to make the following diffs more readable; it doesn't make any other changes (beyond class names).
* Make thrown potions use the correct appearence
This was caused by potions now using metadata.
* Add missing api doc for cSplashPotionEntity::GetItem
* Fix compile error in SplashPotionEntity.cpp
* Fix fix of cSplashPotionEntity API doc
* Temporarilly disable fall damage particles
These were causing issues in 1.9 due to the changed effect ID.
* Properly send a kick packet when connecting with an invalid version
This means that the client no longer waits on the server screen with no indication whatsoever. However, right now the server list ping isn't implemented for unknown versions, so it'll only load "Old" on the ping.
I also added a GetVarIntSize method to cByteBuffer. This helps clean up part of the code here (and I think it could clean up other parts), but it may make sense for it to be moved elsewhere (or declared in a different way).
* Handle server list pings from unrecognized versions
This isn't the cleanest way of writing it (it feels odd to use ProtocolRecognizer to send packets, and the addition of m_InPingForUnrecognizedVersion feels like the wrong technique), but it works and I can't think of a better way (apart from creating a full separate protocol class to handle only the ping... which would be worse).
* Use cPacketizer for the disconnect packet
This also should fix clang build errors.
* Add 1.9.3 / 1.9.4 support
* Fix incorrect indentation in APIDesc
2016-05-14 15:12:42 -04:00
|
|
|
size_t cByteBuffer::GetVarIntSize(UInt32 a_Value)
|
|
|
|
{
|
|
|
|
size_t Count = 0;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
// If the value cannot be expressed in 7 bits, it needs to take up another byte
|
|
|
|
Count++;
|
|
|
|
a_Value >>= 7;
|
|
|
|
} while (a_Value != 0);
|
|
|
|
|
|
|
|
return Count;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|