perf_tests.dart 14.6 KB
Newer Older
1 2 3 4 5
// Copyright (c) 2016 The Chromium 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';
6
import 'dart:convert' show json;
7
import 'dart:io';
8 9 10

import '../framework/adb.dart';
import '../framework/framework.dart';
11
import '../framework/ios.dart';
12 13
import '../framework/utils.dart';

14
TaskFunction createComplexLayoutScrollPerfTest() {
15 16 17 18
  return new PerfTest(
    '${flutterDirectory.path}/dev/benchmarks/complex_layout',
    'test_driver/scroll_perf.dart',
    'complex_layout_scroll_perf',
19
  ).run;
20 21
}

22
TaskFunction createComplexLayoutScrollMemoryTest() {
23 24 25
  return new MemoryTest(
    '${flutterDirectory.path}/dev/benchmarks/complex_layout',
    'com.yourcompany.complexLayout',
26
    testTarget: 'test_driver/scroll_perf.dart',
27
  ).run;
28 29
}

30
TaskFunction createFlutterGalleryStartupTest() {
31 32
  return new StartupTest(
    '${flutterDirectory.path}/examples/flutter_gallery',
33
  ).run;
34 35
}

36
TaskFunction createComplexLayoutStartupTest() {
37 38
  return new StartupTest(
    '${flutterDirectory.path}/dev/benchmarks/complex_layout',
39
  ).run;
40 41
}

42 43
TaskFunction createFlutterGalleryCompileTest() {
  return new CompileTest('${flutterDirectory.path}/examples/flutter_gallery').run;
44 45
}

46 47
TaskFunction createComplexLayoutCompileTest() {
  return new CompileTest('${flutterDirectory.path}/dev/benchmarks/complex_layout').run;
48 49
}

50 51 52
TaskFunction createHelloWorldMemoryTest() {
  return new MemoryTest(
    '${flutterDirectory.path}/examples/hello_world',
53
    'io.flutter.examples.hello_world',
54
  ).run;
55 56
}

57 58 59
TaskFunction createGalleryNavigationMemoryTest() {
  return new MemoryTest(
    '${flutterDirectory.path}/examples/flutter_gallery',
60
    'io.flutter.demo.gallery',
61
    testTarget: 'test_driver/memory_nav.dart',
62
  ).run;
63 64
}

65 66 67
TaskFunction createGalleryBackButtonMemoryTest() {
  return new AndroidBackButtonMemoryTest(
    '${flutterDirectory.path}/examples/flutter_gallery',
68 69
    'io.flutter.demo.gallery',
    'io.flutter.demo.gallery.MainActivity',
70
  ).run;
71 72
}

73 74 75 76
TaskFunction createFlutterViewStartupTest() {
  return new StartupTest(
      '${flutterDirectory.path}/examples/flutter_view',
      reportMetrics: false,
77 78 79 80 81 82 83 84 85 86 87
  ).run;
}

TaskFunction createBasicMaterialCompileTest() {
  return () async {
    const String sampleAppName = 'sample_flutter_app';
    final Directory sampleDir = dir('${Directory.systemTemp.path}/$sampleAppName');

    if (await sampleDir.exists())
      rmTree(sampleDir);

88 89 90 91 92 93 94
    await inDirectory(Directory.systemTemp, () async {
      await flutter('create', options: <String>[sampleAppName]);
    });

    if (!(await sampleDir.exists()))
      throw 'Failed to create default Flutter app in ${sampleDir.path}';

95 96
    return new CompileTest(sampleDir.path).run();
  };
97 98
}

99

100 101
/// Measure application startup performance.
class StartupTest {
102
  static const Duration _startupTimeout = const Duration(minutes: 5);
103

104
  const StartupTest(this.testDirectory, { this.reportMetrics: true });
105 106

  final String testDirectory;
107
  final bool reportMetrics;
108

109
  Future<TaskResult> run() async {
110
    return await inDirectory(testDirectory, () async {
111
      final String deviceId = (await devices.workingDevice).deviceId;
112
      await flutter('packages', options: <String>['get']);
113

114
      if (deviceOperatingSystem == DeviceOperatingSystem.ios)
115
        await prepareProvisioningCertificates(testDirectory);
116 117

      await flutter('run', options: <String>[
118
        '--verbose',
119 120 121 122 123
        '--profile',
        '--trace-startup',
        '-d',
        deviceId,
      ]).timeout(_startupTimeout);
124
      final Map<String, dynamic> data = json.decode(file('$testDirectory/build/start_up_info.json').readAsStringSync());
125

126 127
      if (!reportMetrics)
        return new TaskResult.success(data);
128

129 130 131 132 133 134 135 136 137 138
      return new TaskResult.success(data, benchmarkScoreKeys: <String>[
        'timeToFirstFrameMicros',
      ]);
    });
  }
}

