RC: Add codec control and impl for init/create

Bug: aomedia:450252793

Change-Id: Ida28381348130ff0cf7a6c84781099baeb2817e7
diff --git a/aom/aomcx.h b/aom/aomcx.h
index cae2a79..e7bfe96 100644
--- a/aom/aomcx.h
+++ b/aom/aomcx.h
@@ -18,6 +18,7 @@
  */
 #include "aom/aom.h"
 #include "aom/aom_encoder.h"
+#include "aom/aom_ext_ratectrl.h"
 #include "aom/aom_external_partition.h"
 
 /*!\file
@@ -1617,6 +1618,12 @@
    */
   AV1E_SET_ENABLE_ADAPTIVE_SHARPNESS = 172,
 
+  /*!\brief Codec control function to enable external rate control library.
+   *
+   * args: a pointer to aom_rc_funcs_t that contains implementation of callbacks
+   */
+  AV1E_SET_EXTERNAL_RATE_CONTROL = 173,
+
   // Any new encoder control IDs should be added above.
   // Maximum allowed encoder control ID is 229.
   // No encoder control ID should be added below.
@@ -2360,6 +2367,9 @@
 AOM_CTRL_USE_TYPE(AV1E_SET_ENABLE_ADAPTIVE_SHARPNESS, unsigned int)
 #define AOM_CTRL_AV1E_SET_ENABLE_ADAPTIVE_SHARPNESS
 
+AOM_CTRL_USE_TYPE(AV1E_SET_EXTERNAL_RATE_CONTROL, aom_rc_funcs_t *)
+#define AOM_CTRL_AV1E_SET_EXTERNAL_RATE_CONTROL
+
 /*!\endcond */
 /*! @} - end defgroup aom_encoder */
 #ifdef __cplusplus
diff --git a/av1/av1.cmake b/av1/av1.cmake
index 42e7530..b2f1bdf 100644
--- a/av1/av1.cmake
+++ b/av1/av1.cmake
@@ -127,6 +127,7 @@
             "${AOM_ROOT}/av1/encoder/allintra_vis.c"
             "${AOM_ROOT}/av1/encoder/allintra_vis.h"
             "${AOM_ROOT}/av1/encoder/enc_enums.h"
+            "${AOM_ROOT}/av1/encoder/av1_ext_ratectrl.c"
             "${AOM_ROOT}/av1/encoder/av1_ext_ratectrl.h"
             "${AOM_ROOT}/av1/encoder/av1_fwd_txfm1d.c"
             "${AOM_ROOT}/av1/encoder/av1_fwd_txfm1d.h"
diff --git a/av1/av1_cx_iface.c b/av1/av1_cx_iface.c
index b49c1aa..295ff0d 100644
--- a/av1/av1_cx_iface.c
+++ b/av1/av1_cx_iface.c
@@ -34,6 +34,7 @@
 #include "av1/common/enums.h"
 #include "av1/common/quant_common.h"
 #include "av1/common/scale.h"
+#include "av1/encoder/av1_ext_ratectrl.h"
 #include "av1/encoder/bitstream.h"
 #include "av1/encoder/enc_enums.h"
 #include "av1/encoder/encoder.h"
