ffmpeg 發佈hls流

     原本主要講述如何利用ffmpeg將輸入視頻流經過轉碼的方式轉成m3u8文件。如何經過http的方法將切邊推送給客戶端,不在本文中講述。this

輸入視頻流能夠是rtsp流,也能夠是http,還能夠是文件等等。轉碼的基本流程以下圖所示:spa

圖1. 生產hls視頻流code

視頻流解複用能夠得到packet,對應的實現方法是av_read_frame。 orm

       下面給出代碼:視頻

   1. 初始化ffmpegblog

void Init()
{
      av_register_all();
      avfilter_register_all();
      avformat_network_init();
      av_log_set_level(AV_LOG_ERROR);
}

 初始化ffmepg是必須的,不然調用相關的ffmpeg會返回錯誤。get

2. 打開視頻流input

  

int OpenInput(char *fileName)
{
	 context = avformat_alloc_context();
	 context->interrupt_callback.callback = interrupt_cb;
	int ret = avformat_open_input(&context, fileName, nullptr,nullptr);
	if(ret < 0)
	{
		return  ret;
	}
	ret = avformat_find_stream_info(context,nullptr);
	av_dump_format(context, 0, fileName, 0);
	if(ret >= 0) 
	{
		cout <<"open input stream successfully" << endl;
	}
	return ret;
}

 3.建立hls輸出上下文源碼

int OpenOutput(char *fileName)
{
    int ret = 0;
     ret  = avformat_alloc_output_context2(&outputContext, nullptr, "hls", fileName);
    if(ret < 0)
    {
        goto Error;
    }
    ret = avio_open2(&outputContext->pb, fileName, AVIO_FLAG_READ_WRITE,nullptr, nullptr);    
    if(ret < 0)
    {
        goto Error;
    }

    av_opt_set(outputContext->priv_data, "hls_time" ,"5" , AV_OPT_SEARCH_CHILDREN);
    //av_opt_set(outputContext->priv_data, "hls_list_size" ,"10" , AV_OPT_SEARCH_CHILDREN);
    av_opt_set(outputContext->priv_data, "hls_wrap" ,"5" , AV_OPT_SEARCH_CHILDREN);

    for(int i = 0; i < context->nb_streams; i++)
    {
        AVStream * stream = avformat_new_stream(outputContext, context->streams[i]->codec->codec);
        ret = avcodec_copy_context(stream->codec, context->streams[i]->codec);
        //stream->codec->codec_tag = 0;
        //stream->index = 0;
        if(ret < 0)
        {
            goto Error;
        }
    }
     av_dump_format(outputContext, 0, fileName, 1);
     ret = avformat_write_header(outputContext, nullptr);
    if(ret < 0)
    {
        goto Error;
    }
    if(ret >= 0)
    cout <<"open output stream successfully" << endl;
    return ret ;
Error:
    if(outputContext)
    {
        for(int i = 0; i < outputContext->nb_streams; i++)
        {
            avcodec_close(outputContext->streams[i]->codec);
        }
        avformat_close_input(&outputContext);
    }
    return ret ;
}

 4.解複用string

    

AVPacket *ReadPacketFromSource()
{
	std::shared_ptr<AVPacket> packet(static_cast<AVPacket*>(av_malloc(sizeof(AVPacket))), [&](AVPacket *p) { av_free_packet(p); av_freep(&p); });
	av_init_packet(packet.get());
	lastFrameRealtime = av_gettime();
	int ret = av_read_frame(context, packet.get());
	if(ret >= 0)
	{
		return packet.get();
	}
	else
	{
		return nullptr;
	}
}

 av_read_frame返回packet,packet是通過解複用獲得的裸碼流.

5. 寫packet到輸出context 

av_write_frame(outputContext, packet);

 demo

int _tmain(int argc, _TCHAR* argv[])
{
	string fileInput= "D:\\record\\langxi\\langxi.ts";
	string fileOutput="D:\\test\\file\\live\\wgg2\\test.m3u8";
	Init();
	if(OpenInput((char *)fileInput.c_str()) < 0)
	{
		cout << "Open file Input failed!" << endl;
		this_thread::sleep_for(chrono::seconds(10));
		return 0;
	}
	if(OpenOutput((char *)fileOutput.c_str()) < 0)
	{
		cout << "Open file Output failed!" << endl;
		this_thread::sleep_for(chrono::seconds(10));
		return 0;
	}
	auto timebase = av_q2d(context->streams[0]->time_base);
	int count = 0;
	auto in_stream = context->streams[0];
	auto out_stream = outputContext->streams[0];
	while(true)
	{
		AVPacket *packet = ReadPacketFromSource();
		if(packet)
		{
		   packet->pts = av_rescale_q_rnd(packet->pts, in_stream->time_base, out_stream->time_base, 
             AVRounding(AV_ROUND_NEAR_INF|AV_ROUND_PASS_MINMAX));         packet->dts = av_rescale_q_rnd(packet->dts, in_stream->time_base, out_stream->time_base, AVRounding(AV_ROUND_NEAR_INF|AV_ROUND_PASS_MINMAX));         packet->duration = av_rescale_q(packet->duration, in_stream->time_base, out_stream->time_base);         packet->pos = -1;    int ret = av_write_frame(outputContext, packet); } else { cout <<"write packet end!"<< endl; break; } } CloseInput(); CloseOutput(); cout <<"Transcode file end!" << endl; this_thread::sleep_for(chrono::hours(10)); return 0; }

  若有問題,加羣流媒體/Ffmpeg/音視頻 127903734交流,羣裏有demo源碼.

視頻下載地址:http://www.chungen90.com/?news_3/

 Demo下載地址: http://www.chungen90.com/?news_2

相關文章
相關標籤/搜索