mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-22 17:30:15 +00:00 
			
		
		
		
	This makes breaking changes to add underscores to many non-standard function names provided by the c library. MODE=tiny is now tinier and we now use smaller locks that are better for tiny apps in this mode. Some headers have been renamed to be in the same folder as the build package, so it'll be easier to know which build dependency is needed. Certain old misguided interfaces have been removed. Intel intrinsics headers are now listed in libc/isystem (but not in the amalgamation) to help further improve open source compatibility. Header complexity has also been reduced. Lastly, more shell scripts are now available.
		
			
				
	
	
		
			534 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			534 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | |
| │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8                                :vi│
 | |
| ╞══════════════════════════════════════════════════════════════════════════════╡
 | |
| │ Copyright (C) 1997, 1999, 2001 Lucent Technologies                           │
 | |
| │ All Rights Reserved                                                          │
 | |
| │                                                                              │
 | |
| │ Permission to use, copy, modify, and distribute this software and            │
 | |
| │ its documentation for any purpose and without fee is hereby                  │
 | |
| │ granted, provided that the above copyright notice appear in all              │
 | |
| │ copies and that both that the copyright notice and this                      │
 | |
| │ permission notice and warranty disclaimer appear in supporting               │
 | |
| │ documentation, and that the name of Lucent or any of its entities            │
 | |
| │ not be used in advertising or publicity pertaining to                        │
 | |
| │ distribution of the software without specific, written prior                 │
 | |
| │ permission.                                                                  │
 | |
| │                                                                              │
 | |
| │ LUCENT DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,                │
 | |
| │ INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS.             │
 | |
| │ IN NO EVENT SHALL LUCENT OR ANY OF ITS ENTITIES BE LIABLE FOR ANY            │
 | |
| │ SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES                    │
 | |
| │ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER              │
 | |
| │ IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,               │
 | |
| │ ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF               │
 | |
| │ THIS SOFTWARE.                                                               │
 | |
| ╚─────────────────────────────────────────────────────────────────────────────*/
 | |
| #include "libc/assert.h"
 | |
| #include "libc/fmt/fmt.h"
 | |
| #include "libc/fmt/fmt.internal.h"
 | |
| #include "libc/fmt/internal.h"
 | |
| #include "libc/macros.internal.h"
 | |
| #include "libc/intrin/bsr.h"
 | |
| #include "libc/str/str.h"
 | |
| #include "third_party/gdtoa/gdtoa.h"
 | |
| 
 | |
| /**
 | |
|  * @fileoverview Floating-Point Formatting
 | |
|  *
 | |
|  * This implements most of ANSI C's printf floating-point directives.
 | |
|  * omitting L, with %.0g and %.0G giving the shortest decimal string
 | |
|  * that rounds to the number being converted, and with negative
 | |
|  * precisions allowed for %f.
 | |
|  */
 | |
| 
 | |
| struct FPBits {
 | |
|   uint32_t bits[4];
 | |
|   const FPI *fpi;
 | |
|   int sign;
 | |
|   int ex;  // exponent
 | |
|   int kind;
 | |
| };
 | |
| 
 | |
| union U {
 | |
|   double d;
 | |
|   uint64_t q;
 | |
|   long double ld;
 | |
|   unsigned int ui[4];
 | |
|   unsigned short us[5];
 | |
| };
 | |
| 
 | |
| static const FPI kFpiDbl = {
 | |
|     .nbits = 53,
 | |
|     .emin = 1 - 1023 - 53 + 1,
 | |
|     .emax = 2046 - 1023 - 53 + 1,
 | |
|     .rounding = FPI_Round_near,
 | |
|     .sudden_underflow = 0,
 | |
| };
 | |
| 
 | |
| static const FPI kFpiLdbl = {
 | |
|     .nbits = 64,
 | |
|     .emin = 1 - 16383 - 64 + 1,
 | |
|     .emax = 32766 - 16383 - 64 + 1,
 | |
|     .rounding = FPI_Round_near,
 | |
|     .sudden_underflow = 0,
 | |
| };
 | |
| 
 | |
| static const char kSpecialFloats[2][2][4] = {
 | |
|     {"INF", "inf"},
 | |
|     {"NAN", "nan"},
 | |
| };
 | |
