summaryrefslogtreecommitdiffstats
path: root/stream.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2015-10-17 22:15:30 -0700
committerKaz Kylheku <kaz@kylheku.com>2015-10-17 22:15:30 -0700
commit3041727333dc9d45f7f4d799ca22813a6f8a07dc (patch)
tree236272ac3ea11604ce5f713303543e099d945d95 /stream.c
parente5b96f7555634de5e3b62d4df1a0a0d1a21799b9 (diff)
downloadtxr-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.c31
1 files changed, 26 insertions, 5 deletions
diff --git a/stream.c b/stream.c
index 496342ae..375ca684 100644
--- a/stream.c
+++ b/stream.c
@@ -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':