Commit afb6f218 authored by Davis King's avatar Davis King

Changed the default serialization for integer and byte types so that it runs about 3

times faster.  The behavior should be exactly the same.  The change is basically just
a switch to using the streambuf directly.
parent 8a0b380b
...@@ -167,8 +167,8 @@ namespace dlib ...@@ -167,8 +167,8 @@ namespace dlib
!*/ !*/
{ {
COMPILE_TIME_ASSERT(sizeof(T) <= 8); COMPILE_TIME_ASSERT(sizeof(T) <= 8);
unsigned char buf[8]; unsigned char buf[9];
unsigned char size = 0; unsigned char size = sizeof(T);
unsigned char neg; unsigned char neg;
if (item < 0) if (item < 0)
{ {
...@@ -180,25 +180,22 @@ namespace dlib ...@@ -180,25 +180,22 @@ namespace dlib
neg = 0; neg = 0;
} }
for (unsigned char i = 0; i < sizeof(T); ++i) for (unsigned char i = 1; i <= sizeof(T); ++i)
{ {
buf[i] = static_cast<unsigned char>(item&0xFF); buf[i] = static_cast<unsigned char>(item&0xFF);
item >>= 8; item >>= 8;
if (item == 0) { size = i+1; break; } if (item == 0) { size = i; break; }
} }
if (size == 0)
size = sizeof(T);
size |= neg;
out.write(reinterpret_cast<char*>(&size),1);
size &= 0x7F; // strip off the neg flag
out.write(reinterpret_cast<char*>(buf),size);
// check if there was an error std::streambuf* sbuf = out.rdbuf();
if (!out) buf[0] = size|neg;
if (sbuf->sputn(reinterpret_cast<char*>(buf),size+1) != size+1)
{
out.setstate(std::ios::eofbit | std::ios::badbit);
return true; return true;
else }
return false;
return false;
} }
// ------------------------------------------------------------------------------------ // ------------------------------------------------------------------------------------
...@@ -229,11 +226,20 @@ namespace dlib ...@@ -229,11 +226,20 @@ namespace dlib
unsigned char size; unsigned char size;
bool is_negative; bool is_negative;
std::streambuf* sbuf = in.rdbuf();
item = 0; item = 0;
in.read(reinterpret_cast<char*>(&size),1); int ch = sbuf->sbumpc();
// check if an error occurred if (ch != EOF)
if (!in) {
size = static_cast<unsigned char>(ch);
}
else
{
in.setstate(std::ios::badbit);
return true; return true;
}
if (size&0x80) if (size&0x80)
is_negative = true; is_negative = true;
else else
...@@ -242,13 +248,16 @@ namespace dlib ...@@ -242,13 +248,16 @@ namespace dlib
// check if the serialized object is too big // check if the serialized object is too big
if (size > sizeof(T)) if (size > sizeof(T))
{
return true; return true;
}
in.read(reinterpret_cast<char*>(&buf),size); if (sbuf->sgetn(reinterpret_cast<char*>(&buf),size) != size)
{
// check if there was an error reading from in. in.setstate(std::ios::badbit);
if (!in)
return true; return true;
}
for (unsigned char i = size-1; true; --i) for (unsigned char i = size-1; true; --i)
{ {
...@@ -286,26 +295,25 @@ namespace dlib ...@@ -286,26 +295,25 @@ namespace dlib
!*/ !*/
{ {
COMPILE_TIME_ASSERT(sizeof(T) <= 8); COMPILE_TIME_ASSERT(sizeof(T) <= 8);
unsigned char buf[8]; unsigned char buf[9];
unsigned char size = 0; unsigned char size = sizeof(T);
for (unsigned char i = 0; i < sizeof(T); ++i) for (unsigned char i = 1; i <= sizeof(T); ++i)
{ {
buf[i] = static_cast<unsigned char>(item&0xFF); buf[i] = static_cast<unsigned char>(item&0xFF);
item >>= 8; item >>= 8;
if (item == 0) { size = i+1; break; } if (item == 0) { size = i; break; }
} }
if (size == 0)
size = sizeof(T);
out.write(reinterpret_cast<char*>(&size),1);
out.write(reinterpret_cast<char*>(buf),size);
// check if there was an error std::streambuf* sbuf = out.rdbuf();
if (!out) buf[0] = size;
if (sbuf->sputn(reinterpret_cast<char*>(buf),size+1) != size+1)
{
out.setstate(std::ios::eofbit | std::ios::badbit);
return true; return true;
else }
return false;
return false;
} }
// ------------------------------------------------------------------------------------ // ------------------------------------------------------------------------------------
...@@ -335,20 +343,33 @@ namespace dlib ...@@ -335,20 +343,33 @@ namespace dlib
unsigned char size; unsigned char size;
item = 0; item = 0;
in.read(reinterpret_cast<char*>(&size),1);
std::streambuf* sbuf = in.rdbuf();
int ch = sbuf->sbumpc();
if (ch != EOF)
{
size = static_cast<unsigned char>(ch);
}
else
{
in.setstate(std::ios::badbit);
return true;
}
// mask out the 3 reserved bits // mask out the 3 reserved bits
size &= 0x8F; size &= 0x8F;
// check if an error occurred // check if an error occurred
if (!in || size > sizeof(T)) if (size > sizeof(T))
return true; return true;
in.read(reinterpret_cast<char*>(&buf),size); if (sbuf->sgetn(reinterpret_cast<char*>(&buf),size) != size)
{
// check if the serialized object is too big to fit into something of type T. in.setstate(std::ios::badbit);
// or if there was an error reading from in.
if (!in)
return true; return true;
}
for (unsigned char i = size-1; true; --i) for (unsigned char i = size-1; true; --i)
{ {
...@@ -371,11 +392,40 @@ namespace dlib ...@@ -371,11 +392,40 @@ namespace dlib
inline void deserialize (T& item, std::istream& in) \ inline void deserialize (T& item, std::istream& in) \
{ if (ser_helper::unpack_int(item,in)) throw serialization_error("Error deserializing object of type " + std::string(#T)); } { if (ser_helper::unpack_int(item,in)) throw serialization_error("Error deserializing object of type " + std::string(#T)); }
template <typename T>
inline bool pack_byte (
const T& ch,
std::ostream& out
)
{
std::streambuf* sbuf = out.rdbuf();
return (sbuf->sputc((char)ch) == EOF);
}
template <typename T>
inline bool unpack_byte (
T& ch,
std::istream& in
)
{
std::streambuf* sbuf = in.rdbuf();
int temp = sbuf->sbumpc();
if (temp != EOF)
{
ch = static_cast<T>(temp);
return false;
}
else
{
return true;
}
}
#define USE_DEFAULT_BYTE_SERIALIZATION_FOR(T) \ #define USE_DEFAULT_BYTE_SERIALIZATION_FOR(T) \
inline void serialize (const T& item,std::ostream& out) \ inline void serialize (const T& item,std::ostream& out) \
{ out.write(reinterpret_cast<const char*>(&item),1); if (!out) throw serialization_error("Error serializing object of type " + std::string(#T)); } \ { if (pack_byte(item,out)) throw serialization_error("Error serializing object of type " + std::string(#T)); } \
inline void deserialize (T& item, std::istream& in) \ inline void deserialize (T& item, std::istream& in) \
{ in.read(reinterpret_cast<char*>(&item),1); if (!in) throw serialization_error("Error deserializing object of type " + std::string(#T)); } { if (unpack_byte(item,in)) throw serialization_error("Error deserializing object of type " + std::string(#T)); }
// ---------------------------------------------------------------------------------------- // ----------------------------------------------------------------------------------------
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment