FFmpeg:Example:HLS
HLS 저장 방법.
Options
-
-hls_playlist_type vod
- 이렇게 하면 모든 조각이 동일한 매니페스트에 기록되고 매니페스트에
#EXT-X-PLAYLIST-TYPE:VOD
태그가 추가됩니다.
-
-hls_list_size 0
- 기본값은 5 이며, 이는 출력 매니페스트에 기본적으로 최대 5개의 세그먼트가 포함됨을 의미합니다. 0으로 설정하면 ffmpeg가 모든 조각을 포함하도록 지시합니다.
-
-strftime 1
- 파일 이름에 로컬타임 기준으로 strftime 포맷이 적용된다.
- e.g.
ffmpeg -i in.nut -strftime 1 -hls_segment_filename 'file-%Y%m%d-%s.ts' out.m3u8
-
-hls_flags second_level_segment_index
- 세그먼트 파일명에
%d
를 사용하고 싶다면 적용할 수 있는 옵션. 주의할 점은 strftime 포맷이 적용된 후 다시 Index 가 추가되므로%%d
로 사용해야 한다. - e.g.
ffmpeg -i in.nut -strftime 1 -hls_flags second_level_segment_index -hls_segment_filename 'file-%Y%m%d-%%04d.ts' out.m3u8
-
-
-strftime_mkdir 1
-
-strftime 1
옵션과 함께 사용되며, 하위 디렉토리를 함께 생성한다. - e.g.
ffmpeg -i in.nut -strftime 1 -strftime_mkdir 1 -hls_segment_filename '%Y%m%d/file-%Y%m%d-%s.ts' out.m3u8
Code
/*
*
* Example created by muzie (mailto:[email protected])
*
* */
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
extern "C" {
#include <libavcodec/avcodec.h>
#include <libavformat/avformat.h>
#include <libavformat/avio.h>
#include <libavdevice/avdevice.h>
}
int
main (void)
{
avdevice_register_all();
av_register_all();
avcodec_register_all();
avformat_network_init();
// Initialize
int vidx = 0, aidx = 0; // Video, Audio Index
AVFormatContext* ctx = avformat_alloc_context();
AVDictionary *dicts = NULL;
AVFormatContext* oc = NULL;
avformat_alloc_output_context2(&oc, NULL, "hls", "playlist.m3u8"); // apple hls. If you just want to segment file use "segment"
int rc = av_dict_set(&dicts, "rtsp_transport", "tcp", 0); // default udp. Set tcp interleaved mode
if (rc < 0)
{
return EXIT_FAILURE;
}
/*
rc = av_dict_set(&dicts, "stimeout", "1 * 1000 * 1000", 0); // timeout option
if (rc < 0)
{
return -1;
}
*/
//open rtsp
if (avformat_open_input(&ctx, "rtsp://211.189.132.118/nbr-media/media.nmp?ch=T142",NULL, &dicts) != 0)
{
return EXIT_FAILURE;
}
// get context
if (avformat_find_stream_info(ctx, NULL) < 0)
{
return EXIT_FAILURE;
}
//search video stream , audio stream
for (int i = 0 ; i < ctx->nb_streams ; i++)
{
if (ctx->streams[i]->codec->codec_type == AVMEDIA_TYPE_VIDEO)
vidx = i;
if (ctx->streams[i]->codec->codec_type == AVMEDIA_TYPE_AUDIO)
aidx = i;
}
AVPacket packet;
//open output file
if (oc == NULL)
{
return EXIT_FAILURE;
}
AVStream* vstream = NULL;
AVStream* astream = NULL;
vstream = avformat_new_stream(oc, ctx->streams[vidx]->codec->codec);
astream = avformat_new_stream(oc, ctx->streams[aidx]->codec->codec);
avcodec_copy_context(vstream->codec, ctx->streams[vidx]->codec);
vstream->time_base = (AVRational){1, 90000};
vstream->sample_aspect_ratio = ctx->streams[vidx]->codec->sample_aspect_ratio;
avcodec_copy_context(astream->codec, ctx->streams[aidx]->codec);
astream->time_base = (AVRational){1, 16000};
int cnt = 0;
long long nGap = 0;
long long nGap2 = 1;
av_read_play(ctx);//play RTSP
int i = (1 << 4); // omit endlist
int j = (1 << 1); // delete segment.
// libavformat/hlsenc.c 's description shows that no longer available files will be deleted but it doesnt works as described.
av_opt_set(oc->priv_data, "hls_segment_filename", "file%03d.ts", 0);
av_opt_set_int(oc->priv_data, "hls_list_size", 3, 0);
av_opt_set_int(oc->priv_data, "hls_time", 3, 0);
av_opt_set_int(oc->priv_data, "hls_flags", i|j, 0);
avformat_write_header(oc, NULL);
while (true)
{
av_init_packet(&packet);
int nRecvPacket = av_read_frame(ctx, &packet);
if (packet.stream_index == vidx) // video frame
{
vstream->id = vidx;
packet.pts = av_rescale_q(nGap, (AVRational){1, 10000}, oc->streams[packet.stream_index]->time_base);
nGap += 333;
cnt++;
}
else if (packet.stream_index == aidx) // audio frame
{
astream->id = aidx;
packet.pts = av_rescale_q(nGap2, (AVRational){1, 10000}, oc->streams[packet.stream_index]->time_base);
nGap2 += 666;
}
packet.dts = packet.pts;// generally, dts is same as pts. it only differ when the stream has b-frame
av_write_frame(oc,&packet);
av_packet_unref(&packet);
if (cnt > 3000) //
break;
}
av_read_pause(ctx);
av_write_trailer(oc);
avformat_free_context(oc);
av_dict_free(&dicts);
return (EXIT_SUCCESS);
}