| 
 | |
| static void dfpbits(union U *u, struct FPBits *b) {
 | |
|   int ex, i;
 | |
|   uint32_t *bits;
 | |
|   b->fpi = &kFpiDbl;
 | |
|   b->sign = u->ui[1] & 0x80000000L;
 | |
|   bits = b->bits;
 | |
|   bits[1] = u->ui[1] & 0xfffff;
 | |
|   bits[0] = u->ui[0];
 | |
|   if ((ex = (u->ui[1] & 0x7ff00000L) >> 20) != 0) {
 | |
|     if (ex == 0x7ff) {
 | |
|       // Infinity or NaN
 | |
|       i = bits[0] | bits[1] ? STRTOG_NaN : STRTOG_Infinite;
 | |
|     } else {
 | |
|       i = STRTOG_Normal;
 | |
|       bits[1] |= 0x100000;
 | |
|     }
 | |
|   } else if (bits[0] | bits[1]) {
 | |
|     i = STRTOG_Denormal;
 | |
|     ex = 1;
 | |
|   } else {
 | |
|     i = STRTOG_Zero;
 | |
|   }
 | |
|   b->kind = i;
 | |
|   b->ex = ex - (0x3ff + 52);
 | |
| }
 | |
| 
 | |
| static void xfpbits(union U *u, struct FPBits *b) {
 | |
|   uint32_t *bits;
 | |
|   int ex, i;
 | |
|   b->fpi = &kFpiLdbl;
 | |
|   b->sign = u->us[4] & 0x8000;
 | |
|   bits = b->bits;
 | |
|   bits[1] = ((unsigned)u->us[3] << 16) | u->us[2];
 | |
|   bits[0] = ((unsigned)u->us[1] << 16) | u->us[0];
 | |
|   if ((ex = u->us[4] & 0x7fff) != 0) {
 | |
|     i = STRTOG_Normal;
 | |
|     if (ex == 0x7fff)  // Infinity or NaN
 | |
|       i = bits[0] | bits[1] ? STRTOG_NaN : STRTOG_Infinite;
 | |
|   } else if (bits[0] | bits[1]) {
 | |
|     i = STRTOG_Denormal;
 | |
|     ex = 1;
 | |
|   } else {
 | |
|     i = STRTOG_Zero;
 | |
|   }
 | |
|   b->kind = i;
 | |
|   b->ex = ex - (0x3fff + 63);
 | |
| }
 | |
| 
 | |
| // returns number of hex digits minus 1, or 0 for zero
 | |
