1999-10-07 06:11:44 +04:00
|
|
|
/* -*- Mode: Java; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Netscape Public
|
|
|
|
* License Version 1.1 (the "License"); you may not use this file
|
|
|
|
* except in compliance with the License. You may obtain a copy of
|
|
|
|
* the License at http://www.mozilla.org/NPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS
|
|
|
|
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express oqr
|
|
|
|
* implied. See the License for the specific language governing
|
|
|
|
* rights and limitations under the License.
|
|
|
|
*
|
|
|
|
* The Original Code is Mozilla Communicator client code, released
|
|
|
|
* March 31, 1998.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is Netscape
|
|
|
|
* Communications Corporation. Portions created by Netscape are
|
|
|
|
* Copyright (C) 1998 Netscape Communications Corporation. All
|
|
|
|
* Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
*
|
|
|
|
* Patrick C. Beard <beard@netscape.com>
|
|
|
|
*
|
|
|
|
* Alternatively, the contents of this file may be used under the
|
|
|
|
* terms of the GNU Public License (the "GPL"), in which case the
|
|
|
|
* provisions of the GPL are applicable instead of those above.
|
|
|
|
* If you wish to allow use of your version of this file only
|
|
|
|
* under the terms of the GPL and not to allow others to use your
|
|
|
|
* version of this file under the NPL, indicate your decision by
|
|
|
|
* deleting the provisions above and replace them with the notice
|
|
|
|
* and other provisions required by the GPL. If you do not delete
|
|
|
|
* the provisions above, a recipient may use your version of this
|
|
|
|
* file under either the NPL or the GPL.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import java.io.*;
|
|
|
|
import java.util.*;
|
|
|
|
|
1999-10-14 03:42:49 +04:00
|
|
|
class Type {
|
|
|
|
String mName;
|
|
|
|
int mSize;
|
|
|
|
|
1999-10-15 19:56:05 +04:00
|
|
|
Type(String name, int size) {
|
1999-10-14 03:42:49 +04:00
|
|
|
mName = name;
|
|
|
|
mSize = size;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int hashCode() {
|
|
|
|
return mName.hashCode() + mSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean equals(Object obj) {
|
|
|
|
if (obj instanceof Type) {
|
|
|
|
Type t = (Type) obj;
|
|
|
|
return (t.mSize == mSize && t.mName.equals(mName));
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String toString() {
|
1999-10-15 19:56:05 +04:00
|
|
|
return "<A HREF=\"#" + mName + "_" + mSize + "\"><" + mName + "></A> (" + mSize + ")";
|
|
|
|
}
|
|
|
|
|
|
|
|
static class Comparator implements QuickSort.Comparator {
|
|
|
|
public int compare(Object obj1, Object obj2) {
|
|
|
|
Type t1 = (Type) obj1, t2 = (Type) obj2;
|
|
|
|
return (t1.mSize - t2.mSize);
|
1999-10-14 03:42:49 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-10-07 06:11:44 +04:00
|
|
|
class Leak {
|
|
|
|
String mAddress;
|
|
|
|
Object[] mReferences;
|
1999-10-12 22:59:52 +04:00
|
|
|
Object[] mCrawl;
|
1999-10-07 06:11:44 +04:00
|
|
|
int mRefCount;
|
1999-10-14 03:42:49 +04:00
|
|
|
Type mType;
|
1999-10-07 06:11:44 +04:00
|
|
|
|
1999-10-14 03:42:49 +04:00
|
|
|
Leak(String addr, Type type, Object[] refs, Object[] crawl) {
|
1999-10-07 06:11:44 +04:00
|
|
|
mAddress = addr;
|
|
|
|
mReferences = refs;
|
1999-10-12 22:59:52 +04:00
|
|
|
mCrawl = crawl;
|
1999-10-07 06:11:44 +04:00
|
|
|
mRefCount = 0;
|
|
|
|
mType = type;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String toString() {
|
1999-10-14 03:42:49 +04:00
|
|
|
return ("<A HREF=\"#" + mAddress + "\">" + mAddress + "</A> [" + mRefCount + "] " + mType);
|
1999-10-07 06:11:44 +04:00
|
|
|
}
|
|
|
|
|
1999-10-15 19:56:05 +04:00
|
|
|
static class ByCount implements QuickSort.Comparator {
|
1999-10-07 06:11:44 +04:00
|
|
|
public int compare(Object obj1, Object obj2) {
|
|
|
|
Leak l1 = (Leak) obj1, l2 = (Leak) obj2;
|
1999-10-15 19:56:05 +04:00
|
|
|
// return (l1.mRefCount - l2.mRefCount);
|
|
|
|
return (l1.mType.mSize - l2.mType.mSize);
|
1999-10-07 06:11:44 +04:00
|
|
|
}
|
|
|
|
}
|
1999-10-15 19:56:05 +04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Sorts in order of decreasing size.
|
|
|
|
*/
|
|
|
|
static class BySize implements QuickSort.Comparator {
|
|
|
|
public int compare(Object obj1, Object obj2) {
|
|
|
|
Leak l1 = (Leak) obj1, l2 = (Leak) obj2;
|
|
|
|
return (l2.mType.mSize - l1.mType.mSize);
|
|
|
|
}
|
1999-10-08 08:12:11 +04:00
|
|
|
}
|
1999-10-07 06:11:44 +04:00
|
|
|
}
|
|
|
|
|
1999-10-12 22:59:52 +04:00
|
|
|
class FileTable {
|
|
|
|
static class Line {
|
|
|
|
int mOffset;
|
|
|
|
int mLength;
|
|
|
|
|
|
|
|
Line(int offset, int length) {
|
|
|
|
mOffset = offset;
|
|
|
|
mLength = length;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Line[] mLines;
|
|
|
|
|
|
|
|
FileTable(String path) throws IOException {
|
|
|
|
Vector lines = new Vector();
|
|
|
|
BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream(path)));
|
|
|
|
int offset = 0;
|
|
|
|
for (String line = reader.readLine(); line != null; line = reader.readLine()) {
|
|
|
|
// always add 1 for the line feed.
|
|
|
|
int length = 1 + line.length();
|
|
|
|
lines.addElement(new Line(offset, length));
|
|
|
|
offset += length;
|
|
|
|
}
|
|
|
|
reader.close();
|
|
|
|
int size = lines.size();
|
|
|
|
mLines = new Line[size];
|
|
|
|
lines.copyInto(mLines);
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getLine(int offset) {
|
|
|
|
// use binary search to find the line which spans this offset.
|
|
|
|
int length = mLines.length;
|
|
|
|
int minIndex = 0, maxIndex = length - 1;
|
|
|
|
int index = maxIndex / 2;
|
1999-10-14 03:42:49 +04:00
|
|
|
while (minIndex <= maxIndex) {
|
1999-10-12 22:59:52 +04:00
|
|
|
Line line = mLines[index];
|
|
|
|
if (offset < line.mOffset) {
|
|
|
|
maxIndex = (index - 1);
|
|
|
|
index = (minIndex + maxIndex) / 2;
|
|
|
|
} else {
|
1999-10-14 03:42:49 +04:00
|
|
|
if (offset < (line.mOffset + line.mLength)) {
|
|
|
|
return index;
|
|
|
|
}
|
1999-10-12 22:59:52 +04:00
|
|
|
minIndex = (index + 1);
|
|
|
|
index = (minIndex + maxIndex) / 2;
|
|
|
|
}
|
|
|
|
}
|
1999-10-14 03:42:49 +04:00
|
|
|
// this case shouldn't happen, but provides a helpful value to detect errors.
|
|
|
|
return -1;
|
1999-10-12 22:59:52 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-10-07 06:11:44 +04:00
|
|
|
public class leaksoup {
|
1999-10-14 03:42:49 +04:00
|
|
|
/**
|
|
|
|
* Set by the "-blame" option when generating leak reports.
|
|
|
|
*/
|
|
|
|
static boolean USE_BLAME = false;
|
|
|
|
|
1999-10-07 06:11:44 +04:00
|
|
|
public static void main(String[] args) {
|
1999-10-08 08:12:11 +04:00
|
|
|
if (args.length == 0) {
|
|
|
|
System.out.println("usage: leaksoup leaks");
|
|
|
|
System.exit(1);
|
|
|
|
}
|
|
|
|
|
1999-10-14 03:42:49 +04:00
|
|
|
String inputName = null;
|
|
|
|
|
|
|
|
for (int i = 0; i < args.length; i++) {
|
|
|
|
String arg = args[i];
|
|
|
|
if (arg.charAt(0) == '-') {
|
|
|
|
if (arg.equals("-blame"))
|
|
|
|
USE_BLAME = true;
|
|
|
|
} else {
|
|
|
|
inputName = arg;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-10-07 06:11:44 +04:00
|
|
|
try {
|
|
|
|
Vector vec = new Vector();
|
|
|
|
Hashtable table = new Hashtable();
|
1999-10-15 19:56:05 +04:00
|
|
|
Hashtable types = new Hashtable();
|
1999-10-08 08:12:11 +04:00
|
|
|
Histogram hist = new Histogram();
|
|
|
|
BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream(inputName)));
|
1999-10-07 06:11:44 +04:00
|
|
|
String line = reader.readLine();
|
|
|
|
while (line != null) {
|
|
|
|
if (line.startsWith("0x")) {
|
|
|
|
String addr = line.substring(0, 10);
|
1999-10-14 03:42:49 +04:00
|
|
|
String name = line.substring(line.indexOf('<') + 1, line.indexOf('>'));
|
1999-10-07 06:11:44 +04:00
|
|
|
int size;
|
|
|
|
try {
|
1999-10-08 08:12:11 +04:00
|
|
|
String str = line.substring(line.indexOf('(') + 1, line.indexOf(')')).trim();
|
1999-10-07 06:11:44 +04:00
|
|
|
size = Integer.parseInt(str);
|
|
|
|
} catch (NumberFormatException nfe) {
|
|
|
|
size = 0;
|
|
|
|
}
|
1999-10-15 19:56:05 +04:00
|
|
|
|
|
|
|
// generate a unique type for this object.
|
|
|
|
String key = name + "_" + size;
|
|
|
|
Type type = (Type) types.get(key);
|
|
|
|
if (type == null) {
|
|
|
|
type = new Type(name, size);
|
|
|
|
types.put(key, type);
|
|
|
|
}
|
1999-10-12 22:59:52 +04:00
|
|
|
|
|
|
|
// read in fields.
|
1999-10-07 06:11:44 +04:00
|
|
|
vec.setSize(0);
|
|
|
|
for (line = reader.readLine(); line != null && line.charAt(0) == '\t'; line = reader.readLine())
|
|
|
|
vec.addElement(line.substring(1, 11));
|
|
|
|
Object[] refs = new Object[vec.size()];
|
|
|
|
vec.copyInto(refs);
|
1999-10-12 22:59:52 +04:00
|
|
|
|
|
|
|
// read in stack crawl.
|
|
|
|
vec.setSize(0);
|
|
|
|
for (line = reader.readLine(); line != null && !line.equals("Leaked composite object at:"); line = reader.readLine())
|
|
|
|
vec.addElement(line.intern());
|
|
|
|
Object[] crawl = new Object[vec.size()];
|
|
|
|
vec.copyInto(crawl);
|
|
|
|
|
|
|
|
// record the leak.
|
1999-10-14 03:42:49 +04:00
|
|
|
table.put(addr, new Leak(addr, type, refs, crawl));
|
|
|
|
|
|
|
|
// count the leak types in a histogram.
|
1999-10-08 08:12:11 +04:00
|
|
|
hist.record(type);
|
1999-10-07 06:11:44 +04:00
|
|
|
} else {
|
|
|
|
line = reader.readLine();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
reader.close();
|
|
|
|
|
|
|
|
Leak[] leaks = new Leak[table.size()];
|
|
|
|
int leakCount = 0;
|
|
|
|
long totalSize = 0;
|
|
|
|
|
|
|
|
// now, we have a table full of leaked objects, lets derive reference counts, and build the graph.
|
|
|
|
Enumeration e = table.elements();
|
|
|
|
while (e.hasMoreElements()) {
|
|
|
|
Leak leak = (Leak) e.nextElement();
|
|
|
|
Object[] refs = leak.mReferences;
|
|
|
|
int count = refs.length;
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
String addr = (String) refs[i];
|
|
|
|
Leak ref = (Leak) table.get(addr);
|
|
|
|
if (ref != null) {
|
|
|
|
// increase the ref count.
|
|
|
|
ref.mRefCount++;
|
|
|
|
// change string to ref itself.
|
|
|
|
refs[i] = ref;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
leaks[leakCount++] = leak;
|
1999-10-14 03:42:49 +04:00
|
|
|
totalSize += leak.mType.mSize;
|
1999-10-07 06:11:44 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// be nice to the GC.
|
|
|
|
table.clear();
|
|
|
|
table = null;
|
|
|
|
|
1999-10-14 03:42:49 +04:00
|
|
|
// store the leak report in inputName + ".html"
|
|
|
|
PrintStream out = new PrintStream(new FileOutputStream(inputName + ".html"));
|
1999-10-15 19:56:05 +04:00
|
|
|
|
|
|
|
Date now = new Date();
|
|
|
|
out.println("<TITLE>Leaks as of " + now + "</TITLE>");
|
|
|
|
|
|
|
|
// print leak summary.
|
|
|
|
out.println("<H2>Leak Summary</H2>");
|
|
|
|
out.println("total objects leaked = " + leakCount + "<BR>");
|
|
|
|
out.println("total memory leaked = " + totalSize + " bytes.<BR>");
|
1999-10-07 06:11:44 +04:00
|
|
|
|
1999-10-08 08:12:11 +04:00
|
|
|
// print the object histogram report.
|
|
|
|
printHistogram(out, hist);
|
|
|
|
|
|
|
|
// print the leak report.
|
1999-10-15 19:56:05 +04:00
|
|
|
printLeaks(out, leaks);
|
1999-10-08 08:12:11 +04:00
|
|
|
|
|
|
|
out.close();
|
1999-10-07 06:11:44 +04:00
|
|
|
} catch (Exception e) {
|
|
|
|
e.printStackTrace(System.err);
|
|
|
|
}
|
|
|
|
}
|
1999-10-08 08:12:11 +04:00
|
|
|
|
1999-10-15 19:56:05 +04:00
|
|
|
/**
|
|
|
|
* Sorts the bins of a histogram by (count * typeSize) to show the
|
|
|
|
* most pressing leaks.
|
|
|
|
*/
|
|
|
|
static class HistComparator implements QuickSort.Comparator {
|
|
|
|
Histogram hist;
|
|
|
|
|
|
|
|
HistComparator(Histogram hist) {
|
|
|
|
this.hist = hist;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int compare(Object obj1, Object obj2) {
|
|
|
|
Type t1 = (Type) obj1, t2 = (Type) obj2;
|
1999-10-19 07:03:12 +04:00
|
|
|
return (hist.count(t1) * t1.mSize - hist.count(t2) * t2.mSize);
|
1999-10-15 19:56:05 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void printHistogram(PrintStream out, Histogram hist) throws IOException {
|
|
|
|
// sort the objects by histogram count.
|
|
|
|
Object[] objects = hist.objects();
|
|
|
|
QuickSort sorter = new QuickSort(new HistComparator(hist));
|
|
|
|
sorter.sort(objects);
|
|
|
|
|
|
|
|
out.println("<H2>Leak Histogram:</H2>");
|
|
|
|
out.println("<PRE>");
|
|
|
|
int count = objects.length;
|
|
|
|
while (count > 0) {
|
|
|
|
Object object = objects[--count];
|
|
|
|
out.println(object.toString() + " : " + hist.count(object));
|
|
|
|
}
|
|
|
|
out.println("</PRE>");
|
|
|
|
}
|
|
|
|
|
1999-10-12 22:59:52 +04:00
|
|
|
static final String MOZILLA_BASE = "mozilla/";
|
|
|
|
static final String LXR_BASE = "http://lxr.mozilla.org/seamonkey/source/";
|
1999-10-14 03:42:49 +04:00
|
|
|
static final String BONSAI_BASE = "http://cvs-mirror.mozilla.org/webtools/bonsai/cvsblame.cgi?file=mozilla/";
|
1999-10-12 22:59:52 +04:00
|
|
|
|
|
|
|
static String getFileLocation(Hashtable fileTables, String line) throws IOException {
|
|
|
|
int leftBracket = line.indexOf('[');
|
|
|
|
if (leftBracket == -1)
|
|
|
|
return line;
|
|
|
|
int comma = line.indexOf(',', leftBracket + 1);
|
|
|
|
int rightBracket = line.indexOf(']', leftBracket + 1);
|
|
|
|
String macPath = line.substring(leftBracket + 1, comma);
|
|
|
|
String path = macPath.replace(':', '/');
|
|
|
|
int mozillaIndex = path.indexOf(MOZILLA_BASE);
|
|
|
|
String locationURL;
|
|
|
|
if (mozillaIndex > -1)
|
1999-10-14 03:42:49 +04:00
|
|
|
locationURL = (USE_BLAME ? BONSAI_BASE : LXR_BASE) + path.substring(path.indexOf(MOZILLA_BASE) + MOZILLA_BASE.length());
|
1999-10-12 22:59:52 +04:00
|
|
|
else
|
|
|
|
locationURL = "file:///" + path;
|
|
|
|
int offset = 0;
|
|
|
|
try {
|
|
|
|
offset = Integer.parseInt(line.substring(comma + 1, rightBracket));
|
|
|
|
} catch (NumberFormatException nfe) {
|
|
|
|
return line;
|
|
|
|
}
|
|
|
|
FileTable table = (FileTable) fileTables.get(path);
|
|
|
|
if (table == null) {
|
|
|
|
table = new FileTable("/" + path);
|
|
|
|
fileTables.put(path, table);
|
|
|
|
}
|
1999-10-14 03:42:49 +04:00
|
|
|
int lineNumber = 1 + table.getLine(offset);
|
|
|
|
// return line.substring(0, leftBracket) + "[" + locationURL + "#" + lineNumber + "]";
|
|
|
|
return "<A HREF=\"" + locationURL + "#" + lineNumber + "\"TARGET=\"SOURCE\">" + line.substring(0, leftBracket) + "</A>";
|
1999-10-12 22:59:52 +04:00
|
|
|
}
|
|
|
|
|
1999-10-15 19:56:05 +04:00
|
|
|
static void printLeaks(PrintStream out, Leak[] leaks) throws IOException {
|
|
|
|
// sort the leaks by size.
|
|
|
|
QuickSort bySize = new QuickSort(new Leak.BySize());
|
|
|
|
bySize.sort(leaks);
|
|
|
|
|
|
|
|
out.println("<H2>Leak Roots</H2>");
|
1999-10-08 08:12:11 +04:00
|
|
|
|
1999-10-14 03:42:49 +04:00
|
|
|
out.println("<PRE>");
|
1999-10-08 08:12:11 +04:00
|
|
|
|
1999-10-15 19:56:05 +04:00
|
|
|
int leakCount = leaks.length;
|
|
|
|
for (int i = 0; i < leakCount; i++) {
|
|
|
|
Leak leak = leaks[i];
|
|
|
|
if (leak.mRefCount == 0)
|
|
|
|
out.println(leak);
|
|
|
|
}
|
|
|
|
|
1999-10-12 22:59:52 +04:00
|
|
|
Hashtable fileTables = new Hashtable();
|
1999-10-15 19:56:05 +04:00
|
|
|
Type anchorType = null;
|
1999-10-12 22:59:52 +04:00
|
|
|
|
1999-10-15 19:56:05 +04:00
|
|
|
// now, print the report, sorted by type size.
|
1999-10-08 08:12:11 +04:00
|
|
|
for (int i = 0; i < leakCount; i++) {
|
|
|
|
Leak leak = leaks[i];
|
1999-10-15 19:56:05 +04:00
|
|
|
if (anchorType != leak.mType) {
|
|
|
|
anchorType = leak.mType;
|
|
|
|
out.println("\n<HR>");
|
|
|
|
out.println("<A NAME=\"" + anchorType.mName + "_" + anchorType.mSize + "\"></A>");
|
|
|
|
out.println("<H3>" + anchorType + " Leaks</H3>");
|
|
|
|
}
|
1999-10-14 03:42:49 +04:00
|
|
|
out.println("<A NAME=\"" + leak.mAddress + "\"></A>");
|
1999-10-08 08:12:11 +04:00
|
|
|
out.println(leak);
|
1999-10-12 22:59:52 +04:00
|
|
|
// print object's fields:
|
1999-10-08 08:12:11 +04:00
|
|
|
Object[] refs = leak.mReferences;
|
|
|
|
int count = refs.length;
|
|
|
|
for (int j = 0; j < count; j++)
|
|
|
|
out.println("\t" + refs[j]);
|
1999-10-12 22:59:52 +04:00
|
|
|
// print object's stack crawl:
|
|
|
|
Object[] crawl = leak.mCrawl;
|
|
|
|
count = crawl.length;
|
|
|
|
for (int j = 0; j < count; j++) {
|
|
|
|
String location = getFileLocation(fileTables, (String) crawl[j]);
|
|
|
|
out.println(location);
|
|
|
|
}
|
1999-10-08 08:12:11 +04:00
|
|
|
}
|
1999-10-14 03:42:49 +04:00
|
|
|
|
1999-10-15 19:56:05 +04:00
|
|
|
fileTables.clear();
|
1999-10-08 08:12:11 +04:00
|
|
|
|
1999-10-14 03:42:49 +04:00
|
|
|
out.println("</PRE>");
|
1999-10-08 08:12:11 +04:00
|
|
|
}
|
1999-10-07 06:11:44 +04:00
|
|
|
}
|