3276 lines
112 KiB
C
Executable File
3276 lines
112 KiB
C
Executable File
/*
|
||
* Copyright (c) 2008-2016 Allwinner Technology Co. Ltd.
|
||
* All rights reserved.
|
||
*
|
||
* File : xplayer.c
|
||
* Description : xplayer
|
||
* History :
|
||
* Author : AL3
|
||
* Date : 2015/05/05
|
||
* Comment : first version
|
||
*
|
||
*/
|
||
|
||
#include "cdx_log.h"
|
||
#include "xplayer.h"
|
||
#include "demuxComponent.h"
|
||
#include <string.h>
|
||
#include <inttypes.h>
|
||
|
||
#include <version.h>
|
||
|
||
#include "player.h"
|
||
#include "demuxComponent.h"
|
||
#include "AwMessageQueue.h"
|
||
#include "layerControl.h"
|
||
#include "soundControl.h"
|
||
#include "subtitleControl.h"
|
||
#include "deinterlace.h"
|
||
#include "CdxParser.h"
|
||
#include <sys/time.h>
|
||
|
||
// pause then start to display, the cmcc apk call seekTo, it will flush the buffering cache(livemod)
|
||
// or seek some frames in cache ( vod) , it will discontinue
|
||
#define PAUSE_THEN_SEEK_BUG (1)
|
||
|
||
//* player status.
|
||
static const int XPLAYER_STATUS_IDLE = 0;
|
||
static const int XPLAYER_STATUS_INITIALIZED = 1<<0;
|
||
static const int XPLAYER_STATUS_PREPARING = 1<<1;
|
||
static const int XPLAYER_STATUS_PREPARED = 1<<2;
|
||
static const int XPLAYER_STATUS_STARTED = 1<<3;
|
||
static const int XPLAYER_STATUS_PAUSED = 1<<4;
|
||
static const int XPLAYER_STATUS_STOPPED = 1<<5;
|
||
static const int XPLAYER_STATUS_COMPLETE = 1<<6;
|
||
static const int XPLAYER_STATUS_ERROR = 1<<7;
|
||
|
||
//* command id.
|
||
static const int XPLAYER_COMMAND_SET_SOURCE = 0x101;
|
||
static const int XPLAYER_COMMAND_SET_SURFACE = 0x102;
|
||
static const int XPLAYER_COMMAND_SET_AUDIOSINK = 0x103;
|
||
static const int XPLAYER_COMMAND_PREPARE = 0x104;
|
||
static const int XPLAYER_COMMAND_START = 0x105;
|
||
static const int XPLAYER_COMMAND_STOP = 0x106;
|
||
static const int XPLAYER_COMMAND_PAUSE = 0x107;
|
||
static const int XPLAYER_COMMAND_RESET = 0x108;
|
||
static const int XPLAYER_COMMAND_QUIT = 0x109;
|
||
static const int XPLAYER_COMMAND_SEEK = 0x10a;
|
||
static const int XPLAYER_COMMAND_RESETURL = 0x10b;
|
||
static const int XPLAYER_COMMAND_SETSPEED = 0x10c;
|
||
static const int XPLAYER_COMMAND_SET_SUBCTRL = 0x10d;
|
||
static const int XPLAYER_COMMAND_SET_DI = 0x10e;
|
||
static const int XPLAYER_COMMAND_SET_PLAYRATE = 0x10f;
|
||
|
||
typedef struct PlayerContext
|
||
{
|
||
AwMessageQueue* mMessageQueue;
|
||
Player* mPlayer;
|
||
DemuxComp* mDemux;
|
||
pthread_t mThreadId;
|
||
int mThreadCreated;
|
||
uid_t mUID; //* no use.
|
||
|
||
//* data source.
|
||
char* mSourceUrl; //* file path or network stream url.
|
||
CdxStreamT* mSourceStream; //* outside streaming source like miracast.
|
||
|
||
int mSourceFd; //* file descriptor.
|
||
int64_t mSourceFdOffset;
|
||
int64_t mSourceFdLength;
|
||
|
||
//* media information.
|
||
MediaInfo* mMediaInfo;
|
||
|
||
//* text codec format of the subtitle, used to transform subtitle text to
|
||
//* utf8 when the subtitle text codec format is unknown.
|
||
char mDefaultTextFormat[32];
|
||
|
||
//* whether enable subtitle show.
|
||
int mIsSubtitleDisable;
|
||
|
||
//* file descriptor of .idx file of index+sub subtitle.
|
||
//* we save the .idx file's fd here because application set .idx file and .sub file
|
||
//* seperately, we need to wait for the .sub file's fd, see
|
||
//* INVOKE_ID_ADD_EXTERNAL_SOURCE_FD command in invoke() method.
|
||
int mIndexFileHasBeenSet;
|
||
int mIndexFileFdOfIndexSubtitle;
|
||
|
||
//* for status and synchronize control.
|
||
int mStatus;
|
||
pthread_mutex_t mMutexMediaInfo; //* for media info protection.
|
||
//* for mStatus protection in start/stop/pause operation and complete/seek finish callback.
|
||
pthread_mutex_t mMutexStatus;
|
||
sem_t mSemSetDataSource;
|
||
sem_t mSemPrepare;
|
||
sem_t mSemStart;
|
||
sem_t mSemStop;
|
||
sem_t mSemPause;
|
||
sem_t mSemQuit;
|
||
sem_t mSemReset;
|
||
sem_t mSemSeek;
|
||
sem_t mSemSetSurface;
|
||
sem_t mSemSetAudioSink;
|
||
sem_t mSemPrepareFinish;//* for signal prepare finish, used in prepare().
|
||
sem_t mSemSetSpeed;
|
||
sem_t mSemSetSubCtrl;
|
||
sem_t mSemSetDeinterlace;
|
||
sem_t mSemSetPlayBackSettings;
|
||
|
||
//* status control.
|
||
int mSetDataSourceReply;
|
||
int mPrepareReply;
|
||
int mStartReply;
|
||
int mStopReply;
|
||
int mPauseReply;
|
||
int mResetReply;
|
||
int mSeekReply;
|
||
int mSetSurfaceReply;
|
||
int mSetAudioSinkReply;
|
||
int mPrepareFinishResult; //* save the prepare result for prepare().
|
||
int mSetSpeedReply;
|
||
int mSetSubCtrlReply;
|
||
int mSetDeinterlaceReply;
|
||
int mSetPlayBackSettingsReply;
|
||
|
||
int mPrepareSync; //* synchroized prarare() call, don't call back to user.
|
||
int mSeeking;
|
||
|
||
//* use to check whether seek callback is for current seek operation or previous.
|
||
int mSeekTime;
|
||
int mSeekSync; //* internal seek, don't call back to user.
|
||
int mLoop;
|
||
int mKeepLastFrame;
|
||
int mVideoSizeWidth; //* use to record videoSize which had send to app
|
||
int mVideoSizeHeight;
|
||
|
||
enum AwApplicationType mApplicationType;
|
||
|
||
void* mHTTPService;
|
||
|
||
//* record the id of subtitle which is displaying
|
||
//* we set the Nums to 64 .(32 may be not enough)
|
||
unsigned int mSubtitleDisplayIds[64];
|
||
int mSubtitleDisplayIdsUpdateIndex;
|
||
|
||
//* save the currentSelectTrackIndex;
|
||
int mCurrentSelectTrackIndex;
|
||
int mRawOccupyFlag;
|
||
|
||
int mLivemode;
|
||
int mPauseLivemode;
|
||
int mbIsDiagnose;
|
||
int64_t mPauseTimeStamp; //us
|
||
int64_t mShiftTimeStamp; //us
|
||
int mDisplayRatio;
|
||
int64_t mCurShiftTimeStamp; //us
|
||
int64_t mTimeShiftDuration; //ms
|
||
|
||
int mSeekTobug;
|
||
|
||
// the cmcc player should change pause state when buffer start, to fix getposition bug
|
||
int mDemuxNotifyPause;
|
||
int64_t mDemuxPauseTimeStamp;
|
||
|
||
//*
|
||
|
||
int mPreSeekTimeMs;
|
||
|
||
int mSpeed;
|
||
int mbFast;
|
||
int mFastTime;
|
||
int mSpeedThreadRuning;
|
||
|
||
int mScaledownFlag;
|
||
|
||
int mWidthScaledownRatio;
|
||
int mHeightScaledownRatio;
|
||
|
||
XPlayerNotifyCallback mCallback;
|
||
void* pUser;
|
||
}PlayerContext;
|
||
|
||
static void* XPlayerThread(void* arg);
|
||
static int ShiftTimeMode(int Shiftedms, char *buf);
|
||
static void clearMediaInfo(XPlayer* p);
|
||
static int callbackProcess(void *player, int messageId, void* param);
|
||
|
||
struct AwMessage {
|
||
AWMESSAGE_COMMON_MEMBERS
|
||
uintptr_t params[8];
|
||
};
|
||
|
||
XPlayer* XPlayerCreate()
|
||
{
|
||
logd("XPlayerCreate.");
|
||
XPlayer* mPriData;
|
||
LogVersionInfo();
|
||
mPriData = (PlayerContext*)malloc(sizeof(PlayerContext));
|
||
memset(mPriData,0x00,sizeof(PlayerContext));
|
||
|
||
mPriData->mUID = -1;
|
||
mPriData->mSourceUrl = NULL;
|
||
mPriData->mSourceFd = -1;
|
||
mPriData->mSourceFdOffset = 0;
|
||
mPriData->mSourceFdLength = 0;
|
||
mPriData->mSourceStream = NULL;
|
||
mPriData->mStatus = XPLAYER_STATUS_IDLE;
|
||
mPriData->mSeeking = 0;
|
||
mPriData->mSeekSync = 0;
|
||
mPriData->mLoop = 0;
|
||
mPriData->mKeepLastFrame = 0;
|
||
mPriData->mMediaInfo = NULL;
|
||
mPriData->mMessageQueue = NULL;
|
||
mPriData->mVideoSizeWidth = 0;
|
||
mPriData->mVideoSizeHeight= 0;
|
||
mPriData->mScaledownFlag =0;
|
||
mPriData->mWidthScaledownRatio = 0;
|
||
mPriData->mHeightScaledownRatio = 0;
|
||
mPriData->mCurrentSelectTrackIndex = -1;
|
||
|
||
mPriData->mDemuxNotifyPause = 0;
|
||
|
||
#if PAUSE_THEN_SEEK_BUG
|
||
mPriData->mSeekTobug = 0;
|
||
#endif
|
||
|
||
pthread_mutex_init(&mPriData->mMutexMediaInfo, NULL);
|
||
pthread_mutex_init(&mPriData->mMutexStatus, NULL);
|
||
sem_init(&mPriData->mSemSetDataSource, 0, 0);
|
||
sem_init(&mPriData->mSemPrepare, 0, 0);
|
||
sem_init(&mPriData->mSemStart, 0, 0);
|
||
sem_init(&mPriData->mSemStop, 0, 0);
|
||
sem_init(&mPriData->mSemPause, 0, 0);
|
||
sem_init(&mPriData->mSemReset, 0, 0);
|
||
sem_init(&mPriData->mSemQuit, 0, 0);
|
||
sem_init(&mPriData->mSemSeek, 0, 0);
|
||
sem_init(&mPriData->mSemSetSurface, 0, 0);
|
||
sem_init(&mPriData->mSemSetAudioSink, 0, 0);
|
||
sem_init(&mPriData->mSemSetSubCtrl, 0, 0);
|
||
sem_init(&mPriData->mSemSetDeinterlace, 0, 0);
|
||
|
||
sem_init(&mPriData->mSemPrepareFinish, 0, 0); //* for signal prepare finish, used in prepare().
|
||
sem_init(&mPriData->mSemSetSpeed, 0, 0);
|
||
sem_init(&mPriData->mSemSetPlayBackSettings, 0, 0);
|
||
|
||
mPriData->mMessageQueue = AwMessageQueueCreate(64, "XPlayer");
|
||
mPriData->mPlayer = PlayerCreate();
|
||
mPriData->mDemux = DemuxCompCreate();
|
||
|
||
if(mPriData->mPlayer != NULL)
|
||
PlayerSetCallback(mPriData->mPlayer, callbackProcess, (void*)mPriData);
|
||
|
||
if(mPriData->mDemux != NULL)
|
||
{
|
||
DemuxCompSetCallback(mPriData->mDemux, callbackProcess, (void*)mPriData);
|
||
DemuxCompSetPlayer(mPriData->mDemux, mPriData->mPlayer);
|
||
}
|
||
|
||
if(pthread_create(&mPriData->mThreadId, NULL, XPlayerThread, mPriData) == 0)
|
||
mPriData->mThreadCreated = 1;
|
||
else
|
||
mPriData->mThreadCreated = 0;
|
||
|
||
strcpy(mPriData->mDefaultTextFormat, "GBK");
|
||
|
||
mPriData->mIndexFileFdOfIndexSubtitle = -1;
|
||
mPriData->mIndexFileHasBeenSet = 0;
|
||
memset(mPriData->mSubtitleDisplayIds,0xff,64*sizeof(unsigned int));
|
||
mPriData->mSubtitleDisplayIdsUpdateIndex = 0;
|
||
mPriData->mApplicationType = APP_DEFAULT;
|
||
mPriData->mRawOccupyFlag = 0;
|
||
|
||
mPriData->mSpeed = 1;
|
||
mPriData->mbFast = 0;
|
||
mPriData->mFastTime = 0;
|
||
mPriData->mSpeedThreadRuning = 0;
|
||
|
||
return mPriData;
|
||
}
|
||
|
||
|
||
void XPlayerDestroy(XPlayer* p)
|
||
{
|
||
AwMessage msg;
|
||
XPlayer* mPriData;
|
||
int param_occupy[3] = {1,0,0};
|
||
int param_release[3] = {0,0,0};
|
||
logw("XPlayerDestroy");
|
||
|
||
mPriData = (XPlayer*)p;
|
||
if(mPriData->mThreadCreated)
|
||
{
|
||
void* status;
|
||
|
||
XPlayerReset(p); //* stop demux and player.
|
||
|
||
//* send a quit message to quit the main thread.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_QUIT;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemQuit;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemQuit, -1);
|
||
pthread_join(mPriData->mThreadId, &status);
|
||
}
|
||
|
||
if(mPriData->mDemux != NULL)
|
||
DemuxCompDestroy(mPriData->mDemux);
|
||
|
||
if(mPriData->mPlayer != NULL)
|
||
PlayerDestroy(mPriData->mPlayer);
|
||
|
||
if(mPriData->mMessageQueue != NULL)
|
||
{
|
||
AwMessageQueueDestroy(mPriData->mMessageQueue);
|
||
mPriData->mMessageQueue = NULL;
|
||
}
|
||
|
||
pthread_mutex_destroy(&mPriData->mMutexMediaInfo);
|
||
pthread_mutex_destroy(&mPriData->mMutexStatus);
|
||
sem_destroy(&mPriData->mSemSetDataSource);
|
||
sem_destroy(&mPriData->mSemPrepare);
|
||
sem_destroy(&mPriData->mSemStart);
|
||
sem_destroy(&mPriData->mSemStop);
|
||
sem_destroy(&mPriData->mSemPause);
|
||
sem_destroy(&mPriData->mSemReset);
|
||
sem_destroy(&mPriData->mSemQuit);
|
||
sem_destroy(&mPriData->mSemSeek);
|
||
sem_destroy(&mPriData->mSemSetSurface);
|
||
sem_destroy(&mPriData->mSemSetAudioSink);
|
||
sem_destroy(&mPriData->mSemPrepareFinish);
|
||
sem_destroy(&mPriData->mSemSetSpeed);
|
||
sem_destroy(&mPriData->mSemSetSubCtrl);
|
||
sem_destroy(&mPriData->mSemSetDeinterlace);
|
||
sem_destroy(&mPriData->mSemSetPlayBackSettings);
|
||
|
||
if(mPriData->mMediaInfo != NULL)
|
||
clearMediaInfo(p);
|
||
|
||
if(mPriData->mSourceUrl != NULL)
|
||
free(mPriData->mSourceUrl);
|
||
|
||
if(mPriData->mSourceFd != -1)
|
||
close(mPriData->mSourceFd);
|
||
|
||
if(mPriData->mIndexFileFdOfIndexSubtitle != -1)
|
||
close(mPriData->mIndexFileFdOfIndexSubtitle);
|
||
|
||
if (mPriData)
|
||
{
|
||
free(mPriData);
|
||
mPriData = NULL;
|
||
}
|
||
}
|
||
|
||
int XPlayerConfig(XPlayer* p, const XPlayerConfig_t *config)
|
||
{
|
||
p->mLivemode = config->livemode;
|
||
p->mApplicationType = config->appType;
|
||
return 0;
|
||
}
|
||
|
||
int XPlayerInitCheck(XPlayer* p)
|
||
{
|
||
logv("initCheck");
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
if(mPriData->mPlayer == NULL || mPriData->mDemux == NULL || mPriData->mThreadCreated == 0)
|
||
{
|
||
loge("initCheck() fail, XPlayer::mplayer = %p, XPlayer::mDemux = %p",
|
||
mPriData->mPlayer, mPriData->mDemux);
|
||
return -1;
|
||
}
|
||
else
|
||
return 0;
|
||
}
|
||
|
||
|
||
int XPlayerSetUID(XPlayer* p, int nUid)
|
||
{
|
||
(void)nUid;
|
||
(void)p;
|
||
logv("setUID(), uid = %d", nUid);
|
||
//mPriData->mUID = uid;
|
||
return 0;
|
||
}
|
||
|
||
int XPlayerSetHdcpOps(XPlayer* p, struct HdcpOpsS* pHdcp)
|
||
{
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
if(mPriData->mStatus != XPLAYER_STATUS_IDLE &&
|
||
mPriData->mStatus != XPLAYER_STATUS_INITIALIZED)
|
||
{
|
||
logw("set hdcp ops incorrect. status: %d", mPriData->mStatus);
|
||
return -1;
|
||
}
|
||
|
||
if(p->mDemux)
|
||
DemuxCompSetHdcpOps(p->mDemux, pHdcp);
|
||
return 0;
|
||
}
|
||
|
||
|
||
int XPlayerSetDataSourceUrl(XPlayer* p,
|
||
const char* pUrl,
|
||
void* httpService,
|
||
const CdxKeyedVectorT* pHeaders)
|
||
{
|
||
AwMessage msg;
|
||
int ret;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
if(pUrl == NULL)
|
||
{
|
||
loge("setDataSource(url), url=NULL");
|
||
return -1;
|
||
}
|
||
|
||
logd("setDataSource(url), url='%s'", pUrl);
|
||
|
||
DemuxCompSetLiveMode(mPriData->mDemux, p->mLivemode);
|
||
|
||
mPriData->mHTTPService = httpService;
|
||
PlayerConfigDispErrorFrame(mPriData->mPlayer, 0);
|
||
|
||
//* send a set data source message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_SET_SOURCE;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemSetDataSource;
|
||
msg.params[1] = (uintptr_t)&mPriData->mSetDataSourceReply;
|
||
msg.params[2] = SOURCE_TYPE_URL;
|
||
msg.params[3] = (uintptr_t)pUrl;
|
||
msg.params[4] = (uintptr_t)pHeaders;
|
||
msg.params[5] = (uintptr_t)(mPriData->mHTTPService);
|
||
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemSetDataSource, -1);
|
||
|
||
if(mPriData->mSetDataSourceReply != 0)
|
||
return mPriData->mSetDataSourceReply;
|
||
|
||
//* for local file, I think we should ack like file descriptor source,
|
||
//* so here we call prepare().
|
||
//* local file list: 'bdmv://---', '/---'
|
||
if (strncmp(pUrl, "bdmv://", 7) == 0 || strncmp(pUrl, "file://", 7) == 0 || pUrl[0] == '/')
|
||
{
|
||
//* for local file source set as a file descriptor,
|
||
//* the application will call invoke() by command INVOKE_ID_GET_TRACK_INFO
|
||
//* to get track info, so we need call prepare() here to obtain media information before
|
||
//* application call prepareAsync().
|
||
//* here I think for local file source set as a url, we should ack the same as the file
|
||
//* descriptor case.
|
||
ret = XPlayerPrepare(p);
|
||
if (ret != 0)
|
||
{
|
||
loge("prepare failure, ret(%d)", ret);
|
||
}
|
||
return ret;
|
||
}
|
||
else
|
||
return 0;
|
||
}
|
||
|
||
|
||
//* Warning: The filedescriptor passed into this method will only be valid until
|
||
//* the method returns, if you want to keep it, dup it!
|
||
int XPlayerSetDataSourceFd(XPlayer* p, int fd, int64_t offset, int64_t length)
|
||
{
|
||
AwMessage msg;
|
||
int ret;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
logv("setDataSource(fd), fd=%d", fd);
|
||
|
||
PlayerConfigDispErrorFrame(mPriData->mPlayer, 0);
|
||
//* send a set data source message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_SET_SOURCE;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemSetDataSource;
|
||
msg.params[1] = (uintptr_t)&mPriData->mSetDataSourceReply;
|
||
msg.params[2] = SOURCE_TYPE_FD;
|
||
msg.params[3] = fd;
|
||
msg.params[4] = (uintptr_t)(offset>>32); //* params[4] = high 32 bits of offset.
|
||
msg.params[5] = (uintptr_t)(offset & 0xffffffff);//* params[5] = low 32 bits of offset.
|
||
msg.params[6] = (uintptr_t)(length>>32); //* params[6] = high 32 bits of length.
|
||
msg.params[7] = (uintptr_t)(length & 0xffffffff);//* params[7] = low 32 bits of length.
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemSetDataSource, -1);
|
||
|
||
if(mPriData->mSetDataSourceReply != 0)
|
||
return mPriData->mSetDataSourceReply;
|
||
|
||
//* for local files, the application will call invoke() by command INVOKE_ID_GET_TRACK_INFO
|
||
//* to get track info, so we need call prepare() here to obtain media information before
|
||
//* application call prepareAsync().
|
||
ret = XPlayerPrepare(p);
|
||
if (ret != 0)
|
||
{
|
||
loge("prepare failure, ret(%d)", ret);
|
||
}
|
||
return ret;
|
||
}
|
||
|
||
int XPlayerSetDataSourceStream(XPlayer* p, const char* streamStr)
|
||
{
|
||
AwMessage msg;
|
||
logd("setDataSource(IStreamSource)");
|
||
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
PlayerConfigDispErrorFrame(mPriData->mPlayer, 1);
|
||
|
||
//* send a set data source message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_SET_SOURCE;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemSetDataSource;
|
||
msg.params[1] = (uintptr_t)&mPriData->mSetDataSourceReply;
|
||
msg.params[2] = SOURCE_TYPE_ISTREAMSOURCE;
|
||
msg.params[3] = (uintptr_t)streamStr;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemSetDataSource, -1);
|
||
return mPriData->mSetDataSourceReply;
|
||
}
|
||
|
||
//* set video layer control ops
|
||
int XPlayerSetVideoSurfaceTexture(XPlayer* p, const LayerCtrl* surfaceTexture)
|
||
{
|
||
AwMessage msg;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
logd("setVideoSurfaceTexture, surface = %p", surfaceTexture);
|
||
|
||
//* send a set surface message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_SET_SURFACE;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemSetSurface;
|
||
msg.params[1] = (uintptr_t)&mPriData->mSetSurfaceReply;
|
||
msg.params[2] = (uintptr_t)surfaceTexture;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemSetSurface, -1);
|
||
|
||
return mPriData->mSetSurfaceReply;
|
||
}
|
||
|
||
void XPlayerSetAudioSink(XPlayer* p, const SoundCtrl* audioSink)
|
||
{
|
||
AwMessage msg;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
logv("setAudioSink");
|
||
|
||
//* send a set audio sink message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_SET_AUDIOSINK;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemSetAudioSink;
|
||
msg.params[1] = (uintptr_t)&mPriData->mSetAudioSinkReply;
|
||
msg.params[2] = (uintptr_t)audioSink;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemSetAudioSink, -1);
|
||
|
||
return;
|
||
}
|
||
|
||
int XPlayerSetPlaybackSettings(XPlayer* p,const XAudioPlaybackRate *rate)
|
||
{
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
//* send a set set playback message.
|
||
AwMessage msg;
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_SET_PLAYRATE;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemSetPlayBackSettings;
|
||
msg.params[1] = (uintptr_t)&mPriData->mSetPlayBackSettingsReply;
|
||
msg.params[2] = (uintptr_t)rate;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemSetPlayBackSettings, -1);
|
||
|
||
if (rate->mSpeed == 0.f && mPriData->mStatus == XPLAYER_STATUS_STARTED)
|
||
{
|
||
logd("PlayerPause");
|
||
XPlayerPause(p);
|
||
}
|
||
else if (rate->mSpeed != 0.f && (mPriData->mStatus == XPLAYER_STATUS_PAUSED
|
||
|| mPriData->mStatus == XPLAYER_STATUS_PREPARED))
|
||
{
|
||
XPlayerStart(p);
|
||
}
|
||
|
||
return mPriData->mSetPlayBackSettingsReply;
|
||
}
|
||
|
||
int XPlayerGetPlaybackSettings(XPlayer* p,XAudioPlaybackRate *rate)
|
||
{
|
||
logd("XPlayerGetPlaybackSettings");
|
||
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
int ret = PlayerGetPlayBackSettings(mPriData->mPlayer,rate);
|
||
|
||
if (ret == 0)
|
||
{
|
||
if (mPriData->mStatus == XPLAYER_STATUS_PAUSED)
|
||
{
|
||
rate->mSpeed = 0.f;
|
||
}
|
||
}
|
||
return ret;
|
||
}
|
||
|
||
void XPlayerSetSubCtrl(XPlayer* p, const SubCtrl* subctrl)
|
||
{
|
||
AwMessage msg;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
logv("setSubRender");
|
||
//* send a set audio sink message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_SET_SUBCTRL;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemSetSubCtrl;
|
||
msg.params[1] = (uintptr_t)&mPriData->mSetSubCtrlReply;
|
||
msg.params[2] = (uintptr_t)subctrl;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemSetSubCtrl, -1);
|
||
|
||
return;
|
||
}
|
||
|
||
void XPlayerSetDeinterlace(XPlayer* p, const Deinterlace* di)
|
||
{
|
||
AwMessage msg;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
logd("set deinterlace");
|
||
//* send a set audio sink message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_SET_DI;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemSetDeinterlace;
|
||
msg.params[1] = (uintptr_t)&mPriData->mSetDeinterlaceReply;
|
||
msg.params[2] = (uintptr_t)di;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemSetDeinterlace, -1);
|
||
|
||
return;
|
||
}
|
||
|
||
int XPlayerSetNotifyCallback(XPlayer* p,
|
||
XPlayerNotifyCallback notifier,
|
||
void* pUserData)
|
||
{
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
mPriData->mCallback = notifier;
|
||
mPriData->pUser = pUserData;
|
||
|
||
return 0;
|
||
}
|
||
|
||
int XPlayerPrepareAsync(XPlayer* p)
|
||
{
|
||
AwMessage msg;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
logd("prepareAsync");
|
||
|
||
//* send a prepare.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_PREPARE;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemPrepare;
|
||
msg.params[1] = (uintptr_t)&mPriData->mPrepareReply;
|
||
msg.params[2] = 0;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemPrepare, -1);
|
||
|
||
return mPriData->mPrepareReply;
|
||
}
|
||
|
||
|
||
int XPlayerPrepare(XPlayer* p)
|
||
{
|
||
AwMessage msg;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
logd("prepare");
|
||
|
||
//* clear the mSemPrepareFinish semaphore.
|
||
while(sem_trywait(&mPriData->mSemPrepareFinish) == 0);
|
||
|
||
//* send a prepare message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_PREPARE;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemPrepare;
|
||
msg.params[1] = (uintptr_t)&mPriData->mPrepareReply;
|
||
msg.params[2] = 1; //* params[2] = mPrepareSync.
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemPrepare, -1);
|
||
|
||
if(mPriData->mPrepareReply == 0)
|
||
{
|
||
//* wait for the prepare finish.
|
||
SemTimedWait(&mPriData->mSemPrepareFinish, -1);
|
||
return mPriData->mPrepareFinishResult;
|
||
}
|
||
|
||
//* call DemuxCompPrepareAsync() fail, or status error.
|
||
return mPriData->mPrepareReply;
|
||
}
|
||
|
||
int XPlayerStart(XPlayer* p)
|
||
{
|
||
AwMessage msg;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
logd("start");
|
||
|
||
// in washu cmcc for livemode 1, we should call seekTo when paused longer than 30s
|
||
mPriData->mLivemode = DemuxCompGetLiveMode(mPriData->mDemux);
|
||
if(mPriData->mStatus == XPLAYER_STATUS_PAUSED &&
|
||
mPriData->mLivemode == 1 &&
|
||
mPriData->mApplicationType == APP_CMCC_WASU)
|
||
{
|
||
struct timeval tv;
|
||
gettimeofday(&tv, NULL);
|
||
int shiftTimeMs = (tv.tv_sec * 1000000ll + tv.tv_usec - mPriData->mPauseTimeStamp)/1000;
|
||
if(shiftTimeMs > 30*1000 && !mPriData->mSeekTobug && (mPriData->mPauseLivemode!=2))
|
||
{
|
||
logd("pause time longer than 30s in livemode 1, seekTo(%d)", shiftTimeMs);
|
||
XPlayerSeekTo(p, shiftTimeMs);
|
||
}
|
||
}
|
||
|
||
#if (PAUSE_THEN_SEEK_BUG)
|
||
if(mPriData->mSeekTobug && mPriData->mApplicationType == APP_CMCC_WASU)
|
||
{
|
||
mPriData->mSeekTobug = 0;
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_SEEK_COMPLETE, 0, 0);
|
||
}
|
||
#endif
|
||
|
||
//* send a start message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_START;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemStart;
|
||
msg.params[1] = (uintptr_t)&mPriData->mStartReply;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemStart, -1);
|
||
return mPriData->mStartReply;
|
||
}
|
||
|
||
|
||
int XPlayerStop(XPlayer* p)
|
||
{
|
||
AwMessage msg;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
logd("stop");
|
||
|
||
//* send a stop message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_STOP;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemStop;
|
||
msg.params[1] = (uintptr_t)&mPriData->mStopReply;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemStop, -1);
|
||
|
||
logd("=== stop end");
|
||
return mPriData->mStopReply;
|
||
}
|
||
|
||
|
||
int XPlayerPause(XPlayer* p)
|
||
{
|
||
AwMessage msg;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
logd("pause");
|
||
|
||
mPriData->mLivemode = DemuxCompGetLiveMode(mPriData->mDemux);
|
||
mPriData->mPauseLivemode = mPriData->mLivemode;
|
||
if(mPriData->mLivemode == 1)
|
||
{
|
||
struct timeval tv;
|
||
gettimeofday(&tv, NULL);
|
||
mPriData->mPauseTimeStamp = tv.tv_sec * 1000000ll + tv.tv_usec;
|
||
logd("livemode1, get current position inmPauseTimeStamp = %" PRId64 "",
|
||
mPriData->mPauseTimeStamp);
|
||
}
|
||
else if(mPriData->mLivemode == 2)
|
||
{
|
||
int msec;
|
||
XPlayerGetCurrentPosition(p, &msec);
|
||
logd("get current position in pause: %d", msec);
|
||
|
||
struct timeval tv;
|
||
gettimeofday(&tv, NULL);
|
||
mPriData->mPauseTimeStamp = tv.tv_sec * 1000000ll + tv.tv_usec - (int64_t)msec*1000;
|
||
logd("get current position in pause, nowUs = %ld, mPauseTimeStamp = %" PRId64 "",
|
||
tv.tv_sec * 1000000 + tv.tv_usec, mPriData->mPauseTimeStamp);
|
||
}
|
||
|
||
//* send a pause message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_PAUSE;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemPause;
|
||
msg.params[1] = (uintptr_t)&mPriData->mPauseReply;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemPause, -1);
|
||
return mPriData->mPauseReply;
|
||
}
|
||
|
||
int XPlayerSeekTo(XPlayer* p, int nSeekTimeMs)
|
||
{
|
||
AwMessage msg;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
logd("seekTo [%dms]", nSeekTimeMs);
|
||
|
||
mPriData->mLivemode = DemuxCompGetLiveMode(mPriData->mDemux);
|
||
|
||
#if (PAUSE_THEN_SEEK_BUG)
|
||
if((mPriData->mStatus == XPLAYER_STATUS_PAUSED
|
||
/*|| mPriData->mStatus == XPLAYER_STATUS_STARTED */) &&
|
||
mPriData->mApplicationType == APP_CMCC_WASU)
|
||
{
|
||
int pos;
|
||
XPlayerGetCurrentPosition(p, &pos);
|
||
int diff = pos - nSeekTimeMs;
|
||
logd("diff = %d", diff);
|
||
if(-1000 < diff && diff < 1000)
|
||
{
|
||
if(mPriData->mLivemode == 1 && mPriData->mStatus == XPLAYER_STATUS_PAUSED)
|
||
{
|
||
// if livemod1, we should change it to livemode2
|
||
struct timeval tv;
|
||
gettimeofday(&tv, NULL);
|
||
DemuxCompSetLiveMode(mPriData->mDemux, 2);
|
||
mPriData->mShiftTimeStamp = tv.tv_sec * 1000000ll + tv.tv_usec - nSeekTimeMs*1000ll;
|
||
}
|
||
mPriData->mSeekTobug = 1;
|
||
return XPlayerStart(p);
|
||
}
|
||
}
|
||
#endif
|
||
|
||
if(mPriData->mLivemode == 1 || mPriData->mLivemode == 2)
|
||
{
|
||
logd("++++++ reset url");
|
||
struct timeval tv;
|
||
gettimeofday(&tv, NULL);
|
||
mPriData->mShiftTimeStamp = tv.tv_sec * 1000000ll + tv.tv_usec;
|
||
mPriData->mPreSeekTimeMs = nSeekTimeMs + 20000; //* for timeshift
|
||
|
||
//* send a start message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_RESETURL;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemSeek;
|
||
msg.params[1] = (uintptr_t)&mPriData->mSeekReply;
|
||
msg.params[2] = nSeekTimeMs; //* params[2] = mSeekTime.
|
||
//* params[3] = mSeekSync.
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemSeek, -1);
|
||
}
|
||
else
|
||
{
|
||
logd("seek");
|
||
//* send a start message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_SEEK;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemSeek;
|
||
msg.params[1] = (uintptr_t)&mPriData->mSeekReply;
|
||
msg.params[2] = nSeekTimeMs; //* params[2] = mSeekTime.
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemSeek, -1);
|
||
|
||
}
|
||
|
||
return mPriData->mSeekReply;
|
||
}
|
||
|
||
int XPlayerReset(XPlayer* p)
|
||
{
|
||
AwMessage msg;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
logw("reset...");
|
||
|
||
//* send a start message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_RESET;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemReset;
|
||
msg.params[1] = (uintptr_t)&mPriData->mResetReply;
|
||
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemReset, -1);
|
||
return mPriData->mResetReply;
|
||
}
|
||
|
||
int XPlayerSetSpeed(XPlayer* p, int nSpeed)
|
||
{
|
||
AwMessage msg;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
logw("reset...");
|
||
mPriData->mSpeed = nSpeed;
|
||
//PlayerSetSpeed(mPriData->mPlayer, nSpeed);
|
||
|
||
if(nSpeed == 1)
|
||
{
|
||
mPriData->mbFast = 0;
|
||
mPriData->mSpeed = 1;
|
||
mPriData->mFastTime = 0;
|
||
XPlayerStart(p);
|
||
PlayerSetDiscardAudio(mPriData->mPlayer, 0);
|
||
return 0;
|
||
}
|
||
|
||
//* send a start message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_SETSPEED;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemSetSpeed;
|
||
msg.params[1] = (uintptr_t)&mPriData->mSetSpeedReply;
|
||
msg.params[2] = nSpeed;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
SemTimedWait(&mPriData->mSemSetSpeed, -1);
|
||
|
||
return mPriData->mSetSpeedReply;
|
||
}
|
||
|
||
int XPlayerIsPlaying(XPlayer* p)
|
||
{
|
||
logi("isPlaying");
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
if(mPriData->mStatus == XPLAYER_STATUS_STARTED ||
|
||
(mPriData->mStatus == XPLAYER_STATUS_COMPLETE && mPriData->mLoop != 0))
|
||
return 1;
|
||
|
||
return 0;
|
||
}
|
||
|
||
MediaInfo* XPlayerGetMediaInfo(XPlayer* p)
|
||
{
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
if(mPriData->mStatus == XPLAYER_STATUS_PREPARING ||
|
||
mPriData->mStatus == XPLAYER_STATUS_INITIALIZED||
|
||
mPriData->mStatus == XPLAYER_STATUS_IDLE )
|
||
{
|
||
loge("cannot get mediainfo in this status");
|
||
return NULL;
|
||
}
|
||
|
||
return mPriData->mMediaInfo;
|
||
}
|
||
|
||
int XPlayerGetCurrentPosition(XPlayer* p, int* msec)
|
||
{
|
||
int64_t nPositionUs;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
logv("getCurrentPosition");
|
||
|
||
if(mPriData->mStatus == XPLAYER_STATUS_PREPARED ||
|
||
mPriData->mStatus == XPLAYER_STATUS_STARTED ||
|
||
mPriData->mStatus == XPLAYER_STATUS_PAUSED ||
|
||
mPriData->mStatus == XPLAYER_STATUS_COMPLETE)
|
||
{
|
||
if(mPriData->mSeeking != 0)
|
||
{
|
||
*msec = mPriData->mSeekTime;
|
||
return 0;
|
||
}
|
||
|
||
mPriData->mLivemode = DemuxCompGetLiveMode(mPriData->mDemux);
|
||
if(mPriData->mLivemode == 1)
|
||
{
|
||
if(mPriData->mStatus == XPLAYER_STATUS_PAUSED)
|
||
{
|
||
struct timeval tv;
|
||
gettimeofday(&tv, NULL);
|
||
*msec = (tv.tv_sec * 1000000ll + tv.tv_usec - mPriData->mPauseTimeStamp)/1000;
|
||
}
|
||
else if(mPriData->mDemuxNotifyPause)
|
||
{
|
||
struct timeval tv;
|
||
gettimeofday(&tv, NULL);
|
||
*msec = (tv.tv_sec * 1000000ll + tv.tv_usec - mPriData->mDemuxPauseTimeStamp)/1000;
|
||
}
|
||
else
|
||
{
|
||
*msec = 0;
|
||
}
|
||
|
||
if(*msec < 0) *msec = 0;
|
||
|
||
return 0;
|
||
}
|
||
else if(mPriData->mLivemode == 2)
|
||
{
|
||
struct timeval tv;
|
||
gettimeofday(&tv, NULL);
|
||
if(mPriData->mStatus == XPLAYER_STATUS_PAUSED)
|
||
{
|
||
*msec = (tv.tv_sec * 1000000 + tv.tv_usec - mPriData->mPauseTimeStamp)/1000;
|
||
logd("livemode2, nowUs = %ld, mPauseTimeStamp = %" PRId64 "",
|
||
tv.tv_sec * 1000000 + tv.tv_usec, mPriData->mPauseTimeStamp);
|
||
}
|
||
else
|
||
{
|
||
//cmcc livemode2 only support ts
|
||
nPositionUs = PlayerGetPositionCMCC(mPriData->mPlayer);
|
||
logd("-- nPositionUs = %" PRId64 "", nPositionUs);
|
||
*msec = (tv.tv_sec * 1000000LL + tv.tv_usec -
|
||
mPriData->mShiftTimeStamp + nPositionUs)/1000;
|
||
}
|
||
|
||
if(*msec < 0)
|
||
{
|
||
logd("positon < 0 ,check it!!");
|
||
*msec = 0;
|
||
}
|
||
logd("livemode = 2, getCurrentPosition %d", *msec);
|
||
|
||
return 0;
|
||
}
|
||
|
||
//* in complete status, the prepare() method maybe called and it change the media info.
|
||
pthread_mutex_lock(&mPriData->mMutexMediaInfo);
|
||
if(mPriData->mMediaInfo != NULL)
|
||
{
|
||
//* ts stream's pts is not started at 0.
|
||
if(mPriData->mMediaInfo->eContainerType == CONTAINER_TYPE_TS ||
|
||
mPriData->mMediaInfo->eContainerType == CONTAINER_TYPE_BD ||
|
||
mPriData->mMediaInfo->eContainerType == CONTAINER_TYPE_HLS)
|
||
nPositionUs = PlayerGetPosition(mPriData->mPlayer);
|
||
else //* generally, stream pts is started at 0 except ts stream.
|
||
nPositionUs = PlayerGetPts(mPriData->mPlayer);
|
||
*msec = (nPositionUs + 500)/1000;
|
||
pthread_mutex_unlock(&mPriData->mMutexMediaInfo);
|
||
|
||
return 0;
|
||
}
|
||
else
|
||
{
|
||
loge("getCurrentPosition() fail, mMediaInfo==NULL.");
|
||
*msec = 0;
|
||
pthread_mutex_unlock(&mPriData->mMutexMediaInfo);
|
||
return 0;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
*msec = 0;
|
||
|
||
if(mPriData->mStatus == XPLAYER_STATUS_ERROR)
|
||
return -1;
|
||
else
|
||
return 0;
|
||
}
|
||
|
||
return -1;
|
||
}
|
||
|
||
|
||
int XPlayerGetDuration(XPlayer* p, int *msec)
|
||
{
|
||
logv("getDuration");
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
if(mPriData->mStatus == XPLAYER_STATUS_PREPARED ||
|
||
mPriData->mStatus == XPLAYER_STATUS_STARTED ||
|
||
mPriData->mStatus == XPLAYER_STATUS_PAUSED ||
|
||
mPriData->mStatus == XPLAYER_STATUS_STOPPED ||
|
||
mPriData->mStatus == XPLAYER_STATUS_COMPLETE)
|
||
{
|
||
//* in complete status, the prepare() method maybe called and it change the media info.
|
||
pthread_mutex_lock(&mPriData->mMutexMediaInfo);
|
||
if(mPriData->mMediaInfo != NULL)
|
||
*msec = mPriData->mMediaInfo->nDurationMs;
|
||
else
|
||
{
|
||
loge("getCurrentPosition() fail, mPriData->mMediaInfo==NULL.");
|
||
*msec = 0;
|
||
}
|
||
pthread_mutex_unlock(&mPriData->mMutexMediaInfo);
|
||
return 0;
|
||
}
|
||
|
||
loge("invalid getDuration() call, player not in valid status.");
|
||
return -1;
|
||
}
|
||
|
||
|
||
int XPlayerSetLooping(XPlayer* p, int loop)
|
||
{
|
||
logd("setLooping");
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
if(mPriData->mStatus == XPLAYER_STATUS_ERROR)
|
||
return -1;
|
||
|
||
mPriData->mLoop = loop;
|
||
return 0;
|
||
}
|
||
|
||
int updateVideoInfo(XPlayer* p)
|
||
{
|
||
//* get media information.
|
||
MediaInfo* mi;
|
||
VideoStreamInfo* vi;
|
||
int ret;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
pthread_mutex_lock(&mPriData->mMutexMediaInfo);
|
||
|
||
mi = DemuxCompGetMediaInfo(mPriData->mDemux);
|
||
if(mi == NULL)
|
||
{
|
||
loge("can not get media info from demux.");
|
||
pthread_mutex_unlock(&mPriData->mMutexMediaInfo);
|
||
return -1;
|
||
}
|
||
clearMediaInfo(p);
|
||
mPriData->mMediaInfo = mi;
|
||
|
||
#if !AWPLAYER_CONFIG_DISABLE_VIDEO
|
||
if(mi->pVideoStreamInfo != NULL)
|
||
{
|
||
#if !defined(CONF_3D_ENABLE)
|
||
if(mi->pVideoStreamInfo->bIs3DStream == 1)
|
||
mi->pVideoStreamInfo->bIs3DStream = 0;
|
||
#endif
|
||
|
||
ret = PlayerSetVideoStreamInfo(mPriData->mPlayer, mi->pVideoStreamInfo);
|
||
if(ret != 0)
|
||
{
|
||
logw("PlayerSetVideoStreamInfo() fail, video stream not supported.");
|
||
}
|
||
}
|
||
#endif
|
||
|
||
pthread_mutex_unlock(&mPriData->mMutexMediaInfo);
|
||
return 0;
|
||
}
|
||
|
||
/*
|
||
*ret 1 means need to scale down
|
||
* 0 is do nothing
|
||
* -1 is unknown error
|
||
* */
|
||
/*A80 general 4K -> do nothing (0)*/
|
||
/*H3/H8 general 4K -> Scale down (1)*/
|
||
/*H3 H265 4K -> do nothing (0)*/
|
||
int getScaledownFlag(const MediaInfo *mi)
|
||
{
|
||
int ret = 0; /* default not scaledown */
|
||
|
||
if(mi->pVideoStreamInfo->nWidth >= WIDTH_4K || mi->pVideoStreamInfo->nHeight >= HEIGHT_4K)
|
||
{
|
||
if (mi->pVideoStreamInfo->eCodecFormat == VIDEO_CODEC_FORMAT_H265)
|
||
{
|
||
#if !defined(CONF_H265_4K_P2P) /* only H3/H5 support H.265 p2p */
|
||
ret = 1;
|
||
#endif
|
||
}
|
||
else if (mi->pVideoStreamInfo->eCodecFormat == VIDEO_CODEC_FORMAT_H264)
|
||
{
|
||
#if !defined(CONF_H264_4K_P2P) /* only A80 support H.264 p2p */
|
||
ret = 1;
|
||
#endif
|
||
}
|
||
else /* all other codec format should scaledown for 4K size */
|
||
{
|
||
ret = 1;
|
||
}
|
||
}
|
||
|
||
return ret;
|
||
}
|
||
|
||
int XPlayerSetScaleDownRatio(XPlayer* p,int widthRatio,int heightRatio){
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
if((mPriData != NULL)&& (mPriData->mStatus==XPLAYER_STATUS_IDLE
|
||
|| mPriData->mStatus==XPLAYER_STATUS_INITIALIZED
|
||
|| mPriData->mStatus==XPLAYER_STATUS_PREPARING) ){
|
||
mPriData->mWidthScaledownRatio = widthRatio;
|
||
mPriData->mHeightScaledownRatio = heightRatio;
|
||
return 0;
|
||
}else{
|
||
if(mPriData != NULL){
|
||
loge("XPlayerSetScaleDownRatio err,mPriData->mStatus = %d",mPriData->mStatus);
|
||
}
|
||
return -1;
|
||
}
|
||
}
|
||
|
||
/*
|
||
*ret 0 means supported, -1 is unsupported.
|
||
* */
|
||
int checkVideoSupported(const MediaInfo *mi)
|
||
{
|
||
int ret = 0; //default is supported
|
||
|
||
/*A80/H8 H265 4K unsupported*/
|
||
/*H3/H5 H265 4K supported*/
|
||
if (mi->pVideoStreamInfo->nWidth >= WIDTH_4K ||
|
||
mi->pVideoStreamInfo->nHeight >= HEIGHT_4K)
|
||
{
|
||
#if !defined(CONF_H265_4K)
|
||
if (mi->pVideoStreamInfo->eCodecFormat == VIDEO_CODEC_FORMAT_H265)
|
||
{
|
||
loge("Not support H265 4K video !!");
|
||
ret = -1;
|
||
}
|
||
#endif
|
||
}
|
||
else if (mi->pVideoStreamInfo->nWidth >= WIDTH_1080P ||
|
||
mi->pVideoStreamInfo->nHeight >= HEIGHT_1080P)
|
||
{
|
||
/* for all chip, WMV1/WMV2/VP6 specs unsupported, play effect too bad... */
|
||
if (mi->pVideoStreamInfo->eCodecFormat == VIDEO_CODEC_FORMAT_WMV1
|
||
|| mi->pVideoStreamInfo->eCodecFormat == VIDEO_CODEC_FORMAT_WMV2
|
||
|| mi->pVideoStreamInfo->eCodecFormat == VIDEO_CODEC_FORMAT_VP6)
|
||
{
|
||
loge("Not support WMV1/WMV2/VP6 1080P video !!");
|
||
ret = -1;
|
||
}
|
||
}
|
||
else /* < 1080P */
|
||
{
|
||
/* we support this*/
|
||
}
|
||
|
||
logv("check video support ret = [%d]", ret);
|
||
return ret;
|
||
}
|
||
|
||
static int initializePlayer(XPlayer* p)
|
||
{
|
||
//* get media information.
|
||
MediaInfo* mi;
|
||
VideoStreamInfo* vi;
|
||
AudioStreamInfo* ai;
|
||
SubtitleStreamInfo* si;
|
||
int i;
|
||
int nDefaultAudioIndex;
|
||
int nDefaultSubtitleIndex;
|
||
int ret;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
pthread_mutex_lock(&mPriData->mMutexMediaInfo);
|
||
|
||
mi = DemuxCompGetMediaInfo(mPriData->mDemux);
|
||
if(mi == NULL)
|
||
{
|
||
loge("can not get media info from demux.");
|
||
pthread_mutex_unlock(&mPriData->mMutexMediaInfo);
|
||
return -1;
|
||
}
|
||
|
||
PlayerSetFirstPts(mPriData->mPlayer,mi->nFirstPts);
|
||
mPriData->mMediaInfo = mi;
|
||
if(mi->pVideoStreamInfo != NULL)
|
||
{
|
||
/*detect if support*/
|
||
if(checkVideoSupported(mi))
|
||
{
|
||
loge("this video is outof specs, unsupported.");
|
||
return -1;
|
||
}
|
||
if(mPriData->mWidthScaledownRatio == 0 && mPriData->mHeightScaledownRatio == 0){
|
||
/*if the app not set scaledownRatio,use the default*/
|
||
/*check if need scaledown*/
|
||
if(getScaledownFlag(mi)==1) /*1 means need to scale down*/
|
||
{
|
||
ret = PlayerConfigVideoScaleDownRatio(mPriData->mPlayer, 1, 1);
|
||
if(ret != 0)
|
||
{
|
||
logw("PlayerConfigVideoScaleDownRatio() fail, ret = %d",ret);
|
||
}
|
||
else
|
||
mPriData->mScaledownFlag = 1;
|
||
}
|
||
}else{
|
||
ret = PlayerConfigVideoScaleDownRatio(mPriData->mPlayer,mPriData->mWidthScaledownRatio,mPriData->mHeightScaledownRatio);
|
||
if(ret != 0)
|
||
{
|
||
logw("PlayerConfigVideoScaleDownRatio() fail, ret = %d",ret);
|
||
}
|
||
else
|
||
mPriData->mScaledownFlag = 1;
|
||
}
|
||
}
|
||
|
||
//* initialize the player.
|
||
#if !AWPLAYER_CONFIG_DISABLE_VIDEO
|
||
if(mi->pVideoStreamInfo != NULL)
|
||
{
|
||
#if !defined(CONF_3D_ENABLE)
|
||
if(mi->pVideoStreamInfo->bIs3DStream == 1)
|
||
mi->pVideoStreamInfo->bIs3DStream = 0;
|
||
#endif
|
||
//* set the rotation
|
||
int nRotateDegree;
|
||
int nRotation = atoi((const char*)mPriData->mMediaInfo->cRotation);
|
||
|
||
if(nRotation == 0)
|
||
nRotateDegree = 0;
|
||
else if(nRotation == 90)
|
||
nRotateDegree = 1;
|
||
else if(nRotation == 180)
|
||
nRotateDegree = 2;
|
||
else if(nRotation == 270)
|
||
nRotateDegree = 3;
|
||
else
|
||
nRotateDegree = 0;
|
||
|
||
ret = PlayerConfigVideoRotateDegree(mPriData->mPlayer, nRotateDegree);
|
||
if(ret != 0)
|
||
{
|
||
logw("PlayerConfigVideoRotateDegree() fail, ret = %d",ret);
|
||
}
|
||
|
||
//* set the video streamInfo
|
||
ret = PlayerSetVideoStreamInfo(mPriData->mPlayer, mi->pVideoStreamInfo);
|
||
if(ret != 0)
|
||
{
|
||
logw("PlayerSetVideoStreamInfo() fail, video stream not supported.");
|
||
}
|
||
|
||
ret = PlayerConfigDropDelayFrame(mPriData->mPlayer, 1);
|
||
if(ret != 0)
|
||
{
|
||
logw("PlayerConfigDropDelayFrame() fail.");
|
||
}
|
||
}
|
||
|
||
#endif
|
||
|
||
#if !AWPLAYER_CONFIG_DISABLE_AUDIO
|
||
if(mi->pAudioStreamInfo != NULL)
|
||
{
|
||
nDefaultAudioIndex = -1;
|
||
for(i=0; i<mi->nAudioStreamNum; i++)
|
||
{
|
||
if(PlayerCanSupportAudioStream(mPriData->mPlayer, &mi->pAudioStreamInfo[i]))
|
||
{
|
||
nDefaultAudioIndex = i;
|
||
break;
|
||
}
|
||
}
|
||
|
||
if(nDefaultAudioIndex < 0)
|
||
{
|
||
logw("no audio stream supported.");
|
||
nDefaultAudioIndex = 0;
|
||
}
|
||
|
||
ret = PlayerSetAudioStreamInfo(mPriData->mPlayer, mi->pAudioStreamInfo,
|
||
mi->nAudioStreamNum, nDefaultAudioIndex);
|
||
if(ret != 0)
|
||
{
|
||
logw("PlayerSetAudioStreamInfo() fail, audio stream not supported.");
|
||
}
|
||
}
|
||
#endif
|
||
|
||
if(PlayerHasVideo(mPriData->mPlayer) == 0 && PlayerHasAudio(mPriData->mPlayer) == 0)
|
||
{
|
||
loge("neither video nor audio stream can be played.");
|
||
pthread_mutex_unlock(&mPriData->mMutexMediaInfo);
|
||
return -1;
|
||
}
|
||
|
||
#if !AWPLAYER_CONFIG_DISABLE_SUBTITLE
|
||
//* set subtitle stream to the text decoder.
|
||
if(mi->pSubtitleStreamInfo != NULL)
|
||
{
|
||
nDefaultSubtitleIndex = -1;
|
||
for(i=0; i<mi->nSubtitleStreamNum; i++)
|
||
{
|
||
if(PlayerCanSupportSubtitleStream(mPriData->mPlayer, &mi->pSubtitleStreamInfo[i]))
|
||
{
|
||
nDefaultSubtitleIndex = i;
|
||
break;
|
||
}
|
||
}
|
||
|
||
if(nDefaultSubtitleIndex < 0)
|
||
{
|
||
logw("no subtitle stream supported.");
|
||
nDefaultSubtitleIndex = 0;
|
||
}
|
||
|
||
ret = PlayerSetSubtitleStreamInfo(mPriData->mPlayer,
|
||
mi->pSubtitleStreamInfo, mi->nSubtitleStreamNum, nDefaultSubtitleIndex);
|
||
if(ret != 0)
|
||
{
|
||
logw("PlayerSetSubtitleStreamInfo() fail, subtitle stream not supported.");
|
||
}
|
||
}
|
||
#endif
|
||
|
||
//* report not seekable.
|
||
if(mi->bSeekable == 0)
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_INFO,
|
||
AW_MEDIA_INFO_NOT_SEEKABLE, 0);
|
||
|
||
pthread_mutex_unlock(&mPriData->mMutexMediaInfo);
|
||
|
||
return 0;
|
||
}
|
||
|
||
|
||
static void clearMediaInfo(XPlayer* p)
|
||
{
|
||
int i;
|
||
VideoStreamInfo* v;
|
||
AudioStreamInfo* a;
|
||
SubtitleStreamInfo* s;
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
|
||
if(mPriData->mMediaInfo != NULL)
|
||
{
|
||
//* free video stream info.
|
||
if(mPriData->mMediaInfo->pVideoStreamInfo != NULL)
|
||
{
|
||
for(i=0; i<mPriData->mMediaInfo->nVideoStreamNum; i++)
|
||
{
|
||
v = &mPriData->mMediaInfo->pVideoStreamInfo[i];
|
||
if(v->pCodecSpecificData != NULL && v->nCodecSpecificDataLen > 0)
|
||
free(v->pCodecSpecificData);
|
||
}
|
||
free(mPriData->mMediaInfo->pVideoStreamInfo);
|
||
mPriData->mMediaInfo->pVideoStreamInfo = NULL;
|
||
}
|
||
|
||
//* free audio stream info.
|
||
if(mPriData->mMediaInfo->pAudioStreamInfo != NULL)
|
||
{
|
||
for(i=0; i<mPriData->mMediaInfo->nAudioStreamNum; i++)
|
||
{
|
||
a = &mPriData->mMediaInfo->pAudioStreamInfo[i];
|
||
if(a->pCodecSpecificData != NULL && a->nCodecSpecificDataLen > 0)
|
||
free(a->pCodecSpecificData);
|
||
}
|
||
free(mPriData->mMediaInfo->pAudioStreamInfo);
|
||
mPriData->mMediaInfo->pAudioStreamInfo = NULL;
|
||
}
|
||
|
||
//* free subtitle stream info.
|
||
if(mPriData->mMediaInfo->pSubtitleStreamInfo != NULL)
|
||
{
|
||
for(i=0; i<mPriData->mMediaInfo->nSubtitleStreamNum; i++)
|
||
{
|
||
s = &mPriData->mMediaInfo->pSubtitleStreamInfo[i];
|
||
if(s->pUrl != NULL)
|
||
{
|
||
free(s->pUrl);
|
||
s->pUrl = NULL;
|
||
}
|
||
if(s->fd >= 0)
|
||
{
|
||
close(s->fd);
|
||
s->fd = -1;
|
||
}
|
||
if(s->fdSub >= 0)
|
||
{
|
||
close(s->fdSub);
|
||
s->fdSub = -1;
|
||
}
|
||
}
|
||
free(mPriData->mMediaInfo->pSubtitleStreamInfo);
|
||
mPriData->mMediaInfo->pSubtitleStreamInfo = NULL;
|
||
}
|
||
|
||
//* free the media info.
|
||
free(mPriData->mMediaInfo);
|
||
mPriData->mMediaInfo = NULL;
|
||
}
|
||
|
||
return;
|
||
}
|
||
|
||
static void* sendSpeedCmdThread(void* arg)
|
||
{
|
||
PlayerContext* mPriData = (PlayerContext*)arg;
|
||
mPriData->mSpeedThreadRuning = 1;
|
||
|
||
sleep(1);
|
||
|
||
while(mPriData->mbFast)
|
||
{
|
||
if(mPriData->mSpeed == 0 ||
|
||
mPriData->mStatus == XPLAYER_STATUS_PAUSED ||
|
||
mPriData->mStatus == XPLAYER_STATUS_STOPPED ||
|
||
mPriData->mStatus == XPLAYER_STATUS_COMPLETE ||
|
||
mPriData->mStatus == XPLAYER_STATUS_ERROR )
|
||
{
|
||
break;
|
||
}
|
||
|
||
if (mPriData->mFastTime == 0 && mPriData->mSpeed < 0)
|
||
{
|
||
AwMessage msg;
|
||
//* send a pause message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_PAUSE;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemPause;
|
||
msg.params[1] = (uintptr_t)&mPriData->mPauseReply;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
break;
|
||
}
|
||
|
||
mPriData->mFastTime += mPriData->mSpeed*1000;
|
||
|
||
if (mPriData->mFastTime < 0)
|
||
mPriData->mFastTime = 0;
|
||
|
||
if(mPriData->mFastTime >= 0 && mPriData->mbFast )
|
||
{
|
||
AwMessage msg;
|
||
|
||
//* send a seek message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_SEEK;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemSeek;
|
||
msg.params[1] = (uintptr_t)&mPriData->mSeekReply;
|
||
msg.params[2] = mPriData->mFastTime; //* params[2] = mSeekTime.
|
||
msg.params[3] = 0; //* params[3] = mSeekSync.
|
||
logd("==== video key frame in fast mode, mFastTime: %d, mSpeed: %d",
|
||
mPriData->mFastTime, mPriData->mSpeed);
|
||
logd("current time %lld ms",CdxGetNowUs()/1000);
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
}
|
||
else if(mPriData->mFastTime <= 0)
|
||
{
|
||
PlayerSetDiscardAudio(mPriData->mPlayer, 0);
|
||
mPriData->mbFast = 0;
|
||
}
|
||
|
||
if(mPriData->mSpeed == 0 ||
|
||
mPriData->mStatus == XPLAYER_STATUS_PAUSED ||
|
||
mPriData->mStatus == XPLAYER_STATUS_STOPPED ||
|
||
mPriData->mStatus == XPLAYER_STATUS_COMPLETE ||
|
||
mPriData->mStatus == XPLAYER_STATUS_ERROR )
|
||
{
|
||
break;
|
||
}
|
||
sleep(1);
|
||
}
|
||
mPriData->mSpeedThreadRuning = 0;
|
||
pthread_exit(0);
|
||
}
|
||
|
||
static void* XPlayerThread(void* arg)
|
||
{
|
||
AwMessage msg;
|
||
int ret;
|
||
sem_t* pReplySem;
|
||
int* pReplyValue;
|
||
PlayerContext* mPriData = (PlayerContext*)arg;
|
||
|
||
while(1)
|
||
{
|
||
if(AwMessageQueueGetMessage(mPriData->mMessageQueue, &msg) < 0)
|
||
{
|
||
loge("get message fail.");
|
||
continue;
|
||
}
|
||
|
||
pReplySem = (sem_t*)msg.params[0];
|
||
pReplyValue = (int*)msg.params[1];
|
||
|
||
if(msg.messageId == XPLAYER_COMMAND_SET_SOURCE)
|
||
{
|
||
logi("process message XPLAYER_COMMAND_SET_SOURCE.");
|
||
//* check status.
|
||
if(mPriData->mStatus != XPLAYER_STATUS_IDLE &&
|
||
mPriData->mStatus != XPLAYER_STATUS_INITIALIZED)
|
||
{
|
||
loge("invalid setDataSource() operation, player not in IDLE or INITIALIZED status");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = -1;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
if((int)msg.params[2] == SOURCE_TYPE_URL)
|
||
{
|
||
CdxKeyedVectorT* pHeaders;
|
||
//* data source is a url string.
|
||
if(mPriData->mSourceUrl != NULL)
|
||
free(mPriData->mSourceUrl);
|
||
mPriData->mSourceUrl = strdup((char*)msg.params[3]);
|
||
pHeaders = (CdxKeyedVectorT*) msg.params[4];
|
||
|
||
ret = DemuxCompSetUrlSource(mPriData->mDemux,
|
||
(void*)msg.params[5], mPriData->mSourceUrl, pHeaders);
|
||
if(ret == 0)
|
||
{
|
||
mPriData->mStatus = XPLAYER_STATUS_INITIALIZED;
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = 0;
|
||
}
|
||
else
|
||
{
|
||
loge("DemuxCompSetUrlSource() return fail.");
|
||
mPriData->mStatus = XPLAYER_STATUS_IDLE;
|
||
free(mPriData->mSourceUrl);
|
||
mPriData->mSourceUrl = NULL;
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = -1;
|
||
}
|
||
}
|
||
else if((int)msg.params[2] == SOURCE_TYPE_FD)
|
||
{
|
||
//* data source is a file descriptor.
|
||
int fd;
|
||
int64_t nOffset;
|
||
int64_t nLength;
|
||
|
||
fd = msg.params[3];
|
||
nOffset = msg.params[4];
|
||
nOffset<<=32;
|
||
nOffset |= msg.params[5];
|
||
nLength = msg.params[6];
|
||
nLength<<=32;
|
||
nLength |= msg.params[7];
|
||
|
||
if(mPriData->mSourceFd != -1)
|
||
close(mPriData->mSourceFd);
|
||
|
||
mPriData->mSourceFd = dup(fd);
|
||
mPriData->mSourceFdOffset = nOffset;
|
||
mPriData->mSourceFdLength = nLength;
|
||
ret = DemuxCompSetFdSource(mPriData->mDemux,
|
||
mPriData->mSourceFd, mPriData->mSourceFdOffset,
|
||
mPriData->mSourceFdLength);
|
||
if(ret == 0)
|
||
{
|
||
mPriData->mStatus = XPLAYER_STATUS_INITIALIZED;
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
}
|
||
else
|
||
{
|
||
loge("DemuxCompSetFdSource() return fail.");
|
||
mPriData->mStatus = XPLAYER_STATUS_IDLE;
|
||
close(mPriData->mSourceFd);
|
||
mPriData->mSourceFd = -1;
|
||
mPriData->mSourceFdOffset = 0;
|
||
mPriData->mSourceFdLength = 0;
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)-1;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
//* data source is a IStreamSource interface.
|
||
char *uri = (char *)msg.params[3];
|
||
int ret;
|
||
void *handle;
|
||
ret = sscanf(uri, "customer://%p", &handle);
|
||
if (ret != 1)
|
||
{
|
||
CDX_LOGE("sscanf failure...(%s)", uri);
|
||
mPriData->mSourceStream = NULL;
|
||
}
|
||
else
|
||
{
|
||
mPriData->mSourceStream = (CdxStreamT *)handle;
|
||
}
|
||
ret = DemuxCompSetStreamSource(mPriData->mDemux, uri);
|
||
if(ret == 0)
|
||
{
|
||
mPriData->mStatus = XPLAYER_STATUS_INITIALIZED;
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
}
|
||
else
|
||
{
|
||
loge("DemuxCompSetStreamSource() return fail.");
|
||
mPriData->mStatus = XPLAYER_STATUS_IDLE;
|
||
if(mPriData->mSourceStream)
|
||
{
|
||
CdxStreamClose(mPriData->mSourceStream);
|
||
mPriData->mSourceStream = NULL;
|
||
}
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)-1;
|
||
}
|
||
}
|
||
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
} //* end XPLAYER_COMMAND_SET_SOURCE.
|
||
else if(msg.messageId == XPLAYER_COMMAND_SET_SURFACE)
|
||
{
|
||
LayerCtrl* lc;
|
||
|
||
logd("process message XPLAYER_COMMAND_SET_SURFACE.");
|
||
|
||
//* set native window before delete the old one.
|
||
//* because the player's render thread may use the old surface
|
||
//* before it receive the new surface.
|
||
lc = (LayerCtrl*)msg.params[2];
|
||
|
||
ret = PlayerSetWindow(mPriData->mPlayer, lc);
|
||
if(ret == 0)
|
||
{
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
}
|
||
else
|
||
{
|
||
loge("PlayerSetWindow() return fail.");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)-1;
|
||
}
|
||
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
} //* end XPLAYER_COMMAND_SET_SURFACE.
|
||
else if(msg.messageId == XPLAYER_COMMAND_SET_AUDIOSINK)
|
||
{
|
||
void* audioSink;
|
||
|
||
logv("process message XPLAYER_COMMAND_SET_AUDIOSINK.");
|
||
|
||
audioSink = (SoundCtrl*)msg.params[2];
|
||
PlayerSetAudioSink(mPriData->mPlayer, audioSink);
|
||
|
||
//* super class MediaPlayerInterface has mAudioSink.
|
||
//MediaPlayerInterface::setAudioSink(audioSink);
|
||
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
} //* end XPLAYER_COMMAND_SET_AUDIOSINK.
|
||
else if (msg.messageId == XPLAYER_COMMAND_SET_PLAYRATE)
|
||
{
|
||
XAudioPlaybackRate *rate;
|
||
rate = (XAudioPlaybackRate*)msg.params[2];
|
||
|
||
//do set playback setting in playback
|
||
int ret = PlayerSetPlayBackSettings(mPriData->mPlayer,rate);
|
||
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = ret;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
|
||
continue;
|
||
}//* end XPLAYER_COMMAND_SET_PLAYRATE.
|
||
else if(msg.messageId == XPLAYER_COMMAND_SET_SUBCTRL)
|
||
{
|
||
SubCtrl* subctrl;
|
||
|
||
logd("==== process message XPLAYER_COMMAND_SET_SUBCTRL.");
|
||
|
||
subctrl = (SubCtrl*)msg.params[2];
|
||
PlayerSetSubCtrl(mPriData->mPlayer, subctrl);
|
||
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
} //* end XPLAYER_COMMAND_SET_SUBCTRL.
|
||
else if(msg.messageId == XPLAYER_COMMAND_SET_DI)
|
||
{
|
||
Deinterlace* di;
|
||
|
||
logd("==== process message XPLAYER_COMMAND_SET_SUBCTRL.");
|
||
|
||
di = (Deinterlace*)msg.params[2];
|
||
PlayerSetDeinterlace(mPriData->mPlayer, di);
|
||
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
} //* end XPLAYER_COMMAND_SET_DI
|
||
else if(msg.messageId == XPLAYER_COMMAND_PREPARE)
|
||
{
|
||
logd("process message XPLAYER_COMMAND_PREPARE. mPriData->mStatus: %d",
|
||
mPriData->mStatus);
|
||
if(mPriData->mStatus == XPLAYER_STATUS_PREPARED)
|
||
{
|
||
//* for data source set by fd(file descriptor), the prepare() method
|
||
//* is called in setDataSource(), so the player is already in PREPARED
|
||
//* status, here we just notify a prepared message.
|
||
|
||
//* when app call prepareAsync(), we callback video-size to app here,
|
||
if(mPriData->mMediaInfo->pVideoStreamInfo != NULL)
|
||
{
|
||
if(mPriData->mMediaInfo->pVideoStreamInfo->nWidth !=
|
||
mPriData->mVideoSizeWidth
|
||
&& mPriData->mMediaInfo->pVideoStreamInfo->nHeight !=
|
||
mPriData->mVideoSizeHeight)
|
||
{
|
||
int nRotation;
|
||
|
||
nRotation = atoi((const char*)mPriData->mMediaInfo->cRotation);
|
||
if((nRotation%180)==0)//* when the rotation is 0 and 180
|
||
{
|
||
mPriData->mVideoSizeWidth =
|
||
mPriData->mMediaInfo->pVideoStreamInfo->nWidth;
|
||
mPriData->mVideoSizeHeight =
|
||
mPriData->mMediaInfo->pVideoStreamInfo->nHeight;
|
||
}
|
||
else
|
||
{
|
||
//* when the rotation is 90 and 270,
|
||
//* we should exchange nHeight and nwidth
|
||
mPriData->mVideoSizeWidth =
|
||
mPriData->mMediaInfo->pVideoStreamInfo->nHeight;
|
||
mPriData->mVideoSizeHeight =
|
||
mPriData->mMediaInfo->pVideoStreamInfo->nWidth;
|
||
}
|
||
|
||
logi("xxxxxxxxxx video size: width = %d, height = %d",
|
||
mPriData->mVideoSizeWidth, mPriData->mVideoSizeHeight);
|
||
|
||
int size[2];
|
||
size[0] = mPriData->mVideoSizeWidth;
|
||
size[1] = mPriData->mVideoSizeHeight;
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_SET_VIDEO_SIZE,
|
||
0,
|
||
size);
|
||
}
|
||
}
|
||
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_PREPARED, 0, 0);
|
||
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
if(mPriData->mStatus != XPLAYER_STATUS_INITIALIZED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_STOPPED)
|
||
{
|
||
logd("invalid prepareAsync() call, player not in initialized or stopped status.");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)-1;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
mPriData->mStatus = XPLAYER_STATUS_PREPARING;
|
||
mPriData->mPrepareSync = msg.params[2];
|
||
ret = DemuxCompPrepareAsync(mPriData->mDemux);
|
||
if(ret != 0)
|
||
{
|
||
loge("DemuxCompPrepareAsync return fail immediately.");
|
||
mPriData->mStatus = XPLAYER_STATUS_IDLE;
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)-1;
|
||
}
|
||
else
|
||
{
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
}
|
||
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
} //* end XPLAYER_COMMAND_PREPARE.
|
||
else if(msg.messageId == XPLAYER_COMMAND_SETSPEED)
|
||
{
|
||
logd("process message XPLAYER_COMMAND_SETSPEED.");
|
||
if(mPriData->mStatus != XPLAYER_STATUS_PREPARED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_STARTED )
|
||
{
|
||
logd("invalid start() call, player not in prepared, \
|
||
started, paused or complete status.");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)-1;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
if(mPriData->mMediaInfo == NULL || mPriData->mMediaInfo->bSeekable == 0)
|
||
{
|
||
if(mPriData->mMediaInfo == NULL)
|
||
{
|
||
loge("setspeed fail because mMediaInfo == NULL.");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = -1;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
else
|
||
{
|
||
loge("media not seekable. cannot fast");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = -1;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
}
|
||
|
||
if(mPriData->mSeeking)
|
||
{
|
||
DemuxCompCancelSeek(mPriData->mDemux);
|
||
mPriData->mSeeking = 0;
|
||
}
|
||
|
||
//* protect mSeeking and mSeekTime from being changed by the seek finish callback.
|
||
pthread_mutex_lock(&mPriData->mMutexStatus);
|
||
|
||
int curTime;
|
||
XPlayerGetCurrentPosition(mPriData, &curTime);
|
||
|
||
mPriData->mSeeking = 1;
|
||
mPriData->mbFast = 1;
|
||
mPriData->mFastTime = curTime + mPriData->mSpeed*1000;
|
||
mPriData->mSeekTime = mPriData->mFastTime;
|
||
mPriData->mSeekSync = 0;
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
|
||
//PlayerFastForeward(mPriData->mPlayer);
|
||
PlayerSetDiscardAudio(mPriData->mPlayer, 1);
|
||
|
||
if(PlayerGetStatus(mPriData->mPlayer) == PLAYER_STATUS_STOPPED)
|
||
{
|
||
//* if in prepared status, the player is in stopped status,
|
||
//* this will make the player not record the nSeekTime at PlayerReset() operation
|
||
//* called at seek finish callback.
|
||
PlayerStart(mPriData->mPlayer);
|
||
}
|
||
PlayerPause(mPriData->mPlayer);
|
||
DemuxCompSeekTo(mPriData->mDemux, mPriData->mSeekTime);
|
||
|
||
if (mPriData->mSpeedThreadRuning == 0)
|
||
{
|
||
pthread_t pid;
|
||
int ret = pthread_create(&pid, NULL, sendSpeedCmdThread, (void*)mPriData );
|
||
|
||
if (ret != 0)
|
||
{
|
||
loge("create sendSpeedCmdThread error!");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = 0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
}
|
||
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}//* end XPLAYER_COMMAND_SETSPEED.
|
||
else if(msg.messageId == XPLAYER_COMMAND_START)
|
||
{
|
||
logd("process message XPLAYER_COMMAND_START.");
|
||
if(mPriData->mStatus != XPLAYER_STATUS_PREPARED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_STARTED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_PAUSED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_COMPLETE)
|
||
{
|
||
logd("invalid start() call, player not in prepared, \
|
||
started, paused or complete status.");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)-1;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
if(mPriData->mbFast == 1)
|
||
{
|
||
PlayerSetDiscardAudio(mPriData->mPlayer, 0);
|
||
mPriData->mbFast = 0;
|
||
mPriData->mSpeed = 1;
|
||
mPriData->mFastTime = 0;
|
||
}
|
||
|
||
//* synchronize with the seek or complete callback and the status may be changed.
|
||
pthread_mutex_lock(&mPriData->mMutexStatus);
|
||
|
||
if(mPriData->mStatus == XPLAYER_STATUS_STARTED)
|
||
{
|
||
if(PlayerGetStatus(mPriData->mPlayer) == PLAYER_STATUS_PAUSED
|
||
&& mPriData->mSeeking == 0)
|
||
{
|
||
//* player is paused for buffering, start it.
|
||
//* see XPLAYER_MESSAGE_DEMUX_PAUSE_PLAYER callback message.
|
||
PlayerStart(mPriData->mPlayer);
|
||
DemuxCompStart(mPriData->mDemux);
|
||
}
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
logv("player already in started status.");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
if(mPriData->mSeeking)
|
||
{
|
||
//* player and demux will be started at the seek callback.
|
||
mPriData->mStatus = XPLAYER_STATUS_STARTED;
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
//* for complete status, we seek to the begin of the file.
|
||
if(mPriData->mStatus == XPLAYER_STATUS_COMPLETE)
|
||
{
|
||
AwMessage newMsg;
|
||
|
||
if(mPriData->mMediaInfo->bSeekable)
|
||
{
|
||
memset(&newMsg, 0, sizeof(AwMessage));
|
||
newMsg.messageId = XPLAYER_COMMAND_SEEK;
|
||
newMsg.params[0] = 0;
|
||
newMsg.params[1] = 0;
|
||
newMsg.params[2] = mPriData->mSeekTime; //* params[2] = mSeekTime.
|
||
newMsg.params[3] = 1; //* params[3] = mSeekSync.
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &newMsg);
|
||
|
||
mPriData->mStatus = XPLAYER_STATUS_STARTED;
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
else
|
||
{
|
||
//* post a stop message.
|
||
memset(&newMsg, 0, sizeof(AwMessage));
|
||
newMsg.messageId = XPLAYER_COMMAND_STOP;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &newMsg);
|
||
|
||
//* post a prepare message.
|
||
memset(&newMsg, 0, sizeof(AwMessage));
|
||
newMsg.messageId = XPLAYER_COMMAND_PREPARE;
|
||
newMsg.params[0] = 0;
|
||
newMsg.params[1] = 0;
|
||
newMsg.params[2] = 1; //* params[2] = mPrepareSync.
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &newMsg);
|
||
|
||
//* post a start message.
|
||
memset(&newMsg, 0, sizeof(AwMessage));
|
||
newMsg.messageId = XPLAYER_COMMAND_START;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &newMsg);
|
||
|
||
//* should I reply 0 to the user at this moment?
|
||
//* or just set the semaphore and reply variable to the start message to
|
||
//* make it reply when start message done?
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
}
|
||
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
|
||
if(mPriData->mApplicationType == APP_STREAMING)
|
||
{
|
||
PlayerFast(mPriData->mPlayer, 0);
|
||
}
|
||
else
|
||
{
|
||
PlayerStart(mPriData->mPlayer);
|
||
}
|
||
|
||
DemuxCompStart(mPriData->mDemux);
|
||
mPriData->mStatus = XPLAYER_STATUS_STARTED;
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
|
||
} //* end XPLAYER_COMMAND_START.
|
||
else if(msg.messageId == XPLAYER_COMMAND_STOP)
|
||
{
|
||
logv("process message XPLAYER_COMMAND_STOP.");
|
||
if(mPriData->mStatus != XPLAYER_STATUS_PREPARED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_STARTED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_PAUSED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_COMPLETE &&
|
||
mPriData->mStatus != XPLAYER_STATUS_STOPPED)
|
||
{
|
||
logd("invalid stop() call, player not in prepared, paused, \
|
||
started, stopped or complete status.");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)-1;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
if(mPriData->mStatus == XPLAYER_STATUS_STOPPED)
|
||
{
|
||
logv("player already in stopped status.");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
//* the prepare callback may happen at this moment.
|
||
//* so the mStatus may be changed to PREPARED asynchronizely.
|
||
if(mPriData->mStatus == XPLAYER_STATUS_PREPARING)
|
||
{
|
||
logw("stop() called at preparing status, cancel demux prepare.");
|
||
DemuxCompCancelPrepare(mPriData->mDemux);
|
||
}
|
||
|
||
if(mPriData->mSeeking)
|
||
{
|
||
DemuxCompCancelSeek(mPriData->mDemux);
|
||
mPriData->mSeeking = 0;
|
||
}
|
||
|
||
DemuxCompStop(mPriData->mDemux);
|
||
PlayerStop(mPriData->mPlayer);
|
||
PlayerClear(mPriData->mPlayer); //* clear all media information in player.
|
||
|
||
//*clear the mSubtitleDisplayIds
|
||
memset(mPriData->mSubtitleDisplayIds,0xff,64*sizeof(unsigned int));
|
||
mPriData->mSubtitleDisplayIdsUpdateIndex = 0;
|
||
|
||
mPriData->mStatus = XPLAYER_STATUS_STOPPED;
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
} //* end XPLAYER_COMMAND_STOP.
|
||
else if(msg.messageId == XPLAYER_COMMAND_PAUSE)
|
||
{
|
||
logv("process message XPLAYER_COMMAND_PAUSE.");
|
||
if(mPriData->mStatus != XPLAYER_STATUS_STARTED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_PAUSED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_COMPLETE)
|
||
{
|
||
logd("invalid pause() call, player not in started, paused or complete status.");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)-1;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
if(mPriData->mStatus == XPLAYER_STATUS_PAUSED)
|
||
{
|
||
logv("player already in paused or complete status.");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
//* sync with the seek, complete or pause_player/resume_player call back.
|
||
pthread_mutex_lock(&mPriData->mMutexStatus);
|
||
|
||
if(mPriData->mSeeking)
|
||
{
|
||
//* player and demux will be paused at the seek callback.
|
||
mPriData->mStatus = XPLAYER_STATUS_PAUSED;
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
PlayerPause(mPriData->mPlayer);
|
||
mPriData->mStatus = XPLAYER_STATUS_PAUSED;
|
||
|
||
//* sync with the seek, complete or pause_player/resume_player call back.
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
} //* end XPLAYER_COMMAND_PAUSE.
|
||
else if(msg.messageId == XPLAYER_COMMAND_RESET)
|
||
{
|
||
logv("process message XPLAYER_COMMAND_RESET.");
|
||
//* the prepare callback may happen at this moment.
|
||
//* so the mStatus may be changed to PREPARED asynchronizely.
|
||
if(mPriData->mStatus == XPLAYER_STATUS_PREPARING)
|
||
{
|
||
logw("reset() called at preparing status, cancel demux prepare.");
|
||
DemuxCompCancelPrepare(mPriData->mDemux);
|
||
}
|
||
|
||
if(mPriData->mSeeking)
|
||
{
|
||
DemuxCompCancelSeek(mPriData->mDemux);
|
||
mPriData->mSeeking = 0;
|
||
}
|
||
|
||
//* stop and clear the demux.
|
||
//* this will stop the seeking if demux is currently processing seeking message.
|
||
//* it will clear the data source keep inside, this is important for the IStreamSource.
|
||
DemuxCompStop(mPriData->mDemux);
|
||
DemuxCompClear(mPriData->mDemux);
|
||
|
||
//* stop and clear the player.
|
||
PlayerStop(mPriData->mPlayer);
|
||
PlayerClear(mPriData->mPlayer); //* it will clear media info config to the player.
|
||
|
||
//* clear suface.
|
||
if(mPriData->mKeepLastFrame == 0)
|
||
{
|
||
|
||
}
|
||
|
||
//* clear data source.
|
||
if(mPriData->mSourceUrl != NULL)
|
||
{
|
||
free(mPriData->mSourceUrl);
|
||
mPriData->mSourceUrl = NULL;
|
||
}
|
||
if(mPriData->mSourceFd != -1)
|
||
{
|
||
close(mPriData->mSourceFd);
|
||
mPriData->mSourceFd = -1;
|
||
mPriData->mSourceFdOffset = 0;
|
||
mPriData->mSourceFdLength = 0;
|
||
}
|
||
mPriData->mSourceStream = NULL;
|
||
|
||
//* clear media info.
|
||
clearMediaInfo(mPriData);
|
||
|
||
//* clear loop setting.
|
||
mPriData->mLoop = 0;
|
||
|
||
//* clear the mSubtitleDisplayIds
|
||
memset(mPriData->mSubtitleDisplayIds,0xff,64*sizeof(unsigned int));
|
||
mPriData->mSubtitleDisplayIdsUpdateIndex = 0;
|
||
|
||
//* set status to IDLE.
|
||
mPriData->mStatus = XPLAYER_STATUS_IDLE;
|
||
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
else if(msg.messageId == XPLAYER_COMMAND_SEEK)
|
||
{
|
||
logd("process message XPLAYER_COMMAND_SEEK.");
|
||
if(mPriData->mStatus != XPLAYER_STATUS_PREPARED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_STARTED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_PAUSED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_COMPLETE)
|
||
{
|
||
logd("invalid seekTo() call, player not in prepared, \
|
||
started, paused or complete status.");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)-1;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
//* the application will call seekTo() when player is in complete status.
|
||
//* after seekTo(), the player should still stay on complete status until
|
||
//* application call start().
|
||
//* cts test requires this implement.
|
||
if(mPriData->mStatus == XPLAYER_STATUS_COMPLETE)
|
||
{
|
||
//* protect mSeeking and mSeekTime from being changed by the seek finish callback.
|
||
pthread_mutex_lock(&mPriData->mMutexStatus);
|
||
mPriData->mSeekTime = msg.params[2];
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_SEEK_COMPLETE, 0, 0);
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
if(mPriData->mMediaInfo == NULL || mPriData->mMediaInfo->bSeekable == 0)
|
||
{
|
||
if(mPriData->mMediaInfo == NULL)
|
||
{
|
||
loge("seekTo fail because mMediaInfo == NULL.");
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_SEEK_COMPLETE, 0, 0);
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = -1;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
else
|
||
{
|
||
loge("media not seekable.");
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_SEEK_COMPLETE, 0, 0);
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = 0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
}
|
||
|
||
if(mPriData->mSeeking)
|
||
{
|
||
DemuxCompCancelSeek(mPriData->mDemux);
|
||
mPriData->mSeeking = 0;
|
||
}
|
||
|
||
//* protect mSeeking and mSeekTime from being changed by the seek finish callback.
|
||
pthread_mutex_lock(&mPriData->mMutexStatus);
|
||
mPriData->mSeeking = 1;
|
||
mPriData->mSeekTime = msg.params[2];
|
||
mPriData->mSeekSync = msg.params[3];
|
||
logv("seekTo %.2f secs", mPriData->mSeekTime / 1E3);
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
|
||
if(PlayerGetStatus(mPriData->mPlayer) == PLAYER_STATUS_STOPPED)
|
||
{
|
||
//* if in prepared status, the player is in stopped status,
|
||
//* this will make the player not record the nSeekTime at PlayerReset() operation
|
||
//* called at seek finish callback.
|
||
PlayerStart(mPriData->mPlayer);
|
||
}
|
||
PlayerPause(mPriData->mPlayer);
|
||
DemuxCompSeekTo(mPriData->mDemux, mPriData->mSeekTime);
|
||
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
} //* end XPLAYER_COMMAND_SEEK.
|
||
else if(msg.messageId == XPLAYER_COMMAND_RESETURL)
|
||
{
|
||
logd("process message XPLAYER_COMMAND_RESETURL.");
|
||
if(mPriData->mStatus != XPLAYER_STATUS_PREPARED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_STARTED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_PAUSED &&
|
||
mPriData->mStatus != XPLAYER_STATUS_COMPLETE)
|
||
{
|
||
logd("invalid seekTo() call, player not in prepared, started, \
|
||
paused or complete status.");
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)-1;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
|
||
if(mPriData->mSeeking)
|
||
{
|
||
DemuxCompCancelSeek(mPriData->mDemux);
|
||
mPriData->mSeeking = 0;
|
||
}
|
||
|
||
//* protect mSeeking and mSeekTime from being changed by the seek finish callback.
|
||
pthread_mutex_lock(&mPriData->mMutexStatus);
|
||
mPriData->mSeeking = 1;
|
||
mPriData->mSeekTime = msg.params[2];
|
||
mPriData->mSeekSync = msg.params[3];
|
||
logd("seekTo %.2f secs", mPriData->mSeekTime / 1E3);
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
|
||
if(PlayerGetStatus(mPriData->mPlayer) == PLAYER_STATUS_STOPPED)
|
||
{
|
||
//* if in prepared status, the player is in stopped status,
|
||
//* this will make the player not record the nSeekTime at PlayerReset() operation
|
||
//* called at seek finish callback.
|
||
PlayerStart(mPriData->mPlayer);
|
||
}
|
||
PlayerPause(mPriData->mPlayer);
|
||
DemuxCompSeekToResetUrl(mPriData->mDemux, mPriData->mSeekTime);
|
||
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
continue;
|
||
}
|
||
else if(msg.messageId == XPLAYER_COMMAND_QUIT)
|
||
{
|
||
if(pReplyValue != NULL)
|
||
*pReplyValue = (int)0;
|
||
if(pReplySem != NULL)
|
||
sem_post(pReplySem);
|
||
break; //* break the thread.
|
||
} //* end XPLAYER_COMMAND_QUIT.
|
||
else
|
||
{
|
||
logw("unknow message with id %d, ignore.", msg.messageId);
|
||
}
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
int XPlayerSwitchSubtitle(XPlayer* p, int nStreamIndex)
|
||
{
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
if(mPriData->mPlayer == NULL)
|
||
{
|
||
return -1;
|
||
}
|
||
|
||
return PlayerSwitchSubtitle(mPriData->mPlayer, nStreamIndex);
|
||
}
|
||
|
||
int XPlayerSwitchAudio(XPlayer* p, int nStreamIndex)
|
||
{
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
if(mPriData->mPlayer == NULL)
|
||
{
|
||
return -1;
|
||
}
|
||
|
||
return PlayerSwitchAudio(mPriData->mPlayer, nStreamIndex);
|
||
}
|
||
|
||
static int setSubtitleStream(PlayerContext* mPriData, int nStreamNum,
|
||
SubtitleStreamInfo* pNewStreamInfo)
|
||
{
|
||
int i;
|
||
SubtitleStreamInfo* pStreamInfo;
|
||
if(pNewStreamInfo == NULL || nStreamNum == 0)
|
||
{
|
||
loge("PlayerProbeSubtitleStreamInfo failed!");
|
||
return -1;
|
||
}
|
||
|
||
pthread_mutex_lock(&mPriData->mMutexMediaInfo);
|
||
|
||
//* set reference video size.
|
||
if(mPriData->mMediaInfo->pVideoStreamInfo != NULL)
|
||
{
|
||
for(i=0; i<nStreamNum; i++)
|
||
{
|
||
if(pNewStreamInfo[i].nReferenceVideoFrameRate == 0)
|
||
pNewStreamInfo[i].nReferenceVideoFrameRate =
|
||
mPriData->mMediaInfo->pVideoStreamInfo->nFrameRate;
|
||
if(pNewStreamInfo[i].nReferenceVideoHeight == 0 ||
|
||
pNewStreamInfo[i].nReferenceVideoWidth == 0)
|
||
{
|
||
pNewStreamInfo[i].nReferenceVideoHeight =
|
||
mPriData->mMediaInfo->pVideoStreamInfo->nHeight;
|
||
pNewStreamInfo[i].nReferenceVideoWidth =
|
||
mPriData->mMediaInfo->pVideoStreamInfo->nWidth;
|
||
}
|
||
}
|
||
}
|
||
|
||
//* add stream info to the mMediaInfo,
|
||
//* put external subtitle streams ahead of the embedded subtitle streams.
|
||
if(mPriData->mMediaInfo->pSubtitleStreamInfo == NULL)
|
||
{
|
||
mPriData->mMediaInfo->pSubtitleStreamInfo = pNewStreamInfo;
|
||
mPriData->mMediaInfo->nSubtitleStreamNum = nStreamNum;
|
||
pNewStreamInfo = NULL;
|
||
nStreamNum = 0;
|
||
}
|
||
else
|
||
{
|
||
pStreamInfo = (SubtitleStreamInfo*)malloc(sizeof(SubtitleStreamInfo)*
|
||
(mPriData->mMediaInfo->nSubtitleStreamNum + nStreamNum));
|
||
if(pStreamInfo == NULL)
|
||
{
|
||
loge("invode::INVOKE_ID_ADD_EXTERNAL_SOURCE fail, can not malloc memory.");
|
||
for(i=0; i<nStreamNum; i++)
|
||
{
|
||
if(pNewStreamInfo[i].pUrl != NULL)
|
||
{
|
||
free(pNewStreamInfo[i].pUrl);
|
||
pNewStreamInfo[i].pUrl = NULL;
|
||
}
|
||
|
||
if(pNewStreamInfo[i].fd >= 0)
|
||
{
|
||
close(pNewStreamInfo[i].fd);
|
||
pNewStreamInfo[i].fd = -1;
|
||
}
|
||
|
||
if(pNewStreamInfo[i].fdSub >= 0)
|
||
{
|
||
close(pNewStreamInfo[i].fdSub);
|
||
pNewStreamInfo[i].fdSub = -1;
|
||
}
|
||
}
|
||
|
||
free(pNewStreamInfo);
|
||
pthread_mutex_unlock(&mPriData->mMutexMediaInfo);
|
||
return -1;
|
||
}
|
||
|
||
//* make the internal subtitle in front of external subtitle
|
||
memcpy(&pStreamInfo[mPriData->mMediaInfo->nSubtitleStreamNum],
|
||
pNewStreamInfo, sizeof(SubtitleStreamInfo)*nStreamNum);
|
||
memcpy(pStreamInfo, mPriData->mMediaInfo->pSubtitleStreamInfo,
|
||
sizeof(SubtitleStreamInfo)*mPriData->mMediaInfo->nSubtitleStreamNum);
|
||
|
||
free(mPriData->mMediaInfo->pSubtitleStreamInfo);
|
||
mPriData->mMediaInfo->pSubtitleStreamInfo = pStreamInfo;
|
||
mPriData->mMediaInfo->nSubtitleStreamNum += nStreamNum;
|
||
|
||
free(pNewStreamInfo);
|
||
pNewStreamInfo = NULL;
|
||
nStreamNum = 0;
|
||
}
|
||
|
||
//* re-arrange the stream index.
|
||
for(i=0; i<mPriData->mMediaInfo->nSubtitleStreamNum; i++)
|
||
mPriData->mMediaInfo->pSubtitleStreamInfo[i].nStreamIndex = i;
|
||
|
||
//* set subtitle stream info to player again.
|
||
//* here mMediaInfo != NULL, so initializePlayer() had been called.
|
||
if(mPriData->mPlayer != NULL)
|
||
{
|
||
int nDefaultSubtitleIndex = -1;
|
||
for(i=0; i<mPriData->mMediaInfo->nSubtitleStreamNum; i++)
|
||
{
|
||
if(PlayerCanSupportSubtitleStream(mPriData->mPlayer,
|
||
&mPriData->mMediaInfo->pSubtitleStreamInfo[i]))
|
||
{
|
||
nDefaultSubtitleIndex = i;
|
||
break;
|
||
}
|
||
}
|
||
|
||
if(nDefaultSubtitleIndex < 0)
|
||
{
|
||
logw("no subtitle stream supported.");
|
||
nDefaultSubtitleIndex = 0;
|
||
}
|
||
|
||
if(0 != PlayerSetSubtitleStreamInfo(mPriData->mPlayer,
|
||
mPriData->mMediaInfo->pSubtitleStreamInfo,
|
||
mPriData->mMediaInfo->nSubtitleStreamNum,
|
||
nDefaultSubtitleIndex))
|
||
{
|
||
logw("PlayerSetSubtitleStreamInfo() fail, subtitle stream not supported.");
|
||
}
|
||
}
|
||
|
||
pthread_mutex_unlock(&mPriData->mMutexMediaInfo);
|
||
|
||
return 0;
|
||
}
|
||
|
||
|
||
int XPlayerSetExternalSubUrl(XPlayer* p, const char* fileName)
|
||
{
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
SubtitleStreamInfo* pStreamInfo;
|
||
SubtitleStreamInfo* pNewStreamInfo = NULL;
|
||
int nStreamNum = 0;
|
||
int i;
|
||
|
||
if(mPriData->mStatus != XPLAYER_STATUS_PREPARED || mPriData->mMediaInfo == NULL)
|
||
{
|
||
loge("can not add external text source, player not in prepared status \
|
||
or there is no media stream.");
|
||
return -1;
|
||
}
|
||
|
||
//* probe subtitle info
|
||
PlayerProbeSubtitleStreamInfo(fileName, &nStreamNum, &pNewStreamInfo);
|
||
|
||
if(pNewStreamInfo == NULL || nStreamNum == 0)
|
||
{
|
||
loge("PlayerProbeSubtitleStreamInfo failed!");
|
||
return -1;
|
||
}
|
||
|
||
return setSubtitleStream(mPriData, nStreamNum, pNewStreamInfo);
|
||
}
|
||
|
||
|
||
int XPlayerSetExternalSubFd(XPlayer* p, int fd, int64_t offset, int64_t len, int fdSub)
|
||
{
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
SubtitleStreamInfo* pStreamInfo;
|
||
SubtitleStreamInfo* pNewStreamInfo = NULL;
|
||
int nStreamNum = 0;
|
||
int i;
|
||
|
||
if(mPriData->mStatus != XPLAYER_STATUS_PREPARED || mPriData->mMediaInfo == NULL)
|
||
{
|
||
loge("can not add external text source, player not in prepared status \
|
||
or there is no media stream.");
|
||
return -1;
|
||
}
|
||
|
||
//* probe subtitle info
|
||
PlayerProbeSubtitleStreamInfoFd(fd, offset, len, &nStreamNum, &pNewStreamInfo);
|
||
|
||
if(nStreamNum > 0 && pNewStreamInfo[0].eCodecFormat == SUBTITLE_CODEC_IDXSUB)
|
||
{
|
||
if(fdSub >= 0)
|
||
{
|
||
//* for index+sub subtitle,
|
||
//* we set the .sub file's descriptor to pNewStreamInfo[i].fdSub.
|
||
for(i=0; i<nStreamNum; i++)
|
||
pNewStreamInfo[i].fdSub = dup(fdSub);
|
||
}
|
||
else
|
||
{
|
||
loge("index sub subtitle stream without .sub file fd.");
|
||
for(i=0; i<nStreamNum; i++)
|
||
{
|
||
if(pNewStreamInfo[i].fd >= 0)
|
||
{
|
||
close(pNewStreamInfo[i].fd);
|
||
pNewStreamInfo[i].fd = -1;
|
||
}
|
||
}
|
||
free(pNewStreamInfo);
|
||
pNewStreamInfo = NULL;
|
||
nStreamNum = 0;
|
||
}
|
||
}
|
||
|
||
//* fdSub is the file descriptor of .sub file of a index+sub subtitle.
|
||
if(fdSub >= 0)
|
||
{
|
||
//* close the file descriptor of .idx file, we dup it when
|
||
//* INVOKE_ID_ADD_EXTERNAL_SOURCE_FD is called to set the .idx file
|
||
if(fd >= 0)
|
||
close(fd);
|
||
}
|
||
|
||
if(pNewStreamInfo == NULL || nStreamNum == 0)
|
||
{
|
||
loge("PlayerProbeSubtitleStreamInfo failed!");
|
||
return -1;
|
||
}
|
||
|
||
return setSubtitleStream(mPriData, nStreamNum, pNewStreamInfo);
|
||
}
|
||
|
||
int XPlayerSetSubCharset(XPlayer* p, const char* strFormat)
|
||
{
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
if(strFormat != NULL && (strlen(strFormat) < 31))
|
||
{
|
||
strcpy(mPriData->mDefaultTextFormat, strFormat);
|
||
}
|
||
else
|
||
{
|
||
strcpy(mPriData->mDefaultTextFormat, "UTF-8");
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
int XPlayerGetSubCharset(XPlayer* p, char *charset)
|
||
{
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
if(mPriData->mPlayer == NULL)
|
||
{
|
||
return -1;
|
||
}
|
||
|
||
strcpy(charset, mPriData->mDefaultTextFormat);
|
||
|
||
return 0;
|
||
}
|
||
|
||
int XPlayerSetSubDelay(XPlayer* p, int nTimeMs)
|
||
{
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
if(mPriData->mPlayer!=NULL)
|
||
return PlayerSetSubtitleShowTimeAdjustment(mPriData->mPlayer, nTimeMs);
|
||
else
|
||
return -1;
|
||
}
|
||
|
||
int XPlayerGetSubDelay(XPlayer* p)
|
||
{
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
if(mPriData->mPlayer!=NULL)
|
||
return PlayerGetSubtitleShowTimeAdjustment(mPriData->mPlayer);
|
||
else
|
||
return -1;
|
||
}
|
||
|
||
static int callbackProcess(void *p, int messageId, void* param)
|
||
{
|
||
PlayerContext* mPriData = (PlayerContext*)p;
|
||
switch(messageId)
|
||
{
|
||
case DEMUX_NOTIFY_PREPARED:
|
||
{
|
||
uintptr_t tmpPtr = (uintptr_t)param;
|
||
int err = tmpPtr;
|
||
if(err != 0)
|
||
{
|
||
//* demux prepare return fail.
|
||
//* notify a media error event.
|
||
mPriData->mStatus = XPLAYER_STATUS_ERROR;
|
||
if(mPriData->mPrepareSync == 0)
|
||
{
|
||
if(err == DEMUX_ERROR_IO)
|
||
{
|
||
mPriData->mCallback(mPriData->pUser,
|
||
AWPLAYER_MEDIA_ERROR, AW_MEDIA_ERROR_IO, 0);
|
||
}
|
||
else if(err == DEMUX_ERROR_USER_CANCEL)
|
||
{
|
||
logd("preparing canceled by user");
|
||
}
|
||
else
|
||
{
|
||
mPriData->mCallback(mPriData->pUser,
|
||
AWPLAYER_MEDIA_ERROR, AW_MEDIA_ERROR_UNKNOWN, 0);
|
||
}
|
||
}
|
||
else
|
||
{
|
||
if(err == DEMUX_ERROR_IO)
|
||
mPriData->mPrepareFinishResult = AW_MEDIA_ERROR_IO;
|
||
else
|
||
mPriData->mPrepareFinishResult = -1;
|
||
sem_post(&mPriData->mSemPrepareFinish);
|
||
}
|
||
}
|
||
else
|
||
{
|
||
//* demux prepare success, initialize the player.
|
||
if(initializePlayer(p) == 0)
|
||
{
|
||
//* initialize player success, notify a prepared event.
|
||
mPriData->mStatus = XPLAYER_STATUS_PREPARED;
|
||
if(mPriData->mPrepareSync == 0)
|
||
{
|
||
|
||
if(mPriData->mMediaInfo->pVideoStreamInfo!=NULL)
|
||
{
|
||
if(mPriData->mMediaInfo->pVideoStreamInfo->nWidth !=
|
||
mPriData->mVideoSizeWidth
|
||
&& mPriData->mMediaInfo->pVideoStreamInfo->nHeight !=
|
||
mPriData->mVideoSizeHeight)
|
||
{
|
||
int nRotation;
|
||
|
||
nRotation = atoi((const char*)mPriData->mMediaInfo->cRotation);
|
||
if((nRotation%180)==0)//* when the rotation is 0 and 180
|
||
{
|
||
mPriData->mVideoSizeWidth =
|
||
mPriData->mMediaInfo->pVideoStreamInfo->nWidth;
|
||
mPriData->mVideoSizeHeight =
|
||
mPriData->mMediaInfo->pVideoStreamInfo->nHeight;
|
||
}
|
||
else
|
||
{
|
||
//* when the rotation is 90 and 270,
|
||
//* we should exchange nHeight and nwidth
|
||
mPriData->mVideoSizeWidth =
|
||
mPriData->mMediaInfo->pVideoStreamInfo->nHeight;
|
||
mPriData->mVideoSizeHeight =
|
||
mPriData->mMediaInfo->pVideoStreamInfo->nWidth;
|
||
}
|
||
logi("xxxxxxxxxx video size: width = %d, height = %d",
|
||
mPriData->mVideoSizeWidth, mPriData->mVideoSizeHeight);
|
||
int size[2];
|
||
size[0] = mPriData->mVideoSizeWidth;
|
||
size[1] = mPriData->mVideoSizeHeight;
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_SET_VIDEO_SIZE,
|
||
0, size);
|
||
}
|
||
else
|
||
{
|
||
//<2F><>ʱmVideoSizeWidth<74><68><EFBFBD><EFBFBD>0
|
||
logi("xxxxxxxxxx video size: width = 0, height = 0");
|
||
mPriData->mCallback(mPriData->pUser,
|
||
AWPLAYER_MEDIA_SET_VIDEO_SIZE, 0, (int []){0, 0});
|
||
}
|
||
}
|
||
else
|
||
{
|
||
//<2F><>ʱmVideoSizeWidth<74><68><EFBFBD><EFBFBD>0
|
||
logi("xxxxxxxxxx video size: width = 0, height = 0");
|
||
mPriData->mCallback(mPriData->pUser,
|
||
AWPLAYER_MEDIA_SET_VIDEO_SIZE, 0, (int []){0, 0});
|
||
}
|
||
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_PREPARED, 0, 0);
|
||
}
|
||
else
|
||
{
|
||
mPriData->mPrepareFinishResult = 0;
|
||
sem_post(&mPriData->mSemPrepareFinish);
|
||
}
|
||
}
|
||
else
|
||
{
|
||
//* initialize player fail, notify a media error event.
|
||
mPriData->mStatus = XPLAYER_STATUS_ERROR;
|
||
if(mPriData->mPrepareSync == 0)
|
||
mPriData->mCallback(mPriData->pUser,
|
||
AWPLAYER_MEDIA_ERROR, DEMUX_ERROR_UNKNOWN, 0);
|
||
else
|
||
{
|
||
mPriData->mPrepareFinishResult = -1;
|
||
sem_post(&mPriData->mSemPrepareFinish);
|
||
}
|
||
}
|
||
}
|
||
|
||
break;
|
||
}
|
||
|
||
case DEMUX_NOTIFY_EOS:
|
||
{
|
||
if(mPriData->mLivemode == 2) //* timeshift
|
||
{
|
||
int seekTimeMs = 0;
|
||
struct timeval tv;
|
||
gettimeofday(&tv, NULL);
|
||
mPriData->mCurShiftTimeStamp = tv.tv_sec * 1000000ll + tv.tv_usec;
|
||
logd("mCurShiftTimeStamp=%" PRId64 " ms", mPriData->mCurShiftTimeStamp/1000);
|
||
seekTimeMs = (mPriData->mCurShiftTimeStamp - mPriData->mShiftTimeStamp)/1000
|
||
- mPriData->mTimeShiftDuration + mPriData->mPreSeekTimeMs;
|
||
logd("===== seekTimeMs=%d ms, mPriData->mCurShiftTimeStamp=%" PRId64 " us,"
|
||
"mPriData->mShiftTimeStamp=%" PRId64 " us,"
|
||
" mPriData->mTimeShiftDuration=%" PRId64 " ms, mPreSeekTimeMs=%d ms",
|
||
seekTimeMs,
|
||
mPriData->mCurShiftTimeStamp, mPriData->mShiftTimeStamp,
|
||
mPriData->mTimeShiftDuration, mPriData->mPreSeekTimeMs);
|
||
XPlayerSeekTo(p, seekTimeMs);
|
||
}
|
||
else
|
||
{
|
||
logw("eos...");
|
||
PlayerSetEos(mPriData->mPlayer);
|
||
}
|
||
break;
|
||
}
|
||
|
||
case DEMUX_NOTIFY_IOERROR:
|
||
{
|
||
loge("io error...");
|
||
//* should we report a MEDIA_INFO event of "MEDIA_INFO_NETWORK_ERROR" and
|
||
//* try reconnect for sometimes before a MEDIA_ERROR_IO event reported ?
|
||
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_ERROR, AW_MEDIA_ERROR_IO, 0);
|
||
break;
|
||
}
|
||
|
||
case DEMUX_NOTIFY_CACHE_STAT:
|
||
{
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_BUFFERING_UPDATE,
|
||
0, param);
|
||
break;
|
||
}
|
||
|
||
case DEMUX_NOTIFY_BUFFER_START:
|
||
{
|
||
logd("MEDIA_INFO_BUFFERING_START,mDisplayRatio = %d",mPriData->mDisplayRatio);
|
||
|
||
// for live mode 1, getposition when network broken
|
||
mPriData->mDemuxNotifyPause = 1;
|
||
struct timeval tv;
|
||
gettimeofday(&tv, NULL);
|
||
mPriData->mDemuxPauseTimeStamp = tv.tv_sec * 1000000ll + tv.tv_usec;
|
||
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_INFO,
|
||
AW_MEDIA_INFO_BUFFERING_START, param);
|
||
break;
|
||
}
|
||
|
||
case DEMUX_NOTIFY_BUFFER_END:
|
||
{
|
||
logd("MEDIA_INFO_BUFFERING_END ...");
|
||
|
||
mPriData->mDemuxNotifyPause = 0;
|
||
mPriData->mDemuxPauseTimeStamp = 0;
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_INFO,
|
||
AW_MEDIA_INFO_BUFFERING_END, 0);
|
||
break;
|
||
}
|
||
|
||
case DEMUX_NOTIFY_PAUSE_PLAYER:
|
||
{
|
||
logd("XPLAYER_MESSAGE_DEMUX_PAUSE_PLAYER, mStatus = %d", mPriData->mStatus);
|
||
//* be careful to check whether there is any player callback lock the mMutexStatus,
|
||
//* if so, the PlayerPause() call may fall into dead lock if the player
|
||
//* callback is requesting mMutexStatus.
|
||
//* currently we do not lock mMutexStatus in any player callback.
|
||
pthread_mutex_lock(&mPriData->mMutexStatus);
|
||
if(mPriData->mStatus == XPLAYER_STATUS_STARTED)
|
||
PlayerPause(mPriData->mPlayer);
|
||
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
break;
|
||
}
|
||
|
||
case DEMUX_NOTIFY_RESUME_PLAYER:
|
||
{
|
||
//* be careful to check whether there is any player callback lock the mMutexStatus,
|
||
//* if so, the PlayerPause() call may fall into dead lock if the player
|
||
//* callback is requesting mMutexStatus.
|
||
//* currently we do not lock mMutexStatus in any player callback.
|
||
pthread_mutex_lock(&mPriData->mMutexStatus);
|
||
if (mPriData->mStatus == XPLAYER_STATUS_STARTED /*&& bBuffering == false*/)
|
||
{
|
||
PlayerStart(mPriData->mPlayer);
|
||
}
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
break;
|
||
}
|
||
|
||
#if !(defined(CONF_NEW_BDMV_STREAM))
|
||
|
||
case DEMUX_IOREQ_ACCESS:
|
||
{
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_EXTEND_MEDIA_INFO,
|
||
AW_EX_IOREQ_ACCESS, param);
|
||
break;
|
||
}
|
||
|
||
case DEMUX_IOREQ_OPEN:
|
||
{
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_EXTEND_MEDIA_INFO,
|
||
AW_EX_IOREQ_OPEN, param);
|
||
break;
|
||
}
|
||
|
||
case DEMUX_IOREQ_OPENDIR:
|
||
{
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_EXTEND_MEDIA_INFO,
|
||
AW_EX_IOREQ_OPENDIR, param);
|
||
break;
|
||
}
|
||
|
||
case DEMUX_IOREQ_READDIR:
|
||
{
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_EXTEND_MEDIA_INFO,
|
||
AW_EX_IOREQ_READDIR, param);
|
||
break;
|
||
}
|
||
|
||
case DEMUX_IOREQ_CLOSEDIR:
|
||
{
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_EXTEND_MEDIA_INFO,
|
||
AW_EX_IOREQ_CLOSEDIR, param);
|
||
break;
|
||
}
|
||
|
||
#endif
|
||
|
||
case PLAYBACK_NOTIFY_EOS:
|
||
{
|
||
mPriData->mStatus = XPLAYER_STATUS_COMPLETE;
|
||
if(mPriData->mLoop == 0)
|
||
{
|
||
logd("player notify eos.");
|
||
mPriData->mSeekTime = 0; //* clear the seek flag.
|
||
mPriData->mbFast = 0;
|
||
mPriData->mSpeed = 1;
|
||
mPriData->mFastTime = 0;
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_PLAYBACK_COMPLETE, 0, 0);
|
||
}
|
||
else
|
||
{
|
||
AwMessage msg;
|
||
|
||
logv("player notify eos, loop is set, send start command.");
|
||
mPriData->mSeekTime = 0; //* seek to the file start and replay.
|
||
//* send a start message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_START;
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
}
|
||
break;
|
||
}
|
||
|
||
case PLAYBACK_NOTIFY_FIRST_PICTURE:
|
||
{
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_INFO,
|
||
AW_MEDIA_INFO_RENDERING_START, 0);
|
||
DemuxCompNotifyFirstFrameShowed(mPriData->mDemux);
|
||
|
||
break;
|
||
}
|
||
case DEMUX_NOTIFY_RESET_PLAYER:
|
||
{
|
||
logd("DEMUX_NOTIFY_RESET_PLAYER");
|
||
mPriData->mLivemode = DemuxCompGetLiveMode(mPriData->mDemux);
|
||
if(mPriData->mLivemode == 1)
|
||
{
|
||
mPriData->mSeekTime = 0;
|
||
}
|
||
pthread_mutex_lock(&mPriData->mMutexStatus);
|
||
PlayerReset(mPriData->mPlayer, ((int64_t)mPriData->mSeekTime)*1000);
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
break;
|
||
}
|
||
case DEMUX_NOTIFY_SEEK_FINISH:
|
||
{
|
||
logd("DEMUX_NOTIFY_SEEK_FINISH");
|
||
int seekResult;
|
||
int nSeekTimeMs;
|
||
int nFinalSeekTimeMs;
|
||
|
||
//* be careful to check whether there is any player callback lock the mMutexStatus,
|
||
//* if so, the PlayerPause() call may fall into dead lock if the player
|
||
//* callback is requesting mMutexStatus.
|
||
//* currently we do not lock mMutexStatus in any player callback.
|
||
pthread_mutex_lock(&mPriData->mMutexStatus);
|
||
|
||
seekResult = ((int*)param)[0];
|
||
nSeekTimeMs = ((int*)param)[1];
|
||
nFinalSeekTimeMs = ((int*)param)[2];
|
||
|
||
if (seekResult == 0)
|
||
{
|
||
PlayerReset(mPriData->mPlayer, ((int64_t)nFinalSeekTimeMs)*1000);
|
||
|
||
if (nSeekTimeMs == mPriData->mSeekTime)
|
||
{
|
||
mPriData->mSeeking = 0;
|
||
if (mPriData->mStatus == XPLAYER_STATUS_STARTED)
|
||
{
|
||
PlayerStart(mPriData->mPlayer);
|
||
DemuxCompStart(mPriData->mDemux);
|
||
}
|
||
}
|
||
else
|
||
{
|
||
logv("seek time not match, there may be another seek operation happening.");
|
||
}
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_SEEK_COMPLETE, 0, 0);
|
||
}
|
||
else if(seekResult == DEMUX_ERROR_USER_CANCEL)
|
||
{
|
||
// do nothing , do not start player
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_SEEK_COMPLETE, 0, 0);
|
||
}
|
||
else
|
||
{
|
||
pthread_mutex_unlock(&mPriData->mMutexStatus);
|
||
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_ERROR, AW_MEDIA_ERROR_IO, 0);
|
||
}
|
||
break;
|
||
}
|
||
|
||
case PLAYBACK_NOTIFY_SUBTITLE_ITEM_AVAILABLE:
|
||
{
|
||
|
||
//logd("subtitle available. id = %d, pSubtitleItem = %p",nSubtitleId,pSubtitleItem);
|
||
|
||
break;
|
||
}
|
||
|
||
case PLAYBACK_NOTIFY_SUBTITLE_ITEM_EXPIRED:
|
||
{
|
||
logd("subtitle expired.");
|
||
unsigned int nSubtitleId;
|
||
int i;
|
||
|
||
break;
|
||
}
|
||
|
||
case PLAYBACK_NOTIFY_VIDEO_SIZE:
|
||
{
|
||
int nWidth = ((int*)param)[0];
|
||
int nHeight = ((int*)param)[1];
|
||
logd("*************decoded nWidth = %d,nHeight = %d********",nWidth,nHeight);
|
||
int decodedSize[2];
|
||
decodedSize[0] = nWidth;
|
||
decodedSize[1] = nHeight;
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_DECODED_VIDEO_SIZE,
|
||
0, decodedSize);
|
||
//* if had scale down, we should zoom the widht and height
|
||
if(mPriData->mScaledownFlag == 1)
|
||
{
|
||
nWidth = 2*nWidth;
|
||
nHeight = 2*nHeight;
|
||
}
|
||
|
||
//if(nWidth != mVideoSizeWidth || nHeight != mVideoSizeHeight)
|
||
{
|
||
logi("xxxxxxxxxx video size : width = %d, height = %d", nWidth, nHeight);
|
||
|
||
mPriData->mVideoSizeWidth = nWidth;
|
||
mPriData->mVideoSizeHeight = nHeight;
|
||
int size[2];
|
||
size[0] = mPriData->mVideoSizeWidth;
|
||
size[1] = mPriData->mVideoSizeHeight;
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_SET_VIDEO_SIZE,
|
||
0, size);
|
||
}
|
||
break;
|
||
}
|
||
|
||
case PLAYBACK_NOTIFY_AUDIORAWPLAY:
|
||
{
|
||
break;
|
||
}
|
||
case PLAYBACK_NOTIFY_VIDEO_CROP:
|
||
//* TODO
|
||
break;
|
||
|
||
case PLAYBACK_NOTIFY_VIDEO_UNSUPPORTED:
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_ERROR,
|
||
AW_MEDIA_ERROR_UNSUPPORTED, 0);
|
||
|
||
break;
|
||
|
||
case PLAYBACK_NOTIFY_AUDIO_UNSUPPORTED:
|
||
if(mPriData->mMediaInfo->nVideoStreamNum == 0)
|
||
{
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_ERROR,
|
||
AW_MEDIA_ERROR_UNSUPPORTED, 0);
|
||
}
|
||
//* TODO
|
||
break;
|
||
|
||
case PLAYBACK_NOTIFY_SUBTITLE_UNSUPPORTED:
|
||
//* TODO
|
||
break;
|
||
|
||
case DEMUX_VIDEO_STREAM_CHANGE:
|
||
updateVideoInfo(p);
|
||
break;
|
||
case DEMUX_AUDIO_STREAM_CHANGE:
|
||
logw("it is not supported now.");
|
||
break;
|
||
|
||
case PLAYBACK_NOTIFY_SET_SECURE_BUFFER_COUNT:
|
||
|
||
if(mPriData->mDemux != NULL)
|
||
DemuxCompSetSecureBufferCount(mPriData->mDemux,param);
|
||
else
|
||
loge("the mDemux is null when set secure buffer count");
|
||
|
||
logw("it is not supported now.");
|
||
break;
|
||
|
||
case PLAYBACK_NOTIFY_SET_SECURE_BUFFERS:
|
||
|
||
if(mPriData->mDemux != NULL)
|
||
DemuxCompSetSecureBuffers(mPriData->mDemux,param);
|
||
else
|
||
loge("the mDemux is null when set secure buffers");
|
||
|
||
break;
|
||
|
||
case PLAYBACK_NOTIFY_VIDEO_RENDER_FRAME:
|
||
if(mPriData->mbFast)
|
||
{
|
||
logd("==== video key frame in fast mode, mFastTime: %d, mSpeed: %d",
|
||
mPriData->mFastTime, mPriData->mSpeed);
|
||
if(mPriData->mSpeed == 0)
|
||
{
|
||
break;
|
||
}
|
||
|
||
int sleepTime = (mPriData->mSpeed > 0) ? 2000000/mPriData->mSpeed :
|
||
(-2000000/mPriData->mSpeed);
|
||
//usleep(sleepTime);
|
||
|
||
mPriData->mFastTime += mPriData->mSpeed*1000;
|
||
if(mPriData->mFastTime > 0 && mPriData->mbFast)
|
||
{
|
||
AwMessage msg;
|
||
|
||
//* send a seek message.
|
||
memset(&msg, 0, sizeof(AwMessage));
|
||
msg.messageId = XPLAYER_COMMAND_SEEK;
|
||
msg.params[0] = (uintptr_t)&mPriData->mSemSeek;
|
||
msg.params[1] = (uintptr_t)&mPriData->mSeekReply;
|
||
msg.params[2] = mPriData->mFastTime; //* params[2] = mSeekTime.
|
||
msg.params[3] = 0; //* params[3] = mSeekSync.
|
||
|
||
AwMessageQueuePostMessage(mPriData->mMessageQueue, &msg);
|
||
}
|
||
else if(mPriData->mFastTime <= 0)
|
||
{
|
||
PlayerSetDiscardAudio(mPriData->mPlayer, 0);
|
||
mPriData->mbFast = 0;
|
||
}
|
||
}
|
||
break;
|
||
|
||
case DEMUX_NOTIFY_HLS_DOWNLOAD_START:
|
||
{
|
||
logd("--- hls message, DEMUX_NOTIFY_HLS_DOWNLOAD_START");
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_INFO,
|
||
AW_MEDIA_INFO_DOWNLOAD_START, param);
|
||
break;
|
||
}
|
||
case DEMUX_NOTIFY_HLS_DOWNLOAD_END:
|
||
{
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_INFO,
|
||
AW_MEDIA_INFO_DOWNLOAD_END, param);
|
||
break;
|
||
}
|
||
case DEMUX_NOTIFY_HLS_DOWNLOAD_ERROR:
|
||
{
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_INFO,
|
||
AW_MEDIA_INFO_DOWNLOAD_ERROR, param);
|
||
|
||
break;
|
||
}
|
||
case DEMUX_NOTIFY_LOG_RECORDER:
|
||
{
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_LOG_RECORDER, 0, param);
|
||
break;
|
||
}
|
||
case DEMUX_NOTIFY_TIMESHIFT_DURATION:
|
||
{
|
||
if(mPriData->mLivemode == 2)
|
||
{
|
||
mPriData->mTimeShiftDuration = *(cdx_int64*)param;
|
||
logd("xxxxxxxxx mPriData->mTimeShiftDuration=%" PRId64 "",
|
||
mPriData->mTimeShiftDuration);
|
||
}
|
||
break;
|
||
}
|
||
case DEMUX_NOTIFY_METADATA:
|
||
{
|
||
mPriData->mCallback(mPriData->pUser, AWPLAYER_MEDIA_META_DATA,
|
||
0, param);
|
||
break;
|
||
}
|
||
default:
|
||
{
|
||
logw("message 0x%x not handled.", messageId);
|
||
break;
|
||
}
|
||
}
|
||
|
||
return 0;
|
||
}
|