ox 2.14.13 → 2.14.15

Sign up to get free protection for your applications and to get access to all the features.
data/ext/ox/dump.c CHANGED
@@ -3,84 +3,83 @@
3
3
  * All rights reserved.
4
4
  */
5
5
 
6
- #include <stdlib.h>
7
6
  #include <errno.h>
8
- #include <time.h>
9
7
  #include <stdio.h>
8
+ #include <stdlib.h>
10
9
  #include <string.h>
10
+ #include <time.h>
11
11
 
12
12
  #include "base64.h"
13
13
  #include "cache8.h"
14
14
  #include "ox.h"
15
15
 
16
- #define USE_B64 0
16
+ #define USE_B64 0
17
17
  #define MAX_DEPTH 1000
18
18
 
19
- typedef unsigned long ulong;
19
+ typedef unsigned long ulong;
20
20
 
21
21
  typedef struct _str {
22
- const char *str;
23
- size_t len;
22
+ const char *str;
23
+ size_t len;
24
24
  } *Str;
25
25
 
26
26
  typedef struct _element {
27
- struct _str clas;
28
- struct _str attr;
29
- unsigned long id;
30
- int indent; /* < 0 indicates no \n */
31
- int closed;
32
- char type;
27
+ struct _str clas;
28
+ struct _str attr;
29
+ unsigned long id;
30
+ int indent; /* < 0 indicates no \n */
31
+ int closed;
32
+ char type;
33
33
  } *Element;
34
34
 
35
35
  typedef struct _out {
36
- void (*w_start)(struct _out *out, Element e);
37
- void (*w_end)(struct _out *out, Element e);
38
- void (*w_time)(struct _out *out, VALUE obj);
39
- char *buf;
40
- char *end;
41
- char *cur;
42
- Cache8 circ_cache;
43
- unsigned long circ_cnt;
44
- int indent;
45
- int depth; /* used by dumpHash */
46
- Options opts;
47
- VALUE obj;
36
+ void (*w_start)(struct _out *out, Element e);
37
+ void (*w_end)(struct _out *out, Element e);
38
+ void (*w_time)(struct _out *out, VALUE obj);
39
+ char *buf;
40
+ char *end;
41
+ char *cur;
42
+ Cache8 circ_cache;
43
+ unsigned long circ_cnt;
44
+ int indent;
45
+ int depth; /* used by dumpHash */
46
+ Options opts;
47
+ VALUE obj;
48
48
  } *Out;
49
49
 
50
- static void dump_obj_to_xml(VALUE obj, Options copts, Out out);
50
+ static void dump_obj_to_xml(VALUE obj, Options copts, Out out);
51
51
 
52
- static void dump_first_obj(VALUE obj, Out out);
53
- static void dump_obj(ID aid, VALUE obj, int depth, Out out);
54
- static void dump_gen_doc(VALUE obj, int depth, Out out);
55
- static void dump_gen_element(VALUE obj, int depth, Out out);
56
- static void dump_gen_instruct(VALUE obj, int depth, Out out);
57
- static int dump_gen_attr(VALUE key, VALUE value, VALUE ov);
58
- static int dump_gen_nodes(VALUE obj, int depth, Out out);
59
- static void dump_gen_val_node(VALUE obj, int depth,
60
- const char *pre, size_t plen,
61
- const char *suf, size_t slen, Out out);
52
+ static void dump_first_obj(VALUE obj, Out out);
53
+ static void dump_obj(ID aid, VALUE obj, int depth, Out out);
54
+ static void dump_gen_doc(VALUE obj, int depth, Out out);
55
+ static void dump_gen_element(VALUE obj, int depth, Out out);
56
+ static void dump_gen_instruct(VALUE obj, int depth, Out out);
57
+ static int dump_gen_attr(VALUE key, VALUE value, VALUE ov);
58
+ static int dump_gen_nodes(VALUE obj, int depth, Out out);
59
+ static void
60
+ dump_gen_val_node(VALUE obj, int depth, const char *pre, size_t plen, const char *suf, size_t slen, Out out);
62
61
 
63
- static void dump_start(Out out, Element e);
64
- static void dump_end(Out out, Element e);
62
+ static void dump_start(Out out, Element e);
63
+ static void dump_end(Out out, Element e);
65
64
 
66
- static void grow(Out out, size_t len);
65
+ static void grow(Out out, size_t len);
67
66
 
68
- static void dump_value(Out out, const char *value, size_t size);
69
- static void dump_str_value(Out out, const char *value, size_t size, const char *table);
70
- static int dump_var(ID key, VALUE value, VALUE ov);
71
- static void dump_num(Out out, VALUE obj);
72
- static void dump_date(Out out, VALUE obj);
73
- static void dump_time_thin(Out out, VALUE obj);
74
- static void dump_time_xsd(Out out, VALUE obj);
75
- static int dump_hash(VALUE key, VALUE value, VALUE ov);
67
+ static void dump_value(Out out, const char *value, size_t size);
68
+ static void dump_str_value(Out out, const char *value, size_t size, const char *table);
69
+ static int dump_var(ID key, VALUE value, VALUE ov);
70
+ static void dump_num(Out out, VALUE obj);
71
+ static void dump_date(Out out, VALUE obj);
72
+ static void dump_time_thin(Out out, VALUE obj);
73
+ static void dump_time_xsd(Out out, VALUE obj);
74
+ static int dump_hash(VALUE key, VALUE value, VALUE ov);
76
75
 
77
- static int is_xml_friendly(const uchar *str, int len, const char *table);
76
+ static int is_xml_friendly(const uchar *str, int len, const char *table);
78
77
 
79
- static const char hex_chars[17] = "0123456789abcdef";
78
+ static const char hex_chars[17] = "0123456789abcdef";
80
79
 
81
80
  // The : character is equivalent to 10. Used for replacement characters up to 10
82
81
  // characters long such as '&#x10FFFF;'.
83
- static const char xml_friendly_chars[257] = "\
82
+ static const char xml_friendly_chars[257] = "\
84
83
  :::::::::11::1::::::::::::::::::\
85
84
  11611156111111111111111111114141\
86
85
  11111111111111111111111111111111\
@@ -90,7 +89,7 @@ static const char xml_friendly_chars[257] = "\
90
89
  11111111111111111111111111111111\
91
90
  11111111111111111111111111111111";
92
91
 
93
- static const char xml_quote_chars[257] = "\
92
+ static const char xml_quote_chars[257] = "\
94
93
  :::::::::11::1::::::::::::::::::\
95
94
  11611151111111111111111111114141\
96
95
  11111111111111111111111111111111\
@@ -100,7 +99,7 @@ static const char xml_quote_chars[257] = "\
100
99
  11111111111111111111111111111111\
101
100
  11111111111111111111111111111111";
102
101
 
103
- static const char xml_element_chars[257] = "\
102
+ static const char xml_element_chars[257] = "\
104
103
  :::::::::11::1::::::::::::::::::\
105
104
  11111151111111111111111111114141\
106
105
  11111111111111111111111111111111\
@@ -110,916 +109,902 @@ static const char xml_element_chars[257] = "\
110
109
  11111111111111111111111111111111\
111
110
  11111111111111111111111111111111";
112
111
 
