diff options
Diffstat (limited to 'drivers/soc/tegra/flowctrl.c')
-rw-r--r-- | drivers/soc/tegra/flowctrl.c | 224 |
1 files changed, 224 insertions, 0 deletions
diff --git a/drivers/soc/tegra/flowctrl.c b/drivers/soc/tegra/flowctrl.c new file mode 100644 index 000000000000..0e345c05fc65 --- /dev/null +++ b/drivers/soc/tegra/flowctrl.c | |||
@@ -0,0 +1,224 @@ | |||
1 | /* | ||
2 | * drivers/soc/tegra/flowctrl.c | ||
3 | * | ||
4 | * Functions and macros to control the flowcontroller | ||
5 | * | ||
6 | * Copyright (c) 2010-2012, NVIDIA Corporation. All rights reserved. | ||
7 | * | ||
8 | * This program is free software; you can redistribute it and/or modify it | ||
9 | * under the terms and conditions of the GNU General Public License, | ||
10 | * version 2, as published by the Free Software Foundation. | ||
11 | * | ||
12 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
13 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
14 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
15 | * more details. | ||
16 | * | ||
17 | * You should have received a copy of the GNU General Public License | ||
18 | * along with this program. If not, see <http://www.gnu.org/licenses/>. | ||
19 | */ | ||
20 | |||
21 | #include <linux/cpumask.h> | ||
22 | #include <linux/init.h> | ||
23 | #include <linux/io.h> | ||
24 | #include <linux/kernel.h> | ||
25 | #include <linux/of.h> | ||
26 | #include <linux/of_address.h> | ||
27 | #include <linux/platform_device.h> | ||
28 | |||
29 | #include <soc/tegra/common.h> | ||
30 | #include <soc/tegra/flowctrl.h> | ||
31 | #include <soc/tegra/fuse.h> | ||
32 | |||
33 | static u8 flowctrl_offset_halt_cpu[] = { | ||
34 | FLOW_CTRL_HALT_CPU0_EVENTS, | ||
35 | FLOW_CTRL_HALT_CPU1_EVENTS, | ||
36 | FLOW_CTRL_HALT_CPU1_EVENTS + 8, | ||
37 | FLOW_CTRL_HALT_CPU1_EVENTS + 16, | ||
38 | }; | ||
39 | |||
40 | static u8 flowctrl_offset_cpu_csr[] = { | ||
41 | FLOW_CTRL_CPU0_CSR, | ||
42 | FLOW_CTRL_CPU1_CSR, | ||
43 | FLOW_CTRL_CPU1_CSR + 8, | ||
44 | FLOW_CTRL_CPU1_CSR + 16, | ||
45 | }; | ||
46 | |||
47 | static void __iomem *tegra_flowctrl_base; | ||
48 | |||
49 | static void flowctrl_update(u8 offset, u32 value) | ||
50 | { | ||
51 | if (WARN_ONCE(IS_ERR_OR_NULL(tegra_flowctrl_base), | ||
52 | "Tegra flowctrl not initialised!\n")) | ||
53 | return; | ||
54 | |||
55 | writel(value, tegra_flowctrl_base + offset); | ||
56 | |||
57 | /* ensure the update has reached the flow controller */ | ||
58 | wmb(); | ||
59 | readl_relaxed(tegra_flowctrl_base + offset); | ||
60 | } | ||
61 | |||
62 | u32 flowctrl_read_cpu_csr(unsigned int cpuid) | ||
63 | { | ||
64 | u8 offset = flowctrl_offset_cpu_csr[cpuid]; | ||
65 | |||
66 | if (WARN_ONCE(IS_ERR_OR_NULL(tegra_flowctrl_base), | ||
67 | "Tegra flowctrl not initialised!\n")) | ||
68 | return 0; | ||
69 | |||
70 | return readl(tegra_flowctrl_base + offset); | ||
71 | } | ||
72 | |||
73 | void flowctrl_write_cpu_csr(unsigned int cpuid, u32 value) | ||
74 | { | ||
75 | return flowctrl_update(flowctrl_offset_cpu_csr[cpuid], value); | ||
76 | } | ||
77 | |||
78 | void flowctrl_write_cpu_halt(unsigned int cpuid, u32 value) | ||
79 | { | ||
80 | return flowctrl_update(flowctrl_offset_halt_cpu[cpuid], value); | ||
81 | } | ||
82 | |||
83 | void flowctrl_cpu_suspend_enter(unsigned int cpuid) | ||
84 | { | ||
85 | unsigned int reg; | ||
86 | int i; | ||
87 | |||
88 | reg = flowctrl_read_cpu_csr(cpuid); | ||
89 | switch (tegra_get_chip_id()) { | ||
90 | case TEGRA20: | ||
91 | /* clear wfe bitmap */ | ||
92 | reg &= ~TEGRA20_FLOW_CTRL_CSR_WFE_BITMAP; | ||
93 | /* clear wfi bitmap */ | ||
94 | reg &= ~TEGRA20_FLOW_CTRL_CSR_WFI_BITMAP; | ||
95 | /* pwr gating on wfe */ | ||
96 | reg |= TEGRA20_FLOW_CTRL_CSR_WFE_CPU0 << cpuid; | ||
97 | break; | ||
98 | case TEGRA30: | ||
99 | case TEGRA114: | ||
100 | case TEGRA124: | ||
101 | /* clear wfe bitmap */ | ||
102 | reg &= ~TEGRA30_FLOW_CTRL_CSR_WFE_BITMAP; | ||
103 | /* clear wfi bitmap */ | ||
104 | reg &= ~TEGRA30_FLOW_CTRL_CSR_WFI_BITMAP; | ||
105 | /* pwr gating on wfi */ | ||
106 | reg |= TEGRA30_FLOW_CTRL_CSR_WFI_CPU0 << cpuid; | ||
107 | break; | ||
108 | } | ||
109 | reg |= FLOW_CTRL_CSR_INTR_FLAG; /* clear intr flag */ | ||
110 | reg |= FLOW_CTRL_CSR_EVENT_FLAG; /* clear event flag */ | ||
111 | reg |= FLOW_CTRL_CSR_ENABLE; /* pwr gating */ | ||
112 | flowctrl_write_cpu_csr(cpuid, reg); | ||
113 | |||
114 | for (i = 0; i < num_possible_cpus(); i++) { | ||
115 | if (i == cpuid) | ||
116 | continue; | ||
117 | reg = flowctrl_read_cpu_csr(i); | ||
118 | reg |= FLOW_CTRL_CSR_EVENT_FLAG; | ||
119 | reg |= FLOW_CTRL_CSR_INTR_FLAG; | ||
120 | flowctrl_write_cpu_csr(i, reg); | ||
121 | } | ||
122 | } | ||
123 | |||
124 | void flowctrl_cpu_suspend_exit(unsigned int cpuid) | ||
125 | { | ||
126 | unsigned int reg; | ||
127 | |||
128 | /* Disable powergating via flow controller for CPU0 */ | ||
129 | reg = flowctrl_read_cpu_csr(cpuid); | ||
130 | switch (tegra_get_chip_id()) { | ||
131 | case TEGRA20: | ||
132 | /* clear wfe bitmap */ | ||
133 | reg &= ~TEGRA20_FLOW_CTRL_CSR_WFE_BITMAP; | ||
134 | /* clear wfi bitmap */ | ||
135 | reg &= ~TEGRA20_FLOW_CTRL_CSR_WFI_BITMAP; | ||
136 | break; | ||
137 | case TEGRA30: | ||
138 | case TEGRA114: | ||
139 | case TEGRA124: | ||
140 | /* clear wfe bitmap */ | ||
141 | reg &= ~TEGRA30_FLOW_CTRL_CSR_WFE_BITMAP; | ||
142 | /* clear wfi bitmap */ | ||
143 | reg &= ~TEGRA30_FLOW_CTRL_CSR_WFI_BITMAP; | ||
144 | break; | ||
145 | } | ||
146 | reg &= ~FLOW_CTRL_CSR_ENABLE; /* clear enable */ | ||
147 | reg |= FLOW_CTRL_CSR_INTR_FLAG; /* clear intr */ | ||
148 | reg |= FLOW_CTRL_CSR_EVENT_FLAG; /* clear event */ | ||
149 | flowctrl_write_cpu_csr(cpuid, reg); | ||
150 | } | ||
151 | |||
152 | static int tegra_flowctrl_probe(struct platform_device *pdev) | ||
153 | { | ||
154 | void __iomem *base = tegra_flowctrl_base; | ||
155 | struct resource *res; | ||
156 | |||
157 | res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | ||
158 | tegra_flowctrl_base = devm_ioremap_resource(&pdev->dev, res); | ||
159 | if (IS_ERR(tegra_flowctrl_base)) | ||
160 | return PTR_ERR(base); | ||
161 | |||
162 | iounmap(base); | ||
163 | |||
164 | return 0; | ||
165 | } | ||
166 | |||
167 | static const struct of_device_id tegra_flowctrl_match[] = { | ||
168 | { .compatible = "nvidia,tegra210-flowctrl" }, | ||
169 | { .compatible = "nvidia,tegra124-flowctrl" }, | ||
170 | { .compatible = "nvidia,tegra114-flowctrl" }, | ||
171 | { .compatible = "nvidia,tegra30-flowctrl" }, | ||
172 | { .compatible = "nvidia,tegra20-flowctrl" }, | ||
173 | { } | ||
174 | }; | ||
175 | |||
176 | static struct platform_driver tegra_flowctrl_driver = { | ||
177 | .driver = { | ||
178 | .name = "tegra-flowctrl", | ||
179 | .suppress_bind_attrs = true, | ||
180 | .of_match_table = tegra_flowctrl_match, | ||
181 | }, | ||
182 | .probe = tegra_flowctrl_probe, | ||
183 | }; | ||
184 | builtin_platform_driver(tegra_flowctrl_driver); | ||
185 | |||
186 | static int __init tegra_flowctrl_init(void) | ||
187 | { | ||
188 | struct resource res; | ||
189 | struct device_node *np; | ||
190 | |||
191 | if (!soc_is_tegra()) | ||
192 | return 0; | ||
193 | |||
194 | np = of_find_matching_node(NULL, tegra_flowctrl_match); | ||
195 | if (np) { | ||
196 | if (of_address_to_resource(np, 0, &res) < 0) { | ||
197 | pr_err("failed to get flowctrl register\n"); | ||
198 | return -ENXIO; | ||
199 | } | ||
200 | of_node_put(np); | ||
201 | } else if (IS_ENABLED(CONFIG_ARM)) { | ||
202 | /* | ||
203 | * Hardcoded fallback for 32-bit Tegra | ||
204 | * devices if device tree node is missing. | ||
205 | */ | ||
206 | res.start = 0x60007000; | ||
207 | res.end = 0x60007fff; | ||
208 | res.flags = IORESOURCE_MEM; | ||
209 | } else { | ||
210 | /* | ||
211 | * At this point we're running on a Tegra, | ||
212 | * that doesn't support the flow controller | ||
213 | * (eg. Tegra186), so just return. | ||
214 | */ | ||
215 | return 0; | ||
216 | } | ||
217 | |||
218 | tegra_flowctrl_base = ioremap_nocache(res.start, resource_size(&res)); | ||
219 | if (!tegra_flowctrl_base) | ||
220 | return -ENXIO; | ||
221 | |||
222 | return 0; | ||
223 | } | ||
224 | early_initcall(tegra_flowctrl_init); | ||