#include "Errors.h" #include "string_utils.h" #include "google/protobuf/compiler/plugin.pb.h" #include "google/protobuf/io/zero_copy_stream_impl.h" #include "google/protobuf/text_format.h" #include #include #include #include #include using namespace android::javastream_proto; using namespace google::protobuf; using namespace google::protobuf::compiler; using namespace google::protobuf::io; using namespace std; const int FIELD_TYPE_SHIFT = 32; const uint64_t FIELD_TYPE_DOUBLE = 1L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_FLOAT = 2L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_INT32 = 3L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_INT64 = 4L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_UINT32 = 5L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_UINT64 = 6L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_SINT32 = 7L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_SINT64 = 8L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_FIXED32 = 9L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_FIXED64 = 10L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_SFIXED32 = 11L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_SFIXED64 = 12L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_BOOL = 13L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_STRING = 14L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_BYTES = 15L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_ENUM = 16L << FIELD_TYPE_SHIFT; const uint64_t FIELD_TYPE_OBJECT = 17L << FIELD_TYPE_SHIFT; const int FIELD_COUNT_SHIFT = 40; const uint64_t FIELD_COUNT_SINGLE = 1L << FIELD_COUNT_SHIFT; const uint64_t FIELD_COUNT_REPEATED = 2L << FIELD_COUNT_SHIFT; const uint64_t FIELD_COUNT_PACKED = 5L << FIELD_COUNT_SHIFT; /** * See if this is the file for this request, and not one of the imported ones. */ static bool should_generate_for_file(const CodeGeneratorRequest& request, const string& file) { const int N = request.file_to_generate_size(); for (int i=0; i 0) { result = replace_string(package, '.', '/'); result += '/'; } result += class_name; result += ".java"; return result; } static string indent_more(const string& indent) { return indent + " "; } /** * Write the constants for an enum. */ static void write_enum(stringstream& text, const EnumDescriptorProto& enu, const string& indent) { const int N = enu.value_size(); text << indent << "// enum " << enu.name() << endl; for (int i=0; i"; case FieldDescriptorProto::TYPE_MESSAGE: return field.type_name(); case FieldDescriptorProto::TYPE_BYTES: return "bytes"; case FieldDescriptorProto::TYPE_UINT32: return "uint32"; case FieldDescriptorProto::TYPE_ENUM: return field.type_name(); case FieldDescriptorProto::TYPE_SFIXED32: return "sfixed32"; case FieldDescriptorProto::TYPE_SFIXED64: return "sfixed64"; case FieldDescriptorProto::TYPE_SINT32: return "sint32"; case FieldDescriptorProto::TYPE_SINT64: return "sint64"; default: // won't happen return "void"; } } static uint64_t get_field_id(const FieldDescriptorProto& field) { // Number uint64_t result = (uint32_t)field.number(); // Type switch (field.type()) { case FieldDescriptorProto::TYPE_DOUBLE: result |= FIELD_TYPE_DOUBLE; break; case FieldDescriptorProto::TYPE_FLOAT: result |= FIELD_TYPE_FLOAT; break; case FieldDescriptorProto::TYPE_INT64: result |= FIELD_TYPE_INT64; break; case FieldDescriptorProto::TYPE_UINT64: result |= FIELD_TYPE_UINT64; break; case FieldDescriptorProto::TYPE_INT32: result |= FIELD_TYPE_INT32; break; case FieldDescriptorProto::TYPE_FIXED64: result |= FIELD_TYPE_FIXED64; break; case FieldDescriptorProto::TYPE_FIXED32: result |= FIELD_TYPE_FIXED32; break; case FieldDescriptorProto::TYPE_BOOL: result |= FIELD_TYPE_BOOL; break; case FieldDescriptorProto::TYPE_STRING: result |= FIELD_TYPE_STRING; break; case FieldDescriptorProto::TYPE_MESSAGE: result |= FIELD_TYPE_OBJECT; break; case FieldDescriptorProto::TYPE_BYTES: result |= FIELD_TYPE_BYTES; break; case FieldDescriptorProto::TYPE_UINT32: result |= FIELD_TYPE_UINT32; break; case FieldDescriptorProto::TYPE_ENUM: result |= FIELD_TYPE_ENUM; break; case FieldDescriptorProto::TYPE_SFIXED32: result |= FIELD_TYPE_SFIXED32; break; case FieldDescriptorProto::TYPE_SFIXED64: result |= FIELD_TYPE_SFIXED64; break; case FieldDescriptorProto::TYPE_SINT32: result |= FIELD_TYPE_SINT32; break; case FieldDescriptorProto::TYPE_SINT64: result |= FIELD_TYPE_SINT64; break; default: ; } // Count if (field.options().packed()) { result |= FIELD_COUNT_PACKED; } else if (field.label() == FieldDescriptorProto::LABEL_REPEATED) { result |= FIELD_COUNT_REPEATED; } else { result |= FIELD_COUNT_SINGLE; } return result; } /** * Write a field. */ static void write_field(stringstream& text, const FieldDescriptorProto& field, const string& indent) { string optional_comment = field.label() == FieldDescriptorProto::LABEL_OPTIONAL ? "optional " : ""; string repeated_comment = field.label() == FieldDescriptorProto::LABEL_REPEATED ? "repeated " : ""; string proto_type = get_proto_type(field); string packed_comment = field.options().packed() ? " [packed=true]" : ""; text << indent << "// " << optional_comment << repeated_comment << proto_type << ' ' << field.name() << " = " << field.number() << packed_comment << ';' << endl; text << indent << "public static final long " << make_constant_name(field.name()) << " = 0x"; ios::fmtflags fmt(text.flags()); text << setfill('0') << setw(16) << hex << get_field_id(field); text.flags(fmt); text << "L;" << endl; text << endl; } /** * Write a Message constants class. */ static void write_message(stringstream& text, const DescriptorProto& message, const string& indent) { int N; const string indented = indent_more(indent); text << indent << "// message " << message.name() << endl; text << indent << "public final class " << message.name() << " {" << endl; text << endl; // Enums N = message.enum_type_size(); for (int i=0; i& enums, const vector& messages) { stringstream text; string const package_name = make_java_package(file_descriptor); string const outer_class_name = make_outer_class_name(file_descriptor); text << "// Generated by protoc-gen-javastream. DO NOT MODIFY." << endl; text << "// source: " << file_descriptor.name() << endl << endl; if (package_name.size() > 0) { if (package_name.size() > 0) { text << "package " << package_name << ";" << endl; text << endl; } } // This bit of policy is android api rules specific: Raw proto classes // must never be in the API text << "/** @hide */" << endl; // text << "@android.annotation.TestApi" << endl; if (generate_outer) { text << "public final class " << outer_class_name << " {" << endl; text << endl; } size_t N; const string indented = generate_outer ? indent_more("") : string(); N = enums.size(); for (size_t i=0; iadd_file(); file_response->set_name(filename); file_response->set_content(text.str()); } /** * Write one file per class. Put all of the enums into the "outer" class. */ static void write_multiple_files(CodeGeneratorResponse* response, const FileDescriptorProto& file_descriptor) { // If there is anything to put in the outer class file, create one if (file_descriptor.enum_type_size() > 0) { vector enums; int N = file_descriptor.enum_type_size(); for (int i=0; i messages; write_file(response, file_descriptor, make_file_name(file_descriptor, make_outer_class_name(file_descriptor)), true, enums, messages); } // For each of the message types, make a file int N = file_descriptor.message_type_size(); for (int i=0; i enums; vector messages; messages.push_back(file_descriptor.message_type(i)); write_file(response, file_descriptor, make_file_name(file_descriptor, file_descriptor.message_type(i).name()), false, enums, messages); } } static void write_single_file(CodeGeneratorResponse* response, const FileDescriptorProto& file_descriptor) { int N; vector enums; N = file_descriptor.enum_type_size(); for (int i=0; i messages; N = file_descriptor.message_type_size(); for (int i=0; i