| static int fpiprec(struct FPBits *b) {
 | |
|   FPI *fpi;
 | |
|   int i, j, k, m;
 | |
|   uint32_t *bits;
 | |
|   if (b->kind == STRTOG_Zero) return (b->ex = 0);
 | |
|   fpi = b->fpi;
 | |
|   bits = b->bits;
 | |
|   for (k = (fpi->nbits - 1) >> 2; k > 0; --k) {
 | |
|     if ((bits[k >> 3] >> 4 * (k & 7)) & 0xf) {
 | |
|       m = k >> 3;
 | |
|       for (i = 0; i <= m; ++i)
 | |
|         if (bits[i]) {
 | |
|           if (i > 0) {
 | |
|             k -= 8 * i;
 | |
|             b->ex += 32 * i;
 | |
|             for (j = i; j <= m; ++j) {
 | |
|               bits[j - i] = bits[j];
 | |
|             }
 | |
|           }
 | |
|           break;
 | |
|         }
 | |
|       for (i = 0; i < 28 && !((bits[0] >> i) & 0xf); i += 4) donothing;
 | |
|       if (i) {
 | |
|         b->ex += i;
 | |
|         m = k >> 3;
 | |
|         k -= (i >> 2);
 | |
|         for (j = 0;; ++j) {
 | |
|           bits[j] >>= i;
 | |
|           if (j == m) break;
 | |
|           bits[j] |= bits[j + 1] << (32 - i);
 | |
|         }
 | |
|       }
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
|   return k;
 | |
| }
 | |
| 
 | |
| // round to prec hex digits after the "."
 | |
| // prec1 = incoming precision (after ".")
 | |
| static int bround(struct FPBits *b, int prec, int prec1) {
 | |
|   uint32_t *bits, t;
 | |
|   int i, inc, j, k, m, n;
 | |
|   m = prec1 - prec;
 | |
|   bits = b->bits;
 | |
|   inc = 0;
 | |
|   k = m - 1;
 | |
|   if ((t = bits[k >> 3] >> (j = (k & 7) * 4)) & 8) {
 | |
|     if (t & 7) goto inc1;
 | |
|     if (j && bits[k >> 3] << (32 - j)) goto inc1;
 | |
|     while (k >= 8) {
 | |
|       k -= 8;
 | |
|       if (bits[k >> 3]) {
 | |
|       inc1:
 | |
|         inc = 1;
 | |
|         goto haveinc;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| haveinc:
 | |
|   b->ex += m * 4;
 | |
|   i = m >> 3;
 | |
|   k = prec1 >> 3;
 | |
|   j = i;
 | |
|   if ((n = 4 * (m & 7)))
 | |
|     for (;; ++j) {
 | |
|       bits[j - i] = bits[j] >> n;
 | |
|       if (j == k) break;
 | |
|       bits[j - i] |= bits[j + 1] << (32 - n);
 | |
|     }
 | |
|   else
 | |
|     for (;; ++j) {
 | |
|       bits[j - i] = bits[j];
 | |
|       if (j == k) break;
 | |
|     }
 | |
|   k = prec >> 3;
 | |
|   if (inc) {
 | |
|     for (j = 0; !(++bits[j] & 0xffffffff); ++j) donothing;
 | |
|     if (j > k) {
 | |
|     onebit:
 | |
|       bits[0] = 1;
 | |
|       b->ex += 4 * prec;
 | |
|       return 1;
 | |
|     }
 | |
|     if ((j = prec & 7) < 7 && bits[k] >> (j + 1) * 4) goto onebit;
 | |
|   }
 | |
|   for (i = 0; !(bits[i >> 3] & (0xf << 4 * (i & 7))); ++i) donothing;
 | |
|   if (i) {
 | |
|     b->ex += 4 * i;
 | |
|     prec -= i;
 | |
|     j = i >> 3;
 | |
|     i &= 7;
 | |
|     i *= 4;
 | |
|     for (m = j;; ++m) {
 | |
|       bits[m - j] = bits[m] >> i;
 | |
|       if (m == k) break;
 | |
|       bits[m - j] |= bits[m + 1] << (32 - i);
 | |
|     }
 | |
|   }
 | |
|   return prec;
 | |
| }
 | |
| 
 | |
| int __fmt_dtoa(int (*out)(const char *, void *, size_t), void *arg, int d,
 | |
|                int flags, int prec, int sign, int width, bool longdouble,
 | |
|                char qchar, unsigned char signbit, const char *alphabet,
 | |
|                va_list va) {
 | |
|   double x;
 | |
|   union U u;
 | |
|   struct FPBits fpb;
 | |
|   char *s, *q, *se, *s0, special[8];
 | |
|   int c, k, i1, ui, bw, bex, sgn, prec1, decpt;
 | |
|   x = 0;
 | |
|   switch (d) {
 | |
|     case 'F':
 | |
|     case 'f':
 | |
|       if (!(flags & FLAGS_PRECISION)) prec = 6;
 | |
|       if (!longdouble) {
 | |
|         x = va_arg(va, double);
 | |
|         s = s0 = dtoa(x, 3, prec, &decpt, &fpb.sign, &se);
 | |
|         if (decpt == 9999) {
 | |
|           if (s && s[0] == 'N') {
 | |
|             fpb.kind = STRTOG_NaN;
 | |
|           } else {
 | |
|             fpb.kind = STRTOG_Infinite;
 | |
|           }
 | |
|         }
 | |
|       } else {
 | |
|         u.ld = va_arg(va, long double);
 | |
|         xfpbits(&u, &fpb);
 | |
|         s = s0 =
 | |
|             gdtoa(fpb.fpi, fpb.ex, fpb.bits, &fpb.kind, 3, prec, &decpt, &se);
 | |
|       }
 | |
|       if (decpt == 9999) {
 | |
|       Format9999:
 | |
|         if (s0) freedtoa(s0);
 | |
|         bzero(special, sizeof(special));
 | |
|         s = q = special;
 | |
|         if (fpb.sign) {
 | |
|           *q++ = '-';
 | |
|         } else if (flags & FLAGS_PLUS) {
 | |
|           *q++ = '+';
 | |
|         } else if (flags & FLAGS_SPACE) {
 | |
|           *q++ = ' ';
 | |
|         }
 | |
|         memcpy(q, kSpecialFloats[fpb.kind == STRTOG_NaN][d >= 'a'], 4);
 | |
|         flags &= ~(FLAGS_PRECISION | FLAGS_PLUS | FLAGS_HASH | FLAGS_SPACE);
 | |
|         prec = 0;
 | |
|         return __fmt_stoa(out, arg, s, flags, prec, width, signbit, qchar);
 | |
|       }
 | |
|     FormatReal:
 | |
|       if (fpb.sign /* && (x || sign) */) sign = '-';
 | |
|       if (prec > 0) width -= prec;
 | |
|       if (width > 0) {
 | |
|         if (sign) --width;
 | |
|         if (decpt <= 0) {
 | |
|           --width;
 | |
|           if (prec > 0) --width;
 | |
|         } else {
 | |
|           if (s == se) decpt = 1;
 | |
|           width -= decpt;
 | |
|           if (prec > 0 || (flags & FLAGS_HASH)) --width;
 | |
|         }
 | |
|       }
 | |
|       if (width > 0 && !(flags & FLAGS_LEFT)) {
 | |
|         if ((flags & FLAGS_ZEROPAD)) {
 | |
|           if (sign) __FMT_PUT(sign);
 | |
|           sign = 0;
 | |
|           do __FMT_PUT('0');
 | |
|           while (--width > 0);
 | |
|         } else
 | |
|           do __FMT_PUT(' ');
 | |
|           while (--width > 0);
 | |
|       }
 | |
|       if (sign) __FMT_PUT(sign);
 | |
|       if (decpt <= 0) {
 | |
|         __FMT_PUT('0');
 | |
|         if (prec > 0 || (flags & FLAGS_HASH)) __FMT_PUT('.');
 | |
|         while (decpt < 0) {
 | |
|           __FMT_PUT('0');
 | |
|           prec--;
 | |
|           decpt++;
 | |
|         }
 | |
|       } else {
 | |
|         do {
 | |
|           if ((c = *s)) {
 | |
|             s++;
 | |
|           } else {
 | |
|             c = '0';
 | |
|           }
 | |
|           __FMT_PUT(c);
 | |
|         } while (--decpt > 0);
 | |
|         if (prec > 0 || (flags & FLAGS_HASH)) __FMT_PUT('.');
 | |
|       }
 | |
|       while (--prec >= 0) {
 | |
|         if ((c = *s)) {
 | |
|           s++;
 | |
|         } else {
 | |
|           c = '0';
 | |
|         }
 | |
|         __FMT_PUT(c);
 | |
|       }
 | |
|       while (--width >= 0) __FMT_PUT(' ');
 | |
|       if (s0) freedtoa(s0);
 | |
|       break;
 | |
| 
 | |
|     case 'G':
 | |
|     case 'g':
 | |
|       if (!(flags & FLAGS_PRECISION)) prec = 6;
 | |
|       if (prec < 1) prec = 1;
 | |
|       if (!longdouble) {
 | |
|         x = va_arg(va, double);
 | |
|         s = s0 = dtoa(x, 2, prec, &decpt, &fpb.sign, &se);
 | |
|         if (decpt == 9999) {
 | |
|           if (s && s[0] == 'N') {
 | |
|             fpb.kind = STRTOG_NaN;
 | |
|           } else {
 | |
|             fpb.kind = STRTOG_Infinite;
 | |
|           }
 | |
|         }
 | |
|       } else {
 | |
|         u.ld = va_arg(va, long double);
 | |
|         xfpbits(&u, &fpb);
 | |
|         s = s0 = gdtoa(fpb.fpi, fpb.ex, fpb.bits, &fpb.kind, prec ? 2 : 0, prec,
 | |
|                        &decpt, &se);
 | |
|       }
 | |
|       if (decpt == 9999) goto Format9999;
 | |
|       c = se - s;
 | |
|       prec1 = prec;
 | |
|       if (!prec) {
 | |
|         prec = c;
 | |
|         prec1 = c + (s[1] || (flags & FLAGS_HASH) ? 5 : 4);
 | |
|         // %.0g gives 10 rather than 1e1
 | |
|       }
 | |
|       if (decpt > -4 && decpt <= prec1) {
 | |
|         if ((flags & FLAGS_HASH))
 | |
|           prec -= decpt;
 | |
|         else
 | |
|           prec = c - decpt;
 | |
|         if (prec < 0) prec = 0;
 | |
|         goto FormatReal;
 | |
|       }
 | |
|       d -= 2;
 | |
|       if (!(flags & FLAGS_HASH) && prec > c) prec = c;
 | |
|       --prec;
 | |
|       goto FormatExpo;
 | |
| 
 | |
|     case 'e':
 | |
|     case 'E':
 | |
|       if (!(flags & FLAGS_PRECISION)) prec = 6;
 | |
|       if (prec < 0) prec = 0;
 | |
|       if (!longdouble) {
 | |
|         x = va_arg(va, double);
 | |
|         s = s0 = dtoa(x, 2, prec + 1, &decpt, &fpb.sign, &se);
 | |
|         if (decpt == 9999) {
 | |
|           if (s && s[0] == 'N') {
 | |
|             fpb.kind = STRTOG_NaN;
 | |
|           } else {
 | |
|             fpb.kind = STRTOG_Infinite;
 | |
|           }
 | |
|         }
 | |
|       } else {
 | |
|         u.ld = va_arg(va, long double);
 | |
|         xfpbits(&u, &fpb);
 | |
|         s = s0 = gdtoa(fpb.fpi, fpb.ex, fpb.bits, &fpb.kind, prec ? 2 : 0, prec,
 | |
|                        &decpt, &se);
 | |
|       }
 | |
|       if (decpt == 9999) goto Format9999;
 | |
|     FormatExpo:
 | |
|       if (fpb.sign /* && (x || sign) */) sign = '-';
 | |
|       if ((width -= prec + 5) > 0) {
 | |
|         if (sign) --width;
 | |
|         if (prec || (flags & FLAGS_HASH)) --width;
 | |
|       }
 | |
|       if ((c = --decpt) < 0) c = -c;
 | |
|       while (c >= 100) {
 | |
|         --width;
 | |
|         c /= 10;
 | |
|       }
 | |
|       if (width > 0 && !(flags & FLAGS_LEFT)) {
 | |
|         if ((flags & FLAGS_ZEROPAD)) {
 | |
|           if (sign) __FMT_PUT(sign);
 | |
|           sign = 0;
 | |
|           do __FMT_PUT('0');
 | |
|           while (--width > 0);
 | |
|         } else
 | |
|           do __FMT_PUT(' ');
 | |
|           while (--width > 0);
 | |
|       }
 | |
|       if (sign) __FMT_PUT(sign);
 | |
|       __FMT_PUT(*s++);
 | |
|       if (prec || (flags & FLAGS_HASH)) __FMT_PUT('.');
 | |
|       while (--prec >= 0) {
 | |
|         if ((c = *s))
 | |
|           s++;
 | |
|         else
 | |
|           c = '0';
 | |
|         __FMT_PUT(c);
 | |
|       }
 | |
|       __FMT_PUT(d);
 | |
|       if (decpt < 0) {
 | |
|         __FMT_PUT('-');
 | |
|         decpt = -decpt;
 | |
|       } else
 | |
|         __FMT_PUT('+');
 | |
|       for (c = 2, k = 10; 10 * k <= decpt; c++, k *= 10) donothing;
 | |
|       for (;;) {
 | |
|         i1 = decpt / k;
 | |
|         __FMT_PUT(i1 + '0');
 | |
|         if (--c <= 0) break;
 | |
|         decpt -= i1 * k;
 | |
|         decpt *= 10;
 | |
|       }
 | |
|       while (--width >= 0) __FMT_PUT(' ');
 | |
|       freedtoa(s0);
 | |
|       break;
 | |
| 
 | |
|     case 'A':
 | |
|       alphabet = "0123456789ABCDEFPX";
 | |
|       goto FormatBinary;
 | |
|     case 'a':
 | |
|       alphabet = "0123456789abcdefpx";
 | |
|     FormatBinary:
 | |
|       if (longdouble) {
 | |
|         u.ld = va_arg(va, long double);
 | |
|         xfpbits(&u, &fpb);
 | |
|       } else {
 | |
|         u.d = va_arg(va, double);
 | |
|         dfpbits(&u, &fpb);
 | |
|       }
 | |
|       if (fpb.kind == STRTOG_Infinite || fpb.kind == STRTOG_NaN) {
 | |
|         s0 = 0;
 | |
|         goto Format9999;
 | |
|       }
 | |
|       prec1 = fpiprec(&fpb);
 | |
|       if ((flags & FLAGS_PRECISION) && prec < prec1) {
 | |
|         prec1 = bround(&fpb, prec, prec1);
 | |
|       }
 | |
|       bw = 1;
 | |
|       bex = fpb.ex + 4 * prec1;
 | |
|       if (bex) {
 | |
|         if ((i1 = bex) < 0) i1 = -i1;
 | |
|         while (i1 >= 10) {
 | |
|           ++bw;
 | |
|           i1 /= 10;
 | |
|         }
 | |
|       }
 | |
|       if (fpb.sign /* && (sign || fpb.kind != STRTOG_Zero) */) {
 | |
|         sign = '-';
 | |
|       }
 | |
|       if ((width -= bw + 5) > 0) {
 | |
|         if (sign) --width;
 | |
|         if (prec1 || (flags & FLAGS_HASH)) --width;
 | |
|       }
 | |
|       if ((width -= prec1) > 0 && !(flags & FLAGS_LEFT) &&
 | |
|           !(flags & FLAGS_ZEROPAD)) {
 | |
|         do __FMT_PUT(' ');
 | |
|         while (--width > 0);
 | |
|       }
 | |
|       if (sign) __FMT_PUT(sign);
 | |
|       __FMT_PUT('0');
 | |
|       __FMT_PUT(alphabet[17]);
 | |
|       if ((flags & FLAGS_ZEROPAD) && width > 0 && !(flags & FLAGS_LEFT)) {
 | |
|         do __FMT_PUT('0');
 | |
|         while (--width > 0);
 | |
|       }
 | |
|       i1 = prec1 & 7;
 | |
|       k = prec1 >> 3;
 | |
|       __FMT_PUT(alphabet[(fpb.bits[k] >> 4 * i1) & 0xf]);
 | |
|       if (prec1 > 0 || (flags & FLAGS_HASH)) __FMT_PUT('.');
 | |
|       if (prec1 > 0) {
 | |
|         prec -= prec1;
 | |
|         while (prec1 > 0) {
 | |
|           if (--i1 < 0) {
 | |
|             if (--k < 0) break;
 | |
|             i1 = 7;
 | |
|           }
 | |
|           __FMT_PUT(alphabet[(fpb.bits[k] >> 4 * i1) & 0xf]);
 | |
|           --prec1;
 | |
|         }
 | |
|         if ((flags & FLAGS_HASH) && prec > 0) {
 | |
|           do __FMT_PUT(0);
 | |
|           while (--prec > 0);
 | |
|         }
 | |
|       }
 | |
|       __FMT_PUT(alphabet[16]);
 | |
|       if (bex < 0) {
 | |
|         __FMT_PUT('-');
 | |
|         bex = -bex;
 | |
|       } else
 | |
|         __FMT_PUT('+');
 | |
|       for (c = 1; 10 * c <= bex; c *= 10) donothing;
 | |
|       for (;;) {
 | |
|         i1 = bex / c;
 | |
|         __FMT_PUT('0' + i1);
 | |
|         if (!--bw) break;
 | |
|         bex -= i1 * c;
 | |
|         bex *= 10;
 | |
|       }
 | |
|       while (--width >= 0) __FMT_PUT(' ');
 | |
|       break;
 | |
|     default:
 | |
|       unreachable;
 | |
|   }
 | |
|   return 0;
 | |
| }
 |