needy_debugger 1.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (187) hide show
  1. data/.gitignore +14 -0
  2. data/.travis.yml +8 -0
  3. data/AUTHORS +10 -0
  4. data/CHANGELOG.md +68 -0
  5. data/CONTRIBUTING.md +1 -0
  6. data/Gemfile +3 -0
  7. data/LICENSE +23 -0
  8. data/OLDER_CHANGELOG +334 -0
  9. data/OLD_CHANGELOG +5655 -0
  10. data/OLD_README +122 -0
  11. data/README.md +141 -0
  12. data/Rakefile +78 -0
  13. data/bin/rdebug +397 -0
  14. data/doc/.cvsignore +42 -0
  15. data/doc/Makefile.am +63 -0
  16. data/doc/emacs-notes.txt +38 -0
  17. data/doc/hanoi.rb +35 -0
  18. data/doc/primes.rb +28 -0
  19. data/doc/rdebug-emacs.texi +1030 -0
  20. data/doc/ruby-debug.texi +3791 -0
  21. data/doc/test-tri2.rb +18 -0
  22. data/doc/tri3.rb +8 -0
  23. data/doc/triangle.rb +12 -0
  24. data/emacs/Makefile.am +130 -0
  25. data/emacs/rdebug-annotate.el +385 -0
  26. data/emacs/rdebug-breaks.el +407 -0
  27. data/emacs/rdebug-cmd.el +92 -0
  28. data/emacs/rdebug-core.el +502 -0
  29. data/emacs/rdebug-dbg.el +62 -0
  30. data/emacs/rdebug-error.el +79 -0
  31. data/emacs/rdebug-fns.el +111 -0
  32. data/emacs/rdebug-frames.el +230 -0
  33. data/emacs/rdebug-gud.el +242 -0
  34. data/emacs/rdebug-help.el +104 -0
  35. data/emacs/rdebug-info.el +83 -0
  36. data/emacs/rdebug-layouts.el +180 -0
  37. data/emacs/rdebug-locring.el +118 -0
  38. data/emacs/rdebug-output.el +106 -0
  39. data/emacs/rdebug-regexp.el +118 -0
  40. data/emacs/rdebug-secondary.el +260 -0
  41. data/emacs/rdebug-shortkey.el +175 -0
  42. data/emacs/rdebug-source.el +568 -0
  43. data/emacs/rdebug-track.el +392 -0
  44. data/emacs/rdebug-varbuf.el +150 -0
  45. data/emacs/rdebug-vars.el +125 -0
  46. data/emacs/rdebug-watch.el +132 -0
  47. data/emacs/rdebug.el +326 -0
  48. data/emacs/test/elk-test.el +242 -0
  49. data/emacs/test/test-annotate.el +103 -0
  50. data/emacs/test/test-cmd.el +116 -0
  51. data/emacs/test/test-core.el +104 -0
  52. data/emacs/test/test-fns.el +65 -0
  53. data/emacs/test/test-frames.el +62 -0
  54. data/emacs/test/test-gud.el +35 -0
  55. data/emacs/test/test-indent.el +58 -0
  56. data/emacs/test/test-regexp.el +144 -0
  57. data/emacs/test/test-shortkey.el +61 -0
  58. data/ext/ruby_debug/192/breakpoint.c +586 -0
  59. data/ext/ruby_debug/192/ruby_debug.c +2645 -0
  60. data/ext/ruby_debug/192/ruby_debug.h +148 -0
  61. data/ext/ruby_debug/193/breakpoint.c +586 -0
  62. data/ext/ruby_debug/193/ruby_debug.c +2626 -0
  63. data/ext/ruby_debug/193/ruby_debug.h +148 -0
  64. data/ext/ruby_debug/200/breakpoint.c +586 -0
  65. data/ext/ruby_debug/200/ruby_debug.c +2692 -0
  66. data/ext/ruby_debug/200/ruby_debug.h +148 -0
  67. data/ext/ruby_debug/extconf.rb +94 -0
  68. data/lib/debugger.rb +5 -0
  69. data/lib/debugger/version.rb +5 -0
  70. data/lib/ruby-debug-base.rb +305 -0
  71. data/lib/ruby-debug.rb +177 -0
  72. data/lib/ruby-debug/command.rb +227 -0
  73. data/lib/ruby-debug/commands/breakpoints.rb +153 -0
  74. data/lib/ruby-debug/commands/catchpoint.rb +55 -0
  75. data/lib/ruby-debug/commands/condition.rb +49 -0
  76. data/lib/ruby-debug/commands/continue.rb +38 -0
  77. data/lib/ruby-debug/commands/control.rb +107 -0
  78. data/lib/ruby-debug/commands/display.rb +120 -0
  79. data/lib/ruby-debug/commands/edit.rb +48 -0
  80. data/lib/ruby-debug/commands/enable.rb +202 -0
  81. data/lib/ruby-debug/commands/eval.rb +176 -0
  82. data/lib/ruby-debug/commands/finish.rb +42 -0
  83. data/lib/ruby-debug/commands/frame.rb +301 -0
  84. data/lib/ruby-debug/commands/help.rb +56 -0
  85. data/lib/ruby-debug/commands/info.rb +467 -0
  86. data/lib/ruby-debug/commands/irb.rb +123 -0
  87. data/lib/ruby-debug/commands/jump.rb +66 -0
  88. data/lib/ruby-debug/commands/kill.rb +51 -0
  89. data/lib/ruby-debug/commands/list.rb +94 -0
  90. data/lib/ruby-debug/commands/method.rb +84 -0
  91. data/lib/ruby-debug/commands/quit.rb +50 -0
  92. data/lib/ruby-debug/commands/reload.rb +40 -0
  93. data/lib/ruby-debug/commands/save.rb +90 -0
  94. data/lib/ruby-debug/commands/set.rb +223 -0
  95. data/lib/ruby-debug/commands/show.rb +247 -0
  96. data/lib/ruby-debug/commands/skip.rb +35 -0
  97. data/lib/ruby-debug/commands/source.rb +36 -0
  98. data/lib/ruby-debug/commands/stepping.rb +81 -0
  99. data/lib/ruby-debug/commands/threads.rb +189 -0
  100. data/lib/ruby-debug/commands/tmate.rb +36 -0
  101. data/lib/ruby-debug/commands/trace.rb +57 -0
  102. data/lib/ruby-debug/commands/variables.rb +199 -0
  103. data/lib/ruby-debug/debugger.rb +5 -0
  104. data/lib/ruby-debug/helper.rb +69 -0
  105. data/lib/ruby-debug/interface.rb +232 -0
  106. data/lib/ruby-debug/processor.rb +474 -0
  107. data/man/rdebug.1 +241 -0
  108. data/needy_debugger.gemspec +31 -0
  109. data/old_scripts/Makefile.am +14 -0
  110. data/old_scripts/README.md +2 -0
  111. data/old_scripts/autogen.sh +4 -0
  112. data/old_scripts/configure.ac +12 -0
  113. data/old_scripts/rdbg.rb +33 -0
  114. data/old_scripts/runner.sh +7 -0
  115. data/old_scripts/svn2cl_usermap +3 -0
  116. data/test/.cvsignore +1 -0
  117. data/test/breakpoints_test.rb +365 -0
  118. data/test/conditions_test.rb +76 -0
  119. data/test/continue_test.rb +28 -0
  120. data/test/display_test.rb +141 -0
  121. data/test/edit_test.rb +55 -0
  122. data/test/eval_test.rb +92 -0
  123. data/test/examples/breakpoint1.rb +15 -0
  124. data/test/examples/breakpoint2.rb +7 -0
  125. data/test/examples/conditions.rb +4 -0
  126. data/test/examples/continue.rb +4 -0
  127. data/test/examples/display.rb +5 -0
  128. data/test/examples/edit.rb +3 -0
  129. data/test/examples/edit2.rb +3 -0
  130. data/test/examples/eval.rb +4 -0
  131. data/test/examples/finish.rb +20 -0
  132. data/test/examples/frame.rb +31 -0
  133. data/test/examples/help.rb +2 -0
  134. data/test/examples/info.rb +48 -0
  135. data/test/examples/info2.rb +3 -0
  136. data/test/examples/irb.rb +6 -0
  137. data/test/examples/jump.rb +14 -0
  138. data/test/examples/kill.rb +2 -0
  139. data/test/examples/list.rb +12 -0
  140. data/test/examples/method.rb +15 -0
  141. data/test/examples/post_mortem.rb +19 -0
  142. data/test/examples/quit.rb +2 -0
  143. data/test/examples/reload.rb +6 -0
  144. data/test/examples/restart.rb +6 -0
  145. data/test/examples/save.rb +3 -0
  146. data/test/examples/set.rb +3 -0
  147. data/test/examples/set_annotate.rb +12 -0
  148. data/test/examples/settings.rb +1 -0
  149. data/test/examples/show.rb +2 -0
  150. data/test/examples/source.rb +3 -0
  151. data/test/examples/stepping.rb +21 -0
  152. data/test/examples/thread.rb +32 -0
  153. data/test/examples/tmate.rb +10 -0
  154. data/test/examples/trace.rb +7 -0
  155. data/test/examples/trace_threads.rb +20 -0
  156. data/test/examples/variables.rb +26 -0
  157. data/test/finish_test.rb +48 -0
  158. data/test/frame_test.rb +140 -0
  159. data/test/help_test.rb +50 -0
  160. data/test/info_test.rb +325 -0
  161. data/test/irb_test.rb +81 -0
  162. data/test/jump_test.rb +70 -0
  163. data/test/kill_test.rb +47 -0
  164. data/test/list_test.rb +145 -0
  165. data/test/method_test.rb +70 -0
  166. data/test/post_mortem_test.rb +25 -0
  167. data/test/quit_test.rb +55 -0
  168. data/test/reload_test.rb +43 -0
  169. data/test/restart_test.rb +143 -0
  170. data/test/save_test.rb +92 -0
  171. data/test/set_test.rb +177 -0
  172. data/test/show_test.rb +292 -0
  173. data/test/source_test.rb +44 -0
  174. data/test/stepping_test.rb +118 -0
  175. data/test/support/breakpoint.rb +12 -0
  176. data/test/support/context.rb +14 -0
  177. data/test/support/matchers.rb +67 -0
  178. data/test/support/mocha_extensions.rb +71 -0
  179. data/test/support/processor.rb +7 -0
  180. data/test/support/test_dsl.rb +205 -0
  181. data/test/support/test_interface.rb +66 -0
  182. data/test/test_helper.rb +8 -0
  183. data/test/thread_test.rb +122 -0
  184. data/test/tmate_test.rb +43 -0
  185. data/test/trace_test.rb +154 -0
  186. data/test/variables_test.rb +114 -0
  187. metadata +352 -0
