From 1718cde75e4c0db7b74376b847d58559d646ce1e Mon Sep 17 00:00:00 2001 From: Thorvald Natvig Date: Thu, 26 Mar 2009 21:46:04 +0100 Subject: Make Mumble compile with protobuf. --- src/PacketDataStream.h | 406 +++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 406 insertions(+) create mode 100644 src/PacketDataStream.h (limited to 'src/PacketDataStream.h') diff --git a/src/PacketDataStream.h b/src/PacketDataStream.h new file mode 100644 index 000000000..948cb03dd --- /dev/null +++ b/src/PacketDataStream.h @@ -0,0 +1,406 @@ +/* Copyright (C) 2005-2009, Thorvald Natvig + + All rights reserved. + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions + are met: + + - Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + - Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. + - Neither the name of the Mumble Developers nor the names of its + contributors may be used to endorse or promote products derived from this + software without specific prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR + CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, + EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, + PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR + PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF + LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING + NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*/ + +#ifndef _PACKETDATASTREAM_H +#define _PACKETDATASTREAM_H + +/* + * GCC doesn't yet do inter-object-file inlining, so unfortunately, this all has to be defined here. + */ + +class PacketDataStream { + private: + Q_DISABLE_COPY(PacketDataStream) + private: + unsigned char *data; + quint32 maxsize; + quint32 offset; + quint32 overshoot; + bool ok; + public: + quint32 size() const { + return offset; + } + + quint32 capacity() const { + return maxsize; + } + + bool isValid() const { + return ok; + } + + quint32 left() const { + return maxsize - offset; + } + + quint32 undersize() const { + return overshoot; + } + + void append(const quint64 v) { +#ifndef QT_NO_DEBUG + Q_ASSERT(v <= 0xff); +#endif + if (offset < maxsize) + data[offset++] = static_cast(v); + else { + ok = false; + overshoot++; + } + }; + + void append(const char *d, quint32 len) { + if (left() >= len) { + memcpy(& data[offset], d, len); + offset += len; + } else { + int l = left(); + memset(& data[offset], 0, l); + offset += l; + overshoot += len - l; + ok = false; + } + } + + void skip(quint32 len) { + if (left() >= len) + offset += len; + else + ok = false; + } + + quint64 next() { + if (offset < maxsize) + return data[offset++]; + else { + ok = false; + return 0; + } + }; + + void rewind() { + offset = 0; + } + + void truncate() { + maxsize = offset; + } + + const unsigned char *dataPtr() const { + return reinterpret_cast(& data[offset]); + } + + const char *charPtr() const { + return reinterpret_cast(& data[offset]); + } + + QByteArray dataBlock(quint32 len) { + if (len <= left()) { + QByteArray a(charPtr(), len); + offset +=len; + return a; + } else { + ok = false; + return QByteArray(); + } + } + + protected: + void setup(unsigned char *d, int msize) { + data = d; + offset = 0; + overshoot = 0; + maxsize = msize; + ok = true; + } + public: + PacketDataStream(const char *d, int msize) { + setup(const_cast(reinterpret_cast(d)), msize); + }; + + PacketDataStream(char *d, int msize) { + setup(reinterpret_cast(d), msize); + }; + + PacketDataStream(unsigned char *d, int msize) { + setup(d, msize); + }; + + PacketDataStream &operator <<(const quint64 value) { + quint64 i = value; + + if ((i & 0x8000000000000000LL) && (~i < 0x100000000LL)) { + // Signed number. + i = ~i; + if (i <= 0x3) { + // Shortcase for -1 to -4 + append(0xFC | i); + return *this; + } else { + append(0xF8); + } + } + if (i < 0x80) { + // Need top bit clear + append(i); + } else if (i < 0x4000) { + // Need top two bits clear + append((i >> 8) | 0x80); + append(i & 0xFF); + } else if (i < 0x200000) { + // Need top three bits clear + append((i >> 16) | 0xC0); + append((i >> 8) & 0xFF); + append(i & 0xFF); + } else if (i < 0x10000000) { + // Need top four bits clear + append((i >> 24) | 0xE0); + append((i >> 16) & 0xFF); + append((i >> 8) & 0xFF); + append(i & 0xFF); + } else if (i < 0x100000000LL) { + // It's a full 32-bit integer. + append(0xF0); + append((i >> 24) & 0xFF); + append((i >> 16) & 0xFF); + append((i >> 8) & 0xFF); + append(i & 0xFF); + } else { + // It's a 64-bit value. + append(0xF4); + append((i >> 56) & 0xFF); + append((i >> 48) & 0xFF); + append((i >> 40) & 0xFF); + append((i >> 32) & 0xFF); + append((i >> 24) & 0xFF); + append((i >> 16) & 0xFF); + append((i >> 8) & 0xFF); + append(i & 0xFF); + } + return *this; + } + + PacketDataStream &operator >>(quint64 &i) { + quint64 v = next(); + + if ((v & 0x80) == 0x00) { + i=(v & 0x7F); + } else if ((v & 0xC0) == 0x80) { + i=(v & 0x3F) << 8 | next(); + } else if ((v & 0xF0) == 0xF0) { + switch (v & 0xFC) { + case 0xF0: + i=next() << 24 | next() << 16 | next() << 8 | next(); + break; + case 0xF4: + i=next() << 56 | next() << 48 | next() << 40 | next() << 32 | next() << 24 | next() << 16 | next() << 8 | next(); + break; + case 0xF8: + *this >> i; + i = ~i; + break; + case 0xFC: + i=v & 0x03; + i = ~i; + break; + default: + ok = false; + i = 0; + break; + } + } else if ((v & 0xF0) == 0xE0) { + i=(v & 0x0F) << 24 | next() << 16 | next() << 8 | next(); + } else if ((v & 0xE0) == 0xC0) { + i=(v & 0x1F) << 16 | next() << 8 | next(); + } + return *this; + } + + PacketDataStream &operator <<(const QByteArray &a) { + *this << a.size(); + append(a.constData(), a.size()); + return *this; + } + + PacketDataStream &operator >>(QByteArray &a) { + quint32 len; + *this >> len; + if (len > left()) { + len = left(); + ok = false; + } + a = QByteArray(reinterpret_cast(& data[offset]), len); + offset+=len; + return *this; + } + + PacketDataStream &operator <<(const QString &s) { + return *this << s.toUtf8(); + } + + // Using the data directly instead of through qbuff avoids a copy. + PacketDataStream &operator >>(QString &s) { + quint32 len; + *this >> len; + if (len > left()) { + len = left(); + ok = false; + } + s = QString::fromUtf8(reinterpret_cast(& data[offset]), len); + offset+=len; + return *this; + } + + PacketDataStream &operator <<(const bool b) { + quint32 v = b ? 1 : 0; + return *this << v; + } + + PacketDataStream &operator >>(bool &b) { + quint32 v; + *this >> v; + b = v ? true : false; + return *this; + } + +#define INTMAPOPERATOR(type) \ + PacketDataStream &operator <<(const type v) { \ + return *this << static_cast(v); \ + } \ + PacketDataStream &operator >>(type &v) { \ + quint64 vv; \ + *this >> vv; \ + v = static_cast(vv); \ + return *this; \ + } + + + INTMAPOPERATOR(int); + INTMAPOPERATOR(unsigned int); + INTMAPOPERATOR(short); + INTMAPOPERATOR(unsigned short); + INTMAPOPERATOR(char); + INTMAPOPERATOR(unsigned char); + + union double64u { + quint64 ui; + double d; + }; + + PacketDataStream &operator <<(const double v) { + double64u u; + u.d = v; + return *this << u.ui; + } + + PacketDataStream &operator >>(double &v) { + double64u u; + *this >> u.ui; + v = u.d; + return *this; + } + + template + PacketDataStream &operator <<(const QList &l) { + *this << l.size(); + for (int i=0;i < l.size();i++) + *this << l.at(i); + return *this; + } + + template + PacketDataStream &operator >>(QList &l) { + l.clear(); + quint32 len; + *this >> len; + if (len > left()) { + len = left(); + ok = false; + } + for (quint32 i=0;i> t; + l.append(t); + } + return *this; + } + + + template + PacketDataStream &operator <<(const QSet &s) { + *this << s.size(); + for (typename QSet::const_iterator i=s.constBegin();i!=s.constEnd();++i) + *this << *i; + return *this; + } + + template + PacketDataStream &operator >>(QSet &s) { + s.clear(); + quint32 len; + *this >> len; + if (len > left()) { + len = left(); + ok = false; + } + for (quint32 i=0;i> t; + s.insert(t); + } + return *this; + } + + template + PacketDataStream &operator <<(const QPair &p) { + return *this << p.first << p.second; + } + + template + PacketDataStream &operator >>(QPair &p) { + return *this >> p.first >> p.second; + } + +}; + +#else +class PacketDataStream; +#endif -- cgit v1.2.3