line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package Devel::Leak::Object; |
2
|
|
|
|
|
|
|
|
3
|
8
|
|
|
8
|
|
153800
|
use 5.005; |
|
8
|
|
|
|
|
24
|
|
|
8
|
|
|
|
|
1788
|
|
4
|
|
|
|
|
|
|
# We abuse refs a LOT |
5
|
7
|
|
|
7
|
|
41
|
use strict qw{ vars subs }; |
|
7
|
|
|
|
|
15
|
|
|
7
|
|
|
|
|
1262
|
|
6
|
4
|
|
|
4
|
|
24
|
use Carp (); |
|
4
|
|
|
|
|
12
|
|
|
4
|
|
|
|
|
71
|
|
7
|
4
|
|
|
4
|
|
21
|
use Scalar::Util (); |
|
4
|
|
|
|
|
8
|
|
|
4
|
|
|
|
|
86
|
|
8
|
|
|
|
|
|
|
|
9
|
4
|
|
|
4
|
|
20
|
use vars qw{ $VERSION @ISA @EXPORT_OK }; |
|
4
|
|
|
|
|
7
|
|
|
4
|
|
|
|
|
306
|
|
10
|
4
|
|
|
4
|
|
21
|
use vars qw{ %OBJECT_COUNT %TRACKED %DESTROY_ORIGINAL %DESTROY_STUBBED %DESTROY_NEXT %IGNORE_CLASS %OBJECT_COUNT_CHECKPOINT }; |
|
4
|
|
|
|
|
7
|
|
|
4
|
|
|
|
|
749
|
|
11
|
|
|
|
|
|
|
BEGIN { |
12
|
4
|
|
|
4
|
|
9
|
$VERSION = '1.01'; |
13
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
# Set up exports |
15
|
4
|
|
|
|
|
26
|
require Exporter; |
16
|
4
|
|
|
|
|
549
|
@ISA = qw(Exporter); |
17
|
4
|
|
|
|
|
18
|
@EXPORT_OK = qw(track bless status checkpoint); |
18
|
|
|
|
|
|
|
|
19
|
|
|
|
|
|
|
# Set up state storage (primary for clarity) |
20
|
4
|
|
|
|
|
11
|
%OBJECT_COUNT = (); |
21
|
4
|
|
|
|
|
7
|
%OBJECT_COUNT_CHECKPOINT = (); |
22
|
4
|
|
|
|
|
7
|
%TRACKED = (); |
23
|
4
|
|
|
|
|
7
|
%DESTROY_ORIGINAL = (); |
24
|
4
|
|
|
|
|
6
|
%DESTROY_STUBBED = (); |
25
|
4
|
|
|
|
|
59
|
%DESTROY_NEXT = (); |
26
|
4
|
|
|
|
|
6221
|
%IGNORE_CLASS = (); |
27
|
|
|
|
|
|
|
} |
28
|
|
|
|
|
|
|
|
29
|
|
|
|
|
|
|
sub import { |
30
|
4
|
|
|
4
|
|
40
|
my $class = shift; |
31
|
4
|
|
|
|
|
13
|
my @import = (); |
32
|
4
|
|
|
|
|
21
|
while ( @_ ) { |
33
|
1
|
|
|
|
|
1
|
my $function = shift; |
34
|
1
|
50
|
|
|
|
8
|
unless ( $function =~ /^GLOBAL_(.*)$/ ) { |
35
|
0
|
|
|
|
|
0
|
push @import, $function; |
36
|
0
|
|
|
|
|
0
|
next; |
37
|
|
|
|
|
|
|
} |
38
|
1
|
|
|
|
|
4
|
my $global = $1; |
39
|
1
|
|
|
|
|
1
|
*{'CORE::GLOBAL::' . $global} = \&{$global}; |
|
1
|
|
|
|
|
13
|
|
|
1
|
|
|
|
|
3
|
|
40
|
|
|
|
|
|
|
} |
41
|
4
|
|
|
|
|
2341
|
return $class->SUPER::import(@import); |
42
|
|
|
|
|
|
|
} |
43
|
|
|
|
|
|
|
|
44
|
|
|
|
|
|
|
sub bless { |
45
|
7
|
|
|
7
|
0
|
23645
|
my $reference = shift; |
46
|
7
|
100
|
|
|
|
26
|
my $class = @_ ? shift : scalar caller; |
47
|
7
|
|
|
|
|
24
|
my $object = CORE::bless($reference, $class); |
48
|
7
|
|
|
|
|
17
|
Devel::Leak::Object::track($object); |
49
|
7
|
|
|
|
|
74
|
return $object; |
50
|
|
|
|
|
|
|
}; |
51
|
|
|
|
|
|
|
|
52
|
|
|
|
|
|
|
sub track { |
53
|
15
|
|
|
15
|
0
|
4566
|
my $object = shift; |
54
|
15
|
|
|
|
|
83
|
my $class = Scalar::Util::blessed($object); |
55
|
15
|
50
|
|
|
|
51
|
unless ( defined $class ) { |
56
|
0
|
|
|
|
|
0
|
Carp::carp("Devel::Leak::Object::track was passed a non-object"); |
57
|
|
|
|
|
|
|
} |
58
|
15
|
50
|
|
|
|
49
|
return if (defined($IGNORE_CLASS{$class})); |
59
|
15
|
|
|
|
|
41
|
my $address = Scalar::Util::refaddr($object); |
60
|
15
|
100
|
|
|
|
49
|
if ( $TRACKED{$address} ) { |
61
|
2
|
|
100
|
|
|
12
|
$TRACKED{$address}->{class} ||= ''; # avoid warnings about uninitialised strings |
62
|
2
|
50
|
|
|
|
8
|
if ( $class eq $TRACKED{$address}->{class} ) { |
63
|
|
|
|
|
|
|
# Reblessing into the same class, ignore |
64
|
0
|
|
|
|
|
0
|
return $OBJECT_COUNT{$class}; |
65
|
|
|
|
|
|
|
} else { |
66
|
|
|
|
|
|
|
# Reblessing into a different class |
67
|
2
|
|
|
|
|
7
|
$OBJECT_COUNT{$TRACKED{$address}->{class}}--; |
68
|
|
|
|
|
|
|
} |
69
|
|
|
|
|
|
|
} |
70
|
|
|
|
|
|
|
|
71
|
|
|
|
|
|
|
# Set or over-write the class name for the tracked object |
72
|
15
|
|
|
|
|
54
|
my ($package, $srcfile, $srcline, $subroutine) = caller(1); |
73
|
15
|
|
100
|
|
|
59
|
$package ||= ''; |
74
|
15
|
|
100
|
|
|
47
|
$subroutine ||= ''; |
75
|
|
|
|
|
|
|
#don't just tell us that we called it from our own new.. |
76
|
15
|
100
|
|
|
|
41
|
if ($package eq $class) { |
77
|
3
|
|
|
|
|
13
|
my ($next_package, $next_srcfile, $next_srcline, $next_subroutine) = caller(2); |
78
|
3
|
100
|
|
|
|
20
|
if ($next_subroutine eq $class.'::new') { |
79
|
2
|
|
|
|
|
6
|
($package, $srcfile, $srcline, $subroutine) = ($next_package, $next_srcfile, $next_srcline, $next_subroutine); |
80
|
|
|
|
|
|
|
} |
81
|
|
|
|
|
|
|
} |
82
|
15
|
|
|
|
|
95
|
$TRACKED{$address} = { class => $class, file => $srcfile, line => $srcline, package=>$package, subroutine=>$subroutine }; |
83
|
|
|
|
|
|
|
|
84
|
|
|
|
|
|
|
# If needed, initialise the new class |
85
|
15
|
100
|
|
|
|
72
|
unless ( $DESTROY_STUBBED{$class} ) { |
86
|
11
|
50
|
100
|
|
|
13
|
if ( exists ${$class.'::'}{DESTROY} and *{$class.'::DESTROY'}{CODE} ) { |
|
11
|
|
|
|
|
62
|
|
|
1
|
|
|
|
|
6
|
|
87
|
|
|
|
|
|
|
# Stash the pre-existing DESTROY function |
88
|
0
|
|
|
|
|
0
|
$DESTROY_ORIGINAL{$class} = \&{$class . '::DESTROY'}; |
|
0
|
|
|
|
|
0
|
|
89
|
|
|
|
|
|
|
} |
90
|
11
|
|
|
|
|
23
|
$DESTROY_STUBBED{$class} = 1; |
91
|
4
|
50
|
33
|
4
|
|
32
|
eval <<"END_DESTROY"; |
|
4
|
50
|
33
|
4
|
|
8
|
|
|
4
|
50
|
|
1
|
|
1608
|
|
|
11
|
50
|
|
|
|
925
|
|
|
4
|
50
|
|
|
|
1838
|
|
|
4
|
100
|
|
|
|
11
|
|
|
4
|
100
|
|
|
|
13
|
|
|
0
|
50
|
|
|
|
0
|
|
|
0
|
100
|
|
|
|
0
|
|
|
4
|
50
|
|
|
|
11
|
|
|
0
|
50
|
|
|
|
0
|
|
|
4
|
50
|
|
|
|
532
|
|
|
4
|
50
|
|
|
|
10
|
|
|
3
|
100
|
|
|
|
9
|
|
|
0
|
50
|
|
|
|
0
|
|
|
3
|
50
|
|
|
|
4
|
|
|
3
|
100
|
|
|
|
10
|
|
|
0
|
0
|
|
|
|
0
|
|
|
3
|
0
|
|
|
|
10
|
|
|
3
|
0
|
|
|
|
8
|
|
|
0
|
0
|
|
|
|
0
|
|
|
0
|
0
|
|
|
|
0
|
|
|
1
|
0
|
|
|
|
1
|
|
|
4
|
0
|
|
|
|
12
|
|
|
1
|
0
|
|
|
|
3
|
|
|
4
|
0
|
|
|
|
8
|
|
|
4
|
50
|
|
|
|
13
|
|
|
0
|
50
|
|
|
|
0
|
|
|
0
|
50
|
|
|
|
0
|
|
|
4
|
50
|
|
|
|
11
|
|
|
2
|
50
|
|
|
|
2
|
|
|
2
|
50
|
|
|
|
10
|
|
|
2
|
50
|
|
|
|
11
|
|
|
3
|
50
|
|
|
|
10
|
|
|
0
|
50
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
1
|
|
|
|
|
2
|
|
|
4
|
|
|
|
|
12
|
|
|
0
|
|
|
|
|
0
|
|
|
4
|
|
|
|
|
8
|
|
|
4
|
|
|
|
|
14
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
4
|
|
|
|
|
9
|
|
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
5
|
|
|
3
|
|
|
|
|
18
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
1
|
|
|
|
|
6
|
|
|
1
|
|
|
|
|
5
|
|
|
1
|
|
|
|
|
8
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
1
|
|
|
|
|
3
|
|
|
0
|
|
|
|
|
0
|
|
|
1
|
|
|
|
|
5
|
|
|
1
|
|
|
|
|
4
|
|
|
1
|
|
|
|
|
4
|
|
|
0
|
|
|
|
|
0
|
|
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
5
|
|
|
0
|
|
|
|
|
0
|
|
|
1
|
|
|
|
|
5
|
|
|
1
|
|
|
|
|
4
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
1
|
|
|
|
|
5
|
|
|
0
|
|
|
|
|
0
|
|
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
5
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
1
|
|
|
|
|
3
|
|
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
|
1
|
|
|
|
|
24
|
|
92
|
|
|
|
|
|
|
package $class;\ |
93
|
|
|
|
|
|
|
no warnings; |
94
|
|
|
|
|
|
|
sub DESTROY { |
95
|
|
|
|
|
|
|
my \$class = Scalar::Util::blessed(\$_[0]); |
96
|
|
|
|
|
|
|
my \$address = Scalar::Util::refaddr(\$_[0]); |
97
|
|
|
|
|
|
|
unless ( defined \$class ) { |
98
|
|
|
|
|
|
|
Carp::carp("Unexpected error: First param to DESTROY is no an object"); |
99
|
|
|
|
|
|
|
return; |
100
|
|
|
|
|
|
|
} |
101
|
|
|
|
|
|
|
unless ( defined \$class ) { |
102
|
|
|
|
|
|
|
die "Unexpected error: First param to DESTROY is no an object"; |
103
|
|
|
|
|
|
|
} |
104
|
|
|
|
|
|
|
|
105
|
|
|
|
|
|
|
# Don't do anything unless tracking for the specific object is set |
106
|
|
|
|
|
|
|
my \$original = \$Devel::Leak::Object::TRACKED{\$address}->{class}; |
107
|
|
|
|
|
|
|
if ( \$original ) { |
108
|
|
|
|
|
|
|
### TODO - We COULD add a check that $class eq |
109
|
|
|
|
|
|
|
# \$Devel::Leak::Object::TRACKED{\$address}->{class} |
110
|
|
|
|
|
|
|
# and then not decrement unless it is the same. |
111
|
|
|
|
|
|
|
# However, in practice it should ALWAYS be the same if |
112
|
|
|
|
|
|
|
# we already have \$Devel::Leak::Object::TRACKED{\$address} |
113
|
|
|
|
|
|
|
# true still, and if for some reason this is wrong, we get |
114
|
|
|
|
|
|
|
# a false positive in the leak counting. |
115
|
|
|
|
|
|
|
# This additional check may be able to be added at a later |
116
|
|
|
|
|
|
|
# date if it turns out to be needed. |
117
|
|
|
|
|
|
|
# if ( \$class eq \$Devel::Leak::Object::TRACKED{\$address} ) { ... } |
118
|
|
|
|
|
|
|
if ( \$class ne \$original ) { |
119
|
|
|
|
|
|
|
warn "Object class '\$class' does not match original ".\$Devel::Leak::Object::TRACKED{\$address}->{class}; |
120
|
|
|
|
|
|
|
} |
121
|
|
|
|
|
|
|
\$Devel::Leak::Object::OBJECT_COUNT{\$original}--; |
122
|
|
|
|
|
|
|
if ( \$Devel::Leak::Object::OBJECT_COUNT{\$original} < 0 ) { |
123
|
|
|
|
|
|
|
warn "Object count for ".\$Devel::Leak::Object::TRACKED{\$address}->{class}." negative (\$Devel::Leak::Object::OBJECT_COUNT{\$original})"; |
124
|
|
|
|
|
|
|
} |
125
|
|
|
|
|
|
|
delete \$Devel::Leak::Object::TRACKED{\$address}; |
126
|
|
|
|
|
|
|
|
127
|
|
|
|
|
|
|
# Hand of to the regular DESTROY method, or pass up to the SUPERclass if there isn't one |
128
|
|
|
|
|
|
|
if ( \$Devel::Leak::Object::DESTROY_ORIGINAL{\$original} ) { |
129
|
|
|
|
|
|
|
goto \&{\$Devel::Leak::Object::DESTROY_ORIGINAL{\$original}}; |
130
|
|
|
|
|
|
|
} |
131
|
|
|
|
|
|
|
} else { |
132
|
|
|
|
|
|
|
\$original = \$class; |
133
|
|
|
|
|
|
|
} |
134
|
|
|
|
|
|
|
|
135
|
|
|
|
|
|
|
# If we don't have the DESTROY_NEXT for this class, populate it |
136
|
|
|
|
|
|
|
unless ( \$Devel::Leak::Object::DESTROY_NEXT{\$original} ) { |
137
|
|
|
|
|
|
|
Devel::Leak::Object::make_next(\$original); |
138
|
|
|
|
|
|
|
} |
139
|
|
|
|
|
|
|
my \$super = \$Devel::Leak::Object::DESTROY_NEXT{\$original}->{'$class'}; |
140
|
|
|
|
|
|
|
unless (( defined \$super ) or (defined(\$Devel::Leak::Object::IGNORE_CLASS{\$class}))) { |
141
|
|
|
|
|
|
|
warn "Failed to find super-method for class \$class in package $class"; |
142
|
|
|
|
|
|
|
\$Devel::Leak::Object::IGNORE_CLASS{\$class} = 1; |
143
|
|
|
|
|
|
|
} |
144
|
|
|
|
|
|
|
if ( \$super ) { |
145
|
|
|
|
|
|
|
goto \&{\$super.'::DESTROY'}; |
146
|
|
|
|
|
|
|
} |
147
|
|
|
|
|
|
|
return; |
148
|
|
|
|
|
|
|
} |
149
|
|
|
|
|
|
|
END_DESTROY |
150
|
11
|
50
|
|
|
|
45
|
if ( $@ ) { |
151
|
0
|
|
|
|
|
0
|
die "Failed to generate DESTROY method for $class: $@"; |
152
|
|
|
|
|
|
|
} |
153
|
|
|
|
|
|
|
|
154
|
|
|
|
|
|
|
# Pre-emptively populate the DESTROY_NEXT map |
155
|
11
|
100
|
|
|
|
34
|
unless ( $DESTROY_NEXT{$class} ) { |
156
|
10
|
|
|
|
|
30
|
make_next($class); |
157
|
|
|
|
|
|
|
} |
158
|
|
|
|
|
|
|
} |
159
|
|
|
|
|
|
|
|
160
|
15
|
|
|
|
|
53
|
$OBJECT_COUNT{$TRACKED{$address}->{class}}++; |
161
|
|
|
|
|
|
|
} |
162
|
|
|
|
|
|
|
|
163
|
|
|
|
|
|
|
sub make_next { |
164
|
14
|
|
|
14
|
0
|
1417
|
my $class = shift; |
165
|
|
|
|
|
|
|
|
166
|
|
|
|
|
|
|
# Build the %DESTROY_NEXT entries to support DESTROY_stub |
167
|
14
|
|
|
|
|
37
|
$DESTROY_NEXT{$class} = {}; |
168
|
14
|
|
|
|
|
66
|
my @stack = ( $class ); |
169
|
11
|
|
|
|
|
32
|
my %seen = ( UNIVERSAL => 1 ); |
170
|
11
|
|
|
|
|
20
|
my @queue = (); |
171
|
14
|
|
|
|
|
570
|
while ( my $c = shift @stack ) { |
172
|
19
|
50
|
|
|
|
56
|
next if $seen{$c}++; |
173
|
|
|
|
|
|
|
|
174
|
|
|
|
|
|
|
# Does the class have it's own DESTROY method |
175
|
|
|
|
|
|
|
my $has_destroy = $DESTROY_STUBBED{$c} |
176
|
|
|
|
|
|
|
? !! exists $DESTROY_ORIGINAL{$c} |
177
|
22
|
100
|
66
|
|
|
62
|
: !! (exists ${"${c}::"}{DESTROY} and *{"${c}::DESTROY"}{CODE}); |
178
|
22
|
100
|
|
|
|
47
|
if ( $has_destroy ) { |
179
|
|
|
|
|
|
|
# Everything in the queue has this class as it's next call |
180
|
5
|
|
|
|
|
17
|
while ( @queue ) { |
181
|
5
|
|
|
|
|
22
|
$DESTROY_NEXT{$class}->{shift(@queue)} = $c; |
182
|
|
|
|
|
|
|
} |
183
|
|
|
|
|
|
|
} else { |
184
|
|
|
|
|
|
|
# This class goes onto the queue |
185
|
20
|
|
|
|
|
34
|
push @queue, $c; |
186
|
|
|
|
|
|
|
} |
187
|
|
|
|
|
|
|
|
188
|
|
|
|
|
|
|
# Add the @ISA to the search stack. |
189
|
22
|
|
|
|
|
28
|
unshift @stack, @{"${c}::ISA"}; |
|
19
|
|
|
|
|
123
|
|
190
|
|
|
|
|
|
|
} |
191
|
|
|
|
|
|
|
|
192
|
|
|
|
|
|
|
# Any else has no target to go to |
193
|
14
|
|
|
|
|
43
|
while ( @queue ) { |
194
|
15
|
|
|
|
|
60
|
$DESTROY_NEXT{$class}->{shift @queue} = ''; |
195
|
|
|
|
|
|
|
} |
196
|
|
|
|
|
|
|
|
197
|
11
|
|
|
|
|
70
|
return 1; |
198
|
|
|
|
|
|
|
} |
199
|
|
|
|
|
|
|
|
200
|
|
|
|
|
|
|
sub checkpoint { |
201
|
0
|
|
|
4
|
0
|
0
|
my $first; |
202
|
0
|
|
|
|
|
0
|
for (sort keys %OBJECT_COUNT) { |
203
|
3
|
50
|
|
|
|
8
|
next unless $OBJECT_COUNT{$_}; # Don't list class with count zero |
204
|
0
|
|
33
|
|
|
0
|
$OBJECT_COUNT_CHECKPOINT{$_} ||= 0; |
205
|
3
|
50
|
|
|
|
4
|
next unless ($OBJECT_COUNT{$_} > $OBJECT_COUNT_CHECKPOINT{$_}); |
206
|
|
|
|
|
|
|
|
207
|
3
|
50
|
|
|
|
8
|
print STDERR "checkpoint:\n" unless ($first++);; |
208
|
0
|
|
|
|
|
0
|
printf STDERR "\t%-40s %d\n", $_, $OBJECT_COUNT{$_}-$OBJECT_COUNT_CHECKPOINT{$_}; |
209
|
|
|
|
|
|
|
|
210
|
0
|
|
|
|
|
0
|
$OBJECT_COUNT_CHECKPOINT{$_} = $OBJECT_COUNT{$_}; |
211
|
|
|
|
|
|
|
} |
212
|
|
|
|
|
|
|
} |
213
|
|
|
|
|
|
|
|
214
|
|
|
|
|
|
|
sub status { |
215
|
7
|
|
|
4
|
0
|
225
|
print STDERR "Tracked objects by class:\n"; |
216
|
6
|
|
|
|
|
42
|
for (sort keys %OBJECT_COUNT) { |
217
|
14
|
100
|
|
|
|
53
|
next if ($_ eq ''); #Don't know what these are.. |
218
|
12
|
100
|
|
|
|
44
|
next unless $OBJECT_COUNT{$_}; # Don't list class with count zero |
219
|
7
|
|
|
|
|
1410
|
printf STDERR "\t%-40s %d\n", $_, $OBJECT_COUNT{$_}; |
220
|
|
|
|
|
|
|
} |
221
|
8
|
50
|
|
|
|
48
|
if($Devel::Leak::Object::TRACKSOURCELINES) { |
222
|
4
|
|
|
|
|
15
|
print STDERR "\nSources of leaks:\n"; |
223
|
0
|
|
|
|
|
0
|
my %classes = (); |
224
|
0
|
|
|
|
|
0
|
foreach my $obj (values(%TRACKED)) { |
225
|
|
|
|
|
|
|
#TODO: no, I don't know why there are some undefined |
226
|
4
|
100
|
|
|
|
10
|
next unless defined($obj->{class}); |
227
|
0
|
|
0
|
|
|
0
|
$classes{$obj->{class}} ||= {}; |
228
|
4
|
|
|
|
|
12
|
my $line = $obj->{file}.' line: '.$obj->{line}; #.' ('.$obj->{package}.' -> '.$obj->{subroutine}.')'; |
229
|
4
|
|
|
|
|
11
|
$classes{$obj->{class}}->{$line}++; |
230
|
|
|
|
|
|
|
} |
231
|
3
|
|
|
|
|
9
|
foreach my $class (sort keys(%classes)) { |
232
|
0
|
|
|
|
|
0
|
printf STDERR "%s\n", $class; |
233
|
3
|
|
|
|
|
5
|
my %lines = %{$classes{$class}}; |
|
3
|
|
|
|
|
19
|
|
234
|
0
|
|
|
|
|
0
|
foreach my $line (sort keys(%lines)) { |
235
|
3
|
|
|
|
|
11
|
printf STDERR "%6d from %s\n", $lines{$line}, $line; |
236
|
|
|
|
|
|
|
} |
237
|
|
|
|
|
|
|
} |
238
|
|
|
|
|
|
|
} |
239
|
|
|
|
|
|
|
} |
240
|
|
|
|
|
|
|
|
241
|
|
|
|
|
|
|
END { |
242
|
4
|
|
|
4
|
|
3071
|
status(); |
243
|
|
|
|
|
|
|
} |
244
|
|
|
|
|
|
|
|
245
|
|
|
|
|
|
|
1; |
246
|
|
|
|
|
|
|
|
247
|
|
|
|
|
|
|
__END__ |