blob: 7dde86778d39957a47f9090dcae120e85a49a79d [file] [log] [blame]
// Copyright (c) 2020, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
/// This library contains utilities for reading and analyzing snapshot profiles
/// produced by `--write-v8-snapshot-profile-to` VM flag.
library vm_snapshot_analysis.v8_profile;
import 'package:collection/collection.dart';
import 'package:vm_snapshot_analysis/src/dominators.dart' as dominators;
import 'package:vm_snapshot_analysis/name.dart';
import 'package:vm_snapshot_analysis/program_info.dart';
/// This class represents snapshot graph.
///
/// Note that we do not eagerly deserialize the graph, instead we provide helper
/// methods and wrapper objects to work with serialized representation.
class Snapshot {
final Meta meta;
final int nodeCount;
final int edgeCount;
/// Serialized flat representation of nodes in the graph. Each node occupies
/// [meta.nodeFieldCount] consecutive elements of the list.
final List _nodes;
/// Serialized flat representation of edges between nodes. Each edge occupies
/// [meta.edgeFieldCount] consecutive elements of the list. All outgoing edges
/// for a node are serialized consecutively, number of outgoing edges is given
/// by the value at index [meta.nodeEdgeCountIndex] inside the node.
final List _edges;
/// Auxiliary array which gives starting index of edges (in the [_edges] list)
/// for the given node index.
final List<int> _edgesStartIndexForNode;
late final List<int> _dominators = _computeDominators(this);
final List strings;
Snapshot._(this.meta, this.nodeCount, this.edgeCount, this._nodes,
this._edges, this.strings, this._edgesStartIndexForNode);
/// Return node with the given index.
Node nodeAt(int index) {
assert(index >= 0, 'Node index should be positive: $index');
return Node._(snapshot: this, index: index);
}
/// Return all nodes in the snapshot.
Iterable<Node> get nodes => Iterable.generate(nodeCount, nodeAt);
/// Return dominator node for the given node [n].
Node dominatorOf(Node n) {
return nodeAt(_dominators[n.index]);
}
/// Returns true if the given JSON object is likely to be a serialized
/// snapshot using V8 heap snapshot format.
static bool isV8HeapSnapshot(Object m) =>
m is Map<String, dynamic> && m.containsKey('snapshot');
/// Construct [Snapshot] object from the given JSON object.
factory Snapshot.fromJson(Map<String, dynamic> m) {
// Extract meta information first.
final meta = Meta._fromJson(m['snapshot']['meta']);
final nodes = (m['nodes'] as List<dynamic>).cast<int>();
// Build an array of starting indexes of edges for each node.
final edgesStartIndexForNode = <int>[0];
int nextStartIndex = 0;
for (var i = meta.nodeEdgeCountIndex;
i < nodes.length;
i += meta.nodeFieldCount) {
nextStartIndex += nodes[i];
edgesStartIndexForNode.add(nextStartIndex);
}
return Snapshot._(
meta,
m['snapshot']['node_count'],
m['snapshot']['edge_count'],
m['nodes'],
m['edges'],
m['strings'],
edgesStartIndexForNode);
}
@override
String toString() {
final buffer = StringBuffer();
buffer
..write("Node count: ")
..writeln(nodeCount)
..write("Edge count: ")
..writeln(edgeCount);
buffer.write("Nodes:");
for (final node in nodes) {
buffer
..writeln()
..write(node.index)
..write(': ')
..writeln(node);
}
return buffer.toString();
}
}
/// Meta-information about the serialized snapshot.
///
/// Describes the structure of serialized nodes and edges by giving indexes of
/// the various fields.
class Meta {
final int nodeTypeIndex;
final int nodeNameIndex;
final int nodeIdIndex;
final int nodeSelfSizeIndex;
final int nodeEdgeCountIndex;
final int nodeFieldCount;
final int edgeTypeIndex;
final int edgeNameOrIndexIndex;
final int edgeToNodeIndex;
final int edgeFieldCount;
final List<String> nodeTypes;
final List<String> edgeTypes;
Meta._(
{required this.nodeTypeIndex,
required this.nodeNameIndex,
required this.nodeIdIndex,
required this.nodeSelfSizeIndex,
required this.nodeEdgeCountIndex,
required this.nodeFieldCount,
required this.edgeTypeIndex,
required this.edgeNameOrIndexIndex,
required this.edgeToNodeIndex,
required this.edgeFieldCount,
required this.nodeTypes,
required this.edgeTypes});
factory Meta._fromJson(Map<String, dynamic> m) {
final nodeFields = m['node_fields'];
final nodeTypes = m['node_types'].first.cast<String>();
final edgeFields = m['edge_fields'];
final edgeTypes = m['edge_types'].first.cast<String>();
return Meta._(
nodeTypeIndex: nodeFields.indexOf('type'),
nodeNameIndex: nodeFields.indexOf('name'),
nodeIdIndex: nodeFields.indexOf('id'),
nodeSelfSizeIndex: nodeFields.indexOf('self_size'),
nodeEdgeCountIndex: nodeFields.indexOf('edge_count'),
nodeFieldCount: nodeFields.length,
edgeTypeIndex: edgeFields.indexOf('type'),
edgeNameOrIndexIndex: edgeFields.indexOf('name_or_index'),
edgeToNodeIndex: edgeFields.indexOf('to_node'),
edgeFieldCount: edgeFields.length,
nodeTypes: nodeTypes,
edgeTypes: edgeTypes);
}
}
/// Edge from [Node] to [Node] in the [Snapshot] graph.
class Edge {
final Snapshot snapshot;
/// Index of this [Edge] within the [snapshot].
final int index;
Edge._({required this.snapshot, required this.index});
String get type => snapshot
.meta.edgeTypes[snapshot._edges[_offset + snapshot.meta.edgeTypeIndex]];
Node get target {
return Node._(
snapshot: snapshot,
index: snapshot._edges[_offset + snapshot.meta.edgeToNodeIndex] ~/
snapshot.meta.nodeFieldCount);
}
String get name {
final nameOrIndex =
snapshot._edges[_offset + snapshot.meta.edgeNameOrIndexIndex];
return type == 'property' ? snapshot.strings[nameOrIndex] : '@$nameOrIndex';
}
@override
String toString() {
final nameOrIndex =
snapshot._edges[_offset + snapshot.meta.edgeNameOrIndexIndex];
return {
'type': type,
'nameOrIndex':
type == 'property' ? snapshot.strings[nameOrIndex] : nameOrIndex,
'toNode': target.index,
}.toString();
}
/// Offset into [Snapshot._edges] list at which this edge begins.
int get _offset => index * snapshot.meta.edgeFieldCount;
}
/// Node in the [Snapshot] graph.
class Node {
final Snapshot snapshot;
/// Index of this [Node] within the [snapshot].
final int index;
Node._({required this.snapshot, required this.index});
int get edgeCount =>
snapshot._nodes[_offset + snapshot.meta.nodeEdgeCountIndex];
String get type => snapshot
.meta.nodeTypes[snapshot._nodes[_offset + snapshot.meta.nodeTypeIndex]];
String get name =>
snapshot.strings[snapshot._nodes[_offset + snapshot.meta.nodeNameIndex]];
int get selfSize =>
snapshot._nodes[_offset + snapshot.meta.nodeSelfSizeIndex];
int get id => snapshot._nodes[_offset + snapshot.meta.nodeIdIndex];
/// Returns all outgoing edges for this node.
Iterable<Edge> get edges sync* {
var firstEdgeIndex = snapshot._edgesStartIndexForNode[index];
for (var i = 0, n = edgeCount; i < n; i++) {
yield Edge._(snapshot: snapshot, index: firstEdgeIndex + i);
}
}
@override
String toString() {
return {
'type': type,
'name': name,
'id': id,
'selfSize': selfSize,
'edges': edges.toList(),
}.toString();
}
/// Returns the target of an outgoing edge with the given name (if any).
Node? operator [](String edgeName) =>
this.edges.firstWhereOrNull((e) => e.name == edgeName)?.target;
@override
bool operator ==(Object other) {
return other is Node && other.index == index;
}
@override
int get hashCode => this.index.hashCode;
/// Offset into [Snapshot._nodes] list at which this node begins.
int get _offset => index * snapshot.meta.nodeFieldCount;
}
/// Class representing information about V8 snapshot profile in relation
/// to a [ProgramInfo] structure that was derived from it.
class SnapshotInfo {
final Snapshot snapshot;
final List<ProgramInfoNode> infoNodes;
final Map<int, int> _ownerOf;
SnapshotInfo._(this.snapshot, this.infoNodes, this._ownerOf);
ProgramInfoNode ownerOf(Node node) =>
infoNodes[_ownerOf[node.index] ?? ProgramInfo.unknownId];
}
ProgramInfo toProgramInfo(Snapshot snap,
{bool collapseAnonymousClosures = false}) {
return _ProgramInfoBuilder(
collapseAnonymousClosures: collapseAnonymousClosures)
.build(snap);
}
class _ProgramInfoBuilder {
final bool collapseAnonymousClosures;
final program = ProgramInfo();
final List<ProgramInfoNode> infoNodes = [];
/// Mapping between snapshot [Node] index and id of [ProgramInfoNode] which
/// own this node.
final Map<int, int> ownerOf = {};
/// Mapping between snapshot [Node] indices and corresponding
/// [ProgramInfoNode] objects. Note that multiple snapshot nodes might be
/// mapped to a single [ProgramInfoNode] (e.g. when anonymous closures are
/// collapsed).
final Map<int, ProgramInfoNode> infoNodeByIndex = {};
// Mapping between package names and corresponding [ProgramInfoNode] objects
// representing those packages.
final Map<String, ProgramInfoNode> infoNodeForPackage = {};
/// Owners of some [Node] are determined by the program structure and not
/// by their reachability through the graph. For example, an owner of a
/// function is a class that contains it, even though the function can
/// also be reachable from another function through object pool.
final Set<int> nodesWithFrozenOwner = {};
/// Cache used to optimize common ancestor operation on [ProgramInfoNode] ids.
/// See [findCommonAncestor] method.
final Map<int, int> commonAncestorCache = {};
_ProgramInfoBuilder({required this.collapseAnonymousClosures});
/// Recover [ProgramInfo] structure from the snapshot profile.
///
/// This is done via a simple graph traversal: first all nodes representing
/// objects with clear ownership (like libraries, classes, functions) are
/// discovered and corresponding [ProgramInfoNode] objects are created for
/// them. Then the rest of the snapshot is attributed to one of these nodes
/// based on reachability (ignoring reachability from normal snapshot roots):
/// let `R(n)` be a set of [ProgramInfoNode] objects from which a given
/// snapshot node `n` is reachable. Then we define an owner of `n` to be
/// a lowest common ancestor of all nodes in `R(n)`.
///
/// Nodes which are not reachable from any normal [ProgramInfoNode] are
/// attributed to special `@unknown` [ProgramInfoNode].
ProgramInfo build(Snapshot snap) {
infoNodes.add(program.root);
infoNodes.add(program.stubs);
infoNodes.add(program.unknown);
// Create ProgramInfoNode for every snapshot node representing an element
// of the program structure (e.g. a library, a class, a function).
snap.nodes.forEach(getInfoNodeFor);
// Propagate the ownership information across the edges.
final worklist = ownerOf.keys.toList();
while (worklist.isNotEmpty) {
final node = snap.nodeAt(worklist.removeLast());
final sourceOwner = ownerOf[node.index];
for (var e in node.edges) {
final target = e.target;
if (!nodesWithFrozenOwner.contains(target.index)) {
final targetOwner = ownerOf[target.index];
final updatedOwner = findCommonAncestor(sourceOwner, targetOwner);
if (updatedOwner != targetOwner) {
ownerOf[target.index] = updatedOwner;
worklist.add(target.index);
}
}
}
}
// Now attribute sizes from the snapshot to nodes that own them.
for (var node in snap.nodes) {
if (node.selfSize > 0) {
final owner = infoNodes[ownerOf[node.index] ?? ProgramInfo.unknownId];
owner.size = (owner.size ?? 0) + node.selfSize;
}
}
program.snapshotInfo = SnapshotInfo._(snap, infoNodes, ownerOf);
return program;
}
ProgramInfoNode? getInfoNodeFor(Node node) {
var info = infoNodeByIndex[node.index];
if (info == null) {
info = createInfoNodeFor(node);
if (info != null) {
// Snapshot nodes which represent the program structure can't change
// their owner during iteration - their owner is frozen and is given
// by the program structure.
// Note that [ProgramInfoNode] owns its corresponding [Snapshot] node
// because we want the size of the snapshot node to be attributed to
// the info node itself.
nodesWithFrozenOwner.add(node.index);
ownerOf[node.index] = info.id;
// Handle some nodes specially.
switch (node.type) {
case 'Code':
// Freeze ownership of the Instructions object.
final instructions = node['<instructions>']!;
nodesWithFrozenOwner.add(instructions.index);
ownerOf[instructions.index] =
findCommonAncestor(ownerOf[instructions.index], info.id);
break;
case 'Library':
// Freeze ownership of the Script objects owned by this library.
final scripts = node['owned_scripts_'];
if (scripts != null) {
for (var e in scripts.edges) {
if (e.target.type == 'Script') {
nodesWithFrozenOwner.add(e.target.index);
ownerOf[e.target.index] =
findCommonAncestor(ownerOf[e.target.index]!, info.id);
}
}
}
break;
}
}
}
return info;
}
ProgramInfoNode? createInfoNodeFor(Node node) {
switch (node.type) {
case 'Code':
final owner = node['owner_']!;
if (owner.type != 'Type') {
final ownerNode =
owner.type == 'Null' ? program.stubs : getInfoNodeFor(owner)!;
if (owner.type == 'Function') {
// For normal functions we just attribute Code object and all
// objects dominated by it to the function itself.
return ownerNode;
}
// For stubs we create a dummy functionNode that is going to own
// all objects dominated by it.
return makeInfoNode(node.index,
name: node.name, parent: ownerNode, type: NodeType.functionNode);
}
break;
case 'Function':
if (node.name != '<anonymous signature>') {
var owner = node['owner_']!;
// Artificial nodes may not have a data_ field.
var data = node['data_'];
if (data != null && data.type == 'ClosureData') {
owner = data['parent_function_']!;
}
return makeInfoNode(node.index,
name: node.name,
parent: getInfoNodeFor(owner)!,
type: NodeType.functionNode);
}
break;
case 'PatchClass':
return getInfoNodeFor(node['patched_class_']!);
case 'Class':
// Default to root node. Some builtin classes (void, dynamic) don't have
// any information about their library written out.
var ownerNode = program.root;
if (node['library_'] != null) {
ownerNode = getInfoNodeFor(node['library_']!) ?? ownerNode;
}
return makeInfoNode(node.index,
name: node.name, parent: ownerNode, type: NodeType.classNode);
case 'Library':
// Create fake owner node for the package which contains this library.
final packageName = packageOf(node.name);
return makeInfoNode(node.index,
name: node.name,
parent: packageName != node.name
? packageOwner(packageName)
: program.root,
type: NodeType.libraryNode);
case 'Field':
return makeInfoNode(node.index,
name: node.name,
parent: getInfoNodeFor(node['owner_']!)!,
type: NodeType.other);
}
return null;
}
ProgramInfoNode makeInfoNode(int? index,
{required ProgramInfoNode parent,
required String name,
required NodeType type}) {
name = Name(name).scrubbed;
if (collapseAnonymousClosures) {
name = Name.collapse(name);
}
final node = program.makeNode(name: name, parent: parent, type: type);
if (node.id == infoNodes.length) {
infoNodes.add(node);
}
if (index != null) {
assert(!infoNodeByIndex.containsKey(index));
infoNodeByIndex[index] = node;
}
return node;
}
ProgramInfoNode packageOwner(String packageName) =>
infoNodeForPackage.putIfAbsent(
packageName,
() => makeInfoNode(null,
name: packageName,
type: NodeType.packageNode,
parent: program.root));
/// Create a single key from two node ids.
/// Note that this operation is commutative, because common ancestor of A and
/// B is the same as common ancestor of B and A.
static int ancestorCacheKey(int a, int b) {
if (a > b) {
return b << 32 | a;
} else {
return a << 32 | b;
}
}
/// Returns id of a common ancestor between [ProgramInfoNode] with [idA] and
/// [idB]. At least either [idA] or [idB] are expected to be not null.
int findCommonAncestor(int? idA, int? idB) {
if (idA == null) {
return idB!;
}
if (idB == null) {
return idA;
}
if (idA == idB) {
return idA;
}
// If either are shared - then result is shared.
if (idA == ProgramInfo.rootId || idB == ProgramInfo.rootId) {
return ProgramInfo.rootId;
}
final infoA = infoNodes[idA];
final infoB = infoNodes[idB];
final key = ancestorCacheKey(idA, idB);
var ancestor = commonAncestorCache[key];
if (ancestor == null) {
commonAncestorCache[key] =
ancestor = findCommonAncestorImpl(infoA, infoB).id;
}
return ancestor;
}
static List<ProgramInfoNode> pathToRoot(ProgramInfoNode node) {
final path = <ProgramInfoNode>[];
for (ProgramInfoNode? n = node; n != null; n = n.parent) {
path.add(n);
}
return path;
}
static ProgramInfoNode findCommonAncestorImpl(
ProgramInfoNode a, ProgramInfoNode b) {
final pathA = pathToRoot(a);
final pathB = pathToRoot(b);
var i = pathA.length - 1, j = pathB.length - 1;
while (i > 0 && j > 0 && (pathA[i - 1] == pathB[j - 1])) {
i--;
j--;
}
assert(pathA[i] == pathB[j]);
return pathA[i];
}
}
final bucketLegend = '''
--------------------------------------------------------------------------------
IMPORTANT: Dart AOT snapshot is a serialized representation of Dart VM heap.
Outside of few specific cases (e.g. an object representing a library clearly
originates from the library it represents) there is no well defined relationship
between snapshot bytes and a specific method/class/library to which these
bytes can be attributed with certainty. This snapshot analysis tool tries
to attribute bytes to specific program structure elements based on their
reachability from objects with well defined origin - meaning that this analysis
has some margin of error and imprecision.
- @other bucket denotes bytes attributed to entities outside of the current
granularity. For example, when breaking down the size by method name there
might be bytes which exist outside of any specific symbol - in which case
they will be attributed to @other.
- @stubs bucket accumulates bytes attributed to stubs (pieces of machine code
produced by the VM for internal purposes).
- @shared bucket accumulates bytes shared between otherwise unrelated program
entities
- @unknown bucket accumulates bytes which are not reachable from any program
structure nodes (usually VM internal objects).
--------------------------------------------------------------------------------
''';
/// Compute dominator tree of the graph.
///
/// The code for dominator tree computation is taken verbatim from the
/// native compiler (see runtime/vm/compiler/backend/flow_graph.cc).
List<int> _computeDominators(Snapshot snap) {
final predecessors = List<Object?>.filled(snap.nodeCount, null);
void addPred(int n, int p) {
final pred = predecessors[n];
if (pred == null) {
predecessors[n] = p;
} else if (pred is int) {
predecessors[n] = <int>[pred, p];
} else {
(pred as List<int>).add(p);
}
}
Iterable<int> predOf(int n) sync* {
final ps = predecessors[n];
if (ps is int) {
yield ps;
} else if (ps is List<int>) {
yield* ps;
}
}
return dominators.computeDominators(
size: snap.nodeCount,
root: snap.nodes.first.index,
succ: (n) => snap.nodeAt(n).edges.map((e) => e.target.index),
predOf: predOf,
handleEdge: addPred);
}