chromium/media/gpu/windows/h264_video_rate_control_wrapper.cc

// Copyright 2024 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#ifdef UNSAFE_BUFFERS_BUILD
// TODO(crbug.com/40285824): Remove this and convert code to safer constructs.
#pragma allow_unsafe_buffers
#endif

#include "h264_video_rate_control_wrapper.h"

#include "media/gpu/h264_ratectrl_rtc.h"

namespace media {

template <>
int H264RateControl::GetLoopfilterLevel() const {
  DCHECK(impl_);
  return impl_->GetLoopfilterLevel();
}

template <>
void H264RateControl::PostEncodeUpdate(uint64_t encoded_frame_size,
                                       const FrameParams& frame_params) {
  DCHECK(impl_);
  return impl_->PostEncodeUpdate(encoded_frame_size,
                                 ConvertFrameParams(frame_params));
}

template <>
H264RateControlConfigRTC H264RateControl::ConvertControlConfig(
    const RateControlConfig& config) {
  // Limit max delay for intra frame with HRD buffer size (500ms-1s for camera
  // video, 1s-10s for desktop sharing).
  constexpr base::TimeDelta kHRDBufferDelayCamera = base::Milliseconds(1000);
  constexpr base::TimeDelta kHRDBufferDelayDisplay = base::Milliseconds(3000);
  H264RateControlConfigRTC rc_config;

  // Coded width and heght.
  rc_config.frame_size.SetSize(config.width, config.height);
  // Maximum GOP duration in milliseconds. It is set to maximum value.
  rc_config.gop_max_duration = base::TimeDelta::Max();
  // Source frame rate.
  rc_config.frame_rate_max = static_cast<float>(config.framerate);
  // Number of temopral layers.
  rc_config.num_temporal_layers = config.ts_number_layers;
  // Type of the video content (camera or display).
  rc_config.content_type = config.content_type;
  rc_config.fixed_delta_qp = true;
  rc_config.ease_hrd_reduction = true;
  for (int tid = 0; tid < config.ts_number_layers; ++tid) {
    rc_config.layer_settings.emplace_back();
    rc_config.layer_settings[tid].avg_bitrate =
        config.layer_target_bitrate[tid] * 1000;
    // Peak bitrate is set to 1.5 times the average bitrate.
    rc_config.layer_settings[tid].peak_bitrate =
        config.layer_target_bitrate[tid] * 1000 * 3 / 2;
    base::TimeDelta buffer_delay;
    if (config.content_type ==
        VideoEncodeAccelerator::Config::ContentType::kCamera) {
      buffer_delay = kHRDBufferDelayCamera;
    } else {
      buffer_delay = kHRDBufferDelayDisplay;
    }
    size_t buffer_size = static_cast<size_t>(
        rc_config.layer_settings[tid].avg_bitrate *
        buffer_delay.InMilliseconds() / base::Time::kMillisecondsPerSecond / 8);
    rc_config.layer_settings[tid].hrd_buffer_size = buffer_size;
    rc_config.layer_settings[tid].min_qp = config.min_quantizers[tid];
    rc_config.layer_settings[tid].max_qp = config.max_quantizers[tid];
    rc_config.layer_settings[tid].frame_rate = static_cast<float>(
        config.framerate / (1 << (config.ts_number_layers - tid - 1)));

    if (tid > 0) {
      DCHECK_GT(rc_config.layer_settings[tid].avg_bitrate,
                rc_config.layer_settings[tid - 1].avg_bitrate);
    }

  }
  return rc_config;
}

template <>
H264FrameParamsRTC H264RateControl::ConvertFrameParams(
    const FrameParams& frame_params) {
  H264FrameParamsRTC rc_params;
  rc_params.temporal_layer_id = frame_params.temporal_layer_id;
  rc_params.keyframe =
      frame_params.frame_type == FrameParams::FrameType::kKeyFrame;
  rc_params.timestamp = base::Milliseconds(frame_params.timestamp);
  return rc_params;
}

}  // namespace media