diff options
author | Rafael J. Wysocki <rafael.j.wysocki@intel.com> | 2013-11-07 13:26:02 -0500 |
---|---|---|
committer | Rafael J. Wysocki <rafael.j.wysocki@intel.com> | 2013-11-07 13:26:02 -0500 |
commit | 15d4cb9013ab7cdf10760aeabd07b007d635b321 (patch) | |
tree | 6c49842895e8f925881e4d0ccbc597b7081c8508 /arch | |
parent | adf9684588ec1769b77a9e1c7c68fbad49ec5e43 (diff) | |
parent | fbbcdc0744dace5fcc8147d11c5fb0791126848a (diff) |
Merge branch 'pm-cpufreq'
* pm-cpufreq:
intel_pstate: skip the driver if ACPI has power mgmt option
cpufreq: ondemand: Remove redundant return statement
cpufreq: move freq change notifications to cpufreq core
cpufreq: distinguish drivers that do asynchronous notifications
cpufreq/intel_pstate: Add static declarations to internal functions
cpufreq: arm_big_little: reconfigure switcher behavior at run time
cpufreq: arm_big_little: add in-kernel switching (IKS) support
ARM: vexpress/TC2: register vexpress-spc cpufreq device
cpufreq: arm_big_little: add vexpress SPC interface driver
ARM: vexpress/TC2: add cpu clock support
ARM: vexpress/TC2: add support for CPU DVFS
Diffstat (limited to 'arch')
-rw-r--r-- | arch/arm/mach-vexpress/Kconfig | 12 | ||||
-rw-r--r-- | arch/arm/mach-vexpress/Makefile | 3 | ||||
-rw-r--r-- | arch/arm/mach-vexpress/spc.c | 366 | ||||
-rw-r--r-- | arch/arm/mach-vexpress/spc.h | 2 | ||||
-rw-r--r-- | arch/arm/mach-vexpress/tc2_pm.c | 7 |
5 files changed, 385 insertions, 5 deletions
diff --git a/arch/arm/mach-vexpress/Kconfig b/arch/arm/mach-vexpress/Kconfig index 365795447804..c77170c04fd0 100644 --- a/arch/arm/mach-vexpress/Kconfig +++ b/arch/arm/mach-vexpress/Kconfig | |||
@@ -66,10 +66,22 @@ config ARCH_VEXPRESS_DCSCB | |||
66 | This is needed to provide CPU and cluster power management | 66 | This is needed to provide CPU and cluster power management |
67 | on RTSM implementing big.LITTLE. | 67 | on RTSM implementing big.LITTLE. |
68 | 68 | ||
69 | config ARCH_VEXPRESS_SPC | ||
70 | bool "Versatile Express Serial Power Controller (SPC)" | ||
71 | select ARCH_HAS_CPUFREQ | ||
72 | select ARCH_HAS_OPP | ||
73 | select PM_OPP | ||
74 | help | ||
75 | The TC2 (A15x2 A7x3) versatile express core tile integrates a logic | ||
76 | block called Serial Power Controller (SPC) that provides the interface | ||
77 | between the dual cluster test-chip and the M3 microcontroller that | ||
78 | carries out power management. | ||
79 | |||
69 | config ARCH_VEXPRESS_TC2_PM | 80 | config ARCH_VEXPRESS_TC2_PM |
70 | bool "Versatile Express TC2 power management" | 81 | bool "Versatile Express TC2 power management" |
71 | depends on MCPM | 82 | depends on MCPM |
72 | select ARM_CCI | 83 | select ARM_CCI |
84 | select ARCH_VEXPRESS_SPC | ||
73 | help | 85 | help |
74 | Support for CPU and cluster power management on Versatile Express | 86 | Support for CPU and cluster power management on Versatile Express |
75 | with a TC2 (A15x2 A7x3) big.LITTLE core tile. | 87 | with a TC2 (A15x2 A7x3) big.LITTLE core tile. |
diff --git a/arch/arm/mach-vexpress/Makefile b/arch/arm/mach-vexpress/Makefile index 505e64ab3eae..0997e0b7494c 100644 --- a/arch/arm/mach-vexpress/Makefile +++ b/arch/arm/mach-vexpress/Makefile | |||
@@ -8,7 +8,8 @@ obj-y := v2m.o | |||
8 | obj-$(CONFIG_ARCH_VEXPRESS_CA9X4) += ct-ca9x4.o | 8 | obj-$(CONFIG_ARCH_VEXPRESS_CA9X4) += ct-ca9x4.o |
9 | obj-$(CONFIG_ARCH_VEXPRESS_DCSCB) += dcscb.o dcscb_setup.o | 9 | obj-$(CONFIG_ARCH_VEXPRESS_DCSCB) += dcscb.o dcscb_setup.o |
10 | CFLAGS_dcscb.o += -march=armv7-a | 10 | CFLAGS_dcscb.o += -march=armv7-a |
11 | obj-$(CONFIG_ARCH_VEXPRESS_TC2_PM) += tc2_pm.o spc.o | 11 | obj-$(CONFIG_ARCH_VEXPRESS_SPC) += spc.o |
12 | obj-$(CONFIG_ARCH_VEXPRESS_TC2_PM) += tc2_pm.o | ||
12 | CFLAGS_tc2_pm.o += -march=armv7-a | 13 | CFLAGS_tc2_pm.o += -march=armv7-a |
13 | obj-$(CONFIG_SMP) += platsmp.o | 14 | obj-$(CONFIG_SMP) += platsmp.o |
14 | obj-$(CONFIG_HOTPLUG_CPU) += hotplug.o | 15 | obj-$(CONFIG_HOTPLUG_CPU) += hotplug.o |
diff --git a/arch/arm/mach-vexpress/spc.c b/arch/arm/mach-vexpress/spc.c index eefb029197ca..033d34dcbd3f 100644 --- a/arch/arm/mach-vexpress/spc.c +++ b/arch/arm/mach-vexpress/spc.c | |||
@@ -17,14 +17,31 @@ | |||
17 | * GNU General Public License for more details. | 17 | * GNU General Public License for more details. |
18 | */ | 18 | */ |
19 | 19 | ||
20 | #include <linux/clk-provider.h> | ||
21 | #include <linux/clkdev.h> | ||
22 | #include <linux/cpu.h> | ||
23 | #include <linux/delay.h> | ||
20 | #include <linux/err.h> | 24 | #include <linux/err.h> |
25 | #include <linux/interrupt.h> | ||
21 | #include <linux/io.h> | 26 | #include <linux/io.h> |
27 | #include <linux/platform_device.h> | ||
28 | #include <linux/pm_opp.h> | ||
22 | #include <linux/slab.h> | 29 | #include <linux/slab.h> |
30 | #include <linux/semaphore.h> | ||
23 | 31 | ||
24 | #include <asm/cacheflush.h> | 32 | #include <asm/cacheflush.h> |
25 | 33 | ||
26 | #define SPCLOG "vexpress-spc: " | 34 | #define SPCLOG "vexpress-spc: " |
27 | 35 | ||
36 | #define PERF_LVL_A15 0x00 | ||
37 | #define PERF_REQ_A15 0x04 | ||
38 | #define PERF_LVL_A7 0x08 | ||
39 | #define PERF_REQ_A7 0x0c | ||
40 | #define COMMS 0x10 | ||
41 | #define COMMS_REQ 0x14 | ||
42 | #define PWC_STATUS 0x18 | ||
43 | #define PWC_FLAG 0x1c | ||
44 | |||
28 | /* SPC wake-up IRQs status and mask */ | 45 | /* SPC wake-up IRQs status and mask */ |
29 | #define WAKE_INT_MASK 0x24 | 46 | #define WAKE_INT_MASK 0x24 |
30 | #define WAKE_INT_RAW 0x28 | 47 | #define WAKE_INT_RAW 0x28 |
@@ -36,12 +53,45 @@ | |||
36 | #define A15_BX_ADDR0 0x68 | 53 | #define A15_BX_ADDR0 0x68 |
37 | #define A7_BX_ADDR0 0x78 | 54 | #define A7_BX_ADDR0 0x78 |
38 | 55 | ||
56 | /* SPC system config interface registers */ | ||
57 | #define SYSCFG_WDATA 0x70 | ||
58 | #define SYSCFG_RDATA 0x74 | ||
59 | |||
60 | /* A15/A7 OPP virtual register base */ | ||
61 | #define A15_PERFVAL_BASE 0xC10 | ||
62 | #define A7_PERFVAL_BASE 0xC30 | ||
63 | |||
64 | /* Config interface control bits */ | ||
65 | #define SYSCFG_START (1 << 31) | ||
66 | #define SYSCFG_SCC (6 << 20) | ||
67 | #define SYSCFG_STAT (14 << 20) | ||
68 | |||
39 | /* wake-up interrupt masks */ | 69 | /* wake-up interrupt masks */ |
40 | #define GBL_WAKEUP_INT_MSK (0x3 << 10) | 70 | #define GBL_WAKEUP_INT_MSK (0x3 << 10) |
41 | 71 | ||
42 | /* TC2 static dual-cluster configuration */ | 72 | /* TC2 static dual-cluster configuration */ |
43 | #define MAX_CLUSTERS 2 | 73 | #define MAX_CLUSTERS 2 |
44 | 74 | ||
75 | /* | ||
76 | * Even though the SPC takes max 3-5 ms to complete any OPP/COMMS | ||
77 | * operation, the operation could start just before jiffie is about | ||
78 | * to be incremented. So setting timeout value of 20ms = 2jiffies@100Hz | ||
79 | */ | ||
80 | #define TIMEOUT_US 20000 | ||
81 | |||
82 | #define MAX_OPPS 8 | ||
83 | #define CA15_DVFS 0 | ||
84 | #define CA7_DVFS 1 | ||
85 | #define SPC_SYS_CFG 2 | ||
86 | #define STAT_COMPLETE(type) ((1 << 0) << (type << 2)) | ||
87 | #define STAT_ERR(type) ((1 << 1) << (type << 2)) | ||
88 | #define RESPONSE_MASK(type) (STAT_COMPLETE(type) | STAT_ERR(type)) | ||
89 | |||
90 | struct ve_spc_opp { | ||
91 | unsigned long freq; | ||
92 | unsigned long u_volt; | ||
93 | }; | ||
94 | |||
45 | struct ve_spc_drvdata { | 95 | struct ve_spc_drvdata { |
46 | void __iomem *baseaddr; | 96 | void __iomem *baseaddr; |
47 | /* | 97 | /* |
@@ -49,6 +99,12 @@ struct ve_spc_drvdata { | |||
49 | * It corresponds to A15 processors MPIDR[15:8] bitfield | 99 | * It corresponds to A15 processors MPIDR[15:8] bitfield |
50 | */ | 100 | */ |
51 | u32 a15_clusid; | 101 | u32 a15_clusid; |
102 | uint32_t cur_rsp_mask; | ||
103 | uint32_t cur_rsp_stat; | ||
104 | struct semaphore sem; | ||
105 | struct completion done; | ||
106 | struct ve_spc_opp *opps[MAX_CLUSTERS]; | ||
107 | int num_opps[MAX_CLUSTERS]; | ||
52 | }; | 108 | }; |
53 | 109 | ||
54 | static struct ve_spc_drvdata *info; | 110 | static struct ve_spc_drvdata *info; |
@@ -157,8 +213,197 @@ void ve_spc_powerdown(u32 cluster, bool enable) | |||
157 | writel_relaxed(enable, info->baseaddr + pwdrn_reg); | 213 | writel_relaxed(enable, info->baseaddr + pwdrn_reg); |
158 | } | 214 | } |
159 | 215 | ||
160 | int __init ve_spc_init(void __iomem *baseaddr, u32 a15_clusid) | 216 | static int ve_spc_get_performance(int cluster, u32 *freq) |
217 | { | ||
218 | struct ve_spc_opp *opps = info->opps[cluster]; | ||
219 | u32 perf_cfg_reg = 0; | ||
220 | u32 perf; | ||
221 | |||
222 | perf_cfg_reg = cluster_is_a15(cluster) ? PERF_LVL_A15 : PERF_LVL_A7; | ||
223 | |||
224 | perf = readl_relaxed(info->baseaddr + perf_cfg_reg); | ||
225 | if (perf >= info->num_opps[cluster]) | ||
226 | return -EINVAL; | ||
227 | |||
228 | opps += perf; | ||
229 | *freq = opps->freq; | ||
230 | |||
231 | return 0; | ||
232 | } | ||
233 | |||
234 | /* find closest match to given frequency in OPP table */ | ||
235 | static int ve_spc_round_performance(int cluster, u32 freq) | ||
236 | { | ||
237 | int idx, max_opp = info->num_opps[cluster]; | ||
238 | struct ve_spc_opp *opps = info->opps[cluster]; | ||
239 | u32 fmin = 0, fmax = ~0, ftmp; | ||
240 | |||
241 | freq /= 1000; /* OPP entries in kHz */ | ||
242 | for (idx = 0; idx < max_opp; idx++, opps++) { | ||
243 | ftmp = opps->freq; | ||
244 | if (ftmp >= freq) { | ||
245 | if (ftmp <= fmax) | ||
246 | fmax = ftmp; | ||
247 | } else { | ||
248 | if (ftmp >= fmin) | ||
249 | fmin = ftmp; | ||
250 | } | ||
251 | } | ||
252 | if (fmax != ~0) | ||
253 | return fmax * 1000; | ||
254 | else | ||
255 | return fmin * 1000; | ||
256 | } | ||
257 | |||
258 | static int ve_spc_find_performance_index(int cluster, u32 freq) | ||
259 | { | ||
260 | int idx, max_opp = info->num_opps[cluster]; | ||
261 | struct ve_spc_opp *opps = info->opps[cluster]; | ||
262 | |||
263 | for (idx = 0; idx < max_opp; idx++, opps++) | ||
264 | if (opps->freq == freq) | ||
265 | break; | ||
266 | return (idx == max_opp) ? -EINVAL : idx; | ||
267 | } | ||
268 | |||
269 | static int ve_spc_waitforcompletion(int req_type) | ||
270 | { | ||
271 | int ret = wait_for_completion_interruptible_timeout( | ||
272 | &info->done, usecs_to_jiffies(TIMEOUT_US)); | ||
273 | if (ret == 0) | ||
274 | ret = -ETIMEDOUT; | ||
275 | else if (ret > 0) | ||
276 | ret = info->cur_rsp_stat & STAT_COMPLETE(req_type) ? 0 : -EIO; | ||
277 | return ret; | ||
278 | } | ||
279 | |||
280 | static int ve_spc_set_performance(int cluster, u32 freq) | ||
281 | { | ||
282 | u32 perf_cfg_reg, perf_stat_reg; | ||
283 | int ret, perf, req_type; | ||
284 | |||
285 | if (cluster_is_a15(cluster)) { | ||
286 | req_type = CA15_DVFS; | ||
287 | perf_cfg_reg = PERF_LVL_A15; | ||
288 | perf_stat_reg = PERF_REQ_A15; | ||
289 | } else { | ||
290 | req_type = CA7_DVFS; | ||
291 | perf_cfg_reg = PERF_LVL_A7; | ||
292 | perf_stat_reg = PERF_REQ_A7; | ||
293 | } | ||
294 | |||
295 | perf = ve_spc_find_performance_index(cluster, freq); | ||
296 | |||
297 | if (perf < 0) | ||
298 | return perf; | ||
299 | |||
300 | if (down_timeout(&info->sem, usecs_to_jiffies(TIMEOUT_US))) | ||
301 | return -ETIME; | ||
302 | |||
303 | init_completion(&info->done); | ||
304 | info->cur_rsp_mask = RESPONSE_MASK(req_type); | ||
305 | |||
306 | writel(perf, info->baseaddr + perf_cfg_reg); | ||
307 | ret = ve_spc_waitforcompletion(req_type); | ||
308 | |||
309 | info->cur_rsp_mask = 0; | ||
310 | up(&info->sem); | ||
311 | |||
312 | return ret; | ||
313 | } | ||
314 | |||
315 | static int ve_spc_read_sys_cfg(int func, int offset, uint32_t *data) | ||
316 | { | ||
317 | int ret; | ||
318 | |||
319 | if (down_timeout(&info->sem, usecs_to_jiffies(TIMEOUT_US))) | ||
320 | return -ETIME; | ||
321 | |||
322 | init_completion(&info->done); | ||
323 | info->cur_rsp_mask = RESPONSE_MASK(SPC_SYS_CFG); | ||
324 | |||
325 | /* Set the control value */ | ||
326 | writel(SYSCFG_START | func | offset >> 2, info->baseaddr + COMMS); | ||
327 | ret = ve_spc_waitforcompletion(SPC_SYS_CFG); | ||
328 | |||
329 | if (ret == 0) | ||
330 | *data = readl(info->baseaddr + SYSCFG_RDATA); | ||
331 | |||
332 | info->cur_rsp_mask = 0; | ||
333 | up(&info->sem); | ||
334 | |||
335 | return ret; | ||
336 | } | ||
337 | |||
338 | static irqreturn_t ve_spc_irq_handler(int irq, void *data) | ||
339 | { | ||
340 | struct ve_spc_drvdata *drv_data = data; | ||
341 | uint32_t status = readl_relaxed(drv_data->baseaddr + PWC_STATUS); | ||
342 | |||
343 | if (info->cur_rsp_mask & status) { | ||
344 | info->cur_rsp_stat = status; | ||
345 | complete(&drv_data->done); | ||
346 | } | ||
347 | |||
348 | return IRQ_HANDLED; | ||
349 | } | ||
350 | |||
351 | /* | ||
352 | * +--------------------------+ | ||
353 | * | 31 20 | 19 0 | | ||
354 | * +--------------------------+ | ||
355 | * | u_volt | freq(kHz) | | ||
356 | * +--------------------------+ | ||
357 | */ | ||
358 | #define MULT_FACTOR 20 | ||
359 | #define VOLT_SHIFT 20 | ||
360 | #define FREQ_MASK (0xFFFFF) | ||
361 | static int ve_spc_populate_opps(uint32_t cluster) | ||
161 | { | 362 | { |
363 | uint32_t data = 0, off, ret, idx; | ||
364 | struct ve_spc_opp *opps; | ||
365 | |||
366 | opps = kzalloc(sizeof(*opps) * MAX_OPPS, GFP_KERNEL); | ||
367 | if (!opps) | ||
368 | return -ENOMEM; | ||
369 | |||
370 | info->opps[cluster] = opps; | ||
371 | |||
372 | off = cluster_is_a15(cluster) ? A15_PERFVAL_BASE : A7_PERFVAL_BASE; | ||
373 | for (idx = 0; idx < MAX_OPPS; idx++, off += 4, opps++) { | ||
374 | ret = ve_spc_read_sys_cfg(SYSCFG_SCC, off, &data); | ||
375 | if (!ret) { | ||
376 | opps->freq = (data & FREQ_MASK) * MULT_FACTOR; | ||
377 | opps->u_volt = data >> VOLT_SHIFT; | ||
378 | } else { | ||
379 | break; | ||
380 | } | ||
381 | } | ||
382 | info->num_opps[cluster] = idx; | ||
383 | |||
384 | return ret; | ||
385 | } | ||
386 | |||
387 | static int ve_init_opp_table(struct device *cpu_dev) | ||
388 | { | ||
389 | int cluster = topology_physical_package_id(cpu_dev->id); | ||
390 | int idx, ret = 0, max_opp = info->num_opps[cluster]; | ||
391 | struct ve_spc_opp *opps = info->opps[cluster]; | ||
392 | |||
393 | for (idx = 0; idx < max_opp; idx++, opps++) { | ||
394 | ret = dev_pm_opp_add(cpu_dev, opps->freq * 1000, opps->u_volt); | ||
395 | if (ret) { | ||
396 | dev_warn(cpu_dev, "failed to add opp %lu %lu\n", | ||
397 | opps->freq, opps->u_volt); | ||
398 | return ret; | ||
399 | } | ||
400 | } | ||
401 | return ret; | ||
402 | } | ||
403 | |||
404 | int __init ve_spc_init(void __iomem *baseaddr, u32 a15_clusid, int irq) | ||
405 | { | ||
406 | int ret; | ||
162 | info = kzalloc(sizeof(*info), GFP_KERNEL); | 407 | info = kzalloc(sizeof(*info), GFP_KERNEL); |
163 | if (!info) { | 408 | if (!info) { |
164 | pr_err(SPCLOG "unable to allocate mem\n"); | 409 | pr_err(SPCLOG "unable to allocate mem\n"); |
@@ -168,6 +413,25 @@ int __init ve_spc_init(void __iomem *baseaddr, u32 a15_clusid) | |||
168 | info->baseaddr = baseaddr; | 413 | info->baseaddr = baseaddr; |
169 | info->a15_clusid = a15_clusid; | 414 | info->a15_clusid = a15_clusid; |
170 | 415 | ||
416 | if (irq <= 0) { | ||
417 | pr_err(SPCLOG "Invalid IRQ %d\n", irq); | ||
418 | kfree(info); | ||
419 | return -EINVAL; | ||
420 | } | ||
421 | |||
422 | init_completion(&info->done); | ||
423 | |||
424 | readl_relaxed(info->baseaddr + PWC_STATUS); | ||
425 | |||
426 | ret = request_irq(irq, ve_spc_irq_handler, IRQF_TRIGGER_HIGH | ||
427 | | IRQF_ONESHOT, "vexpress-spc", info); | ||
428 | if (ret) { | ||
429 | pr_err(SPCLOG "IRQ %d request failed\n", irq); | ||
430 | kfree(info); | ||
431 | return -ENODEV; | ||
432 | } | ||
433 | |||
434 | sema_init(&info->sem, 1); | ||
171 | /* | 435 | /* |
172 | * Multi-cluster systems may need this data when non-coherent, during | 436 | * Multi-cluster systems may need this data when non-coherent, during |
173 | * cluster power-up/power-down. Make sure driver info reaches main | 437 | * cluster power-up/power-down. Make sure driver info reaches main |
@@ -178,3 +442,103 @@ int __init ve_spc_init(void __iomem *baseaddr, u32 a15_clusid) | |||
178 | 442 | ||
179 | return 0; | 443 | return 0; |
180 | } | 444 | } |
445 | |||
446 | struct clk_spc { | ||
447 | struct clk_hw hw; | ||
448 | int cluster; | ||
449 | }; | ||
450 | |||
451 | #define to_clk_spc(spc) container_of(spc, struct clk_spc, hw) | ||
452 | static unsigned long spc_recalc_rate(struct clk_hw *hw, | ||
453 | unsigned long parent_rate) | ||
454 | { | ||
455 | struct clk_spc *spc = to_clk_spc(hw); | ||
456 | u32 freq; | ||
457 | |||
458 | if (ve_spc_get_performance(spc->cluster, &freq)) | ||
459 | return -EIO; | ||
460 | |||
461 | return freq * 1000; | ||
462 | } | ||
463 | |||
464 | static long spc_round_rate(struct clk_hw *hw, unsigned long drate, | ||
465 | unsigned long *parent_rate) | ||
466 | { | ||
467 | struct clk_spc *spc = to_clk_spc(hw); | ||
468 | |||
469 | return ve_spc_round_performance(spc->cluster, drate); | ||
470 | } | ||
471 | |||
472 | static int spc_set_rate(struct clk_hw *hw, unsigned long rate, | ||
473 | unsigned long parent_rate) | ||
474 | { | ||
475 | struct clk_spc *spc = to_clk_spc(hw); | ||
476 | |||
477 | return ve_spc_set_performance(spc->cluster, rate / 1000); | ||
478 | } | ||
479 | |||
480 | static struct clk_ops clk_spc_ops = { | ||
481 | .recalc_rate = spc_recalc_rate, | ||
482 | .round_rate = spc_round_rate, | ||
483 | .set_rate = spc_set_rate, | ||
484 | }; | ||
485 | |||
486 | static struct clk *ve_spc_clk_register(struct device *cpu_dev) | ||
487 | { | ||
488 | struct clk_init_data init; | ||
489 | struct clk_spc *spc; | ||
490 | |||
491 | spc = kzalloc(sizeof(*spc), GFP_KERNEL); | ||
492 | if (!spc) { | ||
493 | pr_err("could not allocate spc clk\n"); | ||
494 | return ERR_PTR(-ENOMEM); | ||
495 | } | ||
496 | |||
497 | spc->hw.init = &init; | ||
498 | spc->cluster = topology_physical_package_id(cpu_dev->id); | ||
499 | |||
500 | init.name = dev_name(cpu_dev); | ||
501 | init.ops = &clk_spc_ops; | ||
502 | init.flags = CLK_IS_ROOT | CLK_GET_RATE_NOCACHE; | ||
503 | init.num_parents = 0; | ||
504 | |||
505 | return devm_clk_register(cpu_dev, &spc->hw); | ||
506 | } | ||
507 | |||
508 | static int __init ve_spc_clk_init(void) | ||
509 | { | ||
510 | int cpu; | ||
511 | struct clk *clk; | ||
512 | |||
513 | if (!info) | ||
514 | return 0; /* Continue only if SPC is initialised */ | ||
515 | |||
516 | if (ve_spc_populate_opps(0) || ve_spc_populate_opps(1)) { | ||
517 | pr_err("failed to build OPP table\n"); | ||
518 | return -ENODEV; | ||
519 | } | ||
520 | |||
521 | for_each_possible_cpu(cpu) { | ||
522 | struct device *cpu_dev = get_cpu_device(cpu); | ||
523 | if (!cpu_dev) { | ||
524 | pr_warn("failed to get cpu%d device\n", cpu); | ||
525 | continue; | ||
526 | } | ||
527 | clk = ve_spc_clk_register(cpu_dev); | ||
528 | if (IS_ERR(clk)) { | ||
529 | pr_warn("failed to register cpu%d clock\n", cpu); | ||
530 | continue; | ||
531 | } | ||
532 | if (clk_register_clkdev(clk, NULL, dev_name(cpu_dev))) { | ||
533 | pr_warn("failed to register cpu%d clock lookup\n", cpu); | ||
534 | continue; | ||
535 | } | ||
536 | |||
537 | if (ve_init_opp_table(cpu_dev)) | ||
538 | pr_warn("failed to initialise cpu%d opp table\n", cpu); | ||
539 | } | ||
540 | |||
541 | platform_device_register_simple("vexpress-spc-cpufreq", -1, NULL, 0); | ||
542 | return 0; | ||
543 | } | ||
544 | module_init(ve_spc_clk_init); | ||
diff --git a/arch/arm/mach-vexpress/spc.h b/arch/arm/mach-vexpress/spc.h index 5f7e4a446a17..dbd44c3720f9 100644 --- a/arch/arm/mach-vexpress/spc.h +++ b/arch/arm/mach-vexpress/spc.h | |||
@@ -15,7 +15,7 @@ | |||
15 | #ifndef __SPC_H_ | 15 | #ifndef __SPC_H_ |
16 | #define __SPC_H_ | 16 | #define __SPC_H_ |
17 | 17 | ||
18 | int __init ve_spc_init(void __iomem *base, u32 a15_clusid); | 18 | int __init ve_spc_init(void __iomem *base, u32 a15_clusid, int irq); |
19 | void ve_spc_global_wakeup_irq(bool set); | 19 | void ve_spc_global_wakeup_irq(bool set); |
20 | void ve_spc_cpu_wakeup_irq(u32 cluster, u32 cpu, bool set); | 20 | void ve_spc_cpu_wakeup_irq(u32 cluster, u32 cpu, bool set); |
21 | void ve_spc_set_resume_addr(u32 cluster, u32 cpu, u32 addr); | 21 | void ve_spc_set_resume_addr(u32 cluster, u32 cpu, u32 addr); |
diff --git a/arch/arm/mach-vexpress/tc2_pm.c b/arch/arm/mach-vexpress/tc2_pm.c index e6eb48192912..d38130aba464 100644 --- a/arch/arm/mach-vexpress/tc2_pm.c +++ b/arch/arm/mach-vexpress/tc2_pm.c | |||
@@ -16,6 +16,7 @@ | |||
16 | #include <linux/io.h> | 16 | #include <linux/io.h> |
17 | #include <linux/kernel.h> | 17 | #include <linux/kernel.h> |
18 | #include <linux/of_address.h> | 18 | #include <linux/of_address.h> |
19 | #include <linux/of_irq.h> | ||
19 | #include <linux/spinlock.h> | 20 | #include <linux/spinlock.h> |
20 | #include <linux/errno.h> | 21 | #include <linux/errno.h> |
21 | #include <linux/irqchip/arm-gic.h> | 22 | #include <linux/irqchip/arm-gic.h> |
@@ -311,7 +312,7 @@ static void __naked tc2_pm_power_up_setup(unsigned int affinity_level) | |||
311 | 312 | ||
312 | static int __init tc2_pm_init(void) | 313 | static int __init tc2_pm_init(void) |
313 | { | 314 | { |
314 | int ret; | 315 | int ret, irq; |
315 | void __iomem *scc; | 316 | void __iomem *scc; |
316 | u32 a15_cluster_id, a7_cluster_id, sys_info; | 317 | u32 a15_cluster_id, a7_cluster_id, sys_info; |
317 | struct device_node *np; | 318 | struct device_node *np; |
@@ -336,13 +337,15 @@ static int __init tc2_pm_init(void) | |||
336 | tc2_nr_cpus[a15_cluster_id] = (sys_info >> 16) & 0xf; | 337 | tc2_nr_cpus[a15_cluster_id] = (sys_info >> 16) & 0xf; |
337 | tc2_nr_cpus[a7_cluster_id] = (sys_info >> 20) & 0xf; | 338 | tc2_nr_cpus[a7_cluster_id] = (sys_info >> 20) & 0xf; |
338 | 339 | ||
340 | irq = irq_of_parse_and_map(np, 0); | ||
341 | |||
339 | /* | 342 | /* |
340 | * A subset of the SCC registers is also used to communicate | 343 | * A subset of the SCC registers is also used to communicate |
341 | * with the SPC (power controller). We need to be able to | 344 | * with the SPC (power controller). We need to be able to |
342 | * drive it very early in the boot process to power up | 345 | * drive it very early in the boot process to power up |
343 | * processors, so we initialize the SPC driver here. | 346 | * processors, so we initialize the SPC driver here. |
344 | */ | 347 | */ |
345 | ret = ve_spc_init(scc + SPC_BASE, a15_cluster_id); | 348 | ret = ve_spc_init(scc + SPC_BASE, a15_cluster_id, irq); |
346 | if (ret) | 349 | if (ret) |
347 | return ret; | 350 | return ret; |
348 | 351 | ||