google-protobuf 3.25.5-x86-linux → 4.30.1-x86-linux

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 (68) hide show
  1. checksums.yaml +4 -4
  2. data/ext/google/protobuf_c/convert.c +39 -18
  3. data/ext/google/protobuf_c/defs.c +529 -32
  4. data/ext/google/protobuf_c/extconf.rb +13 -1
  5. data/ext/google/protobuf_c/glue.c +79 -0
  6. data/ext/google/protobuf_c/map.c +74 -29
  7. data/ext/google/protobuf_c/map.h +7 -3
  8. data/ext/google/protobuf_c/message.c +119 -119
  9. data/ext/google/protobuf_c/message.h +2 -6
  10. data/ext/google/protobuf_c/protobuf.c +32 -18
  11. data/ext/google/protobuf_c/protobuf.h +3 -7
  12. data/ext/google/protobuf_c/repeated_field.c +58 -23
  13. data/ext/google/protobuf_c/repeated_field.h +6 -2
  14. data/ext/google/protobuf_c/ruby-upb.c +14931 -11575
  15. data/ext/google/protobuf_c/ruby-upb.h +8701 -5832
  16. data/ext/google/protobuf_c/shared_convert.c +10 -5
  17. data/ext/google/protobuf_c/shared_convert.h +2 -2
  18. data/ext/google/protobuf_c/shared_message.c +3 -31
  19. data/ext/google/protobuf_c/shared_message.h +0 -4
  20. data/ext/google/protobuf_c/third_party/utf8_range/utf8_range.c +207 -0
  21. data/ext/google/protobuf_c/third_party/utf8_range/utf8_range.h +9 -8
  22. data/ext/google/protobuf_c/third_party/utf8_range/utf8_range_neon.inc +117 -0
  23. data/ext/google/protobuf_c/third_party/utf8_range/utf8_range_sse.inc +272 -0
  24. data/lib/google/3.0/protobuf_c.so +0 -0
  25. data/lib/google/3.1/protobuf_c.so +0 -0
  26. data/lib/google/3.2/protobuf_c.so +0 -0
  27. data/lib/google/3.3/protobuf_c.so +0 -0
  28. data/lib/google/3.4/protobuf_c.so +0 -0
  29. data/lib/google/protobuf/any_pb.rb +1 -22
  30. data/lib/google/protobuf/api_pb.rb +1 -24
  31. data/lib/google/protobuf/descriptor_pb.rb +4 -23
  32. data/lib/google/protobuf/duration_pb.rb +1 -22
  33. data/lib/google/protobuf/empty_pb.rb +1 -22
  34. data/lib/google/protobuf/ffi/descriptor.rb +14 -4
  35. data/lib/google/protobuf/ffi/descriptor_pool.rb +3 -1
  36. data/lib/google/protobuf/ffi/enum_descriptor.rb +13 -1
  37. data/lib/google/protobuf/ffi/ffi.rb +7 -6
  38. data/lib/google/protobuf/ffi/field_descriptor.rb +23 -2
  39. data/lib/google/protobuf/ffi/file_descriptor.rb +13 -13
  40. data/lib/google/protobuf/ffi/internal/arena.rb +0 -6
  41. data/lib/google/protobuf/ffi/internal/convert.rb +17 -30
  42. data/lib/google/protobuf/ffi/map.rb +50 -24
  43. data/lib/google/protobuf/ffi/message.rb +188 -67
  44. data/lib/google/protobuf/ffi/method_descriptor.rb +124 -0
  45. data/lib/google/protobuf/ffi/object_cache.rb +3 -3
  46. data/lib/google/protobuf/ffi/oneof_descriptor.rb +13 -1
  47. data/lib/google/protobuf/ffi/repeated_field.rb +47 -19
  48. data/lib/google/protobuf/ffi/service_descriptor.rb +117 -0
  49. data/lib/google/protobuf/field_mask_pb.rb +1 -22
  50. data/lib/google/protobuf/internal/object_cache.rb +99 -0
  51. data/lib/google/protobuf/plugin_pb.rb +2 -24
  52. data/lib/google/protobuf/repeated_field.rb +4 -5
  53. data/lib/google/protobuf/source_context_pb.rb +1 -22
  54. data/lib/google/protobuf/struct_pb.rb +1 -22
  55. data/lib/google/protobuf/timestamp_pb.rb +1 -22
  56. data/lib/google/protobuf/type_pb.rb +1 -24
  57. data/lib/google/protobuf/wrappers_pb.rb +1 -22
  58. data/lib/google/protobuf.rb +1 -1
  59. data/lib/google/protobuf_ffi.rb +5 -3
  60. data/lib/google/protobuf_native.rb +0 -1
  61. data/lib/google/tasks/ffi.rake +1 -3
  62. metadata +30 -15
  63. data/ext/google/protobuf_c/third_party/utf8_range/naive.c +0 -92
  64. data/ext/google/protobuf_c/third_party/utf8_range/range2-neon.c +0 -157
  65. data/ext/google/protobuf_c/third_party/utf8_range/range2-sse.c +0 -170
  66. data/lib/google/2.7/protobuf_c.so +0 -0
  67. data/lib/google/protobuf/descriptor_dsl.rb +0 -465
  68. data/lib/google/protobuf/object_cache.rb +0 -97
