1 // SPDX-License-Identifier: GPL-2.0-only
3 * Copyright (c) 2014, The Linux Foundation. All rights reserved.
7 #include <linux/kernel.h>
8 #include <linux/module.h>
10 #include <linux/time.h>
11 #include <linux/delay.h>
12 #include <linux/clk.h>
13 #include <linux/slab.h>
14 #include <linux/platform_device.h>
15 #include <linux/phy/phy.h>
18 #define UNIPHY_PLL_REFCLK_CFG 0x000
19 #define UNIPHY_PLL_PWRGEN_CFG 0x014
20 #define UNIPHY_PLL_GLB_CFG 0x020
21 #define UNIPHY_PLL_SDM_CFG0 0x038
22 #define UNIPHY_PLL_SDM_CFG1 0x03C
23 #define UNIPHY_PLL_SDM_CFG2 0x040
24 #define UNIPHY_PLL_SDM_CFG3 0x044
25 #define UNIPHY_PLL_SDM_CFG4 0x048
26 #define UNIPHY_PLL_SSC_CFG0 0x04C
27 #define UNIPHY_PLL_SSC_CFG1 0x050
28 #define UNIPHY_PLL_SSC_CFG2 0x054
29 #define UNIPHY_PLL_SSC_CFG3 0x058
30 #define UNIPHY_PLL_LKDET_CFG0 0x05C
31 #define UNIPHY_PLL_LKDET_CFG1 0x060
32 #define UNIPHY_PLL_LKDET_CFG2 0x064
33 #define UNIPHY_PLL_CAL_CFG0 0x06C
34 #define UNIPHY_PLL_CAL_CFG8 0x08C
35 #define UNIPHY_PLL_CAL_CFG9 0x090
36 #define UNIPHY_PLL_CAL_CFG10 0x094
37 #define UNIPHY_PLL_CAL_CFG11 0x098
38 #define UNIPHY_PLL_STATUS 0x0C0
40 #define SATA_PHY_SER_CTRL 0x100
41 #define SATA_PHY_TX_DRIV_CTRL0 0x104
42 #define SATA_PHY_TX_DRIV_CTRL1 0x108
43 #define SATA_PHY_TX_IMCAL0 0x11C
44 #define SATA_PHY_TX_IMCAL2 0x124
45 #define SATA_PHY_RX_IMCAL0 0x128
46 #define SATA_PHY_EQUAL 0x13C
47 #define SATA_PHY_OOB_TERM 0x144
48 #define SATA_PHY_CDR_CTRL0 0x148
49 #define SATA_PHY_CDR_CTRL1 0x14C
50 #define SATA_PHY_CDR_CTRL2 0x150
51 #define SATA_PHY_CDR_CTRL3 0x154
52 #define SATA_PHY_PI_CTRL0 0x168
53 #define SATA_PHY_POW_DWN_CTRL0 0x180
54 #define SATA_PHY_POW_DWN_CTRL1 0x184
55 #define SATA_PHY_TX_DATA_CTRL 0x188
56 #define SATA_PHY_ALIGNP 0x1A4
57 #define SATA_PHY_TX_IMCAL_STAT 0x1E4
58 #define SATA_PHY_RX_IMCAL_STAT 0x1E8
60 #define UNIPHY_PLL_LOCK BIT(0)
61 #define SATA_PHY_TX_CAL BIT(0)
62 #define SATA_PHY_RX_CAL BIT(0)
64 /* default timeout set to 1 sec */
65 #define TIMEOUT_MS 10000
66 #define DELAY_INTERVAL_US 100
68 struct qcom_apq8064_sata_phy
{
74 /* Helper function to do poll and timeout */
75 static int read_poll_timeout(void __iomem
*addr
, u32 mask
)
77 unsigned long timeout
= jiffies
+ msecs_to_jiffies(TIMEOUT_MS
);
80 if (readl_relaxed(addr
) & mask
)
83 usleep_range(DELAY_INTERVAL_US
, DELAY_INTERVAL_US
+ 50);
84 } while (!time_after(jiffies
, timeout
));
86 return (readl_relaxed(addr
) & mask
) ? 0 : -ETIMEDOUT
;
89 static int qcom_apq8064_sata_phy_init(struct phy
*generic_phy
)
91 struct qcom_apq8064_sata_phy
*phy
= phy_get_drvdata(generic_phy
);
92 void __iomem
*base
= phy
->mmio
;
95 /* SATA phy initialization */
96 writel_relaxed(0x01, base
+ SATA_PHY_SER_CTRL
);
97 writel_relaxed(0xB1, base
+ SATA_PHY_POW_DWN_CTRL0
);
98 /* Make sure the power down happens before power up */
100 usleep_range(10, 60);
102 writel_relaxed(0x01, base
+ SATA_PHY_POW_DWN_CTRL0
);
103 writel_relaxed(0x3E, base
+ SATA_PHY_POW_DWN_CTRL1
);
104 writel_relaxed(0x01, base
+ SATA_PHY_RX_IMCAL0
);
105 writel_relaxed(0x01, base
+ SATA_PHY_TX_IMCAL0
);
106 writel_relaxed(0x02, base
+ SATA_PHY_TX_IMCAL2
);
108 /* Write UNIPHYPLL registers to configure PLL */
109 writel_relaxed(0x04, base
+ UNIPHY_PLL_REFCLK_CFG
);
110 writel_relaxed(0x00, base
+ UNIPHY_PLL_PWRGEN_CFG
);
112 writel_relaxed(0x0A, base
+ UNIPHY_PLL_CAL_CFG0
);
113 writel_relaxed(0xF3, base
+ UNIPHY_PLL_CAL_CFG8
);
114 writel_relaxed(0x01, base
+ UNIPHY_PLL_CAL_CFG9
);
115 writel_relaxed(0xED, base
+ UNIPHY_PLL_CAL_CFG10
);
116 writel_relaxed(0x02, base
+ UNIPHY_PLL_CAL_CFG11
);
118 writel_relaxed(0x36, base
+ UNIPHY_PLL_SDM_CFG0
);
119 writel_relaxed(0x0D, base
+ UNIPHY_PLL_SDM_CFG1
);
120 writel_relaxed(0xA3, base
+ UNIPHY_PLL_SDM_CFG2
);
121 writel_relaxed(0xF0, base
+ UNIPHY_PLL_SDM_CFG3
);
122 writel_relaxed(0x00, base
+ UNIPHY_PLL_SDM_CFG4
);
124 writel_relaxed(0x19, base
+ UNIPHY_PLL_SSC_CFG0
);
125 writel_relaxed(0xE1, base
+ UNIPHY_PLL_SSC_CFG1
);
126 writel_relaxed(0x00, base
+ UNIPHY_PLL_SSC_CFG2
);
127 writel_relaxed(0x11, base
+ UNIPHY_PLL_SSC_CFG3
);
129 writel_relaxed(0x04, base
+ UNIPHY_PLL_LKDET_CFG0
);
130 writel_relaxed(0xFF, base
+ UNIPHY_PLL_LKDET_CFG1
);
132 writel_relaxed(0x02, base
+ UNIPHY_PLL_GLB_CFG
);
133 /* make sure global config LDO power down happens before power up */
136 writel_relaxed(0x03, base
+ UNIPHY_PLL_GLB_CFG
);
137 writel_relaxed(0x05, base
+ UNIPHY_PLL_LKDET_CFG2
);
140 ret
= read_poll_timeout(base
+ UNIPHY_PLL_STATUS
, UNIPHY_PLL_LOCK
);
142 dev_err(phy
->dev
, "poll timeout UNIPHY_PLL_STATUS\n");
147 ret
= read_poll_timeout(base
+ SATA_PHY_TX_IMCAL_STAT
, SATA_PHY_TX_CAL
);
149 dev_err(phy
->dev
, "poll timeout SATA_PHY_TX_IMCAL_STAT\n");
154 ret
= read_poll_timeout(base
+ SATA_PHY_RX_IMCAL_STAT
, SATA_PHY_RX_CAL
);
156 dev_err(phy
->dev
, "poll timeout SATA_PHY_RX_IMCAL_STAT\n");
160 /* SATA phy calibrated succesfully, power up to functional mode */
161 writel_relaxed(0x3E, base
+ SATA_PHY_POW_DWN_CTRL1
);
162 writel_relaxed(0x01, base
+ SATA_PHY_RX_IMCAL0
);
163 writel_relaxed(0x01, base
+ SATA_PHY_TX_IMCAL0
);
165 writel_relaxed(0x00, base
+ SATA_PHY_POW_DWN_CTRL1
);
166 writel_relaxed(0x59, base
+ SATA_PHY_CDR_CTRL0
);
167 writel_relaxed(0x04, base
+ SATA_PHY_CDR_CTRL1
);
168 writel_relaxed(0x00, base
+ SATA_PHY_CDR_CTRL2
);
169 writel_relaxed(0x00, base
+ SATA_PHY_PI_CTRL0
);
170 writel_relaxed(0x00, base
+ SATA_PHY_CDR_CTRL3
);
171 writel_relaxed(0x01, base
+ SATA_PHY_POW_DWN_CTRL0
);
173 writel_relaxed(0x11, base
+ SATA_PHY_TX_DATA_CTRL
);
174 writel_relaxed(0x43, base
+ SATA_PHY_ALIGNP
);
175 writel_relaxed(0x04, base
+ SATA_PHY_OOB_TERM
);
177 writel_relaxed(0x01, base
+ SATA_PHY_EQUAL
);
178 writel_relaxed(0x09, base
+ SATA_PHY_TX_DRIV_CTRL0
);
179 writel_relaxed(0x09, base
+ SATA_PHY_TX_DRIV_CTRL1
);
184 static int qcom_apq8064_sata_phy_exit(struct phy
*generic_phy
)
186 struct qcom_apq8064_sata_phy
*phy
= phy_get_drvdata(generic_phy
);
187 void __iomem
*base
= phy
->mmio
;
190 writel_relaxed(0xF8, base
+ SATA_PHY_POW_DWN_CTRL0
);
191 writel_relaxed(0xFE, base
+ SATA_PHY_POW_DWN_CTRL1
);
193 /* Power down PLL block */
194 writel_relaxed(0x00, base
+ UNIPHY_PLL_GLB_CFG
);
199 static const struct phy_ops qcom_apq8064_sata_phy_ops
= {
200 .init
= qcom_apq8064_sata_phy_init
,
201 .exit
= qcom_apq8064_sata_phy_exit
,
202 .owner
= THIS_MODULE
,
205 static int qcom_apq8064_sata_phy_probe(struct platform_device
*pdev
)
207 struct qcom_apq8064_sata_phy
*phy
;
208 struct device
*dev
= &pdev
->dev
;
209 struct resource
*res
;
210 struct phy_provider
*phy_provider
;
211 struct phy
*generic_phy
;
214 phy
= devm_kzalloc(dev
, sizeof(*phy
), GFP_KERNEL
);
218 res
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
219 phy
->mmio
= devm_ioremap_resource(dev
, res
);
220 if (IS_ERR(phy
->mmio
))
221 return PTR_ERR(phy
->mmio
);
223 generic_phy
= devm_phy_create(dev
, NULL
, &qcom_apq8064_sata_phy_ops
);
224 if (IS_ERR(generic_phy
)) {
225 dev_err(dev
, "%s: failed to create phy\n", __func__
);
226 return PTR_ERR(generic_phy
);
230 phy_set_drvdata(generic_phy
, phy
);
231 platform_set_drvdata(pdev
, phy
);
233 phy
->cfg_clk
= devm_clk_get(dev
, "cfg");
234 if (IS_ERR(phy
->cfg_clk
)) {
235 dev_err(dev
, "Failed to get sata cfg clock\n");
236 return PTR_ERR(phy
->cfg_clk
);
239 ret
= clk_prepare_enable(phy
->cfg_clk
);
243 phy_provider
= devm_of_phy_provider_register(dev
, of_phy_simple_xlate
);
244 if (IS_ERR(phy_provider
)) {
245 clk_disable_unprepare(phy
->cfg_clk
);
246 dev_err(dev
, "%s: failed to register phy\n", __func__
);
247 return PTR_ERR(phy_provider
);
253 static int qcom_apq8064_sata_phy_remove(struct platform_device
*pdev
)
255 struct qcom_apq8064_sata_phy
*phy
= platform_get_drvdata(pdev
);
257 clk_disable_unprepare(phy
->cfg_clk
);
262 static const struct of_device_id qcom_apq8064_sata_phy_of_match
[] = {
263 { .compatible
= "qcom,apq8064-sata-phy" },
266 MODULE_DEVICE_TABLE(of
, qcom_apq8064_sata_phy_of_match
);
268 static struct platform_driver qcom_apq8064_sata_phy_driver
= {
269 .probe
= qcom_apq8064_sata_phy_probe
,
270 .remove
= qcom_apq8064_sata_phy_remove
,
272 .name
= "qcom-apq8064-sata-phy",
273 .of_match_table
= qcom_apq8064_sata_phy_of_match
,
276 module_platform_driver(qcom_apq8064_sata_phy_driver
);
278 MODULE_DESCRIPTION("QCOM apq8064 SATA PHY driver");
279 MODULE_LICENSE("GPL v2");