ref: a7eb8b11e4070683ded68913802c259f193afb9f
dir: /sys/src/libc/port/frexp.c/
#include <u.h> #include <libc.h> /* * this is big/little endian non-portable * it gets the endian from the FPdbleword * union in u.h. */ #define MASK 0x7ffL #define SHIFT 20 #define BIAS 1022L #define SIG 52 double frexp(double d, int *ep) { FPdbleword x, a; *ep = 0; /* order matters: only isNaN can operate on NaN */ if(isNaN(d) || isInf(d, 0) || d == 0) return d; x.x = d; a.x = fabs(d); if((a.hi >> SHIFT) == 0){ /* normalize subnormal numbers */ x.x = (double)(1ULL<<SIG) * d; *ep = -SIG; } *ep += ((x.hi >> SHIFT) & MASK) - BIAS; x.hi &= ~(MASK << SHIFT); x.hi |= BIAS << SHIFT; return x.x; } double ldexp(double d, int deltae) { int e, bits; FPdbleword x; ulong z; if(d == 0) return 0; x.x = d; e = (x.hi >> SHIFT) & MASK; if(deltae >= 0 || e+deltae >= 1){ /* no underflow */ e += deltae; if(e >= MASK){ /* overflow */ if(d < 0) return Inf(-1); return Inf(1); } }else{ /* underflow gracefully */ deltae = -deltae; /* need to shift d right deltae */ if(e > 1){ /* shift e-1 by exponent manipulation */ deltae -= e-1; e = 1; } if(deltae > 0 && e==1){ /* shift 1 by switch from 1.fff to 0.1ff */ deltae--; e = 0; x.lo >>= 1; x.lo |= (x.hi&1)<<31; z = x.hi & ((1<<SHIFT)-1); x.hi &= ~((1<<SHIFT)-1); x.hi |= (1<<(SHIFT-1)) | (z>>1); } while(deltae > 0){ /* shift bits down */ bits = deltae; if(bits > SHIFT) bits = SHIFT; x.lo >>= bits; x.lo |= (x.hi&((1<<bits)-1)) << (32-bits); z = x.hi & ((1<<SHIFT)-1); x.hi &= ~((1<<SHIFT)-1); x.hi |= z>>bits; deltae -= bits; } } x.hi &= ~(MASK << SHIFT); x.hi |= (long)e << SHIFT; return x.x; } double modf(double d, double *ip) { FPdbleword x; int e; x.x = d; e = (x.hi >> SHIFT) & MASK; if(e == MASK){ *ip = d; if(x.lo != 0 || (x.hi & 0xfffffL) != 0) /* NaN */ return d; /* ±Inf */ x.hi &= 0x80000000L; return x.x; } if(d < 1) { if(d < 0) { x.x = modf(-d, ip); *ip = -*ip; return -x.x; } *ip = 0; return d; } e -= BIAS; if(e <= SHIFT+1) { x.hi &= ~(0x1fffffL >> e); x.lo = 0; } else if(e <= SHIFT+33) x.lo &= ~(0x7fffffffL >> (e-SHIFT-2)); *ip = x.x; return d - x.x; }