oj 2.18.3 → 3.13.14

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (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 */