blob: 85cd3868f1a2559240b2d4681f315c2aa067779e [file] [log] [blame]
// Protocol Buffers - Google's data interchange format
// Copyright 2023 Google LLC. All rights reserved.
//
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file or at
// https://developers.google.com/open-source/licenses/bsd
#include "upb/wire/reader.h"
#include <stddef.h>
#include <stdint.h>
#include "upb/wire/eps_copy_input_stream.h"
#include "upb/wire/types.h"
// Must be last.
#include "upb/port/def.inc"
UPB_NOINLINE UPB_PRIVATE(_upb_WireReader_LongVarint)
UPB_PRIVATE(_upb_WireReader_ReadLongVarint)(
const char* ptr, uint64_t val, upb_EpsCopyInputStream* stream) {
for (int i = 1; i < 10; i++) {
uint64_t byte = (uint8_t)ptr[i];
val += (byte - 1) << (i * 7);
if (!(byte & 0x80)) {
return (UPB_PRIVATE(_upb_WireReader_LongVarint)){ptr + i + 1, val};
}
}
return (UPB_PRIVATE(_upb_WireReader_LongVarint)){
UPB_PRIVATE(upb_EpsCopyInputStream_ReturnError)(stream), 0};
}
UPB_NOINLINE UPB_PRIVATE(_upb_WireReader_LongVarint)
UPB_PRIVATE(_upb_WireReader_ReadLongTag)(const char* ptr, uint64_t val,
upb_EpsCopyInputStream* stream) {
for (int i = 1; i < 5; i++) {
uint64_t byte = (uint8_t)ptr[i];
val += (byte - 1) << (i * 7);
if (!(byte & 0x80)) {
if (val > UINT32_MAX) break;
return (UPB_PRIVATE(_upb_WireReader_LongVarint)){ptr + i + 1, val};
}
}
return (UPB_PRIVATE(_upb_WireReader_LongVarint)){
UPB_PRIVATE(upb_EpsCopyInputStream_ReturnError)(stream), 0};
}
UPB_NOINLINE UPB_PRIVATE(_upb_WireReader_LongVarint)
UPB_PRIVATE(_upb_WireReader_ReadLongSize)(const char* ptr, uint64_t val,
upb_EpsCopyInputStream* stream) {
for (int i = 1; i < 5; i++) {
uint64_t byte = (uint8_t)ptr[i];
val += (byte - 1) << (i * 7);
if (!(byte & 0x80)) {
if (val > INT32_MAX) break;
return (UPB_PRIVATE(_upb_WireReader_LongVarint)){ptr + i + 1, val};
}
}
return (UPB_PRIVATE(_upb_WireReader_LongVarint)){
UPB_PRIVATE(upb_EpsCopyInputStream_ReturnError)(stream), 0};
}
const char* UPB_PRIVATE(_upb_WireReader_SkipGroup)(
const char* ptr, uint32_t tag, int depth_limit,
upb_EpsCopyInputStream* stream) {
if (--depth_limit < 0) {
return UPB_PRIVATE(upb_EpsCopyInputStream_ReturnError)(stream);
}
uint32_t end_group_tag = (tag & ~7ULL) | kUpb_WireType_EndGroup;
while (!upb_EpsCopyInputStream_IsDone(stream, &ptr)) {
uint32_t tag;
ptr = upb_WireReader_ReadTag(ptr, &tag, stream);
if (!ptr) break;
if (tag == end_group_tag) return ptr;
ptr = _upb_WireReader_SkipValue(ptr, tag, depth_limit, stream);
if (!ptr) break;
}
// Encountered limit end before end group tag.
return UPB_PRIVATE(upb_EpsCopyInputStream_ReturnError)(stream);
}