diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2015-10-17 22:15:30 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2015-10-17 22:15:30 -0700 |
commit | 3041727333dc9d45f7f4d799ca22813a6f8a07dc (patch) | |
tree | 236272ac3ea11604ce5f713303543e099d945d95 /stream.c | |
parent | e5b96f7555634de5e3b62d4df1a0a0d1a21799b9 (diff) | |
download | txr-3041727333dc9d45f7f4d799ca22813a6f8a07dc.tar.gz txr-3041727333dc9d45f7f4d799ca22813a6f8a07dc.tar.bz2 txr-3041727333dc9d45f7f4d799ca22813a6f8a07dc.zip |
Support binary specifier in format.
* stream.c (formatv): New 'b' case along side 'o', handling
binary in straightforward way for bignums, and with binary
formatting loop code for fixnums.
* txr.1: Documented b.
Diffstat (limited to 'stream.c')
-rw-r--r-- | stream.c | 31 |
1 files changed, 26 insertions, 5 deletions
@@ -2302,16 +2302,37 @@ val formatv(val stream_in, val fmtstr, struct args *al) } } goto output_num; - case 'o': + case 'o': case 'b': obj = args_get_checked(name, al, &arg_ix); if (bignump(obj)) { - int nchars = mp_radix_size(mp(obj), 8); + int rad = ch == '0' ? 8 : 2; + int nchars = mp_radix_size(mp(obj), rad); if (nchars >= convert(int, sizeof (num_buf))) pnum = coerce(char *, chk_malloc(nchars + 1)); - mp_toradix(mp(obj), coerce(unsigned char *, pnum), 8); - } else { - value = c_num(obj); + mp_toradix(mp(obj), coerce(unsigned char *, pnum), rad); + } else if (ch == 'o') { + cnum value = c_num(obj); sprintf(num_buf, num_fmt->oct, value); + } else { + cnum val = c_num(obj); + int s = (val < 0); + int i = sizeof num_buf; + + value = s ? -val : val; + + num_buf[--i] = 0; + + while (i > 0) { + num_buf[--i] = ((value & 1) ? '1' : '0'); + value >>= 1; + if (!value) + break; + } + + if (s && i > 0) + num_buf[--i] = '-'; + + memmove(num_buf, num_buf + i, sizeof num_buf - i); } goto output_num; case 'f': case 'e': |