|
|
|
// Protocol Buffers - Google's data interchange format
|
|
|
|
// Copyright 2023 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 "google/protobuf/compiler/hpb/gen_repeated_fields.h"
|
|
|
|
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "google/protobuf/descriptor.pb.h"
|
|
|
|
#include "absl/strings/string_view.h"
|
|
|
|
#include "google/protobuf/compiler/hpb/context.h"
|
|
|
|
#include "google/protobuf/compiler/hpb/gen_accessors.h"
|
|
|
|
#include "google/protobuf/compiler/hpb/gen_enums.h"
|
|
|
|
#include "google/protobuf/compiler/hpb/gen_extensions.h"
|
|
|
|
#include "google/protobuf/compiler/hpb/gen_utils.h"
|
|
|
|
#include "google/protobuf/compiler/hpb/names.h"
|
|
|
|
#include "google/protobuf/descriptor.h"
|
Created proper `names.h` headers for all upb generators.
The goal of the `names.h` convention is to have a single canonical place where a code generator can define the set of symbols it exports to other code generators, and a canonical place where the name mangling logic is implemented.
Each upb code generator now has its own `names.h` file defining the symbols that it owns & exports:
* `third_party/upb/upb_generator/c/names.h` (for `foo.upb.h` files)
* `third_party/upb/upb_generator/minitable/names.h` (for `foo.upb_minitable.h` files)
* `third_party/upb/upb_generator/reflection/names.h` (for `foo.upbdefs.h` files)
This is a significant improvement over the previous situation where the name mangling functions were co-mingled in `common.h`/`mangle.h`, or sprinkled throughout the generators, with no clear structure for which code generator owns which symbols.
With this structure in place, the visibility lists for the various `names.h` files provide a clear dependency graph for how different generators depend on each other. In general, we want to keep dependencies on the "C" code generator to a minimum, since it is the largest and most complicated of upb's generated APIs, and is also the most prone to symbol name clashes.
Note that upb's `names.h` headers are somewhat unusual, in that we do not want them to depend on C++'s reflection or upb's reflection. Most `names.h` headers in protobuf would use types like `proto2::Descriptor`, but we don't want upb to depend on C++ reflection, especially during its bootstrapping process. We also don't want to force users to build upb defs just to use these name mangling functions. So we use only plain string types like `absl::string_view` and `std::string`.
PiperOrigin-RevId: 672397247
4 months ago
|
|
|
#include "upb_generator/c/names.h"
|
|
|
|
#include "upb_generator/common.h"
|
|
|
|
#include "upb_generator/file_layout.h"
|
|
|
|
|
|
|
|
namespace google::protobuf::hpb_generator {
|
|
|
|
|
|
|
|
namespace protobuf = ::proto2;
|
|
|
|
|
|
|
|
// Adds using accessors to reuse base Access class members from a Proxy/CProxy.
|
|
|
|
void WriteRepeatedFieldUsingAccessors(const protobuf::FieldDescriptor* field,
|
|
|
|
absl::string_view class_name,
|
|
|
|
absl::string_view resolved_field_name,
|
|
|
|
Context& ctx, bool read_only) {
|
|
|
|
if (field->cpp_type() == protobuf::FieldDescriptor::CPPTYPE_MESSAGE) {
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
using $0Access::$1;
|
|
|
|
using $0Access::$1_size;
|
|
|
|
)cc",
|
|
|
|
class_name, resolved_field_name);
|
|
|
|
if (!read_only) {
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
using $0Access::add_$1;
|
|
|
|
using $0Access::add_alias_$1;
|
|
|
|
using $0Access::mutable_$1;
|
|
|
|
)cc",
|
|
|
|
class_name, resolved_field_name);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
using $0Access::$1;
|
|
|
|
using $0Access::$1_size;
|
|
|
|
)cc",
|
|
|
|
class_name, resolved_field_name);
|
|
|
|
if (!read_only) {
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
using $0Access::add_$1;
|
|
|
|
using $0Access::mutable_$1;
|
|
|
|
using $0Access::resize_$1;
|
|
|
|
using $0Access::set_$1;
|
|
|
|
)cc",
|
|
|
|
class_name, resolved_field_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void WriteRepeatedFieldsInMessageHeader(const protobuf::Descriptor* desc,
|
|
|
|
const protobuf::FieldDescriptor* field,
|
|
|
|
absl::string_view resolved_field_name,
|
|
|
|
absl::string_view resolved_upbc_name,
|
|
|
|
Context& ctx) {
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
inline size_t $1_size() const {
|
|
|
|
size_t len;
|
|
|
|
$0_$2(msg_, &len);
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
)cc",
|
|
|
|
upb::generator::CApiMessageType(desc->full_name()), resolved_field_name,
|
|
|
|
resolved_upbc_name);
|
|
|
|
|
|
|
|
if (field->cpp_type() == protobuf::FieldDescriptor::CPPTYPE_MESSAGE) {
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
$1 $2(size_t index) const;
|
|
|
|
const ::hpb::RepeatedField<const $4>::CProxy $2() const;
|
|
|
|
::hpb::Ptr<::hpb::RepeatedField<$4>> mutable_$2();
|
|
|
|
absl::StatusOr<$0> add_$2();
|
|
|
|
/**
|
|
|
|
* Re-points submsg of repeated field to given target.
|
|
|
|
*
|
|
|
|
* REQUIRES: both messages must be in the same arena.
|
|
|
|
*/
|
|
|
|
bool add_alias_$2($0 target);
|
|
|
|
$0 mutable_$2(size_t index) const;
|
|
|
|
)cc",
|
|
|
|
MessagePtrConstType(field, /* const */ false), // $0
|
|
|
|
MessagePtrConstType(field, /* const */ true), // $1
|
|
|
|
resolved_field_name, // $2
|
|
|
|
resolved_upbc_name, // $3
|
|
|
|
MessageBaseType(field, /* maybe_const */ false) // $4
|
|
|
|
);
|
|
|
|
} else if (field->cpp_type() == protobuf::FieldDescriptor::CPPTYPE_STRING) {
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
$0 $1(size_t index) const;
|
|
|
|
const ::hpb::RepeatedField<$0>::CProxy $1() const;
|
|
|
|
::hpb::Ptr<::hpb::RepeatedField<$0>> mutable_$1();
|
|
|
|
bool add_$1($0 val);
|
|
|
|
void set_$1(size_t index, $0 val);
|
|
|
|
bool resize_$1(size_t len);
|
|
|
|
)cc",
|
|
|
|
CppConstType(field), resolved_field_name);
|
|
|
|
} else {
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
$0 $1(size_t index) const;
|
|
|
|
const ::hpb::RepeatedField<$0>::CProxy $1() const;
|
|
|
|
::hpb::Ptr<::hpb::RepeatedField<$0>> mutable_$1();
|
|
|
|
bool add_$1($0 val);
|
|
|
|
void set_$1(size_t index, $0 val);
|
|
|
|
bool resize_$1(size_t len);
|
|
|
|
)cc",
|
|
|
|
CppConstType(field), resolved_field_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void WriteRepeatedMessageAccessor(const protobuf::Descriptor* desc,
|
|
|
|
const protobuf::FieldDescriptor* field,
|
|
|
|
const absl::string_view resolved_field_name,
|
|
|
|
const absl::string_view class_name,
|
|
|
|
Context& ctx) {
|
|
|
|
const char arena_expression[] = "arena_";
|
|
|
|
absl::string_view upbc_name = field->name();
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
$1 $0::$2(size_t index) const {
|
|
|
|
size_t len;
|
|
|
|
auto* ptr = $3_$5(msg_, &len);
|
|
|
|
assert(index < len);
|
|
|
|
return ::hpb::interop::upb::MakeCHandle<$4>(
|
|
|
|
(upb_Message*)*(ptr + index), arena_);
|
|
|
|
}
|
|
|
|
)cc",
|
|
|
|
class_name, MessagePtrConstType(field, /* is_const */ true),
|
|
|
|
resolved_field_name, upb::generator::CApiMessageType(desc->full_name()),
|
|
|
|
MessageBaseType(field, /* maybe_const */ false), upbc_name);
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
absl::StatusOr<$1> $0::add_$2() {
|
|
|
|
auto new_msg = $3_add_$6(msg_, $5);
|
|
|
|
if (!new_msg) {
|
|
|
|
return ::hpb::MessageAllocationError();
|
|
|
|
}
|
|
|
|
return hpb::interop::upb::MakeHandle<$4>((upb_Message *)new_msg, $5);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool $0::add_alias_$2($1 target) {
|
|
|
|
ABSL_CHECK(upb_Arena_IsFused(arena_, hpb::interop::upb::GetArena(target)));
|
|
|
|
size_t size = 0;
|
|
|
|
$3_$2(msg_, &size);
|
|
|
|
auto elements = $3_resize_$2(msg_, size + 1, arena_);
|
|
|
|
if (!elements) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
elements[size] = ($9 *)hpb::interop::upb::GetMessage(target);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
)cc",
|
|
|
|
class_name, MessagePtrConstType(field, /* const */ false),
|
|
|
|
resolved_field_name, upb::generator::CApiMessageType(desc->full_name()),
|
|
|
|
MessageBaseType(field, /* maybe_const */ false), arena_expression,
|
|
|
|
upbc_name, ClassName(desc), field->index(),
|
|
|
|
upb::generator::CApiMessageType(field->message_type()->full_name()));
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
$1 $0::mutable_$2(size_t index) const {
|
|
|
|
size_t len;
|
|
|
|
auto* ptr = $3_$6(msg_, &len);
|
|
|
|
assert(index < len);
|
|
|
|
return hpb::interop::upb::MakeHandle<$4>((upb_Message*)*(ptr + index), $5);
|
|
|
|
}
|
|
|
|
)cc",
|
|
|
|
class_name, MessagePtrConstType(field, /* is_const */ false),
|
|
|
|
resolved_field_name, upb::generator::CApiMessageType(desc->full_name()),
|
|
|
|
MessageBaseType(field, /* maybe_const */ false), arena_expression,
|
|
|
|
upbc_name);
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
const ::hpb::RepeatedField<const $1>::CProxy $0::$2() const {
|
|
|
|
size_t size;
|
|
|
|
const upb_Array* arr = _$3_$4_$5(msg_, &size);
|
|
|
|
return ::hpb::RepeatedField<const $1>::CProxy(arr, arena_);
|
|
|
|
};
|
|
|
|
::hpb::Ptr<::hpb::RepeatedField<$1>> $0::mutable_$2() {
|
|
|
|
size_t size;
|
|
|
|
upb_Array* arr = _$3_$4_$6(msg_, &size, arena_);
|
|
|
|
return ::hpb::RepeatedField<$1>::Proxy(arr, arena_);
|
|
|
|
}
|
|
|
|
)cc",
|
|
|
|
class_name, // $0
|
|
|
|
MessageBaseType(field, /* maybe_const */ false), // $1
|
|
|
|
resolved_field_name, // $2
|
|
|
|
upb::generator::CApiMessageType(desc->full_name()), // $3
|
|
|
|
upbc_name, // $4
|
|
|
|
upb::generator::kRepeatedFieldArrayGetterPostfix, // $5
|
|
|
|
upb::generator::kRepeatedFieldMutableArrayGetterPostfix // $6
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WriteRepeatedStringAccessor(const protobuf::Descriptor* desc,
|
|
|
|
const protobuf::FieldDescriptor* field,
|
|
|
|
const absl::string_view resolved_field_name,
|
|
|
|
const absl::string_view class_name,
|
|
|
|
Context& ctx) {
|
|
|
|
absl::string_view upbc_name = field->name();
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
$1 $0::$2(size_t index) const {
|
|
|
|
size_t len;
|
|
|
|
auto* ptr = $3_mutable_$4(msg_, &len);
|
|
|
|
assert(index < len);
|
|
|
|
return hpb::interop::upb::FromUpbStringView(*(ptr + index));
|
|
|
|
}
|
|
|
|
)cc",
|
|
|
|
class_name, CppConstType(field), resolved_field_name,
|
|
|
|
upb::generator::CApiMessageType(desc->full_name()), upbc_name);
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
bool $0::resize_$1(size_t len) {
|
|
|
|
return $2_resize_$3(msg_, len, arena_);
|
|
|
|
}
|
|
|
|
)cc",
|
|
|
|
class_name, resolved_field_name,
|
|
|
|
upb::generator::CApiMessageType(desc->full_name()), upbc_name);
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
bool $0::add_$2($1 val) {
|
|
|
|
return $3_add_$4(msg_,
|
|
|
|
hpb::interop::upb::CopyToUpbStringView(val, arena_),
|
|
|
|
arena_);
|
|
|
|
}
|
|
|
|
)cc",
|
|
|
|
class_name, CppConstType(field), resolved_field_name,
|
|
|
|
upb::generator::CApiMessageType(desc->full_name()), upbc_name);
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
void $0::set_$2(size_t index, $1 val) {
|
|
|
|
size_t len;
|
|
|
|
auto* ptr = $3_mutable_$4(msg_, &len);
|
|
|
|
assert(index < len);
|
|
|
|
*(ptr + index) = hpb::interop::upb::CopyToUpbStringView(val, arena_);
|
|
|
|
}
|
|
|
|
)cc",
|
|
|
|
class_name, CppConstType(field), resolved_field_name,
|
|
|
|
upb::generator::CApiMessageType(desc->full_name()), upbc_name);
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
const ::hpb::RepeatedField<$1>::CProxy $0::$2() const {
|
|
|
|
size_t size;
|
|
|
|
const upb_Array* arr = _$3_$4_$5(msg_, &size);
|
|
|
|
return ::hpb::RepeatedField<$1>::CProxy(arr, arena_);
|
|
|
|
};
|
|
|
|
::hpb::Ptr<::hpb::RepeatedField<$1>> $0::mutable_$2() {
|
|
|
|
size_t size;
|
|
|
|
upb_Array* arr = _$3_$4_$6(msg_, &size, arena_);
|
|
|
|
return ::hpb::RepeatedField<$1>::Proxy(arr, arena_);
|
|
|
|
}
|
|
|
|
)cc",
|
|
|
|
class_name, // $0
|
|
|
|
CppConstType(field), // $1
|
|
|
|
resolved_field_name, // $2
|
|
|
|
upb::generator::CApiMessageType(desc->full_name()), // $3
|
|
|
|
upbc_name, // $4
|
|
|
|
upb::generator::kRepeatedFieldArrayGetterPostfix, // $5
|
|
|
|
upb::generator::kRepeatedFieldMutableArrayGetterPostfix // $6
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WriteRepeatedScalarAccessor(const protobuf::Descriptor* desc,
|
|
|
|
const protobuf::FieldDescriptor* field,
|
|
|
|
const absl::string_view resolved_field_name,
|
|
|
|
const absl::string_view class_name,
|
|
|
|
Context& ctx) {
|
|
|
|
absl::string_view upbc_name = field->name();
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
$1 $0::$2(size_t index) const {
|
|
|
|
size_t len;
|
|
|
|
auto* ptr = $3_mutable_$4(msg_, &len);
|
|
|
|
assert(index < len);
|
|
|
|
return *(ptr + index);
|
|
|
|
}
|
|
|
|
)cc",
|
|
|
|
class_name, CppConstType(field), resolved_field_name,
|
|
|
|
upb::generator::CApiMessageType(desc->full_name()), upbc_name);
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
bool $0::resize_$1(size_t len) {
|
|
|
|
return $2_resize_$3(msg_, len, arena_);
|
|
|
|
}
|
|
|
|
)cc",
|
|
|
|
class_name, resolved_field_name,
|
|
|
|
upb::generator::CApiMessageType(desc->full_name()), upbc_name);
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
bool $0::add_$2($1 val) { return $3_add_$4(msg_, val, arena_); }
|
|
|
|
)cc",
|
|
|
|
class_name, CppConstType(field), resolved_field_name,
|
|
|
|
upb::generator::CApiMessageType(desc->full_name()), upbc_name);
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
void $0::set_$2(size_t index, $1 val) {
|
|
|
|
size_t len;
|
|
|
|
auto* ptr = $3_mutable_$4(msg_, &len);
|
|
|
|
assert(index < len);
|
|
|
|
*(ptr + index) = val;
|
|
|
|
}
|
|
|
|
)cc",
|
|
|
|
class_name, CppConstType(field), resolved_field_name,
|
|
|
|
upb::generator::CApiMessageType(desc->full_name()), upbc_name);
|
|
|
|
ctx.EmitLegacy(
|
|
|
|
R"cc(
|
|
|
|
const ::hpb::RepeatedField<$1>::CProxy $0::$2() const {
|
|
|
|
size_t size;
|
|
|
|
const upb_Array* arr = _$3_$4_$5(msg_, &size);
|
|
|
|
return ::hpb::RepeatedField<$1>::CProxy(arr, arena_);
|
|
|
|
};
|
|
|
|
::hpb::Ptr<::hpb::RepeatedField<$1>> $0::mutable_$2() {
|
|
|
|
size_t size;
|
|
|
|
upb_Array* arr = _$3_$4_$6(msg_, &size, arena_);
|
|
|
|
return ::hpb::RepeatedField<$1>::Proxy(arr, arena_);
|
|
|
|
}
|
|
|
|
)cc",
|
|
|
|
class_name, // $0
|
|
|
|
CppConstType(field), // $1
|
|
|
|
resolved_field_name, // $2
|
|
|
|
upb::generator::CApiMessageType(desc->full_name()), // $3
|
|
|
|
upbc_name, // $4
|
|
|
|
upb::generator::kRepeatedFieldArrayGetterPostfix, // $5
|
|
|
|
upb::generator::kRepeatedFieldMutableArrayGetterPostfix // $6
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace protobuf
|
|
|
|
} // namespace google::hpb_generator
|