slub page alloc fallback: Enable interrupts for GFP_WAIT.
[wrt350n-kernel.git] / include / asm-mn10300 / system.h
blob8214fb7e7fe4747a5cd6962201efeefa2d00e094
1 /* MN10300 System definitions
3 * Copyright (C) 2007 Red Hat, Inc. All Rights Reserved.
4 * Written by David Howells (dhowells@redhat.com)
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public Licence
8 * as published by the Free Software Foundation; either version
9 * 2 of the Licence, or (at your option) any later version.
11 #ifndef _ASM_SYSTEM_H
12 #define _ASM_SYSTEM_H
14 #include <asm/cpu-regs.h>
16 #ifdef __KERNEL__
17 #ifndef __ASSEMBLY__
19 #include <linux/kernel.h>
21 struct task_struct;
22 struct thread_struct;
24 extern asmlinkage
25 struct task_struct *__switch_to(struct thread_struct *prev,
26 struct thread_struct *next,
27 struct task_struct *prev_task);
29 /* context switching is now performed out-of-line in switch_to.S */
30 #define switch_to(prev, next, last) \
31 do { \
32 current->thread.wchan = (u_long) __builtin_return_address(0); \
33 (last) = __switch_to(&(prev)->thread, &(next)->thread, (prev)); \
34 mb(); \
35 current->thread.wchan = 0; \
36 } while (0)
38 #define arch_align_stack(x) (x)
40 #define nop() asm volatile ("nop")
42 #endif /* !__ASSEMBLY__ */
45 * Force strict CPU ordering.
46 * And yes, this is required on UP too when we're talking
47 * to devices.
49 * For now, "wmb()" doesn't actually do anything, as all
50 * Intel CPU's follow what Intel calls a *Processor Order*,
51 * in which all writes are seen in the program order even
52 * outside the CPU.
54 * I expect future Intel CPU's to have a weaker ordering,
55 * but I'd also expect them to finally get their act together
56 * and add some real memory barriers if so.
58 * Some non intel clones support out of order store. wmb() ceases to be a
59 * nop for these.
62 #define mb() asm volatile ("": : :"memory")
63 #define rmb() mb()
64 #define wmb() asm volatile ("": : :"memory")
66 #ifdef CONFIG_SMP
67 #define smp_mb() mb()
68 #define smp_rmb() rmb()
69 #define smp_wmb() wmb()
70 #else
71 #define smp_mb() barrier()
72 #define smp_rmb() barrier()
73 #define smp_wmb() barrier()
74 #endif
76 #define set_mb(var, value) do { var = value; mb(); } while (0)
77 #define set_wmb(var, value) do { var = value; wmb(); } while (0)
79 #define read_barrier_depends() do {} while (0)
80 #define smp_read_barrier_depends() do {} while (0)
82 /*****************************************************************************/
84 * interrupt control
85 * - "disabled": run in IM1/2
86 * - level 0 - GDB stub
87 * - level 1 - virtual serial DMA (if present)
88 * - level 5 - normal interrupt priority
89 * - level 6 - timer interrupt
90 * - "enabled": run in IM7
92 #ifdef CONFIG_MN10300_TTYSM
93 #define MN10300_CLI_LEVEL EPSW_IM_2
94 #else
95 #define MN10300_CLI_LEVEL EPSW_IM_1
96 #endif
98 #define local_save_flags(x) \
99 do { \
100 typecheck(unsigned long, x); \
101 asm volatile( \
102 " mov epsw,%0 \n" \
103 : "=d"(x) \
104 ); \
105 } while (0)
107 #define local_irq_disable() \
108 do { \
109 asm volatile( \
110 " and %0,epsw \n" \
111 " or %1,epsw \n" \
112 " nop \n" \
113 " nop \n" \
114 " nop \n" \
116 : "i"(~EPSW_IM), "i"(EPSW_IE | MN10300_CLI_LEVEL) \
117 ); \
118 } while (0)
120 #define local_irq_save(x) \
121 do { \
122 local_save_flags(x); \
123 local_irq_disable(); \
124 } while (0)
127 * we make sure local_irq_enable() doesn't cause priority inversion
129 #ifndef __ASSEMBLY__
131 extern unsigned long __mn10300_irq_enabled_epsw;
133 #endif
135 #define local_irq_enable() \
136 do { \
137 unsigned long tmp; \
139 asm volatile( \
140 " mov epsw,%0 \n" \
141 " and %1,%0 \n" \
142 " or %2,%0 \n" \
143 " mov %0,epsw \n" \
144 : "=&d"(tmp) \
145 : "i"(~EPSW_IM), "r"(__mn10300_irq_enabled_epsw) \
146 ); \
147 } while (0)
149 #define local_irq_restore(x) \
150 do { \
151 typecheck(unsigned long, x); \
152 asm volatile( \
153 " mov %0,epsw \n" \
154 " nop \n" \
155 " nop \n" \
156 " nop \n" \
158 : "d"(x) \
159 : "memory", "cc" \
160 ); \
161 } while (0)
163 #define irqs_disabled() \
164 ({ \
165 unsigned long flags; \
166 local_save_flags(flags); \
167 (flags & EPSW_IM) <= MN10300_CLI_LEVEL; \
170 /* hook to save power by halting the CPU
171 * - called from the idle loop
172 * - must reenable interrupts (which takes three instruction cycles to complete)
174 #define safe_halt() \
175 do { \
176 asm volatile(" or %0,epsw \n" \
177 " nop \n" \
178 " nop \n" \
179 " bset %2,(%1) \n" \
181 : "i"(EPSW_IE|EPSW_IM), "n"(&CPUM), "i"(CPUM_SLEEP)\
182 : "cc" \
183 ); \
184 } while (0)
186 #define STI or EPSW_IE|EPSW_IM,epsw
187 #define CLI and ~EPSW_IM,epsw; or EPSW_IE|MN10300_CLI_LEVEL,epsw; nop; nop; nop
189 /*****************************************************************************/
191 * MN10300 doesn't actually have an exchange instruction
193 #ifndef __ASSEMBLY__
195 struct __xchg_dummy { unsigned long a[100]; };
196 #define __xg(x) ((struct __xchg_dummy *)(x))
198 static inline
199 unsigned long __xchg(volatile unsigned long *m, unsigned long val)
201 unsigned long retval;
202 unsigned long flags;
204 local_irq_save(flags);
205 retval = *m;
206 *m = val;
207 local_irq_restore(flags);
208 return retval;
211 #define xchg(ptr, v) \
212 ((__typeof__(*(ptr))) __xchg((unsigned long *)(ptr), \
213 (unsigned long)(v)))
215 static inline unsigned long __cmpxchg(volatile unsigned long *m,
216 unsigned long old, unsigned long new)
218 unsigned long retval;
219 unsigned long flags;
221 local_irq_save(flags);
222 retval = *m;
223 if (retval == old)
224 *m = new;
225 local_irq_restore(flags);
226 return retval;
229 #define cmpxchg(ptr, o, n) \
230 ((__typeof__(*(ptr))) __cmpxchg((unsigned long *)(ptr), \
231 (unsigned long)(o), \
232 (unsigned long)(n)))
234 #endif /* !__ASSEMBLY__ */
236 #endif /* __KERNEL__ */
237 #endif /* _ASM_SYSTEM_H */