.. | .. |
---|
| 1 | +// SPDX-License-Identifier: GPL-2.0-only |
---|
1 | 2 | /* |
---|
2 | 3 | * Copyright (c) 2015-2016 MediaTek Inc. |
---|
3 | 4 | * Author: Yong Wu <yong.wu@mediatek.com> |
---|
4 | | - * |
---|
5 | | - * This program is free software; you can redistribute it and/or modify |
---|
6 | | - * it under the terms of the GNU General Public License version 2 as |
---|
7 | | - * published by the Free Software Foundation. |
---|
8 | | - * |
---|
9 | | - * This program is distributed in the hope that it will be useful, |
---|
10 | | - * but WITHOUT ANY WARRANTY; without even the implied warranty of |
---|
11 | | - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
---|
12 | | - * GNU General Public License for more details. |
---|
13 | 5 | */ |
---|
14 | 6 | #include <linux/clk.h> |
---|
15 | 7 | #include <linux/component.h> |
---|
.. | .. |
---|
23 | 15 | #include <linux/pm_runtime.h> |
---|
24 | 16 | #include <soc/mediatek/smi.h> |
---|
25 | 17 | #include <dt-bindings/memory/mt2701-larb-port.h> |
---|
| 18 | +#include <dt-bindings/memory/mtk-memory-port.h> |
---|
26 | 19 | |
---|
27 | 20 | /* mt8173 */ |
---|
28 | 21 | #define SMI_LARB_MMU_EN 0xf00 |
---|
| 22 | + |
---|
| 23 | +/* mt8167 */ |
---|
| 24 | +#define MT8167_SMI_LARB_MMU_EN 0xfc0 |
---|
29 | 25 | |
---|
30 | 26 | /* mt2701 */ |
---|
31 | 27 | #define REG_SMI_SECUR_CON_BASE 0x5c0 |
---|
.. | .. |
---|
48 | 44 | /* mt2712 */ |
---|
49 | 45 | #define SMI_LARB_NONSEC_CON(id) (0x380 + ((id) * 4)) |
---|
50 | 46 | #define F_MMU_EN BIT(0) |
---|
| 47 | +#define BANK_SEL(id) ({ \ |
---|
| 48 | + u32 _id = (id) & 0x3; \ |
---|
| 49 | + (_id << 8 | _id << 10 | _id << 12 | _id << 14); \ |
---|
| 50 | +}) |
---|
| 51 | + |
---|
| 52 | +/* SMI COMMON */ |
---|
| 53 | +#define SMI_BUS_SEL 0x220 |
---|
| 54 | +#define SMI_BUS_LARB_SHIFT(larbid) ((larbid) << 1) |
---|
| 55 | +/* All are MMU0 defaultly. Only specialize mmu1 here. */ |
---|
| 56 | +#define F_MMU1_LARB(larbid) (0x1 << SMI_BUS_LARB_SHIFT(larbid)) |
---|
| 57 | + |
---|
| 58 | +enum mtk_smi_gen { |
---|
| 59 | + MTK_SMI_GEN1, |
---|
| 60 | + MTK_SMI_GEN2 |
---|
| 61 | +}; |
---|
| 62 | + |
---|
| 63 | +struct mtk_smi_common_plat { |
---|
| 64 | + enum mtk_smi_gen gen; |
---|
| 65 | + bool has_gals; |
---|
| 66 | + u32 bus_sel; /* Balance some larbs to enter mmu0 or mmu1 */ |
---|
| 67 | +}; |
---|
51 | 68 | |
---|
52 | 69 | struct mtk_smi_larb_gen { |
---|
53 | | - bool need_larbid; |
---|
54 | 70 | int port_in_larb[MTK_LARB_NR_MAX + 1]; |
---|
55 | | - void (*config_port)(struct device *); |
---|
| 71 | + void (*config_port)(struct device *dev); |
---|
| 72 | + unsigned int larb_direct_to_common_mask; |
---|
| 73 | + bool has_gals; |
---|
56 | 74 | }; |
---|
57 | 75 | |
---|
58 | 76 | struct mtk_smi { |
---|
59 | 77 | struct device *dev; |
---|
60 | 78 | struct clk *clk_apb, *clk_smi; |
---|
| 79 | + struct clk *clk_gals0, *clk_gals1; |
---|
61 | 80 | struct clk *clk_async; /*only needed by mt2701*/ |
---|
62 | | - void __iomem *smi_ao_base; |
---|
| 81 | + union { |
---|
| 82 | + void __iomem *smi_ao_base; /* only for gen1 */ |
---|
| 83 | + void __iomem *base; /* only for gen2 */ |
---|
| 84 | + }; |
---|
| 85 | + const struct mtk_smi_common_plat *plat; |
---|
63 | 86 | }; |
---|
64 | 87 | |
---|
65 | 88 | struct mtk_smi_larb { /* larb: local arbiter */ |
---|
.. | .. |
---|
69 | 92 | const struct mtk_smi_larb_gen *larb_gen; |
---|
70 | 93 | int larbid; |
---|
71 | 94 | u32 *mmu; |
---|
| 95 | + unsigned char *bank; |
---|
72 | 96 | }; |
---|
73 | 97 | |
---|
74 | | -enum mtk_smi_gen { |
---|
75 | | - MTK_SMI_GEN1, |
---|
76 | | - MTK_SMI_GEN2 |
---|
77 | | -}; |
---|
78 | | - |
---|
79 | | -static int mtk_smi_enable(const struct mtk_smi *smi) |
---|
| 98 | +static int mtk_smi_clk_enable(const struct mtk_smi *smi) |
---|
80 | 99 | { |
---|
81 | 100 | int ret; |
---|
82 | 101 | |
---|
83 | | - ret = pm_runtime_get_sync(smi->dev); |
---|
84 | | - if (ret < 0) |
---|
85 | | - return ret; |
---|
86 | | - |
---|
87 | 102 | ret = clk_prepare_enable(smi->clk_apb); |
---|
88 | 103 | if (ret) |
---|
89 | | - goto err_put_pm; |
---|
| 104 | + return ret; |
---|
90 | 105 | |
---|
91 | 106 | ret = clk_prepare_enable(smi->clk_smi); |
---|
92 | 107 | if (ret) |
---|
93 | 108 | goto err_disable_apb; |
---|
94 | 109 | |
---|
| 110 | + ret = clk_prepare_enable(smi->clk_gals0); |
---|
| 111 | + if (ret) |
---|
| 112 | + goto err_disable_smi; |
---|
| 113 | + |
---|
| 114 | + ret = clk_prepare_enable(smi->clk_gals1); |
---|
| 115 | + if (ret) |
---|
| 116 | + goto err_disable_gals0; |
---|
| 117 | + |
---|
95 | 118 | return 0; |
---|
96 | 119 | |
---|
| 120 | +err_disable_gals0: |
---|
| 121 | + clk_disable_unprepare(smi->clk_gals0); |
---|
| 122 | +err_disable_smi: |
---|
| 123 | + clk_disable_unprepare(smi->clk_smi); |
---|
97 | 124 | err_disable_apb: |
---|
98 | 125 | clk_disable_unprepare(smi->clk_apb); |
---|
99 | | -err_put_pm: |
---|
100 | | - pm_runtime_put_sync(smi->dev); |
---|
101 | 126 | return ret; |
---|
102 | 127 | } |
---|
103 | 128 | |
---|
104 | | -static void mtk_smi_disable(const struct mtk_smi *smi) |
---|
| 129 | +static void mtk_smi_clk_disable(const struct mtk_smi *smi) |
---|
105 | 130 | { |
---|
| 131 | + clk_disable_unprepare(smi->clk_gals1); |
---|
| 132 | + clk_disable_unprepare(smi->clk_gals0); |
---|
106 | 133 | clk_disable_unprepare(smi->clk_smi); |
---|
107 | 134 | clk_disable_unprepare(smi->clk_apb); |
---|
108 | | - pm_runtime_put_sync(smi->dev); |
---|
109 | 135 | } |
---|
110 | 136 | |
---|
111 | 137 | int mtk_smi_larb_get(struct device *larbdev) |
---|
112 | 138 | { |
---|
113 | | - struct mtk_smi_larb *larb = dev_get_drvdata(larbdev); |
---|
114 | | - const struct mtk_smi_larb_gen *larb_gen = larb->larb_gen; |
---|
115 | | - struct mtk_smi *common = dev_get_drvdata(larb->smi_common_dev); |
---|
116 | | - int ret; |
---|
| 139 | + int ret = pm_runtime_resume_and_get(larbdev); |
---|
117 | 140 | |
---|
118 | | - /* Enable the smi-common's power and clocks */ |
---|
119 | | - ret = mtk_smi_enable(common); |
---|
120 | | - if (ret) |
---|
121 | | - return ret; |
---|
122 | | - |
---|
123 | | - /* Enable the larb's power and clocks */ |
---|
124 | | - ret = mtk_smi_enable(&larb->smi); |
---|
125 | | - if (ret) { |
---|
126 | | - mtk_smi_disable(common); |
---|
127 | | - return ret; |
---|
128 | | - } |
---|
129 | | - |
---|
130 | | - /* Configure the iommu info for this larb */ |
---|
131 | | - larb_gen->config_port(larbdev); |
---|
132 | | - |
---|
133 | | - return 0; |
---|
| 141 | + return (ret < 0) ? ret : 0; |
---|
134 | 142 | } |
---|
135 | 143 | EXPORT_SYMBOL_GPL(mtk_smi_larb_get); |
---|
136 | 144 | |
---|
137 | 145 | void mtk_smi_larb_put(struct device *larbdev) |
---|
138 | 146 | { |
---|
139 | | - struct mtk_smi_larb *larb = dev_get_drvdata(larbdev); |
---|
140 | | - struct mtk_smi *common = dev_get_drvdata(larb->smi_common_dev); |
---|
141 | | - |
---|
142 | | - /* |
---|
143 | | - * Don't de-configure the iommu info for this larb since there may be |
---|
144 | | - * several modules in this larb. |
---|
145 | | - * The iommu info will be reset after power off. |
---|
146 | | - */ |
---|
147 | | - |
---|
148 | | - mtk_smi_disable(&larb->smi); |
---|
149 | | - mtk_smi_disable(common); |
---|
| 147 | + pm_runtime_put_sync(larbdev); |
---|
150 | 148 | } |
---|
151 | 149 | EXPORT_SYMBOL_GPL(mtk_smi_larb_put); |
---|
152 | 150 | |
---|
.. | .. |
---|
154 | 152 | mtk_smi_larb_bind(struct device *dev, struct device *master, void *data) |
---|
155 | 153 | { |
---|
156 | 154 | struct mtk_smi_larb *larb = dev_get_drvdata(dev); |
---|
157 | | - struct mtk_smi_iommu *smi_iommu = data; |
---|
| 155 | + struct mtk_smi_larb_iommu *larb_mmu = data; |
---|
158 | 156 | unsigned int i; |
---|
159 | 157 | |
---|
160 | | - if (larb->larb_gen->need_larbid) { |
---|
161 | | - larb->mmu = &smi_iommu->larb_imu[larb->larbid].mmu; |
---|
162 | | - return 0; |
---|
163 | | - } |
---|
164 | | - |
---|
165 | | - /* |
---|
166 | | - * If there is no larbid property, Loop to find the corresponding |
---|
167 | | - * iommu information. |
---|
168 | | - */ |
---|
169 | | - for (i = 0; i < smi_iommu->larb_nr; i++) { |
---|
170 | | - if (dev == smi_iommu->larb_imu[i].dev) { |
---|
171 | | - /* The 'mmu' may be updated in iommu-attach/detach. */ |
---|
172 | | - larb->mmu = &smi_iommu->larb_imu[i].mmu; |
---|
| 158 | + for (i = 0; i < MTK_LARB_NR_MAX; i++) { |
---|
| 159 | + if (dev == larb_mmu[i].dev) { |
---|
| 160 | + larb->larbid = i; |
---|
| 161 | + larb->mmu = &larb_mmu[i].mmu; |
---|
| 162 | + larb->bank = larb_mmu[i].bank; |
---|
173 | 163 | return 0; |
---|
174 | 164 | } |
---|
175 | 165 | } |
---|
176 | 166 | return -ENODEV; |
---|
177 | 167 | } |
---|
178 | 168 | |
---|
179 | | -static void mtk_smi_larb_config_port_mt2712(struct device *dev) |
---|
| 169 | +static void mtk_smi_larb_config_port_gen2_general(struct device *dev) |
---|
180 | 170 | { |
---|
181 | 171 | struct mtk_smi_larb *larb = dev_get_drvdata(dev); |
---|
182 | 172 | u32 reg; |
---|
183 | 173 | int i; |
---|
184 | 174 | |
---|
185 | | - /* |
---|
186 | | - * larb 8/9 is the bdpsys larb, the iommu_en is enabled defaultly. |
---|
187 | | - * Don't need to set it again. |
---|
188 | | - */ |
---|
189 | | - if (larb->larbid == 8 || larb->larbid == 9) |
---|
| 175 | + if (BIT(larb->larbid) & larb->larb_gen->larb_direct_to_common_mask) |
---|
190 | 176 | return; |
---|
191 | 177 | |
---|
192 | 178 | for_each_set_bit(i, (unsigned long *)larb->mmu, 32) { |
---|
193 | 179 | reg = readl_relaxed(larb->base + SMI_LARB_NONSEC_CON(i)); |
---|
194 | 180 | reg |= F_MMU_EN; |
---|
| 181 | + reg |= BANK_SEL(larb->bank[i]); |
---|
195 | 182 | writel(reg, larb->base + SMI_LARB_NONSEC_CON(i)); |
---|
196 | 183 | } |
---|
197 | 184 | } |
---|
.. | .. |
---|
201 | 188 | struct mtk_smi_larb *larb = dev_get_drvdata(dev); |
---|
202 | 189 | |
---|
203 | 190 | writel(*larb->mmu, larb->base + SMI_LARB_MMU_EN); |
---|
| 191 | +} |
---|
| 192 | + |
---|
| 193 | +static void mtk_smi_larb_config_port_mt8167(struct device *dev) |
---|
| 194 | +{ |
---|
| 195 | + struct mtk_smi_larb *larb = dev_get_drvdata(dev); |
---|
| 196 | + |
---|
| 197 | + writel(*larb->mmu, larb->base + MT8167_SMI_LARB_MMU_EN); |
---|
204 | 198 | } |
---|
205 | 199 | |
---|
206 | 200 | static void mtk_smi_larb_config_port_gen1(struct device *dev) |
---|
.. | .. |
---|
250 | 244 | .config_port = mtk_smi_larb_config_port_mt8173, |
---|
251 | 245 | }; |
---|
252 | 246 | |
---|
| 247 | +static const struct mtk_smi_larb_gen mtk_smi_larb_mt8167 = { |
---|
| 248 | + /* mt8167 do not need the port in larb */ |
---|
| 249 | + .config_port = mtk_smi_larb_config_port_mt8167, |
---|
| 250 | +}; |
---|
| 251 | + |
---|
253 | 252 | static const struct mtk_smi_larb_gen mtk_smi_larb_mt2701 = { |
---|
254 | | - .need_larbid = true, |
---|
255 | 253 | .port_in_larb = { |
---|
256 | 254 | LARB0_PORT_OFFSET, LARB1_PORT_OFFSET, |
---|
257 | 255 | LARB2_PORT_OFFSET, LARB3_PORT_OFFSET |
---|
.. | .. |
---|
260 | 258 | }; |
---|
261 | 259 | |
---|
262 | 260 | static const struct mtk_smi_larb_gen mtk_smi_larb_mt2712 = { |
---|
263 | | - .need_larbid = true, |
---|
264 | | - .config_port = mtk_smi_larb_config_port_mt2712, |
---|
| 261 | + .config_port = mtk_smi_larb_config_port_gen2_general, |
---|
| 262 | + .larb_direct_to_common_mask = BIT(8) | BIT(9), /* bdpsys */ |
---|
| 263 | +}; |
---|
| 264 | + |
---|
| 265 | +static const struct mtk_smi_larb_gen mtk_smi_larb_mt6779 = { |
---|
| 266 | + .config_port = mtk_smi_larb_config_port_gen2_general, |
---|
| 267 | + .larb_direct_to_common_mask = |
---|
| 268 | + BIT(4) | BIT(6) | BIT(11) | BIT(12) | BIT(13), |
---|
| 269 | + /* DUMMY | IPU0 | IPU1 | CCU | MDLA */ |
---|
| 270 | +}; |
---|
| 271 | + |
---|
| 272 | +static const struct mtk_smi_larb_gen mtk_smi_larb_mt8183 = { |
---|
| 273 | + .has_gals = true, |
---|
| 274 | + .config_port = mtk_smi_larb_config_port_gen2_general, |
---|
| 275 | + .larb_direct_to_common_mask = BIT(2) | BIT(3) | BIT(7), |
---|
| 276 | + /* IPU0 | IPU1 | CCU */ |
---|
| 277 | +}; |
---|
| 278 | + |
---|
| 279 | +static const struct mtk_smi_larb_gen mtk_smi_larb_mt8192 = { |
---|
| 280 | + .config_port = mtk_smi_larb_config_port_gen2_general, |
---|
265 | 281 | }; |
---|
266 | 282 | |
---|
267 | 283 | static const struct of_device_id mtk_smi_larb_of_ids[] = { |
---|
| 284 | + { |
---|
| 285 | + .compatible = "mediatek,mt8167-smi-larb", |
---|
| 286 | + .data = &mtk_smi_larb_mt8167 |
---|
| 287 | + }, |
---|
268 | 288 | { |
---|
269 | 289 | .compatible = "mediatek,mt8173-smi-larb", |
---|
270 | 290 | .data = &mtk_smi_larb_mt8173 |
---|
.. | .. |
---|
277 | 297 | .compatible = "mediatek,mt2712-smi-larb", |
---|
278 | 298 | .data = &mtk_smi_larb_mt2712 |
---|
279 | 299 | }, |
---|
| 300 | + { |
---|
| 301 | + .compatible = "mediatek,mt6779-smi-larb", |
---|
| 302 | + .data = &mtk_smi_larb_mt6779 |
---|
| 303 | + }, |
---|
| 304 | + { |
---|
| 305 | + .compatible = "mediatek,mt8183-smi-larb", |
---|
| 306 | + .data = &mtk_smi_larb_mt8183 |
---|
| 307 | + }, |
---|
| 308 | + { |
---|
| 309 | + .compatible = "mediatek,mt8192-smi-larb", |
---|
| 310 | + .data = &mtk_smi_larb_mt8192 |
---|
| 311 | + }, |
---|
280 | 312 | {} |
---|
281 | 313 | }; |
---|
282 | 314 | |
---|
.. | .. |
---|
287 | 319 | struct device *dev = &pdev->dev; |
---|
288 | 320 | struct device_node *smi_node; |
---|
289 | 321 | struct platform_device *smi_pdev; |
---|
290 | | - int err; |
---|
291 | 322 | |
---|
292 | 323 | larb = devm_kzalloc(dev, sizeof(*larb), GFP_KERNEL); |
---|
293 | 324 | if (!larb) |
---|
.. | .. |
---|
306 | 337 | larb->smi.clk_smi = devm_clk_get(dev, "smi"); |
---|
307 | 338 | if (IS_ERR(larb->smi.clk_smi)) |
---|
308 | 339 | return PTR_ERR(larb->smi.clk_smi); |
---|
309 | | - larb->smi.dev = dev; |
---|
310 | 340 | |
---|
311 | | - if (larb->larb_gen->need_larbid) { |
---|
312 | | - err = of_property_read_u32(dev->of_node, "mediatek,larb-id", |
---|
313 | | - &larb->larbid); |
---|
314 | | - if (err) { |
---|
315 | | - dev_err(dev, "missing larbid property\n"); |
---|
316 | | - return err; |
---|
317 | | - } |
---|
| 341 | + if (larb->larb_gen->has_gals) { |
---|
| 342 | + /* The larbs may still haven't gals even if the SoC support.*/ |
---|
| 343 | + larb->smi.clk_gals0 = devm_clk_get(dev, "gals"); |
---|
| 344 | + if (PTR_ERR(larb->smi.clk_gals0) == -ENOENT) |
---|
| 345 | + larb->smi.clk_gals0 = NULL; |
---|
| 346 | + else if (IS_ERR(larb->smi.clk_gals0)) |
---|
| 347 | + return PTR_ERR(larb->smi.clk_gals0); |
---|
318 | 348 | } |
---|
| 349 | + larb->smi.dev = dev; |
---|
319 | 350 | |
---|
320 | 351 | smi_node = of_parse_phandle(dev->of_node, "mediatek,smi", 0); |
---|
321 | 352 | if (!smi_node) |
---|
.. | .. |
---|
344 | 375 | return 0; |
---|
345 | 376 | } |
---|
346 | 377 | |
---|
| 378 | +static int __maybe_unused mtk_smi_larb_resume(struct device *dev) |
---|
| 379 | +{ |
---|
| 380 | + struct mtk_smi_larb *larb = dev_get_drvdata(dev); |
---|
| 381 | + const struct mtk_smi_larb_gen *larb_gen = larb->larb_gen; |
---|
| 382 | + int ret; |
---|
| 383 | + |
---|
| 384 | + /* Power on smi-common. */ |
---|
| 385 | + ret = pm_runtime_resume_and_get(larb->smi_common_dev); |
---|
| 386 | + if (ret < 0) { |
---|
| 387 | + dev_err(dev, "Failed to pm get for smi-common(%d).\n", ret); |
---|
| 388 | + return ret; |
---|
| 389 | + } |
---|
| 390 | + |
---|
| 391 | + ret = mtk_smi_clk_enable(&larb->smi); |
---|
| 392 | + if (ret < 0) { |
---|
| 393 | + dev_err(dev, "Failed to enable clock(%d).\n", ret); |
---|
| 394 | + pm_runtime_put_sync(larb->smi_common_dev); |
---|
| 395 | + return ret; |
---|
| 396 | + } |
---|
| 397 | + |
---|
| 398 | + /* Configure the basic setting for this larb */ |
---|
| 399 | + larb_gen->config_port(dev); |
---|
| 400 | + |
---|
| 401 | + return 0; |
---|
| 402 | +} |
---|
| 403 | + |
---|
| 404 | +static int __maybe_unused mtk_smi_larb_suspend(struct device *dev) |
---|
| 405 | +{ |
---|
| 406 | + struct mtk_smi_larb *larb = dev_get_drvdata(dev); |
---|
| 407 | + |
---|
| 408 | + mtk_smi_clk_disable(&larb->smi); |
---|
| 409 | + pm_runtime_put_sync(larb->smi_common_dev); |
---|
| 410 | + return 0; |
---|
| 411 | +} |
---|
| 412 | + |
---|
| 413 | +static const struct dev_pm_ops smi_larb_pm_ops = { |
---|
| 414 | + SET_RUNTIME_PM_OPS(mtk_smi_larb_suspend, mtk_smi_larb_resume, NULL) |
---|
| 415 | + SET_LATE_SYSTEM_SLEEP_PM_OPS(pm_runtime_force_suspend, |
---|
| 416 | + pm_runtime_force_resume) |
---|
| 417 | +}; |
---|
| 418 | + |
---|
347 | 419 | static struct platform_driver mtk_smi_larb_driver = { |
---|
348 | 420 | .probe = mtk_smi_larb_probe, |
---|
349 | 421 | .remove = mtk_smi_larb_remove, |
---|
350 | 422 | .driver = { |
---|
351 | 423 | .name = "mtk-smi-larb", |
---|
352 | 424 | .of_match_table = mtk_smi_larb_of_ids, |
---|
| 425 | + .pm = &smi_larb_pm_ops, |
---|
353 | 426 | } |
---|
| 427 | +}; |
---|
| 428 | + |
---|
| 429 | +static const struct mtk_smi_common_plat mtk_smi_common_gen1 = { |
---|
| 430 | + .gen = MTK_SMI_GEN1, |
---|
| 431 | +}; |
---|
| 432 | + |
---|
| 433 | +static const struct mtk_smi_common_plat mtk_smi_common_gen2 = { |
---|
| 434 | + .gen = MTK_SMI_GEN2, |
---|
| 435 | +}; |
---|
| 436 | + |
---|
| 437 | +static const struct mtk_smi_common_plat mtk_smi_common_mt6779 = { |
---|
| 438 | + .gen = MTK_SMI_GEN2, |
---|
| 439 | + .has_gals = true, |
---|
| 440 | + .bus_sel = F_MMU1_LARB(1) | F_MMU1_LARB(2) | F_MMU1_LARB(4) | |
---|
| 441 | + F_MMU1_LARB(5) | F_MMU1_LARB(6) | F_MMU1_LARB(7), |
---|
| 442 | +}; |
---|
| 443 | + |
---|
| 444 | +static const struct mtk_smi_common_plat mtk_smi_common_mt8183 = { |
---|
| 445 | + .gen = MTK_SMI_GEN2, |
---|
| 446 | + .has_gals = true, |
---|
| 447 | + .bus_sel = F_MMU1_LARB(1) | F_MMU1_LARB(2) | F_MMU1_LARB(5) | |
---|
| 448 | + F_MMU1_LARB(7), |
---|
| 449 | +}; |
---|
| 450 | + |
---|
| 451 | +static const struct mtk_smi_common_plat mtk_smi_common_mt8192 = { |
---|
| 452 | + .gen = MTK_SMI_GEN2, |
---|
| 453 | + .has_gals = true, |
---|
| 454 | + .bus_sel = F_MMU1_LARB(1) | F_MMU1_LARB(2) | F_MMU1_LARB(5) | |
---|
| 455 | + F_MMU1_LARB(6), |
---|
354 | 456 | }; |
---|
355 | 457 | |
---|
356 | 458 | static const struct of_device_id mtk_smi_common_of_ids[] = { |
---|
357 | 459 | { |
---|
358 | 460 | .compatible = "mediatek,mt8173-smi-common", |
---|
359 | | - .data = (void *)MTK_SMI_GEN2 |
---|
| 461 | + .data = &mtk_smi_common_gen2, |
---|
| 462 | + }, |
---|
| 463 | + { |
---|
| 464 | + .compatible = "mediatek,mt8167-smi-common", |
---|
| 465 | + .data = &mtk_smi_common_gen2, |
---|
360 | 466 | }, |
---|
361 | 467 | { |
---|
362 | 468 | .compatible = "mediatek,mt2701-smi-common", |
---|
363 | | - .data = (void *)MTK_SMI_GEN1 |
---|
| 469 | + .data = &mtk_smi_common_gen1, |
---|
364 | 470 | }, |
---|
365 | 471 | { |
---|
366 | 472 | .compatible = "mediatek,mt2712-smi-common", |
---|
367 | | - .data = (void *)MTK_SMI_GEN2 |
---|
| 473 | + .data = &mtk_smi_common_gen2, |
---|
| 474 | + }, |
---|
| 475 | + { |
---|
| 476 | + .compatible = "mediatek,mt6779-smi-common", |
---|
| 477 | + .data = &mtk_smi_common_mt6779, |
---|
| 478 | + }, |
---|
| 479 | + { |
---|
| 480 | + .compatible = "mediatek,mt8183-smi-common", |
---|
| 481 | + .data = &mtk_smi_common_mt8183, |
---|
| 482 | + }, |
---|
| 483 | + { |
---|
| 484 | + .compatible = "mediatek,mt8192-smi-common", |
---|
| 485 | + .data = &mtk_smi_common_mt8192, |
---|
368 | 486 | }, |
---|
369 | 487 | {} |
---|
370 | 488 | }; |
---|
.. | .. |
---|
374 | 492 | struct device *dev = &pdev->dev; |
---|
375 | 493 | struct mtk_smi *common; |
---|
376 | 494 | struct resource *res; |
---|
377 | | - enum mtk_smi_gen smi_gen; |
---|
378 | 495 | int ret; |
---|
379 | 496 | |
---|
380 | 497 | common = devm_kzalloc(dev, sizeof(*common), GFP_KERNEL); |
---|
381 | 498 | if (!common) |
---|
382 | 499 | return -ENOMEM; |
---|
383 | 500 | common->dev = dev; |
---|
| 501 | + common->plat = of_device_get_match_data(dev); |
---|
384 | 502 | |
---|
385 | 503 | common->clk_apb = devm_clk_get(dev, "apb"); |
---|
386 | 504 | if (IS_ERR(common->clk_apb)) |
---|
.. | .. |
---|
390 | 508 | if (IS_ERR(common->clk_smi)) |
---|
391 | 509 | return PTR_ERR(common->clk_smi); |
---|
392 | 510 | |
---|
| 511 | + if (common->plat->has_gals) { |
---|
| 512 | + common->clk_gals0 = devm_clk_get(dev, "gals0"); |
---|
| 513 | + if (IS_ERR(common->clk_gals0)) |
---|
| 514 | + return PTR_ERR(common->clk_gals0); |
---|
| 515 | + |
---|
| 516 | + common->clk_gals1 = devm_clk_get(dev, "gals1"); |
---|
| 517 | + if (IS_ERR(common->clk_gals1)) |
---|
| 518 | + return PTR_ERR(common->clk_gals1); |
---|
| 519 | + } |
---|
| 520 | + |
---|
393 | 521 | /* |
---|
394 | 522 | * for mtk smi gen 1, we need to get the ao(always on) base to config |
---|
395 | 523 | * m4u port, and we need to enable the aync clock for transform the smi |
---|
396 | 524 | * clock into emi clock domain, but for mtk smi gen2, there's no smi ao |
---|
397 | 525 | * base. |
---|
398 | 526 | */ |
---|
399 | | - smi_gen = (enum mtk_smi_gen)of_device_get_match_data(dev); |
---|
400 | | - if (smi_gen == MTK_SMI_GEN1) { |
---|
| 527 | + if (common->plat->gen == MTK_SMI_GEN1) { |
---|
401 | 528 | res = platform_get_resource(pdev, IORESOURCE_MEM, 0); |
---|
402 | 529 | common->smi_ao_base = devm_ioremap_resource(dev, res); |
---|
403 | 530 | if (IS_ERR(common->smi_ao_base)) |
---|
.. | .. |
---|
410 | 537 | ret = clk_prepare_enable(common->clk_async); |
---|
411 | 538 | if (ret) |
---|
412 | 539 | return ret; |
---|
| 540 | + } else { |
---|
| 541 | + res = platform_get_resource(pdev, IORESOURCE_MEM, 0); |
---|
| 542 | + common->base = devm_ioremap_resource(dev, res); |
---|
| 543 | + if (IS_ERR(common->base)) |
---|
| 544 | + return PTR_ERR(common->base); |
---|
413 | 545 | } |
---|
414 | 546 | pm_runtime_enable(dev); |
---|
415 | 547 | platform_set_drvdata(pdev, common); |
---|
.. | .. |
---|
422 | 554 | return 0; |
---|
423 | 555 | } |
---|
424 | 556 | |
---|
| 557 | +static int __maybe_unused mtk_smi_common_resume(struct device *dev) |
---|
| 558 | +{ |
---|
| 559 | + struct mtk_smi *common = dev_get_drvdata(dev); |
---|
| 560 | + u32 bus_sel = common->plat->bus_sel; |
---|
| 561 | + int ret; |
---|
| 562 | + |
---|
| 563 | + ret = mtk_smi_clk_enable(common); |
---|
| 564 | + if (ret) { |
---|
| 565 | + dev_err(common->dev, "Failed to enable clock(%d).\n", ret); |
---|
| 566 | + return ret; |
---|
| 567 | + } |
---|
| 568 | + |
---|
| 569 | + if (common->plat->gen == MTK_SMI_GEN2 && bus_sel) |
---|
| 570 | + writel(bus_sel, common->base + SMI_BUS_SEL); |
---|
| 571 | + return 0; |
---|
| 572 | +} |
---|
| 573 | + |
---|
| 574 | +static int __maybe_unused mtk_smi_common_suspend(struct device *dev) |
---|
| 575 | +{ |
---|
| 576 | + struct mtk_smi *common = dev_get_drvdata(dev); |
---|
| 577 | + |
---|
| 578 | + mtk_smi_clk_disable(common); |
---|
| 579 | + return 0; |
---|
| 580 | +} |
---|
| 581 | + |
---|
| 582 | +static const struct dev_pm_ops smi_common_pm_ops = { |
---|
| 583 | + SET_RUNTIME_PM_OPS(mtk_smi_common_suspend, mtk_smi_common_resume, NULL) |
---|
| 584 | + SET_LATE_SYSTEM_SLEEP_PM_OPS(pm_runtime_force_suspend, |
---|
| 585 | + pm_runtime_force_resume) |
---|
| 586 | +}; |
---|
| 587 | + |
---|
425 | 588 | static struct platform_driver mtk_smi_common_driver = { |
---|
426 | 589 | .probe = mtk_smi_common_probe, |
---|
427 | 590 | .remove = mtk_smi_common_remove, |
---|
428 | 591 | .driver = { |
---|
429 | 592 | .name = "mtk-smi-common", |
---|
430 | 593 | .of_match_table = mtk_smi_common_of_ids, |
---|
| 594 | + .pm = &smi_common_pm_ops, |
---|
431 | 595 | } |
---|
| 596 | +}; |
---|
| 597 | + |
---|
| 598 | +static struct platform_driver * const smidrivers[] = { |
---|
| 599 | + &mtk_smi_common_driver, |
---|
| 600 | + &mtk_smi_larb_driver, |
---|
432 | 601 | }; |
---|
433 | 602 | |
---|
434 | 603 | static int __init mtk_smi_init(void) |
---|
435 | 604 | { |
---|
436 | | - int ret; |
---|
437 | | - |
---|
438 | | - ret = platform_driver_register(&mtk_smi_common_driver); |
---|
439 | | - if (ret != 0) { |
---|
440 | | - pr_err("Failed to register SMI driver\n"); |
---|
441 | | - return ret; |
---|
442 | | - } |
---|
443 | | - |
---|
444 | | - ret = platform_driver_register(&mtk_smi_larb_driver); |
---|
445 | | - if (ret != 0) { |
---|
446 | | - pr_err("Failed to register SMI-LARB driver\n"); |
---|
447 | | - goto err_unreg_smi; |
---|
448 | | - } |
---|
449 | | - return ret; |
---|
450 | | - |
---|
451 | | -err_unreg_smi: |
---|
452 | | - platform_driver_unregister(&mtk_smi_common_driver); |
---|
453 | | - return ret; |
---|
| 605 | + return platform_register_drivers(smidrivers, ARRAY_SIZE(smidrivers)); |
---|
454 | 606 | } |
---|
455 | | - |
---|
456 | 607 | module_init(mtk_smi_init); |
---|
| 608 | + |
---|
| 609 | +static void __exit mtk_smi_exit(void) |
---|
| 610 | +{ |
---|
| 611 | + platform_unregister_drivers(smidrivers, ARRAY_SIZE(smidrivers)); |
---|
| 612 | +} |
---|
| 613 | +module_exit(mtk_smi_exit); |
---|
| 614 | + |
---|
| 615 | +MODULE_DESCRIPTION("MediaTek SMI driver"); |
---|
| 616 | +MODULE_LICENSE("GPL v2"); |
---|