mediaplayer.cpp revision 89fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5
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
2789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <utils/IServiceManager.h>
2889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <utils/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
3389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <utils/MemoryBase.h>
3489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
3589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectnamespace android {
3689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
3789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// client singleton for binder interface to service
3889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectMutex MediaPlayer::sServiceLock;
3989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectsp<IMediaPlayerService> MediaPlayer::sMediaPlayerService;
4089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectsp<MediaPlayer::DeathNotifier> MediaPlayer::sDeathNotifier;
4189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectSortedVector< wp<MediaPlayer> > MediaPlayer::sObitRecipients;
4289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
4389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// establish binder interface to service
4489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectconst sp<IMediaPlayerService>& MediaPlayer::getMediaPlayerService()
4589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
4689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(sServiceLock);
4789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (sMediaPlayerService.get() == 0) {
4889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        sp<IServiceManager> sm = defaultServiceManager();
4989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        sp<IBinder> binder;
5089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        do {
5189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            binder = sm->getService(String16("media.player"));
5289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            if (binder != 0)
5389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                break;
5489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGW("MediaPlayerService not published, waiting...");
5589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            usleep(500000); // 0.5 s
5689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } while(true);
5789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (sDeathNotifier == NULL) {
5889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            sDeathNotifier = new DeathNotifier();
5989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
6089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        binder->linkToDeath(sDeathNotifier);
6189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        sMediaPlayerService = interface_cast<IMediaPlayerService>(binder);
6289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
6389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGE_IF(sMediaPlayerService==0, "no MediaPlayerService!?");
6489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return sMediaPlayerService;
6589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
6689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
6789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayer::addObitRecipient(const wp<MediaPlayer>& recipient)
6889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
6989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(sServiceLock);
7089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sObitRecipients.add(recipient);
7189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
7289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
7389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayer::removeObitRecipient(const wp<MediaPlayer>& recipient)
7489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
7589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(sServiceLock);
7689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sObitRecipients.remove(recipient);
7789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
7889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
7989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectMediaPlayer::MediaPlayer()
8089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
8189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("constructor");
8289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mListener = NULL;
8389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mCookie = NULL;
8489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mDuration = -1;
8589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mStreamType = AudioSystem::MUSIC;
8689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mCurrentPosition = -1;
8789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mSeekPosition = -1;
8889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mCurrentState = MEDIA_PLAYER_IDLE;
8989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mPrepareSync = false;
9089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mPrepareStatus = NO_ERROR;
9189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLoop = false;
9289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLeftVolume = mRightVolume = 1.0;
9389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mVideoWidth = mVideoHeight = 0;
9489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
9589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
9689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayer::onFirstRef()
9789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
9889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    addObitRecipient(this);
9989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
10089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
10189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectMediaPlayer::~MediaPlayer()
10289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
10389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("destructor");
10489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    removeObitRecipient(this);
10589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    disconnect();
10689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    IPCThreadState::self()->flushCommands();
10789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
10889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
10989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayer::disconnect()
11089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
11189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("disconnect");
11289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<IMediaPlayer> p;
11389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    {
11489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        Mutex::Autolock _l(mLock);
11589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p = mPlayer;
11689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mPlayer.clear();
11789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
11889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
11989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p != 0) {
12089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p->disconnect();
12189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
12289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
12389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
12489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// always call with lock held
12589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayer::clear_l()
12689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
12789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mDuration = -1;
12889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mCurrentPosition = -1;
12989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mSeekPosition = -1;
13089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mVideoWidth = mVideoHeight = 0;
13189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
13289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
13389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setListener(const sp<MediaPlayerListener>& listener)
13489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
13589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("setListener");
13689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
13789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mListener = listener;
13889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
13989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
14089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
14189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
14289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setDataSource(const sp<IMediaPlayer>& player)
14389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
14489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    status_t err = UNKNOWN_ERROR;
14589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<IMediaPlayer> p;
14689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    { // scope for the lock
14789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        Mutex::Autolock _l(mLock);
14889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
14989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if ( !( mCurrentState & ( MEDIA_PLAYER_IDLE | MEDIA_PLAYER_STATE_ERROR ) ) ) {
15089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGE("setDataSource called in state %d", mCurrentState);
15189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            return INVALID_OPERATION;
15289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
15389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
15489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        clear_l();
15589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p = mPlayer;
15689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mPlayer = player;
15789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (player != 0) {
15889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_INITIALIZED;
15989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            err = NO_ERROR;
16089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else {
16189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGE("Unable to to create media player");
16289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
16389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
16489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
16589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p != 0) {
16689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p->disconnect();
16789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
16889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
16989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return err;
17089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
17189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
17289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setDataSource(const char *url)
17389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
17489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("setDataSource(%s)", url);
17589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    status_t err = BAD_VALUE;
17689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (url != NULL) {
17789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        const sp<IMediaPlayerService>& service(getMediaPlayerService());
17889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (service != 0) {
17989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            sp<IMediaPlayer> player(service->create(getpid(), this, url));
18089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            err = setDataSource(player);
18189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
18289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
18389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return err;
18489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
18589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
18689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setDataSource(int fd, int64_t offset, int64_t length)
18789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
18889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("setDataSource(%d, %lld, %lld)", fd, offset, length);
18989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    status_t err = UNKNOWN_ERROR;
19089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    const sp<IMediaPlayerService>& service(getMediaPlayerService());
19189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (service != 0) {
19289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        sp<IMediaPlayer> player(service->create(getpid(), this, fd, offset, length));
19389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        err = setDataSource(player);
19489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
19589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return err;
19689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
19789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
19889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setVideoSurface(const sp<Surface>& surface)
19989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
20089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("setVideoSurface");
20189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
20289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer == 0) return NO_INIT;
20389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return  mPlayer->setVideoSurface(surface->getISurface());
20489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
20589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
20689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// must call with lock held
20789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::prepareAsync_l()
20889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
20989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if ( (mPlayer != 0) && ( mCurrentState & ( MEDIA_PLAYER_INITIALIZED | MEDIA_PLAYER_STOPPED) ) ) {
21089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mPlayer->setAudioStreamType(mStreamType);
21189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mCurrentState = MEDIA_PLAYER_PREPARING;
21289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return mPlayer->prepareAsync();
21389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
21489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGE("prepareAsync called in state %d", mCurrentState);
21589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return INVALID_OPERATION;
21689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
21789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
21889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::prepare()
21989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
22089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("prepare");
22189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
22289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPrepareSync) return -EALREADY;
22389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mPrepareSync = true;
22489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    status_t ret = prepareAsync_l();
22589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (ret != NO_ERROR) return ret;
22689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
22789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPrepareSync) {
22889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mSignal.wait(mLock);  // wait for prepare done
22989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mPrepareSync = false;
23089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
23189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("prepare complete - status=%d", mPrepareStatus);
23289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return mPrepareStatus;
23389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
23489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
23589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::prepareAsync()
23689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
23789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("prepareAsync");
23889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
23989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return prepareAsync_l();
24089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
24189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
24289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::start()
24389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
24489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("start");
24589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
24689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mCurrentState & MEDIA_PLAYER_STARTED)
24789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return NO_ERROR;
24889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if ( (mPlayer != 0) && ( mCurrentState & ( MEDIA_PLAYER_PREPARED |
24989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                    MEDIA_PLAYER_PLAYBACK_COMPLETE | MEDIA_PLAYER_PAUSED ) ) ) {
25089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mPlayer->setLooping(mLoop);
25189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mPlayer->setVolume(mLeftVolume, mRightVolume);
25289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mCurrentState = MEDIA_PLAYER_STARTED;
25389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        status_t ret = mPlayer->start();
25489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (ret != NO_ERROR) {
25589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_STATE_ERROR;
25689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else {
25789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            if (mCurrentState == MEDIA_PLAYER_PLAYBACK_COMPLETE) {
25889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                LOGV("playback completed immediately following start()");
25989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            }
26089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
26189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return ret;
26289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
26389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGE("start called in state %d", mCurrentState);
26489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return INVALID_OPERATION;
26589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
26689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
26789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::stop()
26889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
26989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("stop");
27089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
27189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mCurrentState & MEDIA_PLAYER_STOPPED) return NO_ERROR;
27289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if ( (mPlayer != 0) && ( mCurrentState & ( MEDIA_PLAYER_STARTED | MEDIA_PLAYER_PREPARED |
27389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                    MEDIA_PLAYER_PAUSED | MEDIA_PLAYER_PLAYBACK_COMPLETE ) ) ) {
27489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        status_t ret = mPlayer->stop();
27589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (ret != NO_ERROR) {
27689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_STATE_ERROR;
27789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else {
27889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_STOPPED;
27989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
28089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return ret;
28189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
28289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGE("stop called in state %d", mCurrentState);
28389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return INVALID_OPERATION;
28489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
28589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
28689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::pause()
28789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
28889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("pause");
28989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
29089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mCurrentState & MEDIA_PLAYER_PAUSED)
29189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return NO_ERROR;
29289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if ((mPlayer != 0) && (mCurrentState & MEDIA_PLAYER_STARTED)) {
29389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        status_t ret = mPlayer->pause();
29489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (ret != NO_ERROR) {
29589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_STATE_ERROR;
29689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else {
29789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_PAUSED;
29889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
29989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return ret;
30089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
30189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGE("pause called in state %d", mCurrentState);
30289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return INVALID_OPERATION;
30389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
30489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
30589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectbool MediaPlayer::isPlaying()
30689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
30789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
30889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer != 0) {
30989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        bool temp = false;
31089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mPlayer->isPlaying(&temp);
31189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("isPlaying: %d", temp);
31289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if ((mCurrentState & MEDIA_PLAYER_STARTED) && ! temp) {
31389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGE("internal/external state mismatch corrected");
31489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_PAUSED;
31589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
31689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return temp;
31789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
31889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("isPlaying: no active player");
31989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return false;
32089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
32189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
32289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::getVideoWidth(int *w)
32389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
32489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("getVideoWidth");
32589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
32689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer == 0) return INVALID_OPERATION;
32789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    *w = mVideoWidth;
32889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
32989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
33089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
33189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::getVideoHeight(int *h)
33289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
33389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("getVideoHeight");
33489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
33589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer == 0) return INVALID_OPERATION;
33689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    *h = mVideoHeight;
33789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
33889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
33989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
34089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::getCurrentPosition(int *msec)
34189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
34289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("getCurrentPosition");
34389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
34489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer != 0) {
34589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (mCurrentPosition >= 0) {
34689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGV("Using cached seek position: %d", mCurrentPosition);
34789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            *msec = mCurrentPosition;
34889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            return NO_ERROR;
34989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
35089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return mPlayer->getCurrentPosition(msec);
35189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
35289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return INVALID_OPERATION;
35389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
35489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
35589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::getDuration_l(int *msec)
35689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
35789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("getDuration");
35889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    bool isValidState = (mCurrentState & (MEDIA_PLAYER_PREPARED | MEDIA_PLAYER_STARTED | MEDIA_PLAYER_PAUSED | MEDIA_PLAYER_STOPPED | MEDIA_PLAYER_PLAYBACK_COMPLETE));
35989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer != 0 && isValidState) {
36089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        status_t ret = NO_ERROR;
36189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (mDuration <= 0)
36289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            ret = mPlayer->getDuration(&mDuration);
36389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (msec)
36489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            *msec = mDuration;
36589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return ret;
36689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
36789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGE("Attempt to call getDuration without a valid mediaplayer");
36889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return INVALID_OPERATION;
36989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
37089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
37189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::getDuration(int *msec)
37289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
37389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
37489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return getDuration_l(msec);
37589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
37689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
37789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::seekTo_l(int msec)
37889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
37989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("seekTo %d", msec);
38089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if ((mPlayer != 0) && ( mCurrentState & ( MEDIA_PLAYER_STARTED | MEDIA_PLAYER_PREPARED | MEDIA_PLAYER_PAUSED |  MEDIA_PLAYER_PLAYBACK_COMPLETE) ) ) {
38189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if ( msec < 0 ) {
38289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGW("Attempt to seek to invalid position: %d", msec);
38389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            msec = 0;
38489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else if ((mDuration > 0) && (msec > mDuration)) {
38589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGW("Attempt to seek to past end of file: request = %d, EOF = %d", msec, mDuration);
38689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            msec = mDuration;
38789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
38889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        // cache duration
38989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mCurrentPosition = msec;
39089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (mSeekPosition < 0) {
39189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            getDuration_l(NULL);
39289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mSeekPosition = msec;
39389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            return mPlayer->seekTo(msec);
39489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
39589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        else {
39689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGV("Seek in progress - queue up seekTo[%d]", msec);
39789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            return NO_ERROR;
39889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
39989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
40089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGE("Attempt to perform seekTo in wrong state: mPlayer=%p, mCurrentState=%u", mPlayer.get(), mCurrentState);
40189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return INVALID_OPERATION;
40289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
40389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
40489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::seekTo(int msec)
40589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
40689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
40789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return seekTo_l(msec);
40889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
40989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
41089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::reset()
41189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
41289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("reset");
41389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
41489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLoop = false;
41589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mCurrentState == MEDIA_PLAYER_IDLE) return NO_ERROR;
41689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mPrepareSync = false;
41789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer != 0) {
41889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        status_t ret = mPlayer->reset();
41989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (ret != NO_ERROR) {
42089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGE("reset() failed with return code (%d)", ret);
42189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_STATE_ERROR;
42289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else {
42389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_IDLE;
42489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
42589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return ret;
42689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
42789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    clear_l();
42889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
42989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
43089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
43189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setAudioStreamType(int type)
43289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
43389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("MediaPlayer::setAudioStreamType");
43489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
43589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mStreamType == type) return NO_ERROR;
43689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mCurrentState & ( MEDIA_PLAYER_PREPARED | MEDIA_PLAYER_STARTED |
43789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                MEDIA_PLAYER_PAUSED | MEDIA_PLAYER_PLAYBACK_COMPLETE ) ) {
43889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        // Can't change the stream type after prepare
43989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGE("setAudioStream called in state %d", mCurrentState);
44089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return INVALID_OPERATION;
44189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
44289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // cache
44389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mStreamType = type;
44489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return OK;
44589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
44689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
44789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setLooping(int loop)
44889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
44989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("MediaPlayer::setLooping");
45089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
45189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLoop = (loop != 0);
45289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer != 0) {
45389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return mPlayer->setLooping(loop);
45489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
45589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return OK;
45689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
45789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
45889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectbool MediaPlayer::isLooping() {
45989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("isLooping");
46089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
46189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer != 0) {
46289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return mLoop;
46389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
46489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("isLooping: no active player");
46589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return false;
46689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
46789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
46889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayer::setVolume(float leftVolume, float rightVolume)
46989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
47089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("MediaPlayer::setVolume(%f, %f)", leftVolume, rightVolume);
47189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
47289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLeftVolume = leftVolume;
47389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mRightVolume = rightVolume;
47489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer != 0) {
47589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return mPlayer->setVolume(leftVolume, rightVolume);
47689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
47789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return OK;
47889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
47989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
48089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayer::notify(int msg, int ext1, int ext2)
48189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
48289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("message received msg=%d, ext1=%d, ext2=%d", msg, ext1, ext2);
48389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    bool send = true;
48489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
48589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // TODO: In the future, we might be on the same thread if the app is
48689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // running in the same process as the media server. In that case,
48789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // this will deadlock.
48889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLock.lock();
48989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mPlayer == 0) {
49089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("notify(%d, %d, %d) callback on disconnected mediaplayer", msg, ext1, ext2);
49189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mLock.unlock();   // release the lock when done.
49289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return;
49389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
49489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
49589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    switch (msg) {
49689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    case MEDIA_NOP: // interface test message
49789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
49889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    case MEDIA_PREPARED:
49989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("prepared");
50089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mCurrentState = MEDIA_PLAYER_PREPARED;
50189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (mPrepareSync) {
50289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGV("signal application thread");
50389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mPrepareSync = false;
50489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mPrepareStatus = NO_ERROR;
50589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mSignal.signal();
50689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
50789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
50889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    case MEDIA_PLAYBACK_COMPLETE:
50989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("playback complete");
51089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (!mLoop) {
51189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentState = MEDIA_PLAYER_PLAYBACK_COMPLETE;
51289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
51389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
51489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    case MEDIA_ERROR:
51589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        // Always log errors
51689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGE("error (%d, %d)", ext1, ext2);
51789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mCurrentState = MEDIA_PLAYER_STATE_ERROR;
51889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (mPrepareSync)
51989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        {
52089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGV("signal application thread");
52189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mPrepareSync = false;
52289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mPrepareStatus = ext1;
52389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mSignal.signal();
52489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            send = false;
52589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
52689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
52789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    case MEDIA_SEEK_COMPLETE:
52889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("Received seek complete");
52989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (mSeekPosition != mCurrentPosition) {
53089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGV("Executing queued seekTo(%d)", mSeekPosition);
53189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mSeekPosition = -1;
53289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            seekTo_l(mCurrentPosition);
53389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
53489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        else {
53589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            LOGV("All seeks complete - return to regularly scheduled program");
53689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mCurrentPosition = mSeekPosition = -1;
53789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
53889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
53989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    case MEDIA_BUFFERING_UPDATE:
54089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("buffering %d", ext1);
54189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
54289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    case MEDIA_SET_VIDEO_SIZE:
54389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("New video size %d x %d", ext1, ext2);
54489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mVideoWidth = ext1;
54589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mVideoHeight = ext2;
54689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
54789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    default:
54889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("unrecognized message: (%d, %d, %d)", msg, ext1, ext2);
54989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        break;
55089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
55189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
55289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerListener> listener = mListener;
55389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLock.unlock();
55489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
55589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // this prevents re-entrant calls into client code
55689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if ((listener != 0) && send) {
55789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        Mutex::Autolock _l(mNotifyLock);
55889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("callback application");
55989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        listener->notify(msg, ext1, ext2);
56089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGV("back from callback");
56189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
56289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
56389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
56489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayer::DeathNotifier::binderDied(const wp<IBinder>& who) {
56589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGW("MediaPlayer server died!");
56689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
56789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // Need to do this with the lock held
56889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    SortedVector< wp<MediaPlayer> > list;
56989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    {
57089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        Mutex::Autolock _l(MediaPlayer::sServiceLock);
57189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        MediaPlayer::sMediaPlayerService.clear();
57289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        list = sObitRecipients;
57389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
57489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
57589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // Notify application when media server dies.
57689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // Don't hold the static lock during callback in case app
57789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // makes a call that needs the lock.
57889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    size_t count = list.size();
57989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    for (size_t iter = 0; iter < count; ++iter) {
58089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        sp<MediaPlayer> player = list[iter].promote();
58189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if ((player != 0) && (player->mPlayer != 0)) {
58289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            player->notify(MEDIA_ERROR, MEDIA_ERROR_SERVER_DIED, 0);
58389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
58489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
58589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
58689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
58789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectMediaPlayer::DeathNotifier::~DeathNotifier()
58889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
58989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(sServiceLock);
59089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sObitRecipients.clear();
59189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (sMediaPlayerService != 0) {
59289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        sMediaPlayerService->asBinder()->unlinkToDeath(this);
59389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
59489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
59589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
59689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project/*static*/ sp<IMemory> MediaPlayer::decode(const char* url, uint32_t *pSampleRate, int* pNumChannels, int* pFormat)
59789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
59889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("decode(%s)", url);
59989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<IMemory> p;
60089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    const sp<IMediaPlayerService>& service = getMediaPlayerService();
60189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (service != 0) {
60289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p = sMediaPlayerService->decode(url, pSampleRate, pNumChannels, pFormat);
60389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    } else {
60489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGE("Unable to locate media service");
60589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
60689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return p;
60789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
60889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
60989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
61089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project/*static*/ sp<IMemory> MediaPlayer::decode(int fd, int64_t offset, int64_t length, uint32_t *pSampleRate, int* pNumChannels, int* pFormat)
61189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
61289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    LOGV("decode(%d, %lld, %lld)", fd, offset, length);
61389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<IMemory> p;
61489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    const sp<IMediaPlayerService>& service = getMediaPlayerService();
61589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (service != 0) {
61689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p = sMediaPlayerService->decode(fd, offset, length, pSampleRate, pNumChannels, pFormat);
61789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    } else {
61889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        LOGE("Unable to locate media service");
61989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
62089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return p;
62189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
62289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
62389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
62489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}; // namespace android
625