| /* |
| * Copyright (C) 2010 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #define USE_LOG SLAndroidLogLevel_Verbose |
| |
| #include "sles_allinclusive.h" |
| #include <media/IMediaPlayerService.h> |
| |
| //-------------------------------------------------------------------------------------------------- |
| // FIXME abstract out the diff between CMediaPlayer and CAudioPlayer |
| |
| void android_StreamPlayer_realize_l(CAudioPlayer *ap, const notif_cbf_t cbf, void* notifUser) { |
| SL_LOGI("android_StreamPlayer_realize_l(%p)", ap); |
| |
| AudioPlayback_Parameters ap_params; |
| ap_params.sessionId = ap->mSessionId; |
| ap_params.streamType = ap->mStreamType; |
| ap_params.trackcb = NULL; |
| ap_params.trackcbUser = NULL; |
| android::StreamPlayer* splr = new android::StreamPlayer(&ap_params, false /*hasVideo*/); |
| ap->mAPlayer = splr; |
| splr->init(cbf, notifUser); |
| } |
| |
| |
| //-------------------------------------------------------------------------------------------------- |
| namespace android { |
| |
| StreamSourceAppProxy::StreamSourceAppProxy( |
| const void* user, bool userIsAudioPlayer, |
| void *context, const void *caller) : |
| mUser(user), |
| mUserIsAudioPlayer(userIsAudioPlayer), |
| mAndroidBufferQueue(NULL), |
| mAppContext(context), |
| mCaller(caller) |
| { |
| SL_LOGI("StreamSourceAppProxy::StreamSourceAppProxy()"); |
| |
| if (mUserIsAudioPlayer) { |
| mAndroidBufferQueue = &((CAudioPlayer*)mUser)->mAndroidBufferQueue; |
| } else { |
| mAndroidBufferQueue = &((CMediaPlayer*)mUser)->mAndroidBufferQueue; |
| } |
| } |
| |
| StreamSourceAppProxy::~StreamSourceAppProxy() { |
| SL_LOGI("StreamSourceAppProxy::~StreamSourceAppProxy()"); |
| mListener.clear(); |
| mBuffers.clear(); |
| } |
| |
| //-------------------------------------------------- |
| // IStreamSource implementation |
| void StreamSourceAppProxy::setListener(const sp<IStreamListener> &listener) { |
| Mutex::Autolock _l(mLock); |
| mListener = listener; |
| } |
| |
| void StreamSourceAppProxy::setBuffers(const Vector<sp<IMemory> > &buffers) { |
| mBuffers = buffers; |
| } |
| |
| void StreamSourceAppProxy::onBufferAvailable(size_t index) { |
| //SL_LOGD("StreamSourceAppProxy::onBufferAvailable(%d)", index); |
| |
| CHECK_LT(index, mBuffers.size()); |
| sp<IMemory> mem = mBuffers.itemAt(index); |
| SLAint64 length = (SLAint64) mem->size(); |
| |
| { |
| Mutex::Autolock _l(mLock); |
| mAvailableBuffers.push_back(index); |
| } |
| //SL_LOGD("onBufferAvailable() now %d buffers available in queue", mAvailableBuffers.size()); |
| |
| // a new shared mem buffer is available: let's try to fill immediately |
| pullFromBuffQueue(); |
| } |
| |
| void StreamSourceAppProxy::receivedCmd_l(IStreamListener::Command cmd, const sp<AMessage> &msg) { |
| if (mListener != 0) { |
| mListener->issueCommand(cmd, false /* synchronous */, msg); |
| } |
| } |
| |
| void StreamSourceAppProxy::receivedBuffer_l(size_t buffIndex, size_t buffLength) { |
| if (mListener != 0) { |
| mListener->queueBuffer(buffIndex, buffLength); |
| } |
| } |
| |
| //-------------------------------------------------- |
| // consumption from ABQ |
| void StreamSourceAppProxy::pullFromBuffQueue() { |
| |
| size_t bufferId; |
| void* bufferLoc; |
| size_t buffSize; |
| |
| slAndroidBufferQueueCallback callback = NULL; |
| void* callbackPContext = NULL; |
| AdvancedBufferHeader *oldFront = NULL; |
| |
| // retrieve data from the buffer queue |
| interface_lock_exclusive(mAndroidBufferQueue); |
| |
| if (mAndroidBufferQueue->mState.count != 0) { |
| // SL_LOGD("nbBuffers in ABQ = %lu, buffSize=%lu",abq->mState.count, buffSize); |
| assert(mAndroidBufferQueue->mFront != mAndroidBufferQueue->mRear); |
| |
| oldFront = mAndroidBufferQueue->mFront; |
| AdvancedBufferHeader *newFront = &oldFront[1]; |
| |
| // consume events when starting to read data from a buffer for the first time |
| if (oldFront->mDataSizeConsumed == 0) { |
| if (oldFront->mItems.mTsCmdData.mTsCmdCode & ANDROID_MP2TSEVENT_EOS) { |
| receivedCmd_l(IStreamListener::EOS); |
| } else if (oldFront->mItems.mTsCmdData.mTsCmdCode & ANDROID_MP2TSEVENT_DISCONTINUITY) { |
| receivedCmd_l(IStreamListener::DISCONTINUITY); |
| } else if (oldFront->mItems.mTsCmdData.mTsCmdCode & ANDROID_MP2TSEVENT_DISCON_NEWPTS) { |
| sp<AMessage> msg = new AMessage(); |
| msg->setInt64(IStreamListener::kKeyResumeAtPTS, |
| (int64_t)oldFront->mItems.mTsCmdData.mPts); |
| receivedCmd_l(IStreamListener::DISCONTINUITY, msg /*msg*/); |
| } |
| oldFront->mItems.mTsCmdData.mTsCmdCode = ANDROID_MP2TSEVENT_NONE; |
| } |
| |
| { |
| // we're going to change the shared mem buffer queue, so lock it |
| Mutex::Autolock _l(mLock); |
| if (!mAvailableBuffers.empty()) { |
| bufferId = *mAvailableBuffers.begin(); |
| CHECK_LT(bufferId, mBuffers.size()); |
| sp<IMemory> mem = mBuffers.itemAt(bufferId); |
| bufferLoc = mem->pointer(); |
| buffSize = mem->size(); |
| |
| char *pSrc = ((char*)oldFront->mDataBuffer) + oldFront->mDataSizeConsumed; |
| if (oldFront->mDataSizeConsumed + buffSize < oldFront->mDataSize) { |
| // more available than requested, copy as much as requested |
| // consume data: 1/ copy to given destination |
| memcpy(bufferLoc, pSrc, buffSize); |
| // 2/ keep track of how much has been consumed |
| oldFront->mDataSizeConsumed += buffSize; |
| // 3/ notify shared mem listener that new data is available |
| receivedBuffer_l(bufferId, buffSize); |
| mAvailableBuffers.erase(mAvailableBuffers.begin()); |
| } else { |
| // requested as much available or more: consume the whole of the current |
| // buffer and move to the next |
| size_t consumed = oldFront->mDataSize - oldFront->mDataSizeConsumed; |
| //SL_LOGD("consuming rest of buffer: enqueueing=%ld", consumed); |
| oldFront->mDataSizeConsumed = oldFront->mDataSize; |
| |
| // move queue to next |
| if (newFront == &mAndroidBufferQueue-> |
| mBufferArray[mAndroidBufferQueue->mNumBuffers + 1]) { |
| // reached the end, circle back |
| newFront = mAndroidBufferQueue->mBufferArray; |
| } |
| mAndroidBufferQueue->mFront = newFront; |
| mAndroidBufferQueue->mState.count--; |
| mAndroidBufferQueue->mState.index++; |
| |
| if (consumed > 0) { |
| // consume data: 1/ copy to given destination |
| memcpy(bufferLoc, pSrc, consumed); |
| // 2/ keep track of how much has been consumed |
| // here nothing to do because we are done with this buffer |
| // 3/ notify StreamPlayer that new data is available |
| receivedBuffer_l(bufferId, consumed); |
| mAvailableBuffers.erase(mAvailableBuffers.begin()); |
| } |
| |
| // data has been consumed, and the buffer queue state has been updated |
| // we will notify the client if applicable |
| callback = mAndroidBufferQueue->mCallback; |
| // save callback data |
| callbackPContext = mAndroidBufferQueue->mContext; |
| } |
| //SL_LOGD("onBufferAvailable() %d buffers available after enqueue", |
| // mAvailableBuffers.size()); |
| } |
| } |
| |
| |
| } else { // empty queue |
| SL_LOGD("ABQ empty, starving!"); |
| // signal we're at the end of the content, but don't pause (see note in function) |
| if (mUserIsAudioPlayer) { |
| // FIXME declare this external |
| //audioPlayer_dispatch_headAtEnd_lockPlay((CAudioPlayer*)user, |
| // false /*set state to paused?*/, false); |
| } else { |
| // FIXME implement headAtEnd dispatch for CMediaPlayer |
| } |
| } |
| |
| interface_unlock_exclusive(mAndroidBufferQueue); |
| |
| // notify client |
| if (NULL != callback) { |
| // oldFront was only initialized in the code path where callback is initialized |
| // so no need to check if it's valid |
| (*callback)(&mAndroidBufferQueue->mItf, callbackPContext, |
| (void *)oldFront->mDataBuffer,/* pBufferData */ |
| oldFront->mDataSize, /* dataSize */ |
| // here a buffer is only dequeued when fully consumed |
| oldFront->mDataSize, /* dataUsed */ |
| // no messages during playback |
| 0, /* itemsLength */ |
| NULL /* pItems */); |
| } |
| } |
| |
| |
| //-------------------------------------------------------------------------------------------------- |
| StreamPlayer::StreamPlayer(AudioPlayback_Parameters* params, bool hasVideo) : |
| GenericMediaPlayer(params, hasVideo), |
| mAppProxy(0) |
| { |
| SL_LOGI("StreamPlayer::StreamPlayer()"); |
| |
| mPlaybackParams = *params; |
| |
| } |
| |
| StreamPlayer::~StreamPlayer() { |
| SL_LOGI("StreamPlayer::~StreamPlayer()"); |
| |
| mAppProxy.clear(); |
| } |
| |
| |
| void StreamPlayer::onMessageReceived(const sp<AMessage> &msg) { |
| switch (msg->what()) { |
| case kWhatQueueRefilled: |
| onQueueRefilled(); |
| break; |
| |
| default: |
| GenericMediaPlayer::onMessageReceived(msg); |
| break; |
| } |
| } |
| |
| |
| void StreamPlayer::registerQueueCallback( |
| const void* user, bool userIsAudioPlayer, |
| void *context, |
| const void *caller) { |
| SL_LOGI("StreamPlayer::registerQueueCallback"); |
| Mutex::Autolock _l(mAppProxyLock); |
| |
| mAppProxy = new StreamSourceAppProxy( |
| user, userIsAudioPlayer, |
| context, caller); |
| |
| CHECK(mAppProxy != 0); |
| SL_LOGI("StreamPlayer::registerQueueCallback end"); |
| } |
| |
| |
| /** |
| * Called with a lock on ABQ |
| */ |
| void StreamPlayer::queueRefilled_l() { |
| // async notification that the ABQ was refilled |
| (new AMessage(kWhatQueueRefilled, id()))->post(); |
| } |
| |
| |
| void StreamPlayer::appClear_l() { |
| // the user of StreamPlayer has cleared its AndroidBufferQueue: a discontinuity is expected |
| Mutex::Autolock _l(mAppProxyLock); |
| if (mAppProxy != 0) { |
| mAppProxy->receivedCmd_l(IStreamListener::DISCONTINUITY); |
| } |
| } |
| |
| |
| //-------------------------------------------------- |
| // Event handlers |
| void StreamPlayer::onPrepare() { |
| SL_LOGI("StreamPlayer::onPrepare()"); |
| Mutex::Autolock _l(mAppProxyLock); |
| if (mAppProxy != 0) { |
| mPlayer = mMediaPlayerService->create(getpid(), mPlayerClient /*IMediaPlayerClient*/, |
| mAppProxy /*IStreamSource*/, mPlaybackParams.sessionId); |
| // blocks until mPlayer is prepared |
| GenericMediaPlayer::onPrepare(); |
| SL_LOGI("StreamPlayer::onPrepare() done"); |
| } else { |
| SL_LOGE("Nothing to do here because there is no registered callback"); |
| } |
| } |
| |
| |
| void StreamPlayer::onQueueRefilled() { |
| //SL_LOGD("StreamPlayer::onQueueRefilled()"); |
| Mutex::Autolock _l(mAppProxyLock); |
| if (mAppProxy != 0) { |
| mAppProxy->pullFromBuffQueue(); |
| } |
| } |
| |
| } // namespace android |