oj 2.18.3 → 3.13.14

Sign up to get free protection for your applications and to get access to all the features.
Files changed (182) hide show
  1. checksums.yaml +5 -5
  2. data/CHANGELOG.md +1324 -0
  3. data/README.md +51 -204
  4. data/RELEASE_NOTES.md +61 -0
  5. data/ext/oj/buf.h +49 -72
  6. data/ext/oj/cache.c +326 -0
  7. data/ext/oj/cache.h +21 -0
  8. data/ext/oj/cache8.c +61 -64
  9. data/ext/oj/cache8.h +12 -39
  10. data/ext/oj/circarray.c +37 -68
  11. data/ext/oj/circarray.h +16 -42
  12. data/ext/oj/code.c +221 -0
  13. data/ext/oj/code.h +40 -0
  14. data/ext/oj/compat.c +231 -107
  15. data/ext/oj/custom.c +1125 -0
  16. data/ext/oj/debug.c +132 -0
  17. data/ext/oj/dump.c +935 -2513
  18. data/ext/oj/dump.h +108 -0
  19. data/ext/oj/dump_compat.c +936 -0
  20. data/ext/oj/dump_leaf.c +164 -0
  21. data/ext/oj/dump_object.c +761 -0
  22. data/ext/oj/dump_strict.c +410 -0
  23. data/ext/oj/encode.h +7 -42
  24. data/ext/oj/encoder.c +43 -0
  25. data/ext/oj/err.c +40 -54
  26. data/ext/oj/err.h +52 -46
  27. data/ext/oj/extconf.rb +21 -30
  28. data/ext/oj/fast.c +1097 -1080
  29. data/ext/oj/intern.c +301 -0
  30. data/ext/oj/intern.h +26 -0
  31. data/ext/oj/mimic_json.c +893 -0
  32. data/ext/oj/object.c +549 -620
  33. data/ext/oj/odd.c +155 -167
  34. data/ext/oj/odd.h +37 -63
  35. data/ext/oj/oj.c +1661 -2063
  36. data/ext/oj/oj.h +341 -270
  37. data/ext/oj/parse.c +974 -737
  38. data/ext/oj/parse.h +105 -97
  39. data/ext/oj/parser.c +1526 -0
  40. data/ext/oj/parser.h +90 -0
  41. data/ext/oj/rails.c +1504 -0
  42. data/ext/oj/rails.h +18 -0
  43. data/ext/oj/reader.c +141 -163
  44. data/ext/oj/reader.h +75 -113
  45. data/ext/oj/resolve.c +45 -93
  46. data/ext/oj/resolve.h +7 -34
  47. data/ext/oj/rxclass.c +143 -0
  48. data/ext/oj/rxclass.h +26 -0
  49. data/ext/oj/saj.c +447 -511
  50. data/ext/oj/saj2.c +348 -0
  51. data/ext/oj/scp.c +91 -138
  52. data/ext/oj/sparse.c +793 -644
  53. data/ext/oj/stream_writer.c +331 -0
  54. data/ext/oj/strict.c +145 -109
  55. data/ext/oj/string_writer.c +493 -0
  56. data/ext/oj/trace.c +72 -0
  57. data/ext/oj/trace.h +28 -0
  58. data/ext/oj/usual.c +1254 -0
  59. data/ext/oj/util.c +136 -0
  60. data/ext/oj/util.h +20 -0
  61. data/ext/oj/val_stack.c +62 -70
  62. data/ext/oj/val_stack.h +95 -129
  63. data/ext/oj/validate.c +51 -0
  64. data/ext/oj/wab.c +622 -0
  65. data/lib/oj/bag.rb +1 -0
  66. data/lib/oj/easy_hash.rb +17 -8
  67. data/lib/oj/error.rb +10 -11
  68. data/lib/oj/json.rb +176 -0
  69. data/lib/oj/mimic.rb +158 -19
  70. data/lib/oj/state.rb +132 -0
  71. data/lib/oj/version.rb +2 -2
  72. data/lib/oj.rb +1 -31
  73. data/pages/Advanced.md +22 -0
  74. data/pages/Compatibility.md +25 -0
  75. data/pages/Custom.md +23 -0
  76. data/pages/Encoding.md +65 -0
  77. data/pages/JsonGem.md +94 -0
  78. data/pages/Modes.md +161 -0
  79. data/pages/Options.md +327 -0
  80. data/pages/Parser.md +309 -0
  81. data/pages/Rails.md +167 -0
  82. data/pages/Security.md +20 -0
  83. data/pages/WAB.md +13 -0
  84. data/test/activerecord/result_test.rb +32 -0
  85. data/test/activesupport4/decoding_test.rb +108 -0
  86. data/test/activesupport4/encoding_test.rb +531 -0
  87. data/test/activesupport4/test_helper.rb +41 -0
  88. data/test/activesupport5/abstract_unit.rb +45 -0
  89. data/test/activesupport5/decoding_test.rb +133 -0
  90. data/test/activesupport5/encoding_test.rb +500 -0
  91. data/test/activesupport5/encoding_test_cases.rb +98 -0
  92. data/test/activesupport5/test_helper.rb +72 -0
  93. data/test/activesupport5/time_zone_test_helpers.rb +39 -0
  94. data/test/activesupport6/abstract_unit.rb +44 -0
  95. data/test/activesupport6/decoding_test.rb +133 -0
  96. data/test/activesupport6/encoding_test.rb +507 -0
  97. data/test/activesupport6/encoding_test_cases.rb +98 -0
  98. data/test/activesupport6/test_common.rb +17 -0
  99. data/test/activesupport6/test_helper.rb +163 -0
  100. data/test/activesupport6/time_zone_test_helpers.rb +39 -0
  101. data/test/activesupport7/abstract_unit.rb +49 -0
  102. data/test/activesupport7/decoding_test.rb +125 -0
  103. data/test/activesupport7/encoding_test.rb +486 -0
  104. data/test/activesupport7/encoding_test_cases.rb +104 -0
  105. data/test/activesupport7/time_zone_test_helpers.rb +47 -0
  106. data/test/bar.rb +9 -0
  107. data/test/baz.rb +16 -0
  108. data/test/bug.rb +11 -46
  109. data/test/foo.rb +69 -16
  110. data/test/helper.rb +10 -1
  111. data/test/isolated/shared.rb +12 -8
  112. data/test/isolated/test_mimic_rails_after.rb +3 -3
  113. data/test/isolated/test_mimic_rails_before.rb +3 -3
  114. data/test/json_gem/json_addition_test.rb +216 -0
  115. data/test/json_gem/json_common_interface_test.rb +153 -0
  116. data/test/json_gem/json_encoding_test.rb +107 -0
  117. data/test/json_gem/json_ext_parser_test.rb +20 -0
  118. data/test/json_gem/json_fixtures_test.rb +35 -0
  119. data/test/json_gem/json_generator_test.rb +397 -0
  120. data/test/json_gem/json_generic_object_test.rb +90 -0
  121. data/test/json_gem/json_parser_test.rb +470 -0
  122. data/test/json_gem/json_string_matching_test.rb +42 -0
  123. data/test/json_gem/test_helper.rb +26 -0
  124. data/test/mem.rb +33 -0
  125. data/test/perf.rb +1 -1
  126. data/test/perf_compat.rb +30 -28
  127. data/test/perf_dump.rb +50 -0
  128. data/test/perf_object.rb +1 -1
  129. data/test/perf_once.rb +58 -0
  130. data/test/perf_parser.rb +189 -0
  131. data/test/perf_scp.rb +11 -10
  132. data/test/perf_strict.rb +30 -19
  133. data/test/perf_wab.rb +131 -0
  134. data/test/prec.rb +23 -0
  135. data/test/sample.rb +0 -1
  136. data/test/sample_json.rb +1 -1
  137. data/test/test_compat.rb +219 -102
  138. data/test/test_custom.rb +533 -0
  139. data/test/test_fast.rb +107 -35
  140. data/test/test_file.rb +19 -25
  141. data/test/test_generate.rb +21 -0
  142. data/test/test_hash.rb +11 -1
  143. data/test/test_integer_range.rb +72 -0
  144. data/test/test_null.rb +376 -0
  145. data/test/test_object.rb +357 -70
  146. data/test/test_parser.rb +27 -0
  147. data/test/test_parser_saj.rb +245 -0
  148. data/test/test_parser_usual.rb +217 -0
  149. data/test/test_rails.rb +35 -0
  150. data/test/test_saj.rb +1 -1
  151. data/test/test_scp.rb +39 -2
  152. data/test/test_strict.rb +186 -7
  153. data/test/test_various.rb +160 -774
  154. data/test/test_wab.rb +307 -0
  155. data/test/test_writer.rb +90 -2
  156. data/test/tests.rb +24 -0
  157. data/test/tests_mimic.rb +14 -0
  158. data/test/tests_mimic_addition.rb +7 -0
  159. data/test/zoo.rb +13 -0
  160. metadata +194 -56
  161. data/ext/oj/hash.c +0 -163
  162. data/ext/oj/hash.h +0 -46
  163. data/ext/oj/hash_test.c +0 -512
  164. data/test/activesupport_datetime_test.rb +0 -23
  165. data/test/bug2.rb +0 -10
  166. data/test/bug3.rb +0 -46
  167. data/test/bug_fast.rb +0 -32
  168. data/test/bug_load.rb +0 -24
  169. data/test/crash.rb +0 -111
  170. data/test/curl/curl_oj.rb +0 -46
  171. data/test/curl/get_oj.rb +0 -24
  172. data/test/curl/just_curl.rb +0 -31
  173. data/test/curl/just_oj.rb +0 -51
  174. data/test/example.rb +0 -11
  175. data/test/io.rb +0 -48
  176. data/test/isolated/test_mimic_rails_datetime.rb +0 -27
  177. data/test/mod.rb +0 -16
  178. data/test/rails.rb +0 -50
  179. data/test/russian.rb +0 -18
  180. data/test/struct.rb +0 -29
  181. data/test/test_serializer.rb +0 -59
  182. data/test/write_timebars.rb +0 -31