@@ -0,0 +1,148 @@
1
+ /* Context info */
2
+ enum ctx_stop_reason {CTX_STOP_NONE, CTX_STOP_STEP, CTX_STOP_BREAKPOINT,
3
+ CTX_STOP_CATCHPOINT};
4
+
5
+ /* Context flags */
6
+ #define CTX_FL_SUSPEND (1<<1)
7
+ #define CTX_FL_TRACING (1<<2)
8
+ #define CTX_FL_SKIPPED (1<<3)
9
+ #define CTX_FL_IGNORE (1<<4)
10
+ #define CTX_FL_DEAD (1<<5)
11
+ #define CTX_FL_WAS_RUNNING (1<<6)
12
+ #define CTX_FL_ENABLE_BKPT (1<<7)
13
+ #define CTX_FL_STEPPED (1<<8)
14
+ #define CTX_FL_FORCE_MOVE (1<<9)
15
+ #define CTX_FL_CATCHING (1<<10)
16
+
17
+ #define CTX_FL_TEST(c,f) ((c)->flags & (f))
18
+ #define CTX_FL_SET(c,f) do { (c)->flags |= (f); } while (0)
19
+ #define CTX_FL_UNSET(c,f) do { (c)->flags &= ~(f); } while (0)
20
+
21
+ typedef struct {
22
+ struct iseq_catch_table_entry tmp_catch_table;
23
+ struct iseq_catch_table_entry *old_catch_table;
24
+ int old_catch_table_size;
25
+ VALUE mod_name;
26
+ VALUE errinfo;
27
+ } debug_catch_t;
28
+
29
+ typedef struct {
30
+ struct rb_iseq_struct *iseq;
31
+ struct iseq_catch_table_entry *catch_table;
32
+ int catch_table_size;
33
+ } iseq_catch_t;
34
+
35
+ typedef struct {
36
+ int argc; /* Number of arguments a frame should have. */
37
+ VALUE binding;
38
+ ID id;
39
+ ID orig_id;
40
+ int line;
41
+ const char * file;
42
+ short dead;
43
+ VALUE self;
44
+ VALUE arg_ary;
45
+ union {
46
+ struct {
47
+ rb_control_frame_t *cfp;
48
+ VALUE *bp;
49
+ struct rb_iseq_struct *block_iseq;
50
+ VALUE *block_pc;
51
+ VALUE *last_pc;
52
+ } runtime;
53
+ struct {
54
+ VALUE args;
55
+ VALUE locals;
56
+ VALUE arg_ary;
57
+ } copy;
58
+ } info;
59
+ } debug_frame_t;
60
+
61
+ typedef struct {
62
+ VALUE thread_id;
63
+ int thnum;
64
+ int flags;
65
+ enum ctx_stop_reason stop_reason;
66
+ int stop_next;
67
+ int dest_frame;
68
+ int stop_line;
69
+ int stop_frame;
70
+ int stack_size;
71
+ int stack_len;
72
+ debug_frame_t *frames;
73
+ const char * last_file;
74
+ int last_line;
75
+ VALUE breakpoint;
76
+ debug_catch_t catch_table;
77
+ VALUE saved_jump_ins[2];
78
+ rb_control_frame_t *jump_cfp;
79
+ VALUE *jump_pc;
80
+ iseq_catch_t *old_iseq_catch;
81
+ volatile int thread_pause;
82
+ } debug_context_t;
83
+
84
+ /* variables in ruby_debug.c */
85
+ extern VALUE mDebugger;
86
+ extern VALUE rdebug_breakpoints;
87
+ extern VALUE rdebug_catchpoints;
88
+ extern VALUE rdebug_threads_tbl;
89
+
90
+ /* routines in ruby_debug.c */
91
+ extern int filename_cmp(VALUE source, char *file);
92
+
93
+ #define IS_STARTED (rdebug_threads_tbl != Qnil)
94
+ static inline void
95
+ debug_check_started()
96
+ {
97
+ if(!IS_STARTED)
98
+ {
99
+ rb_raise(rb_eRuntimeError, "Debugger.start is not called yet.");
100
+ }
101
+ }
102
+
103
+ static inline int
104
+ classname_cmp(VALUE name, VALUE klass)
105
+ {
106
+ VALUE mod_name;
107
+ VALUE class_name = (Qnil == name) ? rb_str_new2("main") : name;
108
+ if (klass == Qnil) return(0);
109
+ mod_name = rb_mod_name(klass);
110
+ return (mod_name != Qnil
111
+ && rb_str_cmp(class_name, mod_name) == 0);
112
+ }
113
+
114
+ /* Breakpoint information */
115
+ enum bp_type {BP_POS_TYPE, BP_METHOD_TYPE};
116
+ enum hit_condition {HIT_COND_NONE, HIT_COND_GE, HIT_COND_EQ, HIT_COND_MOD};
117
+
118
+ typedef struct {
119
+ int id;
120
+ enum bp_type type;
121
+ VALUE source;
122
+ union
123
+ {
124
+ int line;
125
+ ID mid;
126
+ } pos;
127
+ VALUE expr;
128
+ VALUE enabled;
129
+ int hit_count;
130
+ int hit_value;
131
+ enum hit_condition hit_condition;
132
+ } debug_breakpoint_t;
133
+
134
+ /* routines in breakpoint.c */
135
+ extern int check_breakpoint_expression(VALUE breakpoint, VALUE binding);
136
+ extern int check_breakpoint_hit_condition(VALUE breakpoint);
137
+ extern VALUE check_breakpoints_by_method(debug_context_t *debug_context,
138
+ VALUE klass, ID mid, VALUE self);
139
+ extern VALUE check_breakpoints_by_pos(debug_context_t *debug_context,
140
+ char *file, int line);
141
+ extern VALUE create_breakpoint_from_args(int argc, VALUE *argv, int id);
142
+ extern VALUE context_breakpoint(VALUE self);
143
+ extern VALUE context_set_breakpoint(int argc, VALUE *argv, VALUE self);
144
+ extern VALUE rdebug_add_catchpoint(VALUE self, VALUE value);
145
+ extern VALUE debug_catchpoints(VALUE self);
146
+ extern VALUE rdebug_remove_breakpoint(VALUE self, VALUE id_value);
147
+
148
+ extern void Init_breakpoint();
@@ -0,0 +1,586 @@
1
+ #include <ruby.h>
2
+ #include <stdio.h>
3
+ #include <vm_core.h>
4
+ #include <iseq.h>
5
+ #include "ruby_debug.h"
6
+
7
+ VALUE rdebug_breakpoints = Qnil;
8
+ VALUE rdebug_catchpoints;
9
+
10
+ static VALUE cBreakpoint;
11
+ static ID idEval;
12
+
13
+ static VALUE
14
+ eval_expression(VALUE args)
15
+ {
16
+ return rb_funcall2(rb_mKernel, idEval, 2, RARRAY_PTR(args));
17
+ }
18
+
19
+ int
20
+ check_breakpoint_hit_condition(VALUE breakpoint)
21
+ {
22
+ debug_breakpoint_t *debug_breakpoint;
23
+
24
+ if(breakpoint == Qnil)
25
+ return 0;
26
+ Data_Get_Struct(breakpoint, debug_breakpoint_t, debug_breakpoint);
27
+
28
+ debug_breakpoint->hit_count++;
29
+ if (!Qtrue == debug_breakpoint->enabled) return 0;
30
+ switch(debug_breakpoint->hit_condition)
31
+ {
32
+ case HIT_COND_NONE:
33
+ return 1;
34
+ case HIT_COND_GE:
35
+ {
36
+ if(debug_breakpoint->hit_count >= debug_breakpoint->hit_value)
37
+ return 1;
38
+ break;
39
+ }
40
+ case HIT_COND_EQ:
41
+ {
42
+ if(debug_breakpoint->hit_count == debug_breakpoint->hit_value)
43
+ return 1;
44
+ break;
45
+ }
46
+ case HIT_COND_MOD:
47
+ {
48
+ if(debug_breakpoint->hit_count % debug_breakpoint->hit_value == 0)
49
+ return 1;
50
+ break;
51
+ }
52
+ }
53
+ return 0;
54
+ }
55
+
56
+ static int
57
+ check_breakpoint_by_pos(VALUE breakpoint, char *file, int line)
58
+ {
59
+ debug_breakpoint_t *debug_breakpoint;
60
+
61
+ if(breakpoint == Qnil)
62
+ return 0;
63
+ Data_Get_Struct(breakpoint, debug_breakpoint_t, debug_breakpoint);
64
+ if (!Qtrue == debug_breakpoint->enabled) return 0;
65
+ if(debug_breakpoint->type != BP_POS_TYPE)
66
+ return 0;
67
+ if(debug_breakpoint->pos.line != line)
68
+ return 0;
69
+ if(filename_cmp(debug_breakpoint->source, file))
70
+ return 1;
71
+ return 0;
72
+ }
73
+
74
+ int
75
+ check_breakpoint_by_method(VALUE breakpoint, VALUE klass, ID mid, VALUE self)
76
+ {
77
+ debug_breakpoint_t *debug_breakpoint;
78
+
79
+ if(breakpoint == Qnil)
80
+ return 0;
81
+ Data_Get_Struct(breakpoint, debug_breakpoint_t, debug_breakpoint);
82
+ if (!Qtrue == debug_breakpoint->enabled) return 0;
83
+ if(debug_breakpoint->type != BP_METHOD_TYPE)
84
+ return 0;
85
+ if(debug_breakpoint->pos.mid != mid)
86
+ return 0;
87
+ if(classname_cmp(debug_breakpoint->source, klass))
88
+ return 1;
89
+ if ((rb_type(self) == T_CLASS) &&
90
+ classname_cmp(debug_breakpoint->source, self))
91
+ return 1;
92
+ return 0;
93
+ }
94
+
95
+ VALUE
96
+ check_breakpoints_by_pos(debug_context_t *debug_context, char *file, int line)
97
+ {
98
+ VALUE breakpoint;
99
+ int i;
100
+
101
+ if(!CTX_FL_TEST(debug_context, CTX_FL_ENABLE_BKPT))
102
+ return Qnil;
103
+
104
+ if(check_breakpoint_by_pos(debug_context->breakpoint, file, line))
105
+ return debug_context->breakpoint;
106
+
107
+ if(RARRAY_LEN(rdebug_breakpoints) == 0)
108
+ return Qnil;
109
+ for(i = 0; i < RARRAY_LEN(rdebug_breakpoints); i++)
110
+ {
111
+ breakpoint = rb_ary_entry(rdebug_breakpoints, i);
112
+ if(check_breakpoint_by_pos(breakpoint, file, line))
113
+ return breakpoint;
114
+ }
115
+ return Qnil;
116
+ }
117
+
118
+ VALUE
119
+ check_breakpoints_by_method(debug_context_t *debug_context, VALUE klass, ID mid, VALUE self)
120
+ {
121
+ VALUE breakpoint;
122
+ int i;
123
+
124
+ if(!CTX_FL_TEST(debug_context, CTX_FL_ENABLE_BKPT))
125
+ return Qnil;
126
+
127
+ if(check_breakpoint_by_method(debug_context->breakpoint, klass, mid, self))
128
+ return debug_context->breakpoint;
129
+
130
+ if(RARRAY_LEN(rdebug_breakpoints) == 0)
131
+ return Qnil;
132
+ for(i = 0; i < RARRAY_LEN(rdebug_breakpoints); i++)
133
+ {
134
+ breakpoint = rb_ary_entry(rdebug_breakpoints, i);
135
+ if(check_breakpoint_by_method(breakpoint, klass, mid, self))
136
+ return breakpoint;
137
+ }
138
+ return Qnil;
139
+ }
140
+
141
+ int
142
+ check_breakpoint_expression(VALUE breakpoint, VALUE binding)
143
+ {
144
+ debug_breakpoint_t *debug_breakpoint;
145
+ VALUE args, expr_result;
146
+
147
+ Data_Get_Struct(breakpoint, debug_breakpoint_t, debug_breakpoint);
148
+ if(NIL_P(debug_breakpoint->expr))
149
+ return 1;
150
+
151
+ args = rb_ary_new3(2, debug_breakpoint->expr, binding);
152
+ expr_result = rb_protect(eval_expression, args, 0);
153
+ return RTEST(expr_result);
154
+ }
155
+
156
+ static void
157
+ breakpoint_mark(void *data)
158
+ {
159
+ debug_breakpoint_t *breakpoint;
160
+ breakpoint = (debug_breakpoint_t *)data;
161
+ rb_gc_mark(breakpoint->source);
162
+ rb_gc_mark(breakpoint->expr);
163
+ }
164
+
165
+ VALUE
166
+ create_breakpoint_from_args(int argc, VALUE *argv, int id)
167
+ {
168
+ VALUE source, pos, expr;
169
+ debug_breakpoint_t *breakpoint;
170
+ int type;
171
+
172
+ if(rb_scan_args(argc, argv, "21", &source, &pos, &expr) == 2)
173
+ {
174
+ expr = Qnil;
175
+ }
176
+ type = FIXNUM_P(pos) ? BP_POS_TYPE : BP_METHOD_TYPE;
177
+ if(type == BP_POS_TYPE)
178
+ source = StringValue(source);
179
+ else
180
+ pos = StringValue(pos);
181
+ breakpoint = ALLOC(debug_breakpoint_t);
182
+ breakpoint->id = id;
183
+ breakpoint->source = source;
184
+ breakpoint->type = type;
185
+ if(type == BP_POS_TYPE)
186
+ breakpoint->pos.line = FIX2INT(pos);
187
+ else
188
+ breakpoint->pos.mid = rb_intern(RSTRING_PTR(pos));
189
+ breakpoint->enabled = Qtrue;
190
+ breakpoint->expr = NIL_P(expr) ? expr: StringValue(expr);
191
+ breakpoint->hit_count = 0;
192
+ breakpoint->hit_value = 0;
193
+ breakpoint->hit_condition = HIT_COND_NONE;
194
+ return Data_Wrap_Struct(cBreakpoint, breakpoint_mark, xfree, breakpoint);
195
+ }
196
+
197
+ /*
198
+ * call-seq:
199
+ * Debugger.remove_breakpoint(id) -> breakpoint
200
+ *
201
+ * Removes breakpoint by its id.
202
+ * <i>id</i> is an identificator of a breakpoint.
203
+ */
204
+ VALUE
205
+ rdebug_remove_breakpoint(VALUE self, VALUE id_value)
206
+ {
207
+ int i;
208
+ int id;
209
+ VALUE breakpoint;
210
+ debug_breakpoint_t *debug_breakpoint;
211
+
212
+ id = FIX2INT(id_value);
213
+
214
+ for( i = 0; i < RARRAY_LEN(rdebug_breakpoints); i += 1 )
215
+ {
216
+ breakpoint = rb_ary_entry(rdebug_breakpoints, i);
217
+ Data_Get_Struct(breakpoint, debug_breakpoint_t, debug_breakpoint);
218
+ if(debug_breakpoint->id == id)
219
+ {
220
+ rb_ary_delete_at(rdebug_breakpoints, i);
221
+ return breakpoint;
222
+ }
223
+ }
224
+ return Qnil;
225
+ }
226
+
227
+ /*
228
+ * call-seq:
229
+ * Debugger.catchpoints -> hash
230
+ *
231
+ * Returns a current catchpoints, which is a hash exception names that will
232
+ * trigger a debugger when raised. The values are the number of times taht
233
+ * catchpoint was hit, initially 0.
234
+ */
235
+ VALUE
236
+ debug_catchpoints(VALUE self)
237
+ {
238
+ debug_check_started();
239
+
240
+ return rdebug_catchpoints;
241
+ }
242
+
243
+ /*
244
+ * call-seq:
245
+ * Debugger.catchpoint(string) -> string
246
+ *
247
+ * Sets catchpoint. Returns the string passed.
248
+ */
249
+ VALUE
250
+ rdebug_add_catchpoint(VALUE self, VALUE value)
251
+ {
252
+ debug_check_started();
253
+
254
+ if (TYPE(value) != T_STRING) {
255
+ rb_raise(rb_eTypeError, "value of a catchpoint must be String");
256
+ }
257
+ rb_hash_aset(rdebug_catchpoints, rb_str_dup(value), INT2FIX(0));
258
+ return value;
259
+ }
260
+
261
+ /*
262
+ * call-seq:
263
+ * context.breakpoint -> breakpoint
264
+ *
265
+ * Returns a context-specific temporary Breakpoint object.
266
+ */
267
+ VALUE
268
+ context_breakpoint(VALUE self)
269
+ {
270
+ debug_context_t *debug_context;
271
+
272
+ debug_check_started();
273
+
274
+ Data_Get_Struct(self, debug_context_t, debug_context);
275
+ return debug_context->breakpoint;
276
+ }
277
+
278
+ /*
279
+ * call-seq:
280
+ * context.set_breakpoint(source, pos, condition = nil) -> breakpoint
281
+ *
282
+ * Sets a context-specific temporary breakpoint, which can be used to implement
283
+ * 'Run to Cursor' debugger function. When this breakpoint is reached, it will be
284
+ * cleared out.
285
+ *
286
+ * <i>source</i> is a name of a file or a class.
287
+ * <i>pos</i> is a line number or a method name if <i>source</i> is a class name.
288
+ * <i>condition</i> is a string which is evaluated to +true+ when this breakpoint
289
+ * is activated.
290
+ */
291
+ VALUE
292
+ context_set_breakpoint(int argc, VALUE *argv, VALUE self)
293
+ {
294
+ VALUE result;
295
+ debug_context_t *debug_context;
296
+
297
+ debug_check_started();
298
+
299
+ Data_Get_Struct(self, debug_context_t, debug_context);
300
+ result = create_breakpoint_from_args(argc, argv, 0);
301
+ debug_context->breakpoint = result;
302
+ return result;
303
+ }
304
+
305
+ /*
306
+ * call-seq:
307
+ * breakpoint.enabled?
308
+ *
309
+ * Returns whether breakpoint is enabled or not.
310
+ */
311
+ static VALUE
312
+ breakpoint_enabled(VALUE self)
313
+ {
314
+ debug_breakpoint_t *breakpoint;
315
+
316
+ Data_Get_Struct(self, debug_breakpoint_t, breakpoint);
317
+ return breakpoint->enabled;
318
+ }
319
+
320
+ /*
321
+ * call-seq:
322
+ * breakpoint.enabled = bool
323
+ *
324
+ * Enables or disables breakpoint.
325
+ */
326
+ static VALUE
327
+ breakpoint_set_enabled(VALUE self, VALUE bool)
328
+ {
329
+ debug_breakpoint_t *breakpoint;
330
+
331
+ Data_Get_Struct(self, debug_breakpoint_t, breakpoint);
332
+ return breakpoint->enabled = bool;
333
+ }
334
+
335
+ /*
336
+ * call-seq:
337
+ * breakpoint.source -> string
338
+ *
339
+ * Returns a source of the breakpoint.
340
+ */
341
+ static VALUE
342
+ breakpoint_source(VALUE self)
343
+ {
344
+ debug_breakpoint_t *breakpoint;
345
+
346
+ Data_Get_Struct(self, debug_breakpoint_t, breakpoint);
347
+ return breakpoint->source;
348
+ }
349
+
350
+ /*
351
+ * call-seq:
352
+ * breakpoint.source = string
353
+ *
354
+ * Sets the source of the breakpoint.
355
+ */
356
+ static VALUE
357
+ breakpoint_set_source(VALUE self, VALUE value)
358
+ {
359
+ debug_breakpoint_t *breakpoint;
360
+
361
+ Data_Get_Struct(self, debug_breakpoint_t, breakpoint);
362
+ breakpoint->source = StringValue(value);
363
+ return value;
364
+ }
365
+
366
+ /*
367
+ * call-seq:
368
+ * breakpoint.pos -> string or int
369
+ *
370
+ * Returns the position of this breakpoint.
371
+ */
372
+ static VALUE
373
+ breakpoint_pos(VALUE self)
374
+ {
375
+ debug_breakpoint_t *breakpoint;
376
+
377
+ Data_Get_Struct(self, debug_breakpoint_t, breakpoint);
378
+ if(breakpoint->type == BP_METHOD_TYPE)
379
+ return rb_str_new2(rb_id2name(breakpoint->pos.mid));
380
+ else
381
+ return INT2FIX(breakpoint->pos.line);
382
+ }
383
+
384
+ /*
385
+ * call-seq:
386
+ * breakpoint.pos = string or int
387
+ *
388
+ * Sets the position of this breakpoint.
389
+ */
390
+ static VALUE
391
+ breakpoint_set_pos(VALUE self, VALUE value)
392
+ {
393
+ debug_breakpoint_t *breakpoint;
394
+
395
+ Data_Get_Struct(self, debug_breakpoint_t, breakpoint);
396
+ if(breakpoint->type == BP_METHOD_TYPE)
397
+ {
398
+ breakpoint->pos.mid = rb_to_id(StringValue(value));
399
+ }
400
+ else
401
+ breakpoint->pos.line = FIX2INT(value);
402
+ return value;
403
+ }
404
+
405
+ /*
406
+ * call-seq:
407
+ * breakpoint.expr -> string
408
+ *
409
+ * Returns a codition expression when this breakpoint should be activated.
410
+ */
411
+ static VALUE
412
+ breakpoint_expr(VALUE self)
413
+ {
414
+ debug_breakpoint_t *breakpoint;
415
+
416
+ Data_Get_Struct(self, debug_breakpoint_t, breakpoint);
417
+ return breakpoint->expr;
418
+ }
419
+
420
+ /*
421
+ * call-seq:
422
+ * breakpoint.expr = string | nil
423
+ *
424
+ * Sets the codition expression when this breakpoint should be activated.
425
+ */
426
+ static VALUE
427
+ breakpoint_set_expr(VALUE self, VALUE expr)
428
+ {
429
+ debug_breakpoint_t *breakpoint;
430
+
431
+ Data_Get_Struct(self, debug_breakpoint_t, breakpoint);
432
+ breakpoint->expr = NIL_P(expr) ? expr: StringValue(expr);
433
+ return expr;
434
+ }
435
+
436
+ /*
437
+ * call-seq:
438
+ * breakpoint.id -> int
439
+ *
440
+ * Returns id of the breakpoint.
441
+ */
442
+ static VALUE
443
+ breakpoint_id(VALUE self)
444
+ {
445
+ debug_breakpoint_t *breakpoint;
446
+
447
+ Data_Get_Struct(self, debug_breakpoint_t, breakpoint);
448
+ return INT2FIX(breakpoint->id);
449
+ }
450
+
451
+ /*
452
+ * call-seq:
453
+ * breakpoint.hit_count -> int
454
+ *
455
+ * Returns the hit count of the breakpoint.
456
+ */
457
+ static VALUE
458
+ breakpoint_hit_count(VALUE self)
459
+ {
460
+ debug_breakpoint_t *breakpoint;
461
+
462
+ Data_Get_Struct(self, debug_breakpoint_t, breakpoint);
463
+ return INT2FIX(breakpoint->hit_count);
464
+ }
465
+
466
+ /*
467
+ * call-seq:
468
+ * breakpoint.hit_value -> int
469
+ *
470
+ * Returns the hit value of the breakpoint.
471
+ */
472
+ static VALUE
473
+ breakpoint_hit_value(VALUE self)
474
+ {
475
+ debug_breakpoint_t *breakpoint;
476
+
477
+ Data_Get_Struct(self, debug_breakpoint_t, breakpoint);
478
+ return INT2FIX(breakpoint->hit_value);
479
+ }
480
+
481
+ /*
482
+ * call-seq:
483
+ * breakpoint.hit_value = int
484
+ *
485
+ * Sets the hit value of the breakpoint.
486
+ */
487
+ static VALUE
488
+ breakpoint_set_hit_value(VALUE self, VALUE value)
489
+ {
490
+ debug_breakpoint_t *breakpoint;
491
+
492
+ Data_Get_Struct(self, debug_breakpoint_t, breakpoint);
493
+ breakpoint->hit_value = FIX2INT(value);
494
+ return value;
495
+ }
496
+
497
+ /*
498
+ * call-seq:
499
+ * breakpoint.hit_condition -> symbol
500
+ *
501
+ * Returns the hit condition of the breakpoint:
502
+ *
503
+ * +nil+ if it is an unconditional breakpoint, or
504
+ * :greater_or_equal, :equal, :modulo
505
+ */
506
+ static VALUE
507
+ breakpoint_hit_condition(VALUE self)
508
+ {
509
+ debug_breakpoint_t *breakpoint;
510
+
511
+ Data_Get_Struct(self, debug_breakpoint_t, breakpoint);
512
+ switch(breakpoint->hit_condition)
513
+ {
514
+ case HIT_COND_GE:
515
+ return ID2SYM(rb_intern("greater_or_equal"));
516
+ case HIT_COND_EQ:
517
+ return ID2SYM(rb_intern("equal"));
518
+ case HIT_COND_MOD:
519
+ return ID2SYM(rb_intern("modulo"));
520
+ case HIT_COND_NONE:
521
+ default:
522
+ return Qnil;
523
+ }
524
+ }
525
+
526
+ /*
527
+ * call-seq:
528
+ * breakpoint.hit_condition = symbol
529
+ *
530
+ * Sets the hit condition of the breakpoint which must be one of the following values:
531
+ *
532
+ * +nil+ if it is an unconditional breakpoint, or
533
+ * :greater_or_equal(:ge), :equal(:eq), :modulo(:mod)
534
+ */
535
+ static VALUE
536
+ breakpoint_set_hit_condition(VALUE self, VALUE value)
537
+ {
538
+ debug_breakpoint_t *breakpoint;
539
+ ID id_value;
540
+
541
+ Data_Get_Struct(self, debug_breakpoint_t, breakpoint);
542
+ id_value = rb_to_id(value);
543
+
544
+ if(rb_intern("greater_or_equal") == id_value || rb_intern("ge") == id_value)
545
+ breakpoint->hit_condition = HIT_COND_GE;
546
+ else if(rb_intern("equal") == id_value || rb_intern("eq") == id_value)
547
+ breakpoint->hit_condition = HIT_COND_EQ;
548
+ else if(rb_intern("modulo") == id_value || rb_intern("mod") == id_value)
549
+ breakpoint->hit_condition = HIT_COND_MOD;
550
+ else
551
+ rb_raise(rb_eArgError, "Invalid condition parameter");
552
+ return value;
553
+ }
554
+
555
+ /*
556
+ * Document-class: Breakpoint
557
+ *
558
+ * == Summary
559
+ *
560
+ * This class represents a breakpoint. It defines position of the breakpoint and
561
+ * condition when this breakpoint should be triggered.
562
+ */
563
+ void
564
+ Init_breakpoint()
565
+ {
566
+ cBreakpoint = rb_define_class_under(mDebugger, "Breakpoint", rb_cObject);
567
+ rb_define_method(cBreakpoint, "enabled=", breakpoint_set_enabled, 1);
568
+ rb_define_method(cBreakpoint, "enabled?", breakpoint_enabled, 0);
569
+ rb_define_method(cBreakpoint, "expr", breakpoint_expr, 0);
570
+ rb_define_method(cBreakpoint, "expr=", breakpoint_set_expr, 1);
571
+ rb_define_method(cBreakpoint, "hit_condition", breakpoint_hit_condition, 0);
572
+ rb_define_method(cBreakpoint, "hit_condition=", breakpoint_set_hit_condition, 1);
573
+ rb_define_method(cBreakpoint, "hit_count", breakpoint_hit_count, 0);
574
+ rb_define_method(cBreakpoint, "hit_value", breakpoint_hit_value, 0);
575
+ rb_define_method(cBreakpoint, "hit_value=", breakpoint_set_hit_value, 1);
576
+ rb_define_method(cBreakpoint, "id", breakpoint_id, 0);
577
+ rb_define_method(cBreakpoint, "pos", breakpoint_pos, 0);
578
+ rb_define_method(cBreakpoint, "pos=", breakpoint_set_pos, 1);
579
+ rb_define_method(cBreakpoint, "source", breakpoint_source, 0);
580
+ rb_define_method(cBreakpoint, "source=", breakpoint_set_source, 1);
581
+ idEval = rb_intern("eval");
582
+ rdebug_catchpoints = rb_hash_new();
583
+
584
+ }
585
+
586
+