aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--arch/sh/include/asm/suspend.h27
-rw-r--r--arch/sh/kernel/asm-offsets.c10
-rw-r--r--arch/sh/kernel/cpu/shmobile/pm.c54
-rw-r--r--arch/sh/kernel/cpu/shmobile/sleep.S244
4 files changed, 171 insertions, 164 deletions
diff --git a/arch/sh/include/asm/suspend.h b/arch/sh/include/asm/suspend.h
index fab58cc2ecd9..8e2c55dc5fe6 100644
--- a/arch/sh/include/asm/suspend.h
+++ b/arch/sh/include/asm/suspend.h
@@ -34,6 +34,33 @@ extern struct atomic_notifier_head sh_mobile_post_sleep_notifier_list;
34void sh_mobile_register_self_refresh(unsigned long flags, 34void sh_mobile_register_self_refresh(unsigned long flags,
35 void *pre_start, void *pre_end, 35 void *pre_start, void *pre_end,
36 void *post_start, void *post_end); 36 void *post_start, void *post_end);
37
38/* register structure for address/data information */
39struct sh_sleep_regs {
40 unsigned long stbcr;
41};
42
43/* data area for low-level sleep code */
44struct sh_sleep_data {
45 /* current sleep mode (SUSP_SH_...) */
46 unsigned long mode;
47
48 /* addresses of board specific self-refresh snippets */
49 unsigned long sf_pre;
50 unsigned long sf_post;
51
52 /* register state saved and restored by the assembly code */
53 unsigned long vbr;
54 unsigned long spc;
55 unsigned long sr;
56
57 /* structure for keeping register addresses */
58 struct sh_sleep_regs addr;
59
60 /* structure for saving/restoring register state */
61 struct sh_sleep_regs data;
62};
63
37#endif 64#endif
38 65
39/* flags passed to assembly suspend code */ 66/* flags passed to assembly suspend code */
diff --git a/arch/sh/kernel/asm-offsets.c b/arch/sh/kernel/asm-offsets.c
index d218e808294e..9bdeff962e1c 100644
--- a/arch/sh/kernel/asm-offsets.c
+++ b/arch/sh/kernel/asm-offsets.c
@@ -34,5 +34,15 @@ int main(void)
34 DEFINE(PBE_NEXT, offsetof(struct pbe, next)); 34 DEFINE(PBE_NEXT, offsetof(struct pbe, next));
35 DEFINE(SWSUSP_ARCH_REGS_SIZE, sizeof(struct swsusp_arch_regs)); 35 DEFINE(SWSUSP_ARCH_REGS_SIZE, sizeof(struct swsusp_arch_regs));
36#endif 36#endif
37
38 DEFINE(SH_SLEEP_MODE, offsetof(struct sh_sleep_data, mode));
39 DEFINE(SH_SLEEP_SF_PRE, offsetof(struct sh_sleep_data, sf_pre));
40 DEFINE(SH_SLEEP_SF_POST, offsetof(struct sh_sleep_data, sf_post));
41 DEFINE(SH_SLEEP_VBR, offsetof(struct sh_sleep_data, vbr));
42 DEFINE(SH_SLEEP_SPC, offsetof(struct sh_sleep_data, spc));
43 DEFINE(SH_SLEEP_SR, offsetof(struct sh_sleep_data, sr));
44 DEFINE(SH_SLEEP_BASE_ADDR, offsetof(struct sh_sleep_data, addr));
45 DEFINE(SH_SLEEP_BASE_DATA, offsetof(struct sh_sleep_data, data));
46 DEFINE(SH_SLEEP_REG_STBCR, offsetof(struct sh_sleep_regs, stbcr));
37 return 0; 47 return 0;
38} 48}
diff --git a/arch/sh/kernel/cpu/shmobile/pm.c b/arch/sh/kernel/cpu/shmobile/pm.c
index b424747e4252..cb3d28f2968c 100644
--- a/arch/sh/kernel/cpu/shmobile/pm.c
+++ b/arch/sh/kernel/cpu/shmobile/pm.c
@@ -42,13 +42,14 @@ ATOMIC_NOTIFIER_HEAD(sh_mobile_post_sleep_notifier_list);
42 42
43#define ILRAM_BASE 0xe5200000 43#define ILRAM_BASE 0xe5200000
44 44
45extern const unsigned char sh_mobile_standby[];
46extern const unsigned int sh_mobile_standby_size;
47
48void sh_mobile_call_standby(unsigned long mode) 45void sh_mobile_call_standby(unsigned long mode)
49{ 46{
50 void *onchip_mem = (void *)ILRAM_BASE; 47 void *onchip_mem = (void *)ILRAM_BASE;
51 void (*standby_onchip_mem)(unsigned long, unsigned long) = onchip_mem; 48 struct sh_sleep_data *sdp = onchip_mem;
49 void (*standby_onchip_mem)(unsigned long, unsigned long);
50
51 /* code located directly after data structure */
52 standby_onchip_mem = (void *)(sdp + 1);
52 53
53 atomic_notifier_call_chain(&sh_mobile_pre_sleep_notifier_list, 54 atomic_notifier_call_chain(&sh_mobile_pre_sleep_notifier_list,
54 mode, NULL); 55 mode, NULL);
@@ -60,10 +61,48 @@ void sh_mobile_call_standby(unsigned long mode)
60 mode, NULL); 61 mode, NULL);
61} 62}
62 63
64extern char sh_mobile_sleep_enter_start;
65extern char sh_mobile_sleep_enter_end;
66
67extern char sh_mobile_sleep_resume_start;
68extern char sh_mobile_sleep_resume_end;
69
63void sh_mobile_register_self_refresh(unsigned long flags, 70void sh_mobile_register_self_refresh(unsigned long flags,
64 void *pre_start, void *pre_end, 71 void *pre_start, void *pre_end,
65 void *post_start, void *post_end) 72 void *post_start, void *post_end)
66{ 73{
74 void *onchip_mem = (void *)ILRAM_BASE;
75 void *vp;
76 struct sh_sleep_data *sdp;
77 int n;
78
79 /* part 0: data area */
80 sdp = onchip_mem;
81 sdp->addr.stbcr = 0xa4150020; /* STBCR */
82 vp = sdp + 1;
83
84 /* part 1: common code to enter sleep mode */
85 n = &sh_mobile_sleep_enter_end - &sh_mobile_sleep_enter_start;
86 memcpy(vp, &sh_mobile_sleep_enter_start, n);
87 vp += roundup(n, 4);
88
89 /* part 2: board specific code to enter self-refresh mode */
90 n = pre_end - pre_start;
91 memcpy(vp, pre_start, n);
92 sdp->sf_pre = (unsigned long)vp;
93 vp += roundup(n, 4);
94
95 /* part 3: board specific code to resume from self-refresh mode */
96 n = post_end - post_start;
97 memcpy(vp, post_start, n);
98 sdp->sf_post = (unsigned long)vp;
99 vp += roundup(n, 4);
100
101 /* part 4: common code to resume from sleep mode */
102 WARN_ON(vp > (onchip_mem + 0x600));
103 vp = onchip_mem + 0x600; /* located at interrupt vector */
104 n = &sh_mobile_sleep_resume_end - &sh_mobile_sleep_resume_start;
105 memcpy(vp, &sh_mobile_sleep_resume_start, n);
67} 106}
68 107
69static int sh_pm_enter(suspend_state_t state) 108static int sh_pm_enter(suspend_state_t state)
@@ -83,13 +122,6 @@ static struct platform_suspend_ops sh_pm_ops = {
83 122
84static int __init sh_pm_init(void) 123static int __init sh_pm_init(void)
85{ 124{
86 void *onchip_mem = (void *)ILRAM_BASE;
87
88 /* Copy the assembly snippet to the otherwise ununsed ILRAM */
89 memcpy(onchip_mem, sh_mobile_standby, sh_mobile_standby_size);
90 wmb();
91 ctrl_barrier();
92
93 suspend_set_ops(&sh_pm_ops); 125 suspend_set_ops(&sh_pm_ops);
94 sh_mobile_setup_cpuidle(); 126 sh_mobile_setup_cpuidle();
95 return 0; 127 return 0;
diff --git a/arch/sh/kernel/cpu/shmobile/sleep.S b/arch/sh/kernel/cpu/shmobile/sleep.S
index a439e6c7824f..d3221d9b88be 100644
--- a/arch/sh/kernel/cpu/shmobile/sleep.S
+++ b/arch/sh/kernel/cpu/shmobile/sleep.S
@@ -20,79 +20,49 @@
20 * Kernel mode register usage, see entry.S: 20 * Kernel mode register usage, see entry.S:
21 * k0 scratch 21 * k0 scratch
22 * k1 scratch 22 * k1 scratch
23 * k4 scratch
24 */ 23 */
25#define k0 r0 24#define k0 r0
26#define k1 r1 25#define k1 r1
27#define k4 r4
28 26
29/* manage self-refresh and enter standby mode. 27/* manage self-refresh and enter standby mode. must be self-contained.
30 * this code will be copied to on-chip memory and executed from there. 28 * this code will be copied to on-chip memory and executed from there.
31 */ 29 */
30 .balign 4
31ENTRY(sh_mobile_sleep_enter_start)
32 32
33 .balign 4096,0,4096 33 /* save mode flags */
34ENTRY(sh_mobile_standby) 34 mov.l r4, @(SH_SLEEP_MODE, r5)
35 35
36 /* save original vbr */ 36 /* save original vbr */
37 stc vbr, r1 37 stc vbr, r0
38 mova saved_vbr, r0 38 mov.l r0, @(SH_SLEEP_VBR, r5)
39 mov.l r1, @r0
40 39
41 /* point vbr to our on-chip memory page */ 40 /* point vbr to our on-chip memory page */
42 ldc r5, vbr 41 ldc r5, vbr
43 42
44 /* save return address */ 43 /* save return address */
45 mova saved_spc, r0 44 sts pr, r0
46 sts pr, r5 45 mov.l r0, @(SH_SLEEP_SPC, r5)
47 mov.l r5, @r0
48 46
49 /* save sr */ 47 /* save sr */
50 mova saved_sr, r0 48 stc sr, r0
51 stc sr, r5 49 mov.l r0, @(SH_SLEEP_SR, r5)
52 mov.l r5, @r0
53 50
54 /* save mode flags */ 51 /* save stbcr */
55 mova saved_mode, r0 52 bsr save_register
56 mov.l r4, @r0 53 mov #SH_SLEEP_REG_STBCR, r0
57
58 /* put mode flags in r0 */
59 mov r4, r0
60 54
55 /* call self-refresh entering code if needed */
56 mov.l @(SH_SLEEP_MODE, r5), r0
61 tst #SUSP_SH_SF, r0 57 tst #SUSP_SH_SF, r0
62 bt skip_set_sf 58 bt skip_set_sf
63#ifdef CONFIG_CPU_SUBTYPE_SH7724 59
64 /* DBSC: put memory in self-refresh mode */ 60 mov.l @(SH_SLEEP_SF_PRE, r5), r0
65 mov.l dben_reg, r4 61 jsr @r0
66 mov.l dben_data0, r1 62 nop
67 mov.l r1, @r4
68
69 mov.l dbrfpdn0_reg, r4
70 mov.l dbrfpdn0_data0, r1
71 mov.l r1, @r4
72
73 mov.l dbcmdcnt_reg, r4
74 mov.l dbcmdcnt_data0, r1
75 mov.l r1, @r4
76
77 mov.l dbcmdcnt_reg, r4
78 mov.l dbcmdcnt_data1, r1
79 mov.l r1, @r4
80
81 mov.l dbrfpdn0_reg, r4
82 mov.l dbrfpdn0_data1, r1
83 mov.l r1, @r4
84#else
85 /* SBSC: disable power down and put in self-refresh mode */
86 mov.l 1f, r4
87 mov.l 2f, r1
88 mov.l @r4, r2
89 or r1, r2
90 mov.l 3f, r3
91 and r3, r2
92 mov.l r2, @r4
93#endif
94 63
95skip_set_sf: 64skip_set_sf:
65 mov.l @(SH_SLEEP_MODE, r5), r0
96 tst #SUSP_SH_STANDBY, r0 66 tst #SUSP_SH_STANDBY, r0
97 bt test_rstandby 67 bt test_rstandby
98 68
@@ -123,124 +93,92 @@ force_sleep:
123 93
124do_sleep: 94do_sleep:
125 /* setup and enter selected standby mode */ 95 /* setup and enter selected standby mode */
126 mov.l 5f, r4 96 bsr get_register
127 mov.l r1, @r4 97 mov #SH_SLEEP_REG_STBCR, r0
98 mov.l r1, @r0
128again: 99again:
129 sleep 100 sleep
130 bra again 101 bra again
131 nop 102 nop
132 103
133restore_jump_vbr: 104save_register:
105 add #SH_SLEEP_BASE_ADDR, r0
106 mov.l @(r0, r5), r1
107 add #-SH_SLEEP_BASE_ADDR, r0
108 mov.l @r1, r1
109 add #SH_SLEEP_BASE_DATA, r0
110 mov.l r1, @(r0, r5)
111 add #-SH_SLEEP_BASE_DATA, r0
112 rts
113 nop
114
115get_register:
116 add #SH_SLEEP_BASE_ADDR, r0
117 mov.l @(r0, r5), r0
118 rts
119 nop
120ENTRY(sh_mobile_sleep_enter_end)
121
122 .balign 4
123ENTRY(sh_mobile_sleep_resume_start)
124
125 /* figure out start address */
126 bsr 0f
127 nop
1280:
129 sts pr, k1
130 mov.l 1f, k0
131 and k0, k1
132
133 /* store pointer to data area in VBR */
134 ldc k1, vbr
135
136 /* setup sr with saved sr */
137 mov.l @(SH_SLEEP_SR, k1), k0
138 ldc k0, sr
139
140 /* now: user register set! */
141 stc vbr, r5
142
134 /* setup spc with return address to c code */ 143 /* setup spc with return address to c code */
135 mov.l saved_spc, k0 144 mov.l @(SH_SLEEP_SPC, r5), r0
136 ldc k0, spc 145 ldc r0, spc
137 146
138 /* restore vbr */ 147 /* restore vbr */
139 mov.l saved_vbr, k0 148 mov.l @(SH_SLEEP_VBR, r5), r0
140 ldc k0, vbr 149 ldc r0, vbr
141 150
142 /* setup ssr with saved sr */ 151 /* setup ssr with saved sr */
143 mov.l saved_sr, k0 152 mov.l @(SH_SLEEP_SR, r5), r0
144 ldc k0, ssr 153 ldc r0, ssr
145
146 /* get mode flags */
147 mov.l saved_mode, k0
148 154
149done_sleep: 155 /* restore sleep mode register */
150 /* reset standby mode to sleep mode */ 156 bsr restore_register
151 mov.l 5f, k4 157 mov #SH_SLEEP_REG_STBCR, r0
152 mov #0x00, k1
153 mov.l k1, @k4
154 158
155 tst #SUSP_SH_SF, k0 159 /* call self-refresh resume code if needed */
160 mov.l @(SH_SLEEP_MODE, r5), r0
161 tst #SUSP_SH_SF, r0
156 bt skip_restore_sf 162 bt skip_restore_sf
157 163
158#ifdef CONFIG_CPU_SUBTYPE_SH7724 164 mov.l @(SH_SLEEP_SF_POST, r5), r0
159 /* DBSC: put memory in auto-refresh mode */ 165 jsr @r0
160 mov.l dbrfpdn0_reg, k4 166 nop
161 mov.l dbrfpdn0_data0, k1 167
162 mov.l k1, @k4
163
164 nop /* sleep 140 ns */
165 nop
166 nop
167 nop
168
169 mov.l dbcmdcnt_reg, k4
170 mov.l dbcmdcnt_data0, k1
171 mov.l k1, @k4
172
173 mov.l dbcmdcnt_reg, k4
174 mov.l dbcmdcnt_data1, k1
175 mov.l k1, @k4
176
177 mov.l dben_reg, k4
178 mov.l dben_data1, k1
179 mov.l k1, @k4
180
181 mov.l dbrfpdn0_reg, k4
182 mov.l dbrfpdn0_data2, k1
183 mov.l k1, @k4
184#else
185 /* SBSC: set auto-refresh mode */
186 mov.l 1f, k4
187 mov.l @k4, k0
188 mov.l 4f, k1
189 and k1, k0
190 mov.l k0, @k4
191 mov.l 6f, k4
192 mov.l 8f, k0
193 mov.l @k4, k1
194 mov #-1, k4
195 add k4, k1
196 or k1, k0
197 mov.l 7f, k1
198 mov.l k0, @k1
199#endif
200skip_restore_sf: 168skip_restore_sf:
201 /* jump to vbr vector */ 169 rte
202 mov.l saved_vbr, k0
203 mov.l offset_vbr, k4
204 add k4, k0
205 jmp @k0
206 nop 170 nop
207 171
208 .balign 4 172restore_register:
209saved_mode: .long 0 173 add #SH_SLEEP_BASE_DATA, r0
210saved_spc: .long 0 174 mov.l @(r0, r5), r1
211saved_sr: .long 0 175 add #-SH_SLEEP_BASE_DATA, r0
212saved_vbr: .long 0 176 add #SH_SLEEP_BASE_ADDR, r0
213offset_vbr: .long 0x600 177 mov.l @(r0, r5), r0
214#ifdef CONFIG_CPU_SUBTYPE_SH7724 178 mov.l r1, @r0
215dben_reg: .long 0xfd000010 /* DBEN */ 179 rts
216dben_data0: .long 0
217dben_data1: .long 1
218dbrfpdn0_reg: .long 0xfd000040 /* DBRFPDN0 */
219dbrfpdn0_data0: .long 0
220dbrfpdn0_data1: .long 1
221dbrfpdn0_data2: .long 0x00010000
222dbcmdcnt_reg: .long 0xfd000014 /* DBCMDCNT */
223dbcmdcnt_data0: .long 2
224dbcmdcnt_data1: .long 4
225#else
2261: .long 0xfe400008 /* SDCR0 */
2272: .long 0x00000400
2283: .long 0xffff7fff
2294: .long 0xfffffbff
230#endif
2315: .long 0xa4150020 /* STBCR */
2326: .long 0xfe40001c /* RTCOR */
2337: .long 0xfe400018 /* RTCNT */
2348: .long 0xa55a0000
235
236
237/* interrupt vector @ 0x600 */
238 .balign 0x400,0,0x400
239 .long 0xdeadbeef
240 .balign 0x200,0,0x200
241 bra restore_jump_vbr
242 nop 180 nop
243sh_mobile_standby_end:
244 181
245ENTRY(sh_mobile_standby_size) 182 .balign 4
246 .long sh_mobile_standby_end - sh_mobile_standby 1831: .long ~0x7ff
184ENTRY(sh_mobile_sleep_resume_end)