mirror of
https://github.com/pragma-/pbot.git
synced 2025-01-25 19:44:26 +01:00
Update modules/qrpn (#65)
- Fixed itenlog erroneously disallowing negative arguments - Removed badly behaving exact literal check - Fixed 2 not being a prime number - Fixed whitespace issues - Reordered checks in evaluate_literal for speed and factored out the annoying QRPN_NOT_A_UNIT non-error error code - Added a least significant digit to the definition of a year
This commit is contained in:
parent
da7d31229b
commit
dd250ccf83
@ -93,7 +93,8 @@ static unsigned long long ceil_isqrt(unsigned long long n) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int isprime(const unsigned long long n) {
|
static int isprime(const unsigned long long n) {
|
||||||
if (n < 2 || !(n % 2)) return 0;
|
if (2 == n) return 1;
|
||||||
|
else if (1 == n || !(n % 2)) return 0;
|
||||||
|
|
||||||
const unsigned long long stop = ceil_isqrt(n);
|
const unsigned long long stop = ceil_isqrt(n);
|
||||||
|
|
||||||
@ -222,7 +223,7 @@ static const struct named_quantity named_quantities[] = {
|
|||||||
{ .value = 0.514444444, .units = { 1, 0, -1, 0, 0, 0, 0 }, .name = "knot", .abrv = "kt", },
|
{ .value = 0.514444444, .units = { 1, 0, -1, 0, 0, 0, 0 }, .name = "knot", .abrv = "kt", },
|
||||||
{ .value = 1609.34, .units = { 1, 0, 0, 0, 0, 0, 0 }, .name = "mile" },
|
{ .value = 1609.34, .units = { 1, 0, 0, 0, 0, 0, 0 }, .name = "mile" },
|
||||||
{ .value = 1609.34 / 3600, .units = { 1, 0, -1, 0, 0, 0, 0 }, .abrv = "mph" },
|
{ .value = 1609.34 / 3600, .units = { 1, 0, -1, 0, 0, 0, 0 }, .abrv = "mph" },
|
||||||
{ .value = 86400.0 * 365.242, .units = { 0, 0, 1, 0, 0, 0, 0 }, .name = "year", .abrv = "a" },
|
{ .value = 86400.0 * 365.2425, .units = { 0, 0, 1, 0, 0, 0, 0 }, .name = "year", .abrv = "a" },
|
||||||
{ .value = 1852.0 * 3, .units = { 1, 0, 0, 0, 0, 0, 0 }, .name = "league" },
|
{ .value = 1852.0 * 3, .units = { 1, 0, 0, 0, 0, 0, 0 }, .name = "league" },
|
||||||
{ .value = 9.8066, .units = { 1, 0, -2, 0, 0, 0, 0 }, .name = "g" },
|
{ .value = 9.8066, .units = { 1, 0, -2, 0, 0, 0, 0 }, .name = "g" },
|
||||||
{ .value = 0.01, .units = { 1, 0, -2, 0, 0, 0, 0 }, .name = "gal", .abrv = "Gal" },
|
{ .value = 0.01, .units = { 1, 0, -2, 0, 0, 0, 0 }, .name = "gal", .abrv = "Gal" },
|
||||||
@ -316,7 +317,7 @@ static double datestr_to_unix_seconds(const char * const datestr) {
|
|||||||
char * after = NULL;
|
char * after = NULL;
|
||||||
microseconds_after_decimal = llrint(strtod(datestr, &after) * 1000000);
|
microseconds_after_decimal = llrint(strtod(datestr, &after) * 1000000);
|
||||||
if (after && *after != '\0')
|
if (after && *after != '\0')
|
||||||
fprintf(stdout, "warning: %s: ignoring \"%s\"\n", __func__, after);
|
fprintf(stderr, "warning: %s: ignoring \"%s\"\n", __func__, after);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (seconds * 1000000 + microseconds_after_decimal) * 1e-6;
|
return (seconds * 1000000 + microseconds_after_decimal) * 1e-6;
|
||||||
@ -335,7 +336,7 @@ static int evaluate_unit(struct quantity stack[static QRPN_STACK_SIZE_MAX], int
|
|||||||
|
|
||||||
/* loop over all known units and prefixes looking for a match */
|
/* loop over all known units and prefixes looking for a match */
|
||||||
for (const struct named_quantity * possible_quantity = named_quantities; !quantity && possible_quantity < named_quantities + named_quantity_count; possible_quantity++)
|
for (const struct named_quantity * possible_quantity = named_quantities; !quantity && possible_quantity < named_quantities + named_quantity_count; possible_quantity++)
|
||||||
/* loop over [full name, abbreviation, alt spelling] of each unit. this is a bit of a mess */
|
/* loop over [full name, abbreviation, alt spelling] of each unit. this is a bit of a mess */
|
||||||
for (int ipass = 0; !quantity && ipass < 3; ipass++) {
|
for (int ipass = 0; !quantity && ipass < 3; ipass++) {
|
||||||
const char * const unit_name = 2 == ipass ? possible_quantity->alt_spelling : 1 == ipass ? possible_quantity->abrv : possible_quantity->name;
|
const char * const unit_name = 2 == ipass ? possible_quantity->alt_spelling : 1 == ipass ? possible_quantity->abrv : possible_quantity->name;
|
||||||
if (!unit_name) continue;
|
if (!unit_name) continue;
|
||||||
@ -365,7 +366,7 @@ static int evaluate_unit(struct quantity stack[static QRPN_STACK_SIZE_MAX], int
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* not finding anything above is only an error if there was both a numerator and denominator */
|
/* not finding anything above is only an error if there was both a numerator and denominator */
|
||||||
if (!quantity) return slash ? QRPN_ERROR_TOKEN_UNRECOGNIZED : QRPN_NOT_A_UNIT;
|
if (!quantity) return QRPN_ERROR_TOKEN_UNRECOGNIZED;
|
||||||
|
|
||||||
if (quantity->flags & FLAG_UNIT_ENTERS_AS_OPERAND) {
|
if (quantity->flags & FLAG_UNIT_ENTERS_AS_OPERAND) {
|
||||||
if (S >= QRPN_STACK_SIZE_MAX) return QRPN_ERROR_TOO_MUCH_STACK;
|
if (S >= QRPN_STACK_SIZE_MAX) return QRPN_ERROR_TOO_MUCH_STACK;
|
||||||
@ -391,55 +392,51 @@ static int evaluate_unit(struct quantity stack[static QRPN_STACK_SIZE_MAX], int
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int evaluate_literal(struct quantity * stack, int S, const char * const token) {
|
static int evaluate_literal(struct quantity * stack, int S, const char * const token) {
|
||||||
const int unit_ret = evaluate_unit(stack, S, token, 1);
|
|
||||||
|
|
||||||
/* if evaluate unit either errored or succeeded, we're done... */
|
|
||||||
if (unit_ret != QRPN_NOT_A_UNIT) return unit_ret;
|
|
||||||
|
|
||||||
/* otherwise, token was not a unit name, parse it as a simple literal */
|
|
||||||
struct quantity tmp = { 0 };
|
struct quantity tmp = { 0 };
|
||||||
double d = 0, m = 0, s = 0;
|
|
||||||
|
|
||||||
if (strpbrk(token + 1, "d°") && sscanf(token, "%lf%*[d°]%lf%*[m']%lf%*[s\"]", &d, &m, &s)) {
|
char * endptr = NULL;
|
||||||
const double leading = strtod(token, NULL);
|
const double dv = strtod(token, &endptr);
|
||||||
tmp.value = copysign(fabs(d) + m / 60.0 + s / 3600.0, leading) * M_PI / 180.0;
|
|
||||||
}
|
if (endptr == token) {
|
||||||
else if (strpbrk(token, "T") && strpbrk(token, "Z")) {
|
if (!strcmp(token, "pi"))
|
||||||
tmp.value = datestr_to_unix_seconds(token);
|
tmp.value = M_PI;
|
||||||
tmp.units[2] = 1;
|
else if (!strcmp(token, "-pi"))
|
||||||
}
|
tmp.value = -M_PI;
|
||||||
else if (!strcmp(token, "pi"))
|
else if (!strcmp(token, "i"))
|
||||||
tmp.value = M_PI;
|
tmp.value = I;
|
||||||
else if (!strcmp(token, "-pi"))
|
else if (!strcmp(token, "-i"))
|
||||||
tmp.value = -M_PI;
|
tmp.value = -I;
|
||||||
else if (!strcmp(token, "i"))
|
else return evaluate_unit(stack, S, token, 1);
|
||||||
tmp.value = I;
|
} else {
|
||||||
else if (!strcmp(token, "-i"))
|
/* otherwise, token was not a unit name, parse it as a simple literal */
|
||||||
tmp.value = -I;
|
double d = 0, m = 0, s = 0;
|
||||||
else {
|
|
||||||
char * endptr = NULL;
|
if (strpbrk(token + 1, "d°") && sscanf(token, "%lf%*[d°]%lf%*[m']%lf%*[s\"]", &d, &m, &s))
|
||||||
const double dv = strtod(token, &endptr);
|
tmp.value = copysign(fabs(d) + m / 60.0 + s / 3600.0, d) * M_PI / 180.0;
|
||||||
if (fabs(dv) >= (double)(1ULL << 53)) {
|
else if (strpbrk(token, "T") && strpbrk(token, "Z")) {
|
||||||
const long long llv = strtoll(token, NULL, 10);
|
tmp.value = datestr_to_unix_seconds(token);
|
||||||
if ((long long)dv != llv) return QRPN_ERROR_INEXACT_LITERAL;
|
tmp.units[2] = 1;
|
||||||
}
|
}
|
||||||
tmp.value = dv;
|
else {
|
||||||
if (!strcmp(endptr, "i"))
|
tmp.value = dv;
|
||||||
tmp.value *= I;
|
|
||||||
else if (endptr == token)
|
|
||||||
return QRPN_ERROR_TOKEN_UNRECOGNIZED;
|
|
||||||
else if (endptr[0] != '\0' && endptr[1] == '\0') {
|
|
||||||
double prefix_scale = 1.0;
|
|
||||||
/* only allow k, M, G to be used in this position */
|
|
||||||
if ('k' == endptr[0]) prefix_scale = 1e3;
|
|
||||||
else if ('M' == endptr[0]) prefix_scale = 1e6;
|
|
||||||
else if ('G' == endptr[0]) prefix_scale = 1e9;
|
|
||||||
|
|
||||||
/* special case: trailing 'f' from floating point literals copied and pasted from C code should be ignored */
|
if (!strcmp(endptr, "i"))
|
||||||
else if ('f' == endptr[0]) prefix_scale = 1.0;
|
tmp.value *= I;
|
||||||
else return QRPN_ERROR_TOKEN_UNRECOGNIZED;
|
else if (endptr == token)
|
||||||
|
return evaluate_unit(stack, S, token, 1);
|
||||||
|
else if (endptr[0] != '\0' && endptr[1] == '\0') {
|
||||||
|
double prefix_scale = 1.0;
|
||||||
|
/* only allow k, M, G to be used in this position */
|
||||||
|
if ('k' == endptr[0]) prefix_scale = 1e3;
|
||||||
|
else if ('M' == endptr[0]) prefix_scale = 1e6;
|
||||||
|
else if ('G' == endptr[0]) prefix_scale = 1e9;
|
||||||
|
|
||||||
tmp.value *= prefix_scale;
|
/* special case: trailing 'f' from floating point literals copied and pasted from C code should be ignored */
|
||||||
|
else if ('f' == endptr[0]) prefix_scale = 1.0;
|
||||||
|
else return QRPN_ERROR_TOKEN_UNRECOGNIZED;
|
||||||
|
|
||||||
|
tmp.value *= prefix_scale;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -898,7 +895,7 @@ int qrpn_evaluate_token(struct quantity * const stack, int S, const char * const
|
|||||||
else if (!strcmp(token, "log2")) return evaluate_one_argument_must_be_unitless_real_nonnegative(stack, S, log2);
|
else if (!strcmp(token, "log2")) return evaluate_one_argument_must_be_unitless_real_nonnegative(stack, S, log2);
|
||||||
else if (!strcmp(token, "log10")) return evaluate_one_argument_must_be_unitless_real_nonnegative(stack, S, log10);
|
else if (!strcmp(token, "log10")) return evaluate_one_argument_must_be_unitless_real_nonnegative(stack, S, log10);
|
||||||
else if (!strcmp(token, "tenlog")) return evaluate_one_argument_must_be_unitless_real_nonnegative(stack, S, tenlog);
|
else if (!strcmp(token, "tenlog")) return evaluate_one_argument_must_be_unitless_real_nonnegative(stack, S, tenlog);
|
||||||
else if (!strcmp(token, "itenlog")) return evaluate_one_argument_must_be_unitless_real_nonnegative(stack, S, itenlog);
|
else if (!strcmp(token, "itenlog")) return evaluate_one_argument_must_be_unitless_real(stack, S, itenlog);
|
||||||
else if (!strcmp(token, "square")) {
|
else if (!strcmp(token, "square")) {
|
||||||
if (S < 1) return QRPN_ERROR_NOT_ENOUGH_STACK;
|
if (S < 1) return QRPN_ERROR_NOT_ENOUGH_STACK;
|
||||||
|
|
||||||
@ -959,8 +956,8 @@ int qrpn_evaluate_token(struct quantity * const stack, int S, const char * const
|
|||||||
}
|
}
|
||||||
else if (!strcmp(token, "print")) {
|
else if (!strcmp(token, "print")) {
|
||||||
if (S < 1) return QRPN_ERROR_NOT_ENOUGH_STACK;
|
if (S < 1) return QRPN_ERROR_NOT_ENOUGH_STACK;
|
||||||
fprintf_quantity(stdout, stack[S - 1]);
|
fprintf_quantity(stderr, stack[S - 1]);
|
||||||
fprintf(stdout, "\n");
|
fprintf(stderr, "\n");
|
||||||
return S;
|
return S;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -1222,7 +1219,7 @@ int qrpn_try_string(const struct quantity stack[static QRPN_STACK_SIZE_MAX], con
|
|||||||
__attribute((weak)) int main(const int argc, const char ** const argv) {
|
__attribute((weak)) int main(const int argc, const char ** const argv) {
|
||||||
if (isatty(STDIN_FILENO) && argc < 2) {
|
if (isatty(STDIN_FILENO) && argc < 2) {
|
||||||
/* never reached */
|
/* never reached */
|
||||||
fprintf(stdout, "%s: Evaluates an RPN expression with units\n", argv[0]);
|
fprintf(stderr, "%s: Evaluates an RPN expression with units\n", argv[0]);
|
||||||
exit(EXIT_FAILURE);
|
exit(EXIT_FAILURE);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1231,7 +1228,7 @@ __attribute((weak)) int main(const int argc, const char ** const argv) {
|
|||||||
int S = qrpn_evaluate_tokens(stack, 0, argv + 1, 0);
|
int S = qrpn_evaluate_tokens(stack, 0, argv + 1, 0);
|
||||||
|
|
||||||
if (S < 0) {
|
if (S < 0) {
|
||||||
fprintf(stdout, "error: %s\n", qrpn_strerror(S));
|
fprintf(stderr, "error: %s\n", qrpn_strerror(S));
|
||||||
exit(EXIT_FAILURE);
|
exit(EXIT_FAILURE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -44,4 +44,3 @@ void fprintf_quantity(FILE * fh, const struct quantity quantity);
|
|||||||
#define QRPN_ERROR_TOO_MUCH_STACK -11
|
#define QRPN_ERROR_TOO_MUCH_STACK -11
|
||||||
#define QRPN_ERROR_UNMATCHED_CONTROL_STATEMENT -12
|
#define QRPN_ERROR_UNMATCHED_CONTROL_STATEMENT -12
|
||||||
#define QRPN_ERROR_INEXACT_LITERAL -13
|
#define QRPN_ERROR_INEXACT_LITERAL -13
|
||||||
#define QRPN_NOT_A_UNIT -14
|
|
||||||
|
Loading…
Reference in New Issue
Block a user