113
- inline static int
114
- is_xml_friendly(const uchar *str, int len, const char *table) {
112
+ inline static int is_xml_friendly(const uchar *str, int len, const char *table) {
115
113
  for (; 0 < len; str++, len--) {
116
- if ('1' != table[*str]) {
117
- return 0;
118
- }
114
+ if ('1' != table[*str]) {
115
+ return 0;
116
+ }
119
117
  }
120
118
  return 1;
121
119
  }
122
120
 
123
- inline static size_t
124
- xml_str_len(const uchar *str, size_t len, const char *table) {
125
- size_t size = 0;
121
+ inline static size_t xml_str_len(const uchar *str, size_t len, const char *table) {
122
+ size_t size = 0;
126
123
 
127
124
  for (; 0 < len; str++, len--) {
128
- size += xml_friendly_chars[*str];
125
+ size += xml_friendly_chars[*str];
129
126
  }
130
127
  return size - len * (size_t)'0';
131
128
  }
132
129
 
133
- inline static void
134
- dump_hex(uchar c, Out out) {
135
- uchar d = (c >> 4) & 0x0F;
130
+ inline static void dump_hex(uchar c, Out out) {
131
+ uchar d = (c >> 4) & 0x0F;
136
132
 
137
133
  *out->cur++ = hex_chars[d];
138
- d = c & 0x0F;
134
+ d = c & 0x0F;
139
135
  *out->cur++ = hex_chars[d];
140
136
  }
141
137
 
142
- static Type
143
- obj_class_code(VALUE obj) {
144
- VALUE clas = rb_obj_class(obj);
138
+ static Type obj_class_code(VALUE obj) {
139
+ VALUE clas = rb_obj_class(obj);
145
140
 
146
141
  switch (rb_type(obj)) {
147
- case T_NIL: return NilClassCode;
148
- case T_ARRAY: return ArrayCode;
149
- case T_HASH: return HashCode;
150
- case T_TRUE: return TrueClassCode;
151
- case T_FALSE: return FalseClassCode;
152
- case T_FIXNUM: return FixnumCode;
153
- case T_FLOAT: return FloatCode;
154
- case T_STRING: return (is_xml_friendly((uchar*)StringValuePtr(obj), (int)RSTRING_LEN(obj), xml_element_chars)) ? StringCode : String64Code;
155
- case T_SYMBOL:
156
- {
157
- const char *sym = rb_id2name(SYM2ID(obj));
158
-
159
- return (is_xml_friendly((uchar*)sym, (int)strlen(sym), xml_element_chars)) ? SymbolCode : Symbol64Code;
160
- }
161
- case T_DATA: return (rb_cTime == clas) ? TimeCode : ((ox_date_class == clas) ? DateCode : 0);
162
- case T_STRUCT: return (rb_cRange == clas) ? RangeCode : StructCode;
163
- case T_OBJECT: return (ox_document_clas == clas || ox_element_clas == clas) ? RawCode : ObjectCode;
164
- case T_REGEXP: return RegexpCode;
165
- case T_BIGNUM: return BignumCode;
142
+ case T_NIL: return NilClassCode;
143
+ case T_ARRAY: return ArrayCode;
144
+ case T_HASH: return HashCode;
145
+ case T_TRUE: return TrueClassCode;
146
+ case T_FALSE: return FalseClassCode;
147
+ case T_FIXNUM: return FixnumCode;
148
+ case T_FLOAT: return FloatCode;
149
+ case T_STRING:
150
+ return (is_xml_friendly((uchar *)StringValuePtr(obj), (int)RSTRING_LEN(obj), xml_element_chars)) ? StringCode
151
+ : String64Code;
152
+ case T_SYMBOL: {
153
+ const char *sym = rb_id2name(SYM2ID(obj));
154
+
155
+ return (is_xml_friendly((uchar *)sym, (int)strlen(sym), xml_element_chars)) ? SymbolCode : Symbol64Code;
156
+ }
157
+ case T_DATA: return (rb_cTime == clas) ? TimeCode : ((ox_date_class == clas) ? DateCode : 0);
158
+ case T_STRUCT: return (rb_cRange == clas) ? RangeCode : StructCode;
159
+ case T_OBJECT: return (ox_document_clas == clas || ox_element_clas == clas) ? RawCode : ObjectCode;
160
+ case T_REGEXP: return RegexpCode;
161
+ case T_BIGNUM: return BignumCode;
166
162
  #ifdef T_COMPLEX
167
- case T_COMPLEX: return ComplexCode;
163
+ case T_COMPLEX: return ComplexCode;
168
164
  #endif
169
165
  #ifdef T_RATIONAL
170
- case T_RATIONAL: return RationalCode;
166
+ case T_RATIONAL: return RationalCode;
171
167
  #endif
172
- case T_CLASS: return ClassCode;
173
- default: return 0;
168
+ case T_CLASS: return ClassCode;
169
+ default: return 0;
174
170
  }
175
171
  }
176
172
 
177
- inline static void
178
- fill_indent(Out out, int cnt) {
173
+ inline static void fill_indent(Out out, int cnt) {
179
174
  if (0 <= cnt) {
180
- *out->cur++ = '\n';
181
- if (0 < out->opts->margin_len) {
182
- memcpy(out->cur, out->opts->margin, out->opts->margin_len);
183
- out->cur += out->opts->margin_len;
184
- }
185
- for (; 0 < cnt; cnt--) {
186
- *out->cur++ = ' ';
187
- }
175
+ *out->cur++ = '\n';
176
+ if (0 < out->opts->margin_len) {
177
+ memcpy(out->cur, out->opts->margin, out->opts->margin_len);
178
+ out->cur += out->opts->margin_len;
179
+ }
180
+ for (; 0 < cnt; cnt--) {
181
+ *out->cur++ = ' ';
182
+ }
188
183
  }
189
184
  }
190
185
 
191
- inline static void
192
- fill_value(Out out, const char *value, size_t len) {
186
+ inline static void fill_value(Out out, const char *value, size_t len) {
193
187
  if (6 < len) {
194
- memcpy(out->cur, value, len);
195
- out->cur += len;
188
+ memcpy(out->cur, value, len);
189
+ out->cur += len;
196
190
  } else {
197
- for (; 0 < len; len--, value++) {
198
- *out->cur++ = *value;
199
- }
191
+ for (; 0 < len; len--, value++) {
192
+ *out->cur++ = *value;
193
+ }
200
194
  }
201
195
  }
202
196
 
203
- inline static void
204
- fill_attr(Out out, char name, const char *value, size_t len) {
197
+ inline static void fill_attr(Out out, char name, const char *value, size_t len) {
205
198
  *out->cur++ = ' ';
206
199
  *out->cur++ = name;
207
200
  *out->cur++ = '=';
208
201
  *out->cur++ = '"';
209
202
  if (6 < len) {
210
- memcpy(out->cur, value, len);
211
- out->cur += len;
203
+ memcpy(out->cur, value, len);
204
+ out->cur += len;
212
205
  } else {
213
- for (; 0 < len; len--, value++) {
214
- *out->cur++ = *value;
215
- }
206
+ for (; 0 < len; len--, value++) {
207
+ *out->cur++ = *value;
208
+ }
216
209
  }
217
210
  *out->cur++ = '"';
218
211
  }
219
212
 
220
- inline static const char*
221
- ulong2str(ulong num, char *end) {
222
- char *b;
213
+ inline static const char *ulong2str(ulong num, char *end) {
214
+ char *b;
223
215
 
224
216
  *end-- = '\0';
225
217
  for (b = end; 0 < num || b == end; num /= 10, b--) {
226
- *b = (num % 10) + '0';
218
+ *b = (num % 10) + '0';
227
219
  }
228
220
  b++;
229
221
 
230
222
  return b;
231
223
  }
232
224
 
233
- static int
234
- check_circular(Out out, VALUE obj, Element e) {
235
- slot_t *slot;
236
- slot_t id;
237
- int result;
225
+ static int check_circular(Out out, VALUE obj, Element e) {
226
+ slot_t *slot;
227
+ slot_t id;
228
+ int result;
238
229
 
239
230
  if (0 == (id = ox_cache8_get(out->circ_cache, obj, &slot))) {
240
- out->circ_cnt++;
241
- id = out->circ_cnt;
242
- *slot = id;
243
- e->id = id;
244
- result = 0;
231
+ out->circ_cnt++;
232
+ id = out->circ_cnt;
233
+ *slot = id;
234
+ e->id = id;
235
+ result = 0;
245
236
  } else {
246
- e->type = RefCode; e->clas.len = 0; e->clas.str = 0;
247
- e->closed = 1;
248
- e->id = id;
249
- out->w_start(out, e);
250
- result = 1;
237
+ e->type = RefCode;
238
+ e->clas.len = 0;
239
+ e->clas.str = 0;
240
+ e->closed = 1;
241
+ e->id = id;
242
+ out->w_start(out, e);
243
+ result = 1;
251
244
  }
252
245
  return result;
253
246
  }
254
247
 
255
- static void
256
- grow(Out out, size_t len) {
257
- size_t size = out->end - out->buf;
258
- long pos = out->cur - out->buf;
248
+ static void grow(Out out, size_t len) {
249
+ size_t size = out->end - out->buf;
250
+ long pos = out->cur - out->buf;
259
251
 
260
252
  size *= 2;
261
253
  if (size <= len * 2 + pos) {
262
- size += len;
254
+ size += len;
263
255
  }
264
256
  REALLOC_N(out->buf, char, size + 10); /* 10 extra for terminator character plus extra (paranoid) */
265
257
  out->end = out->buf + size;
266
258
  out->cur = out->buf + pos;
267
259
  }
268
260
 
269
- static void
270
- dump_start(Out out, Element e) {
271
- size_t size = e->indent + 4 + out->opts->margin_len;
261
+ static void dump_start(Out out, Element e) {
262
+ size_t size = e->indent + 4 + out->opts->margin_len;
272
263
 
273
264
  if (0 < e->attr.len) { /* a="attr" */
274
- size += e->attr.len + 5;
265
+ size += e->attr.len + 5;
275
266
  }
276
267
  if (0 < e->clas.len) { /* c="class" */
277
- size += e->clas.len + 5;
268
+ size += e->clas.len + 5;
278
269
  }
279
270
  if (0 < e->id) { /* i="id" */
280
- size += 24; /* over estimate, 19 digits */
271
+ size += 24; /* over estimate, 19 digits */
281
272
  }
282
273
  if (out->end - out->cur <= (long)size) {
283
- grow(out, size);
274
+ grow(out, size);
284
275
  }
285
276
  if (out->buf + out->opts->margin_len < out->cur) {
286
- fill_indent(out, e->indent);
277
+ fill_indent(out, e->indent);
287
278
  }
288
279
  *out->cur++ = '<';
289
280
  *out->cur++ = e->type;
290
281
  if (0 < e->attr.len) {
291
- fill_attr(out, 'a', e->attr.str, e->attr.len);
282
+ fill_attr(out, 'a', e->attr.str, e->attr.len);
292
283
  }
293
- if ((ObjectCode == e->type || ExceptionCode == e->type || StructCode == e->type || ClassCode == e->type) && 0 < e->clas.len) {
294
- fill_attr(out, 'c', e->clas.str, e->clas.len);
284
+ if ((ObjectCode == e->type || ExceptionCode == e->type || StructCode == e->type || ClassCode == e->type) &&
285
+ 0 < e->clas.len) {
286
+ fill_attr(out, 'c', e->clas.str, e->clas.len);
295
287
  }
296
288
  if (0 < e->id) {
297
- char buf[32];
298
- char *end = buf + sizeof(buf) - 1;
299
- const char *s = ulong2str(e->id, end);
289
+ char buf[32];
290
+ char *end = buf + sizeof(buf) - 1;
291
+ const char *s = ulong2str(e->id, end);
300
292
 
301
- fill_attr(out, 'i', s, end - s);
293
+ fill_attr(out, 'i', s, end - s);
302
294
  }
303
295
  if (e->closed) {
304
- if (out->opts->no_empty) {
305
- *out->cur++ = '>';
306
- *out->cur++ = '<';
307
- *out->cur++ = '/';
308
- *out->cur++ = e->type;
309
- } else {
310
- *out->cur++ = '/';
311
- }
296
+ if (out->opts->no_empty) {
297
+ *out->cur++ = '>';
298
+ *out->cur++ = '<';
299
+ *out->cur++ = '/';
300
+ *out->cur++ = e->type;
301
+ } else {
302
+ *out->cur++ = '/';
303
+ }
312
304
  }
313
305
  *out->cur++ = '>';
314
- *out->cur = '\0';
306
+ *out->cur = '\0';
315
307
  }
316
308
 
317
- static void
318
- dump_end(Out out, Element e) {
319
- size_t size = e->indent + 5 + out->opts->margin_len;
309
+ static void dump_end(Out out, Element e) {
310
+ size_t size = e->indent + 5 + out->opts->margin_len;
320
311
 
321
312
  if (out->end - out->cur <= (long)size) {
322
- grow(out, size);
313
+ grow(out, size);
323
314
  }
324
315
  fill_indent(out, e->indent);
325
316
  *out->cur++ = '<';
326
317
  *out->cur++ = '/';
327
318
  *out->cur++ = e->type;
328
319
  *out->cur++ = '>';
329
- *out->cur = '\0';
320
+ *out->cur = '\0';
330
321
  }
331
322
 
332
- inline static void
333
- dump_value(Out out, const char *value, size_t size) {
323
+ inline static void dump_value(Out out, const char *value, size_t size) {
334
324
  if (out->end - out->cur <= (long)size) {
335
- grow(out, size);
325
+ grow(out, size);
336
326
  }
337
327
  if (6 < size) {
338
- memcpy(out->cur, value, size);
339
- out->cur += size;
328
+ memcpy(out->cur, value, size);
329
+ out->cur += size;
340
330
  } else {
341
- for (; 0 < size; size--, value++) {
342
- *out->cur++ = *value;
343
- }
331
+ for (; 0 < size; size--, value++) {
332
+ *out->cur++ = *value;
333
+ }
344
334
  }
345
335
  *out->cur = '\0';
346
336
  }
347
337
 
348
- inline static void
349
- dump_str_value(Out out, const char *value, size_t size, const char *table) {
350
- size_t xsize = xml_str_len((const uchar*)value, size, table);
338
+ inline static void dump_str_value(Out out, const char *value, size_t size, const char *table) {
339
+ size_t xsize = xml_str_len((const uchar *)value, size, table);
351
340
 
352
341
  if (out->end - out->cur <= (long)xsize) {
353
- grow(out, xsize);
342
+ grow(out, xsize);
354
343
  }
355
344
  for (; 0 < size; size--, value++) {
356
- if ('1' == table[(uchar)*value]) {
357
- *out->cur++ = *value;
358
- } else {
359
- switch (*value) {
360
- case '"':
361
- *out->cur++ = '&';
362
- *out->cur++ = 'q';
363
- *out->cur++ = 'u';
364
- *out->cur++ = 'o';
365
- *out->cur++ = 't';
366
- *out->cur++ = ';';
367
- break;
368
- case '&':
369
- *out->cur++ = '&';
370
- *out->cur++ = 'a';
371
- *out->cur++ = 'm';
372
- *out->cur++ = 'p';
373
- *out->cur++ = ';';
374
- break;
375
- case '\'':
376
- *out->cur++ = '&';
377
- *out->cur++ = 'a';
378
- *out->cur++ = 'p';
379
- *out->cur++ = 'o';
380
- *out->cur++ = 's';
381
- *out->cur++ = ';';
382
- break;
383
- case '<':
384
- *out->cur++ = '&';
385
- *out->cur++ = 'l';
386
- *out->cur++ = 't';
387
- *out->cur++ = ';';
388
- break;
389
- case '>':
390
- *out->cur++ = '&';
391
- *out->cur++ = 'g';
392
- *out->cur++ = 't';
393
- *out->cur++ = ';';
394
- break;
395
- default:
396
- // Must be one of the invalid characters.
397
- if (StrictEffort == out->opts->effort) {
398
- rb_raise(ox_syntax_error_class, "'\\#x%02x' is not a valid XML character.", *value);
399
- }
400
- if (Yes == out->opts->allow_invalid) {
401
- *out->cur++ = '&';
402
- *out->cur++ = '#';
403
- *out->cur++ = 'x';
404
- *out->cur++ = '0';
405
- *out->cur++ = '0';
406
- dump_hex(*value, out);
407
- *out->cur++ = ';';
408
- } else if ('\0' != *out->opts->inv_repl) {
409
- // If the empty string then ignore. The first character of
410
- // the replacement is the length.
411
- memcpy(out->cur, out->opts->inv_repl + 1, (size_t)*out->opts->inv_repl);
412
- out->cur += *out->opts->inv_repl;
413
- }
414
- break;
415
- }
416
- }
345
+ if ('1' == table[(uchar)*value]) {
346
+ *out->cur++ = *value;
347
+ } else {
348
+ switch (*value) {
349
+ case '"':
350
+ *out->cur++ = '&';
351
+ *out->cur++ = 'q';
352
+ *out->cur++ = 'u';
353
+ *out->cur++ = 'o';
354
+ *out->cur++ = 't';
355
+ *out->cur++ = ';';
356
+ break;
357
+ case '&':
358
+ *out->cur++ = '&';
359
+ *out->cur++ = 'a';
360
+ *out->cur++ = 'm';
361
+ *out->cur++ = 'p';
362
+ *out->cur++ = ';';
363
+ break;
364
+ case '\'':
365
+ *out->cur++ = '&';
366
+ *out->cur++ = 'a';
367
+ *out->cur++ = 'p';
368
+ *out->cur++ = 'o';
369
+ *out->cur++ = 's';
370
+ *out->cur++ = ';';
371
+ break;
372
+ case '<':
373
+ *out->cur++ = '&';
374
+ *out->cur++ = 'l';
375
+ *out->cur++ = 't';
376
+ *out->cur++ = ';';
377
+ break;
378
+ case '>':
379
+ *out->cur++ = '&';
380
+ *out->cur++ = 'g';
381
+ *out->cur++ = 't';
382
+ *out->cur++ = ';';
383
+ break;
384
+ default:
385
+ // Must be one of the invalid characters.
386
+ if (StrictEffort == out->opts->effort) {
387
+ rb_raise(ox_syntax_error_class, "'\\#x%02x' is not a valid XML character.", *value);
388
+ }
389
+ if (Yes == out->opts->allow_invalid) {
390
+ *out->cur++ = '&';
391
+ *out->cur++ = '#';
392
+ *out->cur++ = 'x';
393
+ *out->cur++ = '0';
394
+ *out->cur++ = '0';
395
+ dump_hex(*value, out);
396
+ *out->cur++ = ';';
397
+ } else if ('\0' != *out->opts->inv_repl) {
398
+ // If the empty string then ignore. The first character of
399
+ // the replacement is the length.
400
+ memcpy(out->cur, out->opts->inv_repl + 1, (size_t)*out->opts->inv_repl);
401
+ out->cur += *out->opts->inv_repl;
402
+ }
403
+ break;
404
+ }
405
+ }
417
406
  }
418
407
  *out->cur = '\0';
419
408
  }
420
409
 
421
- inline static void
422
- dump_num(Out out, VALUE obj) {
423
- char buf[32];
424
- char *b = buf + sizeof(buf) - 1;
425
- long num = NUM2LONG(obj);
426
- int neg = 0;
410
+ inline static void dump_num(Out out, VALUE obj) {
411
+ char buf[32];
412
+ char *b = buf + sizeof(buf) - 1;
413
+ long num = NUM2LONG(obj);
414
+ int neg = 0;
427
415
 
428
416
  if (0 > num) {
429
- neg = 1;
430
- num = -num;
417
+ neg = 1;
418
+ num = -num;
431
419
  }
432
420
  *b-- = '\0';
433
421
  if (0 < num) {
434
- for (; 0 < num; num /= 10, b--) {
435
- *b = (num % 10) + '0';
436
- }
437
- if (neg) {
438
- *b = '-';
439
- } else {
440
- b++;
441
- }
422
+ for (; 0 < num; num /= 10, b--) {
423
+ *b = (num % 10) + '0';
424
+ }
425
+ if (neg) {
426
+ *b = '-';
427
+ } else {
428
+ b++;
429
+ }
442
430
  } else {
443
- *b = '0';
431
+ *b = '0';
444
432
  }
445
433
  if (out->end - out->cur <= (long)(sizeof(buf) - (b - buf))) {
446
- grow(out, sizeof(buf) - (b - buf));
434
+ grow(out, sizeof(buf) - (b - buf));
447
435
  }
448
436
  for (; '\0' != *b; b++) {
449
- *out->cur++ = *b;
437
+ *out->cur++ = *b;
450
438
  }
451
439
  *out->cur = '\0';
452
440
  }
453
441
 
454
- static void
455
- dump_time_thin(Out out, VALUE obj) {
456
- char buf[64];
457
- char *b = buf + sizeof(buf) - 1;
442
+ static void dump_time_thin(Out out, VALUE obj) {
443
+ char buf[64];
444
+ char *b = buf + sizeof(buf) - 1;
458
445
  #if HAVE_RB_TIME_TIMESPEC
459
- struct timespec ts = rb_time_timespec(obj);
460
- time_t sec = ts.tv_sec;
461
- long nsec = ts.tv_nsec;
446
+ struct timespec ts = rb_time_timespec(obj);
447
+ time_t sec = ts.tv_sec;
448
+ long nsec = ts.tv_nsec;
462
449
  #else
463
- time_t sec = NUM2LONG(rb_funcall2(obj, ox_tv_sec_id, 0, 0));
464
- long nsec = NUM2LONG(rb_funcall2(obj, ox_tv_nsec_id, 0, 0));
465
- //long nsec = NUM2LONG(rb_funcall2(obj, ox_tv_usec_id, 0, 0)) * 1000;
450
+ time_t sec = NUM2LONG(rb_funcall2(obj, ox_tv_sec_id, 0, 0));
451
+ long nsec = NUM2LONG(rb_funcall2(obj, ox_tv_nsec_id, 0, 0));
452
+ // long nsec = NUM2LONG(rb_funcall2(obj, ox_tv_usec_id, 0, 0)) * 1000;
466
453
  #endif
467
- char *dot = b - 10;
468
- long size;
454
+ char *dot = b - 10;
455
+ long size;
469
456
 
470
457
  *b-- = '\0';
471
458
  for (; dot < b; b--, nsec /= 10) {
472
- *b = '0' + (nsec % 10);
459
+ *b = '0' + (nsec % 10);
473
460
  }
474
461
  *b-- = '.';
475
462
  for (; 0 < sec; b--, sec /= 10) {
476
- *b = '0' + (sec % 10);
463
+ *b = '0' + (sec % 10);
477
464
  }
478
465
  b++;
479
466
  size = sizeof(buf) - (b - buf) - 1;
480
467
  if (out->end - out->cur <= size) {
481
- grow(out, size);
468
+ grow(out, size);
482
469
  }
483
470
  memcpy(out->cur, b, size);
484
471
  out->cur += size;
485
472
  }
486
473
 
487
- static void
488
- dump_date(Out out, VALUE obj) {
489
- char buf[64];
490
- char *b = buf + sizeof(buf) - 1;
491
- long jd = NUM2LONG(rb_funcall2(obj, ox_jd_id, 0, 0));
492
- long size;
474
+ static void dump_date(Out out, VALUE obj) {
475
+ char buf[64];
476
+ char *b = buf + sizeof(buf) - 1;
477
+ long jd = NUM2LONG(rb_funcall2(obj, ox_jd_id, 0, 0));
478
+ long size;
493
479
 
494
480
  *b-- = '\0';
495
481
  for (; 0 < jd; b--, jd /= 10) {
496
- *b = '0' + (jd % 10);
482
+ *b = '0' + (jd % 10);
497
483
  }
498
484
  b++;
499
485
  if ('\0' == *b) {
500
- b--;
501
- *b = '0';
486
+ b--;
487
+ *b = '0';
502
488
  }
503
489
  size = sizeof(buf) - (b - buf) - 1;
504
490
  if (out->end - out->cur <= size) {
505
- grow(out, size);
491
+ grow(out, size);
506
492
  }
507
493
  memcpy(out->cur, b, size);
508
494
  out->cur += size;
509
495
  }
510
496
 
511
- static void
512
- dump_time_xsd(Out out, VALUE obj) {
513
- struct tm *tm;
497
+ static void dump_time_xsd(Out out, VALUE obj) {
498
+ struct tm *tm;
514
499
  #if HAVE_RB_TIME_TIMESPEC
515
- struct timespec ts = rb_time_timespec(obj);
516
- time_t sec = ts.tv_sec;
517
- long nsec = ts.tv_nsec;
500
+ struct timespec ts = rb_time_timespec(obj);
501
+ time_t sec = ts.tv_sec;
502
+ long nsec = ts.tv_nsec;
518
503
  #else
519
- time_t sec = NUM2LONG(rb_funcall2(obj, ox_tv_sec_id, 0, 0));
520
- long nsec = NUM2LONG(rb_funcall2(obj, ox_tv_nsec_id, 0, 0));
521
- //long nsec = NUM2LONG(rb_funcall2(obj, ox_tv_usec_id, 0, 0)) * 1000;
504
+ time_t sec = NUM2LONG(rb_funcall2(obj, ox_tv_sec_id, 0, 0));
505
+ long nsec = NUM2LONG(rb_funcall2(obj, ox_tv_nsec_id, 0, 0));
506
+ // long nsec = NUM2LONG(rb_funcall2(obj, ox_tv_usec_id, 0, 0)) * 1000;
522
507
  #endif
523
- int tzhour, tzmin;
524
- char tzsign = '+';
508
+ int tzhour, tzmin;
509
+ char tzsign = '+';
525
510
 
526
511
  if (out->end - out->cur <= 33) {
527
- grow(out, 33);
512
+ grow(out, 33);
528
513
  }
529
514
  /* 2010-07-09T10:47:45.895826+09:00 */
530
515
  tm = localtime(&sec);
531
516
  #if HAVE_ST_TM_GMTOFF
532
517
  if (0 > tm->tm_gmtoff) {
533
- tzsign = '-';
534
- tzhour = (int)(tm->tm_gmtoff / -3600);
535
- tzmin = (int)(tm->tm_gmtoff / -60) - (tzhour * 60);
518
+ tzsign = '-';
519
+ tzhour = (int)(tm->tm_gmtoff / -3600);
520
+ tzmin = (int)(tm->tm_gmtoff / -60) - (tzhour * 60);
536
521
  } else {
537
- tzhour = (int)(tm->tm_gmtoff / 3600);
538
- tzmin = (int)(tm->tm_gmtoff / 60) - (tzhour * 60);
522
+ tzhour = (int)(tm->tm_gmtoff / 3600);
523
+ tzmin = (int)(tm->tm_gmtoff / 60) - (tzhour * 60);
539
524
  }
540
525
  #else
541
526
  tzhour = 0;
542
- tzmin = 0;
527
+ tzmin = 0;
543
528
  #endif
544
529
  /* TBD replace with more efficient printer */
545
- out->cur += sprintf(out->cur, "%04d-%02d-%02dT%02d:%02d:%02d.%06ld%c%02d:%02d",
546
- tm->tm_year + 1900, tm->tm_mon + 1, tm->tm_mday,
547
- tm->tm_hour, tm->tm_min, tm->tm_sec, nsec / 1000,
548
- tzsign, tzhour, tzmin);
530
+ out->cur += sprintf(out->cur,
531
+ "%04d-%02d-%02dT%02d:%02d:%02d.%06ld%c%02d:%02d",
532
+ tm->tm_year + 1900,
533
+ tm->tm_mon + 1,
534
+ tm->tm_mday,
535
+ tm->tm_hour,
536
+ tm->tm_min,
537
+ tm->tm_sec,
538
+ nsec / 1000,
539
+ tzsign,
540
+ tzhour,
541
+ tzmin);
549
542
  }
550
543
 
551
- static void
552
- dump_first_obj(VALUE obj, Out out) {
553
- char buf[128];
554
- Options copts = out->opts;
555
- int cnt;
544
+ static void dump_first_obj(VALUE obj, Out out) {
545
+ char buf[128];
546
+ Options copts = out->opts;
547
+ int cnt;
556
548
 
557
549
  if (Yes == copts->with_xml) {
558
- if (0 < copts->margin_len) {
559
- dump_value(out, copts->margin, copts->margin_len);
560
- }
561
- if ('\0' == *copts->encoding) {
562
- dump_value(out, "<?xml version=\"1.0\"?>", 21);
563
- } else {
564
- cnt = snprintf(buf, sizeof(buf), "<?xml version=\"1.0\" encoding=\"%s\"?>", copts->encoding);
565
- dump_value(out, buf, cnt);
566
- }
550
+ if (0 < copts->margin_len) {
551
+ dump_value(out, copts->margin, copts->margin_len);
552
+ }
553
+ if ('\0' == *copts->encoding) {
554
+ dump_value(out, "<?xml version=\"1.0\"?>", 21);
555
+ } else {
556
+ cnt = snprintf(buf, sizeof(buf), "<?xml version=\"1.0\" encoding=\"%s\"?>", copts->encoding);
557
+ dump_value(out, buf, cnt);
558
+ }
567
559
  }
568
560
  if (Yes == copts->with_instruct) {
569
- if (out->buf < out->cur) {
570
- dump_value(out, "\n", 1);
571
- }
572
- if (0 < copts->margin_len) {
573
- dump_value(out, copts->margin, copts->margin_len);
574
- }
575
- cnt = snprintf(buf, sizeof(buf), "<?ox version=\"1.0\" mode=\"object\"%s%s?>",
576
- (Yes == copts->circular) ? " circular=\"yes\"" : ((No == copts->circular) ? " circular=\"no\"" : ""),
577
- (Yes == copts->xsd_date) ? " xsd_date=\"yes\"" : ((No == copts->xsd_date) ? " xsd_date=\"no\"" : ""));
578
- dump_value(out, buf, cnt);
561
+ if (out->buf < out->cur) {
562
+ dump_value(out, "\n", 1);
563
+ }
564
+ if (0 < copts->margin_len) {
565
+ dump_value(out, copts->margin, copts->margin_len);
566
+ }
567
+ cnt = snprintf(
568
+ buf,
569
+ sizeof(buf),
570
+ "<?ox version=\"1.0\" mode=\"object\"%s%s?>",
571
+ (Yes == copts->circular) ? " circular=\"yes\"" : ((No == copts->circular) ? " circular=\"no\"" : ""),
572
+ (Yes == copts->xsd_date) ? " xsd_date=\"yes\"" : ((No == copts->xsd_date) ? " xsd_date=\"no\"" : ""));
573
+ dump_value(out, buf, cnt);
579
574
  }
580
575
  if (Yes == copts->with_dtd) {
581
- if (0 < copts->margin_len) {
582
- dump_value(out, copts->margin, copts->margin_len);
583
- }
584
- cnt = snprintf(buf, sizeof(buf), "%s<!DOCTYPE %c SYSTEM \"ox.dtd\">", (out->buf < out->cur) ? "\n" : "", obj_class_code(obj));
585
- dump_value(out, buf, cnt);
576
+ if (0 < copts->margin_len) {
577
+ dump_value(out, copts->margin, copts->margin_len);
578
+ }
579
+ cnt = snprintf(buf,
580
+ sizeof(buf),
581
+ "%s<!DOCTYPE %c SYSTEM \"ox.dtd\">",
582
+ (out->buf < out->cur) ? "\n" : "",
583
+ obj_class_code(obj));
584
+ dump_value(out, buf, cnt);
586
585
  }
587
586
  if (0 < copts->margin_len) {
588
- dump_value(out, copts->margin, copts->margin_len);
587
+ dump_value(out, copts->margin, copts->margin_len);
589
588
  }
590
589
  dump_obj(0, obj, 0, out);
591
590
  }
592
591
 
593
- static void
594
- dump_obj(ID aid, VALUE obj, int depth, Out out) {
595
- struct _element e;
596
- VALUE prev_obj = out->obj;
597
- char value_buf[64];
598
- int cnt;
592
+ static void dump_obj(ID aid, VALUE obj, int depth, Out out) {
593
+ struct _element e;
594
+ VALUE prev_obj = out->obj;
595
+ char value_buf[64];
596
+ int cnt;
599
597
 
600
598
  if (MAX_DEPTH < depth) {
601
- rb_raise(rb_eSysStackError, "maximum depth exceeded");
599
+ rb_raise(rb_eSysStackError, "maximum depth exceeded");
602
600
  }
603
601
  out->obj = obj;
604
602
  if (0 == aid) {
605
- e.attr.str = 0;
606
- e.attr.len = 0;
603
+ e.attr.str = 0;
604
+ e.attr.len = 0;
607
605
  } else {
608
- e.attr.str = rb_id2name(aid);
609
- // Ruby 2.3 started to return NULL for some IDs so check for
610
- // NULL. Ignore if NULL aid.
611
- if (NULL == e.attr.str) {
612
- return;
613
- }
614
- e.attr.len = strlen(e.attr.str);
606
+ e.attr.str = rb_id2name(aid);
607
+ // Ruby 2.3 started to return NULL for some IDs so check for
608
+ // NULL. Ignore if NULL aid.
609
+ if (NULL == e.attr.str) {
610
+ return;
611
+ }
612
+ e.attr.len = strlen(e.attr.str);
615
613
  }
616
614
  e.closed = 0;
617
615
  if (0 == depth) {
618
- e.indent = (0 <= out->indent) ? 0 : -1;
616
+ e.indent = (0 <= out->indent) ? 0 : -1;
619
617
  } else if (0 > out->indent) {
620
- e.indent = -1;
618
+ e.indent = -1;
621
619
  } else if (0 == out->indent) {
622
- e.indent = 0;
620
+ e.indent = 0;
623
621
  } else {
624
- e.indent = depth * out->indent;
622
+ e.indent = depth * out->indent;
625
623
  }
626
- e.id = 0;
624
+ e.id = 0;
627
625
  e.clas.len = 0;
628
626
  e.clas.str = 0;
629
627
  switch (rb_type(obj)) {
630
628
  case T_NIL:
631
- e.type = NilClassCode;
632
- e.closed = 1;
633
- out->w_start(out, &e);
634
- break;
629
+ e.type = NilClassCode;
630
+ e.closed = 1;
631
+ out->w_start(out, &e);
632
+ break;
635
633
  case T_ARRAY:
636
- if (0 != out->circ_cache && check_circular(out, obj, &e)) {
637
- break;
638
- }
639
- cnt = (int)RARRAY_LEN(obj);
640
- e.type = ArrayCode;
641
- e.closed = (0 >= cnt);
642
- out->w_start(out, &e);
643
- if (!e.closed) {
644
- const VALUE *np = RARRAY_PTR(obj);
645
- int i;
646
- int d2 = depth + 1;
647
-
648
- for (i = cnt; 0 < i; i--, np++) {
649
- dump_obj(0, *np, d2, out);
650
- }
651
- out->w_end(out, &e);
652
- }
653
- break;
634
+ if (0 != out->circ_cache && check_circular(out, obj, &e)) {
635
+ break;
636
+ }
637
+ cnt = (int)RARRAY_LEN(obj);
638
+ e.type = ArrayCode;
639
+ e.closed = (0 >= cnt);
640
+ out->w_start(out, &e);
641
+ if (!e.closed) {
642
+ const VALUE *np = RARRAY_PTR(obj);
643
+ int i;
644
+ int d2 = depth + 1;
645
+
646
+ for (i = cnt; 0 < i; i--, np++) {
647
+ dump_obj(0, *np, d2, out);
648
+ }
649
+ out->w_end(out, &e);
650
+ }
651
+ break;
654
652
  case T_HASH:
655
- if (0 != out->circ_cache && check_circular(out, obj, &e)) {
656
- break;
657
- }
658
- cnt = (int)RHASH_SIZE(obj);
659
- e.type = HashCode;
660
- e.closed = (0 >= cnt);
661
- out->w_start(out, &e);
662
- if (0 < cnt) {
663
- unsigned int od = out->depth;
664
-
665
- out->depth = depth + 1;
666
- rb_hash_foreach(obj, dump_hash, (VALUE)out);
667
- out->depth = od;
668
- out->w_end(out, &e);
669
- }
670
- break;
653
+ if (0 != out->circ_cache && check_circular(out, obj, &e)) {
654
+ break;
655
+ }
656
+ cnt = (int)RHASH_SIZE(obj);
657
+ e.type = HashCode;
658
+ e.closed = (0 >= cnt);
659
+ out->w_start(out, &e);
660
+ if (0 < cnt) {
661
+ unsigned int od = out->depth;
662
+
663
+ out->depth = depth + 1;
664
+ rb_hash_foreach(obj, dump_hash, (VALUE)out);
665
+ out->depth = od;
666
+ out->w_end(out, &e);
667
+ }
668
+ break;
671
669
  case T_TRUE:
672
- e.type = TrueClassCode;
673
- e.closed = 1;
674
- out->w_start(out, &e);
675
- break;
670
+ e.type = TrueClassCode;
671
+ e.closed = 1;
672
+ out->w_start(out, &e);
673
+ break;
676
674
  case T_FALSE:
677
- e.type = FalseClassCode;
678
- e.closed = 1;
679
- out->w_start(out, &e);
680
- break;
675
+ e.type = FalseClassCode;
676
+ e.closed = 1;
677
+ out->w_start(out, &e);
678
+ break;
681
679
  case T_FIXNUM:
682
- e.type = FixnumCode;
683
- out->w_start(out, &e);
684
- dump_num(out, obj);
685
- e.indent = -1;
686
- out->w_end(out, &e);
687
- break;
680
+ e.type = FixnumCode;
681
+ out->w_start(out, &e);
682
+ dump_num(out, obj);
683
+ e.indent = -1;
684
+ out->w_end(out, &e);
685
+ break;
688
686
  case T_FLOAT:
689
- e.type = FloatCode;
690
- cnt = snprintf(value_buf, sizeof(value_buf), "%0.16g", rb_num2dbl(obj));
691
- out->w_start(out, &e);
692
- dump_value(out, value_buf, cnt);
693
- e.indent = -1;
694
- out->w_end(out, &e);
695
- break;
696
- case T_STRING:
697
- {
698
- const char *str;
699
-
700
- if (0 != out->circ_cache && check_circular(out, obj, &e)) {
701
- break;
702
- }
703
- str = StringValuePtr(obj);
704
- cnt = (int)RSTRING_LEN(obj);
687
+ e.type = FloatCode;
688
+ cnt = snprintf(value_buf, sizeof(value_buf), "%0.16g", rb_num2dbl(obj));
689
+ out->w_start(out, &e);
690
+ dump_value(out, value_buf, cnt);
691
+ e.indent = -1;
692
+ out->w_end(out, &e);
693
+ break;
694
+ case T_STRING: {
695
+ const char *str;
696
+
697
+ if (0 != out->circ_cache && check_circular(out, obj, &e)) {
698
+ break;
699
+ }
700
+ str = StringValuePtr(obj);
701
+ cnt = (int)RSTRING_LEN(obj);
705
702
  #if USE_B64
706
- if (is_xml_friendly((uchar*)str, cnt)) {
707
- e.type = StringCode;
708
- out->w_start(out, &e);
709
- dump_str_value(out, str, cnt, '<');
710
- e.indent = -1;
711
- out->w_end(out, &e);
712
- } else {
713
- ulong size = b64_size(cnt);
714
- char *b64 = ALLOCA_N(char, size + 1);
715
-
716
- e.type = String64Code;
717
- to_base64((uchar*)str, cnt, b64);
718
- out->w_start(out, &e);
719
- dump_value(out, b64, size);
720
- e.indent = -1;
721
- out->w_end(out, &e);
722
- }
703
+ if (is_xml_friendly((uchar *)str, cnt)) {
704
+ e.type = StringCode;
705
+ out->w_start(out, &e);
706
+ dump_str_value(out, str, cnt, '<');
707
+ e.indent = -1;
708
+ out->w_end(out, &e);
709
+ } else {
710
+ ulong size = b64_size(cnt);
711
+ char *b64 = ALLOCA_N(char, size + 1);
712
+
713
+ e.type = String64Code;
714
+ to_base64((uchar *)str, cnt, b64);
715
+ out->w_start(out, &e);
716
+ dump_value(out, b64, size);
717
+ e.indent = -1;
718
+ out->w_end(out, &e);
719
+ }
723
720
  #else
724
- e.type = StringCode;
725
- out->w_start(out, &e);
726
- dump_str_value(out, str, cnt, xml_element_chars);
727
- e.indent = -1;
728
- out->w_end(out, &e);
721
+ e.type = StringCode;
722
+ out->w_start(out, &e);
723
+ dump_str_value(out, str, cnt, xml_element_chars);
724
+ e.indent = -1;
725
+ out->w_end(out, &e);
729
726
  #endif
730
- break;
727
+ break;
731
728
  }
732
- case T_SYMBOL:
733
- {
734
- const char *sym = rb_id2name(SYM2ID(obj));
729
+ case T_SYMBOL: {
730
+ const char *sym = rb_id2name(SYM2ID(obj));
735
731
 
736
- cnt = (int)strlen(sym);
732
+ cnt = (int)strlen(sym);
737
733
  #if USE_B64
738
- if (is_xml_friendly((uchar*)sym, cnt)) {
739
- e.type = SymbolCode;
740
- out->w_start(out, &e);
741
- dump_str_value(out, sym, cnt, '<');
742
- e.indent = -1;
743
- out->w_end(out, &e);
744
- } else {
745
- ulong size = b64_size(cnt);
746
- char *b64 = ALLOCA_N(char, size + 1);
747
-
748
- e.type = Symbol64Code;
749
- to_base64((uchar*)sym, cnt, b64);
750
- out->w_start(out, &e);
751
- dump_value(out, b64, size);
752
- e.indent = -1;
753
- out->w_end(out, &e);
754
- }
734
+ if (is_xml_friendly((uchar *)sym, cnt)) {
735
+ e.type = SymbolCode;
736
+ out->w_start(out, &e);
737
+ dump_str_value(out, sym, cnt, '<');
738
+ e.indent = -1;
739
+ out->w_end(out, &e);
740
+ } else {
741
+ ulong size = b64_size(cnt);
742
+ char *b64 = ALLOCA_N(char, size + 1);
743
+
744
+ e.type = Symbol64Code;
745
+ to_base64((uchar *)sym, cnt, b64);
746
+ out->w_start(out, &e);
747
+ dump_value(out, b64, size);
748
+ e.indent = -1;
749
+ out->w_end(out, &e);
750
+ }
755
751
  #else
756
- e.type = SymbolCode;
757
- out->w_start(out, &e);
758
- dump_str_value(out, sym, cnt, xml_element_chars);
759
- e.indent = -1;
760
- out->w_end(out, &e);
752
+ e.type = SymbolCode;
753
+ out->w_start(out, &e);
754
+ dump_str_value(out, sym, cnt, xml_element_chars);
755
+ e.indent = -1;
756
+ out->w_end(out, &e);
761
757
  #endif
762
- break;
763
- }
764
- case T_DATA:
765
- {
766
- VALUE clas;
767
-
768
- clas = rb_obj_class(obj);
769
- if (rb_cTime == clas) {
770
- e.type = TimeCode;
771
- out->w_start(out, &e);
772
- out->w_time(out, obj);
773
- e.indent = -1;
774
- out->w_end(out, &e);
775
- } else {
776
- const char *classname = rb_class2name(clas);
777
-
778
- if (0 == strcmp("Date", classname)) {
779
- e.type = DateCode;
780
- out->w_start(out, &e);
781
- dump_date(out, obj);
782
- e.indent = -1;
783
- out->w_end(out, &e);
784
- } else if (0 == strcmp("BigDecimal", classname)) {
785
- volatile VALUE rs = rb_funcall(obj, ox_to_s_id, 0);
786
-
787
- e.type = BigDecimalCode;
788
- out->w_start(out, &e);
789
- dump_value(out, StringValuePtr(rs), RSTRING_LEN(rs));
790
- e.indent = -1;
791
- out->w_end(out, &e);
792
- } else {
793
- if (StrictEffort == out->opts->effort) {
794
- rb_raise(rb_eNotImpError, "Failed to dump T_DATA %s\n", classname);
795
- } else {
796
- e.type = NilClassCode;
797
- e.closed = 1;
798
- out->w_start(out, &e);
799
- }
800
- }
801
- }
802
- break;
803
- }
804
- case T_STRUCT:
805
- {
758
+ break;
759
+ }
760
+ case T_DATA: {
761
+ VALUE clas;
762
+
763
+ clas = rb_obj_class(obj);
764
+ if (rb_cTime == clas) {
765
+ e.type = TimeCode;
766
+ out->w_start(out, &e);
767
+ out->w_time(out, obj);
768
+ e.indent = -1;
769
+ out->w_end(out, &e);
770
+ } else {
771
+ const char *classname = rb_class2name(clas);
772
+
773
+ if (0 == strcmp("Date", classname)) {
774
+ e.type = DateCode;
775
+ out->w_start(out, &e);
776
+ dump_date(out, obj);
777
+ e.indent = -1;
778
+ out->w_end(out, &e);
779
+ } else if (0 == strcmp("BigDecimal", classname)) {
780
+ volatile VALUE rs = rb_funcall(obj, ox_to_s_id, 0);
781
+
782
+ e.type = BigDecimalCode;
783
+ out->w_start(out, &e);
784
+ dump_value(out, StringValuePtr(rs), RSTRING_LEN(rs));
785
+ e.indent = -1;
786
+ out->w_end(out, &e);
787
+ } else {
788
+ if (StrictEffort == out->opts->effort) {
789
+ rb_raise(rb_eNotImpError, "Failed to dump T_DATA %s\n", classname);
790
+ } else {
791
+ e.type = NilClassCode;
792
+ e.closed = 1;
793
+ out->w_start(out, &e);
794
+ }
795
+ }
796
+ }
797
+ break;
798
+ }
799
+ case T_STRUCT: {
806
800
  #ifdef RSTRUCT_GET
807
- VALUE clas;
808
-
809
- if (0 != out->circ_cache && check_circular(out, obj, &e)) {
810
- break;
811
- }
812
- clas = rb_obj_class(obj);
813
- if (rb_cRange == clas) {
814
- VALUE beg = RSTRUCT_GET(obj, 0);
815
- VALUE end = RSTRUCT_GET(obj, 1);
816
- VALUE excl = RSTRUCT_GET(obj, 2);
817
- int d2 = depth + 1;
818
-
819
- e.type = RangeCode; e.clas.len = 5; e.clas.str = "Range";
820
- out->w_start(out, &e);
821
- dump_obj(ox_beg_id, beg, d2, out);
822
- dump_obj(ox_end_id, end, d2, out);
823
- dump_obj(ox_excl_id, excl, d2, out);
824
- out->w_end(out, &e);
825
- } else {
826
- char num_buf[16];
827
- int d2 = depth + 1;
801
+ VALUE clas;
802
+
803
+ if (0 != out->circ_cache && check_circular(out, obj, &e)) {
804
+ break;
805
+ }
806
+ clas = rb_obj_class(obj);
807
+ if (rb_cRange == clas) {
808
+ VALUE beg = RSTRUCT_GET(obj, 0);
809
+ VALUE end = RSTRUCT_GET(obj, 1);
810
+ VALUE excl = RSTRUCT_GET(obj, 2);
811
+ int d2 = depth + 1;
812
+
813
+ e.type = RangeCode;
814
+ e.clas.len = 5;
815
+ e.clas.str = "Range";
816
+ out->w_start(out, &e);
817
+ dump_obj(ox_beg_id, beg, d2, out);
818
+ dump_obj(ox_end_id, end, d2, out);
819
+ dump_obj(ox_excl_id, excl, d2, out);
820
+ out->w_end(out, &e);
821
+ } else {
822
+ char num_buf[16];
823
+ int d2 = depth + 1;
828
824
  #ifdef RUBY_INTEGER_UNIFICATION
829
- long i;
830
- long cnt = NUM2LONG(rb_struct_size(obj));
831
- #else // UNIFY_FIXNUM_AND_INTEGER
832
- int i;
833
- int cnt = (int)RSTRUCT_LEN(obj);
834
- #endif // UNIFY_FIXNUM_AND_INTEGER
835
- e.type = StructCode;
836
- e.clas.str = rb_class2name(clas);
837
- e.clas.len = strlen(e.clas.str);
838
- out->w_start(out, &e);
839
-
840
- for (i = 0; i < cnt; i++) {
841
- VALUE v = RSTRUCT_GET(obj, (int)(i));
842
- dump_obj(rb_intern(ulong2str(i, num_buf + sizeof(num_buf) - 1)), v, d2, out);
843
- }
844
- out->w_end(out, &e);
845
- }
825
+ long i;
826
+ long cnt = NUM2LONG(rb_struct_size(obj));
827
+ #else // UNIFY_FIXNUM_AND_INTEGER
828
+ int i;
829
+ int cnt = (int)RSTRUCT_LEN(obj);
830
+ #endif // UNIFY_FIXNUM_AND_INTEGER
831
+ e.type = StructCode;
832
+ e.clas.str = rb_class2name(clas);
833
+ e.clas.len = strlen(e.clas.str);
834
+ out->w_start(out, &e);
835
+
836
+ for (i = 0; i < cnt; i++) {
837
+ VALUE v = RSTRUCT_GET(obj, (int)(i));
838
+ dump_obj(rb_intern(ulong2str(i, num_buf + sizeof(num_buf) - 1)), v, d2, out);
839
+ }
840
+ out->w_end(out, &e);
841
+ }
846
842
  #else
847
- e.type = NilClassCode;
848
- e.closed = 1;
849
- out->w_start(out, &e);
843
+ e.type = NilClassCode;
844
+ e.closed = 1;
845
+ out->w_start(out, &e);
850
846
  #endif
851
- break;
852
- }
853
- case T_OBJECT:
854
- {
855
- VALUE clas;
856
-
857
- if (0 != out->circ_cache && check_circular(out, obj, &e)) {
858
- break;
859
- }
860
- clas = rb_obj_class(obj);
861
- e.clas.str = rb_class2name(clas);
862
- e.clas.len = strlen(e.clas.str);
863
- if (ox_document_clas == clas) {
864
- e.type = RawCode;
865
- out->w_start(out, &e);
866
- dump_gen_doc(obj, depth + 1, out);
867
- out->w_end(out, &e);
868
- } else if (ox_element_clas == clas) {
869
- e.type = RawCode;
870
- out->w_start(out, &e);
871
- dump_gen_element(obj, depth + 1, out);
872
- out->w_end(out, &e);
873
- } else { /* Object */
847
+ break;
848
+ }
849
+ case T_OBJECT: {
850
+ VALUE clas;
851
+
852
+ if (0 != out->circ_cache && check_circular(out, obj, &e)) {
853
+ break;
854
+ }
855
+ clas = rb_obj_class(obj);
856
+ e.clas.str = rb_class2name(clas);
857
+ e.clas.len = strlen(e.clas.str);
858
+ if (ox_document_clas == clas) {
859
+ e.type = RawCode;
860
+ out->w_start(out, &e);
861
+ dump_gen_doc(obj, depth + 1, out);
862
+ out->w_end(out, &e);
863
+ } else if (ox_element_clas == clas) {
864
+ e.type = RawCode;
865
+ out->w_start(out, &e);
866
+ dump_gen_element(obj, depth + 1, out);
867
+ out->w_end(out, &e);
868
+ } else { /* Object */
874
869
  #if HAVE_RB_IVAR_FOREACH
875
- e.type = (Qtrue == rb_obj_is_kind_of(obj, rb_eException)) ? ExceptionCode : ObjectCode;
876
- cnt = (int)rb_ivar_count(obj);
877
- e.closed = (0 >= cnt);
878
- out->w_start(out, &e);
879
- if (0 < cnt) {
880
- unsigned int od = out->depth;
881
-
882
- out->depth = depth + 1;
883
- rb_ivar_foreach(obj, dump_var, (VALUE)out);
884
- out->depth = od;
885
- out->w_end(out, &e);
886
- }
870
+ e.type = (Qtrue == rb_obj_is_kind_of(obj, rb_eException)) ? ExceptionCode : ObjectCode;
871
+ cnt = (int)rb_ivar_count(obj);
872
+ e.closed = (0 >= cnt);
873
+ out->w_start(out, &e);
874
+ if (0 < cnt) {
875
+ unsigned int od = out->depth;
876
+
877
+ out->depth = depth + 1;
878
+ rb_ivar_foreach(obj, dump_var, (VALUE)out);
879
+ out->depth = od;
880
+ out->w_end(out, &e);
881
+ }
887
882
  #else
888
- volatile VALUE vars = rb_obj_instance_variables(obj);
889
- //volatile VALUE vars = rb_funcall2(obj, rb_intern("instance_variables"), 0, 0);
890
-
891
- e.type = (Qtrue == rb_obj_is_kind_of(obj, rb_eException)) ? ExceptionCode : ObjectCode;
892
- cnt = (int)RARRAY_LEN(vars);
893
- e.closed = (0 >= cnt);
894
- out->w_start(out, &e);
895
- if (0 < cnt) {
896
- const VALUE *np = RARRAY_PTR(vars);
897
- ID vid;
898
- unsigned int od = out->depth;
899
- int i;
900
-
901
- out->depth = depth + 1;
902
- for (i = cnt; 0 < i; i--, np++) {
903
- vid = rb_to_id(*np);
904
- dump_var(vid, rb_ivar_get(obj, vid), out);
905
- }
906
- out->depth = od;
907
- out->w_end(out, &e);
908
- }
883
+ volatile VALUE vars = rb_obj_instance_variables(obj);
884
+ // volatile VALUE vars = rb_funcall2(obj, rb_intern("instance_variables"), 0, 0);
885
+
886
+ e.type = (Qtrue == rb_obj_is_kind_of(obj, rb_eException)) ? ExceptionCode : ObjectCode;
887
+ cnt = (int)RARRAY_LEN(vars);
888
+ e.closed = (0 >= cnt);
889
+ out->w_start(out, &e);
890
+ if (0 < cnt) {
891
+ const VALUE *np = RARRAY_PTR(vars);
892
+ ID vid;
893
+ unsigned int od = out->depth;
894
+ int i;
895
+
896
+ out->depth = depth + 1;
897
+ for (i = cnt; 0 < i; i--, np++) {
898
+ vid = rb_to_id(*np);
899
+ dump_var(vid, rb_ivar_get(obj, vid), out);
900
+ }
901
+ out->depth = od;
902
+ out->w_end(out, &e);
903
+ }
909
904
  #endif
910
- }
911
- break;
905
+ }
906
+ break;
912
907
  }
913
- case T_REGEXP:
914
- {
915
- volatile VALUE rs = rb_funcall2(obj, ox_inspect_id, 0, 0);
916
- const char *s = StringValuePtr(rs);
908
+ case T_REGEXP: {
909
+ volatile VALUE rs = rb_funcall2(obj, ox_inspect_id, 0, 0);
910
+ const char *s = StringValuePtr(rs);
917
911
 
918
- cnt = (int)RSTRING_LEN(rs);
919
- e.type = RegexpCode;
920
- out->w_start(out, &e);
912
+ cnt = (int)RSTRING_LEN(rs);
913
+ e.type = RegexpCode;
914
+ out->w_start(out, &e);
921
915
  #if USE_B64
922
- if (is_xml_friendly((uchar*)s, cnt)) {
923
- /*dump_value(out, "/", 1); */
924
- dump_str_value(out, s, cnt, '<');
925
- } else {
926
- ulong size = b64_size(cnt);
927
- char *b64 = ALLOCA_N(char, size + 1);
928
-
929
- to_base64((uchar*)s, cnt, b64);
930
- dump_value(out, b64, size);
931
- }
916
+ if (is_xml_friendly((uchar *)s, cnt)) {
917
+ /*dump_value(out, "/", 1); */
918
+ dump_str_value(out, s, cnt, '<');
919
+ } else {
920
+ ulong size = b64_size(cnt);
921
+ char *b64 = ALLOCA_N(char, size + 1);
922
+
923
+ to_base64((uchar *)s, cnt, b64);
924
+ dump_value(out, b64, size);
925
+ }
932
926
  #else
933
- dump_str_value(out, s, cnt, xml_element_chars);
927
+ dump_str_value(out, s, cnt, xml_element_chars);
934
928
  #endif
935
- e.indent = -1;
936
- out->w_end(out, &e);
937
- break;
929
+ e.indent = -1;
930
+ out->w_end(out, &e);
931
+ break;
938
932
  }
939
- case T_BIGNUM:
940
- {
941
- volatile VALUE rs = rb_big2str(obj, 10);
933
+ case T_BIGNUM: {
934
+ volatile VALUE rs = rb_big2str(obj, 10);
942
935
 
943
- e.type = BignumCode;
944
- out->w_start(out, &e);
945
- dump_value(out, StringValuePtr(rs), RSTRING_LEN(rs));
946
- e.indent = -1;
947
- out->w_end(out, &e);
948
- break;
936
+ e.type = BignumCode;
937
+ out->w_start(out, &e);
938
+ dump_value(out, StringValuePtr(rs), RSTRING_LEN(rs));
939
+ e.indent = -1;
940
+ out->w_end(out, &e);
941
+ break;
949
942
  }
950
943
  #ifdef T_COMPLEX
951
- case T_COMPLEX:
952
- e.type = ComplexCode;
953
- out->w_start(out, &e);
944
+ case T_COMPLEX: e.type = ComplexCode; out->w_start(out, &e);
954
945
  #ifdef RCOMPLEX
955
- dump_obj(0, RCOMPLEX(obj)->real, depth + 1, out);
956
- dump_obj(0, RCOMPLEX(obj)->imag, depth + 1, out);
946
+ dump_obj(0, RCOMPLEX(obj)->real, depth + 1, out);
947
+ dump_obj(0, RCOMPLEX(obj)->imag, depth + 1, out);
957
948
  #else
958
- dump_obj(0, rb_funcall2(obj, rb_intern("real"), 0, 0), depth + 1, out);
959
- dump_obj(0, rb_funcall2(obj, rb_intern("imag"), 0, 0), depth + 1, out);
949
+ dump_obj(0, rb_funcall2(obj, rb_intern("real"), 0, 0), depth + 1, out);
950
+ dump_obj(0, rb_funcall2(obj, rb_intern("imag"), 0, 0), depth + 1, out);
960
951
  #endif
961
- out->w_end(out, &e);
962
- break;
952
+ out->w_end(out, &e);
953
+ break;
963
954
  #endif
964
955
  #ifdef T_RATIONAL
965
- case T_RATIONAL:
966
- e.type = RationalCode;
967
- out->w_start(out, &e);
956
+ case T_RATIONAL: e.type = RationalCode; out->w_start(out, &e);
968
957
  #ifdef RRATIONAL
969
- dump_obj(0, RRATIONAL(obj)->num, depth + 1, out);
970
- dump_obj(0, RRATIONAL(obj)->den, depth + 1, out);
958
+ dump_obj(0, RRATIONAL(obj)->num, depth + 1, out);
959
+ dump_obj(0, RRATIONAL(obj)->den, depth + 1, out);
971
960
  #else
972
- dump_obj(0, rb_funcall2(obj, rb_intern("numerator"), 0, 0), depth + 1, out);
973
- dump_obj(0, rb_funcall2(obj, rb_intern("denominator"), 0, 0), depth + 1, out);
961
+ dump_obj(0, rb_funcall2(obj, rb_intern("numerator"), 0, 0), depth + 1, out);
962
+ dump_obj(0, rb_funcall2(obj, rb_intern("denominator"), 0, 0), depth + 1, out);
974
963
  #endif
975
- out->w_end(out, &e);
976
- break;
964
+ out->w_end(out, &e);
965
+ break;
977
966
  #endif
978
- case T_CLASS:
979
- {
980
- e.type = ClassCode;
981
- e.clas.str = rb_class2name(obj);
982
- e.clas.len = strlen(e.clas.str);
983
- e.closed = 1;
984
- out->w_start(out, &e);
985
- break;
967
+ case T_CLASS: {
968
+ e.type = ClassCode;
969
+ e.clas.str = rb_class2name(obj);
970
+ e.clas.len = strlen(e.clas.str);
971
+ e.closed = 1;
972
+ out->w_start(out, &e);
973
+ break;
986
974
  }
987
975
  default:
988
- if (StrictEffort == out->opts->effort) {
989
- rb_raise(rb_eNotImpError, "Failed to dump %s Object (%02x)\n",
990
- rb_obj_classname(obj), rb_type(obj));
991
- } else {
992
- e.type = NilClassCode;
993
- e.closed = 1;
994
- out->w_start(out, &e);
995
- }
996
- break;
976
+ if (StrictEffort == out->opts->effort) {
977
+ rb_raise(rb_eNotImpError, "Failed to dump %s Object (%02x)\n", rb_obj_classname(obj), rb_type(obj));
978
+ } else {
979
+ e.type = NilClassCode;
980
+ e.closed = 1;
981
+ out->w_start(out, &e);
982
+ }
983
+ break;
997
984
  }
998
985
  out->obj = prev_obj;
999
986
  }
1000
987
 
1001
- static int
1002
- dump_var(ID key, VALUE value, VALUE ov) {
1003
- Out out = (Out)ov;
988
+ static int dump_var(ID key, VALUE value, VALUE ov) {
989
+ Out out = (Out)ov;
1004
990
 
1005
991
  if (T_DATA == rb_type(value) && key == ox_mesg_id) {
1006
- /* There is a secret recipe that keeps Exception mesg attributes as a
1007
- * T_DATA until it is needed. The safe way around this hack is to call
1008
- * the message() method and use the returned string as the
1009
- * message. Not pretty but it solves the most common use of this
1010
- * hack. If there are others they will have to be handled one at a
1011
- * time.
1012
- */
1013
- value = rb_funcall(out->obj, ox_message_id, 0);
992
+ /* There is a secret recipe that keeps Exception mesg attributes as a
993
+ * T_DATA until it is needed. The safe way around this hack is to call
994
+ * the message() method and use the returned string as the
995
+ * message. Not pretty but it solves the most common use of this
996
+ * hack. If there are others they will have to be handled one at a
997
+ * time.
998
+ */
999
+ value = rb_funcall(out->obj, ox_message_id, 0);
1014
1000
  }
1015
1001
  dump_obj(key, value, out->depth, out);
1016
1002
 
1017
1003
  return ST_CONTINUE;
1018
1004
  }
1019
1005
 
1020
- static int
1021
- dump_hash(VALUE key, VALUE value, VALUE ov) {
1022
- Out out = (Out)ov;
1006
+ static int dump_hash(VALUE key, VALUE value, VALUE ov) {
1007
+ Out out = (Out)ov;
1023
1008
 
1024
1009
  dump_obj(0, key, out->depth, out);
1025
1010
  dump_obj(0, value, out->depth, out);
@@ -1027,199 +1012,190 @@ dump_hash(VALUE key, VALUE value, VALUE ov) {
1027
1012
  return ST_CONTINUE;
1028
1013
  }
1029
1014
 
1030
- static void
1031
- dump_gen_doc(VALUE obj, int depth, Out out) {
1032
- volatile VALUE attrs = rb_attr_get(obj, ox_attributes_id);
1033
- volatile VALUE nodes = rb_attr_get(obj, ox_nodes_id);
1015
+ static void dump_gen_doc(VALUE obj, int depth, Out out) {
1016
+ volatile VALUE attrs = rb_attr_get(obj, ox_attributes_id);
1017
+ volatile VALUE nodes = rb_attr_get(obj, ox_nodes_id);
1034
1018
 
1035
1019
  if ('\0' == *out->opts->encoding && Qnil != attrs) {
1036
- volatile VALUE renc = rb_hash_lookup(attrs, ox_encoding_sym);
1020
+ volatile VALUE renc = rb_hash_lookup(attrs, ox_encoding_sym);
1037
1021
 
1038
- if (Qnil != renc) {
1039
- const char *enc = StringValuePtr(renc);
1022
+ if (Qnil != renc) {
1023
+ const char *enc = StringValuePtr(renc);
1040
1024
 
1041
- strncpy(out->opts->encoding, enc, sizeof(out->opts->encoding) - 1);
1042
- }
1025
+ strncpy(out->opts->encoding, enc, sizeof(out->opts->encoding) - 1);
1026
+ }
1043
1027
  }
1044
1028
  if (Yes == out->opts->with_xml) {
1045
- if (0 < out->opts->margin_len) {
1046
- dump_value(out, out->opts->margin, out->opts->margin_len);
1047
- }
1048
- dump_value(out, "<?xml", 5);
1049
- if (Qnil != attrs) {
1050
- rb_hash_foreach(attrs, dump_gen_attr, (VALUE)out);
1051
- }
1052
- dump_value(out, "?>", 2);
1029
+ if (0 < out->opts->margin_len) {
1030
+ dump_value(out, out->opts->margin, out->opts->margin_len);
1031
+ }
1032
+ dump_value(out, "<?xml", 5);
1033
+ if (Qnil != attrs) {
1034
+ rb_hash_foreach(attrs, dump_gen_attr, (VALUE)out);
1035
+ }
1036
+ dump_value(out, "?>", 2);
1053
1037
  }
1054
1038
  if (Yes == out->opts->with_instruct) {
1055
- if (out->buf < out->cur) {
1056
- dump_value(out, "\n", 1);
1057
- }
1058
- if (0 < out->opts->margin_len) {
1059
- dump_value(out, out->opts->margin, out->opts->margin_len);
1060
- }
1061
- dump_value(out, "<?ox version=\"1.0\" mode=\"generic\"?>", 35);
1039
+ if (out->buf < out->cur) {
1040
+ dump_value(out, "\n", 1);
1041
+ }
1042
+ if (0 < out->opts->margin_len) {
1043
+ dump_value(out, out->opts->margin, out->opts->margin_len);
1044
+ }
1045
+ dump_value(out, "<?ox version=\"1.0\" mode=\"generic\"?>", 35);
1062
1046
  }
1063
1047
  if (Qnil != nodes) {
1064
- dump_gen_nodes(nodes, depth, out);
1048
+ dump_gen_nodes(nodes, depth, out);
1065
1049
  }
1066
1050
  }
1067
1051
 
1068
- static void
1069
- dump_gen_element(VALUE obj, int depth, Out out) {
1070
- volatile VALUE rname = rb_attr_get(obj, ox_at_value_id);
1071
- volatile VALUE attrs = rb_attr_get(obj, ox_attributes_id);
1072
- volatile VALUE nodes = rb_attr_get(obj, ox_nodes_id);
1073
- const char *name = StringValuePtr(rname);
1074
- long nlen = RSTRING_LEN(rname);
1075
- size_t size;
1076
- int indent;
1052
+ static void dump_gen_element(VALUE obj, int depth, Out out) {
1053
+ volatile VALUE rname = rb_attr_get(obj, ox_at_value_id);
1054
+ volatile VALUE attrs = rb_attr_get(obj, ox_attributes_id);
1055
+ volatile VALUE nodes = rb_attr_get(obj, ox_nodes_id);
1056
+ const char *name = StringValuePtr(rname);
1057
+ long nlen = RSTRING_LEN(rname);
1058
+ size_t size;
1059
+ int indent;
1077
1060
 
1078
1061
  if (0 > out->indent) {
1079
- indent = -1;
1062
+ indent = -1;
1080
1063
  } else if (0 == out->indent) {
1081
- indent = 0;
1064
+ indent = 0;
1082
1065
  } else {
1083
- indent = depth * out->indent;
1066
+ indent = depth * out->indent;
1084
1067
  }
1085
1068
  size = indent + 4 + nlen + out->opts->margin_len;
1086
1069
  if (out->end - out->cur <= (long)size) {
1087
- grow(out, size);
1070
+ grow(out, size);
1088
1071
  }
1089
1072
  if (0 == depth && 0 < out->opts->margin_len && 0 < out->indent) {
1090
- memcpy(out->cur, out->opts->margin, out->opts->margin_len);
1091
- out->cur += out->opts->margin_len;
1073
+ memcpy(out->cur, out->opts->margin, out->opts->margin_len);
1074
+ out->cur += out->opts->margin_len;
1092
1075
  }
1093
1076
  fill_indent(out, indent);
1094
1077
  *out->cur++ = '<';
1095
1078
  fill_value(out, name, nlen);
1096
1079
  if (Qnil != attrs) {
1097
- rb_hash_foreach(attrs, dump_gen_attr, (VALUE)out);
1080
+ rb_hash_foreach(attrs, dump_gen_attr, (VALUE)out);
1098
1081
  }
1099
1082
  if (Qnil != nodes && 0 < RARRAY_LEN(nodes)) {
1100
- int do_indent;
1101
-
1102
- *out->cur++ = '>';
1103
- do_indent = dump_gen_nodes(nodes, depth, out);
1104
- if (out->end - out->cur <= (long)size) {
1105
- grow(out, size);
1106
- }
1107
- if (do_indent) {
1108
- fill_indent(out, indent);
1109
- }
1110
- *out->cur++ = '<';
1111
- *out->cur++ = '/';
1112
- fill_value(out, name, nlen);
1083
+ int do_indent;
1084
+
1085
+ *out->cur++ = '>';
1086
+ do_indent = dump_gen_nodes(nodes, depth, out);
1087
+ if (out->end - out->cur <= (long)size) {
1088
+ grow(out, size);
1089
+ }
1090
+ if (do_indent) {
1091
+ fill_indent(out, indent);
1092
+ }
1093
+ *out->cur++ = '<';
1094
+ *out->cur++ = '/';
1095
+ fill_value(out, name, nlen);
1113
1096
  } else if (out->opts->no_empty) {
1114
- *out->cur++ = '>';
1115
- *out->cur++ = '<';
1116
- *out->cur++ = '/';
1117
- fill_value(out, name, nlen);
1097
+ *out->cur++ = '>';
1098
+ *out->cur++ = '<';
1099
+ *out->cur++ = '/';
1100
+ fill_value(out, name, nlen);
1118
1101
  } else {
1119
- *out->cur++ = '/';
1102
+ *out->cur++ = '/';
1120
1103
  }
1121
1104
  *out->cur++ = '>';
1122
- *out->cur = '\0';
1105
+ *out->cur = '\0';
1123
1106
  }
1124
1107
 
1125
- static void
1126
- dump_gen_instruct(VALUE obj, int depth, Out out) {
1127
- volatile VALUE rname = rb_attr_get(obj, ox_at_value_id);
1128
- volatile VALUE attrs = rb_attr_get(obj, ox_attributes_id);
1129
- volatile VALUE rcontent = rb_attr_get(obj, ox_at_content_id);
1130
- const char *name = StringValuePtr(rname);
1131
- const char *content = 0;
1132
- long nlen = RSTRING_LEN(rname);
1133
- long clen = 0;
1134
- size_t size;
1108
+ static void dump_gen_instruct(VALUE obj, int depth, Out out) {
1109
+ volatile VALUE rname = rb_attr_get(obj, ox_at_value_id);
1110
+ volatile VALUE attrs = rb_attr_get(obj, ox_attributes_id);
1111
+ volatile VALUE rcontent = rb_attr_get(obj, ox_at_content_id);
1112
+ const char *name = StringValuePtr(rname);
1113
+ const char *content = 0;
1114
+ long nlen = RSTRING_LEN(rname);
1115
+ long clen = 0;
1116
+ size_t size;
1135
1117
 
1136
1118
  if (T_STRING == rb_type(rcontent)) {
1137
- content = StringValuePtr(rcontent);
1138
- clen = RSTRING_LEN(rcontent);
1139
- size = 4 + nlen + clen;
1119
+ content = StringValuePtr(rcontent);
1120
+ clen = RSTRING_LEN(rcontent);
1121
+ size = 4 + nlen + clen;
1140
1122
  } else {
1141
- size = 4 + nlen;
1123
+ size = 4 + nlen;
1142
1124
  }
1143
1125
  if (out->end - out->cur <= (long)size) {
1144
- grow(out, size);
1126
+ grow(out, size);
1145
1127
  }
1146
1128
  *out->cur++ = '<';
1147
1129
  *out->cur++ = '?';
1148
1130
  fill_value(out, name, nlen);
1149
1131
  if (0 != content) {
1150
- fill_value(out, content, clen);
1132
+ fill_value(out, content, clen);
1151
1133
  } else if (Qnil != attrs) {
1152
- rb_hash_foreach(attrs, dump_gen_attr, (VALUE)out);
1134
+ rb_hash_foreach(attrs, dump_gen_attr, (VALUE)out);
1153
1135
  }
1154
1136
  *out->cur++ = '?';
1155
1137
  *out->cur++ = '>';
1156
- *out->cur = '\0';
1138
+ *out->cur = '\0';
1157
1139
  }
1158
1140
 
1159
- static int
1160
- dump_gen_nodes(VALUE obj, int depth, Out out) {
1161
- long cnt = RARRAY_LEN(obj);
1162
- int indent_needed = 1;
1141
+ static int dump_gen_nodes(VALUE obj, int depth, Out out) {
1142
+ long cnt = RARRAY_LEN(obj);
1143
+ int indent_needed = 1;
1163
1144
 
1164
1145
  if (0 < cnt) {
1165
- const VALUE *np = RARRAY_PTR(obj);
1166
- VALUE clas;
1167
- int d2 = depth + 1;
1168
-
1169
- if (MAX_DEPTH < depth) {
1170
- rb_raise(rb_eSysStackError, "maximum depth exceeded");
1171
- }
1172
- for (; 0 < cnt; cnt--, np++) {
1173
- clas = rb_obj_class(*np);
1174
- if (ox_element_clas == clas) {
1175
- dump_gen_element(*np, d2, out);
1176
- } else if (ox_instruct_clas == clas) {
1177
- dump_gen_instruct(*np, d2, out);
1178
- indent_needed = (1 == cnt) ? 0 : 1;
1179
- } else if (rb_cString == clas) {
1180
- dump_str_value(out, StringValuePtr(*(VALUE*)np), RSTRING_LEN(*np), xml_element_chars);
1181
- indent_needed = (1 == cnt) ? 0 : 1;
1182
- } else if (ox_comment_clas == clas) {
1183
- dump_gen_val_node(*np, d2, "<!-- ", 5, " -->", 4, out);
1184
- } else if (ox_raw_clas == clas) {
1185
- dump_gen_val_node(*np, d2, "", 0, "", 0, out);
1186
- } else if (ox_cdata_clas == clas) {
1187
- dump_gen_val_node(*np, d2, "<![CDATA[", 9, "]]>", 3, out);
1188
- } else if (ox_doctype_clas == clas) {
1189
- dump_gen_val_node(*np, d2, "<!DOCTYPE ", 10, ">", 1, out);
1190
- } else {
1191
- rb_raise(rb_eTypeError, "Unexpected class, %s, while dumping generic XML\n", rb_class2name(clas));
1192
- }
1193
- }
1146
+ const VALUE *np = RARRAY_PTR(obj);
1147
+ VALUE clas;
1148
+ int d2 = depth + 1;
1149
+
1150
+ if (MAX_DEPTH < depth) {
1151
+ rb_raise(rb_eSysStackError, "maximum depth exceeded");
1152
+ }
1153
+ for (; 0 < cnt; cnt--, np++) {
1154
+ clas = rb_obj_class(*np);
1155
+ if (ox_element_clas == clas) {
1156
+ dump_gen_element(*np, d2, out);
1157
+ } else if (ox_instruct_clas == clas) {
1158
+ dump_gen_instruct(*np, d2, out);
1159
+ indent_needed = (1 == cnt) ? 0 : 1;
1160
+ } else if (rb_cString == clas) {
1161
+ dump_str_value(out, StringValuePtr(*(VALUE *)np), RSTRING_LEN(*np), xml_element_chars);
1162
+ indent_needed = (1 == cnt) ? 0 : 1;
1163
+ } else if (ox_comment_clas == clas) {
1164
+ dump_gen_val_node(*np, d2, "<!-- ", 5, " -->", 4, out);
1165
+ } else if (ox_raw_clas == clas) {
1166
+ dump_gen_val_node(*np, d2, "", 0, "", 0, out);
1167
+ } else if (ox_cdata_clas == clas) {
1168
+ dump_gen_val_node(*np, d2, "<![CDATA[", 9, "]]>", 3, out);
1169
+ } else if (ox_doctype_clas == clas) {
1170
+ dump_gen_val_node(*np, d2, "<!DOCTYPE ", 10, ">", 1, out);
1171
+ } else {
1172
+ rb_raise(rb_eTypeError, "Unexpected class, %s, while dumping generic XML\n", rb_class2name(clas));
1173
+ }
1174
+ }
1194
1175
  }
1195
1176
  return indent_needed;
1196
1177
  }
1197
1178
 
1198
- static int
1199
- dump_gen_attr(VALUE key, VALUE value, VALUE ov) {
1200
- Out out = (Out)ov;
1179
+ static int dump_gen_attr(VALUE key, VALUE value, VALUE ov) {
1180
+ Out out = (Out)ov;
1201
1181
 
1202
- const char *ks;
1203
- size_t klen;
1204
- size_t size;
1182
+ const char *ks;
1183
+ size_t klen;
1184
+ size_t size;
1205
1185
 
1206
1186
  switch (rb_type(key)) {
1207
- case T_SYMBOL:
1208
- ks = rb_id2name(SYM2ID(key));
1209
- break;
1210
- case T_STRING:
1211
- ks = StringValuePtr(key);
1212
- break;
1187
+ case T_SYMBOL: ks = rb_id2name(SYM2ID(key)); break;
1188
+ case T_STRING: ks = StringValuePtr(key); break;
1213
1189
  default:
1214
- key = rb_String(key);
1215
- ks = StringValuePtr(key);
1216
- break;
1190
+ key = rb_String(key);
1191
+ ks = StringValuePtr(key);
1192
+ break;
1217
1193
  }
1218
- klen = strlen(ks);
1194
+ klen = strlen(ks);
1219
1195
  value = rb_String(value);
1220
- size = 4 + klen + RSTRING_LEN(value);
1196
+ size = 4 + klen + RSTRING_LEN(value);
1221
1197
  if (out->end - out->cur <= (long)size) {
1222
- grow(out, size);
1198
+ grow(out, size);
1223
1199
  }
1224
1200
  *out->cur++ = ' ';
1225
1201
  fill_value(out, ks, klen);
@@ -1232,30 +1208,28 @@ dump_gen_attr(VALUE key, VALUE value, VALUE ov) {
1232
1208
  }
1233
1209
 
1234
1210
  static void
1235
- dump_gen_val_node(VALUE obj, int depth,
1236
- const char *pre, size_t plen,
1237
- const char *suf, size_t slen, Out out) {
1238
- volatile VALUE v = rb_attr_get(obj, ox_at_value_id);
1239
- const char *val;
1240
- size_t vlen;
1241
- size_t size;
1242
- int indent;
1211
+ dump_gen_val_node(VALUE obj, int depth, const char *pre, size_t plen, const char *suf, size_t slen, Out out) {
1212
+ volatile VALUE v = rb_attr_get(obj, ox_at_value_id);
1213
+ const char *val;
1214
+ size_t vlen;
1215
+ size_t size;
1216
+ int indent;
1243
1217
 
1244
1218
  if (T_STRING != rb_type(v)) {
1245
- return;
1219
+ return;
1246
1220
  }
1247
- val = StringValuePtr(v);
1221
+ val = StringValuePtr(v);
1248
1222
  vlen = RSTRING_LEN(v);
1249
1223
  if (0 > out->indent) {
1250
- indent = -1;
1224
+ indent = -1;
1251
1225
  } else if (0 == out->indent) {
1252
- indent = 0;
1226
+ indent = 0;
1253
1227
  } else {
1254
- indent = depth * out->indent;
1228
+ indent = depth * out->indent;
1255
1229
  }
1256
1230
  size = indent + plen + slen + vlen + out->opts->margin_len;
1257
1231
  if (out->end - out->cur <= (long)size) {
1258
- grow(out, size);
1232
+ grow(out, size);
1259
1233
  }
1260
1234
  fill_indent(out, indent);
1261
1235
  fill_value(out, pre, plen);
@@ -1264,63 +1238,60 @@ dump_gen_val_node(VALUE obj, int depth,
1264
1238
  *out->cur = '\0';
1265
1239
  }
1266
1240
 
1267
- static void
1268
- dump_obj_to_xml(VALUE obj, Options copts, Out out) {
1269
- VALUE clas = rb_obj_class(obj);
1241
+ static void dump_obj_to_xml(VALUE obj, Options copts, Out out) {
1242
+ VALUE clas = rb_obj_class(obj);
1270
1243
 
1271
- out->w_time = (Yes == copts->xsd_date) ? dump_time_xsd : dump_time_thin;
1272
- out->buf = ALLOC_N(char, 65336);
1273
- out->end = out->buf + 65325; /* 10 less than end plus extra for possible errors */
1274
- out->cur = out->buf;
1244
+ out->w_time = (Yes == copts->xsd_date) ? dump_time_xsd : dump_time_thin;
1245
+ out->buf = ALLOC_N(char, 65336);
1246
+ out->end = out->buf + 65325; /* 10 less than end plus extra for possible errors */
1247
+ out->cur = out->buf;
1275
1248
  out->circ_cache = 0;
1276
- out->circ_cnt = 0;
1277
- out->opts = copts;
1278
- out->obj = obj;
1279
- *out->cur = '\0';
1249
+ out->circ_cnt = 0;
1250
+ out->opts = copts;
1251
+ out->obj = obj;
1252
+ *out->cur = '\0';
1280
1253
  if (Yes == copts->circular) {
1281
- ox_cache8_new(&out->circ_cache);
1254
+ ox_cache8_new(&out->circ_cache);
1282
1255
  }
1283
1256
  out->indent = copts->indent;
1284
1257
 
1285
1258
  if (ox_document_clas == clas) {
1286
- dump_gen_doc(obj, -1, out);
1259
+ dump_gen_doc(obj, -1, out);
1287
1260
  } else if (ox_element_clas == clas) {
1288
- dump_gen_element(obj, 0, out);
1261
+ dump_gen_element(obj, 0, out);
1289
1262
  } else {
1290
- out->w_start = dump_start;
1291
- out->w_end = dump_end;
1292
- dump_first_obj(obj, out);
1263
+ out->w_start = dump_start;
1264
+ out->w_end = dump_end;
1265
+ dump_first_obj(obj, out);
1293
1266
  }
1294
1267
  if (0 <= out->indent) {
1295
- dump_value(out, "\n", 1);
1268
+ dump_value(out, "\n", 1);
1296
1269
  }
1297
1270
  if (Yes == copts->circular) {
1298
- ox_cache8_delete(out->circ_cache);
1271
+ ox_cache8_delete(out->circ_cache);
1299
1272
  }
1300
1273
  }
1301
1274
 
1302
- char*
1303
- ox_write_obj_to_str(VALUE obj, Options copts) {
1275
+ char *ox_write_obj_to_str(VALUE obj, Options copts) {
1304
1276
  struct _out out;
1305
1277
 
1306
1278
  dump_obj_to_xml(obj, copts, &out);
1307
1279
  return out.buf;
1308
1280
  }
1309
1281
 
1310
- void
1311
- ox_write_obj_to_file(VALUE obj, const char *path, Options copts) {
1282
+ void ox_write_obj_to_file(VALUE obj, const char *path, Options copts) {
1312
1283
  struct _out out;
1313
- size_t size;
1314
- FILE *f;
1284
+ size_t size;
1285
+ FILE *f;
1315
1286
 
1316
1287
  dump_obj_to_xml(obj, copts, &out);
1317
1288
  size = out.cur - out.buf;
1318
1289
  if (0 == (f = fopen(path, "w"))) {
1319
- rb_raise(rb_eIOError, "%s\n", strerror(errno));
1290
+ rb_raise(rb_eIOError, "%s\n", strerror(errno));
1320
1291
  }
1321
1292
  if (size != fwrite(out.buf, 1, size, f)) {
1322
- int err = ferror(f);
1323
- rb_raise(rb_eIOError, "Write failed. [%d:%s]\n", err, strerror(err));
1293
+ int err = ferror(f);
1294
+ rb_raise(rb_eIOError, "Write failed. [%d:%s]\n", err, strerror(err));
1324
1295
  }
1325
1296
  xfree(out.buf);
1326
1297
  fclose(f);