/// Measures application runtime performance, specifically per-frame
/// performance.
class PerfTest {
139
  const PerfTest(this.testDirectory, this.testTarget, this.timelineFileName);
140 141 142 143 144

  final String testDirectory;
  final String testTarget;
  final String timelineFileName;

145
  Future<TaskResult> run() {
146
    return inDirectory(testDirectory, () async {
147
      final Device device = await devices.workingDevice;
148
      await device.unlock();
149
      final String deviceId = device.deviceId;
150
      await flutter('packages', options: <String>['get']);
151

152
      if (deviceOperatingSystem == DeviceOperatingSystem.ios)
153
        await prepareProvisioningCertificates(testDirectory);
154 155 156 157 158 159 160 161 162 163

      await flutter('drive', options: <String>[
        '-v',
        '--profile',
        '--trace-startup', // Enables "endless" timeline event buffering.
        '-t',
        testTarget,
        '-d',
        deviceId,
      ]);
164
      final Map<String, dynamic> data = json.decode(file('$testDirectory/build/$timelineFileName.timeline_summary.json').readAsStringSync());
165 166 167 168 169 170 171 172

      if (data['frame_count'] < 5) {
        return new TaskResult.failure(
          'Timeline contains too few frames: ${data['frame_count']}. Possibly '
          'trace events are not being captured.',
        );
      }

173 174 175 176
      return new TaskResult.success(data, benchmarkScoreKeys: <String>[
        'average_frame_build_time_millis',
        'worst_frame_build_time_millis',
        'missed_frame_build_budget_count',
177 178 179
        'average_frame_rasterizer_time_millis',
        'worst_frame_rasterizer_time_millis',
        'missed_frame_rasterizer_budget_count',
180 181 182 183 184
      ]);
    });
  }
}

185
/// Measures how long it takes to compile a Flutter app and how big the compiled
186
/// code is.
187 188
class CompileTest {
  const CompileTest(this.testDirectory);
189 190 191

