/usr/share/perl5/Devel/callcount.pm is in libur-perl 0.440-1.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 | package Devel::callsfrom;
use Data::Dumper;
# From perldoc perlvar
# Debugger flags, so you can see what we turn on below.
#
# 0x01 Debug subroutine enter/exit.
#
# 0x02 Line-by-line debugging.
#
# 0x04 Switch off optimizations.
#
# 0x08 Preserve more data for future interactive inspections.
#
# 0x10 Keep info about source lines on which a subroutine is defined.
#
# 0x20 Start with single-step on.
#
# 0x40 Use subroutine address instead of name when reporting.
#
# 0x80 Report "goto &subroutine" as well.
#
# 0x100 Provide informative "file" names for evals based on the place they were com-
# piled.
#
# 0x200 Provide informative names to anonymous subroutines based on the place they
# were compiled.
#
# 0x400 Debug assertion subroutines enter/exit.
#
BEGIN { $^P |= (0x01 | 0x80 | 0x100 | 0x200); };
#BEGIN { $^P |= (0x004 | 0x100 ); };
sub import { }
package DB;
# Any debugger needs to have a sub DB. It doesn't need to do anything.
sub DB{};
# We want to track how deep our subroutines go
our $CALL_DEPTH = 0;
our %CALLED;
our $CALL_WATCH = $ENV{CALL_WATCH};
sub sub {
local $DB::CALL_DEPTH = $DB::CALL_DEPTH+1;
no strict;
no warnings;
my @c0 = caller(0);
my @c1 = caller(-1);
my ($pkg,$file,$line) = @c1;
my $csub = $c0[3] || '-';
my $caller = join(",", $file,$line,$pkg,$csub);
print STDERR ((' ' x $DB::CALL_DEPTH) . $DB::sub{$DB::sub} . ' > ' . $DB::sub . "(@_) : " . $caller . "\n") if $CALL_WATCH;
$DB::CALLED{$DB::sub}{$caller}++;
&{$DB::sub};
}
END {
use strict;
use warnings;
my %counts;
for my $sub (keys %DB::sub) {
my $cases = $DB::CALLED{$sub};
my @callers = keys %$cases;
my $call_count = scalar(@callers);
$counts{$call_count}{$sub} = $cases;
}
my @counts = keys %counts;
my $call_min = $ENV{CALL_MIN};
if (defined $call_min) {
@counts = grep { $_ >= $call_min } @counts;
}
my $call_max = $ENV{CALL_MAX};
if (defined $call_max) {
@counts = grep { $_ <= $call_max } @counts;
}
my $fh;
if (my $fname = $ENV{CALL_COUNT_OUTFILE}) {
open($fh,">$fname");
unless ($fh) { die "failed to open outfile for call count for $0!" };
}
else {
open($fh,">$0.callcount");
$fh or die "failed to open output file $0.callcount: $!";
}
for my $c (sort { $a <=> $b } @counts) {
my $subs = $counts{$c};
for my $sub (sort keys %$subs) {
my $cases = $subs->{$sub};
my @calls = sort keys %$cases;
print $fh join("\t",$c, $sub,$DB::sub{$sub},@calls),"\n";
}
}
}
1;
|