SmartAudio/package/allwinner/tina_multimedia/libcedarx/demo/muxtest/testVideoMuxer.c

717 lines
22 KiB
C
Executable File

/*
* Copyright (c) 2008-2016 Allwinner Technology Co. Ltd.
* All rights reserved.
*
* File : testVideoMuxer.c
* Description : testVideoMuxer
* History :
*
*/
#include <cdx_log.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/time.h>
#include <time.h>
#include "vencoder.h"
#include "CdxMuxer.h"
#include "memoryAdapter.h"
#include "MuxerWriter.h"
#define DEMO_FILE_NAME_LEN 256
#define _ENCODER_TIME_
#ifdef _ENCODER_TIME_
static long long GetNowUs()
{
struct timeval now;
gettimeofday(&now, NULL);
return now.tv_sec * 1000000 + now.tv_usec;
}
extern CdxWriterT *CdxWriterCreat();
long long time1=0;
long long time2=0;
long long time3=0;
long long mux_time1 = 0, mux_time2 = 0, mux_time3 = 0;
long long total_time1 = 0, total_time2 = 0, total_time3 = 0;
#endif
typedef struct {
char intput_file[256];
char output_file[256];
char stream_file[256];
unsigned int encode_frame_num;
unsigned int encode_format;
unsigned int src_size;
unsigned int dst_size;
unsigned int src_width;
unsigned int src_height;
unsigned int dst_width;
unsigned int dst_height;
int bit_rate;
int frame_rate;
int maxKeyFrame;
unsigned char write_unmux;
int muxer_type;
}encode_param_t;
typedef enum {
INPUT,
HELP,
ENCODE_FRAME_NUM,
ENCODE_FORMAT,
OUTPUT,
SRC_SIZE,
DST_SIZE,
MP4,
TS,
INVALID
}ARGUMENT_T;
typedef struct {
char Short[8];
char Name[128];
ARGUMENT_T argument;
char Description[512];
}argument_t;
static const argument_t ArgumentMapping[] =
{
{ "-h", "--help", HELP,
"Print this help" },
{ "-i", "--input", INPUT,
"Input file path" },
{ "-n", "--encode_frame_num", ENCODE_FRAME_NUM,
"After encoder n frames, encoder stop" },
{ "-f", "--encode_format", ENCODE_FORMAT,
"0:h264 encoder, 1:jpeg_encoder" },
{ "-o", "--output", OUTPUT,
"output file path" },
{ "-s", "--srcsize", SRC_SIZE,
"src_size,can be 1080,720,480" },
{ "-d", "--dstsize", DST_SIZE,
"dst_size,can be 1080,720,480" },
{ "-m", "--mp4", MP4,
"output mp4 file path" },
{ "-t", "--ts", TS,
"output mp4 file path" },
};
int yu12_nv12(unsigned int width, unsigned int height,
unsigned char *addr_uv, unsigned char *addr_tmp_uv)
{
unsigned int i, chroma_bytes;
unsigned char *u_addr = NULL;
unsigned char *v_addr = NULL;
unsigned char *tmp_addr = NULL;
chroma_bytes = width*height/4;
u_addr = addr_uv;
v_addr = addr_uv + chroma_bytes;
tmp_addr = addr_tmp_uv;
for(i=0; i<chroma_bytes; i++)
{
*(tmp_addr++) = *(u_addr++);
*(tmp_addr++) = *(v_addr++);
}
memcpy(addr_uv, addr_tmp_uv, chroma_bytes*2);
return 0;
}
ARGUMENT_T GetArgument(char *name)
{
int i = 0;
int num = sizeof(ArgumentMapping) / sizeof(argument_t);
while(i < num)
{
if((0 == strcmp(ArgumentMapping[i].Name, name)) ||
((0 == strcmp(ArgumentMapping[i].Short, name)) &&
(0 != strcmp(ArgumentMapping[i].Short, "--"))))
{
return ArgumentMapping[i].argument;
}
i++;
}
return INVALID;
}
static void PrintDemoUsage(void)
{
int i = 0;
int num = sizeof(ArgumentMapping) / sizeof(argument_t);
printf("Usage:");
while(i < num)
{
printf("%s %-32s %s", ArgumentMapping[i].Short, ArgumentMapping[i].Name,
ArgumentMapping[i].Description);
printf("\n");
i++;
}
}
void demoParseArgument(encode_param_t *encode_param, char *argument, char *value)
{
ARGUMENT_T arg;
arg = GetArgument(argument);
int len = 0;
if(arg != HELP)
len = strlen(value);
if(len > DEMO_FILE_NAME_LEN)
return;
switch(arg)
{
case HELP:
PrintDemoUsage();
exit(-1);
case INPUT:
memset(encode_param->intput_file, 0, sizeof(encode_param->intput_file));
sscanf(value, "%255s", encode_param->intput_file);
logd(" get input file: %s ", encode_param->intput_file);
break;
case ENCODE_FRAME_NUM:
sscanf(value, "%u", &encode_param->encode_frame_num);
break;
case ENCODE_FORMAT:
sscanf(value, "%u", &encode_param->encode_format);
break;
case OUTPUT:
memset(encode_param->output_file, 0, sizeof(encode_param->output_file));
sscanf(value, "%255s", encode_param->output_file);
logd(" get output file: %s ", encode_param->output_file);
encode_param->write_unmux = 1;
break;
case SRC_SIZE:
sscanf(value, "%u", &encode_param->src_size);
logd(" get src_size: %dp ", encode_param->src_size);
if(encode_param->src_size == 1080)
{
encode_param->src_width = 1920;
encode_param->src_height = 1080;
}
else if(encode_param->src_size == 720)
{
encode_param->src_width = 1280;
encode_param->src_height = 720;
}
else if(encode_param->src_size == 480)
{
encode_param->src_width = 640;
encode_param->src_height = 480;
}
else if(encode_param->src_size == 240)
{
encode_param->src_width = 320;
encode_param->src_height = 240;
}
else
{
encode_param->src_width = 1280;
encode_param->src_height = 720;
}
break;
case DST_SIZE:
sscanf(value, "%u", &encode_param->dst_size);
logd(" get dst_size: %dp ", encode_param->dst_size);
if(encode_param->dst_size == 1080)
{
encode_param->dst_width = 1920;
encode_param->dst_height = 1080;
}
else if(encode_param->dst_size == 720)
{
encode_param->dst_width = 1280;
encode_param->dst_height = 720;
}
else if(encode_param->dst_size == 480)
{
encode_param->dst_width = 640;
encode_param->dst_height = 480;
}
else if(encode_param->dst_size == 240)
{
encode_param->dst_width = 320;
encode_param->dst_height = 240;
}
else
{
encode_param->dst_width = 1280;
encode_param->dst_height = 720;
logw("encoder demo only support the size 1080p,720p,480p, \
now use the default size 720p\n");
}
break;
case MP4:
memset(encode_param->stream_file, 0, sizeof(encode_param->stream_file));
sscanf(value, "%255s", encode_param->stream_file);
logd(" get mp4 file: %s ", encode_param->stream_file);
encode_param->muxer_type = CDX_MUXER_MOV;
break;
case TS:
memset(encode_param->stream_file, 0, sizeof(encode_param->stream_file));
sscanf(value, "%255s", encode_param->stream_file);
logd(" get ts file: %s ", encode_param->stream_file);
encode_param->muxer_type = CDX_MUXER_TS;
break;
case INVALID:
default:
logd("unknowed argument : %s", argument);
break;
}
}
void DemoHelpInfo(void)
{
logd(" ==== CedarX2.0 encoder demo help start ===== ");
logd(" -h or --help to show the demo usage");
logd(" demo created by yangcaoyuan, allwinnertech/AL3 ");
logd(" email: yangcaoyuan@allwinnertech.com ");
logd(" ===== CedarX2.0 encoder demo help end ====== ");
}
int main(int argc, char** argv)
{
VencBaseConfig baseConfig;
VencAllocateBufferParam bufferParam;
VideoEncoder* pVideoEnc = NULL;
VencInputBuffer inputBuffer;
VencOutputBuffer outputBuffer;
unsigned char *uv_tmp_buffer = NULL;
int result = 0;
int i = 0;
long long pts = 0;
FILE *in_file = NULL;
FILE *out_file = NULL;
char *input_path = NULL;
char *output_path = NULL;
char *stream_path = NULL;
encode_param_t encode_param;
/********************** Define H264 Paramerter *************************/
VencHeaderData sps_pps_data;
VencH264Param h264Param;
/********************** Define H264 Paramerter *************************/
/*************** Define JPEG Paramerter ******************/
EXIFInfo exifinfo;
int quality = 90;
int jpeg_mode = 1;
/*************** Define JPEG Paramerter **************/
/***************** Define MUXER Paramerter **********************/
MuxerWriterT *mw = NULL;
CdxWriterT *writer = NULL;
CdxMuxerMediaInfoT media_info;
CdxMuxerT *mux = NULL;
#if FS_WRITER
CdxFsCacheMemInfo fs_cache_mem;
int fs_mode = FSWRITEMODE_DIRECT;
#endif
/******************** Define MUXER Paramerter ********************/
memset(&encode_param, 0, sizeof(encode_param));
encode_param.src_width = 1280;
encode_param.src_height = 720;
encode_param.dst_width = 1280;
encode_param.dst_height = 720;
encode_param.bit_rate = 2 * 1024 * 1024;
encode_param.frame_rate = 30;
encode_param.maxKeyFrame = 30;
encode_param.encode_format = VENC_CODEC_H264;
encode_param.muxer_type = CDX_MUXER_MOV;
encode_param.encode_frame_num = 200;
strcpy((char*)encode_param.intput_file, "/mnt/sdcard/DCIM/Camera/720p.yuv");
strcpy((char*)encode_param.output_file, "/mnt/sdcard/DCIM/Camera/720p.264");
strcpy((char*)encode_param.stream_file, "write:///mnt/sdcard/DCIM/Camera/720p_h264.mp4");
//parse the config paramter
if(argc >= 2)
{
for(i = 1; i < (int)argc; i += 2)
{
demoParseArgument(&encode_param, argv[i], argv[i + 1]);
}
}
else
{
printf(" we need more arguments ");
PrintDemoUsage();
return 0;
}
input_path = encode_param.intput_file;
output_path = encode_param.output_file;
stream_path = encode_param.stream_file;
in_file = fopen(input_path, "rb");
if(in_file == NULL)
{
loge("open in_file fail\n");
return -1;
}
if (encode_param.write_unmux)
{
out_file = fopen(output_path, "wb");
if(out_file == NULL)
{
loge("open out_file fail\n");
fclose(in_file);
return -1;
}
}
memset(&baseConfig, 0 ,sizeof(VencBaseConfig));
memset(&bufferParam, 0 ,sizeof(VencAllocateBufferParam));
if ((baseConfig.memops = MemAdapterGetOpsS()) == NULL)
{
loge("baseConfig.memops is NULL");
fclose(in_file);
return -1;
}
CdcMemOpen(baseConfig.memops);
baseConfig.nInputWidth= encode_param.src_width;
baseConfig.nInputHeight = encode_param.src_height;
baseConfig.nStride = encode_param.src_width;
baseConfig.nDstWidth = encode_param.dst_width;
baseConfig.nDstHeight = encode_param.dst_height;
baseConfig.eInputFormat = VENC_PIXEL_YUV420P;
bufferParam.nSizeY = baseConfig.nInputWidth*baseConfig.nInputHeight;
bufferParam.nSizeC = baseConfig.nInputWidth*baseConfig.nInputHeight/2;
bufferParam.nBufferNum = 4;
/******************************* Set H264 Parameters ****************************/
h264Param.bEntropyCodingCABAC = 1;
h264Param.nBitrate = encode_param.bit_rate;
h264Param.nFramerate = encode_param.frame_rate;
h264Param.nCodingMode = VENC_FRAME_CODING;
h264Param.nMaxKeyInterval = encode_param.maxKeyFrame;
h264Param.sProfileLevel.nProfile = VENC_H264ProfileMain;
h264Param.sProfileLevel.nLevel = VENC_H264Level31;
h264Param.sQPRange.nMinqp = 10;
h264Param.sQPRange.nMaxqp = 40;
/******************************* Set H264 Parameters ****************************/
/******************************* Set JPEG Parameters ****************************/
exifinfo.ThumbWidth = 176;
exifinfo.ThumbHeight = 144;
strcpy((char*)exifinfo.CameraMake, "allwinner make test");
strcpy((char*)exifinfo.CameraModel, "allwinner model test");
strcpy((char*)exifinfo.DateTime, "2014:02:21 10:54:05");
strcpy((char*)exifinfo.gpsProcessingMethod, "allwinner gps");
exifinfo.Orientation = 0;
exifinfo.ExposureTime.num = 2;
exifinfo.ExposureTime.den = 1000;
exifinfo.FNumber.num = 20;
exifinfo.FNumber.den = 10;
exifinfo.ISOSpeed = 50;
exifinfo.ExposureBiasValue.num= -4;
exifinfo.ExposureBiasValue.den= 1;
exifinfo.MeteringMode = 1;
exifinfo.FlashUsed = 0;
exifinfo.FocalLength.num = 1400;
exifinfo.FocalLength.den = 100;
exifinfo.DigitalZoomRatio.num = 4;
exifinfo.DigitalZoomRatio.den = 1;
exifinfo.WhiteBalance = 1;
exifinfo.ExposureMode = 1;
exifinfo.enableGpsInfo = 1;
exifinfo.gps_latitude = 23.2368;
exifinfo.gps_longitude = 24.3244;
exifinfo.gps_altitude = 1234.5;
exifinfo.gps_timestamp = (long)time(NULL);
strcpy((char*)exifinfo.CameraSerialNum, "123456789");
strcpy((char*)exifinfo.ImageName, "exif-name-test");
strcpy((char*)exifinfo.ImageDescription, "exif-descriptor-test");
/******************************* Set JPEG Parameters ****************************/
/******************************* Set Muxer Parameters ****************************/
memset(&media_info, 0, sizeof(CdxMuxerMediaInfoT));
media_info.audioNum = 0;
media_info.videoNum = 1;
media_info.video.nWidth = encode_param.dst_width;
media_info.video.nHeight = encode_param.dst_height;
media_info.video.nFrameRate = encode_param.frame_rate;
media_info.video.eCodeType = encode_param.encode_format;
if ((writer = CdxWriterCreat()) == NULL)
{
loge("writer creat failed\n");
fclose(in_file);
fclose(out_file);
return -1;
}
mw = (MuxerWriterT*)writer;
strcpy(mw->file_path, stream_path);
mw->file_mode = FD_FILE_MODE;
MWOpen(writer);
mux = CdxMuxerCreate(encode_param.muxer_type, writer);
CdxMuxerSetMediaInfo(mux, &media_info);
#if FS_WRITER
memset(&fs_cache_mem, 0, sizeof(CdxFsCacheMemInfo));
fs_cache_mem.m_cache_size = 1024 * 1024; // must be less than 512 * 1024
fs_cache_mem.mp_cache = (cdx_uint8*)malloc(fs_cache_mem.m_cache_size);
if (fs_cache_mem.mp_cache == NULL)
{
loge("fs_cache_mem.mp_cache malloc failed\n");
fclose(in_file);
fclose(out_file);
return -1;
}
CdxMuxerControl(mux, SET_CACHE_MEM, &fs_cache_mem);
fs_mode = FSWRITEMODE_CACHETHREAD;
/*
fs_cache_size = 1024 * 1024;
CdxMuxerControl(mux, SET_FS_SIMPLE_CACHE_SIZE, &fs_cache_size);
fs_mode = FSWRITEMODE_SIMPLECACHE;
*/
//fs_mode = FSWRITEMODE_DIRECT;
CdxMuxerControl(mux, SET_FS_WRITE_MODE, &fs_mode);
#endif
CdxMuxerWriteHeader(mux);
/******************************* Set Muxer Parameters ****************************/
pVideoEnc = VideoEncCreate(encode_param.encode_format);
if(encode_param.encode_format == VENC_CODEC_JPEG)
{
VideoEncSetParameter(pVideoEnc, VENC_IndexParamJpegExifInfo, &exifinfo);
VideoEncSetParameter(pVideoEnc, VENC_IndexParamJpegQuality, &quality);
VideoEncSetParameter(pVideoEnc, VENC_IndexParamJpegEncMode, &jpeg_mode);
}
else if(encode_param.encode_format == VENC_CODEC_H264)
{
VideoEncSetParameter(pVideoEnc, VENC_IndexParamH264Param, &h264Param);
int vbvSize = 4*1024*1024;
VideoEncSetParameter(pVideoEnc, VENC_IndexParamSetVbvSize, &vbvSize);
}
VideoEncInit(pVideoEnc, &baseConfig);
if(encode_param.encode_format == VENC_CODEC_H264)
{
VideoEncGetParameter(pVideoEnc, VENC_IndexParamH264SPSPPS, &sps_pps_data);
CdxMuxerWriteExtraData(mux, sps_pps_data.pBuffer, sps_pps_data.nLength, 0);
if (encode_param.write_unmux)
{
fwrite(sps_pps_data.pBuffer, 1, sps_pps_data.nLength, out_file);
}
logd("sps_pps_data.nLength: %d", sps_pps_data.nLength);
}
AllocInputBuffer(pVideoEnc, &bufferParam);
if(baseConfig.eInputFormat == VENC_PIXEL_YUV420SP)
{
uv_tmp_buffer = (unsigned char*)malloc(baseConfig.nInputWidth*baseConfig.nInputHeight/2);
if(uv_tmp_buffer == NULL)
{
loge("malloc uv_tmp_buffer fail\n");
if (encode_param.write_unmux)
{
fclose(out_file);
}
fclose(in_file);
return -1;
}
}
unsigned int testNumber = 0;
total_time1 = GetNowUs();
while(testNumber < encode_param.encode_frame_num)
{
CdxMuxerPacketT pkt;
memset(&pkt, 0, sizeof(CdxMuxerPacketT));
GetOneAllocInputBuffer(pVideoEnc, &inputBuffer);
{
unsigned int size1, size2;
size1 = fread(inputBuffer.pAddrVirY, 1,
baseConfig.nInputWidth*baseConfig.nInputHeight, in_file);
size2 = fread(inputBuffer.pAddrVirC, 1,
baseConfig.nInputWidth*baseConfig.nInputHeight/2, in_file);
if((size1!= baseConfig.nInputWidth*baseConfig.nInputHeight) ||
(size2!= baseConfig.nInputWidth*baseConfig.nInputHeight/2))
{
fseek(in_file, 0, SEEK_SET);
size1 = fread(inputBuffer.pAddrVirY, 1,
baseConfig.nInputWidth*baseConfig.nInputHeight, in_file);
size2 = fread(inputBuffer.pAddrVirC, 1,
baseConfig.nInputWidth*baseConfig.nInputHeight/2, in_file);
}
if(baseConfig.eInputFormat == VENC_PIXEL_YUV420SP)
{
yu12_nv12(baseConfig.nInputWidth, baseConfig.nInputHeight,
inputBuffer.pAddrVirC, uv_tmp_buffer);
}
}
inputBuffer.bEnableCorp = 0;
inputBuffer.sCropInfo.nLeft = 240;
inputBuffer.sCropInfo.nTop = 240;
inputBuffer.sCropInfo.nWidth = 240;
inputBuffer.sCropInfo.nHeight = 240;
FlushCacheAllocInputBuffer(pVideoEnc, &inputBuffer);
pts += 1.0 / encode_param.frame_rate * 1000000;
inputBuffer.nPts = pts;
AddOneInputBuffer(pVideoEnc, &inputBuffer);
time1 = GetNowUs();
VideoEncodeOneFrame(pVideoEnc);
time2 = GetNowUs();
logv("encode frame %d use time is %lldus..\n",testNumber,(time2-time1));
time3 += time2-time1;
AlreadyUsedInputBuffer(pVideoEnc,&inputBuffer);
ReturnOneAllocInputBuffer(pVideoEnc, &inputBuffer);
result = GetOneBitstreamFrame(pVideoEnc, &outputBuffer);
if(result == -1)
{
goto out;
}
/*************************** Write Packet **********************************/
pkt.buflen = outputBuffer.nSize0 + outputBuffer.nSize1;
if ((pkt.buf = (char*)malloc(pkt.buflen)) == NULL)
{
loge("pkt.buf malloc failed in No. %d frame\n", testNumber);
break;
}
memcpy(pkt.buf, outputBuffer.pData0, outputBuffer.nSize0);
if (outputBuffer.nSize1 > 0)
{
memcpy((char*)pkt.buf + outputBuffer.nSize0, outputBuffer.pData1, outputBuffer.nSize1);
}
pkt.pts = outputBuffer.nPts / 1000; // us should be change to ms
pkt.duration = 1.0/ media_info.video.nFrameRate * 1000;
pkt.type = 0;
pkt.streamIndex = 0;
mux_time1 = GetNowUs();
result = CdxMuxerWritePacket(mux, &pkt);
mux_time2 = GetNowUs();
mux_time3 += mux_time2 - mux_time1;
if (result)
{
loge("CdxMuxerWritePacket() failed in No. %d frame\n", testNumber);
break;
}
free(pkt.buf);
pkt.buf = NULL;
/*************************** Write Packet **********************************/
if (encode_param.write_unmux)
{
fwrite(outputBuffer.pData0, 1, outputBuffer.nSize0, out_file);
if(outputBuffer.nSize1)
{
fwrite(outputBuffer.pData1, 1, outputBuffer.nSize1, out_file);
}
}
FreeOneBitStreamFrame(pVideoEnc, &outputBuffer);
if(h264Param.nCodingMode==VENC_FIELD_CODING && encode_param.encode_format==VENC_CODEC_H264)
{
GetOneBitstreamFrame(pVideoEnc, &outputBuffer);
if (encode_param.write_unmux)
{
fwrite(outputBuffer.pData0, 1, outputBuffer.nSize0, out_file);
if(outputBuffer.nSize1)
{
fwrite(outputBuffer.pData1, 1, outputBuffer.nSize1, out_file);
}
}
FreeOneBitStreamFrame(pVideoEnc, &outputBuffer);
}
testNumber++;
}
total_time2 = GetNowUs();
total_time3 += total_time2 - total_time1;
result = CdxMuxerWriteTrailer(mux);
if (result)
{
loge("CdxMuxerWriteTrailer() failed\n");
}
printf("the average encode time is %lldus...\n",time3/testNumber);
printf("the average mux time is %lldus...\n",mux_time3/testNumber);
printf("the average toatal time is %lldus...\n",total_time3/testNumber);
out:
printf("output file is saved: %s\n", (encode_param.stream_file + 8));
if (encode_param.write_unmux)
{
printf("output file is saved: %s\n", encode_param.output_file);
fclose(out_file);
out_file = NULL;
}
fclose(in_file);
in_file = NULL;
if(uv_tmp_buffer)
{
free(uv_tmp_buffer);
}
result = VideoEncUnInit(pVideoEnc);
if( result )
{
loge("VideoEncUnInit error result=%d...\n",result);
}
VideoEncDestroy(pVideoEnc);
pVideoEnc = NULL;
CdxMuxerClose(mux);
if (writer)
{
MWClose(writer);
CdxWriterDestroy(writer);
writer = NULL;
mw = NULL;
}
#if FS_WRITER
if (fs_cache_mem.mp_cache)
{
free(fs_cache_mem.mp_cache);
fs_cache_mem.mp_cache = NULL;
}
#endif
if(baseConfig.memops)
{
CdcMemClose(baseConfig.memops);
baseConfig.memops = NULL;
}
return 0;
}