@@ -44,9 +44,11 @@ static void Message_mark(void* _self) {
44
44
  rb_gc_mark(self->arena);
45
45
  }
46
46
 
47
+ static size_t Message_memsize(const void* _self) { return sizeof(Message); }
48
+
47
49
  static rb_data_type_t Message_type = {
48
50
  "Google::Protobuf::Message",
49
- {Message_mark, RUBY_DEFAULT_FREE, NULL},
51
+ {Message_mark, RUBY_DEFAULT_FREE, Message_memsize},
50
52
  .flags = RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED,
51
53
  };
52
54
 
@@ -78,11 +80,13 @@ const upb_Message* Message_Get(VALUE msg_rb, const upb_MessageDef** m) {
78
80
  }
79
81
 
80
82
  upb_Message* Message_GetMutable(VALUE msg_rb, const upb_MessageDef** m) {
81
- rb_check_frozen(msg_rb);
82
- return (upb_Message*)Message_Get(msg_rb, m);
83
+ const upb_Message* upb_msg = Message_Get(msg_rb, m);
84
+ Protobuf_CheckNotFrozen(msg_rb, upb_Message_IsFrozen(upb_msg));
85
+ return (upb_Message*)upb_msg;
83
86
  }
84
87
 
85
- void Message_InitPtr(VALUE self_, upb_Message* msg, VALUE arena) {
88
+ void Message_InitPtr(VALUE self_, const upb_Message* msg, VALUE arena) {
89
+ PBRUBY_ASSERT(arena != Qnil);
86
90
  Message* self = ruby_to_Message(self_);
87
91
  self->msg = msg;
88
92
  RB_OBJ_WRITE(self_, &self->arena, arena);
@@ -103,7 +107,7 @@ void Message_CheckClass(VALUE klass) {
103
107
  }
104
108
  }
105
109
 
106
- VALUE Message_GetRubyWrapper(upb_Message* msg, const upb_MessageDef* m,
110
+ VALUE Message_GetRubyWrapper(const upb_Message* msg, const upb_MessageDef* m,
107
111
  VALUE arena) {
108
112
  if (msg == NULL) return Qnil;
109
113
 
@@ -114,7 +118,6 @@ VALUE Message_GetRubyWrapper(upb_Message* msg, const upb_MessageDef* m,
114
118
  val = Message_alloc(klass);
115
119
  Message_InitPtr(val, msg, arena);
116
120
  }
117
-
118
121
  return val;
119
122
  }
120
123
 
@@ -246,7 +249,7 @@ static int extract_method_call(VALUE method_name, Message* self,
246
249
  static VALUE Message_oneof_accessor(VALUE _self, const upb_OneofDef* o,
247
250
  int accessor_type) {
248
251
  Message* self = ruby_to_Message(_self);
249
- const upb_FieldDef* oneof_field = upb_Message_WhichOneof(self->msg, o);
252
+ const upb_FieldDef* oneof_field = upb_Message_WhichOneofByDef(self->msg, o);
250
253
 
251
254
  switch (accessor_type) {
252
255
  case METHOD_PRESENCE:
@@ -286,13 +289,42 @@ static void Message_setfield(upb_Message* msg, const upb_FieldDef* f, VALUE val,
286
289
  upb_Message_SetFieldByDef(msg, f, msgval, arena);
287
290
  }
288
291
 
292
+ VALUE Message_getfield_frozen(const upb_Message* msg, const upb_FieldDef* f,
293
+ VALUE arena) {
294
+ upb_MessageValue msgval = upb_Message_GetFieldByDef(msg, f);
295
+ if (upb_FieldDef_IsMap(f)) {
296
+ if (msgval.map_val == NULL) {
297
+ return Map_EmptyFrozen(f);
298
+ }
299
+ const upb_FieldDef* key_f = map_field_key(f);
300
+ const upb_FieldDef* val_f = map_field_value(f);
301
+ upb_CType key_type = upb_FieldDef_CType(key_f);
302
+ TypeInfo value_type_info = TypeInfo_get(val_f);
303
+ return Map_GetRubyWrapper(msgval.map_val, key_type, value_type_info, arena);
304
+ }
305
+ if (upb_FieldDef_IsRepeated(f)) {
306
+ if (msgval.array_val == NULL) {
307
+ return RepeatedField_EmptyFrozen(f);
308
+ }
309
+ return RepeatedField_GetRubyWrapper(msgval.array_val, TypeInfo_get(f),
310
+ arena);
311
+ }
312
+ VALUE ret;
313
+ if (upb_FieldDef_IsSubMessage(f)) {
314
+ const upb_MessageDef* m = upb_FieldDef_MessageSubDef(f);
315
+ ret = Message_GetRubyWrapper(msgval.msg_val, m, arena);
316
+ } else {
317
+ ret = Convert_UpbToRuby(msgval, TypeInfo_get(f), Qnil);
318
+ }
319
+ return ret;
320
+ }
321
+
289
322
  VALUE Message_getfield(VALUE _self, const upb_FieldDef* f) {
290
323
  Message* self = ruby_to_Message(_self);
291
- // This is a special-case: upb_Message_Mutable() for map & array are logically
292
- // const (they will not change what is serialized) but physically
293
- // non-const, as they do allocate a repeated field or map. The logical
294
- // constness means it's ok to do even if the message is frozen.
295
- upb_Message* msg = (upb_Message*)self->msg;
324
+ if (upb_Message_IsFrozen(self->msg)) {
325
+ return Message_getfield_frozen(self->msg, f, self->arena);
326
+ }
327
+ upb_Message* msg = Message_GetMutable(_self, NULL);
296
328
  upb_Arena* arena = Arena_get(self->arena);
297
329
  if (upb_FieldDef_IsMap(f)) {
298
330
  upb_Map* map = upb_Message_Mutable(msg, f, arena).map;
@@ -305,12 +337,12 @@ VALUE Message_getfield(VALUE _self, const upb_FieldDef* f) {
305
337
  upb_Array* arr = upb_Message_Mutable(msg, f, arena).array;
306
338
  return RepeatedField_GetRubyWrapper(arr, TypeInfo_get(f), self->arena);
307
339
  } else if (upb_FieldDef_IsSubMessage(f)) {
308
- if (!upb_Message_HasFieldByDef(self->msg, f)) return Qnil;
340
+ if (!upb_Message_HasFieldByDef(msg, f)) return Qnil;
309
341
  upb_Message* submsg = upb_Message_Mutable(msg, f, arena).msg;
310
342
  const upb_MessageDef* m = upb_FieldDef_MessageSubDef(f);
311
343
  return Message_GetRubyWrapper(submsg, m, self->arena);
312
344
  } else {
313
- upb_MessageValue msgval = upb_Message_GetFieldByDef(self->msg, f);
345
+ upb_MessageValue msgval = upb_Message_GetFieldByDef(msg, f);
314
346
  return Convert_UpbToRuby(msgval, TypeInfo_get(f), self->arena);
315
347
  }
316
348
  }
@@ -330,7 +362,8 @@ static VALUE Message_field_accessor(VALUE _self, const upb_FieldDef* f,
330
362
  if (!upb_FieldDef_HasPresence(f)) {
331
363
  rb_raise(rb_eRuntimeError, "Field does not have presence.");
332
364
  }
333
- return upb_Message_HasFieldByDef(Message_Get(_self, NULL), f);
365
+ return upb_Message_HasFieldByDef(Message_Get(_self, NULL), f) ? Qtrue
366
+ : Qfalse;
334
367
  case METHOD_WRAPPER_GETTER: {
335
368
  Message* self = ruby_to_Message(_self);
336
369
  if (upb_Message_HasFieldByDef(self->msg, f)) {
@@ -434,7 +467,6 @@ static VALUE Message_method_missing(int argc, VALUE* argv, VALUE _self) {
434
467
  if (argc != 2) {
435
468
  rb_raise(rb_eArgError, "Expected 2 arguments, received %d", argc);
436
469
  }
437
- rb_check_frozen(_self);
438
470
  break;
439
471
  default:
440
472
  if (argc != 1) {
@@ -488,7 +520,8 @@ static int Map_initialize_kwarg(VALUE key, VALUE val, VALUE _self) {
488
520
  k = Convert_RubyToUpb(key, "", map_init->key_type, NULL);
489
521
 
490
522
  if (map_init->val_type.type == kUpb_CType_Message && TYPE(val) == T_HASH) {
491
- upb_MiniTable* t = upb_MessageDef_MiniTable(map_init->val_type.def.msgdef);
523
+ const upb_MiniTable* t =
524
+ upb_MessageDef_MiniTable(map_init->val_type.def.msgdef);
492
525
  upb_Message* msg = upb_Message_New(t, map_init->arena);
493
526
  Message_InitFromValue(msg, map_init->val_type.def.msgdef, val,
494
527
  map_init->arena);
@@ -519,7 +552,7 @@ static upb_MessageValue MessageValue_FromValue(VALUE val, TypeInfo info,
519
552
  upb_Arena* arena) {
520
553
  if (info.type == kUpb_CType_Message) {
521
554
  upb_MessageValue msgval;
522
- upb_MiniTable* t = upb_MessageDef_MiniTable(info.def.msgdef);
555
+ const upb_MiniTable* t = upb_MessageDef_MiniTable(info.def.msgdef);
523
556
  upb_Message* msg = upb_Message_New(t, arena);
524
557
  Message_InitFromValue(msg, info.def.msgdef, val, arena);
525
558
  msgval.msg_val = msg;
@@ -635,7 +668,7 @@ static VALUE Message_initialize(int argc, VALUE* argv, VALUE _self) {
635
668
  Message* self = ruby_to_Message(_self);
636
669
  VALUE arena_rb = Arena_new();
637
670
  upb_Arena* arena = Arena_get(arena_rb);
638
- upb_MiniTable* t = upb_MessageDef_MiniTable(self->msgdef);
671
+ const upb_MiniTable* t = upb_MessageDef_MiniTable(self->msgdef);
639
672
  upb_Message* msg = upb_Message_New(t, arena);
640
673
 
641
674
  Message_InitPtr(_self, msg, arena_rb);
@@ -660,28 +693,12 @@ static VALUE Message_dup(VALUE _self) {
660
693
  Message* self = ruby_to_Message(_self);
661
694
  VALUE new_msg = rb_class_new_instance(0, NULL, CLASS_OF(_self));
662
695
  Message* new_msg_self = ruby_to_Message(new_msg);
663
- size_t size = upb_MessageDef_MiniTable(self->msgdef)->size;
664
-
665
- // TODO
666
- // TODO
667
- memcpy((upb_Message*)new_msg_self->msg, self->msg, size);
696
+ const upb_MiniTable* m = upb_MessageDef_MiniTable(self->msgdef);
697
+ upb_Message_ShallowCopy((upb_Message*)new_msg_self->msg, self->msg, m);
668
698
  Arena_fuse(self->arena, Arena_get(new_msg_self->arena));
669
699
  return new_msg;
670
700
  }
671
701
 
672
- // Support function for Message_eq, and also used by other #eq functions.
673
- bool Message_Equal(const upb_Message* m1, const upb_Message* m2,
674
- const upb_MessageDef* m) {
675
- upb_Status status;
676
- upb_Status_Clear(&status);
677
- bool return_value = shared_Message_Equal(m1, m2, m, &status);
678
- if (upb_Status_IsOk(&status)) {
679
- return return_value;
680
- } else {
681
- rb_raise(cParseError, upb_Status_ErrorMessage(&status));
682
- }
683
- }
684
-
685
702
  /*
686
703
  * call-seq:
687
704
  * Message.==(other) => boolean
@@ -698,7 +715,10 @@ static VALUE Message_eq(VALUE _self, VALUE _other) {
698
715
  Message* other = ruby_to_Message(_other);
699
716
  assert(self->msgdef == other->msgdef);
700
717
 
701
- return Message_Equal(self->msg, other->msg, self->msgdef) ? Qtrue : Qfalse;
718
+ const upb_MiniTable* m = upb_MessageDef_MiniTable(self->msgdef);
719
+ const int options = 0;
720
+ return upb_Message_IsEqual(self->msg, other->msg, m, options) ? Qtrue
721
+ : Qfalse;
702
722
  }
703
723
 
704
724
  uint64_t Message_Hash(const upb_Message* msg, const upb_MessageDef* m,
@@ -709,7 +729,8 @@ uint64_t Message_Hash(const upb_Message* msg, const upb_MessageDef* m,
709
729
  if (upb_Status_IsOk(&status)) {
710
730
  return return_value;
711
731
  } else {
712
- rb_raise(cParseError, upb_Status_ErrorMessage(&status));
732
+ rb_raise(cParseError, "Message_Hash(): %s",
733
+ upb_Status_ErrorMessage(&status));
713
734
  }
714
735
  }
715
736
 
@@ -766,58 +787,34 @@ static VALUE Message_CreateHash(const upb_Message* msg,
766
787
  if (!msg) return Qnil;
767
788
 
768
789
  VALUE hash = rb_hash_new();
769
- int n = upb_MessageDef_FieldCount(m);
770
- bool is_proto2;
771
-
772
- // We currently have a few behaviors that are specific to proto2.
773
- // This is unfortunate, we should key behaviors off field attributes (like
774
- // whether a field has presence), not proto2 vs. proto3. We should see if we
775
- // can change this without breaking users.
776
- is_proto2 = upb_MessageDef_Syntax(m) == kUpb_Syntax_Proto2;
777
-
778
- for (int i = 0; i < n; i++) {
779
- const upb_FieldDef* field = upb_MessageDef_Field(m, i);
780
- TypeInfo type_info = TypeInfo_get(field);
781
- upb_MessageValue msgval;
782
- VALUE msg_value;
783
- VALUE msg_key;
784
-
785
- if (!is_proto2 && upb_FieldDef_IsSubMessage(field) &&
786
- !upb_FieldDef_IsRepeated(field) &&
787
- !upb_Message_HasFieldByDef(msg, field)) {
788
- // TODO: Legacy behavior, remove when we fix the is_proto2 differences.
789
- msg_key = ID2SYM(rb_intern(upb_FieldDef_Name(field)));
790
- rb_hash_aset(hash, msg_key, Qnil);
791
- continue;
792
- }
790
+ size_t iter = kUpb_Message_Begin;
791
+ const upb_DefPool* pool = upb_FileDef_Pool(upb_MessageDef_File(m));
792
+ const upb_FieldDef* field;
793
+ upb_MessageValue val;
793
794
 
794
- // Do not include fields that are not present (oneof or optional fields).
795
- if (is_proto2 && upb_FieldDef_HasPresence(field) &&
796
- !upb_Message_HasFieldByDef(msg, field)) {
795
+ while (upb_Message_Next(msg, m, pool, &field, &val, &iter)) {
796
+ if (upb_FieldDef_IsExtension(field)) {
797
+ // TODO: allow extensions once we have decided what naming scheme the
798
+ // symbol should use. eg. :"[pkg.ext]"
797
799
  continue;
798
800
  }
799
801
 
800
- msg_key = ID2SYM(rb_intern(upb_FieldDef_Name(field)));
801
- msgval = upb_Message_GetFieldByDef(msg, field);
802
-
803
- // Proto2 omits empty map/repeated filds also.
802
+ TypeInfo type_info = TypeInfo_get(field);
803
+ VALUE msg_value;
804
804
 
805
805
  if (upb_FieldDef_IsMap(field)) {
806
806
  const upb_MessageDef* entry_m = upb_FieldDef_MessageSubDef(field);
807
807
  const upb_FieldDef* key_f = upb_MessageDef_FindFieldByNumber(entry_m, 1);
808
808
  const upb_FieldDef* val_f = upb_MessageDef_FindFieldByNumber(entry_m, 2);
809
809
  upb_CType key_type = upb_FieldDef_CType(key_f);
810
- msg_value = Map_CreateHash(msgval.map_val, key_type, TypeInfo_get(val_f));
810
+ msg_value = Map_CreateHash(val.map_val, key_type, TypeInfo_get(val_f));
811
811
  } else if (upb_FieldDef_IsRepeated(field)) {
812
- if (is_proto2 &&
813
- (!msgval.array_val || upb_Array_Size(msgval.array_val) == 0)) {
814
- continue;
815
- }
816
- msg_value = RepeatedField_CreateArray(msgval.array_val, type_info);
812
+ msg_value = RepeatedField_CreateArray(val.array_val, type_info);
817
813
  } else {
818
- msg_value = Scalar_CreateHash(msgval, type_info);
814
+ msg_value = Scalar_CreateHash(val, type_info);
819
815
  }
820
816
 
817
+ VALUE msg_key = ID2SYM(rb_intern(upb_FieldDef_Name(field)));
821
818
  rb_hash_aset(hash, msg_key, msg_value);
822
819
  }
823
820
 
@@ -845,43 +842,42 @@ static VALUE Message_to_h(VALUE _self) {
845
842
 
846
843
  /*
847
844
  * call-seq:
848
- * Message.freeze => self
845
+ * Message.frozen? => bool
849
846
  *
850
- * Freezes the message object. We have to intercept this so we can pin the
851
- * Ruby object into memory so we don't forget it's frozen.
847
+ * Returns true if the message is frozen in either Ruby or the underlying
848
+ * representation. Freezes the Ruby message object if it is not already frozen
849
+ * in Ruby but it is frozen in the underlying representation.
852
850
  */
853
- static VALUE Message_freeze(VALUE _self) {
851
+ VALUE Message_frozen(VALUE _self) {
854
852
  Message* self = ruby_to_Message(_self);
855
- if (!RB_OBJ_FROZEN(_self)) {
856
- Arena_Pin(self->arena, _self);
857
- RB_OBJ_FREEZE(_self);
853
+ if (!upb_Message_IsFrozen(self->msg)) {
854
+ PBRUBY_ASSERT(!RB_OBJ_FROZEN(_self));
855
+ return Qfalse;
858
856
  }
859
- return _self;
857
+
858
+ // Lazily freeze the Ruby wrapper.
859
+ if (!RB_OBJ_FROZEN(_self)) RB_OBJ_FREEZE(_self);
860
+ return Qtrue;
860
861
  }
861
862
 
862
863
  /*
863
- * Deep freezes the message object recursively.
864
- * Internal use only.
864
+ * call-seq:
865
+ * Message.freeze => self
866
+ *
867
+ * Freezes the message object. We have to intercept this so we can freeze the
868
+ * underlying representation, not just the Ruby wrapper.
865
869
  */
866
- VALUE Message_internal_deep_freeze(VALUE _self) {
870
+ VALUE Message_freeze(VALUE _self) {
867
871
  Message* self = ruby_to_Message(_self);
868
- Message_freeze(_self);
869
-
870
- int n = upb_MessageDef_FieldCount(self->msgdef);
871
- for (int i = 0; i < n; i++) {
872
- const upb_FieldDef* f = upb_MessageDef_Field(self->msgdef, i);
873
- VALUE field = Message_getfield(_self, f);
874
-
875
- if (field != Qnil) {
876
- if (upb_FieldDef_IsMap(f)) {
877
- Map_internal_deep_freeze(field);
878
- } else if (upb_FieldDef_IsRepeated(f)) {
879
- RepeatedField_internal_deep_freeze(field);
880
- } else if (upb_FieldDef_IsSubMessage(f)) {
881
- Message_internal_deep_freeze(field);
882
- }
883
- }
872
+ if (RB_OBJ_FROZEN(_self)) {
873
+ PBRUBY_ASSERT(upb_Message_IsFrozen(self->msg));
874
+ return _self;
884
875
  }
876
+ if (!upb_Message_IsFrozen(self->msg)) {
877
+ upb_Message_Freeze(Message_GetMutable(_self, NULL),
878
+ upb_MessageDef_MiniTable(self->msgdef));
879
+ }
880
+ RB_OBJ_FREEZE(_self);
885
881
  return _self;
886
882
  }
887
883
 
@@ -987,7 +983,7 @@ VALUE Message_decode_bytes(int size, const char* bytes, int options,
987
983
  rb_raise(cParseError, "Error occurred during parsing");
988
984
  }
989
985
  if (freeze) {
990
- Message_internal_deep_freeze(msg_rb);
986
+ Message_freeze(msg_rb);
991
987
  }
992
988
  return msg_rb;
993
989
  }
@@ -1009,9 +1005,6 @@ static VALUE Message_decode_json(int argc, VALUE* argv, VALUE klass) {
1009
1005
  int options = 0;
1010
1006
  upb_Status status;
1011
1007
 
1012
- // TODO: use this message's pool instead.
1013
- const upb_DefPool* symtab = DescriptorPool_GetSymtab(generated_pool);
1014
-
1015
1008
  if (argc < 1 || argc > 2) {
1016
1009
  rb_raise(rb_eArgError, "Expected 1 or 2 arguments.");
1017
1010
  }
@@ -1045,11 +1038,19 @@ static VALUE Message_decode_json(int argc, VALUE* argv, VALUE klass) {
1045
1038
  }
1046
1039
 
1047
1040
  upb_Status_Clear(&status);
1048
- if (!upb_JsonDecode(RSTRING_PTR(data), RSTRING_LEN(data),
1049
- (upb_Message*)msg->msg, msg->msgdef, symtab, options,
1050
- Arena_get(msg->arena), &status)) {
1051
- rb_raise(cParseError, "Error occurred during parsing: %s",
1052
- upb_Status_ErrorMessage(&status));
1041
+ const upb_DefPool* pool = upb_FileDef_Pool(upb_MessageDef_File(msg->msgdef));
1042
+
1043
+ int result = upb_JsonDecodeDetectingNonconformance(
1044
+ RSTRING_PTR(data), RSTRING_LEN(data), (upb_Message*)msg->msg,
1045
+ msg->msgdef, pool, options, Arena_get(msg->arena), &status);
1046
+
1047
+ switch (result) {
1048
+ case kUpb_JsonDecodeResult_Ok:
1049
+ break;
1050
+ case kUpb_JsonDecodeResult_Error:
1051
+ rb_raise(cParseError, "Error occurred during parsing: %s",
1052
+ upb_Status_ErrorMessage(&status));
1053
+ break;
1053
1054
  }
1054
1055
 
1055
1056
  return msg_rb;
@@ -1125,9 +1126,6 @@ static VALUE Message_encode_json(int argc, VALUE* argv, VALUE klass) {
1125
1126
  size_t size;
1126
1127
  upb_Status status;
1127
1128
 
1128
- // TODO: use this message's pool instead.
1129
- const upb_DefPool* symtab = DescriptorPool_GetSymtab(generated_pool);
1130
-
1131
1129
  if (argc < 1 || argc > 2) {
1132
1130
  rb_raise(rb_eArgError, "Expected 1 or 2 arguments.");
1133
1131
  }
@@ -1162,8 +1160,9 @@ static VALUE Message_encode_json(int argc, VALUE* argv, VALUE klass) {
1162
1160
  }
1163
1161
 
1164
1162
  upb_Status_Clear(&status);
1165
- size = upb_JsonEncode(msg->msg, msg->msgdef, symtab, options, buf,
1166
- sizeof(buf), &status);
1163
+ const upb_DefPool* pool = upb_FileDef_Pool(upb_MessageDef_File(msg->msgdef));
1164
+ size = upb_JsonEncode(msg->msg, msg->msgdef, pool, options, buf, sizeof(buf),
1165
+ &status);
1167
1166
 
1168
1167
  if (!upb_Status_IsOk(&status)) {
1169
1168
  rb_raise(cParseError, "Error occurred during encoding: %s",
@@ -1173,7 +1172,7 @@ static VALUE Message_encode_json(int argc, VALUE* argv, VALUE klass) {
1173
1172
  VALUE ret;
1174
1173
  if (size >= sizeof(buf)) {
1175
1174
  char* buf2 = malloc(size + 1);
1176
- upb_JsonEncode(msg->msg, msg->msgdef, symtab, options, buf2, size + 1,
1175
+ upb_JsonEncode(msg->msg, msg->msgdef, pool, options, buf2, size + 1,
1177
1176
  &status);
1178
1177
  ret = rb_str_new(buf2, size);
1179
1178
  free(buf2);
@@ -1399,6 +1398,7 @@ static void Message_define_class(VALUE klass) {
1399
1398
  rb_define_method(klass, "==", Message_eq, 1);
1400
1399
  rb_define_method(klass, "eql?", Message_eq, 1);
1401
1400
  rb_define_method(klass, "freeze", Message_freeze, 0);
1401
+ rb_define_method(klass, "frozen?", Message_frozen, 0);
1402
1402
  rb_define_method(klass, "hash", Message_hash, 0);
1403
1403
  rb_define_method(klass, "to_h", Message_to_h, 0);
1404
1404
  rb_define_method(klass, "inspect", Message_inspect, 0);
@@ -36,7 +36,7 @@ const upb_Message* Message_GetUpbMessage(VALUE value, const upb_MessageDef* m,
36
36
 
37
37
  // Gets or constructs a Ruby wrapper object for the given message. The wrapper
38
38
  // object will reference |arena| and ensure that it outlives this object.
39
- VALUE Message_GetRubyWrapper(upb_Message* msg, const upb_MessageDef* m,
39
+ VALUE Message_GetRubyWrapper(const upb_Message* msg, const upb_MessageDef* m,
40
40
  VALUE arena);
41
41
 
42
42
  // Gets the given field from this message.
@@ -54,10 +54,6 @@ uint64_t Message_Hash(const upb_Message* msg, const upb_MessageDef* m,
54
54
  upb_Message* Message_deep_copy(const upb_Message* msg, const upb_MessageDef* m,
55
55
  upb_Arena* arena);
56
56
 
57
- // Returns true if these two messages are equal.
58
- bool Message_Equal(const upb_Message* m1, const upb_Message* m2,
59
- const upb_MessageDef* m);
60
-
61
57
  // Checks that this Ruby object is a message, and raises an exception if not.
62
58
  void Message_CheckClass(VALUE klass);
63
59
 
@@ -78,7 +74,7 @@ VALUE Message_decode_bytes(int size, const char* bytes, int options,
78
74
  VALUE klass, bool freeze);
79
75
 
80
76
  // Recursively freeze message
81
- VALUE Message_internal_deep_freeze(VALUE _self);
77
+ VALUE Message_freeze(VALUE _self);
82
78
 
83
79
  // Call at startup to register all types in this module.
84
80
  void Message_register(VALUE protobuf);
@@ -7,8 +7,6 @@
7
7
 
8
8
  #include "protobuf.h"
9
9
 
10
- #include <ruby/version.h>
11
-
12
10
  #include "defs.h"
13
11
  #include "map.h"
14
12
  #include "message.h"
@@ -164,11 +162,23 @@ static void Arena_free(void *data) {
164
162
  xfree(arena);
165
163
  }
166
164
 
165
+ static size_t Arena_memsize(const void *data) {
166
+ const Arena *arena = data;
167
+ size_t fused_count;
168
+ size_t memsize = upb_Arena_SpaceAllocated(arena->arena, &fused_count);
169
+ if (fused_count > 1) {
170
+ // If other arena were fused we attribute an equal
171
+ // share of memory usage to each one.
172
+ memsize /= fused_count;
173
+ }
174
+ return memsize + sizeof(Arena);
175
+ }
176
+
167
177
  static VALUE cArena;
168
178
 
169
179
  const rb_data_type_t Arena_type = {
170
180
  "Google::Protobuf::Internal::Arena",
171
- {Arena_mark, Arena_free, NULL},
181
+ {Arena_mark, Arena_free, Arena_memsize},
172
182
  .flags = RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED,
173
183
  };
174
184
 
@@ -209,15 +219,6 @@ void Arena_fuse(VALUE _arena, upb_Arena *other) {
209
219
 
210
220
  VALUE Arena_new() { return Arena_alloc(cArena); }
211
221
 
212
- void Arena_Pin(VALUE _arena, VALUE obj) {
213
- Arena *arena;
214
- TypedData_Get_Struct(_arena, Arena, &Arena_type, arena);
215
- if (arena->pinned_objs == Qnil) {
216
- RB_OBJ_WRITE(_arena, &arena->pinned_objs, rb_ary_new());
217
- }
218
- rb_ary_push(arena->pinned_objs, obj);
219
- }
220
-
221
222
  void Arena_register(VALUE module) {
222
223
  VALUE internal = rb_define_module_under(module, "Internal");
223
224
  VALUE klass = rb_define_class_under(internal, "Arena", rb_cObject);
@@ -241,16 +242,17 @@ static void ObjectCache_Init(VALUE protobuf) {
241
242
  item_try_add = rb_intern("try_add");
242
243
 
243
244
  rb_gc_register_address(&weak_obj_cache);
245
+ VALUE internal = rb_const_get(protobuf, rb_intern("Internal"));
244
246
  #if SIZEOF_LONG >= SIZEOF_VALUE
245
- VALUE cache_class = rb_const_get(protobuf, rb_intern("ObjectCache"));
247
+ VALUE cache_class = rb_const_get(internal, rb_intern("ObjectCache"));
246
248
  #else
247
- VALUE cache_class = rb_const_get(protobuf, rb_intern("LegacyObjectCache"));
249
+ VALUE cache_class = rb_const_get(internal, rb_intern("LegacyObjectCache"));
248
250
  #endif
249
251
 
250
252
  weak_obj_cache = rb_class_new_instance(0, NULL, cache_class);
251
- rb_const_set(protobuf, rb_intern("OBJECT_CACHE"), weak_obj_cache);
252
- rb_const_set(protobuf, rb_intern("SIZEOF_LONG"), INT2NUM(SIZEOF_LONG));
253
- rb_const_set(protobuf, rb_intern("SIZEOF_VALUE"), INT2NUM(SIZEOF_VALUE));
253
+ rb_const_set(internal, rb_intern("OBJECT_CACHE"), weak_obj_cache);
254
+ rb_const_set(internal, rb_intern("SIZEOF_LONG"), INT2NUM(SIZEOF_LONG));
255
+ rb_const_set(internal, rb_intern("SIZEOF_VALUE"), INT2NUM(SIZEOF_VALUE));
254
256
  }
255
257
 
256
258
  static VALUE ObjectCache_GetKey(const void *key) {
@@ -284,7 +286,8 @@ VALUE ObjectCache_Get(const void *key) {
284
286
  static VALUE Google_Protobuf_discard_unknown(VALUE self, VALUE msg_rb) {
285
287
  const upb_MessageDef *m;
286
288
  upb_Message *msg = Message_GetMutable(msg_rb, &m);
287
- if (!upb_Message_DiscardUnknown(msg, m, 128)) {
289
+ const upb_DefPool* ext_pool = upb_FileDef_Pool(upb_MessageDef_File(m));
290
+ if (!upb_Message_DiscardUnknown(msg, m, ext_pool, 128)) {
288
291
  rb_raise(rb_eRuntimeError, "Messages nested too deeply.");
289
292
  }
290
293
 
@@ -341,3 +344,14 @@ __attribute__((visibility("default"))) void Init_protobuf_c() {
341
344
  rb_define_singleton_method(protobuf, "deep_copy", Google_Protobuf_deep_copy,
342
345
  1);
343
346
  }
347
+
348
+ // -----------------------------------------------------------------------------
349
+ // Utilities
350
+ // -----------------------------------------------------------------------------
351
+
352
+ // Raises a Ruby error if val is frozen in Ruby or UPB.
353
+ void Protobuf_CheckNotFrozen(VALUE val, bool upb_frozen) {
354
+ if (RB_UNLIKELY(rb_obj_frozen_p(val)||upb_frozen)) {
355
+ rb_error_frozen_object(val);
356
+ }
357
+ }
@@ -50,13 +50,6 @@ upb_Arena* Arena_get(VALUE arena);
50
50
  // possible.
51
51
  void Arena_fuse(VALUE arena, upb_Arena* other);
52
52
 
53
- // Pins this Ruby object to the lifetime of this arena, so that as long as the
54
- // arena is alive this object will not be collected.
55
- //
56
- // We use this to guarantee that the "frozen" bit on the object will be
57
- // remembered, even if the user drops their reference to this precise object.
58
- void Arena_Pin(VALUE arena, VALUE obj);
59
-
60
53
  // -----------------------------------------------------------------------------
61
54
  // ObjectCache
62
55
  // -----------------------------------------------------------------------------
@@ -105,6 +98,9 @@ extern VALUE cTypeError;
105
98
  rb_bug("Assertion failed at %s:%d, expr: %s", __FILE__, __LINE__, #expr)
106
99
  #endif
107
100
 
101
+ // Raises a Ruby error if val is frozen in Ruby or upb_frozen is true.
102
+ void Protobuf_CheckNotFrozen(VALUE val, bool upb_frozen);
103
+
108
104
  #define PBRUBY_MAX(x, y) (((x) > (y)) ? (x) : (y))
109
105
 
110
106
  #define UPB_UNUSED(var) (void)var