iodine 0.7.3 → 0.7.4
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 iodine might be problematic. Click here for more details.
- checksums.yaml +4 -4
- data/CHANGELOG.md +12 -2
- data/ext/iodine/fio.c +319 -46
- data/ext/iodine/fio.h +851 -37
- data/ext/iodine/fio_cli.c +17 -16
- data/ext/iodine/fio_json_parser.h +28 -3
- data/ext/iodine/fiobj_ary.c +44 -46
- data/ext/iodine/fiobj_data.c +12 -13
- data/ext/iodine/fiobj_hash.c +28 -27
- data/ext/iodine/fiobj_json.c +40 -33
- data/ext/iodine/fiobj_numbers.c +2 -304
- data/ext/iodine/fiobj_str.c +0 -1
- data/ext/iodine/fiobject.c +268 -38
- data/ext/iodine/iodine_json.c +13 -14
- data/ext/iodine/iodine_store.c +33 -29
- data/iodine.gemspec +1 -3
- data/lib/iodine.rb +2 -2
- data/lib/iodine/json.rb +3 -1
- data/lib/iodine/version.rb +1 -1
- metadata +5 -4
- data/ext/iodine/fio_ary.h +0 -717
data/ext/iodine/fiobj_str.c
CHANGED
data/ext/iodine/fiobject.c
CHANGED
@@ -7,20 +7,25 @@ License: MIT
|
|
7
7
|
This facil.io core library provides wrappers around complex and (or) dynamic
|
8
8
|
types, abstracting some complexity and making dynamic type related tasks easier.
|
9
9
|
*/
|
10
|
+
|
10
11
|
#include <fiobject.h>
|
11
|
-
#undef FIO_FORCE_MALLOC /* just in case */
|
12
|
-
#include <fio.h>
|
13
12
|
|
14
|
-
#
|
13
|
+
#define FIO_ARY_NAME fiobj_stack
|
14
|
+
#define FIO_ARY_TYPE FIOBJ
|
15
|
+
#define FIO_ARY_INVALID FIOBJ_INVALID
|
16
|
+
/* don't free or compare objects, this stack shouldn't have side-effects */
|
17
|
+
#include <fio.h>
|
15
18
|
|
16
19
|
#include <stdarg.h>
|
17
20
|
#include <stdint.h>
|
18
21
|
#include <stdio.h>
|
19
22
|
#include <stdlib.h>
|
23
|
+
#include <string.h>
|
20
24
|
|
21
25
|
/* *****************************************************************************
|
22
|
-
Use the facil.io
|
26
|
+
Use the facil.io features when available, but override when missing.
|
23
27
|
***************************************************************************** */
|
28
|
+
#ifndef fd_data /* defined in fio.c */
|
24
29
|
|
25
30
|
#pragma weak fio_malloc
|
26
31
|
void *fio_malloc(size_t size) {
|
@@ -62,16 +67,236 @@ size_t __attribute__((weak)) FIO_LOG_LEVEL = FIO_LOG_LEVEL_DEBUG;
|
|
62
67
|
size_t __attribute__((weak)) FIO_LOG_LEVEL = FIO_LOG_LEVEL_INFO;
|
63
68
|
#endif
|
64
69
|
|
65
|
-
|
66
|
-
|
70
|
+
/**
|
71
|
+
* We include this in case the parser is used outside of facil.io.
|
72
|
+
*/
|
73
|
+
int64_t __attribute__((weak)) fio_atol(char **pstr) {
|
74
|
+
return strtoll(*pstr, pstr, 0);
|
75
|
+
}
|
76
|
+
#pragma weak fio_atol
|
77
|
+
|
78
|
+
/**
|
79
|
+
* We include this in case the parser is used outside of facil.io.
|
80
|
+
*/
|
81
|
+
double __attribute__((weak)) fio_atof(char **pstr) {
|
82
|
+
return strtod(*pstr, pstr);
|
83
|
+
}
|
84
|
+
#pragma weak fio_atof
|
67
85
|
|
86
|
+
/**
|
87
|
+
* A helper function that writes a signed int64_t to a string.
|
88
|
+
*
|
89
|
+
* No overflow guard is provided, make sure there's at least 68 bytes
|
90
|
+
* available (for base 2).
|
91
|
+
*
|
92
|
+
* Offers special support for base 2 (binary), base 8 (octal), base 10 and base
|
93
|
+
* 16 (hex). An unsupported base will silently default to base 10. Prefixes
|
94
|
+
* are automatically added (i.e., "0x" for hex and "0b" for base 2).
|
95
|
+
*
|
96
|
+
* Returns the number of bytes actually written (excluding the NUL
|
97
|
+
* terminator).
|
98
|
+
*/
|
99
|
+
#pragma weak fio_ltoa
|
100
|
+
size_t __attribute__((weak)) fio_ltoa(char *dest, int64_t num, uint8_t base) {
|
101
|
+
const char notation[] = {'0', '1', '2', '3', '4', '5', '6', '7',
|
102
|
+
'8', '9', 'A', 'B', 'C', 'D', 'E', 'F'};
|
103
|
+
|
104
|
+
size_t len = 0;
|
105
|
+
char buf[48]; /* we only need up to 20 for base 10, but base 3 needs 41... */
|
106
|
+
|
107
|
+
if (!num)
|
108
|
+
goto zero;
|
109
|
+
|
110
|
+
switch (base) {
|
111
|
+
case 1:
|
112
|
+
case 2:
|
113
|
+
/* Base 2 */
|
114
|
+
{
|
115
|
+
uint64_t n = num; /* avoid bit shifting inconsistencies with signed bit */
|
116
|
+
uint8_t i = 0; /* counting bits */
|
117
|
+
dest[len++] = '0';
|
118
|
+
dest[len++] = 'b';
|
119
|
+
|
120
|
+
while ((i < 64) && (n & 0x8000000000000000) == 0) {
|
121
|
+
n = n << 1;
|
122
|
+
i++;
|
123
|
+
}
|
124
|
+
/* make sure the Binary representation doesn't appear signed. */
|
125
|
+
if (i) {
|
126
|
+
dest[len++] = '0';
|
127
|
+
}
|
128
|
+
/* write to dest. */
|
129
|
+
while (i < 64) {
|
130
|
+
dest[len++] = ((n & 0x8000000000000000) ? '1' : '0');
|
131
|
+
n = n << 1;
|
132
|
+
i++;
|
133
|
+
}
|
134
|
+
dest[len] = 0;
|
135
|
+
return len;
|
136
|
+
}
|
137
|
+
case 8:
|
138
|
+
/* Base 8 */
|
139
|
+
{
|
140
|
+
uint64_t l = 0;
|
141
|
+
if (num < 0) {
|
142
|
+
dest[len++] = '-';
|
143
|
+
num = 0 - num;
|
144
|
+
}
|
145
|
+
dest[len++] = '0';
|
146
|
+
|
147
|
+
while (num) {
|
148
|
+
buf[l++] = '0' + (num & 7);
|
149
|
+
num = num >> 3;
|
150
|
+
}
|
151
|
+
while (l) {
|
152
|
+
--l;
|
153
|
+
dest[len++] = buf[l];
|
154
|
+
}
|
155
|
+
dest[len] = 0;
|
156
|
+
return len;
|
157
|
+
}
|
158
|
+
|
159
|
+
case 16:
|
160
|
+
/* Base 16 */
|
161
|
+
{
|
162
|
+
uint64_t n = num; /* avoid bit shifting inconsistencies with signed bit */
|
163
|
+
uint8_t i = 0; /* counting bits */
|
164
|
+
dest[len++] = '0';
|
165
|
+
dest[len++] = 'x';
|
166
|
+
while (i < 8 && (n & 0xFF00000000000000) == 0) {
|
167
|
+
n = n << 8;
|
168
|
+
i++;
|
169
|
+
}
|
170
|
+
/* make sure the Hex representation doesn't appear signed. */
|
171
|
+
if (i && (n & 0x8000000000000000)) {
|
172
|
+
dest[len++] = '0';
|
173
|
+
dest[len++] = '0';
|
174
|
+
}
|
175
|
+
/* write the damn thing */
|
176
|
+
while (i < 8) {
|
177
|
+
uint8_t tmp = (n & 0xF000000000000000) >> 60;
|
178
|
+
dest[len++] = notation[tmp];
|
179
|
+
tmp = (n & 0x0F00000000000000) >> 56;
|
180
|
+
dest[len++] = notation[tmp];
|
181
|
+
i++;
|
182
|
+
n = n << 8;
|
183
|
+
}
|
184
|
+
dest[len] = 0;
|
185
|
+
return len;
|
186
|
+
}
|
187
|
+
case 3:
|
188
|
+
case 4:
|
189
|
+
case 5:
|
190
|
+
case 6:
|
191
|
+
case 7:
|
192
|
+
case 9:
|
193
|
+
/* rare bases */
|
194
|
+
if (num < 0) {
|
195
|
+
dest[len++] = '-';
|
196
|
+
num = 0 - num;
|
197
|
+
}
|
198
|
+
uint64_t l = 0;
|
199
|
+
while (num) {
|
200
|
+
uint64_t t = num / base;
|
201
|
+
buf[l++] = '0' + (num - (t * base));
|
202
|
+
num = t;
|
203
|
+
}
|
204
|
+
while (l) {
|
205
|
+
--l;
|
206
|
+
dest[len++] = buf[l];
|
207
|
+
}
|
208
|
+
dest[len] = 0;
|
209
|
+
return len;
|
210
|
+
|
211
|
+
default:
|
212
|
+
break;
|
213
|
+
}
|
214
|
+
/* Base 10, the default base */
|
215
|
+
|
216
|
+
if (num < 0) {
|
217
|
+
dest[len++] = '-';
|
218
|
+
num = 0 - num;
|
219
|
+
}
|
220
|
+
uint64_t l = 0;
|
221
|
+
while (num) {
|
222
|
+
uint64_t t = num / 10;
|
223
|
+
buf[l++] = '0' + (num - (t * 10));
|
224
|
+
num = t;
|
225
|
+
}
|
226
|
+
while (l) {
|
227
|
+
--l;
|
228
|
+
dest[len++] = buf[l];
|
229
|
+
}
|
230
|
+
dest[len] = 0;
|
231
|
+
return len;
|
232
|
+
|
233
|
+
zero:
|
234
|
+
switch (base) {
|
235
|
+
case 1:
|
236
|
+
case 2:
|
237
|
+
dest[len++] = '0';
|
238
|
+
dest[len++] = 'b';
|
239
|
+
case 16:
|
240
|
+
dest[len++] = '0';
|
241
|
+
dest[len++] = 'x';
|
242
|
+
dest[len++] = '0';
|
243
|
+
}
|
244
|
+
dest[len++] = '0';
|
245
|
+
dest[len] = 0;
|
246
|
+
return len;
|
247
|
+
}
|
248
|
+
|
249
|
+
/**
|
250
|
+
* A helper function that converts between a double to a string.
|
251
|
+
*
|
252
|
+
* No overflow guard is provided, make sure there's at least 130 bytes
|
253
|
+
* available (for base 2).
|
254
|
+
*
|
255
|
+
* Supports base 2, base 10 and base 16. An unsupported base will silently
|
256
|
+
* default to base 10. Prefixes aren't added (i.e., no "0x" or "0b" at the
|
257
|
+
* beginning of the string).
|
258
|
+
*
|
259
|
+
* Returns the number of bytes actually written (excluding the NUL
|
260
|
+
* terminator).
|
261
|
+
*/
|
262
|
+
#pragma weak fio_ftoa
|
263
|
+
size_t __attribute__((weak)) fio_ftoa(char *dest, double num, uint8_t base) {
|
264
|
+
if (base == 2 || base == 16) {
|
265
|
+
/* handle the binary / Hex representation the same as if it were an
|
266
|
+
* int64_t
|
267
|
+
*/
|
268
|
+
int64_t *i = (void *)#
|
269
|
+
return fio_ltoa(dest, *i, base);
|
270
|
+
}
|
271
|
+
|
272
|
+
size_t written = sprintf(dest, "%g", num);
|
273
|
+
uint8_t need_zero = 1;
|
274
|
+
char *start = dest;
|
275
|
+
while (*start) {
|
276
|
+
if (*start == ',') // locale issues?
|
277
|
+
*start = '.';
|
278
|
+
if (*start == '.' || *start == 'e') {
|
279
|
+
need_zero = 0;
|
280
|
+
break;
|
281
|
+
}
|
282
|
+
start++;
|
283
|
+
}
|
284
|
+
if (need_zero) {
|
285
|
+
dest[written++] = '.';
|
286
|
+
dest[written++] = '0';
|
287
|
+
}
|
288
|
+
return written;
|
289
|
+
}
|
290
|
+
|
291
|
+
#endif
|
68
292
|
/* *****************************************************************************
|
69
293
|
the `fiobj_each2` function
|
70
294
|
***************************************************************************** */
|
295
|
+
|
71
296
|
struct task_packet_s {
|
72
297
|
int (*task)(FIOBJ obj, void *arg);
|
73
298
|
void *arg;
|
74
|
-
|
299
|
+
fiobj_stack_s *stack;
|
75
300
|
FIOBJ next;
|
76
301
|
uintptr_t counter;
|
77
302
|
uint8_t stop;
|
@@ -121,7 +346,7 @@ size_t fiobj_each2(FIOBJ o, int (*task)(FIOBJ obj, void *arg), void *arg) {
|
|
121
346
|
if (task(o, arg) == -1)
|
122
347
|
return 1;
|
123
348
|
uintptr_t pos = 0;
|
124
|
-
|
349
|
+
fiobj_stack_s stack = FIO_ARY_INIT;
|
125
350
|
struct task_packet_s packet = {
|
126
351
|
.task = task,
|
127
352
|
.arg = arg,
|
@@ -136,19 +361,20 @@ size_t fiobj_each2(FIOBJ o, int (*task)(FIOBJ obj, void *arg), void *arg) {
|
|
136
361
|
if (packet.stop)
|
137
362
|
goto finish;
|
138
363
|
if (packet.incomplete) {
|
139
|
-
|
140
|
-
|
364
|
+
fiobj_stack_push(&stack, pos);
|
365
|
+
fiobj_stack_push(&stack, o);
|
141
366
|
}
|
142
367
|
|
143
368
|
if (packet.next) {
|
144
|
-
|
145
|
-
|
369
|
+
fiobj_stack_push(&stack, (FIOBJ)0);
|
370
|
+
fiobj_stack_push(&stack, packet.next);
|
146
371
|
}
|
147
|
-
o =
|
148
|
-
|
372
|
+
o = FIOBJ_INVALID;
|
373
|
+
fiobj_stack_pop(&stack, &o);
|
374
|
+
fiobj_stack_pop(&stack, &pos);
|
149
375
|
} while (o);
|
150
376
|
finish:
|
151
|
-
|
377
|
+
fiobj_stack_free(&stack);
|
152
378
|
return packet.counter;
|
153
379
|
}
|
154
380
|
|
@@ -169,8 +395,8 @@ static void fiobj_dealloc_task(FIOBJ o, void *stack_) {
|
|
169
395
|
FIOBJECT2VTBL(o)->dealloc(o, NULL, NULL);
|
170
396
|
return;
|
171
397
|
}
|
172
|
-
|
173
|
-
|
398
|
+
fiobj_stack_s *s = stack_;
|
399
|
+
fiobj_stack_push(s, o);
|
174
400
|
}
|
175
401
|
/**
|
176
402
|
* Decreases an object's reference count, releasing memory and
|
@@ -181,11 +407,11 @@ static void fiobj_dealloc_task(FIOBJ o, void *stack_) {
|
|
181
407
|
* also freed.
|
182
408
|
*/
|
183
409
|
void fiobj_free_complex_object(FIOBJ o) {
|
184
|
-
|
410
|
+
fiobj_stack_s stack = FIO_ARY_INIT;
|
185
411
|
do {
|
186
412
|
FIOBJECT2VTBL(o)->dealloc(o, fiobj_dealloc_task, &stack);
|
187
|
-
} while ((
|
188
|
-
|
413
|
+
} while (!fiobj_stack_pop(&stack, &o));
|
414
|
+
fiobj_stack_free(&stack);
|
189
415
|
}
|
190
416
|
|
191
417
|
/* *****************************************************************************
|
@@ -208,10 +434,10 @@ static inline int fiobj_iseq_simple(const FIOBJ o, const FIOBJ o2) {
|
|
208
434
|
}
|
209
435
|
|
210
436
|
static int fiobj_iseq____internal_complex__task(FIOBJ o, void *ary_) {
|
211
|
-
|
212
|
-
|
437
|
+
fiobj_stack_s *ary = ary_;
|
438
|
+
fiobj_stack_push(ary, o);
|
213
439
|
if (fiobj_hash_key_in_loop())
|
214
|
-
|
440
|
+
fiobj_stack_push(ary, fiobj_hash_key_in_loop());
|
215
441
|
return 0;
|
216
442
|
}
|
217
443
|
|
@@ -220,34 +446,38 @@ int fiobj_iseq____internal_complex__(FIOBJ o, FIOBJ o2) {
|
|
220
446
|
// if (FIOBJECT2VTBL(o)->each && FIOBJECT2VTBL(o)->count(o))
|
221
447
|
// return int fiobj_iseq____internal_complex__(const FIOBJ o, const FIOBJ
|
222
448
|
// o2);
|
223
|
-
|
449
|
+
fiobj_stack_s left = FIO_ARY_INIT, right = FIO_ARY_INIT, queue = FIO_ARY_INIT;
|
224
450
|
do {
|
225
451
|
fiobj_each1(o, 0, fiobj_iseq____internal_complex__task, &left);
|
226
452
|
fiobj_each1(o2, 0, fiobj_iseq____internal_complex__task, &right);
|
227
|
-
while (
|
228
|
-
o =
|
229
|
-
o2 =
|
453
|
+
while (fiobj_stack_count(&left)) {
|
454
|
+
o = FIOBJ_INVALID;
|
455
|
+
o2 = FIOBJ_INVALID;
|
456
|
+
fiobj_stack_pop(&left, &o);
|
457
|
+
fiobj_stack_pop(&right, &o2);
|
230
458
|
if (!fiobj_iseq_simple(o, o2))
|
231
459
|
goto unequal;
|
232
460
|
if (FIOBJ_IS_ALLOCATED(o) && FIOBJECT2VTBL(o)->each &&
|
233
461
|
FIOBJECT2VTBL(o)->count(o)) {
|
234
|
-
|
235
|
-
|
462
|
+
fiobj_stack_push(&queue, o);
|
463
|
+
fiobj_stack_push(&queue, o2);
|
236
464
|
}
|
237
465
|
}
|
238
|
-
|
239
|
-
|
466
|
+
o = FIOBJ_INVALID;
|
467
|
+
o2 = FIOBJ_INVALID;
|
468
|
+
fiobj_stack_pop(&queue, &o2);
|
469
|
+
fiobj_stack_pop(&queue, &o);
|
240
470
|
if (!fiobj_iseq_simple(o, o2))
|
241
471
|
goto unequal;
|
242
472
|
} while (o);
|
243
|
-
|
244
|
-
|
245
|
-
|
473
|
+
fiobj_stack_free(&left);
|
474
|
+
fiobj_stack_free(&right);
|
475
|
+
fiobj_stack_free(&queue);
|
246
476
|
return 1;
|
247
477
|
unequal:
|
248
|
-
|
249
|
-
|
250
|
-
|
478
|
+
fiobj_stack_free(&left);
|
479
|
+
fiobj_stack_free(&right);
|
480
|
+
fiobj_stack_free(&queue);
|
251
481
|
return 0;
|
252
482
|
}
|
253
483
|
|
@@ -262,7 +492,7 @@ void fiobject___noop_dealloc(FIOBJ o, void (*task)(FIOBJ, void *), void *arg) {
|
|
262
492
|
}
|
263
493
|
void fiobject___simple_dealloc(FIOBJ o, void (*task)(FIOBJ, void *),
|
264
494
|
void *arg) {
|
265
|
-
|
495
|
+
fio_free(FIOBJ2PTR(o));
|
266
496
|
(void)task;
|
267
497
|
(void)arg;
|
268
498
|
}
|
data/ext/iodine/iodine_json.c
CHANGED
@@ -2,7 +2,6 @@
|
|
2
2
|
|
3
3
|
#include "fio.h"
|
4
4
|
|
5
|
-
#include "fio_ary.h"
|
6
5
|
#include "fio_json_parser.h"
|
7
6
|
#include "fiobj.h"
|
8
7
|
#include "iodine_fiobj2rb.h"
|
@@ -15,6 +14,9 @@ static VALUE create_additions;
|
|
15
14
|
static VALUE object_class;
|
16
15
|
static VALUE array_class;
|
17
16
|
|
17
|
+
#define FIO_ARY_NAME fio_json_stack
|
18
|
+
#include "fio.h"
|
19
|
+
|
18
20
|
/* *****************************************************************************
|
19
21
|
JSON Callacks - these must be implemented in the C file that uses the parser
|
20
22
|
***************************************************************************** */
|
@@ -28,7 +30,7 @@ typedef struct {
|
|
28
30
|
VALUE key;
|
29
31
|
VALUE top;
|
30
32
|
VALUE target;
|
31
|
-
|
33
|
+
fio_json_stack_s stack;
|
32
34
|
uint8_t is_hash;
|
33
35
|
uint8_t symbolize;
|
34
36
|
} iodine_json_parser_s;
|
@@ -41,9 +43,6 @@ static inline void iodine_json_add2parser(iodine_json_parser_s *p, VALUE o) {
|
|
41
43
|
IodineStore.remove(p->key);
|
42
44
|
p->key = (VALUE)0;
|
43
45
|
} else {
|
44
|
-
// if (p->symbolize) {
|
45
|
-
// o = rb_to_symbol(o);
|
46
|
-
// }
|
47
46
|
p->key = o;
|
48
47
|
IodineStore.add(o);
|
49
48
|
}
|
@@ -98,13 +97,13 @@ static int fio_json_on_start_object(json_parser_s *p) {
|
|
98
97
|
iodine_json_parser_s *pr = (iodine_json_parser_s *)p;
|
99
98
|
if (pr->target) {
|
100
99
|
/* push NULL, don't free the objects */
|
101
|
-
|
100
|
+
fio_json_stack_push(&pr->stack, (void *)pr->top);
|
102
101
|
pr->top = pr->target;
|
103
102
|
pr->target = 0;
|
104
103
|
} else {
|
105
104
|
VALUE h = rb_hash_new();
|
106
105
|
iodine_json_add2parser(pr, h);
|
107
|
-
|
106
|
+
fio_json_stack_push(&pr->stack, (void *)pr->top);
|
108
107
|
pr->top = h;
|
109
108
|
}
|
110
109
|
pr->is_hash = 1;
|
@@ -119,7 +118,7 @@ static void fio_json_on_end_object(json_parser_s *p) {
|
|
119
118
|
IodineStore.remove(pr->key);
|
120
119
|
pr->key = (VALUE)0;
|
121
120
|
}
|
122
|
-
pr->
|
121
|
+
fio_json_stack_pop(&pr->stack, &pr->top);
|
123
122
|
pr->is_hash = (TYPE(pr->top) == T_HASH);
|
124
123
|
}
|
125
124
|
/** an array object was detected */
|
@@ -129,7 +128,7 @@ static int fio_json_on_start_array(json_parser_s *p) {
|
|
129
128
|
return -1;
|
130
129
|
VALUE ary = rb_ary_new();
|
131
130
|
iodine_json_add2parser(pr, ary);
|
132
|
-
|
131
|
+
fio_json_stack_push(&pr->stack, (void *)pr->top);
|
133
132
|
pr->top = ary;
|
134
133
|
pr->is_hash = 0;
|
135
134
|
return 0;
|
@@ -137,7 +136,7 @@ static int fio_json_on_start_array(json_parser_s *p) {
|
|
137
136
|
/** an array closure was detected */
|
138
137
|
static void fio_json_on_end_array(json_parser_s *p) {
|
139
138
|
iodine_json_parser_s *pr = (iodine_json_parser_s *)p;
|
140
|
-
pr->
|
139
|
+
fio_json_stack_pop(&pr->stack, &pr->top);
|
141
140
|
pr->is_hash = (TYPE(pr->top) == T_HASH);
|
142
141
|
}
|
143
142
|
/** the JSON parsing is complete */
|
@@ -148,9 +147,9 @@ static void fio_json_on_error(json_parser_s *p) {
|
|
148
147
|
#if DEBUG
|
149
148
|
FIO_LOG_ERROR("JSON on error called.");
|
150
149
|
#endif
|
151
|
-
IodineStore.remove((VALUE)
|
150
|
+
IodineStore.remove((VALUE)fio_json_stack_get(&pr->stack, 0));
|
152
151
|
IodineStore.remove(pr->key);
|
153
|
-
|
152
|
+
fio_json_stack_free(&pr->stack);
|
154
153
|
*pr = (iodine_json_parser_s){.top = 0};
|
155
154
|
}
|
156
155
|
|
@@ -163,10 +162,10 @@ static inline VALUE iodine_json_convert(VALUE str, fiobj2rb_settings_s s) {
|
|
163
162
|
iodine_json_parser_s p = {.top = 0, .symbolize = s.str2sym};
|
164
163
|
size_t consumed = fio_json_parse(&p.p, RSTRING_PTR(str), RSTRING_LEN(str));
|
165
164
|
if (!consumed || p.p.depth) {
|
166
|
-
IodineStore.remove((VALUE)
|
165
|
+
IodineStore.remove((VALUE)fio_json_stack_get(&p.stack, 0));
|
167
166
|
p.top = FIOBJ_INVALID;
|
168
167
|
}
|
169
|
-
|
168
|
+
fio_json_stack_free(&p.stack);
|
170
169
|
if (p.key) {
|
171
170
|
IodineStore.remove((VALUE)p.key);
|
172
171
|
}
|