google-protobuf 3.22.2 → 4.30.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (69) hide show
  1. checksums.yaml +4 -4
  2. data/ext/google/protobuf_c/Rakefile +3 -0
  3. data/ext/google/protobuf_c/convert.c +60 -86
  4. data/ext/google/protobuf_c/convert.h +3 -28
  5. data/ext/google/protobuf_c/defs.c +692 -72
  6. data/ext/google/protobuf_c/defs.h +3 -28
  7. data/ext/google/protobuf_c/extconf.rb +14 -1
  8. data/ext/google/protobuf_c/glue.c +135 -0
  9. data/ext/google/protobuf_c/map.c +89 -45
  10. data/ext/google/protobuf_c/map.h +12 -30
  11. data/ext/google/protobuf_c/message.c +169 -169
  12. data/ext/google/protobuf_c/message.h +11 -33
  13. data/ext/google/protobuf_c/protobuf.c +65 -188
  14. data/ext/google/protobuf_c/protobuf.h +27 -39
  15. data/ext/google/protobuf_c/repeated_field.c +72 -38
  16. data/ext/google/protobuf_c/repeated_field.h +11 -29
  17. data/ext/google/protobuf_c/ruby-upb.c +15064 -11220
  18. data/ext/google/protobuf_c/ruby-upb.h +10643 -5403
  19. data/ext/google/protobuf_c/shared_convert.c +69 -0
  20. data/ext/google/protobuf_c/shared_convert.h +26 -0
  21. data/ext/google/protobuf_c/shared_message.c +37 -0
  22. data/ext/google/protobuf_c/shared_message.h +21 -0
  23. data/ext/google/protobuf_c/third_party/utf8_range/utf8_range.c +207 -0
  24. data/ext/google/protobuf_c/third_party/utf8_range/utf8_range.h +9 -8
  25. data/ext/google/protobuf_c/third_party/utf8_range/utf8_range_neon.inc +117 -0
  26. data/ext/google/protobuf_c/third_party/utf8_range/utf8_range_sse.inc +272 -0
  27. data/ext/google/protobuf_c/wrap_memcpy.c +3 -26
  28. data/lib/google/protobuf/any_pb.rb +6 -8
  29. data/lib/google/protobuf/api_pb.rb +6 -26
  30. data/lib/google/protobuf/descriptor_pb.rb +21 -252
  31. data/lib/google/protobuf/duration_pb.rb +6 -8
  32. data/lib/google/protobuf/empty_pb.rb +6 -6
  33. data/lib/google/protobuf/ffi/descriptor.rb +175 -0
  34. data/lib/google/protobuf/ffi/descriptor_pool.rb +77 -0
  35. data/lib/google/protobuf/ffi/enum_descriptor.rb +183 -0
  36. data/lib/google/protobuf/ffi/ffi.rb +214 -0
  37. data/lib/google/protobuf/ffi/field_descriptor.rb +340 -0
  38. data/lib/google/protobuf/ffi/file_descriptor.rb +59 -0
  39. data/lib/google/protobuf/ffi/internal/arena.rb +60 -0
  40. data/lib/google/protobuf/ffi/internal/convert.rb +292 -0
  41. data/lib/google/protobuf/ffi/internal/pointer_helper.rb +35 -0
  42. data/lib/google/protobuf/ffi/internal/type_safety.rb +25 -0
  43. data/lib/google/protobuf/ffi/map.rb +433 -0
  44. data/lib/google/protobuf/ffi/message.rb +783 -0
  45. data/lib/google/protobuf/ffi/method_descriptor.rb +124 -0
  46. data/lib/google/protobuf/ffi/object_cache.rb +30 -0
  47. data/lib/google/protobuf/ffi/oneof_descriptor.rb +107 -0
  48. data/lib/google/protobuf/ffi/repeated_field.rb +411 -0
  49. data/lib/google/protobuf/ffi/service_descriptor.rb +117 -0
  50. data/lib/google/protobuf/field_mask_pb.rb +6 -7
  51. data/lib/google/protobuf/internal/object_cache.rb +99 -0
  52. data/lib/google/protobuf/message_exts.rb +3 -26
  53. data/lib/google/protobuf/plugin_pb.rb +6 -31
  54. data/lib/google/protobuf/repeated_field.rb +7 -31
  55. data/lib/google/protobuf/source_context_pb.rb +6 -7
  56. data/lib/google/protobuf/struct_pb.rb +6 -23
  57. data/lib/google/protobuf/timestamp_pb.rb +6 -8
  58. data/lib/google/protobuf/type_pb.rb +6 -71
  59. data/lib/google/protobuf/well_known_types.rb +5 -34
  60. data/lib/google/protobuf/wrappers_pb.rb +6 -31
  61. data/lib/google/protobuf.rb +27 -45
  62. data/lib/google/protobuf_ffi.rb +52 -0
  63. data/lib/google/protobuf_native.rb +19 -0
  64. data/lib/google/tasks/ffi.rake +100 -0
  65. metadata +92 -9
  66. data/ext/google/protobuf_c/third_party/utf8_range/naive.c +0 -92
  67. data/ext/google/protobuf_c/third_party/utf8_range/range2-neon.c +0 -157
  68. data/ext/google/protobuf_c/third_party/utf8_range/range2-sse.c +0 -170
  69. data/lib/google/protobuf/descriptor_dsl.rb +0 -465
@@ -1,36 +1,12 @@
1
1
  // Protocol Buffers - Google's data interchange format
2
2
  // Copyright 2014 Google Inc. All rights reserved.
3
- // https://developers.google.com/protocol-buffers/
4
3
  //
5
- // Redistribution and use in source and binary forms, with or without
6
- // modification, are permitted provided that the following conditions are
7
- // met:
8
- //
9
- // * Redistributions of source code must retain the above copyright
10
- // notice, this list of conditions and the following disclaimer.
11
- // * Redistributions in binary form must reproduce the above
12
- // copyright notice, this list of conditions and the following disclaimer
13
- // in the documentation and/or other materials provided with the
14
- // distribution.
15
- // * Neither the name of Google Inc. nor the names of its
16
- // contributors may be used to endorse or promote products derived from
17
- // this software without specific prior written permission.
18
- //
19
- // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20
- // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21
- // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22
- // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23
- // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24
- // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25
- // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26
- // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27
- // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28
- // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29
- // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
4
+ // Use of this source code is governed by a BSD-style
5
+ // license that can be found in the LICENSE file or at
6
+ // https://developers.google.com/open-source/licenses/bsd
30
7
 
31
8
  #include <ctype.h>
32
9
  #include <errno.h>
33
- #include <ruby/version.h>
34
10
 
35
11
  #include "convert.h"
36
12
  #include "message.h"
@@ -46,6 +22,9 @@ static VALUE get_enumdef_obj(VALUE descriptor_pool, const upb_EnumDef* def);
46
22
  static VALUE get_fielddef_obj(VALUE descriptor_pool, const upb_FieldDef* def);
47
23
  static VALUE get_filedef_obj(VALUE descriptor_pool, const upb_FileDef* def);
48
24
  static VALUE get_oneofdef_obj(VALUE descriptor_pool, const upb_OneofDef* def);
25
+ static VALUE get_servicedef_obj(VALUE descriptor_pool,
26
+ const upb_ServiceDef* def);
27
+ static VALUE get_methoddef_obj(VALUE descriptor_pool, const upb_MethodDef* def);
49
28
 
50
29
  // A distinct object that is not accessible from Ruby. We use this as a
51
30
  // constructor argument to enforce that certain objects cannot be created from
@@ -67,12 +46,14 @@ static VALUE rb_str_maybe_null(const char* s) {
67
46
  }
68
47
  return rb_str_new2(s);
69
48
  }
70
-
49
+ static ID options_instancevar_interned;
71
50
  // -----------------------------------------------------------------------------
72
51
  // DescriptorPool.
73
52
  // -----------------------------------------------------------------------------
74
53
 
75
54
  typedef struct {
55
+ // IMPORTANT: WB_PROTECTED objects must only use the RB_OBJ_WRITE()
56
+ // macro to update VALUE references, as to trigger write barriers.
76
57
  VALUE def_to_descriptor; // Hash table of def* -> Ruby descriptor.
77
58
  upb_DefPool* symtab;
78
59
  } DescriptorPool;
