8354 sync regcomp(3C) with upstream (fix make catalog)
[unleashed/tickless.git] / usr / src / lib / libast / common / sfio / sfcvt.c
blob646a89e2ead980c3bdbda377589afdcf170d0232
1 /***********************************************************************
2 * *
3 * This software is part of the ast package *
4 * Copyright (c) 1985-2010 AT&T Intellectual Property *
5 * and is licensed under the *
6 * Common Public License, Version 1.0 *
7 * by AT&T Intellectual Property *
8 * *
9 * A copy of the License is available at *
10 * http://www.opensource.org/licenses/cpl1.0.txt *
11 * (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9) *
12 * *
13 * Information and Software Systems Research *
14 * AT&T Research *
15 * Florham Park NJ *
16 * *
17 * Glenn Fowler <gsf@research.att.com> *
18 * David Korn <dgk@research.att.com> *
19 * Phong Vo <kpv@research.att.com> *
20 * *
21 ***********************************************************************/
22 #if __STDC__
23 #include "FEATURE/isoc99"
24 #endif
25 #include "sfhdr.h"
27 /* Convert a floating point value to ASCII.
29 ** Written by Kiem-Phong Vo and Glenn Fowler (SFFMT_AFORMAT)
32 static char *lc_inf = "inf", *uc_inf = "INF";
33 static char *lc_nan = "nan", *uc_nan = "NAN";
34 static char *Zero = "0";
35 #define SF_INF ((_Sfi = 3), strncpy(buf, (format & SFFMT_UPPER) ? uc_inf : lc_inf, size))
36 #define SF_NAN ((_Sfi = 3), strncpy(buf, (format & SFFMT_UPPER) ? uc_nan : lc_nan, size))
37 #define SF_ZERO ((_Sfi = 1), strncpy(buf, Zero, size))
38 #define SF_INTPART (SF_IDIGITS/2)
40 #if ! _lib_isnan
41 #if _lib_fpclassify
42 #define isnan(n) (fpclassify(n)==FP_NAN)
43 #define isnanl(n) (fpclassify(n)==FP_NAN)
44 #else
45 #define isnan(n) (memcmp((void*)&n,(void*)&_Sfdnan,sizeof(n))==0)
46 #define isnanl(n) (memcmp((void*)&n,(void*)&_Sflnan,sizeof(n))==0)
47 #endif
48 #else
49 #if ! _lib_isnanl
50 #define isnanl(n) isnan(n)
51 #endif
52 #endif
54 #if ! _lib_signbit && defined(signbit)
55 #undef _lib_signbit
56 #define _lib_signbit 1
57 #endif
59 #if ! _lib_signbit
60 #if ! _ast_fltmax_double
61 static int neg0ld(Sfdouble_t f)
63 Sfdouble_t z = -0.0;
64 return !memcmp(&f, &z, sizeof(f));
66 #endif
67 static int neg0d(double f)
69 double z = -0.0;
70 return !memcmp(&f, &z, sizeof(f));
72 #endif
74 #if ULONG_DIG && ULONG_DIG < (DBL_DIG-1)
75 #define CVT_LDBL_INT long
76 #define CVT_LDBL_MAXINT LONG_MAX
77 #else
78 #if UINT_DIG && UINT_DIG < (DBL_DIG-1)
79 #define CVT_LDBL_INT int
80 #define CVT_LDBL_MAXINT INT_MAX
81 #else
82 #define CVT_LDBL_INT long
83 #define CVT_LDBL_MAXINT SF_MAXLONG
84 #endif
85 #endif
87 #if ULONG_DIG && ULONG_DIG < (DBL_DIG-1)
88 #define CVT_DBL_INT long
89 #define CVT_DBL_MAXINT LONG_MAX
90 #else
91 #if UINT_DIG && UINT_DIG < (DBL_DIG-1)
92 #define CVT_DBL_INT int
93 #define CVT_DBL_MAXINT INT_MAX
94 #else
95 #define CVT_DBL_INT long
96 #define CVT_DBL_MAXINT SF_MAXLONG
97 #endif
98 #endif
100 #if __STD_C
101 char* _sfcvt(Void_t* vp, char* buf, size_t size, int n_digit,
102 int* decpt, int* sign, int* len, int format)
103 #else
104 char* _sfcvt(vp,buf,size,n_digit,decpt,sign,len,format)
105 Void_t* vp; /* pointer to value to convert */
106 char* buf; /* conversion goes here */
107 size_t size; /* size of buf */
108 int n_digit; /* number of digits wanted */
109 int* decpt; /* to return decimal point */
110 int* sign; /* to return sign */
111 int* len; /* return string length */
112 int format; /* conversion format */
113 #endif
115 reg char *sp;
116 reg long n, v;
117 reg char *ep, *b, *endsp, *t;
118 int x;
119 _ast_flt_unsigned_max_t m;
121 static char lx[] = "0123456789abcdef";
122 static char ux[] = "0123456789ABCDEF";
124 *sign = *decpt = 0;
126 #if !_ast_fltmax_double
127 if(format&SFFMT_LDOUBLE)
128 { Sfdouble_t f = *(Sfdouble_t*)vp;
130 if(isnanl(f))
132 #if _lib_signbit
133 if (signbit(f))
134 #else
135 if (f < 0)
136 #endif
137 *sign = 1;
138 return SF_NAN;
140 #if _lib_isinf
141 if (n = isinf(f))
143 #if _lib_signbit
144 if (signbit(f))
145 #else
146 if (n < 0 || f < 0)
147 #endif
148 *sign = 1;
149 return SF_INF;
151 #endif
152 # if _c99_in_the_wild
153 # if _lib_signbit
154 if (signbit(f))
155 # else
156 # if _lib_copysignl
157 if (copysignl(1.0, f) < 0.0)
158 # else
159 # if _lib_copysign
160 if (copysign(1.0, (double)f) < 0.0)
161 # else
162 if (f < 0.0)
163 # endif
164 # endif
165 # endif
166 { f = -f;
167 *sign = 1;
169 # if _lib_fpclassify
170 switch (fpclassify(f))
172 case FP_INFINITE:
173 return SF_INF;
174 case FP_NAN:
175 return SF_NAN;
176 case FP_ZERO:
177 return SF_ZERO;
179 # endif
180 # else
181 # if _lib_signbit
182 if (signbit(f))
183 # else
184 if (f < 0.0 || f == 0.0 && neg0ld(f))
185 # endif
186 { f = -f;
187 *sign = 1;
189 # endif
190 if(f < LDBL_MIN)
191 return SF_ZERO;
192 if(f > LDBL_MAX)
193 return SF_INF;
195 if(format & SFFMT_AFORMAT)
196 { Sfdouble_t g;
197 b = sp = buf;
198 ep = (format & SFFMT_UPPER) ? ux : lx;
199 if(n_digit <= 0 || n_digit >= (size - 9))
200 n_digit = size - 9;
201 endsp = sp + n_digit + 1;
203 g = frexpl(f, &x);
204 *decpt = x;
205 f = ldexpl(g, 8 * sizeof(m) - 3);
207 for (;;)
208 { m = f;
209 x = 8 * sizeof(m);
210 while ((x -= 4) >= 0)
211 { *sp++ = ep[(m >> x) & 0xf];
212 if (sp >= endsp)
213 goto around;
215 f -= m;
216 f = ldexpl(f, 8 * sizeof(m));
220 n = 0;
221 if(f >= (Sfdouble_t)CVT_LDBL_MAXINT)
222 { /* scale to a small enough number to fit an int */
223 v = SF_MAXEXP10-1;
225 { if(f < _Sfpos10[v])
226 v -= 1;
227 else
229 f *= _Sfneg10[v];
230 if((n += (1<<v)) >= SF_IDIGITS)
231 return SF_INF;
233 } while(f >= (Sfdouble_t)CVT_LDBL_MAXINT);
235 else if(f > 0.0 && f < 0.1)
236 { /* scale to avoid excessive multiply by 10 below */
237 v = SF_MAXEXP10-1;
239 { if(f <= _Sfneg10[v])
240 { f *= _Sfpos10[v];
241 if((n += (1<<v)) >= SF_IDIGITS)
242 return SF_INF;
244 else if (--v < 0)
245 break;
246 } while(f < 0.1);
247 n = -n;
249 *decpt = (int)n;
251 b = sp = buf + SF_INTPART;
252 if((v = (CVT_LDBL_INT)f) != 0)
253 { /* translate the integer part */
254 f -= (Sfdouble_t)v;
256 sfucvt(v,sp,n,ep,CVT_LDBL_INT,unsigned CVT_LDBL_INT);
258 n = b-sp;
259 if((*decpt += (int)n) >= SF_IDIGITS)
260 return SF_INF;
261 b = sp;
262 sp = buf + SF_INTPART;
264 else n = 0;
266 /* remaining number of digits to compute; add 1 for later rounding */
267 n = (((format&SFFMT_EFORMAT) || *decpt <= 0) ? 1 : *decpt+1) - n;
268 if(n_digit > 0)
269 { if(n_digit > LDBL_DIG)
270 n_digit = LDBL_DIG;
271 n += n_digit;
274 if((ep = (sp+n)) > (endsp = buf+(size-2)))
275 ep = endsp;
276 if(sp > ep)
277 sp = ep;
278 else
280 if((format&SFFMT_EFORMAT) && *decpt == 0 && f > 0.)
281 { Sfdouble_t d;
282 while((long)(d = f*10.) == 0)
283 { f = d;
284 *decpt -= 1;
288 while(sp < ep)
289 { /* generate fractional digits */
290 if(f <= 0.)
291 { /* fill with 0's */
292 do { *sp++ = '0'; } while(sp < ep);
293 goto done;
295 else if((n = (long)(f *= 10.)) < 10)
296 { *sp++ = '0' + n;
297 f -= n;
299 else /* n == 10 */
300 { do { *sp++ = '9'; } while(sp < ep);
304 } else
305 #endif
306 { double f = *(double*)vp;
308 if(isnan(f))
310 #if _lib_signbit
311 if (signbit(f))
312 #else
313 if (f < 0)
314 #endif
315 *sign = 1;
316 return SF_NAN;
318 #if _lib_isinf
319 if (n = isinf(f))
321 #if _lib_signbit
322 if (signbit(f))
323 #else
324 if (n < 0 || f < 0)
325 #endif
326 *sign = 1;
327 return SF_INF;
329 #endif
330 #if _c99_in_the_wild
331 # if _lib_signbit
332 if (signbit(f))
333 # else
334 # if _lib_copysign
335 if (copysign(1.0, f) < 0.0)
336 # else
337 if (f < 0.0)
338 # endif
339 # endif
340 { f = -f;
341 *sign = 1;
343 # if _lib_fpclassify
344 switch (fpclassify(f))
346 case FP_INFINITE:
347 return SF_INF;
348 case FP_NAN:
349 return SF_NAN;
350 case FP_ZERO:
351 return SF_ZERO;
353 # endif
354 #else
355 # if _lib_signbit
356 if (signbit(f))
357 # else
358 if (f < 0.0 || f == 0.0 && neg0d(f))
359 # endif
360 { f = -f;
361 *sign = 1;
363 #endif
364 if(f < DBL_MIN)
365 return SF_ZERO;
366 if(f > DBL_MAX)
367 return SF_INF;
369 if(format & SFFMT_AFORMAT)
370 { double g;
371 b = sp = buf;
372 ep = (format & SFFMT_UPPER) ? ux : lx;
373 if(n_digit <= 0 || n_digit >= (size - 9))
374 n_digit = size - 9;
375 endsp = sp + n_digit + 1;
377 g = frexp(f, &x);
378 *decpt = x;
379 f = ldexp(g, 8 * sizeof(m) - 3);
381 for (;;)
382 { m = f;
383 x = 8 * sizeof(m);
384 while ((x -= 4) >= 0)
385 { *sp++ = ep[(m >> x) & 0xf];
386 if (sp >= endsp)
387 goto around;
389 f -= m;
390 f = ldexp(f, 8 * sizeof(m));
393 n = 0;
394 if(f >= (double)CVT_DBL_MAXINT)
395 { /* scale to a small enough number to fit an int */
396 v = SF_MAXEXP10-1;
398 { if(f < _Sfpos10[v])
399 v -= 1;
400 else
401 { f *= _Sfneg10[v];
402 if((n += (1<<v)) >= SF_IDIGITS)
403 return SF_INF;
405 } while(f >= (double)CVT_DBL_MAXINT);
407 else if(f > 0.0 && f < 1e-8)
408 { /* scale to avoid excessive multiply by 10 below */
409 v = SF_MAXEXP10-1;
411 { if(f <= _Sfneg10[v])
412 { f *= _Sfpos10[v];
413 if((n += (1<<v)) >= SF_IDIGITS)
414 return SF_INF;
416 else if(--v < 0)
417 break;
418 } while(f < 0.1);
419 n = -n;
421 *decpt = (int)n;
423 b = sp = buf + SF_INTPART;
424 if((v = (CVT_DBL_INT)f) != 0)
425 { /* translate the integer part */
426 f -= (double)v;
428 sfucvt(v,sp,n,ep,CVT_DBL_INT,unsigned CVT_DBL_INT);
430 n = b-sp;
431 if((*decpt += (int)n) >= SF_IDIGITS)
432 return SF_INF;
433 b = sp;
434 sp = buf + SF_INTPART;
436 else n = 0;
438 /* remaining number of digits to compute; add 1 for later rounding */
439 n = (((format&SFFMT_EFORMAT) || *decpt <= 0) ? 1 : *decpt+1) - n;
440 if(n_digit > 0)
441 { if(n_digit > DBL_DIG)
442 n_digit = DBL_DIG;
443 n += n_digit;
446 if((ep = (sp+n)) > (endsp = buf+(size-2)))
447 ep = endsp;
448 if(sp > ep)
449 sp = ep;
450 else
452 if((format&SFFMT_EFORMAT) && *decpt == 0 && f > 0.)
453 { reg double d;
454 while((long)(d = f*10.) == 0)
455 { f = d;
456 *decpt -= 1;
460 while(sp < ep)
461 { /* generate fractional digits */
462 if(f <= 0.)
463 { /* fill with 0's */
464 do { *sp++ = '0'; } while(sp < ep);
465 goto done;
467 else if((n = (long)(f *= 10.)) < 10)
468 { *sp++ = (char)('0' + n);
469 f -= n;
471 else /* n == 10 */
472 { do { *sp++ = '9'; } while(sp < ep);
473 break;
479 if(ep <= b)
480 ep = b+1;
481 else if(ep < endsp)
482 { /* round the last digit */
483 *--sp += 5;
484 while(*sp > '9')
485 { *sp = '0';
486 if(sp > b)
487 *--sp += 1;
488 else
489 { /* next power of 10 */
490 *sp = '1';
491 *decpt += 1;
492 if(!(format&SFFMT_EFORMAT))
493 { /* add one more 0 for %f precision */
494 ep[-1] = '0';
495 ep += 1;
501 done:
502 *--ep = '\0';
503 if(len)
504 *len = ep-b;
505 return b;
506 around:
507 if (((m >> x) & 0xf) >= 8)
508 { t = sp - 1;
509 for (;;)
510 { if (--t <= b)
511 { (*decpt)++;
512 break;
514 switch (*t)
516 case 'f':
517 case 'F':
518 *t = '0';
519 continue;
520 case '9':
521 *t = ep[10];
522 break;
523 default:
524 (*t)++;
525 break;
527 break;
530 ep = sp + 1;
531 goto done;