mediaplayer.cpp revision 5daeb129a2c2ba3d14ccd94af283b5f561c783ea
189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project/* mediaplayer.cpp
289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project**
389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** Copyright 2006, The Android Open Source Project
489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project**
589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** Licensed under the Apache License, Version 2.0 (the "License");
689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** you may not use this file except in compliance with the License.
789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** You may obtain a copy of the License at
889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project**
989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project**     http://www.apache.org/licenses/LICENSE-2.0
1089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project**
1189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** Unless required by applicable law or agreed to in writing, software
1289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** distributed under the License is distributed on an "AS IS" BASIS,
1389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** See the License for the specific language governing permissions and
1589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** limitations under the License.
1689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project*/
1789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
1889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project//#define LOG_NDEBUG 0
1989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#define LOG_TAG "MediaPlayer"
2089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <utils/Log.h>
2189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
2289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <sys/types.h>
2389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <sys/stat.h>
2489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <unistd.h>
2589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <fcntl.h>
2689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
277562408b2261d38415453378b6188f74fda99d88Mathias Agopian#include <binder/IServiceManager.h>
287562408b2261d38415453378b6188f74fda99d88Mathias Agopian#include <binder/IPCThreadState.h>
2989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
3089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <media/mediaplayer.h>
3189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <media/AudioTrack.h>
3289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
333cf613507f1e2f7bd932d921a6e222e426fd3be4Mathias Agopian#include <surfaceflinger/Surface.h>
343cf613507f1e2f7bd932d921a6e222e426fd3be4Mathias Agopian
357562408b2261d38415453378b6188f74fda99d88Mathias Agopian#include <binder/MemoryBase.h>
3689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
372db8455d8f4468a637109d31f319ce02d9d743ecAndreas Huber#include <utils/KeyedVector.h>
382db8455d8f4468a637109d31f319ce02d9d743ecAndreas Huber#include <utils/String8.h>
392db8455d8f4468a637109d31f319ce02d9d743ecAndreas Huber
4089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectnamespace android {
4189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
4289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectMediaPlayer::MediaPlayer()
4389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
4489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("constructor");
4589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mListener = NULL;
4689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mCookie = NULL;
4789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mDuration = -1;
4889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mStreamType = AudioSystem::MUSIC;
4989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mCurrentPosition = -1;
5089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mSeekPosition = -1;
5189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mCurrentState = MEDIA_PLAYER_IDLE;
5289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mPrepareSync = false;
5389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mPrepareStatus = NO_ERROR;
5489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLoop = false;
5589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLeftVolume = mRightVolume = 1.0;
5689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mVideoWidth = mVideoHeight = 0;
571af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams    mLockThreadId = 0;
58a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent    mAudioSessionId = AudioSystem::newAudioSessionId();
5989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
6089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
6189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectMediaPlayer::~MediaPlayer()
6289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
6389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("destructor");
6489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    disconnect();
6589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    IPCThreadState::self()->flushCommands();
6689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
6789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
6889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayer::disconnect()
6989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
7089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("disconnect");
7189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<IMediaPlayer> p;
7289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    {
7389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        Mutex::Autolock _l(mLock);
7489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p = mPlayer;
7589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mPlayer.clear();
7689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
7789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
7889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p != 0) {
7989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p->disconnect();
8089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
8189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
8289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
8389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// always call with lock held
8489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayer::clear_l()
8589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
8689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mDuration = -1;
8789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mCurrentPosition = -1;
8889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mSeekPosition = -1;
8989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mVideoWidth = mVideoHeight = 0;
9089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
9189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
9289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setListener(const sp<MediaPlayerListener>& listener)
9389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
9489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("setListener");
9589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
9689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mListener = listener;
9789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
9889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
9989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
10089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
10189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setDataSource(const sp<IMediaPlayer>& player)
10289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
10389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    status_t err = UNKNOWN_ERROR;
10489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<IMediaPlayer> p;
10589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    { // scope for the lock
10689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        Mutex::Autolock _l(mLock);
10789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
10883ff1438d2d1d5dbf39ca5e6f2e4fa1799e7ba80Marco Nelissen        if ( !( (mCurrentState & MEDIA_PLAYER_IDLE) ||
10983ff1438d2d1d5dbf39ca5e6f2e4fa1799e7ba80Marco Nelissen                (mCurrentState == MEDIA_PLAYER_STATE_ERROR ) ) ) {
11089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGE("setDataSource called in state %d", mCurrentState);
11189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            return INVALID_OPERATION;
11289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
11389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
11489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        clear_l();
11589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p = mPlayer;
11689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mPlayer = player;
11789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (player != 0) {
11889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_INITIALIZED;
11989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            err = NO_ERROR;
12089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else {
12189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGE("Unable to to create media player");
12289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
12389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
12489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
12589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p != 0) {
12689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p->disconnect();
12789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
12889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
12989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return err;
13089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
13189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
1322db8455d8f4468a637109d31f319ce02d9d743ecAndreas Huberstatus_t MediaPlayer::setDataSource(
1332db8455d8f4468a637109d31f319ce02d9d743ecAndreas Huber        const char *url, const KeyedVector<String8, String8> *headers)
13489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
13589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("setDataSource(%s)", url);
13689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    status_t err = BAD_VALUE;
13789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (url != NULL) {
13889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        const sp<IMediaPlayerService>& service(getMediaPlayerService());
13989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (service != 0) {
1402db8455d8f4468a637109d31f319ce02d9d743ecAndreas Huber            sp<IMediaPlayer> player(
141a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent                    service->create(getpid(), this, url, headers, mAudioSessionId));
14289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            err = setDataSource(player);
14389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
14489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
14589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return err;
14689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
14789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
14889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setDataSource(int fd, int64_t offset, int64_t length)
14989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
15089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("setDataSource(%d, %lld, %lld)", fd, offset, length);
15189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    status_t err = UNKNOWN_ERROR;
15289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    const sp<IMediaPlayerService>& service(getMediaPlayerService());
15389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (service != 0) {
154a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent        sp<IMediaPlayer> player(service->create(getpid(), this, fd, offset, length, mAudioSessionId));
15589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        err = setDataSource(player);
15689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
15789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return err;
15889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
15989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
1601d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Cataniastatus_t MediaPlayer::invoke(const Parcel& request, Parcel *reply)
1611d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania{
1621d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania    Mutex::Autolock _l(mLock);
1634023493a827bc9751d8e40795516d0d536a53348Nicolas Catania    const bool hasBeenInitialized =
1644023493a827bc9751d8e40795516d0d536a53348Nicolas Catania            (mCurrentState != MEDIA_PLAYER_STATE_ERROR) &&
1654023493a827bc9751d8e40795516d0d536a53348Nicolas Catania            ((mCurrentState & MEDIA_PLAYER_IDLE) != MEDIA_PLAYER_IDLE);
1664023493a827bc9751d8e40795516d0d536a53348Nicolas Catania    if ((mPlayer != NULL) && hasBeenInitialized) {
1671d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania         LOGV("invoke %d", request.dataSize());
1681d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania         return  mPlayer->invoke(request, reply);
1691d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania    }
1701d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania    LOGE("invoke failed: wrong state %X", mCurrentState);
1711d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania    return INVALID_OPERATION;
1721d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania}
1731d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania
1744e92c7efc68b5d7635a32dbd283ab8558ad9bdd6Andreas Huberstatus_t MediaPlayer::suspend() {
1754e92c7efc68b5d7635a32dbd283ab8558ad9bdd6Andreas Huber    Mutex::Autolock _l(mLock);
1764e92c7efc68b5d7635a32dbd283ab8558ad9bdd6Andreas Huber    return mPlayer->suspend();
1774e92c7efc68b5d7635a32dbd283ab8558ad9bdd6Andreas Huber}
1784e92c7efc68b5d7635a32dbd283ab8558ad9bdd6Andreas Huber
1794e92c7efc68b5d7635a32dbd283ab8558ad9bdd6Andreas Huberstatus_t MediaPlayer::resume() {
1804e92c7efc68b5d7635a32dbd283ab8558ad9bdd6Andreas Huber    Mutex::Autolock _l(mLock);
1814e92c7efc68b5d7635a32dbd283ab8558ad9bdd6Andreas Huber    return mPlayer->resume();
1824e92c7efc68b5d7635a32dbd283ab8558ad9bdd6Andreas Huber}
1834e92c7efc68b5d7635a32dbd283ab8558ad9bdd6Andreas Huber
184a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Cataniastatus_t MediaPlayer::setMetadataFilter(const Parcel& filter)
185a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania{
186a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    LOGD("setMetadataFilter");
1878e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania    Mutex::Autolock lock(mLock);
1888e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania    if (mPlayer == NULL) {
189a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        return NO_INIT;
190a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    }
191a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    return mPlayer->setMetadataFilter(filter);
192a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania}
1931d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania
1948e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Cataniastatus_t MediaPlayer::getMetadata(bool update_only, bool apply_filter, Parcel *metadata)
1958e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania{
1968e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania    LOGD("getMetadata");
1978e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania    Mutex::Autolock lock(mLock);
1988e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania    if (mPlayer == NULL) {
1998e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania        return NO_INIT;
2008e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania    }
2018e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania    return mPlayer->getMetadata(update_only, apply_filter, metadata);
2028e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania}
2038e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania
20489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setVideoSurface(const sp<Surface>& surface)
20589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
20689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("setVideoSurface");
20789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
20889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer == 0) return NO_INIT;
2095daeb129a2c2ba3d14ccd94af283b5f561c783eaAndreas Huber
2105daeb129a2c2ba3d14ccd94af283b5f561c783eaAndreas Huber    status_t err = mPlayer->setVideoISurface(
2115daeb129a2c2ba3d14ccd94af283b5f561c783eaAndreas Huber            surface == NULL ? NULL : surface->getISurface());
2125daeb129a2c2ba3d14ccd94af283b5f561c783eaAndreas Huber
2135daeb129a2c2ba3d14ccd94af283b5f561c783eaAndreas Huber    if (err != OK) {
2145daeb129a2c2ba3d14ccd94af283b5f561c783eaAndreas Huber        return err;
2155daeb129a2c2ba3d14ccd94af283b5f561c783eaAndreas Huber    }
2165daeb129a2c2ba3d14ccd94af283b5f561c783eaAndreas Huber
2175daeb129a2c2ba3d14ccd94af283b5f561c783eaAndreas Huber    return mPlayer->setVideoSurface(surface);
21889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
21989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
22089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// must call with lock held
22189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::prepareAsync_l()
22289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
22389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if ( (mPlayer != 0) && ( mCurrentState & ( MEDIA_PLAYER_INITIALIZED | MEDIA_PLAYER_STOPPED) ) ) {
22489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mPlayer->setAudioStreamType(mStreamType);
22589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mCurrentState = MEDIA_PLAYER_PREPARING;
22689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return mPlayer->prepareAsync();
22789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
22889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGE("prepareAsync called in state %d", mCurrentState);
22989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return INVALID_OPERATION;
23089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
23189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
23265e731f393f704eedab6fbe0af7f8a580c8d4617The Android Open Source Project// TODO: In case of error, prepareAsync provides the caller with 2 error codes,
23365e731f393f704eedab6fbe0af7f8a580c8d4617The Android Open Source Project// one defined in the Android framework and one provided by the implementation
23465e731f393f704eedab6fbe0af7f8a580c8d4617The Android Open Source Project// that generated the error. The sync version of prepare returns only 1 error
23565e731f393f704eedab6fbe0af7f8a580c8d4617The Android Open Source Project// code.
23689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::prepare()
23789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
23889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("prepare");
23989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
2401af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams    mLockThreadId = getThreadId();
2411af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams    if (mPrepareSync) {
2421af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams        mLockThreadId = 0;
2431af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams        return -EALREADY;
2441af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams    }
24589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mPrepareSync = true;
24689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    status_t ret = prepareAsync_l();
2471af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams    if (ret != NO_ERROR) {
2481af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams        mLockThreadId = 0;
2491af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams        return ret;
2501af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams    }
25189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
25289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPrepareSync) {
25389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mSignal.wait(mLock);  // wait for prepare done
25489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mPrepareSync = false;
25589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
25689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("prepare complete - status=%d", mPrepareStatus);
2571af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams    mLockThreadId = 0;
25889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return mPrepareStatus;
25989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
26089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
26189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::prepareAsync()
26289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
26389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("prepareAsync");
26489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
26589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return prepareAsync_l();
26689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
26789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
26889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::start()
26989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
27089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("start");
27189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
27289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mCurrentState & MEDIA_PLAYER_STARTED)
27389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return NO_ERROR;
27489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if ( (mPlayer != 0) && ( mCurrentState & ( MEDIA_PLAYER_PREPARED |
27589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                    MEDIA_PLAYER_PLAYBACK_COMPLETE | MEDIA_PLAYER_PAUSED ) ) ) {
27689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mPlayer->setLooping(mLoop);
27789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mPlayer->setVolume(mLeftVolume, mRightVolume);
2782beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent        mPlayer->setAuxEffectSendLevel(mSendLevel);
27989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mCurrentState = MEDIA_PLAYER_STARTED;
28089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        status_t ret = mPlayer->start();
28189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (ret != NO_ERROR) {
28289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_STATE_ERROR;
28389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else {
28489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            if (mCurrentState == MEDIA_PLAYER_PLAYBACK_COMPLETE) {
28589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                LOGV("playback completed immediately following start()");
28689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            }
28789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
28889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return ret;
28989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
29089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGE("start called in state %d", mCurrentState);
29189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return INVALID_OPERATION;
29289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
29389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
29489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::stop()
29589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
29689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("stop");
29789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
29889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mCurrentState & MEDIA_PLAYER_STOPPED) return NO_ERROR;
29989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if ( (mPlayer != 0) && ( mCurrentState & ( MEDIA_PLAYER_STARTED | MEDIA_PLAYER_PREPARED |
30089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                    MEDIA_PLAYER_PAUSED | MEDIA_PLAYER_PLAYBACK_COMPLETE ) ) ) {
30189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        status_t ret = mPlayer->stop();
30289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (ret != NO_ERROR) {
30389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_STATE_ERROR;
30489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else {
30589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_STOPPED;
30689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
30789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return ret;
30889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
30989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGE("stop called in state %d", mCurrentState);
31089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return INVALID_OPERATION;
31189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
31289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
31389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::pause()
31489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
31589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("pause");
31689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
317698f476590bc9e38d4d1d4155da9efdbedd357c4Marco Nelissen    if (mCurrentState & (MEDIA_PLAYER_PAUSED|MEDIA_PLAYER_PLAYBACK_COMPLETE))
31889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return NO_ERROR;
31989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if ((mPlayer != 0) && (mCurrentState & MEDIA_PLAYER_STARTED)) {
32089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        status_t ret = mPlayer->pause();
32189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (ret != NO_ERROR) {
32289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_STATE_ERROR;
32389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else {
32489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_PAUSED;
32589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
32689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return ret;
32789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
32889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGE("pause called in state %d", mCurrentState);
32989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return INVALID_OPERATION;
33089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
33189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
33289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectbool MediaPlayer::isPlaying()
33389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
33489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
33589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer != 0) {
33689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        bool temp = false;
33789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mPlayer->isPlaying(&temp);
33889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("isPlaying: %d", temp);
33989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if ((mCurrentState & MEDIA_PLAYER_STARTED) && ! temp) {
34089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGE("internal/external state mismatch corrected");
34189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_PAUSED;
34289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
34389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return temp;
34489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
34589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("isPlaying: no active player");
34689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return false;
34789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
34889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
34989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::getVideoWidth(int *w)
35089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
35189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("getVideoWidth");
35289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
35389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer == 0) return INVALID_OPERATION;
35489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    *w = mVideoWidth;
35589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
35689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
35789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
35889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::getVideoHeight(int *h)
35989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
36089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("getVideoHeight");
36189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
36289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer == 0) return INVALID_OPERATION;
36389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    *h = mVideoHeight;
36489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
36589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
36689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
36789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::getCurrentPosition(int *msec)
36889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
36989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("getCurrentPosition");
37089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
37189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer != 0) {
37289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (mCurrentPosition >= 0) {
37389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGV("Using cached seek position: %d", mCurrentPosition);
37489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            *msec = mCurrentPosition;
37589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            return NO_ERROR;
37689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
37789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return mPlayer->getCurrentPosition(msec);
37889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
37989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return INVALID_OPERATION;
38089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
38189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
38289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::getDuration_l(int *msec)
38389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
38489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("getDuration");
38589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    bool isValidState = (mCurrentState & (MEDIA_PLAYER_PREPARED | MEDIA_PLAYER_STARTED | MEDIA_PLAYER_PAUSED | MEDIA_PLAYER_STOPPED | MEDIA_PLAYER_PLAYBACK_COMPLETE));
38689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer != 0 && isValidState) {
38789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        status_t ret = NO_ERROR;
38889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (mDuration <= 0)
38989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            ret = mPlayer->getDuration(&mDuration);
39089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (msec)
39189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            *msec = mDuration;
39289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return ret;
39389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
39489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGE("Attempt to call getDuration without a valid mediaplayer");
39589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return INVALID_OPERATION;
39689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
39789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
39889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::getDuration(int *msec)
39989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
40089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
40189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return getDuration_l(msec);
40289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
40389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
40489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::seekTo_l(int msec)
40589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
40689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("seekTo %d", msec);
40789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if ((mPlayer != 0) && ( mCurrentState & ( MEDIA_PLAYER_STARTED | MEDIA_PLAYER_PREPARED | MEDIA_PLAYER_PAUSED |  MEDIA_PLAYER_PLAYBACK_COMPLETE) ) ) {
40889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if ( msec < 0 ) {
40989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGW("Attempt to seek to invalid position: %d", msec);
41089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            msec = 0;
41189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else if ((mDuration > 0) && (msec > mDuration)) {
41289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGW("Attempt to seek to past end of file: request = %d, EOF = %d", msec, mDuration);
41389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            msec = mDuration;
41489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
41589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        // cache duration
41689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mCurrentPosition = msec;
41789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (mSeekPosition < 0) {
41889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            getDuration_l(NULL);
41989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mSeekPosition = msec;
42089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            return mPlayer->seekTo(msec);
42189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
42289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        else {
42389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGV("Seek in progress - queue up seekTo[%d]", msec);
42489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            return NO_ERROR;
42589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
42689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
42789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGE("Attempt to perform seekTo in wrong state: mPlayer=%p, mCurrentState=%u", mPlayer.get(), mCurrentState);
42889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return INVALID_OPERATION;
42989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
43089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
43189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::seekTo(int msec)
43289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
4335cb07aa071b43a214e4c880b3b7852714e06451bAndreas Huber    mLockThreadId = getThreadId();
43489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
4355cb07aa071b43a214e4c880b3b7852714e06451bAndreas Huber    status_t result = seekTo_l(msec);
4365cb07aa071b43a214e4c880b3b7852714e06451bAndreas Huber    mLockThreadId = 0;
4375cb07aa071b43a214e4c880b3b7852714e06451bAndreas Huber
4385cb07aa071b43a214e4c880b3b7852714e06451bAndreas Huber    return result;
43989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
44089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
44189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::reset()
44289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
44389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("reset");
44489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
44589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLoop = false;
44689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mCurrentState == MEDIA_PLAYER_IDLE) return NO_ERROR;
44789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mPrepareSync = false;
44889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer != 0) {
44989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        status_t ret = mPlayer->reset();
45089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (ret != NO_ERROR) {
45189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGE("reset() failed with return code (%d)", ret);
45289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_STATE_ERROR;
45389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else {
45489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_IDLE;
45589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
45689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return ret;
45789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
45889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    clear_l();
45989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
46089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
46189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
46289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setAudioStreamType(int type)
46389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
46489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("MediaPlayer::setAudioStreamType");
46589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
46689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mStreamType == type) return NO_ERROR;
46789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mCurrentState & ( MEDIA_PLAYER_PREPARED | MEDIA_PLAYER_STARTED |
46889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                MEDIA_PLAYER_PAUSED | MEDIA_PLAYER_PLAYBACK_COMPLETE ) ) {
46989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        // Can't change the stream type after prepare
47089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGE("setAudioStream called in state %d", mCurrentState);
47189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return INVALID_OPERATION;
47289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
47389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // cache
47489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mStreamType = type;
47589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return OK;
47689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
47789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
47889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setLooping(int loop)
47989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
48089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("MediaPlayer::setLooping");
48189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
48289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLoop = (loop != 0);
48389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer != 0) {
48489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return mPlayer->setLooping(loop);
48589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
48689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return OK;
48789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
48889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
48989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectbool MediaPlayer::isLooping() {
49089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("isLooping");
49189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
49289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer != 0) {
49389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return mLoop;
49489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
49589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("isLooping: no active player");
49689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return false;
49789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
49889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
49989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setVolume(float leftVolume, float rightVolume)
50089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
50189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("MediaPlayer::setVolume(%f, %f)", leftVolume, rightVolume);
50289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
50389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLeftVolume = leftVolume;
50489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mRightVolume = rightVolume;
50589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer != 0) {
50689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return mPlayer->setVolume(leftVolume, rightVolume);
50789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
50889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return OK;
50989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
51089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
511a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurentstatus_t MediaPlayer::setAudioSessionId(int sessionId)
512a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent{
513a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent    LOGV("MediaPlayer::setAudioSessionId(%d)", sessionId);
514a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent    Mutex::Autolock _l(mLock);
515a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent    if (!(mCurrentState & MEDIA_PLAYER_IDLE)) {
516a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent        LOGE("setAudioSessionId called in state %d", mCurrentState);
517a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent        return INVALID_OPERATION;
518a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent    }
519a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent    if (sessionId < 0) {
520a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent        return BAD_VALUE;
521a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent    }
522a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent    mAudioSessionId = sessionId;
523a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent    return NO_ERROR;
524a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent}
525a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent
526a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurentint MediaPlayer::getAudioSessionId()
527a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent{
528a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent    Mutex::Autolock _l(mLock);
529a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent    return mAudioSessionId;
530a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent}
531a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent
5322beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurentstatus_t MediaPlayer::setAuxEffectSendLevel(float level)
5332beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent{
5342beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    LOGV("MediaPlayer::setAuxEffectSendLevel(%f)", level);
5352beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    Mutex::Autolock _l(mLock);
5362beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    mSendLevel = level;
5372beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    if (mPlayer != 0) {
5382beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent        return mPlayer->setAuxEffectSendLevel(level);
5392beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    }
5402beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    return OK;
5412beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent}
5422beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent
5432beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurentstatus_t MediaPlayer::attachAuxEffect(int effectId)
5442beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent{
5452beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    LOGV("MediaPlayer::attachAuxEffect(%d)", effectId);
5462beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    Mutex::Autolock _l(mLock);
5472beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    if (mPlayer == 0 ||
5482beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent        (mCurrentState & MEDIA_PLAYER_IDLE) ||
5492beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent        (mCurrentState == MEDIA_PLAYER_STATE_ERROR )) {
5502beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent        LOGE("attachAuxEffect called in state %d", mCurrentState);
5512beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent        return INVALID_OPERATION;
5522beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    }
5532beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent
5542beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    return mPlayer->attachAuxEffect(effectId);
5552beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent}
5562beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent
55789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayer::notify(int msg, int ext1, int ext2)
55889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
55989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("message received msg=%d, ext1=%d, ext2=%d", msg, ext1, ext2);
56089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    bool send = true;
5611af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams    bool locked = false;
56289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
56389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // TODO: In the future, we might be on the same thread if the app is
56489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // running in the same process as the media server. In that case,
56589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // this will deadlock.
566660951867e959ebe98612742ef1f72d33ea7e9a3Nicolas Catania    //
5671af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams    // The threadId hack below works around this for the care of prepare
5685cb07aa071b43a214e4c880b3b7852714e06451bAndreas Huber    // and seekTo within the same process.
5695cb07aa071b43a214e4c880b3b7852714e06451bAndreas Huber    // FIXME: Remember, this is a hack, it's not even a hack that is applied
5705cb07aa071b43a214e4c880b3b7852714e06451bAndreas Huber    // consistently for all use-cases, this needs to be revisited.
5711af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams     if (mLockThreadId != getThreadId()) {
5721af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams        mLock.lock();
5731af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams        locked = true;
574660951867e959ebe98612742ef1f72d33ea7e9a3Nicolas Catania    }
5751af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams
5763b26844e60f8487388e7e62709faf0dada86e7e1Eric Laurent    // Allows calls from JNI in idle state to notify errors
5773b26844e60f8487388e7e62709faf0dada86e7e1Eric Laurent    if (!(msg == MEDIA_ERROR && mCurrentState == MEDIA_PLAYER_IDLE) && mPlayer == 0) {
57889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("notify(%d, %d, %d) callback on disconnected mediaplayer", msg, ext1, ext2);
5791af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams        if (locked) mLock.unlock();   // release the lock when done.
58089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return;
58189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
58289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
58389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    switch (msg) {
58489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    case MEDIA_NOP: // interface test message
58589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
58689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    case MEDIA_PREPARED:
58789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("prepared");
58889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mCurrentState = MEDIA_PLAYER_PREPARED;
58989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (mPrepareSync) {
59089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGV("signal application thread");
59189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mPrepareSync = false;
59289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mPrepareStatus = NO_ERROR;
59389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mSignal.signal();
59489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
59589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
59689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    case MEDIA_PLAYBACK_COMPLETE:
59789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("playback complete");
59889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (!mLoop) {
59989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_PLAYBACK_COMPLETE;
60089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
60189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
60289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    case MEDIA_ERROR:
60365e731f393f704eedab6fbe0af7f8a580c8d4617The Android Open Source Project        // Always log errors.
60465e731f393f704eedab6fbe0af7f8a580c8d4617The Android Open Source Project        // ext1: Media framework error code.
60565e731f393f704eedab6fbe0af7f8a580c8d4617The Android Open Source Project        // ext2: Implementation dependant error code.
60689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGE("error (%d, %d)", ext1, ext2);
60789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mCurrentState = MEDIA_PLAYER_STATE_ERROR;
60889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (mPrepareSync)
60989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        {
61089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGV("signal application thread");
61189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mPrepareSync = false;
61289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mPrepareStatus = ext1;
61389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mSignal.signal();
61489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            send = false;
61589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
61689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
61765e731f393f704eedab6fbe0af7f8a580c8d4617The Android Open Source Project    case MEDIA_INFO:
61865e731f393f704eedab6fbe0af7f8a580c8d4617The Android Open Source Project        // ext1: Media framework error code.
61965e731f393f704eedab6fbe0af7f8a580c8d4617The Android Open Source Project        // ext2: Implementation dependant error code.
62065e731f393f704eedab6fbe0af7f8a580c8d4617The Android Open Source Project        LOGW("info/warning (%d, %d)", ext1, ext2);
62165e731f393f704eedab6fbe0af7f8a580c8d4617The Android Open Source Project        break;
62289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    case MEDIA_SEEK_COMPLETE:
62389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("Received seek complete");
62489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (mSeekPosition != mCurrentPosition) {
62589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGV("Executing queued seekTo(%d)", mSeekPosition);
62689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mSeekPosition = -1;
62789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            seekTo_l(mCurrentPosition);
62889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
62989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        else {
63089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGV("All seeks complete - return to regularly scheduled program");
63189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentPosition = mSeekPosition = -1;
63289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
63389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
63489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    case MEDIA_BUFFERING_UPDATE:
63589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("buffering %d", ext1);
63689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
63789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    case MEDIA_SET_VIDEO_SIZE:
63889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("New video size %d x %d", ext1, ext2);
63989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mVideoWidth = ext1;
64089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mVideoHeight = ext2;
64189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
64289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    default:
64389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("unrecognized message: (%d, %d, %d)", msg, ext1, ext2);
64489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
64589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
64689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
64789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerListener> listener = mListener;
6481af452f333664e8b0a61d96a9b3bb682d8b9a00fJason Sams    if (locked) mLock.unlock();
64989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
65089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // this prevents re-entrant calls into client code
65189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if ((listener != 0) && send) {
65289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        Mutex::Autolock _l(mNotifyLock);
65389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("callback application");
65489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        listener->notify(msg, ext1, ext2);
65589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("back from callback");
65689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
65789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
65889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
65989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project/*static*/ sp<IMemory> MediaPlayer::decode(const char* url, uint32_t *pSampleRate, int* pNumChannels, int* pFormat)
66089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
66189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("decode(%s)", url);
66289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<IMemory> p;
66389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    const sp<IMediaPlayerService>& service = getMediaPlayerService();
66489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (service != 0) {
665dd172fce75b2a1c3cb3a5d3b3bbb5020b1ae8675James Dong        p = service->decode(url, pSampleRate, pNumChannels, pFormat);
66689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    } else {
66789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGE("Unable to locate media service");
66889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
66989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return p;
67089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
67189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
67289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
673dd172fce75b2a1c3cb3a5d3b3bbb5020b1ae8675James Dongvoid MediaPlayer::died()
674dd172fce75b2a1c3cb3a5d3b3bbb5020b1ae8675James Dong{
675dd172fce75b2a1c3cb3a5d3b3bbb5020b1ae8675James Dong    LOGV("died");
676dd172fce75b2a1c3cb3a5d3b3bbb5020b1ae8675James Dong    notify(MEDIA_ERROR, MEDIA_ERROR_SERVER_DIED, 0);
677dd172fce75b2a1c3cb3a5d3b3bbb5020b1ae8675James Dong}
678dd172fce75b2a1c3cb3a5d3b3bbb5020b1ae8675James Dong
67989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project/*static*/ sp<IMemory> MediaPlayer::decode(int fd, int64_t offset, int64_t length, uint32_t *pSampleRate, int* pNumChannels, int* pFormat)
68089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
68189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("decode(%d, %lld, %lld)", fd, offset, length);
68289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<IMemory> p;
68389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    const sp<IMediaPlayerService>& service = getMediaPlayerService();
68489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (service != 0) {
685dd172fce75b2a1c3cb3a5d3b3bbb5020b1ae8675James Dong        p = service->decode(fd, offset, length, pSampleRate, pNumChannels, pFormat);
68689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    } else {
68789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGE("Unable to locate media service");
68889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
68989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return p;
69089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
69189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
69289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
69389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}; // namespace android
694