// 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 #include #include #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/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 false; \ } \ } 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_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(); } }