]> git.proxmox.com Git - mirror_ubuntu-bionic-kernel.git/blob - arch/blackfin/mach-bf609/pm.c
79cdf41096c296667d67dfc8ea690e3344d8025c
[mirror_ubuntu-bionic-kernel.git] / arch / blackfin / mach-bf609 / pm.c
1 /*
2 * Blackfin bf609 power management
3 *
4 * Copyright 2011 Analog Devices Inc.
5 *
6 * Licensed under the GPL-2
7 */
8
9 #include <linux/suspend.h>
10 #include <linux/io.h>
11 #include <linux/interrupt.h>
12 #include <linux/gpio.h>
13 #include <linux/irq.h>
14
15 #include <linux/delay.h>
16
17 #include <asm/dpmc.h>
18 #include <asm/pm.h>
19 #include <mach/pm.h>
20 #include <asm/blackfin.h>
21
22 /***********************************************************/
23 /* */
24 /* Wakeup Actions for DPM_RESTORE */
25 /* */
26 /***********************************************************/
27 #define BITP_ROM_WUA_CHKHDR 24
28 #define BITP_ROM_WUA_DDRLOCK 7
29 #define BITP_ROM_WUA_DDRDLLEN 6
30 #define BITP_ROM_WUA_DDR 5
31 #define BITP_ROM_WUA_CGU 4
32 #define BITP_ROM_WUA_MEMBOOT 2
33 #define BITP_ROM_WUA_EN 1
34
35 #define BITM_ROM_WUA_CHKHDR (0xFF000000)
36 #define ENUM_ROM_WUA_CHKHDR_AD 0xAD000000
37
38 #define BITM_ROM_WUA_DDRLOCK (0x00000080)
39 #define BITM_ROM_WUA_DDRDLLEN (0x00000040)
40 #define BITM_ROM_WUA_DDR (0x00000020)
41 #define BITM_ROM_WUA_CGU (0x00000010)
42 #define BITM_ROM_WUA_MEMBOOT (0x00000002)
43 #define BITM_ROM_WUA_EN (0x00000001)
44
45 /***********************************************************/
46 /* */
47 /* Syscontrol */
48 /* */
49 /***********************************************************/
50 #define BITP_ROM_SYSCTRL_CGU_LOCKINGEN 28 /* unlocks CGU_CTL register */
51 #define BITP_ROM_SYSCTRL_WUA_OVERRIDE 24
52 #define BITP_ROM_SYSCTRL_WUA_DDRDLLEN 20 /* Saves the DDR DLL and PADS registers to the DPM registers */
53 #define BITP_ROM_SYSCTRL_WUA_DDR 19 /* Saves the DDR registers to the DPM registers */
54 #define BITP_ROM_SYSCTRL_WUA_CGU 18 /* Saves the CGU registers into DPM registers */
55 #define BITP_ROM_SYSCTRL_WUA_DPMWRITE 17 /* Saves the Syscontrol structure structure contents into DPM registers */
56 #define BITP_ROM_SYSCTRL_WUA_EN 16 /* reads current PLL and DDR configuration into structure */
57 #define BITP_ROM_SYSCTRL_DDR_WRITE 13 /* writes the DDR registers from Syscontrol structure for wakeup initialization of DDR */
58 #define BITP_ROM_SYSCTRL_DDR_READ 12 /* Read the DDR registers into the Syscontrol structure for storing prior to hibernate */
59 #define BITP_ROM_SYSCTRL_CGU_AUTODIS 11 /* Disables auto handling of UPDT and ALGN fields */
60 #define BITP_ROM_SYSCTRL_CGU_CLKOUTSEL 7 /* access CGU_CLKOUTSEL register */
61 #define BITP_ROM_SYSCTRL_CGU_DIV 6 /* access CGU_DIV register */
62 #define BITP_ROM_SYSCTRL_CGU_STAT 5 /* access CGU_STAT register */
63 #define BITP_ROM_SYSCTRL_CGU_CTL 4 /* access CGU_CTL register */
64 #define BITP_ROM_SYSCTRL_CGU_RTNSTAT 2 /* Update structure STAT field upon error */
65 #define BITP_ROM_SYSCTRL_WRITE 1 /* write registers */
66 #define BITP_ROM_SYSCTRL_READ 0 /* read registers */
67
68 #define BITM_ROM_SYSCTRL_CGU_READ (0x00000001) /* Read CGU registers */
69 #define BITM_ROM_SYSCTRL_CGU_WRITE (0x00000002) /* Write registers */
70 #define BITM_ROM_SYSCTRL_CGU_RTNSTAT (0x00000004) /* Update structure STAT field upon error or after a write operation */
71 #define BITM_ROM_SYSCTRL_CGU_CTL (0x00000010) /* Access CGU_CTL register */
72 #define BITM_ROM_SYSCTRL_CGU_STAT (0x00000020) /* Access CGU_STAT register */
73 #define BITM_ROM_SYSCTRL_CGU_DIV (0x00000040) /* Access CGU_DIV register */
74 #define BITM_ROM_SYSCTRL_CGU_CLKOUTSEL (0x00000080) /* Access CGU_CLKOUTSEL register */
75 #define BITM_ROM_SYSCTRL_CGU_AUTODIS (0x00000800) /* Disables auto handling of UPDT and ALGN fields */
76 #define BITM_ROM_SYSCTRL_DDR_READ (0x00001000) /* Reads the contents of the DDR registers and stores them into the structure */
77 #define BITM_ROM_SYSCTRL_DDR_WRITE (0x00002000) /* Writes the DDR registers from the structure, only really intented for wakeup functionality and not for full DDR configuration */
78 #define BITM_ROM_SYSCTRL_WUA_EN (0x00010000) /* Wakeup entry or exit opertation enable */
79 #define BITM_ROM_SYSCTRL_WUA_DPMWRITE (0x00020000) /* When set indicates a restore of the PLL and DDR is to be performed otherwise a save is required */
80 #define BITM_ROM_SYSCTRL_WUA_CGU (0x00040000) /* Only applicable for a PLL and DDR save operation to the DPM, saves the current settings if cleared or the contents of the structure if set */
81 #define BITM_ROM_SYSCTRL_WUA_DDR (0x00080000) /* Only applicable for a PLL and DDR save operation to the DPM, saves the current settings if cleared or the contents of the structure if set */
82 #define BITM_ROM_SYSCTRL_WUA_DDRDLLEN (0x00100000) /* Enables saving/restoring of the DDR DLLCTL register */
83 #define BITM_ROM_SYSCTRL_WUA_OVERRIDE (0x01000000)
84 #define BITM_ROM_SYSCTRL_CGU_LOCKINGEN (0x10000000) /* Unlocks the CGU_CTL register */
85
86
87 /* Structures for the syscontrol() function */
88 struct STRUCT_ROM_SYSCTRL {
89 uint32_t ulCGU_CTL;
90 uint32_t ulCGU_STAT;
91 uint32_t ulCGU_DIV;
92 uint32_t ulCGU_CLKOUTSEL;
93 uint32_t ulWUA_Flags;
94 uint32_t ulWUA_BootAddr;
95 uint32_t ulWUA_User;
96 uint32_t ulDDR_CTL;
97 uint32_t ulDDR_CFG;
98 uint32_t ulDDR_TR0;
99 uint32_t ulDDR_TR1;
100 uint32_t ulDDR_TR2;
101 uint32_t ulDDR_MR;
102 uint32_t ulDDR_EMR1;
103 uint32_t ulDDR_EMR2;
104 uint32_t ulDDR_PADCTL;
105 uint32_t ulDDR_DLLCTL;
106 uint32_t ulReserved;
107 };
108
109 struct bfin_pm_data {
110 uint32_t magic;
111 uint32_t resume_addr;
112 uint32_t sp;
113 };
114
115 struct bfin_pm_data bf609_pm_data;
116
117 struct STRUCT_ROM_SYSCTRL configvalues;
118 uint32_t dactionflags;
119
120 #define FUNC_ROM_SYSCONTROL 0xC8000080
121 __attribute__((l1_data))
122 static uint32_t (* const bfrom_SysControl)(uint32_t action_flags, struct STRUCT_ROM_SYSCTRL *settings, void *reserved) = (void *)FUNC_ROM_SYSCONTROL;
123
124 __attribute__((l1_text))
125 void bfin_cpu_suspend(void)
126 {
127 __asm__ __volatile__( \
128 ".align 8;" \
129 "idle;" \
130 : : \
131 );
132 }
133
134 __attribute__((l1_text))
135 void bfin_deepsleep(unsigned long mask)
136 {
137 uint32_t dpm0_ctl;
138
139 bfin_write32(DPM0_WAKE_EN, 0x10);
140 bfin_write32(DPM0_WAKE_POL, 0x10);
141 dpm0_ctl = bfin_read32(DPM0_CTL);
142 dpm0_ctl = 0x00000008;
143 bfin_write32(DPM0_CTL, dpm0_ctl);
144 SSYNC();
145 __asm__ __volatile__( \
146 ".align 8;" \
147 "idle;" \
148 : : \
149 );
150 }
151
152 __attribute__((l1_text))
153 void bf609_ddr_sr(void)
154 {
155 uint32_t reg;
156
157 reg = bfin_read_DDR0_CTL();
158 reg |= 0x8;
159 bfin_write_DDR0_CTL(reg);
160
161 while (!(bfin_read_DDR0_STAT() & 0x8))
162 continue;
163 }
164
165 __attribute__((l1_text))
166 void bf609_ddr_sr_exit(void)
167 {
168 uint32_t reg;
169 while (!(bfin_read_DDR0_STAT() & 0x1))
170 continue;
171
172 reg = bfin_read_DDR0_CTL();
173 reg &= ~0x8;
174 bfin_write_DDR0_CTL(reg);
175
176 while ((bfin_read_DDR0_STAT() & 0x8))
177 continue;
178 }
179
180 __attribute__((l1_text))
181 void bfin_hibernate_syscontrol(void)
182 {
183 configvalues.ulWUA_Flags = (0xAD000000 | BITM_ROM_WUA_EN
184 | BITM_ROM_WUA_CGU | BITM_ROM_WUA_DDR | BITM_ROM_WUA_DDRDLLEN);
185
186 dactionflags = (BITM_ROM_SYSCTRL_WUA_EN
187 | BITM_ROM_SYSCTRL_WUA_DPMWRITE | BITM_ROM_SYSCTRL_WUA_CGU
188 | BITM_ROM_SYSCTRL_WUA_DDR | BITM_ROM_SYSCTRL_WUA_DDRDLLEN);
189
190 bfrom_SysControl(dactionflags, &configvalues, NULL);
191
192 bfin_write32(DPM0_RESTORE5, bfin_read32(DPM0_RESTORE5) | 4);
193 }
194
195 #ifndef CONFIG_BF60x
196 # define SIC_SYSIRQ(irq) (irq - (IRQ_CORETMR + 1))
197 #else
198 # define SIC_SYSIRQ(irq) ((irq) - IVG15)
199 #endif
200 void bfin_hibernate(unsigned long mask)
201 {
202 bfin_write32(DPM0_WAKE_EN, 0x10);
203 bfin_write32(DPM0_WAKE_POL, 0x10);
204 bfin_write32(DPM0_PGCNTR, 0x0000FFFF);
205 bfin_write32(DPM0_HIB_DIS, 0xFFFF);
206
207 printk(KERN_DEBUG "hibernate: restore %x pgcnt %x\n", bfin_read32(DPM0_RESTORE0), bfin_read32(DPM0_PGCNTR));
208
209 bf609_hibernate();
210 }
211
212 void bf609_cpu_pm_enter(suspend_state_t state)
213 {
214 int error;
215 unsigned long wakeup = 0;
216 unsigned long wakeup_pol = 0;
217
218 #ifdef CONFIG_PM_BFIN_WAKE_PA15
219 wakeup |= PA15WE;
220 # if CONFIG_PM_BFIN_WAKE_PA15_POL
221 wakeup_pol |= PA15WE;
222 # endif
223 #endif
224
225 #ifdef CONFIG_PM_BFIN_WAKE_PB15
226 wakeup |= PB15WE;
227 # if CONFIG_PM_BFIN_WAKE_PA15_POL
228 wakeup_pol |= PB15WE;
229 # endif
230 #endif
231
232 #ifdef CONFIG_PM_BFIN_WAKE_PC15
233 wakeup |= PC15WE;
234 # if CONFIG_PM_BFIN_WAKE_PC15_POL
235 wakeup_pol |= PC15WE;
236 # endif
237 #endif
238
239 #ifdef CONFIG_PM_BFIN_WAKE_PD06
240 wakeup |= PD06WE;
241 # if CONFIG_PM_BFIN_WAKE_PD06_POL
242 wakeup_pol |= PD06WE;
243 # endif
244 #endif
245
246 #ifdef CONFIG_PM_BFIN_WAKE_PE12
247 wakeup |= PE12WE;
248 # if CONFIG_PM_BFIN_WAKE_PE12_POL
249 wakeup_pol |= PE12WE;
250 # endif
251 #endif
252
253 #ifdef CONFIG_PM_BFIN_WAKE_PG04
254 wakeup |= PG04WE;
255 # if CONFIG_PM_BFIN_WAKE_PG04_POL
256 wakeup_pol |= PG04WE;
257 # endif
258 #endif
259
260 #ifdef CONFIG_PM_BFIN_WAKE_PG13
261 wakeup |= PG13WE;
262 # if CONFIG_PM_BFIN_WAKE_PG13_POL
263 wakeup_pol |= PG13WE;
264 # endif
265 #endif
266
267 #ifdef CONFIG_PM_BFIN_WAKE_USB
268 wakeup |= USBWE;
269 # if CONFIG_PM_BFIN_WAKE_USB_POL
270 wakeup_pol |= USBWE;
271 # endif
272 #endif
273
274 error = irq_set_irq_wake(255, 1);
275 if(error < 0)
276 printk(KERN_DEBUG "Unable to get irq wake\n");
277 error = irq_set_irq_wake(231, 1);
278 if (error < 0)
279 printk(KERN_DEBUG "Unable to get irq wake\n");
280
281 if (state == PM_SUSPEND_STANDBY)
282 bfin_deepsleep(wakeup);
283 else {
284 bfin_hibernate(wakeup);
285 }
286 }
287
288 int bf609_cpu_pm_prepare(void)
289 {
290 return 0;
291 }
292
293 void bf609_cpu_pm_finish(void)
294 {
295
296 }
297
298 static struct bfin_cpu_pm_fns bf609_cpu_pm = {
299 .enter = bf609_cpu_pm_enter,
300 .prepare = bf609_cpu_pm_prepare,
301 .finish = bf609_cpu_pm_finish,
302 };
303
304 static irqreturn_t test_isr(int irq, void *dev_id)
305 {
306 printk(KERN_DEBUG "gpio irq %d\n", irq);
307 return IRQ_HANDLED;
308 }
309
310 static irqreturn_t dpm0_isr(int irq, void *dev_id)
311 {
312 uint32_t wake_stat;
313
314 wake_stat = bfin_read32(DPM0_WAKE_STAT);
315 printk(KERN_DEBUG "enter %s wake stat %08x\n", __func__, wake_stat);
316
317 bfin_write32(DPM0_WAKE_STAT, wake_stat);
318 return IRQ_HANDLED;
319 }
320
321 static int __init bf609_init_pm(void)
322 {
323 int irq;
324 int error;
325
326 #if CONFIG_PM_BFIN_WAKE_PE12
327 irq = gpio_to_irq(GPIO_PE12);
328 if (irq < 0) {
329 error = irq;
330 printk(KERN_DEBUG "Unable to get irq number for GPIO %d, error %d\n",
331 GPIO_PE12, error);
332 }
333
334 error = request_irq(irq, test_isr, IRQF_TRIGGER_RISING | IRQF_NO_SUSPEND, "gpiope12", NULL);
335 if(error < 0)
336 printk(KERN_DEBUG "Unable to get irq\n");
337 #endif
338
339 error = request_irq(IRQ_CGU_EVT, dpm0_isr, IRQF_NO_SUSPEND, "cgu0 event", NULL);
340 if(error < 0)
341 printk(KERN_DEBUG "Unable to get irq\n");
342
343 error = request_irq(IRQ_DPM, dpm0_isr, IRQF_NO_SUSPEND, "dpm0 event", NULL);
344 if (error < 0)
345 printk(KERN_DEBUG "Unable to get irq\n");
346
347 bfin_cpu_pm = &bf609_cpu_pm;
348 return 0;
349 }
350
351 late_initcall(bf609_init_pm);