2#include <ossia/detail/pod_vector.hpp>
3#include <ossia/network/sockets/writers.hpp>
5#include <boost/asio/buffer.hpp>
6#include <boost/asio/error.hpp>
7#include <boost/asio/write.hpp>
12template <
typename Socket>
16 alignas(64) uint8_t m_readbuf[4096];
17 ossia::pod_vector<uint8_t> m_encoded;
18 ossia::pod_vector<char> m_decoded;
20 explicit cobs_decoder(Socket& socket)
23 m_encoded.reserve(1024);
24 m_decoded.reserve(1024);
30 socket.async_read_some(
31 boost::asio::buffer(m_readbuf),
32 [
this, f = std::move(f)](boost::system::error_code ec, std::size_t sz)
mutable {
33 if(!f.validate_stream(ec))
37 process_bytes(f, m_readbuf, sz);
39 receive(std::move(f));
44 void process_bytes(
const F& f,
const uint8_t* data, std::size_t sz)
46 const uint8_t* ptr = data;
47 const uint8_t* end = data + sz;
52 const uint8_t* delim = ptr;
53 while(delim < end && *delim != 0x00)
58 m_encoded.insert(m_encoded.end(), ptr, delim);
66 if(!m_encoded.empty())
69 f((
const unsigned char*)m_decoded.data(), m_decoded.size());
79 const uint8_t* ptr = m_encoded.data();
80 const uint8_t* end = ptr + m_encoded.size();
84 uint8_t code = *ptr++;
88 uint8_t count = code - 1;
92 for(uint8_t j = 0; j < count; ++j)
93 m_decoded.push_back(
static_cast<char>(*ptr++));
96 if(code < 0xFF && ptr < end)
97 m_decoded.push_back(0);
100 return m_decoded.size() > 0;
104template <
typename Socket>
108 ossia::pod_vector<uint8_t> m_buf;
110 void write(
const char* data, std::size_t sz)
113 m_buf.reserve(sz + sz / 254 + 3);
115 const auto* src =
reinterpret_cast<const uint8_t*
>(data);
117 std::size_t code_idx = m_buf.size();
121 for(std::size_t i = 0; i < sz; ++i)
125 m_buf[code_idx] = code;
126 code_idx = m_buf.size();
132 m_buf.push_back(src[i]);
136 m_buf[code_idx] = code;
137 code_idx = m_buf.size();
144 m_buf[code_idx] = code;
145 m_buf.push_back(0x00);
147 this->do_write(socket, boost::asio::buffer(m_buf.data(), m_buf.size()));
150 template <
typename T>
151 void do_write(T& sock,
const boost::asio::const_buffer& buf)
153 boost::asio::write(sock, buf);
156 template <
typename T>
157 void do_write(multi_socket_writer<T>& sock,
const boost::asio::const_buffer& buf)
165 template <
typename Socket>
166 using encoder = cobs_encoder<Socket>;
167 template <
typename Socket>
168 using decoder = cobs_decoder<Socket>;