Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 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 | |
Christopher Tate | 7060b04 | 2014-06-09 19:50:00 -0700 | [diff] [blame] | 17 | package com.android.server.job.controllers; |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 18 | |
Jeff Sharkey | 1b6519b | 2016-04-28 15:33:18 -0600 | [diff] [blame^] | 19 | import android.app.job.JobInfo; |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 20 | import android.content.BroadcastReceiver; |
| 21 | import android.content.Context; |
| 22 | import android.content.Intent; |
| 23 | import android.content.IntentFilter; |
| 24 | import android.net.ConnectivityManager; |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 25 | import android.net.INetworkPolicyListener; |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 26 | import android.net.NetworkInfo; |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 27 | import android.net.NetworkPolicyManager; |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 28 | import android.os.UserHandle; |
Matthew Williams | 9b9244b6 | 2014-05-14 11:06:04 -0700 | [diff] [blame] | 29 | import android.util.Slog; |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 30 | |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 31 | import com.android.internal.annotations.GuardedBy; |
Christopher Tate | 7060b04 | 2014-06-09 19:50:00 -0700 | [diff] [blame] | 32 | import com.android.server.job.JobSchedulerService; |
| 33 | import com.android.server.job.StateChangedListener; |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 34 | |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 35 | import java.io.PrintWriter; |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 36 | import java.util.ArrayList; |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 37 | |
| 38 | /** |
Matthew Williams | 9b9244b6 | 2014-05-14 11:06:04 -0700 | [diff] [blame] | 39 | * Handles changes in connectivity. |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 40 | * <p> |
| 41 | * Each app can have a different default networks or different connectivity |
| 42 | * status due to user-requested network policies, so we need to check |
| 43 | * constraints on a per-UID basis. |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 44 | */ |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 45 | public class ConnectivityController extends StateController implements |
| 46 | ConnectivityManager.OnNetworkActiveListener { |
Christopher Tate | 7060b04 | 2014-06-09 19:50:00 -0700 | [diff] [blame] | 47 | private static final String TAG = "JobScheduler.Conn"; |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 48 | |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 49 | private final ConnectivityManager mConnManager; |
| 50 | private final NetworkPolicyManager mNetPolicyManager; |
| 51 | |
| 52 | @GuardedBy("mLock") |
| 53 | private final ArrayList<JobStatus> mTrackedJobs = new ArrayList<JobStatus>(); |
| 54 | |
Matthew Williams | 9b9244b6 | 2014-05-14 11:06:04 -0700 | [diff] [blame] | 55 | /** Singleton. */ |
| 56 | private static ConnectivityController mSingleton; |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 57 | private static Object sCreationLock = new Object(); |
Matthew Williams | 691e93e | 2014-05-12 15:33:09 -0700 | [diff] [blame] | 58 | |
Christopher Tate | 7060b04 | 2014-06-09 19:50:00 -0700 | [diff] [blame] | 59 | public static ConnectivityController get(JobSchedulerService jms) { |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 60 | synchronized (sCreationLock) { |
| 61 | if (mSingleton == null) { |
Dianne Hackborn | 33d31c5 | 2016-02-16 10:30:33 -0800 | [diff] [blame] | 62 | mSingleton = new ConnectivityController(jms, jms.getContext(), jms.getLock()); |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 63 | } |
| 64 | return mSingleton; |
Matthew Williams | 9b9244b6 | 2014-05-14 11:06:04 -0700 | [diff] [blame] | 65 | } |
Matthew Williams | 9b9244b6 | 2014-05-14 11:06:04 -0700 | [diff] [blame] | 66 | } |
| 67 | |
Dianne Hackborn | 33d31c5 | 2016-02-16 10:30:33 -0800 | [diff] [blame] | 68 | private ConnectivityController(StateChangedListener stateChangedListener, Context context, |
| 69 | Object lock) { |
| 70 | super(stateChangedListener, context, lock); |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 71 | |
| 72 | mConnManager = mContext.getSystemService(ConnectivityManager.class); |
| 73 | mNetPolicyManager = mContext.getSystemService(NetworkPolicyManager.class); |
| 74 | |
| 75 | final IntentFilter intentFilter = new IntentFilter(ConnectivityManager.CONNECTIVITY_ACTION); |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 76 | mContext.registerReceiverAsUser( |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 77 | mConnectivityReceiver, UserHandle.SYSTEM, intentFilter, null, null); |
| 78 | |
| 79 | mNetPolicyManager.registerListener(mNetPolicyListener); |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 80 | } |
| 81 | |
| 82 | @Override |
Dianne Hackborn | b0001f6 | 2016-02-16 10:30:33 -0800 | [diff] [blame] | 83 | public void maybeStartTrackingJobLocked(JobStatus jobStatus, JobStatus lastJob) { |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 84 | if (jobStatus.hasConnectivityConstraint() || jobStatus.hasUnmeteredConstraint() |
| 85 | || jobStatus.hasNotRoamingConstraint()) { |
| 86 | updateConstraintsSatisfied(jobStatus); |
Dianne Hackborn | b0001f6 | 2016-02-16 10:30:33 -0800 | [diff] [blame] | 87 | mTrackedJobs.add(jobStatus); |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 88 | } |
| 89 | } |
| 90 | |
| 91 | @Override |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 92 | public void maybeStopTrackingJobLocked(JobStatus jobStatus, JobStatus incomingJob, |
| 93 | boolean forUpdate) { |
| 94 | if (jobStatus.hasConnectivityConstraint() || jobStatus.hasUnmeteredConstraint() |
| 95 | || jobStatus.hasNotRoamingConstraint()) { |
Dianne Hackborn | b0001f6 | 2016-02-16 10:30:33 -0800 | [diff] [blame] | 96 | mTrackedJobs.remove(jobStatus); |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 97 | } |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 98 | } |
| 99 | |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 100 | private boolean updateConstraintsSatisfied(JobStatus jobStatus) { |
Jeff Sharkey | 1b6519b | 2016-04-28 15:33:18 -0600 | [diff] [blame^] | 101 | final boolean ignoreBlocked = (jobStatus.getFlags() & JobInfo.FLAG_WILL_BE_FOREGROUND) != 0; |
| 102 | final NetworkInfo info = mConnManager.getActiveNetworkInfoForUid(jobStatus.getSourceUid(), |
| 103 | ignoreBlocked); |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 104 | final boolean connected = (info != null) && info.isConnected(); |
| 105 | final boolean unmetered = connected && !info.isMetered(); |
| 106 | final boolean notRoaming = connected && !info.isRoaming(); |
| 107 | |
| 108 | boolean changed = false; |
| 109 | changed |= jobStatus.setConnectivityConstraintSatisfied(connected); |
| 110 | changed |= jobStatus.setUnmeteredConstraintSatisfied(unmetered); |
| 111 | changed |= jobStatus.setNotRoamingConstraintSatisfied(notRoaming); |
| 112 | return changed; |
| 113 | } |
| 114 | |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 115 | /** |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 116 | * Update all jobs tracked by this controller. |
| 117 | * |
| 118 | * @param uid only update jobs belonging to this UID, or {@code -1} to |
| 119 | * update all tracked jobs. |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 120 | */ |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 121 | private void updateTrackedJobs(int uid) { |
Dianne Hackborn | 33d31c5 | 2016-02-16 10:30:33 -0800 | [diff] [blame] | 122 | synchronized (mLock) { |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 123 | boolean changed = false; |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 124 | for (int i = 0; i < mTrackedJobs.size(); i++) { |
| 125 | final JobStatus js = mTrackedJobs.get(i); |
| 126 | if (uid == -1 || uid == js.getSourceUid()) { |
| 127 | changed |= updateConstraintsSatisfied(js); |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 128 | } |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 129 | } |
| 130 | if (changed) { |
| 131 | mStateChangedListener.onControllerStateChanged(); |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 132 | } |
| 133 | } |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 134 | } |
| 135 | |
| 136 | /** |
Christopher Tate | 7060b04 | 2014-06-09 19:50:00 -0700 | [diff] [blame] | 137 | * We know the network has just come up. We want to run any jobs that are ready. |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 138 | */ |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 139 | @Override |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 140 | public synchronized void onNetworkActive() { |
Dianne Hackborn | 33d31c5 | 2016-02-16 10:30:33 -0800 | [diff] [blame] | 141 | synchronized (mLock) { |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 142 | for (int i = 0; i < mTrackedJobs.size(); i++) { |
| 143 | final JobStatus js = mTrackedJobs.get(i); |
Christopher Tate | 7060b04 | 2014-06-09 19:50:00 -0700 | [diff] [blame] | 144 | if (js.isReady()) { |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 145 | if (DEBUG) { |
Christopher Tate | 7060b04 | 2014-06-09 19:50:00 -0700 | [diff] [blame] | 146 | Slog.d(TAG, "Running " + js + " due to network activity."); |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 147 | } |
Christopher Tate | 7060b04 | 2014-06-09 19:50:00 -0700 | [diff] [blame] | 148 | mStateChangedListener.onRunJobNow(js); |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 149 | } |
| 150 | } |
Matthew Williams | 9b9244b6 | 2014-05-14 11:06:04 -0700 | [diff] [blame] | 151 | } |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 152 | } |
| 153 | |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 154 | private BroadcastReceiver mConnectivityReceiver = new BroadcastReceiver() { |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 155 | @Override |
| 156 | public void onReceive(Context context, Intent intent) { |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 157 | updateTrackedJobs(-1); |
| 158 | } |
| 159 | }; |
| 160 | |
| 161 | private INetworkPolicyListener mNetPolicyListener = new INetworkPolicyListener.Stub() { |
| 162 | @Override |
| 163 | public void onUidRulesChanged(int uid, int uidRules) { |
| 164 | updateTrackedJobs(uid); |
| 165 | } |
| 166 | |
| 167 | @Override |
| 168 | public void onMeteredIfacesChanged(String[] meteredIfaces) { |
| 169 | updateTrackedJobs(-1); |
| 170 | } |
| 171 | |
| 172 | @Override |
| 173 | public void onRestrictBackgroundChanged(boolean restrictBackground) { |
| 174 | updateTrackedJobs(-1); |
| 175 | } |
| 176 | |
| 177 | @Override |
| 178 | public void onRestrictBackgroundWhitelistChanged(int uid, boolean whitelisted) { |
| 179 | updateTrackedJobs(uid); |
Matthew Williams | 6de79e2 | 2014-05-01 10:47:00 -0700 | [diff] [blame] | 180 | } |
| 181 | }; |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 182 | |
| 183 | @Override |
Dianne Hackborn | b0001f6 | 2016-02-16 10:30:33 -0800 | [diff] [blame] | 184 | public void dumpControllerStateLocked(PrintWriter pw) { |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 185 | pw.println("Conn."); |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 186 | for (int i = 0; i < mTrackedJobs.size(); i++) { |
| 187 | final JobStatus js = mTrackedJobs.get(i); |
Shreyas Basarge | 0529dac | 2016-03-15 22:19:04 +0000 | [diff] [blame] | 188 | pw.println(String.valueOf(js.getJobId() + "," + js.getUid()) |
Christopher Tate | 7060b04 | 2014-06-09 19:50:00 -0700 | [diff] [blame] | 189 | + ": C=" + js.hasConnectivityConstraint() |
Jeff Sharkey | f07c7b9 | 2016-04-22 09:50:16 -0600 | [diff] [blame] | 190 | + ", UM=" + js.hasUnmeteredConstraint() |
| 191 | + ", NR=" + js.hasNotRoamingConstraint()); |
Matthew Williams | effacfa | 2014-06-05 20:56:40 -0700 | [diff] [blame] | 192 | } |
| 193 | } |
Robert Greenwalt | 6078b50 | 2014-06-11 16:05:07 -0700 | [diff] [blame] | 194 | } |