/* * Copyright (c) 2017, Matias Fontanini * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following disclaimer * in the documentation and/or other materials provided with the * distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * */ #ifndef TINS_MEMORY_HELPERS_H #define TINS_MEMORY_HELPERS_H #include #include #include #include "exceptions.h" #include "endianness.h" namespace Tins { class IPv4Address; class IPv6Address; template class HWAddress; /** * \cond */ namespace Memory { inline void read_data(const uint8_t* buffer, uint8_t* output_buffer, size_t size) { std::memcpy(output_buffer, buffer, size); } template void read_value(const uint8_t* buffer, T& value) { std::memcpy(&value, buffer, sizeof(value)); } inline void write_data(uint8_t* buffer, const uint8_t* ptr, size_t size) { std::memcpy(buffer, ptr, size); } template void write_value(uint8_t* buffer, const T& value) { std::memcpy(buffer, &value, sizeof(value)); } class InputMemoryStream { public: InputMemoryStream(const uint8_t* buffer, size_t total_sz); InputMemoryStream(const std::vector& data); template T read() { T output; read(output); return output; } template T read_le() { return Endian::le_to_host(read()); } template T read_be() { return Endian::be_to_host(read()); } template void read(T& value) { if (!can_read(sizeof(value))) { throw malformed_packet(); } read_value(buffer_, value); skip(sizeof(value)); } void skip(size_t size); bool can_read(size_t byte_count) const; void read(std::vector& value, size_t count); void read(HWAddress<6>& address); void read(IPv4Address& address); void read(IPv6Address& address); void read(void* output_buffer, size_t output_buffer_size); const uint8_t* pointer() const; size_t size() const; void size(size_t new_size); operator bool() const; private: const uint8_t* buffer_; size_t size_; }; class OutputMemoryStream { public: OutputMemoryStream(uint8_t* buffer, size_t total_sz); OutputMemoryStream(std::vector& buffer); template void write(const T& value) { if (TINS_UNLIKELY(size_ < sizeof(value))) { throw serialization_error(); } write_value(buffer_, value); skip(sizeof(value)); } template void write_be(const T& value) { write(Endian::host_to_be(value)); } template void write_le(const T& value) { write(Endian::host_to_le(value)); } template void write(ForwardIterator start, ForwardIterator end) { const size_t length = std::distance(start, end); if (TINS_UNLIKELY(size_ < length)) { throw serialization_error(); } std::memcpy(buffer_, &*start, length); skip(length); } void skip(size_t size); void write(const uint8_t* ptr, size_t length); void write(const HWAddress<6>& address); void write(const IPv4Address& address); void write(const IPv6Address& address); void fill(size_t size, uint8_t value); uint8_t* pointer(); size_t size() const; private: uint8_t* buffer_; size_t size_; }; /** * \endcond */ } // Memory } // Tins #endif // TINS_MEMORY_HELPERS_H