1 /* 2 * ***************************************************************************** 3 * 4 * SPDX-License-Identifier: BSD-2-Clause 5 * 6 * Copyright (c) 2018-2024 Gavin D. Howard and contributors. 7 * 8 * Redistribution and use in source and binary forms, with or without 9 * modification, are permitted provided that the following conditions are met: 10 * 11 * * Redistributions of source code must retain the above copyright notice, this 12 * list of conditions and the following disclaimer. 13 * 14 * * Redistributions in binary form must reproduce the above copyright notice, 15 * this list of conditions and the following disclaimer in the documentation 16 * and/or other materials provided with the distribution. 17 * 18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 19 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 20 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 21 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE 22 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 23 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 24 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 25 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 26 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 27 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 28 * POSSIBILITY OF SUCH DAMAGE. 29 * 30 * ***************************************************************************** 31 * 32 * Definitions for bc's VM. 33 * 34 */ 35 36 #ifndef BC_VM_H 37 #define BC_VM_H 38 39 #include <assert.h> 40 #include <stddef.h> 41 #include <limits.h> 42 43 #include <signal.h> 44 45 #if BC_ENABLE_NLS 46 47 #ifdef _WIN32 48 #error NLS is not supported on Windows. 49 #endif // _WIN32 50 51 #include <nl_types.h> 52 53 #endif // BC_ENABLE_NLS 54 55 #include <version.h> 56 #include <status.h> 57 #include <num.h> 58 #include <lex.h> 59 #include <parse.h> 60 #include <program.h> 61 #include <history.h> 62 #include <bc.h> 63 64 // We don't want to include this file for the library because it's unused. 65 #if !BC_ENABLE_LIBRARY 66 #include <file.h> 67 #endif // !BC_ENABLE_LIBRARY 68 69 // This should be obvious. If neither calculator is enabled, barf. 70 #if !BC_ENABLED && !DC_ENABLED 71 #error Must define BC_ENABLED, DC_ENABLED, or both 72 #endif 73 74 // CHAR_BIT must be at least 6, for various reasons. I might want to bump this 75 // to 8 in the future. 76 #if CHAR_BIT < 6 77 #error CHAR_BIT must be at least 6. 78 #endif 79 80 // Set defaults. 81 82 #ifndef MAINEXEC 83 #define MAINEXEC bc 84 #endif // MAINEXEC 85 86 #ifndef _WIN32 87 #ifndef EXECPREFIX 88 #define EXECPREFIX 89 #endif // EXECPREFIX 90 #else // _WIN32 91 #undef EXECPREFIX 92 #endif // _WIN32 93 94 /** 95 * Generate a string from text. 96 * @parm V The text to generate a string for. 97 */ 98 #define GEN_STR(V) #V 99 100 /** 101 * Help generate a string from text. The preprocessor requires this two-step 102 * process. Trust me. 103 * @parm V The text to generate a string for. 104 */ 105 #define GEN_STR2(V) GEN_STR(V) 106 107 /// The version as a string. VERSION must be defined previously, usually by the 108 /// build system. 109 #define BC_VERSION GEN_STR2(VERSION) 110 111 /// The main executable name as a string. MAINEXEC must be defined previously, 112 /// usually by the build system. 113 #define BC_MAINEXEC GEN_STR2(MAINEXEC) 114 115 /// The build type as a string. BUILD_TYPE must be defined previously, usually 116 /// by the build system. 117 #define BC_BUILD_TYPE GEN_STR2(BUILD_TYPE) 118 119 // We only allow an empty executable prefix on Windows. 120 #ifndef _WIN32 121 #define BC_EXECPREFIX GEN_STR2(EXECPREFIX) 122 #else // _WIN32 123 #define BC_EXECPREFIX "" 124 #endif // _WIN32 125 126 #if !BC_ENABLE_LIBRARY 127 128 #if DC_ENABLED 129 130 /// The flag for the extended register option. 131 #define DC_FLAG_X (UINTMAX_C(1) << 0) 132 133 #endif // DC_ENABLED 134 135 #if BC_ENABLED 136 137 /// The flag for the POSIX warning option. 138 #define BC_FLAG_W (UINTMAX_C(1) << 1) 139 140 /// The flag for the POSIX error option. 141 #define BC_FLAG_S (UINTMAX_C(1) << 2) 142 143 /// The flag for the math library option. 144 #define BC_FLAG_L (UINTMAX_C(1) << 3) 145 146 /// The flag for the global stacks option. 147 #define BC_FLAG_G (UINTMAX_C(1) << 4) 148 149 #endif // BC_ENABLED 150 151 /// The flag for quiet, though this one is reversed; the option clears the flag. 152 #define BC_FLAG_Q (UINTMAX_C(1) << 5) 153 154 /// The flag for interactive. 155 #define BC_FLAG_I (UINTMAX_C(1) << 6) 156 157 /// The flag for prompt. This is also reversed; the option clears the flag. 158 #define BC_FLAG_P (UINTMAX_C(1) << 7) 159 160 /// The flag for read prompt. This is also reversed; the option clears the flag. 161 #define BC_FLAG_R (UINTMAX_C(1) << 8) 162 163 /// The flag for a leading zero. 164 #define BC_FLAG_Z (UINTMAX_C(1) << 9) 165 166 /// The flag for stdin being a TTY. 167 #define BC_FLAG_TTYIN (UINTMAX_C(1) << 10) 168 169 /// The flag for TTY mode. 170 #define BC_FLAG_TTY (UINTMAX_C(1) << 11) 171 172 /// The flag for reset on SIGINT. 173 #define BC_FLAG_SIGINT (UINTMAX_C(1) << 12) 174 175 /// The flag for exiting with expressions. 176 #define BC_FLAG_EXPR_EXIT (UINTMAX_C(1) << 13) 177 178 /// The flag for digit clamping. 179 #define BC_FLAG_DIGIT_CLAMP (UINTMAX_C(1) << 14) 180 181 /// A convenience macro for getting the TTYIN flag. 182 #define BC_TTYIN (vm->flags & BC_FLAG_TTYIN) 183 184 /// A convenience macro for getting the TTY flag. 185 #define BC_TTY (vm->flags & BC_FLAG_TTY) 186 187 /// A convenience macro for getting the SIGINT flag. 188 #define BC_SIGINT (vm->flags & BC_FLAG_SIGINT) 189 190 #if BC_ENABLED 191 192 /// A convenience macro for getting the POSIX error flag. 193 #define BC_S (vm->flags & BC_FLAG_S) 194 195 /// A convenience macro for getting the POSIX warning flag. 196 #define BC_W (vm->flags & BC_FLAG_W) 197 198 /// A convenience macro for getting the math library flag. 199 #define BC_L (vm->flags & BC_FLAG_L) 200 201 /// A convenience macro for getting the global stacks flag. 202 #define BC_G (vm->flags & BC_FLAG_G) 203 204 #endif // BC_ENABLED 205 206 #if DC_ENABLED 207 208 /// A convenience macro for getting the extended register flag. 209 #define DC_X (vm->flags & DC_FLAG_X) 210 211 #endif // DC_ENABLED 212 213 /// A convenience macro for getting the interactive flag. 214 #define BC_I (vm->flags & BC_FLAG_I) 215 216 /// A convenience macro for getting the prompt flag. 217 #define BC_P (vm->flags & BC_FLAG_P) 218 219 /// A convenience macro for getting the read prompt flag. 220 #define BC_R (vm->flags & BC_FLAG_R) 221 222 /// A convenience macro for getting the leading zero flag. 223 #define BC_Z (vm->flags & BC_FLAG_Z) 224 225 /// A convenience macro for getting the expression exit flag. 226 #define BC_EXPR_EXIT (vm->flags & BC_FLAG_EXPR_EXIT) 227 228 /// A convenience macro for getting the digit clamp flag. 229 #define BC_DIGIT_CLAMP (vm->flags & BC_FLAG_DIGIT_CLAMP) 230 231 #if BC_ENABLED 232 233 /// A convenience macro for checking if bc is in POSIX mode. 234 #define BC_IS_POSIX (BC_S || BC_W) 235 236 #if DC_ENABLED 237 238 /// Returns true if bc is running. 239 #define BC_IS_BC (vm->name[0] != 'd') 240 241 /// Returns true if dc is running. 242 #define BC_IS_DC (vm->name[0] == 'd') 243 244 /// Returns the correct read prompt. 245 #define BC_VM_READ_PROMPT (BC_IS_BC ? "read> " : "?> ") 246 247 /// Returns the string for the line length environment variable. 248 #define BC_VM_LINE_LENGTH_STR (BC_IS_BC ? "BC_LINE_LENGTH" : "DC_LINE_LENGTH") 249 250 /// Returns the string for the environment args environment variable. 251 #define BC_VM_ENV_ARGS_STR (BC_IS_BC ? "BC_ENV_ARGS" : "DC_ENV_ARGS") 252 253 /// Returns the string for the expression exit environment variable. 254 #define BC_VM_EXPR_EXIT_STR (BC_IS_BC ? "BC_EXPR_EXIT" : "DC_EXPR_EXIT") 255 256 /// Returns the default for the expression exit environment variable. 257 #define BC_VM_EXPR_EXIT_DEF \ 258 (BC_IS_BC ? BC_DEFAULT_EXPR_EXIT : DC_DEFAULT_EXPR_EXIT) 259 260 /// Returns the string for the digit clamp environment variable. 261 #define BC_VM_DIGIT_CLAMP_STR (BC_IS_BC ? "BC_DIGIT_CLAMP" : "DC_DIGIT_CLAMP") 262 263 /// Returns the default for the digit clamp environment variable. 264 #define BC_VM_DIGIT_CLAMP_DEF \ 265 (BC_IS_BC ? BC_DEFAULT_DIGIT_CLAMP : DC_DEFAULT_DIGIT_CLAMP) 266 267 /// Returns the string for the TTY mode environment variable. 268 #define BC_VM_TTY_MODE_STR (BC_IS_BC ? "BC_TTY_MODE" : "DC_TTY_MODE") 269 270 /// Returns the default for the TTY mode environment variable. 271 #define BC_VM_TTY_MODE_DEF \ 272 (BC_IS_BC ? BC_DEFAULT_TTY_MODE : DC_DEFAULT_TTY_MODE) 273 274 /// Returns the string for the prompt environment variable. 275 #define BC_VM_PROMPT_STR (BC_IS_BC ? "BC_PROMPT" : "DC_PROMPT") 276 277 /// Returns the default for the prompt environment variable. 278 #define BC_VM_PROMPT_DEF (BC_IS_BC ? BC_DEFAULT_PROMPT : DC_DEFAULT_PROMPT) 279 280 /// Returns the string for the SIGINT reset environment variable. 281 #define BC_VM_SIGINT_RESET_STR \ 282 (BC_IS_BC ? "BC_SIGINT_RESET" : "DC_SIGINT_RESET") 283 284 /// Returns the string for the SIGINT reset environment variable. 285 #define BC_VM_SIGINT_RESET_DEF \ 286 (BC_IS_BC ? BC_DEFAULT_SIGINT_RESET : DC_DEFAULT_SIGINT_RESET) 287 288 /// Returns true if the calculator should run stdin. 289 #define BC_VM_RUN_STDIN(has_file) (BC_IS_BC || !(has_file)) 290 291 #else // DC_ENABLED 292 293 /// Returns true if bc is running. 294 #define BC_IS_BC (1) 295 296 /// Returns true if dc is running. 297 #define BC_IS_DC (0) 298 299 /// Returns the correct read prompt. 300 #define BC_VM_READ_PROMPT ("read> ") 301 302 /// Returns the string for the line length environment variable. 303 #define BC_VM_LINE_LENGTH_STR ("BC_LINE_LENGTH") 304 305 /// Returns the string for the environment args environment variable. 306 #define BC_VM_ENV_ARGS_STR ("BC_ENV_ARGS") 307 308 /// Returns the string for the expression exit environment variable. 309 #define BC_VM_EXPR_EXIT_STR ("BC_EXPR_EXIT") 310 311 /// Returns the default for the expression exit environment variable. 312 #define BC_VM_EXPR_EXIT_DEF (BC_DEFAULT_EXPR_EXIT) 313 314 /// Returns the string for the digit clamp environment variable. 315 #define BC_VM_DIGIT_CLAMP_STR ("BC_DIGIT_CLAMP") 316 317 /// Returns the default for the digit clamp environment variable. 318 #define BC_VM_DIGIT_CLAMP_DEF (BC_DEFAULT_DIGIT_CLAMP) 319 320 /// Returns the string for the TTY mode environment variable. 321 #define BC_VM_TTY_MODE_STR ("BC_TTY_MODE") 322 323 /// Returns the default for the TTY mode environment variable. 324 #define BC_VM_TTY_MODE_DEF (BC_DEFAULT_TTY_MODE) 325 326 /// Returns the string for the prompt environment variable. 327 #define BC_VM_PROMPT_STR ("BC_PROMPT") 328 329 /// Returns the default for the SIGINT reset environment variable. 330 #define BC_VM_PROMPT_DEF (BC_DEFAULT_PROMPT) 331 332 /// Returns the string for the SIGINT reset environment variable. 333 #define BC_VM_SIGINT_RESET_STR ("BC_SIGINT_RESET") 334 335 /// Returns the string for the SIGINT reset environment variable. 336 #define BC_VM_SIGINT_RESET_DEF (BC_DEFAULT_SIGINT_RESET) 337 338 /// Returns true if the calculator should run stdin. 339 #define BC_VM_RUN_STDIN(has_file) (BC_IS_BC) 340 341 #endif // DC_ENABLED 342 343 #else // BC_ENABLED 344 345 /// A convenience macro for checking if bc is in POSIX mode. 346 #define BC_IS_POSIX (0) 347 348 /// Returns true if bc is running. 349 #define BC_IS_BC (0) 350 351 /// Returns true if dc is running. 352 #define BC_IS_DC (1) 353 354 /// Returns the correct read prompt. 355 #define BC_VM_READ_PROMPT ("?> ") 356 357 /// Returns the string for the line length environment variable. 358 #define BC_VM_LINE_LENGTH_STR ("DC_LINE_LENGTH") 359 360 /// Returns the string for the environment args environment variable. 361 #define BC_VM_ENV_ARGS_STR ("DC_ENV_ARGS") 362 363 /// Returns the string for the expression exit environment variable. 364 #define BC_VM_EXPR_EXIT_STR ("DC_EXPR_EXIT") 365 366 /// Returns the default for the expression exit environment variable. 367 #define BC_VM_EXPR_EXIT_DEF (DC_DEFAULT_EXPR_EXIT) 368 369 /// Returns the string for the digit clamp environment variable. 370 #define BC_VM_DIGIT_CLAMP_STR ("DC_DIGIT_CLAMP") 371 372 /// Returns the default for the digit clamp environment variable. 373 #define BC_VM_DIGIT_CLAMP_DEF (DC_DEFAULT_DIGIT_CLAMP) 374 375 /// Returns the string for the TTY mode environment variable. 376 #define BC_VM_TTY_MODE_STR ("DC_TTY_MODE") 377 378 /// Returns the default for the TTY mode environment variable. 379 #define BC_VM_TTY_MODE_DEF (DC_DEFAULT_TTY_MODE) 380 381 /// Returns the string for the prompt environment variable. 382 #define BC_VM_PROMPT_STR ("DC_PROMPT") 383 384 /// Returns the default for the SIGINT reset environment variable. 385 #define BC_VM_PROMPT_DEF (DC_DEFAULT_PROMPT) 386 387 /// Returns the string for the SIGINT reset environment variable. 388 #define BC_VM_SIGINT_RESET_STR ("DC_SIGINT_RESET") 389 390 /// Returns the string for the SIGINT reset environment variable. 391 #define BC_VM_SIGINT_RESET_DEF (DC_DEFAULT_SIGINT_RESET) 392 393 /// Returns true if the calculator should run stdin. 394 #define BC_VM_RUN_STDIN(has_file) (!(has_file)) 395 396 #endif // BC_ENABLED 397 398 /// A convenience macro for checking if the prompt is enabled. 399 #define BC_PROMPT (BC_P) 400 401 #else // !BC_ENABLE_LIBRARY 402 403 #define BC_Z (vm->leading_zeroes) 404 405 #define BC_DIGIT_CLAMP (vm->digit_clamp) 406 407 #endif // !BC_ENABLE_LIBRARY 408 409 /** 410 * Returns the max of its two arguments. This evaluates arguments twice, so be 411 * careful what args you give it. 412 * @param a The first argument. 413 * @param b The second argument. 414 * @return The max of the two arguments. 415 */ 416 #define BC_MAX(a, b) ((a) > (b) ? (a) : (b)) 417 418 /** 419 * Returns the min of its two arguments. This evaluates arguments twice, so be 420 * careful what args you give it. 421 * @param a The first argument. 422 * @param b The second argument. 423 * @return The min of the two arguments. 424 */ 425 #define BC_MIN(a, b) ((a) < (b) ? (a) : (b)) 426 427 /// Returns the max obase that is allowed. 428 #define BC_MAX_OBASE ((BcBigDig) (BC_BASE_POW)) 429 430 /// Returns the max array size that is allowed. 431 #define BC_MAX_DIM ((BcBigDig) (SIZE_MAX - 1)) 432 433 /// Returns the max scale that is allowed. 434 #define BC_MAX_SCALE ((BcBigDig) (BC_NUM_BIGDIG_MAX - 1)) 435 436 /// Returns the max string length that is allowed. 437 #define BC_MAX_STRING ((BcBigDig) (BC_NUM_BIGDIG_MAX - 1)) 438 439 /// Returns the max identifier length that is allowed. 440 #define BC_MAX_NAME BC_MAX_STRING 441 442 /// Returns the max number size that is allowed. 443 #define BC_MAX_NUM BC_MAX_SCALE 444 445 #if BC_ENABLE_EXTRA_MATH 446 447 /// Returns the max random integer that can be returned. 448 #define BC_MAX_RAND ((BcBigDig) (((BcRand) 0) - 1)) 449 450 #endif // BC_ENABLE_EXTRA_MATH 451 452 /// Returns the max exponent that is allowed. 453 #define BC_MAX_EXP ((ulong) (BC_NUM_BIGDIG_MAX)) 454 455 /// Returns the max number of variables that is allowed. 456 #define BC_MAX_VARS ((ulong) (SIZE_MAX - 1)) 457 458 #if BC_ENABLE_LINE_LIB 459 460 /// The size of the global buffer. 461 #define BC_VM_BUF_SIZE (1 << 10) 462 463 /// The amount of the global buffer allocated to stdin. 464 #define BC_VM_STDIN_BUF_SIZE (BC_VM_BUF_SIZE - 1) 465 466 #else // BC_ENABLE_LINE_LIB 467 468 /// The size of the global buffer. 469 #define BC_VM_BUF_SIZE (1 << 12) 470 471 /// The amount of the global buffer allocated to stdout. 472 #define BC_VM_STDOUT_BUF_SIZE (1 << 11) 473 474 /// The amount of the global buffer allocated to stderr. 475 #define BC_VM_STDERR_BUF_SIZE (1 << 10) 476 477 /// The amount of the global buffer allocated to stdin. 478 #define BC_VM_STDIN_BUF_SIZE (BC_VM_STDERR_BUF_SIZE - 1) 479 480 #endif // BC_ENABLE_LINE_LIB 481 482 /// The max number of temporary BcNums that can be kept. 483 #define BC_VM_MAX_TEMPS (1 << 9) 484 485 /// The capacity of the one BcNum, which is a constant. 486 #define BC_VM_ONE_CAP (1) 487 488 /** 489 * Returns true if a BcResult is safe for garbage collection. 490 * @param r The BcResult to test. 491 * @return True if @a r is safe to garbage collect. 492 */ 493 #define BC_VM_SAFE_RESULT(r) ((r)->t >= BC_RESULT_TEMP) 494 495 /// The invalid locale catalog return value. 496 #define BC_VM_INVALID_CATALOG ((nl_catd) - 1) 497 498 /** 499 * Returns true if the *unsigned* multiplication overflows. 500 * @param a The first operand. 501 * @param b The second operand. 502 * @param r The product. 503 * @return True if the multiplication of @a a and @a b overflows. 504 */ 505 #define BC_VM_MUL_OVERFLOW(a, b, r) \ 506 ((r) >= SIZE_MAX || ((a) != 0 && (r) / (a) != (b))) 507 508 /// The global vm struct. This holds all of the global data besides the file 509 /// buffers. 510 typedef struct BcVm 511 { 512 /// The current status. This is volatile sig_atomic_t because it is also 513 /// used in the signal handler. See the development manual 514 /// (manuals/development.md#async-signal-safe-signal-handling) for more 515 /// information. 516 volatile sig_atomic_t status; 517 518 /// Non-zero if a jump series is in progress and items should be popped off 519 /// the jmp_bufs vector. This is volatile sig_atomic_t because it is also 520 /// used in the signal handler. See the development manual 521 /// (manuals/development.md#async-signal-safe-signal-handling) for more 522 /// information. 523 volatile sig_atomic_t sig_pop; 524 525 #if !BC_ENABLE_LIBRARY 526 527 /// The parser. 528 BcParse prs; 529 530 /// The program. 531 BcProgram prog; 532 533 /// A buffer for lines for stdin. 534 BcVec line_buf; 535 536 /// A buffer to hold a series of lines from stdin. Sometimes, multiple lines 537 /// are necessary for parsing, such as a comment that spans multiple lines. 538 BcVec buffer; 539 540 /// A parser to parse read expressions. 541 BcParse read_prs; 542 543 /// A buffer for read expressions. 544 BcVec read_buf; 545 546 #endif // !BC_ENABLE_LIBRARY 547 548 /// A vector of jmp_bufs for doing a jump series. This allows exception-type 549 /// error handling, while allowing me to do cleanup on the way. 550 BcVec jmp_bufs; 551 552 /// The number of temps in the temps array. 553 size_t temps_len; 554 555 #if BC_ENABLE_LIBRARY 556 557 /// The vector of contexts for the library. 558 BcVec ctxts; 559 560 /// The vector for creating strings to pass to the client. 561 BcVec out; 562 563 #if BC_ENABLE_EXTRA_MATH 564 565 /// The PRNG. 566 BcRNG rng; 567 568 #endif // BC_ENABLE_EXTRA_MATH 569 570 /// The current error. 571 BclError err; 572 573 /// Whether or not bcl should abort on fatal errors. 574 bool abrt; 575 576 /// Whether or not to print leading zeros. 577 bool leading_zeroes; 578 579 /// Whether or not to clamp digits that are greater than or equal to the 580 /// current ibase. 581 bool digit_clamp; 582 583 /// The number of "references," or times that the library was initialized. 584 unsigned int refs; 585 586 #else // BC_ENABLE_LIBRARY 587 588 /// A pointer to the filename of the current file. This is not owned by the 589 /// BcVm struct. 590 const char* file; 591 592 /// The message printed when SIGINT happens. 593 const char* sigmsg; 594 595 /// Non-zero when signals are "locked." This is volatile sig_atomic_t 596 /// because it is also used in the signal handler. See the development 597 /// manual (manuals/development.md#async-signal-safe-signal-handling) for 598 /// more information. 599 volatile sig_atomic_t sig_lock; 600 601 /// Non-zero when a signal has been received, but not acted on. This is 602 /// volatile sig_atomic_t because it is also used in the signal handler. See 603 /// the development manual 604 /// (manuals/development.md#async-signal-safe-signal-handling) for more 605 /// information. 606 volatile sig_atomic_t sig; 607 608 /// The length of sigmsg. 609 uchar siglen; 610 611 /// The instruction used for returning from a read() call. 612 uchar read_ret; 613 614 /// The flags field used by most macros above. 615 uint16_t flags; 616 617 /// The number of characters printed in the current line. This is used 618 /// because bc has a limit of the number of characters it can print per 619 /// line. 620 uint16_t nchars; 621 622 /// The length of the line we can print. The user can set this if they wish. 623 uint16_t line_len; 624 625 /// True if bc should error if expressions are encountered during option 626 /// parsing, false otherwise. 627 bool no_exprs; 628 629 /// True if bc should exit if expresions are encountered. 630 bool exit_exprs; 631 632 /// True if EOF was encountered. 633 bool eof; 634 635 /// The mode that the program is in. 636 uchar mode; 637 638 #if BC_ENABLED 639 640 /// True if keywords should not be redefined. This is only true for the 641 /// builtin math libraries for bc. 642 bool no_redefine; 643 644 #endif // BC_ENABLED 645 646 /// A vector of filenames to process. 647 BcVec files; 648 649 /// A vector of expressions to process. 650 BcVec exprs; 651 652 /// The name of the calculator under use. This is used by BC_IS_BC and 653 /// BC_IS_DC. 654 const char* name; 655 656 /// The help text for the calculator. 657 const char* help; 658 659 #if BC_ENABLE_HISTORY 660 661 /// The history data. 662 BcHistory history; 663 664 #endif // BC_ENABLE_HISTORY 665 666 /// The function to call to get the next lex token. 667 BcLexNext next; 668 669 /// The function to call to parse. 670 BcParseParse parse; 671 672 /// The function to call to parse expressions. 673 BcParseExpr expr; 674 675 /// The names of the categories of errors. 676 const char* err_ids[BC_ERR_IDX_NELEMS + BC_ENABLED]; 677 678 /// The messages for each error. 679 const char* err_msgs[BC_ERR_NELEMS]; 680 681 #if BC_ENABLE_NLS 682 /// The locale. 683 const char* locale; 684 #endif // BC_ENABLE_NLS 685 686 #endif // BC_ENABLE_LIBRARY 687 688 /// An array of maxes for the globals. 689 BcBigDig maxes[BC_PROG_GLOBALS_LEN + BC_ENABLE_EXTRA_MATH]; 690 691 /// The last base used to parse. 692 BcBigDig last_base; 693 694 /// The last power of last_base used to parse. 695 BcBigDig last_pow; 696 697 /// The last exponent of base that equals last_pow. 698 BcBigDig last_exp; 699 700 /// BC_BASE_POW - last_pow. 701 BcBigDig last_rem; 702 703 #if !BC_ENABLE_LIBRARY 704 705 /// A buffer of environment arguments. This is the actual value of the 706 /// environment variable. 707 char* env_args_buffer; 708 709 /// A vector for environment arguments after parsing. 710 BcVec env_args; 711 712 /// A BcNum set to constant 0. 713 BcNum zero; 714 715 #endif // !BC_ENABLE_LIBRARY 716 717 /// A BcNum set to constant 1. 718 BcNum one; 719 720 /// A BcNum holding the max number held by a BcBigDig plus 1. 721 BcNum max; 722 723 /// A BcNum holding the max number held by a BcBigDig times 2 plus 1. 724 BcNum max2; 725 726 /// The BcDig array for max. 727 BcDig max_num[BC_NUM_BIGDIG_LOG10]; 728 729 /// The BcDig array for max2. 730 BcDig max2_num[BC_NUM_BIGDIG_LOG10]; 731 732 // The BcDig array for the one BcNum. 733 BcDig one_num[BC_VM_ONE_CAP]; 734 735 #if !BC_ENABLE_LIBRARY 736 737 // The BcDig array for the zero BcNum. 738 BcDig zero_num[BC_VM_ONE_CAP]; 739 740 /// The stdout file. 741 BcFile fout; 742 743 /// The stderr file. 744 BcFile ferr; 745 746 #if BC_ENABLE_NLS 747 748 /// The locale catalog. 749 nl_catd catalog; 750 751 #endif // BC_ENABLE_NLS 752 753 /// A pointer to the stdin buffer. 754 char* buf; 755 756 /// The number of items in the input buffer. 757 size_t buf_len; 758 759 /// The slabs vector for constants, strings, function names, and other 760 /// string-like things. 761 BcVec slabs; 762 763 #if BC_ENABLED 764 765 /// An array of booleans for which bc keywords have been redefined if 766 /// BC_REDEFINE_KEYWORDS is non-zero. 767 bool redefined_kws[BC_LEX_NKWS]; 768 769 #endif // BC_ENABLED 770 #endif // !BC_ENABLE_LIBRARY 771 772 BcDig* temps_buf[BC_VM_MAX_TEMPS]; 773 774 #if BC_DEBUG_CODE 775 776 /// The depth for BC_FUNC_ENTER and BC_FUNC_EXIT. 777 size_t func_depth; 778 779 #endif // BC_DEBUG_CODE 780 781 } BcVm; 782 783 /** 784 * Print the copyright banner and help if it's non-NULL. 785 * @param help The help message to print if it's non-NULL. 786 */ 787 void 788 bc_vm_info(const char* const help); 789 790 /** 791 * The entrance point for bc/dc together. 792 * @param argc The count of arguments. 793 * @param argv The argument array. 794 * @return A status. 795 */ 796 BcStatus 797 bc_vm_boot(int argc, const char* argv[]); 798 799 /** 800 * Initializes some of the BcVm global. This is separate to make things easier 801 * on the library code. 802 */ 803 void 804 bc_vm_init(void); 805 806 /** 807 * Frees the BcVm global. 808 */ 809 void 810 bc_vm_shutdown(void); 811 812 /** 813 * Add a temp to the temp array. 814 * @param num The BcDig array to add to the temp array. 815 */ 816 void 817 bc_vm_addTemp(BcDig* num); 818 819 /** 820 * Return the temp on the top of the temp stack, or NULL if there are none. 821 * @return A temp, or NULL if none exist. 822 */ 823 BcDig* 824 bc_vm_takeTemp(void); 825 826 /** 827 * Gets the top temp of the temp stack. This is separate from bc_vm_takeTemp() 828 * to quiet a GCC warning about longjmp() clobbering in bc_num_init(). 829 * @return A temp, or NULL if none exist. 830 */ 831 BcDig* 832 bc_vm_getTemp(void); 833 834 /** 835 * Frees all temporaries. 836 */ 837 void 838 bc_vm_freeTemps(void); 839 840 #if !BC_ENABLE_HISTORY || BC_ENABLE_LINE_LIB || BC_ENABLE_LIBRARY 841 842 /** 843 * Erases the flush argument if history does not exist because it does not 844 * matter if history does not exist. 845 */ 846 #define bc_vm_putchar(c, t) bc_vm_putchar_impl(c) 847 848 #else // !BC_ENABLE_HISTORY || BC_ENABLE_LINE_LIB || BC_ENABLE_LIBRARY 849 850 // This is here to satisfy a clang warning about recursive macros. 851 #define bc_vm_putchar(c, t) bc_vm_putchar_impl(c, t) 852 853 #endif // !BC_ENABLE_HISTORY || BC_ENABLE_LINE_LIB || BC_ENABLE_LIBRARY 854 855 /** 856 * Print to stdout with limited formating. 857 * @param fmt The format string. 858 */ 859 void 860 bc_vm_printf(const char* fmt, ...); 861 862 /** 863 * Puts a char into the stdout buffer. 864 * @param c The character to put on the stdout buffer. 865 * @param type The flush type. 866 */ 867 void 868 bc_vm_putchar(int c, BcFlushType type); 869 870 /** 871 * Multiplies @a n and @a size and throws an allocation error if overflow 872 * occurs. 873 * @param n The number of elements. 874 * @param size The size of each element. 875 * @return The product of @a n and @a size. 876 */ 877 size_t 878 bc_vm_arraySize(size_t n, size_t size); 879 880 /** 881 * Adds @a a and @a b and throws an error if overflow occurs. 882 * @param a The first operand. 883 * @param b The second operand. 884 * @return The sum of @a a and @a b. 885 */ 886 size_t 887 bc_vm_growSize(size_t a, size_t b); 888 889 /** 890 * Allocate @a n bytes and throw an allocation error if allocation fails. 891 * @param n The bytes to allocate. 892 * @return A pointer to the allocated memory. 893 */ 894 void* 895 bc_vm_malloc(size_t n); 896 897 /** 898 * Reallocate @a ptr to be @a n bytes and throw an allocation error if 899 * reallocation fails. 900 * @param ptr The pointer to a memory allocation to reallocate. 901 * @param n The bytes to allocate. 902 * @return A pointer to the reallocated memory. 903 */ 904 void* 905 bc_vm_realloc(void* ptr, size_t n); 906 907 /** 908 * Allocates space for, and duplicates, @a str. 909 * @param str The string to allocate. 910 * @return The allocated string. 911 */ 912 char* 913 bc_vm_strdup(const char* str); 914 915 /** 916 * Reads a line from stdin into BcVm's buffer field. 917 * @param clear True if the buffer should be cleared first, false otherwise. 918 * @return True if a line was read, false otherwise. 919 */ 920 bool 921 bc_vm_readLine(bool clear); 922 923 /** 924 * Reads a line from the command-line expressions into BcVm's buffer field. 925 * @param clear True if the buffer should be cleared first, false otherwise. 926 * @return True if a line was read, false otherwise. 927 */ 928 bool 929 bc_vm_readBuf(bool clear); 930 931 /** 932 * A convenience and portability function for OpenBSD's pledge(). 933 * @param promises The promises to pledge(). 934 * @param execpromises The exec promises to pledge(). 935 */ 936 void 937 bc_pledge(const char* promises, const char* execpromises); 938 939 /** 940 * Returns the value of an environment variable. 941 * @param var The environment variable. 942 * @return The value of the environment variable. 943 */ 944 char* 945 bc_vm_getenv(const char* var); 946 947 /** 948 * Frees an environment variable value. 949 * @param val The value to free. 950 */ 951 void 952 bc_vm_getenvFree(char* val); 953 954 #if BC_DEBUG_CODE 955 956 /** 957 * Start executing a jump series. 958 * @param f The name of the function that started the jump series. 959 */ 960 void 961 bc_vm_jmp(const char* f); 962 963 #else // BC_DEBUG_CODE 964 965 /** 966 * Start executing a jump series. 967 */ 968 void 969 bc_vm_jmp(void); 970 971 #endif // BC_DEBUG_CODE 972 973 #if BC_ENABLE_LIBRARY 974 975 /** 976 * Handle an error. This is the true error handler. It will start a jump series 977 * if an error occurred. POSIX errors will not cause jumps when warnings are on 978 * or no POSIX errors are enabled. 979 * @param e The error. 980 */ 981 void 982 bc_vm_handleError(BcErr e); 983 984 /** 985 * Handle a fatal error. 986 * @param e The error. 987 */ 988 void 989 bc_vm_fatalError(BcErr e); 990 991 /** 992 * A function to call at exit. 993 */ 994 void 995 bc_vm_atexit(void); 996 997 #else // BC_ENABLE_LIBRARY 998 999 /** 1000 * Calculates the number of decimal digits in the argument. 1001 * @param val The value to calculate the number of decimal digits in. 1002 * @return The number of decimal digits in @a val. 1003 */ 1004 size_t 1005 bc_vm_numDigits(size_t val); 1006 1007 #if BC_DEBUG 1008 1009 /** 1010 * Handle an error. This is the true error handler. It will start a jump series 1011 * if an error occurred. POSIX errors will not cause jumps when warnings are on 1012 * or no POSIX errors are enabled. 1013 * @param e The error. 1014 * @param file The source file where the error occurred. 1015 * @param fline The line in the source file where the error occurred. 1016 * @param line The bc source line where the error occurred. 1017 */ 1018 void 1019 bc_vm_handleError(BcErr e, const char* file, int fline, size_t line, ...); 1020 1021 #else // BC_DEBUG 1022 1023 /** 1024 * Handle an error. This is the true error handler. It will start a jump series 1025 * if an error occurred. POSIX errors will not cause jumps when warnings are on 1026 * or no POSIX errors are enabled. 1027 * @param e The error. 1028 * @param line The bc source line where the error occurred. 1029 */ 1030 void 1031 bc_vm_handleError(BcErr e, size_t line, ...); 1032 1033 #endif // BC_DEBUG 1034 1035 /** 1036 * Handle a fatal error. 1037 * @param e The error. 1038 */ 1039 #if !BC_ENABLE_MEMCHECK 1040 BC_NORETURN 1041 #endif // !BC_ENABLE_MEMCHECK 1042 void 1043 bc_vm_fatalError(BcErr e); 1044 1045 /** 1046 * A function to call at exit. 1047 * @param status The exit status. 1048 */ 1049 BcStatus 1050 bc_vm_atexit(BcStatus status); 1051 1052 #endif // BC_ENABLE_LIBRARY 1053 1054 /// A reference to the copyright header. 1055 extern const char bc_copyright[]; 1056 1057 /// A reference to the array of default error category names. 1058 extern const char* bc_errs[]; 1059 1060 /// A reference to the array of error category indices for each error. 1061 extern const uchar bc_err_ids[]; 1062 1063 /// A reference to the array of default error messages. 1064 extern const char* const bc_err_msgs[]; 1065 1066 /// A reference to the pledge() promises at start. 1067 extern const char bc_pledge_start[]; 1068 1069 #if BC_ENABLE_HISTORY 1070 1071 /// A reference to the end pledge() promises when using history. 1072 extern const char bc_pledge_end_history[]; 1073 1074 #endif // BC_ENABLE_HISTORY 1075 1076 /// A reference to the end pledge() promises when *not* using history. 1077 extern const char bc_pledge_end[]; 1078 1079 #if !BC_ENABLE_LIBRARY 1080 1081 /// A reference to the global data. 1082 extern BcVm* vm; 1083 1084 /// The global data. 1085 extern BcVm vm_data; 1086 1087 /// A reference to the global output buffers. 1088 extern char output_bufs[BC_VM_BUF_SIZE]; 1089 1090 #endif // !BC_ENABLE_LIBRARY 1091 1092 #endif // BC_VM_H 1093