blob: 470d7400ac59755c82179739caeb8d200bff03ec [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
Linus Torvalds1da177e2005-04-16 15:20:36 -07002 * Copyright (C) 1995-1999 Gary Thomas, Paul Mackerras, Cort Dougan.
Linus Torvalds1da177e2005-04-16 15:20:36 -07003 */
Kumar Gala5f7c6902005-09-09 15:02:25 -05004#ifndef _ASM_POWERPC_PPC_ASM_H
5#define _ASM_POWERPC_PPC_ASM_H
6
Paul Mackerras40ef8cb2005-10-10 22:50:37 +10007#include <linux/stringify.h>
8#include <linux/config.h>
9
Kumar Gala5f7c6902005-09-09 15:02:25 -050010#ifdef __ASSEMBLY__
Linus Torvalds1da177e2005-04-16 15:20:36 -070011
12/*
13 * Macros for storing registers into and loading registers from
14 * exception frames.
15 */
Kumar Gala5f7c6902005-09-09 15:02:25 -050016#ifdef __powerpc64__
17#define SAVE_GPR(n, base) std n,GPR0+8*(n)(base)
18#define REST_GPR(n, base) ld n,GPR0+8*(n)(base)
19#define SAVE_NVGPRS(base) SAVE_8GPRS(14, base); SAVE_10GPRS(22, base)
20#define REST_NVGPRS(base) REST_8GPRS(14, base); REST_10GPRS(22, base)
21#else
Linus Torvalds1da177e2005-04-16 15:20:36 -070022#define SAVE_GPR(n, base) stw n,GPR0+4*(n)(base)
Linus Torvalds1da177e2005-04-16 15:20:36 -070023#define REST_GPR(n, base) lwz n,GPR0+4*(n)(base)
Linus Torvalds1da177e2005-04-16 15:20:36 -070024#define SAVE_NVGPRS(base) SAVE_GPR(13, base); SAVE_8GPRS(14, base); \
25 SAVE_10GPRS(22, base)
26#define REST_NVGPRS(base) REST_GPR(13, base); REST_8GPRS(14, base); \
27 REST_10GPRS(22, base)
Kumar Gala5f7c6902005-09-09 15:02:25 -050028#endif
29
30
31#define SAVE_2GPRS(n, base) SAVE_GPR(n, base); SAVE_GPR(n+1, base)
32#define SAVE_4GPRS(n, base) SAVE_2GPRS(n, base); SAVE_2GPRS(n+2, base)
33#define SAVE_8GPRS(n, base) SAVE_4GPRS(n, base); SAVE_4GPRS(n+4, base)
34#define SAVE_10GPRS(n, base) SAVE_8GPRS(n, base); SAVE_2GPRS(n+8, base)
35#define REST_2GPRS(n, base) REST_GPR(n, base); REST_GPR(n+1, base)
36#define REST_4GPRS(n, base) REST_2GPRS(n, base); REST_2GPRS(n+2, base)
37#define REST_8GPRS(n, base) REST_4GPRS(n, base); REST_4GPRS(n+4, base)
38#define REST_10GPRS(n, base) REST_8GPRS(n, base); REST_2GPRS(n+8, base)
Linus Torvalds1da177e2005-04-16 15:20:36 -070039
40#define SAVE_FPR(n, base) stfd n,THREAD_FPR0+8*(n)(base)
41#define SAVE_2FPRS(n, base) SAVE_FPR(n, base); SAVE_FPR(n+1, base)
42#define SAVE_4FPRS(n, base) SAVE_2FPRS(n, base); SAVE_2FPRS(n+2, base)
43#define SAVE_8FPRS(n, base) SAVE_4FPRS(n, base); SAVE_4FPRS(n+4, base)
44#define SAVE_16FPRS(n, base) SAVE_8FPRS(n, base); SAVE_8FPRS(n+8, base)
45#define SAVE_32FPRS(n, base) SAVE_16FPRS(n, base); SAVE_16FPRS(n+16, base)
46#define REST_FPR(n, base) lfd n,THREAD_FPR0+8*(n)(base)
47#define REST_2FPRS(n, base) REST_FPR(n, base); REST_FPR(n+1, base)
48#define REST_4FPRS(n, base) REST_2FPRS(n, base); REST_2FPRS(n+2, base)
49#define REST_8FPRS(n, base) REST_4FPRS(n, base); REST_4FPRS(n+4, base)
50#define REST_16FPRS(n, base) REST_8FPRS(n, base); REST_8FPRS(n+8, base)
51#define REST_32FPRS(n, base) REST_16FPRS(n, base); REST_16FPRS(n+16, base)
52
53#define SAVE_VR(n,b,base) li b,THREAD_VR0+(16*(n)); stvx n,b,base
Kumar Gala5f7c6902005-09-09 15:02:25 -050054#define SAVE_2VRS(n,b,base) SAVE_VR(n,b,base); SAVE_VR(n+1,b,base)
55#define SAVE_4VRS(n,b,base) SAVE_2VRS(n,b,base); SAVE_2VRS(n+2,b,base)
56#define SAVE_8VRS(n,b,base) SAVE_4VRS(n,b,base); SAVE_4VRS(n+4,b,base)
57#define SAVE_16VRS(n,b,base) SAVE_8VRS(n,b,base); SAVE_8VRS(n+8,b,base)
58#define SAVE_32VRS(n,b,base) SAVE_16VRS(n,b,base); SAVE_16VRS(n+16,b,base)
Linus Torvalds1da177e2005-04-16 15:20:36 -070059#define REST_VR(n,b,base) li b,THREAD_VR0+(16*(n)); lvx n,b,base
Kumar Gala5f7c6902005-09-09 15:02:25 -050060#define REST_2VRS(n,b,base) REST_VR(n,b,base); REST_VR(n+1,b,base)
61#define REST_4VRS(n,b,base) REST_2VRS(n,b,base); REST_2VRS(n+2,b,base)
62#define REST_8VRS(n,b,base) REST_4VRS(n,b,base); REST_4VRS(n+4,b,base)
63#define REST_16VRS(n,b,base) REST_8VRS(n,b,base); REST_8VRS(n+8,b,base)
64#define REST_32VRS(n,b,base) REST_16VRS(n,b,base); REST_16VRS(n+16,b,base)
Linus Torvalds1da177e2005-04-16 15:20:36 -070065
66#define SAVE_EVR(n,s,base) evmergehi s,s,n; stw s,THREAD_EVR0+4*(n)(base)
Kumar Gala5f7c6902005-09-09 15:02:25 -050067#define SAVE_2EVRS(n,s,base) SAVE_EVR(n,s,base); SAVE_EVR(n+1,s,base)
68#define SAVE_4EVRS(n,s,base) SAVE_2EVRS(n,s,base); SAVE_2EVRS(n+2,s,base)
69#define SAVE_8EVRS(n,s,base) SAVE_4EVRS(n,s,base); SAVE_4EVRS(n+4,s,base)
70#define SAVE_16EVRS(n,s,base) SAVE_8EVRS(n,s,base); SAVE_8EVRS(n+8,s,base)
71#define SAVE_32EVRS(n,s,base) SAVE_16EVRS(n,s,base); SAVE_16EVRS(n+16,s,base)
Linus Torvalds1da177e2005-04-16 15:20:36 -070072#define REST_EVR(n,s,base) lwz s,THREAD_EVR0+4*(n)(base); evmergelo n,s,n
Kumar Gala5f7c6902005-09-09 15:02:25 -050073#define REST_2EVRS(n,s,base) REST_EVR(n,s,base); REST_EVR(n+1,s,base)
74#define REST_4EVRS(n,s,base) REST_2EVRS(n,s,base); REST_2EVRS(n+2,s,base)
75#define REST_8EVRS(n,s,base) REST_4EVRS(n,s,base); REST_4EVRS(n+4,s,base)
76#define REST_16EVRS(n,s,base) REST_8EVRS(n,s,base); REST_8EVRS(n+8,s,base)
77#define REST_32EVRS(n,s,base) REST_16EVRS(n,s,base); REST_16EVRS(n+16,s,base)
Linus Torvalds1da177e2005-04-16 15:20:36 -070078
Michael Ellerman8c716322005-10-24 15:07:27 +100079/* Macros to adjust thread priority for hardware multithreading */
80#define HMT_VERY_LOW or 31,31,31 # very low priority
81#define HMT_LOW or 1,1,1
82#define HMT_MEDIUM_LOW or 6,6,6 # medium low priority
83#define HMT_MEDIUM or 2,2,2
84#define HMT_MEDIUM_HIGH or 5,5,5 # medium high priority
85#define HMT_HIGH or 3,3,3
Kumar Gala5f7c6902005-09-09 15:02:25 -050086
87/* handle instructions that older assemblers may not know */
88#define RFCI .long 0x4c000066 /* rfci instruction */
89#define RFDI .long 0x4c00004e /* rfdi instruction */
90#define RFMCI .long 0x4c00004c /* rfmci instruction */
91
Paul Mackerras40ef8cb2005-10-10 22:50:37 +100092#ifdef CONFIG_PPC64
93
94#define XGLUE(a,b) a##b
95#define GLUE(a,b) XGLUE(a,b)
96
97#define _GLOBAL(name) \
98 .section ".text"; \
99 .align 2 ; \
100 .globl name; \
101 .globl GLUE(.,name); \
102 .section ".opd","aw"; \
103name: \
104 .quad GLUE(.,name); \
105 .quad .TOC.@tocbase; \
106 .quad 0; \
107 .previous; \
108 .type GLUE(.,name),@function; \
109GLUE(.,name):
110
111#define _KPROBE(name) \
112 .section ".kprobes.text","a"; \
113 .align 2 ; \
114 .globl name; \
115 .globl GLUE(.,name); \
116 .section ".opd","aw"; \
117name: \
118 .quad GLUE(.,name); \
119 .quad .TOC.@tocbase; \
120 .quad 0; \
121 .previous; \
122 .type GLUE(.,name),@function; \
123GLUE(.,name):
124
125#define _STATIC(name) \
126 .section ".text"; \
127 .align 2 ; \
128 .section ".opd","aw"; \
129name: \
130 .quad GLUE(.,name); \
131 .quad .TOC.@tocbase; \
132 .quad 0; \
133 .previous; \
134 .type GLUE(.,name),@function; \
135GLUE(.,name):
136
137#else /* 32-bit */
138
139#define _GLOBAL(n) \
140 .text; \
141 .stabs __stringify(n:F-1),N_FUN,0,0,n;\
142 .globl n; \
143n:
144
145#define _KPROBE(n) \
146 .section ".kprobes.text","a"; \
147 .globl n; \
148n:
149
150#endif
151
Kumar Gala5f7c6902005-09-09 15:02:25 -0500152/*
153 * LOADADDR( rn, name )
154 * loads the address of 'name' into 'rn'
155 *
156 * LOADBASE( rn, name )
Paul Mackerras63162222005-10-27 22:44:39 +1000157 * loads the address (possibly without the low 16 bits) of 'name' into 'rn'
Kumar Gala5f7c6902005-09-09 15:02:25 -0500158 * suitable for base+disp addressing
159 */
160#ifdef __powerpc64__
161#define LOADADDR(rn,name) \
162 lis rn,name##@highest; \
163 ori rn,rn,name##@higher; \
164 rldicr rn,rn,32,31; \
165 oris rn,rn,name##@h; \
166 ori rn,rn,name##@l
167
Paul Mackerrasb85a0462005-10-06 10:59:19 +1000168#define LOADBASE(rn,name) \
Paul Mackerras63162222005-10-27 22:44:39 +1000169 ld rn,name@got(r2)
Kumar Gala5f7c6902005-09-09 15:02:25 -0500170
Paul Mackerrasb85a0462005-10-06 10:59:19 +1000171#define OFF(name) 0
Kumar Gala5f7c6902005-09-09 15:02:25 -0500172
173#define SET_REG_TO_CONST(reg, value) \
174 lis reg,(((value)>>48)&0xFFFF); \
175 ori reg,reg,(((value)>>32)&0xFFFF); \
176 rldicr reg,reg,32,31; \
177 oris reg,reg,(((value)>>16)&0xFFFF); \
178 ori reg,reg,((value)&0xFFFF);
179
180#define SET_REG_TO_LABEL(reg, label) \
181 lis reg,(label)@highest; \
182 ori reg,reg,(label)@higher; \
183 rldicr reg,reg,32,31; \
184 oris reg,reg,(label)@h; \
185 ori reg,reg,(label)@l;
Paul Mackerrasb85a0462005-10-06 10:59:19 +1000186
187/* operations for longs and pointers */
188#define LDL ld
189#define STL std
190#define CMPI cmpdi
191
192#else /* 32-bit */
Stephen Rothwell70620182005-10-12 17:44:55 +1000193#define LOADADDR(rn,name) \
Stephen Rothwellcf764852005-10-17 11:46:53 +1000194 lis rn,name@ha; \
Stephen Rothwell70620182005-10-12 17:44:55 +1000195 addi rn,rn,name@l
196
Paul Mackerrasb85a0462005-10-06 10:59:19 +1000197#define LOADBASE(rn,name) \
198 lis rn,name@ha
199
200#define OFF(name) name@l
201
202/* operations for longs and pointers */
203#define LDL lwz
204#define STL stw
205#define CMPI cmpwi
206
Kumar Gala5f7c6902005-09-09 15:02:25 -0500207#endif
208
209/* various errata or part fixups */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700210#ifdef CONFIG_PPC601_SYNC_FIX
211#define SYNC \
212BEGIN_FTR_SECTION \
213 sync; \
214 isync; \
215END_FTR_SECTION_IFSET(CPU_FTR_601)
216#define SYNC_601 \
217BEGIN_FTR_SECTION \
218 sync; \
219END_FTR_SECTION_IFSET(CPU_FTR_601)
220#define ISYNC_601 \
221BEGIN_FTR_SECTION \
222 isync; \
223END_FTR_SECTION_IFSET(CPU_FTR_601)
224#else
225#define SYNC
226#define SYNC_601
227#define ISYNC_601
228#endif
229
Kumar Gala5f7c6902005-09-09 15:02:25 -0500230
Linus Torvalds1da177e2005-04-16 15:20:36 -0700231#ifndef CONFIG_SMP
232#define TLBSYNC
233#else /* CONFIG_SMP */
234/* tlbsync is not implemented on 601 */
235#define TLBSYNC \
236BEGIN_FTR_SECTION \
237 tlbsync; \
238 sync; \
239END_FTR_SECTION_IFCLR(CPU_FTR_601)
240#endif
241
Kumar Gala5f7c6902005-09-09 15:02:25 -0500242
Linus Torvalds1da177e2005-04-16 15:20:36 -0700243/*
244 * This instruction is not implemented on the PPC 603 or 601; however, on
245 * the 403GCX and 405GP tlbia IS defined and tlbie is not.
246 * All of these instructions exist in the 8xx, they have magical powers,
247 * and they must be used.
248 */
249
250#if !defined(CONFIG_4xx) && !defined(CONFIG_8xx)
251#define tlbia \
252 li r4,1024; \
253 mtctr r4; \
254 lis r4,KERNELBASE@h; \
2550: tlbie r4; \
256 addi r4,r4,0x1000; \
257 bdnz 0b
258#endif
259
Kumar Gala5f7c6902005-09-09 15:02:25 -0500260
261#ifdef CONFIG_IBM405_ERR77
262#define PPC405_ERR77(ra,rb) dcbt ra, rb;
263#define PPC405_ERR77_SYNC sync;
264#else
265#define PPC405_ERR77(ra,rb)
266#define PPC405_ERR77_SYNC
267#endif
268
269
270#ifdef CONFIG_IBM440EP_ERR42
271#define PPC440EP_ERR42 isync
272#else
273#define PPC440EP_ERR42
274#endif
275
276
277#if defined(CONFIG_BOOKE)
Paul Mackerras63162222005-10-27 22:44:39 +1000278#define toreal(rd)
279#define fromreal(rd)
280
Linus Torvalds1da177e2005-04-16 15:20:36 -0700281#define tophys(rd,rs) \
282 addis rd,rs,0
283
284#define tovirt(rd,rs) \
285 addis rd,rs,0
286
Kumar Gala5f7c6902005-09-09 15:02:25 -0500287#elif defined(CONFIG_PPC64)
Paul Mackerras63162222005-10-27 22:44:39 +1000288#define toreal(rd) /* we can access c000... in real mode */
289#define fromreal(rd)
290
Kumar Gala5f7c6902005-09-09 15:02:25 -0500291#define tophys(rd,rs) \
Paul Mackerras63162222005-10-27 22:44:39 +1000292 clrldi rd,rs,2
Kumar Gala5f7c6902005-09-09 15:02:25 -0500293
294#define tovirt(rd,rs) \
Paul Mackerras63162222005-10-27 22:44:39 +1000295 rotldi rd,rs,16; \
296 ori rd,rd,((KERNELBASE>>48)&0xFFFF);\
297 rotldi rd,rd,48
Kumar Gala5f7c6902005-09-09 15:02:25 -0500298#else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700299/*
300 * On APUS (Amiga PowerPC cpu upgrade board), we don't know the
301 * physical base address of RAM at compile time.
302 */
Paul Mackerras63162222005-10-27 22:44:39 +1000303#define toreal(rd) tophys(rd,rd)
304#define fromreal(rd) tovirt(rd,rd)
305
Linus Torvalds1da177e2005-04-16 15:20:36 -0700306#define tophys(rd,rs) \
3070: addis rd,rs,-KERNELBASE@h; \
308 .section ".vtop_fixup","aw"; \
309 .align 1; \
310 .long 0b; \
311 .previous
312
313#define tovirt(rd,rs) \
3140: addis rd,rs,KERNELBASE@h; \
315 .section ".ptov_fixup","aw"; \
316 .align 1; \
317 .long 0b; \
318 .previous
Kumar Gala5f7c6902005-09-09 15:02:25 -0500319#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700320
Paul Mackerras40ef8cb2005-10-10 22:50:37 +1000321#ifdef CONFIG_PPC64
322#define RFI rfid
323#define MTMSRD(r) mtmsrd r
Linus Torvalds1da177e2005-04-16 15:20:36 -0700324
325#else
326#define FIX_SRR1(ra, rb)
327#ifndef CONFIG_40x
328#define RFI rfi
329#else
330#define RFI rfi; b . /* Prevent prefetch past rfi */
331#endif
332#define MTMSRD(r) mtmsr r
333#define CLR_TOP32(r)
Matt Porterc9cf73a2005-07-31 22:34:52 -0700334#endif
335
Linus Torvalds1da177e2005-04-16 15:20:36 -0700336/* The boring bits... */
337
338/* Condition Register Bit Fields */
339
340#define cr0 0
341#define cr1 1
342#define cr2 2
343#define cr3 3
344#define cr4 4
345#define cr5 5
346#define cr6 6
347#define cr7 7
348
349
350/* General Purpose Registers (GPRs) */
351
352#define r0 0
353#define r1 1
354#define r2 2
355#define r3 3
356#define r4 4
357#define r5 5
358#define r6 6
359#define r7 7
360#define r8 8
361#define r9 9
362#define r10 10
363#define r11 11
364#define r12 12
365#define r13 13
366#define r14 14
367#define r15 15
368#define r16 16
369#define r17 17
370#define r18 18
371#define r19 19
372#define r20 20
373#define r21 21
374#define r22 22
375#define r23 23
376#define r24 24
377#define r25 25
378#define r26 26
379#define r27 27
380#define r28 28
381#define r29 29
382#define r30 30
383#define r31 31
384
385
386/* Floating Point Registers (FPRs) */
387
388#define fr0 0
389#define fr1 1
390#define fr2 2
391#define fr3 3
392#define fr4 4
393#define fr5 5
394#define fr6 6
395#define fr7 7
396#define fr8 8
397#define fr9 9
398#define fr10 10
399#define fr11 11
400#define fr12 12
401#define fr13 13
402#define fr14 14
403#define fr15 15
404#define fr16 16
405#define fr17 17
406#define fr18 18
407#define fr19 19
408#define fr20 20
409#define fr21 21
410#define fr22 22
411#define fr23 23
412#define fr24 24
413#define fr25 25
414#define fr26 26
415#define fr27 27
416#define fr28 28
417#define fr29 29
418#define fr30 30
419#define fr31 31
420
Kumar Gala5f7c6902005-09-09 15:02:25 -0500421/* AltiVec Registers (VPRs) */
422
Linus Torvalds1da177e2005-04-16 15:20:36 -0700423#define vr0 0
424#define vr1 1
425#define vr2 2
426#define vr3 3
427#define vr4 4
428#define vr5 5
429#define vr6 6
430#define vr7 7
431#define vr8 8
432#define vr9 9
433#define vr10 10
434#define vr11 11
435#define vr12 12
436#define vr13 13
437#define vr14 14
438#define vr15 15
439#define vr16 16
440#define vr17 17
441#define vr18 18
442#define vr19 19
443#define vr20 20
444#define vr21 21
445#define vr22 22
446#define vr23 23
447#define vr24 24
448#define vr25 25
449#define vr26 26
450#define vr27 27
451#define vr28 28
452#define vr29 29
453#define vr30 30
454#define vr31 31
455
Kumar Gala5f7c6902005-09-09 15:02:25 -0500456/* SPE Registers (EVPRs) */
457
Linus Torvalds1da177e2005-04-16 15:20:36 -0700458#define evr0 0
459#define evr1 1
460#define evr2 2
461#define evr3 3
462#define evr4 4
463#define evr5 5
464#define evr6 6
465#define evr7 7
466#define evr8 8
467#define evr9 9
468#define evr10 10
469#define evr11 11
470#define evr12 12
471#define evr13 13
472#define evr14 14
473#define evr15 15
474#define evr16 16
475#define evr17 17
476#define evr18 18
477#define evr19 19
478#define evr20 20
479#define evr21 21
480#define evr22 22
481#define evr23 23
482#define evr24 24
483#define evr25 25
484#define evr26 26
485#define evr27 27
486#define evr28 28
487#define evr29 29
488#define evr30 30
489#define evr31 31
490
491/* some stab codes */
492#define N_FUN 36
493#define N_RSYM 64
494#define N_SLINE 68
495#define N_SO 100
Kumar Gala5f7c6902005-09-09 15:02:25 -0500496
497#define ASM_CONST(x) x
498#else
499 #define __ASM_CONST(x) x##UL
500 #define ASM_CONST(x) __ASM_CONST(x)
501#endif /* __ASSEMBLY__ */
502
503#endif /* _ASM_POWERPC_PPC_ASM_H */