source: EcnlProtoTool/trunk/mruby-1.3.0/src/dump.c@ 331

Last change on this file since 331 was 331, checked in by coas-nagasima, 6 years ago

prototoolに関連するプロジェクトをnewlibからmuslを使うよう変更・更新
ntshellをnewlibの下位の実装から、muslのsyscallの実装に変更・更新
以下のOSSをアップデート
・mruby-1.3.0
・musl-1.1.18
・onigmo-6.1.3
・tcc-0.9.27
以下のOSSを追加
・openssl-1.1.0e
・curl-7.57.0
・zlib-1.2.11
以下のmrbgemsを追加
・iij/mruby-digest
・iij/mruby-env
・iij/mruby-errno
・iij/mruby-iijson
・iij/mruby-ipaddr
・iij/mruby-mock
・iij/mruby-require
・iij/mruby-tls-openssl

  • Property svn:eol-style set to native
  • Property svn:mime-type set to text/x-csrc;charset=UTF-8
File size: 27.7 KB
Line 
1/*
2** dump.c - mruby binary dumper (mrbc binary format)
3**
4** See Copyright Notice in mruby.h
5*/
6
7#include <string.h>
8#include <limits.h>
9#include <mruby/dump.h>
10#include <mruby/string.h>
11#include <mruby/irep.h>
12#include <mruby/numeric.h>
13#include <mruby/debug.h>
14
15#define FLAG_BYTEORDER_NATIVE 2
16#define FLAG_BYTEORDER_NONATIVE 0
17
18#ifdef MRB_USE_FLOAT
19#define MRB_FLOAT_FMT "%.8e"
20#else
21#define MRB_FLOAT_FMT "%.16e"
22#endif
23
24static size_t get_irep_record_size_1(mrb_state *mrb, mrb_irep *irep);
25
26#if UINT32_MAX > SIZE_MAX
27# error This code cannot be built on your environment.
28#endif
29
30static size_t
31write_padding(uint8_t *buf)
32{
33 const size_t align = MRB_DUMP_ALIGNMENT;
34 size_t pad_len = -(intptr_t)buf & (align-1);
35 if (pad_len > 0) {
36 memset(buf, 0, pad_len);
37 }
38 return pad_len;
39}
40
41static size_t
42get_irep_header_size(mrb_state *mrb)
43{
44 size_t size = 0;
45
46 size += sizeof(uint32_t) * 1;
47 size += sizeof(uint16_t) * 3;
48
49 return size;
50}
51
52static ptrdiff_t
53write_irep_header(mrb_state *mrb, mrb_irep *irep, uint8_t *buf)
54{
55 uint8_t *cur = buf;
56
57 cur += uint32_to_bin((uint32_t)get_irep_record_size_1(mrb, irep), cur); /* record size */
58 cur += uint16_to_bin((uint16_t)irep->nlocals, cur); /* number of local variable */
59 cur += uint16_to_bin((uint16_t)irep->nregs, cur); /* number of register variable */
60 cur += uint16_to_bin((uint16_t)irep->rlen, cur); /* number of child irep */
61
62 return cur - buf;
63}
64
65
66static size_t
67get_iseq_block_size(mrb_state *mrb, mrb_irep *irep)
68{
69 size_t size = 0;
70
71 size += sizeof(uint32_t); /* ilen */
72 size += sizeof(uint32_t); /* max padding */
73 size += sizeof(uint32_t) * irep->ilen; /* iseq(n) */
74
75 return size;
76}
77
78static ptrdiff_t
79write_iseq_block(mrb_state *mrb, mrb_irep *irep, uint8_t *buf, uint8_t flags)
80{
81 uint8_t *cur = buf;
82 uint32_t iseq_no;
83
84 cur += uint32_to_bin(irep->ilen, cur); /* number of opcode */
85 cur += write_padding(cur);
86 switch (flags & DUMP_ENDIAN_NAT) {
87 case DUMP_ENDIAN_BIG:
88 if (bigendian_p()) goto native;
89 for (iseq_no = 0; iseq_no < irep->ilen; iseq_no++) {
90 cur += uint32_to_bin(irep->iseq[iseq_no], cur); /* opcode */
91 }
92 break;
93 case DUMP_ENDIAN_LIL:
94 if (!bigendian_p()) goto native;
95 for (iseq_no = 0; iseq_no < irep->ilen; iseq_no++) {
96 cur += uint32l_to_bin(irep->iseq[iseq_no], cur); /* opcode */
97 }
98 break;
99
100 native:
101 case DUMP_ENDIAN_NAT:
102 memcpy(cur, irep->iseq, irep->ilen * sizeof(mrb_code));
103 cur += irep->ilen * sizeof(mrb_code);
104 break;
105 }
106
107 return cur - buf;
108}
109
110
111static size_t
112get_pool_block_size(mrb_state *mrb, mrb_irep *irep)
113{
114 size_t size = 0;
115 size_t pool_no;
116 mrb_value str;
117
118 size += sizeof(uint32_t); /* plen */
119 size += irep->plen * (sizeof(uint8_t) + sizeof(uint16_t)); /* len(n) */
120
121 for (pool_no = 0; pool_no < irep->plen; pool_no++) {
122 int ai = mrb_gc_arena_save(mrb);
123
124 switch (mrb_type(irep->pool[pool_no])) {
125 case MRB_TT_FIXNUM:
126 str = mrb_fixnum_to_str(mrb, irep->pool[pool_no], 10);
127 {
128 mrb_int len = RSTRING_LEN(str);
129 mrb_assert_int_fit(mrb_int, len, size_t, SIZE_MAX);
130 size += (size_t)len;
131 }
132 break;
133
134 case MRB_TT_FLOAT:
135 str = mrb_float_to_str(mrb, irep->pool[pool_no], MRB_FLOAT_FMT);
136 {
137 mrb_int len = RSTRING_LEN(str);
138 mrb_assert_int_fit(mrb_int, len, size_t, SIZE_MAX);
139 size += (size_t)len;
140 }
141 break;
142
143 case MRB_TT_STRING:
144 {
145 mrb_int len = RSTRING_LEN(irep->pool[pool_no]);
146 mrb_assert_int_fit(mrb_int, len, size_t, SIZE_MAX);
147 size += (size_t)len;
148 }
149 break;
150
151 default:
152 break;
153 }
154 mrb_gc_arena_restore(mrb, ai);
155 }
156
157 return size;
158}
159
160static ptrdiff_t
161write_pool_block(mrb_state *mrb, mrb_irep *irep, uint8_t *buf)
162{
163 size_t pool_no;
164 uint8_t *cur = buf;
165 uint16_t len;
166 mrb_value str;
167 const char *char_ptr;
168
169 cur += uint32_to_bin(irep->plen, cur); /* number of pool */
170
171 for (pool_no = 0; pool_no < irep->plen; pool_no++) {
172 int ai = mrb_gc_arena_save(mrb);
173
174 switch (mrb_type(irep->pool[pool_no])) {
175 case MRB_TT_FIXNUM:
176 cur += uint8_to_bin(IREP_TT_FIXNUM, cur); /* data type */
177 str = mrb_fixnum_to_str(mrb, irep->pool[pool_no], 10);
178 break;
179
180 case MRB_TT_FLOAT:
181 cur += uint8_to_bin(IREP_TT_FLOAT, cur); /* data type */
182 str = mrb_float_to_str(mrb, irep->pool[pool_no], MRB_FLOAT_FMT);
183 break;
184
185 case MRB_TT_STRING:
186 cur += uint8_to_bin(IREP_TT_STRING, cur); /* data type */
187 str = irep->pool[pool_no];
188 break;
189
190 default:
191 continue;
192 }
193
194 char_ptr = RSTRING_PTR(str);
195 {
196 mrb_int tlen = RSTRING_LEN(str);
197 mrb_assert_int_fit(mrb_int, tlen, uint16_t, UINT16_MAX);
198 len = (uint16_t)tlen;
199 }
200
201 cur += uint16_to_bin(len, cur); /* data length */
202 memcpy(cur, char_ptr, (size_t)len);
203 cur += len;
204
205 mrb_gc_arena_restore(mrb, ai);
206 }
207
208 return cur - buf;
209}
210
211
212static size_t
213get_syms_block_size(mrb_state *mrb, mrb_irep *irep)
214{
215 size_t size = 0;
216 uint32_t sym_no;
217 mrb_int len;
218
219 size += sizeof(uint32_t); /* slen */
220 for (sym_no = 0; sym_no < irep->slen; sym_no++) {
221 size += sizeof(uint16_t); /* snl(n) */
222 if (irep->syms[sym_no] != 0) {
223 mrb_sym2name_len(mrb, irep->syms[sym_no], &len);
224 size += len + 1; /* sn(n) + null char */
225 }
226 }
227
228 return size;
229}
230
231static ptrdiff_t
232write_syms_block(mrb_state *mrb, mrb_irep *irep, uint8_t *buf)
233{
234 uint32_t sym_no;
235 uint8_t *cur = buf;
236 const char *name;
237
238 cur += uint32_to_bin(irep->slen, cur); /* number of symbol */
239
240 for (sym_no = 0; sym_no < irep->slen; sym_no++) {
241 if (irep->syms[sym_no] != 0) {
242 mrb_int len;
243
244 name = mrb_sym2name_len(mrb, irep->syms[sym_no], &len);
245
246 mrb_assert_int_fit(mrb_int, len, uint16_t, UINT16_MAX);
247 cur += uint16_to_bin((uint16_t)len, cur); /* length of symbol name */
248 memcpy(cur, name, len); /* symbol name */
249 cur += (uint16_t)len;
250 *cur++ = '\0';
251 }
252 else {
253 cur += uint16_to_bin(MRB_DUMP_NULL_SYM_LEN, cur); /* length of symbol name */
254 }
255 }
256
257 return cur - buf;
258}
259
260static size_t
261get_irep_record_size_1(mrb_state *mrb, mrb_irep *irep)
262{
263 size_t size = 0;
264
265 size += get_irep_header_size(mrb);
266 size += get_iseq_block_size(mrb, irep);
267 size += get_pool_block_size(mrb, irep);
268 size += get_syms_block_size(mrb, irep);
269 return size;
270}
271
272static size_t
273get_irep_record_size(mrb_state *mrb, mrb_irep *irep)
274{
275 size_t size = 0;
276 size_t irep_no;
277
278 size = get_irep_record_size_1(mrb, irep);
279 for (irep_no = 0; irep_no < irep->rlen; irep_no++) {
280 size += get_irep_record_size(mrb, irep->reps[irep_no]);
281 }
282 return size;
283}
284
285static int
286write_irep_record(mrb_state *mrb, mrb_irep *irep, uint8_t *bin, size_t *irep_record_size, uint8_t flags)
287{
288 uint32_t i;
289 uint8_t *src = bin;
290
291 if (irep == NULL) {
292 return MRB_DUMP_INVALID_IREP;
293 }
294
295 *irep_record_size = get_irep_record_size_1(mrb, irep);
296 if (*irep_record_size == 0) {
297 return MRB_DUMP_GENERAL_FAILURE;
298 }
299
300 bin += write_irep_header(mrb, irep, bin);
301 bin += write_iseq_block(mrb, irep, bin, flags);
302 bin += write_pool_block(mrb, irep, bin);
303 bin += write_syms_block(mrb, irep, bin);
304
305 for (i = 0; i < irep->rlen; i++) {
306 int result;
307 size_t rsize;
308
309 result = write_irep_record(mrb, irep->reps[i], bin, &rsize, flags);
310 if (result != MRB_DUMP_OK) {
311 return result;
312 }
313 bin += rsize;
314 }
315 *irep_record_size = bin - src;
316 return MRB_DUMP_OK;
317}
318
319static uint32_t
320write_footer(mrb_state *mrb, uint8_t *bin)
321{
322 struct rite_binary_footer footer;
323
324 memcpy(footer.section_ident, RITE_BINARY_EOF, sizeof(footer.section_ident));
325 uint32_to_bin(sizeof(struct rite_binary_footer), footer.section_size);
326 memcpy(bin, &footer, sizeof(struct rite_binary_footer));
327
328 return sizeof(struct rite_binary_footer);
329}
330
331
332static int
333write_section_irep_header(mrb_state *mrb, size_t section_size, uint8_t *bin)
334{
335 struct rite_section_irep_header *header = (struct rite_section_irep_header*)bin;
336
337 memcpy(header->section_ident, RITE_SECTION_IREP_IDENT, sizeof(header->section_ident));
338
339 mrb_assert_int_fit(size_t, section_size, uint32_t, UINT32_MAX);
340 uint32_to_bin((uint32_t)section_size, header->section_size);
341 memcpy(header->rite_version, RITE_VM_VER, sizeof(header->rite_version));
342
343 return MRB_DUMP_OK;
344}
345
346static int
347write_section_irep(mrb_state *mrb, mrb_irep *irep, uint8_t *bin, size_t *len_p, uint8_t flags)
348{
349 int result;
350 size_t rsize = 0;
351 uint8_t *cur = bin;
352
353 if (mrb == NULL || bin == NULL) {
354 return MRB_DUMP_INVALID_ARGUMENT;
355 }
356
357 cur += sizeof(struct rite_section_irep_header);
358
359 result = write_irep_record(mrb, irep, cur, &rsize, flags);
360 if (result != MRB_DUMP_OK) {
361 return result;
362 }
363 *len_p = cur - bin + rsize;
364 write_section_irep_header(mrb, *len_p, bin);
365
366 return MRB_DUMP_OK;
367}
368
369static int
370write_section_lineno_header(mrb_state *mrb, size_t section_size, uint8_t *bin)
371{
372 struct rite_section_lineno_header *header = (struct rite_section_lineno_header*)bin;
373
374 memcpy(header->section_ident, RITE_SECTION_LINENO_IDENT, sizeof(header->section_ident));
375 uint32_to_bin((uint32_t)section_size, header->section_size);
376
377 return MRB_DUMP_OK;
378}
379
380static size_t
381get_lineno_record_size(mrb_state *mrb, mrb_irep *irep)
382{
383 size_t size = 0;
384
385 size += sizeof(uint32_t); /* record size */
386 size += sizeof(uint16_t); /* filename size */
387 if (irep->filename) {
388 size += strlen(irep->filename); /* filename */
389 }
390 size += sizeof(uint32_t); /* niseq */
391 if (irep->lines) {
392 size += sizeof(uint16_t) * irep->ilen; /* lineno */
393 }
394
395 return size;
396}
397
398static size_t
399write_lineno_record_1(mrb_state *mrb, mrb_irep *irep, uint8_t* bin)
400{
401 uint8_t *cur = bin;
402 size_t iseq_no;
403 size_t filename_len;
404 ptrdiff_t diff;
405
406 cur += sizeof(uint32_t); /* record size */
407
408 if (irep->filename) {
409 filename_len = strlen(irep->filename);
410 }
411 else {
412 filename_len = 0;
413 }
414 mrb_assert_int_fit(size_t, filename_len, uint16_t, UINT16_MAX);
415 cur += uint16_to_bin((uint16_t)filename_len, cur); /* filename size */
416
417 if (filename_len) {
418 memcpy(cur, irep->filename, filename_len);
419 cur += filename_len; /* filename */
420 }
421
422 if (irep->lines) {
423 mrb_assert_int_fit(size_t, irep->ilen, uint32_t, UINT32_MAX);
424 cur += uint32_to_bin((uint32_t)(irep->ilen), cur); /* niseq */
425 for (iseq_no = 0; iseq_no < irep->ilen; iseq_no++) {
426 cur += uint16_to_bin(irep->lines[iseq_no], cur); /* opcode */
427 }
428 }
429 else {
430 cur += uint32_to_bin(0, cur); /* niseq */
431 }
432
433 diff = cur - bin;
434 mrb_assert_int_fit(ptrdiff_t, diff, uint32_t, UINT32_MAX);
435
436 uint32_to_bin((uint32_t)diff, bin); /* record size */
437
438 mrb_assert_int_fit(ptrdiff_t, diff, size_t, SIZE_MAX);
439 return (size_t)diff;
440}
441
442static size_t
443write_lineno_record(mrb_state *mrb, mrb_irep *irep, uint8_t* bin)
444{
445 size_t i;
446 size_t rlen, size = 0;
447
448 rlen = write_lineno_record_1(mrb, irep, bin);
449 bin += rlen;
450 size += rlen;
451 for (i=0; i<irep->rlen; i++) {
452 rlen = write_lineno_record(mrb, irep, bin);
453 bin += rlen;
454 size += rlen;
455 }
456 return size;
457}
458
459static int
460write_section_lineno(mrb_state *mrb, mrb_irep *irep, uint8_t *bin)
461{
462 size_t section_size = 0;
463 size_t rlen = 0; /* size of irep record */
464 uint8_t *cur = bin;
465
466 if (mrb == NULL || bin == NULL) {
467 return MRB_DUMP_INVALID_ARGUMENT;
468 }
469
470 cur += sizeof(struct rite_section_lineno_header);
471 section_size += sizeof(struct rite_section_lineno_header);
472
473 rlen = write_lineno_record(mrb, irep, cur);
474 section_size += rlen;
475
476 write_section_lineno_header(mrb, section_size, bin);
477
478 return MRB_DUMP_OK;
479}
480
481static size_t
482get_debug_record_size(mrb_state *mrb, mrb_irep *irep)
483{
484 size_t ret = 0;
485 uint16_t f_idx;
486 size_t i;
487
488 ret += sizeof(uint32_t); /* record size */
489 ret += sizeof(uint16_t); /* file count */
490
491 for (f_idx = 0; f_idx < irep->debug_info->flen; ++f_idx) {
492 mrb_irep_debug_info_file const* file = irep->debug_info->files[f_idx];
493
494 ret += sizeof(uint32_t); /* position */
495 ret += sizeof(uint16_t); /* filename index */
496
497 /* lines */
498 ret += sizeof(uint32_t); /* entry count */
499 ret += sizeof(uint8_t); /* line type */
500 switch (file->line_type) {
501 case mrb_debug_line_ary:
502 ret += sizeof(uint16_t) * (size_t)(file->line_entry_count);
503 break;
504
505 case mrb_debug_line_flat_map:
506 ret += (sizeof(uint32_t) + sizeof(uint16_t)) * (size_t)(file->line_entry_count);
507 break;
508
509 default: mrb_assert(0); break;
510 }
511 }
512 for (i=0; i<irep->rlen; i++) {
513 ret += get_debug_record_size(mrb, irep->reps[i]);
514 }
515
516 return ret;
517}
518
519static int
520find_filename_index(const mrb_sym *ary, int ary_len, mrb_sym s)
521{
522 int i;
523
524 for (i = 0; i < ary_len; ++i) {
525 if (ary[i] == s) { return i; }
526 }
527 return -1;
528}
529
530static size_t
531get_filename_table_size(mrb_state *mrb, mrb_irep *irep, mrb_sym **fp, uint16_t *lp)
532{
533 mrb_sym *filenames = *fp;
534 size_t i, size = 0;
535 mrb_irep_debug_info *di = irep->debug_info;
536
537 mrb_assert(lp);
538 for (i = 0; i < di->flen; ++i) {
539 mrb_irep_debug_info_file *file;
540 mrb_int filename_len;
541
542 file = di->files[i];
543 if (find_filename_index(filenames, *lp, file->filename_sym) == -1) {
544 /* register filename */
545 *lp += 1;
546 *fp = filenames = (mrb_sym *)mrb_realloc(mrb, filenames, sizeof(mrb_sym) * (*lp));
547 filenames[*lp - 1] = file->filename_sym;
548
549 /* filename */
550 mrb_sym2name_len(mrb, file->filename_sym, &filename_len);
551 size += sizeof(uint16_t) + (size_t)filename_len;
552 }
553 }
554 for (i=0; i<irep->rlen; i++) {
555 size += get_filename_table_size(mrb, irep->reps[i], fp, lp);
556 }
557 return size;
558}
559
560static size_t
561write_debug_record_1(mrb_state *mrb, mrb_irep *irep, uint8_t *bin, mrb_sym const* filenames, uint16_t filenames_len)
562{
563 uint8_t *cur;
564 uint16_t f_idx;
565 ptrdiff_t ret;
566
567 cur = bin + sizeof(uint32_t); /* skip record size */
568 cur += uint16_to_bin(irep->debug_info->flen, cur); /* file count */
569
570 for (f_idx = 0; f_idx < irep->debug_info->flen; ++f_idx) {
571 int filename_idx;
572 const mrb_irep_debug_info_file *file = irep->debug_info->files[f_idx];
573
574 /* position */
575 cur += uint32_to_bin(file->start_pos, cur);
576
577 /* filename index */
578 filename_idx = find_filename_index(filenames, filenames_len,
579 file->filename_sym);
580 mrb_assert_int_fit(int, filename_idx, uint16_t, UINT16_MAX);
581 cur += uint16_to_bin((uint16_t)filename_idx, cur);
582
583 /* lines */
584 cur += uint32_to_bin(file->line_entry_count, cur);
585 cur += uint8_to_bin(file->line_type, cur);
586 switch (file->line_type) {
587 case mrb_debug_line_ary: {
588 uint32_t l;
589 for (l = 0; l < file->line_entry_count; ++l) {
590 cur += uint16_to_bin(file->lines.ary[l], cur);
591 }
592 } break;
593
594 case mrb_debug_line_flat_map: {
595 uint32_t line;
596 for (line = 0; line < file->line_entry_count; ++line) {
597 cur += uint32_to_bin(file->lines.flat_map[line].start_pos, cur);
598 cur += uint16_to_bin(file->lines.flat_map[line].line, cur);
599 }
600 } break;
601
602 default: mrb_assert(0); break;
603 }
604 }
605
606 ret = cur - bin;
607 mrb_assert_int_fit(ptrdiff_t, ret, uint32_t, UINT32_MAX);
608 uint32_to_bin(ret, bin);
609
610 mrb_assert_int_fit(ptrdiff_t, ret, size_t, SIZE_MAX);
611 return (size_t)ret;
612}
613
614static size_t
615write_debug_record(mrb_state *mrb, mrb_irep *irep, uint8_t *bin, mrb_sym const* filenames, uint16_t filenames_len)
616{
617 size_t size, len;
618 size_t irep_no;
619
620 size = len = write_debug_record_1(mrb, irep, bin, filenames, filenames_len);
621 bin += len;
622 for (irep_no = 0; irep_no < irep->rlen; irep_no++) {
623 len = write_debug_record(mrb, irep->reps[irep_no], bin, filenames, filenames_len);
624 bin += len;
625 size += len;
626 }
627
628 mrb_assert(size == get_debug_record_size(mrb, irep));
629 return size;
630}
631
632static int
633write_section_debug(mrb_state *mrb, mrb_irep *irep, uint8_t *cur, mrb_sym const *filenames, uint16_t filenames_len)
634{
635 size_t section_size = 0;
636 const uint8_t *bin = cur;
637 struct rite_section_debug_header *header;
638 size_t dlen;
639 uint16_t i;
640 char const *sym; mrb_int sym_len;
641
642 if (mrb == NULL || cur == NULL) {
643 return MRB_DUMP_INVALID_ARGUMENT;
644 }
645
646 header = (struct rite_section_debug_header *)bin;
647 cur += sizeof(struct rite_section_debug_header);
648 section_size += sizeof(struct rite_section_debug_header);
649
650 /* filename table */
651 cur += uint16_to_bin(filenames_len, cur);
652 section_size += sizeof(uint16_t);
653 for (i = 0; i < filenames_len; ++i) {
654 sym = mrb_sym2name_len(mrb, filenames[i], &sym_len);
655 mrb_assert(sym);
656 cur += uint16_to_bin(sym_len, cur);
657 memcpy(cur, sym, sym_len);
658 cur += sym_len;
659 section_size += sizeof(uint16_t) + sym_len;
660 }
661
662 /* debug records */
663 dlen = write_debug_record(mrb, irep, cur, filenames, filenames_len);
664 section_size += dlen;
665
666 memcpy(header->section_ident, RITE_SECTION_DEBUG_IDENT, sizeof(header->section_ident));
667 mrb_assert(section_size <= INT32_MAX);
668 uint32_to_bin(section_size, header->section_size);
669
670 return MRB_DUMP_OK;
671}
672
673static void
674create_lv_sym_table(mrb_state *mrb, const mrb_irep *irep, mrb_sym **syms, uint32_t *syms_len)
675{
676 size_t i;
677
678 if (*syms == NULL) {
679 *syms = (mrb_sym*)mrb_malloc(mrb, sizeof(mrb_sym) * 1);
680 }
681
682 for (i = 0; i + 1 < irep->nlocals; ++i) {
683 mrb_sym const name = irep->lv[i].name;
684 if (name == 0) continue;
685 if (find_filename_index(*syms, *syms_len, name) != -1) continue;
686
687 ++(*syms_len);
688 *syms = (mrb_sym*)mrb_realloc(mrb, *syms, sizeof(mrb_sym) * (*syms_len));
689 (*syms)[*syms_len - 1] = name;
690 }
691
692 for (i = 0; i < irep->rlen; ++i) {
693 create_lv_sym_table(mrb, irep->reps[i], syms, syms_len);
694 }
695}
696
697static int
698write_lv_sym_table(mrb_state *mrb, uint8_t **start, mrb_sym const *syms, uint32_t syms_len)
699{
700 uint8_t *cur = *start;
701 uint32_t i;
702 const char *str;
703 mrb_int str_len;
704
705 cur += uint32_to_bin(syms_len, cur);
706
707 for (i = 0; i < syms_len; ++i) {
708 str = mrb_sym2name_len(mrb, syms[i], &str_len);
709 cur += uint16_to_bin(str_len, cur);
710 memcpy(cur, str, str_len);
711 cur += str_len;
712 }
713
714 *start = cur;
715
716 return MRB_DUMP_OK;
717}
718
719static int
720write_lv_record(mrb_state *mrb, const mrb_irep *irep, uint8_t **start, mrb_sym const *syms, uint32_t syms_len)
721{
722 uint8_t *cur = *start;
723 size_t i;
724
725 for (i = 0; i + 1 < irep->nlocals; ++i) {
726 if (irep->lv[i].name == 0) {
727 cur += uint16_to_bin(RITE_LV_NULL_MARK, cur);
728 cur += uint16_to_bin(0, cur);
729 }
730 else {
731 int const sym_idx = find_filename_index(syms, syms_len, irep->lv[i].name);
732 mrb_assert(sym_idx != -1); /* local variable name must be in syms */
733
734 cur += uint16_to_bin(sym_idx, cur);
735 cur += uint16_to_bin(irep->lv[i].r, cur);
736 }
737 }
738
739 for (i = 0; i < irep->rlen; ++i) {
740 write_lv_record(mrb, irep->reps[i], &cur, syms, syms_len);
741 }
742
743 *start = cur;
744
745 return MRB_DUMP_OK;
746}
747
748static size_t
749get_lv_record_size(mrb_state *mrb, mrb_irep *irep)
750{
751 size_t ret = 0, i;
752
753 ret += (sizeof(uint16_t) + sizeof(uint16_t)) * (irep->nlocals - 1);
754
755 for (i = 0; i < irep->rlen; ++i) {
756 ret += get_lv_record_size(mrb, irep->reps[i]);
757 }
758
759 return ret;
760}
761
762static size_t
763get_lv_section_size(mrb_state *mrb, mrb_irep *irep, mrb_sym const *syms, uint32_t syms_len)
764{
765 size_t ret = 0, i;
766
767 ret += sizeof(uint32_t); /* syms_len */
768 ret += sizeof(uint16_t) * syms_len; /* symbol name lengths */
769 for (i = 0; i < syms_len; ++i) {
770 mrb_int str_len;
771 mrb_sym2name_len(mrb, syms[i], &str_len);
772 ret += str_len;
773 }
774
775 ret += get_lv_record_size(mrb, irep);
776
777 return ret;
778}
779
780static int
781write_section_lv(mrb_state *mrb, mrb_irep *irep, uint8_t *start, mrb_sym const *syms, uint32_t const syms_len)
782{
783 uint8_t *cur = start;
784 struct rite_section_lv_header *header;
785 ptrdiff_t diff;
786 int result = MRB_DUMP_OK;
787
788 if (mrb == NULL || cur == NULL) {
789 return MRB_DUMP_INVALID_ARGUMENT;
790 }
791
792 header = (struct rite_section_lv_header*)cur;
793 cur += sizeof(struct rite_section_lv_header);
794
795 result = write_lv_sym_table(mrb, &cur, syms, syms_len);
796 if (result != MRB_DUMP_OK) {
797 goto lv_section_exit;
798 }
799
800 result = write_lv_record(mrb, irep, &cur, syms, syms_len);
801 if (result != MRB_DUMP_OK) {
802 goto lv_section_exit;
803 }
804
805 memcpy(header->section_ident, RITE_SECTION_LV_IDENT, sizeof(header->section_ident));
806
807 diff = cur - start;
808 mrb_assert_int_fit(ptrdiff_t, diff, size_t, SIZE_MAX);
809 uint32_to_bin(diff, header->section_size);
810
811lv_section_exit:
812 return result;
813}
814
815static int
816write_rite_binary_header(mrb_state *mrb, size_t binary_size, uint8_t *bin, uint8_t flags)
817{
818 struct rite_binary_header *header = (struct rite_binary_header *)bin;
819 uint16_t crc;
820 uint32_t offset;
821
822 switch (flags & DUMP_ENDIAN_NAT) {
823 endian_big:
824 case DUMP_ENDIAN_BIG:
825 memcpy(header->binary_ident, RITE_BINARY_IDENT, sizeof(header->binary_ident));
826 break;
827 endian_little:
828 case DUMP_ENDIAN_LIL:
829 memcpy(header->binary_ident, RITE_BINARY_IDENT_LIL, sizeof(header->binary_ident));
830 break;
831
832 case DUMP_ENDIAN_NAT:
833 if (bigendian_p()) goto endian_big;
834 goto endian_little;
835 break;
836 }
837
838 memcpy(header->binary_version, RITE_BINARY_FORMAT_VER, sizeof(header->binary_version));
839 memcpy(header->compiler_name, RITE_COMPILER_NAME, sizeof(header->compiler_name));
840 memcpy(header->compiler_version, RITE_COMPILER_VERSION, sizeof(header->compiler_version));
841 mrb_assert(binary_size <= UINT32_MAX);
842 uint32_to_bin((uint32_t)binary_size, header->binary_size);
843
844 offset = (&(header->binary_crc[0]) - bin) + sizeof(uint16_t);
845 crc = calc_crc_16_ccitt(bin + offset, binary_size - offset, 0);
846 uint16_to_bin(crc, header->binary_crc);
847
848 return MRB_DUMP_OK;
849}
850
851static mrb_bool
852is_debug_info_defined(mrb_irep *irep)
853{
854 size_t i;
855
856 if (!irep->debug_info) return FALSE;
857 for (i=0; i<irep->rlen; i++) {
858 if (!is_debug_info_defined(irep->reps[i])) return FALSE;
859 }
860 return TRUE;
861}
862
863static mrb_bool
864is_lv_defined(mrb_irep *irep)
865{
866 size_t i;
867
868 if (irep->lv) { return TRUE; }
869
870 for (i = 0; i < irep->rlen; ++i) {
871 if (is_lv_defined(irep->reps[i])) { return TRUE; }
872 }
873
874 return FALSE;
875}
876
877static uint8_t
878dump_flags(uint8_t flags, uint8_t native)
879{
880 if (native == FLAG_BYTEORDER_NATIVE) {
881 if ((flags & DUMP_ENDIAN_NAT) == 0) {
882 return (flags & DUMP_DEBUG_INFO) | DUMP_ENDIAN_NAT;
883 }
884 return flags;
885 }
886 if ((flags & DUMP_ENDIAN_NAT) == 0) {
887 return (flags & DUMP_DEBUG_INFO) | DUMP_ENDIAN_BIG;
888 }
889 return flags;
890}
891
892static int
893dump_irep(mrb_state *mrb, mrb_irep *irep, uint8_t flags, uint8_t **bin, size_t *bin_size)
894{
895 int result = MRB_DUMP_GENERAL_FAILURE;
896 size_t malloc_size;
897 size_t section_irep_size;
898 size_t section_lineno_size = 0, section_lv_size = 0;
899 uint8_t *cur = NULL;
900 mrb_bool const debug_info_defined = is_debug_info_defined(irep), lv_defined = is_lv_defined(irep);
901 mrb_sym *lv_syms = NULL; uint32_t lv_syms_len = 0;
902 mrb_sym *filenames = NULL; uint16_t filenames_len = 0;
903
904 if (mrb == NULL) {
905 *bin = NULL;
906 return MRB_DUMP_GENERAL_FAILURE;
907 }
908
909 section_irep_size = sizeof(struct rite_section_irep_header);
910 section_irep_size += get_irep_record_size(mrb, irep);
911
912 /* DEBUG section size */
913 if (flags & DUMP_DEBUG_INFO) {
914 if (debug_info_defined) {
915 section_lineno_size += sizeof(struct rite_section_debug_header);
916 /* filename table */
917 filenames = (mrb_sym*)mrb_malloc(mrb, sizeof(mrb_sym) + 1);
918
919 /* filename table size */
920 section_lineno_size += sizeof(uint16_t);
921 section_lineno_size += get_filename_table_size(mrb, irep, &filenames, &filenames_len);
922
923 section_lineno_size += get_debug_record_size(mrb, irep);
924 }
925 else {
926 section_lineno_size += sizeof(struct rite_section_lineno_header);
927 section_lineno_size += get_lineno_record_size(mrb, irep);
928 }
929 }
930
931 if (lv_defined) {
932 section_lv_size += sizeof(struct rite_section_lv_header);
933 create_lv_sym_table(mrb, irep, &lv_syms, &lv_syms_len);
934 section_lv_size += get_lv_section_size(mrb, irep, lv_syms, lv_syms_len);
935 }
936
937 malloc_size = sizeof(struct rite_binary_header) +
938 section_irep_size + section_lineno_size + section_lv_size +
939 sizeof(struct rite_binary_footer);
940 cur = *bin = (uint8_t*)mrb_malloc(mrb, malloc_size);
941 cur += sizeof(struct rite_binary_header);
942
943 result = write_section_irep(mrb, irep, cur, &section_irep_size, flags);
944 if (result != MRB_DUMP_OK) {
945 goto error_exit;
946 }
947 cur += section_irep_size;
948 *bin_size = sizeof(struct rite_binary_header) +
949 section_irep_size + section_lineno_size + section_lv_size +
950 sizeof(struct rite_binary_footer);
951
952 /* write DEBUG section */
953 if (flags & DUMP_DEBUG_INFO) {
954 if (debug_info_defined) {
955 result = write_section_debug(mrb, irep, cur, filenames, filenames_len);
956 }
957 else {
958 result = write_section_lineno(mrb, irep, cur);
959 }
960 if (result != MRB_DUMP_OK) {
961 goto error_exit;
962 }
963 cur += section_lineno_size;
964 }
965
966 if (lv_defined) {
967 result = write_section_lv(mrb, irep, cur, lv_syms, lv_syms_len);
968 if (result != MRB_DUMP_OK) {
969 goto error_exit;
970 }
971 cur += section_lv_size;
972 }
973
974 write_footer(mrb, cur);
975 write_rite_binary_header(mrb, *bin_size, *bin, flags);
976
977error_exit:
978 if (result != MRB_DUMP_OK) {
979 mrb_free(mrb, *bin);
980 *bin = NULL;
981 }
982 mrb_free(mrb, lv_syms);
983 mrb_free(mrb, filenames);
984 return result;
985}
986
987int
988mrb_dump_irep(mrb_state *mrb, mrb_irep *irep, uint8_t flags, uint8_t **bin, size_t *bin_size)
989{
990 return dump_irep(mrb, irep, dump_flags(flags, FLAG_BYTEORDER_NONATIVE), bin, bin_size);
991}
992
993#ifndef MRB_DISABLE_STDIO
994
995int
996mrb_dump_irep_binary(mrb_state *mrb, mrb_irep *irep, uint8_t flags, FILE* fp)
997{
998 uint8_t *bin = NULL;
999 size_t bin_size = 0;
1000 int result;
1001
1002 if (fp == NULL) {
1003 return MRB_DUMP_INVALID_ARGUMENT;
1004 }
1005
1006 result = dump_irep(mrb, irep, dump_flags(flags, FLAG_BYTEORDER_NONATIVE), &bin, &bin_size);
1007 if (result == MRB_DUMP_OK) {
1008 if (fwrite(bin, sizeof(bin[0]), bin_size, fp) != bin_size) {
1009 result = MRB_DUMP_WRITE_FAULT;
1010 }
1011 }
1012
1013 mrb_free(mrb, bin);
1014 return result;
1015}
1016
1017static mrb_bool
1018dump_bigendian_p(uint8_t flags)
1019{
1020 switch (flags & DUMP_ENDIAN_NAT) {
1021 case DUMP_ENDIAN_BIG:
1022 return TRUE;
1023 case DUMP_ENDIAN_LIL:
1024 return FALSE;
1025 default:
1026 case DUMP_ENDIAN_NAT:
1027 return bigendian_p();
1028 }
1029}
1030
1031int
1032mrb_dump_irep_cfunc(mrb_state *mrb, mrb_irep *irep, uint8_t flags, FILE *fp, const char *initname)
1033{
1034 uint8_t *bin = NULL;
1035 size_t bin_size = 0, bin_idx = 0;
1036 int result;
1037
1038 if (fp == NULL || initname == NULL || initname[0] == '\0') {
1039 return MRB_DUMP_INVALID_ARGUMENT;
1040 }
1041 flags = dump_flags(flags, FLAG_BYTEORDER_NATIVE);
1042 result = dump_irep(mrb, irep, flags, &bin, &bin_size);
1043 if (result == MRB_DUMP_OK) {
1044 if (!dump_bigendian_p(flags)) {
1045 if (fprintf(fp, "/* dumped in little endian order.\n"
1046 " use `mrbc -E` option for big endian CPU. */\n") < 0) {
1047 mrb_free(mrb, bin);
1048 return MRB_DUMP_WRITE_FAULT;
1049 }
1050 }
1051 else {
1052 if (fprintf(fp, "/* dumped in big endian order.\n"
1053 " use `mrbc -e` option for better performance on little endian CPU. */\n") < 0) {
1054 mrb_free(mrb, bin);
1055 return MRB_DUMP_WRITE_FAULT;
1056 }
1057 }
1058 if (fprintf(fp, "#include <stdint.h>\n") < 0) { /* for uint8_t under at least Darwin */
1059 mrb_free(mrb, bin);
1060 return MRB_DUMP_WRITE_FAULT;
1061 }
1062 if (fprintf(fp,
1063 "extern const uint8_t %s[];\n"
1064 "const uint8_t\n"
1065 "#if defined __GNUC__\n"
1066 "__attribute__((aligned(%u)))\n"
1067 "#elif defined _MSC_VER\n"
1068 "__declspec(align(%u))\n"
1069 "#endif\n"
1070 "%s[] = {",
1071 initname,
1072 (uint16_t)MRB_DUMP_ALIGNMENT, (uint16_t)MRB_DUMP_ALIGNMENT, initname) < 0) {
1073 mrb_free(mrb, bin);
1074 return MRB_DUMP_WRITE_FAULT;
1075 }
1076 while (bin_idx < bin_size) {
1077 if (bin_idx % 16 == 0) {
1078 if (fputs("\n", fp) == EOF) {
1079 mrb_free(mrb, bin);
1080 return MRB_DUMP_WRITE_FAULT;
1081 }
1082 }
1083 if (fprintf(fp, "0x%02x,", bin[bin_idx++]) < 0) {
1084 mrb_free(mrb, bin);
1085 return MRB_DUMP_WRITE_FAULT;
1086 }
1087 }
1088 if (fputs("\n};\n", fp) == EOF) {
1089 mrb_free(mrb, bin);
1090 return MRB_DUMP_WRITE_FAULT;
1091 }
1092 }
1093
1094 mrb_free(mrb, bin);
1095 return result;
1096}
1097
1098#endif /* MRB_DISABLE_STDIO */
Note: See TracBrowser for help on using the repository browser.