Git Inbox Mirror of the ffmpeg-devel mailing list - see https://ffmpeg.org/mailman/listinfo/ffmpeg-devel
 help / color / mirror / Atom feed
From: Gustav Grusell <gustav.grusell@gmail.com>
To: ffmpeg-devel@ffmpeg.org
Subject: Re: [FFmpeg-devel] [PATCH] avcodec/libsvtav1: Add support for multipass encoding
Date: Thu, 6 Oct 2022 08:03:43 +0200
Message-ID: <CANPC24ERoRxKV3WMrhgWRNxF6LyKvagSpPOU1BwX6djXgTyQ2w@mail.gmail.com> (raw)
In-Reply-To: <20220927210605.692742-1-gustav.grusell@gmail.com>

Any chance of someone finding the time to have a look at this?

/Gustav

On Tue, Sep 27, 2022 at 11:06 PM Gustav Grusell <gustav.grusell@gmail.com>
wrote:

> Implements support for 2-pass CRF and 3-pass VBR by implementing
> reading and writing of stats file, and passing the pass number on
> to the encoder. For 3-pass VBR, the first pass should be run with
> '-pass 1', the second with '-pass 3' and the third with '-pass 2'.
>
> Signed-off-by: Gustav Grusell <gustav.grusell@gmail.com>
> ---
>  libavcodec/libsvtav1.c | 82 ++++++++++++++++++++++++++++++++++++++++++
>  1 file changed, 82 insertions(+)
>
> diff --git a/libavcodec/libsvtav1.c b/libavcodec/libsvtav1.c
> index 2f5634cee0..73fece49b7 100644
> --- a/libavcodec/libsvtav1.c
> +++ b/libavcodec/libsvtav1.c
> @@ -24,6 +24,7 @@
>  #include <EbSvtAv1ErrorCodes.h>
>  #include <EbSvtAv1Enc.h>
>
> +#include "libavutil/base64.h"
>  #include "libavutil/common.h"
>  #include "libavutil/frame.h"
>  #include "libavutil/imgutils.h"
> @@ -312,6 +313,22 @@ static int config_enc_params(EbSvtAv1EncConfiguration
> *param,
>          cpb_props->avg_bitrate = avctx->bit_rate;
>      }
>
> +    if (avctx->flags & AV_CODEC_FLAG_PASS2) {
> +        if (param->rate_control_mode == SVT_AV1_RC_MODE_VBR) {
> +            if (avctx->flags & AV_CODEC_FLAG_PASS1) {
> +                param->pass = 2;
> +            } else {
> +                param->pass = 3;
> +            }
> +        } else {
> +            param->pass = 2;
> +        }
> +    } else if (avctx->flags & AV_CODEC_FLAG_PASS1) {
> +        param->pass = 1;
> +    } else {
> +        param->pass = 0;
> +    }
> +
>      return 0;
>  }
>
> @@ -371,6 +388,34 @@ static av_cold int eb_enc_init(AVCodecContext *avctx)
>          return ret;
>      }
>
> +    if (svt_enc->enc_params.pass >= 2) {
> +        int decode_size;
> +
> +        if (!avctx->stats_in) {
> +            av_log(avctx, AV_LOG_ERROR, "No stats file for %s pass\n",
> +                   svt_enc->enc_params.pass == 2 ? "second" : "third");
> +            return AVERROR_INVALIDDATA;
> +        }
> +
> +        svt_enc->enc_params.rc_stats_buffer.sz = strlen(avctx->stats_in)
> * 3 / 4;
> +        svt_enc->enc_params.rc_stats_buffer.buf = av_malloc(svt_enc->
> enc_params.rc_stats_buffer.sz);
> +        if (!svt_enc->enc_params.rc_stats_buffer.buf) {
> +            av_log(avctx, AV_LOG_ERROR,
> +                   "Stat buffer alloc (%"SIZE_SPECIFIER" bytes) failed\n",
> +                   svt_enc->enc_params.rc_stats_buffer.sz);
> +            svt_enc->enc_params.rc_stats_buffer.sz = 0;
> +            return AVERROR(ENOMEM);
> +        }
> +        decode_size =
> av_base64_decode(svt_enc->enc_params.rc_stats_buffer.buf, avctx->stats_in,
> +                                       svt_enc->
> enc_params.rc_stats_buffer.sz);
> +        if (decode_size < 0) {
> +            av_log(avctx, AV_LOG_ERROR, "Stat buffer decode failed\n");
> +            return AVERROR_INVALIDDATA;
> +        }
> +
> +        svt_enc->enc_params.rc_stats_buffer.sz = decode_size;
> +    }
> +
>      svt_ret = svt_av1_enc_set_parameter(svt_enc->svt_handle,
> &svt_enc->enc_params);
>      if (svt_ret != EB_ErrorNone) {
>          return svt_print_error(avctx, svt_ret, "Error setting encoder
> parameters");
> @@ -544,6 +589,38 @@ static int eb_receive_packet(AVCodecContext *avctx,
> AVPacket *pkt)
>      if (headerPtr->flags & EB_BUFFERFLAG_EOS)
>          svt_enc->eos_flag = EOS_RECEIVED;
>
> +    if (svt_enc->eos_flag == EOS_RECEIVED) {
> +        if (svt_enc->enc_params.pass == 1) {
> +            SvtAv1FixedBuf first_pass_stat;
> +            EbErrorType    ret = svt_av1_enc_get_stream_info(
> +                svt_enc->svt_handle,
> +                SVT_AV1_STREAM_INFO_FIRST_PASS_STATS_OUT,
> +                &first_pass_stat);
> +            if (ret == EB_ErrorNone) {
> +                size_t b64_size = AV_BASE64_SIZE(first_pass_stat.sz);
> +                avctx->stats_out = av_malloc(b64_size);
> +                if (!avctx->stats_out) {
> +                    av_log(avctx, AV_LOG_ERROR, "Stat buffer alloc
> (%"SIZE_SPECIFIER" bytes) failed\n",
> +                           b64_size);
> +                    return AVERROR(ENOMEM);
> +                }
> +                av_base64_encode(avctx->stats_out, b64_size,
> first_pass_stat.buf,
> +                                 first_pass_stat.sz);
> +            }
> +        }
> +        if (svt_enc->enc_params.pass == 2) {
> +            size_t b64_size = AV_BASE64_SIZE(svt_enc->
> enc_params.rc_stats_buffer.sz);
> +            avctx->stats_out = av_malloc(b64_size);
> +            if (!avctx->stats_out) {
> +                av_log(avctx, AV_LOG_ERROR, "Stat buffer alloc
> (%"SIZE_SPECIFIER" bytes) failed\n",
> +                       b64_size);
> +                return AVERROR(ENOMEM);
> +            }
> +            av_base64_encode(avctx->stats_out, b64_size,
> svt_enc->enc_params.rc_stats_buffer.buf,
> +                             svt_enc->enc_params.rc_stats_buffer.sz);
> +        }
> +    }
> +
>      ff_side_data_set_encoder_stats(pkt, headerPtr->qp * FF_QP2LAMBDA,
> NULL, 0, pict_type);
>
>      svt_av1_enc_release_out_buffer(&headerPtr);
> @@ -564,6 +641,11 @@ static av_cold int eb_enc_close(AVCodecContext *avctx)
>          av_freep(&svt_enc->in_buf);
>      }
>
> +    if (svt_enc->enc_params.rc_stats_buffer.buf) {
> +        av_freep(&svt_enc->enc_params.rc_stats_buffer.buf);
> +        svt_enc->enc_params.rc_stats_buffer.sz = 0;
> +    }
> +
>      av_buffer_pool_uninit(&svt_enc->pool);
>      av_frame_free(&svt_enc->frame);
>
> --
> 2.34.1
>
>
_______________________________________________
ffmpeg-devel mailing list
ffmpeg-devel@ffmpeg.org
https://ffmpeg.org/mailman/listinfo/ffmpeg-devel

To unsubscribe, visit link above, or email
ffmpeg-devel-request@ffmpeg.org with subject "unsubscribe".

  reply	other threads:[~2022-10-06  6:04 UTC|newest]

Thread overview: 5+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2022-09-27 21:06 Gustav Grusell
2022-10-06  6:03 ` Gustav Grusell [this message]
2022-10-30 13:10 ` Lynne
2022-10-30 22:20   ` Oneric
2022-11-03 15:41     ` Gustav Grusell

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=CANPC24ERoRxKV3WMrhgWRNxF6LyKvagSpPOU1BwX6djXgTyQ2w@mail.gmail.com \
    --to=gustav.grusell@gmail.com \
    --cc=ffmpeg-devel@ffmpeg.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link

Git Inbox Mirror of the ffmpeg-devel mailing list - see https://ffmpeg.org/mailman/listinfo/ffmpeg-devel

This inbox may be cloned and mirrored by anyone:

	git clone --mirror https://master.gitmailbox.com/ffmpegdev/0 ffmpegdev/git/0.git

	# If you have public-inbox 1.1+ installed, you may
	# initialize and index your mirror using the following commands:
	public-inbox-init -V2 ffmpegdev ffmpegdev/ https://master.gitmailbox.com/ffmpegdev \
		ffmpegdev@gitmailbox.com
	public-inbox-index ffmpegdev

Example config snippet for mirrors.


AGPL code for this site: git clone https://public-inbox.org/public-inbox.git