common: Drop linux/bitops.h from common header
[platform/kernel/u-boot.git] / drivers / power / domain / mtk-power-domain.c
1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * Copyright (C) 2018 MediaTek Inc.
4  * Author: Ryder Lee <ryder.lee@mediatek.com>
5  */
6
7 #include <clk.h>
8 #include <common.h>
9 #include <dm.h>
10 #include <malloc.h>
11 #include <power-domain-uclass.h>
12 #include <regmap.h>
13 #include <syscon.h>
14 #include <asm/io.h>
15 #include <asm/processor.h>
16 #include <linux/bitops.h>
17 #include <linux/err.h>
18 #include <linux/iopoll.h>
19
20 #include <dt-bindings/power/mt7623-power.h>
21 #include <dt-bindings/power/mt7629-power.h>
22
23 #define SPM_EN                  (0xb16 << 16 | 0x1)
24 #define SPM_VDE_PWR_CON         0x0210
25 #define SPM_MFG_PWR_CON         0x0214
26 #define SPM_ISP_PWR_CON         0x0238
27 #define SPM_DIS_PWR_CON         0x023c
28 #define SPM_CONN_PWR_CON        0x0280
29 #define SPM_BDP_PWR_CON         0x029c
30 #define SPM_ETH_PWR_CON         0x02a0
31 #define SPM_HIF_PWR_CON         0x02a4
32 #define SPM_IFR_MSC_PWR_CON     0x02a8
33 #define SPM_ETHSYS_PWR_CON      0x2e0
34 #define SPM_HIF0_PWR_CON        0x2e4
35 #define SPM_HIF1_PWR_CON        0x2e8
36 #define SPM_PWR_STATUS          0x60c
37 #define SPM_PWR_STATUS_2ND      0x610
38
39 #define PWR_RST_B_BIT           BIT(0)
40 #define PWR_ISO_BIT             BIT(1)
41 #define PWR_ON_BIT              BIT(2)
42 #define PWR_ON_2ND_BIT          BIT(3)
43 #define PWR_CLK_DIS_BIT         BIT(4)
44
45 #define PWR_STATUS_CONN         BIT(1)
46 #define PWR_STATUS_DISP         BIT(3)
47 #define PWR_STATUS_MFG          BIT(4)
48 #define PWR_STATUS_ISP          BIT(5)
49 #define PWR_STATUS_VDEC         BIT(7)
50 #define PWR_STATUS_BDP          BIT(14)
51 #define PWR_STATUS_ETH          BIT(15)
52 #define PWR_STATUS_HIF          BIT(16)
53 #define PWR_STATUS_IFR_MSC      BIT(17)
54 #define PWR_STATUS_ETHSYS       BIT(24)
55 #define PWR_STATUS_HIF0         BIT(25)
56 #define PWR_STATUS_HIF1         BIT(26)
57
58 /* Infrasys configuration */
59 #define INFRA_TOPDCM_CTRL       0x10
60 #define INFRA_TOPAXI_PROT_EN    0x220
61 #define INFRA_TOPAXI_PROT_STA1  0x228
62
63 #define DCM_TOP_EN              BIT(0)
64
65 enum scp_domain_type {
66         SCPSYS_MT7622,
67         SCPSYS_MT7623,
68         SCPSYS_MT7629,
69 };
70
71 struct scp_domain;
72
73 struct scp_domain_data {
74         struct scp_domain *scpd;
75         u32 sta_mask;
76         int ctl_offs;
77         u32 sram_pdn_bits;
78         u32 sram_pdn_ack_bits;
79         u32 bus_prot_mask;
80 };
81
82 struct scp_domain {
83         void __iomem *base;
84         void __iomem *infracfg;
85         enum scp_domain_type type;
86         struct scp_domain_data *data;
87 };
88
89 static struct scp_domain_data scp_domain_mt7623[] = {
90         [MT7623_POWER_DOMAIN_CONN] = {
91                 .sta_mask = PWR_STATUS_CONN,
92                 .ctl_offs = SPM_CONN_PWR_CON,
93                 .bus_prot_mask = BIT(8) | BIT(2),
94         },
95         [MT7623_POWER_DOMAIN_DISP] = {
96                 .sta_mask = PWR_STATUS_DISP,
97                 .ctl_offs = SPM_DIS_PWR_CON,
98                 .sram_pdn_bits = GENMASK(11, 8),
99                 .bus_prot_mask = BIT(2),
100         },
101         [MT7623_POWER_DOMAIN_MFG] = {
102                 .sta_mask = PWR_STATUS_MFG,
103                 .ctl_offs = SPM_MFG_PWR_CON,
104                 .sram_pdn_bits = GENMASK(11, 8),
105                 .sram_pdn_ack_bits = GENMASK(12, 12),
106         },
107         [MT7623_POWER_DOMAIN_VDEC] = {
108                 .sta_mask = PWR_STATUS_VDEC,
109                 .ctl_offs = SPM_VDE_PWR_CON,
110                 .sram_pdn_bits = GENMASK(11, 8),
111                 .sram_pdn_ack_bits = GENMASK(12, 12),
112         },
113         [MT7623_POWER_DOMAIN_ISP] = {
114                 .sta_mask = PWR_STATUS_ISP,
115                 .ctl_offs = SPM_ISP_PWR_CON,
116                 .sram_pdn_bits = GENMASK(11, 8),
117                 .sram_pdn_ack_bits = GENMASK(13, 12),
118         },
119         [MT7623_POWER_DOMAIN_BDP] = {
120                 .sta_mask = PWR_STATUS_BDP,
121                 .ctl_offs = SPM_BDP_PWR_CON,
122                 .sram_pdn_bits = GENMASK(11, 8),
123         },
124         [MT7623_POWER_DOMAIN_ETH] = {
125                 .sta_mask = PWR_STATUS_ETH,
126                 .ctl_offs = SPM_ETH_PWR_CON,
127                 .sram_pdn_bits = GENMASK(11, 8),
128                 .sram_pdn_ack_bits = GENMASK(15, 12),
129         },
130         [MT7623_POWER_DOMAIN_HIF] = {
131                 .sta_mask = PWR_STATUS_HIF,
132                 .ctl_offs = SPM_HIF_PWR_CON,
133                 .sram_pdn_bits = GENMASK(11, 8),
134                 .sram_pdn_ack_bits = GENMASK(15, 12),
135         },
136         [MT7623_POWER_DOMAIN_IFR_MSC] = {
137                 .sta_mask = PWR_STATUS_IFR_MSC,
138                 .ctl_offs = SPM_IFR_MSC_PWR_CON,
139         },
140 };
141
142 static struct scp_domain_data scp_domain_mt7629[] = {
143         [MT7629_POWER_DOMAIN_ETHSYS] = {
144                 .sta_mask = PWR_STATUS_ETHSYS,
145                 .ctl_offs = SPM_ETHSYS_PWR_CON,
146                 .sram_pdn_bits = GENMASK(11, 8),
147                 .sram_pdn_ack_bits = GENMASK(15, 12),
148                 .bus_prot_mask = (BIT(3) | BIT(17)),
149         },
150         [MT7629_POWER_DOMAIN_HIF0] = {
151                 .sta_mask = PWR_STATUS_HIF0,
152                 .ctl_offs = SPM_HIF0_PWR_CON,
153                 .sram_pdn_bits = GENMASK(11, 8),
154                 .sram_pdn_ack_bits = GENMASK(15, 12),
155                 .bus_prot_mask = GENMASK(25, 24),
156         },
157         [MT7629_POWER_DOMAIN_HIF1] = {
158                 .sta_mask = PWR_STATUS_HIF1,
159                 .ctl_offs = SPM_HIF1_PWR_CON,
160                 .sram_pdn_bits = GENMASK(11, 8),
161                 .sram_pdn_ack_bits = GENMASK(15, 12),
162                 .bus_prot_mask = GENMASK(28, 26),
163         },
164 };
165
166 /**
167  * This function enables the bus protection bits for disabled power
168  * domains so that the system does not hang when some unit accesses the
169  * bus while in power down.
170  */
171 static int mtk_infracfg_set_bus_protection(void __iomem *infracfg,
172                                            u32 mask)
173 {
174         u32 val;
175
176         clrsetbits_le32(infracfg + INFRA_TOPAXI_PROT_EN, mask, mask);
177
178         return readl_poll_timeout(infracfg + INFRA_TOPAXI_PROT_STA1, val,
179                                   (val & mask) == mask, 100);
180 }
181
182 static int mtk_infracfg_clear_bus_protection(void __iomem *infracfg,
183                                              u32 mask)
184 {
185         u32 val;
186
187         clrbits_le32(infracfg + INFRA_TOPAXI_PROT_EN, mask);
188
189         return readl_poll_timeout(infracfg + INFRA_TOPAXI_PROT_STA1, val,
190                                   !(val & mask), 100);
191 }
192
193 static int scpsys_domain_is_on(struct scp_domain_data *data)
194 {
195         struct scp_domain *scpd = data->scpd;
196         u32 sta = readl(scpd->base + SPM_PWR_STATUS) &
197                         data->sta_mask;
198         u32 sta2 = readl(scpd->base + SPM_PWR_STATUS_2ND) &
199                          data->sta_mask;
200
201         /*
202          * A domain is on when both status bits are set. If only one is set
203          * return an error. This happens while powering up a domain
204          */
205         if (sta && sta2)
206                 return true;
207         if (!sta && !sta2)
208                 return false;
209
210         return -EINVAL;
211 }
212
213 static int scpsys_power_on(struct power_domain *power_domain)
214 {
215         struct scp_domain *scpd = dev_get_priv(power_domain->dev);
216         struct scp_domain_data *data = &scpd->data[power_domain->id];
217         void __iomem *ctl_addr = scpd->base + data->ctl_offs;
218         u32 pdn_ack = data->sram_pdn_ack_bits;
219         u32 val;
220         int ret, tmp;
221
222         writel(SPM_EN, scpd->base);
223
224         val = readl(ctl_addr);
225         val |= PWR_ON_BIT;
226         writel(val, ctl_addr);
227
228         val |= PWR_ON_2ND_BIT;
229         writel(val, ctl_addr);
230
231         ret = readx_poll_timeout(scpsys_domain_is_on, data, tmp, tmp > 0,
232                                  100);
233         if (ret < 0)
234                 return ret;
235
236         val &= ~PWR_CLK_DIS_BIT;
237         writel(val, ctl_addr);
238
239         val &= ~PWR_ISO_BIT;
240         writel(val, ctl_addr);
241
242         val |= PWR_RST_B_BIT;
243         writel(val, ctl_addr);
244
245         val &= ~data->sram_pdn_bits;
246         writel(val, ctl_addr);
247
248         ret = readl_poll_timeout(ctl_addr, tmp, !(tmp & pdn_ack), 100);
249         if (ret < 0)
250                 return ret;
251
252         if (data->bus_prot_mask) {
253                 ret = mtk_infracfg_clear_bus_protection(scpd->infracfg,
254                                                         data->bus_prot_mask);
255                 if (ret)
256                         return ret;
257         }
258
259         return 0;
260 }
261
262 static int scpsys_power_off(struct power_domain *power_domain)
263 {
264         struct scp_domain *scpd = dev_get_priv(power_domain->dev);
265         struct scp_domain_data *data = &scpd->data[power_domain->id];
266         void __iomem *ctl_addr = scpd->base + data->ctl_offs;
267         u32 pdn_ack = data->sram_pdn_ack_bits;
268         u32 val;
269         int ret, tmp;
270
271         if (data->bus_prot_mask) {
272                 ret = mtk_infracfg_set_bus_protection(scpd->infracfg,
273                                                       data->bus_prot_mask);
274                 if (ret)
275                         return ret;
276         }
277
278         val = readl(ctl_addr);
279         val |= data->sram_pdn_bits;
280         writel(val, ctl_addr);
281
282         ret = readl_poll_timeout(ctl_addr, tmp, (tmp & pdn_ack) == pdn_ack,
283                                  100);
284         if (ret < 0)
285                 return ret;
286
287         val |= PWR_ISO_BIT;
288         writel(val, ctl_addr);
289
290         val &= ~PWR_RST_B_BIT;
291         writel(val, ctl_addr);
292
293         val |= PWR_CLK_DIS_BIT;
294         writel(val, ctl_addr);
295
296         val &= ~PWR_ON_BIT;
297         writel(val, ctl_addr);
298
299         val &= ~PWR_ON_2ND_BIT;
300         writel(val, ctl_addr);
301
302         ret = readx_poll_timeout(scpsys_domain_is_on, data, tmp, !tmp, 100);
303         if (ret < 0)
304                 return ret;
305
306         return 0;
307 }
308
309 static int scpsys_power_request(struct power_domain *power_domain)
310 {
311         struct scp_domain *scpd = dev_get_priv(power_domain->dev);
312         struct scp_domain_data *data;
313
314         data = &scpd->data[power_domain->id];
315         data->scpd = scpd;
316
317         return 0;
318 }
319
320 static int scpsys_power_free(struct power_domain *power_domain)
321 {
322         return 0;
323 }
324
325 static int mtk_power_domain_hook(struct udevice *dev)
326 {
327         struct scp_domain *scpd = dev_get_priv(dev);
328
329         scpd->type = (enum scp_domain_type)dev_get_driver_data(dev);
330
331         switch (scpd->type) {
332         case SCPSYS_MT7623:
333                 scpd->data = scp_domain_mt7623;
334                 break;
335         case SCPSYS_MT7622:
336         case SCPSYS_MT7629:
337                 scpd->data = scp_domain_mt7629;
338                 break;
339         default:
340                 return -EINVAL;
341         }
342
343         return 0;
344 }
345
346 static int mtk_power_domain_probe(struct udevice *dev)
347 {
348         struct ofnode_phandle_args args;
349         struct scp_domain *scpd = dev_get_priv(dev);
350         struct regmap *regmap;
351         struct clk_bulk bulk;
352         int err;
353
354         scpd->base = dev_read_addr_ptr(dev);
355         if (!scpd->base)
356                 return -ENOENT;
357
358         err = mtk_power_domain_hook(dev);
359         if (err)
360                 return err;
361
362         /* get corresponding syscon phandle */
363         err = dev_read_phandle_with_args(dev, "infracfg", NULL, 0, 0, &args);
364         if (err)
365                 return err;
366
367         regmap = syscon_node_to_regmap(args.node);
368         if (IS_ERR(regmap))
369                 return PTR_ERR(regmap);
370
371         scpd->infracfg = regmap_get_range(regmap, 0);
372         if (!scpd->infracfg)
373                 return -ENOENT;
374
375         /* enable Infra DCM */
376         setbits_le32(scpd->infracfg + INFRA_TOPDCM_CTRL, DCM_TOP_EN);
377
378         err = clk_get_bulk(dev, &bulk);
379         if (err)
380                 return err;
381
382         return clk_enable_bulk(&bulk);
383 }
384
385 static const struct udevice_id mtk_power_domain_ids[] = {
386         {
387                 .compatible = "mediatek,mt7622-scpsys",
388                 .data = SCPSYS_MT7622,
389         },
390         {
391                 .compatible = "mediatek,mt7623-scpsys",
392                 .data = SCPSYS_MT7623,
393         },
394         {
395                 .compatible = "mediatek,mt7629-scpsys",
396                 .data = SCPSYS_MT7629,
397         },
398         { /* sentinel */ }
399 };
400
401 struct power_domain_ops mtk_power_domain_ops = {
402         .rfree = scpsys_power_free,
403         .off = scpsys_power_off,
404         .on = scpsys_power_on,
405         .request = scpsys_power_request,
406 };
407
408 U_BOOT_DRIVER(mtk_power_domain) = {
409         .name = "mtk_power_domain",
410         .id = UCLASS_POWER_DOMAIN,
411         .ops = &mtk_power_domain_ops,
412         .probe = mtk_power_domain_probe,
413         .of_match = mtk_power_domain_ids,
414         .priv_auto_alloc_size = sizeof(struct scp_domain),
415 };