|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157 |
- #if defined(_HAVE_FFMPEG_)
-
- extern "C"
- {
- #include "libavformat/avformat.h"
- }
- #include "hls-fmp4.h"
- #include "hls-m3u8.h"
- #include "hls-param.h"
- #include "mov-format.h"
- #include "mpeg-ps.h"
- #include <assert.h>
-
- static char s_packet[2 * 1024 * 1024];
-
- static void ffmpeg_init()
- {
- avformat_network_init();
- }
-
- static AVFormatContext* ffmpeg_open(const char* url)
- {
- int r;
- AVFormatContext* ic;
- AVDictionary* opt = NULL;
- ic = avformat_alloc_context();
- if (NULL == ic)
- {
- printf("%s(%s): avformat_alloc_context failed.\n", __FUNCTION__, url);
- return NULL;
- }
-
- //if (!av_dict_get(ff->opt, "scan_all_pmts", NULL, AV_DICT_MATCH_CASE)) {
- // av_dict_set(&ff->opt, "scan_all_pmts", "1", AV_DICT_DONT_OVERWRITE);
- // scan_all_pmts_set = 1;
- //}
-
- r = avformat_open_input(&ic, url, NULL, &opt);
- if (0 != r)
- {
- printf("%s: avformat_open_input(%s) => %d\n", __FUNCTION__, url, r);
- return NULL;
- }
-
- //if (scan_all_pmts_set)
- // av_dict_set(&format_opts, "scan_all_pmts", NULL, AV_DICT_MATCH_CASE);
-
- //ff->ic->probesize = 100 * 1024;
- //ff->ic->max_analyze_duration = 5 * AV_TIME_BASE;
-
- /* If not enough info to get the stream parameters, we decode the
- first frames to get it. (used in mpeg case for example) */
- r = avformat_find_stream_info(ic, NULL/*&opt*/);
- if (r < 0) {
- printf("%s(%s): could not find codec parameters\n", __FUNCTION__, url);
- return NULL;
- }
-
- av_dict_free(&opt);
- return ic;
- }
-
- static int hls_init_segment(hls_fmp4_t* hls, hls_m3u8_t* m3u)
- {
- int bytes = hls_fmp4_init_segment(hls, s_packet, sizeof(s_packet));
-
- FILE* fp = fopen("hls/0.mp4", "wb");
- fwrite(s_packet, 1, bytes, fp);
- fclose(fp);
-
- return hls_m3u8_set_x_map(m3u, "hls/0.mp4");
- }
-
- static int hls_segment(void* m3u8, const void* data, size_t bytes, int64_t /*pts*/, int64_t dts, int64_t duration)
- {
- static int i = 0;
- static char name[128] = { 0 };
- snprintf(name, sizeof(name) - 1, "hls/%d.mp4", ++i);
- FILE* fp = fopen(name, "wb");
- fwrite(data, 1, bytes, fp);
- fclose(fp);
-
- return hls_m3u8_add((hls_m3u8_t*)m3u8, name, dts, duration, 0);
- }
-
- void hls_segmenter_fmp4_test(const char* file)
- {
- ffmpeg_init();
-
- AVPacket pkt;
- memset(&pkt, 0, sizeof(pkt));
- //av_init_packet(&pkt);
-
- AVFormatContext* ic = ffmpeg_open(file);
- hls_m3u8_t* m3u = hls_m3u8_create(0, 7);
- hls_fmp4_t* hls = hls_fmp4_create(HLS_DURATION * 1000, hls_segment, m3u);
-
- int track_aac = -1;
- int track_264 = -1;
- int track_265 = -1;
- for (unsigned int i = 0; i < ic->nb_streams; i++)
- {
- AVStream* st = ic->streams[i];
- if (AV_CODEC_ID_AAC == st->codecpar->codec_id)
- track_aac = hls_fmp4_add_audio(hls, MOV_OBJECT_AAC, st->codecpar->channels, st->codecpar->bits_per_coded_sample, st->codecpar->sample_rate, st->codecpar->extradata, st->codecpar->extradata_size);
- else if (AV_CODEC_ID_H264 == st->codecpar->codec_id)
- track_264 = hls_fmp4_add_video(hls, MOV_OBJECT_H264, st->codecpar->width, st->codecpar->height, st->codecpar->extradata, st->codecpar->extradata_size);
- else if(AV_CODEC_ID_H265 == st->codecpar->codec_id)
- track_265 = hls_fmp4_add_video(hls, MOV_OBJECT_HEVC, st->codecpar->width, st->codecpar->height, st->codecpar->extradata, st->codecpar->extradata_size);
- }
-
- // write init segment
- hls_init_segment(hls, m3u);
-
- int r = av_read_frame(ic, &pkt);
- while (0 == r)
- {
- AVStream* st = ic->streams[pkt.stream_index];
- int64_t pts = (int64_t)(pkt.pts * av_q2d(st->time_base) * 1000);
- int64_t dts = (int64_t)(pkt.dts * av_q2d(st->time_base) * 1000);
- if (AV_CODEC_ID_AAC == st->codecpar->codec_id)
- {
- //printf("[A] pts: %08lld, dts: %08lld\n", pts, dts);
- hls_fmp4_input(hls, track_aac, pkt.data, pkt.size, pts, dts, 0);
- }
- else if (AV_CODEC_ID_H264 == st->codecpar->codec_id)
- {
- //printf("[V] pts: %08lld, dts: %08lld\n", pts, dts);
- hls_fmp4_input(hls, track_264, pkt.data, pkt.size, pts, dts, (pkt.flags & AV_PKT_FLAG_KEY) ? MOV_AV_FLAG_KEYFREAME : 0);
- }
- else if (AV_CODEC_ID_H265 == st->codecpar->codec_id)
- {
- //printf("[V] pts: %08lld, dts: %08lld\n", pts, dts);
- hls_fmp4_input(hls, track_265, pkt.data, pkt.size, pts, dts, (pkt.flags & AV_PKT_FLAG_KEY) ? MOV_AV_FLAG_KEYFREAME : 0);
- }
- else
- {
- assert(0);
- }
-
- //av_packet_unref(&pkt);
- r = av_read_frame(ic, &pkt);
- }
-
- avformat_close_input(&ic);
- avformat_free_context(ic);
- hls_fmp4_destroy(hls);
-
- // write m3u8 file
- hls_m3u8_playlist(m3u, 1, s_packet, sizeof(s_packet));
- hls_m3u8_destroy(m3u);
-
- FILE* fp = fopen("playlist.m3u8", "wb");
- fwrite(s_packet, 1, strlen(s_packet), fp);
- fclose(fp);
- }
- #endif
|