blob: 8dd90616f39dfb0ed16ace4a9557043e66cee5d0 [file] [log] [blame]
// 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_extensions.h"
#include <string>
#include <vector>
#include "absl/strings/str_cat.h"
#include "google/protobuf/compiler/hpb/context.h"
#include "google/protobuf/compiler/hpb/gen_utils.h"
#include "google/protobuf/compiler/hpb/names.h"
#include "google/protobuf/descriptor.h"
#include "upb_generator/c/names.h"
namespace google::protobuf::hpb_generator {
namespace protobuf = ::proto2;
std::string ExtensionIdentifierBase(const protobuf::FieldDescriptor* ext) {
assert(ext->is_extension());
std::string ext_scope;
if (ext->extension_scope()) {
return upb::generator::CApiMessageType(ext->extension_scope()->full_name());
} else {
return ToCIdent(ext->file()->package());
}
}
std::string ContainingTypeName(const protobuf::FieldDescriptor* ext) {
return ext->containing_type()->file() != ext->file()
? QualifiedClassName(ext->containing_type())
: ClassName(ext->containing_type());
}
void WriteExtensionIdentifierHeader(const protobuf::FieldDescriptor* ext,
Context& ctx) {
std::string mini_table_name =
absl::StrCat(ExtensionIdentifierBase(ext), "_", ext->name(), "_ext");
std::string linkage = ext->extension_scope() ? "static" : "extern";
ctx.Emit(
{{"linkage", linkage},
{"extendee_type", ContainingTypeName(ext)},
{"extension_type", CppTypeParameterName(ext)},
{"extension_name", ext->name()}},
R"cc(
$linkage$ const ::hpb::internal::ExtensionIdentifier<$extendee_type$,
$extension_type$>
$extension_name$;
)cc");
}
void WriteExtensionIdentifiersHeader(
const std::vector<const protobuf::FieldDescriptor*>& extensions,
Context& ctx) {
for (const auto* ext : extensions) {
if (!ext->extension_scope()) {
WriteExtensionIdentifierHeader(ext, ctx);
}
}
}
void WriteExtensionIdentifier(const protobuf::FieldDescriptor* ext,
Context& ctx) {
std::string mini_table_name =
absl::StrCat(ExtensionIdentifierBase(ext), "_", ext->name(), "_ext");
std::string class_prefix =
ext->extension_scope() ? ClassName(ext->extension_scope()) + "::" : "";
ctx.Emit(
{{"containing_type_name", ContainingTypeName(ext)},
{"mini_table_name", mini_table_name},
{"ext_name", ext->name()},
{"default_value", DefaultValue(ext)},
{"ext_type", CppTypeParameterName(ext)},
{"class_prefix", class_prefix}},
R"cc(
constexpr ::hpb::internal::ExtensionIdentifier<$containing_type_name$,
$ext_type$>
$class_prefix$$ext_name$ =
::hpb::internal::PrivateAccess::InvokeConstructor<
::hpb::internal::ExtensionIdentifier<$containing_type_name$,
$ext_type$>>(
&$mini_table_name$, $default_value$);
)cc");
}
void WriteExtensionIdentifiers(
const std::vector<const protobuf::FieldDescriptor*>& extensions,
Context& ctx) {
for (const auto* ext : extensions) {
if (!ext->extension_scope()) {
WriteExtensionIdentifier(ext, ctx);
}
}
}
} // namespace protobuf
} // namespace google::hpb_generator