2019-05-31 01:09:55 -07:00
// SPDX-License-Identifier: GPL-2.0-only
2005-04-16 15:20:36 -07:00
/* IEEE754 floating point arithmetic
* single precision
*/
/*
* MIPS floating point support
* Copyright ( C ) 1994 - 2000 Algorithmics Ltd .
*/
2014-04-16 00:47:59 +02:00
# include <linux/compiler.h>
2005-04-16 15:20:36 -07:00
# include "ieee754sp.h"
2014-04-16 01:31:11 +02:00
int ieee754sp_class ( union ieee754sp x )
2005-04-16 15:20:36 -07:00
{
COMPXSP ;
EXPLODEXSP ;
return xc ;
}
2015-04-03 23:25:57 +01:00
static inline int ieee754sp_isnan ( union ieee754sp x )
2005-04-16 15:20:36 -07:00
{
2015-04-03 23:25:38 +01:00
return ieee754_class_nan ( ieee754sp_class ( x ) ) ;
2005-04-16 15:20:36 -07:00
}
2014-04-25 00:40:42 +02:00
static inline int ieee754sp_issnan ( union ieee754sp x )
2005-04-16 15:20:36 -07:00
{
2015-11-13 00:47:28 +00:00
int qbit ;
2005-04-16 15:20:36 -07:00
assert ( ieee754sp_isnan ( x ) ) ;
2015-11-13 00:47:28 +00:00
qbit = ( SPMANT ( x ) & SP_MBIT ( SP_FBITS - 1 ) ) = = SP_MBIT ( SP_FBITS - 1 ) ;
return ieee754_csr . nan2008 ^ qbit ;
2005-04-16 15:20:36 -07:00
}
2015-04-03 23:25:34 +01:00
/*
* Raise the Invalid Operation IEEE 754 exception
* and convert the signaling NaN supplied to a quiet NaN .
*/
2014-04-25 03:19:57 +02:00
union ieee754sp __cold ieee754sp_nanxcpt ( union ieee754sp r )
2005-04-16 15:20:36 -07:00
{
2015-04-03 23:25:34 +01:00
assert ( ieee754sp_issnan ( r ) ) ;
2005-04-16 15:20:36 -07:00
2015-04-03 23:25:34 +01:00
ieee754_setcx ( IEEE754_INVALID_OPERATION ) ;
2015-11-13 00:48:48 +00:00
if ( ieee754_csr . nan2008 ) {
2015-11-13 00:47:28 +00:00
SPMANT ( r ) | = SP_MBIT ( SP_FBITS - 1 ) ;
2015-11-13 00:48:48 +00:00
} else {
SPMANT ( r ) & = ~ SP_MBIT ( SP_FBITS - 1 ) ;
if ( ! ieee754sp_isnan ( r ) )
SPMANT ( r ) | = SP_MBIT ( SP_FBITS - 2 ) ;
}
2015-11-13 00:47:28 +00:00
return r ;
2005-04-16 15:20:36 -07:00
}
2017-11-02 12:13:59 +01:00
static unsigned int ieee754sp_get_rounding ( int sn , unsigned int xm )
2005-04-16 15:20:36 -07:00
{
/* inexact must round of 3 bits
*/
if ( xm & ( SP_MBIT ( 3 ) - 1 ) ) {
switch ( ieee754_csr . rm ) {
2014-04-30 11:21:55 +02:00
case FPU_CSR_RZ :
2005-04-16 15:20:36 -07:00
break ;
2014-04-30 11:21:55 +02:00
case FPU_CSR_RN :
2005-04-16 15:20:36 -07:00
xm + = 0x3 + ( ( xm > > 3 ) & 1 ) ;
/* xm += (xm&0x8)?0x4:0x3 */
break ;
2014-04-30 11:21:55 +02:00
case FPU_CSR_RU : /* toward +Infinity */
2005-04-16 15:20:36 -07:00
if ( ! sn ) /* ?? */
xm + = 0x8 ;
break ;
2014-04-30 11:21:55 +02:00
case FPU_CSR_RD : /* toward -Infinity */
2013-01-22 12:59:30 +01:00
if ( sn ) /* ?? */
2005-04-16 15:20:36 -07:00
xm + = 0x8 ;
break ;
}
}
return xm ;
}
/* generate a normal/denormal number with over,under handling
* sn is sign
* xe is an unbiased exponent
* xm is 3 bit extended precision value .
*/
2017-11-02 12:13:59 +01:00
union ieee754sp ieee754sp_format ( int sn , int xe , unsigned int xm )
2005-04-16 15:20:36 -07:00
{
assert ( xm ) ; /* we don't gen exact zeros (probably should) */
2016-02-25 00:44:58 -08:00
assert ( ( xm > > ( SP_FBITS + 1 + 3 ) ) = = 0 ) ; /* no excess */
2005-04-16 15:20:36 -07:00
assert ( xm & ( SP_HIDDEN_BIT < < 3 ) ) ;
if ( xe < SP_EMIN ) {
/* strip lower bits */
int es = SP_EMIN - xe ;
if ( ieee754_csr . nod ) {
2014-04-19 00:36:32 +02:00
ieee754_setcx ( IEEE754_UNDERFLOW ) ;
ieee754_setcx ( IEEE754_INEXACT ) ;
2005-04-16 15:20:36 -07:00
switch ( ieee754_csr . rm ) {
2014-04-30 11:21:55 +02:00
case FPU_CSR_RN :
case FPU_CSR_RZ :
2005-04-16 15:20:36 -07:00
return ieee754sp_zero ( sn ) ;
2014-04-30 11:21:55 +02:00
case FPU_CSR_RU : /* toward +Infinity */
2014-04-16 11:00:12 +02:00
if ( sn = = 0 )
2005-04-16 15:20:36 -07:00
return ieee754sp_min ( 0 ) ;
else
return ieee754sp_zero ( 1 ) ;
2014-04-30 11:21:55 +02:00
case FPU_CSR_RD : /* toward -Infinity */
2014-04-16 11:00:12 +02:00
if ( sn = = 0 )
2005-04-16 15:20:36 -07:00
return ieee754sp_zero ( 0 ) ;
else
return ieee754sp_min ( 1 ) ;
}
}
2014-04-25 10:54:06 +02:00
if ( xe = = SP_EMIN - 1 & &
ieee754sp_get_rounding ( sn , xm ) > > ( SP_FBITS + 1 + 3 ) )
2005-04-16 15:20:36 -07:00
{
/* Not tiny after rounding */
2014-04-19 00:36:32 +02:00
ieee754_setcx ( IEEE754_INEXACT ) ;
2014-04-25 10:54:06 +02:00
xm = ieee754sp_get_rounding ( sn , xm ) ;
2005-04-16 15:20:36 -07:00
xm > > = 1 ;
/* Clear grs bits */
xm & = ~ ( SP_MBIT ( 3 ) - 1 ) ;
xe + + ;
2014-04-16 11:00:12 +02:00
} else {
2005-04-16 15:20:36 -07:00
/* sticky right shift es bits
*/
2016-04-21 14:04:54 +01:00
xm = XSPSRS ( xm , es ) ;
xe + = es ;
2005-04-16 15:20:36 -07:00
assert ( ( xm & ( SP_HIDDEN_BIT < < 3 ) ) = = 0 ) ;
assert ( xe = = SP_EMIN ) ;
}
}
if ( xm & ( SP_MBIT ( 3 ) - 1 ) ) {
2014-04-19 00:36:32 +02:00
ieee754_setcx ( IEEE754_INEXACT ) ;
2005-04-16 15:20:36 -07:00
if ( ( xm & ( SP_HIDDEN_BIT < < 3 ) ) = = 0 ) {
2014-04-19 00:36:32 +02:00
ieee754_setcx ( IEEE754_UNDERFLOW ) ;
2005-04-16 15:20:36 -07:00
}
/* inexact must round of 3 bits
*/
2014-04-25 10:54:06 +02:00
xm = ieee754sp_get_rounding ( sn , xm ) ;
2005-04-16 15:20:36 -07:00
/* adjust exponent for rounding add overflowing
*/
2014-04-22 15:51:55 +02:00
if ( xm > > ( SP_FBITS + 1 + 3 ) ) {
2005-04-16 15:20:36 -07:00
/* add causes mantissa overflow */
xm > > = 1 ;
xe + + ;
}
}
/* strip grs bits */
xm > > = 3 ;
2016-02-25 00:44:58 -08:00
assert ( ( xm > > ( SP_FBITS + 1 ) ) = = 0 ) ; /* no excess */
2005-04-16 15:20:36 -07:00
assert ( xe > = SP_EMIN ) ;
if ( xe > SP_EMAX ) {
2014-04-19 00:36:32 +02:00
ieee754_setcx ( IEEE754_OVERFLOW ) ;
ieee754_setcx ( IEEE754_INEXACT ) ;
2005-04-16 15:20:36 -07:00
/* -O can be table indexed by (rm,sn) */
switch ( ieee754_csr . rm ) {
2014-04-30 11:21:55 +02:00
case FPU_CSR_RN :
2005-04-16 15:20:36 -07:00
return ieee754sp_inf ( sn ) ;
2014-04-30 11:21:55 +02:00
case FPU_CSR_RZ :
2005-04-16 15:20:36 -07:00
return ieee754sp_max ( sn ) ;
2014-04-30 11:21:55 +02:00
case FPU_CSR_RU : /* toward +Infinity */
2005-04-16 15:20:36 -07:00
if ( sn = = 0 )
return ieee754sp_inf ( 0 ) ;
else
return ieee754sp_max ( 1 ) ;
2014-04-30 11:21:55 +02:00
case FPU_CSR_RD : /* toward -Infinity */
2005-04-16 15:20:36 -07:00
if ( sn = = 0 )
return ieee754sp_max ( 0 ) ;
else
return ieee754sp_inf ( 1 ) ;
}
}
/* gen norm/denorm/zero */
if ( ( xm & SP_HIDDEN_BIT ) = = 0 ) {
/* we underflow (tiny/zero) */
assert ( xe = = SP_EMIN ) ;
if ( ieee754_csr . mx & IEEE754_UNDERFLOW )
2014-04-19 00:36:32 +02:00
ieee754_setcx ( IEEE754_UNDERFLOW ) ;
2005-04-16 15:20:36 -07:00
return buildsp ( sn , SP_EMIN - 1 + SP_EBIAS , xm ) ;
} else {
2016-02-25 00:44:58 -08:00
assert ( ( xm > > ( SP_FBITS + 1 ) ) = = 0 ) ; /* no excess */
2005-04-16 15:20:36 -07:00
assert ( xm & SP_HIDDEN_BIT ) ;
return buildsp ( sn , xe + SP_EBIAS , xm & ~ SP_HIDDEN_BIT ) ;
}
}