drm/amd/display: Limit Minimum FreeSync Refresh Rate
authorAustin Zheng <austin.zheng@amd.com>
Thu, 1 Jun 2023 19:16:45 +0000 (15:16 -0400)
committerAlex Deucher <alexander.deucher@amd.com>
Thu, 15 Jun 2023 14:44:49 +0000 (10:44 -0400)
Why:
Some EDIDs report a minimum refresh rate lower than what HW can support

How:
Add a check to calculate minimum supported refresh rate with current timing
and use that as the minimum if a lower one is passed in

Acked-by: Stylon Wang <stylon.wang@amd.com>
Signed-off-by: Austin Zheng <austin.zheng@amd.com>
Reviewed-by: Anthony Koo <Anthony.Koo@amd.com>
Tested-by: Daniel Wheeler <daniel.wheeler@amd.com>
Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
drivers/gpu/drm/amd/display/dc/dc.h
drivers/gpu/drm/amd/display/dc/dcn30/dcn30_resource.c
drivers/gpu/drm/amd/display/dc/dcn302/dcn302_resource.c
drivers/gpu/drm/amd/display/dc/dcn303/dcn303_resource.c
drivers/gpu/drm/amd/display/dc/dcn32/dcn32_resource.c
drivers/gpu/drm/amd/display/dc/dcn321/dcn321_resource.c
drivers/gpu/drm/amd/display/modules/freesync/freesync.c

index 5c906b007e4d53de8f0694fb6d9713221a9892b7..a239dcd8e9fb0d1ed7a6a9e286fc16e37fed317e 100644 (file)
@@ -266,6 +266,7 @@ struct dc_caps {
        uint16_t subvp_pstate_allow_width_us;
        uint16_t subvp_vertical_int_margin_us;
        bool seamless_odm;
+       uint32_t max_v_total;
        uint8_t subvp_drr_vblank_start_margin_us;
 };
 
index f4ee4b3df59628e6e9c1b42eff5d4d64cfa27f0e..1a0284a068b2a74e24a96d79af6f6272dc303ac5 100644 (file)
@@ -2328,6 +2328,7 @@ static bool dcn30_resource_construct(
        dc->caps.color.mpc.ocsc = 1;
 
        dc->caps.dp_hdmi21_pcon_support = true;
+       dc->caps.max_v_total = (1 << 15) - 1;
 
        /* read VBIOS LTTPR caps */
        {
index 93f42132c900f04237c981da84133e757b2b8328..7dc065ea247a1824175ef8bf0fcf94334d86f977 100644 (file)
@@ -1227,6 +1227,7 @@ static bool dcn302_resource_construct(
        dc->caps.force_dp_tps4_for_cp2520 = true;
        dc->caps.extended_aux_timeout_support = true;
        dc->caps.dmcub_support = true;
+       dc->caps.max_v_total = (1 << 15) - 1;
 
        /* Color pipeline capabilities */
        dc->caps.color.dpp.dcn_arch = 1;
index f35514188a5cd17249fb669934c85d6c2e0122cb..6d9761395288db268ca69df92c8ce9a7a557885e 100644 (file)
@@ -1152,6 +1152,7 @@ static bool dcn303_resource_construct(
        dc->caps.force_dp_tps4_for_cp2520 = true;
        dc->caps.extended_aux_timeout_support = true;
        dc->caps.dmcub_support = true;
+       dc->caps.max_v_total = (1 << 15) - 1;
 
        /* Color pipeline capabilities */
        dc->caps.color.dpp.dcn_arch = 1;
index 8c9e15952a49048363546042cd81565324733a1c..19f134caa8adba9125f2f886c6b6639a982d95a2 100644 (file)
@@ -2175,6 +2175,7 @@ static bool dcn32_resource_construct(
        dc->caps.extended_aux_timeout_support = true;
        dc->caps.dmcub_support = true;
        dc->caps.seamless_odm = true;
+       dc->caps.max_v_total = (1 << 15) - 1;
 
        /* Color pipeline capabilities */
        dc->caps.color.dpp.dcn_arch = 1;
index ee07ee3401713acf18340f947d85d525e6b1e76c..ea204742ad35455baa093e38d6fd7945aaa16c9c 100644 (file)
@@ -1718,6 +1718,7 @@ static bool dcn321_resource_construct(
        dc->caps.edp_dsc_support = true;
        dc->caps.extended_aux_timeout_support = true;
        dc->caps.dmcub_support = true;
+       dc->caps.max_v_total = (1 << 15) - 1;
 
        /* Color pipeline capabilities */
        dc->caps.color.dpp.dcn_arch = 1;
index 5798c0eafa1fb279ebc75218b42848e298e8e51e..dbd60811f95da4a8d17f76784758ce363f3e0058 100644 (file)
@@ -1,5 +1,5 @@
 /*
- * Copyright 2016 Advanced Micro Devices, Inc.
+ * Copyright 2016-2023 Advanced Micro Devices, Inc.
  *
  * Permission is hereby granted, free of charge, to any person obtaining a
  * copy of this software and associated documentation files (the "Software"),
@@ -989,6 +989,7 @@ void mod_freesync_build_vrr_params(struct mod_freesync *mod_freesync,
        unsigned int refresh_range = 0;
        unsigned long long min_refresh_in_uhz = 0;
        unsigned long long max_refresh_in_uhz = 0;
+       unsigned long long min_hardware_refresh_in_uhz = 0;
 
        if (mod_freesync == NULL)
                return;
@@ -999,7 +1000,13 @@ void mod_freesync_build_vrr_params(struct mod_freesync *mod_freesync,
        nominal_field_rate_in_uhz =
                        mod_freesync_calc_nominal_field_rate(stream);
 
-       min_refresh_in_uhz = in_config->min_refresh_in_uhz;
+       if (stream->ctx->dc->caps.max_v_total != 0 && stream->timing.h_total != 0) {
+               min_hardware_refresh_in_uhz = div64_u64((stream->timing.pix_clk_100hz * 100000000ULL),
+                       (stream->timing.h_total * stream->ctx->dc->caps.max_v_total));
+       }
+       /* Limit minimum refresh rate to what can be supported by hardware */
+       min_refresh_in_uhz = min_hardware_refresh_in_uhz > in_config->min_refresh_in_uhz ?
+               min_hardware_refresh_in_uhz : in_config->min_refresh_in_uhz;
        max_refresh_in_uhz = in_config->max_refresh_in_uhz;
 
        /* Full range may be larger than current video timing, so cap at nominal */