blob: a6e8fdbfe8febcf7f8a86f61010d9dec6cba2b2d [file] [log] [blame]
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001/*
2 * Idle processing for ARMv7-based Qualcomm SoCs.
3 *
4 * Copyright (C) 2007 Google, Inc.
5 * Copyright (c) 2007-2009, 2011 Code Aurora Forum. All rights reserved.
6 *
7 * This software is licensed under the terms of the GNU General Public
8 * License version 2, as published by the Free Software Foundation, and
9 * may be copied, distributed, and modified under those terms.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 */
17
18#include <linux/linkage.h>
19#include <linux/threads.h>
20#include <asm/assembler.h>
21
22#ifdef CONFIG_MSM_CPU_AVS
23/* 11 general purpose registers (r4-r14), 10 cp15 registers, 3 AVS registers */
24#define CPU_SAVED_STATE_SIZE (4 * 11 + 4 * 10 + 4 * 3)
25#else
26/* 11 general purpose registers (r4-r14), 10 cp15 registers */
27#define CPU_SAVED_STATE_SIZE (4 * 11 + 4 * 10)
28#endif
Maheshkumar Sivasubramanianfa1d0dd2011-07-26 16:02:55 -060029#ifdef CONFIG_ARCH_MSM_KRAIT
30#define SCM_SVC_BOOT 0x1
31#define SCM_CMD_TERMINATE_PC 0x2
32#endif
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070033
34ENTRY(msm_arch_idle)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070035#ifdef CONFIG_MSM_JTAG_V7
Pratik Patel7831c082011-06-08 21:44:37 -070036 stmfd sp!, {lr}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070037 bl msm_save_jtag_debug
38#endif
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070039 wfi
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070040#ifdef CONFIG_MSM_JTAG_V7
41 bl msm_restore_jtag_debug
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070042 ldmfd sp!, {lr}
Pratik Patel7831c082011-06-08 21:44:37 -070043#endif
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070044 bx lr
45
46ENTRY(msm_pm_collapse)
47#if defined(CONFIG_MSM_FIQ_SUPPORT)
48 cpsid f
49#endif
50
51 ldr r0, =saved_state
52#if (NR_CPUS >= 2)
53 mrc p15, 0, r1, c0, c0, 5 /* MPIDR */
54 ands r1, r1, #15 /* What CPU am I */
55 addne r0, r0, #CPU_SAVED_STATE_SIZE
56#endif
57
58 stmia r0!, {r4-r14}
59 mrc p15, 0, r1, c1, c0, 0 /* MMU control */
60 mrc p15, 0, r2, c2, c0, 0 /* TTBR0 */
61 mrc p15, 0, r3, c3, c0, 0 /* dacr */
62#ifdef CONFIG_ARCH_MSM_SCORPION
63 /* This instruction is not valid for non scorpion processors */
64 mrc p15, 3, r4, c15, c0, 3 /* L2CR1 is the L2 cache control reg 1 */
65#endif
66 mrc p15, 0, r5, c10, c2, 0 /* PRRR */
67 mrc p15, 0, r6, c10, c2, 1 /* NMRR */
68 mrc p15, 0, r7, c1, c0, 1 /* ACTLR */
69 mrc p15, 0, r8, c2, c0, 1 /* TTBR1 */
70 mrc p15, 0, r9, c13, c0, 3 /* TPIDRURO */
71 mrc p15, 0, ip, c13, c0, 1 /* context ID */
72 stmia r0!, {r1-r9, ip}
73#ifdef CONFIG_MSM_CPU_AVS
74 mrc p15, 7, r1, c15, c1, 7 /* AVSCSR is the Adaptive Voltage Scaling
75 * Control and Status Register */
76 mrc p15, 7, r2, c15, c0, 6 /* AVSDSCR is the Adaptive Voltage
77 * Scaling Delay Synthesizer Control
78 * Register */
79#ifndef CONFIG_ARCH_MSM_KRAIT
80 mrc p15, 7, r3, c15, c1, 0 /* TSCSR is the Temperature Status and
81 * Control Register
82 */
83#endif
84
85 stmia r0!, {r1-r3}
86#endif
87
88#ifdef CONFIG_MSM_JTAG_V7
89 bl msm_save_jtag_debug
90#endif
Pratik Patel7831c082011-06-08 21:44:37 -070091#ifdef CONFIG_MSM_TRACE_ACROSS_PC
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070092 bl etm_save_reg_check
93#endif
Maheshkumar Sivasubramaniana012e092011-08-18 10:13:03 -060094
95 ldr r0, =msm_pm_flush_l2_flag
96 ldr r0, [r0]
97 mov r1, #0
98 mcr p15, 2, r1, c0, c0, 0 /*CCSELR*/
99 mrc p15, 1, r1, c0, c0, 0 /*CCSIDR*/
100 mov r2, #1
101 and r1, r2, r1, ASR #30 /* Check if the cache is write back */
102 orr r1, r0, r1
103 cmp r1, #1
104 bne skip
105 bl v7_flush_dcache_all
106
107skip: ldr r0, =saved_state
108 ldr r1, =saved_state_end
109 sub r1, r1, r0
110 bl v7_flush_kern_dcache_area
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700111
Maheshkumar Sivasubramanianfa1d0dd2011-07-26 16:02:55 -0600112 mrc p15, 0, r4, c1, c0, 0 /* read current CR */
113 bic r0, r4, #(1 << 2) /* clear dcache bit */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700114 bic r0, r0, #(1 << 12) /* clear icache bit */
115 mcr p15, 0, r0, c1, c0, 0 /* disable d/i cache */
116
117 dsb
Maheshkumar Sivasubramanianfa1d0dd2011-07-26 16:02:55 -0600118#ifdef CONFIG_ARCH_MSM_KRAIT
119 ldr r0, =SCM_SVC_BOOT
120 ldr r1, =SCM_CMD_TERMINATE_PC
121 ldr r2, =0
122 bl scm_call_atomic1
123#else
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700124 wfi
Maheshkumar Sivasubramanianfa1d0dd2011-07-26 16:02:55 -0600125#endif
126 mcr p15, 0, r4, c1, c0, 0 /* restore d/i cache */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700127 isb
128
129#if defined(CONFIG_MSM_FIQ_SUPPORT)
130 cpsie f
131#endif
Pratik Patel7831c082011-06-08 21:44:37 -0700132#ifdef CONFIG_MSM_TRACE_ACROSS_PC
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700133 bl etm_restore_reg_check
134#endif
135#ifdef CONFIG_MSM_JTAG_V7
136 bl msm_restore_jtag_debug
137#endif
138 ldr r0, =saved_state /* restore registers */
139#if (NR_CPUS >= 2)
140 mrc p15, 0, r1, c0, c0, 5 /* MPIDR */
141 ands r1, r1, #15 /* What CPU am I */
142 addne r0, r0, #CPU_SAVED_STATE_SIZE
143#endif
144
145 ldmfd r0, {r4-r14}
146 mov r0, #0 /* return power collapse failed */
147 bx lr
148
149ENTRY(msm_pm_collapse_exit)
150#if 0 /* serial debug */
151 mov r0, #0x80000016
152 mcr p15, 0, r0, c15, c2, 4
153 mov r0, #0xA9000000
154 add r0, r0, #0x00A00000 /* UART1 */
155 /*add r0, r0, #0x00C00000*/ /* UART3 */
156 mov r1, #'A'
157 str r1, [r0, #0x00C]
158#endif
159 ldr r1, =saved_state_end
160 ldr r2, =msm_pm_collapse_exit
161 adr r3, msm_pm_collapse_exit
162 add r1, r1, r3
163 sub r1, r1, r2
164#if (NR_CPUS >= 2)
165 mrc p15, 0, r2, c0, c0, 5 /* MPIDR */
166 ands r2, r2, #15 /* What CPU am I */
167 subeq r1, r1, #CPU_SAVED_STATE_SIZE
168#endif
169
170#ifdef CONFIG_MSM_CPU_AVS
171 ldmdb r1!, {r2-r4}
172#ifndef CONFIG_ARCH_MSM_KRAIT
173 mcr p15, 7, r4, c15, c1, 0 /* TSCSR */
174#endif
175 mcr p15, 7, r3, c15, c0, 6 /* AVSDSCR */
176 mcr p15, 7, r2, c15, c1, 7 /* AVSCSR */
177#endif
178 ldmdb r1!, {r2-r11}
179 mcr p15, 0, r4, c3, c0, 0 /* dacr */
180 mcr p15, 0, r3, c2, c0, 0 /* TTBR0 */
181#ifdef CONFIG_ARCH_MSM_SCORPION
182 /* This instruction is not valid for non scorpion processors */
183 mcr p15, 3, r5, c15, c0, 3 /* L2CR1 */
184#endif
185 mcr p15, 0, r6, c10, c2, 0 /* PRRR */
186 mcr p15, 0, r7, c10, c2, 1 /* NMRR */
187 mcr p15, 0, r8, c1, c0, 1 /* ACTLR */
188 mcr p15, 0, r9, c2, c0, 1 /* TTBR1 */
189 mcr p15, 0, r10, c13, c0, 3 /* TPIDRURO */
190 mcr p15, 0, r11, c13, c0, 1 /* context ID */
191 isb
192 ldmdb r1!, {r4-r14}
193 ldr r0, =msm_pm_pc_pgd
194 ldr r1, =msm_pm_collapse_exit
195 adr r3, msm_pm_collapse_exit
196 add r0, r0, r3
197 sub r0, r0, r1
198 ldr r0, [r0]
199 mrc p15, 0, r1, c2, c0, 0 /* save current TTBR0 */
200 and r3, r1, #0x7f /* mask to get TTB flags */
201 orr r0, r0, r3 /* add TTB flags to switch TTBR value */
202 mcr p15, 0, r0, c2, c0, 0 /* temporary switch TTBR0 */
203 isb
204 mcr p15, 0, r2, c1, c0, 0 /* MMU control */
205 isb
206msm_pm_mapped_pa:
207 /* Switch to virtual */
208 ldr r0, =msm_pm_pa_to_va
209 mov pc, r0
210msm_pm_pa_to_va:
211 mcr p15, 0, r1, c2, c0, 0 /* restore TTBR0 */
212 isb
213 mcr p15, 0, r3, c8, c7, 0 /* UTLBIALL */
214 mcr p15, 0, r3, c7, c5, 6 /* BPIALL */
215 dsb
216
217 isb
218 stmfd sp!, {lr}
219 bl v7_flush_kern_cache_all
Pratik Patel7831c082011-06-08 21:44:37 -0700220#ifdef CONFIG_MSM_TRACE_ACROSS_PC
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700221 bl etm_restore_reg_check
222#endif
223#ifdef CONFIG_MSM_JTAG_V7
224 bl msm_restore_jtag_debug
225#endif
226 ldmfd sp!, {lr}
227 mov r0, #1
228 bx lr
229 nop
230 nop
231 nop
232 nop
233 nop
2341: b 1b
235
236ENTRY(msm_pm_boot_entry)
237 mrc p15, 0, r0, c0, c0, 5 /* MPIDR */
238 and r0, r0, #15 /* what CPU am I */
239
240 ldr r1, =msm_pm_boot_vector
241 ldr r2, =msm_pm_boot_entry
242 adr r3, msm_pm_boot_entry
243 add r1, r1, r3 /* translate virt to phys addr */
244 sub r1, r1, r2
245
246 add r1, r1, r0, LSL #2 /* locate boot vector for our cpu */
247 ldr pc, [r1] /* jump */
248
249ENTRY(msm_pm_write_boot_vector)
250 ldr r2, =msm_pm_boot_vector
251 add r2, r2, r0, LSL #2 /* locate boot vector for our cpu */
252 str r1, [r2]
Maheshkumar Sivasubramaniana012e092011-08-18 10:13:03 -0600253 mov r0, r2
254 ldr r1, =4
255 stmfd sp!, {lr}
256 bl v7_flush_kern_dcache_area
257 ldmfd sp!, {lr}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700258 bx lr
259
Maheshkumar Sivasubramaniana012e092011-08-18 10:13:03 -0600260ENTRY(msm_pm_set_l2_flush_flag)
261 ldr r1, =msm_pm_flush_l2_flag
262 str r0, [r1]
263 bx lr
264
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700265 .data
266
267 .globl msm_pm_pc_pgd
268msm_pm_pc_pgd:
269 .long 0x0
270
271saved_state:
272#if (NR_CPUS >= 2)
273 .space CPU_SAVED_STATE_SIZE * 2 /* This code only supports 2 cores */
274#else
275 .space CPU_SAVED_STATE_SIZE
276#endif
277saved_state_end:
278
279msm_pm_boot_vector:
280 .space 4 * NR_CPUS
Maheshkumar Sivasubramaniana012e092011-08-18 10:13:03 -0600281
282/*
283 * Default the l2 flush flag to 1 so that caches are flushed during power
284 * collapse unless the L2 driver decides to flush them only during L2
285 * Power collapse.
286 */
287msm_pm_flush_l2_flag:
288 .long 0x1