blob: 46dc815605d3c40103a9a40575ea393da53b8133 [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 <stdint.h>
#include <string.h>
#include "upb/message/message.h"
#include "upb/mini_table/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"
static bool upb_DecodeFast_SingleFixed(upb_Decoder* d, const char** ptr,
void* dst, upb_DecodeFast_Type type,
upb_DecodeFastNext* next, void* ctx) {
UPB_UNUSED(ctx);
int valbytes = upb_DecodeFast_ValueBytes(type);
memcpy(dst, *ptr, valbytes);
*ptr += valbytes;
return true;
}
typedef struct {
upb_Decoder* d;
upb_DecodeFast_Type type;
upb_Message* msg;
uint64_t* data;
uint64_t* hasbits;
upb_DecodeFastNext* ret;
} upb_DecodeFast_PackedFixedContext;
static const char* upb_DecodeFast_PackedFixed(upb_EpsCopyInputStream* st,
const char* ptr, int size,
void* ctx) {
upb_DecodeFast_PackedFixedContext* c =
(upb_DecodeFast_PackedFixedContext*)ctx;
int valbytes = upb_DecodeFast_ValueBytes(c->type);
if (size == 0) return ptr; // 0-element packed fields are valid.
if (size % valbytes != 0) {
UPB_DECODEFAST_ERROR(c->d, kUpb_DecodeStatus_Malformed, c->ret);
return NULL;
}
upb_DecodeFastArray arr;
if (!upb_DecodeFast_GetArrayForAppend(c->d, ptr, c->msg, *c->data, c->hasbits,
&arr, c->type, size / valbytes,
c->ret)) {
return NULL;
}
upb_DecodeFast_InlineMemcpy(arr.dst, ptr, size);
arr.dst = UPB_PTR_AT(arr.dst, size, char);
upb_DecodeFastField_SetArraySize(&arr, c->type);
return ptr + size;
}
UPB_FORCEINLINE
void upb_DecodeFast_Fixed(upb_Decoder* d, const char** ptr, upb_Message* msg,
const upb_MiniTable* table, uint64_t* hasbits,
uint64_t* data, upb_DecodeFastNext* ret,
upb_DecodeFast_Type type,
upb_DecodeFast_Cardinality card,
upb_DecodeFast_TagSize tagsize, uint64_t data2) {
if (card == kUpb_DecodeFast_Packed) {
upb_DecodeFast_PackedFixedContext ctx = {
.d = d,
.type = type,
.msg = msg,
.data = data,
.hasbits = hasbits,
.ret = ret,
};
upb_DecodeFast_Packed(d, ptr, type, card, tagsize, data,
&upb_DecodeFast_PackedFixed, ret, &ctx, data2);
} else {
upb_DecodeFast_Unpacked(d, ptr, msg, data, hasbits, ret, type, card,
tagsize, &upb_DecodeFast_SingleFixed, NULL, data2);
}
}
/* Generate all combinations:
* {s,o,r,p} x {f4,f8} x {1bt,2bt} */
#define F(type, card, tagbytes) \
UPB_NOINLINE UPB_PRESERVE_NONE upb_FastDecoder_Return \
UPB_DECODEFAST_FUNCNAME(type, card, tagbytes)(UPB_PARSE_PARAMS) { \
upb_DecodeFastNext next = kUpb_DecodeFastNext_Dispatch; \
upb_DecodeFast_Fixed(d, &ptr, msg, table, &hasbits, &data, &next, \
kUpb_DecodeFast_##type, kUpb_DecodeFast_##card, \
kUpb_DecodeFast_##tagbytes, data2); \
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"