diff options
Diffstat (limited to 'common/recipes-kernel/linux/linux-yocto-4.14.71/4818-drm-amd-powerplay-revise-default-dpm-tables-setup.patch')
-rw-r--r-- | common/recipes-kernel/linux/linux-yocto-4.14.71/4818-drm-amd-powerplay-revise-default-dpm-tables-setup.patch | 424 |
1 files changed, 424 insertions, 0 deletions
diff --git a/common/recipes-kernel/linux/linux-yocto-4.14.71/4818-drm-amd-powerplay-revise-default-dpm-tables-setup.patch b/common/recipes-kernel/linux/linux-yocto-4.14.71/4818-drm-amd-powerplay-revise-default-dpm-tables-setup.patch new file mode 100644 index 00000000..8818771b --- /dev/null +++ b/common/recipes-kernel/linux/linux-yocto-4.14.71/4818-drm-amd-powerplay-revise-default-dpm-tables-setup.patch @@ -0,0 +1,424 @@ +From ae94f990c469147a418075791a822e5094b7b4f5 Mon Sep 17 00:00:00 2001 +From: Evan Quan <evan.quan@amd.com> +Date: Mon, 11 Jun 2018 15:25:37 +0800 +Subject: [PATCH 4818/5725] drm/amd/powerplay: revise default dpm tables setup + +Initialize the soft/hard min/max level correctly and +handle the dpm disabled situation. + +Signed-off-by: Evan Quan <evan.quan@amd.com> +Acked-by: Alex Deucher <alexander.deucher@amd.com> +Signed-off-by: Alex Deucher <alexander.deucher@amd.com> +--- + drivers/gpu/drm/amd/powerplay/hwmgr/vega12_hwmgr.c | 334 ++++++++------------- + 1 file changed, 132 insertions(+), 202 deletions(-) + +diff --git a/drivers/gpu/drm/amd/powerplay/hwmgr/vega12_hwmgr.c b/drivers/gpu/drm/amd/powerplay/hwmgr/vega12_hwmgr.c +index 6e22cb3..8b5c581 100644 +--- a/drivers/gpu/drm/amd/powerplay/hwmgr/vega12_hwmgr.c ++++ b/drivers/gpu/drm/amd/powerplay/hwmgr/vega12_hwmgr.c +@@ -454,37 +454,30 @@ static int vega12_setup_asic_task(struct pp_hwmgr *hwmgr) + */ + static void vega12_init_dpm_state(struct vega12_dpm_state *dpm_state) + { +- dpm_state->soft_min_level = 0xff; +- dpm_state->soft_max_level = 0xff; +- dpm_state->hard_min_level = 0xff; +- dpm_state->hard_max_level = 0xff; ++ dpm_state->soft_min_level = 0x0; ++ dpm_state->soft_max_level = 0xffff; ++ dpm_state->hard_min_level = 0x0; ++ dpm_state->hard_max_level = 0xffff; + } + +-static int vega12_get_number_dpm_level(struct pp_hwmgr *hwmgr, +- PPCLK_e clkID, uint32_t *num_dpm_level) ++static int vega12_get_number_of_dpm_level(struct pp_hwmgr *hwmgr, ++ PPCLK_e clk_id, uint32_t *num_of_levels) + { +- int result; +- /* +- * SMU expects the Clock ID to be in the top 16 bits. +- * Lower 16 bits specify the level however 0xFF is a +- * special argument the returns the total number of levels +- */ +- PP_ASSERT_WITH_CODE(smum_send_msg_to_smc_with_parameter(hwmgr, +- PPSMC_MSG_GetDpmFreqByIndex, (clkID << 16 | 0xFF)) == 0, +- "[GetNumberDpmLevel] Failed to get DPM levels from SMU for CLKID!", +- return -EINVAL); +- +- result = vega12_read_arg_from_smc(hwmgr, num_dpm_level); ++ int ret = 0; + +- PP_ASSERT_WITH_CODE(*num_dpm_level < MAX_REGULAR_DPM_NUMBER, +- "[GetNumberDPMLevel] Number of DPM levels is greater than limit", +- return -EINVAL); ++ ret = smum_send_msg_to_smc_with_parameter(hwmgr, ++ PPSMC_MSG_GetDpmFreqByIndex, ++ (clk_id << 16 | 0xFF)); ++ PP_ASSERT_WITH_CODE(!ret, ++ "[GetNumOfDpmLevel] failed to get dpm levels!", ++ return ret); + +- PP_ASSERT_WITH_CODE(*num_dpm_level != 0, +- "[GetNumberDPMLevel] Number of CLK Levels is zero!", +- return -EINVAL); ++ vega12_read_arg_from_smc(hwmgr, num_of_levels); ++ PP_ASSERT_WITH_CODE(*num_of_levels > 0, ++ "[GetNumOfDpmLevel] number of clk levels is invalid!", ++ return -EINVAL); + +- return result; ++ return ret; + } + + static int vega12_get_dpm_frequency_by_index(struct pp_hwmgr *hwmgr, +@@ -510,6 +503,31 @@ static int vega12_get_dpm_frequency_by_index(struct pp_hwmgr *hwmgr, + return result; + } + ++static int vega12_setup_single_dpm_table(struct pp_hwmgr *hwmgr, ++ struct vega12_single_dpm_table *dpm_table, PPCLK_e clk_id) ++{ ++ int ret = 0; ++ uint32_t i, num_of_levels, clk; ++ ++ ret = vega12_get_number_of_dpm_level(hwmgr, clk_id, &num_of_levels); ++ PP_ASSERT_WITH_CODE(!ret, ++ "[SetupSingleDpmTable] failed to get clk levels!", ++ return ret); ++ ++ dpm_table->count = num_of_levels; ++ ++ for (i = 0; i < num_of_levels; i++) { ++ ret = vega12_get_dpm_frequency_by_index(hwmgr, clk_id, i, &clk); ++ PP_ASSERT_WITH_CODE(!ret, ++ "[SetupSingleDpmTable] failed to get clk of specific level!", ++ return ret); ++ dpm_table->dpm_levels[i].value = clk; ++ dpm_table->dpm_levels[i].enabled = true; ++ } ++ ++ return ret; ++} ++ + /* + * This function is to initialize all DPM state tables + * for SMU based on the dependency table. +@@ -520,224 +538,136 @@ static int vega12_get_dpm_frequency_by_index(struct pp_hwmgr *hwmgr, + */ + static int vega12_setup_default_dpm_tables(struct pp_hwmgr *hwmgr) + { +- uint32_t num_levels, i, clock; + + struct vega12_hwmgr *data = + (struct vega12_hwmgr *)(hwmgr->backend); +- + struct vega12_single_dpm_table *dpm_table; ++ int ret = 0; + + memset(&data->dpm_table, 0, sizeof(data->dpm_table)); + +- /* Initialize Sclk DPM and SOC DPM table based on allow Sclk values */ ++ /* socclk */ + dpm_table = &(data->dpm_table.soc_table); +- +- PP_ASSERT_WITH_CODE(vega12_get_number_dpm_level(hwmgr, PPCLK_SOCCLK, +- &num_levels) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for SOCCLK!", +- return -EINVAL); +- +- dpm_table->count = num_levels; +- +- for (i = 0; i < num_levels; i++) { +- PP_ASSERT_WITH_CODE(vega12_get_dpm_frequency_by_index(hwmgr, +- PPCLK_SOCCLK, i, &clock) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for SOCCLK!", +- return -EINVAL); +- +- dpm_table->dpm_levels[i].value = clock; +- dpm_table->dpm_levels[i].enabled = true; ++ if (data->smu_features[GNLD_DPM_SOCCLK].enabled) { ++ ret = vega12_setup_single_dpm_table(hwmgr, dpm_table, PPCLK_SOCCLK); ++ PP_ASSERT_WITH_CODE(!ret, ++ "[SetupDefaultDpmTable] failed to get socclk dpm levels!", ++ return ret); ++ } else { ++ dpm_table->count = 1; ++ dpm_table->dpm_levels[0].value = data->vbios_boot_state.soc_clock / 100; + } +- + vega12_init_dpm_state(&(dpm_table->dpm_state)); + ++ /* gfxclk */ + dpm_table = &(data->dpm_table.gfx_table); +- +- PP_ASSERT_WITH_CODE(vega12_get_number_dpm_level(hwmgr, PPCLK_GFXCLK, +- &num_levels) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for GFXCLK!", +- return -EINVAL); +- +- dpm_table->count = num_levels; +- for (i = 0; i < num_levels; i++) { +- PP_ASSERT_WITH_CODE(vega12_get_dpm_frequency_by_index(hwmgr, +- PPCLK_GFXCLK, i, &clock) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for GFXCLK!", +- return -EINVAL); +- +- dpm_table->dpm_levels[i].value = clock; +- dpm_table->dpm_levels[i].enabled = true; ++ if (data->smu_features[GNLD_DPM_GFXCLK].enabled) { ++ ret = vega12_setup_single_dpm_table(hwmgr, dpm_table, PPCLK_GFXCLK); ++ PP_ASSERT_WITH_CODE(!ret, ++ "[SetupDefaultDpmTable] failed to get gfxclk dpm levels!", ++ return ret); ++ } else { ++ dpm_table->count = 1; ++ dpm_table->dpm_levels[0].value = data->vbios_boot_state.gfx_clock / 100; + } +- + vega12_init_dpm_state(&(dpm_table->dpm_state)); +- /* Initialize Mclk DPM table based on allow Mclk values */ +- dpm_table = &(data->dpm_table.mem_table); + +- PP_ASSERT_WITH_CODE(vega12_get_number_dpm_level(hwmgr, PPCLK_UCLK, +- &num_levels) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for UCLK!", +- return -EINVAL); +- +- dpm_table->count = num_levels; +- +- for (i = 0; i < num_levels; i++) { +- PP_ASSERT_WITH_CODE(vega12_get_dpm_frequency_by_index(hwmgr, +- PPCLK_UCLK, i, &clock) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for UCLK!", +- return -EINVAL); +- +- dpm_table->dpm_levels[i].value = clock; +- dpm_table->dpm_levels[i].enabled = true; ++ /* memclk */ ++ dpm_table = &(data->dpm_table.mem_table); ++ if (data->smu_features[GNLD_DPM_UCLK].enabled) { ++ ret = vega12_setup_single_dpm_table(hwmgr, dpm_table, PPCLK_UCLK); ++ PP_ASSERT_WITH_CODE(!ret, ++ "[SetupDefaultDpmTable] failed to get memclk dpm levels!", ++ return ret); ++ } else { ++ dpm_table->count = 1; ++ dpm_table->dpm_levels[0].value = data->vbios_boot_state.mem_clock / 100; + } +- + vega12_init_dpm_state(&(dpm_table->dpm_state)); + ++ /* eclk */ + dpm_table = &(data->dpm_table.eclk_table); +- +- PP_ASSERT_WITH_CODE(vega12_get_number_dpm_level(hwmgr, PPCLK_ECLK, +- &num_levels) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for ECLK!", +- return -EINVAL); +- +- dpm_table->count = num_levels; +- +- for (i = 0; i < num_levels; i++) { +- PP_ASSERT_WITH_CODE(vega12_get_dpm_frequency_by_index(hwmgr, +- PPCLK_ECLK, i, &clock) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for ECLK!", +- return -EINVAL); +- +- dpm_table->dpm_levels[i].value = clock; +- dpm_table->dpm_levels[i].enabled = true; ++ if (data->smu_features[GNLD_DPM_VCE].enabled) { ++ ret = vega12_setup_single_dpm_table(hwmgr, dpm_table, PPCLK_ECLK); ++ PP_ASSERT_WITH_CODE(!ret, ++ "[SetupDefaultDpmTable] failed to get eclk dpm levels!", ++ return ret); ++ } else { ++ dpm_table->count = 1; ++ dpm_table->dpm_levels[0].value = data->vbios_boot_state.eclock / 100; + } +- + vega12_init_dpm_state(&(dpm_table->dpm_state)); + ++ /* vclk */ + dpm_table = &(data->dpm_table.vclk_table); +- +- PP_ASSERT_WITH_CODE(vega12_get_number_dpm_level(hwmgr, PPCLK_VCLK, +- &num_levels) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for VCLK!", +- return -EINVAL); +- +- dpm_table->count = num_levels; +- +- for (i = 0; i < num_levels; i++) { +- PP_ASSERT_WITH_CODE(vega12_get_dpm_frequency_by_index(hwmgr, +- PPCLK_VCLK, i, &clock) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for VCLK!", +- return -EINVAL); +- +- dpm_table->dpm_levels[i].value = clock; +- dpm_table->dpm_levels[i].enabled = true; ++ if (data->smu_features[GNLD_DPM_UVD].enabled) { ++ ret = vega12_setup_single_dpm_table(hwmgr, dpm_table, PPCLK_VCLK); ++ PP_ASSERT_WITH_CODE(!ret, ++ "[SetupDefaultDpmTable] failed to get vclk dpm levels!", ++ return ret); ++ } else { ++ dpm_table->count = 1; ++ dpm_table->dpm_levels[0].value = data->vbios_boot_state.vclock / 100; + } +- + vega12_init_dpm_state(&(dpm_table->dpm_state)); + ++ /* dclk */ + dpm_table = &(data->dpm_table.dclk_table); +- +- PP_ASSERT_WITH_CODE(vega12_get_number_dpm_level(hwmgr, PPCLK_DCLK, +- &num_levels) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for DCLK!", +- return -EINVAL); +- +- dpm_table->count = num_levels; +- +- for (i = 0; i < num_levels; i++) { +- PP_ASSERT_WITH_CODE(vega12_get_dpm_frequency_by_index(hwmgr, +- PPCLK_DCLK, i, &clock) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for DCLK!", +- return -EINVAL); +- +- dpm_table->dpm_levels[i].value = clock; +- dpm_table->dpm_levels[i].enabled = true; ++ if (data->smu_features[GNLD_DPM_UVD].enabled) { ++ ret = vega12_setup_single_dpm_table(hwmgr, dpm_table, PPCLK_DCLK); ++ PP_ASSERT_WITH_CODE(!ret, ++ "[SetupDefaultDpmTable] failed to get dclk dpm levels!", ++ return ret); ++ } else { ++ dpm_table->count = 1; ++ dpm_table->dpm_levels[0].value = data->vbios_boot_state.dclock / 100; + } +- + vega12_init_dpm_state(&(dpm_table->dpm_state)); + +- /* Assume there is no headless Vega12 for now */ ++ /* dcefclk */ + dpm_table = &(data->dpm_table.dcef_table); +- +- PP_ASSERT_WITH_CODE(vega12_get_number_dpm_level(hwmgr, +- PPCLK_DCEFCLK, &num_levels) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for DCEFCLK!", +- return -EINVAL); +- +- dpm_table->count = num_levels; +- +- for (i = 0; i < num_levels; i++) { +- PP_ASSERT_WITH_CODE(vega12_get_dpm_frequency_by_index(hwmgr, +- PPCLK_DCEFCLK, i, &clock) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for DCEFCLK!", +- return -EINVAL); +- +- dpm_table->dpm_levels[i].value = clock; +- dpm_table->dpm_levels[i].enabled = true; ++ if (data->smu_features[GNLD_DPM_DCEFCLK].enabled) { ++ ret = vega12_setup_single_dpm_table(hwmgr, dpm_table, PPCLK_DCEFCLK); ++ PP_ASSERT_WITH_CODE(!ret, ++ "[SetupDefaultDpmTable] failed to get dcefclk dpm levels!", ++ return ret); ++ } else { ++ dpm_table->count = 1; ++ dpm_table->dpm_levels[0].value = data->vbios_boot_state.dcef_clock / 100; + } +- + vega12_init_dpm_state(&(dpm_table->dpm_state)); + ++ /* pixclk */ + dpm_table = &(data->dpm_table.pixel_table); +- +- PP_ASSERT_WITH_CODE(vega12_get_number_dpm_level(hwmgr, +- PPCLK_PIXCLK, &num_levels) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for PIXCLK!", +- return -EINVAL); +- +- dpm_table->count = num_levels; +- +- for (i = 0; i < num_levels; i++) { +- PP_ASSERT_WITH_CODE(vega12_get_dpm_frequency_by_index(hwmgr, +- PPCLK_PIXCLK, i, &clock) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for PIXCLK!", +- return -EINVAL); +- +- dpm_table->dpm_levels[i].value = clock; +- dpm_table->dpm_levels[i].enabled = true; +- } +- ++ if (data->smu_features[GNLD_DPM_DCEFCLK].enabled) { ++ ret = vega12_setup_single_dpm_table(hwmgr, dpm_table, PPCLK_PIXCLK); ++ PP_ASSERT_WITH_CODE(!ret, ++ "[SetupDefaultDpmTable] failed to get pixclk dpm levels!", ++ return ret); ++ } else ++ dpm_table->count = 0; + vega12_init_dpm_state(&(dpm_table->dpm_state)); + ++ /* dispclk */ + dpm_table = &(data->dpm_table.display_table); +- +- PP_ASSERT_WITH_CODE(vega12_get_number_dpm_level(hwmgr, +- PPCLK_DISPCLK, &num_levels) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for DISPCLK!", +- return -EINVAL); +- +- dpm_table->count = num_levels; +- +- for (i = 0; i < num_levels; i++) { +- PP_ASSERT_WITH_CODE(vega12_get_dpm_frequency_by_index(hwmgr, +- PPCLK_DISPCLK, i, &clock) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for DISPCLK!", +- return -EINVAL); +- +- dpm_table->dpm_levels[i].value = clock; +- dpm_table->dpm_levels[i].enabled = true; +- } +- ++ if (data->smu_features[GNLD_DPM_DCEFCLK].enabled) { ++ ret = vega12_setup_single_dpm_table(hwmgr, dpm_table, PPCLK_DISPCLK); ++ PP_ASSERT_WITH_CODE(!ret, ++ "[SetupDefaultDpmTable] failed to get dispclk dpm levels!", ++ return ret); ++ } else ++ dpm_table->count = 0; + vega12_init_dpm_state(&(dpm_table->dpm_state)); + ++ /* phyclk */ + dpm_table = &(data->dpm_table.phy_table); +- +- PP_ASSERT_WITH_CODE(vega12_get_number_dpm_level(hwmgr, +- PPCLK_PHYCLK, &num_levels) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for PHYCLK!", +- return -EINVAL); +- +- dpm_table->count = num_levels; +- +- for (i = 0; i < num_levels; i++) { +- PP_ASSERT_WITH_CODE(vega12_get_dpm_frequency_by_index(hwmgr, +- PPCLK_PHYCLK, i, &clock) == 0, +- "[SetupDefaultDPMTables] Failed to get DPM levels from SMU for PHYCLK!", +- return -EINVAL); +- +- dpm_table->dpm_levels[i].value = clock; +- dpm_table->dpm_levels[i].enabled = true; +- } +- ++ if (data->smu_features[GNLD_DPM_DCEFCLK].enabled) { ++ ret = vega12_setup_single_dpm_table(hwmgr, dpm_table, PPCLK_PHYCLK); ++ PP_ASSERT_WITH_CODE(!ret, ++ "[SetupDefaultDpmTable] failed to get phyclk dpm levels!", ++ return ret); ++ } else ++ dpm_table->count = 0; + vega12_init_dpm_state(&(dpm_table->dpm_state)); + + /* save a copy of the default DPM table */ +-- +2.7.4 + |