|  | // Copyright 2014 The Flutter Authors. 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:convert' show JsonEncoder, LineSplitter, json, utf8; | 
|  | import 'dart:io' as io; | 
|  | import 'dart:math' as math; | 
|  |  | 
|  | import 'package:path/path.dart' as path; | 
|  | import 'package:webkit_inspection_protocol/webkit_inspection_protocol.dart'; | 
|  |  | 
|  | /// The number of samples used to extract metrics, such as noise, means, | 
|  | /// max/min values. | 
|  | /// | 
|  | /// Keep this constant in sync with the same constant defined in `dev/benchmarks/macrobenchmarks/lib/src/web/recorder.dart`. | 
|  | const int _kMeasuredSampleCount = 10; | 
|  |  | 
|  | /// Options passed to Chrome when launching it. | 
|  | class ChromeOptions { | 
|  | ChromeOptions({ | 
|  | this.userDataDirectory, | 
|  | this.url, | 
|  | this.windowWidth = 1024, | 
|  | this.windowHeight = 1024, | 
|  | this.headless, | 
|  | this.debugPort, | 
|  | this.enableWasmGC = false, | 
|  | }); | 
|  |  | 
|  | /// If not null passed as `--user-data-dir`. | 
|  | final String? userDataDirectory; | 
|  |  | 
|  | /// If not null launches a Chrome tab at this URL. | 
|  | final String? url; | 
|  |  | 
|  | /// The width of the Chrome window. | 
|  | /// | 
|  | /// This is important for screenshots and benchmarks. | 
|  | final int windowWidth; | 
|  |  | 
|  | /// The height of the Chrome window. | 
|  | /// | 
|  | /// This is important for screenshots and benchmarks. | 
|  | final int windowHeight; | 
|  |  | 
|  | /// Launches code in "headless" mode, which allows running Chrome in | 
|  | /// environments without a display, such as LUCI and Cirrus. | 
|  | final bool? headless; | 
|  |  | 
|  | /// The port Chrome will use for its debugging protocol. | 
|  | /// | 
|  | /// If null, Chrome is launched without debugging. When running in headless | 
|  | /// mode without a debug port, Chrome quits immediately. For most tests it is | 
|  | /// typical to set [headless] to true and set a non-null debug port. | 
|  | final int? debugPort; | 
|  |  | 
|  | /// Whether to enable experimental WasmGC flags | 
|  | final bool enableWasmGC; | 
|  | } | 
|  |  | 
|  | /// A function called when the Chrome process encounters an error. | 
|  | typedef ChromeErrorCallback = void Function(String); | 
|  |  | 
|  | /// Manages a single Chrome process. | 
|  | class Chrome { | 
|  | Chrome._(this._chromeProcess, this._onError, this._debugConnection) { | 
|  | // If the Chrome process quits before it was asked to quit, notify the | 
|  | // error listener. | 
|  | _chromeProcess.exitCode.then((int exitCode) { | 
|  | if (!_isStopped) { | 
|  | _onError('Chrome process exited prematurely with exit code $exitCode'); | 
|  | } | 
|  | }); | 
|  | } | 
|  |  | 
|  | /// Launches Chrome with the give [options]. | 
|  | /// | 
|  | /// The [onError] callback is called with an error message when the Chrome | 
|  | /// process encounters an error. In particular, [onError] is called when the | 
|  | /// Chrome process exits prematurely, i.e. before [stop] is called. | 
|  | static Future<Chrome> launch(ChromeOptions options, { String? workingDirectory, required ChromeErrorCallback onError }) async { | 
|  | if (!io.Platform.isWindows) { | 
|  | final io.ProcessResult versionResult = io.Process.runSync(_findSystemChromeExecutable(), const <String>['--version']); | 
|  | print('Launching ${versionResult.stdout}'); | 
|  | } else { | 
|  | print('Launching Chrome...'); | 
|  | } | 
|  |  | 
|  | final String jsFlags = options.enableWasmGC ? <String>[ | 
|  | '--experimental-wasm-gc', | 
|  | '--experimental-wasm-type-reflection', | 
|  | ].join(' ') : ''; | 
|  | final bool withDebugging = options.debugPort != null; | 
|  | final List<String> args = <String>[ | 
|  | if (options.userDataDirectory != null) | 
|  | '--user-data-dir=${options.userDataDirectory}', | 
|  | if (options.url != null) | 
|  | options.url!, | 
|  | if (io.Platform.environment['CHROME_NO_SANDBOX'] == 'true') | 
|  | '--no-sandbox', | 
|  | if (options.headless ?? false) | 
|  | '--headless', | 
|  | if (withDebugging) | 
|  | '--remote-debugging-port=${options.debugPort}', | 
|  | '--window-size=${options.windowWidth},${options.windowHeight}', | 
|  | '--disable-extensions', | 
|  | '--disable-popup-blocking', | 
|  | // Indicates that the browser is in "browse without sign-in" (Guest session) mode. | 
|  | '--bwsi', | 
|  | '--no-first-run', | 
|  | '--no-default-browser-check', | 
|  | '--disable-default-apps', | 
|  | '--disable-translate', | 
|  | if (jsFlags.isNotEmpty) '--js-flags=$jsFlags', | 
|  | ]; | 
|  |  | 
|  | final io.Process chromeProcess = await _spawnChromiumProcess( | 
|  | _findSystemChromeExecutable(), | 
|  | args, | 
|  | workingDirectory: workingDirectory, | 
|  | ); | 
|  |  | 
|  | WipConnection? debugConnection; | 
|  | if (withDebugging) { | 
|  | debugConnection = await _connectToChromeDebugPort(chromeProcess, options.debugPort!); | 
|  | } | 
|  |  | 
|  | return Chrome._(chromeProcess, onError, debugConnection); | 
|  | } | 
|  |  | 
|  | final io.Process _chromeProcess; | 
|  | final ChromeErrorCallback _onError; | 
|  | final WipConnection? _debugConnection; | 
|  | bool _isStopped = false; | 
|  |  | 
|  | Completer<void> ?_tracingCompleter; | 
|  | StreamSubscription<WipEvent>? _tracingSubscription; | 
|  | List<Map<String, dynamic>>? _tracingData; | 
|  |  | 
|  | /// Starts recording a performance trace. | 
|  | /// | 
|  | /// If there is already a tracing session in progress, throws an error. Call | 
|  | /// [endRecordingPerformance] before starting a new tracing session. | 
|  | /// | 
|  | /// The [label] is for debugging convenience. | 
|  | Future<void> beginRecordingPerformance(String label) async { | 
|  | if (_tracingCompleter != null) { | 
|  | throw StateError( | 
|  | 'Cannot start a new performance trace. A tracing session labeled ' | 
|  | '"$label" is already in progress.' | 
|  | ); | 
|  | } | 
|  | _tracingCompleter = Completer<void>(); | 
|  | _tracingData = <Map<String, dynamic>>[]; | 
|  |  | 
|  | // Subscribe to tracing events prior to calling "Tracing.start". Otherwise, | 
|  | // we'll miss tracing data. | 
|  | _tracingSubscription = _debugConnection?.onNotification.listen((WipEvent event) { | 
|  | // We receive data as a sequence of "Tracing.dataCollected" followed by | 
|  | // "Tracing.tracingComplete" at the end. Until "Tracing.tracingComplete" | 
|  | // is received, the data may be incomplete. | 
|  | if (event.method == 'Tracing.tracingComplete') { | 
|  | _tracingCompleter!.complete(); | 
|  | _tracingSubscription!.cancel(); | 
|  | _tracingSubscription = null; | 
|  | } else if (event.method == 'Tracing.dataCollected') { | 
|  | final dynamic value = event.params?['value']; | 
|  | if (value is! List) { | 
|  | throw FormatException('"Tracing.dataCollected" returned malformed data. ' | 
|  | 'Expected a List but got: ${value.runtimeType}'); | 
|  | } | 
|  | _tracingData?.addAll((event.params?['value'] as List<dynamic>).cast<Map<String, dynamic>>()); | 
|  | } | 
|  | }); | 
|  | await _debugConnection?.sendCommand('Tracing.start', <String, dynamic>{ | 
|  | // The choice of categories is as follows: | 
|  | // | 
|  | // blink: | 
|  | //   provides everything on the UI thread, including scripting, | 
|  | //   style recalculations, layout, painting, and some compositor | 
|  | //   work. | 
|  | // blink.user_timing: | 
|  | //   provides marks recorded using window.performance. We use marks | 
|  | //   to find frames that the benchmark cares to measure. | 
|  | // gpu: | 
|  | //   provides tracing data from the GPU data | 
|  | //   disabled due to https://bugs.chromium.org/p/chromium/issues/detail?id=1068259 | 
|  | // TODO(yjbanov): extract useful GPU data | 
|  | 'categories': 'blink,blink.user_timing', | 
|  | 'transferMode': 'SendAsStream', | 
|  | }); | 
|  | } | 
|  |  | 
|  | /// Stops a performance tracing session started by [beginRecordingPerformance]. | 
|  | /// | 
|  | /// Returns all the collected tracing data unfiltered. | 
|  | Future<List<Map<String, dynamic>>?> endRecordingPerformance() async { | 
|  | await _debugConnection!.sendCommand('Tracing.end'); | 
|  | await _tracingCompleter!.future; | 
|  | final List<Map<String, dynamic>>? data = _tracingData; | 
|  | _tracingCompleter = null; | 
|  | _tracingData = null; | 
|  | return data; | 
|  | } | 
|  |  | 
|  | Future<void> reloadPage({bool ignoreCache = false}) async { | 
|  | await _debugConnection?.page.reload(ignoreCache: ignoreCache); | 
|  | } | 
|  |  | 
|  | /// Stops the Chrome process. | 
|  | void stop() { | 
|  | _isStopped = true; | 
|  | _tracingSubscription?.cancel(); | 
|  | _chromeProcess.kill(); | 
|  | } | 
|  | } | 
|  |  | 
|  | String _findSystemChromeExecutable() { | 
|  | // On some environments, such as the Dart HHH tester, Chrome resides in a | 
|  | // non-standard location and is provided via the following environment | 
|  | // variable. | 
|  | final String? envExecutable = io.Platform.environment['CHROME_EXECUTABLE']; | 
|  | if (envExecutable != null) { | 
|  | return envExecutable; | 
|  | } | 
|  |  | 
|  | if (io.Platform.isLinux) { | 
|  | final io.ProcessResult which = | 
|  | io.Process.runSync('which', <String>['google-chrome']); | 
|  |  | 
|  | if (which.exitCode != 0) { | 
|  | throw Exception('Failed to locate system Chrome installation.'); | 
|  | } | 
|  |  | 
|  | return (which.stdout as String).trim(); | 
|  | } else if (io.Platform.isMacOS) { | 
|  | return '/Applications/Google Chrome.app/Contents/MacOS/Google Chrome'; | 
|  | } else if (io.Platform.isWindows) { | 
|  | const String kWindowsExecutable = r'Google\Chrome\Application\chrome.exe'; | 
|  | final List<String> kWindowsPrefixes = <String?>[ | 
|  | io.Platform.environment['LOCALAPPDATA'], | 
|  | io.Platform.environment['PROGRAMFILES'], | 
|  | io.Platform.environment['PROGRAMFILES(X86)'], | 
|  | ].whereType<String>().toList(); | 
|  | final String windowsPrefix = kWindowsPrefixes.firstWhere((String prefix) { | 
|  | final String expectedPath = path.join(prefix, kWindowsExecutable); | 
|  | return io.File(expectedPath).existsSync(); | 
|  | }, orElse: () => '.'); | 
|  | return path.join(windowsPrefix, kWindowsExecutable); | 
|  | } else { | 
|  | throw Exception('Web benchmarks cannot run on ${io.Platform.operatingSystem}.'); | 
|  | } | 
|  | } | 
|  |  | 
|  | /// Waits for Chrome to print DevTools URI and connects to it. | 
|  | Future<WipConnection> _connectToChromeDebugPort(io.Process chromeProcess, int port) async { | 
|  | final Uri devtoolsUri = await _getRemoteDebuggerUrl(Uri.parse('http://localhost:$port')); | 
|  | print('Connecting to DevTools: $devtoolsUri'); | 
|  | final ChromeConnection chromeConnection = ChromeConnection('localhost', port); | 
|  | final Iterable<ChromeTab> tabs = (await chromeConnection.getTabs()).where((ChromeTab tab) { | 
|  | return tab.url.startsWith('http://localhost'); | 
|  | }); | 
|  | final ChromeTab tab = tabs.single; | 
|  | final WipConnection debugConnection = await tab.connect(); | 
|  | print('Connected to Chrome tab: ${tab.title} (${tab.url})'); | 
|  | return debugConnection; | 
|  | } | 
|  |  | 
|  | /// Gets the Chrome debugger URL for the web page being benchmarked. | 
|  | Future<Uri> _getRemoteDebuggerUrl(Uri base) async { | 
|  | final io.HttpClient client = io.HttpClient(); | 
|  | final io.HttpClientRequest request = await client.getUrl(base.resolve('/json/list')); | 
|  | final io.HttpClientResponse response = await request.close(); | 
|  | final List<dynamic>? jsonObject = await json.fuse(utf8).decoder.bind(response).single as List<dynamic>?; | 
|  | if (jsonObject == null || jsonObject.isEmpty) { | 
|  | return base; | 
|  | } | 
|  | return base.resolve((jsonObject.first as Map<String, dynamic>)['webSocketDebuggerUrl'] as String); | 
|  | } | 
|  |  | 
|  | /// Summarizes a Blink trace down to a few interesting values. | 
|  | class BlinkTraceSummary { | 
|  | BlinkTraceSummary._({ | 
|  | required this.averageBeginFrameTime, | 
|  | required this.averageUpdateLifecyclePhasesTime, | 
|  | }) : averageTotalUIFrameTime = averageBeginFrameTime + averageUpdateLifecyclePhasesTime; | 
|  |  | 
|  | static BlinkTraceSummary? fromJson(List<Map<String, dynamic>> traceJson) { | 
|  | try { | 
|  | // Convert raw JSON data to BlinkTraceEvent objects sorted by timestamp. | 
|  | List<BlinkTraceEvent> events = traceJson | 
|  | .map<BlinkTraceEvent>(BlinkTraceEvent.fromJson) | 
|  | .toList() | 
|  | ..sort((BlinkTraceEvent a, BlinkTraceEvent b) => a.ts! - b.ts!); | 
|  |  | 
|  | Exception noMeasuredFramesFound() => Exception( | 
|  | 'No measured frames found in benchmark tracing data. This likely ' | 
|  | 'indicates a bug in the benchmark. For example, the benchmark failed ' | 
|  | "to pump enough frames. It may also indicate a change in Chrome's " | 
|  | 'tracing data format. Check if Chrome version changed recently and ' | 
|  | 'adjust the parsing code accordingly.', | 
|  | ); | 
|  |  | 
|  | // Use the pid from the first "measured_frame" event since the event is | 
|  | // emitted by the script running on the process we're interested in. | 
|  | // | 
|  | // We previously tried using the "CrRendererMain" event. However, for | 
|  | // reasons unknown, Chrome in the devicelab refuses to emit this event | 
|  | // sometimes, causing to flakes. | 
|  | final BlinkTraceEvent firstMeasuredFrameEvent = events.firstWhere( | 
|  | (BlinkTraceEvent event) => event.isBeginMeasuredFrame, | 
|  | orElse: () => throw noMeasuredFramesFound(), | 
|  | ); | 
|  |  | 
|  | final int tabPid = firstMeasuredFrameEvent.pid!; | 
|  |  | 
|  | // Filter out data from unrelated processes | 
|  | events = events.where((BlinkTraceEvent element) => element.pid == tabPid).toList(); | 
|  |  | 
|  | // Extract frame data. | 
|  | final List<BlinkFrame> frames = <BlinkFrame>[]; | 
|  | int skipCount = 0; | 
|  | BlinkFrame frame = BlinkFrame(); | 
|  | for (final BlinkTraceEvent event in events) { | 
|  | if (event.isBeginFrame) { | 
|  | frame.beginFrame = event; | 
|  | } else if (event.isUpdateAllLifecyclePhases) { | 
|  | frame.updateAllLifecyclePhases = event; | 
|  | if (frame.endMeasuredFrame != null) { | 
|  | frames.add(frame); | 
|  | } else { | 
|  | skipCount += 1; | 
|  | } | 
|  | frame = BlinkFrame(); | 
|  | } else if (event.isBeginMeasuredFrame) { | 
|  | frame.beginMeasuredFrame = event; | 
|  | } else if (event.isEndMeasuredFrame) { | 
|  | frame.endMeasuredFrame = event; | 
|  | } | 
|  | } | 
|  |  | 
|  | print('Extracted ${frames.length} measured frames.'); | 
|  | print('Skipped $skipCount non-measured frames.'); | 
|  |  | 
|  | if (frames.isEmpty) { | 
|  | throw noMeasuredFramesFound(); | 
|  | } | 
|  |  | 
|  | // Compute averages and summarize. | 
|  | return BlinkTraceSummary._( | 
|  | averageBeginFrameTime: _computeAverageDuration(frames.map((BlinkFrame frame) => frame.beginFrame).whereType<BlinkTraceEvent>().toList()), | 
|  | averageUpdateLifecyclePhasesTime: _computeAverageDuration(frames.map((BlinkFrame frame) => frame.updateAllLifecyclePhases).whereType<BlinkTraceEvent>().toList()), | 
|  | ); | 
|  | } catch (_) { | 
|  | final io.File traceFile = io.File('./chrome-trace.json'); | 
|  | io.stderr.writeln('Failed to interpret the Chrome trace contents. The trace was saved in ${traceFile.path}'); | 
|  | traceFile.writeAsStringSync(const JsonEncoder.withIndent('  ').convert(traceJson)); | 
|  | rethrow; | 
|  | } | 
|  | } | 
|  |  | 
|  | /// The average duration of "WebViewImpl::beginFrame" events. | 
|  | /// | 
|  | /// This event contains all of scripting time of an animation frame, plus an | 
|  | /// unknown small amount of work browser does before and after scripting. | 
|  | final Duration averageBeginFrameTime; | 
|  |  | 
|  | /// The average duration of "WebViewImpl::updateAllLifecyclePhases" events. | 
|  | /// | 
|  | /// This event contains style, layout, painting, and compositor computations, | 
|  | /// which are not included in the scripting time. This event does not | 
|  | /// include GPU time, which happens on a separate thread. | 
|  | final Duration averageUpdateLifecyclePhasesTime; | 
|  |  | 
|  | /// The average sum of [averageBeginFrameTime] and | 
|  | /// [averageUpdateLifecyclePhasesTime]. | 
|  | /// | 
|  | /// This value contains the vast majority of work the UI thread performs in | 
|  | /// any given animation frame. | 
|  | final Duration averageTotalUIFrameTime; | 
|  |  | 
|  | @override | 
|  | String toString() => '$BlinkTraceSummary(' | 
|  | 'averageBeginFrameTime: ${averageBeginFrameTime.inMicroseconds / 1000}ms, ' | 
|  | 'averageUpdateLifecyclePhasesTime: ${averageUpdateLifecyclePhasesTime.inMicroseconds / 1000}ms)'; | 
|  | } | 
|  |  | 
|  | /// Contains events pertaining to a single frame in the Blink trace data. | 
|  | class BlinkFrame { | 
|  | /// Corresponds to 'WebViewImpl::beginFrame' event. | 
|  | BlinkTraceEvent? beginFrame; | 
|  |  | 
|  | /// Corresponds to 'WebViewImpl::updateAllLifecyclePhases' event. | 
|  | BlinkTraceEvent? updateAllLifecyclePhases; | 
|  |  | 
|  | /// Corresponds to 'measured_frame' begin event. | 
|  | BlinkTraceEvent? beginMeasuredFrame; | 
|  |  | 
|  | /// Corresponds to 'measured_frame' end event. | 
|  | BlinkTraceEvent? endMeasuredFrame; | 
|  | } | 
|  |  | 
|  | /// Takes a list of events that have non-null [BlinkTraceEvent.tdur] computes | 
|  | /// their average as a [Duration] value. | 
|  | Duration _computeAverageDuration(List<BlinkTraceEvent> events) { | 
|  | // Compute the sum of "tdur" fields of the last _kMeasuredSampleCount events. | 
|  | final double sum = events | 
|  | .skip(math.max(events.length - _kMeasuredSampleCount, 0)) | 
|  | .fold(0.0, (double previousValue, BlinkTraceEvent event) { | 
|  | if (event.tdur == null) { | 
|  | throw FormatException('Trace event lacks "tdur" field: $event'); | 
|  | } | 
|  | return previousValue + event.tdur!; | 
|  | }); | 
|  | final int sampleCount = math.min(events.length, _kMeasuredSampleCount); | 
|  | return Duration(microseconds: sum ~/ sampleCount); | 
|  | } | 
|  |  | 
|  | /// An event collected by the Blink tracer (in Chrome accessible using chrome://tracing). | 
|  | /// | 
|  | /// See also: | 
|  | ///  * https://docs.google.com/document/d/1CvAClvFfyA5R-PhYUmn5OOQtYMH4h6I0nSsKchNAySU/preview | 
|  | class BlinkTraceEvent { | 
|  | /// Parses an event from its JSON representation. | 
|  | /// | 
|  | /// Sample event encoded as JSON (the data is bogus, this just shows the format): | 
|  | /// | 
|  | /// ```json | 
|  | /// { | 
|  | ///   "name": "myName", | 
|  | ///   "cat": "category,list", | 
|  | ///   "ph": "B", | 
|  | ///   "ts": 12345, | 
|  | ///   "pid": 123, | 
|  | ///   "tid": 456, | 
|  | ///   "args": { | 
|  | ///     "someArg": 1, | 
|  | ///     "anotherArg": { | 
|  | ///       "value": "my value" | 
|  | ///     } | 
|  | ///   } | 
|  | /// } | 
|  | /// ``` | 
|  | /// | 
|  | /// For detailed documentation of the format see: | 
|  | /// | 
|  | /// https://docs.google.com/document/d/1CvAClvFfyA5R-PhYUmn5OOQtYMH4h6I0nSsKchNAySU/preview | 
|  | BlinkTraceEvent.fromJson(Map<String, dynamic> json) | 
|  | : args = json['args'] as Map<String, dynamic>, | 
|  | cat = json['cat'] as String, | 
|  | name = json['name'] as String, | 
|  | ph = json['ph'] as String, | 
|  | pid = _readInt(json, 'pid'), | 
|  | tid = _readInt(json, 'tid'), | 
|  | ts = _readInt(json, 'ts'), | 
|  | tts = _readInt(json, 'tts'), | 
|  | tdur = _readInt(json, 'tdur'); | 
|  |  | 
|  | /// Event-specific data. | 
|  | final Map<String, dynamic> args; | 
|  |  | 
|  | /// Event category. | 
|  | final String cat; | 
|  |  | 
|  | /// Event name. | 
|  | final String name; | 
|  |  | 
|  | /// Event "phase". | 
|  | final String ph; | 
|  |  | 
|  | /// Process ID of the process that emitted the event. | 
|  | final int? pid; | 
|  |  | 
|  | /// Thread ID of the thread that emitted the event. | 
|  | final int? tid; | 
|  |  | 
|  | /// Timestamp in microseconds using tracer clock. | 
|  | final int? ts; | 
|  |  | 
|  | /// Timestamp in microseconds using thread clock. | 
|  | final int? tts; | 
|  |  | 
|  | /// Event duration in microseconds. | 
|  | final int? tdur; | 
|  |  | 
|  | /// A "begin frame" event contains all of the scripting time of an animation | 
|  | /// frame (JavaScript, WebAssembly), plus a negligible amount of internal | 
|  | /// browser overhead. | 
|  | /// | 
|  | /// This event does not include non-UI thread scripting, such as web workers, | 
|  | /// service workers, and CSS Paint paintlets. | 
|  | /// | 
|  | /// WebViewImpl::beginFrame was used in earlier versions of Chrome, kept | 
|  | /// for compatibility. | 
|  | /// | 
|  | /// This event is a duration event that has its `tdur` populated. | 
|  | bool get isBeginFrame { | 
|  | return ph == 'X' && ( | 
|  | name == 'WebViewImpl::beginFrame' || | 
|  | name == 'WebFrameWidgetBase::BeginMainFrame' || | 
|  | name == 'WebFrameWidgetImpl::BeginMainFrame' | 
|  | ); | 
|  | } | 
|  |  | 
|  | /// An "update all lifecycle phases" event contains UI thread computations | 
|  | /// related to an animation frame that's outside the scripting phase. | 
|  | /// | 
|  | /// This event includes style recalculation, layer tree update, layout, | 
|  | /// painting, and parts of compositing work. | 
|  | /// | 
|  | /// WebViewImpl::updateAllLifecyclePhases was used in earlier versions of | 
|  | /// Chrome, kept for compatibility. | 
|  | /// | 
|  | /// This event is a duration event that has its `tdur` populated. | 
|  | bool get isUpdateAllLifecyclePhases { | 
|  | return ph == 'X' && ( | 
|  | name == 'WebViewImpl::updateAllLifecyclePhases' || | 
|  | name == 'WebFrameWidgetImpl::UpdateLifecycle' | 
|  | ); | 
|  | } | 
|  |  | 
|  | /// Whether this is the beginning of a "measured_frame" event. | 
|  | /// | 
|  | /// This event is a custom event emitted by our benchmark test harness. | 
|  | /// | 
|  | /// See also: | 
|  | ///  * `recorder.dart`, which emits this event. | 
|  | bool get isBeginMeasuredFrame => ph == 'b' && name == 'measured_frame'; | 
|  |  | 
|  | /// Whether this is the end of a "measured_frame" event. | 
|  | /// | 
|  | /// This event is a custom event emitted by our benchmark test harness. | 
|  | /// | 
|  | /// See also: | 
|  | ///  * `recorder.dart`, which emits this event. | 
|  | bool get isEndMeasuredFrame => ph == 'e' && name == 'measured_frame'; | 
|  |  | 
|  | @override | 
|  | String toString() => '$BlinkTraceEvent(' | 
|  | 'args: ${json.encode(args)}, ' | 
|  | 'cat: $cat, ' | 
|  | 'name: $name, ' | 
|  | 'ph: $ph, ' | 
|  | 'pid: $pid, ' | 
|  | 'tid: $tid, ' | 
|  | 'ts: $ts, ' | 
|  | 'tts: $tts, ' | 
|  | 'tdur: $tdur)'; | 
|  | } | 
|  |  | 
|  | /// Read an integer out of [json] stored under [key]. | 
|  | /// | 
|  | /// Since JSON does not distinguish between `int` and `double`, extra | 
|  | /// validation and conversion is needed. | 
|  | /// | 
|  | /// Returns null if the value is null. | 
|  | int? _readInt(Map<String, dynamic> json, String key) { | 
|  | final num? jsonValue = json[key] as num?; | 
|  | return jsonValue?.toInt(); | 
|  | } | 
|  |  | 
|  | /// Used by [Chrome.launch] to detect a glibc bug and retry launching the | 
|  | /// browser. | 
|  | /// | 
|  | /// Once every few thousands of launches we hit this glibc bug: | 
|  | /// | 
|  | /// https://sourceware.org/bugzilla/show_bug.cgi?id=19329. | 
|  | /// | 
|  | /// When this happens Chrome spits out something like the following then exits with code 127: | 
|  | /// | 
|  | ///     Inconsistency detected by ld.so: ../elf/dl-tls.c: 493: _dl_allocate_tls_init: Assertion `listp->slotinfo[cnt].gen <= GL(dl_tls_generation)' failed! | 
|  | const String _kGlibcError = 'Inconsistency detected by ld.so'; | 
|  |  | 
|  | Future<io.Process> _spawnChromiumProcess(String executable, List<String> args, { String? workingDirectory }) async { | 
|  | // Keep attempting to launch the browser until one of: | 
|  | // - Chrome launched successfully, in which case we just return from the loop. | 
|  | // - The tool detected an unretryable Chrome error, in which case we throw ToolExit. | 
|  | while (true) { | 
|  | final io.Process process = await io.Process.start(executable, args, workingDirectory: workingDirectory); | 
|  |  | 
|  | process.stdout | 
|  | .transform(utf8.decoder) | 
|  | .transform(const LineSplitter()) | 
|  | .listen((String line) { | 
|  | print('[CHROME STDOUT]: $line'); | 
|  | }); | 
|  |  | 
|  | // Wait until the DevTools are listening before trying to connect. This is | 
|  | // only required for flutter_test --platform=chrome and not flutter run. | 
|  | bool hitGlibcBug = false; | 
|  | await process.stderr | 
|  | .transform(utf8.decoder) | 
|  | .transform(const LineSplitter()) | 
|  | .map((String line) { | 
|  | print('[CHROME STDERR]:$line'); | 
|  | if (line.contains(_kGlibcError)) { | 
|  | hitGlibcBug = true; | 
|  | } | 
|  | return line; | 
|  | }) | 
|  | .firstWhere((String line) => line.startsWith('DevTools listening'), orElse: () { | 
|  | if (hitGlibcBug) { | 
|  | print( | 
|  | 'Encountered glibc bug https://sourceware.org/bugzilla/show_bug.cgi?id=19329. ' | 
|  | 'Will try launching browser again.', | 
|  | ); | 
|  | return ''; | 
|  | } | 
|  | print('Failed to launch browser. Command used to launch it: ${args.join(' ')}'); | 
|  | throw Exception( | 
|  | 'Failed to launch browser. Make sure you are using an up-to-date ' | 
|  | 'Chrome or Edge. Otherwise, consider using -d web-server instead ' | 
|  | 'and filing an issue at https://github.com/flutter/flutter/issues.', | 
|  | ); | 
|  | }); | 
|  |  | 
|  | if (!hitGlibcBug) { | 
|  | return process; | 
|  | } | 
|  |  | 
|  | // A precaution that avoids accumulating browser processes, in case the | 
|  | // glibc bug doesn't cause the browser to quit and we keep looping and | 
|  | // launching more processes. | 
|  | unawaited(process.exitCode.timeout(const Duration(seconds: 1), onTimeout: () { | 
|  | process.kill(); | 
|  | return 0; | 
|  | })); | 
|  | } | 
|  | } |