printf: fix printing +-prefixed numbers
authorBernhard Reutner-Fischer <rep.dot.nop@gmail.com>
Fri, 19 Oct 2018 13:25:41 +0000 (15:25 +0200)
committerBernhard Reutner-Fischer <rep.dot.nop@gmail.com>
Fri, 19 Oct 2018 13:27:42 +0000 (15:27 +0200)
Thanks to Cristian Ionescu-Idbohrn for noticing.

Also fix "%d" '  42' to skip leading whitespace.

function                                             old     new   delta
print_direc                                          435     454     +19
bb_strtoll                                            99     103      +4
------------------------------------------------------------------------------
(add/remove: 0/0 grow/shrink: 2/0 up/down: 23/0)               Total: 23 bytes

Signed-off-by: Bernhard Reutner-Fischer <rep.dot.nop@gmail.com>
coreutils/printf.c
libbb/bb_strtonum.c
testsuite/printf.tests

index a666ff7acf43ef770a8218a510c267826259febc..b2429c5cfc50a501da510c76add2e322ed301809 100644 (file)
@@ -191,6 +191,7 @@ static void print_direc(char *format, unsigned fmt_length,
        if (have_width - 1 == have_prec)
                have_width = NULL;
 
+       /* multiconvert sets errno = 0, but %s needs it cleared */
        errno = 0;
 
        switch (format[fmt_length - 1]) {
@@ -199,7 +200,7 @@ static void print_direc(char *format, unsigned fmt_length,
                break;
        case 'd':
        case 'i':
-               llv = my_xstrtoll(argument);
+               llv = my_xstrtoll(skip_whitespace(argument));
  print_long:
                if (!have_width) {
                        if (!have_prec)
@@ -217,7 +218,7 @@ static void print_direc(char *format, unsigned fmt_length,
        case 'u':
        case 'x':
        case 'X':
-               llv = my_xstrtoull(argument);
+               llv = my_xstrtoull(skip_whitespace(argument));
                /* cheat: unsigned long and long have same width, so... */
                goto print_long;
        case 's':
index 2185017b02533ef81e4f9f8844edf8c30a161a63..cb70f10536432257ed99017c0294b34474a7e08e 100644 (file)
@@ -81,7 +81,7 @@ long long FAST_FUNC bb_strtoll(const char *arg, char **endp, int base)
        /* Check for the weird "feature":
         * a "-" string is apparently a valid "number" for strto[u]l[l]!
         * It returns zero and errno is 0! :( */
-       first = (arg[0] != '-' ? arg[0] : arg[1]);
+       first = (arg[0] != '-' && arg[0] != '+' ? arg[0] : arg[1]);
        if (!isalnum(first)) return ret_ERANGE();
 
        errno = 0;
index 963ded94b4a398cb815f20889ef2852838d32503..34a65926eb6a565e3dab9d0d375d8348ba5ace58 100755 (executable)
@@ -79,6 +79,39 @@ testing "printf understands %Ld" \
        "-5\n""0\n" \
        "" ""
 
+testing "printf handles positive numbers for %d" \
+       "${bb}printf '%d\n' 3 +3 '   3' '   +3' 2>&1; echo \$?" \
+       "3\n"\
+"3\n"\
+"3\n"\
+"3\n""0\n" \
+       "" ""
+
+testing "printf handles positive numbers for %i" \
+       "${bb}printf '%i\n' 3 +3 '   3' '   +3' 2>&1; echo \$?" \
+       "3\n"\
+"3\n"\
+"3\n"\
+"3\n""0\n" \
+       "" ""
+
+testing "printf handles positive numbers for %x" \
+       "${bb}printf '%x\n' 42 +42 '   42' '   +42' 2>&1; echo \$?" \
+       "2a\n"\
+"2a\n"\
+"2a\n"\
+"2a\n""0\n" \
+       "" ""
+
+testing "printf handles positive numbers for %f" \
+       "${bb}printf '%0.3f\n' .42 +.42 '   .42' '   +.42' 2>&1; echo \$?" \
+       "0.420\n"\
+"0.420\n"\
+"0.420\n"\
+"0.420\n""0\n" \
+       "" ""
+
+
 # "FIXED" now to act compatibly
 ## We are "more correct" here than bash/coreutils: they happily print -2
 ## as if it is a huge unsigned number