1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * drivers/ata/ahci_tegra.c
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * Copyright (c) 2014, NVIDIA CORPORATION. All rights reserved.
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * Author:
8*4882a593Smuzhiyun * Mikko Perttunen <mperttunen@nvidia.com>
9*4882a593Smuzhiyun */
10*4882a593Smuzhiyun
11*4882a593Smuzhiyun #include <linux/ahci_platform.h>
12*4882a593Smuzhiyun #include <linux/errno.h>
13*4882a593Smuzhiyun #include <linux/kernel.h>
14*4882a593Smuzhiyun #include <linux/module.h>
15*4882a593Smuzhiyun #include <linux/of_device.h>
16*4882a593Smuzhiyun #include <linux/platform_device.h>
17*4882a593Smuzhiyun #include <linux/regulator/consumer.h>
18*4882a593Smuzhiyun #include <linux/reset.h>
19*4882a593Smuzhiyun
20*4882a593Smuzhiyun #include <soc/tegra/fuse.h>
21*4882a593Smuzhiyun #include <soc/tegra/pmc.h>
22*4882a593Smuzhiyun
23*4882a593Smuzhiyun #include "ahci.h"
24*4882a593Smuzhiyun
25*4882a593Smuzhiyun #define DRV_NAME "tegra-ahci"
26*4882a593Smuzhiyun
27*4882a593Smuzhiyun #define SATA_CONFIGURATION_0 0x180
28*4882a593Smuzhiyun #define SATA_CONFIGURATION_0_EN_FPCI BIT(0)
29*4882a593Smuzhiyun #define SATA_CONFIGURATION_0_CLK_OVERRIDE BIT(31)
30*4882a593Smuzhiyun
31*4882a593Smuzhiyun #define SCFG_OFFSET 0x1000
32*4882a593Smuzhiyun
33*4882a593Smuzhiyun #define T_SATA0_CFG_1 0x04
34*4882a593Smuzhiyun #define T_SATA0_CFG_1_IO_SPACE BIT(0)
35*4882a593Smuzhiyun #define T_SATA0_CFG_1_MEMORY_SPACE BIT(1)
36*4882a593Smuzhiyun #define T_SATA0_CFG_1_BUS_MASTER BIT(2)
37*4882a593Smuzhiyun #define T_SATA0_CFG_1_SERR BIT(8)
38*4882a593Smuzhiyun
39*4882a593Smuzhiyun #define T_SATA0_CFG_9 0x24
40*4882a593Smuzhiyun #define T_SATA0_CFG_9_BASE_ADDRESS 0x40020000
41*4882a593Smuzhiyun
42*4882a593Smuzhiyun #define SATA_FPCI_BAR5 0x94
43*4882a593Smuzhiyun #define SATA_FPCI_BAR5_START_MASK (0xfffffff << 4)
44*4882a593Smuzhiyun #define SATA_FPCI_BAR5_START (0x0040020 << 4)
45*4882a593Smuzhiyun #define SATA_FPCI_BAR5_ACCESS_TYPE (0x1)
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun #define SATA_INTR_MASK 0x188
48*4882a593Smuzhiyun #define SATA_INTR_MASK_IP_INT_MASK BIT(16)
49*4882a593Smuzhiyun
50*4882a593Smuzhiyun #define T_SATA0_CFG_35 0x94
51*4882a593Smuzhiyun #define T_SATA0_CFG_35_IDP_INDEX_MASK (0x7ff << 2)
52*4882a593Smuzhiyun #define T_SATA0_CFG_35_IDP_INDEX (0x2a << 2)
53*4882a593Smuzhiyun
54*4882a593Smuzhiyun #define T_SATA0_AHCI_IDP1 0x98
55*4882a593Smuzhiyun #define T_SATA0_AHCI_IDP1_DATA (0x400040)
56*4882a593Smuzhiyun
57*4882a593Smuzhiyun #define T_SATA0_CFG_PHY_1 0x12c
58*4882a593Smuzhiyun #define T_SATA0_CFG_PHY_1_PADS_IDDQ_EN BIT(23)
59*4882a593Smuzhiyun #define T_SATA0_CFG_PHY_1_PAD_PLL_IDDQ_EN BIT(22)
60*4882a593Smuzhiyun
61*4882a593Smuzhiyun #define T_SATA0_NVOOB 0x114
62*4882a593Smuzhiyun #define T_SATA0_NVOOB_COMMA_CNT_MASK (0xff << 16)
63*4882a593Smuzhiyun #define T_SATA0_NVOOB_COMMA_CNT (0x07 << 16)
64*4882a593Smuzhiyun #define T_SATA0_NVOOB_SQUELCH_FILTER_MODE_MASK (0x3 << 24)
65*4882a593Smuzhiyun #define T_SATA0_NVOOB_SQUELCH_FILTER_MODE (0x1 << 24)
66*4882a593Smuzhiyun #define T_SATA0_NVOOB_SQUELCH_FILTER_LENGTH_MASK (0x3 << 26)
67*4882a593Smuzhiyun #define T_SATA0_NVOOB_SQUELCH_FILTER_LENGTH (0x3 << 26)
68*4882a593Smuzhiyun
69*4882a593Smuzhiyun #define T_SATA_CFG_PHY_0 0x120
70*4882a593Smuzhiyun #define T_SATA_CFG_PHY_0_USE_7BIT_ALIGN_DET_FOR_SPD BIT(11)
71*4882a593Smuzhiyun #define T_SATA_CFG_PHY_0_MASK_SQUELCH BIT(24)
72*4882a593Smuzhiyun
73*4882a593Smuzhiyun #define T_SATA0_CFG2NVOOB_2 0x134
74*4882a593Smuzhiyun #define T_SATA0_CFG2NVOOB_2_COMWAKE_IDLE_CNT_LOW_MASK (0x1ff << 18)
75*4882a593Smuzhiyun #define T_SATA0_CFG2NVOOB_2_COMWAKE_IDLE_CNT_LOW (0xc << 18)
76*4882a593Smuzhiyun
77*4882a593Smuzhiyun #define T_SATA0_AHCI_HBA_CAP_BKDR 0x300
78*4882a593Smuzhiyun #define T_SATA0_AHCI_HBA_CAP_BKDR_PARTIAL_ST_CAP BIT(13)
79*4882a593Smuzhiyun #define T_SATA0_AHCI_HBA_CAP_BKDR_SLUMBER_ST_CAP BIT(14)
80*4882a593Smuzhiyun #define T_SATA0_AHCI_HBA_CAP_BKDR_SALP BIT(26)
81*4882a593Smuzhiyun #define T_SATA0_AHCI_HBA_CAP_BKDR_SUPP_PM BIT(17)
82*4882a593Smuzhiyun #define T_SATA0_AHCI_HBA_CAP_BKDR_SNCQ BIT(30)
83*4882a593Smuzhiyun
84*4882a593Smuzhiyun #define T_SATA0_BKDOOR_CC 0x4a4
85*4882a593Smuzhiyun #define T_SATA0_BKDOOR_CC_CLASS_CODE_MASK (0xffff << 16)
86*4882a593Smuzhiyun #define T_SATA0_BKDOOR_CC_CLASS_CODE (0x0106 << 16)
87*4882a593Smuzhiyun #define T_SATA0_BKDOOR_CC_PROG_IF_MASK (0xff << 8)
88*4882a593Smuzhiyun #define T_SATA0_BKDOOR_CC_PROG_IF (0x01 << 8)
89*4882a593Smuzhiyun
90*4882a593Smuzhiyun #define T_SATA0_CFG_SATA 0x54c
91*4882a593Smuzhiyun #define T_SATA0_CFG_SATA_BACKDOOR_PROG_IF_EN BIT(12)
92*4882a593Smuzhiyun
93*4882a593Smuzhiyun #define T_SATA0_CFG_MISC 0x550
94*4882a593Smuzhiyun
95*4882a593Smuzhiyun #define T_SATA0_INDEX 0x680
96*4882a593Smuzhiyun
97*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL1_GEN1 0x690
98*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL1_GEN1_TX_AMP_MASK 0xff
99*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL1_GEN1_TX_AMP_SHIFT 0
100*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL1_GEN1_TX_PEAK_MASK (0xff << 8)
101*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL1_GEN1_TX_PEAK_SHIFT 8
102*4882a593Smuzhiyun
103*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL1_GEN2 0x694
104*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL1_GEN2_TX_AMP_MASK 0xff
105*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL1_GEN2_TX_AMP_SHIFT 0
106*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL1_GEN2_TX_PEAK_MASK (0xff << 12)
107*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL1_GEN2_TX_PEAK_SHIFT 12
108*4882a593Smuzhiyun
109*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL2 0x69c
110*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL2_CDR_CNTL_GEN1 0x23
111*4882a593Smuzhiyun
112*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL11 0x6d0
113*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL11_GEN2_RX_EQ (0x2800 << 16)
114*4882a593Smuzhiyun
115*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL17_0 0x6e8
116*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL17_0_RX_EQ_CTRL_L_GEN1 0x55010000
117*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL18_0 0x6ec
118*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL18_0_RX_EQ_CTRL_L_GEN2 0x55010000
119*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL20_0 0x6f4
120*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL20_0_RX_EQ_CTRL_H_GEN1 0x1
121*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL21_0 0x6f8
122*4882a593Smuzhiyun #define T_SATA0_CHX_PHY_CTRL21_0_RX_EQ_CTRL_H_GEN2 0x1
123*4882a593Smuzhiyun
124*4882a593Smuzhiyun /* AUX Registers */
125*4882a593Smuzhiyun #define SATA_AUX_MISC_CNTL_1_0 0x8
126*4882a593Smuzhiyun #define SATA_AUX_MISC_CNTL_1_0_DEVSLP_OVERRIDE BIT(17)
127*4882a593Smuzhiyun #define SATA_AUX_MISC_CNTL_1_0_SDS_SUPPORT BIT(13)
128*4882a593Smuzhiyun #define SATA_AUX_MISC_CNTL_1_0_DESO_SUPPORT BIT(15)
129*4882a593Smuzhiyun
130*4882a593Smuzhiyun #define SATA_AUX_RX_STAT_INT_0 0xc
131*4882a593Smuzhiyun #define SATA_AUX_RX_STAT_INT_0_SATA_DEVSLP BIT(7)
132*4882a593Smuzhiyun
133*4882a593Smuzhiyun #define SATA_AUX_SPARE_CFG0_0 0x18
134*4882a593Smuzhiyun #define SATA_AUX_SPARE_CFG0_0_MDAT_TIMER_AFTER_PG_VALID BIT(14)
135*4882a593Smuzhiyun
136*4882a593Smuzhiyun #define FUSE_SATA_CALIB 0x124
137*4882a593Smuzhiyun #define FUSE_SATA_CALIB_MASK 0x3
138*4882a593Smuzhiyun
139*4882a593Smuzhiyun struct sata_pad_calibration {
140*4882a593Smuzhiyun u8 gen1_tx_amp;
141*4882a593Smuzhiyun u8 gen1_tx_peak;
142*4882a593Smuzhiyun u8 gen2_tx_amp;
143*4882a593Smuzhiyun u8 gen2_tx_peak;
144*4882a593Smuzhiyun };
145*4882a593Smuzhiyun
146*4882a593Smuzhiyun static const struct sata_pad_calibration tegra124_pad_calibration[] = {
147*4882a593Smuzhiyun {0x18, 0x04, 0x18, 0x0a},
148*4882a593Smuzhiyun {0x0e, 0x04, 0x14, 0x0a},
149*4882a593Smuzhiyun {0x0e, 0x07, 0x1a, 0x0e},
150*4882a593Smuzhiyun {0x14, 0x0e, 0x1a, 0x0e},
151*4882a593Smuzhiyun };
152*4882a593Smuzhiyun
153*4882a593Smuzhiyun struct tegra_ahci_ops {
154*4882a593Smuzhiyun int (*init)(struct ahci_host_priv *hpriv);
155*4882a593Smuzhiyun };
156*4882a593Smuzhiyun
157*4882a593Smuzhiyun struct tegra_ahci_soc {
158*4882a593Smuzhiyun const char *const *supply_names;
159*4882a593Smuzhiyun u32 num_supplies;
160*4882a593Smuzhiyun bool supports_devslp;
161*4882a593Smuzhiyun const struct tegra_ahci_ops *ops;
162*4882a593Smuzhiyun };
163*4882a593Smuzhiyun
164*4882a593Smuzhiyun struct tegra_ahci_priv {
165*4882a593Smuzhiyun struct platform_device *pdev;
166*4882a593Smuzhiyun void __iomem *sata_regs;
167*4882a593Smuzhiyun void __iomem *sata_aux_regs;
168*4882a593Smuzhiyun struct reset_control *sata_rst;
169*4882a593Smuzhiyun struct reset_control *sata_oob_rst;
170*4882a593Smuzhiyun struct reset_control *sata_cold_rst;
171*4882a593Smuzhiyun /* Needs special handling, cannot use ahci_platform */
172*4882a593Smuzhiyun struct clk *sata_clk;
173*4882a593Smuzhiyun struct regulator_bulk_data *supplies;
174*4882a593Smuzhiyun const struct tegra_ahci_soc *soc;
175*4882a593Smuzhiyun };
176*4882a593Smuzhiyun
tegra_ahci_handle_quirks(struct ahci_host_priv * hpriv)177*4882a593Smuzhiyun static void tegra_ahci_handle_quirks(struct ahci_host_priv *hpriv)
178*4882a593Smuzhiyun {
179*4882a593Smuzhiyun struct tegra_ahci_priv *tegra = hpriv->plat_data;
180*4882a593Smuzhiyun u32 val;
181*4882a593Smuzhiyun
182*4882a593Smuzhiyun if (tegra->sata_aux_regs && !tegra->soc->supports_devslp) {
183*4882a593Smuzhiyun val = readl(tegra->sata_aux_regs + SATA_AUX_MISC_CNTL_1_0);
184*4882a593Smuzhiyun val &= ~SATA_AUX_MISC_CNTL_1_0_SDS_SUPPORT;
185*4882a593Smuzhiyun writel(val, tegra->sata_aux_regs + SATA_AUX_MISC_CNTL_1_0);
186*4882a593Smuzhiyun }
187*4882a593Smuzhiyun }
188*4882a593Smuzhiyun
tegra124_ahci_init(struct ahci_host_priv * hpriv)189*4882a593Smuzhiyun static int tegra124_ahci_init(struct ahci_host_priv *hpriv)
190*4882a593Smuzhiyun {
191*4882a593Smuzhiyun struct tegra_ahci_priv *tegra = hpriv->plat_data;
192*4882a593Smuzhiyun struct sata_pad_calibration calib;
193*4882a593Smuzhiyun int ret;
194*4882a593Smuzhiyun u32 val;
195*4882a593Smuzhiyun
196*4882a593Smuzhiyun /* Pad calibration */
197*4882a593Smuzhiyun ret = tegra_fuse_readl(FUSE_SATA_CALIB, &val);
198*4882a593Smuzhiyun if (ret)
199*4882a593Smuzhiyun return ret;
200*4882a593Smuzhiyun
201*4882a593Smuzhiyun calib = tegra124_pad_calibration[val & FUSE_SATA_CALIB_MASK];
202*4882a593Smuzhiyun
203*4882a593Smuzhiyun writel(BIT(0), tegra->sata_regs + SCFG_OFFSET + T_SATA0_INDEX);
204*4882a593Smuzhiyun
205*4882a593Smuzhiyun val = readl(tegra->sata_regs +
206*4882a593Smuzhiyun SCFG_OFFSET + T_SATA0_CHX_PHY_CTRL1_GEN1);
207*4882a593Smuzhiyun val &= ~T_SATA0_CHX_PHY_CTRL1_GEN1_TX_AMP_MASK;
208*4882a593Smuzhiyun val &= ~T_SATA0_CHX_PHY_CTRL1_GEN1_TX_PEAK_MASK;
209*4882a593Smuzhiyun val |= calib.gen1_tx_amp << T_SATA0_CHX_PHY_CTRL1_GEN1_TX_AMP_SHIFT;
210*4882a593Smuzhiyun val |= calib.gen1_tx_peak << T_SATA0_CHX_PHY_CTRL1_GEN1_TX_PEAK_SHIFT;
211*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET +
212*4882a593Smuzhiyun T_SATA0_CHX_PHY_CTRL1_GEN1);
213*4882a593Smuzhiyun
214*4882a593Smuzhiyun val = readl(tegra->sata_regs +
215*4882a593Smuzhiyun SCFG_OFFSET + T_SATA0_CHX_PHY_CTRL1_GEN2);
216*4882a593Smuzhiyun val &= ~T_SATA0_CHX_PHY_CTRL1_GEN2_TX_AMP_MASK;
217*4882a593Smuzhiyun val &= ~T_SATA0_CHX_PHY_CTRL1_GEN2_TX_PEAK_MASK;
218*4882a593Smuzhiyun val |= calib.gen2_tx_amp << T_SATA0_CHX_PHY_CTRL1_GEN1_TX_AMP_SHIFT;
219*4882a593Smuzhiyun val |= calib.gen2_tx_peak << T_SATA0_CHX_PHY_CTRL1_GEN1_TX_PEAK_SHIFT;
220*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET +
221*4882a593Smuzhiyun T_SATA0_CHX_PHY_CTRL1_GEN2);
222*4882a593Smuzhiyun
223*4882a593Smuzhiyun writel(T_SATA0_CHX_PHY_CTRL11_GEN2_RX_EQ,
224*4882a593Smuzhiyun tegra->sata_regs + SCFG_OFFSET + T_SATA0_CHX_PHY_CTRL11);
225*4882a593Smuzhiyun writel(T_SATA0_CHX_PHY_CTRL2_CDR_CNTL_GEN1,
226*4882a593Smuzhiyun tegra->sata_regs + SCFG_OFFSET + T_SATA0_CHX_PHY_CTRL2);
227*4882a593Smuzhiyun
228*4882a593Smuzhiyun writel(0, tegra->sata_regs + SCFG_OFFSET + T_SATA0_INDEX);
229*4882a593Smuzhiyun
230*4882a593Smuzhiyun return 0;
231*4882a593Smuzhiyun }
232*4882a593Smuzhiyun
tegra_ahci_power_on(struct ahci_host_priv * hpriv)233*4882a593Smuzhiyun static int tegra_ahci_power_on(struct ahci_host_priv *hpriv)
234*4882a593Smuzhiyun {
235*4882a593Smuzhiyun struct tegra_ahci_priv *tegra = hpriv->plat_data;
236*4882a593Smuzhiyun int ret;
237*4882a593Smuzhiyun
238*4882a593Smuzhiyun ret = regulator_bulk_enable(tegra->soc->num_supplies,
239*4882a593Smuzhiyun tegra->supplies);
240*4882a593Smuzhiyun if (ret)
241*4882a593Smuzhiyun return ret;
242*4882a593Smuzhiyun
243*4882a593Smuzhiyun ret = tegra_powergate_sequence_power_up(TEGRA_POWERGATE_SATA,
244*4882a593Smuzhiyun tegra->sata_clk,
245*4882a593Smuzhiyun tegra->sata_rst);
246*4882a593Smuzhiyun if (ret)
247*4882a593Smuzhiyun goto disable_regulators;
248*4882a593Smuzhiyun
249*4882a593Smuzhiyun reset_control_assert(tegra->sata_oob_rst);
250*4882a593Smuzhiyun reset_control_assert(tegra->sata_cold_rst);
251*4882a593Smuzhiyun
252*4882a593Smuzhiyun ret = ahci_platform_enable_resources(hpriv);
253*4882a593Smuzhiyun if (ret)
254*4882a593Smuzhiyun goto disable_power;
255*4882a593Smuzhiyun
256*4882a593Smuzhiyun reset_control_deassert(tegra->sata_cold_rst);
257*4882a593Smuzhiyun reset_control_deassert(tegra->sata_oob_rst);
258*4882a593Smuzhiyun
259*4882a593Smuzhiyun return 0;
260*4882a593Smuzhiyun
261*4882a593Smuzhiyun disable_power:
262*4882a593Smuzhiyun clk_disable_unprepare(tegra->sata_clk);
263*4882a593Smuzhiyun
264*4882a593Smuzhiyun tegra_powergate_power_off(TEGRA_POWERGATE_SATA);
265*4882a593Smuzhiyun
266*4882a593Smuzhiyun disable_regulators:
267*4882a593Smuzhiyun regulator_bulk_disable(tegra->soc->num_supplies, tegra->supplies);
268*4882a593Smuzhiyun
269*4882a593Smuzhiyun return ret;
270*4882a593Smuzhiyun }
271*4882a593Smuzhiyun
tegra_ahci_power_off(struct ahci_host_priv * hpriv)272*4882a593Smuzhiyun static void tegra_ahci_power_off(struct ahci_host_priv *hpriv)
273*4882a593Smuzhiyun {
274*4882a593Smuzhiyun struct tegra_ahci_priv *tegra = hpriv->plat_data;
275*4882a593Smuzhiyun
276*4882a593Smuzhiyun ahci_platform_disable_resources(hpriv);
277*4882a593Smuzhiyun
278*4882a593Smuzhiyun reset_control_assert(tegra->sata_rst);
279*4882a593Smuzhiyun reset_control_assert(tegra->sata_oob_rst);
280*4882a593Smuzhiyun reset_control_assert(tegra->sata_cold_rst);
281*4882a593Smuzhiyun
282*4882a593Smuzhiyun clk_disable_unprepare(tegra->sata_clk);
283*4882a593Smuzhiyun tegra_powergate_power_off(TEGRA_POWERGATE_SATA);
284*4882a593Smuzhiyun
285*4882a593Smuzhiyun regulator_bulk_disable(tegra->soc->num_supplies, tegra->supplies);
286*4882a593Smuzhiyun }
287*4882a593Smuzhiyun
tegra_ahci_controller_init(struct ahci_host_priv * hpriv)288*4882a593Smuzhiyun static int tegra_ahci_controller_init(struct ahci_host_priv *hpriv)
289*4882a593Smuzhiyun {
290*4882a593Smuzhiyun struct tegra_ahci_priv *tegra = hpriv->plat_data;
291*4882a593Smuzhiyun int ret;
292*4882a593Smuzhiyun u32 val;
293*4882a593Smuzhiyun
294*4882a593Smuzhiyun ret = tegra_ahci_power_on(hpriv);
295*4882a593Smuzhiyun if (ret) {
296*4882a593Smuzhiyun dev_err(&tegra->pdev->dev,
297*4882a593Smuzhiyun "failed to power on AHCI controller: %d\n", ret);
298*4882a593Smuzhiyun return ret;
299*4882a593Smuzhiyun }
300*4882a593Smuzhiyun
301*4882a593Smuzhiyun /*
302*4882a593Smuzhiyun * Program the following SATA IPFS registers to allow SW accesses to
303*4882a593Smuzhiyun * SATA's MMIO register range.
304*4882a593Smuzhiyun */
305*4882a593Smuzhiyun val = readl(tegra->sata_regs + SATA_FPCI_BAR5);
306*4882a593Smuzhiyun val &= ~(SATA_FPCI_BAR5_START_MASK | SATA_FPCI_BAR5_ACCESS_TYPE);
307*4882a593Smuzhiyun val |= SATA_FPCI_BAR5_START | SATA_FPCI_BAR5_ACCESS_TYPE;
308*4882a593Smuzhiyun writel(val, tegra->sata_regs + SATA_FPCI_BAR5);
309*4882a593Smuzhiyun
310*4882a593Smuzhiyun /* Program the following SATA IPFS register to enable the SATA */
311*4882a593Smuzhiyun val = readl(tegra->sata_regs + SATA_CONFIGURATION_0);
312*4882a593Smuzhiyun val |= SATA_CONFIGURATION_0_EN_FPCI;
313*4882a593Smuzhiyun writel(val, tegra->sata_regs + SATA_CONFIGURATION_0);
314*4882a593Smuzhiyun
315*4882a593Smuzhiyun /* Electrical settings for better link stability */
316*4882a593Smuzhiyun val = T_SATA0_CHX_PHY_CTRL17_0_RX_EQ_CTRL_L_GEN1;
317*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_CHX_PHY_CTRL17_0);
318*4882a593Smuzhiyun val = T_SATA0_CHX_PHY_CTRL18_0_RX_EQ_CTRL_L_GEN2;
319*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_CHX_PHY_CTRL18_0);
320*4882a593Smuzhiyun val = T_SATA0_CHX_PHY_CTRL20_0_RX_EQ_CTRL_H_GEN1;
321*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_CHX_PHY_CTRL20_0);
322*4882a593Smuzhiyun val = T_SATA0_CHX_PHY_CTRL21_0_RX_EQ_CTRL_H_GEN2;
323*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_CHX_PHY_CTRL21_0);
324*4882a593Smuzhiyun
325*4882a593Smuzhiyun /* For SQUELCH Filter & Gen3 drive getting detected as Gen1 drive */
326*4882a593Smuzhiyun
327*4882a593Smuzhiyun val = readl(tegra->sata_regs + SCFG_OFFSET + T_SATA_CFG_PHY_0);
328*4882a593Smuzhiyun val |= T_SATA_CFG_PHY_0_MASK_SQUELCH;
329*4882a593Smuzhiyun val &= ~T_SATA_CFG_PHY_0_USE_7BIT_ALIGN_DET_FOR_SPD;
330*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA_CFG_PHY_0);
331*4882a593Smuzhiyun
332*4882a593Smuzhiyun val = readl(tegra->sata_regs + SCFG_OFFSET + T_SATA0_NVOOB);
333*4882a593Smuzhiyun val &= ~(T_SATA0_NVOOB_COMMA_CNT_MASK |
334*4882a593Smuzhiyun T_SATA0_NVOOB_SQUELCH_FILTER_LENGTH_MASK |
335*4882a593Smuzhiyun T_SATA0_NVOOB_SQUELCH_FILTER_MODE_MASK);
336*4882a593Smuzhiyun val |= (T_SATA0_NVOOB_COMMA_CNT |
337*4882a593Smuzhiyun T_SATA0_NVOOB_SQUELCH_FILTER_LENGTH |
338*4882a593Smuzhiyun T_SATA0_NVOOB_SQUELCH_FILTER_MODE);
339*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_NVOOB);
340*4882a593Smuzhiyun
341*4882a593Smuzhiyun /*
342*4882a593Smuzhiyun * Change CFG2NVOOB_2_COMWAKE_IDLE_CNT_LOW from 83.3 ns to 58.8ns
343*4882a593Smuzhiyun */
344*4882a593Smuzhiyun val = readl(tegra->sata_regs + SCFG_OFFSET + T_SATA0_CFG2NVOOB_2);
345*4882a593Smuzhiyun val &= ~T_SATA0_CFG2NVOOB_2_COMWAKE_IDLE_CNT_LOW_MASK;
346*4882a593Smuzhiyun val |= T_SATA0_CFG2NVOOB_2_COMWAKE_IDLE_CNT_LOW;
347*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_CFG2NVOOB_2);
348*4882a593Smuzhiyun
349*4882a593Smuzhiyun if (tegra->soc->ops && tegra->soc->ops->init)
350*4882a593Smuzhiyun tegra->soc->ops->init(hpriv);
351*4882a593Smuzhiyun
352*4882a593Smuzhiyun /*
353*4882a593Smuzhiyun * Program the following SATA configuration registers to
354*4882a593Smuzhiyun * initialize SATA
355*4882a593Smuzhiyun */
356*4882a593Smuzhiyun val = readl(tegra->sata_regs + SCFG_OFFSET + T_SATA0_CFG_1);
357*4882a593Smuzhiyun val |= (T_SATA0_CFG_1_IO_SPACE | T_SATA0_CFG_1_MEMORY_SPACE |
358*4882a593Smuzhiyun T_SATA0_CFG_1_BUS_MASTER | T_SATA0_CFG_1_SERR);
359*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_CFG_1);
360*4882a593Smuzhiyun val = T_SATA0_CFG_9_BASE_ADDRESS;
361*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_CFG_9);
362*4882a593Smuzhiyun
363*4882a593Smuzhiyun /* Program Class Code and Programming interface for SATA */
364*4882a593Smuzhiyun val = readl(tegra->sata_regs + SCFG_OFFSET + T_SATA0_CFG_SATA);
365*4882a593Smuzhiyun val |= T_SATA0_CFG_SATA_BACKDOOR_PROG_IF_EN;
366*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_CFG_SATA);
367*4882a593Smuzhiyun
368*4882a593Smuzhiyun val = readl(tegra->sata_regs + SCFG_OFFSET + T_SATA0_BKDOOR_CC);
369*4882a593Smuzhiyun val &=
370*4882a593Smuzhiyun ~(T_SATA0_BKDOOR_CC_CLASS_CODE_MASK |
371*4882a593Smuzhiyun T_SATA0_BKDOOR_CC_PROG_IF_MASK);
372*4882a593Smuzhiyun val |= T_SATA0_BKDOOR_CC_CLASS_CODE | T_SATA0_BKDOOR_CC_PROG_IF;
373*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_BKDOOR_CC);
374*4882a593Smuzhiyun
375*4882a593Smuzhiyun val = readl(tegra->sata_regs + SCFG_OFFSET + T_SATA0_CFG_SATA);
376*4882a593Smuzhiyun val &= ~T_SATA0_CFG_SATA_BACKDOOR_PROG_IF_EN;
377*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_CFG_SATA);
378*4882a593Smuzhiyun
379*4882a593Smuzhiyun /* Enabling LPM capabilities through Backdoor Programming */
380*4882a593Smuzhiyun val = readl(tegra->sata_regs + SCFG_OFFSET + T_SATA0_AHCI_HBA_CAP_BKDR);
381*4882a593Smuzhiyun val |= (T_SATA0_AHCI_HBA_CAP_BKDR_PARTIAL_ST_CAP |
382*4882a593Smuzhiyun T_SATA0_AHCI_HBA_CAP_BKDR_SLUMBER_ST_CAP |
383*4882a593Smuzhiyun T_SATA0_AHCI_HBA_CAP_BKDR_SALP |
384*4882a593Smuzhiyun T_SATA0_AHCI_HBA_CAP_BKDR_SUPP_PM);
385*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_AHCI_HBA_CAP_BKDR);
386*4882a593Smuzhiyun
387*4882a593Smuzhiyun /* SATA Second Level Clock Gating configuration
388*4882a593Smuzhiyun * Enabling Gating of Tx/Rx clocks and driving Pad IDDQ and Lane
389*4882a593Smuzhiyun * IDDQ Signals
390*4882a593Smuzhiyun */
391*4882a593Smuzhiyun val = readl(tegra->sata_regs + SCFG_OFFSET + T_SATA0_CFG_35);
392*4882a593Smuzhiyun val &= ~T_SATA0_CFG_35_IDP_INDEX_MASK;
393*4882a593Smuzhiyun val |= T_SATA0_CFG_35_IDP_INDEX;
394*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_CFG_35);
395*4882a593Smuzhiyun
396*4882a593Smuzhiyun val = T_SATA0_AHCI_IDP1_DATA;
397*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_AHCI_IDP1);
398*4882a593Smuzhiyun
399*4882a593Smuzhiyun val = readl(tegra->sata_regs + SCFG_OFFSET + T_SATA0_CFG_PHY_1);
400*4882a593Smuzhiyun val |= (T_SATA0_CFG_PHY_1_PADS_IDDQ_EN |
401*4882a593Smuzhiyun T_SATA0_CFG_PHY_1_PAD_PLL_IDDQ_EN);
402*4882a593Smuzhiyun writel(val, tegra->sata_regs + SCFG_OFFSET + T_SATA0_CFG_PHY_1);
403*4882a593Smuzhiyun
404*4882a593Smuzhiyun /* Enabling IPFS Clock Gating */
405*4882a593Smuzhiyun val = readl(tegra->sata_regs + SATA_CONFIGURATION_0);
406*4882a593Smuzhiyun val &= ~SATA_CONFIGURATION_0_CLK_OVERRIDE;
407*4882a593Smuzhiyun writel(val, tegra->sata_regs + SATA_CONFIGURATION_0);
408*4882a593Smuzhiyun
409*4882a593Smuzhiyun tegra_ahci_handle_quirks(hpriv);
410*4882a593Smuzhiyun
411*4882a593Smuzhiyun /* Unmask SATA interrupts */
412*4882a593Smuzhiyun
413*4882a593Smuzhiyun val = readl(tegra->sata_regs + SATA_INTR_MASK);
414*4882a593Smuzhiyun val |= SATA_INTR_MASK_IP_INT_MASK;
415*4882a593Smuzhiyun writel(val, tegra->sata_regs + SATA_INTR_MASK);
416*4882a593Smuzhiyun
417*4882a593Smuzhiyun return 0;
418*4882a593Smuzhiyun }
419*4882a593Smuzhiyun
tegra_ahci_controller_deinit(struct ahci_host_priv * hpriv)420*4882a593Smuzhiyun static void tegra_ahci_controller_deinit(struct ahci_host_priv *hpriv)
421*4882a593Smuzhiyun {
422*4882a593Smuzhiyun tegra_ahci_power_off(hpriv);
423*4882a593Smuzhiyun }
424*4882a593Smuzhiyun
tegra_ahci_host_stop(struct ata_host * host)425*4882a593Smuzhiyun static void tegra_ahci_host_stop(struct ata_host *host)
426*4882a593Smuzhiyun {
427*4882a593Smuzhiyun struct ahci_host_priv *hpriv = host->private_data;
428*4882a593Smuzhiyun
429*4882a593Smuzhiyun tegra_ahci_controller_deinit(hpriv);
430*4882a593Smuzhiyun }
431*4882a593Smuzhiyun
432*4882a593Smuzhiyun static struct ata_port_operations ahci_tegra_port_ops = {
433*4882a593Smuzhiyun .inherits = &ahci_ops,
434*4882a593Smuzhiyun .host_stop = tegra_ahci_host_stop,
435*4882a593Smuzhiyun };
436*4882a593Smuzhiyun
437*4882a593Smuzhiyun static const struct ata_port_info ahci_tegra_port_info = {
438*4882a593Smuzhiyun .flags = AHCI_FLAG_COMMON | ATA_FLAG_NO_DIPM,
439*4882a593Smuzhiyun .pio_mask = ATA_PIO4,
440*4882a593Smuzhiyun .udma_mask = ATA_UDMA6,
441*4882a593Smuzhiyun .port_ops = &ahci_tegra_port_ops,
442*4882a593Smuzhiyun };
443*4882a593Smuzhiyun
444*4882a593Smuzhiyun static const char *const tegra124_supply_names[] = {
445*4882a593Smuzhiyun "avdd", "hvdd", "vddio", "target-5v", "target-12v"
446*4882a593Smuzhiyun };
447*4882a593Smuzhiyun
448*4882a593Smuzhiyun static const struct tegra_ahci_ops tegra124_ahci_ops = {
449*4882a593Smuzhiyun .init = tegra124_ahci_init,
450*4882a593Smuzhiyun };
451*4882a593Smuzhiyun
452*4882a593Smuzhiyun static const struct tegra_ahci_soc tegra124_ahci_soc = {
453*4882a593Smuzhiyun .supply_names = tegra124_supply_names,
454*4882a593Smuzhiyun .num_supplies = ARRAY_SIZE(tegra124_supply_names),
455*4882a593Smuzhiyun .supports_devslp = false,
456*4882a593Smuzhiyun .ops = &tegra124_ahci_ops,
457*4882a593Smuzhiyun };
458*4882a593Smuzhiyun
459*4882a593Smuzhiyun static const struct tegra_ahci_soc tegra210_ahci_soc = {
460*4882a593Smuzhiyun .supports_devslp = false,
461*4882a593Smuzhiyun };
462*4882a593Smuzhiyun
463*4882a593Smuzhiyun static const struct of_device_id tegra_ahci_of_match[] = {
464*4882a593Smuzhiyun {
465*4882a593Smuzhiyun .compatible = "nvidia,tegra124-ahci",
466*4882a593Smuzhiyun .data = &tegra124_ahci_soc
467*4882a593Smuzhiyun },
468*4882a593Smuzhiyun {
469*4882a593Smuzhiyun .compatible = "nvidia,tegra210-ahci",
470*4882a593Smuzhiyun .data = &tegra210_ahci_soc
471*4882a593Smuzhiyun },
472*4882a593Smuzhiyun {}
473*4882a593Smuzhiyun };
474*4882a593Smuzhiyun MODULE_DEVICE_TABLE(of, tegra_ahci_of_match);
475*4882a593Smuzhiyun
476*4882a593Smuzhiyun static struct scsi_host_template ahci_platform_sht = {
477*4882a593Smuzhiyun AHCI_SHT(DRV_NAME),
478*4882a593Smuzhiyun };
479*4882a593Smuzhiyun
tegra_ahci_probe(struct platform_device * pdev)480*4882a593Smuzhiyun static int tegra_ahci_probe(struct platform_device *pdev)
481*4882a593Smuzhiyun {
482*4882a593Smuzhiyun struct ahci_host_priv *hpriv;
483*4882a593Smuzhiyun struct tegra_ahci_priv *tegra;
484*4882a593Smuzhiyun struct resource *res;
485*4882a593Smuzhiyun int ret;
486*4882a593Smuzhiyun
487*4882a593Smuzhiyun hpriv = ahci_platform_get_resources(pdev, 0);
488*4882a593Smuzhiyun if (IS_ERR(hpriv))
489*4882a593Smuzhiyun return PTR_ERR(hpriv);
490*4882a593Smuzhiyun
491*4882a593Smuzhiyun tegra = devm_kzalloc(&pdev->dev, sizeof(*tegra), GFP_KERNEL);
492*4882a593Smuzhiyun if (!tegra)
493*4882a593Smuzhiyun return -ENOMEM;
494*4882a593Smuzhiyun
495*4882a593Smuzhiyun hpriv->plat_data = tegra;
496*4882a593Smuzhiyun
497*4882a593Smuzhiyun tegra->pdev = pdev;
498*4882a593Smuzhiyun tegra->soc = of_device_get_match_data(&pdev->dev);
499*4882a593Smuzhiyun
500*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
501*4882a593Smuzhiyun tegra->sata_regs = devm_ioremap_resource(&pdev->dev, res);
502*4882a593Smuzhiyun if (IS_ERR(tegra->sata_regs))
503*4882a593Smuzhiyun return PTR_ERR(tegra->sata_regs);
504*4882a593Smuzhiyun
505*4882a593Smuzhiyun /*
506*4882a593Smuzhiyun * AUX registers is optional.
507*4882a593Smuzhiyun */
508*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_MEM, 2);
509*4882a593Smuzhiyun if (res) {
510*4882a593Smuzhiyun tegra->sata_aux_regs = devm_ioremap_resource(&pdev->dev, res);
511*4882a593Smuzhiyun if (IS_ERR(tegra->sata_aux_regs))
512*4882a593Smuzhiyun return PTR_ERR(tegra->sata_aux_regs);
513*4882a593Smuzhiyun }
514*4882a593Smuzhiyun
515*4882a593Smuzhiyun tegra->sata_rst = devm_reset_control_get(&pdev->dev, "sata");
516*4882a593Smuzhiyun if (IS_ERR(tegra->sata_rst)) {
517*4882a593Smuzhiyun dev_err(&pdev->dev, "Failed to get sata reset\n");
518*4882a593Smuzhiyun return PTR_ERR(tegra->sata_rst);
519*4882a593Smuzhiyun }
520*4882a593Smuzhiyun
521*4882a593Smuzhiyun tegra->sata_oob_rst = devm_reset_control_get(&pdev->dev, "sata-oob");
522*4882a593Smuzhiyun if (IS_ERR(tegra->sata_oob_rst)) {
523*4882a593Smuzhiyun dev_err(&pdev->dev, "Failed to get sata-oob reset\n");
524*4882a593Smuzhiyun return PTR_ERR(tegra->sata_oob_rst);
525*4882a593Smuzhiyun }
526*4882a593Smuzhiyun
527*4882a593Smuzhiyun tegra->sata_cold_rst = devm_reset_control_get(&pdev->dev, "sata-cold");
528*4882a593Smuzhiyun if (IS_ERR(tegra->sata_cold_rst)) {
529*4882a593Smuzhiyun dev_err(&pdev->dev, "Failed to get sata-cold reset\n");
530*4882a593Smuzhiyun return PTR_ERR(tegra->sata_cold_rst);
531*4882a593Smuzhiyun }
532*4882a593Smuzhiyun
533*4882a593Smuzhiyun tegra->sata_clk = devm_clk_get(&pdev->dev, "sata");
534*4882a593Smuzhiyun if (IS_ERR(tegra->sata_clk)) {
535*4882a593Smuzhiyun dev_err(&pdev->dev, "Failed to get sata clock\n");
536*4882a593Smuzhiyun return PTR_ERR(tegra->sata_clk);
537*4882a593Smuzhiyun }
538*4882a593Smuzhiyun
539*4882a593Smuzhiyun tegra->supplies = devm_kcalloc(&pdev->dev,
540*4882a593Smuzhiyun tegra->soc->num_supplies,
541*4882a593Smuzhiyun sizeof(*tegra->supplies), GFP_KERNEL);
542*4882a593Smuzhiyun if (!tegra->supplies)
543*4882a593Smuzhiyun return -ENOMEM;
544*4882a593Smuzhiyun
545*4882a593Smuzhiyun regulator_bulk_set_supply_names(tegra->supplies,
546*4882a593Smuzhiyun tegra->soc->supply_names,
547*4882a593Smuzhiyun tegra->soc->num_supplies);
548*4882a593Smuzhiyun
549*4882a593Smuzhiyun ret = devm_regulator_bulk_get(&pdev->dev,
550*4882a593Smuzhiyun tegra->soc->num_supplies,
551*4882a593Smuzhiyun tegra->supplies);
552*4882a593Smuzhiyun if (ret) {
553*4882a593Smuzhiyun dev_err(&pdev->dev, "Failed to get regulators\n");
554*4882a593Smuzhiyun return ret;
555*4882a593Smuzhiyun }
556*4882a593Smuzhiyun
557*4882a593Smuzhiyun ret = tegra_ahci_controller_init(hpriv);
558*4882a593Smuzhiyun if (ret)
559*4882a593Smuzhiyun return ret;
560*4882a593Smuzhiyun
561*4882a593Smuzhiyun ret = ahci_platform_init_host(pdev, hpriv, &ahci_tegra_port_info,
562*4882a593Smuzhiyun &ahci_platform_sht);
563*4882a593Smuzhiyun if (ret)
564*4882a593Smuzhiyun goto deinit_controller;
565*4882a593Smuzhiyun
566*4882a593Smuzhiyun return 0;
567*4882a593Smuzhiyun
568*4882a593Smuzhiyun deinit_controller:
569*4882a593Smuzhiyun tegra_ahci_controller_deinit(hpriv);
570*4882a593Smuzhiyun
571*4882a593Smuzhiyun return ret;
572*4882a593Smuzhiyun };
573*4882a593Smuzhiyun
574*4882a593Smuzhiyun static struct platform_driver tegra_ahci_driver = {
575*4882a593Smuzhiyun .probe = tegra_ahci_probe,
576*4882a593Smuzhiyun .remove = ata_platform_remove_one,
577*4882a593Smuzhiyun .driver = {
578*4882a593Smuzhiyun .name = DRV_NAME,
579*4882a593Smuzhiyun .of_match_table = tegra_ahci_of_match,
580*4882a593Smuzhiyun },
581*4882a593Smuzhiyun /* LP0 suspend support not implemented */
582*4882a593Smuzhiyun };
583*4882a593Smuzhiyun module_platform_driver(tegra_ahci_driver);
584*4882a593Smuzhiyun
585*4882a593Smuzhiyun MODULE_AUTHOR("Mikko Perttunen <mperttunen@nvidia.com>");
586*4882a593Smuzhiyun MODULE_DESCRIPTION("Tegra AHCI SATA driver");
587*4882a593Smuzhiyun MODULE_LICENSE("GPL v2");
588