fp_private.h   [plain text]


/*
 * Copyright (c) 2002 Apple Computer, Inc. All rights reserved.
 *
 * @APPLE_LICENSE_HEADER_START@
 * 
 * The contents of this file constitute Original Code as defined in and
 * are subject to the Apple Public Source License Version 1.1 (the
 * "License").  You may not use this file except in compliance with the
 * License.  Please obtain a copy of the License at
 * http://www.apple.com/publicsource and read it before using this file.
 * 
 * This Original Code and all software distributed under the License are
 * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT.  Please see the
 * License for the specific language governing rights and limitations
 * under the License.
 * 
 * @APPLE_LICENSE_HEADER_END@
 */
/*******************************************************************************
*                                                                              *
*     File fp_private.h                                                        *
*     Masks used for single and double floating point representations          *
*     on PowerPC.                                                              *
*                                                                              *
*******************************************************************************/
#ifndef __FP_PRIVATE__
#define __FP_PRIVATE__
#include "stdint.h"

/******************************************************************************
*       Functions used internally                                             *
******************************************************************************/
double   copysign ( double arg2, double arg1 );
double	 fabs ( double x );
double   nan   ( const char *string );

/* gcc inlines fabs() and fabsf()  */ 
#define      __FABS(x)	__builtin_fabs(x)
#define      __FABSF(x)	__builtin_fabsf(x)

#if defined(__APPLE_CC__)
#define likely(x) __builtin_expect(!!(x), 1) 
#define unlikely(x) __builtin_expect((x), 0) 
#else 
#define likely(x) (x) 
#define unlikely(x) (x) 
#endif 

#include "ppc_intrinsics.h"

#define __FMADD __fmadd
#define __FMADDS __fmadds
#define __FMSUB __fmsub
#define __FNMSUB __fnmsub
#define __FMUL __fmul
#define __FADD __fadd
#define __FSUB __fsub

static inline double __fadd (double a, double b) __attribute__((always_inline));
static inline double
__fadd (double  a, double b)
{
  double result;
  __asm__ ("fadd %0, %1, %2"
           /* outputs:  */ : "=f" (result)
           /* inputs:   */ : "f" (a), "f" (b));
  return result;
}

static inline double __fsub (double a, double b) __attribute__((always_inline));
static inline double
__fsub (double  a, double b)
{
  double result;
  __asm__ ("fsub %0, %1, %2"
           /* outputs:  */ : "=f" (result)
           /* inputs:   */ : "f" (a), "f" (b));
  return result;
}

// The following macros are invoked for side-effect. Not written as inline functions because the
// compiler could discard the code as an optimization.
#define __NOOP \
({  __label__ L1, L2; L1: (void)&&L1;\
    asm volatile ( "nop" ); /* NOOP */ \
    L2: (void)&&L2; \
})  

#define __ENSURE(x, y, z) \
({ \
    double __value, __argx = (x), __argy = (y), __argz = (z); \
    asm volatile ("fmadd %0,%1,%2,%3" : "=f" (__value): "f" (__argx), "f" (__argy), "f" (__argz)); \
    __value; \
}) 
 
#define __PROD(x, y) \
({ \
    double __value, __argx = (x), __argy = (y); \
    asm volatile ("fmul %0,%1,%2" : "=f" (__value): "f" (__argx), "f" (__argy)); \
    __value; \
})  

#define __PROG_INEXACT( x ) (void)__PROD( x, x ) /* Raises INEXACT for suitable choice of x */

#define __PROG_UF_INEXACT( x ) (void)__PROD( x, x ) /* Raises UNDERFLOW and INEXACT for suitable choice of x e.g. MIN_NORMAL */

#define __PROG_OF_INEXACT( x ) (void)__PROD( x, x ) /* Raises OVERFLOW and INEXACT for suitable choice of x e.g. MAX_NORMAL */

/******************************************************************************
*       Single precision                                                      *
******************************************************************************/

#define       fQuietNan           0x00400000

typedef union {
       int32_t       lval;
       float          fval;
} hexsingle;

/******************************************************************************
*       Double precision                                                      *
******************************************************************************/

#define       dQuietNan           0x00080000

#if defined(__BIG_ENDIAN__)

typedef union {
       struct {
		uint32_t hi;
		uint32_t lo;
	} i;
       double            d;
} hexdouble;

#define HEXDOUBLE(hi, lo) { { hi, lo } }

#elif defined(__LITTLE_ENDIAN__)

typedef union {
       struct {
		uint32_t lo;
		uint32_t hi;
	} i;
       double            d;
} hexdouble;

#define HEXDOUBLE(hi, lo) { { lo, hi } }

#else
#error Unknown endianness
#endif

typedef union {
	uint32_t i[4];
	struct {
	    hexdouble hexhead;
	    hexdouble hextail;
	} hh;
	struct {
		double head;
		double tail;
	} dd;
	long double ld;
} hexdbldbl;
	  
#endif      /* __FP_PRIVATE__ */