FFMPEG(三) v4l2 数据编码H264

     系列相关博文:

            FFMPEG(一) 从V4L2捕获摄像头数据

            FFMPEG(二) v4l2 数据格式装换

            FFMPEG(三) v4l2 数据编码H264

    前面已经介绍了linux 系统 使用FFMPEG 库通过V4L2采集摄像头数据,并且输出不同的数据格式,接下来需要处理的就是将采集到的数据进行压缩编码。H264是现在使用最广的一个视频压缩库。下面的代码实现的就是将v4l2采集到的数据,格式转换后再进行H264编码的程序。

/*=============================================================================
 * #     FileName: read_device.c
 * #         Desc: use ffmpeg read a frame data from v4l2, and encode to H264
 * #       Author: licaibiao
 * #   LastChange: 2017-03-28 
 * =============================================================================*/
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <time.h>
#include <sys/time.h>
#include "avformat.h"
#include "avcodec.h"
#include "avdevice.h"
#include <libavutil/imgutils.h>  
#include <libswscale/swscale.h> 

#define	LOOP_NUM			1000
#define OUT_WIDTH			320
#define OUT_HEIGHT			240

char* input_name= "video4linux2";
char* file_name = "/dev/video0";

struct timeval  time_val;
float 	 time_start;
float 	 time_end;

float get_diff_time(struct timeval* start , int update)
{
	float dt;
	struct timeval now;
	gettimeofday(&now, NULL);
	dt = (float)(now.tv_sec  - start->tv_sec);
	dt += (float)(now.tv_usec - start->tv_usec) * 1e-6;
	
	if (update == 1) {
		start->tv_sec = now.tv_sec;
		start->tv_usec = now.tv_usec;
	}
	
	return dt;
}

int flush_encoder(AVFormatContext *fmt_ctx,unsigned int stream_index){
	int ret;
	int got_frame;
	AVPacket enc_pkt;
	if (!(fmt_ctx->streams[stream_index]->codec->codec->capabilities & CODEC_CAP_DELAY))
		return 0;

	while (1) {
		enc_pkt.data = NULL;
		enc_pkt.size = 0;
		av_init_packet(&enc_pkt);
		ret = avcodec_encode_video2 (fmt_ctx->streams[stream_index]->codec, &enc_pkt,
			NULL, &got_frame);
		av_frame_free(NULL);
		if (ret < 0)
			break;
		if (!got_frame){
			ret=0;
			break;
		}
		printf("Flush Encoder: Succeed to encode 1 frame!\tsize:%5d\n",enc_pkt.size);
		ret = av_write_frame(fmt_ctx, &enc_pkt);
		if (ret < 0)
			break;
	}
	return ret;
}


