色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

基于ffmpeg的推流示例

嵌入式技術 ? 來源:嵌入式技術 ? 作者:嵌入式技術 ? 2022-09-29 14:30 ? 次閱讀

基于ffmpeg的推流示例

? 流媒體(streaming media)是指將一連串的媒體數據壓縮后,經過網上分段發送數據,在網上即時傳輸影音以供觀賞的一種技術與過程,此技術使得數據包得以像流水一樣發送;如果不使用此技術,就必須在使用前下載整個媒體文件。流式傳輸可傳送現場影音或預存于服務器上的影片,當觀看者在收看這些影音文件時,影音數據在送達觀看者的計算機后立即由特定播放軟件播放。

RTMP是Real Time Messaging Protocol(實時消息傳輸協議)的首字母縮寫。該協議基于TCP,是一個協議族,包括RTMP基本協議及RTMPT/RTMPS/RTMPE等多種變種。RTMP是一種設計用來進行實時數據通信網絡協議,主要用來在Flash/AIR平臺和支持RTMP協議的流媒體/交互服務器之間進行音視頻和數據通信。支持該協議的軟件包括Adobe Media Server/Ultrant Media Server/red5等。RTMP與HTTP一樣,都屬于TCP/IP四層模型的應用層。

RTMP 是一種基于 TCP 的、用于數據、音頻視頻傳輸的雙向通信協議。大部分具備行業標準的編碼器(如 encoding.com、Bitmovin、Harmonic 和 AWS Elemental 等)都能夠生產 RTMP 數據源。同樣,Twitch、YouTube、Facebook Live 等流媒體服務和 Dacast、Ant Media、Wowza 等直播平臺都能接收 RTMP 推流。
利用FFMPEG進行視頻編碼,通過RTMP將視頻流推送到nginx服務端。FFMPEG推流過程和視頻編碼過程是一樣的,視頻流本身也是屬于文件的一種,所以只需要正常實現視頻編碼即可。

FFMPEG視頻編碼示例:https://blog.csdn.net/weixin_44453694/article/details/123885112
nginx服務器搭建示例:https://blog.csdn.net/weixin_44453694/article/details/126511614
FFMPEG視頻解碼示例:https://blog.csdn.net/weixin_44453694/article/details/127098011

  • 推流示例
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include "libavcodec/avcodec.h"
#include "libavformat/avformat.h"
#include "libswscale/swscale.h"
#include 
#include 
#include 

#include 
#include 
#include 
#include 
#include 

