| /* | 
 |  * 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; | 
 | } |