diff options
Diffstat (limited to 'sys/powerpc/include/asm.h')
| -rw-r--r-- | sys/powerpc/include/asm.h | 267 | 
1 files changed, 267 insertions, 0 deletions
| diff --git a/sys/powerpc/include/asm.h b/sys/powerpc/include/asm.h new file mode 100644 index 000000000000..35a8536776ea --- /dev/null +++ b/sys/powerpc/include/asm.h @@ -0,0 +1,267 @@ +/*- + * SPDX-License-Identifier: BSD-4-Clause + * + * Copyright (C) 1995, 1996 Wolfgang Solfrank. + * Copyright (C) 1995, 1996 TooLs GmbH. + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + *    notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + *    notice, this list of conditions and the following disclaimer in the + *    documentation and/or other materials provided with the distribution. + * 3. All advertising materials mentioning features or use of this software + *    must display the following acknowledgement: + *	This product includes software developed by TooLs GmbH. + * 4. The name of TooLs GmbH may not be used to endorse or promote products + *    derived from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY TOOLS GMBH ``AS IS'' AND ANY EXPRESS OR + * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES + * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. + * IN NO EVENT SHALL TOOLS GMBH BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, + * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; + * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, + * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR + * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF + * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + * + *	$NetBSD: asm.h,v 1.6.18.1 2000/07/25 08:37:14 kleink Exp $ + */ + +#ifndef _MACHINE_ASM_H_ +#define	_MACHINE_ASM_H_ + +#include <sys/cdefs.h> + +#if defined(PIC) && !defined(__powerpc64__) +#define	PIC_PROLOGUE	XXX +#define	PIC_EPILOGUE	XXX +#define	PIC_PLT(x)	x@plt +#ifdef	__STDC__ +#define	PIC_GOT(x)	XXX +#else	/* not __STDC__ */ +#define	PIC_GOT(x)	XXX +#endif	/* __STDC__ */ +#else +#define	PIC_PROLOGUE +#define	PIC_EPILOGUE +#define	PIC_PLT(x)	x +#define PIC_GOT(x)	x +#endif + +#define	CNAME(csym)		csym +#define	ASMNAME(asmsym)		asmsym +#ifdef __powerpc64__ +#define	HIDENAME(asmsym)	__CONCAT(_,asmsym) +#else +#define	HIDENAME(asmsym)	__CONCAT(.,asmsym) +#endif + +#if !defined(_CALL_ELF) || _CALL_ELF == 1 +#ifdef _KERNEL +/* ELFv1 kernel uses global dot symbols */ +#define	DOT_LABEL(name)		__CONCAT(.,name) +#define	TYPE_ENTRY(name)	.size	name,24; \ +				.type	DOT_LABEL(name),@function; \ +				.globl	DOT_LABEL(name); +#define	END_SIZE(name)		.size	DOT_LABEL(name),.-DOT_LABEL(name); +#else /* !_KERNEL */ +/* ELFv1 user code uses local function entry points */ +#define	DOT_LABEL(name)		__CONCAT(.L.,name) +#define	TYPE_ENTRY(name)	.type	name,@function; +#define	END_SIZE(name)		.size	name,.-DOT_LABEL(name); +#endif /* _KERNEL */ +#else +/* ELFv2 doesn't have any of this complication */ +#define	DOT_LABEL(name)		name +#define	TYPE_ENTRY(name)	.type	name,@function; +#define	END_SIZE(name)		.size	name,.-DOT_LABEL(name); +#endif + +#define	_GLOBAL(name) \ +	.data; \ +	.p2align 2; \ +	.globl	name; \ +	name: + +#ifdef __powerpc64__ +#define TOC_NAME_FOR_REF(name)	__CONCAT(.L,name) +#define	TOC_REF(name)	TOC_NAME_FOR_REF(name)@toc +#define TOC_ENTRY(name) \ +	.section ".toc","aw"; \ +	TOC_NAME_FOR_REF(name): \ +        .tc name[TC],name +#endif + +#ifdef __powerpc64__ + +#if !defined(_CALL_ELF) || _CALL_ELF == 1 +#define	_ENTRY(name) \ +	.section ".text"; \ +	.p2align 2; \ +	.globl	name; \ +	.section ".opd","aw"; \ +	.p2align 3; \ +name: \ +	.quad	DOT_LABEL(name),.TOC.@tocbase,0; \ +	.previous; \ +	.p2align 4; \ +	TYPE_ENTRY(name) \ +DOT_LABEL(name): \ +	.cfi_startproc +#define	_NAKED_ENTRY(name)	_ENTRY(name) +#else +#define	_ENTRY(name) \ +	.text; \ +	.p2align 4; \ +	.globl	name; \ +	.type	name,@function; \ +name: \ +	.cfi_startproc; \ +	addis	%r2, %r12, (.TOC.-name)@ha; \ +	addi	%r2, %r2, (.TOC.-name)@l; \ +	.localentry name, .-name; + +/* "Naked" function entry.  No TOC prologue for ELFv2. */ +#define	_NAKED_ENTRY(name) \ +	.text; \ +	.p2align 4; \ +	.globl	name; \ +	.type	name,@function; \ +name: \ +	.cfi_startproc; \ +	.localentry name, .-name; +#endif + +#define	_END(name) \ +	.cfi_endproc; \ +	.long	0; \ +	.byte	0,0,0,0,0,0,0,0; \ +	END_SIZE(name) + +#define	LOAD_ADDR(reg, var) \ +	lis	reg, var@highest; \ +	ori	reg, reg, var@higher; \ +	rldicr	reg, reg, 32, 31; \ +	oris	reg, reg, var@h; \ +	ori	reg, reg, var@l; +#else /* !__powerpc64__ */ +#define	_ENTRY(name) \ +	.text; \ +	.p2align 4; \ +	.globl	name; \ +	.type	name,@function; \ +name: \ +	.cfi_startproc +#define	_END(name) \ +	.cfi_endproc; \ +	.size	name, . - name + +#define _NAKED_ENTRY(name)	_ENTRY(name) + +#define	LOAD_ADDR(reg, var) \ +	lis	reg, var@ha; \ +	ori	reg, reg, var@l; +#endif /* __powerpc64__ */ + +#if defined(PROF) || (defined(_KERNEL) && defined(GPROF)) +# ifdef __powerpc64__ +#   define	_PROF_PROLOGUE	mflr 0;					\ +				std 3,48(1);				\ +				std 4,56(1);				\ +				std 5,64(1);				\ +				std 0,16(1);				\ +				stdu 1,-112(1);				\ +				bl _mcount;				\ +				nop;					\ +				ld 0,112+16(1);				\ +				ld 3,112+48(1);				\ +				ld 4,112+56(1);				\ +				ld 5,112+64(1);				\ +				mtlr 0;					\ +				addi 1,1,112 +# else +#   define	_PROF_PROLOGUE	mflr 0; stw 0,4(1); bl _mcount +# endif +#else +# define	_PROF_PROLOGUE +#endif + +#define	ASEND(y)	_END(ASMNAME(y)) +#define	ASENTRY(y)	_ENTRY(ASMNAME(y)); _PROF_PROLOGUE +#define	END(y)		_END(CNAME(y)) +#define	ENTRY(y)	_ENTRY(CNAME(y)); _PROF_PROLOGUE +#define	GLOBAL(y)	_GLOBAL(CNAME(y)) + +#define	ASENTRY_NOPROF(y)	_ENTRY(ASMNAME(y)) +#define	ENTRY_NOPROF(y)		_ENTRY(CNAME(y)) + +/* Load NIA without affecting branch prediction */ +#define	LOAD_LR_NIA	bcl	20, 31, .+4 + +/* + * Magic sequence to return to native endian. + * Overwrites r0 and r11. + * + * The encoding of the instruction "tdi 0, %r0, 0x48" in opposite endian + * happens to be "b . + 8". This is useful because we can write a sequence + * of instructions that can execute in either endian. + * + * Use a sequence of handcoded instructions that switches contexts to the + * instruction following the sequence, but with the correct PSL_LE bit. + * + * The same sequence works for both BE and LE because the xori will flip + * the bit to the other state, and the code only runs when running in the + * wrong endian. + * + * This sequence is NMI-reentrant. + * + * Do not change the length of this sequence without looking at the users, + * this is used in size-constrained places like the reset vector! + */ +#define	RETURN_TO_NATIVE_ENDIAN						  \ +	tdi	0, %r0, 0x48;	/* Endian swapped: b . + 8		*/\ +	b	1f;		/* Will fall through to here if correct */\ +	.long	0xa600607d;	/* mfmsr %r11				*/\ +	.long	0x00000038;	/* li %r0, 0				*/\ +	.long	0x6401617d;	/* mtmsrd %r0, 1 (L=1 EE,RI bits only)	*/\ +	.long	0x01006b69;	/* xori %r11, %r11, 0x1 (PSL_LE)	*/\ +	.long	0xa602087c;	/* mflr %r0				*/\ +	.long	0x05009f42;	/* LOAD_LR_NIA				*/\ +	.long	0xa6037b7d;	/* 0: mtsrr1 %r11			*/\ +	.long	0xa602687d;	/* mflr	%r11				*/\ +	.long	0x18006b39;	/* addi	%r11, %r11, (1f - 0b)		*/\ +	.long	0xa6037a7d;	/* mtsrr0 %r11				*/\ +	.long	0xa603087c;	/* mtlr %r0				*/\ +	.long	0x2400004c;	/* rfid					*/\ +1:	/* RETURN_TO_NATIVE_ENDIAN */ + +#define	ASMSTR		.asciz + +#define	RCSID(x)	.text; .asciz x + +#undef __FBSDID +#if !defined(lint) && !defined(STRIP_FBSDID) +#define __FBSDID(s)	.ident s +#else +#define __FBSDID(s)	/* nothing */ +#endif /* not lint and not STRIP_FBSDID */ + +#define	WEAK_REFERENCE(sym, alias)				\ +	.weak alias;						\ +	.equ alias,sym + +#ifdef __STDC__ +#define	WARN_REFERENCES(_sym,_msg)				\ +	.section .gnu.warning. ## _sym ; .ascii _msg ; .text +#else +#define	WARN_REFERENCES(_sym,_msg)				\ +	.section .gnu.warning./**/_sym ; .ascii _msg ; .text +#endif /* __STDC__ */ + +#endif /* !_MACHINE_ASM_H_ */ | 