#include 
#include 
#include 
#define VIDEO_DEV "/dev/video0"
char file_name[100]="rtmp://124.21.108.66:8088/live/ashui";//視頻編碼文件名
#define STREAM_FRAME_RATE 25 /*每秒25幀*/
#define STREAM_PIX_FMT AV_PIX_FMT_YUV420P /*圖像格式yuv420p*/
#define STREAM_DURATION 10.0 /*錄制時間10s*/
pthread_mutex_t fastmutex = PTHREAD_MUTEX_INITIALIZER;//互斥鎖
pthread_cond_t cond = PTHREAD_COND_INITIALIZER;//條件變量
int width;
int height;
int size;
int mp4_decode_stat=0;
unsigned char *rgb_buff=NULL;
unsigned char video_flag=1;
void *Video_CollectImage(void *arg);
void *Video_savemp4(void*arg);
typedef enum
{
	false=0,
	true,
}bool;
typedef struct OutputStream
{
	AVStream *st;
	AVCodecContext *enc;
	int64_t next_pts;/*將生成的下一幀的pts*/
	AVFrame *frame;/*保存編解碼數據*/
	AVFrame *tmp_frame;
	struct SwsContext *sws_ctx;
	struct SwrContext *swr_ctx;
}OutputStream;
typedef struct IntputDev
{
	AVCodecContext *pcodecCtx;
	AVCodec *pCodec;
	AVFormatContext *v_ifmtCtx;
	int videoindex;//視頻幀ID
	struct SwsContext *img_convert_ctx;
	AVPacket *in_packet;
	AVFrame *pFrame,*pFrameYUV;
}IntputDev;
IntputDev video_input={0};//視頻輸入流
/*添加一個輸出流*/
static void add_stream(OutputStream *ost, AVFormatContext *oc, AVCodec **codec, enum AVCodecID codec_id)
{
	AVCodecContext *c;
	int i;
	/*查找編碼器*/
	*codec=avcodec_find_encoder(codec_id);
	if(*codec==NULL)
	{
		printf("Could not find encoder for ' %s' n",avcodec_get_name(codec_id));
		exit(1);
	}
	/*向媒體文件添加新流。*/
	ost->st=avformat_new_stream(oc,NULL);
	if(ost->st==NULL)
	{
		printf("Could not allocate stream n");
		exit(1);
	}
	ost->st->id=oc->nb_streams-1;
	/*分配AvcodeContext并將其字段設置為默認值*/
	c=avcodec_alloc_context3(*codec);
	if(c==NULL)
	{
		printf("avcodec_alloc_context3 failed n");
	}
	ost->enc=c;		
	switch((*codec)->type)
	{
		case AVMEDIA_TYPE_AUDIO:
			
			break;	
		case AVMEDIA_TYPE_VIDEO:/*視頻流*/
		
			c->codec_id=codec_id;
			c->bit_rate=2500000;//比特率
			/*分辨率必須是2的倍數。*/
			c->width=width;
			c->height=height;
			/*時基:這是時間的基本單位(秒)
				其中幀時間戳被表示。對于固定fps內容,時基應為1/幀率,時間戳增量應與1相同*/
			ost->st->time_base=(AVRational){1,STREAM_FRAME_RATE};
			c->time_base=ost->st->time_base;
			c->gop_size=12;/*最多每12幀發射一幀*/
			c->pix_fmt=STREAM_PIX_FMT;/*圖像格式*/
			if(c->codec_id == AV_CODEC_ID_MPEG2VIDEO)
			{
				/*
				為了測試,我們還添加了B幀
				*/
				c->max_b_frames=2;
			}
			if(c->codec_id == AV_CODEC_ID_MPEG1VIDEO)
			{
				/*
				需要避免使用某些系數溢出的宏塊。
				這種情況不會發生在普通視頻中,只會發生在這里
				色度平面的運動與亮度平面不匹配。*/
				c->mb_decision=2;
			}
			break;
		default:
			break;
	}
	/*有些格式希望流頭是分開的。*/
	if(oc->oformat->flags & AVFMT_GLOBALHEADER)
	{
		c->flags|=AV_CODEC_FLAG_GLOBAL_HEADER;
	}
}
/*視頻輸出*/
static AVFrame *alloc_picture(enum AVPixelFormat pix_fmt, int width, int height)
{
	AVFrame *picture;
	int ret;
	/*
		分配AVFrame并將其字段設置為默認值。結果
		必須使用av_frame_free()釋放結構。
	*/
	picture=av_frame_alloc();
	if(picture==NULL)
	{
		return  NULL;
	}
	picture->format=pix_fmt;
	picture->width=width;
	picture->height=height;
	/*為幀數據分配緩沖區*/
	ret=av_frame_get_buffer(picture,32);/*緩沖區以32位對齊*/
	if(ret<0)
	{
		printf("Could not allocate frame datan");/*無法分配幀數據*/
		exit(1);
	}
	return picture;
}

