hc
2024-10-22 8ac6c7a54ed1b98d142dce24b11c6de6a1e239a5
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
// SPDX-License-Identifier: GPL-2.0+
/*
 * Mediatek MT7621 PCI PHY Driver
 * Author: Sergio Paracuellos <sergio.paracuellos@gmail.com>
 */
 
#include <dt-bindings/phy/phy.h>
#include <linux/bitops.h>
#include <linux/module.h>
#include <linux/of_address.h>
#include <linux/of_device.h>
#include <linux/phy/phy.h>
#include <linux/platform_device.h>
#include <linux/regmap.h>
#include <linux/sys_soc.h>
#include <mt7621.h>
#include <ralink_regs.h>
 
#define RG_PE1_PIPE_REG                0x02c
#define RG_PE1_PIPE_RST                BIT(12)
#define RG_PE1_PIPE_CMD_FRC            BIT(4)
 
#define RG_P0_TO_P1_WIDTH            0x100
#define RG_PE1_H_LCDDS_REG            0x49c
#define RG_PE1_H_LCDDS_PCW            GENMASK(30, 0)
#define RG_PE1_H_LCDDS_PCW_VAL(x)        ((0x7fffffff & (x)) << 0)
 
#define RG_PE1_FRC_H_XTAL_REG            0x400
#define RG_PE1_FRC_H_XTAL_TYPE            BIT(8)
#define RG_PE1_H_XTAL_TYPE            GENMASK(10, 9)
#define RG_PE1_H_XTAL_TYPE_VAL(x)        ((0x3 & (x)) << 9)
 
#define RG_PE1_FRC_PHY_REG            0x000
#define RG_PE1_FRC_PHY_EN            BIT(4)
#define RG_PE1_PHY_EN                BIT(5)
 
#define RG_PE1_H_PLL_REG            0x490
#define RG_PE1_H_PLL_BC                GENMASK(23, 22)
#define RG_PE1_H_PLL_BC_VAL(x)            ((0x3 & (x)) << 22)
#define RG_PE1_H_PLL_BP                GENMASK(21, 18)
#define RG_PE1_H_PLL_BP_VAL(x)            ((0xf & (x)) << 18)
#define RG_PE1_H_PLL_IR                GENMASK(15, 12)
#define RG_PE1_H_PLL_IR_VAL(x)            ((0xf & (x)) << 12)
#define RG_PE1_H_PLL_IC                GENMASK(11, 8)
#define RG_PE1_H_PLL_IC_VAL(x)            ((0xf & (x)) << 8)
#define RG_PE1_H_PLL_PREDIV            GENMASK(7, 6)
#define RG_PE1_H_PLL_PREDIV_VAL(x)        ((0x3 & (x)) << 6)
#define RG_PE1_PLL_DIVEN            GENMASK(3, 1)
#define RG_PE1_PLL_DIVEN_VAL(x)            ((0x7 & (x)) << 1)
 
#define RG_PE1_H_PLL_FBKSEL_REG            0x4bc
#define RG_PE1_H_PLL_FBKSEL            GENMASK(5, 4)
#define RG_PE1_H_PLL_FBKSEL_VAL(x)        ((0x3 & (x)) << 4)
 
#define    RG_PE1_H_LCDDS_SSC_PRD_REG        0x4a4
#define RG_PE1_H_LCDDS_SSC_PRD            GENMASK(15, 0)
#define RG_PE1_H_LCDDS_SSC_PRD_VAL(x)        ((0xffff & (x)) << 0)
 
#define RG_PE1_H_LCDDS_SSC_DELTA_REG        0x4a8
#define RG_PE1_H_LCDDS_SSC_DELTA        GENMASK(11, 0)
#define RG_PE1_H_LCDDS_SSC_DELTA_VAL(x)        ((0xfff & (x)) << 0)
#define RG_PE1_H_LCDDS_SSC_DELTA1        GENMASK(27, 16)
#define RG_PE1_H_LCDDS_SSC_DELTA1_VAL(x)    ((0xff & (x)) << 16)
 
#define RG_PE1_LCDDS_CLK_PH_INV_REG        0x4a0
#define RG_PE1_LCDDS_CLK_PH_INV            BIT(5)
 
#define RG_PE1_H_PLL_BR_REG            0x4ac
#define RG_PE1_H_PLL_BR                GENMASK(18, 16)
#define RG_PE1_H_PLL_BR_VAL(x)            ((0x7 & (x)) << 16)
 