  final String testDirectory;

192
  Future<TaskResult> run() async {
193
    return await inDirectory(testDirectory, () async {
194
      final Device device = await devices.workingDevice;
195
      await device.unlock();
196
      await flutter('packages', options: <String>['get']);
197

198
      final Map<String, dynamic> metrics = <String, dynamic>{}
199 200 201
        ..addAll(await _compileAot())
        ..addAll(await _compileApp())
        ..addAll(await _compileDebug())
202 203 204
        ..addAll(_suffix(await _compileAot(previewDart2: false), '__dart1'))
        ..addAll(_suffix(await _compileApp(previewDart2: false), '__dart1'))
        ..addAll(_suffix(await _compileDebug(previewDart2: false), '__dart1'));
205

206
      return new TaskResult.success(metrics, benchmarkScoreKeys: metrics.keys.toList());
207 208
    });
  }
209

210 211 212 213 214 215 216
  static Map<String, dynamic> _suffix(Map<String, dynamic> map, String suffix) {
    return new Map<String, dynamic>.fromIterables(
      map.keys.map<String>((String key) => '$key$suffix'),
      map.values,
    );
  }

217
  static Future<Map<String, dynamic>> _compileAot({ bool previewDart2: true }) async {
Ian Hickson's avatar
Ian Hickson committed
218
    // Generate blobs instead of assembly.
219
    await flutter('clean');
220
    final Stopwatch watch = new Stopwatch()..start();
221
    final List<String> options = <String>[
222 223 224 225
      'aot',
      '-v',
      '--release',
      '--no-pub',
Ian Hickson's avatar
Ian Hickson committed
226
      '--target-platform',
227
    ];
Ian Hickson's avatar
Ian Hickson committed
228 229 230 231 232 233 234 235
    switch (deviceOperatingSystem) {
      case DeviceOperatingSystem.ios:
        options.add('ios');
        break;
      case DeviceOperatingSystem.android:
        options.add('android-arm');
        break;
    }
236 237
    if (previewDart2)
      options.add('--preview-dart-2');
238 239
    else
      options.add('--no-preview-dart-2');
240
    setLocalEngineOptionIfNecessary(options);
241
    final String compileLog = await evalFlutter('build', options: options);
242 243 244
    watch.stop();

    final RegExp metricExpression = new RegExp(r'([a-zA-Z]+)\(CodeSize\)\: (\d+)');
245 246 247 248
    final Map<String, dynamic> metrics = <String, dynamic>{};
    for (Match m in metricExpression.allMatches(compileLog)) {
      metrics[_sdkNameToMetricName(m.group(1))] = int.parse(m.group(2));
    }
249
    metrics['aot_snapshot_compile_millis'] = watch.elapsedMilliseconds;
250 251 252 253

    return metrics;
  }

254
  static Future<Map<String, dynamic>> _compileApp({ bool previewDart2: true }) async {
255
    await flutter('clean');
256 257 258 259 260
    final Stopwatch watch = new Stopwatch();
    int releaseSizeInBytes;
    final List<String> options = <String>['--release'];
    if (previewDart2)
      options.add('--preview-dart-2');
261 262
    else
      options.add('--no-preview-dart-2');
263
    setLocalEngineOptionIfNecessary(options);
Ian Hickson's avatar
Ian Hickson committed
264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279
    switch (deviceOperatingSystem) {
      case DeviceOperatingSystem.ios:
        options.insert(0, 'ios');
        await prepareProvisioningCertificates(cwd);
        watch.start();
        await flutter('build', options: options);
        watch.stop();
        // IPAs are created manually AFAICT
        await exec('tar', <String>['-zcf', 'build/app.ipa', 'build/ios/Release-iphoneos/Runner.app/']);
        releaseSizeInBytes = await file('$cwd/build/app.ipa').length();
        break;
      case DeviceOperatingSystem.android:
        options.insert(0, 'apk');
        watch.start();
        await flutter('build', options: options);
        watch.stop();
280 281 282 283 284 285
        File apk = file('$cwd/build/app/outputs/apk/app.apk');
        if (!apk.existsSync()) {
          // Pre Android SDK 26 path
          apk = file('$cwd/build/app/outputs/apk/app-release.apk');
        }
        releaseSizeInBytes = apk.lengthSync();
Ian Hickson's avatar
Ian Hickson committed
286
        break;
287
    }
288

289 290 291 292 293 294
    return <String, dynamic>{
      'release_full_compile_millis': watch.elapsedMilliseconds,
      'release_size_bytes': releaseSizeInBytes,
    };
  }

295
  static Future<Map<String, dynamic>> _compileDebug({ bool previewDart2: true }) async {
296
    await flutter('clean');
297
    final Stopwatch watch = new Stopwatch();
Ian Hickson's avatar
Ian Hickson committed
298 299 300
    final List<String> options = <String>['--debug'];
    if (previewDart2)
      options.add('--preview-dart-2');
301 302
    else
      options.add('--no-preview-dart-2');
303
    setLocalEngineOptionIfNecessary(options);
Ian Hickson's avatar
Ian Hickson committed
304 305 306 307 308 309 310 311
    switch (deviceOperatingSystem) {
      case DeviceOperatingSystem.ios:
        options.insert(0, 'ios');
        await prepareProvisioningCertificates(cwd);
        break;
      case DeviceOperatingSystem.android:
        options.insert(0, 'apk');
        break;
312
    }
Ian Hickson's avatar
Ian Hickson committed
313 314 315
    watch.start();
    await flutter('build', options: options);
    watch.stop();
316 317

    return <String, dynamic>{
318
      'debug_full_compile_millis': watch.elapsedMilliseconds,
319 320 321
    };
  }

322 323 324 325 326 327 328 329 330 331 332 333 334 335
  static String _sdkNameToMetricName(String sdkName) {
    const Map<String, String> kSdkNameToMetricNameMapping = const <String, String> {
      'VMIsolate': 'aot_snapshot_size_vmisolate',
      'Isolate': 'aot_snapshot_size_isolate',
      'ReadOnlyData': 'aot_snapshot_size_rodata',
      'Instructions': 'aot_snapshot_size_instructions',
      'Total': 'aot_snapshot_size_total',
    };

    if (!kSdkNameToMetricNameMapping.containsKey(sdkName))
      throw 'Unrecognized SDK snapshot metric name: $sdkName';

    return kSdkNameToMetricNameMapping[sdkName];
  }
336
}
337