@@ -3105,6 +3106,7 @@
 
   if (ctx->ppi) {
     AV1_PRIMARY *ppi = ctx->ppi;
+    av1_extrc_delete(&ppi->cpi->ext_ratectrl);
     for (int i = 0; i < MAX_PARALLEL_FRAMES - 1; i++) {
       if (ppi->parallel_frames_data[i].cx_data) {
         free(ppi->parallel_frames_data[i].cx_data);
@@ -4193,6 +4195,50 @@
   return update_extra_cfg(ctx, &extra_cfg);
 }
 
+static aom_codec_err_t ctrl_set_external_rate_control(aom_codec_alg_priv_t *ctx,
+                                                      va_list args) {
+  aom_rc_funcs_t funcs = *CAST(AV1E_SET_EXTERNAL_RATE_CONTROL, args);
+  AV1_COMP *cpi = ctx->ppi->cpi;
+  AOM_EXT_RATECTRL *ext_ratectrl = &cpi->ext_ratectrl;
+  const AV1EncoderConfig *oxcf = &cpi->oxcf;
+  if (oxcf->pass == AOM_RC_SECOND_PASS) {
+    const FRAME_INFO *frame_info = &cpi->frame_info;
+    aom_rc_config_t ratectrl_config;
+    aom_codec_err_t codec_status;
+    memset(&ratectrl_config, 0, sizeof(ratectrl_config));
+
+    ratectrl_config.frame_width = frame_info->frame_width;
+    ratectrl_config.frame_height = frame_info->frame_height;
+    ratectrl_config.show_frame_count =
+        cpi->ppi->twopass.firstpass_info.stats_count;
+    ratectrl_config.max_gf_interval = ctx->extra_cfg.max_gf_interval;
+    ratectrl_config.min_gf_interval = ctx->extra_cfg.min_gf_interval;
+    ratectrl_config.target_bitrate_kbps =
+        (int)(oxcf->rc_cfg.target_bandwidth / 1000);
+    ratectrl_config.frame_rate_num = ctx->cfg.g_timebase.den;
+    ratectrl_config.frame_rate_den = ctx->cfg.g_timebase.num;
+    ratectrl_config.overshoot_percent = oxcf->rc_cfg.over_shoot_pct;
+    ratectrl_config.undershoot_percent = oxcf->rc_cfg.under_shoot_pct;
+    ratectrl_config.min_base_q_index = oxcf->rc_cfg.best_allowed_q;
+    ratectrl_config.max_base_q_index = oxcf->rc_cfg.worst_allowed_q;
+    ratectrl_config.base_qp = ctx->extra_cfg.cq_level;
+
+    if (ctx->cfg.rc_end_usage == AOM_VBR) {
+      ratectrl_config.rc_mode = AOM_RC_VBR;
+    } else if (ctx->cfg.rc_end_usage == AOM_Q) {
+      ratectrl_config.rc_mode = AOM_RC_QMODE;
+    } else if (ctx->cfg.rc_end_usage == AOM_CQ) {
+      ratectrl_config.rc_mode = AOM_RC_CQ;
+    }
+
+    codec_status = av1_extrc_create(funcs, ratectrl_config, ext_ratectrl);
+    if (codec_status != AOM_CODEC_OK) {
+      return codec_status;
+    }
+  }
+  return AOM_CODEC_OK;
+}
+
 static aom_codec_err_t encoder_set_option(aom_codec_alg_priv_t *ctx,
                                           const char *name, const char *value) {
   if (ctx == NULL || name == NULL || value == NULL)
@@ -4894,6 +4940,7 @@
   { AV1E_SET_SCREEN_CONTENT_DETECTION_MODE,
     ctrl_set_screen_content_detection_mode },
   { AV1E_SET_ENABLE_ADAPTIVE_SHARPNESS, ctrl_set_enable_adaptive_sharpness },
+  { AV1E_SET_EXTERNAL_RATE_CONTROL, ctrl_set_external_rate_control },
 
   // Getters
   { AOME_GET_LAST_QUANTIZER, ctrl_get_quantizer },
diff --git a/av1/encoder/av1_ext_ratectrl.c b/av1/encoder/av1_ext_ratectrl.c
new file mode 100644
index 0000000..bf24fb9
--- /dev/null
+++ b/av1/encoder/av1_ext_ratectrl.c
@@ -0,0 +1,66 @@
+/*
+ * 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;
+}
+
+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);
+  }
+  return av1_extrc_init(ext_ratectrl);
+}
diff --git a/av1/encoder/encoder.h b/av1/encoder/encoder.h
index b31a7d9..361332a 100644
--- a/av1/encoder/encoder.h
+++ b/av1/encoder/encoder.h
@@ -34,6 +34,7 @@
 #include "av1/common/timing.h"
 
 #include "av1/encoder/aq_cyclicrefresh.h"
+#include "av1/encoder/av1_ext_ratectrl.h"
 #include "av1/encoder/av1_quantize.h"
 #include "av1/encoder/block.h"
 #include "av1/encoder/context_tree.h"
@@ -3706,6 +3707,11 @@
    * ROI map.
    */
   aom_roi_map_t roi;
+
+  /*!
+   * External rate control.
+   */
+  AOM_EXT_RATECTRL ext_ratectrl;
 } AV1_COMP;
 
 /*!
diff --git a/build/cmake/aom_install.cmake b/build/cmake/aom_install.cmake
index 4291782..1ac795e 100644
--- a/build/cmake/aom_install.cmake
+++ b/build/cmake/aom_install.cmake
@@ -20,7 +20,9 @@
 if(CONFIG_AV1_ENCODER)
   list(APPEND AOM_INSTALL_INCS "${AOM_ROOT}/aom/aomcx.h"
               "${AOM_ROOT}/aom/aom_encoder.h"
-              "${AOM_ROOT}/aom/aom_external_partition.h")
+              "${AOM_ROOT}/aom/aom_ext_ratectrl.h"
+              "${AOM_ROOT}/aom/aom_external_partition.h"
+              "${AOM_ROOT}/aom/aom_tpl.h")
 endif()
 
 # Generate aom.pc and setup dependencies to ensure it is created when necessary.
diff --git a/test/encode_test_driver.h b/test/encode_test_driver.h
index a589ae2..739e9a0 100644
--- a/test/encode_test_driver.h
+++ b/test/encode_test_driver.h
@@ -159,6 +159,10 @@
     const aom_codec_err_t res = aom_codec_control(&encoder_, ctrl_id, arg);
     ASSERT_EQ(AOM_CODEC_OK, res) << EncoderError();
   }
+  void Control(int ctrl_id, aom_rc_funcs_t *arg) {
+    const aom_codec_err_t res = aom_codec_control(&encoder_, ctrl_id, arg);
+    ASSERT_EQ(AOM_CODEC_OK, res) << EncoderError();
+  }
 #endif
 
   void SetOption(const char *name, const char *value) {
diff --git a/test/ext_ratectrl_test.cc b/test/ext_ratectrl_test.cc
new file mode 100644
index 0000000..ef7f92e
--- /dev/null
+++ b/test/ext_ratectrl_test.cc
@@ -0,0 +1,121 @@
+/*
+ * 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 "test/codec_factory.h"
+#include "test/encode_test_driver.h"
+#include "test/util.h"
+#include "test/y4m_video_source.h"
+#include "aom/aom_ext_ratectrl.h"
+#include "gtest/gtest.h"
+
+namespace {
+
+const int kFrameNum = 5;
+
+// A mock rate control model.
+struct MockRateCtrlModel {};
+
+// A mock private data for rate control callbacks.
+struct MockRC {};
+struct MockRC g_priv;
+
+// A flag to indicate if create_model() is called.
+bool is_create_model_called = false;
+
+// A flag to indicate if delete_model() is called.
+bool is_delete_model_called = false;
+
+aom_rc_status_t mock_create_model(void *priv,
+                                  const aom_rc_config_t *ratectrl_config,
+                                  aom_rc_model_t *ratectrl_model) {
+  (void)priv;
+  (void)ratectrl_config;
+  EXPECT_NE(ratectrl_model, nullptr);
+  *ratectrl_model = (aom_rc_model_t)(new MockRateCtrlModel());
+  is_create_model_called = true;
+  return AOM_RC_OK;
+}
+
+aom_rc_status_t mock_delete_model(aom_rc_model_t ratectrl_model) {
+  EXPECT_NE(ratectrl_model, nullptr);
+  delete (MockRateCtrlModel *)ratectrl_model;
+  is_delete_model_called = true;
+  return AOM_RC_OK;
+}
+
+aom_rc_status_t mock_send_firstpass_stats(
+    aom_rc_model_t ratectrl_model,
+    const aom_rc_firstpass_stats_t *firstpass_stats) {
+  (void)ratectrl_model;
+  (void)firstpass_stats;
+  return AOM_RC_OK;
+}
+
+aom_rc_status_t mock_get_encodeframe_decision(
+    aom_rc_model_t ratectrl_model, const int frame_gop_index,
+    aom_rc_encodeframe_decision_t *frame_decision) {
+  (void)ratectrl_model;
+  (void)frame_gop_index;
+  (void)frame_decision;
+  return AOM_RC_OK;
+}
+
+aom_rc_status_t mock_update_encodeframe_result(
+    aom_rc_model_t ratectrl_model,
+    const aom_rc_encodeframe_result_t *encode_frame_result) {
+  (void)ratectrl_model;
+  (void)encode_frame_result;
+  return AOM_RC_OK;
+}
+
+class ExtRateCtrlTest : public ::libaom_test::EncoderTest,
+                        public ::libaom_test::CodecTestWith2Params<int, int> {
+ protected:
+  ExtRateCtrlTest() : EncoderTest(GET_PARAM(0)), cpu_used_(GET_PARAM(2)) {
+    aom_rc_funcs_t *rc_funcs = &rc_funcs_;
+    rc_funcs->priv = &g_priv;
+    rc_funcs->create_model = mock_create_model;
+    rc_funcs->delete_model = mock_delete_model;
+    rc_funcs->send_firstpass_stats = mock_send_firstpass_stats;
+    rc_funcs->get_encodeframe_decision = mock_get_encodeframe_decision;
+    rc_funcs->update_encodeframe_result = mock_update_encodeframe_result;
+  }
+  ~ExtRateCtrlTest() override = default;
+
+  void SetUp() override {
+    InitializeConfig(static_cast<libaom_test::TestMode>(GET_PARAM(1)));
+    cfg_.g_threads = 1;
+    cfg_.g_limit = kFrameNum;
+  }
+
+  void PreEncodeFrameHook(::libaom_test::VideoSource *video,
+                          ::libaom_test::Encoder *encoder) override {
+    if (video->frame() == 0) {
+      encoder->Control(AOME_SET_CPUUSED, cpu_used_);
+      encoder->Control(AV1E_SET_EXTERNAL_RATE_CONTROL, &rc_funcs_);
+    }
+  }
+
+  aom_rc_funcs_t rc_funcs_;
+  int cpu_used_;
+};
+
+TEST_P(ExtRateCtrlTest, TestExternalRateCtrl) {
+  ::libaom_test::Y4mVideoSource video("screendata.y4m", 0, kFrameNum);
+  ASSERT_NO_FATAL_FAILURE(RunLoop(&video));
+  EXPECT_TRUE(is_create_model_called);
+  EXPECT_TRUE(is_delete_model_called);
+}
+
+AV1_INSTANTIATE_TEST_SUITE(ExtRateCtrlTest,
+                           ::testing::Values(::libaom_test::kTwoPassGood),
+                           ::testing::Values(0));
+}  // namespace
diff --git a/test/test.cmake b/test/test.cmake
index 4228a25..088ce3c 100644
--- a/test/test.cmake
+++ b/test/test.cmake
@@ -76,6 +76,7 @@
             "${AOM_ROOT}/test/encode_test_driver.cc"
             "${AOM_ROOT}/test/encode_test_driver.h"
             "${AOM_ROOT}/test/end_to_end_psnr_test.cc"
+            "${AOM_ROOT}/test/ext_ratectrl_test.cc"
             "${AOM_ROOT}/test/forced_max_frame_width_height_test.cc"
             "${AOM_ROOT}/test/force_key_frame_test.cc"
             "${AOM_ROOT}/test/gf_pyr_height_test.cc"