1 /** 2 * Copyright (C) 2015 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); you may not 5 * use this file except in compliance with the License. You may obtain a copy 6 * 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, WITHOUT 12 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the 13 * License for the specific language governing permissions and limitations 14 * under the License. 15 */ 16 17 package com.android.server.usage; 18 19 import static android.app.usage.UsageStatsManager.REASON_MAIN_DEFAULT; 20 import static android.app.usage.UsageStatsManager.REASON_MAIN_FORCED; 21 import static android.app.usage.UsageStatsManager.REASON_MAIN_MASK; 22 import static android.app.usage.UsageStatsManager.REASON_MAIN_PREDICTED; 23 import static android.app.usage.UsageStatsManager.REASON_MAIN_USAGE; 24 import static android.app.usage.UsageStatsManager.REASON_SUB_USAGE_USER_INTERACTION; 25 import static android.app.usage.UsageStatsManager.STANDBY_BUCKET_ACTIVE; 26 import static android.app.usage.UsageStatsManager.STANDBY_BUCKET_NEVER; 27 import static android.app.usage.UsageStatsManager.STANDBY_BUCKET_RARE; 28 import static android.app.usage.UsageStatsManager.STANDBY_BUCKET_WORKING_SET; 29 30 import android.app.usage.AppStandbyInfo; 31 import android.app.usage.UsageStatsManager; 32 import android.os.SystemClock; 33 import android.util.ArrayMap; 34 import android.util.AtomicFile; 35 import android.util.Slog; 36 import android.util.SparseArray; 37 import android.util.TimeUtils; 38 import android.util.Xml; 39 40 import com.android.internal.annotations.VisibleForTesting; 41 import com.android.internal.util.FastXmlSerializer; 42 import com.android.internal.util.IndentingPrintWriter; 43 44 import libcore.io.IoUtils; 45 46 import org.xmlpull.v1.XmlPullParser; 47 import org.xmlpull.v1.XmlPullParserException; 48 49 import java.io.BufferedOutputStream; 50 import java.io.BufferedReader; 51 import java.io.File; 52 import java.io.FileInputStream; 53 import java.io.FileOutputStream; 54 import java.io.FileReader; 55 import java.io.IOException; 56 import java.nio.charset.StandardCharsets; 57 import java.util.ArrayList; 58 59 /** 60 * Keeps track of recent active state changes in apps. 61 * Access should be guarded by a lock by the caller. 62 */ 63 public class AppIdleHistory { 64 65 private static final String TAG = "AppIdleHistory"; 66 67 private static final boolean DEBUG = AppStandbyController.DEBUG; 68 69 // History for all users and all packages 70 private SparseArray<ArrayMap<String,AppUsageHistory>> mIdleHistory = new SparseArray<>(); 71 private static final long ONE_MINUTE = 60 * 1000; 72 73 private static final int STANDBY_BUCKET_UNKNOWN = -1; 74 75 @VisibleForTesting 76 static final String APP_IDLE_FILENAME = "app_idle_stats.xml"; 77 private static final String TAG_PACKAGES = "packages"; 78 private static final String TAG_PACKAGE = "package"; 79 private static final String ATTR_NAME = "name"; 80 // Screen on timebase time when app was last used 81 private static final String ATTR_SCREEN_IDLE = "screenIdleTime"; 82 // Elapsed timebase time when app was last used 83 private static final String ATTR_ELAPSED_IDLE = "elapsedIdleTime"; 84 // Elapsed timebase time when the app bucket was last predicted externally 85 private static final String ATTR_LAST_PREDICTED_TIME = "lastPredictedTime"; 86 // The standby bucket for the app 87 private static final String ATTR_CURRENT_BUCKET = "appLimitBucket"; 88 // The reason the app was put in the above bucket 89 private static final String ATTR_BUCKETING_REASON = "bucketReason"; 90 // The last time a job was run for this app 91 private static final String ATTR_LAST_RUN_JOB_TIME = "lastJobRunTime"; 92 // The time when the forced active state can be overridden. 93 private static final String ATTR_BUCKET_ACTIVE_TIMEOUT_TIME = "activeTimeoutTime"; 94 // The time when the forced working_set state can be overridden. 95 private static final String ATTR_BUCKET_WORKING_SET_TIMEOUT_TIME = "workingSetTimeoutTime"; 96 97 // device on time = mElapsedDuration + (timeNow - mElapsedSnapshot) 98 private long mElapsedSnapshot; // Elapsed time snapshot when last write of mDeviceOnDuration 99 private long mElapsedDuration; // Total device on duration since device was "born" 100 101 // screen on time = mScreenOnDuration + (timeNow - mScreenOnSnapshot) 102 private long mScreenOnSnapshot; // Elapsed time snapshot when last write of mScreenOnDuration 103 private long mScreenOnDuration; // Total screen on duration since device was "born" 104 105 private final File mStorageDir; 106 107 private boolean mScreenOn; 108 109 static class AppUsageHistory { 110 // Last used time using elapsed timebase 111 long lastUsedElapsedTime; 112 // Last used time using screen_on timebase 113 long lastUsedScreenTime; 114 // Last predicted time using elapsed timebase 115 long lastPredictedTime; 116 // Last predicted bucket 117 @UsageStatsManager.StandbyBuckets 118 int lastPredictedBucket = STANDBY_BUCKET_UNKNOWN; 119 // Standby bucket 120 @UsageStatsManager.StandbyBuckets 121 int currentBucket; 122 // Reason for setting the standby bucket. The value here is a combination of 123 // one of UsageStatsManager.REASON_MAIN_* and one (or none) of 124 // UsageStatsManager.REASON_SUB_*. Also see REASON_MAIN_MASK and REASON_SUB_MASK. 125 int bucketingReason; 126 // In-memory only, last bucket for which the listeners were informed 127 int lastInformedBucket; 128 // The last time a job was run for this app, using elapsed timebase 129 long lastJobRunTime; 130 // When should the bucket active state timeout, in elapsed timebase, if greater than 131 // lastUsedElapsedTime. 132 // This is used to keep the app in a high bucket regardless of other timeouts and 133 // predictions. 134 long bucketActiveTimeoutTime; 135 // If there's a forced working_set state, this is when it times out. This can be sitting 136 // under any active state timeout, so that it becomes applicable after the active state 137 // timeout expires. 138 long bucketWorkingSetTimeoutTime; 139 } 140 AppIdleHistory(File storageDir, long elapsedRealtime)141 AppIdleHistory(File storageDir, long elapsedRealtime) { 142 mElapsedSnapshot = elapsedRealtime; 143 mScreenOnSnapshot = elapsedRealtime; 144 mStorageDir = storageDir; 145 readScreenOnTime(); 146 } 147 updateDisplay(boolean screenOn, long elapsedRealtime)148 public void updateDisplay(boolean screenOn, long elapsedRealtime) { 149 if (screenOn == mScreenOn) return; 150 151 mScreenOn = screenOn; 152 if (mScreenOn) { 153 mScreenOnSnapshot = elapsedRealtime; 154 } else { 155 mScreenOnDuration += elapsedRealtime - mScreenOnSnapshot; 156 mElapsedDuration += elapsedRealtime - mElapsedSnapshot; 157 mElapsedSnapshot = elapsedRealtime; 158 } 159 if (DEBUG) Slog.d(TAG, "mScreenOnSnapshot=" + mScreenOnSnapshot 160 + ", mScreenOnDuration=" + mScreenOnDuration 161 + ", mScreenOn=" + mScreenOn); 162 } 163 getScreenOnTime(long elapsedRealtime)164 public long getScreenOnTime(long elapsedRealtime) { 165 long screenOnTime = mScreenOnDuration; 166 if (mScreenOn) { 167 screenOnTime += elapsedRealtime - mScreenOnSnapshot; 168 } 169 return screenOnTime; 170 } 171 172 @VisibleForTesting getScreenOnTimeFile()173 File getScreenOnTimeFile() { 174 return new File(mStorageDir, "screen_on_time"); 175 } 176 readScreenOnTime()177 private void readScreenOnTime() { 178 File screenOnTimeFile = getScreenOnTimeFile(); 179 if (screenOnTimeFile.exists()) { 180 try { 181 BufferedReader reader = new BufferedReader(new FileReader(screenOnTimeFile)); 182 mScreenOnDuration = Long.parseLong(reader.readLine()); 183 mElapsedDuration = Long.parseLong(reader.readLine()); 184 reader.close(); 185 } catch (IOException | NumberFormatException e) { 186 } 187 } else { 188 writeScreenOnTime(); 189 } 190 } 191 writeScreenOnTime()192 private void writeScreenOnTime() { 193 AtomicFile screenOnTimeFile = new AtomicFile(getScreenOnTimeFile()); 194 FileOutputStream fos = null; 195 try { 196 fos = screenOnTimeFile.startWrite(); 197 fos.write((Long.toString(mScreenOnDuration) + "\n" 198 + Long.toString(mElapsedDuration) + "\n").getBytes()); 199 screenOnTimeFile.finishWrite(fos); 200 } catch (IOException ioe) { 201 screenOnTimeFile.failWrite(fos); 202 } 203 } 204 205 /** 206 * To be called periodically to keep track of elapsed time when app idle times are written 207 */ writeAppIdleDurations()208 public void writeAppIdleDurations() { 209 final long elapsedRealtime = SystemClock.elapsedRealtime(); 210 // Only bump up and snapshot the elapsed time. Don't change screen on duration. 211 mElapsedDuration += elapsedRealtime - mElapsedSnapshot; 212 mElapsedSnapshot = elapsedRealtime; 213 writeScreenOnTime(); 214 } 215 216 /** 217 * Mark the app as used and update the bucket if necessary. If there is a timeout specified 218 * that's in the future, then the usage event is temporary and keeps the app in the specified 219 * bucket at least until the timeout is reached. This can be used to keep the app in an 220 * elevated bucket for a while until some important task gets to run. 221 * @param appUsageHistory the usage record for the app being updated 222 * @param packageName name of the app being updated, for logging purposes 223 * @param newBucket the bucket to set the app to 224 * @param usageReason the sub-reason for usage, one of REASON_SUB_USAGE_* 225 * @param elapsedRealtime mark as used time if non-zero 226 * @param timeout set the timeout of the specified bucket, if non-zero. Can only be used 227 * with bucket values of ACTIVE and WORKING_SET. 228 * @return 229 */ reportUsage(AppUsageHistory appUsageHistory, String packageName, int newBucket, int usageReason, long elapsedRealtime, long timeout)230 public AppUsageHistory reportUsage(AppUsageHistory appUsageHistory, String packageName, 231 int newBucket, int usageReason, long elapsedRealtime, long timeout) { 232 // Set the timeout if applicable 233 if (timeout > elapsedRealtime) { 234 // Convert to elapsed timebase 235 final long timeoutTime = mElapsedDuration + (timeout - mElapsedSnapshot); 236 if (newBucket == STANDBY_BUCKET_ACTIVE) { 237 appUsageHistory.bucketActiveTimeoutTime = Math.max(timeoutTime, 238 appUsageHistory.bucketActiveTimeoutTime); 239 } else if (newBucket == STANDBY_BUCKET_WORKING_SET) { 240 appUsageHistory.bucketWorkingSetTimeoutTime = Math.max(timeoutTime, 241 appUsageHistory.bucketWorkingSetTimeoutTime); 242 } else { 243 throw new IllegalArgumentException("Cannot set a timeout on bucket=" + 244 newBucket); 245 } 246 } 247 248 if (elapsedRealtime != 0) { 249 appUsageHistory.lastUsedElapsedTime = mElapsedDuration 250 + (elapsedRealtime - mElapsedSnapshot); 251 appUsageHistory.lastUsedScreenTime = getScreenOnTime(elapsedRealtime); 252 } 253 254 if (appUsageHistory.currentBucket > newBucket) { 255 appUsageHistory.currentBucket = newBucket; 256 if (DEBUG) { 257 Slog.d(TAG, "Moved " + packageName + " to bucket=" + appUsageHistory 258 .currentBucket 259 + ", reason=0x0" + Integer.toHexString(appUsageHistory.bucketingReason)); 260 } 261 } 262 appUsageHistory.bucketingReason = REASON_MAIN_USAGE | usageReason; 263 264 return appUsageHistory; 265 } 266 267 /** 268 * Mark the app as used and update the bucket if necessary. If there is a timeout specified 269 * that's in the future, then the usage event is temporary and keeps the app in the specified 270 * bucket at least until the timeout is reached. This can be used to keep the app in an 271 * elevated bucket for a while until some important task gets to run. 272 * @param packageName 273 * @param userId 274 * @param newBucket the bucket to set the app to 275 * @param usageReason sub reason for usage 276 * @param nowElapsed mark as used time if non-zero 277 * @param timeout set the timeout of the specified bucket, if non-zero. Can only be used 278 * with bucket values of ACTIVE and WORKING_SET. 279 * @return 280 */ reportUsage(String packageName, int userId, int newBucket, int usageReason, long nowElapsed, long timeout)281 public AppUsageHistory reportUsage(String packageName, int userId, int newBucket, 282 int usageReason, long nowElapsed, long timeout) { 283 ArrayMap<String, AppUsageHistory> userHistory = getUserHistory(userId); 284 AppUsageHistory history = getPackageHistory(userHistory, packageName, nowElapsed, true); 285 return reportUsage(history, packageName, newBucket, usageReason, nowElapsed, timeout); 286 } 287 getUserHistory(int userId)288 private ArrayMap<String, AppUsageHistory> getUserHistory(int userId) { 289 ArrayMap<String, AppUsageHistory> userHistory = mIdleHistory.get(userId); 290 if (userHistory == null) { 291 userHistory = new ArrayMap<>(); 292 mIdleHistory.put(userId, userHistory); 293 readAppIdleTimes(userId, userHistory); 294 } 295 return userHistory; 296 } 297 getPackageHistory(ArrayMap<String, AppUsageHistory> userHistory, String packageName, long elapsedRealtime, boolean create)298 private AppUsageHistory getPackageHistory(ArrayMap<String, AppUsageHistory> userHistory, 299 String packageName, long elapsedRealtime, boolean create) { 300 AppUsageHistory appUsageHistory = userHistory.get(packageName); 301 if (appUsageHistory == null && create) { 302 appUsageHistory = new AppUsageHistory(); 303 appUsageHistory.lastUsedElapsedTime = getElapsedTime(elapsedRealtime); 304 appUsageHistory.lastUsedScreenTime = getScreenOnTime(elapsedRealtime); 305 appUsageHistory.lastPredictedTime = getElapsedTime(0); 306 appUsageHistory.currentBucket = STANDBY_BUCKET_NEVER; 307 appUsageHistory.bucketingReason = REASON_MAIN_DEFAULT; 308 appUsageHistory.lastInformedBucket = -1; 309 appUsageHistory.lastJobRunTime = Long.MIN_VALUE; // long long time ago 310 userHistory.put(packageName, appUsageHistory); 311 } 312 return appUsageHistory; 313 } 314 onUserRemoved(int userId)315 public void onUserRemoved(int userId) { 316 mIdleHistory.remove(userId); 317 } 318 isIdle(String packageName, int userId, long elapsedRealtime)319 public boolean isIdle(String packageName, int userId, long elapsedRealtime) { 320 ArrayMap<String, AppUsageHistory> userHistory = getUserHistory(userId); 321 AppUsageHistory appUsageHistory = 322 getPackageHistory(userHistory, packageName, elapsedRealtime, true); 323 return appUsageHistory.currentBucket >= STANDBY_BUCKET_RARE; 324 } 325 getAppUsageHistory(String packageName, int userId, long elapsedRealtime)326 public AppUsageHistory getAppUsageHistory(String packageName, int userId, 327 long elapsedRealtime) { 328 ArrayMap<String, AppUsageHistory> userHistory = getUserHistory(userId); 329 AppUsageHistory appUsageHistory = 330 getPackageHistory(userHistory, packageName, elapsedRealtime, true); 331 return appUsageHistory; 332 } 333 setAppStandbyBucket(String packageName, int userId, long elapsedRealtime, int bucket, int reason)334 public void setAppStandbyBucket(String packageName, int userId, long elapsedRealtime, 335 int bucket, int reason) { 336 setAppStandbyBucket(packageName, userId, elapsedRealtime, bucket, reason, false); 337 } 338 setAppStandbyBucket(String packageName, int userId, long elapsedRealtime, int bucket, int reason, boolean resetTimeout)339 public void setAppStandbyBucket(String packageName, int userId, long elapsedRealtime, 340 int bucket, int reason, boolean resetTimeout) { 341 ArrayMap<String, AppUsageHistory> userHistory = getUserHistory(userId); 342 AppUsageHistory appUsageHistory = 343 getPackageHistory(userHistory, packageName, elapsedRealtime, true); 344 appUsageHistory.currentBucket = bucket; 345 appUsageHistory.bucketingReason = reason; 346 347 final long elapsed = getElapsedTime(elapsedRealtime); 348 349 if ((reason & REASON_MAIN_MASK) == REASON_MAIN_PREDICTED) { 350 appUsageHistory.lastPredictedTime = elapsed; 351 appUsageHistory.lastPredictedBucket = bucket; 352 } 353 if (resetTimeout) { 354 appUsageHistory.bucketActiveTimeoutTime = elapsed; 355 appUsageHistory.bucketWorkingSetTimeoutTime = elapsed; 356 } 357 if (DEBUG) { 358 Slog.d(TAG, "Moved " + packageName + " to bucket=" + appUsageHistory.currentBucket 359 + ", reason=0x0" + Integer.toHexString(appUsageHistory.bucketingReason)); 360 } 361 } 362 363 /** 364 * Update the prediction for the app but don't change the actual bucket 365 * @param app The app for which the prediction was made 366 * @param elapsedTimeAdjusted The elapsed time in the elapsed duration timebase 367 * @param bucket The predicted bucket 368 */ updateLastPrediction(AppUsageHistory app, long elapsedTimeAdjusted, int bucket)369 public void updateLastPrediction(AppUsageHistory app, long elapsedTimeAdjusted, int bucket) { 370 app.lastPredictedTime = elapsedTimeAdjusted; 371 app.lastPredictedBucket = bucket; 372 } 373 374 /** 375 * Marks the last time a job was run, with the given elapsedRealtime. The time stored is 376 * based on the elapsed timebase. 377 * @param packageName 378 * @param userId 379 * @param elapsedRealtime 380 */ setLastJobRunTime(String packageName, int userId, long elapsedRealtime)381 public void setLastJobRunTime(String packageName, int userId, long elapsedRealtime) { 382 ArrayMap<String, AppUsageHistory> userHistory = getUserHistory(userId); 383 AppUsageHistory appUsageHistory = 384 getPackageHistory(userHistory, packageName, elapsedRealtime, true); 385 appUsageHistory.lastJobRunTime = getElapsedTime(elapsedRealtime); 386 } 387 388 /** 389 * Returns the time since the last job was run for this app. This can be larger than the 390 * current elapsedRealtime, in case it happened before boot or a really large value if no jobs 391 * were ever run. 392 * @param packageName 393 * @param userId 394 * @param elapsedRealtime 395 * @return 396 */ getTimeSinceLastJobRun(String packageName, int userId, long elapsedRealtime)397 public long getTimeSinceLastJobRun(String packageName, int userId, long elapsedRealtime) { 398 ArrayMap<String, AppUsageHistory> userHistory = getUserHistory(userId); 399 AppUsageHistory appUsageHistory = 400 getPackageHistory(userHistory, packageName, elapsedRealtime, false); 401 // Don't adjust the default, else it'll wrap around to a positive value 402 if (appUsageHistory == null || appUsageHistory.lastJobRunTime == Long.MIN_VALUE) { 403 return Long.MAX_VALUE; 404 } 405 return getElapsedTime(elapsedRealtime) - appUsageHistory.lastJobRunTime; 406 } 407 getAppStandbyBucket(String packageName, int userId, long elapsedRealtime)408 public int getAppStandbyBucket(String packageName, int userId, long elapsedRealtime) { 409 ArrayMap<String, AppUsageHistory> userHistory = getUserHistory(userId); 410 AppUsageHistory appUsageHistory = 411 getPackageHistory(userHistory, packageName, elapsedRealtime, false); 412 return appUsageHistory == null ? STANDBY_BUCKET_NEVER : appUsageHistory.currentBucket; 413 } 414 getAppStandbyBuckets(int userId, boolean appIdleEnabled)415 public ArrayList<AppStandbyInfo> getAppStandbyBuckets(int userId, boolean appIdleEnabled) { 416 ArrayMap<String, AppUsageHistory> userHistory = getUserHistory(userId); 417 int size = userHistory.size(); 418 ArrayList<AppStandbyInfo> buckets = new ArrayList<>(size); 419 for (int i = 0; i < size; i++) { 420 buckets.add(new AppStandbyInfo(userHistory.keyAt(i), 421 appIdleEnabled ? userHistory.valueAt(i).currentBucket : STANDBY_BUCKET_ACTIVE)); 422 } 423 return buckets; 424 } 425 getAppStandbyReason(String packageName, int userId, long elapsedRealtime)426 public int getAppStandbyReason(String packageName, int userId, long elapsedRealtime) { 427 ArrayMap<String, AppUsageHistory> userHistory = getUserHistory(userId); 428 AppUsageHistory appUsageHistory = 429 getPackageHistory(userHistory, packageName, elapsedRealtime, false); 430 return appUsageHistory != null ? appUsageHistory.bucketingReason : 0; 431 } 432 getElapsedTime(long elapsedRealtime)433 public long getElapsedTime(long elapsedRealtime) { 434 return (elapsedRealtime - mElapsedSnapshot + mElapsedDuration); 435 } 436 437 /* Returns the new standby bucket the app is assigned to */ setIdle(String packageName, int userId, boolean idle, long elapsedRealtime)438 public int setIdle(String packageName, int userId, boolean idle, long elapsedRealtime) { 439 ArrayMap<String, AppUsageHistory> userHistory = getUserHistory(userId); 440 AppUsageHistory appUsageHistory = getPackageHistory(userHistory, packageName, 441 elapsedRealtime, true); 442 if (idle) { 443 appUsageHistory.currentBucket = STANDBY_BUCKET_RARE; 444 appUsageHistory.bucketingReason = REASON_MAIN_FORCED; 445 } else { 446 appUsageHistory.currentBucket = STANDBY_BUCKET_ACTIVE; 447 // This is to pretend that the app was just used, don't freeze the state anymore. 448 appUsageHistory.bucketingReason = REASON_MAIN_USAGE | REASON_SUB_USAGE_USER_INTERACTION; 449 } 450 return appUsageHistory.currentBucket; 451 } 452 clearUsage(String packageName, int userId)453 public void clearUsage(String packageName, int userId) { 454 ArrayMap<String, AppUsageHistory> userHistory = getUserHistory(userId); 455 userHistory.remove(packageName); 456 } 457 shouldInformListeners(String packageName, int userId, long elapsedRealtime, int bucket)458 boolean shouldInformListeners(String packageName, int userId, 459 long elapsedRealtime, int bucket) { 460 ArrayMap<String, AppUsageHistory> userHistory = getUserHistory(userId); 461 AppUsageHistory appUsageHistory = getPackageHistory(userHistory, packageName, 462 elapsedRealtime, true); 463 if (appUsageHistory.lastInformedBucket != bucket) { 464 appUsageHistory.lastInformedBucket = bucket; 465 return true; 466 } 467 return false; 468 } 469 470 /** 471 * Returns the index in the arrays of screenTimeThresholds and elapsedTimeThresholds 472 * that corresponds to how long since the app was used. 473 * @param packageName 474 * @param userId 475 * @param elapsedRealtime current time 476 * @param screenTimeThresholds Array of screen times, in ascending order, first one is 0 477 * @param elapsedTimeThresholds Array of elapsed time, in ascending order, first one is 0 478 * @return The index whose values the app's used time exceeds (in both arrays) 479 */ getThresholdIndex(String packageName, int userId, long elapsedRealtime, long[] screenTimeThresholds, long[] elapsedTimeThresholds)480 int getThresholdIndex(String packageName, int userId, long elapsedRealtime, 481 long[] screenTimeThresholds, long[] elapsedTimeThresholds) { 482 ArrayMap<String, AppUsageHistory> userHistory = getUserHistory(userId); 483 AppUsageHistory appUsageHistory = getPackageHistory(userHistory, packageName, 484 elapsedRealtime, false); 485 // If we don't have any state for the app, assume never used 486 if (appUsageHistory == null) return screenTimeThresholds.length - 1; 487 488 long screenOnDelta = getScreenOnTime(elapsedRealtime) - appUsageHistory.lastUsedScreenTime; 489 long elapsedDelta = getElapsedTime(elapsedRealtime) - appUsageHistory.lastUsedElapsedTime; 490 491 if (DEBUG) Slog.d(TAG, packageName 492 + " lastUsedScreen=" + appUsageHistory.lastUsedScreenTime 493 + " lastUsedElapsed=" + appUsageHistory.lastUsedElapsedTime); 494 if (DEBUG) Slog.d(TAG, packageName + " screenOn=" + screenOnDelta 495 + ", elapsed=" + elapsedDelta); 496 for (int i = screenTimeThresholds.length - 1; i >= 0; i--) { 497 if (screenOnDelta >= screenTimeThresholds[i] 498 && elapsedDelta >= elapsedTimeThresholds[i]) { 499 return i; 500 } 501 } 502 return 0; 503 } 504 505 @VisibleForTesting getUserFile(int userId)506 File getUserFile(int userId) { 507 return new File(new File(new File(mStorageDir, "users"), 508 Integer.toString(userId)), APP_IDLE_FILENAME); 509 } 510 511 512 /** 513 * Check if App Idle File exists on disk 514 * @param userId 515 * @return true if file exists 516 */ userFileExists(int userId)517 public boolean userFileExists(int userId) { 518 return getUserFile(userId).exists(); 519 } 520 readAppIdleTimes(int userId, ArrayMap<String, AppUsageHistory> userHistory)521 private void readAppIdleTimes(int userId, ArrayMap<String, AppUsageHistory> userHistory) { 522 FileInputStream fis = null; 523 try { 524 AtomicFile appIdleFile = new AtomicFile(getUserFile(userId)); 525 fis = appIdleFile.openRead(); 526 XmlPullParser parser = Xml.newPullParser(); 527 parser.setInput(fis, StandardCharsets.UTF_8.name()); 528 529 int type; 530 while ((type = parser.next()) != XmlPullParser.START_TAG 531 && type != XmlPullParser.END_DOCUMENT) { 532 // Skip 533 } 534 535 if (type != XmlPullParser.START_TAG) { 536 Slog.e(TAG, "Unable to read app idle file for user " + userId); 537 return; 538 } 539 if (!parser.getName().equals(TAG_PACKAGES)) { 540 return; 541 } 542 while ((type = parser.next()) != XmlPullParser.END_DOCUMENT) { 543 if (type == XmlPullParser.START_TAG) { 544 final String name = parser.getName(); 545 if (name.equals(TAG_PACKAGE)) { 546 final String packageName = parser.getAttributeValue(null, ATTR_NAME); 547 AppUsageHistory appUsageHistory = new AppUsageHistory(); 548 appUsageHistory.lastUsedElapsedTime = 549 Long.parseLong(parser.getAttributeValue(null, ATTR_ELAPSED_IDLE)); 550 appUsageHistory.lastUsedScreenTime = 551 Long.parseLong(parser.getAttributeValue(null, ATTR_SCREEN_IDLE)); 552 appUsageHistory.lastPredictedTime = getLongValue(parser, 553 ATTR_LAST_PREDICTED_TIME, 0L); 554 String currentBucketString = parser.getAttributeValue(null, 555 ATTR_CURRENT_BUCKET); 556 appUsageHistory.currentBucket = currentBucketString == null 557 ? STANDBY_BUCKET_ACTIVE 558 : Integer.parseInt(currentBucketString); 559 String bucketingReason = 560 parser.getAttributeValue(null, ATTR_BUCKETING_REASON); 561 appUsageHistory.lastJobRunTime = getLongValue(parser, 562 ATTR_LAST_RUN_JOB_TIME, Long.MIN_VALUE); 563 appUsageHistory.bucketActiveTimeoutTime = getLongValue(parser, 564 ATTR_BUCKET_ACTIVE_TIMEOUT_TIME, 0L); 565 appUsageHistory.bucketWorkingSetTimeoutTime = getLongValue(parser, 566 ATTR_BUCKET_WORKING_SET_TIMEOUT_TIME, 0L); 567 appUsageHistory.bucketingReason = REASON_MAIN_DEFAULT; 568 if (bucketingReason != null) { 569 try { 570 appUsageHistory.bucketingReason = 571 Integer.parseInt(bucketingReason, 16); 572 } catch (NumberFormatException nfe) { 573 } 574 } 575 appUsageHistory.lastInformedBucket = -1; 576 userHistory.put(packageName, appUsageHistory); 577 } 578 } 579 } 580 } catch (IOException | XmlPullParserException e) { 581 Slog.e(TAG, "Unable to read app idle file for user " + userId); 582 } finally { 583 IoUtils.closeQuietly(fis); 584 } 585 } 586 getLongValue(XmlPullParser parser, String attrName, long defValue)587 private long getLongValue(XmlPullParser parser, String attrName, long defValue) { 588 String value = parser.getAttributeValue(null, attrName); 589 if (value == null) return defValue; 590 return Long.parseLong(value); 591 } 592 writeAppIdleTimes(int userId)593 public void writeAppIdleTimes(int userId) { 594 FileOutputStream fos = null; 595 AtomicFile appIdleFile = new AtomicFile(getUserFile(userId)); 596 try { 597 fos = appIdleFile.startWrite(); 598 final BufferedOutputStream bos = new BufferedOutputStream(fos); 599 600 FastXmlSerializer xml = new FastXmlSerializer(); 601 xml.setOutput(bos, StandardCharsets.UTF_8.name()); 602 xml.startDocument(null, true); 603 xml.setFeature("http://xmlpull.org/v1/doc/features.html#indent-output", true); 604 605 xml.startTag(null, TAG_PACKAGES); 606 607 ArrayMap<String,AppUsageHistory> userHistory = getUserHistory(userId); 608 final int N = userHistory.size(); 609 for (int i = 0; i < N; i++) { 610 String packageName = userHistory.keyAt(i); 611 AppUsageHistory history = userHistory.valueAt(i); 612 xml.startTag(null, TAG_PACKAGE); 613 xml.attribute(null, ATTR_NAME, packageName); 614 xml.attribute(null, ATTR_ELAPSED_IDLE, 615 Long.toString(history.lastUsedElapsedTime)); 616 xml.attribute(null, ATTR_SCREEN_IDLE, 617 Long.toString(history.lastUsedScreenTime)); 618 xml.attribute(null, ATTR_LAST_PREDICTED_TIME, 619 Long.toString(history.lastPredictedTime)); 620 xml.attribute(null, ATTR_CURRENT_BUCKET, 621 Integer.toString(history.currentBucket)); 622 xml.attribute(null, ATTR_BUCKETING_REASON, 623 Integer.toHexString(history.bucketingReason)); 624 if (history.bucketActiveTimeoutTime > 0) { 625 xml.attribute(null, ATTR_BUCKET_ACTIVE_TIMEOUT_TIME, Long.toString(history 626 .bucketActiveTimeoutTime)); 627 } 628 if (history.bucketWorkingSetTimeoutTime > 0) { 629 xml.attribute(null, ATTR_BUCKET_WORKING_SET_TIMEOUT_TIME, Long.toString(history 630 .bucketWorkingSetTimeoutTime)); 631 } 632 if (history.lastJobRunTime != Long.MIN_VALUE) { 633 xml.attribute(null, ATTR_LAST_RUN_JOB_TIME, Long.toString(history 634 .lastJobRunTime)); 635 } 636 xml.endTag(null, TAG_PACKAGE); 637 } 638 639 xml.endTag(null, TAG_PACKAGES); 640 xml.endDocument(); 641 appIdleFile.finishWrite(fos); 642 } catch (Exception e) { 643 appIdleFile.failWrite(fos); 644 Slog.e(TAG, "Error writing app idle file for user " + userId); 645 } 646 } 647 dump(IndentingPrintWriter idpw, int userId, String pkg)648 public void dump(IndentingPrintWriter idpw, int userId, String pkg) { 649 idpw.println("App Standby States:"); 650 idpw.increaseIndent(); 651 ArrayMap<String, AppUsageHistory> userHistory = mIdleHistory.get(userId); 652 final long elapsedRealtime = SystemClock.elapsedRealtime(); 653 final long totalElapsedTime = getElapsedTime(elapsedRealtime); 654 final long screenOnTime = getScreenOnTime(elapsedRealtime); 655 if (userHistory == null) return; 656 final int P = userHistory.size(); 657 for (int p = 0; p < P; p++) { 658 final String packageName = userHistory.keyAt(p); 659 final AppUsageHistory appUsageHistory = userHistory.valueAt(p); 660 if (pkg != null && !pkg.equals(packageName)) { 661 continue; 662 } 663 idpw.print("package=" + packageName); 664 idpw.print(" u=" + userId); 665 idpw.print(" bucket=" + appUsageHistory.currentBucket 666 + " reason=" 667 + UsageStatsManager.reasonToString(appUsageHistory.bucketingReason)); 668 idpw.print(" used="); 669 TimeUtils.formatDuration(totalElapsedTime - appUsageHistory.lastUsedElapsedTime, idpw); 670 idpw.print(" usedScr="); 671 TimeUtils.formatDuration(screenOnTime - appUsageHistory.lastUsedScreenTime, idpw); 672 idpw.print(" lastPred="); 673 TimeUtils.formatDuration(totalElapsedTime - appUsageHistory.lastPredictedTime, idpw); 674 idpw.print(" activeLeft="); 675 TimeUtils.formatDuration(appUsageHistory.bucketActiveTimeoutTime - totalElapsedTime, 676 idpw); 677 idpw.print(" wsLeft="); 678 TimeUtils.formatDuration(appUsageHistory.bucketWorkingSetTimeoutTime - totalElapsedTime, 679 idpw); 680 idpw.print(" lastJob="); 681 TimeUtils.formatDuration(totalElapsedTime - appUsageHistory.lastJobRunTime, idpw); 682 idpw.print(" idle=" + (isIdle(packageName, userId, elapsedRealtime) ? "y" : "n")); 683 idpw.println(); 684 } 685 idpw.println(); 686 idpw.print("totalElapsedTime="); 687 TimeUtils.formatDuration(getElapsedTime(elapsedRealtime), idpw); 688 idpw.println(); 689 idpw.print("totalScreenOnTime="); 690 TimeUtils.formatDuration(getScreenOnTime(elapsedRealtime), idpw); 691 idpw.println(); 692 idpw.decreaseIndent(); 693 } 694 } 695