source: EcnlProtoTool/trunk/mruby-2.1.1/src/backtrace.c@ 439

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

mrubyを2.1.1に更新

  • Property svn:eol-style set to native
  • Property svn:mime-type set to text/x-csrc;charset=UTF-8
File size: 5.7 KB
RevLine 
[270]1/*
2** backtrace.c -
3**
4** See Copyright Notice in mruby.h
5*/
6
[331]7#include <mruby.h>
8#include <mruby/variable.h>
9#include <mruby/proc.h>
10#include <mruby/array.h>
11#include <mruby/string.h>
12#include <mruby/class.h>
13#include <mruby/debug.h>
14#include <mruby/error.h>
15#include <mruby/numeric.h>
16#include <mruby/data.h>
[270]17
18struct backtrace_location {
[439]19 int32_t lineno;
20 mrb_sym method_id;
[270]21 const char *filename;
22};
23
[439]24typedef void (*each_backtrace_func)(mrb_state*, const struct backtrace_location*, void*);
[270]25
[331]26static const mrb_data_type bt_type = { "Backtrace", mrb_free };
[270]27
[439]28mrb_value mrb_exc_inspect(mrb_state *mrb, mrb_value exc);
29mrb_value mrb_unpack_backtrace(mrb_state *mrb, mrb_value backtrace);
30
[270]31static void
[439]32each_backtrace(mrb_state *mrb, ptrdiff_t ciidx, const mrb_code *pc0, each_backtrace_func func, void *data)
[270]33{
[439]34 ptrdiff_t i;
[270]35
36 if (ciidx >= mrb->c->ciend - mrb->c->cibase)
37 ciidx = 10; /* ciidx is broken... */
38
[439]39 for (i=ciidx; i >= 0; i--) {
[270]40 struct backtrace_location loc;
41 mrb_callinfo *ci;
42 mrb_irep *irep;
[439]43 const mrb_code *pc;
[270]44
45 ci = &mrb->c->cibase[i];
46
47 if (!ci->proc) continue;
48 if (MRB_PROC_CFUNC_P(ci->proc)) continue;
49
50 irep = ci->proc->body.irep;
[331]51 if (!irep) continue;
[270]52
53 if (mrb->c->cibase[i].err) {
54 pc = mrb->c->cibase[i].err;
55 }
56 else if (i+1 <= ciidx) {
[439]57 if (!mrb->c->cibase[i + 1].pc) continue;
58 pc = &mrb->c->cibase[i+1].pc[-1];
[270]59 }
60 else {
61 pc = pc0;
62 }
63
[439]64 loc.lineno = mrb_debug_get_line(mrb, irep, pc - irep->iseq);
[270]65 if (loc.lineno == -1) continue;
66
[439]67 loc.filename = mrb_debug_get_filename(mrb, irep, pc - irep->iseq);
[270]68 if (!loc.filename) {
69 loc.filename = "(unknown)";
70 }
71
[331]72 loc.method_id = ci->mid;
[439]73 func(mrb, &loc, data);
[270]74 }
75}
76
[331]77#ifndef MRB_DISABLE_STDIO
78
[270]79static void
[439]80print_backtrace(mrb_state *mrb, struct RObject *exc, mrb_value backtrace)
[270]81{
[439]82 mrb_int i;
83 mrb_int n = RARRAY_LEN(backtrace);
84 mrb_value *loc, mesg;
[331]85 FILE *stream = stderr;
[270]86
[439]87 if (n != 0) {
88 fprintf(stream, "trace (most recent call last):\n");
89 for (i=n-1,loc=&RARRAY_PTR(backtrace)[i]; i>0; i--,loc--) {
90 if (mrb_string_p(*loc)) {
91 fprintf(stream, "\t[%d] %.*s\n",
92 (int)i, (int)RSTRING_LEN(*loc), RSTRING_PTR(*loc));
93 }
[331]94 }
[439]95 if (mrb_string_p(*loc)) {
96 fprintf(stream, "%.*s: ", (int)RSTRING_LEN(*loc), RSTRING_PTR(*loc));
[331]97 }
98 }
[439]99 mesg = mrb_exc_inspect(mrb, mrb_obj_value(exc));
100 fprintf(stream, "%.*s\n", (int)RSTRING_LEN(mesg), RSTRING_PTR(mesg));
[270]101}
102
[331]103/* mrb_print_backtrace
[270]104
[331]105 function to retrieve backtrace information from the last exception.
[270]106*/
107
108MRB_API void
109mrb_print_backtrace(mrb_state *mrb)
110{
[331]111 mrb_value backtrace;
[270]112
[331]113 if (!mrb->exc) {
[270]114 return;
115 }
116
[331]117 backtrace = mrb_obj_iv_get(mrb, mrb->exc, mrb_intern_lit(mrb, "backtrace"));
118 if (mrb_nil_p(backtrace)) return;
[439]119 if (!mrb_array_p(backtrace)) backtrace = mrb_unpack_backtrace(mrb, backtrace);
120 print_backtrace(mrb, mrb->exc, backtrace);
[270]121}
122#else
123
124MRB_API void
125mrb_print_backtrace(mrb_state *mrb)
126{
127}
128
129#endif
130
[331]131static void
[439]132count_backtrace_i(mrb_state *mrb,
133 const struct backtrace_location *loc,
134 void *data)
135{
136 int *lenp = (int*)data;
137
138 (*lenp)++;
139}
140
141static void
[331]142pack_backtrace_i(mrb_state *mrb,
[439]143 const struct backtrace_location *loc,
[331]144 void *data)
[270]145{
[439]146 struct backtrace_location **pptr = (struct backtrace_location**)data;
147 struct backtrace_location *ptr = *pptr;
[270]148
[439]149 *ptr = *loc;
150 *pptr = ptr+1;
[331]151}
[270]152
[331]153static mrb_value
154packed_backtrace(mrb_state *mrb)
155{
156 struct RData *backtrace;
157 ptrdiff_t ciidx = mrb->c->ci - mrb->c->cibase;
[439]158 int len = 0;
159 int size;
[331]160 void *ptr;
161
[439]162 each_backtrace(mrb, ciidx, mrb->c->ci->pc, count_backtrace_i, &len);
163 size = len * sizeof(struct backtrace_location);
164 ptr = mrb_malloc(mrb, size);
[331]165 backtrace = mrb_data_object_alloc(mrb, NULL, ptr, &bt_type);
[439]166 backtrace->flags = (uint32_t)len;
167 each_backtrace(mrb, ciidx, mrb->c->ci->pc, pack_backtrace_i, &ptr);
[331]168 return mrb_obj_value(backtrace);
[270]169}
170
[331]171void
172mrb_keep_backtrace(mrb_state *mrb, mrb_value exc)
173{
[439]174 mrb_sym sym = mrb_intern_lit(mrb, "backtrace");
[331]175 mrb_value backtrace;
[439]176 int ai;
[331]177
[439]178 if (mrb_iv_defined(mrb, exc, sym)) return;
179 ai = mrb_gc_arena_save(mrb);
[331]180 backtrace = packed_backtrace(mrb);
[439]181 mrb_iv_set(mrb, exc, sym, backtrace);
[331]182 mrb_gc_arena_restore(mrb, ai);
183}
184
185mrb_value
186mrb_unpack_backtrace(mrb_state *mrb, mrb_value backtrace)
187{
[439]188 const struct backtrace_location *bt;
[331]189 mrb_int n, i;
[439]190 int ai;
[331]191
[439]192 if (mrb_nil_p(backtrace)) {
193 empty_backtrace:
194 return mrb_ary_new_capa(mrb, 0);
195 }
[331]196 if (mrb_array_p(backtrace)) return backtrace;
197 bt = (struct backtrace_location*)mrb_data_check_get_ptr(mrb, backtrace, &bt_type);
[439]198 if (bt == NULL) goto empty_backtrace;
[331]199 n = (mrb_int)RDATA(backtrace)->flags;
200 backtrace = mrb_ary_new_capa(mrb, n);
[439]201 ai = mrb_gc_arena_save(mrb);
[331]202 for (i = 0; i < n; i++) {
[439]203 const struct backtrace_location *entry = &bt[i];
[331]204 mrb_value btline;
205
[439]206 btline = mrb_format(mrb, "%s:%d", entry->filename, (int)entry->lineno);
[331]207 if (entry->method_id != 0) {
208 mrb_str_cat_lit(mrb, btline, ":in ");
[439]209 mrb_str_cat_cstr(mrb, btline, mrb_sym_name(mrb, entry->method_id));
[331]210 }
211 mrb_ary_push(mrb, backtrace, btline);
212 mrb_gc_arena_restore(mrb, ai);
213 }
214
215 return backtrace;
216}
217
[270]218MRB_API mrb_value
[331]219mrb_exc_backtrace(mrb_state *mrb, mrb_value exc)
[270]220{
[331]221 mrb_sym attr_name;
222 mrb_value backtrace;
[270]223
[331]224 attr_name = mrb_intern_lit(mrb, "backtrace");
225 backtrace = mrb_iv_get(mrb, exc, attr_name);
226 if (mrb_nil_p(backtrace) || mrb_array_p(backtrace)) {
227 return backtrace;
228 }
229 backtrace = mrb_unpack_backtrace(mrb, backtrace);
230 mrb_iv_set(mrb, exc, attr_name, backtrace);
231 return backtrace;
232}
[270]233
[331]234MRB_API mrb_value
235mrb_get_backtrace(mrb_state *mrb)
236{
237 return mrb_unpack_backtrace(mrb, packed_backtrace(mrb));
[270]238}
Note: See TracBrowser for help on using the repository browser.