2205 lines
72 KiB
C
2205 lines
72 KiB
C
|
|
||
|
/*
|
||
|
* Copyright (c) 2008-2016 Allwinner Technology Co. Ltd.
|
||
|
* All rights reserved.
|
||
|
*
|
||
|
* File : vdecoder.c
|
||
|
* Description :
|
||
|
* History :
|
||
|
* Author : xyliu <xyliu@allwinnertech.com>
|
||
|
* Date : 2016/04/13
|
||
|
* Comment :
|
||
|
*
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#include <pthread.h>
|
||
|
#include <memory.h>
|
||
|
#include "vdecoder.h"
|
||
|
#include "sbm.h"
|
||
|
#include "fbm.h"
|
||
|
#include "videoengine.h"
|
||
|
#include "veAdapter.h"
|
||
|
#include "CdcUtil.h"
|
||
|
|
||
|
#include "memoryAdapter.h"
|
||
|
#include "gralloc_metadata/sunxi_metadata_def.h"
|
||
|
|
||
|
#include <log.h>
|
||
|
#include <cdc_version.h>
|
||
|
#include <stdio.h>
|
||
|
|
||
|
#include "sunxi_tr.h"
|
||
|
#include <sys/types.h>
|
||
|
#include <sys/stat.h>
|
||
|
#include <sys/time.h>
|
||
|
#include <sys/ioctl.h>
|
||
|
#include <fcntl.h>
|
||
|
|
||
|
#define DEBUG_SAVE_BITSTREAM (0)
|
||
|
#define DEBUG_SAVE_PICTURE (0)
|
||
|
|
||
|
/* show decoder speed */
|
||
|
#define AW_VDECODER_SPEED_INFO (0)
|
||
|
#define DEBUG_SAVE_FRAME_TIME (0)
|
||
|
#define DEBUG_MAX_FRAME_IN_LIST 16
|
||
|
|
||
|
#define DEBUG_MAKE_SPECIAL_STREAM (0)
|
||
|
#define SPECIAL_STREAM_FILE "/data/camera/special.awsp"
|
||
|
#if DEBUG_SAVE_BITSTREAM
|
||
|
const char* fpStreamPath = "/data/camera/bitstream.dat";
|
||
|
static FILE* fpStream = NULL;
|
||
|
#endif
|
||
|
|
||
|
#define CHECK_METADATA_INFO (0)
|
||
|
|
||
|
//* setup the ve freq
|
||
|
|
||
|
#ifdef CONF_VE_FREQ_ENABLE_SETUP
|
||
|
#define VE_FREQ_SETTING_VALUE (696)
|
||
|
#endif
|
||
|
|
||
|
//* for check the proc info
|
||
|
#define ENABLE_SET_PROC_INFO (0)
|
||
|
#define SET_PROC_INFO_FREQ (60)
|
||
|
|
||
|
extern const char* strCodecFormat[];
|
||
|
|
||
|
const char* strDecodeResult[] =
|
||
|
{
|
||
|
"UNSUPPORTED",
|
||
|
"OK",
|
||
|
"FRAME_DECODED",
|
||
|
"CONTINU",
|
||
|
"KEYFRAME_DECODED",
|
||
|
"NO_FRAME_BUFFER",
|
||
|
"NO_BITSTREAM",
|
||
|
"RESOLUTION_CHANGE"
|
||
|
};
|
||
|
|
||
|
const char* strPixelFormat[] =
|
||
|
{
|
||
|
"DEFAULT", "YUV_PLANER_420", "YUV_PLANER_420",
|
||
|
"YUV_PLANER_444", "YV12", "NV1",
|
||
|
"YUV_MB32_420", "MB32_422", "MB32_444",
|
||
|
};
|
||
|
|
||
|
typedef struct VideoDecoderDebugContext
|
||
|
{
|
||
|
int nFrameNum;
|
||
|
int nFrameNumDuration;
|
||
|
int nThreshHoldNum;
|
||
|
float nMaxCostTime;
|
||
|
float nCostTimeList[DEBUG_MAX_FRAME_IN_LIST];
|
||
|
int64_t nStartTime;
|
||
|
int64_t nDuration;
|
||
|
int64_t nCurrentTime;
|
||
|
int64_t nHardwareCurrentTime;
|
||
|
int64_t nHardwareDuration;
|
||
|
int64_t nHardwareTotalTime;
|
||
|
int64_t nGetOneFrameFinishTime;
|
||
|
float nExtraDecodeTimeList[DEBUG_MAX_FRAME_IN_LIST];
|
||
|
float nFrameTimeList[DEBUG_MAX_FRAME_IN_LIST];
|
||
|
float nFrameSmoothTime;
|
||
|
int nWaitingForDispPicNum[DEBUG_MAX_FRAME_IN_LIST];
|
||
|
int nValidDisplayPic;
|
||
|
int nStartIndexExtra;
|
||
|
unsigned int *pFrameTime;
|
||
|
int nFrameTimeNum;
|
||
|
unsigned int *pDispFrameNum;
|
||
|
float nMax_hardDecodeFps;
|
||
|
float nMin_hardDecodeFps;
|
||
|
}VideoDecoderDebugContext;
|
||
|
|
||
|
typedef struct VideoDecoderContext
|
||
|
{
|
||
|
VConfig vconfig;
|
||
|
VideoStreamInfo videoStreamInfo;
|
||
|
VideoEngine* pVideoEngine;
|
||
|
int nFbmNum;
|
||
|
Fbm* pFbm[2];
|
||
|
int nSbmNum;
|
||
|
SbmInterface* pSbm[2];
|
||
|
VideoStreamDataInfo partialStreamDataInfo[2];
|
||
|
|
||
|
VideoDecoderDebugContext debug;
|
||
|
struct ScMemOpsS *memops;
|
||
|
int nIonFd;
|
||
|
int nSavePicCount;
|
||
|
}VideoDecoderContext;
|
||
|
|
||
|
static int DecideStreamBufferSize(VideoDecoderContext* p);
|
||
|
static void UpdateVideoStreamInfo(VideoDecoderContext* p, VideoPicture* pPicture);
|
||
|
static void CheckConfiguration(VideoDecoderContext* p);
|
||
|
extern void ConvertPixelFormat(VideoPicture* pPictureIn, VideoPicture* pPictureOut);
|
||
|
extern int RotatePicture0Degree(VideoPicture* pPictureIn,
|
||
|
VideoPicture* pPictureOut,
|
||
|
int nGpuYAlign,
|
||
|
int nGpuCAlign);
|
||
|
extern int RotatePicture90Degree(VideoPicture* pPictureIn, VideoPicture* pPictureOut);
|
||
|
extern int RotatePicture180Degree(VideoPicture* pPictureIn, VideoPicture* pPictureOut);
|
||
|
extern int RotatePicture270Degree(VideoPicture* pPictureIn, VideoPicture* pPictureOut);
|
||
|
|
||
|
static void printfMetaDataInfo(struct sunxi_metadata* pSunxi_metadata)
|
||
|
{
|
||
|
|
||
|
logd("metadata--hdr: d_x: %d, %d, %d, d_y: %d, %d, %d, w_x = %d, w_y = %d,\
|
||
|
maxDML = %d, minDML = %d",
|
||
|
pSunxi_metadata->hdr_smetada.disp_master.display_primaries_x[0],
|
||
|
pSunxi_metadata->hdr_smetada.disp_master.display_primaries_x[1],
|
||
|
pSunxi_metadata->hdr_smetada.disp_master.display_primaries_x[2],
|
||
|
pSunxi_metadata->hdr_smetada.disp_master.display_primaries_y[0],
|
||
|
pSunxi_metadata->hdr_smetada.disp_master.display_primaries_y[1],
|
||
|
pSunxi_metadata->hdr_smetada.disp_master.display_primaries_y[2],
|
||
|
pSunxi_metadata->hdr_smetada.disp_master.white_point_x,
|
||
|
pSunxi_metadata->hdr_smetada.disp_master.white_point_y,
|
||
|
pSunxi_metadata->hdr_smetada.disp_master.max_display_mastering_luminance,
|
||
|
pSunxi_metadata->hdr_smetada.disp_master.min_display_mastering_luminance);
|
||
|
logd("metadata--hdr: maxCLL = %d, maxFALL = %d",
|
||
|
pSunxi_metadata->hdr_smetada.maximum_content_light_level ,
|
||
|
pSunxi_metadata->hdr_smetada.maximum_frame_average_light_level);
|
||
|
|
||
|
char* pSignature = (char*)&pSunxi_metadata->afbc_head.signature;
|
||
|
logd("metadata-afbc: signature = %c%c%c%c, filehdrSize = %d, version = %d, bodySize = %d",
|
||
|
pSignature[0],pSignature[1],pSignature[2],pSignature[3],
|
||
|
pSunxi_metadata->afbc_head.filehdr_size,
|
||
|
pSunxi_metadata->afbc_head.version,
|
||
|
pSunxi_metadata->afbc_head.body_size);
|
||
|
logd("metadata-afbc: ncomponents = %d, header_layout = %d, yuvTran = %d, blockSplit = %d",
|
||
|
pSunxi_metadata->afbc_head.ncomponents,
|
||
|
pSunxi_metadata->afbc_head.header_layout,
|
||
|
pSunxi_metadata->afbc_head.yuv_transform,
|
||
|
pSunxi_metadata->afbc_head.block_split);
|
||
|
logd("metadata-afbc: inputbits: %d, %d, %d, %d",
|
||
|
pSunxi_metadata->afbc_head.inputbits[0],
|
||
|
pSunxi_metadata->afbc_head.inputbits[1],
|
||
|
pSunxi_metadata->afbc_head.inputbits[2],
|
||
|
pSunxi_metadata->afbc_head.inputbits[3]);
|
||
|
logd("metadata-afbc: block_w = %d, block_h = %d, w = %d, h = %d",
|
||
|
pSunxi_metadata->afbc_head.block_width,
|
||
|
pSunxi_metadata->afbc_head.block_height,
|
||
|
pSunxi_metadata->afbc_head.width,
|
||
|
pSunxi_metadata->afbc_head.height);
|
||
|
logd("metadata-afbc: left_crop = %d, top_crop = %d, block_layout = %d",
|
||
|
pSunxi_metadata->afbc_head.left_crop,
|
||
|
pSunxi_metadata->afbc_head.top_crop,
|
||
|
pSunxi_metadata->afbc_head.block_layout);
|
||
|
|
||
|
};
|
||
|
|
||
|
static inline int64_t VdecoderGetCurrentTime(void)
|
||
|
{
|
||
|
struct timeval tv;
|
||
|
int64_t time;
|
||
|
gettimeofday(&tv,NULL);
|
||
|
time = tv.tv_sec*1000000 + tv.tv_usec;
|
||
|
return time;
|
||
|
}
|
||
|
|
||
|
VideoDecoder* CreateVideoDecoder(void)
|
||
|
{
|
||
|
LogVersionInfo();
|
||
|
|
||
|
VideoDecoderContext* p;
|
||
|
|
||
|
logd("CreateVideoDecoder ****");
|
||
|
|
||
|
p = (VideoDecoderContext*)malloc(sizeof(VideoDecoderContext));
|
||
|
if(p == NULL)
|
||
|
{
|
||
|
loge("memory alloc fail.");
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
memset(p, 0, sizeof(VideoDecoderContext));
|
||
|
|
||
|
remove("/tmp/h265Sha.txt");
|
||
|
|
||
|
//* the partialStreamDataInfo is for data submit status recording.
|
||
|
p->partialStreamDataInfo[0].nStreamIndex = 0;
|
||
|
p->partialStreamDataInfo[0].nPts = -1;
|
||
|
p->partialStreamDataInfo[1].nStreamIndex = 1;
|
||
|
p->partialStreamDataInfo[1].nPts = -1;
|
||
|
|
||
|
p->nIonFd = open("/dev/ion", O_RDONLY, 0);
|
||
|
if (p->nIonFd <= 0)
|
||
|
{
|
||
|
loge("vdecoder open /dev/ion failed \n");
|
||
|
free(p);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
logv("create a video decoder with handle=%p", p);
|
||
|
|
||
|
#if DEBUG_SAVE_BITSTREAM
|
||
|
fpStream = fopen(fpStreamPath, "wb");
|
||
|
#endif
|
||
|
return (VideoDecoder*)p;
|
||
|
}
|
||
|
|
||
|
void DestroyVideoDecoder(VideoDecoder* pDecoder)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
//int bSecureosEn = 0;
|
||
|
//int bVirMallocSbm = 0;
|
||
|
|
||
|
logv("DestroyVideoDecoder, pDecoder=%p", pDecoder);
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
//bSecureosEn = p->vconfig.bSecureosEn;
|
||
|
//bVirMallocSbm = p->vconfig.bVirMallocSbm;
|
||
|
#if DEBUG_SAVE_FRAME_TIME
|
||
|
if(p->debug.pFrameTime != NULL)
|
||
|
{
|
||
|
FILE *fp = fopen("/data/camera/vdecoderFrameTime.txt", "wt");
|
||
|
if(fp != NULL)
|
||
|
{
|
||
|
int i;
|
||
|
loge(" vdecoder saving frame time. number: %d ", p->debug.nFrameTimeNum);
|
||
|
for(i = 0; i < p->debug.nFrameTimeNum; i++)
|
||
|
{
|
||
|
fprintf(fp, "%u\n", p->debug.pFrameTime[i]);
|
||
|
}
|
||
|
fclose(fp);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
loge("vdecoder.c save frame open file error ");
|
||
|
}
|
||
|
free(p->debug.pFrameTime);
|
||
|
p->debug.pFrameTime = NULL;
|
||
|
}
|
||
|
if(p->debug.pDispFrameNum != NULL)
|
||
|
{
|
||
|
FILE *fp = fopen("/data/camera/vdecoderHadSendDispNum.txt", "wt");
|
||
|
if(fp != NULL)
|
||
|
{
|
||
|
int i;
|
||
|
loge(" vdecoder saving frame time. number: %d ", p->debug.nFrameTimeNum);
|
||
|
for(i = 0; i < p->debug.nFrameTimeNum; i++)
|
||
|
{
|
||
|
fprintf(fp, "%u\n", p->debug.pDispFrameNum[i]);
|
||
|
}
|
||
|
fclose(fp);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
loge("vdecoder.c save frame open file error ");
|
||
|
}
|
||
|
free(p->debug.pDispFrameNum);
|
||
|
p->debug.pDispFrameNum = NULL;
|
||
|
}
|
||
|
|
||
|
#endif //DEBUG_SAVE_FRAME_TIME
|
||
|
|
||
|
//* Destroy the video engine first.
|
||
|
if(p->pVideoEngine != NULL)
|
||
|
{
|
||
|
VideoEngineDestroy(p->pVideoEngine);
|
||
|
}
|
||
|
|
||
|
//* the memory space for codec specific data was allocated in InitializeVideoDecoder.
|
||
|
if(p->videoStreamInfo.pCodecSpecificData != NULL)
|
||
|
free(p->videoStreamInfo.pCodecSpecificData);
|
||
|
|
||
|
if(p->memops)
|
||
|
CdcMemClose(p->memops);
|
||
|
|
||
|
close(p->nIonFd);
|
||
|
p->nIonFd = 0;
|
||
|
|
||
|
free(p);
|
||
|
|
||
|
#if DEBUG_SAVE_BITSTREAM
|
||
|
if(fpStream)
|
||
|
{
|
||
|
fclose(fpStream);
|
||
|
fpStream = NULL;
|
||
|
}
|
||
|
#endif
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
int InitializeVideoDecoder(VideoDecoder* pDecoder, VideoStreamInfo* pVideoInfo, VConfig* pVconfig)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
int nStreamBufferSize;
|
||
|
//int i;
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
//* if the caller had seted the nVeFreq, we should not change it
|
||
|
if(pVconfig->nVeFreq == 0)
|
||
|
{
|
||
|
#ifdef CONF_VE_FREQ_ENABLE_SETUP
|
||
|
pVconfig->nVeFreq = VE_FREQ_SETTING_VALUE;
|
||
|
#else
|
||
|
pVconfig->nVeFreq = 0;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
logd("*** pVconfig->nVeFreq = %d",pVconfig->nVeFreq);
|
||
|
|
||
|
pVconfig->bSetProcInfoEnable = ENABLE_SET_PROC_INFO;
|
||
|
pVconfig->nSetProcInfoFreq = SET_PROC_INFO_FREQ;
|
||
|
|
||
|
//* check codec format.
|
||
|
if(pVideoInfo->eCodecFormat > VIDEO_CODEC_FORMAT_MAX ||
|
||
|
pVideoInfo->eCodecFormat < VIDEO_CODEC_FORMAT_MIN)
|
||
|
{
|
||
|
loge("codec format(0x%x) invalid.", pVideoInfo->eCodecFormat);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
if(pVideoInfo->nHeight > 3240 || pVideoInfo->nWidth > 6144)//>6k
|
||
|
{
|
||
|
loge("pVideoInfo->nWidth = %d, pVideoInfo->nHeight = %d unsupported!!",
|
||
|
pVideoInfo->nWidth, pVideoInfo->nHeight);
|
||
|
return VDECODE_RESULT_UNSUPPORTED;
|
||
|
}
|
||
|
|
||
|
if(pVconfig->bSecureosEn == 1)
|
||
|
pVconfig->memops = SecureMemAdapterGetOpsS();
|
||
|
else
|
||
|
pVconfig->memops = MemAdapterGetOpsS();
|
||
|
|
||
|
p->memops = pVconfig->memops;
|
||
|
|
||
|
if(CdcMemOpen(p->memops) < 0)
|
||
|
return -1;
|
||
|
|
||
|
//* print video stream information.
|
||
|
{
|
||
|
logv("Video Stream Information:");
|
||
|
logv(" codec = %s",strCodecFormat[pVideoInfo->eCodecFormat - VIDEO_CODEC_FORMAT_MIN]);
|
||
|
logv(" width = %d pixels", pVideoInfo->nWidth);
|
||
|
logv(" height = %d pixels", pVideoInfo->nHeight);
|
||
|
logv(" frame rate = %d", pVideoInfo->nFrameRate);
|
||
|
logv(" frame duration = %d us", pVideoInfo->nFrameDuration);
|
||
|
logv(" aspect ratio = %d", pVideoInfo->nAspectRatio);
|
||
|
logv(" is 3D stream = %s", pVideoInfo->bIs3DStream ? "yes" : "no");
|
||
|
logv(" csd data len = %d", pVideoInfo->nCodecSpecificDataLen);
|
||
|
logv(" bIsFrameCtsTestFlag = %d", pVideoInfo->bIsFrameCtsTestFlag);
|
||
|
}
|
||
|
|
||
|
//* save the video stream information.
|
||
|
p->videoStreamInfo.eCodecFormat = pVideoInfo->eCodecFormat;
|
||
|
p->videoStreamInfo.nWidth = pVideoInfo->nWidth;
|
||
|
p->videoStreamInfo.nHeight = pVideoInfo->nHeight;
|
||
|
p->videoStreamInfo.nFrameRate = pVideoInfo->nFrameRate;
|
||
|
p->videoStreamInfo.nFrameDuration = pVideoInfo->nFrameDuration;
|
||
|
p->videoStreamInfo.nAspectRatio = pVideoInfo->nAspectRatio;
|
||
|
p->videoStreamInfo.bIs3DStream = pVideoInfo->bIs3DStream;
|
||
|
p->videoStreamInfo.bIsFramePackage = pVideoInfo->bIsFramePackage;
|
||
|
p->videoStreamInfo.nCodecSpecificDataLen = pVideoInfo->nCodecSpecificDataLen;
|
||
|
p->videoStreamInfo.bSecureStreamFlag = pVideoInfo->bSecureStreamFlag;
|
||
|
p->videoStreamInfo.bSecureStreamFlagLevel1 = pVideoInfo->bSecureStreamFlagLevel1;
|
||
|
p->videoStreamInfo.bIsFrameCtsTestFlag = pVideoInfo->bIsFrameCtsTestFlag;
|
||
|
|
||
|
if(p->videoStreamInfo.nCodecSpecificDataLen > 0)
|
||
|
{
|
||
|
int nSize = p->videoStreamInfo.nCodecSpecificDataLen;
|
||
|
char* pMem = (char*)malloc(nSize);
|
||
|
|
||
|
if(pMem == NULL)
|
||
|
{
|
||
|
p->videoStreamInfo.nCodecSpecificDataLen = 0;
|
||
|
loge("memory alloc fail.");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
memcpy(pMem, pVideoInfo->pCodecSpecificData, nSize);
|
||
|
p->videoStreamInfo.pCodecSpecificData = pMem;
|
||
|
}
|
||
|
|
||
|
if(pVconfig->nDecodeSmoothFrameBufferNum <= 0
|
||
|
|| pVconfig->nDecodeSmoothFrameBufferNum > 16)
|
||
|
{
|
||
|
logw("warning: the nDecodeSmoothFrameBufferNum is %d",
|
||
|
pVconfig->nDecodeSmoothFrameBufferNum);
|
||
|
}
|
||
|
if(pVconfig->nDeInterlaceHoldingFrameBufferNum <= 0
|
||
|
|| pVconfig->nDeInterlaceHoldingFrameBufferNum > 16)
|
||
|
{
|
||
|
logw("warning: the nDeInterlaceHoldingFrameBufferNum is %d",
|
||
|
pVconfig->nDeInterlaceHoldingFrameBufferNum);
|
||
|
}
|
||
|
if(pVconfig->nDisplayHoldingFrameBufferNum <= 0
|
||
|
|| pVconfig->nDisplayHoldingFrameBufferNum > 16)
|
||
|
{
|
||
|
logw("warning: the nDisplayHoldingFrameBufferNum is %d",
|
||
|
pVconfig->nDisplayHoldingFrameBufferNum);
|
||
|
}
|
||
|
if(pVideoInfo->nHeight >= 2880 || pVideoInfo->nWidth >= 5120)//5K
|
||
|
{
|
||
|
// pVconfig->bEnableAfbcFlag = 0;
|
||
|
pVconfig->bSecOutputEn = 1;
|
||
|
pVconfig->bScaleDownEn = 1;
|
||
|
pVconfig->nHorizonScaleDownRatio = 1;
|
||
|
pVconfig->nVerticalScaleDownRatio = 1;
|
||
|
}
|
||
|
|
||
|
//logi("=====enableAfbc = %d, ENABLE_AFBC = %d", pVconfig->bEnableAfbcFlag, ENABLE_AFBC);
|
||
|
memcpy(&p->vconfig, pVconfig, sizeof(VConfig));
|
||
|
|
||
|
//* create stream buffer.
|
||
|
nStreamBufferSize = DecideStreamBufferSize(p);
|
||
|
|
||
|
//* check and fix the configuration for decoder.
|
||
|
//CheckConfiguration(p);
|
||
|
|
||
|
//* create video engine.
|
||
|
p->pVideoEngine = VideoEngineCreate(&p->vconfig, &p->videoStreamInfo);
|
||
|
if(p->pVideoEngine == NULL)
|
||
|
{
|
||
|
loge("create video engine fail.");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
SbmConfig mSbmConfig;
|
||
|
memset(&mSbmConfig, 0, sizeof(SbmConfig));
|
||
|
mSbmConfig.bVirFlag = p->vconfig.bVirMallocSbm;
|
||
|
mSbmConfig.memops = p->memops;
|
||
|
mSbmConfig.nSbmBufferTotalSize = nStreamBufferSize;
|
||
|
mSbmConfig.veOpsS = p->vconfig.veOpsS;
|
||
|
mSbmConfig.pVeOpsSelf = p->vconfig.pVeOpsSelf;
|
||
|
mSbmConfig.bSecureVideoFlag = p->vconfig.bSecureosEn;
|
||
|
|
||
|
int nSbmType = SBM_TYPE_STREAM;
|
||
|
|
||
|
#if(ENABLE_SBM_FRAME_INTERFACE)
|
||
|
if(pVideoInfo->eCodecFormat == VIDEO_CODEC_FORMAT_H265 &&
|
||
|
p->pVideoEngine->bIsSoftwareDecoder == 0)
|
||
|
nSbmType = SBM_TYPE_FRAME;
|
||
|
else if(pVideoInfo->eCodecFormat == VIDEO_CODEC_FORMAT_H264)
|
||
|
nSbmType = SBM_TYPE_FRAME_AVC;
|
||
|
else
|
||
|
nSbmType = SBM_TYPE_STREAM;
|
||
|
#endif
|
||
|
p->pSbm[0] = GetSbmInterface(nSbmType);
|
||
|
|
||
|
if(p->pSbm[0] == NULL)
|
||
|
{
|
||
|
loge("create stream buffer fail.");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
if(SbmInit(p->pSbm[0], &mSbmConfig) != 0)
|
||
|
{
|
||
|
loge(" sbm init failed");
|
||
|
SbmDestroy(p->pSbm[0]);
|
||
|
p->pSbm[0] = NULL;
|
||
|
return -1;
|
||
|
}
|
||
|
VideoEngineSetSbm(p->pVideoEngine, p->pSbm[0], 0);
|
||
|
p->nSbmNum++;
|
||
|
|
||
|
if(p->videoStreamInfo.bIs3DStream)
|
||
|
{
|
||
|
p->pSbm[1] = GetSbmInterface(nSbmType);
|
||
|
if(p->pSbm[1] == NULL)
|
||
|
{
|
||
|
loge("create stream buffer fail.");
|
||
|
return -1;
|
||
|
}
|
||
|
if(SbmInit(p->pSbm[1], &mSbmConfig) != 0)
|
||
|
{
|
||
|
loge(" sbm init failed");
|
||
|
SbmDestroy(p->pSbm[1]);
|
||
|
p->pSbm[1] = NULL;
|
||
|
return -1;
|
||
|
}
|
||
|
VideoEngineSetSbm(p->pVideoEngine, p->pSbm[1], 1);
|
||
|
p->nSbmNum++;
|
||
|
}
|
||
|
|
||
|
#if DEBUG_SAVE_FRAME_TIME
|
||
|
if(p->debug.pFrameTime == NULL)
|
||
|
{
|
||
|
p->debug.pFrameTime = calloc(1, 4*1024*1024);
|
||
|
if(p->debug.pFrameTime == NULL)
|
||
|
{
|
||
|
loge(" save frame time calloc error ");
|
||
|
}
|
||
|
}
|
||
|
if(p->debug.pDispFrameNum == NULL)
|
||
|
{
|
||
|
p->debug.pDispFrameNum = calloc(1, 4*1024*1024);
|
||
|
if(p->debug.pDispFrameNum == NULL)
|
||
|
{
|
||
|
loge(" save frame time calloc error ");
|
||
|
}
|
||
|
}
|
||
|
#endif //DEBUG_SAVE_FRAME_TIM
|
||
|
#if DEBUG_SAVE_BITSTREAM
|
||
|
logd("%d\n", p->videoStreamInfo.nCodecSpecificDataLen);
|
||
|
logd("pts=0\n");
|
||
|
if(p->videoStreamInfo.nCodecSpecificDataLen > 0)
|
||
|
{
|
||
|
fwrite(p->videoStreamInfo.pCodecSpecificData, 1,
|
||
|
p->videoStreamInfo.nCodecSpecificDataLen, fpStream);
|
||
|
}
|
||
|
#endif //DEBUG_SAVE_BITSTREAM
|
||
|
#if DEBUG_MAKE_SPECIAL_STREAM
|
||
|
if(1)
|
||
|
{
|
||
|
remove(SPECIAL_STREAM_FILE); //if the awsp file exist, remove it
|
||
|
FILE *fp = fopen(SPECIAL_STREAM_FILE, "ab");
|
||
|
if(fp == NULL)
|
||
|
{
|
||
|
loge(" make special stream open file error, initial ");
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
char mask[16] = "awspcialstream";
|
||
|
logd(" make special stream save special data size: %d ", \
|
||
|
pVideoInfo->nCodecSpecificDataLen);
|
||
|
fwrite(mask, 14, sizeof(char), fp);
|
||
|
fwrite(&pVideoInfo->eCodecFormat, 1, sizeof(int), fp);
|
||
|
fwrite(&pVideoInfo->nWidth, 1, sizeof(int), fp);
|
||
|
fwrite(&pVideoInfo->nHeight, 1, sizeof(int), fp);
|
||
|
fwrite(&pVideoInfo->bIs3DStream, 1, sizeof(int), fp);
|
||
|
fwrite(&pVideoInfo->nCodecSpecificDataLen, 1, sizeof(int), fp);
|
||
|
if(pVideoInfo->nCodecSpecificDataLen != 0)
|
||
|
{
|
||
|
fwrite(pVideoInfo->pCodecSpecificData, 1, \
|
||
|
pVideoInfo->nCodecSpecificDataLen, fp);
|
||
|
}
|
||
|
fclose(fp);
|
||
|
}
|
||
|
}
|
||
|
#endif//DEBUG_MAKE_SPECIAL_STREAM
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void ResetVideoDecoder(VideoDecoder* pDecoder)
|
||
|
{
|
||
|
int i;
|
||
|
VideoDecoderContext* p;
|
||
|
|
||
|
logv("ResetVideoDecoder, pDecoder=%p", pDecoder);
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
//Version = VeGetIcVersion();
|
||
|
|
||
|
//* reset the video engine.
|
||
|
if(p->pVideoEngine)
|
||
|
{
|
||
|
VideoEngineReset(p->pVideoEngine);
|
||
|
}
|
||
|
|
||
|
//* flush pictures.
|
||
|
if(p->nFbmNum == 0 && p->pVideoEngine != NULL)
|
||
|
{
|
||
|
p->nFbmNum = VideoEngineGetFbmNum(p->pVideoEngine);
|
||
|
}
|
||
|
|
||
|
for(i=0; i<p->nFbmNum; i++)
|
||
|
{
|
||
|
if(p->pFbm[i] == NULL && p->pVideoEngine != NULL)
|
||
|
{
|
||
|
p->pFbm[i] = VideoEngineGetFbm(p->pVideoEngine, i);
|
||
|
}
|
||
|
|
||
|
if(p->pFbm[i] != NULL)
|
||
|
FbmFlush(p->pFbm[i]);
|
||
|
}
|
||
|
|
||
|
//* flush stream data.
|
||
|
for(i=0; i<p->nSbmNum; i++)
|
||
|
{
|
||
|
if(p->pSbm[i] != NULL)
|
||
|
SbmReset(p->pSbm[i]);
|
||
|
}
|
||
|
|
||
|
//* clear the partialStreamDataInfo.
|
||
|
memset(p->partialStreamDataInfo, 0, sizeof(VideoStreamDataInfo)*2);
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
int DecoderSetSpecialData(VideoDecoder* pDecoder, void *pArg)
|
||
|
{
|
||
|
int ret;
|
||
|
VideoDecoderContext* p;
|
||
|
DecoderInterface *pDecoderInterface;
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
if(p == NULL ||
|
||
|
p->pVideoEngine == NULL ||
|
||
|
p->pVideoEngine->pDecoderInterface == NULL)
|
||
|
{
|
||
|
loge(" set decoder special data fail ");
|
||
|
return VDECODE_RESULT_UNSUPPORTED;
|
||
|
}
|
||
|
|
||
|
pDecoderInterface = p->pVideoEngine->pDecoderInterface;
|
||
|
ret = 0;
|
||
|
if(pDecoderInterface->SetSpecialData != NULL)
|
||
|
{
|
||
|
ret = pDecoderInterface->SetSpecialData(pDecoderInterface, pArg);
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static int decoderSendToDisplayPictureNum(VideoDecoder* pDecoder, int nStreamIndex)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
Fbm* pFbm;
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
pFbm = p->pFbm[nStreamIndex];
|
||
|
if(pFbm == NULL)
|
||
|
{
|
||
|
if(p->pVideoEngine != NULL)
|
||
|
pFbm = p->pFbm[nStreamIndex] = VideoEngineGetFbm(p->pVideoEngine, nStreamIndex);
|
||
|
logi("Fbm module of video stream %d not create yet, ValidPictureNum() fail.", nStreamIndex);
|
||
|
return 0;
|
||
|
}
|
||
|
return FbmGetDisplayBufferNum(pFbm);
|
||
|
}
|
||
|
|
||
|
|
||
|
#define DEBUG_VDECODER_SWAP(a,b,tmp) {(tmp)=(a);(a)=(b);(b)=(tmp);}
|
||
|
static void DebugUpDateCostTimeList(float *list, float time)
|
||
|
{
|
||
|
int i;
|
||
|
float tmp;
|
||
|
if(list == NULL)
|
||
|
return;
|
||
|
if(time <= list[0])
|
||
|
return;
|
||
|
list[0] = time;
|
||
|
for(i = 0; i < DEBUG_MAX_FRAME_IN_LIST - 1; i++)
|
||
|
{
|
||
|
if(list[i+1] < time)
|
||
|
DEBUG_VDECODER_SWAP(list[i+1], list[i], tmp)
|
||
|
else
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
static void DebugShowExtraTime(float *list, int index)
|
||
|
{
|
||
|
int i, j;
|
||
|
float f[DEBUG_MAX_FRAME_IN_LIST];
|
||
|
j = index;
|
||
|
for(i = 0; i <= DEBUG_MAX_FRAME_IN_LIST - 1; i++)
|
||
|
{
|
||
|
f[i] = list[j];
|
||
|
j++;
|
||
|
if(j >= DEBUG_MAX_FRAME_IN_LIST - 1)
|
||
|
j = 0;
|
||
|
}
|
||
|
if(1)
|
||
|
{
|
||
|
float *l = f;
|
||
|
logd("time [1-8]: %.3f, %.3f, %.3f, %.3f, %.3f, %.3f, %.3f, %.3f",
|
||
|
l[0],l[1],l[2],l[3], l[4],l[5],l[6],l[7]);
|
||
|
logd(" [9-16]: %.3f, %.3f, %.3f, %.3f, %.3f, %.3f, %.3f, %.3f",
|
||
|
l[8],l[9],l[10],l[11], l[12],l[13],l[14],l[15]);
|
||
|
}
|
||
|
}
|
||
|
static void DebugShowValidPicList(int *list, int index)
|
||
|
{
|
||
|
int i, j;
|
||
|
int f[DEBUG_MAX_FRAME_IN_LIST];
|
||
|
j = index;
|
||
|
for(i = 0; i <= DEBUG_MAX_FRAME_IN_LIST - 1; i++)
|
||
|
{
|
||
|
f[i] = list[j];
|
||
|
j++;
|
||
|
if(j >= DEBUG_MAX_FRAME_IN_LIST - 1)
|
||
|
j = 0;
|
||
|
}
|
||
|
if(1)
|
||
|
{
|
||
|
int *l = f;
|
||
|
logd(" when decoding the last 16 frame, waiting for display number ");
|
||
|
logd("valid frame [1-8]: %d, %d, %d, %d, %d, %d, %d, %d",
|
||
|
l[0],l[1],l[2],l[3], l[4],l[5],l[6],l[7]);
|
||
|
logd(" [9-16]: %d, %d, %d, %d, %d, %d, %d, %d",
|
||
|
l[8],l[9],l[10],l[11], l[12],l[13],l[14],l[15]);
|
||
|
}
|
||
|
}
|
||
|
static void DebugDecoderSpeedInfo(VideoDecoder* pDecoder)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
int nDeltaFrame = 64;
|
||
|
int64_t hwtime;
|
||
|
int64_t nCurrentFrameCostTime;
|
||
|
int nCostTime;
|
||
|
float fCostTime;
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
hwtime = VdecoderGetCurrentTime();
|
||
|
nCurrentFrameCostTime = (hwtime - p->debug.nHardwareCurrentTime);
|
||
|
nCostTime = nCurrentFrameCostTime;
|
||
|
fCostTime = nCostTime/1000.0;
|
||
|
#if DEBUG_SAVE_FRAME_TIME
|
||
|
if(p->debug.pFrameTime != NULL)
|
||
|
{
|
||
|
p->debug.pFrameTime[p->debug.nFrameTimeNum] = nCurrentFrameCostTime;
|
||
|
}
|
||
|
if(p->debug.pDispFrameNum != NULL)
|
||
|
{
|
||
|
int num = decoderSendToDisplayPictureNum(pDecoder, 0);
|
||
|
p->debug.pDispFrameNum[p->debug.nFrameTimeNum] = num;
|
||
|
}
|
||
|
p->debug.nFrameTimeNum++;
|
||
|
#endif //DEBUG_SAVE_FRAME_TIME
|
||
|
if(p->debug.nFrameNum > 8 && fCostTime > p->debug.nCostTimeList[0])
|
||
|
{
|
||
|
if(fCostTime > p->debug.nMaxCostTime)
|
||
|
p->debug.nMaxCostTime = fCostTime;
|
||
|
if(fCostTime >= 34.0)
|
||
|
p->debug.nThreshHoldNum++;
|
||
|
DebugUpDateCostTimeList(p->debug.nCostTimeList, fCostTime);
|
||
|
}
|
||
|
if(p->debug.nFrameNum > 8)
|
||
|
{
|
||
|
int64_t last = p->debug.nGetOneFrameFinishTime;
|
||
|
p->debug.nGetOneFrameFinishTime = VdecoderGetCurrentTime();
|
||
|
if(1)
|
||
|
{
|
||
|
int64_t nExtraTime = 0;
|
||
|
int nExtTimeInt;
|
||
|
float fExtTime;
|
||
|
nExtraTime = p->debug.nGetOneFrameFinishTime - last;
|
||
|
nExtTimeInt = nExtraTime;
|
||
|
fExtTime = nExtTimeInt/1000.0;
|
||
|
p->debug.nExtraDecodeTimeList[p->debug.nStartIndexExtra] = fExtTime - fCostTime;
|
||
|
p->debug.nFrameTimeList[p->debug.nStartIndexExtra] = fCostTime;
|
||
|
p->debug.nWaitingForDispPicNum[p->debug.nStartIndexExtra] = p->debug.nValidDisplayPic;
|
||
|
p->debug.nStartIndexExtra++;
|
||
|
if(p->debug.nStartIndexExtra >= DEBUG_MAX_FRAME_IN_LIST - 1)
|
||
|
p->debug.nStartIndexExtra = 0;
|
||
|
}
|
||
|
}
|
||
|
if(fCostTime < 1.0)
|
||
|
{
|
||
|
logd("warning, maybe drop frame enable, cost time: %.2f ms", fCostTime);
|
||
|
return;
|
||
|
}
|
||
|
p->debug.nHardwareDuration += nCurrentFrameCostTime;
|
||
|
p->debug.nFrameNumDuration += 1;
|
||
|
p->debug.nFrameNum += 1;
|
||
|
if(p->debug.nFrameNumDuration >= nDeltaFrame)
|
||
|
{
|
||
|
int64_t nTime, nTotalDiff, hwDiff, hwAvgTime;
|
||
|
nTime = VdecoderGetCurrentTime();
|
||
|
p->debug.nHardwareTotalTime += p->debug.nHardwareDuration;
|
||
|
p->debug.nDuration = nTime - p->debug.nCurrentTime;
|
||
|
if(1)
|
||
|
{
|
||
|
float PlayerCurrFps, PlayerAverageFps, HardwareCurrFps, HardwareAvgFps;
|
||
|
nTotalDiff = (nTime - p->debug.nStartTime) / 1000; /* ms */
|
||
|
nTime = p->debug.nDuration / 1000; /* ms */
|
||
|
hwDiff = p->debug.nHardwareDuration / 1000; /* ms */
|
||
|
hwAvgTime = p->debug.nHardwareTotalTime / 1000;
|
||
|
PlayerCurrFps = (float)(p->debug.nFrameNumDuration * 1000.0) / nTime;
|
||
|
PlayerAverageFps = (float)(p->debug.nFrameNum * 1000.0) / nTotalDiff;
|
||
|
HardwareCurrFps = (float)(p->debug.nFrameNumDuration * 1000.0) / hwDiff;
|
||
|
HardwareAvgFps = (float)(p->debug.nFrameNum * 1000.0) / hwAvgTime;
|
||
|
|
||
|
if(HardwareCurrFps < p->debug.nMin_hardDecodeFps || p->debug.nMin_hardDecodeFps == 0)
|
||
|
p->debug.nMin_hardDecodeFps = HardwareCurrFps;
|
||
|
|
||
|
if(HardwareCurrFps > p->debug.nMax_hardDecodeFps || p->debug.nMax_hardDecodeFps == 0)
|
||
|
p->debug.nMax_hardDecodeFps = HardwareCurrFps;
|
||
|
|
||
|
logd("hardware speed: avr = %.2f fps, max = %.2f fps, min = %.2f fps",
|
||
|
HardwareAvgFps, p->debug.nMax_hardDecodeFps, p->debug.nMin_hardDecodeFps);
|
||
|
|
||
|
logd("player speed: %.2ffps, \
|
||
|
average: %.2ffps;slowest frame cost time: %.2f ms; \
|
||
|
hardware speed: %.2ffps, average: %.2ffps", \
|
||
|
PlayerCurrFps, PlayerAverageFps, p->debug.nMaxCostTime,\
|
||
|
HardwareCurrFps, HardwareAvgFps);
|
||
|
logv(" 0527 vdecoder hardware speed: %.2f fps, \
|
||
|
slowest frame cost time: %.2f ms, average speed: %.2f fps", \
|
||
|
HardwareCurrFps, p->debug.nMaxCostTime, HardwareAvgFps);
|
||
|
logv(" 0527 vdecoder player current speed: %.2f fps; player average speed: %.2f fps",
|
||
|
PlayerCurrFps, PlayerAverageFps);
|
||
|
}
|
||
|
if(0)
|
||
|
{
|
||
|
float *l = p->debug.nCostTimeList;
|
||
|
logd("cost most time frame list[1-8]: \
|
||
|
%.2f, %.2f, %.2f, %.2f, %.2f, %.2f, %.2f, %.2f", \
|
||
|
l[0],l[1],l[2],l[3], l[4],l[5],l[6],l[7]);
|
||
|
logd("cost most time frame list[9-16]: \
|
||
|
%.2f, %.2f, %.2f, %.2f, %.2f, %.2f, %.2f, %.2f", \
|
||
|
l[8],l[9],l[10],l[11], l[12],l[13],l[14],l[15]);
|
||
|
logd("decoded frame number: %d, cost time >= 34ms frame number: %d", \
|
||
|
p->debug.nFrameNum, p->debug.nThreshHoldNum);
|
||
|
}
|
||
|
if(0)
|
||
|
{
|
||
|
logd("current frame cost time: %.3fms last 16 frame extra time: ", fCostTime);
|
||
|
DebugShowExtraTime(p->debug.nExtraDecodeTimeList, p->debug.nStartIndexExtra);
|
||
|
logd(" last 16 frame decode time: , smooth time: %.3f ", p->debug.nFrameSmoothTime);
|
||
|
p->debug.nFrameSmoothTime = 0;
|
||
|
DebugShowExtraTime(p->debug.nFrameTimeList, p->debug.nStartIndexExtra);
|
||
|
DebugShowValidPicList(p->debug.nWaitingForDispPicNum, p->debug.nStartIndexExtra);
|
||
|
logd(" ");
|
||
|
}
|
||
|
p->debug.nDuration = 0;
|
||
|
p->debug.nHardwareDuration = 0;
|
||
|
p->debug.nFrameNumDuration = 0;
|
||
|
p->debug.nCurrentTime = VdecoderGetCurrentTime();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int DecodeVideoStream(VideoDecoder* pDecoder,
|
||
|
int bEndOfStream,
|
||
|
int bDecodeKeyFrameOnly,
|
||
|
int bDropBFrameIfDelay,
|
||
|
int64_t nCurrentTimeUs)
|
||
|
{
|
||
|
int ret;
|
||
|
VideoDecoderContext* p;
|
||
|
|
||
|
logi("DecodeVideoStream, pDecoder=%p, \
|
||
|
bEndOfStream=%d, bDropBFrameIfDelay=%d, nCurrentTimeUs=%lld", \
|
||
|
pDecoder, bEndOfStream, bDropBFrameIfDelay, nCurrentTimeUs);
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
//eVersion = VeGetIcVersion();
|
||
|
|
||
|
if(p->pVideoEngine == NULL)
|
||
|
return VDECODE_RESULT_UNSUPPORTED;
|
||
|
|
||
|
#if AW_VDECODER_SPEED_INFO
|
||
|
p->debug.nHardwareCurrentTime = VdecoderGetCurrentTime();
|
||
|
if(p->debug.nCurrentTime == 0)
|
||
|
p->debug.nCurrentTime = p->debug.nHardwareCurrentTime;
|
||
|
if(p->debug.nStartTime == 0)
|
||
|
p->debug.nStartTime = p->debug.nHardwareCurrentTime;
|
||
|
p->debug.nValidDisplayPic = ValidPictureNum(pDecoder, 0);
|
||
|
#endif //AW_VDECODER_SPEED_INFO
|
||
|
|
||
|
if(bEndOfStream == 1)
|
||
|
{
|
||
|
if(p->pSbm[0])
|
||
|
SbmSetEos(p->pSbm[0], bEndOfStream);
|
||
|
if(p->pSbm[1])
|
||
|
SbmSetEos(p->pSbm[1], bEndOfStream);
|
||
|
}
|
||
|
|
||
|
if(p->videoStreamInfo.bSecureStreamFlagLevel1)
|
||
|
CdcMemSetup(p->memops);
|
||
|
|
||
|
ret = VideoEngineDecode(p->pVideoEngine,
|
||
|
bEndOfStream,
|
||
|
bDecodeKeyFrameOnly,
|
||
|
bDropBFrameIfDelay,
|
||
|
nCurrentTimeUs);
|
||
|
|
||
|
if(p->videoStreamInfo.bSecureStreamFlagLevel1)
|
||
|
CdcMemShutdown(p->memops);
|
||
|
logv("decode method return %s", strDecodeResult[ret-VDECODE_RESULT_MIN]);
|
||
|
|
||
|
#if AW_VDECODER_SPEED_INFO
|
||
|
if(ret == VDECODE_RESULT_FRAME_DECODED ||
|
||
|
ret == VDECODE_RESULT_KEYFRAME_DECODED)
|
||
|
{
|
||
|
DebugDecoderSpeedInfo(pDecoder);
|
||
|
}
|
||
|
#endif //AW_VDECODER_SPEED_INFO
|
||
|
|
||
|
//* if in thumbnail mode and picture decoded,
|
||
|
//* reset the video engine to flush the picture out to FBM.
|
||
|
if(p->vconfig.bThumbnailMode)
|
||
|
{
|
||
|
if(ret == VDECODE_RESULT_FRAME_DECODED || ret == VDECODE_RESULT_KEYFRAME_DECODED)
|
||
|
{
|
||
|
//VideoEngineReset(p->pVideoEngine);
|
||
|
}
|
||
|
}
|
||
|
else if(bEndOfStream && ret == VDECODE_RESULT_NO_BITSTREAM)
|
||
|
{
|
||
|
VideoEngineReset(p->pVideoEngine);
|
||
|
}
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
int GetVideoStreamInfo(VideoDecoder* pDecoder, VideoStreamInfo* pVideoInfo)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
|
||
|
logv("GetVideoStreamInfo, pDecoder=%p, pVideoInfo=%p", pDecoder, pVideoInfo);
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
//* set the video stream information.
|
||
|
pVideoInfo->eCodecFormat = p->videoStreamInfo.eCodecFormat;
|
||
|
pVideoInfo->nWidth = p->videoStreamInfo.nWidth;
|
||
|
pVideoInfo->nHeight = p->videoStreamInfo.nHeight;
|
||
|
pVideoInfo->nFrameRate = p->videoStreamInfo.nFrameRate;
|
||
|
pVideoInfo->nFrameDuration = p->videoStreamInfo.nFrameDuration;
|
||
|
pVideoInfo->nAspectRatio = p->videoStreamInfo.nAspectRatio;
|
||
|
pVideoInfo->bIs3DStream = p->videoStreamInfo.bIs3DStream;
|
||
|
|
||
|
//* print video stream information.
|
||
|
{
|
||
|
logi("Video Stream Information:");
|
||
|
logi(" codec = %s", strCodecFormat[pVideoInfo->eCodecFormat - VIDEO_CODEC_FORMAT_MIN]);
|
||
|
logi(" width = %d pixels", pVideoInfo->nWidth);
|
||
|
logi(" height = %d pixels", pVideoInfo->nHeight);
|
||
|
logi(" frame rate = %d", pVideoInfo->nFrameRate);
|
||
|
logi(" frame duration = %d us", pVideoInfo->nFrameDuration);
|
||
|
logi(" aspect ratio = %d", pVideoInfo->nAspectRatio);
|
||
|
logi(" is 3D stream = %s", pVideoInfo->bIs3DStream ? "yes" : "no");
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int RequestVideoStreamBuffer(VideoDecoder* pDecoder,
|
||
|
int nRequireSize,
|
||
|
char** ppBuf,
|
||
|
int* pBufSize,
|
||
|
char** ppRingBuf,
|
||
|
int* pRingBufSize,
|
||
|
int nStreamBufIndex)
|
||
|
{
|
||
|
char* pStart;
|
||
|
char* pStreamBufEnd;
|
||
|
char* pMem;
|
||
|
int nFreeSize;
|
||
|
SbmInterface* pSbm;
|
||
|
VideoDecoderContext* p;
|
||
|
|
||
|
logi("RequestVideoStreamBuffer, pDecoder=%p, nRequireSize=%d, nStreamBufIndex=%d",
|
||
|
pDecoder, nRequireSize, nStreamBufIndex);
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
*ppBuf = NULL;
|
||
|
*ppRingBuf = NULL;
|
||
|
*pBufSize = 0;
|
||
|
*pRingBufSize = 0;
|
||
|
|
||
|
pSbm = p->pSbm[nStreamBufIndex];
|
||
|
|
||
|
if(pSbm == NULL)
|
||
|
{
|
||
|
logw("pSbm of video stream %d is NULL, RequestVideoStreamBuffer fail.", nStreamBufIndex);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
//* sometimes AVI parser will pass empty stream frame to help pts calculation.
|
||
|
//* in this case give four bytes even the parser does not need.
|
||
|
if(nRequireSize == 0)
|
||
|
nRequireSize = 4;
|
||
|
|
||
|
//* we've filled partial frame data but not added to the SBM before,
|
||
|
//* we need to calculate the actual buffer pointer by self.
|
||
|
nRequireSize += p->partialStreamDataInfo[nStreamBufIndex].nLength;
|
||
|
if(SbmRequestBuffer(pSbm, nRequireSize, &pMem, &nFreeSize) < 0)
|
||
|
{
|
||
|
logi("request stream buffer fail, \
|
||
|
%d bytes valid data in SBM[%d], total buffer size is %d bytes.",
|
||
|
SbmStreamDataSize(pSbm),
|
||
|
nStreamBufIndex,
|
||
|
SbmBufferSize(pSbm));
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
//* check the free buffer is larger than the partial data we filled before.
|
||
|
if(nFreeSize <= p->partialStreamDataInfo[nStreamBufIndex].nLength)
|
||
|
{
|
||
|
logi("require stream buffer get %d bytes, \
|
||
|
but this buffer has been filled with partial \
|
||
|
frame data of %d bytes before, nStreamBufIndex=%d.",
|
||
|
nFreeSize, p->partialStreamDataInfo[nStreamBufIndex].nLength, nStreamBufIndex);
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
//* calculate the output buffer pos.
|
||
|
pStreamBufEnd = (char*)SbmBufferAddress(pSbm) + SbmBufferSize(pSbm);
|
||
|
pStart = pMem + p->partialStreamDataInfo[nStreamBufIndex].nLength;
|
||
|
if(pStart >= pStreamBufEnd)
|
||
|
pStart -= SbmBufferSize(pSbm);
|
||
|
nFreeSize -= p->partialStreamDataInfo[nStreamBufIndex].nLength;
|
||
|
|
||
|
if(pStart + nFreeSize <= pStreamBufEnd) //* check if buffer ring back.
|
||
|
{
|
||
|
*ppBuf = pStart;
|
||
|
*pBufSize = nFreeSize;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//* the buffer ring back.
|
||
|
*ppBuf = pStart;
|
||
|
*pBufSize = pStreamBufEnd - pStart;
|
||
|
*ppRingBuf = SbmBufferAddress(pSbm);
|
||
|
*pRingBufSize = nFreeSize - *pBufSize;
|
||
|
logi("stream buffer %d ring back.", nStreamBufIndex);
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void FlushSbmData(VideoDecoderContext* p, SbmInterface*pSbm,
|
||
|
VideoStreamDataInfo* pPartialStreamDataInfo)
|
||
|
{
|
||
|
if(p->vconfig.bVirMallocSbm == 0)
|
||
|
{
|
||
|
//* we need to flush data from cache to memory for the VE hardware module.
|
||
|
char* pStreamBufEnd = (char*)SbmBufferAddress(pSbm) + SbmBufferSize(pSbm);
|
||
|
if(pPartialStreamDataInfo->pData + pPartialStreamDataInfo->nLength <= pStreamBufEnd)
|
||
|
{
|
||
|
CdcMemFlushCache(p->memops, \
|
||
|
pPartialStreamDataInfo->pData, \
|
||
|
pPartialStreamDataInfo->nLength);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//* buffer ring back.
|
||
|
int nPartialLen = pStreamBufEnd - pPartialStreamDataInfo->pData;
|
||
|
CdcMemFlushCache(p->memops, pPartialStreamDataInfo->pData, nPartialLen);
|
||
|
CdcMemFlushCache(p->memops, \
|
||
|
SbmBufferAddress(pSbm), \
|
||
|
pPartialStreamDataInfo->nLength - nPartialLen);
|
||
|
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
int SubmitVideoStreamData(VideoDecoder* pDecoder,
|
||
|
VideoStreamDataInfo* pDataInfo,
|
||
|
int nStreamBufIndex)
|
||
|
{
|
||
|
SbmInterface* pSbm;
|
||
|
VideoDecoderContext* p;
|
||
|
VideoStreamDataInfo* pPartialStreamDataInfo;
|
||
|
struct ScMemOpsS *_memops;
|
||
|
char mask[6] = "awsp";
|
||
|
|
||
|
logi("SubmitVideoStreamData, pDecoder=%p, pDataInfo=%p, nStreamBufIndex=%d",
|
||
|
pDecoder, pDataInfo, nStreamBufIndex);
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
_memops = p->memops;
|
||
|
|
||
|
pSbm = p->pSbm[nStreamBufIndex];
|
||
|
|
||
|
if(pSbm == NULL)
|
||
|
{
|
||
|
logw("pSbm of video stream %d is NULL, SubmitVideoStreamData fail.", nStreamBufIndex);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
pPartialStreamDataInfo = &p->partialStreamDataInfo[nStreamBufIndex];
|
||
|
|
||
|
//* chech wheter a new stream frame.
|
||
|
if(pDataInfo->bIsFirstPart)
|
||
|
{
|
||
|
if(pPartialStreamDataInfo->nLength != 0) //* last frame is not complete yet.
|
||
|
{
|
||
|
logv("stream data frame uncomplete.");
|
||
|
#if DEBUG_SAVE_BITSTREAM
|
||
|
logd("lenth=%d\n", pPartialStreamDataInfo->nLength);
|
||
|
logd("pts=%lld\n", pPartialStreamDataInfo->nPts);
|
||
|
char* pWriteAddr = SbmBufferWritePointer(pSbm);
|
||
|
char* pSbmBufferStartAddr = SbmBufferAddress(pSbm);
|
||
|
int nSbmBufferSize = SbmBufferSize(pSbm);
|
||
|
char* pSbmBufferEndAddr = pSbmBufferStartAddr + nSbmBufferSize - 1;
|
||
|
logv("%p, %p, %p, %d",pWriteAddr, pSbmBufferStartAddr,
|
||
|
pSbmBufferEndAddr, nSbmBufferSize);
|
||
|
if(pWriteAddr+pPartialStreamDataInfo->nLength <= pSbmBufferEndAddr)
|
||
|
{
|
||
|
fwrite(pWriteAddr, 1, pPartialStreamDataInfo->nLength, fpStream);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
fwrite(pWriteAddr, 1, pSbmBufferEndAddr - pWriteAddr+1, fpStream);
|
||
|
fwrite(pSbmBufferStartAddr, 1, \
|
||
|
pPartialStreamDataInfo->nLength-(pSbmBufferEndAddr-pWriteAddr+1), \
|
||
|
fpStream);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#if DEBUG_MAKE_SPECIAL_STREAM
|
||
|
if(1)
|
||
|
{
|
||
|
FILE *fp = fopen(SPECIAL_STREAM_FILE, "ab");
|
||
|
if(fp == NULL)
|
||
|
{
|
||
|
loge(" make special stream open file error, save stream ");
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
logd(" make special stream save data size: %d ",
|
||
|
pPartialStreamDataInfo->nLength);
|
||
|
char* pWriteAddr = SbmBufferWritePointer(pSbm);
|
||
|
char* pSbmBufferStartAddr = SbmBufferAddress(pSbm);
|
||
|
int nSbmBufferSize = SbmBufferSize(pSbm);
|
||
|
char* pSbmBufferEndAddr = pSbmBufferStartAddr + nSbmBufferSize - 1;
|
||
|
|
||
|
fwrite(mask, 4, sizeof(char), fp);
|
||
|
fwrite(&nStreamBufIndex, 1, sizeof(int), fp);
|
||
|
fwrite(&pPartialStreamDataInfo->nLength, 1, sizeof(int), fp);
|
||
|
fwrite(&pPartialStreamDataInfo->nPts, 1, sizeof(int64_t), fp);
|
||
|
if(pWriteAddr+pPartialStreamDataInfo->nLength <= pSbmBufferEndAddr)
|
||
|
{
|
||
|
fwrite(pWriteAddr, 1, pPartialStreamDataInfo->nLength, fp);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
fwrite(pWriteAddr, 1, pSbmBufferEndAddr-pWriteAddr+1, fp);
|
||
|
fwrite(pSbmBufferStartAddr, 1, \
|
||
|
pPartialStreamDataInfo->nLength- \
|
||
|
(pSbmBufferEndAddr - pWriteAddr + 1), \
|
||
|
fp);
|
||
|
}
|
||
|
fclose(fp);
|
||
|
}
|
||
|
}
|
||
|
#endif //DEBUG_MAKE_SPECIAL_STREAM
|
||
|
|
||
|
FlushSbmData(p, pSbm, pPartialStreamDataInfo);
|
||
|
SbmAddStream(pSbm, pPartialStreamDataInfo);
|
||
|
}
|
||
|
|
||
|
//* set the data address and pts.
|
||
|
pPartialStreamDataInfo->pData = pDataInfo->pData;
|
||
|
pPartialStreamDataInfo->nLength = pDataInfo->nLength;
|
||
|
pPartialStreamDataInfo->nPts = pDataInfo->nPts;
|
||
|
pPartialStreamDataInfo->nPcr = pDataInfo->nPcr;
|
||
|
pPartialStreamDataInfo->bIsFirstPart = pDataInfo->bIsFirstPart;
|
||
|
pPartialStreamDataInfo->bIsLastPart = 0;
|
||
|
pPartialStreamDataInfo->bVideoInfoFlag = pDataInfo->bVideoInfoFlag;
|
||
|
pPartialStreamDataInfo->pVideoInfo = pDataInfo->pVideoInfo;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
pPartialStreamDataInfo->nLength += pDataInfo->nLength;
|
||
|
if(pPartialStreamDataInfo->nPts == -1 && pDataInfo->nPts != -1)
|
||
|
pPartialStreamDataInfo->nPts = pDataInfo->nPts;
|
||
|
}
|
||
|
|
||
|
//* check whether a stream frame complete.
|
||
|
if(pDataInfo->bIsLastPart)
|
||
|
{
|
||
|
if(pPartialStreamDataInfo->pData != NULL && pPartialStreamDataInfo->nLength != 0)
|
||
|
{
|
||
|
FlushSbmData(p, pSbm, pPartialStreamDataInfo);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//* maybe it is a empty frame for MPEG4 decoder from AVI parser.
|
||
|
logw("empty stream data frame submitted, pData=%p, nLength=%d",
|
||
|
pPartialStreamDataInfo->pData, pPartialStreamDataInfo->nLength);
|
||
|
}
|
||
|
|
||
|
//* submit stream frame to the SBM.
|
||
|
#if DEBUG_SAVE_BITSTREAM
|
||
|
char* pWriteAddr = SbmBufferWritePointer(pSbm);
|
||
|
char* pSbmBufferStartAddr = SbmBufferAddress(pSbm);
|
||
|
int nSbmBufferSize = SbmBufferSize(pSbm);
|
||
|
char* pSbmBufferEndAddr = pSbmBufferStartAddr + nSbmBufferSize - 1;
|
||
|
logv("%p, %p, %p, %d",pWriteAddr, pSbmBufferStartAddr,
|
||
|
pSbmBufferEndAddr, nSbmBufferSize);
|
||
|
if(pWriteAddr+pPartialStreamDataInfo->nLength <= pSbmBufferEndAddr)
|
||
|
{
|
||
|
fwrite(pWriteAddr, 1, pPartialStreamDataInfo->nLength, fpStream);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
fwrite(pWriteAddr, 1, pSbmBufferEndAddr-pWriteAddr+1, fpStream);
|
||
|
fwrite(pSbmBufferStartAddr, 1, \
|
||
|
pPartialStreamDataInfo->nLength- \
|
||
|
(pSbmBufferEndAddr-pWriteAddr+1), fpStream);
|
||
|
}
|
||
|
logd("lenth=%d\n", pPartialStreamDataInfo->nLength);
|
||
|
logd("pts=%lld\n", pPartialStreamDataInfo->nPts);
|
||
|
#endif
|
||
|
|
||
|
#if DEBUG_MAKE_SPECIAL_STREAM
|
||
|
if(1)
|
||
|
{
|
||
|
FILE *fp = fopen(SPECIAL_STREAM_FILE, "ab");
|
||
|
if(fp == NULL)
|
||
|
{
|
||
|
loge(" make special stream open file error, save stream ");
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
logd(" make special stream save data size: %d ",\
|
||
|
pPartialStreamDataInfo->nLength);
|
||
|
char* pWriteAddr = SbmBufferWritePointer(pSbm);
|
||
|
char* pSbmBufferStartAddr = SbmBufferAddress(pSbm);
|
||
|
int nSbmBufferSize = SbmBufferSize(pSbm);
|
||
|
char* pSbmBufferEndAddr = pSbmBufferStartAddr + nSbmBufferSize - 1;
|
||
|
|
||
|
fwrite(mask, 4, sizeof(char), fp);
|
||
|
fwrite(&nStreamBufIndex, 1, sizeof(int), fp);
|
||
|
fwrite(&pPartialStreamDataInfo->nLength, 1, sizeof(int), fp);
|
||
|
fwrite(&pPartialStreamDataInfo->nPts, 1, sizeof(int64_t), fp);
|
||
|
if(pWriteAddr+pPartialStreamDataInfo->nLength <= pSbmBufferEndAddr)
|
||
|
{
|
||
|
fwrite(pWriteAddr, 1, pPartialStreamDataInfo->nLength, fp);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
fwrite(pWriteAddr, 1, pSbmBufferEndAddr - pWriteAddr+1, fp);
|
||
|
fwrite(pSbmBufferStartAddr, 1, \
|
||
|
pPartialStreamDataInfo->nLength-\
|
||
|
(pSbmBufferEndAddr - pWriteAddr + 1), fp);
|
||
|
}
|
||
|
fclose(fp);
|
||
|
}
|
||
|
}
|
||
|
#endif// DEBUG_MAKE_SPECIAL_STREAM
|
||
|
|
||
|
SbmAddStream(pSbm, pPartialStreamDataInfo);
|
||
|
|
||
|
//* clear status of stream data info.
|
||
|
pPartialStreamDataInfo->pData = NULL;
|
||
|
pPartialStreamDataInfo->nLength = 0;
|
||
|
pPartialStreamDataInfo->nPts = -1;
|
||
|
pPartialStreamDataInfo->nPcr = -1;
|
||
|
pPartialStreamDataInfo->bIsLastPart = 0;
|
||
|
pPartialStreamDataInfo->bIsFirstPart = 0;
|
||
|
pPartialStreamDataInfo->bVideoInfoFlag = 0;
|
||
|
pPartialStreamDataInfo->pVideoInfo = NULL;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int VideoStreamBufferSize(VideoDecoder* pDecoder, int nStreamBufIndex)
|
||
|
{
|
||
|
VideoDecoderContext* p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
logi("VideoStreamBufferSize, nStreamBufIndex=%d", nStreamBufIndex);
|
||
|
|
||
|
if(p->pSbm[nStreamBufIndex] == NULL)
|
||
|
return 0;
|
||
|
|
||
|
return SbmBufferSize(p->pSbm[nStreamBufIndex]);
|
||
|
}
|
||
|
|
||
|
int VideoStreamDataSize(VideoDecoder* pDecoder, int nStreamBufIndex)
|
||
|
{
|
||
|
VideoDecoderContext* p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
logi("VideoStreamDataSize, nStreamBufIndex=%d", nStreamBufIndex);
|
||
|
|
||
|
if(p->pSbm[nStreamBufIndex] == NULL)
|
||
|
return 0;
|
||
|
|
||
|
return SbmStreamDataSize(p->pSbm[nStreamBufIndex]);
|
||
|
}
|
||
|
|
||
|
int VideoStreamFrameNum(VideoDecoder* pDecoder, int nStreamBufIndex)
|
||
|
{
|
||
|
VideoDecoderContext* p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
logi("VideoStreamFrameNum, nStreamBufIndex=%d", nStreamBufIndex);
|
||
|
|
||
|
if(p->pSbm[nStreamBufIndex] == NULL)
|
||
|
return 0;
|
||
|
|
||
|
return SbmStreamFrameNum(p->pSbm[nStreamBufIndex]);
|
||
|
}
|
||
|
|
||
|
void* VideoStreamDataInfoPointer(VideoDecoder* pDecoder, int nStreamBufIndex)
|
||
|
{
|
||
|
VideoDecoderContext* p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
logi("VideoStreamDataSize, nStreamBufIndex=%d", nStreamBufIndex);
|
||
|
|
||
|
if(p->pSbm[nStreamBufIndex] == NULL)
|
||
|
return NULL;
|
||
|
|
||
|
return SbmBufferDataInfo(p->pSbm[nStreamBufIndex]);
|
||
|
}
|
||
|
|
||
|
static void savePicture(VideoDecoderContext* p, VideoPicture* pPicture)
|
||
|
{
|
||
|
p->nSavePicCount++;
|
||
|
if(p->nSavePicCount <= 10)
|
||
|
{
|
||
|
|
||
|
if(pPicture->pData0 == NULL)
|
||
|
{
|
||
|
logw("save picture failed: pPicture->pData0 is null");
|
||
|
return ;
|
||
|
}
|
||
|
|
||
|
typedef struct PixelFormatMapping
|
||
|
{
|
||
|
unsigned char name[10];
|
||
|
int nIndex ;
|
||
|
} PixelFormatMapping;
|
||
|
int i=0;
|
||
|
char name[128];
|
||
|
FILE *fp;
|
||
|
int nMappSize;
|
||
|
|
||
|
PixelFormatMapping mPixelFormatMapping[] =
|
||
|
{{"non", 0},
|
||
|
{"yuv420", 1},
|
||
|
{"yuv422", 2},
|
||
|
{"yuv444", 3},
|
||
|
{"yv12", 4},
|
||
|
{"nv21", 5},
|
||
|
{"nv12", 6},
|
||
|
{"mb32", 7}
|
||
|
};
|
||
|
|
||
|
nMappSize = sizeof(mPixelFormatMapping)/sizeof(PixelFormatMapping);
|
||
|
for(; i < nMappSize; i++)
|
||
|
{
|
||
|
if(mPixelFormatMapping[i].nIndex == pPicture->ePixelFormat)
|
||
|
break;
|
||
|
}
|
||
|
if(i >= nMappSize)
|
||
|
i = 0;
|
||
|
|
||
|
sprintf(name, "/mnt/UDISK/pic_%s_%dx%d_%d.dat",mPixelFormatMapping[i].name,
|
||
|
pPicture->nWidth, pPicture->nHeight,p->nSavePicCount);
|
||
|
fp = fopen(name, "ab");
|
||
|
if(fp != NULL)
|
||
|
{
|
||
|
logd("saving picture, size: %d x %d, format: %d, count: %d",
|
||
|
pPicture->nWidth, pPicture->nHeight, pPicture->ePixelFormat,
|
||
|
p->nSavePicCount);
|
||
|
|
||
|
int nSaveLen;
|
||
|
if(pPicture->bEnableAfbcFlag)
|
||
|
nSaveLen = pPicture->nAfbcSize;
|
||
|
else
|
||
|
nSaveLen = pPicture->nWidth * pPicture->nHeight * 3/2;
|
||
|
|
||
|
CdcMemFlushCache(p->memops,pPicture->pData0, nSaveLen);
|
||
|
fwrite(pPicture->pData0, 1, nSaveLen, fp);
|
||
|
fclose(fp);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
loge("saving picture failed: open file error, frame number = %d", p->nSavePicCount);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
VideoPicture* RequestPicture(VideoDecoder* pDecoder, int nStreamIndex)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
Fbm* pFbm;
|
||
|
VideoPicture* pPicture;
|
||
|
|
||
|
logi("RequestPicture, nStreamIndex=%d", nStreamIndex);
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
pFbm = p->pFbm[nStreamIndex];
|
||
|
if(pFbm == NULL)
|
||
|
{
|
||
|
//* get FBM handle from video engine.
|
||
|
if(p->pVideoEngine != NULL)
|
||
|
pFbm = p->pFbm[nStreamIndex] = VideoEngineGetFbm(p->pVideoEngine, nStreamIndex);
|
||
|
|
||
|
if(pFbm == NULL)
|
||
|
{
|
||
|
logi("Fbm module of video stream %d not create yet, \
|
||
|
RequestPicture fail.", nStreamIndex);
|
||
|
return NULL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//* request picture.
|
||
|
pPicture = FbmRequestPicture(pFbm);
|
||
|
|
||
|
if(pPicture != NULL)
|
||
|
{
|
||
|
if(pPicture->colour_primaries == 9)
|
||
|
{
|
||
|
pPicture->matrix_coeffs = VIDEO_MATRIX_COEFFS_BT2020;
|
||
|
}
|
||
|
|
||
|
#if(CHECK_METADATA_INFO)
|
||
|
printfMetaDataInfo((struct sunxi_metadata*)pPicture->pMetaData);
|
||
|
#endif
|
||
|
|
||
|
logv("**** color_primaries = %d, matrix_coeffs = %d",
|
||
|
pPicture->colour_primaries,pPicture->matrix_coeffs);
|
||
|
//* set stream index to the picture, it will be useful when picture returned.
|
||
|
pPicture->nStreamIndex = nStreamIndex;
|
||
|
|
||
|
//* update video stream information.
|
||
|
UpdateVideoStreamInfo(p, pPicture);
|
||
|
logv("picture w: %d, h: %d, addr: %x, top: %d, \
|
||
|
bottom: %d, left: %d, right: %d, stride: %d", \
|
||
|
pPicture->nWidth, pPicture->nHeight, (size_addr)pPicture,
|
||
|
pPicture->nTopOffset, pPicture->nBottomOffset,
|
||
|
pPicture->nLeftOffset, pPicture->nRightOffset, pPicture->nLineStride);
|
||
|
|
||
|
#if DEBUG_SAVE_PICTURE
|
||
|
savePicture(p, pPicture);
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
return pPicture;
|
||
|
}
|
||
|
|
||
|
int ReturnPicture(VideoDecoder* pDecoder, VideoPicture* pPicture)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
Fbm* pFbm;
|
||
|
int nStreamIndex;
|
||
|
int ret;
|
||
|
|
||
|
logi("ReturnPicture, pPicture=%p", pPicture);
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
if(pPicture == NULL)
|
||
|
return 0;
|
||
|
nStreamIndex = pPicture->nStreamIndex;
|
||
|
if(nStreamIndex < 0 || nStreamIndex > 2)
|
||
|
{
|
||
|
loge("invalid stream index(%d), pPicture->nStreamIndex must had been \
|
||
|
changed by someone incorrectly, ReturnPicture fail.", nStreamIndex);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
pFbm = p->pFbm[nStreamIndex];
|
||
|
if(pFbm == NULL)
|
||
|
{
|
||
|
logw("pFbm is NULL when returning a picture, ReturnPicture fail.");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
ret = FbmReturnPicture(pFbm, pPicture);
|
||
|
if(ret != 0)
|
||
|
{
|
||
|
logw("FbmReturnPicture return fail,\
|
||
|
it means the picture being returned it not one of this FBM.");
|
||
|
}
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
VideoPicture* NextPictureInfo(VideoDecoder* pDecoder, int nStreamIndex)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
Fbm* pFbm;
|
||
|
VideoPicture* pPicture;
|
||
|
|
||
|
logi("RequestPicture, nStreamIndex=%d", nStreamIndex);
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
pFbm = p->pFbm[nStreamIndex];
|
||
|
if(pFbm == NULL)
|
||
|
{
|
||
|
//* get FBM handle from video engine.
|
||
|
if(p->pVideoEngine != NULL)
|
||
|
pFbm = p->pFbm[nStreamIndex] = VideoEngineGetFbm(p->pVideoEngine, nStreamIndex);
|
||
|
|
||
|
logi("Fbm module of video stream %d not create yet, NextPictureInfo() fail.", nStreamIndex);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
//* request picture.
|
||
|
pPicture = FbmNextPictureInfo(pFbm);
|
||
|
|
||
|
if(pPicture != NULL)
|
||
|
{
|
||
|
//* set stream index to the picture, it will be useful when picture returned.
|
||
|
pPicture->nStreamIndex = nStreamIndex;
|
||
|
|
||
|
//* update video stream information.
|
||
|
UpdateVideoStreamInfo(p, pPicture);
|
||
|
}
|
||
|
|
||
|
return pPicture;
|
||
|
}
|
||
|
|
||
|
int TotalPictureBufferNum(VideoDecoder* pDecoder, int nStreamIndex)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
Fbm* pFbm;
|
||
|
|
||
|
logv("TotalPictureBufferNum, nStreamIndex=%d", nStreamIndex);
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
pFbm = p->pFbm[nStreamIndex];
|
||
|
if(pFbm == NULL)
|
||
|
{
|
||
|
//* get FBM handle from video engine.
|
||
|
if(p->pVideoEngine != NULL)
|
||
|
pFbm = p->pFbm[nStreamIndex] = VideoEngineGetFbm(p->pVideoEngine, nStreamIndex);
|
||
|
|
||
|
logi("Fbm module of video stream %d not create yet, \
|
||
|
TotalPictureBufferNum() fail.", nStreamIndex);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
return FbmTotalBufferNum(pFbm);
|
||
|
}
|
||
|
|
||
|
int EmptyPictureBufferNum(VideoDecoder* pDecoder, int nStreamIndex)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
Fbm* pFbm;
|
||
|
|
||
|
logv("EmptyPictureBufferNum, nStreamIndex=%d", nStreamIndex);
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
pFbm = p->pFbm[nStreamIndex];
|
||
|
if(pFbm == NULL)
|
||
|
{
|
||
|
//* get FBM handle from video engine.
|
||
|
if(p->pVideoEngine != NULL)
|
||
|
pFbm = p->pFbm[nStreamIndex] = VideoEngineGetFbm(p->pVideoEngine, nStreamIndex);
|
||
|
|
||
|
logi("Fbm module of video stream %d not create yet, \
|
||
|
EmptyPictureBufferNum() fail.", nStreamIndex);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
return FbmEmptyBufferNum(pFbm);
|
||
|
}
|
||
|
|
||
|
int ValidPictureNum(VideoDecoder* pDecoder, int nStreamIndex)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
Fbm* pFbm;
|
||
|
|
||
|
// logv("ValidPictureNum, nStreamIndex=%d", nStreamIndex);
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
pFbm = p->pFbm[nStreamIndex];
|
||
|
if(pFbm == NULL)
|
||
|
{
|
||
|
//* get FBM handle from video engine.
|
||
|
if(p->pVideoEngine != NULL)
|
||
|
pFbm = p->pFbm[nStreamIndex] = VideoEngineGetFbm(p->pVideoEngine, nStreamIndex);
|
||
|
|
||
|
logi("Fbm module of video stream %d not create yet, ValidPictureNum() fail.", nStreamIndex);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
return FbmValidPictureNum(pFbm);
|
||
|
}
|
||
|
|
||
|
//***********************************************//
|
||
|
//***********************************************//
|
||
|
//***added by xyliu at 2015-12-23****************//
|
||
|
/*if you want to scale the output video frame, but you donot know the real size of video,
|
||
|
cannot set the scale params (nHorizonScaleDownRatio,nVerticalScaleDownRatio,
|
||
|
bRotationEn), so you can set value to ExtraVidScaleInfo
|
||
|
*/
|
||
|
int ConfigExtraScaleInfo(VideoDecoder* pDecoder, int nWidthTh,
|
||
|
int nHeightTh, int nHorizonScaleRatio, int nVerticalScaleRatio)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
int ret;
|
||
|
DecoderInterface *pDecoderInterface;
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
logd("*************config ConfigExtraScaleInfo\n");
|
||
|
|
||
|
if(p == NULL ||p->pVideoEngine == NULL ||
|
||
|
p->pVideoEngine->pDecoderInterface == NULL)
|
||
|
{
|
||
|
loge(" set decoder extra scale info fail ");
|
||
|
return VDECODE_RESULT_UNSUPPORTED;
|
||
|
}
|
||
|
pDecoderInterface = p->pVideoEngine->pDecoderInterface;
|
||
|
ret = 0;
|
||
|
if(pDecoderInterface->SetExtraScaleInfo != NULL)
|
||
|
{
|
||
|
ret = pDecoderInterface->SetExtraScaleInfo(pDecoderInterface, \
|
||
|
nWidthTh, nHeightTh,nHorizonScaleRatio, nVerticalScaleRatio);
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
int ReopenVideoEngine(VideoDecoder* pDecoder, VConfig* pVConfig, VideoStreamInfo* pStreamInfo)
|
||
|
{
|
||
|
int i;
|
||
|
VideoDecoderContext* p;
|
||
|
int ret = 0;
|
||
|
|
||
|
logv("ReopenVideoEngine");
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
if(p->pVideoEngine == NULL)
|
||
|
{
|
||
|
logw("video decoder is not initialized yet, ReopenVideoEngine() fail.");
|
||
|
return -1;
|
||
|
}
|
||
|
//* create a video engine again,
|
||
|
//* before that, we should set vconfig and videoStreamInfo again.
|
||
|
if(p->videoStreamInfo.pCodecSpecificData != NULL)
|
||
|
{
|
||
|
free(p->videoStreamInfo.pCodecSpecificData);
|
||
|
p->videoStreamInfo.pCodecSpecificData = NULL;
|
||
|
}
|
||
|
|
||
|
ret = VideoEngineReopen(p->pVideoEngine,pVConfig, pStreamInfo);
|
||
|
if(ret < 0)
|
||
|
{
|
||
|
loge("VideoEngineReopen result is %d\n", ret);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
p->pFbm[0] = p->pFbm[1] = NULL;
|
||
|
p->nFbmNum = 0;
|
||
|
|
||
|
memcpy(&p->vconfig,pVConfig,sizeof(VConfig));
|
||
|
memcpy(&p->videoStreamInfo,pStreamInfo,sizeof(VideoStreamInfo));
|
||
|
pStreamInfo->bReOpenEngine = 1;
|
||
|
p->vconfig.pVeOpsSelf = p->pVideoEngine->pVeOpsSelf;
|
||
|
p->vconfig.veOpsS = p->pVideoEngine->veOpsS;
|
||
|
|
||
|
if(pStreamInfo->pCodecSpecificData)
|
||
|
{
|
||
|
p->videoStreamInfo.pCodecSpecificData = (char*)malloc(pStreamInfo->nCodecSpecificDataLen);
|
||
|
if(p->videoStreamInfo.pCodecSpecificData == NULL)
|
||
|
{
|
||
|
loge("malloc video codec specific data fail!");
|
||
|
return -1;
|
||
|
}
|
||
|
memcpy(p->videoStreamInfo.pCodecSpecificData,
|
||
|
pStreamInfo->pCodecSpecificData,
|
||
|
pStreamInfo->nCodecSpecificDataLen);
|
||
|
p->videoStreamInfo.nCodecSpecificDataLen = pStreamInfo->nCodecSpecificDataLen;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
p->videoStreamInfo.pCodecSpecificData = NULL;
|
||
|
p->videoStreamInfo.nCodecSpecificDataLen = 0;
|
||
|
}
|
||
|
|
||
|
//* set sbm module to the video engine.
|
||
|
for(i=0; i<p->nSbmNum; i++)
|
||
|
VideoEngineSetSbm(p->pVideoEngine, p->pSbm[i], i);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int RotatePictureHw(VideoDecoder* pDecoder,
|
||
|
VideoPicture* pPictureIn,
|
||
|
VideoPicture* pPictureOut,
|
||
|
int nRotateDegree)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
return VideoEngineRotatePicture(p->pVideoEngine, pPictureIn, pPictureOut, nRotateDegree);
|
||
|
}
|
||
|
|
||
|
int RotatePicture(struct ScMemOpsS* memOps,
|
||
|
VideoPicture* pPictureIn,
|
||
|
VideoPicture* pPictureOut,
|
||
|
int nRotateDegree,
|
||
|
int nGpuYAlign,
|
||
|
int nGpuCAlign)
|
||
|
{
|
||
|
VideoPicture* pOrgPicture = NULL;
|
||
|
VideoPicture* pPictureMid = NULL;
|
||
|
int bMallocBuffer = 0;
|
||
|
int nMemSizeY = 0;
|
||
|
int nMemSizeC = 0;
|
||
|
int ret = 0;
|
||
|
|
||
|
if(pPictureOut == NULL)
|
||
|
{
|
||
|
return -1;
|
||
|
}
|
||
|
GetBufferSize(pPictureIn->ePixelFormat,
|
||
|
pPictureIn->nWidth, pPictureIn->nHeight,
|
||
|
&nMemSizeY, &nMemSizeC, NULL, NULL, 0);
|
||
|
CdcMemFlushCache(memOps, (void*)pPictureIn->pData0, nMemSizeY+2*nMemSizeC);
|
||
|
pPictureOut->nPts = pPictureIn->nPts;
|
||
|
pOrgPicture = pPictureIn;
|
||
|
|
||
|
if(pPictureOut->ePixelFormat != pPictureIn->ePixelFormat)
|
||
|
{
|
||
|
if(nRotateDegree == 0)
|
||
|
{
|
||
|
ConvertPixelFormat(pPictureIn, pPictureOut);
|
||
|
return 0;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
pPictureMid = malloc(sizeof(VideoPicture));
|
||
|
pPictureMid->ePixelFormat = pPictureOut->ePixelFormat;
|
||
|
pPictureMid->nWidth = pPictureIn->nWidth;
|
||
|
pPictureMid->nHeight = pPictureIn->nHeight;
|
||
|
if((pPictureMid->ePixelFormat==PIXEL_FORMAT_YV12)
|
||
|
|| (pPictureMid->ePixelFormat==PIXEL_FORMAT_NV21))
|
||
|
{
|
||
|
|
||
|
int nHeight16Align = (pPictureIn->nHeight+15) &~ 15;
|
||
|
int nMemSizeY = pPictureIn->nLineStride*nHeight16Align;
|
||
|
int nMemSizeC = nMemSizeY>>2;
|
||
|
pPictureMid->pData0 = malloc(nMemSizeY+nMemSizeC*2);
|
||
|
if(pPictureMid->pData0 == NULL)
|
||
|
{
|
||
|
logd("malloc memory for pPictureMid failed\n");
|
||
|
}
|
||
|
ConvertPixelFormat(pPictureIn, pPictureMid);
|
||
|
pOrgPicture = pPictureMid;
|
||
|
bMallocBuffer = 1;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if(nRotateDegree == 0)
|
||
|
{
|
||
|
ret = RotatePicture0Degree(pOrgPicture, pPictureOut,nGpuYAlign,nGpuCAlign);
|
||
|
}
|
||
|
else if(nRotateDegree == 90)
|
||
|
{
|
||
|
ret = RotatePicture90Degree(pOrgPicture, pPictureOut);
|
||
|
}
|
||
|
else if(nRotateDegree == 180)
|
||
|
{
|
||
|
ret = RotatePicture180Degree(pOrgPicture, pPictureOut);
|
||
|
}
|
||
|
else if(nRotateDegree == 270)
|
||
|
{
|
||
|
ret = RotatePicture270Degree(pOrgPicture, pPictureOut);
|
||
|
}
|
||
|
if(pPictureMid != NULL)
|
||
|
{
|
||
|
if(bMallocBuffer)
|
||
|
free(pPictureMid->pData0);
|
||
|
free(pPictureMid);
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
#if 0 //remove the function for iommu mode
|
||
|
VideoPicture* AllocatePictureBuffer(struct ScMemOpsS* memOps,
|
||
|
int nWidth, int nHeight, int nLineStride, int ePixelFormat)
|
||
|
{
|
||
|
VideoPicture* p;
|
||
|
|
||
|
p = (VideoPicture*)malloc(sizeof(VideoPicture));
|
||
|
|
||
|
logi("AllocatePictureBuffer, nWidth=%d, nHeight=%d, nLineStride=%d, ePixelFormat=%s",
|
||
|
nWidth, nHeight, nLineStride, strPixelFormat[ePixelFormat]);
|
||
|
|
||
|
if(p == NULL)
|
||
|
{
|
||
|
logw("memory alloc fail, allocate %d bytes in AllocatePictureBuffer()", \
|
||
|
(int)sizeof(VideoPicture));
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
memset(p, 0, sizeof(VideoPicture));
|
||
|
p->nWidth = nWidth;
|
||
|
p->nHeight = nHeight;
|
||
|
p->nLineStride = nLineStride;
|
||
|
p->ePixelFormat = ePixelFormat;
|
||
|
int nAlignValue = 0;
|
||
|
|
||
|
if(FbmAllocatePictureBuffer(memOps, p, &nAlignValue, nWidth, nHeight) != 0)
|
||
|
{
|
||
|
logw("memory alloc fail, FbmAllocatePictureBuffer() fail in AllocatePictureBuffer().");
|
||
|
free(p);
|
||
|
p = NULL;
|
||
|
}
|
||
|
|
||
|
return p;
|
||
|
}
|
||
|
|
||
|
int FreePictureBuffer(struct ScMemOpsS* memOps, VideoPicture* pPicture)
|
||
|
{
|
||
|
//VideoDecoderContext* p;
|
||
|
|
||
|
logi("FreePictureBuffer, pPicture=%p", pPicture);
|
||
|
|
||
|
if(pPicture == NULL)
|
||
|
{
|
||
|
loge("invaid picture, FreePictureBuffer fail.");
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
FbmFreePictureBuffer(memOps, pPicture);
|
||
|
free(pPicture);
|
||
|
return 0;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
static int DecideStreamBufferSize(VideoDecoderContext* p)
|
||
|
{
|
||
|
//int nVideoHeight;
|
||
|
int nVideoWidth;
|
||
|
int eCodecFormat;
|
||
|
int nBufferSize = 8*1024*1024;
|
||
|
|
||
|
if(p->vconfig.nVbvBufferSize>=1*1024*1024 && p->vconfig.nVbvBufferSize<=32*1024*1024)
|
||
|
{
|
||
|
nBufferSize = p->vconfig.nVbvBufferSize;
|
||
|
return nBufferSize;
|
||
|
}
|
||
|
|
||
|
//* we decide stream buffer size by resolution and codec format.
|
||
|
//nVideoHeight = p->videoStreamInfo.nHeight;
|
||
|
nVideoWidth = p->videoStreamInfo.nWidth;
|
||
|
eCodecFormat = p->videoStreamInfo.eCodecFormat;
|
||
|
|
||
|
//* for skia create sbm
|
||
|
if((p->vconfig.eOutputPixelFormat == PIXEL_FORMAT_RGBA)
|
||
|
&& (eCodecFormat == VIDEO_CODEC_FORMAT_MJPEG)
|
||
|
&& p->vconfig.nVbvBufferSize)
|
||
|
{
|
||
|
nBufferSize = p->vconfig.nVbvBufferSize;
|
||
|
return nBufferSize;
|
||
|
}
|
||
|
|
||
|
//* if resolution is unknown, treat it as full HD source.
|
||
|
if(nVideoWidth == 0)
|
||
|
nVideoWidth = 1920;
|
||
|
|
||
|
if(nVideoWidth < 480)
|
||
|
nBufferSize = 2*1024*1024;
|
||
|
else if (nVideoWidth < 720)
|
||
|
nBufferSize = 4*1024*1024;
|
||
|
else if(nVideoWidth < 1080)
|
||
|
nBufferSize = 6*1024*1024;
|
||
|
else if(nVideoWidth < 2160)
|
||
|
nBufferSize = 8*1024*1024;
|
||
|
else
|
||
|
nBufferSize = 32*1024*1024;
|
||
|
|
||
|
if(eCodecFormat == VIDEO_CODEC_FORMAT_MJPEG ||
|
||
|
eCodecFormat == VIDEO_CODEC_FORMAT_MPEG1 ||
|
||
|
eCodecFormat == VIDEO_CODEC_FORMAT_MPEG2)
|
||
|
{
|
||
|
nBufferSize += 2*1024*1024; //* for old codec format, compress rate is low.
|
||
|
}
|
||
|
if(eCodecFormat == VIDEO_CODEC_FORMAT_RX)
|
||
|
{
|
||
|
nBufferSize = 16*1024*1024;
|
||
|
}
|
||
|
|
||
|
if(nVideoWidth >= 3840 && nBufferSize<=32*1024*1024)
|
||
|
{
|
||
|
nBufferSize = 32*1024*1024;
|
||
|
}
|
||
|
|
||
|
return nBufferSize;
|
||
|
}
|
||
|
|
||
|
static void UpdateVideoStreamInfo(VideoDecoderContext* p, VideoPicture* pPicture)
|
||
|
{
|
||
|
//* update video resolution.
|
||
|
if(p->vconfig.bRotationEn == 0 ||
|
||
|
p->vconfig.nRotateDegree == 0 ||
|
||
|
p->vconfig.nRotateDegree == 3)
|
||
|
{
|
||
|
//* the picture was not rotated or rotated by 180 degree.
|
||
|
if(p->vconfig.bScaleDownEn == 0)
|
||
|
{
|
||
|
//* the picture was not scaled.
|
||
|
if(p->videoStreamInfo.nWidth != pPicture->nWidth ||
|
||
|
p->videoStreamInfo.nHeight != pPicture->nHeight)
|
||
|
{
|
||
|
p->videoStreamInfo.nWidth = pPicture->nWidth;
|
||
|
p->videoStreamInfo.nHeight = pPicture->nHeight;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//* the picture was scaled.
|
||
|
if(p->videoStreamInfo.nWidth != \
|
||
|
(pPicture->nWidth<<p->vconfig.nHorizonScaleDownRatio) ||
|
||
|
p->videoStreamInfo.nHeight != \
|
||
|
(pPicture->nHeight<<p->vconfig.nVerticalScaleDownRatio))
|
||
|
{
|
||
|
p->videoStreamInfo.nWidth = \
|
||
|
pPicture->nWidth << p->vconfig.nHorizonScaleDownRatio;
|
||
|
p->videoStreamInfo.nHeight = \
|
||
|
pPicture->nHeight << p->vconfig.nVerticalScaleDownRatio;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//* the picture was rotated by 90 or 270 degree.
|
||
|
if(p->vconfig.bScaleDownEn == 0)
|
||
|
{
|
||
|
//* the picture was not scaled.
|
||
|
if(p->videoStreamInfo.nWidth != pPicture->nHeight ||
|
||
|
p->videoStreamInfo.nHeight != pPicture->nWidth)
|
||
|
{
|
||
|
p->videoStreamInfo.nWidth = pPicture->nHeight;
|
||
|
p->videoStreamInfo.nHeight = pPicture->nWidth;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
//* the picture was scaled.
|
||
|
if(p->videoStreamInfo.nWidth != \
|
||
|
(pPicture->nHeight<<p->vconfig.nHorizonScaleDownRatio) ||
|
||
|
p->videoStreamInfo.nHeight != \
|
||
|
(pPicture->nWidth<<p->vconfig.nVerticalScaleDownRatio))
|
||
|
{
|
||
|
p->videoStreamInfo.nWidth = \
|
||
|
pPicture->nHeight << p->vconfig.nHorizonScaleDownRatio;
|
||
|
p->videoStreamInfo.nHeight = \
|
||
|
pPicture->nWidth << p->vconfig.nVerticalScaleDownRatio;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//* update frame rate.
|
||
|
if(p->videoStreamInfo.nFrameRate != pPicture->nFrameRate && pPicture->nFrameRate != 0)
|
||
|
p->videoStreamInfo.nFrameRate = pPicture->nFrameRate;
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
//*****************************************************************************************//
|
||
|
//***added new interface function for new display structure********************************//
|
||
|
//******************************************************************************************//
|
||
|
FbmBufInfo* GetVideoFbmBufInfo(VideoDecoder* pDecoder)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
if(p == NULL)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
if(p->pVideoEngine != NULL)
|
||
|
{
|
||
|
int nStreamIndex = 0;
|
||
|
if(VideoEngineGetFbm(p->pVideoEngine, nStreamIndex) == NULL)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
return FbmGetVideoBufferInfo(&(p->pVideoEngine->fbmInfo));
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
VideoPicture* SetVideoFbmBufAddress(VideoDecoder* pDecoder,
|
||
|
VideoPicture* pVideoPicture, int bForbidUseFlag)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
if(p == NULL)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
if((CdcIonGetMemType() == MEMORY_IOMMU) && p->vconfig.bGpuBufValid)
|
||
|
{
|
||
|
|
||
|
struct user_iommu_param sUserIommuBuf;
|
||
|
#if 0
|
||
|
int ret;
|
||
|
logd("ycy p->nIonFd:%d, pPrivate:%p\n", p->nIonFd, pVideoPicture->pPrivate);
|
||
|
ret = pVideoPicture->nBufFd = CdcIonGetFd(p->nIonFd, (uintptr_t)pVideoPicture->pPrivate);
|
||
|
if(ret < 0)
|
||
|
return NULL;
|
||
|
#endif
|
||
|
|
||
|
sUserIommuBuf.fd = pVideoPicture->nBufFd;
|
||
|
if(p->pVideoEngine->bIsSoftwareDecoder == 0)
|
||
|
{
|
||
|
CdcVeGetIommuAddr(p->vconfig.veOpsS, p->vconfig.pVeOpsSelf, &sUserIommuBuf);
|
||
|
}
|
||
|
pVideoPicture->phyYBufAddr = sUserIommuBuf.iommu_addr;
|
||
|
pVideoPicture->phyCBufAddr = pVideoPicture->phyYBufAddr +
|
||
|
(pVideoPicture->nHeight * pVideoPicture->nLineStride);
|
||
|
}
|
||
|
if(p->pVideoEngine != NULL)
|
||
|
{
|
||
|
return FbmSetFbmBufAddress(&(p->pVideoEngine->fbmInfo), pVideoPicture, bForbidUseFlag);
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
int SetVideoFbmBufRelease(VideoDecoder* pDecoder)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
if(p == NULL)
|
||
|
{
|
||
|
return -1;
|
||
|
}
|
||
|
if(p->pVideoEngine != NULL)
|
||
|
{
|
||
|
return FbmSetNewDispRelease(&(p->pVideoEngine->fbmInfo));
|
||
|
}
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
VideoPicture* RequestReleasePicture(VideoDecoder* pDecoder)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
VideoPicture* pVpicture = NULL;
|
||
|
|
||
|
if(p == NULL)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
if(p->pVideoEngine != NULL)
|
||
|
{
|
||
|
//return FbmRequestReleasePicture(&(p->pVideoEngine->fbmInfo));
|
||
|
pVpicture = FbmRequestReleasePicture(&(p->pVideoEngine->fbmInfo));
|
||
|
if((CdcIonGetMemType()==MEMORY_IOMMU)&&(pVpicture!=NULL))
|
||
|
{
|
||
|
struct user_iommu_param sUserIommuBuf;
|
||
|
sUserIommuBuf.fd = pVpicture->nBufFd;
|
||
|
VideoFbmInfo* pFbmInfo = (VideoFbmInfo*)(p->pFbm[0]->pFbmInfo);
|
||
|
if(pFbmInfo->bIsSoftDecoderFlag==0)
|
||
|
{
|
||
|
CdcVeFreeIommuAddr(p->pVideoEngine->veOpsS,
|
||
|
p->pVideoEngine->pVeOpsSelf, &sUserIommuBuf);
|
||
|
}
|
||
|
}
|
||
|
return pVpicture;
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
VideoPicture* ReturnRelasePicture(VideoDecoder* pDecoder, \
|
||
|
VideoPicture* pVpicture, int bForbidUseFlag)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
if(p == NULL)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
if(p->pVideoEngine != NULL)
|
||
|
{
|
||
|
if((CdcIonGetMemType()==MEMORY_IOMMU)&& (p->vconfig.bGpuBufValid))
|
||
|
{
|
||
|
struct user_iommu_param sUserIommuBuf;
|
||
|
sUserIommuBuf.fd = pVpicture->nBufFd;
|
||
|
if(p->pVideoEngine->bIsSoftwareDecoder == 0)
|
||
|
{
|
||
|
CdcVeGetIommuAddr(p->vconfig.veOpsS, p->vconfig.pVeOpsSelf, &sUserIommuBuf);
|
||
|
}
|
||
|
pVpicture->phyYBufAddr = sUserIommuBuf.iommu_addr;
|
||
|
pVpicture->phyCBufAddr = pVpicture->phyYBufAddr +
|
||
|
(pVpicture->nHeight * pVpicture->nLineStride);
|
||
|
}
|
||
|
return FbmReturnReleasePicture(&(p->pVideoEngine->fbmInfo), pVpicture, bForbidUseFlag);
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
int SetDecodePerformCmd(VideoDecoder* pDecoder, enum EVDECODERSETPERFORMCMD performCmd)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
int ret;
|
||
|
DecoderInterface *pDecoderInterface;
|
||
|
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
if(p == NULL ||p->pVideoEngine == NULL ||
|
||
|
p->pVideoEngine->pDecoderInterface == NULL)
|
||
|
{
|
||
|
loge("SetDecodeDebugCmd fail\n");
|
||
|
return VDECODE_RESULT_UNSUPPORTED;
|
||
|
}
|
||
|
//eVersion = VeGetIcVersion();
|
||
|
pDecoderInterface = p->pVideoEngine->pDecoderInterface;
|
||
|
ret = 0;
|
||
|
if(pDecoderInterface->SetExtraScaleInfo != NULL)
|
||
|
{
|
||
|
ret = pDecoderInterface->SetPerformCmd(pDecoderInterface,performCmd);
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
int GetDecodePerformInfo(VideoDecoder* pDecoder,
|
||
|
enum EVDECODERGETPERFORMCMD performCmd, VDecodePerformaceInfo** performInfo)
|
||
|
{
|
||
|
VideoDecoderContext* p;
|
||
|
int ret;
|
||
|
DecoderInterface *pDecoderInterface;
|
||
|
p = (VideoDecoderContext*)pDecoder;
|
||
|
|
||
|
if(p == NULL ||p->pVideoEngine == NULL ||
|
||
|
p->pVideoEngine->pDecoderInterface == NULL)
|
||
|
{
|
||
|
loge("GetDecodePerformInfo fail\n");
|
||
|
return VDECODE_RESULT_UNSUPPORTED;
|
||
|
}
|
||
|
pDecoderInterface = p->pVideoEngine->pDecoderInterface;
|
||
|
ret = 0;
|
||
|
if(pDecoderInterface->SetExtraScaleInfo != NULL)
|
||
|
{
|
||
|
ret = pDecoderInterface->GetPerformInfo(pDecoderInterface,performCmd, performInfo);
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
void *VideoDecoderPallocIonBuf(VideoDecoder* pDecoder, int nSize)
|
||
|
{
|
||
|
VideoDecoderContext* p= (VideoDecoderContext *)pDecoder;
|
||
|
return CdcMemPalloc(p->vconfig.memops, nSize, (void *)p->vconfig.veOpsS, p->vconfig.pVeOpsSelf);
|
||
|
}
|
||
|
void VideoDecoderFreeIonBuf(VideoDecoder* pDecoder, void *pIonBuf)
|
||
|
{
|
||
|
VideoDecoderContext* p= (VideoDecoderContext *)pDecoder;
|
||
|
CdcMemPfree(p->vconfig.memops, pIonBuf, (void *)p->vconfig.veOpsS, p->vconfig.pVeOpsSelf);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
//*just for remove warning of cppcheck
|
||
|
static void vdecoderUseFunction()
|
||
|
{
|
||
|
CEDARC_UNUSE(vdecoderUseFunction);
|
||
|
CEDARC_UNUSE(CreateVideoDecoder);
|
||
|
CEDARC_UNUSE(DestroyVideoDecoder);
|
||
|
CEDARC_UNUSE(InitializeVideoDecoder);
|
||
|
CEDARC_UNUSE(ResetVideoDecoder);
|
||
|
CEDARC_UNUSE(DecodeVideoStream);
|
||
|
CEDARC_UNUSE(DecoderSetSpecialData);
|
||
|
CEDARC_UNUSE(GetVideoStreamInfo);
|
||
|
CEDARC_UNUSE(RequestVideoStreamBuffer);
|
||
|
CEDARC_UNUSE(SubmitVideoStreamData);
|
||
|
CEDARC_UNUSE(VideoStreamBufferSize);
|
||
|
CEDARC_UNUSE(VideoStreamDataSize);
|
||
|
CEDARC_UNUSE(VideoStreamFrameNum);
|
||
|
CEDARC_UNUSE(VideoStreamFrameNum);
|
||
|
CEDARC_UNUSE(RequestPicture);
|
||
|
CEDARC_UNUSE(ReturnPicture);
|
||
|
CEDARC_UNUSE(NextPictureInfo);
|
||
|
CEDARC_UNUSE(TotalPictureBufferNum);
|
||
|
CEDARC_UNUSE(EmptyPictureBufferNum);
|
||
|
CEDARC_UNUSE(ValidPictureNum);
|
||
|
CEDARC_UNUSE(ReopenVideoEngine);
|
||
|
CEDARC_UNUSE(RotatePicture);
|
||
|
CEDARC_UNUSE(RotatePictureHw);
|
||
|
CEDARC_UNUSE(AllocatePictureBuffer);
|
||
|
CEDARC_UNUSE(FreePictureBuffer);
|
||
|
CEDARC_UNUSE(ReturnRelasePicture);
|
||
|
CEDARC_UNUSE(RequestReleasePicture);
|
||
|
CEDARC_UNUSE(SetVideoFbmBufRelease);
|
||
|
CEDARC_UNUSE(SetVideoFbmBufAddress);
|
||
|
CEDARC_UNUSE(GetVideoFbmBufInfo);
|
||
|
CEDARC_UNUSE(ConfigExtraScaleInfo);
|
||
|
CEDARC_UNUSE(SetDecodePerformCmd);
|
||
|
CEDARC_UNUSE(GetDecodePerformInfo);
|
||
|
CEDARC_UNUSE(VideoStreamDataInfoPointer);
|
||
|
CEDARC_UNUSE(VideoDecoderPallocIonBuf);
|
||
|
CEDARC_UNUSE(VideoDecoderFreeIonBuf);
|
||
|
}
|