static void open_video(AVFormatContext *oc, AVCodec *codec, OutputStream *ost, AVDictionary *opt_arg)
{
	int ret;
	AVCodecContext *c=ost->enc;
	AVDictionary *opt=NULL;
	av_dict_copy(&opt,opt_arg, 0);
	/*初始化AvcodeContext以使用給定的AVCodec*/
	ret=avcodec_open2(c, codec,&opt);
	/*釋放為AVDictionary結構分配的所有內存*以及所有鍵和值。*/
	av_dict_free(&opt);
	if(ret<0)
	{
		printf("could not open video codec :%sn",av_err2str(ret));//無法打開視頻編解碼器
		exit(1);
	}
	/*視頻輸出*/
	ost->frame=alloc_picture(AV_PIX_FMT_YUV420P,c->width,c->height);
	if(ost->frame==NULL)
	{
		printf("could not allocate video framen");/*無法分配視頻幀*/
		exit(1);
	}
	printf("ost->frame alloc success fmt=%d w=%d h=%dn",c->pix_fmt,c->width,c->height);
	ost->tmp_frame=NULL;
	if(c->pix_fmt!=AV_PIX_FMT_YUV420P)
	{
		ost->tmp_frame=alloc_picture(AV_PIX_FMT_YUV420P,c->width, c->height);/*視頻幀格式*/
		if(ost->tmp_frame==NULL)
		{
			printf("conld not allocate temporary picturen");/*無法分配臨時幀*/
			exit(1);
		}
	}
	/*根據提供的編解碼器上下文中的值填充參數結構。*/
	ret=avcodec_parameters_from_context(ost->st->codecpar,c);
	if(ret)
	{
		printf("Could not copy the stream parametersn");/*無法復制流參數*/
		exit(1);
	}

}
static void log_packet(const AVFormatContext * fmt_ctx, const AVPacket * pkt)
{
	AVRational *time_base=&fmt_ctx->streams[pkt->stream_index]->time_base;
    printf("pts:%s pts_time:%s dts:%s dts_time:%s duration:%s duration_time:%s stream_index:%dn",
           									av_ts2str(pkt->pts), av_ts2timestr(pkt->pts, time_base),
           									av_ts2str(pkt->dts), av_ts2timestr(pkt->dts, time_base),
           									av_ts2str(pkt->duration), av_ts2timestr(pkt->duration, time_base),
           									pkt->stream_index);
										
}

