diff options
author | Alexei Frolov <frolv@google.com> | 2019-11-27 14:38:39 -0800 |
---|---|---|
committer | Alexei Frolov <frolv@google.com> | 2019-12-02 18:39:29 +0000 |
commit | 82d3cb35d37f9f4011a47f0b33acf55fd2e3f8c7 (patch) | |
tree | 08d6a42c30760a5644d7b9550dfbf177303d32ad /pw_varint/varint.cc | |
parent | af744f59aa1767b2266a6521991d8e2ab25c6861 (diff) | |
download | pigweed-82d3cb35d37f9f4011a47f0b33acf55fd2e3f8c7.tar.gz |
pw_varint: Add varint module
This change adds a pw_varint module containing functions for encoding
and decoding variable-length integers.
Change-Id: I50bdf6d9d6762bffb93ee638683de53afed9c849
Diffstat (limited to 'pw_varint/varint.cc')
-rw-r--r-- | pw_varint/varint.cc | 77 |
1 files changed, 77 insertions, 0 deletions
diff --git a/pw_varint/varint.cc b/pw_varint/varint.cc new file mode 100644 index 000000000..3d0e759d7 --- /dev/null +++ b/pw_varint/varint.cc @@ -0,0 +1,77 @@ +// Copyright 2019 The Pigweed Authors +// +// Licensed under the Apache License, Version 2.0 (the "License"); you may not +// use this file except in compliance with the License. You may obtain a copy +// of the License at +// +// https://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +// License for the specific language governing permissions and limitations +// under the License. + +#include "pw_varint/varint.h" + +#include <algorithm> + +namespace pw::varint { +namespace { + +constexpr int64_t ZigZagDecode64(uint64_t n) { + return static_cast<int64_t>((n >> 1) ^ (~(n & 1) + 1)); +} + +} // namespace + +size_t EncodeLittleEndianBase128(uint64_t integer, + const span<uint8_t>& output) { + size_t written = 0; + do { + if (written >= output.size()) { + return 0; + } + + // Grab 7 bits; the eighth bit is set to 1 to indicate more data coming. + output[written++] = static_cast<uint8_t>(integer) | '\x80'; + integer >>= 7; + } while (integer != 0u); + + output[written - 1] &= '\x7f'; // clear the top bit of the last byte + return written; +} + +size_t DecodeVarint(const span<const uint8_t>& input, int64_t* value) { + const size_t bytes = DecodeVarint(input, reinterpret_cast<uint64_t*>(value)); + *value = ZigZagDecode64(*value); + return bytes; +} + +size_t DecodeVarint(const span<const uint8_t>& input, uint64_t* value) { + uint64_t decoded_value = 0; + uint_fast8_t count = 0; + + // The largest 64-bit ints require 10 B. + const size_t max_count = std::min(kMaxVarintSizeBytes, input.size()); + + while (true) { + if (count >= max_count) { + return 0; + } + + // Add the bottom seven bits of the next byte to the result. + decoded_value |= static_cast<uint64_t>(input[count] & '\x7f') + << (7 * count); + + // Stop decoding if the top bit is not set. + if ((input[count++] & '\x80') == 0) { + break; + } + } + + *value = decoded_value; + return count; +} + +} // namespace pw::varint |