| // Copyright (c) 2024 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 "upb_generator/common/cpp_to_upb_def.h" |
| |
| #include <string> |
| |
| #include "google/protobuf/descriptor.upb.h" |
| #include "absl/log/absl_check.h" |
| #include "google/protobuf/descriptor.h" |
| #include "upb/base/status.hpp" |
| #include "upb/mem/arena.hpp" |
| #include "upb/mini_table/field.h" |
| #include "upb/reflection/def.hpp" |
| |
| namespace upb::generator { |
| |
| using google::protobuf::Descriptor; |
| using google::protobuf::EnumDescriptor; |
| using google::protobuf::FieldDescriptor; |
| using google::protobuf::FileDescriptor; |
| |
| // Internal helper that takes a filedesc and emits a upb proto; |
| // used for defpool tracking and dependency management |
| google_protobuf_FileDescriptorProto* ToUpbProto(const FileDescriptor* file, |
| upb::Arena* arena) { |
| google::protobuf::FileDescriptorProto proto; |
| file->CopyTo(&proto); |
| std::string serialized_proto = proto.SerializeAsString(); |
| google_protobuf_FileDescriptorProto* upb_proto = google_protobuf_FileDescriptorProto_parse( |
| serialized_proto.data(), serialized_proto.size(), arena->ptr()); |
| ABSL_CHECK(upb_proto) << "Failed to parse proto"; |
| return upb_proto; |
| } |
| |
| void AddFile(const FileDescriptor* file, upb::DefPool* pool) { |
| // Avoid adding the same file twice. |
| if (pool->FindFileByName(file->name().c_str())) return; |
| |
| // Like a google::protobuf::DescriptorPool, a upb::DefPool requires that all |
| // dependencies are added first. |
| for (int i = 0; i < file->dependency_count(); i++) { |
| AddFile(file->dependency(i), pool); |
| } |
| |
| upb::Arena tmp_arena; |
| upb::Status status; |
| ABSL_CHECK(pool->AddFile(ToUpbProto(file, &tmp_arena), &status)) |
| << status.error_message(); |
| } |
| |
| upb::MessageDefPtr FindMessageDef(upb::DefPool& pool, |
| const Descriptor* descriptor) { |
| const std::string& name = descriptor->full_name(); |
| upb::MessageDefPtr message_def = pool.FindMessageByName(name.c_str()); |
| ABSL_CHECK(message_def) << "No message named " << name; |
| return message_def; |
| } |
| |
| upb::EnumDefPtr FindEnumDef(upb::DefPool& pool, |
| const EnumDescriptor* enum_descriptor) { |
| const std::string& name = enum_descriptor->full_name(); |
| upb::EnumDefPtr enum_def = pool.FindEnumByName(name.c_str()); |
| ABSL_CHECK(enum_def) << "No enum named " << name; |
| return enum_def; |
| } |
| |
| upb::FieldDefPtr FindBaseFieldDef(upb::DefPool& pool, |
| const FieldDescriptor* field) { |
| ABSL_CHECK(!field->is_extension()); |
| upb::MessageDefPtr message_def = |
| FindMessageDef(pool, field->containing_type()); |
| upb::FieldDefPtr field_def = message_def.FindFieldByNumber(field->number()); |
| ABSL_CHECK(field_def) << "No field with number " << field->number() |
| << " in message " << message_def.full_name(); |
| return field_def; |
| } |
| |
| upb::FieldDefPtr FindExtensionDef(upb::DefPool& pool, |
| const FieldDescriptor* field) { |
| ABSL_CHECK(field->is_extension()); |
| return pool.FindExtensionByName(field->full_name().c_str()); |
| } |
| |
| const FieldDescriptor* FindFieldDescriptor( |
| const Descriptor* message, const upb_MiniTableField* field_def) { |
| int field_number = upb_MiniTableField_Number(field_def); |
| const FieldDescriptor* field = message->FindFieldByNumber(field_number); |
| ABSL_CHECK(field) << "No field in message " << message->full_name() |
| << " with number " << field_number; |
| return field; |
| } |
| |
| } // namespace upb::generator |