原本主要講述如何利用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