387 lines
11 KiB
C
387 lines
11 KiB
C
#include <sys/mman.h>
|
|
#include <fcntl.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <sys/ioctl.h>
|
|
|
|
#include "hi_common.h"
|
|
#include "hi_comm_video.h"
|
|
#include "hi_comm_sys.h"
|
|
#include "mpi_sys.h"
|
|
#include "hi_comm_vb.h"
|
|
#include "mpi_vb.h"
|
|
#include "hi_comm_vo.h"
|
|
#include "mpi_vo.h"
|
|
#include "mpi_vgs.h"
|
|
|
|
#define MAX_FRM_CNT 256
|
|
|
|
typedef struct hiDUMP_MEMBUF_S
|
|
{
|
|
VB_BLK hBlock;
|
|
VB_POOL hPool;
|
|
HI_U32 u32PoolId;
|
|
|
|
HI_U32 u32PhyAddr;
|
|
HI_U8* pVirAddr;
|
|
HI_S32 s32Mdev;
|
|
} DUMP_MEMBUF_S;
|
|
|
|
|
|
/* sp420 转存为 p420 ; sp422 转存为 p422 */
|
|
void sample_yuv_dump(VIDEO_FRAME_S* pVBuf, FILE* pfd)
|
|
{
|
|
unsigned int w, h;
|
|
char* pVBufVirt_Y;
|
|
char* pVBufVirt_C;
|
|
char* pMemContent;
|
|
unsigned char TmpBuff[4096]; //如果这个值太小,图像很大的话存不了
|
|
HI_U32 phy_addr, Ysize, Csize;
|
|
PIXEL_FORMAT_E enPixelFormat = pVBuf->enPixelFormat;
|
|
HI_U32 u32UvHeight;/* 存为planar 格式时的UV分量的高度 */
|
|
|
|
Ysize = (pVBuf->u32Stride[0]) * (pVBuf->u32Height);
|
|
if (PIXEL_FORMAT_YUV_SEMIPLANAR_420 == enPixelFormat)
|
|
{
|
|
Csize = (pVBuf->u32Stride[1]) * (pVBuf->u32Height) / 2;
|
|
u32UvHeight = pVBuf->u32Height / 2;
|
|
}
|
|
else if (PIXEL_FORMAT_YUV_SEMIPLANAR_422 == enPixelFormat)
|
|
{
|
|
Csize = (pVBuf->u32Stride[1]) * (pVBuf->u32Height);
|
|
u32UvHeight = pVBuf->u32Height;
|
|
}
|
|
else
|
|
{
|
|
Csize = 0;
|
|
u32UvHeight = 0;
|
|
}
|
|
|
|
phy_addr = pVBuf->u32PhyAddr[0];
|
|
|
|
//printf("phy_addr:%x, size:%d\n", phy_addr, size);
|
|
pVBufVirt_Y = (HI_CHAR*) HI_MPI_SYS_Mmap(phy_addr, Ysize);
|
|
if (NULL == pVBufVirt_Y)
|
|
{
|
|
return;
|
|
}
|
|
|
|
/* save Y ----------------------------------------------------------------*/
|
|
fprintf(stderr, "saving......Y......");
|
|
fflush(stderr);
|
|
for (h = 0; h < pVBuf->u32Height; h++)
|
|
{
|
|
pMemContent = pVBufVirt_Y + h * pVBuf->u32Stride[0];
|
|
fwrite(pMemContent, pVBuf->u32Width, 1, pfd);
|
|
}
|
|
fflush(pfd);
|
|
|
|
|
|
if (PIXEL_FORMAT_SINGLE != enPixelFormat)
|
|
{
|
|
pVBufVirt_C = (HI_CHAR*) HI_MPI_SYS_Mmap(pVBuf->u32PhyAddr[1], Csize);
|
|
if (NULL == pVBufVirt_C)
|
|
{
|
|
HI_MPI_SYS_Munmap(pVBufVirt_Y, Ysize);
|
|
return;
|
|
}
|
|
/* save U ----------------------------------------------------------------*/
|
|
fprintf(stderr, "U......");
|
|
fflush(stderr);
|
|
for (h = 0; h < u32UvHeight; h++)
|
|
{
|
|
pMemContent = pVBufVirt_C + h * pVBuf->u32Stride[1];
|
|
|
|
pMemContent += 1;
|
|
|
|
for (w = 0; w < pVBuf->u32Width / 2; w++)
|
|
{
|
|
TmpBuff[w] = *pMemContent;
|
|
pMemContent += 2;
|
|
}
|
|
fwrite(TmpBuff, pVBuf->u32Width / 2, 1, pfd);
|
|
}
|
|
fflush(pfd);
|
|
|
|
/* save V ----------------------------------------------------------------*/
|
|
fprintf(stderr, "V......");
|
|
fflush(stderr);
|
|
for (h = 0; h < u32UvHeight; h++)
|
|
{
|
|
pMemContent = pVBufVirt_C + h * pVBuf->u32Stride[1];
|
|
|
|
for (w = 0; w < pVBuf->u32Width / 2; w++)
|
|
{
|
|
TmpBuff[w] = *pMemContent;
|
|
pMemContent += 2;
|
|
}
|
|
fwrite(TmpBuff, pVBuf->u32Width / 2, 1, pfd);
|
|
}
|
|
fflush(pfd);
|
|
HI_MPI_SYS_Munmap(pVBufVirt_C, Csize);
|
|
}
|
|
|
|
fprintf(stderr, "done %d!\n", pVBuf->u32TimeRef);
|
|
fflush(stderr);
|
|
|
|
HI_MPI_SYS_Munmap(pVBufVirt_Y, Ysize);
|
|
|
|
}
|
|
|
|
HI_S32 SAMPLE_MISC_VoDump(VO_DEV VoDev, VO_CHN VoChn, HI_U32 u32Cnt)
|
|
{
|
|
HI_S32 i, s32Ret;
|
|
VIDEO_FRAME_INFO_S stFrame;
|
|
//VIDEO_FRAME_INFO_S astFrame[256];
|
|
HI_CHAR szYuvName[128];
|
|
HI_CHAR szPixFrm[10];
|
|
FILE* pfd;
|
|
VB_POOL hPool = VB_INVALID_POOLID;
|
|
HI_U32 u32BlkSize = 0;
|
|
|
|
DUMP_MEMBUF_S stMem = {0};
|
|
VIDEO_FRAME_INFO_S stFrmInfo;
|
|
VGS_HANDLE hHandle;
|
|
VGS_TASK_ATTR_S stTask;
|
|
HI_U32 u32LumaSize = 0;
|
|
HI_U32 u32ChrmSize = 0;
|
|
HI_U32 u32PicLStride = 0;
|
|
HI_U32 u32PicCStride = 0;
|
|
HI_U32 u32Width = 0;
|
|
HI_U32 u32Height = 0;
|
|
HI_BOOL bSendToVgs = HI_FALSE;
|
|
|
|
printf("\nNOTICE: This tool only can be used for TESTING !!!\n");
|
|
printf("usage: ./vou_chn_dump [vodev] [vochn] [frmcnt]. sample: ./vou_chn_dump 0 0 5\n\n");
|
|
|
|
/* Get Frame to make file name*/
|
|
s32Ret = HI_MPI_VO_GetChnFrame(VoDev, VoChn, &stFrame, 0);
|
|
if (HI_SUCCESS != s32Ret)
|
|
{
|
|
printf("HI_MPI_VO(%d)_GetChnFrame errno %#x\n", VoDev, s32Ret);
|
|
return -1;
|
|
}
|
|
|
|
/* make file name */
|
|
if (PIXEL_FORMAT_YUV_SEMIPLANAR_420 == stFrame.stVFrame.enPixelFormat)
|
|
{
|
|
strcpy(szPixFrm, "p420");
|
|
}
|
|
else if (PIXEL_FORMAT_YUV_SEMIPLANAR_422 == stFrame.stVFrame.enPixelFormat)
|
|
{
|
|
strcpy(szPixFrm, "p422");
|
|
}
|
|
else if (PIXEL_FORMAT_SINGLE == stFrame.stVFrame.enPixelFormat)
|
|
{
|
|
strcpy(szPixFrm, "single");
|
|
}
|
|
else
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
sprintf(szYuvName, "./vo(%d,%d)_%d_%d_%s_%d.yuv", VoDev, VoChn,
|
|
stFrame.stVFrame.u32Width, stFrame.stVFrame.u32Height, szPixFrm, u32Cnt);
|
|
printf("Dump YUV frame of vo(%d,%d) to file: \"%s\"\n", VoDev, VoChn, szYuvName);
|
|
|
|
bSendToVgs = (stFrame.stVFrame.enCompressMode > 0) || (stFrame.stVFrame.enVideoFormat > 0);
|
|
|
|
HI_MPI_VO_ReleaseChnFrame(VoDev, VoChn, &stFrame);
|
|
|
|
/* open file */
|
|
pfd = fopen(szYuvName, "wb");
|
|
|
|
if (NULL == pfd)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
u32PicLStride = stFrame.stVFrame.u32Stride[0];
|
|
u32PicCStride = stFrame.stVFrame.u32Stride[0];
|
|
u32LumaSize = stFrame.stVFrame.u32Stride[0] * stFrame.stVFrame.u32Height;
|
|
u32Width = stFrame.stVFrame.u32Width;
|
|
u32Height = stFrame.stVFrame.u32Height;
|
|
if (PIXEL_FORMAT_SINGLE == stFrame.stVFrame.enPixelFormat)
|
|
{
|
|
u32ChrmSize = 0;
|
|
}
|
|
else
|
|
{
|
|
u32ChrmSize = (stFrame.stVFrame.u32Stride[0] * stFrame.stVFrame.u32Height) >> (1 + (PIXEL_FORMAT_YUV_SEMIPLANAR_420 == stFrame.stVFrame.enPixelFormat));
|
|
}
|
|
|
|
if (bSendToVgs)
|
|
{
|
|
if (PIXEL_FORMAT_SINGLE == stFrame.stVFrame.enPixelFormat)
|
|
{
|
|
u32BlkSize = stFrame.stVFrame.u32Stride[0] * stFrame.stVFrame.u32Height;
|
|
}
|
|
else
|
|
{
|
|
u32BlkSize = (PIXEL_FORMAT_YUV_SEMIPLANAR_420 == stFrame.stVFrame.enPixelFormat)
|
|
? (stFrame.stVFrame.u32Stride[0] * stFrame.stVFrame.u32Height * 3 >> 1) : (stFrame.stVFrame.u32Stride[0] * stFrame.stVFrame.u32Height * 2);
|
|
}
|
|
|
|
/*create comm vb pool*/
|
|
hPool = HI_MPI_VB_CreatePool( u32BlkSize, 2, NULL);
|
|
if (hPool == VB_INVALID_POOLID)
|
|
{
|
|
printf("HI_MPI_VB_CreatePool failed! \n");
|
|
goto END1;
|
|
}
|
|
|
|
stMem.hPool = hPool;
|
|
}
|
|
|
|
/* get VO frame */
|
|
for (i = 0; i < u32Cnt; i++)
|
|
{
|
|
s32Ret = HI_MPI_VO_GetChnFrame(VoDev, VoChn, &stFrame, 20);
|
|
if (HI_SUCCESS != s32Ret)
|
|
{
|
|
printf("get vo(%d,%d) frame err\n", VoDev, VoChn);
|
|
printf("only get %d frame\n", i);
|
|
break;
|
|
}
|
|
|
|
if (bSendToVgs)
|
|
{
|
|
while ((stMem.hBlock = HI_MPI_VB_GetBlock(stMem.hPool, u32BlkSize, NULL)) == VB_INVALID_HANDLE)
|
|
{
|
|
;
|
|
}
|
|
|
|
stMem.u32PhyAddr = HI_MPI_VB_Handle2PhysAddr(stMem.hBlock);
|
|
|
|
|
|
stMem.pVirAddr = (HI_U8*) HI_MPI_SYS_Mmap( stMem.u32PhyAddr, u32BlkSize );
|
|
if (stMem.pVirAddr == NULL)
|
|
{
|
|
printf("Mem dev may not open\n");
|
|
HI_MPI_VB_ReleaseBlock(stMem.hBlock);
|
|
goto END2;
|
|
}
|
|
|
|
memset(&stFrmInfo.stVFrame, 0, sizeof(VIDEO_FRAME_S));
|
|
stFrmInfo.stVFrame.u32PhyAddr[0] = stMem.u32PhyAddr;
|
|
stFrmInfo.stVFrame.u32PhyAddr[1] = stFrmInfo.stVFrame.u32PhyAddr[0] + u32LumaSize;
|
|
stFrmInfo.stVFrame.u32PhyAddr[2] = stFrmInfo.stVFrame.u32PhyAddr[1] + u32ChrmSize;
|
|
|
|
stFrmInfo.stVFrame.pVirAddr[0] = stMem.pVirAddr;
|
|
stFrmInfo.stVFrame.pVirAddr[1] = (HI_U8*) stFrmInfo.stVFrame.pVirAddr[0] + u32LumaSize;
|
|
stFrmInfo.stVFrame.pVirAddr[2] = (HI_U8*) stFrmInfo.stVFrame.pVirAddr[1] + u32ChrmSize;
|
|
|
|
stFrmInfo.stVFrame.u32Width = u32Width;
|
|
stFrmInfo.stVFrame.u32Height = u32Height;
|
|
stFrmInfo.stVFrame.u32Stride[0] = u32PicLStride;
|
|
stFrmInfo.stVFrame.u32Stride[1] = u32PicCStride;
|
|
stFrmInfo.stVFrame.u32Stride[2] = u32PicCStride;
|
|
|
|
stFrmInfo.stVFrame.enCompressMode = COMPRESS_MODE_NONE;
|
|
stFrmInfo.stVFrame.enPixelFormat = stFrame.stVFrame.enPixelFormat;
|
|
stFrmInfo.stVFrame.enVideoFormat = VIDEO_FORMAT_LINEAR;
|
|
|
|
stFrmInfo.stVFrame.u64pts = (i * 40);
|
|
stFrmInfo.stVFrame.u32TimeRef = (i * 2);
|
|
|
|
stFrmInfo.u32PoolId = hPool;
|
|
|
|
s32Ret = HI_MPI_VGS_BeginJob(&hHandle);
|
|
if (s32Ret != HI_SUCCESS)
|
|
{
|
|
printf("HI_MPI_VGS_BeginJob failed\n");
|
|
HI_MPI_VB_ReleaseBlock(stMem.hBlock);
|
|
HI_MPI_VO_ReleaseChnFrame(VoDev, VoChn, &stFrame);
|
|
goto END2;
|
|
}
|
|
|
|
memcpy(&stTask.stImgIn, &stFrame.stVFrame, sizeof(VIDEO_FRAME_INFO_S));
|
|
memcpy(&stTask.stImgOut , &stFrmInfo, sizeof(VIDEO_FRAME_INFO_S));
|
|
s32Ret = HI_MPI_VGS_AddScaleTask(hHandle, &stTask);
|
|
if (s32Ret != HI_SUCCESS)
|
|
{
|
|
printf("HI_MPI_VGS_AddScaleTask failed\n");
|
|
HI_MPI_VGS_CancelJob(hHandle);
|
|
HI_MPI_VB_ReleaseBlock(stMem.hBlock);
|
|
HI_MPI_VO_ReleaseChnFrame(VoDev, VoChn, &stFrame);
|
|
goto END2;
|
|
}
|
|
|
|
s32Ret = HI_MPI_VGS_EndJob(hHandle);
|
|
if (s32Ret != HI_SUCCESS)
|
|
{
|
|
printf("HI_MPI_VGS_EndJob failed\n");
|
|
HI_MPI_VGS_CancelJob(hHandle);
|
|
HI_MPI_VB_ReleaseBlock(stMem.hBlock);
|
|
HI_MPI_VO_ReleaseChnFrame(VoDev, VoChn, &stFrame);
|
|
goto END2;
|
|
}
|
|
|
|
/* save VO frame to file */
|
|
sample_yuv_dump(&stFrmInfo.stVFrame, pfd);
|
|
|
|
HI_MPI_VB_ReleaseBlock(stMem.hBlock);
|
|
}
|
|
else
|
|
{
|
|
/* save VO frame to file */
|
|
sample_yuv_dump(&stFrame.stVFrame, pfd);
|
|
}
|
|
|
|
/* release frame after using */
|
|
s32Ret = HI_MPI_VO_ReleaseChnFrame(VoDev, VoChn, &stFrame);
|
|
if (HI_SUCCESS != s32Ret)
|
|
{
|
|
printf("Release vo(%d,%d) frame err\n", VoDev, VoChn);
|
|
printf("only get %d frame\n", i);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
END2:
|
|
if (hPool != VB_INVALID_POOLID)
|
|
{
|
|
HI_MPI_VB_DestroyPool( hPool );
|
|
}
|
|
|
|
END1:
|
|
|
|
fclose(pfd);
|
|
|
|
return 0;
|
|
}
|
|
|
|
HI_S32 main(int argc, char* argv[])
|
|
{
|
|
VO_DEV VoDev = 0;
|
|
VO_CHN VoChn = 0;
|
|
HI_U32 u32FrmCnt = 1;
|
|
|
|
/* VO设备号*/
|
|
if (argc > 1)
|
|
{
|
|
VoDev = atoi(argv[1]);
|
|
}
|
|
|
|
/* VO通道号 */
|
|
if (argc > 2)
|
|
{
|
|
VoChn = atoi(argv[2]);
|
|
}
|
|
|
|
/* 需要采集的帧数目*/
|
|
if (argc > 3)
|
|
{
|
|
u32FrmCnt = atoi(argv[3]);
|
|
}
|
|
|
|
SAMPLE_MISC_VoDump(VoDev, VoChn, u32FrmCnt);
|
|
|
|
return HI_SUCCESS;
|
|
}
|
|
|
|
|