#ifndef BYEBUG #define BYEBUG #include #include /* flags */ #define CTX_FL_DEAD (1<<1) /* this context belonged to a dead thread */ #define CTX_FL_ENABLE_BKPT (1<<2) /* can check for breakpoints */ #define CTX_FL_FORCE_MOVE (1<<3) /* don't stop unless we've changed line */ #define CTX_FL_IGNORE (1<<4) /* this context belongs to ignored thread */ #define CTX_FL_SUSPEND (1<<5) /* thread currently suspended */ #define CTX_FL_TRACING (1<<6) /* call at_tracing method */ #define CTX_FL_WAS_RUNNING (1<<7) /* thread was previously running */ #define CTX_FL_STOP_ON_RET (1<<8) /* can stop on method 'end' */ /* macro functions */ #define CTX_FL_TEST(c,f) ((c)->flags & (f)) #define CTX_FL_SET(c,f) do { (c)->flags |= (f); } while (0) #define CTX_FL_UNSET(c,f) do { (c)->flags &= ~(f); } while (0) /* types */ typedef enum { CTX_STOP_NONE, CTX_STOP_STEP, CTX_STOP_BREAKPOINT, CTX_STOP_CATCHPOINT } ctx_stop_reason; typedef struct { int calced_stack_size; int flags; ctx_stop_reason stop_reason; VALUE thread; int thnum; int dest_frame; /* next stop's frame if stopped by next */ int lines; /* # of lines in dest_frame before stopping */ int steps; /* # of steps before stopping */ int steps_out; /* # of returns before stopping */ VALUE last_file; VALUE last_line; VALUE backtrace; /* [[loc, self, klass, binding], ...] */ } debug_context_t; enum frame_component { LOCATION, SELF, CLASS, BINDING }; struct call_with_inspection_data { debug_context_t *dc; VALUE context_obj; ID id; int argc; VALUE *argv; }; typedef struct { st_table *tbl; } threads_table_t; /* functions from locker.c */ extern int is_in_locked(VALUE thread_id); extern void add_to_locked(VALUE thread); extern VALUE remove_from_locked(); /* functions from threads.c */ extern void Init_threads_table(VALUE mByebug); extern VALUE create_threads_table(void); extern void check_threads_table(void); extern void thread_context_lookup(VALUE thread, VALUE *context); extern void halt_while_other_thread_is_active(debug_context_t *dc); /* global variables */ extern VALUE locker; extern VALUE threads; /* functions */ extern void Init_context(VALUE mByebug); extern VALUE context_create(VALUE thread); extern VALUE context_dup(debug_context_t *context); extern void reset_stepping_stop_points(debug_context_t *context); extern VALUE call_with_debug_inspector(struct call_with_inspection_data *data); extern VALUE context_backtrace_set(const rb_debug_inspector_t *inspector, void *data); static inline int classname_cmp(VALUE name, VALUE klass) { VALUE mod_name; VALUE class_name = (Qnil == name) ? rb_str_new2("main") : name; if (klass == Qnil) return(0); mod_name = rb_mod_name(klass); return (mod_name != Qnil && rb_str_cmp(class_name, mod_name) == 0); } /* breakpoints & catchpoints */ enum bp_type { BP_POS_TYPE, BP_METHOD_TYPE }; enum hit_condition { HIT_COND_NONE, HIT_COND_GE, HIT_COND_EQ, HIT_COND_MOD }; typedef struct { int id; enum bp_type type; VALUE source; union { int line; ID mid; } pos; VALUE expr; VALUE enabled; int hit_count; int hit_value; enum hit_condition hit_condition; } breakpoint_t; extern void Init_breakpoint(VALUE mByebug); extern VALUE catchpoint_hit_count(VALUE catchpoints, VALUE exception, VALUE *exception_name); extern VALUE find_breakpoint_by_pos(VALUE breakpoints, VALUE source, VALUE pos, VALUE binding); extern VALUE find_breakpoint_by_method(VALUE breakpoints, VALUE klass, VALUE mid, VALUE binding, VALUE self); #endif