data/ext/oj/odd.c CHANGED
@@ -1,109 +1,107 @@
1
- /* odd.c
2
- * Copyright (c) 2011, Peter Ohler
3
- * All rights reserved.
4
- *
5
- * Redistribution and use in source and binary forms, with or without
6
- * modification, are permitted provided that the following conditions are met:
7
- *
8
- * - Redistributions of source code must retain the above copyright notice, this
9
- * list of conditions and the following disclaimer.
10
- *
11
- * - Redistributions in binary form must reproduce the above copyright notice,
12
- * this list of conditions and the following disclaimer in the documentation
13
- * and/or other materials provided with the distribution.
14
- *
15
- * - Neither the name of Peter Ohler nor the names of its contributors may be
16
- * used to endorse or promote products derived from this software without
17
- * specific prior written permission.
18
- *
19
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
20
- * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21
- * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
22
- * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
23
- * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
24
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
25
- * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
26
- * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
27
- * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28
- * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29
- */
1
+ // Copyright (c) 2011 Peter Ohler. All rights reserved.
2
+ // Licensed under the MIT License. See LICENSE file in the project root for license details.
3
+
4
+ #include "odd.h"
30
5
 
31
6
  #include <string.h>
32
7
 
33
- #include "odd.h"
8
+ static Odd odds = NULL;
9
+ static ID sec_id;
10
+ static ID sec_fraction_id;
11
+ static ID to_f_id;
12
+ static ID numerator_id;
13
+ static ID denominator_id;
14
+ static ID rational_id;
34
15
 
