/*
* 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.Heap;
import java.io.IOException;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;
class SiteHandler implements AhatHandler {
private static final String ALLOCATION_SITE_ID = "frames";
private static final String SITES_CALLED_ID = "called";
private static final String OBJECTS_ALLOCATED_ID = "objects";
private AhatSnapshot mSnapshot;
public SiteHandler(AhatSnapshot snapshot) {
mSnapshot = snapshot;
}
@Override
public void handle(Doc doc, Query query) throws IOException {
int stackId = query.getInt("stack", 0);
int depth = query.getInt("depth", -1);
Site site = mSnapshot.getSite(stackId, depth);
doc.title("Site %s", site.getName());
doc.section("Allocation Site");
SitePrinter.printSite(mSnapshot, doc, query, ALLOCATION_SITE_ID, site);
doc.section("Sites Called from Here");
List<Site> children = site.getChildren();
if (children.isEmpty()) {
doc.println(DocString.text("(none)"));
} else {
Collections.sort(children, new Sort.SiteBySize("app"));
HeapTable.TableConfig<Site> table = new HeapTable.TableConfig<Site>() {
public String getHeapsDescription() {
return "Reachable Bytes Allocated on Heap";
}
public long getSize(Site element, Heap heap) {
return element.getSize(heap.getName());
}
public List<HeapTable.ValueConfig<Site>> getValueConfigs() {
HeapTable.ValueConfig<Site> value = new HeapTable.ValueConfig<Site>() {
public String getDescription() {
return "Child Site";
}
public DocString render(Site element) {
return DocString.link(
DocString.formattedUri("site?stack=%d&depth=%d",
element.getStackId(), element.getStackDepth()),
DocString.text(element.getName()));
}
};
return Collections.singletonList(value);
}
};
HeapTable.render(doc, query, SITES_CALLED_ID, table, mSnapshot, children);
}
doc.section("Objects Allocated");
doc.table(
new Column("Reachable Bytes Allocated", Column.Align.RIGHT),
new Column("Instances", Column.Align.RIGHT),
new Column("Heap"),
new Column("Class"));
List<Site.ObjectsInfo> infos = site.getObjectsInfos();
Comparator<Site.ObjectsInfo> compare = new Sort.WithPriority<Site.ObjectsInfo>(
new Sort.ObjectsInfoByHeapName(),
new Sort.ObjectsInfoBySize(),
new Sort.ObjectsInfoByClassName());
Collections.sort(infos, compare);
SubsetSelector<Site.ObjectsInfo> selector
= new SubsetSelector(query, OBJECTS_ALLOCATED_ID, infos);
for (Site.ObjectsInfo info : selector.selected()) {
String className = AhatSnapshot.getClassName(info.classObj);
doc.row(
DocString.format("%,14d", info.numBytes),
DocString.link(
DocString.formattedUri("objects?stack=%d&depth=%d&heap=%s&class=%s",
site.getStackId(), site.getStackDepth(), info.heap.getName(), className),
DocString.format("%,14d", info.numInstances)),
DocString.text(info.heap.getName()),
Value.render(mSnapshot, info.classObj));
}
doc.end();
selector.render(doc);
}
}