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