Chromium Code Reviews| Index: runtime/observatory/lib/src/elements/memory/profile.dart |
| diff --git a/runtime/observatory/lib/src/elements/memory/profile.dart b/runtime/observatory/lib/src/elements/memory/profile.dart |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..ae1d49a4f57590c387f3d7732523476821e007da |
| --- /dev/null |
| +++ b/runtime/observatory/lib/src/elements/memory/profile.dart |
| @@ -0,0 +1,246 @@ |
| +// Copyright (c) 2014, the Dart project authors. Please see the AUTHORS file |
|
siva
2017/08/03 18:14:51
2017
cbernaschina
2017/08/03 22:32:53
Done.
|
| +// 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: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/rendering_scheduler.dart'; |
| +import 'package:observatory/src/elements/helpers/tag.dart'; |
| +import 'package:observatory/utils.dart'; |
| + |
| +enum _SortingField { |
| + accumulatedSize, |
| + accumulatedInstances, |
| + currentSize, |
| + currentInstances, |
| + className, |
| +} |
| + |
| +enum _SortingDirection { ascending, descending } |
| + |
| +class MemoryProfileElement extends HtmlElement implements Renderable { |
| + static const tag = const Tag<MemoryProfileElement>('memory-profile', |
| + dependencies: const [ClassRefElement.tag, VirtualCollectionElement.tag]); |
| + |
| + RenderingScheduler<MemoryProfileElement> _r; |
| + |
| + Stream<RenderedEvent<MemoryProfileElement>> get onRendered => _r.onRendered; |
| + |
| + M.IsolateRef _isolate; |
| + M.EventRepository _events; |
| + M.AllocationProfileRepository _repository; |
| + M.AllocationProfile _profile; |
| + M.EditorRepository _editor; |
| + StreamSubscription _gcSubscription; |
| + _SortingField _sortingField = _SortingField.accumulatedInstances; |
| + _SortingDirection _sortingDirection = _SortingDirection.descending; |
| + |
| + M.IsolateRef get isolate => _isolate; |
| + |
| + factory MemoryProfileElement(M.IsolateRef isolate, M.EditorRepository editor, |
| + M.EventRepository events, M.AllocationProfileRepository repository, |
| + {RenderingQueue queue}) { |
| + assert(isolate != null); |
| + assert(events != null); |
| + assert(editor != null); |
| + assert(repository != null); |
| + MemoryProfileElement e = document.createElement(tag.name); |
| + e._r = new RenderingScheduler(e, queue: queue); |
| + e._isolate = isolate; |
| + e._editor = editor; |
| + e._events = events; |
| + e._repository = repository; |
| + return e; |
| + } |
| + |
| + MemoryProfileElement.created() : super.created(); |
| + |
| + @override |
| + attached() { |
| + super.attached(); |
| + _r.enable(); |
| + _refresh(); |
| + _gcSubscription = _events.onGCEvent.listen((e) { |
| + if (e.isolate.id == _isolate.id) { |
| + _refresh(); |
| + } |
| + }); |
| + } |
| + |
| + @override |
| + detached() { |
| + super.detached(); |
| + _r.disable(notify: true); |
| + children = []; |
| + _gcSubscription.cancel(); |
| + } |
| + |
| + Future reload({bool gc = false, bool reset = false}) async { |
| + return _refresh(gc: gc, reset: reset); |
| + } |
| + |
| + void render() { |
| + if (_profile == null) { |
| + children = [ |
| + new DivElement() |
| + ..classes = ['content-centered-big'] |
| + ..children = [new HeadingElement.h2()..text = 'Loading...'] |
| + ]; |
| + } else { |
| + children = [ |
| + new VirtualCollectionElement( |
| + _createCollectionLine, _updateCollectionLine, |
| + createHeader: _createCollectionHeader, |
| + items: _profile.members.toList()..sort(_createSorter()), |
| + queue: _r.queue) |
| + ]; |
| + } |
| + } |
| + |
| + _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.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 = ['name'] |
| + ]; |
| + |
| + List<HtmlElement> _createCollectionHeader() { |
| + final resetAccumulators = new ButtonElement(); |
| + return [ |
| + new DivElement() |
| + ..classes = ['collection-item'] |
| + ..children = [ |
| + new SpanElement() |
| + ..classes = ['group'] |
| + ..children = [ |
| + new Text('Since Last '), |
| + resetAccumulators |
| + ..text = 'Reset↺' |
| + ..title = 'Reset' |
| + ..onClick.listen((_) async { |
| + resetAccumulators.disabled = true; |
| + await _refresh(reset: true); |
| + resetAccumulators.disabled = false; |
| + }) |
| + ], |
| + new SpanElement() |
| + ..classes = ['group'] |
| + ..text = '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 ['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] = new ClassRefElement(_isolate, item.clazz, queue: _r.queue) |
| + ..classes = ['name']; |
| + Element.clickEvent.forTarget(e.children[4], useCapture: true).listen((e) { |
| + if (_editor.canOpenClass) { |
| + e.preventDefault(); |
| + _editor.openClass(isolate, item.clazz); |
| + } |
| + }); |
| + } |
| + |
| + Future _refresh({bool gc: false, bool reset: false}) async { |
| + _profile = null; |
| + _r.dirty(); |
| + _profile = await _repository.get(_isolate, gc: gc, reset: reset); |
| + _r.dirty(); |
| + } |
| + |
| + 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; |
| +} |