1 /*
2  * Copyright (C) 2014 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;
18 
19 import java.util.Calendar;
20 
21 import android.app.ActivityManager;
22 import android.app.job.JobInfo;
23 import android.app.job.JobParameters;
24 import android.app.job.JobScheduler;
25 import android.app.job.JobService;
26 import android.content.ComponentName;
27 import android.content.Context;
28 import android.os.RemoteException;
29 import android.util.Slog;
30 import java.util.concurrent.TimeUnit;
31 
32 public class MountServiceIdler extends JobService {
33     private static final String TAG = "MountServiceIdler";
34 
35     private static ComponentName sIdleService =
36             new ComponentName("android", MountServiceIdler.class.getName());
37 
38     private static int MOUNT_JOB_ID = 808;
39 
40     private boolean mStarted;
41     private JobParameters mJobParams;
42     private Runnable mFinishCallback = new Runnable() {
43         @Override
44         public void run() {
45             Slog.i(TAG, "Got mount service completion callback");
46             synchronized (mFinishCallback) {
47                 if (mStarted) {
48                     jobFinished(mJobParams, false);
49                     mStarted = false;
50                 }
51             }
52             // ... and try again right away or tomorrow
53             scheduleIdlePass(MountServiceIdler.this);
54         }
55     };
56 
57     @Override
onStartJob(JobParameters params)58     public boolean onStartJob(JobParameters params) {
59         // First have the activity manager do its idle maintenance.  (Yes this job
60         // is really more than just mount, some day it should be renamed to be system
61         // idleer).
62         try {
63             ActivityManager.getService().performIdleMaintenance();
64         } catch (RemoteException e) {
65         }
66         // The mount service will run an fstrim operation asynchronously
67         // on a designated separate thread, so we provide it with a callback
68         // that lets us cleanly end our idle timeslice.  It's safe to call
69         // finishIdle() from any thread.
70         mJobParams = params;
71         StorageManagerService ms = StorageManagerService.sSelf;
72         if (ms != null) {
73             synchronized (mFinishCallback) {
74                 mStarted = true;
75             }
76             ms.runIdleMaint(mFinishCallback);
77         }
78         return ms != null;
79     }
80 
81     @Override
onStopJob(JobParameters params)82     public boolean onStopJob(JobParameters params) {
83         // Once we kick off the fstrim we aren't actually interruptible; just note
84         // that we don't need to call jobFinished(), and let everything happen in
85         // the callback from the mount service.
86         StorageManagerService ms = StorageManagerService.sSelf;
87         if (ms != null) {
88             ms.abortIdleMaint(mFinishCallback);
89             synchronized (mFinishCallback) {
90                 mStarted = false;
91             }
92         }
93         return false;
94     }
95 
96     /**
97      * Schedule the idle job that will ping the mount service
98      */
scheduleIdlePass(Context context)99     public static void scheduleIdlePass(Context context) {
100         JobScheduler tm = (JobScheduler) context.getSystemService(Context.JOB_SCHEDULER_SERVICE);
101 
102         final long today3AM = offsetFromTodayMidnight(0, 3).getTimeInMillis();
103         final long today4AM = offsetFromTodayMidnight(0, 4).getTimeInMillis();
104         final long tomorrow3AM = offsetFromTodayMidnight(1, 3).getTimeInMillis();
105 
106         long nextScheduleTime;
107         if (System.currentTimeMillis() > today3AM && System.currentTimeMillis() < today4AM) {
108             nextScheduleTime = TimeUnit.SECONDS.toMillis(10);
109         } else {
110             nextScheduleTime = tomorrow3AM - System.currentTimeMillis(); // 3AM tomorrow
111         }
112 
113         JobInfo.Builder builder = new JobInfo.Builder(MOUNT_JOB_ID, sIdleService);
114         builder.setRequiresDeviceIdle(true);
115         builder.setRequiresBatteryNotLow(true);
116         builder.setRequiresCharging(true);
117         builder.setMinimumLatency(nextScheduleTime);
118         tm.schedule(builder.build());
119     }
120 
offsetFromTodayMidnight(int nDays, int nHours)121     private static Calendar offsetFromTodayMidnight(int nDays, int nHours) {
122         Calendar calendar = Calendar.getInstance();
123         calendar.setTimeInMillis(System.currentTimeMillis());
124         calendar.set(Calendar.HOUR_OF_DAY, nHours);
125         calendar.set(Calendar.MINUTE, 0);
126         calendar.set(Calendar.SECOND, 0);
127         calendar.set(Calendar.MILLISECOND, 0);
128         calendar.add(Calendar.DAY_OF_MONTH, nDays);
129         return calendar;
130     }
131 }
132