2018-11-03 11:00:21 -06:00
|
|
|
/* vi: set sw=4 ts=4: */
|
|
|
|
/*
|
|
|
|
* Licensed under GPLv2 or later, see file LICENSE in this source tree.
|
2018-12-18 10:03:14 -07:00
|
|
|
* Adapted from https://github.com/gavinhoward/bc
|
|
|
|
* Original code copyright (c) 2018 Gavin D. Howard and contributors.
|
2018-11-03 11:00:21 -06:00
|
|
|
*/
|
2019-01-25 14:24:03 +01:00
|
|
|
//TODO:
|
2021-06-12 14:03:24 +02:00
|
|
|
// maybe implement a^b for non-integer b? (see zbc_num_p())
|
2018-12-30 15:56:36 +01:00
|
|
|
|
|
|
|
#define DEBUG_LEXER 0
|
|
|
|
#define DEBUG_COMPILE 0
|
|
|
|
#define DEBUG_EXEC 0
|
|
|
|
// This can be left enabled for production as well:
|
|
|
|
#define SANITY_CHECKS 1
|
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
//config:config BC
|
2018-12-28 03:20:17 +01:00
|
|
|
//config: bool "bc (45 kb)"
|
2018-11-03 11:00:21 -06:00
|
|
|
//config: default y
|
2018-12-28 15:13:23 +01:00
|
|
|
//config: select FEATURE_DC_BIG
|
2018-11-03 11:00:21 -06:00
|
|
|
//config: help
|
|
|
|
//config: bc is a command-line, arbitrary-precision calculator with a
|
|
|
|
//config: Turing-complete language. See the GNU bc manual
|
|
|
|
//config: (https://www.gnu.org/software/bc/manual/bc.html) and bc spec
|
2018-12-13 16:35:52 +01:00
|
|
|
//config: (http://pubs.opengroup.org/onlinepubs/9699919799/utilities/bc.html).
|
2018-11-03 11:00:21 -06:00
|
|
|
//config:
|
2018-12-13 16:35:52 +01:00
|
|
|
//config: This bc has five differences to the GNU bc:
|
|
|
|
//config: 1) The period (.) is a shortcut for "last", as in the BSD bc.
|
2018-11-03 11:00:21 -06:00
|
|
|
//config: 2) Arrays are copied before being passed as arguments to
|
|
|
|
//config: functions. This behavior is required by the bc spec.
|
|
|
|
//config: 3) Arrays can be passed to the builtin "length" function to get
|
2018-12-13 16:35:52 +01:00
|
|
|
//config: the number of elements in the array. This prints "1":
|
|
|
|
//config: a[0] = 0; length(a[])
|
2018-11-03 11:00:21 -06:00
|
|
|
//config: 4) The precedence of the boolean "not" operator (!) is equal to
|
2018-12-13 16:35:52 +01:00
|
|
|
//config: that of the unary minus (-) negation operator. This still
|
2018-11-03 11:00:21 -06:00
|
|
|
//config: allows POSIX-compliant scripts to work while somewhat
|
|
|
|
//config: preserving expected behavior (versus C) and making parsing
|
|
|
|
//config: easier.
|
2018-12-13 16:35:52 +01:00
|
|
|
//config: 5) "read()" accepts expressions, not only numeric literals.
|
2018-11-03 11:00:21 -06:00
|
|
|
//config:
|
|
|
|
//config:config DC
|
2018-12-28 03:20:17 +01:00
|
|
|
//config: bool "dc (36 kb)"
|
2018-11-03 11:00:21 -06:00
|
|
|
//config: default y
|
|
|
|
//config: help
|
|
|
|
//config: dc is a reverse-polish notation command-line calculator which
|
|
|
|
//config: supports unlimited precision arithmetic. See the FreeBSD man page
|
|
|
|
//config: (https://www.unix.com/man-page/FreeBSD/1/dc/) and GNU dc manual
|
2018-12-13 16:35:52 +01:00
|
|
|
//config: (https://www.gnu.org/software/bc/manual/dc-1.05/html_mono/dc.html).
|
2018-11-03 11:00:21 -06:00
|
|
|
//config:
|
|
|
|
//config: This dc has a few differences from the two above:
|
2018-12-13 16:35:52 +01:00
|
|
|
//config: 1) When printing a byte stream (command "P"), this dc follows what
|
2018-11-03 11:00:21 -06:00
|
|
|
//config: the FreeBSD dc does.
|
2018-12-13 16:35:52 +01:00
|
|
|
//config: 2) Implements the GNU extensions for divmod ("~") and
|
2018-11-03 11:00:21 -06:00
|
|
|
//config: modular exponentiation ("|").
|
2018-12-13 16:35:52 +01:00
|
|
|
//config: 3) Implements all FreeBSD extensions, except for "J" and "M".
|
2018-11-03 11:00:21 -06:00
|
|
|
//config: 4) Like the FreeBSD dc, this dc supports extended registers.
|
|
|
|
//config: However, they are implemented differently. When it encounters
|
|
|
|
//config: whitespace where a register should be, it skips the whitespace.
|
|
|
|
//config: If the character following is not a lowercase letter, an error
|
|
|
|
//config: is issued. Otherwise, the register name is parsed by the
|
2018-12-27 22:52:13 +01:00
|
|
|
//config: following regex: [a-z][a-z0-9_]*
|
2018-11-03 11:00:21 -06:00
|
|
|
//config: This generally means that register names will be surrounded by
|
2018-12-13 16:35:52 +01:00
|
|
|
//config: whitespace. Examples:
|
|
|
|
//config: l idx s temp L index S temp2 < do_thing
|
2018-11-03 11:00:21 -06:00
|
|
|
//config: Also note that, like the FreeBSD dc, extended registers are not
|
|
|
|
//config: allowed unless the "-x" option is given.
|
|
|
|
//config:
|
2018-12-27 22:52:13 +01:00
|
|
|
//config:if BC || DC # for menuconfig indenting
|
|
|
|
//config:
|
|
|
|
//config:config FEATURE_DC_BIG
|
|
|
|
//config: bool "Use bc code base for dc (larger, more features)"
|
|
|
|
//config: default y
|
2018-12-06 11:31:14 +01:00
|
|
|
//config:
|
|
|
|
//config:config FEATURE_DC_LIBM
|
|
|
|
//config: bool "Enable power and exp functions (requires libm)"
|
|
|
|
//config: default y
|
2018-12-27 22:52:13 +01:00
|
|
|
//config: depends on DC && !BC && !FEATURE_DC_BIG
|
2018-12-06 11:31:14 +01:00
|
|
|
//config: help
|
|
|
|
//config: Enable power and exp functions.
|
|
|
|
//config: NOTE: This will require libm to be present for linking.
|
|
|
|
//config:
|
2018-12-27 22:52:13 +01:00
|
|
|
//config:config FEATURE_BC_INTERACTIVE
|
2018-12-13 16:35:52 +01:00
|
|
|
//config: bool "Interactive mode (+4kb)"
|
2018-11-03 11:00:21 -06:00
|
|
|
//config: default y
|
2018-12-27 22:52:13 +01:00
|
|
|
//config: depends on BC || (DC && FEATURE_DC_BIG)
|
2018-11-03 11:00:21 -06:00
|
|
|
//config: help
|
2018-12-13 16:35:52 +01:00
|
|
|
//config: Enable interactive mode: when started on a tty,
|
|
|
|
//config: ^C interrupts execution and returns to command line,
|
|
|
|
//config: errors also return to command line instead of exiting,
|
|
|
|
//config: line editing with history is available.
|
|
|
|
//config:
|
|
|
|
//config: With this option off, input can still be taken from tty,
|
|
|
|
//config: but all errors are fatal, ^C is fatal,
|
|
|
|
//config: tty is treated exactly the same as any other
|
|
|
|
//config: standard input (IOW: no line editing).
|
2018-11-03 11:00:21 -06:00
|
|
|
//config:
|
|
|
|
//config:config FEATURE_BC_LONG_OPTIONS
|
|
|
|
//config: bool "Enable bc/dc long options"
|
|
|
|
//config: default y
|
2018-12-27 22:52:13 +01:00
|
|
|
//config: depends on BC || (DC && FEATURE_DC_BIG)
|
|
|
|
//config:
|
|
|
|
//config:endif
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
//applet:IF_BC(APPLET(bc, BB_DIR_USR_BIN, BB_SUID_DROP))
|
|
|
|
//applet:IF_DC(APPLET(dc, BB_DIR_USR_BIN, BB_SUID_DROP))
|
|
|
|
|
|
|
|
//kbuild:lib-$(CONFIG_BC) += bc.o
|
|
|
|
//kbuild:lib-$(CONFIG_DC) += bc.o
|
|
|
|
|
2018-12-02 20:34:03 +01:00
|
|
|
//See www.gnu.org/software/bc/manual/bc.html
|
2018-11-03 11:00:21 -06:00
|
|
|
//usage:#define bc_trivial_usage
|
2021-06-15 10:00:18 +02:00
|
|
|
//usage: "[-sqlw] [FILE]..."
|
2018-11-03 11:00:21 -06:00
|
|
|
//usage:
|
2018-12-02 20:34:03 +01:00
|
|
|
//usage:#define bc_full_usage "\n"
|
|
|
|
//usage: "\nArbitrary precision calculator"
|
|
|
|
//usage: "\n"
|
2018-12-06 18:41:59 +01:00
|
|
|
///////: "\n -i Interactive" - has no effect for now
|
|
|
|
//usage: "\n -q Quiet"
|
2021-04-14 15:15:45 +02:00
|
|
|
//usage: "\n -l Load standard library"
|
2018-12-02 20:34:03 +01:00
|
|
|
//usage: "\n -s Be POSIX compatible"
|
|
|
|
//usage: "\n -w Warn if extensions are used"
|
|
|
|
///////: "\n -v Version"
|
2018-12-06 18:41:59 +01:00
|
|
|
//usage: "\n"
|
2018-12-06 09:20:32 +01:00
|
|
|
//usage: "\n$BC_LINE_LENGTH changes output width"
|
2018-11-03 11:00:21 -06:00
|
|
|
//usage:
|
|
|
|
//usage:#define bc_example_usage
|
|
|
|
//usage: "3 + 4.129\n"
|
|
|
|
//usage: "1903 - 2893\n"
|
|
|
|
//usage: "-129 * 213.28935\n"
|
|
|
|
//usage: "12 / -1932\n"
|
|
|
|
//usage: "12 % 12\n"
|
|
|
|
//usage: "34 ^ 189\n"
|
|
|
|
//usage: "scale = 13\n"
|
|
|
|
//usage: "ibase = 2\n"
|
|
|
|
//usage: "obase = A\n"
|
|
|
|
//usage:
|
|
|
|
//usage:#define dc_trivial_usage
|
2018-12-27 22:52:13 +01:00
|
|
|
//usage: IF_FEATURE_DC_BIG("[-x] ")"[-eSCRIPT]... [-fFILE]... [FILE]..."
|
2018-11-03 11:00:21 -06:00
|
|
|
//usage:
|
2018-12-06 11:31:14 +01:00
|
|
|
//usage:#define dc_full_usage "\n"
|
|
|
|
//usage: "\nTiny RPN calculator. Operations:"
|
2021-02-26 13:26:48 +01:00
|
|
|
//usage: "\nArithmetic: + - * / % ^"
|
|
|
|
//usage: IF_FEATURE_DC_BIG(
|
|
|
|
//usage: "\n~ - divide with remainder"
|
|
|
|
//usage: "\n| - modular exponentiation"
|
|
|
|
//usage: "\nv - square root"
|
2021-02-26 17:11:55 +01:00
|
|
|
//////// "\nA-F - digits 10..15
|
|
|
|
//////// "\n_NNN - push negative number -NNN
|
|
|
|
//////// "\n[string] - push string (in FreeBSD, \[, \] and \\ are escapes, not implemented here and in GNU)
|
2021-02-26 13:26:48 +01:00
|
|
|
//////// "\nR - DC_LEX_POP pop and discard
|
|
|
|
//////// "\nc - DC_LEX_CLEAR_STACK clear stack
|
2021-02-26 17:11:55 +01:00
|
|
|
//////// "\nd - DC_LEX_DUPLICATE duplicate top-of-stack
|
|
|
|
//////// "\nr - DC_LEX_SWAP swap top-of-stack
|
2021-02-26 13:26:48 +01:00
|
|
|
//////// "\n:r - DC_LEX_COLON pop index, pop value, store to array 'r'
|
|
|
|
//////// "\n;r - DC_LEX_SCOLON pop index, fetch from array 'r', push
|
2021-02-26 21:20:18 +01:00
|
|
|
//////// "\nLr - DC_LEX_LOAD_POP pop register 'r', push
|
2021-02-26 13:26:48 +01:00
|
|
|
//////// "\nSr - DC_LEX_STORE_PUSH pop, push to register 'r'
|
|
|
|
//////// "\nlr - DC_LEX_LOAD read register 'r', push
|
|
|
|
//////// "\nsr - DC_LEX_OP_ASSIGN pop, assign to register 'r'
|
|
|
|
//////// "\n? - DC_LEX_READ read line and execute
|
|
|
|
//////// "\nx - DC_LEX_EXECUTE pop string and execute
|
2021-02-26 17:11:55 +01:00
|
|
|
//////// "\n<r - XC_LEX_OP_REL_GT pop, pop, execute register 'r' if top-of-stack was less
|
|
|
|
//////// "\n>r - XC_LEX_OP_REL_LT pop, pop, execute register 'r' if top-of-stack was greater
|
2021-02-26 13:26:48 +01:00
|
|
|
//////// "\n=r - XC_LEX_OP_REL_EQ pop, pop, execute register 'r' if equal
|
2021-02-26 17:11:55 +01:00
|
|
|
//////// "\n !<r !>r !=r - negated forms
|
|
|
|
//////// "\n >tef - "if greater execute register 't' else execute 'f'"
|
2021-02-26 13:26:48 +01:00
|
|
|
//////// "\nQ - DC_LEX_NQUIT pop, "break N" from macro invocations
|
|
|
|
//////// "\nq - DC_LEX_QUIT "break 2" (if less than 2 levels of macros, exit dc)
|
|
|
|
//////// "\nX - DC_LEX_SCALE_FACTOR pop, push number of fractional digits
|
|
|
|
//////// "\nZ - DC_LEX_LENGTH pop, push number of digits it has (or number of characters in string)
|
2021-02-26 14:48:04 +01:00
|
|
|
//////// "\na - DC_LEX_ASCIIFY pop, push low-order byte as char or 1st char of string
|
2021-02-26 17:11:55 +01:00
|
|
|
//////// "\n( - DC_LEX_LPAREN (FreeBSD, not in GNU) pop, pop, if top-of-stack was less push 1 else push 0
|
|
|
|
//////// "\n{ - DC_LEX_LBRACE (FreeBSD, not in GNU) pop, pop, if top-of-stack was less-or-equal push 1 else push 0
|
|
|
|
//////// "\nG - DC_LEX_EQ_NO_REG (FreeBSD, not in GNU) pop, pop, if equal push 1 else push 0
|
|
|
|
//////// "\nN - DC_LEX_OP_BOOL_NOT (FreeBSD, not in GNU) pop, if 0 push 1 else push 0
|
|
|
|
//////// FreeBSD also has J and M commands, used internally by bc
|
2021-02-26 13:26:48 +01:00
|
|
|
//////// "\nn - DC_LEX_PRINT_POP pop, print without newline
|
|
|
|
//////// "\nP - DC_LEX_PRINT_STREAM pop, print string or hex bytes
|
|
|
|
//usage: )
|
2018-12-24 00:50:32 +01:00
|
|
|
//usage: "\np - print top of the stack without popping"
|
2018-12-06 18:41:59 +01:00
|
|
|
//usage: "\nf - print entire stack"
|
2021-02-26 13:26:48 +01:00
|
|
|
//////// "\nz - DC_LEX_STACK_LEVEL push stack depth
|
|
|
|
//////// "\nK - DC_LEX_SCALE push precision
|
|
|
|
//////// "\nI - DC_LEX_IBASE push input radix
|
|
|
|
//////// "\nO - DC_LEX_OBASE push output radix
|
|
|
|
//usage: IF_FEATURE_DC_BIG(
|
|
|
|
//usage: "\nk - pop the value and set precision"
|
2018-12-06 18:41:59 +01:00
|
|
|
//usage: "\ni - pop the value and set input radix"
|
2021-02-26 13:26:48 +01:00
|
|
|
//usage: )
|
2018-12-06 18:41:59 +01:00
|
|
|
//usage: "\no - pop the value and set output radix"
|
|
|
|
//usage: "\nExamples: dc -e'2 2 + p' -> 4, dc -e'8 8 * 2 2 + / p' -> 16"
|
2018-11-03 11:00:21 -06:00
|
|
|
//usage:
|
|
|
|
//usage:#define dc_example_usage
|
2018-12-06 18:41:59 +01:00
|
|
|
//usage: "$ dc -e'2 2 + p'\n"
|
2018-11-03 11:00:21 -06:00
|
|
|
//usage: "4\n"
|
2018-12-06 18:41:59 +01:00
|
|
|
//usage: "$ dc -e'8 8 \\* 2 2 + / p'\n"
|
2018-11-03 11:00:21 -06:00
|
|
|
//usage: "16\n"
|
2018-12-06 18:41:59 +01:00
|
|
|
//usage: "$ dc -e'0 1 & p'\n"
|
2018-11-03 11:00:21 -06:00
|
|
|
//usage: "0\n"
|
2018-12-06 18:41:59 +01:00
|
|
|
//usage: "$ dc -e'0 1 | p'\n"
|
2018-11-03 11:00:21 -06:00
|
|
|
//usage: "1\n"
|
2018-12-06 18:41:59 +01:00
|
|
|
//usage: "$ echo '72 9 / 8 * p' | dc\n"
|
2018-11-03 11:00:21 -06:00
|
|
|
//usage: "64\n"
|
|
|
|
|
|
|
|
#include "libbb.h"
|
2018-12-06 10:29:12 +01:00
|
|
|
#include "common_bufsiz.h"
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 22:52:13 +01:00
|
|
|
#if !ENABLE_BC && !ENABLE_FEATURE_DC_BIG
|
2018-12-06 11:31:14 +01:00
|
|
|
# include "dc.c"
|
|
|
|
#else
|
|
|
|
|
2018-12-14 17:51:17 +01:00
|
|
|
#if DEBUG_LEXER
|
2018-12-16 16:03:03 +01:00
|
|
|
static uint8_t lex_indent;
|
2018-12-14 17:51:17 +01:00
|
|
|
#define dbg_lex(...) \
|
|
|
|
do { \
|
|
|
|
fprintf(stderr, "%*s", lex_indent, ""); \
|
|
|
|
bb_error_msg(__VA_ARGS__); \
|
|
|
|
} while (0)
|
|
|
|
#define dbg_lex_enter(...) \
|
|
|
|
do { \
|
|
|
|
dbg_lex(__VA_ARGS__); \
|
|
|
|
lex_indent++; \
|
|
|
|
} while (0)
|
|
|
|
#define dbg_lex_done(...) \
|
|
|
|
do { \
|
|
|
|
lex_indent--; \
|
|
|
|
dbg_lex(__VA_ARGS__); \
|
|
|
|
} while (0)
|
2018-12-14 16:48:34 +01:00
|
|
|
#else
|
2018-12-14 17:51:17 +01:00
|
|
|
# define dbg_lex(...) ((void)0)
|
|
|
|
# define dbg_lex_enter(...) ((void)0)
|
|
|
|
# define dbg_lex_done(...) ((void)0)
|
2018-12-14 16:48:34 +01:00
|
|
|
#endif
|
|
|
|
|
2018-12-19 12:35:27 +01:00
|
|
|
#if DEBUG_COMPILE
|
|
|
|
# define dbg_compile(...) bb_error_msg(__VA_ARGS__)
|
|
|
|
#else
|
|
|
|
# define dbg_compile(...) ((void)0)
|
|
|
|
#endif
|
|
|
|
|
2018-12-15 20:06:59 +01:00
|
|
|
#if DEBUG_EXEC
|
|
|
|
# define dbg_exec(...) bb_error_msg(__VA_ARGS__)
|
|
|
|
#else
|
|
|
|
# define dbg_exec(...) ((void)0)
|
|
|
|
#endif
|
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
typedef enum BcStatus {
|
2018-12-04 20:05:28 +01:00
|
|
|
BC_STATUS_SUCCESS = 0,
|
|
|
|
BC_STATUS_FAILURE = 1,
|
2018-11-03 11:00:21 -06:00
|
|
|
} BcStatus;
|
|
|
|
|
2018-12-30 15:56:36 +01:00
|
|
|
#define BC_VEC_INVALID_IDX ((size_t) -1)
|
|
|
|
#define BC_VEC_START_CAP (1 << 5)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-10 15:37:14 +01:00
|
|
|
typedef void (*BcVecFree)(void *) FAST_FUNC;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
typedef struct BcVec {
|
|
|
|
char *v;
|
|
|
|
size_t len;
|
|
|
|
size_t cap;
|
|
|
|
size_t size;
|
|
|
|
BcVecFree dtor;
|
|
|
|
} BcVec;
|
|
|
|
|
|
|
|
typedef signed char BcDig;
|
|
|
|
|
|
|
|
typedef struct BcNum {
|
|
|
|
BcDig *restrict num;
|
|
|
|
size_t rdx;
|
|
|
|
size_t len;
|
|
|
|
size_t cap;
|
|
|
|
bool neg;
|
|
|
|
} BcNum;
|
|
|
|
|
2018-12-31 19:42:13 +01:00
|
|
|
#define BC_NUM_MAX_IBASE 36
|
2018-12-06 12:34:39 +01:00
|
|
|
// larger value might speed up BIGNUM calculations a bit:
|
2018-12-30 15:56:36 +01:00
|
|
|
#define BC_NUM_DEF_SIZE 16
|
2020-12-29 18:50:56 +01:00
|
|
|
#define BC_NUM_PRINT_WIDTH 70
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-30 15:56:36 +01:00
|
|
|
#define BC_NUM_KARATSUBA_LEN 32
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
typedef enum BcInst {
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_BC
|
2018-11-03 11:00:21 -06:00
|
|
|
BC_INST_INC_PRE,
|
|
|
|
BC_INST_DEC_PRE,
|
|
|
|
BC_INST_INC_POST,
|
|
|
|
BC_INST_DEC_POST,
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-12-24 13:20:57 +01:00
|
|
|
XC_INST_NEG, // order
|
|
|
|
|
2018-12-24 17:06:37 +01:00
|
|
|
XC_INST_REL_EQ, // should
|
|
|
|
XC_INST_REL_LE, // match
|
|
|
|
XC_INST_REL_GE, // LEX
|
|
|
|
XC_INST_REL_NE, // constants
|
|
|
|
XC_INST_REL_LT, // for
|
|
|
|
XC_INST_REL_GT, // these
|
|
|
|
|
|
|
|
XC_INST_POWER, // operations
|
|
|
|
XC_INST_MULTIPLY, // |
|
|
|
|
XC_INST_DIVIDE, // |
|
|
|
|
XC_INST_MODULUS, // |
|
|
|
|
XC_INST_PLUS, // |
|
|
|
|
XC_INST_MINUS, // |
|
2018-12-24 13:20:57 +01:00
|
|
|
|
|
|
|
XC_INST_BOOL_NOT, // |
|
|
|
|
XC_INST_BOOL_OR, // |
|
|
|
|
XC_INST_BOOL_AND, // |
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-24 13:20:57 +01:00
|
|
|
BC_INST_ASSIGN_POWER, // |
|
|
|
|
BC_INST_ASSIGN_MULTIPLY,// |
|
|
|
|
BC_INST_ASSIGN_DIVIDE, // |
|
|
|
|
BC_INST_ASSIGN_MODULUS, // |
|
|
|
|
BC_INST_ASSIGN_PLUS, // |
|
|
|
|
BC_INST_ASSIGN_MINUS, // |
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-12-24 13:20:57 +01:00
|
|
|
XC_INST_ASSIGN, // V
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
XC_INST_NUM,
|
|
|
|
XC_INST_VAR,
|
|
|
|
XC_INST_ARRAY_ELEM,
|
|
|
|
XC_INST_ARRAY,
|
|
|
|
XC_INST_SCALE_FUNC,
|
2018-12-24 00:50:32 +01:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
XC_INST_IBASE, // order of these constans should match other enums
|
|
|
|
XC_INST_OBASE, // order of these constans should match other enums
|
|
|
|
XC_INST_SCALE, // order of these constans should match other enums
|
2018-12-24 00:50:32 +01:00
|
|
|
IF_BC(BC_INST_LAST,) // order of these constans should match other enums
|
2018-12-24 04:26:07 +01:00
|
|
|
XC_INST_LENGTH,
|
|
|
|
XC_INST_READ,
|
|
|
|
XC_INST_SQRT,
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
XC_INST_PRINT,
|
|
|
|
XC_INST_PRINT_POP,
|
|
|
|
XC_INST_STR,
|
|
|
|
XC_INST_PRINT_STR,
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-22 02:23:08 +01:00
|
|
|
BC_INST_HALT,
|
2018-11-03 11:00:21 -06:00
|
|
|
BC_INST_JUMP,
|
|
|
|
BC_INST_JUMP_ZERO,
|
|
|
|
|
|
|
|
BC_INST_CALL,
|
|
|
|
BC_INST_RET0,
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-12-24 04:26:07 +01:00
|
|
|
XC_INST_RET,
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
XC_INST_POP,
|
2018-12-22 01:34:10 +01:00
|
|
|
#if ENABLE_DC
|
2018-12-24 04:26:07 +01:00
|
|
|
DC_INST_POP_EXEC,
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
DC_INST_MODEXP,
|
|
|
|
DC_INST_DIVMOD,
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
DC_INST_EXECUTE,
|
|
|
|
DC_INST_EXEC_COND,
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
DC_INST_ASCIIFY,
|
|
|
|
DC_INST_PRINT_STREAM,
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
DC_INST_PRINT_STACK,
|
|
|
|
DC_INST_CLEAR_STACK,
|
|
|
|
DC_INST_STACK_LEN,
|
|
|
|
DC_INST_DUPLICATE,
|
|
|
|
DC_INST_SWAP,
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
DC_INST_LOAD,
|
|
|
|
DC_INST_PUSH_VAR,
|
|
|
|
DC_INST_PUSH_TO_VAR,
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
DC_INST_QUIT,
|
|
|
|
DC_INST_NQUIT,
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
DC_INST_INVALID = -1,
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
} BcInst;
|
|
|
|
|
|
|
|
typedef struct BcId {
|
|
|
|
char *name;
|
|
|
|
size_t idx;
|
|
|
|
} BcId;
|
|
|
|
|
|
|
|
typedef struct BcFunc {
|
|
|
|
BcVec code;
|
2018-12-20 16:24:18 +01:00
|
|
|
IF_BC(BcVec labels;)
|
|
|
|
IF_BC(BcVec autos;)
|
2018-12-21 16:22:26 +01:00
|
|
|
IF_BC(BcVec strs;)
|
|
|
|
IF_BC(BcVec consts;)
|
2018-12-20 16:24:18 +01:00
|
|
|
IF_BC(size_t nparams;)
|
2019-01-04 13:58:46 +01:00
|
|
|
IF_BC(bool voidfunc;)
|
2018-11-03 11:00:21 -06:00
|
|
|
} BcFunc;
|
|
|
|
|
|
|
|
typedef enum BcResultType {
|
2018-12-24 23:41:31 +01:00
|
|
|
XC_RESULT_TEMP,
|
2019-01-04 13:58:46 +01:00
|
|
|
IF_BC(BC_RESULT_VOID,) // same as TEMP, but INST_PRINT will ignore it
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 23:41:31 +01:00
|
|
|
XC_RESULT_VAR,
|
|
|
|
XC_RESULT_ARRAY_ELEM,
|
|
|
|
XC_RESULT_ARRAY,
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 23:41:31 +01:00
|
|
|
XC_RESULT_STR,
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 23:41:31 +01:00
|
|
|
//code uses "inst - XC_INST_IBASE + XC_RESULT_IBASE" construct,
|
|
|
|
XC_RESULT_IBASE, // relative order should match for: XC_INST_IBASE
|
|
|
|
XC_RESULT_OBASE, // relative order should match for: XC_INST_OBASE
|
|
|
|
XC_RESULT_SCALE, // relative order should match for: XC_INST_SCALE
|
2018-12-24 00:50:32 +01:00
|
|
|
IF_BC(BC_RESULT_LAST,) // relative order should match for: BC_INST_LAST
|
2018-12-24 23:41:31 +01:00
|
|
|
XC_RESULT_CONSTANT,
|
|
|
|
IF_BC(BC_RESULT_ONE,)
|
2018-11-03 11:00:21 -06:00
|
|
|
} BcResultType;
|
|
|
|
|
|
|
|
typedef union BcResultData {
|
|
|
|
BcNum n;
|
|
|
|
BcVec v;
|
|
|
|
BcId id;
|
|
|
|
} BcResultData;
|
|
|
|
|
|
|
|
typedef struct BcResult {
|
|
|
|
BcResultType t;
|
|
|
|
BcResultData d;
|
|
|
|
} BcResult;
|
|
|
|
|
|
|
|
typedef struct BcInstPtr {
|
|
|
|
size_t func;
|
2018-12-21 23:01:26 +01:00
|
|
|
size_t inst_idx;
|
2018-11-03 11:00:21 -06:00
|
|
|
} BcInstPtr;
|
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
typedef enum BcType {
|
|
|
|
BC_TYPE_VAR,
|
|
|
|
BC_TYPE_ARRAY,
|
|
|
|
BC_TYPE_REF,
|
|
|
|
} BcType;
|
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
typedef enum BcLexType {
|
2018-12-24 12:25:20 +01:00
|
|
|
XC_LEX_EOF,
|
|
|
|
XC_LEX_INVALID,
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 15:05:49 +01:00
|
|
|
XC_LEX_NLINE,
|
|
|
|
XC_LEX_WHITESPACE,
|
|
|
|
XC_LEX_STR,
|
|
|
|
XC_LEX_NAME,
|
|
|
|
XC_LEX_NUMBER,
|
2018-12-24 15:00:56 +01:00
|
|
|
|
2018-12-24 14:14:23 +01:00
|
|
|
XC_LEX_1st_op,
|
|
|
|
XC_LEX_NEG = XC_LEX_1st_op, // order
|
2018-12-24 13:20:57 +01:00
|
|
|
|
2018-12-24 17:06:37 +01:00
|
|
|
XC_LEX_OP_REL_EQ, // should
|
|
|
|
XC_LEX_OP_REL_LE, // match
|
|
|
|
XC_LEX_OP_REL_GE, // INST
|
|
|
|
XC_LEX_OP_REL_NE, // constants
|
|
|
|
XC_LEX_OP_REL_LT, // for
|
|
|
|
XC_LEX_OP_REL_GT, // these
|
|
|
|
|
|
|
|
XC_LEX_OP_POWER, // operations
|
|
|
|
XC_LEX_OP_MULTIPLY, // |
|
|
|
|
XC_LEX_OP_DIVIDE, // |
|
|
|
|
XC_LEX_OP_MODULUS, // |
|
|
|
|
XC_LEX_OP_PLUS, // |
|
|
|
|
XC_LEX_OP_MINUS, // |
|
|
|
|
XC_LEX_OP_last = XC_LEX_OP_MINUS,
|
2018-12-24 15:00:56 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-24 13:20:57 +01:00
|
|
|
BC_LEX_OP_BOOL_NOT, // |
|
|
|
|
BC_LEX_OP_BOOL_OR, // |
|
|
|
|
BC_LEX_OP_BOOL_AND, // |
|
|
|
|
|
|
|
|
BC_LEX_OP_ASSIGN_POWER, // |
|
|
|
|
BC_LEX_OP_ASSIGN_MULTIPLY, // |
|
|
|
|
BC_LEX_OP_ASSIGN_DIVIDE, // |
|
|
|
|
BC_LEX_OP_ASSIGN_MODULUS, // |
|
|
|
|
BC_LEX_OP_ASSIGN_PLUS, // |
|
|
|
|
BC_LEX_OP_ASSIGN_MINUS, // |
|
|
|
|
|
|
|
|
BC_LEX_OP_ASSIGN, // V
|
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
BC_LEX_OP_INC,
|
|
|
|
BC_LEX_OP_DEC,
|
|
|
|
|
2018-12-24 17:06:37 +01:00
|
|
|
BC_LEX_LPAREN, // () are 0x28 and 0x29
|
|
|
|
BC_LEX_RPAREN, // must be LPAREN+1: code uses (c - '(' + BC_LEX_LPAREN)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
BC_LEX_LBRACKET, // [] are 0x5B and 0x5D
|
2018-11-03 11:00:21 -06:00
|
|
|
BC_LEX_COMMA,
|
2018-12-24 17:06:37 +01:00
|
|
|
BC_LEX_RBRACKET, // must be LBRACKET+2: code uses (c - '[' + BC_LEX_LBRACKET)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 17:06:37 +01:00
|
|
|
BC_LEX_LBRACE, // {} are 0x7B and 0x7D
|
2018-11-03 11:00:21 -06:00
|
|
|
BC_LEX_SCOLON,
|
2018-12-24 17:06:37 +01:00
|
|
|
BC_LEX_RBRACE, // must be LBRACE+2: code uses (c - '{' + BC_LEX_LBRACE)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-05 00:22:34 +01:00
|
|
|
BC_LEX_KEY_1st_keyword,
|
|
|
|
BC_LEX_KEY_AUTO = BC_LEX_KEY_1st_keyword,
|
2018-11-03 11:00:21 -06:00
|
|
|
BC_LEX_KEY_BREAK,
|
|
|
|
BC_LEX_KEY_CONTINUE,
|
|
|
|
BC_LEX_KEY_DEFINE,
|
|
|
|
BC_LEX_KEY_ELSE,
|
|
|
|
BC_LEX_KEY_FOR,
|
|
|
|
BC_LEX_KEY_HALT,
|
2018-12-24 04:26:07 +01:00
|
|
|
// code uses "type - BC_LEX_KEY_IBASE + XC_INST_IBASE" construct,
|
2018-12-30 15:56:36 +01:00
|
|
|
BC_LEX_KEY_IBASE, // relative order should match for: XC_INST_IBASE
|
|
|
|
BC_LEX_KEY_OBASE, // relative order should match for: XC_INST_OBASE
|
2018-11-03 11:00:21 -06:00
|
|
|
BC_LEX_KEY_IF,
|
2018-12-30 15:56:36 +01:00
|
|
|
BC_LEX_KEY_LAST, // relative order should match for: BC_INST_LAST
|
2018-11-03 11:00:21 -06:00
|
|
|
BC_LEX_KEY_LENGTH,
|
|
|
|
BC_LEX_KEY_LIMITS,
|
|
|
|
BC_LEX_KEY_PRINT,
|
|
|
|
BC_LEX_KEY_QUIT,
|
|
|
|
BC_LEX_KEY_READ,
|
|
|
|
BC_LEX_KEY_RETURN,
|
|
|
|
BC_LEX_KEY_SCALE,
|
|
|
|
BC_LEX_KEY_SQRT,
|
|
|
|
BC_LEX_KEY_WHILE,
|
2018-12-24 15:00:56 +01:00
|
|
|
#endif // ENABLE_BC
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_DC
|
2018-12-24 15:00:56 +01:00
|
|
|
DC_LEX_OP_BOOL_NOT = XC_LEX_OP_last + 1,
|
|
|
|
DC_LEX_OP_ASSIGN,
|
|
|
|
|
|
|
|
DC_LEX_LPAREN,
|
|
|
|
DC_LEX_SCOLON,
|
|
|
|
DC_LEX_READ,
|
|
|
|
DC_LEX_IBASE,
|
|
|
|
DC_LEX_SCALE,
|
|
|
|
DC_LEX_OBASE,
|
|
|
|
DC_LEX_LENGTH,
|
|
|
|
DC_LEX_PRINT,
|
|
|
|
DC_LEX_QUIT,
|
|
|
|
DC_LEX_SQRT,
|
|
|
|
DC_LEX_LBRACE,
|
|
|
|
|
2018-12-24 12:25:20 +01:00
|
|
|
DC_LEX_EQ_NO_REG,
|
|
|
|
DC_LEX_OP_MODEXP,
|
|
|
|
DC_LEX_OP_DIVMOD,
|
|
|
|
|
|
|
|
DC_LEX_COLON,
|
|
|
|
DC_LEX_ELSE,
|
|
|
|
DC_LEX_EXECUTE,
|
|
|
|
DC_LEX_PRINT_STACK,
|
|
|
|
DC_LEX_CLEAR_STACK,
|
|
|
|
DC_LEX_STACK_LEVEL,
|
|
|
|
DC_LEX_DUPLICATE,
|
|
|
|
DC_LEX_SWAP,
|
|
|
|
DC_LEX_POP,
|
|
|
|
|
|
|
|
DC_LEX_ASCIIFY,
|
|
|
|
DC_LEX_PRINT_STREAM,
|
|
|
|
|
|
|
|
// code uses "t - DC_LEX_STORE_IBASE + XC_INST_IBASE" construct,
|
|
|
|
DC_LEX_STORE_IBASE, // relative order should match for: XC_INST_IBASE
|
|
|
|
DC_LEX_STORE_OBASE, // relative order should match for: XC_INST_OBASE
|
|
|
|
DC_LEX_STORE_SCALE, // relative order should match for: XC_INST_SCALE
|
|
|
|
DC_LEX_LOAD,
|
|
|
|
DC_LEX_LOAD_POP,
|
|
|
|
DC_LEX_STORE_PUSH,
|
|
|
|
DC_LEX_PRINT_POP,
|
|
|
|
DC_LEX_NQUIT,
|
|
|
|
DC_LEX_SCALE_FACTOR,
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
} BcLexType;
|
2018-12-05 00:22:34 +01:00
|
|
|
// must match order of BC_LEX_KEY_foo etc above
|
|
|
|
#if ENABLE_BC
|
|
|
|
struct BcLexKeyword {
|
|
|
|
char name8[8];
|
|
|
|
};
|
2018-12-25 21:44:10 +01:00
|
|
|
#define LEX_KW_ENTRY(a, b) \
|
2018-12-06 12:59:40 +01:00
|
|
|
{ .name8 = a /*, .posix = b */ }
|
2020-11-30 14:58:02 +01:00
|
|
|
static const struct BcLexKeyword bc_lex_kws[20] ALIGN8 = {
|
2018-12-25 21:44:10 +01:00
|
|
|
LEX_KW_ENTRY("auto" , 1), // 0
|
|
|
|
LEX_KW_ENTRY("break" , 1), // 1
|
|
|
|
LEX_KW_ENTRY("continue", 0), // 2 note: this one has no terminating NUL
|
|
|
|
LEX_KW_ENTRY("define" , 1), // 3
|
|
|
|
LEX_KW_ENTRY("else" , 0), // 4
|
|
|
|
LEX_KW_ENTRY("for" , 1), // 5
|
|
|
|
LEX_KW_ENTRY("halt" , 0), // 6
|
|
|
|
LEX_KW_ENTRY("ibase" , 1), // 7
|
|
|
|
LEX_KW_ENTRY("obase" , 1), // 8
|
|
|
|
LEX_KW_ENTRY("if" , 1), // 9
|
|
|
|
LEX_KW_ENTRY("last" , 0), // 10
|
|
|
|
LEX_KW_ENTRY("length" , 1), // 11
|
|
|
|
LEX_KW_ENTRY("limits" , 0), // 12
|
|
|
|
LEX_KW_ENTRY("print" , 0), // 13
|
|
|
|
LEX_KW_ENTRY("quit" , 1), // 14
|
|
|
|
LEX_KW_ENTRY("read" , 0), // 15
|
|
|
|
LEX_KW_ENTRY("return" , 1), // 16
|
|
|
|
LEX_KW_ENTRY("scale" , 1), // 17
|
|
|
|
LEX_KW_ENTRY("sqrt" , 1), // 18
|
|
|
|
LEX_KW_ENTRY("while" , 1), // 19
|
2018-12-05 00:22:34 +01:00
|
|
|
};
|
2018-12-25 21:44:10 +01:00
|
|
|
#undef LEX_KW_ENTRY
|
2018-12-17 10:42:31 +01:00
|
|
|
#define STRING_else (bc_lex_kws[4].name8)
|
|
|
|
#define STRING_for (bc_lex_kws[5].name8)
|
2018-12-24 00:50:32 +01:00
|
|
|
#define STRING_if (bc_lex_kws[9].name8)
|
|
|
|
#define STRING_while (bc_lex_kws[19].name8)
|
2018-12-05 00:22:34 +01:00
|
|
|
enum {
|
|
|
|
POSIX_KWORD_MASK = 0
|
2018-12-14 16:30:56 +01:00
|
|
|
| (1 << 0) // 0
|
|
|
|
| (1 << 1) // 1
|
|
|
|
| (0 << 2) // 2
|
|
|
|
| (1 << 3) // 3
|
|
|
|
| (0 << 4) // 4
|
|
|
|
| (1 << 5) // 5
|
|
|
|
| (0 << 6) // 6
|
|
|
|
| (1 << 7) // 7
|
|
|
|
| (1 << 8) // 8
|
2018-12-24 00:50:32 +01:00
|
|
|
| (1 << 9) // 9
|
|
|
|
| (0 << 10) // 10
|
|
|
|
| (1 << 11) // 11
|
|
|
|
| (0 << 12) // 12
|
2018-12-14 16:30:56 +01:00
|
|
|
| (0 << 13) // 13
|
|
|
|
| (1 << 14) // 14
|
|
|
|
| (0 << 15) // 15
|
|
|
|
| (1 << 16) // 16
|
|
|
|
| (1 << 17) // 17
|
|
|
|
| (1 << 18) // 18
|
|
|
|
| (1 << 19) // 19
|
2018-12-05 00:22:34 +01:00
|
|
|
};
|
2018-12-25 21:44:10 +01:00
|
|
|
#define keyword_is_POSIX(i) ((1 << (i)) & POSIX_KWORD_MASK)
|
2018-12-17 16:54:37 +01:00
|
|
|
|
|
|
|
// This is a bit array that corresponds to token types. An entry is
|
|
|
|
// true if the token is valid in an expression, false otherwise.
|
|
|
|
// Used to figure out when expr parsing should stop *without error message*
|
|
|
|
// - 0 element indicates this condition. 1 means "this token is to be eaten
|
2018-12-24 00:50:32 +01:00
|
|
|
// as part of the expression", it can then still be determined to be invalid
|
2018-12-17 16:54:37 +01:00
|
|
|
// by later processing.
|
|
|
|
enum {
|
|
|
|
#define EXBITS(a,b,c,d,e,f,g,h) \
|
|
|
|
((uint64_t)((a << 0)+(b << 1)+(c << 2)+(d << 3)+(e << 4)+(f << 5)+(g << 6)+(h << 7)))
|
2018-12-24 00:50:32 +01:00
|
|
|
BC_PARSE_EXPRS_BITS = 0 // corresponding BC_LEX_xyz:
|
2018-12-24 15:00:56 +01:00
|
|
|
+ (EXBITS(0,0,0,0,0,1,1,1) << (0*8)) // 0: EOF INVAL NL WS STR NAME NUM -
|
2018-12-24 17:06:37 +01:00
|
|
|
+ (EXBITS(1,1,1,1,1,1,1,1) << (1*8)) // 8: == <= >= != < > ^ *
|
|
|
|
+ (EXBITS(1,1,1,1,1,1,1,1) << (2*8)) // 16: / % + - ! || && ^=
|
2018-12-24 15:00:56 +01:00
|
|
|
+ (EXBITS(1,1,1,1,1,1,1,1) << (3*8)) // 24: *= /= %= += -= = ++ --
|
|
|
|
+ (EXBITS(1,1,0,0,0,0,0,0) << (4*8)) // 32: ( ) [ , ] { ; }
|
2018-12-24 13:20:57 +01:00
|
|
|
+ (EXBITS(0,0,0,0,0,0,0,1) << (5*8)) // 40: auto break cont define else for halt ibase
|
|
|
|
+ (EXBITS(1,0,1,1,0,0,0,1) << (6*8)) // 48: obase if last length limits print quit read
|
|
|
|
+ (EXBITS(0,1,1,0,0,0,0,0) << (7*8)) // 56: return scale sqrt while
|
2018-12-17 16:54:37 +01:00
|
|
|
#undef EXBITS
|
|
|
|
};
|
2018-12-25 01:21:16 +01:00
|
|
|
static ALWAYS_INLINE long lex_allowed_in_bc_expr(unsigned i)
|
2018-12-17 16:54:37 +01:00
|
|
|
{
|
|
|
|
#if ULONG_MAX > 0xffffffff
|
|
|
|
// 64-bit version (will not work correctly for 32-bit longs!)
|
|
|
|
return BC_PARSE_EXPRS_BITS & (1UL << i);
|
|
|
|
#else
|
|
|
|
// 32-bit version
|
|
|
|
unsigned long m = (uint32_t)BC_PARSE_EXPRS_BITS;
|
|
|
|
if (i >= 32) {
|
|
|
|
m = (uint32_t)(BC_PARSE_EXPRS_BITS >> 32);
|
|
|
|
i &= 31;
|
|
|
|
}
|
|
|
|
return m & (1UL << i);
|
2018-12-05 00:22:34 +01:00
|
|
|
#endif
|
2018-12-17 16:54:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// This is an array of data for operators that correspond to
|
2018-12-24 14:14:23 +01:00
|
|
|
// [XC_LEX_1st_op...] token types.
|
2018-12-30 15:56:36 +01:00
|
|
|
static const uint8_t bc_ops_prec_and_assoc[] ALIGN1 = {
|
2018-12-17 16:54:37 +01:00
|
|
|
#define OP(p,l) ((int)(l) * 0x10 + (p))
|
|
|
|
OP(1, false), // neg
|
2018-12-24 17:06:37 +01:00
|
|
|
OP(6, true ), OP( 6, true ), OP( 6, true ), OP( 6, true ), OP( 6, true ), OP( 6, true ), // == <= >= != < >
|
2018-12-17 16:54:37 +01:00
|
|
|
OP(2, false), // pow
|
|
|
|
OP(3, true ), OP( 3, true ), OP( 3, true ), // mul div mod
|
|
|
|
OP(4, true ), OP( 4, true ), // + -
|
|
|
|
OP(1, false), // not
|
|
|
|
OP(7, true ), OP( 7, true ), // or and
|
|
|
|
OP(5, false), OP( 5, false ), OP( 5, false ), OP( 5, false ), OP( 5, false ), // ^= *= /= %= +=
|
|
|
|
OP(5, false), OP( 5, false ), // -= =
|
2018-12-24 13:20:57 +01:00
|
|
|
OP(0, false), OP( 0, false ), // inc dec
|
2018-12-17 16:54:37 +01:00
|
|
|
#undef OP
|
|
|
|
};
|
2018-12-30 15:56:36 +01:00
|
|
|
#define bc_operation_PREC(i) (bc_ops_prec_and_assoc[i] & 0x0f)
|
|
|
|
#define bc_operation_LEFT(i) (bc_ops_prec_and_assoc[i] & 0x10)
|
2018-12-17 16:54:37 +01:00
|
|
|
#endif // ENABLE_BC
|
|
|
|
|
|
|
|
#if ENABLE_DC
|
2018-12-24 01:02:59 +01:00
|
|
|
static const //BcLexType - should be this type
|
|
|
|
uint8_t
|
2018-12-26 21:17:12 +01:00
|
|
|
dc_char_to_LEX[] ALIGN1 = {
|
2018-12-27 18:23:58 +01:00
|
|
|
// %&'(
|
2018-12-24 15:00:56 +01:00
|
|
|
XC_LEX_OP_MODULUS, XC_LEX_INVALID, XC_LEX_INVALID, DC_LEX_LPAREN,
|
2018-12-27 18:23:58 +01:00
|
|
|
// )*+,
|
2018-12-24 14:14:23 +01:00
|
|
|
XC_LEX_INVALID, XC_LEX_OP_MULTIPLY, XC_LEX_OP_PLUS, XC_LEX_INVALID,
|
2018-12-27 18:23:58 +01:00
|
|
|
// -./
|
2018-12-24 14:14:23 +01:00
|
|
|
XC_LEX_OP_MINUS, XC_LEX_INVALID, XC_LEX_OP_DIVIDE,
|
2018-12-27 18:23:58 +01:00
|
|
|
// 0123456789
|
2018-12-24 12:25:20 +01:00
|
|
|
XC_LEX_INVALID, XC_LEX_INVALID, XC_LEX_INVALID, XC_LEX_INVALID,
|
|
|
|
XC_LEX_INVALID, XC_LEX_INVALID, XC_LEX_INVALID, XC_LEX_INVALID,
|
|
|
|
XC_LEX_INVALID, XC_LEX_INVALID,
|
2018-12-27 18:23:58 +01:00
|
|
|
// :;<=>?@
|
2018-12-24 15:00:56 +01:00
|
|
|
DC_LEX_COLON, DC_LEX_SCOLON, XC_LEX_OP_REL_GT, XC_LEX_OP_REL_EQ,
|
|
|
|
XC_LEX_OP_REL_LT, DC_LEX_READ, XC_LEX_INVALID,
|
2018-12-27 18:23:58 +01:00
|
|
|
// ABCDEFGH
|
2018-12-24 12:25:20 +01:00
|
|
|
XC_LEX_INVALID, XC_LEX_INVALID, XC_LEX_INVALID, XC_LEX_INVALID,
|
|
|
|
XC_LEX_INVALID, XC_LEX_INVALID, DC_LEX_EQ_NO_REG, XC_LEX_INVALID,
|
2018-12-27 18:23:58 +01:00
|
|
|
// IJKLMNOP
|
2018-12-24 15:00:56 +01:00
|
|
|
DC_LEX_IBASE, XC_LEX_INVALID, DC_LEX_SCALE, DC_LEX_LOAD_POP,
|
|
|
|
XC_LEX_INVALID, DC_LEX_OP_BOOL_NOT, DC_LEX_OBASE, DC_LEX_PRINT_STREAM,
|
2019-01-04 15:54:40 +01:00
|
|
|
// QRSTUVWX
|
|
|
|
DC_LEX_NQUIT, DC_LEX_POP, DC_LEX_STORE_PUSH, XC_LEX_INVALID,
|
|
|
|
XC_LEX_INVALID, XC_LEX_INVALID, XC_LEX_INVALID, DC_LEX_SCALE_FACTOR,
|
|
|
|
// YZ
|
|
|
|
XC_LEX_INVALID, DC_LEX_LENGTH,
|
|
|
|
// [\]
|
|
|
|
XC_LEX_INVALID, XC_LEX_INVALID, XC_LEX_INVALID,
|
2018-12-27 18:23:58 +01:00
|
|
|
// ^_`
|
2018-12-24 14:14:23 +01:00
|
|
|
XC_LEX_OP_POWER, XC_LEX_NEG, XC_LEX_INVALID,
|
2018-12-27 18:23:58 +01:00
|
|
|
// abcdefgh
|
2018-12-24 12:25:20 +01:00
|
|
|
DC_LEX_ASCIIFY, XC_LEX_INVALID, DC_LEX_CLEAR_STACK, DC_LEX_DUPLICATE,
|
|
|
|
DC_LEX_ELSE, DC_LEX_PRINT_STACK, XC_LEX_INVALID, XC_LEX_INVALID,
|
2018-12-27 18:23:58 +01:00
|
|
|
// ijklmnop
|
2018-12-24 12:25:20 +01:00
|
|
|
DC_LEX_STORE_IBASE, XC_LEX_INVALID, DC_LEX_STORE_SCALE, DC_LEX_LOAD,
|
2018-12-24 15:00:56 +01:00
|
|
|
XC_LEX_INVALID, DC_LEX_PRINT_POP, DC_LEX_STORE_OBASE, DC_LEX_PRINT,
|
2018-12-27 18:23:58 +01:00
|
|
|
// qrstuvwx
|
2018-12-24 15:00:56 +01:00
|
|
|
DC_LEX_QUIT, DC_LEX_SWAP, DC_LEX_OP_ASSIGN, XC_LEX_INVALID,
|
|
|
|
XC_LEX_INVALID, DC_LEX_SQRT, XC_LEX_INVALID, DC_LEX_EXECUTE,
|
2018-12-27 18:23:58 +01:00
|
|
|
// yz
|
2018-12-24 12:25:20 +01:00
|
|
|
XC_LEX_INVALID, DC_LEX_STACK_LEVEL,
|
2018-12-27 18:23:58 +01:00
|
|
|
// {|}~
|
2018-12-24 15:00:56 +01:00
|
|
|
DC_LEX_LBRACE, DC_LEX_OP_MODEXP, XC_LEX_INVALID, DC_LEX_OP_DIVMOD,
|
2018-12-24 01:02:59 +01:00
|
|
|
};
|
2018-12-24 04:26:07 +01:00
|
|
|
static const //BcInst - should be this type. Using signed narrow type since DC_INST_INVALID is -1
|
2018-12-17 16:54:37 +01:00
|
|
|
int8_t
|
2018-12-27 18:23:58 +01:00
|
|
|
dc_LEX_to_INST[] ALIGN1 = { //starts at XC_LEX_OP_POWER // corresponding XC/DC_LEX_xyz:
|
|
|
|
XC_INST_POWER, XC_INST_MULTIPLY, // XC_LEX_OP_POWER XC_LEX_OP_MULTIPLY
|
|
|
|
XC_INST_DIVIDE, XC_INST_MODULUS, // XC_LEX_OP_DIVIDE XC_LEX_OP_MODULUS
|
|
|
|
XC_INST_PLUS, XC_INST_MINUS, // XC_LEX_OP_PLUS XC_LEX_OP_MINUS
|
2018-12-24 15:29:08 +01:00
|
|
|
XC_INST_BOOL_NOT, // DC_LEX_OP_BOOL_NOT
|
|
|
|
DC_INST_INVALID, // DC_LEX_OP_ASSIGN
|
|
|
|
XC_INST_REL_GT, // DC_LEX_LPAREN
|
|
|
|
DC_INST_INVALID, // DC_LEX_SCOLON
|
|
|
|
DC_INST_INVALID, // DC_LEX_READ
|
|
|
|
XC_INST_IBASE, // DC_LEX_IBASE
|
|
|
|
XC_INST_SCALE, // DC_LEX_SCALE
|
|
|
|
XC_INST_OBASE, // DC_LEX_OBASE
|
|
|
|
XC_INST_LENGTH, // DC_LEX_LENGTH
|
|
|
|
XC_INST_PRINT, // DC_LEX_PRINT
|
|
|
|
DC_INST_QUIT, // DC_LEX_QUIT
|
|
|
|
XC_INST_SQRT, // DC_LEX_SQRT
|
|
|
|
XC_INST_REL_GE, // DC_LEX_LBRACE
|
|
|
|
XC_INST_REL_EQ, // DC_LEX_EQ_NO_REG
|
2018-12-27 18:23:58 +01:00
|
|
|
DC_INST_MODEXP, DC_INST_DIVMOD, // DC_LEX_OP_MODEXP DC_LEX_OP_DIVMOD
|
|
|
|
DC_INST_INVALID, DC_INST_INVALID, // DC_LEX_COLON DC_LEX_ELSE
|
|
|
|
DC_INST_EXECUTE, // DC_LEX_EXECUTE
|
|
|
|
DC_INST_PRINT_STACK, DC_INST_CLEAR_STACK, // DC_LEX_PRINT_STACK DC_LEX_CLEAR_STACK
|
|
|
|
DC_INST_STACK_LEN, DC_INST_DUPLICATE, // DC_LEX_STACK_LEVEL DC_LEX_DUPLICATE
|
|
|
|
DC_INST_SWAP, XC_INST_POP, // DC_LEX_SWAP DC_LEX_POP
|
|
|
|
DC_INST_ASCIIFY, DC_INST_PRINT_STREAM, // DC_LEX_ASCIIFY DC_LEX_PRINT_STREAM
|
|
|
|
DC_INST_INVALID, DC_INST_INVALID, // DC_LEX_STORE_IBASE DC_LEX_STORE_OBASE
|
|
|
|
DC_INST_INVALID, DC_INST_INVALID, // DC_LEX_STORE_SCALE DC_LEX_LOAD
|
|
|
|
DC_INST_INVALID, DC_INST_INVALID, // DC_LEX_LOAD_POP DC_LEX_STORE_PUSH
|
|
|
|
XC_INST_PRINT, DC_INST_NQUIT, // DC_LEX_PRINT_POP DC_LEX_NQUIT
|
|
|
|
XC_INST_SCALE_FUNC, // DC_LEX_SCALE_FACTOR
|
2018-12-24 15:00:56 +01:00
|
|
|
// DC_INST_INVALID in this table either means that corresponding LEX
|
|
|
|
// is not possible for dc, or that it does not compile one-to-one
|
|
|
|
// to a single INST.
|
2018-12-17 16:54:37 +01:00
|
|
|
};
|
|
|
|
#endif // ENABLE_DC
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
typedef struct BcParse {
|
2018-12-25 22:20:14 +01:00
|
|
|
smallint lex; // was BcLexType // first member is most used
|
|
|
|
smallint lex_last; // was BcLexType
|
2018-12-25 21:44:10 +01:00
|
|
|
size_t lex_line;
|
2018-12-25 22:20:14 +01:00
|
|
|
const char *lex_inbuf;
|
|
|
|
const char *lex_next_at; // last lex_next() was called at this string
|
2018-12-25 22:32:41 +01:00
|
|
|
const char *lex_filename;
|
|
|
|
FILE *lex_input_fp;
|
2018-12-25 21:44:10 +01:00
|
|
|
BcVec lex_strnumbuf;
|
2018-12-25 19:38:13 +01:00
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
BcFunc *func;
|
|
|
|
size_t fidx;
|
2018-12-20 16:24:18 +01:00
|
|
|
IF_BC(size_t in_funcdef;)
|
2018-12-25 22:20:14 +01:00
|
|
|
IF_BC(BcVec exits;)
|
|
|
|
IF_BC(BcVec conds;)
|
|
|
|
IF_BC(BcVec ops;)
|
2018-11-03 11:00:21 -06:00
|
|
|
} BcParse;
|
|
|
|
|
|
|
|
typedef struct BcProgram {
|
|
|
|
size_t len;
|
2018-12-20 16:24:18 +01:00
|
|
|
size_t nchars;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-20 16:24:18 +01:00
|
|
|
size_t scale;
|
2018-11-03 11:00:21 -06:00
|
|
|
size_t ib_t;
|
|
|
|
size_t ob_t;
|
|
|
|
|
|
|
|
BcVec results;
|
2018-12-19 12:35:27 +01:00
|
|
|
BcVec exestack;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
BcVec fns;
|
2018-12-20 20:34:09 +01:00
|
|
|
IF_BC(BcVec fn_map;)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
BcVec vars;
|
|
|
|
BcVec var_map;
|
|
|
|
|
|
|
|
BcVec arrs;
|
|
|
|
BcVec arr_map;
|
|
|
|
|
2018-12-21 16:22:26 +01:00
|
|
|
IF_DC(BcVec strs;)
|
|
|
|
IF_DC(BcVec consts;)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
BcNum zero;
|
2018-12-20 16:24:18 +01:00
|
|
|
IF_BC(BcNum one;)
|
|
|
|
IF_BC(BcNum last;)
|
2018-11-03 11:00:21 -06:00
|
|
|
} BcProgram;
|
|
|
|
|
2018-12-25 22:20:14 +01:00
|
|
|
struct globals {
|
|
|
|
BcParse prs; // first member is most used
|
|
|
|
|
|
|
|
// For error messages. Can be set to current parsed line,
|
|
|
|
// or [TODO] to current executing line (can be before last parsed one)
|
2019-01-02 16:30:24 +01:00
|
|
|
size_t err_line;
|
2018-12-25 22:20:14 +01:00
|
|
|
|
|
|
|
BcVec input_buffer;
|
|
|
|
|
2018-12-27 22:52:13 +01:00
|
|
|
IF_FEATURE_BC_INTERACTIVE(smallint ttyin;)
|
2018-12-25 22:20:14 +01:00
|
|
|
IF_FEATURE_CLEAN_UP(smallint exiting;)
|
|
|
|
|
|
|
|
BcProgram prog;
|
|
|
|
|
|
|
|
BcVec files;
|
|
|
|
|
|
|
|
char *env_args;
|
|
|
|
|
|
|
|
#if ENABLE_FEATURE_EDITING
|
|
|
|
line_input_t *line_input_state;
|
|
|
|
#endif
|
|
|
|
} FIX_ALIASING;
|
|
|
|
#define G (*ptr_to_globals)
|
|
|
|
#define INIT_G() do { \
|
|
|
|
SET_PTR_TO_GLOBALS(xzalloc(sizeof(G))); \
|
|
|
|
} while (0)
|
|
|
|
#define FREE_G() do { \
|
|
|
|
FREE_PTR_TO_GLOBALS(); \
|
|
|
|
} while (0)
|
|
|
|
#define G_posix (ENABLE_BC && (option_mask32 & BC_FLAG_S))
|
|
|
|
#define G_warn (ENABLE_BC && (option_mask32 & BC_FLAG_W))
|
|
|
|
#define G_exreg (ENABLE_DC && (option_mask32 & DC_FLAG_X))
|
2018-12-27 22:52:13 +01:00
|
|
|
#if ENABLE_FEATURE_BC_INTERACTIVE
|
2018-12-25 22:20:14 +01:00
|
|
|
# define G_interrupt bb_got_signal
|
|
|
|
# define G_ttyin G.ttyin
|
|
|
|
#else
|
|
|
|
# define G_interrupt 0
|
|
|
|
# define G_ttyin 0
|
|
|
|
#endif
|
|
|
|
#if ENABLE_FEATURE_CLEAN_UP
|
|
|
|
# define G_exiting G.exiting
|
|
|
|
#else
|
|
|
|
# define G_exiting 0
|
|
|
|
#endif
|
|
|
|
#define IS_BC (ENABLE_BC && (!ENABLE_DC || applet_name[0] == 'b'))
|
|
|
|
#define IS_DC (ENABLE_DC && (!ENABLE_BC || applet_name[0] != 'b'))
|
|
|
|
|
|
|
|
#if ENABLE_BC
|
|
|
|
# define BC_PARSE_REL (1 << 0)
|
|
|
|
# define BC_PARSE_PRINT (1 << 1)
|
|
|
|
# define BC_PARSE_ARRAY (1 << 2)
|
|
|
|
# define BC_PARSE_NOCALL (1 << 3)
|
|
|
|
#endif
|
|
|
|
|
2018-12-30 15:56:36 +01:00
|
|
|
#define BC_PROG_MAIN 0
|
|
|
|
#define BC_PROG_READ 1
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_DC
|
2018-12-30 15:56:36 +01:00
|
|
|
#define BC_PROG_REQ_FUNCS 2
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-06 18:41:59 +01:00
|
|
|
#define BC_FLAG_W (1 << 0)
|
|
|
|
#define BC_FLAG_V (1 << 1)
|
|
|
|
#define BC_FLAG_S (1 << 2)
|
|
|
|
#define BC_FLAG_Q (1 << 3)
|
|
|
|
#define BC_FLAG_L (1 << 4)
|
2018-12-27 18:23:58 +01:00
|
|
|
#define BC_FLAG_I ((1 << 5) * ENABLE_DC)
|
|
|
|
#define DC_FLAG_X ((1 << 6) * ENABLE_DC)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-07 15:50:14 +01:00
|
|
|
#define BC_MAX_OBASE ((unsigned) 999)
|
|
|
|
#define BC_MAX_DIM ((unsigned) INT_MAX)
|
|
|
|
#define BC_MAX_SCALE ((unsigned) UINT_MAX)
|
|
|
|
#define BC_MAX_STRING ((unsigned) UINT_MAX - 1)
|
|
|
|
#define BC_MAX_NUM BC_MAX_STRING
|
|
|
|
// Unused apart from "limits" message. Just show a "biggish number" there.
|
|
|
|
//#define BC_MAX_EXP ((unsigned long) LONG_MAX)
|
|
|
|
//#define BC_MAX_VARS ((unsigned long) SIZE_MAX - 1)
|
|
|
|
#define BC_MAX_EXP_STR "999999999"
|
|
|
|
#define BC_MAX_VARS_STR "999999999"
|
|
|
|
|
|
|
|
#define BC_MAX_OBASE_STR "999"
|
|
|
|
|
|
|
|
#if INT_MAX == 2147483647
|
|
|
|
# define BC_MAX_DIM_STR "2147483647"
|
|
|
|
#elif INT_MAX == 9223372036854775807
|
|
|
|
# define BC_MAX_DIM_STR "9223372036854775807"
|
|
|
|
#else
|
|
|
|
# error Strange INT_MAX
|
|
|
|
#endif
|
|
|
|
|
2019-09-05 23:40:38 +08:00
|
|
|
#if UINT_MAX == 4294967295U
|
2018-12-07 15:50:14 +01:00
|
|
|
# define BC_MAX_SCALE_STR "4294967295"
|
|
|
|
# define BC_MAX_STRING_STR "4294967294"
|
2019-09-05 23:40:38 +08:00
|
|
|
#elif UINT_MAX == 18446744073709551615U
|
2018-12-07 15:50:14 +01:00
|
|
|
# define BC_MAX_SCALE_STR "18446744073709551615"
|
|
|
|
# define BC_MAX_STRING_STR "18446744073709551614"
|
|
|
|
#else
|
|
|
|
# error Strange UINT_MAX
|
|
|
|
#endif
|
|
|
|
#define BC_MAX_NUM_STR BC_MAX_STRING_STR
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 17:56:09 +01:00
|
|
|
// In configurations where errors abort instead of propagating error
|
|
|
|
// return code up the call chain, functions returning BC_STATUS
|
|
|
|
// actually don't return anything, they always succeed and return "void".
|
|
|
|
// A macro wrapper is provided, which makes this statement work:
|
|
|
|
// s = zbc_func(...)
|
|
|
|
// and makes it visible to the compiler that s is always zero,
|
|
|
|
// allowing compiler to optimize dead code after the statement.
|
|
|
|
//
|
|
|
|
// To make code more readable, each such function has a "z"
|
|
|
|
// ("always returning zero") prefix, i.e. zbc_foo or zdc_foo.
|
|
|
|
//
|
2018-12-27 22:52:13 +01:00
|
|
|
#if ENABLE_FEATURE_BC_INTERACTIVE || ENABLE_FEATURE_CLEAN_UP
|
2018-12-11 19:04:44 +01:00
|
|
|
# define ERRORS_ARE_FATAL 0
|
2018-12-11 17:56:09 +01:00
|
|
|
# define ERRORFUNC /*nothing*/
|
2018-12-17 10:34:02 +01:00
|
|
|
# define IF_ERROR_RETURN_POSSIBLE(a) a
|
|
|
|
# define BC_STATUS BcStatus
|
2018-12-11 17:56:09 +01:00
|
|
|
# define RETURN_STATUS(v) return (v)
|
2018-12-17 10:34:02 +01:00
|
|
|
# define COMMA_SUCCESS /*nothing*/
|
2018-12-11 17:56:09 +01:00
|
|
|
#else
|
2018-12-11 19:04:44 +01:00
|
|
|
# define ERRORS_ARE_FATAL 1
|
2018-12-11 17:56:09 +01:00
|
|
|
# define ERRORFUNC NORETURN
|
2018-12-17 10:34:02 +01:00
|
|
|
# define IF_ERROR_RETURN_POSSIBLE(a) /*nothing*/
|
|
|
|
# define BC_STATUS void
|
2018-12-11 17:56:09 +01:00
|
|
|
# define RETURN_STATUS(v) do { ((void)(v)); return; } while (0)
|
2018-12-17 10:34:02 +01:00
|
|
|
# define COMMA_SUCCESS ,BC_STATUS_SUCCESS
|
2018-12-11 17:56:09 +01:00
|
|
|
#endif
|
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
//
|
|
|
|
// Utility routines
|
|
|
|
//
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-30 15:56:36 +01:00
|
|
|
#define BC_MAX(a, b) ((a) > (b) ? (a) : (b))
|
|
|
|
#define BC_MIN(a, b) ((a) < (b) ? (a) : (b))
|
|
|
|
|
2018-12-03 14:28:51 +01:00
|
|
|
static void fflush_and_check(void)
|
|
|
|
{
|
|
|
|
fflush_all();
|
|
|
|
if (ferror(stdout) || ferror(stderr))
|
libbb: reduce the overhead of single parameter bb_error_msg() calls
Back in 2007, commit 0c97c9d43707 ("'simple' error message functions by
Loic Grenie") introduced bb_simple_perror_msg() to allow for a lower
overhead call to bb_perror_msg() when only a string was being printed
with no parameters. This saves space for some CPU architectures because
it avoids the overhead of a call to a variadic function. However there
has never been a simple version of bb_error_msg(), and since 2007 many
new calls to bb_perror_msg() have been added that only take a single
parameter and so could have been using bb_simple_perror_message().
This changeset introduces 'simple' versions of bb_info_msg(),
bb_error_msg(), bb_error_msg_and_die(), bb_herror_msg() and
bb_herror_msg_and_die(), and replaces all calls that only take a
single parameter, or use something like ("%s", arg), with calls to the
corresponding 'simple' version.
Since it is likely that single parameter calls to the variadic functions
may be accidentally reintroduced in the future a new debugging config
option WARN_SIMPLE_MSG has been introduced. This uses some macro magic
which will cause any such calls to generate a warning, but this is
turned off by default to avoid use of the unpleasant macros in normal
circumstances.
This is a large changeset due to the number of calls that have been
replaced. The only files that contain changes other than simple
substitution of function calls are libbb.h, libbb/herror_msg.c,
libbb/verror_msg.c and libbb/xfuncs_printf.c. In miscutils/devfsd.c,
networking/udhcp/common.h and util-linux/mdev.c additonal macros have
been added for logging so that single parameter and multiple parameter
logging variants exist.
The amount of space saved varies considerably by architecture, and was
found to be as follows (for 'defconfig' using GCC 7.4):
Arm: -92 bytes
MIPS: -52 bytes
PPC: -1836 bytes
x86_64: -938 bytes
Note that for the MIPS architecture only an exception had to be made
disabling the 'simple' calls for 'udhcp' (in networking/udhcp/common.h)
because it made these files larger on MIPS.
Signed-off-by: James Byrne <james.byrne@origamienergy.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2019-07-02 11:35:03 +02:00
|
|
|
bb_simple_perror_msg_and_die("output error");
|
2018-12-03 14:28:51 +01:00
|
|
|
}
|
|
|
|
|
2018-12-06 00:29:22 +01:00
|
|
|
#if ENABLE_FEATURE_CLEAN_UP
|
2018-12-06 23:06:57 +01:00
|
|
|
#define QUIT_OR_RETURN_TO_MAIN \
|
2018-12-06 00:29:22 +01:00
|
|
|
do { \
|
2018-12-27 22:52:13 +01:00
|
|
|
IF_FEATURE_BC_INTERACTIVE(G_ttyin = 0;) /* do not loop in main loop anymore */ \
|
2018-12-06 23:06:57 +01:00
|
|
|
G_exiting = 1; \
|
2018-12-06 00:29:22 +01:00
|
|
|
return BC_STATUS_FAILURE; \
|
|
|
|
} while (0)
|
|
|
|
#else
|
2018-12-03 14:02:35 +01:00
|
|
|
static void quit(void) NORETURN;
|
|
|
|
static void quit(void)
|
|
|
|
{
|
2018-12-03 14:28:51 +01:00
|
|
|
if (ferror(stdin))
|
libbb: reduce the overhead of single parameter bb_error_msg() calls
Back in 2007, commit 0c97c9d43707 ("'simple' error message functions by
Loic Grenie") introduced bb_simple_perror_msg() to allow for a lower
overhead call to bb_perror_msg() when only a string was being printed
with no parameters. This saves space for some CPU architectures because
it avoids the overhead of a call to a variadic function. However there
has never been a simple version of bb_error_msg(), and since 2007 many
new calls to bb_perror_msg() have been added that only take a single
parameter and so could have been using bb_simple_perror_message().
This changeset introduces 'simple' versions of bb_info_msg(),
bb_error_msg(), bb_error_msg_and_die(), bb_herror_msg() and
bb_herror_msg_and_die(), and replaces all calls that only take a
single parameter, or use something like ("%s", arg), with calls to the
corresponding 'simple' version.
Since it is likely that single parameter calls to the variadic functions
may be accidentally reintroduced in the future a new debugging config
option WARN_SIMPLE_MSG has been introduced. This uses some macro magic
which will cause any such calls to generate a warning, but this is
turned off by default to avoid use of the unpleasant macros in normal
circumstances.
This is a large changeset due to the number of calls that have been
replaced. The only files that contain changes other than simple
substitution of function calls are libbb.h, libbb/herror_msg.c,
libbb/verror_msg.c and libbb/xfuncs_printf.c. In miscutils/devfsd.c,
networking/udhcp/common.h and util-linux/mdev.c additonal macros have
been added for logging so that single parameter and multiple parameter
logging variants exist.
The amount of space saved varies considerably by architecture, and was
found to be as follows (for 'defconfig' using GCC 7.4):
Arm: -92 bytes
MIPS: -52 bytes
PPC: -1836 bytes
x86_64: -938 bytes
Note that for the MIPS architecture only an exception had to be made
disabling the 'simple' calls for 'udhcp' (in networking/udhcp/common.h)
because it made these files larger on MIPS.
Signed-off-by: James Byrne <james.byrne@origamienergy.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2019-07-02 11:35:03 +02:00
|
|
|
bb_simple_perror_msg_and_die("input error");
|
2018-12-03 14:28:51 +01:00
|
|
|
fflush_and_check();
|
2018-12-16 16:03:03 +01:00
|
|
|
dbg_exec("quit(): exiting with exitcode SUCCESS");
|
2018-12-03 14:28:51 +01:00
|
|
|
exit(0);
|
2018-12-03 14:02:35 +01:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define QUIT_OR_RETURN_TO_MAIN quit()
|
|
|
|
#endif
|
2018-12-03 14:02:35 +01:00
|
|
|
|
2018-12-05 17:48:01 +01:00
|
|
|
static void bc_verror_msg(const char *fmt, va_list p)
|
|
|
|
{
|
2018-12-14 16:30:56 +01:00
|
|
|
const char *sv = sv; // for compiler
|
2018-12-25 22:32:41 +01:00
|
|
|
if (G.prs.lex_filename) {
|
2018-12-05 17:48:01 +01:00
|
|
|
sv = applet_name;
|
2019-01-02 16:30:24 +01:00
|
|
|
applet_name = xasprintf("%s: %s:%lu", applet_name,
|
|
|
|
G.prs.lex_filename, (unsigned long)G.err_line
|
|
|
|
);
|
2018-12-05 17:48:01 +01:00
|
|
|
}
|
|
|
|
bb_verror_msg(fmt, p, NULL);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (G.prs.lex_filename) {
|
2018-12-05 17:48:01 +01:00
|
|
|
free((char*)applet_name);
|
|
|
|
applet_name = sv;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-10 19:46:53 +01:00
|
|
|
static NOINLINE ERRORFUNC int bc_error_fmt(const char *fmt, ...)
|
2018-12-03 16:06:02 +01:00
|
|
|
{
|
|
|
|
va_list p;
|
|
|
|
|
|
|
|
va_start(p, fmt);
|
2018-12-05 17:48:01 +01:00
|
|
|
bc_verror_msg(fmt, p);
|
2018-12-03 16:06:02 +01:00
|
|
|
va_end(p);
|
2018-12-05 16:39:22 +01:00
|
|
|
|
2018-12-17 10:34:02 +01:00
|
|
|
if (ENABLE_FEATURE_CLEAN_UP || G_ttyin)
|
|
|
|
IF_ERROR_RETURN_POSSIBLE(return BC_STATUS_FAILURE);
|
|
|
|
exit(1);
|
2018-12-03 16:06:02 +01:00
|
|
|
}
|
|
|
|
|
2018-12-06 11:43:17 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-24 18:11:41 +01:00
|
|
|
static NOINLINE BC_STATUS zbc_posix_error_fmt(const char *fmt, ...)
|
2018-12-04 20:51:40 +01:00
|
|
|
{
|
|
|
|
va_list p;
|
|
|
|
|
2018-12-05 16:03:46 +01:00
|
|
|
// Are non-POSIX constructs totally ok?
|
2018-12-04 20:58:40 +01:00
|
|
|
if (!(option_mask32 & (BC_FLAG_S|BC_FLAG_W)))
|
2018-12-24 18:11:41 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS); // yes
|
2018-12-04 20:51:40 +01:00
|
|
|
|
|
|
|
va_start(p, fmt);
|
2018-12-05 17:48:01 +01:00
|
|
|
bc_verror_msg(fmt, p);
|
2018-12-04 20:51:40 +01:00
|
|
|
va_end(p);
|
|
|
|
|
|
|
|
// Do we treat non-POSIX constructs as errors?
|
2018-12-04 20:58:40 +01:00
|
|
|
if (!(option_mask32 & BC_FLAG_S))
|
2018-12-24 18:11:41 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS); // no, it's a warning
|
|
|
|
|
2018-12-17 10:34:02 +01:00
|
|
|
if (ENABLE_FEATURE_CLEAN_UP || G_ttyin)
|
2018-12-24 18:11:41 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_FAILURE);
|
2018-12-17 10:34:02 +01:00
|
|
|
exit(1);
|
2018-12-04 20:51:40 +01:00
|
|
|
}
|
2018-12-24 18:11:41 +01:00
|
|
|
#define zbc_posix_error_fmt(...) (zbc_posix_error_fmt(__VA_ARGS__) COMMA_SUCCESS)
|
2018-12-06 11:43:17 +01:00
|
|
|
#endif
|
2018-12-04 20:51:40 +01:00
|
|
|
|
2018-12-05 16:03:46 +01:00
|
|
|
// We use error functions with "return bc_error(FMT[, PARAMS])" idiom.
|
|
|
|
// This idiom begs for tail-call optimization, but for it to work,
|
2018-12-06 10:29:12 +01:00
|
|
|
// function must not have caller-cleaned parameters on stack.
|
|
|
|
// Unfortunately, vararg function API does exactly that on most arches.
|
|
|
|
// Thus, use these shims for the cases when we have no vararg PARAMS:
|
2018-12-10 19:46:53 +01:00
|
|
|
static ERRORFUNC int bc_error(const char *msg)
|
2018-12-05 16:03:46 +01:00
|
|
|
{
|
2018-12-17 10:34:02 +01:00
|
|
|
IF_ERROR_RETURN_POSSIBLE(return) bc_error_fmt("%s", msg);
|
2018-12-05 16:03:46 +01:00
|
|
|
}
|
2019-01-01 21:50:14 +01:00
|
|
|
static ERRORFUNC int bc_error_at(const char *msg)
|
|
|
|
{
|
|
|
|
const char *err_at = G.prs.lex_next_at;
|
|
|
|
if (err_at) {
|
|
|
|
IF_ERROR_RETURN_POSSIBLE(return) bc_error_fmt(
|
|
|
|
"%s at '%.*s'",
|
|
|
|
msg,
|
|
|
|
(int)(strchrnul(err_at, '\n') - err_at),
|
|
|
|
err_at
|
|
|
|
);
|
|
|
|
}
|
|
|
|
IF_ERROR_RETURN_POSSIBLE(return) bc_error_fmt("%s", msg);
|
|
|
|
}
|
2018-12-10 19:46:53 +01:00
|
|
|
static ERRORFUNC int bc_error_bad_character(char c)
|
2018-12-05 16:03:46 +01:00
|
|
|
{
|
2018-12-17 11:58:20 +01:00
|
|
|
if (!c)
|
|
|
|
IF_ERROR_RETURN_POSSIBLE(return) bc_error("NUL character");
|
2018-12-17 10:34:02 +01:00
|
|
|
IF_ERROR_RETURN_POSSIBLE(return) bc_error_fmt("bad character '%c'", c);
|
2018-12-05 16:03:46 +01:00
|
|
|
}
|
2019-06-08 12:39:30 +02:00
|
|
|
#if ENABLE_BC
|
2019-01-01 21:50:14 +01:00
|
|
|
static ERRORFUNC int bc_error_bad_function_definition(void)
|
|
|
|
{
|
|
|
|
IF_ERROR_RETURN_POSSIBLE(return) bc_error_at("bad function definition");
|
|
|
|
}
|
2019-06-08 12:39:30 +02:00
|
|
|
#endif
|
2018-12-10 19:46:53 +01:00
|
|
|
static ERRORFUNC int bc_error_bad_expression(void)
|
2018-12-05 18:12:27 +01:00
|
|
|
{
|
2019-01-01 21:50:14 +01:00
|
|
|
IF_ERROR_RETURN_POSSIBLE(return) bc_error_at("bad expression");
|
|
|
|
}
|
|
|
|
static ERRORFUNC int bc_error_bad_assignment(void)
|
|
|
|
{
|
|
|
|
IF_ERROR_RETURN_POSSIBLE(return) bc_error_at(
|
|
|
|
"bad assignment: left side must be variable or array element"
|
|
|
|
);
|
2018-12-05 18:12:27 +01:00
|
|
|
}
|
2018-12-10 19:46:53 +01:00
|
|
|
static ERRORFUNC int bc_error_bad_token(void)
|
2018-12-05 18:12:27 +01:00
|
|
|
{
|
2019-01-01 21:50:14 +01:00
|
|
|
IF_ERROR_RETURN_POSSIBLE(return) bc_error_at("bad token");
|
2018-12-05 18:12:27 +01:00
|
|
|
}
|
2018-12-10 19:46:53 +01:00
|
|
|
static ERRORFUNC int bc_error_stack_has_too_few_elements(void)
|
2018-12-05 18:12:27 +01:00
|
|
|
{
|
2018-12-17 10:34:02 +01:00
|
|
|
IF_ERROR_RETURN_POSSIBLE(return) bc_error("stack has too few elements");
|
2018-12-05 18:12:27 +01:00
|
|
|
}
|
2018-12-10 19:46:53 +01:00
|
|
|
static ERRORFUNC int bc_error_variable_is_wrong_type(void)
|
2018-12-05 16:03:46 +01:00
|
|
|
{
|
2018-12-17 10:34:02 +01:00
|
|
|
IF_ERROR_RETURN_POSSIBLE(return) bc_error("variable is wrong type");
|
2018-12-05 16:03:46 +01:00
|
|
|
}
|
2018-12-10 19:46:53 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-24 18:11:41 +01:00
|
|
|
static BC_STATUS zbc_POSIX_requires(const char *msg)
|
2018-12-05 16:03:46 +01:00
|
|
|
{
|
2018-12-24 18:11:41 +01:00
|
|
|
RETURN_STATUS(zbc_posix_error_fmt("POSIX requires %s", msg));
|
2018-12-05 16:03:46 +01:00
|
|
|
}
|
2018-12-24 18:11:41 +01:00
|
|
|
#define zbc_POSIX_requires(...) (zbc_POSIX_requires(__VA_ARGS__) COMMA_SUCCESS)
|
|
|
|
static BC_STATUS zbc_POSIX_does_not_allow(const char *msg)
|
2018-12-05 16:03:46 +01:00
|
|
|
{
|
2018-12-24 18:11:41 +01:00
|
|
|
RETURN_STATUS(zbc_posix_error_fmt("%s%s", "POSIX does not allow ", msg));
|
2018-12-05 16:03:46 +01:00
|
|
|
}
|
2018-12-24 18:11:41 +01:00
|
|
|
#define zbc_POSIX_does_not_allow(...) (zbc_POSIX_does_not_allow(__VA_ARGS__) COMMA_SUCCESS)
|
|
|
|
static BC_STATUS zbc_POSIX_does_not_allow_bool_ops_this_is_bad(const char *msg)
|
2018-12-05 16:03:46 +01:00
|
|
|
{
|
2018-12-25 21:28:25 +01:00
|
|
|
RETURN_STATUS(zbc_posix_error_fmt("%s%s %s", "POSIX does not allow ", "boolean operators; this is bad:", msg));
|
2018-12-05 16:03:46 +01:00
|
|
|
}
|
2018-12-24 18:11:41 +01:00
|
|
|
#define zbc_POSIX_does_not_allow_bool_ops_this_is_bad(...) (zbc_POSIX_does_not_allow_bool_ops_this_is_bad(__VA_ARGS__) COMMA_SUCCESS)
|
|
|
|
static BC_STATUS zbc_POSIX_does_not_allow_empty_X_expression_in_for(const char *msg)
|
2018-12-05 16:03:46 +01:00
|
|
|
{
|
2018-12-25 21:28:25 +01:00
|
|
|
RETURN_STATUS(zbc_posix_error_fmt("%san empty %s expression in 'for()'", "POSIX does not allow ", msg));
|
2018-12-05 16:03:46 +01:00
|
|
|
}
|
2018-12-24 18:11:41 +01:00
|
|
|
#define zbc_POSIX_does_not_allow_empty_X_expression_in_for(...) (zbc_POSIX_does_not_allow_empty_X_expression_in_for(__VA_ARGS__) COMMA_SUCCESS)
|
2018-12-10 19:46:53 +01:00
|
|
|
#endif
|
2018-12-05 16:03:46 +01:00
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
static void bc_vec_grow(BcVec *v, size_t n)
|
|
|
|
{
|
|
|
|
size_t cap = v->cap * 2;
|
|
|
|
while (cap < v->len + n) cap *= 2;
|
|
|
|
v->v = xrealloc(v->v, v->size * cap);
|
|
|
|
v->cap = cap;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_vec_init(BcVec *v, size_t esize, BcVecFree dtor)
|
|
|
|
{
|
|
|
|
v->size = esize;
|
|
|
|
v->cap = BC_VEC_START_CAP;
|
|
|
|
v->len = 0;
|
|
|
|
v->dtor = dtor;
|
|
|
|
v->v = xmalloc(esize * BC_VEC_START_CAP);
|
|
|
|
}
|
|
|
|
|
2018-12-04 21:46:47 +01:00
|
|
|
static void bc_char_vec_init(BcVec *v)
|
|
|
|
{
|
|
|
|
bc_vec_init(v, sizeof(char), NULL);
|
|
|
|
}
|
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
static void bc_vec_expand(BcVec *v, size_t req)
|
|
|
|
{
|
|
|
|
if (v->cap < req) {
|
|
|
|
v->v = xrealloc(v->v, v->size * req);
|
|
|
|
v->cap = req;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-06 13:10:56 +01:00
|
|
|
static void bc_vec_pop(BcVec *v)
|
|
|
|
{
|
|
|
|
v->len--;
|
|
|
|
if (v->dtor)
|
|
|
|
v->dtor(v->v + (v->size * v->len));
|
|
|
|
}
|
2018-12-06 12:34:39 +01:00
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
static void bc_vec_npop(BcVec *v, size_t n)
|
|
|
|
{
|
|
|
|
if (!v->dtor)
|
|
|
|
v->len -= n;
|
|
|
|
else {
|
|
|
|
size_t len = v->len - n;
|
|
|
|
while (v->len > len) v->dtor(v->v + (v->size * --v->len));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-04 21:46:47 +01:00
|
|
|
static void bc_vec_pop_all(BcVec *v)
|
|
|
|
{
|
|
|
|
bc_vec_npop(v, v->len);
|
|
|
|
}
|
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
static size_t bc_vec_npush(BcVec *v, size_t n, const void *data)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-21 22:16:17 +01:00
|
|
|
size_t len = v->len;
|
2019-01-25 14:24:03 +01:00
|
|
|
if (len + n > v->cap) bc_vec_grow(v, n);
|
|
|
|
memmove(v->v + (v->size * len), data, v->size * n);
|
|
|
|
v->len = len + n;
|
2018-12-21 22:16:17 +01:00
|
|
|
return len;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
static size_t bc_vec_push(BcVec *v, const void *data)
|
|
|
|
{
|
|
|
|
return bc_vec_npush(v, 1, data);
|
|
|
|
//size_t len = v->len;
|
|
|
|
//if (len >= v->cap) bc_vec_grow(v, 1);
|
|
|
|
//memmove(v->v + (v->size * len), data, v->size);
|
|
|
|
//v->len = len + 1;
|
|
|
|
//return len;
|
|
|
|
}
|
|
|
|
|
2018-12-21 23:13:48 +01:00
|
|
|
// G.prog.results often needs "pop old operand, push result" idiom.
|
|
|
|
// Can do this without a few extra ops
|
|
|
|
static size_t bc_result_pop_and_push(const void *data)
|
|
|
|
{
|
|
|
|
BcVec *v = &G.prog.results;
|
|
|
|
char *last;
|
|
|
|
size_t len = v->len - 1;
|
|
|
|
|
|
|
|
last = v->v + (v->size * len);
|
|
|
|
if (v->dtor)
|
|
|
|
v->dtor(last);
|
|
|
|
memmove(last, data, v->size);
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2018-12-21 22:16:17 +01:00
|
|
|
static size_t bc_vec_pushByte(BcVec *v, char data)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-21 22:16:17 +01:00
|
|
|
return bc_vec_push(v, &data);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-21 22:16:17 +01:00
|
|
|
static size_t bc_vec_pushZeroByte(BcVec *v)
|
2018-12-05 16:55:08 +01:00
|
|
|
{
|
2018-12-21 22:16:17 +01:00
|
|
|
//return bc_vec_pushByte(v, '\0');
|
2018-12-05 16:55:08 +01:00
|
|
|
// better:
|
2018-12-21 22:16:17 +01:00
|
|
|
return bc_vec_push(v, &const_int_0);
|
2018-12-05 16:55:08 +01:00
|
|
|
}
|
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
static void bc_vec_pushAt(BcVec *v, const void *data, size_t idx)
|
|
|
|
{
|
|
|
|
if (idx == v->len)
|
|
|
|
bc_vec_push(v, data);
|
|
|
|
else {
|
|
|
|
char *ptr;
|
|
|
|
|
|
|
|
if (v->len == v->cap) bc_vec_grow(v, 1);
|
|
|
|
|
|
|
|
ptr = v->v + v->size * idx;
|
|
|
|
|
|
|
|
memmove(ptr + v->size, ptr, v->size * (v->len++ - idx));
|
|
|
|
memmove(ptr, data, v->size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_vec_string(BcVec *v, size_t len, const char *str)
|
|
|
|
{
|
2018-12-04 21:46:47 +01:00
|
|
|
bc_vec_pop_all(v);
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_vec_expand(v, len + 1);
|
|
|
|
memcpy(v->v, str, len);
|
|
|
|
v->len = len;
|
|
|
|
|
2018-12-05 16:55:08 +01:00
|
|
|
bc_vec_pushZeroByte(v);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
static void *bc_vec_item(const BcVec *v, size_t idx)
|
|
|
|
{
|
|
|
|
return v->v + v->size * idx;
|
|
|
|
}
|
|
|
|
|
2018-12-21 16:22:26 +01:00
|
|
|
static void *bc_vec_item_rev(const BcVec *v, size_t idx)
|
|
|
|
{
|
|
|
|
return v->v + v->size * (v->len - idx - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *bc_vec_top(const BcVec *v)
|
2018-12-09 00:03:57 +01:00
|
|
|
{
|
2018-12-21 16:22:26 +01:00
|
|
|
return v->v + v->size * (v->len - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static FAST_FUNC void bc_vec_free(void *vec)
|
|
|
|
{
|
|
|
|
BcVec *v = (BcVec *) vec;
|
|
|
|
bc_vec_pop_all(v);
|
|
|
|
free(v->v);
|
2018-12-09 00:03:57 +01:00
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BcFunc* xc_program_func(size_t idx)
|
2018-12-09 00:03:57 +01:00
|
|
|
{
|
|
|
|
return bc_vec_item(&G.prog.fns, idx);
|
|
|
|
}
|
2018-12-19 15:13:14 +01:00
|
|
|
// BC_PROG_MAIN is zeroth element, so:
|
2018-12-27 18:23:58 +01:00
|
|
|
#define xc_program_func_BC_PROG_MAIN() ((BcFunc*)(G.prog.fns.v))
|
2018-12-09 00:03:57 +01:00
|
|
|
|
2018-12-21 16:22:26 +01:00
|
|
|
#if ENABLE_BC
|
|
|
|
static BcFunc* bc_program_current_func(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-21 16:22:26 +01:00
|
|
|
BcInstPtr *ip = bc_vec_top(&G.prog.exestack);
|
2018-12-27 18:23:58 +01:00
|
|
|
BcFunc *func = xc_program_func(ip->func);
|
2018-12-21 16:22:26 +01:00
|
|
|
return func;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-21 16:22:26 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static char** xc_program_str(size_t idx)
|
2018-12-06 13:10:56 +01:00
|
|
|
{
|
2018-12-21 16:22:26 +01:00
|
|
|
#if ENABLE_BC
|
|
|
|
if (IS_BC) {
|
|
|
|
BcFunc *func = bc_program_current_func();
|
|
|
|
return bc_vec_item(&func->strs, idx);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
IF_DC(return bc_vec_item(&G.prog.strs, idx);)
|
2018-12-06 13:10:56 +01:00
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static char** xc_program_const(size_t idx)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-21 16:22:26 +01:00
|
|
|
#if ENABLE_BC
|
|
|
|
if (IS_BC) {
|
|
|
|
BcFunc *func = bc_program_current_func();
|
|
|
|
return bc_vec_item(&func->consts, idx);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
IF_DC(return bc_vec_item(&G.prog.consts, idx);)
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-05 21:15:46 +01:00
|
|
|
static int bc_id_cmp(const void *e1, const void *e2)
|
|
|
|
{
|
|
|
|
return strcmp(((const BcId *) e1)->name, ((const BcId *) e2)->name);
|
|
|
|
}
|
|
|
|
|
2018-12-10 15:37:14 +01:00
|
|
|
static FAST_FUNC void bc_id_free(void *id)
|
2018-12-05 21:15:46 +01:00
|
|
|
{
|
|
|
|
free(((BcId *) id)->name);
|
|
|
|
}
|
|
|
|
|
2018-12-19 13:55:53 +01:00
|
|
|
static size_t bc_map_find_ge(const BcVec *v, const void *ptr)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
size_t low = 0, high = v->len;
|
|
|
|
|
|
|
|
while (low < high) {
|
|
|
|
size_t mid = (low + high) / 2;
|
|
|
|
BcId *id = bc_vec_item(v, mid);
|
|
|
|
int result = bc_id_cmp(ptr, id);
|
|
|
|
|
|
|
|
if (result == 0)
|
|
|
|
return mid;
|
2018-12-19 13:19:44 +01:00
|
|
|
if (result < 0)
|
2018-11-03 11:00:21 -06:00
|
|
|
high = mid;
|
|
|
|
else
|
|
|
|
low = mid + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return low;
|
|
|
|
}
|
|
|
|
|
2018-12-03 20:35:16 +01:00
|
|
|
static int bc_map_insert(BcVec *v, const void *ptr, size_t *i)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-19 13:55:53 +01:00
|
|
|
size_t n = *i = bc_map_find_ge(v, ptr);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-03 20:35:16 +01:00
|
|
|
if (n == v->len)
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_vec_push(v, ptr);
|
2018-12-03 20:35:16 +01:00
|
|
|
else if (!bc_id_cmp(ptr, bc_vec_item(v, n)))
|
|
|
|
return 0; // "was not inserted"
|
2018-11-03 11:00:21 -06:00
|
|
|
else
|
2018-12-03 20:35:16 +01:00
|
|
|
bc_vec_pushAt(v, ptr, n);
|
|
|
|
return 1; // "was inserted"
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-19 13:55:53 +01:00
|
|
|
static size_t bc_map_find_exact(const BcVec *v, const void *ptr)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-19 13:55:53 +01:00
|
|
|
size_t i = bc_map_find_ge(v, ptr);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (i >= v->len) return BC_VEC_INVALID_IDX;
|
|
|
|
return bc_id_cmp(ptr, bc_vec_item(v, i)) ? BC_VEC_INVALID_IDX : i;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_num_setToZero(BcNum *n, size_t scale)
|
|
|
|
{
|
|
|
|
n->len = 0;
|
|
|
|
n->neg = false;
|
|
|
|
n->rdx = scale;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_num_zero(BcNum *n)
|
|
|
|
{
|
|
|
|
bc_num_setToZero(n, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_num_one(BcNum *n)
|
|
|
|
{
|
|
|
|
bc_num_setToZero(n, 0);
|
|
|
|
n->len = 1;
|
|
|
|
n->num[0] = 1;
|
|
|
|
}
|
|
|
|
|
2018-12-09 12:04:44 +01:00
|
|
|
// Note: this also sets BcNum to zero
|
2018-12-05 21:03:16 +01:00
|
|
|
static void bc_num_init(BcNum *n, size_t req)
|
|
|
|
{
|
|
|
|
req = req >= BC_NUM_DEF_SIZE ? req : BC_NUM_DEF_SIZE;
|
2018-12-09 12:04:44 +01:00
|
|
|
//memset(n, 0, sizeof(BcNum)); - cleared by assignments below
|
2018-12-05 21:03:16 +01:00
|
|
|
n->num = xmalloc(req);
|
|
|
|
n->cap = req;
|
2018-12-09 12:04:44 +01:00
|
|
|
n->rdx = 0;
|
|
|
|
n->len = 0;
|
|
|
|
n->neg = false;
|
2018-12-05 21:03:16 +01:00
|
|
|
}
|
|
|
|
|
2018-12-09 11:44:20 +01:00
|
|
|
static void bc_num_init_DEF_SIZE(BcNum *n)
|
|
|
|
{
|
|
|
|
bc_num_init(n, BC_NUM_DEF_SIZE);
|
|
|
|
}
|
|
|
|
|
2018-12-05 21:03:16 +01:00
|
|
|
static void bc_num_expand(BcNum *n, size_t req)
|
|
|
|
{
|
|
|
|
req = req >= BC_NUM_DEF_SIZE ? req : BC_NUM_DEF_SIZE;
|
|
|
|
if (req > n->cap) {
|
|
|
|
n->num = xrealloc(n->num, req);
|
|
|
|
n->cap = req;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-10 15:37:14 +01:00
|
|
|
static FAST_FUNC void bc_num_free(void *num)
|
2018-12-05 21:03:16 +01:00
|
|
|
{
|
|
|
|
free(((BcNum *) num)->num);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_num_copy(BcNum *d, BcNum *s)
|
|
|
|
{
|
|
|
|
if (d != s) {
|
|
|
|
bc_num_expand(d, s->cap);
|
|
|
|
d->len = s->len;
|
|
|
|
d->neg = s->neg;
|
|
|
|
d->rdx = s->rdx;
|
|
|
|
memcpy(d->num, s->num, sizeof(BcDig) * d->len);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-12 12:35:15 +02:00
|
|
|
static void bc_num_init_and_copy(BcNum *d, BcNum *s)
|
|
|
|
{
|
|
|
|
bc_num_init(d, s->len);
|
|
|
|
bc_num_copy(d, s);
|
|
|
|
}
|
|
|
|
|
2018-12-22 21:52:30 +01:00
|
|
|
static BC_STATUS zbc_num_ulong_abs(BcNum *n, unsigned long *result_p)
|
2018-12-05 21:03:16 +01:00
|
|
|
{
|
|
|
|
size_t i;
|
2018-12-22 21:37:46 +01:00
|
|
|
unsigned long result;
|
2018-12-05 21:03:16 +01:00
|
|
|
|
2018-12-22 21:37:46 +01:00
|
|
|
result = 0;
|
|
|
|
i = n->len;
|
|
|
|
while (i > n->rdx) {
|
|
|
|
unsigned long prev = result;
|
|
|
|
result = result * 10 + n->num[--i];
|
|
|
|
// Even overflowed N*10 can still satisfy N*10>=N. For example,
|
|
|
|
// 0x1ff00000 * 10 is 0x13f600000,
|
|
|
|
// or 0x3f600000 truncated to 32 bits. Which is larger.
|
|
|
|
// However, (N*10)/8 < N check is always correct.
|
|
|
|
if ((result / 8) < prev)
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error("overflow"));
|
2018-12-05 21:03:16 +01:00
|
|
|
}
|
2018-12-07 15:10:05 +01:00
|
|
|
*result_p = result;
|
2018-12-05 21:03:16 +01:00
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-12-05 21:03:16 +01:00
|
|
|
}
|
2018-12-22 21:52:30 +01:00
|
|
|
#define zbc_num_ulong_abs(...) (zbc_num_ulong_abs(__VA_ARGS__) COMMA_SUCCESS)
|
|
|
|
|
|
|
|
static BC_STATUS zbc_num_ulong(BcNum *n, unsigned long *result_p)
|
|
|
|
{
|
|
|
|
if (n->neg) RETURN_STATUS(bc_error("negative number"));
|
|
|
|
|
|
|
|
RETURN_STATUS(zbc_num_ulong_abs(n, result_p));
|
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_num_ulong(...) (zbc_num_ulong(__VA_ARGS__) COMMA_SUCCESS)
|
2018-12-05 21:03:16 +01:00
|
|
|
|
2018-12-21 22:36:04 +01:00
|
|
|
#if ULONG_MAX == 0xffffffffUL // 10 digits: 4294967295
|
|
|
|
# define ULONG_NUM_BUFSIZE (10 > BC_NUM_DEF_SIZE ? 10 : BC_NUM_DEF_SIZE)
|
|
|
|
#elif ULONG_MAX == 0xffffffffffffffffULL // 20 digits: 18446744073709551615
|
|
|
|
# define ULONG_NUM_BUFSIZE (20 > BC_NUM_DEF_SIZE ? 20 : BC_NUM_DEF_SIZE)
|
|
|
|
#endif
|
|
|
|
// minimum BC_NUM_DEF_SIZE, so that bc_num_expand() in bc_num_ulong2num()
|
|
|
|
// would not hit realloc() code path - not good if num[] is not malloced
|
|
|
|
|
2018-12-05 21:03:16 +01:00
|
|
|
static void bc_num_ulong2num(BcNum *n, unsigned long val)
|
|
|
|
{
|
|
|
|
BcDig *ptr;
|
|
|
|
|
|
|
|
bc_num_zero(n);
|
|
|
|
|
|
|
|
if (val == 0) return;
|
|
|
|
|
2018-12-21 22:36:04 +01:00
|
|
|
bc_num_expand(n, ULONG_NUM_BUFSIZE);
|
2018-12-10 12:22:15 +01:00
|
|
|
|
|
|
|
ptr = n->num;
|
|
|
|
for (;;) {
|
|
|
|
n->len++;
|
|
|
|
*ptr++ = val % 10;
|
|
|
|
val /= 10;
|
|
|
|
if (val == 0) break;
|
|
|
|
}
|
2018-12-05 21:03:16 +01:00
|
|
|
}
|
|
|
|
|
2018-12-18 16:24:07 +01:00
|
|
|
static void bc_num_subArrays(BcDig *restrict a, BcDig *restrict b, size_t len)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
size_t i, j;
|
2018-12-02 19:43:34 +01:00
|
|
|
for (i = 0; i < len; ++i) {
|
2018-12-18 12:23:16 +01:00
|
|
|
a[i] -= b[i];
|
|
|
|
for (j = i; a[j] < 0;) {
|
|
|
|
a[j++] += 10;
|
|
|
|
a[j] -= 1;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t bc_num_compare(BcDig *restrict a, BcDig *restrict b, size_t len)
|
|
|
|
{
|
2018-12-18 00:39:24 +01:00
|
|
|
size_t i = len;
|
|
|
|
for (;;) {
|
|
|
|
int c;
|
|
|
|
if (i == 0)
|
|
|
|
return 0;
|
|
|
|
i--;
|
|
|
|
c = a[i] - b[i];
|
|
|
|
if (c != 0) {
|
|
|
|
i++;
|
|
|
|
if (c < 0)
|
|
|
|
return -i;
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
#define BC_NUM_NEG(n, neg) ((((ssize_t)(n)) ^ -((ssize_t)(neg))) + (neg))
|
|
|
|
#define BC_NUM_ONE(n) ((n)->len == 1 && (n)->rdx == 0 && (n)->num[0] == 1)
|
|
|
|
#define BC_NUM_INT(n) ((n)->len - (n)->rdx)
|
|
|
|
//#define BC_NUM_AREQ(a, b) (BC_MAX((a)->rdx, (b)->rdx) + BC_MAX(BC_NUM_INT(a), BC_NUM_INT(b)) + 1)
|
|
|
|
static /*ALWAYS_INLINE*/ size_t BC_NUM_AREQ(BcNum *a, BcNum *b)
|
|
|
|
{
|
|
|
|
return BC_MAX(a->rdx, b->rdx) + BC_MAX(BC_NUM_INT(a), BC_NUM_INT(b)) + 1;
|
|
|
|
}
|
|
|
|
//#define BC_NUM_MREQ(a, b, scale) (BC_NUM_INT(a) + BC_NUM_INT(b) + BC_MAX((scale), (a)->rdx + (b)->rdx) + 1)
|
|
|
|
static /*ALWAYS_INLINE*/ size_t BC_NUM_MREQ(BcNum *a, BcNum *b, size_t scale)
|
|
|
|
{
|
|
|
|
return BC_NUM_INT(a) + BC_NUM_INT(b) + BC_MAX(scale, a->rdx + b->rdx) + 1;
|
|
|
|
}
|
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
static ssize_t bc_num_cmp(BcNum *a, BcNum *b)
|
|
|
|
{
|
|
|
|
size_t i, min, a_int, b_int, diff;
|
|
|
|
BcDig *max_num, *min_num;
|
2018-12-12 11:51:32 +01:00
|
|
|
bool a_max, neg;
|
2018-11-03 11:00:21 -06:00
|
|
|
ssize_t cmp;
|
|
|
|
|
|
|
|
if (a == b) return 0;
|
|
|
|
if (a->len == 0) return BC_NUM_NEG(!!b->len, !b->neg);
|
|
|
|
if (b->len == 0) return BC_NUM_NEG(1, a->neg);
|
2018-12-12 11:51:32 +01:00
|
|
|
|
|
|
|
if (a->neg != b->neg) // signs of a and b differ
|
|
|
|
// +a,-b = a>b = 1 or -a,+b = a<b = -1
|
|
|
|
return (int)b->neg - (int)a->neg;
|
|
|
|
neg = a->neg; // 1 if both negative, 0 if both positive
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
a_int = BC_NUM_INT(a);
|
|
|
|
b_int = BC_NUM_INT(b);
|
|
|
|
a_int -= b_int;
|
|
|
|
|
2019-11-23 17:25:21 +01:00
|
|
|
if (a_int != 0) {
|
|
|
|
if (neg) return - (ssize_t) a_int;
|
|
|
|
return (ssize_t) a_int;
|
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-12 11:51:32 +01:00
|
|
|
a_max = (a->rdx > b->rdx);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (a_max) {
|
|
|
|
min = b->rdx;
|
|
|
|
diff = a->rdx - b->rdx;
|
|
|
|
max_num = a->num + diff;
|
|
|
|
min_num = b->num;
|
2018-12-12 11:51:32 +01:00
|
|
|
// neg = (a_max == neg); - NOP (maps 1->1 and 0->0)
|
|
|
|
} else {
|
2018-11-03 11:00:21 -06:00
|
|
|
min = a->rdx;
|
|
|
|
diff = b->rdx - a->rdx;
|
|
|
|
max_num = b->num + diff;
|
|
|
|
min_num = a->num;
|
2018-12-12 11:51:32 +01:00
|
|
|
neg = !neg; // same as "neg = (a_max == neg)"
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
cmp = bc_num_compare(max_num, min_num, b_int + min);
|
2018-12-12 11:51:32 +01:00
|
|
|
if (cmp != 0) return BC_NUM_NEG(cmp, neg);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 19:43:34 +01:00
|
|
|
for (max_num -= diff, i = diff - 1; i < diff; --i) {
|
2018-12-12 11:51:32 +01:00
|
|
|
if (max_num[i]) return BC_NUM_NEG(1, neg);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_num_truncate(BcNum *n, size_t places)
|
|
|
|
{
|
|
|
|
if (places == 0) return;
|
|
|
|
|
|
|
|
n->rdx -= places;
|
|
|
|
|
|
|
|
if (n->len != 0) {
|
|
|
|
n->len -= places;
|
|
|
|
memmove(n->num, n->num + places, n->len * sizeof(BcDig));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_num_extend(BcNum *n, size_t places)
|
|
|
|
{
|
|
|
|
size_t len = n->len + places;
|
|
|
|
|
|
|
|
if (places != 0) {
|
|
|
|
if (n->cap < len) bc_num_expand(n, len);
|
|
|
|
|
|
|
|
memmove(n->num + places, n->num, sizeof(BcDig) * n->len);
|
|
|
|
memset(n->num, 0, sizeof(BcDig) * places);
|
|
|
|
|
|
|
|
n->len += places;
|
|
|
|
n->rdx += places;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_num_clean(BcNum *n)
|
|
|
|
{
|
|
|
|
while (n->len > 0 && n->num[n->len - 1] == 0) --n->len;
|
|
|
|
if (n->len == 0)
|
|
|
|
n->neg = false;
|
|
|
|
else if (n->len < n->rdx)
|
|
|
|
n->len = n->rdx;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_num_retireMul(BcNum *n, size_t scale, bool neg1, bool neg2)
|
|
|
|
{
|
|
|
|
if (n->rdx < scale)
|
|
|
|
bc_num_extend(n, scale - n->rdx);
|
|
|
|
else
|
|
|
|
bc_num_truncate(n, n->rdx - scale);
|
|
|
|
|
|
|
|
bc_num_clean(n);
|
|
|
|
if (n->len != 0) n->neg = !neg1 != !neg2;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_num_split(BcNum *restrict n, size_t idx, BcNum *restrict a,
|
|
|
|
BcNum *restrict b)
|
|
|
|
{
|
|
|
|
if (idx < n->len) {
|
|
|
|
b->len = n->len - idx;
|
|
|
|
a->len = idx;
|
|
|
|
a->rdx = b->rdx = 0;
|
|
|
|
|
|
|
|
memcpy(b->num, n->num + idx, b->len * sizeof(BcDig));
|
|
|
|
memcpy(a->num, n->num, idx * sizeof(BcDig));
|
2018-12-18 16:24:07 +01:00
|
|
|
} else {
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_zero(b);
|
|
|
|
bc_num_copy(a, n);
|
|
|
|
}
|
|
|
|
|
|
|
|
bc_num_clean(a);
|
|
|
|
bc_num_clean(b);
|
|
|
|
}
|
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
static BC_STATUS zbc_num_shift(BcNum *n, size_t places)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-11 15:29:32 +01:00
|
|
|
if (places == 0 || n->len == 0) RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-12-07 15:50:14 +01:00
|
|
|
|
|
|
|
// This check makes sense only if size_t is (much) larger than BC_MAX_NUM.
|
|
|
|
if (SIZE_MAX > (BC_MAX_NUM | 0xff)) {
|
|
|
|
if (places + n->len > BC_MAX_NUM)
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error("number too long: must be [1,"BC_MAX_NUM_STR"]"));
|
2018-12-07 15:50:14 +01:00
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (n->rdx >= places)
|
|
|
|
n->rdx -= places;
|
|
|
|
else {
|
|
|
|
bc_num_extend(n, places - n->rdx);
|
|
|
|
n->rdx = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bc_num_clean(n);
|
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_num_shift(...) (zbc_num_shift(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
typedef BC_STATUS (*BcNumBinaryOp)(BcNum *, BcNum *, BcNum *, size_t) FAST_FUNC;
|
|
|
|
|
|
|
|
static BC_STATUS zbc_num_binary(BcNum *a, BcNum *b, BcNum *c, size_t scale,
|
|
|
|
BcNumBinaryOp op, size_t req)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-24 05:00:36 +01:00
|
|
|
BcStatus s;
|
|
|
|
BcNum num2, *ptr_a, *ptr_b;
|
|
|
|
bool init = false;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
if (c == a) {
|
|
|
|
ptr_a = &num2;
|
|
|
|
memcpy(ptr_a, c, sizeof(BcNum));
|
|
|
|
init = true;
|
|
|
|
} else
|
|
|
|
ptr_a = a;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
if (c == b) {
|
|
|
|
ptr_b = &num2;
|
|
|
|
if (c != a) {
|
|
|
|
memcpy(ptr_b, c, sizeof(BcNum));
|
|
|
|
init = true;
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
ptr_b = b;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
if (init)
|
|
|
|
bc_num_init(c, req);
|
|
|
|
else
|
|
|
|
bc_num_expand(c, req);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
s = BC_STATUS_SUCCESS;
|
|
|
|
IF_ERROR_RETURN_POSSIBLE(s =) op(ptr_a, ptr_b, c, scale);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
if (init) bc_num_free(&num2);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
RETURN_STATUS(s);
|
|
|
|
}
|
|
|
|
#define zbc_num_binary(...) (zbc_num_binary(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
static FAST_FUNC BC_STATUS zbc_num_a(BcNum *a, BcNum *b, BcNum *restrict c, size_t scale);
|
|
|
|
static FAST_FUNC BC_STATUS zbc_num_s(BcNum *a, BcNum *b, BcNum *restrict c, size_t scale);
|
|
|
|
static FAST_FUNC BC_STATUS zbc_num_p(BcNum *a, BcNum *b, BcNum *restrict c, size_t scale);
|
|
|
|
static FAST_FUNC BC_STATUS zbc_num_m(BcNum *a, BcNum *b, BcNum *restrict c, size_t scale);
|
|
|
|
static FAST_FUNC BC_STATUS zbc_num_d(BcNum *a, BcNum *b, BcNum *restrict c, size_t scale);
|
|
|
|
static FAST_FUNC BC_STATUS zbc_num_rem(BcNum *a, BcNum *b, BcNum *restrict c, size_t scale);
|
|
|
|
|
|
|
|
static FAST_FUNC BC_STATUS zbc_num_add(BcNum *a, BcNum *b, BcNum *c, size_t scale)
|
|
|
|
{
|
|
|
|
BcNumBinaryOp op = (!a->neg == !b->neg) ? zbc_num_a : zbc_num_s;
|
|
|
|
(void) scale;
|
|
|
|
RETURN_STATUS(zbc_num_binary(a, b, c, false, op, BC_NUM_AREQ(a, b)));
|
|
|
|
}
|
|
|
|
|
|
|
|
static FAST_FUNC BC_STATUS zbc_num_sub(BcNum *a, BcNum *b, BcNum *c, size_t scale)
|
|
|
|
{
|
|
|
|
BcNumBinaryOp op = (!a->neg == !b->neg) ? zbc_num_s : zbc_num_a;
|
|
|
|
(void) scale;
|
|
|
|
RETURN_STATUS(zbc_num_binary(a, b, c, true, op, BC_NUM_AREQ(a, b)));
|
|
|
|
}
|
|
|
|
|
|
|
|
static FAST_FUNC BC_STATUS zbc_num_mul(BcNum *a, BcNum *b, BcNum *c, size_t scale)
|
|
|
|
{
|
|
|
|
size_t req = BC_NUM_MREQ(a, b, scale);
|
|
|
|
RETURN_STATUS(zbc_num_binary(a, b, c, scale, zbc_num_m, req));
|
|
|
|
}
|
|
|
|
|
|
|
|
static FAST_FUNC BC_STATUS zbc_num_div(BcNum *a, BcNum *b, BcNum *c, size_t scale)
|
|
|
|
{
|
|
|
|
size_t req = BC_NUM_MREQ(a, b, scale);
|
|
|
|
RETURN_STATUS(zbc_num_binary(a, b, c, scale, zbc_num_d, req));
|
|
|
|
}
|
|
|
|
|
|
|
|
static FAST_FUNC BC_STATUS zbc_num_mod(BcNum *a, BcNum *b, BcNum *c, size_t scale)
|
|
|
|
{
|
|
|
|
size_t req = BC_NUM_MREQ(a, b, scale);
|
|
|
|
RETURN_STATUS(zbc_num_binary(a, b, c, scale, zbc_num_rem, req));
|
|
|
|
}
|
|
|
|
|
|
|
|
static FAST_FUNC BC_STATUS zbc_num_pow(BcNum *a, BcNum *b, BcNum *c, size_t scale)
|
|
|
|
{
|
|
|
|
RETURN_STATUS(zbc_num_binary(a, b, c, scale, zbc_num_p, a->len * b->len + 1));
|
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static const BcNumBinaryOp zxc_program_ops[] = {
|
2018-12-24 05:00:36 +01:00
|
|
|
zbc_num_pow, zbc_num_mul, zbc_num_div, zbc_num_mod, zbc_num_add, zbc_num_sub,
|
|
|
|
};
|
|
|
|
#define zbc_num_add(...) (zbc_num_add(__VA_ARGS__) COMMA_SUCCESS)
|
|
|
|
#define zbc_num_sub(...) (zbc_num_sub(__VA_ARGS__) COMMA_SUCCESS)
|
|
|
|
#define zbc_num_mul(...) (zbc_num_mul(__VA_ARGS__) COMMA_SUCCESS)
|
|
|
|
#define zbc_num_div(...) (zbc_num_div(__VA_ARGS__) COMMA_SUCCESS)
|
|
|
|
#define zbc_num_mod(...) (zbc_num_mod(__VA_ARGS__) COMMA_SUCCESS)
|
|
|
|
#define zbc_num_pow(...) (zbc_num_pow(__VA_ARGS__) COMMA_SUCCESS)
|
|
|
|
|
|
|
|
static BC_STATUS zbc_num_inv(BcNum *a, BcNum *b, size_t scale)
|
|
|
|
{
|
|
|
|
BcNum one;
|
|
|
|
BcDig num[2];
|
|
|
|
|
|
|
|
one.cap = 2;
|
|
|
|
one.num = num;
|
|
|
|
bc_num_one(&one);
|
|
|
|
|
|
|
|
RETURN_STATUS(zbc_num_div(&one, a, b, scale));
|
|
|
|
}
|
|
|
|
#define zbc_num_inv(...) (zbc_num_inv(__VA_ARGS__) COMMA_SUCCESS)
|
|
|
|
|
|
|
|
static FAST_FUNC BC_STATUS zbc_num_a(BcNum *a, BcNum *b, BcNum *restrict c, size_t sub)
|
|
|
|
{
|
|
|
|
BcDig *ptr, *ptr_a, *ptr_b, *ptr_c;
|
|
|
|
size_t i, max, min_rdx, min_int, diff, a_int, b_int;
|
|
|
|
unsigned carry;
|
|
|
|
|
|
|
|
// Because this function doesn't need to use scale (per the bc spec),
|
|
|
|
// I am hijacking it to say whether it's doing an add or a subtract.
|
|
|
|
|
|
|
|
if (a->len == 0) {
|
|
|
|
bc_num_copy(c, b);
|
|
|
|
if (sub && c->len) c->neg = !c->neg;
|
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
|
|
|
}
|
|
|
|
if (b->len == 0) {
|
|
|
|
bc_num_copy(c, a);
|
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
|
|
|
}
|
|
|
|
|
|
|
|
c->neg = a->neg;
|
|
|
|
c->rdx = BC_MAX(a->rdx, b->rdx);
|
|
|
|
min_rdx = BC_MIN(a->rdx, b->rdx);
|
|
|
|
c->len = 0;
|
|
|
|
|
|
|
|
if (a->rdx > b->rdx) {
|
|
|
|
diff = a->rdx - b->rdx;
|
|
|
|
ptr = a->num;
|
|
|
|
ptr_a = a->num + diff;
|
|
|
|
ptr_b = b->num;
|
|
|
|
} else {
|
|
|
|
diff = b->rdx - a->rdx;
|
|
|
|
ptr = b->num;
|
|
|
|
ptr_a = a->num;
|
2018-11-03 11:00:21 -06:00
|
|
|
ptr_b = b->num + diff;
|
|
|
|
}
|
|
|
|
|
2018-12-18 12:23:16 +01:00
|
|
|
ptr_c = c->num;
|
|
|
|
for (i = 0; i < diff; ++i, ++c->len)
|
|
|
|
ptr_c[i] = ptr[i];
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
ptr_c += diff;
|
|
|
|
a_int = BC_NUM_INT(a);
|
|
|
|
b_int = BC_NUM_INT(b);
|
|
|
|
|
|
|
|
if (a_int > b_int) {
|
|
|
|
min_int = b_int;
|
|
|
|
max = a_int;
|
|
|
|
ptr = ptr_a;
|
2018-12-18 12:23:16 +01:00
|
|
|
} else {
|
2018-11-03 11:00:21 -06:00
|
|
|
min_int = a_int;
|
|
|
|
max = b_int;
|
|
|
|
ptr = ptr_b;
|
|
|
|
}
|
|
|
|
|
2018-12-18 12:23:16 +01:00
|
|
|
carry = 0;
|
|
|
|
for (i = 0; i < min_rdx + min_int; ++i) {
|
|
|
|
unsigned in = (unsigned)ptr_a[i] + (unsigned)ptr_b[i] + carry;
|
2018-11-03 11:00:21 -06:00
|
|
|
carry = in / 10;
|
|
|
|
ptr_c[i] = (BcDig)(in % 10);
|
|
|
|
}
|
2018-12-18 12:23:16 +01:00
|
|
|
for (; i < max + min_rdx; ++i) {
|
|
|
|
unsigned in = (unsigned)ptr[i] + carry;
|
2018-11-03 11:00:21 -06:00
|
|
|
carry = in / 10;
|
|
|
|
ptr_c[i] = (BcDig)(in % 10);
|
|
|
|
}
|
2018-12-18 12:23:16 +01:00
|
|
|
c->len += i;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (carry != 0) c->num[c->len++] = (BcDig) carry;
|
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS); // can't make void, see zbc_num_binary()
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
static FAST_FUNC BC_STATUS zbc_num_s(BcNum *a, BcNum *b, BcNum *restrict c, size_t sub)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
ssize_t cmp;
|
|
|
|
BcNum *minuend, *subtrahend;
|
|
|
|
size_t start;
|
|
|
|
bool aneg, bneg, neg;
|
|
|
|
|
|
|
|
// Because this function doesn't need to use scale (per the bc spec),
|
|
|
|
// I am hijacking it to say whether it's doing an add or a subtract.
|
|
|
|
|
|
|
|
if (a->len == 0) {
|
|
|
|
bc_num_copy(c, b);
|
|
|
|
if (sub && c->len) c->neg = !c->neg;
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-11 19:04:44 +01:00
|
|
|
if (b->len == 0) {
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_copy(c, a);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
aneg = a->neg;
|
|
|
|
bneg = b->neg;
|
|
|
|
a->neg = b->neg = false;
|
|
|
|
|
|
|
|
cmp = bc_num_cmp(a, b);
|
|
|
|
|
|
|
|
a->neg = aneg;
|
|
|
|
b->neg = bneg;
|
|
|
|
|
|
|
|
if (cmp == 0) {
|
|
|
|
bc_num_setToZero(c, BC_MAX(a->rdx, b->rdx));
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-11 19:04:44 +01:00
|
|
|
if (cmp > 0) {
|
2018-11-03 11:00:21 -06:00
|
|
|
neg = a->neg;
|
|
|
|
minuend = a;
|
|
|
|
subtrahend = b;
|
2018-12-18 12:23:16 +01:00
|
|
|
} else {
|
2018-11-03 11:00:21 -06:00
|
|
|
neg = b->neg;
|
|
|
|
if (sub) neg = !neg;
|
|
|
|
minuend = b;
|
|
|
|
subtrahend = a;
|
|
|
|
}
|
|
|
|
|
|
|
|
bc_num_copy(c, minuend);
|
|
|
|
c->neg = neg;
|
|
|
|
|
|
|
|
if (c->rdx < subtrahend->rdx) {
|
|
|
|
bc_num_extend(c, subtrahend->rdx - c->rdx);
|
|
|
|
start = 0;
|
2018-12-18 12:23:16 +01:00
|
|
|
} else
|
2018-11-03 11:00:21 -06:00
|
|
|
start = c->rdx - subtrahend->rdx;
|
|
|
|
|
2018-12-02 19:43:34 +01:00
|
|
|
bc_num_subArrays(c->num + start, subtrahend->num, subtrahend->len);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
bc_num_clean(c);
|
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS); // can't make void, see zbc_num_binary()
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
static FAST_FUNC BC_STATUS zbc_num_k(BcNum *restrict a, BcNum *restrict b,
|
2018-11-03 11:00:21 -06:00
|
|
|
BcNum *restrict c)
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_num_k(...) (zbc_num_k(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
2021-01-04 14:41:20 +01:00
|
|
|
size_t max, max2;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcNum l1, h1, l2, h2, m2, m1, z0, z1, z2, temp;
|
2018-12-05 18:56:14 +01:00
|
|
|
bool aone;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (a->len == 0 || b->len == 0) {
|
|
|
|
bc_num_zero(c);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-05 18:56:14 +01:00
|
|
|
aone = BC_NUM_ONE(a);
|
|
|
|
if (aone || BC_NUM_ONE(b)) {
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_copy(c, aone ? b : a);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2021-01-06 14:00:53 +01:00
|
|
|
if (a->len < BC_NUM_KARATSUBA_LEN
|
2018-12-18 12:23:16 +01:00
|
|
|
|| b->len < BC_NUM_KARATSUBA_LEN
|
2021-01-06 14:00:53 +01:00
|
|
|
/* || a->len + b->len < BC_NUM_KARATSUBA_LEN - redundant check */
|
2018-12-18 12:23:16 +01:00
|
|
|
) {
|
2018-12-05 18:56:14 +01:00
|
|
|
size_t i, j, len;
|
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_expand(c, a->len + b->len + 1);
|
|
|
|
|
|
|
|
memset(c->num, 0, sizeof(BcDig) * c->cap);
|
2018-12-05 18:56:14 +01:00
|
|
|
c->len = len = 0;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 19:43:34 +01:00
|
|
|
for (i = 0; i < b->len; ++i) {
|
2018-12-18 12:23:16 +01:00
|
|
|
unsigned carry = 0;
|
2018-12-02 19:43:34 +01:00
|
|
|
for (j = 0; j < a->len; ++j) {
|
2018-12-05 19:05:32 +01:00
|
|
|
unsigned in = c->num[i + j];
|
2018-12-18 12:23:16 +01:00
|
|
|
in += (unsigned)a->num[j] * (unsigned)b->num[i] + carry;
|
2018-12-05 19:05:32 +01:00
|
|
|
// note: compilers prefer _unsigned_ div/const
|
2018-11-03 11:00:21 -06:00
|
|
|
carry = in / 10;
|
|
|
|
c->num[i + j] = (BcDig)(in % 10);
|
|
|
|
}
|
|
|
|
|
|
|
|
c->num[i + j] += (BcDig) carry;
|
|
|
|
len = BC_MAX(len, i + j + !!carry);
|
2018-12-05 19:00:58 +01:00
|
|
|
|
2018-12-27 22:52:13 +01:00
|
|
|
#if ENABLE_FEATURE_BC_INTERACTIVE
|
2018-12-05 19:00:58 +01:00
|
|
|
// a=2^1000000
|
|
|
|
// a*a <- without check below, this will not be interruptible
|
|
|
|
if (G_interrupt) return BC_STATUS_FAILURE;
|
2018-12-11 19:04:44 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
c->len = len;
|
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2021-01-04 14:41:20 +01:00
|
|
|
max = BC_MAX(a->len, b->len);
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_init(&l1, max);
|
|
|
|
bc_num_init(&h1, max);
|
|
|
|
bc_num_init(&l2, max);
|
|
|
|
bc_num_init(&h2, max);
|
|
|
|
bc_num_init(&m1, max);
|
|
|
|
bc_num_init(&m2, max);
|
|
|
|
bc_num_init(&z0, max);
|
|
|
|
bc_num_init(&z1, max);
|
|
|
|
bc_num_init(&z2, max);
|
|
|
|
bc_num_init(&temp, max + max);
|
|
|
|
|
2021-01-04 14:41:20 +01:00
|
|
|
max2 = (max + 1) / 2;
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_split(a, max2, &l1, &h1);
|
|
|
|
bc_num_split(b, max2, &l2, &h2);
|
|
|
|
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_add(&h1, &l1, &m1, 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_add(&h2, &l2, &m2, 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_k(&h1, &h2, &z0);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_k(&m1, &m2, &z1);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_k(&l1, &l2, &z2);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_sub(&z1, &z0, &temp, 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_sub(&temp, &z2, &z1, 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
s = zbc_num_shift(&z0, max2 * 2);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-11 15:29:32 +01:00
|
|
|
s = zbc_num_shift(&z1, max2);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_add(&z0, &z1, &temp, 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_add(&temp, &z2, c, 0);
|
2018-12-18 16:24:07 +01:00
|
|
|
err:
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_free(&temp);
|
|
|
|
bc_num_free(&z2);
|
|
|
|
bc_num_free(&z1);
|
|
|
|
bc_num_free(&z0);
|
|
|
|
bc_num_free(&m2);
|
|
|
|
bc_num_free(&m1);
|
|
|
|
bc_num_free(&h2);
|
|
|
|
bc_num_free(&l2);
|
|
|
|
bc_num_free(&h1);
|
|
|
|
bc_num_free(&l1);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
static FAST_FUNC BC_STATUS zbc_num_m(BcNum *a, BcNum *b, BcNum *restrict c, size_t scale)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcNum cpa, cpb;
|
|
|
|
size_t maxrdx = BC_MAX(a->rdx, b->rdx);
|
|
|
|
|
|
|
|
scale = BC_MAX(scale, a->rdx);
|
|
|
|
scale = BC_MAX(scale, b->rdx);
|
|
|
|
scale = BC_MIN(a->rdx + b->rdx, scale);
|
|
|
|
maxrdx = BC_MAX(maxrdx, scale);
|
|
|
|
|
2021-06-12 12:35:15 +02:00
|
|
|
bc_num_init_and_copy(&cpa, a);
|
|
|
|
bc_num_init_and_copy(&cpb, b);
|
2018-11-03 11:00:21 -06:00
|
|
|
cpa.neg = cpb.neg = false;
|
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
s = zbc_num_shift(&cpa, maxrdx);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-11 15:29:32 +01:00
|
|
|
s = zbc_num_shift(&cpb, maxrdx);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_k(&cpa, &cpb, c);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
|
|
|
maxrdx += scale;
|
|
|
|
bc_num_expand(c, c->len + maxrdx);
|
|
|
|
|
|
|
|
if (c->len < maxrdx) {
|
|
|
|
memset(c->num + c->len, 0, (c->cap - c->len) * sizeof(BcDig));
|
|
|
|
c->len += maxrdx;
|
|
|
|
}
|
|
|
|
|
|
|
|
c->rdx = maxrdx;
|
|
|
|
bc_num_retireMul(c, scale, a->neg, b->neg);
|
2018-12-18 16:24:07 +01:00
|
|
|
err:
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_free(&cpb);
|
|
|
|
bc_num_free(&cpa);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_num_m(...) (zbc_num_m(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
static FAST_FUNC BC_STATUS zbc_num_d(BcNum *a, BcNum *b, BcNum *restrict c, size_t scale)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-18 13:15:55 +01:00
|
|
|
BcStatus s;
|
2018-11-03 11:00:21 -06:00
|
|
|
size_t len, end, i;
|
|
|
|
BcNum cp;
|
|
|
|
|
|
|
|
if (b->len == 0)
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(bc_error("divide by zero"));
|
|
|
|
if (a->len == 0) {
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_setToZero(c, scale);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-11 19:04:44 +01:00
|
|
|
if (BC_NUM_ONE(b)) {
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_copy(c, a);
|
|
|
|
bc_num_retireMul(c, scale, a->neg, b->neg);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
bc_num_init(&cp, BC_NUM_MREQ(a, b, scale));
|
|
|
|
bc_num_copy(&cp, a);
|
|
|
|
len = b->len;
|
|
|
|
|
|
|
|
if (len > cp.len) {
|
|
|
|
bc_num_expand(&cp, len + 2);
|
|
|
|
bc_num_extend(&cp, len - cp.len);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (b->rdx > cp.rdx) bc_num_extend(&cp, b->rdx - cp.rdx);
|
|
|
|
cp.rdx -= b->rdx;
|
|
|
|
if (scale > cp.rdx) bc_num_extend(&cp, scale - cp.rdx);
|
|
|
|
|
|
|
|
if (b->rdx == b->len) {
|
2018-12-18 12:43:21 +01:00
|
|
|
for (;;) {
|
|
|
|
if (len == 0) break;
|
|
|
|
len--;
|
|
|
|
if (b->num[len] != 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
len++;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (cp.cap == cp.len) bc_num_expand(&cp, cp.len + 1);
|
|
|
|
|
|
|
|
// We want an extra zero in front to make things simpler.
|
|
|
|
cp.num[cp.len++] = 0;
|
|
|
|
end = cp.len - len;
|
|
|
|
|
|
|
|
bc_num_expand(c, cp.len);
|
|
|
|
|
|
|
|
bc_num_zero(c);
|
|
|
|
memset(c->num + end, 0, (c->cap - end) * sizeof(BcDig));
|
|
|
|
c->rdx = cp.rdx;
|
|
|
|
c->len = cp.len;
|
|
|
|
|
2018-12-18 13:15:55 +01:00
|
|
|
s = BC_STATUS_SUCCESS;
|
|
|
|
for (i = end - 1; i < end; --i) {
|
|
|
|
BcDig *n, q;
|
2018-11-03 11:00:21 -06:00
|
|
|
n = cp.num + i;
|
2018-12-18 13:15:55 +01:00
|
|
|
for (q = 0; n[len] != 0 || bc_num_compare(n, b->num, len) >= 0; ++q)
|
|
|
|
bc_num_subArrays(n, b->num, len);
|
2018-11-03 11:00:21 -06:00
|
|
|
c->num[i] = q;
|
2018-12-27 22:52:13 +01:00
|
|
|
#if ENABLE_FEATURE_BC_INTERACTIVE
|
2018-12-05 19:21:34 +01:00
|
|
|
// a=2^100000
|
|
|
|
// scale=40000
|
|
|
|
// 1/a <- without check below, this will not be interruptible
|
|
|
|
if (G_interrupt) {
|
|
|
|
s = BC_STATUS_FAILURE;
|
|
|
|
break;
|
|
|
|
}
|
2018-12-11 19:04:44 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-02 19:43:34 +01:00
|
|
|
bc_num_retireMul(c, scale, a->neg, b->neg);
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_free(&cp);
|
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_num_d(...) (zbc_num_d(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
static FAST_FUNC BC_STATUS zbc_num_r(BcNum *a, BcNum *b, BcNum *restrict c,
|
2018-11-03 11:00:21 -06:00
|
|
|
BcNum *restrict d, size_t scale, size_t ts)
|
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcNum temp;
|
|
|
|
bool neg;
|
|
|
|
|
2018-12-04 20:05:28 +01:00
|
|
|
if (b->len == 0)
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(bc_error("divide by zero"));
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (a->len == 0) {
|
|
|
|
bc_num_setToZero(d, ts);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
bc_num_init(&temp, d->cap);
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_d(a, b, c, scale);
|
2018-12-05 19:21:34 +01:00
|
|
|
if (s) goto err;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (scale != 0) scale = ts;
|
|
|
|
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_m(c, b, &temp, scale);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_sub(a, &temp, d, scale);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
|
|
|
if (ts > d->rdx && d->len) bc_num_extend(d, ts - d->rdx);
|
|
|
|
|
|
|
|
neg = d->neg;
|
|
|
|
bc_num_retireMul(d, ts, a->neg, b->neg);
|
|
|
|
d->neg = neg;
|
2018-12-18 16:24:07 +01:00
|
|
|
err:
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_free(&temp);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_num_r(...) (zbc_num_r(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
static FAST_FUNC BC_STATUS zbc_num_rem(BcNum *a, BcNum *b, BcNum *restrict c, size_t scale)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcNum c1;
|
|
|
|
size_t ts = BC_MAX(scale + b->rdx, a->rdx), len = BC_NUM_MREQ(a, b, ts);
|
|
|
|
|
|
|
|
bc_num_init(&c1, len);
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_r(a, b, &c1, c, scale, ts);
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_free(&c1);
|
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_num_rem(...) (zbc_num_rem(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
static FAST_FUNC BC_STATUS zbc_num_p(BcNum *a, BcNum *b, BcNum *restrict c, size_t scale)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s = BC_STATUS_SUCCESS;
|
|
|
|
BcNum copy;
|
|
|
|
unsigned long pow;
|
|
|
|
size_t i, powrdx, resrdx;
|
2021-06-12 12:19:20 +02:00
|
|
|
size_t a_rdx;
|
2018-12-18 12:55:40 +01:00
|
|
|
bool neg;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-22 23:14:48 +01:00
|
|
|
// GNU bc does not allow 2^2.0 - we do
|
|
|
|
for (i = 0; i < b->rdx; i++)
|
|
|
|
if (b->num[i] != 0)
|
|
|
|
RETURN_STATUS(bc_error("not an integer"));
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2021-06-12 14:03:24 +02:00
|
|
|
// a^b for non-integer b (for a>0) can be implemented as exp(ln(a)*b).
|
|
|
|
// Possibly better precision would be given by a^int(b) * exp(ln(a)*frac(b)).
|
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
if (b->len == 0) {
|
|
|
|
bc_num_one(c);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-11 19:04:44 +01:00
|
|
|
if (a->len == 0) {
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_setToZero(c, scale);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-11 19:04:44 +01:00
|
|
|
if (BC_NUM_ONE(b)) {
|
2018-11-03 11:00:21 -06:00
|
|
|
if (!b->neg)
|
|
|
|
bc_num_copy(c, a);
|
|
|
|
else
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_inv(a, c, scale);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
neg = b->neg;
|
2018-12-22 21:52:30 +01:00
|
|
|
s = zbc_num_ulong_abs(b, &pow);
|
2018-12-11 19:04:44 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-22 21:45:18 +01:00
|
|
|
// b is not used beyond this point
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2021-06-12 12:35:15 +02:00
|
|
|
bc_num_init_and_copy(©, a);
|
2021-06-12 12:19:20 +02:00
|
|
|
a_rdx = a->rdx; // pull it into a CPU register (hopefully)
|
|
|
|
// a is not used beyond this point
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-07 16:22:45 +01:00
|
|
|
if (!neg) {
|
2021-06-12 12:19:20 +02:00
|
|
|
unsigned long new_scale;
|
|
|
|
if (a_rdx > scale)
|
|
|
|
scale = a_rdx;
|
|
|
|
new_scale = a_rdx * pow;
|
|
|
|
// Don't fall for multiplication overflow. Example:
|
|
|
|
// 0.01^2147483648 a_rdx:2 pow:0x80000000, 32bit mul is 0.
|
|
|
|
//not that it matters with current algorithm, it would OOM on such large powers,
|
|
|
|
//but it can be improved to detect zero results etc. Example: with scale=0,
|
|
|
|
//result of 0.01^N for any N>1 is 0: 0.01^2 = 0.0001 ~= 0.00 (trunc to scale)
|
|
|
|
//then this would matter:
|
2021-06-12 14:03:24 +02:00
|
|
|
// if a_rdx != 0 and new_scale < pow, we had overflow,
|
|
|
|
// correct "new_scale" value is larger than ULONG_MAX,
|
|
|
|
// thus larger than any possible current value of "scale",
|
2021-06-12 12:19:20 +02:00
|
|
|
// thus "scale = new_scale" should not be done:
|
2021-06-12 14:03:24 +02:00
|
|
|
if (a_rdx == 0 || new_scale >= pow)
|
2021-06-12 12:19:20 +02:00
|
|
|
if (new_scale < scale)
|
|
|
|
scale = new_scale;
|
2018-12-07 16:22:45 +01:00
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2021-06-12 12:19:20 +02:00
|
|
|
for (powrdx = a_rdx; !(pow & 1); pow >>= 1) {
|
2018-11-03 11:00:21 -06:00
|
|
|
powrdx <<= 1;
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_mul(©, ©, ©, powrdx);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-11 19:12:13 +01:00
|
|
|
// Not needed: zbc_num_mul() has a check for ^C:
|
2018-12-05 19:00:58 +01:00
|
|
|
//if (G_interrupt) {
|
|
|
|
// s = BC_STATUS_FAILURE;
|
|
|
|
// goto err;
|
|
|
|
//}
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
bc_num_copy(c, ©);
|
|
|
|
|
2018-12-02 19:43:34 +01:00
|
|
|
for (resrdx = powrdx, pow >>= 1; pow != 0; pow >>= 1) {
|
2018-11-03 11:00:21 -06:00
|
|
|
powrdx <<= 1;
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_mul(©, ©, ©, powrdx);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
|
|
|
if (pow & 1) {
|
|
|
|
resrdx += powrdx;
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_mul(c, ©, c, resrdx);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
}
|
2018-12-11 19:12:13 +01:00
|
|
|
// Not needed: zbc_num_mul() has a check for ^C:
|
2018-12-05 19:00:58 +01:00
|
|
|
//if (G_interrupt) {
|
|
|
|
// s = BC_STATUS_FAILURE;
|
|
|
|
// goto err;
|
|
|
|
//}
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (neg) {
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_inv(c, c, scale);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (c->rdx > scale) bc_num_truncate(c, c->rdx - scale);
|
|
|
|
|
|
|
|
// We can't use bc_num_clean() here.
|
2018-12-18 12:55:40 +01:00
|
|
|
for (i = 0; i < c->len; ++i)
|
|
|
|
if (c->num[i] != 0)
|
|
|
|
goto skip;
|
|
|
|
bc_num_setToZero(c, scale);
|
|
|
|
skip:
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-18 12:55:40 +01:00
|
|
|
err:
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_free(©);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_num_p(...) (zbc_num_p(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2021-10-10 13:50:53 +02:00
|
|
|
static NOINLINE BC_STATUS zbc_num_sqrt(BcNum *a, BcNum *restrict b, size_t scale)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
2018-12-24 05:00:36 +01:00
|
|
|
BcNum num1, num2, half, f, fprime, *x0, *x1, *temp;
|
2018-12-29 16:23:34 +01:00
|
|
|
BcDig half_digs[1];
|
2020-12-29 16:54:37 +01:00
|
|
|
size_t pow, len, digs, digs1, resrdx, req, times;
|
|
|
|
ssize_t cmp, cmp1, cmp2;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
req = BC_MAX(scale, a->rdx) + ((BC_NUM_INT(a) + 1) >> 1) + 1;
|
|
|
|
bc_num_expand(b, req);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
if (a->len == 0) {
|
|
|
|
bc_num_setToZero(b, scale);
|
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
|
|
|
}
|
|
|
|
if (a->neg) {
|
|
|
|
RETURN_STATUS(bc_error("negative number"));
|
|
|
|
}
|
|
|
|
if (BC_NUM_ONE(a)) {
|
|
|
|
bc_num_one(b);
|
|
|
|
bc_num_extend(b, scale);
|
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
scale = BC_MAX(scale, a->rdx) + 1;
|
|
|
|
len = a->len + scale;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
bc_num_init(&num1, len);
|
|
|
|
bc_num_init(&num2, len);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-29 16:23:34 +01:00
|
|
|
half.cap = ARRAY_SIZE(half_digs);
|
|
|
|
half.num = half_digs;
|
2018-12-24 05:00:36 +01:00
|
|
|
bc_num_one(&half);
|
2018-12-29 16:23:34 +01:00
|
|
|
half_digs[0] = 5;
|
2018-12-24 05:00:36 +01:00
|
|
|
half.rdx = 1;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
bc_num_init(&f, len);
|
|
|
|
bc_num_init(&fprime, len);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
x0 = &num1;
|
|
|
|
x1 = &num2;
|
2018-12-10 11:54:18 +01:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
bc_num_one(x0);
|
|
|
|
pow = BC_NUM_INT(a);
|
|
|
|
|
|
|
|
if (pow) {
|
|
|
|
if (pow & 1)
|
|
|
|
x0->num[0] = 2;
|
|
|
|
else
|
|
|
|
x0->num[0] = 6;
|
|
|
|
|
|
|
|
pow -= 2 - (pow & 1);
|
|
|
|
|
|
|
|
bc_num_extend(x0, pow);
|
|
|
|
|
|
|
|
// Make sure to move the radix back.
|
|
|
|
x0->rdx -= pow;
|
2018-12-10 11:54:18 +01:00
|
|
|
}
|
|
|
|
|
2020-12-29 16:54:37 +01:00
|
|
|
x0->rdx = digs = digs1 = times = 0;
|
2018-12-24 05:00:36 +01:00
|
|
|
resrdx = scale + 2;
|
2020-12-29 16:54:37 +01:00
|
|
|
len = x0->len + resrdx - 1;
|
|
|
|
cmp = 1;
|
|
|
|
cmp1 = cmp2 = SSIZE_MAX;
|
|
|
|
do {
|
2018-12-24 05:00:36 +01:00
|
|
|
s = zbc_num_div(a, x0, &f, resrdx);
|
|
|
|
if (s) goto err;
|
|
|
|
s = zbc_num_add(x0, &f, &fprime, resrdx);
|
|
|
|
if (s) goto err;
|
|
|
|
s = zbc_num_mul(&fprime, &half, x1, resrdx);
|
|
|
|
if (s) goto err;
|
2018-12-10 11:54:18 +01:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
cmp = bc_num_cmp(x1, x0);
|
|
|
|
digs = x1->len - (unsigned long long) llabs(cmp);
|
2018-12-10 11:54:18 +01:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
if (cmp == cmp2 && digs == digs1)
|
|
|
|
times += 1;
|
|
|
|
else
|
|
|
|
times = 0;
|
2018-12-10 11:54:18 +01:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
resrdx += times > 4;
|
2018-12-10 11:54:18 +01:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
cmp2 = cmp1;
|
|
|
|
cmp1 = cmp;
|
|
|
|
digs1 = digs;
|
|
|
|
|
|
|
|
temp = x0;
|
|
|
|
x0 = x1;
|
|
|
|
x1 = temp;
|
2020-12-29 16:54:37 +01:00
|
|
|
} while (cmp != 0 || digs < len);
|
2018-12-24 05:00:36 +01:00
|
|
|
|
|
|
|
bc_num_copy(b, x0);
|
|
|
|
scale -= 1;
|
2020-12-29 16:54:37 +01:00
|
|
|
if (b->rdx > scale)
|
|
|
|
bc_num_truncate(b, b->rdx - scale);
|
2018-12-24 05:00:36 +01:00
|
|
|
err:
|
|
|
|
bc_num_free(&fprime);
|
|
|
|
bc_num_free(&f);
|
|
|
|
bc_num_free(&num2);
|
|
|
|
bc_num_free(&num1);
|
|
|
|
RETURN_STATUS(s);
|
2018-12-10 11:54:18 +01:00
|
|
|
}
|
2018-12-24 05:00:36 +01:00
|
|
|
#define zbc_num_sqrt(...) (zbc_num_sqrt(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
static BC_STATUS zbc_num_divmod(BcNum *a, BcNum *b, BcNum *c, BcNum *d,
|
2018-11-03 11:00:21 -06:00
|
|
|
size_t scale)
|
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcNum num2, *ptr_a;
|
|
|
|
bool init = false;
|
|
|
|
size_t ts = BC_MAX(scale + b->rdx, a->rdx), len = BC_NUM_MREQ(a, b, ts);
|
|
|
|
|
|
|
|
if (c == a) {
|
|
|
|
memcpy(&num2, c, sizeof(BcNum));
|
|
|
|
ptr_a = &num2;
|
|
|
|
bc_num_init(c, len);
|
|
|
|
init = true;
|
2018-12-18 12:23:16 +01:00
|
|
|
} else {
|
2018-11-03 11:00:21 -06:00
|
|
|
ptr_a = a;
|
|
|
|
bc_num_expand(c, len);
|
|
|
|
}
|
|
|
|
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_r(ptr_a, b, c, d, scale, ts);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (init) bc_num_free(&num2);
|
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_num_divmod(...) (zbc_num_divmod(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_DC
|
2018-12-19 19:35:40 +01:00
|
|
|
static BC_STATUS zdc_num_modexp(BcNum *a, BcNum *b, BcNum *c, BcNum *restrict d)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcNum base, exp, two, temp;
|
2018-12-29 16:23:34 +01:00
|
|
|
BcDig two_digs[1];
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-04 20:05:28 +01:00
|
|
|
if (c->len == 0)
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(bc_error("divide by zero"));
|
2018-12-04 20:05:28 +01:00
|
|
|
if (a->rdx || b->rdx || c->rdx)
|
2018-12-22 23:14:48 +01:00
|
|
|
RETURN_STATUS(bc_error("not an integer"));
|
2018-12-04 20:05:28 +01:00
|
|
|
if (b->neg)
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(bc_error("negative number"));
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
bc_num_expand(d, c->len);
|
|
|
|
bc_num_init(&base, c->len);
|
|
|
|
bc_num_init(&exp, b->len);
|
|
|
|
bc_num_init(&temp, b->len);
|
|
|
|
|
2018-12-22 23:59:21 +01:00
|
|
|
two.cap = ARRAY_SIZE(two_digs);
|
|
|
|
two.num = two_digs;
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_one(&two);
|
2018-12-22 23:59:21 +01:00
|
|
|
two_digs[0] = 2;
|
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_one(d);
|
|
|
|
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_rem(a, c, &base, 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
bc_num_copy(&exp, b);
|
|
|
|
|
|
|
|
while (exp.len != 0) {
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_divmod(&exp, &two, &exp, &temp, 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
|
|
|
if (BC_NUM_ONE(&temp)) {
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_mul(d, &base, &temp, 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_rem(&temp, c, d, 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
}
|
|
|
|
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_mul(&base, &base, &temp, 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_rem(&temp, c, &base, 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
}
|
2018-12-18 12:23:16 +01:00
|
|
|
err:
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_free(&temp);
|
|
|
|
bc_num_free(&exp);
|
|
|
|
bc_num_free(&base);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-24 05:00:36 +01:00
|
|
|
#define zdc_num_modexp(...) (zdc_num_modexp(__VA_ARGS__) COMMA_SUCCESS)
|
|
|
|
#endif // ENABLE_DC
|
|
|
|
|
|
|
|
static FAST_FUNC void bc_string_free(void *string)
|
|
|
|
{
|
|
|
|
free(*(char**)string);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_func_init(BcFunc *f)
|
|
|
|
{
|
|
|
|
bc_char_vec_init(&f->code);
|
|
|
|
IF_BC(bc_vec_init(&f->labels, sizeof(size_t), NULL);)
|
|
|
|
IF_BC(bc_vec_init(&f->autos, sizeof(BcId), bc_id_free);)
|
|
|
|
IF_BC(bc_vec_init(&f->strs, sizeof(char *), bc_string_free);)
|
|
|
|
IF_BC(bc_vec_init(&f->consts, sizeof(char *), bc_string_free);)
|
|
|
|
IF_BC(f->nparams = 0;)
|
|
|
|
}
|
|
|
|
|
|
|
|
static FAST_FUNC void bc_func_free(void *func)
|
|
|
|
{
|
|
|
|
BcFunc *f = (BcFunc *) func;
|
|
|
|
bc_vec_free(&f->code);
|
|
|
|
IF_BC(bc_vec_free(&f->labels);)
|
|
|
|
IF_BC(bc_vec_free(&f->autos);)
|
|
|
|
IF_BC(bc_vec_free(&f->strs);)
|
|
|
|
IF_BC(bc_vec_free(&f->consts);)
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_array_expand(BcVec *a, size_t len);
|
|
|
|
|
|
|
|
static void bc_array_init(BcVec *a, bool nums)
|
|
|
|
{
|
|
|
|
if (nums)
|
|
|
|
bc_vec_init(a, sizeof(BcNum), bc_num_free);
|
|
|
|
else
|
|
|
|
bc_vec_init(a, sizeof(BcVec), bc_vec_free);
|
|
|
|
bc_array_expand(a, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_array_expand(BcVec *a, size_t len)
|
|
|
|
{
|
|
|
|
if (a->dtor == bc_num_free
|
|
|
|
// && a->size == sizeof(BcNum) - always true
|
|
|
|
) {
|
|
|
|
BcNum n;
|
|
|
|
while (len > a->len) {
|
|
|
|
bc_num_init_DEF_SIZE(&n);
|
|
|
|
bc_vec_push(a, &n);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
BcVec v;
|
|
|
|
while (len > a->len) {
|
|
|
|
bc_array_init(&v, true);
|
|
|
|
bc_vec_push(a, &v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_array_copy(BcVec *d, const BcVec *s)
|
|
|
|
{
|
|
|
|
BcNum *dnum, *snum;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
bc_vec_pop_all(d);
|
|
|
|
bc_vec_expand(d, s->cap);
|
|
|
|
d->len = s->len;
|
|
|
|
|
|
|
|
dnum = (void*)d->v;
|
|
|
|
snum = (void*)s->v;
|
|
|
|
for (i = 0; i < s->len; i++, dnum++, snum++) {
|
2021-06-12 12:35:15 +02:00
|
|
|
bc_num_init_and_copy(dnum, snum);
|
2018-12-24 05:00:36 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if ENABLE_DC
|
|
|
|
static void dc_result_copy(BcResult *d, BcResult *src)
|
|
|
|
{
|
|
|
|
d->t = src->t;
|
|
|
|
|
|
|
|
switch (d->t) {
|
2018-12-24 23:41:31 +01:00
|
|
|
case XC_RESULT_TEMP:
|
|
|
|
case XC_RESULT_IBASE:
|
|
|
|
case XC_RESULT_SCALE:
|
|
|
|
case XC_RESULT_OBASE:
|
2021-06-12 12:35:15 +02:00
|
|
|
bc_num_init_and_copy(&d->d.n, &src->d.n);
|
2018-12-24 05:00:36 +01:00
|
|
|
break;
|
2018-12-24 23:41:31 +01:00
|
|
|
case XC_RESULT_VAR:
|
|
|
|
case XC_RESULT_ARRAY:
|
|
|
|
case XC_RESULT_ARRAY_ELEM:
|
2018-12-24 05:00:36 +01:00
|
|
|
d->d.id.name = xstrdup(src->d.id.name);
|
|
|
|
break;
|
2018-12-24 23:41:31 +01:00
|
|
|
case XC_RESULT_CONSTANT:
|
|
|
|
case XC_RESULT_STR:
|
2018-12-24 05:00:36 +01:00
|
|
|
memcpy(&d->d.n, &src->d.n, sizeof(BcNum));
|
|
|
|
break;
|
2019-01-04 13:58:46 +01:00
|
|
|
default: // placate compiler
|
|
|
|
// BC_RESULT_VOID, BC_RESULT_LAST, BC_RESULT_ONE - do not happen
|
|
|
|
break;
|
2018-12-24 05:00:36 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif // ENABLE_DC
|
|
|
|
|
|
|
|
static FAST_FUNC void bc_result_free(void *result)
|
|
|
|
{
|
|
|
|
BcResult *r = (BcResult *) result;
|
|
|
|
|
|
|
|
switch (r->t) {
|
2018-12-24 23:41:31 +01:00
|
|
|
case XC_RESULT_TEMP:
|
2019-01-04 13:58:46 +01:00
|
|
|
IF_BC(case BC_RESULT_VOID:)
|
2018-12-24 23:41:31 +01:00
|
|
|
case XC_RESULT_IBASE:
|
|
|
|
case XC_RESULT_SCALE:
|
|
|
|
case XC_RESULT_OBASE:
|
2018-12-24 05:00:36 +01:00
|
|
|
bc_num_free(&r->d.n);
|
|
|
|
break;
|
2018-12-24 23:41:31 +01:00
|
|
|
case XC_RESULT_VAR:
|
|
|
|
case XC_RESULT_ARRAY:
|
|
|
|
case XC_RESULT_ARRAY_ELEM:
|
2018-12-24 05:00:36 +01:00
|
|
|
free(r->d.id.name);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// Do nothing.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int bad_input_byte(char c)
|
|
|
|
{
|
|
|
|
if ((c < ' ' && c != '\t' && c != '\r' && c != '\n') // also allow '\v' '\f'?
|
|
|
|
|| c > 0x7e
|
|
|
|
) {
|
|
|
|
bc_error_fmt("illegal character 0x%02x", c);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_read_line(BcVec *vec, FILE *fp)
|
2018-12-24 05:00:36 +01:00
|
|
|
{
|
|
|
|
again:
|
2018-12-26 18:46:03 +01:00
|
|
|
bc_vec_pop_all(vec);
|
2018-12-24 05:00:36 +01:00
|
|
|
fflush_and_check();
|
|
|
|
|
2018-12-27 22:52:13 +01:00
|
|
|
#if ENABLE_FEATURE_BC_INTERACTIVE
|
2018-12-24 05:00:36 +01:00
|
|
|
if (G_interrupt) { // ^C was pressed
|
|
|
|
if (fp != stdin) {
|
|
|
|
// ^C while running a script (bc SCRIPT): die.
|
|
|
|
// We do not return to interactive prompt:
|
|
|
|
// user might be running us from a shell,
|
|
|
|
// and SCRIPT might be intended to terminate
|
|
|
|
// (e.g. contain a "halt" stmt).
|
|
|
|
// ^C dropping user into a bc prompt instead of
|
|
|
|
// the shell would be unexpected.
|
|
|
|
xfunc_die();
|
|
|
|
}
|
2021-01-04 14:41:20 +01:00
|
|
|
// There was ^C while running calculations
|
2018-12-24 05:00:36 +01:00
|
|
|
G_interrupt = 0;
|
2021-01-04 14:41:20 +01:00
|
|
|
// GNU bc says "interrupted execution." (to stdout, not stderr)
|
2018-12-24 05:00:36 +01:00
|
|
|
// GNU dc says "Interrupt!"
|
2021-01-04 14:41:20 +01:00
|
|
|
puts("\ninterrupted execution");
|
2018-12-24 05:00:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
# if ENABLE_FEATURE_EDITING
|
|
|
|
if (G_ttyin && fp == stdin) {
|
|
|
|
int n, i;
|
2020-12-21 21:36:58 +01:00
|
|
|
if (!G.line_input_state)
|
|
|
|
G.line_input_state = new_line_input_t(DO_HISTORY);
|
2018-12-24 05:00:36 +01:00
|
|
|
# define line_buf bb_common_bufsiz1
|
|
|
|
n = read_line_input(G.line_input_state, "", line_buf, COMMON_BUFSIZE);
|
|
|
|
if (n <= 0) { // read errors or EOF, or ^D, or ^C
|
2021-01-04 14:41:20 +01:00
|
|
|
//GNU bc prints this on ^C:
|
|
|
|
//if (n == 0) // ^C
|
|
|
|
// puts("(interrupt) Exiting bc.");
|
|
|
|
bc_vec_pushZeroByte(vec);
|
2018-12-24 05:00:36 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
i = 0;
|
|
|
|
for (;;) {
|
|
|
|
char c = line_buf[i++];
|
2019-01-01 02:19:02 +01:00
|
|
|
if (c == '\0') break;
|
2018-12-24 05:00:36 +01:00
|
|
|
if (bad_input_byte(c)) goto again;
|
|
|
|
}
|
2018-12-26 18:46:03 +01:00
|
|
|
bc_vec_string(vec, n, line_buf);
|
2018-12-24 05:00:36 +01:00
|
|
|
# undef line_buf
|
|
|
|
} else
|
|
|
|
# endif
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
int c;
|
|
|
|
bool bad_chars = 0;
|
|
|
|
|
|
|
|
do {
|
2019-01-01 02:19:02 +01:00
|
|
|
get_char:
|
2018-12-27 22:52:13 +01:00
|
|
|
#if ENABLE_FEATURE_BC_INTERACTIVE
|
2018-12-24 05:00:36 +01:00
|
|
|
if (G_interrupt) {
|
|
|
|
// ^C was pressed: ignore entire line, get another one
|
2019-01-01 02:19:02 +01:00
|
|
|
goto again;
|
2018-12-24 05:00:36 +01:00
|
|
|
}
|
|
|
|
#endif
|
2019-01-01 02:19:02 +01:00
|
|
|
c = fgetc(fp);
|
|
|
|
if (c == '\0')
|
|
|
|
goto get_char;
|
2018-12-24 05:00:36 +01:00
|
|
|
if (c == EOF) {
|
|
|
|
if (ferror(fp))
|
libbb: reduce the overhead of single parameter bb_error_msg() calls
Back in 2007, commit 0c97c9d43707 ("'simple' error message functions by
Loic Grenie") introduced bb_simple_perror_msg() to allow for a lower
overhead call to bb_perror_msg() when only a string was being printed
with no parameters. This saves space for some CPU architectures because
it avoids the overhead of a call to a variadic function. However there
has never been a simple version of bb_error_msg(), and since 2007 many
new calls to bb_perror_msg() have been added that only take a single
parameter and so could have been using bb_simple_perror_message().
This changeset introduces 'simple' versions of bb_info_msg(),
bb_error_msg(), bb_error_msg_and_die(), bb_herror_msg() and
bb_herror_msg_and_die(), and replaces all calls that only take a
single parameter, or use something like ("%s", arg), with calls to the
corresponding 'simple' version.
Since it is likely that single parameter calls to the variadic functions
may be accidentally reintroduced in the future a new debugging config
option WARN_SIMPLE_MSG has been introduced. This uses some macro magic
which will cause any such calls to generate a warning, but this is
turned off by default to avoid use of the unpleasant macros in normal
circumstances.
This is a large changeset due to the number of calls that have been
replaced. The only files that contain changes other than simple
substitution of function calls are libbb.h, libbb/herror_msg.c,
libbb/verror_msg.c and libbb/xfuncs_printf.c. In miscutils/devfsd.c,
networking/udhcp/common.h and util-linux/mdev.c additonal macros have
been added for logging so that single parameter and multiple parameter
logging variants exist.
The amount of space saved varies considerably by architecture, and was
found to be as follows (for 'defconfig' using GCC 7.4):
Arm: -92 bytes
MIPS: -52 bytes
PPC: -1836 bytes
x86_64: -938 bytes
Note that for the MIPS architecture only an exception had to be made
disabling the 'simple' calls for 'udhcp' (in networking/udhcp/common.h)
because it made these files larger on MIPS.
Signed-off-by: James Byrne <james.byrne@origamienergy.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2019-07-02 11:35:03 +02:00
|
|
|
bb_simple_perror_msg_and_die("input error");
|
2018-12-24 05:00:36 +01:00
|
|
|
// Note: EOF does not append '\n'
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
bad_chars |= bad_input_byte(c);
|
|
|
|
bc_vec_pushByte(vec, (char)c);
|
|
|
|
} while (c != '\n');
|
|
|
|
|
|
|
|
if (bad_chars) {
|
|
|
|
// Bad chars on this line
|
2018-12-25 22:32:41 +01:00
|
|
|
if (!G.prs.lex_filename) { // stdin
|
2018-12-24 05:00:36 +01:00
|
|
|
// ignore entire line, get another one
|
|
|
|
goto again;
|
|
|
|
}
|
2018-12-25 22:32:41 +01:00
|
|
|
bb_perror_msg_and_die("file '%s' is not text", G.prs.lex_filename);
|
2018-12-24 05:00:36 +01:00
|
|
|
}
|
|
|
|
bc_vec_pushZeroByte(vec);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Parsing routines
|
|
|
|
//
|
|
|
|
|
2018-12-29 02:24:19 +01:00
|
|
|
// "Input numbers may contain the characters 0-9 and A-Z.
|
|
|
|
// (Note: They must be capitals. Lower case letters are variable names.)
|
|
|
|
// Single digit numbers always have the value of the digit regardless of
|
|
|
|
// the value of ibase. (i.e. A = 10.) For multi-digit numbers, bc changes
|
|
|
|
// all input digits greater or equal to ibase to the value of ibase-1.
|
|
|
|
// This makes the number ZZZ always be the largest 3 digit number of the
|
|
|
|
// input base."
|
|
|
|
static bool xc_num_strValid(const char *val)
|
|
|
|
{
|
|
|
|
bool radix = false;
|
2018-12-24 05:00:36 +01:00
|
|
|
for (;;) {
|
|
|
|
BcDig c = *val++;
|
|
|
|
if (c == '\0')
|
|
|
|
break;
|
|
|
|
if (c == '.') {
|
|
|
|
if (radix) return false;
|
|
|
|
radix = true;
|
|
|
|
continue;
|
|
|
|
}
|
2018-12-29 02:24:19 +01:00
|
|
|
if ((c < '0' || c > '9') && (c < 'A' || c > 'Z'))
|
2018-12-24 05:00:36 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
// Note: n is already "bc_num_zero()"ed,
|
|
|
|
// leading zeroes in "val" are removed
|
|
|
|
static void bc_num_parseDecimal(BcNum *n, const char *val)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-24 05:00:36 +01:00
|
|
|
size_t len, i;
|
|
|
|
const char *ptr;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
len = strlen(val);
|
|
|
|
if (len == 0)
|
|
|
|
return;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-29 02:40:03 +01:00
|
|
|
bc_num_expand(n, len + 1); // +1 for e.g. "A" converting into 10
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
ptr = strchr(val, '.');
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
n->rdx = 0;
|
|
|
|
if (ptr != NULL)
|
|
|
|
n->rdx = (size_t)((val + len) - (ptr + 1));
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
for (i = 0; val[i]; ++i) {
|
|
|
|
if (val[i] != '0' && val[i] != '.') {
|
|
|
|
// Not entirely zero value - convert it, and exit
|
2018-12-29 02:24:19 +01:00
|
|
|
if (len == 1) {
|
2018-12-29 02:40:03 +01:00
|
|
|
unsigned c = val[0] - '0';
|
|
|
|
n->len = 1;
|
|
|
|
if (c > 9) { // A-Z => 10-36
|
|
|
|
n->len = 2;
|
2018-12-29 02:24:19 +01:00
|
|
|
c -= ('A' - '9' - 1);
|
2018-12-29 02:40:03 +01:00
|
|
|
n->num[1] = c/10;
|
|
|
|
c = c%10;
|
|
|
|
}
|
2018-12-29 02:24:19 +01:00
|
|
|
n->num[0] = c;
|
|
|
|
break;
|
|
|
|
}
|
2018-12-24 05:00:36 +01:00
|
|
|
i = len - 1;
|
|
|
|
for (;;) {
|
2018-12-29 02:24:19 +01:00
|
|
|
char c = val[i] - '0';
|
|
|
|
if (c > 9) // A-Z => 9
|
|
|
|
c = 9;
|
|
|
|
n->num[n->len] = c;
|
|
|
|
n->len++;
|
2018-12-24 05:00:36 +01:00
|
|
|
skip_dot:
|
|
|
|
if (i == 0) break;
|
|
|
|
if (val[--i] == '.') goto skip_dot;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// if for() exits without hitting if(), the value is entirely zero
|
2018-12-21 16:22:26 +01:00
|
|
|
}
|
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
// Note: n is already "bc_num_zero()"ed,
|
|
|
|
// leading zeroes in "val" are removed
|
|
|
|
static void bc_num_parseBase(BcNum *n, const char *val, unsigned base_t)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-24 05:00:36 +01:00
|
|
|
BcStatus s;
|
2018-12-31 19:42:13 +01:00
|
|
|
BcNum mult, result;
|
|
|
|
BcNum temp;
|
2018-12-24 05:00:36 +01:00
|
|
|
BcNum base;
|
2018-12-31 19:42:13 +01:00
|
|
|
BcDig temp_digs[ULONG_NUM_BUFSIZE];
|
2018-12-24 05:00:36 +01:00
|
|
|
BcDig base_digs[ULONG_NUM_BUFSIZE];
|
2018-12-31 19:42:13 +01:00
|
|
|
size_t digits;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
bc_num_init_DEF_SIZE(&mult);
|
2018-12-31 19:42:13 +01:00
|
|
|
|
|
|
|
temp.cap = ARRAY_SIZE(temp_digs);
|
|
|
|
temp.num = temp_digs;
|
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
base.cap = ARRAY_SIZE(base_digs);
|
|
|
|
base.num = base_digs;
|
|
|
|
bc_num_ulong2num(&base, base_t);
|
2018-12-31 19:42:13 +01:00
|
|
|
base_t--;
|
2018-12-05 21:03:16 +01:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
for (;;) {
|
2018-12-31 19:42:13 +01:00
|
|
|
unsigned v;
|
2018-12-31 19:50:06 +01:00
|
|
|
char c;
|
2018-12-31 19:42:13 +01:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
c = *val++;
|
|
|
|
if (c == '\0') goto int_err;
|
|
|
|
if (c == '.') break;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-31 19:42:13 +01:00
|
|
|
v = (unsigned)(c <= '9' ? c - '0' : c - 'A' + 10);
|
|
|
|
if (v > base_t) v = base_t;
|
2018-12-24 05:00:36 +01:00
|
|
|
|
|
|
|
s = zbc_num_mul(n, &base, &mult, 0);
|
|
|
|
if (s) goto int_err;
|
|
|
|
bc_num_ulong2num(&temp, v);
|
|
|
|
s = zbc_num_add(&mult, &temp, n, 0);
|
|
|
|
if (s) goto int_err;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
bc_num_init(&result, base.len);
|
|
|
|
//bc_num_zero(&result); - already is
|
|
|
|
bc_num_one(&mult);
|
2018-12-05 21:03:16 +01:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
digits = 0;
|
|
|
|
for (;;) {
|
2018-12-31 19:42:13 +01:00
|
|
|
unsigned v;
|
2018-12-31 19:50:06 +01:00
|
|
|
char c;
|
2018-12-31 19:42:13 +01:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
c = *val++;
|
|
|
|
if (c == '\0') break;
|
|
|
|
digits++;
|
2018-12-05 21:03:16 +01:00
|
|
|
|
2018-12-31 19:42:13 +01:00
|
|
|
v = (unsigned)(c <= '9' ? c - '0' : c - 'A' + 10);
|
|
|
|
if (v > base_t) v = base_t;
|
2018-12-24 05:00:36 +01:00
|
|
|
|
|
|
|
s = zbc_num_mul(&result, &base, &result, 0);
|
|
|
|
if (s) goto err;
|
|
|
|
bc_num_ulong2num(&temp, v);
|
|
|
|
s = zbc_num_add(&result, &temp, &result, 0);
|
|
|
|
if (s) goto err;
|
|
|
|
s = zbc_num_mul(&mult, &base, &mult, 0);
|
|
|
|
if (s) goto err;
|
2018-12-05 21:03:16 +01:00
|
|
|
}
|
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
s = zbc_num_div(&result, &mult, &result, digits);
|
|
|
|
if (s) goto err;
|
|
|
|
s = zbc_num_add(n, &result, n, digits);
|
|
|
|
if (s) goto err;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
if (n->len != 0) {
|
|
|
|
if (n->rdx < digits)
|
|
|
|
bc_num_extend(n, digits - n->rdx);
|
|
|
|
} else
|
|
|
|
bc_num_zero(n);
|
|
|
|
err:
|
|
|
|
bc_num_free(&result);
|
|
|
|
int_err:
|
|
|
|
bc_num_free(&mult);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_num_parse(BcNum *n, const char *val, unsigned base_t)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-31 19:42:13 +01:00
|
|
|
size_t i;
|
|
|
|
|
2018-12-29 02:24:19 +01:00
|
|
|
if (!xc_num_strValid(val))
|
2018-12-24 05:00:36 +01:00
|
|
|
RETURN_STATUS(bc_error("bad number string"));
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
bc_num_zero(n);
|
2018-12-31 19:42:13 +01:00
|
|
|
while (*val == '0')
|
|
|
|
val++;
|
|
|
|
for (i = 0; ; ++i) {
|
|
|
|
if (val[i] == '\0')
|
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
|
|
|
if (val[i] != '.' && val[i] != '0')
|
|
|
|
break;
|
|
|
|
}
|
2018-12-24 05:00:36 +01:00
|
|
|
|
2018-12-31 19:42:13 +01:00
|
|
|
if (base_t == 10 || val[1] == '\0')
|
|
|
|
// Decimal, or single-digit number
|
2018-12-24 05:00:36 +01:00
|
|
|
bc_num_parseDecimal(n, val);
|
|
|
|
else
|
|
|
|
bc_num_parseBase(n, val, base_t);
|
|
|
|
|
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_num_parse(...) (zxc_num_parse(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-26 18:32:43 +01:00
|
|
|
// p->lex_inbuf points to the current string to be parsed.
|
|
|
|
// if p->lex_inbuf points to '\0', it's either EOF or it points after
|
|
|
|
// last processed line's terminating '\n' (and more reading needs to be done
|
|
|
|
// to get next character).
|
|
|
|
//
|
|
|
|
// If you are in a situation where that is a possibility, call peek_inbuf().
|
|
|
|
// If necessary, it performs more reading and changes p->lex_inbuf,
|
|
|
|
// then it returns *p->lex_inbuf (which will be '\0' only if it's EOF).
|
|
|
|
// After it, just referencing *p->lex_inbuf is valid, and if it wasn't '\0',
|
|
|
|
// it's ok to do p->lex_inbuf++ once without end-of-buffer checking.
|
|
|
|
//
|
|
|
|
// eat_inbuf() is equvalent to "peek_inbuf(); if (c) p->lex_inbuf++":
|
|
|
|
// it returns current char and advances the pointer (if not EOF).
|
|
|
|
// After eat_inbuf(), referencing p->lex_inbuf[-1] and *p->lex_inbuf is valid.
|
|
|
|
//
|
|
|
|
// In many cases, you can use fast *p->lex_inbuf instead of peek_inbuf():
|
|
|
|
// unless prev char might have been '\n', *p->lex_inbuf is '\0' ONLY
|
|
|
|
// on real EOF, not end-of-buffer.
|
2018-12-26 18:46:03 +01:00
|
|
|
//
|
|
|
|
// bc cases to test interactively:
|
|
|
|
// 1 #comment\ - prints "1<newline>" at once (comment is not continued)
|
|
|
|
// 1 #comment/* - prints "1<newline>" at once
|
|
|
|
// 1 #comment" - prints "1<newline>" at once
|
|
|
|
// 1\#comment - error at once (\ is not a line continuation)
|
|
|
|
// 1 + /*"*/2 - prints "3<newline>" at once
|
|
|
|
// 1 + /*#*/2 - prints "3<newline>" at once
|
|
|
|
// "str\" - prints "str\" at once
|
|
|
|
// "str#" - prints "str#" at once
|
|
|
|
// "str/*" - prints "str/*" at once
|
|
|
|
// "str#\ - waits for second line
|
|
|
|
// end" - ...prints "str#\<newline>end"
|
2018-12-26 18:32:43 +01:00
|
|
|
static char peek_inbuf(void)
|
|
|
|
{
|
2018-12-31 18:48:10 +01:00
|
|
|
if (*G.prs.lex_inbuf == '\0'
|
|
|
|
&& G.prs.lex_input_fp
|
|
|
|
) {
|
|
|
|
xc_read_line(&G.input_buffer, G.prs.lex_input_fp);
|
|
|
|
G.prs.lex_inbuf = G.input_buffer.v;
|
2022-08-18 16:23:55 +02:00
|
|
|
// lex_next_at may point to now-freed data, update it:
|
|
|
|
G.prs.lex_next_at = G.prs.lex_inbuf;
|
2018-12-31 18:48:10 +01:00
|
|
|
if (G.input_buffer.len <= 1) // on EOF, len is 1 (NUL byte)
|
|
|
|
G.prs.lex_input_fp = NULL;
|
2018-12-26 18:32:43 +01:00
|
|
|
}
|
|
|
|
return *G.prs.lex_inbuf;
|
|
|
|
}
|
|
|
|
static char eat_inbuf(void)
|
|
|
|
{
|
|
|
|
char c = peek_inbuf();
|
|
|
|
if (c) G.prs.lex_inbuf++;
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_lex_lineComment(void)
|
2018-12-26 18:32:43 +01:00
|
|
|
{
|
|
|
|
BcParse *p = &G.prs;
|
|
|
|
char c;
|
|
|
|
|
|
|
|
// Try: echo -n '#foo' | bc
|
|
|
|
p->lex = XC_LEX_WHITESPACE;
|
|
|
|
|
2018-12-31 18:48:10 +01:00
|
|
|
// Not peek_inbuf(): we depend on input being done in whole lines:
|
2018-12-26 18:32:43 +01:00
|
|
|
// '\0' which isn't the EOF can only be seen after '\n'.
|
|
|
|
while ((c = *p->lex_inbuf) != '\n' && c != '\0')
|
|
|
|
p->lex_inbuf++;
|
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_lex_whitespace(void)
|
2018-12-26 18:32:43 +01:00
|
|
|
{
|
|
|
|
BcParse *p = &G.prs;
|
|
|
|
|
|
|
|
p->lex = XC_LEX_WHITESPACE;
|
|
|
|
for (;;) {
|
|
|
|
// We depend here on input being done in whole lines:
|
|
|
|
// '\0' which isn't the EOF can only be seen after '\n'.
|
|
|
|
char c = *p->lex_inbuf;
|
|
|
|
if (c == '\n') // this is XC_LEX_NLINE, not XC_LEX_WHITESPACE
|
|
|
|
break;
|
|
|
|
if (!isspace(c))
|
|
|
|
break;
|
|
|
|
p->lex_inbuf++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_lex_number(char last)
|
2018-12-26 18:32:43 +01:00
|
|
|
{
|
|
|
|
BcParse *p = &G.prs;
|
|
|
|
bool pt;
|
2018-12-29 02:40:03 +01:00
|
|
|
char last_valid_ch;
|
2018-12-26 18:32:43 +01:00
|
|
|
|
|
|
|
bc_vec_pop_all(&p->lex_strnumbuf);
|
|
|
|
bc_vec_pushByte(&p->lex_strnumbuf, last);
|
|
|
|
|
2018-12-29 02:40:03 +01:00
|
|
|
// bc: "Input numbers may contain the characters 0-9 and A-Z.
|
2018-12-29 02:24:19 +01:00
|
|
|
// (Note: They must be capitals. Lower case letters are variable names.)
|
|
|
|
// Single digit numbers always have the value of the digit regardless of
|
|
|
|
// the value of ibase. (i.e. A = 10.) For multi-digit numbers, bc changes
|
|
|
|
// all input digits greater or equal to ibase to the value of ibase-1.
|
|
|
|
// This makes the number ZZZ always be the largest 3 digit number of the
|
|
|
|
// input base."
|
2018-12-29 02:40:03 +01:00
|
|
|
// dc only allows A-F, the rules about single-char and multi-char are the same.
|
|
|
|
last_valid_ch = (IS_BC ? 'Z' : 'F');
|
2018-12-26 18:32:43 +01:00
|
|
|
pt = (last == '.');
|
|
|
|
p->lex = XC_LEX_NUMBER;
|
|
|
|
for (;;) {
|
|
|
|
// We depend here on input being done in whole lines:
|
|
|
|
// '\0' which isn't the EOF can only be seen after '\n'.
|
|
|
|
char c = *p->lex_inbuf;
|
|
|
|
check_c:
|
|
|
|
if (c == '\0')
|
|
|
|
break;
|
|
|
|
if (c == '\\' && p->lex_inbuf[1] == '\n') {
|
|
|
|
p->lex_inbuf += 2;
|
|
|
|
p->lex_line++;
|
2019-01-01 21:50:14 +01:00
|
|
|
dbg_lex("++p->lex_line=%zd", p->lex_line);
|
2018-12-26 18:32:43 +01:00
|
|
|
c = peek_inbuf(); // force next line to be read
|
|
|
|
goto check_c;
|
|
|
|
}
|
2018-12-29 02:40:03 +01:00
|
|
|
if (!isdigit(c) && (c < 'A' || c > last_valid_ch)) {
|
2018-12-26 18:32:43 +01:00
|
|
|
if (c != '.') break;
|
|
|
|
// if '.' was already seen, stop on second one:
|
|
|
|
if (pt) break;
|
|
|
|
pt = true;
|
|
|
|
}
|
2018-12-29 02:24:19 +01:00
|
|
|
// c is one of "0-9A-Z."
|
2018-12-26 18:32:43 +01:00
|
|
|
last = c;
|
|
|
|
bc_vec_push(&p->lex_strnumbuf, p->lex_inbuf);
|
|
|
|
p->lex_inbuf++;
|
|
|
|
}
|
|
|
|
if (last == '.') // remove trailing '.' if any
|
|
|
|
bc_vec_pop(&p->lex_strnumbuf);
|
|
|
|
bc_vec_pushZeroByte(&p->lex_strnumbuf);
|
|
|
|
|
|
|
|
G.err_line = G.prs.lex_line;
|
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_lex_number(...) (zxc_lex_number(__VA_ARGS__) COMMA_SUCCESS)
|
2018-12-26 18:32:43 +01:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_lex_name(void)
|
2018-12-26 18:32:43 +01:00
|
|
|
{
|
|
|
|
BcParse *p = &G.prs;
|
|
|
|
size_t i;
|
|
|
|
const char *buf;
|
|
|
|
|
|
|
|
p->lex = XC_LEX_NAME;
|
|
|
|
|
|
|
|
// Since names can't cross lines with \<newline>,
|
|
|
|
// we depend on the fact that whole line is in the buffer
|
|
|
|
i = 0;
|
|
|
|
buf = p->lex_inbuf - 1;
|
|
|
|
for (;;) {
|
|
|
|
char c = buf[i];
|
|
|
|
if ((c < 'a' || c > 'z') && !isdigit(c) && c != '_') break;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0 // We do not protect against people with gigabyte-long names
|
|
|
|
// This check makes sense only if size_t is (much) larger than BC_MAX_STRING.
|
|
|
|
if (SIZE_MAX > (BC_MAX_STRING | 0xff)) {
|
|
|
|
if (i > BC_MAX_STRING)
|
|
|
|
return bc_error("name too long: must be [1,"BC_MAX_STRING_STR"]");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
bc_vec_string(&p->lex_strnumbuf, i, buf);
|
|
|
|
|
|
|
|
// Increment the index. We minus 1 because it has already been incremented.
|
|
|
|
p->lex_inbuf += i - 1;
|
|
|
|
|
|
|
|
//return BC_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
IF_BC(static BC_STATUS zbc_lex_token(void);)
|
|
|
|
IF_DC(static BC_STATUS zdc_lex_token(void);)
|
2018-12-22 18:24:19 +01:00
|
|
|
#define zbc_lex_token(...) (zbc_lex_token(__VA_ARGS__) COMMA_SUCCESS)
|
|
|
|
#define zdc_lex_token(...) (zdc_lex_token(__VA_ARGS__) COMMA_SUCCESS)
|
2018-12-22 06:00:25 +01:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_lex_next(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 22:32:41 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
|
2019-01-01 21:50:14 +01:00
|
|
|
G.err_line = p->lex_line;
|
2018-12-25 22:32:41 +01:00
|
|
|
p->lex_last = p->lex;
|
2018-12-26 18:59:42 +01:00
|
|
|
//why?
|
|
|
|
// if (p->lex_last == XC_LEX_EOF)
|
|
|
|
// RETURN_STATUS(bc_error("end of file"));
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
// Loop until failure or we don't have whitespace. This
|
|
|
|
// is so the parser doesn't get inundated with whitespace.
|
2018-12-24 15:05:49 +01:00
|
|
|
// Comments are also XC_LEX_WHITESPACE tokens and eaten here.
|
2018-12-12 13:58:55 +01:00
|
|
|
s = BC_STATUS_SUCCESS;
|
2018-11-03 11:00:21 -06:00
|
|
|
do {
|
2018-12-26 18:32:43 +01:00
|
|
|
if (*p->lex_inbuf == '\0') {
|
2018-12-25 22:32:41 +01:00
|
|
|
p->lex = XC_LEX_EOF;
|
2018-12-26 18:32:43 +01:00
|
|
|
if (peek_inbuf() == '\0')
|
2018-12-17 09:51:43 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
|
|
|
}
|
2018-12-25 23:15:59 +01:00
|
|
|
p->lex_next_at = p->lex_inbuf;
|
2018-12-14 17:51:17 +01:00
|
|
|
dbg_lex("next string to parse:'%.*s'",
|
2018-12-25 22:32:41 +01:00
|
|
|
(int)(strchrnul(p->lex_next_at, '\n') - p->lex_next_at),
|
|
|
|
p->lex_next_at
|
2018-12-25 18:37:52 +01:00
|
|
|
);
|
2018-12-22 06:00:25 +01:00
|
|
|
if (IS_BC) {
|
2018-12-25 20:40:55 +01:00
|
|
|
IF_BC(s = zbc_lex_token());
|
2018-12-22 06:00:25 +01:00
|
|
|
} else {
|
2018-12-25 20:40:55 +01:00
|
|
|
IF_DC(s = zdc_lex_token());
|
2018-12-22 06:00:25 +01:00
|
|
|
}
|
2018-12-25 22:32:41 +01:00
|
|
|
} while (!s && p->lex == XC_LEX_WHITESPACE);
|
|
|
|
dbg_lex("p->lex from string:%d", p->lex);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-12 13:58:55 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_lex_next(...) (zxc_lex_next(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 19:43:03 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_lex_skip_if_at_NLINE(void)
|
2018-12-16 17:30:35 +01:00
|
|
|
{
|
2018-12-25 22:32:41 +01:00
|
|
|
if (G.prs.lex == XC_LEX_NLINE)
|
2018-12-27 18:23:58 +01:00
|
|
|
RETURN_STATUS(zxc_lex_next());
|
2018-12-16 17:30:35 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_lex_skip_if_at_NLINE(...) (zbc_lex_skip_if_at_NLINE(__VA_ARGS__) COMMA_SUCCESS)
|
2018-12-16 17:30:35 +01:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_lex_next_and_skip_NLINE(void)
|
2018-12-16 17:30:35 +01:00
|
|
|
{
|
|
|
|
BcStatus s;
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-16 17:30:35 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
// if(cond)<newline>stmt is accepted too (but not 2+ newlines)
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_lex_skip_if_at_NLINE();
|
2018-12-16 17:30:35 +01:00
|
|
|
RETURN_STATUS(s);
|
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_lex_next_and_skip_NLINE(...) (zbc_lex_next_and_skip_NLINE(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_lex_identifier(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 22:32:41 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
2018-12-05 00:22:34 +01:00
|
|
|
unsigned i;
|
2018-12-25 23:15:59 +01:00
|
|
|
const char *buf = p->lex_inbuf - 1;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-05 00:22:34 +01:00
|
|
|
for (i = 0; i < ARRAY_SIZE(bc_lex_kws); ++i) {
|
|
|
|
const char *keyword8 = bc_lex_kws[i].name8;
|
|
|
|
unsigned j = 0;
|
|
|
|
while (buf[j] != '\0' && buf[j] == keyword8[j]) {
|
|
|
|
j++;
|
|
|
|
if (j == 8) goto match;
|
|
|
|
}
|
|
|
|
if (keyword8[j] != '\0')
|
|
|
|
continue;
|
|
|
|
match:
|
|
|
|
// buf starts with keyword bc_lex_kws[i]
|
2022-08-30 16:41:17 +02:00
|
|
|
if (isalnum(buf[j]) || buf[j] == '_')
|
2018-12-20 16:48:50 +01:00
|
|
|
continue; // "ifz" does not match "if" keyword, "if." does
|
2018-12-25 22:32:41 +01:00
|
|
|
p->lex = BC_LEX_KEY_1st_keyword + i;
|
2018-12-25 21:44:10 +01:00
|
|
|
if (!keyword_is_POSIX(i)) {
|
2018-12-24 18:11:41 +01:00
|
|
|
s = zbc_posix_error_fmt("%sthe '%.8s' keyword", "POSIX does not allow ", bc_lex_kws[i].name8);
|
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-05 00:22:34 +01:00
|
|
|
|
|
|
|
// We minus 1 because the index has already been incremented.
|
2018-12-25 23:15:59 +01:00
|
|
|
p->lex_inbuf += j - 1;
|
2018-12-12 13:58:55 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_lex_name();
|
2018-12-18 03:16:48 +01:00
|
|
|
s = BC_STATUS_SUCCESS;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex_strnumbuf.len > 2) {
|
2018-12-05 18:31:19 +01:00
|
|
|
// Prevent this:
|
|
|
|
// >>> qwe=1
|
2018-12-25 21:28:25 +01:00
|
|
|
// bc: POSIX only allows one character names; this is bad: 'qwe=1
|
2018-12-05 18:31:19 +01:00
|
|
|
// '
|
|
|
|
unsigned len = strchrnul(buf, '\n') - buf;
|
2018-12-25 21:28:25 +01:00
|
|
|
s = zbc_posix_error_fmt("POSIX only allows one character names; this is bad: '%.*s'", len, buf);
|
2018-12-05 18:31:19 +01:00
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-12 13:58:55 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_lex_identifier(...) (zbc_lex_identifier(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_lex_string(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 22:32:41 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
p->lex = XC_LEX_STR;
|
2018-12-26 18:32:43 +01:00
|
|
|
bc_vec_pop_all(&p->lex_strnumbuf);
|
2018-12-18 14:03:20 +01:00
|
|
|
for (;;) {
|
2018-12-26 18:32:43 +01:00
|
|
|
char c = peek_inbuf(); // strings can cross lines
|
2018-12-18 14:03:20 +01:00
|
|
|
if (c == '\0') {
|
2018-12-26 21:01:41 +01:00
|
|
|
RETURN_STATUS(bc_error("unterminated string"));
|
2018-12-18 14:03:20 +01:00
|
|
|
}
|
|
|
|
if (c == '"')
|
|
|
|
break;
|
2019-01-01 21:50:14 +01:00
|
|
|
if (c == '\n') {
|
2018-12-26 18:32:43 +01:00
|
|
|
p->lex_line++;
|
2019-01-01 21:50:14 +01:00
|
|
|
dbg_lex("++p->lex_line=%zd", p->lex_line);
|
|
|
|
}
|
2018-12-26 18:32:43 +01:00
|
|
|
bc_vec_push(&p->lex_strnumbuf, p->lex_inbuf);
|
|
|
|
p->lex_inbuf++;
|
2018-12-07 15:50:14 +01:00
|
|
|
}
|
2018-12-26 18:32:43 +01:00
|
|
|
bc_vec_pushZeroByte(&p->lex_strnumbuf);
|
|
|
|
p->lex_inbuf++;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
G.err_line = p->lex_line;
|
2018-12-12 16:08:46 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_lex_string(...) (zbc_lex_string(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-26 18:32:43 +01:00
|
|
|
static void parse_lex_by_checking_eq_sign(unsigned with_and_without)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 22:32:41 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-12-25 23:15:59 +01:00
|
|
|
if (*p->lex_inbuf == '=') {
|
2018-12-26 18:32:43 +01:00
|
|
|
// ^^^ not using peek_inbuf() since '==' etc can't be split across lines
|
2018-12-25 23:15:59 +01:00
|
|
|
p->lex_inbuf++;
|
2018-12-14 16:48:34 +01:00
|
|
|
with_and_without >>= 8; // store "with" value
|
|
|
|
} // else store "without" value
|
2018-12-25 22:32:41 +01:00
|
|
|
p->lex = (with_and_without & 0xff);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-26 18:32:43 +01:00
|
|
|
#define parse_lex_by_checking_eq_sign(with, without) \
|
|
|
|
parse_lex_by_checking_eq_sign(((with)<<8)|(without))
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_lex_comment(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 22:32:41 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
p->lex = XC_LEX_WHITESPACE;
|
2018-12-26 18:32:43 +01:00
|
|
|
// here lex_inbuf is at '*' of opening comment delimiter
|
2018-12-03 19:12:29 +01:00
|
|
|
for (;;) {
|
2018-12-26 18:32:43 +01:00
|
|
|
char c;
|
|
|
|
|
|
|
|
p->lex_inbuf++;
|
|
|
|
c = peek_inbuf();
|
2018-12-03 19:12:29 +01:00
|
|
|
check_star:
|
|
|
|
if (c == '*') {
|
2018-12-26 18:32:43 +01:00
|
|
|
p->lex_inbuf++;
|
2018-12-26 21:01:41 +01:00
|
|
|
c = *p->lex_inbuf; // no need to peek_inbuf()
|
2018-12-03 19:12:29 +01:00
|
|
|
if (c == '/')
|
|
|
|
break;
|
|
|
|
goto check_star;
|
|
|
|
}
|
|
|
|
if (c == '\0') {
|
2018-12-25 21:28:25 +01:00
|
|
|
RETURN_STATUS(bc_error("unterminated comment"));
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2019-01-01 21:50:14 +01:00
|
|
|
if (c == '\n') {
|
2018-12-26 18:32:43 +01:00
|
|
|
p->lex_line++;
|
2019-01-01 21:50:14 +01:00
|
|
|
dbg_lex("++p->lex_line=%zd", p->lex_line);
|
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-26 18:32:43 +01:00
|
|
|
p->lex_inbuf++; // skip trailing '/'
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
G.err_line = p->lex_line;
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_lex_comment(...) (zbc_lex_comment(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-22 18:24:19 +01:00
|
|
|
#undef zbc_lex_token
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_lex_token(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 22:32:41 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s = BC_STATUS_SUCCESS;
|
2018-12-26 18:32:43 +01:00
|
|
|
char c = eat_inbuf();
|
2018-12-25 23:15:59 +01:00
|
|
|
char c2;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
// This is the workhorse of the lexer.
|
|
|
|
switch (c) {
|
2018-12-26 19:24:15 +01:00
|
|
|
// case '\0': // probably never reached
|
|
|
|
// p->lex_inbuf--;
|
|
|
|
// p->lex = XC_LEX_EOF;
|
|
|
|
// break;
|
|
|
|
case '\n':
|
|
|
|
p->lex_line++;
|
2019-01-01 21:50:14 +01:00
|
|
|
dbg_lex("++p->lex_line=%zd", p->lex_line);
|
2018-12-26 19:24:15 +01:00
|
|
|
p->lex = XC_LEX_NLINE;
|
|
|
|
break;
|
|
|
|
case '\t':
|
|
|
|
case '\v':
|
|
|
|
case '\f':
|
|
|
|
case '\r':
|
|
|
|
case ' ':
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_lex_whitespace();
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case '!':
|
|
|
|
parse_lex_by_checking_eq_sign(XC_LEX_OP_REL_NE, BC_LEX_OP_BOOL_NOT);
|
|
|
|
if (p->lex == BC_LEX_OP_BOOL_NOT) {
|
|
|
|
s = zbc_POSIX_does_not_allow_bool_ops_this_is_bad("!");
|
2018-12-24 18:11:41 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-26 19:24:15 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '"':
|
|
|
|
s = zbc_lex_string();
|
|
|
|
break;
|
|
|
|
case '#':
|
|
|
|
s = zbc_POSIX_does_not_allow("'#' script comments");
|
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_lex_lineComment();
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case '%':
|
|
|
|
parse_lex_by_checking_eq_sign(BC_LEX_OP_ASSIGN_MODULUS, XC_LEX_OP_MODULUS);
|
|
|
|
break;
|
|
|
|
case '&':
|
|
|
|
c2 = *p->lex_inbuf;
|
|
|
|
if (c2 == '&') {
|
|
|
|
s = zbc_POSIX_does_not_allow_bool_ops_this_is_bad("&&");
|
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
p->lex_inbuf++;
|
|
|
|
p->lex = BC_LEX_OP_BOOL_AND;
|
|
|
|
} else {
|
|
|
|
p->lex = XC_LEX_INVALID;
|
|
|
|
s = bc_error_bad_character('&');
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '(':
|
|
|
|
case ')':
|
|
|
|
p->lex = (BcLexType)(c - '(' + BC_LEX_LPAREN);
|
|
|
|
break;
|
|
|
|
case '*':
|
|
|
|
parse_lex_by_checking_eq_sign(BC_LEX_OP_ASSIGN_MULTIPLY, XC_LEX_OP_MULTIPLY);
|
|
|
|
break;
|
|
|
|
case '+':
|
|
|
|
c2 = *p->lex_inbuf;
|
|
|
|
if (c2 == '+') {
|
|
|
|
p->lex_inbuf++;
|
|
|
|
p->lex = BC_LEX_OP_INC;
|
|
|
|
} else
|
|
|
|
parse_lex_by_checking_eq_sign(BC_LEX_OP_ASSIGN_PLUS, XC_LEX_OP_PLUS);
|
|
|
|
break;
|
|
|
|
case ',':
|
|
|
|
p->lex = BC_LEX_COMMA;
|
|
|
|
break;
|
|
|
|
case '-':
|
|
|
|
c2 = *p->lex_inbuf;
|
|
|
|
if (c2 == '-') {
|
|
|
|
p->lex_inbuf++;
|
|
|
|
p->lex = BC_LEX_OP_DEC;
|
|
|
|
} else
|
|
|
|
parse_lex_by_checking_eq_sign(BC_LEX_OP_ASSIGN_MINUS, XC_LEX_OP_MINUS);
|
|
|
|
break;
|
|
|
|
case '.':
|
|
|
|
if (isdigit(*p->lex_inbuf))
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_number(c);
|
2018-12-26 19:24:15 +01:00
|
|
|
else {
|
|
|
|
p->lex = BC_LEX_KEY_LAST;
|
|
|
|
s = zbc_POSIX_does_not_allow("'.' as 'last'");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '/':
|
|
|
|
c2 = *p->lex_inbuf;
|
|
|
|
if (c2 == '*')
|
|
|
|
s = zbc_lex_comment();
|
|
|
|
else
|
|
|
|
parse_lex_by_checking_eq_sign(BC_LEX_OP_ASSIGN_DIVIDE, XC_LEX_OP_DIVIDE);
|
|
|
|
break;
|
|
|
|
case '0':
|
|
|
|
case '1':
|
|
|
|
case '2':
|
|
|
|
case '3':
|
|
|
|
case '4':
|
|
|
|
case '5':
|
|
|
|
case '6':
|
|
|
|
case '7':
|
|
|
|
case '8':
|
|
|
|
case '9':
|
|
|
|
case 'A':
|
|
|
|
case 'B':
|
|
|
|
case 'C':
|
|
|
|
case 'D':
|
|
|
|
case 'E':
|
|
|
|
case 'F':
|
2018-12-29 02:24:19 +01:00
|
|
|
case 'G':
|
|
|
|
case 'H':
|
|
|
|
case 'I':
|
|
|
|
case 'J':
|
|
|
|
case 'K':
|
|
|
|
case 'L':
|
|
|
|
case 'M':
|
|
|
|
case 'N':
|
|
|
|
case 'O':
|
|
|
|
case 'P':
|
|
|
|
case 'Q':
|
|
|
|
case 'R':
|
|
|
|
case 'S':
|
|
|
|
case 'T':
|
|
|
|
case 'U':
|
|
|
|
case 'V':
|
|
|
|
case 'W':
|
|
|
|
case 'X':
|
|
|
|
case 'Y':
|
|
|
|
case 'Z':
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_number(c);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case ';':
|
|
|
|
p->lex = BC_LEX_SCOLON;
|
|
|
|
break;
|
|
|
|
case '<':
|
|
|
|
parse_lex_by_checking_eq_sign(XC_LEX_OP_REL_LE, XC_LEX_OP_REL_LT);
|
|
|
|
break;
|
|
|
|
case '=':
|
|
|
|
parse_lex_by_checking_eq_sign(XC_LEX_OP_REL_EQ, BC_LEX_OP_ASSIGN);
|
|
|
|
break;
|
|
|
|
case '>':
|
|
|
|
parse_lex_by_checking_eq_sign(XC_LEX_OP_REL_GE, XC_LEX_OP_REL_GT);
|
|
|
|
break;
|
|
|
|
case '[':
|
|
|
|
case ']':
|
|
|
|
p->lex = (BcLexType)(c - '[' + BC_LEX_LBRACKET);
|
|
|
|
break;
|
|
|
|
case '\\':
|
|
|
|
if (*p->lex_inbuf == '\n') {
|
|
|
|
p->lex = XC_LEX_WHITESPACE;
|
|
|
|
p->lex_inbuf++;
|
|
|
|
} else
|
|
|
|
s = bc_error_bad_character(c);
|
|
|
|
break;
|
|
|
|
case '^':
|
|
|
|
parse_lex_by_checking_eq_sign(BC_LEX_OP_ASSIGN_POWER, XC_LEX_OP_POWER);
|
|
|
|
break;
|
|
|
|
case 'a':
|
|
|
|
case 'b':
|
|
|
|
case 'c':
|
|
|
|
case 'd':
|
|
|
|
case 'e':
|
|
|
|
case 'f':
|
|
|
|
case 'g':
|
|
|
|
case 'h':
|
|
|
|
case 'i':
|
|
|
|
case 'j':
|
|
|
|
case 'k':
|
|
|
|
case 'l':
|
|
|
|
case 'm':
|
|
|
|
case 'n':
|
|
|
|
case 'o':
|
|
|
|
case 'p':
|
|
|
|
case 'q':
|
|
|
|
case 'r':
|
|
|
|
case 's':
|
|
|
|
case 't':
|
|
|
|
case 'u':
|
|
|
|
case 'v':
|
|
|
|
case 'w':
|
|
|
|
case 'x':
|
|
|
|
case 'y':
|
|
|
|
case 'z':
|
|
|
|
s = zbc_lex_identifier();
|
|
|
|
break;
|
|
|
|
case '{':
|
|
|
|
case '}':
|
|
|
|
p->lex = (BcLexType)(c - '{' + BC_LEX_LBRACE);
|
|
|
|
break;
|
|
|
|
case '|':
|
|
|
|
c2 = *p->lex_inbuf;
|
|
|
|
if (c2 == '|') {
|
|
|
|
s = zbc_POSIX_does_not_allow_bool_ops_this_is_bad("||");
|
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
p->lex_inbuf++;
|
|
|
|
p->lex = BC_LEX_OP_BOOL_OR;
|
|
|
|
} else {
|
2018-12-25 22:32:41 +01:00
|
|
|
p->lex = XC_LEX_INVALID;
|
2018-12-05 16:03:46 +01:00
|
|
|
s = bc_error_bad_character(c);
|
2018-12-26 19:24:15 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
p->lex = XC_LEX_INVALID;
|
|
|
|
s = bc_error_bad_character(c);
|
|
|
|
break;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-12 13:58:55 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-22 18:24:19 +01:00
|
|
|
#define zbc_lex_token(...) (zbc_lex_token(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
#endif // ENABLE_BC
|
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_DC
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zdc_lex_register(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 22:32:41 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-12-25 23:15:59 +01:00
|
|
|
if (G_exreg && isspace(*p->lex_inbuf)) {
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_lex_whitespace(); // eats whitespace (but not newline)
|
|
|
|
p->lex_inbuf++; // xc_lex_name() expects this
|
|
|
|
xc_lex_name();
|
2018-12-24 01:53:55 +01:00
|
|
|
} else {
|
2018-12-25 22:32:41 +01:00
|
|
|
bc_vec_pop_all(&p->lex_strnumbuf);
|
2018-12-25 23:15:59 +01:00
|
|
|
bc_vec_push(&p->lex_strnumbuf, p->lex_inbuf++);
|
2018-12-25 22:32:41 +01:00
|
|
|
bc_vec_pushZeroByte(&p->lex_strnumbuf);
|
|
|
|
p->lex = XC_LEX_NAME;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zdc_lex_register(...) (zdc_lex_register(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zdc_lex_string(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 22:32:41 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-12-26 18:32:43 +01:00
|
|
|
size_t depth;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
p->lex = XC_LEX_STR;
|
|
|
|
bc_vec_pop_all(&p->lex_strnumbuf);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-18 13:48:37 +01:00
|
|
|
depth = 1;
|
|
|
|
for (;;) {
|
2018-12-26 18:32:43 +01:00
|
|
|
char c = peek_inbuf();
|
2018-12-18 13:48:37 +01:00
|
|
|
if (c == '\0') {
|
2018-12-26 18:32:43 +01:00
|
|
|
RETURN_STATUS(bc_error("unterminated string"));
|
2018-12-18 13:48:37 +01:00
|
|
|
}
|
2018-12-26 18:32:43 +01:00
|
|
|
if (c == '[') depth++;
|
|
|
|
if (c == ']')
|
|
|
|
if (--depth == 0)
|
|
|
|
break;
|
2019-01-01 21:50:14 +01:00
|
|
|
if (c == '\n') {
|
2018-12-26 18:32:43 +01:00
|
|
|
p->lex_line++;
|
2019-01-01 21:50:14 +01:00
|
|
|
dbg_lex("++p->lex_line=%zd", p->lex_line);
|
|
|
|
}
|
2018-12-26 18:32:43 +01:00
|
|
|
bc_vec_push(&p->lex_strnumbuf, p->lex_inbuf);
|
|
|
|
p->lex_inbuf++;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-25 22:32:41 +01:00
|
|
|
bc_vec_pushZeroByte(&p->lex_strnumbuf);
|
2018-12-26 18:32:43 +01:00
|
|
|
p->lex_inbuf++; // skip trailing ']'
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
G.err_line = p->lex_line;
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zdc_lex_string(...) (zdc_lex_string(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-22 18:24:19 +01:00
|
|
|
#undef zdc_lex_token
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zdc_lex_token(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-17 11:58:20 +01:00
|
|
|
static const //BcLexType - should be this type, but narrower type saves size:
|
|
|
|
uint8_t
|
2018-12-26 21:17:12 +01:00
|
|
|
dc_lex_regs[] ALIGN1 = {
|
2018-12-24 14:14:23 +01:00
|
|
|
XC_LEX_OP_REL_EQ, XC_LEX_OP_REL_LE, XC_LEX_OP_REL_GE, XC_LEX_OP_REL_NE,
|
2018-12-24 15:00:56 +01:00
|
|
|
XC_LEX_OP_REL_LT, XC_LEX_OP_REL_GT, DC_LEX_SCOLON, DC_LEX_COLON,
|
|
|
|
DC_LEX_ELSE, DC_LEX_LOAD, DC_LEX_LOAD_POP, DC_LEX_OP_ASSIGN,
|
2018-12-24 12:25:20 +01:00
|
|
|
DC_LEX_STORE_PUSH,
|
2018-12-17 11:58:20 +01:00
|
|
|
};
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-12-24 01:53:55 +01:00
|
|
|
BcStatus s;
|
|
|
|
char c, c2;
|
2018-11-03 11:00:21 -06:00
|
|
|
size_t i;
|
|
|
|
|
2018-12-05 00:22:34 +01:00
|
|
|
for (i = 0; i < ARRAY_SIZE(dc_lex_regs); ++i) {
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex_last == dc_lex_regs[i])
|
2018-12-25 20:40:55 +01:00
|
|
|
RETURN_STATUS(zdc_lex_register());
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-24 01:53:55 +01:00
|
|
|
s = BC_STATUS_SUCCESS;
|
2018-12-26 18:32:43 +01:00
|
|
|
c = eat_inbuf();
|
2018-12-11 23:50:14 +01:00
|
|
|
if (c >= '%' && c <= '~'
|
2018-12-25 22:32:41 +01:00
|
|
|
&& (p->lex = dc_char_to_LEX[c - '%']) != XC_LEX_INVALID
|
2018-12-11 23:50:14 +01:00
|
|
|
) {
|
2018-12-12 13:58:55 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// This is the workhorse of the lexer.
|
|
|
|
switch (c) {
|
2018-12-26 19:24:15 +01:00
|
|
|
// case '\0': // probably never reached
|
|
|
|
// p->lex = XC_LEX_EOF;
|
|
|
|
// break;
|
|
|
|
case '\n':
|
|
|
|
// '\n' is XC_LEX_NLINE, not XC_LEX_WHITESPACE
|
|
|
|
// (and "case '\n':" is not just empty here)
|
|
|
|
// only to allow interactive dc have a way to exit
|
|
|
|
// "parse" stage of "parse,execute" loop
|
|
|
|
// on <enter>, not on _next_ token (which would mean
|
|
|
|
// commands are not executed on pressing <enter>).
|
|
|
|
// IOW: typing "1p<enter>" should print "1" _at once_,
|
|
|
|
// not after some more input.
|
|
|
|
p->lex_line++;
|
2019-01-01 21:50:14 +01:00
|
|
|
dbg_lex("++p->lex_line=%zd", p->lex_line);
|
2018-12-26 19:24:15 +01:00
|
|
|
p->lex = XC_LEX_NLINE;
|
|
|
|
break;
|
|
|
|
case '\t':
|
|
|
|
case '\v':
|
|
|
|
case '\f':
|
|
|
|
case '\r':
|
|
|
|
case ' ':
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_lex_whitespace();
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case '!':
|
|
|
|
c2 = *p->lex_inbuf;
|
|
|
|
if (c2 == '=')
|
|
|
|
p->lex = XC_LEX_OP_REL_NE;
|
|
|
|
else if (c2 == '<')
|
|
|
|
p->lex = XC_LEX_OP_REL_LE;
|
|
|
|
else if (c2 == '>')
|
|
|
|
p->lex = XC_LEX_OP_REL_GE;
|
|
|
|
else
|
|
|
|
RETURN_STATUS(bc_error_bad_character(c));
|
|
|
|
p->lex_inbuf++;
|
|
|
|
break;
|
|
|
|
case '#':
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_lex_lineComment();
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case '.':
|
|
|
|
if (isdigit(*p->lex_inbuf))
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_number(c);
|
2018-12-26 19:24:15 +01:00
|
|
|
else
|
2018-12-05 16:03:46 +01:00
|
|
|
s = bc_error_bad_character(c);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case '0':
|
|
|
|
case '1':
|
|
|
|
case '2':
|
|
|
|
case '3':
|
|
|
|
case '4':
|
|
|
|
case '5':
|
|
|
|
case '6':
|
|
|
|
case '7':
|
|
|
|
case '8':
|
|
|
|
case '9':
|
|
|
|
case 'A':
|
|
|
|
case 'B':
|
|
|
|
case 'C':
|
|
|
|
case 'D':
|
|
|
|
case 'E':
|
|
|
|
case 'F':
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_number(c);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case '[':
|
|
|
|
s = zdc_lex_string();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
p->lex = XC_LEX_INVALID;
|
|
|
|
s = bc_error_bad_character(c);
|
|
|
|
break;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-12 13:58:55 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-22 18:24:19 +01:00
|
|
|
#define zdc_lex_token(...) (zdc_lex_token(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
#endif // ENABLE_DC
|
|
|
|
|
2019-01-09 11:17:19 +01:00
|
|
|
static void xc_parse_push(unsigned i)
|
2018-12-14 17:51:17 +01:00
|
|
|
{
|
2018-12-26 20:02:09 +01:00
|
|
|
BcVec *code = &G.prs.func->code;
|
|
|
|
dbg_compile("%s:%d pushing bytecode %zd:%d", __func__, __LINE__, code->len, i);
|
2019-01-09 11:17:19 +01:00
|
|
|
bc_vec_pushByte(code, (uint8_t)i);
|
2018-12-14 17:51:17 +01:00
|
|
|
}
|
2018-12-06 13:10:56 +01:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_parse_pushName(char *name)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-26 20:30:47 +01:00
|
|
|
#if 1
|
|
|
|
BcVec *code = &G.prs.func->code;
|
|
|
|
size_t pos = code->len;
|
|
|
|
size_t len = strlen(name) + 1;
|
|
|
|
|
|
|
|
bc_vec_expand(code, pos + len);
|
|
|
|
strcpy(code->v + pos, name);
|
|
|
|
code->len = pos + len;
|
|
|
|
#else
|
|
|
|
// Smaller code, but way slow:
|
2018-12-26 20:02:09 +01:00
|
|
|
do {
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(*name);
|
2018-12-26 20:02:09 +01:00
|
|
|
} while (*name++);
|
2018-12-26 20:30:47 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-26 20:30:47 +01:00
|
|
|
// Indexes < 0xfc are encoded verbatim, else first byte is
|
|
|
|
// 0xfc, 0xfd, 0xfe or 0xff, encoding "1..4 bytes",
|
|
|
|
// followed by that many bytes, lsb first.
|
|
|
|
// (The above describes 32-bit case).
|
|
|
|
#define SMALL_INDEX_LIMIT (0x100 - sizeof(size_t))
|
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
static void bc_vec_pushIndex(BcVec *v, size_t idx)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-12 16:44:34 +01:00
|
|
|
size_t mask;
|
|
|
|
unsigned amt;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-18 02:23:53 +01:00
|
|
|
dbg_lex("%s:%d pushing index %zd", __func__, __LINE__, idx);
|
2018-12-26 20:30:47 +01:00
|
|
|
if (idx < SMALL_INDEX_LIMIT) {
|
2019-01-25 14:24:03 +01:00
|
|
|
bc_vec_pushByte(v, idx);
|
2019-01-02 16:30:24 +01:00
|
|
|
return;
|
2018-12-26 20:30:47 +01:00
|
|
|
}
|
|
|
|
|
2018-12-12 16:44:34 +01:00
|
|
|
mask = ((size_t)0xff) << (sizeof(idx) * 8 - 8);
|
|
|
|
amt = sizeof(idx);
|
2018-12-29 02:24:19 +01:00
|
|
|
for (;;) {
|
2018-12-12 16:44:34 +01:00
|
|
|
if (idx & mask) break;
|
|
|
|
mask >>= 8;
|
|
|
|
amt--;
|
2018-12-29 02:24:19 +01:00
|
|
|
}
|
|
|
|
// amt is at least 1 here - "one byte of length data follows"
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
bc_vec_pushByte(v, (SMALL_INDEX_LIMIT - 1) + amt);
|
2018-12-12 16:44:34 +01:00
|
|
|
|
2019-01-02 16:30:24 +01:00
|
|
|
do {
|
2019-01-25 14:24:03 +01:00
|
|
|
bc_vec_pushByte(v, (unsigned char)idx);
|
2018-12-12 16:44:34 +01:00
|
|
|
idx >>= 8;
|
2019-01-02 16:30:24 +01:00
|
|
|
} while (idx != 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
static void xc_parse_pushIndex(size_t idx)
|
|
|
|
{
|
|
|
|
bc_vec_pushIndex(&G.prs.func->code, idx);
|
|
|
|
}
|
|
|
|
|
2019-01-09 11:17:19 +01:00
|
|
|
static void xc_parse_pushInst_and_Index(unsigned inst, size_t idx)
|
|
|
|
{
|
|
|
|
xc_parse_push(inst);
|
|
|
|
xc_parse_pushIndex(idx);
|
|
|
|
}
|
|
|
|
|
2018-12-19 19:43:03 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-25 20:40:55 +01:00
|
|
|
static void bc_parse_pushJUMP(size_t idx)
|
2018-12-17 00:07:48 +01:00
|
|
|
{
|
2019-01-09 11:17:19 +01:00
|
|
|
xc_parse_pushInst_and_Index(BC_INST_JUMP, idx);
|
2018-12-17 00:07:48 +01:00
|
|
|
}
|
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static void bc_parse_pushJUMP_ZERO(size_t idx)
|
2018-12-17 00:07:48 +01:00
|
|
|
{
|
2019-01-09 11:17:19 +01:00
|
|
|
xc_parse_pushInst_and_Index(BC_INST_JUMP_ZERO, idx);
|
2018-12-17 00:07:48 +01:00
|
|
|
}
|
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_pushSTR(void)
|
2018-12-19 19:10:40 +01:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-12-25 22:32:41 +01:00
|
|
|
char *str = xstrdup(p->lex_strnumbuf.v);
|
2018-12-19 19:10:40 +01:00
|
|
|
|
2019-01-09 11:17:19 +01:00
|
|
|
xc_parse_pushInst_and_Index(XC_INST_STR, p->func->strs.len);
|
2018-12-21 16:22:26 +01:00
|
|
|
bc_vec_push(&p->func->strs, &str);
|
2018-12-19 19:10:40 +01:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
RETURN_STATUS(zxc_lex_next());
|
2018-12-19 19:10:40 +01:00
|
|
|
}
|
2018-12-22 01:34:10 +01:00
|
|
|
#define zbc_parse_pushSTR(...) (zbc_parse_pushSTR(__VA_ARGS__) COMMA_SUCCESS)
|
2018-12-19 19:43:03 +01:00
|
|
|
#endif
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_parse_pushNUM(void)
|
2018-12-19 19:43:03 +01:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-12-25 22:32:41 +01:00
|
|
|
char *num = xstrdup(p->lex_strnumbuf.v);
|
2018-12-21 16:22:26 +01:00
|
|
|
#if ENABLE_BC && ENABLE_DC
|
2018-12-21 22:16:17 +01:00
|
|
|
size_t idx = bc_vec_push(IS_BC ? &p->func->consts : &G.prog.consts, &num);
|
2018-12-21 16:22:26 +01:00
|
|
|
#elif ENABLE_BC
|
2018-12-21 22:16:17 +01:00
|
|
|
size_t idx = bc_vec_push(&p->func->consts, &num);
|
2018-12-21 16:22:26 +01:00
|
|
|
#else // DC
|
2018-12-21 22:16:17 +01:00
|
|
|
size_t idx = bc_vec_push(&G.prog.consts, &num);
|
2018-12-21 16:22:26 +01:00
|
|
|
#endif
|
2019-01-09 11:17:19 +01:00
|
|
|
xc_parse_pushInst_and_Index(XC_INST_NUM, idx);
|
2018-12-19 19:43:03 +01:00
|
|
|
}
|
2018-12-19 19:10:40 +01:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_parse_text_init(const char *text)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-27 18:23:58 +01:00
|
|
|
G.prs.func = xc_program_func(G.prs.fidx);
|
|
|
|
G.prs.lex_inbuf = text;
|
|
|
|
G.prs.lex = G.prs.lex_last = XC_LEX_INVALID;
|
|
|
|
RETURN_STATUS(zxc_lex_next());
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_parse_text_init(...) (zxc_parse_text_init(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-06 12:54:26 +01:00
|
|
|
// Called when parsing or execution detects a failure,
|
|
|
|
// resets execution structures.
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_program_reset(void)
|
2018-12-06 12:54:26 +01:00
|
|
|
{
|
|
|
|
BcFunc *f;
|
|
|
|
BcInstPtr *ip;
|
|
|
|
|
2018-12-19 12:35:27 +01:00
|
|
|
bc_vec_npop(&G.prog.exestack, G.prog.exestack.len - 1);
|
2018-12-06 12:54:26 +01:00
|
|
|
bc_vec_pop_all(&G.prog.results);
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
f = xc_program_func_BC_PROG_MAIN();
|
2018-12-19 12:35:27 +01:00
|
|
|
ip = bc_vec_top(&G.prog.exestack);
|
2018-12-21 23:01:26 +01:00
|
|
|
ip->inst_idx = f->code.len;
|
2018-12-06 12:54:26 +01:00
|
|
|
}
|
|
|
|
|
2018-12-22 06:00:25 +01:00
|
|
|
// Called when parsing code detects a failure,
|
2018-12-04 19:11:02 +01:00
|
|
|
// resets parsing structures.
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_parse_reset(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
if (p->fidx != BC_PROG_MAIN) {
|
2018-12-22 06:00:25 +01:00
|
|
|
bc_func_free(p->func);
|
|
|
|
bc_func_init(p->func);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 15:13:14 +01:00
|
|
|
p->fidx = BC_PROG_MAIN;
|
2018-12-27 18:23:58 +01:00
|
|
|
p->func = xc_program_func_BC_PROG_MAIN();
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-26 18:46:03 +01:00
|
|
|
p->lex_inbuf += strlen(p->lex_inbuf);
|
2018-12-25 22:32:41 +01:00
|
|
|
p->lex = XC_LEX_EOF;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-20 16:24:18 +01:00
|
|
|
IF_BC(bc_vec_pop_all(&p->exits);)
|
|
|
|
IF_BC(bc_vec_pop_all(&p->conds);)
|
|
|
|
IF_BC(bc_vec_pop_all(&p->ops);)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_reset();
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_parse_free(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-26 19:24:15 +01:00
|
|
|
IF_BC(bc_vec_free(&G.prs.exits);)
|
|
|
|
IF_BC(bc_vec_free(&G.prs.conds);)
|
|
|
|
IF_BC(bc_vec_free(&G.prs.ops);)
|
2018-12-26 18:32:43 +01:00
|
|
|
bc_vec_free(&G.prs.lex_strnumbuf);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_parse_create(size_t fidx)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
memset(p, 0, sizeof(BcParse));
|
|
|
|
|
2018-12-26 19:24:15 +01:00
|
|
|
bc_char_vec_init(&p->lex_strnumbuf);
|
2018-12-20 16:24:18 +01:00
|
|
|
IF_BC(bc_vec_init(&p->exits, sizeof(size_t), NULL);)
|
|
|
|
IF_BC(bc_vec_init(&p->conds, sizeof(size_t), NULL);)
|
|
|
|
IF_BC(bc_vec_init(&p->ops, sizeof(BcLexType), NULL);)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 19:35:40 +01:00
|
|
|
p->fidx = fidx;
|
2018-12-27 18:23:58 +01:00
|
|
|
p->func = xc_program_func(fidx);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_program_add_fn(void)
|
2018-12-20 20:34:09 +01:00
|
|
|
{
|
2018-12-21 00:10:26 +01:00
|
|
|
//size_t idx;
|
2018-12-20 20:34:09 +01:00
|
|
|
BcFunc f;
|
|
|
|
bc_func_init(&f);
|
2018-12-21 22:16:17 +01:00
|
|
|
//idx =
|
2018-12-20 20:34:09 +01:00
|
|
|
bc_vec_push(&G.prog.fns, &f);
|
2018-12-21 00:10:26 +01:00
|
|
|
//return idx;
|
2018-12-20 20:34:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#if ENABLE_BC
|
|
|
|
|
2018-12-19 19:43:03 +01:00
|
|
|
// Note: takes ownership of 'name' (must be malloced)
|
|
|
|
static size_t bc_program_addFunc(char *name)
|
|
|
|
{
|
|
|
|
size_t idx;
|
|
|
|
BcId entry, *entry_ptr;
|
|
|
|
int inserted;
|
|
|
|
|
|
|
|
entry.name = name;
|
|
|
|
entry.idx = G.prog.fns.len;
|
|
|
|
|
|
|
|
inserted = bc_map_insert(&G.prog.fn_map, &entry, &idx);
|
|
|
|
if (!inserted) free(name);
|
|
|
|
|
|
|
|
entry_ptr = bc_vec_item(&G.prog.fn_map, idx);
|
|
|
|
idx = entry_ptr->idx;
|
|
|
|
|
|
|
|
if (!inserted) {
|
2018-12-19 20:05:50 +01:00
|
|
|
// There is already a function with this name.
|
|
|
|
// It'll be redefined now, clear old definition.
|
2018-12-27 18:23:58 +01:00
|
|
|
BcFunc *func = xc_program_func(entry_ptr->idx);
|
2018-12-19 20:05:50 +01:00
|
|
|
bc_func_free(func);
|
|
|
|
bc_func_init(func);
|
2018-12-19 19:43:03 +01:00
|
|
|
} else {
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_add_fn();
|
2018-12-19 19:43:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
2018-12-24 23:22:40 +01:00
|
|
|
#define BC_PARSE_TOP_OP(p) (*(BcLexType*)bc_vec_top(&(p)->ops))
|
2018-12-05 21:15:46 +01:00
|
|
|
// We can calculate the conversion between tokens and exprs by subtracting the
|
|
|
|
// position of the first operator in the lex enum and adding the position of the
|
|
|
|
// first in the expr enum. Note: This only works for binary operators.
|
2018-12-24 14:14:23 +01:00
|
|
|
#define BC_TOKEN_2_INST(t) ((char) ((t) - XC_LEX_OP_POWER + XC_INST_POWER))
|
2018-12-05 21:15:46 +01:00
|
|
|
|
2019-01-08 19:29:35 +01:00
|
|
|
static BC_STATUS zbc_parse_expr(uint8_t flags);
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_expr(...) (zbc_parse_expr(__VA_ARGS__) COMMA_SUCCESS)
|
2018-12-17 16:54:37 +01:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_stmt_possibly_auto(bool auto_allowed);
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_stmt_possibly_auto(...) (zbc_parse_stmt_possibly_auto(__VA_ARGS__) COMMA_SUCCESS)
|
2018-12-16 16:03:03 +01:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_stmt(void)
|
2018-12-16 16:03:03 +01:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
RETURN_STATUS(zbc_parse_stmt_possibly_auto(false));
|
2018-12-16 16:03:03 +01:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_stmt(...) (zbc_parse_stmt(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_stmt_allow_NLINE_before(const char *after_X)
|
2018-12-16 20:41:32 +01:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-12-17 00:07:48 +01:00
|
|
|
// "if(cond)<newline>stmt" is accepted too, but not 2+ newlines.
|
|
|
|
// Same for "else", "while()", "for()".
|
2018-12-25 20:40:55 +01:00
|
|
|
BcStatus s = zbc_lex_next_and_skip_NLINE();
|
2018-12-17 00:07:48 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex == XC_LEX_NLINE)
|
2018-12-16 20:41:32 +01:00
|
|
|
RETURN_STATUS(bc_error_fmt("no statement after '%s'", after_X));
|
2018-12-17 00:07:48 +01:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
RETURN_STATUS(zbc_parse_stmt());
|
2018-12-16 20:41:32 +01:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_stmt_allow_NLINE_before(...) (zbc_parse_stmt_allow_NLINE_before(__VA_ARGS__) COMMA_SUCCESS)
|
2018-12-16 20:41:32 +01:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static void bc_parse_operator(BcLexType type, size_t start, size_t *nexprs)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-12-30 15:56:36 +01:00
|
|
|
char l, r = bc_operation_PREC(type - XC_LEX_1st_op);
|
|
|
|
bool left = bc_operation_LEFT(type - XC_LEX_1st_op);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
while (p->ops.len > start) {
|
2018-12-14 23:12:48 +01:00
|
|
|
BcLexType t = BC_PARSE_TOP_OP(p);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (t == BC_LEX_LPAREN) break;
|
|
|
|
|
2018-12-30 15:56:36 +01:00
|
|
|
l = bc_operation_PREC(t - XC_LEX_1st_op);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (l >= r && (l != r || !left)) break;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(BC_TOKEN_2_INST(t));
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_vec_pop(&p->ops);
|
2018-12-24 14:14:23 +01:00
|
|
|
*nexprs -= (t != BC_LEX_OP_BOOL_NOT && t != XC_LEX_NEG);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
bc_vec_push(&p->ops, &type);
|
|
|
|
}
|
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_rightParen(size_t ops_bgn, size_t *nexs)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcLexType top;
|
|
|
|
|
2018-12-04 20:05:28 +01:00
|
|
|
if (p->ops.len <= ops_bgn)
|
2018-12-12 13:58:55 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_expression());
|
2018-11-03 11:00:21 -06:00
|
|
|
top = BC_PARSE_TOP_OP(p);
|
|
|
|
|
|
|
|
while (top != BC_LEX_LPAREN) {
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(BC_TOKEN_2_INST(top));
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
bc_vec_pop(&p->ops);
|
2018-12-24 14:14:23 +01:00
|
|
|
*nexs -= (top != BC_LEX_OP_BOOL_NOT && top != XC_LEX_NEG);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-04 20:05:28 +01:00
|
|
|
if (p->ops.len <= ops_bgn)
|
2018-12-12 13:58:55 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_expression());
|
2018-11-03 11:00:21 -06:00
|
|
|
top = BC_PARSE_TOP_OP(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
bc_vec_pop(&p->ops);
|
|
|
|
|
2018-12-25 16:39:01 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_rightParen(...) (zbc_parse_rightParen(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_params(uint8_t flags)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
size_t nparams;
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
dbg_lex("%s:%d p->lex:%d", __func__, __LINE__, p->lex);
|
2018-12-17 16:54:37 +01:00
|
|
|
flags = (flags & ~(BC_PARSE_PRINT | BC_PARSE_REL)) | BC_PARSE_ARRAY;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 16:08:46 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-17 16:54:37 +01:00
|
|
|
nparams = 0;
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_RPAREN) {
|
2018-12-17 16:54:37 +01:00
|
|
|
for (;;) {
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_expr(flags);
|
2018-12-17 16:54:37 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
nparams++;
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_COMMA) {
|
|
|
|
if (p->lex == BC_LEX_RPAREN)
|
2018-12-17 16:54:37 +01:00
|
|
|
break;
|
|
|
|
RETURN_STATUS(bc_error_bad_token());
|
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 16:08:46 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-09 11:17:19 +01:00
|
|
|
xc_parse_pushInst_and_Index(BC_INST_CALL, nparams);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-12 16:08:46 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_params(...) (zbc_parse_params(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 13:19:44 +01:00
|
|
|
// Note: takes ownership of 'name' (must be malloced)
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_call(char *name, uint8_t flags)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
BcId entry, *entry_ptr;
|
|
|
|
size_t idx;
|
|
|
|
|
|
|
|
entry.name = name;
|
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_params(flags);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_RPAREN) {
|
2018-12-05 16:03:46 +01:00
|
|
|
s = bc_error_bad_token();
|
2018-11-03 11:00:21 -06:00
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2018-12-19 13:55:53 +01:00
|
|
|
idx = bc_map_find_exact(&G.prog.fn_map, &entry);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (idx == BC_VEC_INVALID_IDX) {
|
2018-12-19 13:55:53 +01:00
|
|
|
// No such function exists, create an empty one
|
2018-12-19 14:57:23 +01:00
|
|
|
bc_program_addFunc(name);
|
2018-12-19 13:55:53 +01:00
|
|
|
idx = bc_map_find_exact(&G.prog.fn_map, &entry);
|
2018-12-12 16:08:46 +01:00
|
|
|
} else
|
2018-11-03 11:00:21 -06:00
|
|
|
free(name);
|
|
|
|
|
2018-12-02 18:26:38 +01:00
|
|
|
entry_ptr = bc_vec_item(&G.prog.fn_map, idx);
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_pushIndex(entry_ptr->idx);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
RETURN_STATUS(zxc_lex_next());
|
2018-12-18 16:24:07 +01:00
|
|
|
err:
|
2018-11-03 11:00:21 -06:00
|
|
|
free(name);
|
2018-12-12 16:08:46 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_call(...) (zbc_parse_call(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_name(BcInst *type, uint8_t flags)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
char *name;
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
name = xstrdup(p->lex_strnumbuf.v);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex == BC_LEX_LBRACKET) {
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex == BC_LEX_RBRACKET) {
|
2018-11-03 11:00:21 -06:00
|
|
|
if (!(flags & BC_PARSE_ARRAY)) {
|
2018-12-05 16:03:46 +01:00
|
|
|
s = bc_error_bad_expression();
|
2018-11-03 11:00:21 -06:00
|
|
|
goto err;
|
|
|
|
}
|
2018-12-24 04:26:07 +01:00
|
|
|
*type = XC_INST_ARRAY;
|
2018-12-12 16:08:46 +01:00
|
|
|
} else {
|
2018-12-24 04:26:07 +01:00
|
|
|
*type = XC_INST_ARRAY_ELEM;
|
2018-11-03 11:00:21 -06:00
|
|
|
flags &= ~(BC_PARSE_PRINT | BC_PARSE_REL);
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_expr(flags);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(*type);
|
|
|
|
xc_parse_pushName(name);
|
2018-12-16 20:32:58 +01:00
|
|
|
free(name);
|
2018-12-25 22:32:41 +01:00
|
|
|
} else if (p->lex == BC_LEX_LPAREN) {
|
2018-11-03 11:00:21 -06:00
|
|
|
if (flags & BC_PARSE_NOCALL) {
|
2018-12-05 16:03:46 +01:00
|
|
|
s = bc_error_bad_token();
|
2018-11-03 11:00:21 -06:00
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
*type = BC_INST_CALL;
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_call(name, flags);
|
2018-12-12 16:08:46 +01:00
|
|
|
} else {
|
2018-12-24 04:26:07 +01:00
|
|
|
*type = XC_INST_VAR;
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(XC_INST_VAR);
|
|
|
|
xc_parse_pushName(name);
|
2018-12-16 20:32:58 +01:00
|
|
|
free(name);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-12 16:08:46 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-12-18 16:24:07 +01:00
|
|
|
err:
|
2018-11-03 11:00:21 -06:00
|
|
|
free(name);
|
2018-12-12 16:08:46 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_name(...) (zbc_parse_name(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_read(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 14:54:38 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_LPAREN) RETURN_STATUS(bc_error_bad_token());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 14:54:38 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_RPAREN) RETURN_STATUS(bc_error_bad_token());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(XC_INST_READ);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 17:07:51 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_read(...) (zbc_parse_read(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_builtin(BcLexType type, uint8_t flags, BcInst *prev)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 16:08:46 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_LPAREN) RETURN_STATUS(bc_error_bad_token());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
flags = (flags & ~(BC_PARSE_PRINT | BC_PARSE_REL)) | BC_PARSE_ARRAY;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 16:08:46 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_expr(flags);
|
2018-12-12 16:08:46 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_RPAREN) RETURN_STATUS(bc_error_bad_token());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
*prev = (type == BC_LEX_KEY_LENGTH) ? XC_INST_LENGTH : XC_INST_SQRT;
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(*prev);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 17:07:51 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_builtin(...) (zbc_parse_builtin(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_scale(BcInst *type, uint8_t flags)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 16:08:46 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_LPAREN) {
|
2018-12-24 04:26:07 +01:00
|
|
|
*type = XC_INST_SCALE;
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(XC_INST_SCALE);
|
2018-12-12 16:08:46 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
*type = XC_INST_SCALE_FUNC;
|
2018-11-03 11:00:21 -06:00
|
|
|
flags &= ~(BC_PARSE_PRINT | BC_PARSE_REL);
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 16:08:46 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_expr(flags);
|
2018-12-12 16:08:46 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_RPAREN)
|
2018-12-12 16:08:46 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_token());
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(XC_INST_SCALE_FUNC);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
RETURN_STATUS(zxc_lex_next());
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_scale(...) (zbc_parse_scale(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2019-01-01 21:50:14 +01:00
|
|
|
static BC_STATUS zbc_parse_incdec(BcInst *prev, size_t *nexs, uint8_t flags)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
BcLexType type;
|
|
|
|
char inst;
|
|
|
|
BcInst etype = *prev;
|
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
if (etype == XC_INST_VAR || etype == XC_INST_ARRAY_ELEM
|
|
|
|
|| etype == XC_INST_SCALE || etype == BC_INST_LAST
|
|
|
|
|| etype == XC_INST_IBASE || etype == XC_INST_OBASE
|
2018-12-18 16:24:07 +01:00
|
|
|
) {
|
2018-12-25 22:32:41 +01:00
|
|
|
*prev = inst = BC_INST_INC_POST + (p->lex != BC_LEX_OP_INC);
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(inst);
|
|
|
|
s = zxc_lex_next();
|
2018-12-18 16:24:07 +01:00
|
|
|
} else {
|
2018-12-25 22:32:41 +01:00
|
|
|
*prev = inst = BC_INST_INC_PRE + (p->lex != BC_LEX_OP_INC);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 16:08:46 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
type = p->lex;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
// Because we parse the next part of the expression
|
|
|
|
// right here, we need to increment this.
|
2019-01-01 21:50:14 +01:00
|
|
|
*nexs = *nexs + 1;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
switch (type) {
|
2018-12-26 19:24:15 +01:00
|
|
|
case XC_LEX_NAME:
|
|
|
|
s = zbc_parse_name(prev, flags | BC_PARSE_NOCALL);
|
|
|
|
break;
|
|
|
|
case BC_LEX_KEY_IBASE:
|
|
|
|
case BC_LEX_KEY_LAST:
|
|
|
|
case BC_LEX_KEY_OBASE:
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(type - BC_LEX_KEY_IBASE + XC_INST_IBASE);
|
|
|
|
s = zxc_lex_next();
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case BC_LEX_KEY_SCALE:
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-26 19:24:15 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
if (p->lex == BC_LEX_LPAREN)
|
2018-12-05 16:03:46 +01:00
|
|
|
s = bc_error_bad_token();
|
2018-12-26 19:24:15 +01:00
|
|
|
else
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(XC_INST_SCALE);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
s = bc_error_bad_token();
|
|
|
|
break;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
if (!s) xc_parse_push(inst);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-12 16:08:46 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_incdec(...) (zbc_parse_incdec(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2019-01-01 21:50:14 +01:00
|
|
|
static int bc_parse_inst_isLeaf(BcInst p)
|
2018-12-24 23:22:40 +01:00
|
|
|
{
|
|
|
|
return (p >= XC_INST_NUM && p <= XC_INST_SQRT)
|
|
|
|
|| p == BC_INST_INC_POST
|
|
|
|
|| p == BC_INST_DEC_POST
|
|
|
|
;
|
|
|
|
}
|
2019-01-01 21:50:14 +01:00
|
|
|
#define BC_PARSE_LEAF(prev, bin_last, rparen) \
|
|
|
|
(!(bin_last) && ((rparen) || bc_parse_inst_isLeaf(prev)))
|
2018-12-24 23:22:40 +01:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_minus(BcInst *prev, size_t ops_bgn,
|
2019-01-01 21:50:14 +01:00
|
|
|
bool rparen, bool bin_last, size_t *nexprs)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
BcLexType type;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 14:54:38 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2019-01-01 21:50:14 +01:00
|
|
|
type = BC_PARSE_LEAF(*prev, bin_last, rparen) ? XC_LEX_OP_MINUS : XC_LEX_NEG;
|
2018-12-14 23:32:51 +01:00
|
|
|
*prev = BC_TOKEN_2_INST(type);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
// We can just push onto the op stack because this is the largest
|
|
|
|
// precedence operator that gets pushed. Inc/dec does not.
|
2018-12-24 14:14:23 +01:00
|
|
|
if (type != XC_LEX_OP_MINUS)
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_vec_push(&p->ops, &type);
|
|
|
|
else
|
2018-12-25 20:40:55 +01:00
|
|
|
bc_parse_operator(type, ops_bgn, nexprs);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-12 14:54:38 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_minus(...) (zbc_parse_minus(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_print(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
BcLexType type;
|
|
|
|
|
2018-12-16 21:08:30 +01:00
|
|
|
for (;;) {
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-16 21:08:30 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
type = p->lex;
|
2018-12-24 15:05:49 +01:00
|
|
|
if (type == XC_LEX_STR) {
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_pushSTR();
|
2018-12-08 23:36:28 +01:00
|
|
|
} else {
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_expr(0);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-16 21:08:30 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(XC_INST_PRINT_POP);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_COMMA)
|
2018-12-16 21:08:30 +01:00
|
|
|
break;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-16 21:08:30 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_print(...) (zbc_parse_print(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_return(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
BcLexType t;
|
|
|
|
|
2018-12-16 16:03:03 +01:00
|
|
|
dbg_lex_enter("%s:%d entered", __func__, __LINE__);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 15:19:54 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
t = p->lex;
|
2019-01-03 23:34:36 +01:00
|
|
|
if (t == XC_LEX_NLINE || t == BC_LEX_SCOLON || t == BC_LEX_RBRACE)
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(BC_INST_RET0);
|
2018-11-03 11:00:21 -06:00
|
|
|
else {
|
2019-01-04 13:58:46 +01:00
|
|
|
//TODO: if (p->func->voidfunc) ERROR
|
2019-01-03 23:34:36 +01:00
|
|
|
s = zbc_parse_expr(0);
|
2018-12-12 15:19:54 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2019-01-03 23:34:36 +01:00
|
|
|
if (t != BC_LEX_LPAREN // "return EXPR", no ()
|
|
|
|
|| p->lex_last != BC_LEX_RPAREN // example: "return (a) + b"
|
|
|
|
) {
|
2018-12-24 18:11:41 +01:00
|
|
|
s = zbc_POSIX_requires("parentheses around return expressions");
|
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(XC_INST_RET);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-16 16:03:03 +01:00
|
|
|
dbg_lex_done("%s:%d done", __func__, __LINE__);
|
2018-12-12 15:19:54 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_return(...) (zbc_parse_return(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static void rewrite_label_to_current(size_t idx)
|
2018-12-17 00:07:48 +01:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-12-17 00:07:48 +01:00
|
|
|
size_t *label = bc_vec_item(&p->func->labels, idx);
|
|
|
|
*label = p->func->code.len;
|
|
|
|
}
|
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_if(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
2018-12-16 21:40:54 +01:00
|
|
|
size_t ip_idx;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-14 17:51:17 +01:00
|
|
|
dbg_lex_enter("%s:%d entered", __func__, __LINE__);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 15:19:54 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_LPAREN) RETURN_STATUS(bc_error_bad_token());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 15:19:54 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_expr(BC_PARSE_REL);
|
2018-12-12 15:19:54 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_RPAREN) RETURN_STATUS(bc_error_bad_token());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-21 22:16:17 +01:00
|
|
|
// Encode "if zero, jump to ..."
|
|
|
|
// Pushed value (destination of the jump) is uninitialized,
|
|
|
|
// will be rewritten to be address of "end of if()" or of "else".
|
|
|
|
ip_idx = bc_vec_push(&p->func->labels, &ip_idx);
|
2018-12-25 20:40:55 +01:00
|
|
|
bc_parse_pushJUMP_ZERO(ip_idx);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_stmt_allow_NLINE_before(STRING_if);
|
2018-12-16 16:03:03 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-16 19:21:57 +01:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
dbg_lex("%s:%d in if after stmt: p->lex:%d", __func__, __LINE__, p->lex);
|
|
|
|
if (p->lex == BC_LEX_KEY_ELSE) {
|
2018-12-16 21:40:54 +01:00
|
|
|
size_t ip2_idx;
|
2018-12-16 21:21:27 +01:00
|
|
|
|
2018-12-21 22:16:17 +01:00
|
|
|
// Encode "after then_stmt, jump to end of if()"
|
|
|
|
ip2_idx = bc_vec_push(&p->func->labels, &ip2_idx);
|
|
|
|
dbg_lex("%s:%d after if() then_stmt: BC_INST_JUMP to %zd", __func__, __LINE__, ip2_idx);
|
2018-12-25 20:40:55 +01:00
|
|
|
bc_parse_pushJUMP(ip2_idx);
|
2018-12-16 19:21:57 +01:00
|
|
|
|
2018-12-18 02:23:53 +01:00
|
|
|
dbg_lex("%s:%d rewriting 'if_zero' label to jump to 'else'-> %zd", __func__, __LINE__, p->func->code.len);
|
2018-12-25 20:40:55 +01:00
|
|
|
rewrite_label_to_current(ip_idx);
|
2018-12-16 19:21:57 +01:00
|
|
|
|
2018-12-16 21:40:54 +01:00
|
|
|
ip_idx = ip2_idx;
|
2018-12-16 19:21:57 +01:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_stmt_allow_NLINE_before(STRING_else);
|
2018-12-16 16:03:03 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-18 02:23:53 +01:00
|
|
|
dbg_lex("%s:%d rewriting label to jump after 'if' body-> %zd", __func__, __LINE__, p->func->code.len);
|
2018-12-25 20:40:55 +01:00
|
|
|
rewrite_label_to_current(ip_idx);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-14 23:00:24 +01:00
|
|
|
dbg_lex_done("%s:%d done", __func__, __LINE__);
|
2018-12-16 16:03:03 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_if(...) (zbc_parse_if(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_while(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
2018-12-16 23:02:22 +01:00
|
|
|
size_t cond_idx;
|
2018-12-16 23:35:04 +01:00
|
|
|
size_t ip_idx;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 15:19:54 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_LPAREN) RETURN_STATUS(bc_error_bad_token());
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 15:19:54 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-21 22:16:17 +01:00
|
|
|
cond_idx = bc_vec_push(&p->func->labels, &p->func->code.len);
|
2018-12-16 23:35:04 +01:00
|
|
|
ip_idx = cond_idx + 1;
|
2018-12-16 23:02:22 +01:00
|
|
|
bc_vec_push(&p->conds, &cond_idx);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-16 23:35:04 +01:00
|
|
|
bc_vec_push(&p->exits, &ip_idx);
|
|
|
|
bc_vec_push(&p->func->labels, &ip_idx);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_expr(BC_PARSE_REL);
|
2018-12-12 15:19:54 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_RPAREN) RETURN_STATUS(bc_error_bad_token());
|
2018-12-16 17:30:35 +01:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
bc_parse_pushJUMP_ZERO(ip_idx);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_stmt_allow_NLINE_before(STRING_while);
|
2018-12-16 16:03:03 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
|
2018-12-18 02:23:53 +01:00
|
|
|
dbg_lex("%s:%d BC_INST_JUMP to %zd", __func__, __LINE__, cond_idx);
|
2018-12-25 20:40:55 +01:00
|
|
|
bc_parse_pushJUMP(cond_idx);
|
2018-12-16 16:03:03 +01:00
|
|
|
|
2018-12-18 02:23:53 +01:00
|
|
|
dbg_lex("%s:%d rewriting label-> %zd", __func__, __LINE__, p->func->code.len);
|
2018-12-25 20:40:55 +01:00
|
|
|
rewrite_label_to_current(ip_idx);
|
2018-12-16 16:03:03 +01:00
|
|
|
|
|
|
|
bc_vec_pop(&p->exits);
|
|
|
|
bc_vec_pop(&p->conds);
|
|
|
|
|
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_while(...) (zbc_parse_while(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_for(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
size_t cond_idx, exit_idx, body_idx, update_idx;
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
dbg_lex("%s:%d p->lex:%d", __func__, __LINE__, p->lex);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 15:19:54 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_LPAREN) RETURN_STATUS(bc_error_bad_token());
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 15:19:54 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_SCOLON) {
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_expr(0);
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(XC_INST_POP);
|
2018-12-21 20:29:34 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
} else {
|
2018-12-24 18:11:41 +01:00
|
|
|
s = zbc_POSIX_does_not_allow_empty_X_expression_in_for("init");
|
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-21 20:29:34 +01:00
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_SCOLON) RETURN_STATUS(bc_error_bad_token());
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 15:19:54 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-21 22:16:17 +01:00
|
|
|
cond_idx = bc_vec_push(&p->func->labels, &p->func->code.len);
|
2018-11-03 11:00:21 -06:00
|
|
|
update_idx = cond_idx + 1;
|
|
|
|
body_idx = update_idx + 1;
|
|
|
|
exit_idx = body_idx + 1;
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_SCOLON)
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_expr(BC_PARSE_REL);
|
2018-12-21 00:35:22 +01:00
|
|
|
else {
|
2018-12-27 18:23:58 +01:00
|
|
|
// Set this for the next call to xc_parse_pushNUM().
|
2018-12-21 00:35:22 +01:00
|
|
|
// This is safe to set because the current token is a semicolon,
|
|
|
|
// which has no string requirement.
|
2018-12-25 22:32:41 +01:00
|
|
|
bc_vec_string(&p->lex_strnumbuf, 1, "1");
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_pushNUM();
|
2018-12-24 18:11:41 +01:00
|
|
|
s = zbc_POSIX_does_not_allow_empty_X_expression_in_for("condition");
|
2018-12-21 00:35:22 +01:00
|
|
|
}
|
2018-12-12 15:19:54 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-21 20:29:34 +01:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_SCOLON) RETURN_STATUS(bc_error_bad_token());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 15:19:54 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
bc_parse_pushJUMP_ZERO(exit_idx);
|
|
|
|
bc_parse_pushJUMP(body_idx);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
bc_vec_push(&p->conds, &update_idx);
|
|
|
|
bc_vec_push(&p->func->labels, &p->func->code.len);
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_RPAREN) {
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_expr(0);
|
2018-12-21 20:29:34 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_RPAREN) RETURN_STATUS(bc_error_bad_token());
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(XC_INST_POP);
|
2018-12-21 20:29:34 +01:00
|
|
|
} else {
|
2018-12-24 18:11:41 +01:00
|
|
|
s = zbc_POSIX_does_not_allow_empty_X_expression_in_for("update");
|
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-21 20:29:34 +01:00
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
bc_parse_pushJUMP(cond_idx);
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_vec_push(&p->func->labels, &p->func->code.len);
|
|
|
|
|
2018-12-16 23:35:04 +01:00
|
|
|
bc_vec_push(&p->exits, &exit_idx);
|
|
|
|
bc_vec_push(&p->func->labels, &exit_idx);
|
2018-12-16 17:30:35 +01:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_stmt_allow_NLINE_before(STRING_for);
|
2018-12-16 16:03:03 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
|
2018-12-18 02:23:53 +01:00
|
|
|
dbg_lex("%s:%d BC_INST_JUMP to %zd", __func__, __LINE__, update_idx);
|
2018-12-25 20:40:55 +01:00
|
|
|
bc_parse_pushJUMP(update_idx);
|
2018-12-16 16:03:03 +01:00
|
|
|
|
2018-12-18 02:23:53 +01:00
|
|
|
dbg_lex("%s:%d rewriting label-> %zd", __func__, __LINE__, p->func->code.len);
|
2018-12-25 20:40:55 +01:00
|
|
|
rewrite_label_to_current(exit_idx);
|
2018-12-16 16:03:03 +01:00
|
|
|
|
|
|
|
bc_vec_pop(&p->exits);
|
|
|
|
bc_vec_pop(&p->conds);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-12 15:19:54 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_for(...) (zbc_parse_for(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_break_or_continue(BcLexType type)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (type == BC_LEX_KEY_BREAK) {
|
2018-12-16 23:18:28 +01:00
|
|
|
if (p->exits.len == 0) // none of the enclosing blocks is a loop
|
|
|
|
RETURN_STATUS(bc_error_bad_token());
|
2018-12-16 23:24:25 +01:00
|
|
|
i = *(size_t*)bc_vec_top(&p->exits);
|
2018-12-16 23:18:28 +01:00
|
|
|
} else {
|
2018-12-16 23:24:25 +01:00
|
|
|
i = *(size_t*)bc_vec_top(&p->conds);
|
2018-12-16 23:18:28 +01:00
|
|
|
}
|
2018-12-25 20:40:55 +01:00
|
|
|
bc_parse_pushJUMP(i);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
RETURN_STATUS(zxc_lex_next());
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_break_or_continue(...) (zbc_parse_break_or_continue(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
static BC_STATUS zbc_func_insert(BcFunc *f, char *name, BcType type)
|
2018-12-24 05:00:36 +01:00
|
|
|
{
|
|
|
|
BcId *autoid;
|
|
|
|
BcId a;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
autoid = (void*)f->autos.v;
|
|
|
|
for (i = 0; i < f->autos.len; i++, autoid++) {
|
2019-01-01 21:50:14 +01:00
|
|
|
if (strcmp(name, autoid->name) == 0
|
2019-01-25 14:24:03 +01:00
|
|
|
&& type == (BcType) autoid->idx
|
2019-01-01 21:50:14 +01:00
|
|
|
) {
|
2018-12-25 21:28:25 +01:00
|
|
|
RETURN_STATUS(bc_error("duplicate function parameter or auto name"));
|
2019-01-01 21:50:14 +01:00
|
|
|
}
|
2018-12-24 05:00:36 +01:00
|
|
|
}
|
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
a.idx = type;
|
2018-12-24 05:00:36 +01:00
|
|
|
a.name = name;
|
|
|
|
|
|
|
|
bc_vec_push(&f->autos, &a);
|
|
|
|
|
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
|
|
|
}
|
|
|
|
#define zbc_func_insert(...) (zbc_func_insert(__VA_ARGS__) COMMA_SUCCESS)
|
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_funcdef(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
2019-01-25 14:24:03 +01:00
|
|
|
bool comma, voidfunc;
|
2018-11-03 11:00:21 -06:00
|
|
|
char *name;
|
|
|
|
|
2018-12-18 02:23:53 +01:00
|
|
|
dbg_lex_enter("%s:%d entered", __func__, __LINE__);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 14:54:38 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != XC_LEX_NAME)
|
2019-01-01 21:50:14 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_function_definition());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2019-01-04 13:58:46 +01:00
|
|
|
// To be maximally both POSIX and GNU-compatible,
|
|
|
|
// "void" is not treated as a normal keyword:
|
|
|
|
// you can have variable named "void", and even a function
|
|
|
|
// named "void": "define void() { return 6; }" is ok.
|
|
|
|
// _Only_ "define void f() ..." syntax treats "void"
|
|
|
|
// specially.
|
|
|
|
voidfunc = (strcmp(p->lex_strnumbuf.v, "void") == 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 14:54:38 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2019-01-04 13:58:46 +01:00
|
|
|
|
|
|
|
voidfunc = (voidfunc && p->lex == XC_LEX_NAME);
|
|
|
|
if (voidfunc) {
|
|
|
|
s = zxc_lex_next();
|
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
}
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_LPAREN)
|
2019-01-01 21:50:14 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_function_definition());
|
2019-01-04 13:58:46 +01:00
|
|
|
|
|
|
|
p->fidx = bc_program_addFunc(xstrdup(p->lex_strnumbuf.v));
|
|
|
|
p->func = xc_program_func(p->fidx);
|
|
|
|
p->func->voidfunc = voidfunc;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 14:54:38 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2019-01-04 13:58:46 +01:00
|
|
|
comma = false;
|
2018-12-25 22:32:41 +01:00
|
|
|
while (p->lex != BC_LEX_RPAREN) {
|
2019-01-25 14:24:03 +01:00
|
|
|
BcType t = BC_TYPE_VAR;
|
|
|
|
|
|
|
|
if (p->lex == XC_LEX_OP_MULTIPLY) {
|
|
|
|
t = BC_TYPE_REF;
|
|
|
|
s = zxc_lex_next();
|
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
s = zbc_POSIX_does_not_allow("references");
|
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
}
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != XC_LEX_NAME)
|
2019-01-01 21:50:14 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_function_definition());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
++p->func->nparams;
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
name = xstrdup(p->lex_strnumbuf.v);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
if (p->lex == BC_LEX_LBRACKET) {
|
|
|
|
if (t == BC_TYPE_VAR) t = BC_TYPE_ARRAY;
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_RBRACKET) {
|
2019-01-01 21:50:14 +01:00
|
|
|
s = bc_error_bad_function_definition();
|
2018-11-03 11:00:21 -06:00
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
}
|
2019-01-25 14:24:03 +01:00
|
|
|
else if (t == BC_TYPE_REF) {
|
|
|
|
s = bc_error_at("vars can't be references");
|
|
|
|
goto err;
|
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
comma = p->lex == BC_LEX_COMMA;
|
2018-11-03 11:00:21 -06:00
|
|
|
if (comma) {
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
}
|
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
s = zbc_func_insert(p->func, name, t);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
}
|
|
|
|
|
2019-01-01 21:50:14 +01:00
|
|
|
if (comma) RETURN_STATUS(bc_error_bad_function_definition());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 14:54:38 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_LBRACE) {
|
2018-12-24 18:11:41 +01:00
|
|
|
s = zbc_POSIX_requires("the left brace be on the same line as the function header");
|
2018-12-24 18:28:56 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-16 17:30:35 +01:00
|
|
|
// Prevent "define z()<newline>" from being interpreted as function with empty stmt as body
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_lex_skip_if_at_NLINE();
|
2018-12-16 17:30:35 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2019-01-04 13:58:46 +01:00
|
|
|
// GNU bc requires a {} block even if function body has single stmt, enforce this
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_LBRACE)
|
2018-12-17 00:07:48 +01:00
|
|
|
RETURN_STATUS(bc_error("function { body } expected"));
|
2018-12-16 17:06:07 +01:00
|
|
|
|
|
|
|
p->in_funcdef++; // to determine whether "return" stmt is allowed, and such
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_stmt_possibly_auto(true);
|
2018-12-16 17:06:07 +01:00
|
|
|
p->in_funcdef--;
|
2018-12-16 16:03:03 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(BC_INST_RET0);
|
2018-12-19 15:13:14 +01:00
|
|
|
|
|
|
|
// Subsequent code generation is into main program
|
|
|
|
p->fidx = BC_PROG_MAIN;
|
2018-12-27 18:23:58 +01:00
|
|
|
p->func = xc_program_func_BC_PROG_MAIN();
|
2018-12-16 16:03:03 +01:00
|
|
|
|
2018-12-18 02:23:53 +01:00
|
|
|
dbg_lex_done("%s:%d done", __func__, __LINE__);
|
2018-12-12 14:54:38 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-12-18 16:24:07 +01:00
|
|
|
err:
|
2018-12-18 02:23:53 +01:00
|
|
|
dbg_lex_done("%s:%d done (error)", __func__, __LINE__);
|
2018-11-03 11:00:21 -06:00
|
|
|
free(name);
|
2018-12-12 14:54:38 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_funcdef(...) (zbc_parse_funcdef(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_auto(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
char *name;
|
|
|
|
|
2018-12-16 16:03:03 +01:00
|
|
|
dbg_lex_enter("%s:%d entered", __func__, __LINE__);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 14:54:38 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 16:39:01 +01:00
|
|
|
for (;;) {
|
2019-01-25 14:24:03 +01:00
|
|
|
BcType t;
|
2018-12-25 16:39:01 +01:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != XC_LEX_NAME)
|
2019-01-01 21:50:14 +01:00
|
|
|
RETURN_STATUS(bc_error_at("bad 'auto' syntax"));
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
name = xstrdup(p->lex_strnumbuf.v);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
t = BC_TYPE_VAR;
|
|
|
|
if (p->lex == BC_LEX_LBRACKET) {
|
|
|
|
t = BC_TYPE_ARRAY;
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_RBRACKET) {
|
2019-01-01 21:50:14 +01:00
|
|
|
s = bc_error_at("bad 'auto' syntax");
|
2018-11-03 11:00:21 -06:00
|
|
|
goto err;
|
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
}
|
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
s = zbc_func_insert(p->func, name, t);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex == XC_LEX_NLINE
|
|
|
|
|| p->lex == BC_LEX_SCOLON
|
|
|
|
//|| p->lex == BC_LEX_RBRACE // allow "define f() {auto a}"
|
2018-12-25 16:39:01 +01:00
|
|
|
) {
|
|
|
|
break;
|
|
|
|
}
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != BC_LEX_COMMA)
|
2019-01-01 21:50:14 +01:00
|
|
|
RETURN_STATUS(bc_error_at("bad 'auto' syntax"));
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next(); // skip comma
|
2018-12-25 16:39:01 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-16 16:03:03 +01:00
|
|
|
dbg_lex_done("%s:%d done", __func__, __LINE__);
|
2018-12-25 16:39:01 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-12-18 16:24:07 +01:00
|
|
|
err:
|
2018-11-03 11:00:21 -06:00
|
|
|
free(name);
|
2018-12-16 16:03:03 +01:00
|
|
|
dbg_lex_done("%s:%d done (ERROR)", __func__, __LINE__);
|
2018-12-12 14:54:38 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_auto(...) (zbc_parse_auto(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-16 16:03:03 +01:00
|
|
|
#undef zbc_parse_stmt_possibly_auto
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_stmt_possibly_auto(bool auto_allowed)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s = BC_STATUS_SUCCESS;
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
dbg_lex_enter("%s:%d entered, p->lex:%d", __func__, __LINE__, p->lex);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex == XC_LEX_NLINE) {
|
2018-12-24 15:05:49 +01:00
|
|
|
dbg_lex_done("%s:%d done (seen XC_LEX_NLINE)", __func__, __LINE__);
|
2019-01-08 18:08:48 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex == BC_LEX_SCOLON) {
|
2018-12-16 16:03:03 +01:00
|
|
|
dbg_lex_done("%s:%d done (seen BC_LEX_SCOLON)", __func__, __LINE__);
|
2019-01-08 18:08:48 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex == BC_LEX_LBRACE) {
|
2018-12-16 16:03:03 +01:00
|
|
|
dbg_lex("%s:%d BC_LEX_LBRACE: (auto_allowed:%d)", __func__, __LINE__, auto_allowed);
|
|
|
|
do {
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 15:19:54 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
} while (p->lex == XC_LEX_NLINE);
|
|
|
|
if (auto_allowed && p->lex == BC_LEX_KEY_AUTO) {
|
2018-12-16 16:03:03 +01:00
|
|
|
dbg_lex("%s:%d calling zbc_parse_auto()", __func__, __LINE__);
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_auto();
|
2018-12-16 16:03:03 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
}
|
2018-12-25 22:32:41 +01:00
|
|
|
while (p->lex != BC_LEX_RBRACE) {
|
2018-12-16 16:03:03 +01:00
|
|
|
dbg_lex("%s:%d block parsing loop", __func__, __LINE__);
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_stmt();
|
2018-12-16 16:03:03 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2019-01-08 18:08:48 +01:00
|
|
|
// Check that next token is a correct stmt delimiter -
|
|
|
|
// disallows "print 1 print 2" and such.
|
|
|
|
if (p->lex == BC_LEX_RBRACE)
|
|
|
|
break;
|
|
|
|
if (p->lex != BC_LEX_SCOLON
|
|
|
|
&& p->lex != XC_LEX_NLINE
|
|
|
|
) {
|
|
|
|
RETURN_STATUS(bc_error_at("bad statement terminator"));
|
|
|
|
}
|
|
|
|
s = zxc_lex_next();
|
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-16 16:03:03 +01:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-16 16:03:03 +01:00
|
|
|
dbg_lex_done("%s:%d done (seen BC_LEX_RBRACE)", __func__, __LINE__);
|
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
dbg_lex("%s:%d p->lex:%d", __func__, __LINE__, p->lex);
|
|
|
|
switch (p->lex) {
|
2018-12-26 19:24:15 +01:00
|
|
|
case XC_LEX_OP_MINUS:
|
|
|
|
case BC_LEX_OP_INC:
|
|
|
|
case BC_LEX_OP_DEC:
|
|
|
|
case BC_LEX_OP_BOOL_NOT:
|
|
|
|
case BC_LEX_LPAREN:
|
|
|
|
case XC_LEX_NAME:
|
|
|
|
case XC_LEX_NUMBER:
|
|
|
|
case BC_LEX_KEY_IBASE:
|
|
|
|
case BC_LEX_KEY_LAST:
|
|
|
|
case BC_LEX_KEY_LENGTH:
|
|
|
|
case BC_LEX_KEY_OBASE:
|
|
|
|
case BC_LEX_KEY_READ:
|
|
|
|
case BC_LEX_KEY_SCALE:
|
|
|
|
case BC_LEX_KEY_SQRT:
|
|
|
|
s = zbc_parse_expr(BC_PARSE_PRINT);
|
|
|
|
break;
|
|
|
|
case XC_LEX_STR:
|
|
|
|
s = zbc_parse_pushSTR();
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(XC_INST_PRINT_STR);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case BC_LEX_KEY_BREAK:
|
|
|
|
case BC_LEX_KEY_CONTINUE:
|
|
|
|
s = zbc_parse_break_or_continue(p->lex);
|
|
|
|
break;
|
|
|
|
case BC_LEX_KEY_FOR:
|
|
|
|
s = zbc_parse_for();
|
|
|
|
break;
|
|
|
|
case BC_LEX_KEY_HALT:
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(BC_INST_HALT);
|
|
|
|
s = zxc_lex_next();
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case BC_LEX_KEY_IF:
|
|
|
|
s = zbc_parse_if();
|
|
|
|
break;
|
|
|
|
case BC_LEX_KEY_LIMITS:
|
|
|
|
// "limits" is a compile-time command,
|
|
|
|
// the output is produced at _parse time_.
|
|
|
|
printf(
|
|
|
|
"BC_BASE_MAX = "BC_MAX_OBASE_STR "\n"
|
|
|
|
"BC_DIM_MAX = "BC_MAX_DIM_STR "\n"
|
|
|
|
"BC_SCALE_MAX = "BC_MAX_SCALE_STR "\n"
|
|
|
|
"BC_STRING_MAX = "BC_MAX_STRING_STR"\n"
|
2019-01-04 16:26:19 +01:00
|
|
|
// "BC_NUM_MAX = "BC_MAX_NUM_STR "\n" - GNU bc does not show this
|
2018-12-26 19:24:15 +01:00
|
|
|
"MAX Exponent = "BC_MAX_EXP_STR "\n"
|
|
|
|
"Number of vars = "BC_MAX_VARS_STR "\n"
|
|
|
|
);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case BC_LEX_KEY_PRINT:
|
|
|
|
s = zbc_parse_print();
|
|
|
|
break;
|
|
|
|
case BC_LEX_KEY_QUIT:
|
|
|
|
// "quit" is a compile-time command. For example,
|
|
|
|
// "if (0 == 1) quit" terminates when parsing the statement,
|
|
|
|
// not when it is executed
|
|
|
|
QUIT_OR_RETURN_TO_MAIN;
|
|
|
|
case BC_LEX_KEY_RETURN:
|
|
|
|
if (!p->in_funcdef)
|
|
|
|
RETURN_STATUS(bc_error("'return' not in a function"));
|
|
|
|
s = zbc_parse_return();
|
|
|
|
break;
|
|
|
|
case BC_LEX_KEY_WHILE:
|
|
|
|
s = zbc_parse_while();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
s = bc_error_bad_token();
|
|
|
|
break;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-14 17:51:17 +01:00
|
|
|
dbg_lex_done("%s:%d done", __func__, __LINE__);
|
2018-12-12 15:19:54 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_stmt_possibly_auto(...) (zbc_parse_stmt_possibly_auto(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zbc_parse_stmt_or_funcdef(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
|
2018-12-14 17:51:17 +01:00
|
|
|
dbg_lex_enter("%s:%d entered", __func__, __LINE__);
|
2019-01-08 18:08:48 +01:00
|
|
|
//why?
|
|
|
|
// if (p->lex == XC_LEX_EOF)
|
|
|
|
// s = bc_error("end of file");
|
|
|
|
// else
|
|
|
|
if (p->lex == BC_LEX_KEY_DEFINE) {
|
2018-12-25 22:32:41 +01:00
|
|
|
dbg_lex("%s:%d p->lex:BC_LEX_KEY_DEFINE", __func__, __LINE__);
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_funcdef();
|
2018-12-14 17:51:17 +01:00
|
|
|
} else {
|
2018-12-25 22:32:41 +01:00
|
|
|
dbg_lex("%s:%d p->lex:%d (not BC_LEX_KEY_DEFINE)", __func__, __LINE__, p->lex);
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_stmt();
|
2018-12-14 17:51:17 +01:00
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-14 17:51:17 +01:00
|
|
|
dbg_lex_done("%s:%d done", __func__, __LINE__);
|
2018-12-12 16:08:46 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_parse_stmt_or_funcdef(...) (zbc_parse_stmt_or_funcdef(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2019-01-08 19:29:35 +01:00
|
|
|
#undef zbc_parse_expr
|
|
|
|
static BC_STATUS zbc_parse_expr(uint8_t flags)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-12-24 04:26:07 +01:00
|
|
|
BcInst prev = XC_INST_PRINT;
|
2018-11-03 11:00:21 -06:00
|
|
|
size_t nexprs = 0, ops_bgn = p->ops.len;
|
2018-12-07 12:57:32 +01:00
|
|
|
unsigned nparens, nrelops;
|
2019-01-01 21:50:14 +01:00
|
|
|
bool paren_first, rprn, assign, bin_last, incdec;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-14 23:00:24 +01:00
|
|
|
dbg_lex_enter("%s:%d entered", __func__, __LINE__);
|
2018-12-25 22:32:41 +01:00
|
|
|
paren_first = (p->lex == BC_LEX_LPAREN);
|
2018-11-03 11:00:21 -06:00
|
|
|
nparens = nrelops = 0;
|
2019-01-01 21:50:14 +01:00
|
|
|
rprn = assign = incdec = false;
|
2018-11-03 11:00:21 -06:00
|
|
|
bin_last = true;
|
|
|
|
|
2018-12-25 01:21:16 +01:00
|
|
|
for (;;) {
|
2018-12-25 01:43:52 +01:00
|
|
|
bool get_token;
|
2018-12-25 01:21:16 +01:00
|
|
|
BcStatus s;
|
2018-12-25 22:32:41 +01:00
|
|
|
BcLexType t = p->lex;
|
2018-12-25 01:21:16 +01:00
|
|
|
|
|
|
|
if (!lex_allowed_in_bc_expr(t))
|
|
|
|
break;
|
2018-12-25 01:16:37 +01:00
|
|
|
|
2018-12-22 01:34:10 +01:00
|
|
|
dbg_lex("%s:%d t:%d", __func__, __LINE__, t);
|
2018-12-25 01:43:52 +01:00
|
|
|
get_token = false;
|
2018-12-25 01:21:16 +01:00
|
|
|
s = BC_STATUS_SUCCESS;
|
2018-11-03 11:00:21 -06:00
|
|
|
switch (t) {
|
2018-12-26 19:24:15 +01:00
|
|
|
case BC_LEX_OP_INC:
|
|
|
|
case BC_LEX_OP_DEC:
|
|
|
|
dbg_lex("%s:%d LEX_OP_INC/DEC", __func__, __LINE__);
|
2019-01-08 19:29:35 +01:00
|
|
|
if (incdec) RETURN_STATUS(bc_error_bad_assignment());
|
2019-01-01 21:50:14 +01:00
|
|
|
s = zbc_parse_incdec(&prev, &nexprs, flags);
|
|
|
|
incdec = true;
|
2018-12-26 19:24:15 +01:00
|
|
|
rprn = bin_last = false;
|
|
|
|
//get_token = false; - already is
|
|
|
|
break;
|
|
|
|
case XC_LEX_OP_MINUS:
|
|
|
|
dbg_lex("%s:%d LEX_OP_MINUS", __func__, __LINE__);
|
2019-01-01 21:50:14 +01:00
|
|
|
s = zbc_parse_minus(&prev, ops_bgn, rprn, bin_last, &nexprs);
|
2018-12-26 19:24:15 +01:00
|
|
|
rprn = false;
|
|
|
|
//get_token = false; - already is
|
|
|
|
bin_last = (prev == XC_INST_MINUS);
|
2019-01-01 21:50:14 +01:00
|
|
|
if (bin_last) incdec = false;
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case BC_LEX_OP_ASSIGN_POWER:
|
|
|
|
case BC_LEX_OP_ASSIGN_MULTIPLY:
|
|
|
|
case BC_LEX_OP_ASSIGN_DIVIDE:
|
|
|
|
case BC_LEX_OP_ASSIGN_MODULUS:
|
|
|
|
case BC_LEX_OP_ASSIGN_PLUS:
|
|
|
|
case BC_LEX_OP_ASSIGN_MINUS:
|
|
|
|
case BC_LEX_OP_ASSIGN:
|
|
|
|
dbg_lex("%s:%d LEX_ASSIGNxyz", __func__, __LINE__);
|
|
|
|
if (prev != XC_INST_VAR && prev != XC_INST_ARRAY_ELEM
|
|
|
|
&& prev != XC_INST_SCALE && prev != XC_INST_IBASE
|
|
|
|
&& prev != XC_INST_OBASE && prev != BC_INST_LAST
|
|
|
|
) {
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_assignment());
|
2018-12-26 19:24:15 +01:00
|
|
|
}
|
|
|
|
// Fallthrough.
|
|
|
|
case XC_LEX_OP_POWER:
|
|
|
|
case XC_LEX_OP_MULTIPLY:
|
|
|
|
case XC_LEX_OP_DIVIDE:
|
|
|
|
case XC_LEX_OP_MODULUS:
|
|
|
|
case XC_LEX_OP_PLUS:
|
|
|
|
case XC_LEX_OP_REL_EQ:
|
|
|
|
case XC_LEX_OP_REL_LE:
|
|
|
|
case XC_LEX_OP_REL_GE:
|
|
|
|
case XC_LEX_OP_REL_NE:
|
|
|
|
case XC_LEX_OP_REL_LT:
|
|
|
|
case XC_LEX_OP_REL_GT:
|
|
|
|
case BC_LEX_OP_BOOL_NOT:
|
|
|
|
case BC_LEX_OP_BOOL_OR:
|
|
|
|
case BC_LEX_OP_BOOL_AND:
|
|
|
|
dbg_lex("%s:%d LEX_OP_xyz", __func__, __LINE__);
|
2019-01-01 21:50:14 +01:00
|
|
|
if (t == BC_LEX_OP_BOOL_NOT) {
|
|
|
|
if (!bin_last && p->lex_last != BC_LEX_OP_BOOL_NOT)
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_expression());
|
2019-01-01 21:50:14 +01:00
|
|
|
} else if (prev == XC_INST_BOOL_NOT) {
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_expression());
|
2018-12-26 19:24:15 +01:00
|
|
|
}
|
2019-01-01 21:50:14 +01:00
|
|
|
|
2018-12-26 19:24:15 +01:00
|
|
|
nrelops += (t >= XC_LEX_OP_REL_EQ && t <= XC_LEX_OP_REL_GT);
|
|
|
|
prev = BC_TOKEN_2_INST(t);
|
|
|
|
bc_parse_operator(t, ops_bgn, &nexprs);
|
2019-01-01 21:50:14 +01:00
|
|
|
rprn = incdec = false;
|
|
|
|
get_token = true;
|
2018-12-26 19:24:15 +01:00
|
|
|
bin_last = (t != BC_LEX_OP_BOOL_NOT);
|
|
|
|
break;
|
|
|
|
case BC_LEX_LPAREN:
|
|
|
|
dbg_lex("%s:%d LEX_LPAREN", __func__, __LINE__);
|
2019-01-01 21:50:14 +01:00
|
|
|
if (BC_PARSE_LEAF(prev, bin_last, rprn))
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_expression());
|
2018-12-26 19:24:15 +01:00
|
|
|
bc_vec_push(&p->ops, &t);
|
|
|
|
nparens++;
|
|
|
|
get_token = true;
|
2019-01-01 21:50:14 +01:00
|
|
|
rprn = incdec = false;
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case BC_LEX_RPAREN:
|
|
|
|
dbg_lex("%s:%d LEX_RPAREN", __func__, __LINE__);
|
2019-01-08 19:32:38 +01:00
|
|
|
//why?
|
|
|
|
// if (p->lex_last == BC_LEX_LPAREN) {
|
|
|
|
// RETURN_STATUS(bc_error_at("empty expression"));
|
|
|
|
// }
|
2018-12-26 19:24:15 +01:00
|
|
|
if (bin_last || prev == XC_INST_BOOL_NOT)
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_expression());
|
2018-12-26 19:24:15 +01:00
|
|
|
if (nparens == 0) {
|
|
|
|
goto exit_loop;
|
|
|
|
}
|
|
|
|
s = zbc_parse_rightParen(ops_bgn, &nexprs);
|
|
|
|
nparens--;
|
|
|
|
get_token = true;
|
2019-01-01 21:50:14 +01:00
|
|
|
rprn = true;
|
|
|
|
bin_last = incdec = false;
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case XC_LEX_NAME:
|
|
|
|
dbg_lex("%s:%d LEX_NAME", __func__, __LINE__);
|
2019-01-01 21:50:14 +01:00
|
|
|
if (BC_PARSE_LEAF(prev, bin_last, rprn))
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_expression());
|
2018-12-26 19:24:15 +01:00
|
|
|
s = zbc_parse_name(&prev, flags & ~BC_PARSE_NOCALL);
|
2019-01-01 21:50:14 +01:00
|
|
|
rprn = (prev == BC_INST_CALL);
|
|
|
|
bin_last = false;
|
2018-12-26 19:24:15 +01:00
|
|
|
//get_token = false; - already is
|
|
|
|
nexprs++;
|
|
|
|
break;
|
|
|
|
case XC_LEX_NUMBER:
|
|
|
|
dbg_lex("%s:%d LEX_NUMBER", __func__, __LINE__);
|
2019-01-01 21:50:14 +01:00
|
|
|
if (BC_PARSE_LEAF(prev, bin_last, rprn))
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_expression());
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_pushNUM();
|
2018-12-26 19:24:15 +01:00
|
|
|
prev = XC_INST_NUM;
|
|
|
|
get_token = true;
|
|
|
|
rprn = bin_last = false;
|
|
|
|
nexprs++;
|
|
|
|
break;
|
|
|
|
case BC_LEX_KEY_IBASE:
|
|
|
|
case BC_LEX_KEY_LAST:
|
|
|
|
case BC_LEX_KEY_OBASE:
|
|
|
|
dbg_lex("%s:%d LEX_IBASE/LAST/OBASE", __func__, __LINE__);
|
2019-01-01 21:50:14 +01:00
|
|
|
if (BC_PARSE_LEAF(prev, bin_last, rprn))
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_expression());
|
2018-12-26 19:24:15 +01:00
|
|
|
prev = (char) (t - BC_LEX_KEY_IBASE + XC_INST_IBASE);
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push((char) prev);
|
2018-12-26 19:24:15 +01:00
|
|
|
get_token = true;
|
|
|
|
rprn = bin_last = false;
|
|
|
|
nexprs++;
|
|
|
|
break;
|
|
|
|
case BC_LEX_KEY_LENGTH:
|
|
|
|
case BC_LEX_KEY_SQRT:
|
|
|
|
dbg_lex("%s:%d LEX_LEN/SQRT", __func__, __LINE__);
|
2019-01-01 21:50:14 +01:00
|
|
|
if (BC_PARSE_LEAF(prev, bin_last, rprn))
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_expression());
|
2018-12-26 19:24:15 +01:00
|
|
|
s = zbc_parse_builtin(t, flags, &prev);
|
|
|
|
get_token = true;
|
2019-01-01 21:50:14 +01:00
|
|
|
rprn = bin_last = incdec = false;
|
2018-12-26 19:24:15 +01:00
|
|
|
nexprs++;
|
|
|
|
break;
|
|
|
|
case BC_LEX_KEY_READ:
|
|
|
|
dbg_lex("%s:%d LEX_READ", __func__, __LINE__);
|
2019-01-01 21:50:14 +01:00
|
|
|
if (BC_PARSE_LEAF(prev, bin_last, rprn))
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_expression());
|
2018-12-26 19:24:15 +01:00
|
|
|
s = zbc_parse_read();
|
|
|
|
prev = XC_INST_READ;
|
|
|
|
get_token = true;
|
2019-01-01 21:50:14 +01:00
|
|
|
rprn = bin_last = incdec = false;
|
2018-12-26 19:24:15 +01:00
|
|
|
nexprs++;
|
|
|
|
break;
|
|
|
|
case BC_LEX_KEY_SCALE:
|
|
|
|
dbg_lex("%s:%d LEX_SCALE", __func__, __LINE__);
|
2019-01-01 21:50:14 +01:00
|
|
|
if (BC_PARSE_LEAF(prev, bin_last, rprn))
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_expression());
|
2018-12-26 19:24:15 +01:00
|
|
|
s = zbc_parse_scale(&prev, flags);
|
|
|
|
//get_token = false; - already is
|
|
|
|
rprn = bin_last = false;
|
|
|
|
nexprs++;
|
|
|
|
break;
|
|
|
|
default:
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_token());
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-25 01:16:37 +01:00
|
|
|
if (s || G_interrupt) // error, or ^C: stop parsing
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_FAILURE);
|
2018-12-25 01:16:37 +01:00
|
|
|
if (get_token) {
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2019-01-08 19:29:35 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 01:16:37 +01:00
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-25 01:16:37 +01:00
|
|
|
exit_loop:
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
while (p->ops.len > ops_bgn) {
|
2018-12-25 01:21:16 +01:00
|
|
|
BcLexType top = BC_PARSE_TOP_OP(p);
|
2018-12-24 17:06:37 +01:00
|
|
|
assign = (top >= BC_LEX_OP_ASSIGN_POWER && top <= BC_LEX_OP_ASSIGN);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (top == BC_LEX_LPAREN || top == BC_LEX_RPAREN)
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_expression());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(BC_TOKEN_2_INST(top));
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 14:14:23 +01:00
|
|
|
nexprs -= (top != BC_LEX_OP_BOOL_NOT && top != XC_LEX_NEG);
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_vec_pop(&p->ops);
|
|
|
|
}
|
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
if (prev == XC_INST_BOOL_NOT || nexprs != 1)
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_expression());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (!(flags & BC_PARSE_REL) && nrelops) {
|
2018-12-25 01:16:37 +01:00
|
|
|
BcStatus s;
|
2018-12-24 18:11:41 +01:00
|
|
|
s = zbc_POSIX_does_not_allow("comparison operators outside if or loops");
|
2019-01-08 19:29:35 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-18 16:24:07 +01:00
|
|
|
} else if ((flags & BC_PARSE_REL) && nrelops > 1) {
|
2018-12-25 01:16:37 +01:00
|
|
|
BcStatus s;
|
2018-12-24 18:11:41 +01:00
|
|
|
s = zbc_POSIX_requires("exactly one comparison operator per condition");
|
2019-01-08 19:29:35 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (flags & BC_PARSE_PRINT) {
|
2018-12-25 01:16:37 +01:00
|
|
|
if (paren_first || !assign)
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(XC_INST_PRINT);
|
|
|
|
xc_parse_push(XC_INST_POP);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-14 23:00:24 +01:00
|
|
|
dbg_lex_done("%s:%d done", __func__, __LINE__);
|
2019-01-08 19:29:35 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2019-01-08 19:29:35 +01:00
|
|
|
#define zbc_parse_expr(...) (zbc_parse_expr(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
#endif // ENABLE_BC
|
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_DC
|
2018-12-05 21:15:46 +01:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zdc_parse_register(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 14:54:38 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex != XC_LEX_NAME) RETURN_STATUS(bc_error_bad_token());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_pushName(p->lex_strnumbuf.v);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-12 14:54:38 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zdc_parse_register(...) (zdc_parse_register(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static void dc_parse_string(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-12-20 20:34:09 +01:00
|
|
|
char *str;
|
2018-12-19 14:57:23 +01:00
|
|
|
size_t len = G.prog.strs.len;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-20 20:34:09 +01:00
|
|
|
dbg_lex_enter("%s:%d entered", __func__, __LINE__);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
str = xstrdup(p->lex_strnumbuf.v);
|
2019-01-09 11:17:19 +01:00
|
|
|
xc_parse_pushInst_and_Index(XC_INST_STR, len);
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_push(&G.prog.strs, &str);
|
2018-12-20 20:34:09 +01:00
|
|
|
|
2019-09-05 10:50:13 +02:00
|
|
|
// Add an empty function so that if zdc_program_execStr ever needs to
|
|
|
|
// parse the string into code (from the 'x' command) there's somewhere
|
|
|
|
// to store the bytecode.
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_add_fn();
|
|
|
|
p->func = xc_program_func(p->fidx);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-20 20:34:09 +01:00
|
|
|
dbg_lex_done("%s:%d done", __func__, __LINE__);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zdc_parse_mem(uint8_t inst, bool name, bool store)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(inst);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (name) {
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zdc_parse_register();
|
2018-12-12 14:54:38 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (store) {
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(DC_INST_SWAP);
|
|
|
|
xc_parse_push(XC_INST_ASSIGN);
|
|
|
|
xc_parse_push(XC_INST_POP);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-22 16:40:38 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zdc_parse_mem(...) (zdc_parse_mem(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zdc_parse_cond(uint8_t inst)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcStatus s;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(inst);
|
|
|
|
xc_parse_push(DC_INST_EXEC_COND);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zdc_parse_register();
|
2018-12-12 14:54:38 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-12 14:54:38 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-22 18:04:08 +01:00
|
|
|
// Note that 'else' part can not be on the next line:
|
|
|
|
// echo -e '[1p]sa [2p]sb 2 1>a eb' | dc - OK, prints "2"
|
|
|
|
// echo -e '[1p]sa [2p]sb 2 1>a\neb' | dc - parse error
|
2018-12-25 22:32:41 +01:00
|
|
|
if (p->lex == DC_LEX_ELSE) {
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zdc_parse_register();
|
2018-12-12 14:54:38 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-22 18:04:08 +01:00
|
|
|
} else {
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push('\0');
|
2018-12-22 18:04:08 +01:00
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-12 14:54:38 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zdc_parse_cond(...) (zdc_parse_cond(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zdc_parse_token(BcLexType t)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-22 16:40:38 +01:00
|
|
|
BcStatus s;
|
2018-11-03 11:00:21 -06:00
|
|
|
uint8_t inst;
|
2018-12-22 16:40:38 +01:00
|
|
|
bool assign, get_token;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-20 20:34:09 +01:00
|
|
|
dbg_lex_enter("%s:%d entered", __func__, __LINE__);
|
2018-12-22 16:40:38 +01:00
|
|
|
s = BC_STATUS_SUCCESS;
|
|
|
|
get_token = true;
|
2018-11-03 11:00:21 -06:00
|
|
|
switch (t) {
|
2018-12-26 19:24:15 +01:00
|
|
|
case XC_LEX_OP_REL_EQ:
|
|
|
|
case XC_LEX_OP_REL_LE:
|
|
|
|
case XC_LEX_OP_REL_GE:
|
|
|
|
case XC_LEX_OP_REL_NE:
|
|
|
|
case XC_LEX_OP_REL_LT:
|
|
|
|
case XC_LEX_OP_REL_GT:
|
|
|
|
dbg_lex("%s:%d LEX_OP_REL_xyz", __func__, __LINE__);
|
|
|
|
s = zdc_parse_cond(t - XC_LEX_OP_REL_EQ + XC_INST_REL_EQ);
|
|
|
|
get_token = false;
|
|
|
|
break;
|
|
|
|
case DC_LEX_SCOLON:
|
|
|
|
case DC_LEX_COLON:
|
|
|
|
dbg_lex("%s:%d LEX_[S]COLON", __func__, __LINE__);
|
|
|
|
s = zdc_parse_mem(XC_INST_ARRAY_ELEM, true, t == DC_LEX_COLON);
|
|
|
|
break;
|
|
|
|
case XC_LEX_STR:
|
|
|
|
dbg_lex("%s:%d LEX_STR", __func__, __LINE__);
|
|
|
|
dc_parse_string();
|
|
|
|
break;
|
|
|
|
case XC_LEX_NEG:
|
|
|
|
dbg_lex("%s:%d LEX_NEG", __func__, __LINE__);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_lex_next();
|
2018-12-26 19:24:15 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
if (G.prs.lex != XC_LEX_NUMBER)
|
2018-12-22 16:40:38 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_token());
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_pushNUM();
|
|
|
|
xc_parse_push(XC_INST_NEG);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case XC_LEX_NUMBER:
|
|
|
|
dbg_lex("%s:%d LEX_NUMBER", __func__, __LINE__);
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_pushNUM();
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case DC_LEX_READ:
|
|
|
|
dbg_lex("%s:%d LEX_KEY_READ", __func__, __LINE__);
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(XC_INST_READ);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case DC_LEX_OP_ASSIGN:
|
|
|
|
case DC_LEX_STORE_PUSH:
|
|
|
|
dbg_lex("%s:%d LEX_OP_ASSIGN/STORE_PUSH", __func__, __LINE__);
|
|
|
|
assign = (t == DC_LEX_OP_ASSIGN);
|
|
|
|
inst = assign ? XC_INST_VAR : DC_INST_PUSH_TO_VAR;
|
|
|
|
s = zdc_parse_mem(inst, true, assign);
|
|
|
|
break;
|
|
|
|
case DC_LEX_LOAD:
|
|
|
|
case DC_LEX_LOAD_POP:
|
|
|
|
dbg_lex("%s:%d LEX_OP_LOAD[_POP]", __func__, __LINE__);
|
|
|
|
inst = t == DC_LEX_LOAD_POP ? DC_INST_PUSH_VAR : DC_INST_LOAD;
|
|
|
|
s = zdc_parse_mem(inst, true, false);
|
|
|
|
break;
|
|
|
|
case DC_LEX_STORE_IBASE:
|
|
|
|
case DC_LEX_STORE_SCALE:
|
|
|
|
case DC_LEX_STORE_OBASE:
|
|
|
|
dbg_lex("%s:%d LEX_OP_STORE_I/OBASE/SCALE", __func__, __LINE__);
|
|
|
|
inst = t - DC_LEX_STORE_IBASE + XC_INST_IBASE;
|
|
|
|
s = zdc_parse_mem(inst, false, true);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
dbg_lex_done("%s:%d done (bad token)", __func__, __LINE__);
|
|
|
|
RETURN_STATUS(bc_error_bad_token());
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
if (!s && get_token) s = zxc_lex_next();
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-20 20:34:09 +01:00
|
|
|
dbg_lex_done("%s:%d done", __func__, __LINE__);
|
2018-12-12 14:54:38 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zdc_parse_token(...) (zdc_parse_token(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zdc_parse_expr(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-12-24 15:29:08 +01:00
|
|
|
int i;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2019-01-04 05:41:47 +01:00
|
|
|
if (p->lex == XC_LEX_NLINE)
|
|
|
|
RETURN_STATUS(zxc_lex_next());
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
i = (int)p->lex - (int)XC_LEX_OP_POWER;
|
2018-12-24 15:29:08 +01:00
|
|
|
if (i >= 0) {
|
|
|
|
BcInst inst = dc_LEX_to_INST[i];
|
|
|
|
if (inst != DC_INST_INVALID) {
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(inst);
|
|
|
|
RETURN_STATUS(zxc_lex_next());
|
2018-12-24 15:29:08 +01:00
|
|
|
}
|
2018-12-22 18:04:08 +01:00
|
|
|
}
|
2018-12-25 22:32:41 +01:00
|
|
|
RETURN_STATUS(zdc_parse_token(p->lex));
|
2018-12-22 18:04:08 +01:00
|
|
|
}
|
|
|
|
#define zdc_parse_expr(...) (zdc_parse_expr(__VA_ARGS__) COMMA_SUCCESS)
|
2018-12-18 14:11:35 +01:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
static BC_STATUS zdc_parse_exprs_until_eof(void)
|
2018-12-22 18:04:08 +01:00
|
|
|
{
|
2018-12-25 20:40:55 +01:00
|
|
|
BcParse *p = &G.prs;
|
2018-12-25 22:32:41 +01:00
|
|
|
dbg_lex_enter("%s:%d entered, p->lex:%d", __func__, __LINE__, p->lex);
|
|
|
|
while (p->lex != XC_LEX_EOF) {
|
2018-12-25 20:40:55 +01:00
|
|
|
BcStatus s = zdc_parse_expr();
|
2018-12-18 14:11:35 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-20 20:34:09 +01:00
|
|
|
dbg_lex_done("%s:%d done", __func__, __LINE__);
|
2018-12-18 14:11:35 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-22 18:04:08 +01:00
|
|
|
#define zdc_parse_exprs_until_eof(...) (zdc_parse_exprs_until_eof(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
#endif // ENABLE_DC
|
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
//
|
|
|
|
// Execution engine
|
|
|
|
//
|
|
|
|
|
2018-12-30 15:56:36 +01:00
|
|
|
#define BC_PROG_STR(n) (!(n)->num && !(n)->cap)
|
|
|
|
#define BC_PROG_NUM(r, n) \
|
|
|
|
((r)->t != XC_RESULT_ARRAY && (r)->t != XC_RESULT_STR && !BC_PROG_STR(n))
|
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
#define STACK_HAS_MORE_THAN(s, n) ((s)->len > ((size_t)(n)))
|
|
|
|
#define STACK_HAS_EQUAL_OR_MORE_THAN(s, n) ((s)->len >= ((size_t)(n)))
|
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
static size_t xc_program_index(char *code, size_t *bgn)
|
|
|
|
{
|
|
|
|
unsigned char *bytes = (void*)(code + *bgn);
|
|
|
|
unsigned amt;
|
|
|
|
unsigned i;
|
|
|
|
size_t res;
|
|
|
|
|
|
|
|
amt = *bytes++;
|
|
|
|
if (amt < SMALL_INDEX_LIMIT) {
|
|
|
|
*bgn += 1;
|
|
|
|
return amt;
|
|
|
|
}
|
|
|
|
amt -= (SMALL_INDEX_LIMIT - 1); // amt is 1 or more here
|
|
|
|
*bgn += amt + 1;
|
|
|
|
|
|
|
|
res = 0;
|
|
|
|
i = 0;
|
|
|
|
do {
|
|
|
|
res |= (size_t)(*bytes++) << i;
|
|
|
|
i += 8;
|
|
|
|
} while (--amt != 0);
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *xc_program_name(char *code, size_t *bgn)
|
|
|
|
{
|
|
|
|
code += *bgn;
|
|
|
|
*bgn += strlen(code) + 1;
|
|
|
|
|
|
|
|
return xstrdup(code);
|
|
|
|
}
|
|
|
|
|
|
|
|
static BcVec* xc_program_dereference(BcVec *vec)
|
|
|
|
{
|
|
|
|
BcVec *v;
|
|
|
|
size_t vidx, nidx, i = 0;
|
|
|
|
|
|
|
|
//assert(vec->size == sizeof(uint8_t));
|
|
|
|
|
|
|
|
vidx = xc_program_index(vec->v, &i);
|
|
|
|
nidx = xc_program_index(vec->v, &i);
|
|
|
|
|
|
|
|
v = bc_vec_item(&G.prog.arrs, vidx);
|
|
|
|
v = bc_vec_item(v, nidx);
|
|
|
|
|
|
|
|
//assert(v->size != sizeof(uint8_t));
|
|
|
|
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
static BcVec* xc_program_search(char *id, BcType type)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcId e, *ptr;
|
|
|
|
BcVec *v, *map;
|
|
|
|
size_t i;
|
2018-12-03 20:35:16 +01:00
|
|
|
int new;
|
2019-01-25 14:24:03 +01:00
|
|
|
bool var = (type == BC_TYPE_VAR);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:26:38 +01:00
|
|
|
v = var ? &G.prog.vars : &G.prog.arrs;
|
|
|
|
map = var ? &G.prog.var_map : &G.prog.arr_map;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
e.name = id;
|
|
|
|
e.idx = v->len;
|
2018-12-03 20:35:16 +01:00
|
|
|
new = bc_map_insert(map, &e, &i); // 1 if insertion was successful
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (new) {
|
2018-12-19 17:15:04 +01:00
|
|
|
BcVec v2;
|
|
|
|
bc_array_init(&v2, var);
|
|
|
|
bc_vec_push(v, &v2);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
ptr = bc_vec_item(map, i);
|
|
|
|
if (new) ptr->name = xstrdup(e.name);
|
2018-12-02 19:27:48 +01:00
|
|
|
return bc_vec_item(v, ptr->idx);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-21 22:43:53 +01:00
|
|
|
// 'num' need not be initialized on entry
|
2018-12-29 14:52:30 +01:00
|
|
|
static BC_STATUS zxc_program_num(BcResult *r, BcNum **num)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
switch (r->t) {
|
2018-12-26 19:24:15 +01:00
|
|
|
case XC_RESULT_STR:
|
|
|
|
case XC_RESULT_TEMP:
|
2019-01-04 13:58:46 +01:00
|
|
|
IF_BC(case BC_RESULT_VOID:)
|
2018-12-26 19:24:15 +01:00
|
|
|
case XC_RESULT_IBASE:
|
|
|
|
case XC_RESULT_SCALE:
|
|
|
|
case XC_RESULT_OBASE:
|
|
|
|
*num = &r->d.n;
|
|
|
|
break;
|
|
|
|
case XC_RESULT_CONSTANT: {
|
|
|
|
BcStatus s;
|
|
|
|
char *str;
|
|
|
|
size_t len;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
str = *xc_program_const(r->d.id.idx);
|
2018-12-26 19:24:15 +01:00
|
|
|
len = strlen(str);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-26 19:24:15 +01:00
|
|
|
bc_num_init(&r->d.n, len);
|
|
|
|
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_num_parse(&r->d.n, str, G.prog.ib_t);
|
2018-12-26 19:24:15 +01:00
|
|
|
if (s) {
|
|
|
|
bc_num_free(&r->d.n);
|
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-26 19:24:15 +01:00
|
|
|
*num = &r->d.n;
|
|
|
|
r->t = XC_RESULT_TEMP;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case XC_RESULT_VAR:
|
|
|
|
case XC_RESULT_ARRAY:
|
|
|
|
case XC_RESULT_ARRAY_ELEM: {
|
2019-01-25 14:24:03 +01:00
|
|
|
BcType type = (r->t == XC_RESULT_VAR) ? BC_TYPE_VAR : BC_TYPE_ARRAY;
|
|
|
|
BcVec *v = xc_program_search(r->d.id.name, type);
|
|
|
|
void *p = bc_vec_top(v);
|
|
|
|
|
2018-12-26 19:24:15 +01:00
|
|
|
if (r->t == XC_RESULT_ARRAY_ELEM) {
|
2019-01-25 14:24:03 +01:00
|
|
|
size_t idx = r->d.id.idx;
|
|
|
|
|
2018-12-29 16:23:34 +01:00
|
|
|
v = p;
|
2019-01-25 14:24:03 +01:00
|
|
|
if (v->size == sizeof(uint8_t))
|
|
|
|
v = xc_program_dereference(v);
|
|
|
|
//assert(v->size == sizeof(BcNum));
|
|
|
|
if (v->len <= idx)
|
|
|
|
bc_array_expand(v, idx + 1);
|
|
|
|
*num = bc_vec_item(v, idx);
|
2018-12-29 16:23:34 +01:00
|
|
|
} else {
|
|
|
|
*num = p;
|
|
|
|
}
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
}
|
2018-12-20 16:24:18 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-26 19:24:15 +01:00
|
|
|
case BC_RESULT_LAST:
|
|
|
|
*num = &G.prog.last;
|
|
|
|
break;
|
|
|
|
case BC_RESULT_ONE:
|
|
|
|
*num = &G.prog.one;
|
|
|
|
break;
|
2018-12-20 16:24:18 +01:00
|
|
|
#endif
|
2018-12-24 00:50:32 +01:00
|
|
|
#if SANITY_CHECKS
|
2018-12-26 19:24:15 +01:00
|
|
|
default:
|
|
|
|
// Testing the theory that dc does not reach LAST/ONE
|
|
|
|
bb_error_msg_and_die("BUG:%d", r->t);
|
2018-12-24 00:50:32 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_program_num(...) (zxc_program_num(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_program_binOpPrep(BcResult **l, BcNum **ln,
|
2018-11-03 11:00:21 -06:00
|
|
|
BcResult **r, BcNum **rn, bool assign)
|
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcResultType lt, rt;
|
|
|
|
|
2018-12-19 17:09:01 +01:00
|
|
|
if (!STACK_HAS_MORE_THAN(&G.prog.results, 1))
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error_stack_has_too_few_elements());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:26:38 +01:00
|
|
|
*r = bc_vec_item_rev(&G.prog.results, 0);
|
|
|
|
*l = bc_vec_item_rev(&G.prog.results, 1);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_program_num(*l, ln);
|
2018-12-11 15:29:32 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_program_num(*r, rn);
|
2018-12-11 15:29:32 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-29 14:52:30 +01:00
|
|
|
lt = (*l)->t;
|
|
|
|
rt = (*r)->t;
|
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
// We run this again under these conditions in case any vector has been
|
|
|
|
// reallocated out from under the BcNums or arrays we had.
|
2018-12-24 23:41:31 +01:00
|
|
|
if (lt == rt && (lt == XC_RESULT_VAR || lt == XC_RESULT_ARRAY_ELEM)) {
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_program_num(*l, ln);
|
2018-12-11 15:29:32 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-24 23:41:31 +01:00
|
|
|
if (!BC_PROG_NUM((*l), (*ln)) && (!assign || (*l)->t != XC_RESULT_VAR))
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error_variable_is_wrong_type());
|
2018-12-04 20:05:28 +01:00
|
|
|
if (!assign && !BC_PROG_NUM((*r), (*ln)))
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error_variable_is_wrong_type());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_program_binOpPrep(...) (zxc_program_binOpPrep(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_program_binOpRetire(BcResult *r)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-24 23:41:31 +01:00
|
|
|
r->t = XC_RESULT_TEMP;
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_pop(&G.prog.results);
|
2018-12-21 23:13:48 +01:00
|
|
|
bc_result_pop_and_push(r);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-24 17:15:34 +01:00
|
|
|
// Note: *r and *n need not be initialized by caller
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_program_prep(BcResult **r, BcNum **n)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
|
2018-12-19 17:09:01 +01:00
|
|
|
if (!STACK_HAS_MORE_THAN(&G.prog.results, 0))
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error_stack_has_too_few_elements());
|
2018-12-02 18:26:38 +01:00
|
|
|
*r = bc_vec_top(&G.prog.results);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_program_num(*r, n);
|
2018-12-11 15:29:32 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-04 20:05:28 +01:00
|
|
|
if (!BC_PROG_NUM((*r), (*n)))
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error_variable_is_wrong_type());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_program_prep(...) (zxc_program_prep(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_program_retire(BcResult *r, BcResultType t)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
r->t = t;
|
2018-12-21 23:13:48 +01:00
|
|
|
bc_result_pop_and_push(r);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_program_op(char inst)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcResult *opd1, *opd2, res;
|
2018-12-24 17:15:34 +01:00
|
|
|
BcNum *n1, *n2;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_binOpPrep(&opd1, &n1, &opd2, &n2, false);
|
2018-12-11 19:42:05 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-09 11:44:20 +01:00
|
|
|
bc_num_init_DEF_SIZE(&res.d.n);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
s = BC_STATUS_SUCCESS;
|
2018-12-27 18:23:58 +01:00
|
|
|
IF_ERROR_RETURN_POSSIBLE(s =) zxc_program_ops[inst - XC_INST_POWER](n1, n2, &res.d.n, G.prog.scale);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_binOpRetire(&res);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 19:42:05 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-12-18 16:24:07 +01:00
|
|
|
err:
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_free(&res.d.n);
|
2018-12-11 19:42:05 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_program_op(...) (zxc_program_op(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_program_read(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
2018-12-25 19:38:13 +01:00
|
|
|
BcParse sv_parse;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcVec buf;
|
|
|
|
BcInstPtr ip;
|
2018-12-12 00:29:24 +01:00
|
|
|
BcFunc *f;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-13 19:23:45 +01:00
|
|
|
bc_char_vec_init(&buf);
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_read_line(&buf, stdin);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
f = xc_program_func(BC_PROG_READ);
|
2018-12-25 19:38:13 +01:00
|
|
|
bc_vec_pop_all(&f->code);
|
|
|
|
|
|
|
|
sv_parse = G.prs; // struct copy
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_create(BC_PROG_READ);
|
2018-12-26 18:32:43 +01:00
|
|
|
//G.err_line = G.prs.lex_line = 1; - not needed, error line info is not printed for read()
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_parse_text_init(buf.v);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto exec_err;
|
2018-12-22 03:38:52 +01:00
|
|
|
if (IS_BC) {
|
2018-12-25 20:40:55 +01:00
|
|
|
IF_BC(s = zbc_parse_expr(0));
|
2018-12-22 03:38:52 +01:00
|
|
|
} else {
|
2018-12-25 20:40:55 +01:00
|
|
|
IF_DC(s = zdc_parse_exprs_until_eof());
|
2018-12-22 03:38:52 +01:00
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto exec_err;
|
2018-12-25 22:32:41 +01:00
|
|
|
if (G.prs.lex != XC_LEX_NLINE && G.prs.lex != XC_LEX_EOF) {
|
2019-01-01 21:50:14 +01:00
|
|
|
s = bc_error_at("bad read() expression");
|
2018-11-03 11:00:21 -06:00
|
|
|
goto exec_err;
|
|
|
|
}
|
2018-12-31 18:48:10 +01:00
|
|
|
xc_parse_push(XC_INST_RET);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
ip.func = BC_PROG_READ;
|
2018-12-21 23:01:26 +01:00
|
|
|
ip.inst_idx = 0;
|
2018-12-19 12:35:27 +01:00
|
|
|
bc_vec_push(&G.prog.exestack, &ip);
|
2018-12-25 19:38:13 +01:00
|
|
|
|
2018-12-18 16:24:07 +01:00
|
|
|
exec_err:
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_free();
|
2018-12-25 19:38:13 +01:00
|
|
|
G.prs = sv_parse; // struct copy
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_vec_free(&buf);
|
2018-12-12 16:08:46 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_program_read(...) (zxc_program_read(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_program_printString(const char *str)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_DC
|
2019-01-02 05:03:53 +01:00
|
|
|
if (!str[0] && IS_DC) {
|
2018-12-11 21:21:14 +01:00
|
|
|
// Example: echo '[]ap' | dc
|
|
|
|
// should print two bytes: 0x00, 0x0A
|
2018-11-30 23:13:42 +01:00
|
|
|
bb_putchar('\0');
|
2018-11-03 11:00:21 -06:00
|
|
|
return;
|
|
|
|
}
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-12-11 19:52:25 +01:00
|
|
|
while (*str) {
|
2019-01-02 05:03:53 +01:00
|
|
|
char c = *str++;
|
|
|
|
if (c == '\\') {
|
|
|
|
static const char esc[] ALIGN1 = "nabfrt""e\\";
|
|
|
|
char *n;
|
|
|
|
|
2018-12-11 19:52:25 +01:00
|
|
|
c = *str++;
|
2019-09-05 10:53:21 +02:00
|
|
|
n = strchr(esc, c); // note: if c is NUL, n = \0 at end of esc
|
|
|
|
if (!n || !c) {
|
2019-01-02 05:03:53 +01:00
|
|
|
// Just print the backslash and following character
|
2018-12-11 19:52:25 +01:00
|
|
|
bb_putchar('\\');
|
|
|
|
++G.prog.nchars;
|
2019-09-05 10:53:21 +02:00
|
|
|
// But if we're at the end of the string, stop
|
|
|
|
if (!c) break;
|
2019-01-02 05:03:53 +01:00
|
|
|
} else {
|
|
|
|
if (n - esc == 0) // "\n" ?
|
|
|
|
G.prog.nchars = SIZE_MAX;
|
|
|
|
c = "\n\a\b\f\r\t""\\\\""\\"[n - esc];
|
|
|
|
// n a b f r t e \ \<end of line>
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
}
|
2019-01-02 05:03:53 +01:00
|
|
|
putchar(c);
|
2018-12-11 19:52:25 +01:00
|
|
|
++G.prog.nchars;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
static void bc_num_printNewline(void)
|
|
|
|
{
|
|
|
|
if (G.prog.nchars == G.prog.len - 1) {
|
|
|
|
bb_putchar('\\');
|
|
|
|
bb_putchar('\n');
|
|
|
|
G.prog.nchars = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if ENABLE_DC
|
2018-12-19 19:35:40 +01:00
|
|
|
static FAST_FUNC void dc_num_printChar(size_t num, size_t width, bool radix)
|
2018-12-11 19:04:44 +01:00
|
|
|
{
|
|
|
|
(void) radix;
|
|
|
|
bb_putchar((char) num);
|
|
|
|
G.prog.nchars += width;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static FAST_FUNC void bc_num_printDigits(size_t num, size_t width, bool radix)
|
|
|
|
{
|
|
|
|
size_t exp, pow;
|
|
|
|
|
|
|
|
bc_num_printNewline();
|
|
|
|
bb_putchar(radix ? '.' : ' ');
|
|
|
|
++G.prog.nchars;
|
|
|
|
|
|
|
|
bc_num_printNewline();
|
|
|
|
for (exp = 0, pow = 1; exp < width - 1; ++exp, pow *= 10)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for (exp = 0; exp < width; pow /= 10, ++G.prog.nchars, ++exp) {
|
|
|
|
size_t dig;
|
|
|
|
bc_num_printNewline();
|
|
|
|
dig = num / pow;
|
|
|
|
num -= dig * pow;
|
|
|
|
bb_putchar(((char) dig) + '0');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static FAST_FUNC void bc_num_printHex(size_t num, size_t width, bool radix)
|
|
|
|
{
|
|
|
|
if (radix) {
|
|
|
|
bc_num_printNewline();
|
|
|
|
bb_putchar('.');
|
2018-12-18 19:20:04 +01:00
|
|
|
G.prog.nchars++;
|
2018-12-11 19:04:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bc_num_printNewline();
|
|
|
|
bb_putchar(bb_hexdigits_upcase[num]);
|
|
|
|
G.prog.nchars += width;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_num_printDecimal(BcNum *n)
|
|
|
|
{
|
|
|
|
size_t i, rdx = n->rdx - 1;
|
|
|
|
|
2018-12-18 19:20:04 +01:00
|
|
|
if (n->neg) {
|
|
|
|
bb_putchar('-');
|
|
|
|
G.prog.nchars++;
|
|
|
|
}
|
2018-12-11 19:04:44 +01:00
|
|
|
|
|
|
|
for (i = n->len - 1; i < n->len; --i)
|
|
|
|
bc_num_printHex((size_t) n->num[i], 1, i == rdx);
|
|
|
|
}
|
|
|
|
|
2018-12-24 05:00:36 +01:00
|
|
|
typedef void (*BcNumDigitOp)(size_t, size_t, bool) FAST_FUNC;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_num_printNum(BcNum *n, unsigned base_t, size_t width, BcNumDigitOp print)
|
2018-12-11 19:04:44 +01:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcVec stack;
|
2018-12-18 17:00:35 +01:00
|
|
|
BcNum base;
|
2018-12-21 22:36:04 +01:00
|
|
|
BcDig base_digs[ULONG_NUM_BUFSIZE];
|
2018-12-11 19:04:44 +01:00
|
|
|
BcNum intp, fracp, digit, frac_len;
|
|
|
|
unsigned long dig, *ptr;
|
|
|
|
size_t i;
|
|
|
|
bool radix;
|
|
|
|
|
|
|
|
if (n->len == 0) {
|
|
|
|
print(0, width, false);
|
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
|
|
|
}
|
|
|
|
|
|
|
|
bc_vec_init(&stack, sizeof(long), NULL);
|
2021-06-12 12:35:15 +02:00
|
|
|
bc_num_init_and_copy(&intp, n);
|
2018-12-11 19:04:44 +01:00
|
|
|
bc_num_init(&fracp, n->rdx);
|
|
|
|
bc_num_init(&digit, width);
|
|
|
|
bc_num_init(&frac_len, BC_NUM_INT(n));
|
|
|
|
bc_num_one(&frac_len);
|
2018-12-21 22:36:04 +01:00
|
|
|
base.cap = ARRAY_SIZE(base_digs);
|
|
|
|
base.num = base_digs;
|
2018-12-18 17:00:35 +01:00
|
|
|
bc_num_ulong2num(&base, base_t);
|
2018-12-11 19:04:44 +01:00
|
|
|
|
|
|
|
bc_num_truncate(&intp, intp.rdx);
|
|
|
|
s = zbc_num_sub(n, &intp, &fracp, 0);
|
|
|
|
if (s) goto err;
|
|
|
|
|
|
|
|
while (intp.len != 0) {
|
2018-12-18 17:00:35 +01:00
|
|
|
s = zbc_num_divmod(&intp, &base, &intp, &digit, 0);
|
2018-12-11 19:04:44 +01:00
|
|
|
if (s) goto err;
|
|
|
|
s = zbc_num_ulong(&digit, &dig);
|
|
|
|
if (s) goto err;
|
|
|
|
bc_vec_push(&stack, &dig);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < stack.len; ++i) {
|
|
|
|
ptr = bc_vec_item_rev(&stack, i);
|
|
|
|
print(*ptr, width, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!n->rdx) goto err;
|
|
|
|
|
|
|
|
for (radix = true; frac_len.len <= n->rdx; radix = false) {
|
2018-12-18 17:00:35 +01:00
|
|
|
s = zbc_num_mul(&fracp, &base, &fracp, n->rdx);
|
2018-12-11 19:04:44 +01:00
|
|
|
if (s) goto err;
|
|
|
|
s = zbc_num_ulong(&fracp, &dig);
|
|
|
|
if (s) goto err;
|
|
|
|
bc_num_ulong2num(&intp, dig);
|
|
|
|
s = zbc_num_sub(&fracp, &intp, &fracp, 0);
|
|
|
|
if (s) goto err;
|
|
|
|
print(dig, width, radix);
|
2018-12-18 17:00:35 +01:00
|
|
|
s = zbc_num_mul(&frac_len, &base, &frac_len, 0);
|
2018-12-11 19:04:44 +01:00
|
|
|
if (s) goto err;
|
|
|
|
}
|
2018-12-18 16:24:07 +01:00
|
|
|
err:
|
2018-12-11 19:04:44 +01:00
|
|
|
bc_num_free(&frac_len);
|
|
|
|
bc_num_free(&digit);
|
|
|
|
bc_num_free(&fracp);
|
|
|
|
bc_num_free(&intp);
|
|
|
|
bc_vec_free(&stack);
|
|
|
|
RETURN_STATUS(s);
|
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_num_printNum(...) (zxc_num_printNum(__VA_ARGS__) COMMA_SUCCESS)
|
2018-12-11 19:04:44 +01:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_num_printBase(BcNum *n)
|
2018-12-11 19:04:44 +01:00
|
|
|
{
|
|
|
|
BcStatus s;
|
2018-12-18 13:22:23 +01:00
|
|
|
size_t width;
|
2018-12-11 19:04:44 +01:00
|
|
|
BcNumDigitOp print;
|
|
|
|
bool neg = n->neg;
|
|
|
|
|
|
|
|
if (neg) {
|
|
|
|
bb_putchar('-');
|
|
|
|
G.prog.nchars++;
|
|
|
|
}
|
|
|
|
|
|
|
|
n->neg = false;
|
|
|
|
|
2018-12-31 19:42:13 +01:00
|
|
|
if (G.prog.ob_t <= 16) {
|
2018-12-11 19:04:44 +01:00
|
|
|
width = 1;
|
|
|
|
print = bc_num_printHex;
|
2018-12-17 21:05:09 +01:00
|
|
|
} else {
|
2018-12-18 13:22:23 +01:00
|
|
|
unsigned i = G.prog.ob_t - 1;
|
|
|
|
width = 0;
|
|
|
|
for (;;) {
|
|
|
|
width++;
|
|
|
|
i /= 10;
|
|
|
|
if (i == 0)
|
|
|
|
break;
|
|
|
|
}
|
2018-12-11 19:04:44 +01:00
|
|
|
print = bc_num_printDigits;
|
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_num_printNum(n, G.prog.ob_t, width, print);
|
2018-12-11 19:04:44 +01:00
|
|
|
n->neg = neg;
|
|
|
|
|
|
|
|
RETURN_STATUS(s);
|
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_num_printBase(...) (zxc_num_printBase(__VA_ARGS__) COMMA_SUCCESS)
|
2018-12-11 19:04:44 +01:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_num_print(BcNum *n, bool newline)
|
2018-12-11 19:04:44 +01:00
|
|
|
{
|
|
|
|
BcStatus s = BC_STATUS_SUCCESS;
|
|
|
|
|
|
|
|
bc_num_printNewline();
|
|
|
|
|
|
|
|
if (n->len == 0) {
|
|
|
|
bb_putchar('0');
|
|
|
|
++G.prog.nchars;
|
2018-12-17 21:05:09 +01:00
|
|
|
} else if (G.prog.ob_t == 10)
|
2018-12-11 19:04:44 +01:00
|
|
|
bc_num_printDecimal(n);
|
|
|
|
else
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_num_printBase(n);
|
2018-12-11 19:04:44 +01:00
|
|
|
|
|
|
|
if (newline) {
|
|
|
|
bb_putchar('\n');
|
|
|
|
G.prog.nchars = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
RETURN_STATUS(s);
|
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_num_print(...) (zxc_num_print(__VA_ARGS__) COMMA_SUCCESS)
|
2018-12-11 19:04:44 +01:00
|
|
|
|
2019-01-04 13:58:46 +01:00
|
|
|
#if !ENABLE_DC
|
|
|
|
// for bc, idx is always 0
|
|
|
|
#define xc_program_print(inst, idx) \
|
|
|
|
xc_program_print(inst)
|
|
|
|
#endif
|
|
|
|
static BC_STATUS xc_program_print(char inst, size_t idx)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-10 20:17:24 +01:00
|
|
|
BcStatus s;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcResult *r;
|
2018-12-10 12:33:40 +01:00
|
|
|
BcNum *num;
|
2019-01-04 13:58:46 +01:00
|
|
|
IF_NOT_DC(size_t idx = 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 17:09:01 +01:00
|
|
|
if (!STACK_HAS_MORE_THAN(&G.prog.results, idx))
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error_stack_has_too_few_elements());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:26:38 +01:00
|
|
|
r = bc_vec_item_rev(&G.prog.results, idx);
|
2019-01-04 13:58:46 +01:00
|
|
|
#if ENABLE_BC
|
|
|
|
if (inst == XC_INST_PRINT && r->t == BC_RESULT_VOID)
|
|
|
|
// void function's result on stack, ignore
|
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
|
|
|
#endif
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_program_num(r, &num);
|
2018-12-11 15:29:32 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (BC_PROG_NUM(r, num)) {
|
2019-01-04 13:58:46 +01:00
|
|
|
s = zxc_num_print(num, /*newline:*/ inst == XC_INST_PRINT);
|
2018-12-20 16:24:18 +01:00
|
|
|
#if ENABLE_BC
|
|
|
|
if (!s && IS_BC) bc_num_copy(&G.prog.last, num);
|
|
|
|
#endif
|
2018-12-18 16:24:07 +01:00
|
|
|
} else {
|
2018-12-10 12:33:40 +01:00
|
|
|
char *str;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 23:41:31 +01:00
|
|
|
idx = (r->t == XC_RESULT_STR) ? r->d.id.idx : num->rdx;
|
2018-12-27 18:23:58 +01:00
|
|
|
str = *xc_program_str(idx);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
if (inst == XC_INST_PRINT_STR) {
|
2019-01-02 05:03:53 +01:00
|
|
|
char *nl;
|
|
|
|
G.prog.nchars += printf("%s", str);
|
|
|
|
nl = strrchr(str, '\n');
|
|
|
|
if (nl)
|
|
|
|
G.prog.nchars = strlen(nl + 1);
|
2018-12-18 16:24:07 +01:00
|
|
|
} else {
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_printString(str);
|
2019-01-02 05:03:53 +01:00
|
|
|
if (inst == XC_INST_PRINT)
|
|
|
|
bb_putchar('\n');
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-04 13:58:46 +01:00
|
|
|
if (!s && inst != XC_INST_PRINT) bc_vec_pop(&G.prog.results);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2019-01-04 13:58:46 +01:00
|
|
|
#define zxc_program_print(...) (xc_program_print(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_program_negate(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcResult res, *ptr;
|
2018-12-24 17:15:34 +01:00
|
|
|
BcNum *num;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_prep(&ptr, &num);
|
2018-12-11 15:29:32 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2021-06-12 12:35:15 +02:00
|
|
|
bc_num_init_and_copy(&res.d.n, num);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (res.d.n.len) res.d.n.neg = !res.d.n.neg;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_retire(&res, XC_RESULT_TEMP);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_program_negate(...) (zxc_program_negate(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_program_logical(char inst)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcResult *opd1, *opd2, res;
|
|
|
|
BcNum *n1, *n2;
|
2018-12-12 00:50:23 +01:00
|
|
|
ssize_t cond;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_binOpPrep(&opd1, &n1, &opd2, &n2, false);
|
2018-12-11 19:37:00 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-11 19:29:35 +01:00
|
|
|
|
2018-12-09 11:44:20 +01:00
|
|
|
bc_num_init_DEF_SIZE(&res.d.n);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
if (inst == XC_INST_BOOL_AND)
|
2018-12-02 18:26:38 +01:00
|
|
|
cond = bc_num_cmp(n1, &G.prog.zero) && bc_num_cmp(n2, &G.prog.zero);
|
2018-12-24 04:26:07 +01:00
|
|
|
else if (inst == XC_INST_BOOL_OR)
|
2018-12-02 18:26:38 +01:00
|
|
|
cond = bc_num_cmp(n1, &G.prog.zero) || bc_num_cmp(n2, &G.prog.zero);
|
2018-11-03 11:00:21 -06:00
|
|
|
else {
|
2018-12-12 00:50:23 +01:00
|
|
|
cond = bc_num_cmp(n1, n2);
|
2018-11-03 11:00:21 -06:00
|
|
|
switch (inst) {
|
2018-12-24 04:26:07 +01:00
|
|
|
case XC_INST_REL_EQ:
|
2018-12-12 00:50:23 +01:00
|
|
|
cond = (cond == 0);
|
2018-12-11 19:29:35 +01:00
|
|
|
break;
|
2018-12-24 04:26:07 +01:00
|
|
|
case XC_INST_REL_LE:
|
2018-12-12 00:50:23 +01:00
|
|
|
cond = (cond <= 0);
|
2018-12-11 19:29:35 +01:00
|
|
|
break;
|
2018-12-24 04:26:07 +01:00
|
|
|
case XC_INST_REL_GE:
|
2018-12-12 00:50:23 +01:00
|
|
|
cond = (cond >= 0);
|
2018-12-11 19:29:35 +01:00
|
|
|
break;
|
2018-12-24 04:26:07 +01:00
|
|
|
case XC_INST_REL_LT:
|
2018-12-12 00:50:23 +01:00
|
|
|
cond = (cond < 0);
|
2018-12-11 19:29:35 +01:00
|
|
|
break;
|
2018-12-24 04:26:07 +01:00
|
|
|
case XC_INST_REL_GT:
|
2018-12-12 00:50:23 +01:00
|
|
|
cond = (cond > 0);
|
|
|
|
break;
|
2018-12-24 04:26:07 +01:00
|
|
|
default: // = case XC_INST_REL_NE:
|
2018-12-12 00:50:23 +01:00
|
|
|
//cond = (cond != 0); - not needed
|
2018-12-11 19:29:35 +01:00
|
|
|
break;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-11 19:29:35 +01:00
|
|
|
if (cond) bc_num_one(&res.d.n);
|
|
|
|
//else bc_num_zero(&res.d.n); - already is
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_binOpRetire(&res);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 19:37:00 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_program_logical(...) (zxc_program_logical(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_DC
|
2018-12-19 19:35:40 +01:00
|
|
|
static BC_STATUS zdc_program_assignStr(BcResult *r, BcVec *v, bool push)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcNum n2;
|
|
|
|
BcResult res;
|
|
|
|
|
|
|
|
memset(&n2, 0, sizeof(BcNum));
|
|
|
|
n2.rdx = res.d.id.idx = r->d.id.idx;
|
2018-12-24 23:41:31 +01:00
|
|
|
res.t = XC_RESULT_STR;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (!push) {
|
2018-12-19 17:09:01 +01:00
|
|
|
if (!STACK_HAS_MORE_THAN(&G.prog.results, 1))
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error_stack_has_too_few_elements());
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_vec_pop(v);
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_pop(&G.prog.results);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-21 23:13:48 +01:00
|
|
|
bc_result_pop_and_push(&res);
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_vec_push(v, &n2);
|
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-19 19:35:40 +01:00
|
|
|
#define zdc_program_assignStr(...) (zdc_program_assignStr(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
#endif // ENABLE_DC
|
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
static BC_STATUS zxc_program_popResultAndCopyToVar(char *name, BcType t)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcResult *ptr, r;
|
2019-01-25 14:24:03 +01:00
|
|
|
BcVec *vec;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcNum *n;
|
2019-01-25 14:24:03 +01:00
|
|
|
bool var = (t == BC_TYPE_VAR);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 17:09:01 +01:00
|
|
|
if (!STACK_HAS_MORE_THAN(&G.prog.results, 0))
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error_stack_has_too_few_elements());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:26:38 +01:00
|
|
|
ptr = bc_vec_top(&G.prog.results);
|
2019-01-25 14:24:03 +01:00
|
|
|
if ((ptr->t == XC_RESULT_ARRAY) == var)
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error_variable_is_wrong_type());
|
2019-01-25 14:24:03 +01:00
|
|
|
vec = xc_program_search(name, t);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_DC
|
2019-01-25 14:24:03 +01:00
|
|
|
if (ptr->t == XC_RESULT_STR) {
|
|
|
|
if (!var)
|
|
|
|
RETURN_STATUS(bc_error_variable_is_wrong_type());
|
|
|
|
RETURN_STATUS(zdc_program_assignStr(ptr, vec, true));
|
|
|
|
}
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_program_num(ptr, &n);
|
2018-12-11 15:29:32 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
// Do this once more to make sure that pointers were not invalidated.
|
2019-01-25 14:24:03 +01:00
|
|
|
vec = xc_program_search(name, t);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (var) {
|
2018-12-09 11:44:20 +01:00
|
|
|
bc_num_init_DEF_SIZE(&r.d.n);
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_copy(&r.d.n, n);
|
2018-12-18 16:24:07 +01:00
|
|
|
} else {
|
2019-01-25 14:24:03 +01:00
|
|
|
BcVec *v = (BcVec*) n;
|
|
|
|
bool ref, ref_size;
|
|
|
|
|
|
|
|
ref = (v->size == sizeof(BcVec) && t != BC_TYPE_ARRAY);
|
|
|
|
ref_size = (v->size == sizeof(uint8_t));
|
|
|
|
|
|
|
|
if (ref || (ref_size && t == BC_TYPE_REF)) {
|
|
|
|
bc_vec_init(&r.d.v, sizeof(uint8_t), NULL);
|
|
|
|
if (ref) {
|
|
|
|
size_t vidx, idx;
|
|
|
|
BcId id;
|
|
|
|
|
|
|
|
id.name = ptr->d.id.name;
|
|
|
|
v = xc_program_search(ptr->d.id.name, BC_TYPE_REF);
|
|
|
|
|
|
|
|
// Make sure the pointer was not invalidated.
|
|
|
|
vec = xc_program_search(name, t);
|
|
|
|
|
|
|
|
vidx = bc_map_find_exact(&G.prog.arr_map, &id);
|
|
|
|
//assert(vidx != BC_VEC_INVALID_IDX);
|
|
|
|
vidx = ((BcId*) bc_vec_item(&G.prog.arr_map, vidx))->idx;
|
|
|
|
idx = v->len - 1;
|
|
|
|
|
|
|
|
bc_vec_pushIndex(&r.d.v, vidx);
|
|
|
|
bc_vec_pushIndex(&r.d.v, idx);
|
|
|
|
}
|
|
|
|
// If we get here, we are copying a ref to a ref.
|
|
|
|
else bc_vec_npush(&r.d.v, v->len, v->v);
|
|
|
|
|
|
|
|
// We need to return early.
|
|
|
|
goto ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ref_size && t != BC_TYPE_REF)
|
|
|
|
v = xc_program_dereference(v);
|
|
|
|
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_array_init(&r.d.v, true);
|
2019-01-25 14:24:03 +01:00
|
|
|
bc_array_copy(&r.d.v, v);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2019-01-25 14:24:03 +01:00
|
|
|
ret:
|
|
|
|
bc_vec_push(vec, &r.d);
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_pop(&G.prog.results);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2019-01-04 00:21:29 +01:00
|
|
|
#define zxc_program_popResultAndCopyToVar(...) (zxc_program_popResultAndCopyToVar(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_program_assign(char inst)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcResult *left, *right, res;
|
2018-12-24 17:15:34 +01:00
|
|
|
BcNum *l, *r;
|
|
|
|
bool assign = (inst == XC_INST_ASSIGN);
|
|
|
|
bool ib, sc;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_binOpPrep(&left, &l, &right, &r, assign);
|
2018-12-11 19:04:44 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 23:41:31 +01:00
|
|
|
ib = left->t == XC_RESULT_IBASE;
|
|
|
|
sc = left->t == XC_RESULT_SCALE;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_DC
|
2018-12-24 23:41:31 +01:00
|
|
|
if (right->t == XC_RESULT_STR) {
|
2018-11-03 11:00:21 -06:00
|
|
|
BcVec *v;
|
|
|
|
|
2018-12-24 23:41:31 +01:00
|
|
|
if (left->t != XC_RESULT_VAR)
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(bc_error_variable_is_wrong_type());
|
2019-01-25 14:24:03 +01:00
|
|
|
v = xc_program_search(left->d.id.name, BC_TYPE_VAR);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 19:35:40 +01:00
|
|
|
RETURN_STATUS(zdc_program_assignStr(right, v, false));
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2019-01-04 13:58:46 +01:00
|
|
|
if (left->t == XC_RESULT_CONSTANT
|
|
|
|
|| left->t == XC_RESULT_TEMP
|
|
|
|
IF_BC(|| left->t == BC_RESULT_VOID)
|
|
|
|
) {
|
2019-01-01 21:50:14 +01:00
|
|
|
RETURN_STATUS(bc_error_bad_assignment());
|
2019-01-04 13:58:46 +01:00
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_BC
|
2018-11-03 11:00:21 -06:00
|
|
|
if (assign)
|
|
|
|
bc_num_copy(l, r);
|
2018-12-11 19:04:44 +01:00
|
|
|
else {
|
|
|
|
s = BC_STATUS_SUCCESS;
|
2018-12-27 18:23:58 +01:00
|
|
|
IF_ERROR_RETURN_POSSIBLE(s =) zxc_program_ops[inst - BC_INST_ASSIGN_POWER](l, r, l, G.prog.scale);
|
2018-12-11 19:04:44 +01:00
|
|
|
}
|
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-02 18:49:16 +01:00
|
|
|
#else
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_copy(l, r);
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 23:41:31 +01:00
|
|
|
if (ib || sc || left->t == XC_RESULT_OBASE) {
|
2022-02-06 19:53:10 +01:00
|
|
|
static const char *const msg[] ALIGN_PTR = {
|
2018-12-24 23:41:31 +01:00
|
|
|
"bad ibase; must be [2,16]", //XC_RESULT_IBASE
|
|
|
|
"bad obase; must be [2,"BC_MAX_OBASE_STR"]", //XC_RESULT_OBASE
|
|
|
|
"bad scale; must be [0,"BC_MAX_SCALE_STR"]", //XC_RESULT_SCALE
|
2018-12-04 20:05:28 +01:00
|
|
|
};
|
2018-11-03 11:00:21 -06:00
|
|
|
size_t *ptr;
|
2018-12-17 21:05:09 +01:00
|
|
|
size_t max;
|
|
|
|
unsigned long val;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
s = zbc_num_ulong(l, &val);
|
2018-12-11 19:04:44 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-24 23:41:31 +01:00
|
|
|
s = left->t - XC_RESULT_IBASE;
|
2018-11-03 11:00:21 -06:00
|
|
|
if (sc) {
|
|
|
|
max = BC_MAX_SCALE;
|
2018-12-02 18:26:38 +01:00
|
|
|
ptr = &G.prog.scale;
|
2018-12-17 21:05:09 +01:00
|
|
|
} else {
|
|
|
|
if (val < 2)
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(bc_error(msg[s]));
|
2018-11-03 11:00:21 -06:00
|
|
|
max = ib ? BC_NUM_MAX_IBASE : BC_MAX_OBASE;
|
2018-12-02 18:26:38 +01:00
|
|
|
ptr = ib ? &G.prog.ib_t : &G.prog.ob_t;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-04 20:05:28 +01:00
|
|
|
if (val > max)
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(bc_error(msg[s]));
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
*ptr = (size_t) val;
|
|
|
|
s = BC_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2021-06-12 12:35:15 +02:00
|
|
|
bc_num_init_and_copy(&res.d.n, l);
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_binOpRetire(&res);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_program_assign(...) (zxc_program_assign(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 20:57:17 +01:00
|
|
|
#if !ENABLE_DC
|
2018-12-27 18:23:58 +01:00
|
|
|
#define xc_program_pushVar(code, bgn, pop, copy) \
|
|
|
|
xc_program_pushVar(code, bgn)
|
2018-12-02 20:57:17 +01:00
|
|
|
// for bc, 'pop' and 'copy' are always false
|
|
|
|
#endif
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS xc_program_pushVar(char *code, size_t *bgn,
|
2018-11-03 11:00:21 -06:00
|
|
|
bool pop, bool copy)
|
|
|
|
{
|
|
|
|
BcResult r;
|
2018-12-27 18:23:58 +01:00
|
|
|
char *name = xc_program_name(code, bgn);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 23:41:31 +01:00
|
|
|
r.t = XC_RESULT_VAR;
|
2018-11-03 11:00:21 -06:00
|
|
|
r.d.id.name = name;
|
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_DC
|
2018-12-18 17:14:34 +01:00
|
|
|
if (pop || copy) {
|
2019-01-25 14:24:03 +01:00
|
|
|
BcVec *v = xc_program_search(name, BC_TYPE_VAR);
|
2018-12-02 20:57:17 +01:00
|
|
|
BcNum *num = bc_vec_top(v);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-18 17:14:34 +01:00
|
|
|
free(name);
|
2018-12-19 17:09:01 +01:00
|
|
|
if (!STACK_HAS_MORE_THAN(v, 1 - copy)) {
|
2018-12-18 17:14:34 +01:00
|
|
|
RETURN_STATUS(bc_error_stack_has_too_few_elements());
|
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-18 17:14:34 +01:00
|
|
|
if (!BC_PROG_STR(num)) {
|
2018-12-24 23:41:31 +01:00
|
|
|
r.t = XC_RESULT_TEMP;
|
2018-12-18 17:14:34 +01:00
|
|
|
bc_num_init_DEF_SIZE(&r.d.n);
|
|
|
|
bc_num_copy(&r.d.n, num);
|
|
|
|
} else {
|
2018-12-24 23:41:31 +01:00
|
|
|
r.t = XC_RESULT_STR;
|
2018-12-18 17:14:34 +01:00
|
|
|
r.d.id.idx = num->rdx;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-18 17:14:34 +01:00
|
|
|
|
|
|
|
if (!copy) bc_vec_pop(v);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
#endif // ENABLE_DC
|
|
|
|
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_push(&G.prog.results, &r);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 15:45:15 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_program_pushVar(...) (xc_program_pushVar(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-18 16:24:07 +01:00
|
|
|
static BC_STATUS zbc_program_pushArray(char *code, size_t *bgn, char inst)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s = BC_STATUS_SUCCESS;
|
|
|
|
BcResult r;
|
|
|
|
BcNum *num;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
r.d.id.name = xc_program_name(code, bgn);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
if (inst == XC_INST_ARRAY) {
|
2018-12-24 23:41:31 +01:00
|
|
|
r.t = XC_RESULT_ARRAY;
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_push(&G.prog.results, &r);
|
2018-12-18 16:24:07 +01:00
|
|
|
} else {
|
2018-11-03 11:00:21 -06:00
|
|
|
BcResult *operand;
|
|
|
|
unsigned long temp;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_prep(&operand, &num);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-11 15:29:32 +01:00
|
|
|
s = zbc_num_ulong(num, &temp);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
|
|
|
if (temp > BC_MAX_DIM) {
|
2018-12-07 15:50:14 +01:00
|
|
|
s = bc_error("array too long; must be [1,"BC_MAX_DIM_STR"]");
|
2018-11-03 11:00:21 -06:00
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
r.d.id.idx = (size_t) temp;
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_retire(&r, XC_RESULT_ARRAY_ELEM);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-18 16:24:07 +01:00
|
|
|
err:
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) free(r.d.id.name);
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_program_pushArray(...) (zbc_program_pushArray(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-11 15:29:32 +01:00
|
|
|
static BC_STATUS zbc_program_incdec(char inst)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcResult *ptr, res, copy;
|
2018-12-24 17:15:34 +01:00
|
|
|
BcNum *num;
|
2018-11-03 11:00:21 -06:00
|
|
|
char inst2 = inst;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_prep(&ptr, &num);
|
2018-12-11 15:29:32 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (inst == BC_INST_INC_POST || inst == BC_INST_DEC_POST) {
|
2018-12-24 23:41:31 +01:00
|
|
|
copy.t = XC_RESULT_TEMP;
|
2021-06-12 12:35:15 +02:00
|
|
|
bc_num_init_and_copy(©.d.n, num);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
res.t = BC_RESULT_ONE;
|
2018-12-24 17:15:34 +01:00
|
|
|
inst = (inst == BC_INST_INC_PRE || inst == BC_INST_INC_POST)
|
|
|
|
? BC_INST_ASSIGN_PLUS
|
|
|
|
: BC_INST_ASSIGN_MINUS;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_push(&G.prog.results, &res);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_assign(inst);
|
2018-12-11 19:04:44 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (inst2 == BC_INST_INC_POST || inst2 == BC_INST_DEC_POST) {
|
2018-12-21 23:13:48 +01:00
|
|
|
bc_result_pop_and_push(©);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_program_incdec(...) (zbc_program_incdec(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
static BC_STATUS zbc_program_call(char *code, size_t *idx)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcInstPtr ip;
|
2018-12-19 12:35:27 +01:00
|
|
|
size_t i, nparams;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcId *a;
|
2019-01-04 00:05:07 +01:00
|
|
|
BcFunc *func;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
nparams = xc_program_index(code, idx);
|
|
|
|
ip.func = xc_program_index(code, idx);
|
|
|
|
func = xc_program_func(ip.func);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-03 21:10:57 +01:00
|
|
|
if (func->code.len == 0) {
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error("undefined function"));
|
2018-12-03 21:10:57 +01:00
|
|
|
}
|
|
|
|
if (nparams != func->nparams) {
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error_fmt("function has %u parameters, but called with %u", func->nparams, nparams));
|
2018-12-03 21:10:57 +01:00
|
|
|
}
|
2019-01-04 00:05:07 +01:00
|
|
|
ip.inst_idx = 0;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
for (i = 0; i < nparams; ++i) {
|
2019-01-04 00:05:07 +01:00
|
|
|
BcResult *arg;
|
2018-12-10 20:41:05 +01:00
|
|
|
BcStatus s;
|
2019-01-25 14:24:03 +01:00
|
|
|
bool arr;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
a = bc_vec_item(&func->autos, nparams - 1 - i);
|
2018-12-02 18:26:38 +01:00
|
|
|
arg = bc_vec_top(&G.prog.results);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
arr = (a->idx == BC_TYPE_ARRAY || a->idx == BC_TYPE_REF);
|
|
|
|
|
|
|
|
if (arr != (arg->t == XC_RESULT_ARRAY) // array/variable mismatch
|
2019-01-04 00:05:07 +01:00
|
|
|
// || arg->t == XC_RESULT_STR - impossible, f("str") is not a legal syntax (strings are not bc expressions)
|
|
|
|
) {
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error_variable_is_wrong_type());
|
2019-01-04 00:05:07 +01:00
|
|
|
}
|
2019-01-25 14:24:03 +01:00
|
|
|
s = zxc_program_popResultAndCopyToVar(a->name, (BcType) a->idx);
|
2018-12-11 15:29:32 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-19 17:55:23 +01:00
|
|
|
a = bc_vec_item(&func->autos, i);
|
|
|
|
for (; i < func->autos.len; i++, a++) {
|
2018-12-02 19:27:48 +01:00
|
|
|
BcVec *v;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2019-01-25 14:24:03 +01:00
|
|
|
v = xc_program_search(a->name, (BcType) a->idx);
|
|
|
|
if (a->idx == BC_TYPE_VAR) {
|
2018-12-19 17:15:04 +01:00
|
|
|
BcNum n2;
|
|
|
|
bc_num_init_DEF_SIZE(&n2);
|
|
|
|
bc_vec_push(v, &n2);
|
2018-12-18 16:24:07 +01:00
|
|
|
} else {
|
2019-01-25 14:24:03 +01:00
|
|
|
//assert(a->idx == BC_TYPE_ARRAY);
|
2018-12-19 17:15:04 +01:00
|
|
|
BcVec v2;
|
|
|
|
bc_array_init(&v2, true);
|
|
|
|
bc_vec_push(v, &v2);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-19 12:35:27 +01:00
|
|
|
bc_vec_push(&G.prog.exestack, &ip);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_program_call(...) (zbc_program_call(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
static BC_STATUS zbc_program_return(char inst)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcResult res;
|
|
|
|
BcFunc *f;
|
2018-12-19 17:55:23 +01:00
|
|
|
BcId *a;
|
2018-11-03 11:00:21 -06:00
|
|
|
size_t i;
|
2018-12-19 12:35:27 +01:00
|
|
|
BcInstPtr *ip = bc_vec_top(&G.prog.exestack);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2019-01-04 13:58:46 +01:00
|
|
|
f = xc_program_func(ip->func);
|
|
|
|
|
|
|
|
res.t = XC_RESULT_TEMP;
|
2018-12-24 04:26:07 +01:00
|
|
|
if (inst == XC_INST_RET) {
|
2019-01-04 06:18:00 +01:00
|
|
|
// bc needs this for e.g. RESULT_CONSTANT ("return 5")
|
|
|
|
// because bc constants are per-function.
|
|
|
|
// TODO: maybe avoid if value is already RESULT_TEMP?
|
2018-12-10 20:41:05 +01:00
|
|
|
BcStatus s;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcNum *num;
|
2018-12-02 18:26:38 +01:00
|
|
|
BcResult *operand = bc_vec_top(&G.prog.results);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_program_num(operand, &num);
|
2018-12-11 15:29:32 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2021-06-12 12:35:15 +02:00
|
|
|
bc_num_init_and_copy(&res.d.n, num);
|
2019-01-04 00:34:52 +01:00
|
|
|
bc_vec_pop(&G.prog.results);
|
2018-12-18 16:24:07 +01:00
|
|
|
} else {
|
2019-01-04 13:58:46 +01:00
|
|
|
if (f->voidfunc)
|
|
|
|
res.t = BC_RESULT_VOID;
|
2018-12-09 11:44:20 +01:00
|
|
|
bc_num_init_DEF_SIZE(&res.d.n);
|
2018-12-09 12:04:44 +01:00
|
|
|
//bc_num_zero(&res.d.n); - already is
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2019-01-04 00:21:29 +01:00
|
|
|
bc_vec_push(&G.prog.results, &res);
|
|
|
|
|
|
|
|
bc_vec_pop(&G.prog.exestack);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
// We need to pop arguments as well, so this takes that into account.
|
2018-12-19 17:55:23 +01:00
|
|
|
a = (void*)f->autos.v;
|
|
|
|
for (i = 0; i < f->autos.len; i++, a++) {
|
2018-11-03 11:00:21 -06:00
|
|
|
BcVec *v;
|
2019-01-25 14:24:03 +01:00
|
|
|
v = xc_program_search(a->name, (BcType) a->idx);
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_vec_pop(v);
|
|
|
|
}
|
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-17 10:34:02 +01:00
|
|
|
#define zbc_program_return(...) (zbc_program_return(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
#endif // ENABLE_BC
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static unsigned long xc_program_scale(BcNum *n)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
return (unsigned long) n->rdx;
|
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static unsigned long xc_program_len(BcNum *n)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-10 12:57:01 +01:00
|
|
|
size_t len = n->len;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2021-02-26 14:05:28 +01:00
|
|
|
if (n->rdx != len)
|
|
|
|
// length(100): rdx 0 len 3, return 3
|
|
|
|
// length(0.01-0.01): rdx 2 len 0, return 2
|
|
|
|
// dc: 0.01 0.01 - Zp: rdx 2 len 0, return 1
|
|
|
|
return len != 0 ? len : (IS_BC ? n->rdx : 1);
|
|
|
|
|
|
|
|
// length(0): return 1
|
|
|
|
// length(0.000nnn): count nnn
|
2018-12-10 12:57:01 +01:00
|
|
|
for (;;) {
|
|
|
|
if (len == 0) break;
|
|
|
|
len--;
|
|
|
|
if (n->num[len] != 0) break;
|
|
|
|
}
|
2021-02-26 14:05:28 +01:00
|
|
|
return len + 1;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_program_builtin(char inst)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcResult *opnd;
|
2018-12-24 17:15:34 +01:00
|
|
|
BcNum *num;
|
2018-11-03 11:00:21 -06:00
|
|
|
BcResult res;
|
2018-12-24 17:15:34 +01:00
|
|
|
bool len = (inst == XC_INST_LENGTH);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 17:09:01 +01:00
|
|
|
if (!STACK_HAS_MORE_THAN(&G.prog.results, 0))
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(bc_error_stack_has_too_few_elements());
|
2018-12-02 18:26:38 +01:00
|
|
|
opnd = bc_vec_top(&G.prog.results);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_program_num(opnd, &num);
|
2018-12-11 19:04:44 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_DC
|
2018-12-04 20:05:28 +01:00
|
|
|
if (!BC_PROG_NUM(opnd, num) && !len)
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(bc_error_variable_is_wrong_type());
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-09 11:44:20 +01:00
|
|
|
bc_num_init_DEF_SIZE(&res.d.n);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 04:26:07 +01:00
|
|
|
if (inst == XC_INST_SQRT)
|
2018-12-18 16:24:07 +01:00
|
|
|
s = zbc_num_sqrt(num, &res.d.n, G.prog.scale);
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_BC
|
2021-02-26 14:05:28 +01:00
|
|
|
else if (len && opnd->t == XC_RESULT_ARRAY) {
|
2018-12-02 18:44:40 +01:00
|
|
|
bc_num_ulong2num(&res.d.n, (unsigned long) ((BcVec *) num)->len);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
|
|
|
#if ENABLE_DC
|
2021-02-26 14:05:28 +01:00
|
|
|
else if (len && !BC_PROG_NUM(opnd, num)) {
|
2018-11-03 11:00:21 -06:00
|
|
|
char **str;
|
2018-12-24 23:41:31 +01:00
|
|
|
size_t idx = opnd->t == XC_RESULT_STR ? opnd->d.id.idx : num->rdx;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
str = xc_program_str(idx);
|
2018-12-02 18:44:40 +01:00
|
|
|
bc_num_ulong2num(&res.d.n, strlen(*str));
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
else {
|
2021-02-26 14:05:28 +01:00
|
|
|
//TODO: length(.00) and scale(.00) should return 2, they return 1 and 0 now
|
|
|
|
//(don't forget to check that dc Z and X commands do not break)
|
2018-12-27 18:23:58 +01:00
|
|
|
bc_num_ulong2num(&res.d.n, len ? xc_program_len(num) : xc_program_scale(num));
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_retire(&res, XC_RESULT_TEMP);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_program_builtin(...) (zxc_program_builtin(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_DC
|
2018-12-19 19:35:40 +01:00
|
|
|
static BC_STATUS zdc_program_divmod(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcResult *opd1, *opd2, res, res2;
|
2018-12-24 17:15:34 +01:00
|
|
|
BcNum *n1, *n2;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_binOpPrep(&opd1, &n1, &opd2, &n2, false);
|
2018-12-11 19:04:44 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-09 11:44:20 +01:00
|
|
|
bc_num_init_DEF_SIZE(&res.d.n);
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_init(&res2.d.n, n2->len);
|
|
|
|
|
2018-12-11 19:12:13 +01:00
|
|
|
s = zbc_num_divmod(n1, n2, &res2.d.n, &res.d.n, G.prog.scale);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_binOpRetire(&res2);
|
2018-12-24 23:41:31 +01:00
|
|
|
res.t = XC_RESULT_TEMP;
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_push(&G.prog.results, &res);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-12-18 16:24:07 +01:00
|
|
|
err:
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_free(&res2.d.n);
|
|
|
|
bc_num_free(&res.d.n);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-19 19:35:40 +01:00
|
|
|
#define zdc_program_divmod(...) (zdc_program_divmod(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 19:35:40 +01:00
|
|
|
static BC_STATUS zdc_program_modexp(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcResult *r1, *r2, *r3, res;
|
|
|
|
BcNum *n1, *n2, *n3;
|
|
|
|
|
2018-12-19 17:09:01 +01:00
|
|
|
if (!STACK_HAS_MORE_THAN(&G.prog.results, 2))
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(bc_error_stack_has_too_few_elements());
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_binOpPrep(&r2, &n2, &r3, &n3, false);
|
2018-12-11 19:04:44 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:26:38 +01:00
|
|
|
r1 = bc_vec_item_rev(&G.prog.results, 2);
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_program_num(r1, &n1);
|
2018-12-11 19:04:44 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-12-04 20:05:28 +01:00
|
|
|
if (!BC_PROG_NUM(r1, n1))
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(bc_error_variable_is_wrong_type());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
// Make sure that the values have their pointers updated, if necessary.
|
2018-12-24 23:41:31 +01:00
|
|
|
if (r1->t == XC_RESULT_VAR || r1->t == XC_RESULT_ARRAY_ELEM) {
|
2018-11-03 11:00:21 -06:00
|
|
|
if (r1->t == r2->t) {
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_program_num(r2, &n2);
|
2018-12-11 19:04:44 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
if (r1->t == r3->t) {
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_program_num(r3, &n3);
|
2018-12-11 19:04:44 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bc_num_init(&res.d.n, n3->len);
|
2018-12-19 19:35:40 +01:00
|
|
|
s = zdc_num_modexp(n1, n2, n3, &res.d.n);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
|
|
|
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_pop(&G.prog.results);
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_binOpRetire(&res);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-12-18 16:24:07 +01:00
|
|
|
err:
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_free(&res.d.n);
|
2018-12-11 19:04:44 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-19 19:35:40 +01:00
|
|
|
#define zdc_program_modexp(...) (zdc_program_modexp(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 19:35:40 +01:00
|
|
|
static void dc_program_stackLen(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcResult res;
|
2018-12-02 18:26:38 +01:00
|
|
|
size_t len = G.prog.results.len;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-24 23:41:31 +01:00
|
|
|
res.t = XC_RESULT_TEMP;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-09 11:44:20 +01:00
|
|
|
bc_num_init_DEF_SIZE(&res.d.n);
|
2018-12-02 18:44:40 +01:00
|
|
|
bc_num_ulong2num(&res.d.n, len);
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_push(&G.prog.results, &res);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-19 19:35:40 +01:00
|
|
|
static BC_STATUS zdc_program_asciify(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcResult *r, res;
|
2018-12-09 13:21:54 +01:00
|
|
|
BcNum *num, n;
|
2018-12-20 20:34:09 +01:00
|
|
|
char **strs;
|
|
|
|
char *str;
|
|
|
|
char c;
|
|
|
|
size_t idx;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 17:09:01 +01:00
|
|
|
if (!STACK_HAS_MORE_THAN(&G.prog.results, 0))
|
2018-12-11 20:57:53 +01:00
|
|
|
RETURN_STATUS(bc_error_stack_has_too_few_elements());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-30 15:56:36 +01:00
|
|
|
r = bc_vec_top(&G.prog.results);
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_program_num(r, &num);
|
2018-12-11 20:57:53 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (BC_PROG_NUM(r, num)) {
|
2018-12-22 21:52:30 +01:00
|
|
|
unsigned long val;
|
2018-12-18 17:00:35 +01:00
|
|
|
BcNum strmb;
|
2018-12-21 22:36:04 +01:00
|
|
|
BcDig strmb_digs[ULONG_NUM_BUFSIZE];
|
2018-12-18 17:00:35 +01:00
|
|
|
|
2018-12-09 11:44:20 +01:00
|
|
|
bc_num_init_DEF_SIZE(&n);
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_copy(&n, num);
|
|
|
|
bc_num_truncate(&n, n.rdx);
|
|
|
|
|
2018-12-21 22:36:04 +01:00
|
|
|
strmb.cap = ARRAY_SIZE(strmb_digs);
|
|
|
|
strmb.num = strmb_digs;
|
2018-12-18 17:00:35 +01:00
|
|
|
bc_num_ulong2num(&strmb, 0x100);
|
|
|
|
|
2018-12-30 15:56:36 +01:00
|
|
|
s = zbc_num_mod(&n, &strmb, &n, 0);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto num_err;
|
2018-12-11 15:29:32 +01:00
|
|
|
s = zbc_num_ulong(&n, &val);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto num_err;
|
|
|
|
|
|
|
|
c = (char) val;
|
|
|
|
|
|
|
|
bc_num_free(&n);
|
2018-12-18 16:24:07 +01:00
|
|
|
} else {
|
2018-12-20 20:34:09 +01:00
|
|
|
char *sp;
|
2018-12-24 23:41:31 +01:00
|
|
|
idx = (r->t == XC_RESULT_STR) ? r->d.id.idx : num->rdx;
|
2018-12-27 18:23:58 +01:00
|
|
|
sp = *xc_program_str(idx);
|
2018-12-20 20:34:09 +01:00
|
|
|
c = sp[0];
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-20 20:34:09 +01:00
|
|
|
strs = (void*)G.prog.strs.v;
|
|
|
|
for (idx = 0; idx < G.prog.strs.len; idx++) {
|
|
|
|
if (strs[idx][0] == c && strs[idx][1] == '\0') {
|
|
|
|
goto dup;
|
|
|
|
}
|
|
|
|
}
|
2018-12-10 20:17:24 +01:00
|
|
|
str = xzalloc(2);
|
2018-11-03 11:00:21 -06:00
|
|
|
str[0] = c;
|
2018-12-10 20:17:24 +01:00
|
|
|
//str[1] = '\0'; - already is
|
2019-09-05 10:50:13 +02:00
|
|
|
idx = bc_vec_push(&G.prog.strs, &str);
|
|
|
|
// Add an empty function so that if zdc_program_execStr ever needs to
|
|
|
|
// parse the string into code (from the 'x' command) there's somewhere
|
|
|
|
// to store the bytecode.
|
|
|
|
xc_program_add_fn();
|
2018-12-20 20:34:09 +01:00
|
|
|
dup:
|
2018-12-24 23:41:31 +01:00
|
|
|
res.t = XC_RESULT_STR;
|
2018-12-20 20:34:09 +01:00
|
|
|
res.d.id.idx = idx;
|
2018-12-21 23:13:48 +01:00
|
|
|
bc_result_pop_and_push(&res);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 20:57:53 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-12-18 16:24:07 +01:00
|
|
|
num_err:
|
2018-11-03 11:00:21 -06:00
|
|
|
bc_num_free(&n);
|
2018-12-11 20:57:53 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-19 19:35:40 +01:00
|
|
|
#define zdc_program_asciify(...) (zdc_program_asciify(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 19:35:40 +01:00
|
|
|
static BC_STATUS zdc_program_printStream(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcResult *r;
|
2018-12-19 19:35:40 +01:00
|
|
|
BcNum *n;
|
2018-11-03 11:00:21 -06:00
|
|
|
size_t idx;
|
|
|
|
|
2018-12-19 17:09:01 +01:00
|
|
|
if (!STACK_HAS_MORE_THAN(&G.prog.results, 0))
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error_stack_has_too_few_elements());
|
2018-12-02 18:26:38 +01:00
|
|
|
r = bc_vec_top(&G.prog.results);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_program_num(r, &n);
|
2018-12-11 15:29:32 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-17 21:14:05 +01:00
|
|
|
if (BC_PROG_NUM(r, n)) {
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_num_printNum(n, 0x100, 1, dc_num_printChar);
|
2018-12-17 21:14:05 +01:00
|
|
|
} else {
|
2018-12-19 19:35:40 +01:00
|
|
|
char *str;
|
2018-12-24 23:41:31 +01:00
|
|
|
idx = (r->t == XC_RESULT_STR) ? r->d.id.idx : n->rdx;
|
2018-12-27 18:23:58 +01:00
|
|
|
str = *xc_program_str(idx);
|
2021-02-03 20:47:14 +01:00
|
|
|
fputs_stdout(str);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-19 19:35:40 +01:00
|
|
|
#define zdc_program_printStream(...) (zdc_program_printStream(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 19:35:40 +01:00
|
|
|
static BC_STATUS zdc_program_nquit(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s;
|
|
|
|
BcResult *opnd;
|
2018-12-24 17:15:34 +01:00
|
|
|
BcNum *num;
|
2018-11-03 11:00:21 -06:00
|
|
|
unsigned long val;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_prep(&opnd, &num);
|
2018-12-11 15:29:32 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
s = zbc_num_ulong(num, &val);
|
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_pop(&G.prog.results);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 12:35:27 +01:00
|
|
|
if (G.prog.exestack.len < val)
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(bc_error_stack_has_too_few_elements());
|
2018-12-19 12:35:27 +01:00
|
|
|
if (G.prog.exestack.len == val) {
|
2018-12-06 23:06:57 +01:00
|
|
|
QUIT_OR_RETURN_TO_MAIN;
|
2018-12-06 00:29:22 +01:00
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 12:35:27 +01:00
|
|
|
bc_vec_npop(&G.prog.exestack, val);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-11 15:29:32 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-19 19:35:40 +01:00
|
|
|
#define zdc_program_nquit(...) (zdc_program_nquit(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-19 19:35:40 +01:00
|
|
|
static BC_STATUS zdc_program_execStr(char *code, size_t *bgn, bool cond)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcStatus s = BC_STATUS_SUCCESS;
|
|
|
|
BcResult *r;
|
|
|
|
BcFunc *f;
|
|
|
|
BcInstPtr ip;
|
|
|
|
size_t fidx, sidx;
|
|
|
|
|
2018-12-19 17:09:01 +01:00
|
|
|
if (!STACK_HAS_MORE_THAN(&G.prog.results, 0))
|
2018-12-12 21:39:10 +01:00
|
|
|
RETURN_STATUS(bc_error_stack_has_too_few_elements());
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:26:38 +01:00
|
|
|
r = bc_vec_top(&G.prog.results);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (cond) {
|
2018-12-09 02:54:06 +01:00
|
|
|
BcNum *n = n; // for compiler
|
|
|
|
bool exec;
|
|
|
|
char *name;
|
2018-12-27 18:23:58 +01:00
|
|
|
char *then_name = xc_program_name(code, bgn);
|
2018-12-09 02:54:06 +01:00
|
|
|
char *else_name = NULL;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-26 20:02:09 +01:00
|
|
|
if (code[*bgn] == '\0')
|
2018-11-03 11:00:21 -06:00
|
|
|
(*bgn) += 1;
|
|
|
|
else
|
2018-12-27 18:23:58 +01:00
|
|
|
else_name = xc_program_name(code, bgn);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
exec = r->d.n.len != 0;
|
2018-12-09 02:54:06 +01:00
|
|
|
name = then_name;
|
|
|
|
if (!exec && else_name != NULL) {
|
2018-11-03 11:00:21 -06:00
|
|
|
exec = true;
|
|
|
|
name = else_name;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (exec) {
|
2018-12-02 19:27:48 +01:00
|
|
|
BcVec *v;
|
2019-01-25 14:24:03 +01:00
|
|
|
v = xc_program_search(name, BC_TYPE_VAR);
|
2018-11-03 11:00:21 -06:00
|
|
|
n = bc_vec_top(v);
|
|
|
|
}
|
|
|
|
|
|
|
|
free(then_name);
|
|
|
|
free(else_name);
|
|
|
|
|
|
|
|
if (!exec) goto exit;
|
|
|
|
if (!BC_PROG_STR(n)) {
|
2018-12-05 16:03:46 +01:00
|
|
|
s = bc_error_variable_is_wrong_type();
|
2018-11-03 11:00:21 -06:00
|
|
|
goto exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
sidx = n->rdx;
|
2018-12-09 02:54:06 +01:00
|
|
|
} else {
|
2018-12-24 23:41:31 +01:00
|
|
|
if (r->t == XC_RESULT_STR) {
|
2018-11-03 11:00:21 -06:00
|
|
|
sidx = r->d.id.idx;
|
2018-12-24 23:41:31 +01:00
|
|
|
} else if (r->t == XC_RESULT_VAR) {
|
2018-12-09 02:54:06 +01:00
|
|
|
BcNum *n;
|
2018-12-29 14:52:30 +01:00
|
|
|
s = zxc_program_num(r, &n);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s || !BC_PROG_STR(n)) goto exit;
|
|
|
|
sidx = n->rdx;
|
2018-12-09 02:54:06 +01:00
|
|
|
} else
|
2019-09-05 10:46:22 +02:00
|
|
|
goto exit_nopop;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
fidx = sidx + BC_PROG_REQ_FUNCS;
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
f = xc_program_func(fidx);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
|
|
|
if (f->code.len == 0) {
|
2018-12-25 19:38:13 +01:00
|
|
|
BcParse sv_parse;
|
2018-12-19 19:35:40 +01:00
|
|
|
char *str;
|
|
|
|
|
2018-12-25 19:38:13 +01:00
|
|
|
sv_parse = G.prs; // struct copy
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_create(fidx);
|
|
|
|
str = *xc_program_str(sidx);
|
|
|
|
s = zxc_parse_text_init(str);
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-22 18:04:08 +01:00
|
|
|
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zdc_parse_exprs_until_eof();
|
2018-11-03 11:00:21 -06:00
|
|
|
if (s) goto err;
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_push(DC_INST_POP_EXEC);
|
2018-12-25 22:32:41 +01:00
|
|
|
if (G.prs.lex != XC_LEX_EOF)
|
2018-12-05 16:03:46 +01:00
|
|
|
s = bc_error_bad_expression();
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_free();
|
2018-12-25 19:38:13 +01:00
|
|
|
G.prs = sv_parse; // struct copy
|
|
|
|
if (s) {
|
2018-12-19 19:35:40 +01:00
|
|
|
err:
|
|
|
|
bc_vec_pop_all(&f->code);
|
|
|
|
goto exit;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-21 23:01:26 +01:00
|
|
|
ip.inst_idx = 0;
|
2018-11-03 11:00:21 -06:00
|
|
|
ip.func = fidx;
|
|
|
|
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_pop(&G.prog.results);
|
2018-12-19 12:35:27 +01:00
|
|
|
bc_vec_push(&G.prog.exestack, &ip);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-12 21:39:10 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-12-18 16:24:07 +01:00
|
|
|
exit:
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_pop(&G.prog.results);
|
2019-09-05 10:46:22 +02:00
|
|
|
exit_nopop:
|
2018-12-12 21:39:10 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-19 19:35:40 +01:00
|
|
|
#define zdc_program_execStr(...) (zdc_program_execStr(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
#endif // ENABLE_DC
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_program_pushGlobal(char inst)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcResult res;
|
|
|
|
unsigned long val;
|
|
|
|
|
2018-12-24 23:41:31 +01:00
|
|
|
res.t = inst - XC_INST_IBASE + XC_RESULT_IBASE;
|
2018-12-24 04:26:07 +01:00
|
|
|
if (inst == XC_INST_IBASE)
|
2018-12-02 18:26:38 +01:00
|
|
|
val = (unsigned long) G.prog.ib_t;
|
2018-12-24 04:26:07 +01:00
|
|
|
else if (inst == XC_INST_SCALE)
|
2018-12-02 18:26:38 +01:00
|
|
|
val = (unsigned long) G.prog.scale;
|
2018-11-03 11:00:21 -06:00
|
|
|
else
|
2018-12-02 18:26:38 +01:00
|
|
|
val = (unsigned long) G.prog.ob_t;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-09 11:44:20 +01:00
|
|
|
bc_num_init_DEF_SIZE(&res.d.n);
|
2018-12-02 18:44:40 +01:00
|
|
|
bc_num_ulong2num(&res.d.n, val);
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_push(&G.prog.results, &res);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_program_exec(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
|
|
|
BcResult r, *ptr;
|
2018-12-19 12:35:27 +01:00
|
|
|
BcInstPtr *ip = bc_vec_top(&G.prog.exestack);
|
2018-12-27 18:23:58 +01:00
|
|
|
BcFunc *func = xc_program_func(ip->func);
|
2018-11-03 11:00:21 -06:00
|
|
|
char *code = func->code.v;
|
|
|
|
|
2018-12-21 20:29:34 +01:00
|
|
|
dbg_exec("func:%zd bytes:%zd ip:%zd results.len:%d",
|
2018-12-21 23:01:26 +01:00
|
|
|
ip->func, func->code.len, ip->inst_idx, G.prog.results.len);
|
|
|
|
while (ip->inst_idx < func->code.len) {
|
2018-12-10 20:17:24 +01:00
|
|
|
BcStatus s = BC_STATUS_SUCCESS;
|
2018-12-21 23:01:26 +01:00
|
|
|
char inst = code[ip->inst_idx++];
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-21 23:01:26 +01:00
|
|
|
dbg_exec("inst at %zd:%d results.len:%d", ip->inst_idx - 1, inst, G.prog.results.len);
|
2018-11-03 11:00:21 -06:00
|
|
|
switch (inst) {
|
2019-01-04 06:18:00 +01:00
|
|
|
case XC_INST_RET:
|
|
|
|
if (IS_DC) { // end of '?' reached
|
|
|
|
bc_vec_pop(&G.prog.exestack);
|
|
|
|
goto read_updated_ip;
|
|
|
|
}
|
|
|
|
// bc: fall through
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_BC
|
2019-01-04 06:18:00 +01:00
|
|
|
case BC_INST_RET0:
|
|
|
|
dbg_exec("BC_INST_RET[0]:");
|
|
|
|
s = zbc_program_return(inst);
|
|
|
|
goto read_updated_ip;
|
2018-12-26 19:24:15 +01:00
|
|
|
case BC_INST_JUMP_ZERO: {
|
|
|
|
BcNum *num;
|
|
|
|
bool zero;
|
|
|
|
dbg_exec("BC_INST_JUMP_ZERO:");
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_prep(&ptr, &num);
|
2018-12-26 19:24:15 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
zero = (bc_num_cmp(num, &G.prog.zero) == 0);
|
|
|
|
bc_vec_pop(&G.prog.results);
|
|
|
|
if (!zero) {
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_index(code, &ip->inst_idx);
|
2018-11-03 11:00:21 -06:00
|
|
|
break;
|
|
|
|
}
|
2018-12-26 19:24:15 +01:00
|
|
|
// else: fall through
|
|
|
|
}
|
|
|
|
case BC_INST_JUMP: {
|
2018-12-27 18:23:58 +01:00
|
|
|
size_t idx = xc_program_index(code, &ip->inst_idx);
|
2018-12-26 19:24:15 +01:00
|
|
|
size_t *addr = bc_vec_item(&func->labels, idx);
|
|
|
|
dbg_exec("BC_INST_JUMP: to %ld", (long)*addr);
|
|
|
|
ip->inst_idx = *addr;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case BC_INST_CALL:
|
|
|
|
dbg_exec("BC_INST_CALL:");
|
|
|
|
s = zbc_program_call(code, &ip->inst_idx);
|
|
|
|
goto read_updated_ip;
|
|
|
|
case BC_INST_INC_PRE:
|
|
|
|
case BC_INST_DEC_PRE:
|
|
|
|
case BC_INST_INC_POST:
|
|
|
|
case BC_INST_DEC_POST:
|
|
|
|
dbg_exec("BC_INST_INCDEC:");
|
|
|
|
s = zbc_program_incdec(inst);
|
|
|
|
break;
|
|
|
|
case BC_INST_HALT:
|
|
|
|
dbg_exec("BC_INST_HALT:");
|
|
|
|
QUIT_OR_RETURN_TO_MAIN;
|
|
|
|
break;
|
|
|
|
case XC_INST_BOOL_OR:
|
|
|
|
case XC_INST_BOOL_AND:
|
2018-11-03 11:00:21 -06:00
|
|
|
#endif // ENABLE_BC
|
2018-12-26 19:24:15 +01:00
|
|
|
case XC_INST_REL_EQ:
|
|
|
|
case XC_INST_REL_LE:
|
|
|
|
case XC_INST_REL_GE:
|
|
|
|
case XC_INST_REL_NE:
|
|
|
|
case XC_INST_REL_LT:
|
|
|
|
case XC_INST_REL_GT:
|
|
|
|
dbg_exec("BC_INST_BOOL:");
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_logical(inst);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case XC_INST_READ:
|
|
|
|
dbg_exec("XC_INST_READ:");
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_read();
|
2018-12-26 19:24:15 +01:00
|
|
|
goto read_updated_ip;
|
|
|
|
case XC_INST_VAR:
|
|
|
|
dbg_exec("XC_INST_VAR:");
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_pushVar(code, &ip->inst_idx, false, false);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case XC_INST_ARRAY_ELEM:
|
|
|
|
case XC_INST_ARRAY:
|
|
|
|
dbg_exec("XC_INST_ARRAY[_ELEM]:");
|
|
|
|
s = zbc_program_pushArray(code, &ip->inst_idx, inst);
|
|
|
|
break;
|
2018-12-24 00:50:32 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-26 19:24:15 +01:00
|
|
|
case BC_INST_LAST:
|
|
|
|
dbg_exec("BC_INST_LAST:");
|
|
|
|
r.t = BC_RESULT_LAST;
|
|
|
|
bc_vec_push(&G.prog.results, &r);
|
|
|
|
break;
|
2018-12-24 00:50:32 +01:00
|
|
|
#endif
|
2018-12-26 19:24:15 +01:00
|
|
|
case XC_INST_IBASE:
|
|
|
|
case XC_INST_OBASE:
|
|
|
|
case XC_INST_SCALE:
|
|
|
|
dbg_exec("XC_INST_internalvar(%d):", inst - XC_INST_IBASE);
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_pushGlobal(inst);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case XC_INST_SCALE_FUNC:
|
|
|
|
case XC_INST_LENGTH:
|
|
|
|
case XC_INST_SQRT:
|
|
|
|
dbg_exec("BC_INST_builtin:");
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_builtin(inst);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case XC_INST_NUM:
|
|
|
|
dbg_exec("XC_INST_NUM:");
|
|
|
|
r.t = XC_RESULT_CONSTANT;
|
2018-12-27 18:23:58 +01:00
|
|
|
r.d.id.idx = xc_program_index(code, &ip->inst_idx);
|
2018-12-26 19:24:15 +01:00
|
|
|
bc_vec_push(&G.prog.results, &r);
|
|
|
|
break;
|
|
|
|
case XC_INST_POP:
|
|
|
|
dbg_exec("XC_INST_POP:");
|
|
|
|
if (!STACK_HAS_MORE_THAN(&G.prog.results, 0))
|
|
|
|
s = bc_error_stack_has_too_few_elements();
|
|
|
|
else
|
|
|
|
bc_vec_pop(&G.prog.results);
|
|
|
|
break;
|
|
|
|
case XC_INST_PRINT:
|
|
|
|
case XC_INST_PRINT_POP:
|
|
|
|
case XC_INST_PRINT_STR:
|
2019-01-04 13:58:46 +01:00
|
|
|
dbg_exec("XC_INST_PRINTxyz(%d):", inst - XC_INST_PRINT);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_print(inst, 0);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case XC_INST_STR:
|
|
|
|
dbg_exec("XC_INST_STR:");
|
|
|
|
r.t = XC_RESULT_STR;
|
2018-12-27 18:23:58 +01:00
|
|
|
r.d.id.idx = xc_program_index(code, &ip->inst_idx);
|
2018-12-26 19:24:15 +01:00
|
|
|
bc_vec_push(&G.prog.results, &r);
|
|
|
|
break;
|
|
|
|
case XC_INST_POWER:
|
|
|
|
case XC_INST_MULTIPLY:
|
|
|
|
case XC_INST_DIVIDE:
|
|
|
|
case XC_INST_MODULUS:
|
|
|
|
case XC_INST_PLUS:
|
|
|
|
case XC_INST_MINUS:
|
|
|
|
dbg_exec("BC_INST_binaryop:");
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_op(inst);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
case XC_INST_BOOL_NOT: {
|
|
|
|
BcNum *num;
|
|
|
|
dbg_exec("XC_INST_BOOL_NOT:");
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_prep(&ptr, &num);
|
2018-12-26 19:24:15 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
|
|
|
bc_num_init_DEF_SIZE(&r.d.n);
|
|
|
|
if (bc_num_cmp(num, &G.prog.zero) == 0)
|
|
|
|
bc_num_one(&r.d.n);
|
|
|
|
//else bc_num_zero(&r.d.n); - already is
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_retire(&r, XC_RESULT_TEMP);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case XC_INST_NEG:
|
|
|
|
dbg_exec("XC_INST_NEG:");
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_negate();
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-26 19:24:15 +01:00
|
|
|
case BC_INST_ASSIGN_POWER:
|
|
|
|
case BC_INST_ASSIGN_MULTIPLY:
|
|
|
|
case BC_INST_ASSIGN_DIVIDE:
|
|
|
|
case BC_INST_ASSIGN_MODULUS:
|
|
|
|
case BC_INST_ASSIGN_PLUS:
|
|
|
|
case BC_INST_ASSIGN_MINUS:
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-12-26 19:24:15 +01:00
|
|
|
case XC_INST_ASSIGN:
|
|
|
|
dbg_exec("BC_INST_ASSIGNxyz:");
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_assign(inst);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_DC
|
2018-12-26 19:24:15 +01:00
|
|
|
case DC_INST_POP_EXEC:
|
|
|
|
dbg_exec("DC_INST_POP_EXEC:");
|
|
|
|
bc_vec_pop(&G.prog.exestack);
|
|
|
|
goto read_updated_ip;
|
|
|
|
case DC_INST_MODEXP:
|
|
|
|
dbg_exec("DC_INST_MODEXP:");
|
|
|
|
s = zdc_program_modexp();
|
|
|
|
break;
|
|
|
|
case DC_INST_DIVMOD:
|
|
|
|
dbg_exec("DC_INST_DIVMOD:");
|
|
|
|
s = zdc_program_divmod();
|
|
|
|
break;
|
|
|
|
case DC_INST_EXECUTE:
|
|
|
|
case DC_INST_EXEC_COND:
|
|
|
|
dbg_exec("DC_INST_EXEC[_COND]:");
|
|
|
|
s = zdc_program_execStr(code, &ip->inst_idx, inst == DC_INST_EXEC_COND);
|
|
|
|
goto read_updated_ip;
|
|
|
|
case DC_INST_PRINT_STACK: {
|
|
|
|
size_t idx;
|
|
|
|
dbg_exec("DC_INST_PRINT_STACK:");
|
|
|
|
for (idx = 0; idx < G.prog.results.len; ++idx) {
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_print(XC_INST_PRINT, idx);
|
2018-12-26 19:24:15 +01:00
|
|
|
if (s) break;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case DC_INST_CLEAR_STACK:
|
|
|
|
dbg_exec("DC_INST_CLEAR_STACK:");
|
|
|
|
bc_vec_pop_all(&G.prog.results);
|
|
|
|
break;
|
|
|
|
case DC_INST_STACK_LEN:
|
|
|
|
dbg_exec("DC_INST_STACK_LEN:");
|
|
|
|
dc_program_stackLen();
|
|
|
|
break;
|
|
|
|
case DC_INST_DUPLICATE:
|
|
|
|
dbg_exec("DC_INST_DUPLICATE:");
|
|
|
|
if (!STACK_HAS_MORE_THAN(&G.prog.results, 0))
|
|
|
|
RETURN_STATUS(bc_error_stack_has_too_few_elements());
|
|
|
|
ptr = bc_vec_top(&G.prog.results);
|
|
|
|
dc_result_copy(&r, ptr);
|
|
|
|
bc_vec_push(&G.prog.results, &r);
|
|
|
|
break;
|
|
|
|
case DC_INST_SWAP: {
|
|
|
|
BcResult *ptr2;
|
|
|
|
dbg_exec("DC_INST_SWAP:");
|
|
|
|
if (!STACK_HAS_MORE_THAN(&G.prog.results, 1))
|
|
|
|
RETURN_STATUS(bc_error_stack_has_too_few_elements());
|
|
|
|
ptr = bc_vec_item_rev(&G.prog.results, 0);
|
|
|
|
ptr2 = bc_vec_item_rev(&G.prog.results, 1);
|
|
|
|
memcpy(&r, ptr, sizeof(BcResult));
|
|
|
|
memcpy(ptr, ptr2, sizeof(BcResult));
|
|
|
|
memcpy(ptr2, &r, sizeof(BcResult));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case DC_INST_ASCIIFY:
|
|
|
|
dbg_exec("DC_INST_ASCIIFY:");
|
|
|
|
s = zdc_program_asciify();
|
|
|
|
break;
|
|
|
|
case DC_INST_PRINT_STREAM:
|
|
|
|
dbg_exec("DC_INST_PRINT_STREAM:");
|
|
|
|
s = zdc_program_printStream();
|
|
|
|
break;
|
|
|
|
case DC_INST_LOAD:
|
|
|
|
case DC_INST_PUSH_VAR: {
|
|
|
|
bool copy = inst == DC_INST_LOAD;
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_pushVar(code, &ip->inst_idx, true, copy);
|
2018-12-26 19:24:15 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case DC_INST_PUSH_TO_VAR: {
|
2018-12-27 18:23:58 +01:00
|
|
|
char *name = xc_program_name(code, &ip->inst_idx);
|
2019-01-25 14:24:03 +01:00
|
|
|
s = zxc_program_popResultAndCopyToVar(name, BC_TYPE_VAR);
|
2018-12-26 19:24:15 +01:00
|
|
|
free(name);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case DC_INST_QUIT:
|
|
|
|
dbg_exec("DC_INST_QUIT:");
|
|
|
|
if (G.prog.exestack.len <= 2)
|
|
|
|
QUIT_OR_RETURN_TO_MAIN;
|
|
|
|
bc_vec_npop(&G.prog.exestack, 2);
|
|
|
|
goto read_updated_ip;
|
|
|
|
case DC_INST_NQUIT:
|
|
|
|
dbg_exec("DC_INST_NQUIT:");
|
|
|
|
s = zdc_program_nquit();
|
|
|
|
//goto read_updated_ip; - just fall through to it
|
2018-11-03 11:00:21 -06:00
|
|
|
#endif // ENABLE_DC
|
2018-12-19 12:35:27 +01:00
|
|
|
read_updated_ip:
|
2018-12-26 19:24:15 +01:00
|
|
|
// Instruction stack has changed, read new pointers
|
|
|
|
ip = bc_vec_top(&G.prog.exestack);
|
2018-12-27 18:23:58 +01:00
|
|
|
func = xc_program_func(ip->func);
|
2018-12-26 19:24:15 +01:00
|
|
|
code = func->code.v;
|
|
|
|
dbg_exec("func:%zd bytes:%zd ip:%zd", ip->func, func->code.len, ip->inst_idx);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-04 19:11:02 +01:00
|
|
|
if (s || G_interrupt) {
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_reset();
|
2018-12-12 21:39:10 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-12-04 19:11:02 +01:00
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-17 01:22:53 +01:00
|
|
|
fflush_and_check();
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
|
|
|
|
2018-12-12 21:39:10 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_SUCCESS);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_program_exec(...) (zxc_program_exec(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_vm_process(const char *text)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-14 17:51:17 +01:00
|
|
|
BcStatus s;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-14 17:51:17 +01:00
|
|
|
dbg_lex_enter("%s:%d entered", __func__, __LINE__);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_parse_text_init(text); // does the first zxc_lex_next()
|
2018-12-12 21:39:10 +01:00
|
|
|
if (s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
while (G.prs.lex != XC_LEX_EOF) {
|
2018-12-22 19:23:46 +01:00
|
|
|
BcInstPtr *ip;
|
|
|
|
BcFunc *f;
|
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
dbg_lex("%s:%d G.prs.lex:%d, parsing...", __func__, __LINE__, G.prs.lex);
|
2018-12-22 06:00:25 +01:00
|
|
|
if (IS_BC) {
|
2018-12-25 18:37:52 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zbc_parse_stmt_or_funcdef();
|
2018-12-25 18:37:52 +01:00
|
|
|
if (s) goto err;
|
|
|
|
|
2018-12-25 19:37:23 +01:00
|
|
|
// Check that next token is a correct stmt delimiter -
|
|
|
|
// disallows "print 1 print 2" and such.
|
2018-12-25 22:32:41 +01:00
|
|
|
if (G.prs.lex != BC_LEX_SCOLON
|
|
|
|
&& G.prs.lex != XC_LEX_NLINE
|
|
|
|
&& G.prs.lex != XC_LEX_EOF
|
2018-12-25 19:37:23 +01:00
|
|
|
) {
|
2019-01-01 21:50:14 +01:00
|
|
|
bc_error_at("bad statement terminator");
|
2018-12-25 19:37:23 +01:00
|
|
|
goto err;
|
2018-12-25 18:37:52 +01:00
|
|
|
}
|
2018-12-25 19:37:23 +01:00
|
|
|
// The above logic is fragile. Check these examples:
|
2018-12-27 18:23:58 +01:00
|
|
|
// - interactive read() still works
|
2018-12-25 18:37:52 +01:00
|
|
|
#endif
|
2018-12-22 06:00:25 +01:00
|
|
|
} else {
|
2018-12-25 18:37:52 +01:00
|
|
|
#if ENABLE_DC
|
2018-12-25 20:40:55 +01:00
|
|
|
s = zdc_parse_expr();
|
2018-12-25 18:37:52 +01:00
|
|
|
#endif
|
2018-12-22 06:00:25 +01:00
|
|
|
}
|
|
|
|
if (s || G_interrupt) {
|
2018-12-23 00:13:15 +01:00
|
|
|
err:
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_reset(); // includes xc_program_reset()
|
2018-12-22 06:00:25 +01:00
|
|
|
RETURN_STATUS(BC_STATUS_FAILURE);
|
|
|
|
}
|
2018-12-25 19:37:23 +01:00
|
|
|
|
2018-12-22 02:23:08 +01:00
|
|
|
dbg_lex("%s:%d executing...", __func__, __LINE__);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_program_exec();
|
2018-12-16 16:03:03 +01:00
|
|
|
if (s) {
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_reset();
|
2018-12-16 16:03:03 +01:00
|
|
|
break;
|
|
|
|
}
|
2018-12-22 19:23:46 +01:00
|
|
|
|
|
|
|
ip = (void*)G.prog.exestack.v;
|
|
|
|
#if SANITY_CHECKS
|
|
|
|
if (G.prog.exestack.len != 1) // should have only main's IP
|
libbb: reduce the overhead of single parameter bb_error_msg() calls
Back in 2007, commit 0c97c9d43707 ("'simple' error message functions by
Loic Grenie") introduced bb_simple_perror_msg() to allow for a lower
overhead call to bb_perror_msg() when only a string was being printed
with no parameters. This saves space for some CPU architectures because
it avoids the overhead of a call to a variadic function. However there
has never been a simple version of bb_error_msg(), and since 2007 many
new calls to bb_perror_msg() have been added that only take a single
parameter and so could have been using bb_simple_perror_message().
This changeset introduces 'simple' versions of bb_info_msg(),
bb_error_msg(), bb_error_msg_and_die(), bb_herror_msg() and
bb_herror_msg_and_die(), and replaces all calls that only take a
single parameter, or use something like ("%s", arg), with calls to the
corresponding 'simple' version.
Since it is likely that single parameter calls to the variadic functions
may be accidentally reintroduced in the future a new debugging config
option WARN_SIMPLE_MSG has been introduced. This uses some macro magic
which will cause any such calls to generate a warning, but this is
turned off by default to avoid use of the unpleasant macros in normal
circumstances.
This is a large changeset due to the number of calls that have been
replaced. The only files that contain changes other than simple
substitution of function calls are libbb.h, libbb/herror_msg.c,
libbb/verror_msg.c and libbb/xfuncs_printf.c. In miscutils/devfsd.c,
networking/udhcp/common.h and util-linux/mdev.c additonal macros have
been added for logging so that single parameter and multiple parameter
logging variants exist.
The amount of space saved varies considerably by architecture, and was
found to be as follows (for 'defconfig' using GCC 7.4):
Arm: -92 bytes
MIPS: -52 bytes
PPC: -1836 bytes
x86_64: -938 bytes
Note that for the MIPS architecture only an exception had to be made
disabling the 'simple' calls for 'udhcp' (in networking/udhcp/common.h)
because it made these files larger on MIPS.
Signed-off-by: James Byrne <james.byrne@origamienergy.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2019-07-02 11:35:03 +02:00
|
|
|
bb_simple_error_msg_and_die("BUG:call stack");
|
2018-12-22 19:23:46 +01:00
|
|
|
if (ip->func != BC_PROG_MAIN)
|
libbb: reduce the overhead of single parameter bb_error_msg() calls
Back in 2007, commit 0c97c9d43707 ("'simple' error message functions by
Loic Grenie") introduced bb_simple_perror_msg() to allow for a lower
overhead call to bb_perror_msg() when only a string was being printed
with no parameters. This saves space for some CPU architectures because
it avoids the overhead of a call to a variadic function. However there
has never been a simple version of bb_error_msg(), and since 2007 many
new calls to bb_perror_msg() have been added that only take a single
parameter and so could have been using bb_simple_perror_message().
This changeset introduces 'simple' versions of bb_info_msg(),
bb_error_msg(), bb_error_msg_and_die(), bb_herror_msg() and
bb_herror_msg_and_die(), and replaces all calls that only take a
single parameter, or use something like ("%s", arg), with calls to the
corresponding 'simple' version.
Since it is likely that single parameter calls to the variadic functions
may be accidentally reintroduced in the future a new debugging config
option WARN_SIMPLE_MSG has been introduced. This uses some macro magic
which will cause any such calls to generate a warning, but this is
turned off by default to avoid use of the unpleasant macros in normal
circumstances.
This is a large changeset due to the number of calls that have been
replaced. The only files that contain changes other than simple
substitution of function calls are libbb.h, libbb/herror_msg.c,
libbb/verror_msg.c and libbb/xfuncs_printf.c. In miscutils/devfsd.c,
networking/udhcp/common.h and util-linux/mdev.c additonal macros have
been added for logging so that single parameter and multiple parameter
logging variants exist.
The amount of space saved varies considerably by architecture, and was
found to be as follows (for 'defconfig' using GCC 7.4):
Arm: -92 bytes
MIPS: -52 bytes
PPC: -1836 bytes
x86_64: -938 bytes
Note that for the MIPS architecture only an exception had to be made
disabling the 'simple' calls for 'udhcp' (in networking/udhcp/common.h)
because it made these files larger on MIPS.
Signed-off-by: James Byrne <james.byrne@origamienergy.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2019-07-02 11:35:03 +02:00
|
|
|
bb_simple_error_msg_and_die("BUG:not MAIN");
|
2018-12-22 19:23:46 +01:00
|
|
|
#endif
|
2018-12-27 18:23:58 +01:00
|
|
|
f = xc_program_func_BC_PROG_MAIN();
|
2018-12-21 16:22:26 +01:00
|
|
|
// bc discards strings, constants and code after each
|
|
|
|
// top-level statement in the "main program".
|
|
|
|
// This prevents "yes 1 | bc" from growing its memory
|
|
|
|
// without bound. This can be done because data stack
|
|
|
|
// is empty and thus can't hold any references to
|
|
|
|
// strings or constants, there is no generated code
|
|
|
|
// which can hold references (after we discard one
|
|
|
|
// we just executed). Code of functions can have references,
|
|
|
|
// but bc stores function strings/constants in per-function
|
|
|
|
// storage.
|
|
|
|
if (IS_BC) {
|
2018-12-21 20:29:34 +01:00
|
|
|
#if SANITY_CHECKS
|
2018-12-22 14:18:47 +01:00
|
|
|
if (G.prog.results.len != 0) // should be empty
|
libbb: reduce the overhead of single parameter bb_error_msg() calls
Back in 2007, commit 0c97c9d43707 ("'simple' error message functions by
Loic Grenie") introduced bb_simple_perror_msg() to allow for a lower
overhead call to bb_perror_msg() when only a string was being printed
with no parameters. This saves space for some CPU architectures because
it avoids the overhead of a call to a variadic function. However there
has never been a simple version of bb_error_msg(), and since 2007 many
new calls to bb_perror_msg() have been added that only take a single
parameter and so could have been using bb_simple_perror_message().
This changeset introduces 'simple' versions of bb_info_msg(),
bb_error_msg(), bb_error_msg_and_die(), bb_herror_msg() and
bb_herror_msg_and_die(), and replaces all calls that only take a
single parameter, or use something like ("%s", arg), with calls to the
corresponding 'simple' version.
Since it is likely that single parameter calls to the variadic functions
may be accidentally reintroduced in the future a new debugging config
option WARN_SIMPLE_MSG has been introduced. This uses some macro magic
which will cause any such calls to generate a warning, but this is
turned off by default to avoid use of the unpleasant macros in normal
circumstances.
This is a large changeset due to the number of calls that have been
replaced. The only files that contain changes other than simple
substitution of function calls are libbb.h, libbb/herror_msg.c,
libbb/verror_msg.c and libbb/xfuncs_printf.c. In miscutils/devfsd.c,
networking/udhcp/common.h and util-linux/mdev.c additonal macros have
been added for logging so that single parameter and multiple parameter
logging variants exist.
The amount of space saved varies considerably by architecture, and was
found to be as follows (for 'defconfig' using GCC 7.4):
Arm: -92 bytes
MIPS: -52 bytes
PPC: -1836 bytes
x86_64: -938 bytes
Note that for the MIPS architecture only an exception had to be made
disabling the 'simple' calls for 'udhcp' (in networking/udhcp/common.h)
because it made these files larger on MIPS.
Signed-off-by: James Byrne <james.byrne@origamienergy.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2019-07-02 11:35:03 +02:00
|
|
|
bb_simple_error_msg_and_die("BUG:data stack");
|
2018-12-21 20:29:34 +01:00
|
|
|
#endif
|
2018-12-21 16:22:26 +01:00
|
|
|
IF_BC(bc_vec_pop_all(&f->strs);)
|
|
|
|
IF_BC(bc_vec_pop_all(&f->consts);)
|
2019-01-08 18:08:48 +01:00
|
|
|
// We are at SCOLON/NLINE, skip it:
|
|
|
|
s = zxc_lex_next();
|
|
|
|
if (s) goto err;
|
2018-12-22 18:04:08 +01:00
|
|
|
} else {
|
2018-12-22 19:23:46 +01:00
|
|
|
if (G.prog.results.len == 0
|
|
|
|
&& G.prog.vars.len == 0
|
|
|
|
) {
|
|
|
|
// If stack is empty and no registers exist (TODO: or they are all empty),
|
|
|
|
// we can get rid of accumulated strings and constants.
|
|
|
|
// In this example dc process should not grow
|
|
|
|
// its memory consumption with time:
|
|
|
|
// yes 1pc | dc
|
|
|
|
IF_DC(bc_vec_pop_all(&G.prog.strs);)
|
|
|
|
IF_DC(bc_vec_pop_all(&G.prog.consts);)
|
|
|
|
}
|
|
|
|
// The code is discarded always (below), thus this example
|
|
|
|
// should also not grow its memory consumption with time,
|
|
|
|
// even though its data stack is not empty:
|
|
|
|
// { echo 1; yes dk; } | dc
|
2018-12-21 16:22:26 +01:00
|
|
|
}
|
2018-12-22 19:23:46 +01:00
|
|
|
// We drop generated and executed code for both bc and dc:
|
|
|
|
bc_vec_pop_all(&f->code);
|
|
|
|
ip->inst_idx = 0;
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2019-01-04 05:41:47 +01:00
|
|
|
|
2018-12-14 17:51:17 +01:00
|
|
|
dbg_lex_done("%s:%d done", __func__, __LINE__);
|
2018-12-12 21:39:10 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_vm_process(...) (zxc_vm_process(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_vm_execute_FILE(FILE *fp, const char *filename)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-13 19:58:58 +01:00
|
|
|
// So far bc/dc have no way to include a file from another file,
|
2018-12-25 22:32:41 +01:00
|
|
|
// therefore we know G.prs.lex_filename == NULL on entry
|
2018-12-13 19:58:58 +01:00
|
|
|
//const char *sv_file;
|
2018-12-05 16:39:22 +01:00
|
|
|
BcStatus s;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-25 22:32:41 +01:00
|
|
|
G.prs.lex_filename = filename;
|
|
|
|
G.prs.lex_input_fp = fp;
|
2018-12-26 18:32:43 +01:00
|
|
|
G.err_line = G.prs.lex_line = 1;
|
2019-01-01 21:50:14 +01:00
|
|
|
dbg_lex("p->lex_line reset to 1");
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-17 09:51:43 +01:00
|
|
|
do {
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_vm_process("");
|
2018-12-17 09:51:43 +01:00
|
|
|
// We do not stop looping on errors here if reading stdin.
|
|
|
|
// Example: start interactive bc and enter "return".
|
|
|
|
// It should say "'return' not in a function"
|
|
|
|
// but should not exit.
|
2018-12-25 22:32:41 +01:00
|
|
|
} while (G.prs.lex_input_fp == stdin);
|
|
|
|
G.prs.lex_filename = NULL;
|
2018-12-12 21:39:10 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_vm_execute_FILE(...) (zxc_vm_execute_FILE(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_vm_file(const char *file)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-02 20:16:52 +01:00
|
|
|
BcStatus s;
|
2018-12-17 09:51:43 +01:00
|
|
|
FILE *fp;
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-17 09:51:43 +01:00
|
|
|
fp = xfopen_for_read(file);
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_vm_execute_FILE(fp, file);
|
2018-12-17 09:51:43 +01:00
|
|
|
fclose(fp);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-12 22:48:19 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_vm_file(...) (zxc_vm_file(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-05 16:21:43 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-14 00:12:13 +01:00
|
|
|
static void bc_vm_info(void)
|
|
|
|
{
|
|
|
|
printf("%s "BB_VER"\n"
|
2018-12-18 10:03:14 -07:00
|
|
|
"Adapted from https://github.com/gavinhoward/bc\n"
|
|
|
|
"Original code (c) 2018 Gavin D. Howard and contributors\n"
|
2018-12-14 00:12:13 +01:00
|
|
|
, applet_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_args(char **argv)
|
|
|
|
{
|
|
|
|
unsigned opts;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
GETOPT_RESET();
|
|
|
|
#if ENABLE_FEATURE_BC_LONG_OPTIONS
|
|
|
|
opts = option_mask32 |= getopt32long(argv, "wvsqli",
|
|
|
|
"warn\0" No_argument "w"
|
|
|
|
"version\0" No_argument "v"
|
|
|
|
"standard\0" No_argument "s"
|
|
|
|
"quiet\0" No_argument "q"
|
|
|
|
"mathlib\0" No_argument "l"
|
|
|
|
"interactive\0" No_argument "i"
|
|
|
|
);
|
|
|
|
#else
|
|
|
|
opts = option_mask32 |= getopt32(argv, "wvsqli");
|
|
|
|
#endif
|
|
|
|
if (getenv("POSIXLY_CORRECT"))
|
|
|
|
option_mask32 |= BC_FLAG_S;
|
|
|
|
|
|
|
|
if (opts & BC_FLAG_V) {
|
|
|
|
bc_vm_info();
|
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = optind; argv[i]; ++i)
|
|
|
|
bc_vec_push(&G.files, argv + i);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bc_vm_envArgs(void)
|
|
|
|
{
|
|
|
|
BcVec v;
|
|
|
|
char *buf;
|
|
|
|
char *env_args = getenv("BC_ENV_ARGS");
|
|
|
|
|
|
|
|
if (!env_args) return;
|
|
|
|
|
|
|
|
G.env_args = xstrdup(env_args);
|
|
|
|
buf = G.env_args;
|
|
|
|
|
|
|
|
bc_vec_init(&v, sizeof(char *), NULL);
|
|
|
|
|
|
|
|
while (*(buf = skip_whitespace(buf)) != '\0') {
|
|
|
|
bc_vec_push(&v, &buf);
|
|
|
|
buf = skip_non_whitespace(buf);
|
|
|
|
if (!*buf)
|
|
|
|
break;
|
|
|
|
*buf++ = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
// NULL terminate, and pass argv[] so that first arg is argv[1]
|
|
|
|
if (sizeof(int) == sizeof(char*)) {
|
|
|
|
bc_vec_push(&v, &const_int_0);
|
|
|
|
} else {
|
|
|
|
static char *const nullptr = NULL;
|
|
|
|
bc_vec_push(&v, &nullptr);
|
|
|
|
}
|
|
|
|
bc_args(((char **)v.v) - 1);
|
|
|
|
|
|
|
|
bc_vec_free(&v);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char bc_lib[] ALIGN1 = {
|
2018-12-05 16:21:43 +01:00
|
|
|
"scale=20"
|
|
|
|
"\n" "define e(x){"
|
|
|
|
"\n" "auto b,s,n,r,d,i,p,f,v"
|
2018-12-14 09:53:50 +01:00
|
|
|
////////////////"if(x<0)return(1/e(-x))" // and drop 'n' and x<0 logic below
|
|
|
|
//^^^^^^^^^^^^^^^^ this would work, and is even more precise than GNU bc:
|
|
|
|
//e(-.998896): GNU:.36828580434569428695
|
|
|
|
// above code:.36828580434569428696
|
|
|
|
// actual value:.3682858043456942869594...
|
|
|
|
// but for now let's be "GNU compatible"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "b=ibase"
|
|
|
|
"\n" "ibase=A"
|
|
|
|
"\n" "if(x<0){"
|
|
|
|
"\n" "n=1"
|
|
|
|
"\n" "x=-x"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "s=scale"
|
2018-12-16 21:46:11 +01:00
|
|
|
"\n" "r=6+s+.44*x"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "scale=scale(x)+1"
|
|
|
|
"\n" "while(x>1){"
|
|
|
|
"\n" "d+=1"
|
|
|
|
"\n" "x/=2"
|
|
|
|
"\n" "scale+=1"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "scale=r"
|
|
|
|
"\n" "r=x+1"
|
|
|
|
"\n" "p=x"
|
|
|
|
"\n" "f=v=1"
|
2018-12-14 01:01:01 +01:00
|
|
|
"\n" "for(i=2;v;++i){"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "p*=x"
|
|
|
|
"\n" "f*=i"
|
|
|
|
"\n" "v=p/f"
|
|
|
|
"\n" "r+=v"
|
|
|
|
"\n" "}"
|
2018-12-14 01:01:01 +01:00
|
|
|
"\n" "while(d--)r*=r"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "scale=s"
|
|
|
|
"\n" "ibase=b"
|
2018-12-14 01:01:01 +01:00
|
|
|
"\n" "if(n)return(1/r)"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "return(r/1)"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "define l(x){"
|
|
|
|
"\n" "auto b,s,r,p,a,q,i,v"
|
|
|
|
"\n" "b=ibase"
|
|
|
|
"\n" "ibase=A"
|
|
|
|
"\n" "if(x<=0){"
|
|
|
|
"\n" "r=(1-10^scale)/1"
|
|
|
|
"\n" "ibase=b"
|
|
|
|
"\n" "return(r)"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "s=scale"
|
|
|
|
"\n" "scale+=6"
|
|
|
|
"\n" "p=2"
|
|
|
|
"\n" "while(x>=2){"
|
|
|
|
"\n" "p*=2"
|
|
|
|
"\n" "x=sqrt(x)"
|
|
|
|
"\n" "}"
|
2018-12-16 21:46:11 +01:00
|
|
|
"\n" "while(x<=.5){"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "p*=2"
|
|
|
|
"\n" "x=sqrt(x)"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "r=a=(x-1)/(x+1)"
|
|
|
|
"\n" "q=a*a"
|
2018-12-14 01:01:01 +01:00
|
|
|
"\n" "v=1"
|
|
|
|
"\n" "for(i=3;v;i+=2){"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "a*=q"
|
|
|
|
"\n" "v=a/i"
|
|
|
|
"\n" "r+=v"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "r*=p"
|
|
|
|
"\n" "scale=s"
|
|
|
|
"\n" "ibase=b"
|
|
|
|
"\n" "return(r/1)"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "define s(x){"
|
2018-12-14 11:27:09 +01:00
|
|
|
"\n" "auto b,s,r,a,q,i"
|
|
|
|
"\n" "if(x<0)return(-s(-x))"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "b=ibase"
|
|
|
|
"\n" "ibase=A"
|
|
|
|
"\n" "s=scale"
|
|
|
|
"\n" "scale=1.1*s+2"
|
|
|
|
"\n" "a=a(1)"
|
|
|
|
"\n" "scale=0"
|
|
|
|
"\n" "q=(x/a+2)/4"
|
2018-12-14 09:53:50 +01:00
|
|
|
"\n" "x-=4*q*a"
|
2018-12-14 01:01:01 +01:00
|
|
|
"\n" "if(q%2)x=-x"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "scale=s+2"
|
|
|
|
"\n" "r=a=x"
|
|
|
|
"\n" "q=-x*x"
|
2018-12-14 01:01:01 +01:00
|
|
|
"\n" "for(i=3;a;i+=2){"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "a*=q/(i*(i-1))"
|
|
|
|
"\n" "r+=a"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "scale=s"
|
|
|
|
"\n" "ibase=b"
|
|
|
|
"\n" "return(r/1)"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "define c(x){"
|
|
|
|
"\n" "auto b,s"
|
|
|
|
"\n" "b=ibase"
|
|
|
|
"\n" "ibase=A"
|
|
|
|
"\n" "s=scale"
|
|
|
|
"\n" "scale*=1.2"
|
|
|
|
"\n" "x=s(2*a(1)+x)"
|
|
|
|
"\n" "scale=s"
|
|
|
|
"\n" "ibase=b"
|
|
|
|
"\n" "return(x/1)"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "define a(x){"
|
|
|
|
"\n" "auto b,s,r,n,a,m,t,f,i,u"
|
|
|
|
"\n" "b=ibase"
|
|
|
|
"\n" "ibase=A"
|
|
|
|
"\n" "n=1"
|
|
|
|
"\n" "if(x<0){"
|
|
|
|
"\n" "n=-1"
|
|
|
|
"\n" "x=-x"
|
|
|
|
"\n" "}"
|
2018-12-14 01:01:01 +01:00
|
|
|
"\n" "if(scale<65){"
|
|
|
|
"\n" "if(x==1)return(.7853981633974483096156608458198757210492923498437764552437361480/n)"
|
|
|
|
"\n" "if(x==.2)return(.1973955598498807583700497651947902934475851037878521015176889402/n)"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "}"
|
|
|
|
"\n" "s=scale"
|
|
|
|
"\n" "if(x>.2){"
|
|
|
|
"\n" "scale+=5"
|
|
|
|
"\n" "a=a(.2)"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "scale=s+3"
|
|
|
|
"\n" "while(x>.2){"
|
|
|
|
"\n" "m+=1"
|
|
|
|
"\n" "x=(x-.2)/(1+.2*x)"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "r=u=x"
|
|
|
|
"\n" "f=-x*x"
|
|
|
|
"\n" "t=1"
|
2018-12-14 01:01:01 +01:00
|
|
|
"\n" "for(i=3;t;i+=2){"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "u*=f"
|
|
|
|
"\n" "t=u/i"
|
|
|
|
"\n" "r+=t"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "scale=s"
|
|
|
|
"\n" "ibase=b"
|
|
|
|
"\n" "return((m*a+r)/n)"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "define j(n,x){"
|
|
|
|
"\n" "auto b,s,o,a,i,v,f"
|
|
|
|
"\n" "b=ibase"
|
|
|
|
"\n" "ibase=A"
|
|
|
|
"\n" "s=scale"
|
|
|
|
"\n" "scale=0"
|
|
|
|
"\n" "n/=1"
|
|
|
|
"\n" "if(n<0){"
|
|
|
|
"\n" "n=-n"
|
2018-12-14 09:53:50 +01:00
|
|
|
"\n" "o=n%2"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "}"
|
|
|
|
"\n" "a=1"
|
|
|
|
"\n" "for(i=2;i<=n;++i)a*=i"
|
|
|
|
"\n" "scale=1.5*s"
|
|
|
|
"\n" "a=(x^n)/2^n/a"
|
|
|
|
"\n" "r=v=1"
|
|
|
|
"\n" "f=-x*x/4"
|
2018-12-14 10:10:37 +01:00
|
|
|
"\n" "scale+=length(a)-scale(a)"
|
2018-12-14 01:01:01 +01:00
|
|
|
"\n" "for(i=1;v;++i){"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "v=v*f/i/(n+i)"
|
|
|
|
"\n" "r+=v"
|
|
|
|
"\n" "}"
|
|
|
|
"\n" "scale=s"
|
|
|
|
"\n" "ibase=b"
|
2018-12-14 01:01:01 +01:00
|
|
|
"\n" "if(o)a=-a"
|
2018-12-05 16:21:43 +01:00
|
|
|
"\n" "return(a*r/1)"
|
|
|
|
"\n" "}"
|
|
|
|
};
|
|
|
|
#endif // ENABLE_BC
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BC_STATUS zxc_vm_exec(void)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-19 18:09:31 +01:00
|
|
|
char **fname;
|
2018-12-06 18:41:59 +01:00
|
|
|
BcStatus s;
|
2018-11-03 11:00:21 -06:00
|
|
|
size_t i;
|
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-04 20:58:40 +01:00
|
|
|
if (option_mask32 & BC_FLAG_L) {
|
2018-12-05 16:21:43 +01:00
|
|
|
// We know that internal library is not buggy,
|
|
|
|
// thus error checking is normally disabled.
|
|
|
|
# define DEBUG_LIB 0
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_vm_process(bc_lib);
|
2018-12-12 22:48:19 +01:00
|
|
|
if (DEBUG_LIB && s) RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-06 18:41:59 +01:00
|
|
|
s = BC_STATUS_SUCCESS;
|
2018-12-19 18:09:31 +01:00
|
|
|
fname = (void*)G.files.v;
|
|
|
|
for (i = 0; i < G.files.len; i++) {
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_vm_file(*fname++);
|
2018-12-19 18:09:31 +01:00
|
|
|
if (ENABLE_FEATURE_CLEAN_UP && !G_ttyin && s) {
|
|
|
|
// Debug config, non-interactive mode:
|
|
|
|
// return all the way back to main.
|
|
|
|
// Non-debug builds do not come here
|
|
|
|
// in non-interactive mode, they exit.
|
|
|
|
RETURN_STATUS(s);
|
|
|
|
}
|
2018-12-04 20:51:40 +01:00
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-10 20:56:08 +01:00
|
|
|
if (IS_BC || (option_mask32 & BC_FLAG_I))
|
2018-12-27 18:23:58 +01:00
|
|
|
s = zxc_vm_execute_FILE(stdin, /*filename:*/ NULL);
|
2018-12-06 18:41:59 +01:00
|
|
|
|
2018-12-12 22:48:19 +01:00
|
|
|
RETURN_STATUS(s);
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-27 18:23:58 +01:00
|
|
|
#define zxc_vm_exec(...) (zxc_vm_exec(__VA_ARGS__) COMMA_SUCCESS)
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 17:18:52 +01:00
|
|
|
#if ENABLE_FEATURE_CLEAN_UP
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_program_free(void)
|
2018-12-02 17:18:52 +01:00
|
|
|
{
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_free(&G.prog.fns);
|
2018-12-20 20:34:09 +01:00
|
|
|
IF_BC(bc_vec_free(&G.prog.fn_map);)
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_free(&G.prog.vars);
|
|
|
|
bc_vec_free(&G.prog.var_map);
|
|
|
|
bc_vec_free(&G.prog.arrs);
|
|
|
|
bc_vec_free(&G.prog.arr_map);
|
2018-12-21 16:22:26 +01:00
|
|
|
IF_DC(bc_vec_free(&G.prog.strs);)
|
|
|
|
IF_DC(bc_vec_free(&G.prog.consts);)
|
2018-12-02 18:26:38 +01:00
|
|
|
bc_vec_free(&G.prog.results);
|
2018-12-19 12:35:27 +01:00
|
|
|
bc_vec_free(&G.prog.exestack);
|
2018-12-20 20:34:09 +01:00
|
|
|
IF_BC(bc_num_free(&G.prog.last);)
|
2018-12-27 18:08:30 +01:00
|
|
|
//IF_BC(bc_num_free(&G.prog.zero);)
|
2018-12-20 16:24:18 +01:00
|
|
|
IF_BC(bc_num_free(&G.prog.one);)
|
2018-12-17 09:51:43 +01:00
|
|
|
bc_vec_free(&G.input_buffer);
|
2018-12-02 17:18:52 +01:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static void xc_program_init(void)
|
2018-12-02 17:18:52 +01:00
|
|
|
{
|
|
|
|
BcInstPtr ip;
|
|
|
|
|
2018-12-14 16:30:56 +01:00
|
|
|
// memset(&G.prog, 0, sizeof(G.prog)); - already is
|
2018-12-02 17:18:52 +01:00
|
|
|
memset(&ip, 0, sizeof(BcInstPtr));
|
|
|
|
|
2018-12-14 16:30:56 +01:00
|
|
|
// G.prog.nchars = G.prog.scale = 0; - already is
|
2018-12-02 17:18:52 +01:00
|
|
|
G.prog.ib_t = 10;
|
|
|
|
G.prog.ob_t = 10;
|
|
|
|
|
2018-12-20 16:24:18 +01:00
|
|
|
IF_BC(bc_num_init_DEF_SIZE(&G.prog.last);)
|
|
|
|
//IF_BC(bc_num_zero(&G.prog.last);) - already is
|
2018-12-02 17:18:52 +01:00
|
|
|
|
2018-12-27 18:08:30 +01:00
|
|
|
//bc_num_init_DEF_SIZE(&G.prog.zero); - not needed
|
2018-12-09 12:04:44 +01:00
|
|
|
//bc_num_zero(&G.prog.zero); - already is
|
2018-12-02 17:18:52 +01:00
|
|
|
|
2018-12-20 16:24:18 +01:00
|
|
|
IF_BC(bc_num_init_DEF_SIZE(&G.prog.one);)
|
|
|
|
IF_BC(bc_num_one(&G.prog.one);)
|
2018-12-02 17:18:52 +01:00
|
|
|
|
|
|
|
bc_vec_init(&G.prog.fns, sizeof(BcFunc), bc_func_free);
|
2018-12-20 20:34:09 +01:00
|
|
|
IF_BC(bc_vec_init(&G.prog.fn_map, sizeof(BcId), bc_id_free);)
|
2018-12-02 17:18:52 +01:00
|
|
|
|
2018-12-20 20:34:09 +01:00
|
|
|
if (IS_BC) {
|
2018-12-22 01:34:10 +01:00
|
|
|
// Names are chosen simply to be distinct and never match
|
2018-12-21 00:10:26 +01:00
|
|
|
// a valid function name (and be short)
|
|
|
|
IF_BC(bc_program_addFunc(xstrdup(""))); // func #0: main
|
2018-12-22 01:34:10 +01:00
|
|
|
IF_BC(bc_program_addFunc(xstrdup("1"))); // func #1: for read()
|
2018-12-20 20:34:09 +01:00
|
|
|
} else {
|
2018-12-22 01:34:10 +01:00
|
|
|
// in dc, functions have no names
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_add_fn();
|
|
|
|
xc_program_add_fn();
|
2018-12-20 20:34:09 +01:00
|
|
|
}
|
2018-12-02 17:18:52 +01:00
|
|
|
|
|
|
|
bc_vec_init(&G.prog.vars, sizeof(BcVec), bc_vec_free);
|
2018-12-04 21:54:33 +01:00
|
|
|
bc_vec_init(&G.prog.var_map, sizeof(BcId), bc_id_free);
|
2018-12-02 17:18:52 +01:00
|
|
|
|
|
|
|
bc_vec_init(&G.prog.arrs, sizeof(BcVec), bc_vec_free);
|
2018-12-04 21:54:33 +01:00
|
|
|
bc_vec_init(&G.prog.arr_map, sizeof(BcId), bc_id_free);
|
2018-12-02 17:18:52 +01:00
|
|
|
|
2018-12-21 16:22:26 +01:00
|
|
|
IF_DC(bc_vec_init(&G.prog.strs, sizeof(char *), bc_string_free);)
|
|
|
|
IF_DC(bc_vec_init(&G.prog.consts, sizeof(char *), bc_string_free);)
|
2018-12-02 17:18:52 +01:00
|
|
|
bc_vec_init(&G.prog.results, sizeof(BcResult), bc_result_free);
|
2018-12-19 12:35:27 +01:00
|
|
|
bc_vec_init(&G.prog.exestack, sizeof(BcInstPtr), NULL);
|
|
|
|
bc_vec_push(&G.prog.exestack, &ip);
|
2018-12-16 16:03:03 +01:00
|
|
|
|
2018-12-17 09:51:43 +01:00
|
|
|
bc_char_vec_init(&G.input_buffer);
|
2018-12-02 17:18:52 +01:00
|
|
|
}
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2020-12-21 21:36:58 +01:00
|
|
|
static unsigned xc_vm_envLen(const char *var)
|
|
|
|
{
|
|
|
|
char *lenv;
|
|
|
|
unsigned len;
|
|
|
|
|
|
|
|
lenv = getenv(var);
|
|
|
|
len = BC_NUM_PRINT_WIDTH;
|
2020-12-29 18:50:56 +01:00
|
|
|
if (lenv) {
|
|
|
|
len = bb_strtou(lenv, NULL, 10);
|
|
|
|
if (len == 0 || len > INT_MAX)
|
|
|
|
len = INT_MAX;
|
|
|
|
if (errno)
|
|
|
|
len = BC_NUM_PRINT_WIDTH;
|
|
|
|
}
|
2020-12-21 21:36:58 +01:00
|
|
|
|
2020-12-29 18:50:56 +01:00
|
|
|
// dc (GNU bc 1.07.1) 1.4.1 seems to use width
|
|
|
|
// 1 char wider than bc from the same package.
|
|
|
|
// Both default width, and xC_LINE_LENGTH=N are wider:
|
|
|
|
// "DC_LINE_LENGTH=5 dc -e'123456 p'" prints:
|
|
|
|
// |1234\ |
|
|
|
|
// |56 |
|
|
|
|
// "echo '123456' | BC_LINE_LENGTH=5 bc" prints:
|
|
|
|
// |123\ |
|
|
|
|
// |456 |
|
|
|
|
// Do the same, but it might be a bug in GNU package
|
|
|
|
if (IS_BC)
|
|
|
|
len--;
|
|
|
|
|
|
|
|
if (len < 2)
|
|
|
|
len = IS_BC ? BC_NUM_PRINT_WIDTH - 1 : BC_NUM_PRINT_WIDTH;
|
2020-12-21 21:36:58 +01:00
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static int xc_vm_init(const char *env_len)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-30 15:56:36 +01:00
|
|
|
G.prog.len = xc_vm_envLen(env_len);
|
2018-12-02 15:48:37 +01:00
|
|
|
bc_vec_init(&G.files, sizeof(char *), NULL);
|
2018-12-30 15:56:36 +01:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_program_init();
|
2018-12-13 22:49:59 +01:00
|
|
|
IF_BC(if (IS_BC) bc_vm_envArgs();)
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_parse_create(BC_PROG_MAIN);
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-13 16:35:52 +01:00
|
|
|
//TODO: in GNU bc, the check is (isatty(0) && isatty(1)),
|
|
|
|
//-i option unconditionally enables this regardless of isatty():
|
2018-12-06 09:20:32 +01:00
|
|
|
if (isatty(0)) {
|
2018-12-27 22:52:13 +01:00
|
|
|
#if ENABLE_FEATURE_BC_INTERACTIVE
|
2018-12-06 09:20:32 +01:00
|
|
|
G_ttyin = 1;
|
2018-12-04 21:21:32 +01:00
|
|
|
// With SA_RESTART, most system calls will restart
|
|
|
|
// (IOW: they won't fail with EINTR).
|
|
|
|
// In particular, this means ^C won't cause
|
|
|
|
// stdout to get into "error state" if SIGINT hits
|
|
|
|
// within write() syscall.
|
2018-12-13 16:35:52 +01:00
|
|
|
//
|
|
|
|
// The downside is that ^C while tty input is taken
|
2018-12-04 21:21:32 +01:00
|
|
|
// will only be handled after [Enter] since read()
|
|
|
|
// from stdin is not interrupted by ^C either,
|
|
|
|
// it restarts, thus fgetc() does not return on ^C.
|
2018-12-13 16:35:52 +01:00
|
|
|
// (This problem manifests only if line editing is disabled)
|
2018-12-04 21:21:32 +01:00
|
|
|
signal_SA_RESTART_empty_mask(SIGINT, record_signo);
|
|
|
|
|
|
|
|
// Without SA_RESTART, this exhibits a bug:
|
|
|
|
// "while (1) print 1" and try ^C-ing it.
|
|
|
|
// Intermittently, instead of returning to input line,
|
|
|
|
// you'll get "output error: Interrupted system call"
|
|
|
|
// and exit.
|
|
|
|
//signal_no_SA_RESTART_empty_mask(SIGINT, record_signo);
|
2018-12-04 19:11:02 +01:00
|
|
|
#endif
|
2018-12-06 18:41:59 +01:00
|
|
|
return 1; // "tty"
|
2018-12-04 19:11:02 +01:00
|
|
|
}
|
2018-12-06 18:41:59 +01:00
|
|
|
return 0; // "not a tty"
|
|
|
|
}
|
2018-12-06 00:46:09 +01:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
static BcStatus xc_vm_run(void)
|
2018-12-06 18:41:59 +01:00
|
|
|
{
|
2018-12-27 18:23:58 +01:00
|
|
|
BcStatus st = zxc_vm_exec();
|
2018-12-02 17:18:52 +01:00
|
|
|
#if ENABLE_FEATURE_CLEAN_UP
|
2018-12-06 23:06:57 +01:00
|
|
|
if (G_exiting) // it was actually "halt" or "quit"
|
|
|
|
st = EXIT_SUCCESS;
|
2018-12-30 15:56:36 +01:00
|
|
|
|
|
|
|
bc_vec_free(&G.files);
|
|
|
|
xc_program_free();
|
|
|
|
xc_parse_free();
|
|
|
|
free(G.env_args);
|
2018-12-06 10:29:12 +01:00
|
|
|
# if ENABLE_FEATURE_EDITING
|
|
|
|
free_line_input_t(G.line_input_state);
|
|
|
|
# endif
|
2018-12-06 00:29:22 +01:00
|
|
|
FREE_G();
|
2018-12-02 17:18:52 +01:00
|
|
|
#endif
|
2018-12-16 16:03:03 +01:00
|
|
|
dbg_exec("exiting with exitcode %d", st);
|
2018-11-03 11:00:21 -06:00
|
|
|
return st;
|
|
|
|
}
|
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_BC
|
2018-12-02 14:35:32 +01:00
|
|
|
int bc_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
|
2018-12-06 00:46:09 +01:00
|
|
|
int bc_main(int argc UNUSED_PARAM, char **argv)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-06 18:41:59 +01:00
|
|
|
int is_tty;
|
|
|
|
|
2018-12-02 15:48:37 +01:00
|
|
|
INIT_G();
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
is_tty = xc_vm_init("BC_LINE_LENGTH");
|
2018-12-06 18:41:59 +01:00
|
|
|
|
|
|
|
bc_args(argv);
|
|
|
|
|
|
|
|
if (is_tty && !(option_mask32 & BC_FLAG_Q))
|
|
|
|
bc_vm_info();
|
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
return xc_vm_run();
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-02 18:49:16 +01:00
|
|
|
#if ENABLE_DC
|
2018-12-02 14:35:32 +01:00
|
|
|
int dc_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
|
2018-12-06 00:46:09 +01:00
|
|
|
int dc_main(int argc UNUSED_PARAM, char **argv)
|
2018-11-03 11:00:21 -06:00
|
|
|
{
|
2018-12-06 18:41:59 +01:00
|
|
|
int noscript;
|
|
|
|
|
2018-12-02 15:48:37 +01:00
|
|
|
INIT_G();
|
2018-12-14 16:30:56 +01:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
xc_vm_init("DC_LINE_LENGTH");
|
2018-12-06 18:41:59 +01:00
|
|
|
|
|
|
|
// Run -e'SCRIPT' and -fFILE in order of appearance, then handle FILEs
|
|
|
|
noscript = BC_FLAG_I;
|
|
|
|
for (;;) {
|
|
|
|
int n = getopt(argc, argv, "e:f:x");
|
|
|
|
if (n <= 0)
|
|
|
|
break;
|
|
|
|
switch (n) {
|
|
|
|
case 'e':
|
|
|
|
noscript = 0;
|
2018-12-27 18:23:58 +01:00
|
|
|
n = zxc_vm_process(optarg);
|
2018-12-06 18:41:59 +01:00
|
|
|
if (n) return n;
|
|
|
|
break;
|
|
|
|
case 'f':
|
|
|
|
noscript = 0;
|
2018-12-27 18:23:58 +01:00
|
|
|
n = zxc_vm_file(optarg);
|
2018-12-12 21:39:10 +01:00
|
|
|
if (n) return n;
|
2018-12-06 18:41:59 +01:00
|
|
|
break;
|
|
|
|
case 'x':
|
|
|
|
option_mask32 |= DC_FLAG_X;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
bb_show_usage();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
argv += optind;
|
|
|
|
|
|
|
|
while (*argv) {
|
|
|
|
noscript = 0;
|
|
|
|
bc_vec_push(&G.files, argv++);
|
|
|
|
}
|
|
|
|
|
|
|
|
option_mask32 |= noscript; // set BC_FLAG_I if we need to interpret stdin
|
2018-11-03 11:00:21 -06:00
|
|
|
|
2018-12-27 18:23:58 +01:00
|
|
|
return xc_vm_run();
|
2018-11-03 11:00:21 -06:00
|
|
|
}
|
2018-12-02 18:49:16 +01:00
|
|
|
#endif
|
2018-12-06 11:31:14 +01:00
|
|
|
|
2018-12-27 22:52:13 +01:00
|
|
|
#endif // DC_BIG
|