line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
# -*- coding: utf-8 -*- |
2
|
|
|
|
|
|
|
# Copyright (C) 2011-2012, 2014-2015 Rocky Bernstein <rocky@cpan.org> |
3
|
12
|
|
|
12
|
|
83
|
use strict; use warnings; |
|
12
|
|
|
12
|
|
32
|
|
|
12
|
|
|
|
|
334
|
|
|
12
|
|
|
|
|
62
|
|
|
12
|
|
|
|
|
40
|
|
|
12
|
|
|
|
|
389
|
|
4
|
12
|
|
|
12
|
|
70
|
use rlib '../../..'; |
|
12
|
|
|
|
|
34
|
|
|
12
|
|
|
|
|
93
|
|
5
|
12
|
|
|
12
|
|
4437
|
use Devel::Trepan::DB::LineCache; # for map_file |
|
12
|
|
|
|
|
36
|
|
|
12
|
|
|
|
|
2232
|
|
6
|
12
|
|
|
12
|
|
90
|
use Devel::Trepan::Complete; |
|
12
|
|
|
|
|
33
|
|
|
12
|
|
|
|
|
1157
|
|
7
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
package Devel::Trepan::CmdProcessor; |
9
|
12
|
|
|
12
|
|
100
|
use English qw( -no_match_vars ); |
|
12
|
|
|
|
|
27
|
|
|
12
|
|
|
|
|
100
|
|
10
|
|
|
|
|
|
|
|
11
|
|
|
|
|
|
|
sub frame_complete($$;$) |
12
|
|
|
|
|
|
|
{ |
13
|
0
|
|
|
0
|
0
|
|
my ($self, $prefix, $direction) = @_; |
14
|
0
|
0
|
|
|
|
|
$direction = 1 unless defined $direction; |
15
|
0
|
|
|
|
|
|
my ($low, $high) = $self->frame_low_high($direction); |
16
|
0
|
|
|
|
|
|
my @ary = ($low..$high); |
17
|
0
|
|
|
|
|
|
Devel::Trepan::Complete::complete_token(\@ary, $prefix); |
18
|
|
|
|
|
|
|
} |
19
|
|
|
|
|
|
|
|
20
|
|
|
|
|
|
|
sub print_stack_entry() |
21
|
|
|
|
|
|
|
{ |
22
|
0
|
|
|
0
|
0
|
|
my ($self, $frame, $i, $prefix, $opts) = @_; |
23
|
0
|
0
|
|
|
|
|
$opts->{maxstack} = 1e9 unless defined $opts->{maxstack}; |
24
|
|
|
|
|
|
|
# Set the separator so arrays print nice. |
25
|
0
|
|
|
|
|
|
local $LIST_SEPARATOR = ', '; |
26
|
|
|
|
|
|
|
|
27
|
|
|
|
|
|
|
# Get the file name. |
28
|
0
|
|
|
|
|
|
my $file = $self->canonic_file($frame->{file}); |
29
|
0
|
0
|
|
|
|
|
$file = '??' unless defined $file; |
30
|
|
|
|
|
|
|
|
31
|
|
|
|
|
|
|
# Put in a filename header if short is off. |
32
|
0
|
0
|
|
|
|
|
$file = ($file eq '-e') ? $file : "file `$file'" unless $opts->{short}; |
|
|
0
|
|
|
|
|
|
33
|
|
|
|
|
|
|
|
34
|
0
|
|
|
|
|
|
my $not_last_frame = $i != ($self->{stack_size}-1); |
35
|
0
|
|
|
|
|
|
my $s = ''; |
36
|
|
|
|
|
|
|
my $args = |
37
|
|
|
|
|
|
|
defined $frame->{args} |
38
|
0
|
0
|
|
|
|
|
? "(@{ $frame->{args} })" |
|
0
|
|
|
|
|
|
|
39
|
|
|
|
|
|
|
: ''; |
40
|
0
|
0
|
|
|
|
|
if ($not_last_frame) { |
41
|
|
|
|
|
|
|
# Grab and stringify the arguments if they are there. |
42
|
|
|
|
|
|
|
|
43
|
|
|
|
|
|
|
# Shorten them up if $opts->{maxwidth} says they're too long. |
44
|
|
|
|
|
|
|
$args = substr($args, 0, $opts->{maxwidth}-3) . '...' |
45
|
0
|
0
|
|
|
|
|
if length($args) > $opts->{maxwidth}; |
46
|
|
|
|
|
|
|
|
47
|
|
|
|
|
|
|
# Get the actual sub's name, and shorten to $maxwidth's requirement. |
48
|
0
|
0
|
|
|
|
|
if (exists($frame->{fn})) { |
49
|
0
|
|
|
|
|
|
$s = $frame->{fn}; |
50
|
|
|
|
|
|
|
$s = substr($s, 0, $opts->{maxwidth}-3) . '...' |
51
|
0
|
0
|
|
|
|
|
if length($s) > $opts->{maxwidth}; |
52
|
|
|
|
|
|
|
} |
53
|
|
|
|
|
|
|
} |
54
|
|
|
|
|
|
|
|
55
|
|
|
|
|
|
|
# Short report uses trimmed file and sub names. |
56
|
0
|
|
|
|
|
|
my $want_array; |
57
|
0
|
0
|
|
|
|
|
if (exists($frame->{wantarray})) { |
58
|
0
|
|
|
|
|
|
$want_array = "$frame->{wantarray} = "; |
59
|
|
|
|
|
|
|
} else { |
60
|
0
|
|
|
|
|
|
$not_last_frame = 0; |
61
|
0
|
|
|
|
|
|
$want_array = '' ; |
62
|
|
|
|
|
|
|
} |
63
|
|
|
|
|
|
|
|
64
|
0
|
|
0
|
|
|
|
my $lineno = $frame->{line} || '??'; |
65
|
0
|
0
|
0
|
|
|
|
my $addr = $opts->{displayop} && $frame->{addr} ? sprintf("0x%x ", $frame->{addr}) : ''; |
66
|
0
|
0
|
|
|
|
|
if ($opts->{short}) { |
67
|
0
|
|
|
|
|
|
my $fn = $s; # @_ >= 4 ? $_[3] : $s; |
68
|
0
|
|
|
|
|
|
my $msg = sprintf("%s%s%s%s from %s:%d", |
69
|
|
|
|
|
|
|
$want_array, $addr, $fn, $args, $file, $lineno); |
70
|
0
|
|
|
|
|
|
$self->msg($msg); |
71
|
|
|
|
|
|
|
} else { |
72
|
|
|
|
|
|
|
# Non-short report includes full names. |
73
|
|
|
|
|
|
|
# Lose the DB::DB hook call if frame is 0. |
74
|
0
|
0
|
|
|
|
|
my $call_str = $not_last_frame ? "$addr$want_array$s$args in " : $addr; |
75
|
0
|
|
|
|
|
|
my $prefix_call = "$prefix$call_str"; |
76
|
0
|
|
|
|
|
|
my $file_line = $file . " at line $lineno"; |
77
|
|
|
|
|
|
|
|
78
|
0
|
0
|
|
|
|
|
if (length($prefix_call) + length($file_line) <= $opts->{maxwidth}) { |
79
|
0
|
|
|
|
|
|
$self->msg($prefix_call . $file_line); |
80
|
|
|
|
|
|
|
} else { |
81
|
0
|
|
|
|
|
|
$self->msg($prefix_call); |
82
|
0
|
|
|
|
|
|
$self->msg("\t" . $file_line); |
83
|
|
|
|
|
|
|
} |
84
|
|
|
|
|
|
|
} |
85
|
|
|
|
|
|
|
} |
86
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
sub print_stack_trace_from_to($$$$$) |
88
|
|
|
|
|
|
|
{ |
89
|
0
|
|
|
0
|
0
|
|
my ($self, $from, $to, $frames, $opts) = @_; |
90
|
0
|
|
|
|
|
|
for (my $i=$from; $i <= $to; $i++) { |
91
|
0
|
0
|
|
|
|
|
my $prefix = ($i == $opts->{current_pos}) ? '-->' : ' '; |
92
|
0
|
|
|
|
|
|
$prefix .= sprintf ' #%d ', $i; |
93
|
0
|
|
|
|
|
|
$self->print_stack_entry($frames->[$i], $i, $prefix, $opts); |
94
|
|
|
|
|
|
|
} |
95
|
|
|
|
|
|
|
} |
96
|
|
|
|
|
|
|
|
97
|
|
|
|
|
|
|
# Print `count' frame entries |
98
|
|
|
|
|
|
|
sub print_stack_trace($$$) |
99
|
|
|
|
|
|
|
{ |
100
|
0
|
|
|
0
|
0
|
|
my ($self, $frames, $opts)=@_; |
101
|
0
|
|
0
|
|
|
|
$opts ||= {maxstack=>1e9, count=>1e9}; |
102
|
|
|
|
|
|
|
# $opts = DEFAULT_STACK_TRACE_SETTINGS.merge(opts); |
103
|
0
|
|
|
|
|
|
my $halfstack = $opts->{maxstack} / 2; |
104
|
0
|
|
|
|
|
|
my $n = scalar @{$frames}; |
|
0
|
|
|
|
|
|
|
105
|
0
|
0
|
|
|
|
|
$n = $opts->{count} if $opts->{count} < $n; |
106
|
0
|
0
|
|
|
|
|
if ($n > ($halfstack * 2)) { |
107
|
0
|
|
|
|
|
|
$self->print_stack_trace_from_to(0, $halfstack-1, $frames, $opts); |
108
|
0
|
|
|
|
|
|
my $msg = sprintf "... %d levels ...", ($n - $halfstack*2); |
109
|
0
|
|
|
|
|
|
$self->msg($msg); |
110
|
0
|
|
|
|
|
|
$self->print_stack_trace_from_to($n - $halfstack, $n-1, $frames, $opts); |
111
|
|
|
|
|
|
|
} else { |
112
|
0
|
|
|
|
|
|
$self->print_stack_trace_from_to(0, $n-1, $frames, $opts); |
113
|
|
|
|
|
|
|
} |
114
|
|
|
|
|
|
|
} |
115
|
|
|
|
|
|
|
|
116
|
|
|
|
|
|
|
1; |