asm.h revision 1.29.2.2 1 /* $NetBSD: asm.h,v 1.29.2.2 2011/06/12 00:24:04 rmind Exp $ */
2
3 /*
4 * Copyright (C) 1995, 1996 Wolfgang Solfrank.
5 * Copyright (C) 1995, 1996 TooLs GmbH.
6 * All rights reserved.
7 *
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions
10 * are met:
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in the
15 * documentation and/or other materials provided with the distribution.
16 * 3. All advertising materials mentioning features or use of this software
17 * must display the following acknowledgement:
18 * This product includes software developed by TooLs GmbH.
19 * 4. The name of TooLs GmbH may not be used to endorse or promote products
20 * derived from this software without specific prior written permission.
21 *
22 * THIS SOFTWARE IS PROVIDED BY TOOLS GMBH ``AS IS'' AND ANY EXPRESS OR
23 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
24 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
25 * IN NO EVENT SHALL TOOLS GMBH BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
26 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
27 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
28 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
29 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
30 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
31 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32 */
33
34 #ifndef _PPC_ASM_H_
35 #define _PPC_ASM_H_
36
37 #ifdef _LP64
38
39 /* ppc64 is always PIC, r2 is always the TOC */
40
41 #define PIC_PLT(x) .x
42
43 #else
44
45 #ifdef PIC
46 #define PIC_PROLOGUE XXX
47 #define PIC_EPILOGUE XXX
48 #define PIC_PLT(x) x+32768@plt
49 #ifdef __STDC__
50 #define PIC_TOCNAME(name) .LCTOC_##name
51 #else
52 #define PIC_TOCNAME(name) .LCTOC_/**/name
53 #endif
54 #define PIC_TOCSETUP(name, reg) \
55 .pushsection ".got2","aw" ;\
56 PIC_TOCNAME(name) = . + 32768 ;\
57 .popsection ;\
58 bcl 20,31,1001f ;\
59 1001: mflr reg ;\
60 addis reg,reg,PIC_TOCNAME(name)-1001b@ha ;\
61 addi reg,reg,PIC_TOCNAME(name)-1001b@l
62 #define PIC_GOTSETUP(reg) \
63 bcl 20,31,2002f ;\
64 2002: mflr reg ;\
65 addis reg,reg,_GLOBAL_OFFSET_TABLE_-2002b@ha ;\
66 addi reg,reg,_GLOBAL_OFFSET_TABLE_-2002b@l
67 #ifdef __STDC__
68 #define PIC_GOT(x) XXX
69 #define PIC_GOTOFF(x) XXX
70 #else /* not __STDC__ */
71 #define PIC_GOT(x) XXX
72 #define PIC_GOTOFF(x) XXX
73 #endif /* __STDC__ */
74 #else
75 #define PIC_PROLOGUE
76 #define PIC_EPILOGUE
77 #define PIC_PLT(x) x
78 #define PIC_GOT(x) x
79 #define PIC_GOTOFF(x) x
80 #define PIC_GOTSETUP(r)
81 #define PIC_TOCSETUP(n, r)
82 #endif
83
84 #endif
85
86 #define _C_LABEL(x) x
87 #define _ASM_LABEL(x) x
88
89 #define _GLOBAL(x) \
90 .data; .align 2; .globl x; x:
91
92 #ifdef GPROF
93 # define _PROF_PROLOGUE mflr 0; stw 0,4(1); bl _mcount
94 #else
95 # define _PROF_PROLOGUE
96 #endif
97
98 #ifdef _LP64
99
100 #define SF_HEADER_SZ 48
101 #define SF_PARAM_SZ 64
102 #define SF_SZ (SF_HEADER_SZ + SF_PARAM_SZ)
103
104 #define SF_SP 0
105 #define SF_CR 8
106 #define SF_LR 16
107 #define SF_PARAM SF_HEADER_SZ
108
109 #define ENTRY(y) \
110 .globl y; \
111 .section ".opd","aw"; \
112 .align 3; \
113 y: .quad .y,.TOC.@tocbase,0; \
114 .previous; \
115 .size y,24; \
116 .type .y,@function; \
117 .globl .y; \
118 .align 3; \
119 .y:
120
121 #define CALL(y) \
122 bl .y; \
123 nop
124
125 #define ENTRY_NOPROFILE(y) ENTRY(y)
126 #define ASENTRY(y) ENTRY(y)
127 #else
128
129 #define _ENTRY(x) \
130 .text; .align 2; .globl x; .type x,@function; x:
131
132 #define ENTRY(y) _ENTRY(_C_LABEL(y)); _PROF_PROLOGUE
133 #define END(y) .size _C_LABEL(y),.-_C_LABEL(y)
134
135 #define ENTRY_NOPROFILE(y) _ENTRY(_C_LABEL(y))
136
137 #define CALL(y) \
138 bl y
139
140 #define ASENTRY(y) _ENTRY(_ASM_LABEL(y)); _PROF_PROLOGUE
141 #endif
142
143 #define GLOBAL(y) _GLOBAL(_C_LABEL(y))
144
145 #define ASMSTR .asciz
146
147 #undef __RCSID
148 #define RCSID(x) __RCSID(x)
149 #define __RCSID(x) .pushsection .ident; .asciz x; .popsection
150
151 #ifdef __ELF__
152 #define WEAK_ALIAS(alias,sym) \
153 .weak alias; \
154 alias = sym
155 #endif
156 /*
157 * STRONG_ALIAS: create a strong alias.
158 */
159 #define STRONG_ALIAS(alias,sym) \
160 .globl alias; \
161 alias = sym
162
163 #ifdef __STDC__
164 #define WARN_REFERENCES(sym,msg) \
165 .pushsection .gnu.warning. ## sym; \
166 .ascii msg; \
167 .popsection
168 #else
169 #define WARN_REFERENCES(sym,msg) \
170 .pushsection .gnu.warning./**/sym; \
171 .ascii msg; \
172 .popsection
173 #endif /* __STDC__ */
174
175 #ifdef _KERNEL
176 /*
177 * Get cpu_info pointer for current processor. Always in SPRG0. *ALWAYS*
178 */
179 #define GET_CPUINFO(r) mfsprg r,0
180 /*
181 * IN:
182 * R4[er] = first free byte beyond end/esym.
183 *
184 * OUT:
185 * R1[sp] = new kernel stack
186 * R4[er] = kernelend
187 */
188
189 #ifdef CI_INTSTK
190 #define INIT_CPUINFO_INTSTK(er,tmp1) \
191 addi er,er,INTSTK; \
192 stptr er,CI_INTSTK(tmp1)
193 #else
194 #define INIT_CPUINFO_INTSTK(er,tmp1) /* nothing */
195 #endif
196
197 /*
198 * We use lis/ori instead of lis/addi in case tmp2 is r0.
199 */
200 #define INIT_CPUINFO(er,sp,tmp1,tmp2) \
201 li tmp1,PAGE_MASK; \
202 add er,er,tmp1; \
203 andc er,er,tmp1; /* page align */ \
204 lis tmp1,_C_LABEL(cpu_info)@ha; \
205 addi tmp1,tmp1,_C_LABEL(cpu_info)@l; \
206 mtsprg0 tmp1; /* save for later use */ \
207 INIT_CPUINFO_INTSTK(er,tmp1); \
208 lis tmp2,_C_LABEL(emptyidlespin)@h; \
209 ori tmp2,tmp2,_C_LABEL(emptyidlespin)@l; \
210 stptr tmp2,CI_IDLESPIN(tmp1); \
211 li tmp2,-1; \
212 stint tmp2,CI_IDEPTH(tmp1); \
213 li tmp2,0; \
214 lis %r13,_C_LABEL(lwp0)@h; \
215 ori %r13,%r13,_C_LABEL(lwp0)@l; \
216 stptr er,L_PCB(%r13); /* XXXuvm_lwp_getuarea */ \
217 stptr tmp1,L_CPU(%r13); \
218 addi er,er,USPACE; /* stackpointer for lwp0 */ \
219 addi sp,er,-FRAMELEN-CALLFRAMELEN; /* stackpointer for lwp0 */ \
220 stptr sp,L_MD_UTF(%r13); /* save in lwp0.l_md.md_utf */ \
221 /* er = end of mem reserved for kernel */ \
222 li tmp2,0; \
223 stptr tmp2,-CALLFRAMELEN(er); /* end of stack chain */ \
224 stptru tmp2,-CALLFRAMELEN(sp) /* end of stack chain */
225
226 #endif
227
228 /* Condition Register Bit Fields */
229
230 #if defined(_REGNAMES)
231 #if defined(_KERNEL) || defined(_STANDALONE)
232 #define cr0 0
233 #define cr1 1
234 #define cr2 2
235 #define cr3 3
236 #define cr4 4
237 #define cr5 5
238 #define cr6 6
239 #define cr7 7
240 #endif
241
242 /* General Purpose Registers (GPRs) */
243
244 #if defined(_KERNEL) || defined(_STANDALONE)
245 #define r0 0
246 #define r1 1
247 #define r2 2
248 #define r3 3
249 #define r4 4
250 #define r5 5
251 #define r6 6
252 #define r7 7
253 #define r8 8
254 #define r9 9
255 #define r10 10
256 #define r11 11
257 #define r12 12
258 #define r13 13
259 #define r14 14
260 #define r15 15
261 #define r16 16
262 #define r17 17
263 #define r18 18
264 #define r19 19
265 #define r20 20
266 #define r21 21
267 #define r22 22
268 #define r23 23
269 #define r24 24
270 #define r25 25
271 #define r26 26
272 #define r27 27
273 #define r28 28
274 #define r29 29
275 #define r30 30
276 #define r31 31
277 #endif
278
279 /* Floating Point Registers (FPRs) */
280
281 #if defined(_KERNEL) || defined(_STANDALONE)
282 #define fr0 0
283 #define fr1 1
284 #define fr2 2
285 #define fr3 3
286 #define fr4 4
287 #define fr5 5
288 #define fr6 6
289 #define fr7 7
290 #define fr8 8
291 #define fr9 9
292 #define fr10 10
293 #define fr11 11
294 #define fr12 12
295 #define fr13 13
296 #define fr14 14
297 #define fr15 15
298 #define fr16 16
299 #define fr17 17
300 #define fr18 18
301 #define fr19 19
302 #define fr20 20
303 #define fr21 21
304 #define fr22 22
305 #define fr23 23
306 #define fr24 24
307 #define fr25 25
308 #define fr26 26
309 #define fr27 27
310 #define fr28 28
311 #define fr29 29
312 #define fr30 30
313 #define fr31 31
314 #endif
315 #endif /* _REGNAMES */
316
317 /*
318 * Add some psuedo instructions to made sharing of assembly versions of
319 * ILP32 and LP64 code possible.
320 */
321 #define ldint lwz /* not needed but for completeness */
322 #define ldintu lwzu /* not needed but for completeness */
323 #define stint stw /* not needed but for completeness */
324 #define stintu stwu /* not needed but for completeness */
325
326 #ifndef _LP64
327
328 #define ldlong lwz /* load "C" long */
329 #define ldlongu lwzu /* load "C" long with udpate */
330 #define stlong stw /* load "C" long */
331 #define stlongu stwu /* load "C" long with udpate */
332 #define ldptr lwz /* load "C" pointer */
333 #define ldptru lwzu /* load "C" pointer with udpate */
334 #define stptr stw /* load "C" pointer */
335 #define stptru stwu /* load "C" pointer with udpate */
336 #define ldreg lwz /* load PPC general register */
337 #define ldregu lwzu /* load PPC general register with udpate */
338 #define streg stw /* load PPC general register */
339 #define stregu stwu /* load PPC general register with udpate */
340 #define SZREG 4 /* 4 byte registers */
341
342 #define lptrarx lwarx /* load "C" pointer with reservation */
343 #define llongarx lwarx /* load "C" long with reservation */
344 #define lregarx lwarx /* load PPC general register with reservation */
345
346 #define stptrcx stwcx /* store "C" pointer conditional */
347 #define stlongcx stwcx /* store "C" long conditional */
348 #define stregcx stwcx /* store PPC general register conditional */
349
350 #define clrrptri clrrwi /* clear right "C" pointer immediate */
351 #define clrrlongi clrrwi /* clear right "C" long immediate */
352 #define clrrregi clrrwi /* clear right PPC general register immediate */
353
354 #else
355
356 #define ldlong ld /* load "C" long */
357 #define ldlongu ldu /* load "C" long with update */
358 #define stlong std /* store "C" long */
359 #define stlongu stdu /* store "C" long with update */
360 #define ldptr ld /* load "C" pointer */
361 #define ldptru ldu /* load "C" pointer with update */
362 #define stptr std /* store "C" pointer */
363 #define stptru stdu /* store "C" pointer with update */
364 #define ldreg ld /* load PPC general register */
365 #define ldregu ldu /* load PPC general register with update */
366 #define streg std /* store PPC general register */
367 #define stregu stdu /* store PPC general register with update */
368 /* redefined this to force an error on PPC64 to catch their use. */
369 #define lmw lmd /* load multiple PPC general registers */
370 #define stmw stmd /* store multiple PPC general registers */
371 #define SZREG 8 /* 8 byte registers */
372
373 #define lptrarx ldarx /* load "C" pointer with reservation */
374 #define llongarx ldarx /* load "C" long with reservation */
375 #define lregarx ldarx /* load PPC general register with reservation */
376
377 #define stptrcx stdcx /* store "C" pointer conditional */
378 #define stlongcx stdcx /* store "C" long conditional */
379 #define stregax stdcx /* store PPC general register conditional */
380
381 #define clrrptri clrrdi /* clear right "C" pointer immediate */
382 #define clrrlongi clrrdi /* clear right "C" long immediate */
383 #define clrrregi clrrdi /* clear right PPC general register immediate */
384
385 #endif
386
387 #ifdef _LOCORE
388 .macro stmd r,dst
389 i = 0
390 .rept 32-\r
391 std i+\r, i*8+\dst
392 i = i + 1
393 .endr
394 .endm
395
396 .macro lmd r,dst
397 i = 0
398 .rept 32-\r
399 ld i+\r, i*8+\dst
400 i = i + 1
401 .endr
402 .endm
403 #endif
404
405 #endif /* !_PPC_ASM_H_ */
406