1 /* 2 * Copyright (C) 2013 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 package com.android.tests.applaunch; 17 18 import static org.junit.Assert.assertNotNull; 19 20 import android.accounts.Account; 21 import android.accounts.AccountManager; 22 import android.app.ActivityManager; 23 import android.app.ActivityManager.ProcessErrorStateInfo; 24 import android.app.IActivityManager; 25 import android.app.UiAutomation; 26 import android.content.Context; 27 import android.content.Intent; 28 import android.content.pm.PackageManager; 29 import android.content.pm.PackageManager.NameNotFoundException; 30 import android.content.pm.ResolveInfo; 31 import android.os.Bundle; 32 import android.os.ParcelFileDescriptor; 33 import android.os.RemoteException; 34 import android.os.SystemClock; 35 import android.os.UserHandle; 36 import android.support.test.uiautomator.UiDevice; 37 import android.test.InstrumentationTestCase; 38 import android.test.InstrumentationTestRunner; 39 import android.util.Log; 40 41 import androidx.test.rule.logging.AtraceLogger; 42 43 import java.io.BufferedReader; 44 import java.io.BufferedWriter; 45 import java.io.File; 46 import java.io.FileInputStream; 47 import java.io.FileOutputStream; 48 import java.io.FileWriter; 49 import java.io.IOException; 50 import java.io.InputStream; 51 import java.io.InputStreamReader; 52 import java.io.OutputStreamWriter; 53 import java.nio.file.Paths; 54 import java.time.format.DateTimeFormatter; 55 import java.time.ZonedDateTime; 56 import java.time.ZoneOffset; 57 import java.util.ArrayList; 58 import java.util.Arrays; 59 import java.util.HashMap; 60 import java.util.HashSet; 61 import java.util.LinkedHashMap; 62 import java.util.List; 63 import java.util.Map; 64 import java.util.Set; 65 66 /** 67 * This test is intended to measure the time it takes for the apps to start. 68 * Names of the applications are passed in command line, and the 69 * test starts each application, and reports the start up time in milliseconds. 70 * The instrumentation expects the following key to be passed on the command line: 71 * apps - A list of applications to start and their corresponding result keys 72 * in the following format: 73 * -e apps <app name>^<result key>|<app name>^<result key> 74 */ 75 @Deprecated 76 public class AppLaunch extends InstrumentationTestCase { 77 78 private static final int JOIN_TIMEOUT = 10000; 79 private static final String TAG = AppLaunch.class.getSimpleName(); 80 81 // optional parameter: comma separated list of required account types before proceeding 82 // with the app launch 83 private static final String KEY_REQUIRED_ACCOUNTS = "required_accounts"; 84 private static final String KEY_APPS = "apps"; 85 private static final String KEY_IORAP_TRIAL_LAUNCH = "iorap_trial_launch"; 86 private static final String KEY_IORAP_COMPILER_FILTERS = "iorap_compiler_filters"; 87 private static final String KEY_TRIAL_LAUNCH = "trial_launch"; 88 private static final String KEY_LAUNCH_ITERATIONS = "launch_iterations"; 89 private static final String KEY_LAUNCH_ORDER = "launch_order"; 90 private static final String KEY_DROP_CACHE = "drop_cache"; 91 private static final String KEY_SIMPLEPERF_CMD = "simpleperf_cmd"; 92 private static final String KEY_SIMPLEPERF_APP = "simpleperf_app"; 93 private static final String KEY_CYCLE_CLEAN = "cycle_clean"; 94 private static final String KEY_TRACE_ALL = "trace_all"; 95 private static final String KEY_TRACE_ITERATIONS = "trace_iterations"; 96 private static final String KEY_LAUNCH_DIRECTORY = "launch_directory"; 97 private static final String KEY_TRACE_DIRECTORY = "trace_directory"; 98 private static final String KEY_TRACE_CATEGORY = "trace_categories"; 99 private static final String KEY_TRACE_BUFFERSIZE = "trace_bufferSize"; 100 private static final String KEY_TRACE_DUMPINTERVAL = "tracedump_interval"; 101 private static final String KEY_COMPILER_FILTERS = "compiler_filters"; 102 private static final String KEY_FORCE_STOP_APP = "force_stop_app"; 103 private static final String ENABLE_SCREEN_RECORDING = "enable_screen_recording"; 104 private static final int MAX_RECORDING_PARTS = 5; 105 private static final long VIDEO_TAIL_BUFFER = 500; 106 107 private static final String SIMPLEPERF_APP_CMD = 108 "simpleperf --log fatal stat --csv -e cpu-cycles,major-faults --app %s & %s"; 109 private static final String WEARABLE_ACTION_GOOGLE = 110 "com.google.android.wearable.action.GOOGLE"; 111 private static final int INITIAL_LAUNCH_IDLE_TIMEOUT = 5000; // 5s to allow app to idle 112 private static final int POST_LAUNCH_IDLE_TIMEOUT = 750; // 750ms idle for non initial launches 113 private static final int BEFORE_FORCE_STOP_SLEEP_TIMEOUT = 1000; // 1s before force stopping 114 private static final int BEFORE_KILL_APP_SLEEP_TIMEOUT = 1000; // 1s before killing 115 private static final int BETWEEN_LAUNCH_SLEEP_TIMEOUT = 3000; // 3s between launching apps 116 private static final int PROFILE_SAVE_SLEEP_TIMEOUT = 1000; // Allow 1s for the profile to save 117 private static final int IORAP_TRACE_DURATION_TIMEOUT = 7000; // Allow 7s for trace to complete. 118 private static final int IORAP_TRIAL_LAUNCH_ITERATIONS = 5; // min 5 launches to merge traces. 119 private static final int IORAP_COMPILE_CMD_TIMEOUT = 60; // in seconds: 1 minutes 120 private static final int IORAP_COMPILE_MIN_TRACES = 1; // configure iorapd to need 1 trace. 121 private static final int IORAP_COMPILE_RETRIES = 3; // retry compiler 3 times if it fails. 122 private static final String LAUNCH_SUB_DIRECTORY = "launch_logs"; 123 private static final String LAUNCH_FILE = "applaunch.txt"; 124 private static final String TRACE_SUB_DIRECTORY = "atrace_logs"; 125 private static final String DEFAULT_TRACE_CATEGORIES = 126 "sched,freq,gfx,view,dalvik,webview,input,wm,disk,am,wm,binder_driver,hal,ss"; 127 private static final String DEFAULT_TRACE_BUFFER_SIZE = "20000"; 128 private static final String DEFAULT_TRACE_DUMP_INTERVAL = "10"; 129 private static final String TRIAL_LAUNCH = "TRIAL_LAUNCH"; 130 private static final String IORAP_TRIAL_LAUNCH = "IORAP_TRIAL_LAUNCH"; 131 private static final String IORAP_TRIAL_LAUNCH_FIRST = "IORAP_TRIAL_LAUNCH_FIRST"; 132 private static final String IORAP_TRIAL_LAUNCH_LAST = "IORAP_TRIAL_LAUNCH_LAST"; 133 private static final String DELIMITER = ","; 134 private static final String DROP_CACHE_SCRIPT = "/data/local/tmp/dropCache.sh"; 135 private static final String APP_LAUNCH_CMD = "am start -W -n"; 136 private static final String SUCCESS_MESSAGE = "Status: ok"; 137 private static final String TOTAL_TIME_MESSAGE = "TotalTime:"; 138 private static final String COMPILE_SUCCESS = "Success"; 139 private static final String LAUNCH_ITERATION = "LAUNCH_ITERATION-%d"; 140 private static final String TRACE_ITERATION = "TRACE_ITERATION-%d"; 141 private static final String LAUNCH_ITERATION_PREFIX = "LAUNCH_ITERATION"; 142 private static final String TRACE_ITERATION_PREFIX = "TRACE_ITERATION"; 143 private static final String LAUNCH_ORDER_CYCLIC = "cyclic"; 144 private static final String LAUNCH_ORDER_SEQUENTIAL = "sequential"; 145 private static final String COMPILE_CMD = "cmd package compile -f -m %s %s"; 146 private static final String IORAP_COMPILE_CMD = "dumpsys iorapd --compile-package %s"; 147 private static final String IORAP_MAINTENANCE_CMD = 148 "dumpsys iorapd --purge-package %s"; 149 private static final String IORAP_DUMPSYS_CMD = "dumpsys iorapd"; 150 private static final String SPEED_PROFILE_FILTER = "speed-profile"; 151 private static final String VERIFY_FILTER = "verify"; 152 private static final String LAUNCH_SCRIPT_NAME = "appLaunch"; 153 154 private Map<String, Intent> mNameToIntent; 155 private List<LaunchOrder> mLaunchOrderList = new ArrayList<LaunchOrder>(); 156 private RecordingThread mCurrentThread; 157 private Map<String, String> mNameToResultKey; 158 private Map<String, Map<String, List<AppLaunchResult>>> mNameToLaunchTime; 159 private IActivityManager mAm; 160 private File launchSubDir = null; 161 private String mSimplePerfCmd = null; 162 private String mLaunchOrder = null; 163 private boolean mDropCache = false; 164 private int mLaunchIterations = 10; 165 private boolean mForceStopApp = true; 166 private boolean mEnableRecording = false; 167 private int mTraceLaunchCount = 0; 168 private String mTraceDirectoryStr = null; 169 private Bundle mResult = new Bundle(); 170 private Set<String> mRequiredAccounts; 171 private boolean mTrialLaunch = false; 172 private boolean mIorapTrialLaunch = false; 173 private BufferedWriter mBufferedWriter = null; 174 private boolean mSimplePerfAppOnly = false; 175 private String[] mCompilerFilters = null; 176 private List<String> mIorapCompilerFilters = null; 177 private String mLastAppName = ""; 178 private boolean mCycleCleanUp = false; 179 private boolean mTraceAll = false; 180 private boolean mIterationCycle = false; 181 private UiDevice mDevice; 182 183 enum IorapStatus { 184 UNDEFINED, 185 ENABLED, 186 DISABLED 187 } 188 private IorapStatus mIorapStatus = IorapStatus.UNDEFINED; 189 private long mCycleTime = 0; 190 private StringBuilder mCycleTimes = new StringBuilder(); 191 192 @Override setUp()193 protected void setUp() throws Exception { 194 super.setUp(); 195 getInstrumentation().getUiAutomation().setRotation(UiAutomation.ROTATION_FREEZE_0); 196 } 197 198 @Override tearDown()199 protected void tearDown() throws Exception { 200 getInstrumentation().getUiAutomation().setRotation(UiAutomation.ROTATION_UNFREEZE); 201 super.tearDown(); 202 } 203 addLaunchResult(LaunchOrder launch, AppLaunchResult result)204 private void addLaunchResult(LaunchOrder launch, AppLaunchResult result) { 205 mNameToLaunchTime.get(launch.getApp()).get(launch.getCompilerFilter()).add(result); 206 } 207 hasFailureOnFirstLaunch(LaunchOrder launch)208 private boolean hasFailureOnFirstLaunch(LaunchOrder launch) { 209 List<AppLaunchResult> results = 210 mNameToLaunchTime.get(launch.getApp()).get(launch.getCompilerFilter()); 211 return (results.size() > 0) && (results.get(0).mLaunchTime < 0); 212 } 213 testMeasureStartUpTime()214 public void testMeasureStartUpTime() throws RemoteException, NameNotFoundException, 215 IOException, InterruptedException { 216 InstrumentationTestRunner instrumentation = 217 (InstrumentationTestRunner)getInstrumentation(); 218 Bundle args = instrumentation.getArguments(); 219 mAm = ActivityManager.getService(); 220 String launchDirectory = args.getString(KEY_LAUNCH_DIRECTORY); 221 222 createMappings(); 223 parseArgs(args); 224 checkAccountSignIn(); 225 226 // Root directory for applaunch file to log the app launch output 227 // Will be useful in case of simpleperf command is used 228 File launchRootDir = null; 229 if (null != launchDirectory && !launchDirectory.isEmpty()) { 230 launchRootDir = new File(launchDirectory); 231 if (!launchRootDir.exists() && !launchRootDir.mkdirs()) { 232 throw new IOException("Unable to create the destination directory " 233 + launchRootDir + ". Try disabling selinux."); 234 } 235 } 236 237 try { 238 launchSubDir = new File(launchRootDir, LAUNCH_SUB_DIRECTORY); 239 240 if (!launchSubDir.exists() && !launchSubDir.mkdirs()) { 241 throw new IOException("Unable to create the lauch file sub directory " 242 + launchSubDir + ". Try disabling selinux."); 243 } 244 File file = new File(launchSubDir, LAUNCH_FILE); 245 FileOutputStream outputStream = new FileOutputStream(file); 246 mBufferedWriter = new BufferedWriter(new OutputStreamWriter( 247 outputStream)); 248 249 // Root directory for trace file during the launches 250 File rootTrace = null; 251 File rootTraceSubDir = null; 252 int traceBufferSize = 0; 253 int traceDumpInterval = 0; 254 Set<String> traceCategoriesSet = null; 255 if (null != mTraceDirectoryStr && !mTraceDirectoryStr.isEmpty()) { 256 rootTrace = new File(mTraceDirectoryStr); 257 if (!rootTrace.exists() && !rootTrace.mkdirs()) { 258 throw new IOException("Unable to create the trace directory"); 259 } 260 rootTraceSubDir = new File(rootTrace, TRACE_SUB_DIRECTORY); 261 if (!rootTraceSubDir.exists() && !rootTraceSubDir.mkdirs()) { 262 throw new IOException("Unable to create the trace sub directory"); 263 } 264 assertNotNull("Trace iteration parameter is mandatory", 265 args.getString(KEY_TRACE_ITERATIONS)); 266 mTraceLaunchCount = Integer.parseInt(args.getString(KEY_TRACE_ITERATIONS)); 267 String traceCategoriesStr = args 268 .getString(KEY_TRACE_CATEGORY, DEFAULT_TRACE_CATEGORIES); 269 traceBufferSize = Integer.parseInt(args.getString(KEY_TRACE_BUFFERSIZE, 270 DEFAULT_TRACE_BUFFER_SIZE)); 271 traceDumpInterval = Integer.parseInt(args.getString(KEY_TRACE_DUMPINTERVAL, 272 DEFAULT_TRACE_DUMP_INTERVAL)); 273 traceCategoriesSet = new HashSet<String>(); 274 if (!traceCategoriesStr.isEmpty()) { 275 String[] traceCategoriesSplit = traceCategoriesStr.split(DELIMITER); 276 for (int i = 0; i < traceCategoriesSplit.length; i++) { 277 traceCategoriesSet.add(traceCategoriesSplit[i]); 278 } 279 } 280 } 281 282 // Get the app launch order based on launch order, trial launch, 283 // launch iterations and trace iterations 284 setLaunchOrder(); 285 286 for (LaunchOrder launch : mLaunchOrderList) { 287 toggleIorapStatus(launch.getIorapEnabled()); 288 dropCache(/*override*/false); 289 290 Log.v(TAG, "Launch reason: " + launch.getLaunchReason()); 291 292 // App launch times for trial launch will not be used for final 293 // launch time calculations. 294 if (launch.getLaunchReason().equals(TRIAL_LAUNCH)) { 295 mIterationCycle = false; 296 // In the "applaunch.txt" file, trail launches is referenced using 297 // "TRIAL_LAUNCH" 298 Intent startIntent = mNameToIntent.get(launch.getApp()); 299 if (startIntent == null) { 300 Log.w(TAG, "App does not exist: " + launch.getApp()); 301 mResult.putString(mNameToResultKey.get(launch.getApp()), 302 "App does not exist"); 303 continue; 304 } 305 String appPkgName = startIntent.getComponent().getPackageName(); 306 if (SPEED_PROFILE_FILTER.equals(launch.getCompilerFilter())) { 307 assertTrue(String.format("Not able to compile the app : %s", appPkgName), 308 compileApp(VERIFY_FILTER, appPkgName)); 309 } else if (launch.getCompilerFilter() != null) { 310 assertTrue(String.format("Not able to compile the app : %s", appPkgName), 311 compileApp(launch.getCompilerFilter(), appPkgName)); 312 } 313 // We only need to run a trial for the speed-profile filter, but we always 314 // run one for "applaunch.txt" consistency. 315 AppLaunchResult launchResult = 316 startApp(launch.getApp(), launch.getLaunchReason()); 317 if (launchResult.mLaunchTime < 0) { 318 addLaunchResult(launch, new AppLaunchResult()); 319 // simply pass the app if launch isn't successful 320 // error should have already been logged by startApp 321 continue; 322 } 323 sleep(INITIAL_LAUNCH_IDLE_TIMEOUT); 324 if (SPEED_PROFILE_FILTER.equals(launch.getCompilerFilter())) { 325 // Send SIGUSR1 to force dumping a profile. 326 String sendSignalCommand = 327 String.format("killall -s SIGUSR1 %s", appPkgName); 328 getInstrumentation().getUiAutomation().executeShellCommand( 329 sendSignalCommand); 330 // killall is async, wait one second to let the app save the profile. 331 sleep(PROFILE_SAVE_SLEEP_TIMEOUT); 332 assertTrue(String.format("Not able to compile the app : %s", appPkgName), 333 compileApp(launch.getCompilerFilter(), appPkgName)); 334 } 335 } 336 else if (launch.getLaunchReason().startsWith(IORAP_TRIAL_LAUNCH)) { 337 mIterationCycle = false; 338 339 // In the "applaunch.txt" file, iorap-trial launches is referenced using 340 // "IORAP_TRIAL_LAUNCH" or "IORAP_TRIAL_LAUNCH_LAST" 341 Intent startIntent = mNameToIntent.get(launch.getApp()); 342 if (startIntent == null) { 343 Log.w(TAG, "App does not exist: " + launch.getApp()); 344 mResult.putString(mNameToResultKey.get(launch.getApp()), 345 "App does not exist"); 346 continue; 347 } 348 String appPkgName = startIntent.getComponent().getPackageName(); 349 350 if (launch.getLaunchReason().equals(IORAP_TRIAL_LAUNCH_FIRST)) { 351 // delete any iorap-traces associated with this package. 352 purgeIorapPackage(appPkgName); 353 } 354 dropCache(/*override*/true); // iorap-trial runs must have drop cache. 355 356 AppLaunchResult launchResult = 357 startApp(launch.getApp(), launch.getLaunchReason()); 358 if (launchResult.mLaunchTime < 0) { 359 addLaunchResult(launch, new AppLaunchResult()); 360 // simply pass the app if launch isn't successful 361 // error should have already been logged by startApp 362 continue; 363 } 364 // wait for slightly more than 5s (iorapd.perfetto.trace_duration_ms) for the trace buffers to complete. 365 sleep(IORAP_TRACE_DURATION_TIMEOUT); 366 367 if (launch.getLaunchReason().equals(IORAP_TRIAL_LAUNCH_LAST)) { 368 // run the iorap compiler and wait for iorap to compile fully. 369 // this throws an exception if it fails. 370 compileAppForIorapWithRetries(appPkgName, IORAP_COMPILE_RETRIES); 371 } 372 } 373 374 // App launch times used for final calculation 375 else if (launch.getLaunchReason().contains(LAUNCH_ITERATION_PREFIX)) { 376 mIterationCycle = true; 377 AppLaunchResult launchResults = null; 378 if (hasFailureOnFirstLaunch(launch)) { 379 // skip if the app has failures while launched first 380 Log.w(TAG, "Has failures on first launch: " + launch.getApp()); 381 forceStopApp(launch.getApp()); 382 continue; 383 } 384 AtraceLogger atraceLogger = null; 385 if (mTraceAll) { 386 Log.i(TAG, "Started tracing " + launch.getApp()); 387 atraceLogger = AtraceLogger 388 .getAtraceLoggerInstance(getInstrumentation()); 389 } 390 try { 391 // Start the trace 392 if (atraceLogger != null) { 393 atraceLogger.atraceStart(traceCategoriesSet, traceBufferSize, 394 traceDumpInterval, rootTraceSubDir, 395 String.format("%s-%s-%s", launch.getApp(), 396 launch.getCompilerFilter(), launch.getLaunchReason())); 397 } 398 // In the "applaunch.txt" file app launches are referenced using 399 // "LAUNCH_ITERATION - ITERATION NUM" 400 launchResults = startApp(launch.getApp(), launch.getLaunchReason()); 401 if (launchResults.mLaunchTime < 0) { 402 addLaunchResult(launch, new AppLaunchResult()); 403 // if it fails once, skip the rest of the launches 404 continue; 405 } else { 406 mCycleTime += launchResults.mLaunchTime; 407 addLaunchResult(launch, launchResults); 408 } 409 sleep(POST_LAUNCH_IDLE_TIMEOUT); 410 } finally { 411 // Stop the trace 412 if (atraceLogger != null) { 413 Log.i(TAG, "Stopped tracing " + launch.getApp()); 414 atraceLogger.atraceStop(); 415 } 416 } 417 418 } 419 420 // App launch times for trace launch will not be used for final 421 // launch time calculations. 422 else if (launch.getLaunchReason().contains(TRACE_ITERATION_PREFIX)) { 423 mIterationCycle = false; 424 AtraceLogger atraceLogger = AtraceLogger 425 .getAtraceLoggerInstance(getInstrumentation()); 426 // Start the trace 427 try { 428 atraceLogger.atraceStart(traceCategoriesSet, traceBufferSize, 429 traceDumpInterval, rootTraceSubDir, 430 String.format("%s-%s-%s", launch.getApp(), 431 launch.getCompilerFilter(), launch.getLaunchReason())); 432 startApp(launch.getApp(), launch.getLaunchReason()); 433 sleep(POST_LAUNCH_IDLE_TIMEOUT); 434 } finally { 435 // Stop the trace 436 atraceLogger.atraceStop(); 437 } 438 } 439 if(mForceStopApp) { 440 sleep(BEFORE_FORCE_STOP_SLEEP_TIMEOUT); 441 forceStopApp(launch.getApp()); 442 sleep(BEFORE_KILL_APP_SLEEP_TIMEOUT); 443 // Close again for good measure (just in case). 444 forceStopApp(launch.getApp()); 445 // Kill the backgrounded process in the case forceStopApp only sent it to 446 // background. 447 killBackgroundApp(launch.getApp()); 448 } else { 449 startHomeIntent(); 450 } 451 sleep(BETWEEN_LAUNCH_SLEEP_TIMEOUT); 452 453 // If cycle clean up is enabled and last app launched is 454 // current app then the cycle is completed and eligible for 455 // cleanup. 456 if (LAUNCH_ORDER_CYCLIC.equalsIgnoreCase(mLaunchOrder) && mCycleCleanUp 457 && launch.getApp().equalsIgnoreCase(mLastAppName)) { 458 // Kill all the apps and drop all the cache 459 cleanUpAfterCycle(); 460 if (mIterationCycle) { 461 // Save the previous cycle time and reset the cycle time to 0 462 mCycleTimes.append(String.format("%d,", mCycleTime)); 463 mCycleTime = 0; 464 } 465 } 466 } 467 } finally { 468 if (null != mBufferedWriter) { 469 mBufferedWriter.close(); 470 } 471 } 472 473 if (mCycleTimes.length() != 0) { 474 mResult.putString("Cycle_Times", mCycleTimes.toString()); 475 } 476 for (String app : mNameToResultKey.keySet()) { 477 for (String compilerFilter : mCompilerFilters) { 478 StringBuilder launchTimes = new StringBuilder(); 479 StringBuilder cpuCycles = new StringBuilder(); 480 StringBuilder majorFaults = new StringBuilder(); 481 for (AppLaunchResult result : mNameToLaunchTime.get(app).get(compilerFilter)) { 482 launchTimes.append(result.mLaunchTime); 483 launchTimes.append(","); 484 if (mSimplePerfAppOnly) { 485 cpuCycles.append(result.mCpuCycles); 486 cpuCycles.append(","); 487 majorFaults.append(result.mMajorFaults); 488 majorFaults.append(","); 489 } 490 } 491 String filterName = (compilerFilter == null) ? "" : ("-" + compilerFilter); 492 mResult.putString(mNameToResultKey.get(app) + filterName, launchTimes.toString()); 493 if (mSimplePerfAppOnly) { 494 mResult.putString(mNameToResultKey.get(app) + filterName + "-cpuCycles", 495 cpuCycles.toString()); 496 mResult.putString(mNameToResultKey.get(app) + filterName + "-majorFaults", 497 majorFaults.toString()); 498 } 499 } 500 } 501 instrumentation.sendStatus(0, mResult); 502 } 503 504 /** 505 * Compile the app package using compilerFilter and return true or false 506 * based on status of the compilation command. 507 */ compileApp(String compilerFilter, String appPkgName)508 private boolean compileApp(String compilerFilter, String appPkgName) throws IOException { 509 try (ParcelFileDescriptor result = getInstrumentation().getUiAutomation(). 510 executeShellCommand(String.format(COMPILE_CMD, compilerFilter, appPkgName)); 511 BufferedReader bufferedReader = new BufferedReader(new InputStreamReader( 512 new FileInputStream(result.getFileDescriptor())))) { 513 String line; 514 while ((line = bufferedReader.readLine()) != null) { 515 if (line.contains(COMPILE_SUCCESS)) { 516 return true; 517 } 518 } 519 return false; 520 } 521 } 522 523 /** 524 * Compile the app package using compilerFilter, 525 * retrying if the compilation command fails in between. 526 */ compileAppForIorapWithRetries(String appPkgName, int retries)527 private void compileAppForIorapWithRetries(String appPkgName, int retries) throws IOException { 528 for (int i = 0; i < retries; ++i) { 529 if (compileAppForIorap(appPkgName)) { 530 return; 531 } 532 sleep(1000); 533 } 534 535 throw new IllegalStateException("compileAppForIorapWithRetries: timed out after " 536 + retries + " retries"); 537 } 538 539 /** 540 * Compile the app package using compilerFilter and return true or false 541 * based on status of the compilation command. 542 */ compileAppForIorap(String appPkgName)543 private boolean compileAppForIorap(String appPkgName) throws IOException { 544 String logcatTimestamp = getTimeNowForLogcat(); 545 546 getInstrumentation().getUiAutomation(). 547 executeShellCommand(String.format(IORAP_COMPILE_CMD, appPkgName)); 548 549 int i = 0; 550 for (i = 0; i < IORAP_COMPILE_CMD_TIMEOUT; ++i) { 551 IorapCompilationStatus status = waitForIorapCompiled(appPkgName); 552 if (status == IorapCompilationStatus.COMPLETE) { 553 Log.v(TAG, "compileAppForIorap: success"); 554 logDumpsysIorapd(appPkgName); 555 break; 556 } else if (status == IorapCompilationStatus.INSUFFICIENT_TRACES) { 557 Log.e(TAG, "compileAppForIorap: failed due to insufficient traces"); 558 logDumpsysIorapd(appPkgName); 559 throw new IllegalStateException( 560 "compileAppForIorap: failed due to insufficient traces"); 561 } // else INCOMPLETE. keep asking iorapd if it's done yet. 562 sleep(1000); 563 } 564 565 if (i == IORAP_COMPILE_CMD_TIMEOUT) { 566 Log.e(TAG, "compileAppForIorap: failed due to timeout"); 567 logDumpsysIorapd(appPkgName); 568 return false; 569 } 570 571 return true; 572 } 573 574 /** Save the contents of $(adb shell dumpsys iorapd) to the launch_logs directory. */ logDumpsysIorapd(String packageName)575 private void logDumpsysIorapd(String packageName) throws IOException { 576 InstrumentationTestRunner instrumentation = 577 (InstrumentationTestRunner)getInstrumentation(); 578 Bundle args = instrumentation.getArguments(); 579 580 String launchDirectory = args.getString(KEY_LAUNCH_DIRECTORY); 581 582 // Root directory for applaunch file to log the app launch output 583 // Will be useful in case of simpleperf command is used 584 File launchRootDir = null; 585 if (null != launchDirectory && !launchDirectory.isEmpty()) { 586 launchRootDir = new File(launchDirectory); 587 if (!launchRootDir.exists() && !launchRootDir.mkdirs()) { 588 throw new IOException("Unable to create the destination directory " 589 + launchRootDir + ". Try disabling selinux."); 590 } 591 } else { 592 Log.w(TAG, "logDumpsysIorapd: Missing launch-directory arg"); 593 return; 594 } 595 596 File launchSubDir = new File(launchRootDir, LAUNCH_SUB_DIRECTORY); 597 598 if (!launchSubDir.exists() && !launchSubDir.mkdirs()) { 599 throw new IOException("Unable to create the lauch file sub directory " 600 + launchSubDir + ". Try disabling selinux."); 601 } 602 String path = "iorapd_dumpsys_" + packageName + "_" + System.nanoTime() + ".txt"; 603 File file = new File(launchSubDir, path); 604 try (FileOutputStream outputStream = new FileOutputStream(file); 605 BufferedWriter writer = new BufferedWriter( 606 new OutputStreamWriter(outputStream)); 607 ParcelFileDescriptor result = getInstrumentation().getUiAutomation(). 608 executeShellCommand(IORAP_DUMPSYS_CMD); 609 BufferedReader bufferedReader = new BufferedReader(new InputStreamReader( 610 new FileInputStream(result.getFileDescriptor())))) { 611 String line; 612 while ((line = bufferedReader.readLine()) != null) { 613 writer.write(line + "\n"); 614 } 615 } 616 617 Log.v(TAG, "logDumpsysIorapd: Saved to file: " + path); 618 } 619 620 enum IorapCompilationStatus { 621 INCOMPLETE, 622 COMPLETE, 623 INSUFFICIENT_TRACES, 624 } waitForIorapCompiled(String appPkgName)625 private IorapCompilationStatus waitForIorapCompiled(String appPkgName) throws IOException { 626 try (ParcelFileDescriptor result = getInstrumentation().getUiAutomation(). 627 executeShellCommand(IORAP_DUMPSYS_CMD); 628 BufferedReader bufferedReader = new BufferedReader(new InputStreamReader( 629 new FileInputStream(result.getFileDescriptor())))) { 630 String line; 631 String prevLine = ""; 632 while ((line = bufferedReader.readLine()) != null) { 633 // Match the indented VersionedComponentName string. 634 // " com.google.android.deskclock/com.android.deskclock.DeskClock@62000712" 635 // Note: spaces are meaningful here. 636 if (prevLine.contains(" " + appPkgName) && prevLine.contains("@")) { 637 // pre-requisite: 638 // Compiled Status: Raw traces pending compilation (3) 639 if (line.contains("Compiled Status: Usable compiled trace")) { 640 return IorapCompilationStatus.COMPLETE; 641 } else if (line.contains("Compiled Status: ") && 642 line.contains("more traces for compilation")) { 643 // Compiled Status: Need 1 more traces for compilation 644 // No amount of waiting will help here because there were 645 // insufficient traces made. 646 return IorapCompilationStatus.INSUFFICIENT_TRACES; 647 } 648 } 649 650 prevLine = line; 651 } 652 return IorapCompilationStatus.INCOMPLETE; 653 } 654 } 655 makeReasonForIorapTrialLaunch(int launchCount)656 private String makeReasonForIorapTrialLaunch(int launchCount) { 657 String reason = IORAP_TRIAL_LAUNCH; 658 if (launchCount == 0) { 659 reason = IORAP_TRIAL_LAUNCH_FIRST; 660 } 661 if (launchCount == IORAP_TRIAL_LAUNCH_ITERATIONS - 1) { 662 reason = IORAP_TRIAL_LAUNCH_LAST; 663 } 664 return reason; 665 } 666 shouldIncludeIorap(String compilerFilter)667 private boolean shouldIncludeIorap(String compilerFilter) { 668 if (!mIorapTrialLaunch) { 669 return false; 670 } 671 672 // No iorap compiler filters specified: treat all compiler filters as ok. 673 if (mIorapCompilerFilters == null) { 674 return true; 675 } 676 677 // iorap compiler filters specified: the compilerFilter must be in the allowlist. 678 if (mIorapCompilerFilters.indexOf(compilerFilter) != -1) { 679 return true; 680 } 681 682 return false; 683 } 684 685 /** 686 * If launch order is "cyclic" then apps will be launched one after the 687 * other for each iteration count. 688 * If launch order is "sequential" then each app will be launched for given number 689 * iterations at once before launching the other apps. 690 */ setLaunchOrder()691 private void setLaunchOrder() { 692 if (LAUNCH_ORDER_CYCLIC.equalsIgnoreCase(mLaunchOrder)) { 693 for (String compilerFilter : mCompilerFilters) { 694 if (mTrialLaunch) { 695 for (String app : mNameToResultKey.keySet()) { 696 mLaunchOrderList.add(new LaunchOrder(app, compilerFilter, TRIAL_LAUNCH, /*iorapEnabled*/false)); 697 } 698 } 699 if (shouldIncludeIorap(compilerFilter)) { 700 for (int launchCount = 0; launchCount < IORAP_TRIAL_LAUNCH_ITERATIONS; ++launchCount) { 701 for (String app : mNameToResultKey.keySet()) { 702 String reason = makeReasonForIorapTrialLaunch(launchCount); 703 mLaunchOrderList.add( 704 new LaunchOrder(app, compilerFilter, 705 reason, 706 /*iorapEnabled*/true)); 707 } 708 } 709 } 710 for (int launchCount = 0; launchCount < mLaunchIterations; launchCount++) { 711 for (String app : mNameToResultKey.keySet()) { 712 mLaunchOrderList.add(new LaunchOrder(app, compilerFilter, 713 String.format(LAUNCH_ITERATION, launchCount), 714 shouldIncludeIorap(compilerFilter))); 715 } 716 } 717 if (mTraceDirectoryStr != null && !mTraceDirectoryStr.isEmpty()) { 718 for (int traceCount = 0; traceCount < mTraceLaunchCount; traceCount++) { 719 for (String app : mNameToResultKey.keySet()) { 720 mLaunchOrderList.add(new LaunchOrder(app, compilerFilter, 721 String.format(TRACE_ITERATION, traceCount), 722 shouldIncludeIorap(compilerFilter))); 723 } 724 } 725 } 726 } 727 } else if (LAUNCH_ORDER_SEQUENTIAL.equalsIgnoreCase(mLaunchOrder)) { 728 for (String compilerFilter : mCompilerFilters) { 729 for (String app : mNameToResultKey.keySet()) { 730 if (mTrialLaunch) { 731 mLaunchOrderList.add(new LaunchOrder(app, compilerFilter, TRIAL_LAUNCH, /*iorapEnabled*/false)); 732 } 733 if (shouldIncludeIorap(compilerFilter)) { 734 for (int launchCount = 0; launchCount < IORAP_TRIAL_LAUNCH_ITERATIONS; ++launchCount) { 735 String reason = makeReasonForIorapTrialLaunch(launchCount); 736 mLaunchOrderList.add( 737 new LaunchOrder(app, compilerFilter, 738 reason, 739 /*iorapEnabled*/true)); 740 } 741 } 742 for (int launchCount = 0; launchCount < mLaunchIterations; launchCount++) { 743 mLaunchOrderList.add(new LaunchOrder(app, compilerFilter, 744 String.format(LAUNCH_ITERATION, launchCount), 745 shouldIncludeIorap(compilerFilter))); 746 } 747 if (mTraceDirectoryStr != null && !mTraceDirectoryStr.isEmpty()) { 748 for (int traceCount = 0; traceCount < mTraceLaunchCount; traceCount++) { 749 mLaunchOrderList.add(new LaunchOrder(app, compilerFilter, 750 String.format(TRACE_ITERATION, traceCount), 751 shouldIncludeIorap(compilerFilter))); 752 } 753 } 754 } 755 } 756 } else { 757 assertTrue("Launch order is not valid parameter", false); 758 } 759 } 760 dropCache(boolean override)761 private void dropCache(boolean override) { 762 if (mDropCache || override) { 763 assertNotNull("Issue in dropping the cache", 764 getInstrumentation().getUiAutomation() 765 .executeShellCommand(DROP_CACHE_SCRIPT)); 766 } 767 } 768 769 // [[ $(adb shell whoami) == "root" ]] checkIfRoot()770 private boolean checkIfRoot() throws IOException { 771 String total = ""; 772 try (ParcelFileDescriptor result = getInstrumentation().getUiAutomation(). 773 executeShellCommand("whoami"); 774 BufferedReader bufferedReader = new BufferedReader(new InputStreamReader( 775 new FileInputStream(result.getFileDescriptor())))) { 776 String line; 777 while ((line = bufferedReader.readLine()) != null) { 778 total = total + line; 779 } 780 } 781 return total.contains("root"); 782 } 783 stopIorapd()784 private void stopIorapd() { 785 getInstrumentation().getUiAutomation() 786 .executeShellCommand("stop iorapd"); 787 sleep(100); // give it extra time to fully stop. 788 } 789 startIorapd()790 private void startIorapd() { 791 String logcatTimeNow = getTimeNowForLogcat(); 792 Log.v(TAG, "startIorapd, logcat time: " + logcatTimeNow); 793 794 getInstrumentation().getUiAutomation() 795 .executeShellCommand("start iorapd"); 796 797 int maxAttempts = 100; 798 int attempt = 0; 799 do { 800 // Ensure that IorapForwardingService fully reconnects to iorapd before proceeding. 801 String needle = "Connected to iorapd native service"; 802 String logcatLines = getLogcatSinceTime(logcatTimeNow); 803 804 if (logcatLines.contains(needle)) { 805 break; 806 } 807 808 sleep(1000); 809 attempt++; 810 } while (attempt < maxAttempts); 811 812 if (attempt == maxAttempts) { 813 Log.e(TAG, "Timed out after waiting for iorapd to start"); 814 } 815 // Wait a little bit longer for iorapd to settle. 816 sleep(1000); 817 } 818 819 // Delete all db rows and files associated with a package in iorapd. 820 // Effectively deletes any raw or compiled trace files, unoptimizing the package in iorap. purgeIorapPackage(String packageName)821 private void purgeIorapPackage(String packageName) { 822 try { 823 if (!checkIfRoot()) { 824 throw new AssertionError("must be root to toggle iorapd; try adb root?"); 825 } 826 } catch (IOException e) { 827 throw new AssertionError(e); 828 } 829 830 Log.v(TAG, "Purge iorap package: " + packageName); 831 getInstrumentation().getUiAutomation() 832 .executeShellCommand(String.format(IORAP_MAINTENANCE_CMD, packageName)); 833 Log.v(TAG, "Executed: " + String.format(IORAP_MAINTENANCE_CMD, packageName)); 834 } 835 executeShellCommandWithTempFile(String cmd)836 String executeShellCommandWithTempFile(String cmd) { 837 Log.v(TAG, "executeShellCommandWithTempFile, cmd: " + cmd); 838 try { 839 //File outputDir = 840 // InstrumentationRegistry.getInstrumentation().getContext().getCacheDir(); 841 File outputFile = File.createTempFile("exec_shell_command", ".sh"); 842 843 try { 844 outputFile.setWritable(true); 845 outputFile.setExecutable(true, /*ownersOnly*/false); 846 847 String scriptPath = outputFile.toString(); 848 849 // If this works correctly, the next log-line will print 'Success'. 850 try (BufferedWriter writer = new BufferedWriter(new FileWriter(scriptPath))) { 851 writer.write(cmd); 852 } 853 854 String resultString = ""; 855 try (ParcelFileDescriptor result = getInstrumentation().getUiAutomation(). 856 executeShellCommand(scriptPath); 857 BufferedReader bufferedReader = new BufferedReader(new InputStreamReader( 858 new FileInputStream(result.getFileDescriptor())))) { 859 String line; 860 while ((line = bufferedReader.readLine()) != null) { 861 resultString += line + "\n"; 862 } 863 } 864 865 return resultString; 866 } finally { 867 outputFile.delete(); 868 } 869 } catch (IOException e) { 870 throw new AssertionError("Failed to execute shell command: " + cmd, e); 871 } 872 } 873 874 // Get the 'now' timestamp usable with $(adb logcat -v utc -T "time string") getTimeNowForLogcat()875 String getTimeNowForLogcat() { 876 ZonedDateTime utc = ZonedDateTime.now(ZoneOffset.UTC); 877 878 // YYYY-MM-DD hh:mm:ss.mmm 879 return utc.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSS")); 880 } 881 getLogcatSinceTime(String logcatTime)882 String getLogcatSinceTime(String logcatTime) { 883 // The time has spaces in it but must be passed as a single arg. 884 // Therefore use a temp script file. 885 return executeShellCommandWithTempFile( 886 String.format("logcat -d -v threadtime -v utc -T '%s'", logcatTime)); 887 } 888 889 /** 890 * Toggle iorapd-based readahead and trace-collection. 891 * If iorapd is already enabled and enable is true, does nothing. 892 * If iorapd is already disabled and enable is false, does nothing. 893 */ toggleIorapStatus(boolean enable)894 private void toggleIorapStatus(boolean enable) { 895 boolean currentlyEnabled = false; 896 Log.v(TAG, "toggleIorapStatus " + Boolean.toString(enable)); 897 898 // Do nothing if we are already enabled or disabled. 899 if (mIorapStatus == IorapStatus.ENABLED && enable) { 900 return; 901 } else if (mIorapStatus == IorapStatus.DISABLED && !enable) { 902 return; 903 } 904 905 try { 906 if (!checkIfRoot()) { 907 throw new AssertionError("must be root to toggle iorapd; try adb root?"); 908 } 909 } catch (IOException e) { 910 throw new AssertionError(e); 911 } 912 913 getInstrumentation().getUiAutomation() 914 .executeShellCommand(String.format("setprop iorapd.perfetto.enable %b", enable)); 915 getInstrumentation().getUiAutomation() 916 .executeShellCommand(String.format("setprop iorapd.readahead.enable %b", enable)); 917 getInstrumentation().getUiAutomation() 918 .executeShellCommand(String.format( 919 "setprop iorapd.maintenance.min_traces %d", IORAP_COMPILE_MIN_TRACES)); 920 // this last command blocks until iorapd refreshes its system properties 921 getInstrumentation().getUiAutomation() 922 .executeShellCommand(String.format("dumpsys iorapd --refresh-properties")); 923 924 if (enable) { 925 mIorapStatus = IorapStatus.ENABLED; 926 } else { 927 mIorapStatus = IorapStatus.DISABLED; 928 } 929 } 930 parseArgs(Bundle args)931 private void parseArgs(Bundle args) { 932 mNameToResultKey = new LinkedHashMap<String, String>(); 933 mNameToLaunchTime = new HashMap<>(); 934 String launchIterations = args.getString(KEY_LAUNCH_ITERATIONS); 935 if (launchIterations != null) { 936 mLaunchIterations = Integer.parseInt(launchIterations); 937 } 938 String forceStopApp = args.getString(KEY_FORCE_STOP_APP); 939 940 if (forceStopApp != null) { 941 mForceStopApp = Boolean.parseBoolean(forceStopApp); 942 } 943 944 String enableRecording = args.getString(ENABLE_SCREEN_RECORDING); 945 946 if (enableRecording != null) { 947 mEnableRecording = Boolean.parseBoolean(enableRecording); 948 } 949 String appList = args.getString(KEY_APPS); 950 if (appList == null) 951 return; 952 953 String appNames[] = appList.split("\\|"); 954 for (String pair : appNames) { 955 String[] parts = pair.split("\\^"); 956 if (parts.length != 2) { 957 Log.e(TAG, "The apps key is incorrectly formatted"); 958 fail(); 959 } 960 961 mNameToResultKey.put(parts[0], parts[1]); 962 mNameToLaunchTime.put(parts[0], null); 963 mLastAppName = parts[0]; 964 } 965 String requiredAccounts = args.getString(KEY_REQUIRED_ACCOUNTS); 966 if (requiredAccounts != null) { 967 mRequiredAccounts = new HashSet<String>(); 968 for (String accountType : requiredAccounts.split(",")) { 969 mRequiredAccounts.add(accountType); 970 } 971 } 972 973 String compilerFilterList = args.getString(KEY_COMPILER_FILTERS); 974 if (compilerFilterList != null) { 975 // If a compiler filter is passed, we make a trial launch to force compilation 976 // of the apps. 977 mTrialLaunch = true; 978 mCompilerFilters = compilerFilterList.split("\\|"); 979 } else { 980 // Just pass a null compiler filter to use the current state of the app. 981 mCompilerFilters = new String[1]; 982 } 983 984 String iorapCompilerFilterList = args.getString(KEY_IORAP_COMPILER_FILTERS); 985 if (iorapCompilerFilterList != null) { 986 // Passing in iorap compiler filters implies an iorap trial launch. 987 mIorapTrialLaunch = true; 988 mIorapCompilerFilters = Arrays.asList(iorapCompilerFilterList.split("\\|")); 989 } 990 991 // Pre-populate the results map to avoid null checks. 992 for (String app : mNameToLaunchTime.keySet()) { 993 HashMap<String, List<AppLaunchResult>> map = new HashMap<>(); 994 mNameToLaunchTime.put(app, map); 995 for (String compilerFilter : mCompilerFilters) { 996 map.put(compilerFilter, new ArrayList<>()); 997 } 998 } 999 1000 mTraceDirectoryStr = args.getString(KEY_TRACE_DIRECTORY); 1001 mDropCache = Boolean.parseBoolean(args.getString(KEY_DROP_CACHE)); 1002 mSimplePerfCmd = args.getString(KEY_SIMPLEPERF_CMD); 1003 mLaunchOrder = args.getString(KEY_LAUNCH_ORDER, LAUNCH_ORDER_CYCLIC); 1004 mSimplePerfAppOnly = Boolean.parseBoolean(args.getString(KEY_SIMPLEPERF_APP)); 1005 mCycleCleanUp = Boolean.parseBoolean(args.getString(KEY_CYCLE_CLEAN)); 1006 mTraceAll = Boolean.parseBoolean(args.getString(KEY_TRACE_ALL)); 1007 mTrialLaunch = mTrialLaunch || Boolean.parseBoolean(args.getString(KEY_TRIAL_LAUNCH)); 1008 mIorapTrialLaunch = mIorapTrialLaunch || 1009 Boolean.parseBoolean(args.getString(KEY_IORAP_TRIAL_LAUNCH)); 1010 1011 if (mSimplePerfCmd != null && mSimplePerfAppOnly) { 1012 Log.w(TAG, String.format("Passing both %s and %s is not supported, ignoring %s", 1013 KEY_SIMPLEPERF_CMD, KEY_SIMPLEPERF_APP, KEY_SIMPLEPERF_CMD)); 1014 } 1015 } 1016 hasLeanback(Context context)1017 private boolean hasLeanback(Context context) { 1018 return context.getPackageManager().hasSystemFeature(PackageManager.FEATURE_LEANBACK); 1019 } 1020 createMappings()1021 private void createMappings() { 1022 mNameToIntent = new LinkedHashMap<String, Intent>(); 1023 1024 PackageManager pm = getInstrumentation().getContext() 1025 .getPackageManager(); 1026 Intent intentToResolve = new Intent(Intent.ACTION_MAIN); 1027 intentToResolve.addCategory(hasLeanback(getInstrumentation().getContext()) ? 1028 Intent.CATEGORY_LEANBACK_LAUNCHER : 1029 Intent.CATEGORY_LAUNCHER); 1030 List<ResolveInfo> ris = pm.queryIntentActivities(intentToResolve, 0); 1031 resolveLoop(ris, intentToResolve, pm); 1032 // For Wear 1033 intentToResolve = new Intent(WEARABLE_ACTION_GOOGLE); 1034 ris = pm.queryIntentActivities(intentToResolve, 0); 1035 resolveLoop(ris, intentToResolve, pm); 1036 } 1037 resolveLoop(List<ResolveInfo> ris, Intent intentToResolve, PackageManager pm)1038 private void resolveLoop(List<ResolveInfo> ris, Intent intentToResolve, PackageManager pm) { 1039 if (ris == null || ris.isEmpty()) { 1040 Log.i(TAG, "Could not find any apps"); 1041 } else { 1042 for (ResolveInfo ri : ris) { 1043 Intent startIntent = new Intent(intentToResolve); 1044 startIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK 1045 | Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED); 1046 startIntent.setClassName(ri.activityInfo.packageName, 1047 ri.activityInfo.name); 1048 String appName = ri.loadLabel(pm).toString(); 1049 if (appName != null) { 1050 // Support launching intent using package name or app name 1051 mNameToIntent.put(ri.activityInfo.packageName, startIntent); 1052 mNameToIntent.put(appName, startIntent); 1053 } 1054 } 1055 } 1056 } 1057 startApp(String appName, String launchReason)1058 private AppLaunchResult startApp(String appName, String launchReason) 1059 throws NameNotFoundException, RemoteException { 1060 Log.i(TAG, "Starting " + appName); 1061 if(mEnableRecording) { 1062 startRecording(appName, launchReason); 1063 } 1064 1065 Intent startIntent = mNameToIntent.get(appName); 1066 if (startIntent == null) { 1067 Log.w(TAG, "App does not exist: " + appName); 1068 mResult.putString(mNameToResultKey.get(appName), "App does not exist"); 1069 return new AppLaunchResult(); 1070 } 1071 AppLaunchRunnable runnable = new AppLaunchRunnable(startIntent, launchReason); 1072 Thread t = new Thread(runnable); 1073 t.start(); 1074 try { 1075 t.join(JOIN_TIMEOUT); 1076 } catch (InterruptedException e) { 1077 // ignore 1078 } 1079 1080 if(mEnableRecording) { 1081 stopRecording(); 1082 } 1083 return runnable.getResult(); 1084 } 1085 checkAccountSignIn()1086 private void checkAccountSignIn() { 1087 // ensure that the device has the required account types before starting test 1088 // e.g. device must have a valid Google account sign in to measure a meaningful launch time 1089 // for Gmail 1090 if (mRequiredAccounts == null || mRequiredAccounts.isEmpty()) { 1091 return; 1092 } 1093 final AccountManager am = 1094 (AccountManager) getInstrumentation().getTargetContext().getSystemService( 1095 Context.ACCOUNT_SERVICE); 1096 Account[] accounts = am.getAccounts(); 1097 // use set here in case device has multiple accounts of the same type 1098 Set<String> foundAccounts = new HashSet<String>(); 1099 for (Account account : accounts) { 1100 if (mRequiredAccounts.contains(account.type)) { 1101 foundAccounts.add(account.type); 1102 } 1103 } 1104 // check if account type matches, if not, fail test with message on what account types 1105 // are missing 1106 if (mRequiredAccounts.size() != foundAccounts.size()) { 1107 mRequiredAccounts.removeAll(foundAccounts); 1108 StringBuilder sb = new StringBuilder("Device missing these accounts:"); 1109 for (String account : mRequiredAccounts) { 1110 sb.append(' '); 1111 sb.append(account); 1112 } 1113 fail(sb.toString()); 1114 } 1115 } 1116 startHomeIntent()1117 private void startHomeIntent() { 1118 Intent homeIntent = new Intent(Intent.ACTION_MAIN); 1119 homeIntent.addCategory(Intent.CATEGORY_HOME); 1120 homeIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK 1121 | Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED); 1122 getInstrumentation().getContext().startActivity(homeIntent); 1123 sleep(POST_LAUNCH_IDLE_TIMEOUT); 1124 } 1125 cleanUpAfterCycle()1126 private void cleanUpAfterCycle() { 1127 // Kill all the apps 1128 for (String appName : mNameToIntent.keySet()) { 1129 Log.w(TAG, String.format("killing %s", appName)); 1130 forceStopApp(appName); 1131 } 1132 // Drop all the cache. 1133 assertNotNull("Issue in dropping the cache", 1134 getInstrumentation().getUiAutomation() 1135 .executeShellCommand(DROP_CACHE_SCRIPT)); 1136 } 1137 forceStopApp(String appName)1138 private void forceStopApp(String appName) { 1139 Intent startIntent = mNameToIntent.get(appName); 1140 if (startIntent != null) { 1141 String packageName = startIntent.getComponent().getPackageName(); 1142 try { 1143 mAm.forceStopPackage(packageName, UserHandle.USER_CURRENT); 1144 } catch (RemoteException e) { 1145 Log.w(TAG, "Error closing app", e); 1146 } 1147 } 1148 } 1149 killBackgroundApp(String appName)1150 private void killBackgroundApp(String appName) { 1151 Intent startIntent = mNameToIntent.get(appName); 1152 if (startIntent != null) { 1153 String packageName = startIntent.getComponent().getPackageName(); 1154 try { 1155 mAm.killBackgroundProcesses(packageName, UserHandle.USER_CURRENT); 1156 } catch (RemoteException e) { 1157 Log.w(TAG, "Error closing app", e); 1158 } 1159 } 1160 } 1161 sleep(int time)1162 private void sleep(int time) { 1163 try { 1164 Thread.sleep(time); 1165 } catch (InterruptedException e) { 1166 // ignore 1167 } 1168 } 1169 reportError(String appName, String processName)1170 private void reportError(String appName, String processName) { 1171 ActivityManager am = (ActivityManager) getInstrumentation() 1172 .getContext().getSystemService(Context.ACTIVITY_SERVICE); 1173 List<ProcessErrorStateInfo> crashes = am.getProcessesInErrorState(); 1174 if (crashes != null) { 1175 for (ProcessErrorStateInfo crash : crashes) { 1176 if (!crash.processName.equals(processName)) 1177 continue; 1178 1179 Log.w(TAG, appName + " crashed: " + crash.shortMsg); 1180 mResult.putString(mNameToResultKey.get(appName), crash.shortMsg); 1181 return; 1182 } 1183 } 1184 1185 mResult.putString(mNameToResultKey.get(appName), 1186 "Crashed for unknown reason"); 1187 Log.w(TAG, appName 1188 + " not found in process list, most likely it is crashed"); 1189 } 1190 1191 private class LaunchOrder { 1192 private String mApp; 1193 private String mCompilerFilter; 1194 private String mLaunchReason; 1195 private boolean mIorapEnabled; 1196 LaunchOrder(String app, String compilerFilter, String launchReason, boolean iorapEnabled)1197 LaunchOrder(String app, String compilerFilter, String launchReason, boolean iorapEnabled) { 1198 mApp = app; 1199 mCompilerFilter = compilerFilter; 1200 mLaunchReason = launchReason; 1201 mIorapEnabled = iorapEnabled; 1202 } 1203 getApp()1204 public String getApp() { 1205 return mApp; 1206 } 1207 setApp(String app)1208 public void setApp(String app) { 1209 mApp = app; 1210 } 1211 getCompilerFilter()1212 public String getCompilerFilter() { 1213 return mCompilerFilter; 1214 } 1215 getLaunchReason()1216 public String getLaunchReason() { 1217 return mLaunchReason; 1218 } 1219 setLaunchReason(String launchReason)1220 public void setLaunchReason(String launchReason) { 1221 mLaunchReason = launchReason; 1222 } 1223 setIorapEnabled(boolean iorapEnabled)1224 public void setIorapEnabled(boolean iorapEnabled) { 1225 mIorapEnabled = iorapEnabled; 1226 } 1227 getIorapEnabled()1228 public boolean getIorapEnabled() { 1229 return mIorapEnabled; 1230 } 1231 } 1232 1233 private class AppLaunchResult { 1234 long mLaunchTime; 1235 long mCpuCycles; 1236 long mMajorFaults; 1237 AppLaunchResult()1238 AppLaunchResult() { 1239 mLaunchTime = -1L; 1240 mCpuCycles = -1L; 1241 mMajorFaults = -1L; 1242 } 1243 AppLaunchResult(String launchTime, String cpuCycles, String majorFaults)1244 AppLaunchResult(String launchTime, String cpuCycles, String majorFaults) { 1245 try { 1246 mLaunchTime = Long.parseLong(launchTime, 10); 1247 mCpuCycles = Long.parseLong(cpuCycles, 10); 1248 mMajorFaults = Long.parseLong(majorFaults, 10); 1249 } catch (NumberFormatException e) { 1250 Log.e(TAG, "Error parsing result", e); 1251 } 1252 } 1253 } 1254 1255 private class AppLaunchRunnable implements Runnable { 1256 private Intent mLaunchIntent; 1257 private AppLaunchResult mLaunchResult; 1258 private String mLaunchReason; 1259 AppLaunchRunnable(Intent intent, String launchReason)1260 public AppLaunchRunnable(Intent intent, String launchReason) { 1261 mLaunchIntent = intent; 1262 mLaunchReason = launchReason; 1263 mLaunchResult = new AppLaunchResult(); 1264 } 1265 getResult()1266 public AppLaunchResult getResult() { 1267 return mLaunchResult; 1268 } 1269 run()1270 public void run() { 1271 File launchFile = null; 1272 try { 1273 String packageName = mLaunchIntent.getComponent().getPackageName(); 1274 String componentName = mLaunchIntent.getComponent().flattenToShortString(); 1275 if (mForceStopApp) { 1276 mAm.forceStopPackage(packageName, UserHandle.USER_CURRENT); 1277 } 1278 String launchCmd = String.format("%s %s", APP_LAUNCH_CMD, componentName); 1279 if (mSimplePerfAppOnly) { 1280 try { 1281 // executeShellCommand cannot handle shell specific actions, like '&'. 1282 // Therefore, we create a file containing the command and make that 1283 // the command to launch. 1284 launchFile = File.createTempFile(LAUNCH_SCRIPT_NAME, ".sh"); 1285 launchFile.setExecutable(true); 1286 try (FileOutputStream stream = new FileOutputStream(launchFile); 1287 BufferedWriter writer = 1288 new BufferedWriter(new OutputStreamWriter(stream))) { 1289 String cmd = String.format(SIMPLEPERF_APP_CMD, packageName, launchCmd); 1290 // In the file, we need to escape any "$". 1291 cmd = cmd.replace("$", "\\$"); 1292 writer.write(cmd); 1293 } 1294 launchCmd = launchFile.getAbsolutePath(); 1295 } catch (IOException e) { 1296 Log.w(TAG, "Error writing the launch command", e); 1297 return; 1298 } 1299 } else if (null != mSimplePerfCmd) { 1300 launchCmd = String.format("%s %s", mSimplePerfCmd, launchCmd); 1301 } 1302 Log.v(TAG, "Final launch cmd:" + launchCmd); 1303 ParcelFileDescriptor parcelDesc = getInstrumentation().getUiAutomation() 1304 .executeShellCommand(launchCmd); 1305 mLaunchResult = parseLaunchTimeAndWrite(parcelDesc, String.format 1306 ("App Launch :%s %s", componentName, mLaunchReason)); 1307 } catch (RemoteException e) { 1308 Log.w(TAG, "Error launching app", e); 1309 } finally { 1310 if (launchFile != null) { 1311 launchFile.delete(); 1312 } 1313 } 1314 } 1315 1316 /** 1317 * Method to parse the launch time info and write the result to file 1318 * 1319 * @param parcelDesc 1320 * @return 1321 */ parseLaunchTimeAndWrite(ParcelFileDescriptor parcelDesc, String headerInfo)1322 private AppLaunchResult parseLaunchTimeAndWrite(ParcelFileDescriptor parcelDesc, 1323 String headerInfo) { 1324 String launchTime = "-1"; 1325 String cpuCycles = "-1"; 1326 String majorFaults = "-1"; 1327 boolean launchSuccess = false; 1328 try { 1329 InputStream inputStream = new FileInputStream(parcelDesc.getFileDescriptor()); 1330 /* SAMPLE OUTPUT : Cold launch 1331 Starting: Intent { cmp=com.google.android.calculator/com.android.calculator2.Calculator } 1332 Status: ok 1333 LaunchState: COLD 1334 Activity: com.google.android.calculator/com.android.calculator2.Calculator 1335 TotalTime: 357 1336 WaitTime: 377 1337 Complete*/ 1338 /* SAMPLE OUTPUT : Hot launch 1339 Starting: Intent { cmp=com.google.android.calculator/com.android.calculator2.Calculator } 1340 Warning: Activity not started, its current task has been brought to the front 1341 Status: ok 1342 LaunchState: HOT 1343 Activity: com.google.android.calculator/com.android.calculator2.CalculatorGoogle 1344 TotalTime: 60 1345 WaitTime: 67 1346 Complete*/ 1347 /* WITH SIMPLEPERF : 1348 Performance counter statistics, 1349 6595722690,cpu-cycles,4.511040,GHz,(100%), 1350 0,major-faults,0.000,/sec,(100%), 1351 Total test time,1.462129,seconds,*/ 1352 BufferedReader bufferedReader = new BufferedReader(new InputStreamReader( 1353 inputStream)); 1354 String line; 1355 mBufferedWriter.newLine(); 1356 mBufferedWriter.write(headerInfo); 1357 mBufferedWriter.newLine(); 1358 while ((line = bufferedReader.readLine()) != null) { 1359 mBufferedWriter.write(line); 1360 mBufferedWriter.newLine(); 1361 if (line.startsWith(SUCCESS_MESSAGE)) { 1362 launchSuccess = true; 1363 } 1364 if (!launchSuccess) { 1365 continue; 1366 } 1367 // Parse TotalTime which is the launch time 1368 if (line.startsWith(TOTAL_TIME_MESSAGE)) { 1369 String launchSplit[] = line.split(":"); 1370 launchTime = launchSplit[1].trim(); 1371 } 1372 1373 if (mSimplePerfAppOnly) { 1374 if (line.contains(",cpu-cycles,")) { 1375 cpuCycles = line.split(",")[0].trim(); 1376 } else if (line.contains(",major-faults,")) { 1377 majorFaults = line.split(",")[0].trim(); 1378 } 1379 } 1380 } 1381 mBufferedWriter.flush(); 1382 inputStream.close(); 1383 } catch (IOException e) { 1384 Log.w(TAG, "Error parsing launch time and writing to file", e); 1385 } 1386 return new AppLaunchResult(launchTime, cpuCycles, majorFaults); 1387 } 1388 1389 } 1390 1391 /** 1392 * Start the screen recording while launching the app. 1393 * 1394 * @param appName 1395 * @param launchReason 1396 */ startRecording(String appName, String launchReason)1397 private void startRecording(String appName, String launchReason) { 1398 Log.v(TAG, "Started Recording"); 1399 mCurrentThread = new RecordingThread("test-screen-record", 1400 String.format("%s_%s", appName, launchReason)); 1401 mCurrentThread.start(); 1402 } 1403 1404 /** 1405 * Stop already started screen recording. 1406 */ stopRecording()1407 private void stopRecording() { 1408 // Skip if not directory. 1409 if (launchSubDir == null) { 1410 return; 1411 } 1412 1413 // Add some extra time to the video end. 1414 SystemClock.sleep(VIDEO_TAIL_BUFFER); 1415 // Ctrl + C all screen record processes. 1416 mCurrentThread.cancel(); 1417 // Wait for the thread to completely die. 1418 try { 1419 mCurrentThread.join(); 1420 } catch (InterruptedException ex) { 1421 Log.e(TAG, "Interrupted when joining the recording thread.", ex); 1422 } 1423 Log.v(TAG, "Stopped Recording"); 1424 } 1425 1426 /** Returns the recording's name for part {@code part} of launch description. */ getOutputFile(String description, int part)1427 private File getOutputFile(String description, int part) { 1428 // Omit the iteration number for the first iteration. 1429 final String fileName = 1430 String.format( 1431 "%s-video%s.mp4", description, part == 1 ? "" : part); 1432 return Paths.get(launchSubDir.getAbsolutePath(), description).toFile(); 1433 } 1434 1435 1436 /** 1437 * Encapsulates the start and stop screen recording logic. 1438 * Copied from ScreenRecordCollector. 1439 */ 1440 private class RecordingThread extends Thread { 1441 private final String mDescription; 1442 private final List<File> mRecordings; 1443 1444 private boolean mContinue; 1445 RecordingThread(String name, String description)1446 public RecordingThread(String name, String description) { 1447 super(name); 1448 1449 mContinue = true; 1450 mRecordings = new ArrayList<>(); 1451 1452 assertNotNull("No test description provided for recording.", description); 1453 mDescription = description; 1454 } 1455 1456 @Override run()1457 public void run() { 1458 try { 1459 // Start at i = 1 to encode parts as X.mp4, X2.mp4, X3.mp4, etc. 1460 for (int i = 1; i <= MAX_RECORDING_PARTS && mContinue; i++) { 1461 File output = getOutputFile(mDescription, i); 1462 Log.d( 1463 TAG, 1464 String.format("Recording screen to %s", output.getAbsolutePath())); 1465 mRecordings.add(output); 1466 // Make sure not to block on this background command in the main thread so 1467 // that the test continues to run, but block in this thread so it does not 1468 // trigger a new screen recording session before the prior one completes. 1469 getDevice().executeShellCommand( 1470 String.format("screenrecord %s", output.getAbsolutePath())); 1471 } 1472 } catch (IOException e) { 1473 throw new RuntimeException("Caught exception while screen recording."); 1474 } 1475 } 1476 cancel()1477 public void cancel() { 1478 mContinue = false; 1479 1480 // Identify the screenrecord PIDs and send SIGINT 2 (Ctrl + C) to each. 1481 try { 1482 String[] pids = getDevice().executeShellCommand( 1483 "pidof screenrecord").split(" "); 1484 for (String pid : pids) { 1485 // Avoid empty process ids, because of weird splitting behavior. 1486 if (pid.isEmpty()) { 1487 continue; 1488 } 1489 1490 getDevice().executeShellCommand( 1491 String.format("kill -2 %s", pid)); 1492 Log.d( 1493 TAG, 1494 String.format("Sent SIGINT 2 to screenrecord process (%s)", pid)); 1495 } 1496 } catch (IOException e) { 1497 throw new RuntimeException("Failed to kill screen recording process."); 1498 } 1499 } 1500 getRecordings()1501 public List<File> getRecordings() { 1502 return mRecordings; 1503 } 1504 } 1505 getDevice()1506 public UiDevice getDevice() { 1507 if (mDevice == null) { 1508 mDevice = UiDevice.getInstance(getInstrumentation()); 1509 } 1510 return mDevice; 1511 } 1512 } 1513