osb/source/core/StarVlqEncoding.hpp

107 lines
2.9 KiB
C++
Raw Normal View History

2023-06-20 14:33:09 +10:00
#ifndef STAR_VLQ_ENCODING_HPP
#define STAR_VLQ_ENCODING_HPP
#include "StarMemory.hpp"
namespace Star {
// Write an unsigned integer as a VLQ (Variable Length Quantity). Writes the
// integer in 7 byte chunks, with the 8th bit of each octet indicates whether
// another chunk follows. Endianness independent, as the chunks are always
// written most significant first. Returns number of octet written (writes a
// maximum of a 64 bit integer, so a maximum of 10)
template <typename OutputIterator>
size_t writeVlqU(uint64_t x, OutputIterator out) {
size_t i;
for (i = 9; i > 0; --i) {
if (x & ((uint64_t)(127) << (i * 7)))
break;
}
for (size_t j = 0; j < i; ++j)
*out++ = (uint8_t)((x >> ((i - j) * 7)) & 127) | 128;
*out++ = (uint8_t)(x & 127);
return i + 1;
}
inline size_t vlqUSize(uint64_t x) {
size_t i;
for (i = 9; i > 0; --i) {
if (x & ((uint64_t)(127) << (i * 7)))
break;
}
return i + 1;
}
// Read a VLQ (Variable Length Quantity) encoded unsigned integer. Returns
// number of bytes read. Reads a *maximum of 10 bytes*, cannot read a larger
// than 64 bit integer! If no end marker is found within 'maxBytes' or 10
// bytes, whichever is smaller, then will return NPos to signal error.
template <typename InputIterator>
size_t readVlqU(uint64_t& x, InputIterator in, size_t maxBytes = 10) {
x = 0;
for (size_t i = 0; i < min<size_t>(10, maxBytes); ++i) {
uint8_t oct = *in++;
x = (x << 7) | (uint64_t)(oct & 127);
if (!(oct & 128))
return i + 1;
}
return NPos;
}
// Write a VLQ (Variable Length Quantity) encoded signed integer. Encoded by
// making the sign bit the least significant bit in the integer. Returns
// number of bytes written.
template <typename OutputIterator>
size_t writeVlqI(int64_t v, OutputIterator out) {
uint64_t target;
// If negative, then add 1 to properly encode -2^63
if (v < 0)
target = ((-(v + 1)) << 1) | 1;
else
target = v << 1;
return writeVlqU(target, out);
}
inline size_t vlqISize(int64_t v) {
uint64_t target;
// If negative, then add 1 to properly encode -2^63
if (v < 0)
target = ((-(v + 1)) << 1) | 1;
else
target = v << 1;
return vlqUSize(target);
}
// Read a VLQ (Variable Length Quantity) encoded signed integer. Returns
// number of bytes read. Reads a *maximum of 10 bytes*, cannot read a larger
// than 64 bit integer! If no end marker is found within 'maxBytes' or 10
// bytes, whichever is smaller, then will return NPos to signal error.
template <typename InputIterator>
size_t readVlqI(int64_t& v, InputIterator in, size_t maxBytes = 10) {
uint64_t source;
size_t bytes = readVlqU(source, in, maxBytes);
if (bytes == NPos)
return NPos;
bool negative = (source & 1);
// If negative, then need to undo the +1 transformation to encode -2^63
if (negative)
v = -(int64_t)(source >> 1) - 1;
else
v = (int64_t)(source >> 1);
return bytes;
}
}
#endif