abi.h   [plain text]


/*	$NetBSD: abi.h,v 1.1 2001/06/19 00:26:29 fvdl Exp $	*/

/*
 * Written by Frank van der Linden (fvdl@wasabisystems.com)
 */

/*
 * The x86-64 ABI specifies that float, double and long double
 * arguments are passed in SSE2 (xmm) registers. Unfortunately,
 * there is no way to push those on to the FP stack, which is
 * where he fancier instructions get their arguments from.
 *
 * Define some prologues and epilogues to store and retrieve
 * xmm regs to local variables.
 */

//
// Define some overrideable constants
// Default settings are for minimum stack frame, all arguments are long double
//
// LOCAL_STACK_SIZE -- number of bytes in addition to the return pointer that the stack frame holds
// FIRST_ARG_SIZE -- number of bytes used on the stack by the first argument (if passed on the stack)
// SECOND_ARG_SIZE -- number of bytes used on the stack by the second argument (if passed on the stack)
//
#ifndef LOCAL_STACK_SIZE
	#define LOCAL_STACK_SIZE	0
#endif
#ifndef FIRST_ARG_SIZE
	#define FIRST_ARG_SIZE		16
#endif
#ifndef SECOND_ARG_SIZE
	#define SECOND_ARG_SIZE		16
#endif


#if defined( __LP64__ )
	#define FRAME_SIZE			8		/* sizeof( void*) */
	#define STACKP				%rsp
	#define BASEP				%rbp
	#define AX_P				%rax
	#define BX_P				%rbx
	#define CX_P				%rcx
	#define DX_P				%rdx
	#define SIGN_EXTEND_EAX		cdqe
	#define ADDP				addq
	#define CALLP				callq
    #define FISTPP              fistpll
    #define FISTTPP             fisttpll
	#define MOVP				movq
    #define NEGP                negq
	#define SUBP				subq
	#define XORP				xorq
#else	
	#define FRAME_SIZE			4		/* sizeof( void*) */
	#define STACKP				%esp
	#define BASEP				%ebp
	#define AX_P				%eax
	#define BX_P				%ebx
	#define CX_P				%ecx
	#define DX_P				%edx
	#define SIGN_EXTEND_EAX		
	#define ADDP				addl
    #define FISTPP              fistpl
    #define FISTTPP             fisttpl
	#define MOVP				movl
    #define NEGP                negl
	#define SUBP				subl
	#define XORP				xorl
	#define CALLP				calll
#endif

#define FIRST_ARG_OFFSET		(FRAME_SIZE + LOCAL_STACK_SIZE)
#define SECOND_ARG_OFFSET		(FIRST_ARG_OFFSET + FIRST_ARG_SIZE)
#define THIRD_ARG_OFFSET		(SECOND_ARG_OFFSET + SECOND_ARG_SIZE)

/*
#ifdef __LP64__

#error these interfaces appear to violate the red zone

    #define ARG_LONG_DOUBLE_ONE		-4(%esp)  
    #define ARG_LONG_DOUBLE_TWO		-20(%esp) 
    #define ARG_DOUBLE_ONE		-8(%rsp)
    #define ARG_DOUBLE_TWO		-16(%rsp)
    #define ARG_FLOAT_ONE		-4(%rsp)
    #define ARG_FLOAT_TWO		-8(%rsp)

    #define XMM_ONE_ARG_LONG_DOUBLE_PROLOGUE
    #define XMM_TWO_ARG_LONG_DOUBLE_PROLOGUE

    #define XMM_ONE_ARG_DOUBLE_PROLOGUE \
        movsd	%xmm0, ARG_DOUBLE_ONE
        
    #define XMM_TWO_ARG_DOUBLE_PROLOGUE \
        movsd	%xmm0, ARG_DOUBLE_ONE ; \
        movsd	%xmm1, ARG_DOUBLE_TWO

    #define XMM_ONE_ARG_FLOAT_PROLOGUE \
        movss	%xmm0, ARG_FLOAT_ONE

    #define XMM_TWO_ARG_FLOAT_PROLOGUE \
        movss	%xmm0, ARG_FLOAT_ONE ; \
        movss	%xmm1, ARG_FLOAT_TWO

    #define XMM_LONG_DOUBLE_EPILOGUE

    #define XMM_DOUBLE_EPILOGUE \
        fstpl ARG_DOUBLE_ONE ; \
        movsd ARG_DOUBLE_ONE, %xmm0

    #define XMM_FLOAT_EPILOGUE \
        fstps ARG_FLOAT_ONE ; \
        movss ARG_FLOAT_ONE, %xmm0
    
#else

    #define ARG_LONG_DOUBLE_ONE		4(%esp)
    #define ARG_LONG_DOUBLE_TWO		20(%esp)
    #define ARG_DOUBLE_ONE		4(%esp)
    #define ARG_DOUBLE_TWO		12(%esp)
    #define ARG_FLOAT_ONE		4(%esp)
    #define ARG_FLOAT_TWO		8(%esp)

    #define XMM_ONE_ARG_LONG_DOUBLE_PROLOGUE
    #define XMM_TWO_ARG_LONG_DOUBLE_PROLOGUE
    #define XMM_ONE_ARG_DOUBLE_PROLOGUE
    #define XMM_TWO_ARG_DOUBLE_PROLOGUE
    #define XMM_ONE_ARG_FLOAT_PROLOGUE
    #define XMM_TWO_ARG_FLOAT_PROLOGUE

    #define XMM_LONG_DOUBLE_EPILOGUE
    #define XMM_DOUBLE_EPILOGUE
    #define XMM_FLOAT_EPILOGUE

#endif
*/