338
/// Measure application memory usage.
339
class MemoryTest {
340
  const MemoryTest(this.testDirectory, this.packageName, { this.testTarget });
341 342 343 344

  final String testDirectory;
  final String packageName;

345 346 347 348 349
  /// Path to a flutter driver script that will run after starting the app.
  ///
  /// If not specified, then the test will start the app, gather statistics, and then exit.
  final String testTarget;

350
  Future<TaskResult> run() {
351
    return inDirectory(testDirectory, () async {
352
      final Device device = await devices.workingDevice;
353
      await device.unlock();
354
      final String deviceId = device.deviceId;
355 356
      await flutter('packages', options: <String>['get']);

357
      if (deviceOperatingSystem == DeviceOperatingSystem.ios)
358
        await prepareProvisioningCertificates(testDirectory);
359

360
      final int observatoryPort = await findAvailablePort();
361

362
      final List<String> runOptions = <String>[
363 364 365 366 367
        '-v',
        '--profile',
        '--trace-startup', // wait for the first frame to render
        '-d',
        deviceId,
368 369
        '--observatory-port',
        observatoryPort.toString(),
370 371 372 373
      ];
      if (testTarget != null)
        runOptions.addAll(<String>['-t', testTarget]);
      await flutter('run', options: runOptions);
374

375
      final Map<String, dynamic> startData = await device.getMemoryStats(packageName);
376

377
      final Map<String, dynamic> data = <String, dynamic>{
378 379 380
         'start_total_kb': startData['total_kb'],
      };

381 382 383 384 385 386 387
      if (testTarget != null) {
        await flutter('drive', options: <String>[
          '-v',
          '-t',
          testTarget,
          '-d',
          deviceId,
388 389
          '--use-existing-app=http://localhost:$observatoryPort',
        ]);
390

391
        final Map<String, dynamic> endData = await device.getMemoryStats(packageName);
392 393 394 395
        data['end_total_kb'] = endData['total_kb'];
        data['diff_total_kb'] = endData['total_kb'] - startData['total_kb'];
      }

396 397
      await device.stop(packageName);

398
      return new TaskResult.success(data, benchmarkScoreKeys: data.keys.toList());
399 400 401
    });
  }
}
402 403 404 405

/// Measure application memory usage after pausing and resuming the app
/// with the Android back button.
class AndroidBackButtonMemoryTest {
406 407
  const AndroidBackButtonMemoryTest(this.testDirectory, this.packageName, this.activityName);

408 409
  final String testDirectory;
  final String packageName;
410
  final String activityName;
411

412
  Future<TaskResult> run() {
413 414 415 416 417
    return inDirectory(testDirectory, () async {
      if (deviceOperatingSystem != DeviceOperatingSystem.android) {
        throw 'This test is only supported on Android';
      }

418
      final AndroidDevice device = await devices.workingDevice;
419
      await device.unlock();
420
      final String deviceId = device.deviceId;
421 422 423 424 425 426 427 428 429 430
      await flutter('packages', options: <String>['get']);

      await flutter('run', options: <String>[
        '-v',
        '--profile',
        '--trace-startup', // wait for the first frame to render
        '-d',
        deviceId,
      ]);

431
      final Map<String, dynamic> startData = await device.getMemoryStats(packageName);
432

433
      final Map<String, dynamic> data = <String, dynamic>{
434 435 436 437 438
         'start_total_kb': startData['total_kb'],
      };

      // Perform a series of back button suspend and resume cycles.
      for (int i = 0; i < 10; i++) {
439
        await device.shellExec('input', <String>['keyevent', 'KEYCODE_BACK']);
440
        await new Future<Null>.delayed(const Duration(milliseconds: 1000));
441
        final String output = await device.shellEval('am', <String>['start', '-n', '$packageName/$activityName']);
442 443 444
        print(output);
        if (output.contains('Error'))
          return new TaskResult.failure('unable to launch activity');
445
        await new Future<Null>.delayed(const Duration(milliseconds: 1000));
446 447
      }

448
      final Map<String, dynamic> endData = await device.getMemoryStats(packageName);
449 450 451 452 453 454 455 456 457
      data['end_total_kb'] = endData['total_kb'];
      data['diff_total_kb'] = endData['total_kb'] - startData['total_kb'];

      await device.stop(packageName);

      return new TaskResult.success(data, benchmarkScoreKeys: data.keys.toList());
    });
  }
}