/*寫入數據*/
static int write_frame(AVFormatContext *fmt_ctx, const AVRational *time_base, AVStream *st, AVPacket *pkt)
{
	/*數據包中的有效時間字段(時間戳/持續時間)從一個時基轉換為另一個時基。*/
	av_packet_rescale_ts(pkt,*time_base,st->time_base);
	pkt->stream_index=st->index;
	/*打印信息*/
	//log_packet(fmt_ctx, pkt);
	return av_interleaved_write_frame(fmt_ctx, pkt);/*將數據包寫入輸出媒體文件,確保正確的交織。*/
}
static int write_video_frame(AVFormatContext * oc, OutputStream * ost,AVFrame *frame)
{
	int ret;
	AVCodecContext *c;
	int got_packet=0;
	AVPacket pkt={0};
	if(frame==(void *)-1)return 1;
	c=ost->enc;
	/*使用默認值初始化數據包*/
	av_init_packet(&pkt);
	
	/*對一幀視頻進行編碼。*/
	ret=avcodec_encode_video2(c,&pkt,frame,&got_packet);
	if(ret<0)
	{
		printf("Error encoding video frame:%sn",av_err2str(ret));//編碼視頻流錯誤
		exit(1);
	}
	/*
	printf("--------vidoe pkt.pts=%sn",av_ts2str(pkt.pts));
	printf("------st.num=%d st.den=%d codec.num=%d codec.den=%d-------n",ost->st->time_base.num,
															  ost->st->time_base.den,
															  c->time_base.num,
															  c->time_base.den);
	*/
	if(got_packet)
	{

		ret=write_frame(oc,&c->time_base,ost->st,&pkt);/*寫入流數據*/
	}
	else
	{
		ret=0;
	}
	if(ret<0)
	{
		printf("Error while writing video frame:%sn",av_err2str(ret));/*寫入流出錯*/
		exit(1);
	}
	return (frame||got_packet)?0:1;
}
static AVFrame *get_video_frame(OutputStream *ost,IntputDev* input, int *got_pic)
{
	
	int ret,got_picture;
	AVCodecContext *c=ost->enc;
	AVFrame *ret_frame=NULL;
	/*在各自的時基中比較兩個時間戳。*/
	if(av_compare_ts(ost->next_pts,c->time_base,STREAM_DURATION, (AVRational){1,1})>=0)
	{
		//return  (void*)-1;
	}
	/*確保幀數據可寫,盡可能避免數據復制。*/
	if(av_frame_make_writable(ost->frame)<0)
	{
		exit(1);
	}
	/*此函數返回文件中存儲的內容,并且不驗證是否存在解碼器的有效幀。*/
	if(av_read_frame(input->v_ifmtCtx,input->in_packet)>=0)
	{
		if(input->in_packet->stream_index == input->videoindex)
		{
			/*解碼一幀視頻數據。輸入一個壓縮編碼的結構體AVPacket,輸出一個解碼后的結構體AVFrame*/
			ret=avcodec_decode_video2(input->pcodecCtx, input->pFrame,&got_picture,input->in_packet);
			*got_pic=got_picture;
			if(ret<0)
			{
				printf("Decode Error.n");
				av_packet_unref(input->in_packet);
				return NULL;
			}
			if(got_picture)
			{
				sws_scale(input->img_convert_ctx, (const unsigned char * const *)input->pFrame->data,input->pFrame->linesize,0,input->pcodecCtx->height,input->pFrameYUV->data,input->pFrameYUV->linesize);
				sws_scale(input->img_convert_ctx, (const unsigned char * const *)input->pFrame->data,input->pFrame->linesize,0,input->pcodecCtx->height,ost->frame->data,ost->frame->linesize);

				pthread_mutex_lock(&fastmutex);//互斥鎖上鎖
				memcpy(rgb_buff,input->pFrameYUV->data[0],size);
				pthread_cond_broadcast(&cond);//廣播喚醒所有線程
				pthread_mutex_unlock(&fastmutex);//互斥鎖解鎖
				ost->frame->pts=ost->next_pts++;

				//水印添加處理
				//frame->frame->format=AV_PIX_FMT_YUV420P;
				//AVFrame *frame_out=av_frame_alloc();
				//unsigned char *frame_buffer_out;
				//frame_buffer_out=(unsigned char *)av_malloc(size);
				//av_image_fill_arrays(frame_out->data,frame_out->linesize,frame_buffer_out,AV_PIX_FMT_YUV420P,width,height,32);
				//添加水印,調用libavfilter庫實現
				//time_t sec;
				//sec=time(NULL);
				//struct tm* today = localtime(&sec);
				//char sys_time[64];
				//strftime(sys_time, sizeof(sys_time), "%Y/%m/%d %H:%M:%S", today);  
				//waterMark(ost->frame,frame_out,width,height,sys_time);
				//yuv420p,y表示亮度,uv表示像素顏色
				//ost->frame=frame_out;
				//ost->frame->pts=ost->next_pts++;

				ret_frame=ost->frame;
			}
		}
		av_packet_unref(input->in_packet);
	}
	return ret_frame;
}
static void close_stream(AVFormatContext * oc, OutputStream * ost)
{
	avcodec_free_context(&ost->enc);
	av_frame_free(&ost->frame);
	av_frame_free(&ost->tmp_frame);
	sws_freeContext(ost->sws_ctx);
	swr_free(&ost->swr_ctx);
}

int main()
{
	/*創建攝像頭采集線程*/
	pthread_t pthid[2];
    pthread_create(&pthid[0],NULL,Video_CollectImage, NULL);
	pthread_detach(pthid[0]);/*設置分離屬性*/
	sleep(1);
	while(1)
	{
		if(width!=0 && height!=0 && size!=0)break;
		if(video_flag==0)return 0;
	}
	printf("image:%d * %d,%dn",width,height,size);
	unsigned char *rgb_data=malloc(size);
	/*創建mp4視頻編碼線程*/
	pthread_create(&pthid[1],NULL,Video_savemp4, NULL);
	pthread_detach(pthid[1]);/*設置分離屬性*/
	int count=0;
	mp4_decode_stat=1;
	pause();
    pthread_mutex_destroy(&fastmutex);/*銷毀互斥鎖*/
    pthread_cond_destroy(&cond);/*銷毀條件變量*/
	free(rgb_buff);
	free(rgb_data);
	return 0;
}

