mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-26 19:16:41 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			171 lines
		
	
	
	
		
			6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			171 lines
		
	
	
	
		
			6 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 2021 Justine Alexandra Roberts Tunney                              │
 | |
| │                                                                              │
 | |
| │ Permission to use, copy, modify, and/or distribute this software for         │
 | |
| │ any purpose with or without fee is hereby granted, provided that the         │
 | |
| │ above copyright notice and this permission notice appear in all copies.      │
 | |
| │                                                                              │
 | |
| │ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL                │
 | |
| │ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED                │
 | |
| │ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE             │
 | |
| │ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, 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/alg/reverse.internal.h"
 | |
| #include "libc/assert.h"
 | |
| #include "libc/fmt/conv.h"
 | |
| #include "libc/fmt/divmod10.internal.h"
 | |
| #include "libc/fmt/fmt.internal.h"
 | |
| #include "libc/fmt/internal.h"
 | |
| #include "libc/limits.h"
 | |
| 
 | |
| #define BUFFER_SIZE 144
 | |
| 
 | |
| uint128_t __udivmodti4(uint128_t, uint128_t, uint128_t *);
 | |
| 
 | |
| static int __fmt_ntoa_format(int out(const char *, void *, size_t), void *arg,
 | |
|                              char *buf, unsigned len, bool negative,
 | |
|                              unsigned log2base, unsigned prec, unsigned width,
 | |
|                              unsigned char flags) {
 | |
|   unsigned i;
 | |
|   /* pad leading zeros */
 | |
|   if (!(flags & FLAGS_LEFT)) {
 | |
|     if (width && (flags & FLAGS_ZEROPAD) &&
 | |
|         (negative || (flags & (FLAGS_PLUS | FLAGS_SPACE)))) {
 | |
|       width--;
 | |
|     }
 | |
|     while ((len < prec) && (len < BUFFER_SIZE)) {
 | |
|       buf[len++] = '0';
 | |
|     }
 | |
|     while ((flags & FLAGS_ZEROPAD) && (len < width) && (len < BUFFER_SIZE)) {
 | |
|       buf[len++] = '0';
 | |
|     }
 | |
|   }
 | |
|   /* handle hash */
 | |
|   if (flags & FLAGS_HASH) {
 | |
|     if (!(flags & FLAGS_PRECISION) && len &&
 | |
|         ((len == prec) || (len == width)) && buf[len - 1] == '0') {
 | |
|       len--;
 | |
|       if (len && (log2base == 4 || log2base == 1) && buf[len - 1] == '0') {
 | |
|         len--;
 | |
|       }
 | |
|     }
 | |
|     if (log2base == 4 && len < BUFFER_SIZE) {
 | |
|       buf[len++] = 'x';
 | |
|     } else if (log2base == 1 && len < BUFFER_SIZE) {
 | |
|       buf[len++] = 'b';
 | |
|     }
 | |
|     if (len < BUFFER_SIZE) {
 | |
|       buf[len++] = '0';
 | |
|     }
 | |
|   }
 | |
|   if (len < BUFFER_SIZE) {
 | |
|     if (negative) {
 | |
|       buf[len++] = '-';
 | |
|     } else if (flags & FLAGS_PLUS) {
 | |
|       buf[len++] = '+'; /* ignore the space if the '+' exists */
 | |
|     } else if (flags & FLAGS_SPACE) {
 | |
|       buf[len++] = ' ';
 | |
|     }
 | |
|   }
 | |
|   /* pad spaces up to given width */
 | |
|   if (!(flags & FLAGS_LEFT) && !(flags & FLAGS_ZEROPAD)) {
 | |
|     if (len < width) {
 | |
|       if (__fmt_pad(out, arg, width - len) == -1) return -1;
 | |
|     }
 | |
|   }
 | |
|   reverse(buf, len);
 | |
|   if (out(buf, arg, len) == -1) return -1;
 | |
|   /* append pad spaces up to given width */
 | |
|   if (flags & FLAGS_LEFT) {
 | |
|     if (len < width) {
 | |
|       if (__fmt_pad(out, arg, width - len) == -1) return -1;
 | |
|     }
 | |
|   }
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| int __fmt_ntoa2(int out(const char *, void *, size_t), void *arg,
 | |
|                 uint128_t value, bool neg, unsigned log2base, unsigned prec,
 | |
|                 unsigned width, unsigned flags, const char *alphabet) {
 | |
|   uint128_t remainder;
 | |
|   unsigned len, count, digit;
 | |
|   char buf[BUFFER_SIZE];
 | |
|   len = 0;
 | |
|   if (!value) flags &= ~FLAGS_HASH;
 | |
|   if (value || !(flags & FLAGS_PRECISION)) {
 | |
|     count = 0;
 | |
|     do {
 | |
|       if (!log2base) {
 | |
|         if (value <= UINT64_MAX) {
 | |
|           value = DivMod10(value, &digit);
 | |
|         } else {
 | |
|           value = __udivmodti4(value, 10, &remainder);
 | |
|           digit = remainder;
 | |
|         }
 | |
|       } else {
 | |
|         digit = value;
 | |
|         digit &= (1u << log2base) - 1;
 | |
|         value >>= log2base;
 | |
|       }
 | |
|       if ((flags & FLAGS_GROUPING) && count == 3) {
 | |
|         buf[len++] = ',';
 | |
|         count = 1;
 | |
|       } else {
 | |
|         count++;
 | |
|       }
 | |
|       buf[len++] = alphabet[digit];
 | |
|     } while (value);
 | |
|     assert(count <= BUFFER_SIZE);
 | |
|   }
 | |
|   return __fmt_ntoa_format(out, arg, buf, len, neg, log2base, prec, width,
 | |
|                            flags);
 | |
| }
 | |
| 
 | |
| int __fmt_ntoa(int out(const char *, void *, size_t), void *arg, va_list va,
 | |
|                unsigned char signbit, unsigned long log2base,
 | |
|                unsigned long prec, unsigned long width, unsigned char flags,
 | |
|                const char *lang) {
 | |
|   bool neg;
 | |
|   uint128_t value, sign;
 | |
| 
 | |
|   /* ignore '0' flag when prec is given */
 | |
|   if (flags & FLAGS_PRECISION) {
 | |
|     flags &= ~FLAGS_ZEROPAD;
 | |
|   }
 | |
| 
 | |
|   /* no plus / space flag for u, x, X, o, b */
 | |
|   if (!(flags & FLAGS_ISSIGNED)) {
 | |
|     flags &= ~(FLAGS_PLUS | FLAGS_SPACE);
 | |
|   }
 | |
| 
 | |
|   if (signbit > 63) {
 | |
|     value = va_arg(va, uint128_t);
 | |
|   } else {
 | |
|     value = va_arg(va, uint64_t);
 | |
|   }
 | |
| 
 | |
|   neg = 0;
 | |
|   sign = 1;
 | |
|   sign <<= signbit;
 | |
|   value &= sign | (sign - 1);
 | |
|   if (flags & FLAGS_ISSIGNED) {
 | |
|     if (value != sign) {
 | |
|       if (value & sign) {
 | |
|         value = ~value + 1;
 | |
|         value &= sign | (sign - 1);
 | |
|         neg = 1;
 | |
|       }
 | |
|       value &= sign - 1;
 | |
|     } else {
 | |
|       neg = 1;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return __fmt_ntoa2(out, arg, value, neg, log2base, prec, width, flags, lang);
 | |
| }
 |