/*** This file is part of snapcast Copyright (C) 2014-2023 Johannes Pohl This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . ***/ #ifndef MESSAGE_MESSAGE_HPP #define MESSAGE_MESSAGE_HPP // local headers #include "common/endian.hpp" #include "common/time_defs.hpp" // standard headers #include #include #include #include #ifndef WINDOWS #include #endif #include #include /* template > class vectorwrapbuf : public std::basic_streambuf { public: vectorwrapbuf(std::vector &vec) { this->setg(vec.data(), vec.data(), vec.data() + vec.size()); } }; */ struct membuf : public std::basic_streambuf { membuf(char* begin, char* end) { this->setg(begin, begin, end); } }; enum class message_type : uint16_t { kBase = 0, kCodecHeader = 1, kWireChunk = 2, kServerSettings = 3, kTime = 4, kHello = 5, // kStreamTags = 6, kClientInfo = 7, kFirst = kBase, kLast = kClientInfo }; static std::ostream& operator<<(std::ostream& os, const message_type& msg_type) { switch (msg_type) { case message_type::kBase: os << "Base"; break; case message_type::kCodecHeader: os << "CodecHeader"; break; case message_type::kWireChunk: os << "WireChunk"; break; case message_type::kServerSettings: os << "ServerSettings"; break; case message_type::kTime: os << "Time"; break; case message_type::kHello: os << "Hello"; break; case message_type::kClientInfo: os << "ClientInfo"; break; default: os << "Unknown"; } return os; } struct tv { tv() { timeval t; chronos::steadytimeofday(&t); sec = t.tv_sec; usec = t.tv_usec; } tv(timeval tv) : sec(tv.tv_sec), usec(tv.tv_usec){}; tv(int32_t _sec, int32_t _usec) : sec(_sec), usec(_usec){}; int32_t sec; int32_t usec; tv operator+(const tv& other) const { tv result(*this); result.sec += other.sec; result.usec += other.usec; if (result.usec > 1000000) { result.sec += result.usec / 1000000; result.usec %= 1000000; } return result; } tv operator-(const tv& other) const { tv result(*this); result.sec -= other.sec; result.usec -= other.usec; while (result.usec < 0) { result.sec -= 1; result.usec += 1000000; } return result; } }; namespace msg { const size_t max_size = 1000000; struct BaseMessage; using message_ptr = std::shared_ptr; struct BaseMessage { BaseMessage() : BaseMessage(message_type::kBase) { } BaseMessage(message_type type_) : type(type_), id(0), refersTo(0), size(0) { } virtual ~BaseMessage() = default; virtual void read(std::istream& stream) { readVal(stream, type); readVal(stream, id); readVal(stream, refersTo); readVal(stream, sent.sec); readVal(stream, sent.usec); readVal(stream, received.sec); readVal(stream, received.usec); readVal(stream, size); } void deserialize(char* payload) { membuf databuf(payload, payload + BaseMessage::getSize()); std::istream is(&databuf); read(is); } void deserialize(const BaseMessage& baseMessage, char* payload) { type = baseMessage.type; id = baseMessage.id; refersTo = baseMessage.refersTo; sent = baseMessage.sent; received = baseMessage.received; size = baseMessage.size; membuf databuf(payload, payload + size); std::istream is(&databuf); read(is); } virtual void serialize(std::ostream& stream) const { writeVal(stream, type); writeVal(stream, id); writeVal(stream, refersTo); writeVal(stream, sent.sec); writeVal(stream, sent.usec); writeVal(stream, received.sec); writeVal(stream, received.usec); size = getSize(); writeVal(stream, size); doserialize(stream); } virtual uint32_t getSize() const { return 3 * sizeof(uint16_t) + 2 * sizeof(tv) + sizeof(uint32_t); }; message_type type; mutable uint16_t id; uint16_t refersTo; tv received; mutable tv sent; mutable uint32_t size; protected: void writeVal(std::ostream& stream, const bool& val) const { char c = val ? 1 : 0; writeVal(stream, c); } void writeVal(std::ostream& stream, const char& val) const { stream.write(reinterpret_cast(&val), sizeof(char)); } void writeVal(std::ostream& stream, const uint16_t& val) const { uint16_t v = SWAP_16(val); stream.write(reinterpret_cast(&v), sizeof(uint16_t)); } void writeVal(std::ostream& stream, const message_type& val) const { uint16_t v = SWAP_16(static_cast(val)); stream.write(reinterpret_cast(&v), sizeof(uint16_t)); } void writeVal(std::ostream& stream, const int16_t& val) const { uint16_t v = SWAP_16(val); stream.write(reinterpret_cast(&v), sizeof(int16_t)); } void writeVal(std::ostream& stream, const uint32_t& val) const { uint32_t v = SWAP_32(val); stream.write(reinterpret_cast(&v), sizeof(uint32_t)); } void writeVal(std::ostream& stream, const int32_t& val) const { uint32_t v = SWAP_32(val); stream.write(reinterpret_cast(&v), sizeof(int32_t)); } void writeVal(std::ostream& stream, const char* payload, const uint32_t& size) const { writeVal(stream, size); stream.write(payload, size); } void writeVal(std::ostream& stream, const std::string& val) const { auto len = static_cast(val.size()); writeVal(stream, val.c_str(), len); } void readVal(std::istream& stream, bool& val) const { char c; readVal(stream, c); val = (c != 0); } void readVal(std::istream& stream, char& val) const { stream.read(reinterpret_cast(&val), sizeof(char)); } void readVal(std::istream& stream, uint16_t& val) const { stream.read(reinterpret_cast(&val), sizeof(uint16_t)); // cppcheck-suppress selfAssignment val = SWAP_16(val); } void readVal(std::istream& stream, message_type& val) const { stream.read(reinterpret_cast(&val), sizeof(uint16_t)); val = static_cast(SWAP_16(static_cast(val))); } void readVal(std::istream& stream, int16_t& val) const { stream.read(reinterpret_cast(&val), sizeof(int16_t)); // cppcheck-suppress selfAssignment val = SWAP_16(val); } void readVal(std::istream& stream, uint32_t& val) const { stream.read(reinterpret_cast(&val), sizeof(uint32_t)); // cppcheck-suppress selfAssignment val = SWAP_32(val); } void readVal(std::istream& stream, int32_t& val) const { stream.read(reinterpret_cast(&val), sizeof(int32_t)); // cppcheck-suppress selfAssignment val = SWAP_32(val); } void readVal(std::istream& stream, char** payload, uint32_t& size) const { readVal(stream, size); *payload = (char*)realloc(*payload, size); stream.read(*payload, size); } void readVal(std::istream& stream, std::string& val) const { uint32_t len; readVal(stream, len); val.resize(len); stream.read(&val[0], len); } virtual void doserialize(std::ostream& /*stream*/) const {}; }; } // namespace msg #endif