/* * Copyright (C) 2015 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import dalvik.system.VMDebug; import java.io.IOException; import java.lang.ref.PhantomReference; import java.lang.ref.ReferenceQueue; import java.lang.ref.SoftReference; import java.lang.ref.WeakReference; import org.apache.harmony.dalvik.ddmc.DdmVmInternal; /** * Program used to create a heap dump for test purposes. */ public class Main { // Keep a reference to the DumpedStuff instance so that it is not garbage // collected before we take the heap dump. public static DumpedStuff stuff; public static class ObjectTree { public ObjectTree left; public ObjectTree right; public ObjectTree(ObjectTree left, ObjectTree right) { this.left = left; this.right = right; } } public static class AddedObject { } public static class RemovedObject { } public static class UnchangedObject { } public static class ModifiedObject { public int value; public String modifiedRefField; public String unmodifiedRefField; } public static class StackSmasher { public StackSmasher child; } // We will take a heap dump that includes a single instance of this // DumpedStuff class. Objects stored as fields in this class can be easily // found in the hprof dump by searching for the instance of the DumpedStuff // class and reading the desired field. public static class DumpedStuff { public String basicString = "hello, world"; public String nonAscii = "Sigma (Ʃ) is not ASCII"; public String embeddedZero = "embedded\0..."; // Non-ASCII for string compression purposes. public char[] charArray = "char thing".toCharArray(); public String nullString = null; public Object anObject = new Object(); public ReferenceQueue<Object> referenceQueue = new ReferenceQueue<Object>(); public PhantomReference aPhantomReference = new PhantomReference(anObject, referenceQueue); public WeakReference aWeakReference = new WeakReference(anObject, referenceQueue); public WeakReference aNullReferentReference = new WeakReference(null, referenceQueue); public SoftReference aSoftReference = new SoftReference(new Object()); public byte[] bigArray; public ObjectTree[] gcPathArray = new ObjectTree[]{null, null, new ObjectTree( new ObjectTree(null, new ObjectTree(null, null)), new ObjectTree(null, null)), null}; public Object[] basicStringRef; public AddedObject addedObject; public UnchangedObject unchangedObject = new UnchangedObject(); public RemovedObject removedObject; public ModifiedObject modifiedObject; public StackSmasher stackSmasher; public StackSmasher stackSmasherAdded; public static String modifiedStaticField; public int[] modifiedArray; DumpedStuff(boolean baseline) { int N = baseline ? 400000 : 1000000; bigArray = new byte[N]; for (int i = 0; i < N; i++) { bigArray[i] = (byte)((i*i) & 0xFF); } addedObject = baseline ? null : new AddedObject(); removedObject = baseline ? new RemovedObject() : null; modifiedObject = new ModifiedObject(); modifiedObject.value = baseline ? 5 : 8; modifiedObject.modifiedRefField = baseline ? "A1" : "A2"; modifiedObject.unmodifiedRefField = "B"; modifiedStaticField = baseline ? "C1" : "C2"; modifiedArray = baseline ? new int[]{0,1,2,3} : new int[]{3,1,2,0}; // Deep matching dominator trees shouldn't smash the stack when we try // to diff them. Make some deep dominator trees to help test it. for (int i = 0; i < 10000; i++) { StackSmasher smasher = new StackSmasher(); smasher.child = stackSmasher; stackSmasher = smasher; if (!baseline) { smasher = new StackSmasher(); smasher.child = stackSmasherAdded; stackSmasherAdded = smasher; } } gcPathArray[2].right.left = gcPathArray[2].left.right; } } public static void main(String[] args) throws IOException { if (args.length < 1) { System.err.println("no output file specified"); return; } String file = args[0]; // If a --base argument is provided, it means we should generate a // baseline hprof file suitable for using in testing diff. boolean baseline = args.length > 1 && args[1].equals("--base"); // Enable allocation tracking so we get stack traces in the heap dump. DdmVmInternal.enableRecentAllocations(true); // Allocate the instance of DumpedStuff. stuff = new DumpedStuff(baseline); // Create a bunch of unreachable objects pointing to basicString for the // reverseReferencesAreNotUnreachable test for (int i = 0; i < 100; i++) { stuff.basicStringRef = new Object[]{stuff.basicString}; } // Take a heap dump that will include that instance of DumpedStuff. System.err.println("Dumping hprof data to " + file); VMDebug.dumpHprofData(file); } }