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