| // 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" |
| |
| static bool upb_DecodeFast_SingleFixed(upb_Decoder* d, const char** ptr, |
| void* dst, upb_DecodeFast_Type type, |
| upb_DecodeFastNext* next) { |
| 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) { |
| 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); |
| } |
| |
| _upb_Decoder_Trace(c->d, 'F'); |
| return ptr + size; |
| } |
| |
| UPB_FORCEINLINE |
| void upb_DecodeFast_Fixed(upb_Decoder* d, const char** ptr, upb_Message* msg, |
| intptr_t table, uint64_t* hasbits, uint64_t* data, |
| upb_DecodeFastNext* ret, upb_DecodeFast_Type type, |
| upb_DecodeFast_Cardinality card, |
| upb_DecodeFast_TagSize tagsize) { |
| if (card == kUpb_DecodeFast_Packed) { |
| upb_DecodeFast_PackedFixedContext ctx = { |
| .d = d, |
| .type = type, |
| .msg = msg, |
| .data = data, |
| .hasbits = hasbits, |
| .ret = ret, |
| }; |
| upb_DecodeFast_Delimited(d, ptr, type, card, tagsize, data, |
| &upb_DecodeFast_PackedFixed, ret, &ctx); |
| } else { |
| upb_DecodeFast_Unpacked(d, ptr, msg, data, hasbits, ret, type, card, |
| tagsize, &upb_DecodeFast_SingleFixed); |
| } |
| } |
| |
| /* 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) { \ |
| 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); \ |
| 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" |