Devel::WxProf - heavy-weight subroutine profiler and graphical profile analyzer


Devel-WxProf documentation Contained in the Devel-WxProf distribution.

Index


Code Index:

NAME

Top

Devel::WxProf - heavy-weight subroutine profiler and graphical profile analyzer

SYNOPSIS

Top

    perl -d:WxProf program.pl
    wxprofile tmon.out

    # or if you like it better:
    dprofpp

    # or even
    perl -d:DProf program.pl
    wxprofile tmon.out

DESCRIPTION

Top

Devel::WxProf package is a heavy-weight subroutine profiler for perl.

It collects information on the execution time of a Perl script - more specifically: on the subs called from that script.

To profile a Perl script run the perl interpreter with the -d debugging switch. The profiler uses the debugging hooks. So to profile script test.pl the following command should be used:

    perl -d:WxProf test.pl

When the script terminates the profiler will dump the profile information to a file called wxmon.out.

Use wxprofile (wxprofile) to display the information collected.

Devel::WxProf uses the (new) output format of Devel::DProf. You may thus use dprofpp to analyze the data, or analyze data collected using Devel::DProf.

Note that real (wall) time is reported as user time, and system tyme is not reported at all, which means that dprofpp is likely to produce slightly different output for profile data created by Devel::WxProf and Devel::DProf.

Devel::DProf usually reports times at 1/100s granularity, so one-shot data collected by Devel::DProf is likely to be useless (and results largely random).

ENVIRONMENT

Top

PERL_DPROF_OUT_FILE_NAME

Filename to save profile data to, default is tmon.out

BUGS AND LIMITATIONS

Top

Many.

Devel::WxProf measures wall clock times. These may be happily inaccurate - especially if run under a system with heavy load, or a program using interactive dialogs. Wall clock (also called stopwatch) times do not nesseccarily represnt the processing time required to run a program.

Devel::WxProf does not profile anonymous subroutines (yet).

Valid profiling data is not saved until the application terminates and runs this modules END{} block. Applications which cause END{} blocks not to run (such as call _exit or exec) will leave a corrupt and/or incomplete temporary data file.

WxProf reports time in your system's high resolution timer's clock ticks - usually micro- or nanoseconds. The exact times reported by WxProf are badly inaccurate.

If the program being profiled contains subroutines which do not return in a normal manner (such as by throwing an exception), the timing data is estimated and may be attributed incorrectly. The time data might even get corrupted.

WxProf (and wxprofile) are heavy-weight tools. Use with care. You should not try to profile benchmarks with WxProf - try it on single runs instead.

SEE ALSO

Top

    Devel::Profile
    Devel::DProfLB
    Devel::DProf
    dprofpp

DEVELOPEMENT

Top

I wrote Devel::WxProf because I needed a fine-grained one-shot profiler (and because I saw that cool treemap in kcachegrind). I actually wrote it for myself. I'd be pleased if you find it useful, but I probably won't put much time into bugfixes. Send me a test and a patch if you want to speed things up. If you're really out for boosting development, I'll set up a repository I can open up...

LICENSE AND COPYRIGHT

Top

AUTHOR

Top

Martin Kutter <martin.kutter fen-net.de>

REPOSITORY INFORMATION

Top

 $Rev: 583 $
 $LastChangedBy: kutterma $
 $Id: $
 $HeadURL: $


Devel-WxProf documentation Contained in the Devel-WxProf distribution.

package Devel::WxProf;
BEGIN {
    # begin block to make sure we don't profile yet...
    use strict; use warnings;
    use version; our $VERSION = qv(0.0.1);
}

package DB;
no strict;
use Time::HiRes qw(time gettimeofday tv_interval);

my @prof_stack =    (); # call stack, to account for subs that haven't returned
my $prof_pid;           # process id of process being profiled
my $tmpfile;            # temporary data file

my $monfile = $ENV{PERL_DPROF_OUT_FILE_NAME} || 'tmon.out';

my $resolution;         # tick resolution
my $overhead;

my %sub_id_of =     (); # associate sub names with IDs
my $sub_id_counter = 0; # ID counter

my $last_real_time;     # records the last time

our $sub;               # needs to be defined

my $hz;                 # clock ticks per second
my $start;              # debugging start time

my $wall_ticks;         # wall clock ticks

sub DB {}
sub sub {}
sub get_time () {
    int( ( time() - $start) * $hz );
}

sub _prof {
    $wall_ticks = get_time();
}

sub calibrate {
    my $start = time();
    for (1..10000) {
        _prof();
        _prof();
    }
    my $end = time();
    $overhead = int( ($end - $start) / 10000 * $hz);
}

