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 * All bc status codes and cross-platform portability. 33 * 34 */ 35 36 #ifndef BC_STATUS_H 37 #define BC_STATUS_H 38 39 #ifdef _WIN32 40 #include <Windows.h> 41 #include <BaseTsd.h> 42 #include <stdio.h> 43 #include <io.h> 44 #endif // _WIN32 45 46 #include <stdint.h> 47 #include <sys/types.h> 48 49 // This is used by configure.sh to test for OpenBSD. 50 #ifdef BC_TEST_OPENBSD 51 #ifdef __OpenBSD__ 52 #error On OpenBSD without _BSD_SOURCE 53 #endif // __OpenBSD__ 54 #endif // BC_TEST_OPENBSD 55 56 // This is used by configure.sh to test for FreeBSD. 57 #ifdef BC_TEST_FREEBSD 58 #ifdef __FreeBSD__ 59 #error On FreeBSD with _POSIX_C_SOURCE 60 #endif // __FreeBSD__ 61 #endif // BC_TEST_FREEBSD 62 63 // This is used by configure.sh to test for macOS. 64 #ifdef BC_TEST_APPLE 65 #ifdef __APPLE__ 66 #error On macOS without _DARWIN_C_SOURCE 67 #endif // __APPLE__ 68 #endif // BC_TEST_APPLE 69 70 // Windows has deprecated isatty() and the rest of these. Or doesn't have them. 71 // So these are just fixes for Windows. 72 #ifdef _WIN32 73 74 // This one is special. Windows did not like me defining an 75 // inline function that was not given a definition in a header 76 // file. This suppresses that by making inline functions non-inline. 77 #define inline 78 79 #define restrict __restrict 80 #define strdup _strdup 81 #define write(f, b, s) _write((f), (b), (unsigned int) (s)) 82 #define read(f, b, s) _read((f), (b), (unsigned int) (s)) 83 #define close _close 84 #define open(f, n, m) \ 85 _sopen_s((f), (n), (m) | _O_BINARY, _SH_DENYNO, _S_IREAD | _S_IWRITE) 86 #define sigjmp_buf jmp_buf 87 #define sigsetjmp(j, s) setjmp(j) 88 #define siglongjmp longjmp 89 #define isatty _isatty 90 #define STDIN_FILENO _fileno(stdin) 91 #define STDOUT_FILENO _fileno(stdout) 92 #define STDERR_FILENO _fileno(stderr) 93 #define S_ISDIR(m) ((m) & (_S_IFDIR)) 94 #define O_RDONLY _O_RDONLY 95 #define stat _stat 96 #define fstat _fstat 97 #define BC_FILE_SEP '\\' 98 99 #else // _WIN32 100 #define BC_FILE_SEP '/' 101 #endif // _WIN32 102 103 #ifndef BC_ENABLED 104 #define BC_ENABLED (1) 105 #endif // BC_ENABLED 106 107 #ifndef DC_ENABLED 108 #define DC_ENABLED (1) 109 #endif // DC_ENABLED 110 111 #ifndef BC_ENABLE_EXTRA_MATH 112 #define BC_ENABLE_EXTRA_MATH (1) 113 #endif // BC_ENABLE_EXTRA_MATH 114 115 #ifndef BC_ENABLE_LIBRARY 116 #define BC_ENABLE_LIBRARY (0) 117 #endif // BC_ENABLE_LIBRARY 118 119 #ifndef BC_ENABLE_HISTORY 120 #define BC_ENABLE_HISTORY (1) 121 #endif // BC_ENABLE_HISTORY 122 123 #ifndef BC_ENABLE_EDITLINE 124 #define BC_ENABLE_EDITLINE (0) 125 #endif // BC_ENABLE_EDITLINE 126 127 #ifndef BC_ENABLE_READLINE 128 #define BC_ENABLE_READLINE (0) 129 #endif // BC_ENABLE_READLINE 130 131 #ifndef BC_ENABLE_NLS 132 #define BC_ENABLE_NLS (0) 133 #endif // BC_ENABLE_NLS 134 135 #ifdef __OpenBSD__ 136 #if BC_ENABLE_READLINE 137 #error Cannot use readline on OpenBSD 138 #endif // BC_ENABLE_READLINE 139 #endif // __OpenBSD__ 140 141 #if BC_ENABLE_EDITLINE && BC_ENABLE_READLINE 142 #error Must enable only one of editline or readline, not both. 143 #endif // BC_ENABLE_EDITLINE && BC_ENABLE_READLINE 144 145 #if BC_ENABLE_EDITLINE || BC_ENABLE_READLINE 146 #define BC_ENABLE_LINE_LIB (1) 147 #else // BC_ENABLE_EDITLINE || BC_ENABLE_READLINE 148 #define BC_ENABLE_LINE_LIB (0) 149 #endif // BC_ENABLE_EDITLINE || BC_ENABLE_READLINE 150 151 // This is error checking for fuzz builds. 152 #if BC_ENABLE_AFL 153 #ifndef __AFL_HAVE_MANUAL_CONTROL 154 #error Must compile with afl-clang-fast or afl-clang-lto for fuzzing 155 #endif // __AFL_HAVE_MANUAL_CONTROL 156 #endif // BC_ENABLE_AFL 157 158 #ifndef BC_ENABLE_MEMCHECK 159 #define BC_ENABLE_MEMCHECK (0) 160 #endif // BC_ENABLE_MEMCHECK 161 162 /** 163 * Mark a variable as unused. 164 * @param e The variable to mark as unused. 165 */ 166 #define BC_UNUSED(e) ((void) (e)) 167 168 // If users want, they can define this to something like __builtin_expect(e, 1). 169 // It might give a performance improvement. 170 #ifndef BC_LIKELY 171 172 /** 173 * Mark a branch expression as likely. 174 * @param e The expression to mark as likely. 175 */ 176 #define BC_LIKELY(e) (e) 177 178 #endif // BC_LIKELY 179 180 // If users want, they can define this to something like __builtin_expect(e, 0). 181 // It might give a performance improvement. 182 #ifndef BC_UNLIKELY 183 184 /** 185 * Mark a branch expression as unlikely. 186 * @param e The expression to mark as unlikely. 187 */ 188 #define BC_UNLIKELY(e) (e) 189 190 #endif // BC_UNLIKELY 191 192 /** 193 * Mark a branch expression as an error, if true. 194 * @param e The expression to mark as an error, if true. 195 */ 196 #define BC_ERR(e) BC_UNLIKELY(e) 197 198 /** 199 * Mark a branch expression as not an error, if true. 200 * @param e The expression to mark as not an error, if true. 201 */ 202 #define BC_NO_ERR(s) BC_LIKELY(s) 203 204 // Disable extra debug code by default. 205 #ifndef BC_DEBUG_CODE 206 #define BC_DEBUG_CODE (0) 207 #endif // BC_DEBUG_CODE 208 209 #if defined(__clang__) 210 #define BC_CLANG (1) 211 #else // defined(__clang__) 212 #define BC_CLANG (0) 213 #endif // defined(__clang__) 214 215 #if defined(__GNUC__) && !BC_CLANG 216 #define BC_GCC (1) 217 #else // defined(__GNUC__) && !BC_CLANG 218 #define BC_GCC (0) 219 #endif // defined(__GNUC__) && !BC_CLANG 220 221 // We want to be able to use _Noreturn on C11 compilers. 222 #if __STDC_VERSION__ >= 201112L 223 224 #include <stdnoreturn.h> 225 #define BC_NORETURN _Noreturn 226 #define BC_C11 (1) 227 228 #else // __STDC_VERSION__ 229 230 #if BC_CLANG 231 #if __has_attribute(noreturn) 232 #define BC_NORETURN __attribute((noreturn)) 233 #else // __has_attribute(noreturn) 234 #define BC_NORETURN 235 #endif // __has_attribute(noreturn) 236 237 #else // BC_CLANG 238 239 #define BC_NORETURN 240 241 #endif // BC_CLANG 242 243 #define BC_MUST_RETURN 244 #define BC_C11 (0) 245 246 #endif // __STDC_VERSION__ 247 248 #define BC_HAS_UNREACHABLE (0) 249 #define BC_HAS_COMPUTED_GOTO (0) 250 251 // GCC and Clang complain if fallthroughs are not marked with their special 252 // attribute. Jerks. This creates a define for marking the fallthroughs that is 253 // nothing on other compilers. 254 #if BC_CLANG || BC_GCC 255 256 #if defined(__has_attribute) 257 258 #if __has_attribute(fallthrough) 259 #define BC_FALLTHROUGH __attribute__((fallthrough)); 260 #else // __has_attribute(fallthrough) 261 #define BC_FALLTHROUGH 262 #endif // __has_attribute(fallthrough) 263 264 #if BC_GCC 265 266 #if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 5) 267 #undef BC_HAS_UNREACHABLE 268 #define BC_HAS_UNREACHABLE (1) 269 #endif // __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 5) 270 271 #else // BC_GCC 272 273 #if __clang_major__ >= 4 274 #undef BC_HAS_UNREACHABLE 275 #define BC_HAS_UNREACHABLE (1) 276 #endif // __clang_major__ >= 4 277 278 #endif // BC_GCC 279 280 #else // defined(__has_attribute) 281 #define BC_FALLTHROUGH 282 #endif // defined(__has_attribute) 283 #else // BC_CLANG || BC_GCC 284 #define BC_FALLTHROUGH 285 #endif // BC_CLANG || BC_GCC 286 287 #if BC_HAS_UNREACHABLE 288 289 #define BC_UNREACHABLE __builtin_unreachable(); 290 291 #else // BC_HAS_UNREACHABLE 292 293 #ifdef _WIN32 294 295 #define BC_UNREACHABLE __assume(0); 296 297 #else // _WIN32 298 299 #define BC_UNREACHABLE 300 301 #endif // _WIN32 302 303 #endif // BC_HAS_UNREACHABLE 304 305 #if BC_GCC 306 307 #if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 5) 308 309 #undef BC_HAS_COMPUTED_GOTO 310 #define BC_HAS_COMPUTED_GOTO (1) 311 312 #endif // __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 5) 313 314 #endif // BC_GCC 315 316 #if BC_CLANG 317 318 #if __clang_major__ >= 4 319 320 #undef BC_HAS_COMPUTED_GOTO 321 #define BC_HAS_COMPUTED_GOTO (1) 322 323 #endif // __clang_major__ >= 4 324 325 #endif // BC_CLANG 326 327 #ifdef BC_NO_COMPUTED_GOTO 328 329 #undef BC_HAS_COMPUTED_GOTO 330 #define BC_HAS_COMPUTED_GOTO (0) 331 332 #endif // BC_NO_COMPUTED_GOTO 333 334 #if BC_GCC 335 #ifdef __OpenBSD__ 336 // The OpenBSD GCC doesn't like inline. 337 #define inline 338 #endif // __OpenBSD__ 339 #endif // BC_GCC 340 341 // Workarounds for AIX's POSIX incompatibility. 342 #ifndef SIZE_MAX 343 #define SIZE_MAX __SIZE_MAX__ 344 #endif // SIZE_MAX 345 #ifndef UINTMAX_C 346 #define UINTMAX_C __UINTMAX_C 347 #endif // UINTMAX_C 348 #ifndef UINT32_C 349 #define UINT32_C __UINT32_C 350 #endif // UINT32_C 351 #ifndef UINT_FAST32_MAX 352 #define UINT_FAST32_MAX __UINT_FAST32_MAX__ 353 #endif // UINT_FAST32_MAX 354 #ifndef UINT16_MAX 355 #define UINT16_MAX __UINT16_MAX__ 356 #endif // UINT16_MAX 357 #ifndef SIG_ATOMIC_MAX 358 #define SIG_ATOMIC_MAX __SIG_ATOMIC_MAX__ 359 #endif // SIG_ATOMIC_MAX 360 361 // Yes, this has to be here. 362 #include <bcl.h> 363 364 // All of these set defaults for settings. 365 366 #if BC_ENABLED 367 368 #ifndef BC_DEFAULT_BANNER 369 #define BC_DEFAULT_BANNER (0) 370 #endif // BC_DEFAULT_BANNER 371 372 #endif // BC_ENABLED 373 374 #ifndef BC_DEFAULT_SIGINT_RESET 375 #define BC_DEFAULT_SIGINT_RESET (1) 376 #endif // BC_DEFAULT_SIGINT_RESET 377 378 #ifndef BC_DEFAULT_TTY_MODE 379 #define BC_DEFAULT_TTY_MODE (1) 380 #endif // BC_DEFAULT_TTY_MODE 381 382 #ifndef BC_DEFAULT_PROMPT 383 #define BC_DEFAULT_PROMPT BC_DEFAULT_TTY_MODE 384 #endif // BC_DEFAULT_PROMPT 385 386 #ifndef BC_DEFAULT_EXPR_EXIT 387 #define BC_DEFAULT_EXPR_EXIT (1) 388 #endif // BC_DEFAULT_EXPR_EXIT 389 390 #ifndef BC_DEFAULT_DIGIT_CLAMP 391 #define BC_DEFAULT_DIGIT_CLAMP (0) 392 #endif // BC_DEFAULT_DIGIT_CLAMP 393 394 // All of these set defaults for settings. 395 #ifndef DC_DEFAULT_SIGINT_RESET 396 #define DC_DEFAULT_SIGINT_RESET (1) 397 #endif // DC_DEFAULT_SIGINT_RESET 398 399 #ifndef DC_DEFAULT_TTY_MODE 400 #define DC_DEFAULT_TTY_MODE (0) 401 #endif // DC_DEFAULT_TTY_MODE 402 403 #ifndef DC_DEFAULT_HISTORY 404 #define DC_DEFAULT_HISTORY DC_DEFAULT_TTY_MODE 405 #endif // DC_DEFAULT_HISTORY 406 407 #ifndef DC_DEFAULT_PROMPT 408 #define DC_DEFAULT_PROMPT DC_DEFAULT_TTY_MODE 409 #endif // DC_DEFAULT_PROMPT 410 411 #ifndef DC_DEFAULT_EXPR_EXIT 412 #define DC_DEFAULT_EXPR_EXIT (1) 413 #endif // DC_DEFAULT_EXPR_EXIT 414 415 #ifndef DC_DEFAULT_DIGIT_CLAMP 416 #define DC_DEFAULT_DIGIT_CLAMP (0) 417 #endif // DC_DEFAULT_DIGIT_CLAMP 418 419 /// Statuses, which mark either which category of error happened, or some other 420 /// status that matters. 421 typedef enum BcStatus 422 { 423 /// Normal status. 424 BC_STATUS_SUCCESS = 0, 425 426 /// Math error. 427 BC_STATUS_ERROR_MATH, 428 429 /// Parse (and lex) error. 430 BC_STATUS_ERROR_PARSE, 431 432 /// Runtime error. 433 BC_STATUS_ERROR_EXEC, 434 435 /// Fatal error. 436 BC_STATUS_ERROR_FATAL, 437 438 /// EOF status. 439 BC_STATUS_EOF, 440 441 /// Quit status. This means that bc/dc is in the process of quitting. 442 BC_STATUS_QUIT, 443 444 } BcStatus; 445 446 /// Errors, which are more specific errors. 447 typedef enum BcErr 448 { 449 // Math errors. 450 451 /// Negative number used when not allowed. 452 BC_ERR_MATH_NEGATIVE, 453 454 /// Non-integer used when not allowed. 455 BC_ERR_MATH_NON_INTEGER, 456 457 /// Conversion to a hardware integer would overflow. 458 BC_ERR_MATH_OVERFLOW, 459 460 /// Divide by zero. 461 BC_ERR_MATH_DIVIDE_BY_ZERO, 462 463 // Fatal errors. 464 465 /// An allocation or reallocation failed. 466 BC_ERR_FATAL_ALLOC_ERR, 467 468 /// I/O failure. 469 BC_ERR_FATAL_IO_ERR, 470 471 /// File error, such as permissions or file does not exist. 472 BC_ERR_FATAL_FILE_ERR, 473 474 /// File is binary, not text, error. 475 BC_ERR_FATAL_BIN_FILE, 476 477 /// Attempted to read a directory as a file error. 478 BC_ERR_FATAL_PATH_DIR, 479 480 /// Invalid option error. 481 BC_ERR_FATAL_OPTION, 482 483 /// Option with required argument not given an argument. 484 BC_ERR_FATAL_OPTION_NO_ARG, 485 486 /// Option with no argument given an argument. 487 BC_ERR_FATAL_OPTION_ARG, 488 489 /// Option argument is invalid. 490 BC_ERR_FATAL_ARG, 491 492 // Runtime errors. 493 494 /// Invalid ibase value. 495 BC_ERR_EXEC_IBASE, 496 497 /// Invalid obase value. 498 BC_ERR_EXEC_OBASE, 499 500 /// Invalid scale value. 501 BC_ERR_EXEC_SCALE, 502 503 /// Invalid expression parsed by read(). 504 BC_ERR_EXEC_READ_EXPR, 505 506 /// read() used within an expression given to a read() call. 507 BC_ERR_EXEC_REC_READ, 508 509 /// Type error. 510 BC_ERR_EXEC_TYPE, 511 512 /// Stack has too few elements error. 513 BC_ERR_EXEC_STACK, 514 515 /// Register stack has too few elements error. 516 BC_ERR_EXEC_STACK_REGISTER, 517 518 /// Wrong number of arguments error. 519 BC_ERR_EXEC_PARAMS, 520 521 /// Undefined function error. 522 BC_ERR_EXEC_UNDEF_FUNC, 523 524 /// Void value used in an expression error. 525 BC_ERR_EXEC_VOID_VAL, 526 527 // Parse (and lex) errors. 528 529 /// EOF encountered when not expected error. 530 BC_ERR_PARSE_EOF, 531 532 /// Invalid character error. 533 BC_ERR_PARSE_CHAR, 534 535 /// Invalid string (no ending quote) error. 536 BC_ERR_PARSE_STRING, 537 538 /// Invalid comment (no end found) error. 539 BC_ERR_PARSE_COMMENT, 540 541 /// Invalid token encountered error. 542 BC_ERR_PARSE_TOKEN, 543 544 #if BC_ENABLED 545 546 /// Invalid expression error. 547 BC_ERR_PARSE_EXPR, 548 549 /// Expression is empty error. 550 BC_ERR_PARSE_EMPTY_EXPR, 551 552 /// Print statement is invalid error. 553 BC_ERR_PARSE_PRINT, 554 555 /// Function definition is invalid error. 556 BC_ERR_PARSE_FUNC, 557 558 /// Assignment is invalid error. 559 BC_ERR_PARSE_ASSIGN, 560 561 /// No auto identifiers given for an auto statement error. 562 BC_ERR_PARSE_NO_AUTO, 563 564 /// Duplicate local (parameter or auto) error. 565 BC_ERR_PARSE_DUP_LOCAL, 566 567 /// Invalid block (within braces) error. 568 BC_ERR_PARSE_BLOCK, 569 570 /// Invalid return statement for void functions. 571 BC_ERR_PARSE_RET_VOID, 572 573 /// Reference attached to a variable, not an array, error. 574 BC_ERR_PARSE_REF_VAR, 575 576 // POSIX-only errors. 577 578 /// Name length greater than 1 error. 579 BC_ERR_POSIX_NAME_LEN, 580 581 /// Non-POSIX comment used error. 582 BC_ERR_POSIX_COMMENT, 583 584 /// Non-POSIX keyword error. 585 BC_ERR_POSIX_KW, 586 587 /// Non-POSIX . (last) error. 588 BC_ERR_POSIX_DOT, 589 590 /// Non-POSIX return error. 591 BC_ERR_POSIX_RET, 592 593 /// Non-POSIX boolean operator used error. 594 BC_ERR_POSIX_BOOL, 595 596 /// POSIX relation operator used outside if, while, or for statements error. 597 BC_ERR_POSIX_REL_POS, 598 599 /// Multiple POSIX relation operators used in an if, while, or for statement 600 /// error. 601 BC_ERR_POSIX_MULTIREL, 602 603 /// Empty statements in POSIX for loop error. 604 BC_ERR_POSIX_FOR, 605 606 /// POSIX's grammar does not allow a function definition right after a 607 /// semicolon. 608 BC_ERR_POSIX_FUNC_AFTER_SEMICOLON, 609 610 /// Non-POSIX exponential (scientific or engineering) number used error. 611 BC_ERR_POSIX_EXP_NUM, 612 613 /// Non-POSIX array reference error. 614 BC_ERR_POSIX_REF, 615 616 /// Non-POSIX void error. 617 BC_ERR_POSIX_VOID, 618 619 /// Non-POSIX brace position used error. 620 BC_ERR_POSIX_BRACE, 621 622 /// String used in expression. 623 BC_ERR_POSIX_EXPR_STRING, 624 625 #endif // BC_ENABLED 626 627 // Number of elements. 628 BC_ERR_NELEMS, 629 630 #if BC_ENABLED 631 632 /// A marker for the start of POSIX errors. 633 BC_ERR_POSIX_START = BC_ERR_POSIX_NAME_LEN, 634 635 /// A marker for the end of POSIX errors. 636 BC_ERR_POSIX_END = BC_ERR_POSIX_EXPR_STRING, 637 638 #endif // BC_ENABLED 639 640 } BcErr; 641 642 // The indices of each category of error in bc_errs[], and used in bc_err_ids[] 643 // to associate actual errors with their categories. 644 645 /// Math error category. 646 #define BC_ERR_IDX_MATH (0) 647 648 /// Parse (and lex) error category. 649 #define BC_ERR_IDX_PARSE (1) 650 651 /// Runtime error category. 652 #define BC_ERR_IDX_EXEC (2) 653 654 /// Fatal error category. 655 #define BC_ERR_IDX_FATAL (3) 656 657 /// Number of categories. 658 #define BC_ERR_IDX_NELEMS (4) 659 660 // If bc is enabled, we add an extra category for POSIX warnings. 661 #if BC_ENABLED 662 663 /// POSIX warning category. 664 #define BC_ERR_IDX_WARN (BC_ERR_IDX_NELEMS) 665 666 #endif // BC_ENABLED 667 668 /** 669 * The mode bc is in. This is basically what input it is processing. 670 */ 671 typedef enum BcMode 672 { 673 /// Expressions mode. 674 BC_MODE_EXPRS, 675 676 /// File mode. 677 BC_MODE_FILE, 678 679 /// stdin mode. 680 BC_MODE_STDIN, 681 682 } BcMode; 683 684 /// Do a longjmp(). This is what to use when activating an "exception", i.e., a 685 /// longjmp(). With debug code, it will print the name of the function it jumped 686 /// from. 687 #if BC_DEBUG_CODE 688 #define BC_JMP bc_vm_jmp(__func__) 689 #else // BC_DEBUG_CODE 690 #define BC_JMP bc_vm_jmp() 691 #endif // BC_DEBUG_CODE 692 693 #if !BC_ENABLE_LIBRARY 694 695 /// Returns true if an exception is in flight, false otherwise. 696 #define BC_SIG_EXC(vm) \ 697 BC_UNLIKELY((vm)->status != (sig_atomic_t) BC_STATUS_SUCCESS || (vm)->sig) 698 699 /// Returns true if there is *no* exception in flight, false otherwise. 700 #define BC_NO_SIG_EXC(vm) \ 701 BC_LIKELY((vm)->status == (sig_atomic_t) BC_STATUS_SUCCESS && !(vm)->sig) 702 703 #ifndef _WIN32 704 #define BC_SIG_INTERRUPT(vm) \ 705 BC_UNLIKELY((vm)->sig != 0 && (vm)->sig != SIGWINCH) 706 #else // _WIN32 707 #define BC_SIG_INTERRUPT(vm) BC_UNLIKELY((vm)->sig != 0) 708 #endif // _WIN32 709 710 #if BC_DEBUG 711 712 /// Assert that signals are locked. There are non-async-signal-safe functions in 713 /// bc, and they *must* have signals locked. Other functions are expected to 714 /// *not* have signals locked, for reasons. So this is a pre-built assert 715 /// (no-op in non-debug mode) that check that signals are locked. 716 #define BC_SIG_ASSERT_LOCKED \ 717 do \ 718 { \ 719 assert(vm->sig_lock); \ 720 } \ 721 while (0) 722 723 /// Assert that signals are unlocked. There are non-async-signal-safe functions 724 /// in bc, and they *must* have signals locked. Other functions are expected to 725 /// *not* have signals locked, for reasons. So this is a pre-built assert 726 /// (no-op in non-debug mode) that check that signals are unlocked. 727 #define BC_SIG_ASSERT_NOT_LOCKED \ 728 do \ 729 { \ 730 assert(vm->sig_lock == 0); \ 731 } \ 732 while (0) 733 734 #else // BC_DEBUG 735 736 /// Assert that signals are locked. There are non-async-signal-safe functions in 737 /// bc, and they *must* have signals locked. Other functions are expected to 738 /// *not* have signals locked, for reasons. So this is a pre-built assert 739 /// (no-op in non-debug mode) that check that signals are locked. 740 #define BC_SIG_ASSERT_LOCKED 741 742 /// Assert that signals are unlocked. There are non-async-signal-safe functions 743 /// in bc, and they *must* have signals locked. Other functions are expected to 744 /// *not* have signals locked, for reasons. So this is a pre-built assert 745 /// (no-op in non-debug mode) that check that signals are unlocked. 746 #define BC_SIG_ASSERT_NOT_LOCKED 747 748 #endif // BC_DEBUG 749 750 /// Locks signals. 751 #define BC_SIG_LOCK \ 752 do \ 753 { \ 754 BC_SIG_ASSERT_NOT_LOCKED; \ 755 vm->sig_lock = 1; \ 756 } \ 757 while (0) 758 759 /// Unlocks signals. If a signal happened, then this will cause a jump. 760 #define BC_SIG_UNLOCK \ 761 do \ 762 { \ 763 BC_SIG_ASSERT_LOCKED; \ 764 vm->sig_lock = 0; \ 765 if (vm->sig) BC_JMP; \ 766 } \ 767 while (0) 768 769 /// Locks signals, regardless of if they are already locked. This is really only 770 /// used after labels that longjmp() goes to after the jump because the cleanup 771 /// code must have signals locked, and BC_LONGJMP_CONT will unlock signals if it 772 /// doesn't jump. 773 #define BC_SIG_MAYLOCK \ 774 do \ 775 { \ 776 vm->sig_lock = 1; \ 777 } \ 778 while (0) 779 780 /// Unlocks signals, regardless of if they were already unlocked. If a signal 781 /// happened, then this will cause a jump. 782 #define BC_SIG_MAYUNLOCK \ 783 do \ 784 { \ 785 vm->sig_lock = 0; \ 786 if (vm->sig) BC_JMP; \ 787 } \ 788 while (0) 789 790 /** 791 * Locks signals, but stores the old lock state, to be restored later by 792 * BC_SIG_TRYUNLOCK. 793 * @param v The variable to store the old lock state to. 794 */ 795 #define BC_SIG_TRYLOCK(v) \ 796 do \ 797 { \ 798 v = vm->sig_lock; \ 799 vm->sig_lock = 1; \ 800 } \ 801 while (0) 802 803 /** 804 * Restores the previous state of a signal lock, and if it is now unlocked, 805 * initiates an exception/jump. 806 * @param v The old lock state. 807 */ 808 #define BC_SIG_TRYUNLOCK(v) \ 809 do \ 810 { \ 811 vm->sig_lock = (v); \ 812 if (!(v) && vm->sig) BC_JMP; \ 813 } \ 814 while (0) 815 816 /// Stops a stack unwinding. Technically, a stack unwinding needs to be done 817 /// manually, but it will always be done unless certain flags are cleared. This 818 /// clears the flags. 819 #define BC_LONGJMP_STOP \ 820 do \ 821 { \ 822 vm->sig_pop = 0; \ 823 vm->sig = 0; \ 824 } \ 825 while (0) 826 827 /** 828 * Sets a jump like BC_SETJMP, but unlike BC_SETJMP, it assumes signals are 829 * locked and will just set the jump. This does *not* have a call to 830 * bc_vec_grow() because it is assumed that BC_SETJMP_LOCKED(l) is used *after* 831 * the initializations that need the setjmp(). 832 * param l The label to jump to on a longjmp(). 833 */ 834 #define BC_SETJMP_LOCKED(vm, l) \ 835 do \ 836 { \ 837 sigjmp_buf sjb; \ 838 BC_SIG_ASSERT_LOCKED; \ 839 if (sigsetjmp(sjb, 0)) \ 840 { \ 841 assert(BC_SIG_EXC(vm)); \ 842 goto l; \ 843 } \ 844 bc_vec_push(&vm->jmp_bufs, &sjb); \ 845 } \ 846 while (0) 847 848 /// Used after cleanup labels set by BC_SETJMP and BC_SETJMP_LOCKED to jump to 849 /// the next place. This is what continues the stack unwinding. This basically 850 /// copies BC_SIG_UNLOCK into itself, but that is because its condition for 851 /// jumping is BC_SIG_EXC, not just that a signal happened. 852 #define BC_LONGJMP_CONT(vm) \ 853 do \ 854 { \ 855 BC_SIG_ASSERT_LOCKED; \ 856 if (!vm->sig_pop) bc_vec_pop(&vm->jmp_bufs); \ 857 vm->sig_lock = 0; \ 858 if (BC_SIG_EXC(vm)) BC_JMP; \ 859 } \ 860 while (0) 861 862 #else // !BC_ENABLE_LIBRARY 863 864 #define BC_SIG_LOCK 865 #define BC_SIG_UNLOCK 866 #define BC_SIG_MAYLOCK 867 #define BC_SIG_TRYLOCK(lock) 868 #define BC_SIG_TRYUNLOCK(lock) 869 #define BC_SIG_ASSERT_LOCKED 870 871 /// Returns true if an exception is in flight, false otherwise. 872 #define BC_SIG_EXC(vm) \ 873 BC_UNLIKELY(vm->status != (sig_atomic_t) BC_STATUS_SUCCESS) 874 875 /// Returns true if there is *no* exception in flight, false otherwise. 876 #define BC_NO_SIG_EXC(vm) \ 877 BC_LIKELY(vm->status == (sig_atomic_t) BC_STATUS_SUCCESS) 878 879 /// Used after cleanup labels set by BC_SETJMP and BC_SETJMP_LOCKED to jump to 880 /// the next place. This is what continues the stack unwinding. This basically 881 /// copies BC_SIG_UNLOCK into itself, but that is because its condition for 882 /// jumping is BC_SIG_EXC, not just that a signal happened. 883 #define BC_LONGJMP_CONT(vm) \ 884 do \ 885 { \ 886 bc_vec_pop(&vm->jmp_bufs); \ 887 if (BC_SIG_EXC(vm)) BC_JMP; \ 888 } \ 889 while (0) 890 891 #endif // !BC_ENABLE_LIBRARY 892 893 /** 894 * Sets a jump, and sets it up as well so that if a longjmp() happens, bc will 895 * immediately goto a label where some cleanup code is. This one assumes that 896 * signals are not locked and will lock them, set the jump, and unlock them. 897 * Setting the jump also includes pushing the jmp_buf onto the jmp_buf stack. 898 * This grows the jmp_bufs vector first to prevent a fatal error from happening 899 * after the setjmp(). This is done because BC_SETJMP(l) is assumed to be used 900 * *before* the actual initialization calls that need the setjmp(). 901 * param l The label to jump to on a longjmp(). 902 */ 903 #define BC_SETJMP(vm, l) \ 904 do \ 905 { \ 906 sigjmp_buf sjb; \ 907 BC_SIG_LOCK; \ 908 bc_vec_grow(&vm->jmp_bufs, 1); \ 909 if (sigsetjmp(sjb, 0)) \ 910 { \ 911 assert(BC_SIG_EXC(vm)); \ 912 goto l; \ 913 } \ 914 bc_vec_push(&vm->jmp_bufs, &sjb); \ 915 BC_SIG_UNLOCK; \ 916 } \ 917 while (0) 918 919 /// Unsets a jump. It always assumes signals are locked. This basically just 920 /// pops a jmp_buf off of the stack of jmp_bufs, and since the jump mechanism 921 /// always jumps to the location at the top of the stack, this effectively 922 /// undoes a setjmp(). 923 #define BC_UNSETJMP(vm) \ 924 do \ 925 { \ 926 BC_SIG_ASSERT_LOCKED; \ 927 bc_vec_pop(&vm->jmp_bufs); \ 928 } \ 929 while (0) 930 931 #if BC_ENABLE_LIBRARY 932 933 #define BC_SETJMP_LOCKED(vm, l) BC_SETJMP(vm, l) 934 935 // Various convenience macros for calling the bc's error handling routine. 936 937 /** 938 * Call bc's error handling routine. 939 * @param e The error. 940 * @param l The line of the script that the error happened. 941 * @param ... Extra arguments for error messages as necessary. 942 */ 943 #define bc_error(e, l, ...) (bc_vm_handleError((e))) 944 945 /** 946 * Call bc's error handling routine. 947 * @param e The error. 948 */ 949 #define bc_err(e) (bc_vm_handleError((e))) 950 951 /** 952 * Call bc's error handling routine. 953 * @param e The error. 954 */ 955 #define bc_verr(e, ...) (bc_vm_handleError((e))) 956 957 #else // BC_ENABLE_LIBRARY 958 959 // Various convenience macros for calling the bc's error handling routine. 960 961 /** 962 * Call bc's error handling routine. 963 * @param e The error. 964 * @param l The line of the script that the error happened. 965 * @param ... Extra arguments for error messages as necessary. 966 */ 967 #if BC_DEBUG 968 #define bc_error(e, l, ...) \ 969 (bc_vm_handleError((e), __FILE__, __LINE__, (l), __VA_ARGS__)) 970 #else // BC_DEBUG 971 #define bc_error(e, l, ...) (bc_vm_handleError((e), (l), __VA_ARGS__)) 972 #endif // BC_DEBUG 973 974 /** 975 * Call bc's error handling routine. 976 * @param e The error. 977 */ 978 #if BC_DEBUG 979 #define bc_err(e) (bc_vm_handleError((e), __FILE__, __LINE__, 0)) 980 #else // BC_DEBUG 981 #define bc_err(e) (bc_vm_handleError((e), 0)) 982 #endif // BC_DEBUG 983 984 /** 985 * Call bc's error handling routine. 986 * @param e The error. 987 */ 988 #if BC_DEBUG 989 #define bc_verr(e, ...) \ 990 (bc_vm_handleError((e), __FILE__, __LINE__, 0, __VA_ARGS__)) 991 #else // BC_DEBUG 992 #define bc_verr(e, ...) (bc_vm_handleError((e), 0, __VA_ARGS__)) 993 #endif // BC_DEBUG 994 995 #endif // BC_ENABLE_LIBRARY 996 997 /** 998 * Returns true if status @a s is an error, false otherwise. 999 * @param s The status to test. 1000 * @return True if @a s is an error, false otherwise. 1001 */ 1002 #define BC_STATUS_IS_ERROR(s) \ 1003 ((s) >= BC_STATUS_ERROR_MATH && (s) <= BC_STATUS_ERROR_FATAL) 1004 1005 // Convenience macros that can be placed at the beginning and exits of functions 1006 // for easy marking of where functions are entered and exited. 1007 #if BC_DEBUG_CODE 1008 #define BC_FUNC_ENTER \ 1009 do \ 1010 { \ 1011 size_t bc_func_enter_i; \ 1012 for (bc_func_enter_i = 0; bc_func_enter_i < vm->func_depth; \ 1013 ++bc_func_enter_i) \ 1014 { \ 1015 bc_file_puts(&vm->ferr, bc_flush_none, " "); \ 1016 } \ 1017 vm->func_depth += 1; \ 1018 bc_file_printf(&vm->ferr, "Entering %s\n", __func__); \ 1019 bc_file_flush(&vm->ferr, bc_flush_none); \ 1020 } \ 1021 while (0); 1022 1023 #define BC_FUNC_EXIT \ 1024 do \ 1025 { \ 1026 size_t bc_func_enter_i; \ 1027 vm->func_depth -= 1; \ 1028 for (bc_func_enter_i = 0; bc_func_enter_i < vm->func_depth; \ 1029 ++bc_func_enter_i) \ 1030 { \ 1031 bc_file_puts(&vm->ferr, bc_flush_none, " "); \ 1032 } \ 1033 bc_file_printf(&vm->ferr, "Leaving %s\n", __func__); \ 1034 bc_file_flush(&vm->ferr, bc_flush_none); \ 1035 } \ 1036 while (0); 1037 #else // BC_DEBUG_CODE 1038 #define BC_FUNC_ENTER 1039 #define BC_FUNC_EXIT 1040 #endif // BC_DEBUG_CODE 1041 1042 #endif // BC_STATUS_H 1043