aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorRex Zhu <Rex.Zhu@amd.com>2016-06-07 06:39:06 -0400
committerAlex Deucher <alexander.deucher@amd.com>2016-06-21 10:22:42 -0400
commit270d013659ddab52a6fd0eacae452c422d08aa39 (patch)
tree48cb2696e73000bdbaca57c452de782d69bbf223
parent92d1576859577b94eafaea9b64f78ab99fe20a78 (diff)
drm/amd/powerplay: enable clock stretch feature for polaris
Power saving feature which reduces the amount of voltage needed for specific engine clocks. Signed-off-by: Rex Zhu <Rex.Zhu@amd.com> Reviewed-by: Alex Deucher <alexander.deucher@amd.com> Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
-rw-r--r--drivers/gpu/drm/amd/powerplay/hwmgr/polaris10_hwmgr.c120
1 files changed, 23 insertions, 97 deletions
diff --git a/drivers/gpu/drm/amd/powerplay/hwmgr/polaris10_hwmgr.c b/drivers/gpu/drm/amd/powerplay/hwmgr/polaris10_hwmgr.c
index f730ec8b529d..64ee78f7d41e 100644
--- a/drivers/gpu/drm/amd/powerplay/hwmgr/polaris10_hwmgr.c
+++ b/drivers/gpu/drm/amd/powerplay/hwmgr/polaris10_hwmgr.c
@@ -1759,12 +1759,9 @@ static int polaris10_populate_smc_initailial_state(struct pp_hwmgr *hwmgr)
1759 1759
1760static int polaris10_populate_clock_stretcher_data_table(struct pp_hwmgr *hwmgr) 1760static int polaris10_populate_clock_stretcher_data_table(struct pp_hwmgr *hwmgr)
1761{ 1761{
1762 uint32_t ro, efuse, efuse2, clock_freq, volt_without_cks, 1762 uint32_t ro, efuse, volt_without_cks, volt_with_cks, value, max, min;
1763 volt_with_cks, value;
1764 uint16_t clock_freq_u16;
1765 struct polaris10_hwmgr *data = (struct polaris10_hwmgr *)(hwmgr->backend); 1763 struct polaris10_hwmgr *data = (struct polaris10_hwmgr *)(hwmgr->backend);
1766 uint8_t type, i, j, cks_setting, stretch_amount, stretch_amount2, 1764 uint8_t i, stretch_amount, stretch_amount2, volt_offset = 0;
1767 volt_offset = 0;
1768 struct phm_ppt_v1_information *table_info = 1765 struct phm_ppt_v1_information *table_info =
1769 (struct phm_ppt_v1_information *)(hwmgr->pptable); 1766 (struct phm_ppt_v1_information *)(hwmgr->pptable);
1770 struct phm_ppt_v1_clock_voltage_dependency_table *sclk_table = 1767 struct phm_ppt_v1_clock_voltage_dependency_table *sclk_table =
@@ -1776,50 +1773,38 @@ static int polaris10_populate_clock_stretcher_data_table(struct pp_hwmgr *hwmgr)
1776 * if the part is SS or FF. if RO >= 1660MHz, part is FF. 1773 * if the part is SS or FF. if RO >= 1660MHz, part is FF.
1777 */ 1774 */
1778 efuse = cgs_read_ind_register(hwmgr->device, CGS_IND_REG__SMC, 1775 efuse = cgs_read_ind_register(hwmgr->device, CGS_IND_REG__SMC,
1779 ixSMU_EFUSE_0 + (146 * 4)); 1776 ixSMU_EFUSE_0 + (67 * 4));
1780 efuse2 = cgs_read_ind_register(hwmgr->device, CGS_IND_REG__SMC,
1781 ixSMU_EFUSE_0 + (148 * 4));
1782 efuse &= 0xFF000000; 1777 efuse &= 0xFF000000;
1783 efuse = efuse >> 24; 1778 efuse = efuse >> 24;
1784 efuse2 &= 0xF;
1785 1779
1786 if (efuse2 == 1) 1780 if (hwmgr->chip_id == CHIP_POLARIS10) {
1787 ro = (2300 - 1350) * efuse / 255 + 1350; 1781 min = 1000;
1788 else 1782 max = 2300;
1789 ro = (2500 - 1000) * efuse / 255 + 1000; 1783 } else {
1790 1784 min = 1100;
1791 if (ro >= 1660) 1785 max = 2100;
1792 type = 0; 1786 }
1793 else
1794 type = 1;
1795 1787
1796 /* Populate Stretch amount */ 1788 ro = efuse * (max -min)/255 + min;
1797 data->smc_state_table.ClockStretcherAmount = stretch_amount;
1798 1789
1799 /* Populate Sclk_CKS_masterEn0_7 and Sclk_voltageOffset */ 1790 /* Populate Sclk_CKS_masterEn0_7 and Sclk_voltageOffset */
1800 for (i = 0; i < sclk_table->count; i++) { 1791 for (i = 0; i < sclk_table->count; i++) {
1801 data->smc_state_table.Sclk_CKS_masterEn0_7 |= 1792 data->smc_state_table.Sclk_CKS_masterEn0_7 |=
1802 sclk_table->entries[i].cks_enable << i; 1793 sclk_table->entries[i].cks_enable << i;
1803 volt_without_cks = (uint32_t)((14041 * 1794
1804 (sclk_table->entries[i].clk/100) / 10000 + 3571 + 75 - ro) * 1000 / 1795 volt_without_cks = (uint32_t)(((ro - 40) * 1000 - 2753594 - sclk_table->entries[i].clk/100 * 136418 /1000) / \
1805 (4026 - (13924 * (sclk_table->entries[i].clk/100) / 10000))); 1796 (sclk_table->entries[i].clk/100 * 1132925 /10000 - 242418)/100);
1806 volt_with_cks = (uint32_t)((13946 * 1797
1807 (sclk_table->entries[i].clk/100) / 10000 + 3320 + 45 - ro) * 1000 / 1798 volt_with_cks = (uint32_t)((ro * 1000 -2396351 - sclk_table->entries[i].clk/100 * 329021/1000) / \
1808 (3664 - (11454 * (sclk_table->entries[i].clk/100) / 10000))); 1799 (sclk_table->entries[i].clk/10000 * 649434 /1000 - 18005)/10);
1800
1809 if (volt_without_cks >= volt_with_cks) 1801 if (volt_without_cks >= volt_with_cks)
1810 volt_offset = (uint8_t)(((volt_without_cks - volt_with_cks + 1802 volt_offset = (uint8_t)(((volt_without_cks - volt_with_cks +
1811 sclk_table->entries[i].cks_voffset) * 100 / 625) + 1); 1803 sclk_table->entries[i].cks_voffset) * 100 / 625) + 1);
1804
1812 data->smc_state_table.Sclk_voltageOffset[i] = volt_offset; 1805 data->smc_state_table.Sclk_voltageOffset[i] = volt_offset;
1813 } 1806 }
1814 1807
1815 PHM_WRITE_INDIRECT_FIELD(hwmgr->device, CGS_IND_REG__SMC, PWR_CKS_ENABLE,
1816 STRETCH_ENABLE, 0x0);
1817 PHM_WRITE_INDIRECT_FIELD(hwmgr->device, CGS_IND_REG__SMC, PWR_CKS_ENABLE,
1818 masterReset, 0x1);
1819 /* PHM_WRITE_INDIRECT_FIELD(hwmgr->device, CGS_IND_REG__SMC, PWR_CKS_ENABLE, staticEnable, 0x1); */
1820 PHM_WRITE_INDIRECT_FIELD(hwmgr->device, CGS_IND_REG__SMC, PWR_CKS_ENABLE,
1821 masterReset, 0x0);
1822
1823 /* Populate CKS Lookup Table */ 1808 /* Populate CKS Lookup Table */
1824 if (stretch_amount == 1 || stretch_amount == 2 || stretch_amount == 5) 1809 if (stretch_amount == 1 || stretch_amount == 2 || stretch_amount == 5)
1825 stretch_amount2 = 0; 1810 stretch_amount2 = 0;
@@ -1833,69 +1818,6 @@ static int polaris10_populate_clock_stretcher_data_table(struct pp_hwmgr *hwmgr)
1833 return -EINVAL); 1818 return -EINVAL);
1834 } 1819 }
1835 1820
1836 value = cgs_read_ind_register(hwmgr->device, CGS_IND_REG__SMC,
1837 ixPWR_CKS_CNTL);
1838 value &= 0xFFC2FF87;
1839 data->smc_state_table.CKS_LOOKUPTable.CKS_LOOKUPTableEntry[0].minFreq =
1840 polaris10_clock_stretcher_lookup_table[stretch_amount2][0];
1841 data->smc_state_table.CKS_LOOKUPTable.CKS_LOOKUPTableEntry[0].maxFreq =
1842 polaris10_clock_stretcher_lookup_table[stretch_amount2][1];
1843 clock_freq_u16 = (uint16_t)(PP_SMC_TO_HOST_UL(data->smc_state_table.
1844 GraphicsLevel[data->smc_state_table.GraphicsDpmLevelCount - 1].SclkSetting.SclkFrequency) / 100);
1845 if (polaris10_clock_stretcher_lookup_table[stretch_amount2][0] < clock_freq_u16
1846 && polaris10_clock_stretcher_lookup_table[stretch_amount2][1] > clock_freq_u16) {
1847 /* Program PWR_CKS_CNTL. CKS_USE_FOR_LOW_FREQ */
1848 value |= (polaris10_clock_stretcher_lookup_table[stretch_amount2][3]) << 16;
1849 /* Program PWR_CKS_CNTL. CKS_LDO_REFSEL */
1850 value |= (polaris10_clock_stretcher_lookup_table[stretch_amount2][2]) << 18;
1851 /* Program PWR_CKS_CNTL. CKS_STRETCH_AMOUNT */
1852 value |= (polaris10_clock_stretch_amount_conversion
1853 [polaris10_clock_stretcher_lookup_table[stretch_amount2][3]]
1854 [stretch_amount]) << 3;
1855 }
1856 CONVERT_FROM_HOST_TO_SMC_US(data->smc_state_table.CKS_LOOKUPTable.CKS_LOOKUPTableEntry[0].minFreq);
1857 CONVERT_FROM_HOST_TO_SMC_US(data->smc_state_table.CKS_LOOKUPTable.CKS_LOOKUPTableEntry[0].maxFreq);
1858 data->smc_state_table.CKS_LOOKUPTable.CKS_LOOKUPTableEntry[0].setting =
1859 polaris10_clock_stretcher_lookup_table[stretch_amount2][2] & 0x7F;
1860 data->smc_state_table.CKS_LOOKUPTable.CKS_LOOKUPTableEntry[0].setting |=
1861 (polaris10_clock_stretcher_lookup_table[stretch_amount2][3]) << 7;
1862
1863 cgs_write_ind_register(hwmgr->device, CGS_IND_REG__SMC,
1864 ixPWR_CKS_CNTL, value);
1865
1866 /* Populate DDT Lookup Table */
1867 for (i = 0; i < 4; i++) {
1868 /* Assign the minimum and maximum VID stored
1869 * in the last row of Clock Stretcher Voltage Table.
1870 */
1871 data->smc_state_table.ClockStretcherDataTable.ClockStretcherDataTableEntry[i].minVID =
1872 (uint8_t) polaris10_clock_stretcher_ddt_table[type][i][2];
1873 data->smc_state_table.ClockStretcherDataTable.ClockStretcherDataTableEntry[i].maxVID =
1874 (uint8_t) polaris10_clock_stretcher_ddt_table[type][i][3];
1875 /* Loop through each SCLK and check the frequency
1876 * to see if it lies within the frequency for clock stretcher.
1877 */
1878 for (j = 0; j < data->smc_state_table.GraphicsDpmLevelCount; j++) {
1879 cks_setting = 0;
1880 clock_freq = PP_SMC_TO_HOST_UL(
1881 data->smc_state_table.GraphicsLevel[j].SclkSetting.SclkFrequency);
1882 /* Check the allowed frequency against the sclk level[j].
1883 * Sclk's endianness has already been converted,
1884 * and it's in 10Khz unit,
1885 * as opposed to Data table, which is in Mhz unit.
1886 */
1887 if (clock_freq >= (polaris10_clock_stretcher_ddt_table[type][i][0]) * 100) {
1888 cks_setting |= 0x2;
1889 if (clock_freq < (polaris10_clock_stretcher_ddt_table[type][i][1]) * 100)
1890 cks_setting |= 0x1;
1891 }
1892 data->smc_state_table.ClockStretcherDataTable.ClockStretcherDataTableEntry[i].setting
1893 |= cks_setting << (j * 2);
1894 }
1895 CONVERT_FROM_HOST_TO_SMC_US(
1896 data->smc_state_table.ClockStretcherDataTable.ClockStretcherDataTableEntry[i].setting);
1897 }
1898
1899 value = cgs_read_ind_register(hwmgr->device, CGS_IND_REG__SMC, ixPWR_CKS_CNTL); 1821 value = cgs_read_ind_register(hwmgr->device, CGS_IND_REG__SMC, ixPWR_CKS_CNTL);
1900 value &= 0xFFFFFFFE; 1822 value &= 0xFFFFFFFE;
1901 cgs_write_ind_register(hwmgr->device, CGS_IND_REG__SMC, ixPWR_CKS_CNTL, value); 1823 cgs_write_ind_register(hwmgr->device, CGS_IND_REG__SMC, ixPWR_CKS_CNTL, value);
@@ -3062,6 +2984,10 @@ int polaris10_hwmgr_backend_init(struct pp_hwmgr *hwmgr)
3062 data->vddci_control = POLARIS10_VOLTAGE_CONTROL_BY_SVID2; 2984 data->vddci_control = POLARIS10_VOLTAGE_CONTROL_BY_SVID2;
3063 } 2985 }
3064 2986
2987 if (table_info->cac_dtp_table->usClockStretchAmount != 0)
2988 phm_cap_set(hwmgr->platform_descriptor.platformCaps,
2989 PHM_PlatformCaps_ClockStretcher);
2990
3065 polaris10_set_features_platform_caps(hwmgr); 2991 polaris10_set_features_platform_caps(hwmgr);
3066 2992
3067 polaris10_init_dpm_defaults(hwmgr); 2993 polaris10_init_dpm_defaults(hwmgr);