void *Video_CollectImage(void *arg)
{
	int res=0;
	AVFrame *Input_pFrame=NULL;
	AVFrame *Output_pFrame=NULL;
	printf("pth:%sn",avcodec_configuration());
	/*注冊設備*/
	avdevice_register_all();
	/*查找輸入格式*/
	AVInputFormat *ifmt=av_find_input_format("video4linux2");
	if(ifmt==NULL)
	{
		printf("av_find_input_format failedn");
		video_flag=0;
		return 0;
	}
	/*打開輸入流并讀取頭部信息*/
	AVFormatContext *ps=NULL;
	//分配一個AVFormatContext。
	ps=avformat_alloc_context();	
	res=avformat_open_input(&ps,VIDEO_DEV,ifmt,NULL);
	if(res)
	{
		printf("open input failedn");
		video_flag=0;
		return 0;
	}
	/*查找流信息*/
	res=avformat_find_stream_info(ps,NULL);
	if(res)
	{
		printf("find stream failedn");
		video_flag=0;
		return 0;
	}
	/*打印有關輸入或輸出格式信息*/
	av_dump_format(ps, 0, "video4linux2", 0);
	/*尋找視頻流*/
	int videostream=-1;
	videostream=av_find_best_stream(ps,AVMEDIA_TYPE_VIDEO,-1,-1,NULL,0);
	printf("videostram=%dn",videostream);
	/*尋找編解碼器*/
	AVCodec *video_avcodec=NULL;/*保存解碼器信息*/
	AVStream *stream = ps->streams[videostream];
	AVCodecContext *context=stream->codec;
	video_avcodec=avcodec_find_decoder(context->codec_id);
	if(video_avcodec==NULL)
	{
		printf("find video decodec failedn");
		video_flag=0;
		return 0;
	}	
	/*初始化音視頻解碼器*/
	res=avcodec_open2(context,video_avcodec,NULL);
	if(res)
	{
		printf("avcodec_open2 failedn");
		video_flag=0;
		return 0;
	}	
	AVPacket *packet=av_malloc(sizeof(AVPacket));/*分配包*/
	AVFrame *frame=av_frame_alloc();/*分配視頻幀*/
	AVFrame *frameyuv=av_frame_alloc();/*申請YUV空間*/
	/*分配空間,進行圖像轉換*/
	width=context->width;
	height=context->height;
	int fmt=context->pix_fmt;/*流格式*/
	size=av_image_get_buffer_size(AV_PIX_FMT_YUV420P,width,height,16);
	unsigned char *buff=NULL;
	printf("w=%d,h=%d,size=%dn",width,height,size);	
	buff=av_malloc(size);
	rgb_buff=malloc(size);//保存RGB顏色數據
	/*存儲一幀圖像數據*/
	av_image_fill_arrays(frameyuv->data,frameyuv->linesize,buff,AV_PIX_FMT_YUV420P,width,height, 16);	
	/*轉換上下文,使用sws_scale()執行縮放/轉換操作。*/
	struct SwsContext *swsctx=sws_getContext(width,height, fmt,width,height, AV_PIX_FMT_YUV420P,SWS_BICUBIC,NULL,NULL,NULL);
	/*視頻輸入流信息*/
	video_input.img_convert_ctx=swsctx;//格式轉換上下文
	video_input.in_packet=packet;//數據包
	video_input.pcodecCtx=context;
	video_input.pCodec=video_avcodec;/*保存解碼器信息*/
	video_input.v_ifmtCtx=ps;//輸入流并讀取頭部信息
	video_input.videoindex=videostream;/*視頻流*/
	video_input.pFrame=frame;/*視頻幀*/
	video_input.pFrameYUV=frameyuv;/*申請YUV空間*/

	//水印添加處理
	frameyuv->width=width;
	frameyuv->height=height;
	frameyuv->format=AV_PIX_FMT_YUV420P;
	AVFrame *frame_out=av_frame_alloc();
	unsigned char *frame_buffer_out;
	frame_buffer_out=(unsigned char *)av_malloc(size);
	av_image_fill_arrays(frame_out->data,frame_out->linesize,frame_buffer_out,AV_PIX_FMT_YUV420P,width,height,16);
	/*讀幀*/
	char *p=NULL;

	int go=0;
	int Framecount=0;
	time_t sec,sec2;
	char sys_time[64]; 
	while(video_flag)
	{
		if(!mp4_decode_stat)
		{
			res=av_read_frame(ps,packet);//讀取數據
			if(res>=0)
			{
				if(packet->stream_index == AVMEDIA_TYPE_VIDEO)//視頻流
				{
					/*解碼一幀視頻數據。輸入一個壓縮編碼的結構體AVPacket,輸出一個解碼后的結構體AVFrame*/
					res=avcodec_decode_video2(ps->streams[videostream]->codec,frame,&go,packet);
					if(res<0)
					{
						printf("avcodec_decode_video2 failedn");
						break;
					}
					if(go)
					{
						/*轉換像素的函數*/
						sws_scale(swsctx,(const uint8_t * const*)frame->data,frame->linesize,0,context->height,frameyuv->data,frameyuv->linesize);
						
						//添加水印,調用libavfilter庫實現
						
						sec=time(NULL);
						if(sec!=sec2)
						{
							struct tm* today = localtime(&sec);
							strftime(sys_time, sizeof(sys_time), "%Y/%m/%d %H:%M:%S", today);  
						}
						waterMark(frameyuv,frame_out,width,height,sys_time);
						//yuv420p,y表示亮度,uv表示像素顏色
						p=frame_buffer_out;
						memcpy(p,frame_out->data[0],frame_out->height*frame_out->width);//y,占用空間w*h
						p+=frame_out->width*frame_out->height;
						memcpy(p,frame_out->data[1],frame_out->height/2*frame_out->width/2);//u,占用空間(w/2)*(h/2)
						p+=frame_out->height/2*frame_out->width/2;
						memcpy(p,frame_out->data[2],frame_out->height/2*frame_out->width/2);//v,占用空間(w/2)*(h/2)
						p+=frame_out->height/2*frame_out->width/2;
						pthread_mutex_lock(&fastmutex);//互斥鎖上鎖
						memcpy(rgb_buff,frame_buffer_out,size);
						pthread_cond_broadcast(&cond);//廣播喚醒所有線程
						pthread_mutex_unlock(&fastmutex);//互斥鎖解鎖
					}
				}
			}
		}
	}
	sws_freeContext(swsctx);/*釋放上下文*/
	av_frame_free(&frameyuv);/*釋放YUV空間*/
	av_packet_unref(packet);/*釋放包*/
	av_frame_free(&frame);/*釋放視頻幀*/
	avformat_close_input(&ps);/*關閉流*/

	sws_freeContext(video_input.img_convert_ctx);
	avcodec_close(video_input.pcodecCtx);
	av_free(video_input.pFrameYUV);
	av_free(video_input.pFrame);
	avformat_close_input(&video_input.v_ifmtCtx);
	video_flag=0;
	pthread_exit(NULL);	
}
/*MP4格式數據保存*/
void *Video_savemp4(void*arg)
{
	while(1)
	{
		if(mp4_decode_stat)
		{
			int res;
			AVFormatContext *oc=NULL;
			AVDictionary *opt=NULL;
			/* 創建的AVFormatContext結構體。*/
			avformat_alloc_output_context2(&oc,NULL,"flv",NULL);//通過文件名創建
			if(oc==NULL)
			{
				printf("為輸出格式分配AVFormatContext失敗n");
				avformat_alloc_output_context2(&oc,NULL,"flv",NULL);//通過文件名創建
				return 0;
			}
			if(oc==NULL)return (void*)1;
			/*輸出流信息*/
			AVOutputFormat *ofmt=oc->oformat;
			printf("ofmt->video_codec=%dn",ofmt->video_codec);
			int have_video=1;
			int encode_video=0;
			OutputStream video_st={0};
			if(ofmt->video_codec !=AV_CODEC_ID_NONE)
			{
				/*添加一個輸出流*/
				add_stream(&video_st,oc,&video_input.pCodec,ofmt->video_codec);
				have_video=1;
				encode_video=1;
			}
			printf("w=%d,h=%d,size=%dn",width,height,size);
			/*視頻幀處理*/
			if(have_video)open_video(oc,video_input.pCodec,&video_st,opt);
			printf("打開輸出文件成功rn");
			/*打印有關輸入或輸出格式信息*/
			printf("file_name=%sn",file_name);
			av_dump_format(oc, 0,file_name,1);
			if(!(ofmt->flags & AVFMT_NOFILE))
			{
				/*打開輸出文件,成功之后創建的AVFormatContext結構體*/
				res=avio_open(&oc->pb,file_name,AVIO_FLAG_WRITE);
				if(res<0)
				{
					printf("%s open failed :%sn",file_name,av_err2str(res));
					return  (void*)1;
				}
			}
			/*寫入流數據頭*/
			res=avformat_write_header(oc,&opt);
			if(res<0)
			{
				printf("open output faile:%sn",av_err2str(res));
				return  (void*)1;
			}
			if(res<0)
			{
				printf("open output faile:%sn",av_err2str(res));
				return  (void*)1;
			}
			int got_pic;
			while(encode_video)
			{
				/*獲取流數據*/
				AVFrame *frame=get_video_frame(&video_st,&video_input,&got_pic);
				if(!got_pic || frame==NULL)
				{
					usleep(100);
					continue;
				}
				encode_video=!write_video_frame(oc,&video_st,frame);
			}
			/*將流數據寫入輸出媒體文件并釋放文件私有數據。*/
			av_write_trailer(oc);
			/*關閉AVIOContext*s訪問的資源,釋放它,并將指向它的指針設置為NULL*/
			if(!(ofmt->flags & AVFMT_NOFILE))avio_closep(&oc->pb);
			/*釋放AVFormatContext及其所有流。*/
			avformat_free_context(oc);
			/*關閉流*/
			if(have_video)close_stream(oc, &video_st);
			mp4_decode_stat=0;
		}
	}
}
poYBAGM1O4CAP4w9AAghhM7d2QE062.png#pic_center

