1 /* 2 * Copyright (C) 2015 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.backup; 18 19 import static com.android.server.pm.PackageManagerService.PLATFORM_PACKAGE_NAME; 20 21 import android.app.AlarmManager; 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.Bundle; 29 import android.os.RemoteException; 30 import android.util.Slog; 31 import android.util.SparseBooleanArray; 32 import android.util.SparseLongArray; 33 34 import com.android.internal.annotations.GuardedBy; 35 import com.android.internal.annotations.VisibleForTesting; 36 37 import java.util.Random; 38 39 /** 40 * Job for scheduling key/value backup work. This module encapsulates all 41 * of the policy around when those backup passes are executed. 42 */ 43 public class KeyValueBackupJob extends JobService { 44 private static final String TAG = "KeyValueBackupJob"; 45 private static ComponentName sKeyValueJobService = 46 new ComponentName(PLATFORM_PACKAGE_NAME, KeyValueBackupJob.class.getName()); 47 48 private static final String USER_ID_EXTRA_KEY = "userId"; 49 50 // Once someone asks for a backup, this is how long we hold off until we find 51 // an on-charging opportunity. If we hit this max latency we will run the operation 52 // regardless. Privileged callers can always trigger an immediate pass via 53 // BackupManager.backupNow(). 54 private static final long MAX_DEFERRAL = AlarmManager.INTERVAL_DAY; 55 56 @GuardedBy("KeyValueBackupJob.class") 57 private static final SparseBooleanArray sScheduledForUserId = new SparseBooleanArray(); 58 @GuardedBy("KeyValueBackupJob.class") 59 private static final SparseLongArray sNextScheduledForUserId = new SparseLongArray(); 60 61 @VisibleForTesting 62 public static final int MIN_JOB_ID = 52417896; 63 @VisibleForTesting 64 public static final int MAX_JOB_ID = 52418896; 65 schedule(int userId, Context ctx, UserBackupManagerService userBackupManagerService)66 public static void schedule(int userId, Context ctx, 67 UserBackupManagerService userBackupManagerService) { 68 schedule(userId, ctx, 0, userBackupManagerService); 69 } 70 schedule(int userId, Context ctx, long delay, UserBackupManagerService userBackupManagerService)71 public static void schedule(int userId, Context ctx, long delay, 72 UserBackupManagerService userBackupManagerService) { 73 synchronized (KeyValueBackupJob.class) { 74 if (sScheduledForUserId.get(userId) 75 || !userBackupManagerService.isFrameworkSchedulingEnabled()) { 76 return; 77 } 78 79 final long interval; 80 final long fuzz; 81 final int networkType; 82 final boolean needsCharging; 83 84 final BackupManagerConstants constants = userBackupManagerService.getConstants(); 85 synchronized (constants) { 86 interval = constants.getKeyValueBackupIntervalMilliseconds(); 87 fuzz = constants.getKeyValueBackupFuzzMilliseconds(); 88 networkType = constants.getKeyValueBackupRequiredNetworkType(); 89 needsCharging = constants.getKeyValueBackupRequireCharging(); 90 } 91 if (delay <= 0) { 92 delay = interval + new Random().nextInt((int) fuzz); 93 } 94 Slog.d(TAG, "Scheduling k/v pass in " + (delay / 1000 / 60) + " minutes"); 95 96 JobInfo.Builder builder = new JobInfo.Builder(getJobIdForUserId(userId), 97 sKeyValueJobService) 98 .setMinimumLatency(delay) 99 .setRequiredNetworkType(networkType) 100 .setRequiresCharging(needsCharging) 101 .setOverrideDeadline(MAX_DEFERRAL); 102 103 Bundle extraInfo = new Bundle(); 104 extraInfo.putInt(USER_ID_EXTRA_KEY, userId); 105 builder.setTransientExtras(extraInfo); 106 107 JobScheduler js = (JobScheduler) ctx.getSystemService(Context.JOB_SCHEDULER_SERVICE); 108 js.schedule(builder.build()); 109 110 sScheduledForUserId.put(userId, true); 111 sNextScheduledForUserId.put(userId, System.currentTimeMillis() + delay); 112 } 113 } 114 cancel(int userId, Context ctx)115 public static void cancel(int userId, Context ctx) { 116 synchronized (KeyValueBackupJob.class) { 117 JobScheduler js = (JobScheduler) ctx.getSystemService( 118 Context.JOB_SCHEDULER_SERVICE); 119 js.cancel(getJobIdForUserId(userId)); 120 121 clearScheduledForUserId(userId); 122 } 123 } 124 nextScheduled(int userId)125 public static long nextScheduled(int userId) { 126 synchronized (KeyValueBackupJob.class) { 127 return sNextScheduledForUserId.get(userId); 128 } 129 } 130 131 @VisibleForTesting isScheduled(int userId)132 public static boolean isScheduled(int userId) { 133 synchronized (KeyValueBackupJob.class) { 134 return sScheduledForUserId.get(userId); 135 } 136 } 137 138 @Override onStartJob(JobParameters params)139 public boolean onStartJob(JobParameters params) { 140 int userId = params.getTransientExtras().getInt(USER_ID_EXTRA_KEY); 141 142 synchronized (KeyValueBackupJob.class) { 143 clearScheduledForUserId(userId); 144 } 145 146 // Time to run a key/value backup! 147 BackupManagerService service = BackupManagerService.getInstance(); 148 try { 149 service.backupNowForUser(userId); 150 } catch (RemoteException e) {} 151 152 // This was just a trigger; ongoing wakelock management is done by the 153 // rest of the backup system. 154 return false; 155 } 156 157 @Override onStopJob(JobParameters params)158 public boolean onStopJob(JobParameters params) { 159 // Intentionally empty; the job starting was just a trigger 160 return false; 161 } 162 163 @GuardedBy("KeyValueBackupJob.class") clearScheduledForUserId(int userId)164 private static void clearScheduledForUserId(int userId) { 165 sScheduledForUserId.delete(userId); 166 sNextScheduledForUserId.delete(userId); 167 } 168 169 @VisibleForTesting getJobIdForUserId(int userId)170 static int getJobIdForUserId(int userId) { 171 return JobIdManager.getJobIdForUserId(MIN_JOB_ID, MAX_JOB_ID, userId); 172 } 173 } 174