@@ -97,7 +78,7 @@ static void DescriptorPool_free(void* _self) {
97
78
  static const rb_data_type_t DescriptorPool_type = {
98
79
  "Google::Protobuf::DescriptorPool",
99
80
  {DescriptorPool_mark, DescriptorPool_free, NULL},
100
- .flags = RUBY_TYPED_FREE_IMMEDIATELY,
81
+ .flags = RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED,
101
82
  };
102
83
 
103
84
  static DescriptorPool* ruby_to_DescriptorPool(VALUE val) {
@@ -125,11 +106,9 @@ static VALUE DescriptorPool_alloc(VALUE klass) {
125
106
  self->def_to_descriptor = Qnil;
126
107
  ret = TypedData_Wrap_Struct(klass, &DescriptorPool_type, self);
127
108
 
128
- self->def_to_descriptor = rb_hash_new();
109
+ RB_OBJ_WRITE(ret, &self->def_to_descriptor, rb_hash_new());
129
110
  self->symtab = upb_DefPool_New();
130
- ObjectCache_Add(self->symtab, ret);
131
-
132
- return ret;
111
+ return ObjectCache_TryAdd(self->symtab, ret);
133
112
  }
134
113
 
135
114
  /*
@@ -167,25 +146,37 @@ VALUE DescriptorPool_add_serialized_file(VALUE _self,
167
146
  * call-seq:
168
147
  * DescriptorPool.lookup(name) => descriptor
169
148
  *
170
- * Finds a Descriptor or EnumDescriptor by name and returns it, or nil if none
171
- * exists with the given name.
149
+ * Finds a Descriptor, EnumDescriptor, FieldDescriptor or ServiceDescriptor by
150
+ * name and returns it, or nil if none exists with the given name.
172
151
  */
173
152
  static VALUE DescriptorPool_lookup(VALUE _self, VALUE name) {
174
153
  DescriptorPool* self = ruby_to_DescriptorPool(_self);
175
154
  const char* name_str = get_str(name);
176
155
  const upb_MessageDef* msgdef;
177
156
  const upb_EnumDef* enumdef;
157
+ const upb_FieldDef* fielddef;
158
+ const upb_ServiceDef* servicedef;
178
159
 
179
160
  msgdef = upb_DefPool_FindMessageByName(self->symtab, name_str);
180
161
  if (msgdef) {
181
162
  return get_msgdef_obj(_self, msgdef);
182
163
  }
183
164
 
165
+ fielddef = upb_DefPool_FindExtensionByName(self->symtab, name_str);
166
+ if (fielddef) {
167
+ return get_fielddef_obj(_self, fielddef);
168
+ }
169
+
184
170
  enumdef = upb_DefPool_FindEnumByName(self->symtab, name_str);
185
171
  if (enumdef) {
186
172
  return get_enumdef_obj(_self, enumdef);
187
173
  }
188
174
 
175
+ servicedef = upb_DefPool_FindServiceByName(self->symtab, name_str);
176
+ if (servicedef) {
177
+ return get_servicedef_obj(_self, servicedef);
178
+ }
179
+
189
180
  return Qnil;
190
181
  }
191
182
 
@@ -215,6 +206,7 @@ static void DescriptorPool_register(VALUE module) {
215
206
 
216
207
  rb_gc_register_address(&generated_pool);
217
208
  generated_pool = rb_class_new_instance(0, NULL, klass);
209
+ options_instancevar_interned = rb_intern("options");
218
210
  }
219
211
 
220
212
  // -----------------------------------------------------------------------------
@@ -223,6 +215,8 @@ static void DescriptorPool_register(VALUE module) {
223
215
 
224
216
  typedef struct {
225
217
  const upb_MessageDef* msgdef;
218
+ // IMPORTANT: WB_PROTECTED objects must only use the RB_OBJ_WRITE()
219
+ // macro to update VALUE references, as to trigger write barriers.
226
220
  VALUE klass;
227
221
  VALUE descriptor_pool;
228
222
  } Descriptor;
@@ -238,7 +232,7 @@ static void Descriptor_mark(void* _self) {
238
232
  static const rb_data_type_t Descriptor_type = {
239
233
  "Google::Protobuf::Descriptor",
240
234
  {Descriptor_mark, RUBY_DEFAULT_FREE, NULL},
241
- .flags = RUBY_TYPED_FREE_IMMEDIATELY,
235
+ .flags = RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED,
242
236
  };
243
237
 
244
238
  static Descriptor* ruby_to_Descriptor(VALUE val) {
@@ -247,6 +241,48 @@ static Descriptor* ruby_to_Descriptor(VALUE val) {
247
241
  return ret;
248
242
  }
249
243
 
244
+ // Decode and return a frozen instance of a Descriptor Option for the given pool
245
+ static VALUE decode_options(VALUE self, const char* option_type, int size,
246
+ const char* bytes, VALUE descriptor_pool) {
247
+ VALUE options_rb = rb_ivar_get(self, options_instancevar_interned);
248
+ if (options_rb != Qnil) {
249
+ return options_rb;
250
+ }
251
+
252
+ static const char* prefix = "google.protobuf.";
253
+ char fullname
254
+ [/*strlen(prefix)*/ 16 +
255
+ /*strln(longest option type supported e.g. "MessageOptions")*/ 14 +
256
+ /*null terminator*/ 1];
257
+
258
+ snprintf(fullname, sizeof(fullname), "%s%s", prefix, option_type);
259
+ const upb_MessageDef* msgdef = upb_DefPool_FindMessageByName(
260
+ ruby_to_DescriptorPool(descriptor_pool)->symtab, fullname);
261
+ if (!msgdef) {
262
+ rb_raise(rb_eRuntimeError, "Cannot find %s in DescriptorPool", option_type);
263
+ }
264
+
265
+ VALUE desc_rb = get_msgdef_obj(descriptor_pool, msgdef);
266
+ const Descriptor* desc = ruby_to_Descriptor(desc_rb);
267
+
268
+ options_rb = Message_decode_bytes(size, bytes, 0, desc->klass, false);
269
+
270
+ // Strip features from the options proto to keep it internal.
271
+ const upb_MessageDef* decoded_desc = NULL;
272
+ upb_Message* options = Message_GetMutable(options_rb, &decoded_desc);
273
+ PBRUBY_ASSERT(options != NULL);
274
+ PBRUBY_ASSERT(decoded_desc == msgdef);
275
+ const upb_FieldDef* field =
276
+ upb_MessageDef_FindFieldByName(decoded_desc, "features");
277
+ PBRUBY_ASSERT(field != NULL);
278
+ upb_Message_ClearFieldByDef(options, field);
279
+
280
+ Message_freeze(options_rb);
281
+
282
+ rb_ivar_set(self, options_instancevar_interned, options_rb);
283
+ return options_rb;
284
+ }
285
+
250
286
  /*
251
287
  * call-seq:
252
288
  * Descriptor.new => descriptor
@@ -280,7 +316,7 @@ static VALUE Descriptor_initialize(VALUE _self, VALUE cookie,
280
316
  "Descriptor objects may not be created from Ruby.");
281
317
  }
282
318
 
283
- self->descriptor_pool = descriptor_pool;
319
+ RB_OBJ_WRITE(_self, &self->descriptor_pool, descriptor_pool);
284
320
  self->msgdef = (const upb_MessageDef*)NUM2ULL(ptr);
285
321
 
286
322
  return Qnil;
@@ -390,11 +426,52 @@ static VALUE Descriptor_lookup_oneof(VALUE _self, VALUE name) {
390
426
  static VALUE Descriptor_msgclass(VALUE _self) {
391
427
  Descriptor* self = ruby_to_Descriptor(_self);
392
428
  if (self->klass == Qnil) {
393
- self->klass = build_class_from_descriptor(_self);
429
+ RB_OBJ_WRITE(_self, &self->klass, build_class_from_descriptor(_self));
394
430
  }
395
431
  return self->klass;
396
432
  }
397
433
 
434
+ /*
435
+ * call-seq:
436
+ * Descriptor.options => options
437
+ *
438
+ * Returns the `MessageOptions` for this `Descriptor`.
439
+ */
440
+ static VALUE Descriptor_options(VALUE _self) {
441
+ Descriptor* self = ruby_to_Descriptor(_self);
442
+ const google_protobuf_MessageOptions* opts =
443
+ upb_MessageDef_Options(self->msgdef);
444
+ upb_Arena* arena = upb_Arena_New();
445
+ size_t size;
446
+ char* serialized =
447
+ google_protobuf_MessageOptions_serialize(opts, arena, &size);
448
+ VALUE message_options = decode_options(_self, "MessageOptions", size,
449
+ serialized, self->descriptor_pool);
450
+ upb_Arena_Free(arena);
451
+ return message_options;
452
+ }
453
+
454
+ /*
455
+ * call-seq:
456
+ * Descriptor.to_proto => DescriptorProto
457
+ *
458
+ * Returns the `DescriptorProto` of this `Descriptor`.
459
+ */
460
+ static VALUE Descriptor_to_proto(VALUE _self) {
461
+ Descriptor* self = ruby_to_Descriptor(_self);
462
+ upb_Arena* arena = upb_Arena_New();
463
+ google_protobuf_DescriptorProto* proto =
464
+ upb_MessageDef_ToProto(self->msgdef, arena);
465
+ size_t size;
466
+ const char* serialized =
467
+ google_protobuf_DescriptorProto_serialize(proto, arena, &size);
468
+ VALUE proto_class = rb_path2class("Google::Protobuf::DescriptorProto");
469
+ VALUE proto_rb =
470
+ Message_decode_bytes(size, serialized, 0, proto_class, false);
471
+ upb_Arena_Free(arena);
472
+ return proto_rb;
473
+ }
474
+
398
475
  static void Descriptor_register(VALUE module) {
399
476
  VALUE klass = rb_define_class_under(module, "Descriptor", rb_cObject);
400
477
  rb_define_alloc_func(klass, Descriptor_alloc);
@@ -406,6 +483,8 @@ static void Descriptor_register(VALUE module) {
406
483
  rb_define_method(klass, "msgclass", Descriptor_msgclass, 0);
407
484
  rb_define_method(klass, "name", Descriptor_name, 0);
408
485
  rb_define_method(klass, "file_descriptor", Descriptor_file_descriptor, 0);
486
+ rb_define_method(klass, "options", Descriptor_options, 0);
487
+ rb_define_method(klass, "to_proto", Descriptor_to_proto, 0);
409
488
  rb_include_module(klass, rb_mEnumerable);
410
489
  rb_gc_register_address(&cDescriptor);
411
490
  cDescriptor = klass;
@@ -417,6 +496,8 @@ static void Descriptor_register(VALUE module) {
417
496
 
418
497
  typedef struct {
419
498
  const upb_FileDef* filedef;
499
+ // IMPORTANT: WB_PROTECTED objects must only use the RB_OBJ_WRITE()
500
+ // macro to update VALUE references, as to trigger write barriers.
420
501
  VALUE descriptor_pool; // Owns the upb_FileDef.
421
502
  } FileDescriptor;
422
503
 
@@ -430,7 +511,7 @@ static void FileDescriptor_mark(void* _self) {
430
511
  static const rb_data_type_t FileDescriptor_type = {
431
512
  "Google::Protobuf::FileDescriptor",
432
513
  {FileDescriptor_mark, RUBY_DEFAULT_FREE, NULL},
433
- .flags = RUBY_TYPED_FREE_IMMEDIATELY,
514
+ .flags = RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED,
434
515
  };
435
516
 
436
517
  static FileDescriptor* ruby_to_FileDescriptor(VALUE val) {
@@ -451,7 +532,7 @@ static VALUE FileDescriptor_alloc(VALUE klass) {
451
532
  * call-seq:
452
533
  * FileDescriptor.new => file
453
534
  *
454
- * Returns a new file descriptor. The syntax must be set before it's passed
535
+ * Returns a new file descriptor. May
455
536
  * to a builder.
456
537
  */
457
538
  static VALUE FileDescriptor_initialize(VALUE _self, VALUE cookie,
@@ -463,7 +544,7 @@ static VALUE FileDescriptor_initialize(VALUE _self, VALUE cookie,
463
544
  "Descriptor objects may not be created from Ruby.");
464
545
  }
465
546
 
466
- self->descriptor_pool = descriptor_pool;
547
+ RB_OBJ_WRITE(_self, &self->descriptor_pool, descriptor_pool);
467
548
  self->filedef = (const upb_FileDef*)NUM2ULL(ptr);
468
549
 
469
550
  return Qnil;
@@ -483,24 +564,44 @@ static VALUE FileDescriptor_name(VALUE _self) {
483
564
 
484
565
  /*
485
566
  * call-seq:
486
- * FileDescriptor.syntax => syntax
567
+ * FileDescriptor.options => options
487
568
  *
488
- * Returns this file descriptors syntax.
569
+ * Returns the `FileOptions` for this `FileDescriptor`.
570
+ */
571
+ static VALUE FileDescriptor_options(VALUE _self) {
572
+ FileDescriptor* self = ruby_to_FileDescriptor(_self);
573
+ const google_protobuf_FileOptions* opts = upb_FileDef_Options(self->filedef);
574
+ upb_Arena* arena = upb_Arena_New();
575
+ size_t size;
576
+ char* serialized = google_protobuf_FileOptions_serialize(opts, arena, &size);
577
+ VALUE file_options = decode_options(_self, "FileOptions", size, serialized,
578
+ self->descriptor_pool);
579
+ upb_Arena_Free(arena);
580
+ return file_options;
581
+ }
582
+
583
+ /*
584
+ * call-seq:
585
+ * FileDescriptor.to_proto => FileDescriptorProto
489
586
  *
490
- * Valid syntax versions are:
491
- * :proto2 or :proto3.
587
+ * Returns the `FileDescriptorProto` of this `FileDescriptor`.
492
588
  */
493
- static VALUE FileDescriptor_syntax(VALUE _self) {
589
+ static VALUE FileDescriptor_to_proto(VALUE _self) {
494
590
  FileDescriptor* self = ruby_to_FileDescriptor(_self);
591
+ upb_Arena* arena = upb_Arena_New();
592
+ google_protobuf_FileDescriptorProto* file_proto =
593
+ upb_FileDef_ToProto(self->filedef, arena);
495
594
 
496
- switch (upb_FileDef_Syntax(self->filedef)) {
497
- case kUpb_Syntax_Proto3:
498
- return ID2SYM(rb_intern("proto3"));
499
- case kUpb_Syntax_Proto2:
500
- return ID2SYM(rb_intern("proto2"));
501
- default:
502
- return Qnil;
503
- }
595
+ size_t size;
596
+ const char* serialized =
597
+ google_protobuf_FileDescriptorProto_serialize(file_proto, arena, &size);
598
+
599
+ VALUE file_proto_class =
600
+ rb_path2class("Google::Protobuf::FileDescriptorProto");
601
+ VALUE proto_rb =
602
+ Message_decode_bytes(size, serialized, 0, file_proto_class, false);
603
+ upb_Arena_Free(arena);
604
+ return proto_rb;
504
605
  }
505
606
 
506
607
  static void FileDescriptor_register(VALUE module) {
@@ -508,7 +609,8 @@ static void FileDescriptor_register(VALUE module) {
508
609
  rb_define_alloc_func(klass, FileDescriptor_alloc);
509
610
  rb_define_method(klass, "initialize", FileDescriptor_initialize, 3);
510
611
  rb_define_method(klass, "name", FileDescriptor_name, 0);
511
- rb_define_method(klass, "syntax", FileDescriptor_syntax, 0);
612
+ rb_define_method(klass, "options", FileDescriptor_options, 0);
613
+ rb_define_method(klass, "to_proto", FileDescriptor_to_proto, 0);
512
614
  rb_gc_register_address(&cFileDescriptor);
513
615
  cFileDescriptor = klass;
514
616
  }
@@ -519,6 +621,8 @@ static void FileDescriptor_register(VALUE module) {
519
621
 
520
622
  typedef struct {
521
623
  const upb_FieldDef* fielddef;
624
+ // IMPORTANT: WB_PROTECTED objects must only use the RB_OBJ_WRITE()
625
+ // macro to update VALUE references, as to trigger write barriers.
522
626
  VALUE descriptor_pool; // Owns the upb_FieldDef.
523
627
  } FieldDescriptor;
524
628
 
@@ -532,7 +636,7 @@ static void FieldDescriptor_mark(void* _self) {
532
636
  static const rb_data_type_t FieldDescriptor_type = {
533
637
  "Google::Protobuf::FieldDescriptor",
534
638
  {FieldDescriptor_mark, RUBY_DEFAULT_FREE, NULL},
535
- .flags = RUBY_TYPED_FREE_IMMEDIATELY,
639
+ .flags = RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED,
536
640
  };
537
641
 
538
642
  static FieldDescriptor* ruby_to_FieldDescriptor(VALUE val) {
@@ -557,7 +661,7 @@ static VALUE FieldDescriptor_alloc(VALUE klass) {
557
661
 
558
662
  /*
559
663
  * call-seq:
560
- * EnumDescriptor.new(c_only_cookie, pool, ptr) => EnumDescriptor
664
+ * FieldDescriptor.new(c_only_cookie, pool, ptr) => FieldDescriptor
561
665
  *
562
666
  * Creates a descriptor wrapper object. May only be called from C.
563
667
  */
@@ -570,7 +674,7 @@ static VALUE FieldDescriptor_initialize(VALUE _self, VALUE cookie,
570
674
  "Descriptor objects may not be created from Ruby.");
571
675
  }
572
676
 
573
- self->descriptor_pool = descriptor_pool;
677
+ RB_OBJ_WRITE(_self, &self->descriptor_pool, descriptor_pool);
574
678
  self->fielddef = (const upb_FieldDef*)NUM2ULL(ptr);
575
679
 
576
680
  return Qnil;
@@ -595,7 +699,7 @@ upb_CType ruby_to_fieldtype(VALUE type) {
595
699
 
596
700
  #define CONVERT(upb, ruby) \
597
701
  if (SYM2ID(type) == rb_intern(#ruby)) { \
598
- return kUpb_CType_##upb; \
702
+ return kUpb_CType_##upb; \
599
703
  }
600
704
 
601
705
  CONVERT(Float, float);
@@ -618,7 +722,7 @@ upb_CType ruby_to_fieldtype(VALUE type) {
618
722
 
619
723
  static VALUE descriptortype_to_ruby(upb_FieldType type) {
620
724
  switch (type) {
621
- #define CONVERT(upb, ruby) \
725
+ #define CONVERT(upb, ruby) \
622
726
  case kUpb_FieldType_##upb: \
623
727
  return ID2SYM(rb_intern(#ruby));
624
728
  CONVERT(Float, float);
@@ -668,7 +772,7 @@ static VALUE FieldDescriptor__type(VALUE _self) {
668
772
  static VALUE FieldDescriptor_default(VALUE _self) {
669
773
  FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
670
774
  const upb_FieldDef* f = self->fielddef;
671
- upb_MessageValue default_val = {0};
775
+ upb_MessageValue default_val = upb_MessageValue_Zero();
672
776
  if (upb_FieldDef_IsSubMessage(f)) {
673
777
  return Qnil;
674
778
  } else if (!upb_FieldDef_IsRepeated(f)) {
@@ -677,6 +781,28 @@ static VALUE FieldDescriptor_default(VALUE _self) {
677
781
  return Convert_UpbToRuby(default_val, TypeInfo_get(self->fielddef), Qnil);
678
782
  }
679
783
 
784
+ /*
785
+ * call-seq:
786
+ * FieldDescriptor.has_presence? => bool
787
+ *
788
+ * Returns whether this field tracks presence.
789
+ */
790
+ static VALUE FieldDescriptor_has_presence(VALUE _self) {
791
+ FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
792
+ return upb_FieldDef_HasPresence(self->fielddef) ? Qtrue : Qfalse;
793
+ }
794
+
795
+ /*
796
+ * call-seq:
797
+ * FieldDescriptor.is_packed? => bool
798
+ *
799
+ * Returns whether this is a repeated field that uses packed encoding.
800
+ */
801
+ static VALUE FieldDescriptor_is_packed(VALUE _self) {
802
+ FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
803
+ return upb_FieldDef_IsPacked(self->fielddef) ? Qtrue : Qfalse;
804
+ }
805
+
680
806
  /*
681
807
  * call-seq:
682
808
  * FieldDescriptor.json_name => json_name
@@ -703,7 +829,7 @@ static VALUE FieldDescriptor_label(VALUE _self) {
703
829
  FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
704
830
  switch (upb_FieldDef_Label(self->fielddef)) {
705
831
  #define CONVERT(upb, ruby) \
706
- case kUpb_Label_##upb: \
832
+ case kUpb_Label_##upb: \
707
833
  return ID2SYM(rb_intern(#ruby));
708
834
 
709
835
  CONVERT(Optional, optional);
@@ -803,7 +929,7 @@ static VALUE FieldDescriptor_get(VALUE _self, VALUE msg_rb) {
803
929
  static VALUE FieldDescriptor_has(VALUE _self, VALUE msg_rb) {
804
930
  FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
805
931
  const upb_MessageDef* m;
806
- const upb_MessageDef* msg = Message_Get(msg_rb, &m);
932
+ const upb_Message* msg = Message_Get(msg_rb, &m);
807
933
 
808
934
  if (m != upb_FieldDef_ContainingType(self->fielddef)) {
809
935
  rb_raise(cTypeError, "has method called on wrong message type");
@@ -823,7 +949,7 @@ static VALUE FieldDescriptor_has(VALUE _self, VALUE msg_rb) {
823
949
  static VALUE FieldDescriptor_clear(VALUE _self, VALUE msg_rb) {
824
950
  FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
825
951
  const upb_MessageDef* m;
826
- upb_MessageDef* msg = Message_GetMutable(msg_rb, &m);
952
+ upb_Message* msg = Message_GetMutable(msg_rb, &m);
827
953
 
828
954
  if (m != upb_FieldDef_ContainingType(self->fielddef)) {
829
955
  rb_raise(cTypeError, "has method called on wrong message type");
@@ -844,7 +970,7 @@ static VALUE FieldDescriptor_clear(VALUE _self, VALUE msg_rb) {
844
970
  static VALUE FieldDescriptor_set(VALUE _self, VALUE msg_rb, VALUE value) {
845
971
  FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
846
972
  const upb_MessageDef* m;
847
- upb_MessageDef* msg = Message_GetMutable(msg_rb, &m);
973
+ upb_Message* msg = Message_GetMutable(msg_rb, &m);
848
974
  upb_Arena* arena = Arena_get(Message_GetArena(msg_rb));
849
975
  upb_MessageValue msgval;
850
976
 
@@ -858,6 +984,46 @@ static VALUE FieldDescriptor_set(VALUE _self, VALUE msg_rb, VALUE value) {
858
984
  return Qnil;
859
985
  }
860
986
 
987
+ /*
988
+ * call-seq:
989
+ * FieldDescriptor.options => options
990
+ *
991
+ * Returns the `FieldOptions` for this `FieldDescriptor`.
992
+ */
993
+ static VALUE FieldDescriptor_options(VALUE _self) {
994
+ FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
995
+ const google_protobuf_FieldOptions* opts =
996
+ upb_FieldDef_Options(self->fielddef);
997
+ upb_Arena* arena = upb_Arena_New();
998
+ size_t size;
999
+ char* serialized = google_protobuf_FieldOptions_serialize(opts, arena, &size);
1000
+ VALUE field_options = decode_options(_self, "FieldOptions", size, serialized,
1001
+ self->descriptor_pool);
1002
+ upb_Arena_Free(arena);
1003
+ return field_options;
1004
+ }
1005
+
1006
+ /*
1007
+ * call-seq:
1008
+ * FieldDescriptor.to_proto => FieldDescriptorProto
1009
+ *
1010
+ * Returns the `FieldDescriptorProto` of this `FieldDescriptor`.
1011
+ */
1012
+ static VALUE FieldDescriptor_to_proto(VALUE _self) {
1013
+ FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
1014
+ upb_Arena* arena = upb_Arena_New();
1015
+ google_protobuf_FieldDescriptorProto* proto =
1016
+ upb_FieldDef_ToProto(self->fielddef, arena);
1017
+ size_t size;
1018
+ const char* serialized =
1019
+ google_protobuf_FieldDescriptorProto_serialize(proto, arena, &size);
1020
+ VALUE proto_class = rb_path2class("Google::Protobuf::FieldDescriptorProto");
1021
+ VALUE proto_rb =
1022
+ Message_decode_bytes(size, serialized, 0, proto_class, false);
1023
+ upb_Arena_Free(arena);
1024
+ return proto_rb;
1025
+ }
1026
+
861
1027
  static void FieldDescriptor_register(VALUE module) {
862
1028
  VALUE klass = rb_define_class_under(module, "FieldDescriptor", rb_cObject);
863
1029
  rb_define_alloc_func(klass, FieldDescriptor_alloc);
@@ -865,6 +1031,8 @@ static void FieldDescriptor_register(VALUE module) {
865
1031
  rb_define_method(klass, "name", FieldDescriptor_name, 0);
866
1032
  rb_define_method(klass, "type", FieldDescriptor__type, 0);
867
1033
  rb_define_method(klass, "default", FieldDescriptor_default, 0);
1034
+ rb_define_method(klass, "has_presence?", FieldDescriptor_has_presence, 0);
1035
+ rb_define_method(klass, "is_packed?", FieldDescriptor_is_packed, 0);
868
1036
  rb_define_method(klass, "json_name", FieldDescriptor_json_name, 0);
869
1037
  rb_define_method(klass, "label", FieldDescriptor_label, 0);
870
1038
  rb_define_method(klass, "number", FieldDescriptor_number, 0);
@@ -874,6 +1042,8 @@ static void FieldDescriptor_register(VALUE module) {
874
1042
  rb_define_method(klass, "clear", FieldDescriptor_clear, 1);
875
1043
  rb_define_method(klass, "get", FieldDescriptor_get, 1);
876
1044
  rb_define_method(klass, "set", FieldDescriptor_set, 2);
1045
+ rb_define_method(klass, "options", FieldDescriptor_options, 0);
1046
+ rb_define_method(klass, "to_proto", FieldDescriptor_to_proto, 0);
877
1047
  rb_gc_register_address(&cFieldDescriptor);
878
1048
  cFieldDescriptor = klass;
879
1049
  }
@@ -884,6 +1054,8 @@ static void FieldDescriptor_register(VALUE module) {
884
1054
 
885
1055
  typedef struct {
886
1056
  const upb_OneofDef* oneofdef;
1057
+ // IMPORTANT: WB_PROTECTED objects must only use the RB_OBJ_WRITE()
1058
+ // macro to update VALUE references, as to trigger write barriers.
887
1059
  VALUE descriptor_pool; // Owns the upb_OneofDef.
888
1060
  } OneofDescriptor;
889
1061
 
@@ -897,7 +1069,7 @@ static void OneofDescriptor_mark(void* _self) {
897
1069
  static const rb_data_type_t OneofDescriptor_type = {
898
1070
  "Google::Protobuf::OneofDescriptor",
899
1071
  {OneofDescriptor_mark, RUBY_DEFAULT_FREE, NULL},
900
- .flags = RUBY_TYPED_FREE_IMMEDIATELY,
1072
+ .flags = RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED,
901
1073
  };
902
1074
 
903
1075
  static OneofDescriptor* ruby_to_OneofDescriptor(VALUE val) {
@@ -936,7 +1108,7 @@ static VALUE OneofDescriptor_initialize(VALUE _self, VALUE cookie,
936
1108
  "Descriptor objects may not be created from Ruby.");
937
1109
  }
938
1110
 
939
- self->descriptor_pool = descriptor_pool;
1111
+ RB_OBJ_WRITE(_self, &self->descriptor_pool, descriptor_pool);
940
1112
  self->oneofdef = (const upb_OneofDef*)NUM2ULL(ptr);
941
1113
 
942
1114
  return Qnil;
@@ -971,12 +1143,54 @@ static VALUE OneofDescriptor_each(VALUE _self) {
971
1143
  return Qnil;
972
1144
  }
973
1145
 
1146
+ /*
1147
+ * call-seq:
1148
+ * OneofDescriptor.options => options
1149
+ *
1150
+ * Returns the `OneofOptions` for this `OneofDescriptor`.
1151
+ */
1152
+ static VALUE OneOfDescriptor_options(VALUE _self) {
1153
+ OneofDescriptor* self = ruby_to_OneofDescriptor(_self);
1154
+ const google_protobuf_OneofOptions* opts =
1155
+ upb_OneofDef_Options(self->oneofdef);
1156
+ upb_Arena* arena = upb_Arena_New();
1157
+ size_t size;
1158
+ char* serialized = google_protobuf_OneofOptions_serialize(opts, arena, &size);
1159
+ VALUE oneof_options = decode_options(_self, "OneofOptions", size, serialized,
1160
+ self->descriptor_pool);
1161
+ upb_Arena_Free(arena);
1162
+ return oneof_options;
1163
+ }
1164
+
1165
+ /*
1166
+ * call-seq:
1167
+ * OneofDescriptor.to_proto => OneofDescriptorProto
1168
+ *
1169
+ * Returns the `OneofDescriptorProto` of this `OneofDescriptor`.
1170
+ */
1171
+ static VALUE OneOfDescriptor_to_proto(VALUE _self) {
1172
+ OneofDescriptor* self = ruby_to_OneofDescriptor(_self);
1173
+ upb_Arena* arena = upb_Arena_New();
1174
+ google_protobuf_OneofDescriptorProto* proto =
1175
+ upb_OneofDef_ToProto(self->oneofdef, arena);
1176
+ size_t size;
1177
+ const char* serialized =
1178
+ google_protobuf_OneofDescriptorProto_serialize(proto, arena, &size);
1179
+ VALUE proto_class = rb_path2class("Google::Protobuf::OneofDescriptorProto");
1180
+ VALUE proto_rb =
1181
+ Message_decode_bytes(size, serialized, 0, proto_class, false);
1182
+ upb_Arena_Free(arena);
1183
+ return proto_rb;
1184
+ }
1185
+
974
1186
  static void OneofDescriptor_register(VALUE module) {
975
1187
  VALUE klass = rb_define_class_under(module, "OneofDescriptor", rb_cObject);
976
1188
  rb_define_alloc_func(klass, OneofDescriptor_alloc);
977
1189
  rb_define_method(klass, "initialize", OneofDescriptor_initialize, 3);
978
1190
  rb_define_method(klass, "name", OneofDescriptor_name, 0);
979
1191
  rb_define_method(klass, "each", OneofDescriptor_each, 0);
1192
+ rb_define_method(klass, "options", OneOfDescriptor_options, 0);
1193
+ rb_define_method(klass, "to_proto", OneOfDescriptor_to_proto, 0);
980
1194
  rb_include_module(klass, rb_mEnumerable);
981
1195
  rb_gc_register_address(&cOneofDescriptor);
982
1196
  cOneofDescriptor = klass;
@@ -988,6 +1202,8 @@ static void OneofDescriptor_register(VALUE module) {
988
1202
 
989
1203
  typedef struct {
990
1204
  const upb_EnumDef* enumdef;
1205
+ // IMPORTANT: WB_PROTECTED objects must only use the RB_OBJ_WRITE()
1206
+ // macro to update VALUE references, as to trigger write barriers.
991
1207
  VALUE module; // begins as nil
992
1208
  VALUE descriptor_pool; // Owns the upb_EnumDef.
993
1209
  } EnumDescriptor;
@@ -1003,7 +1219,7 @@ static void EnumDescriptor_mark(void* _self) {
1003
1219
  static const rb_data_type_t EnumDescriptor_type = {
1004
1220
  "Google::Protobuf::EnumDescriptor",
1005
1221
  {EnumDescriptor_mark, RUBY_DEFAULT_FREE, NULL},
1006
- .flags = RUBY_TYPED_FREE_IMMEDIATELY,
1222
+ .flags = RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED,
1007
1223
  };
1008
1224
 
1009
1225
  static EnumDescriptor* ruby_to_EnumDescriptor(VALUE val) {
@@ -1042,7 +1258,7 @@ static VALUE EnumDescriptor_initialize(VALUE _self, VALUE cookie,
1042
1258
  "Descriptor objects may not be created from Ruby.");
1043
1259
  }
1044
1260
 
1045
- self->descriptor_pool = descriptor_pool;
1261
+ RB_OBJ_WRITE(_self, &self->descriptor_pool, descriptor_pool);
1046
1262
  self->enumdef = (const upb_EnumDef*)NUM2ULL(ptr);
1047
1263
 
1048
1264
  return Qnil;
@@ -1060,6 +1276,17 @@ static VALUE EnumDescriptor_file_descriptor(VALUE _self) {
1060
1276
  upb_EnumDef_File(self->enumdef));
1061
1277
  }
1062
1278
 
1279
+ /*
1280
+ * call-seq:
1281
+ * EnumDescriptor.is_closed? => bool
1282
+ *
1283
+ * Returns whether this enum is open or closed.
1284
+ */
1285
+ static VALUE EnumDescriptor_is_closed(VALUE _self) {
1286
+ EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
1287
+ return upb_EnumDef_IsClosed(self->enumdef) ? Qtrue : Qfalse;
1288
+ }
1289
+
1063
1290
  /*
1064
1291
  * call-seq:
1065
1292
  * EnumDescriptor.name => name
@@ -1081,7 +1308,7 @@ static VALUE EnumDescriptor_name(VALUE _self) {
1081
1308
  static VALUE EnumDescriptor_lookup_name(VALUE _self, VALUE name) {
1082
1309
  EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
1083
1310
  const char* name_str = rb_id2name(SYM2ID(name));
1084
- const upb_EnumValueDef *ev =
1311
+ const upb_EnumValueDef* ev =
1085
1312
  upb_EnumDef_FindValueByName(self->enumdef, name_str);
1086
1313
  if (ev) {
1087
1314
  return INT2NUM(upb_EnumValueDef_Number(ev));
@@ -1100,7 +1327,8 @@ static VALUE EnumDescriptor_lookup_name(VALUE _self, VALUE name) {
1100
1327
  static VALUE EnumDescriptor_lookup_value(VALUE _self, VALUE number) {
1101
1328
  EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
1102
1329
  int32_t val = NUM2INT(number);
1103
- const upb_EnumValueDef* ev = upb_EnumDef_FindValueByNumber(self->enumdef, val);
1330
+ const upb_EnumValueDef* ev =
1331
+ upb_EnumDef_FindValueByNumber(self->enumdef, val);
1104
1332
  if (ev) {
1105
1333
  return ID2SYM(rb_intern(upb_EnumValueDef_Name(ev)));
1106
1334
  } else {
@@ -1138,11 +1366,52 @@ static VALUE EnumDescriptor_each(VALUE _self) {
1138
1366
  static VALUE EnumDescriptor_enummodule(VALUE _self) {
1139
1367
  EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
1140
1368
  if (self->module == Qnil) {
1141
- self->module = build_module_from_enumdesc(_self);
1369
+ RB_OBJ_WRITE(_self, &self->module, build_module_from_enumdesc(_self));
1142
1370
  }
1143
1371
  return self->module;
1144
1372
  }
1145
1373
 
1374
+ /*
1375
+ * call-seq:
1376
+ * EnumDescriptor.options => options
1377
+ *
1378
+ * Returns the `EnumOptions` for this `EnumDescriptor`.
1379
+ */
1380
+ static VALUE EnumDescriptor_options(VALUE _self) {
1381
+ EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
1382
+ const google_protobuf_EnumOptions* opts = upb_EnumDef_Options(self->enumdef);
1383
+ upb_Arena* arena = upb_Arena_New();
1384
+ size_t size;
1385
+ char* serialized = google_protobuf_EnumOptions_serialize(opts, arena, &size);
1386
+ VALUE enum_options = decode_options(_self, "EnumOptions", size, serialized,
1387
+ self->descriptor_pool);
1388
+ upb_Arena_Free(arena);
1389
+ return enum_options;
1390
+ }
1391
+
1392
+ /*
1393
+ * call-seq:
1394
+ * EnumDescriptor.to_proto => EnumDescriptorProto
1395
+ *
1396
+ * Returns the `EnumDescriptorProto` of this `EnumDescriptor`.
1397
+ */
1398
+ static VALUE EnumDescriptor_to_proto(VALUE _self) {
1399
+ EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
1400
+ upb_Arena* arena = upb_Arena_New();
1401
+ google_protobuf_EnumDescriptorProto* proto =
1402
+ upb_EnumDef_ToProto(self->enumdef, arena);
1403
+
1404
+ size_t size;
1405
+ const char* serialized =
1406
+ google_protobuf_EnumDescriptorProto_serialize(proto, arena, &size);
1407
+
1408
+ VALUE proto_class = rb_path2class("Google::Protobuf::EnumDescriptorProto");
1409
+ VALUE proto_rb =
1410
+ Message_decode_bytes(size, serialized, 0, proto_class, false);
1411
+ upb_Arena_Free(arena);
1412
+ return proto_rb;
1413
+ }
1414
+
1146
1415
  static void EnumDescriptor_register(VALUE module) {
1147
1416
  VALUE klass = rb_define_class_under(module, "EnumDescriptor", rb_cObject);
1148
1417
  rb_define_alloc_func(klass, EnumDescriptor_alloc);
@@ -1153,11 +1422,350 @@ static void EnumDescriptor_register(VALUE module) {
1153
1422
  rb_define_method(klass, "each", EnumDescriptor_each, 0);
1154
1423
  rb_define_method(klass, "enummodule", EnumDescriptor_enummodule, 0);
1155
1424
  rb_define_method(klass, "file_descriptor", EnumDescriptor_file_descriptor, 0);
1425
+ rb_define_method(klass, "is_closed?", EnumDescriptor_is_closed, 0);
1426
+ rb_define_method(klass, "options", EnumDescriptor_options, 0);
1427
+ rb_define_method(klass, "to_proto", EnumDescriptor_to_proto, 0);
1156
1428
  rb_include_module(klass, rb_mEnumerable);
1157
1429
  rb_gc_register_address(&cEnumDescriptor);
1158
1430
  cEnumDescriptor = klass;
1159
1431
  }
1160
1432
 
1433
+ // -----------------------------------------------------------------------------
1434
+ // ServiceDescriptor
1435
+ // -----------------------------------------------------------------------------
1436
+
1437
+ typedef struct {
1438
+ const upb_ServiceDef* servicedef;
1439
+ // IMPORTANT: WB_PROTECTED objects must only use the RB_OBJ_WRITE()
1440
+ // macro to update VALUE references, as to trigger write barriers.
1441
+ VALUE module; // begins as nil
1442
+ VALUE descriptor_pool; // Owns the upb_ServiceDef.
1443
+ } ServiceDescriptor;
1444
+
1445
+ static VALUE cServiceDescriptor = Qnil;
1446
+
1447
+ static void ServiceDescriptor_mark(void* _self) {
1448
+ ServiceDescriptor* self = _self;
1449
+ rb_gc_mark(self->module);
1450
+ rb_gc_mark(self->descriptor_pool);
1451
+ }
1452
+
1453
+ static const rb_data_type_t ServiceDescriptor_type = {
1454
+ "Google::Protobuf::ServicDescriptor",
1455
+ {ServiceDescriptor_mark, RUBY_DEFAULT_FREE, NULL},
1456
+ .flags = RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED,
1457
+ };
1458
+
1459
+ static ServiceDescriptor* ruby_to_ServiceDescriptor(VALUE val) {
1460
+ ServiceDescriptor* ret;
1461
+ TypedData_Get_Struct(val, ServiceDescriptor, &ServiceDescriptor_type, ret);
1462
+ return ret;
1463
+ }
1464
+
1465
+ static VALUE ServiceDescriptor_alloc(VALUE klass) {
1466
+ ServiceDescriptor* self = ALLOC(ServiceDescriptor);
1467
+ VALUE ret = TypedData_Wrap_Struct(klass, &ServiceDescriptor_type, self);
1468
+ self->servicedef = NULL;
1469
+ self->module = Qnil;
1470
+ self->descriptor_pool = Qnil;
1471
+ return ret;
1472
+ }
1473
+
1474
+ /*
1475
+ * call-seq:
1476
+ * ServiceDescriptor.new(c_only_cookie, ptr) => ServiceDescriptor
1477
+ *
1478
+ * Creates a descriptor wrapper object. May only be called from C.
1479
+ */
1480
+ static VALUE ServiceDescriptor_initialize(VALUE _self, VALUE cookie,
1481
+ VALUE descriptor_pool, VALUE ptr) {
1482
+ ServiceDescriptor* self = ruby_to_ServiceDescriptor(_self);
1483
+
1484
+ if (cookie != c_only_cookie) {
1485
+ rb_raise(rb_eRuntimeError,
1486
+ "Descriptor objects may not be created from Ruby.");
1487
+ }
1488
+
1489
+ RB_OBJ_WRITE(_self, &self->descriptor_pool, descriptor_pool);
1490
+ self->servicedef = (const upb_ServiceDef*)NUM2ULL(ptr);
1491
+
1492
+ return Qnil;
1493
+ }
1494
+
1495
+ /*
1496
+ * call-seq:
1497
+ * ServiceDescriptor.name => name
1498
+ *
1499
+ * Returns the name of this service.
1500
+ */
1501
+ static VALUE ServiceDescriptor_name(VALUE _self) {
1502
+ ServiceDescriptor* self = ruby_to_ServiceDescriptor(_self);
1503
+ return rb_str_maybe_null(upb_ServiceDef_FullName(self->servicedef));
1504
+ }
1505
+
1506
+ /*
1507
+ * call-seq:
1508
+ * ServiceDescriptor.file_descriptor
1509
+ *
1510
+ * Returns the FileDescriptor object this service belongs to.
1511
+ */
1512
+ static VALUE ServiceDescriptor_file_descriptor(VALUE _self) {
1513
+ ServiceDescriptor* self = ruby_to_ServiceDescriptor(_self);
1514
+ return get_filedef_obj(self->descriptor_pool,
1515
+ upb_ServiceDef_File(self->servicedef));
1516
+ }
1517
+
1518
+ /*
1519
+ * call-seq:
1520
+ * ServiceDescriptor.each(&block)
1521
+ *
1522
+ * Iterates over methods in this service, yielding to the block on each one.
1523
+ */
1524
+ static VALUE ServiceDescriptor_each(VALUE _self) {
1525
+ ServiceDescriptor* self = ruby_to_ServiceDescriptor(_self);
1526
+
1527
+ int n = upb_ServiceDef_MethodCount(self->servicedef);
1528
+ for (int i = 0; i < n; i++) {
1529
+ const upb_MethodDef* method = upb_ServiceDef_Method(self->servicedef, i);
1530
+ VALUE obj = get_methoddef_obj(self->descriptor_pool, method);
1531
+ rb_yield(obj);
1532
+ }
1533
+ return Qnil;
1534
+ }
1535
+
1536
+ /*
1537
+ * call-seq:
1538
+ * ServiceDescriptor.options => options
1539
+ *
1540
+ * Returns the `ServiceOptions` for this `ServiceDescriptor`.
1541
+ */
1542
+ static VALUE ServiceDescriptor_options(VALUE _self) {
1543
+ ServiceDescriptor* self = ruby_to_ServiceDescriptor(_self);
1544
+ const google_protobuf_ServiceOptions* opts =
1545
+ upb_ServiceDef_Options(self->servicedef);
1546
+ upb_Arena* arena = upb_Arena_New();
1547
+ size_t size;
1548
+ char* serialized =
1549
+ google_protobuf_ServiceOptions_serialize(opts, arena, &size);
1550
+ VALUE service_options = decode_options(_self, "ServiceOptions", size,
1551
+ serialized, self->descriptor_pool);
1552
+ upb_Arena_Free(arena);
1553
+ return service_options;
1554
+ }
1555
+
1556
+ /*
1557
+ * call-seq:
1558
+ * ServiceDescriptor.to_proto => ServiceDescriptorProto
1559
+ *
1560
+ * Returns the `ServiceDescriptorProto` of this `ServiceDescriptor`.
1561
+ */
1562
+ static VALUE ServiceDescriptor_to_proto(VALUE _self) {
1563
+ ServiceDescriptor* self = ruby_to_ServiceDescriptor(_self);
1564
+ upb_Arena* arena = upb_Arena_New();
1565
+ google_protobuf_ServiceDescriptorProto* proto =
1566
+ upb_ServiceDef_ToProto(self->servicedef, arena);
1567
+ size_t size;
1568
+ const char* serialized =
1569
+ google_protobuf_ServiceDescriptorProto_serialize(proto, arena, &size);
1570
+ VALUE proto_class = rb_path2class("Google::Protobuf::ServiceDescriptorProto");
1571
+ VALUE proto_rb =
1572
+ Message_decode_bytes(size, serialized, 0, proto_class, false);
1573
+ upb_Arena_Free(arena);
1574
+ return proto_rb;
1575
+ }
1576
+
1577
+ static void ServiceDescriptor_register(VALUE module) {
1578
+ VALUE klass = rb_define_class_under(module, "ServiceDescriptor", rb_cObject);
1579
+ rb_define_alloc_func(klass, ServiceDescriptor_alloc);
1580
+ rb_define_method(klass, "initialize", ServiceDescriptor_initialize, 3);
1581
+ rb_define_method(klass, "name", ServiceDescriptor_name, 0);
1582
+ rb_define_method(klass, "each", ServiceDescriptor_each, 0);
1583
+ rb_define_method(klass, "file_descriptor", ServiceDescriptor_file_descriptor,
1584
+ 0);
1585
+ rb_define_method(klass, "options", ServiceDescriptor_options, 0);
1586
+ rb_define_method(klass, "to_proto", ServiceDescriptor_to_proto, 0);
1587
+ rb_include_module(klass, rb_mEnumerable);
1588
+ rb_gc_register_address(&cServiceDescriptor);
1589
+ cServiceDescriptor = klass;
1590
+ }
1591
+
1592
+ // -----------------------------------------------------------------------------
1593
+ // MethodDescriptor
1594
+ // -----------------------------------------------------------------------------
1595
+
1596
+ typedef struct {
1597
+ const upb_MethodDef* methoddef;
1598
+ // IMPORTANT: WB_PROTECTED objects must only use the RB_OBJ_WRITE()
1599
+ // macro to update VALUE references, as to trigger write barriers.
1600
+ VALUE module; // begins as nil
1601
+ VALUE descriptor_pool; // Owns the upb_MethodDef.
1602
+ } MethodDescriptor;
1603
+
1604
+ static VALUE cMethodDescriptor = Qnil;
1605
+
1606
+ static void MethodDescriptor_mark(void* _self) {
1607
+ MethodDescriptor* self = _self;
1608
+ rb_gc_mark(self->module);
1609
+ rb_gc_mark(self->descriptor_pool);
1610
+ }
1611
+
1612
+ static const rb_data_type_t MethodDescriptor_type = {
1613
+ "Google::Protobuf::MethodDescriptor",
1614
+ {MethodDescriptor_mark, RUBY_DEFAULT_FREE, NULL},
1615
+ .flags = RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED,
1616
+ };
1617
+
1618
+ static MethodDescriptor* ruby_to_MethodDescriptor(VALUE val) {
1619
+ MethodDescriptor* ret;
1620
+ TypedData_Get_Struct(val, MethodDescriptor, &MethodDescriptor_type, ret);
1621
+ return ret;
1622
+ }
1623
+
1624
+ static VALUE MethodDescriptor_alloc(VALUE klass) {
1625
+ MethodDescriptor* self = ALLOC(MethodDescriptor);
1626
+ VALUE ret = TypedData_Wrap_Struct(klass, &MethodDescriptor_type, self);
1627
+ self->methoddef = NULL;
1628
+ self->module = Qnil;
1629
+ self->descriptor_pool = Qnil;
1630
+ return ret;
1631
+ }
1632
+
1633
+ /*
1634
+ * call-seq:
1635
+ * MethodDescriptor.new(c_only_cookie, ptr) => MethodDescriptor
1636
+ *
1637
+ * Creates a descriptor wrapper object. May only be called from C.
1638
+ */
1639
+ static VALUE MethodDescriptor_initialize(VALUE _self, VALUE cookie,
1640
+ VALUE descriptor_pool, VALUE ptr) {
1641
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1642
+
1643
+ if (cookie != c_only_cookie) {
1644
+ rb_raise(rb_eRuntimeError,
1645
+ "Descriptor objects may not be created from Ruby.");
1646
+ }
1647
+
1648
+ RB_OBJ_WRITE(_self, &self->descriptor_pool, descriptor_pool);
1649
+ self->methoddef = (const upb_MethodDef*)NUM2ULL(ptr);
1650
+
1651
+ return Qnil;
1652
+ }
1653
+
1654
+ /*
1655
+ * call-seq:
1656
+ * MethodDescriptor.name => name
1657
+ *
1658
+ * Returns the name of this method
1659
+ */
1660
+ static VALUE MethodDescriptor_name(VALUE _self) {
1661
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1662
+ return rb_str_maybe_null(upb_MethodDef_Name(self->methoddef));
1663
+ }
1664
+
1665
+ /*
1666
+ * call-seq:
1667
+ * MethodDescriptor.options => options
1668
+ *
1669
+ * Returns the `MethodOptions` for this `MethodDescriptor`.
1670
+ */
1671
+ static VALUE MethodDescriptor_options(VALUE _self) {
1672
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1673
+ const google_protobuf_MethodOptions* opts =
1674
+ upb_MethodDef_Options(self->methoddef);
1675
+ upb_Arena* arena = upb_Arena_New();
1676
+ size_t size;
1677
+ char* serialized =
1678
+ google_protobuf_MethodOptions_serialize(opts, arena, &size);
1679
+ VALUE method_options = decode_options(_self, "MethodOptions", size,
1680
+ serialized, self->descriptor_pool);
1681
+ upb_Arena_Free(arena);
1682
+ return method_options;
1683
+ }
1684
+
1685
+ /*
1686
+ * call-seq:
1687
+ * MethodDescriptor.input_type => Descriptor
1688
+ *
1689
+ * Returns the `Descriptor` for the request message type of this method
1690
+ */
1691
+ static VALUE MethodDescriptor_input_type(VALUE _self) {
1692
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1693
+ const upb_MessageDef* type = upb_MethodDef_InputType(self->methoddef);
1694
+ return get_msgdef_obj(self->descriptor_pool, type);
1695
+ }
1696
+
1697
+ /*
1698
+ * call-seq:
1699
+ * MethodDescriptor.output_type => Descriptor
1700
+ *
1701
+ * Returns the `Descriptor` for the response message type of this method
1702
+ */
1703
+ static VALUE MethodDescriptor_output_type(VALUE _self) {
1704
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1705
+ const upb_MessageDef* type = upb_MethodDef_OutputType(self->methoddef);
1706
+ return get_msgdef_obj(self->descriptor_pool, type);
1707
+ }
1708
+
1709
+ /*
1710
+ * call-seq:
1711
+ * MethodDescriptor.client_streaming => bool
1712
+ *
1713
+ * Returns whether or not this is a streaming request method
1714
+ */
1715
+ static VALUE MethodDescriptor_client_streaming(VALUE _self) {
1716
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1717
+ return upb_MethodDef_ClientStreaming(self->methoddef) ? Qtrue : Qfalse;
1718
+ }
1719
+
1720
+ /*
1721
+ * call-seq:
1722
+ * MethodDescriptor.to_proto => MethodDescriptorProto
1723
+ *
1724
+ * Returns the `MethodDescriptorProto` of this `MethodDescriptor`.
1725
+ */
1726
+ static VALUE MethodDescriptor_to_proto(VALUE _self) {
1727
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1728
+ upb_Arena* arena = upb_Arena_New();
1729
+ google_protobuf_MethodDescriptorProto* proto =
1730
+ upb_MethodDef_ToProto(self->methoddef, arena);
1731
+ size_t size;
1732
+ const char* serialized =
1733
+ google_protobuf_MethodDescriptorProto_serialize(proto, arena, &size);
1734
+ VALUE proto_class = rb_path2class("Google::Protobuf::MethodDescriptorProto");
1735
+ VALUE proto_rb =
1736
+ Message_decode_bytes(size, serialized, 0, proto_class, false);
1737
+ upb_Arena_Free(arena);
1738
+ return proto_rb;
1739
+ }
1740
+
1741
+ /*
1742
+ * call-seq:
1743
+ * MethodDescriptor.server_streaming => bool
1744
+ *
1745
+ * Returns whether or not this is a streaming response method
1746
+ */
1747
+ static VALUE MethodDescriptor_server_streaming(VALUE _self) {
1748
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1749
+ return upb_MethodDef_ServerStreaming(self->methoddef) ? Qtrue : Qfalse;
1750
+ }
1751
+
1752
+ static void MethodDescriptor_register(VALUE module) {
1753
+ VALUE klass = rb_define_class_under(module, "MethodDescriptor", rb_cObject);
1754
+ rb_define_alloc_func(klass, MethodDescriptor_alloc);
1755
+ rb_define_method(klass, "initialize", MethodDescriptor_initialize, 3);
1756
+ rb_define_method(klass, "name", MethodDescriptor_name, 0);
1757
+ rb_define_method(klass, "options", MethodDescriptor_options, 0);
1758
+ rb_define_method(klass, "input_type", MethodDescriptor_input_type, 0);
1759
+ rb_define_method(klass, "output_type", MethodDescriptor_output_type, 0);
1760
+ rb_define_method(klass, "client_streaming", MethodDescriptor_client_streaming,
1761
+ 0);
1762
+ rb_define_method(klass, "server_streaming", MethodDescriptor_server_streaming,
1763
+ 0);
1764
+ rb_define_method(klass, "to_proto", MethodDescriptor_to_proto, 0);
1765
+ rb_gc_register_address(&cMethodDescriptor);
1766
+ cMethodDescriptor = klass;
1767
+ }
1768
+
1161
1769
  static VALUE get_def_obj(VALUE _descriptor_pool, const void* ptr, VALUE klass) {
1162
1770
  DescriptorPool* descriptor_pool = ruby_to_DescriptorPool(_descriptor_pool);
1163
1771
  VALUE key = ULL2NUM((intptr_t)ptr);
@@ -1199,6 +1807,16 @@ static VALUE get_oneofdef_obj(VALUE descriptor_pool, const upb_OneofDef* def) {
1199
1807
  return get_def_obj(descriptor_pool, def, cOneofDescriptor);
1200
1808
  }
1201
1809
 
1810
+ static VALUE get_servicedef_obj(VALUE descriptor_pool,
1811
+ const upb_ServiceDef* def) {
1812
+ return get_def_obj(descriptor_pool, def, cServiceDescriptor);
1813
+ }
1814
+
1815
+ static VALUE get_methoddef_obj(VALUE descriptor_pool,
1816
+ const upb_MethodDef* def) {
1817
+ return get_def_obj(descriptor_pool, def, cMethodDescriptor);
1818
+ }
1819
+
1202
1820
  // -----------------------------------------------------------------------------
1203
1821
  // Shared functions
1204
1822
  // -----------------------------------------------------------------------------
@@ -1274,6 +1892,8 @@ void Defs_register(VALUE module) {
1274
1892
  FieldDescriptor_register(module);
1275
1893
  OneofDescriptor_register(module);
1276
1894
  EnumDescriptor_register(module);
1895
+ ServiceDescriptor_register(module);
1896
+ MethodDescriptor_register(module);
1277
1897
 
1278
1898
  rb_gc_register_address(&c_only_cookie);
1279
1899
  c_only_cookie = rb_class_new_instance(0, NULL, rb_cObject);