審核編輯:湯梓紅
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 服務器
    +關注

    關注

    12

    文章

    9287

    瀏覽量

    85847
  • ffmpeg
    +關注

    關注

    0

    文章

    46

    瀏覽量

    7408
  • rtmp
    +關注

    關注

    0

    文章

    7

    瀏覽量

    1614
收藏 人收藏

    評論

    相關推薦

    如何移植FFmpeg

    ?FFmpeg是一款專門用于處理數字音頻和視頻,支持錄制、轉換,并能將這些媒體內容轉化為實時數據的開源計算機程序。它遵循LGPL或GPL許可協議,為用戶提供了涵蓋音視頻錄制、格式轉換及流媒體分發
    的頭像 發表于 06-07 15:28 ?1721次閱讀
    如何移植<b class='flag-5'>FFmpeg</b>

    【米爾-芯馳D9360商顯板試用評測】使用ffmpeg實現遠程視頻監控

    】 通過ffmpeg實現采集視頻,并給服務器,就可以實現遠程的視頻查看。也可以定義在服務器中對他進行傳發、錄像。
    發表于 04-11 13:07

    樹莓派 ffmpeg硬編碼問題

    想寫個監控軟件玩,同時練習下ffmpeg編程。推送端使用了ffmpeg采集視頻并推送rtmp,利用nginx搭建RTMP視頻直播,現在功能都已經實現了,但推送端使用的是軟編碼,cpu使用率較高。現在改成硬編碼。請問相關資料。或
    發表于 08-09 17:04

    視頻直播常見問題與解決辦法匯總【系列三—直播

    摘要: 直播初始化設置 要進行直播,您必須開啟直播加速 > 配置直播域名 > 直播
    發表于 03-22 18:29

    樹莓派視頻(部分轉)

    一個可以24小時運行著的播放腳本,比如下面rtmp="地址/碼"while truedo for i in $(seq 0 188) do
    發表于 05-04 21:34

    【HarmonyOS HiSpark AI Camera試用連載之 】萌新闖關之直播原理講解_序章

    文章先介紹一下直播的原理和開發環境的準備,話不多說,直接開整。使用到的工具介紹:1.FFmpeg,2.嗶哩嗶哩直播姬1.FFmpeg是一套可以用來記錄、轉換數字音頻、視頻,并能將其
    發表于 12-31 01:27

    【米爾MYD-YT507開發板試用體驗】ubuntu下FFmpeg攝像頭流體驗

    是否正常運行就是在瀏覽器中輸入本機IP。出現這樣提示就說明可以進行操作了。在開發板命令窗口輸入如下命令:ffmpeg -i /dev/video0 -vcodec libx264 -acodec
    發表于 07-21 20:11

    【觸覺智能 Purple Pi開發板試用】視頻采集編碼與開發

    編碼開發在完成視頻采集編碼之后,那么就可以再結合網絡進一步將編碼的數據,通過視頻流媒體協議發送出去,形成實時視頻,這樣就可以在線觀看直播了。由于ffmpeg可以把視頻文件進行網絡
    發表于 10-11 01:40

    rv1126 rtsp延時問題

    目前需要使用1126 rtsp,有沒有什么方向能夠減少直播過程中的延時
    發表于 06-08 16:40

    如何查看rtsp服務是否實時

    查看 rtsp 服務是否實時
    發表于 09-18 07:36

    采用TCP傳輸碼的時候如果碼服務器停止ffmpeg阻塞在av_read_frame如何解決?

    采用TCP傳輸碼的時候如果碼服務器停止ffmpeg阻塞在av_read_frame
    發表于 09-19 07:22

    rtsp使用ffmpeg和opencv可以正常解碼,但是使用sail.Decoder無法正常解碼是怎么回事?

    rtsp使用ffmpeg和opencv可以正常解碼,但是使用sail.Decoder無法正常解碼
    發表于 09-19 06:03

    【悟空派H3開發板免費體驗】04 流體驗

    地址和串流密鑰,組合在一起,就是你的地址了。 回到開發板,輸入下面的命令 ffmpeg -f v4l2 -i /dev/video0 -c:v libx264 -preset ultrafast
    發表于 10-09 23:01

    介紹FFmpeg是什么?與關于FFmpeg的問題回答

    FFmpeg誕生于十幾年前,最初是作為一個MPlayer播放器的一個子項目出現。因為當時的播放器有需要支持各種各樣解碼的需求, 其中有一位Mplayer的開發者看到了這樣的需求,于是編寫了FFmpeg
    的頭像 發表于 07-28 09:10 ?2.4w次閱讀
    介紹<b class='flag-5'>FFmpeg</b>是什么?與關于<b class='flag-5'>FFmpeg</b>的問題回答

    音視頻圖像處理神器FFmpeg有用的命令示例

    ffmpeg由Fabrice Bellard創建,他與QEMU和QuickJS ( JSLinux背后的引擎)的創建者相同。
    的頭像 發表于 05-04 14:59 ?1044次閱讀
    音視頻圖像處理神器<b class='flag-5'>FFmpeg</b>有用的命令<b class='flag-5'>示例</b>
    主站蜘蛛池模板: 日本 稀土矿| 极品美女穴| 耻辱诊察室1一4集动漫在线观看| 国产成人精品视频频| 久久精品国产亚洲AV未满十八| 魔乳 堕乳漫画acg产卵| 无码内射成人免费喷射| 99精品久久久久久久| 九九热在线视频| 少妇两个奶头喷出奶水了怎么办| 亚洲伊人精品| 国产精品99久久久久久AV蜜臀 | 一区二区三区福利视频| 俄罗斯乌克兰战争原因| 美女露出乳胸扒开尿口| 夜夜国产亚洲视频香蕉| 国产精品99久久免费黑人人妻| 欧美精品熟妇乱| 91精品国产色综合久久不| 国内精品日本久久久久影院| 日韩av片无码一区二区不卡电影| 538久久视频在线| 久久久久久88色偷偷| 亚洲免费无l码中文在线视频| 国产精品久久久久久久久久免费| 欧美午夜不卡在线观看| 99国产这里只有精品视频| 脔到她哭H粗话HWWW男男动漫| 优菈的乳液狂飙天堂W98| 黄色三级视频在线| 亚洲精品午夜VA久久成人 | 饥渴的护士自慰被发现| 色偷偷男人的天堂a v| 爱看吧孕妇网| 欧美激情精品久久久久久不卡| 13一18TV处流血TV| 久久免费精品一区二区| 孕妇bbwbbwbbwbbw超清| 久久九九免费| 18女下面流水不遮网站免费| 美女在线永久免费网站|