blob: 656d9ca057e5c1641d5b64640f7e8452a78a3770 [file] [log] [blame]
chenhui zhaod17799f2015-11-20 17:13:59 +08001/*
2 * RCPM(Run Control/Power Management) support
3 *
4 * Copyright 2012-2015 Freescale Semiconductor Inc.
5 *
6 * Author: Chenhui Zhao <chenhui.zhao@freescale.com>
7 *
8 * This program is free software; you can redistribute it and/or modify it
9 * under the terms of the GNU General Public License as published by the
10 * Free Software Foundation; either version 2 of the License, or (at your
11 * option) any later version.
12 */
13
14#define pr_fmt(fmt) "%s: " fmt, __func__
15
16#include <linux/types.h>
17#include <linux/errno.h>
18#include <linux/of_address.h>
19#include <linux/export.h>
20
21#include <asm/io.h>
22#include <linux/fsl/guts.h>
23#include <asm/cputhreads.h>
24#include <asm/fsl_pm.h>
25
26static struct ccsr_rcpm_v1 __iomem *rcpm_v1_regs;
27static struct ccsr_rcpm_v2 __iomem *rcpm_v2_regs;
28static unsigned int fsl_supported_pm_modes;
29
30static void rcpm_v1_irq_mask(int cpu)
31{
32 int hw_cpu = get_hard_smp_processor_id(cpu);
33 unsigned int mask = 1 << hw_cpu;
34
35 setbits32(&rcpm_v1_regs->cpmimr, mask);
36 setbits32(&rcpm_v1_regs->cpmcimr, mask);
37 setbits32(&rcpm_v1_regs->cpmmcmr, mask);
38 setbits32(&rcpm_v1_regs->cpmnmimr, mask);
39}
40
41static void rcpm_v2_irq_mask(int cpu)
42{
43 int hw_cpu = get_hard_smp_processor_id(cpu);
44 unsigned int mask = 1 << hw_cpu;
45
46 setbits32(&rcpm_v2_regs->tpmimr0, mask);
47 setbits32(&rcpm_v2_regs->tpmcimr0, mask);
48 setbits32(&rcpm_v2_regs->tpmmcmr0, mask);
49 setbits32(&rcpm_v2_regs->tpmnmimr0, mask);
50}
51
52static void rcpm_v1_irq_unmask(int cpu)
53{
54 int hw_cpu = get_hard_smp_processor_id(cpu);
55 unsigned int mask = 1 << hw_cpu;
56
57 clrbits32(&rcpm_v1_regs->cpmimr, mask);
58 clrbits32(&rcpm_v1_regs->cpmcimr, mask);
59 clrbits32(&rcpm_v1_regs->cpmmcmr, mask);
60 clrbits32(&rcpm_v1_regs->cpmnmimr, mask);
61}
62
63static void rcpm_v2_irq_unmask(int cpu)
64{
65 int hw_cpu = get_hard_smp_processor_id(cpu);
66 unsigned int mask = 1 << hw_cpu;
67
68 clrbits32(&rcpm_v2_regs->tpmimr0, mask);
69 clrbits32(&rcpm_v2_regs->tpmcimr0, mask);
70 clrbits32(&rcpm_v2_regs->tpmmcmr0, mask);
71 clrbits32(&rcpm_v2_regs->tpmnmimr0, mask);
72}
73
74static void rcpm_v1_set_ip_power(bool enable, u32 mask)
75{
76 if (enable)
77 setbits32(&rcpm_v1_regs->ippdexpcr, mask);
78 else
79 clrbits32(&rcpm_v1_regs->ippdexpcr, mask);
80}
81
82static void rcpm_v2_set_ip_power(bool enable, u32 mask)
83{
84 if (enable)
85 setbits32(&rcpm_v2_regs->ippdexpcr[0], mask);
86 else
87 clrbits32(&rcpm_v2_regs->ippdexpcr[0], mask);
88}
89
90static void rcpm_v1_cpu_enter_state(int cpu, int state)
91{
92 int hw_cpu = get_hard_smp_processor_id(cpu);
93 unsigned int mask = 1 << hw_cpu;
94
95 switch (state) {
96 case E500_PM_PH10:
97 setbits32(&rcpm_v1_regs->cdozcr, mask);
98 break;
99 case E500_PM_PH15:
100 setbits32(&rcpm_v1_regs->cnapcr, mask);
101 break;
102 default:
103 pr_warn("Unknown cpu PM state (%d)\n", state);
104 break;
105 }
106}
107
108static void rcpm_v2_cpu_enter_state(int cpu, int state)
109{
110 int hw_cpu = get_hard_smp_processor_id(cpu);
111 u32 mask = 1 << cpu_core_index_of_thread(cpu);
112
113 switch (state) {
114 case E500_PM_PH10:
115 /* one bit corresponds to one thread for PH10 of 6500 */
116 setbits32(&rcpm_v2_regs->tph10setr0, 1 << hw_cpu);
117 break;
118 case E500_PM_PH15:
119 setbits32(&rcpm_v2_regs->pcph15setr, mask);
120 break;
121 case E500_PM_PH20:
122 setbits32(&rcpm_v2_regs->pcph20setr, mask);
123 break;
124 case E500_PM_PH30:
125 setbits32(&rcpm_v2_regs->pcph30setr, mask);
126 break;
127 default:
128 pr_warn("Unknown cpu PM state (%d)\n", state);
129 }
130}
131
132static void rcpm_v1_cpu_die(int cpu)
133{
134 rcpm_v1_cpu_enter_state(cpu, E500_PM_PH15);
135}
136
137#ifdef CONFIG_PPC64
138static void qoriq_disable_thread(int cpu)
139{
140 int thread = cpu_thread_in_core(cpu);
141
142 book3e_stop_thread(thread);
143}
144#endif
145
146static void rcpm_v2_cpu_die(int cpu)
147{
148#ifdef CONFIG_PPC64
149 int primary;
150
151 if (threads_per_core == 2) {
152 primary = cpu_first_thread_sibling(cpu);
153 if (cpu_is_offline(primary) && cpu_is_offline(primary + 1)) {
154 /* if both threads are offline, put the cpu in PH20 */
155 rcpm_v2_cpu_enter_state(cpu, E500_PM_PH20);
156 } else {
157 /* if only one thread is offline, disable the thread */
158 qoriq_disable_thread(cpu);
159 }
160 }
161#endif
162
163 if (threads_per_core == 1)
164 rcpm_v2_cpu_enter_state(cpu, E500_PM_PH20);
165}
166
167static void rcpm_v1_cpu_exit_state(int cpu, int state)
168{
169 int hw_cpu = get_hard_smp_processor_id(cpu);
170 unsigned int mask = 1 << hw_cpu;
171
172 switch (state) {
173 case E500_PM_PH10:
174 clrbits32(&rcpm_v1_regs->cdozcr, mask);
175 break;
176 case E500_PM_PH15:
177 clrbits32(&rcpm_v1_regs->cnapcr, mask);
178 break;
179 default:
180 pr_warn("Unknown cpu PM state (%d)\n", state);
181 break;
182 }
183}
184
185static void rcpm_v1_cpu_up_prepare(int cpu)
186{
187 rcpm_v1_cpu_exit_state(cpu, E500_PM_PH15);
188 rcpm_v1_irq_unmask(cpu);
189}
190
191static void rcpm_v2_cpu_exit_state(int cpu, int state)
192{
193 int hw_cpu = get_hard_smp_processor_id(cpu);
194 u32 mask = 1 << cpu_core_index_of_thread(cpu);
195
196 switch (state) {
197 case E500_PM_PH10:
198 setbits32(&rcpm_v2_regs->tph10clrr0, 1 << hw_cpu);
199 break;
200 case E500_PM_PH15:
201 setbits32(&rcpm_v2_regs->pcph15clrr, mask);
202 break;
203 case E500_PM_PH20:
204 setbits32(&rcpm_v2_regs->pcph20clrr, mask);
205 break;
206 case E500_PM_PH30:
207 setbits32(&rcpm_v2_regs->pcph30clrr, mask);
208 break;
209 default:
210 pr_warn("Unknown cpu PM state (%d)\n", state);
211 }
212}
213
214static void rcpm_v2_cpu_up_prepare(int cpu)
215{
216 rcpm_v2_cpu_exit_state(cpu, E500_PM_PH20);
217 rcpm_v2_irq_unmask(cpu);
218}
219
220static int rcpm_v1_plat_enter_state(int state)
221{
222 u32 *pmcsr_reg = &rcpm_v1_regs->powmgtcsr;
223 int ret = 0;
224 int result;
225
226 switch (state) {
227 case PLAT_PM_SLEEP:
228 setbits32(pmcsr_reg, RCPM_POWMGTCSR_SLP);
229
230 /* Upon resume, wait for RCPM_POWMGTCSR_SLP bit to be clear. */
231 result = spin_event_timeout(
232 !(in_be32(pmcsr_reg) & RCPM_POWMGTCSR_SLP), 10000, 10);
233 if (!result) {
234 pr_err("timeout waiting for SLP bit to be cleared\n");
235 ret = -ETIMEDOUT;
236 }
237 break;
238 default:
239 pr_warn("Unknown platform PM state (%d)", state);
240 ret = -EINVAL;
241 }
242
243 return ret;
244}
245
246static int rcpm_v2_plat_enter_state(int state)
247{
248 u32 *pmcsr_reg = &rcpm_v2_regs->powmgtcsr;
249 int ret = 0;
250 int result;
251
252 switch (state) {
253 case PLAT_PM_LPM20:
254 /* clear previous LPM20 status */
255 setbits32(pmcsr_reg, RCPM_POWMGTCSR_P_LPM20_ST);
256 /* enter LPM20 status */
257 setbits32(pmcsr_reg, RCPM_POWMGTCSR_LPM20_RQ);
258
259 /* At this point, the device is in LPM20 status. */
260
261 /* resume ... */
262 result = spin_event_timeout(
263 !(in_be32(pmcsr_reg) & RCPM_POWMGTCSR_LPM20_ST), 10000, 10);
264 if (!result) {
265 pr_err("timeout waiting for LPM20 bit to be cleared\n");
266 ret = -ETIMEDOUT;
267 }
268 break;
269 default:
270 pr_warn("Unknown platform PM state (%d)\n", state);
271 ret = -EINVAL;
272 }
273
274 return ret;
275}
276
277static int rcpm_v1_plat_enter_sleep(void)
278{
279 return rcpm_v1_plat_enter_state(PLAT_PM_SLEEP);
280}
281
282static int rcpm_v2_plat_enter_sleep(void)
283{
284 return rcpm_v2_plat_enter_state(PLAT_PM_LPM20);
285}
286
287static void rcpm_common_freeze_time_base(u32 *tben_reg, int freeze)
288{
289 static u32 mask;
290
291 if (freeze) {
292 mask = in_be32(tben_reg);
293 clrbits32(tben_reg, mask);
294 } else {
295 setbits32(tben_reg, mask);
296 }
297
298 /* read back to push the previous write */
299 in_be32(tben_reg);
300}
301
302static void rcpm_v1_freeze_time_base(bool freeze)
303{
304 rcpm_common_freeze_time_base(&rcpm_v1_regs->ctbenr, freeze);
305}
306
307static void rcpm_v2_freeze_time_base(bool freeze)
308{
309 rcpm_common_freeze_time_base(&rcpm_v2_regs->pctbenr, freeze);
310}
311
312static unsigned int rcpm_get_pm_modes(void)
313{
314 return fsl_supported_pm_modes;
315}
316
317static const struct fsl_pm_ops qoriq_rcpm_v1_ops = {
318 .irq_mask = rcpm_v1_irq_mask,
319 .irq_unmask = rcpm_v1_irq_unmask,
320 .cpu_enter_state = rcpm_v1_cpu_enter_state,
321 .cpu_exit_state = rcpm_v1_cpu_exit_state,
322 .cpu_up_prepare = rcpm_v1_cpu_up_prepare,
323 .cpu_die = rcpm_v1_cpu_die,
324 .plat_enter_sleep = rcpm_v1_plat_enter_sleep,
325 .set_ip_power = rcpm_v1_set_ip_power,
326 .freeze_time_base = rcpm_v1_freeze_time_base,
327 .get_pm_modes = rcpm_get_pm_modes,
328};
329
330static const struct fsl_pm_ops qoriq_rcpm_v2_ops = {
331 .irq_mask = rcpm_v2_irq_mask,
332 .irq_unmask = rcpm_v2_irq_unmask,
333 .cpu_enter_state = rcpm_v2_cpu_enter_state,
334 .cpu_exit_state = rcpm_v2_cpu_exit_state,
335 .cpu_up_prepare = rcpm_v2_cpu_up_prepare,
336 .cpu_die = rcpm_v2_cpu_die,
337 .plat_enter_sleep = rcpm_v2_plat_enter_sleep,
338 .set_ip_power = rcpm_v2_set_ip_power,
339 .freeze_time_base = rcpm_v2_freeze_time_base,
340 .get_pm_modes = rcpm_get_pm_modes,
341};
342
343static const struct of_device_id rcpm_matches[] = {
344 {
345 .compatible = "fsl,qoriq-rcpm-1.0",
346 .data = &qoriq_rcpm_v1_ops,
347 },
348 {
349 .compatible = "fsl,qoriq-rcpm-2.0",
350 .data = &qoriq_rcpm_v2_ops,
351 },
352 {
353 .compatible = "fsl,qoriq-rcpm-2.1",
354 .data = &qoriq_rcpm_v2_ops,
355 },
356 {},
357};
358
359int __init fsl_rcpm_init(void)
360{
361 struct device_node *np;
362 const struct of_device_id *match;
363 void __iomem *base;
364
365 np = of_find_matching_node_and_match(NULL, rcpm_matches, &match);
366 if (!np)
367 return 0;
368
369 base = of_iomap(np, 0);
370 of_node_put(np);
371 if (!base) {
372 pr_err("of_iomap() error.\n");
373 return -ENOMEM;
374 }
375
376 rcpm_v1_regs = base;
377 rcpm_v2_regs = base;
378
379 /* support sleep by default */
380 fsl_supported_pm_modes = FSL_PM_SLEEP;
381
382 qoriq_pm_ops = match->data;
383
384 return 0;
385}