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.

@@ -29,7 +29,6 @@ License: MIT
29
29
  #include <string.h>
30
30
  #include <sys/stat.h>
31
31
 
32
- #define FIO_OVERRIDE_MALLOC 1
33
32
  #define FIO_INCLUDE_STR
34
33
  #define FIO_STR_NO_REF
35
34
  #include <fio.h>
@@ -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
- #include <fio_ary.h>
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 allocator when available
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
- #define FIO_OVERRIDE_MALLOC 1
66
- #include <fio.h>
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 *)&num;
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
- fio_ary_s *stack;
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
- fio_ary_s stack = FIO_ARY_INIT;
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
- fio_ary_push(&stack, (void *)pos);
140
- fio_ary_push(&stack, (void *)o);
364
+ fiobj_stack_push(&stack, pos);
365
+ fiobj_stack_push(&stack, o);
141
366
  }
142
367
 
143
368
  if (packet.next) {
144
- fio_ary_push(&stack, (void *)0);
145
- fio_ary_push(&stack, (void *)packet.next);
369
+ fiobj_stack_push(&stack, (FIOBJ)0);
370
+ fiobj_stack_push(&stack, packet.next);
146
371
  }
147
- o = (FIOBJ)fio_ary_pop(&stack);
148
- pos = (uintptr_t)fio_ary_pop(&stack);
372
+ o = FIOBJ_INVALID;
373
+ fiobj_stack_pop(&stack, &o);
374
+ fiobj_stack_pop(&stack, &pos);
149
375
  } while (o);
150
376
  finish:
151
- fio_ary_free(&stack);
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
- fio_ary_s *s = stack_;
173
- fio_ary_push(s, (void *)o);
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
- fio_ary_s stack = FIO_ARY_INIT;
410
+ fiobj_stack_s stack = FIO_ARY_INIT;
185
411
  do {
186
412
  FIOBJECT2VTBL(o)->dealloc(o, fiobj_dealloc_task, &stack);
187
- } while ((o = (FIOBJ)fio_ary_pop(&stack)));
188
- fio_ary_free(&stack);
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
- fio_ary_s *ary = ary_;
212
- fio_ary_push(ary, (void *)o);
437
+ fiobj_stack_s *ary = ary_;
438
+ fiobj_stack_push(ary, o);
213
439
  if (fiobj_hash_key_in_loop())
214
- fio_ary_push(ary, (void *)fiobj_hash_key_in_loop());
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
- fio_ary_s left = FIO_ARY_INIT, right = FIO_ARY_INIT, queue = FIO_ARY_INIT;
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 (fio_ary_count(&left)) {
228
- o = (FIOBJ)fio_ary_pop(&left);
229
- o2 = (FIOBJ)fio_ary_pop(&right);
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
- fio_ary_push(&queue, (void *)o);
235
- fio_ary_push(&queue, (void *)o2);
462
+ fiobj_stack_push(&queue, o);
463
+ fiobj_stack_push(&queue, o2);
236
464
  }
237
465
  }
238
- o2 = (FIOBJ)fio_ary_pop(&queue);
239
- o = (FIOBJ)fio_ary_pop(&queue);
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
- fio_ary_free(&left);
244
- fio_ary_free(&right);
245
- fio_ary_free(&queue);
473
+ fiobj_stack_free(&left);
474
+ fiobj_stack_free(&right);
475
+ fiobj_stack_free(&queue);
246
476
  return 1;
247
477
  unequal:
248
- fio_ary_free(&left);
249
- fio_ary_free(&right);
250
- fio_ary_free(&queue);
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
- free(FIOBJ2PTR(o));
495
+ fio_free(FIOBJ2PTR(o));
266
496
  (void)task;
267
497
  (void)arg;
268
498
  }
@@ -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
- fio_ary_s stack;
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
- fio_ary_push(&pr->stack, (void *)pr->top);
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
- fio_ary_push(&pr->stack, (void *)pr->top);
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->top = (VALUE)fio_ary_pop(&pr->stack);
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
- fio_ary_push(&pr->stack, (void *)pr->top);
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->top = (VALUE)fio_ary_pop(&pr->stack);
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)fio_ary_index(&pr->stack, 0));
150
+ IodineStore.remove((VALUE)fio_json_stack_get(&pr->stack, 0));
152
151
  IodineStore.remove(pr->key);
153
- fio_ary_free(&pr->stack);
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)fio_ary_index(&p.stack, 0));
165
+ IodineStore.remove((VALUE)fio_json_stack_get(&p.stack, 0));
167
166
  p.top = FIOBJ_INVALID;
168
167
  }
169
- fio_ary_free(&p.stack);
168
+ fio_json_stack_free(&p.stack);
170
169
  if (p.key) {
171
170
  IodineStore.remove((VALUE)p.key);
172
171
  }