Protocol Buffers - Google's data interchange format (grpc依赖)
https://developers.google.com/protocol-buffers/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
480 lines
13 KiB
480 lines
13 KiB
// Protocol Buffers - Google's data interchange format |
|
// Copyright 2023 Google LLC. All rights reserved. |
|
// https://developers.google.com/protocol-buffers/ |
|
// |
|
// Redistribution and use in source and binary forms, with or without |
|
// modification, are permitted provided that the following conditions are |
|
// met: |
|
// |
|
// * Redistributions of source code must retain the above copyright |
|
// notice, this list of conditions and the following disclaimer. |
|
// * Redistributions in binary form must reproduce the above |
|
// copyright notice, this list of conditions and the following disclaimer |
|
// in the documentation and/or other materials provided with the |
|
// distribution. |
|
// * Neither the name of Google LLC nor the names of its |
|
// contributors may be used to endorse or promote products derived from |
|
// this software without specific prior written permission. |
|
// |
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
|
|
|
#include "upb/text/encode.h" |
|
|
|
#include <ctype.h> |
|
#include <float.h> |
|
#include <inttypes.h> |
|
#include <stdarg.h> |
|
#include <string.h> |
|
|
|
#include "upb/collections/internal/map_sorter.h" |
|
#include "upb/collections/map.h" |
|
#include "upb/lex/round_trip.h" |
|
#include "upb/port/vsnprintf_compat.h" |
|
#include "upb/reflection/message.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" |
|
|
|
typedef struct { |
|
char *buf, *ptr, *end; |
|
size_t overflow; |
|
int indent_depth; |
|
int options; |
|
const upb_DefPool* ext_pool; |
|
_upb_mapsorter sorter; |
|
} txtenc; |
|
|
|
static void txtenc_msg(txtenc* e, const upb_Message* msg, |
|
const upb_MessageDef* m); |
|
|
|
static void txtenc_putbytes(txtenc* e, const void* data, size_t len) { |
|
size_t have = e->end - e->ptr; |
|
if (UPB_LIKELY(have >= len)) { |
|
memcpy(e->ptr, data, len); |
|
e->ptr += len; |
|
} else { |
|
if (have) { |
|
memcpy(e->ptr, data, have); |
|
e->ptr += have; |
|
} |
|
e->overflow += (len - have); |
|
} |
|
} |
|
|
|
static void txtenc_putstr(txtenc* e, const char* str) { |
|
txtenc_putbytes(e, str, strlen(str)); |
|
} |
|
|
|
static void txtenc_printf(txtenc* e, const char* fmt, ...) { |
|
size_t n; |
|
size_t have = e->end - e->ptr; |
|
va_list args; |
|
|
|
va_start(args, fmt); |
|
n = _upb_vsnprintf(e->ptr, have, fmt, args); |
|
va_end(args); |
|
|
|
if (UPB_LIKELY(have > n)) { |
|
e->ptr += n; |
|
} else { |
|
e->ptr = UPB_PTRADD(e->ptr, have); |
|
e->overflow += (n - have); |
|
} |
|
} |
|
|
|
static void txtenc_indent(txtenc* e) { |
|
if ((e->options & UPB_TXTENC_SINGLELINE) == 0) { |
|
int i = e->indent_depth; |
|
while (i-- > 0) { |
|
txtenc_putstr(e, " "); |
|
} |
|
} |
|
} |
|
|
|
static void txtenc_endfield(txtenc* e) { |
|
if (e->options & UPB_TXTENC_SINGLELINE) { |
|
txtenc_putstr(e, " "); |
|
} else { |
|
txtenc_putstr(e, "\n"); |
|
} |
|
} |
|
|
|
static void txtenc_enum(int32_t val, const upb_FieldDef* f, txtenc* e) { |
|
const upb_EnumDef* e_def = upb_FieldDef_EnumSubDef(f); |
|
const upb_EnumValueDef* ev = upb_EnumDef_FindValueByNumber(e_def, val); |
|
|
|
if (ev) { |
|
txtenc_printf(e, "%s", upb_EnumValueDef_Name(ev)); |
|
} else { |
|
txtenc_printf(e, "%" PRId32, val); |
|
} |
|
} |
|
|
|
static void txtenc_string(txtenc* e, upb_StringView str, bool bytes) { |
|
const char* ptr = str.data; |
|
const char* end = ptr + str.size; |
|
txtenc_putstr(e, "\""); |
|
|
|
while (ptr < end) { |
|
switch (*ptr) { |
|
case '\n': |
|
txtenc_putstr(e, "\\n"); |
|
break; |
|
case '\r': |
|
txtenc_putstr(e, "\\r"); |
|
break; |
|
case '\t': |
|
txtenc_putstr(e, "\\t"); |
|
break; |
|
case '\"': |
|
txtenc_putstr(e, "\\\""); |
|
break; |
|
case '\'': |
|
txtenc_putstr(e, "\\'"); |
|
break; |
|
case '\\': |
|
txtenc_putstr(e, "\\\\"); |
|
break; |
|
default: |
|
if ((bytes || (uint8_t)*ptr < 0x80) && !isprint(*ptr)) { |
|
txtenc_printf(e, "\\%03o", (int)(uint8_t)*ptr); |
|
} else { |
|
txtenc_putbytes(e, ptr, 1); |
|
} |
|
break; |
|
} |
|
ptr++; |
|
} |
|
|
|
txtenc_putstr(e, "\""); |
|
} |
|
|
|
static void txtenc_field(txtenc* e, upb_MessageValue val, |
|
const upb_FieldDef* f) { |
|
txtenc_indent(e); |
|
const upb_CType type = upb_FieldDef_CType(f); |
|
const bool is_ext = upb_FieldDef_IsExtension(f); |
|
const char* full = upb_FieldDef_FullName(f); |
|
const char* name = upb_FieldDef_Name(f); |
|
|
|
if (type == kUpb_CType_Message) { |
|
if (is_ext) { |
|
txtenc_printf(e, "[%s] {", full); |
|
} else { |
|
txtenc_printf(e, "%s {", name); |
|
} |
|
txtenc_endfield(e); |
|
e->indent_depth++; |
|
txtenc_msg(e, val.msg_val, upb_FieldDef_MessageSubDef(f)); |
|
e->indent_depth--; |
|
txtenc_indent(e); |
|
txtenc_putstr(e, "}"); |
|
txtenc_endfield(e); |
|
return; |
|
} |
|
|
|
if (is_ext) { |
|
txtenc_printf(e, "[%s]: ", full); |
|
} else { |
|
txtenc_printf(e, "%s: ", name); |
|
} |
|
|
|
switch (type) { |
|
case kUpb_CType_Bool: |
|
txtenc_putstr(e, val.bool_val ? "true" : "false"); |
|
break; |
|
case kUpb_CType_Float: { |
|
char buf[32]; |
|
_upb_EncodeRoundTripFloat(val.float_val, buf, sizeof(buf)); |
|
txtenc_putstr(e, buf); |
|
break; |
|
} |
|
case kUpb_CType_Double: { |
|
char buf[32]; |
|
_upb_EncodeRoundTripDouble(val.double_val, buf, sizeof(buf)); |
|
txtenc_putstr(e, buf); |
|
break; |
|
} |
|
case kUpb_CType_Int32: |
|
txtenc_printf(e, "%" PRId32, val.int32_val); |
|
break; |
|
case kUpb_CType_UInt32: |
|
txtenc_printf(e, "%" PRIu32, val.uint32_val); |
|
break; |
|
case kUpb_CType_Int64: |
|
txtenc_printf(e, "%" PRId64, val.int64_val); |
|
break; |
|
case kUpb_CType_UInt64: |
|
txtenc_printf(e, "%" PRIu64, val.uint64_val); |
|
break; |
|
case kUpb_CType_String: |
|
txtenc_string(e, val.str_val, false); |
|
break; |
|
case kUpb_CType_Bytes: |
|
txtenc_string(e, val.str_val, true); |
|
break; |
|
case kUpb_CType_Enum: |
|
txtenc_enum(val.int32_val, f, e); |
|
break; |
|
default: |
|
UPB_UNREACHABLE(); |
|
} |
|
|
|
txtenc_endfield(e); |
|
} |
|
|
|
/* |
|
* Arrays print as simple repeated elements, eg. |
|
* |
|
* foo_field: 1 |
|
* foo_field: 2 |
|
* foo_field: 3 |
|
*/ |
|
static void txtenc_array(txtenc* e, const upb_Array* arr, |
|
const upb_FieldDef* f) { |
|
size_t i; |
|
size_t size = upb_Array_Size(arr); |
|
|
|
for (i = 0; i < size; i++) { |
|
txtenc_field(e, upb_Array_Get(arr, i), f); |
|
} |
|
} |
|
|
|
static void txtenc_mapentry(txtenc* e, upb_MessageValue key, |
|
upb_MessageValue val, const upb_FieldDef* f) { |
|
const upb_MessageDef* entry = upb_FieldDef_MessageSubDef(f); |
|
const upb_FieldDef* key_f = upb_MessageDef_Field(entry, 0); |
|
const upb_FieldDef* val_f = upb_MessageDef_Field(entry, 1); |
|
txtenc_indent(e); |
|
txtenc_printf(e, "%s {", upb_FieldDef_Name(f)); |
|
txtenc_endfield(e); |
|
e->indent_depth++; |
|
|
|
txtenc_field(e, key, key_f); |
|
txtenc_field(e, val, val_f); |
|
|
|
e->indent_depth--; |
|
txtenc_indent(e); |
|
txtenc_putstr(e, "}"); |
|
txtenc_endfield(e); |
|
} |
|
|
|
/* |
|
* Maps print as messages of key/value, etc. |
|
* |
|
* foo_map: { |
|
* key: "abc" |
|
* value: 123 |
|
* } |
|
* foo_map: { |
|
* key: "def" |
|
* value: 456 |
|
* } |
|
*/ |
|
static void txtenc_map(txtenc* e, const upb_Map* map, const upb_FieldDef* f) { |
|
if (e->options & UPB_TXTENC_NOSORT) { |
|
size_t iter = kUpb_Map_Begin; |
|
upb_MessageValue key, val; |
|
while (upb_Map_Next(map, &key, &val, &iter)) { |
|
txtenc_mapentry(e, key, val, f); |
|
} |
|
} else { |
|
const upb_MessageDef* entry = upb_FieldDef_MessageSubDef(f); |
|
const upb_FieldDef* key_f = upb_MessageDef_Field(entry, 0); |
|
_upb_sortedmap sorted; |
|
upb_MapEntry ent; |
|
|
|
_upb_mapsorter_pushmap(&e->sorter, upb_FieldDef_Type(key_f), map, &sorted); |
|
while (_upb_sortedmap_next(&e->sorter, map, &sorted, &ent)) { |
|
upb_MessageValue key, val; |
|
memcpy(&key, &ent.data.k, sizeof(key)); |
|
memcpy(&val, &ent.data.v, sizeof(val)); |
|
txtenc_mapentry(e, key, val, f); |
|
} |
|
_upb_mapsorter_popmap(&e->sorter, &sorted); |
|
} |
|
} |
|
|
|
#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 |
|
* } |
|
*/ |
|
static const char* txtenc_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; |
|
|
|
txtenc_indent(e); |
|
txtenc_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)); |
|
txtenc_printf(e, "%" PRIu64, val); |
|
break; |
|
} |
|
case kUpb_WireType_32Bit: { |
|
uint32_t val; |
|
ptr = upb_WireReader_ReadFixed32(ptr, &val); |
|
txtenc_printf(e, "0x%08" PRIu32, val); |
|
break; |
|
} |
|
case kUpb_WireType_64Bit: { |
|
uint64_t val; |
|
ptr = upb_WireReader_ReadFixed64(ptr, &val); |
|
txtenc_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. |
|
txtenc_putstr(e, "{"); |
|
txtenc_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 (txtenc_unknown(e, sub_ptr, &sub_stream, -1)) { |
|
ptr = upb_EpsCopyInputStream_Skip(stream, ptr, size); |
|
e->indent_depth--; |
|
txtenc_indent(e); |
|
txtenc_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); |
|
assert(ptr); |
|
txtenc_string(e, (upb_StringView){.data = str, .size = size}, true); |
|
} |
|
break; |
|
} |
|
case kUpb_WireType_StartGroup: |
|
txtenc_putstr(e, "{"); |
|
txtenc_endfield(e); |
|
e->indent_depth++; |
|
CHK(ptr = txtenc_unknown(e, ptr, stream, |
|
upb_WireReader_GetFieldNumber(tag))); |
|
e->indent_depth--; |
|
txtenc_indent(e); |
|
txtenc_putstr(e, "}"); |
|
break; |
|
default: |
|
return NULL; |
|
} |
|
txtenc_endfield(e); |
|
} |
|
|
|
return end_group == 0 && !upb_EpsCopyInputStream_IsError(stream) ? ptr : NULL; |
|
} |
|
|
|
#undef CHK |
|
|
|
static void txtenc_msg(txtenc* e, const upb_Message* msg, |
|
const upb_MessageDef* m) { |
|
size_t iter = kUpb_Message_Begin; |
|
const upb_FieldDef* f; |
|
upb_MessageValue val; |
|
|
|
while (upb_Message_Next(msg, m, e->ext_pool, &f, &val, &iter)) { |
|
if (upb_FieldDef_IsMap(f)) { |
|
txtenc_map(e, val.map_val, f); |
|
} else if (upb_FieldDef_IsRepeated(f)) { |
|
txtenc_array(e, val.array_val, f); |
|
} else { |
|
txtenc_field(e, val, f); |
|
} |
|
} |
|
|
|
if ((e->options & UPB_TXTENC_SKIPUNKNOWN) == 0) { |
|
size_t size; |
|
const char* ptr = upb_Message_GetUnknown(msg, &size); |
|
if (size != 0) { |
|
char* start = e->ptr; |
|
upb_EpsCopyInputStream stream; |
|
upb_EpsCopyInputStream_Init(&stream, &ptr, size, true); |
|
if (!txtenc_unknown(e, ptr, &stream, -1)) { |
|
/* Unknown failed to parse, back up and don't print it at all. */ |
|
e->ptr = start; |
|
} |
|
} |
|
} |
|
} |
|
|
|
size_t txtenc_nullz(txtenc* e, size_t size) { |
|
size_t ret = e->ptr - e->buf + e->overflow; |
|
|
|
if (size > 0) { |
|
if (e->ptr == e->end) e->ptr--; |
|
*e->ptr = '\0'; |
|
} |
|
|
|
return ret; |
|
} |
|
|
|
size_t upb_TextEncode(const upb_Message* msg, const upb_MessageDef* m, |
|
const upb_DefPool* ext_pool, int options, char* buf, |
|
size_t size) { |
|
txtenc e; |
|
|
|
e.buf = buf; |
|
e.ptr = buf; |
|
e.end = UPB_PTRADD(buf, size); |
|
e.overflow = 0; |
|
e.indent_depth = 0; |
|
e.options = options; |
|
e.ext_pool = ext_pool; |
|
_upb_mapsorter_init(&e.sorter); |
|
|
|
txtenc_msg(&e, msg, m); |
|
_upb_mapsorter_destroy(&e.sorter); |
|
return txtenc_nullz(&e, size); |
|
}
|
|
|