#define    RG_PE1_MSTCKDIV_REG            0x414
#define RG_PE1_MSTCKDIV                GENMASK(7, 6)
#define RG_PE1_MSTCKDIV_VAL(x)            ((0x3 & (x)) << 6)
 
#define RG_PE1_FRC_MSTCKDIV            BIT(5)
 
#define XTAL_MODE_SEL_SHIFT            6
#define XTAL_MODE_SEL_MASK            0x7
 
#define MAX_PHYS    2
 
/**
 * struct mt7621_pci_phy - Mt7621 Pcie PHY core
 * @dev: pointer to device
 * @regmap: kernel regmap pointer
 * @phy: pointer to the kernel PHY device
 * @port_base: base register
 * @has_dual_port: if the phy has dual ports.
 * @bypass_pipe_rst: mark if 'mt7621_bypass_pipe_rst'
 * needs to be executed. Depends on chip revision.
 */
struct mt7621_pci_phy {
   struct device *dev;
   struct regmap *regmap;
   struct phy *phy;
   void __iomem *port_base;
   bool has_dual_port;
   bool bypass_pipe_rst;
};
 
static inline u32 phy_read(struct mt7621_pci_phy *phy, u32 reg)
{
   u32 val;
 
   regmap_read(phy->regmap, reg, &val);
 
   return val;
}
 
static inline void phy_write(struct mt7621_pci_phy *phy, u32 val, u32 reg)
{
   regmap_write(phy->regmap, reg, val);
}
 
static inline void mt7621_phy_rmw(struct mt7621_pci_phy *phy,
                 u32 reg, u32 clr, u32 set)
{
   u32 val = phy_read(phy, reg);
 
   val &= ~clr;
   val |= set;
   phy_write(phy, val, reg);
}
 
static void mt7621_bypass_pipe_rst(struct mt7621_pci_phy *phy)
{
   mt7621_phy_rmw(phy, RG_PE1_PIPE_REG, 0, RG_PE1_PIPE_RST);
   mt7621_phy_rmw(phy, RG_PE1_PIPE_REG, 0, RG_PE1_PIPE_CMD_FRC);
 
   if (phy->has_dual_port) {
       mt7621_phy_rmw(phy, RG_PE1_PIPE_REG + RG_P0_TO_P1_WIDTH,
                  0, RG_PE1_PIPE_RST);
       mt7621_phy_rmw(phy, RG_PE1_PIPE_REG + RG_P0_TO_P1_WIDTH,
                  0, RG_PE1_PIPE_CMD_FRC);
   }
}
 
