1 /* 2 * Copyright (C) 2019 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.car.trust; 18 19 import static android.car.Car.PERMISSION_CAR_ENROLL_TRUST; 20 21 import android.annotation.IntDef; 22 import android.annotation.NonNull; 23 import android.annotation.Nullable; 24 import android.annotation.RequiresPermission; 25 import android.annotation.SystemApi; 26 import android.bluetooth.BluetoothDevice; 27 import android.car.Car; 28 import android.car.CarManagerBase; 29 import android.os.Bundle; 30 import android.os.Handler; 31 import android.os.IBinder; 32 import android.os.Looper; 33 import android.os.Message; 34 import android.os.RemoteException; 35 import android.util.Log; 36 37 import com.android.internal.annotations.GuardedBy; 38 39 import java.lang.annotation.Retention; 40 import java.lang.annotation.RetentionPolicy; 41 import java.lang.ref.WeakReference; 42 import java.util.Collections; 43 import java.util.List; 44 45 46 /** 47 * APIs to help enroll a remote device as a trusted device that can be used to authenticate a user 48 * in the head unit. 49 * <p> 50 * The call sequence to add a new trusted device from the client should be as follows: 51 * <ol> 52 * <li> setEnrollmentCallback() 53 * <li> setBleCallback(bleCallback) 54 * <li> startEnrollmentAdvertising() 55 * <li> wait for onEnrollmentAdvertisingStarted() or 56 * <li> wait for onBleEnrollmentDeviceConnected() and check if the device connected is the right 57 * one. 58 * <li> initiateEnrollmentHandshake() 59 * <li> wait for onAuthStringAvailable() to get the pairing code to display to the user 60 * <li> enrollmentHandshakeAccepted() after user confirms the pairing code 61 * <li> wait for onEscrowTokenAdded() 62 * <li> Authenticate user's credentials by showing the lock screen 63 * <li> activateToken() 64 * <li> wait for onEscrowTokenActiveStateChanged() to add the device as a trusted device and show 65 * in the list 66 * </ol> 67 * 68 * @hide 69 */ 70 @SystemApi 71 public final class CarTrustAgentEnrollmentManager extends CarManagerBase { 72 private static final String TAG = "CarTrustEnrollMgr"; 73 private static final String KEY_HANDLE = "handle"; 74 private static final String KEY_ACTIVE = "active"; 75 private static final int MSG_ENROLL_ADVERTISING_STARTED = 0; 76 private static final int MSG_ENROLL_ADVERTISING_FAILED = 1; 77 private static final int MSG_ENROLL_DEVICE_CONNECTED = 2; 78 private static final int MSG_ENROLL_DEVICE_DISCONNECTED = 3; 79 private static final int MSG_ENROLL_HANDSHAKE_FAILURE = 4; 80 private static final int MSG_ENROLL_AUTH_STRING_AVAILABLE = 5; 81 private static final int MSG_ENROLL_TOKEN_ADDED = 6; 82 private static final int MSG_ENROLL_TOKEN_STATE_CHANGED = 7; 83 private static final int MSG_ENROLL_TOKEN_REMOVED = 8; 84 85 private final ICarTrustAgentEnrollment mEnrollmentService; 86 private Object mListenerLock = new Object(); 87 @GuardedBy("mListenerLock") 88 private CarTrustAgentEnrollmentCallback mEnrollmentCallback; 89 @GuardedBy("mListenerLock") 90 private CarTrustAgentBleCallback mBleCallback; 91 @GuardedBy("mListenerLock") 92 private final ListenerToEnrollmentService mListenerToEnrollmentService = 93 new ListenerToEnrollmentService(this); 94 private final ListenerToBleService mListenerToBleService = new ListenerToBleService(this); 95 private final EventCallbackHandler mEventCallbackHandler; 96 97 /** 98 * Enrollment Handshake failed. 99 */ 100 public static final int ENROLLMENT_HANDSHAKE_FAILURE = 1; 101 /** 102 * Enrollment of a new device is not allowed. This happens when either the whole feature is 103 * disabled or just the enrollment is disabled. Useful when feature needs to be disabled 104 * in a lost/stolen phone scenario. 105 */ 106 public static final int ENROLLMENT_NOT_ALLOWED = 2; 107 108 /** @hide */ 109 @Retention(RetentionPolicy.SOURCE) 110 @IntDef({ENROLLMENT_HANDSHAKE_FAILURE, 111 ENROLLMENT_NOT_ALLOWED}) 112 public @interface TrustedDeviceEnrollmentError { 113 } 114 115 116 /** @hide */ CarTrustAgentEnrollmentManager(Car car, IBinder service)117 public CarTrustAgentEnrollmentManager(Car car, IBinder service) { 118 super(car); 119 mEnrollmentService = ICarTrustAgentEnrollment.Stub.asInterface(service); 120 mEventCallbackHandler = new EventCallbackHandler(this, getEventHandler().getLooper()); 121 } 122 123 /** @hide */ 124 @Override onCarDisconnected()125 public synchronized void onCarDisconnected() { 126 } 127 128 /** 129 * Starts broadcasting enrollment UUID on BLE. 130 * Phones can scan and connect for the enrollment process to begin. 131 */ 132 @RequiresPermission(PERMISSION_CAR_ENROLL_TRUST) startEnrollmentAdvertising()133 public void startEnrollmentAdvertising() { 134 try { 135 mEnrollmentService.startEnrollmentAdvertising(); 136 } catch (RemoteException e) { 137 handleRemoteExceptionFromCarService(e); 138 } 139 } 140 141 /** 142 * Stops Enrollment advertising. 143 */ 144 @RequiresPermission(PERMISSION_CAR_ENROLL_TRUST) stopEnrollmentAdvertising()145 public void stopEnrollmentAdvertising() { 146 try { 147 mEnrollmentService.stopEnrollmentAdvertising(); 148 } catch (RemoteException e) { 149 handleRemoteExceptionFromCarService(e); 150 } 151 } 152 153 /** 154 * Confirms that the enrollment handshake has been accepted by the user. This should be called 155 * after the user has confirmed the verification code displayed on the UI. 156 * 157 * @param device the remote Bluetooth device that will receive the signal. 158 */ 159 @RequiresPermission(PERMISSION_CAR_ENROLL_TRUST) enrollmentHandshakeAccepted(BluetoothDevice device)160 public void enrollmentHandshakeAccepted(BluetoothDevice device) { 161 try { 162 mEnrollmentService.enrollmentHandshakeAccepted(device); 163 } catch (RemoteException e) { 164 handleRemoteExceptionFromCarService(e); 165 } 166 } 167 168 /** 169 * Provides an option to quit enrollment if the pairing code doesn't match for example. 170 */ 171 @RequiresPermission(PERMISSION_CAR_ENROLL_TRUST) terminateEnrollmentHandshake()172 public void terminateEnrollmentHandshake() { 173 try { 174 mEnrollmentService.terminateEnrollmentHandshake(); 175 } catch (RemoteException e) { 176 handleRemoteExceptionFromCarService(e); 177 } 178 } 179 180 /** 181 * Returns {@code true} if the escrow token associated with the given handle is active. 182 * <p> 183 * When a new escrow token has been added as part of the Trusted device enrollment, the client 184 * will receive {@link CarTrustAgentEnrollmentCallback#onEscrowTokenAdded(long)} and 185 * {@link CarTrustAgentEnrollmentCallback#onEscrowTokenActiveStateChanged(long, boolean)} 186 * callbacks. This method provides a way to query for the token state at a later point of time. 187 * 188 * @param handle the handle corresponding to the escrow token 189 * @param uid user id associated with the token 190 * @return true if the token is active, false if not 191 */ 192 @RequiresPermission(PERMISSION_CAR_ENROLL_TRUST) isEscrowTokenActive(long handle, int uid)193 public boolean isEscrowTokenActive(long handle, int uid) { 194 try { 195 return mEnrollmentService.isEscrowTokenActive(handle, uid); 196 } catch (RemoteException e) { 197 return handleRemoteExceptionFromCarService(e, false); 198 } 199 } 200 201 /** 202 * Remove the escrow token that is associated with the given handle and uid. 203 * 204 * @param handle the handle associated with the escrow token 205 * @param uid user id associated with the token 206 */ 207 @RequiresPermission(PERMISSION_CAR_ENROLL_TRUST) removeEscrowToken(long handle, int uid)208 public void removeEscrowToken(long handle, int uid) { 209 try { 210 mEnrollmentService.removeEscrowToken(handle, uid); 211 } catch (RemoteException e) { 212 handleRemoteExceptionFromCarService(e); 213 } 214 } 215 216 /** 217 * Remove all of the trusted devices associated with the given user. 218 * 219 * @param uid User id to remove the devices for 220 */ 221 @RequiresPermission(PERMISSION_CAR_ENROLL_TRUST) removeAllTrustedDevices(int uid)222 public void removeAllTrustedDevices(int uid) { 223 try { 224 mEnrollmentService.removeAllTrustedDevices(uid); 225 } catch (RemoteException e) { 226 handleRemoteExceptionFromCarService(e); 227 } 228 } 229 230 /** 231 * Enable or Disable Trusted device enrollment. Once disabled, head unit will not broadcast 232 * for enrollment until enabled back. 233 * 234 * @param isEnabled {@code true} enables enrollment. 235 */ 236 @RequiresPermission(PERMISSION_CAR_ENROLL_TRUST) setTrustedDeviceEnrollmentEnabled(boolean isEnabled)237 public void setTrustedDeviceEnrollmentEnabled(boolean isEnabled) { 238 try { 239 mEnrollmentService.setTrustedDeviceEnrollmentEnabled(isEnabled); 240 } catch (RemoteException e) { 241 handleRemoteExceptionFromCarService(e); 242 } 243 } 244 245 /** 246 * Enable or disable Unlocking with a trusted device. Once disabled, head unit will not 247 * broadcast until enabled back. 248 * 249 * @param isEnabled {@code true} enables unlock. 250 */ 251 @RequiresPermission(PERMISSION_CAR_ENROLL_TRUST) setTrustedDeviceUnlockEnabled(boolean isEnabled)252 public void setTrustedDeviceUnlockEnabled(boolean isEnabled) { 253 try { 254 mEnrollmentService.setTrustedDeviceUnlockEnabled(isEnabled); 255 } catch (RemoteException e) { 256 handleRemoteExceptionFromCarService(e); 257 } 258 } 259 260 /** 261 * Register for enrollment event callbacks. 262 * 263 * @param callback The callback methods to call, null to unregister 264 */ 265 @RequiresPermission(PERMISSION_CAR_ENROLL_TRUST) setEnrollmentCallback(@ullable CarTrustAgentEnrollmentCallback callback)266 public void setEnrollmentCallback(@Nullable CarTrustAgentEnrollmentCallback callback) { 267 if (callback == null) { 268 unregisterEnrollmentCallback(); 269 } else { 270 registerEnrollmentCallback(callback); 271 } 272 } 273 registerEnrollmentCallback(CarTrustAgentEnrollmentCallback callback)274 private void registerEnrollmentCallback(CarTrustAgentEnrollmentCallback callback) { 275 synchronized (mListenerLock) { 276 if (callback != null && mEnrollmentCallback == null) { 277 try { 278 mEnrollmentService.registerEnrollmentCallback(mListenerToEnrollmentService); 279 mEnrollmentCallback = callback; 280 } catch (RemoteException e) { 281 handleRemoteExceptionFromCarService(e); 282 } 283 } 284 } 285 } 286 unregisterEnrollmentCallback()287 private void unregisterEnrollmentCallback() { 288 synchronized (mListenerLock) { 289 if (mEnrollmentCallback != null) { 290 try { 291 mEnrollmentService.unregisterEnrollmentCallback(mListenerToEnrollmentService); 292 } catch (RemoteException e) { 293 handleRemoteExceptionFromCarService(e); 294 } 295 mEnrollmentCallback = null; 296 } 297 } 298 } 299 300 /** 301 * Register for general BLE callbacks 302 * 303 * @param callback The callback methods to call, null to unregister 304 */ 305 @RequiresPermission(PERMISSION_CAR_ENROLL_TRUST) setBleCallback(@ullable CarTrustAgentBleCallback callback)306 public void setBleCallback(@Nullable CarTrustAgentBleCallback callback) { 307 if (callback == null) { 308 unregisterBleCallback(); 309 } else { 310 registerBleCallback(callback); 311 } 312 } 313 registerBleCallback(CarTrustAgentBleCallback callback)314 private void registerBleCallback(CarTrustAgentBleCallback callback) { 315 synchronized (mListenerLock) { 316 if (callback != null && mBleCallback == null) { 317 try { 318 mEnrollmentService.registerBleCallback(mListenerToBleService); 319 mBleCallback = callback; 320 } catch (RemoteException e) { 321 handleRemoteExceptionFromCarService(e); 322 } 323 } 324 } 325 } 326 unregisterBleCallback()327 private void unregisterBleCallback() { 328 synchronized (mListenerLock) { 329 if (mBleCallback != null) { 330 try { 331 mEnrollmentService.unregisterBleCallback(mListenerToBleService); 332 } catch (RemoteException e) { 333 handleRemoteExceptionFromCarService(e); 334 } 335 mBleCallback = null; 336 } 337 } 338 } 339 340 /** 341 * Provides a list that contains information about the enrolled devices for the given user id. 342 * <p> 343 * Each enrollment handle corresponds to a trusted device for the given user. 344 * 345 * @param uid user id. 346 * @return list of the Enrollment handles and user names for the user id. 347 */ 348 @RequiresPermission(PERMISSION_CAR_ENROLL_TRUST) 349 @NonNull getEnrolledDeviceInfoForUser(int uid)350 public List<TrustedDeviceInfo> getEnrolledDeviceInfoForUser(int uid) { 351 try { 352 return mEnrollmentService.getEnrolledDeviceInfosForUser(uid); 353 } catch (RemoteException e) { 354 return handleRemoteExceptionFromCarService(e, Collections.emptyList()); 355 } 356 } 357 getEventCallbackHandler()358 private Handler getEventCallbackHandler() { 359 return mEventCallbackHandler; 360 } 361 362 /** 363 * Callback interface for Trusted device enrollment applications to implement. The applications 364 * get notified on various enrollment state change events. 365 */ 366 public interface CarTrustAgentEnrollmentCallback { 367 /** 368 * Communicate about failure/timeouts in the handshake process. BluetoothDevice will be 369 * null when the returned error code is {@link #ENROLLMENT_NOT_ALLOWED}. 370 * 371 * @param device the remote device trying to enroll 372 * @param errorCode information on what failed. 373 */ onEnrollmentHandshakeFailure(@ullable BluetoothDevice device, @TrustedDeviceEnrollmentError int errorCode)374 void onEnrollmentHandshakeFailure(@Nullable BluetoothDevice device, 375 @TrustedDeviceEnrollmentError int errorCode); 376 377 /** 378 * Present the pairing/authentication string to the user. 379 * 380 * @param device the remote device trying to enroll 381 * @param authString the authentication string to show to the user to confirm across 382 * both devices 383 */ onAuthStringAvailable(BluetoothDevice device, String authString)384 void onAuthStringAvailable(BluetoothDevice device, String authString); 385 386 /** 387 * Escrow token was received and the Trust Agent framework has generated a corresponding 388 * handle. 389 * 390 * @param handle the handle associated with the escrow token. 391 */ onEscrowTokenAdded(long handle)392 void onEscrowTokenAdded(long handle); 393 394 /** 395 * Escrow token was removed as a result of a call to {@link #removeEscrowToken(long handle, 396 * int uid)}. The peer device associated with this token is not trusted for authentication 397 * anymore. 398 * 399 * @param handle the handle associated with the escrow token. 400 */ onEscrowTokenRemoved(long handle)401 void onEscrowTokenRemoved(long handle); 402 403 404 /** 405 * Escrow token's active state changed. 406 * 407 * @param handle the handle associated with the escrow token 408 * @param active True if token has been activated, false if not. 409 */ onEscrowTokenActiveStateChanged(long handle, boolean active)410 void onEscrowTokenActiveStateChanged(long handle, boolean active); 411 } 412 413 /** 414 * Callback interface for Trusted device enrollment applications to implement. The applications 415 * get notified on various BLE state change events that happen during trusted device enrollment. 416 */ 417 public interface CarTrustAgentBleCallback { 418 /** 419 * Indicates a remote device connected on BLE. 420 */ onBleEnrollmentDeviceConnected(BluetoothDevice device)421 void onBleEnrollmentDeviceConnected(BluetoothDevice device); 422 423 /** 424 * Indicates a remote device disconnected on BLE. 425 */ onBleEnrollmentDeviceDisconnected(BluetoothDevice device)426 void onBleEnrollmentDeviceDisconnected(BluetoothDevice device); 427 428 /** 429 * Indicates that the device is broadcasting for trusted device enrollment on BLE. 430 */ onEnrollmentAdvertisingStarted()431 void onEnrollmentAdvertisingStarted(); 432 433 /** 434 * Indicates a failure in BLE broadcasting for enrollment. 435 */ onEnrollmentAdvertisingFailed()436 void onEnrollmentAdvertisingFailed(); 437 } 438 439 private static final class ListenerToEnrollmentService extends 440 ICarTrustAgentEnrollmentCallback.Stub { 441 private final WeakReference<CarTrustAgentEnrollmentManager> mMgr; 442 ListenerToEnrollmentService(CarTrustAgentEnrollmentManager mgr)443 ListenerToEnrollmentService(CarTrustAgentEnrollmentManager mgr) { 444 mMgr = new WeakReference<>(mgr); 445 } 446 447 /** 448 * Communicate about failure/timeouts in the handshake process. 449 */ 450 @Override onEnrollmentHandshakeFailure(BluetoothDevice device, @TrustedDeviceEnrollmentError int errorCode)451 public void onEnrollmentHandshakeFailure(BluetoothDevice device, 452 @TrustedDeviceEnrollmentError int errorCode) { 453 CarTrustAgentEnrollmentManager enrollmentManager = mMgr.get(); 454 if (enrollmentManager == null) { 455 return; 456 } 457 enrollmentManager.getEventCallbackHandler().sendMessage( 458 enrollmentManager.getEventCallbackHandler().obtainMessage( 459 MSG_ENROLL_HANDSHAKE_FAILURE, new AuthInfo(device, null, errorCode))); 460 } 461 462 /** 463 * Present the pairing/authentication string to the user. 464 */ 465 @Override onAuthStringAvailable(BluetoothDevice device, String authString)466 public void onAuthStringAvailable(BluetoothDevice device, String authString) { 467 CarTrustAgentEnrollmentManager enrollmentManager = mMgr.get(); 468 if (enrollmentManager == null) { 469 return; 470 } 471 enrollmentManager.getEventCallbackHandler().sendMessage( 472 enrollmentManager.getEventCallbackHandler().obtainMessage( 473 MSG_ENROLL_AUTH_STRING_AVAILABLE, new AuthInfo(device, authString, 0))); 474 } 475 476 /** 477 * Escrow token was received and the Trust Agent framework has generated a corresponding 478 * handle. 479 */ 480 @Override onEscrowTokenAdded(long handle)481 public void onEscrowTokenAdded(long handle) { 482 CarTrustAgentEnrollmentManager enrollmentManager = mMgr.get(); 483 if (enrollmentManager == null) { 484 return; 485 } 486 Message message = enrollmentManager.getEventCallbackHandler().obtainMessage( 487 MSG_ENROLL_TOKEN_ADDED); 488 Bundle data = new Bundle(); 489 data.putLong(KEY_HANDLE, handle); 490 message.setData(data); 491 enrollmentManager.getEventCallbackHandler().sendMessage(message); 492 } 493 494 /** 495 * Escrow token was removed. 496 */ 497 @Override onEscrowTokenRemoved(long handle)498 public void onEscrowTokenRemoved(long handle) { 499 CarTrustAgentEnrollmentManager enrollmentManager = mMgr.get(); 500 if (enrollmentManager == null) { 501 return; 502 } 503 Message message = enrollmentManager.getEventCallbackHandler().obtainMessage( 504 MSG_ENROLL_TOKEN_REMOVED); 505 Bundle data = new Bundle(); 506 data.putLong(KEY_HANDLE, handle); 507 message.setData(data); 508 enrollmentManager.getEventCallbackHandler().sendMessage(message); 509 } 510 511 /** 512 * Escrow token's active state changed. 513 */ 514 @Override onEscrowTokenActiveStateChanged(long handle, boolean active)515 public void onEscrowTokenActiveStateChanged(long handle, boolean active) { 516 CarTrustAgentEnrollmentManager enrollmentManager = mMgr.get(); 517 if (enrollmentManager == null) { 518 return; 519 } 520 Message message = enrollmentManager.getEventCallbackHandler().obtainMessage( 521 MSG_ENROLL_TOKEN_STATE_CHANGED); 522 Bundle data = new Bundle(); 523 data.putLong(KEY_HANDLE, handle); 524 data.putBoolean(KEY_ACTIVE, active); 525 message.setData(data); 526 enrollmentManager.getEventCallbackHandler().sendMessage(message); 527 } 528 } 529 530 private static final class ListenerToBleService extends ICarTrustAgentBleCallback.Stub { 531 private final WeakReference<CarTrustAgentEnrollmentManager> mMgr; 532 ListenerToBleService(CarTrustAgentEnrollmentManager mgr)533 ListenerToBleService(CarTrustAgentEnrollmentManager mgr) { 534 mMgr = new WeakReference<>(mgr); 535 } 536 537 /** 538 * Called when the GATT server is started and BLE is successfully advertising for 539 * enrollment. 540 */ onEnrollmentAdvertisingStarted()541 public void onEnrollmentAdvertisingStarted() { 542 CarTrustAgentEnrollmentManager enrollmentManager = mMgr.get(); 543 if (enrollmentManager == null) { 544 return; 545 } 546 enrollmentManager.getEventCallbackHandler().sendMessage( 547 enrollmentManager.getEventCallbackHandler().obtainMessage( 548 MSG_ENROLL_ADVERTISING_STARTED)); 549 } 550 551 /** 552 * Called when the BLE enrollment advertisement fails to start. 553 * see AdvertiseCallback#ADVERTISE_FAILED_* for possible error codes. 554 */ onEnrollmentAdvertisingFailed()555 public void onEnrollmentAdvertisingFailed() { 556 CarTrustAgentEnrollmentManager enrollmentManager = mMgr.get(); 557 if (enrollmentManager == null) { 558 return; 559 } 560 enrollmentManager.getEventCallbackHandler().sendMessage( 561 enrollmentManager.getEventCallbackHandler().obtainMessage( 562 MSG_ENROLL_ADVERTISING_FAILED)); 563 } 564 565 /** 566 * Called when a remote device is connected on BLE. 567 */ onBleEnrollmentDeviceConnected(BluetoothDevice device)568 public void onBleEnrollmentDeviceConnected(BluetoothDevice device) { 569 CarTrustAgentEnrollmentManager enrollmentManager = mMgr.get(); 570 if (enrollmentManager == null) { 571 return; 572 } 573 enrollmentManager.getEventCallbackHandler().sendMessage( 574 enrollmentManager.getEventCallbackHandler().obtainMessage( 575 MSG_ENROLL_DEVICE_CONNECTED, device)); 576 } 577 578 /** 579 * Called when a remote device is disconnected on BLE. 580 */ onBleEnrollmentDeviceDisconnected(BluetoothDevice device)581 public void onBleEnrollmentDeviceDisconnected(BluetoothDevice device) { 582 CarTrustAgentEnrollmentManager enrollmentManager = mMgr.get(); 583 if (enrollmentManager == null) { 584 return; 585 } 586 enrollmentManager.getEventCallbackHandler().sendMessage( 587 enrollmentManager.getEventCallbackHandler().obtainMessage( 588 MSG_ENROLL_DEVICE_DISCONNECTED, device)); 589 } 590 } 591 592 /** 593 * Callback Handler to handle dispatching the enrollment state changes to the corresponding 594 * listeners 595 */ 596 private static final class EventCallbackHandler extends Handler { 597 private final WeakReference<CarTrustAgentEnrollmentManager> mEnrollmentManager; 598 EventCallbackHandler(CarTrustAgentEnrollmentManager manager, Looper looper)599 EventCallbackHandler(CarTrustAgentEnrollmentManager manager, Looper looper) { 600 super(looper); 601 mEnrollmentManager = new WeakReference<>(manager); 602 } 603 604 @Override handleMessage(Message message)605 public void handleMessage(Message message) { 606 CarTrustAgentEnrollmentManager enrollmentManager = mEnrollmentManager.get(); 607 if (enrollmentManager == null) { 608 return; 609 } 610 switch (message.what) { 611 case MSG_ENROLL_ADVERTISING_STARTED: 612 case MSG_ENROLL_ADVERTISING_FAILED: 613 case MSG_ENROLL_DEVICE_CONNECTED: 614 case MSG_ENROLL_DEVICE_DISCONNECTED: 615 enrollmentManager.dispatchBleCallback(message); 616 break; 617 case MSG_ENROLL_HANDSHAKE_FAILURE: 618 case MSG_ENROLL_AUTH_STRING_AVAILABLE: 619 case MSG_ENROLL_TOKEN_ADDED: 620 case MSG_ENROLL_TOKEN_STATE_CHANGED: 621 case MSG_ENROLL_TOKEN_REMOVED: 622 enrollmentManager.dispatchEnrollmentCallback(message); 623 break; 624 default: 625 Log.e(TAG, "Unknown message:" + message.what); 626 break; 627 } 628 } 629 } 630 631 /** 632 * Dispatch BLE related state change callbacks 633 * 634 * @param message Message to handle and dispatch 635 */ dispatchBleCallback(Message message)636 private void dispatchBleCallback(Message message) { 637 CarTrustAgentBleCallback bleCallback; 638 synchronized (mListenerLock) { 639 bleCallback = mBleCallback; 640 } 641 if (bleCallback == null) { 642 return; 643 } 644 switch (message.what) { 645 case MSG_ENROLL_ADVERTISING_STARTED: 646 bleCallback.onEnrollmentAdvertisingStarted(); 647 break; 648 case MSG_ENROLL_ADVERTISING_FAILED: 649 bleCallback.onEnrollmentAdvertisingFailed(); 650 break; 651 case MSG_ENROLL_DEVICE_CONNECTED: 652 bleCallback.onBleEnrollmentDeviceConnected((BluetoothDevice) message.obj); 653 break; 654 case MSG_ENROLL_DEVICE_DISCONNECTED: 655 bleCallback.onBleEnrollmentDeviceDisconnected((BluetoothDevice) message.obj); 656 break; 657 default: 658 break; 659 } 660 } 661 662 /** 663 * Dispatch Enrollment related state changes to the listener. 664 * 665 * @param message Message to handle and dispatch 666 */ dispatchEnrollmentCallback(Message message)667 private void dispatchEnrollmentCallback(Message message) { 668 CarTrustAgentEnrollmentCallback enrollmentCallback; 669 synchronized (mListenerLock) { 670 enrollmentCallback = mEnrollmentCallback; 671 } 672 if (enrollmentCallback == null) { 673 return; 674 } 675 AuthInfo auth; 676 Bundle data; 677 switch (message.what) { 678 case MSG_ENROLL_HANDSHAKE_FAILURE: 679 auth = (AuthInfo) message.obj; 680 enrollmentCallback.onEnrollmentHandshakeFailure(auth.mDevice, auth.mErrorCode); 681 break; 682 case MSG_ENROLL_AUTH_STRING_AVAILABLE: 683 auth = (AuthInfo) message.obj; 684 if (auth.mDevice != null && auth.mAuthString != null) { 685 enrollmentCallback.onAuthStringAvailable(auth.mDevice, auth.mAuthString); 686 } 687 break; 688 case MSG_ENROLL_TOKEN_ADDED: 689 data = message.getData(); 690 if (data == null) { 691 break; 692 } 693 enrollmentCallback.onEscrowTokenAdded(data.getLong(KEY_HANDLE)); 694 break; 695 case MSG_ENROLL_TOKEN_STATE_CHANGED: 696 data = message.getData(); 697 if (data == null) { 698 break; 699 } 700 enrollmentCallback.onEscrowTokenActiveStateChanged(data.getLong(KEY_HANDLE), 701 data.getBoolean(KEY_ACTIVE)); 702 break; 703 case MSG_ENROLL_TOKEN_REMOVED: 704 data = message.getData(); 705 if (data == null) { 706 break; 707 } 708 enrollmentCallback.onEscrowTokenRemoved(data.getLong(KEY_HANDLE)); 709 break; 710 default: 711 break; 712 } 713 } 714 715 /** 716 * Container class to pass information through a Message to the handler. 717 */ 718 private static class AuthInfo { 719 final BluetoothDevice mDevice; 720 @Nullable 721 final String mAuthString; 722 final int mErrorCode; 723 AuthInfo(BluetoothDevice device, @Nullable String authString, @TrustedDeviceEnrollmentError int errorCode)724 AuthInfo(BluetoothDevice device, @Nullable String authString, 725 @TrustedDeviceEnrollmentError int errorCode) { 726 mDevice = device; 727 mAuthString = authString; 728 mErrorCode = errorCode; 729 } 730 } 731 } 732