MediaPlayer_to_android.cpp revision 72042d4448cee63528c619537321ba73944c6382
1eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi/*
2eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi * Copyright (C) 2010 The Android Open Source Project
3eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi *
4eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi * Licensed under the Apache License, Version 2.0 (the "License");
5eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi * you may not use this file except in compliance with the License.
6eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi * You may obtain a copy of the License at
7eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi *
8eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi *      http://www.apache.org/licenses/LICENSE-2.0
9eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi *
10eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi * Unless required by applicable law or agreed to in writing, software
11eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi * distributed under the License is distributed on an "AS IS" BASIS,
12eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi * See the License for the specific language governing permissions and
14eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi * limitations under the License.
15eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi */
16eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
17eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi#include "sles_allinclusive.h"
18eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi#include "utils/RefBase.h"
19eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi#include "android_prompts.h"
202b06e20ae32388f6e1dfd088d9773c34e6b1cb45Jean-Michel Trivi// LocAVPlayer and StreamPlayer derive from GenericMediaPlayer,
212b06e20ae32388f6e1dfd088d9773c34e6b1cb45Jean-Michel Trivi//    so no need to #include "android_GenericMediaPlayer.h"
222b06e20ae32388f6e1dfd088d9773c34e6b1cb45Jean-Michel Trivi#include "android_LocAVPlayer.h"
232b06e20ae32388f6e1dfd088d9773c34e6b1cb45Jean-Michel Trivi#include "android_StreamPlayer.h"
24eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
25e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi
26581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi//-----------------------------------------------------------------------------
2737dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivistatic void player_handleMediaPlayerEventNotifications(int event, int data1, int data2, void* user)
28581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi{
29c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
30c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten    // FIXME This code is derived from similar code in sfplayer_handlePrefetchEvent.  The two
31c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten    // versions are quite similar, but still different enough that they need to be separate.
32c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten    // At some point they should be re-factored and merged if feasible.
33c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten    // As with other OpenMAX AL implementation code, this copy mostly uses SL_ symbols
34c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten    // rather than XA_ unless the difference is significant.
35c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
36581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi    if (NULL == user) {
37581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi        return;
38581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi    }
39581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi
40581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi    CMediaPlayer* mp = (CMediaPlayer*) user;
419d8a98601357c0669dca4de63e43196c0a70553dGlenn Kasten    if (!android::CallbackProtector::enterCbIfOk(mp->mCallbackProtector)) {
429d8a98601357c0669dca4de63e43196c0a70553dGlenn Kasten        // it is not safe to enter the callback (the media player is about to go away)
439d8a98601357c0669dca4de63e43196c0a70553dGlenn Kasten        return;
449d8a98601357c0669dca4de63e43196c0a70553dGlenn Kasten    }
45c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten    union {
46c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        char c[sizeof(int)];
47c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        int i;
48c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten    } u;
49c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten    u.i = event;
50c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten    SL_LOGV("player_handleMediaPlayerEventNotifications(event='%c%c%c%c' (%d), data1=%d, data2=%d, "
51c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            "user=%p) from AVPlayer", u.c[3], u.c[2], u.c[1], u.c[0], event, data1, data2, user);
52581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi    switch(event) {
53581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi
5437dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi      case android::GenericPlayer::kEventPrepared: {
557f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten        SL_LOGV("Received GenericPlayer::kEventPrepared for CMediaPlayer %p", mp);
56c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
57c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        // assume no callback
58c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        slPrefetchCallback callback = NULL;
597f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten        void* callbackPContext;
607f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten        XAuint32 events;
61c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
62c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        object_lock_exclusive(&mp->mObject);
637f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten
647f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten        // mark object as prepared; same state is used for successful or unsuccessful prepare
657f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten        assert(mp->mAndroidObjState == ANDROID_PREPARING);
66c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        mp->mAndroidObjState = ANDROID_READY;
67c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
687f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten        if (PLAYER_SUCCESS == data1) {
697f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten            // Most of successful prepare completion for mp->mAVPlayer
707f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten            // is handled by GenericPlayer and its subclasses.
717f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten        } else {
727f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten            // AVPlayer prepare() failed prefetching, there is no event in XAPrefetchStatus to
737f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten            //  indicate a prefetch error, so we signal it by sending simultaneously two events:
747f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten            //  - SL_PREFETCHEVENT_FILLLEVELCHANGE with a level of 0
757f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten            //  - SL_PREFETCHEVENT_STATUSCHANGE with a status of SL_PREFETCHSTATUS_UNDERFLOW
767f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten            SL_LOGE(ERROR_PLAYER_PREFETCH_d, data1);
777f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten            if (IsInterfaceInitialized(&mp->mObject, MPH_XAPREFETCHSTATUS)) {
787f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten                mp->mPrefetchStatus.mLevel = 0;
797f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten                mp->mPrefetchStatus.mStatus = SL_PREFETCHSTATUS_UNDERFLOW;
807f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten                if (!(~mp->mPrefetchStatus.mCallbackEventsMask &
817f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten                        (SL_PREFETCHEVENT_FILLLEVELCHANGE | SL_PREFETCHEVENT_STATUSCHANGE))) {
827f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten                    callback = mp->mPrefetchStatus.mCallback;
837f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten                    callbackPContext = mp->mPrefetchStatus.mContext;
847f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten                    events = SL_PREFETCHEVENT_FILLLEVELCHANGE | SL_PREFETCHEVENT_STATUSCHANGE;
857f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten                }
86c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            }
87581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi        }
887f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten
89c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        object_unlock_exclusive(&mp->mObject);
90c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
91c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        // callback with no lock held
92c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (NULL != callback) {
937f250a17c145382b866d5d4d7ef23d65fada6236Glenn Kasten            (*callback)(&mp->mPrefetchStatus.mItf, callbackPContext, events);
94581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi        }
95c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
9637dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        break;
9737dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi      }
9837dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
9937dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi      case android::GenericPlayer::kEventHasVideoSize: {
10037dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        SL_LOGV("Received AVPlayer::kEventHasVideoSize (%d,%d) for CMediaPlayer %p",
10137dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi                data1, data2, mp);
10237dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
10337dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        object_lock_exclusive(&mp->mObject);
10437dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
10537dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        // remove an existing video info entry (here we only have one video stream)
10637dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        for(size_t i=0 ; i < mp->mStreamInfo.mStreamInfoTable.size() ; i++) {
10737dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi            if (XA_DOMAINTYPE_VIDEO == mp->mStreamInfo.mStreamInfoTable.itemAt(i).domain) {
10837dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi                mp->mStreamInfo.mStreamInfoTable.removeAt(i);
10937dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi                break;
11037dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi            }
11137dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        }
11237dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        // update the stream information with a new video info entry
11337dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        StreamInfo streamInfo;
11437dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        streamInfo.domain = XA_DOMAINTYPE_VIDEO;
11537dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        streamInfo.videoInfo.codecId = 0;// unknown, we don't have that info FIXME
11637dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        streamInfo.videoInfo.width = (XAuint32)data1;
11737dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        streamInfo.videoInfo.height = (XAuint32)data2;
11837dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        streamInfo.videoInfo.bitRate = 0;// unknown, we don't have that info FIXME
119cb441acdda6f8e81d44fcdaadd4ff7ab3d3e367bGlenn Kasten        streamInfo.videoInfo.frameRate = 0;
12037dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        streamInfo.videoInfo.duration = XA_TIME_UNKNOWN;
12137dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        StreamInfo &contInfo = mp->mStreamInfo.mStreamInfoTable.editItemAt(0);
12237dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        contInfo.containerInfo.numStreams = 1;
12337dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        ssize_t index = mp->mStreamInfo.mStreamInfoTable.add(streamInfo);
12437dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
125e2e8fa36bd7448b59fbcdf141e0b6d21e5401d91Glenn Kasten        // callback is unconditional; there is no bitmask of enabled events
12637dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        xaStreamEventChangeCallback callback = mp->mStreamInfo.mCallback;
12737dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        void* callbackPContext = mp->mStreamInfo.mContext;
12837dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
12937dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        object_unlock_exclusive(&mp->mObject);
13037dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
13185133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi        // enqueue notification (outside of lock) that the stream information has been updated
13237dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        if ((NULL != callback) && (index >= 0)) {
133dd177e2d3923d4653eaa4226f07b89a999907970Glenn Kasten#ifndef USE_ASYNCHRONOUS_STREAMCBEVENT_PROPERTYCHANGE_CALLBACK
13437dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi            (*callback)(&mp->mStreamInfo.mItf, XA_STREAMCBEVENT_PROPERTYCHANGE /*eventId*/,
13537dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi                    1 /*streamIndex, only one stream supported here, 0 is reserved*/,
13637dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi                    NULL /*pEventData, always NULL in OpenMAX AL 1.0.1*/,
13737dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi                    callbackPContext /*pContext*/);
13885133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi#else
13985133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi            SLresult res = EnqueueAsyncCallback_piipp(mp, callback,
14085133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi                    /*p1*/ &mp->mStreamInfo.mItf,
14185133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi                    /*i1*/ XA_STREAMCBEVENT_PROPERTYCHANGE /*eventId*/,
14285133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi                    /*i2*/ 1 /*streamIndex, only one stream supported here, 0 is reserved*/,
14385133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi                    /*p2*/ NULL /*pEventData, always NULL in OpenMAX AL 1.0.1*/,
14485133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi                    /*p3*/ callbackPContext /*pContext*/);
145dd177e2d3923d4653eaa4226f07b89a999907970Glenn Kasten            LOGW_IF(SL_RESULT_SUCCESS != res,
146dd177e2d3923d4653eaa4226f07b89a999907970Glenn Kasten                        "Callback %p(%p, XA_STREAMCBEVENT_PROPERTYCHANGE, 1, NULL, %p) dropped",
147dd177e2d3923d4653eaa4226f07b89a999907970Glenn Kasten                        callback, &mp->mStreamInfo.mItf, callbackPContext);
14885133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi#endif
149581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi        }
150581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi        break;
15137dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi      }
152581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi
15392b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      case android::GenericPlayer::kEventEndOfStream: {
15492b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        SL_LOGV("Received AVPlayer::kEventEndOfStream for CMediaPlayer %p", mp);
15592b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi
15692b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        object_lock_exclusive(&mp->mObject);
15792b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        // should be xaPlayCallback but we're sharing the itf between SL and AL
15892b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        slPlayCallback playCallback = NULL;
15992b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        void * playContext = NULL;
16092b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        // XAPlayItf callback or no callback?
16192b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        if (mp->mPlay.mEventFlags & XA_PLAYEVENT_HEADATEND) {
16292b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi            playCallback = mp->mPlay.mCallback;
16392b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi            playContext = mp->mPlay.mContext;
16492b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        }
165e37c62f0691f9a137afae60f9d6dbd1a65d36aedGlenn Kasten        mp->mPlay.mState = XA_PLAYSTATE_PAUSED;
16692b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        object_unlock_exclusive(&mp->mObject);
16792b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi
16885133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi        // enqueue callback with no lock held
16992b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        if (NULL != playCallback) {
170dd177e2d3923d4653eaa4226f07b89a999907970Glenn Kasten#ifndef USE_ASYNCHRONOUS_PLAY_CALLBACK
17192b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi            (*playCallback)(&mp->mPlay.mItf, playContext, XA_PLAYEVENT_HEADATEND);
17285133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi#else
17385133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi            SLresult res = EnqueueAsyncCallback_ppi(mp, playCallback, &mp->mPlay.mItf, playContext,
17485133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi                    XA_PLAYEVENT_HEADATEND);
17585133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi            LOGW_IF(SL_RESULT_SUCCESS != res,
176c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten                    "Callback %p(%p, %p, SL_PLAYEVENT_HEADATEND) dropped", playCallback,
17785133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi                    &mp->mPlay.mItf, playContext);
17885133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi#endif
17992b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        }
180581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi        break;
18192b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      }
18292b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi
18399b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      case android::GenericPlayer::kEventChannelCount: {
18499b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        SL_LOGV("kEventChannelCount channels = %d", data1);
18599b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        object_lock_exclusive(&mp->mObject);
18699b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        if (UNKNOWN_NUMCHANNELS == mp->mNumChannels && UNKNOWN_NUMCHANNELS != data1) {
18799b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten            mp->mNumChannels = data1;
18899b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten            android_Player_volumeUpdate(mp);
18999b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        }
19099b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        object_unlock_exclusive(&mp->mObject);
19199b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      }
19299b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      break;
19399b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten
19499b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      case android::GenericPlayer::kEventPrefetchFillLevelUpdate: {
19599b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        SL_LOGV("kEventPrefetchFillLevelUpdate");
196c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (!IsInterfaceInitialized(&mp->mObject, MPH_XAPREFETCHSTATUS)) {
197c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            break;
198c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        }
199c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        slPrefetchCallback callback = NULL;
200c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        void* callbackPContext = NULL;
201c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
202c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        // SLPrefetchStatusItf callback or no callback?
203c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        interface_lock_exclusive(&mp->mPrefetchStatus);
204c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (mp->mPrefetchStatus.mCallbackEventsMask & SL_PREFETCHEVENT_FILLLEVELCHANGE) {
205c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            callback = mp->mPrefetchStatus.mCallback;
206c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            callbackPContext = mp->mPrefetchStatus.mContext;
207c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        }
208c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        mp->mPrefetchStatus.mLevel = (SLpermille)data1;
209c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        interface_unlock_exclusive(&mp->mPrefetchStatus);
210c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
211c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        // callback with no lock held
212c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (NULL != callback) {
213c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            (*callback)(&mp->mPrefetchStatus.mItf, callbackPContext,
214c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten                    SL_PREFETCHEVENT_FILLLEVELCHANGE);
215c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        }
21699b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      }
21799b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      break;
21899b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten
21999b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      case android::GenericPlayer::kEventPrefetchStatusChange: {
22099b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        SL_LOGV("kEventPrefetchStatusChange");
221c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (!IsInterfaceInitialized(&mp->mObject, MPH_XAPREFETCHSTATUS)) {
222c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            break;
223c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        }
224c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        slPrefetchCallback callback = NULL;
225c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        void* callbackPContext = NULL;
226c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
227c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        // SLPrefetchStatusItf callback or no callback?
228c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        object_lock_exclusive(&mp->mObject);
229c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (mp->mPrefetchStatus.mCallbackEventsMask & SL_PREFETCHEVENT_STATUSCHANGE) {
230c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            callback = mp->mPrefetchStatus.mCallback;
231c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            callbackPContext = mp->mPrefetchStatus.mContext;
232c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        }
233c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (data1 >= android::kStatusIntermediate) {
234c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            mp->mPrefetchStatus.mStatus = SL_PREFETCHSTATUS_SUFFICIENTDATA;
235c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        } else if (data1 < android::kStatusIntermediate) {
236c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            mp->mPrefetchStatus.mStatus = SL_PREFETCHSTATUS_UNDERFLOW;
237c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        }
238c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        object_unlock_exclusive(&mp->mObject);
239c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
240c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        // callback with no lock held
241c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (NULL != callback) {
242c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            (*callback)(&mp->mPrefetchStatus.mItf, callbackPContext, SL_PREFETCHEVENT_STATUSCHANGE);
243c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        }
24499b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      }
24599b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      break;
24699b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten
247bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten      case android::GenericPlayer::kEventPlay: {
248bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        SL_LOGV("kEventPlay");
249bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten
250bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        interface_lock_shared(&mp->mPlay);
251bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        slPlayCallback callback = mp->mPlay.mCallback;
252bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        void* callbackPContext = mp->mPlay.mContext;
253bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        interface_unlock_shared(&mp->mPlay);
254bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten
255bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        if (NULL != callback) {
256bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten            (*callback)(&mp->mPlay.mItf, callbackPContext, (SLuint32) data1); // SL_PLAYEVENT_HEAD*
257bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        }
258bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten      }
259bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten      break;
26099b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten
2615e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten      case android::GenericPlayer::kEventErrorAfterPrepare: {
2625e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        SL_LOGV("kEventErrorAfterPrepare");
2635e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten
2645e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        // assume no callback
2655e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        slPrefetchCallback callback = NULL;
2665e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        void* callbackPContext = NULL;
2675e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten
2685e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        object_lock_exclusive(&mp->mObject);
2695e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        if (IsInterfaceInitialized(&mp->mObject, MPH_XAPREFETCHSTATUS)) {
2705e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten            mp->mPrefetchStatus.mLevel = 0;
2715e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten            mp->mPrefetchStatus.mStatus = SL_PREFETCHSTATUS_UNDERFLOW;
2725e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten            if (!(~mp->mPrefetchStatus.mCallbackEventsMask &
2735e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten                    (SL_PREFETCHEVENT_FILLLEVELCHANGE | SL_PREFETCHEVENT_STATUSCHANGE))) {
2745e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten                callback = mp->mPrefetchStatus.mCallback;
2755e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten                callbackPContext = mp->mPrefetchStatus.mContext;
2765e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten            }
2775e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        }
2785e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        object_unlock_exclusive(&mp->mObject);
2795e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten
2805e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        // FIXME there's interesting information in data1, but no API to convey it to client
2815e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        SL_LOGE("Error after prepare: %d", data1);
2825e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten
2835e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        // callback with no lock held
2845e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        if (NULL != callback) {
2855e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten            (*callback)(&mp->mPrefetchStatus.mItf, callbackPContext,
2865e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten                    SL_PREFETCHEVENT_FILLLEVELCHANGE | SL_PREFETCHEVENT_STATUSCHANGE);
2875e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        }
2885e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten
2895e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten      }
2905e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten      break;
2915e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten
29292b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      default: {
29392b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        SL_LOGE("Received unknown event %d, data %d from AVPlayer", event, data1);
29492b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      }
295581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi    }
2969d8a98601357c0669dca4de63e43196c0a70553dGlenn Kasten
2979d8a98601357c0669dca4de63e43196c0a70553dGlenn Kasten    mp->mCallbackProtector->exitCb();
298581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi}
299581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi
300eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
301eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi//-----------------------------------------------------------------------------
302d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel TriviXAresult android_Player_checkSourceSink(CMediaPlayer *mp) {
303d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
304d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    XAresult result = XA_RESULT_SUCCESS;
305d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
306d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    const SLDataSource *pSrc    = &mp->mDataSource.u.mSource;
307d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    const SLDataSink *pAudioSnk = &mp->mAudioSink.u.mSink;
308d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
309d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    // format check:
310d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    const SLuint32 sourceLocatorType = *(SLuint32 *)pSrc->pLocator;
311d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    const SLuint32 sourceFormatType  = *(SLuint32 *)pSrc->pFormat;
312d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    const SLuint32 audioSinkLocatorType = *(SLuint32 *)pAudioSnk->pLocator;
313d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    //const SLuint32 sinkFormatType = *(SLuint32 *)pAudioSnk->pFormat;
314d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
315d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    // Source check
316d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    switch(sourceLocatorType) {
317d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
318d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    case XA_DATALOCATOR_ANDROIDBUFFERQUEUE: {
319d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        switch (sourceFormatType) {
320d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        case XA_DATAFORMAT_MIME: {
321d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi            SLDataFormat_MIME *df_mime = (SLDataFormat_MIME *) pSrc->pFormat;
322d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi            if (SL_CONTAINERTYPE_MPEG_TS != df_mime->containerType) {
323d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi                SL_LOGE("Cannot create player with XA_DATALOCATOR_ANDROIDBUFFERQUEUE data source "
324d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi                        "that is not fed MPEG-2 TS data");
325d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi                return SL_RESULT_CONTENT_UNSUPPORTED;
326d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi            }
327d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        } break;
328d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        default:
329d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi            SL_LOGE("Cannot create player with XA_DATALOCATOR_ANDROIDBUFFERQUEUE data source "
330d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi                    "without SL_DATAFORMAT_MIME format");
331d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi            return XA_RESULT_CONTENT_UNSUPPORTED;
332d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        }
333d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    } break;
334d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
335d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    case XA_DATALOCATOR_URI: // intended fall-through
336d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    case XA_DATALOCATOR_ANDROIDFD:
337d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        break;
338d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
339d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    default:
340d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        SL_LOGE("Cannot create media player with data locator type 0x%x",
341d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi                (unsigned) sourceLocatorType);
342d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        return SL_RESULT_PARAMETER_INVALID;
343d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    }// switch (locatorType)
344d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
345d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    // Audio sink check: only playback is supported here
346d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    switch(audioSinkLocatorType) {
347d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
348d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    case XA_DATALOCATOR_OUTPUTMIX:
349d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        break;
350d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
351d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    default:
352d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        SL_LOGE("Cannot create media player with audio sink data locator of type 0x%x",
353d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi                (unsigned) audioSinkLocatorType);
354d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        return XA_RESULT_PARAMETER_INVALID;
355d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    }// switch (locaaudioSinkLocatorTypeorType)
356d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
357d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    return result;
358d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi}
359d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
360d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
361d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi//-----------------------------------------------------------------------------
362eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel TriviXAresult android_Player_create(CMediaPlayer *mp) {
363eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
364eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    XAresult result = XA_RESULT_SUCCESS;
365eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
366eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    // FIXME verify data source
367eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    const SLDataSource *pDataSrc = &mp->mDataSource.u.mSource;
368eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    // FIXME verify audio data sink
369eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    const SLDataSink *pAudioSnk = &mp->mAudioSink.u.mSink;
370eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    // FIXME verify image data sink
371eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    const SLDataSink *pVideoSnk = &mp->mImageVideoSink.u.mSink;
372eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
37397876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi    XAuint32 sourceLocator = *(XAuint32 *)pDataSrc->pLocator;
374eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    switch(sourceLocator) {
37597876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi    // FIXME support Android simple buffer queue as well
376f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    case XA_DATALOCATOR_ANDROIDBUFFERQUEUE:
377b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi        mp->mAndroidObjType = AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE;
378eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi        break;
379f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    case XA_DATALOCATOR_URI: // intended fall-through
38097876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi    case SL_DATALOCATOR_ANDROIDFD:
381b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi        mp->mAndroidObjType = AUDIOVIDEOPLAYER_FROM_URIFD;
38297876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        break;
383f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    case XA_DATALOCATOR_ADDRESS: // intended fall-through
384eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    default:
38572042d4448cee63528c619537321ba73944c6382Glenn Kasten        mp->mAndroidObjType = INVALID_TYPE;
386a8179ea15c4ff78db589d742b135649f0eda7ef2Glenn Kasten        SL_LOGE("Unable to create MediaPlayer for data source locator 0x%x", sourceLocator);
387eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi        result = XA_RESULT_PARAMETER_INVALID;
388eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi        break;
389eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    }
390eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
39164621eac543d714d4d3f7cb9c24205f2ddc59201Glenn Kasten    // FIXME duplicates an initialization also done by higher level
392f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    mp->mAndroidObjState = ANDROID_UNINITIALIZED;
393f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    mp->mStreamType = ANDROID_DEFAULT_OUTPUT_STREAM_TYPE;
394f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    mp->mSessionId = android::AudioSystem::newAudioSessionId();
395eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
39672042d4448cee63528c619537321ba73944c6382Glenn Kasten    // placeholder: not necessary yet as session ID lifetime doesn't extend beyond player
39772042d4448cee63528c619537321ba73944c6382Glenn Kasten    // android::AudioSystem::acquireAudioSessionId(mp->mSessionId);
39872042d4448cee63528c619537321ba73944c6382Glenn Kasten
399485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    mp->mCallbackProtector = new android::CallbackProtector();
400485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten
401eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    return result;
402eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi}
403eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
404eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
405eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi//-----------------------------------------------------------------------------
406eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi// FIXME abstract out the diff between CMediaPlayer and CAudioPlayer
407eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel TriviXAresult android_Player_realize(CMediaPlayer *mp, SLboolean async) {
408ecc4fe22e076c4e5c891d823b01db1a683ba6690Glenn Kasten    SL_LOGV("android_Player_realize_l(%p)", mp);
409eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    XAresult result = XA_RESULT_SUCCESS;
410eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
411eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    const SLDataSource *pDataSrc = &mp->mDataSource.u.mSource;
412eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    const SLuint32 sourceLocator = *(SLuint32 *)pDataSrc->pLocator;
413eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
414eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    AudioPlayback_Parameters ap_params;
415f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    ap_params.sessionId = mp->mSessionId;
416f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    ap_params.streamType = mp->mStreamType;
417eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
418f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    switch(mp->mAndroidObjType) {
419b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi    case AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE: {
4207133228a478e16458b659946f2180ecddd13fda7Glenn Kasten        mp->mAVPlayer = new android::StreamPlayer(&ap_params, true /*hasVideo*/,
4217133228a478e16458b659946f2180ecddd13fda7Glenn Kasten                &mp->mAndroidBufferQueue, mp->mCallbackProtector);
422581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi        mp->mAVPlayer->init(player_handleMediaPlayerEventNotifications, (void*)mp);
423eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi        }
424eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi        break;
425b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi    case AUDIOVIDEOPLAYER_FROM_URIFD: {
42668d56b8ebaf60184a3aef988e3d2b09ed8b88c05Jean-Michel Trivi        mp->mAVPlayer = new android::LocAVPlayer(&ap_params, true /*hasVideo*/);
42797876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        mp->mAVPlayer->init(player_handleMediaPlayerEventNotifications, (void*)mp);
42897876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        switch (mp->mDataSource.mLocator.mLocatorType) {
42997876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        case XA_DATALOCATOR_URI:
43097876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi            ((android::LocAVPlayer*)mp->mAVPlayer.get())->setDataSource(
43197876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi                    (const char*)mp->mDataSource.mLocator.mURI.URI);
43297876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi            break;
43397876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        case XA_DATALOCATOR_ANDROIDFD: {
43497876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi            int64_t offset = (int64_t)mp->mDataSource.mLocator.mFD.offset;
43597876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi            ((android::LocAVPlayer*)mp->mAVPlayer.get())->setDataSource(
43697876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi                    (int)mp->mDataSource.mLocator.mFD.fd,
43797876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi                    offset == SL_DATALOCATOR_ANDROIDFD_USE_FILE_SIZE ?
43897876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi                            (int64_t)PLAYER_FD_FIND_FILE_SIZE : offset,
43997876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi                    (int64_t)mp->mDataSource.mLocator.mFD.length);
44097876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi            }
44197876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi            break;
44297876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        default:
443a8179ea15c4ff78db589d742b135649f0eda7ef2Glenn Kasten            SL_LOGE("Invalid or unsupported data locator type %u for data source",
44497876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi                    mp->mDataSource.mLocator.mLocatorType);
44597876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi            result = XA_RESULT_PARAMETER_INVALID;
44697876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        }
44797876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        }
44897876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        break;
449321f2cffd7dd560bf2e5c898be6953e19bed8496Jean-Michel Trivi    case INVALID_TYPE: // intended fall-through
450eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    default:
451f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi        SL_LOGE("Unable to realize MediaPlayer, invalid internal Android object type");
452eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi        result = XA_RESULT_PARAMETER_INVALID;
453eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi        break;
454eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    }
455eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
45635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    if (XA_RESULT_SUCCESS == result) {
45735ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten
45835ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        // if there is a video sink
45935ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        if (XA_DATALOCATOR_NATIVEDISPLAY ==
46035ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten                mp->mImageVideoSink.mLocator.mLocatorType) {
46135ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            ANativeWindow *nativeWindow = (ANativeWindow *)
46235ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten                    mp->mImageVideoSink.mLocator.mNativeDisplay.hWindow;
46335ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            // we already verified earlier that hWindow is non-NULL
46435ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            assert(nativeWindow != NULL);
46535ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            result = android_Player_setNativeWindow(mp, nativeWindow);
46635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        }
46735ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten
46835ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    }
46935ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten
470eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    return result;
471eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi}
472eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
473485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten// Called with a lock on MediaPlayer, and blocks until safe to destroy
474485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn KastenXAresult android_Player_preDestroy(CMediaPlayer *mp) {
475485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    SL_LOGV("android_Player_preDestroy(%p)", mp);
476485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten
477b66dfcb9e7b944c45927314ef2282d6cc95cfa0aGlenn Kasten    // Not yet clear why this order is important, but it reduces detected deadlocks
478485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    object_unlock_exclusive(&mp->mObject);
479485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    if (mp->mCallbackProtector != 0) {
480485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten        mp->mCallbackProtector->requestCbExitAndWait();
481485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    }
482485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    object_lock_exclusive(&mp->mObject);
483485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten
484b66dfcb9e7b944c45927314ef2282d6cc95cfa0aGlenn Kasten    if (mp->mAVPlayer != 0) {
485b66dfcb9e7b944c45927314ef2282d6cc95cfa0aGlenn Kasten        mp->mAVPlayer->preDestroy();
486b66dfcb9e7b944c45927314ef2282d6cc95cfa0aGlenn Kasten    }
487b66dfcb9e7b944c45927314ef2282d6cc95cfa0aGlenn Kasten    SL_LOGV("android_Player_preDestroy(%p) after mAVPlayer->preDestroy()", mp);
488b66dfcb9e7b944c45927314ef2282d6cc95cfa0aGlenn Kasten
489485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    return XA_RESULT_SUCCESS;
490485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten}
491485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten
49297876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi//-----------------------------------------------------------------------------
49397876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel TriviXAresult android_Player_destroy(CMediaPlayer *mp) {
494ecc4fe22e076c4e5c891d823b01db1a683ba6690Glenn Kasten    SL_LOGV("android_Player_destroy(%p)", mp);
49597876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi
496485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    mp->mAVPlayer.clear();
49772042d4448cee63528c619537321ba73944c6382Glenn Kasten
49872042d4448cee63528c619537321ba73944c6382Glenn Kasten    // placeholder: not necessary yet as session ID lifetime doesn't extend beyond player
49972042d4448cee63528c619537321ba73944c6382Glenn Kasten    // android::AudioSystem::releaseAudioSessionId(mp->mSessionId);
50072042d4448cee63528c619537321ba73944c6382Glenn Kasten
501485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    mp->mCallbackProtector.clear();
50297876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi
503485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    // explicit destructor
504485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    mp->mAVPlayer.~sp();
505485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    mp->mCallbackProtector.~sp();
506485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten
507485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    return XA_RESULT_SUCCESS;
50897876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi}
509eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
510ad1ab1d13a9b043202b9d5cdc1d8c4ef66cbbca8Glenn Kasten
511bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kastenvoid android_Player_usePlayEventMask(CMediaPlayer *mp) {
512bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten    if (mp->mAVPlayer != 0) {
513bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        IPlay *pPlayItf = &mp->mPlay;
514bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        mp->mAVPlayer->setPlayEvents((int32_t) pPlayItf->mEventFlags,
515bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten                (int32_t) pPlayItf->mMarkerPosition, (int32_t) pPlayItf->mPositionUpdatePeriod);
516bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten    }
517bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten}
518bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten
519bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten
52070c49ae2867094072a4365423417ea452bf82231Jean-Michel TriviXAresult android_Player_getDuration(IPlay *pPlayItf, XAmillisecond *pDurMsec) {
52170c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi    CMediaPlayer *avp = (CMediaPlayer *)pPlayItf->mThis;
52270c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi
52370c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi    switch (avp->mAndroidObjType) {
52470c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi
525b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi    case AUDIOVIDEOPLAYER_FROM_URIFD: {
5263d332ff421e7179c36fb652771cc8ded53383729Glenn Kasten        int dur = ANDROID_UNKNOWN_TIME;
52770c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi        if (avp->mAVPlayer != 0) {
52870c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi            avp->mAVPlayer->getDurationMsec(&dur);
52970c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi        }
53035a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        if (dur == ANDROID_UNKNOWN_TIME) {
53135a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi            *pDurMsec = XA_TIME_UNKNOWN;
53270c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi        } else {
53370c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi            *pDurMsec = (XAmillisecond)dur;
53470c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi        }
53570c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi    } break;
53670c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi
5373d332ff421e7179c36fb652771cc8ded53383729Glenn Kasten    case AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE: // intended fall-through
53870c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi    default:
5393d332ff421e7179c36fb652771cc8ded53383729Glenn Kasten        *pDurMsec = XA_TIME_UNKNOWN;
54035a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        break;
54135a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    }
54235a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi
5433d332ff421e7179c36fb652771cc8ded53383729Glenn Kasten    return XA_RESULT_SUCCESS;
54435a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi}
54535a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi
54635a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi
54735a5a30fdad179ccf38d8d756590411326159a89Jean-Michel TriviXAresult android_Player_getPosition(IPlay *pPlayItf, XAmillisecond *pPosMsec) {
54835a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    SL_LOGD("android_Player_getPosition()");
54935a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    XAresult result = XA_RESULT_SUCCESS;
55035a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    CMediaPlayer *avp = (CMediaPlayer *)pPlayItf->mThis;
55135a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi
55235a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    switch (avp->mAndroidObjType) {
55335a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi
55435a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    case AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE: // intended fall-through
55535a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    case AUDIOVIDEOPLAYER_FROM_URIFD: {
556136965c92c625d52a6cbad42f82a2091d7769c9cGlenn Kasten        int pos = ANDROID_UNKNOWN_TIME;
55735a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        if (avp->mAVPlayer != 0) {
55835a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi            avp->mAVPlayer->getPositionMsec(&pos);
55935a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        }
56035a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        if (pos == ANDROID_UNKNOWN_TIME) {
561136965c92c625d52a6cbad42f82a2091d7769c9cGlenn Kasten            *pPosMsec = 0;
56235a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        } else {
56335a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi            *pPosMsec = (XAmillisecond)pos;
56435a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        }
56535a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    } break;
56635a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi
56735a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    default:
56835a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        // we shouldn't be here
56935a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        assert(false);
57070c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi        break;
57170c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi    }
57270c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi
57370c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi    return result;
57470c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi}
57570c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi
57637dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
57737dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi//-----------------------------------------------------------------------------
57837dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi/**
57999b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten * pre-condition: mp != NULL
58037dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi */
58199b927751677abfb60a388d65dfeed1fed1db12cGlenn Kastenvoid android_Player_volumeUpdate(CMediaPlayer* mp)
58237dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi{
58399b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten    android::GenericPlayer* avp = mp->mAVPlayer.get();
58499b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten    if (avp != NULL) {
58599b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        float volumes[2];
58699b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        // MediaPlayer does not currently support EffectSend or MuteSolo
58799b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        android_player_volumeUpdate(volumes, &mp->mVolume, mp->mNumChannels, 1.0f, NULL);
58899b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        float leftVol = volumes[0], rightVol = volumes[1];
58999b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        avp->setVolume(leftVol, rightVol);
59099b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten    }
59137dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi}
59237dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
5930e47a0657162bbff48886ea2f5c68d9edb607768Jean-Michel Trivi//-----------------------------------------------------------------------------
5940e47a0657162bbff48886ea2f5c68d9edb607768Jean-Michel Trivi/**
59536b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel Trivi * pre-condition: gp != 0
596f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi */
59736b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel TriviXAresult android_Player_setPlayState(const android::sp<android::GenericPlayer> &gp,
59836b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel Trivi        SLuint32 playState,
599b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi        AndroidObjectState* pObjState)
600eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi{
601eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    XAresult result = XA_RESULT_SUCCESS;
602b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi    AndroidObjectState objState = *pObjState;
603eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
604eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    switch (playState) {
605eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi     case SL_PLAYSTATE_STOPPED: {
606f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi         SL_LOGV("setting AVPlayer to SL_PLAYSTATE_STOPPED");
60736b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel Trivi         gp->stop();
60897876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi         }
60997876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi         break;
610eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi     case SL_PLAYSTATE_PAUSED: {
611f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi         SL_LOGV("setting AVPlayer to SL_PLAYSTATE_PAUSED");
612eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         switch(objState) {
613e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi         case ANDROID_UNINITIALIZED:
614581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi             *pObjState = ANDROID_PREPARING;
61536b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel Trivi             gp->prepare();
616581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi             break;
617e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi         case ANDROID_PREPARING:
618eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi             break;
619e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi         case ANDROID_READY:
62036b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel Trivi             gp->pause();
621581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi             break;
622eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         default:
623f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi             SL_LOGE("Android object in invalid state");
624eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi             break;
625eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         }
62697876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi         }
62797876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi         break;
628eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi     case SL_PLAYSTATE_PLAYING: {
629f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi         SL_LOGV("setting AVPlayer to SL_PLAYSTATE_PLAYING");
630eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         switch(objState) {
631e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi         case ANDROID_UNINITIALIZED:
632581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi             *pObjState = ANDROID_PREPARING;
63336b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel Trivi             gp->prepare();
634581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi             // intended fall through
635e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi         case ANDROID_PREPARING:
636581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi             // intended fall through
637e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi         case ANDROID_READY:
63836b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel Trivi             gp->play();
639581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi             break;
640eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         default:
641f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi             SL_LOGE("Android object in invalid state");
642eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi             break;
643eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         }
64497876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi         }
64597876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi         break;
646eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi     default:
647eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         // checked by caller, should not happen
648eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         break;
649eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi     }
650eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
651eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    return result;
652eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi}
653eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
654eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
65592b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi/**
65692b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi * pre-condition: mp != NULL
65792b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi */
65892b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel TriviXAresult android_Player_seek(CMediaPlayer *mp, SLmillisecond posMsec) {
65992b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    XAresult result = XA_RESULT_SUCCESS;
66092b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    switch (mp->mAndroidObjType) {
66192b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      case AUDIOVIDEOPLAYER_FROM_URIFD:
66292b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        if (mp->mAVPlayer !=0) {
66392b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi            mp->mAVPlayer->seek(posMsec);
66492b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        }
66592b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        break;
66692b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      case AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE: // intended fall-through
66792b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      default: {
668f5ff1a75f55677163bd9a8bd804e8f5c33ef592cGlenn Kasten          result = XA_RESULT_FEATURE_UNSUPPORTED;
66992b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      }
67092b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    }
67192b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    return result;
67292b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi}
67392b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi
67492b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi
67592b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi/**
67692b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi * pre-condition: mp != NULL
67792b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi */
67892b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel TriviXAresult android_Player_loop(CMediaPlayer *mp, SLboolean loopEnable) {
67992b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    XAresult result = XA_RESULT_SUCCESS;
68092b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    switch (mp->mAndroidObjType) {
68192b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      case AUDIOVIDEOPLAYER_FROM_URIFD:
68292b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        if (mp->mAVPlayer !=0) {
68392b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi            mp->mAVPlayer->loop(loopEnable);
68492b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        }
68592b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        break;
68692b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      case AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE: // intended fall-through
68792b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      default: {
688f5ff1a75f55677163bd9a8bd804e8f5c33ef592cGlenn Kasten          result = XA_RESULT_FEATURE_UNSUPPORTED;
68992b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      }
69092b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    }
69192b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    return result;
69292b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi}
69392b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi
69492b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi
695eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi//-----------------------------------------------------------------------------
696e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivivoid android_Player_androidBufferQueue_clear_l(CMediaPlayer *mp) {
697b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi    if ((mp->mAndroidObjType == AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE)
698b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi            && (mp->mAVPlayer != 0)) {
699e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi        android::StreamPlayer* splr = static_cast<android::StreamPlayer*>(mp->mAVPlayer.get());
700e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi        splr->appClear_l();
701e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi    }
702e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi}
703e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi
704e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi
70570c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivivoid android_Player_androidBufferQueue_onRefilled_l(CMediaPlayer *mp) {
706b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi    if ((mp->mAndroidObjType == AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE)
707b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi            && (mp->mAVPlayer != 0)) {
70870c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi        android::StreamPlayer* splr = static_cast<android::StreamPlayer*>(mp->mAVPlayer.get());
709bc0e642e6c1a51b3ae3a02d490d94b03e718e6b5Jean-Michel Trivi        splr->queueRefilled();
710eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    }
711eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi}
712eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
713eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
71435ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten/*
71535ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten *  pre-conditions:
71635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten *      mp != NULL
71735ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten *      mp->mAVPlayer != 0 (player is realized)
71835ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten *      nativeWindow can be NULL, but if NULL it is treated as an error
71935ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten */
72035ac702ee1ad91e5c8748c12450222d50b366a52Glenn KastenSLresult android_Player_setNativeWindow(CMediaPlayer *mp, ANativeWindow *nativeWindow)
72135ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten{
72235ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    assert(mp != NULL);
72335ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    assert(mp->mAVPlayer != 0);
72435ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    if (nativeWindow == NULL) {
72535ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        SL_LOGE("ANativeWindow is NULL");
72635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        return SL_RESULT_PARAMETER_INVALID;
72735ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    }
72835ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    SLresult result;
72935ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    int err;
73035ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    int value;
73135ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    // this could crash if app passes in a bad parameter, but that's OK
73235ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    err = (*nativeWindow->query)(nativeWindow, NATIVE_WINDOW_CONCRETE_TYPE, &value);
73335ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    if (0 != err) {
73435ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        SL_LOGE("Query NATIVE_WINDOW_CONCRETE_TYPE on ANativeWindow * %p failed; "
73535ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten                "errno %d", nativeWindow, err);
73635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        result = SL_RESULT_PARAMETER_INVALID;
73735ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    } else {
73835ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        switch (value) {
73935ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        case NATIVE_WINDOW_SURFACE: {                // Surface
74035ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            SL_LOGV("Displaying on ANativeWindow of type NATIVE_WINDOW_SURFACE");
74135ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            android::sp<android::Surface> nativeSurface(
74235ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten                    static_cast<android::Surface *>(nativeWindow));
743241b9c06493479dc632a8851097c193b724a2b41Andreas Huber            mp->mAVPlayer->setVideoSurfaceTexture(
744241b9c06493479dc632a8851097c193b724a2b41Andreas Huber                    nativeSurface->getSurfaceTexture());
74535ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            result = SL_RESULT_SUCCESS;
74635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            } break;
74735ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        case NATIVE_WINDOW_SURFACE_TEXTURE_CLIENT: { // SurfaceTextureClient
74835ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            SL_LOGV("Displaying on ANativeWindow of type NATIVE_WINDOW_SURFACE_TEXTURE_CLIENT");
74935ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            android::sp<android::SurfaceTextureClient> surfaceTextureClient(
75035ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten                    static_cast<android::SurfaceTextureClient *>(nativeWindow));
75135ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            android::sp<android::ISurfaceTexture> nativeSurfaceTexture(
75235ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten                    surfaceTextureClient->getISurfaceTexture());
75335ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            mp->mAVPlayer->setVideoSurfaceTexture(nativeSurfaceTexture);
75435ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            result = SL_RESULT_SUCCESS;
75535ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            } break;
75635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        case NATIVE_WINDOW_FRAMEBUFFER:              // FramebufferNativeWindow
75735ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            // fall through
75835ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        default:
75935ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            SL_LOGE("ANativeWindow * %p has unknown or unsupported concrete type %d",
76035ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten                    nativeWindow, value);
76135ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            result = SL_RESULT_PARAMETER_INVALID;
76235ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            break;
76335ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        }
76435ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    }
76535ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    return result;
76635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten}
767