MediaPlayer_to_android.cpp revision e2e8fa36bd7448b59fbcdf141e0b6d21e5401d91
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: {
55c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
56c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        SL_LOGV("Received AVPlayer::kEventPrepared for CMediaPlayer %p", mp);
57c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
58c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        // assume no callback
59c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        slPrefetchCallback callback = NULL;
60c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        void* callbackPContext = NULL;
61c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
62c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        object_lock_exclusive(&mp->mObject);
63c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        // mark object as prepared; same state is used for successfully or unsuccessful prepare
64c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        mp->mAndroidObjState = ANDROID_READY;
65c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
66c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        // AVPlayer prepare() failed prefetching, there is no event in XAPrefetchStatus to
67c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        //  indicate a prefetch error, so we signal it by sending simulataneously two events:
68c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        //  - SL_PREFETCHEVENT_FILLLEVELCHANGE with a level of 0
69c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        //  - SL_PREFETCHEVENT_STATUSCHANGE with a status of SL_PREFETCHSTATUS_UNDERFLOW
70c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (PLAYER_SUCCESS != data1 && IsInterfaceInitialized(&mp->mObject, MPH_XAPREFETCHSTATUS)) {
71c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            mp->mPrefetchStatus.mLevel = 0;
72c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            mp->mPrefetchStatus.mStatus = SL_PREFETCHSTATUS_UNDERFLOW;
73c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            if (!(~mp->mPrefetchStatus.mCallbackEventsMask &
74c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten                    (SL_PREFETCHEVENT_FILLLEVELCHANGE | SL_PREFETCHEVENT_STATUSCHANGE))) {
75c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten                callback = mp->mPrefetchStatus.mCallback;
76c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten                callbackPContext = mp->mPrefetchStatus.mContext;
77c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            }
78581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi        }
79c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        object_unlock_exclusive(&mp->mObject);
80c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
81c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        // callback with no lock held
82c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (NULL != callback) {
83c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            (*callback)(&mp->mPrefetchStatus.mItf, callbackPContext,
84c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten                    SL_PREFETCHEVENT_FILLLEVELCHANGE | SL_PREFETCHEVENT_STATUSCHANGE);
85581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi        }
86c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
8737dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        break;
8837dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi      }
8937dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
9037dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi      case android::GenericPlayer::kEventHasVideoSize: {
9137dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        SL_LOGV("Received AVPlayer::kEventHasVideoSize (%d,%d) for CMediaPlayer %p",
9237dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi                data1, data2, mp);
9337dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
9437dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        object_lock_exclusive(&mp->mObject);
9537dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
9637dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        // remove an existing video info entry (here we only have one video stream)
9737dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        for(size_t i=0 ; i < mp->mStreamInfo.mStreamInfoTable.size() ; i++) {
9837dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi            if (XA_DOMAINTYPE_VIDEO == mp->mStreamInfo.mStreamInfoTable.itemAt(i).domain) {
9937dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi                mp->mStreamInfo.mStreamInfoTable.removeAt(i);
10037dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi                break;
10137dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi            }
10237dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        }
10337dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        // update the stream information with a new video info entry
10437dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        StreamInfo streamInfo;
10537dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        streamInfo.domain = XA_DOMAINTYPE_VIDEO;
10637dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        streamInfo.videoInfo.codecId = 0;// unknown, we don't have that info FIXME
10737dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        streamInfo.videoInfo.width = (XAuint32)data1;
10837dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        streamInfo.videoInfo.height = (XAuint32)data2;
10937dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        streamInfo.videoInfo.bitRate = 0;// unknown, we don't have that info FIXME
110cb441acdda6f8e81d44fcdaadd4ff7ab3d3e367bGlenn Kasten        streamInfo.videoInfo.frameRate = 0;
11137dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        streamInfo.videoInfo.duration = XA_TIME_UNKNOWN;
11237dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        StreamInfo &contInfo = mp->mStreamInfo.mStreamInfoTable.editItemAt(0);
11337dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        contInfo.containerInfo.numStreams = 1;
11437dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        ssize_t index = mp->mStreamInfo.mStreamInfoTable.add(streamInfo);
11537dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
116e2e8fa36bd7448b59fbcdf141e0b6d21e5401d91Glenn Kasten        // callback is unconditional; there is no bitmask of enabled events
11737dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        xaStreamEventChangeCallback callback = mp->mStreamInfo.mCallback;
11837dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        void* callbackPContext = mp->mStreamInfo.mContext;
11937dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
12037dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        object_unlock_exclusive(&mp->mObject);
12137dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
12285133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi        // enqueue notification (outside of lock) that the stream information has been updated
12337dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi        if ((NULL != callback) && (index >= 0)) {
124dd177e2d3923d4653eaa4226f07b89a999907970Glenn Kasten#ifndef USE_ASYNCHRONOUS_STREAMCBEVENT_PROPERTYCHANGE_CALLBACK
12537dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi            (*callback)(&mp->mStreamInfo.mItf, XA_STREAMCBEVENT_PROPERTYCHANGE /*eventId*/,
12637dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi                    1 /*streamIndex, only one stream supported here, 0 is reserved*/,
12737dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi                    NULL /*pEventData, always NULL in OpenMAX AL 1.0.1*/,
12837dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi                    callbackPContext /*pContext*/);
12985133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi#else
13085133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi            SLresult res = EnqueueAsyncCallback_piipp(mp, callback,
13185133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi                    /*p1*/ &mp->mStreamInfo.mItf,
13285133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi                    /*i1*/ XA_STREAMCBEVENT_PROPERTYCHANGE /*eventId*/,
13385133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi                    /*i2*/ 1 /*streamIndex, only one stream supported here, 0 is reserved*/,
13485133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi                    /*p2*/ NULL /*pEventData, always NULL in OpenMAX AL 1.0.1*/,
13585133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi                    /*p3*/ callbackPContext /*pContext*/);
136dd177e2d3923d4653eaa4226f07b89a999907970Glenn Kasten            LOGW_IF(SL_RESULT_SUCCESS != res,
137dd177e2d3923d4653eaa4226f07b89a999907970Glenn Kasten                        "Callback %p(%p, XA_STREAMCBEVENT_PROPERTYCHANGE, 1, NULL, %p) dropped",
138dd177e2d3923d4653eaa4226f07b89a999907970Glenn Kasten                        callback, &mp->mStreamInfo.mItf, callbackPContext);
13985133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi#endif
140581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi        }
141581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi        break;
14237dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi      }
143581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi
14492b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      case android::GenericPlayer::kEventEndOfStream: {
14592b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        SL_LOGV("Received AVPlayer::kEventEndOfStream for CMediaPlayer %p", mp);
14692b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi
14792b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        object_lock_exclusive(&mp->mObject);
14892b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        // should be xaPlayCallback but we're sharing the itf between SL and AL
14992b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        slPlayCallback playCallback = NULL;
15092b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        void * playContext = NULL;
15192b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        // XAPlayItf callback or no callback?
15292b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        if (mp->mPlay.mEventFlags & XA_PLAYEVENT_HEADATEND) {
15392b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi            playCallback = mp->mPlay.mCallback;
15492b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi            playContext = mp->mPlay.mContext;
15592b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        }
156e37c62f0691f9a137afae60f9d6dbd1a65d36aedGlenn Kasten        mp->mPlay.mState = XA_PLAYSTATE_PAUSED;
15792b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        object_unlock_exclusive(&mp->mObject);
15892b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi
15985133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi        // enqueue callback with no lock held
16092b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        if (NULL != playCallback) {
161dd177e2d3923d4653eaa4226f07b89a999907970Glenn Kasten#ifndef USE_ASYNCHRONOUS_PLAY_CALLBACK
16292b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi            (*playCallback)(&mp->mPlay.mItf, playContext, XA_PLAYEVENT_HEADATEND);
16385133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi#else
16485133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi            SLresult res = EnqueueAsyncCallback_ppi(mp, playCallback, &mp->mPlay.mItf, playContext,
16585133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi                    XA_PLAYEVENT_HEADATEND);
16685133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi            LOGW_IF(SL_RESULT_SUCCESS != res,
167c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten                    "Callback %p(%p, %p, SL_PLAYEVENT_HEADATEND) dropped", playCallback,
16885133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi                    &mp->mPlay.mItf, playContext);
16985133c817f6f387cd7d072988a8818f18bb53702Jean-Michel Trivi#endif
17092b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        }
171581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi        break;
17292b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      }
17392b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi
17499b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      case android::GenericPlayer::kEventChannelCount: {
17599b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        SL_LOGV("kEventChannelCount channels = %d", data1);
17699b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        object_lock_exclusive(&mp->mObject);
17799b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        if (UNKNOWN_NUMCHANNELS == mp->mNumChannels && UNKNOWN_NUMCHANNELS != data1) {
17899b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten            mp->mNumChannels = data1;
17999b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten            android_Player_volumeUpdate(mp);
18099b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        }
18199b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        object_unlock_exclusive(&mp->mObject);
18299b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      }
18399b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      break;
18499b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten
18599b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      case android::GenericPlayer::kEventPrefetchFillLevelUpdate: {
18699b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        SL_LOGV("kEventPrefetchFillLevelUpdate");
187c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (!IsInterfaceInitialized(&mp->mObject, MPH_XAPREFETCHSTATUS)) {
188c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            break;
189c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        }
190c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        slPrefetchCallback callback = NULL;
191c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        void* callbackPContext = NULL;
192c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
193c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        // SLPrefetchStatusItf callback or no callback?
194c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        interface_lock_exclusive(&mp->mPrefetchStatus);
195c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (mp->mPrefetchStatus.mCallbackEventsMask & SL_PREFETCHEVENT_FILLLEVELCHANGE) {
196c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            callback = mp->mPrefetchStatus.mCallback;
197c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            callbackPContext = mp->mPrefetchStatus.mContext;
198c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        }
199c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        mp->mPrefetchStatus.mLevel = (SLpermille)data1;
200c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        interface_unlock_exclusive(&mp->mPrefetchStatus);
201c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
202c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        // callback with no lock held
203c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (NULL != callback) {
204c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            (*callback)(&mp->mPrefetchStatus.mItf, callbackPContext,
205c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten                    SL_PREFETCHEVENT_FILLLEVELCHANGE);
206c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        }
20799b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      }
20899b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      break;
20999b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten
21099b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      case android::GenericPlayer::kEventPrefetchStatusChange: {
21199b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        SL_LOGV("kEventPrefetchStatusChange");
212c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (!IsInterfaceInitialized(&mp->mObject, MPH_XAPREFETCHSTATUS)) {
213c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            break;
214c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        }
215c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        slPrefetchCallback callback = NULL;
216c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        void* callbackPContext = NULL;
217c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
218c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        // SLPrefetchStatusItf callback or no callback?
219c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        object_lock_exclusive(&mp->mObject);
220c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (mp->mPrefetchStatus.mCallbackEventsMask & SL_PREFETCHEVENT_STATUSCHANGE) {
221c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            callback = mp->mPrefetchStatus.mCallback;
222c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            callbackPContext = mp->mPrefetchStatus.mContext;
223c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        }
224c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (data1 >= android::kStatusIntermediate) {
225c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            mp->mPrefetchStatus.mStatus = SL_PREFETCHSTATUS_SUFFICIENTDATA;
226c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        } else if (data1 < android::kStatusIntermediate) {
227c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            mp->mPrefetchStatus.mStatus = SL_PREFETCHSTATUS_UNDERFLOW;
228c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        }
229c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        object_unlock_exclusive(&mp->mObject);
230c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten
231c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        // callback with no lock held
232c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        if (NULL != callback) {
233c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten            (*callback)(&mp->mPrefetchStatus.mItf, callbackPContext, SL_PREFETCHEVENT_STATUSCHANGE);
234c623c89c0a32c5fc77c998f1742d58e7be69e8c1Glenn Kasten        }
23599b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      }
23699b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten      break;
23799b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten
238bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten      case android::GenericPlayer::kEventPlay: {
239bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        SL_LOGV("kEventPlay");
240bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten
241bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        interface_lock_shared(&mp->mPlay);
242bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        slPlayCallback callback = mp->mPlay.mCallback;
243bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        void* callbackPContext = mp->mPlay.mContext;
244bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        interface_unlock_shared(&mp->mPlay);
245bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten
246bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        if (NULL != callback) {
247bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten            (*callback)(&mp->mPlay.mItf, callbackPContext, (SLuint32) data1); // SL_PLAYEVENT_HEAD*
248bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        }
249bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten      }
250bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten      break;
25199b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten
2525e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten      case android::GenericPlayer::kEventErrorAfterPrepare: {
2535e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        SL_LOGV("kEventErrorAfterPrepare");
2545e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten
2555e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        // assume no callback
2565e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        slPrefetchCallback callback = NULL;
2575e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        void* callbackPContext = NULL;
2585e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten
2595e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        object_lock_exclusive(&mp->mObject);
2605e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        if (IsInterfaceInitialized(&mp->mObject, MPH_XAPREFETCHSTATUS)) {
2615e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten            mp->mPrefetchStatus.mLevel = 0;
2625e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten            mp->mPrefetchStatus.mStatus = SL_PREFETCHSTATUS_UNDERFLOW;
2635e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten            if (!(~mp->mPrefetchStatus.mCallbackEventsMask &
2645e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten                    (SL_PREFETCHEVENT_FILLLEVELCHANGE | SL_PREFETCHEVENT_STATUSCHANGE))) {
2655e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten                callback = mp->mPrefetchStatus.mCallback;
2665e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten                callbackPContext = mp->mPrefetchStatus.mContext;
2675e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten            }
2685e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        }
2695e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        object_unlock_exclusive(&mp->mObject);
2705e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten
2715e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        // FIXME there's interesting information in data1, but no API to convey it to client
2725e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        SL_LOGE("Error after prepare: %d", data1);
2735e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten
2745e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        // callback with no lock held
2755e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        if (NULL != callback) {
2765e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten            (*callback)(&mp->mPrefetchStatus.mItf, callbackPContext,
2775e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten                    SL_PREFETCHEVENT_FILLLEVELCHANGE | SL_PREFETCHEVENT_STATUSCHANGE);
2785e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten        }
2795e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten
2805e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten      }
2815e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten      break;
2825e4d65e369f28746767aba11b618dee314bb8197Glenn Kasten
28392b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      default: {
28492b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        SL_LOGE("Received unknown event %d, data %d from AVPlayer", event, data1);
28592b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      }
286581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi    }
2879d8a98601357c0669dca4de63e43196c0a70553dGlenn Kasten
2889d8a98601357c0669dca4de63e43196c0a70553dGlenn Kasten    mp->mCallbackProtector->exitCb();
289581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi}
290581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi
291eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
292eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi//-----------------------------------------------------------------------------
293d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel TriviXAresult android_Player_checkSourceSink(CMediaPlayer *mp) {
294d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
295d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    XAresult result = XA_RESULT_SUCCESS;
296d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
297d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    const SLDataSource *pSrc    = &mp->mDataSource.u.mSource;
298d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    const SLDataSink *pAudioSnk = &mp->mAudioSink.u.mSink;
299d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
300d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    // format check:
301d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    const SLuint32 sourceLocatorType = *(SLuint32 *)pSrc->pLocator;
302d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    const SLuint32 sourceFormatType  = *(SLuint32 *)pSrc->pFormat;
303d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    const SLuint32 audioSinkLocatorType = *(SLuint32 *)pAudioSnk->pLocator;
304d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    //const SLuint32 sinkFormatType = *(SLuint32 *)pAudioSnk->pFormat;
305d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
306d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    // Source check
307d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    switch(sourceLocatorType) {
308d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
309d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    case XA_DATALOCATOR_ANDROIDBUFFERQUEUE: {
310d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        switch (sourceFormatType) {
311d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        case XA_DATAFORMAT_MIME: {
312d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi            SLDataFormat_MIME *df_mime = (SLDataFormat_MIME *) pSrc->pFormat;
313d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi            if (SL_CONTAINERTYPE_MPEG_TS != df_mime->containerType) {
314d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi                SL_LOGE("Cannot create player with XA_DATALOCATOR_ANDROIDBUFFERQUEUE data source "
315d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi                        "that is not fed MPEG-2 TS data");
316d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi                return SL_RESULT_CONTENT_UNSUPPORTED;
317d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi            }
318d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        } break;
319d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        default:
320d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi            SL_LOGE("Cannot create player with XA_DATALOCATOR_ANDROIDBUFFERQUEUE data source "
321d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi                    "without SL_DATAFORMAT_MIME format");
322d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi            return XA_RESULT_CONTENT_UNSUPPORTED;
323d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        }
324d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    } break;
325d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
326d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    case XA_DATALOCATOR_URI: // intended fall-through
327d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    case XA_DATALOCATOR_ANDROIDFD:
328d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        break;
329d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
330d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    default:
331d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        SL_LOGE("Cannot create media player with data locator type 0x%x",
332d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi                (unsigned) sourceLocatorType);
333d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        return SL_RESULT_PARAMETER_INVALID;
334d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    }// switch (locatorType)
335d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
336d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    // Audio sink check: only playback is supported here
337d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    switch(audioSinkLocatorType) {
338d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
339d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    case XA_DATALOCATOR_OUTPUTMIX:
340d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        break;
341d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
342d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    default:
343d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        SL_LOGE("Cannot create media player with audio sink data locator of type 0x%x",
344d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi                (unsigned) audioSinkLocatorType);
345d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi        return XA_RESULT_PARAMETER_INVALID;
346d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    }// switch (locaaudioSinkLocatorTypeorType)
347d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
348d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi    return result;
349d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi}
350d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
351d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi
352d158d31a6bbb06426b71c3d097b7768bc3fb79a3Jean-Michel Trivi//-----------------------------------------------------------------------------
353eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel TriviXAresult android_Player_create(CMediaPlayer *mp) {
354eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
355eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    XAresult result = XA_RESULT_SUCCESS;
356eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
357eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    // FIXME verify data source
358eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    const SLDataSource *pDataSrc = &mp->mDataSource.u.mSource;
359eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    // FIXME verify audio data sink
360eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    const SLDataSink *pAudioSnk = &mp->mAudioSink.u.mSink;
361eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    // FIXME verify image data sink
362eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    const SLDataSink *pVideoSnk = &mp->mImageVideoSink.u.mSink;
363eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
36497876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi    XAuint32 sourceLocator = *(XAuint32 *)pDataSrc->pLocator;
365eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    switch(sourceLocator) {
36697876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi    // FIXME support Android simple buffer queue as well
367f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    case XA_DATALOCATOR_ANDROIDBUFFERQUEUE:
368b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi        mp->mAndroidObjType = AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE;
369eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi        break;
370f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    case XA_DATALOCATOR_URI: // intended fall-through
37197876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi    case SL_DATALOCATOR_ANDROIDFD:
372b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi        mp->mAndroidObjType = AUDIOVIDEOPLAYER_FROM_URIFD;
37397876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        break;
374f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    case XA_DATALOCATOR_ADDRESS: // intended fall-through
375eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    default:
376a8179ea15c4ff78db589d742b135649f0eda7ef2Glenn Kasten        SL_LOGE("Unable to create MediaPlayer for data source locator 0x%x", sourceLocator);
377eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi        result = XA_RESULT_PARAMETER_INVALID;
378eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi        break;
379eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    }
380eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
38164621eac543d714d4d3f7cb9c24205f2ddc59201Glenn Kasten    // FIXME duplicates an initialization also done by higher level
382f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    mp->mAndroidObjState = ANDROID_UNINITIALIZED;
383f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    mp->mStreamType = ANDROID_DEFAULT_OUTPUT_STREAM_TYPE;
384f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    mp->mSessionId = android::AudioSystem::newAudioSessionId();
385eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
386485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    mp->mCallbackProtector = new android::CallbackProtector();
387485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten
388eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    return result;
389eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi}
390eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
391eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
392eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi//-----------------------------------------------------------------------------
393eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi// FIXME abstract out the diff between CMediaPlayer and CAudioPlayer
394eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel TriviXAresult android_Player_realize(CMediaPlayer *mp, SLboolean async) {
395ecc4fe22e076c4e5c891d823b01db1a683ba6690Glenn Kasten    SL_LOGV("android_Player_realize_l(%p)", mp);
396eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    XAresult result = XA_RESULT_SUCCESS;
397eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
398eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    const SLDataSource *pDataSrc = &mp->mDataSource.u.mSource;
399eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    const SLuint32 sourceLocator = *(SLuint32 *)pDataSrc->pLocator;
400eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
401eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    AudioPlayback_Parameters ap_params;
402f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    ap_params.sessionId = mp->mSessionId;
403f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    ap_params.streamType = mp->mStreamType;
404eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
405f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi    switch(mp->mAndroidObjType) {
406b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi    case AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE: {
4077133228a478e16458b659946f2180ecddd13fda7Glenn Kasten        mp->mAVPlayer = new android::StreamPlayer(&ap_params, true /*hasVideo*/,
4087133228a478e16458b659946f2180ecddd13fda7Glenn Kasten                &mp->mAndroidBufferQueue, mp->mCallbackProtector);
409581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi        mp->mAVPlayer->init(player_handleMediaPlayerEventNotifications, (void*)mp);
410eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi        }
411eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi        break;
412b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi    case AUDIOVIDEOPLAYER_FROM_URIFD: {
41368d56b8ebaf60184a3aef988e3d2b09ed8b88c05Jean-Michel Trivi        mp->mAVPlayer = new android::LocAVPlayer(&ap_params, true /*hasVideo*/);
41497876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        mp->mAVPlayer->init(player_handleMediaPlayerEventNotifications, (void*)mp);
41597876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        switch (mp->mDataSource.mLocator.mLocatorType) {
41697876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        case XA_DATALOCATOR_URI:
41797876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi            ((android::LocAVPlayer*)mp->mAVPlayer.get())->setDataSource(
41897876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi                    (const char*)mp->mDataSource.mLocator.mURI.URI);
41997876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi            break;
42097876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        case XA_DATALOCATOR_ANDROIDFD: {
42197876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi            int64_t offset = (int64_t)mp->mDataSource.mLocator.mFD.offset;
42297876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi            ((android::LocAVPlayer*)mp->mAVPlayer.get())->setDataSource(
42397876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi                    (int)mp->mDataSource.mLocator.mFD.fd,
42497876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi                    offset == SL_DATALOCATOR_ANDROIDFD_USE_FILE_SIZE ?
42597876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi                            (int64_t)PLAYER_FD_FIND_FILE_SIZE : offset,
42697876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi                    (int64_t)mp->mDataSource.mLocator.mFD.length);
42797876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi            }
42897876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi            break;
42997876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        default:
430a8179ea15c4ff78db589d742b135649f0eda7ef2Glenn Kasten            SL_LOGE("Invalid or unsupported data locator type %u for data source",
43197876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi                    mp->mDataSource.mLocator.mLocatorType);
43297876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi            result = XA_RESULT_PARAMETER_INVALID;
43397876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        }
43497876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        }
43597876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi        break;
436321f2cffd7dd560bf2e5c898be6953e19bed8496Jean-Michel Trivi    case INVALID_TYPE: // intended fall-through
437eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    default:
438f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi        SL_LOGE("Unable to realize MediaPlayer, invalid internal Android object type");
439eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi        result = XA_RESULT_PARAMETER_INVALID;
440eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi        break;
441eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    }
442eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
44335ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    if (XA_RESULT_SUCCESS == result) {
44435ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten
44535ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        // if there is a video sink
44635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        if (XA_DATALOCATOR_NATIVEDISPLAY ==
44735ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten                mp->mImageVideoSink.mLocator.mLocatorType) {
44835ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            ANativeWindow *nativeWindow = (ANativeWindow *)
44935ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten                    mp->mImageVideoSink.mLocator.mNativeDisplay.hWindow;
45035ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            // we already verified earlier that hWindow is non-NULL
45135ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            assert(nativeWindow != NULL);
45235ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            result = android_Player_setNativeWindow(mp, nativeWindow);
45335ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        }
45435ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten
45535ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    }
45635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten
457eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    return result;
458eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi}
459eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
460485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten// Called with a lock on MediaPlayer, and blocks until safe to destroy
461485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn KastenXAresult android_Player_preDestroy(CMediaPlayer *mp) {
462485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    SL_LOGV("android_Player_preDestroy(%p)", mp);
463485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten
464b66dfcb9e7b944c45927314ef2282d6cc95cfa0aGlenn Kasten    // Not yet clear why this order is important, but it reduces detected deadlocks
465485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    object_unlock_exclusive(&mp->mObject);
466485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    if (mp->mCallbackProtector != 0) {
467485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten        mp->mCallbackProtector->requestCbExitAndWait();
468485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    }
469485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    object_lock_exclusive(&mp->mObject);
470485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten
471b66dfcb9e7b944c45927314ef2282d6cc95cfa0aGlenn Kasten    if (mp->mAVPlayer != 0) {
472b66dfcb9e7b944c45927314ef2282d6cc95cfa0aGlenn Kasten        mp->mAVPlayer->preDestroy();
473b66dfcb9e7b944c45927314ef2282d6cc95cfa0aGlenn Kasten    }
474b66dfcb9e7b944c45927314ef2282d6cc95cfa0aGlenn Kasten    SL_LOGV("android_Player_preDestroy(%p) after mAVPlayer->preDestroy()", mp);
475b66dfcb9e7b944c45927314ef2282d6cc95cfa0aGlenn Kasten
476485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    return XA_RESULT_SUCCESS;
477485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten}
478485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten
47997876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi//-----------------------------------------------------------------------------
48097876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel TriviXAresult android_Player_destroy(CMediaPlayer *mp) {
481ecc4fe22e076c4e5c891d823b01db1a683ba6690Glenn Kasten    SL_LOGV("android_Player_destroy(%p)", mp);
48297876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi
483485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    mp->mAVPlayer.clear();
484485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    mp->mCallbackProtector.clear();
48597876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi
486485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    // explicit destructor
487485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    mp->mAVPlayer.~sp();
488485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    mp->mCallbackProtector.~sp();
489485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten
490485a038f9f0f898227b8ab4218e94c5d56b6ed0bGlenn Kasten    return XA_RESULT_SUCCESS;
49197876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi}
492eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
493ad1ab1d13a9b043202b9d5cdc1d8c4ef66cbbca8Glenn Kasten
494bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kastenvoid android_Player_usePlayEventMask(CMediaPlayer *mp) {
495bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten    if (mp->mAVPlayer != 0) {
496bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        IPlay *pPlayItf = &mp->mPlay;
497bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten        mp->mAVPlayer->setPlayEvents((int32_t) pPlayItf->mEventFlags,
498bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten                (int32_t) pPlayItf->mMarkerPosition, (int32_t) pPlayItf->mPositionUpdatePeriod);
499bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten    }
500bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten}
501bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten
502bcfe680db1e392f3bb29382c2e15e89c3af783edGlenn Kasten
50370c49ae2867094072a4365423417ea452bf82231Jean-Michel TriviXAresult android_Player_getDuration(IPlay *pPlayItf, XAmillisecond *pDurMsec) {
50470c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi    CMediaPlayer *avp = (CMediaPlayer *)pPlayItf->mThis;
50570c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi
50670c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi    switch (avp->mAndroidObjType) {
50770c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi
508b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi    case AUDIOVIDEOPLAYER_FROM_URIFD: {
5093d332ff421e7179c36fb652771cc8ded53383729Glenn Kasten        int dur = ANDROID_UNKNOWN_TIME;
51070c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi        if (avp->mAVPlayer != 0) {
51170c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi            avp->mAVPlayer->getDurationMsec(&dur);
51270c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi        }
51335a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        if (dur == ANDROID_UNKNOWN_TIME) {
51435a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi            *pDurMsec = XA_TIME_UNKNOWN;
51570c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi        } else {
51670c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi            *pDurMsec = (XAmillisecond)dur;
51770c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi        }
51870c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi    } break;
51970c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi
5203d332ff421e7179c36fb652771cc8ded53383729Glenn Kasten    case AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE: // intended fall-through
52170c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi    default:
5223d332ff421e7179c36fb652771cc8ded53383729Glenn Kasten        *pDurMsec = XA_TIME_UNKNOWN;
52335a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        break;
52435a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    }
52535a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi
5263d332ff421e7179c36fb652771cc8ded53383729Glenn Kasten    return XA_RESULT_SUCCESS;
52735a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi}
52835a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi
52935a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi
53035a5a30fdad179ccf38d8d756590411326159a89Jean-Michel TriviXAresult android_Player_getPosition(IPlay *pPlayItf, XAmillisecond *pPosMsec) {
53135a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    SL_LOGD("android_Player_getPosition()");
53235a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    XAresult result = XA_RESULT_SUCCESS;
53335a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    CMediaPlayer *avp = (CMediaPlayer *)pPlayItf->mThis;
53435a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi
53535a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    switch (avp->mAndroidObjType) {
53635a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi
53735a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    case AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE: // intended fall-through
53835a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    case AUDIOVIDEOPLAYER_FROM_URIFD: {
539136965c92c625d52a6cbad42f82a2091d7769c9cGlenn Kasten        int pos = ANDROID_UNKNOWN_TIME;
54035a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        if (avp->mAVPlayer != 0) {
54135a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi            avp->mAVPlayer->getPositionMsec(&pos);
54235a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        }
54335a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        if (pos == ANDROID_UNKNOWN_TIME) {
544136965c92c625d52a6cbad42f82a2091d7769c9cGlenn Kasten            *pPosMsec = 0;
54535a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        } else {
54635a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi            *pPosMsec = (XAmillisecond)pos;
54735a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        }
54835a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    } break;
54935a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi
55035a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi    default:
55135a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        // we shouldn't be here
55235a5a30fdad179ccf38d8d756590411326159a89Jean-Michel Trivi        assert(false);
55370c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi        break;
55470c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi    }
55570c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi
55670c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi    return result;
55770c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi}
55870c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi
55937dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
56037dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi//-----------------------------------------------------------------------------
56137dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi/**
56299b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten * pre-condition: mp != NULL
56337dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi */
56499b927751677abfb60a388d65dfeed1fed1db12cGlenn Kastenvoid android_Player_volumeUpdate(CMediaPlayer* mp)
56537dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi{
56699b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten    android::GenericPlayer* avp = mp->mAVPlayer.get();
56799b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten    if (avp != NULL) {
56899b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        float volumes[2];
56999b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        // MediaPlayer does not currently support EffectSend or MuteSolo
57099b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        android_player_volumeUpdate(volumes, &mp->mVolume, mp->mNumChannels, 1.0f, NULL);
57199b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        float leftVol = volumes[0], rightVol = volumes[1];
57299b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten        avp->setVolume(leftVol, rightVol);
57399b927751677abfb60a388d65dfeed1fed1db12cGlenn Kasten    }
57437dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi}
57537dc2fccf3f122b79ebd554de209d0a3c94ae161Jean-Michel Trivi
5760e47a0657162bbff48886ea2f5c68d9edb607768Jean-Michel Trivi//-----------------------------------------------------------------------------
5770e47a0657162bbff48886ea2f5c68d9edb607768Jean-Michel Trivi/**
57836b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel Trivi * pre-condition: gp != 0
579f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi */
58036b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel TriviXAresult android_Player_setPlayState(const android::sp<android::GenericPlayer> &gp,
58136b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel Trivi        SLuint32 playState,
582b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi        AndroidObjectState* pObjState)
583eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi{
584eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    XAresult result = XA_RESULT_SUCCESS;
585b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi    AndroidObjectState objState = *pObjState;
586eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
587eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    switch (playState) {
588eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi     case SL_PLAYSTATE_STOPPED: {
589f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi         SL_LOGV("setting AVPlayer to SL_PLAYSTATE_STOPPED");
59036b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel Trivi         gp->stop();
59197876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi         }
59297876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi         break;
593eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi     case SL_PLAYSTATE_PAUSED: {
594f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi         SL_LOGV("setting AVPlayer to SL_PLAYSTATE_PAUSED");
595eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         switch(objState) {
596e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi         case ANDROID_UNINITIALIZED:
597581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi             *pObjState = ANDROID_PREPARING;
59836b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel Trivi             gp->prepare();
599581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi             break;
600e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi         case ANDROID_PREPARING:
601eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi             break;
602e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi         case ANDROID_READY:
60336b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel Trivi             gp->pause();
604581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi             break;
605eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         default:
606f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi             SL_LOGE("Android object in invalid state");
607eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi             break;
608eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         }
60997876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi         }
61097876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi         break;
611eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi     case SL_PLAYSTATE_PLAYING: {
612f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi         SL_LOGV("setting AVPlayer to SL_PLAYSTATE_PLAYING");
613eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         switch(objState) {
614e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi         case ANDROID_UNINITIALIZED:
615581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi             *pObjState = ANDROID_PREPARING;
61636b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel Trivi             gp->prepare();
617581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi             // intended fall through
618e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi         case ANDROID_PREPARING:
619581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi             // intended fall through
620e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi         case ANDROID_READY:
62136b700a829b7a02b873b4cd0cdb0a95342b20a31Jean-Michel Trivi             gp->play();
622581a0f550f15f6fc22199cb85775a220f668b480Jean-Michel Trivi             break;
623eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         default:
624f271eea20f9fff6c101213b34652399f457bcd50Jean-Michel Trivi             SL_LOGE("Android object in invalid state");
625eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi             break;
626eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         }
62797876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi         }
62897876858aa17c7f24c6a1d60be09a57bc1824ba3Jean-Michel Trivi         break;
629eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi     default:
630eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         // checked by caller, should not happen
631eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi         break;
632eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi     }
633eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
634eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    return result;
635eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi}
636eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
637eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
63892b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi/**
63992b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi * pre-condition: mp != NULL
64092b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi */
64192b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel TriviXAresult android_Player_seek(CMediaPlayer *mp, SLmillisecond posMsec) {
64292b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    XAresult result = XA_RESULT_SUCCESS;
64392b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    switch (mp->mAndroidObjType) {
64492b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      case AUDIOVIDEOPLAYER_FROM_URIFD:
64592b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        if (mp->mAVPlayer !=0) {
64692b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi            mp->mAVPlayer->seek(posMsec);
64792b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        }
64892b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        break;
64992b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      case AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE: // intended fall-through
65092b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      default: {
651f5ff1a75f55677163bd9a8bd804e8f5c33ef592cGlenn Kasten          result = XA_RESULT_FEATURE_UNSUPPORTED;
65292b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      }
65392b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    }
65492b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    return result;
65592b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi}
65692b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi
65792b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi
65892b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi/**
65992b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi * pre-condition: mp != NULL
66092b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi */
66192b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel TriviXAresult android_Player_loop(CMediaPlayer *mp, SLboolean loopEnable) {
66292b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    XAresult result = XA_RESULT_SUCCESS;
66392b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    switch (mp->mAndroidObjType) {
66492b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      case AUDIOVIDEOPLAYER_FROM_URIFD:
66592b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        if (mp->mAVPlayer !=0) {
66692b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi            mp->mAVPlayer->loop(loopEnable);
66792b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        }
66892b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi        break;
66992b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      case AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE: // intended fall-through
67092b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      default: {
671f5ff1a75f55677163bd9a8bd804e8f5c33ef592cGlenn Kasten          result = XA_RESULT_FEATURE_UNSUPPORTED;
67292b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi      }
67392b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    }
67492b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi    return result;
67592b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi}
67692b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi
67792b245bf8828db9e469febebbe8774c00570b5b9Jean-Michel Trivi
678eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi//-----------------------------------------------------------------------------
679e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivivoid android_Player_androidBufferQueue_clear_l(CMediaPlayer *mp) {
680b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi    if ((mp->mAndroidObjType == AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE)
681b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi            && (mp->mAVPlayer != 0)) {
682e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi        android::StreamPlayer* splr = static_cast<android::StreamPlayer*>(mp->mAVPlayer.get());
683e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi        splr->appClear_l();
684e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi    }
685e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi}
686e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi
687e7bfcdc183454ec959ff51342f0973cabba219b2Jean-Michel Trivi
68870c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivivoid android_Player_androidBufferQueue_onRefilled_l(CMediaPlayer *mp) {
689b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi    if ((mp->mAndroidObjType == AUDIOVIDEOPLAYER_FROM_TS_ANDROIDBUFFERQUEUE)
690b05ea38e5131001884aa226f90fd50cf594a23f3Jean-Michel Trivi            && (mp->mAVPlayer != 0)) {
69170c49ae2867094072a4365423417ea452bf82231Jean-Michel Trivi        android::StreamPlayer* splr = static_cast<android::StreamPlayer*>(mp->mAVPlayer.get());
692bc0e642e6c1a51b3ae3a02d490d94b03e718e6b5Jean-Michel Trivi        splr->queueRefilled();
693eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi    }
694eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi}
695eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
696eae4df541ba1d46f65d37e959baf2127aa632c93Jean-Michel Trivi
69735ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten/*
69835ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten *  pre-conditions:
69935ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten *      mp != NULL
70035ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten *      mp->mAVPlayer != 0 (player is realized)
70135ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten *      nativeWindow can be NULL, but if NULL it is treated as an error
70235ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten */
70335ac702ee1ad91e5c8748c12450222d50b366a52Glenn KastenSLresult android_Player_setNativeWindow(CMediaPlayer *mp, ANativeWindow *nativeWindow)
70435ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten{
70535ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    assert(mp != NULL);
70635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    assert(mp->mAVPlayer != 0);
70735ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    if (nativeWindow == NULL) {
70835ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        SL_LOGE("ANativeWindow is NULL");
70935ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        return SL_RESULT_PARAMETER_INVALID;
71035ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    }
71135ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    SLresult result;
71235ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    int err;
71335ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    int value;
71435ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    // this could crash if app passes in a bad parameter, but that's OK
71535ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    err = (*nativeWindow->query)(nativeWindow, NATIVE_WINDOW_CONCRETE_TYPE, &value);
71635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    if (0 != err) {
71735ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        SL_LOGE("Query NATIVE_WINDOW_CONCRETE_TYPE on ANativeWindow * %p failed; "
71835ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten                "errno %d", nativeWindow, err);
71935ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        result = SL_RESULT_PARAMETER_INVALID;
72035ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    } else {
72135ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        switch (value) {
72235ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        case NATIVE_WINDOW_SURFACE: {                // Surface
72335ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            SL_LOGV("Displaying on ANativeWindow of type NATIVE_WINDOW_SURFACE");
72435ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            android::sp<android::Surface> nativeSurface(
72535ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten                    static_cast<android::Surface *>(nativeWindow));
72635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            mp->mAVPlayer->setVideoSurface(nativeSurface);
72735ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            result = SL_RESULT_SUCCESS;
72835ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            } break;
72935ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        case NATIVE_WINDOW_SURFACE_TEXTURE_CLIENT: { // SurfaceTextureClient
73035ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            SL_LOGV("Displaying on ANativeWindow of type NATIVE_WINDOW_SURFACE_TEXTURE_CLIENT");
73135ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            android::sp<android::SurfaceTextureClient> surfaceTextureClient(
73235ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten                    static_cast<android::SurfaceTextureClient *>(nativeWindow));
73335ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            android::sp<android::ISurfaceTexture> nativeSurfaceTexture(
73435ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten                    surfaceTextureClient->getISurfaceTexture());
73535ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            mp->mAVPlayer->setVideoSurfaceTexture(nativeSurfaceTexture);
73635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            result = SL_RESULT_SUCCESS;
73735ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            } break;
73835ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        case NATIVE_WINDOW_FRAMEBUFFER:              // FramebufferNativeWindow
73935ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            // fall through
74035ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        default:
74135ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            SL_LOGE("ANativeWindow * %p has unknown or unsupported concrete type %d",
74235ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten                    nativeWindow, value);
74335ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            result = SL_RESULT_PARAMETER_INVALID;
74435ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten            break;
74535ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten        }
74635ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    }
74735ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten    return result;
74835ac702ee1ad91e5c8748c12450222d50b366a52Glenn Kasten}
749