diff options
Diffstat (limited to 'shell/math.c')
| -rw-r--r-- | shell/math.c | 42 |
1 files changed, 41 insertions, 1 deletions
diff --git a/shell/math.c b/shell/math.c index 611b3beab..eaf4f2453 100644 --- a/shell/math.c +++ b/shell/math.c | |||
| @@ -513,6 +513,46 @@ static const char op_tokens[] ALIGN1 = { | |||
| 513 | }; | 513 | }; |
| 514 | #define ptr_to_rparen (&op_tokens[sizeof(op_tokens)-7]) | 514 | #define ptr_to_rparen (&op_tokens[sizeof(op_tokens)-7]) |
| 515 | 515 | ||
| 516 | #if ENABLE_FEATURE_SH_MATH_BASE | ||
| 517 | static arith_t strto_arith_t(const char *nptr, char **endptr) | ||
| 518 | { | ||
| 519 | unsigned base; | ||
| 520 | arith_t n; | ||
| 521 | |||
| 522 | # if ENABLE_FEATURE_SH_MATH_64 | ||
| 523 | n = strtoull(nptr, endptr, 0); | ||
| 524 | # else | ||
| 525 | n = strtoul(nptr, endptr, 0); | ||
| 526 | # endif | ||
| 527 | if (**endptr != '#' | ||
| 528 | || (*nptr < '1' || *nptr > '9') | ||
| 529 | || (n < 2 || n > 64) | ||
| 530 | ) { | ||
| 531 | return n; | ||
| 532 | } | ||
| 533 | |||
| 534 | /* It's "N#nnnn" or "NN#nnnn" syntax, NN can't start with 0, | ||
| 535 | * NN is in 2..64 range. | ||
| 536 | */ | ||
| 537 | base = (unsigned)n; | ||
| 538 | n = 0; | ||
| 539 | nptr = *endptr + 1; | ||
| 540 | /* bash allows "N#" (empty "nnnn" part) */ | ||
| 541 | while (isdigit(*nptr)) { | ||
| 542 | /* bash does not check for overflows */ | ||
| 543 | n = n * base + (*nptr++ - '0'); | ||
| 544 | } | ||
| 545 | *endptr = (char*)nptr; | ||
| 546 | return n; | ||
| 547 | } | ||
| 548 | #else /* !ENABLE_FEATURE_SH_MATH_BASE */ | ||
| 549 | # if ENABLE_FEATURE_SH_MATH_64 | ||
| 550 | # define strto_arith_t(nptr, endptr) strtoull(nptr, endptr, 0) | ||
| 551 | # else | ||
| 552 | # define strto_arith_t(nptr, endptr) strtoul(nptr, endptr, 0) | ||
| 553 | # endif | ||
| 554 | #endif | ||
| 555 | |||
| 516 | static arith_t FAST_FUNC | 556 | static arith_t FAST_FUNC |
| 517 | evaluate_string(arith_state_t *math_state, const char *expr) | 557 | evaluate_string(arith_state_t *math_state, const char *expr) |
| 518 | { | 558 | { |
| @@ -591,7 +631,7 @@ evaluate_string(arith_state_t *math_state, const char *expr) | |||
| 591 | /* Number */ | 631 | /* Number */ |
| 592 | numstackptr->var = NULL; | 632 | numstackptr->var = NULL; |
| 593 | errno = 0; | 633 | errno = 0; |
| 594 | numstackptr->val = strto_arith_t(expr, (char**) &expr, 0); | 634 | numstackptr->val = strto_arith_t(expr, (char**) &expr); |
| 595 | if (errno) | 635 | if (errno) |
| 596 | numstackptr->val = 0; /* bash compat */ | 636 | numstackptr->val = 0; /* bash compat */ |
| 597 | goto num; | 637 | goto num; |
