1 /*
2  * Copyright (C) 2018 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 
17 package com.android.server.job.controllers;
18 
19 import static android.text.format.DateUtils.HOUR_IN_MILLIS;
20 import static android.text.format.DateUtils.MINUTE_IN_MILLIS;
21 import static android.text.format.DateUtils.SECOND_IN_MILLIS;
22 
23 import static com.android.server.job.JobSchedulerService.ACTIVE_INDEX;
24 import static com.android.server.job.JobSchedulerService.FREQUENT_INDEX;
25 import static com.android.server.job.JobSchedulerService.NEVER_INDEX;
26 import static com.android.server.job.JobSchedulerService.RARE_INDEX;
27 import static com.android.server.job.JobSchedulerService.WORKING_INDEX;
28 import static com.android.server.job.JobSchedulerService.sElapsedRealtimeClock;
29 
30 import android.annotation.NonNull;
31 import android.annotation.Nullable;
32 import android.annotation.UserIdInt;
33 import android.app.ActivityManager;
34 import android.app.ActivityManagerInternal;
35 import android.app.AlarmManager;
36 import android.app.AppGlobals;
37 import android.app.IUidObserver;
38 import android.app.usage.UsageStatsManagerInternal;
39 import android.app.usage.UsageStatsManagerInternal.AppIdleStateChangeListener;
40 import android.content.BroadcastReceiver;
41 import android.content.ContentResolver;
42 import android.content.Context;
43 import android.content.Intent;
44 import android.content.IntentFilter;
45 import android.database.ContentObserver;
46 import android.net.Uri;
47 import android.os.BatteryManager;
48 import android.os.BatteryManagerInternal;
49 import android.os.Handler;
50 import android.os.Looper;
51 import android.os.Message;
52 import android.os.RemoteException;
53 import android.os.UserHandle;
54 import android.provider.Settings;
55 import android.util.ArrayMap;
56 import android.util.ArraySet;
57 import android.util.KeyValueListParser;
58 import android.util.Log;
59 import android.util.Slog;
60 import android.util.SparseArray;
61 import android.util.SparseBooleanArray;
62 import android.util.SparseSetArray;
63 import android.util.proto.ProtoOutputStream;
64 
65 import com.android.internal.annotations.VisibleForTesting;
66 import com.android.internal.os.BackgroundThread;
67 import com.android.internal.util.IndentingPrintWriter;
68 import com.android.server.LocalServices;
69 import com.android.server.job.ConstantsProto;
70 import com.android.server.job.JobSchedulerService;
71 import com.android.server.job.StateControllerProto;
72 
73 import java.util.ArrayList;
74 import java.util.List;
75 import java.util.Objects;
76 import java.util.function.Consumer;
77 import java.util.function.Predicate;
78 
79 /**
80  * Controller that tracks whether an app has exceeded its standby bucket quota.
81  *
82  * With initial defaults, each app in each bucket is given 10 minutes to run within its respective
83  * time window. Active jobs can run indefinitely, working set jobs can run for 10 minutes within a
84  * 2 hour window, frequent jobs get to run 10 minutes in an 8 hour window, and rare jobs get to run
85  * 10 minutes in a 24 hour window. The windows are rolling, so as soon as a job would have some
86  * quota based on its bucket, it will be eligible to run. When a job's bucket changes, its new
87  * quota is immediately applied to it.
88  *
89  * Job and session count limits are included to prevent abuse/spam. Each bucket has its own limit on
90  * the number of jobs or sessions that can run within the window. Regardless of bucket, apps will
91  * not be allowed to run more than 20 jobs within the past 10 minutes.
92  *
93  * Jobs are throttled while an app is not in a foreground state. All jobs are allowed to run
94  * freely when an app enters the foreground state and are restricted when the app leaves the
95  * foreground state. However, jobs that are started while the app is in the TOP state do not count
96  * towards any quota and are not restricted regardless of the app's state change.
97  *
98  * Jobs will not be throttled when the device is charging. The device is considered to be charging
99  * once the {@link BatteryManager#ACTION_CHARGING} intent has been broadcast.
100  *
101  * Note: all limits are enforced per bucket window unless explicitly stated otherwise.
102  * All stated values are configurable and subject to change. See {@link QcConstants} for current
103  * defaults.
104  *
105  * Test: atest com.android.server.job.controllers.QuotaControllerTest
106  */
107 public final class QuotaController extends StateController {
108     private static final String TAG = "JobScheduler.Quota";
109     private static final boolean DEBUG = JobSchedulerService.DEBUG
110             || Log.isLoggable(TAG, Log.DEBUG);
111 
112     private static final String ALARM_TAG_CLEANUP = "*job.cleanup*";
113     private static final String ALARM_TAG_QUOTA_CHECK = "*job.quota_check*";
114 
115     /**
116      * A sparse array of ArrayMaps, which is suitable for holding (userId, packageName)->object
117      * associations.
118      */
119     private static class UserPackageMap<T> {
120         private final SparseArray<ArrayMap<String, T>> mData = new SparseArray<>();
121 
add(int userId, @NonNull String packageName, @Nullable T obj)122         public void add(int userId, @NonNull String packageName, @Nullable T obj) {
123             ArrayMap<String, T> data = mData.get(userId);
124             if (data == null) {
125                 data = new ArrayMap<String, T>();
126                 mData.put(userId, data);
127             }
128             data.put(packageName, obj);
129         }
130 
clear()131         public void clear() {
132             for (int i = 0; i < mData.size(); ++i) {
133                 mData.valueAt(i).clear();
134             }
135         }
136 
137         /** Removes all the data for the user, if there was any. */
delete(int userId)138         public void delete(int userId) {
139             mData.delete(userId);
140         }
141 
142         /** Removes the data for the user and package, if there was any. */
delete(int userId, @NonNull String packageName)143         public void delete(int userId, @NonNull String packageName) {
144             ArrayMap<String, T> data = mData.get(userId);
145             if (data != null) {
146                 data.remove(packageName);
147             }
148         }
149 
150         @Nullable
get(int userId, @NonNull String packageName)151         public T get(int userId, @NonNull String packageName) {
152             ArrayMap<String, T> data = mData.get(userId);
153             if (data != null) {
154                 return data.get(packageName);
155             }
156             return null;
157         }
158 
159         /** @see SparseArray#indexOfKey */
indexOfKey(int userId)160         public int indexOfKey(int userId) {
161             return mData.indexOfKey(userId);
162         }
163 
164         /** Returns the userId at the given index. */
keyAt(int index)165         public int keyAt(int index) {
166             return mData.keyAt(index);
167         }
168 
169         /** Returns the package name at the given index. */
170         @NonNull
keyAt(int userIndex, int packageIndex)171         public String keyAt(int userIndex, int packageIndex) {
172             return mData.valueAt(userIndex).keyAt(packageIndex);
173         }
174 
175         /** Returns the size of the outer (userId) array. */
numUsers()176         public int numUsers() {
177             return mData.size();
178         }
179 
numPackagesForUser(int userId)180         public int numPackagesForUser(int userId) {
181             ArrayMap<String, T> data = mData.get(userId);
182             return data == null ? 0 : data.size();
183         }
184 
185         /** Returns the value T at the given user and index. */
186         @Nullable
valueAt(int userIndex, int packageIndex)187         public T valueAt(int userIndex, int packageIndex) {
188             return mData.valueAt(userIndex).valueAt(packageIndex);
189         }
190 
forEach(Consumer<T> consumer)191         public void forEach(Consumer<T> consumer) {
192             for (int i = numUsers() - 1; i >= 0; --i) {
193                 ArrayMap<String, T> data = mData.valueAt(i);
194                 for (int j = data.size() - 1; j >= 0; --j) {
195                     consumer.accept(data.valueAt(j));
196                 }
197             }
198         }
199     }
200 
201     /**
202      * Standardize the output of userId-packageName combo.
203      */
string(int userId, String packageName)204     private static String string(int userId, String packageName) {
205         return "<" + userId + ">" + packageName;
206     }
207 
208     private static final class Package {
209         public final String packageName;
210         public final int userId;
211 
Package(int userId, String packageName)212         Package(int userId, String packageName) {
213             this.userId = userId;
214             this.packageName = packageName;
215         }
216 
217         @Override
toString()218         public String toString() {
219             return string(userId, packageName);
220         }
221 
writeToProto(ProtoOutputStream proto, long fieldId)222         public void writeToProto(ProtoOutputStream proto, long fieldId) {
223             final long token = proto.start(fieldId);
224 
225             proto.write(StateControllerProto.QuotaController.Package.USER_ID, userId);
226             proto.write(StateControllerProto.QuotaController.Package.NAME, packageName);
227 
228             proto.end(token);
229         }
230 
231         @Override
equals(Object obj)232         public boolean equals(Object obj) {
233             if (obj instanceof Package) {
234                 Package other = (Package) obj;
235                 return userId == other.userId && Objects.equals(packageName, other.packageName);
236             } else {
237                 return false;
238             }
239         }
240 
241         @Override
hashCode()242         public int hashCode() {
243             return packageName.hashCode() + userId;
244         }
245     }
246 
hashLong(long val)247     private static int hashLong(long val) {
248         return (int) (val ^ (val >>> 32));
249     }
250 
251     @VisibleForTesting
252     static class ExecutionStats {
253         /**
254          * The time after which this record should be considered invalid (out of date), in the
255          * elapsed realtime timebase.
256          */
257         public long expirationTimeElapsed;
258 
259         public long windowSizeMs;
260         public int jobCountLimit;
261         public int sessionCountLimit;
262 
263         /** The total amount of time the app ran in its respective bucket window size. */
264         public long executionTimeInWindowMs;
265         public int bgJobCountInWindow;
266 
267         /** The total amount of time the app ran in the last {@link #MAX_PERIOD_MS}. */
268         public long executionTimeInMaxPeriodMs;
269         public int bgJobCountInMaxPeriod;
270 
271         /**
272          * The number of {@link TimingSession}s within the bucket window size. This will include
273          * sessions that started before the window as long as they end within the window.
274          */
275         public int sessionCountInWindow;
276 
277         /**
278          * The time after which the app will be under the bucket quota and can start running jobs
279          * again. This is only valid if
280          * {@link #executionTimeInWindowMs} >= {@link #mAllowedTimePerPeriodMs},
281          * {@link #executionTimeInMaxPeriodMs} >= {@link #mMaxExecutionTimeMs},
282          * {@link #bgJobCountInWindow} >= {@link #jobCountLimit}, or
283          * {@link #sessionCountInWindow} >= {@link #sessionCountLimit}.
284          */
285         public long inQuotaTimeElapsed;
286 
287         /**
288          * The time after which {@link #jobCountInRateLimitingWindow} should be considered invalid,
289          * in the elapsed realtime timebase.
290          */
291         public long jobRateLimitExpirationTimeElapsed;
292 
293         /**
294          * The number of jobs that ran in at least the last {@link #mRateLimitingWindowMs}.
295          * It may contain a few stale entries since cleanup won't happen exactly every
296          * {@link #mRateLimitingWindowMs}.
297          */
298         public int jobCountInRateLimitingWindow;
299 
300         /**
301          * The time after which {@link #sessionCountInRateLimitingWindow} should be considered
302          * invalid, in the elapsed realtime timebase.
303          */
304         public long sessionRateLimitExpirationTimeElapsed;
305 
306         /**
307          * The number of {@link TimingSession}s that ran in at least the last
308          * {@link #mRateLimitingWindowMs}. It may contain a few stale entries since cleanup won't
309          * happen exactly every {@link #mRateLimitingWindowMs}. This should only be considered
310          * valid before elapsed realtime has reached {@link #sessionRateLimitExpirationTimeElapsed}.
311          */
312         public int sessionCountInRateLimitingWindow;
313 
314         @Override
toString()315         public String toString() {
316             return "expirationTime=" + expirationTimeElapsed + ", "
317                     + "windowSizeMs=" + windowSizeMs + ", "
318                     + "jobCountLimit=" + jobCountLimit + ", "
319                     + "sessionCountLimit=" + sessionCountLimit + ", "
320                     + "executionTimeInWindow=" + executionTimeInWindowMs + ", "
321                     + "bgJobCountInWindow=" + bgJobCountInWindow + ", "
322                     + "executionTimeInMaxPeriod=" + executionTimeInMaxPeriodMs + ", "
323                     + "bgJobCountInMaxPeriod=" + bgJobCountInMaxPeriod + ", "
324                     + "sessionCountInWindow=" + sessionCountInWindow + ", "
325                     + "inQuotaTime=" + inQuotaTimeElapsed + ", "
326                     + "jobCountExpirationTime=" + jobRateLimitExpirationTimeElapsed + ", "
327                     + "jobCountInRateLimitingWindow=" + jobCountInRateLimitingWindow + ", "
328                     + "sessionCountExpirationTime=" + sessionRateLimitExpirationTimeElapsed + ", "
329                     + "sessionCountInRateLimitingWindow=" + sessionCountInRateLimitingWindow;
330         }
331 
332         @Override
equals(Object obj)333         public boolean equals(Object obj) {
334             if (obj instanceof ExecutionStats) {
335                 ExecutionStats other = (ExecutionStats) obj;
336                 return this.expirationTimeElapsed == other.expirationTimeElapsed
337                         && this.windowSizeMs == other.windowSizeMs
338                         && this.jobCountLimit == other.jobCountLimit
339                         && this.sessionCountLimit == other.sessionCountLimit
340                         && this.executionTimeInWindowMs == other.executionTimeInWindowMs
341                         && this.bgJobCountInWindow == other.bgJobCountInWindow
342                         && this.executionTimeInMaxPeriodMs == other.executionTimeInMaxPeriodMs
343                         && this.sessionCountInWindow == other.sessionCountInWindow
344                         && this.bgJobCountInMaxPeriod == other.bgJobCountInMaxPeriod
345                         && this.inQuotaTimeElapsed == other.inQuotaTimeElapsed
346                         && this.jobRateLimitExpirationTimeElapsed
347                                 == other.jobRateLimitExpirationTimeElapsed
348                         && this.jobCountInRateLimitingWindow == other.jobCountInRateLimitingWindow
349                         && this.sessionRateLimitExpirationTimeElapsed
350                                 == other.sessionRateLimitExpirationTimeElapsed
351                         && this.sessionCountInRateLimitingWindow
352                                 == other.sessionCountInRateLimitingWindow;
353             } else {
354                 return false;
355             }
356         }
357 
358         @Override
hashCode()359         public int hashCode() {
360             int result = 0;
361             result = 31 * result + hashLong(expirationTimeElapsed);
362             result = 31 * result + hashLong(windowSizeMs);
363             result = 31 * result + hashLong(jobCountLimit);
364             result = 31 * result + hashLong(sessionCountLimit);
365             result = 31 * result + hashLong(executionTimeInWindowMs);
366             result = 31 * result + bgJobCountInWindow;
367             result = 31 * result + hashLong(executionTimeInMaxPeriodMs);
368             result = 31 * result + bgJobCountInMaxPeriod;
369             result = 31 * result + sessionCountInWindow;
370             result = 31 * result + hashLong(inQuotaTimeElapsed);
371             result = 31 * result + hashLong(jobRateLimitExpirationTimeElapsed);
372             result = 31 * result + jobCountInRateLimitingWindow;
373             result = 31 * result + hashLong(sessionRateLimitExpirationTimeElapsed);
374             result = 31 * result + sessionCountInRateLimitingWindow;
375             return result;
376         }
377     }
378 
379     /** List of all tracked jobs keyed by source package-userId combo. */
380     private final UserPackageMap<ArraySet<JobStatus>> mTrackedJobs = new UserPackageMap<>();
381 
382     /** Timer for each package-userId combo. */
383     private final UserPackageMap<Timer> mPkgTimers = new UserPackageMap<>();
384 
385     /** List of all timing sessions for a package-userId combo, in chronological order. */
386     private final UserPackageMap<List<TimingSession>> mTimingSessions = new UserPackageMap<>();
387 
388     /**
389      * List of alarm listeners for each package that listen for when each package comes back within
390      * quota.
391      */
392     private final UserPackageMap<QcAlarmListener> mInQuotaAlarmListeners = new UserPackageMap<>();
393 
394     /** Cached calculation results for each app, with the standby buckets as the array indices. */
395     private final UserPackageMap<ExecutionStats[]> mExecutionStatsCache = new UserPackageMap<>();
396 
397     /** List of UIDs currently in the foreground. */
398     private final SparseBooleanArray mForegroundUids = new SparseBooleanArray();
399 
400     /** Cached mapping of UIDs (for all users) to a list of packages in the UID. */
401     private final SparseSetArray<String> mUidToPackageCache = new SparseSetArray<>();
402 
403     /**
404      * List of jobs that started while the UID was in the TOP state. There will be no more than
405      * 16 ({@link JobSchedulerService#MAX_JOB_CONTEXTS_COUNT}) running at once, so an ArraySet is
406      * fine.
407      */
408     private final ArraySet<JobStatus> mTopStartedJobs = new ArraySet<>();
409 
410     private final ActivityManagerInternal mActivityManagerInternal;
411     private final AlarmManager mAlarmManager;
412     private final ChargingTracker mChargeTracker;
413     private final Handler mHandler;
414     private final QcConstants mQcConstants;
415 
416     private volatile boolean mInParole;
417 
418     /**
419      * If the QuotaController should throttle apps based on their standby bucket and job activity.
420      * If false, all jobs will have their CONSTRAINT_WITHIN_QUOTA bit set to true immediately and
421      * indefinitely.
422      */
423     private boolean mShouldThrottle;
424 
425     /** How much time each app will have to run jobs within their standby bucket window. */
426     private long mAllowedTimePerPeriodMs = QcConstants.DEFAULT_ALLOWED_TIME_PER_PERIOD_MS;
427 
428     /**
429      * The maximum amount of time an app can have its jobs running within a {@link #MAX_PERIOD_MS}
430      * window.
431      */
432     private long mMaxExecutionTimeMs = QcConstants.DEFAULT_MAX_EXECUTION_TIME_MS;
433 
434     /**
435      * How much time the app should have before transitioning from out-of-quota to in-quota.
436      * This should not affect processing if the app is already in-quota.
437      */
438     private long mQuotaBufferMs = QcConstants.DEFAULT_IN_QUOTA_BUFFER_MS;
439 
440     /**
441      * {@link #mAllowedTimePerPeriodMs} - {@link #mQuotaBufferMs}. This can be used to determine
442      * when an app will have enough quota to transition from out-of-quota to in-quota.
443      */
444     private long mAllowedTimeIntoQuotaMs = mAllowedTimePerPeriodMs - mQuotaBufferMs;
445 
446     /**
447      * {@link #mMaxExecutionTimeMs} - {@link #mQuotaBufferMs}. This can be used to determine when an
448      * app will have enough quota to transition from out-of-quota to in-quota.
449      */
450     private long mMaxExecutionTimeIntoQuotaMs = mMaxExecutionTimeMs - mQuotaBufferMs;
451 
452     /** The period of time used to rate limit recently run jobs. */
453     private long mRateLimitingWindowMs = QcConstants.DEFAULT_RATE_LIMITING_WINDOW_MS;
454 
455     /** The maximum number of jobs that can run within the past {@link #mRateLimitingWindowMs}. */
456     private int mMaxJobCountPerRateLimitingWindow =
457             QcConstants.DEFAULT_MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW;
458 
459     /**
460      * The maximum number of {@link TimingSession}s that can run within the past {@link
461      * #mRateLimitingWindowMs}.
462      */
463     private int mMaxSessionCountPerRateLimitingWindow =
464             QcConstants.DEFAULT_MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW;
465 
466     private long mNextCleanupTimeElapsed = 0;
467     private final AlarmManager.OnAlarmListener mSessionCleanupAlarmListener =
468             new AlarmManager.OnAlarmListener() {
469                 @Override
470                 public void onAlarm() {
471                     mHandler.obtainMessage(MSG_CLEAN_UP_SESSIONS).sendToTarget();
472                 }
473             };
474 
475     private final IUidObserver mUidObserver = new IUidObserver.Stub() {
476         @Override
477         public void onUidStateChanged(int uid, int procState, long procStateSeq) {
478             mHandler.obtainMessage(MSG_UID_PROCESS_STATE_CHANGED, uid, procState).sendToTarget();
479         }
480 
481         @Override
482         public void onUidGone(int uid, boolean disabled) {
483         }
484 
485         @Override
486         public void onUidActive(int uid) {
487         }
488 
489         @Override
490         public void onUidIdle(int uid, boolean disabled) {
491         }
492 
493         @Override
494         public void onUidCachedChanged(int uid, boolean cached) {
495         }
496     };
497 
498     private final BroadcastReceiver mPackageAddedReceiver = new BroadcastReceiver() {
499         @Override
500         public void onReceive(Context context, Intent intent) {
501             if (intent == null) {
502                 return;
503             }
504             if (intent.getBooleanExtra(Intent.EXTRA_REPLACING, false)) {
505                 return;
506             }
507             final int uid = intent.getIntExtra(Intent.EXTRA_UID, -1);
508             synchronized (mLock) {
509                 mUidToPackageCache.remove(uid);
510             }
511         }
512     };
513 
514     /**
515      * The rolling window size for each standby bucket. Within each window, an app will have 10
516      * minutes to run its jobs.
517      */
518     private final long[] mBucketPeriodsMs = new long[]{
519             QcConstants.DEFAULT_WINDOW_SIZE_ACTIVE_MS,
520             QcConstants.DEFAULT_WINDOW_SIZE_WORKING_MS,
521             QcConstants.DEFAULT_WINDOW_SIZE_FREQUENT_MS,
522             QcConstants.DEFAULT_WINDOW_SIZE_RARE_MS
523     };
524 
525     /** The maximum period any bucket can have. */
526     private static final long MAX_PERIOD_MS = 24 * 60 * MINUTE_IN_MILLIS;
527 
528     /**
529      * The maximum number of jobs based on its standby bucket. For each max value count in the
530      * array, the app will not be allowed to run more than that many number of jobs within the
531      * latest time interval of its rolling window size.
532      *
533      * @see #mBucketPeriodsMs
534      */
535     private final int[] mMaxBucketJobCounts = new int[]{
536             QcConstants.DEFAULT_MAX_JOB_COUNT_ACTIVE,
537             QcConstants.DEFAULT_MAX_JOB_COUNT_WORKING,
538             QcConstants.DEFAULT_MAX_JOB_COUNT_FREQUENT,
539             QcConstants.DEFAULT_MAX_JOB_COUNT_RARE
540     };
541 
542     /**
543      * The maximum number of {@link TimingSession}s based on its standby bucket. For each max value
544      * count in the array, the app will not be allowed to have more than that many number of
545      * {@link TimingSession}s within the latest time interval of its rolling window size.
546      *
547      * @see #mBucketPeriodsMs
548      */
549     private final int[] mMaxBucketSessionCounts = new int[]{
550             QcConstants.DEFAULT_MAX_SESSION_COUNT_ACTIVE,
551             QcConstants.DEFAULT_MAX_SESSION_COUNT_WORKING,
552             QcConstants.DEFAULT_MAX_SESSION_COUNT_FREQUENT,
553             QcConstants.DEFAULT_MAX_SESSION_COUNT_RARE
554     };
555 
556     /**
557      * Treat two distinct {@link TimingSession}s as the same if they start and end within this
558      * amount of time of each other.
559      */
560     private long mTimingSessionCoalescingDurationMs =
561             QcConstants.DEFAULT_TIMING_SESSION_COALESCING_DURATION_MS;
562 
563     /** An app has reached its quota. The message should contain a {@link Package} object. */
564     private static final int MSG_REACHED_QUOTA = 0;
565     /** Drop any old timing sessions. */
566     private static final int MSG_CLEAN_UP_SESSIONS = 1;
567     /** Check if a package is now within its quota. */
568     private static final int MSG_CHECK_PACKAGE = 2;
569     /** Process state for a UID has changed. */
570     private static final int MSG_UID_PROCESS_STATE_CHANGED = 3;
571 
QuotaController(JobSchedulerService service)572     public QuotaController(JobSchedulerService service) {
573         super(service);
574         mHandler = new QcHandler(mContext.getMainLooper());
575         mChargeTracker = new ChargingTracker();
576         mChargeTracker.startTracking();
577         mActivityManagerInternal = LocalServices.getService(ActivityManagerInternal.class);
578         mAlarmManager = (AlarmManager) mContext.getSystemService(Context.ALARM_SERVICE);
579         mQcConstants = new QcConstants(mHandler);
580 
581         final IntentFilter filter = new IntentFilter(Intent.ACTION_PACKAGE_ADDED);
582         mContext.registerReceiverAsUser(mPackageAddedReceiver, UserHandle.ALL, filter, null, null);
583 
584         // Set up the app standby bucketing tracker
585         UsageStatsManagerInternal usageStats = LocalServices.getService(
586                 UsageStatsManagerInternal.class);
587         usageStats.addAppIdleStateChangeListener(new StandbyTracker());
588 
589         try {
590             ActivityManager.getService().registerUidObserver(mUidObserver,
591                     ActivityManager.UID_OBSERVER_PROCSTATE,
592                     ActivityManager.PROCESS_STATE_FOREGROUND_SERVICE, null);
593         } catch (RemoteException e) {
594             // ignored; both services live in system_server
595         }
596 
597         mShouldThrottle = !mConstants.USE_HEARTBEATS;
598     }
599 
600     @Override
onSystemServicesReady()601     public void onSystemServicesReady() {
602         mQcConstants.start(mContext.getContentResolver());
603     }
604 
605     @Override
maybeStartTrackingJobLocked(JobStatus jobStatus, JobStatus lastJob)606     public void maybeStartTrackingJobLocked(JobStatus jobStatus, JobStatus lastJob) {
607         final int userId = jobStatus.getSourceUserId();
608         final String pkgName = jobStatus.getSourcePackageName();
609         // Still need to track jobs even if mShouldThrottle is false in case it's set to true at
610         // some point.
611         ArraySet<JobStatus> jobs = mTrackedJobs.get(userId, pkgName);
612         if (jobs == null) {
613             jobs = new ArraySet<>();
614             mTrackedJobs.add(userId, pkgName, jobs);
615         }
616         jobs.add(jobStatus);
617         jobStatus.setTrackingController(JobStatus.TRACKING_QUOTA);
618         if (mShouldThrottle) {
619             final boolean isWithinQuota = isWithinQuotaLocked(jobStatus);
620             setConstraintSatisfied(jobStatus, isWithinQuota);
621             if (!isWithinQuota) {
622                 maybeScheduleStartAlarmLocked(userId, pkgName,
623                         getEffectiveStandbyBucket(jobStatus));
624             }
625         } else {
626             // QuotaController isn't throttling, so always set to true.
627             jobStatus.setQuotaConstraintSatisfied(true);
628         }
629     }
630 
631     @Override
prepareForExecutionLocked(JobStatus jobStatus)632     public void prepareForExecutionLocked(JobStatus jobStatus) {
633         if (DEBUG) {
634             Slog.d(TAG, "Prepping for " + jobStatus.toShortString());
635         }
636 
637         final int uid = jobStatus.getSourceUid();
638         if (mActivityManagerInternal.getUidProcessState(uid) <= ActivityManager.PROCESS_STATE_TOP) {
639             if (DEBUG) {
640                 Slog.d(TAG, jobStatus.toShortString() + " is top started job");
641             }
642             mTopStartedJobs.add(jobStatus);
643             // Top jobs won't count towards quota so there's no need to involve the Timer.
644             return;
645         }
646 
647         final int userId = jobStatus.getSourceUserId();
648         final String packageName = jobStatus.getSourcePackageName();
649         Timer timer = mPkgTimers.get(userId, packageName);
650         if (timer == null) {
651             timer = new Timer(uid, userId, packageName);
652             mPkgTimers.add(userId, packageName, timer);
653         }
654         timer.startTrackingJobLocked(jobStatus);
655     }
656 
657     @Override
maybeStopTrackingJobLocked(JobStatus jobStatus, JobStatus incomingJob, boolean forUpdate)658     public void maybeStopTrackingJobLocked(JobStatus jobStatus, JobStatus incomingJob,
659             boolean forUpdate) {
660         if (jobStatus.clearTrackingController(JobStatus.TRACKING_QUOTA)) {
661             Timer timer = mPkgTimers.get(jobStatus.getSourceUserId(),
662                     jobStatus.getSourcePackageName());
663             if (timer != null) {
664                 timer.stopTrackingJob(jobStatus);
665             }
666             ArraySet<JobStatus> jobs = mTrackedJobs.get(jobStatus.getSourceUserId(),
667                     jobStatus.getSourcePackageName());
668             if (jobs != null) {
669                 jobs.remove(jobStatus);
670             }
671             mTopStartedJobs.remove(jobStatus);
672         }
673     }
674 
675     @Override
onConstantsUpdatedLocked()676     public void onConstantsUpdatedLocked() {
677         if (mShouldThrottle == mConstants.USE_HEARTBEATS) {
678             mShouldThrottle = !mConstants.USE_HEARTBEATS;
679 
680             // Update job bookkeeping out of band.
681             BackgroundThread.getHandler().post(() -> {
682                 synchronized (mLock) {
683                     maybeUpdateAllConstraintsLocked();
684                 }
685             });
686         }
687     }
688 
689     @Override
onAppRemovedLocked(String packageName, int uid)690     public void onAppRemovedLocked(String packageName, int uid) {
691         if (packageName == null) {
692             Slog.wtf(TAG, "Told app removed but given null package name.");
693             return;
694         }
695         final int userId = UserHandle.getUserId(uid);
696         mTrackedJobs.delete(userId, packageName);
697         Timer timer = mPkgTimers.get(userId, packageName);
698         if (timer != null) {
699             if (timer.isActive()) {
700                 Slog.wtf(TAG, "onAppRemovedLocked called before Timer turned off.");
701                 timer.dropEverythingLocked();
702             }
703             mPkgTimers.delete(userId, packageName);
704         }
705         mTimingSessions.delete(userId, packageName);
706         QcAlarmListener alarmListener = mInQuotaAlarmListeners.get(userId, packageName);
707         if (alarmListener != null) {
708             mAlarmManager.cancel(alarmListener);
709             mInQuotaAlarmListeners.delete(userId, packageName);
710         }
711         mExecutionStatsCache.delete(userId, packageName);
712         mForegroundUids.delete(uid);
713         mUidToPackageCache.remove(uid);
714     }
715 
716     @Override
onUserRemovedLocked(int userId)717     public void onUserRemovedLocked(int userId) {
718         mTrackedJobs.delete(userId);
719         mPkgTimers.delete(userId);
720         mTimingSessions.delete(userId);
721         mInQuotaAlarmListeners.delete(userId);
722         mExecutionStatsCache.delete(userId);
723         mUidToPackageCache.clear();
724     }
725 
isUidInForeground(int uid)726     private boolean isUidInForeground(int uid) {
727         if (UserHandle.isCore(uid)) {
728             return true;
729         }
730         synchronized (mLock) {
731             return mForegroundUids.get(uid);
732         }
733     }
734 
735     /** @return true if the job was started while the app was in the TOP state. */
isTopStartedJobLocked(@onNull final JobStatus jobStatus)736     private boolean isTopStartedJobLocked(@NonNull final JobStatus jobStatus) {
737         return mTopStartedJobs.contains(jobStatus);
738     }
739 
740     /**
741      * Returns an appropriate standby bucket for the job, taking into account any standby
742      * exemptions.
743      */
getEffectiveStandbyBucket(@onNull final JobStatus jobStatus)744     private int getEffectiveStandbyBucket(@NonNull final JobStatus jobStatus) {
745         if (jobStatus.uidActive || jobStatus.getJob().isExemptedFromAppStandby()) {
746             // Treat these cases as if they're in the ACTIVE bucket so that they get throttled
747             // like other ACTIVE apps.
748             return ACTIVE_INDEX;
749         }
750         return jobStatus.getStandbyBucket();
751     }
752 
753     @VisibleForTesting
isWithinQuotaLocked(@onNull final JobStatus jobStatus)754     boolean isWithinQuotaLocked(@NonNull final JobStatus jobStatus) {
755         final int standbyBucket = getEffectiveStandbyBucket(jobStatus);
756         // A job is within quota if one of the following is true:
757         //   1. it was started while the app was in the TOP state
758         //   2. the app is currently in the foreground
759         //   3. the app overall is within its quota
760         return isTopStartedJobLocked(jobStatus)
761                 || isUidInForeground(jobStatus.getSourceUid())
762                 || isWithinQuotaLocked(
763                 jobStatus.getSourceUserId(), jobStatus.getSourcePackageName(), standbyBucket);
764     }
765 
766     @VisibleForTesting
isWithinQuotaLocked(final int userId, @NonNull final String packageName, final int standbyBucket)767     boolean isWithinQuotaLocked(final int userId, @NonNull final String packageName,
768             final int standbyBucket) {
769         if (standbyBucket == NEVER_INDEX) return false;
770         // This check is needed in case the flag is toggled after a job has been registered.
771         if (!mShouldThrottle) return true;
772 
773         // Quota constraint is not enforced while charging or when parole is on.
774         if (mChargeTracker.isCharging() || mInParole) {
775             return true;
776         }
777 
778         ExecutionStats stats = getExecutionStatsLocked(userId, packageName, standbyBucket);
779         return getRemainingExecutionTimeLocked(stats) > 0
780                 && isUnderJobCountQuotaLocked(stats, standbyBucket)
781                 && isUnderSessionCountQuotaLocked(stats, standbyBucket);
782     }
783 
isUnderJobCountQuotaLocked(@onNull ExecutionStats stats, final int standbyBucket)784     private boolean isUnderJobCountQuotaLocked(@NonNull ExecutionStats stats,
785             final int standbyBucket) {
786         final long now = sElapsedRealtimeClock.millis();
787         final boolean isUnderAllowedTimeQuota =
788                 (stats.jobRateLimitExpirationTimeElapsed <= now
789                         || stats.jobCountInRateLimitingWindow < mMaxJobCountPerRateLimitingWindow);
790         return isUnderAllowedTimeQuota
791                 && (stats.bgJobCountInWindow < mMaxBucketJobCounts[standbyBucket]);
792     }
793 
isUnderSessionCountQuotaLocked(@onNull ExecutionStats stats, final int standbyBucket)794     private boolean isUnderSessionCountQuotaLocked(@NonNull ExecutionStats stats,
795             final int standbyBucket) {
796         final long now = sElapsedRealtimeClock.millis();
797         final boolean isUnderAllowedTimeQuota = (stats.sessionRateLimitExpirationTimeElapsed <= now
798                 || stats.sessionCountInRateLimitingWindow < mMaxSessionCountPerRateLimitingWindow);
799         return isUnderAllowedTimeQuota
800                 && stats.sessionCountInWindow < mMaxBucketSessionCounts[standbyBucket];
801     }
802 
803     @VisibleForTesting
getRemainingExecutionTimeLocked(@onNull final JobStatus jobStatus)804     long getRemainingExecutionTimeLocked(@NonNull final JobStatus jobStatus) {
805         return getRemainingExecutionTimeLocked(jobStatus.getSourceUserId(),
806                 jobStatus.getSourcePackageName(),
807                 getEffectiveStandbyBucket(jobStatus));
808     }
809 
810     @VisibleForTesting
getRemainingExecutionTimeLocked(final int userId, @NonNull final String packageName)811     long getRemainingExecutionTimeLocked(final int userId, @NonNull final String packageName) {
812         final int standbyBucket = JobSchedulerService.standbyBucketForPackage(packageName,
813                 userId, sElapsedRealtimeClock.millis());
814         return getRemainingExecutionTimeLocked(userId, packageName, standbyBucket);
815     }
816 
817     /**
818      * Returns the amount of time, in milliseconds, that this job has remaining to run based on its
819      * current standby bucket. Time remaining could be negative if the app was moved from a less
820      * restricted to a more restricted bucket.
821      */
getRemainingExecutionTimeLocked(final int userId, @NonNull final String packageName, final int standbyBucket)822     private long getRemainingExecutionTimeLocked(final int userId,
823             @NonNull final String packageName, final int standbyBucket) {
824         if (standbyBucket == NEVER_INDEX) {
825             return 0;
826         }
827         return getRemainingExecutionTimeLocked(
828                 getExecutionStatsLocked(userId, packageName, standbyBucket));
829     }
830 
getRemainingExecutionTimeLocked(@onNull ExecutionStats stats)831     private long getRemainingExecutionTimeLocked(@NonNull ExecutionStats stats) {
832         return Math.min(mAllowedTimePerPeriodMs - stats.executionTimeInWindowMs,
833                 mMaxExecutionTimeMs - stats.executionTimeInMaxPeriodMs);
834     }
835 
836     /**
837      * Returns the amount of time, in milliseconds, until the package would have reached its
838      * duration quota, assuming it has a job counting towards its quota the entire time. This takes
839      * into account any {@link TimingSession}s that may roll out of the window as the job is
840      * running.
841      */
842     @VisibleForTesting
getTimeUntilQuotaConsumedLocked(final int userId, @NonNull final String packageName)843     long getTimeUntilQuotaConsumedLocked(final int userId, @NonNull final String packageName) {
844         final long nowElapsed = sElapsedRealtimeClock.millis();
845         final int standbyBucket = JobSchedulerService.standbyBucketForPackage(
846                 packageName, userId, nowElapsed);
847         if (standbyBucket == NEVER_INDEX) {
848             return 0;
849         }
850         List<TimingSession> sessions = mTimingSessions.get(userId, packageName);
851         if (sessions == null || sessions.size() == 0) {
852             return mAllowedTimePerPeriodMs;
853         }
854 
855         final ExecutionStats stats = getExecutionStatsLocked(userId, packageName, standbyBucket);
856         final long startWindowElapsed = nowElapsed - stats.windowSizeMs;
857         final long startMaxElapsed = nowElapsed - MAX_PERIOD_MS;
858         final long allowedTimeRemainingMs = mAllowedTimePerPeriodMs - stats.executionTimeInWindowMs;
859         final long maxExecutionTimeRemainingMs =
860                 mMaxExecutionTimeMs - stats.executionTimeInMaxPeriodMs;
861 
862         // Regular ACTIVE case. Since the bucket size equals the allowed time, the app jobs can
863         // essentially run until they reach the maximum limit.
864         if (stats.windowSizeMs == mAllowedTimePerPeriodMs) {
865             return calculateTimeUntilQuotaConsumedLocked(
866                     sessions, startMaxElapsed, maxExecutionTimeRemainingMs);
867         }
868 
869         // Need to check both max time and period time in case one is less than the other.
870         // For example, max time remaining could be less than bucket time remaining, but sessions
871         // contributing to the max time remaining could phase out enough that we'd want to use the
872         // bucket value.
873         return Math.min(
874                 calculateTimeUntilQuotaConsumedLocked(
875                         sessions, startMaxElapsed, maxExecutionTimeRemainingMs),
876                 calculateTimeUntilQuotaConsumedLocked(
877                         sessions, startWindowElapsed, allowedTimeRemainingMs));
878     }
879 
880     /**
881      * Calculates how much time it will take, in milliseconds, until the quota is fully consumed.
882      *
883      * @param windowStartElapsed The start of the window, in the elapsed realtime timebase.
884      * @param deadSpaceMs        How much time can be allowed to count towards the quota
885      */
calculateTimeUntilQuotaConsumedLocked(@onNull List<TimingSession> sessions, final long windowStartElapsed, long deadSpaceMs)886     private long calculateTimeUntilQuotaConsumedLocked(@NonNull List<TimingSession> sessions,
887             final long windowStartElapsed, long deadSpaceMs) {
888         long timeUntilQuotaConsumedMs = 0;
889         long start = windowStartElapsed;
890         for (int i = 0; i < sessions.size(); ++i) {
891             TimingSession session = sessions.get(i);
892 
893             if (session.endTimeElapsed < windowStartElapsed) {
894                 // Outside of window. Ignore.
895                 continue;
896             } else if (session.startTimeElapsed <= windowStartElapsed) {
897                 // Overlapping session. Can extend time by portion of session in window.
898                 timeUntilQuotaConsumedMs += session.endTimeElapsed - windowStartElapsed;
899                 start = session.endTimeElapsed;
900             } else {
901                 // Completely within the window. Can only consider if there's enough dead space
902                 // to get to the start of the session.
903                 long diff = session.startTimeElapsed - start;
904                 if (diff > deadSpaceMs) {
905                     break;
906                 }
907                 timeUntilQuotaConsumedMs += diff
908                         + (session.endTimeElapsed - session.startTimeElapsed);
909                 deadSpaceMs -= diff;
910                 start = session.endTimeElapsed;
911             }
912         }
913         // Will be non-zero if the loop didn't look at any sessions.
914         timeUntilQuotaConsumedMs += deadSpaceMs;
915         if (timeUntilQuotaConsumedMs > mMaxExecutionTimeMs) {
916             Slog.wtf(TAG, "Calculated quota consumed time too high: " + timeUntilQuotaConsumedMs);
917         }
918         return timeUntilQuotaConsumedMs;
919     }
920 
921     /** Returns the execution stats of the app in the most recent window. */
922     @VisibleForTesting
923     @NonNull
getExecutionStatsLocked(final int userId, @NonNull final String packageName, final int standbyBucket)924     ExecutionStats getExecutionStatsLocked(final int userId, @NonNull final String packageName,
925             final int standbyBucket) {
926         return getExecutionStatsLocked(userId, packageName, standbyBucket, true);
927     }
928 
929     @NonNull
getExecutionStatsLocked(final int userId, @NonNull final String packageName, final int standbyBucket, final boolean refreshStatsIfOld)930     private ExecutionStats getExecutionStatsLocked(final int userId,
931             @NonNull final String packageName, final int standbyBucket,
932             final boolean refreshStatsIfOld) {
933         if (standbyBucket == NEVER_INDEX) {
934             Slog.wtf(TAG, "getExecutionStatsLocked called for a NEVER app.");
935             return new ExecutionStats();
936         }
937         ExecutionStats[] appStats = mExecutionStatsCache.get(userId, packageName);
938         if (appStats == null) {
939             appStats = new ExecutionStats[mBucketPeriodsMs.length];
940             mExecutionStatsCache.add(userId, packageName, appStats);
941         }
942         ExecutionStats stats = appStats[standbyBucket];
943         if (stats == null) {
944             stats = new ExecutionStats();
945             appStats[standbyBucket] = stats;
946         }
947         if (refreshStatsIfOld) {
948             final long bucketWindowSizeMs = mBucketPeriodsMs[standbyBucket];
949             final int jobCountLimit = mMaxBucketJobCounts[standbyBucket];
950             final int sessionCountLimit = mMaxBucketSessionCounts[standbyBucket];
951             Timer timer = mPkgTimers.get(userId, packageName);
952             if ((timer != null && timer.isActive())
953                     || stats.expirationTimeElapsed <= sElapsedRealtimeClock.millis()
954                     || stats.windowSizeMs != bucketWindowSizeMs
955                     || stats.jobCountLimit != jobCountLimit
956                     || stats.sessionCountLimit != sessionCountLimit) {
957                 // The stats are no longer valid.
958                 stats.windowSizeMs = bucketWindowSizeMs;
959                 stats.jobCountLimit = jobCountLimit;
960                 stats.sessionCountLimit = sessionCountLimit;
961                 updateExecutionStatsLocked(userId, packageName, stats);
962             }
963         }
964 
965         return stats;
966     }
967 
968     @VisibleForTesting
updateExecutionStatsLocked(final int userId, @NonNull final String packageName, @NonNull ExecutionStats stats)969     void updateExecutionStatsLocked(final int userId, @NonNull final String packageName,
970             @NonNull ExecutionStats stats) {
971         stats.executionTimeInWindowMs = 0;
972         stats.bgJobCountInWindow = 0;
973         stats.executionTimeInMaxPeriodMs = 0;
974         stats.bgJobCountInMaxPeriod = 0;
975         stats.sessionCountInWindow = 0;
976         stats.inQuotaTimeElapsed = 0;
977 
978         Timer timer = mPkgTimers.get(userId, packageName);
979         final long nowElapsed = sElapsedRealtimeClock.millis();
980         stats.expirationTimeElapsed = nowElapsed + MAX_PERIOD_MS;
981         if (timer != null && timer.isActive()) {
982             stats.executionTimeInWindowMs =
983                     stats.executionTimeInMaxPeriodMs = timer.getCurrentDuration(nowElapsed);
984             stats.bgJobCountInWindow = stats.bgJobCountInMaxPeriod = timer.getBgJobCount();
985             // If the timer is active, the value will be stale at the next method call, so
986             // invalidate now.
987             stats.expirationTimeElapsed = nowElapsed;
988             if (stats.executionTimeInWindowMs >= mAllowedTimeIntoQuotaMs) {
989                 stats.inQuotaTimeElapsed = Math.max(stats.inQuotaTimeElapsed,
990                         nowElapsed - mAllowedTimeIntoQuotaMs + stats.windowSizeMs);
991             }
992             if (stats.executionTimeInMaxPeriodMs >= mMaxExecutionTimeIntoQuotaMs) {
993                 stats.inQuotaTimeElapsed = Math.max(stats.inQuotaTimeElapsed,
994                         nowElapsed - mMaxExecutionTimeIntoQuotaMs + MAX_PERIOD_MS);
995             }
996         }
997 
998         List<TimingSession> sessions = mTimingSessions.get(userId, packageName);
999         if (sessions == null || sessions.size() == 0) {
1000             return;
1001         }
1002 
1003         final long startWindowElapsed = nowElapsed - stats.windowSizeMs;
1004         final long startMaxElapsed = nowElapsed - MAX_PERIOD_MS;
1005         int sessionCountInWindow = 0;
1006         // The minimum time between the start time and the beginning of the sessions that were
1007         // looked at --> how much time the stats will be valid for.
1008         long emptyTimeMs = Long.MAX_VALUE;
1009         // Sessions are non-overlapping and in order of occurrence, so iterating backwards will get
1010         // the most recent ones.
1011         final int loopStart = sessions.size() - 1;
1012         for (int i = loopStart; i >= 0; --i) {
1013             TimingSession session = sessions.get(i);
1014 
1015             // Window management.
1016             if (startWindowElapsed < session.endTimeElapsed) {
1017                 final long start;
1018                 if (startWindowElapsed < session.startTimeElapsed) {
1019                     start = session.startTimeElapsed;
1020                     emptyTimeMs =
1021                             Math.min(emptyTimeMs, session.startTimeElapsed - startWindowElapsed);
1022                 } else {
1023                     // The session started before the window but ended within the window. Only
1024                     // include the portion that was within the window.
1025                     start = startWindowElapsed;
1026                     emptyTimeMs = 0;
1027                 }
1028 
1029                 stats.executionTimeInWindowMs += session.endTimeElapsed - start;
1030                 stats.bgJobCountInWindow += session.bgJobCount;
1031                 if (stats.executionTimeInWindowMs >= mAllowedTimeIntoQuotaMs) {
1032                     stats.inQuotaTimeElapsed = Math.max(stats.inQuotaTimeElapsed,
1033                             start + stats.executionTimeInWindowMs - mAllowedTimeIntoQuotaMs
1034                                     + stats.windowSizeMs);
1035                 }
1036                 if (stats.bgJobCountInWindow >= stats.jobCountLimit) {
1037                     stats.inQuotaTimeElapsed = Math.max(stats.inQuotaTimeElapsed,
1038                             session.endTimeElapsed + stats.windowSizeMs);
1039                 }
1040                 if (i == loopStart
1041                         || (sessions.get(i + 1).startTimeElapsed - session.endTimeElapsed)
1042                                 > mTimingSessionCoalescingDurationMs) {
1043                     // Coalesce sessions if they are very close to each other in time
1044                     sessionCountInWindow++;
1045 
1046                     if (sessionCountInWindow >= stats.sessionCountLimit) {
1047                         stats.inQuotaTimeElapsed = Math.max(stats.inQuotaTimeElapsed,
1048                                 session.endTimeElapsed + stats.windowSizeMs);
1049                     }
1050                 }
1051             }
1052 
1053             // Max period check.
1054             if (startMaxElapsed < session.startTimeElapsed) {
1055                 stats.executionTimeInMaxPeriodMs +=
1056                         session.endTimeElapsed - session.startTimeElapsed;
1057                 stats.bgJobCountInMaxPeriod += session.bgJobCount;
1058                 emptyTimeMs = Math.min(emptyTimeMs, session.startTimeElapsed - startMaxElapsed);
1059                 if (stats.executionTimeInMaxPeriodMs >= mMaxExecutionTimeIntoQuotaMs) {
1060                     stats.inQuotaTimeElapsed = Math.max(stats.inQuotaTimeElapsed,
1061                             session.startTimeElapsed + stats.executionTimeInMaxPeriodMs
1062                                     - mMaxExecutionTimeIntoQuotaMs + MAX_PERIOD_MS);
1063                 }
1064             } else if (startMaxElapsed < session.endTimeElapsed) {
1065                 // The session started before the window but ended within the window. Only include
1066                 // the portion that was within the window.
1067                 stats.executionTimeInMaxPeriodMs += session.endTimeElapsed - startMaxElapsed;
1068                 stats.bgJobCountInMaxPeriod += session.bgJobCount;
1069                 emptyTimeMs = 0;
1070                 if (stats.executionTimeInMaxPeriodMs >= mMaxExecutionTimeIntoQuotaMs) {
1071                     stats.inQuotaTimeElapsed = Math.max(stats.inQuotaTimeElapsed,
1072                             startMaxElapsed + stats.executionTimeInMaxPeriodMs
1073                                     - mMaxExecutionTimeIntoQuotaMs + MAX_PERIOD_MS);
1074                 }
1075             } else {
1076                 // This session ended before the window. No point in going any further.
1077                 break;
1078             }
1079         }
1080         stats.expirationTimeElapsed = nowElapsed + emptyTimeMs;
1081         stats.sessionCountInWindow = sessionCountInWindow;
1082     }
1083 
1084     /** Invalidate ExecutionStats for all apps. */
1085     @VisibleForTesting
invalidateAllExecutionStatsLocked()1086     void invalidateAllExecutionStatsLocked() {
1087         final long nowElapsed = sElapsedRealtimeClock.millis();
1088         mExecutionStatsCache.forEach((appStats) -> {
1089             if (appStats != null) {
1090                 for (int i = 0; i < appStats.length; ++i) {
1091                     ExecutionStats stats = appStats[i];
1092                     if (stats != null) {
1093                         stats.expirationTimeElapsed = nowElapsed;
1094                     }
1095                 }
1096             }
1097         });
1098     }
1099 
1100     @VisibleForTesting
invalidateAllExecutionStatsLocked(final int userId, @NonNull final String packageName)1101     void invalidateAllExecutionStatsLocked(final int userId,
1102             @NonNull final String packageName) {
1103         ExecutionStats[] appStats = mExecutionStatsCache.get(userId, packageName);
1104         if (appStats != null) {
1105             final long nowElapsed = sElapsedRealtimeClock.millis();
1106             for (int i = 0; i < appStats.length; ++i) {
1107                 ExecutionStats stats = appStats[i];
1108                 if (stats != null) {
1109                     stats.expirationTimeElapsed = nowElapsed;
1110                 }
1111             }
1112         }
1113     }
1114 
1115     @VisibleForTesting
incrementJobCount(final int userId, @NonNull final String packageName, int count)1116     void incrementJobCount(final int userId, @NonNull final String packageName, int count) {
1117         final long now = sElapsedRealtimeClock.millis();
1118         ExecutionStats[] appStats = mExecutionStatsCache.get(userId, packageName);
1119         if (appStats == null) {
1120             appStats = new ExecutionStats[mBucketPeriodsMs.length];
1121             mExecutionStatsCache.add(userId, packageName, appStats);
1122         }
1123         for (int i = 0; i < appStats.length; ++i) {
1124             ExecutionStats stats = appStats[i];
1125             if (stats == null) {
1126                 stats = new ExecutionStats();
1127                 appStats[i] = stats;
1128             }
1129             if (stats.jobRateLimitExpirationTimeElapsed <= now) {
1130                 stats.jobRateLimitExpirationTimeElapsed = now + mRateLimitingWindowMs;
1131                 stats.jobCountInRateLimitingWindow = 0;
1132             }
1133             stats.jobCountInRateLimitingWindow += count;
1134         }
1135     }
1136 
incrementTimingSessionCount(final int userId, @NonNull final String packageName)1137     private void incrementTimingSessionCount(final int userId, @NonNull final String packageName) {
1138         final long now = sElapsedRealtimeClock.millis();
1139         ExecutionStats[] appStats = mExecutionStatsCache.get(userId, packageName);
1140         if (appStats == null) {
1141             appStats = new ExecutionStats[mBucketPeriodsMs.length];
1142             mExecutionStatsCache.add(userId, packageName, appStats);
1143         }
1144         for (int i = 0; i < appStats.length; ++i) {
1145             ExecutionStats stats = appStats[i];
1146             if (stats == null) {
1147                 stats = new ExecutionStats();
1148                 appStats[i] = stats;
1149             }
1150             if (stats.sessionRateLimitExpirationTimeElapsed <= now) {
1151                 stats.sessionRateLimitExpirationTimeElapsed = now + mRateLimitingWindowMs;
1152                 stats.sessionCountInRateLimitingWindow = 0;
1153             }
1154             stats.sessionCountInRateLimitingWindow++;
1155         }
1156     }
1157 
1158     @VisibleForTesting
saveTimingSession(final int userId, @NonNull final String packageName, @NonNull final TimingSession session)1159     void saveTimingSession(final int userId, @NonNull final String packageName,
1160             @NonNull final TimingSession session) {
1161         synchronized (mLock) {
1162             List<TimingSession> sessions = mTimingSessions.get(userId, packageName);
1163             if (sessions == null) {
1164                 sessions = new ArrayList<>();
1165                 mTimingSessions.add(userId, packageName, sessions);
1166             }
1167             sessions.add(session);
1168             // Adding a new session means that the current stats are now incorrect.
1169             invalidateAllExecutionStatsLocked(userId, packageName);
1170 
1171             maybeScheduleCleanupAlarmLocked();
1172         }
1173     }
1174 
1175     private final class EarliestEndTimeFunctor implements Consumer<List<TimingSession>> {
1176         public long earliestEndElapsed = Long.MAX_VALUE;
1177 
1178         @Override
accept(List<TimingSession> sessions)1179         public void accept(List<TimingSession> sessions) {
1180             if (sessions != null && sessions.size() > 0) {
1181                 earliestEndElapsed = Math.min(earliestEndElapsed, sessions.get(0).endTimeElapsed);
1182             }
1183         }
1184 
reset()1185         void reset() {
1186             earliestEndElapsed = Long.MAX_VALUE;
1187         }
1188     }
1189 
1190     private final EarliestEndTimeFunctor mEarliestEndTimeFunctor = new EarliestEndTimeFunctor();
1191 
1192     /** Schedule a cleanup alarm if necessary and there isn't already one scheduled. */
1193     @VisibleForTesting
maybeScheduleCleanupAlarmLocked()1194     void maybeScheduleCleanupAlarmLocked() {
1195         if (mNextCleanupTimeElapsed > sElapsedRealtimeClock.millis()) {
1196             // There's already an alarm scheduled. Just stick with that one. There's no way we'll
1197             // end up scheduling an earlier alarm.
1198             if (DEBUG) {
1199                 Slog.v(TAG, "Not scheduling cleanup since there's already one at "
1200                         + mNextCleanupTimeElapsed + " (in " + (mNextCleanupTimeElapsed
1201                         - sElapsedRealtimeClock.millis()) + "ms)");
1202             }
1203             return;
1204         }
1205         mEarliestEndTimeFunctor.reset();
1206         mTimingSessions.forEach(mEarliestEndTimeFunctor);
1207         final long earliestEndElapsed = mEarliestEndTimeFunctor.earliestEndElapsed;
1208         if (earliestEndElapsed == Long.MAX_VALUE) {
1209             // Couldn't find a good time to clean up. Maybe this was called after we deleted all
1210             // timing sessions.
1211             if (DEBUG) {
1212                 Slog.d(TAG, "Didn't find a time to schedule cleanup");
1213             }
1214             return;
1215         }
1216         // Need to keep sessions for all apps up to the max period, regardless of their current
1217         // standby bucket.
1218         long nextCleanupElapsed = earliestEndElapsed + MAX_PERIOD_MS;
1219         if (nextCleanupElapsed - mNextCleanupTimeElapsed <= 10 * MINUTE_IN_MILLIS) {
1220             // No need to clean up too often. Delay the alarm if the next cleanup would be too soon
1221             // after it.
1222             nextCleanupElapsed += 10 * MINUTE_IN_MILLIS;
1223         }
1224         mNextCleanupTimeElapsed = nextCleanupElapsed;
1225         mAlarmManager.set(AlarmManager.ELAPSED_REALTIME, nextCleanupElapsed, ALARM_TAG_CLEANUP,
1226                 mSessionCleanupAlarmListener, mHandler);
1227         if (DEBUG) {
1228             Slog.d(TAG, "Scheduled next cleanup for " + mNextCleanupTimeElapsed);
1229         }
1230     }
1231 
handleNewChargingStateLocked()1232     private void handleNewChargingStateLocked() {
1233         final long nowElapsed = sElapsedRealtimeClock.millis();
1234         final boolean isCharging = mChargeTracker.isCharging();
1235         if (DEBUG) {
1236             Slog.d(TAG, "handleNewChargingStateLocked: " + isCharging);
1237         }
1238         // Deal with Timers first.
1239         mPkgTimers.forEach((t) -> t.onStateChangedLocked(nowElapsed, isCharging));
1240         // Now update jobs.
1241         maybeUpdateAllConstraintsLocked();
1242     }
1243 
maybeUpdateAllConstraintsLocked()1244     private void maybeUpdateAllConstraintsLocked() {
1245         boolean changed = false;
1246         for (int u = 0; u < mTrackedJobs.numUsers(); ++u) {
1247             final int userId = mTrackedJobs.keyAt(u);
1248             for (int p = 0; p < mTrackedJobs.numPackagesForUser(userId); ++p) {
1249                 final String packageName = mTrackedJobs.keyAt(u, p);
1250                 changed |= maybeUpdateConstraintForPkgLocked(userId, packageName);
1251             }
1252         }
1253         if (changed) {
1254             mStateChangedListener.onControllerStateChanged();
1255         }
1256     }
1257 
1258     /**
1259      * Update the CONSTRAINT_WITHIN_QUOTA bit for all of the Jobs for a given package.
1260      *
1261      * @return true if at least one job had its bit changed
1262      */
maybeUpdateConstraintForPkgLocked(final int userId, @NonNull final String packageName)1263     private boolean maybeUpdateConstraintForPkgLocked(final int userId,
1264             @NonNull final String packageName) {
1265         ArraySet<JobStatus> jobs = mTrackedJobs.get(userId, packageName);
1266         if (jobs == null || jobs.size() == 0) {
1267             return false;
1268         }
1269 
1270         // Quota is the same for all jobs within a package.
1271         final int realStandbyBucket = jobs.valueAt(0).getStandbyBucket();
1272         final boolean realInQuota = isWithinQuotaLocked(userId, packageName, realStandbyBucket);
1273         boolean changed = false;
1274         for (int i = jobs.size() - 1; i >= 0; --i) {
1275             final JobStatus js = jobs.valueAt(i);
1276             if (isTopStartedJobLocked(js)) {
1277                 // Job was started while the app was in the TOP state so we should allow it to
1278                 // finish.
1279                 changed |= js.setQuotaConstraintSatisfied(true);
1280             } else if (realStandbyBucket != ACTIVE_INDEX
1281                     && realStandbyBucket == getEffectiveStandbyBucket(js)) {
1282                 // An app in the ACTIVE bucket may be out of quota while the job could be in quota
1283                 // for some reason. Therefore, avoid setting the real value here and check each job
1284                 // individually.
1285                 changed |= setConstraintSatisfied(js, realInQuota);
1286             } else {
1287                 // This job is somehow exempted. Need to determine its own quota status.
1288                 changed |= setConstraintSatisfied(js, isWithinQuotaLocked(js));
1289             }
1290         }
1291         if (!realInQuota) {
1292             // Don't want to use the effective standby bucket here since that bump the bucket to
1293             // ACTIVE for one of the jobs, which doesn't help with other jobs that aren't
1294             // exempted.
1295             maybeScheduleStartAlarmLocked(userId, packageName, realStandbyBucket);
1296         } else {
1297             QcAlarmListener alarmListener = mInQuotaAlarmListeners.get(userId, packageName);
1298             if (alarmListener != null && alarmListener.isWaiting()) {
1299                 mAlarmManager.cancel(alarmListener);
1300                 // Set the trigger time to 0 so that the alarm doesn't think it's still waiting.
1301                 alarmListener.setTriggerTime(0);
1302             }
1303         }
1304         return changed;
1305     }
1306 
1307     private class UidConstraintUpdater implements Consumer<JobStatus> {
1308         private final UserPackageMap<Integer> mToScheduleStartAlarms = new UserPackageMap<>();
1309         public boolean wasJobChanged;
1310 
1311         @Override
accept(JobStatus jobStatus)1312         public void accept(JobStatus jobStatus) {
1313             wasJobChanged |= setConstraintSatisfied(jobStatus, isWithinQuotaLocked(jobStatus));
1314             final int userId = jobStatus.getSourceUserId();
1315             final String packageName = jobStatus.getSourcePackageName();
1316             final int realStandbyBucket = jobStatus.getStandbyBucket();
1317             if (isWithinQuotaLocked(userId, packageName, realStandbyBucket)) {
1318                 QcAlarmListener alarmListener = mInQuotaAlarmListeners.get(userId, packageName);
1319                 if (alarmListener != null && alarmListener.isWaiting()) {
1320                     mAlarmManager.cancel(alarmListener);
1321                     // Set the trigger time to 0 so that the alarm doesn't think it's still waiting.
1322                     alarmListener.setTriggerTime(0);
1323                 }
1324             } else {
1325                 mToScheduleStartAlarms.add(userId, packageName, realStandbyBucket);
1326             }
1327         }
1328 
postProcess()1329         void postProcess() {
1330             for (int u = 0; u < mToScheduleStartAlarms.numUsers(); ++u) {
1331                 final int userId = mToScheduleStartAlarms.keyAt(u);
1332                 for (int p = 0; p < mToScheduleStartAlarms.numPackagesForUser(userId); ++p) {
1333                     final String packageName = mToScheduleStartAlarms.keyAt(u, p);
1334                     final int standbyBucket = mToScheduleStartAlarms.get(userId, packageName);
1335                     maybeScheduleStartAlarmLocked(userId, packageName, standbyBucket);
1336                 }
1337             }
1338         }
1339 
reset()1340         void reset() {
1341             wasJobChanged = false;
1342             mToScheduleStartAlarms.clear();
1343         }
1344     }
1345 
1346     private final UidConstraintUpdater mUpdateUidConstraints = new UidConstraintUpdater();
1347 
maybeUpdateConstraintForUidLocked(final int uid)1348     private boolean maybeUpdateConstraintForUidLocked(final int uid) {
1349         mService.getJobStore().forEachJobForSourceUid(uid, mUpdateUidConstraints);
1350 
1351         mUpdateUidConstraints.postProcess();
1352         boolean changed = mUpdateUidConstraints.wasJobChanged;
1353         mUpdateUidConstraints.reset();
1354         return changed;
1355     }
1356 
1357     /**
1358      * Maybe schedule a non-wakeup alarm for the next time this package will have quota to run
1359      * again. This should only be called if the package is already out of quota.
1360      */
1361     @VisibleForTesting
maybeScheduleStartAlarmLocked(final int userId, @NonNull final String packageName, final int standbyBucket)1362     void maybeScheduleStartAlarmLocked(final int userId, @NonNull final String packageName,
1363             final int standbyBucket) {
1364         if (standbyBucket == NEVER_INDEX) {
1365             return;
1366         }
1367 
1368         final String pkgString = string(userId, packageName);
1369         ExecutionStats stats = getExecutionStatsLocked(userId, packageName, standbyBucket);
1370         final boolean isUnderJobCountQuota = isUnderJobCountQuotaLocked(stats, standbyBucket);
1371         final boolean isUnderTimingSessionCountQuota = isUnderSessionCountQuotaLocked(stats,
1372                 standbyBucket);
1373 
1374         QcAlarmListener alarmListener = mInQuotaAlarmListeners.get(userId, packageName);
1375         if (stats.executionTimeInWindowMs < mAllowedTimePerPeriodMs
1376                 && stats.executionTimeInMaxPeriodMs < mMaxExecutionTimeMs
1377                 && isUnderJobCountQuota
1378                 && isUnderTimingSessionCountQuota) {
1379             // Already in quota. Why was this method called?
1380             if (DEBUG) {
1381                 Slog.e(TAG, "maybeScheduleStartAlarmLocked called for " + pkgString
1382                         + " even though it already has "
1383                         + getRemainingExecutionTimeLocked(userId, packageName, standbyBucket)
1384                         + "ms in its quota.");
1385             }
1386             if (alarmListener != null) {
1387                 // Cancel any pending alarm.
1388                 mAlarmManager.cancel(alarmListener);
1389                 // Set the trigger time to 0 so that the alarm doesn't think it's still waiting.
1390                 alarmListener.setTriggerTime(0);
1391             }
1392             mHandler.obtainMessage(MSG_CHECK_PACKAGE, userId, 0, packageName).sendToTarget();
1393             return;
1394         }
1395 
1396         if (alarmListener == null) {
1397             alarmListener = new QcAlarmListener(userId, packageName);
1398             mInQuotaAlarmListeners.add(userId, packageName, alarmListener);
1399         }
1400 
1401         // The time this app will have quota again.
1402         long inQuotaTimeElapsed = stats.inQuotaTimeElapsed;
1403         if (!isUnderJobCountQuota && stats.bgJobCountInWindow < stats.jobCountLimit) {
1404             // App hit the rate limit.
1405             inQuotaTimeElapsed = Math.max(inQuotaTimeElapsed,
1406                     stats.jobRateLimitExpirationTimeElapsed);
1407         }
1408         if (!isUnderTimingSessionCountQuota
1409                 && stats.sessionCountInWindow < stats.sessionCountLimit) {
1410             // App hit the rate limit.
1411             inQuotaTimeElapsed = Math.max(inQuotaTimeElapsed,
1412                     stats.sessionRateLimitExpirationTimeElapsed);
1413         }
1414         // Only schedule the alarm if:
1415         // 1. There isn't one currently scheduled
1416         // 2. The new alarm is significantly earlier than the previous alarm (which could be the
1417         // case if the package moves into a higher standby bucket). If it's earlier but not
1418         // significantly so, then we essentially delay the job a few extra minutes.
1419         // 3. The alarm is after the current alarm by more than the quota buffer.
1420         // TODO: this might be overengineering. Simplify if proven safe.
1421         if (!alarmListener.isWaiting()
1422                 || inQuotaTimeElapsed < alarmListener.getTriggerTimeElapsed() - 3 * MINUTE_IN_MILLIS
1423                 || alarmListener.getTriggerTimeElapsed() < inQuotaTimeElapsed) {
1424             if (DEBUG) {
1425                 Slog.d(TAG, "Scheduling start alarm for " + pkgString);
1426             }
1427             // If the next time this app will have quota is at least 3 minutes before the
1428             // alarm is supposed to go off, reschedule the alarm.
1429             mAlarmManager.set(AlarmManager.ELAPSED_REALTIME, inQuotaTimeElapsed,
1430                     ALARM_TAG_QUOTA_CHECK, alarmListener, mHandler);
1431             alarmListener.setTriggerTime(inQuotaTimeElapsed);
1432         } else if (DEBUG) {
1433             Slog.d(TAG, "No need to schedule start alarm for " + pkgString);
1434         }
1435     }
1436 
setConstraintSatisfied(@onNull JobStatus jobStatus, boolean isWithinQuota)1437     private boolean setConstraintSatisfied(@NonNull JobStatus jobStatus, boolean isWithinQuota) {
1438         if (!isWithinQuota && jobStatus.getWhenStandbyDeferred() == 0) {
1439             // Mark that the job is being deferred due to buckets.
1440             jobStatus.setWhenStandbyDeferred(sElapsedRealtimeClock.millis());
1441         }
1442         return jobStatus.setQuotaConstraintSatisfied(isWithinQuota);
1443     }
1444 
1445     private final class ChargingTracker extends BroadcastReceiver {
1446         /**
1447          * Track whether we're charging. This has a slightly different definition than that of
1448          * BatteryController.
1449          */
1450         private boolean mCharging;
1451 
ChargingTracker()1452         ChargingTracker() {
1453         }
1454 
startTracking()1455         public void startTracking() {
1456             IntentFilter filter = new IntentFilter();
1457 
1458             // Charging/not charging.
1459             filter.addAction(BatteryManager.ACTION_CHARGING);
1460             filter.addAction(BatteryManager.ACTION_DISCHARGING);
1461             mContext.registerReceiver(this, filter);
1462 
1463             // Initialise tracker state.
1464             BatteryManagerInternal batteryManagerInternal =
1465                     LocalServices.getService(BatteryManagerInternal.class);
1466             mCharging = batteryManagerInternal.isPowered(BatteryManager.BATTERY_PLUGGED_ANY);
1467         }
1468 
isCharging()1469         public boolean isCharging() {
1470             return mCharging;
1471         }
1472 
1473         @Override
onReceive(Context context, Intent intent)1474         public void onReceive(Context context, Intent intent) {
1475             synchronized (mLock) {
1476                 final String action = intent.getAction();
1477                 if (BatteryManager.ACTION_CHARGING.equals(action)) {
1478                     if (DEBUG) {
1479                         Slog.d(TAG, "Received charging intent, fired @ "
1480                                 + sElapsedRealtimeClock.millis());
1481                     }
1482                     mCharging = true;
1483                     handleNewChargingStateLocked();
1484                 } else if (BatteryManager.ACTION_DISCHARGING.equals(action)) {
1485                     if (DEBUG) {
1486                         Slog.d(TAG, "Disconnected from power.");
1487                     }
1488                     mCharging = false;
1489                     handleNewChargingStateLocked();
1490                 }
1491             }
1492         }
1493     }
1494 
1495     @VisibleForTesting
1496     static final class TimingSession {
1497         // Start timestamp in elapsed realtime timebase.
1498         public final long startTimeElapsed;
1499         // End timestamp in elapsed realtime timebase.
1500         public final long endTimeElapsed;
1501         // How many background jobs ran during this session.
1502         public final int bgJobCount;
1503 
1504         private final int mHashCode;
1505 
TimingSession(long startElapsed, long endElapsed, int bgJobCount)1506         TimingSession(long startElapsed, long endElapsed, int bgJobCount) {
1507             this.startTimeElapsed = startElapsed;
1508             this.endTimeElapsed = endElapsed;
1509             this.bgJobCount = bgJobCount;
1510 
1511             int hashCode = 0;
1512             hashCode = 31 * hashCode + hashLong(startTimeElapsed);
1513             hashCode = 31 * hashCode + hashLong(endTimeElapsed);
1514             hashCode = 31 * hashCode + bgJobCount;
1515             mHashCode = hashCode;
1516         }
1517 
1518         @Override
toString()1519         public String toString() {
1520             return "TimingSession{" + startTimeElapsed + "->" + endTimeElapsed + ", " + bgJobCount
1521                     + "}";
1522         }
1523 
1524         @Override
equals(Object obj)1525         public boolean equals(Object obj) {
1526             if (obj instanceof TimingSession) {
1527                 TimingSession other = (TimingSession) obj;
1528                 return startTimeElapsed == other.startTimeElapsed
1529                         && endTimeElapsed == other.endTimeElapsed
1530                         && bgJobCount == other.bgJobCount;
1531             } else {
1532                 return false;
1533             }
1534         }
1535 
1536         @Override
hashCode()1537         public int hashCode() {
1538             return mHashCode;
1539         }
1540 
dump(IndentingPrintWriter pw)1541         public void dump(IndentingPrintWriter pw) {
1542             pw.print(startTimeElapsed);
1543             pw.print(" -> ");
1544             pw.print(endTimeElapsed);
1545             pw.print(" (");
1546             pw.print(endTimeElapsed - startTimeElapsed);
1547             pw.print("), ");
1548             pw.print(bgJobCount);
1549             pw.print(" bg jobs.");
1550             pw.println();
1551         }
1552 
dump(@onNull ProtoOutputStream proto, long fieldId)1553         public void dump(@NonNull ProtoOutputStream proto, long fieldId) {
1554             final long token = proto.start(fieldId);
1555 
1556             proto.write(StateControllerProto.QuotaController.TimingSession.START_TIME_ELAPSED,
1557                     startTimeElapsed);
1558             proto.write(StateControllerProto.QuotaController.TimingSession.END_TIME_ELAPSED,
1559                     endTimeElapsed);
1560             proto.write(StateControllerProto.QuotaController.TimingSession.BG_JOB_COUNT,
1561                     bgJobCount);
1562 
1563             proto.end(token);
1564         }
1565     }
1566 
1567     private final class Timer {
1568         private final Package mPkg;
1569         private final int mUid;
1570 
1571         // List of jobs currently running for this app that started when the app wasn't in the
1572         // foreground.
1573         private final ArraySet<JobStatus> mRunningBgJobs = new ArraySet<>();
1574         private long mStartTimeElapsed;
1575         private int mBgJobCount;
1576 
Timer(int uid, int userId, String packageName)1577         Timer(int uid, int userId, String packageName) {
1578             mPkg = new Package(userId, packageName);
1579             mUid = uid;
1580         }
1581 
startTrackingJobLocked(@onNull JobStatus jobStatus)1582         void startTrackingJobLocked(@NonNull JobStatus jobStatus) {
1583             if (isTopStartedJobLocked(jobStatus)) {
1584                 // We intentionally don't pay attention to fg state changes after a TOP job has
1585                 // started.
1586                 if (DEBUG) {
1587                     Slog.v(TAG,
1588                             "Timer ignoring " + jobStatus.toShortString() + " because isTop");
1589                 }
1590                 return;
1591             }
1592             if (DEBUG) {
1593                 Slog.v(TAG, "Starting to track " + jobStatus.toShortString());
1594             }
1595             // Always track jobs, even when charging.
1596             mRunningBgJobs.add(jobStatus);
1597             if (shouldTrackLocked()) {
1598                 mBgJobCount++;
1599                 incrementJobCount(mPkg.userId, mPkg.packageName, 1);
1600                 if (mRunningBgJobs.size() == 1) {
1601                     // Started tracking the first job.
1602                     mStartTimeElapsed = sElapsedRealtimeClock.millis();
1603                     // Starting the timer means that all cached execution stats are now incorrect.
1604                     invalidateAllExecutionStatsLocked(mPkg.userId, mPkg.packageName);
1605                     scheduleCutoff();
1606                 }
1607             }
1608         }
1609 
stopTrackingJob(@onNull JobStatus jobStatus)1610         void stopTrackingJob(@NonNull JobStatus jobStatus) {
1611             if (DEBUG) {
1612                 Slog.v(TAG, "Stopping tracking of " + jobStatus.toShortString());
1613             }
1614             synchronized (mLock) {
1615                 if (mRunningBgJobs.size() == 0) {
1616                     // maybeStopTrackingJobLocked can be called when an app cancels a job, so a
1617                     // timer may not be running when it's asked to stop tracking a job.
1618                     if (DEBUG) {
1619                         Slog.d(TAG, "Timer isn't tracking any jobs but still told to stop");
1620                     }
1621                     return;
1622                 }
1623                 if (mRunningBgJobs.remove(jobStatus)
1624                         && !mChargeTracker.isCharging() && mRunningBgJobs.size() == 0) {
1625                     emitSessionLocked(sElapsedRealtimeClock.millis());
1626                     cancelCutoff();
1627                 }
1628             }
1629         }
1630 
1631         /**
1632          * Stops tracking all jobs and cancels any pending alarms. This should only be called if
1633          * the Timer is not going to be used anymore.
1634          */
dropEverythingLocked()1635         void dropEverythingLocked() {
1636             mRunningBgJobs.clear();
1637             cancelCutoff();
1638         }
1639 
emitSessionLocked(long nowElapsed)1640         private void emitSessionLocked(long nowElapsed) {
1641             if (mBgJobCount <= 0) {
1642                 // Nothing to emit.
1643                 return;
1644             }
1645             TimingSession ts = new TimingSession(mStartTimeElapsed, nowElapsed, mBgJobCount);
1646             saveTimingSession(mPkg.userId, mPkg.packageName, ts);
1647             mBgJobCount = 0;
1648             // Don't reset the tracked jobs list as we need to keep tracking the current number
1649             // of jobs.
1650             // However, cancel the currently scheduled cutoff since it's not currently useful.
1651             cancelCutoff();
1652             incrementTimingSessionCount(mPkg.userId, mPkg.packageName);
1653         }
1654 
1655         /**
1656          * Returns true if the Timer is actively tracking, as opposed to passively ref counting
1657          * during charging.
1658          */
isActive()1659         public boolean isActive() {
1660             synchronized (mLock) {
1661                 return mBgJobCount > 0;
1662             }
1663         }
1664 
isRunning(JobStatus jobStatus)1665         boolean isRunning(JobStatus jobStatus) {
1666             return mRunningBgJobs.contains(jobStatus);
1667         }
1668 
getCurrentDuration(long nowElapsed)1669         long getCurrentDuration(long nowElapsed) {
1670             synchronized (mLock) {
1671                 return !isActive() ? 0 : nowElapsed - mStartTimeElapsed;
1672             }
1673         }
1674 
getBgJobCount()1675         int getBgJobCount() {
1676             synchronized (mLock) {
1677                 return mBgJobCount;
1678             }
1679         }
1680 
shouldTrackLocked()1681         private boolean shouldTrackLocked() {
1682             return !mChargeTracker.isCharging() && !mForegroundUids.get(mUid);
1683         }
1684 
onStateChangedLocked(long nowElapsed, boolean isQuotaFree)1685         void onStateChangedLocked(long nowElapsed, boolean isQuotaFree) {
1686             if (isQuotaFree) {
1687                 emitSessionLocked(nowElapsed);
1688             } else if (!isActive() && shouldTrackLocked()) {
1689                 // Start timing from unplug.
1690                 if (mRunningBgJobs.size() > 0) {
1691                     mStartTimeElapsed = nowElapsed;
1692                     // NOTE: this does have the unfortunate consequence that if the device is
1693                     // repeatedly plugged in and unplugged, or an app changes foreground state
1694                     // very frequently, the job count for a package may be artificially high.
1695                     mBgJobCount = mRunningBgJobs.size();
1696                     incrementJobCount(mPkg.userId, mPkg.packageName, mBgJobCount);
1697                     // Starting the timer means that all cached execution stats are now
1698                     // incorrect.
1699                     invalidateAllExecutionStatsLocked(mPkg.userId, mPkg.packageName);
1700                     // Schedule cutoff since we're now actively tracking for quotas again.
1701                     scheduleCutoff();
1702                 }
1703             }
1704         }
1705 
rescheduleCutoff()1706         void rescheduleCutoff() {
1707             cancelCutoff();
1708             scheduleCutoff();
1709         }
1710 
scheduleCutoff()1711         private void scheduleCutoff() {
1712             // Each package can only be in one standby bucket, so we only need to have one
1713             // message per timer. We only need to reschedule when restarting timer or when
1714             // standby bucket changes.
1715             synchronized (mLock) {
1716                 if (!isActive()) {
1717                     return;
1718                 }
1719                 Message msg = mHandler.obtainMessage(MSG_REACHED_QUOTA, mPkg);
1720                 final long timeRemainingMs = getTimeUntilQuotaConsumedLocked(mPkg.userId,
1721                         mPkg.packageName);
1722                 if (DEBUG) {
1723                     Slog.i(TAG, "Job for " + mPkg + " has " + timeRemainingMs + "ms left.");
1724                 }
1725                 // If the job was running the entire time, then the system would be up, so it's
1726                 // fine to use uptime millis for these messages.
1727                 mHandler.sendMessageDelayed(msg, timeRemainingMs);
1728             }
1729         }
1730 
cancelCutoff()1731         private void cancelCutoff() {
1732             mHandler.removeMessages(MSG_REACHED_QUOTA, mPkg);
1733         }
1734 
dump(IndentingPrintWriter pw, Predicate<JobStatus> predicate)1735         public void dump(IndentingPrintWriter pw, Predicate<JobStatus> predicate) {
1736             pw.print("Timer{");
1737             pw.print(mPkg);
1738             pw.print("} ");
1739             if (isActive()) {
1740                 pw.print("started at ");
1741                 pw.print(mStartTimeElapsed);
1742                 pw.print(" (");
1743                 pw.print(sElapsedRealtimeClock.millis() - mStartTimeElapsed);
1744                 pw.print("ms ago)");
1745             } else {
1746                 pw.print("NOT active");
1747             }
1748             pw.print(", ");
1749             pw.print(mBgJobCount);
1750             pw.print(" running bg jobs");
1751             pw.println();
1752             pw.increaseIndent();
1753             for (int i = 0; i < mRunningBgJobs.size(); i++) {
1754                 JobStatus js = mRunningBgJobs.valueAt(i);
1755                 if (predicate.test(js)) {
1756                     pw.println(js.toShortString());
1757                 }
1758             }
1759             pw.decreaseIndent();
1760         }
1761 
dump(ProtoOutputStream proto, long fieldId, Predicate<JobStatus> predicate)1762         public void dump(ProtoOutputStream proto, long fieldId, Predicate<JobStatus> predicate) {
1763             final long token = proto.start(fieldId);
1764 
1765             mPkg.writeToProto(proto, StateControllerProto.QuotaController.Timer.PKG);
1766             proto.write(StateControllerProto.QuotaController.Timer.IS_ACTIVE, isActive());
1767             proto.write(StateControllerProto.QuotaController.Timer.START_TIME_ELAPSED,
1768                     mStartTimeElapsed);
1769             proto.write(StateControllerProto.QuotaController.Timer.BG_JOB_COUNT, mBgJobCount);
1770             for (int i = 0; i < mRunningBgJobs.size(); i++) {
1771                 JobStatus js = mRunningBgJobs.valueAt(i);
1772                 if (predicate.test(js)) {
1773                     js.writeToShortProto(proto,
1774                             StateControllerProto.QuotaController.Timer.RUNNING_JOBS);
1775                 }
1776             }
1777 
1778             proto.end(token);
1779         }
1780     }
1781 
1782     /**
1783      * Tracking of app assignments to standby buckets
1784      */
1785     final class StandbyTracker extends AppIdleStateChangeListener {
1786 
1787         @Override
onAppIdleStateChanged(final String packageName, final @UserIdInt int userId, boolean idle, int bucket, int reason)1788         public void onAppIdleStateChanged(final String packageName, final @UserIdInt int userId,
1789                 boolean idle, int bucket, int reason) {
1790             // Update job bookkeeping out of band.
1791             BackgroundThread.getHandler().post(() -> {
1792                 final int bucketIndex = JobSchedulerService.standbyBucketToBucketIndex(bucket);
1793                 if (DEBUG) {
1794                     Slog.i(TAG, "Moving pkg " + string(userId, packageName) + " to bucketIndex "
1795                             + bucketIndex);
1796                 }
1797                 synchronized (mLock) {
1798                     ArraySet<JobStatus> jobs = mTrackedJobs.get(userId, packageName);
1799                     if (jobs == null || jobs.size() == 0) {
1800                         return;
1801                     }
1802                     for (int i = jobs.size() - 1; i >= 0; i--) {
1803                         JobStatus js = jobs.valueAt(i);
1804                         js.setStandbyBucket(bucketIndex);
1805                     }
1806                     Timer timer = mPkgTimers.get(userId, packageName);
1807                     if (timer != null && timer.isActive()) {
1808                         timer.rescheduleCutoff();
1809                     }
1810                     if (!mShouldThrottle || maybeUpdateConstraintForPkgLocked(userId,
1811                             packageName)) {
1812                         mStateChangedListener.onControllerStateChanged();
1813                     }
1814                 }
1815             });
1816         }
1817 
1818         @Override
onParoleStateChanged(final boolean isParoleOn)1819         public void onParoleStateChanged(final boolean isParoleOn) {
1820             mInParole = isParoleOn;
1821             if (DEBUG) {
1822                 Slog.i(TAG, "Global parole state now " + (isParoleOn ? "ON" : "OFF"));
1823             }
1824             // Update job bookkeeping out of band.
1825             BackgroundThread.getHandler().post(() -> {
1826                 synchronized (mLock) {
1827                     maybeUpdateAllConstraintsLocked();
1828                 }
1829             });
1830         }
1831     }
1832 
1833     private final class DeleteTimingSessionsFunctor implements Consumer<List<TimingSession>> {
1834         private final Predicate<TimingSession> mTooOld = new Predicate<TimingSession>() {
1835             public boolean test(TimingSession ts) {
1836                 return ts.endTimeElapsed <= sElapsedRealtimeClock.millis() - MAX_PERIOD_MS;
1837             }
1838         };
1839 
1840         @Override
accept(List<TimingSession> sessions)1841         public void accept(List<TimingSession> sessions) {
1842             if (sessions != null) {
1843                 // Remove everything older than MAX_PERIOD_MS time ago.
1844                 sessions.removeIf(mTooOld);
1845             }
1846         }
1847     }
1848 
1849     private final DeleteTimingSessionsFunctor mDeleteOldSessionsFunctor =
1850             new DeleteTimingSessionsFunctor();
1851 
1852     @VisibleForTesting
deleteObsoleteSessionsLocked()1853     void deleteObsoleteSessionsLocked() {
1854         mTimingSessions.forEach(mDeleteOldSessionsFunctor);
1855     }
1856 
1857     private class QcHandler extends Handler {
QcHandler(Looper looper)1858         QcHandler(Looper looper) {
1859             super(looper);
1860         }
1861 
1862         @Override
handleMessage(Message msg)1863         public void handleMessage(Message msg) {
1864             synchronized (mLock) {
1865                 switch (msg.what) {
1866                     case MSG_REACHED_QUOTA: {
1867                         Package pkg = (Package) msg.obj;
1868                         if (DEBUG) {
1869                             Slog.d(TAG, "Checking if " + pkg + " has reached its quota.");
1870                         }
1871 
1872                         long timeRemainingMs = getRemainingExecutionTimeLocked(pkg.userId,
1873                                 pkg.packageName);
1874                         if (timeRemainingMs <= 50) {
1875                             // Less than 50 milliseconds left. Start process of shutting down jobs.
1876                             if (DEBUG) Slog.d(TAG, pkg + " has reached its quota.");
1877                             if (maybeUpdateConstraintForPkgLocked(pkg.userId, pkg.packageName)) {
1878                                 mStateChangedListener.onControllerStateChanged();
1879                             }
1880                         } else {
1881                             // This could potentially happen if an old session phases out while a
1882                             // job is currently running.
1883                             // Reschedule message
1884                             Message rescheduleMsg = obtainMessage(MSG_REACHED_QUOTA, pkg);
1885                             timeRemainingMs = getTimeUntilQuotaConsumedLocked(pkg.userId,
1886                                     pkg.packageName);
1887                             if (DEBUG) {
1888                                 Slog.d(TAG, pkg + " has " + timeRemainingMs + "ms left.");
1889                             }
1890                             sendMessageDelayed(rescheduleMsg, timeRemainingMs);
1891                         }
1892                         break;
1893                     }
1894                     case MSG_CLEAN_UP_SESSIONS:
1895                         if (DEBUG) {
1896                             Slog.d(TAG, "Cleaning up timing sessions.");
1897                         }
1898                         deleteObsoleteSessionsLocked();
1899                         maybeScheduleCleanupAlarmLocked();
1900 
1901                         break;
1902                     case MSG_CHECK_PACKAGE: {
1903                         String packageName = (String) msg.obj;
1904                         int userId = msg.arg1;
1905                         if (DEBUG) {
1906                             Slog.d(TAG, "Checking pkg " + string(userId, packageName));
1907                         }
1908                         if (maybeUpdateConstraintForPkgLocked(userId, packageName)) {
1909                             mStateChangedListener.onControllerStateChanged();
1910                         }
1911                         break;
1912                     }
1913                     case MSG_UID_PROCESS_STATE_CHANGED: {
1914                         final int uid = msg.arg1;
1915                         final int procState = msg.arg2;
1916                         final int userId = UserHandle.getUserId(uid);
1917                         final long nowElapsed = sElapsedRealtimeClock.millis();
1918 
1919                         synchronized (mLock) {
1920                             boolean isQuotaFree;
1921                             if (procState <= ActivityManager.PROCESS_STATE_FOREGROUND_SERVICE) {
1922                                 mForegroundUids.put(uid, true);
1923                                 isQuotaFree = true;
1924                             } else {
1925                                 mForegroundUids.delete(uid);
1926                                 isQuotaFree = false;
1927                             }
1928                             // Update Timers first.
1929                             if (mPkgTimers.indexOfKey(userId) >= 0) {
1930                                 ArraySet<String> packages = mUidToPackageCache.get(uid);
1931                                 if (packages == null) {
1932                                     try {
1933                                         String[] pkgs = AppGlobals.getPackageManager()
1934                                                 .getPackagesForUid(uid);
1935                                         if (pkgs != null) {
1936                                             for (String pkg : pkgs) {
1937                                                 mUidToPackageCache.add(uid, pkg);
1938                                             }
1939                                             packages = mUidToPackageCache.get(uid);
1940                                         }
1941                                     } catch (RemoteException e) {
1942                                         Slog.wtf(TAG, "Failed to get package list", e);
1943                                     }
1944                                 }
1945                                 if (packages != null) {
1946                                     for (int i = packages.size() - 1; i >= 0; --i) {
1947                                         Timer t = mPkgTimers.get(userId, packages.valueAt(i));
1948                                         if (t != null) {
1949                                             t.onStateChangedLocked(nowElapsed, isQuotaFree);
1950                                         }
1951                                     }
1952                                 }
1953                             }
1954                             if (maybeUpdateConstraintForUidLocked(uid)) {
1955                                 mStateChangedListener.onControllerStateChanged();
1956                             }
1957                         }
1958                         break;
1959                     }
1960                 }
1961             }
1962         }
1963     }
1964 
1965     private class QcAlarmListener implements AlarmManager.OnAlarmListener {
1966         private final int mUserId;
1967         private final String mPackageName;
1968         private volatile long mTriggerTimeElapsed;
1969 
QcAlarmListener(int userId, String packageName)1970         QcAlarmListener(int userId, String packageName) {
1971             mUserId = userId;
1972             mPackageName = packageName;
1973         }
1974 
isWaiting()1975         boolean isWaiting() {
1976             return mTriggerTimeElapsed > 0;
1977         }
1978 
setTriggerTime(long timeElapsed)1979         void setTriggerTime(long timeElapsed) {
1980             mTriggerTimeElapsed = timeElapsed;
1981         }
1982 
getTriggerTimeElapsed()1983         long getTriggerTimeElapsed() {
1984             return mTriggerTimeElapsed;
1985         }
1986 
1987         @Override
onAlarm()1988         public void onAlarm() {
1989             mHandler.obtainMessage(MSG_CHECK_PACKAGE, mUserId, 0, mPackageName).sendToTarget();
1990             mTriggerTimeElapsed = 0;
1991         }
1992     }
1993 
1994     @VisibleForTesting
1995     class QcConstants extends ContentObserver {
1996         private ContentResolver mResolver;
1997         private final KeyValueListParser mParser = new KeyValueListParser(',');
1998 
1999         private static final String KEY_ALLOWED_TIME_PER_PERIOD_MS = "allowed_time_per_period_ms";
2000         private static final String KEY_IN_QUOTA_BUFFER_MS = "in_quota_buffer_ms";
2001         private static final String KEY_WINDOW_SIZE_ACTIVE_MS = "window_size_active_ms";
2002         private static final String KEY_WINDOW_SIZE_WORKING_MS = "window_size_working_ms";
2003         private static final String KEY_WINDOW_SIZE_FREQUENT_MS = "window_size_frequent_ms";
2004         private static final String KEY_WINDOW_SIZE_RARE_MS = "window_size_rare_ms";
2005         private static final String KEY_MAX_EXECUTION_TIME_MS = "max_execution_time_ms";
2006         private static final String KEY_MAX_JOB_COUNT_ACTIVE = "max_job_count_active";
2007         private static final String KEY_MAX_JOB_COUNT_WORKING = "max_job_count_working";
2008         private static final String KEY_MAX_JOB_COUNT_FREQUENT = "max_job_count_frequent";
2009         private static final String KEY_MAX_JOB_COUNT_RARE = "max_job_count_rare";
2010         private static final String KEY_RATE_LIMITING_WINDOW_MS = "rate_limiting_window_ms";
2011         private static final String KEY_MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW =
2012                 "max_job_count_per_rate_limiting_window";
2013         private static final String KEY_MAX_SESSION_COUNT_ACTIVE = "max_session_count_active";
2014         private static final String KEY_MAX_SESSION_COUNT_WORKING = "max_session_count_working";
2015         private static final String KEY_MAX_SESSION_COUNT_FREQUENT = "max_session_count_frequent";
2016         private static final String KEY_MAX_SESSION_COUNT_RARE = "max_session_count_rare";
2017         private static final String KEY_MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW =
2018                 "max_session_count_per_rate_limiting_window";
2019         private static final String KEY_TIMING_SESSION_COALESCING_DURATION_MS =
2020                 "timing_session_coalescing_duration_ms";
2021 
2022         private static final long DEFAULT_ALLOWED_TIME_PER_PERIOD_MS =
2023                 10 * 60 * 1000L; // 10 minutes
2024         private static final long DEFAULT_IN_QUOTA_BUFFER_MS =
2025                 30 * 1000L; // 30 seconds
2026         private static final long DEFAULT_WINDOW_SIZE_ACTIVE_MS =
2027                 DEFAULT_ALLOWED_TIME_PER_PERIOD_MS; // ACTIVE apps can run jobs at any time
2028         private static final long DEFAULT_WINDOW_SIZE_WORKING_MS =
2029                 2 * 60 * 60 * 1000L; // 2 hours
2030         private static final long DEFAULT_WINDOW_SIZE_FREQUENT_MS =
2031                 8 * 60 * 60 * 1000L; // 8 hours
2032         private static final long DEFAULT_WINDOW_SIZE_RARE_MS =
2033                 24 * 60 * 60 * 1000L; // 24 hours
2034         private static final long DEFAULT_MAX_EXECUTION_TIME_MS =
2035                 4 * HOUR_IN_MILLIS;
2036         private static final long DEFAULT_RATE_LIMITING_WINDOW_MS =
2037                 10 * MINUTE_IN_MILLIS;
2038         private static final int DEFAULT_MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW = 20;
2039         private static final int DEFAULT_MAX_JOB_COUNT_ACTIVE = // 20/window = 120/hr = 1/session
2040                 DEFAULT_MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW;
2041         private static final int DEFAULT_MAX_JOB_COUNT_WORKING = // 120/window = 60/hr = 12/session
2042                 (int) (60.0 * DEFAULT_WINDOW_SIZE_WORKING_MS / HOUR_IN_MILLIS);
2043         private static final int DEFAULT_MAX_JOB_COUNT_FREQUENT = // 200/window = 25/hr = 25/session
2044                 (int) (25.0 * DEFAULT_WINDOW_SIZE_FREQUENT_MS / HOUR_IN_MILLIS);
2045         private static final int DEFAULT_MAX_JOB_COUNT_RARE = // 48/window = 2/hr = 16/session
2046                 (int) (2.0 * DEFAULT_WINDOW_SIZE_RARE_MS / HOUR_IN_MILLIS);
2047         private static final int DEFAULT_MAX_SESSION_COUNT_ACTIVE =
2048                 20; // 120/hr
2049         private static final int DEFAULT_MAX_SESSION_COUNT_WORKING =
2050                 10; // 5/hr
2051         private static final int DEFAULT_MAX_SESSION_COUNT_FREQUENT =
2052                 8; // 1/hr
2053         private static final int DEFAULT_MAX_SESSION_COUNT_RARE =
2054                 3; // .125/hr
2055         private static final int DEFAULT_MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW = 20;
2056         private static final long DEFAULT_TIMING_SESSION_COALESCING_DURATION_MS = 5000; // 5 seconds
2057 
2058         /** How much time each app will have to run jobs within their standby bucket window. */
2059         public long ALLOWED_TIME_PER_PERIOD_MS = DEFAULT_ALLOWED_TIME_PER_PERIOD_MS;
2060 
2061         /**
2062          * How much time the package should have before transitioning from out-of-quota to in-quota.
2063          * This should not affect processing if the package is already in-quota.
2064          */
2065         public long IN_QUOTA_BUFFER_MS = DEFAULT_IN_QUOTA_BUFFER_MS;
2066 
2067         /**
2068          * The quota window size of the particular standby bucket. Apps in this standby bucket are
2069          * expected to run only {@link #ALLOWED_TIME_PER_PERIOD_MS} within the past
2070          * WINDOW_SIZE_MS.
2071          */
2072         public long WINDOW_SIZE_ACTIVE_MS = DEFAULT_WINDOW_SIZE_ACTIVE_MS;
2073 
2074         /**
2075          * The quota window size of the particular standby bucket. Apps in this standby bucket are
2076          * expected to run only {@link #ALLOWED_TIME_PER_PERIOD_MS} within the past
2077          * WINDOW_SIZE_MS.
2078          */
2079         public long WINDOW_SIZE_WORKING_MS = DEFAULT_WINDOW_SIZE_WORKING_MS;
2080 
2081         /**
2082          * The quota window size of the particular standby bucket. Apps in this standby bucket are
2083          * expected to run only {@link #ALLOWED_TIME_PER_PERIOD_MS} within the past
2084          * WINDOW_SIZE_MS.
2085          */
2086         public long WINDOW_SIZE_FREQUENT_MS = DEFAULT_WINDOW_SIZE_FREQUENT_MS;
2087 
2088         /**
2089          * The quota window size of the particular standby bucket. Apps in this standby bucket are
2090          * expected to run only {@link #ALLOWED_TIME_PER_PERIOD_MS} within the past
2091          * WINDOW_SIZE_MS.
2092          */
2093         public long WINDOW_SIZE_RARE_MS = DEFAULT_WINDOW_SIZE_RARE_MS;
2094 
2095         /**
2096          * The maximum amount of time an app can have its jobs running within a 24 hour window.
2097          */
2098         public long MAX_EXECUTION_TIME_MS = DEFAULT_MAX_EXECUTION_TIME_MS;
2099 
2100         /**
2101          * The maximum number of jobs an app can run within this particular standby bucket's
2102          * window size.
2103          */
2104         public int MAX_JOB_COUNT_ACTIVE = DEFAULT_MAX_JOB_COUNT_ACTIVE;
2105 
2106         /**
2107          * The maximum number of jobs an app can run within this particular standby bucket's
2108          * window size.
2109          */
2110         public int MAX_JOB_COUNT_WORKING = DEFAULT_MAX_JOB_COUNT_WORKING;
2111 
2112         /**
2113          * The maximum number of jobs an app can run within this particular standby bucket's
2114          * window size.
2115          */
2116         public int MAX_JOB_COUNT_FREQUENT = DEFAULT_MAX_JOB_COUNT_FREQUENT;
2117 
2118         /**
2119          * The maximum number of jobs an app can run within this particular standby bucket's
2120          * window size.
2121          */
2122         public int MAX_JOB_COUNT_RARE = DEFAULT_MAX_JOB_COUNT_RARE;
2123 
2124         /** The period of time used to rate limit recently run jobs. */
2125         public long RATE_LIMITING_WINDOW_MS = DEFAULT_RATE_LIMITING_WINDOW_MS;
2126 
2127         /**
2128          * The maximum number of jobs that can run within the past {@link #RATE_LIMITING_WINDOW_MS}.
2129          */
2130         public int MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW =
2131                 DEFAULT_MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW;
2132 
2133         /**
2134          * The maximum number of {@link TimingSession}s an app can run within this particular
2135          * standby bucket's window size.
2136          */
2137         public int MAX_SESSION_COUNT_ACTIVE = DEFAULT_MAX_SESSION_COUNT_ACTIVE;
2138 
2139         /**
2140          * The maximum number of {@link TimingSession}s an app can run within this particular
2141          * standby bucket's window size.
2142          */
2143         public int MAX_SESSION_COUNT_WORKING = DEFAULT_MAX_SESSION_COUNT_WORKING;
2144 
2145         /**
2146          * The maximum number of {@link TimingSession}s an app can run within this particular
2147          * standby bucket's window size.
2148          */
2149         public int MAX_SESSION_COUNT_FREQUENT = DEFAULT_MAX_SESSION_COUNT_FREQUENT;
2150 
2151         /**
2152          * The maximum number of {@link TimingSession}s an app can run within this particular
2153          * standby bucket's window size.
2154          */
2155         public int MAX_SESSION_COUNT_RARE = DEFAULT_MAX_SESSION_COUNT_RARE;
2156 
2157         /**
2158          * The maximum number of {@link TimingSession}s that can run within the past
2159          * {@link #ALLOWED_TIME_PER_PERIOD_MS}.
2160          */
2161         public int MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW =
2162                 DEFAULT_MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW;
2163 
2164         /**
2165          * Treat two distinct {@link TimingSession}s as the same if they start and end within this
2166          * amount of time of each other.
2167          */
2168         public long TIMING_SESSION_COALESCING_DURATION_MS =
2169                 DEFAULT_TIMING_SESSION_COALESCING_DURATION_MS;
2170 
2171         // Safeguards
2172 
2173         /** The minimum number of jobs that any bucket will be allowed to run within its window. */
2174         private static final int MIN_BUCKET_JOB_COUNT = 10;
2175 
2176         /**
2177          * The minimum number of {@link TimingSession}s that any bucket will be allowed to run
2178          * within its window.
2179          */
2180         private static final int MIN_BUCKET_SESSION_COUNT = 1;
2181 
2182         /** The minimum value that {@link #MAX_EXECUTION_TIME_MS} can have. */
2183         private static final long MIN_MAX_EXECUTION_TIME_MS = 60 * MINUTE_IN_MILLIS;
2184 
2185         /** The minimum value that {@link #MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW} can have. */
2186         private static final int MIN_MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW = 10;
2187 
2188         /** The minimum value that {@link #MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW} can have. */
2189         private static final int MIN_MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW = 10;
2190 
2191         /** The minimum value that {@link #RATE_LIMITING_WINDOW_MS} can have. */
2192         private static final long MIN_RATE_LIMITING_WINDOW_MS = 30 * SECOND_IN_MILLIS;
2193 
QcConstants(Handler handler)2194         QcConstants(Handler handler) {
2195             super(handler);
2196         }
2197 
start(ContentResolver resolver)2198         private void start(ContentResolver resolver) {
2199             mResolver = resolver;
2200             mResolver.registerContentObserver(Settings.Global.getUriFor(
2201                     Settings.Global.JOB_SCHEDULER_QUOTA_CONTROLLER_CONSTANTS), false, this);
2202             updateConstants();
2203         }
2204 
2205         @Override
onChange(boolean selfChange, Uri uri)2206         public void onChange(boolean selfChange, Uri uri) {
2207             final String constants = Settings.Global.getString(
2208                     mResolver, Settings.Global.JOB_SCHEDULER_QUOTA_CONTROLLER_CONSTANTS);
2209 
2210             try {
2211                 mParser.setString(constants);
2212             } catch (Exception e) {
2213                 // Failed to parse the settings string, log this and move on with defaults.
2214                 Slog.e(TAG, "Bad jobscheduler quota controller settings", e);
2215             }
2216 
2217             ALLOWED_TIME_PER_PERIOD_MS = mParser.getDurationMillis(
2218                     KEY_ALLOWED_TIME_PER_PERIOD_MS, DEFAULT_ALLOWED_TIME_PER_PERIOD_MS);
2219             IN_QUOTA_BUFFER_MS = mParser.getDurationMillis(
2220                     KEY_IN_QUOTA_BUFFER_MS, DEFAULT_IN_QUOTA_BUFFER_MS);
2221             WINDOW_SIZE_ACTIVE_MS = mParser.getDurationMillis(
2222                     KEY_WINDOW_SIZE_ACTIVE_MS, DEFAULT_WINDOW_SIZE_ACTIVE_MS);
2223             WINDOW_SIZE_WORKING_MS = mParser.getDurationMillis(
2224                     KEY_WINDOW_SIZE_WORKING_MS, DEFAULT_WINDOW_SIZE_WORKING_MS);
2225             WINDOW_SIZE_FREQUENT_MS = mParser.getDurationMillis(
2226                     KEY_WINDOW_SIZE_FREQUENT_MS, DEFAULT_WINDOW_SIZE_FREQUENT_MS);
2227             WINDOW_SIZE_RARE_MS = mParser.getDurationMillis(
2228                     KEY_WINDOW_SIZE_RARE_MS, DEFAULT_WINDOW_SIZE_RARE_MS);
2229             MAX_EXECUTION_TIME_MS = mParser.getDurationMillis(
2230                     KEY_MAX_EXECUTION_TIME_MS, DEFAULT_MAX_EXECUTION_TIME_MS);
2231             MAX_JOB_COUNT_ACTIVE = mParser.getInt(
2232                     KEY_MAX_JOB_COUNT_ACTIVE, DEFAULT_MAX_JOB_COUNT_ACTIVE);
2233             MAX_JOB_COUNT_WORKING = mParser.getInt(
2234                     KEY_MAX_JOB_COUNT_WORKING, DEFAULT_MAX_JOB_COUNT_WORKING);
2235             MAX_JOB_COUNT_FREQUENT = mParser.getInt(
2236                     KEY_MAX_JOB_COUNT_FREQUENT, DEFAULT_MAX_JOB_COUNT_FREQUENT);
2237             MAX_JOB_COUNT_RARE = mParser.getInt(
2238                     KEY_MAX_JOB_COUNT_RARE, DEFAULT_MAX_JOB_COUNT_RARE);
2239             RATE_LIMITING_WINDOW_MS = mParser.getLong(
2240                     KEY_RATE_LIMITING_WINDOW_MS, DEFAULT_RATE_LIMITING_WINDOW_MS);
2241             MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW = mParser.getInt(
2242                     KEY_MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW,
2243                     DEFAULT_MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW);
2244             MAX_SESSION_COUNT_ACTIVE = mParser.getInt(
2245                     KEY_MAX_SESSION_COUNT_ACTIVE, DEFAULT_MAX_SESSION_COUNT_ACTIVE);
2246             MAX_SESSION_COUNT_WORKING = mParser.getInt(
2247                     KEY_MAX_SESSION_COUNT_WORKING, DEFAULT_MAX_SESSION_COUNT_WORKING);
2248             MAX_SESSION_COUNT_FREQUENT = mParser.getInt(
2249                     KEY_MAX_SESSION_COUNT_FREQUENT, DEFAULT_MAX_SESSION_COUNT_FREQUENT);
2250             MAX_SESSION_COUNT_RARE = mParser.getInt(
2251                     KEY_MAX_SESSION_COUNT_RARE, DEFAULT_MAX_SESSION_COUNT_RARE);
2252             MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW = mParser.getInt(
2253                     KEY_MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW,
2254                     DEFAULT_MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW);
2255             TIMING_SESSION_COALESCING_DURATION_MS = mParser.getLong(
2256                     KEY_TIMING_SESSION_COALESCING_DURATION_MS,
2257                     DEFAULT_TIMING_SESSION_COALESCING_DURATION_MS);
2258 
2259             updateConstants();
2260         }
2261 
2262         @VisibleForTesting
updateConstants()2263         void updateConstants() {
2264             synchronized (mLock) {
2265                 boolean changed = false;
2266 
2267                 long newMaxExecutionTimeMs = Math.max(MIN_MAX_EXECUTION_TIME_MS,
2268                         Math.min(MAX_PERIOD_MS, MAX_EXECUTION_TIME_MS));
2269                 if (mMaxExecutionTimeMs != newMaxExecutionTimeMs) {
2270                     mMaxExecutionTimeMs = newMaxExecutionTimeMs;
2271                     mMaxExecutionTimeIntoQuotaMs = mMaxExecutionTimeMs - mQuotaBufferMs;
2272                     changed = true;
2273                 }
2274                 long newAllowedTimeMs = Math.min(mMaxExecutionTimeMs,
2275                         Math.max(MINUTE_IN_MILLIS, ALLOWED_TIME_PER_PERIOD_MS));
2276                 if (mAllowedTimePerPeriodMs != newAllowedTimeMs) {
2277                     mAllowedTimePerPeriodMs = newAllowedTimeMs;
2278                     mAllowedTimeIntoQuotaMs = mAllowedTimePerPeriodMs - mQuotaBufferMs;
2279                     changed = true;
2280                 }
2281                 long newQuotaBufferMs = Math.max(0,
2282                         Math.min(5 * MINUTE_IN_MILLIS, IN_QUOTA_BUFFER_MS));
2283                 if (mQuotaBufferMs != newQuotaBufferMs) {
2284                     mQuotaBufferMs = newQuotaBufferMs;
2285                     mAllowedTimeIntoQuotaMs = mAllowedTimePerPeriodMs - mQuotaBufferMs;
2286                     mMaxExecutionTimeIntoQuotaMs = mMaxExecutionTimeMs - mQuotaBufferMs;
2287                     changed = true;
2288                 }
2289                 long newActivePeriodMs = Math.max(mAllowedTimePerPeriodMs,
2290                         Math.min(MAX_PERIOD_MS, WINDOW_SIZE_ACTIVE_MS));
2291                 if (mBucketPeriodsMs[ACTIVE_INDEX] != newActivePeriodMs) {
2292                     mBucketPeriodsMs[ACTIVE_INDEX] = newActivePeriodMs;
2293                     changed = true;
2294                 }
2295                 long newWorkingPeriodMs = Math.max(mAllowedTimePerPeriodMs,
2296                         Math.min(MAX_PERIOD_MS, WINDOW_SIZE_WORKING_MS));
2297                 if (mBucketPeriodsMs[WORKING_INDEX] != newWorkingPeriodMs) {
2298                     mBucketPeriodsMs[WORKING_INDEX] = newWorkingPeriodMs;
2299                     changed = true;
2300                 }
2301                 long newFrequentPeriodMs = Math.max(mAllowedTimePerPeriodMs,
2302                         Math.min(MAX_PERIOD_MS, WINDOW_SIZE_FREQUENT_MS));
2303                 if (mBucketPeriodsMs[FREQUENT_INDEX] != newFrequentPeriodMs) {
2304                     mBucketPeriodsMs[FREQUENT_INDEX] = newFrequentPeriodMs;
2305                     changed = true;
2306                 }
2307                 long newRarePeriodMs = Math.max(mAllowedTimePerPeriodMs,
2308                         Math.min(MAX_PERIOD_MS, WINDOW_SIZE_RARE_MS));
2309                 if (mBucketPeriodsMs[RARE_INDEX] != newRarePeriodMs) {
2310                     mBucketPeriodsMs[RARE_INDEX] = newRarePeriodMs;
2311                     changed = true;
2312                 }
2313                 long newRateLimitingWindowMs = Math.min(MAX_PERIOD_MS,
2314                         Math.max(MIN_RATE_LIMITING_WINDOW_MS, RATE_LIMITING_WINDOW_MS));
2315                 if (mRateLimitingWindowMs != newRateLimitingWindowMs) {
2316                     mRateLimitingWindowMs = newRateLimitingWindowMs;
2317                     changed = true;
2318                 }
2319                 int newMaxJobCountPerRateLimitingWindow = Math.max(
2320                         MIN_MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW,
2321                         MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW);
2322                 if (mMaxJobCountPerRateLimitingWindow != newMaxJobCountPerRateLimitingWindow) {
2323                     mMaxJobCountPerRateLimitingWindow = newMaxJobCountPerRateLimitingWindow;
2324                     changed = true;
2325                 }
2326                 int newActiveMaxJobCount = Math.max(MIN_BUCKET_JOB_COUNT, MAX_JOB_COUNT_ACTIVE);
2327                 if (mMaxBucketJobCounts[ACTIVE_INDEX] != newActiveMaxJobCount) {
2328                     mMaxBucketJobCounts[ACTIVE_INDEX] = newActiveMaxJobCount;
2329                     changed = true;
2330                 }
2331                 int newWorkingMaxJobCount = Math.max(MIN_BUCKET_JOB_COUNT, MAX_JOB_COUNT_WORKING);
2332                 if (mMaxBucketJobCounts[WORKING_INDEX] != newWorkingMaxJobCount) {
2333                     mMaxBucketJobCounts[WORKING_INDEX] = newWorkingMaxJobCount;
2334                     changed = true;
2335                 }
2336                 int newFrequentMaxJobCount = Math.max(MIN_BUCKET_JOB_COUNT, MAX_JOB_COUNT_FREQUENT);
2337                 if (mMaxBucketJobCounts[FREQUENT_INDEX] != newFrequentMaxJobCount) {
2338                     mMaxBucketJobCounts[FREQUENT_INDEX] = newFrequentMaxJobCount;
2339                     changed = true;
2340                 }
2341                 int newRareMaxJobCount = Math.max(MIN_BUCKET_JOB_COUNT, MAX_JOB_COUNT_RARE);
2342                 if (mMaxBucketJobCounts[RARE_INDEX] != newRareMaxJobCount) {
2343                     mMaxBucketJobCounts[RARE_INDEX] = newRareMaxJobCount;
2344                     changed = true;
2345                 }
2346                 int newMaxSessionCountPerRateLimitPeriod = Math.max(
2347                         MIN_MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW,
2348                         MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW);
2349                 if (mMaxSessionCountPerRateLimitingWindow != newMaxSessionCountPerRateLimitPeriod) {
2350                     mMaxSessionCountPerRateLimitingWindow = newMaxSessionCountPerRateLimitPeriod;
2351                     changed = true;
2352                 }
2353                 int newActiveMaxSessionCount =
2354                         Math.max(MIN_BUCKET_SESSION_COUNT, MAX_SESSION_COUNT_ACTIVE);
2355                 if (mMaxBucketSessionCounts[ACTIVE_INDEX] != newActiveMaxSessionCount) {
2356                     mMaxBucketSessionCounts[ACTIVE_INDEX] = newActiveMaxSessionCount;
2357                     changed = true;
2358                 }
2359                 int newWorkingMaxSessionCount =
2360                         Math.max(MIN_BUCKET_SESSION_COUNT, MAX_SESSION_COUNT_WORKING);
2361                 if (mMaxBucketSessionCounts[WORKING_INDEX] != newWorkingMaxSessionCount) {
2362                     mMaxBucketSessionCounts[WORKING_INDEX] = newWorkingMaxSessionCount;
2363                     changed = true;
2364                 }
2365                 int newFrequentMaxSessionCount =
2366                         Math.max(MIN_BUCKET_SESSION_COUNT, MAX_SESSION_COUNT_FREQUENT);
2367                 if (mMaxBucketSessionCounts[FREQUENT_INDEX] != newFrequentMaxSessionCount) {
2368                     mMaxBucketSessionCounts[FREQUENT_INDEX] = newFrequentMaxSessionCount;
2369                     changed = true;
2370                 }
2371                 int newRareMaxSessionCount =
2372                         Math.max(MIN_BUCKET_SESSION_COUNT, MAX_SESSION_COUNT_RARE);
2373                 if (mMaxBucketSessionCounts[RARE_INDEX] != newRareMaxSessionCount) {
2374                     mMaxBucketSessionCounts[RARE_INDEX] = newRareMaxSessionCount;
2375                     changed = true;
2376                 }
2377                 long newSessionCoalescingDurationMs = Math.min(15 * MINUTE_IN_MILLIS,
2378                         Math.max(0, TIMING_SESSION_COALESCING_DURATION_MS));
2379                 if (mTimingSessionCoalescingDurationMs != newSessionCoalescingDurationMs) {
2380                     mTimingSessionCoalescingDurationMs = newSessionCoalescingDurationMs;
2381                     changed = true;
2382                 }
2383 
2384                 if (changed && mShouldThrottle) {
2385                     // Update job bookkeeping out of band.
2386                     BackgroundThread.getHandler().post(() -> {
2387                         synchronized (mLock) {
2388                             invalidateAllExecutionStatsLocked();
2389                             maybeUpdateAllConstraintsLocked();
2390                         }
2391                     });
2392                 }
2393             }
2394         }
2395 
dump(IndentingPrintWriter pw)2396         private void dump(IndentingPrintWriter pw) {
2397             pw.println();
2398             pw.println("QuotaController:");
2399             pw.increaseIndent();
2400             pw.printPair(KEY_ALLOWED_TIME_PER_PERIOD_MS, ALLOWED_TIME_PER_PERIOD_MS).println();
2401             pw.printPair(KEY_IN_QUOTA_BUFFER_MS, IN_QUOTA_BUFFER_MS).println();
2402             pw.printPair(KEY_WINDOW_SIZE_ACTIVE_MS, WINDOW_SIZE_ACTIVE_MS).println();
2403             pw.printPair(KEY_WINDOW_SIZE_WORKING_MS, WINDOW_SIZE_WORKING_MS).println();
2404             pw.printPair(KEY_WINDOW_SIZE_FREQUENT_MS, WINDOW_SIZE_FREQUENT_MS).println();
2405             pw.printPair(KEY_WINDOW_SIZE_RARE_MS, WINDOW_SIZE_RARE_MS).println();
2406             pw.printPair(KEY_MAX_EXECUTION_TIME_MS, MAX_EXECUTION_TIME_MS).println();
2407             pw.printPair(KEY_MAX_JOB_COUNT_ACTIVE, MAX_JOB_COUNT_ACTIVE).println();
2408             pw.printPair(KEY_MAX_JOB_COUNT_WORKING, MAX_JOB_COUNT_WORKING).println();
2409             pw.printPair(KEY_MAX_JOB_COUNT_FREQUENT, MAX_JOB_COUNT_FREQUENT).println();
2410             pw.printPair(KEY_MAX_JOB_COUNT_RARE, MAX_JOB_COUNT_RARE).println();
2411             pw.printPair(KEY_RATE_LIMITING_WINDOW_MS, RATE_LIMITING_WINDOW_MS).println();
2412             pw.printPair(KEY_MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW,
2413                     MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW).println();
2414             pw.printPair(KEY_MAX_SESSION_COUNT_ACTIVE, MAX_SESSION_COUNT_ACTIVE).println();
2415             pw.printPair(KEY_MAX_SESSION_COUNT_WORKING, MAX_SESSION_COUNT_WORKING).println();
2416             pw.printPair(KEY_MAX_SESSION_COUNT_FREQUENT, MAX_SESSION_COUNT_FREQUENT).println();
2417             pw.printPair(KEY_MAX_SESSION_COUNT_RARE, MAX_SESSION_COUNT_RARE).println();
2418             pw.printPair(KEY_MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW,
2419                     MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW).println();
2420             pw.printPair(KEY_TIMING_SESSION_COALESCING_DURATION_MS,
2421                     TIMING_SESSION_COALESCING_DURATION_MS).println();
2422             pw.decreaseIndent();
2423         }
2424 
dump(ProtoOutputStream proto)2425         private void dump(ProtoOutputStream proto) {
2426             final long qcToken = proto.start(ConstantsProto.QUOTA_CONTROLLER);
2427             proto.write(ConstantsProto.QuotaController.ALLOWED_TIME_PER_PERIOD_MS,
2428                     ALLOWED_TIME_PER_PERIOD_MS);
2429             proto.write(ConstantsProto.QuotaController.IN_QUOTA_BUFFER_MS, IN_QUOTA_BUFFER_MS);
2430             proto.write(ConstantsProto.QuotaController.ACTIVE_WINDOW_SIZE_MS,
2431                     WINDOW_SIZE_ACTIVE_MS);
2432             proto.write(ConstantsProto.QuotaController.WORKING_WINDOW_SIZE_MS,
2433                     WINDOW_SIZE_WORKING_MS);
2434             proto.write(ConstantsProto.QuotaController.FREQUENT_WINDOW_SIZE_MS,
2435                     WINDOW_SIZE_FREQUENT_MS);
2436             proto.write(ConstantsProto.QuotaController.RARE_WINDOW_SIZE_MS, WINDOW_SIZE_RARE_MS);
2437             proto.write(ConstantsProto.QuotaController.MAX_EXECUTION_TIME_MS,
2438                     MAX_EXECUTION_TIME_MS);
2439             proto.write(ConstantsProto.QuotaController.MAX_JOB_COUNT_ACTIVE, MAX_JOB_COUNT_ACTIVE);
2440             proto.write(ConstantsProto.QuotaController.MAX_JOB_COUNT_WORKING,
2441                     MAX_JOB_COUNT_WORKING);
2442             proto.write(ConstantsProto.QuotaController.MAX_JOB_COUNT_FREQUENT,
2443                     MAX_JOB_COUNT_FREQUENT);
2444             proto.write(ConstantsProto.QuotaController.MAX_JOB_COUNT_RARE, MAX_JOB_COUNT_RARE);
2445             proto.write(ConstantsProto.QuotaController.RATE_LIMITING_WINDOW_MS,
2446                     RATE_LIMITING_WINDOW_MS);
2447             proto.write(ConstantsProto.QuotaController.MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW,
2448                     MAX_JOB_COUNT_PER_RATE_LIMITING_WINDOW);
2449             proto.write(ConstantsProto.QuotaController.MAX_SESSION_COUNT_ACTIVE,
2450                     MAX_SESSION_COUNT_ACTIVE);
2451             proto.write(ConstantsProto.QuotaController.MAX_SESSION_COUNT_WORKING,
2452                     MAX_SESSION_COUNT_WORKING);
2453             proto.write(ConstantsProto.QuotaController.MAX_SESSION_COUNT_FREQUENT,
2454                     MAX_SESSION_COUNT_FREQUENT);
2455             proto.write(ConstantsProto.QuotaController.MAX_SESSION_COUNT_RARE,
2456                     MAX_SESSION_COUNT_RARE);
2457             proto.write(ConstantsProto.QuotaController.MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW,
2458                     MAX_SESSION_COUNT_PER_RATE_LIMITING_WINDOW);
2459             proto.write(ConstantsProto.QuotaController.TIMING_SESSION_COALESCING_DURATION_MS,
2460                     TIMING_SESSION_COALESCING_DURATION_MS);
2461             proto.end(qcToken);
2462         }
2463     }
2464 
2465     //////////////////////// TESTING HELPERS /////////////////////////////
2466 
2467     @VisibleForTesting
getAllowedTimePerPeriodMs()2468     long getAllowedTimePerPeriodMs() {
2469         return mAllowedTimePerPeriodMs;
2470     }
2471 
2472     @VisibleForTesting
2473     @NonNull
getBucketMaxJobCounts()2474     int[] getBucketMaxJobCounts() {
2475         return mMaxBucketJobCounts;
2476     }
2477 
2478     @VisibleForTesting
2479     @NonNull
getBucketMaxSessionCounts()2480     int[] getBucketMaxSessionCounts() {
2481         return mMaxBucketSessionCounts;
2482     }
2483 
2484     @VisibleForTesting
2485     @NonNull
getBucketWindowSizes()2486     long[] getBucketWindowSizes() {
2487         return mBucketPeriodsMs;
2488     }
2489 
2490     @VisibleForTesting
2491     @NonNull
getForegroundUids()2492     SparseBooleanArray getForegroundUids() {
2493         return mForegroundUids;
2494     }
2495 
2496     @VisibleForTesting
2497     @NonNull
getHandler()2498     Handler getHandler() {
2499         return mHandler;
2500     }
2501 
2502     @VisibleForTesting
getInQuotaBufferMs()2503     long getInQuotaBufferMs() {
2504         return mQuotaBufferMs;
2505     }
2506 
2507     @VisibleForTesting
getMaxExecutionTimeMs()2508     long getMaxExecutionTimeMs() {
2509         return mMaxExecutionTimeMs;
2510     }
2511 
2512     @VisibleForTesting
getMaxJobCountPerRateLimitingWindow()2513     int getMaxJobCountPerRateLimitingWindow() {
2514         return mMaxJobCountPerRateLimitingWindow;
2515     }
2516 
2517     @VisibleForTesting
getMaxSessionCountPerRateLimitingWindow()2518     int getMaxSessionCountPerRateLimitingWindow() {
2519         return mMaxSessionCountPerRateLimitingWindow;
2520     }
2521 
2522     @VisibleForTesting
getRateLimitingWindowMs()2523     long getRateLimitingWindowMs() {
2524         return mRateLimitingWindowMs;
2525     }
2526 
2527     @VisibleForTesting
getTimingSessionCoalescingDurationMs()2528     long getTimingSessionCoalescingDurationMs() {
2529         return mTimingSessionCoalescingDurationMs;
2530     }
2531 
2532     @VisibleForTesting
2533     @Nullable
getTimingSessions(int userId, String packageName)2534     List<TimingSession> getTimingSessions(int userId, String packageName) {
2535         return mTimingSessions.get(userId, packageName);
2536     }
2537 
2538     @VisibleForTesting
2539     @NonNull
getQcConstants()2540     QcConstants getQcConstants() {
2541         return mQcConstants;
2542     }
2543 
2544     //////////////////////////// DATA DUMP //////////////////////////////
2545 
2546     @Override
dumpControllerStateLocked(final IndentingPrintWriter pw, final Predicate<JobStatus> predicate)2547     public void dumpControllerStateLocked(final IndentingPrintWriter pw,
2548             final Predicate<JobStatus> predicate) {
2549         pw.println("Is throttling: " + mShouldThrottle);
2550         pw.println("Is charging: " + mChargeTracker.isCharging());
2551         pw.println("In parole: " + mInParole);
2552         pw.println("Current elapsed time: " + sElapsedRealtimeClock.millis());
2553         pw.println();
2554 
2555         pw.print("Foreground UIDs: ");
2556         pw.println(mForegroundUids.toString());
2557         pw.println();
2558 
2559         pw.println("Cached UID->package map:");
2560         pw.increaseIndent();
2561         for (int i = 0; i < mUidToPackageCache.size(); ++i) {
2562             final int uid = mUidToPackageCache.keyAt(i);
2563             pw.print(uid);
2564             pw.print(": ");
2565             pw.println(mUidToPackageCache.get(uid));
2566         }
2567         pw.decreaseIndent();
2568         pw.println();
2569 
2570         mTrackedJobs.forEach((jobs) -> {
2571             for (int j = 0; j < jobs.size(); j++) {
2572                 final JobStatus js = jobs.valueAt(j);
2573                 if (!predicate.test(js)) {
2574                     continue;
2575                 }
2576                 pw.print("#");
2577                 js.printUniqueId(pw);
2578                 pw.print(" from ");
2579                 UserHandle.formatUid(pw, js.getSourceUid());
2580                 if (mTopStartedJobs.contains(js)) {
2581                     pw.print(" (TOP)");
2582                 }
2583                 pw.println();
2584 
2585                 pw.increaseIndent();
2586                 pw.print(JobStatus.bucketName(getEffectiveStandbyBucket(js)));
2587                 pw.print(", ");
2588                 if (js.isConstraintSatisfied(JobStatus.CONSTRAINT_WITHIN_QUOTA)) {
2589                     pw.print("within quota");
2590                 } else {
2591                     pw.print("not within quota");
2592                 }
2593                 pw.print(", ");
2594                 pw.print(getRemainingExecutionTimeLocked(js));
2595                 pw.print("ms remaining in quota");
2596                 pw.decreaseIndent();
2597                 pw.println();
2598             }
2599         });
2600 
2601         pw.println();
2602         for (int u = 0; u < mPkgTimers.numUsers(); ++u) {
2603             final int userId = mPkgTimers.keyAt(u);
2604             for (int p = 0; p < mPkgTimers.numPackagesForUser(userId); ++p) {
2605                 final String pkgName = mPkgTimers.keyAt(u, p);
2606                 mPkgTimers.valueAt(u, p).dump(pw, predicate);
2607                 pw.println();
2608                 List<TimingSession> sessions = mTimingSessions.get(userId, pkgName);
2609                 if (sessions != null) {
2610                     pw.increaseIndent();
2611                     pw.println("Saved sessions:");
2612                     pw.increaseIndent();
2613                     for (int j = sessions.size() - 1; j >= 0; j--) {
2614                         TimingSession session = sessions.get(j);
2615                         session.dump(pw);
2616                     }
2617                     pw.decreaseIndent();
2618                     pw.decreaseIndent();
2619                     pw.println();
2620                 }
2621             }
2622         }
2623 
2624         pw.println("Cached execution stats:");
2625         pw.increaseIndent();
2626         for (int u = 0; u < mExecutionStatsCache.numUsers(); ++u) {
2627             final int userId = mExecutionStatsCache.keyAt(u);
2628             for (int p = 0; p < mExecutionStatsCache.numPackagesForUser(userId); ++p) {
2629                 final String pkgName = mExecutionStatsCache.keyAt(u, p);
2630                 ExecutionStats[] stats = mExecutionStatsCache.valueAt(u, p);
2631 
2632                 pw.println(string(userId, pkgName));
2633                 pw.increaseIndent();
2634                 for (int i = 0; i < stats.length; ++i) {
2635                     ExecutionStats executionStats = stats[i];
2636                     if (executionStats != null) {
2637                         pw.print(JobStatus.bucketName(i));
2638                         pw.print(": ");
2639                         pw.println(executionStats);
2640                     }
2641                 }
2642                 pw.decreaseIndent();
2643             }
2644         }
2645         pw.decreaseIndent();
2646 
2647         pw.println();
2648         pw.println("In quota alarms:");
2649         pw.increaseIndent();
2650         for (int u = 0; u < mInQuotaAlarmListeners.numUsers(); ++u) {
2651             final int userId = mInQuotaAlarmListeners.keyAt(u);
2652             for (int p = 0; p < mInQuotaAlarmListeners.numPackagesForUser(userId); ++p) {
2653                 final String pkgName = mInQuotaAlarmListeners.keyAt(u, p);
2654                 QcAlarmListener alarmListener = mInQuotaAlarmListeners.valueAt(u, p);
2655 
2656                 pw.print(string(userId, pkgName));
2657                 pw.print(": ");
2658                 if (alarmListener.isWaiting()) {
2659                     pw.println(alarmListener.getTriggerTimeElapsed());
2660                 } else {
2661                     pw.println("NOT WAITING");
2662                 }
2663             }
2664         }
2665         pw.decreaseIndent();
2666     }
2667 
2668     @Override
dumpControllerStateLocked(ProtoOutputStream proto, long fieldId, Predicate<JobStatus> predicate)2669     public void dumpControllerStateLocked(ProtoOutputStream proto, long fieldId,
2670             Predicate<JobStatus> predicate) {
2671         final long token = proto.start(fieldId);
2672         final long mToken = proto.start(StateControllerProto.QUOTA);
2673 
2674         proto.write(StateControllerProto.QuotaController.IS_CHARGING, mChargeTracker.isCharging());
2675         proto.write(StateControllerProto.QuotaController.IS_IN_PAROLE, mInParole);
2676         proto.write(StateControllerProto.QuotaController.ELAPSED_REALTIME,
2677                 sElapsedRealtimeClock.millis());
2678 
2679         for (int i = 0; i < mForegroundUids.size(); ++i) {
2680             proto.write(StateControllerProto.QuotaController.FOREGROUND_UIDS,
2681                     mForegroundUids.keyAt(i));
2682         }
2683 
2684         mTrackedJobs.forEach((jobs) -> {
2685             for (int j = 0; j < jobs.size(); j++) {
2686                 final JobStatus js = jobs.valueAt(j);
2687                 if (!predicate.test(js)) {
2688                     continue;
2689                 }
2690                 final long jsToken = proto.start(
2691                         StateControllerProto.QuotaController.TRACKED_JOBS);
2692                 js.writeToShortProto(proto,
2693                         StateControllerProto.QuotaController.TrackedJob.INFO);
2694                 proto.write(StateControllerProto.QuotaController.TrackedJob.SOURCE_UID,
2695                         js.getSourceUid());
2696                 proto.write(
2697                         StateControllerProto.QuotaController.TrackedJob.EFFECTIVE_STANDBY_BUCKET,
2698                         getEffectiveStandbyBucket(js));
2699                 proto.write(StateControllerProto.QuotaController.TrackedJob.IS_TOP_STARTED_JOB,
2700                         mTopStartedJobs.contains(js));
2701                 proto.write(StateControllerProto.QuotaController.TrackedJob.HAS_QUOTA,
2702                         js.isConstraintSatisfied(JobStatus.CONSTRAINT_WITHIN_QUOTA));
2703                 proto.write(StateControllerProto.QuotaController.TrackedJob.REMAINING_QUOTA_MS,
2704                         getRemainingExecutionTimeLocked(js));
2705                 proto.end(jsToken);
2706             }
2707         });
2708 
2709         for (int u = 0; u < mPkgTimers.numUsers(); ++u) {
2710             final int userId = mPkgTimers.keyAt(u);
2711             for (int p = 0; p < mPkgTimers.numPackagesForUser(userId); ++p) {
2712                 final String pkgName = mPkgTimers.keyAt(u, p);
2713                 final long psToken = proto.start(
2714                         StateControllerProto.QuotaController.PACKAGE_STATS);
2715                 mPkgTimers.valueAt(u, p).dump(proto,
2716                         StateControllerProto.QuotaController.PackageStats.TIMER, predicate);
2717 
2718                 List<TimingSession> sessions = mTimingSessions.get(userId, pkgName);
2719                 if (sessions != null) {
2720                     for (int j = sessions.size() - 1; j >= 0; j--) {
2721                         TimingSession session = sessions.get(j);
2722                         session.dump(proto,
2723                                 StateControllerProto.QuotaController.PackageStats.SAVED_SESSIONS);
2724                     }
2725                 }
2726 
2727                 ExecutionStats[] stats = mExecutionStatsCache.get(userId, pkgName);
2728                 if (stats != null) {
2729                     for (int bucketIndex = 0; bucketIndex < stats.length; ++bucketIndex) {
2730                         ExecutionStats es = stats[bucketIndex];
2731                         if (es == null) {
2732                             continue;
2733                         }
2734                         final long esToken = proto.start(
2735                                 StateControllerProto.QuotaController.PackageStats.EXECUTION_STATS);
2736                         proto.write(
2737                                 StateControllerProto.QuotaController.ExecutionStats.STANDBY_BUCKET,
2738                                 bucketIndex);
2739                         proto.write(
2740                                 StateControllerProto.QuotaController.ExecutionStats.EXPIRATION_TIME_ELAPSED,
2741                                 es.expirationTimeElapsed);
2742                         proto.write(
2743                                 StateControllerProto.QuotaController.ExecutionStats.WINDOW_SIZE_MS,
2744                                 es.windowSizeMs);
2745                         proto.write(
2746                                 StateControllerProto.QuotaController.ExecutionStats.JOB_COUNT_LIMIT,
2747                                 es.jobCountLimit);
2748                         proto.write(
2749                                 StateControllerProto.QuotaController.ExecutionStats.SESSION_COUNT_LIMIT,
2750                                 es.sessionCountLimit);
2751                         proto.write(
2752                                 StateControllerProto.QuotaController.ExecutionStats.EXECUTION_TIME_IN_WINDOW_MS,
2753                                 es.executionTimeInWindowMs);
2754                         proto.write(
2755                                 StateControllerProto.QuotaController.ExecutionStats.BG_JOB_COUNT_IN_WINDOW,
2756                                 es.bgJobCountInWindow);
2757                         proto.write(
2758                                 StateControllerProto.QuotaController.ExecutionStats.EXECUTION_TIME_IN_MAX_PERIOD_MS,
2759                                 es.executionTimeInMaxPeriodMs);
2760                         proto.write(
2761                                 StateControllerProto.QuotaController.ExecutionStats.BG_JOB_COUNT_IN_MAX_PERIOD,
2762                                 es.bgJobCountInMaxPeriod);
2763                         proto.write(
2764                                 StateControllerProto.QuotaController.ExecutionStats.SESSION_COUNT_IN_WINDOW,
2765                                 es.sessionCountInWindow);
2766                         proto.write(
2767                                 StateControllerProto.QuotaController.ExecutionStats.IN_QUOTA_TIME_ELAPSED,
2768                                 es.inQuotaTimeElapsed);
2769                         proto.write(
2770                                 StateControllerProto.QuotaController.ExecutionStats.JOB_COUNT_EXPIRATION_TIME_ELAPSED,
2771                                 es.jobRateLimitExpirationTimeElapsed);
2772                         proto.write(
2773                                 StateControllerProto.QuotaController.ExecutionStats.JOB_COUNT_IN_RATE_LIMITING_WINDOW,
2774                                 es.jobCountInRateLimitingWindow);
2775                         proto.write(
2776                                 StateControllerProto.QuotaController.ExecutionStats.SESSION_COUNT_EXPIRATION_TIME_ELAPSED,
2777                                 es.sessionRateLimitExpirationTimeElapsed);
2778                         proto.write(
2779                                 StateControllerProto.QuotaController.ExecutionStats.SESSION_COUNT_IN_RATE_LIMITING_WINDOW,
2780                                 es.sessionCountInRateLimitingWindow);
2781                         proto.end(esToken);
2782                     }
2783                 }
2784 
2785                 QcAlarmListener alarmListener = mInQuotaAlarmListeners.get(userId, pkgName);
2786                 if (alarmListener != null) {
2787                     final long alToken = proto.start(
2788                             StateControllerProto.QuotaController.PackageStats.IN_QUOTA_ALARM_LISTENER);
2789                     proto.write(StateControllerProto.QuotaController.AlarmListener.IS_WAITING,
2790                             alarmListener.isWaiting());
2791                     proto.write(
2792                             StateControllerProto.QuotaController.AlarmListener.TRIGGER_TIME_ELAPSED,
2793                             alarmListener.getTriggerTimeElapsed());
2794                     proto.end(alToken);
2795                 }
2796 
2797                 proto.end(psToken);
2798             }
2799         }
2800 
2801         proto.end(mToken);
2802         proto.end(token);
2803     }
2804 
2805     @Override
dumpConstants(IndentingPrintWriter pw)2806     public void dumpConstants(IndentingPrintWriter pw) {
2807         mQcConstants.dump(pw);
2808     }
2809 
2810     @Override
dumpConstants(ProtoOutputStream proto)2811     public void dumpConstants(ProtoOutputStream proto) {
2812         mQcConstants.dump(proto);
2813     }
2814 }
2815