blob: 204362fe0707bd1722d58a99d32cbe5809c8b36e [file]
// Protocol Buffers - Google's data interchange format
// Copyright 2025 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 <stddef.h>
#include <stdint.h>
#include "upb/base/string_view.h"
#include "upb/wire/decode.h"
#include "upb/wire/decode_fast/cardinality.h"
#include "upb/wire/decode_fast/combinations.h"
#include "upb/wire/decode_fast/dispatch.h"
#include "upb/wire/decode_fast/field_parsers.h"
#include "upb/wire/eps_copy_input_stream.h"
#include "upb/wire/internal/decoder.h"
#include "utf8_range.h"
// Must be last.
#include "upb/port/def.inc"
UPB_FORCEINLINE
bool upb_DecodeFast_SingleStringAlias(upb_Decoder* d, const char** ptr,
void* dst, upb_DecodeFast_Type type,
upb_DecodeFastNext* next, void* ctx) {
UPB_UNUSED(ctx);
bool validate_utf8 = type == kUpb_DecodeFast_String;
upb_StringView* sv = dst;
int size;
if (!upb_DecodeFast_DecodeSize(d, ptr, &size, next)) return false;
*ptr = upb_EpsCopyInputStream_ReadStringAlwaysAlias(EPS(d), *ptr, size, sv);
if (validate_utf8 && !utf8_range_IsValid(sv->data, sv->size)) {
return UPB_DECODEFAST_ERROR(d, kUpb_DecodeStatus_BadUtf8, next);
}
return true;
}
UPB_FORCEINLINE
bool upb_DecodeFast_SingleStringCopy(upb_Decoder* d, const char** ptr,
void* dst, upb_DecodeFast_Type type,
upb_DecodeFastNext* next, void* ctx) {
bool validate_utf8 = type == kUpb_DecodeFast_String;
upb_StringView* sv = dst;
int size;
if (!upb_DecodeFast_DecodeSize(d, ptr, &size, next)) return false;
if (!_upb_Decoder_ReadString(d, ptr, size, sv, validate_utf8)) {
sv->size = 0;
return UPB_DECODEFAST_ERROR(d, kUpb_DecodeStatus_OutOfMemory, next);
}
return true;
}
#define F_COPY(type, card, tagsize) \
UPB_NOINLINE UPB_PRESERVE_NONE static upb_FastDecoder_Return \
UPB_DECODEFAST_FUNCNAME(type, card, tagsize##_Copy)(UPB_PARSE_PARAMS) { \
upb_DecodeFastNext next = kUpb_DecodeFastNext_Dispatch; \
upb_DecodeFast_Unpacked(d, &ptr, msg, &data, &hasbits, &next, \
kUpb_DecodeFast_##type, kUpb_DecodeFast_##card, \
kUpb_DecodeFast_##tagsize, \
&upb_DecodeFast_SingleStringCopy, NULL, data2); \
UPB_DECODEFAST_NEXT(next); \
}
#define F(type, card, tagsize) \
F_COPY(type, card, tagsize) \
upb_FastDecoder_Return UPB_PRESERVE_NONE UPB_DECODEFAST_FUNCNAME( \
type, card, tagsize)(UPB_PARSE_PARAMS) { \
if (UPB_UNLIKELY((d->options & kUpb_DecodeOption_AliasString) == 0)) { \
UPB_MUSTTAIL return UPB_DECODEFAST_FUNCNAME( \
type, card, tagsize##_Copy)(UPB_PARSE_ARGS); \
} \
upb_DecodeFastNext next = kUpb_DecodeFastNext_Dispatch; \
upb_DecodeFast_Unpacked(d, &ptr, msg, &data, &hasbits, &next, \
kUpb_DecodeFast_##type, kUpb_DecodeFast_##card, \
kUpb_DecodeFast_##tagsize, \
&upb_DecodeFast_SingleStringAlias, NULL, data2); \
UPB_DECODEFAST_NEXT(next); \
}
UPB_DECODEFAST_CARDINALITIES(UPB_DECODEFAST_TAGSIZES, F, String)
UPB_DECODEFAST_CARDINALITIES(UPB_DECODEFAST_TAGSIZES, F, Bytes)
#undef F
#undef F_COPY
#undef F
#undef FASTDECODE_LONGSTRING
#undef FASTDECODE_COPYSTRING
#undef FASTDECODE_STRING