| /* |
| * 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.ahat.heapdump.AhatHeap; |
| import com.android.ahat.heapdump.AhatSnapshot; |
| import com.android.ahat.heapdump.Diffable; |
| import java.util.ArrayList; |
| import java.util.HashMap; |
| import java.util.List; |
| import java.util.Map; |
| |
| /** |
| * Class for rendering a table that includes sizes of some kind for each heap. |
| */ |
| class HeapTable { |
| /** |
| * Configuration for a value column of a heap table. |
| */ |
| public interface ValueConfig<T> { |
| String getDescription(); |
| DocString render(T element); |
| } |
| |
| /** |
| * Configuration for the HeapTable. |
| */ |
| public interface TableConfig<T> { |
| String getHeapsDescription(); |
| long getSize(T element, AhatHeap heap); |
| List<ValueConfig<T>> getValueConfigs(); |
| } |
| |
| /** |
| * Render the table to the given document. |
| * @param query - The page query. |
| * @param id - A unique identifier for the table on the page. |
| */ |
| public static <T extends Diffable<T>> void render(Doc doc, Query query, String id, |
| TableConfig<T> config, AhatSnapshot snapshot, List<T> elements) { |
| // Only show the heaps that have non-zero entries. |
| List<AhatHeap> heaps = new ArrayList<AhatHeap>(); |
| for (AhatHeap heap : snapshot.getHeaps()) { |
| if (hasNonZeroEntry(heap, config, elements)) { |
| heaps.add(heap); |
| } |
| } |
| |
| List<ValueConfig<T>> values = config.getValueConfigs(); |
| |
| // Print the heap and values descriptions. |
| List<Column> subcols = new ArrayList<Column>(); |
| for (AhatHeap heap : heaps) { |
| subcols.add(new Column(heap.getName(), Column.Align.RIGHT)); |
| subcols.add(new Column("Δ", Column.Align.RIGHT, snapshot.isDiffed())); |
| } |
| boolean showTotal = heaps.size() > 1; |
| subcols.add(new Column("Total", Column.Align.RIGHT, showTotal)); |
| subcols.add(new Column("Δ", Column.Align.RIGHT, showTotal && snapshot.isDiffed())); |
| List<Column> cols = new ArrayList<Column>(); |
| for (ValueConfig value : values) { |
| cols.add(new Column(value.getDescription())); |
| } |
| doc.table(DocString.text(config.getHeapsDescription()), subcols, cols); |
| |
| // Print the entries up to the selected limit. |
| SubsetSelector<T> selector = new SubsetSelector(query, id, elements); |
| ArrayList<DocString> vals = new ArrayList<DocString>(); |
| for (T elem : selector.selected()) { |
| T base = elem.getBaseline(); |
| vals.clear(); |
| long total = 0; |
| long basetotal = 0; |
| for (AhatHeap heap : heaps) { |
| long size = config.getSize(elem, heap); |
| long basesize = config.getSize(base, heap.getBaseline()); |
| total += size; |
| basetotal += basesize; |
| vals.add(DocString.size(size, elem.isPlaceHolder())); |
| vals.add(DocString.delta(elem.isPlaceHolder(), base.isPlaceHolder(), size, basesize)); |
| } |
| vals.add(DocString.size(total, elem.isPlaceHolder())); |
| vals.add(DocString.delta(elem.isPlaceHolder(), base.isPlaceHolder(), total, basetotal)); |
| |
| for (ValueConfig<T> value : values) { |
| vals.add(value.render(elem)); |
| } |
| doc.row(vals.toArray(new DocString[0])); |
| } |
| |
| // Print a summary of the remaining entries if there are any. |
| List<T> remaining = selector.remaining(); |
| if (!remaining.isEmpty()) { |
| Map<AhatHeap, Long> summary = new HashMap<AhatHeap, Long>(); |
| Map<AhatHeap, Long> basesummary = new HashMap<AhatHeap, Long>(); |
| for (AhatHeap heap : heaps) { |
| summary.put(heap, 0L); |
| basesummary.put(heap, 0L); |
| } |
| |
| for (T elem : remaining) { |
| for (AhatHeap heap : heaps) { |
| long size = config.getSize(elem, heap); |
| summary.put(heap, summary.get(heap) + size); |
| |
| long basesize = config.getSize(elem.getBaseline(), heap.getBaseline()); |
| basesummary.put(heap, basesummary.get(heap) + basesize); |
| } |
| } |
| |
| vals.clear(); |
| long total = 0; |
| long basetotal = 0; |
| for (AhatHeap heap : heaps) { |
| long size = summary.get(heap); |
| long basesize = basesummary.get(heap); |
| total += size; |
| basetotal += basesize; |
| vals.add(DocString.size(size, false)); |
| vals.add(DocString.delta(false, false, size, basesize)); |
| } |
| vals.add(DocString.size(total, false)); |
| vals.add(DocString.delta(false, false, total, basetotal)); |
| |
| for (ValueConfig<T> value : values) { |
| vals.add(DocString.text("...")); |
| } |
| doc.row(vals.toArray(new DocString[0])); |
| } |
| doc.end(); |
| selector.render(doc); |
| } |
| |
| // Returns true if the given heap has a non-zero size entry. |
| public static <T extends Diffable<T>> boolean hasNonZeroEntry(AhatHeap heap, |
| TableConfig<T> config, List<T> elements) { |
| AhatHeap baseheap = heap.getBaseline(); |
| if (!heap.getSize().isZero() || !baseheap.getSize().isZero()) { |
| for (T element : elements) { |
| if (config.getSize(element, heap) > 0 || |
| config.getSize(element.getBaseline(), baseheap) > 0) { |
| return true; |
| } |
| } |
| } |
| return false; |
| } |
| } |
| |