https://github.com/root-project/root
Tip revision: 15f7c582488836affc446314a9785fc1b9eb4b74 authored by Fons Rademakers on 10 March 2008, 16:11:36 UTC
tag patch release v5-18-00b.
tag patch release v5-18-00b.
Tip revision: 15f7c58
memprobe.in
#!@perl@
################################################################################
#
# Authors : D.Bertini and M.Ivanov
# Date: 23/10/2000
# Updated: 10/05/2001 D.Bertini, v2.0 port to gdb5.0 + glibc2.2
#
#
# To activate the memory checker you have to set in the .rootrc file
# the resource Root.MemCheck to 1 (e.g.: Root.MemCheck: 1) and you
# have to link with libNew.so (e.g. use root-config --new --libs) or
# use rootn.exe. When all this is the case you will find at the end
# of the program execution a file "memcheck.out" in the directory
# where you started your ROOT program. Alternatively you can set
# the resource Root.MemCheckFile to the name of a file to which
# the leak information will be written. The contents of this
# "memcheck.out" file can be analyzed and transformed into printable
# text via the memprobe program (in $ROOTSYS/bin).
#
################################################################################
if ( ( $#ARGV < 0 )) {
Usage();
}
use Getopt::Std;
getopt "h,e,m,d,f";
#Usage("exec file is mandatory !\n") if ( !($opt_h or $opt_e) );
if ( $opt_h ) { Usage();}
if ( $opt_e ) { $ExeFile=$opt_e;
}else {
if (!$opt_h){ Usage("an exec file must be provided !\n");}
}
if ( $opt_m ) { $MemStatFile=$opt_m;
}else{
$MemStatFile="memcheck.out";
}
if ( $opt_d ) { $MemDescFile=$opt_d;
}else{
$MemDescFile="memcheckdesc.out";
}
if ( $opt_f ) { $MemFiltFile=$opt_f;
}else{
$MemFiltFile="analfilter";
}
sub Usage
{
my ($txt) = @_;
print "MemProbe v2.0 Usage:\n";
print "\t $0 -e exec-file [options] [files] \n ";
print "\t-e \t\t\tname of analyzed program\n";
print "Options: \n";
print "\t-h \t\t\tthis usage help\n";
print "\t-m \t\tname of analyzed program output file\n";
print "\t-d \t\tname of the stat description file\n";
print "\t-f \t\tname of the filter file\n";
print "\nWarning: $txt\n" if $txt;
exit 2;
}
# old version
#$ExeFile = shift @ARGV;
#$MemStatFile = $#ARGV >= 0 ? shift @ARGV : "memcheck.out";
open (MEMSTAT, $MemStatFile) or die "-E- Could not open leaks data file $MemStatFile: $!";
if ($#ARGV >= 0) {
$BreakOn = shift @ARGV;
# Rest in @ARGV are program arguments
}
##############################################################################
# Read filter expression
##############################################################################
open (FILTERFILE, $MemFiltFile);
@FILTER = <FILTERFILE>;
close (FILTERFILE);
for ($j = 0; $j <= $#FILTER; ++$j){
chop $FILTER[$j];
# print $FILTER[$j];
}
##############################################################################
# Find leak adresses
##############################################################################
$n = $u = 0;
$nfull = $ufull = 0;
while (<MEMSTAT>) {
# remove newline
chop $_ ;
if (/\s*(stack:)\s*/) {
$addrfull = $'; #obtain stack info
$_ = $addrfull;
s/(st)//g;
s/(0x)/:0x/g;
@arr = split(/:/); #obtain different leak points
for ($i = 1; $i < $#arr; ++$i){
$addr = $arr[$i];
$u++ if not exists $Size{$addr};
$Size{$addr} += 0;
$n++;
}
}
}
close (MEMSTAT);
##############################################################################
# FIND debug info for addresses - information stored
# in file leak.desc.C
##############################################################################
# redirect standard output to the trash
open (STDERR,">/dev/null");
# Redirect list of commands to a file
# using unix pipes (needed for RH.7.1 & gdb5.0)
open (myFile,">commands");
# Change set listsize 2 to something else to show more lines
print myFile "set prompt\nset listsize 1\nset height 0\nset verbose off\n";
if (defined($BreakOn)) {
print myFile "break $BreakOn\n";
print myFile "run ", join(" ", @ARGV), " \n";
}
else{
print myFile "break main \n ";
print myFile "run ", join(" ", @ARGV), " \n";
}
foreach (sort keys %Size) {
print myFile "l \*$_\n ";
}
if (defined($BreakOn)) {
print myFile "kill\n";
}
close (myFile);
##############################################################################
# Calling now gdb services in batch mode using
# dumped list of commands.
# ==> works with gdb-4.18 gdb-5.0 release versions
##############################################################################
open (PIPE, "| gdb -n -q $ExeFile < commands > $MemDescFile")
or die "-E- Cannot start gdb !!!";
close (PIPE);
##############################################################################
# ASIGN debug info to address
##############################################################################
open (DBGinfo, $MemDescFile) or die "-E- Could not open desc file $MemDescFile: $!";
$addr =0;
while (<DBGinfo>){
# if (/(0x[0-9a-f]+)\s*/){
if (/(^0x[0-9a-f]+)/){
$addr = $1;
#print $addr;
/is in /;
# print $';
# print "\n";
$filename{$addr} =$';
# print "$addr $filename{$addr}\n";
}else{
$line{$addr} = $_ if ! /^\s*l/;
}
for ($j = 0; $j <= $#FILTER; ++$j){
if ( ( $line{$addr} =~ /(${FILTER[$j]})/ ) ||
( $filename{$addr} =~ /(${FILTER[$j]})/ )){
$line{$addr} = "filtered";
}
}
}
close (DBGinfo);
##############################################################################
# FIND unique leak stack sequences
##############################################################################
open (MEMSTAT, $MemStatFile) or die "-E- Could not open leaks data file $MemStatFile: $!";
while (<MEMSTAT>) {
# remove newline
chop $_ ;
if (/\s*(stack:)\s*/) {
$addrfull = $'; #obtain stack info
$info =$`;
$info =~ s/size //; #obtain mem info for given stack sequence
$_ = $addrfull;
s/(st)//g;
s/(0x)/:0x/g;
s/ //g;
@arr = split(/:/); #obtain leak points
$addrfull ="stack";
$FilterOut=0;
for ($i = 1; ($i <= $#arr)&&($FilterOut==0); ++$i){
$addr = $arr[$i];
if ( $line{$addr} =~ /filtered/)
{
$FilterOut=1;
}
if ( $filename{$addr} =~ /\)/)
{
$addrfull .= ":";
$addrfull .= $addr;
}
}
if ($FilterOut==0){
$_= $info;
@meminfo = split(/:/);
$ufull++ if not exists $Sizefull{$addrfull};
$SizeTotal{$addrfull} += $meminfo[1];
$CountTotal{$addrfull}+= $meminfo[0];
$SizeLeak{$addrfull} += $meminfo[3];
$CountLeak{$addrfull}+= $meminfo[2];
$nfull++;
}
}
}
print STDERR "total memory corrupted at point==> ($nfull) \n";
print STDERR "unique allocations ==> ($ufull) \n";
close (MEMSTAT);
##############################################################################
# PRINT output leak information to leak.info.C output file
##############################################################################
open (LEAKinfo,">leak.info");
open (MULTI,">multidelete.info");
open (MEMSTAT,">memcheck.info");
# sort by size of leak
sub bysize {
$SizeLeak{$b} <=> $SizeLeak{$a}; # presuming integers
}
sub bysizetotal {
$SizeTotal{$b} <=> $SizeTotal{$a}; # presuming integers
}
foreach (sort bysizetotal keys %SizeTotal) {
print MEMSTAT "Count: $CountTotal{$_}($CountLeak{$_}) Size: $SizeTotal{$_}($SizeLeak{$_}) \n";
s/stack://g;
@arr = split(/:/); #obtain leak points
for ($i = 0; $i <= $#arr; ++$i){
$addr = $arr[$i];
print MEMSTAT "$filename{$addr}$line{$addr}";
}
print MEMSTAT "\n\n";
}
foreach (sort bysize keys %SizeLeak) {
if ($CountLeak{$_}<0){
print MULTI "Multiple deallocation :$CountLeak{$_}\n";
s/stack://g;
@arr = split(/:/); #obtain leak points
for ($i = 0; $i <= $#arr; ++$i){
$addr = $arr[$i];
print MULTI "$filename{$addr}$line{$addr}";
}
print MULTI "\n\n";
}
if ($SizeLeak{$_}!=0){
print LEAKinfo "Leaked allocations: $CountLeak{$_} \t Leaked size: $SizeLeak{$_} \n";
s/stack://g;
@arr = split(/:/); #obtain leak points
for ($i = 0; $i <= $#arr; ++$i){
$addr = $arr[$i];
print LEAKinfo "$filename{$addr}$line{$addr}";
}
print LEAKinfo "\n\n";
}
}
close(LEAKinfo);
close(MULTI);
close(MEMSTAT);
$remove = 'rm -rf commands';
exec $remove;
die " -E- no commands file !";