X-Git-Url: http://git.tdb.fi/?p=libs%2Fnet.git;a=blobdiff_plain;f=source%2Fnet%2Fprotocol.cpp;h=52678af2d9009cf9279d88ce3e1856d0fee1be9b;hp=d56ac77a09f70eff7137da98beef2eb7112d8978;hb=HEAD;hpb=debe1004676d5431e571d9c4361072661dcc88c4 diff --git a/source/net/protocol.cpp b/source/net/protocol.cpp index d56ac77..5e82a80 100644 --- a/source/net/protocol.cpp +++ b/source/net/protocol.cpp @@ -1,232 +1,184 @@ +#include "protocol.h" #include #include #include +#include #include -#include "protocol.h" using namespace std; -namespace { +namespace Msp { +namespace Net { -using Msp::Net::buffer_error; +Protocol::Protocol(): + header_def(0) +{ + PacketDefBuilder>(*this, header_def, Serializer()) + .fields(&PacketHeader::type, &PacketHeader::length); +} -template -class Assembler +unsigned Protocol::get_next_packet_class_id() { -public: - static char *assemble(const T &v, char *, char *); - static const char *disassemble(T &, const char *, const char *); -}; + static unsigned next_id = 1; + return next_id++; +} -template -class Assembler > +void Protocol::add_packet(unique_ptr pdef) { -public: - static char *assemble(const vector &v, char *, char *); - static const char *disassemble(vector &, const char *, const char *); -}; + unique_ptr &ptr = packet_class_defs[pdef->get_class_id()]; + if(ptr) + packet_id_defs.erase(ptr->get_id()); + ptr = move(pdef); + if(unsigned id = ptr->get_id()) + packet_id_defs[id] = ptr.get(); +} -template -char *Assembler::assemble(const T &v, char *data, char *end) +const Protocol::PacketDefBase &Protocol::get_packet_by_class_id(unsigned id) const { - // XXX Assumes little-endian - const char *ptr = reinterpret_cast(&v)+sizeof(T); - for(unsigned i=0; i -char *Assembler::assemble(const string &v, char *data, char *end) +const Protocol::PacketDefBase &Protocol::get_packet_by_id(unsigned id) const { - data = Assembler::assemble(v.size(), data, end); - if(end-data(v.size())) - throw buffer_error("overflow"); - memcpy(data, v.data(), v.size()); - return data+v.size(); + return *get_item(packet_id_defs, id); } -template -char *Assembler >::assemble(const vector &v, char *data, char *end) +unsigned Protocol::get_max_packet_id() const { - data = Assembler::assemble(v.size(), data, end); - for(typename vector::const_iterator i=v.begin(); i!=v.end(); ++i) - data = Assembler::assemble(*i, data, end); - return data; + if(packet_id_defs.empty()) + return 0; + return prev(packet_id_defs.end())->first; } -template -const char *Assembler::disassemble(T &v, const char *data, const char *end) +size_t Protocol::dispatch(ReceiverBase &rcv, const char *buf, size_t size, unsigned base_id) const { - char *ptr = reinterpret_cast(&v)+sizeof(T); - for(unsigned i=0; isize) + throw bad_packet("truncated"); + const PacketDefBase &pdef = get_packet_by_id(header.type-base_id); + if(DynamicReceiver *drcv = dynamic_cast(&rcv)) { - if(data==end) - throw buffer_error("underflow"); - *--ptr = *data++; + Variant pkt; + ptr = pdef.deserialize(pkt, ptr, ptr+header.length); + drcv->receive(pdef.get_id(), pkt); } - return data; + else + ptr = pdef.dispatch(rcv, ptr, ptr+header.length); + return ptr-buf; } -template<> -const char *Assembler::disassemble(string &v, const char *data, const char *end) +bool Protocol::get_packet_header(PacketHeader &header, const char *buf, size_t size) const { - unsigned short size; - data = Assembler::disassemble(size, data, end); - if(end-data -const char *Assembler >::disassemble(vector &v, const char *data, const char *end) +size_t Protocol::get_packet_size(const char *buf, size_t size) const +{ + PacketHeader header; + return (get_packet_header(header, buf, size) ? header.length : 0); +} + +uint64_t Protocol::get_hash() const { - /* We assume that the vector is in pristine state - this holds because the - only code path leading here is from PacketDef