void captureFrame(void){
    AVFormatContext *fmtCtx = NULL;    
    AVInputFormat 	*inputFmt;
    AVPacket 		*packet;
	AVCodecContext	*pCodecCtx;
	AVCodec 		*pCodec; 
	struct SwsContext *sws_ctx;  
    FILE *fp; 
	int i;
	int ret;
	int videoindex;
	
	enum AVPixelFormat dst_pix_fmt = AV_PIX_FMT_YUV420P;
    const char *dst_size = NULL;  
    const char *src_size = NULL;  
    uint8_t *src_data[4];   
    uint8_t *dst_data[4];  
    int src_linesize[4];  
    int dst_linesize[4];  
    int src_bufsize;  
    int dst_bufsize;  
    int src_w ;  
    int src_h ;  
    int dst_w = OUT_WIDTH;   
    int dst_h = OUT_HEIGHT;
	
    inputFmt = av_find_input_format (input_name);    
   
    if (inputFmt == NULL)    {        
        printf("can not find_input_format\n");        
        return;    
    }    

    if (avformat_open_input ( &fmtCtx, file_name, inputFmt, NULL) < 0){
        printf("can not open_input_file\n");         return;    
    }
	
	av_dump_format(fmtCtx, 0, file_name, 0);

	videoindex= -1;
	for(i=0; i<fmtCtx->nb_streams; i++) 
		if(fmtCtx->streams[i]->codec->codec_type==AVMEDIA_TYPE_VIDEO){
			videoindex=i;
			break;
		}
	if(videoindex==-1){
		printf("Didn't find a video stream.\n");
		return -1;
	}

	pCodecCtx = fmtCtx->streams[videoindex]->codec;
	pCodec    = avcodec_find_decoder(pCodecCtx->codec_id); 

    printf("picture width   =  %d \n", pCodecCtx->width);
    printf("picture height  =  %d \n", pCodecCtx->height);
    printf("Pixel   Format  =  %d \n", pCodecCtx->pix_fmt);
		
	sws_ctx = sws_getContext( pCodecCtx->width, pCodecCtx->height, pCodecCtx->pix_fmt, dst_w, dst_h, dst_pix_fmt,  
                             SWS_BILINEAR, NULL, NULL, NULL); 

	src_bufsize = av_image_alloc(src_data, src_linesize, pCodecCtx->width, pCodecCtx->height, pCodecCtx->pix_fmt, 16);
	dst_bufsize = av_image_alloc(dst_data, dst_linesize, dst_w, dst_h, dst_pix_fmt, 1);
	
	packet = (AVPacket *)av_malloc(sizeof(AVPacket));  

	/* set out format */
	AVFormatContext		*outFormatCtx;
	AVOutputFormat		*outfmt;
	AVCodecContext		*outCodecCtx;  
	AVStream			*video_st;
	AVDictionary 		*param = 0;
	AVCodec				*outCodec;
	AVFrame				*outFrame;
	AVPacket 			outpkt;

	uint8_t *picture_buf; 
	char	*out_file  = "ds.h264";
	int 	picture_size;
	int 	y_size;
	int		got_picture;
	int 	loop = 0;

	outFormatCtx = avformat_alloc_context();
	outfmt = av_guess_format(NULL, out_file, NULL);  
	outFormatCtx->oformat = outfmt; 
	if (avio_open(&outFormatCtx->pb,out_file, AVIO_FLAG_READ_WRITE) < 0){  
        printf("Failed to open output file! \n");  
        return -1;  
    }
	video_st = avformat_new_stream(outFormatCtx, 0); 
	if (video_st==NULL){
		printf(" creat new stream err \n ");  
        return -1;  
    }  
	
	outCodecCtx = video_st->codec;
	outCodecCtx->codec_id = outfmt->video_codec;
	outCodecCtx->codec_type = AVMEDIA_TYPE_VIDEO;
	outCodecCtx->pix_fmt = AV_PIX_FMT_YUV420P;
	outCodecCtx->width = dst_w;    
    outCodecCtx->height = dst_h;  
    outCodecCtx->bit_rate = 2000000;    
    outCodecCtx->gop_size=10;  
  
    outCodecCtx->time_base.num = 1;    
    outCodecCtx->time_base.den = 25;   
    outCodecCtx->qmin = 10;  
    outCodecCtx->qmax = 51;  
	outCodecCtx->max_b_frames=3; 
	
	if(pCodecCtx->codec_id == AV_CODEC_ID_H264) { 
		av_dict_set(¶m,"preset", "faster", 0);
		//av_dict_set(¶m,"preset", "slow", 0);
		av_dict_set(¶m,"tune", "zerolatency", 0);
	}

	av_dump_format(outFormatCtx, 0, out_file, 1);
	
	outCodec = avcodec_find_encoder(outCodecCtx->codec_id);
    if (!outCodec){  
        printf("Can not find encoder! \n");  
        return -1;  
    } 

    if (avcodec_open2(outCodecCtx, outCodec, ¶m) < 0){  
        printf("Failed to open encoder! \n");  
        return -1;  
    }  

	outFrame = av_frame_alloc();
	picture_size = avpicture_get_size(outCodecCtx->pix_fmt, outCodecCtx->width, outCodecCtx->height);
	picture_buf = (uint8_t *)av_malloc(picture_size); 
	avpicture_fill((AVPicture *)outFrame, picture_buf, outCodecCtx->pix_fmt, outCodecCtx->width, outCodecCtx->height);
	outFrame->format = outCodecCtx->pix_fmt;
	outFrame->width  = outCodecCtx->width;
	outFrame->height = outCodecCtx->height;	
	avformat_write_header(outFormatCtx,NULL);
	av_new_packet(&outpkt,picture_size);
	y_size = outCodecCtx->width * outCodecCtx->height; 
	
	time_start = get_diff_time(&time_val, 1);
	while(loop++ < LOOP_NUM){
    	av_read_frame(fmtCtx, packet);
		memcpy(src_data[0], packet->data, packet->size);
		sws_scale(sws_ctx, src_data,  src_linesize, 0, pCodecCtx->height, dst_data, dst_linesize); 
    	

		outFrame->data[0] = dst_data[0];
		outFrame->data[1] = dst_data[0] + y_size;
		outFrame->data[2] = dst_data[0] + y_size*5/4;
		
		outFrame->pts=(loop -1)*(video_st->time_base.den)/((video_st->time_base.num)*25);
		ret = avcodec_encode_video2(outCodecCtx, &outpkt, outFrame, &got_picture);
		if(ret < 0)
		{
			printf("Failed to encode! \n");
			return -1;
		} 

		if(got_picture==1){
			outpkt.stream_index = video_st->index;
			ret = av_write_frame(outFormatCtx, &outpkt);
			av_free_packet(&outpkt); 
		}
	}
 
	time_end = get_diff_time(&time_val, 0);
	printf("\n\nencoder %d frame  spend time = %f \n\n",loop, time_end);
	
	ret = flush_encoder(outFormatCtx,0);
	if(ret < 0){
		printf("Flushing encoder failed\n");  
        return -1;
	}

	av_write_trailer(outFormatCtx);  
	if (video_st){  
        avcodec_close(video_st->codec);  
        av_free(outFrame);  
        av_free(picture_buf);  
    }  
    avio_close(outFormatCtx->pb);  
    avformat_free_context(outFormatCtx); 
    
    av_free_packet(packet);    
	av_freep(&dst_data[0]);
	sws_freeContext(sws_ctx);
    avformat_close_input(&fmtCtx);
 } 

int main(void){    
	av_register_all(); 
    avcodec_register_all();    
    avdevice_register_all();    
    captureFrame();    
    return 0;
}

    在我的设备中,X264编码消耗的时间太长。编码VGA(640*480)格式数据 编码1000 帧需要耗时76.771431秒,编码QVGA(320*240)格式还好只需要34.012341 秒。如果是需要编码高清的视频,那么编码的时间将会更长,视频播放的时候就是跟快进一样的效果。出现这个问题,可能是安装X264库的时候禁止了使用底层汇编引起的,也可能是X264软件编码的时间本来就会耗费这么长时间。这个待以后有时间在分析定位。

    工程代码可以到这里下载:FFMPEG  V4L2数据编码成H264格式

------------------------------------------2022.08.21------------------------------------------

该博客将停止更新 

新的文章内容和附件工程文件

请到 liwen01 博客首页信息查询

------------------------------------------2022.08.21------------------------------------------

  • 3
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 15
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 15
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

li_wen01

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值