summaryrefslogtreecommitdiffstats
path: root/lib.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2015-03-20 07:15:34 -0700
committerKaz Kylheku <kaz@kylheku.com>2015-03-20 07:15:34 -0700
commit622f87134be4d14aefac13f3c3013d08ebe7c5d0 (patch)
treeaa26b914161f2a50b9bf78103d8fbdc591a0cea2 /lib.c
parent772733d9558e63afe8f33ab9f230407242bed2ce (diff)
downloadtxr-622f87134be4d14aefac13f3c3013d08ebe7c5d0.tar.gz
txr-622f87134be4d14aefac13f3c3013d08ebe7c5d0.tar.bz2
txr-622f87134be4d14aefac13f3c3013d08ebe7c5d0.zip
* lib.c (int_str): Workaround for wcstol recognizing the 0x prefix when
radix is 16. Also, thrown an error if radix is not in the range 0 to 36. * txr.1: Document int-str's radix range restriction.
Diffstat (limited to 'lib.c')
-rw-r--r--lib.c28
1 files changed, 27 insertions, 1 deletions
diff --git a/lib.c b/lib.c
index 8fd4034d..9be90ea6 100644
--- a/lib.c
+++ b/lib.c
@@ -3036,10 +3036,36 @@ val int_str(val str, val base)
{
const wchar_t *wcs = c_str(str);
wchar_t *ptr;
+ long value;
cnum b = c_num(default_arg(base, num_fast(10)));
+ /* Standard C idiocy: if base is 16, strtoul and its siblings
+ still recognize the 0x prefix. */
+ if (b == 16) {
+ switch (wcs[0]) {
+ case '+':
+ case '-':
+ switch (wcs[1]) {
+ case '0':
+ switch (wcs[2]) {
+ case 'x': case 'X':
+ return zero;
+ }
+ }
+ break;
+ case '0':
+ switch (wcs[1]) {
+ case 'x': case 'X':
+ return zero;
+ }
+ break;
+ }
+ } else if (b < 2 || b > 36) {
+ uw_throwf(error_s, lit("int-str: invalid base ~s"), base, nao);
+ }
+
/* TODO: detect if we have wcstoll */
- long value = wcstol(wcs, &ptr, b ? b : 10);
+ value = wcstol(wcs, &ptr, b ? b : 10);
if (value == 0 && ptr == wcs)
return nil;