Commit 488f211d authored by Evan Quan's avatar Evan Quan Committed by Alex Deucher

drm/amd/pm: correct the power limits reporting on OOB supported

As OOB(out-of-band) interface may be used to update the power limits.
Thus to make sure the power limits reporting of our driver always
reflects the correct values, the internal cache must be aligned
carefully.

V2: add support for out-of-band of other ASICs
    align cached current power limit with OOB imposed
Signed-off-by: default avatarEvan Quan <evan.quan@amd.com>
Reviewed-By: default avatarHarish Kasiviswanathan <Harish.Kasiviswanathan@amd.com>
Signed-off-by: default avatarAlex Deucher <alexander.deucher@amd.com>
parent c89d2a2f
...@@ -765,7 +765,10 @@ struct pptable_funcs { ...@@ -765,7 +765,10 @@ struct pptable_funcs {
/** /**
* @get_power_limit: Get the device's power limits. * @get_power_limit: Get the device's power limits.
*/ */
int (*get_power_limit)(struct smu_context *smu); int (*get_power_limit)(struct smu_context *smu,
uint32_t *current_power_limit,
uint32_t *default_power_limit,
uint32_t *max_power_limit);
/** /**
* @get_ppt_limit: Get the device's ppt limits. * @get_ppt_limit: Get the device's ppt limits.
......
...@@ -688,7 +688,10 @@ static int smu_late_init(void *handle) ...@@ -688,7 +688,10 @@ static int smu_late_init(void *handle)
return ret; return ret;
} }
ret = smu_get_asic_power_limits(smu); ret = smu_get_asic_power_limits(smu,
&smu->current_power_limit,
&smu->default_power_limit,
&smu->max_power_limit);
if (ret) { if (ret) {
dev_err(adev->dev, "Failed to get asic power limits!\n"); dev_err(adev->dev, "Failed to get asic power limits!\n");
return ret; return ret;
...@@ -2238,6 +2241,15 @@ int smu_get_power_limit(void *handle, ...@@ -2238,6 +2241,15 @@ int smu_get_power_limit(void *handle,
} else { } else {
switch (limit_level) { switch (limit_level) {
case SMU_PPT_LIMIT_CURRENT: case SMU_PPT_LIMIT_CURRENT:
if ((smu->adev->asic_type == CHIP_ALDEBARAN) ||
(smu->adev->asic_type == CHIP_SIENNA_CICHLID) ||
(smu->adev->asic_type == CHIP_NAVY_FLOUNDER) ||
(smu->adev->asic_type == CHIP_DIMGREY_CAVEFISH) ||
(smu->adev->asic_type == CHIP_BEIGE_GOBY))
ret = smu_get_asic_power_limits(smu,
&smu->current_power_limit,
NULL,
NULL);
*limit = smu->current_power_limit; *limit = smu->current_power_limit;
break; break;
case SMU_PPT_LIMIT_DEFAULT: case SMU_PPT_LIMIT_DEFAULT:
......
...@@ -1194,7 +1194,10 @@ static int arcturus_get_fan_parameters(struct smu_context *smu) ...@@ -1194,7 +1194,10 @@ static int arcturus_get_fan_parameters(struct smu_context *smu)
return 0; return 0;
} }
static int arcturus_get_power_limit(struct smu_context *smu) static int arcturus_get_power_limit(struct smu_context *smu,
uint32_t *current_power_limit,
uint32_t *default_power_limit,
uint32_t *max_power_limit)
{ {
struct smu_11_0_powerplay_table *powerplay_table = struct smu_11_0_powerplay_table *powerplay_table =
(struct smu_11_0_powerplay_table *)smu->smu_table.power_play_table; (struct smu_11_0_powerplay_table *)smu->smu_table.power_play_table;
...@@ -1210,17 +1213,24 @@ static int arcturus_get_power_limit(struct smu_context *smu) ...@@ -1210,17 +1213,24 @@ static int arcturus_get_power_limit(struct smu_context *smu)
power_limit = power_limit =
pptable->SocketPowerLimitAc[PPT_THROTTLER_PPT0]; pptable->SocketPowerLimitAc[PPT_THROTTLER_PPT0];
} }
smu->current_power_limit = smu->default_power_limit = power_limit;
if (smu->od_enabled) { if (current_power_limit)
od_percent = le32_to_cpu(powerplay_table->overdrive_table.max[SMU_11_0_ODSETTING_POWERPERCENTAGE]); *current_power_limit = power_limit;
if (default_power_limit)
*default_power_limit = power_limit;
dev_dbg(smu->adev->dev, "ODSETTING_POWERPERCENTAGE: %d (default: %d)\n", od_percent, power_limit); if (max_power_limit) {
if (smu->od_enabled) {
od_percent = le32_to_cpu(powerplay_table->overdrive_table.max[SMU_11_0_ODSETTING_POWERPERCENTAGE]);
power_limit *= (100 + od_percent); dev_dbg(smu->adev->dev, "ODSETTING_POWERPERCENTAGE: %d (default: %d)\n", od_percent, power_limit);
power_limit /= 100;
power_limit *= (100 + od_percent);
power_limit /= 100;
}
*max_power_limit = power_limit;
} }
smu->max_power_limit = power_limit;
return 0; return 0;
} }
......
...@@ -2136,7 +2136,10 @@ static int navi10_display_disable_memory_clock_switch(struct smu_context *smu, ...@@ -2136,7 +2136,10 @@ static int navi10_display_disable_memory_clock_switch(struct smu_context *smu,
return ret; return ret;
} }
static int navi10_get_power_limit(struct smu_context *smu) static int navi10_get_power_limit(struct smu_context *smu,
uint32_t *current_power_limit,
uint32_t *default_power_limit,
uint32_t *max_power_limit)
{ {
struct smu_11_0_powerplay_table *powerplay_table = struct smu_11_0_powerplay_table *powerplay_table =
(struct smu_11_0_powerplay_table *)smu->smu_table.power_play_table; (struct smu_11_0_powerplay_table *)smu->smu_table.power_play_table;
...@@ -2153,18 +2156,25 @@ static int navi10_get_power_limit(struct smu_context *smu) ...@@ -2153,18 +2156,25 @@ static int navi10_get_power_limit(struct smu_context *smu)
power_limit = power_limit =
pptable->SocketPowerLimitAc[PPT_THROTTLER_PPT0]; pptable->SocketPowerLimitAc[PPT_THROTTLER_PPT0];
} }
smu->current_power_limit = smu->default_power_limit = power_limit;
if (smu->od_enabled && if (current_power_limit)
navi10_od_feature_is_supported(od_settings, SMU_11_0_ODCAP_POWER_LIMIT)) { *current_power_limit = power_limit;
od_percent = le32_to_cpu(powerplay_table->overdrive_table.max[SMU_11_0_ODSETTING_POWERPERCENTAGE]); if (default_power_limit)
*default_power_limit = power_limit;
dev_dbg(smu->adev->dev, "ODSETTING_POWERPERCENTAGE: %d (default: %d)\n", od_percent, power_limit); if (max_power_limit) {
if (smu->od_enabled &&
navi10_od_feature_is_supported(od_settings, SMU_11_0_ODCAP_POWER_LIMIT)) {
od_percent = le32_to_cpu(powerplay_table->overdrive_table.max[SMU_11_0_ODSETTING_POWERPERCENTAGE]);
power_limit *= (100 + od_percent); dev_dbg(smu->adev->dev, "ODSETTING_POWERPERCENTAGE: %d (default: %d)\n", od_percent, power_limit);
power_limit /= 100;
power_limit *= (100 + od_percent);
power_limit /= 100;
}
*max_power_limit = power_limit;
} }
smu->max_power_limit = power_limit;
return 0; return 0;
} }
......
...@@ -1791,7 +1791,10 @@ static int sienna_cichlid_display_disable_memory_clock_switch(struct smu_context ...@@ -1791,7 +1791,10 @@ static int sienna_cichlid_display_disable_memory_clock_switch(struct smu_context
return ret; return ret;
} }
static int sienna_cichlid_get_power_limit(struct smu_context *smu) static int sienna_cichlid_get_power_limit(struct smu_context *smu,
uint32_t *current_power_limit,
uint32_t *default_power_limit,
uint32_t *max_power_limit)
{ {
struct smu_11_0_7_powerplay_table *powerplay_table = struct smu_11_0_7_powerplay_table *powerplay_table =
(struct smu_11_0_7_powerplay_table *)smu->smu_table.power_play_table; (struct smu_11_0_7_powerplay_table *)smu->smu_table.power_play_table;
...@@ -1804,17 +1807,23 @@ static int sienna_cichlid_get_power_limit(struct smu_context *smu) ...@@ -1804,17 +1807,23 @@ static int sienna_cichlid_get_power_limit(struct smu_context *smu)
power_limit = power_limit =
table_member[PPT_THROTTLER_PPT0]; table_member[PPT_THROTTLER_PPT0];
} }
smu->current_power_limit = smu->default_power_limit = power_limit;
if (smu->od_enabled) { if (current_power_limit)
od_percent = le32_to_cpu(powerplay_table->overdrive_table.max[SMU_11_0_7_ODSETTING_POWERPERCENTAGE]); *current_power_limit = power_limit;
if (default_power_limit)
*default_power_limit = power_limit;
dev_dbg(smu->adev->dev, "ODSETTING_POWERPERCENTAGE: %d (default: %d)\n", od_percent, power_limit); if (max_power_limit) {
if (smu->od_enabled) {
od_percent = le32_to_cpu(powerplay_table->overdrive_table.max[SMU_11_0_7_ODSETTING_POWERPERCENTAGE]);
power_limit *= (100 + od_percent); dev_dbg(smu->adev->dev, "ODSETTING_POWERPERCENTAGE: %d (default: %d)\n", od_percent, power_limit);
power_limit /= 100;
power_limit *= (100 + od_percent);
power_limit /= 100;
}
*max_power_limit = power_limit;
} }
smu->max_power_limit = power_limit;
return 0; return 0;
} }
......
...@@ -2071,7 +2071,10 @@ static int vangogh_mode2_reset(struct smu_context *smu) ...@@ -2071,7 +2071,10 @@ static int vangogh_mode2_reset(struct smu_context *smu)
return vangogh_mode_reset(smu, SMU_RESET_MODE_2); return vangogh_mode_reset(smu, SMU_RESET_MODE_2);
} }
static int vangogh_get_power_limit(struct smu_context *smu) static int vangogh_get_power_limit(struct smu_context *smu,
uint32_t *current_power_limit,
uint32_t *default_power_limit,
uint32_t *max_power_limit)
{ {
struct smu_11_5_power_context *power_context = struct smu_11_5_power_context *power_context =
smu->smu_power.power_context; smu->smu_power.power_context;
...@@ -2087,8 +2090,12 @@ static int vangogh_get_power_limit(struct smu_context *smu) ...@@ -2087,8 +2090,12 @@ static int vangogh_get_power_limit(struct smu_context *smu)
return ret; return ret;
} }
/* convert from milliwatt to watt */ /* convert from milliwatt to watt */
smu->current_power_limit = smu->default_power_limit = ppt_limit / 1000; if (current_power_limit)
smu->max_power_limit = 29; *current_power_limit = ppt_limit / 1000;
if (default_power_limit)
*default_power_limit = ppt_limit / 1000;
if (max_power_limit)
*max_power_limit = 29;
ret = smu_cmn_send_smc_msg(smu, SMU_MSG_GetFastPPTLimit, &ppt_limit); ret = smu_cmn_send_smc_msg(smu, SMU_MSG_GetFastPPTLimit, &ppt_limit);
if (ret) { if (ret) {
......
...@@ -1146,7 +1146,10 @@ static int aldebaran_read_sensor(struct smu_context *smu, ...@@ -1146,7 +1146,10 @@ static int aldebaran_read_sensor(struct smu_context *smu,
return ret; return ret;
} }
static int aldebaran_get_power_limit(struct smu_context *smu) static int aldebaran_get_power_limit(struct smu_context *smu,
uint32_t *current_power_limit,
uint32_t *default_power_limit,
uint32_t *max_power_limit)
{ {
PPTable_t *pptable = smu->smu_table.driver_pptable; PPTable_t *pptable = smu->smu_table.driver_pptable;
uint32_t power_limit = 0; uint32_t power_limit = 0;
...@@ -1166,9 +1169,15 @@ static int aldebaran_get_power_limit(struct smu_context *smu) ...@@ -1166,9 +1169,15 @@ static int aldebaran_get_power_limit(struct smu_context *smu)
power_limit = pptable->PptLimit; power_limit = pptable->PptLimit;
} }
smu->current_power_limit = smu->default_power_limit = power_limit; if (current_power_limit)
if (pptable) *current_power_limit = power_limit;
smu->max_power_limit = pptable->PptLimit; if (default_power_limit)
*default_power_limit = power_limit;
if (max_power_limit) {
if (pptable)
*max_power_limit = pptable->PptLimit;
}
return 0; return 0;
} }
......
...@@ -82,7 +82,7 @@ ...@@ -82,7 +82,7 @@
#define smu_i2c_fini(smu, control) smu_ppt_funcs(i2c_fini, 0, smu, control) #define smu_i2c_fini(smu, control) smu_ppt_funcs(i2c_fini, 0, smu, control)
#define smu_get_unique_id(smu) smu_ppt_funcs(get_unique_id, 0, smu) #define smu_get_unique_id(smu) smu_ppt_funcs(get_unique_id, 0, smu)
#define smu_log_thermal_throttling(smu) smu_ppt_funcs(log_thermal_throttling_event, 0, smu) #define smu_log_thermal_throttling(smu) smu_ppt_funcs(log_thermal_throttling_event, 0, smu)
#define smu_get_asic_power_limits(smu) smu_ppt_funcs(get_power_limit, 0, smu) #define smu_get_asic_power_limits(smu, current, default, max) smu_ppt_funcs(get_power_limit, 0, smu, current, default, max)
#define smu_get_pp_feature_mask(smu, buf) smu_ppt_funcs(get_pp_feature_mask, 0, smu, buf) #define smu_get_pp_feature_mask(smu, buf) smu_ppt_funcs(get_pp_feature_mask, 0, smu, buf)
#define smu_set_pp_feature_mask(smu, new_mask) smu_ppt_funcs(set_pp_feature_mask, 0, smu, new_mask) #define smu_set_pp_feature_mask(smu, new_mask) smu_ppt_funcs(set_pp_feature_mask, 0, smu, new_mask)
#define smu_gfx_ulv_control(smu, enablement) smu_ppt_funcs(gfx_ulv_control, 0, smu, enablement) #define smu_gfx_ulv_control(smu, enablement) smu_ppt_funcs(gfx_ulv_control, 0, smu, enablement)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment