123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115 |
- //
- // Created by xu fulong on 2022/9/9.
- //
- #include "ff_rtmp_pusher.h"
- #define PUSH_TAG "HttpPusher"
- int FFRtmpPusher::open(const char *inputPath, const char *outputPath) {
- int ret;
- avformat_network_init();
- ret = avformat_open_input(&inFormatCtx, inputPath, nullptr, nullptr);
- if (ret < 0) {
- LOGE(PUSH_TAG, "avformat_open_input err=%s", av_err2str(ret));
- return ret;
- }
- avformat_find_stream_info(inFormatCtx, nullptr);
- av_dump_format(inFormatCtx, 0, inputPath, 0);
- ret = avformat_alloc_output_context2(&outFormatCtx, nullptr, "flv", outputPath);
- if (ret < 0 || !outFormatCtx) {
- LOGE(PUSH_TAG, "alloc format_context err=%s", av_err2str(ret));
- return ret;
- }
- // Flv video: h264 audio: aac/mp3
- // If not h264, should transocde to h264
- for (int i = 0; i < inFormatCtx->nb_streams; ++i) {
- AVStream *in_stream = inFormatCtx->streams[i];
- const auto *codec = avcodec_find_encoder(in_stream->codecpar->codec_id);
- AVStream *out_stream = avformat_new_stream(outFormatCtx, codec);
- avcodec_parameters_copy(out_stream->codecpar, in_stream->codecpar);
- out_stream->codecpar->codec_tag = 0;
- if (in_stream->codecpar->codec_type == AVMEDIA_TYPE_VIDEO) {
- video_index = i;
- } else if (in_stream->codecpar->codec_type == AVMEDIA_TYPE_AUDIO) {
- if (audio_index == -1) {
- audio_index = i;
- }
- }
- }
- if (!(outFormatCtx->oformat->flags & AVFMT_NOFILE)) {
- ret = avio_open2(&outFormatCtx->pb, outputPath, AVIO_FLAG_WRITE, nullptr, nullptr);
- if (ret < 0) {
- LOGE(PUSH_TAG, "avio open error=%s", av_err2str(ret));
- return ret;
- }
- }
- ret = avformat_write_header(outFormatCtx, nullptr);
- if (ret < 0) {
- LOGE(PUSH_TAG, "avformat_write_header err=%s", av_err2str(ret));
- }
- return ret;
- }
- void rescale(AVFormatContext *in_format_ctx, AVFormatContext *out_format_ctx, AVPacket *packet) {
- AVStream *in_stream = in_format_ctx->streams[packet->stream_index];
- AVStream *out_stream = out_format_ctx->streams[packet->stream_index];
- packet->pts = av_rescale_q(packet->pts, in_stream->time_base, out_stream->time_base);
- packet->dts = av_rescale_q(packet->dts, in_stream->time_base, out_stream->time_base);
- packet->duration = av_rescale_q(packet->duration, in_stream->time_base, out_stream->time_base);
- packet->pos = -1;
- }
- int FFRtmpPusher::push() {
- int ret;
- int64_t startTime = av_gettime();
- while (true) {
- ret = av_read_frame(inFormatCtx, &packet);
- if (ret < 0) {
- LOGE(PUSH_TAG, "av_read_frame err=%s", av_err2str(ret));
- break;
- }
- if (packet.stream_index != video_index && packet.stream_index != audio_index)
- continue;
- // sync
- AVRational time_base = inFormatCtx->streams[packet.stream_index]->time_base;
- int64_t pts_time = av_rescale_q(packet.pts, time_base, AV_TIME_BASE_Q);
- int64_t cur_time = av_gettime() - startTime;
- if (pts_time > cur_time) {
- av_usleep((unsigned int)(pts_time - cur_time));
- }
- rescale(inFormatCtx, outFormatCtx, &packet);
- ret = av_interleaved_write_frame(outFormatCtx, &packet);
- if (ret < 0) {
- LOGE(PUSH_TAG, "write frame err=%s", av_err2str(ret));
- break;
- }
- av_packet_unref(&packet);
- }
- return ret;
- }
- void FFRtmpPusher::close() {
- if (outFormatCtx) {
- av_write_trailer(outFormatCtx);
- avformat_close_input(&outFormatCtx);
- outFormatCtx = nullptr;
- }
- if (inFormatCtx) {
- avformat_close_input(&inFormatCtx);
- inFormatCtx = nullptr;
- }
- }
|