xmm_nextafter.c   [plain text]


/*
 *  xmm_nextafter.c
 *  xmmLibm
 *
 *  Created by Ian Ollmann on 8/19/05.
 *  Copyright 2005 Apple Computer. All rights reserved.
 *
 */

#include "fenv.h"
#include "xmmLibm_prefix.h"

static inline double _nextafter( double x, double y ) ALWAYS_INLINE;
static inline double _nextafter( double x, double y )
{
    static const double smallest = 0x0.0000000000001p-1022;
    static const double tiny = 0x1.0000000000000p-1022;

    //must be a x or y is NaN
    if( EXPECT_FALSE( x != x ) )
        return x + x;
    
    if( EXPECT_TRUE( x < y ) )
    {
		if( EXPECT_FALSE( x == - __builtin_inf() ) )
			return -0x1.fffffffffffffp1023;

        int oldmxcsr = _mm_getcsr();
        int newmxcsr = (oldmxcsr & ~ROUND_MASK ) | ROUND_TO_INFINITY;
        _mm_setcsr( newmxcsr );
         
        x += smallest;
    
		int test = __builtin_fabs( x ) < tiny;
		oldmxcsr |= -test & ( UNDERFLOW_FLAG | INEXACT_FLAG );
	
        oldmxcsr |= _mm_getcsr() & ALL_FLAGS;
        _mm_setcsr( oldmxcsr );
        return x;
    }

    if( EXPECT_TRUE( x > y ) )
    {
		if( EXPECT_FALSE( x == __builtin_inf() ) )
			return 0x1.fffffffffffffp1023;

        int oldmxcsr = _mm_getcsr();
        int newmxcsr = (oldmxcsr & ~ROUND_MASK ) | ROUND_TO_NEG_INFINITY;
        _mm_setcsr( newmxcsr );
         
        x -= smallest;
    
		int test = __builtin_fabs( x ) < tiny;
		oldmxcsr |= -test & ( UNDERFLOW_FLAG | INEXACT_FLAG );

        oldmxcsr |= _mm_getcsr() & ALL_FLAGS;
        _mm_setcsr( oldmxcsr );
        return x;
    }

    if( EXPECT_TRUE( x == y ) )
        return y;
        
    return y + y;
}

#if defined( BUILDING_FOR_CARBONCORE_LEGACY )


float nextafterf( float x, float y )
{
    static const float smallest = 0x0.000002p-126f;
    static const float tiny = 0x1.000000p-126f;

    //must be a x or y is NaN
    if( EXPECT_FALSE( x != x ) )
        return x + x;

    if( EXPECT_TRUE( x < y ) )
    {
		if( EXPECT_FALSE( x == - __builtin_inff() ) )
			return -0x1.fffffep127f;

        int oldmxcsr = _mm_getcsr();
        int newmxcsr = (oldmxcsr & ~ROUND_MASK ) | ROUND_TO_INFINITY;
        _mm_setcsr( newmxcsr );
         
        x += smallest;

		int test = __builtin_fabsf( x ) < tiny;
		oldmxcsr |= -test & ( UNDERFLOW_FLAG | INEXACT_FLAG );
    
        oldmxcsr |= _mm_getcsr() & ALL_FLAGS;
        _mm_setcsr( oldmxcsr );
        return x;
    }
    
    if( EXPECT_TRUE( x > y ) )
    {
		if( EXPECT_FALSE( x == __builtin_inff() ) )
			return 0x1.fffffep127f;

        int oldmxcsr = _mm_getcsr();
        int newmxcsr = (oldmxcsr & ~ROUND_MASK ) | ROUND_TO_NEG_INFINITY;
        _mm_setcsr( newmxcsr );
         
        x -= smallest;

		int test = __builtin_fabsf( x ) < tiny;
		oldmxcsr |= -test & ( UNDERFLOW_FLAG | INEXACT_FLAG );
    
        oldmxcsr |= _mm_getcsr() & ALL_FLAGS;
        _mm_setcsr( oldmxcsr );
        return x;
    }

    if( EXPECT_TRUE( x == y ) )
        return y;
        
    return y + y;
}

double nextafterd( double x, double y )
{
	return _nextafter( x, y );
}

#else

double nextafter( double x, double y )
{
	return _nextafter( x, y );
}


long double nextafterl( long double x, long double y )
{
    static const long double smallest = 0x0.0000000000000002p-16382L;
    static const long double tiny = 0x1.0000000000000000p-16382L;
    struct
	{
			uint32_t	control;
			uint32_t	status;
			uint32_t	tag;
			uint32_t	ip;
			uint32_t	stuff;
			uint32_t	op;
			uint32_t	stuff2;
	}env;

    //must be a x or y is NaN
    if( EXPECT_FALSE( x != x ) )
        return x + x;

    if( EXPECT_TRUE( x < y ) )
    {
		if( EXPECT_FALSE( x == - __builtin_infl() ) )
			return -0x1.fffffffffffffffep16383L;
	
        asm volatile( "fnstenv %0" : "=m" (*&env ));
		uint16_t newcontrol = (env.control & 0xf3ff) | 0x0800;
        asm volatile( "fldcw %0" : : "m" (*&newcontrol ));
         
        long double result = x + smallest;
    
		int test = __builtin_fabsl( x ) < tiny;
		env.status |= -test & 0x30;
        asm volatile( "fldenv %0" : : "m" (*&env ));
        return result;
    }

    if( EXPECT_TRUE( x > y ) )
    {
		if( EXPECT_FALSE( x == __builtin_infl() ) )
			return 0x1.fffffffffffffffep16383L;

        asm volatile( "fnstenv %0" : "=m" (*&env ));
		uint16_t newcontrol = (env.control & 0xf3ff) | 0x0400;
        asm volatile( "fldcw %0" : : "m" (*&newcontrol ));
         
        long double result = x - smallest;
    
		int test = __builtin_fabsl( x ) < tiny;
		env.status |= -test & 0x30;
        asm volatile( "fldenv %0" : : "m" (*&env ));
        return result;
    }

    if( EXPECT_TRUE( x == y ) )
        return y;
        
    return y + y;
}

float nexttowardf( float x, long double y )
{
    return nextafterl( x, y );
}

double nexttoward( double x, long double y )
{
    return nextafterl( x, y );
}

long double nexttowardl( long double x, long double y )
{
    return nextafterl( x, y );
}

#endif