35
- static struct _Odd _odds[4]; // bump up if new initial Odd classes are added
36
- static struct _Odd *odds = _odds;
37
- static long odd_cnt = 0;
38
- static ID sec_id;
39
- static ID sec_fraction_id;
40
- static ID to_f_id;
41
- static ID numerator_id;
42
- static ID denominator_id;
43
- static ID rational_id;
44
- static VALUE rational_class;
45
-
46
- static void
47
- set_class(Odd odd, const char *classname) {
48
- const char **np;
49
- ID *idp;
16
+ static void set_class(Odd odd, const char *classname) {
17
+ const char **np;
18
+ ID *idp;
50
19
 
51
20
  odd->classname = classname;
52
- odd->clen = strlen(classname);
53
- odd->clas = rb_const_get(rb_cObject, rb_intern(classname));
21
+ odd->clen = strlen(classname);
22
+ odd->clas = rb_const_get(rb_cObject, rb_intern(classname));
23
+ rb_gc_register_mark_object(odd->clas);
54
24
  odd->create_obj = odd->clas;
25
+ rb_gc_register_mark_object(odd->create_obj);
55
26
  odd->create_op = rb_intern("new");
56
27
  odd->is_module = (T_MODULE == rb_type(odd->clas));
57
- odd->raw = 0;
28
+ odd->raw = 0;
58
29
  for (np = odd->attr_names, idp = odd->attrs; 0 != *np; np++, idp++) {
59
- *idp = rb_intern(*np);
30
+ *idp = rb_intern(*np);
60
31
  }
61
32
  *idp = 0;
62
33
  }
63
34
 
64
- static VALUE
65
- get_datetime_secs(VALUE obj) {
66
- VALUE rsecs = rb_funcall(obj, sec_id, 0);
67
- VALUE rfrac = rb_funcall(obj, sec_fraction_id, 0);
68
- long sec = NUM2LONG(rsecs);
69
- long long num = rb_num2ll(rb_funcall(rfrac, numerator_id, 0));
70
- long long den = rb_num2ll(rb_funcall(rfrac, denominator_id, 0));
71
-
72
- #if DATETIME_1_8
73
- num *= 86400;
74
- #endif
35
+ static VALUE get_datetime_secs(VALUE obj) {
36
+ volatile VALUE rsecs = rb_funcall(obj, sec_id, 0);
37
+ volatile VALUE rfrac = rb_funcall(obj, sec_fraction_id, 0);
38
+ long sec = NUM2LONG(rsecs);
39
+ long long num = NUM2LL(rb_funcall(rfrac, numerator_id, 0));
40
+ long long den = NUM2LL(rb_funcall(rfrac, denominator_id, 0));
41
+
75
42
  num += sec * den;
76
43
 
77
44
  return rb_funcall(rb_cObject, rational_id, 2, rb_ll2inum(num), rb_ll2inum(den));
78
45
  }
79
46
 
80
- void
81
- oj_odd_init() {
82
- Odd odd;
83
- const char **np;
47
+ static void print_odd(Odd odd) {
48
+ const char **np;
49
+ int i;
50
+
51
+ printf(" %s {\n", odd->classname);
52
+ printf(" attr_cnt: %d %p\n", odd->attr_cnt, (void *)odd->attr_names);
53
+ printf(" attr_names: %p\n", (void *)*odd->attr_names);
54
+ printf(" attr_names: %c\n", **odd->attr_names);
55
+ for (i = odd->attr_cnt, np = odd->attr_names; 0 < i; i--, np++) {
56
+ printf(" %d %s\n", i, *np);
57
+ }
58
+ printf(" }\n");
59
+ }
60
+
61
+ void print_all_odds(const char *label) {
62
+ Odd odd;
63
+ printf("@ %s {\n", label);
64
+ for (odd = odds; NULL != odd; odd = odd->next) {
65
+ print_odd(odd);
66
+ }
67
+ printf("}\n");
68
+ }
69
+
70
+ static Odd odd_create(void) {
71
+ Odd odd = ALLOC(struct _odd);
72
+
73
+ memset(odd, 0, sizeof(struct _odd));
74
+ odd->next = odds;
75
+ odds = odd;
76
+
77
+ return odd;
78
+ }
79
+
80
+ void oj_odd_init(void) {
81
+ Odd odd;
82
+ const char **np;
84
83
 
85
- sec_id = rb_intern("sec");
84
+ sec_id = rb_intern("sec");
86
85
  sec_fraction_id = rb_intern("sec_fraction");
87
- to_f_id = rb_intern("to_f");
88
- numerator_id = rb_intern("numerator");
89
- denominator_id = rb_intern("denominator");
90
- rational_id = rb_intern("Rational");
91
- rational_class = rb_const_get(rb_cObject, rational_id);
92
-
93
- memset(_odds, 0, sizeof(_odds));
94
- odd = odds;
86
+ to_f_id = rb_intern("to_f");
87
+ numerator_id = rb_intern("numerator");
88
+ denominator_id = rb_intern("denominator");
89
+ rational_id = rb_intern("Rational");
90
+
95
91
  // Rational
96
- np = odd->attr_names;
92
+ odd = odd_create();
93
+ np = odd->attr_names;
97
94
  *np++ = "numerator";
98
95
  *np++ = "denominator";
99
- *np = 0;
96
+ *np = 0;
100
97
  set_class(odd, "Rational");
101
98
  odd->create_obj = rb_cObject;
102
- odd->create_op = rational_id;
103
- odd->attr_cnt = 2;
99
+ odd->create_op = rational_id;
100
+ odd->attr_cnt = 2;
101
+
104
102
  // Date
105
- odd++;
106
- np = odd->attr_names;
103
+ odd = odd_create();
104
+ np = odd->attr_names;
107
105
  *np++ = "year";
108
106
  *np++ = "month";
109
107
  *np++ = "day";
@@ -111,9 +109,10 @@ oj_odd_init() {
111
109
  *np++ = 0;
112
110
  set_class(odd, "Date");
113
111
  odd->attr_cnt = 4;
112
+
114
113
  // DateTime
115
- odd++;
116
- np = odd->attr_names;
114
+ odd = odd_create();
115
+ np = odd->attr_names;
117
116
  *np++ = "year";
118
117
  *np++ = "month";
119
118
  *np++ = "day";
@@ -124,132 +123,121 @@ oj_odd_init() {
124
123
  *np++ = "start";
125
124
  *np++ = 0;
126
125
  set_class(odd, "DateTime");
127
- odd->attr_cnt = 8;
126
+ odd->attr_cnt = 8;
128
127
  odd->attrFuncs[5] = get_datetime_secs;
128
+
129
129
  // Range
130
- odd++;
131
- np = odd->attr_names;
130
+ odd = odd_create();
131
+ np = odd->attr_names;
132
132
  *np++ = "begin";
133
133
  *np++ = "end";
134
134
  *np++ = "exclude_end?";
135
135
  *np++ = 0;
136
136
  set_class(odd, "Range");
137
137
  odd->attr_cnt = 3;
138
-
139
- odd_cnt = odd - odds + 1;
140
138
  }
141
139
 
142
- Odd
143
- oj_get_odd(VALUE clas) {
144
- Odd odd;
145
- const char *classname = NULL;
146
-
147
- for (odd = odds + odd_cnt - 1; odds <= odd; odd--) {
148
- if (clas == odd->clas) {
149
- return odd;
150
- }
151
- if (odd->is_module) {
152
- if (NULL == classname) {
153
- classname = rb_class2name(clas);
154
- }
155
- if (0 == strncmp(odd->classname, classname, odd->clen) &&
156
- ':' == classname[odd->clen]) {
157
- return odd;
158
- }
159
- }
140
+ Odd oj_get_odd(VALUE clas) {
141
+ Odd odd;
142
+ const char *classname = NULL;
143
+
144
+ for (odd = odds; NULL != odd; odd = odd->next) {
145
+ if (clas == odd->clas) {
146
+ return odd;
147
+ }
148
+ if (odd->is_module) {
149
+ if (NULL == classname) {
150
+ classname = rb_class2name(clas);
151
+ }
152
+ if (0 == strncmp(odd->classname, classname, odd->clen) && ':' == classname[odd->clen]) {
153
+ return odd;
154
+ }
155
+ }
160
156
  }
161
157
  return NULL;
162
158
  }
163
159
 
164
- Odd
165
- oj_get_oddc(const char *classname, size_t len) {
166
- Odd odd;
167
-
168
- for (odd = odds + odd_cnt - 1; odds <= odd; odd--) {
169
- if (len == odd->clen && 0 == strncmp(classname, odd->classname, len)) {
170
- return odd;
171
- }
172
- if (odd->is_module &&
173
- 0 == strncmp(odd->classname, classname, odd->clen) &&
174
- ':' == classname[odd->clen]) {
175
- return odd;
176
- }
160
+ Odd oj_get_oddc(const char *classname, size_t len) {
161
+ Odd odd;
162
+
163
+ for (odd = odds; NULL != odd; odd = odd->next) {
164
+ if (len == odd->clen && 0 == strncmp(classname, odd->classname, len)) {
165
+ return odd;
166
+ }
167
+ if (odd->is_module && 0 == strncmp(odd->classname, classname, odd->clen) && ':' == classname[odd->clen]) {
168
+ return odd;
169
+ }
177
170
  }
178
- return 0;
171
+ return NULL;
179
172
  }
180
173
 
181
- OddArgs
182
- oj_odd_alloc_args(Odd odd) {
183
- OddArgs oa = ALLOC_N(struct _OddArgs, 1);
184
- VALUE *a;
185
- int i;
174
+ OddArgs oj_odd_alloc_args(Odd odd) {
175
+ OddArgs oa = ALLOC_N(struct _oddArgs, 1);
176
+ VALUE *a;
177
+ int i;
186
178
 
187
179
  oa->odd = odd;
188
180
  for (i = odd->attr_cnt, a = oa->args; 0 < i; i--, a++) {
189
- *a = Qnil;
181
+ *a = Qnil;
190
182
  }
191
183
  return oa;
192
184
  }
193
185
 
194
- void
195
- oj_odd_free(OddArgs args) {
186
+ void oj_odd_free(OddArgs args) {
196
187
  xfree(args);
197
188
  }
198
189
 
199
- int
200
- oj_odd_set_arg(OddArgs args, const char *key, size_t klen, VALUE value) {
201
- const char **np;
202
- VALUE *vp;
203
- int i;
190
+ int oj_odd_set_arg(OddArgs args, const char *key, size_t klen, VALUE value) {
191
+ const char **np;
192
+ VALUE *vp;
193
+ int i;
204
194
 
205
195
  for (i = args->odd->attr_cnt, np = args->odd->attr_names, vp = args->args; 0 < i; i--, np++, vp++) {
206
- if (0 == strncmp(key, *np, klen) && '\0' == *((*np) + klen)) {
207
- *vp = value;
208
- return 0;
209
- }
196
+ if (0 == strncmp(key, *np, klen) && '\0' == *((*np) + klen)) {
197
+ *vp = value;
198
+ return 0;
199
+ }
210
200
  }
211
201
  return -1;
212
202
  }
213
203
 
214
- void
215
- oj_reg_odd(VALUE clas, VALUE create_object, VALUE create_method, int mcnt, VALUE *members, bool raw) {
216
- Odd odd;
217
- const char **np;
218
- ID *ap;
219
- AttrGetFunc *fp;
204
+ void oj_reg_odd(VALUE clas, VALUE create_object, VALUE create_method, int mcnt, VALUE *members, bool raw) {
205
+ Odd odd;
206
+ const char **np;
207
+ ID *ap;
208
+ AttrGetFunc *fp;
220
209
 
221
- if (_odds == odds) {
222
- odds = ALLOC_N(struct _Odd, odd_cnt + 1);
223
-
224
- memcpy(odds, _odds, sizeof(struct _Odd) * odd_cnt);
225
- } else {
226
- REALLOC_N(odds, struct _Odd, odd_cnt + 1);
227
- }
228
- odd = odds + odd_cnt;
210
+ odd = odd_create();
229
211
  odd->clas = clas;
230
- odd->classname = strdup(rb_class2name(clas));
231
- odd->clen = strlen(odd->classname);
212
+ rb_gc_register_mark_object(odd->clas);
213
+ if (NULL == (odd->classname = strdup(rb_class2name(clas)))) {
214
+ rb_raise(rb_eNoMemError, "for class name.");
215
+ }
216
+ odd->clen = strlen(odd->classname);
232
217
  odd->create_obj = create_object;
218
+ rb_gc_register_mark_object(odd->create_obj);
233
219
  odd->create_op = SYM2ID(create_method);
234
- odd->attr_cnt = mcnt;
220
+ odd->attr_cnt = mcnt;
235
221
  odd->is_module = (T_MODULE == rb_type(clas));
236
- odd->raw = raw;
222
+ odd->raw = raw;
237
223
  for (ap = odd->attrs, np = odd->attr_names, fp = odd->attrFuncs; 0 < mcnt; mcnt--, ap++, np++, members++, fp++) {
238
- *fp = 0;
239
- switch (rb_type(*members)) {
240
- case T_STRING:
241
- *np = strdup(rb_string_value_ptr(members));
242
- break;
243
- case T_SYMBOL:
244
- *np = rb_id2name(SYM2ID(*members));
245
- break;
246
- default:
247
- rb_raise(rb_eArgError, "registered member identifiers must be Strings or Symbols.");
248
- break;
249
- }
250
- *ap = rb_intern(*np);
224
+ *fp = 0;
225
+ switch (rb_type(*members)) {
226
+ case T_STRING:
227
+ if (NULL == (*np = strdup(RSTRING_PTR(*members)))) {
228
+ rb_raise(rb_eNoMemError, "for attribute name.");
229
+ }
230
+ break;
231
+ case T_SYMBOL:
232
+ // The symbol can move and invalidate the name so make a copy.
233
+ if (NULL == (*np = strdup(rb_id2name(SYM2ID(*members))))) {
234
+ rb_raise(rb_eNoMemError, "for attribute name.");
235
+ }
236
+ break;
237
+ default: rb_raise(rb_eArgError, "registered member identifiers must be Strings or Symbols."); break;
238
+ }
239
+ *ap = rb_intern(*np);
251
240
  }
252
241
  *np = 0;
253
242
  *ap = 0;
254
- odd_cnt++;
255
243
  }
data/ext/oj/odd.h CHANGED
@@ -1,69 +1,43 @@
1
- /* odd.h
2
- * Copyright (c) 2011, Peter Ohler
3
- * All rights reserved.
4
- *
5
- * Redistribution and use in source and binary forms, with or without
6
- * modification, are permitted provided that the following conditions are met:
7
- *
8
- * - Redistributions of source code must retain the above copyright notice, this
9
- * list of conditions and the following disclaimer.
10
- *
11
- * - Redistributions in binary form must reproduce the above copyright notice,
12
- * this list of conditions and the following disclaimer in the documentation
13
- * and/or other materials provided with the distribution.
14
- *
15
- * - Neither the name of Peter Ohler nor the names of its contributors may be
16
- * used to endorse or promote products derived from this software without
17
- * specific prior written permission.
18
- *
19
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
20
- * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21
- * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
22
- * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
23
- * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
24
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
25
- * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
26
- * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
27
- * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28
- * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29
- */
1
+ // Copyright (c) 2011 Peter Ohler. All rights reserved.
2
+ // Licensed under the MIT License. See LICENSE file in the project root for license details.
30
3
 
31
- #ifndef __OJ_ODD_H__
32
- #define __OJ_ODD_H__
4
+ #ifndef OJ_ODD_H
5
+ #define OJ_ODD_H
33
6
 
34
7
  #include <stdbool.h>
35
8
 
36
9
  #include "ruby.h"
37
10
 
38
- #define MAX_ODD_ARGS 10
39
-
40
- typedef VALUE (*AttrGetFunc)(VALUE obj);
41
-
42
- typedef struct _Odd {
43
- const char *classname;
44
- size_t clen;
45
- VALUE clas; // Ruby class or module
46
- VALUE create_obj;
47
- ID create_op;
48
- int attr_cnt;
49
- bool is_module;
50
- bool raw;
51
- const char *attr_names[MAX_ODD_ARGS]; // 0 terminated attr IDs
52
- ID attrs[MAX_ODD_ARGS]; // 0 terminated attr IDs
53
- AttrGetFunc attrFuncs[MAX_ODD_ARGS];
54
- } *Odd;
55
-
56
- typedef struct _OddArgs {
57
- Odd odd;
58
- VALUE args[MAX_ODD_ARGS];
59
- } *OddArgs;
60
-
61
- extern void oj_odd_init(void);
62
- extern Odd oj_get_odd(VALUE clas);
63
- extern Odd oj_get_oddc(const char *classname, size_t len);
64
- extern OddArgs oj_odd_alloc_args(Odd odd);
65
- extern void oj_odd_free(OddArgs args);
66
- extern int oj_odd_set_arg(OddArgs args, const char *key, size_t klen, VALUE value);
67
- extern void oj_reg_odd(VALUE clas, VALUE create_object, VALUE create_method, int mcnt, VALUE *members, bool raw);
68
-
69
- #endif /* __OJ_ODD_H__ */
11
+ #define MAX_ODD_ARGS 10
12
+
13
+ typedef VALUE (*AttrGetFunc)(VALUE obj);
14
+
15
+ typedef struct _odd {
16
+ struct _odd *next;
17
+ const char * classname;
18
+ size_t clen;
19
+ VALUE clas; // Ruby class or module
20
+ VALUE create_obj;
21
+ ID create_op;
22
+ int attr_cnt;
23
+ bool is_module;
24
+ bool raw;
25
+ const char * attr_names[MAX_ODD_ARGS]; // NULL terminated attr names
26
+ ID attrs[MAX_ODD_ARGS]; // 0 terminated attr IDs
27
+ AttrGetFunc attrFuncs[MAX_ODD_ARGS];
28
+ } * Odd;
29
+
30
+ typedef struct _oddArgs {
31
+ Odd odd;
32
+ VALUE args[MAX_ODD_ARGS];
33
+ } * OddArgs;
34
+
35
+ extern void oj_odd_init(void);
36
+ extern Odd oj_get_odd(VALUE clas);
37
+ extern Odd oj_get_oddc(const char *classname, size_t len);
38
+ extern OddArgs oj_odd_alloc_args(Odd odd);
39
+ extern void oj_odd_free(OddArgs args);
40
+ extern int oj_odd_set_arg(OddArgs args, const char *key, size_t klen, VALUE value);
41
+ extern void oj_reg_odd(VALUE clas, VALUE create_object, VALUE create_method, int mcnt, VALUE *members, bool raw);
42
+
43
+ #endif /* OJ_ODD_H */