blob: 02a12542d23e955f19791b6678f7fe808990a2fc [file] [log] [blame]
/*
* Copyright (C) 2023 The Android Open Source Project
*
* 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
*
* http://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 "perfetto/public/abi/pb_decoder_abi.h"
#include <limits>
#include <type_traits>
#include "perfetto/public/pb_utils.h"
namespace {
template <typename T>
bool Uint64RepresentableIn(uint64_t val) {
return val <= std::numeric_limits<T>::max();
}
} // namespace
struct PerfettoPbDecoderField PerfettoPbDecoderParseField(
struct PerfettoPbDecoder* decoder) {
struct PerfettoPbDecoderField field;
const uint8_t* read_ptr = decoder->read_ptr;
if (read_ptr >= decoder->end_ptr) {
field.status = PERFETTO_PB_DECODER_DONE;
return field;
}
field.status = PERFETTO_PB_DECODER_ERROR;
uint64_t tag;
const uint8_t* end_of_tag =
PerfettoPbParseVarInt(read_ptr, decoder->end_ptr, &tag);
if (end_of_tag == read_ptr) {
field.status = PERFETTO_PB_DECODER_ERROR;
return field;
}
read_ptr = end_of_tag;
constexpr uint8_t kFieldTypeNumBits = 3;
field.wire_type = tag & ((1 << kFieldTypeNumBits) - 1);
uint64_t id = tag >> kFieldTypeNumBits;
static_assert(std::is_same<uint32_t, decltype(field.id)>::value);
if (id > std::numeric_limits<uint32_t>::max()) {
field.status = PERFETTO_PB_DECODER_ERROR;
return field;
}
field.id = static_cast<uint32_t>(id);
switch (field.wire_type) {
case PERFETTO_PB_WIRE_TYPE_DELIMITED: {
uint64_t len;
const uint8_t* end_of_len =
PerfettoPbParseVarInt(read_ptr, decoder->end_ptr, &len);
if (end_of_len == read_ptr || !Uint64RepresentableIn<size_t>(len)) {
field.status = PERFETTO_PB_DECODER_ERROR;
return field;
}
read_ptr = end_of_len;
field.value.delimited.len = static_cast<size_t>(len);
field.value.delimited.start = read_ptr;
read_ptr += len;
if (read_ptr > decoder->end_ptr) {
field.status = PERFETTO_PB_DECODER_ERROR;
return field;
}
field.status = PERFETTO_PB_DECODER_OK;
decoder->read_ptr = read_ptr;
break;
}
case PERFETTO_PB_WIRE_TYPE_VARINT: {
uint64_t val;
const uint8_t* end_of_val =
PerfettoPbParseVarInt(read_ptr, decoder->end_ptr, &val);
if (end_of_val == read_ptr) {
field.status = PERFETTO_PB_DECODER_ERROR;
return field;
}
read_ptr = end_of_val;
field.value.integer64 = val;
field.status = PERFETTO_PB_DECODER_OK;
decoder->read_ptr = read_ptr;
break;
}
case PERFETTO_PB_WIRE_TYPE_FIXED32: {
const uint8_t* end_of_val = read_ptr + sizeof(uint32_t);
if (end_of_val > decoder->end_ptr) {
field.status = PERFETTO_PB_DECODER_ERROR;
return field;
}
// Little endian on the wire.
field.value.integer32 = read_ptr[0] |
(static_cast<uint32_t>(read_ptr[1]) << 8) |
(static_cast<uint32_t>(read_ptr[2]) << 16) |
(static_cast<uint32_t>(read_ptr[3]) << 24);
read_ptr = end_of_val;
decoder->read_ptr = read_ptr;
field.status = PERFETTO_PB_DECODER_OK;
break;
}
case PERFETTO_PB_WIRE_TYPE_FIXED64: {
const uint8_t* end_of_val = read_ptr + sizeof(uint64_t);
if (end_of_val > decoder->end_ptr) {
field.status = PERFETTO_PB_DECODER_ERROR;
return field;
}
// Little endian on the wire.
field.value.integer64 = read_ptr[0] |
(static_cast<uint64_t>(read_ptr[1]) << 8) |
(static_cast<uint64_t>(read_ptr[2]) << 16) |
(static_cast<uint64_t>(read_ptr[3]) << 24) |
(static_cast<uint64_t>(read_ptr[4]) << 32) |
(static_cast<uint64_t>(read_ptr[5]) << 40) |
(static_cast<uint64_t>(read_ptr[6]) << 48) |
(static_cast<uint64_t>(read_ptr[7]) << 56);
read_ptr = end_of_val;
decoder->read_ptr = read_ptr;
field.status = PERFETTO_PB_DECODER_OK;
break;
}
default:
field.status = PERFETTO_PB_DECODER_ERROR;
return field;
}
return field;
}
uint32_t PerfettoPbDecoderSkipField(struct PerfettoPbDecoder* decoder) {
const uint8_t* read_ptr = decoder->read_ptr;
if (read_ptr >= decoder->end_ptr) {
return PERFETTO_PB_DECODER_DONE;
}
uint64_t tag;
const uint8_t* end_of_tag =
PerfettoPbParseVarInt(decoder->read_ptr, decoder->end_ptr, &tag);
if (end_of_tag == read_ptr) {
return PERFETTO_PB_DECODER_ERROR;
}
read_ptr = end_of_tag;
constexpr uint8_t kFieldTypeNumBits = 3;
uint32_t wire_type = tag & ((1 << kFieldTypeNumBits) - 1);
switch (wire_type) {
case PERFETTO_PB_WIRE_TYPE_DELIMITED: {
uint64_t len;
const uint8_t* end_of_len =
PerfettoPbParseVarInt(read_ptr, decoder->end_ptr, &len);
if (end_of_len == read_ptr) {
return PERFETTO_PB_DECODER_ERROR;
}
read_ptr = end_of_len;
read_ptr += len;
if (read_ptr > decoder->end_ptr) {
return PERFETTO_PB_DECODER_ERROR;
}
decoder->read_ptr = read_ptr;
return PERFETTO_PB_DECODER_OK;
}
case PERFETTO_PB_WIRE_TYPE_VARINT: {
uint64_t val;
const uint8_t* end_of_val =
PerfettoPbParseVarInt(read_ptr, decoder->end_ptr, &val);
if (end_of_val == read_ptr) {
return PERFETTO_PB_DECODER_ERROR;
}
read_ptr = end_of_val;
decoder->read_ptr = read_ptr;
return PERFETTO_PB_DECODER_OK;
}
case PERFETTO_PB_WIRE_TYPE_FIXED32: {
const uint8_t* end_of_val = read_ptr + sizeof(uint32_t);
if (end_of_val > decoder->end_ptr) {
return PERFETTO_PB_DECODER_ERROR;
}
read_ptr = end_of_val;
decoder->read_ptr = read_ptr;
return PERFETTO_PB_DECODER_OK;
}
case PERFETTO_PB_WIRE_TYPE_FIXED64: {
const uint8_t* end_of_val = read_ptr + sizeof(uint64_t);
if (read_ptr > decoder->end_ptr) {
return PERFETTO_PB_DECODER_ERROR;
}
read_ptr = end_of_val;
decoder->read_ptr = read_ptr;
return PERFETTO_PB_DECODER_OK;
}
default:
break;
}
return PERFETTO_PB_DECODER_ERROR;
}