1 /* 2 * Copyright (C) 2011 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.wifi.p2p; 18 19 import android.annotation.IntDef; 20 import android.annotation.NonNull; 21 import android.annotation.Nullable; 22 import android.annotation.RequiresPermission; 23 import android.annotation.SdkConstant; 24 import android.annotation.SdkConstant.SdkConstantType; 25 import android.annotation.SystemService; 26 import android.annotation.UnsupportedAppUsage; 27 import android.content.Context; 28 import android.net.NetworkInfo; 29 import android.net.wifi.WpsInfo; 30 import android.net.wifi.p2p.nsd.WifiP2pDnsSdServiceInfo; 31 import android.net.wifi.p2p.nsd.WifiP2pDnsSdServiceResponse; 32 import android.net.wifi.p2p.nsd.WifiP2pServiceInfo; 33 import android.net.wifi.p2p.nsd.WifiP2pServiceRequest; 34 import android.net.wifi.p2p.nsd.WifiP2pServiceResponse; 35 import android.net.wifi.p2p.nsd.WifiP2pUpnpServiceInfo; 36 import android.net.wifi.p2p.nsd.WifiP2pUpnpServiceResponse; 37 import android.os.Binder; 38 import android.os.Build; 39 import android.os.Bundle; 40 import android.os.Handler; 41 import android.os.Looper; 42 import android.os.Message; 43 import android.os.Messenger; 44 import android.os.RemoteException; 45 import android.text.TextUtils; 46 import android.util.Log; 47 48 import com.android.internal.util.AsyncChannel; 49 import com.android.internal.util.Protocol; 50 51 import dalvik.system.CloseGuard; 52 53 import java.lang.annotation.Retention; 54 import java.lang.annotation.RetentionPolicy; 55 import java.util.HashMap; 56 import java.util.List; 57 import java.util.Map; 58 59 /** 60 * This class provides the API for managing Wi-Fi peer-to-peer connectivity. This lets an 61 * application discover available peers, setup connection to peers and query for the list of peers. 62 * When a p2p connection is formed over wifi, the device continues to maintain the uplink 63 * connection over mobile or any other available network for internet connectivity on the device. 64 * 65 * <p> The API is asynchronous and responses to requests from an application are on listener 66 * callbacks provided by the application. The application needs to do an initialization with 67 * {@link #initialize} before doing any p2p operation. 68 * 69 * <p> Most application calls need a {@link ActionListener} instance for receiving callbacks 70 * {@link ActionListener#onSuccess} or {@link ActionListener#onFailure}. Action callbacks 71 * indicate whether the initiation of the action was a success or a failure. 72 * Upon failure, the reason of failure can be one of {@link #ERROR}, {@link #P2P_UNSUPPORTED} 73 * or {@link #BUSY}. 74 * 75 * <p> An application can initiate discovery of peers with {@link #discoverPeers}. An initiated 76 * discovery request from an application stays active until the device starts connecting to a peer 77 * ,forms a p2p group or there is an explicit {@link #stopPeerDiscovery}. 78 * Applications can listen to {@link #WIFI_P2P_DISCOVERY_CHANGED_ACTION} to know if a peer-to-peer 79 * discovery is running or stopped. Additionally, {@link #WIFI_P2P_PEERS_CHANGED_ACTION} indicates 80 * if the peer list has changed. 81 * 82 * <p> When an application needs to fetch the current list of peers, it can request the list 83 * of peers with {@link #requestPeers}. When the peer list is available 84 * {@link PeerListListener#onPeersAvailable} is called with the device list. 85 * 86 * <p> An application can initiate a connection request to a peer through {@link #connect}. See 87 * {@link WifiP2pConfig} for details on setting up the configuration. For communication with legacy 88 * Wi-Fi devices that do not support p2p, an app can create a group using {@link #createGroup} 89 * which creates an access point whose details can be fetched with {@link #requestGroupInfo}. 90 * 91 * <p> After a successful group formation through {@link #createGroup} or through {@link #connect}, 92 * use {@link #requestConnectionInfo} to fetch the connection details. The connection info 93 * {@link WifiP2pInfo} contains the address of the group owner 94 * {@link WifiP2pInfo#groupOwnerAddress} and a flag {@link WifiP2pInfo#isGroupOwner} to indicate 95 * if the current device is a p2p group owner. A p2p client can thus communicate with 96 * the p2p group owner through a socket connection. If the current device is the p2p group owner, 97 * {@link WifiP2pInfo#groupOwnerAddress} is anonymized unless the caller holds the 98 * {@code android.Manifest.permission#LOCAL_MAC_ADDRESS} permission. 99 * 100 * <p> With peer discovery using {@link #discoverPeers}, an application discovers the neighboring 101 * peers, but has no good way to figure out which peer to establish a connection with. For example, 102 * if a game application is interested in finding all the neighboring peers that are also running 103 * the same game, it has no way to find out until after the connection is setup. Pre-association 104 * service discovery is meant to address this issue of filtering the peers based on the running 105 * services. 106 * 107 * <p>With pre-association service discovery, an application can advertise a service for a 108 * application on a peer device prior to a connection setup between the devices. 109 * Currently, DNS based service discovery (Bonjour) and Upnp are the higher layer protocols 110 * supported. Get Bonjour resources at dns-sd.org and Upnp resources at upnp.org 111 * As an example, a video application can discover a Upnp capable media renderer 112 * prior to setting up a Wi-fi p2p connection with the device. 113 * 114 * <p> An application can advertise a Upnp or a Bonjour service with a call to 115 * {@link #addLocalService}. After a local service is added, 116 * the framework automatically responds to a peer application discovering the service prior 117 * to establishing a p2p connection. A call to {@link #removeLocalService} removes a local 118 * service and {@link #clearLocalServices} can be used to clear all local services. 119 * 120 * <p> An application that is looking for peer devices that support certain services 121 * can do so with a call to {@link #discoverServices}. Prior to initiating the discovery, 122 * application can add service discovery request with a call to {@link #addServiceRequest}, 123 * remove a service discovery request with a call to {@link #removeServiceRequest} or clear 124 * all requests with a call to {@link #clearServiceRequests}. When no service requests remain, 125 * a previously running service discovery will stop. 126 * 127 * The application is notified of a result of service discovery request through listener callbacks 128 * set through {@link #setDnsSdResponseListeners} for Bonjour or 129 * {@link #setUpnpServiceResponseListener} for Upnp. 130 * 131 * <p class="note"><strong>Note:</strong> 132 * Registering an application handler with {@link #initialize} requires the permissions 133 * {@link android.Manifest.permission#ACCESS_WIFI_STATE} and 134 * {@link android.Manifest.permission#CHANGE_WIFI_STATE} to perform any further peer-to-peer 135 * operations. 136 * 137 * {@see WifiP2pConfig} 138 * {@see WifiP2pInfo} 139 * {@see WifiP2pGroup} 140 * {@see WifiP2pDevice} 141 * {@see WifiP2pDeviceList} 142 * {@see android.net.wifi.WpsInfo} 143 */ 144 @SystemService(Context.WIFI_P2P_SERVICE) 145 public class WifiP2pManager { 146 private static final String TAG = "WifiP2pManager"; 147 /** 148 * Broadcast intent action to indicate whether Wi-Fi p2p is enabled or disabled. An 149 * extra {@link #EXTRA_WIFI_STATE} provides the state information as int. 150 * 151 * @see #EXTRA_WIFI_STATE 152 */ 153 @SdkConstant(SdkConstantType.BROADCAST_INTENT_ACTION) 154 public static final String WIFI_P2P_STATE_CHANGED_ACTION = 155 "android.net.wifi.p2p.STATE_CHANGED"; 156 157 /** 158 * The lookup key for an int that indicates whether Wi-Fi p2p is enabled or disabled. 159 * Retrieve it with {@link android.content.Intent#getIntExtra(String,int)}. 160 * 161 * @see #WIFI_P2P_STATE_DISABLED 162 * @see #WIFI_P2P_STATE_ENABLED 163 */ 164 public static final String EXTRA_WIFI_STATE = "wifi_p2p_state"; 165 166 /** @hide */ 167 @IntDef({ 168 WIFI_P2P_STATE_DISABLED, 169 WIFI_P2P_STATE_ENABLED}) 170 @Retention(RetentionPolicy.SOURCE) 171 public @interface WifiP2pState { 172 } 173 174 /** 175 * Wi-Fi p2p is disabled. 176 * 177 * @see #WIFI_P2P_STATE_CHANGED_ACTION 178 */ 179 public static final int WIFI_P2P_STATE_DISABLED = 1; 180 181 /** 182 * Wi-Fi p2p is enabled. 183 * 184 * @see #WIFI_P2P_STATE_CHANGED_ACTION 185 */ 186 public static final int WIFI_P2P_STATE_ENABLED = 2; 187 188 /** 189 * Broadcast intent action indicating that the state of Wi-Fi p2p connectivity 190 * has changed. One extra {@link #EXTRA_WIFI_P2P_INFO} provides the p2p connection info in 191 * the form of a {@link WifiP2pInfo} object. Another extra {@link #EXTRA_NETWORK_INFO} provides 192 * the network info in the form of a {@link android.net.NetworkInfo}. A third extra provides 193 * the details of the group. 194 * 195 * All of these permissions are required to receive this broadcast: 196 * {@link android.Manifest.permission#ACCESS_FINE_LOCATION} and 197 * {@link android.Manifest.permission#ACCESS_WIFI_STATE} 198 * 199 * @see #EXTRA_WIFI_P2P_INFO 200 * @see #EXTRA_NETWORK_INFO 201 * @see #EXTRA_WIFI_P2P_GROUP 202 */ 203 @SdkConstant(SdkConstantType.BROADCAST_INTENT_ACTION) 204 public static final String WIFI_P2P_CONNECTION_CHANGED_ACTION = 205 "android.net.wifi.p2p.CONNECTION_STATE_CHANGE"; 206 207 /** 208 * The lookup key for a {@link android.net.wifi.p2p.WifiP2pInfo} object 209 * Retrieve with {@link android.content.Intent#getParcelableExtra(String)}. 210 */ 211 public static final String EXTRA_WIFI_P2P_INFO = "wifiP2pInfo"; 212 213 /** 214 * The lookup key for a {@link android.net.NetworkInfo} object associated with the 215 * p2p network. Retrieve with 216 * {@link android.content.Intent#getParcelableExtra(String)}. 217 */ 218 public static final String EXTRA_NETWORK_INFO = "networkInfo"; 219 220 /** 221 * The lookup key for a {@link android.net.wifi.p2p.WifiP2pGroup} object 222 * associated with the p2p network. Retrieve with 223 * {@link android.content.Intent#getParcelableExtra(String)}. 224 */ 225 public static final String EXTRA_WIFI_P2P_GROUP = "p2pGroupInfo"; 226 227 /** 228 * Broadcast intent action indicating that the available peer list has changed. This 229 * can be sent as a result of peers being found, lost or updated. 230 * 231 * All of these permissions are required to receive this broadcast: 232 * {@link android.Manifest.permission#ACCESS_FINE_LOCATION} and 233 * {@link android.Manifest.permission#ACCESS_WIFI_STATE} 234 * 235 * <p> An extra {@link #EXTRA_P2P_DEVICE_LIST} provides the full list of 236 * current peers. The full list of peers can also be obtained any time with 237 * {@link #requestPeers}. 238 * 239 * @see #EXTRA_P2P_DEVICE_LIST 240 */ 241 @SdkConstant(SdkConstantType.BROADCAST_INTENT_ACTION) 242 public static final String WIFI_P2P_PEERS_CHANGED_ACTION = 243 "android.net.wifi.p2p.PEERS_CHANGED"; 244 245 /** 246 * The lookup key for a {@link android.net.wifi.p2p.WifiP2pDeviceList} object representing 247 * the new peer list when {@link #WIFI_P2P_PEERS_CHANGED_ACTION} broadcast is sent. 248 * 249 * <p>Retrieve with {@link android.content.Intent#getParcelableExtra(String)}. 250 */ 251 public static final String EXTRA_P2P_DEVICE_LIST = "wifiP2pDeviceList"; 252 253 /** 254 * Broadcast intent action indicating that peer discovery has either started or stopped. 255 * One extra {@link #EXTRA_DISCOVERY_STATE} indicates whether discovery has started 256 * or stopped. 257 * 258 * <p>Note that discovery will be stopped during a connection setup. If the application tries 259 * to re-initiate discovery during this time, it can fail. 260 */ 261 @SdkConstant(SdkConstantType.BROADCAST_INTENT_ACTION) 262 public static final String WIFI_P2P_DISCOVERY_CHANGED_ACTION = 263 "android.net.wifi.p2p.DISCOVERY_STATE_CHANGE"; 264 265 /** 266 * The lookup key for an int that indicates whether p2p discovery has started or stopped. 267 * Retrieve it with {@link android.content.Intent#getIntExtra(String,int)}. 268 * 269 * @see #WIFI_P2P_DISCOVERY_STARTED 270 * @see #WIFI_P2P_DISCOVERY_STOPPED 271 */ 272 public static final String EXTRA_DISCOVERY_STATE = "discoveryState"; 273 274 /** @hide */ 275 @IntDef({ 276 WIFI_P2P_DISCOVERY_STOPPED, 277 WIFI_P2P_DISCOVERY_STARTED}) 278 @Retention(RetentionPolicy.SOURCE) 279 public @interface WifiP2pDiscoveryState { 280 } 281 282 /** 283 * p2p discovery has stopped 284 * 285 * @see #WIFI_P2P_DISCOVERY_CHANGED_ACTION 286 */ 287 public static final int WIFI_P2P_DISCOVERY_STOPPED = 1; 288 289 /** 290 * p2p discovery has started 291 * 292 * @see #WIFI_P2P_DISCOVERY_CHANGED_ACTION 293 */ 294 public static final int WIFI_P2P_DISCOVERY_STARTED = 2; 295 296 /** 297 * Broadcast intent action indicating that this device details have changed. 298 * 299 * <p> An extra {@link #EXTRA_WIFI_P2P_DEVICE} provides this device details. 300 * The valid device details can also be obtained with 301 * {@link #requestDeviceInfo(Channel, DeviceInfoListener)} when p2p is enabled. 302 * To get information notifications on P2P getting enabled refers 303 * {@link #WIFI_P2P_STATE_ENABLED}. 304 * 305 * <p> The {@link #EXTRA_WIFI_P2P_DEVICE} extra contains an anonymized version of the device's 306 * MAC address. Callers holding the {@code android.Manifest.permission#LOCAL_MAC_ADDRESS} 307 * permission can use {@link #requestDeviceInfo} to obtain the actual MAC address of this 308 * device. 309 * 310 * All of these permissions are required to receive this broadcast: 311 * {@link android.Manifest.permission#ACCESS_FINE_LOCATION} and 312 * {@link android.Manifest.permission#ACCESS_WIFI_STATE} 313 * 314 * @see #EXTRA_WIFI_P2P_DEVICE 315 */ 316 @SdkConstant(SdkConstantType.BROADCAST_INTENT_ACTION) 317 public static final String WIFI_P2P_THIS_DEVICE_CHANGED_ACTION = 318 "android.net.wifi.p2p.THIS_DEVICE_CHANGED"; 319 320 /** 321 * The lookup key for a {@link android.net.wifi.p2p.WifiP2pDevice} object 322 * Retrieve with {@link android.content.Intent#getParcelableExtra(String)}. 323 */ 324 public static final String EXTRA_WIFI_P2P_DEVICE = "wifiP2pDevice"; 325 326 /** 327 * Broadcast intent action indicating that remembered persistent groups have changed. 328 * @hide 329 */ 330 public static final String WIFI_P2P_PERSISTENT_GROUPS_CHANGED_ACTION = 331 "android.net.wifi.p2p.PERSISTENT_GROUPS_CHANGED"; 332 333 /** 334 * The lookup key for a handover message returned by the WifiP2pService. 335 * @hide 336 */ 337 public static final String EXTRA_HANDOVER_MESSAGE = 338 "android.net.wifi.p2p.EXTRA_HANDOVER_MESSAGE"; 339 340 /** 341 * The lookup key for a calling package name from WifiP2pManager 342 * @hide 343 */ 344 public static final String CALLING_PACKAGE = 345 "android.net.wifi.p2p.CALLING_PACKAGE"; 346 347 /** 348 * The lookup key for a calling package binder from WifiP2pManager 349 * @hide 350 */ 351 public static final String CALLING_BINDER = 352 "android.net.wifi.p2p.CALLING_BINDER"; 353 354 IWifiP2pManager mService; 355 356 private static final int BASE = Protocol.BASE_WIFI_P2P_MANAGER; 357 358 /** @hide */ 359 public static final int DISCOVER_PEERS = BASE + 1; 360 /** @hide */ 361 public static final int DISCOVER_PEERS_FAILED = BASE + 2; 362 /** @hide */ 363 public static final int DISCOVER_PEERS_SUCCEEDED = BASE + 3; 364 365 /** @hide */ 366 public static final int STOP_DISCOVERY = BASE + 4; 367 /** @hide */ 368 public static final int STOP_DISCOVERY_FAILED = BASE + 5; 369 /** @hide */ 370 public static final int STOP_DISCOVERY_SUCCEEDED = BASE + 6; 371 372 /** @hide */ 373 public static final int CONNECT = BASE + 7; 374 /** @hide */ 375 public static final int CONNECT_FAILED = BASE + 8; 376 /** @hide */ 377 public static final int CONNECT_SUCCEEDED = BASE + 9; 378 379 /** @hide */ 380 public static final int CANCEL_CONNECT = BASE + 10; 381 /** @hide */ 382 public static final int CANCEL_CONNECT_FAILED = BASE + 11; 383 /** @hide */ 384 public static final int CANCEL_CONNECT_SUCCEEDED = BASE + 12; 385 386 /** @hide */ 387 @UnsupportedAppUsage 388 public static final int CREATE_GROUP = BASE + 13; 389 /** @hide */ 390 public static final int CREATE_GROUP_FAILED = BASE + 14; 391 /** @hide */ 392 public static final int CREATE_GROUP_SUCCEEDED = BASE + 15; 393 394 /** @hide */ 395 public static final int REMOVE_GROUP = BASE + 16; 396 /** @hide */ 397 public static final int REMOVE_GROUP_FAILED = BASE + 17; 398 /** @hide */ 399 public static final int REMOVE_GROUP_SUCCEEDED = BASE + 18; 400 401 /** @hide */ 402 public static final int REQUEST_PEERS = BASE + 19; 403 /** @hide */ 404 public static final int RESPONSE_PEERS = BASE + 20; 405 406 /** @hide */ 407 public static final int REQUEST_CONNECTION_INFO = BASE + 21; 408 /** @hide */ 409 public static final int RESPONSE_CONNECTION_INFO = BASE + 22; 410 411 /** @hide */ 412 public static final int REQUEST_GROUP_INFO = BASE + 23; 413 /** @hide */ 414 public static final int RESPONSE_GROUP_INFO = BASE + 24; 415 416 /** @hide */ 417 public static final int ADD_LOCAL_SERVICE = BASE + 28; 418 /** @hide */ 419 public static final int ADD_LOCAL_SERVICE_FAILED = BASE + 29; 420 /** @hide */ 421 public static final int ADD_LOCAL_SERVICE_SUCCEEDED = BASE + 30; 422 423 /** @hide */ 424 public static final int REMOVE_LOCAL_SERVICE = BASE + 31; 425 /** @hide */ 426 public static final int REMOVE_LOCAL_SERVICE_FAILED = BASE + 32; 427 /** @hide */ 428 public static final int REMOVE_LOCAL_SERVICE_SUCCEEDED = BASE + 33; 429 430 /** @hide */ 431 public static final int CLEAR_LOCAL_SERVICES = BASE + 34; 432 /** @hide */ 433 public static final int CLEAR_LOCAL_SERVICES_FAILED = BASE + 35; 434 /** @hide */ 435 public static final int CLEAR_LOCAL_SERVICES_SUCCEEDED = BASE + 36; 436 437 /** @hide */ 438 public static final int ADD_SERVICE_REQUEST = BASE + 37; 439 /** @hide */ 440 public static final int ADD_SERVICE_REQUEST_FAILED = BASE + 38; 441 /** @hide */ 442 public static final int ADD_SERVICE_REQUEST_SUCCEEDED = BASE + 39; 443 444 /** @hide */ 445 public static final int REMOVE_SERVICE_REQUEST = BASE + 40; 446 /** @hide */ 447 public static final int REMOVE_SERVICE_REQUEST_FAILED = BASE + 41; 448 /** @hide */ 449 public static final int REMOVE_SERVICE_REQUEST_SUCCEEDED = BASE + 42; 450 451 /** @hide */ 452 public static final int CLEAR_SERVICE_REQUESTS = BASE + 43; 453 /** @hide */ 454 public static final int CLEAR_SERVICE_REQUESTS_FAILED = BASE + 44; 455 /** @hide */ 456 public static final int CLEAR_SERVICE_REQUESTS_SUCCEEDED = BASE + 45; 457 458 /** @hide */ 459 public static final int DISCOVER_SERVICES = BASE + 46; 460 /** @hide */ 461 public static final int DISCOVER_SERVICES_FAILED = BASE + 47; 462 /** @hide */ 463 public static final int DISCOVER_SERVICES_SUCCEEDED = BASE + 48; 464 465 /** @hide */ 466 public static final int PING = BASE + 49; 467 468 /** @hide */ 469 public static final int RESPONSE_SERVICE = BASE + 50; 470 471 /** @hide */ 472 public static final int SET_DEVICE_NAME = BASE + 51; 473 /** @hide */ 474 public static final int SET_DEVICE_NAME_FAILED = BASE + 52; 475 /** @hide */ 476 public static final int SET_DEVICE_NAME_SUCCEEDED = BASE + 53; 477 478 /** @hide */ 479 public static final int DELETE_PERSISTENT_GROUP = BASE + 54; 480 /** @hide */ 481 public static final int DELETE_PERSISTENT_GROUP_FAILED = BASE + 55; 482 /** @hide */ 483 public static final int DELETE_PERSISTENT_GROUP_SUCCEEDED = BASE + 56; 484 485 /** @hide */ 486 public static final int REQUEST_PERSISTENT_GROUP_INFO = BASE + 57; 487 /** @hide */ 488 public static final int RESPONSE_PERSISTENT_GROUP_INFO = BASE + 58; 489 490 /** @hide */ 491 public static final int SET_WFD_INFO = BASE + 59; 492 /** @hide */ 493 public static final int SET_WFD_INFO_FAILED = BASE + 60; 494 /** @hide */ 495 public static final int SET_WFD_INFO_SUCCEEDED = BASE + 61; 496 497 /** @hide */ 498 public static final int START_WPS = BASE + 62; 499 /** @hide */ 500 public static final int START_WPS_FAILED = BASE + 63; 501 /** @hide */ 502 public static final int START_WPS_SUCCEEDED = BASE + 64; 503 504 /** @hide */ 505 public static final int START_LISTEN = BASE + 65; 506 /** @hide */ 507 public static final int START_LISTEN_FAILED = BASE + 66; 508 /** @hide */ 509 public static final int START_LISTEN_SUCCEEDED = BASE + 67; 510 511 /** @hide */ 512 public static final int STOP_LISTEN = BASE + 68; 513 /** @hide */ 514 public static final int STOP_LISTEN_FAILED = BASE + 69; 515 /** @hide */ 516 public static final int STOP_LISTEN_SUCCEEDED = BASE + 70; 517 518 /** @hide */ 519 public static final int SET_CHANNEL = BASE + 71; 520 /** @hide */ 521 public static final int SET_CHANNEL_FAILED = BASE + 72; 522 /** @hide */ 523 public static final int SET_CHANNEL_SUCCEEDED = BASE + 73; 524 525 /** @hide */ 526 public static final int GET_HANDOVER_REQUEST = BASE + 75; 527 /** @hide */ 528 public static final int GET_HANDOVER_SELECT = BASE + 76; 529 /** @hide */ 530 public static final int RESPONSE_GET_HANDOVER_MESSAGE = BASE + 77; 531 /** @hide */ 532 public static final int INITIATOR_REPORT_NFC_HANDOVER = BASE + 78; 533 /** @hide */ 534 public static final int RESPONDER_REPORT_NFC_HANDOVER = BASE + 79; 535 /** @hide */ 536 public static final int REPORT_NFC_HANDOVER_SUCCEEDED = BASE + 80; 537 /** @hide */ 538 public static final int REPORT_NFC_HANDOVER_FAILED = BASE + 81; 539 540 /** @hide */ 541 public static final int FACTORY_RESET = BASE + 82; 542 /** @hide */ 543 public static final int FACTORY_RESET_FAILED = BASE + 83; 544 /** @hide */ 545 public static final int FACTORY_RESET_SUCCEEDED = BASE + 84; 546 547 /** @hide */ 548 public static final int REQUEST_ONGOING_PEER_CONFIG = BASE + 85; 549 /** @hide */ 550 public static final int RESPONSE_ONGOING_PEER_CONFIG = BASE + 86; 551 /** @hide */ 552 public static final int SET_ONGOING_PEER_CONFIG = BASE + 87; 553 /** @hide */ 554 public static final int SET_ONGOING_PEER_CONFIG_FAILED = BASE + 88; 555 /** @hide */ 556 public static final int SET_ONGOING_PEER_CONFIG_SUCCEEDED = BASE + 89; 557 558 /** @hide */ 559 public static final int REQUEST_P2P_STATE = BASE + 90; 560 /** @hide */ 561 public static final int RESPONSE_P2P_STATE = BASE + 91; 562 563 /** @hide */ 564 public static final int REQUEST_DISCOVERY_STATE = BASE + 92; 565 /** @hide */ 566 public static final int RESPONSE_DISCOVERY_STATE = BASE + 93; 567 568 /** @hide */ 569 public static final int REQUEST_NETWORK_INFO = BASE + 94; 570 /** @hide */ 571 public static final int RESPONSE_NETWORK_INFO = BASE + 95; 572 573 /** @hide */ 574 public static final int UPDATE_CHANNEL_INFO = BASE + 96; 575 576 /** @hide */ 577 public static final int REQUEST_DEVICE_INFO = BASE + 97; 578 /** @hide */ 579 public static final int RESPONSE_DEVICE_INFO = BASE + 98; 580 581 /** 582 * Create a new WifiP2pManager instance. Applications use 583 * {@link android.content.Context#getSystemService Context.getSystemService()} to retrieve 584 * the standard {@link android.content.Context#WIFI_P2P_SERVICE Context.WIFI_P2P_SERVICE}. 585 * @param service the Binder interface 586 * @hide - hide this because it takes in a parameter of type IWifiP2pManager, which 587 * is a system private class. 588 */ 589 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023) WifiP2pManager(IWifiP2pManager service)590 public WifiP2pManager(IWifiP2pManager service) { 591 mService = service; 592 } 593 594 /** 595 * Passed with {@link ActionListener#onFailure}. 596 * Indicates that the operation failed due to an internal error. 597 */ 598 public static final int ERROR = 0; 599 600 /** 601 * Passed with {@link ActionListener#onFailure}. 602 * Indicates that the operation failed because p2p is unsupported on the device. 603 */ 604 public static final int P2P_UNSUPPORTED = 1; 605 606 /** 607 * Passed with {@link ActionListener#onFailure}. 608 * Indicates that the operation failed because the framework is busy and 609 * unable to service the request 610 */ 611 public static final int BUSY = 2; 612 613 /** 614 * Passed with {@link ActionListener#onFailure}. 615 * Indicates that the {@link #discoverServices} failed because no service 616 * requests are added. Use {@link #addServiceRequest} to add a service 617 * request. 618 */ 619 public static final int NO_SERVICE_REQUESTS = 3; 620 621 /** Interface for callback invocation when framework channel is lost */ 622 public interface ChannelListener { 623 /** 624 * The channel to the framework has been disconnected. 625 * Application could try re-initializing using {@link #initialize} 626 */ onChannelDisconnected()627 public void onChannelDisconnected(); 628 } 629 630 /** Interface for callback invocation on an application action */ 631 public interface ActionListener { 632 /** The operation succeeded */ onSuccess()633 public void onSuccess(); 634 /** 635 * The operation failed 636 * @param reason The reason for failure could be one of {@link #P2P_UNSUPPORTED}, 637 * {@link #ERROR} or {@link #BUSY} 638 */ onFailure(int reason)639 public void onFailure(int reason); 640 } 641 642 /** Interface for callback invocation when peer list is available */ 643 public interface PeerListListener { 644 /** 645 * The requested peer list is available 646 * @param peers List of available peers 647 */ onPeersAvailable(WifiP2pDeviceList peers)648 public void onPeersAvailable(WifiP2pDeviceList peers); 649 } 650 651 /** Interface for callback invocation when connection info is available */ 652 public interface ConnectionInfoListener { 653 /** 654 * The requested connection info is available 655 * @param info Wi-Fi p2p connection info 656 */ onConnectionInfoAvailable(WifiP2pInfo info)657 public void onConnectionInfoAvailable(WifiP2pInfo info); 658 } 659 660 /** Interface for callback invocation when group info is available */ 661 public interface GroupInfoListener { 662 /** 663 * The requested p2p group info is available 664 * @param group Wi-Fi p2p group info 665 */ onGroupInfoAvailable(WifiP2pGroup group)666 public void onGroupInfoAvailable(WifiP2pGroup group); 667 } 668 669 /** 670 * Interface for callback invocation when service discovery response other than 671 * Upnp or Bonjour is received 672 */ 673 public interface ServiceResponseListener { 674 675 /** 676 * The requested service response is available. 677 * 678 * @param protocolType protocol type. currently only 679 * {@link WifiP2pServiceInfo#SERVICE_TYPE_VENDOR_SPECIFIC}. 680 * @param responseData service discovery response data based on the requested 681 * service protocol type. The format depends on the service type. 682 * @param srcDevice source device. 683 */ onServiceAvailable(int protocolType, byte[] responseData, WifiP2pDevice srcDevice)684 public void onServiceAvailable(int protocolType, 685 byte[] responseData, WifiP2pDevice srcDevice); 686 } 687 688 /** 689 * Interface for callback invocation when Bonjour service discovery response 690 * is received 691 */ 692 public interface DnsSdServiceResponseListener { 693 694 /** 695 * The requested Bonjour service response is available. 696 * 697 * <p>This function is invoked when the device with the specified Bonjour 698 * registration type returned the instance name. 699 * @param instanceName instance name.<br> 700 * e.g) "MyPrinter". 701 * @param registrationType <br> 702 * e.g) "_ipp._tcp.local." 703 * @param srcDevice source device. 704 */ onDnsSdServiceAvailable(String instanceName, String registrationType, WifiP2pDevice srcDevice)705 public void onDnsSdServiceAvailable(String instanceName, 706 String registrationType, WifiP2pDevice srcDevice); 707 708 } 709 710 /** 711 * Interface for callback invocation when Bonjour TXT record is available 712 * for a service 713 */ 714 public interface DnsSdTxtRecordListener { 715 /** 716 * The requested Bonjour service response is available. 717 * 718 * <p>This function is invoked when the device with the specified full 719 * service domain service returned TXT record. 720 * 721 * @param fullDomainName full domain name. <br> 722 * e.g) "MyPrinter._ipp._tcp.local.". 723 * @param txtRecordMap TXT record data as a map of key/value pairs 724 * @param srcDevice source device. 725 */ onDnsSdTxtRecordAvailable(String fullDomainName, Map<String, String> txtRecordMap, WifiP2pDevice srcDevice)726 public void onDnsSdTxtRecordAvailable(String fullDomainName, 727 Map<String, String> txtRecordMap, 728 WifiP2pDevice srcDevice); 729 } 730 731 /** 732 * Interface for callback invocation when upnp service discovery response 733 * is received 734 * */ 735 public interface UpnpServiceResponseListener { 736 737 /** 738 * The requested upnp service response is available. 739 * 740 * <p>This function is invoked when the specified device or service is found. 741 * 742 * @param uniqueServiceNames The list of unique service names.<br> 743 * e.g) uuid:6859dede-8574-59ab-9332-123456789012::urn:schemas-upnp-org:device: 744 * MediaServer:1 745 * @param srcDevice source device. 746 */ onUpnpServiceAvailable(List<String> uniqueServiceNames, WifiP2pDevice srcDevice)747 public void onUpnpServiceAvailable(List<String> uniqueServiceNames, 748 WifiP2pDevice srcDevice); 749 } 750 751 752 /** Interface for callback invocation when stored group info list is available {@hide}*/ 753 public interface PersistentGroupInfoListener { 754 /** 755 * The requested stored p2p group info list is available 756 * @param groups Wi-Fi p2p group info list 757 */ onPersistentGroupInfoAvailable(WifiP2pGroupList groups)758 public void onPersistentGroupInfoAvailable(WifiP2pGroupList groups); 759 } 760 761 /** 762 * Interface for callback invocation when Handover Request or Select Message is available 763 * @hide 764 */ 765 public interface HandoverMessageListener { onHandoverMessageAvailable(String handoverMessage)766 public void onHandoverMessageAvailable(String handoverMessage); 767 } 768 769 /** Interface for callback invocation when p2p state is available 770 * in response to {@link #requestP2pState}. 771 */ 772 public interface P2pStateListener { 773 /** 774 * The requested p2p state is available. 775 * @param state Wi-Fi p2p state 776 * @see #WIFI_P2P_STATE_DISABLED 777 * @see #WIFI_P2P_STATE_ENABLED 778 */ onP2pStateAvailable(@ifiP2pState int state)779 void onP2pStateAvailable(@WifiP2pState int state); 780 } 781 782 /** Interface for callback invocation when p2p state is available 783 * in response to {@link #requestDiscoveryState}. 784 */ 785 public interface DiscoveryStateListener { 786 /** 787 * The requested p2p discovery state is available. 788 * @param state Wi-Fi p2p discovery state 789 * @see #WIFI_P2P_DISCOVERY_STARTED 790 * @see #WIFI_P2P_DISCOVERY_STOPPED 791 */ onDiscoveryStateAvailable(@ifiP2pDiscoveryState int state)792 void onDiscoveryStateAvailable(@WifiP2pDiscoveryState int state); 793 } 794 795 /** Interface for callback invocation when {@link android.net.NetworkInfo} is available 796 * in response to {@link #requestNetworkInfo}. 797 */ 798 public interface NetworkInfoListener { 799 /** 800 * The requested {@link android.net.NetworkInfo} is available 801 * @param networkInfo Wi-Fi p2p {@link android.net.NetworkInfo} 802 */ onNetworkInfoAvailable(@onNull NetworkInfo networkInfo)803 void onNetworkInfoAvailable(@NonNull NetworkInfo networkInfo); 804 } 805 806 /** 807 * Interface for callback invocation when ongoing peer info is available 808 * @hide 809 */ 810 public interface OngoingPeerInfoListener { 811 /** 812 * The requested ongoing WifiP2pConfig is available 813 * @param peerConfig WifiP2pConfig for current connecting session 814 */ onOngoingPeerAvailable(WifiP2pConfig peerConfig)815 void onOngoingPeerAvailable(WifiP2pConfig peerConfig); 816 } 817 818 /** Interface for callback invocation when {@link android.net.wifi.p2p.WifiP2pDevice} 819 * is available in response to {@link #requestDeviceInfo(Channel, DeviceInfoListener)}. 820 */ 821 public interface DeviceInfoListener { 822 /** 823 * The requested {@link android.net.wifi.p2p.WifiP2pDevice} is available. 824 * @param wifiP2pDevice Wi-Fi p2p {@link android.net.wifi.p2p.WifiP2pDevice} 825 */ onDeviceInfoAvailable(@ullable WifiP2pDevice wifiP2pDevice)826 void onDeviceInfoAvailable(@Nullable WifiP2pDevice wifiP2pDevice); 827 } 828 829 /** 830 * A channel that connects the application to the Wifi p2p framework. 831 * Most p2p operations require a Channel as an argument. An instance of Channel is obtained 832 * by doing a call on {@link #initialize} 833 */ 834 public static class Channel implements AutoCloseable { 835 /** @hide */ Channel(Context context, Looper looper, ChannelListener l, Binder binder, WifiP2pManager p2pManager)836 public Channel(Context context, Looper looper, ChannelListener l, Binder binder, 837 WifiP2pManager p2pManager) { 838 mAsyncChannel = new AsyncChannel(); 839 mHandler = new P2pHandler(looper); 840 mChannelListener = l; 841 mContext = context; 842 mBinder = binder; 843 mP2pManager = p2pManager; 844 845 mCloseGuard.open("close"); 846 } 847 private final static int INVALID_LISTENER_KEY = 0; 848 private final WifiP2pManager mP2pManager; 849 private ChannelListener mChannelListener; 850 private ServiceResponseListener mServRspListener; 851 private DnsSdServiceResponseListener mDnsSdServRspListener; 852 private DnsSdTxtRecordListener mDnsSdTxtListener; 853 private UpnpServiceResponseListener mUpnpServRspListener; 854 private HashMap<Integer, Object> mListenerMap = new HashMap<Integer, Object>(); 855 private final Object mListenerMapLock = new Object(); 856 private int mListenerKey = 0; 857 858 private final CloseGuard mCloseGuard = CloseGuard.get(); 859 860 /** 861 * Close the current P2P connection and indicate to the P2P service that connections 862 * created by the app can be removed. 863 */ close()864 public void close() { 865 if (mP2pManager == null) { 866 Log.w(TAG, "Channel.close(): Null mP2pManager!?"); 867 } else { 868 try { 869 mP2pManager.mService.close(mBinder); 870 } catch (RemoteException e) { 871 throw e.rethrowFromSystemServer(); 872 } 873 } 874 875 mAsyncChannel.disconnect(); 876 mCloseGuard.close(); 877 } 878 879 /** @hide */ 880 @Override finalize()881 protected void finalize() throws Throwable { 882 try { 883 if (mCloseGuard != null) { 884 mCloseGuard.warnIfOpen(); 885 } 886 887 close(); 888 } finally { 889 super.finalize(); 890 } 891 } 892 893 /* package */ final Binder mBinder; 894 895 @UnsupportedAppUsage 896 private AsyncChannel mAsyncChannel; 897 private P2pHandler mHandler; 898 Context mContext; 899 class P2pHandler extends Handler { P2pHandler(Looper looper)900 P2pHandler(Looper looper) { 901 super(looper); 902 } 903 904 @Override handleMessage(Message message)905 public void handleMessage(Message message) { 906 Object listener = getListener(message.arg2); 907 switch (message.what) { 908 case AsyncChannel.CMD_CHANNEL_DISCONNECTED: 909 if (mChannelListener != null) { 910 mChannelListener.onChannelDisconnected(); 911 mChannelListener = null; 912 } 913 break; 914 /* ActionListeners grouped together */ 915 case DISCOVER_PEERS_FAILED: 916 case STOP_DISCOVERY_FAILED: 917 case DISCOVER_SERVICES_FAILED: 918 case CONNECT_FAILED: 919 case CANCEL_CONNECT_FAILED: 920 case CREATE_GROUP_FAILED: 921 case REMOVE_GROUP_FAILED: 922 case ADD_LOCAL_SERVICE_FAILED: 923 case REMOVE_LOCAL_SERVICE_FAILED: 924 case CLEAR_LOCAL_SERVICES_FAILED: 925 case ADD_SERVICE_REQUEST_FAILED: 926 case REMOVE_SERVICE_REQUEST_FAILED: 927 case CLEAR_SERVICE_REQUESTS_FAILED: 928 case SET_DEVICE_NAME_FAILED: 929 case DELETE_PERSISTENT_GROUP_FAILED: 930 case SET_WFD_INFO_FAILED: 931 case START_WPS_FAILED: 932 case START_LISTEN_FAILED: 933 case STOP_LISTEN_FAILED: 934 case SET_CHANNEL_FAILED: 935 case REPORT_NFC_HANDOVER_FAILED: 936 case FACTORY_RESET_FAILED: 937 case SET_ONGOING_PEER_CONFIG_FAILED: 938 if (listener != null) { 939 ((ActionListener) listener).onFailure(message.arg1); 940 } 941 break; 942 /* ActionListeners grouped together */ 943 case DISCOVER_PEERS_SUCCEEDED: 944 case STOP_DISCOVERY_SUCCEEDED: 945 case DISCOVER_SERVICES_SUCCEEDED: 946 case CONNECT_SUCCEEDED: 947 case CANCEL_CONNECT_SUCCEEDED: 948 case CREATE_GROUP_SUCCEEDED: 949 case REMOVE_GROUP_SUCCEEDED: 950 case ADD_LOCAL_SERVICE_SUCCEEDED: 951 case REMOVE_LOCAL_SERVICE_SUCCEEDED: 952 case CLEAR_LOCAL_SERVICES_SUCCEEDED: 953 case ADD_SERVICE_REQUEST_SUCCEEDED: 954 case REMOVE_SERVICE_REQUEST_SUCCEEDED: 955 case CLEAR_SERVICE_REQUESTS_SUCCEEDED: 956 case SET_DEVICE_NAME_SUCCEEDED: 957 case DELETE_PERSISTENT_GROUP_SUCCEEDED: 958 case SET_WFD_INFO_SUCCEEDED: 959 case START_WPS_SUCCEEDED: 960 case START_LISTEN_SUCCEEDED: 961 case STOP_LISTEN_SUCCEEDED: 962 case SET_CHANNEL_SUCCEEDED: 963 case REPORT_NFC_HANDOVER_SUCCEEDED: 964 case FACTORY_RESET_SUCCEEDED: 965 case SET_ONGOING_PEER_CONFIG_SUCCEEDED: 966 if (listener != null) { 967 ((ActionListener) listener).onSuccess(); 968 } 969 break; 970 case RESPONSE_PEERS: 971 WifiP2pDeviceList peers = (WifiP2pDeviceList) message.obj; 972 if (listener != null) { 973 ((PeerListListener) listener).onPeersAvailable(peers); 974 } 975 break; 976 case RESPONSE_CONNECTION_INFO: 977 WifiP2pInfo wifiP2pInfo = (WifiP2pInfo) message.obj; 978 if (listener != null) { 979 ((ConnectionInfoListener) listener).onConnectionInfoAvailable(wifiP2pInfo); 980 } 981 break; 982 case RESPONSE_GROUP_INFO: 983 WifiP2pGroup group = (WifiP2pGroup) message.obj; 984 if (listener != null) { 985 ((GroupInfoListener) listener).onGroupInfoAvailable(group); 986 } 987 break; 988 case RESPONSE_SERVICE: 989 WifiP2pServiceResponse resp = (WifiP2pServiceResponse) message.obj; 990 handleServiceResponse(resp); 991 break; 992 case RESPONSE_PERSISTENT_GROUP_INFO: 993 WifiP2pGroupList groups = (WifiP2pGroupList) message.obj; 994 if (listener != null) { 995 ((PersistentGroupInfoListener) listener). 996 onPersistentGroupInfoAvailable(groups); 997 } 998 break; 999 case RESPONSE_GET_HANDOVER_MESSAGE: 1000 Bundle handoverBundle = (Bundle) message.obj; 1001 if (listener != null) { 1002 String handoverMessage = handoverBundle != null 1003 ? handoverBundle.getString(EXTRA_HANDOVER_MESSAGE) 1004 : null; 1005 ((HandoverMessageListener) listener) 1006 .onHandoverMessageAvailable(handoverMessage); 1007 } 1008 break; 1009 case RESPONSE_ONGOING_PEER_CONFIG: 1010 WifiP2pConfig peerConfig = (WifiP2pConfig) message.obj; 1011 if (listener != null) { 1012 ((OngoingPeerInfoListener) listener) 1013 .onOngoingPeerAvailable(peerConfig); 1014 } 1015 break; 1016 case RESPONSE_P2P_STATE: 1017 if (listener != null) { 1018 ((P2pStateListener) listener) 1019 .onP2pStateAvailable(message.arg1); 1020 } 1021 break; 1022 case RESPONSE_DISCOVERY_STATE: 1023 if (listener != null) { 1024 ((DiscoveryStateListener) listener) 1025 .onDiscoveryStateAvailable(message.arg1); 1026 } 1027 break; 1028 case RESPONSE_NETWORK_INFO: 1029 if (listener != null) { 1030 ((NetworkInfoListener) listener) 1031 .onNetworkInfoAvailable((NetworkInfo) message.obj); 1032 } 1033 break; 1034 case RESPONSE_DEVICE_INFO: 1035 if (listener != null) { 1036 ((DeviceInfoListener) listener) 1037 .onDeviceInfoAvailable((WifiP2pDevice) message.obj); 1038 } 1039 break; 1040 default: 1041 Log.d(TAG, "Ignored " + message); 1042 break; 1043 } 1044 } 1045 } 1046 handleServiceResponse(WifiP2pServiceResponse resp)1047 private void handleServiceResponse(WifiP2pServiceResponse resp) { 1048 if (resp instanceof WifiP2pDnsSdServiceResponse) { 1049 handleDnsSdServiceResponse((WifiP2pDnsSdServiceResponse)resp); 1050 } else if (resp instanceof WifiP2pUpnpServiceResponse) { 1051 if (mUpnpServRspListener != null) { 1052 handleUpnpServiceResponse((WifiP2pUpnpServiceResponse)resp); 1053 } 1054 } else { 1055 if (mServRspListener != null) { 1056 mServRspListener.onServiceAvailable(resp.getServiceType(), 1057 resp.getRawData(), resp.getSrcDevice()); 1058 } 1059 } 1060 } 1061 handleUpnpServiceResponse(WifiP2pUpnpServiceResponse resp)1062 private void handleUpnpServiceResponse(WifiP2pUpnpServiceResponse resp) { 1063 mUpnpServRspListener.onUpnpServiceAvailable(resp.getUniqueServiceNames(), 1064 resp.getSrcDevice()); 1065 } 1066 handleDnsSdServiceResponse(WifiP2pDnsSdServiceResponse resp)1067 private void handleDnsSdServiceResponse(WifiP2pDnsSdServiceResponse resp) { 1068 if (resp.getDnsType() == WifiP2pDnsSdServiceInfo.DNS_TYPE_PTR) { 1069 if (mDnsSdServRspListener != null) { 1070 mDnsSdServRspListener.onDnsSdServiceAvailable( 1071 resp.getInstanceName(), 1072 resp.getDnsQueryName(), 1073 resp.getSrcDevice()); 1074 } 1075 } else if (resp.getDnsType() == WifiP2pDnsSdServiceInfo.DNS_TYPE_TXT) { 1076 if (mDnsSdTxtListener != null) { 1077 mDnsSdTxtListener.onDnsSdTxtRecordAvailable( 1078 resp.getDnsQueryName(), 1079 resp.getTxtRecord(), 1080 resp.getSrcDevice()); 1081 } 1082 } else { 1083 Log.e(TAG, "Unhandled resp " + resp); 1084 } 1085 } 1086 1087 @UnsupportedAppUsage putListener(Object listener)1088 private int putListener(Object listener) { 1089 if (listener == null) return INVALID_LISTENER_KEY; 1090 int key; 1091 synchronized (mListenerMapLock) { 1092 do { 1093 key = mListenerKey++; 1094 } while (key == INVALID_LISTENER_KEY); 1095 mListenerMap.put(key, listener); 1096 } 1097 return key; 1098 } 1099 getListener(int key)1100 private Object getListener(int key) { 1101 if (key == INVALID_LISTENER_KEY) return null; 1102 synchronized (mListenerMapLock) { 1103 return mListenerMap.remove(key); 1104 } 1105 } 1106 } 1107 checkChannel(Channel c)1108 private static void checkChannel(Channel c) { 1109 if (c == null) throw new IllegalArgumentException("Channel needs to be initialized"); 1110 } 1111 checkServiceInfo(WifiP2pServiceInfo info)1112 private static void checkServiceInfo(WifiP2pServiceInfo info) { 1113 if (info == null) throw new IllegalArgumentException("service info is null"); 1114 } 1115 checkServiceRequest(WifiP2pServiceRequest req)1116 private static void checkServiceRequest(WifiP2pServiceRequest req) { 1117 if (req == null) throw new IllegalArgumentException("service request is null"); 1118 } 1119 checkP2pConfig(WifiP2pConfig c)1120 private static void checkP2pConfig(WifiP2pConfig c) { 1121 if (c == null) throw new IllegalArgumentException("config cannot be null"); 1122 if (TextUtils.isEmpty(c.deviceAddress)) { 1123 throw new IllegalArgumentException("deviceAddress cannot be empty"); 1124 } 1125 } 1126 1127 /** 1128 * Registers the application with the Wi-Fi framework. This function 1129 * must be the first to be called before any p2p operations are performed. 1130 * 1131 * @param srcContext is the context of the source 1132 * @param srcLooper is the Looper on which the callbacks are receivied 1133 * @param listener for callback at loss of framework communication. Can be null. 1134 * @return Channel instance that is necessary for performing any further p2p operations 1135 */ initialize(Context srcContext, Looper srcLooper, ChannelListener listener)1136 public Channel initialize(Context srcContext, Looper srcLooper, ChannelListener listener) { 1137 Binder binder = new Binder(); 1138 Channel channel = initalizeChannel(srcContext, srcLooper, listener, getMessenger(binder), 1139 binder); 1140 return channel; 1141 } 1142 1143 /** 1144 * Registers the application with the Wi-Fi framework. Enables system-only functionality. 1145 * @hide 1146 */ initializeInternal(Context srcContext, Looper srcLooper, ChannelListener listener)1147 public Channel initializeInternal(Context srcContext, Looper srcLooper, 1148 ChannelListener listener) { 1149 return initalizeChannel(srcContext, srcLooper, listener, getP2pStateMachineMessenger(), 1150 null); 1151 } 1152 initalizeChannel(Context srcContext, Looper srcLooper, ChannelListener listener, Messenger messenger, Binder binder)1153 private Channel initalizeChannel(Context srcContext, Looper srcLooper, ChannelListener listener, 1154 Messenger messenger, Binder binder) { 1155 if (messenger == null) return null; 1156 1157 Channel c = new Channel(srcContext, srcLooper, listener, binder, this); 1158 if (c.mAsyncChannel.connectSync(srcContext, c.mHandler, messenger) 1159 == AsyncChannel.STATUS_SUCCESSFUL) { 1160 Bundle bundle = new Bundle(); 1161 bundle.putString(CALLING_PACKAGE, c.mContext.getOpPackageName()); 1162 bundle.putBinder(CALLING_BINDER, binder); 1163 c.mAsyncChannel.sendMessage(UPDATE_CHANNEL_INFO, 0, 1164 c.putListener(null), bundle); 1165 return c; 1166 } else { 1167 c.close(); 1168 return null; 1169 } 1170 } 1171 1172 /** 1173 * Initiate peer discovery. A discovery process involves scanning for available Wi-Fi peers 1174 * for the purpose of establishing a connection. 1175 * 1176 * <p> The function call immediately returns after sending a discovery request 1177 * to the framework. The application is notified of a success or failure to initiate 1178 * discovery through listener callbacks {@link ActionListener#onSuccess} or 1179 * {@link ActionListener#onFailure}. 1180 * 1181 * <p> The discovery remains active until a connection is initiated or 1182 * a p2p group is formed. Register for {@link #WIFI_P2P_PEERS_CHANGED_ACTION} intent to 1183 * determine when the framework notifies of a change as peers are discovered. 1184 * 1185 * <p> Upon receiving a {@link #WIFI_P2P_PEERS_CHANGED_ACTION} intent, an application 1186 * can request for the list of peers using {@link #requestPeers}. 1187 * 1188 * @param c is the channel created at {@link #initialize} 1189 * @param listener for callbacks on success or failure. Can be null. 1190 */ 1191 @RequiresPermission(android.Manifest.permission.ACCESS_FINE_LOCATION) discoverPeers(Channel c, ActionListener listener)1192 public void discoverPeers(Channel c, ActionListener listener) { 1193 checkChannel(c); 1194 c.mAsyncChannel.sendMessage(DISCOVER_PEERS, 0, c.putListener(listener)); 1195 } 1196 1197 /** 1198 * Stop an ongoing peer discovery 1199 * 1200 * <p> The function call immediately returns after sending a stop request 1201 * to the framework. The application is notified of a success or failure to initiate 1202 * stop through listener callbacks {@link ActionListener#onSuccess} or 1203 * {@link ActionListener#onFailure}. 1204 * 1205 * @param c is the channel created at {@link #initialize} 1206 * @param listener for callbacks on success or failure. Can be null. 1207 */ stopPeerDiscovery(Channel c, ActionListener listener)1208 public void stopPeerDiscovery(Channel c, ActionListener listener) { 1209 checkChannel(c); 1210 c.mAsyncChannel.sendMessage(STOP_DISCOVERY, 0, c.putListener(listener)); 1211 } 1212 1213 /** 1214 * Start a p2p connection to a device with the specified configuration. 1215 * 1216 * <p> The function call immediately returns after sending a connection request 1217 * to the framework. The application is notified of a success or failure to initiate 1218 * connect through listener callbacks {@link ActionListener#onSuccess} or 1219 * {@link ActionListener#onFailure}. 1220 * 1221 * <p> Register for {@link #WIFI_P2P_CONNECTION_CHANGED_ACTION} intent to 1222 * determine when the framework notifies of a change in connectivity. 1223 * 1224 * <p> If the current device is not part of a p2p group, a connect request initiates 1225 * a group negotiation with the peer. 1226 * 1227 * <p> If the current device is part of an existing p2p group or has created 1228 * a p2p group with {@link #createGroup}, an invitation to join the group is sent to 1229 * the peer device. 1230 * 1231 * @param c is the channel created at {@link #initialize} 1232 * @param config options as described in {@link WifiP2pConfig} class 1233 * @param listener for callbacks on success or failure. Can be null. 1234 */ 1235 @RequiresPermission(android.Manifest.permission.ACCESS_FINE_LOCATION) connect(Channel c, WifiP2pConfig config, ActionListener listener)1236 public void connect(Channel c, WifiP2pConfig config, ActionListener listener) { 1237 checkChannel(c); 1238 checkP2pConfig(config); 1239 c.mAsyncChannel.sendMessage(CONNECT, 0, c.putListener(listener), config); 1240 } 1241 1242 /** 1243 * Cancel any ongoing p2p group negotiation 1244 * 1245 * <p> The function call immediately returns after sending a connection cancellation request 1246 * to the framework. The application is notified of a success or failure to initiate 1247 * cancellation through listener callbacks {@link ActionListener#onSuccess} or 1248 * {@link ActionListener#onFailure}. 1249 * 1250 * @param c is the channel created at {@link #initialize} 1251 * @param listener for callbacks on success or failure. Can be null. 1252 */ cancelConnect(Channel c, ActionListener listener)1253 public void cancelConnect(Channel c, ActionListener listener) { 1254 checkChannel(c); 1255 c.mAsyncChannel.sendMessage(CANCEL_CONNECT, 0, c.putListener(listener)); 1256 } 1257 1258 /** 1259 * Create a p2p group with the current device as the group owner. This essentially creates 1260 * an access point that can accept connections from legacy clients as well as other p2p 1261 * devices. 1262 * 1263 * <p class="note"><strong>Note:</strong> 1264 * This function would normally not be used unless the current device needs 1265 * to form a p2p connection with a legacy client 1266 * 1267 * <p> The function call immediately returns after sending a group creation request 1268 * to the framework. The application is notified of a success or failure to initiate 1269 * group creation through listener callbacks {@link ActionListener#onSuccess} or 1270 * {@link ActionListener#onFailure}. 1271 * 1272 * <p> Application can request for the group details with {@link #requestGroupInfo}. 1273 * 1274 * @param c is the channel created at {@link #initialize} 1275 * @param listener for callbacks on success or failure. Can be null. 1276 */ 1277 @RequiresPermission(android.Manifest.permission.ACCESS_FINE_LOCATION) createGroup(Channel c, ActionListener listener)1278 public void createGroup(Channel c, ActionListener listener) { 1279 checkChannel(c); 1280 c.mAsyncChannel.sendMessage(CREATE_GROUP, WifiP2pGroup.PERSISTENT_NET_ID, 1281 c.putListener(listener)); 1282 } 1283 1284 /** 1285 * Create a p2p group with the current device as the group owner. This essentially creates 1286 * an access point that can accept connections from legacy clients as well as other p2p 1287 * devices. 1288 * 1289 * <p> An app should use {@link WifiP2pConfig.Builder} to build the configuration 1290 * for a group. 1291 * 1292 * <p class="note"><strong>Note:</strong> 1293 * This function would normally not be used unless the current device needs 1294 * to form a p2p group as a Group Owner and allow peers to join it as either 1295 * Group Clients or legacy Wi-Fi STAs. 1296 * 1297 * <p> The function call immediately returns after sending a group creation request 1298 * to the framework. The application is notified of a success or failure to initiate 1299 * group creation through listener callbacks {@link ActionListener#onSuccess} or 1300 * {@link ActionListener#onFailure}. 1301 * 1302 * <p> Application can request for the group details with {@link #requestGroupInfo}. 1303 * 1304 * @param c is the channel created at {@link #initialize}. 1305 * @param config the configuration of a p2p group. 1306 * @param listener for callbacks on success or failure. Can be null. 1307 */ 1308 @RequiresPermission(android.Manifest.permission.ACCESS_FINE_LOCATION) createGroup(@onNull Channel c, @Nullable WifiP2pConfig config, @Nullable ActionListener listener)1309 public void createGroup(@NonNull Channel c, 1310 @Nullable WifiP2pConfig config, 1311 @Nullable ActionListener listener) { 1312 checkChannel(c); 1313 c.mAsyncChannel.sendMessage(CREATE_GROUP, 0, 1314 c.putListener(listener), config); 1315 } 1316 1317 /** 1318 * Remove the current p2p group. 1319 * 1320 * <p> The function call immediately returns after sending a group removal request 1321 * to the framework. The application is notified of a success or failure to initiate 1322 * group removal through listener callbacks {@link ActionListener#onSuccess} or 1323 * {@link ActionListener#onFailure}. 1324 * 1325 * @param c is the channel created at {@link #initialize} 1326 * @param listener for callbacks on success or failure. Can be null. 1327 */ removeGroup(Channel c, ActionListener listener)1328 public void removeGroup(Channel c, ActionListener listener) { 1329 checkChannel(c); 1330 c.mAsyncChannel.sendMessage(REMOVE_GROUP, 0, c.putListener(listener)); 1331 } 1332 1333 /** 1334 * Force p2p to enter or exit listen state 1335 * 1336 * @param c is the channel created at {@link #initialize(Context, Looper, ChannelListener)} 1337 * @param enable enables or disables listening 1338 * @param listener for callbacks on success or failure. Can be null. 1339 * 1340 * @hide 1341 */ 1342 @RequiresPermission(android.Manifest.permission.NETWORK_SETTINGS) listen(Channel c, boolean enable, ActionListener listener)1343 public void listen(Channel c, boolean enable, ActionListener listener) { 1344 checkChannel(c); 1345 c.mAsyncChannel.sendMessage(enable ? START_LISTEN : STOP_LISTEN, 1346 0, c.putListener(listener)); 1347 } 1348 1349 /** @hide */ 1350 @UnsupportedAppUsage setWifiP2pChannels(Channel c, int lc, int oc, ActionListener listener)1351 public void setWifiP2pChannels(Channel c, int lc, int oc, ActionListener listener) { 1352 checkChannel(c); 1353 Bundle p2pChannels = new Bundle(); 1354 p2pChannels.putInt("lc", lc); 1355 p2pChannels.putInt("oc", oc); 1356 c.mAsyncChannel.sendMessage(SET_CHANNEL, 0, c.putListener(listener), p2pChannels); 1357 } 1358 1359 /** 1360 * Start a Wi-Fi Protected Setup (WPS) session. 1361 * 1362 * <p> The function call immediately returns after sending a request to start a 1363 * WPS session. Currently, this is only valid if the current device is running 1364 * as a group owner to allow any new clients to join the group. The application 1365 * is notified of a success or failure to initiate WPS through listener callbacks 1366 * {@link ActionListener#onSuccess} or {@link ActionListener#onFailure}. 1367 * @hide 1368 */ 1369 @UnsupportedAppUsage startWps(Channel c, WpsInfo wps, ActionListener listener)1370 public void startWps(Channel c, WpsInfo wps, ActionListener listener) { 1371 checkChannel(c); 1372 c.mAsyncChannel.sendMessage(START_WPS, 0, c.putListener(listener), wps); 1373 } 1374 1375 /** 1376 * Register a local service for service discovery. If a local service is registered, 1377 * the framework automatically responds to a service discovery request from a peer. 1378 * 1379 * <p> The function call immediately returns after sending a request to add a local 1380 * service to the framework. The application is notified of a success or failure to 1381 * add service through listener callbacks {@link ActionListener#onSuccess} or 1382 * {@link ActionListener#onFailure}. 1383 * 1384 * <p>The service information is set through {@link WifiP2pServiceInfo}.<br> 1385 * or its subclass calls {@link WifiP2pUpnpServiceInfo#newInstance} or 1386 * {@link WifiP2pDnsSdServiceInfo#newInstance} for a Upnp or Bonjour service 1387 * respectively 1388 * 1389 * <p>The service information can be cleared with calls to 1390 * {@link #removeLocalService} or {@link #clearLocalServices}. 1391 * 1392 * @param c is the channel created at {@link #initialize} 1393 * @param servInfo is a local service information. 1394 * @param listener for callbacks on success or failure. Can be null. 1395 */ 1396 @RequiresPermission(android.Manifest.permission.ACCESS_FINE_LOCATION) addLocalService(Channel c, WifiP2pServiceInfo servInfo, ActionListener listener)1397 public void addLocalService(Channel c, WifiP2pServiceInfo servInfo, ActionListener listener) { 1398 checkChannel(c); 1399 checkServiceInfo(servInfo); 1400 c.mAsyncChannel.sendMessage(ADD_LOCAL_SERVICE, 0, c.putListener(listener), servInfo); 1401 } 1402 1403 /** 1404 * Remove a registered local service added with {@link #addLocalService} 1405 * 1406 * <p> The function call immediately returns after sending a request to remove a 1407 * local service to the framework. The application is notified of a success or failure to 1408 * add service through listener callbacks {@link ActionListener#onSuccess} or 1409 * {@link ActionListener#onFailure}. 1410 * 1411 * @param c is the channel created at {@link #initialize} 1412 * @param servInfo is the local service information. 1413 * @param listener for callbacks on success or failure. Can be null. 1414 */ removeLocalService(Channel c, WifiP2pServiceInfo servInfo, ActionListener listener)1415 public void removeLocalService(Channel c, WifiP2pServiceInfo servInfo, 1416 ActionListener listener) { 1417 checkChannel(c); 1418 checkServiceInfo(servInfo); 1419 c.mAsyncChannel.sendMessage(REMOVE_LOCAL_SERVICE, 0, c.putListener(listener), servInfo); 1420 } 1421 1422 /** 1423 * Clear all registered local services of service discovery. 1424 * 1425 * <p> The function call immediately returns after sending a request to clear all 1426 * local services to the framework. The application is notified of a success or failure to 1427 * add service through listener callbacks {@link ActionListener#onSuccess} or 1428 * {@link ActionListener#onFailure}. 1429 * 1430 * @param c is the channel created at {@link #initialize} 1431 * @param listener for callbacks on success or failure. Can be null. 1432 */ clearLocalServices(Channel c, ActionListener listener)1433 public void clearLocalServices(Channel c, ActionListener listener) { 1434 checkChannel(c); 1435 c.mAsyncChannel.sendMessage(CLEAR_LOCAL_SERVICES, 0, c.putListener(listener)); 1436 } 1437 1438 /** 1439 * Register a callback to be invoked on receiving service discovery response. 1440 * Used only for vendor specific protocol right now. For Bonjour or Upnp, use 1441 * {@link #setDnsSdResponseListeners} or {@link #setUpnpServiceResponseListener} 1442 * respectively. 1443 * 1444 * <p> see {@link #discoverServices} for the detail. 1445 * 1446 * @param c is the channel created at {@link #initialize} 1447 * @param listener for callbacks on receiving service discovery response. 1448 */ setServiceResponseListener(Channel c, ServiceResponseListener listener)1449 public void setServiceResponseListener(Channel c, 1450 ServiceResponseListener listener) { 1451 checkChannel(c); 1452 c.mServRspListener = listener; 1453 } 1454 1455 /** 1456 * Register a callback to be invoked on receiving Bonjour service discovery 1457 * response. 1458 * 1459 * <p> see {@link #discoverServices} for the detail. 1460 * 1461 * @param c 1462 * @param servListener is for listening to a Bonjour service response 1463 * @param txtListener is for listening to a Bonjour TXT record response 1464 */ setDnsSdResponseListeners(Channel c, DnsSdServiceResponseListener servListener, DnsSdTxtRecordListener txtListener)1465 public void setDnsSdResponseListeners(Channel c, 1466 DnsSdServiceResponseListener servListener, DnsSdTxtRecordListener txtListener) { 1467 checkChannel(c); 1468 c.mDnsSdServRspListener = servListener; 1469 c.mDnsSdTxtListener = txtListener; 1470 } 1471 1472 /** 1473 * Register a callback to be invoked on receiving upnp service discovery 1474 * response. 1475 * 1476 * <p> see {@link #discoverServices} for the detail. 1477 * 1478 * @param c is the channel created at {@link #initialize} 1479 * @param listener for callbacks on receiving service discovery response. 1480 */ setUpnpServiceResponseListener(Channel c, UpnpServiceResponseListener listener)1481 public void setUpnpServiceResponseListener(Channel c, 1482 UpnpServiceResponseListener listener) { 1483 checkChannel(c); 1484 c.mUpnpServRspListener = listener; 1485 } 1486 1487 /** 1488 * Initiate service discovery. A discovery process involves scanning for 1489 * requested services for the purpose of establishing a connection to a peer 1490 * that supports an available service. 1491 * 1492 * <p> The function call immediately returns after sending a request to start service 1493 * discovery to the framework. The application is notified of a success or failure to initiate 1494 * discovery through listener callbacks {@link ActionListener#onSuccess} or 1495 * {@link ActionListener#onFailure}. 1496 * 1497 * <p> The services to be discovered are specified with calls to {@link #addServiceRequest}. 1498 * 1499 * <p>The application is notified of the response against the service discovery request 1500 * through listener callbacks registered by {@link #setServiceResponseListener} or 1501 * {@link #setDnsSdResponseListeners}, or {@link #setUpnpServiceResponseListener}. 1502 * 1503 * @param c is the channel created at {@link #initialize} 1504 * @param listener for callbacks on success or failure. Can be null. 1505 */ 1506 @RequiresPermission(android.Manifest.permission.ACCESS_FINE_LOCATION) discoverServices(Channel c, ActionListener listener)1507 public void discoverServices(Channel c, ActionListener listener) { 1508 checkChannel(c); 1509 c.mAsyncChannel.sendMessage(DISCOVER_SERVICES, 0, c.putListener(listener)); 1510 } 1511 1512 /** 1513 * Add a service discovery request. 1514 * 1515 * <p> The function call immediately returns after sending a request to add service 1516 * discovery request to the framework. The application is notified of a success or failure to 1517 * add service through listener callbacks {@link ActionListener#onSuccess} or 1518 * {@link ActionListener#onFailure}. 1519 * 1520 * <p>After service discovery request is added, you can initiate service discovery by 1521 * {@link #discoverServices}. 1522 * 1523 * <p>The added service requests can be cleared with calls to 1524 * {@link #removeServiceRequest(Channel, WifiP2pServiceRequest, ActionListener)} or 1525 * {@link #clearServiceRequests(Channel, ActionListener)}. 1526 * 1527 * @param c is the channel created at {@link #initialize} 1528 * @param req is the service discovery request. 1529 * @param listener for callbacks on success or failure. Can be null. 1530 */ addServiceRequest(Channel c, WifiP2pServiceRequest req, ActionListener listener)1531 public void addServiceRequest(Channel c, 1532 WifiP2pServiceRequest req, ActionListener listener) { 1533 checkChannel(c); 1534 checkServiceRequest(req); 1535 c.mAsyncChannel.sendMessage(ADD_SERVICE_REQUEST, 0, 1536 c.putListener(listener), req); 1537 } 1538 1539 /** 1540 * Remove a specified service discovery request added with {@link #addServiceRequest} 1541 * 1542 * <p> The function call immediately returns after sending a request to remove service 1543 * discovery request to the framework. The application is notified of a success or failure to 1544 * add service through listener callbacks {@link ActionListener#onSuccess} or 1545 * {@link ActionListener#onFailure}. 1546 * 1547 * @param c is the channel created at {@link #initialize} 1548 * @param req is the service discovery request. 1549 * @param listener for callbacks on success or failure. Can be null. 1550 */ removeServiceRequest(Channel c, WifiP2pServiceRequest req, ActionListener listener)1551 public void removeServiceRequest(Channel c, WifiP2pServiceRequest req, 1552 ActionListener listener) { 1553 checkChannel(c); 1554 checkServiceRequest(req); 1555 c.mAsyncChannel.sendMessage(REMOVE_SERVICE_REQUEST, 0, 1556 c.putListener(listener), req); 1557 } 1558 1559 /** 1560 * Clear all registered service discovery requests. 1561 * 1562 * <p> The function call immediately returns after sending a request to clear all 1563 * service discovery requests to the framework. The application is notified of a success 1564 * or failure to add service through listener callbacks {@link ActionListener#onSuccess} or 1565 * {@link ActionListener#onFailure}. 1566 * 1567 * @param c is the channel created at {@link #initialize} 1568 * @param listener for callbacks on success or failure. Can be null. 1569 */ clearServiceRequests(Channel c, ActionListener listener)1570 public void clearServiceRequests(Channel c, ActionListener listener) { 1571 checkChannel(c); 1572 c.mAsyncChannel.sendMessage(CLEAR_SERVICE_REQUESTS, 1573 0, c.putListener(listener)); 1574 } 1575 1576 /** 1577 * Request the current list of peers. 1578 * 1579 * @param c is the channel created at {@link #initialize} 1580 * @param listener for callback when peer list is available. Can be null. 1581 */ 1582 @RequiresPermission(android.Manifest.permission.ACCESS_FINE_LOCATION) requestPeers(Channel c, PeerListListener listener)1583 public void requestPeers(Channel c, PeerListListener listener) { 1584 checkChannel(c); 1585 c.mAsyncChannel.sendMessage(REQUEST_PEERS, 0, c.putListener(listener)); 1586 } 1587 1588 /** 1589 * Request device connection info. 1590 * 1591 * @param c is the channel created at {@link #initialize} 1592 * @param listener for callback when connection info is available. Can be null. 1593 */ requestConnectionInfo(Channel c, ConnectionInfoListener listener)1594 public void requestConnectionInfo(Channel c, ConnectionInfoListener listener) { 1595 checkChannel(c); 1596 c.mAsyncChannel.sendMessage(REQUEST_CONNECTION_INFO, 0, c.putListener(listener)); 1597 } 1598 1599 /** 1600 * Request p2p group info. 1601 * 1602 * @param c is the channel created at {@link #initialize} 1603 * @param listener for callback when group info is available. Can be null. 1604 */ 1605 @RequiresPermission(android.Manifest.permission.ACCESS_FINE_LOCATION) requestGroupInfo(Channel c, GroupInfoListener listener)1606 public void requestGroupInfo(Channel c, GroupInfoListener listener) { 1607 checkChannel(c); 1608 c.mAsyncChannel.sendMessage(REQUEST_GROUP_INFO, 0, c.putListener(listener)); 1609 } 1610 1611 /** 1612 * Set p2p device name. 1613 * @hide 1614 * @param c is the channel created at {@link #initialize} 1615 * @param listener for callback when group info is available. Can be null. 1616 */ 1617 @UnsupportedAppUsage setDeviceName(Channel c, String devName, ActionListener listener)1618 public void setDeviceName(Channel c, String devName, ActionListener listener) { 1619 checkChannel(c); 1620 WifiP2pDevice d = new WifiP2pDevice(); 1621 d.deviceName = devName; 1622 c.mAsyncChannel.sendMessage(SET_DEVICE_NAME, 0, c.putListener(listener), d); 1623 } 1624 1625 /** @hide */ 1626 @UnsupportedAppUsage setWFDInfo( Channel c, WifiP2pWfdInfo wfdInfo, ActionListener listener)1627 public void setWFDInfo( 1628 Channel c, WifiP2pWfdInfo wfdInfo, 1629 ActionListener listener) { 1630 checkChannel(c); 1631 try { 1632 mService.checkConfigureWifiDisplayPermission(); 1633 } catch (RemoteException e) { 1634 e.rethrowFromSystemServer(); 1635 } 1636 c.mAsyncChannel.sendMessage(SET_WFD_INFO, 0, c.putListener(listener), wfdInfo); 1637 } 1638 1639 1640 /** 1641 * Delete a stored persistent group from the system settings. 1642 * 1643 * <p> The function call immediately returns after sending a persistent group removal request 1644 * to the framework. The application is notified of a success or failure to initiate 1645 * group removal through listener callbacks {@link ActionListener#onSuccess} or 1646 * {@link ActionListener#onFailure}. 1647 * 1648 * <p>The persistent p2p group list stored in the system can be obtained by 1649 * {@link #requestPersistentGroupInfo(Channel, PersistentGroupInfoListener)} and 1650 * a network id can be obtained by {@link WifiP2pGroup#getNetworkId()}. 1651 * 1652 * @param c is the channel created at {@link #initialize} 1653 * @param netId he network id of the p2p group. 1654 * @param listener for callbacks on success or failure. Can be null. 1655 * @hide 1656 */ 1657 @UnsupportedAppUsage deletePersistentGroup(Channel c, int netId, ActionListener listener)1658 public void deletePersistentGroup(Channel c, int netId, ActionListener listener) { 1659 checkChannel(c); 1660 c.mAsyncChannel.sendMessage(DELETE_PERSISTENT_GROUP, netId, c.putListener(listener)); 1661 } 1662 1663 /** 1664 * Request a list of all the persistent p2p groups stored in system. 1665 * 1666 * @param c is the channel created at {@link #initialize} 1667 * @param listener for callback when persistent group info list is available. Can be null. 1668 * @hide 1669 */ 1670 @UnsupportedAppUsage requestPersistentGroupInfo(Channel c, PersistentGroupInfoListener listener)1671 public void requestPersistentGroupInfo(Channel c, PersistentGroupInfoListener listener) { 1672 checkChannel(c); 1673 c.mAsyncChannel.sendMessage(REQUEST_PERSISTENT_GROUP_INFO, 0, c.putListener(listener)); 1674 } 1675 1676 /** @hide */ 1677 public static final int MIRACAST_DISABLED = 0; 1678 /** @hide */ 1679 public static final int MIRACAST_SOURCE = 1; 1680 /** @hide */ 1681 public static final int MIRACAST_SINK = 2; 1682 /** Internal use only @hide */ 1683 @UnsupportedAppUsage setMiracastMode(int mode)1684 public void setMiracastMode(int mode) { 1685 try { 1686 mService.setMiracastMode(mode); 1687 } catch (RemoteException e) { 1688 throw e.rethrowFromSystemServer(); 1689 } 1690 } 1691 1692 /** 1693 * Get a reference to WifiP2pService handler. This is used to establish 1694 * an AsyncChannel communication with WifiService 1695 * 1696 * @param binder A binder for the service to associate with this client. 1697 * 1698 * @return Messenger pointing to the WifiP2pService handler 1699 * @hide 1700 */ getMessenger(Binder binder)1701 public Messenger getMessenger(Binder binder) { 1702 try { 1703 return mService.getMessenger(binder); 1704 } catch (RemoteException e) { 1705 throw e.rethrowFromSystemServer(); 1706 } 1707 } 1708 1709 /** 1710 * Get a reference to P2pStateMachine handler. This is used to establish 1711 * a priveleged AsyncChannel communication with WifiP2pService. 1712 * 1713 * @return Messenger pointing to the WifiP2pService handler 1714 * @hide 1715 */ getP2pStateMachineMessenger()1716 public Messenger getP2pStateMachineMessenger() { 1717 try { 1718 return mService.getP2pStateMachineMessenger(); 1719 } catch (RemoteException e) { 1720 throw e.rethrowFromSystemServer(); 1721 } 1722 } 1723 1724 /** 1725 * Get a handover request message for use in WFA NFC Handover transfer. 1726 * @hide 1727 */ getNfcHandoverRequest(Channel c, HandoverMessageListener listener)1728 public void getNfcHandoverRequest(Channel c, HandoverMessageListener listener) { 1729 checkChannel(c); 1730 c.mAsyncChannel.sendMessage(GET_HANDOVER_REQUEST, 0, c.putListener(listener)); 1731 } 1732 1733 1734 /** 1735 * Get a handover select message for use in WFA NFC Handover transfer. 1736 * @hide 1737 */ getNfcHandoverSelect(Channel c, HandoverMessageListener listener)1738 public void getNfcHandoverSelect(Channel c, HandoverMessageListener listener) { 1739 checkChannel(c); 1740 c.mAsyncChannel.sendMessage(GET_HANDOVER_SELECT, 0, c.putListener(listener)); 1741 } 1742 1743 /** 1744 * @hide 1745 */ initiatorReportNfcHandover(Channel c, String handoverSelect, ActionListener listener)1746 public void initiatorReportNfcHandover(Channel c, String handoverSelect, 1747 ActionListener listener) { 1748 checkChannel(c); 1749 Bundle bundle = new Bundle(); 1750 bundle.putString(EXTRA_HANDOVER_MESSAGE, handoverSelect); 1751 c.mAsyncChannel.sendMessage(INITIATOR_REPORT_NFC_HANDOVER, 0, 1752 c.putListener(listener), bundle); 1753 } 1754 1755 1756 /** 1757 * @hide 1758 */ responderReportNfcHandover(Channel c, String handoverRequest, ActionListener listener)1759 public void responderReportNfcHandover(Channel c, String handoverRequest, 1760 ActionListener listener) { 1761 checkChannel(c); 1762 Bundle bundle = new Bundle(); 1763 bundle.putString(EXTRA_HANDOVER_MESSAGE, handoverRequest); 1764 c.mAsyncChannel.sendMessage(RESPONDER_REPORT_NFC_HANDOVER, 0, 1765 c.putListener(listener), bundle); 1766 } 1767 1768 /** 1769 * Removes all saved p2p groups. 1770 * 1771 * @param c is the channel created at {@link #initialize}. 1772 * @param listener for callback on success or failure. Can be null. 1773 * @hide 1774 */ 1775 @RequiresPermission(android.Manifest.permission.NETWORK_SETTINGS) factoryReset(@onNull Channel c, @Nullable ActionListener listener)1776 public void factoryReset(@NonNull Channel c, @Nullable ActionListener listener) { 1777 checkChannel(c); 1778 c.mAsyncChannel.sendMessage(FACTORY_RESET, 0, c.putListener(listener)); 1779 } 1780 1781 /** 1782 * Request saved WifiP2pConfig which used for an ongoing peer connection 1783 * 1784 * @param c is the channel created at {@link #initialize} 1785 * @param listener for callback when ongoing peer config updated. Can't be null. 1786 * 1787 * @hide 1788 */ 1789 @RequiresPermission(android.Manifest.permission.NETWORK_STACK) requestOngoingPeerConfig(@onNull Channel c, @NonNull OngoingPeerInfoListener listener)1790 public void requestOngoingPeerConfig(@NonNull Channel c, 1791 @NonNull OngoingPeerInfoListener listener) { 1792 checkChannel(c); 1793 c.mAsyncChannel.sendMessage(REQUEST_ONGOING_PEER_CONFIG, 1794 Binder.getCallingUid(), c.putListener(listener)); 1795 } 1796 1797 /** 1798 * Set saved WifiP2pConfig which used for an ongoing peer connection 1799 * 1800 * @param c is the channel created at {@link #initialize} 1801 * @param config used for change an ongoing peer connection 1802 * @param listener for callback when ongoing peer config updated. Can be null. 1803 * 1804 * @hide 1805 */ 1806 @RequiresPermission(android.Manifest.permission.NETWORK_STACK) setOngoingPeerConfig(@onNull Channel c, @NonNull WifiP2pConfig config, @Nullable ActionListener listener)1807 public void setOngoingPeerConfig(@NonNull Channel c, @NonNull WifiP2pConfig config, 1808 @Nullable ActionListener listener) { 1809 checkChannel(c); 1810 checkP2pConfig(config); 1811 c.mAsyncChannel.sendMessage(SET_ONGOING_PEER_CONFIG, 0, 1812 c.putListener(listener), config); 1813 } 1814 1815 /** 1816 * Request p2p enabled state. 1817 * 1818 * <p> This state indicates whether Wi-Fi p2p is enabled or disabled. 1819 * The valid value is one of {@link #WIFI_P2P_STATE_DISABLED} or 1820 * {@link #WIFI_P2P_STATE_ENABLED}. The state is returned using the 1821 * {@link P2pStateListener} listener. 1822 * 1823 * <p> This state is also included in the {@link #WIFI_P2P_STATE_CHANGED_ACTION} 1824 * broadcast event with extra {@link #EXTRA_WIFI_STATE}. 1825 * 1826 * @param c is the channel created at {@link #initialize}. 1827 * @param listener for callback when p2p state is available.. 1828 */ requestP2pState(@onNull Channel c, @NonNull P2pStateListener listener)1829 public void requestP2pState(@NonNull Channel c, 1830 @NonNull P2pStateListener listener) { 1831 checkChannel(c); 1832 if (listener == null) throw new IllegalArgumentException("This listener cannot be null."); 1833 c.mAsyncChannel.sendMessage(REQUEST_P2P_STATE, 0, c.putListener(listener)); 1834 } 1835 1836 /** 1837 * Request p2p discovery state. 1838 * 1839 * <p> This state indicates whether p2p discovery has started or stopped. 1840 * The valid value is one of {@link #WIFI_P2P_DISCOVERY_STARTED} or 1841 * {@link #WIFI_P2P_DISCOVERY_STOPPED}. The state is returned using the 1842 * {@link DiscoveryStateListener} listener. 1843 * 1844 * <p> This state is also included in the {@link #WIFI_P2P_DISCOVERY_CHANGED_ACTION} 1845 * broadcast event with extra {@link #EXTRA_DISCOVERY_STATE}. 1846 * 1847 * @param c is the channel created at {@link #initialize}. 1848 * @param listener for callback when discovery state is available.. 1849 */ requestDiscoveryState(@onNull Channel c, @NonNull DiscoveryStateListener listener)1850 public void requestDiscoveryState(@NonNull Channel c, 1851 @NonNull DiscoveryStateListener listener) { 1852 checkChannel(c); 1853 if (listener == null) throw new IllegalArgumentException("This listener cannot be null."); 1854 c.mAsyncChannel.sendMessage(REQUEST_DISCOVERY_STATE, 0, c.putListener(listener)); 1855 } 1856 1857 /** 1858 * Request network info. 1859 * 1860 * <p> This method provides the network info in the form of a {@link android.net.NetworkInfo}. 1861 * {@link android.net.NetworkInfo#isAvailable()} indicates the p2p availability and 1862 * {@link android.net.NetworkInfo#getDetailedState()} reports the current fine-grained state 1863 * of the network. This {@link android.net.NetworkInfo} is returned using the 1864 * {@link NetworkInfoListener} listener. 1865 * 1866 * <p> This information is also included in the {@link #WIFI_P2P_CONNECTION_CHANGED_ACTION} 1867 * broadcast event with extra {@link #EXTRA_NETWORK_INFO}. 1868 * 1869 * @param c is the channel created at {@link #initialize}. 1870 * @param listener for callback when network info is available.. 1871 */ requestNetworkInfo(@onNull Channel c, @NonNull NetworkInfoListener listener)1872 public void requestNetworkInfo(@NonNull Channel c, 1873 @NonNull NetworkInfoListener listener) { 1874 checkChannel(c); 1875 if (listener == null) throw new IllegalArgumentException("This listener cannot be null."); 1876 c.mAsyncChannel.sendMessage(REQUEST_NETWORK_INFO, 0, c.putListener(listener)); 1877 } 1878 1879 /** 1880 * Request Device Info 1881 * 1882 * <p> This method provides the device info 1883 * in the form of a {@link android.net.wifi.p2p.WifiP2pDevice}. 1884 * Valid {@link android.net.wifi.p2p.WifiP2pDevice} is returned when p2p is enabled. 1885 * To get information notifications on P2P getting enabled refers 1886 * {@link #WIFI_P2P_STATE_ENABLED}. 1887 * 1888 * <p> This {@link android.net.wifi.p2p.WifiP2pDevice} is returned using the 1889 * {@link DeviceInfoListener} listener. 1890 * 1891 * <p> {@link android.net.wifi.p2p.WifiP2pDevice#deviceAddress} is only available if the caller 1892 * holds the {@code android.Manifest.permission#LOCAL_MAC_ADDRESS} permission, and holds the 1893 * anonymized MAC address (02:00:00:00:00:00) otherwise. 1894 * 1895 * <p> This information is also included in the {@link #WIFI_P2P_THIS_DEVICE_CHANGED_ACTION} 1896 * broadcast event with extra {@link #EXTRA_WIFI_P2P_DEVICE}. 1897 * 1898 * @param c is the channel created at {@link #initialize(Context, Looper, ChannelListener)}. 1899 * @param listener for callback when network info is available. 1900 */ 1901 @RequiresPermission(android.Manifest.permission.ACCESS_FINE_LOCATION) requestDeviceInfo(@onNull Channel c, @NonNull DeviceInfoListener listener)1902 public void requestDeviceInfo(@NonNull Channel c, @NonNull DeviceInfoListener listener) { 1903 checkChannel(c); 1904 if (listener == null) throw new IllegalArgumentException("This listener cannot be null."); 1905 c.mAsyncChannel.sendMessage(REQUEST_DEVICE_INFO, 0, c.putListener(listener)); 1906 } 1907 } 1908