On Thu, Mar 9, 2023 at 6:54 AM Lazar, Lijo <lijo.la...@amd.com> wrote:
>
>
>
> On 3/9/2023 8:11 AM, Quan, Evan wrote:
> > [AMD Official Use Only - General]
> >
> >
> >
> >> -----Original Message-----
> >> From: Deucher, Alexander <alexander.deuc...@amd.com>
> >> Sent: Wednesday, March 8, 2023 11:20 PM
> >> To: amd-gfx@lists.freedesktop.org
> >> Cc: Deucher, Alexander <alexander.deuc...@amd.com>; Błażej Szczygieł
> >> <mumei6...@gmail.com>; Quan, Evan <evan.q...@amd.com>
> >> Subject: [PATCH 2/2] drm/amd/pm: Fix navi10 incorrect OD volage after
> >> resume
> >>
> >> Always setup overdrive tables after resume. Preserve only some
> >> user-defined settings in user_overdrive_table if they're set.
> >>
> >> Copy restored user_overdrive_table into od_table to get correct
> >> values.
> >>
> >> On cold boot, BTC was triggered and GfxVfCurve was calibrated. We
> >> got VfCurve settings (a). On resuming back, BTC will be triggered
> >> again and GfxVfCurve will be recalibrated. VfCurve settings (b)
> >> got may be different from those of cold boot.  So if we reuse
> >> those VfCurve settings (a) got on cold boot on suspend, we can
> >> run into discrepencies.
> >>
> >> Based on the sienna cichlid patch from Błażej Szczygieł
> >> <mumei6...@gmail.com>
> >>
> >> Cc: Błażej Szczygieł <mumei6...@gmail.com>
> >> Cc: Evan Quan <evan.q...@amd.com>
> >> Signed-off-by: Alex Deucher <alexander.deuc...@amd.com>
> >> ---
> >>   .../gpu/drm/amd/pm/swsmu/smu11/navi10_ppt.c   | 47
> >> +++++++++++++++----
> >>   1 file changed, 37 insertions(+), 10 deletions(-)
> >>
> >> diff --git a/drivers/gpu/drm/amd/pm/swsmu/smu11/navi10_ppt.c
> >> b/drivers/gpu/drm/amd/pm/swsmu/smu11/navi10_ppt.c
> >> index 95da6dd1cc65..68201d8e1c72 100644
> >> --- a/drivers/gpu/drm/amd/pm/swsmu/smu11/navi10_ppt.c
> >> +++ b/drivers/gpu/drm/amd/pm/swsmu/smu11/navi10_ppt.c
> >> @@ -2510,16 +2510,9 @@ static int navi10_set_default_od_settings(struct
> >> smu_context *smu)
> >>              (OverDriveTable_t *)smu->smu_table.boot_overdrive_table;
> >>      OverDriveTable_t *user_od_table =
> >>              (OverDriveTable_t *)smu->smu_table.user_overdrive_table;
> >> +    OverDriveTable_t user_od_table_bak;
> >>      int ret = 0;
> >>
> >> -    /*
> >> -     * For S3/S4/Runpm resume, no need to setup those overdrive
> >> tables again as
> >> -     *   - either they already have the default OD settings got during 
> >> cold
> >> bootup
> >> -     *   - or they have some user customized OD settings which cannot be
> >> overwritten
> >> -     */
> >> -    if (smu->adev->in_suspend)
> >> -            return 0;
> >> -
> >>      ret = smu_cmn_update_table(smu, SMU_TABLE_OVERDRIVE, 0,
> >> (void *)boot_od_table, false);
> >>      if (ret) {
> >>              dev_err(smu->adev->dev, "Failed to get overdrive table!\n");
> >> @@ -2553,7 +2546,27 @@ static int navi10_set_default_od_settings(struct
> >> smu_context *smu)
> >>      navi10_dump_od_table(smu, boot_od_table);
> >>
> >>      memcpy(od_table, boot_od_table, sizeof(OverDriveTable_t));
> >> -    memcpy(user_od_table, boot_od_table, sizeof(OverDriveTable_t));
> >> +
> >> +    /*
> >> +     * For S3/S4/Runpm resume, we need to setup those overdrive
> >> tables again,
> >> +     * but we have to preserve user defined values in "user_od_table".
> >> +     */
> >> +    if (!smu->adev->in_suspend) {
> >> +            memcpy(user_od_table, boot_od_table,
> >> sizeof(OverDriveTable_t));
> >> +            smu->user_dpm_profile.user_od = false;
> >> +    } else if (smu->user_dpm_profile.user_od) {
> >> +            memcpy(&user_od_table_bak, user_od_table,
> >> sizeof(OverDriveTable_t));
> >> +            memcpy(user_od_table, boot_od_table,
> >> sizeof(OverDriveTable_t));
> >> +            user_od_table->GfxclkFmin =
> >> user_od_table_bak.GfxclkFmin;
> >> +            user_od_table->GfxclkFmax =
> >> user_od_table_bak.GfxclkFmax;
> >> +            user_od_table->UclkFmax = user_od_table_bak.UclkFmax;
> >> +            user_od_table->GfxclkFreq1 =
> >> user_od_table_bak.GfxclkFreq1;
> >> +            user_od_table->GfxclkVolt1 =
> >> user_od_table_bak.GfxclkVolt1;
> >> +            user_od_table->GfxclkFreq2 =
> >> user_od_table_bak.GfxclkFreq2;
> >> +            user_od_table->GfxclkVolt2 =
> >> user_od_table_bak.GfxclkVolt2;
> >> +            user_od_table->GfxclkFreq3 =
> >> user_od_table_bak.GfxclkFreq3;
> >> +            user_od_table->GfxclkVolt3 =
> >> user_od_table_bak.GfxclkVolt3;
> >> +    }
> > Thing is a little tricky for navi10...
> > For navi2x, the vfcurve settings(GfxVfCurve.a, GfxVfCurve.b, GfxVfCurve.c) 
> > are not configurable by user. We do not expose them to user.
> > So, we can just load the new vfcurve settings on resuming back without 
> > worry about overriding user's settings.
> >
> > Unlike navi2x, user can customize the vfcurve settings(by setting 
> > GfxclkFreq/GfxVolt pairs) on navi10. More specifically:
> > - On cold boot, btc was triggered and vfcurve line was calibrated
> > - Driver calculated the target voltage(via 
> > navi10_overdrive_get_gfx_clk_base_voltage) for the point 
> > frequencies(GfxclkFreq1, GfxclkFreq2, GfxclkFreq3) and expose them to user
> >     - e.g. point1 frequency/voltage:  500Mhz/ 0.75v
> > - Then user customized the vfcurve line by setting a new target voltage for 
> > the point frequency.
> >     - e.g. 500Mhz / 0.76v  --> 10mv added
> > - On resuming back, the vfcurve line was recalibrated. The target voltage 
> > for the point1 frequency may be changed to for example 0.745v(for 500Mhz). 
> > Under such scenario, if we just restore user's settings(0.76v which will 
> > add 15mv),  that might not fit user's original intention.
> >
> > So, for this case:
> > 1. Either we add some new variables to record the voltage offset(10mv) from 
> > user's settings. And we restore the target voltage with new vfcurve line 
> > and voltage offset(that is 0.745v + 10mv = 0.755v for the case above). But 
> > still we cannot know whether it truely reflects user's original intention.
> > 2. Or we just restore back to the new vfcurve line calibrated and remind 
> > user that original setting was dropped and they need to set new ones.
> >
>
> As per the current driver logic, user may choose to override (voltage,
> frequency) points to define a new curve. Since the user is defining the
> curve, it's better to restore the same curve.
>
> BTW, this patch doesn't seem to have any effect as the curve will be
> restored properly otherwise also.
>

