1 /* 2 * ***************************************************************************** 3 * 4 * SPDX-License-Identifier: BSD-2-Clause 5 * 6 * Copyright (c) 2018-2021 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 BC_ENABLE_NLS 83 #define BC_ENABLE_NLS (0) 84 #endif // BC_ENABLE_NLS 85 86 #ifndef MAINEXEC 87 #define MAINEXEC bc 88 #endif // MAINEXEC 89 90 #ifndef _WIN32 91 #ifndef EXECPREFIX 92 #define EXECPREFIX 93 #endif // EXECPREFIX 94 #else // _WIN32 95 #undef EXECPREFIX 96 #endif // _WIN32 97 98 /** 99 * Generate a string from text. 100 * @parm V The text to generate a string for. 101 */ 102 #define GEN_STR(V) #V 103 104 /** 105 * Help generate a string from text. The preprocessor requires this two-step 106 * process. Trust me. 107 * @parm V The text to generate a string for. 108 */ 109 #define GEN_STR2(V) GEN_STR(V) 110 111 /// The version as a string. VERSION must be defined previously, usually by the 112 /// build system. 113 #define BC_VERSION GEN_STR2(VERSION) 114 115 /// The main executable name as a string. MAINEXEC must be defined previously, 116 /// usually by the build system. 117 #define BC_MAINEXEC GEN_STR2(MAINEXEC) 118 119 /// The build type as a string. BUILD_TYPE must be defined previously, usually 120 /// by the build system. 121 #define BC_BUILD_TYPE GEN_STR2(BUILD_TYPE) 122 123 // We only allow an empty executable prefix on Windows. 124 #ifndef _WIN32 125 #define BC_EXECPREFIX GEN_STR2(EXECPREFIX) 126 #else // _WIN32 127 #define BC_EXECPREFIX "" 128 #endif // _WIN32 129 130 #if !BC_ENABLE_LIBRARY 131 132 #if DC_ENABLED 133 134 /// The flag for the extended register option. 135 #define DC_FLAG_X (UINTMAX_C(1) << 0) 136 137 #endif // DC_ENABLED 138 139 #if BC_ENABLED 140 141 /// The flag for the POSIX warning option. 142 #define BC_FLAG_W (UINTMAX_C(1) << 1) 143 144 /// The flag for the POSIX error option. 145 #define BC_FLAG_S (UINTMAX_C(1) << 2) 146 147 /// The flag for the math library option. 148 #define BC_FLAG_L (UINTMAX_C(1) << 3) 149 150 /// The flag for the global stacks option. 151 #define BC_FLAG_G (UINTMAX_C(1) << 4) 152 153 #endif // BC_ENABLED 154 155 /// The flag for quiet, though this one is reversed; the option clears the flag. 156 #define BC_FLAG_Q (UINTMAX_C(1) << 5) 157 158 /// The flag for interactive. 159 #define BC_FLAG_I (UINTMAX_C(1) << 6) 160 161 /// The flag for prompt. This is also reversed; the option clears the flag. 162 #define BC_FLAG_P (UINTMAX_C(1) << 7) 163 164 /// The flag for read prompt. This is also reversed; the option clears the flag. 165 #define BC_FLAG_R (UINTMAX_C(1) << 8) 166 167 /// The flag for a leading zero. 168 #define BC_FLAG_Z (UINTMAX_C(1) << 9) 169 170 /// The flag for stdin being a TTY. 171 #define BC_FLAG_TTYIN (UINTMAX_C(1) << 10) 172 173 /// The flag for TTY mode. 174 #define BC_FLAG_TTY (UINTMAX_C(1) << 11) 175 176 /// The flag for reset on SIGINT. 177 #define BC_FLAG_SIGINT (UINTMAX_C(1) << 12) 178 179 /// The flag for exiting with expressions. 180 #define BC_FLAG_EXPR_EXIT (UINTMAX_C(1) << 13) 181 182 /// A convenience macro for getting the TTYIN flag. 183 #define BC_TTYIN (vm.flags & BC_FLAG_TTYIN) 184 185 /// A convenience macro for getting the TTY flag. 186 #define BC_TTY (vm.flags & BC_FLAG_TTY) 187 188 /// A convenience macro for getting the SIGINT flag. 189 #define BC_SIGINT (vm.flags & BC_FLAG_SIGINT) 190 191 #if BC_ENABLED 192 193 /// A convenience macro for getting the POSIX error flag. 194 #define BC_S (vm.flags & BC_FLAG_S) 195 196 /// A convenience macro for getting the POSIX warning flag. 197 #define BC_W (vm.flags & BC_FLAG_W) 198 199 /// A convenience macro for getting the math library flag. 200 #define BC_L (vm.flags & BC_FLAG_L) 201 202 /// A convenience macro for getting the global stacks flag. 203 #define BC_G (vm.flags & BC_FLAG_G) 204 205 #endif // BC_ENABLED 206 207 #if DC_ENABLED 208 209 /// A convenience macro for getting the extended register flag. 210 #define DC_X (vm.flags & DC_FLAG_X) 211 212 #endif // DC_ENABLED 213 214 /// A convenience macro for getting the interactive flag. 215 #define BC_I (vm.flags & BC_FLAG_I) 216 217 /// A convenience macro for getting the prompt flag. 218 #define BC_P (vm.flags & BC_FLAG_P) 219 220 /// A convenience macro for getting the read prompt flag. 221 #define BC_R (vm.flags & BC_FLAG_R) 222 223 /// A convenience macro for getting the leading zero flag. 224 #define BC_Z (vm.flags & BC_FLAG_Z) 225 226 /// A convenience macro for getting the expression exit flag. 227 #define BC_EXPR_EXIT (vm.flags & BC_FLAG_EXPR_EXIT) 228 229 #if BC_ENABLED 230 231 /// A convenience macro for checking if bc is in POSIX mode. 232 #define BC_IS_POSIX (BC_S || BC_W) 233 234 #if DC_ENABLED 235 236 /// Returns true if bc is running. 237 #define BC_IS_BC (vm.name[0] != 'd') 238 239 /// Returns true if dc is running. 240 #define BC_IS_DC (vm.name[0] == 'd') 241 242 #else // DC_ENABLED 243 244 /// Returns true if bc is running. 245 #define BC_IS_BC (1) 246 247 /// Returns true if dc is running. 248 #define BC_IS_DC (0) 249 250 #endif // DC_ENABLED 251 252 #else // BC_ENABLED 253 254 /// A convenience macro for checking if bc is in POSIX mode. 255 #define BC_IS_POSIX (0) 256 257 /// Returns true if bc is running. 258 #define BC_IS_BC (0) 259 260 /// Returns true if dc is running. 261 #define BC_IS_DC (1) 262 263 #endif // BC_ENABLED 264 265 /// A convenience macro for checking if the prompt is enabled. 266 #define BC_PROMPT (BC_P) 267 268 #else // !BC_ENABLE_LIBRARY 269 270 #define BC_Z (vm.leading_zeroes) 271 272 #endif // !BC_ENABLE_LIBRARY 273 274 /** 275 * Returns the max of its two arguments. This evaluates arguments twice, so be 276 * careful what args you give it. 277 * @param a The first argument. 278 * @param b The second argument. 279 * @return The max of the two arguments. 280 */ 281 #define BC_MAX(a, b) ((a) > (b) ? (a) : (b)) 282 283 /** 284 * Returns the min of its two arguments. This evaluates arguments twice, so be 285 * careful what args you give it. 286 * @param a The first argument. 287 * @param b The second argument. 288 * @return The min of the two arguments. 289 */ 290 #define BC_MIN(a, b) ((a) < (b) ? (a) : (b)) 291 292 /// Returns the max obase that is allowed. 293 #define BC_MAX_OBASE ((BcBigDig) (BC_BASE_POW)) 294 295 /// Returns the max array size that is allowed. 296 #define BC_MAX_DIM ((BcBigDig) (SIZE_MAX - 1)) 297 298 /// Returns the max scale that is allowed. 299 #define BC_MAX_SCALE ((BcBigDig) (BC_NUM_BIGDIG_MAX - 1)) 300 301 /// Returns the max string length that is allowed. 302 #define BC_MAX_STRING ((BcBigDig) (BC_NUM_BIGDIG_MAX - 1)) 303 304 /// Returns the max identifier length that is allowed. 305 #define BC_MAX_NAME BC_MAX_STRING 306 307 /// Returns the max number size that is allowed. 308 #define BC_MAX_NUM BC_MAX_SCALE 309 310 #if BC_ENABLE_EXTRA_MATH 311 312 /// Returns the max random integer that can be returned. 313 #define BC_MAX_RAND ((BcBigDig) (((BcRand) 0) - 1)) 314 315 #endif // BC_ENABLE_EXTRA_MATH 316 317 /// Returns the max exponent that is allowed. 318 #define BC_MAX_EXP ((ulong) (BC_NUM_BIGDIG_MAX)) 319 320 /// Returns the max number of variables that is allowed. 321 #define BC_MAX_VARS ((ulong) (SIZE_MAX - 1)) 322 323 #if BC_ENABLE_LINE_LIB 324 325 /// The size of the global buffer. 326 #define BC_VM_BUF_SIZE (1 << 10) 327 328 /// The amount of the global buffer allocated to stdin. 329 #define BC_VM_STDIN_BUF_SIZE (BC_VM_BUF_SIZE - 1) 330 331 #else // BC_ENABLE_LINE_LIB 332 333 /// The size of the global buffer. 334 #define BC_VM_BUF_SIZE (1 << 12) 335 336 /// The amount of the global buffer allocated to stdout. 337 #define BC_VM_STDOUT_BUF_SIZE (1 << 11) 338 339 /// The amount of the global buffer allocated to stderr. 340 #define BC_VM_STDERR_BUF_SIZE (1 << 10) 341 342 /// The amount of the global buffer allocated to stdin. 343 #define BC_VM_STDIN_BUF_SIZE (BC_VM_STDERR_BUF_SIZE - 1) 344 345 #endif // BC_ENABLE_LINE_LIB 346 347 /// The max number of temporary BcNums that can be kept. 348 #define BC_VM_MAX_TEMPS (1 << 9) 349 350 /// The capacity of the one BcNum, which is a constant. 351 #define BC_VM_ONE_CAP (1) 352 353 /** 354 * Returns true if a BcResult is safe for garbage collection. 355 * @param r The BcResult to test. 356 * @return True if @a r is safe to garbage collect. 357 */ 358 #define BC_VM_SAFE_RESULT(r) ((r)->t >= BC_RESULT_TEMP) 359 360 /// The invalid locale catalog return value. 361 #define BC_VM_INVALID_CATALOG ((nl_catd) -1) 362 363 /** 364 * Returns true if the *unsigned* multiplication overflows. 365 * @param a The first operand. 366 * @param b The second operand. 367 * @param r The product. 368 * @return True if the multiplication of @a a and @a b overflows. 369 */ 370 #define BC_VM_MUL_OVERFLOW(a, b, r) \ 371 ((r) >= SIZE_MAX || ((a) != 0 && (r) / (a) != (b))) 372 373 /// The global vm struct. This holds all of the global data besides the file 374 /// buffers. 375 typedef struct BcVm 376 { 377 /// The current status. This is volatile sig_atomic_t because it is also 378 /// used in the signal handler. See the development manual 379 /// (manuals/development.md#async-signal-safe-signal-handling) for more 380 /// information. 381 volatile sig_atomic_t status; 382 383 /// Non-zero if a jump series is in progress and items should be popped off 384 /// the jmp_bufs vector. This is volatile sig_atomic_t because it is also 385 /// used in the signal handler. See the development manual 386 /// (manuals/development.md#async-signal-safe-signal-handling) for more 387 /// information. 388 volatile sig_atomic_t sig_pop; 389 390 #if !BC_ENABLE_LIBRARY 391 392 /// The parser. 393 BcParse prs; 394 395 /// The program. 396 BcProgram prog; 397 398 /// A buffer for lines for stdin. 399 BcVec line_buf; 400 401 /// A buffer to hold a series of lines from stdin. Sometimes, multiple lines 402 /// are necessary for parsing, such as a comment that spans multiple lines. 403 BcVec buffer; 404 405 /// A parser to parse read expressions. 406 BcParse read_prs; 407 408 /// A buffer for read expressions. 409 BcVec read_buf; 410 411 #endif // !BC_ENABLE_LIBRARY 412 413 /// A vector of jmp_bufs for doing a jump series. This allows exception-type 414 /// error handling, while allowing me to do cleanup on the way. 415 BcVec jmp_bufs; 416 417 /// The number of temps in the temps array. 418 size_t temps_len; 419 420 #if BC_ENABLE_LIBRARY 421 422 /// The vector of contexts for the library. 423 BcVec ctxts; 424 425 /// The vector for creating strings to pass to the client. 426 BcVec out; 427 428 /// The PRNG. 429 BcRNG rng; 430 431 /// The current error. 432 BclError err; 433 434 /// Whether or not bcl should abort on fatal errors. 435 bool abrt; 436 437 /// Whether or not to print leading zeros. 438 bool leading_zeroes; 439 440 /// The number of "references," or times that the library was initialized. 441 unsigned int refs; 442 443 /// Non-zero if bcl is running. This is volatile sig_atomic_t because it is 444 /// also used in the signal handler. See the development manual 445 /// (manuals/development.md#async-signal-safe-signal-handling) for more 446 /// information. 447 volatile sig_atomic_t running; 448 449 #endif // BC_ENABLE_LIBRARY 450 451 #if !BC_ENABLE_LIBRARY 452 453 /// A pointer to the filename of the current file. This is not owned by the 454 /// BcVm struct. 455 const char* file; 456 457 /// The message printed when SIGINT happens. 458 const char* sigmsg; 459 460 #endif // !BC_ENABLE_LIBRARY 461 462 /// Non-zero when signals are "locked." This is volatile sig_atomic_t 463 /// because it is also used in the signal handler. See the development 464 /// manual (manuals/development.md#async-signal-safe-signal-handling) for 465 /// more information. 466 volatile sig_atomic_t sig_lock; 467 468 /// Non-zero when a signal has been received, but not acted on. This is 469 /// volatile sig_atomic_t because it is also used in the signal handler. See 470 /// the development manual 471 /// (manuals/development.md#async-signal-safe-signal-handling) for more 472 /// information. 473 volatile sig_atomic_t sig; 474 475 #if !BC_ENABLE_LIBRARY 476 477 /// The length of sigmsg. 478 uchar siglen; 479 480 /// The instruction used for returning from a read() call. 481 uchar read_ret; 482 483 /// The flags field used by most macros above. 484 uint16_t flags; 485 486 /// The number of characters printed in the current line. This is used 487 /// because bc has a limit of the number of characters it can print per 488 /// line. 489 uint16_t nchars; 490 491 /// The length of the line we can print. The user can set this if they wish. 492 uint16_t line_len; 493 494 /// True if bc should error if expressions are encountered during option 495 /// parsing, false otherwise. 496 bool no_exprs; 497 498 /// True if bc should exit if expresions are encountered. 499 bool exit_exprs; 500 501 /// True if EOF was encountered. 502 bool eof; 503 504 /// True if bc is currently reading from stdin. 505 bool is_stdin; 506 507 #if BC_ENABLED 508 509 /// True if keywords should not be redefined. This is only true for the 510 /// builtin math libraries for bc. 511 bool no_redefine; 512 513 #endif // BC_ENABLED 514 515 #endif // !BC_ENABLE_LIBRARY 516 517 /// An array of maxes for the globals. 518 BcBigDig maxes[BC_PROG_GLOBALS_LEN + BC_ENABLE_EXTRA_MATH]; 519 520 #if !BC_ENABLE_LIBRARY 521 522 /// A vector of filenames to process. 523 BcVec files; 524 525 /// A vector of expressions to process. 526 BcVec exprs; 527 528 /// The name of the calculator under use. This is used by BC_IS_BC and 529 /// BC_IS_DC. 530 const char* name; 531 532 /// The help text for the calculator. 533 const char* help; 534 535 #if BC_ENABLE_HISTORY 536 537 /// The history data. 538 BcHistory history; 539 540 #endif // BC_ENABLE_HISTORY 541 542 /// The function to call to get the next lex token. 543 BcLexNext next; 544 545 /// The function to call to parse. 546 BcParseParse parse; 547 548 /// The function to call to parse expressions. 549 BcParseExpr expr; 550 551 /// The text to display to label functions in error messages. 552 const char* func_header; 553 554 /// The names of the categories of errors. 555 const char* err_ids[BC_ERR_IDX_NELEMS + BC_ENABLED]; 556 557 /// The messages for each error. 558 const char* err_msgs[BC_ERR_NELEMS]; 559 560 #if BC_ENABLE_NLS 561 /// The locale. 562 const char* locale; 563 #endif // BC_ENABLE_NLS 564 565 #endif // !BC_ENABLE_LIBRARY 566 567 /// The last base used to parse. 568 BcBigDig last_base; 569 570 /// The last power of last_base used to parse. 571 BcBigDig last_pow; 572 573 /// The last exponent of base that equals last_pow. 574 BcBigDig last_exp; 575 576 /// BC_BASE_POW - last_pow. 577 BcBigDig last_rem; 578 579 #if !BC_ENABLE_LIBRARY 580 581 /// A buffer of environment arguments. This is the actual value of the 582 /// environment variable. 583 char* env_args_buffer; 584 585 /// A vector for environment arguments after parsing. 586 BcVec env_args; 587 588 /// A BcNum set to constant 0. 589 BcNum zero; 590 591 #endif // !BC_ENABLE_LIBRARY 592 593 /// A BcNum set to constant 1. 594 BcNum one; 595 596 /// A BcNum holding the max number held by a BcBigDig plus 1. 597 BcNum max; 598 599 /// A BcNum holding the max number held by a BcBigDig times 2 plus 1. 600 BcNum max2; 601 602 /// The BcDig array for max. 603 BcDig max_num[BC_NUM_BIGDIG_LOG10]; 604 605 /// The BcDig array for max2. 606 BcDig max2_num[BC_NUM_BIGDIG_LOG10]; 607 608 // The BcDig array for the one BcNum. 609 BcDig one_num[BC_VM_ONE_CAP]; 610 611 #if !BC_ENABLE_LIBRARY 612 613 // The BcDig array for the zero BcNum. 614 BcDig zero_num[BC_VM_ONE_CAP]; 615 616 /// The stdout file. 617 BcFile fout; 618 619 /// The stderr file. 620 BcFile ferr; 621 622 #if BC_ENABLE_NLS 623 624 /// The locale catalog. 625 nl_catd catalog; 626 627 #endif // BC_ENABLE_NLS 628 629 /// A pointer to the stdin buffer. 630 char* buf; 631 632 /// The number of items in the input buffer. 633 size_t buf_len; 634 635 /// The slab for constants in the main function. This is separate for 636 /// garbage collection reasons. 637 BcVec main_const_slab; 638 639 //// The slab for all other strings for the main function. 640 BcVec main_slabs; 641 642 /// The slab for function names, strings in other functions, and constants 643 /// in other functions. 644 BcVec other_slabs; 645 646 #if BC_ENABLED 647 648 /// An array of booleans for which bc keywords have been redefined if 649 /// BC_REDEFINE_KEYWORDS is non-zero. 650 bool redefined_kws[BC_LEX_NKWS]; 651 652 #endif // BC_ENABLED 653 #endif // !BC_ENABLE_LIBRARY 654 655 #if BC_DEBUG_CODE 656 657 /// The depth for BC_FUNC_ENTER and BC_FUNC_EXIT. 658 size_t func_depth; 659 660 #endif // BC_DEBUG_CODE 661 662 } BcVm; 663 664 /** 665 * Print the copyright banner and help if it's non-NULL. 666 * @param help The help message to print if it's non-NULL. 667 */ 668 void 669 bc_vm_info(const char* const help); 670 671 /** 672 * The entrance point for bc/dc together. 673 * @param argc The count of arguments. 674 * @param argv The argument array. 675 */ 676 void 677 bc_vm_boot(int argc, char* argv[]); 678 679 /** 680 * Initializes some of the BcVm global. This is separate to make things easier 681 * on the library code. 682 */ 683 void 684 bc_vm_init(void); 685 686 /** 687 * Frees the BcVm global. 688 */ 689 void 690 bc_vm_shutdown(void); 691 692 /** 693 * Add a temp to the temp array. 694 * @param num The BcDig array to add to the temp array. 695 */ 696 void 697 bc_vm_addTemp(BcDig* num); 698 699 /** 700 * Dish out a temp, or NULL if there are none. 701 * @return A temp, or NULL if none exist. 702 */ 703 BcDig* 704 bc_vm_takeTemp(void); 705 706 /** 707 * Frees all temporaries. 708 */ 709 void 710 bc_vm_freeTemps(void); 711 712 #if !BC_ENABLE_HISTORY || BC_ENABLE_LINE_LIB 713 714 /** 715 * Erases the flush argument if history does not exist because it does not 716 * matter if history does not exist. 717 */ 718 #define bc_vm_putchar(c, t) bc_vm_putchar(c) 719 720 #endif // !BC_ENABLE_HISTORY || BC_ENABLE_LINE_LIB 721 722 /** 723 * Print to stdout with limited formating. 724 * @param fmt The format string. 725 */ 726 void 727 bc_vm_printf(const char* fmt, ...); 728 729 /** 730 * Puts a char into the stdout buffer. 731 * @param c The character to put on the stdout buffer. 732 * @param type The flush type. 733 */ 734 void 735 bc_vm_putchar(int c, BcFlushType type); 736 737 /** 738 * Multiplies @a n and @a size and throws an allocation error if overflow 739 * occurs. 740 * @param n The number of elements. 741 * @param size The size of each element. 742 * @return The product of @a n and @a size. 743 */ 744 size_t 745 bc_vm_arraySize(size_t n, size_t size); 746 747 /** 748 * Adds @a a and @a b and throws an error if overflow occurs. 749 * @param a The first operand. 750 * @param b The second operand. 751 * @return The sum of @a a and @a b. 752 */ 753 size_t 754 bc_vm_growSize(size_t a, size_t b); 755 756 /** 757 * Allocate @a n bytes and throw an allocation error if allocation fails. 758 * @param n The bytes to allocate. 759 * @return A pointer to the allocated memory. 760 */ 761 void* 762 bc_vm_malloc(size_t n); 763 764 /** 765 * Reallocate @a ptr to be @a n bytes and throw an allocation error if 766 * reallocation fails. 767 * @param ptr The pointer to a memory allocation to reallocate. 768 * @param n The bytes to allocate. 769 * @return A pointer to the reallocated memory. 770 */ 771 void* 772 bc_vm_realloc(void* ptr, size_t n); 773 774 /** 775 * Allocates space for, and duplicates, @a str. 776 * @param str The string to allocate. 777 * @return The allocated string. 778 */ 779 char* 780 bc_vm_strdup(const char* str); 781 782 /** 783 * Reads a line from stdin into BcVm's buffer field. 784 * @param clear True if the buffer should be cleared first, false otherwise. 785 * @return True if a line was read, false otherwise. 786 */ 787 bool 788 bc_vm_readLine(bool clear); 789 790 /** 791 * Reads a line from the command-line expressions into BcVm's buffer field. 792 * @param clear True if the buffer should be cleared first, false otherwise. 793 * @return True if a line was read, false otherwise. 794 */ 795 bool 796 bc_vm_readBuf(bool clear); 797 798 /** 799 * A convenience and portability function for OpenBSD's pledge(). 800 * @param promises The promises to pledge(). 801 * @param execpromises The exec promises to pledge(). 802 */ 803 void 804 bc_pledge(const char* promises, const char* execpromises); 805 806 /** 807 * Returns the value of an environment variable. 808 * @param var The environment variable. 809 * @return The value of the environment variable. 810 */ 811 char* 812 bc_vm_getenv(const char* var); 813 814 /** 815 * Frees an environment variable value. 816 * @param val The value to free. 817 */ 818 void 819 bc_vm_getenvFree(char* val); 820 821 #if BC_DEBUG_CODE 822 823 /** 824 * Start executing a jump series. 825 * @param f The name of the function that started the jump series. 826 */ 827 void 828 bc_vm_jmp(const char* f); 829 #else // BC_DEBUG_CODE 830 831 /** 832 * Start executing a jump series. 833 */ 834 void 835 bc_vm_jmp(void); 836 837 #endif // BC_DEBUG_CODE 838 839 #if BC_ENABLE_LIBRARY 840 841 /** 842 * Handle an error. This is the true error handler. It will start a jump series 843 * if an error occurred. POSIX errors will not cause jumps when warnings are on 844 * or no POSIX errors are enabled. 845 * @param e The error. 846 */ 847 void 848 bc_vm_handleError(BcErr e); 849 850 /** 851 * Handle a fatal error. 852 * @param e The error. 853 */ 854 void 855 bc_vm_fatalError(BcErr e); 856 857 /** 858 * A function to call at exit. 859 */ 860 void 861 bc_vm_atexit(void); 862 863 #else // BC_ENABLE_LIBRARY 864 865 /** 866 * Handle an error. This is the true error handler. It will start a jump series 867 * if an error occurred. POSIX errors will not cause jumps when warnings are on 868 * or no POSIX errors are enabled. 869 * @param e The error. 870 * @param line The source line where the error occurred. 871 */ 872 void 873 bc_vm_handleError(BcErr e, size_t line, ...); 874 875 /** 876 * Handle a fatal error. 877 * @param e The error. 878 */ 879 #if !BC_ENABLE_MEMCHECK 880 BC_NORETURN 881 #endif // !BC_ENABLE_MEMCHECK 882 void 883 bc_vm_fatalError(BcErr e); 884 885 /** 886 * A function to call at exit. 887 * @param status The exit status. 888 */ 889 int 890 bc_vm_atexit(int status); 891 #endif // BC_ENABLE_LIBRARY 892 893 /// A reference to the copyright header. 894 extern const char bc_copyright[]; 895 896 /// A reference to the format string for source code line printing. 897 extern const char* const bc_err_line; 898 899 /// A reference to the format string for source code function printing. 900 extern const char* const bc_err_func_header; 901 902 /// A reference to the array of default error category names. 903 extern const char* bc_errs[]; 904 905 /// A reference to the array of error category indices for each error. 906 extern const uchar bc_err_ids[]; 907 908 /// A reference to the array of default error messages. 909 extern const char* const bc_err_msgs[]; 910 911 /// A reference to the pledge() promises at start. 912 extern const char bc_pledge_start[]; 913 914 #if BC_ENABLE_HISTORY 915 916 /// A reference to the end pledge() promises when using history. 917 extern const char bc_pledge_end_history[]; 918 919 #endif // BC_ENABLE_HISTORY 920 921 /// A reference to the end pledge() promises when *not* using history. 922 extern const char bc_pledge_end[]; 923 924 /// A reference to the global data. 925 extern BcVm vm; 926 927 /// A reference to the global output buffers. 928 extern char output_bufs[BC_VM_BUF_SIZE]; 929 930 #endif // BC_VM_H 931