google-protobuf 3.14.0-x64-mingw32 → 3.15.0.rc.1-x64-mingw32
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.
Potentially problematic release.
This version of google-protobuf might be problematic. Click here for more details.
- checksums.yaml +4 -4
- data/ext/google/protobuf_c/convert.c +349 -0
- data/ext/google/protobuf_c/convert.h +72 -0
- data/ext/google/protobuf_c/defs.c +924 -731
- data/ext/google/protobuf_c/defs.h +107 -0
- data/ext/google/protobuf_c/extconf.rb +4 -5
- data/ext/google/protobuf_c/map.c +308 -456
- data/ext/google/protobuf_c/map.h +66 -0
- data/ext/google/protobuf_c/message.c +889 -433
- data/ext/google/protobuf_c/message.h +98 -0
- data/ext/google/protobuf_c/protobuf.c +325 -65
- data/ext/google/protobuf_c/protobuf.h +42 -597
- data/ext/google/protobuf_c/repeated_field.c +291 -321
- data/ext/google/protobuf_c/repeated_field.h +62 -0
- data/ext/google/protobuf_c/ruby-upb.c +8915 -0
- data/ext/google/protobuf_c/{upb.h → ruby-upb.h} +1362 -3687
- data/ext/google/protobuf_c/third_party/wyhash/wyhash.h +145 -0
- data/lib/google/2.3/protobuf_c.so +0 -0
- data/lib/google/2.4/protobuf_c.so +0 -0
- data/lib/google/2.5/protobuf_c.so +0 -0
- data/lib/google/2.6/protobuf_c.so +0 -0
- data/lib/google/2.7/protobuf_c.so +0 -0
- data/lib/google/3.0/protobuf_c.so +0 -0
- data/tests/basic.rb +35 -6
- metadata +19 -13
- data/ext/google/protobuf_c/encode_decode.c +0 -1795
- data/ext/google/protobuf_c/storage.c +0 -1198
- data/ext/google/protobuf_c/upb.c +0 -13817
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 0d50209c660ab2facb21d43ef84e79c92dcd115f5f2789c6868bc5574d7d1cc7
|
4
|
+
data.tar.gz: e4d1bbd6d933b47736c200a9839b8a6c84eff8f0506879cb326a64fc9bd43afb
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: d71040a8ff1277d3a8d1bec2c3074eafaf480c2d023569e60849bb0a9532bd3363aa804776e6d1bae4744a78b081ad6685b2da18be4ebf52d793bdcb491d2cef
|
7
|
+
data.tar.gz: 763ef294fe993c3306029383139dd6be95a2ab06b7b698bfc4aa81af9f6b42f405be6d99ac0c492ad32fb3070f10a73f182dd26885371f919bd857216941a763
|
@@ -0,0 +1,349 @@
|
|
1
|
+
// Protocol Buffers - Google's data interchange format
|
2
|
+
// Copyright 2008 Google Inc. All rights reserved.
|
3
|
+
// https://developers.google.com/protocol-buffers/
|
4
|
+
//
|
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.
|
30
|
+
|
31
|
+
// -----------------------------------------------------------------------------
|
32
|
+
// Ruby <-> upb data conversion functions.
|
33
|
+
//
|
34
|
+
// This file Also contains a few other assorted algorithms on upb_msgval.
|
35
|
+
//
|
36
|
+
// None of the algorithms in this file require any access to the internal
|
37
|
+
// representation of Ruby or upb objects.
|
38
|
+
// -----------------------------------------------------------------------------
|
39
|
+
|
40
|
+
#include "convert.h"
|
41
|
+
|
42
|
+
#include "message.h"
|
43
|
+
#include "protobuf.h"
|
44
|
+
#include "third_party/wyhash/wyhash.h"
|
45
|
+
|
46
|
+
static upb_strview Convert_StringData(VALUE str, upb_arena *arena) {
|
47
|
+
upb_strview ret;
|
48
|
+
if (arena) {
|
49
|
+
char *ptr = upb_arena_malloc(arena, RSTRING_LEN(str));
|
50
|
+
memcpy(ptr, RSTRING_PTR(str), RSTRING_LEN(str));
|
51
|
+
ret.data = ptr;
|
52
|
+
} else {
|
53
|
+
// Data is only needed temporarily (within map lookup).
|
54
|
+
ret.data = RSTRING_PTR(str);
|
55
|
+
}
|
56
|
+
ret.size = RSTRING_LEN(str);
|
57
|
+
return ret;
|
58
|
+
}
|
59
|
+
|
60
|
+
static bool is_ruby_num(VALUE value) {
|
61
|
+
return (TYPE(value) == T_FLOAT ||
|
62
|
+
TYPE(value) == T_FIXNUM ||
|
63
|
+
TYPE(value) == T_BIGNUM);
|
64
|
+
}
|
65
|
+
|
66
|
+
static void Convert_CheckInt(const char* name, upb_fieldtype_t type,
|
67
|
+
VALUE val) {
|
68
|
+
if (!is_ruby_num(val)) {
|
69
|
+
rb_raise(cTypeError,
|
70
|
+
"Expected number type for integral field '%s' (given %s).", name,
|
71
|
+
rb_class2name(CLASS_OF(val)));
|
72
|
+
}
|
73
|
+
|
74
|
+
// NUM2{INT,UINT,LL,ULL} macros do the appropriate range checks on upper
|
75
|
+
// bound; we just need to do precision checks (i.e., disallow rounding) and
|
76
|
+
// check for < 0 on unsigned types.
|
77
|
+
if (TYPE(val) == T_FLOAT) {
|
78
|
+
double dbl_val = NUM2DBL(val);
|
79
|
+
if (floor(dbl_val) != dbl_val) {
|
80
|
+
rb_raise(rb_eRangeError,
|
81
|
+
"Non-integral floating point value assigned to integer field "
|
82
|
+
"'%s' (given %s).",
|
83
|
+
name, rb_class2name(CLASS_OF(val)));
|
84
|
+
}
|
85
|
+
}
|
86
|
+
if (type == UPB_TYPE_UINT32 || type == UPB_TYPE_UINT64) {
|
87
|
+
if (NUM2DBL(val) < 0) {
|
88
|
+
rb_raise(
|
89
|
+
rb_eRangeError,
|
90
|
+
"Assigning negative value to unsigned integer field '%s' (given %s).",
|
91
|
+
name, rb_class2name(CLASS_OF(val)));
|
92
|
+
}
|
93
|
+
}
|
94
|
+
}
|
95
|
+
|
96
|
+
static int32_t Convert_ToEnum(VALUE value, const char* name,
|
97
|
+
const upb_enumdef* e) {
|
98
|
+
int32_t val;
|
99
|
+
|
100
|
+
switch (TYPE(value)) {
|
101
|
+
case T_FLOAT:
|
102
|
+
case T_FIXNUM:
|
103
|
+
case T_BIGNUM:
|
104
|
+
Convert_CheckInt(name, UPB_TYPE_INT32, value);
|
105
|
+
val = NUM2INT(value);
|
106
|
+
break;
|
107
|
+
case T_STRING:
|
108
|
+
if (!upb_enumdef_ntoi(e, RSTRING_PTR(value), RSTRING_LEN(value), &val)) {
|
109
|
+
goto unknownval;
|
110
|
+
}
|
111
|
+
break;
|
112
|
+
case T_SYMBOL:
|
113
|
+
if (!upb_enumdef_ntoiz(e, rb_id2name(SYM2ID(value)), &val)) {
|
114
|
+
goto unknownval;
|
115
|
+
}
|
116
|
+
break;
|
117
|
+
default:
|
118
|
+
rb_raise(cTypeError,
|
119
|
+
"Expected number or symbol type for enum field '%s'.", name);
|
120
|
+
}
|
121
|
+
|
122
|
+
return val;
|
123
|
+
|
124
|
+
unknownval:
|
125
|
+
rb_raise(rb_eRangeError, "Unknown symbol value for enum field '%s'.", name);
|
126
|
+
}
|
127
|
+
|
128
|
+
upb_msgval Convert_RubyToUpb(VALUE value, const char* name, TypeInfo type_info,
|
129
|
+
upb_arena* arena) {
|
130
|
+
upb_msgval ret;
|
131
|
+
|
132
|
+
switch (type_info.type) {
|
133
|
+
case UPB_TYPE_FLOAT:
|
134
|
+
if (!is_ruby_num(value)) {
|
135
|
+
rb_raise(cTypeError, "Expected number type for float field '%s' (given %s).",
|
136
|
+
name, rb_class2name(CLASS_OF(value)));
|
137
|
+
}
|
138
|
+
ret.float_val = NUM2DBL(value);
|
139
|
+
break;
|
140
|
+
case UPB_TYPE_DOUBLE:
|
141
|
+
if (!is_ruby_num(value)) {
|
142
|
+
rb_raise(cTypeError, "Expected number type for double field '%s' (given %s).",
|
143
|
+
name, rb_class2name(CLASS_OF(value)));
|
144
|
+
}
|
145
|
+
ret.double_val = NUM2DBL(value);
|
146
|
+
break;
|
147
|
+
case UPB_TYPE_BOOL: {
|
148
|
+
if (value == Qtrue) {
|
149
|
+
ret.bool_val = 1;
|
150
|
+
} else if (value == Qfalse) {
|
151
|
+
ret.bool_val = 0;
|
152
|
+
} else {
|
153
|
+
rb_raise(cTypeError, "Invalid argument for boolean field '%s' (given %s).",
|
154
|
+
name, rb_class2name(CLASS_OF(value)));
|
155
|
+
}
|
156
|
+
break;
|
157
|
+
}
|
158
|
+
case UPB_TYPE_STRING: {
|
159
|
+
VALUE utf8 = rb_enc_from_encoding(rb_utf8_encoding());
|
160
|
+
if (CLASS_OF(value) == rb_cSymbol) {
|
161
|
+
value = rb_funcall(value, rb_intern("to_s"), 0);
|
162
|
+
} else if (CLASS_OF(value) != rb_cString) {
|
163
|
+
rb_raise(cTypeError, "Invalid argument for string field '%s' (given %s).",
|
164
|
+
name, rb_class2name(CLASS_OF(value)));
|
165
|
+
}
|
166
|
+
|
167
|
+
if (rb_obj_encoding(value) != utf8) {
|
168
|
+
// Note: this will not duplicate underlying string data unless necessary.
|
169
|
+
value = rb_str_encode(value, utf8, 0, Qnil);
|
170
|
+
|
171
|
+
if (rb_enc_str_coderange(value) == ENC_CODERANGE_BROKEN) {
|
172
|
+
rb_raise(rb_eEncodingError, "String is invalid UTF-8");
|
173
|
+
}
|
174
|
+
}
|
175
|
+
|
176
|
+
ret.str_val = Convert_StringData(value, arena);
|
177
|
+
break;
|
178
|
+
}
|
179
|
+
case UPB_TYPE_BYTES: {
|
180
|
+
VALUE bytes = rb_enc_from_encoding(rb_ascii8bit_encoding());
|
181
|
+
if (CLASS_OF(value) != rb_cString) {
|
182
|
+
rb_raise(cTypeError, "Invalid argument for bytes field '%s' (given %s).",
|
183
|
+
name, rb_class2name(CLASS_OF(value)));
|
184
|
+
}
|
185
|
+
|
186
|
+
if (rb_obj_encoding(value) != bytes) {
|
187
|
+
// Note: this will not duplicate underlying string data unless necessary.
|
188
|
+
// TODO(haberman): is this really necessary to get raw bytes?
|
189
|
+
value = rb_str_encode(value, bytes, 0, Qnil);
|
190
|
+
}
|
191
|
+
|
192
|
+
ret.str_val = Convert_StringData(value, arena);
|
193
|
+
break;
|
194
|
+
}
|
195
|
+
case UPB_TYPE_MESSAGE:
|
196
|
+
ret.msg_val =
|
197
|
+
Message_GetUpbMessage(value, type_info.def.msgdef, name, arena);
|
198
|
+
break;
|
199
|
+
case UPB_TYPE_ENUM:
|
200
|
+
ret.int32_val = Convert_ToEnum(value, name, type_info.def.enumdef);
|
201
|
+
break;
|
202
|
+
case UPB_TYPE_INT32:
|
203
|
+
case UPB_TYPE_INT64:
|
204
|
+
case UPB_TYPE_UINT32:
|
205
|
+
case UPB_TYPE_UINT64:
|
206
|
+
Convert_CheckInt(name, type_info.type, value);
|
207
|
+
switch (type_info.type) {
|
208
|
+
case UPB_TYPE_INT32:
|
209
|
+
ret.int32_val = NUM2INT(value);
|
210
|
+
break;
|
211
|
+
case UPB_TYPE_INT64:
|
212
|
+
ret.int64_val = NUM2LL(value);
|
213
|
+
break;
|
214
|
+
case UPB_TYPE_UINT32:
|
215
|
+
ret.uint32_val = NUM2UINT(value);
|
216
|
+
break;
|
217
|
+
case UPB_TYPE_UINT64:
|
218
|
+
ret.uint64_val = NUM2ULL(value);
|
219
|
+
break;
|
220
|
+
default:
|
221
|
+
break;
|
222
|
+
}
|
223
|
+
break;
|
224
|
+
default:
|
225
|
+
break;
|
226
|
+
}
|
227
|
+
|
228
|
+
return ret;
|
229
|
+
}
|
230
|
+
|
231
|
+
VALUE Convert_UpbToRuby(upb_msgval upb_val, TypeInfo type_info, VALUE arena) {
|
232
|
+
switch (type_info.type) {
|
233
|
+
case UPB_TYPE_FLOAT:
|
234
|
+
return DBL2NUM(upb_val.float_val);
|
235
|
+
case UPB_TYPE_DOUBLE:
|
236
|
+
return DBL2NUM(upb_val.double_val);
|
237
|
+
case UPB_TYPE_BOOL:
|
238
|
+
return upb_val.bool_val ? Qtrue : Qfalse;
|
239
|
+
case UPB_TYPE_INT32:
|
240
|
+
return INT2NUM(upb_val.int32_val);
|
241
|
+
case UPB_TYPE_INT64:
|
242
|
+
return LL2NUM(upb_val.int64_val);
|
243
|
+
case UPB_TYPE_UINT32:
|
244
|
+
return UINT2NUM(upb_val.uint32_val);
|
245
|
+
case UPB_TYPE_UINT64:
|
246
|
+
return ULL2NUM(upb_val.int64_val);
|
247
|
+
case UPB_TYPE_ENUM: {
|
248
|
+
const char* name =
|
249
|
+
upb_enumdef_iton(type_info.def.enumdef, upb_val.int32_val);
|
250
|
+
if (name) {
|
251
|
+
return ID2SYM(rb_intern(name));
|
252
|
+
} else {
|
253
|
+
return INT2NUM(upb_val.int32_val);
|
254
|
+
}
|
255
|
+
}
|
256
|
+
case UPB_TYPE_STRING: {
|
257
|
+
VALUE str_rb = rb_str_new(upb_val.str_val.data, upb_val.str_val.size);
|
258
|
+
rb_enc_associate(str_rb, rb_utf8_encoding());
|
259
|
+
rb_obj_freeze(str_rb);
|
260
|
+
return str_rb;
|
261
|
+
}
|
262
|
+
case UPB_TYPE_BYTES: {
|
263
|
+
VALUE str_rb = rb_str_new(upb_val.str_val.data, upb_val.str_val.size);
|
264
|
+
rb_enc_associate(str_rb, rb_ascii8bit_encoding());
|
265
|
+
rb_obj_freeze(str_rb);
|
266
|
+
return str_rb;
|
267
|
+
}
|
268
|
+
case UPB_TYPE_MESSAGE:
|
269
|
+
return Message_GetRubyWrapper((upb_msg*)upb_val.msg_val,
|
270
|
+
type_info.def.msgdef, arena);
|
271
|
+
default:
|
272
|
+
rb_raise(rb_eRuntimeError, "Convert_UpbToRuby(): Unexpected type %d",
|
273
|
+
(int)type_info.type);
|
274
|
+
}
|
275
|
+
}
|
276
|
+
|
277
|
+
upb_msgval Msgval_DeepCopy(upb_msgval msgval, TypeInfo type_info,
|
278
|
+
upb_arena* arena) {
|
279
|
+
upb_msgval new_msgval;
|
280
|
+
|
281
|
+
switch (type_info.type) {
|
282
|
+
default:
|
283
|
+
memcpy(&new_msgval, &msgval, sizeof(msgval));
|
284
|
+
break;
|
285
|
+
case UPB_TYPE_STRING:
|
286
|
+
case UPB_TYPE_BYTES: {
|
287
|
+
size_t n = msgval.str_val.size;
|
288
|
+
char *mem = upb_arena_malloc(arena, n);
|
289
|
+
new_msgval.str_val.data = mem;
|
290
|
+
new_msgval.str_val.size = n;
|
291
|
+
memcpy(mem, msgval.str_val.data, n);
|
292
|
+
break;
|
293
|
+
}
|
294
|
+
case UPB_TYPE_MESSAGE:
|
295
|
+
new_msgval.msg_val =
|
296
|
+
Message_deep_copy(msgval.msg_val, type_info.def.msgdef, arena);
|
297
|
+
break;
|
298
|
+
}
|
299
|
+
|
300
|
+
return new_msgval;
|
301
|
+
}
|
302
|
+
|
303
|
+
bool Msgval_IsEqual(upb_msgval val1, upb_msgval val2, TypeInfo type_info) {
|
304
|
+
switch (type_info.type) {
|
305
|
+
case UPB_TYPE_BOOL:
|
306
|
+
return memcmp(&val1, &val2, 1) == 0;
|
307
|
+
case UPB_TYPE_FLOAT:
|
308
|
+
case UPB_TYPE_INT32:
|
309
|
+
case UPB_TYPE_UINT32:
|
310
|
+
case UPB_TYPE_ENUM:
|
311
|
+
return memcmp(&val1, &val2, 4) == 0;
|
312
|
+
case UPB_TYPE_DOUBLE:
|
313
|
+
case UPB_TYPE_INT64:
|
314
|
+
case UPB_TYPE_UINT64:
|
315
|
+
return memcmp(&val1, &val2, 8) == 0;
|
316
|
+
case UPB_TYPE_STRING:
|
317
|
+
case UPB_TYPE_BYTES:
|
318
|
+
return val1.str_val.size != val2.str_val.size ||
|
319
|
+
memcmp(val1.str_val.data, val2.str_val.data,
|
320
|
+
val1.str_val.size) == 0;
|
321
|
+
case UPB_TYPE_MESSAGE:
|
322
|
+
return Message_Equal(val1.msg_val, val2.msg_val, type_info.def.msgdef);
|
323
|
+
default:
|
324
|
+
rb_raise(rb_eRuntimeError, "Internal error, unexpected type");
|
325
|
+
}
|
326
|
+
}
|
327
|
+
|
328
|
+
uint64_t Msgval_GetHash(upb_msgval val, TypeInfo type_info, uint64_t seed) {
|
329
|
+
switch (type_info.type) {
|
330
|
+
case UPB_TYPE_BOOL:
|
331
|
+
return wyhash(&val, 1, seed, _wyp);
|
332
|
+
case UPB_TYPE_FLOAT:
|
333
|
+
case UPB_TYPE_INT32:
|
334
|
+
case UPB_TYPE_UINT32:
|
335
|
+
case UPB_TYPE_ENUM:
|
336
|
+
return wyhash(&val, 4, seed, _wyp);
|
337
|
+
case UPB_TYPE_DOUBLE:
|
338
|
+
case UPB_TYPE_INT64:
|
339
|
+
case UPB_TYPE_UINT64:
|
340
|
+
return wyhash(&val, 8, seed, _wyp);
|
341
|
+
case UPB_TYPE_STRING:
|
342
|
+
case UPB_TYPE_BYTES:
|
343
|
+
return wyhash(val.str_val.data, val.str_val.size, seed, _wyp);
|
344
|
+
case UPB_TYPE_MESSAGE:
|
345
|
+
return Message_Hash(val.msg_val, type_info.def.msgdef, seed);
|
346
|
+
default:
|
347
|
+
rb_raise(rb_eRuntimeError, "Internal error, unexpected type");
|
348
|
+
}
|
349
|
+
}
|
@@ -0,0 +1,72 @@
|
|
1
|
+
// Protocol Buffers - Google's data interchange format
|
2
|
+
// Copyright 2008 Google Inc. All rights reserved.
|
3
|
+
// https://developers.google.com/protocol-buffers/
|
4
|
+
//
|
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.
|
30
|
+
|
31
|
+
#ifndef RUBY_PROTOBUF_CONVERT_H_
|
32
|
+
#define RUBY_PROTOBUF_CONVERT_H_
|
33
|
+
|
34
|
+
#include <ruby/ruby.h>
|
35
|
+
|
36
|
+
#include "protobuf.h"
|
37
|
+
#include "ruby-upb.h"
|
38
|
+
|
39
|
+
// Converts |ruby_val| to a upb_msgval according to |type_info|.
|
40
|
+
//
|
41
|
+
// The |arena| parameter indicates the lifetime of the container where this
|
42
|
+
// value will be assigned. It is used as follows:
|
43
|
+
// - If type is string or bytes, the string data will be copied into |arena|.
|
44
|
+
// - If type is message, and we need to auto-construct a message due to implicit
|
45
|
+
// conversions (eg. Time -> Google::Protobuf::Timestamp), the new message
|
46
|
+
// will be created in |arena|.
|
47
|
+
// - If type is message and the Ruby value is a message instance, we will fuse
|
48
|
+
// the message's arena into |arena|, to ensure that this message outlives the
|
49
|
+
// container.
|
50
|
+
upb_msgval Convert_RubyToUpb(VALUE ruby_val, const char *name,
|
51
|
+
TypeInfo type_info, upb_arena *arena);
|
52
|
+
|
53
|
+
// Converts |upb_val| to a Ruby VALUE according to |type_info|. This may involve
|
54
|
+
// creating a Ruby wrapper object.
|
55
|
+
//
|
56
|
+
// The |arena| parameter indicates the arena that owns the lifetime of
|
57
|
+
// |upb_val|. Any Ruby wrapper object that is created will reference |arena|
|
58
|
+
// and ensure it outlives the wrapper.
|
59
|
+
VALUE Convert_UpbToRuby(upb_msgval upb_val, TypeInfo type_info, VALUE arena);
|
60
|
+
|
61
|
+
// Creates a deep copy of |msgval| in |arena|.
|
62
|
+
upb_msgval Msgval_DeepCopy(upb_msgval msgval, TypeInfo type_info,
|
63
|
+
upb_arena *arena);
|
64
|
+
|
65
|
+
// Returns true if |val1| and |val2| are equal. Their type is given by
|
66
|
+
// |type_info|.
|
67
|
+
bool Msgval_IsEqual(upb_msgval val1, upb_msgval val2, TypeInfo type_info);
|
68
|
+
|
69
|
+
// Returns a hash value for the given upb_msgval.
|
70
|
+
uint64_t Msgval_GetHash(upb_msgval val, TypeInfo type_info, uint64_t seed);
|
71
|
+
|
72
|
+
#endif // RUBY_PROTOBUF_CONVERT_H_
|
@@ -30,8 +30,35 @@
|
|
30
30
|
|
31
31
|
#include <ctype.h>
|
32
32
|
#include <errno.h>
|
33
|
+
#include <ruby/version.h>
|
34
|
+
|
35
|
+
#include "convert.h"
|
36
|
+
#include "message.h"
|
33
37
|
#include "protobuf.h"
|
34
38
|
|
39
|
+
static VALUE Builder_build(VALUE _self);
|
40
|
+
|
41
|
+
static VALUE cMessageBuilderContext;
|
42
|
+
static VALUE cOneofBuilderContext;
|
43
|
+
static VALUE cEnumBuilderContext;
|
44
|
+
static VALUE cBuilder;
|
45
|
+
|
46
|
+
// -----------------------------------------------------------------------------
|
47
|
+
// Global map from upb {msg,enum}defs to wrapper Descriptor/EnumDescriptor
|
48
|
+
// instances.
|
49
|
+
// -----------------------------------------------------------------------------
|
50
|
+
|
51
|
+
static VALUE get_msgdef_obj(VALUE descriptor_pool, const upb_msgdef* def);
|
52
|
+
static VALUE get_enumdef_obj(VALUE descriptor_pool, const upb_enumdef* def);
|
53
|
+
static VALUE get_fielddef_obj(VALUE descriptor_pool, const upb_fielddef* def);
|
54
|
+
static VALUE get_filedef_obj(VALUE descriptor_pool, const upb_filedef* def);
|
55
|
+
static VALUE get_oneofdef_obj(VALUE descriptor_pool, const upb_oneofdef* def);
|
56
|
+
|
57
|
+
// A distinct object that is not accessible from Ruby. We use this as a
|
58
|
+
// constructor argument to enforce that certain objects cannot be created from
|
59
|
+
// Ruby.
|
60
|
+
VALUE c_only_cookie = Qnil;
|
61
|
+
|
35
62
|
// -----------------------------------------------------------------------------
|
36
63
|
// Common utilities.
|
37
64
|
// -----------------------------------------------------------------------------
|
@@ -48,6 +75,10 @@ static VALUE rb_str_maybe_null(const char* s) {
|
|
48
75
|
return rb_str_new2(s);
|
49
76
|
}
|
50
77
|
|
78
|
+
// -----------------------------------------------------------------------------
|
79
|
+
// Backward compatibility code.
|
80
|
+
// -----------------------------------------------------------------------------
|
81
|
+
|
51
82
|
static void rewrite_enum_default(const upb_symtab* symtab,
|
52
83
|
google_protobuf_FileDescriptorProto* file,
|
53
84
|
google_protobuf_FieldDescriptorProto* field) {
|
@@ -205,221 +236,70 @@ static void rewrite_nesting(VALUE msg_ent, google_protobuf_DescriptorProto* msg,
|
|
205
236
|
}
|
206
237
|
}
|
207
238
|
|
208
|
-
/* We have to do some relatively complicated logic here for backward
|
209
|
-
* compatibility.
|
210
|
-
*
|
211
|
-
* In descriptor.proto, messages are nested inside other messages if that is
|
212
|
-
* what the original .proto file looks like. For example, suppose we have this
|
213
|
-
* foo.proto:
|
214
|
-
*
|
215
|
-
* package foo;
|
216
|
-
* message Bar {
|
217
|
-
* message Baz {}
|
218
|
-
* }
|
219
|
-
*
|
220
|
-
* The descriptor for this must look like this:
|
221
|
-
*
|
222
|
-
* file {
|
223
|
-
* name: "test.proto"
|
224
|
-
* package: "foo"
|
225
|
-
* message_type {
|
226
|
-
* name: "Bar"
|
227
|
-
* nested_type {
|
228
|
-
* name: "Baz"
|
229
|
-
* }
|
230
|
-
* }
|
231
|
-
* }
|
232
|
-
*
|
233
|
-
* However, the Ruby generated code has always generated messages in a flat,
|
234
|
-
* non-nested way:
|
235
|
-
*
|
236
|
-
* Google::Protobuf::DescriptorPool.generated_pool.build do
|
237
|
-
* add_message "foo.Bar" do
|
238
|
-
* end
|
239
|
-
* add_message "foo.Bar.Baz" do
|
240
|
-
* end
|
241
|
-
* end
|
242
|
-
*
|
243
|
-
* Here we need to do a translation where we turn this generated code into the
|
244
|
-
* above descriptor. We need to infer that "foo" is the package name, and not
|
245
|
-
* a message itself.
|
246
|
-
*
|
247
|
-
* We delegate to Ruby to compute the transformation, for more concice and
|
248
|
-
* readable code than we can do in C */
|
249
|
-
static void rewrite_names(VALUE _file_builder,
|
250
|
-
google_protobuf_FileDescriptorProto* file_proto) {
|
251
|
-
FileBuilderContext* file_builder = ruby_to_FileBuilderContext(_file_builder);
|
252
|
-
upb_arena *arena = file_builder->arena;
|
253
|
-
// Build params (package, msg_names, enum_names).
|
254
|
-
VALUE package = Qnil;
|
255
|
-
VALUE msg_names = rb_ary_new();
|
256
|
-
VALUE enum_names = rb_ary_new();
|
257
|
-
size_t msg_count, enum_count, i;
|
258
|
-
VALUE new_package, nesting, msg_ents, enum_ents;
|
259
|
-
google_protobuf_DescriptorProto** msgs;
|
260
|
-
google_protobuf_EnumDescriptorProto** enums;
|
261
|
-
|
262
|
-
if (google_protobuf_FileDescriptorProto_has_package(file_proto)) {
|
263
|
-
upb_strview package_str =
|
264
|
-
google_protobuf_FileDescriptorProto_package(file_proto);
|
265
|
-
package = rb_str_new(package_str.data, package_str.size);
|
266
|
-
}
|
267
|
-
|
268
|
-
msgs = google_protobuf_FileDescriptorProto_mutable_message_type(file_proto,
|
269
|
-
&msg_count);
|
270
|
-
for (i = 0; i < msg_count; i++) {
|
271
|
-
upb_strview name = google_protobuf_DescriptorProto_name(msgs[i]);
|
272
|
-
rb_ary_push(msg_names, rb_str_new(name.data, name.size));
|
273
|
-
}
|
274
|
-
|
275
|
-
enums = google_protobuf_FileDescriptorProto_mutable_enum_type(file_proto,
|
276
|
-
&enum_count);
|
277
|
-
for (i = 0; i < enum_count; i++) {
|
278
|
-
upb_strview name = google_protobuf_EnumDescriptorProto_name(enums[i]);
|
279
|
-
rb_ary_push(enum_names, rb_str_new(name.data, name.size));
|
280
|
-
}
|
281
|
-
|
282
|
-
{
|
283
|
-
// Call Ruby code to calculate package name and nesting.
|
284
|
-
VALUE args[3] = { package, msg_names, enum_names };
|
285
|
-
VALUE internal = rb_eval_string("Google::Protobuf::Internal");
|
286
|
-
VALUE ret = rb_funcallv(internal, rb_intern("fixup_descriptor"), 3, args);
|
287
|
-
|
288
|
-
new_package = rb_ary_entry(ret, 0);
|
289
|
-
nesting = rb_ary_entry(ret, 1);
|
290
|
-
}
|
291
|
-
|
292
|
-
// Rewrite package and names.
|
293
|
-
if (new_package != Qnil) {
|
294
|
-
upb_strview new_package_str =
|
295
|
-
FileBuilderContext_strdup(_file_builder, new_package);
|
296
|
-
google_protobuf_FileDescriptorProto_set_package(file_proto,
|
297
|
-
new_package_str);
|
298
|
-
}
|
299
|
-
|
300
|
-
for (i = 0; i < msg_count; i++) {
|
301
|
-
upb_strview name = google_protobuf_DescriptorProto_name(msgs[i]);
|
302
|
-
remove_path(&name);
|
303
|
-
google_protobuf_DescriptorProto_set_name(msgs[i], name);
|
304
|
-
}
|
305
|
-
|
306
|
-
for (i = 0; i < enum_count; i++) {
|
307
|
-
upb_strview name = google_protobuf_EnumDescriptorProto_name(enums[i]);
|
308
|
-
remove_path(&name);
|
309
|
-
google_protobuf_EnumDescriptorProto_set_name(enums[i], name);
|
310
|
-
}
|
311
|
-
|
312
|
-
// Rewrite nesting.
|
313
|
-
msg_ents = rb_hash_aref(nesting, ID2SYM(rb_intern("msgs")));
|
314
|
-
enum_ents = rb_hash_aref(nesting, ID2SYM(rb_intern("enums")));
|
315
|
-
|
316
|
-
Check_Type(msg_ents, T_ARRAY);
|
317
|
-
Check_Type(enum_ents, T_ARRAY);
|
318
|
-
|
319
|
-
for (i = 0; i < (size_t)RARRAY_LEN(msg_ents); i++) {
|
320
|
-
VALUE msg_ent = rb_ary_entry(msg_ents, i);
|
321
|
-
VALUE pos = rb_hash_aref(msg_ent, ID2SYM(rb_intern("pos")));
|
322
|
-
msgs[i] = msgs[NUM2INT(pos)];
|
323
|
-
rewrite_nesting(msg_ent, msgs[i], msgs, enums, arena);
|
324
|
-
}
|
325
|
-
|
326
|
-
for (i = 0; i < (size_t)RARRAY_LEN(enum_ents); i++) {
|
327
|
-
VALUE enum_pos = rb_ary_entry(enum_ents, i);
|
328
|
-
enums[i] = enums[NUM2INT(enum_pos)];
|
329
|
-
}
|
330
|
-
|
331
|
-
google_protobuf_FileDescriptorProto_resize_message_type(
|
332
|
-
file_proto, RARRAY_LEN(msg_ents), arena);
|
333
|
-
google_protobuf_FileDescriptorProto_resize_enum_type(
|
334
|
-
file_proto, RARRAY_LEN(enum_ents), arena);
|
335
|
-
}
|
336
|
-
|
337
239
|
// -----------------------------------------------------------------------------
|
338
240
|
// DescriptorPool.
|
339
241
|
// -----------------------------------------------------------------------------
|
340
242
|
|
341
|
-
|
342
|
-
|
343
|
-
|
344
|
-
|
345
|
-
|
346
|
-
|
347
|
-
name* ruby_to_ ## name(VALUE val) { \
|
348
|
-
name* ret; \
|
349
|
-
TypedData_Get_Struct(val, name, &_ ## name ## _type, ret); \
|
350
|
-
return ret; \
|
351
|
-
} \
|
352
|
-
|
353
|
-
#define DEFINE_SELF(type, var, rb_var) \
|
354
|
-
type* var = ruby_to_ ## type(rb_var)
|
243
|
+
typedef struct {
|
244
|
+
VALUE def_to_descriptor; // Hash table of def* -> Ruby descriptor.
|
245
|
+
upb_symtab* symtab;
|
246
|
+
} DescriptorPool;
|
247
|
+
|
248
|
+
VALUE cDescriptorPool = Qnil;
|
355
249
|
|
356
250
|
// Global singleton DescriptorPool. The user is free to create others, but this
|
357
251
|
// is used by generated code.
|
358
252
|
VALUE generated_pool = Qnil;
|
359
253
|
|
360
|
-
|
361
|
-
|
362
|
-
void DescriptorPool_mark(void* _self) {
|
254
|
+
static void DescriptorPool_mark(void* _self) {
|
363
255
|
DescriptorPool* self = _self;
|
364
256
|
rb_gc_mark(self->def_to_descriptor);
|
365
257
|
}
|
366
258
|
|
367
|
-
void DescriptorPool_free(void* _self) {
|
259
|
+
static void DescriptorPool_free(void* _self) {
|
368
260
|
DescriptorPool* self = _self;
|
369
|
-
|
370
261
|
upb_symtab_free(self->symtab);
|
371
|
-
upb_handlercache_free(self->fill_handler_cache);
|
372
|
-
upb_handlercache_free(self->pb_serialize_handler_cache);
|
373
|
-
upb_handlercache_free(self->json_serialize_handler_cache);
|
374
|
-
upb_handlercache_free(self->json_serialize_handler_preserve_cache);
|
375
|
-
upb_pbcodecache_free(self->fill_method_cache);
|
376
|
-
upb_json_codecache_free(self->json_fill_method_cache);
|
377
|
-
|
378
262
|
xfree(self);
|
379
263
|
}
|
380
264
|
|
265
|
+
static const rb_data_type_t DescriptorPool_type = {
|
266
|
+
"Google::Protobuf::DescriptorPool",
|
267
|
+
{DescriptorPool_mark, DescriptorPool_free, NULL},
|
268
|
+
.flags = RUBY_TYPED_FREE_IMMEDIATELY,
|
269
|
+
};
|
270
|
+
|
271
|
+
static DescriptorPool* ruby_to_DescriptorPool(VALUE val) {
|
272
|
+
DescriptorPool* ret;
|
273
|
+
TypedData_Get_Struct(val, DescriptorPool, &DescriptorPool_type, ret);
|
274
|
+
return ret;
|
275
|
+
}
|
276
|
+
|
277
|
+
// Exposed to other modules in defs.h.
|
278
|
+
const upb_symtab *DescriptorPool_GetSymtab(VALUE desc_pool_rb) {
|
279
|
+
DescriptorPool *pool = ruby_to_DescriptorPool(desc_pool_rb);
|
280
|
+
return pool->symtab;
|
281
|
+
}
|
282
|
+
|
381
283
|
/*
|
382
284
|
* call-seq:
|
383
285
|
* DescriptorPool.new => pool
|
384
286
|
*
|
385
287
|
* Creates a new, empty, descriptor pool.
|
386
288
|
*/
|
387
|
-
VALUE DescriptorPool_alloc(VALUE klass) {
|
289
|
+
static VALUE DescriptorPool_alloc(VALUE klass) {
|
388
290
|
DescriptorPool* self = ALLOC(DescriptorPool);
|
389
291
|
VALUE ret;
|
390
292
|
|
391
293
|
self->def_to_descriptor = Qnil;
|
392
|
-
ret = TypedData_Wrap_Struct(klass, &
|
294
|
+
ret = TypedData_Wrap_Struct(klass, &DescriptorPool_type, self);
|
393
295
|
|
394
296
|
self->def_to_descriptor = rb_hash_new();
|
395
297
|
self->symtab = upb_symtab_new();
|
396
|
-
self->
|
397
|
-
upb_handlercache_new(add_handlers_for_message, (void*)ret);
|
398
|
-
self->pb_serialize_handler_cache = upb_pb_encoder_newcache();
|
399
|
-
self->json_serialize_handler_cache = upb_json_printer_newcache(false);
|
400
|
-
self->json_serialize_handler_preserve_cache =
|
401
|
-
upb_json_printer_newcache(true);
|
402
|
-
self->fill_method_cache = upb_pbcodecache_new(self->fill_handler_cache);
|
403
|
-
self->json_fill_method_cache = upb_json_codecache_new();
|
298
|
+
ObjectCache_Add(self->symtab, ret, _upb_symtab_arena(self->symtab));
|
404
299
|
|
405
300
|
return ret;
|
406
301
|
}
|
407
302
|
|
408
|
-
void DescriptorPool_register(VALUE module) {
|
409
|
-
VALUE klass = rb_define_class_under(
|
410
|
-
module, "DescriptorPool", rb_cObject);
|
411
|
-
rb_define_alloc_func(klass, DescriptorPool_alloc);
|
412
|
-
rb_define_method(klass, "build", DescriptorPool_build, -1);
|
413
|
-
rb_define_method(klass, "lookup", DescriptorPool_lookup, 1);
|
414
|
-
rb_define_singleton_method(klass, "generated_pool",
|
415
|
-
DescriptorPool_generated_pool, 0);
|
416
|
-
rb_gc_register_address(&cDescriptorPool);
|
417
|
-
cDescriptorPool = klass;
|
418
|
-
|
419
|
-
rb_gc_register_address(&generated_pool);
|
420
|
-
generated_pool = rb_class_new_instance(0, NULL, klass);
|
421
|
-
}
|
422
|
-
|
423
303
|
/*
|
424
304
|
* call-seq:
|
425
305
|
* DescriptorPool.build(&block)
|
@@ -430,7 +310,7 @@ void DescriptorPool_register(VALUE module) {
|
|
430
310
|
* Builder#add_enum within the block as appropriate. This is the recommended,
|
431
311
|
* idiomatic way to define new message and enum types.
|
432
312
|
*/
|
433
|
-
VALUE DescriptorPool_build(int argc, VALUE* argv, VALUE _self) {
|
313
|
+
static VALUE DescriptorPool_build(int argc, VALUE* argv, VALUE _self) {
|
434
314
|
VALUE ctx = rb_class_new_instance(1, &_self, cBuilder);
|
435
315
|
VALUE block = rb_block_proc();
|
436
316
|
rb_funcall_with_block(ctx, rb_intern("instance_eval"), 0, NULL, block);
|
@@ -445,8 +325,8 @@ VALUE DescriptorPool_build(int argc, VALUE* argv, VALUE _self) {
|
|
445
325
|
* Finds a Descriptor or EnumDescriptor by name and returns it, or nil if none
|
446
326
|
* exists with the given name.
|
447
327
|
*/
|
448
|
-
VALUE DescriptorPool_lookup(VALUE _self, VALUE name) {
|
449
|
-
|
328
|
+
static VALUE DescriptorPool_lookup(VALUE _self, VALUE name) {
|
329
|
+
DescriptorPool* self = ruby_to_DescriptorPool(_self);
|
450
330
|
const char* name_str = get_str(name);
|
451
331
|
const upb_msgdef* msgdef;
|
452
332
|
const upb_enumdef* enumdef;
|
@@ -473,31 +353,53 @@ VALUE DescriptorPool_lookup(VALUE _self, VALUE name) {
|
|
473
353
|
* register types in this pool for convenience so that they do not have to hold
|
474
354
|
* a reference to a private pool instance.
|
475
355
|
*/
|
476
|
-
VALUE DescriptorPool_generated_pool(VALUE _self) {
|
356
|
+
static VALUE DescriptorPool_generated_pool(VALUE _self) {
|
477
357
|
return generated_pool;
|
478
358
|
}
|
479
359
|
|
360
|
+
static void DescriptorPool_register(VALUE module) {
|
361
|
+
VALUE klass = rb_define_class_under(
|
362
|
+
module, "DescriptorPool", rb_cObject);
|
363
|
+
rb_define_alloc_func(klass, DescriptorPool_alloc);
|
364
|
+
rb_define_method(klass, "build", DescriptorPool_build, -1);
|
365
|
+
rb_define_method(klass, "lookup", DescriptorPool_lookup, 1);
|
366
|
+
rb_define_singleton_method(klass, "generated_pool",
|
367
|
+
DescriptorPool_generated_pool, 0);
|
368
|
+
rb_gc_register_address(&cDescriptorPool);
|
369
|
+
cDescriptorPool = klass;
|
370
|
+
|
371
|
+
rb_gc_register_address(&generated_pool);
|
372
|
+
generated_pool = rb_class_new_instance(0, NULL, klass);
|
373
|
+
}
|
374
|
+
|
480
375
|
// -----------------------------------------------------------------------------
|
481
376
|
// Descriptor.
|
482
377
|
// -----------------------------------------------------------------------------
|
483
378
|
|
484
|
-
|
379
|
+
typedef struct {
|
380
|
+
const upb_msgdef* msgdef;
|
381
|
+
VALUE klass;
|
382
|
+
VALUE descriptor_pool;
|
383
|
+
} Descriptor;
|
384
|
+
|
385
|
+
VALUE cDescriptor = Qnil;
|
485
386
|
|
486
|
-
void Descriptor_mark(void* _self) {
|
387
|
+
static void Descriptor_mark(void* _self) {
|
487
388
|
Descriptor* self = _self;
|
488
389
|
rb_gc_mark(self->klass);
|
489
390
|
rb_gc_mark(self->descriptor_pool);
|
490
|
-
if (self->layout && self->layout->empty_template) {
|
491
|
-
layout_mark(self->layout, self->layout->empty_template);
|
492
|
-
}
|
493
391
|
}
|
494
392
|
|
495
|
-
|
496
|
-
Descriptor
|
497
|
-
|
498
|
-
|
499
|
-
|
500
|
-
|
393
|
+
static const rb_data_type_t Descriptor_type = {
|
394
|
+
"Google::Protobuf::Descriptor",
|
395
|
+
{Descriptor_mark, RUBY_DEFAULT_FREE, NULL},
|
396
|
+
.flags = RUBY_TYPED_FREE_IMMEDIATELY,
|
397
|
+
};
|
398
|
+
|
399
|
+
static Descriptor* ruby_to_Descriptor(VALUE val) {
|
400
|
+
Descriptor* ret;
|
401
|
+
TypedData_Get_Struct(val, Descriptor, &Descriptor_type, ret);
|
402
|
+
return ret;
|
501
403
|
}
|
502
404
|
|
503
405
|
/*
|
@@ -509,42 +411,24 @@ void Descriptor_free(void* _self) {
|
|
509
411
|
* it is added to a pool, after which it becomes immutable (as part of a
|
510
412
|
* finalization process).
|
511
413
|
*/
|
512
|
-
VALUE Descriptor_alloc(VALUE klass) {
|
414
|
+
static VALUE Descriptor_alloc(VALUE klass) {
|
513
415
|
Descriptor* self = ALLOC(Descriptor);
|
514
|
-
VALUE ret = TypedData_Wrap_Struct(klass, &
|
416
|
+
VALUE ret = TypedData_Wrap_Struct(klass, &Descriptor_type, self);
|
515
417
|
self->msgdef = NULL;
|
516
418
|
self->klass = Qnil;
|
517
419
|
self->descriptor_pool = Qnil;
|
518
|
-
self->layout = NULL;
|
519
420
|
return ret;
|
520
421
|
}
|
521
422
|
|
522
|
-
void Descriptor_register(VALUE module) {
|
523
|
-
VALUE klass = rb_define_class_under(
|
524
|
-
module, "Descriptor", rb_cObject);
|
525
|
-
rb_define_alloc_func(klass, Descriptor_alloc);
|
526
|
-
rb_define_method(klass, "initialize", Descriptor_initialize, 3);
|
527
|
-
rb_define_method(klass, "each", Descriptor_each, 0);
|
528
|
-
rb_define_method(klass, "lookup", Descriptor_lookup, 1);
|
529
|
-
rb_define_method(klass, "each_oneof", Descriptor_each_oneof, 0);
|
530
|
-
rb_define_method(klass, "lookup_oneof", Descriptor_lookup_oneof, 1);
|
531
|
-
rb_define_method(klass, "msgclass", Descriptor_msgclass, 0);
|
532
|
-
rb_define_method(klass, "name", Descriptor_name, 0);
|
533
|
-
rb_define_method(klass, "file_descriptor", Descriptor_file_descriptor, 0);
|
534
|
-
rb_include_module(klass, rb_mEnumerable);
|
535
|
-
rb_gc_register_address(&cDescriptor);
|
536
|
-
cDescriptor = klass;
|
537
|
-
}
|
538
|
-
|
539
423
|
/*
|
540
424
|
* call-seq:
|
541
425
|
* Descriptor.new(c_only_cookie, ptr) => Descriptor
|
542
426
|
*
|
543
427
|
* Creates a descriptor wrapper object. May only be called from C.
|
544
428
|
*/
|
545
|
-
VALUE Descriptor_initialize(VALUE _self, VALUE cookie,
|
546
|
-
|
547
|
-
|
429
|
+
static VALUE Descriptor_initialize(VALUE _self, VALUE cookie,
|
430
|
+
VALUE descriptor_pool, VALUE ptr) {
|
431
|
+
Descriptor* self = ruby_to_Descriptor(_self);
|
548
432
|
|
549
433
|
if (cookie != c_only_cookie) {
|
550
434
|
rb_raise(rb_eRuntimeError,
|
@@ -563,8 +447,8 @@ VALUE Descriptor_initialize(VALUE _self, VALUE cookie,
|
|
563
447
|
*
|
564
448
|
* Returns the FileDescriptor object this message belongs to.
|
565
449
|
*/
|
566
|
-
VALUE Descriptor_file_descriptor(VALUE _self) {
|
567
|
-
|
450
|
+
static VALUE Descriptor_file_descriptor(VALUE _self) {
|
451
|
+
Descriptor* self = ruby_to_Descriptor(_self);
|
568
452
|
return get_filedef_obj(self->descriptor_pool, upb_msgdef_file(self->msgdef));
|
569
453
|
}
|
570
454
|
|
@@ -575,8 +459,8 @@ VALUE Descriptor_file_descriptor(VALUE _self) {
|
|
575
459
|
* Returns the name of this message type as a fully-qualified string (e.g.,
|
576
460
|
* My.Package.MessageType).
|
577
461
|
*/
|
578
|
-
VALUE Descriptor_name(VALUE _self) {
|
579
|
-
|
462
|
+
static VALUE Descriptor_name(VALUE _self) {
|
463
|
+
Descriptor* self = ruby_to_Descriptor(_self);
|
580
464
|
return rb_str_maybe_null(upb_msgdef_fullname(self->msgdef));
|
581
465
|
}
|
582
466
|
|
@@ -586,8 +470,8 @@ VALUE Descriptor_name(VALUE _self) {
|
|
586
470
|
*
|
587
471
|
* Iterates over fields in this message type, yielding to the block on each one.
|
588
472
|
*/
|
589
|
-
VALUE Descriptor_each(VALUE _self) {
|
590
|
-
|
473
|
+
static VALUE Descriptor_each(VALUE _self) {
|
474
|
+
Descriptor* self = ruby_to_Descriptor(_self);
|
591
475
|
|
592
476
|
upb_msg_field_iter it;
|
593
477
|
for (upb_msg_field_begin(&it, self->msgdef);
|
@@ -607,8 +491,8 @@ VALUE Descriptor_each(VALUE _self) {
|
|
607
491
|
* Returns the field descriptor for the field with the given name, if present,
|
608
492
|
* or nil if none.
|
609
493
|
*/
|
610
|
-
VALUE Descriptor_lookup(VALUE _self, VALUE name) {
|
611
|
-
|
494
|
+
static VALUE Descriptor_lookup(VALUE _self, VALUE name) {
|
495
|
+
Descriptor* self = ruby_to_Descriptor(_self);
|
612
496
|
const char* s = get_str(name);
|
613
497
|
const upb_fielddef* field = upb_msgdef_ntofz(self->msgdef, s);
|
614
498
|
if (field == NULL) {
|
@@ -624,8 +508,8 @@ VALUE Descriptor_lookup(VALUE _self, VALUE name) {
|
|
624
508
|
* Invokes the given block for each oneof in this message type, passing the
|
625
509
|
* corresponding OneofDescriptor.
|
626
510
|
*/
|
627
|
-
VALUE Descriptor_each_oneof(VALUE _self) {
|
628
|
-
|
511
|
+
static VALUE Descriptor_each_oneof(VALUE _self) {
|
512
|
+
Descriptor* self = ruby_to_Descriptor(_self);
|
629
513
|
|
630
514
|
upb_msg_oneof_iter it;
|
631
515
|
for (upb_msg_oneof_begin(&it, self->msgdef);
|
@@ -645,8 +529,8 @@ VALUE Descriptor_each_oneof(VALUE _self) {
|
|
645
529
|
* Returns the oneof descriptor for the oneof with the given name, if present,
|
646
530
|
* or nil if none.
|
647
531
|
*/
|
648
|
-
VALUE Descriptor_lookup_oneof(VALUE _self, VALUE name) {
|
649
|
-
|
532
|
+
static VALUE Descriptor_lookup_oneof(VALUE _self, VALUE name) {
|
533
|
+
Descriptor* self = ruby_to_Descriptor(_self);
|
650
534
|
const char* s = get_str(name);
|
651
535
|
const upb_oneofdef* oneof = upb_msgdef_ntooz(self->msgdef, s);
|
652
536
|
if (oneof == NULL) {
|
@@ -661,32 +545,62 @@ VALUE Descriptor_lookup_oneof(VALUE _self, VALUE name) {
|
|
661
545
|
*
|
662
546
|
* Returns the Ruby class created for this message type.
|
663
547
|
*/
|
664
|
-
VALUE Descriptor_msgclass(VALUE _self) {
|
665
|
-
|
548
|
+
static VALUE Descriptor_msgclass(VALUE _self) {
|
549
|
+
Descriptor* self = ruby_to_Descriptor(_self);
|
666
550
|
if (self->klass == Qnil) {
|
667
551
|
self->klass = build_class_from_descriptor(_self);
|
668
552
|
}
|
669
553
|
return self->klass;
|
670
554
|
}
|
671
555
|
|
556
|
+
static void Descriptor_register(VALUE module) {
|
557
|
+
VALUE klass = rb_define_class_under(
|
558
|
+
module, "Descriptor", rb_cObject);
|
559
|
+
rb_define_alloc_func(klass, Descriptor_alloc);
|
560
|
+
rb_define_method(klass, "initialize", Descriptor_initialize, 3);
|
561
|
+
rb_define_method(klass, "each", Descriptor_each, 0);
|
562
|
+
rb_define_method(klass, "lookup", Descriptor_lookup, 1);
|
563
|
+
rb_define_method(klass, "each_oneof", Descriptor_each_oneof, 0);
|
564
|
+
rb_define_method(klass, "lookup_oneof", Descriptor_lookup_oneof, 1);
|
565
|
+
rb_define_method(klass, "msgclass", Descriptor_msgclass, 0);
|
566
|
+
rb_define_method(klass, "name", Descriptor_name, 0);
|
567
|
+
rb_define_method(klass, "file_descriptor", Descriptor_file_descriptor, 0);
|
568
|
+
rb_include_module(klass, rb_mEnumerable);
|
569
|
+
rb_gc_register_address(&cDescriptor);
|
570
|
+
cDescriptor = klass;
|
571
|
+
}
|
572
|
+
|
672
573
|
// -----------------------------------------------------------------------------
|
673
574
|
// FileDescriptor.
|
674
575
|
// -----------------------------------------------------------------------------
|
675
576
|
|
676
|
-
|
577
|
+
typedef struct {
|
578
|
+
const upb_filedef* filedef;
|
579
|
+
VALUE descriptor_pool; // Owns the upb_filedef.
|
580
|
+
} FileDescriptor;
|
677
581
|
|
678
|
-
|
582
|
+
static VALUE cFileDescriptor = Qnil;
|
583
|
+
|
584
|
+
static void FileDescriptor_mark(void* _self) {
|
679
585
|
FileDescriptor* self = _self;
|
680
586
|
rb_gc_mark(self->descriptor_pool);
|
681
587
|
}
|
682
588
|
|
683
|
-
|
684
|
-
|
589
|
+
static const rb_data_type_t FileDescriptor_type = {
|
590
|
+
"Google::Protobuf::FileDescriptor",
|
591
|
+
{FileDescriptor_mark, RUBY_DEFAULT_FREE, NULL},
|
592
|
+
.flags = RUBY_TYPED_FREE_IMMEDIATELY,
|
593
|
+
};
|
594
|
+
|
595
|
+
static FileDescriptor* ruby_to_FileDescriptor(VALUE val) {
|
596
|
+
FileDescriptor* ret;
|
597
|
+
TypedData_Get_Struct(val, FileDescriptor, &FileDescriptor_type, ret);
|
598
|
+
return ret;
|
685
599
|
}
|
686
600
|
|
687
|
-
VALUE FileDescriptor_alloc(VALUE klass) {
|
601
|
+
static VALUE FileDescriptor_alloc(VALUE klass) {
|
688
602
|
FileDescriptor* self = ALLOC(FileDescriptor);
|
689
|
-
VALUE ret = TypedData_Wrap_Struct(klass, &
|
603
|
+
VALUE ret = TypedData_Wrap_Struct(klass, &FileDescriptor_type, self);
|
690
604
|
self->descriptor_pool = Qnil;
|
691
605
|
self->filedef = NULL;
|
692
606
|
return ret;
|
@@ -699,9 +613,9 @@ VALUE FileDescriptor_alloc(VALUE klass) {
|
|
699
613
|
* Returns a new file descriptor. The syntax must be set before it's passed
|
700
614
|
* to a builder.
|
701
615
|
*/
|
702
|
-
VALUE FileDescriptor_initialize(VALUE _self, VALUE cookie,
|
616
|
+
static VALUE FileDescriptor_initialize(VALUE _self, VALUE cookie,
|
703
617
|
VALUE descriptor_pool, VALUE ptr) {
|
704
|
-
|
618
|
+
FileDescriptor* self = ruby_to_FileDescriptor(_self);
|
705
619
|
|
706
620
|
if (cookie != c_only_cookie) {
|
707
621
|
rb_raise(rb_eRuntimeError,
|
@@ -714,25 +628,14 @@ VALUE FileDescriptor_initialize(VALUE _self, VALUE cookie,
|
|
714
628
|
return Qnil;
|
715
629
|
}
|
716
630
|
|
717
|
-
void FileDescriptor_register(VALUE module) {
|
718
|
-
VALUE klass = rb_define_class_under(
|
719
|
-
module, "FileDescriptor", rb_cObject);
|
720
|
-
rb_define_alloc_func(klass, FileDescriptor_alloc);
|
721
|
-
rb_define_method(klass, "initialize", FileDescriptor_initialize, 3);
|
722
|
-
rb_define_method(klass, "name", FileDescriptor_name, 0);
|
723
|
-
rb_define_method(klass, "syntax", FileDescriptor_syntax, 0);
|
724
|
-
rb_gc_register_address(&cFileDescriptor);
|
725
|
-
cFileDescriptor = klass;
|
726
|
-
}
|
727
|
-
|
728
631
|
/*
|
729
632
|
* call-seq:
|
730
633
|
* FileDescriptor.name => name
|
731
634
|
*
|
732
635
|
* Returns the name of the file.
|
733
636
|
*/
|
734
|
-
VALUE FileDescriptor_name(VALUE _self) {
|
735
|
-
|
637
|
+
static VALUE FileDescriptor_name(VALUE _self) {
|
638
|
+
FileDescriptor* self = ruby_to_FileDescriptor(_self);
|
736
639
|
const char* name = upb_filedef_name(self->filedef);
|
737
640
|
return name == NULL ? Qnil : rb_str_new2(name);
|
738
641
|
}
|
@@ -746,8 +649,8 @@ VALUE FileDescriptor_name(VALUE _self) {
|
|
746
649
|
* Valid syntax versions are:
|
747
650
|
* :proto2 or :proto3.
|
748
651
|
*/
|
749
|
-
VALUE FileDescriptor_syntax(VALUE _self) {
|
750
|
-
|
652
|
+
static VALUE FileDescriptor_syntax(VALUE _self) {
|
653
|
+
FileDescriptor* self = ruby_to_FileDescriptor(_self);
|
751
654
|
|
752
655
|
switch (upb_filedef_syntax(self->filedef)) {
|
753
656
|
case UPB_SYNTAX_PROTO3: return ID2SYM(rb_intern("proto3"));
|
@@ -756,19 +659,43 @@ VALUE FileDescriptor_syntax(VALUE _self) {
|
|
756
659
|
}
|
757
660
|
}
|
758
661
|
|
662
|
+
static void FileDescriptor_register(VALUE module) {
|
663
|
+
VALUE klass = rb_define_class_under(
|
664
|
+
module, "FileDescriptor", rb_cObject);
|
665
|
+
rb_define_alloc_func(klass, FileDescriptor_alloc);
|
666
|
+
rb_define_method(klass, "initialize", FileDescriptor_initialize, 3);
|
667
|
+
rb_define_method(klass, "name", FileDescriptor_name, 0);
|
668
|
+
rb_define_method(klass, "syntax", FileDescriptor_syntax, 0);
|
669
|
+
rb_gc_register_address(&cFileDescriptor);
|
670
|
+
cFileDescriptor = klass;
|
671
|
+
}
|
672
|
+
|
759
673
|
// -----------------------------------------------------------------------------
|
760
674
|
// FieldDescriptor.
|
761
675
|
// -----------------------------------------------------------------------------
|
762
676
|
|
763
|
-
|
677
|
+
typedef struct {
|
678
|
+
const upb_fielddef* fielddef;
|
679
|
+
VALUE descriptor_pool; // Owns the upb_fielddef.
|
680
|
+
} FieldDescriptor;
|
681
|
+
|
682
|
+
static VALUE cFieldDescriptor = Qnil;
|
764
683
|
|
765
|
-
void FieldDescriptor_mark(void* _self) {
|
684
|
+
static void FieldDescriptor_mark(void* _self) {
|
766
685
|
FieldDescriptor* self = _self;
|
767
686
|
rb_gc_mark(self->descriptor_pool);
|
768
687
|
}
|
769
688
|
|
770
|
-
|
771
|
-
|
689
|
+
static const rb_data_type_t FieldDescriptor_type = {
|
690
|
+
"Google::Protobuf::FieldDescriptor",
|
691
|
+
{FieldDescriptor_mark, RUBY_DEFAULT_FREE, NULL},
|
692
|
+
.flags = RUBY_TYPED_FREE_IMMEDIATELY,
|
693
|
+
};
|
694
|
+
|
695
|
+
static FieldDescriptor* ruby_to_FieldDescriptor(VALUE val) {
|
696
|
+
FieldDescriptor* ret;
|
697
|
+
TypedData_Get_Struct(val, FieldDescriptor, &FieldDescriptor_type, ret);
|
698
|
+
return ret;
|
772
699
|
}
|
773
700
|
|
774
701
|
/*
|
@@ -778,42 +705,22 @@ void FieldDescriptor_free(void* _self) {
|
|
778
705
|
* Returns a new field descriptor. Its name, type, etc. must be set before it is
|
779
706
|
* added to a message type.
|
780
707
|
*/
|
781
|
-
VALUE FieldDescriptor_alloc(VALUE klass) {
|
708
|
+
static VALUE FieldDescriptor_alloc(VALUE klass) {
|
782
709
|
FieldDescriptor* self = ALLOC(FieldDescriptor);
|
783
|
-
VALUE ret = TypedData_Wrap_Struct(klass, &
|
710
|
+
VALUE ret = TypedData_Wrap_Struct(klass, &FieldDescriptor_type, self);
|
784
711
|
self->fielddef = NULL;
|
785
712
|
return ret;
|
786
713
|
}
|
787
714
|
|
788
|
-
void FieldDescriptor_register(VALUE module) {
|
789
|
-
VALUE klass = rb_define_class_under(
|
790
|
-
module, "FieldDescriptor", rb_cObject);
|
791
|
-
rb_define_alloc_func(klass, FieldDescriptor_alloc);
|
792
|
-
rb_define_method(klass, "initialize", FieldDescriptor_initialize, 3);
|
793
|
-
rb_define_method(klass, "name", FieldDescriptor_name, 0);
|
794
|
-
rb_define_method(klass, "type", FieldDescriptor_type, 0);
|
795
|
-
rb_define_method(klass, "default", FieldDescriptor_default, 0);
|
796
|
-
rb_define_method(klass, "label", FieldDescriptor_label, 0);
|
797
|
-
rb_define_method(klass, "number", FieldDescriptor_number, 0);
|
798
|
-
rb_define_method(klass, "submsg_name", FieldDescriptor_submsg_name, 0);
|
799
|
-
rb_define_method(klass, "subtype", FieldDescriptor_subtype, 0);
|
800
|
-
rb_define_method(klass, "has?", FieldDescriptor_has, 1);
|
801
|
-
rb_define_method(klass, "clear", FieldDescriptor_clear, 1);
|
802
|
-
rb_define_method(klass, "get", FieldDescriptor_get, 1);
|
803
|
-
rb_define_method(klass, "set", FieldDescriptor_set, 2);
|
804
|
-
rb_gc_register_address(&cFieldDescriptor);
|
805
|
-
cFieldDescriptor = klass;
|
806
|
-
}
|
807
|
-
|
808
715
|
/*
|
809
716
|
* call-seq:
|
810
717
|
* EnumDescriptor.new(c_only_cookie, pool, ptr) => EnumDescriptor
|
811
718
|
*
|
812
719
|
* Creates a descriptor wrapper object. May only be called from C.
|
813
720
|
*/
|
814
|
-
VALUE FieldDescriptor_initialize(VALUE _self, VALUE cookie,
|
815
|
-
|
816
|
-
|
721
|
+
static VALUE FieldDescriptor_initialize(VALUE _self, VALUE cookie,
|
722
|
+
VALUE descriptor_pool, VALUE ptr) {
|
723
|
+
FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
|
817
724
|
|
818
725
|
if (cookie != c_only_cookie) {
|
819
726
|
rb_raise(rb_eRuntimeError,
|
@@ -832,11 +739,12 @@ VALUE FieldDescriptor_initialize(VALUE _self, VALUE cookie,
|
|
832
739
|
*
|
833
740
|
* Returns the name of this field.
|
834
741
|
*/
|
835
|
-
VALUE FieldDescriptor_name(VALUE _self) {
|
836
|
-
|
742
|
+
static VALUE FieldDescriptor_name(VALUE _self) {
|
743
|
+
FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
|
837
744
|
return rb_str_maybe_null(upb_fielddef_name(self->fielddef));
|
838
745
|
}
|
839
746
|
|
747
|
+
// Non-static, exposed to other .c files.
|
840
748
|
upb_fieldtype_t ruby_to_fieldtype(VALUE type) {
|
841
749
|
if (TYPE(type) != T_SYMBOL) {
|
842
750
|
rb_raise(rb_eArgError, "Expected symbol for field type.");
|
@@ -865,27 +773,7 @@ upb_fieldtype_t ruby_to_fieldtype(VALUE type) {
|
|
865
773
|
return 0;
|
866
774
|
}
|
867
775
|
|
868
|
-
|
869
|
-
switch (type) {
|
870
|
-
#define CONVERT(upb, ruby) \
|
871
|
-
case UPB_TYPE_ ## upb : return ID2SYM(rb_intern( # ruby ));
|
872
|
-
CONVERT(FLOAT, float);
|
873
|
-
CONVERT(DOUBLE, double);
|
874
|
-
CONVERT(BOOL, bool);
|
875
|
-
CONVERT(STRING, string);
|
876
|
-
CONVERT(BYTES, bytes);
|
877
|
-
CONVERT(MESSAGE, message);
|
878
|
-
CONVERT(ENUM, enum);
|
879
|
-
CONVERT(INT32, int32);
|
880
|
-
CONVERT(INT64, int64);
|
881
|
-
CONVERT(UINT32, uint32);
|
882
|
-
CONVERT(UINT64, uint64);
|
883
|
-
#undef CONVERT
|
884
|
-
}
|
885
|
-
return Qnil;
|
886
|
-
}
|
887
|
-
|
888
|
-
upb_descriptortype_t ruby_to_descriptortype(VALUE type) {
|
776
|
+
static upb_descriptortype_t ruby_to_descriptortype(VALUE type) {
|
889
777
|
if (TYPE(type) != T_SYMBOL) {
|
890
778
|
rb_raise(rb_eArgError, "Expected symbol for field type.");
|
891
779
|
}
|
@@ -920,7 +808,7 @@ upb_descriptortype_t ruby_to_descriptortype(VALUE type) {
|
|
920
808
|
return 0;
|
921
809
|
}
|
922
810
|
|
923
|
-
VALUE descriptortype_to_ruby(upb_descriptortype_t type) {
|
811
|
+
static VALUE descriptortype_to_ruby(upb_descriptortype_t type) {
|
924
812
|
switch (type) {
|
925
813
|
#define CONVERT(upb, ruby) \
|
926
814
|
case UPB_DESCRIPTOR_TYPE_ ## upb : return ID2SYM(rb_intern( # ruby ));
|
@@ -947,29 +835,6 @@ VALUE descriptortype_to_ruby(upb_descriptortype_t type) {
|
|
947
835
|
return Qnil;
|
948
836
|
}
|
949
837
|
|
950
|
-
VALUE ruby_to_label(VALUE label) {
|
951
|
-
upb_label_t upb_label;
|
952
|
-
bool converted = false;
|
953
|
-
|
954
|
-
#define CONVERT(upb, ruby) \
|
955
|
-
if (SYM2ID(label) == rb_intern( # ruby )) { \
|
956
|
-
upb_label = UPB_LABEL_ ## upb; \
|
957
|
-
converted = true; \
|
958
|
-
}
|
959
|
-
|
960
|
-
CONVERT(OPTIONAL, optional);
|
961
|
-
CONVERT(REQUIRED, required);
|
962
|
-
CONVERT(REPEATED, repeated);
|
963
|
-
|
964
|
-
#undef CONVERT
|
965
|
-
|
966
|
-
if (!converted) {
|
967
|
-
rb_raise(rb_eArgError, "Unknown field label.");
|
968
|
-
}
|
969
|
-
|
970
|
-
return upb_label;
|
971
|
-
}
|
972
|
-
|
973
838
|
/*
|
974
839
|
* call-seq:
|
975
840
|
* FieldDescriptor.type => type
|
@@ -980,8 +845,8 @@ VALUE ruby_to_label(VALUE label) {
|
|
980
845
|
* :int32, :int64, :uint32, :uint64, :float, :double, :bool, :string,
|
981
846
|
* :bytes, :message.
|
982
847
|
*/
|
983
|
-
VALUE
|
984
|
-
|
848
|
+
static VALUE FieldDescriptor__type(VALUE _self) {
|
849
|
+
FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
|
985
850
|
return descriptortype_to_ruby(upb_fielddef_descriptortype(self->fielddef));
|
986
851
|
}
|
987
852
|
|
@@ -991,9 +856,16 @@ VALUE FieldDescriptor_type(VALUE _self) {
|
|
991
856
|
*
|
992
857
|
* Returns this field's default, as a Ruby object, or nil if not yet set.
|
993
858
|
*/
|
994
|
-
VALUE FieldDescriptor_default(VALUE _self) {
|
995
|
-
|
996
|
-
|
859
|
+
static VALUE FieldDescriptor_default(VALUE _self) {
|
860
|
+
FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
|
861
|
+
const upb_fielddef *f = self->fielddef;
|
862
|
+
upb_msgval default_val = {0};
|
863
|
+
if (upb_fielddef_issubmsg(f)) {
|
864
|
+
return Qnil;
|
865
|
+
} else if (!upb_fielddef_isseq(f)) {
|
866
|
+
default_val = upb_fielddef_default(f);
|
867
|
+
}
|
868
|
+
return Convert_UpbToRuby(default_val, TypeInfo_get(self->fielddef), Qnil);
|
997
869
|
}
|
998
870
|
|
999
871
|
/*
|
@@ -1005,8 +877,8 @@ VALUE FieldDescriptor_default(VALUE _self) {
|
|
1005
877
|
* Valid field labels are:
|
1006
878
|
* :optional, :repeated
|
1007
879
|
*/
|
1008
|
-
VALUE FieldDescriptor_label(VALUE _self) {
|
1009
|
-
|
880
|
+
static VALUE FieldDescriptor_label(VALUE _self) {
|
881
|
+
FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
|
1010
882
|
switch (upb_fielddef_label(self->fielddef)) {
|
1011
883
|
#define CONVERT(upb, ruby) \
|
1012
884
|
case UPB_LABEL_ ## upb : return ID2SYM(rb_intern( # ruby ));
|
@@ -1027,8 +899,8 @@ VALUE FieldDescriptor_label(VALUE _self) {
|
|
1027
899
|
*
|
1028
900
|
* Returns the tag number for this field.
|
1029
901
|
*/
|
1030
|
-
VALUE FieldDescriptor_number(VALUE _self) {
|
1031
|
-
|
902
|
+
static VALUE FieldDescriptor_number(VALUE _self) {
|
903
|
+
FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
|
1032
904
|
return INT2NUM(upb_fielddef_number(self->fielddef));
|
1033
905
|
}
|
1034
906
|
|
@@ -1041,8 +913,8 @@ VALUE FieldDescriptor_number(VALUE _self) {
|
|
1041
913
|
* name will be resolved within the context of the pool to which the containing
|
1042
914
|
* message type is added.
|
1043
915
|
*/
|
1044
|
-
VALUE FieldDescriptor_submsg_name(VALUE _self) {
|
1045
|
-
|
916
|
+
static VALUE FieldDescriptor_submsg_name(VALUE _self) {
|
917
|
+
FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
|
1046
918
|
switch (upb_fielddef_type(self->fielddef)) {
|
1047
919
|
case UPB_TYPE_ENUM:
|
1048
920
|
return rb_str_new2(
|
@@ -1064,8 +936,8 @@ VALUE FieldDescriptor_submsg_name(VALUE _self) {
|
|
1064
936
|
* called *until* the containing message type is added to a pool (and thus
|
1065
937
|
* resolved).
|
1066
938
|
*/
|
1067
|
-
VALUE FieldDescriptor_subtype(VALUE _self) {
|
1068
|
-
|
939
|
+
static VALUE FieldDescriptor_subtype(VALUE _self) {
|
940
|
+
FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
|
1069
941
|
switch (upb_fielddef_type(self->fielddef)) {
|
1070
942
|
case UPB_TYPE_ENUM:
|
1071
943
|
return get_enumdef_obj(self->descriptor_pool,
|
@@ -1085,14 +957,19 @@ VALUE FieldDescriptor_subtype(VALUE _self) {
|
|
1085
957
|
* Returns the value set for this field on the given message. Raises an
|
1086
958
|
* exception if message is of the wrong type.
|
1087
959
|
*/
|
1088
|
-
VALUE FieldDescriptor_get(VALUE _self, VALUE msg_rb) {
|
1089
|
-
|
1090
|
-
|
1091
|
-
|
1092
|
-
|
960
|
+
static VALUE FieldDescriptor_get(VALUE _self, VALUE msg_rb) {
|
961
|
+
FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
|
962
|
+
const upb_msgdef *m;
|
963
|
+
const upb_msgdef *msg = Message_Get(msg_rb, &m);
|
964
|
+
VALUE arena = Message_GetArena(msg_rb);
|
965
|
+
upb_msgval msgval;
|
966
|
+
|
967
|
+
if (m != upb_fielddef_containingtype(self->fielddef)) {
|
1093
968
|
rb_raise(cTypeError, "get method called on wrong message type");
|
1094
969
|
}
|
1095
|
-
|
970
|
+
|
971
|
+
msgval = upb_msg_get(msg, self->fielddef);
|
972
|
+
return Convert_UpbToRuby(msgval, TypeInfo_get(self->fielddef), arena);
|
1096
973
|
}
|
1097
974
|
|
1098
975
|
/*
|
@@ -1102,17 +979,18 @@ VALUE FieldDescriptor_get(VALUE _self, VALUE msg_rb) {
|
|
1102
979
|
* Returns whether the value is set on the given message. Raises an
|
1103
980
|
* exception when calling for fields that do not have presence.
|
1104
981
|
*/
|
1105
|
-
VALUE FieldDescriptor_has(VALUE _self, VALUE msg_rb) {
|
1106
|
-
|
1107
|
-
|
1108
|
-
|
1109
|
-
|
982
|
+
static VALUE FieldDescriptor_has(VALUE _self, VALUE msg_rb) {
|
983
|
+
FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
|
984
|
+
const upb_msgdef *m;
|
985
|
+
const upb_msgdef *msg = Message_Get(msg_rb, &m);
|
986
|
+
|
987
|
+
if (m != upb_fielddef_containingtype(self->fielddef)) {
|
1110
988
|
rb_raise(cTypeError, "has method called on wrong message type");
|
1111
989
|
} else if (!upb_fielddef_haspresence(self->fielddef)) {
|
1112
990
|
rb_raise(rb_eArgError, "does not track presence");
|
1113
991
|
}
|
1114
992
|
|
1115
|
-
return
|
993
|
+
return upb_msg_has(msg, self->fielddef) ? Qtrue : Qfalse;
|
1116
994
|
}
|
1117
995
|
|
1118
996
|
/*
|
@@ -1121,15 +999,16 @@ VALUE FieldDescriptor_has(VALUE _self, VALUE msg_rb) {
|
|
1121
999
|
*
|
1122
1000
|
* Clears the field from the message if it's set.
|
1123
1001
|
*/
|
1124
|
-
VALUE FieldDescriptor_clear(VALUE _self, VALUE msg_rb) {
|
1125
|
-
|
1126
|
-
|
1127
|
-
|
1128
|
-
|
1002
|
+
static VALUE FieldDescriptor_clear(VALUE _self, VALUE msg_rb) {
|
1003
|
+
FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
|
1004
|
+
const upb_msgdef *m;
|
1005
|
+
upb_msgdef *msg = Message_GetMutable(msg_rb, &m);
|
1006
|
+
|
1007
|
+
if (m != upb_fielddef_containingtype(self->fielddef)) {
|
1129
1008
|
rb_raise(cTypeError, "has method called on wrong message type");
|
1130
1009
|
}
|
1131
1010
|
|
1132
|
-
|
1011
|
+
upb_msg_clearfield(msg, self->fielddef);
|
1133
1012
|
return Qnil;
|
1134
1013
|
}
|
1135
1014
|
|
@@ -1141,30 +1020,69 @@ VALUE FieldDescriptor_clear(VALUE _self, VALUE msg_rb) {
|
|
1141
1020
|
* message. Raises an exception if message is of the wrong type. Performs the
|
1142
1021
|
* ordinary type-checks for field setting.
|
1143
1022
|
*/
|
1144
|
-
VALUE FieldDescriptor_set(VALUE _self, VALUE msg_rb, VALUE value) {
|
1145
|
-
|
1146
|
-
|
1147
|
-
|
1148
|
-
|
1023
|
+
static VALUE FieldDescriptor_set(VALUE _self, VALUE msg_rb, VALUE value) {
|
1024
|
+
FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
|
1025
|
+
const upb_msgdef *m;
|
1026
|
+
upb_msgdef *msg = Message_GetMutable(msg_rb, &m);
|
1027
|
+
upb_arena *arena = Arena_get(Message_GetArena(msg_rb));
|
1028
|
+
upb_msgval msgval;
|
1029
|
+
|
1030
|
+
if (m != upb_fielddef_containingtype(self->fielddef)) {
|
1149
1031
|
rb_raise(cTypeError, "set method called on wrong message type");
|
1150
1032
|
}
|
1151
|
-
|
1033
|
+
|
1034
|
+
msgval = Convert_RubyToUpb(value, upb_fielddef_name(self->fielddef),
|
1035
|
+
TypeInfo_get(self->fielddef), arena);
|
1036
|
+
upb_msg_set(msg, self->fielddef, msgval, arena);
|
1152
1037
|
return Qnil;
|
1153
1038
|
}
|
1154
1039
|
|
1040
|
+
static void FieldDescriptor_register(VALUE module) {
|
1041
|
+
VALUE klass = rb_define_class_under(
|
1042
|
+
module, "FieldDescriptor", rb_cObject);
|
1043
|
+
rb_define_alloc_func(klass, FieldDescriptor_alloc);
|
1044
|
+
rb_define_method(klass, "initialize", FieldDescriptor_initialize, 3);
|
1045
|
+
rb_define_method(klass, "name", FieldDescriptor_name, 0);
|
1046
|
+
rb_define_method(klass, "type", FieldDescriptor__type, 0);
|
1047
|
+
rb_define_method(klass, "default", FieldDescriptor_default, 0);
|
1048
|
+
rb_define_method(klass, "label", FieldDescriptor_label, 0);
|
1049
|
+
rb_define_method(klass, "number", FieldDescriptor_number, 0);
|
1050
|
+
rb_define_method(klass, "submsg_name", FieldDescriptor_submsg_name, 0);
|
1051
|
+
rb_define_method(klass, "subtype", FieldDescriptor_subtype, 0);
|
1052
|
+
rb_define_method(klass, "has?", FieldDescriptor_has, 1);
|
1053
|
+
rb_define_method(klass, "clear", FieldDescriptor_clear, 1);
|
1054
|
+
rb_define_method(klass, "get", FieldDescriptor_get, 1);
|
1055
|
+
rb_define_method(klass, "set", FieldDescriptor_set, 2);
|
1056
|
+
rb_gc_register_address(&cFieldDescriptor);
|
1057
|
+
cFieldDescriptor = klass;
|
1058
|
+
}
|
1059
|
+
|
1155
1060
|
// -----------------------------------------------------------------------------
|
1156
1061
|
// OneofDescriptor.
|
1157
1062
|
// -----------------------------------------------------------------------------
|
1158
1063
|
|
1159
|
-
|
1064
|
+
typedef struct {
|
1065
|
+
const upb_oneofdef* oneofdef;
|
1066
|
+
VALUE descriptor_pool; // Owns the upb_oneofdef.
|
1067
|
+
} OneofDescriptor;
|
1068
|
+
|
1069
|
+
static VALUE cOneofDescriptor = Qnil;
|
1160
1070
|
|
1161
|
-
void OneofDescriptor_mark(void* _self) {
|
1071
|
+
static void OneofDescriptor_mark(void* _self) {
|
1162
1072
|
OneofDescriptor* self = _self;
|
1163
1073
|
rb_gc_mark(self->descriptor_pool);
|
1164
1074
|
}
|
1165
1075
|
|
1166
|
-
|
1167
|
-
|
1076
|
+
static const rb_data_type_t OneofDescriptor_type = {
|
1077
|
+
"Google::Protobuf::OneofDescriptor",
|
1078
|
+
{OneofDescriptor_mark, RUBY_DEFAULT_FREE, NULL},
|
1079
|
+
.flags = RUBY_TYPED_FREE_IMMEDIATELY,
|
1080
|
+
};
|
1081
|
+
|
1082
|
+
static OneofDescriptor* ruby_to_OneofDescriptor(VALUE val) {
|
1083
|
+
OneofDescriptor* ret;
|
1084
|
+
TypedData_Get_Struct(val, OneofDescriptor, &OneofDescriptor_type, ret);
|
1085
|
+
return ret;
|
1168
1086
|
}
|
1169
1087
|
|
1170
1088
|
/*
|
@@ -1174,35 +1092,23 @@ void OneofDescriptor_free(void* _self) {
|
|
1174
1092
|
* Creates a new, empty, oneof descriptor. The oneof may only be modified prior
|
1175
1093
|
* to being added to a message descriptor which is subsequently added to a pool.
|
1176
1094
|
*/
|
1177
|
-
VALUE OneofDescriptor_alloc(VALUE klass) {
|
1095
|
+
static VALUE OneofDescriptor_alloc(VALUE klass) {
|
1178
1096
|
OneofDescriptor* self = ALLOC(OneofDescriptor);
|
1179
|
-
VALUE ret = TypedData_Wrap_Struct(klass, &
|
1097
|
+
VALUE ret = TypedData_Wrap_Struct(klass, &OneofDescriptor_type, self);
|
1180
1098
|
self->oneofdef = NULL;
|
1181
1099
|
self->descriptor_pool = Qnil;
|
1182
1100
|
return ret;
|
1183
1101
|
}
|
1184
1102
|
|
1185
|
-
void OneofDescriptor_register(VALUE module) {
|
1186
|
-
VALUE klass = rb_define_class_under(
|
1187
|
-
module, "OneofDescriptor", rb_cObject);
|
1188
|
-
rb_define_alloc_func(klass, OneofDescriptor_alloc);
|
1189
|
-
rb_define_method(klass, "initialize", OneofDescriptor_initialize, 3);
|
1190
|
-
rb_define_method(klass, "name", OneofDescriptor_name, 0);
|
1191
|
-
rb_define_method(klass, "each", OneofDescriptor_each, 0);
|
1192
|
-
rb_include_module(klass, rb_mEnumerable);
|
1193
|
-
rb_gc_register_address(&cOneofDescriptor);
|
1194
|
-
cOneofDescriptor = klass;
|
1195
|
-
}
|
1196
|
-
|
1197
1103
|
/*
|
1198
1104
|
* call-seq:
|
1199
1105
|
* OneofDescriptor.new(c_only_cookie, pool, ptr) => OneofDescriptor
|
1200
1106
|
*
|
1201
1107
|
* Creates a descriptor wrapper object. May only be called from C.
|
1202
1108
|
*/
|
1203
|
-
VALUE OneofDescriptor_initialize(VALUE _self, VALUE cookie,
|
1109
|
+
static VALUE OneofDescriptor_initialize(VALUE _self, VALUE cookie,
|
1204
1110
|
VALUE descriptor_pool, VALUE ptr) {
|
1205
|
-
|
1111
|
+
OneofDescriptor* self = ruby_to_OneofDescriptor(_self);
|
1206
1112
|
|
1207
1113
|
if (cookie != c_only_cookie) {
|
1208
1114
|
rb_raise(rb_eRuntimeError,
|
@@ -1221,8 +1127,8 @@ VALUE OneofDescriptor_initialize(VALUE _self, VALUE cookie,
|
|
1221
1127
|
*
|
1222
1128
|
* Returns the name of this oneof.
|
1223
1129
|
*/
|
1224
|
-
VALUE OneofDescriptor_name(VALUE _self) {
|
1225
|
-
|
1130
|
+
static VALUE OneofDescriptor_name(VALUE _self) {
|
1131
|
+
OneofDescriptor* self = ruby_to_OneofDescriptor(_self);
|
1226
1132
|
return rb_str_maybe_null(upb_oneofdef_name(self->oneofdef));
|
1227
1133
|
}
|
1228
1134
|
|
@@ -1232,8 +1138,8 @@ VALUE OneofDescriptor_name(VALUE _self) {
|
|
1232
1138
|
*
|
1233
1139
|
* Iterates through fields in this oneof, yielding to the block on each one.
|
1234
1140
|
*/
|
1235
|
-
VALUE OneofDescriptor_each(VALUE _self) {
|
1236
|
-
|
1141
|
+
static VALUE OneofDescriptor_each(VALUE _self) {
|
1142
|
+
OneofDescriptor* self = ruby_to_OneofDescriptor(_self);
|
1237
1143
|
upb_oneof_iter it;
|
1238
1144
|
for (upb_oneof_begin(&it, self->oneofdef);
|
1239
1145
|
!upb_oneof_done(&it);
|
@@ -1245,40 +1151,72 @@ VALUE OneofDescriptor_each(VALUE _self) {
|
|
1245
1151
|
return Qnil;
|
1246
1152
|
}
|
1247
1153
|
|
1154
|
+
static void OneofDescriptor_register(VALUE module) {
|
1155
|
+
VALUE klass = rb_define_class_under(
|
1156
|
+
module, "OneofDescriptor", rb_cObject);
|
1157
|
+
rb_define_alloc_func(klass, OneofDescriptor_alloc);
|
1158
|
+
rb_define_method(klass, "initialize", OneofDescriptor_initialize, 3);
|
1159
|
+
rb_define_method(klass, "name", OneofDescriptor_name, 0);
|
1160
|
+
rb_define_method(klass, "each", OneofDescriptor_each, 0);
|
1161
|
+
rb_include_module(klass, rb_mEnumerable);
|
1162
|
+
rb_gc_register_address(&cOneofDescriptor);
|
1163
|
+
cOneofDescriptor = klass;
|
1164
|
+
}
|
1165
|
+
|
1248
1166
|
// -----------------------------------------------------------------------------
|
1249
1167
|
// EnumDescriptor.
|
1250
1168
|
// -----------------------------------------------------------------------------
|
1251
1169
|
|
1252
|
-
|
1170
|
+
typedef struct {
|
1171
|
+
const upb_enumdef* enumdef;
|
1172
|
+
VALUE module; // begins as nil
|
1173
|
+
VALUE descriptor_pool; // Owns the upb_enumdef.
|
1174
|
+
} EnumDescriptor;
|
1175
|
+
|
1176
|
+
static VALUE cEnumDescriptor = Qnil;
|
1253
1177
|
|
1254
|
-
void EnumDescriptor_mark(void* _self) {
|
1178
|
+
static void EnumDescriptor_mark(void* _self) {
|
1255
1179
|
EnumDescriptor* self = _self;
|
1256
1180
|
rb_gc_mark(self->module);
|
1257
1181
|
rb_gc_mark(self->descriptor_pool);
|
1258
1182
|
}
|
1259
1183
|
|
1260
|
-
|
1261
|
-
|
1184
|
+
static const rb_data_type_t EnumDescriptor_type = {
|
1185
|
+
"Google::Protobuf::EnumDescriptor",
|
1186
|
+
{EnumDescriptor_mark, RUBY_DEFAULT_FREE, NULL},
|
1187
|
+
.flags = RUBY_TYPED_FREE_IMMEDIATELY,
|
1188
|
+
};
|
1189
|
+
|
1190
|
+
static EnumDescriptor* ruby_to_EnumDescriptor(VALUE val) {
|
1191
|
+
EnumDescriptor* ret;
|
1192
|
+
TypedData_Get_Struct(val, EnumDescriptor, &EnumDescriptor_type, ret);
|
1193
|
+
return ret;
|
1262
1194
|
}
|
1263
1195
|
|
1264
|
-
VALUE EnumDescriptor_alloc(VALUE klass) {
|
1196
|
+
static VALUE EnumDescriptor_alloc(VALUE klass) {
|
1265
1197
|
EnumDescriptor* self = ALLOC(EnumDescriptor);
|
1266
|
-
VALUE ret = TypedData_Wrap_Struct(klass, &
|
1198
|
+
VALUE ret = TypedData_Wrap_Struct(klass, &EnumDescriptor_type, self);
|
1267
1199
|
self->enumdef = NULL;
|
1268
1200
|
self->module = Qnil;
|
1269
1201
|
self->descriptor_pool = Qnil;
|
1270
1202
|
return ret;
|
1271
1203
|
}
|
1272
1204
|
|
1205
|
+
// Exposed to other modules in defs.h.
|
1206
|
+
const upb_enumdef *EnumDescriptor_GetEnumDef(VALUE enum_desc_rb) {
|
1207
|
+
EnumDescriptor *desc = ruby_to_EnumDescriptor(enum_desc_rb);
|
1208
|
+
return desc->enumdef;
|
1209
|
+
}
|
1210
|
+
|
1273
1211
|
/*
|
1274
1212
|
* call-seq:
|
1275
1213
|
* EnumDescriptor.new(c_only_cookie, ptr) => EnumDescriptor
|
1276
1214
|
*
|
1277
1215
|
* Creates a descriptor wrapper object. May only be called from C.
|
1278
1216
|
*/
|
1279
|
-
VALUE EnumDescriptor_initialize(VALUE _self, VALUE cookie,
|
1280
|
-
|
1281
|
-
|
1217
|
+
static VALUE EnumDescriptor_initialize(VALUE _self, VALUE cookie,
|
1218
|
+
VALUE descriptor_pool, VALUE ptr) {
|
1219
|
+
EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
|
1282
1220
|
|
1283
1221
|
if (cookie != c_only_cookie) {
|
1284
1222
|
rb_raise(rb_eRuntimeError,
|
@@ -1291,30 +1229,14 @@ VALUE EnumDescriptor_initialize(VALUE _self, VALUE cookie,
|
|
1291
1229
|
return Qnil;
|
1292
1230
|
}
|
1293
1231
|
|
1294
|
-
void EnumDescriptor_register(VALUE module) {
|
1295
|
-
VALUE klass = rb_define_class_under(
|
1296
|
-
module, "EnumDescriptor", rb_cObject);
|
1297
|
-
rb_define_alloc_func(klass, EnumDescriptor_alloc);
|
1298
|
-
rb_define_method(klass, "initialize", EnumDescriptor_initialize, 3);
|
1299
|
-
rb_define_method(klass, "name", EnumDescriptor_name, 0);
|
1300
|
-
rb_define_method(klass, "lookup_name", EnumDescriptor_lookup_name, 1);
|
1301
|
-
rb_define_method(klass, "lookup_value", EnumDescriptor_lookup_value, 1);
|
1302
|
-
rb_define_method(klass, "each", EnumDescriptor_each, 0);
|
1303
|
-
rb_define_method(klass, "enummodule", EnumDescriptor_enummodule, 0);
|
1304
|
-
rb_define_method(klass, "file_descriptor", EnumDescriptor_file_descriptor, 0);
|
1305
|
-
rb_include_module(klass, rb_mEnumerable);
|
1306
|
-
rb_gc_register_address(&cEnumDescriptor);
|
1307
|
-
cEnumDescriptor = klass;
|
1308
|
-
}
|
1309
|
-
|
1310
1232
|
/*
|
1311
1233
|
* call-seq:
|
1312
1234
|
* EnumDescriptor.file_descriptor
|
1313
1235
|
*
|
1314
1236
|
* Returns the FileDescriptor object this enum belongs to.
|
1315
1237
|
*/
|
1316
|
-
VALUE EnumDescriptor_file_descriptor(VALUE _self) {
|
1317
|
-
|
1238
|
+
static VALUE EnumDescriptor_file_descriptor(VALUE _self) {
|
1239
|
+
EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
|
1318
1240
|
return get_filedef_obj(self->descriptor_pool,
|
1319
1241
|
upb_enumdef_file(self->enumdef));
|
1320
1242
|
}
|
@@ -1325,8 +1247,8 @@ VALUE EnumDescriptor_file_descriptor(VALUE _self) {
|
|
1325
1247
|
*
|
1326
1248
|
* Returns the name of this enum type.
|
1327
1249
|
*/
|
1328
|
-
VALUE EnumDescriptor_name(VALUE _self) {
|
1329
|
-
|
1250
|
+
static VALUE EnumDescriptor_name(VALUE _self) {
|
1251
|
+
EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
|
1330
1252
|
return rb_str_maybe_null(upb_enumdef_fullname(self->enumdef));
|
1331
1253
|
}
|
1332
1254
|
|
@@ -1337,8 +1259,8 @@ VALUE EnumDescriptor_name(VALUE _self) {
|
|
1337
1259
|
* Returns the numeric value corresponding to the given key name (as a Ruby
|
1338
1260
|
* symbol), or nil if none.
|
1339
1261
|
*/
|
1340
|
-
VALUE EnumDescriptor_lookup_name(VALUE _self, VALUE name) {
|
1341
|
-
|
1262
|
+
static VALUE EnumDescriptor_lookup_name(VALUE _self, VALUE name) {
|
1263
|
+
EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
|
1342
1264
|
const char* name_str= rb_id2name(SYM2ID(name));
|
1343
1265
|
int32_t val = 0;
|
1344
1266
|
if (upb_enumdef_ntoiz(self->enumdef, name_str, &val)) {
|
@@ -1355,8 +1277,8 @@ VALUE EnumDescriptor_lookup_name(VALUE _self, VALUE name) {
|
|
1355
1277
|
* Returns the key name (as a Ruby symbol) corresponding to the integer value,
|
1356
1278
|
* or nil if none.
|
1357
1279
|
*/
|
1358
|
-
VALUE EnumDescriptor_lookup_value(VALUE _self, VALUE number) {
|
1359
|
-
|
1280
|
+
static VALUE EnumDescriptor_lookup_value(VALUE _self, VALUE number) {
|
1281
|
+
EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
|
1360
1282
|
int32_t val = NUM2INT(number);
|
1361
1283
|
const char* name = upb_enumdef_iton(self->enumdef, val);
|
1362
1284
|
if (name != NULL) {
|
@@ -1373,8 +1295,8 @@ VALUE EnumDescriptor_lookup_value(VALUE _self, VALUE number) {
|
|
1373
1295
|
* Iterates over key => value mappings in this enum's definition, yielding to
|
1374
1296
|
* the block with (key, value) arguments for each one.
|
1375
1297
|
*/
|
1376
|
-
VALUE EnumDescriptor_each(VALUE _self) {
|
1377
|
-
|
1298
|
+
static VALUE EnumDescriptor_each(VALUE _self) {
|
1299
|
+
EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
|
1378
1300
|
|
1379
1301
|
upb_enum_iter it;
|
1380
1302
|
for (upb_enum_begin(&it, self->enumdef);
|
@@ -1394,53 +1316,365 @@ VALUE EnumDescriptor_each(VALUE _self) {
|
|
1394
1316
|
*
|
1395
1317
|
* Returns the Ruby module corresponding to this enum type.
|
1396
1318
|
*/
|
1397
|
-
VALUE EnumDescriptor_enummodule(VALUE _self) {
|
1398
|
-
|
1319
|
+
static VALUE EnumDescriptor_enummodule(VALUE _self) {
|
1320
|
+
EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
|
1399
1321
|
if (self->module == Qnil) {
|
1400
1322
|
self->module = build_module_from_enumdesc(_self);
|
1401
1323
|
}
|
1402
1324
|
return self->module;
|
1403
1325
|
}
|
1404
1326
|
|
1327
|
+
static void EnumDescriptor_register(VALUE module) {
|
1328
|
+
VALUE klass = rb_define_class_under(
|
1329
|
+
module, "EnumDescriptor", rb_cObject);
|
1330
|
+
rb_define_alloc_func(klass, EnumDescriptor_alloc);
|
1331
|
+
rb_define_method(klass, "initialize", EnumDescriptor_initialize, 3);
|
1332
|
+
rb_define_method(klass, "name", EnumDescriptor_name, 0);
|
1333
|
+
rb_define_method(klass, "lookup_name", EnumDescriptor_lookup_name, 1);
|
1334
|
+
rb_define_method(klass, "lookup_value", EnumDescriptor_lookup_value, 1);
|
1335
|
+
rb_define_method(klass, "each", EnumDescriptor_each, 0);
|
1336
|
+
rb_define_method(klass, "enummodule", EnumDescriptor_enummodule, 0);
|
1337
|
+
rb_define_method(klass, "file_descriptor", EnumDescriptor_file_descriptor, 0);
|
1338
|
+
rb_include_module(klass, rb_mEnumerable);
|
1339
|
+
rb_gc_register_address(&cEnumDescriptor);
|
1340
|
+
cEnumDescriptor = klass;
|
1341
|
+
}
|
1342
|
+
|
1343
|
+
// -----------------------------------------------------------------------------
|
1344
|
+
// FileBuilderContext.
|
1345
|
+
// -----------------------------------------------------------------------------
|
1346
|
+
|
1347
|
+
typedef struct {
|
1348
|
+
upb_arena *arena;
|
1349
|
+
google_protobuf_FileDescriptorProto* file_proto;
|
1350
|
+
VALUE descriptor_pool;
|
1351
|
+
} FileBuilderContext;
|
1352
|
+
|
1353
|
+
static VALUE cFileBuilderContext = Qnil;
|
1354
|
+
|
1355
|
+
static void FileBuilderContext_mark(void* _self) {
|
1356
|
+
FileBuilderContext* self = _self;
|
1357
|
+
rb_gc_mark(self->descriptor_pool);
|
1358
|
+
}
|
1359
|
+
|
1360
|
+
static void FileBuilderContext_free(void* _self) {
|
1361
|
+
FileBuilderContext* self = _self;
|
1362
|
+
upb_arena_free(self->arena);
|
1363
|
+
xfree(self);
|
1364
|
+
}
|
1365
|
+
|
1366
|
+
static const rb_data_type_t FileBuilderContext_type = {
|
1367
|
+
"Google::Protobuf::Internal::FileBuilderContext",
|
1368
|
+
{FileBuilderContext_mark, FileBuilderContext_free, NULL},
|
1369
|
+
.flags = RUBY_TYPED_FREE_IMMEDIATELY,
|
1370
|
+
};
|
1371
|
+
|
1372
|
+
static FileBuilderContext* ruby_to_FileBuilderContext(VALUE val) {
|
1373
|
+
FileBuilderContext* ret;
|
1374
|
+
TypedData_Get_Struct(val, FileBuilderContext, &FileBuilderContext_type, ret);
|
1375
|
+
return ret;
|
1376
|
+
}
|
1377
|
+
|
1378
|
+
static upb_strview FileBuilderContext_strdup2(VALUE _self, const char *str) {
|
1379
|
+
FileBuilderContext* self = ruby_to_FileBuilderContext(_self);
|
1380
|
+
upb_strview ret;
|
1381
|
+
char *data;
|
1382
|
+
|
1383
|
+
ret.size = strlen(str);
|
1384
|
+
data = upb_malloc(upb_arena_alloc(self->arena), ret.size + 1);
|
1385
|
+
ret.data = data;
|
1386
|
+
memcpy(data, str, ret.size);
|
1387
|
+
/* Null-terminate required by rewrite_enum_defaults() above. */
|
1388
|
+
data[ret.size] = '\0';
|
1389
|
+
return ret;
|
1390
|
+
}
|
1391
|
+
|
1392
|
+
static upb_strview FileBuilderContext_strdup(VALUE _self, VALUE rb_str) {
|
1393
|
+
return FileBuilderContext_strdup2(_self, get_str(rb_str));
|
1394
|
+
}
|
1395
|
+
|
1396
|
+
static upb_strview FileBuilderContext_strdup_sym(VALUE _self, VALUE rb_sym) {
|
1397
|
+
Check_Type(rb_sym, T_SYMBOL);
|
1398
|
+
return FileBuilderContext_strdup(_self, rb_id2str(SYM2ID(rb_sym)));
|
1399
|
+
}
|
1400
|
+
|
1401
|
+
static VALUE FileBuilderContext_alloc(VALUE klass) {
|
1402
|
+
FileBuilderContext* self = ALLOC(FileBuilderContext);
|
1403
|
+
VALUE ret = TypedData_Wrap_Struct(klass, &FileBuilderContext_type, self);
|
1404
|
+
self->arena = upb_arena_new();
|
1405
|
+
self->file_proto = google_protobuf_FileDescriptorProto_new(self->arena);
|
1406
|
+
self->descriptor_pool = Qnil;
|
1407
|
+
return ret;
|
1408
|
+
}
|
1409
|
+
|
1410
|
+
/*
|
1411
|
+
* call-seq:
|
1412
|
+
* FileBuilderContext.new(descriptor_pool) => context
|
1413
|
+
*
|
1414
|
+
* Create a new file builder context for the given file descriptor and
|
1415
|
+
* builder context. This class is intended to serve as a DSL context to be used
|
1416
|
+
* with #instance_eval.
|
1417
|
+
*/
|
1418
|
+
static VALUE FileBuilderContext_initialize(VALUE _self, VALUE descriptor_pool,
|
1419
|
+
VALUE name, VALUE options) {
|
1420
|
+
FileBuilderContext* self = ruby_to_FileBuilderContext(_self);
|
1421
|
+
self->descriptor_pool = descriptor_pool;
|
1422
|
+
|
1423
|
+
google_protobuf_FileDescriptorProto_set_name(
|
1424
|
+
self->file_proto, FileBuilderContext_strdup(_self, name));
|
1425
|
+
|
1426
|
+
// Default syntax for Ruby is proto3.
|
1427
|
+
google_protobuf_FileDescriptorProto_set_syntax(
|
1428
|
+
self->file_proto,
|
1429
|
+
FileBuilderContext_strdup(_self, rb_str_new2("proto3")));
|
1430
|
+
|
1431
|
+
if (options != Qnil) {
|
1432
|
+
VALUE syntax;
|
1433
|
+
|
1434
|
+
Check_Type(options, T_HASH);
|
1435
|
+
syntax = rb_hash_lookup2(options, ID2SYM(rb_intern("syntax")), Qnil);
|
1436
|
+
|
1437
|
+
if (syntax != Qnil) {
|
1438
|
+
VALUE syntax_str;
|
1439
|
+
|
1440
|
+
Check_Type(syntax, T_SYMBOL);
|
1441
|
+
syntax_str = rb_id2str(SYM2ID(syntax));
|
1442
|
+
google_protobuf_FileDescriptorProto_set_syntax(
|
1443
|
+
self->file_proto, FileBuilderContext_strdup(_self, syntax_str));
|
1444
|
+
}
|
1445
|
+
}
|
1446
|
+
|
1447
|
+
return Qnil;
|
1448
|
+
}
|
1449
|
+
|
1450
|
+
static void MessageBuilderContext_add_synthetic_oneofs(VALUE _self);
|
1451
|
+
|
1452
|
+
/*
|
1453
|
+
* call-seq:
|
1454
|
+
* FileBuilderContext.add_message(name, &block)
|
1455
|
+
*
|
1456
|
+
* Creates a new, empty descriptor with the given name, and invokes the block in
|
1457
|
+
* the context of a MessageBuilderContext on that descriptor. The block can then
|
1458
|
+
* call, e.g., MessageBuilderContext#optional and MessageBuilderContext#repeated
|
1459
|
+
* methods to define the message fields.
|
1460
|
+
*
|
1461
|
+
* This is the recommended, idiomatic way to build message definitions.
|
1462
|
+
*/
|
1463
|
+
static VALUE FileBuilderContext_add_message(VALUE _self, VALUE name) {
|
1464
|
+
VALUE args[2] = { _self, name };
|
1465
|
+
VALUE ctx = rb_class_new_instance(2, args, cMessageBuilderContext);
|
1466
|
+
VALUE block = rb_block_proc();
|
1467
|
+
rb_funcall_with_block(ctx, rb_intern("instance_eval"), 0, NULL, block);
|
1468
|
+
MessageBuilderContext_add_synthetic_oneofs(ctx);
|
1469
|
+
return Qnil;
|
1470
|
+
}
|
1471
|
+
|
1472
|
+
/* We have to do some relatively complicated logic here for backward
|
1473
|
+
* compatibility.
|
1474
|
+
*
|
1475
|
+
* In descriptor.proto, messages are nested inside other messages if that is
|
1476
|
+
* what the original .proto file looks like. For example, suppose we have this
|
1477
|
+
* foo.proto:
|
1478
|
+
*
|
1479
|
+
* package foo;
|
1480
|
+
* message Bar {
|
1481
|
+
* message Baz {}
|
1482
|
+
* }
|
1483
|
+
*
|
1484
|
+
* The descriptor for this must look like this:
|
1485
|
+
*
|
1486
|
+
* file {
|
1487
|
+
* name: "test.proto"
|
1488
|
+
* package: "foo"
|
1489
|
+
* message_type {
|
1490
|
+
* name: "Bar"
|
1491
|
+
* nested_type {
|
1492
|
+
* name: "Baz"
|
1493
|
+
* }
|
1494
|
+
* }
|
1495
|
+
* }
|
1496
|
+
*
|
1497
|
+
* However, the Ruby generated code has always generated messages in a flat,
|
1498
|
+
* non-nested way:
|
1499
|
+
*
|
1500
|
+
* Google::Protobuf::DescriptorPool.generated_pool.build do
|
1501
|
+
* add_message "foo.Bar" do
|
1502
|
+
* end
|
1503
|
+
* add_message "foo.Bar.Baz" do
|
1504
|
+
* end
|
1505
|
+
* end
|
1506
|
+
*
|
1507
|
+
* Here we need to do a translation where we turn this generated code into the
|
1508
|
+
* above descriptor. We need to infer that "foo" is the package name, and not
|
1509
|
+
* a message itself.
|
1510
|
+
*
|
1511
|
+
* We delegate to Ruby to compute the transformation, for more concice and
|
1512
|
+
* readable code than we can do in C */
|
1513
|
+
static void rewrite_names(VALUE _file_builder,
|
1514
|
+
google_protobuf_FileDescriptorProto* file_proto) {
|
1515
|
+
FileBuilderContext* file_builder = ruby_to_FileBuilderContext(_file_builder);
|
1516
|
+
upb_arena *arena = file_builder->arena;
|
1517
|
+
// Build params (package, msg_names, enum_names).
|
1518
|
+
VALUE package = Qnil;
|
1519
|
+
VALUE msg_names = rb_ary_new();
|
1520
|
+
VALUE enum_names = rb_ary_new();
|
1521
|
+
size_t msg_count, enum_count, i;
|
1522
|
+
VALUE new_package, nesting, msg_ents, enum_ents;
|
1523
|
+
google_protobuf_DescriptorProto** msgs;
|
1524
|
+
google_protobuf_EnumDescriptorProto** enums;
|
1525
|
+
|
1526
|
+
if (google_protobuf_FileDescriptorProto_has_package(file_proto)) {
|
1527
|
+
upb_strview package_str =
|
1528
|
+
google_protobuf_FileDescriptorProto_package(file_proto);
|
1529
|
+
package = rb_str_new(package_str.data, package_str.size);
|
1530
|
+
}
|
1531
|
+
|
1532
|
+
msgs = google_protobuf_FileDescriptorProto_mutable_message_type(file_proto,
|
1533
|
+
&msg_count);
|
1534
|
+
for (i = 0; i < msg_count; i++) {
|
1535
|
+
upb_strview name = google_protobuf_DescriptorProto_name(msgs[i]);
|
1536
|
+
rb_ary_push(msg_names, rb_str_new(name.data, name.size));
|
1537
|
+
}
|
1538
|
+
|
1539
|
+
enums = google_protobuf_FileDescriptorProto_mutable_enum_type(file_proto,
|
1540
|
+
&enum_count);
|
1541
|
+
for (i = 0; i < enum_count; i++) {
|
1542
|
+
upb_strview name = google_protobuf_EnumDescriptorProto_name(enums[i]);
|
1543
|
+
rb_ary_push(enum_names, rb_str_new(name.data, name.size));
|
1544
|
+
}
|
1545
|
+
|
1546
|
+
{
|
1547
|
+
// Call Ruby code to calculate package name and nesting.
|
1548
|
+
VALUE args[3] = { package, msg_names, enum_names };
|
1549
|
+
VALUE internal = rb_eval_string("Google::Protobuf::Internal");
|
1550
|
+
VALUE ret = rb_funcallv(internal, rb_intern("fixup_descriptor"), 3, args);
|
1551
|
+
|
1552
|
+
new_package = rb_ary_entry(ret, 0);
|
1553
|
+
nesting = rb_ary_entry(ret, 1);
|
1554
|
+
}
|
1555
|
+
|
1556
|
+
// Rewrite package and names.
|
1557
|
+
if (new_package != Qnil) {
|
1558
|
+
upb_strview new_package_str =
|
1559
|
+
FileBuilderContext_strdup(_file_builder, new_package);
|
1560
|
+
google_protobuf_FileDescriptorProto_set_package(file_proto,
|
1561
|
+
new_package_str);
|
1562
|
+
}
|
1563
|
+
|
1564
|
+
for (i = 0; i < msg_count; i++) {
|
1565
|
+
upb_strview name = google_protobuf_DescriptorProto_name(msgs[i]);
|
1566
|
+
remove_path(&name);
|
1567
|
+
google_protobuf_DescriptorProto_set_name(msgs[i], name);
|
1568
|
+
}
|
1569
|
+
|
1570
|
+
for (i = 0; i < enum_count; i++) {
|
1571
|
+
upb_strview name = google_protobuf_EnumDescriptorProto_name(enums[i]);
|
1572
|
+
remove_path(&name);
|
1573
|
+
google_protobuf_EnumDescriptorProto_set_name(enums[i], name);
|
1574
|
+
}
|
1575
|
+
|
1576
|
+
// Rewrite nesting.
|
1577
|
+
msg_ents = rb_hash_aref(nesting, ID2SYM(rb_intern("msgs")));
|
1578
|
+
enum_ents = rb_hash_aref(nesting, ID2SYM(rb_intern("enums")));
|
1579
|
+
|
1580
|
+
Check_Type(msg_ents, T_ARRAY);
|
1581
|
+
Check_Type(enum_ents, T_ARRAY);
|
1582
|
+
|
1583
|
+
for (i = 0; i < (size_t)RARRAY_LEN(msg_ents); i++) {
|
1584
|
+
VALUE msg_ent = rb_ary_entry(msg_ents, i);
|
1585
|
+
VALUE pos = rb_hash_aref(msg_ent, ID2SYM(rb_intern("pos")));
|
1586
|
+
msgs[i] = msgs[NUM2INT(pos)];
|
1587
|
+
rewrite_nesting(msg_ent, msgs[i], msgs, enums, arena);
|
1588
|
+
}
|
1589
|
+
|
1590
|
+
for (i = 0; i < (size_t)RARRAY_LEN(enum_ents); i++) {
|
1591
|
+
VALUE enum_pos = rb_ary_entry(enum_ents, i);
|
1592
|
+
enums[i] = enums[NUM2INT(enum_pos)];
|
1593
|
+
}
|
1594
|
+
|
1595
|
+
google_protobuf_FileDescriptorProto_resize_message_type(
|
1596
|
+
file_proto, RARRAY_LEN(msg_ents), arena);
|
1597
|
+
google_protobuf_FileDescriptorProto_resize_enum_type(
|
1598
|
+
file_proto, RARRAY_LEN(enum_ents), arena);
|
1599
|
+
}
|
1600
|
+
|
1601
|
+
/*
|
1602
|
+
* call-seq:
|
1603
|
+
* FileBuilderContext.add_enum(name, &block)
|
1604
|
+
*
|
1605
|
+
* Creates a new, empty enum descriptor with the given name, and invokes the
|
1606
|
+
* block in the context of an EnumBuilderContext on that descriptor. The block
|
1607
|
+
* can then call EnumBuilderContext#add_value to define the enum values.
|
1608
|
+
*
|
1609
|
+
* This is the recommended, idiomatic way to build enum definitions.
|
1610
|
+
*/
|
1611
|
+
static VALUE FileBuilderContext_add_enum(VALUE _self, VALUE name) {
|
1612
|
+
VALUE args[2] = { _self, name };
|
1613
|
+
VALUE ctx = rb_class_new_instance(2, args, cEnumBuilderContext);
|
1614
|
+
VALUE block = rb_block_proc();
|
1615
|
+
rb_funcall_with_block(ctx, rb_intern("instance_eval"), 0, NULL, block);
|
1616
|
+
return Qnil;
|
1617
|
+
}
|
1618
|
+
|
1619
|
+
static void FileBuilderContext_build(VALUE _self) {
|
1620
|
+
FileBuilderContext* self = ruby_to_FileBuilderContext(_self);
|
1621
|
+
DescriptorPool* pool = ruby_to_DescriptorPool(self->descriptor_pool);
|
1622
|
+
upb_status status;
|
1623
|
+
|
1624
|
+
rewrite_enum_defaults(pool->symtab, self->file_proto);
|
1625
|
+
rewrite_names(_self, self->file_proto);
|
1626
|
+
|
1627
|
+
upb_status_clear(&status);
|
1628
|
+
if (!upb_symtab_addfile(pool->symtab, self->file_proto, &status)) {
|
1629
|
+
rb_raise(cTypeError, "Unable to add defs to DescriptorPool: %s",
|
1630
|
+
upb_status_errmsg(&status));
|
1631
|
+
}
|
1632
|
+
}
|
1633
|
+
|
1634
|
+
static void FileBuilderContext_register(VALUE module) {
|
1635
|
+
VALUE klass = rb_define_class_under(module, "FileBuilderContext", rb_cObject);
|
1636
|
+
rb_define_alloc_func(klass, FileBuilderContext_alloc);
|
1637
|
+
rb_define_method(klass, "initialize", FileBuilderContext_initialize, 3);
|
1638
|
+
rb_define_method(klass, "add_message", FileBuilderContext_add_message, 1);
|
1639
|
+
rb_define_method(klass, "add_enum", FileBuilderContext_add_enum, 1);
|
1640
|
+
rb_gc_register_address(&cFileBuilderContext);
|
1641
|
+
cFileBuilderContext = klass;
|
1642
|
+
}
|
1643
|
+
|
1405
1644
|
// -----------------------------------------------------------------------------
|
1406
1645
|
// MessageBuilderContext.
|
1407
1646
|
// -----------------------------------------------------------------------------
|
1408
1647
|
|
1409
|
-
|
1410
|
-
|
1648
|
+
typedef struct {
|
1649
|
+
google_protobuf_DescriptorProto* msg_proto;
|
1650
|
+
VALUE file_builder;
|
1651
|
+
} MessageBuilderContext;
|
1652
|
+
|
1653
|
+
static VALUE cMessageBuilderContext = Qnil;
|
1411
1654
|
|
1412
|
-
void MessageBuilderContext_mark(void* _self) {
|
1655
|
+
static void MessageBuilderContext_mark(void* _self) {
|
1413
1656
|
MessageBuilderContext* self = _self;
|
1414
1657
|
rb_gc_mark(self->file_builder);
|
1415
1658
|
}
|
1416
1659
|
|
1417
|
-
|
1418
|
-
MessageBuilderContext
|
1419
|
-
|
1420
|
-
|
1660
|
+
static const rb_data_type_t MessageBuilderContext_type = {
|
1661
|
+
"Google::Protobuf::Internal::MessageBuilderContext",
|
1662
|
+
{MessageBuilderContext_mark, RUBY_DEFAULT_FREE, NULL},
|
1663
|
+
.flags = RUBY_TYPED_FREE_IMMEDIATELY,
|
1664
|
+
};
|
1421
1665
|
|
1422
|
-
|
1423
|
-
MessageBuilderContext*
|
1424
|
-
|
1425
|
-
|
1426
|
-
self->file_builder = Qnil;
|
1666
|
+
static MessageBuilderContext* ruby_to_MessageBuilderContext(VALUE val) {
|
1667
|
+
MessageBuilderContext* ret;
|
1668
|
+
TypedData_Get_Struct(val, MessageBuilderContext, &MessageBuilderContext_type,
|
1669
|
+
ret);
|
1427
1670
|
return ret;
|
1428
1671
|
}
|
1429
1672
|
|
1430
|
-
|
1431
|
-
|
1432
|
-
|
1433
|
-
|
1434
|
-
|
1435
|
-
MessageBuilderContext_initialize, 2);
|
1436
|
-
rb_define_method(klass, "optional", MessageBuilderContext_optional, -1);
|
1437
|
-
rb_define_method(klass, "proto3_optional", MessageBuilderContext_proto3_optional, -1);
|
1438
|
-
rb_define_method(klass, "required", MessageBuilderContext_required, -1);
|
1439
|
-
rb_define_method(klass, "repeated", MessageBuilderContext_repeated, -1);
|
1440
|
-
rb_define_method(klass, "map", MessageBuilderContext_map, -1);
|
1441
|
-
rb_define_method(klass, "oneof", MessageBuilderContext_oneof, 1);
|
1442
|
-
rb_gc_register_address(&cMessageBuilderContext);
|
1443
|
-
cMessageBuilderContext = klass;
|
1673
|
+
static VALUE MessageBuilderContext_alloc(VALUE klass) {
|
1674
|
+
MessageBuilderContext* self = ALLOC(MessageBuilderContext);
|
1675
|
+
VALUE ret = TypedData_Wrap_Struct(klass, &MessageBuilderContext_type, self);
|
1676
|
+
self->file_builder = Qnil;
|
1677
|
+
return ret;
|
1444
1678
|
}
|
1445
1679
|
|
1446
1680
|
/*
|
@@ -1451,10 +1685,9 @@ void MessageBuilderContext_register(VALUE module) {
|
|
1451
1685
|
* builder context. This class is intended to serve as a DSL context to be used
|
1452
1686
|
* with #instance_eval.
|
1453
1687
|
*/
|
1454
|
-
VALUE MessageBuilderContext_initialize(VALUE _self,
|
1455
|
-
|
1456
|
-
|
1457
|
-
DEFINE_SELF(MessageBuilderContext, self, _self);
|
1688
|
+
static VALUE MessageBuilderContext_initialize(VALUE _self, VALUE _file_builder,
|
1689
|
+
VALUE name) {
|
1690
|
+
MessageBuilderContext* self = ruby_to_MessageBuilderContext(_self);
|
1458
1691
|
FileBuilderContext* file_builder = ruby_to_FileBuilderContext(_file_builder);
|
1459
1692
|
google_protobuf_FileDescriptorProto* file_proto = file_builder->file_proto;
|
1460
1693
|
|
@@ -1472,7 +1705,7 @@ static void msgdef_add_field(VALUE msgbuilder_rb, upb_label_t label, VALUE name,
|
|
1472
1705
|
VALUE type, VALUE number, VALUE type_class,
|
1473
1706
|
VALUE options, int oneof_index,
|
1474
1707
|
bool proto3_optional) {
|
1475
|
-
|
1708
|
+
MessageBuilderContext* self = ruby_to_MessageBuilderContext(msgbuilder_rb);
|
1476
1709
|
FileBuilderContext* file_context =
|
1477
1710
|
ruby_to_FileBuilderContext(self->file_builder);
|
1478
1711
|
google_protobuf_FieldDescriptorProto* field_proto;
|
@@ -1527,9 +1760,16 @@ static void msgdef_add_field(VALUE msgbuilder_rb, upb_label_t label, VALUE name,
|
|
1527
1760
|
}
|
1528
1761
|
}
|
1529
1762
|
|
1763
|
+
#if RUBY_API_VERSION_CODE >= 20700
|
1764
|
+
static VALUE make_mapentry(VALUE _message_builder, VALUE types, int argc,
|
1765
|
+
const VALUE* argv, VALUE blockarg) {
|
1766
|
+
(void)blockarg;
|
1767
|
+
#else
|
1530
1768
|
static VALUE make_mapentry(VALUE _message_builder, VALUE types, int argc,
|
1531
1769
|
VALUE* argv) {
|
1532
|
-
|
1770
|
+
#endif
|
1771
|
+
MessageBuilderContext* message_builder =
|
1772
|
+
ruby_to_MessageBuilderContext(_message_builder);
|
1533
1773
|
VALUE type_class = rb_ary_entry(types, 2);
|
1534
1774
|
FileBuilderContext* file_context =
|
1535
1775
|
ruby_to_FileBuilderContext(message_builder->file_builder);
|
@@ -1596,8 +1836,8 @@ VALUE MessageBuilderContext_optional(int argc, VALUE* argv, VALUE _self) {
|
|
1596
1836
|
* FieldDescriptor#type=) and the type_class must be a string, if present (as
|
1597
1837
|
* accepted by FieldDescriptor#submsg_name=).
|
1598
1838
|
*/
|
1599
|
-
VALUE MessageBuilderContext_proto3_optional(int argc, VALUE* argv,
|
1600
|
-
|
1839
|
+
static VALUE MessageBuilderContext_proto3_optional(int argc, VALUE* argv,
|
1840
|
+
VALUE _self) {
|
1601
1841
|
VALUE name, type, number;
|
1602
1842
|
VALUE type_class, options = Qnil;
|
1603
1843
|
|
@@ -1630,7 +1870,8 @@ VALUE MessageBuilderContext_proto3_optional(int argc, VALUE* argv,
|
|
1630
1870
|
* completeness. Any attempt to add a message type with required fields to a
|
1631
1871
|
* pool will currently result in an error.
|
1632
1872
|
*/
|
1633
|
-
VALUE MessageBuilderContext_required(int argc, VALUE* argv,
|
1873
|
+
static VALUE MessageBuilderContext_required(int argc, VALUE* argv,
|
1874
|
+
VALUE _self) {
|
1634
1875
|
VALUE name, type, number;
|
1635
1876
|
VALUE type_class, options = Qnil;
|
1636
1877
|
|
@@ -1658,7 +1899,8 @@ VALUE MessageBuilderContext_required(int argc, VALUE* argv, VALUE _self) {
|
|
1658
1899
|
* symbol (as accepted by FieldDescriptor#type=) and the type_class must be a
|
1659
1900
|
* string, if present (as accepted by FieldDescriptor#submsg_name=).
|
1660
1901
|
*/
|
1661
|
-
VALUE MessageBuilderContext_repeated(int argc, VALUE* argv,
|
1902
|
+
static VALUE MessageBuilderContext_repeated(int argc, VALUE* argv,
|
1903
|
+
VALUE _self) {
|
1662
1904
|
VALUE name, type, number, type_class;
|
1663
1905
|
|
1664
1906
|
if (argc < 3) {
|
@@ -1687,8 +1929,8 @@ VALUE MessageBuilderContext_repeated(int argc, VALUE* argv, VALUE _self) {
|
|
1687
1929
|
* type_class must be a string, if present (as accepted by
|
1688
1930
|
* FieldDescriptor#submsg_name=).
|
1689
1931
|
*/
|
1690
|
-
VALUE MessageBuilderContext_map(int argc, VALUE* argv, VALUE _self) {
|
1691
|
-
|
1932
|
+
static VALUE MessageBuilderContext_map(int argc, VALUE* argv, VALUE _self) {
|
1933
|
+
MessageBuilderContext* self = ruby_to_MessageBuilderContext(_self);
|
1692
1934
|
VALUE name, key_type, value_type, number, type_class;
|
1693
1935
|
VALUE mapentry_desc_name;
|
1694
1936
|
FileBuilderContext* file_builder;
|
@@ -1717,14 +1959,6 @@ VALUE MessageBuilderContext_map(int argc, VALUE* argv, VALUE _self) {
|
|
1717
1959
|
|
1718
1960
|
file_builder = ruby_to_FileBuilderContext(self->file_builder);
|
1719
1961
|
|
1720
|
-
// TODO(haberman): remove this restriction, maps are supported in proto2.
|
1721
|
-
if (upb_strview_eql(
|
1722
|
-
google_protobuf_FileDescriptorProto_syntax(file_builder->file_proto),
|
1723
|
-
upb_strview_makez("proto2"))) {
|
1724
|
-
rb_raise(rb_eArgError,
|
1725
|
-
"Cannot add a native map field using proto2 syntax.");
|
1726
|
-
}
|
1727
|
-
|
1728
1962
|
// Create a new message descriptor for the map entry message, and create a
|
1729
1963
|
// repeated submessage field here with that type.
|
1730
1964
|
msg_name = google_protobuf_DescriptorProto_name(self->msg_proto);
|
@@ -1768,8 +2002,8 @@ VALUE MessageBuilderContext_map(int argc, VALUE* argv, VALUE _self) {
|
|
1768
2002
|
*
|
1769
2003
|
* This is the recommended, idiomatic way to build oneof definitions.
|
1770
2004
|
*/
|
1771
|
-
VALUE MessageBuilderContext_oneof(VALUE _self, VALUE name) {
|
1772
|
-
|
2005
|
+
static VALUE MessageBuilderContext_oneof(VALUE _self, VALUE name) {
|
2006
|
+
MessageBuilderContext* self = ruby_to_MessageBuilderContext(_self);
|
1773
2007
|
size_t oneof_count;
|
1774
2008
|
FileBuilderContext* file_context =
|
1775
2009
|
ruby_to_FileBuilderContext(self->file_builder);
|
@@ -1795,8 +2029,8 @@ VALUE MessageBuilderContext_oneof(VALUE _self, VALUE name) {
|
|
1795
2029
|
return Qnil;
|
1796
2030
|
}
|
1797
2031
|
|
1798
|
-
void MessageBuilderContext_add_synthetic_oneofs(VALUE _self) {
|
1799
|
-
|
2032
|
+
static void MessageBuilderContext_add_synthetic_oneofs(VALUE _self) {
|
2033
|
+
MessageBuilderContext* self = ruby_to_MessageBuilderContext(_self);
|
1800
2034
|
FileBuilderContext* file_context =
|
1801
2035
|
ruby_to_FileBuilderContext(self->file_builder);
|
1802
2036
|
size_t field_count, oneof_count;
|
@@ -1845,42 +2079,59 @@ void MessageBuilderContext_add_synthetic_oneofs(VALUE _self) {
|
|
1845
2079
|
}
|
1846
2080
|
}
|
1847
2081
|
|
2082
|
+
static void MessageBuilderContext_register(VALUE module) {
|
2083
|
+
VALUE klass = rb_define_class_under(
|
2084
|
+
module, "MessageBuilderContext", rb_cObject);
|
2085
|
+
rb_define_alloc_func(klass, MessageBuilderContext_alloc);
|
2086
|
+
rb_define_method(klass, "initialize",
|
2087
|
+
MessageBuilderContext_initialize, 2);
|
2088
|
+
rb_define_method(klass, "optional", MessageBuilderContext_optional, -1);
|
2089
|
+
rb_define_method(klass, "proto3_optional", MessageBuilderContext_proto3_optional, -1);
|
2090
|
+
rb_define_method(klass, "required", MessageBuilderContext_required, -1);
|
2091
|
+
rb_define_method(klass, "repeated", MessageBuilderContext_repeated, -1);
|
2092
|
+
rb_define_method(klass, "map", MessageBuilderContext_map, -1);
|
2093
|
+
rb_define_method(klass, "oneof", MessageBuilderContext_oneof, 1);
|
2094
|
+
rb_gc_register_address(&cMessageBuilderContext);
|
2095
|
+
cMessageBuilderContext = klass;
|
2096
|
+
}
|
2097
|
+
|
1848
2098
|
// -----------------------------------------------------------------------------
|
1849
2099
|
// OneofBuilderContext.
|
1850
2100
|
// -----------------------------------------------------------------------------
|
1851
2101
|
|
1852
|
-
|
1853
|
-
|
2102
|
+
typedef struct {
|
2103
|
+
int oneof_index;
|
2104
|
+
VALUE message_builder;
|
2105
|
+
} OneofBuilderContext;
|
2106
|
+
|
2107
|
+
static VALUE cOneofBuilderContext = Qnil;
|
1854
2108
|
|
1855
2109
|
void OneofBuilderContext_mark(void* _self) {
|
1856
2110
|
OneofBuilderContext* self = _self;
|
1857
2111
|
rb_gc_mark(self->message_builder);
|
1858
2112
|
}
|
1859
2113
|
|
1860
|
-
|
1861
|
-
|
2114
|
+
static const rb_data_type_t OneofBuilderContext_type = {
|
2115
|
+
"Google::Protobuf::Internal::OneofBuilderContext",
|
2116
|
+
{OneofBuilderContext_mark, RUBY_DEFAULT_FREE, NULL},
|
2117
|
+
.flags = RUBY_TYPED_FREE_IMMEDIATELY,
|
2118
|
+
};
|
2119
|
+
|
2120
|
+
static OneofBuilderContext* ruby_to_OneofBuilderContext(VALUE val) {
|
2121
|
+
OneofBuilderContext* ret;
|
2122
|
+
TypedData_Get_Struct(val, OneofBuilderContext, &OneofBuilderContext_type,
|
2123
|
+
ret);
|
2124
|
+
return ret;
|
1862
2125
|
}
|
1863
2126
|
|
1864
|
-
VALUE OneofBuilderContext_alloc(VALUE klass) {
|
2127
|
+
static VALUE OneofBuilderContext_alloc(VALUE klass) {
|
1865
2128
|
OneofBuilderContext* self = ALLOC(OneofBuilderContext);
|
1866
|
-
VALUE ret = TypedData_Wrap_Struct(
|
1867
|
-
klass, &_OneofBuilderContext_type, self);
|
2129
|
+
VALUE ret = TypedData_Wrap_Struct(klass, &OneofBuilderContext_type, self);
|
1868
2130
|
self->oneof_index = 0;
|
1869
2131
|
self->message_builder = Qnil;
|
1870
2132
|
return ret;
|
1871
2133
|
}
|
1872
2134
|
|
1873
|
-
void OneofBuilderContext_register(VALUE module) {
|
1874
|
-
VALUE klass = rb_define_class_under(
|
1875
|
-
module, "OneofBuilderContext", rb_cObject);
|
1876
|
-
rb_define_alloc_func(klass, OneofBuilderContext_alloc);
|
1877
|
-
rb_define_method(klass, "initialize",
|
1878
|
-
OneofBuilderContext_initialize, 2);
|
1879
|
-
rb_define_method(klass, "optional", OneofBuilderContext_optional, -1);
|
1880
|
-
rb_gc_register_address(&cOneofBuilderContext);
|
1881
|
-
cOneofBuilderContext = klass;
|
1882
|
-
}
|
1883
|
-
|
1884
2135
|
/*
|
1885
2136
|
* call-seq:
|
1886
2137
|
* OneofBuilderContext.new(oneof_index, message_builder) => context
|
@@ -1889,10 +2140,9 @@ void OneofBuilderContext_register(VALUE module) {
|
|
1889
2140
|
* builder context. This class is intended to serve as a DSL context to be used
|
1890
2141
|
* with #instance_eval.
|
1891
2142
|
*/
|
1892
|
-
VALUE OneofBuilderContext_initialize(VALUE _self,
|
1893
|
-
|
1894
|
-
|
1895
|
-
DEFINE_SELF(OneofBuilderContext, self, _self);
|
2143
|
+
static VALUE OneofBuilderContext_initialize(VALUE _self, VALUE oneof_index,
|
2144
|
+
VALUE message_builder) {
|
2145
|
+
OneofBuilderContext* self = ruby_to_OneofBuilderContext(_self);
|
1896
2146
|
self->oneof_index = NUM2INT(oneof_index);
|
1897
2147
|
self->message_builder = message_builder;
|
1898
2148
|
return Qnil;
|
@@ -1908,8 +2158,8 @@ VALUE OneofBuilderContext_initialize(VALUE _self,
|
|
1908
2158
|
* (as accepted by FieldDescriptor#type=) and the type_class must be a string,
|
1909
2159
|
* if present (as accepted by FieldDescriptor#submsg_name=).
|
1910
2160
|
*/
|
1911
|
-
VALUE OneofBuilderContext_optional(int argc, VALUE* argv, VALUE _self) {
|
1912
|
-
|
2161
|
+
static VALUE OneofBuilderContext_optional(int argc, VALUE* argv, VALUE _self) {
|
2162
|
+
OneofBuilderContext* self = ruby_to_OneofBuilderContext(_self);
|
1913
2163
|
VALUE name, type, number;
|
1914
2164
|
VALUE type_class, options = Qnil;
|
1915
2165
|
|
@@ -1921,41 +2171,53 @@ VALUE OneofBuilderContext_optional(int argc, VALUE* argv, VALUE _self) {
|
|
1921
2171
|
return Qnil;
|
1922
2172
|
}
|
1923
2173
|
|
2174
|
+
static void OneofBuilderContext_register(VALUE module) {
|
2175
|
+
VALUE klass = rb_define_class_under(
|
2176
|
+
module, "OneofBuilderContext", rb_cObject);
|
2177
|
+
rb_define_alloc_func(klass, OneofBuilderContext_alloc);
|
2178
|
+
rb_define_method(klass, "initialize",
|
2179
|
+
OneofBuilderContext_initialize, 2);
|
2180
|
+
rb_define_method(klass, "optional", OneofBuilderContext_optional, -1);
|
2181
|
+
rb_gc_register_address(&cOneofBuilderContext);
|
2182
|
+
cOneofBuilderContext = klass;
|
2183
|
+
}
|
2184
|
+
|
1924
2185
|
// -----------------------------------------------------------------------------
|
1925
2186
|
// EnumBuilderContext.
|
1926
2187
|
// -----------------------------------------------------------------------------
|
1927
2188
|
|
1928
|
-
|
1929
|
-
|
2189
|
+
typedef struct {
|
2190
|
+
google_protobuf_EnumDescriptorProto* enum_proto;
|
2191
|
+
VALUE file_builder;
|
2192
|
+
} EnumBuilderContext;
|
2193
|
+
|
2194
|
+
static VALUE cEnumBuilderContext = Qnil;
|
1930
2195
|
|
1931
2196
|
void EnumBuilderContext_mark(void* _self) {
|
1932
2197
|
EnumBuilderContext* self = _self;
|
1933
2198
|
rb_gc_mark(self->file_builder);
|
1934
2199
|
}
|
1935
2200
|
|
1936
|
-
|
1937
|
-
|
2201
|
+
static const rb_data_type_t EnumBuilderContext_type = {
|
2202
|
+
"Google::Protobuf::Internal::EnumBuilderContext",
|
2203
|
+
{EnumBuilderContext_mark, RUBY_DEFAULT_FREE, NULL},
|
2204
|
+
.flags = RUBY_TYPED_FREE_IMMEDIATELY,
|
2205
|
+
};
|
2206
|
+
|
2207
|
+
static EnumBuilderContext* ruby_to_EnumBuilderContext(VALUE val) {
|
2208
|
+
EnumBuilderContext* ret;
|
2209
|
+
TypedData_Get_Struct(val, EnumBuilderContext, &EnumBuilderContext_type, ret);
|
2210
|
+
return ret;
|
1938
2211
|
}
|
1939
2212
|
|
1940
|
-
VALUE EnumBuilderContext_alloc(VALUE klass) {
|
2213
|
+
static VALUE EnumBuilderContext_alloc(VALUE klass) {
|
1941
2214
|
EnumBuilderContext* self = ALLOC(EnumBuilderContext);
|
1942
|
-
VALUE ret = TypedData_Wrap_Struct(
|
1943
|
-
klass, &_EnumBuilderContext_type, self);
|
2215
|
+
VALUE ret = TypedData_Wrap_Struct(klass, &EnumBuilderContext_type, self);
|
1944
2216
|
self->enum_proto = NULL;
|
1945
2217
|
self->file_builder = Qnil;
|
1946
2218
|
return ret;
|
1947
2219
|
}
|
1948
2220
|
|
1949
|
-
void EnumBuilderContext_register(VALUE module) {
|
1950
|
-
VALUE klass = rb_define_class_under(
|
1951
|
-
module, "EnumBuilderContext", rb_cObject);
|
1952
|
-
rb_define_alloc_func(klass, EnumBuilderContext_alloc);
|
1953
|
-
rb_define_method(klass, "initialize", EnumBuilderContext_initialize, 2);
|
1954
|
-
rb_define_method(klass, "value", EnumBuilderContext_value, 2);
|
1955
|
-
rb_gc_register_address(&cEnumBuilderContext);
|
1956
|
-
cEnumBuilderContext = klass;
|
1957
|
-
}
|
1958
|
-
|
1959
2221
|
/*
|
1960
2222
|
* call-seq:
|
1961
2223
|
* EnumBuilderContext.new(file_builder) => context
|
@@ -1963,9 +2225,9 @@ void EnumBuilderContext_register(VALUE module) {
|
|
1963
2225
|
* Create a new builder context around the given enum descriptor. This class is
|
1964
2226
|
* intended to serve as a DSL context to be used with #instance_eval.
|
1965
2227
|
*/
|
1966
|
-
VALUE EnumBuilderContext_initialize(VALUE _self, VALUE _file_builder,
|
1967
|
-
|
1968
|
-
|
2228
|
+
static VALUE EnumBuilderContext_initialize(VALUE _self, VALUE _file_builder,
|
2229
|
+
VALUE name) {
|
2230
|
+
EnumBuilderContext* self = ruby_to_EnumBuilderContext(_self);
|
1969
2231
|
FileBuilderContext* file_builder = ruby_to_FileBuilderContext(_file_builder);
|
1970
2232
|
google_protobuf_FileDescriptorProto* file_proto = file_builder->file_proto;
|
1971
2233
|
|
@@ -1986,8 +2248,8 @@ VALUE EnumBuilderContext_initialize(VALUE _self, VALUE _file_builder,
|
|
1986
2248
|
* Adds the given name => number mapping to the enum type. Name must be a Ruby
|
1987
2249
|
* symbol.
|
1988
2250
|
*/
|
1989
|
-
VALUE EnumBuilderContext_value(VALUE _self, VALUE name, VALUE number) {
|
1990
|
-
|
2251
|
+
static VALUE EnumBuilderContext_value(VALUE _self, VALUE name, VALUE number) {
|
2252
|
+
EnumBuilderContext* self = ruby_to_EnumBuilderContext(_self);
|
1991
2253
|
FileBuilderContext* file_builder =
|
1992
2254
|
ruby_to_FileBuilderContext(self->file_builder);
|
1993
2255
|
google_protobuf_EnumValueDescriptorProto* enum_value;
|
@@ -2003,196 +2265,53 @@ VALUE EnumBuilderContext_value(VALUE _self, VALUE name, VALUE number) {
|
|
2003
2265
|
return Qnil;
|
2004
2266
|
}
|
2005
2267
|
|
2006
|
-
|
2007
|
-
|
2008
|
-
|
2009
|
-
|
2010
|
-
|
2011
|
-
|
2012
|
-
|
2013
|
-
|
2014
|
-
void FileBuilderContext_mark(void* _self) {
|
2015
|
-
FileBuilderContext* self = _self;
|
2016
|
-
rb_gc_mark(self->descriptor_pool);
|
2017
|
-
}
|
2018
|
-
|
2019
|
-
void FileBuilderContext_free(void* _self) {
|
2020
|
-
FileBuilderContext* self = _self;
|
2021
|
-
upb_arena_free(self->arena);
|
2022
|
-
xfree(self);
|
2023
|
-
}
|
2024
|
-
|
2025
|
-
upb_strview FileBuilderContext_strdup2(VALUE _self, const char *str) {
|
2026
|
-
DEFINE_SELF(FileBuilderContext, self, _self);
|
2027
|
-
upb_strview ret;
|
2028
|
-
char *data;
|
2029
|
-
|
2030
|
-
ret.size = strlen(str);
|
2031
|
-
data = upb_malloc(upb_arena_alloc(self->arena), ret.size + 1);
|
2032
|
-
ret.data = data;
|
2033
|
-
memcpy(data, str, ret.size);
|
2034
|
-
/* Null-terminate required by rewrite_enum_defaults() above. */
|
2035
|
-
data[ret.size] = '\0';
|
2036
|
-
return ret;
|
2037
|
-
}
|
2038
|
-
|
2039
|
-
upb_strview FileBuilderContext_strdup(VALUE _self, VALUE rb_str) {
|
2040
|
-
return FileBuilderContext_strdup2(_self, get_str(rb_str));
|
2041
|
-
}
|
2042
|
-
|
2043
|
-
upb_strview FileBuilderContext_strdup_sym(VALUE _self, VALUE rb_sym) {
|
2044
|
-
Check_Type(rb_sym, T_SYMBOL);
|
2045
|
-
return FileBuilderContext_strdup(_self, rb_id2str(SYM2ID(rb_sym)));
|
2046
|
-
}
|
2047
|
-
|
2048
|
-
VALUE FileBuilderContext_alloc(VALUE klass) {
|
2049
|
-
FileBuilderContext* self = ALLOC(FileBuilderContext);
|
2050
|
-
VALUE ret = TypedData_Wrap_Struct(klass, &_FileBuilderContext_type, self);
|
2051
|
-
self->arena = upb_arena_new();
|
2052
|
-
self->file_proto = google_protobuf_FileDescriptorProto_new(self->arena);
|
2053
|
-
self->descriptor_pool = Qnil;
|
2054
|
-
return ret;
|
2055
|
-
}
|
2056
|
-
|
2057
|
-
void FileBuilderContext_register(VALUE module) {
|
2058
|
-
VALUE klass = rb_define_class_under(module, "FileBuilderContext", rb_cObject);
|
2059
|
-
rb_define_alloc_func(klass, FileBuilderContext_alloc);
|
2060
|
-
rb_define_method(klass, "initialize", FileBuilderContext_initialize, 3);
|
2061
|
-
rb_define_method(klass, "add_message", FileBuilderContext_add_message, 1);
|
2062
|
-
rb_define_method(klass, "add_enum", FileBuilderContext_add_enum, 1);
|
2063
|
-
rb_gc_register_address(&cFileBuilderContext);
|
2064
|
-
cFileBuilderContext = klass;
|
2065
|
-
}
|
2066
|
-
|
2067
|
-
/*
|
2068
|
-
* call-seq:
|
2069
|
-
* FileBuilderContext.new(descriptor_pool) => context
|
2070
|
-
*
|
2071
|
-
* Create a new file builder context for the given file descriptor and
|
2072
|
-
* builder context. This class is intended to serve as a DSL context to be used
|
2073
|
-
* with #instance_eval.
|
2074
|
-
*/
|
2075
|
-
VALUE FileBuilderContext_initialize(VALUE _self, VALUE descriptor_pool,
|
2076
|
-
VALUE name, VALUE options) {
|
2077
|
-
DEFINE_SELF(FileBuilderContext, self, _self);
|
2078
|
-
self->descriptor_pool = descriptor_pool;
|
2079
|
-
|
2080
|
-
google_protobuf_FileDescriptorProto_set_name(
|
2081
|
-
self->file_proto, FileBuilderContext_strdup(_self, name));
|
2082
|
-
|
2083
|
-
// Default syntax for Ruby is proto3.
|
2084
|
-
google_protobuf_FileDescriptorProto_set_syntax(
|
2085
|
-
self->file_proto,
|
2086
|
-
FileBuilderContext_strdup(_self, rb_str_new2("proto3")));
|
2087
|
-
|
2088
|
-
if (options != Qnil) {
|
2089
|
-
VALUE syntax;
|
2090
|
-
|
2091
|
-
Check_Type(options, T_HASH);
|
2092
|
-
syntax = rb_hash_lookup2(options, ID2SYM(rb_intern("syntax")), Qnil);
|
2093
|
-
|
2094
|
-
if (syntax != Qnil) {
|
2095
|
-
VALUE syntax_str;
|
2096
|
-
|
2097
|
-
Check_Type(syntax, T_SYMBOL);
|
2098
|
-
syntax_str = rb_id2str(SYM2ID(syntax));
|
2099
|
-
google_protobuf_FileDescriptorProto_set_syntax(
|
2100
|
-
self->file_proto, FileBuilderContext_strdup(_self, syntax_str));
|
2101
|
-
}
|
2102
|
-
}
|
2103
|
-
|
2104
|
-
return Qnil;
|
2105
|
-
}
|
2106
|
-
|
2107
|
-
/*
|
2108
|
-
* call-seq:
|
2109
|
-
* FileBuilderContext.add_message(name, &block)
|
2110
|
-
*
|
2111
|
-
* Creates a new, empty descriptor with the given name, and invokes the block in
|
2112
|
-
* the context of a MessageBuilderContext on that descriptor. The block can then
|
2113
|
-
* call, e.g., MessageBuilderContext#optional and MessageBuilderContext#repeated
|
2114
|
-
* methods to define the message fields.
|
2115
|
-
*
|
2116
|
-
* This is the recommended, idiomatic way to build message definitions.
|
2117
|
-
*/
|
2118
|
-
VALUE FileBuilderContext_add_message(VALUE _self, VALUE name) {
|
2119
|
-
VALUE args[2] = { _self, name };
|
2120
|
-
VALUE ctx = rb_class_new_instance(2, args, cMessageBuilderContext);
|
2121
|
-
VALUE block = rb_block_proc();
|
2122
|
-
rb_funcall_with_block(ctx, rb_intern("instance_eval"), 0, NULL, block);
|
2123
|
-
MessageBuilderContext_add_synthetic_oneofs(ctx);
|
2124
|
-
return Qnil;
|
2125
|
-
}
|
2126
|
-
|
2127
|
-
/*
|
2128
|
-
* call-seq:
|
2129
|
-
* FileBuilderContext.add_enum(name, &block)
|
2130
|
-
*
|
2131
|
-
* Creates a new, empty enum descriptor with the given name, and invokes the
|
2132
|
-
* block in the context of an EnumBuilderContext on that descriptor. The block
|
2133
|
-
* can then call EnumBuilderContext#add_value to define the enum values.
|
2134
|
-
*
|
2135
|
-
* This is the recommended, idiomatic way to build enum definitions.
|
2136
|
-
*/
|
2137
|
-
VALUE FileBuilderContext_add_enum(VALUE _self, VALUE name) {
|
2138
|
-
VALUE args[2] = { _self, name };
|
2139
|
-
VALUE ctx = rb_class_new_instance(2, args, cEnumBuilderContext);
|
2140
|
-
VALUE block = rb_block_proc();
|
2141
|
-
rb_funcall_with_block(ctx, rb_intern("instance_eval"), 0, NULL, block);
|
2142
|
-
return Qnil;
|
2143
|
-
}
|
2144
|
-
|
2145
|
-
void FileBuilderContext_build(VALUE _self) {
|
2146
|
-
DEFINE_SELF(FileBuilderContext, self, _self);
|
2147
|
-
DescriptorPool* pool = ruby_to_DescriptorPool(self->descriptor_pool);
|
2148
|
-
upb_status status;
|
2149
|
-
|
2150
|
-
rewrite_enum_defaults(pool->symtab, self->file_proto);
|
2151
|
-
rewrite_names(_self, self->file_proto);
|
2152
|
-
|
2153
|
-
upb_status_clear(&status);
|
2154
|
-
if (!upb_symtab_addfile(pool->symtab, self->file_proto, &status)) {
|
2155
|
-
rb_raise(cTypeError, "Unable to add defs to DescriptorPool: %s",
|
2156
|
-
upb_status_errmsg(&status));
|
2157
|
-
}
|
2268
|
+
static void EnumBuilderContext_register(VALUE module) {
|
2269
|
+
VALUE klass = rb_define_class_under(
|
2270
|
+
module, "EnumBuilderContext", rb_cObject);
|
2271
|
+
rb_define_alloc_func(klass, EnumBuilderContext_alloc);
|
2272
|
+
rb_define_method(klass, "initialize", EnumBuilderContext_initialize, 2);
|
2273
|
+
rb_define_method(klass, "value", EnumBuilderContext_value, 2);
|
2274
|
+
rb_gc_register_address(&cEnumBuilderContext);
|
2275
|
+
cEnumBuilderContext = klass;
|
2158
2276
|
}
|
2159
2277
|
|
2160
2278
|
// -----------------------------------------------------------------------------
|
2161
2279
|
// Builder.
|
2162
2280
|
// -----------------------------------------------------------------------------
|
2163
2281
|
|
2164
|
-
|
2282
|
+
typedef struct {
|
2283
|
+
VALUE descriptor_pool;
|
2284
|
+
VALUE default_file_builder;
|
2285
|
+
} Builder;
|
2165
2286
|
|
2166
|
-
|
2287
|
+
static VALUE cBuilder = Qnil;
|
2288
|
+
|
2289
|
+
static void Builder_mark(void* _self) {
|
2167
2290
|
Builder* self = _self;
|
2168
2291
|
rb_gc_mark(self->descriptor_pool);
|
2169
2292
|
rb_gc_mark(self->default_file_builder);
|
2170
2293
|
}
|
2171
2294
|
|
2172
|
-
|
2173
|
-
|
2295
|
+
static const rb_data_type_t Builder_type = {
|
2296
|
+
"Google::Protobuf::Internal::Builder",
|
2297
|
+
{Builder_mark, RUBY_DEFAULT_FREE, NULL},
|
2298
|
+
.flags = RUBY_TYPED_FREE_IMMEDIATELY,
|
2299
|
+
};
|
2300
|
+
|
2301
|
+
static Builder* ruby_to_Builder(VALUE val) {
|
2302
|
+
Builder* ret;
|
2303
|
+
TypedData_Get_Struct(val, Builder, &Builder_type, ret);
|
2304
|
+
return ret;
|
2174
2305
|
}
|
2175
2306
|
|
2176
|
-
VALUE Builder_alloc(VALUE klass) {
|
2307
|
+
static VALUE Builder_alloc(VALUE klass) {
|
2177
2308
|
Builder* self = ALLOC(Builder);
|
2178
|
-
VALUE ret = TypedData_Wrap_Struct(
|
2179
|
-
klass, &_Builder_type, self);
|
2309
|
+
VALUE ret = TypedData_Wrap_Struct(klass, &Builder_type, self);
|
2180
2310
|
self->descriptor_pool = Qnil;
|
2181
2311
|
self->default_file_builder = Qnil;
|
2182
2312
|
return ret;
|
2183
2313
|
}
|
2184
2314
|
|
2185
|
-
void Builder_register(VALUE module) {
|
2186
|
-
VALUE klass = rb_define_class_under(module, "Builder", rb_cObject);
|
2187
|
-
rb_define_alloc_func(klass, Builder_alloc);
|
2188
|
-
rb_define_method(klass, "initialize", Builder_initialize, 1);
|
2189
|
-
rb_define_method(klass, "add_file", Builder_add_file, -1);
|
2190
|
-
rb_define_method(klass, "add_message", Builder_add_message, 1);
|
2191
|
-
rb_define_method(klass, "add_enum", Builder_add_enum, 1);
|
2192
|
-
rb_gc_register_address(&cBuilder);
|
2193
|
-
cBuilder = klass;
|
2194
|
-
}
|
2195
|
-
|
2196
2315
|
/*
|
2197
2316
|
* call-seq:
|
2198
2317
|
* Builder.new(descriptor_pool) => builder
|
@@ -2201,8 +2320,8 @@ void Builder_register(VALUE module) {
|
|
2201
2320
|
* descriptors and atomically register them into a pool in a way that allows for
|
2202
2321
|
* (co)recursive type references.
|
2203
2322
|
*/
|
2204
|
-
VALUE Builder_initialize(VALUE _self, VALUE pool) {
|
2205
|
-
|
2323
|
+
static VALUE Builder_initialize(VALUE _self, VALUE pool) {
|
2324
|
+
Builder* self = ruby_to_Builder(_self);
|
2206
2325
|
self->descriptor_pool = pool;
|
2207
2326
|
self->default_file_builder = Qnil; // Created lazily if needed.
|
2208
2327
|
return Qnil;
|
@@ -2219,8 +2338,8 @@ VALUE Builder_initialize(VALUE _self, VALUE pool) {
|
|
2219
2338
|
*
|
2220
2339
|
* This is the recommended, idiomatic way to build file descriptors.
|
2221
2340
|
*/
|
2222
|
-
VALUE Builder_add_file(int argc, VALUE* argv, VALUE _self) {
|
2223
|
-
|
2341
|
+
static VALUE Builder_add_file(int argc, VALUE* argv, VALUE _self) {
|
2342
|
+
Builder* self = ruby_to_Builder(_self);
|
2224
2343
|
VALUE name, options;
|
2225
2344
|
VALUE ctx;
|
2226
2345
|
VALUE block;
|
@@ -2240,7 +2359,7 @@ VALUE Builder_add_file(int argc, VALUE* argv, VALUE _self) {
|
|
2240
2359
|
}
|
2241
2360
|
|
2242
2361
|
static VALUE Builder_get_default_file(VALUE _self) {
|
2243
|
-
|
2362
|
+
Builder* self = ruby_to_Builder(_self);
|
2244
2363
|
|
2245
2364
|
/* Lazily create only if legacy builder-level methods are called. */
|
2246
2365
|
if (self->default_file_builder == Qnil) {
|
@@ -2264,7 +2383,7 @@ static VALUE Builder_get_default_file(VALUE _self) {
|
|
2264
2383
|
* files generated by protoc which don't add messages within "add_file" block.
|
2265
2384
|
* Descriptors created this way get assigned to a default empty FileDescriptor.
|
2266
2385
|
*/
|
2267
|
-
VALUE Builder_add_message(VALUE _self, VALUE name) {
|
2386
|
+
static VALUE Builder_add_message(VALUE _self, VALUE name) {
|
2268
2387
|
VALUE file_builder = Builder_get_default_file(_self);
|
2269
2388
|
rb_funcall_with_block(file_builder, rb_intern("add_message"), 1, &name,
|
2270
2389
|
rb_block_proc());
|
@@ -2283,7 +2402,7 @@ VALUE Builder_add_message(VALUE _self, VALUE name) {
|
|
2283
2402
|
* Enum descriptors created this way get assigned to a default empty
|
2284
2403
|
* FileDescriptor.
|
2285
2404
|
*/
|
2286
|
-
VALUE Builder_add_enum(VALUE _self, VALUE name) {
|
2405
|
+
static VALUE Builder_add_enum(VALUE _self, VALUE name) {
|
2287
2406
|
VALUE file_builder = Builder_get_default_file(_self);
|
2288
2407
|
rb_funcall_with_block(file_builder, rb_intern("add_enum"), 1, &name,
|
2289
2408
|
rb_block_proc());
|
@@ -2292,8 +2411,8 @@ VALUE Builder_add_enum(VALUE _self, VALUE name) {
|
|
2292
2411
|
|
2293
2412
|
/* This method is hidden from Ruby, and only called directly from
|
2294
2413
|
* DescriptorPool_build(). */
|
2295
|
-
VALUE Builder_build(VALUE _self) {
|
2296
|
-
|
2414
|
+
static VALUE Builder_build(VALUE _self) {
|
2415
|
+
Builder* self = ruby_to_Builder(_self);
|
2297
2416
|
|
2298
2417
|
if (self->default_file_builder != Qnil) {
|
2299
2418
|
FileBuilderContext_build(self->default_file_builder);
|
@@ -2303,8 +2422,19 @@ VALUE Builder_build(VALUE _self) {
|
|
2303
2422
|
return Qnil;
|
2304
2423
|
}
|
2305
2424
|
|
2425
|
+
static void Builder_register(VALUE module) {
|
2426
|
+
VALUE klass = rb_define_class_under(module, "Builder", rb_cObject);
|
2427
|
+
rb_define_alloc_func(klass, Builder_alloc);
|
2428
|
+
rb_define_method(klass, "initialize", Builder_initialize, 1);
|
2429
|
+
rb_define_method(klass, "add_file", Builder_add_file, -1);
|
2430
|
+
rb_define_method(klass, "add_message", Builder_add_message, 1);
|
2431
|
+
rb_define_method(klass, "add_enum", Builder_add_enum, 1);
|
2432
|
+
rb_gc_register_address(&cBuilder);
|
2433
|
+
cBuilder = klass;
|
2434
|
+
}
|
2435
|
+
|
2306
2436
|
static VALUE get_def_obj(VALUE _descriptor_pool, const void* ptr, VALUE klass) {
|
2307
|
-
|
2437
|
+
DescriptorPool* descriptor_pool = ruby_to_DescriptorPool(_descriptor_pool);
|
2308
2438
|
VALUE key = ULL2NUM((intptr_t)ptr);
|
2309
2439
|
VALUE def;
|
2310
2440
|
|
@@ -2319,48 +2449,111 @@ static VALUE get_def_obj(VALUE _descriptor_pool, const void* ptr, VALUE klass) {
|
|
2319
2449
|
VALUE args[3] = { c_only_cookie, _descriptor_pool, key };
|
2320
2450
|
def = rb_class_new_instance(3, args, klass);
|
2321
2451
|
rb_hash_aset(descriptor_pool->def_to_descriptor, key, def);
|
2322
|
-
|
2323
|
-
// For message defs, we now eagerly get/create descriptors for all
|
2324
|
-
// submessages. We will need these anyway to parse or serialize this
|
2325
|
-
// message type. But more importantly, we must do this now so that
|
2326
|
-
// add_handlers_for_message() (which calls get_msgdef_obj()) does *not*
|
2327
|
-
// need to create a Ruby object or insert into a Ruby Hash. We need to
|
2328
|
-
// avoid triggering GC, which can switch Ruby threads and re-enter our
|
2329
|
-
// C extension from a different thread. This wreaks havoc on our state
|
2330
|
-
// if we were in the middle of building handlers.
|
2331
|
-
if (klass == cDescriptor) {
|
2332
|
-
const upb_msgdef *m = ptr;
|
2333
|
-
upb_msg_field_iter it;
|
2334
|
-
for (upb_msg_field_begin(&it, m);
|
2335
|
-
!upb_msg_field_done(&it);
|
2336
|
-
upb_msg_field_next(&it)) {
|
2337
|
-
const upb_fielddef* f = upb_msg_iter_field(&it);
|
2338
|
-
if (upb_fielddef_issubmsg(f)) {
|
2339
|
-
get_msgdef_obj(_descriptor_pool, upb_fielddef_msgsubdef(f));
|
2340
|
-
}
|
2341
|
-
}
|
2342
|
-
}
|
2343
2452
|
}
|
2344
2453
|
|
2345
2454
|
return def;
|
2346
2455
|
}
|
2347
2456
|
|
2348
|
-
VALUE get_msgdef_obj(VALUE descriptor_pool, const upb_msgdef* def) {
|
2457
|
+
static VALUE get_msgdef_obj(VALUE descriptor_pool, const upb_msgdef* def) {
|
2349
2458
|
return get_def_obj(descriptor_pool, def, cDescriptor);
|
2350
2459
|
}
|
2351
2460
|
|
2352
|
-
VALUE get_enumdef_obj(VALUE descriptor_pool, const upb_enumdef* def) {
|
2461
|
+
static VALUE get_enumdef_obj(VALUE descriptor_pool, const upb_enumdef* def) {
|
2353
2462
|
return get_def_obj(descriptor_pool, def, cEnumDescriptor);
|
2354
2463
|
}
|
2355
2464
|
|
2356
|
-
VALUE get_fielddef_obj(VALUE descriptor_pool, const upb_fielddef* def) {
|
2465
|
+
static VALUE get_fielddef_obj(VALUE descriptor_pool, const upb_fielddef* def) {
|
2357
2466
|
return get_def_obj(descriptor_pool, def, cFieldDescriptor);
|
2358
2467
|
}
|
2359
2468
|
|
2360
|
-
VALUE get_filedef_obj(VALUE descriptor_pool, const upb_filedef* def) {
|
2469
|
+
static VALUE get_filedef_obj(VALUE descriptor_pool, const upb_filedef* def) {
|
2361
2470
|
return get_def_obj(descriptor_pool, def, cFileDescriptor);
|
2362
2471
|
}
|
2363
2472
|
|
2364
|
-
VALUE get_oneofdef_obj(VALUE descriptor_pool, const upb_oneofdef* def) {
|
2473
|
+
static VALUE get_oneofdef_obj(VALUE descriptor_pool, const upb_oneofdef* def) {
|
2365
2474
|
return get_def_obj(descriptor_pool, def, cOneofDescriptor);
|
2366
2475
|
}
|
2476
|
+
|
2477
|
+
// -----------------------------------------------------------------------------
|
2478
|
+
// Shared functions
|
2479
|
+
// -----------------------------------------------------------------------------
|
2480
|
+
|
2481
|
+
// Functions exposed to other modules in defs.h.
|
2482
|
+
|
2483
|
+
VALUE Descriptor_DefToClass(const upb_msgdef *m) {
|
2484
|
+
const upb_symtab *symtab = upb_filedef_symtab(upb_msgdef_file(m));
|
2485
|
+
VALUE pool = ObjectCache_Get(symtab);
|
2486
|
+
PBRUBY_ASSERT(pool != Qnil);
|
2487
|
+
VALUE desc_rb = get_msgdef_obj(pool, m);
|
2488
|
+
const Descriptor* desc = ruby_to_Descriptor(desc_rb);
|
2489
|
+
return desc->klass;
|
2490
|
+
}
|
2491
|
+
|
2492
|
+
const upb_msgdef *Descriptor_GetMsgDef(VALUE desc_rb) {
|
2493
|
+
const Descriptor* desc = ruby_to_Descriptor(desc_rb);
|
2494
|
+
return desc->msgdef;
|
2495
|
+
}
|
2496
|
+
|
2497
|
+
VALUE TypeInfo_InitArg(int argc, VALUE *argv, int skip_arg) {
|
2498
|
+
if (argc > skip_arg) {
|
2499
|
+
if (argc > 1 + skip_arg) {
|
2500
|
+
rb_raise(rb_eArgError, "Expected a maximum of %d arguments.", skip_arg + 1);
|
2501
|
+
}
|
2502
|
+
return argv[skip_arg];
|
2503
|
+
} else {
|
2504
|
+
return Qnil;
|
2505
|
+
}
|
2506
|
+
}
|
2507
|
+
|
2508
|
+
TypeInfo TypeInfo_FromClass(int argc, VALUE* argv, int skip_arg,
|
2509
|
+
VALUE* type_class, VALUE* init_arg) {
|
2510
|
+
TypeInfo ret = {ruby_to_fieldtype(argv[skip_arg])};
|
2511
|
+
|
2512
|
+
if (ret.type == UPB_TYPE_MESSAGE || ret.type == UPB_TYPE_ENUM) {
|
2513
|
+
*init_arg = TypeInfo_InitArg(argc, argv, skip_arg + 2);
|
2514
|
+
|
2515
|
+
if (argc < 2 + skip_arg) {
|
2516
|
+
rb_raise(rb_eArgError, "Expected at least %d arguments for message/enum.",
|
2517
|
+
2 + skip_arg);
|
2518
|
+
}
|
2519
|
+
|
2520
|
+
VALUE klass = argv[1 + skip_arg];
|
2521
|
+
VALUE desc = MessageOrEnum_GetDescriptor(klass);
|
2522
|
+
*type_class = klass;
|
2523
|
+
|
2524
|
+
if (desc == Qnil) {
|
2525
|
+
rb_raise(rb_eArgError,
|
2526
|
+
"Type class has no descriptor. Please pass a "
|
2527
|
+
"class or enum as returned by the DescriptorPool.");
|
2528
|
+
}
|
2529
|
+
|
2530
|
+
if (ret.type == UPB_TYPE_MESSAGE) {
|
2531
|
+
ret.def.msgdef = ruby_to_Descriptor(desc)->msgdef;
|
2532
|
+
Message_CheckClass(klass);
|
2533
|
+
} else {
|
2534
|
+
PBRUBY_ASSERT(ret.type == UPB_TYPE_ENUM);
|
2535
|
+
ret.def.enumdef = ruby_to_EnumDescriptor(desc)->enumdef;
|
2536
|
+
}
|
2537
|
+
} else {
|
2538
|
+
*init_arg = TypeInfo_InitArg(argc, argv, skip_arg + 1);
|
2539
|
+
}
|
2540
|
+
|
2541
|
+
return ret;
|
2542
|
+
}
|
2543
|
+
|
2544
|
+
void Defs_register(VALUE module) {
|
2545
|
+
DescriptorPool_register(module);
|
2546
|
+
Descriptor_register(module);
|
2547
|
+
FileDescriptor_register(module);
|
2548
|
+
FieldDescriptor_register(module);
|
2549
|
+
OneofDescriptor_register(module);
|
2550
|
+
EnumDescriptor_register(module);
|
2551
|
+
FileBuilderContext_register(module);
|
2552
|
+
MessageBuilderContext_register(module);
|
2553
|
+
OneofBuilderContext_register(module);
|
2554
|
+
EnumBuilderContext_register(module);
|
2555
|
+
Builder_register(module);
|
2556
|
+
|
2557
|
+
rb_gc_register_address(&c_only_cookie);
|
2558
|
+
c_only_cookie = rb_class_new_instance(0, NULL, rb_cObject);
|
2559
|
+
}
|