static void mt7621_set_phy_for_ssc(struct mt7621_pci_phy *phy)
{
   struct device *dev = phy->dev;
   u32 xtal_mode;
 
   xtal_mode = (rt_sysc_r32(SYSC_REG_SYSTEM_CONFIG0)
            >> XTAL_MODE_SEL_SHIFT) & XTAL_MODE_SEL_MASK;
 
   /* Set PCIe Port PHY to disable SSC */
   /* Debug Xtal Type */
   mt7621_phy_rmw(phy, RG_PE1_FRC_H_XTAL_REG,
              RG_PE1_FRC_H_XTAL_TYPE | RG_PE1_H_XTAL_TYPE,
              RG_PE1_FRC_H_XTAL_TYPE | RG_PE1_H_XTAL_TYPE_VAL(0x00));
 
   /* disable port */
   mt7621_phy_rmw(phy, RG_PE1_FRC_PHY_REG,
              RG_PE1_PHY_EN, RG_PE1_FRC_PHY_EN);
 
   if (phy->has_dual_port) {
       mt7621_phy_rmw(phy, RG_PE1_FRC_PHY_REG + RG_P0_TO_P1_WIDTH,
                  RG_PE1_PHY_EN, RG_PE1_FRC_PHY_EN);
   }
 
   if (xtal_mode <= 5 && xtal_mode >= 3) { /* 40MHz Xtal */
       /* Set Pre-divider ratio (for host mode) */
       mt7621_phy_rmw(phy, RG_PE1_H_PLL_REG,
                  RG_PE1_H_PLL_PREDIV,
                  RG_PE1_H_PLL_PREDIV_VAL(0x01));
       dev_info(dev, "Xtal is 40MHz\n");
   } else if (xtal_mode >= 6) { /* 25MHz Xal */
       mt7621_phy_rmw(phy, RG_PE1_H_PLL_REG,
                  RG_PE1_H_PLL_PREDIV,
                  RG_PE1_H_PLL_PREDIV_VAL(0x00));
       /* Select feedback clock */
       mt7621_phy_rmw(phy, RG_PE1_H_PLL_FBKSEL_REG,
                  RG_PE1_H_PLL_FBKSEL,
                  RG_PE1_H_PLL_FBKSEL_VAL(0x01));
       /* DDS NCPO PCW (for host mode) */
       mt7621_phy_rmw(phy, RG_PE1_H_LCDDS_SSC_PRD_REG,
                  RG_PE1_H_LCDDS_SSC_PRD,
                  RG_PE1_H_LCDDS_SSC_PRD_VAL(0x18000000));
       /* DDS SSC dither period control */
       mt7621_phy_rmw(phy, RG_PE1_H_LCDDS_SSC_PRD_REG,
                  RG_PE1_H_LCDDS_SSC_PRD,
                  RG_PE1_H_LCDDS_SSC_PRD_VAL(0x18d));
       /* DDS SSC dither amplitude control */
       mt7621_phy_rmw(phy, RG_PE1_H_LCDDS_SSC_DELTA_REG,
                  RG_PE1_H_LCDDS_SSC_DELTA |
                  RG_PE1_H_LCDDS_SSC_DELTA1,
                  RG_PE1_H_LCDDS_SSC_DELTA_VAL(0x4a) |
                  RG_PE1_H_LCDDS_SSC_DELTA1_VAL(0x4a));
       dev_info(dev, "Xtal is 25MHz\n");
   } else { /* 20MHz Xtal */
       mt7621_phy_rmw(phy, RG_PE1_H_PLL_REG,
                  RG_PE1_H_PLL_PREDIV,
                  RG_PE1_H_PLL_PREDIV_VAL(0x00));
 
       dev_info(dev, "Xtal is 20MHz\n");
   }
 
   /* DDS clock inversion */
   mt7621_phy_rmw(phy, RG_PE1_LCDDS_CLK_PH_INV_REG,
              RG_PE1_LCDDS_CLK_PH_INV, RG_PE1_LCDDS_CLK_PH_INV);
 
   /* Set PLL bits */
   mt7621_phy_rmw(phy, RG_PE1_H_PLL_REG,
              RG_PE1_H_PLL_BC | RG_PE1_H_PLL_BP | RG_PE1_H_PLL_IR |
              RG_PE1_H_PLL_IC | RG_PE1_PLL_DIVEN,
              RG_PE1_H_PLL_BC_VAL(0x02) | RG_PE1_H_PLL_BP_VAL(0x06) |
              RG_PE1_H_PLL_IR_VAL(0x02) | RG_PE1_H_PLL_IC_VAL(0x01) |
              RG_PE1_PLL_DIVEN_VAL(0x02));
 
   mt7621_phy_rmw(phy, RG_PE1_H_PLL_BR_REG,
              RG_PE1_H_PLL_BR, RG_PE1_H_PLL_BR_VAL(0x00));
 
   if (xtal_mode <= 5 && xtal_mode >= 3) { /* 40MHz Xtal */
       /* set force mode enable of da_pe1_mstckdiv */
       mt7621_phy_rmw(phy, RG_PE1_MSTCKDIV_REG,
                  RG_PE1_MSTCKDIV | RG_PE1_FRC_MSTCKDIV,
                  RG_PE1_MSTCKDIV_VAL(0x01) | RG_PE1_FRC_MSTCKDIV);
   }
}
 
static int mt7621_pci_phy_init(struct phy *phy)
{
   struct mt7621_pci_phy *mphy = phy_get_drvdata(phy);
 
   if (mphy->bypass_pipe_rst)
       mt7621_bypass_pipe_rst(mphy);
 
   mt7621_set_phy_for_ssc(mphy);
 
   return 0;
}
 
static int mt7621_pci_phy_power_on(struct phy *phy)
{
   struct mt7621_pci_phy *mphy = phy_get_drvdata(phy);
 
   /* Enable PHY and disable force mode */
   mt7621_phy_rmw(mphy, RG_PE1_FRC_PHY_REG,
              RG_PE1_FRC_PHY_EN, RG_PE1_PHY_EN);
 
   if (mphy->has_dual_port) {
       mt7621_phy_rmw(mphy, RG_PE1_FRC_PHY_REG + RG_P0_TO_P1_WIDTH,
                  RG_PE1_FRC_PHY_EN, RG_PE1_PHY_EN);
   }
 
   return 0;
}
 
