mirror of
https://github.com/fraillt/bitsery.git
synced 2026-06-08 08:13:56 +00:00
220 lines
8.0 KiB
C++
220 lines
8.0 KiB
C++
//MIT License
|
|
//
|
|
//Copyright (c) 2017 Mindaugas Vinkelis
|
|
//
|
|
//Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
//of this software and associated documentation files (the "Software"), to deal
|
|
//in the Software without restriction, including without limitation the rights
|
|
//to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
//copies of the Software, and to permit persons to whom the Software is
|
|
//furnished to do so, subject to the following conditions:
|
|
//
|
|
//The above copyright notice and this permission notice shall be included in all
|
|
//copies or substantial portions of the Software.
|
|
//
|
|
//THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
//IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
//FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
//AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
//LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
//OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
//SOFTWARE.
|
|
|
|
|
|
|
|
#ifndef BITSERY_BUFFER_WRITER_H
|
|
#define BITSERY_BUFFER_WRITER_H
|
|
|
|
#include "common.h"
|
|
#include <cassert>
|
|
#include <algorithm>
|
|
#include <iterator>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
namespace bitsery {
|
|
|
|
struct MeasureSize {
|
|
|
|
template<size_t SIZE, typename T>
|
|
void writeBytes(const T &) {
|
|
static_assert(std::is_integral<T>(), "");
|
|
static_assert(sizeof(T) == SIZE, "");
|
|
_bitsCount += details::BITS_SIZE<T>;
|
|
}
|
|
|
|
template<typename T>
|
|
void writeBits(const T &, size_t bitsCount) {
|
|
static_assert(std::is_integral<T>() && std::is_unsigned<T>(), "");
|
|
assert(bitsCount <= details::BITS_SIZE<T>);
|
|
_bitsCount += bitsCount;
|
|
}
|
|
|
|
template<size_t SIZE, typename T>
|
|
void writeBuffer(const T *, size_t count) {
|
|
static_assert(std::is_integral<T>(), "");
|
|
static_assert(sizeof(T) == SIZE, "");
|
|
_bitsCount += details::BITS_SIZE<T> * count;
|
|
}
|
|
|
|
void align() {
|
|
_bitsCount += 8 - (_bitsCount % 8);
|
|
}
|
|
|
|
void flush() {
|
|
|
|
}
|
|
|
|
//get size in bytes
|
|
size_t getSize() const {
|
|
return _bitsCount / 8;
|
|
}
|
|
|
|
private:
|
|
size_t _bitsCount{};
|
|
|
|
};
|
|
|
|
template<typename Config>
|
|
struct BasicBufferWriter {
|
|
using ValueType = typename Config::BufferValueType;
|
|
using ScratchType = typename Config::BufferScrathType;
|
|
|
|
explicit BasicBufferWriter(std::vector<ValueType> &buffer) : _outIt{std::back_inserter(buffer)} {
|
|
static_assert(std::is_unsigned<ValueType>(), "Config::BufferValueType must be unsigned");
|
|
static_assert(std::is_unsigned<ScratchType>(), "Config::BufferScrathType must be unsigned");
|
|
static_assert(sizeof(ValueType) * 2 == sizeof(ScratchType),
|
|
"ScratchType must be 2x bigger than value type");
|
|
static_assert(sizeof(ValueType) == 1, "currently only supported BufferValueType is 1 byte");
|
|
}
|
|
|
|
BasicBufferWriter(const BasicBufferWriter &) = delete;
|
|
|
|
BasicBufferWriter &operator=(const BasicBufferWriter &) = delete;
|
|
|
|
BasicBufferWriter(BasicBufferWriter &&) noexcept = default;
|
|
|
|
BasicBufferWriter &operator=(BasicBufferWriter &&) noexcept = default;
|
|
|
|
~BasicBufferWriter() noexcept = default;
|
|
|
|
template<size_t SIZE, typename T>
|
|
void writeBytes(const T &v) {
|
|
static_assert(std::is_integral<T>(), "");
|
|
static_assert(sizeof(T) == SIZE, "");
|
|
|
|
if (!_scratchBits) {
|
|
directWrite(&v, 1);
|
|
} else {
|
|
using UT = typename std::make_unsigned<T>::type;
|
|
writeBits(reinterpret_cast<const UT &>(v), details::BITS_SIZE<T>);
|
|
}
|
|
}
|
|
|
|
template<size_t SIZE, typename T>
|
|
void writeBuffer(const T *buf, size_t count) {
|
|
static_assert(std::is_integral<T>(), "");
|
|
static_assert(sizeof(T) == SIZE, "");
|
|
if (!_scratchBits) {
|
|
directWrite(buf, count);
|
|
} else {
|
|
using UT = typename std::make_unsigned<T>::type;
|
|
//todo improve implementation
|
|
const auto end = buf + count;
|
|
for (auto it = buf; it != end; ++it)
|
|
writeBits(reinterpret_cast<const UT &>(*it), details::BITS_SIZE<T>);
|
|
}
|
|
}
|
|
|
|
template<typename T>
|
|
void writeBits(const T &v, size_t bitsCount) {
|
|
static_assert(std::is_integral<T>() && std::is_unsigned<T>(), "");
|
|
assert(0 < bitsCount && bitsCount <= details::BITS_SIZE<T>);
|
|
assert(v <= ((1ULL << bitsCount) - 1));
|
|
writeBitsInternal(v, bitsCount);
|
|
}
|
|
|
|
void align() {
|
|
if (_scratchBits)
|
|
writeBitsInternal(ValueType{}, details::BITS_SIZE<ValueType> - _scratchBits);
|
|
}
|
|
|
|
void flush() {
|
|
if (_scratchBits) {
|
|
auto tmp = static_cast<ValueType>( _scratch & _MASK );
|
|
directWrite(&tmp, 1);
|
|
_scratch >>= _scratchBits;
|
|
_scratchBits -= _scratchBits;
|
|
}
|
|
}
|
|
|
|
|
|
private:
|
|
|
|
template<typename T>
|
|
void directWrite(T &&v, size_t count) {
|
|
_directWriteSwapTag(std::forward<T>(v), count, std::integral_constant<bool,
|
|
Config::NetworkEndianness != details::getSystemEndianness()>{});
|
|
}
|
|
|
|
template<typename T>
|
|
void _directWriteSwapTag(const T *v, size_t count, std::true_type) {
|
|
std::for_each(v, std::next(v, count), [this](const T &v) {
|
|
const auto res = details::swap(v);
|
|
std::copy_n(reinterpret_cast<const ValueType *>(&res), sizeof(T), _outIt);
|
|
});
|
|
}
|
|
|
|
template<typename T>
|
|
void _directWriteSwapTag(const T *v, size_t count, std::false_type) {
|
|
std::copy_n(reinterpret_cast<const ValueType *>(v), count * sizeof(T), _outIt);
|
|
}
|
|
|
|
template<typename T>
|
|
void writeBitsInternal(const T &v, size_t size) {
|
|
constexpr size_t valueSize = details::BITS_SIZE<ValueType>;
|
|
auto value = v;
|
|
auto bitsLeft = size;
|
|
while (bitsLeft > 0) {
|
|
auto bits = std::min(bitsLeft, valueSize);
|
|
_scratch |= static_cast<ScratchType>( value ) << _scratchBits;
|
|
_scratchBits += bits;
|
|
if (_scratchBits >= valueSize) {
|
|
auto tmp = static_cast<ValueType>(_scratch & _MASK);
|
|
directWrite(&tmp, 1);
|
|
_scratch >>= valueSize;
|
|
_scratchBits -= valueSize;
|
|
|
|
value >>= valueSize;
|
|
}
|
|
bitsLeft -= bits;
|
|
}
|
|
}
|
|
|
|
//overload for ValueType, for better performance
|
|
void writeBitsInternal(const ValueType &v, size_t size) {
|
|
if (size > 0) {
|
|
_scratch |= static_cast<ScratchType>( v ) << _scratchBits;
|
|
_scratchBits += size;
|
|
if (_scratchBits >= details::BITS_SIZE<ValueType>) {
|
|
auto tmp = static_cast<ValueType>(_scratch & _MASK);
|
|
directWrite(&tmp, 1);
|
|
_scratch >>= details::BITS_SIZE<ValueType>;
|
|
_scratchBits -= details::BITS_SIZE<ValueType>;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
const ValueType _MASK = std::numeric_limits<ValueType>::max();
|
|
std::back_insert_iterator<std::vector<ValueType>> _outIt;
|
|
ScratchType _scratch{};
|
|
size_t _scratchBits{};
|
|
};
|
|
|
|
//helper type
|
|
using BufferWriter = BasicBufferWriter<DefaultConfig>;
|
|
}
|
|
|
|
#endif //BITSERY_BUFFER_WRITER_H
|