blob: 94934a2831f841ca5f9ada2ee45ab1c56886c9ea [file] [log] [blame]
/*
* 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.
*/
package com.android.ahat;
import com.android.tools.perflib.heap.ArrayInstance;
import com.android.tools.perflib.heap.ClassInstance;
import com.android.tools.perflib.heap.ClassObj;
import com.android.tools.perflib.heap.Field;
import com.android.tools.perflib.heap.Heap;
import com.android.tools.perflib.heap.Instance;
import com.android.tools.perflib.heap.RootObj;
import com.android.tools.perflib.heap.Type;
import java.awt.image.BufferedImage;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Map;
/**
* Utilities for extracting information from hprof instances.
*/
class InstanceUtils {
/**
* Returns true if the given instance is an instance of a class with the
* given name.
*/
private static boolean isInstanceOfClass(Instance inst, String className) {
ClassObj cls = (inst == null) ? null : inst.getClassObj();
return (cls != null && className.equals(cls.getClassName()));
}
/**
* Read the byte[] value from an hprof Instance.
* Returns null if the instance is not a byte array.
*/
private static byte[] asByteArray(Instance inst) {
if (!(inst instanceof ArrayInstance)) {
return null;
}
ArrayInstance array = (ArrayInstance) inst;
if (array.getArrayType() != Type.BYTE) {
return null;
}
Object[] objs = array.getValues();
byte[] bytes = new byte[objs.length];
for (int i = 0; i < objs.length; i++) {
Byte b = (Byte) objs[i];
bytes[i] = b.byteValue();
}
return bytes;
}
/**
* Read the string value from an hprof Instance.
* Returns null if the object can't be interpreted as a string.
*/
public static String asString(Instance inst) {
return asString(inst, -1);
}
/**
* Read the string value from an hprof Instance.
* Returns null if the object can't be interpreted as a string.
* The returned string is truncated to maxChars characters.
* If maxChars is negative, the returned string is not truncated.
*/
public static String asString(Instance inst, int maxChars) {
// The inst object could either be a java.lang.String or a char[]. If it
// is a char[], use that directly as the value, otherwise use the value
// field of the string object. The field accesses for count and offset
// later on will work okay regardless of what type the inst object is.
Object value = inst;
if (isInstanceOfClass(inst, "java.lang.String")) {
value = getField(inst, "value");
}
if (!(value instanceof ArrayInstance)) {
return null;
}
ArrayInstance chars = (ArrayInstance) value;
if (chars.getArrayType() != Type.CHAR) {
return null;
}
int numChars = chars.getLength();
int count = getIntField(inst, "count", numChars);
if (count == 0) {
return "";
}
if (0 <= maxChars && maxChars < count) {
count = maxChars;
}
int offset = getIntField(inst, "offset", 0);
int end = offset + count - 1;
if (offset >= 0 && offset < numChars && end >= 0 && end < numChars) {
return new String(chars.asCharArray(offset, count));
}
return null;
}
/**
* Read the bitmap data for the given android.graphics.Bitmap object.
* Returns null if the object isn't for android.graphics.Bitmap or the
* bitmap data couldn't be read.
*/
public static BufferedImage asBitmap(Instance inst) {
if (!isInstanceOfClass(inst, "android.graphics.Bitmap")) {
return null;
}
Integer width = getIntField(inst, "mWidth", null);
if (width == null) {
return null;
}
Integer height = getIntField(inst, "mHeight", null);
if (height == null) {
return null;
}
byte[] buffer = getByteArrayField(inst, "mBuffer");
if (buffer == null) {
return null;
}
// Convert the raw data to an image
// Convert BGRA to ABGR
int[] abgr = new int[height * width];
for (int i = 0; i < abgr.length; i++) {
abgr[i] = (
(((int) buffer[i * 4 + 3] & 0xFF) << 24)
+ (((int) buffer[i * 4 + 0] & 0xFF) << 16)
+ (((int) buffer[i * 4 + 1] & 0xFF) << 8)
+ ((int) buffer[i * 4 + 2] & 0xFF));
}
BufferedImage bitmap = new BufferedImage(
width, height, BufferedImage.TYPE_4BYTE_ABGR);
bitmap.setRGB(0, 0, width, height, abgr, 0, width);
return bitmap;
}
/**
* Read a field of an instance.
* Returns null if the field value is null or if the field couldn't be read.
*/
public static Object getField(Instance inst, String fieldName) {
if (!(inst instanceof ClassInstance)) {
return null;
}
ClassInstance clsinst = (ClassInstance) inst;
Object value = null;
int count = 0;
for (ClassInstance.FieldValue field : clsinst.getValues()) {
if (fieldName.equals(field.getField().getName())) {
value = field.getValue();
count++;
}
}
return count == 1 ? value : null;
}
/**
* Read a reference field of an instance.
* Returns null if the field value is null, or if the field couldn't be read.
*/
public static Instance getRefField(Instance inst, String fieldName) {
Object value = getField(inst, fieldName);
if (!(value instanceof Instance)) {
return null;
}
return (Instance) value;
}
/**
* Read an int field of an instance.
* The field is assumed to be an int type.
* Returns <code>def</code> if the field value is not an int or could not be
* read.
*/
private static Integer getIntField(Instance inst, String fieldName, Integer def) {
Object value = getField(inst, fieldName);
if (!(value instanceof Integer)) {
return def;
}
return (Integer) value;
}
/**
* Read a long field of an instance.
* The field is assumed to be a long type.
* Returns <code>def</code> if the field value is not an long or could not
* be read.
*/
private static Long getLongField(Instance inst, String fieldName, Long def) {
Object value = getField(inst, fieldName);
if (!(value instanceof Long)) {
return def;
}
return (Long) value;
}
/**
* Read the given field from the given instance.
* The field is assumed to be a byte[] field.
* Returns null if the field value is null, not a byte[] or could not be read.
*/
private static byte[] getByteArrayField(Instance inst, String fieldName) {
Object value = getField(inst, fieldName);
if (!(value instanceof Instance)) {
return null;
}
return asByteArray((Instance) value);
}
// Return the bitmap instance associated with this object, or null if there
// is none. This works for android.graphics.Bitmap instances and their
// underlying Byte[] instances.
public static Instance getAssociatedBitmapInstance(Instance inst) {
ClassObj cls = inst.getClassObj();
if (cls == null) {
return null;
}
if ("android.graphics.Bitmap".equals(cls.getClassName())) {
return inst;
}
if (inst instanceof ArrayInstance) {
ArrayInstance array = (ArrayInstance) inst;
if (array.getArrayType() == Type.BYTE && inst.getHardReverseReferences().size() == 1) {
Instance ref = inst.getHardReverseReferences().get(0);
ClassObj clsref = ref.getClassObj();
if (clsref != null && "android.graphics.Bitmap".equals(clsref.getClassName())) {
return ref;
}
}
}
return null;
}
private static boolean isJavaLangRefReference(Instance inst) {
ClassObj cls = (inst == null) ? null : inst.getClassObj();
while (cls != null) {
if ("java.lang.ref.Reference".equals(cls.getClassName())) {
return true;
}
cls = cls.getSuperClassObj();
}
return false;
}
public static Instance getReferent(Instance inst) {
if (isJavaLangRefReference(inst)) {
return getRefField(inst, "referent");
}
return null;
}
/**
* Assuming inst represents a DexCache object, return the dex location for
* that dex cache. Returns null if the given instance doesn't represent a
* DexCache object or the location could not be found.
* If maxChars is non-negative, the returned location is truncated to
* maxChars in length.
*/
public static String getDexCacheLocation(Instance inst, int maxChars) {
if (isInstanceOfClass(inst, "java.lang.DexCache")) {
Instance location = getRefField(inst, "location");
if (location != null) {
return asString(location, maxChars);
}
}
return null;
}
public static class NativeAllocation {
public long size;
public Heap heap;
public long pointer;
public Instance referent;
public NativeAllocation(long size, Heap heap, long pointer, Instance referent) {
this.size = size;
this.heap = heap;
this.pointer = pointer;
this.referent = referent;
}
}
/**
* Assuming inst represents a NativeAllocation, return information about the
* native allocation. Returns null if the given instance doesn't represent a
* native allocation.
*/
public static NativeAllocation getNativeAllocation(Instance inst) {
if (!isInstanceOfClass(inst, "libcore.util.NativeAllocationRegistry$CleanerThunk")) {
return null;
}
Long pointer = InstanceUtils.getLongField(inst, "nativePtr", null);
if (pointer == null) {
return null;
}
// Search for the registry field of inst.
// Note: We know inst as an instance of ClassInstance because we already
// read the nativePtr field from it.
Instance registry = null;
for (ClassInstance.FieldValue field : ((ClassInstance) inst).getValues()) {
Object fieldValue = field.getValue();
if (fieldValue instanceof Instance) {
Instance fieldInst = (Instance) fieldValue;
if (isInstanceOfClass(fieldInst, "libcore.util.NativeAllocationRegistry")) {
registry = fieldInst;
break;
}
}
}
if (registry == null) {
return null;
}
Long size = InstanceUtils.getLongField(registry, "size", null);
if (size == null) {
return null;
}
Instance referent = null;
for (Instance ref : inst.getHardReverseReferences()) {
if (isInstanceOfClass(ref, "sun.misc.Cleaner")) {
referent = InstanceUtils.getReferent(ref);
if (referent != null) {
break;
}
}
}
if (referent == null) {
return null;
}
return new NativeAllocation(size, inst.getHeap(), pointer, referent);
}
public static class PathElement {
public final Instance instance;
public final String field;
public boolean isDominator;
public PathElement(Instance instance, String field) {
this.instance = instance;
this.field = field;
this.isDominator = false;
}
}
/**
* Returns a sample path from a GC root to this instance.
* The given instance is included as the last element of the path with an
* empty field description.
*/
public static List<PathElement> getPathFromGcRoot(Instance inst) {
List<PathElement> path = new ArrayList<PathElement>();
Instance dom = inst;
for (PathElement elem = new PathElement(inst, ""); elem != null;
elem = getNextPathElementToGcRoot(elem.instance)) {
if (elem.instance == dom) {
elem.isDominator = true;
dom = dom.getImmediateDominator();
}
path.add(elem);
}
Collections.reverse(path);
return path;
}
/**
* Returns the next instance to GC root from this object and a string
* description of which field of that object refers to the given instance.
* Returns null if the given instance has no next instance to the gc root.
*/
private static PathElement getNextPathElementToGcRoot(Instance inst) {
Instance parent = inst.getNextInstanceToGcRoot();
if (parent == null || parent instanceof RootObj) {
return null;
}
// Search the parent for the reference to the child.
// TODO: This seems terribly inefficient. Can we use data structures to
// help us here?
String description = ".???";
if (parent instanceof ArrayInstance) {
ArrayInstance array = (ArrayInstance)parent;
Object[] values = array.getValues();
for (int i = 0; i < values.length; i++) {
if (values[i] instanceof Instance) {
Instance ref = (Instance)values[i];
if (ref.getId() == inst.getId()) {
description = String.format("[%d]", i);
break;
}
}
}
} else if (parent instanceof ClassObj) {
ClassObj cls = (ClassObj)parent;
for (Map.Entry<Field, Object> entries : cls.getStaticFieldValues().entrySet()) {
if (entries.getValue() instanceof Instance) {
Instance ref = (Instance)entries.getValue();
if (ref.getId() == inst.getId()) {
description = "." + entries.getKey().getName();
break;
}
}
}
} else if (parent instanceof ClassInstance) {
ClassInstance obj = (ClassInstance)parent;
for (ClassInstance.FieldValue fields : obj.getValues()) {
if (fields.getValue() instanceof Instance) {
Instance ref = (Instance)fields.getValue();
if (ref.getId() == inst.getId()) {
description = "." + fields.getField().getName();
break;
}
}
}
}
return new PathElement(parent, description);
}
}