static int mt7621_pci_phy_power_off(struct phy *phy)
{
   struct mt7621_pci_phy *mphy = phy_get_drvdata(phy);
 
   /* Disable PHY */
   mt7621_phy_rmw(mphy, RG_PE1_FRC_PHY_REG,
              RG_PE1_PHY_EN, RG_PE1_FRC_PHY_EN);
 
   if (mphy->has_dual_port) {
       mt7621_phy_rmw(mphy, RG_PE1_FRC_PHY_REG + RG_P0_TO_P1_WIDTH,
                  RG_PE1_PHY_EN, RG_PE1_FRC_PHY_EN);
   }
 
   return 0;
}
 
static int mt7621_pci_phy_exit(struct phy *phy)
{
   return 0;
}
 
static const struct phy_ops mt7621_pci_phy_ops = {
   .init        = mt7621_pci_phy_init,
   .exit        = mt7621_pci_phy_exit,
   .power_on    = mt7621_pci_phy_power_on,
   .power_off    = mt7621_pci_phy_power_off,
   .owner        = THIS_MODULE,
};
 
static struct phy *mt7621_pcie_phy_of_xlate(struct device *dev,
                       struct of_phandle_args *args)
{
   struct mt7621_pci_phy *mt7621_phy = dev_get_drvdata(dev);
 
   if (WARN_ON(args->args[0] >= MAX_PHYS))
       return ERR_PTR(-ENODEV);
 
   mt7621_phy->has_dual_port = args->args[0];
 
   dev_info(dev, "PHY for 0x%08x (dual port = %d)\n",
        (unsigned int)mt7621_phy->port_base, mt7621_phy->has_dual_port);
 
   return mt7621_phy->phy;
}
 
static const struct soc_device_attribute mt7621_pci_quirks_match[] = {
   { .soc_id = "mt7621", .revision = "E2" }
};
 
static const struct regmap_config mt7621_pci_phy_regmap_config = {
   .reg_bits = 32,
   .val_bits = 32,
   .reg_stride = 4,
   .max_register = 0x700,
};
 
static int mt7621_pci_phy_probe(struct platform_device *pdev)
{
   struct device *dev = &pdev->dev;
   const struct soc_device_attribute *attr;
   struct phy_provider *provider;
   struct mt7621_pci_phy *phy;
   struct resource *res;
 
   phy = devm_kzalloc(dev, sizeof(*phy), GFP_KERNEL);
   if (!phy)
       return -ENOMEM;
 
   attr = soc_device_match(mt7621_pci_quirks_match);
   if (attr)
       phy->bypass_pipe_rst = true;
 
   phy->dev = dev;
   platform_set_drvdata(pdev, phy);
 
   res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
   if (!res) {
       dev_err(dev, "failed to get address resource\n");
       return -ENXIO;
   }
 
   phy->port_base = devm_ioremap_resource(dev, res);
   if (IS_ERR(phy->port_base)) {
       dev_err(dev, "failed to remap phy regs\n");
       return PTR_ERR(phy->port_base);
   }
 
   phy->regmap = devm_regmap_init_mmio(phy->dev, phy->port_base,
                       &mt7621_pci_phy_regmap_config);
   if (IS_ERR(phy->regmap))
       return PTR_ERR(phy->regmap);
 
   phy->phy = devm_phy_create(dev, dev->of_node, &mt7621_pci_phy_ops);
   if (IS_ERR(phy)) {
       dev_err(dev, "failed to create phy\n");
       return PTR_ERR(phy);
   }
 
   phy_set_drvdata(phy->phy, phy);
 
   provider = devm_of_phy_provider_register(dev, mt7621_pcie_phy_of_xlate);
 
   return PTR_ERR_OR_ZERO(provider);
}
 
static const struct of_device_id mt7621_pci_phy_ids[] = {
   { .compatible = "mediatek,mt7621-pci-phy" },
   {},
};
MODULE_DEVICE_TABLE(of, mt7621_pci_ids);
 
static struct platform_driver mt7621_pci_phy_driver = {
   .probe = mt7621_pci_phy_probe,
   .driver = {
       .name = "mt7621-pci-phy",
       .of_match_table = of_match_ptr(mt7621_pci_phy_ids),
   },
};
 
builtin_platform_driver(mt7621_pci_phy_driver);
 
MODULE_AUTHOR("Sergio Paracuellos <sergio.paracuellos@gmail.com>");
MODULE_DESCRIPTION("MediaTek MT7621 PCIe PHY driver");
MODULE_LICENSE("GPL v2");