1 page.title=Bluetooth 2 @jd:body 3 4 <div id="qv-wrapper"> 5 <div id="qv"> 6 <h2>Key Classes</h2> 7 <ol> 8 <li>{@link android.bluetooth.BluetoothAdapter}</li> 9 <li>{@link android.bluetooth.BluetoothDevice}</li> 10 <li>{@link android.bluetooth.BluetoothSocket}</li> 11 <li>{@link android.bluetooth.BluetoothServerSocket}</li> 12 </ol> 13 14 <h2>In this document</h2> 15 <ol> 16 <li><a href="#TheBasics">The Basics</a></li> 17 <li><a href="#Permissions">Bluetooth Permissions</a></li> 18 <li><a href="#SettingUp">Setting Up Bluetooth</a></li> 19 <li><a href="#FindingDevices">Finding Devices</a> 20 <ol> 21 <li><a href="#QueryingPairedDevices">Querying paired devices</a></li> 22 <li><a href="#DiscoveringDevices">Discovering devices</a> 23 <ol><li><a href="#EnablingDiscoverability">Enabling 24 discoverability</a></li></ol> 25 </li> 26 </ol> 27 </li> 28 <li><a href="#ConnectingDevices">Connecting Devices</a> 29 <ol> 30 <li><a href="#ConnectingAsAServer">Connecting as a server</a></li> 31 <li><a href="#ConnectingAsAClient">Connecting as a client</a></li> 32 </ol> 33 </li> 34 <li><a href="#ManagingAConnection">Managing a Connection</a></li> 35 </ol> 36 37 <h2>See also</h2> 38 <ol> 39 <li><a href="{@docRoot}resources/samples/BluetoothChat/index.html">Bluetooth Chat sample 40 app</a></li> 41 </ol> 42 43 </div> 44 </div> 45 46 47 <p>The Android platform includes support for the Bluetooth network stack, 48 which allows a device to wirelessly exchange data with other Bluetooth devices. 49 The application framework provides access to the Bluetooth functionality through 50 the Android Bluetooth APIs. These APIs let applications wirelessly 51 connect to other Bluetooth devices, enabling point-to-point and multipoint 52 wireless features.</p> 53 54 <p>Using the Bluetooth APIs, an Android application can perform the 55 following:</p> 56 <ul> 57 <li>Scan for other Bluetooth devices</li> 58 <li>Query the local Bluetooth adapter for paired Bluetooth devices</li> 59 <li>Establish RFCOMM channels</li> 60 <li>Connect to other devices through service discovery</li> 61 <li>Transfer data to and from other devices</li> 62 <li>Manage multiple connections</li> 63 </ul> 64 65 66 <h2 id="TheBasics">The Basics</h2> 67 68 <p>This document describes how to us the Android Bluetooth APIs to accomplish 69 the four major tasks necessary to communicate using Bluetooth: setting up 70 Bluetooth, finding devices that are either paired or available in the local 71 area, connecting devices, and transferring data between devices.</p> 72 73 <p>All of the Bluetooth APIs are available in the {@link android.bluetooth} 74 package. Here's a summary of the classes you will need to create Bluetooth 75 connections:</p> 76 77 <dl> 78 <dt>{@link android.bluetooth.BluetoothAdapter}</dt> 79 <dd>Represents the local Bluetooth adapter (Bluetooth radio). The 80 {@link android.bluetooth.BluetoothAdapter} is the entry-point for all Bluetooth 81 interaction. Using this, 82 you can discover other Bluetooth devices, query a list of bonded (paired) 83 devices, instantiate a {@link android.bluetooth.BluetoothDevice} using a known 84 MAC address, and create a {@link android.bluetooth.BluetoothServerSocket} to 85 listen for communications 86 from other devices.</dd> 87 88 <dt>{@link android.bluetooth.BluetoothDevice}</dt> 89 <dd>Represents a remote Bluetooth device. Use this to request a connection 90 with a remote device through a {@link android.bluetooth.BluetoothSocket} or 91 query information about the 92 device such as its name, address, class, and bonding state.</dd> 93 94 <dt>{@link android.bluetooth.BluetoothSocket}</dt> 95 <dd>Represents the interface for a Bluetooth socket (similar to a TCP 96 {@link java.net.Socket}). This is the connection point that allows 97 an application to exchange data with another Bluetooth device via InputStream 98 and OutputStream.</dd> 99 100 <dt>{@link android.bluetooth.BluetoothServerSocket}</dt> 101 <dd>Represents an open server socket that listens for incoming requests 102 (similar to a TCP {@link java.net.ServerSocket}). In order to connect two 103 Android devices, one device must open a server socket with this class. When a 104 remote Bluetooth device makes a connection request to the this device, the 105 {@link android.bluetooth.BluetoothServerSocket} will return a connected {@link 106 android.bluetooth.BluetoothSocket} when the 107 connection is accepted.</dd> 108 109 <dt>{@link android.bluetooth.BluetoothClass}</dt> 110 <dd>Describes the general characteristics and capabilities of a Bluetooth 111 device. This is a read-only set of properties that define the device's major and 112 minor device classes and its services. However, this does not reliably describe 113 all Bluetooth profiles and services supported by the device, but is useful as a 114 hint to the device type.</dd> 115 </dl> 116 117 118 119 120 <h2 id="Permissions">Bluetooth Permissions</h2> 121 122 <p>In order to use Bluetooth features in your application, you need to declare 123 at least one of two Bluetooth permissions: {@link 124 android.Manifest.permission#BLUETOOTH} and {@link 125 android.Manifest.permission#BLUETOOTH_ADMIN}.</p> 126 127 <p>You must request the {@link android.Manifest.permission#BLUETOOTH} permission 128 in order to perform any Bluetooth communication, such as requesting a 129 connection, accepting a connection, and transferring data.</p> 130 131 <p>You must request the {@link android.Manifest.permission#BLUETOOTH_ADMIN} 132 permission in order to initiate device discovery or manipulate Bluetooth 133 settings. Most applications need this permission solely for the 134 ability to discover local Bluetooth devices. The other abilities granted by this 135 permission should not be used, unless the application is a "power manager" that 136 will modify Bluetooth settings upon user request. <strong>Note:</strong> If you 137 use {@link android.Manifest.permission#BLUETOOTH_ADMIN} permission, then must 138 also have the {@link android.Manifest.permission#BLUETOOTH} permission.</p> 139 140 <p>Declare the Bluetooth permission(s) in your application manifest file. For 141 example:</p> 142 143 <pre> 144 <manifest ... > 145 <uses-permission android:name="android.permission.BLUETOOTH" /> 146 ... 147 </manifest> 148 </pre> 149 150 <p>See the <a 151 href="{@docRoot}guide/topics/manifest/uses-permission-element.html"><uses-permission></a> 152 reference for more information about declaring application permissions.</p> 153 154 155 <h2 id="SettingUp">Setting Up Bluetooth</h2> 156 157 <div class="figure" style="width:200px"> 158 <img src="{@docRoot}images/bt_enable_request.png" /> 159 <strong>Figure 1:</strong> The enabling Bluetooth dialog. 160 </div> 161 162 <p>Before your application can communicate over Bluetooth, you need to verify 163 that Bluetooth is supported on the device, and if so, ensure that it is enabled.</p> 164 165 <p>If Bluetooth is not supported, then you should gracefully disable any 166 Bluetooth features. If Bluetooth is supported, but disabled, then you can request that the 167 user enable Bluetooth without leaving your application. This setup is 168 accomplished in two steps, using the {@link android.bluetooth.BluetoothAdapter}.</p> 169 170 171 <ol> 172 <li>Get the {@link android.bluetooth.BluetoothAdapter} 173 <p>The {@link android.bluetooth.BluetoothAdapter} is required for any and all Bluetooth 174 activity. To get the {@link android.bluetooth.BluetoothAdapter}, call the static {@link 175 android.bluetooth.BluetoothAdapter#getDefaultAdapter()} method. This returns a 176 {@link android.bluetooth.BluetoothAdapter} that represents the device's own 177 Bluetooth adapter (the Bluetooth radio). There's one Bluetooth adapter for the 178 entire system, and your application can interact with it using this object. If 179 {@link android.bluetooth.BluetoothAdapter#getDefaultAdapter()} returns null, 180 then the device does not support Bluetooth and your story ends here. For example:</p> 181 <pre> 182 BluetoothAdapter mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter(); 183 if (mBluetoothAdapter == null) { 184 // Device does not support Bluetooth 185 } 186 </pre> 187 </li> 188 189 <li>Enable Bluetooth 190 <p>Next, you need to ensure that Bluetooth is enabled. Call {@link 191 android.bluetooth.BluetoothAdapter#isEnabled()} to check whether Bluetooth is 192 currently enable. If this method returns false, then Bluetooth is disabled. To 193 request that Bluetooth be enabled, call {@link 194 android.app.Activity#startActivityForResult(Intent,int) startActivityForResult()} 195 with the {@link android.bluetooth.BluetoothAdapter#ACTION_REQUEST_ENABLE} action Intent. 196 This will issue a request to enable Bluetooth through the system settings (without 197 stopping your application). For example:</p> 198 <pre> 199 if (!mBluetoothAdapter.isEnabled()) { 200 Intent enableBtIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE); 201 startActivityForResult(enableBtIntent, REQUEST_ENABLE_BT); 202 } 203 </pre> 204 205 <p>A dialog will appear requesting user permission to enable Bluetooth, as shown 206 in Figure 1. If the user responds "Yes," the system will begin to enable Bluetooth 207 and focus will return to your application once the process completes (or fails).</p> 208 <p>If enabling Bluetooth succeeds, your Activity will receive the {@link 209 android.app.Activity#RESULT_OK} result code in the {@link 210 android.app.Activity#onActivityResult(int,int,Intent) onActivityResult()} 211 callback. If Bluetooth was not enabled 212 due to an error (or the user responded "No") then the result code will be {@link 213 android.app.Activity#RESULT_CANCELED}.</p> 214 </li> 215 </ol> 216 217 <p>Optionally, your application can also listen for the 218 {@link android.bluetooth.BluetoothAdapter#ACTION_STATE_CHANGED} broadcast Intent, which 219 the system will broadcast whenever the Bluetooth state has changed. This broadcast contains 220 the extra fields {@link android.bluetooth.BluetoothAdapter#EXTRA_STATE} and {@link 221 android.bluetooth.BluetoothAdapter#EXTRA_PREVIOUS_STATE}, containing the new and old 222 Bluetooth states, respectively. Possible values for these extra fields are 223 {@link android.bluetooth.BluetoothAdapter#STATE_TURNING_ON}, {@link 224 android.bluetooth.BluetoothAdapter#STATE_ON}, {@link 225 android.bluetooth.BluetoothAdapter#STATE_TURNING_OFF}, and {@link 226 android.bluetooth.BluetoothAdapter#STATE_OFF}. Listening for this 227 broadcast can be useful to detect changes made to the Bluetooth state while your 228 app is running.</p> 229 230 <p class="note"><strong>Tip:</strong> Enabling discoverability will automatically 231 enable Bluetooth. If you plan to consistently enable device discoverability before 232 performing Bluetooth activity, you can skip 233 step 2 above. Read about <a href="#EnablingDiscoverability">enabling discoverability</a>, 234 below.</p> 235 236 237 <h2 id="FindingDevices">Finding Devices</h2> 238 239 <p>Using the {@link android.bluetooth.BluetoothAdapter}, you can find remote Bluetooth 240 devices either through device discovery or by querying the list of paired (bonded) 241 devices.</p> 242 243 <p>Device discovery is a scanning procedure that searches the local area for 244 Bluetooth enabled devices and then requesting some information about each one 245 (this is sometimes referred to as "discovering," "inquiring" or "scanning"). 246 However, a Bluetooth device within the local area will respond to a discovery 247 request only if it is currently enabled to be discoverable. If a device is 248 discoverable, it will respond to the discovery request by sharing some 249 information, such as the device name, class, and its unique MAC address. Using 250 this information, the device performing discovery can then choose to initiate a 251 connection to the discovered device.</p> 252 253 <p>Once a connection is made with a remote device for the first time, a pairing 254 request is automatically presented to the user. When a device is 255 paired, the basic information about that device (such as the device name, class, 256 and MAC address) is saved and can be read using the Bluetooth APIs. Using the 257 known MAC address for a remote device, a connection can be initiated with it at 258 any time without performing discovery (assuming the device is within range).</p> 259 260 <p>Remember there is a difference between being paired and being connected. To 261 be paired means that two devices are aware of each other's existence, have a 262 shared link-key that can be used for authentication, and are capable of 263 establishing an encrypted connection with each other. To be connected means that 264 the devices currently share an RFCOMM channel and are able to transmit data with 265 each other. The current Android Bluetooth API's require devices to be paired 266 before an RFCOMM connection can be established. (Pairing is automatically performed 267 when you initiate an encrypted connection with the Bluetooth APIs.)</p> 268 269 <p>The following sections describe how to find devices that have been paired, or 270 discover new devices using device discovery.</p> 271 272 <p class="note"><strong>Note:</strong> Android-powered devices are not 273 discoverable by default. A user can make 274 the device discoverable for a limited time through the system settings, or an 275 application can request that the user enable discoverability without leaving the 276 application. How to <a href="#EnablingDiscoverability">enable discoverability</a> 277 is discussed below.</p> 278 279 280 <h3 id="QueryingPairedDevices">Querying paired devices</h3> 281 282 <p>Before performing device discovery, its worth querying the set 283 of paired devices to see if the desired device is already known. To do so, 284 call {@link android.bluetooth.BluetoothAdapter#getBondedDevices()}. This 285 will return a Set of {@link android.bluetooth.BluetoothDevice}s representing 286 paired devices. For example, you can query all paired devices and then add then 287 show the name of each device to the user, using an ArrayAdapter:</p> 288 <pre> 289 Set<BluetoothDevice> pairedDevices = mBluetoothAdapter.getBondedDevices(); 290 // If there are paired devices 291 if (pairedDevices.size() > 0) { 292 // Loop through paired devices 293 for (BluetoothDevice device : pairedDevices) { 294 // Add the name and address to an array adapter to show in a ListView 295 mArrayAdapter.add(device.getName() + "\n" + device.getAddress()); 296 } 297 } 298 </pre> 299 300 <p>All that's needed from the {@link android.bluetooth.BluetoothDevice} object 301 in order to initiate a connection is the MAC address. In this example, it's saved 302 as a part of an ArrayAdapter that's shown to the user. The MAC address can later 303 be extracted in order to initiate the connection. You can learn more about creating 304 a connection in the section about <a href="#ConnectingDevices">Connecting Devices</a>.</p> 305 306 307 <h3 id="DiscoveringDevices">Discovering devices</h3> 308 309 <p>To start discovering devices, simply call {@link 310 android.bluetooth.BluetoothAdapter#startDiscovery()}. The 311 process is asynchronous and the method will immediately return with a boolean 312 indicating whether discovery has successfully started. The discovery process 313 usually involves an inquiry scan of about 12 seconds, followed by a page scan of 314 each found device to retrieve its Bluetooth name.</p> 315 316 <p>Your application must register a BroadcastReceiver for the 317 {@link android.bluetooth.BluetoothDevice#ACTION_FOUND} Intent in 318 order to receive information about each 319 device discovered. For each device, the system will broadcast the 320 {@link android.bluetooth.BluetoothDevice#ACTION_FOUND} Intent. This 321 Intent carries the extra fields 322 {@link android.bluetooth.BluetoothDevice#EXTRA_DEVICE} and 323 {@link android.bluetooth.BluetoothDevice#EXTRA_CLASS}, containing a 324 {@link android.bluetooth.BluetoothDevice} and a {@link 325 android.bluetooth.BluetoothClass}, respectively. For example, here's how you can 326 register to handle the broadcast when devices are discovered:</p> 327 <pre> 328 // Create a BroadcastReceiver for ACTION_FOUND 329 private final BroadcastReceiver mReceiver = new BroadcastReceiver() { 330 public void onReceive(Context context, Intent intent) { 331 String action = intent.getAction(); 332 // When discovery finds a device 333 if (BluetoothDevice.ACTION_FOUND.equals(action)) { 334 // Get the BluetoothDevice object from the Intent 335 BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE); 336 // Add the name and address to an array adapter to show in a ListView 337 mArrayAdapter.add(device.getName() + "\n" + device.getAddress()); 338 } 339 } 340 }; 341 // Register the BroadcastReceiver 342 IntentFilter filter = new IntentFilter(BluetoothDevice.ACTION_FOUND); 343 registerReceiver(mReceiver, filter); // Don't forget to unregister during onDestroy 344 </pre> 345 346 <p>All that's needed from the {@link android.bluetooth.BluetoothDevice} object 347 in order to initiate a 348 connection is the MAC address. In this example, it's saved as a part of an 349 ArrayAdapter that's shown to the user. The MAC address can later be extracted in 350 order to initiate the connection. You can learn more about creating a connection 351 in the section about <a href="#ConnectingDevices">Connecting Devices</a>.</p> 352 353 <p class="caution"><strong>Caution:</strong> Performing device discovery is 354 a heavy procedure for the Bluetooth 355 adapter and will consume a lot of its resources. Once you have found a device to 356 connect, be certain that you always stop discovery with 357 {@link android.bluetooth.BluetoothAdapter#cancelDiscovery()} before 358 attempting a connection. Also, if you 359 already hold a connection with a device, then performing discovery can 360 significantly reduce the bandwidth available for the connection, so you should 361 not perform discovery while connected.</p> 362 363 <h4 id="EnablingDiscoverability">Enabling discoverability</h4> 364 365 <p>If you would like to make the local device discoverable to other devices, 366 call {@link android.app.Activity#startActivityForResult(Intent,int)} with the 367 {@link android.bluetooth.BluetoothAdapter#ACTION_REQUEST_DISCOVERABLE} action Intent. 368 This will issue a request to enable discoverable mode through the system settings (without 369 stopping your application). By default, the device will become discoverable for 370 120 seconds. You can define a different duration by adding the 371 {@link android.bluetooth.BluetoothAdapter#EXTRA_DISCOVERABLE_DURATION} Intent extra 372 (maximum duration is 300 seconds). For example:</p> 373 <pre> 374 Intent discoverableIntent = new 375 Intent(BluetoothAdapter.ACTION_REQUEST_DISCOVERABLE); 376 discoverableIntent.putExtra(BluetoothAdapter.EXTRA_DISCOVERABLE_DURATION, 300); 377 startActivity(discoverableIntent); 378 </pre> 379 380 <div class="figure" style="width:200px"> 381 <img src="{@docRoot}images/bt_enable_discoverable.png" /> 382 <strong>Figure 2:</strong> The enabling discoverability dialog. 383 </div> 384 385 <p>A dialog will be displayed, requesting user permission to make the device 386 discoverable, as shown in Figure 2. If the user responds "Yes," then the device 387 will become discoverable for the specified amount of time. Your Activity will 388 then receive a call to the {@link android.app.Activity#onActivityResult(int,int,Intent) 389 onActivityResult())} callback, with the result code equal to the duration that the device 390 is discoverable. If the user responded "No" or if an error occurred, the result code will 391 be Activity.RESULT_CANCELLED.</p> 392 393 <p class="note"><strong>Note:</strong> If Bluetooth has not been enabled on the device, 394 then enabling device discoverability will automatically enable Bluetooth.</p> 395 396 <p>The device will silently remain in discoverable mode for the allotted time. 397 If you would like to be notified when the discoverable mode has changed, you can 398 register a BroadcastReceiver for the {@link 399 android.bluetooth.BluetoothAdapter#ACTION_SCAN_MODE_CHANGED} 400 Intent. This will contain the extra fields {@link 401 android.bluetooth.BluetoothAdapter#EXTRA_SCAN_MODE} and 402 {@link android.bluetooth.BluetoothAdapter#EXTRA_PREVIOUS_SCAN_MODE}, which tell you the 403 new and old scan mode, respectively. Possible values for each are 404 {@link android.bluetooth.BluetoothAdapter#SCAN_MODE_CONNECTABLE_DISCOVERABLE}, 405 {@link android.bluetooth.BluetoothAdapter#SCAN_MODE_CONNECTABLE}, or {@link 406 android.bluetooth.BluetoothAdapter#SCAN_MODE_NONE}, 407 which indicate that the device is either in discoverable mode, not in 408 discoverable mode but still able to receive connections, or not in discoverable 409 mode and unable to receive connections, respectively.</p> 410 411 <p>You do not need to enable device discoverability if you will be initiating 412 the connection to a remote device. Enabling discoverability is only necessary when 413 you want your application to host a server socket that will accept incoming 414 connections, because the remote devices must be able to discover the device 415 before it can initiate the connection.</p> 416 417 418 419 <h2 id="ConnectingDevices">Connecting Devices</h2> 420 421 <p>In order to create a connection between your application on two devices, you 422 must implement both the server-side and client-side mechanisms, because one 423 device must open a server socket and the other one must initiate the connection 424 (using the server device's MAC address to initiate a connection). The server and 425 client are considered connected to each other when they each have a connected 426 {@link android.bluetooth.BluetoothSocket} on the same RFCOMM channel. At this 427 point, each device can obtain input and output streams and data transfer can 428 begin, which is discussed in the section about <a 429 href="#ManagingConnection">Managing a Connection</a>. This section describes how 430 to initiate the connection between two devices.</p> 431 432 <p>The server device and the client device each obtain the required {@link 433 android.bluetooth.BluetoothSocket} in different ways. The server will receive it 434 when an incoming connection is accepted. The client will receive it when it 435 opens an RFCOMM channel to the server.</p> 436 437 <div class="figure" style="width:200px"> 438 <img src="{@docRoot}images/bt_pairing_request.png" /> 439 <strong>Figure 3:</strong> The Bluetooth pairing dialog. 440 </div> 441 442 <p>One implementation technique is to automatically prepare each device as a 443 server, so that each one has a server socket open and listening for connections. 444 Then either device can initiate a connection with the other and become the 445 client. Alternatively, one device can explicitly "host" the connection and open 446 a server socket on demand and the other device can simply initiate the 447 connection.</p> 448 449 <p class="note"><strong>Note:</strong> If the two devices have not been previously paired, 450 then the Android framework will automatically show a pairing request notification or 451 dialog to the user during the connection procedure, as shown in Figure 3. So 452 when attempting to connect devices, 453 your application does not need to be concerned about whether or not the devices are 454 paired. Your RFCOMM connection attempt will block until the user has successfully paired, 455 or will fail if the user rejects pairing, or if pairing fails or times out. </p> 456 457 458 <h3 id="ConnectingAsAServer">Connecting as a server</h3> 459 460 <p>When you want to connect two devices, one must act as a server by holding an 461 open {@link android.bluetooth.BluetoothServerSocket}. The purpose of the server 462 socket is to listen for incoming connection requests and when one is accepted, 463 provide a connected {@link android.bluetooth.BluetoothSocket}. When the {@link 464 android.bluetooth.BluetoothSocket} is acquired from the {@link 465 android.bluetooth.BluetoothServerSocket}, 466 the {@link android.bluetooth.BluetoothServerSocket} can (and should) be 467 discarded, unless you want to accept more connections.</p> 468 469 <div class="sidebox-wrapper"> 470 <div class="sidebox"> 471 <h2>About UUID</h2> 472 473 <p>A Universally Unique Identifier (UUID) is a standardized 128-bit format for a string 474 ID used to uniquely identify information. The point of a UUID is that it's big 475 enough that you can select any random and it won't clash. In this case, it's 476 used to uniquely identify your application's Bluetooth service. To get a UUID to 477 use with your application, you can use one of the many random UUID generators on 478 the web, then initialize a {@link java.util.UUID} with {@link 479 java.util.UUID#fromString(String)}.</p> 480 </div> 481 </div> 482 483 <p>Here's the basic procedure to set up a server socket and accept a 484 connection:</p> 485 486 <ol> 487 <li>Get a {@link android.bluetooth.BluetoothServerSocket} by calling the 488 {@link 489 android.bluetooth.BluetoothAdapter#listenUsingRfcommWithServiceRecord(String, 490 UUID)}. 491 <p>The string is an identifiable name of your service, which the system will 492 automatically write to a new Service Discovery Protocol (SDP) database entry on 493 the device (the name is arbitrary and can simply be your application name). The 494 UUID is also included in the SDP entry and will be the basis for the connection 495 agreement with the client device. That is, when the client attempts to connect 496 with this device, it will carry a UUID that uniquely identifies the service with 497 which it wants to connect. These UUIDs must match in order for the connection to 498 be accepted (in the next step).</p> 499 </li> 500 501 <li>Start listening for connection requests by calling 502 {@link android.bluetooth.BluetoothServerSocket#accept()}. 503 <p>This is a blocking call. It will return when either a connection has been 504 accepted or an exception has occurred. A connection is accepted only when a 505 remote device has sent a connection request with a UUID matching the one 506 registered with this listening server socket. When successful, {@link 507 android.bluetooth.BluetoothServerSocket#accept()} will 508 return a connected {@link android.bluetooth.BluetoothSocket}.</p> 509 </li> 510 511 <li>Unless you want to accept additional connections, call 512 {@link android.bluetooth.BluetoothServerSocket#close()}. 513 <p>This releases the server socket and all its resources, but does <em>not</em> close the 514 connected {@link android.bluetooth.BluetoothSocket} that's been returned by {@link 515 android.bluetooth.BluetoothServerSocket#accept()}. Unlike TCP/IP, RFCOMM only allows one 516 connected client per channel at a time, so in most cases it makes sense to call {@link 517 android.bluetooth.BluetoothServerSocket#close()} on the {@link 518 android.bluetooth.BluetoothServerSocket} immediately after accepting a connected 519 socket.</p> 520 </li> 521 </ol> 522 523 <p>The {@link android.bluetooth.BluetoothServerSocket#accept()} call should not 524 be executed in the main Activity UI thread because it is a blocking call and 525 will prevent any other interaction with the application. It usually makes 526 sense to do all work with a {@link android.bluetooth.BluetoothServerSocket} or {@link 527 android.bluetooth.BluetoothSocket} in a new 528 thread managed by your application. To abort a blocked call such as {@link 529 android.bluetooth.BluetoothServerSocket#accept()}, call {@link 530 android.bluetooth.BluetoothServerSocket#close()} on the {@link 531 android.bluetooth.BluetoothServerSocket} (or {@link 532 android.bluetooth.BluetoothSocket}) from another thread and the blocked call will 533 immediately return. Note that all methods on a {@link 534 android.bluetooth.BluetoothServerSocket} or {@link android.bluetooth.BluetoothSocket} 535 are thread-safe.</p> 536 537 <h4>Example</h4> 538 539 <p>Here's a simplified thread for the server component that accepts incoming 540 connections:</p> 541 <pre> 542 private class AcceptThread extends Thread { 543 private final BluetoothServerSocket mmServerSocket; 544 545 public AcceptThread() { 546 // Use a temporary object that is later assigned to mmServerSocket, 547 // because mmServerSocket is final 548 BluetoothServerSocket tmp = null; 549 try { 550 // MY_UUID is the app's UUID string, also used by the client code 551 tmp = mAdapter.listenUsingRfcommWithServiceRecord(NAME, MY_UUID); 552 } catch (IOException e) { } 553 mmServerSocket = tmp; 554 } 555 556 public void run() { 557 BluetoothSocket socket = null; 558 // Keep listening until exception occurs or a socket is returned 559 while (true) { 560 try { 561 socket = mmServerSocket.accept(); 562 } catch (IOException e) { 563 break; 564 } 565 // If a connection was accepted 566 if (socket != null) { 567 // Do work to manage the connection (in a separate thread) 568 manageConnectedSocket(socket); 569 mmServerSocket.close(); 570 break; 571 } 572 } 573 } 574 575 /** Will cancel the listening socket, and cause the thread to finish */ 576 public void cancel() { 577 try { 578 mmServerSocket.close(); 579 } catch (IOException e) { } 580 } 581 } 582 </pre> 583 584 <p>In this example, only one incoming connection is desired, so as soon as a 585 connection is accepted and the {@link android.bluetooth.BluetoothSocket} is 586 acquired, the application 587 sends the acquired {@link android.bluetooth.BluetoothSocket} to a separate 588 thread, closes the 589 {@link android.bluetooth.BluetoothServerSocket} and breaks the loop.</p> 590 591 <p>Note that when {@link android.bluetooth.BluetoothServerSocket#accept()} 592 returns the {@link android.bluetooth.BluetoothSocket}, the socket is already 593 connected, so you should <em>not</em> call {@link 594 android.bluetooth.BluetoothSocket#connect()} (as you do from the 595 client-side).</p> 596 597 <p><code>manageConnectedSocket()</code> is a fictional method in the application 598 that will 599 initiate the thread for transferring data, which is discussed in the section 600 about <a href="#ManagingConnection">Managing a Connection</a>.</p> 601 602 <p>You should usually close your {@link android.bluetooth.BluetoothServerSocket} 603 as soon as you are done listening for incoming connections. In this example, {@link 604 android.bluetooth.BluetoothServerSocket#close()} is called as soon 605 as the {@link android.bluetooth.BluetoothSocket} is acquired. You may also want 606 to provide a public method in your thread that can close the private {@link 607 android.bluetooth.BluetoothSocket} in the event that you need to stop listening on the 608 server socket.</p> 609 610 611 <h3 id="ConnectingAsAClient">Connecting as a client</h3> 612 613 <p>In order to initiate a connection with a remote device (a device holding an 614 open 615 server socket), you must first obtain a {@link 616 android.bluetooth.BluetoothDevice} object that represents the remote device. 617 (Getting a {@link android.bluetooth.BluetoothDevice} is covered in the above 618 section about <a 619 href="#FindingDevices">Finding Devices</a>.) You must then use the 620 {@link android.bluetooth.BluetoothDevice} to acquire a {@link 621 android.bluetooth.BluetoothSocket} and initiate the connection.</p> 622 623 <p>Here's the basic procedure:</p> 624 625 <ol> 626 <li>Using the {@link android.bluetooth.BluetoothDevice}, get a {@link 627 android.bluetooth.BluetoothSocket} by calling {@link 628 android.bluetooth.BluetoothDevice#createRfcommSocketToServiceRecord(UUID)}. 629 <p>This initializes a {@link android.bluetooth.BluetoothSocket} that will 630 connect to the {@link android.bluetooth.BluetoothDevice}. The UUID passed here 631 must match the UUID used by the server device when it opened its 632 {@link android.bluetooth.BluetoothServerSocket} (with {@link 633 android.bluetooth.BluetoothAdapter#listenUsingRfcommWithServiceRecord(String, 634 UUID)}). Using the same UUID is simply a matter of hard-coding the UUID string 635 into your application and then referencing it from both the server and client 636 code.</p> 637 </li> 638 639 <li>Initiate the connection by calling {@link 640 android.bluetooth.BluetoothSocket#connect()}. 641 <p>Upon this call, the system will perform an SDP lookup on the remote device in 642 order to match the UUID. If the lookup is successful and the remote device 643 accepts the connection, it will share the RFCOMM channel to use during the 644 connection and {@link 645 android.bluetooth.BluetoothSocket#connect()} will return. This method is a 646 blocking call. If, for 647 any reason, the connection fails or the {@link 648 android.bluetooth.BluetoothSocket#connect()} method times out (after about 649 12 seconds), then it will throw an exception.</p> 650 <p>Because {@link 651 android.bluetooth.BluetoothSocket#connect()} is a blocking call, this connection 652 procedure should always be performed in a thread separate from the main Activity 653 thread.</p> 654 <p class="note">Note: You should always ensure that the device is not performing 655 device discovery when you call {@link 656 android.bluetooth.BluetoothSocket#connect()}. If discovery is in progress, then 657 the 658 connection attempt will be significantly slowed and is more likely to fail.</p> 659 </li> 660 </ol> 661 662 <h4>Example</h4> 663 664 <p>Here is a basic example of a thread that initiates a Bluetooth 665 connection:</p> 666 <pre> 667 private class ConnectThread extends Thread { 668 private final BluetoothSocket mmSocket; 669 private final BluetoothDevice mmDevice; 670 671 public ConnectThread(BluetoothDevice device) { 672 // Use a temporary object that is later assigned to mmSocket, 673 // because mmSocket is final 674 BluetoothSocket tmp = null; 675 mmDevice = device; 676 677 // Get a BluetoothSocket to connect with the given BluetoothDevice 678 try { 679 // MY_UUID is the app's UUID string, also used by the server code 680 tmp = device.createRfcommSocketToServiceRecord(MY_UUID); 681 } catch (IOException e) { } 682 mmSocket = tmp; 683 } 684 685 public void run() { 686 // Cancel discovery because it will slow down the connection 687 mAdapter.cancelDiscovery(); 688 689 try { 690 // Connect the device through the socket. This will block 691 // until it succeeds or throws an exception 692 mmSocket.connect(); 693 } catch (IOException connectException) { 694 // Unable to connect; close the socket and get out 695 try { 696 mmSocket.close(); 697 } catch (IOException closeException) { } 698 return; 699 } 700 701 // Do work to manage the connection (in a separate thread) 702 manageConnectedSocket(mmSocket); 703 } 704 705 /** Will cancel an in-progress connection, and close the socket */ 706 public void cancel() { 707 try { 708 mmSocket.close(); 709 } catch (IOException e) { } 710 } 711 } 712 </pre> 713 714 <p>Notice that {@link android.bluetooth.BluetoothAdapter#cancelDiscovery()} is called 715 before the connection is made. You should always do this before connecting and it is safe 716 to call without actually checking whether it is running or not (but if you do want to 717 check, call {@link android.bluetooth.BluetoothAdapter#isDiscovering()}).</p> 718 719 <p><code>manageConnectedSocket()</code> is a fictional method in the application 720 that will initiate the thread for transferring data, which is discussed in the section 721 about <a href="#ManagingConnection">Managing a Connection</a>.</p> 722 723 <p>When you're done with your {@link android.bluetooth.BluetoothSocket}, always 724 call {@link android.bluetooth.BluetoothSocket#close()} to clean up. 725 Doing so will immediately close the connected socket and clean up all internal 726 resources.</p> 727 728 729 <h2 id="ManagingAConnection">Managing a Connection</h2> 730 731 <p>When you have successfully connected two (or more) devices, each one will 732 have a connected {@link android.bluetooth.BluetoothSocket}. This is where the fun 733 begins because you can share data between devices. Using the {@link 734 android.bluetooth.BluetoothSocket}, the general procedure to transfer arbitrary data is 735 simple:</p> 736 <ol> 737 <li>Get the {@link java.io.InputStream} and {@link java.io.OutputStream} that 738 handle transmissions through the socket, via {@link 739 android.bluetooth.BluetoothSocket#getInputStream()} and 740 {@link android.bluetooth.BluetoothSocket#getOutputStream}, respectively.</li> 741 742 <li>Read and write data to the streams with {@link 743 java.io.InputStream#read(byte[])} and {@link java.io.OutputStream#write(byte[])}.</li> 744 </ol> 745 746 <p>That's it.</p> 747 748 <p>There are, of course, implementation details to consider. First and foremost, 749 you should use a dedicated thread for all stream reading and writing. This is 750 important because both {@link java.io.InputStream#read(byte[])} and {@link 751 java.io.OutputStream#write(byte[])} methods are blocking calls. {@link 752 java.io.InputStream#read(byte[])} will block until there is something to read 753 from the stream. {@link java.io.OutputStream#write(byte[])} does not usually 754 block, but can block for flow control if the remote device is not calling {@link 755 java.io.InputStream#read(byte[])} quickly enough and the intermediate buffers are full. 756 So, your main loop in the thread should be dedicated to reading from the {@link 757 java.io.InputStream}. A separate public method in the thread can be used to initiate 758 writes to the {@link java.io.OutputStream}.</p> 759 760 <h4>Example</h4> 761 762 <p>Here's an example of how this might look:</p> 763 <pre> 764 private class ConnectedThread extends Thread { 765 private final BluetoothSocket mmSocket; 766 private final InputStream mmInStream; 767 private final OutputStream mmOutStream; 768 769 public ConnectedThread(BluetoothSocket socket) { 770 mmSocket = socket; 771 InputStream tmpIn = null; 772 OutputStream tmpOut = null; 773 774 // Get the input and output streams, using temp objects because 775 // member streams are final 776 try { 777 tmpIn = socket.getInputStream(); 778 tmpOut = socket.getOutputStream(); 779 } catch (IOException e) { } 780 781 mmInStream = tmpIn; 782 mmOutStream = tmpOut; 783 } 784 785 public void run() { 786 byte[] buffer = new byte[1024]; // buffer store for the stream 787 int bytes; // bytes returned from read() 788 789 // Keep listening to the InputStream until an exception occurs 790 while (true) { 791 try { 792 // Read from the InputStream 793 bytes = mmInStream.read(buffer); 794 // Send the obtained bytes to the UI Activity 795 mHandler.obtainMessage(MESSAGE_READ, bytes, -1, buffer) 796 .sendToTarget(); 797 } catch (IOException e) { 798 break; 799 } 800 } 801 } 802 803 /* Call this from the main Activity to send data to the remote device */ 804 public void write(byte[] bytes) { 805 try { 806 mmOutStream.write(bytes); 807 } catch (IOException e) { } 808 } 809 810 /* Call this from the main Activity to shutdown the connection */ 811 public void cancel() { 812 try { 813 mmSocket.close(); 814 } catch (IOException e) { } 815 } 816 } 817 </pre> 818 819 <p>The constructor acquires the necessary streams and once executed, the thread 820 will wait for data to come through the InputStream. When {@link 821 java.io.InputStream#read(byte[])} returns with 822 bytes from the stream, the data is sent to the main Activity using a member 823 Handler from the parent class. Then it goes back and waits for more bytes from 824 the stream.</p> 825 826 <p>Sending outgoing data is as simple as calling the thread's 827 <code>write()</code> method from the main Activity and passing in the bytes to 828 be sent. This method then simply calls {@link 829 java.io.OutputStream#write(byte[])} to send the data to the remote device.</p> 830 831 <p>The thread's <code>cancel()</code> method is important so that the connection 832 can be 833 terminated at any time by closing the {@link android.bluetooth.BluetoothSocket}. 834 This should always be called when you're done using the Bluetooth 835 connection.</p> 836 837 <div class="special"> 838 <p>For a complete demonstration using the Bluetooth APIs, see the <a 839 href="{@docRoot}resources/samples/BluetoothChat/index.html">Bluetooth Chat sample app</a>.</p> 840 </div> 841