// 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 #include "google/protobuf/descriptor.pb.h" #include "google/protobuf/compiler/code_generator.h" #include "google/protobuf/compiler/plugin.h" #include "google/protobuf/descriptor.h" #include "upb/protos_generator/gen_enums.h" #include "upb/protos_generator/gen_extensions.h" #include "upb/protos_generator/gen_messages.h" #include "upb/protos_generator/gen_utils.h" #include "upb/protos_generator/names.h" #include "upb/protos_generator/output.h" #include "upb/upbc/file_layout.h" namespace protos_generator { namespace { namespace protoc = ::google::protobuf::compiler; namespace protobuf = ::google::protobuf; using FileDescriptor = ::google::protobuf::FileDescriptor; void WriteSource(const protobuf::FileDescriptor* file, Output& output, bool fasttable_enabled); void WriteHeader(const protobuf::FileDescriptor* file, Output& output); void WriteForwardingHeader(const protobuf::FileDescriptor* file, Output& output); void WriteMessageImplementations(const protobuf::FileDescriptor* file, Output& output); void WriteTypedefForwardingHeader( const protobuf::FileDescriptor* file, const std::vector& file_messages, Output& output); void WriteHeaderMessageForwardDecls( const protobuf::FileDescriptor* file, Output& output); class Generator : public protoc::CodeGenerator { public: ~Generator() override {} bool Generate(const protobuf::FileDescriptor* file, const std::string& parameter, protoc::GeneratorContext* context, std::string* error) const override; uint64_t GetSupportedFeatures() const override { return FEATURE_PROTO3_OPTIONAL; } }; bool Generator::Generate(const protobuf::FileDescriptor* file, const std::string& parameter, protoc::GeneratorContext* context, std::string* error) const { bool fasttable_enabled = false; std::vector> params; google::protobuf::compiler::ParseGeneratorParameter(parameter, ¶ms); for (const auto& pair : params) { if (pair.first == "fasttable") { fasttable_enabled = true; } else { *error = "Unknown parameter: " + pair.first; return false; } } // Write model.upb.fwd.h Output forwarding_header_output( context->Open(ForwardingHeaderFilename(file))); WriteForwardingHeader(file, forwarding_header_output); // Write model.upb.proto.h Output header_output(context->Open(CppHeaderFilename(file))); WriteHeader(file, header_output); // Write model.upb.proto.cc Output cc_output(context->Open(CppSourceFilename(file))); WriteSource(file, cc_output, fasttable_enabled); return true; } // The forwarding header defines Access/Proxy/CProxy for message classes // used to include when referencing dependencies to prevent transitive // dependency headers from being included. void WriteForwardingHeader(const protobuf::FileDescriptor* file, Output& output) { EmitFileWarning(file, output); output( R"cc( #ifndef $0_UPB_FWD_H_ #define $0_UPB_FWD_H_ )cc", ToPreproc(file->name())); output("\n"); for (int i = 0; i < file->public_dependency_count(); ++i) { output("#include \"$0\"\n", ForwardingHeaderFilename(file->public_dependency(i))); } if (file->public_dependency_count() > 0) { output("\n"); } const std::vector this_file_messages = SortedMessages(file); WriteTypedefForwardingHeader(file, this_file_messages, output); output("#endif /* $0_UPB_FWD_H_ */\n", ToPreproc(file->name())); } void WriteHeader(const protobuf::FileDescriptor* file, Output& output) { EmitFileWarning(file, output); output( R"cc( #ifndef $0_UPB_PROTO_H_ #define $0_UPB_PROTO_H_ #include "upb/protos/protos.h" #include "upb/protos/protos_internal.h" #include "upb/protos/repeated_field.h" #include "absl/strings/string_view.h" #include "absl/status/statusor.h" )cc", ToPreproc(file->name())); // Import headers for proto public dependencies. for (int i = 0; i < file->public_dependency_count(); i++) { if (i == 0) { output("// Public Imports.\n"); } output("#include \"$0\"\n", CppHeaderFilename(file->public_dependency(i))); if (i == file->public_dependency_count() - 1) { output("\n"); } } output("#include \"upb/upb/port/def.inc\"\n"); const std::vector this_file_messages = SortedMessages(file); const std::vector this_file_exts = SortedExtensions(file); if (!this_file_messages.empty()) { output("\n"); } WriteHeaderMessageForwardDecls(file, output); WriteStartNamespace(file, output); std::vector this_file_enums = SortedEnums(file); // Write Class and Enums. WriteEnumDeclarations(this_file_enums, output); output("\n"); for (auto message : this_file_messages) { WriteMessageClassDeclarations(message, this_file_exts, this_file_enums, output); } output("\n"); WriteExtensionIdentifiersHeader(this_file_exts, output); output("\n"); WriteEndNamespace(file, output); output("\n#include \"upb/upb/port/undef.inc\"\n\n"); // End of "C" section. output("#endif /* $0_UPB_PROTO_H_ */\n", ToPreproc(file->name())); } // Writes a .upb.cc source file. void WriteSource(const protobuf::FileDescriptor* file, Output& output, bool fasttable_enabled) { EmitFileWarning(file, output); output( R"cc( #include #include "absl/strings/string_view.h" #include "upb/protos/protos.h" #include "$0" )cc", CppHeaderFilename(file)); for (int i = 0; i < file->dependency_count(); i++) { output("#include \"$0\"\n", CppHeaderFilename(file->dependency(i))); } output("#include \"upb/upb/port/def.inc\"\n"); WriteStartNamespace(file, output); WriteMessageImplementations(file, output); const std::vector this_file_exts = SortedExtensions(file); WriteExtensionIdentifiers(this_file_exts, output); WriteEndNamespace(file, output); output("#include \"upb/upb/port/undef.inc\"\n\n"); } void WriteMessageImplementations(const protobuf::FileDescriptor* file, Output& output) { const std::vector file_exts = SortedExtensions(file); const std::vector this_file_messages = SortedMessages(file); for (auto message : this_file_messages) { WriteMessageImplementation(message, file_exts, output); } } void WriteTypedefForwardingHeader( const protobuf::FileDescriptor* file, const std::vector& file_messages, Output& output) { WriteStartNamespace(file, output); // Forward-declare types defined in this file. for (auto message : file_messages) { output( R"cc( class $0; namespace internal { class $0Access; class $0Proxy; class $0CProxy; } // namespace internal )cc", ClassName(message)); } output("\n"); WriteEndNamespace(file, output); } /// Writes includes for upb C minitables and fwd.h for transitive typedefs. void WriteHeaderMessageForwardDecls( const protobuf::FileDescriptor* file, Output& output) { // Import forward-declaration of types defined in this file. output("#include \"$0\"\n", UpbCFilename(file)); output("#include \"$0\"\n", ForwardingHeaderFilename(file)); // Import forward-declaration of types in dependencies. for (int i = 0; i < file->dependency_count(); ++i) { output("#include \"$0\"\n", ForwardingHeaderFilename(file->dependency(i))); } output("\n"); } } // namespace } // namespace protos_generator int main(int argc, char** argv) { protos_generator::Generator generator_cc; return google::protobuf::compiler::PluginMain(argc, argv, &generator_cc); }