// 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 #include #include #include "absl/container/flat_hash_set.h" #include "absl/strings/string_view.h" #include "absl/strings/substitute.h" #include "google/protobuf/compiler/code_generator.h" #include "google/protobuf/compiler/plugin.h" #include "google/protobuf/descriptor.h" #include "upb/mem/arena.hpp" #include "upb/reflection/def.hpp" #include "upb_generator/file_layout.h" #include "upb_generator/plugin.h" #include "upb_generator/reflection/context.h" #include "upb_generator/reflection/header.h" #include "upb_generator/reflection/source.h" namespace upb { namespace generator { namespace reflection { void GenerateFile(upb::FileDefPtr file, const Options& options, google::protobuf::compiler::GeneratorContext* context) { GenerateReflectionHeader(file, options, context); GenerateReflectionSource(file, options, context); } bool ParseOptions(absl::string_view parameter, Options* options, std::string* error) { for (const auto& pair : ParseGeneratorParameter(parameter)) { if (pair.first == "dllexport_decl") { options->dllexport_decl = pair.second; } else { *error = absl::Substitute("Unknown parameter: $0", pair.first); return false; } } return true; } class ReflectionGenerator : public google::protobuf::compiler::CodeGenerator { bool Generate(const google::protobuf::FileDescriptor* file, const std::string& parameter, google::protobuf::compiler::GeneratorContext* generator_context, std::string* error) const override { std::vector files{file}; return GenerateAll(files, parameter, generator_context, error); } bool GenerateAll(const std::vector& files, const std::string& parameter, google::protobuf::compiler::GeneratorContext* generator_context, std::string* error) const override { Options options; if (!ParseOptions(parameter, &options, error)) { return false; } upb::Arena arena; DefPoolPair pools; absl::flat_hash_set files_seen; for (const auto* file : files) { PopulateDefPool(file, &arena, &pools, &files_seen); upb::FileDefPtr upb_file = pools.GetFile(file->name()); GenerateFile(upb_file, options, generator_context); } return true; } uint64_t GetSupportedFeatures() const override { return FEATURE_PROTO3_OPTIONAL | FEATURE_SUPPORTS_EDITIONS; } google::protobuf::Edition GetMinimumEdition() const override { return google::protobuf::Edition::EDITION_PROTO2; } google::protobuf::Edition GetMaximumEdition() const override { return google::protobuf::Edition::EDITION_2024; } }; } // namespace reflection } // namespace generator } // namespace upb int main(int argc, char** argv) { upb::generator::reflection::ReflectionGenerator generator; return google::protobuf::compiler::PluginMain(argc, argv, &generator); }