3948 lines
127 KiB
C++
Executable File
3948 lines
127 KiB
C++
Executable File
|
|
#include "log.h"
|
|
|
|
#include "player_i.h"
|
|
#include "audioDecComponent.h"
|
|
#include "videoDecComponent.h"
|
|
#include "subtitleDecComponent.h"
|
|
#include "audioRenderComponent.h"
|
|
#include "videoRenderComponent.h"
|
|
#include "subtitleRenderComponent.h"
|
|
#include "messageQueue.h"
|
|
#include "avtimer.h"
|
|
#include "bitrateEstimater.h"
|
|
#include "framerateEstimater.h"
|
|
|
|
#define CONFIG_DISABLE_VIDEO (0)
|
|
#define CONFIG_DISABLE_AUDIO (0)
|
|
#define CONFIG_DISABLE_SUBTITLE (0)
|
|
|
|
#define CdxAbs(a) ((a)>0 ? (a) : -(a))
|
|
#if(CONFIG_DTV==OPTION_DTV_YES)
|
|
#define IS_DTMB_STREAM (1)
|
|
#else
|
|
#define IS_DTMB_STREAM (0)
|
|
#endif
|
|
|
|
typedef struct PlayerContext
|
|
{
|
|
VideoStreamInfo* pVideoStreamInfo;
|
|
|
|
AudioStreamInfo* pAudioStreamInfo;
|
|
|
|
SubtitleStreamInfo* pSubtitleStreamInfo;
|
|
|
|
int nAudioStreamNum;
|
|
int nAudioStreamSelected;
|
|
int nSubtitleStreamNum;
|
|
int nSubtitleStreamSelected;
|
|
void* pNativeWindow;
|
|
void* pAudioSink;
|
|
|
|
VideoRenderComp* pVideoRender;
|
|
AudioRenderComp* pAudioRender;
|
|
SubtitleRenderComp* pSubtitleRender;
|
|
VideoDecComp* pVideoDecComp;
|
|
AudioDecComp* pAudioDecComp;
|
|
SubtitleDecComp* pSubtitleDecComp;
|
|
AvTimer* pAvTimer;
|
|
|
|
BitrateEstimater* pVideoBitrateEstimater;
|
|
BitrateEstimater* pAudioBitrateEstimater;
|
|
FramerateEstimater* pVideoFramerateEstimater;
|
|
|
|
enum EPLAYERSTATUS eStatus;
|
|
|
|
PlayerCallback callback;
|
|
void* pUserData;
|
|
|
|
//* buffer for crashed media stream.
|
|
#if CONFIG_DISABLE_VIDEO
|
|
void* pVideoStreamBuffer;
|
|
int nVideoStreamBufferSize;
|
|
#endif
|
|
#if CONFIG_DISABLE_AUDIO
|
|
void* pAudioStreamBuffer;
|
|
int nAudioStreamBufferSize;
|
|
#endif
|
|
#if CONFIG_DISABLE_SUBTITLE
|
|
void* pSubtitleStreamBuffer;
|
|
int nSubtitleStreamBufferSize;
|
|
#endif
|
|
|
|
#if DROP_3D_SECOND_VIDEO_STREAM
|
|
void* pVideoSecondStreamBuffer;
|
|
int nVideoSecondStreamBufferSize;
|
|
#endif
|
|
|
|
int bProcessingCommand;
|
|
int64_t nFirstVideoRenderPts;
|
|
int64_t nFirstAudioRenderPts;
|
|
int64_t nTimeBase;
|
|
int64_t nTimeOffset;
|
|
pthread_mutex_t timerMutex;
|
|
int64_t nLastTimeTimerAdjusted;
|
|
int onResetNotSync;
|
|
int64_t ptsBeforeReset;
|
|
|
|
int64_t nPreVideoPts;
|
|
int64_t nPreAudioPts;
|
|
int nUnsyncVideoFrameCnt;
|
|
int64_t nLastInputPts; //* for pts loop back detect in subtitle pts notify process.
|
|
int64_t nFirstPts;
|
|
|
|
int bStreamEosSet;
|
|
int bVideoRenderEosReceived;
|
|
int bAudioRenderEosReceived;
|
|
pthread_mutex_t eosMutex;
|
|
|
|
int bInFastMode;
|
|
|
|
VConfig sVideoConfig;
|
|
|
|
int sVideoCropWindow[4];
|
|
|
|
//* audio decoder mutex, currently for synchronization of the switch audio
|
|
//* operation and the demux sending job.
|
|
pthread_mutex_t audioDecoderMutex;
|
|
pthread_mutex_t subtitleDecoderMutex;
|
|
|
|
int bVideoCrash;
|
|
int bAudioCrash;
|
|
int bSubtitleCrash;
|
|
|
|
//*
|
|
int bUnSurpportVideoFlag;
|
|
void* pUnSurpportVideoBuffer;
|
|
int nUnSurpportVideoBufferSize;
|
|
|
|
float volume;
|
|
int bDiscardAudio;
|
|
LayerControlOpsT* pLayerControlOps;
|
|
SoundControlOpsT* pSoundControlOps;
|
|
}PlayerContext;
|
|
|
|
|
|
static int CallbackProcess(void* pSelf, int eMessageId, void* param);
|
|
static int PlayerInitialVideo(PlayerContext* p);
|
|
static int PlayerInitialAudio(PlayerContext* p);
|
|
static int PlayerInitialSubtitle(PlayerContext* p);
|
|
|
|
|
|
Player* PlayerCreate(void)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
AddVDPlugin();
|
|
p = (PlayerContext*)malloc(sizeof(PlayerContext));
|
|
if(p == NULL)
|
|
{
|
|
loge("memory alloc fail.");
|
|
return NULL;
|
|
}
|
|
memset(p, 0, sizeof(*p));
|
|
//p->volume = -1.0;
|
|
p->volume = 0.0;
|
|
p->nTimeOffset = 0;
|
|
p->nTimeBase = -1;
|
|
p->eStatus = PLAYER_STATUS_STOPPED;
|
|
p->pAvTimer = AvTimerCreate();
|
|
p->onResetNotSync = 0;
|
|
pthread_mutex_init(&p->eosMutex, NULL);
|
|
pthread_mutex_init(&p->timerMutex, NULL);
|
|
pthread_mutex_init(&p->audioDecoderMutex, NULL);
|
|
pthread_mutex_init(&p->subtitleDecoderMutex, NULL);
|
|
|
|
#if CONFIG_DISABLE_VIDEO
|
|
p->nVideoStreamBufferSize = 256*1024;
|
|
p->pVideoStreamBuffer = malloc(p->nVideoStreamBufferSize);
|
|
if(p->pVideoStreamBuffer == NULL)
|
|
{
|
|
loge("allocate memory fail.");
|
|
return NULL;
|
|
}
|
|
#endif
|
|
#if CONFIG_DISABLE_AUDIO
|
|
p->nAudioStreamBufferSize = 128*1024;
|
|
p->pAudioStreamBuffer = malloc(p->nAudioStreamBufferSize);
|
|
if(p->pAudioStreamBuffer == NULL)
|
|
{
|
|
loge("allocate memory fail.");
|
|
return NULL;
|
|
}
|
|
#endif
|
|
#if CONFIG_DISABLE_SUBTITLE
|
|
p->nSubtitleStreamBufferSize = 256*1024;
|
|
p->pSubtitleStreamBuffer = malloc(p->nSubtitleStreamBufferSize);
|
|
if(p->pSubtitleStreamBuffer == NULL)
|
|
{
|
|
loge("allocate memory fail.");
|
|
return NULL;
|
|
}
|
|
#endif
|
|
|
|
#if DROP_3D_SECOND_VIDEO_STREAM
|
|
p->nVideoSecondStreamBufferSize = 256*1024;
|
|
p->pVideoSecondStreamBuffer = malloc(p->nVideoSecondStreamBufferSize);
|
|
if(p->pVideoSecondStreamBuffer == NULL)
|
|
{
|
|
loge("allocate memory fail.");
|
|
return NULL;
|
|
}
|
|
#endif
|
|
|
|
p->sVideoConfig.bDispErrorFrame = 1; // set the default value of the bDispErrorFrame is 1
|
|
return (Player*)p;
|
|
}
|
|
|
|
|
|
int PlayerSetControlOps(Player* pl, LayerControlOpsT* layerOps, SoundControlOpsT* soundOps)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
p->pLayerControlOps = layerOps;
|
|
p->pSoundControlOps = soundOps;
|
|
logd("=== set player control ops");
|
|
|
|
return 0;
|
|
}
|
|
|
|
void PlayerDestroy(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
PlayerClear(pl);
|
|
|
|
if(p->pAvTimer != NULL)
|
|
{
|
|
AvTimerDestroy(p->pAvTimer);
|
|
p->pAvTimer = NULL;
|
|
}
|
|
|
|
pthread_mutex_destroy(&p->eosMutex);
|
|
pthread_mutex_destroy(&p->timerMutex);
|
|
pthread_mutex_destroy(&p->audioDecoderMutex);
|
|
pthread_mutex_destroy(&p->subtitleDecoderMutex);
|
|
|
|
|
|
#if CONFIG_DISABLE_VIDEO
|
|
if(p->pVideoStreamBuffer != NULL)
|
|
{
|
|
free(p->pVideoStreamBuffer);
|
|
p->pVideoStreamBuffer = NULL;
|
|
}
|
|
#endif
|
|
#if CONFIG_DISABLE_AUDIO
|
|
if(p->pAudioStreamBuffer != NULL)
|
|
{
|
|
free(p->pAudioStreamBuffer);
|
|
p->pAudioStreamBuffer = NULL;
|
|
}
|
|
#endif
|
|
#if CONFIG_DISABLE_SUBTITLE
|
|
if(p->pSubtitleStreamBuffer != NULL)
|
|
{
|
|
free(p->pSubtitleStreamBuffer);
|
|
p->pSubtitleStreamBuffer = NULL;
|
|
}
|
|
#endif
|
|
|
|
#if DROP_3D_SECOND_VIDEO_STREAM
|
|
if(p->pVideoSecondStreamBuffer != NULL)
|
|
{
|
|
free(p->pVideoSecondStreamBuffer);
|
|
p->pVideoSecondStreamBuffer = NULL;
|
|
}
|
|
#endif
|
|
|
|
if(p->pUnSurpportVideoBuffer != NULL)
|
|
{
|
|
free(p->pUnSurpportVideoBuffer);
|
|
p->pUnSurpportVideoBuffer = NULL;
|
|
}
|
|
|
|
free(p);
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
int PlayerSetVideoStreamInfo(Player* pl, VideoStreamInfo* pStreamInfo)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
#if CONFIG_DISABLE_VIDEO
|
|
return 0;
|
|
#else
|
|
if(p->pVideoStreamInfo != NULL)
|
|
{
|
|
if(p->pVideoStreamInfo->pCodecSpecificData != NULL &&
|
|
p->pVideoStreamInfo->nCodecSpecificDataLen > 0)
|
|
{
|
|
free(p->pVideoStreamInfo->pCodecSpecificData);
|
|
p->pVideoStreamInfo->pCodecSpecificData = NULL;
|
|
p->pVideoStreamInfo->nCodecSpecificDataLen = 0;
|
|
}
|
|
|
|
free(p->pVideoStreamInfo);
|
|
p->pVideoStreamInfo = NULL;
|
|
}
|
|
|
|
if(pStreamInfo)
|
|
{
|
|
p->pVideoStreamInfo = (VideoStreamInfo*)malloc(sizeof(VideoStreamInfo));
|
|
if(p->pVideoStreamInfo == NULL)
|
|
{
|
|
loge("malloc memory fail.");
|
|
return -1;
|
|
}
|
|
|
|
memcpy(p->pVideoStreamInfo, pStreamInfo, sizeof(VideoStreamInfo));
|
|
if(pStreamInfo->pCodecSpecificData != NULL &&
|
|
pStreamInfo->nCodecSpecificDataLen > 0)
|
|
{
|
|
p->pVideoStreamInfo->pCodecSpecificData = (char*)malloc(pStreamInfo->nCodecSpecificDataLen);
|
|
if(p->pVideoStreamInfo->pCodecSpecificData == NULL)
|
|
{
|
|
loge("malloc memory fail.");
|
|
free(p->pVideoStreamInfo);
|
|
p->pVideoStreamInfo = NULL;
|
|
return -1;
|
|
}
|
|
memcpy(p->pVideoStreamInfo->pCodecSpecificData,
|
|
pStreamInfo->pCodecSpecificData,
|
|
pStreamInfo->nCodecSpecificDataLen);
|
|
}
|
|
}
|
|
|
|
if(PlayerInitialVideo(p) == 0)
|
|
{
|
|
if(p->eStatus != PLAYER_STATUS_STOPPED)
|
|
{
|
|
VideoRenderCompStart(p->pVideoRender);
|
|
if(p->eStatus == PLAYER_STATUS_PAUSED)
|
|
VideoRenderCompPause(p->pVideoRender);
|
|
VideoDecCompStart(p->pVideoDecComp);
|
|
if(p->eStatus == PLAYER_STATUS_PAUSED)
|
|
VideoDecCompPause(p->pVideoDecComp);
|
|
}
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
return -1;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
int PlayerSetAudioStreamInfo(Player* pl, AudioStreamInfo* pStreamInfo, int nStreamNum, int nDefaultStream)
|
|
{
|
|
PlayerContext* p;
|
|
int i;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(nStreamNum == 0)
|
|
{
|
|
loge("have no audio stream, wouldn't init audio player");
|
|
return 0;
|
|
}
|
|
|
|
#if CONFIG_DISABLE_AUDIO
|
|
return 0;
|
|
#else
|
|
if(p->pAudioStreamInfo != NULL && p->nAudioStreamNum > 0)
|
|
{
|
|
for(i=0; i<p->nAudioStreamNum; i++)
|
|
{
|
|
if(p->pAudioStreamInfo[i].pCodecSpecificData != NULL &&
|
|
p->pAudioStreamInfo[i].nCodecSpecificDataLen > 0)
|
|
{
|
|
free(p->pAudioStreamInfo[i].pCodecSpecificData);
|
|
p->pAudioStreamInfo[i].pCodecSpecificData = NULL;
|
|
p->pAudioStreamInfo[i].nCodecSpecificDataLen = 0;
|
|
}
|
|
}
|
|
|
|
free(p->pAudioStreamInfo);
|
|
p->pAudioStreamInfo = NULL;
|
|
}
|
|
|
|
p->nAudioStreamNum = 0;
|
|
p->nAudioStreamSelected = 0;
|
|
|
|
if(pStreamInfo != NULL && nStreamNum > 0)
|
|
{
|
|
p->pAudioStreamInfo = (AudioStreamInfo*)malloc(sizeof(AudioStreamInfo) * nStreamNum);
|
|
if(p->pAudioStreamInfo == NULL)
|
|
{
|
|
loge("malloc memory fail.");
|
|
return -1;
|
|
}
|
|
|
|
memcpy(p->pAudioStreamInfo, pStreamInfo, sizeof(AudioStreamInfo)*nStreamNum);
|
|
|
|
for(i=0; i<nStreamNum; i++)
|
|
{
|
|
if(pStreamInfo[i].pCodecSpecificData != NULL &&
|
|
pStreamInfo[i].nCodecSpecificDataLen > 0)
|
|
{
|
|
p->pAudioStreamInfo[i].pCodecSpecificData = (char*)malloc(pStreamInfo[i].nCodecSpecificDataLen);
|
|
if(p->pAudioStreamInfo[i].pCodecSpecificData == NULL)
|
|
{
|
|
loge("malloc memory fail.");
|
|
break;
|
|
}
|
|
memcpy(p->pAudioStreamInfo[i].pCodecSpecificData,
|
|
pStreamInfo[i].pCodecSpecificData,
|
|
pStreamInfo[i].nCodecSpecificDataLen);
|
|
}
|
|
}
|
|
|
|
if(i != nStreamNum)
|
|
{
|
|
i--;
|
|
for(; i>=0; i--)
|
|
{
|
|
if(p->pAudioStreamInfo[i].pCodecSpecificData != NULL &&
|
|
p->pAudioStreamInfo[i].nCodecSpecificDataLen > 0)
|
|
{
|
|
free(p->pAudioStreamInfo[i].pCodecSpecificData);
|
|
p->pAudioStreamInfo[i].pCodecSpecificData = NULL;
|
|
p->pAudioStreamInfo[i].nCodecSpecificDataLen = 0;
|
|
}
|
|
}
|
|
|
|
free(p->pAudioStreamInfo);
|
|
p->pAudioStreamInfo = NULL;
|
|
return -1;
|
|
}
|
|
|
|
p->nAudioStreamSelected = nDefaultStream;
|
|
p->nAudioStreamNum = nStreamNum;
|
|
}
|
|
|
|
if(PlayerInitialAudio(p) == 0)
|
|
{
|
|
if(p->eStatus != PLAYER_STATUS_STOPPED)
|
|
{
|
|
AudioRenderCompStart(p->pAudioRender);
|
|
if(p->eStatus == PLAYER_STATUS_PAUSED)
|
|
AudioRenderCompPause(p->pAudioRender);
|
|
AudioDecCompStart(p->pAudioDecComp);
|
|
if(p->eStatus == PLAYER_STATUS_PAUSED)
|
|
AudioDecCompPause(p->pAudioDecComp);
|
|
}
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
return -1;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
int PlayerAddAudioStream(Player* pl, AudioStreamInfo* pStreamInfo)
|
|
{
|
|
PlayerContext* p;
|
|
AudioStreamInfo* newArr;
|
|
int nStreamCount;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->pAudioStreamInfo != NULL && p->nAudioStreamNum > 0)
|
|
{
|
|
nStreamCount = p->nAudioStreamNum + 1;
|
|
newArr = (AudioStreamInfo*)malloc(sizeof(AudioStreamInfo)*nStreamCount);
|
|
if(newArr == NULL)
|
|
{
|
|
loge("malloc memory fail.");
|
|
return -1;
|
|
}
|
|
|
|
memcpy(newArr, p->pAudioStreamInfo, p->nAudioStreamNum*sizeof(AudioStreamInfo));
|
|
memcpy(&newArr[nStreamCount-1], pStreamInfo, sizeof(AudioStreamInfo));
|
|
if(pStreamInfo->pCodecSpecificData != NULL && pStreamInfo->nCodecSpecificDataLen > 0)
|
|
{
|
|
newArr[nStreamCount-1].pCodecSpecificData = (char*)malloc(pStreamInfo->nCodecSpecificDataLen);
|
|
if(newArr[nStreamCount-1].pCodecSpecificData == NULL)
|
|
{
|
|
loge("malloc memory fail.");
|
|
free(newArr);
|
|
return -1;
|
|
}
|
|
memcpy(newArr[nStreamCount-1].pCodecSpecificData,
|
|
pStreamInfo->pCodecSpecificData,
|
|
pStreamInfo->nCodecSpecificDataLen);
|
|
}
|
|
|
|
if(p->pAudioDecComp != NULL)
|
|
{
|
|
if(AudioDecCompGetStatus(p->pAudioDecComp) == PLAYER_STATUS_STARTED)
|
|
{
|
|
AudioDecCompPause(p->pAudioDecComp);
|
|
ret = AudioDecCompAddAudioStream(p->pAudioDecComp, &newArr[nStreamCount-1]);
|
|
AudioDecCompStart(p->pAudioDecComp);
|
|
}
|
|
else
|
|
ret = AudioDecCompAddAudioStream(p->pAudioDecComp, &newArr[nStreamCount-1]);
|
|
|
|
if(ret != 0)
|
|
{
|
|
if(newArr[nStreamCount-1].pCodecSpecificData != NULL &&
|
|
newArr[nStreamCount-1].nCodecSpecificDataLen != 0)
|
|
{
|
|
free(newArr[nStreamCount-1].pCodecSpecificData);
|
|
newArr[nStreamCount-1].nCodecSpecificDataLen = 0;
|
|
newArr[nStreamCount-1].pCodecSpecificData = NULL;
|
|
}
|
|
free(newArr);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
free(p->pAudioStreamInfo);
|
|
p->pAudioStreamInfo = newArr;
|
|
p->nAudioStreamNum = nStreamCount;
|
|
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
return PlayerSetAudioStreamInfo(pl, pStreamInfo, 1, 0);
|
|
}
|
|
}
|
|
|
|
|
|
int PlayerGetAudioStreamCnt(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
p = (PlayerContext*)pl;
|
|
return p->nAudioStreamNum;
|
|
}
|
|
|
|
|
|
int PlayerCanSupportAudioStream(Player* pl, AudioStreamInfo* pStreamInfo)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
CEDARX_UNUSE(pStreamInfo);
|
|
|
|
p = (PlayerContext*)pl;
|
|
return 1;
|
|
}
|
|
|
|
int PlayerCanSupportVideoStream(Player* pl, VideoStreamInfo* pStreamInfo)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
CEDARX_UNUSE(pStreamInfo);
|
|
|
|
p = (PlayerContext*)pl;
|
|
return 1;
|
|
}
|
|
|
|
int PlayerSetDiscardAudio(Player* pl, int f)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
p->bDiscardAudio = f;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int PlayerSetWindow(Player* pl, void* pNativeWindow)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
p->pNativeWindow = pNativeWindow;
|
|
|
|
if(p->pVideoRender != NULL)
|
|
{
|
|
return VideoRenderCompSetWindow(p->pVideoRender, pNativeWindow);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int PlayerSetAudioSink(Player* pl, void* pAudioSink)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
p->pAudioSink = pAudioSink;
|
|
if(p->pAudioRender != NULL)
|
|
{
|
|
return AudioRenderCompSetAudioSink(p->pAudioRender, pAudioSink);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int PlayerSetCallback(Player* pl, PlayerCallback callback, void* pUserData)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
p->callback = callback;
|
|
p->pUserData = pUserData;
|
|
return 0;
|
|
}
|
|
|
|
|
|
int PlayerHasVideo(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pVideoRender != NULL)
|
|
return 1;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
|
|
int PlayerHasAudio(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pAudioRender != NULL)
|
|
return 1;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
int PlayerStart(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
logd("player start");
|
|
|
|
if(p->eStatus == PLAYER_STATUS_STARTED)
|
|
{
|
|
loge("invalid start operation, player already in started status.");
|
|
return -1;
|
|
}
|
|
|
|
p->bProcessingCommand = 1;
|
|
|
|
if(p->eStatus == PLAYER_STATUS_PAUSED)
|
|
{
|
|
if(p->bInFastMode)
|
|
{
|
|
//* set timer to the max value to make the audio render to discard all data.
|
|
p->pAvTimer->SetTime(p->pAvTimer, 0x7000000000000000LL);
|
|
}
|
|
else
|
|
{
|
|
if(p->nTimeBase != -1)
|
|
p->pAvTimer->Start(p->pAvTimer);
|
|
}
|
|
|
|
//* resume components.
|
|
if(p->pVideoDecComp != NULL)
|
|
VideoDecCompStart(p->pVideoDecComp);
|
|
if(p->pAudioDecComp != NULL)
|
|
AudioDecCompStart(p->pAudioDecComp);
|
|
if(p->pSubtitleDecComp != NULL)
|
|
SubtitleDecCompStart(p->pSubtitleDecComp);
|
|
if(p->pVideoRender != NULL)
|
|
VideoRenderCompStart(p->pVideoRender);
|
|
if(p->pAudioRender != NULL)
|
|
AudioRenderCompStart(p->pAudioRender);
|
|
if(p->pSubtitleRender != NULL)
|
|
SubtitleRenderCompStart(p->pSubtitleRender);
|
|
}
|
|
else
|
|
{
|
|
if((p->pVideoDecComp == NULL || p->pVideoRender == NULL) &&
|
|
(p->pAudioDecComp == NULL || p->pAudioRender == NULL))
|
|
{
|
|
loge("neither video nor audio stream exits.");
|
|
return -1;
|
|
}
|
|
|
|
p->nFirstVideoRenderPts = -1;
|
|
p->nFirstAudioRenderPts = -1;
|
|
p->bStreamEosSet = 0;
|
|
p->bVideoRenderEosReceived = 0;
|
|
p->bAudioRenderEosReceived = 0;
|
|
p->nLastTimeTimerAdjusted = 0;
|
|
p->nPreVideoPts = -1;
|
|
p->nPreAudioPts = -1;
|
|
p->nUnsyncVideoFrameCnt = 0;
|
|
p->nLastInputPts = -1;
|
|
|
|
//* nTimeBase==-1 means timer will be started at the first pts callback.
|
|
p->nTimeBase = -1;
|
|
|
|
if(p->pVideoBitrateEstimater)
|
|
BitrateEstimaterReset(p->pVideoBitrateEstimater);
|
|
if(p->pAudioBitrateEstimater)
|
|
BitrateEstimaterReset(p->pAudioBitrateEstimater);
|
|
if(p->pVideoFramerateEstimater)
|
|
FramerateEstimaterReset(p->pVideoFramerateEstimater);
|
|
|
|
p->pAvTimer->Stop(p->pAvTimer);
|
|
if(p->bInFastMode)
|
|
{
|
|
//* set timer to a big value(but not easy to loop back) to make the audio render to discard all data.
|
|
p->pAvTimer->SetTime(p->pAvTimer, 0x7000000000000000LL);
|
|
}
|
|
else
|
|
p->pAvTimer->SetTime(p->pAvTimer, 0);
|
|
p->pAvTimer->SetSpeed(p->pAvTimer, 1000);
|
|
|
|
//* start components.
|
|
//* must start the decoder components first, because render component thread will call methods like
|
|
//* AudioDecCompRequestPcmData() which need access audio or subtitle decoders inside decoder components,
|
|
//* if we start the render components first, decoders are not created yet and this will cause the
|
|
//* render thread crash when calling methods like AudioDecCompRequestPcmData().
|
|
if(p->pVideoDecComp != NULL)
|
|
VideoDecCompStart(p->pVideoDecComp);
|
|
if(p->pAudioDecComp != NULL)
|
|
AudioDecCompStart(p->pAudioDecComp);
|
|
if(p->pSubtitleDecComp != NULL)
|
|
SubtitleDecCompStart(p->pSubtitleDecComp);
|
|
if(p->pVideoRender != NULL)
|
|
VideoRenderCompStart(p->pVideoRender);
|
|
if(p->pAudioRender != NULL)
|
|
AudioRenderCompStart(p->pAudioRender);
|
|
if(p->pSubtitleRender != NULL)
|
|
SubtitleRenderCompStart(p->pSubtitleRender);
|
|
}
|
|
|
|
p->bProcessingCommand = 0;
|
|
p->eStatus = PLAYER_STATUS_STARTED;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int PlayerStop(Player* pl) //* media stream information is still kept by the player.
|
|
{
|
|
PlayerContext* p;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
logi("player stop");
|
|
|
|
if(p->eStatus == PLAYER_STATUS_STOPPED)
|
|
{
|
|
loge("invalid stop operation, player already in stopped status.");
|
|
return -1;
|
|
}
|
|
|
|
p->bProcessingCommand = 1;
|
|
|
|
//* stop components.
|
|
//* must stop the render components first, because render component thread will call methods like
|
|
//* AudioDecCompRequestPcmData() which need access audio or subtitle decoders inside decoder components,
|
|
//* if we stop the decoder components first, decoders are destroyed and this will cause the render
|
|
//* thread crash when calling methods like AudioDecCompRequestPcmData().
|
|
if(p->pSubtitleRender != NULL)
|
|
SubtitleRenderCompStop(p->pSubtitleRender);
|
|
if(p->pAudioRender != NULL)
|
|
AudioRenderCompStop(p->pAudioRender);
|
|
if(p->pVideoRender != NULL)
|
|
VideoRenderCompStop(p->pVideoRender);
|
|
if(p->pSubtitleDecComp != NULL)
|
|
SubtitleDecCompStop(p->pSubtitleDecComp);
|
|
if(p->pAudioDecComp != NULL)
|
|
AudioDecCompStop(p->pAudioDecComp);
|
|
if(p->pVideoDecComp != NULL)
|
|
VideoDecCompStop(p->pVideoDecComp);
|
|
|
|
p->pAvTimer->Stop(p->pAvTimer);
|
|
p->pAvTimer->SetTime(p->pAvTimer, 0);
|
|
p->nTimeBase = -1;
|
|
p->nTimeOffset = 0;
|
|
p->bProcessingCommand = 0;
|
|
p->bInFastMode = 0;
|
|
p->eStatus = PLAYER_STATUS_STOPPED;
|
|
p->bVideoCrash = 0;
|
|
p->bAudioCrash = 0;
|
|
p->bSubtitleCrash = 0;
|
|
p->bDiscardAudio = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int PlayerPause(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->eStatus != PLAYER_STATUS_STARTED)
|
|
{
|
|
loge("invalid pause operation, player not in started status.");
|
|
return -1;
|
|
}
|
|
|
|
p->bProcessingCommand = 1;
|
|
|
|
//* pause components.
|
|
if(p->pVideoRender != NULL)
|
|
VideoRenderCompPause(p->pVideoRender);
|
|
if(p->pAudioRender != NULL)
|
|
AudioRenderCompPause(p->pAudioRender);
|
|
if(p->pSubtitleRender != NULL)
|
|
SubtitleRenderCompPause(p->pSubtitleRender);
|
|
if(p->pVideoDecComp != NULL)
|
|
VideoDecCompPause(p->pVideoDecComp);
|
|
if(p->pAudioDecComp != NULL)
|
|
AudioDecCompPause(p->pAudioDecComp);
|
|
if(p->pSubtitleDecComp != NULL)
|
|
SubtitleDecCompPause(p->pSubtitleDecComp);
|
|
|
|
p->pAvTimer->Stop(p->pAvTimer);
|
|
|
|
p->bProcessingCommand = 0;
|
|
p->eStatus = PLAYER_STATUS_PAUSED;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
enum EPLAYERSTATUS PlayerGetStatus(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
int ret;
|
|
p = (PlayerContext*)pl;
|
|
return p->eStatus;
|
|
}
|
|
|
|
|
|
int PlayerReset(Player* pl, int64_t nSeekTimeUs) //* for seek operation, mute be called under paused status.
|
|
{
|
|
PlayerContext* p;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
//* nTimeBase==-1 means timer will be started at the first pts callback.
|
|
p->nTimeBase = -1;
|
|
p->nTimeOffset = nSeekTimeUs;
|
|
logd("in PlayerReset():p->nTimeOffset = %lld us",p->nTimeOffset);
|
|
if(p->eStatus == PLAYER_STATUS_STOPPED)
|
|
{
|
|
loge("invalid reset operation, should be called under pause status.");
|
|
return -1;
|
|
}
|
|
|
|
p->bProcessingCommand = 1;
|
|
|
|
if(p->eStatus == PLAYER_STATUS_STARTED)
|
|
{
|
|
//* pause components.
|
|
if(p->pVideoRender != NULL)
|
|
VideoRenderCompPause(p->pVideoRender);
|
|
if(p->pAudioRender != NULL)
|
|
AudioRenderCompPause(p->pAudioRender);
|
|
if(p->pSubtitleRender != NULL)
|
|
SubtitleRenderCompPause(p->pSubtitleRender);
|
|
if(p->pVideoDecComp != NULL)
|
|
VideoDecCompPause(p->pVideoDecComp);
|
|
if(p->pAudioDecComp != NULL)
|
|
AudioDecCompPause(p->pAudioDecComp);
|
|
if(p->pSubtitleDecComp != NULL)
|
|
SubtitleDecCompPause(p->pSubtitleDecComp);
|
|
}
|
|
|
|
//* reset components.
|
|
if(p->pVideoDecComp != NULL)
|
|
VideoDecCompReset(p->pVideoDecComp);
|
|
if(p->pAudioDecComp != NULL)
|
|
AudioDecCompReset(p->pAudioDecComp, nSeekTimeUs);
|
|
if(p->pSubtitleDecComp != NULL)
|
|
SubtitleDecCompReset(p->pSubtitleDecComp, nSeekTimeUs);
|
|
if(p->pVideoRender != NULL)
|
|
VideoRenderCompReset(p->pVideoRender);
|
|
if(p->pAudioRender != NULL)
|
|
AudioRenderCompReset(p->pAudioRender);
|
|
if(p->pSubtitleRender != NULL)
|
|
SubtitleRenderCompReset(p->pSubtitleRender);
|
|
|
|
p->pAvTimer->Stop(p->pAvTimer);
|
|
p->pAvTimer->SetTime(p->pAvTimer, 0);
|
|
|
|
p->nFirstVideoRenderPts = -1;
|
|
p->nFirstAudioRenderPts = -1;
|
|
p->bStreamEosSet = 0;
|
|
p->bVideoRenderEosReceived = 0;
|
|
p->bAudioRenderEosReceived = 0;
|
|
p->nLastTimeTimerAdjusted = 0;
|
|
p->nPreVideoPts = -1;
|
|
p->nPreAudioPts = -1;
|
|
p->nUnsyncVideoFrameCnt = 0;
|
|
p->nLastInputPts = -1;
|
|
p->bVideoCrash = 0;
|
|
p->bAudioCrash = 0;
|
|
p->bSubtitleCrash = 0;
|
|
|
|
if(p->pVideoBitrateEstimater)
|
|
BitrateEstimaterReset(p->pVideoBitrateEstimater);
|
|
if(p->pAudioBitrateEstimater)
|
|
BitrateEstimaterReset(p->pAudioBitrateEstimater);
|
|
if(p->pVideoFramerateEstimater)
|
|
FramerateEstimaterReset(p->pVideoFramerateEstimater);
|
|
|
|
p->bProcessingCommand = 0;
|
|
|
|
if(p->eStatus == PLAYER_STATUS_STARTED)
|
|
{
|
|
//* start components.
|
|
//* must start the decoder components first, because render component thread will call methods like
|
|
//* AudioDecCompRequestPcmData() which need access audio or subtitle decoders inside decoder components,
|
|
//* if we start the render components first, decoders are not created yet and this will cause the
|
|
//* render thread crash when calling methods like AudioDecCompRequestPcmData().
|
|
if(p->pVideoRender != NULL)
|
|
VideoRenderCompStart(p->pVideoRender);
|
|
if(p->pAudioRender != NULL)
|
|
AudioRenderCompStart(p->pAudioRender);
|
|
if(p->pSubtitleRender != NULL)
|
|
SubtitleRenderCompStart(p->pSubtitleRender);
|
|
if(p->pVideoDecComp != NULL)
|
|
VideoDecCompStart(p->pVideoDecComp);
|
|
if(p->pAudioDecComp != NULL)
|
|
AudioDecCompStart(p->pAudioDecComp);
|
|
if(p->pSubtitleDecComp != NULL)
|
|
SubtitleDecCompStart(p->pSubtitleDecComp);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int PlayerClear(Player* pl) //* must be called under stopped status, all stream information cleared.
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->eStatus != PLAYER_STATUS_STOPPED)
|
|
{
|
|
loge("invalid clear operation, player is not in stopped status.");
|
|
return -1;
|
|
}
|
|
|
|
p->bProcessingCommand = 1;
|
|
p->nFirstVideoRenderPts = -1;
|
|
p->nFirstAudioRenderPts = -1;
|
|
p->bStreamEosSet = 0;
|
|
p->bVideoRenderEosReceived = 0;
|
|
p->bAudioRenderEosReceived = 0;
|
|
p->bVideoCrash = 0;
|
|
p->bAudioCrash = 0;
|
|
p->bSubtitleCrash = 0;
|
|
p->sVideoCropWindow[0] = 0;
|
|
p->sVideoCropWindow[1] = 0;
|
|
p->sVideoCropWindow[2] = 0;
|
|
p->sVideoCropWindow[3] = 0;
|
|
|
|
if(p->pVideoRender != NULL)
|
|
{
|
|
VideoRenderCompDestroy(p->pVideoRender);
|
|
p->pVideoRender = NULL;
|
|
}
|
|
|
|
if(p->pAudioRender != NULL)
|
|
{
|
|
AudioRenderCompDestroy(p->pAudioRender);
|
|
p->pAudioRender = NULL;
|
|
}
|
|
|
|
if(p->pSubtitleRender != NULL)
|
|
{
|
|
SubtitleRenderCompDestroy(p->pSubtitleRender);
|
|
p->pSubtitleRender = NULL;
|
|
}
|
|
|
|
if(p->pVideoDecComp != NULL)
|
|
{
|
|
VideoDecCompDestroy(p->pVideoDecComp);
|
|
p->pVideoDecComp = NULL;
|
|
}
|
|
|
|
if(p->pAudioDecComp != NULL)
|
|
{
|
|
AudioDecCompDestroy(p->pAudioDecComp);
|
|
p->pAudioDecComp = NULL;
|
|
}
|
|
|
|
if(p->pSubtitleDecComp != NULL)
|
|
{
|
|
SubtitleDecCompDestroy(p->pSubtitleDecComp);
|
|
p->pSubtitleDecComp = NULL;
|
|
}
|
|
|
|
p->bProcessingCommand = 0;
|
|
|
|
if(p->pVideoBitrateEstimater != NULL)
|
|
{
|
|
BitrateEstimaterDestroy(p->pVideoBitrateEstimater);
|
|
p->pVideoBitrateEstimater = NULL;
|
|
}
|
|
|
|
if(p->pAudioBitrateEstimater != NULL)
|
|
{
|
|
BitrateEstimaterDestroy(p->pAudioBitrateEstimater);
|
|
p->pAudioBitrateEstimater = NULL;
|
|
}
|
|
|
|
if(p->pVideoFramerateEstimater)
|
|
{
|
|
FramerateEstimaterDestroy(p->pVideoFramerateEstimater);
|
|
p->pVideoFramerateEstimater = NULL;
|
|
}
|
|
|
|
if(p->pAvTimer != NULL)
|
|
{
|
|
p->pAvTimer->Stop(p->pAvTimer);
|
|
p->pAvTimer->SetTime(p->pAvTimer, 0);
|
|
}
|
|
|
|
if(p->pVideoStreamInfo != NULL)
|
|
{
|
|
if(p->pVideoStreamInfo->pCodecSpecificData != NULL &&
|
|
p->pVideoStreamInfo->nCodecSpecificDataLen > 0)
|
|
{
|
|
free(p->pVideoStreamInfo->pCodecSpecificData);
|
|
p->pVideoStreamInfo->pCodecSpecificData = NULL;
|
|
p->pVideoStreamInfo->nCodecSpecificDataLen = 0;
|
|
}
|
|
|
|
free(p->pVideoStreamInfo);
|
|
p->pVideoStreamInfo = NULL;
|
|
}
|
|
|
|
if(p->pAudioStreamInfo != NULL && p->nAudioStreamNum > 0)
|
|
{
|
|
int i;
|
|
for(i=0; i<p->nAudioStreamNum; i++)
|
|
{
|
|
if(p->pAudioStreamInfo[i].pCodecSpecificData != NULL &&
|
|
p->pAudioStreamInfo[i].nCodecSpecificDataLen > 0)
|
|
{
|
|
free(p->pAudioStreamInfo[i].pCodecSpecificData);
|
|
p->pAudioStreamInfo[i].pCodecSpecificData = NULL;
|
|
p->pAudioStreamInfo[i].nCodecSpecificDataLen = 0;
|
|
}
|
|
}
|
|
|
|
free(p->pAudioStreamInfo);
|
|
p->pAudioStreamInfo = NULL;
|
|
}
|
|
|
|
if(p->pSubtitleStreamInfo != NULL && p->nSubtitleStreamNum > 0)
|
|
{
|
|
int i;
|
|
for(i=0; i<p->nSubtitleStreamNum; i++)
|
|
{
|
|
if(p->pSubtitleStreamInfo[i].pUrl != NULL)
|
|
{
|
|
free(p->pSubtitleStreamInfo[i].pUrl);
|
|
p->pSubtitleStreamInfo[i].pUrl = NULL;
|
|
}
|
|
if(p->pSubtitleStreamInfo[i].pCodecSpecificData != NULL &&
|
|
p->pSubtitleStreamInfo[i].nCodecSpecificDataLen > 0)
|
|
{
|
|
free(p->pSubtitleStreamInfo[i].pCodecSpecificData);
|
|
p->pSubtitleStreamInfo[i].pCodecSpecificData = NULL;
|
|
p->pSubtitleStreamInfo[i].nCodecSpecificDataLen = 0;
|
|
}
|
|
}
|
|
|
|
free(p->pSubtitleStreamInfo);
|
|
p->pSubtitleStreamInfo = NULL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int PlayerSetEos(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pVideoDecComp != NULL)
|
|
VideoDecCompSetEOS(p->pVideoDecComp);
|
|
if(p->pAudioDecComp != NULL)
|
|
AudioDecCompSetEOS(p->pAudioDecComp);
|
|
if(p->pSubtitleDecComp != NULL)
|
|
SubtitleDecCompSetEOS(p->pSubtitleDecComp);
|
|
|
|
p->bStreamEosSet = 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int PlayerSetFirstPts(Player* pl, int64_t nFirstPts)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
p->nFirstPts = nFirstPts;
|
|
if(p->pSubtitleRender != NULL)
|
|
SubtitleRenderCompSetVideoOrAudioFirstPts(p->pSubtitleRender,p->nFirstPts);
|
|
|
|
return 0;
|
|
}
|
|
int PlayerRequestStreamBuffer(Player* pl,
|
|
int nRequireSize,
|
|
void** ppBuf,
|
|
int* pBufSize,
|
|
void** ppRingBuf,
|
|
int* pRingBufSize,
|
|
enum EMEDIATYPE eMediaType,
|
|
int nStreamIndex)
|
|
{
|
|
PlayerContext* p;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
logi("request stream buffer, eMediaType = %d, require size = %d", eMediaType, nRequireSize);
|
|
|
|
*ppBuf = NULL;
|
|
*pBufSize = 0;
|
|
*ppRingBuf = NULL;
|
|
*pRingBufSize = 0;
|
|
|
|
if(p->eStatus == PLAYER_STATUS_STOPPED)
|
|
{
|
|
loge("can not request stream buffer at stopped status.");
|
|
return -1;
|
|
}
|
|
|
|
if(eMediaType == MEDIA_TYPE_VIDEO)
|
|
{
|
|
if(p->bUnSurpportVideoFlag == 1)
|
|
{
|
|
if(nRequireSize > p->nUnSurpportVideoBufferSize)
|
|
nRequireSize = p->nUnSurpportVideoBufferSize;
|
|
*pBufSize = nRequireSize;
|
|
*ppBuf = p->pUnSurpportVideoBuffer;
|
|
return 0;
|
|
}
|
|
|
|
#if CONFIG_DISABLE_VIDEO
|
|
if(nRequireSize > p->nVideoStreamBufferSize)
|
|
nRequireSize = p->nVideoStreamBufferSize;
|
|
*pBufSize = nRequireSize;
|
|
*ppBuf = p->pVideoStreamBuffer;
|
|
return 0;
|
|
#else
|
|
|
|
#if DROP_3D_SECOND_VIDEO_STREAM
|
|
if(nStreamIndex == 1)
|
|
{
|
|
if(nRequireSize > p->nVideoSecondStreamBufferSize)
|
|
nRequireSize = p->nVideoSecondStreamBufferSize;
|
|
*pBufSize = nRequireSize;
|
|
*ppBuf = p->pVideoSecondStreamBuffer;
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
if(p->pVideoDecComp == NULL)
|
|
{
|
|
loge("no video decode component, can not request video stream buffer.");
|
|
return -1;
|
|
}
|
|
|
|
ret = VideoDecCompRequestStreamBuffer(p->pVideoDecComp,
|
|
nRequireSize,
|
|
(char**)ppBuf,
|
|
pBufSize,
|
|
(char**)ppRingBuf,
|
|
pRingBufSize,
|
|
nStreamIndex);
|
|
|
|
return ret;
|
|
#endif
|
|
}
|
|
else if(eMediaType == MEDIA_TYPE_AUDIO)
|
|
{
|
|
#if CONFIG_DISABLE_AUDIO
|
|
if(nRequireSize > p->nAudioStreamBufferSize)
|
|
nRequireSize = p->nAudioStreamBufferSize;
|
|
*pBufSize = nRequireSize;
|
|
*ppBuf = p->pAudioStreamBuffer;
|
|
return 0;
|
|
#else
|
|
pthread_mutex_lock(&p->audioDecoderMutex);
|
|
|
|
if(p->pAudioDecComp == NULL)
|
|
{
|
|
loge("no audio decode component, can not request audio stream buffer.");
|
|
pthread_mutex_unlock(&p->audioDecoderMutex);
|
|
return -1;
|
|
}
|
|
|
|
ret = AudioDecCompRequestStreamBuffer(p->pAudioDecComp,
|
|
nRequireSize,
|
|
(char**)ppBuf,
|
|
pBufSize,
|
|
(char**)ppRingBuf,
|
|
pRingBufSize,
|
|
nStreamIndex);
|
|
|
|
pthread_mutex_unlock(&p->audioDecoderMutex);
|
|
return ret;
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
#if CONFIG_DISABLE_SUBTITLE
|
|
if(nRequireSize > p->nSubtitleStreamBufferSize)
|
|
nRequireSize = p->nSubtitleStreamBufferSize;
|
|
*pBufSize = nRequireSize;
|
|
*ppBuf = p->pSubtitleStreamBuffer;
|
|
return 0;
|
|
#else
|
|
pthread_mutex_lock(&p->subtitleDecoderMutex);
|
|
|
|
if(p->pSubtitleDecComp == NULL)
|
|
{
|
|
loge("no subtitle decode component, can not request subtitle stream buffer.");
|
|
pthread_mutex_unlock(&p->subtitleDecoderMutex);
|
|
return -1;
|
|
}
|
|
|
|
ret = SubtitleDecCompRequestStreamBuffer(p->pSubtitleDecComp,
|
|
nRequireSize,
|
|
(char**)ppBuf,
|
|
pBufSize,
|
|
nStreamIndex);
|
|
|
|
pthread_mutex_unlock(&p->subtitleDecoderMutex);
|
|
return ret;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
|
|
int PlayerSubmitStreamData(Player* pl,
|
|
MediaStreamDataInfo* pDataInfo,
|
|
enum EMEDIATYPE eMediaType,
|
|
int nStreamIndex)
|
|
{
|
|
PlayerContext* p;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
logi("submit stream data, eMediaType = %d", eMediaType);
|
|
|
|
if(p->eStatus == PLAYER_STATUS_STOPPED)
|
|
{
|
|
loge("can not submit stream data at stopped status.");
|
|
return -1;
|
|
}
|
|
|
|
if(pDataInfo->nPts != -1)
|
|
p->nLastInputPts = pDataInfo->nPts; //* save the last input pts.
|
|
|
|
if(eMediaType == MEDIA_TYPE_VIDEO)
|
|
{
|
|
if(p->bUnSurpportVideoFlag == 1)
|
|
return 0;
|
|
|
|
#if CONFIG_DISABLE_VIDEO
|
|
return 0;
|
|
#else
|
|
|
|
#if DROP_3D_SECOND_VIDEO_STREAM
|
|
if(nStreamIndex == 1)
|
|
return 0;
|
|
#endif
|
|
VideoStreamDataInfo dataInfo;
|
|
|
|
if(p->pVideoDecComp == NULL)
|
|
{
|
|
loge("no video decode component, can not submit video stream data.");
|
|
return -1;
|
|
}
|
|
|
|
logi("submit video data, pts = %lld ms, curTime = %lld ms", pDataInfo->nPts/1000,
|
|
p->pAvTimer->GetTime(p->pAvTimer)/1000);
|
|
|
|
if(nStreamIndex == 0 && p->pVideoBitrateEstimater)
|
|
BitrateEstimaterUpdate(p->pVideoBitrateEstimater, pDataInfo->nPts, pDataInfo->nLength);
|
|
|
|
//* process the video resolution change
|
|
if(pDataInfo->nStreamChangeFlag == 1)
|
|
{
|
|
if(pDataInfo->nStreamChangeNum != 1)
|
|
{
|
|
loge("the videoNum is not 1,");
|
|
abort();
|
|
}
|
|
|
|
VideoStreamInfo* pVideoChangeInfo = (VideoStreamInfo*)pDataInfo->pStreamInfo;
|
|
VideoStreamInfo* pVideoStreamInfo = (VideoStreamInfo*)malloc(sizeof(VideoStreamInfo));
|
|
memset(pVideoStreamInfo,0,sizeof(VideoStreamInfo));
|
|
|
|
pVideoStreamInfo->eCodecFormat = pVideoChangeInfo->eCodecFormat;
|
|
pVideoStreamInfo->nWidth = pVideoChangeInfo->nWidth;
|
|
pVideoStreamInfo->nHeight = pVideoChangeInfo->nHeight;
|
|
pVideoStreamInfo->nFrameRate = pVideoChangeInfo->nFrameRate;
|
|
pVideoStreamInfo->nFrameDuration = pVideoChangeInfo->nFrameDuration;
|
|
pVideoStreamInfo->nAspectRatio = pVideoChangeInfo->nAspectRatio;
|
|
pVideoStreamInfo->bIs3DStream = pVideoChangeInfo->bIs3DStream;
|
|
pVideoStreamInfo->bIsFramePackage = pVideoChangeInfo->bIsFramePackage;
|
|
|
|
if(pVideoChangeInfo->nCodecSpecificDataLen > 0)
|
|
{
|
|
pVideoStreamInfo->pCodecSpecificData = (char*)malloc(pVideoChangeInfo->nCodecSpecificDataLen);
|
|
if(pVideoStreamInfo->pCodecSpecificData == NULL)
|
|
{
|
|
loge("malloc pCodecSpecificData failed");
|
|
free(pVideoChangeInfo->pCodecSpecificData);
|
|
free(pVideoChangeInfo);
|
|
return -1;
|
|
}
|
|
|
|
memcpy(pVideoStreamInfo->pCodecSpecificData,pVideoChangeInfo->pCodecSpecificData,
|
|
pVideoChangeInfo->nCodecSpecificDataLen);
|
|
pVideoStreamInfo->nCodecSpecificDataLen = pVideoChangeInfo->nCodecSpecificDataLen;
|
|
|
|
free(pVideoChangeInfo->pCodecSpecificData);
|
|
pVideoChangeInfo->pCodecSpecificData = NULL;
|
|
}
|
|
else
|
|
{
|
|
pVideoStreamInfo->pCodecSpecificData = NULL;
|
|
pVideoStreamInfo->nCodecSpecificDataLen = 0;
|
|
}
|
|
|
|
free(pVideoChangeInfo);
|
|
pDataInfo->pStreamInfo = NULL;
|
|
|
|
dataInfo.bVideoInfoFlag = 1;
|
|
dataInfo.pVideoInfo = pVideoStreamInfo;
|
|
}
|
|
else
|
|
{
|
|
dataInfo.pVideoInfo = NULL;
|
|
dataInfo.bVideoInfoFlag = 0;
|
|
}
|
|
|
|
dataInfo.pData = pDataInfo->pData;
|
|
dataInfo.nLength = pDataInfo->nLength;
|
|
dataInfo.nPts = pDataInfo->nPts;
|
|
dataInfo.nPcr = pDataInfo->nPcr;
|
|
dataInfo.bIsFirstPart = pDataInfo->bIsFirstPart;
|
|
dataInfo.bIsLastPart = pDataInfo->bIsLastPart;
|
|
dataInfo.nStreamIndex = nStreamIndex;
|
|
|
|
ret = VideoDecCompSubmitStreamData(p->pVideoDecComp, &dataInfo, nStreamIndex);
|
|
return ret;
|
|
#endif
|
|
}
|
|
else if(eMediaType == MEDIA_TYPE_AUDIO)
|
|
{
|
|
#if CONFIG_DISABLE_AUDIO
|
|
return 0;
|
|
#else
|
|
AudioStreamDataInfo dataInfo;
|
|
|
|
pthread_mutex_lock(&p->audioDecoderMutex);
|
|
|
|
if(p->pAudioDecComp == NULL)
|
|
{
|
|
loge("no audio decode component, can not submit audio stream data.");
|
|
pthread_mutex_unlock(&p->audioDecoderMutex);
|
|
return -1;
|
|
}
|
|
|
|
logi("submit audio data, pts = %lld ms, curTime = %lld ms", pDataInfo->nPts/1000,
|
|
p->pAvTimer->GetTime(p->pAvTimer)/1000);
|
|
|
|
if(nStreamIndex == p->nAudioStreamSelected)
|
|
BitrateEstimaterUpdate(p->pAudioBitrateEstimater, pDataInfo->nPts, pDataInfo->nLength);
|
|
|
|
//* process the video resolution change.
|
|
//* now, we just free the buffer , not to process it really
|
|
if(pDataInfo->nStreamChangeFlag == 1)
|
|
{
|
|
AudioStreamInfo* pAudioChangeInfo = (AudioStreamInfo*)pDataInfo->pStreamInfo;
|
|
for(int i = 0; i < pDataInfo->nStreamChangeNum; i++)
|
|
{
|
|
if(pAudioChangeInfo[i].pCodecSpecificData != NULL)
|
|
free(pAudioChangeInfo[i].pCodecSpecificData);
|
|
}
|
|
free(pAudioChangeInfo);
|
|
}
|
|
|
|
|
|
dataInfo.pData = pDataInfo->pData;
|
|
dataInfo.nLength = pDataInfo->nLength;
|
|
dataInfo.nPts = pDataInfo->nPts;
|
|
dataInfo.nPcr = pDataInfo->nPcr;
|
|
dataInfo.nPcr = pDataInfo->nPcr;
|
|
dataInfo.bIsFirstPart = pDataInfo->bIsFirstPart;
|
|
dataInfo.bIsLastPart = pDataInfo->bIsLastPart;
|
|
|
|
ret = AudioDecCompSubmitStreamData(p->pAudioDecComp, &dataInfo, nStreamIndex);
|
|
pthread_mutex_unlock(&p->audioDecoderMutex);
|
|
|
|
return ret;
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
#if CONFIG_DISABLE_SUBTITLE
|
|
return 0;
|
|
#else
|
|
SubtitleStreamDataInfo dataInfo;
|
|
|
|
pthread_mutex_lock(&p->subtitleDecoderMutex);
|
|
|
|
if(p->pSubtitleDecComp == NULL)
|
|
{
|
|
loge("no subtitle decode component, can not submit subtitle stream data.");
|
|
pthread_mutex_unlock(&p->subtitleDecoderMutex);
|
|
return -1;
|
|
}
|
|
|
|
logi("submit subtitle data, pts = %lld ms, curTime = %lld ms", pDataInfo->nPts/1000,
|
|
p->pAvTimer->GetTime(p->pAvTimer)/1000);
|
|
|
|
dataInfo.pData = pDataInfo->pData;
|
|
dataInfo.nLength = pDataInfo->nLength;
|
|
dataInfo.nPts = pDataInfo->nPts;
|
|
dataInfo.nPcr = pDataInfo->nPcr;
|
|
dataInfo.nPcr = pDataInfo->nPcr;
|
|
dataInfo.nDuration = pDataInfo->nDuration;
|
|
|
|
ret = SubtitleDecCompSubmitStreamData(p->pSubtitleDecComp, &dataInfo, nStreamIndex);
|
|
pthread_mutex_unlock(&p->subtitleDecoderMutex);
|
|
|
|
return ret;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
int64_t PlayerGetPositionCMCC(Player* pl) //* current time position in us.
|
|
{
|
|
PlayerContext* p;
|
|
int64_t nCurTime;
|
|
int64_t nCurPosition;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->bInFastMode)
|
|
return p->nPreVideoPts;
|
|
|
|
if(p->nTimeBase != -1)
|
|
{
|
|
nCurTime = p->pAvTimer->GetTime(p->pAvTimer);
|
|
nCurPosition = p->nTimeOffset - nCurTime + p->nTimeBase ;
|
|
}
|
|
else
|
|
{
|
|
//* waiting timer to be start at the first pts callback.
|
|
//* current timer value is not valid.
|
|
nCurPosition = p->nTimeOffset;
|
|
}
|
|
return nCurPosition;
|
|
}
|
|
|
|
int64_t PlayerGetPosition(Player* pl) //* current time position in us.
|
|
{
|
|
PlayerContext* p;
|
|
int64_t nCurTime;
|
|
int64_t nCurPosition;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->bInFastMode)
|
|
return p->nPreVideoPts;
|
|
|
|
if(p->nTimeBase != -1)
|
|
{
|
|
nCurTime = p->pAvTimer->GetTime(p->pAvTimer);
|
|
nCurPosition = nCurTime - p->nTimeBase + p->nTimeOffset;
|
|
}
|
|
else
|
|
{
|
|
//* waiting timer to be start at the first pts callback.
|
|
//* current timer value is not valid.
|
|
nCurPosition = p->nTimeOffset;
|
|
}
|
|
logd("in PlayerGetPosition,nCurPosition = %lld us , p->nTimeBase = %d",nCurPosition,p->nTimeBase);
|
|
return nCurPosition;
|
|
}
|
|
|
|
|
|
int64_t PlayerGetPts(Player* pl) //* current pts in us.
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->bInFastMode)
|
|
return p->nPreVideoPts;
|
|
|
|
if(p->nTimeBase != -1)
|
|
{
|
|
return p->pAvTimer->GetTime(p->pAvTimer);
|
|
}
|
|
else
|
|
{
|
|
//* waiting timer to be start at the first pts callback.
|
|
//* current timer value is not valid.
|
|
return p->nTimeOffset;
|
|
}
|
|
}
|
|
|
|
|
|
int PlayerSet3DMode(Player* pl, enum EPICTURE3DMODE ePicture3DMode, enum EDISPLAY3DMODE eDisplay3DMode)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->pVideoRender != NULL)
|
|
{
|
|
return VideoRenderSet3DMode(p->pVideoRender, ePicture3DMode, eDisplay3DMode);
|
|
}
|
|
else
|
|
{
|
|
loge("no video render, can not set 3D mode.");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
|
|
int PlayerGet3DMode(Player* pl, enum EPICTURE3DMODE* ePicture3DMode, enum EDISPLAY3DMODE* eDisplay3DMode)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->pVideoRender != NULL)
|
|
{
|
|
return VideoRenderGet3DMode(p->pVideoRender, ePicture3DMode, eDisplay3DMode);
|
|
}
|
|
else
|
|
{
|
|
loge("no video render, can not get 3D mode.");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
|
|
int PlayerConfigVideoScaleDownRatio(Player* pl, int nHorizonRatio, int nVerticalRatio)
|
|
{
|
|
PlayerContext* p;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->eStatus != PLAYER_STATUS_STOPPED)
|
|
{
|
|
loge("player not in stopped status, can not config video scale down ratio.");
|
|
return -1;
|
|
}
|
|
|
|
if(nHorizonRatio != 0 || nVerticalRatio != 0)
|
|
p->sVideoConfig.bScaleDownEn = 1;
|
|
else
|
|
p->sVideoConfig.bScaleDownEn = 0;
|
|
p->sVideoConfig.nHorizonScaleDownRatio = nHorizonRatio;
|
|
p->sVideoConfig.nVerticalScaleDownRatio = nVerticalRatio;
|
|
return 0;
|
|
}
|
|
|
|
|
|
int PlayerConfigVideoRotateDegree(Player* pl, int nDegree)
|
|
{
|
|
PlayerContext* p;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->eStatus != PLAYER_STATUS_STOPPED)
|
|
{
|
|
loge("player not in stopped status, can not config video scale down ratio.");
|
|
return -1;
|
|
}
|
|
if(nDegree != 0)
|
|
p->sVideoConfig.bRotationEn = 1;
|
|
else
|
|
p->sVideoConfig.bRotationEn = 0;
|
|
p->sVideoConfig.nRotateDegree = nDegree;
|
|
return 0;
|
|
}
|
|
|
|
|
|
int PlayerConfigVideoDeinterlace(Player* pl, int bDeinterlace)
|
|
{
|
|
PlayerContext* p;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->eStatus != PLAYER_STATUS_STOPPED)
|
|
{
|
|
loge("player not in stopped status, can not config video deinterlace flag.");
|
|
return -1;
|
|
}
|
|
|
|
p->sVideoConfig.bSupportMaf = bDeinterlace;
|
|
return 0;
|
|
}
|
|
|
|
int PlayerConfigTvStreamFlag(Player* pl, int bFlag)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->eStatus != PLAYER_STATUS_STOPPED)
|
|
{
|
|
loge("player not in stopped status, can not config video deinterlace flag.");
|
|
return -1;
|
|
}
|
|
|
|
p->sVideoConfig.bIsTvStream = bFlag;
|
|
return 0;
|
|
}
|
|
|
|
int PlayerConfigDispErrorFrame(Player* pl, int bDispErrorFrame)
|
|
{
|
|
PlayerContext* p;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
p->sVideoConfig.bDispErrorFrame = bDispErrorFrame;
|
|
return 0;
|
|
}
|
|
|
|
int PlayerConfigDropLaytedFrame(Player* pl, int bDropLaytedFrame)
|
|
{
|
|
PlayerContext* p;
|
|
p = (PlayerContext*)pl;
|
|
|
|
logd("PlayerConfigDropLaytedFrame\n");
|
|
if(p->pVideoDecComp != NULL)
|
|
{
|
|
logd("VideoDecCompSetDropDelayFrames\n");
|
|
return VideoDecCompSetDropDelayFrames(p->pVideoDecComp, bDropLaytedFrame); //* only treat the major video stream.
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int PlayerConfigSetMemoryThresh(Player* pl, int nMemoryThresh)
|
|
{
|
|
PlayerContext* p;
|
|
p = (PlayerContext*)pl;
|
|
|
|
|
|
if(p->pVideoDecComp != NULL)
|
|
{
|
|
|
|
return VideoDecCompSetMemortThresh(p->pVideoDecComp, nMemoryThresh); //* only treat the major video stream.
|
|
}
|
|
return 0;
|
|
}
|
|
int PlayerGetVideoBitrate(Player* pl) //* estimated by pts and stream data size.
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pVideoBitrateEstimater != NULL)
|
|
{
|
|
return BitrateEstimaterGetBitrate(p->pVideoBitrateEstimater);
|
|
}
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
|
|
//* estimated by pts.
|
|
int PlayerGetVideoFrameRate(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pVideoFramerateEstimater != NULL)
|
|
{
|
|
return FramerateEstimaterGetFramerate(p->pVideoFramerateEstimater);
|
|
}
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
|
|
//* estimated by pts.
|
|
int PlayerGetVideoFrameDuration(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pVideoFramerateEstimater != NULL)
|
|
{
|
|
return FramerateEstimaterGetFrameDuration(p->pVideoFramerateEstimater);
|
|
}
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
|
|
//* how much video stream data has been buffered.
|
|
int PlayerGetVideoStreamDataSize(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pVideoDecComp != NULL)
|
|
return VideoDecCompStreamDataSize(p->pVideoDecComp, 0); //* only treat the major video stream.
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
int PlayerGetVideoStreamBufferSize(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pVideoDecComp != NULL)
|
|
return VideoDecCompStreamBufferSize(p->pVideoDecComp, 0); //* only treat the major video stream.
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
|
|
//* how many stream frame in buffer.
|
|
int PlayerGetVideoStreamFrameNum(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pVideoDecComp != NULL)
|
|
return VideoDecCompStreamFrameNum(p->pVideoDecComp, 0); //* only treat the major video stream.
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
|
|
//* how many picture has been decoded and waiting to show.
|
|
int PlayerGetValidPictureNum(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pVideoDecComp != NULL)
|
|
return VideoDecCompValidPictureNum(p->pVideoDecComp, 0); //* only treat the major video stream.
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
|
|
int PlayerGetAudioBitrate(Player* pl) //* estimated by pts and stream data size.
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pAudioBitrateEstimater != NULL)
|
|
{
|
|
return BitrateEstimaterGetBitrate(p->pAudioBitrateEstimater);
|
|
}
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
|
|
//* get audio sample rate, channel count and how many bits per sample of pcm data.
|
|
int PlayerGetAudioParam(Player* pl, int* pSampleRate, int* pChannelCount, int* pBitsPerSample)
|
|
{
|
|
PlayerContext* p;
|
|
unsigned int nSampleRate;
|
|
unsigned int nChannelCount;
|
|
unsigned int nBitRate;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pAudioDecComp != NULL &&
|
|
AudioDecCompGetAudioSampleRate(p->pAudioDecComp, &nSampleRate, &nChannelCount, &nBitRate) == 0)
|
|
{
|
|
*pSampleRate = nSampleRate;
|
|
*pChannelCount = nChannelCount;
|
|
*pBitsPerSample = 16;
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
*pSampleRate = 0;
|
|
*pChannelCount = 0;
|
|
*pBitsPerSample = 0;
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
|
|
//* how much audio stream data has been buffered.
|
|
int PlayerGetAudioStreamDataSize(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pAudioDecComp != NULL)
|
|
return AudioDecCompStreamDataSize(p->pAudioDecComp, p->nAudioStreamSelected);
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
|
|
//* how many stream frame in buffer.
|
|
int PlayerGetAudioStreamFrameNum(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pAudioDecComp != NULL)
|
|
return AudioDecCompStreamFrameNum(p->pAudioDecComp, p->nAudioStreamSelected);
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
|
|
//* how much audio pcm data has been decoded to the pcm buffer.
|
|
int PlayerGetAudioPcmDataSize(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pAudioDecComp != NULL)
|
|
return AudioDecCompPcmDataSize(p->pAudioDecComp, p->nAudioStreamSelected);
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
|
|
//* how much audio pcm data cached in audio device.
|
|
int PlayerGetAudioCacheTimeInSoundDevice(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pAudioRender != NULL)
|
|
return AudioRenderCompCacheTimeUs(p->pAudioRender);
|
|
else
|
|
return 0;
|
|
}
|
|
//hkw for IPTV
|
|
int PlayerStopAudio(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pAudioRender == NULL || p->pAudioDecComp == NULL)
|
|
{
|
|
logw("no audio decoder or render, switch audio fail.");
|
|
return -1;
|
|
}
|
|
|
|
pthread_mutex_lock(&p->audioDecoderMutex);
|
|
|
|
p->bAudioCrash = 0;
|
|
|
|
//* 1. stop the audio render.
|
|
AudioRenderCompStop(p->pAudioRender);
|
|
|
|
//* 2. stop the audio decoder.
|
|
AudioDecCompStop(p->pAudioDecComp);
|
|
pthread_mutex_unlock(&p->audioDecoderMutex);
|
|
|
|
return 0;
|
|
}
|
|
int PlayerStartAudio(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pAudioRender == NULL || p->pAudioDecComp == NULL)
|
|
{
|
|
logw("no audio decoder or render, switch audio fail.");
|
|
return -1;
|
|
}
|
|
|
|
pthread_mutex_lock(&p->audioDecoderMutex);
|
|
|
|
if(p->eStatus != PLAYER_STATUS_STOPPED)
|
|
{
|
|
//* 4. start the audio decoder and render.
|
|
AudioDecCompStart(p->pAudioDecComp);
|
|
AudioRenderCompStart(p->pAudioRender);
|
|
|
|
if(p->eStatus == PLAYER_STATUS_PAUSED)
|
|
{
|
|
//* 5. pause the audio decoder and render.
|
|
AudioRenderCompPause(p->pAudioRender);
|
|
AudioDecCompPause(p->pAudioDecComp);
|
|
}
|
|
}
|
|
|
|
if(p->bStreamEosSet)
|
|
{
|
|
p->bAudioRenderEosReceived = 0;
|
|
AudioDecCompSetEOS(p->pAudioDecComp);
|
|
}
|
|
|
|
pthread_mutex_unlock(&p->audioDecoderMutex);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int PlayerSwitchAudio(Player* pl, int nStreamIndex)
|
|
{
|
|
PlayerContext* p;
|
|
int nAudioStreamCount;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pAudioRender == NULL || p->pAudioDecComp == NULL)
|
|
{
|
|
logw("no audio decoder or render, switch audio fail.");
|
|
return -1;
|
|
}
|
|
|
|
if(nStreamIndex < 0)
|
|
{
|
|
logw("invalid audio stream index %d, switch audio stream fail.", nStreamIndex);
|
|
return -1;
|
|
}
|
|
|
|
nAudioStreamCount = AudioDecCompGetAudioStreamCnt(p->pAudioDecComp);
|
|
if(nAudioStreamCount <= nStreamIndex)
|
|
{
|
|
logw("only %d audio streams, can not switch to audio stream %d.", nAudioStreamCount, nStreamIndex);
|
|
return -1;
|
|
}
|
|
|
|
if(AudioDecCompCurrentStreamIndex(p->pAudioDecComp) == nStreamIndex)
|
|
{
|
|
logw("selected audio stream is the one being played, do nothing for switch audio.");
|
|
return 0;
|
|
}
|
|
|
|
pthread_mutex_lock(&p->audioDecoderMutex);
|
|
|
|
p->bAudioCrash = 0;
|
|
|
|
//* 1. stop the audio render.
|
|
AudioRenderCompStop(p->pAudioRender);
|
|
|
|
//* 2. stop the audio decoder.
|
|
AudioDecCompStop(p->pAudioDecComp);
|
|
|
|
//* 3. audio decoder switch audio stream.
|
|
AudioDecCompSwitchStream(p->pAudioDecComp, nStreamIndex);
|
|
p->nAudioStreamSelected = nStreamIndex;
|
|
|
|
if(p->eStatus != PLAYER_STATUS_STOPPED)
|
|
{
|
|
//* 4. start the audio decoder and render.
|
|
AudioDecCompStart(p->pAudioDecComp);
|
|
AudioRenderCompStart(p->pAudioRender);
|
|
|
|
if(p->eStatus == PLAYER_STATUS_PAUSED)
|
|
{
|
|
//* 5. pause the audio decoder and render.
|
|
AudioRenderCompPause(p->pAudioRender);
|
|
AudioDecCompPause(p->pAudioDecComp);
|
|
}
|
|
}
|
|
|
|
if(p->bStreamEosSet)
|
|
{
|
|
p->bAudioRenderEosReceived = 0;
|
|
AudioDecCompSetEOS(p->pAudioDecComp);
|
|
}
|
|
|
|
pthread_mutex_unlock(&p->audioDecoderMutex);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
//* get audio balance, return 1 means left channel, 2 means right channel, 3 means stereo.
|
|
int PlayerGetAudioBalance(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->pAudioRender != NULL)
|
|
return AudioRenderCompGetAudioOutBalance(p->pAudioRender);
|
|
else
|
|
return 3;
|
|
}
|
|
|
|
|
|
//* set audio balance, 1 means left channel, 2 means right channel, 3 means stereo.
|
|
int PlayerSetAudioBalance(Player* pl, int nAudioBalance)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->pAudioRender != NULL)
|
|
return AudioRenderCompSetAudioOutBalance(p->pAudioRender, nAudioBalance);
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
|
|
int PlayerGetAudioStreamInfo(Player* pl, int* pStreamNum, AudioStreamInfo** ppStreamInfo)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
*pStreamNum = p->nAudioStreamNum;
|
|
*ppStreamInfo = p->pAudioStreamInfo;
|
|
return 0;
|
|
}
|
|
|
|
|
|
int PlayerSetAudioMute(Player* pl, int bMute)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->pAudioRender != NULL)
|
|
return AudioRenderCompSetAudioMute(p->pAudioRender, bMute);
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
|
|
int PlayerGetAudioMuteFlag(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->pAudioRender != NULL)
|
|
return AudioRenderCompGetAudioMuteFlag(p->pAudioRender);
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
int PlayerSetAudioForceWriteToDeviceFlag(Player* pl, int bForceFlag)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
logd("*** PlayerSetAudioForceWriteToDeviceFlag, pAudioRender = %p, flag = %d",
|
|
p->pAudioRender,bForceFlag);
|
|
if(p->pAudioRender != NULL)
|
|
return AudioRenderCompSetForceWriteToDeviceFlag(p->pAudioRender,bForceFlag);
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
//*****************************************************************************//
|
|
//* TODO.
|
|
//*****************************************************************************//
|
|
|
|
//* set video window.
|
|
int PlayerSetVideoRegion(Player* pl, int x, int y, int nWidth, int nHeight)
|
|
{
|
|
CEDARX_UNUSE(pl);
|
|
CEDARX_UNUSE(x);
|
|
CEDARX_UNUSE(y);
|
|
CEDARX_UNUSE(nWidth);
|
|
CEDARX_UNUSE(nHeight);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
//* set picture display ratio, full stream, letter box or other modes.
|
|
int PlayerSetDisplayRatio(Player* pl, enum EDISPLAYRATIO eDisplayRatio)
|
|
{
|
|
CEDARX_UNUSE(pl);
|
|
CEDARX_UNUSE(eDisplayRatio);
|
|
return 0;
|
|
}
|
|
|
|
//*****************************************************************************//
|
|
//* TODO end.
|
|
//*****************************************************************************//
|
|
|
|
int PlayerSetSubtitleStreamInfo(Player* pl, SubtitleStreamInfo* pStreamInfo, int nStreamNum, int nDefaultStream)
|
|
{
|
|
PlayerContext* p;
|
|
int i;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(nStreamNum == 0)
|
|
{
|
|
loge("have no subtitle stream, wouldn't init audio player");
|
|
return 0;
|
|
}
|
|
|
|
#if CONFIG_DISABLE_SUBTITLE
|
|
return 0;
|
|
#else
|
|
if(p->pSubtitleStreamInfo != NULL && p->nSubtitleStreamNum > 0)
|
|
{
|
|
for(i=0; i<p->nSubtitleStreamNum; i++)
|
|
{
|
|
if(p->pSubtitleStreamInfo[i].pUrl != NULL)
|
|
{
|
|
free(p->pSubtitleStreamInfo[i].pUrl);
|
|
p->pSubtitleStreamInfo[i].pUrl = NULL;
|
|
}
|
|
if(p->pSubtitleStreamInfo[i].pCodecSpecificData)
|
|
{
|
|
free(p->pSubtitleStreamInfo[i].pCodecSpecificData);
|
|
p->pSubtitleStreamInfo[i].pCodecSpecificData = NULL;
|
|
p->pSubtitleStreamInfo[i].nCodecSpecificDataLen = 0;
|
|
}
|
|
}
|
|
|
|
free(p->pSubtitleStreamInfo);
|
|
p->pSubtitleStreamInfo = NULL;
|
|
}
|
|
|
|
p->nSubtitleStreamNum = 0;
|
|
p->nSubtitleStreamSelected = 0;
|
|
|
|
if(pStreamInfo != NULL && nStreamNum > 0)
|
|
{
|
|
p->pSubtitleStreamInfo = (SubtitleStreamInfo*)malloc(sizeof(SubtitleStreamInfo) * nStreamNum);
|
|
if(p->pSubtitleStreamInfo == NULL)
|
|
{
|
|
loge("malloc memory fail.");
|
|
return -1;
|
|
}
|
|
|
|
memcpy(p->pSubtitleStreamInfo, pStreamInfo, sizeof(SubtitleStreamInfo)*nStreamNum);
|
|
|
|
for(i=0; i<nStreamNum; i++)
|
|
{
|
|
if(pStreamInfo[i].pUrl != NULL)
|
|
{
|
|
p->pSubtitleStreamInfo[i].pUrl = strdup(pStreamInfo[i].pUrl);
|
|
if(p->pSubtitleStreamInfo[i].pUrl == NULL)
|
|
{
|
|
loge("malloc memory fail.");
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(pStreamInfo[i].pCodecSpecificData != NULL &&
|
|
pStreamInfo[i].nCodecSpecificDataLen > 0)
|
|
{
|
|
p->pSubtitleStreamInfo[i].nCodecSpecificDataLen = pStreamInfo[i].nCodecSpecificDataLen;
|
|
p->pSubtitleStreamInfo[i].pCodecSpecificData = (char*)malloc(pStreamInfo[i].nCodecSpecificDataLen);
|
|
if(p->pSubtitleStreamInfo[i].pCodecSpecificData == NULL)
|
|
{
|
|
loge("malloc memory fail.");
|
|
break;
|
|
}
|
|
memcpy(p->pSubtitleStreamInfo[i].pCodecSpecificData,
|
|
pStreamInfo[i].pCodecSpecificData,
|
|
pStreamInfo[i].nCodecSpecificDataLen);
|
|
}
|
|
}
|
|
|
|
if(i != nStreamNum)
|
|
{
|
|
i--;
|
|
for(; i>=0; i--)
|
|
{
|
|
if(p->pSubtitleStreamInfo[i].pUrl != NULL)
|
|
{
|
|
free(p->pSubtitleStreamInfo[i].pUrl);
|
|
p->pSubtitleStreamInfo[i].pUrl = NULL;
|
|
}
|
|
if(p->pSubtitleStreamInfo[i].pCodecSpecificData)
|
|
{
|
|
free(p->pSubtitleStreamInfo[i].pCodecSpecificData);
|
|
p->pSubtitleStreamInfo[i].pCodecSpecificData = NULL;
|
|
p->pSubtitleStreamInfo[i].nCodecSpecificDataLen = 0;
|
|
}
|
|
}
|
|
|
|
free(p->pSubtitleStreamInfo);
|
|
p->pSubtitleStreamInfo = NULL;
|
|
return -1;
|
|
}
|
|
|
|
p->nSubtitleStreamSelected = nDefaultStream;
|
|
p->nSubtitleStreamNum = nStreamNum;
|
|
}
|
|
|
|
if(PlayerInitialSubtitle(p) == 0)
|
|
{
|
|
if(p->eStatus != PLAYER_STATUS_STOPPED)
|
|
{
|
|
SubtitleRenderCompStart(p->pSubtitleRender);
|
|
if(p->eStatus == PLAYER_STATUS_PAUSED)
|
|
SubtitleRenderCompPause(p->pSubtitleRender);
|
|
SubtitleDecCompStart(p->pSubtitleDecComp);
|
|
if(p->eStatus == PLAYER_STATUS_PAUSED)
|
|
SubtitleDecCompPause(p->pSubtitleDecComp);
|
|
}
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
return -1;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
int PlayerAddSubtitleStream(Player* pl, SubtitleStreamInfo* pStreamInfo)
|
|
{
|
|
PlayerContext* p;
|
|
SubtitleStreamInfo* newArr;
|
|
int nStreamCount;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->pSubtitleStreamInfo != NULL && p->nSubtitleStreamNum > 0)
|
|
{
|
|
nStreamCount = p->nSubtitleStreamNum + 1;
|
|
newArr = (SubtitleStreamInfo*)malloc(sizeof(SubtitleStreamInfo)*nStreamCount);
|
|
if(newArr == NULL)
|
|
{
|
|
loge("malloc memory fail.");
|
|
return -1;
|
|
}
|
|
|
|
memcpy(newArr, p->pSubtitleStreamInfo, p->nSubtitleStreamNum*sizeof(SubtitleStreamInfo));
|
|
memcpy(&newArr[nStreamCount-1], pStreamInfo, sizeof(SubtitleStreamInfo));
|
|
if(pStreamInfo->pUrl != NULL)
|
|
{
|
|
newArr[nStreamCount-1].pUrl = strdup(pStreamInfo->pUrl);
|
|
if(newArr[nStreamCount-1].pUrl == NULL)
|
|
{
|
|
loge("malloc memory fail.");
|
|
free(newArr);
|
|
return -1;
|
|
}
|
|
}
|
|
if(pStreamInfo->pCodecSpecificData != NULL && pStreamInfo->nCodecSpecificDataLen > 0)
|
|
{
|
|
newArr[nStreamCount-1].nCodecSpecificDataLen = pStreamInfo->nCodecSpecificDataLen;
|
|
newArr[nStreamCount-1].pCodecSpecificData = (char*)malloc(pStreamInfo->nCodecSpecificDataLen);
|
|
if(newArr[nStreamCount-1].pCodecSpecificData == NULL)
|
|
{
|
|
loge("malloc memory fail.");
|
|
free(newArr);
|
|
return -1;
|
|
}
|
|
memcpy(newArr[nStreamCount-1].pCodecSpecificData,
|
|
pStreamInfo->pCodecSpecificData,
|
|
pStreamInfo->nCodecSpecificDataLen);
|
|
}
|
|
|
|
if(p->pSubtitleDecComp != NULL)
|
|
{
|
|
if(SubtitleDecCompGetStatus(p->pSubtitleDecComp) == PLAYER_STATUS_STARTED)
|
|
{
|
|
SubtitleDecCompPause(p->pSubtitleDecComp);
|
|
ret = SubtitleDecCompAddSubtitleStream(p->pSubtitleDecComp, &newArr[nStreamCount-1]);
|
|
SubtitleDecCompStart(p->pSubtitleDecComp);
|
|
}
|
|
else
|
|
ret = SubtitleDecCompAddSubtitleStream(p->pSubtitleDecComp, &newArr[nStreamCount-1]);
|
|
|
|
if(ret != 0)
|
|
{
|
|
if(newArr[nStreamCount-1].pUrl != NULL)
|
|
{
|
|
free(newArr[nStreamCount-1].pUrl);
|
|
newArr[nStreamCount-1].pUrl = NULL;
|
|
}
|
|
if(pStreamInfo->pCodecSpecificData != NULL && pStreamInfo->nCodecSpecificDataLen > 0)
|
|
{
|
|
free(newArr[nStreamCount-1].pCodecSpecificData);
|
|
newArr[nStreamCount-1].pCodecSpecificData = NULL;
|
|
newArr[nStreamCount-1].nCodecSpecificDataLen = 0;
|
|
}
|
|
free(newArr);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
free(p->pSubtitleStreamInfo);
|
|
p->pSubtitleStreamInfo = newArr;
|
|
p->nSubtitleStreamNum = nStreamCount;
|
|
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
return PlayerSetSubtitleStreamInfo(pl, pStreamInfo, 1, 0);
|
|
}
|
|
}
|
|
|
|
|
|
int PlayerCanSupportSubtitleStream(Player* pl, SubtitleStreamInfo* pStreamInfo)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
CEDARX_UNUSE(pStreamInfo);
|
|
p = (PlayerContext*)pl;
|
|
return 1;
|
|
}
|
|
|
|
|
|
int PlayerGetSubtitleStreamCnt(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
p = (PlayerContext*)pl;
|
|
return p->nSubtitleStreamNum;
|
|
}
|
|
|
|
|
|
int PlayerGetSubtitleStreamInfo(Player* pl, int* pStreamNum, SubtitleStreamInfo** ppStreamInfo)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
*pStreamNum = p->nSubtitleStreamNum;
|
|
*ppStreamInfo = p->pSubtitleStreamInfo;
|
|
return 0;
|
|
}
|
|
|
|
//* stream info array is allocated inside, user need to free the memory by calling free(*ppStreamInfo);
|
|
int PlayerProbeSubtitleStreamInfo(const char* strFileName,
|
|
int* pStreamNum,
|
|
SubtitleStreamInfo** ppStreamInfo)
|
|
{
|
|
return ProbeSubtitleStream(strFileName, ppStreamInfo, pStreamNum);
|
|
}
|
|
|
|
//* stream info array is allocated inside, user need to free the memory by calling free(*ppStreamInfo);
|
|
int PlayerProbeSubtitleStreamInfoFd(int fd, int offset, int len, int* pStreamNum, SubtitleStreamInfo** ppStreamInfo)
|
|
{
|
|
return ProbeSubtitleStreamFd(fd, offset, len, ppStreamInfo, pStreamNum);
|
|
}
|
|
|
|
int PlayerSwitchSubtitle(Player* pl, int nStreamIndex)
|
|
{
|
|
PlayerContext* p;
|
|
int nSubtitleStreamCount;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pSubtitleRender == NULL || p->pSubtitleDecComp == NULL)
|
|
{
|
|
logw("no subtitle decoder or render, switch subtitle fail.");
|
|
return -1;
|
|
}
|
|
|
|
if(nStreamIndex < 0)
|
|
{
|
|
logw("invalid subtitle stream index %d, switch subtitle stream fail.", nStreamIndex);
|
|
return -1;
|
|
}
|
|
|
|
nSubtitleStreamCount = SubtitleDecCompGetSubtitleStreamCnt(p->pSubtitleDecComp);
|
|
if(nSubtitleStreamCount <= nStreamIndex)
|
|
{
|
|
logw("only %d subtitle streams, can not switch to subtitle stream %d.", nSubtitleStreamCount, nStreamIndex);
|
|
return -1;
|
|
}
|
|
|
|
if(SubtitleDecCompCurrentStreamIndex(p->pSubtitleDecComp) == nStreamIndex)
|
|
{
|
|
logw("selected subtitle stream is the one being played, do nothing for switch subtitle.");
|
|
return 0;
|
|
}
|
|
|
|
pthread_mutex_lock(&p->subtitleDecoderMutex);
|
|
|
|
p->bSubtitleCrash = 0;
|
|
|
|
//* 1. stop the subtitle render.
|
|
SubtitleRenderCompStop(p->pSubtitleRender);
|
|
|
|
//* 2. stop the subtitle decoder.
|
|
SubtitleDecCompStop(p->pSubtitleDecComp);
|
|
|
|
//* 3. subtitle decoder switch subtitle stream.
|
|
SubtitleDecCompSwitchStream(p->pSubtitleDecComp, nStreamIndex);
|
|
p->nSubtitleStreamSelected = nStreamIndex;
|
|
|
|
if(p->eStatus != PLAYER_STATUS_STOPPED)
|
|
{
|
|
//* 4. start the subtitle decoder and render.
|
|
SubtitleDecCompStart(p->pSubtitleDecComp);
|
|
SubtitleRenderCompStart(p->pSubtitleRender);
|
|
|
|
if(p->eStatus == PLAYER_STATUS_PAUSED)
|
|
{
|
|
//* 5. pause the subtitle decoder and render.
|
|
SubtitleRenderCompPause(p->pSubtitleRender);
|
|
SubtitleDecCompPause(p->pSubtitleDecComp);
|
|
}
|
|
}
|
|
|
|
#if( CONFIG_ALI_YUNOS == OPTION_ALI_YUNOS_YES)
|
|
SubtitleRenderCompSwitchStream(p->pSubtitleRender,nStreamIndex);
|
|
#endif
|
|
|
|
if(p->bStreamEosSet)
|
|
{
|
|
SubtitleDecCompSetEOS(p->pAudioDecComp);
|
|
}
|
|
|
|
pthread_mutex_unlock(&p->subtitleDecoderMutex);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
//* adjust subtitle show time.
|
|
int PlayerSetSubtitleShowTimeAdjustment(Player* pl, int nTimeMs)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pSubtitleRender != NULL)
|
|
return SubtitleRenderSetShowTimeAdjustment(p->pSubtitleRender, nTimeMs);
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
//* get the adjustment of subtitle show time, in unit of ms.
|
|
int PlayerGetSubtitleShowTimeAdjustment(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->pSubtitleRender != NULL)
|
|
return SubtitleRenderGetShowTimeAdjustment(p->pSubtitleRender);
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
|
|
//* show video layer.
|
|
int PlayerVideoShow(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pVideoRender != NULL)
|
|
return VideoRenderVideoHide(p->pVideoRender, 0);
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
|
|
//* Hide video layer.
|
|
int PlayerVideoHide(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pVideoRender != NULL)
|
|
return VideoRenderVideoHide(p->pVideoRender, 1);
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
//* set whether keep last picture when player stopped.
|
|
//* the last picture is hold in default.
|
|
int PlayerSetHoldLastPicture(Player* pl, int bHold)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
if(p->pVideoRender != NULL)
|
|
return VideoRenderSetHoldLastPicture(p->pVideoRender, bHold);
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
int PlayerGetPictureSize(Player* pl, int* pWidth, int* pHeight)
|
|
{
|
|
PlayerContext* p;
|
|
VideoStreamInfo vsi;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->pVideoDecComp != NULL)
|
|
{
|
|
ret = VideoDecCompGetVideoStreamInfo(p->pVideoDecComp, &vsi);
|
|
if(ret < 0)
|
|
return -1;
|
|
|
|
*pWidth = vsi.nWidth;
|
|
*pHeight = vsi.nHeight;
|
|
return 0;
|
|
}
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
int PlayerGetPictureCrop(Player* pl, int* pLeftOff, int* pTopOff, int* pCropWidth, int* pCropHeight)
|
|
{
|
|
PlayerContext* p;
|
|
p = (PlayerContext*)pl;
|
|
*pLeftOff = p->sVideoCropWindow[0];
|
|
*pTopOff = p->sVideoCropWindow[1];
|
|
*pCropWidth = p->sVideoCropWindow[2];
|
|
*pCropHeight = p->sVideoCropWindow[3];
|
|
return 0;
|
|
}
|
|
|
|
|
|
//* set player playback in fast mode.
|
|
//* in fast mode, video is showed directly without any synchronization, and
|
|
//* audio is discard.
|
|
int PlayerFast(Player* pl, int bDecodeKeyframeOnly)
|
|
{
|
|
PlayerContext* p;
|
|
int ret;
|
|
|
|
p = (PlayerContext*)pl;
|
|
|
|
PlayerPause(pl);
|
|
PlayerReset(pl);
|
|
if(p->pVideoDecComp && bDecodeKeyframeOnly)
|
|
VideoDecCompSetDecodeKeyFrameOnly(p->pVideoDecComp, 1);
|
|
p->bInFastMode = 1;
|
|
PlayerStart(pl);
|
|
return 0;
|
|
}
|
|
|
|
//* return from fast mode, the player is set to started status when return.
|
|
int PlayerStopFast(Player* pl)
|
|
{
|
|
PlayerContext* p;
|
|
enum EPLAYERSTATUS eStatus;
|
|
|
|
p = (PlayerContext*)pl;
|
|
eStatus = p->eStatus;
|
|
PlayerPause(pl);
|
|
if(p->pVideoDecComp)
|
|
VideoDecCompSetDecodeKeyFrameOnly(p->pVideoDecComp, 0);
|
|
PlayerReset(pl);
|
|
p->bInFastMode = 0;
|
|
if(eStatus == PLAYER_STATUS_STARTED)
|
|
PlayerStart(pl);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static int CallbackProcess(void* pSelf, int eMessageId, void* param)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pSelf;
|
|
|
|
switch(eMessageId)
|
|
{
|
|
case PLAYER_VIDEO_DECODER_NOTIFY_EOS:
|
|
{
|
|
if(p->pVideoRender != NULL)
|
|
VideoRenderCompSetEOS(p->pVideoRender);
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_AUDIO_DECODER_NOTIFY_EOS:
|
|
{
|
|
if(p->pAudioRender != NULL)
|
|
AudioRenderCompSetEOS(p->pAudioRender);
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_SUBTITLE_DECODER_NOTIFY_EOS:
|
|
{
|
|
if(p->pSubtitleRender != NULL)
|
|
SubtitleRenderCompSetEOS(p->pSubtitleRender);
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_VIDEO_RENDER_NOTIFY_EOS:
|
|
{
|
|
pthread_mutex_lock(&p->eosMutex);
|
|
p->bVideoRenderEosReceived = 1;
|
|
if(p->bAudioRenderEosReceived == 1 || PlayerHasAudio((Player*)p) == 0 || p->bAudioCrash == 1)
|
|
{
|
|
//*When recive eos, we should stop the avTimer,
|
|
//*or the PlayerGetPts() function will return wrong values.
|
|
if(p->pAvTimer != NULL)
|
|
p->pAvTimer->Stop(p->pAvTimer);
|
|
|
|
if(p->callback != NULL)
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_EOS, NULL);
|
|
}
|
|
pthread_mutex_unlock(&p->eosMutex);
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_AUDIO_RENDER_NOTIFY_EOS:
|
|
{
|
|
pthread_mutex_lock(&p->eosMutex);
|
|
p->bAudioRenderEosReceived = 1;
|
|
if(p->bVideoRenderEosReceived == 1 || PlayerHasVideo((Player*)p) == 0 || p->bVideoCrash == 1)
|
|
{
|
|
//*When recive eos, we should stop the avTimer,
|
|
//*or the PlayerGetPts() function will return wrong values.
|
|
if(p->pAvTimer != NULL)
|
|
p->pAvTimer->Stop(p->pAvTimer);
|
|
|
|
if(p->callback != NULL)
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_EOS, NULL);
|
|
}
|
|
pthread_mutex_unlock(&p->eosMutex);
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_SUBTITLE_RENDER_NOTIFY_EOS:
|
|
{
|
|
//* unconcerned.
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_VIDEO_RENDER_NOTIFY_VIDEO_SIZE:
|
|
{
|
|
if(p->callback != NULL)
|
|
{
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_VIDEO_SIZE, param);
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_FIRST_PICTURE, NULL);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_VIDEO_RENDER_NOTIFY_VIDEO_CROP:
|
|
{
|
|
p->sVideoCropWindow[0] = ((int*)param)[0];
|
|
p->sVideoCropWindow[1] = ((int*)param)[1];
|
|
p->sVideoCropWindow[2] = ((int*)param)[2];
|
|
p->sVideoCropWindow[3] = ((int*)param)[3];
|
|
|
|
if(p->callback != NULL) {
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_VIDEO_CROP, param);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_VIDEO_RENDER_NOTIFY_VIDEO_BUFFER:
|
|
{
|
|
if(p->callback)
|
|
{
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_VIDEO_PIC_DATA, param);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_AUDIO_RENDER_NOTIFY_AUDIO_BUFFER:
|
|
{
|
|
if(p->callback)
|
|
{
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_AUDIO_PCM_DATA, param);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_VIDEO_RENDER_NOTIFY_FIRST_PICTURE:
|
|
{
|
|
p->nFirstVideoRenderPts = *((int64_t*)param);
|
|
|
|
logd("first video pts = %lld", p->nFirstVideoRenderPts);
|
|
|
|
if(p->bInFastMode)
|
|
{
|
|
//* in fast mode, video is showed without any synchronization,
|
|
//* audio data is discard if overtime.
|
|
p->nPreVideoPts = p->nFirstVideoRenderPts;
|
|
return 0;
|
|
}
|
|
|
|
if(PlayerHasAudio((Player*)p) == 0)
|
|
{
|
|
pthread_mutex_lock(&p->timerMutex);
|
|
p->nTimeBase = p->nFirstVideoRenderPts;
|
|
p->pAvTimer->SetTime(p->pAvTimer, p->nTimeBase);
|
|
p->pAvTimer->Start(p->pAvTimer);
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
logd("no audio, set time to %lld and start timer.", p->nTimeBase);
|
|
}
|
|
else
|
|
{
|
|
int nWaitTime = 0;
|
|
|
|
while(p->pAvTimer->GetStatus(p->pAvTimer) != TIMER_STATUS_START)
|
|
{
|
|
//* wait for audio first frame.
|
|
if(p->bProcessingCommand || nWaitTime >= CONFIG_MAX_WAIT_TIME_FOR_SYNC || p->bAudioCrash || p->bAudioRenderEosReceived || p->bDiscardAudio)
|
|
{
|
|
//* need process new command, or wait too long.
|
|
logw("break audio video first sync.");
|
|
break;
|
|
}
|
|
|
|
#if CONFIG_USE_LARGER_PTS_AS_START_TIME
|
|
//* if the video first pts is smaller (too much) than the audio first pts, discard
|
|
//* this frame to catch up the audio first frame.
|
|
if(p->nFirstAudioRenderPts != -1)
|
|
{
|
|
if(p->nFirstAudioRenderPts > (p->nFirstVideoRenderPts + 50000))
|
|
return TIMER_DROP_VIDEO_DATA;
|
|
}
|
|
#endif
|
|
usleep(10*1000);
|
|
nWaitTime += 10000;
|
|
}
|
|
|
|
if(p->bProcessingCommand && nWaitTime < CONFIG_MAX_WAIT_TIME_FOR_SYNC && p->bAudioCrash == 0)
|
|
{
|
|
//* the waiting process for first audio video sync is broken by new command.
|
|
//* tell the video render notify the first pts message later.
|
|
p->nFirstVideoRenderPts = -1;
|
|
return TIMER_NEED_NOTIFY_AGAIN;
|
|
}
|
|
|
|
pthread_mutex_lock(&p->timerMutex);
|
|
if(p->pAvTimer->GetStatus(p->pAvTimer) != TIMER_STATUS_START)
|
|
{
|
|
p->nTimeBase = p->nFirstVideoRenderPts;
|
|
p->pAvTimer->SetTime(p->pAvTimer, p->nTimeBase);
|
|
p->pAvTimer->Start(p->pAvTimer);
|
|
}
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
}
|
|
|
|
p->nPreVideoPts = p->nFirstVideoRenderPts;
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_AUDIO_RENDER_NOTIFY_FIRST_FRAME:
|
|
{
|
|
if(p->bDiscardAudio)
|
|
return TIMER_DROP_AUDIO_DATA;
|
|
|
|
if(p->bInFastMode)
|
|
{
|
|
//* discard audio in fast mode for IPTV
|
|
if(p->bDiscardAudio)
|
|
return TIMER_DROP_AUDIO_DATA;
|
|
|
|
//* in fast mode, video is showed without any synchronization,
|
|
//* audio data is discard if overtime.
|
|
int64_t nAudioPts = *((int64_t*)param);
|
|
//if(p->nPreVideoPts == -1 || nAudioPts < (p->nPreVideoPts - 150*1000))
|
|
//{
|
|
// logw("TIMER_DROP_AUDIO_DATA");
|
|
// return TIMER_DROP_AUDIO_DATA;
|
|
//}
|
|
//else
|
|
{
|
|
p->nFirstAudioRenderPts = *((int64_t*)param);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
p->nFirstAudioRenderPts = *((int64_t*)param);
|
|
|
|
logd("first audio pts = %lld", p->nFirstAudioRenderPts);
|
|
|
|
if(PlayerHasVideo((Player*)p) == 0)
|
|
{
|
|
pthread_mutex_lock(&p->timerMutex);
|
|
p->nTimeBase = p->nFirstAudioRenderPts;
|
|
p->pAvTimer->SetTime(p->pAvTimer, p->nTimeBase);
|
|
p->pAvTimer->Start(p->pAvTimer);
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
}
|
|
else
|
|
{
|
|
int nWaitTime = 0;
|
|
int eTimerStatus;
|
|
|
|
while(p->nFirstVideoRenderPts == -1)
|
|
{
|
|
if(p->bProcessingCommand || nWaitTime >= CONFIG_MAX_WAIT_TIME_FOR_SYNC || p->bVideoCrash || p->bVideoRenderEosReceived)
|
|
{
|
|
//* need process new command, or wait too long.
|
|
logw("break audio video first sync.");
|
|
break;
|
|
}
|
|
|
|
usleep(10*1000);
|
|
nWaitTime += 10000;
|
|
}
|
|
|
|
if(p->bProcessingCommand && nWaitTime < CONFIG_MAX_WAIT_TIME_FOR_SYNC && p->bVideoCrash == 0)
|
|
{
|
|
//* the waiting process for first audio video sync is broken by new command.
|
|
//* tell the audio render notify the first pts message later.
|
|
p->nFirstAudioRenderPts = -1;
|
|
return TIMER_NEED_NOTIFY_AGAIN;
|
|
}
|
|
|
|
pthread_mutex_lock(&p->timerMutex);
|
|
eTimerStatus = p->pAvTimer->GetStatus(p->pAvTimer);
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
|
|
if(eTimerStatus != TIMER_STATUS_START) //* timer not started yet.
|
|
{
|
|
if(p->nFirstVideoRenderPts != -1)
|
|
{
|
|
if(p->nFirstVideoRenderPts <= p->nFirstAudioRenderPts)
|
|
{
|
|
if((p->nFirstAudioRenderPts - p->nFirstVideoRenderPts) <=
|
|
CONFIG_THREASHOLD_OF_PTS_DIFFERENCE_TO_JUDGE_PTS_jUMP)
|
|
{
|
|
//* time difference seems to be normal. (less than 'threshold' seconds.)
|
|
|
|
#if CONFIG_USE_LARGER_PTS_AS_START_TIME
|
|
//* wait some time for the video first pts run to nFirstAudioRenderPts
|
|
//* by discarding video first picture.
|
|
nWaitTime = 0;
|
|
while(p->nFirstAudioRenderPts > (p->nFirstVideoRenderPts + 50000))
|
|
{
|
|
if(p->bProcessingCommand || nWaitTime >= CONFIG_MAX_WAIT_TIME_FOR_SYNC)
|
|
{
|
|
//* need process new command, or wait too long, break waiting.
|
|
logw("break audio first frame waiting for clock.");
|
|
break;
|
|
}
|
|
|
|
usleep(50000);
|
|
nWaitTime += 50000;
|
|
}
|
|
|
|
//* set smaller as the start time.
|
|
pthread_mutex_lock(&p->timerMutex);
|
|
if(p->nFirstVideoRenderPts <= p->nFirstAudioRenderPts)
|
|
p->nTimeBase = p->nFirstVideoRenderPts;
|
|
else
|
|
p->nTimeBase = p->nFirstAudioRenderPts;
|
|
p->pAvTimer->SetTime(p->pAvTimer, p->nTimeBase);
|
|
p->pAvTimer->Start(p->pAvTimer);
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
#else
|
|
//* start timer and let audio wait sometime for clock run to nFirstAudioRenderPts.
|
|
pthread_mutex_lock(&p->timerMutex);
|
|
p->nTimeBase = p->nFirstVideoRenderPts;
|
|
p->pAvTimer->SetTime(p->pAvTimer, p->nTimeBase);
|
|
p->pAvTimer->Start(p->pAvTimer);
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
|
|
nWaitTime = 0;
|
|
while(p->nFirstAudioRenderPts >
|
|
(p->pAvTimer->GetTime(p->pAvTimer) + 50000))
|
|
{
|
|
if(p->bProcessingCommand || nWaitTime >= CONFIG_MAX_WAIT_TIME_FOR_SYNC)
|
|
{
|
|
//* need process new command, or wait too long, break waiting.
|
|
logw("break audio first frame waiting for clock.");
|
|
break;
|
|
}
|
|
|
|
usleep(50000);
|
|
nWaitTime += 50000;
|
|
}
|
|
#endif //* end of CONFIG_USE_LARGER_PTS_AS_START_TIME
|
|
}
|
|
else
|
|
{
|
|
//* time difference is more than 2 seconds, to prevend from pts jump
|
|
//* case, we set the base time to the audio first pts and let the
|
|
//* synchronize job to be done at pts call back process later.
|
|
logw("nFirstAudioRenderPts = %lld, nFirstVideoRenderPts == %lld, difference too big.", \
|
|
p->nFirstAudioRenderPts, p->nFirstVideoRenderPts);
|
|
|
|
//* reset the timer value to audio pts.
|
|
p->nTimeBase = p->nFirstAudioRenderPts;
|
|
pthread_mutex_lock(&p->timerMutex);
|
|
p->pAvTimer->SetTime(p->pAvTimer, p->nTimeBase);
|
|
p->pAvTimer->Start(p->pAvTimer);
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
|
|
//* What is the effect if time difference is more than 2 seconds at
|
|
//* the file start? It seems that video display will flush
|
|
//* quickly to catch the timer.
|
|
//* Any method to resolve this problem except enlarge the 2 seconds
|
|
//* limits? Oh, Live is so hard to me.
|
|
}
|
|
}
|
|
else //* p->nFirstVideoRenderPts > p->nFirstAudioRenderPts
|
|
{
|
|
//* audio first pts is smaller,
|
|
//* if CONFIG_USE_LARGER_PTS_AS_START_TIME is set,
|
|
//* choose video first pts as the base time, discard audio data before first picture;
|
|
//* else
|
|
//* choose the audio first pts as base time.
|
|
|
|
if((p->nFirstVideoRenderPts - p->nFirstAudioRenderPts) <=
|
|
CONFIG_THREASHOLD_OF_PTS_DIFFERENCE_TO_JUDGE_PTS_jUMP)
|
|
{
|
|
//* time difference seems to be normal.
|
|
#if CONFIG_USE_LARGER_PTS_AS_START_TIME
|
|
//* audio discard some data to follow the first picture pts.
|
|
if(p->nFirstVideoRenderPts - p->nFirstAudioRenderPts > 50000)
|
|
{
|
|
//* difference is more than 50ms.
|
|
//* do not start the timer, audio first frame callback will to come again after
|
|
//* some data is dropped.
|
|
return TIMER_DROP_AUDIO_DATA;
|
|
}
|
|
else
|
|
{
|
|
//* difference is less than 50ms, start the timer.
|
|
pthread_mutex_lock(&p->timerMutex);
|
|
p->nTimeBase = p->nFirstVideoRenderPts;
|
|
p->pAvTimer->SetTime(p->pAvTimer, p->nTimeBase);
|
|
p->pAvTimer->Start(p->pAvTimer);
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
}
|
|
#else
|
|
//* choose the audio first pts as the base time.
|
|
pthread_mutex_lock(&p->timerMutex);
|
|
p->nTimeBase = p->nFirstAudioRenderPts;
|
|
p->pAvTimer->SetTime(p->pAvTimer, p->nTimeBase);
|
|
p->pAvTimer->Start(p->pAvTimer);
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
#endif
|
|
}
|
|
else //* time difference is more than threshold to judge pts jump.
|
|
{
|
|
//* time difference is unnormal, to prevend from pts jump case,
|
|
//* we set the base time to the audio first pts and let the
|
|
//* synchronize job to be done at pts call back process later.
|
|
|
|
logw("nFirstAudioRenderPts = %lld, nFirstVideoRenderPts == %lld, difference too big.", \
|
|
p->nFirstAudioRenderPts, p->nFirstVideoRenderPts);
|
|
|
|
pthread_mutex_lock(&p->timerMutex);
|
|
p->nTimeBase = p->nFirstAudioRenderPts;
|
|
p->pAvTimer->SetTime(p->pAvTimer, p->nTimeBase);
|
|
p->pAvTimer->Start(p->pAvTimer);
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
|
|
//* What is the effect if time difference is more than 'theashold' seconds at
|
|
//* the file begin? It seems that video display will stuff
|
|
//* for several seconds to wait the timer.
|
|
//* Any method to resolve this problem except enlarge the 'theashold' seconds
|
|
//* limits? I hate those guys who make media files with media
|
|
//* streams interleaved so bad, don't let me have a gun!
|
|
}
|
|
}
|
|
}
|
|
else //* p->nFirstVideoRenderPts == -1
|
|
{
|
|
//* wait for video first frame too long, start timer anyway.
|
|
pthread_mutex_lock(&p->timerMutex);
|
|
p->nTimeBase = p->nFirstAudioRenderPts;
|
|
p->pAvTimer->SetTime(p->pAvTimer, p->nTimeBase);
|
|
p->pAvTimer->Start(p->pAvTimer);
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//* in case when video thread wait for audio first frame too long at its first frame call
|
|
//* back process, the timer was force started even first audio frame not received yet.
|
|
//* another case that the timer is in start status before audio first frame received is
|
|
//* that audio stream was just switched.
|
|
|
|
int64_t nCurTime;
|
|
|
|
nCurTime = p->pAvTimer->GetTime(p->pAvTimer);
|
|
if(nCurTime <= p->nFirstAudioRenderPts)
|
|
{
|
|
//* audio wait sometime for clock run to nFirstAudioRenderPts.
|
|
nWaitTime = 0;
|
|
if(p->nFirstAudioRenderPts - nCurTime <=
|
|
CONFIG_THREASHOLD_OF_PTS_DIFFERENCE_TO_JUDGE_PTS_jUMP)
|
|
{
|
|
//* time difference seems to be normal. (less than 2 seconds.)
|
|
//* audio wait sometime for clock run to nFirstAudioRenderPts.
|
|
while(p->nFirstAudioRenderPts >
|
|
(p->pAvTimer->GetTime(p->pAvTimer) + 50000))
|
|
{
|
|
if(p->bProcessingCommand || nWaitTime >= CONFIG_MAX_WAIT_TIME_FOR_SYNC)
|
|
{
|
|
//* need process new command, or wait too long, break waiting.
|
|
logw("break audio first frame waiting for clock.");
|
|
break;
|
|
}
|
|
|
|
usleep(50000);
|
|
nWaitTime += 50000;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//* time difference is more than 'threshold' seconds, to prevend from pts jump
|
|
//* case, we set the base time to the audio first pts and let the
|
|
//* synchronize job to be done at pts call back process later.
|
|
logw("nFirstAudioRenderPts = %lld, nFirstVideoRenderPts == %lld, difference too big.", \
|
|
p->nFirstAudioRenderPts, p->nFirstVideoRenderPts);
|
|
|
|
//* reset the timer value to audio pts.
|
|
p->nTimeBase = p->nFirstAudioRenderPts;
|
|
pthread_mutex_lock(&p->timerMutex);
|
|
p->pAvTimer->SetTime(p->pAvTimer, p->nTimeBase);
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
|
|
//* What is the effect if time difference is more than 2 seconds at
|
|
//* audio stream switch case? It seems that video display will flush
|
|
//* quickly to catch the timer. So we should control audio stream
|
|
//* carefully for multi audio stream at the audioDecComponent.cpp module.
|
|
}
|
|
}
|
|
else //* curTime > p->nFirstAudioRenderPts
|
|
{
|
|
//* audio first pts is smaller, we should discard some audio data
|
|
//* to catch up the timer.
|
|
if(nCurTime - p->nFirstAudioRenderPts <= CONFIG_THREASHOLD_OF_PTS_DIFFERENCE_TO_JUDGE_PTS_jUMP)
|
|
{
|
|
//* time difference seems to be normal. (less than 'threshold' seconds)
|
|
//* audio discard some data to catch up the timer.
|
|
if(p->nFirstAudioRenderPts + 50000 < nCurTime)
|
|
{
|
|
//* difference is more than 50ms.
|
|
//* discard audio data.
|
|
return TIMER_DROP_AUDIO_DATA;
|
|
}
|
|
}
|
|
else //* time difference is more than 2 seconds.
|
|
{
|
|
//* time difference is unnormal, to prevend from pts jump case,
|
|
//* we set the base time to the audio first pts and let the
|
|
//* synchronize job to be done at pts call back process later.
|
|
|
|
logw("timer had been started, current time = %lld, \
|
|
nFirstVideoRenderPts == %lld, difference too big.", \
|
|
nCurTime, p->nFirstVideoRenderPts);
|
|
|
|
p->nTimeBase = p->nFirstAudioRenderPts;
|
|
p->pAvTimer->SetTime(p->pAvTimer, p->nTimeBase);
|
|
|
|
//* What is the effect if time difference is more than 2 seconds at
|
|
//* audio stream switch case? It seems that video display will stuff
|
|
//* for several seconds to wait the timer. So we should control audio
|
|
//* stream carefully for multi audio stream at the audioDecComponent.cpp
|
|
//* module.
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
p->nPreAudioPts = p->nFirstAudioRenderPts;
|
|
p->nLastTimeTimerAdjusted = p->nFirstAudioRenderPts;
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_VIDEO_RENDER_NOTIFY_PICTURE_PTS:
|
|
{
|
|
int64_t nVideoPts;
|
|
int64_t nCurTime;
|
|
int64_t nTimeDiff;
|
|
int nWaitTimeMs;
|
|
|
|
if(p->bInFastMode)
|
|
{
|
|
//* in fast mode, video is showed without any synchronization,
|
|
//* audio data is discard if overtime.
|
|
p->nPreVideoPts = *((int64_t*)param);
|
|
return 0;
|
|
}
|
|
|
|
nVideoPts = *((int64_t*)param);
|
|
if(p->pVideoFramerateEstimater != NULL)
|
|
FramerateEstimaterUpdate(p->pVideoFramerateEstimater, nVideoPts);
|
|
|
|
nCurTime = p->pAvTimer->GetTime(p->pAvTimer);
|
|
nTimeDiff = nVideoPts - nCurTime;
|
|
|
|
logv("notify video pts = %lld ms, curTime = %lld ms, diff = %lld ms",
|
|
nVideoPts/1000, nCurTime/1000, nTimeDiff/1000);
|
|
|
|
if (p->onResetNotSync)
|
|
{
|
|
if (CdxAbs(nVideoPts - p->ptsBeforeReset) <= 2000000)
|
|
{
|
|
nWaitTimeMs = 0;
|
|
p->nPreVideoPts = nVideoPts;
|
|
return nWaitTimeMs;
|
|
}
|
|
else
|
|
{
|
|
p->onResetNotSync = 0;
|
|
}
|
|
}
|
|
|
|
if(!IS_DTMB_STREAM || (PlayerHasAudio((Player*)p) && (p->bAudioCrash == 0)))
|
|
{
|
|
if(CdxAbs(nTimeDiff) <= CONFIG_THREASHOLD_OF_PTS_DIFFERENCE_TO_JUDGE_PTS_jUMP)
|
|
{
|
|
//* time difference seems to be normal, video render thread will
|
|
//* wait to sync the timer if nTimeDiff>0 or hurry up to catch the
|
|
//* timer if nTimeDiff<0.
|
|
p->nUnsyncVideoFrameCnt = 0;
|
|
nWaitTimeMs = (int)(nTimeDiff/1000);
|
|
}
|
|
else
|
|
{
|
|
//* time difference unnormal.
|
|
if(PlayerHasAudio((Player*)p) == 0)
|
|
{
|
|
//* if no audio, just reset the timer, it should be a pts jump event.
|
|
if(p->nUnsyncVideoFrameCnt > 2)
|
|
{
|
|
nWaitTimeMs = 0;
|
|
p->pAvTimer->SetTime(p->pAvTimer, nVideoPts);
|
|
}
|
|
else
|
|
{
|
|
nWaitTimeMs = 40;
|
|
p->nUnsyncVideoFrameCnt++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//* it maybe a pts jump event,
|
|
//* if one of the video pts or audio pts jump first and the other
|
|
//* not jump yet, the video pts will get a big difference to the timer,
|
|
//* we just go for several frames for a while to see whether the other
|
|
//* guy (video pts or audio pts) will jump also.
|
|
//* here we go ahead for 30 frames at maximum.
|
|
if(p->nUnsyncVideoFrameCnt > 30)
|
|
{
|
|
//* we have go for more than 30 frames and the pts still not synchronized,
|
|
//* it seems not a pts jump event, it seem like a pts error or packet lost,
|
|
//* we force the video to hurry up(nWaitTimeMs<0) or wait(nWaitTime>0) to
|
|
//* sync to the timer.
|
|
//* (but for save, don't wait too long for a frame, 2 seconds)
|
|
if(nTimeDiff < -CONFIG_THREASHOLD_OF_PTS_DIFFERENCE_TO_JUDGE_PTS_jUMP)
|
|
nWaitTimeMs = -2000;
|
|
else
|
|
nWaitTimeMs = 2000;
|
|
}
|
|
else
|
|
{
|
|
//* wait for one frame duration.
|
|
if(p->pVideoFramerateEstimater != NULL)
|
|
nWaitTimeMs = FramerateEstimaterGetFrameDuration(p->pVideoFramerateEstimater)/1000;
|
|
else
|
|
nWaitTimeMs = 40;
|
|
p->nUnsyncVideoFrameCnt++;
|
|
}
|
|
|
|
//add by xuqi, for IPTV
|
|
nWaitTimeMs = 0;
|
|
}
|
|
logv("%d video frames not sync, force sync. vpts(%.3f) jztime(%.3f)",
|
|
p->nUnsyncVideoFrameCnt, nVideoPts/1000000.0, nCurTime/1000000.0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if(CdxAbs(nTimeDiff) <= CONFIG_THREASHOLD_OF_PTS_DIFFERENCE_TO_JUDGE_PTS_jUMP_DTMB)
|
|
{
|
|
p->nUnsyncVideoFrameCnt = 0;
|
|
nWaitTimeMs = (int)(nTimeDiff/1000);
|
|
}
|
|
else
|
|
{
|
|
//* time difference unnormal.
|
|
if(PlayerHasAudio((Player*)p) == 0 || (PlayerHasAudio((Player*)p) == 1 && p->bAudioCrash == 1))
|
|
{
|
|
if(p->nUnsyncVideoFrameCnt > 2)
|
|
{
|
|
nWaitTimeMs = 0;
|
|
p->pAvTimer->SetTime(p->pAvTimer, nVideoPts);
|
|
}
|
|
else
|
|
{
|
|
nWaitTimeMs = 40;
|
|
p->nUnsyncVideoFrameCnt++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if(p->nUnsyncVideoFrameCnt > 30)
|
|
{
|
|
if(nTimeDiff < -CONFIG_THREASHOLD_OF_PTS_DIFFERENCE_TO_JUDGE_PTS_jUMP)
|
|
nWaitTimeMs = -2000;
|
|
else
|
|
nWaitTimeMs = 2000;
|
|
}
|
|
else
|
|
{
|
|
//* wait for one frame duration.
|
|
if(p->pVideoFramerateEstimater != NULL)
|
|
nWaitTimeMs = FramerateEstimaterGetFrameDuration(p->pVideoFramerateEstimater)/1000;
|
|
else
|
|
nWaitTimeMs = 40;
|
|
p->nUnsyncVideoFrameCnt++;
|
|
}
|
|
|
|
nWaitTimeMs = 0;
|
|
}
|
|
logv("%d video frames not sync, force sync. vpts(%.3f) jztime(%.3f)",
|
|
p->nUnsyncVideoFrameCnt, nVideoPts/1000000.0, nCurTime/1000000.0);
|
|
}
|
|
}
|
|
|
|
p->nPreVideoPts = nVideoPts;
|
|
|
|
return nWaitTimeMs;
|
|
}
|
|
|
|
case PLAYER_AUDIO_RENDER_NOTIFY_PTS_AND_CACHETIME:
|
|
{
|
|
int64_t nAudioPts;
|
|
int64_t nCachedTimeInSoundDevice;
|
|
int64_t nTimeDiff;
|
|
int64_t nCurTime;
|
|
int64_t nCurAudioTime;
|
|
|
|
|
|
pthread_mutex_lock(&p->timerMutex);
|
|
|
|
nAudioPts = ((int64_t*)param)[0];
|
|
nCachedTimeInSoundDevice = ((int64_t*)param)[1];
|
|
nCurTime = p->pAvTimer->GetTime(p->pAvTimer);
|
|
nCurAudioTime = nAudioPts - nCachedTimeInSoundDevice;
|
|
nTimeDiff = nCurAudioTime - nCurTime;
|
|
|
|
if(p->bDiscardAudio)
|
|
{
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
return TIMER_DROP_AUDIO_DATA;
|
|
}
|
|
|
|
if(p->bInFastMode)
|
|
{
|
|
p->nPreAudioPts = nAudioPts;
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
|
|
//* discard audio in fast mode for IPTV
|
|
if(p->bDiscardAudio)
|
|
return TIMER_DROP_AUDIO_DATA;
|
|
|
|
//* in fast mode, video is showed without any synchronization,
|
|
//* audio data is discard if overtime.
|
|
//if(p->nPreAudioPts < (p->nPreVideoPts - 200*1000))
|
|
//{
|
|
// logw("TIMER_DROP_AUDIO_DATA");
|
|
// return TIMER_DROP_AUDIO_DATA;
|
|
//}
|
|
//else
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
}
|
|
|
|
logv("notify audio pts = %lld ms, curTime = %lld ms, diff = %lld ms, cacheTime = %lld ms",
|
|
nAudioPts/1000, nCurTime/1000, nTimeDiff/1000, nCachedTimeInSoundDevice/1000);
|
|
|
|
//* check whether audio pts jump.
|
|
if(CdxAbs(nTimeDiff) > CONFIG_THREASHOLD_OF_PTS_DIFFERENCE_TO_JUDGE_PTS_jUMP)
|
|
{
|
|
//* Time difference is unnormal, there may be a pts jump event in the streams.
|
|
//* Any way, we just adjust the time base to keep the play position goes normal.
|
|
if((nAudioPts > p->nPreAudioPts + CONFIG_THREASHOLD_OF_PTS_DIFFERENCE_TO_JUDGE_PTS_jUMP) ||
|
|
(nAudioPts + CONFIG_THREASHOLD_OF_PTS_DIFFERENCE_TO_JUDGE_PTS_jUMP < p->nPreAudioPts))
|
|
{
|
|
int64_t nCurPosition;
|
|
|
|
logw("audio pts jump, maybe pts loop back or data lost.");
|
|
//* update timebase.
|
|
nCurPosition = nCurTime - p->nTimeBase;
|
|
p->nTimeBase = nCurAudioTime - nCurPosition;
|
|
}
|
|
}
|
|
|
|
if(CdxAbs(nTimeDiff) > 100000) //* difference is more than 100 ms.
|
|
{
|
|
//* time difference is too big, we can not adjust the timer speed to make it
|
|
//* become synchronize in a short time,
|
|
//* so, just reset the timer, this will make the video display flush or stuff.
|
|
logw("reset the timer to %.3f, time difference is %.3f",
|
|
nCurAudioTime/1000000.0, nTimeDiff/1000000.0);
|
|
p->pAvTimer->SetTime(p->pAvTimer, nCurAudioTime);
|
|
p->nLastTimeTimerAdjusted = nAudioPts;
|
|
|
|
//for IPTV
|
|
if(CdxAbs(nTimeDiff) > 2000000)
|
|
{
|
|
p->onResetNotSync = 1;
|
|
p->ptsBeforeReset = nCurTime;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if((nAudioPts > p->nLastTimeTimerAdjusted + 1000000) ||
|
|
(p->nLastTimeTimerAdjusted > nAudioPts))
|
|
{
|
|
//* adjust timer speed to make the timer follow audio.
|
|
if(nTimeDiff > 50000)
|
|
p->pAvTimer->SetSpeed(p->pAvTimer, 1006);
|
|
else if(nTimeDiff > 40000)
|
|
p->pAvTimer->SetSpeed(p->pAvTimer, 1005);
|
|
else if(nTimeDiff > 30000)
|
|
p->pAvTimer->SetSpeed(p->pAvTimer, 1004);
|
|
else if(nTimeDiff > 20000)
|
|
p->pAvTimer->SetSpeed(p->pAvTimer, 1003);
|
|
else if(nTimeDiff > 10000)
|
|
p->pAvTimer->SetSpeed(p->pAvTimer, 1002);
|
|
else if(nTimeDiff > 5000)
|
|
p->pAvTimer->SetSpeed(p->pAvTimer, 1001);
|
|
else if(nTimeDiff < -50000)
|
|
p->pAvTimer->SetSpeed(p->pAvTimer, 994);
|
|
else if(nTimeDiff < -40000)
|
|
p->pAvTimer->SetSpeed(p->pAvTimer, 995);
|
|
else if(nTimeDiff < -30000)
|
|
p->pAvTimer->SetSpeed(p->pAvTimer, 996);
|
|
else if(nTimeDiff < -20000)
|
|
p->pAvTimer->SetSpeed(p->pAvTimer, 997);
|
|
else if(nTimeDiff < -10000)
|
|
p->pAvTimer->SetSpeed(p->pAvTimer, 998);
|
|
else if(nTimeDiff < -5000)
|
|
p->pAvTimer->SetSpeed(p->pAvTimer, 999);
|
|
else
|
|
p->pAvTimer->SetSpeed(p->pAvTimer, 1000);
|
|
|
|
p->nLastTimeTimerAdjusted = nAudioPts;
|
|
}
|
|
}
|
|
|
|
pthread_mutex_unlock(&p->timerMutex);
|
|
p->nPreAudioPts = nAudioPts;
|
|
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_SUBTITLE_RENDER_NOTIFY_ITEM_PTS_AND_DURATION:
|
|
{
|
|
//* check whether a subtitle item is time to show or expired.
|
|
int64_t nPts;
|
|
int64_t nEndTime;
|
|
int64_t nDuration;
|
|
int64_t nCurTime;
|
|
int64_t nCurInputPts;
|
|
int eTimerStatus;
|
|
int nStreamIndex;
|
|
int bExternal;
|
|
int bHasBeenShowed;
|
|
int nWaitTimeMs;
|
|
|
|
nPts = ((int64_t*)param)[0];
|
|
nDuration = ((int64_t*)param)[1];
|
|
nEndTime = nPts + nDuration;
|
|
|
|
bHasBeenShowed = (int)((int64_t*)param)[2];
|
|
nCurTime = p->pAvTimer->GetTime(p->pAvTimer);
|
|
|
|
if(p->bInFastMode)
|
|
{
|
|
//* in fast mode, video is showed without any synchronization,
|
|
//* audio and subtitle data is discard.
|
|
return -1; //* means discard subtitle item.
|
|
}
|
|
|
|
eTimerStatus = p->pAvTimer->GetStatus(p->pAvTimer);
|
|
if(eTimerStatus != TIMER_STATUS_START)
|
|
return 100; //* not started yet, wait 100ms for first audio and video frame arrive.
|
|
|
|
if(nCurTime >= nPts && nCurTime < nEndTime)
|
|
return 0; //* show this item.
|
|
|
|
nStreamIndex = SubtitleDecCompCurrentStreamIndex(p->pSubtitleDecComp);
|
|
bExternal = SubtitleDecCompIsExternalSubtitle(p->pSubtitleDecComp, nStreamIndex);
|
|
|
|
if(nCurTime < nPts)
|
|
{
|
|
if(bHasBeenShowed)
|
|
{
|
|
logw("subtitle item had been showed but now pts seems not match, may be timer loop back.");
|
|
return -1; //* means discard subtitle item.
|
|
}
|
|
else if(bExternal)
|
|
{
|
|
//* when (nPts-nCurTime) less than 1000 us, the nWaitTimeMs will be 0,
|
|
//* player.cpp will return 0 to subtitleRender.
|
|
//* if return 0, subtitleRender will send the subItem to display(it is not the time to the subItem to display),
|
|
//* and when check the subItem whether had expired , it will return -1 because (nCurTime < nPts) and (bHasBeenShowed==1)
|
|
//* so we will see that the subItem will disappear.
|
|
//* here we enlarge 999 to avoid the unexpected wrong.
|
|
nWaitTimeMs = (int)((nPts-nCurTime+999)/1000);
|
|
return nWaitTimeMs;
|
|
}
|
|
else
|
|
{
|
|
if(nPts-nCurTime > CONFIG_THREASHOLD_OF_PTS_DIFFERENCE_TO_JUDGE_PTS_jUMP_FOR_SUBTITLE)
|
|
{
|
|
//* time difference more than 'threshold_subtitle' second, it should not happen in embedded subtitle
|
|
//* except it is a pts loop back or
|
|
//* it is a bad interleave file(subtitle is put ahead the video and audio too much).
|
|
//* check whether it is a pts loop back case,
|
|
//* if it is, the subtitle is expired, should drop it.
|
|
|
|
//* if it is cased by a clock loop back,
|
|
//* we should have nPts > nLastInputPts > nCurTime;
|
|
//* (except user submit too much stream of m3u8/dash source, in which case pts always loop back in a short time.)
|
|
if(nPts > p->nLastInputPts)
|
|
{
|
|
//* PTS loop back like this:
|
|
//* small big small big
|
|
//* pts: |------------| |-------------|
|
|
//* | | |
|
|
//* nPts | |
|
|
//* curTime |
|
|
//* nLastInputPts
|
|
return -1; //* means discard subtitle item.
|
|
}
|
|
else
|
|
{
|
|
//* to prevent from subtitle stream stuff cased by invalid judgement,
|
|
//* check whether the subtitle stream buffer is full,
|
|
//* if so, discard this subtitle item.
|
|
int nStreamDataSize;
|
|
int nStreamBufferSize;
|
|
int nFullness;
|
|
|
|
nStreamDataSize = SubtitleDecCompStreamBufferSize(p->pSubtitleDecComp, nStreamIndex);
|
|
nStreamBufferSize = SubtitleDecCompStreamDataSize(p->pSubtitleDecComp, nStreamIndex);
|
|
nFullness = nStreamDataSize*100/nStreamBufferSize;
|
|
if(nFullness >= 90)
|
|
return -1; //* means discard subtitle item.
|
|
}
|
|
}
|
|
|
|
//* when (nPts-nCurTime) less than 1000 us, the nWaitTimeMs will be 0,
|
|
//* player.cpp will return 0 to subtitleRender.
|
|
//* if return 0, subtitleRender will send the subItem to display(it is not the time to the subItem to display),
|
|
//* and when check the subItem whether had expired , it will return -1 because (nCurTime < nPts) and (bHasBeenShowed==1)
|
|
//* so we will see that the subItem will disappear.
|
|
//* here we enlarge 999 to avoid the unexpected wrong.
|
|
nWaitTimeMs = (int)((nPts-nCurTime+999)/1000);
|
|
return nWaitTimeMs;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//* nCurTime >= nEndTime.
|
|
if(bHasBeenShowed)
|
|
return -1; //* means discard subtitle item.
|
|
else if(bExternal)
|
|
{
|
|
return -1; //* means discard subtitle item.
|
|
}
|
|
else
|
|
{
|
|
if(nCurTime-nEndTime > 2000000) //* here we set the threshold as 2 seconds.
|
|
{
|
|
//* time difference more than 'threshold_subtitle' second, check whether it is a loop back case,
|
|
//* if it is, wait for the timer loop back.
|
|
|
|
//* if it is cased by a clock loop back,
|
|
//* we should have nCurTime > nLastInputPts > nPts; otherwise it
|
|
//* should be a normal case that the subtitle is expired. we should have nLastInputPts >= nCurTime > nEndTime;
|
|
if(p->nLastInputPts >= nCurTime)
|
|
{
|
|
return -1;
|
|
}
|
|
else
|
|
{
|
|
//* PTS loop back like this:
|
|
//* small big small big
|
|
//* pts: |------------| |-------------|
|
|
//* | | |
|
|
//* curTime | |
|
|
//* nPts |
|
|
//* nLastInputPts
|
|
|
|
//* to prevent from subtitle stream stuff cased by invalid judgement,
|
|
//* check whether the subtitle stream buffer is full,
|
|
//* if so, discard this subtitle item.
|
|
int nStreamDataSize;
|
|
int nStreamBufferSize;
|
|
int nFullness;
|
|
nStreamDataSize = SubtitleDecCompStreamBufferSize(p->pSubtitleDecComp, nStreamIndex);
|
|
nStreamBufferSize = SubtitleDecCompStreamDataSize(p->pSubtitleDecComp, nStreamIndex);
|
|
nFullness = nStreamDataSize*100/nStreamBufferSize;
|
|
if(nFullness >= 90)
|
|
return -1; //* means discard subtitle item.
|
|
|
|
return 100; //* wait 100ms for the timer loop back.
|
|
}
|
|
}
|
|
|
|
return -1; //* means discard subtitle item.
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case PLAYER_SUBTITLE_RENDER_NOTIFY_ITEM_AVAILABLE:
|
|
{
|
|
//* notify to draw a subtitle item.
|
|
//* subtitle_id = ((unsigned int*)param)[0];
|
|
//* pSubtitleItem = (SubtitleItem*)((unsigned int*)param)[1];
|
|
if(p->callback)
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_SUBTITLE_ITEM_AVAILABLE, param);
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_SUBTITLE_RENDER_NOTIFY_ITEM_EXPIRED:
|
|
{
|
|
//* notify to clear a subtitle item.
|
|
//* subtitle_id = (unsigned int)param;
|
|
if(p->callback)
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_SUBTITLE_ITEM_EXPIRED, param);
|
|
return 0;
|
|
}
|
|
|
|
case PLAYER_VIDEO_DECODER_NOTIFY_STREAM_UNDERFLOW:
|
|
case PLAYER_AUDIO_DECODER_NOTIFY_STREAM_UNDERFLOW:
|
|
case PLAYER_SUBTITLE_DECODER_NOTIFY_STREAM_UNDERFLOW:
|
|
break;
|
|
|
|
case PLAYER_VIDEO_DECODER_NOTIFY_CRASH:
|
|
|
|
pthread_mutex_lock(&p->eosMutex);
|
|
|
|
p->bVideoCrash = 1;
|
|
if(p->callback != NULL)
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_VIDEO_UNSUPPORTED, NULL);
|
|
|
|
if(p->bAudioRenderEosReceived == 1 || PlayerHasAudio((Player*)p) == 0 || p->bAudioCrash == 1)
|
|
{
|
|
if(p->callback != NULL)
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_EOS, NULL);
|
|
}
|
|
pthread_mutex_unlock(&p->eosMutex);
|
|
break;
|
|
|
|
case PLAYER_VIDEO_RENDER_NOTIFY_VIDEO_FRAME:
|
|
if(p->callback != NULL)
|
|
{
|
|
logv("===== notify render key frame in fast mode");
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_VIDEO_RENDER_FRAME, NULL);
|
|
}
|
|
break;
|
|
|
|
case PLAYER_AUDIO_DECODER_NOTIFY_CRASH:
|
|
|
|
pthread_mutex_lock(&p->eosMutex);
|
|
|
|
p->bAudioCrash = 1;
|
|
if(p->callback != NULL)
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_AUDIO_UNSUPPORTED, NULL);
|
|
|
|
if(p->bVideoRenderEosReceived == 1 || PlayerHasVideo((Player*)p) == 0 || p->bVideoCrash == 1)
|
|
{
|
|
if(p->callback != NULL)
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_EOS, NULL);
|
|
}
|
|
pthread_mutex_unlock(&p->eosMutex);
|
|
break;
|
|
|
|
case PLAYER_SUBTITLE_DECODER_NOTIFY_CRASH:
|
|
|
|
pthread_mutex_lock(&p->eosMutex);
|
|
|
|
p->bSubtitleCrash = 1;
|
|
if(p->callback != NULL)
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_SUBTITLE_UNSUPPORTED, NULL);
|
|
|
|
pthread_mutex_unlock(&p->eosMutex);
|
|
break;
|
|
|
|
case PLAYER_AUDIO_DECODER_NOTIFY_AUDIORAWPLAY:
|
|
loge("%s,1",__func__);
|
|
if(p->callback != NULL)
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_AUDIORAWPLAY, param);
|
|
loge("%s,2",__func__);
|
|
break;
|
|
|
|
case PLAYER_VIDEO_DECODER_NOTIFY_SET_SECURE_BUFFER_COUNT:
|
|
|
|
if(p->callback != NULL)
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_SET_SECURE_BUFFER_COUNT, param);
|
|
break;
|
|
|
|
case PLAYER_VIDEO_DECODER_NOTIFY_SET_SECURE_BUFFERS:
|
|
|
|
if(p->callback != NULL)
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_SET_SECURE_BUFFERS, param);
|
|
break;
|
|
|
|
case PLAYER_AUDIO_RENDER_NOTIFY_AUDIO_INFO:
|
|
|
|
if(p->callback != NULL)
|
|
{
|
|
int nAudioInfo[4];
|
|
nAudioInfo[0] = p->nAudioStreamSelected;
|
|
nAudioInfo[1] = ((int *)param)[0];
|
|
nAudioInfo[2] = ((int *)param)[1];
|
|
nAudioInfo[3] = ((int *)param)[2] > 0 ? ((int *)param)[2] : 320*1000;
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_AUDIO_INFO, (void *)nAudioInfo);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
static int PlayerInitialVideo(PlayerContext* p)
|
|
{
|
|
if(p->pVideoRender != NULL)
|
|
{
|
|
VideoRenderCompDestroy(p->pVideoRender);
|
|
p->pVideoRender = NULL;
|
|
}
|
|
|
|
if(p->pVideoDecComp != NULL)
|
|
{
|
|
VideoDecCompDestroy(p->pVideoDecComp);
|
|
p->pVideoDecComp = NULL;
|
|
}
|
|
|
|
if(p->pVideoBitrateEstimater != NULL)
|
|
{
|
|
BitrateEstimaterDestroy(p->pVideoBitrateEstimater);
|
|
p->pVideoBitrateEstimater = NULL;
|
|
}
|
|
|
|
if(p->pVideoFramerateEstimater != NULL)
|
|
{
|
|
FramerateEstimaterDestroy(p->pVideoFramerateEstimater);
|
|
p->pVideoFramerateEstimater = NULL;
|
|
}
|
|
|
|
p->sVideoCropWindow[0] = 0;
|
|
p->sVideoCropWindow[1] = 0;
|
|
p->sVideoCropWindow[2] = 0;
|
|
p->sVideoCropWindow[3] = 0;
|
|
|
|
p->pVideoDecComp = VideoDecCompCreate();
|
|
|
|
#if (OUTPUT_PIXEL_FORMAT == OUTPUT_PIXEL_FORMAT_YV12)
|
|
p->sVideoConfig.eOutputPixelFormat = PIXEL_FORMAT_YV12;
|
|
#elif (OUTPUT_PIXEL_FORMAT == OUTPUT_PIXEL_FORMAT_NV21)
|
|
p->sVideoConfig.eOutputPixelFormat = PIXEL_FORMAT_NV21;
|
|
#elif (OUTPUT_PIXEL_FORMAT == OUTPUT_PIXEL_FORMAT_MB32)
|
|
p->sVideoConfig.eOutputPixelFormat = PIXEL_FORMAT_YUV_MB32_420;
|
|
#endif
|
|
|
|
//* We should set format to nv21 if it is 3D stream except chip-1639,
|
|
//* Because new-display-double-stream need nv21 format
|
|
#if(NEW_DISPLAY_DOUBLE_STREAM_NEED_NV21 == 1 && USE_NEW_DISPLAY == 1)
|
|
if(p->pVideoStreamInfo->bIs3DStream == 1)
|
|
p->sVideoConfig.eOutputPixelFormat = PIXEL_FORMAT_NV21;
|
|
#endif
|
|
|
|
p->sVideoConfig.bSecureosEn = p->pVideoStreamInfo->bSecureStreamFlagLevel1;
|
|
|
|
p->sVideoConfig.nDisplayHoldingFrameBufferNum = NUM_OF_PICTURES_KEEP_IN_LIST;
|
|
p->sVideoConfig.nDeInterlaceHoldingFrameBufferNum = NUM_OF_PICTURES_KEEPPED_BY_DEINTERLACE;
|
|
p->sVideoConfig.nRotateHoldingFrameBufferNum = NUM_OF_PICTURES_KEEPPED_BY_ROTATE;
|
|
p->sVideoConfig.nDecodeSmoothFrameBufferNum = NUM_OF_PICTURES_FOR_EXTRA_SMOOTH;
|
|
|
|
#if (CONFIG_CHIP == OPTION_CHIP_C500)
|
|
p->sVideoConfig.nVbvBufferSize = 2*1024*1024;
|
|
p->sVideoConfig.nDeInterlaceHoldingFrameBufferNum = 0;
|
|
p->sVideoConfig.nDisplayHoldingFrameBufferNum = 0;
|
|
p->sVideoConfig.nRotateHoldingFrameBufferNum = 0;
|
|
p->sVideoConfig.nDecodeSmoothFrameBufferNum = 1;
|
|
logd("set vbv buf [2M] in c500");
|
|
#endif
|
|
|
|
|
|
if(p->pVideoStreamInfo->nWidth >= (3840-256) || p->pVideoStreamInfo->nHeight >= (2160-256))
|
|
p->sVideoConfig.nDecodeSmoothFrameBufferNum = 0;
|
|
|
|
//* Because in the function of VideoDecCompSetVideoStreamInfo, we will call
|
|
//* the callback function to do something, so we must setCallback before it.
|
|
VideoDecCompSetCallback(p->pVideoDecComp, CallbackProcess, (void*)p);
|
|
VideoDecCompSetTimer(p->pVideoDecComp, p->pAvTimer);
|
|
|
|
p->bUnSurpportVideoFlag = 0;
|
|
if(VideoDecCompSetVideoStreamInfo(p->pVideoDecComp, p->pVideoStreamInfo, &p->sVideoConfig) != 0)
|
|
{
|
|
loge("video stream is not supported.");
|
|
|
|
if(p->callback != NULL)
|
|
{
|
|
p->callback(p->pUserData, PLAYER_NOTIFY_VIDEO_UNSUPPORTED, NULL);
|
|
}
|
|
|
|
return -1;
|
|
#if 0
|
|
/* in this case the docoder did process, but audio and subtitle will be work */
|
|
|
|
p->bUnSurpportVideoFlag = 1;
|
|
|
|
if(p->pUnSurpportVideoBuffer != NULL)
|
|
free(p->pUnSurpportVideoBuffer);
|
|
|
|
p->nUnSurpportVideoBufferSize = 256*1024;
|
|
p->pUnSurpportVideoBuffer = malloc(p->nUnSurpportVideoBufferSize);
|
|
if(p->pUnSurpportVideoBuffer == NULL)
|
|
{
|
|
loge("malloc buffer failed");
|
|
return -1;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
p->pVideoRender = VideoRenderCompCreate();
|
|
if(p->pVideoRender == NULL)
|
|
{
|
|
loge("create video render fail.");
|
|
VideoDecCompDestroy(p->pVideoDecComp);
|
|
p->pVideoDecComp = NULL;
|
|
return -1;
|
|
}
|
|
|
|
if(p->pLayerControlOps)
|
|
{
|
|
VideoRenderCompSetLayerCtlOps(p->pVideoRender, p->pLayerControlOps);
|
|
}
|
|
|
|
//* set video stream info for videorender
|
|
VideoRenderCompSetVideoStreamInfo(p->pVideoRender, p->pVideoStreamInfo);
|
|
|
|
//* tall the render whether the output buffer is secure(should be protect by native window) or not
|
|
VideoRenderCompSetProtecedFlag(p->pVideoRender,p->pVideoStreamInfo->bSecureStreamFlag);
|
|
VideoRenderCompSetCallback(p->pVideoRender, CallbackProcess, (void*)p);
|
|
VideoRenderCompSetTimer(p->pVideoRender, p->pAvTimer);
|
|
VideoRenderCompSetDecodeComp(p->pVideoRender, p->pVideoDecComp);
|
|
|
|
#if(CONFIG_CMCC==OPTION_CMCC_YES)
|
|
VideoRenderCompSetSyncFirstPictureFlag(p->pVideoRender, 1);
|
|
#else
|
|
VideoRenderCompSetSyncFirstPictureFlag(p->pVideoRender, 0);
|
|
#endif
|
|
if(p->pVideoStreamInfo->bIs3DStream)
|
|
VideoRenderSet3DMode(p->pVideoRender, PICTURE_3D_MODE_TWO_SEPERATED_PICTURE, DISPLAY_3D_MODE_2D);
|
|
if(p->pNativeWindow != NULL)
|
|
VideoRenderCompSetWindow(p->pVideoRender, p->pNativeWindow);
|
|
|
|
p->pVideoBitrateEstimater = BitrateEstimaterCreate();
|
|
p->pVideoFramerateEstimater = FramerateEstimaterCreate();
|
|
|
|
VideoRenderCompSetFrameRateEstimater(p->pVideoRender, p->pVideoFramerateEstimater);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static int PlayerInitialAudio(PlayerContext* p)
|
|
{
|
|
int ret;
|
|
|
|
if(p->pAudioRender != NULL)
|
|
{
|
|
AudioRenderCompDestroy(p->pAudioRender);
|
|
p->pAudioRender = NULL;
|
|
}
|
|
|
|
if(p->pAudioDecComp != NULL)
|
|
{
|
|
AudioDecCompDestroy(p->pAudioDecComp);
|
|
p->pAudioDecComp = NULL;
|
|
}
|
|
|
|
if(p->pAudioBitrateEstimater != NULL)
|
|
{
|
|
BitrateEstimaterDestroy(p->pAudioBitrateEstimater);
|
|
p->pAudioBitrateEstimater = NULL;
|
|
}
|
|
|
|
p->pAudioDecComp = AudioDecCompCreate();
|
|
ret = AudioDecCompSetAudioStreamInfo(p->pAudioDecComp,
|
|
p->pAudioStreamInfo,
|
|
p->nAudioStreamNum,
|
|
p->nAudioStreamSelected);
|
|
|
|
if(ret != 0)
|
|
{
|
|
loge("selected audio stream is not supported, call SwitchAudio() to select another stream.");
|
|
/* in this case the docoder did process, but video and subtitle will be work */
|
|
}
|
|
|
|
AudioDecCompSetCallback(p->pAudioDecComp, CallbackProcess, (void*)p);
|
|
AudioDecCompSetTimer(p->pAudioDecComp, p->pAvTimer);
|
|
|
|
p->pAudioRender = AudioRenderCompCreate();
|
|
if(p->pAudioRender == NULL)
|
|
{
|
|
loge("create audio render fail.");
|
|
AudioDecCompDestroy(p->pAudioDecComp);
|
|
p->pAudioDecComp = NULL;
|
|
return -1;
|
|
}
|
|
|
|
if(p->pSoundControlOps)
|
|
{
|
|
AudioRenderSetSoundCtlOps(p->pAudioRender, p->pSoundControlOps);
|
|
}
|
|
|
|
AudioRenderCompSetVolume(p->pAudioRender, p->volume);
|
|
AudioRenderCompSetCallback(p->pAudioRender, CallbackProcess, (void*)p);
|
|
AudioRenderCompSetTimer(p->pAudioRender, p->pAvTimer);
|
|
AudioRenderCompSetDecodeComp(p->pAudioRender, p->pAudioDecComp);
|
|
if(p->pAudioSink != NULL)
|
|
AudioRenderCompSetAudioSink(p->pAudioRender, p->pAudioSink);
|
|
|
|
p->pAudioBitrateEstimater = BitrateEstimaterCreate();
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static int PlayerInitialSubtitle(PlayerContext* p)
|
|
{
|
|
int ret;
|
|
|
|
if(p->pSubtitleRender != NULL)
|
|
{
|
|
SubtitleRenderCompDestroy(p->pSubtitleRender);
|
|
p->pSubtitleRender = NULL;
|
|
}
|
|
|
|
if(p->pSubtitleDecComp != NULL)
|
|
{
|
|
SubtitleDecCompDestroy(p->pSubtitleDecComp);
|
|
p->pSubtitleDecComp = NULL;
|
|
}
|
|
|
|
p->pSubtitleDecComp = SubtitleDecCompCreate();
|
|
ret = SubtitleDecCompSetSubtitleStreamInfo(p->pSubtitleDecComp,
|
|
p->pSubtitleStreamInfo,
|
|
p->nSubtitleStreamNum,
|
|
p->nSubtitleStreamSelected);
|
|
|
|
if(ret != 0)
|
|
{
|
|
loge("selected subtitle stream is not supported, call SwitchSubtitle() to select another stream.");
|
|
/* in this case the docoder did process, but video and audio will be work */
|
|
}
|
|
|
|
SubtitleDecCompSetCallback(p->pSubtitleDecComp, CallbackProcess, (void*)p);
|
|
SubtitleDecCompSetTimer(p->pSubtitleDecComp, p->pAvTimer);
|
|
|
|
p->pSubtitleRender = SubtitleRenderCompCreate();
|
|
if(p->pSubtitleRender == NULL)
|
|
{
|
|
loge("create subtitle render fail.");
|
|
SubtitleDecCompDestroy(p->pSubtitleDecComp);
|
|
p->pSubtitleDecComp = NULL;
|
|
return -1;
|
|
}
|
|
SubtitleRenderCompSetVideoOrAudioFirstPts(p->pSubtitleRender, p->nFirstPts);
|
|
SubtitleRenderCompSetCallback(p->pSubtitleRender, CallbackProcess, (void*)p);
|
|
SubtitleRenderCompSetTimer(p->pSubtitleRender, p->pAvTimer);
|
|
SubtitleRenderCompSetDecodeComp(p->pSubtitleRender, p->pSubtitleDecComp);
|
|
|
|
#if( CONFIG_ALI_YUNOS == OPTION_ALI_YUNOS_YES)
|
|
ret = SubtitleRenderCompSetSubtitleStreamInfo(p->pSubtitleRender,
|
|
p->pSubtitleStreamInfo,
|
|
p->nSubtitleStreamNum,
|
|
p->nSubtitleStreamSelected);
|
|
if(ret != 0)
|
|
{
|
|
loge("Set Subtitle StreamInfo to RenderComp fail.");
|
|
}
|
|
#endif
|
|
|
|
return 0;
|
|
}
|
|
int PlayerSetVolume(Player* pl, float volume)
|
|
{
|
|
PlayerContext* p;
|
|
|
|
p = (PlayerContext*)pl;
|
|
p->volume = volume;
|
|
if(!p->pAudioRender)
|
|
{
|
|
logd("!p->pAudioRender");
|
|
return -1;
|
|
}
|
|
return AudioRenderCompSetVolume(p->pAudioRender, p->volume);
|
|
}
|
|
int PlayerGetVolume(Player* pl, float *volume)
|
|
{
|
|
logd("GetVolume");
|
|
PlayerContext* p;
|
|
p = (PlayerContext*)pl;
|
|
*volume = p->volume;
|
|
if(!p->pAudioRender)
|
|
{
|
|
logd("!p->pAudioRender");
|
|
return 0;
|
|
}
|
|
return AudioRenderCompGetVolume(p->pAudioRender, volume);
|
|
}
|
|
|
|
|
|
//*****************************************************************************//
|
|
//*****************************************************************************//
|
|
//*****************************************************************************//
|
|
|
|
int PlayerConfigExtraScaleInfo(Player* pl, int nWidthTh, int nHeightTh, int nHorizontalScaleRatio, int nVerticalScaleRatio)
|
|
{
|
|
PlayerContext* p;
|
|
p = (PlayerContext*)pl;
|
|
|
|
if(p->pVideoDecComp != NULL)
|
|
{
|
|
//return VideoDecCompSetDropDelayFrames(p->pVideoDecComp, bDropLaytedFrame); //* only treat the major video stream.
|
|
return VideoDecCompSetExtraScaleInfo(p->pVideoDecComp,nWidthTh, nHeightTh, nHorizontalScaleRatio, nVerticalScaleRatio);
|
|
}
|
|
return 0;
|
|
}
|