/* * Copyright (c) 2025, Alliance for Open Media. All rights reserved. * * This source code is subject to the terms of the BSD 2 Clause License and * the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License * was not distributed with this source code in the LICENSE file, you can * obtain it at www.aomedia.org/license/software. If the Alliance for Open * Media Patent License 1.0 was not distributed with this source code in the * PATENTS file, you can obtain it at www.aomedia.org/license/patent. */ #include "aom/aom_ext_ratectrl.h" #include "av1/encoder/av1_ext_ratectrl.h" aom_codec_err_t av1_extrc_init(AOM_EXT_RATECTRL *ext_ratectrl) { if (ext_ratectrl == NULL) { return AOM_CODEC_INVALID_PARAM; } av1_zero(*ext_ratectrl); return AOM_CODEC_OK; } aom_codec_err_t av1_extrc_create(aom_rc_funcs_t funcs, aom_rc_config_t ratectrl_config, AOM_EXT_RATECTRL *ext_ratectrl) { aom_rc_status_t rc_status; aom_rc_firstpass_stats_t *rc_firstpass_stats; if (ext_ratectrl == NULL) { return AOM_CODEC_INVALID_PARAM; } av1_extrc_delete(ext_ratectrl); ext_ratectrl->funcs = funcs; ext_ratectrl->ratectrl_config = ratectrl_config; rc_status = ext_ratectrl->funcs.create_model(ext_ratectrl->funcs.priv, &ext_ratectrl->ratectrl_config, &ext_ratectrl->model); if (rc_status == AOM_RC_ERROR) { return AOM_CODEC_ERROR; } rc_firstpass_stats = &ext_ratectrl->rc_firstpass_stats; rc_firstpass_stats->num_frames = ratectrl_config.show_frame_count; rc_firstpass_stats->frame_stats = aom_malloc(sizeof(*rc_firstpass_stats->frame_stats) * rc_firstpass_stats->num_frames); if (rc_firstpass_stats->frame_stats == NULL) { return AOM_CODEC_MEM_ERROR; } ext_ratectrl->ready = 1; return AOM_CODEC_OK; } static void gen_rc_firstpass_stats(const FIRSTPASS_STATS *stats, aom_rc_frame_stats_t *rc_frame_stats) { rc_frame_stats->frame = stats->frame; rc_frame_stats->weight = stats->weight; rc_frame_stats->intra_error = stats->intra_error; rc_frame_stats->coded_error = stats->coded_error; rc_frame_stats->sr_coded_error = stats->sr_coded_error; rc_frame_stats->frame_avg_wavelet_energy = stats->frame_avg_wavelet_energy; rc_frame_stats->pcnt_inter = stats->pcnt_inter; rc_frame_stats->pcnt_motion = stats->pcnt_motion; rc_frame_stats->pcnt_second_ref = stats->pcnt_second_ref; rc_frame_stats->pcnt_neutral = stats->pcnt_neutral; rc_frame_stats->intra_skip_pct = stats->intra_skip_pct; rc_frame_stats->inactive_zone_rows = stats->inactive_zone_rows; rc_frame_stats->inactive_zone_cols = stats->inactive_zone_cols; rc_frame_stats->MVr = stats->MVr; rc_frame_stats->mvr_abs = stats->mvr_abs; rc_frame_stats->MVc = stats->MVc; rc_frame_stats->mvc_abs = stats->mvc_abs; rc_frame_stats->MVrv = stats->MVrv; rc_frame_stats->MVcv = stats->MVcv; rc_frame_stats->mv_in_out_count = stats->mv_in_out_count; rc_frame_stats->duration = stats->duration; rc_frame_stats->count = stats->count; rc_frame_stats->new_mv_count = stats->new_mv_count; rc_frame_stats->raw_error_stdev = stats->raw_error_stdev; rc_frame_stats->is_flash = stats->is_flash; rc_frame_stats->noise_var = stats->noise_var; rc_frame_stats->cor_coeff = stats->cor_coeff; rc_frame_stats->log_intra_error = stats->log_intra_error; rc_frame_stats->log_coded_error = stats->log_coded_error; } aom_codec_err_t av1_extrc_send_firstpass_stats( AOM_EXT_RATECTRL *ext_ratectrl, const FIRSTPASS_INFO *first_pass_info) { assert(first_pass_info != NULL); assert(ext_ratectrl != NULL); if (ext_ratectrl->ready) { aom_rc_status_t rc_status; aom_rc_firstpass_stats_t *rc_firstpass_stats = &ext_ratectrl->rc_firstpass_stats; assert(rc_firstpass_stats->num_frames == first_pass_info->stats_buf_size); for (int i = 0; i < rc_firstpass_stats->num_frames; ++i) { gen_rc_firstpass_stats(&first_pass_info->stats_buf[i], &rc_firstpass_stats->frame_stats[i]); } rc_status = ext_ratectrl->funcs.send_firstpass_stats(ext_ratectrl->model, rc_firstpass_stats); if (rc_status == AOM_RC_ERROR) { return AOM_CODEC_ERROR; } } return AOM_CODEC_OK; } aom_codec_err_t av1_extrc_send_tpl_stats( AOM_EXT_RATECTRL *ext_ratectrl, const AomTplGopStats *extrc_tpl_gop_stats) { assert(ext_ratectrl != NULL); assert(extrc_tpl_gop_stats != NULL); if (ext_ratectrl->ready && ext_ratectrl->funcs.send_tpl_gop_stats != NULL) { aom_rc_status_t rc_status = ext_ratectrl->funcs.send_tpl_gop_stats( ext_ratectrl->model, extrc_tpl_gop_stats); if (rc_status == AOM_RC_ERROR) { return AOM_CODEC_ERROR; } } return AOM_CODEC_OK; } aom_codec_err_t av1_extrc_get_gop_decision( AOM_EXT_RATECTRL *ext_ratectrl, aom_rc_gop_decision_t *gop_decision) { aom_rc_status_t rc_status; assert(ext_ratectrl != NULL); assert(gop_decision != NULL); if ((ext_ratectrl->funcs.rc_type & AOM_RC_GOP) == 0) { return AOM_CODEC_INVALID_PARAM; } rc_status = ext_ratectrl->funcs.get_gop_decision(ext_ratectrl->model, gop_decision); if (rc_status == AOM_RC_ERROR) { return AOM_CODEC_ERROR; } return AOM_CODEC_OK; } aom_codec_err_t av1_extrc_get_encodeframe_decision( AOM_EXT_RATECTRL *ext_ratectrl, int gop_index, aom_rc_encodeframe_decision_t *encode_frame_decision) { assert(ext_ratectrl != NULL); assert(ext_ratectrl->ready && (ext_ratectrl->funcs.rc_type & AOM_RC_QP) != 0); assert(encode_frame_decision != NULL); aom_rc_status_t rc_status = ext_ratectrl->funcs.get_encodeframe_decision( ext_ratectrl->model, gop_index, encode_frame_decision); if (rc_status == AOM_RC_ERROR) { return AOM_CODEC_ERROR; } return AOM_CODEC_OK; } aom_codec_err_t av1_extrc_update_encodeframe_result( AOM_EXT_RATECTRL *ext_ratectrl, int64_t bit_count, int actual_encoding_qindex) { assert(ext_ratectrl != NULL); if (ext_ratectrl->ready) { aom_rc_status_t rc_status; aom_rc_encodeframe_result_t encode_frame_result; encode_frame_result.bit_count = bit_count; encode_frame_result.actual_encoding_qindex = actual_encoding_qindex; rc_status = ext_ratectrl->funcs.update_encodeframe_result( ext_ratectrl->model, &encode_frame_result); if (rc_status == AOM_RC_ERROR) { return AOM_CODEC_ERROR; } } return AOM_CODEC_OK; } aom_codec_err_t av1_extrc_get_key_frame_decision( AOM_EXT_RATECTRL *ext_ratectrl, aom_rc_key_frame_decision_t *key_frame_decision) { assert(ext_ratectrl != NULL); if (!ext_ratectrl->ready || (ext_ratectrl->funcs.rc_type & AOM_RC_GOP) == 0) { return AOM_CODEC_INVALID_PARAM; } aom_rc_status_t rc_status = ext_ratectrl->funcs.get_key_frame_decision( ext_ratectrl->model, key_frame_decision); return rc_status == AOM_RC_OK ? AOM_CODEC_OK : AOM_CODEC_ERROR; } aom_codec_err_t av1_extrc_delete(AOM_EXT_RATECTRL *ext_ratectrl) { if (ext_ratectrl == NULL) { return AOM_CODEC_INVALID_PARAM; } if (ext_ratectrl->ready) { aom_rc_status_t rc_status = ext_ratectrl->funcs.delete_model(ext_ratectrl->model); if (rc_status == AOM_RC_ERROR) { return AOM_CODEC_ERROR; } aom_free(ext_ratectrl->rc_firstpass_stats.frame_stats); aom_free(ext_ratectrl->sb_params_list); } return av1_extrc_init(ext_ratectrl); }