diff options
author | Rex Zhu <Rex.Zhu@amd.com> | 2016-06-07 06:39:06 -0400 |
---|---|---|
committer | Alex Deucher <alexander.deucher@amd.com> | 2016-06-21 10:22:42 -0400 |
commit | 270d013659ddab52a6fd0eacae452c422d08aa39 (patch) | |
tree | 48cb2696e73000bdbaca57c452de782d69bbf223 | |
parent | 92d1576859577b94eafaea9b64f78ab99fe20a78 (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.c | 120 |
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 | ||
1760 | static int polaris10_populate_clock_stretcher_data_table(struct pp_hwmgr *hwmgr) | 1760 | static 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); |