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 android.net; 18 19 import android.content.Context; 20 import android.os.Handler; 21 import android.os.Looper; 22 import android.os.Message; 23 import android.os.Messenger; 24 import android.util.Log; 25 26 import com.android.internal.util.AsyncChannel; 27 import com.android.internal.util.Protocol; 28 29 import java.util.ArrayList; 30 import java.util.concurrent.atomic.AtomicBoolean; 31 32 /** 33 * A Utility class for handling for communicating between bearer-specific 34 * code and ConnectivityService. 35 * 36 * A bearer may have more than one NetworkAgent if it can simultaneously 37 * support separate networks (IMS / Internet / MMS Apns on cellular, or 38 * perhaps connections with different SSID or P2P for Wi-Fi). 39 * 40 * @hide 41 */ 42 public abstract class NetworkAgent extends Handler { 43 // Guaranteed to be valid (not NETID_UNSET), otherwise registerNetworkAgent() would have thrown 44 // an exception. 45 public final int netId; 46 47 private volatile AsyncChannel mAsyncChannel; 48 private final String LOG_TAG; 49 private static final boolean DBG = true; 50 private static final boolean VDBG = false; 51 private final Context mContext; 52 private final ArrayList<Message>mPreConnectedQueue = new ArrayList<Message>(); 53 private volatile long mLastBwRefreshTime = 0; 54 private static final long BW_REFRESH_MIN_WIN_MS = 500; 55 private boolean mPollLceScheduled = false; 56 private AtomicBoolean mPollLcePending = new AtomicBoolean(false); 57 58 private static final int BASE = Protocol.BASE_NETWORK_AGENT; 59 60 /** 61 * Sent by ConnectivityService to the NetworkAgent to inform it of 62 * suspected connectivity problems on its network. The NetworkAgent 63 * should take steps to verify and correct connectivity. 64 */ 65 public static final int CMD_SUSPECT_BAD = BASE; 66 67 /** 68 * Sent by the NetworkAgent (note the EVENT vs CMD prefix) to 69 * ConnectivityService to pass the current NetworkInfo (connection state). 70 * Sent when the NetworkInfo changes, mainly due to change of state. 71 * obj = NetworkInfo 72 */ 73 public static final int EVENT_NETWORK_INFO_CHANGED = BASE + 1; 74 75 /** 76 * Sent by the NetworkAgent to ConnectivityService to pass the current 77 * NetworkCapabilties. 78 * obj = NetworkCapabilities 79 */ 80 public static final int EVENT_NETWORK_CAPABILITIES_CHANGED = BASE + 2; 81 82 /** 83 * Sent by the NetworkAgent to ConnectivityService to pass the current 84 * NetworkProperties. 85 * obj = NetworkProperties 86 */ 87 public static final int EVENT_NETWORK_PROPERTIES_CHANGED = BASE + 3; 88 89 /* centralize place where base network score, and network score scaling, will be 90 * stored, so as we can consistently compare apple and oranges, or wifi, ethernet and LTE 91 */ 92 public static final int WIFI_BASE_SCORE = 60; 93 94 /** 95 * Sent by the NetworkAgent to ConnectivityService to pass the current 96 * network score. 97 * obj = network score Integer 98 */ 99 public static final int EVENT_NETWORK_SCORE_CHANGED = BASE + 4; 100 101 /** 102 * Sent by the NetworkAgent to ConnectivityService to add new UID ranges 103 * to be forced into this Network. For VPNs only. 104 * obj = UidRange[] to forward 105 */ 106 public static final int EVENT_UID_RANGES_ADDED = BASE + 5; 107 108 /** 109 * Sent by the NetworkAgent to ConnectivityService to remove UID ranges 110 * from being forced into this Network. For VPNs only. 111 * obj = UidRange[] to stop forwarding 112 */ 113 public static final int EVENT_UID_RANGES_REMOVED = BASE + 6; 114 115 /** 116 * Sent by ConnectivityService to the NetworkAgent to inform the agent of the 117 * networks status - whether we could use the network or could not, due to 118 * either a bad network configuration (no internet link) or captive portal. 119 * 120 * arg1 = either {@code VALID_NETWORK} or {@code INVALID_NETWORK} 121 */ 122 public static final int CMD_REPORT_NETWORK_STATUS = BASE + 7; 123 124 public static final int VALID_NETWORK = 1; 125 public static final int INVALID_NETWORK = 2; 126 127 /** 128 * Sent by the NetworkAgent to ConnectivityService to indicate this network was 129 * explicitly selected. This should be sent before the NetworkInfo is marked 130 * CONNECTED so it can be given special treatment at that time. 131 * 132 * obj = boolean indicating whether to use this network even if unvalidated 133 */ 134 public static final int EVENT_SET_EXPLICITLY_SELECTED = BASE + 8; 135 136 /** 137 * Sent by ConnectivityService to the NetworkAgent to inform the agent of 138 * whether the network should in the future be used even if not validated. 139 * This decision is made by the user, but it is the network transport's 140 * responsibility to remember it. 141 * 142 * arg1 = 1 if true, 0 if false 143 */ 144 public static final int CMD_SAVE_ACCEPT_UNVALIDATED = BASE + 9; 145 146 /** Sent by ConnectivityService to the NetworkAgent to inform the agent to pull 147 * the underlying network connection for updated bandwidth information. 148 */ 149 public static final int CMD_REQUEST_BANDWIDTH_UPDATE = BASE + 10; 150 151 /** 152 * Sent by ConnectivityService to the NeworkAgent to inform the agent to avoid 153 * automatically reconnecting to this network (e.g. via autojoin). Happens 154 * when user selects "No" option on the "Stay connected?" dialog box. 155 */ 156 public static final int CMD_PREVENT_AUTOMATIC_RECONNECT = BASE + 11; 157 158 public NetworkAgent(Looper looper, Context context, String logTag, NetworkInfo ni, 159 NetworkCapabilities nc, LinkProperties lp, int score) { 160 this(looper, context, logTag, ni, nc, lp, score, null); 161 } 162 163 public NetworkAgent(Looper looper, Context context, String logTag, NetworkInfo ni, 164 NetworkCapabilities nc, LinkProperties lp, int score, NetworkMisc misc) { 165 super(looper); 166 LOG_TAG = logTag; 167 mContext = context; 168 if (ni == null || nc == null || lp == null) { 169 throw new IllegalArgumentException(); 170 } 171 172 if (VDBG) log("Registering NetworkAgent"); 173 ConnectivityManager cm = (ConnectivityManager)mContext.getSystemService( 174 Context.CONNECTIVITY_SERVICE); 175 netId = cm.registerNetworkAgent(new Messenger(this), new NetworkInfo(ni), 176 new LinkProperties(lp), new NetworkCapabilities(nc), score, misc); 177 } 178 179 @Override 180 public void handleMessage(Message msg) { 181 switch (msg.what) { 182 case AsyncChannel.CMD_CHANNEL_FULL_CONNECTION: { 183 if (mAsyncChannel != null) { 184 log("Received new connection while already connected!"); 185 } else { 186 if (VDBG) log("NetworkAgent fully connected"); 187 AsyncChannel ac = new AsyncChannel(); 188 ac.connected(null, this, msg.replyTo); 189 ac.replyToMessage(msg, AsyncChannel.CMD_CHANNEL_FULLY_CONNECTED, 190 AsyncChannel.STATUS_SUCCESSFUL); 191 synchronized (mPreConnectedQueue) { 192 mAsyncChannel = ac; 193 for (Message m : mPreConnectedQueue) { 194 ac.sendMessage(m); 195 } 196 mPreConnectedQueue.clear(); 197 } 198 } 199 break; 200 } 201 case AsyncChannel.CMD_CHANNEL_DISCONNECT: { 202 if (VDBG) log("CMD_CHANNEL_DISCONNECT"); 203 if (mAsyncChannel != null) mAsyncChannel.disconnect(); 204 break; 205 } 206 case AsyncChannel.CMD_CHANNEL_DISCONNECTED: { 207 if (DBG) log("NetworkAgent channel lost"); 208 // let the client know CS is done with us. 209 unwanted(); 210 synchronized (mPreConnectedQueue) { 211 mAsyncChannel = null; 212 } 213 break; 214 } 215 case CMD_SUSPECT_BAD: { 216 log("Unhandled Message " + msg); 217 break; 218 } 219 case CMD_REQUEST_BANDWIDTH_UPDATE: { 220 long currentTimeMs = System.currentTimeMillis(); 221 if (VDBG) { 222 log("CMD_REQUEST_BANDWIDTH_UPDATE request received."); 223 } 224 if (currentTimeMs >= (mLastBwRefreshTime + BW_REFRESH_MIN_WIN_MS)) { 225 mPollLceScheduled = false; 226 if (mPollLcePending.getAndSet(true) == false) { 227 pollLceData(); 228 } 229 } else { 230 // deliver the request at a later time rather than discard it completely. 231 if (!mPollLceScheduled) { 232 long waitTime = mLastBwRefreshTime + BW_REFRESH_MIN_WIN_MS - 233 currentTimeMs + 1; 234 mPollLceScheduled = sendEmptyMessageDelayed( 235 CMD_REQUEST_BANDWIDTH_UPDATE, waitTime); 236 } 237 } 238 break; 239 } 240 case CMD_REPORT_NETWORK_STATUS: { 241 if (VDBG) { 242 log("CMD_REPORT_NETWORK_STATUS(" + 243 (msg.arg1 == VALID_NETWORK ? "VALID)" : "INVALID)")); 244 } 245 networkStatus(msg.arg1); 246 break; 247 } 248 case CMD_SAVE_ACCEPT_UNVALIDATED: { 249 saveAcceptUnvalidated(msg.arg1 != 0); 250 break; 251 } 252 case CMD_PREVENT_AUTOMATIC_RECONNECT: { 253 preventAutomaticReconnect(); 254 break; 255 } 256 } 257 } 258 259 private void queueOrSendMessage(int what, Object obj) { 260 synchronized (mPreConnectedQueue) { 261 if (mAsyncChannel != null) { 262 mAsyncChannel.sendMessage(what, obj); 263 } else { 264 Message msg = Message.obtain(); 265 msg.what = what; 266 msg.obj = obj; 267 mPreConnectedQueue.add(msg); 268 } 269 } 270 } 271 272 /** 273 * Called by the bearer code when it has new LinkProperties data. 274 */ 275 public void sendLinkProperties(LinkProperties linkProperties) { 276 queueOrSendMessage(EVENT_NETWORK_PROPERTIES_CHANGED, new LinkProperties(linkProperties)); 277 } 278 279 /** 280 * Called by the bearer code when it has new NetworkInfo data. 281 */ 282 public void sendNetworkInfo(NetworkInfo networkInfo) { 283 queueOrSendMessage(EVENT_NETWORK_INFO_CHANGED, new NetworkInfo(networkInfo)); 284 } 285 286 /** 287 * Called by the bearer code when it has new NetworkCapabilities data. 288 */ 289 public void sendNetworkCapabilities(NetworkCapabilities networkCapabilities) { 290 mPollLcePending.set(false); 291 mLastBwRefreshTime = System.currentTimeMillis(); 292 queueOrSendMessage(EVENT_NETWORK_CAPABILITIES_CHANGED, 293 new NetworkCapabilities(networkCapabilities)); 294 } 295 296 /** 297 * Called by the bearer code when it has a new score for this network. 298 */ 299 public void sendNetworkScore(int score) { 300 if (score < 0) { 301 throw new IllegalArgumentException("Score must be >= 0"); 302 } 303 queueOrSendMessage(EVENT_NETWORK_SCORE_CHANGED, new Integer(score)); 304 } 305 306 /** 307 * Called by the VPN code when it wants to add ranges of UIDs to be routed 308 * through the VPN network. 309 */ 310 public void addUidRanges(UidRange[] ranges) { 311 queueOrSendMessage(EVENT_UID_RANGES_ADDED, ranges); 312 } 313 314 /** 315 * Called by the VPN code when it wants to remove ranges of UIDs from being routed 316 * through the VPN network. 317 */ 318 public void removeUidRanges(UidRange[] ranges) { 319 queueOrSendMessage(EVENT_UID_RANGES_REMOVED, ranges); 320 } 321 322 /** 323 * Called by the bearer to indicate this network was manually selected by the user. 324 * This should be called before the NetworkInfo is marked CONNECTED so that this 325 * Network can be given special treatment at that time. If {@code acceptUnvalidated} is 326 * {@code true}, then the system will switch to this network. If it is {@code false} and the 327 * network cannot be validated, the system will ask the user whether to switch to this network. 328 * If the user confirms and selects "don't ask again", then the system will call 329 * {@link #saveAcceptUnvalidated} to persist the user's choice. Thus, if the transport ever 330 * calls this method with {@code acceptUnvalidated} set to {@code false}, it must also implement 331 * {@link #saveAcceptUnvalidated} to respect the user's choice. 332 */ 333 public void explicitlySelected(boolean acceptUnvalidated) { 334 queueOrSendMessage(EVENT_SET_EXPLICITLY_SELECTED, acceptUnvalidated); 335 } 336 337 /** 338 * Called when ConnectivityService has indicated they no longer want this network. 339 * The parent factory should (previously) have received indication of the change 340 * as well, either canceling NetworkRequests or altering their score such that this 341 * network won't be immediately requested again. 342 */ 343 abstract protected void unwanted(); 344 345 /** 346 * Called when ConnectivityService request a bandwidth update. The parent factory 347 * shall try to overwrite this method and produce a bandwidth update if capable. 348 */ 349 protected void pollLceData() { 350 } 351 352 /** 353 * Called when the system determines the usefulness of this network. 354 * 355 * Networks claiming internet connectivity will have their internet 356 * connectivity verified. 357 * 358 * Currently there are two possible values: 359 * {@code VALID_NETWORK} if the system is happy with the connection, 360 * {@code INVALID_NETWORK} if the system is not happy. 361 * TODO - add indications of captive portal-ness and related success/failure, 362 * ie, CAPTIVE_SUCCESS_NETWORK, CAPTIVE_NETWORK for successful login and detection 363 * 364 * This may be called multiple times as the network status changes and may 365 * generate false negatives if we lose ip connectivity before the link is torn down. 366 */ 367 protected void networkStatus(int status) { 368 } 369 370 /** 371 * Called when the user asks to remember the choice to use this network even if unvalidated. 372 * The transport is responsible for remembering the choice, and the next time the user connects 373 * to the network, should explicitlySelected with {@code acceptUnvalidated} set to {@code true}. 374 * This method will only be called if {@link #explicitlySelected} was called with 375 * {@code acceptUnvalidated} set to {@code false}. 376 */ 377 protected void saveAcceptUnvalidated(boolean accept) { 378 } 379 380 /** 381 * Called when the user asks to not stay connected to this network because it was found to not 382 * provide Internet access. Usually followed by call to {@code unwanted}. The transport is 383 * responsible for making sure the device does not automatically reconnect to the same network 384 * after the {@code unwanted} call. 385 */ 386 protected void preventAutomaticReconnect() { 387 } 388 389 protected void log(String s) { 390 Log.d(LOG_TAG, "NetworkAgent: " + s); 391 } 392 } 393