git-svn-id: svn://db.shs.com.ru/libs@622 a8b55f48-bf90-11e4-a774-851b48703e85
This commit is contained in:
@@ -23,6 +23,32 @@ void ChunkStream::setSource(QDataStream & str) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int ChunkStream::read() {
|
||||||
|
switch (version_) {
|
||||||
|
case Version_1:
|
||||||
|
stream_ >> last_id >> last_data;
|
||||||
|
break;
|
||||||
|
case Version_2:
|
||||||
|
last_id = readVInt(stream_);
|
||||||
|
last_data.resize(readVInt(stream_));
|
||||||
|
//piCout << last_id << last_data.size();
|
||||||
|
stream_.readRawData(last_data.data(), last_data.size());
|
||||||
|
break;
|
||||||
|
default: break;
|
||||||
|
}
|
||||||
|
return last_id;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void ChunkStream::readAll() {
|
||||||
|
data_map.clear();
|
||||||
|
while (!atEnd()) {
|
||||||
|
read();
|
||||||
|
data_map[last_id] = last_data;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
ChunkStream::~ChunkStream() {
|
ChunkStream::~ChunkStream() {
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -35,4 +61,53 @@ void ChunkStream::_init() {
|
|||||||
buffer.setBuffer(data_);
|
buffer.setBuffer(data_);
|
||||||
buffer.open(QIODevice::ReadWrite);
|
buffer.open(QIODevice::ReadWrite);
|
||||||
stream_.setDevice(&buffer);
|
stream_.setDevice(&buffer);
|
||||||
|
if (!data_->isEmpty()) {
|
||||||
|
uchar v = data_->at(0);
|
||||||
|
if ((v & 0x80) == 0x80) {
|
||||||
|
v &= 0x7f;
|
||||||
|
switch (v) {
|
||||||
|
case 2: version_ = (uchar)Version_2; stream_.skipRawData(1); break;
|
||||||
|
default: break;
|
||||||
|
}
|
||||||
|
} else
|
||||||
|
version_ = Version_1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
uint ChunkStream::readVInt(QDataStream & s) {
|
||||||
|
if (s.atEnd()) return 0;
|
||||||
|
uchar bytes[4]; s >> bytes[0];
|
||||||
|
uchar abc = 0;
|
||||||
|
for (abc = 0; abc < 3; ++abc) {
|
||||||
|
uchar mask = (0x80 >> abc);
|
||||||
|
if ((bytes[0] & mask) == mask) {
|
||||||
|
//if (s.isEmpty()) return 0;
|
||||||
|
bytes[0] &= (mask - 1);
|
||||||
|
s >> bytes[abc + 1];
|
||||||
|
} else break;
|
||||||
|
}
|
||||||
|
uint ret = 0;
|
||||||
|
for (int i = 0; i <= abc; ++i) {
|
||||||
|
ret += (bytes[i] << (8 * ((int)abc - i)));
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void ChunkStream::writeVInt(QDataStream & s, uint val) {
|
||||||
|
if (val > 0xfffffff) return;
|
||||||
|
if (val <= 0x7f) {
|
||||||
|
s << uchar(val);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (val <= 0x3fff) {
|
||||||
|
s << uchar((val >> 8) | 0x80) << uchar(val & 0xff);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (val <= 0x1fffff) {
|
||||||
|
s << uchar((val >> 16) | 0xc0) << uchar((val >> 8) & 0xff) << uchar(val & 0xff);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
s << uchar((val >> 24) | 0xe0) << uchar((val >> 16) & 0xff) << uchar((val >> 8) & 0xff) << uchar(val & 0xff);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,14 +4,20 @@
|
|||||||
#include <QDataStream>
|
#include <QDataStream>
|
||||||
#include <QBuffer>
|
#include <QBuffer>
|
||||||
#include <QByteArray>
|
#include <QByteArray>
|
||||||
|
#include <QMap>
|
||||||
|
|
||||||
|
|
||||||
class ChunkStream
|
class ChunkStream
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
ChunkStream(const QByteArray & data) {setSource(data);}
|
enum Version {
|
||||||
ChunkStream(QDataStream & str) {setSource(str);}
|
Version_1 /*! First, old version */,
|
||||||
ChunkStream(QByteArray * data = 0) {setSource(data);}
|
Version_2 /*! Second, more optimized version */ = 2,
|
||||||
|
};
|
||||||
|
ChunkStream(const QByteArray & data): version_(Version_2) {setSource(data);}
|
||||||
|
ChunkStream(QDataStream & str): version_(Version_2) {setSource(str);}
|
||||||
|
ChunkStream(QByteArray * data = 0, Version v = Version_2): version_(v) {setSource(data);}
|
||||||
|
ChunkStream(Version v): version_(v) {setSource(0);}
|
||||||
~ChunkStream();
|
~ChunkStream();
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
@@ -30,21 +36,38 @@ public:
|
|||||||
QDataStream & dataStream() {return stream_;}
|
QDataStream & dataStream() {return stream_;}
|
||||||
QByteArray data() const {return tmp_data;}
|
QByteArray data() const {return tmp_data;}
|
||||||
bool atEnd() const {return stream_.atEnd();}
|
bool atEnd() const {return stream_.atEnd();}
|
||||||
|
Version version() const {return (Version)version_;}
|
||||||
|
|
||||||
int read() {stream_ >> last_id >> last_data; return last_id;}
|
int read();
|
||||||
|
void readAll();
|
||||||
int getID() {return last_id;}
|
int getID() {return last_id;}
|
||||||
template <typename T>
|
template <typename T>
|
||||||
T getData() const {T ret; QDataStream s(last_data); s.setVersion(QDataStream::Qt_4_8); s >> ret; return ret;}
|
T getData() const {T ret; QDataStream s(last_data); s.setVersion(QDataStream::Qt_4_8); s >> ret; return ret;}
|
||||||
template <typename T>
|
template <typename T>
|
||||||
void get(T & v) const {v = getData<T>();}
|
void get(T & v) const {v = getData<T>();}
|
||||||
|
template <typename T>
|
||||||
|
const ChunkStream & get(int id, T & v) const {
|
||||||
|
QByteArray ba = data_map.value(id);
|
||||||
|
if (!ba.isEmpty()) {
|
||||||
|
QDataStream s(ba); s.setVersion(QDataStream::Qt_4_8);
|
||||||
|
s >> v;
|
||||||
|
}
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
private:
|
private:
|
||||||
void _init();
|
void _init();
|
||||||
|
|
||||||
|
static uint readVInt(QDataStream & s);
|
||||||
|
static void writeVInt(QDataStream & s, uint val);
|
||||||
|
|
||||||
int last_id;
|
int last_id;
|
||||||
|
uchar version_;
|
||||||
QByteArray * data_, last_data, tmp_data;
|
QByteArray * data_, last_data, tmp_data;
|
||||||
QBuffer buffer;
|
QBuffer buffer;
|
||||||
QDataStream stream_;
|
QDataStream stream_;
|
||||||
|
QMap<int, QByteArray> data_map;
|
||||||
|
|
||||||
|
template <typename T> friend ChunkStream & operator <<(ChunkStream & s, const ChunkStream::Chunk<T> & c);
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
@@ -53,7 +76,19 @@ ChunkStream & operator <<(ChunkStream & s, const ChunkStream::Chunk<T> & c) {
|
|||||||
QDataStream bas(&ba, QIODevice::WriteOnly);
|
QDataStream bas(&ba, QIODevice::WriteOnly);
|
||||||
bas.setVersion(QDataStream::Qt_4_8);
|
bas.setVersion(QDataStream::Qt_4_8);
|
||||||
bas << c.data;
|
bas << c.data;
|
||||||
s.dataStream() << c.id << ba;
|
switch (s.version_) {
|
||||||
|
case ChunkStream::Version_1:
|
||||||
|
s.stream_ << c.id << ba;
|
||||||
|
break;
|
||||||
|
case ChunkStream::Version_2:
|
||||||
|
if (s.data_->isEmpty())
|
||||||
|
s.stream_ << uchar(uchar(s.version_) | 0x80);
|
||||||
|
ChunkStream::writeVInt(s.stream_, c.id);
|
||||||
|
ChunkStream::writeVInt(s.stream_, ba.size());
|
||||||
|
s.stream_.writeRawData(ba.data(), ba.size());
|
||||||
|
break;
|
||||||
|
default: break;
|
||||||
|
}
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user