::disassemble, which creates - a new packet. */ - unsigned short size; - data = Assembler::disassemble(size, data, end); - for(unsigned i=0; i(packet_id_defs.size()); + for(auto &kvp: packet_id_defs) { - T u; - data = Assembler::disassemble(u, data, end); - v.push_back(u); + hash_update<64>(result, kvp.first); + hash_update<64>(result, kvp.second->get_hash()); } - return data; + return result; } -} -namespace Msp { -namespace Net { +/* TODO These assumes the machine is little-endian; are there any relevant +big-endian platforms these days? */ +template +char *Protocol::BasicSerializer::serialize(const T &value, char *buf, char *end) const +{ + if(end-buf(sizeof(T))) + throw buffer_error("overflow"); -Protocol::Protocol(unsigned npi): - next_packet_id(npi) -{ } + const char *ptr = reinterpret_cast(&value)+sizeof(T); + for(size_t i=0; i::iterator i=packet_class_defs.begin(); i!=packet_class_defs.end(); ++i) - delete i->second; + return buf; } -void Protocol::add_packet(PacketDefBase &pdef) +template +const char *Protocol::BasicSerializer::deserialize(T &value, const char *buf, const char *end) const { - PacketDefBase *&ptr = packet_class_defs[pdef.get_class_id()]; - if(ptr) - delete ptr; - ptr = &pdef; - packet_id_defs[pdef.get_id()] = &pdef; -} + if(end-buf(sizeof(T))) + throw buffer_error("underflow"); -const Protocol::PacketDefBase &Protocol::get_packet_by_class(unsigned id) const -{ - return *get_item(packet_class_defs, id); -} + char *ptr = reinterpret_cast(&value)+sizeof(T); + for(size_t i=0; i(data); - unsigned id = (udata[0]<<8)+udata[1]; - unsigned psz = (udata[2]<<8)+udata[3]; - if(psz>size) - throw bad_packet("truncated"); - const PacketDefBase &pdef = get_packet_by_id(id); - const char *ptr = pdef.disassemble(rcv, data+4, data+psz); - return ptr-data; -} +template char *Protocol::BasicSerializer::serialize(const bool &, char *, char *) const; +template char *Protocol::BasicSerializer::serialize(const int8_t &, char *, char *) const; +template char *Protocol::BasicSerializer::serialize(const int16_t &, char *, char *) const; +template char *Protocol::BasicSerializer::serialize(const int32_t &, char *, char *) const; +template char *Protocol::BasicSerializer::serialize(const int64_t &, char *, char *) const; +template char *Protocol::BasicSerializer::serialize(const uint8_t &, char *, char *) const; +template char *Protocol::BasicSerializer::serialize(const uint16_t &, char *, char *) const; +template char *Protocol::BasicSerializer::serialize(const uint32_t &, char *, char *) const; +template char *Protocol::BasicSerializer::serialize(const uint64_t &, char *, char *) const; +template char *Protocol::BasicSerializer::serialize(const float &, char *, char *) const; +template char *Protocol::BasicSerializer::serialize(const double &, char *, char *) const; +template const char *Protocol::BasicSerializer::deserialize(bool &, const char *, const char *) const; +template const char *Protocol::BasicSerializer::deserialize(int8_t &, const char *, const char *) const; +template const char *Protocol::BasicSerializer::deserialize(int16_t &, const char *, const char *) const; +template const char *Protocol::BasicSerializer::deserialize(int32_t &, const char *, const char *) const; +template const char *Protocol::BasicSerializer::deserialize(int64_t &, const char *, const char *) const; +template const char *Protocol::BasicSerializer::deserialize(uint8_t &, const char *, const char *) const; +template const char *Protocol::BasicSerializer::deserialize(uint16_t &, const char *, const char *) const; +template const char *Protocol::BasicSerializer::deserialize(uint32_t &, const char *, const char *) const; +template const char *Protocol::BasicSerializer::deserialize(uint64_t &, const char *, const char *) const; +template const char *Protocol::BasicSerializer::deserialize(float &, const char *, const char *) const; +template const char *Protocol::BasicSerializer::deserialize(double &, const char *, const char *) const; + + +Protocol::StringSerializer::StringSerializer(const Protocol &p): + length_serializer(p) +{ } -unsigned Protocol::get_packet_size(const char *data, unsigned size) const +char *Protocol::StringSerializer::serialize(const string &str, char *buf, char *end) const { - if(size<4) - return 0; - const unsigned char *udata = reinterpret_cast(data); - return (udata[2]<<8)+udata[3]; + buf = length_serializer.serialize(str.size(), buf, end); + if(end-buf(str.size())) + throw buffer_error("overflow"); + copy(str.begin(), str.end(), buf); + return buf+str.size(); } -unsigned Protocol::get_hash() const +const char *Protocol::StringSerializer::deserialize(string &str, const char *buf, const char *end) const { - // TODO - return 123; + uint16_t length; + buf = length_serializer.deserialize(length, buf, end); + if(end-buf(length)) + throw buffer_error("underflow"); + str.assign(buf, buf+length); + return buf+length; } -void Protocol::assemble_header(char *buf, unsigned id, unsigned size) -{ - buf[0] = (id>>8)&0xFF; - buf[1] = id&0xFF; - buf[2] = (size>>8)&0xFF; - buf[3] = size&0xFF; -} -template -char *Protocol::assemble_field(const T &v, char *d, char *e) -{ return Assembler::assemble(v, d, e); } - -template char *Protocol::assemble_field<>(const char &v, char *d, char *e); -template char *Protocol::assemble_field<>(const signed char &v, char *d, char *e); -template char *Protocol::assemble_field<>(const unsigned char &v, char *d, char *e); -template char *Protocol::assemble_field<>(const short &v, char *d, char *e); -template char *Protocol::assemble_field<>(const unsigned short &v, char *d, char *e); -template char *Protocol::assemble_field<>(const int &v, char *d, char *e); -template char *Protocol::assemble_field<>(const unsigned &v, char *d, char *e); -template char *Protocol::assemble_field<>(const long &v, char *d, char *e); -template char *Protocol::assemble_field<>(const unsigned long &v, char *d, char *e); -template char *Protocol::assemble_field<>(const float &v, char *d, char *e); -template char *Protocol::assemble_field<>(const double &v, char *d, char *e); -template char *Protocol::assemble_field<>(const string &v, char *d, char *e); -template char *Protocol::assemble_field<>(const vector &v, char *d, char *e); -template char *Protocol::assemble_field<>(const vector &v, char *d, char *e); -template char *Protocol::assemble_field<>(const vector &v, char *d, char *e); -template char *Protocol::assemble_field<>(const vector &v, char *d, char *e); -template char *Protocol::assemble_field<>(const vector &v, char *d, char *e); -template char *Protocol::assemble_field<>(const vector &v, char *d, char *e); -template char *Protocol::assemble_field<>(const vector &v, char *d, char *e); -template char *Protocol::assemble_field<>(const vector &v, char *d, char *e); -template char *Protocol::assemble_field<>(const vector &v, char *d, char *e); -template char *Protocol::assemble_field<>(const vector &v, char *d, char *e); -template char *Protocol::assemble_field<>(const vector &v, char *d, char *e); -template char *Protocol::assemble_field<>(const vector &v, char *d, char *e); +Protocol::PacketDefBase::PacketDefBase(unsigned i): + id(i) +{ } -template -const char *Protocol::disassemble_field(T &v, const char *d, const char *e) -{ return Assembler::disassemble(v, d, e); } - -template const char *Protocol::disassemble_field<>(char &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(signed char &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(unsigned char &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(short &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(unsigned short &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(int &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(unsigned &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(long &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(unsigned long &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(float &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(double &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(string &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(vector &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(vector &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(vector &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(vector &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(vector &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(vector &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(vector &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(vector &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(vector &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(vector &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(vector &v, const char *d, const char *e); -template const char *Protocol::disassemble_field<>(vector &v, const char *d, const char *e); - -unsigned Protocol::PacketDefBase::next_class_id = 1; + +Protocol::PacketHeader::PacketHeader(uint16_t t, uint16_t l): + type(t), + length(l) +{ } } // namespace Net } // namespace Msp