1 /* 2 * Copyright (C) 2011 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 //#define USE_LOG SLAndroidLogLevel_Verbose 18 19 #include "sles_allinclusive.h" 20 21 #include <media/stagefright/foundation/ADebug.h> 22 #include <sys/stat.h> 23 #include <inttypes.h> 24 25 namespace android { 26 27 //-------------------------------------------------------------------------------------------------- 28 GenericPlayer::GenericPlayer(const AudioPlayback_Parameters* params) : 29 mDataLocatorType(kDataLocatorNone), 30 mNotifyClient(NULL), 31 mNotifyUser(NULL), 32 mStateFlags(0), 33 mPlaybackParams(*params), 34 mDurationMsec(ANDROID_UNKNOWN_TIME), 35 mPlaybackRatePermille(1000), 36 mCacheStatus(kStatusEmpty), 37 mCacheFill(0), 38 mLastNotifiedCacheFill(0), 39 mCacheFillNotifThreshold(100), 40 mEventFlags(0), 41 mMarkerPositionMs(ANDROID_UNKNOWN_TIME), 42 mPositionUpdatePeriodMs(1000), // per spec 43 mOneShotGeneration(0), 44 mDeliveredNewPosMs(ANDROID_UNKNOWN_TIME), 45 mObservedPositionMs(ANDROID_UNKNOWN_TIME) 46 { 47 SL_LOGD("GenericPlayer::GenericPlayer()"); 48 49 mLooper = new android::ALooper(); 50 51 // Post-construction accesses need to be protected by mSettingsLock 52 mAndroidAudioLevels.mFinalVolume[0] = 1.0f; 53 mAndroidAudioLevels.mFinalVolume[1] = 1.0f; 54 } 55 56 57 GenericPlayer::~GenericPlayer() { 58 SL_LOGV("GenericPlayer::~GenericPlayer()"); 59 60 resetDataLocator(); 61 } 62 63 64 void GenericPlayer::init(const notif_cbf_t cbf, void* notifUser) { 65 SL_LOGD("GenericPlayer::init()"); 66 67 { 68 android::Mutex::Autolock autoLock(mNotifyClientLock); 69 mNotifyClient = cbf; 70 mNotifyUser = notifUser; 71 } 72 73 mLooper->registerHandler(this); 74 mLooper->start(false /*runOnCallingThread*/, false /*canCallJava*/, PRIORITY_DEFAULT); 75 } 76 77 78 void GenericPlayer::preDestroy() { 79 SL_LOGD("GenericPlayer::preDestroy()"); 80 { 81 android::Mutex::Autolock autoLock(mNotifyClientLock); 82 mNotifyClient = NULL; 83 mNotifyUser = NULL; 84 } 85 86 mLooper->stop(); 87 mLooper->unregisterHandler(id()); 88 } 89 90 91 void GenericPlayer::setDataSource(const char *uri) { 92 SL_LOGV("GenericPlayer::setDataSource(uri=%s)", uri); 93 resetDataLocator(); 94 95 mDataLocator.uriRef = uri; 96 97 mDataLocatorType = kDataLocatorUri; 98 } 99 100 101 void GenericPlayer::setDataSource(int fd, int64_t offset, int64_t length, bool closeAfterUse) { 102 SL_LOGV("GenericPlayer::setDataSource(fd=%d, offset=%lld, length=%lld, closeAfterUse=%s)", fd, 103 offset, length, closeAfterUse ? "true" : "false"); 104 resetDataLocator(); 105 106 mDataLocator.fdi.fd = fd; 107 108 struct stat sb; 109 int ret = fstat(fd, &sb); 110 if (ret != 0) { 111 SL_LOGE("GenericPlayer::setDataSource: fstat(%d) failed: %d, %s", fd, ret, strerror(errno)); 112 return; 113 } 114 115 if (offset >= sb.st_size) { 116 SL_LOGE("SfPlayer::setDataSource: invalid offset"); 117 return; 118 } 119 mDataLocator.fdi.offset = offset; 120 121 if (PLAYER_FD_FIND_FILE_SIZE == length) { 122 mDataLocator.fdi.length = sb.st_size; 123 } else if (offset + length > sb.st_size) { 124 mDataLocator.fdi.length = sb.st_size - offset; 125 } else { 126 mDataLocator.fdi.length = length; 127 } 128 129 mDataLocator.fdi.mCloseAfterUse = closeAfterUse; 130 131 mDataLocatorType = kDataLocatorFd; 132 } 133 134 135 void GenericPlayer::prepare() { 136 SL_LOGD("GenericPlayer::prepare()"); 137 // do not attempt prepare more than once 138 if (!(mStateFlags & (kFlagPrepared | kFlagPreparedUnsuccessfully))) { 139 sp<AMessage> msg = new AMessage(kWhatPrepare, this); 140 msg->post(); 141 } 142 } 143 144 145 void GenericPlayer::play() { 146 SL_LOGD("GenericPlayer::play()"); 147 sp<AMessage> msg = new AMessage(kWhatPlay, this); 148 msg->post(); 149 } 150 151 152 void GenericPlayer::pause() { 153 SL_LOGD("GenericPlayer::pause()"); 154 sp<AMessage> msg = new AMessage(kWhatPause, this); 155 msg->post(); 156 } 157 158 159 void GenericPlayer::stop() { 160 SL_LOGD("GenericPlayer::stop()"); 161 (new AMessage(kWhatPause, this))->post(); 162 163 // after a stop, playback should resume from the start. 164 seek(0); 165 } 166 167 168 void GenericPlayer::seek(int64_t timeMsec) { 169 SL_LOGV("GenericPlayer::seek %lld", timeMsec); 170 if (timeMsec < 0 && timeMsec != ANDROID_UNKNOWN_TIME) { 171 SL_LOGE("GenericPlayer::seek error, can't seek to negative time %" PRId64 "ms", timeMsec); 172 return; 173 } 174 sp<AMessage> msg = new AMessage(kWhatSeek, this); 175 msg->setInt64(WHATPARAM_SEEK_SEEKTIME_MS, timeMsec); 176 msg->post(); 177 } 178 179 180 void GenericPlayer::loop(bool loop) { 181 SL_LOGV("GenericPlayer::loop %s", loop ? "true" : "false"); 182 sp<AMessage> msg = new AMessage(kWhatLoop, this); 183 msg->setInt32(WHATPARAM_LOOP_LOOPING, (int32_t)loop); 184 msg->post(); 185 } 186 187 188 void GenericPlayer::setBufferingUpdateThreshold(int16_t thresholdPercent) { 189 SL_LOGV("GenericPlayer::setBufferingUpdateThreshold %d", thresholdPercent); 190 sp<AMessage> msg = new AMessage(kWhatBuffUpdateThres, this); 191 msg->setInt32(WHATPARAM_BUFFERING_UPDATETHRESHOLD_PERCENT, (int32_t)thresholdPercent); 192 msg->post(); 193 } 194 195 196 //-------------------------------------------------- 197 void GenericPlayer::getDurationMsec(int* msec) { 198 Mutex::Autolock _l(mSettingsLock); 199 *msec = mDurationMsec; 200 } 201 202 //-------------------------------------------------- 203 void GenericPlayer::setVolume(float leftVol, float rightVol) 204 { 205 { 206 Mutex::Autolock _l(mSettingsLock); 207 mAndroidAudioLevels.mFinalVolume[0] = leftVol; 208 mAndroidAudioLevels.mFinalVolume[1] = rightVol; 209 } 210 // send a message for the volume to be updated by the object which implements the volume 211 (new AMessage(kWhatVolumeUpdate, this))->post(); 212 } 213 214 215 //-------------------------------------------------- 216 void GenericPlayer::attachAuxEffect(int32_t effectId) 217 { 218 SL_LOGV("GenericPlayer::attachAuxEffect(id=%d)", effectId); 219 sp<AMessage> msg = new AMessage(kWhatAttachAuxEffect, this); 220 msg->setInt32(WHATPARAM_ATTACHAUXEFFECT, effectId); 221 msg->post(); 222 } 223 224 225 //-------------------------------------------------- 226 void GenericPlayer::setAuxEffectSendLevel(float level) 227 { 228 SL_LOGV("GenericPlayer::setAuxEffectSendLevel(level=%g)", level); 229 sp<AMessage> msg = new AMessage(kWhatSetAuxEffectSendLevel, this); 230 msg->setFloat(WHATPARAM_SETAUXEFFECTSENDLEVEL, level); 231 msg->post(); 232 } 233 234 235 //-------------------------------------------------- 236 void GenericPlayer::setPlaybackRate(int32_t ratePermille) { 237 SL_LOGV("GenericPlayer::setPlaybackRate(ratePermille=%d)", ratePermille); 238 { 239 Mutex::Autolock _l(mSettingsLock); 240 mPlaybackRatePermille = (int16_t)ratePermille; 241 } 242 } 243 244 //-------------------------------------------------- 245 // Call after changing any of the IPlay settings related to SL_PLAYEVENT_* 246 void GenericPlayer::setPlayEvents(int32_t eventFlags, int32_t markerPositionMs, 247 int32_t positionUpdatePeriodMs) 248 { 249 // Normalize ms that are within the valid unsigned range, but not in the int32_t range 250 if (markerPositionMs < 0) { 251 markerPositionMs = ANDROID_UNKNOWN_TIME; 252 } 253 if (positionUpdatePeriodMs < 0) { 254 positionUpdatePeriodMs = ANDROID_UNKNOWN_TIME; 255 } 256 // markers are delivered accurately, but new position updates are limited to every 100 ms 257 if (positionUpdatePeriodMs < 100) { 258 positionUpdatePeriodMs = 100; 259 } 260 sp<AMessage> msg = new AMessage(kWhatSetPlayEvents, this); 261 msg->setInt32(WHATPARAM_SETPLAYEVENTS_FLAGS, eventFlags); 262 msg->setInt32(WHATPARAM_SETPLAYEVENTS_MARKER, markerPositionMs); 263 msg->setInt32(WHATPARAM_SETPLAYEVENTS_UPDATE, positionUpdatePeriodMs); 264 msg->post(); 265 } 266 267 268 //-------------------------------------------------- 269 /* 270 * post-condition: mDataLocatorType == kDataLocatorNone 271 * 272 */ 273 void GenericPlayer::resetDataLocator() { 274 SL_LOGV("GenericPlayer::resetDataLocator()"); 275 if (mDataLocatorType == kDataLocatorFd && mDataLocator.fdi.mCloseAfterUse) { 276 (void) ::close(mDataLocator.fdi.fd); 277 // would be redundant, as we're about to invalidate the union mDataLocator 278 //mDataLocator.fdi.fd = -1; 279 //mDataLocator.fdi.mCloseAfterUse = false; 280 } 281 mDataLocatorType = kDataLocatorNone; 282 } 283 284 285 void GenericPlayer::notify(const char* event, int data, bool async) { 286 SL_LOGV("GenericPlayer::notify(event=%s, data=%d, async=%s)", event, data, 287 async ? "true" : "false"); 288 sp<AMessage> msg = new AMessage(kWhatNotif, this); 289 msg->setInt32(event, (int32_t)data); 290 if (async) { 291 msg->post(); 292 } else { 293 onNotify(msg); 294 } 295 } 296 297 298 void GenericPlayer::notify(const char* event, int data1, int data2, bool async) { 299 SL_LOGV("GenericPlayer::notify(event=%s, data1=%d, data2=%d, async=%s)", event, data1, data2, 300 async ? "true" : "false"); 301 sp<AMessage> msg = new AMessage(kWhatNotif, this); 302 msg->setRect(event, 0, 0, (int32_t)data1, (int32_t)data2); 303 if (async) { 304 msg->post(); 305 } else { 306 onNotify(msg); 307 } 308 } 309 310 311 //-------------------------------------------------- 312 // AHandler implementation 313 void GenericPlayer::onMessageReceived(const sp<AMessage> &msg) { 314 SL_LOGV("GenericPlayer::onMessageReceived()"); 315 switch (msg->what()) { 316 case kWhatPrepare: 317 SL_LOGV("kWhatPrepare"); 318 onPrepare(); 319 break; 320 321 case kWhatNotif: 322 SL_LOGV("kWhatNotif"); 323 onNotify(msg); 324 break; 325 326 case kWhatPlay: 327 SL_LOGV("kWhatPlay"); 328 onPlay(); 329 break; 330 331 case kWhatPause: 332 SL_LOGV("kWhatPause"); 333 onPause(); 334 break; 335 336 case kWhatSeek: 337 SL_LOGV("kWhatSeek"); 338 onSeek(msg); 339 break; 340 341 case kWhatLoop: 342 SL_LOGV("kWhatLoop"); 343 onLoop(msg); 344 break; 345 346 case kWhatVolumeUpdate: 347 SL_LOGV("kWhatVolumeUpdate"); 348 onVolumeUpdate(); 349 break; 350 351 case kWhatSeekComplete: 352 SL_LOGV("kWhatSeekComplete"); 353 onSeekComplete(); 354 break; 355 356 case kWhatBufferingUpdate: 357 SL_LOGV("kWhatBufferingUpdate"); 358 onBufferingUpdate(msg); 359 break; 360 361 case kWhatBuffUpdateThres: 362 SL_LOGV("kWhatBuffUpdateThres"); 363 onSetBufferingUpdateThreshold(msg); 364 break; 365 366 case kWhatAttachAuxEffect: 367 SL_LOGV("kWhatAttachAuxEffect"); 368 onAttachAuxEffect(msg); 369 break; 370 371 case kWhatSetAuxEffectSendLevel: 372 SL_LOGV("kWhatSetAuxEffectSendLevel"); 373 onSetAuxEffectSendLevel(msg); 374 break; 375 376 case kWhatSetPlayEvents: 377 SL_LOGV("kWhatSetPlayEvents"); 378 onSetPlayEvents(msg); 379 break; 380 381 case kWhatOneShot: 382 SL_LOGV("kWhatOneShot"); 383 onOneShot(msg); 384 break; 385 386 default: 387 SL_LOGE("GenericPlayer::onMessageReceived unknown message %d", msg->what()); 388 TRESPASS(); 389 } 390 } 391 392 393 //-------------------------------------------------- 394 // Event handlers 395 // it is strictly verboten to call those methods outside of the event loop 396 397 void GenericPlayer::onPrepare() { 398 SL_LOGV("GenericPlayer::onPrepare()"); 399 // Subclass is responsible for indicating whether prepare was successful or unsuccessful 400 // by updating mStateFlags accordingly. It must set exactly one of these two flags. 401 assert(!(mStateFlags & kFlagPrepared) != !(mStateFlags & kFlagPreparedUnsuccessfully)); 402 notify(PLAYEREVENT_PREPARED, mStateFlags & kFlagPrepared ? PLAYER_SUCCESS : PLAYER_FAILURE, 403 true /*async*/); 404 SL_LOGD("GenericPlayer::onPrepare() done, mStateFlags=0x%x", mStateFlags); 405 } 406 407 408 void GenericPlayer::onNotify(const sp<AMessage> &msg) { 409 SL_LOGV("GenericPlayer::onNotify()"); 410 notif_cbf_t notifClient; 411 void* notifUser; 412 { 413 android::Mutex::Autolock autoLock(mNotifyClientLock); 414 if (NULL == mNotifyClient) { 415 return; 416 } else { 417 notifClient = mNotifyClient; 418 notifUser = mNotifyUser; 419 } 420 } 421 422 int32_t val1, val2; 423 if (msg->findInt32(PLAYEREVENT_PREFETCHSTATUSCHANGE, &val1)) { 424 SL_LOGV("GenericPlayer notifying %s = %d", PLAYEREVENT_PREFETCHSTATUSCHANGE, val1); 425 notifClient(kEventPrefetchStatusChange, val1, 0, notifUser); 426 // There is exactly one notification per message, hence "else if" instead of "if" 427 } else if (msg->findInt32(PLAYEREVENT_PREFETCHFILLLEVELUPDATE, &val1)) { 428 SL_LOGV("GenericPlayer notifying %s = %d", PLAYEREVENT_PREFETCHFILLLEVELUPDATE, val1); 429 notifClient(kEventPrefetchFillLevelUpdate, val1, 0, notifUser); 430 } else if (msg->findInt32(PLAYEREVENT_ENDOFSTREAM, &val1)) { 431 SL_LOGV("GenericPlayer notifying %s = %d", PLAYEREVENT_ENDOFSTREAM, val1); 432 notifClient(kEventEndOfStream, val1, 0, notifUser); 433 } else if (msg->findInt32(PLAYEREVENT_PREPARED, &val1)) { 434 SL_LOGV("GenericPlayer notifying %s = %d", PLAYEREVENT_PREPARED, val1); 435 notifClient(kEventPrepared, val1, 0, notifUser); 436 } else if (msg->findInt32(PLAYEREVENT_CHANNEL_COUNT, &val1)) { 437 SL_LOGV("GenericPlayer notifying %s = %d", PLAYEREVENT_CHANNEL_COUNT, val1); 438 notifClient(kEventChannelCount, val1, 0, notifUser); 439 } else if (msg->findRect(PLAYEREVENT_VIDEO_SIZE_UPDATE, &val1, &val2, &val1, &val2)) { 440 SL_LOGV("GenericPlayer notifying %s = %d, %d", PLAYEREVENT_VIDEO_SIZE_UPDATE, val1, val2); 441 notifClient(kEventHasVideoSize, val1, val2, notifUser); 442 } else if (msg->findInt32(PLAYEREVENT_PLAY, &val1)) { 443 SL_LOGV("GenericPlayer notifying %s = %d", PLAYEREVENT_PLAY, val1); 444 notifClient(kEventPlay, val1, 0, notifUser); 445 } else if (msg->findInt32(PLAYEREVENT_ERRORAFTERPREPARE, &val1)) { 446 SL_LOGV("GenericPlayer notifying %s = %d", PLAYEREVENT_ERRORAFTERPREPARE, val1); 447 notifClient(kEventErrorAfterPrepare, val1, 0, notifUser); 448 } else { 449 SL_LOGV("GenericPlayer notifying unknown"); 450 } 451 } 452 453 454 void GenericPlayer::onPlay() { 455 SL_LOGD("GenericPlayer::onPlay()"); 456 if ((mStateFlags & (kFlagPrepared | kFlagPlaying)) == kFlagPrepared) { 457 SL_LOGD("starting player"); 458 mStateFlags |= kFlagPlaying; 459 updateOneShot(); 460 } 461 } 462 463 464 void GenericPlayer::onPause() { 465 SL_LOGD("GenericPlayer::onPause()"); 466 if (!(~mStateFlags & (kFlagPrepared | kFlagPlaying))) { 467 SL_LOGV("pausing player"); 468 mStateFlags &= ~kFlagPlaying; 469 updateOneShot(); 470 } 471 } 472 473 474 void GenericPlayer::onSeek(const sp<AMessage> &msg) { 475 SL_LOGV("GenericPlayer::onSeek"); 476 } 477 478 479 void GenericPlayer::onLoop(const sp<AMessage> &msg) { 480 SL_LOGV("GenericPlayer::onLoop"); 481 } 482 483 484 void GenericPlayer::onVolumeUpdate() { 485 SL_LOGV("GenericPlayer::onVolumeUpdate"); 486 } 487 488 489 void GenericPlayer::onSeekComplete() { 490 SL_LOGD("GenericPlayer::onSeekComplete()"); 491 mStateFlags &= ~kFlagSeeking; 492 // avoid spurious or lost events caused by seeking past a marker 493 mDeliveredNewPosMs = ANDROID_UNKNOWN_TIME; 494 mObservedPositionMs = ANDROID_UNKNOWN_TIME; 495 updateOneShot(); 496 } 497 498 499 void GenericPlayer::onBufferingUpdate(const sp<AMessage> &msg) { 500 SL_LOGV("GenericPlayer::onBufferingUpdate"); 501 } 502 503 504 void GenericPlayer::onSetBufferingUpdateThreshold(const sp<AMessage> &msg) { 505 SL_LOGV("GenericPlayer::onSetBufferingUpdateThreshold"); 506 int32_t thresholdPercent = 0; 507 if (msg->findInt32(WHATPARAM_BUFFERING_UPDATETHRESHOLD_PERCENT, &thresholdPercent)) { 508 Mutex::Autolock _l(mSettingsLock); 509 mCacheFillNotifThreshold = (int16_t)thresholdPercent; 510 } 511 } 512 513 514 void GenericPlayer::onAttachAuxEffect(const sp<AMessage> &msg) { 515 SL_LOGV("GenericPlayer::onAttachAuxEffect()"); 516 } 517 518 519 void GenericPlayer::onSetAuxEffectSendLevel(const sp<AMessage> &msg) { 520 SL_LOGV("GenericPlayer::onSetAuxEffectSendLevel()"); 521 } 522 523 524 void GenericPlayer::onSetPlayEvents(const sp<AMessage> &msg) { 525 SL_LOGV("GenericPlayer::onSetPlayEvents()"); 526 int32_t eventFlags, markerPositionMs, positionUpdatePeriodMs; 527 if (msg->findInt32(WHATPARAM_SETPLAYEVENTS_FLAGS, &eventFlags) && 528 msg->findInt32(WHATPARAM_SETPLAYEVENTS_MARKER, &markerPositionMs) && 529 msg->findInt32(WHATPARAM_SETPLAYEVENTS_UPDATE, &positionUpdatePeriodMs)) { 530 mEventFlags = eventFlags; 531 mMarkerPositionMs = markerPositionMs; 532 mPositionUpdatePeriodMs = positionUpdatePeriodMs; 533 updateOneShot(); 534 } 535 } 536 537 538 void GenericPlayer::onOneShot(const sp<AMessage> &msg) { 539 SL_LOGV("GenericPlayer::onOneShot()"); 540 int32_t generation; 541 if (msg->findInt32(WHATPARAM_ONESHOT_GENERATION, &generation)) { 542 if (generation != mOneShotGeneration) { 543 SL_LOGV("GenericPlayer::onOneShot() generation %d cancelled; latest is %d", 544 generation, mOneShotGeneration); 545 return; 546 } 547 updateOneShot(); 548 } 549 } 550 551 552 //------------------------------------------------- 553 void GenericPlayer::notifyStatus() { 554 SL_LOGV("GenericPlayer::notifyStatus"); 555 notify(PLAYEREVENT_PREFETCHSTATUSCHANGE, (int32_t)mCacheStatus, true /*async*/); 556 } 557 558 559 void GenericPlayer::notifyCacheFill() { 560 SL_LOGV("GenericPlayer::notifyCacheFill"); 561 mLastNotifiedCacheFill = mCacheFill; 562 notify(PLAYEREVENT_PREFETCHFILLLEVELUPDATE, (int32_t)mLastNotifiedCacheFill, true/*async*/); 563 } 564 565 566 void GenericPlayer::seekComplete() { 567 SL_LOGV("GenericPlayer::seekComplete"); 568 sp<AMessage> msg = new AMessage(kWhatSeekComplete, this); 569 msg->post(); 570 } 571 572 573 void GenericPlayer::bufferingUpdate(int16_t fillLevelPerMille) { 574 SL_LOGV("GenericPlayer::bufferingUpdate"); 575 sp<AMessage> msg = new AMessage(kWhatBufferingUpdate, this); 576 msg->setInt32(WHATPARAM_BUFFERING_UPDATE, fillLevelPerMille); 577 msg->post(); 578 } 579 580 581 // For the meaning of positionMs, see comment in declaration at android_GenericPlayer.h 582 void GenericPlayer::updateOneShot(int positionMs) 583 { 584 SL_LOGV("GenericPlayer::updateOneShot"); 585 586 // nop until prepared 587 if (!(mStateFlags & kFlagPrepared)) { 588 return; 589 } 590 591 // cancel any pending one-shot(s) 592 ++mOneShotGeneration; 593 594 // don't restart one-shot if player is paused or stopped 595 if (!(mStateFlags & kFlagPlaying)) { 596 return; 597 } 598 599 // get current player position in milliseconds 600 if (positionMs < 0) { 601 positionMs = ANDROID_UNKNOWN_TIME; 602 } 603 if (positionMs == ANDROID_UNKNOWN_TIME) { 604 getPositionMsec(&positionMs); 605 // normalize it 606 if (positionMs < 0) { 607 positionMs = ANDROID_UNKNOWN_TIME; 608 } 609 if (ANDROID_UNKNOWN_TIME == positionMs) { 610 // getPositionMsec is not working for some reason, give up 611 //ALOGV("Does anyone really know what time it is?"); 612 return; 613 } 614 } 615 616 // if we observe the player position going backwards, even without without a seek, then recover 617 if (mObservedPositionMs != ANDROID_UNKNOWN_TIME && positionMs < mObservedPositionMs) { 618 mDeliveredNewPosMs = ANDROID_UNKNOWN_TIME; 619 mObservedPositionMs = positionMs; 620 } 621 622 // delayUs is the expected delay between current position and marker; 623 // the default is infinity in case there are no upcoming marker(s) 624 int64_t delayUs = -1; 625 626 // is there a marker? 627 if ((mEventFlags & SL_PLAYEVENT_HEADATMARKER) && (mMarkerPositionMs != ANDROID_UNKNOWN_TIME)) { 628 // check to see if we have observed the position passing through the marker 629 if (mObservedPositionMs <= mMarkerPositionMs && mMarkerPositionMs <= positionMs) { 630 notify(PLAYEREVENT_PLAY, (int32_t) SL_PLAYEVENT_HEADATMARKER, true /*async*/); 631 } else if (positionMs < mMarkerPositionMs) { 632 delayUs = (mMarkerPositionMs - positionMs) * 1000LL; 633 } 634 } 635 636 // are periodic position updates needed? 637 if ((mEventFlags & SL_PLAYEVENT_HEADATNEWPOS) && 638 (mPositionUpdatePeriodMs != ANDROID_UNKNOWN_TIME)) { 639 // check to see if we have observed the position passing through a virtual marker, where the 640 // virtual marker is at the previously delivered new position plus position update period 641 int32_t virtualMarkerMs; 642 if (mDeliveredNewPosMs != ANDROID_UNKNOWN_TIME) { 643 virtualMarkerMs = mDeliveredNewPosMs + mPositionUpdatePeriodMs; 644 } else if (mObservedPositionMs != ANDROID_UNKNOWN_TIME) { 645 virtualMarkerMs = mObservedPositionMs + mPositionUpdatePeriodMs; 646 // pretend there has been an update in the past 647 mDeliveredNewPosMs = mObservedPositionMs; 648 } else { 649 virtualMarkerMs = positionMs + mPositionUpdatePeriodMs; 650 // pretend there has been an update in the past 651 mDeliveredNewPosMs = positionMs; 652 } 653 // nextVirtualMarkerMs will be set to the position of the next upcoming virtual marker 654 int32_t nextVirtualMarkerMs; 655 if (mObservedPositionMs <= virtualMarkerMs && virtualMarkerMs <= positionMs) { 656 // we did pass through the virtual marker, now compute the next virtual marker 657 mDeliveredNewPosMs = virtualMarkerMs; 658 nextVirtualMarkerMs = virtualMarkerMs + mPositionUpdatePeriodMs; 659 // re-synchronize if we missed an update 660 if (nextVirtualMarkerMs <= positionMs) { 661 SL_LOGW("Missed SL_PLAYEVENT_HEADATNEWPOS for position %d; current position %d", 662 nextVirtualMarkerMs, positionMs); 663 // try to catch up by setting next goal to current position plus update period 664 mDeliveredNewPosMs = positionMs; 665 nextVirtualMarkerMs = positionMs + mPositionUpdatePeriodMs; 666 } 667 notify(PLAYEREVENT_PLAY, (int32_t) SL_PLAYEVENT_HEADATNEWPOS, true /*async*/); 668 } else { 669 // we did not pass through the virtual marker yet, so use same marker again 670 nextVirtualMarkerMs = virtualMarkerMs; 671 } 672 // note that if arithmetic overflow occurred, nextVirtualMarkerMs will be negative 673 if (positionMs < nextVirtualMarkerMs) { 674 int64_t trialDelayUs; 675 trialDelayUs = (nextVirtualMarkerMs - positionMs) * 1000LL; 676 if (trialDelayUs > 0 && (delayUs == -1 || trialDelayUs < delayUs)) { 677 delayUs = trialDelayUs; 678 } 679 } 680 } 681 682 // we have a new observed position 683 mObservedPositionMs = positionMs; 684 685 if (mPlaybackRatePermille == 0) { 686 // playback is frozen, no update expected (and no division by zero below) 687 return; 688 } 689 690 // post the new one-shot message if needed 691 if (advancesPositionInRealTime() && delayUs >= 0) { 692 // scale delay according to playback rate (reported positions won't change, but reported 693 // time will advance slower or faster depending on rate) 694 { 695 Mutex::Autolock _l(mSettingsLock); 696 delayUs = delayUs * 1000 / mPlaybackRatePermille; 697 } 698 699 // 20 ms min delay to avoid near busy waiting 700 if (delayUs < 20000LL) { 701 delayUs = 20000LL; 702 } 703 // 1 minute max delay avoids indefinite memory leaks caused by cancelled one-shots 704 if (delayUs > 60000000LL) { 705 delayUs = 60000000LL; 706 } 707 //SL_LOGI("delayUs = %lld", delayUs); 708 sp<AMessage> msg = new AMessage(kWhatOneShot, this); 709 msg->setInt32(WHATPARAM_ONESHOT_GENERATION, mOneShotGeneration); 710 msg->post(delayUs); 711 } 712 713 } 714 715 } // namespace android 716