/* * 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 #include #include "perfetto/public/pb_utils.h" namespace { template bool Uint64RepresentableIn(uint64_t val) { return val <= std::numeric_limits::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::value); if (id > std::numeric_limits::max()) { field.status = PERFETTO_PB_DECODER_ERROR; return field; } field.id = static_cast(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(len)) { field.status = PERFETTO_PB_DECODER_ERROR; return field; } read_ptr = end_of_len; field.value.delimited.len = static_cast(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(read_ptr[1]) << 8) | (static_cast(read_ptr[2]) << 16) | (static_cast(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(read_ptr[1]) << 8) | (static_cast(read_ptr[2]) << 16) | (static_cast(read_ptr[3]) << 24) | (static_cast(read_ptr[4]) << 32) | (static_cast(read_ptr[5]) << 40) | (static_cast(read_ptr[6]) << 48) | (static_cast(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; }