BEGIN {
    # profile
    # 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 compiled.
    # 0x200   Provide informative names to anonymous subroutines based on the place they were compiled.
    # 0x400   Debug assertion subroutines enter/exit.

    $^P = 0x311;

    $prof_pid = $$;
    $tmpfile  = "wxmon$$.out";
    open(PROF, ">$tmpfile") || die "cannot open $tmpfile: $!\n";

    # calculate hertz
    $resolution = Time::HiRes::clock_getres();
    $hz = 1/$resolution;

    $start = time();
    # TODO: replace with better calibration
    calibrate();

    $start = time();

    $last_real_time = get_time();
    no warnings qw(redefine);
    *DB::sub = \&_sub;
}

END {
    # original process only, if we fork()
    return unless $$ == $prof_pid;

    $wall_ticks = get_time();

    # generate data for any unfinished subs
    if( @prof_stack ){
        print PROF "# the following did not return, due to program termination\n";
        for my $asx (reverse @prof_stack){
            print PROF "- $sub_id_of{$asx}\n";
        }
    }

    {
        # wipe out DB::sub to stop profiling here.
        no warnings qw(redefine);
        *DB::sub = sub {};
    }
    close PROF;

    my $elapsed = int( ( time()-$start) * $hz );

    open(my $fh, ">", $monfile) || die "cannot open $monfile: $!\n";

    # output header
    print $fh "#fOrTyTwO\n";
    # this portion of the header is blindly evaled by dprofpp
    # any valid perl may be placed here
    print $fh "# Devel::WxProf";
    print $fh "# All timing values are given in \n";
    print $fh "\$hz = $hz;\n";
    print $fh "# Estimated profiling overhead per call:\n";
    print $fh "\$over_utime=$overhead; \$over_stime=0; \$over_rtime=$overhead;
\$over_tests=1;
\$rrun_utime=$elapsed; \$rrun_stime=0; \$rrun_rtime=$elapsed;
\$total_marks=$elapsed;

";

    print $fh "PART2\n";

    open my $tmp_fh, "<", $tmpfile;
    while (my $line = <$tmp_fh>) {
        print $fh $line;
    }
    close $tmp_fh;
    close $fh;

    unlink $tmpfile;

    $prof_pid = undef;
}

sub _sub {
    my( $wa, $r, @r );
    my $elapsed;

    $wall_ticks = get_time();
    $elapsed = $wall_ticks - $last_real_time;
    $last_real_time = $wall_ticks;

    my $sx = $sub;
    if( ref $sx ) {
        my @caller_from = caller(0);
        if ($caller_from[0] ne 'DB') {
            my @caller_from = caller(1);
            if (defined($caller_from[3]) && $caller_from[3] eq '(eval)') {
                if (defined($caller_from[7]) && $caller_from[7]==1) {
                    $sx = undef; # a eval from a require
                }
                else {
                    if (!$caller_from[0]) {
                        $sx = undef; # a eval from main - hard to profile, might be "use"
                    }
                    else {
                        $sx = $caller_from[0] . "::__ANON__[$caller_from[2]]";
                    }
                }
            }
            else {
                $sx = undef;
            }
        }
        else {
            $sx = undef;
        }
    }

    if ($sx) {
        if (! exists ($sub_id_of{ $sx })) {
            $sub_id_of{ $sx } = $sub_id_counter++;
            my ($package, $sub_name) = $sx =~m{ (.+)*::(.+) $}x;
            warn $sx if not $package;
            print PROF "& $sub_id_of{ $sx } $package $sub_name\n";
        }

        # do not corrupt data on fork()
        $prof_pid ||= 0;
        my $noprof = $$ != $prof_pid;
        if( $noprof ){
            close PROF;
        }

        # print PROF "+ $wall_ticks $sx\n" unless $noprof;
        print PROF "\@ $elapsed 0 $elapsed\n+ $sub_id_of{ $sx }\n";

        push @prof_stack, $sx;
        my $ss = @prof_stack;

        $wa = wantarray;
        if( $wa ){
            @r = &$sub;
        }
        elsif( defined $wa ){
            $r = &$sub;
        }
        else {
            &$sub;
        }

        if( $ss < @prof_stack ){
            # we took an exception - account for aborted subs

            while( $ss < @prof_stack ){
                my $asx = pop @prof_stack;
                next if $noprof;
                print PROF "\@ $elapsed 0 $elapsed\n- $sub_id_of{ $sx }\n";
            }
        }

        pop @prof_stack;
        print PROF "\@ $elapsed 0 $elapsed\n- $sub_id_of{ $sx }\n";

    }
    else {
        $wa = wantarray;
        if( $wa ){
            @r = &$sub;
        }
        elsif( defined $wa ){
            $r = &$sub;
        }
        else {
            &$sub;
        }
    }

    if( $wa ){
        @r;
    } else {
        $r;
    }

}

1;

__END__