Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(394)

Unified Diff: runtime/observatory/lib/src/elements/memory_dashboard.dart

Issue 2962593002: Added Editor stream and sendObjectToEditor RPC into Service Protocol (Closed)
Patch Set: Added explanation comment Created 3 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: runtime/observatory/lib/src/elements/memory_dashboard.dart
diff --git a/runtime/observatory/lib/src/elements/memory_dashboard.dart b/runtime/observatory/lib/src/elements/memory_dashboard.dart
new file mode 100644
index 0000000000000000000000000000000000000000..28181bb7c5466cb33ccec34c8daf7af7ae95bd87
--- /dev/null
+++ b/runtime/observatory/lib/src/elements/memory_dashboard.dart
@@ -0,0 +1,649 @@
+// Copyright (c) 2014, 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.
+
+import 'dart:async';
+import 'dart:html';
+import 'package:charted/charted.dart';
+import "package:charted/charts/charts.dart";
+import 'package:observatory/models.dart' as M;
+import 'package:observatory/src/elements/class_ref.dart';
+import 'package:observatory/src/elements/containers/virtual_collection.dart';
+import 'package:observatory/src/elements/helpers/nav_bar.dart';
+import 'package:observatory/src/elements/helpers/rendering_scheduler.dart';
+import 'package:observatory/src/elements/helpers/tag.dart';
+import 'package:observatory/src/elements/nav/isolate_menu.dart';
+import 'package:observatory/src/elements/nav/notify.dart';
+import 'package:observatory/src/elements/nav/refresh.dart';
+import 'package:observatory/src/elements/nav/top_menu.dart';
+import 'package:observatory/src/elements/nav/vm_menu.dart';
+import 'package:observatory/utils.dart';
+
+enum _SortingField {
+ accumulatedSize,
+ accumulatedInstances,
+ currentSize,
+ currentInstances,
+ newAccumulatedSize,
+ newAccumulatedInstances,
+ newCurrentSize,
+ newCurrentInstances,
+ oldAccumulatedSize,
+ oldAccumulatedInstances,
+ oldCurrentSize,
+ oldCurrentInstances,
+ className,
+}
+
+enum _SortingDirection { ascending, descending }
+
+class MemoryDashboardElement extends HtmlElement implements Renderable {
+ static const tag = const Tag<MemoryDashboardElement>('memory-dashboard',
+ dependencies: const [
+ ClassRefElement.tag,
+ NavTopMenuElement.tag,
+ NavVMMenuElement.tag,
+ NavIsolateMenuElement.tag,
+ NavRefreshElement.tag,
+ NavNotifyElement.tag,
+ VirtualCollectionElement.tag
+ ]);
+
+ RenderingScheduler<MemoryDashboardElement> _r;
+
+ Stream<RenderedEvent<MemoryDashboardElement>> get onRendered => _r.onRendered;
+
+ M.VM _vm;
+ M.IsolateRef _isolate;
+ M.EventRepository _events;
+ M.NotificationRepository _notifications;
+ M.AllocationProfileRepository _repository;
+ M.AllocationProfile _profile;
+ M.EditorRepository _editor;
+ bool _autoRefresh = false;
+ bool _isCompacted = false;
+ StreamSubscription _gcSubscription;
+ _SortingField _sortingField = _SortingField.className;
+ _SortingDirection _sortingDirection = _SortingDirection.ascending;
+
+ M.VMRef get vm => _vm;
+ M.IsolateRef get isolate => _isolate;
+ M.NotificationRepository get notifications => _notifications;
+
+ factory MemoryDashboardElement(
+ M.VM vm,
+ M.IsolateRef isolate,
+ M.EventRepository events,
+ M.NotificationRepository notifications,
+ M.AllocationProfileRepository repository,
+ M.EditorRepository editor,
+ {RenderingQueue queue}) {
+ assert(vm != null);
+ assert(isolate != null);
+ assert(events != null);
+ assert(notifications != null);
+ assert(repository != null);
+ assert(editor != null);
+ MemoryDashboardElement e = document.createElement(tag.name);
+ e._r = new RenderingScheduler(e, queue: queue);
+ e._vm = vm;
+ e._isolate = isolate;
+ e._events = events;
+ e._notifications = notifications;
+ e._repository = repository;
+ e._editor = editor;
+ return e;
+ }
+
+ MemoryDashboardElement.created() : super.created();
+
+ @override
+ attached() {
+ super.attached();
+ _r.enable();
+ _refresh();
+ _gcSubscription = _events.onGCEvent.listen((e) {
+ if (_autoRefresh && (e.isolate.id == _isolate.id)) {
+ _refresh();
+ }
+ });
+ }
+
+ @override
+ detached() {
+ super.detached();
+ _r.disable(notify: true);
+ children = [];
+ _gcSubscription.cancel();
+ }
+
+ void render() {
+ children = [
+ navBar([
+ new NavRefreshElement(
+ label: 'Download', disabled: _profile == null, queue: _r.queue)
+ ..onRefresh.listen((_) => _downloadCSV()),
+ new NavRefreshElement(label: 'Reset Accumulator', queue: _r.queue)
+ ..onRefresh.listen((_) => _refresh(reset: true)),
+ new NavRefreshElement(label: 'GC', queue: _r.queue)
+ ..onRefresh.listen((_) => _refresh(gc: true)),
+ new NavRefreshElement(queue: _r.queue)
+ ..onRefresh.listen((_) => _refresh()),
+ new DivElement()
+ ..classes = ['nav-option']
+ ..children = [
+ new CheckboxInputElement()
+ ..id = 'allocation-profile-auto-refresh'
+ ..checked = _autoRefresh
+ ..onChange.listen((_) => _autoRefresh = !_autoRefresh),
+ new LabelElement()
+ ..htmlFor = 'allocation-profile-auto-refresh'
+ ..text = 'Auto-refresh on GC'
+ ],
+ new NavNotifyElement(_notifications, queue: _r.queue)
+ ]),
+ new DivElement()
+ ..classes = ['content-centered-big']
+ ..children = [
+ new HeadingElement.h2()..text = 'Allocation Profile',
+ new HRElement()
+ ]
+ ];
+ if (_profile == null) {
+ children.addAll([
+ new DivElement()
+ ..classes = ['content-centered-big']
+ ..children = [new HeadingElement.h2()..text = 'Loading...']
+ ]);
+ } else {
+ final newChartHost = new DivElement()..classes = ['host'];
+ final newChartLegend = new DivElement()..classes = ['legend'];
+ final oldChartHost = new DivElement()..classes = ['host'];
+ final oldChartLegend = new DivElement()..classes = ['legend'];
+ children.addAll([
+ new DivElement()
+ ..classes = ['content-centered-big']
+ ..children = _isCompacted
+ ? []
+ : [
+ new DivElement()
+ ..classes = ['memberList']
+ ..children = [
+ new DivElement()
+ ..classes = ['memberItem']
+ ..children = [
+ new DivElement()
+ ..classes = ['memberName']
+ ..text = 'last forced GC at',
+ new DivElement()
+ ..classes = ['memberValue']
+ ..text = _profile.lastServiceGC == null
+ ? '---'
+ : '${_profile.lastServiceGC}',
+ ],
+ new DivElement()
+ ..classes = ['memberItem']
+ ..children = [
+ new DivElement()
+ ..classes = ['memberName']
+ ..text = 'last accumulator reset at',
+ new DivElement()
+ ..classes = ['memberValue']
+ ..text = _profile.lastAccumulatorReset == null
+ ? '---'
+ : '${_profile.lastAccumulatorReset}',
+ ]
+ ],
+ new HRElement(),
+ ],
+ new DivElement()
+ ..classes = ['content-centered-big', 'compactable']
+ ..children = [
+ new DivElement()
+ ..classes = ['heap-space', 'left']
+ ..children = _isCompacted
+ ? [
+ new HeadingElement.h2()
+ ..text = 'New Generation '
+ '(${_usedCaption(_profile.newSpace)})',
+ ]
+ : [
+ new HeadingElement.h2()..text = 'New Generation',
+ new BRElement(),
+ new DivElement()
+ ..classes = ['memberList']
+ ..children = _createSpaceMembers(_profile.newSpace),
+ new BRElement(),
+ new DivElement()
+ ..classes = ['chart']
+ ..children = [newChartLegend, newChartHost]
+ ],
+ new DivElement()
+ ..classes = ['heap-space', 'right']
+ ..children = _isCompacted
+ ? [
+ new HeadingElement.h2()
+ ..text = '(${_usedCaption(_profile.oldSpace)}) '
+ 'Old Generation',
+ ]
+ : [
+ new HeadingElement.h2()..text = 'Old Generation',
+ new BRElement(),
+ new DivElement()
+ ..classes = ['memberList']
+ ..children = _createSpaceMembers(_profile.oldSpace),
+ new BRElement(),
+ new DivElement()
+ ..classes = ['chart']
+ ..children = [oldChartLegend, oldChartHost]
+ ],
+ new ButtonElement()
+ ..classes = ['compact']
+ ..text = _isCompacted ? 'expand ▼' : 'compact ▲'
+ ..onClick.listen((_) {
+ _isCompacted = !_isCompacted;
+ _r.dirty();
+ }),
+ new HRElement()
+ ],
+ new DivElement()
+ ..classes = _isCompacted ? ['collection', 'expanded'] : ['collection']
+ ..children = [
+ new VirtualCollectionElement(
+ _createCollectionLine, _updateCollectionLine,
+ createHeader: _createCollectionHeader,
+ items: _profile.members.toList()..sort(_createSorter()),
+ queue: _r.queue)
+ ]
+ ]);
+ _renderGraph(newChartHost, newChartLegend, _profile.newSpace);
+ _renderGraph(oldChartHost, oldChartLegend, _profile.oldSpace);
+ }
+ }
+
+ _createSorter() {
+ var getter;
+ switch (_sortingField) {
+ case _SortingField.accumulatedSize:
+ getter = _getAccumulatedSize;
+ break;
+ case _SortingField.accumulatedInstances:
+ getter = _getAccumulatedInstances;
+ break;
+ case _SortingField.currentSize:
+ getter = _getCurrentSize;
+ break;
+ case _SortingField.currentInstances:
+ getter = _getCurrentInstances;
+ break;
+ case _SortingField.newAccumulatedSize:
+ getter = _getNewAccumulatedSize;
+ break;
+ case _SortingField.newAccumulatedInstances:
+ getter = _getNewAccumulatedInstances;
+ break;
+ case _SortingField.newCurrentSize:
+ getter = _getNewCurrentSize;
+ break;
+ case _SortingField.newCurrentInstances:
+ getter = _getNewCurrentInstances;
+ break;
+ case _SortingField.oldAccumulatedSize:
+ getter = _getOldAccumulatedSize;
+ break;
+ case _SortingField.oldAccumulatedInstances:
+ getter = _getOldAccumulatedInstances;
+ break;
+ case _SortingField.oldCurrentSize:
+ getter = _getOldCurrentSize;
+ break;
+ case _SortingField.oldCurrentInstances:
+ getter = _getOldCurrentInstances;
+ break;
+ case _SortingField.className:
+ getter = (M.ClassHeapStats s) => s.clazz.name;
+ break;
+ }
+ switch (_sortingDirection) {
+ case _SortingDirection.ascending:
+ return (a, b) => getter(a).compareTo(getter(b));
+ case _SortingDirection.descending:
+ return (a, b) => getter(b).compareTo(getter(a));
+ }
+ }
+
+ static Element _createCollectionLine() => new DivElement()
+ ..classes = ['collection-item']
+ ..children = [
+ new SpanElement()
+ ..classes = ['bytes']
+ ..text = '0B',
+ new SpanElement()
+ ..classes = ['instances']
+ ..text = '0',
+ new SpanElement()
+ ..classes = ['bytes']
+ ..text = '0B',
+ new SpanElement()
+ ..classes = ['instances']
+ ..text = '0',
+ new SpanElement()
+ ..classes = ['bytes']
+ ..text = '0B',
+ new SpanElement()
+ ..classes = ['instances']
+ ..text = '0',
+ new SpanElement()
+ ..classes = ['bytes']
+ ..text = '0B',
+ new SpanElement()
+ ..classes = ['instances']
+ ..text = '0',
+ new SpanElement()
+ ..classes = ['bytes']
+ ..text = '0B',
+ new SpanElement()
+ ..classes = ['instances']
+ ..text = '0',
+ new SpanElement()
+ ..classes = ['bytes']
+ ..text = '0B',
+ new SpanElement()
+ ..classes = ['instances']
+ ..text = '0',
+ new SpanElement()..classes = ['name']
+ ];
+
+ Element _createCollectionHeader() => new DivElement()
+ ..children = [
+ new DivElement()
+ ..classes = ['collection-item']
+ ..children = [
+ new SpanElement()
+ ..classes = ['group']
+ ..text = 'Accumulated',
+ new SpanElement()
+ ..classes = ['group']
+ ..text = 'Current',
+ new SpanElement()
+ ..classes = ['group']
+ ..text = '(NEW) Accumulated',
+ new SpanElement()
+ ..classes = ['group']
+ ..text = '(NEW) Current',
+ new SpanElement()
+ ..classes = ['group']
+ ..text = '(OLD) Accumulated',
+ new SpanElement()
+ ..classes = ['group']
+ ..text = '(OLD) Current',
+ ],
+ new DivElement()
+ ..classes = ['collection-item']
+ ..children = [
+ _createHeaderButton(const ['bytes'], 'Size',
+ _SortingField.accumulatedSize, _SortingDirection.descending),
+ _createHeaderButton(const ['instances'], 'Instances',
+ _SortingField.accumulatedInstances, _SortingDirection.descending),
+ _createHeaderButton(const ['bytes'], 'Size',
+ _SortingField.currentSize, _SortingDirection.descending),
+ _createHeaderButton(const ['instances'], 'Instances',
+ _SortingField.currentInstances, _SortingDirection.descending),
+ _createHeaderButton(const ['bytes'], 'Size',
+ _SortingField.newAccumulatedSize, _SortingDirection.descending),
+ _createHeaderButton(
+ const ['instances'],
+ 'Instances',
+ _SortingField.newAccumulatedInstances,
+ _SortingDirection.descending),
+ _createHeaderButton(const ['bytes'], 'Size',
+ _SortingField.newCurrentSize, _SortingDirection.descending),
+ _createHeaderButton(const ['instances'], 'Instances',
+ _SortingField.newCurrentInstances, _SortingDirection.descending),
+ _createHeaderButton(const ['bytes'], 'Size',
+ _SortingField.oldAccumulatedSize, _SortingDirection.descending),
+ _createHeaderButton(
+ const ['instances'],
+ 'Instances',
+ _SortingField.oldAccumulatedInstances,
+ _SortingDirection.descending),
+ _createHeaderButton(const ['bytes'], 'Size',
+ _SortingField.oldCurrentSize, _SortingDirection.descending),
+ _createHeaderButton(const ['instances'], 'Instances',
+ _SortingField.oldCurrentInstances, _SortingDirection.descending),
+ _createHeaderButton(const ['name'], 'Class', _SortingField.className,
+ _SortingDirection.ascending)
+ ],
+ ];
+
+ ButtonElement _createHeaderButton(List<String> classes, String text,
+ _SortingField field, _SortingDirection direction) =>
+ new ButtonElement()
+ ..classes = classes
+ ..text = _sortingField != field
+ ? text
+ : _sortingDirection == _SortingDirection.ascending
+ ? '$text▼'
+ : '$text▲'
+ ..onClick.listen((_) => _setSorting(field, direction));
+
+ void _setSorting(_SortingField field, _SortingDirection defaultDirection) {
+ if (_sortingField == field) {
+ switch (_sortingDirection) {
+ case _SortingDirection.descending:
+ _sortingDirection = _SortingDirection.ascending;
+ break;
+ case _SortingDirection.ascending:
+ _sortingDirection = _SortingDirection.descending;
+ break;
+ }
+ } else {
+ _sortingDirection = defaultDirection;
+ _sortingField = field;
+ }
+ _r.dirty();
+ }
+
+ void _updateCollectionLine(Element e, M.ClassHeapStats item, index) {
+ e.children[0].text = Utils.formatSize(_getAccumulatedSize(item));
+ e.children[1].text = '${_getAccumulatedInstances(item)}';
+ e.children[2].text = Utils.formatSize(_getCurrentSize(item));
+ e.children[3].text = '${_getCurrentInstances(item)}';
+ e.children[4].text = Utils.formatSize(_getNewAccumulatedSize(item));
+ e.children[5].text = '${_getNewAccumulatedInstances(item)}';
+ e.children[6].text = Utils.formatSize(_getNewCurrentSize(item));
+ e.children[7].text = '${_getNewCurrentInstances(item)}';
+ e.children[8].text = Utils.formatSize(_getOldAccumulatedSize(item));
+ e.children[9].text = '${_getOldAccumulatedInstances(item)}';
+ e.children[10].text = Utils.formatSize(_getOldCurrentSize(item));
+ e.children[11].text = '${_getOldCurrentInstances(item)}';
+ e.children[12] = new ClassRefElement(_isolate, item.clazz, queue: _r.queue)
+ ..classes = ['name'];
+ Element.clickEvent.forTarget(e.children[12], useCapture: true).listen((e) {
+ e.preventDefault();
+ _editor.sendObject(isolate, item.clazz);
+ window.close();
+ });
+ }
+
+ static String _usedCaption(M.HeapSpace space) =>
+ '${Utils.formatSize(space.used)}'
+ ' of '
+ '${Utils.formatSize(space.capacity)}';
+
+ static List<Element> _createSpaceMembers(M.HeapSpace space) {
+ final used = _usedCaption(space);
+ final ext = '${Utils.formatSize(space.external)}';
+ final collections = '${space.collections}';
+ final avgCollectionTime =
+ '${Utils.formatDurationInMilliseconds(space.avgCollectionTime)} ms';
+ final totalCollectionTime =
+ '${Utils.formatDurationInSeconds(space.totalCollectionTime)} secs';
+ final avgCollectionPeriod =
+ '${Utils.formatDurationInMilliseconds(space.avgCollectionPeriod)} ms';
+ return [
+ new DivElement()
+ ..classes = ['memberItem']
+ ..children = [
+ new DivElement()
+ ..classes = ['memberName']
+ ..text = 'used',
+ new DivElement()
+ ..classes = ['memberValue']
+ ..text = used
+ ],
+ new DivElement()
+ ..classes = ['memberItem']
+ ..children = [
+ new DivElement()
+ ..classes = ['memberName']
+ ..text = 'external',
+ new DivElement()
+ ..classes = ['memberValue']
+ ..text = ext
+ ],
+ new DivElement()
+ ..classes = ['memberItem']
+ ..children = [
+ new DivElement()
+ ..classes = ['memberName']
+ ..text = 'collections',
+ new DivElement()
+ ..classes = ['memberValue']
+ ..text = collections
+ ],
+ new DivElement()
+ ..classes = ['memberItem']
+ ..children = [
+ new DivElement()
+ ..classes = ['memberName']
+ ..text = 'average collection time',
+ new DivElement()
+ ..classes = ['memberValue']
+ ..text = avgCollectionTime
+ ],
+ new DivElement()
+ ..classes = ['memberItem']
+ ..children = [
+ new DivElement()
+ ..classes = ['memberName']
+ ..text = 'cumulative collection time',
+ new DivElement()
+ ..classes = ['memberValue']
+ ..text = totalCollectionTime
+ ],
+ new DivElement()
+ ..classes = ['memberItem']
+ ..children = [
+ new DivElement()
+ ..classes = ['memberName']
+ ..text = 'average time between collections',
+ new DivElement()
+ ..classes = ['memberValue']
+ ..text = avgCollectionPeriod
+ ]
+ ];
+ }
+
+ static final _columns = [
+ new ChartColumnSpec(label: 'Type', type: ChartColumnSpec.TYPE_STRING),
+ new ChartColumnSpec(label: 'Size', formatter: (v) => v.toString())
+ ];
+
+ static void _renderGraph(Element host, Element legend, M.HeapSpace space) {
+ final series = [
+ new ChartSeries("Work", [1], new PieChartRenderer(sortDataByValue: false))
+ ];
+ final rect = host.getBoundingClientRect();
+ final minSize = new Rect.size(rect.width, rect.height);
+ final config = new ChartConfig(series, [0])
+ ..minimumSize = minSize
+ ..legend = new ChartLegend(legend, showValues: true);
+ final data = new ChartData(_columns, [
+ ['Used', space.used],
+ ['Free', space.capacity - space.used],
+ ['External', space.external]
+ ]);
+
+ new LayoutArea(host, data, config,
+ state: new ChartState(), autoUpdate: true)
+ ..draw();
+ }
+
+ Future _refresh({bool gc: false, bool reset: false}) async {
+ _profile = null;
+ _r.dirty();
+ _profile = await _repository.get(_isolate, gc: gc, reset: reset);
+ _r.dirty();
+ }
+
+ void _downloadCSV() {
+ assert(_profile != null);
+ final header = [
+ '"Accumulator Size"',
+ '"Accumulator Instances"',
+ '"Current Size"',
+ '"Current Instances"',
+ '"(NEW) Accumulator Size"',
+ '"(NEW) Accumulator Instances"',
+ '"(NEW) Current Size"',
+ '"(NEW) Current Instances"',
+ '"(OLD) Accumulator Size"',
+ '"(OLD) Accumulator Instances"',
+ '"(OLD) Current Size"',
+ '"(OLD) Current Instances"',
+ 'Class'
+ ].join(',') +
+ '\n';
+ AnchorElement tl = document.createElement('a');
+ tl
+ ..attributes['href'] = 'data:text/plain;charset=utf-8,' +
+ Uri.encodeComponent(header +
+ (_profile.members.toList()..sort(_createSorter()))
+ .map(_csvOut)
+ .join('\n'))
+ ..attributes['download'] = 'heap-profile.csv'
+ ..click();
+ }
+
+ static _csvOut(M.ClassHeapStats s) {
+ return [
+ _getAccumulatedSize(s),
+ _getAccumulatedInstances(s),
+ _getCurrentSize(s),
+ _getCurrentInstances(s),
+ _getNewAccumulatedSize(s),
+ _getNewAccumulatedInstances(s),
+ _getNewCurrentSize(s),
+ _getNewCurrentInstances(s),
+ _getOldAccumulatedSize(s),
+ _getOldAccumulatedInstances(s),
+ _getOldCurrentSize(s),
+ _getOldCurrentInstances(s),
+ s.clazz.name
+ ].join(',');
+ }
+
+ static int _getAccumulatedSize(M.ClassHeapStats s) =>
+ s.newSpace.accumulated.bytes + s.oldSpace.accumulated.bytes;
+ static int _getAccumulatedInstances(M.ClassHeapStats s) =>
+ s.newSpace.accumulated.instances + s.oldSpace.accumulated.instances;
+ static int _getCurrentSize(M.ClassHeapStats s) =>
+ s.newSpace.current.bytes + s.oldSpace.current.bytes;
+ static int _getCurrentInstances(M.ClassHeapStats s) =>
+ s.newSpace.current.instances + s.oldSpace.current.instances;
+ static int _getNewAccumulatedSize(M.ClassHeapStats s) =>
+ s.newSpace.accumulated.bytes;
+ static int _getNewAccumulatedInstances(M.ClassHeapStats s) =>
+ s.newSpace.accumulated.instances;
+ static int _getNewCurrentSize(M.ClassHeapStats s) => s.newSpace.current.bytes;
+ static int _getNewCurrentInstances(M.ClassHeapStats s) =>
+ s.newSpace.current.instances;
+ static int _getOldAccumulatedSize(M.ClassHeapStats s) =>
+ s.oldSpace.accumulated.bytes;
+ static int _getOldAccumulatedInstances(M.ClassHeapStats s) =>
+ s.oldSpace.accumulated.instances;
+ static int _getOldCurrentSize(M.ClassHeapStats s) => s.oldSpace.current.bytes;
+ static int _getOldCurrentInstances(M.ClassHeapStats s) =>
+ s.oldSpace.current.instances;
+}
« no previous file with comments | « runtime/observatory/lib/src/elements/debugger.dart ('k') | runtime/observatory/lib/src/models/repositories/editor.dart » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698