blob: 5b466b99ebd960a2d92e71fadac0f653e5939928 [file] [log] [blame]
// 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 <stdint.h>
#include <string.h>
#include "upb/message/array.h"
#include "upb/message/message.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"
// Must be last.
#include "upb/port/def.inc"
UPB_FORCEINLINE
int upb_DecodeFast_UnpackedFixed(upb_Decoder* d, const char** ptr,
upb_Message* msg, intptr_t table,
uint64_t* hasbits, uint64_t* data,
upb_DecodeFast_Type type,
upb_DecodeFast_Cardinality card,
upb_DecodeFast_TagSize tagsize) {
upb_DecodeFastNext ret;
upb_DecodeFastField field;
int valbytes = upb_DecodeFast_ValueBytes(type);
if (!upb_DecodeFast_CheckPackableTag(type, card, tagsize, data,
kUpb_DecodeFastNext_TailCallPacked,
&ret) ||
!Upb_DecodeFast_GetField(d, *ptr, msg, *data, hasbits, &ret, &field, type,
card)) {
return ret;
}
do {
*ptr += upb_DecodeFast_TagSizeBytes(tagsize);
memcpy(field.dst, *ptr, valbytes);
*ptr += valbytes;
_upb_Decoder_Trace(d, 'F');
} while (upb_DecodeFast_NextRepeated(d, ptr, *data, &ret, &field, type, card,
tagsize));
return kUpb_DecodeFastNext_TailCallDispatch;
}
UPB_FORCEINLINE
int upb_DecodeFast_PackedFixed(upb_Decoder* d, const char** ptr,
upb_Message* msg, intptr_t table,
uint64_t* hasbits, uint64_t* data,
upb_DecodeFast_Type type,
upb_DecodeFast_Cardinality card,
upb_DecodeFast_TagSize tagsize) {
upb_DecodeFastNext ret;
int size;
upb_DecodeFastField field;
uint8_t valbytes = upb_DecodeFast_ValueBytes(type);
const char* data_ptr = *ptr + upb_DecodeFast_TagSizeBytes(tagsize);
if (!upb_DecodeFast_CheckPackableTag(type, card, tagsize, data,
kUpb_DecodeFastNext_TailCallUnpacked,
&ret) ||
!upb_DecodeFast_DecodeShortSizeForImmediateRead(d, &data_ptr, &size,
&ret)) {
return ret;
}
if (size != 0) {
if (size % valbytes != 0) {
UPB_DECODEFAST_ERROR(d, kUpb_DecodeStatus_Malformed, &ret);
return ret;
}
if (!upb_DecodeFast_GetArrayForAppend(d, *ptr, msg, *data, hasbits, &field,
type, size / valbytes, &ret)) {
return ret;
}
upb_DecodeFast_InlineMemcpy(field.dst, data_ptr, size);
upb_DecodeFastField_AddArraySize(&field, size / valbytes);
}
*ptr = data_ptr + size;
_upb_Decoder_Trace(d, 'F');
return kUpb_DecodeFastNext_TailCallDispatch;
}
UPB_FORCEINLINE
int upb_DecodeFast_Fixed(upb_Decoder* d, const char** ptr, upb_Message* msg,
intptr_t table, uint64_t* hasbits, uint64_t* data,
upb_DecodeFast_Type type,
upb_DecodeFast_Cardinality card,
upb_DecodeFast_TagSize tagsize) {
if (card == kUpb_DecodeFast_Packed) {
return upb_DecodeFast_PackedFixed(d, ptr, msg, table, hasbits, data, type,
card, tagsize);
} else {
return upb_DecodeFast_UnpackedFixed(d, ptr, msg, table, hasbits, data, type,
card, tagsize);
}
}
/* Generate all combinations:
* {s,o,r,p} x {f4,f8} x {1bt,2bt} */
#define F(type, card, tagbytes) \
UPB_NOINLINE UPB_PRESERVE_NONE const char* UPB_DECODEFAST_FUNCNAME( \
type, card, tagbytes)(UPB_PARSE_PARAMS) { \
int next = upb_DecodeFast_Fixed( \
d, &ptr, msg, table, &hasbits, &data, kUpb_DecodeFast_##type, \
kUpb_DecodeFast_##card, kUpb_DecodeFast_##tagbytes); \
UPB_DECODEFAST_NEXTMAYBEPACKED( \
next, UPB_DECODEFAST_FUNCNAME(type, Repeated, tagbytes), \
UPB_DECODEFAST_FUNCNAME(type, Packed, tagbytes)); \
}
UPB_DECODEFAST_CARDINALITIES(UPB_DECODEFAST_TAGSIZES, F, Fixed32)
UPB_DECODEFAST_CARDINALITIES(UPB_DECODEFAST_TAGSIZES, F, Fixed64)
#undef F
#undef FASTDECODE_FIXED
#undef FASTDECODE_PACKEDFIXED
#undef FASTDECODE_UNPACKEDFIXED
#include "upb/port/undef.inc"