|
|
|
// Protocol Buffers - Google's data interchange format
|
|
|
|
// Copyright 2024 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/text/internal/encode.h"
|
|
|
|
|
|
|
|
#include <inttypes.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
#include "upb/base/descriptor_constants.h"
|
|
|
|
#include "upb/base/string_view.h"
|
|
|
|
#include "upb/lex/round_trip.h"
|
|
|
|
#include "upb/message/array.h"
|
|
|
|
#include "upb/message/message.h"
|
|
|
|
#include "upb/text/options.h"
|
|
|
|
#include "upb/wire/eps_copy_input_stream.h"
|
|
|
|
#include "upb/wire/reader.h"
|
|
|
|
#include "upb/wire/types.h"
|
|
|
|
|
|
|
|
// Must be last.
|
|
|
|
#include "upb/port/def.inc"
|
|
|
|
|
|
|
|
#define CHK(x) \
|
|
|
|
do { \
|
|
|
|
if (!(x)) { \
|
|
|
|
return NULL; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Unknown fields are printed by number.
|
|
|
|
*
|
|
|
|
* 1001: 123
|
|
|
|
* 1002: "hello"
|
|
|
|
* 1006: 0xdeadbeef
|
|
|
|
* 1003: {
|
|
|
|
* 1: 111
|
|
|
|
* }
|
|
|
|
*/
|
|
|
|
const char* UPB_PRIVATE(_upb_TextEncode_Unknown)(txtenc* e, const char* ptr,
|
|
|
|
upb_EpsCopyInputStream* stream,
|
|
|
|
int groupnum) {
|
|
|
|
// We are guaranteed that the unknown data is valid wire format, and will not
|
|
|
|
// contain tag zero.
|
|
|
|
uint32_t end_group = groupnum > 0
|
|
|
|
? ((groupnum << kUpb_WireReader_WireTypeBits) |
|
|
|
|
kUpb_WireType_EndGroup)
|
|
|
|
: 0;
|
|
|
|
|
|
|
|
while (!upb_EpsCopyInputStream_IsDone(stream, &ptr)) {
|
|
|
|
uint32_t tag;
|
|
|
|
CHK(ptr = upb_WireReader_ReadTag(ptr, &tag));
|
|
|
|
if (tag == end_group) return ptr;
|
|
|
|
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_Indent)(e);
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_Printf)
|
|
|
|
(e, "%d: ", (int)upb_WireReader_GetFieldNumber(tag));
|
|
|
|
|
|
|
|
switch (upb_WireReader_GetWireType(tag)) {
|
|
|
|
case kUpb_WireType_Varint: {
|
|
|
|
uint64_t val;
|
|
|
|
CHK(ptr = upb_WireReader_ReadVarint(ptr, &val));
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_Printf)(e, "%" PRIu64, val);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case kUpb_WireType_32Bit: {
|
|
|
|
uint32_t val;
|
|
|
|
ptr = upb_WireReader_ReadFixed32(ptr, &val);
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_Printf)(e, "0x%08" PRIu32, val);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case kUpb_WireType_64Bit: {
|
|
|
|
uint64_t val;
|
|
|
|
ptr = upb_WireReader_ReadFixed64(ptr, &val);
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_Printf)(e, "0x%016" PRIu64, val);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case kUpb_WireType_Delimited: {
|
|
|
|
int size;
|
|
|
|
char* start = e->ptr;
|
|
|
|
size_t start_overflow = e->overflow;
|
|
|
|
CHK(ptr = upb_WireReader_ReadSize(ptr, &size));
|
|
|
|
CHK(upb_EpsCopyInputStream_CheckDataSizeAvailable(stream, ptr, size));
|
|
|
|
|
|
|
|
// Speculatively try to parse as message.
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_PutStr)(e, "{");
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_EndField)(e);
|
|
|
|
|
|
|
|
// EpsCopyInputStream can't back up, so create a sub-stream for the
|
|
|
|
// speculative parse.
|
|
|
|
upb_EpsCopyInputStream sub_stream;
|
|
|
|
const char* sub_ptr = upb_EpsCopyInputStream_GetAliasedPtr(stream, ptr);
|
|
|
|
upb_EpsCopyInputStream_Init(&sub_stream, &sub_ptr, size, true);
|
|
|
|
|
|
|
|
e->indent_depth++;
|
|
|
|
if (UPB_PRIVATE(_upb_TextEncode_Unknown)(e, sub_ptr, &sub_stream, -1)) {
|
|
|
|
ptr = upb_EpsCopyInputStream_Skip(stream, ptr, size);
|
|
|
|
e->indent_depth--;
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_Indent)(e);
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_PutStr)(e, "}");
|
|
|
|
} else {
|
|
|
|
// Didn't work out, print as raw bytes.
|
|
|
|
e->indent_depth--;
|
|
|
|
e->ptr = start;
|
|
|
|
e->overflow = start_overflow;
|
|
|
|
const char* str = ptr;
|
|
|
|
ptr = upb_EpsCopyInputStream_ReadString(stream, &str, size, NULL);
|
|
|
|
UPB_ASSERT(ptr);
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_Bytes)
|
|
|
|
(e, (upb_StringView){.data = str, .size = size});
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case kUpb_WireType_StartGroup:
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_PutStr)(e, "{");
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_EndField)(e);
|
|
|
|
e->indent_depth++;
|
|
|
|
CHK(ptr = UPB_PRIVATE(_upb_TextEncode_Unknown)(
|
|
|
|
e, ptr, stream, upb_WireReader_GetFieldNumber(tag)));
|
|
|
|
e->indent_depth--;
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_Indent)(e);
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_PutStr)(e, "}");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_EndField)(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
return end_group == 0 && !upb_EpsCopyInputStream_IsError(stream) ? ptr : NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
#undef CHK
|
|
|
|
|
|
|
|
void UPB_PRIVATE(_upb_TextEncode_ParseUnknown)(txtenc* e,
|
|
|
|
const upb_Message* msg) {
|
|
|
|
if ((e->options & UPB_TXTENC_SKIPUNKNOWN) != 0) return;
|
|
|
|
|
|
|
|
uintptr_t iter = kUpb_Message_UnknownBegin;
|
|
|
|
upb_StringView view;
|
|
|
|
while (upb_Message_NextUnknown(msg, &view, &iter)) {
|
|
|
|
char* start = e->ptr;
|
|
|
|
upb_EpsCopyInputStream stream;
|
|
|
|
upb_EpsCopyInputStream_Init(&stream, &view.data, view.size, true);
|
|
|
|
if (!UPB_PRIVATE(_upb_TextEncode_Unknown)(e, view.data, &stream, -1)) {
|
|
|
|
/* Unknown failed to parse, back up and don't print it at all. */
|
|
|
|
e->ptr = start;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void UPB_PRIVATE(_upb_TextEncode_Scalar)(txtenc* e, upb_MessageValue val,
|
|
|
|
upb_CType ctype) {
|
|
|
|
switch (ctype) {
|
|
|
|
case kUpb_CType_Bool:
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_PutStr)(e, val.bool_val ? "true" : "false");
|
|
|
|
break;
|
|
|
|
case kUpb_CType_Float: {
|
|
|
|
char buf[32];
|
|
|
|
_upb_EncodeRoundTripFloat(val.float_val, buf, sizeof(buf));
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_PutStr)(e, buf);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case kUpb_CType_Double: {
|
|
|
|
char buf[32];
|
|
|
|
_upb_EncodeRoundTripDouble(val.double_val, buf, sizeof(buf));
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_PutStr)(e, buf);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case kUpb_CType_Int32:
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_Printf)(e, "%" PRId32, val.int32_val);
|
|
|
|
break;
|
|
|
|
case kUpb_CType_UInt32:
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_Printf)(e, "%" PRIu32, val.uint32_val);
|
|
|
|
break;
|
|
|
|
case kUpb_CType_Int64:
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_Printf)(e, "%" PRId64, val.int64_val);
|
|
|
|
break;
|
|
|
|
case kUpb_CType_UInt64:
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_Printf)(e, "%" PRIu64, val.uint64_val);
|
|
|
|
break;
|
|
|
|
case kUpb_CType_String:
|
|
|
|
UPB_PRIVATE(_upb_HardenedPrintString)
|
|
|
|
(e, val.str_val.data, val.str_val.size);
|
|
|
|
break;
|
|
|
|
case kUpb_CType_Bytes:
|
|
|
|
UPB_PRIVATE(_upb_TextEncode_Bytes)(e, val.str_val);
|
|
|
|
break;
|
|
|
|
case kUpb_CType_Enum:
|
|
|
|
UPB_ASSERT(false); // handled separately in each encoder
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
UPB_UNREACHABLE();
|
|
|
|
}
|
|
|
|
}
|