221 lines
6.9 KiB
C++
221 lines
6.9 KiB
C++
// Copyright 2020 The Abseil 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 "absl/log/internal/proto.h"
|
|
|
|
#include <algorithm>
|
|
#include <cassert>
|
|
#include <cstddef>
|
|
#include <cstdint>
|
|
#include <cstring>
|
|
|
|
#include "absl/base/attributes.h"
|
|
#include "absl/base/config.h"
|
|
#include "absl/types/span.h"
|
|
|
|
namespace absl {
|
|
ABSL_NAMESPACE_BEGIN
|
|
namespace log_internal {
|
|
namespace {
|
|
void EncodeRawVarint(uint64_t value, size_t size, absl::Span<char> *buf) {
|
|
for (size_t s = 0; s < size; s++) {
|
|
(*buf)[s] = static_cast<char>((value & 0x7f) | (s + 1 == size ? 0 : 0x80));
|
|
value >>= 7;
|
|
}
|
|
buf->remove_prefix(size);
|
|
}
|
|
constexpr uint64_t MakeTagType(uint64_t tag, WireType type) {
|
|
return tag << 3 | static_cast<uint64_t>(type);
|
|
}
|
|
} // namespace
|
|
|
|
bool EncodeVarint(uint64_t tag, uint64_t value, absl::Span<char> *buf) {
|
|
const uint64_t tag_type = MakeTagType(tag, WireType::kVarint);
|
|
const size_t tag_type_size = VarintSize(tag_type);
|
|
const size_t value_size = VarintSize(value);
|
|
if (tag_type_size + value_size > buf->size()) {
|
|
buf->remove_suffix(buf->size());
|
|
return false;
|
|
}
|
|
EncodeRawVarint(tag_type, tag_type_size, buf);
|
|
EncodeRawVarint(value, value_size, buf);
|
|
return true;
|
|
}
|
|
|
|
bool Encode64Bit(uint64_t tag, uint64_t value, absl::Span<char> *buf) {
|
|
const uint64_t tag_type = MakeTagType(tag, WireType::k64Bit);
|
|
const size_t tag_type_size = VarintSize(tag_type);
|
|
if (tag_type_size + sizeof(value) > buf->size()) {
|
|
buf->remove_suffix(buf->size());
|
|
return false;
|
|
}
|
|
EncodeRawVarint(tag_type, tag_type_size, buf);
|
|
for (size_t s = 0; s < sizeof(value); s++) {
|
|
(*buf)[s] = static_cast<char>(value & 0xff);
|
|
value >>= 8;
|
|
}
|
|
buf->remove_prefix(sizeof(value));
|
|
return true;
|
|
}
|
|
|
|
bool Encode32Bit(uint64_t tag, uint32_t value, absl::Span<char> *buf) {
|
|
const uint64_t tag_type = MakeTagType(tag, WireType::k32Bit);
|
|
const size_t tag_type_size = VarintSize(tag_type);
|
|
if (tag_type_size + sizeof(value) > buf->size()) {
|
|
buf->remove_suffix(buf->size());
|
|
return false;
|
|
}
|
|
EncodeRawVarint(tag_type, tag_type_size, buf);
|
|
for (size_t s = 0; s < sizeof(value); s++) {
|
|
(*buf)[s] = static_cast<char>(value & 0xff);
|
|
value >>= 8;
|
|
}
|
|
buf->remove_prefix(sizeof(value));
|
|
return true;
|
|
}
|
|
|
|
bool EncodeBytes(uint64_t tag, absl::Span<const char> value,
|
|
absl::Span<char> *buf) {
|
|
const uint64_t tag_type = MakeTagType(tag, WireType::kLengthDelimited);
|
|
const size_t tag_type_size = VarintSize(tag_type);
|
|
uint64_t length = value.size();
|
|
const size_t length_size = VarintSize(length);
|
|
if (tag_type_size + length_size + value.size() > buf->size()) {
|
|
buf->remove_suffix(buf->size());
|
|
return false;
|
|
}
|
|
EncodeRawVarint(tag_type, tag_type_size, buf);
|
|
EncodeRawVarint(length, length_size, buf);
|
|
memcpy(buf->data(), value.data(), value.size());
|
|
buf->remove_prefix(value.size());
|
|
return true;
|
|
}
|
|
|
|
bool EncodeBytesTruncate(uint64_t tag, absl::Span<const char> value,
|
|
absl::Span<char> *buf) {
|
|
const uint64_t tag_type = MakeTagType(tag, WireType::kLengthDelimited);
|
|
const size_t tag_type_size = VarintSize(tag_type);
|
|
uint64_t length = value.size();
|
|
const size_t length_size =
|
|
VarintSize(std::min<uint64_t>(length, buf->size()));
|
|
if (tag_type_size + length_size <= buf->size() &&
|
|
tag_type_size + length_size + value.size() > buf->size()) {
|
|
value.remove_suffix(tag_type_size + length_size + value.size() -
|
|
buf->size());
|
|
length = value.size();
|
|
}
|
|
if (tag_type_size + length_size + value.size() > buf->size()) {
|
|
buf->remove_suffix(buf->size());
|
|
return false;
|
|
}
|
|
EncodeRawVarint(tag_type, tag_type_size, buf);
|
|
EncodeRawVarint(length, length_size, buf);
|
|
memcpy(buf->data(), value.data(), value.size());
|
|
buf->remove_prefix(value.size());
|
|
return true;
|
|
}
|
|
|
|
ABSL_MUST_USE_RESULT absl::Span<char> EncodeMessageStart(
|
|
uint64_t tag, uint64_t max_size, absl::Span<char> *buf) {
|
|
const uint64_t tag_type = MakeTagType(tag, WireType::kLengthDelimited);
|
|
const size_t tag_type_size = VarintSize(tag_type);
|
|
max_size = std::min<uint64_t>(max_size, buf->size());
|
|
const size_t length_size = VarintSize(max_size);
|
|
if (tag_type_size + length_size > buf->size()) {
|
|
buf->remove_suffix(buf->size());
|
|
return absl::Span<char>();
|
|
}
|
|
EncodeRawVarint(tag_type, tag_type_size, buf);
|
|
const absl::Span<char> ret = buf->subspan(0, length_size);
|
|
EncodeRawVarint(0, length_size, buf);
|
|
return ret;
|
|
}
|
|
|
|
void EncodeMessageLength(absl::Span<char> msg, const absl::Span<char> *buf) {
|
|
if (!msg.data()) return;
|
|
assert(buf->data() >= msg.data());
|
|
if (buf->data() < msg.data()) return;
|
|
EncodeRawVarint(
|
|
static_cast<uint64_t>(buf->data() - (msg.data() + msg.size())),
|
|
msg.size(), &msg);
|
|
}
|
|
|
|
namespace {
|
|
uint64_t DecodeVarint(absl::Span<const char> *buf) {
|
|
uint64_t value = 0;
|
|
size_t s = 0;
|
|
while (s < buf->size()) {
|
|
value |= static_cast<uint64_t>(static_cast<unsigned char>((*buf)[s]) & 0x7f)
|
|
<< 7 * s;
|
|
if (!((*buf)[s++] & 0x80)) break;
|
|
}
|
|
buf->remove_prefix(s);
|
|
return value;
|
|
}
|
|
|
|
uint64_t Decode64Bit(absl::Span<const char> *buf) {
|
|
uint64_t value = 0;
|
|
size_t s = 0;
|
|
while (s < buf->size()) {
|
|
value |= static_cast<uint64_t>(static_cast<unsigned char>((*buf)[s]))
|
|
<< 8 * s;
|
|
if (++s == sizeof(value)) break;
|
|
}
|
|
buf->remove_prefix(s);
|
|
return value;
|
|
}
|
|
|
|
uint32_t Decode32Bit(absl::Span<const char> *buf) {
|
|
uint32_t value = 0;
|
|
size_t s = 0;
|
|
while (s < buf->size()) {
|
|
value |= static_cast<uint32_t>(static_cast<unsigned char>((*buf)[s]))
|
|
<< 8 * s;
|
|
if (++s == sizeof(value)) break;
|
|
}
|
|
buf->remove_prefix(s);
|
|
return value;
|
|
}
|
|
} // namespace
|
|
|
|
bool ProtoField::DecodeFrom(absl::Span<const char> *data) {
|
|
if (data->empty()) return false;
|
|
const uint64_t tag_type = DecodeVarint(data);
|
|
tag_ = tag_type >> 3;
|
|
type_ = static_cast<WireType>(tag_type & 0x07);
|
|
switch (type_) {
|
|
case WireType::kVarint:
|
|
value_ = DecodeVarint(data);
|
|
break;
|
|
case WireType::k64Bit:
|
|
value_ = Decode64Bit(data);
|
|
break;
|
|
case WireType::kLengthDelimited: {
|
|
value_ = DecodeVarint(data);
|
|
data_ = data->subspan(
|
|
0, static_cast<size_t>(std::min<uint64_t>(value_, data->size())));
|
|
data->remove_prefix(data_.size());
|
|
break;
|
|
}
|
|
case WireType::k32Bit:
|
|
value_ = Decode32Bit(data);
|
|
break;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
} // namespace log_internal
|
|
ABSL_NAMESPACE_END
|
|
} // namespace absl
|