blob: 44c137995d8b5974a659a540915046e55a9b53db [file] [log] [blame]
// Copyright (c) 2013, 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.
library cpu_profile_table_element;
import 'dart:async';
import 'package:web/web.dart';
import '../../models.dart' as M;
import 'containers/virtual_collection.dart';
import 'cpu_profile/virtual_tree.dart';
import 'function_ref.dart';
import 'helpers/custom_element.dart';
import 'helpers/element_utils.dart';
import 'helpers/nav_bar.dart';
import 'helpers/nav_menu.dart';
import 'helpers/rendering_scheduler.dart';
import 'nav/isolate_menu.dart';
import 'nav/notify.dart';
import 'nav/refresh.dart';
import 'nav/top_menu.dart';
import 'nav/vm_menu.dart';
import 'sample_buffer_control.dart';
import 'stack_trace_tree_config.dart';
import '../../utils.dart';
enum _Table { functions, caller, callee }
enum _SortingField { exclusive, inclusive, caller, callee, method }
enum _SortingDirection { ascending, descending }
class CpuProfileTableElement extends CustomElement implements Renderable {
late RenderingScheduler<CpuProfileTableElement> _r;
Stream<RenderedEvent<CpuProfileTableElement>> get onRendered => _r.onRendered;
late M.VM _vm;
late M.IsolateRef _isolate;
late M.EventRepository _events;
late M.NotificationRepository _notifications;
late M.IsolateSampleProfileRepository _profiles;
late Stream<M.SampleProfileLoadingProgressEvent> _progressStream;
M.SampleProfileLoadingProgress? _progress;
final _sortingField = <_Table, _SortingField>{
_Table.functions: _SortingField.exclusive,
_Table.caller: _SortingField.caller,
_Table.callee: _SortingField.callee,
};
final _sortingDirection = <_Table, _SortingDirection>{
_Table.functions: _SortingDirection.descending,
_Table.caller: _SortingDirection.descending,
_Table.callee: _SortingDirection.descending,
};
String _filter = '';
M.IsolateRef get isolate => _isolate;
M.NotificationRepository get notifications => _notifications;
M.IsolateSampleProfileRepository get profiles => _profiles;
M.VMRef get vm => _vm;
factory CpuProfileTableElement(
M.VM vm,
M.IsolateRef isolate,
M.EventRepository events,
M.NotificationRepository notifications,
M.IsolateSampleProfileRepository profiles, {
RenderingQueue? queue,
}) {
CpuProfileTableElement e = new CpuProfileTableElement.created();
e._r = new RenderingScheduler<CpuProfileTableElement>(e, queue: queue);
e._vm = vm;
e._isolate = isolate;
e._events = events;
e._notifications = notifications;
e._profiles = profiles;
return e;
}
CpuProfileTableElement.created() : super.created('cpu-profile-table');
@override
attached() {
super.attached();
_r.enable();
_request();
}
@override
detached() {
super.detached();
_r.disable(notify: true);
removeChildren();
}
void render() {
var content = <HTMLElement>[
navBar(<HTMLElement>[
new NavTopMenuElement(queue: _r.queue).element,
new NavVMMenuElement(_vm, _events, queue: _r.queue).element,
new NavIsolateMenuElement(_isolate, _events, queue: _r.queue).element,
navMenu('cpu profile (table)'),
(new NavRefreshElement(
queue: _r.queue,
)..onRefresh.listen(_refresh)).element,
(new NavRefreshElement(
label: 'Clear',
queue: _r.queue,
)..onRefresh.listen(_clearCpuSamples)).element,
new NavNotifyElement(_notifications, queue: _r.queue).element,
]),
];
if (_progress == null) {
setChildren(content);
return;
}
content.add(
new SampleBufferControlElement(
_vm,
_progress!,
_progressStream,
showTag: false,
queue: _r.queue,
).element,
);
if (_progress!.status == M.SampleProfileLoadingStatus.loaded) {
content.add(new HTMLBRElement());
content.addAll(_createTables());
content.add(new HTMLBRElement());
content.addAll(_createTree());
}
setChildren(content);
}
M.ProfileFunction? _selected;
VirtualCollectionElement? _functions;
VirtualCollectionElement? _callers;
VirtualCollectionElement? _callees;
List<HTMLElement> _createTables() {
_functions =
_functions ??
new VirtualCollectionElement(
_createFunction,
_updateFunction,
createHeader: _createFunctionHeader,
search: _searchFunction,
queue: _r.queue,
);
// If there's no samples, don't populate the function list.
_functions!.items =
(_progress!.profile.sampleCount != 0)
? _progress!.profile.functions.toList()
: []
..sort(_createSorter(_Table.functions));
_functions!.takeIntoView(_selected);
_callers =
_callers ??
new VirtualCollectionElement(
_createCaller,
_updateCaller,
createHeader: _createCallerHeader,
search: _searchFunction,
queue: _r.queue,
);
_callees =
_callees ??
new VirtualCollectionElement(
_createCallee,
_updateCallee,
createHeader: _createCalleeHeader,
search: _searchFunction,
queue: _r.queue,
);
if (_selected != null) {
_callers!.items = _selected!.callers.keys.toList()
..sort(_createSorter(_Table.caller));
_callees!.items = _selected!.callees.keys.toList()
..sort(_createSorter(_Table.callee));
} else {
_callers!.items = const [];
_callees!.items = const [];
}
return <HTMLElement>[
new HTMLDivElement()
..className = 'profile-trees'
..appendChildren(<HTMLElement>[
new HTMLDivElement()
..className = 'profile-trees-all'
..appendChildren(<HTMLElement>[_functions!.element]),
new HTMLDivElement()
..className = 'profile-trees-current'
..appendChildren(<HTMLElement>[
new HTMLDivElement()
..className = 'profile-trees-caller'
..appendChildren(<HTMLElement>[_callers!.element]),
new HTMLDivElement()
..className = 'profile-trees-selected'
..setChildren(
_selected == null
? [
new HTMLSpanElement()
..textContent = 'No element selected',
]
: [
new FunctionRefElement(
_isolate,
_selected!.function!,
queue: _r.queue,
).element,
],
),
new HTMLDivElement()
..className = 'profile-trees-callee'
..appendChildren(<HTMLElement>[_callees!.element]),
]),
]),
];
}
HTMLElement _createFunction() {
final element = new HTMLDivElement()
..className = 'function-item'
..appendChildren(<HTMLElement>[
new HTMLSpanElement()
..className = 'exclusive'
..textContent = '0%',
new HTMLSpanElement()
..className = 'inclusive'
..textContent = '0%',
new HTMLSpanElement()..className = 'name',
]);
element.onClick.listen((e) {
if (e.target is HTMLAnchorElement) {
return;
}
_selected = _functions!.getItemFromElement(element);
_r.dirty();
});
return element;
}
void _updateFunction(HTMLElement e, itemDynamic, int index) {
M.ProfileFunction item = itemDynamic;
if (item == _selected) {
e.className = 'function-item selected';
} else {
e.className = 'function-item';
}
(e.children.item(0) as HTMLElement).textContent =
Utils.formatPercentNormalized(_getExclusiveT(item));
(e.children.item(1) as HTMLElement).textContent =
Utils.formatPercentNormalized(_getInclusiveT(item));
(e.children.item(2) as HTMLElement).textContent = M.getFunctionFullName(
item.function!,
);
}
List<HTMLElement> _createFunctionHeader() => [
new HTMLDivElement()
..className = 'function-item'
..appendChildren(<HTMLElement>[
_createHeaderButton(
'exclusive',
'Execution(%)',
_Table.functions,
_SortingField.exclusive,
_SortingDirection.descending,
),
_createHeaderButton(
'inclusive',
'Stack(%)',
_Table.functions,
_SortingField.inclusive,
_SortingDirection.descending,
),
_createHeaderButton(
'name',
'Method',
_Table.functions,
_SortingField.method,
_SortingDirection.ascending,
),
]),
];
bool _searchFunction(Pattern pattern, itemDynamic) {
M.ProfileFunction item = itemDynamic;
return M.getFunctionFullName(item.function!).contains(pattern);
}
void _setSorting(
_Table table,
_SortingField field,
_SortingDirection defaultDirection,
) {
if (_sortingField[table] == field) {
switch (_sortingDirection[table]!) {
case _SortingDirection.descending:
_sortingDirection[table] = _SortingDirection.ascending;
break;
case _SortingDirection.ascending:
_sortingDirection[table] = _SortingDirection.descending;
break;
}
} else {
_sortingDirection[table] = defaultDirection;
_sortingField[table] = field;
}
_r.dirty();
}
HTMLElement _createCallee() {
final element = new HTMLDivElement()
..className = 'function-item'
..appendChildren(<HTMLElement>[
new HTMLSpanElement()
..className = 'inclusive'
..textContent = '0%',
new HTMLSpanElement()..className = 'name',
]);
element.onClick.listen((e) {
if (e.target is HTMLAnchorElement) {
return;
}
_selected = _callees!.getItemFromElement(element);
_r.dirty();
});
return element;
}
void _updateCallee(HTMLElement e, item, int index) {
(e.children.item(0) as HTMLElement).textContent =
Utils.formatPercentNormalized(_getCalleeT(item));
(e.children.item(1) as HTMLElement).textContent = M.getFunctionFullName(
item.function,
);
}
List<HTMLElement> _createCalleeHeader() => [
new HTMLDivElement()
..className = 'function-item'
..appendChildren(<HTMLElement>[
_createHeaderButton(
'inclusive',
'Callees(%)',
_Table.callee,
_SortingField.callee,
_SortingDirection.descending,
),
_createHeaderButton(
'name',
'Method',
_Table.callee,
_SortingField.method,
_SortingDirection.ascending,
),
]),
];
HTMLElement _createCaller() {
final element = new HTMLDivElement()
..className = 'function-item'
..appendChildren(<HTMLElement>[
new HTMLSpanElement()
..className = 'inclusive'
..textContent = '0%',
new HTMLSpanElement()..className = 'name',
]);
element.onClick.listen((e) {
if (e.target is HTMLAnchorElement) {
return;
}
_selected = _callers!.getItemFromElement(element);
_r.dirty();
});
return element;
}
void _updateCaller(HTMLElement e, item, int index) {
(e.children.item(0) as HTMLElement).textContent =
Utils.formatPercentNormalized(_getCallerT(item));
(e.children.item(1) as HTMLElement).textContent = M.getFunctionFullName(
item.function,
);
}
List<HTMLElement> _createCallerHeader() => [
new HTMLDivElement()
..className = 'function-item'
..appendChildren(<HTMLElement>[
_createHeaderButton(
'inclusive',
'Callers(%)',
_Table.caller,
_SortingField.caller,
_SortingDirection.descending,
),
_createHeaderButton(
'name',
'Method',
_Table.caller,
_SortingField.method,
_SortingDirection.ascending,
),
]),
];
HTMLButtonElement _createHeaderButton(
String className,
String text,
_Table table,
_SortingField field,
_SortingDirection direction,
) => new HTMLButtonElement()
..className = className
..textContent = _sortingField[table] != field
? text
: _sortingDirection[table] == _SortingDirection.ascending
? '$textâ–¼'
: '$textâ–²'
..onClick.listen((_) => _setSorting(table, field, direction));
List<HTMLElement> _createTree() {
late CpuProfileVirtualTreeElement tree;
return [
(new StackTraceTreeConfigElement(
showMode: false,
showDirection: false,
mode: ProfileTreeMode.function,
direction: M.ProfileTreeDirection.exclusive,
filter: _filter,
queue: _r.queue,
)
..onFilterChange.listen((e) {
_filter = e.element.filter.trim();
tree.filters = _filter.isNotEmpty
? [
_filterTree,
(node) {
return node.name.contains(_filter);
},
]
: [_filterTree];
}))
.element,
new HTMLBRElement(),
(tree =
new CpuProfileVirtualTreeElement(
_isolate,
_progress!.profile,
mode: ProfileTreeMode.function,
direction: M.ProfileTreeDirection.exclusive,
queue: _r.queue,
)
..filters = _filter.isNotEmpty
? [
_filterTree,
(node) {
return node.name.contains(_filter);
},
]
: [_filterTree])
.element,
];
}
bool _filterTree(nodeDynamic) {
M.FunctionCallTreeNode node = nodeDynamic;
return node.profileFunction == _selected;
}
Future _request({bool clear = false, bool forceFetch = false}) async {
_progress = null;
_progressStream = _profiles.get(
isolate,
M.SampleProfileTag.vmOnly,
clear: clear,
forceFetch: forceFetch,
);
_r.dirty();
_progress = (await _progressStream.first).progress;
_r.dirty();
if (M.isSampleProcessRunning(_progress!.status)) {
_progress = (await _progressStream.last).progress;
_r.dirty();
}
}
Future _clearCpuSamples(RefreshEvent e) async {
e.element.disabled = true;
await _request(clear: true);
e.element.disabled = false;
}
Future _refresh(e) async {
e.element.disabled = true;
await _request(forceFetch: true);
e.element.disabled = false;
}
_createSorter(_Table table) {
var getter;
switch (_sortingField[table]!) {
case _SortingField.exclusive:
getter = _getExclusiveT;
break;
case _SortingField.inclusive:
getter = _getInclusiveT;
break;
case _SortingField.callee:
getter = _getCalleeT;
break;
case _SortingField.caller:
getter = _getCallerT;
break;
case _SortingField.method:
getter = (M.ProfileFunction s) => M.getFunctionFullName(s.function!);
break;
}
switch (_sortingDirection[table]!) {
case _SortingDirection.ascending:
int sort(a, b) {
return getter(a).compareTo(getter(b));
}
return sort;
case _SortingDirection.descending:
int sort(a, b) {
return getter(b).compareTo(getter(a));
}
return sort;
}
}
static double _getExclusiveT(M.ProfileFunction f) =>
f.normalizedExclusiveTicks;
static double _getInclusiveT(M.ProfileFunction f) =>
f.normalizedInclusiveTicks;
double _getCalleeT(M.ProfileFunction f) =>
_selected!.callees[f]! /
_selected!.callees.values.reduce((a, b) => a + b);
double _getCallerT(M.ProfileFunction f) =>
_selected!.callers[f]! /
_selected!.callers.values.reduce((a, b) => a + b);
}