xref: /titanic_51/usr/src/head/ieeefp.h (revision ba3594ba9b5dd4c846c472a8d657edcb7c8109ac)
17c478bd9Sstevel@tonic-gate /*
27c478bd9Sstevel@tonic-gate  * CDDL HEADER START
37c478bd9Sstevel@tonic-gate  *
47c478bd9Sstevel@tonic-gate  * The contents of this file are subject to the terms of the
57c478bd9Sstevel@tonic-gate  * Common Development and Distribution License, Version 1.0 only
67c478bd9Sstevel@tonic-gate  * (the "License").  You may not use this file except in compliance
77c478bd9Sstevel@tonic-gate  * with the License.
87c478bd9Sstevel@tonic-gate  *
97c478bd9Sstevel@tonic-gate  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
107c478bd9Sstevel@tonic-gate  * or http://www.opensolaris.org/os/licensing.
117c478bd9Sstevel@tonic-gate  * See the License for the specific language governing permissions
127c478bd9Sstevel@tonic-gate  * and limitations under the License.
137c478bd9Sstevel@tonic-gate  *
147c478bd9Sstevel@tonic-gate  * When distributing Covered Code, include this CDDL HEADER in each
157c478bd9Sstevel@tonic-gate  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
167c478bd9Sstevel@tonic-gate  * If applicable, add the following below this CDDL HEADER, with the
177c478bd9Sstevel@tonic-gate  * fields enclosed by brackets "[]" replaced with your own identifying
187c478bd9Sstevel@tonic-gate  * information: Portions Copyright [yyyy] [name of copyright owner]
197c478bd9Sstevel@tonic-gate  *
207c478bd9Sstevel@tonic-gate  * CDDL HEADER END
217c478bd9Sstevel@tonic-gate  */
227c478bd9Sstevel@tonic-gate /*
23*ba3594baSGarrett D'Amore  * Copyright 2014 Garrett D'Amore <garrett@damore.org>
24*ba3594baSGarrett D'Amore  *
257c478bd9Sstevel@tonic-gate  * Copyright 2005 Sun Microsystems, Inc.  All rights reserved.
267c478bd9Sstevel@tonic-gate  * Use is subject to license terms.
277c478bd9Sstevel@tonic-gate  */
287c478bd9Sstevel@tonic-gate 
297c478bd9Sstevel@tonic-gate /*	Copyright (c) 1988 AT&T */
307c478bd9Sstevel@tonic-gate /*	  All Rights Reserved */
317c478bd9Sstevel@tonic-gate 
327c478bd9Sstevel@tonic-gate 
337c478bd9Sstevel@tonic-gate #ifndef _IEEEFP_H
347c478bd9Sstevel@tonic-gate #define	_IEEEFP_H
357c478bd9Sstevel@tonic-gate 
367c478bd9Sstevel@tonic-gate #ifdef	__cplusplus
377c478bd9Sstevel@tonic-gate extern "C" {
387c478bd9Sstevel@tonic-gate #endif
397c478bd9Sstevel@tonic-gate 
407c478bd9Sstevel@tonic-gate /*
417c478bd9Sstevel@tonic-gate  * Floating point enviornment for machines that support
427c478bd9Sstevel@tonic-gate  * the IEEE 754 floating-point standard.  This file currently
437c478bd9Sstevel@tonic-gate  * supports the 80*87, and SPARC families.
447c478bd9Sstevel@tonic-gate  *
457c478bd9Sstevel@tonic-gate  * This header defines the following interfaces:
467c478bd9Sstevel@tonic-gate  *	1) Classes of floating point numbers
477c478bd9Sstevel@tonic-gate  *	2) Rounding Control
487c478bd9Sstevel@tonic-gate  *	3) Exception Control
497c478bd9Sstevel@tonic-gate  *	4) Exception Handling
507c478bd9Sstevel@tonic-gate  *	5) Utility Macros
517c478bd9Sstevel@tonic-gate  *	6) Full Exception Environment Control
527c478bd9Sstevel@tonic-gate  */
537c478bd9Sstevel@tonic-gate 
547c478bd9Sstevel@tonic-gate /*
557c478bd9Sstevel@tonic-gate  * CLASSES of floating point numbers *************************
567c478bd9Sstevel@tonic-gate  * IEEE floating point values fall into 1 of the following 10
577c478bd9Sstevel@tonic-gate  * classes
587c478bd9Sstevel@tonic-gate  */
597c478bd9Sstevel@tonic-gate typedef	enum	fpclass_t {
607c478bd9Sstevel@tonic-gate 	FP_SNAN = 0,	/* signaling NaN */
617c478bd9Sstevel@tonic-gate 	FP_QNAN = 1,	/* quiet NaN */
627c478bd9Sstevel@tonic-gate 	FP_NINF = 2,	/* negative infinity */
637c478bd9Sstevel@tonic-gate 	FP_PINF = 3,	/* positive infinity */
647c478bd9Sstevel@tonic-gate 	FP_NDENORM = 4, /* negative denormalized non-zero */
657c478bd9Sstevel@tonic-gate 	FP_PDENORM = 5, /* positive denormalized non-zero */
667c478bd9Sstevel@tonic-gate 	FP_NZERO = 6,	/* -0.0 */
677c478bd9Sstevel@tonic-gate 	FP_PZERO = 7,   /* +0.0 */
687c478bd9Sstevel@tonic-gate 	FP_NNORM = 8,	/* negative normalized non-zero */
697c478bd9Sstevel@tonic-gate 	FP_PNORM = 9	/* positive normalized non-zero */
707c478bd9Sstevel@tonic-gate } fpclass_t;
717c478bd9Sstevel@tonic-gate 
727c478bd9Sstevel@tonic-gate extern fpclass_t fpclass(double);	/* get class of double value */
737c478bd9Sstevel@tonic-gate extern int	finite(double);
747c478bd9Sstevel@tonic-gate extern int	unordered(double, double);
757c478bd9Sstevel@tonic-gate 
767c478bd9Sstevel@tonic-gate /*
777c478bd9Sstevel@tonic-gate  * ROUNDING CONTROL ******************************************
787c478bd9Sstevel@tonic-gate  *
797c478bd9Sstevel@tonic-gate  * At all times, floating-point math is done using one of four
807c478bd9Sstevel@tonic-gate  * mutually-exclusive rounding modes.
817c478bd9Sstevel@tonic-gate  */
827c478bd9Sstevel@tonic-gate 
837c478bd9Sstevel@tonic-gate #if defined(__i386) || defined(__amd64)
847c478bd9Sstevel@tonic-gate 
857c478bd9Sstevel@tonic-gate /*
867c478bd9Sstevel@tonic-gate  * NOTE: the values given are chosen to match those used by the
877c478bd9Sstevel@tonic-gate  * 80*87 rounding mode field in the control word.
887c478bd9Sstevel@tonic-gate  */
897c478bd9Sstevel@tonic-gate typedef	enum	fp_rnd {
907c478bd9Sstevel@tonic-gate 	FP_RN = 0,	/* round to nearest representable number, tie -> even */
917c478bd9Sstevel@tonic-gate 	FP_RM = 1,	/* round toward minus infinity */
927c478bd9Sstevel@tonic-gate 	FP_RP = 2,	/* round toward plus infinity */
937c478bd9Sstevel@tonic-gate 	FP_RZ = 3	/* round toward zero (truncate) */
947c478bd9Sstevel@tonic-gate } fp_rnd;
957c478bd9Sstevel@tonic-gate 
967c478bd9Sstevel@tonic-gate #endif
977c478bd9Sstevel@tonic-gate 
987c478bd9Sstevel@tonic-gate #if defined(__sparc)
997c478bd9Sstevel@tonic-gate 
1007c478bd9Sstevel@tonic-gate /*
1017c478bd9Sstevel@tonic-gate  * NOTE: the values given are chosen to match those used by the
1027c478bd9Sstevel@tonic-gate  * RD (Round Direction) field of the FSR (Floating Point State Register).
1037c478bd9Sstevel@tonic-gate  */
1047c478bd9Sstevel@tonic-gate typedef	enum	fp_rnd {
1057c478bd9Sstevel@tonic-gate 	FP_RN = 0,	/* round to nearest representable number, tie -> even */
1067c478bd9Sstevel@tonic-gate 	FP_RZ = 1,	/* round toward zero (truncate) */
1077c478bd9Sstevel@tonic-gate 	FP_RP = 2,	/* round toward plus infinity */
1087c478bd9Sstevel@tonic-gate 	FP_RM = 3	/* round toward minus infinity */
1097c478bd9Sstevel@tonic-gate } fp_rnd;
1107c478bd9Sstevel@tonic-gate 
1117c478bd9Sstevel@tonic-gate #endif
1127c478bd9Sstevel@tonic-gate 
1137c478bd9Sstevel@tonic-gate extern fp_rnd	fpsetround(fp_rnd);	/* set rounding mode, return previous */
1147c478bd9Sstevel@tonic-gate extern fp_rnd	fpgetround(void);	/* return current rounding mode */
1157c478bd9Sstevel@tonic-gate 
1167c478bd9Sstevel@tonic-gate /*
1177c478bd9Sstevel@tonic-gate  * EXCEPTION CONTROL *****************************************
1187c478bd9Sstevel@tonic-gate  *
1197c478bd9Sstevel@tonic-gate  */
1207c478bd9Sstevel@tonic-gate 
1217c478bd9Sstevel@tonic-gate #define	fp_except	int
1227c478bd9Sstevel@tonic-gate 
1237c478bd9Sstevel@tonic-gate #define	FP_DISABLE	0	/* exception will be ignored */
1247c478bd9Sstevel@tonic-gate #define	FP_ENABLE	1	/* exception will cause SIGFPE */
1257c478bd9Sstevel@tonic-gate #define	FP_CLEAR	0	/* exception has not occurred */
1267c478bd9Sstevel@tonic-gate #define	FP_SET		1	/* exception has occurred */
1277c478bd9Sstevel@tonic-gate 
1287c478bd9Sstevel@tonic-gate #if defined(__i386) || defined(__amd64)
1297c478bd9Sstevel@tonic-gate 
1307c478bd9Sstevel@tonic-gate /*
1317c478bd9Sstevel@tonic-gate  * There are six floating point exceptions, which can be individually
1327c478bd9Sstevel@tonic-gate  * ENABLED (== 1) or DISABLED (== 0).  When an exception occurs
1337c478bd9Sstevel@tonic-gate  * (ENABLED or not), the fact is noted by changing an associated
1347c478bd9Sstevel@tonic-gate  * "sticky bit" from CLEAR (==0) to SET (==1).
1357c478bd9Sstevel@tonic-gate  *
1367c478bd9Sstevel@tonic-gate  * NOTE: the bit positions in fp_except are chosen to match those of
1377c478bd9Sstevel@tonic-gate  * the 80*87 control word mask bits.  Although the 87 chips actually
1387c478bd9Sstevel@tonic-gate  * ENABLE exceptions with a mask value of 0 (not 1, as on the 3b), it
1397c478bd9Sstevel@tonic-gate  * is felt that switching these values may create more problems than
1407c478bd9Sstevel@tonic-gate  * it solves.
1417c478bd9Sstevel@tonic-gate  */
1427c478bd9Sstevel@tonic-gate 
1437c478bd9Sstevel@tonic-gate /* an fp_except can have the following (not exclusive) values: */
1447c478bd9Sstevel@tonic-gate #define	FP_X_INV	0x01	/* invalid operation exception */
1457c478bd9Sstevel@tonic-gate #define	FP_X_DNML	0x02	/* denormalization exception */
1467c478bd9Sstevel@tonic-gate #define	FP_X_DZ		0x04	/* divide-by-zero exception */
1477c478bd9Sstevel@tonic-gate #define	FP_X_OFL	0x08	/* overflow exception */
1487c478bd9Sstevel@tonic-gate #define	FP_X_UFL	0x10	/* underflow exception */
1497c478bd9Sstevel@tonic-gate #define	FP_X_IMP	0x20	/* imprecise (loss of precision) */
1507c478bd9Sstevel@tonic-gate 
1517c478bd9Sstevel@tonic-gate #endif
1527c478bd9Sstevel@tonic-gate 
1537c478bd9Sstevel@tonic-gate #if defined(__sparc)
1547c478bd9Sstevel@tonic-gate 
1557c478bd9Sstevel@tonic-gate /*
1567c478bd9Sstevel@tonic-gate  * There are five floating-point exceptions, which can be individually
1577c478bd9Sstevel@tonic-gate  * ENABLED (== 1) or DISABLED (== 0).  When an exception occurs
1587c478bd9Sstevel@tonic-gate  * (ENABLED or not), the fact is noted by changing an associated
1597c478bd9Sstevel@tonic-gate  * "sticky bit" from CLEAR (==0) to SET (==1).
1607c478bd9Sstevel@tonic-gate  *
1617c478bd9Sstevel@tonic-gate  * NOTE: the bit positions in an fp_except are chosen to match that in
1627c478bd9Sstevel@tonic-gate  * the Trap Enable Mask of the FSR (Floating Point State Register).
1637c478bd9Sstevel@tonic-gate  */
1647c478bd9Sstevel@tonic-gate 
1657c478bd9Sstevel@tonic-gate /* an fp_except can have the following (not exclusive) values: */
1667c478bd9Sstevel@tonic-gate #define	FP_X_INV	0x10	/* invalid operation exception */
1677c478bd9Sstevel@tonic-gate #define	FP_X_OFL	0x08	/* overflow exception */
1687c478bd9Sstevel@tonic-gate #define	FP_X_UFL	0x04	/* underflow exception */
1697c478bd9Sstevel@tonic-gate #define	FP_X_DZ		0x02	/* divide-by-zero exception */
1707c478bd9Sstevel@tonic-gate #define	FP_X_IMP	0x01	/* imprecise (loss of precision) */
1717c478bd9Sstevel@tonic-gate 
1727c478bd9Sstevel@tonic-gate #endif
1737c478bd9Sstevel@tonic-gate 
1747c478bd9Sstevel@tonic-gate extern fp_except fpgetmask(void);		/* current exception mask */
1757c478bd9Sstevel@tonic-gate extern fp_except fpsetmask(fp_except);		/* set mask, return previous */
1767c478bd9Sstevel@tonic-gate extern fp_except fpgetsticky(void);		/* return logged exceptions */
1777c478bd9Sstevel@tonic-gate extern fp_except fpsetsticky(fp_except);	/* change logged exceptions */
1787c478bd9Sstevel@tonic-gate 
1797c478bd9Sstevel@tonic-gate /*
1807c478bd9Sstevel@tonic-gate  * UTILITY MACROS ********************************************
1817c478bd9Sstevel@tonic-gate  */
1827c478bd9Sstevel@tonic-gate 
1837c478bd9Sstevel@tonic-gate extern int isnanf(float);
1847c478bd9Sstevel@tonic-gate extern int isnand(double);
1857c478bd9Sstevel@tonic-gate 
1867c478bd9Sstevel@tonic-gate #if defined(__i386) || defined(__amd64)
1877c478bd9Sstevel@tonic-gate 
1887c478bd9Sstevel@tonic-gate /*
1897c478bd9Sstevel@tonic-gate  * EXCEPTION HANDLING ****************************************
1907c478bd9Sstevel@tonic-gate  *
1917c478bd9Sstevel@tonic-gate  * When a signal handler catches an FPE, it will have a freshly initialized
1927c478bd9Sstevel@tonic-gate  * coprocessor.  This allows signal handling routines to make use of
1937c478bd9Sstevel@tonic-gate  * floating point arithmetic, if need be.  The previous state of the 87
1947c478bd9Sstevel@tonic-gate  * chip is available, however.  There are two ways to get at this information,
1957c478bd9Sstevel@tonic-gate  * depending on how the signal handler was set up.
1967c478bd9Sstevel@tonic-gate  *
1977c478bd9Sstevel@tonic-gate  * If the handler was set via signal() or sigset(), the old, SVR3, method
1987c478bd9Sstevel@tonic-gate  * should be used: the signal handler assumes that it has a single parameter,
1997c478bd9Sstevel@tonic-gate  * which is of type struct _fpstackframe, defined below.  By investigating
2007c478bd9Sstevel@tonic-gate  * this parameter, the cause of the FPE may be determined.  By modifying it,
2017c478bd9Sstevel@tonic-gate  * the state of the coprocessor can be changed upon return to the main task.
2027c478bd9Sstevel@tonic-gate  * THIS METHOD IS OBSOLETE, AND MAY NOT BE SUPPORTED IN FUTURE RELEASES.
2037c478bd9Sstevel@tonic-gate  *
2047c478bd9Sstevel@tonic-gate  * If the handler was set via sigaction(), the new, SVR4, method should be
2057c478bd9Sstevel@tonic-gate  * used: the third argument to the handler will be a pointer to a ucontext
2067c478bd9Sstevel@tonic-gate  * structure (see sys/ucontext.h).  The uc_mcontext.fpregs member of the
2077c478bd9Sstevel@tonic-gate  * ucontext structure holds the saved floating-point registers.  This can be
2087c478bd9Sstevel@tonic-gate  * examined and/or modified.  By modifying it, the state of the coprocessor
2097c478bd9Sstevel@tonic-gate  * can be changed upon return to the main task.
2107c478bd9Sstevel@tonic-gate  */
2117c478bd9Sstevel@tonic-gate 
2127c478bd9Sstevel@tonic-gate struct _fpreg {	/* structure of a temp real fp register */
2137c478bd9Sstevel@tonic-gate 	unsigned short significand[4];	/* 64 bit mantissa value */
2147c478bd9Sstevel@tonic-gate 	unsigned short exponent;	/* 15 bit exponent and sign bit */
2157c478bd9Sstevel@tonic-gate };
2167c478bd9Sstevel@tonic-gate 
2177c478bd9Sstevel@tonic-gate #if defined(__i386)
2187c478bd9Sstevel@tonic-gate 
2197c478bd9Sstevel@tonic-gate /*
2207c478bd9Sstevel@tonic-gate  * AMD64 users should use sigaction() as described above.
2217c478bd9Sstevel@tonic-gate  */
2227c478bd9Sstevel@tonic-gate 
2237c478bd9Sstevel@tonic-gate struct _fpstackframe {		/* signal handler's argument */
2247c478bd9Sstevel@tonic-gate 	long signo;		/* signal number arg */
2257c478bd9Sstevel@tonic-gate 	long regs[19];		/* all registers */
2267c478bd9Sstevel@tonic-gate 	struct _fpstate *fpsp;	/* address of saved 387 state */
2277c478bd9Sstevel@tonic-gate 	char *wsp;		/* address of saved Weitek state */
2287c478bd9Sstevel@tonic-gate };
2297c478bd9Sstevel@tonic-gate 
2307c478bd9Sstevel@tonic-gate #endif
2317c478bd9Sstevel@tonic-gate 
2327c478bd9Sstevel@tonic-gate #if defined(__i386) || defined(__amd64)
2337c478bd9Sstevel@tonic-gate 
2347c478bd9Sstevel@tonic-gate #if defined(__amd64)
2357c478bd9Sstevel@tonic-gate #define	_fpstate _fpstate32
2367c478bd9Sstevel@tonic-gate #endif
2377c478bd9Sstevel@tonic-gate 
2387c478bd9Sstevel@tonic-gate struct _fpstate {		/* saved state info from an exception */
2397c478bd9Sstevel@tonic-gate 	unsigned int	cw,	/* control word */
2407c478bd9Sstevel@tonic-gate 			sw,	/* status word after fnclex-not useful */
2417c478bd9Sstevel@tonic-gate 			tag,	/* tag word */
2427c478bd9Sstevel@tonic-gate 			ipoff,	/* %eip register */
2437c478bd9Sstevel@tonic-gate 			cssel,	/* code segment selector */
2447c478bd9Sstevel@tonic-gate 			dataoff, /* data operand address */
2457c478bd9Sstevel@tonic-gate 			datasel; /* data operand selector */
2467c478bd9Sstevel@tonic-gate 	struct _fpreg _st[8];	/* saved register stack */
2477c478bd9Sstevel@tonic-gate 	unsigned int status;	/* status word saved at exception */
2487c478bd9Sstevel@tonic-gate 	unsigned int mxcsr;
2497c478bd9Sstevel@tonic-gate 	unsigned int xstatus;	/* status word saved at exception */
2507c478bd9Sstevel@tonic-gate 	unsigned int __pad[2];
2517c478bd9Sstevel@tonic-gate 	unsigned int xmm[8][4];
2527c478bd9Sstevel@tonic-gate };
2537c478bd9Sstevel@tonic-gate 
2547c478bd9Sstevel@tonic-gate #if defined(__amd64)
2557c478bd9Sstevel@tonic-gate #undef	_fpstate
2567c478bd9Sstevel@tonic-gate #endif
2577c478bd9Sstevel@tonic-gate 
2587c478bd9Sstevel@tonic-gate #endif	/* __i386 || __amd64 */
2597c478bd9Sstevel@tonic-gate 
2607c478bd9Sstevel@tonic-gate /*
2617c478bd9Sstevel@tonic-gate  * The structure of the 80*87 status and control words, and the mxcsr
2627c478bd9Sstevel@tonic-gate  * register are given by the following structures.
2637c478bd9Sstevel@tonic-gate  */
2647c478bd9Sstevel@tonic-gate struct _cw87 {
2657c478bd9Sstevel@tonic-gate 	unsigned
2667c478bd9Sstevel@tonic-gate 		mask:	6,	/* exception masks */
2677c478bd9Sstevel@tonic-gate 		res1:	2,	/* not used */
2687c478bd9Sstevel@tonic-gate 		prec:	2,	/* precision control field */
2697c478bd9Sstevel@tonic-gate 		rnd:	2,	/* rounding control field */
2707c478bd9Sstevel@tonic-gate 		inf:	1,	/* infinity control (not on 387) */
2717c478bd9Sstevel@tonic-gate 		res2:	3;	/* not used */
2727c478bd9Sstevel@tonic-gate };
2737c478bd9Sstevel@tonic-gate 
2747c478bd9Sstevel@tonic-gate struct _sw87 {
2757c478bd9Sstevel@tonic-gate 	unsigned
2767c478bd9Sstevel@tonic-gate 		excp:	6,	/* exception sticky bits */
2777c478bd9Sstevel@tonic-gate 		res1:	1,	/* not used */
2787c478bd9Sstevel@tonic-gate 		errs:	1,	/* error summary-set if unmasked excp */
2797c478bd9Sstevel@tonic-gate 		c012:	3,	/* condition code bits 0..2 */
2807c478bd9Sstevel@tonic-gate 		stkt:	3,	/* stack top pointer */
2817c478bd9Sstevel@tonic-gate 		c3:	1,	/* condition code bit 3 */
2827c478bd9Sstevel@tonic-gate 		busy:	1;	/* coprocessor busy */
2837c478bd9Sstevel@tonic-gate };
2847c478bd9Sstevel@tonic-gate 
2857c478bd9Sstevel@tonic-gate struct _mxcsr {
2867c478bd9Sstevel@tonic-gate 	unsigned
2877c478bd9Sstevel@tonic-gate 		excp:	6,	/* exception sticky bits */
2887c478bd9Sstevel@tonic-gate 		daz:	1,	/* denormals are zeroes */
2897c478bd9Sstevel@tonic-gate 		mask:	6,	/* exception masks */
2907c478bd9Sstevel@tonic-gate 		rnd:	2,	/* rounding control */
2917c478bd9Sstevel@tonic-gate 		fzero:	1;	/* flush to zero */
2927c478bd9Sstevel@tonic-gate };
2937c478bd9Sstevel@tonic-gate 
2947c478bd9Sstevel@tonic-gate #endif
2957c478bd9Sstevel@tonic-gate 
2967c478bd9Sstevel@tonic-gate #ifdef	__cplusplus
2977c478bd9Sstevel@tonic-gate }
2987c478bd9Sstevel@tonic-gate #endif
2997c478bd9Sstevel@tonic-gate 
3007c478bd9Sstevel@tonic-gate #endif	/* _IEEEFP_H */
301