I'll drop this patch then.

Thanks,

Alex

> Thanks,
> Lijo
>
> > BR
> > Evan
> >>
> >>      return 0;
> >>   }
> >> @@ -2733,6 +2746,20 @@ static int navi10_od_edit_dpm_table(struct
> >> smu_context *smu, enum PP_OD_DPM_TABL
> >>      return ret;
> >>   }
> >>
> >> +static int navi10_restore_user_od_settings(struct smu_context *smu)
> >> +{
> >> +    struct smu_table_context *table_context = &smu->smu_table;
> >> +    OverDriveTable_t *od_table = table_context->overdrive_table;
> >> +    OverDriveTable_t *user_od_table = table_context-
> >>> user_overdrive_table;
> >> +    int res;
> >> +
> >> +    res = smu_v11_0_restore_user_od_settings(smu);
> >> +    if (res == 0)
> >> +            memcpy(od_table, user_od_table,
> >> sizeof(OverDriveTable_t));
> >> +
> >> +    return res;
> >> +}
> >> +
> >>   static int navi10_run_btc(struct smu_context *smu)
> >>   {
> >>      int ret = 0;
> >> @@ -3560,7 +3587,7 @@ static const struct pptable_funcs navi10_ppt_funcs
> >> = {
> >>      .set_soft_freq_limited_range =
> >> smu_v11_0_set_soft_freq_limited_range,
> >>      .set_default_od_settings = navi10_set_default_od_settings,
> >>      .od_edit_dpm_table = navi10_od_edit_dpm_table,
> >> -    .restore_user_od_settings = smu_v11_0_restore_user_od_settings,
> >> +    .restore_user_od_settings = navi10_restore_user_od_settings,
> >>      .run_btc = navi10_run_btc,
> >>      .set_power_source = smu_v11_0_set_power_source,
> >>      .get_pp_feature_mask = smu_cmn_get_pp_feature_mask,
> >> --
> >> 2.39.2

Reply via email to