summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--miscutils/bc.c267
1 files changed, 52 insertions, 215 deletions
diff --git a/miscutils/bc.c b/miscutils/bc.c
index 7b20a94..bc5501d 100644
--- a/miscutils/bc.c
+++ b/miscutils/bc.c
@@ -171,141 +171,7 @@ typedef enum BcStatus {
BC_STATUS_SUCCESS = 0,
BC_STATUS_FAILURE = 1,
BC_STATUS_PARSE_EMPTY_EXP = 2, // bc_parse_expr() uses this
-
-// BC_STATUS_ALLOC_ERR,
-// BC_STATUS_INPUT_EOF,
-// BC_STATUS_BIN_FILE,
-// BC_STATUS_PATH_IS_DIR,
-
-// BC_STATUS_LEX_BAD_CHAR,
-// BC_STATUS_LEX_NO_STRING_END,
-// BC_STATUS_LEX_NO_COMMENT_END,
-// BC_STATUS_LEX_EOF,
-#if ENABLE_DC
-// BC_STATUS_LEX_EXTENDED_REG,
-#endif
-// BC_STATUS_PARSE_BAD_TOKEN,
-// BC_STATUS_PARSE_BAD_EXP,
-// BC_STATUS_PARSE_BAD_PRINT,
-// BC_STATUS_PARSE_BAD_FUNC,
-// BC_STATUS_PARSE_BAD_ASSIGN,
-// BC_STATUS_PARSE_NO_AUTO,
-// BC_STATUS_PARSE_DUPLICATE_LOCAL,
-// BC_STATUS_PARSE_NO_BLOCK_END,
-
-// BC_STATUS_MATH_NEGATIVE,
-// BC_STATUS_MATH_NON_INTEGER,
-// BC_STATUS_MATH_OVERFLOW,
-// BC_STATUS_MATH_DIVIDE_BY_ZERO,
-// BC_STATUS_MATH_BAD_STRING,
-
-// BC_STATUS_EXEC_FILE_ERR,
-// BC_STATUS_EXEC_MISMATCHED_PARAMS,
-// BC_STATUS_EXEC_UNDEFINED_FUNC,
-// BC_STATUS_EXEC_FILE_NOT_EXECUTABLE,
-// BC_STATUS_EXEC_NUM_LEN,
-// BC_STATUS_EXEC_NAME_LEN,
-// BC_STATUS_EXEC_STRING_LEN,
-// BC_STATUS_EXEC_ARRAY_LEN,
-// BC_STATUS_EXEC_BAD_IBASE,
-// BC_STATUS_EXEC_BAD_SCALE,
-// BC_STATUS_EXEC_BAD_READ_EXPR,
-// BC_STATUS_EXEC_REC_READ,
-// BC_STATUS_EXEC_BAD_TYPE,
-// BC_STATUS_EXEC_BAD_OBASE,
-// BC_STATUS_EXEC_SIGNAL,
-// BC_STATUS_EXEC_STACK,
-
-// BC_STATUS_VEC_OUT_OF_BOUNDS,
-// BC_STATUS_VEC_ITEM_EXISTS,
- BC_STATUS_BEFORE_POSIX = BC_STATUS_PARSE_EMPTY_EXP,
-#if ENABLE_BC
- BC_STATUS_POSIX_NAME_LEN,
- BC_STATUS_POSIX_COMMENT,
- BC_STATUS_POSIX_BAD_KW,
- BC_STATUS_POSIX_DOT,
- BC_STATUS_POSIX_RET,
- BC_STATUS_POSIX_BOOL,
- BC_STATUS_POSIX_REL_POS,
- BC_STATUS_POSIX_MULTIREL,
- BC_STATUS_POSIX_FOR1,
- BC_STATUS_POSIX_FOR2,
- BC_STATUS_POSIX_FOR3,
- BC_STATUS_POSIX_BRACE,
-#endif
-// BC_STATUS_QUIT,
-// BC_STATUS_LIMITS,
-
-// BC_STATUS_INVALID_OPTION,
} BcStatus;
-// Keep enum above and messages below in sync!
-static const char *const bc_err_msgs[] = {
- NULL,
- NULL,
- NULL,
-
-// "memory allocation error",
-// "I/O error",
-// "file is not text:",
-// "path is a directory:",
-
-// "bad character",
-// "string end could not be found",
-// "comment end could not be found",
-// "end of file",
-#if ENABLE_DC
-// "extended register",
-#endif
-// "bad token",
-// "bad expression",
-// "bad print statement",
-// "bad function definition",
-// "bad assignment: left side must be scale, ibase, "
-// "obase, last, var, or array element",
-// "no auto variable found",
-// "function parameter or auto var has the same name as another",
-// "block end could not be found",
-
-// "negative number",
-// "non integer number",
-// "overflow",
-// "divide by zero",
-// "bad number string",
-
-// "could not open file:",
-// "mismatched parameters", // wrong number of them, to be exact
-// "undefined function",
-// "file is not executable:",
-// "number too long: must be [1, BC_NUM_MAX]",
-// "name too long: must be [1, BC_NAME_MAX]",
-// "string too long: must be [1, BC_STRING_MAX]",
-// "array too long; must be [1, BC_DIM_MAX]",
-// "bad ibase; must be [2, 16]",
-// "bad scale; must be [0, BC_SCALE_MAX]",
-// "bad read() expression",
-// "read() call inside of a read() call",
-// "variable is wrong type",
-// "bad obase; must be [2, BC_BASE_MAX]",
-// "signal caught and not handled",
-// "stack has too few elements",
-
-// "index is out of bounds",
-// "item already exists",
-#if ENABLE_BC
- "POSIX only allows one character names; the following is bad:",
- "POSIX does not allow '#' script comments",
- "POSIX does not allow the following keyword:",
- "POSIX does not allow a period ('.') as a shortcut for the last result",
- "POSIX requires parentheses around return expressions",
- "POSIX does not allow boolean operators; the following is bad:",
- "POSIX does not allow comparison operators outside if or loops",
- "POSIX requires exactly one comparison operator per condition",
- "POSIX does not allow an empty init expression in a for loop",
- "POSIX does not allow an empty condition expression in a for loop",
- "POSIX does not allow an empty update expression in a for loop",
- "POSIX requires the left brace be on the same line as the function header",
-#endif
-};
#define BC_VEC_INVALID_IDX ((size_t) -1)
#define BC_VEC_START_CAP (1 << 5)
@@ -891,17 +757,8 @@ struct globals {
#define IS_BC (ENABLE_BC && (!ENABLE_DC || applet_name[0] == 'b'))
-#if ENABLE_BC
-static BcStatus bc_vm_posixError(BcStatus s, const char *file, size_t line,
- const char *msg);
-#endif
-
static void bc_vm_info(void);
-static const char bc_err_fmt[] = "\nerror: %s\n";
-static const char bc_warn_fmt[] = "\nwarning: %s\n";
-static const char bc_err_line[] = ":%zu\n\n";
-
#if ENABLE_BC
static const BcLexKeyword bc_lex_kws[20] = {
BC_LEX_KW_ENTRY("auto", 4, true),
@@ -1151,6 +1008,25 @@ static int bc_error(const char *fmt, ...)
return BC_STATUS_FAILURE;
}
+static int bc_posix_error(const char *fmt, ...)
+{
+ va_list p;
+
+ if (!(G.flags & (BC_FLAG_S|BC_FLAG_W)))
+ return BC_STATUS_SUCCESS;
+
+ va_start(p, fmt);
+ bb_verror_msg(fmt, p, NULL);
+ va_end(p);
+
+ // Do we treat non-POSIX constructs as errors?
+ if (!(G.flags & BC_FLAG_S))
+ return BC_STATUS_SUCCESS; // no, it's a warning
+ if (!G.ttyin)
+ exit(1);
+ return BC_STATUS_FAILURE;
+}
+
static void bc_vec_grow(BcVec *v, size_t n)
{
size_t cap = v->cap * 2;
@@ -3039,8 +2915,7 @@ static BcStatus bc_lex_identifier(BcLex *l)
l->t.t = BC_LEX_KEY_AUTO + (BcLexType) i;
if (!bc_lex_kws[i].posix) {
- s = bc_vm_posixError(BC_STATUS_POSIX_BAD_KW, l->f, l->line,
- bc_lex_kws[i].name);
+ s = bc_posix_error("POSIX does not allow the following keyword:"); // bc_lex_kws[i].name
if (s) return s;
}
@@ -3054,7 +2929,7 @@ static BcStatus bc_lex_identifier(BcLex *l)
if (s) return s;
if (l->t.v.len - 1 > 1)
- s = bc_vm_posixError(BC_STATUS_POSIX_NAME_LEN, l->f, l->line, buf);
+ s = bc_posix_error("POSIX only allows one character names; the following is bad:"); // buf
return s;
}
@@ -3155,7 +3030,7 @@ static BcStatus bc_lex_token(BcLex *l)
bc_lex_assign(l, BC_LEX_OP_REL_NE, BC_LEX_OP_BOOL_NOT);
if (l->t.t == BC_LEX_OP_BOOL_NOT) {
- s = bc_vm_posixError(BC_STATUS_POSIX_BOOL, l->f, l->line, "!");
+ s = bc_posix_error("POSIX does not allow boolean operators; the following is bad:"); // "!"
if (s) return s;
}
@@ -3170,7 +3045,7 @@ static BcStatus bc_lex_token(BcLex *l)
case '#':
{
- s = bc_vm_posixError(BC_STATUS_POSIX_COMMENT, l->f, l->line, NULL);
+ s = bc_posix_error("POSIX does not allow '#' script comments");
if (s) return s;
bc_lex_lineComment(l);
@@ -3189,7 +3064,7 @@ static BcStatus bc_lex_token(BcLex *l)
c2 = l->buf[l->i];
if (c2 == '&') {
- s = bc_vm_posixError(BC_STATUS_POSIX_BOOL, l->f, l->line, "&&");
+ s = bc_posix_error("POSIX does not allow boolean operators; the following is bad:"); // "&&"
if (s) return s;
++l->i;
@@ -3252,7 +3127,7 @@ static BcStatus bc_lex_token(BcLex *l)
s = bc_lex_number(l, c);
else {
l->t.t = BC_LEX_KEY_LAST;
- s = bc_vm_posixError(BC_STATUS_POSIX_DOT, l->f, l->line, NULL);
+ s = bc_posix_error("POSIX does not allow a period ('.') as a shortcut for the last result");
}
break;
}
@@ -3379,8 +3254,7 @@ static BcStatus bc_lex_token(BcLex *l)
c2 = l->buf[l->i];
if (c2 == '|') {
-
- s = bc_vm_posixError(BC_STATUS_POSIX_BOOL, l->f, l->line, "||");
+ s = bc_posix_error("POSIX does not allow boolean operators; the following is bad:"); // "||"
if (s) return s;
++l->i;
@@ -4106,7 +3980,7 @@ static BcStatus bc_parse_return(BcParse *p)
}
if (!paren || p->l.t.last != BC_LEX_RPAREN) {
- s = bc_vm_posixError(BC_STATUS_POSIX_RET, p->l.f, p->l.line, NULL);
+ s = bc_posix_error("POSIX requires parentheses around return expressions");
if (s) return s;
}
@@ -4313,7 +4187,7 @@ static BcStatus bc_parse_for(BcParse *p)
if (p->l.t.t != BC_LEX_SCOLON)
s = bc_parse_expr(p, 0, bc_parse_next_for);
else
- s = bc_vm_posixError(BC_STATUS_POSIX_FOR1, p->l.f, p->l.line, NULL);
+ s = bc_posix_error("POSIX does not allow an empty init expression in a for loop");
if (s) return s;
if (p->l.t.t != BC_LEX_SCOLON) return bc_error("bad token");
@@ -4330,7 +4204,7 @@ static BcStatus bc_parse_for(BcParse *p)
if (p->l.t.t != BC_LEX_SCOLON)
s = bc_parse_expr(p, BC_PARSE_REL, bc_parse_next_for);
else
- s = bc_vm_posixError(BC_STATUS_POSIX_FOR2, p->l.f, p->l.line, NULL);
+ s = bc_posix_error("POSIX does not allow an empty condition expression in a for loop");
if (s) return s;
if (p->l.t.t != BC_LEX_SCOLON) return bc_error("bad token");
@@ -4351,7 +4225,7 @@ static BcStatus bc_parse_for(BcParse *p)
if (p->l.t.t != BC_LEX_RPAREN)
s = bc_parse_expr(p, 0, bc_parse_next_rel);
else
- s = bc_vm_posixError(BC_STATUS_POSIX_FOR3, p->l.f, p->l.line, NULL);
+ s = bc_posix_error("POSIX does not allow an empty update expression in a for loop");
if (s) return s;
@@ -4475,7 +4349,7 @@ static BcStatus bc_parse_func(BcParse *p)
if (s) return s;
if (p->l.t.t != BC_LEX_LBRACE)
- s = bc_vm_posixError(BC_STATUS_POSIX_BRACE, p->l.f, p->l.line, NULL);
+ s = bc_posix_error("POSIX requires the left brace be on the same line as the function header");
return s;
@@ -4999,11 +4873,11 @@ static BcStatus bc_parse_expr(BcParse *p, uint8_t flags, BcParseNext next)
ok:
if (!(flags & BC_PARSE_REL) && nrelops) {
- s = bc_vm_posixError(BC_STATUS_POSIX_REL_POS, p->l.f, p->l.line, NULL);
+ s = bc_posix_error("POSIX does not allow comparison operators outside if or loops");
if (s) return s;
}
else if ((flags & BC_PARSE_REL) && nrelops > 1) {
- s = bc_vm_posixError(BC_STATUS_POSIX_MULTIREL, p->l.f, p->l.line, NULL);
+ s = bc_posix_error("POSIX requires exactly one comparison operator per condition");
if (s) return s;
}
@@ -6575,10 +6449,6 @@ static void bc_program_reset(void)
// If !tty, no need to check for ^C: we don't have ^C handler,
// we would be killed by a signal and won't reach this place
-
- fflush_and_check(); // make sure buffered stdout is printed
- fputs("ready for more input\n", stderr);
- fflush_and_check();
}
static BcStatus bc_program_exec(void)
@@ -6914,40 +6784,7 @@ static void bc_vm_info(void)
, applet_name);
}
-static BcStatus bc_vm_error(BcStatus s, const char *file, size_t line)
-{
- if (!s || s > BC_STATUS_BEFORE_POSIX) return s;
-
- if (bc_err_msgs[s]) {
- fprintf(stderr, bc_err_fmt, bc_err_msgs[s]);
- fprintf(stderr, " %s", file);
- fprintf(stderr, bc_err_line + 4 * !line, line);
- }
-
-///
- return s * (!G.ttyin || !!strcmp(file, bc_program_stdin_name));
-}
-
#if ENABLE_BC
-static BcStatus bc_vm_posixError(BcStatus s, const char *file, size_t line,
- const char *msg)
-{
- const char *fmt;
-
- if (!(G.flags & (BC_FLAG_S|BC_FLAG_W))) return BC_STATUS_SUCCESS;
- if (s < BC_STATUS_POSIX_NAME_LEN) return BC_STATUS_SUCCESS;
-
- fmt = G_posix ? bc_err_fmt : bc_warn_fmt;
- fprintf(stderr, fmt, bc_err_msgs[s]);
- if (msg) fprintf(stderr, " %s\n", msg);
- fprintf(stderr, " %s", file);
- fprintf(stderr, bc_err_line + 4 * !line, line);
-
- if (G.ttyin || !G_posix)
- s = BC_STATUS_SUCCESS;
- return s;
-}
-
static void bc_vm_envArgs(void)
{
static const char* const bc_args_env_name = "BC_ENV_ARGS";
@@ -7004,24 +6841,18 @@ static BcStatus bc_vm_process(const char *text)
{
BcStatus s = bc_parse_text(&G.prs, text);
- s = bc_vm_error(s, G.prs.l.f, G.prs.l.line);
if (s) return s;
while (G.prs.l.t.t != BC_LEX_EOF) {
-
s = G.prs.parse(&G.prs);
-
- s = bc_vm_error(s, G.prs.l.f, G.prs.l.line);
if (s) return s;
}
if (BC_PARSE_CAN_EXEC(&G.prs)) {
s = bc_program_exec();
fflush_and_check();
- if (s) {
+ if (s)
bc_program_reset();
- s = bc_vm_error(s, G.prs.l.f, 0);
- }
}
return s;
@@ -7115,23 +6946,21 @@ static BcStatus bc_vm_stdin(void)
bc_vec_concat(&buffer, buf.v);
s = bc_vm_process(buffer.v);
- if (s) goto err;
+ if (s) {
+ fflush_and_check();
+ fputs("ready for more input\n", stderr);
+ }
bc_vec_npop(&buffer, buffer.len);
}
if (str) {
- bc_error("string end could not be found");
- s = bc_vm_error(BC_STATUS_FAILURE, G.prs.l.f,
- G.prs.l.line);
+ s = bc_error("string end could not be found");
}
else if (comment) {
- bc_error("comment end could not be found");
- s = bc_vm_error(BC_STATUS_FAILURE, G.prs.l.f,
- G.prs.l.line);
+ s = bc_error("comment end could not be found");
}
-err:
bc_vec_free(&buf);
bc_vec_free(&buffer);
return s;
@@ -7148,7 +6977,8 @@ static BcStatus bc_vm_exec(void)
bc_lex_file(&G.prs.l, bc_lib_name);
s = bc_parse_text(&G.prs, bc_lib);
- while (!s && G.prs.l.t.t != BC_LEX_EOF) s = G.prs.parse(&G.prs);
+ while (!s && G.prs.l.t.t != BC_LEX_EOF)
+ s = G.prs.parse(&G.prs);
if (s) return s;
s = bc_program_exec();
@@ -7158,10 +6988,17 @@ static BcStatus bc_vm_exec(void)
for (i = 0; !s && i < G.files.len; ++i)
s = bc_vm_file(*((char **) bc_vec_item(&G.files, i)));
- if (s) return s;
+ if (s) {
+ if (!G.tty)
+ return s;
+ fflush_and_check();
+ fputs("ready for more input\n", stderr);
+ }
- if (IS_BC || !G.files.len) s = bc_vm_stdin();
- if (!s && !BC_PARSE_CAN_EXEC(&G.prs)) s = bc_vm_process("");
+ if (IS_BC || !G.files.len)
+ s = bc_vm_stdin();
+ if (!s && !BC_PARSE_CAN_EXEC(&G.prs))
+ s = bc_vm_process("");
return s;
}