blob: 055b22df0b5de0cd5d1605ccfe0eaeb0385c955a [file] [log] [blame]
#include "binary_wireformat.h"
#include <cassert>
#include <cstdint>
#include <cstring>
#include <string>
#include "absl/strings/string_view.h"
#include "google/protobuf/endian.h"
#include "google/protobuf/wire_format_lite.h"
namespace google {
namespace protobuf {
namespace conformance {
namespace {
// The maximum number of bytes that it takes to encode a 64-bit varint.
#define VARINT_MAX_LEN 10
size_t vencode64(uint64_t val, int over_encoded_bytes, char* buf) {
if (val == 0) {
buf[0] = 0;
return 1;
}
size_t i = 0;
while (val) {
uint8_t byte = val & 0x7fU;
val >>= 7;
if (val || over_encoded_bytes) byte |= 0x80U;
buf[i++] = byte;
}
while (over_encoded_bytes--) {
assert(i < 10);
uint8_t byte = over_encoded_bytes ? 0x80 : 0;
buf[i++] = byte;
}
return i;
}
Wire MakeFixed32(void* data) {
uint32_t data_le;
std::memcpy(&data_le, data, 4);
data_le = ::google::protobuf::internal::little_endian::FromHost(data_le);
return Wire(std::string(reinterpret_cast<char*>(&data_le), 4));
}
Wire MakeFixed64(void* data) {
uint64_t data_le;
std::memcpy(&data_le, data, 8);
data_le = ::google::protobuf::internal::little_endian::FromHost(data_le);
return Wire(std::string(reinterpret_cast<char*>(&data_le), 8));
}
} // namespace
Wire Varint(uint64_t x) {
char buf[VARINT_MAX_LEN];
size_t len = vencode64(x, 0, buf);
return Wire(std::string(buf, len));
}
// Encodes a varint that is |extra| bytes longer than it needs to be, but still
// valid.
Wire LongVarint(uint64_t x, int extra) {
char buf[VARINT_MAX_LEN];
size_t len = vencode64(x, extra, buf);
return Wire(std::string(buf, len));
}
Wire SInt32(int32_t x) {
return Varint(::google::protobuf::internal::WireFormatLite::ZigZagEncode32(x));
}
Wire SInt64(int64_t x) {
return Varint(::google::protobuf::internal::WireFormatLite::ZigZagEncode64(x));
}
Wire Fixed32(uint32_t x) { return MakeFixed32(&x); }
Wire Fixed64(uint64_t x) { return MakeFixed64(&x); }
Wire Float(float f) { return MakeFixed32(&f); }
Wire Double(double d) { return MakeFixed64(&d); }
Wire LengthPrefixed(absl::string_view data) {
return Wire(Varint(data.size()), data);
}
Wire Tag(uint32_t fieldnum, WireType wire_type) {
return Varint((fieldnum << 3) | static_cast<uint8_t>(wire_type));
}
} // namespace conformance
} // namespace protobuf
} // namespace google