|
158 | 158 | #define GLI_9767_VHS_REV_M 0x1
|
159 | 159 | #define GLI_9767_VHS_REV_W 0x2
|
160 | 160 |
|
| 161 | +#define PCIE_GLI_9767_COM_MAILBOX 0x888 |
| 162 | +#define PCIE_GLI_9767_COM_MAILBOX_SSC_EN BIT(1) |
| 163 | + |
| 164 | +#define PCIE_GLI_9767_CFG 0x8A0 |
| 165 | +#define PCIE_GLI_9767_CFG_LOW_PWR_OFF BIT(12) |
| 166 | + |
161 | 167 | #define PCIE_GLI_9767_PWR_MACRO_CTL 0x8D0
|
162 | 168 | #define PCIE_GLI_9767_PWR_MACRO_CTL_LOW_VOLTAGE GENMASK(3, 0)
|
163 | 169 | #define PCIE_GLI_9767_PWR_MACRO_CTL_LD0_LOW_OUTPUT_VOLTAGE GENMASK(15, 12)
|
|
175 | 181 | #define PCIE_GLI_9767_SCR_CORE_PWR_D3_OFF BIT(21)
|
176 | 182 | #define PCIE_GLI_9767_SCR_CFG_RST_DATA_LINK_DOWN BIT(30)
|
177 | 183 |
|
| 184 | +#define PCIE_GLI_9767_SD_PLL_CTL 0x938 |
| 185 | +#define PCIE_GLI_9767_SD_PLL_CTL_PLL_LDIV GENMASK(9, 0) |
| 186 | +#define PCIE_GLI_9767_SD_PLL_CTL_PLL_PDIV GENMASK(15, 12) |
| 187 | +#define PCIE_GLI_9767_SD_PLL_CTL_PLL_DIR_EN BIT(16) |
| 188 | +#define PCIE_GLI_9767_SD_PLL_CTL_SSC_EN BIT(19) |
| 189 | +#define PCIE_GLI_9767_SD_PLL_CTL_SSC_STEP_SETTING GENMASK(28, 24) |
| 190 | + |
| 191 | +#define PCIE_GLI_9767_SD_PLL_CTL2 0x93C |
| 192 | +#define PCIE_GLI_9767_SD_PLL_CTL2_PLLSSC_PPM GENMASK(31, 16) |
| 193 | + |
178 | 194 | #define GLI_MAX_TUNING_LOOP 40
|
179 | 195 |
|
180 | 196 | /* Genesys Logic chipset */
|
@@ -753,6 +769,123 @@ static inline void gl9767_vhs_write(struct pci_dev *pdev)
|
753 | 769 | pci_write_config_dword(pdev, PCIE_GLI_9767_VHS, vhs_value);
|
754 | 770 | }
|
755 | 771 |
|
| 772 | +static bool gl9767_ssc_enable(struct pci_dev *pdev) |
| 773 | +{ |
| 774 | + u32 value; |
| 775 | + u8 enable; |
| 776 | + |
| 777 | + gl9767_vhs_write(pdev); |
| 778 | + |
| 779 | + pci_read_config_dword(pdev, PCIE_GLI_9767_COM_MAILBOX, &value); |
| 780 | + enable = FIELD_GET(PCIE_GLI_9767_COM_MAILBOX_SSC_EN, value); |
| 781 | + |
| 782 | + gl9767_vhs_read(pdev); |
| 783 | + |
| 784 | + return enable; |
| 785 | +} |
| 786 | + |
| 787 | +static void gl9767_set_ssc(struct pci_dev *pdev, u8 enable, u8 step, u16 ppm) |
| 788 | +{ |
| 789 | + u32 pll; |
| 790 | + u32 ssc; |
| 791 | + |
| 792 | + gl9767_vhs_write(pdev); |
| 793 | + |
| 794 | + pci_read_config_dword(pdev, PCIE_GLI_9767_SD_PLL_CTL, &pll); |
| 795 | + pci_read_config_dword(pdev, PCIE_GLI_9767_SD_PLL_CTL2, &ssc); |
| 796 | + pll &= ~(PCIE_GLI_9767_SD_PLL_CTL_SSC_STEP_SETTING | |
| 797 | + PCIE_GLI_9767_SD_PLL_CTL_SSC_EN); |
| 798 | + ssc &= ~PCIE_GLI_9767_SD_PLL_CTL2_PLLSSC_PPM; |
| 799 | + pll |= FIELD_PREP(PCIE_GLI_9767_SD_PLL_CTL_SSC_STEP_SETTING, step) | |
| 800 | + FIELD_PREP(PCIE_GLI_9767_SD_PLL_CTL_SSC_EN, enable); |
| 801 | + ssc |= FIELD_PREP(PCIE_GLI_9767_SD_PLL_CTL2_PLLSSC_PPM, ppm); |
| 802 | + pci_write_config_dword(pdev, PCIE_GLI_9767_SD_PLL_CTL2, ssc); |
| 803 | + pci_write_config_dword(pdev, PCIE_GLI_9767_SD_PLL_CTL, pll); |
| 804 | + |
| 805 | + gl9767_vhs_read(pdev); |
| 806 | +} |
| 807 | + |
| 808 | +static void gl9767_set_pll(struct pci_dev *pdev, u8 dir, u16 ldiv, u8 pdiv) |
| 809 | +{ |
| 810 | + u32 pll; |
| 811 | + |
| 812 | + gl9767_vhs_write(pdev); |
| 813 | + |
| 814 | + pci_read_config_dword(pdev, PCIE_GLI_9767_SD_PLL_CTL, &pll); |
| 815 | + pll &= ~(PCIE_GLI_9767_SD_PLL_CTL_PLL_LDIV | |
| 816 | + PCIE_GLI_9767_SD_PLL_CTL_PLL_PDIV | |
| 817 | + PCIE_GLI_9767_SD_PLL_CTL_PLL_DIR_EN); |
| 818 | + pll |= FIELD_PREP(PCIE_GLI_9767_SD_PLL_CTL_PLL_LDIV, ldiv) | |
| 819 | + FIELD_PREP(PCIE_GLI_9767_SD_PLL_CTL_PLL_PDIV, pdiv) | |
| 820 | + FIELD_PREP(PCIE_GLI_9767_SD_PLL_CTL_PLL_DIR_EN, dir); |
| 821 | + pci_write_config_dword(pdev, PCIE_GLI_9767_SD_PLL_CTL, pll); |
| 822 | + |
| 823 | + gl9767_vhs_read(pdev); |
| 824 | + |
| 825 | + /* wait for pll stable */ |
| 826 | + usleep_range(1000, 1100); |
| 827 | +} |
| 828 | + |
| 829 | +static void gl9767_set_ssc_pll_205mhz(struct pci_dev *pdev) |
| 830 | +{ |
| 831 | + bool enable = gl9767_ssc_enable(pdev); |
| 832 | + |
| 833 | + /* set pll to 205MHz and ssc */ |
| 834 | + gl9767_set_ssc(pdev, enable, 0x1F, 0xF5C3); |
| 835 | + gl9767_set_pll(pdev, 0x1, 0x246, 0x0); |
| 836 | +} |
| 837 | + |
| 838 | +static void gl9767_disable_ssc_pll(struct pci_dev *pdev) |
| 839 | +{ |
| 840 | + u32 pll; |
| 841 | + |
| 842 | + gl9767_vhs_write(pdev); |
| 843 | + |
| 844 | + pci_read_config_dword(pdev, PCIE_GLI_9767_SD_PLL_CTL, &pll); |
| 845 | + pll &= ~(PCIE_GLI_9767_SD_PLL_CTL_PLL_DIR_EN | PCIE_GLI_9767_SD_PLL_CTL_SSC_EN); |
| 846 | + pci_write_config_dword(pdev, PCIE_GLI_9767_SD_PLL_CTL, pll); |
| 847 | + |
| 848 | + gl9767_vhs_read(pdev); |
| 849 | +} |
| 850 | + |
| 851 | +static void sdhci_gl9767_set_clock(struct sdhci_host *host, unsigned int clock) |
| 852 | +{ |
| 853 | + struct sdhci_pci_slot *slot = sdhci_priv(host); |
| 854 | + struct mmc_ios *ios = &host->mmc->ios; |
| 855 | + struct pci_dev *pdev; |
| 856 | + u32 value; |
| 857 | + u16 clk; |
| 858 | + |
| 859 | + pdev = slot->chip->pdev; |
| 860 | + host->mmc->actual_clock = 0; |
| 861 | + |
| 862 | + gl9767_vhs_write(pdev); |
| 863 | + |
| 864 | + pci_read_config_dword(pdev, PCIE_GLI_9767_CFG, &value); |
| 865 | + value |= PCIE_GLI_9767_CFG_LOW_PWR_OFF; |
| 866 | + pci_write_config_dword(pdev, PCIE_GLI_9767_CFG, value); |
| 867 | + |
| 868 | + gl9767_disable_ssc_pll(pdev); |
| 869 | + sdhci_writew(host, 0, SDHCI_CLOCK_CONTROL); |
| 870 | + |
| 871 | + if (clock == 0) |
| 872 | + return; |
| 873 | + |
| 874 | + clk = sdhci_calc_clk(host, clock, &host->mmc->actual_clock); |
| 875 | + if (clock == 200000000 && ios->timing == MMC_TIMING_UHS_SDR104) { |
| 876 | + host->mmc->actual_clock = 205000000; |
| 877 | + gl9767_set_ssc_pll_205mhz(pdev); |
| 878 | + } |
| 879 | + |
| 880 | + sdhci_enable_clk(host, clk); |
| 881 | + |
| 882 | + pci_read_config_dword(pdev, PCIE_GLI_9767_CFG, &value); |
| 883 | + value &= ~PCIE_GLI_9767_CFG_LOW_PWR_OFF; |
| 884 | + pci_write_config_dword(pdev, PCIE_GLI_9767_CFG, value); |
| 885 | + |
| 886 | + gl9767_vhs_read(pdev); |
| 887 | +} |
| 888 | + |
756 | 889 | static void gli_set_9767(struct sdhci_host *host)
|
757 | 890 | {
|
758 | 891 | u32 value;
|
@@ -1293,7 +1426,7 @@ const struct sdhci_pci_fixes sdhci_gl9763e = {
|
1293 | 1426 | };
|
1294 | 1427 |
|
1295 | 1428 | static const struct sdhci_ops sdhci_gl9767_ops = {
|
1296 |
| - .set_clock = sdhci_set_clock, |
| 1429 | + .set_clock = sdhci_gl9767_set_clock, |
1297 | 1430 | .enable_dma = sdhci_pci_enable_dma,
|
1298 | 1431 | .set_bus_width = sdhci_set_bus_width,
|
1299 | 1432 | .reset = sdhci_gl9767_reset,
|
|
0 commit comments