1 /* 2 * Copyright (C) 2012 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 com.android.bluetooth.avrcp; 18 19 import java.util.Timer; 20 import java.util.TimerTask; 21 22 import android.app.PendingIntent; 23 import android.bluetooth.BluetoothA2dp; 24 import android.bluetooth.BluetoothAvrcp; 25 import android.content.Context; 26 import android.content.Intent; 27 import android.graphics.Bitmap; 28 import android.media.AudioManager; 29 import android.media.IRemoteControlDisplay; 30 import android.media.MediaMetadataRetriever; 31 import android.media.RemoteControlClient; 32 import android.media.RemoteController; 33 import android.media.RemoteController.MetadataEditor; 34 import android.os.Bundle; 35 import android.os.Handler; 36 import android.os.HandlerThread; 37 import android.os.Looper; 38 import android.os.Message; 39 import android.os.ParcelUuid; 40 import android.os.PowerManager; 41 import android.os.PowerManager.WakeLock; 42 import android.os.RemoteException; 43 import android.os.ServiceManager; 44 import android.os.SystemClock; 45 import android.util.Log; 46 import android.view.KeyEvent; 47 48 import com.android.bluetooth.btservice.AdapterService; 49 import com.android.bluetooth.btservice.ProfileService; 50 import com.android.bluetooth.Utils; 51 import com.android.internal.util.IState; 52 import com.android.internal.util.State; 53 import com.android.internal.util.StateMachine; 54 55 import java.lang.ref.WeakReference; 56 import java.util.ArrayList; 57 import java.util.List; 58 import java.util.Set; 59 /** 60 * support Bluetooth AVRCP profile. 61 * support metadata, play status and event notification 62 */ 63 public final class Avrcp { 64 private static final boolean DEBUG = false; 65 private static final String TAG = "Avrcp"; 66 67 private Context mContext; 68 private final AudioManager mAudioManager; 69 private AvrcpMessageHandler mHandler; 70 private RemoteController mRemoteController; 71 private RemoteControllerWeak mRemoteControllerCb; 72 private Metadata mMetadata; 73 private int mTransportControlFlags; 74 private int mCurrentPlayState; 75 private int mPlayStatusChangedNT; 76 private int mTrackChangedNT; 77 private long mTrackNumber; 78 private long mCurrentPosMs; 79 private long mPlayStartTimeMs; 80 private long mSongLengthMs; 81 private long mPlaybackIntervalMs; 82 private int mPlayPosChangedNT; 83 private long mNextPosMs; 84 private long mPrevPosMs; 85 private long mSkipStartTime; 86 private int mFeatures; 87 private int mAbsoluteVolume; 88 private int mLastSetVolume; 89 private int mLastDirection; 90 private final int mVolumeStep; 91 private final int mAudioStreamMax; 92 private boolean mVolCmdInProgress; 93 private int mAbsVolRetryTimes; 94 private int mSkipAmount; 95 96 /* BTRC features */ 97 public static final int BTRC_FEAT_METADATA = 0x01; 98 public static final int BTRC_FEAT_ABSOLUTE_VOLUME = 0x02; 99 public static final int BTRC_FEAT_BROWSE = 0x04; 100 101 /* AVRC response codes, from avrc_defs */ 102 private static final int AVRC_RSP_NOT_IMPL = 8; 103 private static final int AVRC_RSP_ACCEPT = 9; 104 private static final int AVRC_RSP_REJ = 10; 105 private static final int AVRC_RSP_IN_TRANS = 11; 106 private static final int AVRC_RSP_IMPL_STBL = 12; 107 private static final int AVRC_RSP_CHANGED = 13; 108 private static final int AVRC_RSP_INTERIM = 15; 109 110 private static final int MESSAGE_GET_RC_FEATURES = 1; 111 private static final int MESSAGE_GET_PLAY_STATUS = 2; 112 private static final int MESSAGE_GET_ELEM_ATTRS = 3; 113 private static final int MESSAGE_REGISTER_NOTIFICATION = 4; 114 private static final int MESSAGE_PLAY_INTERVAL_TIMEOUT = 5; 115 private static final int MESSAGE_VOLUME_CHANGED = 6; 116 private static final int MESSAGE_ADJUST_VOLUME = 7; 117 private static final int MESSAGE_SET_ABSOLUTE_VOLUME = 8; 118 private static final int MESSAGE_ABS_VOL_TIMEOUT = 9; 119 private static final int MESSAGE_FAST_FORWARD = 10; 120 private static final int MESSAGE_REWIND = 11; 121 private static final int MESSAGE_CHANGE_PLAY_POS = 12; 122 private static final int MESSAGE_SET_A2DP_AUDIO_STATE = 13; 123 private static final int MSG_UPDATE_STATE = 100; 124 private static final int MSG_SET_METADATA = 101; 125 private static final int MSG_SET_TRANSPORT_CONTROLS = 102; 126 private static final int MSG_SET_GENERATION_ID = 104; 127 128 private static final int BUTTON_TIMEOUT_TIME = 2000; 129 private static final int BASE_SKIP_AMOUNT = 2000; 130 private static final int KEY_STATE_PRESS = 1; 131 private static final int KEY_STATE_RELEASE = 0; 132 private static final int SKIP_PERIOD = 400; 133 private static final int SKIP_DOUBLE_INTERVAL = 3000; 134 private static final long MAX_MULTIPLIER_VALUE = 128L; 135 private static final int CMD_TIMEOUT_DELAY = 2000; 136 private static final int MAX_ERROR_RETRY_TIMES = 3; 137 private static final int AVRCP_MAX_VOL = 127; 138 private static final int AVRCP_BASE_VOLUME_STEP = 1; 139 140 static { classInitNative()141 classInitNative(); 142 } 143 Avrcp(Context context)144 private Avrcp(Context context) { 145 mMetadata = new Metadata(); 146 mCurrentPlayState = RemoteControlClient.PLAYSTATE_NONE; // until we get a callback 147 mPlayStatusChangedNT = NOTIFICATION_TYPE_CHANGED; 148 mTrackChangedNT = NOTIFICATION_TYPE_CHANGED; 149 mTrackNumber = -1L; 150 mCurrentPosMs = 0L; 151 mPlayStartTimeMs = -1L; 152 mSongLengthMs = 0L; 153 mPlaybackIntervalMs = 0L; 154 mPlayPosChangedNT = NOTIFICATION_TYPE_CHANGED; 155 mFeatures = 0; 156 mAbsoluteVolume = -1; 157 mLastSetVolume = -1; 158 mLastDirection = 0; 159 mVolCmdInProgress = false; 160 mAbsVolRetryTimes = 0; 161 162 mContext = context; 163 164 initNative(); 165 166 mAudioManager = (AudioManager) context.getSystemService(Context.AUDIO_SERVICE); 167 mAudioStreamMax = mAudioManager.getStreamMaxVolume(AudioManager.STREAM_MUSIC); 168 mVolumeStep = Math.max(AVRCP_BASE_VOLUME_STEP, AVRCP_MAX_VOL/mAudioStreamMax); 169 } 170 start()171 private void start() { 172 HandlerThread thread = new HandlerThread("BluetoothAvrcpHandler"); 173 thread.start(); 174 Looper looper = thread.getLooper(); 175 mHandler = new AvrcpMessageHandler(looper); 176 mRemoteControllerCb = new RemoteControllerWeak(mHandler); 177 mRemoteController = new RemoteController(mContext, mRemoteControllerCb); 178 mAudioManager.registerRemoteController(mRemoteController); 179 mRemoteController.setSynchronizationMode(RemoteController.POSITION_SYNCHRONIZATION_CHECK); 180 } 181 make(Context context)182 public static Avrcp make(Context context) { 183 if (DEBUG) Log.v(TAG, "make"); 184 Avrcp ar = new Avrcp(context); 185 ar.start(); 186 return ar; 187 } 188 doQuit()189 public void doQuit() { 190 mHandler.removeCallbacksAndMessages(null); 191 Looper looper = mHandler.getLooper(); 192 if (looper != null) { 193 looper.quit(); 194 } 195 mAudioManager.unregisterRemoteController(mRemoteController); 196 } 197 cleanup()198 public void cleanup() { 199 cleanupNative(); 200 } 201 202 private static class RemoteControllerWeak implements RemoteController.OnClientUpdateListener { 203 private final WeakReference<Handler> mLocalHandler; 204 RemoteControllerWeak(Handler handler)205 public RemoteControllerWeak(Handler handler) { 206 mLocalHandler = new WeakReference<Handler>(handler); 207 } 208 209 @Override onClientChange(boolean clearing)210 public void onClientChange(boolean clearing) { 211 Handler handler = mLocalHandler.get(); 212 if (handler != null) { 213 handler.obtainMessage(MSG_SET_GENERATION_ID, 214 0, (clearing ? 1 : 0), null).sendToTarget(); 215 } 216 } 217 218 @Override onClientPlaybackStateUpdate(int state)219 public void onClientPlaybackStateUpdate(int state) { 220 // Should never be called with the existing code, but just in case 221 Handler handler = mLocalHandler.get(); 222 if (handler != null) { 223 handler.obtainMessage(MSG_UPDATE_STATE, 0, state, 224 new Long(RemoteControlClient.PLAYBACK_POSITION_INVALID)).sendToTarget(); 225 } 226 } 227 228 @Override onClientPlaybackStateUpdate(int state, long stateChangeTimeMs, long currentPosMs, float speed)229 public void onClientPlaybackStateUpdate(int state, long stateChangeTimeMs, 230 long currentPosMs, float speed) { 231 Handler handler = mLocalHandler.get(); 232 if (handler != null) { 233 handler.obtainMessage(MSG_UPDATE_STATE, 0, state, 234 new Long(currentPosMs)).sendToTarget(); 235 } 236 } 237 238 @Override onClientTransportControlUpdate(int transportControlFlags)239 public void onClientTransportControlUpdate(int transportControlFlags) { 240 Handler handler = mLocalHandler.get(); 241 if (handler != null) { 242 handler.obtainMessage(MSG_SET_TRANSPORT_CONTROLS, 0, transportControlFlags) 243 .sendToTarget(); 244 } 245 } 246 247 @Override onClientMetadataUpdate(MetadataEditor metadataEditor)248 public void onClientMetadataUpdate(MetadataEditor metadataEditor) { 249 Handler handler = mLocalHandler.get(); 250 if (handler != null) { 251 handler.obtainMessage(MSG_SET_METADATA, 0, 0, metadataEditor).sendToTarget(); 252 } 253 } 254 } 255 256 /** Handles Avrcp messages. */ 257 private final class AvrcpMessageHandler extends Handler { AvrcpMessageHandler(Looper looper)258 private AvrcpMessageHandler(Looper looper) { 259 super(looper); 260 } 261 262 @Override handleMessage(Message msg)263 public void handleMessage(Message msg) { 264 switch (msg.what) { 265 case MSG_UPDATE_STATE: 266 updatePlayPauseState(msg.arg2, ((Long) msg.obj).longValue()); 267 break; 268 269 case MSG_SET_METADATA: 270 updateMetadata((MetadataEditor) msg.obj); 271 break; 272 273 case MSG_SET_TRANSPORT_CONTROLS: 274 updateTransportControls(msg.arg2); 275 break; 276 277 case MSG_SET_GENERATION_ID: 278 if (DEBUG) Log.v(TAG, "New genId = " + msg.arg1 + ", clearing = " + msg.arg2); 279 break; 280 281 case MESSAGE_GET_RC_FEATURES: 282 String address = (String) msg.obj; 283 if (DEBUG) Log.v(TAG, "MESSAGE_GET_RC_FEATURES: address="+address+ 284 ", features="+msg.arg1); 285 mFeatures = msg.arg1; 286 mAudioManager.avrcpSupportsAbsoluteVolume(address, isAbsoluteVolumeSupported()); 287 break; 288 289 case MESSAGE_GET_PLAY_STATUS: 290 if (DEBUG) Log.v(TAG, "MESSAGE_GET_PLAY_STATUS"); 291 getPlayStatusRspNative(convertPlayStateToPlayStatus(mCurrentPlayState), 292 (int)mSongLengthMs, (int)getPlayPosition()); 293 break; 294 295 case MESSAGE_GET_ELEM_ATTRS: 296 { 297 String[] textArray; 298 int[] attrIds; 299 byte numAttr = (byte) msg.arg1; 300 ArrayList<Integer> attrList = (ArrayList<Integer>) msg.obj; 301 if (DEBUG) Log.v(TAG, "MESSAGE_GET_ELEM_ATTRS:numAttr=" + numAttr); 302 attrIds = new int[numAttr]; 303 textArray = new String[numAttr]; 304 for (int i = 0; i < numAttr; ++i) { 305 attrIds[i] = attrList.get(i).intValue(); 306 textArray[i] = getAttributeString(attrIds[i]); 307 } 308 getElementAttrRspNative(numAttr, attrIds, textArray); 309 break; 310 } 311 case MESSAGE_REGISTER_NOTIFICATION: 312 if (DEBUG) Log.v(TAG, "MESSAGE_REGISTER_NOTIFICATION:event=" + msg.arg1 + 313 " param=" + msg.arg2); 314 processRegisterNotification(msg.arg1, msg.arg2); 315 break; 316 317 case MESSAGE_PLAY_INTERVAL_TIMEOUT: 318 if (DEBUG) Log.v(TAG, "MESSAGE_PLAY_INTERVAL_TIMEOUT"); 319 mPlayPosChangedNT = NOTIFICATION_TYPE_CHANGED; 320 registerNotificationRspPlayPosNative(mPlayPosChangedNT, (int)getPlayPosition()); 321 break; 322 323 case MESSAGE_VOLUME_CHANGED: 324 if (DEBUG) Log.v(TAG, "MESSAGE_VOLUME_CHANGED: volume=" + ((byte)msg.arg1 & 0x7f) 325 + " ctype=" + msg.arg2); 326 327 if (msg.arg2 == AVRC_RSP_ACCEPT || msg.arg2 == AVRC_RSP_REJ) { 328 if (mVolCmdInProgress == false) { 329 Log.e(TAG, "Unsolicited response, ignored"); 330 break; 331 } 332 removeMessages(MESSAGE_ABS_VOL_TIMEOUT); 333 mVolCmdInProgress = false; 334 mAbsVolRetryTimes = 0; 335 } 336 if (mAbsoluteVolume != msg.arg1 && (msg.arg2 == AVRC_RSP_ACCEPT || 337 msg.arg2 == AVRC_RSP_CHANGED || 338 msg.arg2 == AVRC_RSP_INTERIM)) { 339 byte absVol = (byte)((byte)msg.arg1 & 0x7f); // discard MSB as it is RFD 340 notifyVolumeChanged(absVol); 341 mAbsoluteVolume = absVol; 342 long pecentVolChanged = ((long)absVol * 100) / 0x7f; 343 Log.e(TAG, "percent volume changed: " + pecentVolChanged + "%"); 344 } else if (msg.arg2 == AVRC_RSP_REJ) { 345 Log.e(TAG, "setAbsoluteVolume call rejected"); 346 } 347 break; 348 349 case MESSAGE_ADJUST_VOLUME: 350 if (DEBUG) Log.d(TAG, "MESSAGE_ADJUST_VOLUME: direction=" + msg.arg1); 351 if (mVolCmdInProgress) { 352 if (DEBUG) Log.w(TAG, "There is already a volume command in progress."); 353 break; 354 } 355 // Wait on verification on volume from device, before changing the volume. 356 if (mAbsoluteVolume != -1 && (msg.arg1 == -1 || msg.arg1 == 1)) { 357 int setVol = Math.min(AVRCP_MAX_VOL, 358 Math.max(0, mAbsoluteVolume + msg.arg1*mVolumeStep)); 359 if (setVolumeNative(setVol)) { 360 sendMessageDelayed(obtainMessage(MESSAGE_ABS_VOL_TIMEOUT), 361 CMD_TIMEOUT_DELAY); 362 mVolCmdInProgress = true; 363 mLastDirection = msg.arg1; 364 mLastSetVolume = setVol; 365 } 366 } else { 367 Log.e(TAG, "Unknown direction in MESSAGE_ADJUST_VOLUME"); 368 } 369 break; 370 371 case MESSAGE_SET_ABSOLUTE_VOLUME: 372 if (DEBUG) Log.v(TAG, "MESSAGE_SET_ABSOLUTE_VOLUME"); 373 if (mVolCmdInProgress) { 374 if (DEBUG) Log.w(TAG, "There is already a volume command in progress."); 375 break; 376 } 377 if (setVolumeNative(msg.arg1)) { 378 sendMessageDelayed(obtainMessage(MESSAGE_ABS_VOL_TIMEOUT), CMD_TIMEOUT_DELAY); 379 mVolCmdInProgress = true; 380 mLastSetVolume = msg.arg1; 381 } 382 break; 383 384 case MESSAGE_ABS_VOL_TIMEOUT: 385 if (DEBUG) Log.v(TAG, "MESSAGE_ABS_VOL_TIMEOUT: Volume change cmd timed out."); 386 mVolCmdInProgress = false; 387 if (mAbsVolRetryTimes >= MAX_ERROR_RETRY_TIMES) { 388 mAbsVolRetryTimes = 0; 389 } else { 390 mAbsVolRetryTimes += 1; 391 if (setVolumeNative(mLastSetVolume)) { 392 sendMessageDelayed(obtainMessage(MESSAGE_ABS_VOL_TIMEOUT), 393 CMD_TIMEOUT_DELAY); 394 mVolCmdInProgress = true; 395 } 396 } 397 break; 398 399 case MESSAGE_FAST_FORWARD: 400 case MESSAGE_REWIND: 401 if(msg.what == MESSAGE_FAST_FORWARD) { 402 if((mTransportControlFlags & 403 RemoteControlClient.FLAG_KEY_MEDIA_FAST_FORWARD) != 0) { 404 int keyState = msg.arg1 == KEY_STATE_PRESS ? 405 KeyEvent.ACTION_DOWN : KeyEvent.ACTION_UP; 406 KeyEvent keyEvent = 407 new KeyEvent(keyState, KeyEvent.KEYCODE_MEDIA_FAST_FORWARD); 408 mRemoteController.sendMediaKeyEvent(keyEvent); 409 break; 410 } 411 } else if((mTransportControlFlags & 412 RemoteControlClient.FLAG_KEY_MEDIA_REWIND) != 0) { 413 int keyState = msg.arg1 == KEY_STATE_PRESS ? 414 KeyEvent.ACTION_DOWN : KeyEvent.ACTION_UP; 415 KeyEvent keyEvent = 416 new KeyEvent(keyState, KeyEvent.KEYCODE_MEDIA_REWIND); 417 mRemoteController.sendMediaKeyEvent(keyEvent); 418 break; 419 } 420 421 int skipAmount; 422 if (msg.what == MESSAGE_FAST_FORWARD) { 423 if (DEBUG) Log.v(TAG, "MESSAGE_FAST_FORWARD"); 424 removeMessages(MESSAGE_FAST_FORWARD); 425 skipAmount = BASE_SKIP_AMOUNT; 426 } else { 427 if (DEBUG) Log.v(TAG, "MESSAGE_REWIND"); 428 removeMessages(MESSAGE_REWIND); 429 skipAmount = -BASE_SKIP_AMOUNT; 430 } 431 432 if (hasMessages(MESSAGE_CHANGE_PLAY_POS) && 433 (skipAmount != mSkipAmount)) { 434 Log.w(TAG, "missing release button event:" + mSkipAmount); 435 } 436 437 if ((!hasMessages(MESSAGE_CHANGE_PLAY_POS)) || 438 (skipAmount != mSkipAmount)) { 439 mSkipStartTime = SystemClock.elapsedRealtime(); 440 } 441 442 removeMessages(MESSAGE_CHANGE_PLAY_POS); 443 if (msg.arg1 == KEY_STATE_PRESS) { 444 mSkipAmount = skipAmount; 445 changePositionBy(mSkipAmount * getSkipMultiplier()); 446 Message posMsg = obtainMessage(MESSAGE_CHANGE_PLAY_POS); 447 posMsg.arg1 = 1; 448 sendMessageDelayed(posMsg, SKIP_PERIOD); 449 } 450 451 break; 452 453 case MESSAGE_CHANGE_PLAY_POS: 454 if (DEBUG) Log.v(TAG, "MESSAGE_CHANGE_PLAY_POS:" + msg.arg1); 455 changePositionBy(mSkipAmount * getSkipMultiplier()); 456 if (msg.arg1 * SKIP_PERIOD < BUTTON_TIMEOUT_TIME) { 457 Message posMsg = obtainMessage(MESSAGE_CHANGE_PLAY_POS); 458 posMsg.arg1 = msg.arg1 + 1; 459 sendMessageDelayed(posMsg, SKIP_PERIOD); 460 } 461 break; 462 463 case MESSAGE_SET_A2DP_AUDIO_STATE: 464 if (DEBUG) Log.v(TAG, "MESSAGE_SET_A2DP_AUDIO_STATE:" + msg.arg1); 465 updateA2dpAudioState(msg.arg1); 466 break; 467 } 468 } 469 } 470 updateA2dpAudioState(int state)471 private void updateA2dpAudioState(int state) { 472 boolean isPlaying = (state == BluetoothA2dp.STATE_PLAYING); 473 if (isPlaying != isPlayingState(mCurrentPlayState)) { 474 /* if a2dp is streaming, check to make sure music is active */ 475 if ( (isPlaying) && !mAudioManager.isMusicActive()) 476 return; 477 updatePlayPauseState(isPlaying ? RemoteControlClient.PLAYSTATE_PLAYING : 478 RemoteControlClient.PLAYSTATE_PAUSED, 479 RemoteControlClient.PLAYBACK_POSITION_INVALID); 480 } 481 } 482 updatePlayPauseState(int state, long currentPosMs)483 private void updatePlayPauseState(int state, long currentPosMs) { 484 if (DEBUG) Log.v(TAG, 485 "updatePlayPauseState, old=" + mCurrentPlayState + ", state=" + state); 486 boolean oldPosValid = (mCurrentPosMs != 487 RemoteControlClient.PLAYBACK_POSITION_ALWAYS_UNKNOWN); 488 int oldPlayStatus = convertPlayStateToPlayStatus(mCurrentPlayState); 489 int newPlayStatus = convertPlayStateToPlayStatus(state); 490 491 if ((mCurrentPlayState == RemoteControlClient.PLAYSTATE_PLAYING) && 492 (mCurrentPlayState != state) && oldPosValid) { 493 mCurrentPosMs = getPlayPosition(); 494 } 495 496 if (currentPosMs != RemoteControlClient.PLAYBACK_POSITION_INVALID) { 497 mCurrentPosMs = currentPosMs; 498 } 499 if ((state == RemoteControlClient.PLAYSTATE_PLAYING) && 500 ((currentPosMs != RemoteControlClient.PLAYBACK_POSITION_INVALID) || 501 (mCurrentPlayState != RemoteControlClient.PLAYSTATE_PLAYING))) { 502 mPlayStartTimeMs = SystemClock.elapsedRealtime(); 503 } 504 mCurrentPlayState = state; 505 506 boolean newPosValid = (mCurrentPosMs != 507 RemoteControlClient.PLAYBACK_POSITION_ALWAYS_UNKNOWN); 508 long playPosition = getPlayPosition(); 509 mHandler.removeMessages(MESSAGE_PLAY_INTERVAL_TIMEOUT); 510 /* need send play position changed notification when play status is changed */ 511 if ((mPlayPosChangedNT == NOTIFICATION_TYPE_INTERIM) && 512 ((oldPlayStatus != newPlayStatus) || (oldPosValid != newPosValid) || 513 (newPosValid && ((playPosition >= mNextPosMs) || (playPosition <= mPrevPosMs))))) { 514 mPlayPosChangedNT = NOTIFICATION_TYPE_CHANGED; 515 registerNotificationRspPlayPosNative(mPlayPosChangedNT, (int)playPosition); 516 } 517 if ((mPlayPosChangedNT == NOTIFICATION_TYPE_INTERIM) && newPosValid && 518 (state == RemoteControlClient.PLAYSTATE_PLAYING)) { 519 Message msg = mHandler.obtainMessage(MESSAGE_PLAY_INTERVAL_TIMEOUT); 520 mHandler.sendMessageDelayed(msg, mNextPosMs - playPosition); 521 } 522 523 if ((mPlayStatusChangedNT == NOTIFICATION_TYPE_INTERIM) && (oldPlayStatus != newPlayStatus)) { 524 mPlayStatusChangedNT = NOTIFICATION_TYPE_CHANGED; 525 registerNotificationRspPlayStatusNative(mPlayStatusChangedNT, newPlayStatus); 526 } 527 } 528 updateTransportControls(int transportControlFlags)529 private void updateTransportControls(int transportControlFlags) { 530 mTransportControlFlags = transportControlFlags; 531 } 532 533 class Metadata { 534 private String artist; 535 private String trackTitle; 536 private String albumTitle; 537 Metadata()538 public Metadata() { 539 artist = null; 540 trackTitle = null; 541 albumTitle = null; 542 } 543 toString()544 public String toString() { 545 return "Metadata[artist=" + artist + " trackTitle=" + trackTitle + " albumTitle=" + 546 albumTitle + "]"; 547 } 548 } 549 updateMetadata(MetadataEditor data)550 private void updateMetadata(MetadataEditor data) { 551 String oldMetadata = mMetadata.toString(); 552 mMetadata.artist = data.getString(MediaMetadataRetriever.METADATA_KEY_ARTIST, null); 553 mMetadata.trackTitle = data.getString(MediaMetadataRetriever.METADATA_KEY_TITLE, null); 554 mMetadata.albumTitle = data.getString(MediaMetadataRetriever.METADATA_KEY_ALBUM, null); 555 if (!oldMetadata.equals(mMetadata.toString())) { 556 mTrackNumber++; 557 if (mTrackChangedNT == NOTIFICATION_TYPE_INTERIM) { 558 mTrackChangedNT = NOTIFICATION_TYPE_CHANGED; 559 sendTrackChangedRsp(); 560 } 561 562 if (mCurrentPosMs != RemoteControlClient.PLAYBACK_POSITION_ALWAYS_UNKNOWN) { 563 mCurrentPosMs = 0L; 564 if (mCurrentPlayState == RemoteControlClient.PLAYSTATE_PLAYING) { 565 mPlayStartTimeMs = SystemClock.elapsedRealtime(); 566 } 567 } 568 /* need send play position changed notification when track is changed */ 569 if (mPlayPosChangedNT == NOTIFICATION_TYPE_INTERIM) { 570 mPlayPosChangedNT = NOTIFICATION_TYPE_CHANGED; 571 registerNotificationRspPlayPosNative(mPlayPosChangedNT, 572 (int)getPlayPosition()); 573 mHandler.removeMessages(MESSAGE_PLAY_INTERVAL_TIMEOUT); 574 } 575 } 576 if (DEBUG) Log.v(TAG, "mMetadata=" + mMetadata.toString()); 577 578 mSongLengthMs = data.getLong(MediaMetadataRetriever.METADATA_KEY_DURATION, 579 RemoteControlClient.PLAYBACK_POSITION_INVALID); 580 if (DEBUG) Log.v(TAG, "duration=" + mSongLengthMs); 581 } 582 getRcFeatures(byte[] address, int features)583 private void getRcFeatures(byte[] address, int features) { 584 Message msg = mHandler.obtainMessage(MESSAGE_GET_RC_FEATURES, features, 0, 585 Utils.getAddressStringFromByte(address)); 586 mHandler.sendMessage(msg); 587 } 588 getPlayStatus()589 private void getPlayStatus() { 590 Message msg = mHandler.obtainMessage(MESSAGE_GET_PLAY_STATUS); 591 mHandler.sendMessage(msg); 592 } 593 getElementAttr(byte numAttr, int[] attrs)594 private void getElementAttr(byte numAttr, int[] attrs) { 595 int i; 596 ArrayList<Integer> attrList = new ArrayList<Integer>(); 597 for (i = 0; i < numAttr; ++i) { 598 attrList.add(attrs[i]); 599 } 600 Message msg = mHandler.obtainMessage(MESSAGE_GET_ELEM_ATTRS, numAttr, 0, attrList); 601 mHandler.sendMessage(msg); 602 } 603 registerNotification(int eventId, int param)604 private void registerNotification(int eventId, int param) { 605 Message msg = mHandler.obtainMessage(MESSAGE_REGISTER_NOTIFICATION, eventId, param); 606 mHandler.sendMessage(msg); 607 } 608 processRegisterNotification(int eventId, int param)609 private void processRegisterNotification(int eventId, int param) { 610 switch (eventId) { 611 case EVT_PLAY_STATUS_CHANGED: 612 mPlayStatusChangedNT = NOTIFICATION_TYPE_INTERIM; 613 registerNotificationRspPlayStatusNative(mPlayStatusChangedNT, 614 convertPlayStateToPlayStatus(mCurrentPlayState)); 615 break; 616 617 case EVT_TRACK_CHANGED: 618 mTrackChangedNT = NOTIFICATION_TYPE_INTERIM; 619 sendTrackChangedRsp(); 620 break; 621 622 case EVT_PLAY_POS_CHANGED: 623 long songPosition = getPlayPosition(); 624 mPlayPosChangedNT = NOTIFICATION_TYPE_INTERIM; 625 mPlaybackIntervalMs = (long)param * 1000L; 626 if (mCurrentPosMs != RemoteControlClient.PLAYBACK_POSITION_ALWAYS_UNKNOWN) { 627 mNextPosMs = songPosition + mPlaybackIntervalMs; 628 mPrevPosMs = songPosition - mPlaybackIntervalMs; 629 if (mCurrentPlayState == RemoteControlClient.PLAYSTATE_PLAYING) { 630 Message msg = mHandler.obtainMessage(MESSAGE_PLAY_INTERVAL_TIMEOUT); 631 mHandler.sendMessageDelayed(msg, mPlaybackIntervalMs); 632 } 633 } 634 registerNotificationRspPlayPosNative(mPlayPosChangedNT, (int)songPosition); 635 break; 636 637 } 638 } 639 handlePassthroughCmd(int id, int keyState)640 private void handlePassthroughCmd(int id, int keyState) { 641 switch (id) { 642 case BluetoothAvrcp.PASSTHROUGH_ID_REWIND: 643 rewind(keyState); 644 break; 645 case BluetoothAvrcp.PASSTHROUGH_ID_FAST_FOR: 646 fastForward(keyState); 647 break; 648 } 649 } 650 fastForward(int keyState)651 private void fastForward(int keyState) { 652 Message msg = mHandler.obtainMessage(MESSAGE_FAST_FORWARD, keyState, 0); 653 mHandler.sendMessage(msg); 654 } 655 rewind(int keyState)656 private void rewind(int keyState) { 657 Message msg = mHandler.obtainMessage(MESSAGE_REWIND, keyState, 0); 658 mHandler.sendMessage(msg); 659 } 660 changePositionBy(long amount)661 private void changePositionBy(long amount) { 662 long currentPosMs = getPlayPosition(); 663 if (currentPosMs == -1L) return; 664 long newPosMs = Math.max(0L, currentPosMs + amount); 665 mRemoteController.seekTo(newPosMs); 666 } 667 getSkipMultiplier()668 private int getSkipMultiplier() { 669 long currentTime = SystemClock.elapsedRealtime(); 670 long multi = (long) Math.pow(2, (currentTime - mSkipStartTime)/SKIP_DOUBLE_INTERVAL); 671 return (int) Math.min(MAX_MULTIPLIER_VALUE, multi); 672 } 673 sendTrackChangedRsp()674 private void sendTrackChangedRsp() { 675 byte[] track = new byte[TRACK_ID_SIZE]; 676 /* track is stored in big endian format */ 677 for (int i = 0; i < TRACK_ID_SIZE; ++i) { 678 track[i] = (byte) (mTrackNumber >> (56 - 8 * i)); 679 } 680 registerNotificationRspTrackChangeNative(mTrackChangedNT, track); 681 } 682 getPlayPosition()683 private long getPlayPosition() { 684 long songPosition = -1L; 685 if (mCurrentPosMs != RemoteControlClient.PLAYBACK_POSITION_ALWAYS_UNKNOWN) { 686 if (mCurrentPlayState == RemoteControlClient.PLAYSTATE_PLAYING) { 687 songPosition = SystemClock.elapsedRealtime() - 688 mPlayStartTimeMs + mCurrentPosMs; 689 } else { 690 songPosition = mCurrentPosMs; 691 } 692 } 693 if (DEBUG) Log.v(TAG, "position=" + songPosition); 694 return songPosition; 695 } 696 getAttributeString(int attrId)697 private String getAttributeString(int attrId) { 698 String attrStr = null; 699 switch (attrId) { 700 case MEDIA_ATTR_TITLE: 701 attrStr = mMetadata.trackTitle; 702 break; 703 704 case MEDIA_ATTR_ARTIST: 705 attrStr = mMetadata.artist; 706 break; 707 708 case MEDIA_ATTR_ALBUM: 709 attrStr = mMetadata.albumTitle; 710 break; 711 712 case MEDIA_ATTR_PLAYING_TIME: 713 if (mSongLengthMs != 0L) { 714 attrStr = Long.toString(mSongLengthMs); 715 } 716 break; 717 718 } 719 if (attrStr == null) { 720 attrStr = new String(); 721 } 722 if (DEBUG) Log.v(TAG, "getAttributeString:attrId=" + attrId + " str=" + attrStr); 723 return attrStr; 724 } 725 convertPlayStateToPlayStatus(int playState)726 private int convertPlayStateToPlayStatus(int playState) { 727 int playStatus = PLAYSTATUS_ERROR; 728 switch (playState) { 729 case RemoteControlClient.PLAYSTATE_PLAYING: 730 case RemoteControlClient.PLAYSTATE_BUFFERING: 731 playStatus = PLAYSTATUS_PLAYING; 732 break; 733 734 case RemoteControlClient.PLAYSTATE_STOPPED: 735 case RemoteControlClient.PLAYSTATE_NONE: 736 playStatus = PLAYSTATUS_STOPPED; 737 break; 738 739 case RemoteControlClient.PLAYSTATE_PAUSED: 740 playStatus = PLAYSTATUS_PAUSED; 741 break; 742 743 case RemoteControlClient.PLAYSTATE_FAST_FORWARDING: 744 case RemoteControlClient.PLAYSTATE_SKIPPING_FORWARDS: 745 playStatus = PLAYSTATUS_FWD_SEEK; 746 break; 747 748 case RemoteControlClient.PLAYSTATE_REWINDING: 749 case RemoteControlClient.PLAYSTATE_SKIPPING_BACKWARDS: 750 playStatus = PLAYSTATUS_REV_SEEK; 751 break; 752 753 case RemoteControlClient.PLAYSTATE_ERROR: 754 playStatus = PLAYSTATUS_ERROR; 755 break; 756 757 } 758 return playStatus; 759 } 760 isPlayingState(int playState)761 private boolean isPlayingState(int playState) { 762 boolean isPlaying = false; 763 switch (playState) { 764 case RemoteControlClient.PLAYSTATE_PLAYING: 765 case RemoteControlClient.PLAYSTATE_BUFFERING: 766 isPlaying = true; 767 break; 768 default: 769 isPlaying = false; 770 break; 771 } 772 return isPlaying; 773 } 774 775 /** 776 * This is called from AudioService. It will return whether this device supports abs volume. 777 * NOT USED AT THE MOMENT. 778 */ isAbsoluteVolumeSupported()779 public boolean isAbsoluteVolumeSupported() { 780 return ((mFeatures & BTRC_FEAT_ABSOLUTE_VOLUME) != 0); 781 } 782 783 /** 784 * We get this call from AudioService. This will send a message to our handler object, 785 * requesting our handler to call setVolumeNative() 786 */ adjustVolume(int direction)787 public void adjustVolume(int direction) { 788 Message msg = mHandler.obtainMessage(MESSAGE_ADJUST_VOLUME, direction, 0); 789 mHandler.sendMessage(msg); 790 } 791 setAbsoluteVolume(int volume)792 public void setAbsoluteVolume(int volume) { 793 int avrcpVolume = convertToAvrcpVolume(volume); 794 avrcpVolume = Math.min(AVRCP_MAX_VOL, Math.max(0, avrcpVolume)); 795 mHandler.removeMessages(MESSAGE_ADJUST_VOLUME); 796 Message msg = mHandler.obtainMessage(MESSAGE_SET_ABSOLUTE_VOLUME, avrcpVolume, 0); 797 mHandler.sendMessage(msg); 798 } 799 800 /* Called in the native layer as a btrc_callback to return the volume set on the carkit in the 801 * case when the volume is change locally on the carkit. This notification is not called when 802 * the volume is changed from the phone. 803 * 804 * This method will send a message to our handler to change the local stored volume and notify 805 * AudioService to update the UI 806 */ volumeChangeCallback(int volume, int ctype)807 private void volumeChangeCallback(int volume, int ctype) { 808 Message msg = mHandler.obtainMessage(MESSAGE_VOLUME_CHANGED, volume, ctype); 809 mHandler.sendMessage(msg); 810 } 811 notifyVolumeChanged(int volume)812 private void notifyVolumeChanged(int volume) { 813 volume = convertToAudioStreamVolume(volume); 814 mAudioManager.setStreamVolume(AudioManager.STREAM_MUSIC, volume, 815 AudioManager.FLAG_SHOW_UI | AudioManager.FLAG_BLUETOOTH_ABS_VOLUME); 816 } 817 convertToAudioStreamVolume(int volume)818 private int convertToAudioStreamVolume(int volume) { 819 // Rescale volume to match AudioSystem's volume 820 return (int) Math.round((double) volume*mAudioStreamMax/AVRCP_MAX_VOL); 821 } 822 convertToAvrcpVolume(int volume)823 private int convertToAvrcpVolume(int volume) { 824 return (int) Math.ceil((double) volume*AVRCP_MAX_VOL/mAudioStreamMax); 825 } 826 827 /** 828 * This is called from A2dpStateMachine to set A2dp audio state. 829 */ setA2dpAudioState(int state)830 public void setA2dpAudioState(int state) { 831 Message msg = mHandler.obtainMessage(MESSAGE_SET_A2DP_AUDIO_STATE, state, 0); 832 mHandler.sendMessage(msg); 833 } 834 835 // Do not modify without updating the HAL bt_rc.h files. 836 837 // match up with btrc_play_status_t enum of bt_rc.h 838 final static int PLAYSTATUS_STOPPED = 0; 839 final static int PLAYSTATUS_PLAYING = 1; 840 final static int PLAYSTATUS_PAUSED = 2; 841 final static int PLAYSTATUS_FWD_SEEK = 3; 842 final static int PLAYSTATUS_REV_SEEK = 4; 843 final static int PLAYSTATUS_ERROR = 255; 844 845 // match up with btrc_media_attr_t enum of bt_rc.h 846 final static int MEDIA_ATTR_TITLE = 1; 847 final static int MEDIA_ATTR_ARTIST = 2; 848 final static int MEDIA_ATTR_ALBUM = 3; 849 final static int MEDIA_ATTR_TRACK_NUM = 4; 850 final static int MEDIA_ATTR_NUM_TRACKS = 5; 851 final static int MEDIA_ATTR_GENRE = 6; 852 final static int MEDIA_ATTR_PLAYING_TIME = 7; 853 854 // match up with btrc_event_id_t enum of bt_rc.h 855 final static int EVT_PLAY_STATUS_CHANGED = 1; 856 final static int EVT_TRACK_CHANGED = 2; 857 final static int EVT_TRACK_REACHED_END = 3; 858 final static int EVT_TRACK_REACHED_START = 4; 859 final static int EVT_PLAY_POS_CHANGED = 5; 860 final static int EVT_BATT_STATUS_CHANGED = 6; 861 final static int EVT_SYSTEM_STATUS_CHANGED = 7; 862 final static int EVT_APP_SETTINGS_CHANGED = 8; 863 864 // match up with btrc_notification_type_t enum of bt_rc.h 865 final static int NOTIFICATION_TYPE_INTERIM = 0; 866 final static int NOTIFICATION_TYPE_CHANGED = 1; 867 868 // match up with BTRC_UID_SIZE of bt_rc.h 869 final static int TRACK_ID_SIZE = 8; 870 classInitNative()871 private native static void classInitNative(); initNative()872 private native void initNative(); cleanupNative()873 private native void cleanupNative(); getPlayStatusRspNative(int playStatus, int songLen, int songPos)874 private native boolean getPlayStatusRspNative(int playStatus, int songLen, int songPos); getElementAttrRspNative(byte numAttr, int[] attrIds, String[] textArray)875 private native boolean getElementAttrRspNative(byte numAttr, int[] attrIds, String[] textArray); registerNotificationRspPlayStatusNative(int type, int playStatus)876 private native boolean registerNotificationRspPlayStatusNative(int type, int playStatus); registerNotificationRspTrackChangeNative(int type, byte[] track)877 private native boolean registerNotificationRspTrackChangeNative(int type, byte[] track); registerNotificationRspPlayPosNative(int type, int playPos)878 private native boolean registerNotificationRspPlayPosNative(int type, int playPos); setVolumeNative(int volume)879 private native boolean setVolumeNative(int volume); sendPassThroughCommandNative(int keyCode, int keyState)880 private native boolean sendPassThroughCommandNative(int keyCode, int keyState); 881 882 } 883