ruby-debug 0.1.5-mswin32
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.
- data/CHANGES +16 -0
- data/LICENSE +23 -0
- data/README +63 -0
- data/Rakefile +102 -0
- data/bin/rdebug +13 -0
- data/ext/extconf.rb +18 -0
- data/ext/ruby_debug.c +991 -0
- data/lib/ruby-debug.rb +633 -0
- data/lib/ruby_debug.so +0 -0
- metadata +55 -0
data/lib/ruby-debug.rb
ADDED
@@ -0,0 +1,633 @@
|
|
1
|
+
require 'ruby_debug.so'
|
2
|
+
require 'pp'
|
3
|
+
|
4
|
+
SCRIPT_LINES__ = {} unless defined? SCRIPT_LINES__
|
5
|
+
|
6
|
+
module Debugger
|
7
|
+
class Mutex
|
8
|
+
def initialize
|
9
|
+
@locker = nil
|
10
|
+
@waiting = []
|
11
|
+
@locked = false;
|
12
|
+
end
|
13
|
+
|
14
|
+
def locked?
|
15
|
+
@locked
|
16
|
+
end
|
17
|
+
|
18
|
+
def lock
|
19
|
+
return if Thread.critical
|
20
|
+
return if @locker == Thread.current
|
21
|
+
while (Thread.critical = true; @locked)
|
22
|
+
@waiting.push Thread.current
|
23
|
+
Thread.stop
|
24
|
+
end
|
25
|
+
@locked = true
|
26
|
+
@locker = Thread.current
|
27
|
+
Thread.critical = false
|
28
|
+
self
|
29
|
+
end
|
30
|
+
|
31
|
+
def unlock
|
32
|
+
return if Thread.critical
|
33
|
+
return unless @locked
|
34
|
+
unless @locker == Thread.current
|
35
|
+
raise RuntimeError, "unlocked by other"
|
36
|
+
end
|
37
|
+
Thread.critical = true
|
38
|
+
t = @waiting.shift
|
39
|
+
@locked = false
|
40
|
+
@locker = nil
|
41
|
+
Thread.critical = false
|
42
|
+
t.run if t
|
43
|
+
self
|
44
|
+
end
|
45
|
+
end
|
46
|
+
|
47
|
+
MUTEX = Mutex.new
|
48
|
+
|
49
|
+
@stdout = STDOUT
|
50
|
+
@display = []
|
51
|
+
@trace_proc = nil
|
52
|
+
|
53
|
+
class Context
|
54
|
+
DEBUG_LAST_CMD = []
|
55
|
+
|
56
|
+
private
|
57
|
+
|
58
|
+
begin
|
59
|
+
require 'readline'
|
60
|
+
def readline(prompt, hist)
|
61
|
+
Readline::readline(prompt, hist)
|
62
|
+
end
|
63
|
+
rescue LoadError
|
64
|
+
def readline(prompt, hist)
|
65
|
+
STDOUT.print prompt
|
66
|
+
STDOUT.flush
|
67
|
+
line = STDIN.gets
|
68
|
+
exit unless line
|
69
|
+
line.chomp!
|
70
|
+
line
|
71
|
+
end
|
72
|
+
USE_READLINE = false
|
73
|
+
end
|
74
|
+
|
75
|
+
def stdout
|
76
|
+
Debugger.stdout
|
77
|
+
end
|
78
|
+
|
79
|
+
def at_breakpoint(breakpoint, method_name)
|
80
|
+
n = Debugger.breakpoints.index(breakpoint) + 1
|
81
|
+
stdout.printf "Breakpoint %d, %s at %s:%s\n", n, method_name, breakpoint.source, breakpoint.pos
|
82
|
+
end
|
83
|
+
|
84
|
+
def at_catchpoint
|
85
|
+
frames = Debugger.current_context.frames
|
86
|
+
stdout.printf "%s:%d: `%s' (%s)\n", frames[0].file, frames[0].line, $!, $!.class
|
87
|
+
fs = frames.size
|
88
|
+
tb = caller(0)[-fs..-1]
|
89
|
+
if tb
|
90
|
+
for i in tb
|
91
|
+
stdout.printf "\tfrom %s\n", i
|
92
|
+
end
|
93
|
+
end
|
94
|
+
end
|
95
|
+
|
96
|
+
def at_tracing(file, line)
|
97
|
+
if Debugger.trace_proc
|
98
|
+
Debugger.trace_proc.call(self.thnum, file, line)
|
99
|
+
else
|
100
|
+
stdout.puts sprintf("Tracing(%d):%s:%s %s", self.thnum, file, line, line_at(file, line))
|
101
|
+
end
|
102
|
+
end
|
103
|
+
|
104
|
+
def debug_command(file, line, id, binding)
|
105
|
+
MUTEX.lock
|
106
|
+
frame_pos = 0
|
107
|
+
binding_file = file
|
108
|
+
binding_line = line
|
109
|
+
previous_line = nil
|
110
|
+
stdout.printf "%s:%d: %s", binding_file, binding_line, line_at(binding_file, binding_line)
|
111
|
+
display_expressions(binding)
|
112
|
+
prompt = true
|
113
|
+
while prompt and input = readline("(rdb:%d) " % Debugger.thnum, true)
|
114
|
+
catch(:debug_error) do
|
115
|
+
if input == ""
|
116
|
+
next unless DEBUG_LAST_CMD[0]
|
117
|
+
input = DEBUG_LAST_CMD[0]
|
118
|
+
else
|
119
|
+
DEBUG_LAST_CMD[0] = input
|
120
|
+
end
|
121
|
+
|
122
|
+
case input
|
123
|
+
when /^\s*s(?:tep)?(?:\s+(\d+))?$/
|
124
|
+
self.stop_next = $1 ? $1.to_i : 1
|
125
|
+
prompt = false
|
126
|
+
|
127
|
+
when /^\s*c(?:ont)?$|^\s*r(?:un)?$/
|
128
|
+
prompt = false
|
129
|
+
|
130
|
+
when /^\s*v(?:ar)?\s+/
|
131
|
+
debug_variable_info($', binding)
|
132
|
+
|
133
|
+
when /^\s*w(?:here)?$/, /^\s*f(?:rame)?$/
|
134
|
+
display_frames(frame_pos)
|
135
|
+
|
136
|
+
when /^\s*l(?:ist)?(?:\s+(.+))?$/
|
137
|
+
if not $1
|
138
|
+
b = previous_line ? previous_line + 10 : binding_line - 5
|
139
|
+
e = b + 9
|
140
|
+
elsif $1 == '-'
|
141
|
+
b = previous_line ? previous_line - 10 : binding_line - 5
|
142
|
+
e = b + 9
|
143
|
+
else
|
144
|
+
b, e = $1.split(/[-,]/)
|
145
|
+
if e
|
146
|
+
b = b.to_i
|
147
|
+
e = e.to_i
|
148
|
+
else
|
149
|
+
b = b.to_i - 5
|
150
|
+
e = b + 9
|
151
|
+
end
|
152
|
+
end
|
153
|
+
previous_line = b
|
154
|
+
display_list(b, e, binding_file, binding_line)
|
155
|
+
|
156
|
+
when /^\s*n(?:ext)?(?:\s+(\d+))?$/
|
157
|
+
steps = $1 ? $1.to_i : 1
|
158
|
+
self.step_over steps, self.frames.size - frame_pos
|
159
|
+
prompt = false
|
160
|
+
|
161
|
+
when /^\s*up(?:\s+(\d+))?$/
|
162
|
+
previous_line = nil
|
163
|
+
frame_pos += $1 ? $1.to_i : 1
|
164
|
+
if frame_pos >= self.frames.size
|
165
|
+
frame_pos = self.frames.size - 1
|
166
|
+
stdout.puts "At toplevel"
|
167
|
+
end
|
168
|
+
frame = self.frames[frame_pos]
|
169
|
+
binding, binding_file, binding_line = frame.binding, frame.file, frame.line
|
170
|
+
stdout.print format_frame(frame, frame_pos)
|
171
|
+
|
172
|
+
when /^\s*down(?:\s+(\d+))?$/
|
173
|
+
previous_line = nil
|
174
|
+
frame_pos -= $1 ? $1.to_i : 1
|
175
|
+
if frame_pos < 0
|
176
|
+
frame_pos = 0
|
177
|
+
stdout.print "At stack bottom\n"
|
178
|
+
end
|
179
|
+
frame = self.frames[frame_pos]
|
180
|
+
binding, binding_file, binding_line = frame.binding, frame.file, frame.line
|
181
|
+
stdout.print format_frame(frame, frame_pos)
|
182
|
+
|
183
|
+
when /^\s*fin(?:ish)?$/
|
184
|
+
if frame_pos == self.frames.size
|
185
|
+
stdout.print "\"finish\" not meaningful in the outermost frame.\n"
|
186
|
+
else
|
187
|
+
self.stop_frame = self.frames.size - frame_pos
|
188
|
+
frame_pos = 0
|
189
|
+
prompt = false
|
190
|
+
end
|
191
|
+
|
192
|
+
when /^\s*b(?:reak)?\s+(?:(.+):)?([^.:\s]+)\s*(?:\sif\s+(.+))?$/
|
193
|
+
pos = $2
|
194
|
+
expr = $3
|
195
|
+
if $1
|
196
|
+
klass = debug_silent_eval($1, binding)
|
197
|
+
if klass && !klass.kind_of?(Module)
|
198
|
+
stdout.puts "Unknown class #$1"
|
199
|
+
throw :debug_error
|
200
|
+
end
|
201
|
+
klass = klass.name if klass
|
202
|
+
file = $1
|
203
|
+
end
|
204
|
+
if pos =~ /^\d+$/
|
205
|
+
pname = pos
|
206
|
+
pos = pos.to_i
|
207
|
+
else
|
208
|
+
pname = pos = pos.intern.id2name
|
209
|
+
end
|
210
|
+
Debugger.add_breakpoint klass || file, pos, expr
|
211
|
+
stdout.printf "Set breakpoint %d at %s:%s\n", Debugger.breakpoints.size, klass || file, pname
|
212
|
+
|
213
|
+
when /^\s*b(?:reak)?\s+(.+)[#.]([^.:\s]+)(?:\s+if\s+(.+))?$/
|
214
|
+
pos = $2.intern.id2name
|
215
|
+
expr = $3
|
216
|
+
klass = debug_eval($1, binding)
|
217
|
+
if klass.nil? || !klass.kind_of?(Module)
|
218
|
+
stdout.puts "Unknown class #$1"
|
219
|
+
throw :debug_error
|
220
|
+
end
|
221
|
+
Debugger.add_breakpoint klass.name, pos, expr
|
222
|
+
stdout.printf "Set breakpoint %d at %s.%s\n", Debugger.breakpoints.size, klass, pos
|
223
|
+
|
224
|
+
when /^\s*b(?:reak)?$/
|
225
|
+
unless Debugger.breakpoints.empty?
|
226
|
+
stdout.print "Breakpoints:\n"
|
227
|
+
Debugger.breakpoints.each_with_index do |b, n|
|
228
|
+
if b.expr.nil?
|
229
|
+
stdout.printf " %d %s:%s\n", n+1, b.source, b.pos
|
230
|
+
else
|
231
|
+
stdout.printf " %d %s:%s if %s\n", n+1, b.source, b.pos, b.expr
|
232
|
+
end
|
233
|
+
end
|
234
|
+
stdout.puts
|
235
|
+
else
|
236
|
+
stdout.print "No breakpoints\n"
|
237
|
+
end
|
238
|
+
when /^\s*del(?:ete)?(?:\s+(\d+))?$/
|
239
|
+
pos = $1
|
240
|
+
unless pos
|
241
|
+
input = readline("Clear all breakpoints? (y/n) ", false)
|
242
|
+
if input == "y"
|
243
|
+
Debugger.breakpoints.clear
|
244
|
+
end
|
245
|
+
else
|
246
|
+
pos = pos.to_i
|
247
|
+
unless Debugger.breakpoints.delete_at(pos-1)
|
248
|
+
stdout.printf "Breakpoint %d is not defined\n", pos
|
249
|
+
end
|
250
|
+
end
|
251
|
+
|
252
|
+
when /^\s*th(?:read)?\s+/
|
253
|
+
if Debugger.debug_thread_info($') == :cont
|
254
|
+
prompt = false
|
255
|
+
end
|
256
|
+
|
257
|
+
when /^\s*m(?:ethod)?\s+/
|
258
|
+
debug_method_info($', binding)
|
259
|
+
|
260
|
+
when /^\s*pp\s+/
|
261
|
+
PP.pp(debug_eval($', binding), stdout) rescue stdout.puts $!.message
|
262
|
+
|
263
|
+
when /^\s*p\s+/
|
264
|
+
stdout.printf "%s\n", debug_eval($', binding).inspect
|
265
|
+
|
266
|
+
when /^\s*h(?:elp)?$/
|
267
|
+
debug_print_help()
|
268
|
+
|
269
|
+
when /^\s*q(?:uit)?$/
|
270
|
+
input = readline("Really quit? (y/n) ", false)
|
271
|
+
if input == "y"
|
272
|
+
exit! # exit -> exit!: No graceful way to stop threads...
|
273
|
+
end
|
274
|
+
|
275
|
+
when /^\s*disp(?:lay)?\s+(.+)$/
|
276
|
+
exp = $1
|
277
|
+
display.push [true, exp]
|
278
|
+
stdout.printf "%d: ", display.size
|
279
|
+
display_expression(exp, binding)
|
280
|
+
|
281
|
+
when /^\s*disp(?:lay)?$/
|
282
|
+
display_expressions(binding)
|
283
|
+
|
284
|
+
when /^\s*undisp(?:lay)?(?:\s+(\d+))?$/
|
285
|
+
pos = $1
|
286
|
+
unless pos
|
287
|
+
input = readline("Clear all expressions? (y/n) ", false)
|
288
|
+
if input == "y"
|
289
|
+
for d in display
|
290
|
+
d[0] = false
|
291
|
+
end
|
292
|
+
end
|
293
|
+
else
|
294
|
+
pos = pos.to_i
|
295
|
+
if display[pos-1]
|
296
|
+
display[pos-1][0] = false
|
297
|
+
else
|
298
|
+
stdout.printf "Display expression %d is not defined\n", pos
|
299
|
+
end
|
300
|
+
end
|
301
|
+
|
302
|
+
when /^\s*cat(?:ch)?(?:\s+(.+))?$/
|
303
|
+
if $1
|
304
|
+
excn = $1
|
305
|
+
if excn == 'off'
|
306
|
+
Debugger.catchpoint = nil
|
307
|
+
stdout.print "Clear catchpoint.\n"
|
308
|
+
else
|
309
|
+
Debugger.catchpoint = excn
|
310
|
+
stdout.printf "Set catchpoint %s.\n", excn
|
311
|
+
end
|
312
|
+
else
|
313
|
+
if Debugger.catchpoint
|
314
|
+
stdout.printf "Catchpoint %s.\n", Debugger.catchpoint
|
315
|
+
else
|
316
|
+
stdout.print "No catchpoint.\n"
|
317
|
+
end
|
318
|
+
end
|
319
|
+
|
320
|
+
when /^\s*tr(?:ace)?(?:\s+(on|off))?(?:\s+(all))?$/
|
321
|
+
if defined?( $2 )
|
322
|
+
Debugger.tracing = $1 == 'on'
|
323
|
+
elsif defined?( $1 )
|
324
|
+
self.tracing = $1 == 'on'
|
325
|
+
end
|
326
|
+
if Debugger.tracing || self.tracing
|
327
|
+
stdout.print "Trace on.\n"
|
328
|
+
else
|
329
|
+
stdout.print "Trace off.\n"
|
330
|
+
end
|
331
|
+
|
332
|
+
else
|
333
|
+
v = debug_eval(input, binding)
|
334
|
+
stdout.printf "%s\n", v.inspect
|
335
|
+
end
|
336
|
+
end
|
337
|
+
end
|
338
|
+
MUTEX.unlock
|
339
|
+
Debugger.resume
|
340
|
+
end
|
341
|
+
|
342
|
+
def debug_print_help
|
343
|
+
stdout.print <<EOHELP
|
344
|
+
ruby-debug help v.#{Debugger::VERSION}
|
345
|
+
Commands
|
346
|
+
b[reak] [file|class:]<line|method> [if expr]
|
347
|
+
b[reak] [class.]<line|method> [if expr]
|
348
|
+
set breakpoint to some position,
|
349
|
+
optionally if expr == true
|
350
|
+
cat[ch] <an Exception> set catchpoint to an exception
|
351
|
+
cat[ch] show catchpoint
|
352
|
+
disp[lay] <expression> add expression into display expression list
|
353
|
+
undisp[lay][ nnn] delete one particular or all display expressions
|
354
|
+
b[reak] list breakpoints
|
355
|
+
del[ete][ nnn] delete some or all breakpoints
|
356
|
+
c[ont] run until program ends or hit breakpoint
|
357
|
+
r[un] alias for cont
|
358
|
+
s[tep][ nnn] step (into methods) one line or till line nnn
|
359
|
+
n[ext][ nnn] go over one line or till line nnn
|
360
|
+
w[here] display frames
|
361
|
+
f[rame] alias for where
|
362
|
+
l[ist][ (-|nn-mm)] list program, - list backwards, nn-mm list given lines
|
363
|
+
up[ nn] move to higher frame
|
364
|
+
down[ nn] move to lower frame
|
365
|
+
fin[ish] return to outer frame
|
366
|
+
q[uit] exit from debugger
|
367
|
+
v[ar] g[lobal] show global variables
|
368
|
+
v[ar] l[ocal] show local variables
|
369
|
+
v[ar] i[nstance] <object> show instance variables of object
|
370
|
+
v[ar] c[onst] <object> show constants of object
|
371
|
+
m[ethod] i[nstance] <obj> show methods of object
|
372
|
+
m[ethod] <class|module> show instance methods of class or module
|
373
|
+
th[read] l[ist] list all threads
|
374
|
+
th[read] c[ur[rent]] show current thread
|
375
|
+
th[read] [sw[itch]] <nnn> switch thread context to nnn
|
376
|
+
th[read] stop <nnn> stop thread nnn
|
377
|
+
th[read] resume <nnn> resume thread nnn
|
378
|
+
p expression evaluate expression and print its value
|
379
|
+
pp expression evaluate expression and print its value
|
380
|
+
h[elp] print this help
|
381
|
+
<everything else> evaluate
|
382
|
+
EOHELP
|
383
|
+
end
|
384
|
+
|
385
|
+
def display_expressions(binding)
|
386
|
+
n = 1
|
387
|
+
for d in display
|
388
|
+
if d[0]
|
389
|
+
stdout.printf "%d: ", n
|
390
|
+
display_expression(d[1], binding)
|
391
|
+
end
|
392
|
+
n += 1
|
393
|
+
end
|
394
|
+
end
|
395
|
+
|
396
|
+
def display_expression(exp, binding)
|
397
|
+
stdout.printf "%s = %s\n", exp, debug_silent_eval(exp, binding).to_s
|
398
|
+
end
|
399
|
+
|
400
|
+
def debug_eval(str, binding)
|
401
|
+
begin
|
402
|
+
val = eval(str, binding)
|
403
|
+
rescue StandardError, ScriptError => e
|
404
|
+
at = eval("caller(1)", binding)
|
405
|
+
stdout.printf "%s:%s\n", at.shift, e.to_s.sub(/\(eval\):1:(in `.*?':)?/, '')
|
406
|
+
for i in at
|
407
|
+
stdout.printf "\tfrom %s\n", i
|
408
|
+
end
|
409
|
+
throw :debug_error
|
410
|
+
end
|
411
|
+
end
|
412
|
+
|
413
|
+
def debug_silent_eval(str, binding)
|
414
|
+
begin
|
415
|
+
eval(str, binding)
|
416
|
+
rescue StandardError, ScriptError
|
417
|
+
nil
|
418
|
+
end
|
419
|
+
end
|
420
|
+
|
421
|
+
def debug_variable_info(input, binding)
|
422
|
+
case input
|
423
|
+
when /^\s*g(?:lobal)?\s*$/
|
424
|
+
var_list(global_variables, binding)
|
425
|
+
|
426
|
+
when /^\s*l(?:ocal)?\s*$/
|
427
|
+
var_list(eval("local_variables", binding), binding)
|
428
|
+
|
429
|
+
when /^\s*i(?:nstance)?\s+/
|
430
|
+
obj = debug_eval($', binding)
|
431
|
+
var_list(obj.instance_variables, obj.instance_eval{binding()})
|
432
|
+
|
433
|
+
when /^\s*c(?:onst(?:ant)?)?\s+/
|
434
|
+
obj = debug_eval($', binding)
|
435
|
+
unless obj.kind_of? Module
|
436
|
+
stdout.print "Should be Class/Module: ", $', "\n"
|
437
|
+
else
|
438
|
+
var_list(obj.constants, obj.module_eval{binding()})
|
439
|
+
end
|
440
|
+
end
|
441
|
+
end
|
442
|
+
|
443
|
+
def display_frames(pos)
|
444
|
+
self.frames.each_with_index do |frame, idx|
|
445
|
+
if idx == pos
|
446
|
+
stdout.print "--> "
|
447
|
+
else
|
448
|
+
stdout.print " "
|
449
|
+
end
|
450
|
+
stdout.print format_frame(frame, idx)
|
451
|
+
end
|
452
|
+
end
|
453
|
+
|
454
|
+
def format_frame(frame, pos)
|
455
|
+
bind, file, line, id = frame.binding, frame.file, frame.line, frame.id
|
456
|
+
sprintf "#%d %s:%s%s\n", pos + 1, file, line,
|
457
|
+
(id ? ":in `#{id.id2name}'" : "")
|
458
|
+
end
|
459
|
+
|
460
|
+
def var_list(ary, binding)
|
461
|
+
ary.sort!
|
462
|
+
for v in ary
|
463
|
+
stdout.printf " %s => %s\n", v, eval(v, binding).inspect
|
464
|
+
end
|
465
|
+
end
|
466
|
+
|
467
|
+
def display_list(b, e, file, line)
|
468
|
+
stdout.printf "[%d, %d] in %s\n", b, e, file
|
469
|
+
if lines = SCRIPT_LINES__[file] and lines != true
|
470
|
+
n = 0
|
471
|
+
b.upto(e) do |n|
|
472
|
+
if n > 0 && lines[n-1]
|
473
|
+
if n == line
|
474
|
+
stdout.printf "=> %d %s\n", n, lines[n-1].chomp
|
475
|
+
else
|
476
|
+
stdout.printf " %d %s\n", n, lines[n-1].chomp
|
477
|
+
end
|
478
|
+
end
|
479
|
+
end
|
480
|
+
else
|
481
|
+
stdout.printf "No sourcefile available for %s\n", file
|
482
|
+
end
|
483
|
+
end
|
484
|
+
|
485
|
+
def line_at(file, line)
|
486
|
+
lines = SCRIPT_LINES__[file]
|
487
|
+
if lines
|
488
|
+
return "\n" if lines == true
|
489
|
+
line = lines[line-1]
|
490
|
+
return "\n" unless line
|
491
|
+
return line.gsub(/^\s+/, '')
|
492
|
+
end
|
493
|
+
return "\n"
|
494
|
+
end
|
495
|
+
|
496
|
+
def display
|
497
|
+
Debugger.display
|
498
|
+
end
|
499
|
+
end
|
500
|
+
|
501
|
+
class << self
|
502
|
+
def stdout
|
503
|
+
@stdout
|
504
|
+
end
|
505
|
+
|
506
|
+
def stdout=(v)
|
507
|
+
@stdout = v
|
508
|
+
end
|
509
|
+
|
510
|
+
def set_trace_proc &block
|
511
|
+
@trace_proc = block
|
512
|
+
end
|
513
|
+
|
514
|
+
def trace_proc
|
515
|
+
@trace_proc
|
516
|
+
end
|
517
|
+
|
518
|
+
def display
|
519
|
+
@display
|
520
|
+
end
|
521
|
+
|
522
|
+
def debug_method_info(input, binding)
|
523
|
+
case input
|
524
|
+
when /^i(:?nstance)?\s+/
|
525
|
+
obj = debug_eval($', binding)
|
526
|
+
|
527
|
+
len = 0
|
528
|
+
for v in obj.methods.sort
|
529
|
+
len += v.size + 1
|
530
|
+
if len > 70
|
531
|
+
len = v.size + 1
|
532
|
+
stdout.print "\n"
|
533
|
+
end
|
534
|
+
stdout.print v, " "
|
535
|
+
end
|
536
|
+
stdout.print "\n"
|
537
|
+
|
538
|
+
else
|
539
|
+
obj = debug_eval(input, binding)
|
540
|
+
unless obj.kind_of? Module
|
541
|
+
stdout.print "Should be Class/Module: ", input, "\n"
|
542
|
+
else
|
543
|
+
len = 0
|
544
|
+
for v in obj.instance_methods(false).sort
|
545
|
+
len += v.size + 1
|
546
|
+
if len > 70
|
547
|
+
len = v.size + 1
|
548
|
+
stdout.print "\n"
|
549
|
+
end
|
550
|
+
stdout.print v, " "
|
551
|
+
end
|
552
|
+
stdout.print "\n"
|
553
|
+
end
|
554
|
+
end
|
555
|
+
end
|
556
|
+
|
557
|
+
def display_context(c)
|
558
|
+
if c.thread == Thread.current
|
559
|
+
stdout.print "+"
|
560
|
+
else
|
561
|
+
stdout.print " "
|
562
|
+
end
|
563
|
+
stdout.printf "%d ", c.thnum
|
564
|
+
stdout.print c.thread.inspect, "\t"
|
565
|
+
last_frame = c.frames.first
|
566
|
+
if last_frame
|
567
|
+
@stdout.print last_frame.file,":",last_frame.line
|
568
|
+
end
|
569
|
+
@stdout.print "\n"
|
570
|
+
end
|
571
|
+
|
572
|
+
def display_all_contexts
|
573
|
+
threads = contexts.sort_by{|c| c.thnum}.each do |c|
|
574
|
+
display_context(c)
|
575
|
+
end
|
576
|
+
end
|
577
|
+
|
578
|
+
def get_context(thnum)
|
579
|
+
contexts.find{|c| c.thnum = thnum}
|
580
|
+
end
|
581
|
+
|
582
|
+
def debug_thread_info(input)
|
583
|
+
case input
|
584
|
+
when /^l(?:ist)?/
|
585
|
+
display_all_contexts
|
586
|
+
|
587
|
+
when /^c(?:ur(?:rent)?)?$/
|
588
|
+
display_context(current_context)
|
589
|
+
|
590
|
+
when /^(?:sw(?:itch)?\s+)?(\d+)/
|
591
|
+
c = get_context($1.to_i)
|
592
|
+
if c == current_context
|
593
|
+
stdout.print "It's the current thread.\n"
|
594
|
+
else
|
595
|
+
display_context(c)
|
596
|
+
c.stop_next = 1
|
597
|
+
c.thread.run
|
598
|
+
return :cont
|
599
|
+
end
|
600
|
+
|
601
|
+
when /^stop\s+(\d+)/
|
602
|
+
c = get_context($1.to_i)
|
603
|
+
if c == current_context
|
604
|
+
stdout.print "It's the current thread.\n"
|
605
|
+
elsif c.thread.stop?
|
606
|
+
stdout.print "Already stopped.\n"
|
607
|
+
else
|
608
|
+
display_context(c)
|
609
|
+
c.set_suspend
|
610
|
+
end
|
611
|
+
|
612
|
+
when /^resume\s+(\d+)/
|
613
|
+
c = get_context($1.to_i)
|
614
|
+
if c == current_context
|
615
|
+
stdout.print "It's the current thread.\n"
|
616
|
+
elsif !c.thread.stop?
|
617
|
+
stdout.print "Already running."
|
618
|
+
else
|
619
|
+
display_context(c)
|
620
|
+
c.thread.run
|
621
|
+
end
|
622
|
+
end
|
623
|
+
end
|
624
|
+
end
|
625
|
+
end
|
626
|
+
|
627
|
+
module Kernel
|
628
|
+
def debugger(steps = 1)
|
629
|
+
Debugger.current_context.stop_next = steps
|
630
|
+
end
|
631
|
+
end
|
632
|
+
|
633
|
+
Debugger.start
|