xref: /freebsd/lib/libc/gen/fpclassify.c (revision 8cf5ed5125ca8f57fa6f27892f3fbda25a52d354)
18cf5ed51SMike Barcroft /*-
28cf5ed51SMike Barcroft  * Copyright (c) 2003 Mike Barcroft <mike@FreeBSD.org>
38cf5ed51SMike Barcroft  * Copyright (c) 2002, 2003 David Schultz <dschultz@uclink.Berkeley.EDU>
48cf5ed51SMike Barcroft  * All rights reserved.
58cf5ed51SMike Barcroft  *
68cf5ed51SMike Barcroft  * Redistribution and use in source and binary forms, with or without
78cf5ed51SMike Barcroft  * modification, are permitted provided that the following conditions
88cf5ed51SMike Barcroft  * are met:
98cf5ed51SMike Barcroft  * 1. Redistributions of source code must retain the above copyright
108cf5ed51SMike Barcroft  *    notice, this list of conditions and the following disclaimer.
118cf5ed51SMike Barcroft  * 2. Redistributions in binary form must reproduce the above copyright
128cf5ed51SMike Barcroft  *    notice, this list of conditions and the following disclaimer in the
138cf5ed51SMike Barcroft  *    documentation and/or other materials provided with the distribution.
148cf5ed51SMike Barcroft  *
158cf5ed51SMike Barcroft  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
168cf5ed51SMike Barcroft  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
178cf5ed51SMike Barcroft  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
188cf5ed51SMike Barcroft  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
198cf5ed51SMike Barcroft  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
208cf5ed51SMike Barcroft  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
218cf5ed51SMike Barcroft  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
228cf5ed51SMike Barcroft  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
238cf5ed51SMike Barcroft  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
248cf5ed51SMike Barcroft  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
258cf5ed51SMike Barcroft  * SUCH DAMAGE.
268cf5ed51SMike Barcroft  *
278cf5ed51SMike Barcroft  * $FreeBSD$
288cf5ed51SMike Barcroft  */
298cf5ed51SMike Barcroft 
308cf5ed51SMike Barcroft #include <sys/endian.h>
318cf5ed51SMike Barcroft 
328cf5ed51SMike Barcroft #include <math.h>
338cf5ed51SMike Barcroft #include <stdint.h>
348cf5ed51SMike Barcroft 
358cf5ed51SMike Barcroft #include "fpmath.h"
368cf5ed51SMike Barcroft 
378cf5ed51SMike Barcroft int
388cf5ed51SMike Barcroft __fpclassifyf(float f)
398cf5ed51SMike Barcroft {
408cf5ed51SMike Barcroft 	union IEEEf2bits u;
418cf5ed51SMike Barcroft 
428cf5ed51SMike Barcroft 	u.f = f;
438cf5ed51SMike Barcroft 	if (u.bits.exp == 0) {
448cf5ed51SMike Barcroft 		if (u.bits.man == 0)
458cf5ed51SMike Barcroft 			return (FP_ZERO);
468cf5ed51SMike Barcroft 		return (FP_SUBNORMAL);
478cf5ed51SMike Barcroft 	}
488cf5ed51SMike Barcroft 	if (u.bits.exp == 255) {
498cf5ed51SMike Barcroft 		if (u.bits.man == 0)
508cf5ed51SMike Barcroft 			return (FP_INFINITE);
518cf5ed51SMike Barcroft 		return (FP_NAN);
528cf5ed51SMike Barcroft 	}
538cf5ed51SMike Barcroft 	return (FP_NORMAL);
548cf5ed51SMike Barcroft }
558cf5ed51SMike Barcroft 
568cf5ed51SMike Barcroft int
578cf5ed51SMike Barcroft __fpclassifyd(double d)
588cf5ed51SMike Barcroft {
598cf5ed51SMike Barcroft 	union IEEEd2bits u;
608cf5ed51SMike Barcroft 
618cf5ed51SMike Barcroft 	u.d = d;
628cf5ed51SMike Barcroft 	if (u.bits.exp == 0) {
638cf5ed51SMike Barcroft 		if ((u.bits.manl | u.bits.manh) == 0)
648cf5ed51SMike Barcroft 			return (FP_ZERO);
658cf5ed51SMike Barcroft 		return (FP_SUBNORMAL);
668cf5ed51SMike Barcroft 	}
678cf5ed51SMike Barcroft 	if (u.bits.exp == 2047) {
688cf5ed51SMike Barcroft 		if ((u.bits.manl | u.bits.manh) == 0)
698cf5ed51SMike Barcroft 			return (FP_INFINITE);
708cf5ed51SMike Barcroft 		return (FP_NAN);
718cf5ed51SMike Barcroft 	}
728cf5ed51SMike Barcroft 	return (FP_NORMAL);
738cf5ed51SMike Barcroft }
748cf5ed51SMike Barcroft 
758cf5ed51SMike Barcroft int
768cf5ed51SMike Barcroft __fpclassifyl(long double e)
778cf5ed51SMike Barcroft {
788cf5ed51SMike Barcroft 	union IEEEl2bits u;
798cf5ed51SMike Barcroft 
808cf5ed51SMike Barcroft 	u.e = e;
818cf5ed51SMike Barcroft 	if (u.bits.exp == 0) {
828cf5ed51SMike Barcroft 		if ((u.bits.manl | u.bits.manh) == 0)
838cf5ed51SMike Barcroft 			return (FP_ZERO);
848cf5ed51SMike Barcroft 		return (FP_SUBNORMAL);
858cf5ed51SMike Barcroft 	}
868cf5ed51SMike Barcroft 	mask_nbit_l(u);		/* Mask normalization bit if applicable. */
878cf5ed51SMike Barcroft 	if (u.bits.exp == 32767) {
888cf5ed51SMike Barcroft 		if ((u.bits.manl | u.bits.manh) == 0)
898cf5ed51SMike Barcroft 			return (FP_INFINITE);
908cf5ed51SMike Barcroft 		return (FP_NAN);
918cf5ed51SMike Barcroft 	}
928cf5ed51SMike Barcroft 	return (FP_NORMAL);
938cf5ed51SMike Barcroft }
94