本來主要講述如何利用ffmpeg將輸入視頻流通過轉碼的方式轉成m3u8文件。如何通過http的方法將切邊推送給客戶端,不在本文中講述。
輸入視頻流可以是rtsp流,也可以是http,還可以是文件等等。轉碼的基本流程如下圖所示:
圖1. 生產hls視頻流
視頻流解復用可以獲得packet,對應的實現方法是av_read_frame。
下面給出代碼:
1. 初始化ffmpeg
void Init()
{
av_register_all();
avfilter_register_all();
avformat_network_init();
av_log_set_level(AV_LOG_ERROR);
}
初始化ffmepg是必須的,否則調用相關的ffmpeg會返回錯誤。
2. 打開視頻流
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.解復用
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
