1 /** 2 * Copyright (C) 2018 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 android.annotation.UserIdInt; 20 import android.app.PendingIntent; 21 import android.app.usage.UsageStatsManagerInternal; 22 import android.os.Handler; 23 import android.os.Looper; 24 import android.os.Message; 25 import android.os.SystemClock; 26 import android.os.UserHandle; 27 import android.util.ArrayMap; 28 import android.util.ArraySet; 29 import android.util.Slog; 30 import android.util.SparseArray; 31 32 import com.android.internal.annotations.GuardedBy; 33 import com.android.internal.annotations.VisibleForTesting; 34 35 import java.io.PrintWriter; 36 import java.lang.ref.WeakReference; 37 import java.util.ArrayList; 38 import java.util.Arrays; 39 40 /** 41 * Monitors and informs of any app time limits exceeded. It must be informed when an app 42 * enters the foreground and exits. Used by UsageStatsService. Manages multiple users. 43 * 44 * Test: atest FrameworksServicesTests:AppTimeLimitControllerTests 45 * Test: manual: frameworks/base/tests/UsageStatsTest 46 */ 47 public class AppTimeLimitController { 48 49 private static final String TAG = "AppTimeLimitController"; 50 51 private static final boolean DEBUG = false; 52 53 /** Lock class for this object */ 54 private static class Lock {} 55 56 /** Lock object for the data in this class. */ 57 private final Lock mLock = new Lock(); 58 59 private final MyHandler mHandler; 60 61 private TimeLimitCallbackListener mListener; 62 63 private static final long MAX_OBSERVER_PER_UID = 1000; 64 65 private static final long ONE_MINUTE = 60_000L; 66 67 private static final Integer ONE = new Integer(1); 68 69 /** Collection of data for each user that has reported usage */ 70 @GuardedBy("mLock") 71 private final SparseArray<UserData> mUsers = new SparseArray<>(); 72 73 /** 74 * Collection of data for each app that is registering observers 75 * WARNING: Entries are currently not removed, based on the assumption there are a small 76 * fixed number of apps on device that can register observers. 77 */ 78 @GuardedBy("mLock") 79 private final SparseArray<ObserverAppData> mObserverApps = new SparseArray<>(); 80 81 private class UserData { 82 /** userId of the user */ 83 private @UserIdInt 84 int userId; 85 86 /** Count of the currently active entities */ 87 public final ArrayMap<String, Integer> currentlyActive = new ArrayMap<>(); 88 89 /** Map from entity name for quick lookup */ 90 public final ArrayMap<String, ArrayList<UsageGroup>> observedMap = new ArrayMap<>(); 91 UserData(@serIdInt int userId)92 private UserData(@UserIdInt int userId) { 93 this.userId = userId; 94 } 95 96 @GuardedBy("mLock") isActive(String[] entities)97 boolean isActive(String[] entities) { 98 // TODO: Consider using a bloom filter here if number of actives becomes large 99 final int size = entities.length; 100 for (int i = 0; i < size; i++) { 101 if (currentlyActive.containsKey(entities[i])) { 102 return true; 103 } 104 } 105 return false; 106 } 107 108 @GuardedBy("mLock") addUsageGroup(UsageGroup group)109 void addUsageGroup(UsageGroup group) { 110 final int size = group.mObserved.length; 111 for (int i = 0; i < size; i++) { 112 ArrayList<UsageGroup> list = observedMap.get(group.mObserved[i]); 113 if (list == null) { 114 list = new ArrayList<>(); 115 observedMap.put(group.mObserved[i], list); 116 } 117 list.add(group); 118 } 119 } 120 121 @GuardedBy("mLock") removeUsageGroup(UsageGroup group)122 void removeUsageGroup(UsageGroup group) { 123 final int size = group.mObserved.length; 124 for (int i = 0; i < size; i++) { 125 final String observed = group.mObserved[i]; 126 final ArrayList<UsageGroup> list = observedMap.get(observed); 127 if (list != null) { 128 list.remove(group); 129 if (list.isEmpty()) { 130 // No more observers for this observed entity, remove from map 131 observedMap.remove(observed); 132 } 133 } 134 } 135 } 136 137 @GuardedBy("mLock") dump(PrintWriter pw)138 void dump(PrintWriter pw) { 139 pw.print(" userId="); 140 pw.println(userId); 141 pw.print(" Currently Active:"); 142 final int nActive = currentlyActive.size(); 143 for (int i = 0; i < nActive; i++) { 144 pw.print(currentlyActive.keyAt(i)); 145 pw.print(", "); 146 } 147 pw.println(); 148 pw.print(" Observed Entities:"); 149 final int nEntities = observedMap.size(); 150 for (int i = 0; i < nEntities; i++) { 151 pw.print(observedMap.keyAt(i)); 152 pw.print(", "); 153 } 154 pw.println(); 155 } 156 } 157 158 159 private class ObserverAppData { 160 /** uid of the observing app */ 161 private int uid; 162 163 /** Map of observerId to details of the time limit group */ 164 SparseArray<AppUsageGroup> appUsageGroups = new SparseArray<>(); 165 166 /** Map of observerId to details of the time limit group */ 167 SparseArray<SessionUsageGroup> sessionUsageGroups = new SparseArray<>(); 168 169 /** Map of observerId to details of the app usage limit group */ 170 SparseArray<AppUsageLimitGroup> appUsageLimitGroups = new SparseArray<>(); 171 ObserverAppData(int uid)172 private ObserverAppData(int uid) { 173 this.uid = uid; 174 } 175 176 @GuardedBy("mLock") removeAppUsageGroup(int observerId)177 void removeAppUsageGroup(int observerId) { 178 appUsageGroups.remove(observerId); 179 } 180 181 @GuardedBy("mLock") removeSessionUsageGroup(int observerId)182 void removeSessionUsageGroup(int observerId) { 183 sessionUsageGroups.remove(observerId); 184 } 185 186 @GuardedBy("mLock") removeAppUsageLimitGroup(int observerId)187 void removeAppUsageLimitGroup(int observerId) { 188 appUsageLimitGroups.remove(observerId); 189 } 190 191 @GuardedBy("mLock") dump(PrintWriter pw)192 void dump(PrintWriter pw) { 193 pw.print(" uid="); 194 pw.println(uid); 195 pw.println(" App Usage Groups:"); 196 final int nAppUsageGroups = appUsageGroups.size(); 197 for (int i = 0; i < nAppUsageGroups; i++) { 198 appUsageGroups.valueAt(i).dump(pw); 199 pw.println(); 200 } 201 pw.println(" Session Usage Groups:"); 202 final int nSessionUsageGroups = sessionUsageGroups.size(); 203 for (int i = 0; i < nSessionUsageGroups; i++) { 204 sessionUsageGroups.valueAt(i).dump(pw); 205 pw.println(); 206 } 207 pw.println(" App Usage Limit Groups:"); 208 final int nAppUsageLimitGroups = appUsageLimitGroups.size(); 209 for (int i = 0; i < nAppUsageLimitGroups; i++) { 210 appUsageLimitGroups.valueAt(i).dump(pw); 211 pw.println(); 212 } 213 } 214 } 215 216 /** 217 * Listener interface for being informed when an app group's time limit is reached. 218 */ 219 public interface TimeLimitCallbackListener { 220 /** 221 * Time limit for a group, keyed by the observerId, has been reached. 222 * 223 * @param observerId The observerId of the group whose limit was reached 224 * @param userId The userId 225 * @param timeLimit The original time limit in milliseconds 226 * @param timeElapsed How much time was actually spent on apps in the group, in 227 * milliseconds 228 * @param callbackIntent The PendingIntent to send when the limit is reached 229 */ onLimitReached(int observerId, @UserIdInt int userId, long timeLimit, long timeElapsed, PendingIntent callbackIntent)230 public void onLimitReached(int observerId, @UserIdInt int userId, long timeLimit, 231 long timeElapsed, PendingIntent callbackIntent); 232 233 /** 234 * Session ended for a group, keyed by the observerId, after limit was reached. 235 * 236 * @param observerId The observerId of the group whose limit was reached 237 * @param userId The userId 238 * @param timeElapsed How much time was actually spent on apps in the group, in 239 * milliseconds 240 * @param callbackIntent The PendingIntent to send when the limit is reached 241 */ onSessionEnd(int observerId, @UserIdInt int userId, long timeElapsed, PendingIntent callbackIntent)242 public void onSessionEnd(int observerId, @UserIdInt int userId, long timeElapsed, 243 PendingIntent callbackIntent); 244 } 245 246 abstract class UsageGroup { 247 protected int mObserverId; 248 protected String[] mObserved; 249 protected long mTimeLimitMs; 250 protected long mUsageTimeMs; 251 protected int mActives; 252 protected long mLastKnownUsageTimeMs; 253 protected long mLastUsageEndTimeMs; 254 protected WeakReference<UserData> mUserRef; 255 protected WeakReference<ObserverAppData> mObserverAppRef; 256 protected PendingIntent mLimitReachedCallback; 257 UsageGroup(UserData user, ObserverAppData observerApp, int observerId, String[] observed, long timeLimitMs, PendingIntent limitReachedCallback)258 UsageGroup(UserData user, ObserverAppData observerApp, int observerId, String[] observed, 259 long timeLimitMs, PendingIntent limitReachedCallback) { 260 mUserRef = new WeakReference<>(user); 261 mObserverAppRef = new WeakReference<>(observerApp); 262 mObserverId = observerId; 263 mObserved = observed; 264 mTimeLimitMs = timeLimitMs; 265 mLimitReachedCallback = limitReachedCallback; 266 } 267 268 @GuardedBy("mLock") getTimeLimitMs()269 public long getTimeLimitMs() { return mTimeLimitMs; } 270 271 @GuardedBy("mLock") getUsageTimeMs()272 public long getUsageTimeMs() { return mUsageTimeMs; } 273 274 @GuardedBy("mLock") remove()275 public void remove() { 276 UserData user = mUserRef.get(); 277 if (user != null) { 278 user.removeUsageGroup(this); 279 } 280 // Clear the callback, so any racy inflight message will do nothing 281 mLimitReachedCallback = null; 282 } 283 284 @GuardedBy("mLock") noteUsageStart(long startTimeMs)285 void noteUsageStart(long startTimeMs) { 286 noteUsageStart(startTimeMs, startTimeMs); 287 } 288 289 @GuardedBy("mLock") noteUsageStart(long startTimeMs, long currentTimeMs)290 void noteUsageStart(long startTimeMs, long currentTimeMs) { 291 if (mActives++ == 0) { 292 // If last known usage ended after the start of this usage, there is overlap 293 // between the last usage session and this one. Avoid double counting by only 294 // counting from the end of the last session. This has a rare side effect that some 295 // usage will not be accounted for if the previous session started and stopped 296 // within this current usage. 297 startTimeMs = mLastUsageEndTimeMs > startTimeMs ? mLastUsageEndTimeMs : startTimeMs; 298 mLastKnownUsageTimeMs = startTimeMs; 299 final long timeRemaining = 300 mTimeLimitMs - mUsageTimeMs - currentTimeMs + startTimeMs; 301 if (timeRemaining > 0) { 302 if (DEBUG) { 303 Slog.d(TAG, "Posting timeout for " + mObserverId + " for " 304 + timeRemaining + "ms"); 305 } 306 postCheckTimeoutLocked(this, timeRemaining); 307 } 308 } else { 309 if (mActives > mObserved.length) { 310 // Try to get to a sane state and log the issue 311 mActives = mObserved.length; 312 final UserData user = mUserRef.get(); 313 if (user == null) return; 314 final Object[] array = user.currentlyActive.keySet().toArray(); 315 Slog.e(TAG, 316 "Too many noted usage starts! Observed entities: " + Arrays.toString( 317 mObserved) + " Active Entities: " + Arrays.toString(array)); 318 } 319 } 320 } 321 322 @GuardedBy("mLock") noteUsageStop(long stopTimeMs)323 void noteUsageStop(long stopTimeMs) { 324 if (--mActives == 0) { 325 final boolean limitNotCrossed = mUsageTimeMs < mTimeLimitMs; 326 mUsageTimeMs += stopTimeMs - mLastKnownUsageTimeMs; 327 328 mLastUsageEndTimeMs = stopTimeMs; 329 if (limitNotCrossed && mUsageTimeMs >= mTimeLimitMs) { 330 // Crossed the limit 331 if (DEBUG) Slog.d(TAG, "MTB informing group obs=" + mObserverId); 332 postInformLimitReachedListenerLocked(this); 333 } 334 cancelCheckTimeoutLocked(this); 335 } else { 336 if (mActives < 0) { 337 // Try to get to a sane state and log the issue 338 mActives = 0; 339 final UserData user = mUserRef.get(); 340 if (user == null) return; 341 final Object[] array = user.currentlyActive.keySet().toArray(); 342 Slog.e(TAG, 343 "Too many noted usage stops! Observed entities: " + Arrays.toString( 344 mObserved) + " Active Entities: " + Arrays.toString(array)); 345 } 346 } 347 } 348 349 @GuardedBy("mLock") checkTimeout(long currentTimeMs)350 void checkTimeout(long currentTimeMs) { 351 final UserData user = mUserRef.get(); 352 if (user == null) return; 353 354 long timeRemainingMs = mTimeLimitMs - mUsageTimeMs; 355 356 if (DEBUG) Slog.d(TAG, "checkTimeout timeRemaining=" + timeRemainingMs); 357 358 // Already reached the limit, no need to report again 359 if (timeRemainingMs <= 0) return; 360 361 if (DEBUG) { 362 Slog.d(TAG, "checkTimeout"); 363 } 364 365 // Double check that at least one entity in this group is currently active 366 if (user.isActive(mObserved)) { 367 if (DEBUG) { 368 Slog.d(TAG, "checkTimeout group is active"); 369 } 370 final long timeUsedMs = currentTimeMs - mLastKnownUsageTimeMs; 371 if (timeRemainingMs <= timeUsedMs) { 372 if (DEBUG) Slog.d(TAG, "checkTimeout : Time limit reached"); 373 // Hit the limit, set timeRemaining to zero to avoid checking again 374 mUsageTimeMs += timeUsedMs; 375 mLastKnownUsageTimeMs = currentTimeMs; 376 AppTimeLimitController.this.postInformLimitReachedListenerLocked(this); 377 } else { 378 if (DEBUG) Slog.d(TAG, "checkTimeout : Some more time remaining"); 379 AppTimeLimitController.this.postCheckTimeoutLocked(this, 380 timeRemainingMs - timeUsedMs); 381 } 382 } 383 } 384 385 @GuardedBy("mLock") onLimitReached()386 public void onLimitReached() { 387 UserData user = mUserRef.get(); 388 if (user == null) return; 389 if (mListener != null) { 390 mListener.onLimitReached(mObserverId, user.userId, mTimeLimitMs, mUsageTimeMs, 391 mLimitReachedCallback); 392 } 393 } 394 395 @GuardedBy("mLock") dump(PrintWriter pw)396 void dump(PrintWriter pw) { 397 pw.print(" Group id="); 398 pw.print(mObserverId); 399 pw.print(" timeLimit="); 400 pw.print(mTimeLimitMs); 401 pw.print(" used="); 402 pw.print(mUsageTimeMs); 403 pw.print(" lastKnownUsage="); 404 pw.print(mLastKnownUsageTimeMs); 405 pw.print(" mActives="); 406 pw.print(mActives); 407 pw.print(" observed="); 408 pw.print(Arrays.toString(mObserved)); 409 } 410 } 411 412 class AppUsageGroup extends UsageGroup { AppUsageGroup(UserData user, ObserverAppData observerApp, int observerId, String[] observed, long timeLimitMs, PendingIntent limitReachedCallback)413 public AppUsageGroup(UserData user, ObserverAppData observerApp, int observerId, 414 String[] observed, long timeLimitMs, PendingIntent limitReachedCallback) { 415 super(user, observerApp, observerId, observed, timeLimitMs, limitReachedCallback); 416 } 417 418 @Override 419 @GuardedBy("mLock") remove()420 public void remove() { 421 super.remove(); 422 ObserverAppData observerApp = mObserverAppRef.get(); 423 if (observerApp != null) { 424 observerApp.removeAppUsageGroup(mObserverId); 425 } 426 } 427 428 @Override 429 @GuardedBy("mLock") onLimitReached()430 public void onLimitReached() { 431 super.onLimitReached(); 432 // Unregister since the limit has been met and observer was informed. 433 remove(); 434 } 435 } 436 437 class SessionUsageGroup extends UsageGroup { 438 private long mNewSessionThresholdMs; 439 private PendingIntent mSessionEndCallback; 440 SessionUsageGroup(UserData user, ObserverAppData observerApp, int observerId, String[] observed, long timeLimitMs, PendingIntent limitReachedCallback, long newSessionThresholdMs, PendingIntent sessionEndCallback)441 public SessionUsageGroup(UserData user, ObserverAppData observerApp, int observerId, 442 String[] observed, long timeLimitMs, PendingIntent limitReachedCallback, 443 long newSessionThresholdMs, PendingIntent sessionEndCallback) { 444 super(user, observerApp, observerId, observed, timeLimitMs, limitReachedCallback); 445 this.mNewSessionThresholdMs = newSessionThresholdMs; 446 this.mSessionEndCallback = sessionEndCallback; 447 } 448 449 @Override 450 @GuardedBy("mLock") remove()451 public void remove() { 452 super.remove(); 453 ObserverAppData observerApp = mObserverAppRef.get(); 454 if (observerApp != null) { 455 observerApp.removeSessionUsageGroup(mObserverId); 456 } 457 // Clear the callback, so any racy inflight messages will do nothing 458 mSessionEndCallback = null; 459 } 460 461 @Override 462 @GuardedBy("mLock") noteUsageStart(long startTimeMs, long currentTimeMs)463 public void noteUsageStart(long startTimeMs, long currentTimeMs) { 464 if (mActives == 0) { 465 if (startTimeMs - mLastUsageEndTimeMs > mNewSessionThresholdMs) { 466 // New session has started, clear usage time. 467 mUsageTimeMs = 0; 468 } 469 AppTimeLimitController.this.cancelInformSessionEndListener(this); 470 } 471 super.noteUsageStart(startTimeMs, currentTimeMs); 472 } 473 474 @Override 475 @GuardedBy("mLock") noteUsageStop(long stopTimeMs)476 public void noteUsageStop(long stopTimeMs) { 477 super.noteUsageStop(stopTimeMs); 478 if (mActives == 0) { 479 if (mUsageTimeMs >= mTimeLimitMs) { 480 // Usage has ended. Schedule the session end callback to be triggered once 481 // the new session threshold has been reached 482 AppTimeLimitController.this.postInformSessionEndListenerLocked(this, 483 mNewSessionThresholdMs); 484 } 485 486 } 487 } 488 489 @GuardedBy("mLock") onSessionEnd()490 public void onSessionEnd() { 491 UserData user = mUserRef.get(); 492 if (user == null) return; 493 if (mListener != null) { 494 mListener.onSessionEnd(mObserverId, 495 user.userId, 496 mUsageTimeMs, 497 mSessionEndCallback); 498 } 499 } 500 501 @Override 502 @GuardedBy("mLock") dump(PrintWriter pw)503 void dump(PrintWriter pw) { 504 super.dump(pw); 505 pw.print(" lastUsageEndTime="); 506 pw.print(mLastUsageEndTimeMs); 507 pw.print(" newSessionThreshold="); 508 pw.print(mNewSessionThresholdMs); 509 } 510 } 511 512 class AppUsageLimitGroup extends UsageGroup { AppUsageLimitGroup(UserData user, ObserverAppData observerApp, int observerId, String[] observed, long timeLimitMs, long timeUsedMs, PendingIntent limitReachedCallback)513 public AppUsageLimitGroup(UserData user, ObserverAppData observerApp, int observerId, 514 String[] observed, long timeLimitMs, long timeUsedMs, 515 PendingIntent limitReachedCallback) { 516 super(user, observerApp, observerId, observed, timeLimitMs, limitReachedCallback); 517 mUsageTimeMs = timeUsedMs; 518 } 519 520 @Override 521 @GuardedBy("mLock") remove()522 public void remove() { 523 super.remove(); 524 ObserverAppData observerApp = mObserverAppRef.get(); 525 if (observerApp != null) { 526 observerApp.removeAppUsageLimitGroup(mObserverId); 527 } 528 } 529 530 @GuardedBy("mLock") getTotaUsageLimit()531 long getTotaUsageLimit() { 532 return mTimeLimitMs; 533 } 534 535 @GuardedBy("mLock") getUsageRemaining()536 long getUsageRemaining() { 537 // If there is currently an active session, account for its usage 538 if (mActives > 0) { 539 return mTimeLimitMs - mUsageTimeMs - (getUptimeMillis() - mLastKnownUsageTimeMs); 540 } else { 541 return mTimeLimitMs - mUsageTimeMs; 542 } 543 } 544 } 545 546 547 private class MyHandler extends Handler { 548 static final int MSG_CHECK_TIMEOUT = 1; 549 static final int MSG_INFORM_LIMIT_REACHED_LISTENER = 2; 550 static final int MSG_INFORM_SESSION_END = 3; 551 MyHandler(Looper looper)552 MyHandler(Looper looper) { 553 super(looper); 554 } 555 556 @Override handleMessage(Message msg)557 public void handleMessage(Message msg) { 558 switch (msg.what) { 559 case MSG_CHECK_TIMEOUT: 560 synchronized (mLock) { 561 ((UsageGroup) msg.obj).checkTimeout(getUptimeMillis()); 562 } 563 break; 564 case MSG_INFORM_LIMIT_REACHED_LISTENER: 565 synchronized (mLock) { 566 ((UsageGroup) msg.obj).onLimitReached(); 567 } 568 break; 569 case MSG_INFORM_SESSION_END: 570 synchronized (mLock) { 571 ((SessionUsageGroup) msg.obj).onSessionEnd(); 572 } 573 break; 574 default: 575 super.handleMessage(msg); 576 break; 577 } 578 } 579 } 580 AppTimeLimitController(TimeLimitCallbackListener listener, Looper looper)581 public AppTimeLimitController(TimeLimitCallbackListener listener, Looper looper) { 582 mHandler = new MyHandler(looper); 583 mListener = listener; 584 } 585 586 /** Overrideable by a test */ 587 @VisibleForTesting getUptimeMillis()588 protected long getUptimeMillis() { 589 return SystemClock.uptimeMillis(); 590 } 591 592 /** Overrideable for testing purposes */ 593 @VisibleForTesting getAppUsageObserverPerUidLimit()594 protected long getAppUsageObserverPerUidLimit() { 595 return MAX_OBSERVER_PER_UID; 596 } 597 598 /** Overrideable for testing purposes */ 599 @VisibleForTesting getUsageSessionObserverPerUidLimit()600 protected long getUsageSessionObserverPerUidLimit() { 601 return MAX_OBSERVER_PER_UID; 602 } 603 604 /** Overrideable for testing purposes */ 605 @VisibleForTesting getAppUsageLimitObserverPerUidLimit()606 protected long getAppUsageLimitObserverPerUidLimit() { 607 return MAX_OBSERVER_PER_UID; 608 } 609 610 /** Overrideable for testing purposes */ 611 @VisibleForTesting getMinTimeLimit()612 protected long getMinTimeLimit() { 613 return ONE_MINUTE; 614 } 615 616 @VisibleForTesting getAppUsageGroup(int observerAppUid, int observerId)617 AppUsageGroup getAppUsageGroup(int observerAppUid, int observerId) { 618 synchronized (mLock) { 619 return getOrCreateObserverAppDataLocked(observerAppUid).appUsageGroups.get(observerId); 620 } 621 } 622 623 @VisibleForTesting getSessionUsageGroup(int observerAppUid, int observerId)624 SessionUsageGroup getSessionUsageGroup(int observerAppUid, int observerId) { 625 synchronized (mLock) { 626 return getOrCreateObserverAppDataLocked(observerAppUid).sessionUsageGroups.get( 627 observerId); 628 } 629 } 630 631 @VisibleForTesting getAppUsageLimitGroup(int observerAppUid, int observerId)632 AppUsageLimitGroup getAppUsageLimitGroup(int observerAppUid, int observerId) { 633 synchronized (mLock) { 634 return getOrCreateObserverAppDataLocked(observerAppUid).appUsageLimitGroups.get( 635 observerId); 636 } 637 } 638 639 /** 640 * Returns an object describing the app usage limit for the given package which was set via 641 * {@link #addAppUsageLimitObserver). 642 * If there are multiple limits that apply to the package, the one with the smallest 643 * time remaining will be returned. 644 */ getAppUsageLimit( String packageName, UserHandle user)645 public UsageStatsManagerInternal.AppUsageLimitData getAppUsageLimit( 646 String packageName, UserHandle user) { 647 synchronized (mLock) { 648 final UserData userData = getOrCreateUserDataLocked(user.getIdentifier()); 649 if (userData == null) { 650 return null; 651 } 652 653 final ArrayList<UsageGroup> usageGroups = userData.observedMap.get(packageName); 654 if (usageGroups == null || usageGroups.isEmpty()) { 655 return null; 656 } 657 658 final ArraySet<AppUsageLimitGroup> usageLimitGroups = new ArraySet<>(); 659 for (int i = 0; i < usageGroups.size(); i++) { 660 if (usageGroups.get(i) instanceof AppUsageLimitGroup) { 661 final AppUsageLimitGroup group = (AppUsageLimitGroup) usageGroups.get(i); 662 for (int j = 0; j < group.mObserved.length; j++) { 663 if (group.mObserved[j].equals(packageName)) { 664 usageLimitGroups.add(group); 665 break; 666 } 667 } 668 } 669 } 670 if (usageLimitGroups.isEmpty()) { 671 return null; 672 } 673 674 AppUsageLimitGroup smallestGroup = usageLimitGroups.valueAt(0); 675 for (int i = 1; i < usageLimitGroups.size(); i++) { 676 final AppUsageLimitGroup otherGroup = usageLimitGroups.valueAt(i); 677 if (otherGroup.getUsageRemaining() < smallestGroup.getUsageRemaining()) { 678 smallestGroup = otherGroup; 679 } 680 } 681 return new UsageStatsManagerInternal.AppUsageLimitData( 682 smallestGroup.getTotaUsageLimit(), smallestGroup.getUsageRemaining()); 683 } 684 } 685 686 /** Returns an existing UserData object for the given userId, or creates one */ 687 @GuardedBy("mLock") getOrCreateUserDataLocked(int userId)688 private UserData getOrCreateUserDataLocked(int userId) { 689 UserData userData = mUsers.get(userId); 690 if (userData == null) { 691 userData = new UserData(userId); 692 mUsers.put(userId, userData); 693 } 694 return userData; 695 } 696 697 /** Returns an existing ObserverAppData object for the given uid, or creates one */ 698 @GuardedBy("mLock") getOrCreateObserverAppDataLocked(int uid)699 private ObserverAppData getOrCreateObserverAppDataLocked(int uid) { 700 ObserverAppData appData = mObserverApps.get(uid); 701 if (appData == null) { 702 appData = new ObserverAppData(uid); 703 mObserverApps.put(uid, appData); 704 } 705 return appData; 706 } 707 708 /** Clean up data if user is removed */ onUserRemoved(int userId)709 public void onUserRemoved(int userId) { 710 synchronized (mLock) { 711 // TODO: Remove any inflight delayed messages 712 mUsers.remove(userId); 713 } 714 } 715 716 /** 717 * Check if group has any currently active entities. 718 */ 719 @GuardedBy("mLock") noteActiveLocked(UserData user, UsageGroup group, long currentTimeMs)720 private void noteActiveLocked(UserData user, UsageGroup group, long currentTimeMs) { 721 // TODO: Consider using a bloom filter here if number of actives becomes large 722 final int size = group.mObserved.length; 723 for (int i = 0; i < size; i++) { 724 if (user.currentlyActive.containsKey(group.mObserved[i])) { 725 // Entity is currently active. Start group's usage. 726 group.noteUsageStart(currentTimeMs); 727 } 728 } 729 } 730 731 /** 732 * Registers an app usage observer with the given details. 733 * Existing app usage observer with the same observerId will be removed. 734 */ addAppUsageObserver(int requestingUid, int observerId, String[] observed, long timeLimit, PendingIntent callbackIntent, @UserIdInt int userId)735 public void addAppUsageObserver(int requestingUid, int observerId, String[] observed, 736 long timeLimit, PendingIntent callbackIntent, @UserIdInt int userId) { 737 if (timeLimit < getMinTimeLimit()) { 738 throw new IllegalArgumentException("Time limit must be >= " + getMinTimeLimit()); 739 } 740 synchronized (mLock) { 741 UserData user = getOrCreateUserDataLocked(userId); 742 ObserverAppData observerApp = getOrCreateObserverAppDataLocked(requestingUid); 743 AppUsageGroup group = observerApp.appUsageGroups.get(observerId); 744 if (group != null) { 745 // Remove previous app usage group associated with observerId 746 group.remove(); 747 } 748 749 final int observerIdCount = observerApp.appUsageGroups.size(); 750 if (observerIdCount >= getAppUsageObserverPerUidLimit()) { 751 throw new IllegalStateException( 752 "Too many app usage observers added by uid " + requestingUid); 753 } 754 group = new AppUsageGroup(user, observerApp, observerId, observed, timeLimit, 755 callbackIntent); 756 observerApp.appUsageGroups.append(observerId, group); 757 758 if (DEBUG) { 759 Slog.d(TAG, "addObserver " + observed + " for " + timeLimit); 760 } 761 762 user.addUsageGroup(group); 763 noteActiveLocked(user, group, getUptimeMillis()); 764 } 765 } 766 767 /** 768 * Remove a registered observer by observerId and calling uid. 769 * 770 * @param requestingUid The calling uid 771 * @param observerId The unique observer id for this user 772 * @param userId The user id of the observer 773 */ removeAppUsageObserver(int requestingUid, int observerId, @UserIdInt int userId)774 public void removeAppUsageObserver(int requestingUid, int observerId, @UserIdInt int userId) { 775 synchronized (mLock) { 776 final ObserverAppData observerApp = getOrCreateObserverAppDataLocked(requestingUid); 777 final AppUsageGroup group = observerApp.appUsageGroups.get(observerId); 778 if (group != null) { 779 // Remove previous app usage group associated with observerId 780 group.remove(); 781 } 782 } 783 } 784 785 786 /** 787 * Registers a usage session observer with the given details. 788 * Existing usage session observer with the same observerId will be removed. 789 */ addUsageSessionObserver(int requestingUid, int observerId, String[] observed, long timeLimit, long sessionThresholdTime, PendingIntent limitReachedCallbackIntent, PendingIntent sessionEndCallbackIntent, @UserIdInt int userId)790 public void addUsageSessionObserver(int requestingUid, int observerId, String[] observed, 791 long timeLimit, long sessionThresholdTime, 792 PendingIntent limitReachedCallbackIntent, PendingIntent sessionEndCallbackIntent, 793 @UserIdInt int userId) { 794 if (timeLimit < getMinTimeLimit()) { 795 throw new IllegalArgumentException("Time limit must be >= " + getMinTimeLimit()); 796 } 797 synchronized (mLock) { 798 UserData user = getOrCreateUserDataLocked(userId); 799 ObserverAppData observerApp = getOrCreateObserverAppDataLocked(requestingUid); 800 SessionUsageGroup group = observerApp.sessionUsageGroups.get(observerId); 801 if (group != null) { 802 // Remove previous session usage group associated with observerId 803 group.remove(); 804 } 805 806 final int observerIdCount = observerApp.sessionUsageGroups.size(); 807 if (observerIdCount >= getUsageSessionObserverPerUidLimit()) { 808 throw new IllegalStateException( 809 "Too many app usage observers added by uid " + requestingUid); 810 } 811 group = new SessionUsageGroup(user, observerApp, observerId, observed, timeLimit, 812 limitReachedCallbackIntent, sessionThresholdTime, sessionEndCallbackIntent); 813 observerApp.sessionUsageGroups.append(observerId, group); 814 815 user.addUsageGroup(group); 816 noteActiveLocked(user, group, getUptimeMillis()); 817 } 818 } 819 820 /** 821 * Remove a registered observer by observerId and calling uid. 822 * 823 * @param requestingUid The calling uid 824 * @param observerId The unique observer id for this user 825 * @param userId The user id of the observer 826 */ removeUsageSessionObserver(int requestingUid, int observerId, @UserIdInt int userId)827 public void removeUsageSessionObserver(int requestingUid, int observerId, 828 @UserIdInt int userId) { 829 synchronized (mLock) { 830 final ObserverAppData observerApp = getOrCreateObserverAppDataLocked(requestingUid); 831 final SessionUsageGroup group = observerApp.sessionUsageGroups.get(observerId); 832 if (group != null) { 833 // Remove previous app usage group associated with observerId 834 group.remove(); 835 } 836 } 837 } 838 839 /** 840 * Registers an app usage limit observer with the given details. 841 * Existing app usage limit observer with the same observerId will be removed. 842 */ addAppUsageLimitObserver(int requestingUid, int observerId, String[] observed, long timeLimit, long timeUsed, PendingIntent callbackIntent, @UserIdInt int userId)843 public void addAppUsageLimitObserver(int requestingUid, int observerId, String[] observed, 844 long timeLimit, long timeUsed, PendingIntent callbackIntent, 845 @UserIdInt int userId) { 846 if (timeLimit < getMinTimeLimit()) { 847 throw new IllegalArgumentException("Time limit must be >= " + getMinTimeLimit()); 848 } 849 synchronized (mLock) { 850 UserData user = getOrCreateUserDataLocked(userId); 851 ObserverAppData observerApp = getOrCreateObserverAppDataLocked(requestingUid); 852 AppUsageLimitGroup group = observerApp.appUsageLimitGroups.get(observerId); 853 if (group != null) { 854 // Remove previous app usage group associated with observerId 855 group.remove(); 856 } 857 858 final int observerIdCount = observerApp.appUsageLimitGroups.size(); 859 if (observerIdCount >= getAppUsageLimitObserverPerUidLimit()) { 860 throw new IllegalStateException( 861 "Too many app usage observers added by uid " + requestingUid); 862 } 863 group = new AppUsageLimitGroup(user, observerApp, observerId, observed, timeLimit, 864 timeUsed, timeUsed >= timeLimit ? null : callbackIntent); 865 observerApp.appUsageLimitGroups.append(observerId, group); 866 867 if (DEBUG) { 868 Slog.d(TAG, "addObserver " + observed + " for " + timeLimit); 869 } 870 871 user.addUsageGroup(group); 872 noteActiveLocked(user, group, getUptimeMillis()); 873 } 874 } 875 876 /** 877 * Remove a registered observer by observerId and calling uid. 878 * 879 * @param requestingUid The calling uid 880 * @param observerId The unique observer id for this user 881 * @param userId The user id of the observer 882 */ removeAppUsageLimitObserver(int requestingUid, int observerId, @UserIdInt int userId)883 public void removeAppUsageLimitObserver(int requestingUid, int observerId, 884 @UserIdInt int userId) { 885 synchronized (mLock) { 886 final ObserverAppData observerApp = getOrCreateObserverAppDataLocked(requestingUid); 887 final AppUsageLimitGroup group = observerApp.appUsageLimitGroups.get(observerId); 888 if (group != null) { 889 // Remove previous app usage group associated with observerId 890 group.remove(); 891 } 892 } 893 } 894 895 /** 896 * Called when an entity becomes active. 897 * 898 * @param name The entity that became active 899 * @param userId The user 900 * @param timeAgoMs Time since usage was started 901 */ noteUsageStart(String name, int userId, long timeAgoMs)902 public void noteUsageStart(String name, int userId, long timeAgoMs) 903 throws IllegalArgumentException { 904 synchronized (mLock) { 905 UserData user = getOrCreateUserDataLocked(userId); 906 if (DEBUG) Slog.d(TAG, "Usage entity " + name + " became active"); 907 908 final int index = user.currentlyActive.indexOfKey(name); 909 if (index >= 0) { 910 final Integer count = user.currentlyActive.valueAt(index); 911 if (count != null) { 912 // There are multiple instances of this entity. Just increment the count. 913 user.currentlyActive.setValueAt(index, count + 1); 914 return; 915 } 916 } 917 final long currentTime = getUptimeMillis(); 918 919 user.currentlyActive.put(name, ONE); 920 921 ArrayList<UsageGroup> groups = user.observedMap.get(name); 922 if (groups == null) return; 923 924 final int size = groups.size(); 925 for (int i = 0; i < size; i++) { 926 UsageGroup group = groups.get(i); 927 group.noteUsageStart(currentTime - timeAgoMs, currentTime); 928 } 929 } 930 } 931 932 /** 933 * Called when an entity becomes active. 934 * 935 * @param name The entity that became active 936 * @param userId The user 937 */ noteUsageStart(String name, int userId)938 public void noteUsageStart(String name, int userId) throws IllegalArgumentException { 939 noteUsageStart(name, userId, 0); 940 } 941 942 /** 943 * Called when an entity becomes inactive. 944 * 945 * @param name The entity that became inactive 946 * @param userId The user 947 */ noteUsageStop(String name, int userId)948 public void noteUsageStop(String name, int userId) throws IllegalArgumentException { 949 synchronized (mLock) { 950 UserData user = getOrCreateUserDataLocked(userId); 951 if (DEBUG) Slog.d(TAG, "Usage entity " + name + " became inactive"); 952 953 final int index = user.currentlyActive.indexOfKey(name); 954 if (index < 0) { 955 throw new IllegalArgumentException( 956 "Unable to stop usage for " + name + ", not in use"); 957 } 958 959 final Integer count = user.currentlyActive.valueAt(index); 960 if (!count.equals(ONE)) { 961 // There are multiple instances of this entity. Just decrement the count. 962 user.currentlyActive.setValueAt(index, count - 1); 963 return; 964 } 965 966 user.currentlyActive.removeAt(index); 967 final long currentTime = getUptimeMillis(); 968 969 // Check if any of the groups need to watch for this entity 970 ArrayList<UsageGroup> groups = user.observedMap.get(name); 971 if (groups == null) return; 972 973 final int size = groups.size(); 974 for (int i = 0; i < size; i++) { 975 UsageGroup group = groups.get(i); 976 group.noteUsageStop(currentTime); 977 } 978 979 } 980 } 981 982 @GuardedBy("mLock") postInformLimitReachedListenerLocked(UsageGroup group)983 private void postInformLimitReachedListenerLocked(UsageGroup group) { 984 mHandler.sendMessage(mHandler.obtainMessage(MyHandler.MSG_INFORM_LIMIT_REACHED_LISTENER, 985 group)); 986 } 987 988 @GuardedBy("mLock") postInformSessionEndListenerLocked(SessionUsageGroup group, long timeout)989 private void postInformSessionEndListenerLocked(SessionUsageGroup group, long timeout) { 990 mHandler.sendMessageDelayed( 991 mHandler.obtainMessage(MyHandler.MSG_INFORM_SESSION_END, group), 992 timeout); 993 } 994 995 @GuardedBy("mLock") cancelInformSessionEndListener(SessionUsageGroup group)996 private void cancelInformSessionEndListener(SessionUsageGroup group) { 997 mHandler.removeMessages(MyHandler.MSG_INFORM_SESSION_END, group); 998 } 999 1000 @GuardedBy("mLock") postCheckTimeoutLocked(UsageGroup group, long timeout)1001 private void postCheckTimeoutLocked(UsageGroup group, long timeout) { 1002 mHandler.sendMessageDelayed(mHandler.obtainMessage(MyHandler.MSG_CHECK_TIMEOUT, group), 1003 timeout); 1004 } 1005 1006 @GuardedBy("mLock") cancelCheckTimeoutLocked(UsageGroup group)1007 private void cancelCheckTimeoutLocked(UsageGroup group) { 1008 mHandler.removeMessages(MyHandler.MSG_CHECK_TIMEOUT, group); 1009 } 1010 dump(String[] args, PrintWriter pw)1011 void dump(String[] args, PrintWriter pw) { 1012 if (args != null) { 1013 for (int i = 0; i < args.length; i++) { 1014 String arg = args[i]; 1015 if ("actives".equals(arg)) { 1016 synchronized (mLock) { 1017 final int nUsers = mUsers.size(); 1018 for (int user = 0; user < nUsers; user++) { 1019 final ArrayMap<String, Integer> actives = 1020 mUsers.valueAt(user).currentlyActive; 1021 final int nActive = actives.size(); 1022 for (int active = 0; active < nActive; active++) { 1023 pw.println(actives.keyAt(active)); 1024 } 1025 } 1026 } 1027 return; 1028 } 1029 } 1030 } 1031 1032 synchronized (mLock) { 1033 pw.println("\n App Time Limits"); 1034 final int nUsers = mUsers.size(); 1035 for (int i = 0; i < nUsers; i++) { 1036 pw.print(" User "); 1037 mUsers.valueAt(i).dump(pw); 1038 } 1039 pw.println(); 1040 final int nObserverApps = mObserverApps.size(); 1041 for (int i = 0; i < nObserverApps; i++) { 1042 pw.print(" Observer App "); 1043 mObserverApps.valueAt(i).dump(pw); 1044 } 1045 } 1046 } 1047 } 1048