1#!/usr/bin/env perl
2#***************************************************************************
3#                                  _   _ ____  _
4#  Project                     ___| | | |  _ \| |
5#                             / __| | | | |_) | |
6#                            | (__| |_| |  _ <| |___
7#                             \___|\___/|_| \_\_____|
8#
9# Copyright (C) 1998 - 2013, Daniel Stenberg, <daniel@haxx.se>, et al.
10#
11# This software is licensed as described in the file COPYING, which
12# you should have received as part of this distribution. The terms
13# are also available at https://curl.haxx.se/docs/copyright.html.
14#
15# You may opt to use, copy, modify, merge, publish, distribute and/or sell
16# copies of the Software, and permit persons to whom the Software is
17# furnished to do so, under the terms of the COPYING file.
18#
19# This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
20# KIND, either express or implied.
21#
22###########################################################################
23#
24# Example input:
25#
26# MEM mprintf.c:1094 malloc(32) = e5718
27# MEM mprintf.c:1103 realloc(e5718, 64) = e6118
28# MEM sendf.c:232 free(f6520)
29
30my $mallocs=0;
31my $callocs=0;
32my $reallocs=0;
33my $strdups=0;
34my $wcsdups=0;
35my $showlimit;
36
37while(1) {
38    if($ARGV[0] eq "-v") {
39        $verbose=1;
40        shift @ARGV;
41    }
42    elsif($ARGV[0] eq "-t") {
43        $trace=1;
44        shift @ARGV;
45    }
46    elsif($ARGV[0] eq "-l") {
47        # only show what alloc that caused a memlimit failure
48        $showlimit=1;
49        shift @ARGV;
50    }
51    else {
52        last;
53    }
54}
55
56my $maxmem;
57
58sub newtotal {
59    my ($newtot)=@_;
60    # count a max here
61
62    if($newtot > $maxmem) {
63        $maxmem= $newtot;
64    }
65}
66
67my $file = $ARGV[0];
68
69if(! -f $file) {
70    print "Usage: memanalyze.pl [options] <dump file>\n",
71    "Options:\n",
72    " -l  memlimit failure displayed\n",
73    " -v  Verbose\n",
74    " -t  Trace\n";
75    exit;
76}
77
78open(FILE, "<$file");
79
80if($showlimit) {
81    while(<FILE>) {
82        if(/^LIMIT.*memlimit$/) {
83            print $_;
84            last;
85        }
86    }
87    close(FILE);
88    exit;
89}
90
91
92my $lnum=0;
93while(<FILE>) {
94    chomp $_;
95    $line = $_;
96    $lnum++;
97    if($line =~ /^LIMIT ([^ ]*):(\d*) (.*)/) {
98        # new memory limit test prefix
99        my $i = $3;
100        my ($source, $linenum) = ($1, $2);
101        if($trace && ($i =~ /([^ ]*) reached memlimit/)) {
102            print "LIMIT: $1 returned error at $source:$linenum\n";
103        }
104    }
105    elsif($line =~ /^MEM ([^ ]*):(\d*) (.*)/) {
106        # generic match for the filename+linenumber
107        $source = $1;
108        $linenum = $2;
109        $function = $3;
110
111        if($function =~ /free\((\(nil\)|0x([0-9a-f]*))/) {
112            $addr = $2;
113            if($1 eq "(nil)") {
114                ; # do nothing when free(NULL)
115            }
116            elsif(!exists $sizeataddr{$addr}) {
117                print "FREE ERROR: No memory allocated: $line\n";
118            }
119            elsif(-1 == $sizeataddr{$addr}) {
120                print "FREE ERROR: Memory freed twice: $line\n";
121                print "FREE ERROR: Previously freed at: ".$getmem{$addr}."\n";
122            }
123            else {
124                $totalmem -= $sizeataddr{$addr};
125                if($trace) {
126                    print "FREE: malloc at ".$getmem{$addr}." is freed again at $source:$linenum\n";
127                    printf("FREE: %d bytes freed, left allocated: $totalmem bytes\n", $sizeataddr{$addr});
128                }
129
130                newtotal($totalmem);
131                $frees++;
132
133                $sizeataddr{$addr}=-1; # set -1 to mark as freed
134                $getmem{$addr}="$source:$linenum";
135
136            }
137        }
138        elsif($function =~ /malloc\((\d*)\) = 0x([0-9a-f]*)/) {
139            $size = $1;
140            $addr = $2;
141
142            if($sizeataddr{$addr}>0) {
143                # this means weeeeeirdo
144                print "Mixed debug compile ($source:$linenum at line $lnum), rebuild curl now\n";
145                print "We think $sizeataddr{$addr} bytes are already allocated at that memory address: $addr!\n";
146            }
147
148            $sizeataddr{$addr}=$size;
149            $totalmem += $size;
150
151            if($trace) {
152                print "MALLOC: malloc($size) at $source:$linenum",
153                " makes totally $totalmem bytes\n";
154            }
155
156            newtotal($totalmem);
157            $mallocs++;
158
159            $getmem{$addr}="$source:$linenum";
160        }
161        elsif($function =~ /calloc\((\d*),(\d*)\) = 0x([0-9a-f]*)/) {
162            $size = $1*$2;
163            $addr = $3;
164
165            $arg1 = $1;
166            $arg2 = $2;
167
168            if($sizeataddr{$addr}>0) {
169                # this means weeeeeirdo
170                print "Mixed debug compile, rebuild curl now\n";
171            }
172
173            $sizeataddr{$addr}=$size;
174            $totalmem += $size;
175
176            if($trace) {
177                print "CALLOC: calloc($arg1,$arg2) at $source:$linenum",
178                " makes totally $totalmem bytes\n";
179            }
180
181            newtotal($totalmem);
182            $callocs++;
183
184            $getmem{$addr}="$source:$linenum";
185        }
186        elsif($function =~ /realloc\((\(nil\)|0x([0-9a-f]*)), (\d*)\) = 0x([0-9a-f]*)/) {
187            my ($oldaddr, $newsize, $newaddr) = ($2, $3, $4);
188
189            $totalmem -= $sizeataddr{$oldaddr};
190            if($trace) {
191                printf("REALLOC: %d less bytes and ", $sizeataddr{$oldaddr});
192            }
193            $sizeataddr{$oldaddr}=0;
194
195            $totalmem += $newsize;
196            $sizeataddr{$newaddr}=$newsize;
197
198            if($trace) {
199                printf("%d more bytes ($source:$linenum)\n", $newsize);
200            }
201
202            newtotal($totalmem);
203            $reallocs++;
204
205            $getmem{$oldaddr}="";
206            $getmem{$newaddr}="$source:$linenum";
207        }
208        elsif($function =~ /strdup\(0x([0-9a-f]*)\) \((\d*)\) = 0x([0-9a-f]*)/) {
209            # strdup(a5b50) (8) = df7c0
210
211            $dup = $1;
212            $size = $2;
213            $addr = $3;
214            $getmem{$addr}="$source:$linenum";
215            $sizeataddr{$addr}=$size;
216
217            $totalmem += $size;
218
219            if($trace) {
220                printf("STRDUP: $size bytes at %s, makes totally: %d bytes\n",
221                       $getmem{$addr}, $totalmem);
222            }
223
224            newtotal($totalmem);
225            $strdups++;
226        }
227        elsif($function =~ /wcsdup\(0x([0-9a-f]*)\) \((\d*)\) = 0x([0-9a-f]*)/) {
228            # wcsdup(a5b50) (8) = df7c0
229
230            $dup = $1;
231            $size = $2;
232            $addr = $3;
233            $getmem{$addr}="$source:$linenum";
234            $sizeataddr{$addr}=$size;
235
236            $totalmem += $size;
237
238            if($trace) {
239                printf("WCSDUP: $size bytes at %s, makes totally: %d bytes\n",
240                       $getmem{$addr}, $totalmem);
241            }
242
243            newtotal($totalmem);
244            $wcsdups++;
245        }
246        else {
247            print "Not recognized input line: $function\n";
248        }
249    }
250    # FD url.c:1282 socket() = 5
251    elsif($_ =~ /^FD ([^ ]*):(\d*) (.*)/) {
252        # generic match for the filename+linenumber
253        $source = $1;
254        $linenum = $2;
255        $function = $3;
256
257        if($function =~ /socket\(\) = (\d*)/) {
258            $filedes{$1}=1;
259            $getfile{$1}="$source:$linenum";
260            $openfile++;
261        }
262        elsif($function =~ /socketpair\(\) = (\d*) (\d*)/) {
263            $filedes{$1}=1;
264            $getfile{$1}="$source:$linenum";
265            $openfile++;
266            $filedes{$2}=1;
267            $getfile{$2}="$source:$linenum";
268            $openfile++;
269        }
270        elsif($function =~ /accept\(\) = (\d*)/) {
271            $filedes{$1}=1;
272            $getfile{$1}="$source:$linenum";
273            $openfile++;
274        }
275        elsif($function =~ /sclose\((\d*)\)/) {
276            if($filedes{$1} != 1) {
277                print "Close without open: $line\n";
278            }
279            else {
280                $filedes{$1}=0; # closed now
281                $openfile--;
282            }
283        }
284    }
285    # FILE url.c:1282 fopen("blabla") = 0x5ddd
286    elsif($_ =~ /^FILE ([^ ]*):(\d*) (.*)/) {
287        # generic match for the filename+linenumber
288        $source = $1;
289        $linenum = $2;
290        $function = $3;
291
292        if($function =~ /f[d]*open\(\"(.*)\",\"([^\"]*)\"\) = (\(nil\)|0x([0-9a-f]*))/) {
293            if($3 eq "(nil)") {
294                ;
295            }
296            else {
297                $fopen{$4}=1;
298                $fopenfile{$4}="$source:$linenum";
299                $fopens++;
300            }
301        }
302        # fclose(0x1026c8)
303        elsif($function =~ /fclose\(0x([0-9a-f]*)\)/) {
304            if(!$fopen{$1}) {
305                print "fclose() without fopen(): $line\n";
306            }
307            else {
308                $fopen{$1}=0;
309                $fopens--;
310            }
311        }
312    }
313    # GETNAME url.c:1901 getnameinfo()
314    elsif($_ =~ /^GETNAME ([^ ]*):(\d*) (.*)/) {
315        # not much to do
316    }
317
318    # ADDR url.c:1282 getaddrinfo() = 0x5ddd
319    elsif($_ =~ /^ADDR ([^ ]*):(\d*) (.*)/) {
320        # generic match for the filename+linenumber
321        $source = $1;
322        $linenum = $2;
323        $function = $3;
324
325        if($function =~ /getaddrinfo\(\) = (\(nil\)|0x([0-9a-f]*))/) {
326            my $add = $2;
327            if($add eq "(nil)") {
328                ;
329            }
330            else {
331                $addrinfo{$add}=1;
332                $addrinfofile{$add}="$source:$linenum";
333                $addrinfos++;
334            }
335            if($trace) {
336                printf("GETADDRINFO ($source:$linenum)\n");
337            }
338        }
339        # fclose(0x1026c8)
340        elsif($function =~ /freeaddrinfo\(0x([0-9a-f]*)\)/) {
341            if(!$addrinfo{$1}) {
342                print "freeaddrinfo() without getaddrinfo(): $line\n";
343            }
344            else {
345                $addrinfo{$1}=0;
346                $addrinfos--;
347            }
348            if($trace) {
349                printf("FREEADDRINFO ($source:$linenum)\n");
350            }
351        }
352
353    }
354    else {
355        print "Not recognized prefix line: $line\n";
356    }
357}
358close(FILE);
359
360if($totalmem) {
361    print "Leak detected: memory still allocated: $totalmem bytes\n";
362
363    for(keys %sizeataddr) {
364        $addr = $_;
365        $size = $sizeataddr{$addr};
366        if($size > 0) {
367            print "At $addr, there's $size bytes.\n";
368            print " allocated by ".$getmem{$addr}."\n";
369        }
370    }
371}
372
373if($openfile) {
374    for(keys %filedes) {
375        if($filedes{$_} == 1) {
376            print "Open file descriptor created at ".$getfile{$_}."\n";
377        }
378    }
379}
380
381if($fopens) {
382    print "Open FILE handles left at:\n";
383    for(keys %fopen) {
384        if($fopen{$_} == 1) {
385            print "fopen() called at ".$fopenfile{$_}."\n";
386        }
387    }
388}
389
390if($addrinfos) {
391    print "IPv6-style name resolve data left at:\n";
392    for(keys %addrinfofile) {
393        if($addrinfo{$_} == 1) {
394            print "getaddrinfo() called at ".$addrinfofile{$_}."\n";
395        }
396    }
397}
398
399if($verbose) {
400    print "Mallocs: $mallocs\n",
401    "Reallocs: $reallocs\n",
402    "Callocs: $callocs\n",
403    "Strdups:  $strdups\n",
404    "Wcsdups:  $wcsdups\n",
405    "Frees: $frees\n",
406    "Allocations: ".($mallocs + $callocs + $reallocs + $strdups + $wcsdups)."\n";
407
408    print "Maximum allocated: $maxmem\n";
409}
410