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 programs. 33 * 34 */ 35 36 #ifndef BC_PROGRAM_H 37 #define BC_PROGRAM_H 38 39 #include <assert.h> 40 #include <stddef.h> 41 42 #include <status.h> 43 #include <parse.h> 44 #include <lang.h> 45 #include <num.h> 46 #include <rand.h> 47 48 /// The index of ibase in the globals array. 49 #define BC_PROG_GLOBALS_IBASE (0) 50 51 /// The index of obase in the globals array. 52 #define BC_PROG_GLOBALS_OBASE (1) 53 54 /// The index of scale in the globals array. 55 #define BC_PROG_GLOBALS_SCALE (2) 56 57 #if BC_ENABLE_EXTRA_MATH 58 59 /// The index of the rand max in the maxes array. 60 #define BC_PROG_MAX_RAND (3) 61 62 #endif // BC_ENABLE_EXTRA_MATH 63 64 /// The length of the globals array. 65 #define BC_PROG_GLOBALS_LEN (3 + BC_ENABLE_EXTRA_MATH) 66 67 typedef struct BcProgram { 68 69 /// The array of globals values. 70 BcBigDig globals[BC_PROG_GLOBALS_LEN]; 71 72 /// The array of globals stacks. 73 BcVec globals_v[BC_PROG_GLOBALS_LEN]; 74 75 #if BC_ENABLE_EXTRA_MATH 76 77 /// The pseudo-random number generator. 78 BcRNG rng; 79 80 #endif // BC_ENABLE_EXTRA_MATH 81 82 /// The results stack. 83 BcVec results; 84 85 /// The execution stack. 86 BcVec stack; 87 88 /// A pointer to the current function's constants. 89 BcVec *consts; 90 91 /// A pointer to the current function's strings. 92 BcVec *strs; 93 94 /// The array of functions. 95 BcVec fns; 96 97 /// The map of functions to go with fns. 98 BcVec fn_map; 99 100 /// The array of variables. 101 BcVec vars; 102 103 /// The map of variables to go with vars. 104 BcVec var_map; 105 106 /// The array of arrays. 107 BcVec arrs; 108 109 /// The map of arrays to go with arrs. 110 BcVec arr_map; 111 112 #if DC_ENABLED 113 114 /// A vector of tail calls. These are just integers, which are the number of 115 /// tail calls that have been executed for each function (string) on the 116 /// stack for dc. This is to prevent dc from constantly growing memory use 117 /// because of pushing more and more string executions on the stack. 118 BcVec tail_calls; 119 120 #endif // DC_ENABLED 121 122 /// A BcNum that has the proper base for asciify. 123 BcNum strmb; 124 125 #if BC_ENABLED 126 127 /// The last printed value for bc. 128 BcNum last; 129 130 #endif // BC_ENABLED 131 132 // The BcDig array for strmb. This uses BC_NUM_LONG_LOG10 because it is used 133 // in bc_num_ulong2num(), which attempts to realloc, unless it is big 134 // enough. This is big enough. 135 BcDig strmb_num[BC_NUM_BIGDIG_LOG10]; 136 137 } BcProgram; 138 139 /** 140 * Returns true if the stack @a s has at least @a n items, false otherwise. 141 * @param s The stack to check. 142 * @param n The number of items the stack must have. 143 * @return True if @a s has at least @a n items, false otherwise. 144 */ 145 #define BC_PROG_STACK(s, n) ((s)->len >= ((size_t) (n))) 146 147 /** 148 * Get a pointer to the top value in a global value stack. 149 * @param v The global value stack. 150 * @return A pointer to the top value in @a v. 151 */ 152 #define BC_PROG_GLOBAL_PTR(v) (bc_vec_top(v)) 153 154 /** 155 * Get the top value in a global value stack. 156 * @param v The global value stack. 157 * @return The top value in @a v. 158 */ 159 #define BC_PROG_GLOBAL(v) (*((BcBigDig*) BC_PROG_GLOBAL_PTR(v))) 160 161 /** 162 * Returns the current value of ibase. 163 * @param p The program. 164 * @return The current ibase. 165 */ 166 #define BC_PROG_IBASE(p) ((p)->globals[BC_PROG_GLOBALS_IBASE]) 167 168 /** 169 * Returns the current value of obase. 170 * @param p The program. 171 * @return The current obase. 172 */ 173 #define BC_PROG_OBASE(p) ((p)->globals[BC_PROG_GLOBALS_OBASE]) 174 175 /** 176 * Returns the current value of scale. 177 * @param p The program. 178 * @return The current scale. 179 */ 180 #define BC_PROG_SCALE(p) ((p)->globals[BC_PROG_GLOBALS_SCALE]) 181 182 /// The index for the main function in the functions array.// 183 #define BC_PROG_MAIN (0) 184 185 /// The index for the read function in the functions array. 186 #define BC_PROG_READ (1) 187 188 /** 189 * Retires (completes the execution of) an instruction. Some instructions 190 * require special retirement, but most can use this. This basically pops the 191 * operands while preserving the result (which we assumed was pushed before the 192 * actual operation). 193 * @param p The program. 194 * @param nres The number of results returned by the instruction. 195 * @param nops The number of operands used by the instruction. 196 */ 197 #define bc_program_retire(p, nres, nops) \ 198 (bc_vec_npopAt(&(p)->results, (nops), (p)->results.len - (nres + nops))) 199 200 #if DC_ENABLED 201 202 /// A constant that tells how many functions are required in dc. 203 #define BC_PROG_REQ_FUNCS (2) 204 205 #if !BC_ENABLED 206 207 /// This define disappears the parameter last because for dc only, last is 208 /// always true. 209 #define bc_program_copyToVar(p, name, t, last) \ 210 bc_program_copyToVar(p, name, t) 211 212 #endif // !BC_ENABLED 213 214 #else // DC_ENABLED 215 216 /// This define disappears pop and copy because for bc, 'pop' and 'copy' are 217 /// always false. 218 #define bc_program_pushVar(p, code, bgn, pop, copy) \ 219 bc_program_pushVar(p, code, bgn) 220 221 // In debug mode, we want bc to check the stack, but otherwise, we don't because 222 // the bc language implicitly mandates that the stack should always have enough 223 // items. 224 #ifdef NDEBUG 225 #define BC_PROG_NO_STACK_CHECK 226 #endif // NDEBUG 227 228 #endif // DC_ENABLED 229 230 /** 231 * Returns true if the BcNum @a n is acting as a string. 232 * @param n The BcNum to test. 233 * @return True if @a n is acting as a string, false otherwise. 234 */ 235 #define BC_PROG_STR(n) ((n)->num == NULL && !(n)->cap) 236 237 #if BC_ENABLED 238 239 /** 240 * Returns true if the result @a r and @a n is a number. 241 * @param r The result. 242 * @param n The number corresponding to the result. 243 * @return True if the result holds a number, false otherwise. 244 */ 245 #define BC_PROG_NUM(r, n) \ 246 ((r)->t != BC_RESULT_ARRAY && (r)->t != BC_RESULT_STR && !BC_PROG_STR(n)) 247 248 #else // BC_ENABLED 249 250 /** 251 * Returns true if the result @a r and @a n is a number. 252 * @param r The result. 253 * @param n The number corresponding to the result. 254 * @return True if the result holds a number, false otherwise. 255 */ 256 #define BC_PROG_NUM(r, n) ((r)->t != BC_RESULT_STR && !BC_PROG_STR(n)) 257 258 #endif // BC_ENABLED 259 260 /** 261 * This is a function type for unary operations. Currently, these include 262 * boolean not, negation, and truncation with extra math. 263 * @param r The BcResult to store the result into. 264 * @param n The parameter to the unary operation. 265 */ 266 typedef void (*BcProgramUnary)(BcResult *r, BcNum *n); 267 268 /** 269 * Initializes the BcProgram. 270 * @param p The program to initialize. 271 */ 272 void bc_program_init(BcProgram *p); 273 274 #ifndef NDEBUG 275 276 /** 277 * Frees a BcProgram. This is only used in debug builds because a BcProgram is 278 * only freed on program exit, and we don't care about freeing resources on 279 * exit. 280 * @param p The program to initialize. 281 */ 282 void bc_program_free(BcProgram *p); 283 284 #endif // NDEBUG 285 286 #if BC_DEBUG_CODE 287 #if BC_ENABLED && DC_ENABLED 288 289 /** 290 * Prints the bytecode in a function. This is a debug-only function. 291 * @param p The program. 292 */ 293 void bc_program_code(const BcProgram *p); 294 295 /** 296 * Prints an instruction. This is a debug-only function. 297 * @param p The program. 298 * @param code The bytecode array. 299 * @param bgn A pointer to the current index. It is also updated to the next 300 * index. 301 */ 302 void bc_program_printInst(const BcProgram *p, const char *code, 303 size_t *restrict bgn); 304 305 /** 306 * Prints the stack. This is a debug-only function. 307 * @param p The program. 308 */ 309 void bc_program_printStackDebug(BcProgram* p); 310 311 #endif // BC_ENABLED && DC_ENABLED 312 #endif // BC_DEBUG_CODE 313 314 /** 315 * Returns the index of the variable or array in their respective arrays. 316 * @param p The program. 317 * @param id The BcId of the variable or array. 318 * @param var True if the search should be for a variable, false for an array. 319 * @return The index of the variable or array in the correct array. 320 */ 321 size_t bc_program_search(BcProgram *p, const char* id, bool var); 322 323 /** 324 * Adds a string to a function and returns the string's index in the function. 325 * @param p The program. 326 * @param str The string to add. 327 * @param fidx The index of the function to add to. 328 */ 329 size_t bc_program_addString(BcProgram *p, const char *str, size_t fidx); 330 331 /** 332 * Inserts a function into the program and returns the index of the function in 333 * the fns array. 334 * @param p The program. 335 * @param name The name of the function. 336 * @return The index of the function after insertion. 337 */ 338 size_t bc_program_insertFunc(BcProgram *p, const char *name); 339 340 /** 341 * Resets a program, usually because of resetting after an error. 342 * @param p The program to reset. 343 */ 344 void bc_program_reset(BcProgram *p); 345 346 /** 347 * Executes bc or dc code in the BcProgram. 348 * @param p The program. 349 */ 350 void bc_program_exec(BcProgram *p); 351 352 /** 353 * Negates a copy of a BcNum. This is a BcProgramUnary function. 354 * @param r The BcResult to store the result into. 355 * @param n The parameter to the unary operation. 356 */ 357 void bc_program_negate(BcResult *r, BcNum *n); 358 359 /** 360 * Returns a boolean not of a BcNum. This is a BcProgramUnary function. 361 * @param r The BcResult to store the result into. 362 * @param n The parameter to the unary operation. 363 */ 364 void bc_program_not(BcResult *r, BcNum *n); 365 366 #if BC_ENABLE_EXTRA_MATH 367 368 /** 369 * Truncates a copy of a BcNum. This is a BcProgramUnary function. 370 * @param r The BcResult to store the result into. 371 * @param n The parameter to the unary operation. 372 */ 373 void bc_program_trunc(BcResult *r, BcNum *n); 374 375 #endif // BC_ENABLE_EXTRA_MATH 376 377 /// A reference to an array of binary operator functions. 378 extern const BcNumBinaryOp bc_program_ops[]; 379 380 /// A reference to an array of binary operator allocation request functions. 381 extern const BcNumBinaryOpReq bc_program_opReqs[]; 382 383 /// A reference to an array of unary operator functions. 384 extern const BcProgramUnary bc_program_unarys[]; 385 386 /// A reference to a filename for command-line expressions. 387 extern const char bc_program_exprs_name[]; 388 389 /// A reference to a filename for stdin. 390 extern const char bc_program_stdin_name[]; 391 392 /// A reference to the ready message printed on SIGINT. 393 extern const char bc_program_ready_msg[]; 394 395 /// A reference to the length of the ready message. 396 extern const size_t bc_program_ready_msg_len; 397 398 /// A reference to an array of escape characters for the print statement. 399 extern const char bc_program_esc_chars[]; 400 401 /// A reference to an array of the characters corresponding to the escape 402 /// characters in bc_program_esc_chars. 403 extern const char bc_program_esc_seqs[]; 404 405 #if BC_HAS_COMPUTED_GOTO 406 407 #if BC_DEBUG_CODE 408 409 #define BC_PROG_JUMP(inst, code, ip) \ 410 do { \ 411 inst = (uchar) (code)[(ip)->idx++]; \ 412 bc_file_printf(&vm.ferr, "inst: %s\n", bc_inst_names[inst]); \ 413 bc_file_flush(&vm.ferr, bc_flush_none); \ 414 goto *bc_program_inst_lbls[inst]; \ 415 } while (0) 416 417 #else // BC_DEBUG_CODE 418 419 #define BC_PROG_JUMP(inst, code, ip) \ 420 do { \ 421 inst = (uchar) (code)[(ip)->idx++]; \ 422 goto *bc_program_inst_lbls[inst]; \ 423 } while (0) 424 425 #endif // BC_DEBUG_CODE 426 427 #define BC_PROG_DIRECT_JUMP(l) goto lbl_ ## l; 428 #define BC_PROG_LBL(l) lbl_ ## l 429 #define BC_PROG_FALLTHROUGH 430 431 #if BC_C11 432 433 #define BC_PROG_LBLS_SIZE (sizeof(bc_program_inst_lbls) / sizeof(void*)) 434 #define BC_PROG_LBLS_ASSERT \ 435 static_assert(BC_PROG_LBLS_SIZE == BC_INST_INVALID + 1,\ 436 "bc_program_inst_lbls[] mismatches the instructions") 437 438 #else // BC_C11 439 440 #define BC_PROG_LBLS_ASSERT 441 442 #endif // BC_C11 443 444 #if BC_ENABLED 445 446 #if DC_ENABLED 447 448 #if BC_ENABLE_EXTRA_MATH 449 450 #define BC_PROG_LBLS static const void* const bc_program_inst_lbls[] = { \ 451 &&lbl_BC_INST_INC, \ 452 &&lbl_BC_INST_DEC, \ 453 &&lbl_BC_INST_NEG, \ 454 &&lbl_BC_INST_BOOL_NOT, \ 455 &&lbl_BC_INST_TRUNC, \ 456 &&lbl_BC_INST_POWER, \ 457 &&lbl_BC_INST_MULTIPLY, \ 458 &&lbl_BC_INST_DIVIDE, \ 459 &&lbl_BC_INST_MODULUS, \ 460 &&lbl_BC_INST_PLUS, \ 461 &&lbl_BC_INST_MINUS, \ 462 &&lbl_BC_INST_PLACES, \ 463 &&lbl_BC_INST_LSHIFT, \ 464 &&lbl_BC_INST_RSHIFT, \ 465 &&lbl_BC_INST_REL_EQ, \ 466 &&lbl_BC_INST_REL_LE, \ 467 &&lbl_BC_INST_REL_GE, \ 468 &&lbl_BC_INST_REL_NE, \ 469 &&lbl_BC_INST_REL_LT, \ 470 &&lbl_BC_INST_REL_GT, \ 471 &&lbl_BC_INST_BOOL_OR, \ 472 &&lbl_BC_INST_BOOL_AND, \ 473 &&lbl_BC_INST_ASSIGN_POWER, \ 474 &&lbl_BC_INST_ASSIGN_MULTIPLY, \ 475 &&lbl_BC_INST_ASSIGN_DIVIDE, \ 476 &&lbl_BC_INST_ASSIGN_MODULUS, \ 477 &&lbl_BC_INST_ASSIGN_PLUS, \ 478 &&lbl_BC_INST_ASSIGN_MINUS, \ 479 &&lbl_BC_INST_ASSIGN_PLACES, \ 480 &&lbl_BC_INST_ASSIGN_LSHIFT, \ 481 &&lbl_BC_INST_ASSIGN_RSHIFT, \ 482 &&lbl_BC_INST_ASSIGN, \ 483 &&lbl_BC_INST_ASSIGN_POWER_NO_VAL, \ 484 &&lbl_BC_INST_ASSIGN_MULTIPLY_NO_VAL, \ 485 &&lbl_BC_INST_ASSIGN_DIVIDE_NO_VAL, \ 486 &&lbl_BC_INST_ASSIGN_MODULUS_NO_VAL, \ 487 &&lbl_BC_INST_ASSIGN_PLUS_NO_VAL, \ 488 &&lbl_BC_INST_ASSIGN_MINUS_NO_VAL, \ 489 &&lbl_BC_INST_ASSIGN_PLACES_NO_VAL, \ 490 &&lbl_BC_INST_ASSIGN_LSHIFT_NO_VAL, \ 491 &&lbl_BC_INST_ASSIGN_RSHIFT_NO_VAL, \ 492 &&lbl_BC_INST_ASSIGN_NO_VAL, \ 493 &&lbl_BC_INST_NUM, \ 494 &&lbl_BC_INST_VAR, \ 495 &&lbl_BC_INST_ARRAY_ELEM, \ 496 &&lbl_BC_INST_ARRAY, \ 497 &&lbl_BC_INST_ZERO, \ 498 &&lbl_BC_INST_ONE, \ 499 &&lbl_BC_INST_LAST, \ 500 &&lbl_BC_INST_IBASE, \ 501 &&lbl_BC_INST_OBASE, \ 502 &&lbl_BC_INST_SCALE, \ 503 &&lbl_BC_INST_SEED, \ 504 &&lbl_BC_INST_LENGTH, \ 505 &&lbl_BC_INST_SCALE_FUNC, \ 506 &&lbl_BC_INST_SQRT, \ 507 &&lbl_BC_INST_ABS, \ 508 &&lbl_BC_INST_IRAND, \ 509 &&lbl_BC_INST_ASCIIFY, \ 510 &&lbl_BC_INST_READ, \ 511 &&lbl_BC_INST_RAND, \ 512 &&lbl_BC_INST_MAXIBASE, \ 513 &&lbl_BC_INST_MAXOBASE, \ 514 &&lbl_BC_INST_MAXSCALE, \ 515 &&lbl_BC_INST_MAXRAND, \ 516 &&lbl_BC_INST_PRINT, \ 517 &&lbl_BC_INST_PRINT_POP, \ 518 &&lbl_BC_INST_STR, \ 519 &&lbl_BC_INST_PRINT_STR, \ 520 &&lbl_BC_INST_JUMP, \ 521 &&lbl_BC_INST_JUMP_ZERO, \ 522 &&lbl_BC_INST_CALL, \ 523 &&lbl_BC_INST_RET, \ 524 &&lbl_BC_INST_RET0, \ 525 &&lbl_BC_INST_RET_VOID, \ 526 &&lbl_BC_INST_HALT, \ 527 &&lbl_BC_INST_POP, \ 528 &&lbl_BC_INST_SWAP, \ 529 &&lbl_BC_INST_MODEXP, \ 530 &&lbl_BC_INST_DIVMOD, \ 531 &&lbl_BC_INST_PRINT_STREAM, \ 532 &&lbl_BC_INST_POP_EXEC, \ 533 &&lbl_BC_INST_EXECUTE, \ 534 &&lbl_BC_INST_EXEC_COND, \ 535 &&lbl_BC_INST_PRINT_STACK, \ 536 &&lbl_BC_INST_CLEAR_STACK, \ 537 &&lbl_BC_INST_REG_STACK_LEN, \ 538 &&lbl_BC_INST_STACK_LEN, \ 539 &&lbl_BC_INST_DUPLICATE, \ 540 &&lbl_BC_INST_LOAD, \ 541 &&lbl_BC_INST_PUSH_VAR, \ 542 &&lbl_BC_INST_PUSH_TO_VAR, \ 543 &&lbl_BC_INST_QUIT, \ 544 &&lbl_BC_INST_NQUIT, \ 545 &&lbl_BC_INST_EXEC_STACK_LEN, \ 546 &&lbl_BC_INST_INVALID, \ 547 } 548 549 #else // BC_ENABLE_EXTRA_MATH 550 551 #define BC_PROG_LBLS static const void* const bc_program_inst_lbls[] = { \ 552 &&lbl_BC_INST_INC, \ 553 &&lbl_BC_INST_DEC, \ 554 &&lbl_BC_INST_NEG, \ 555 &&lbl_BC_INST_BOOL_NOT, \ 556 &&lbl_BC_INST_POWER, \ 557 &&lbl_BC_INST_MULTIPLY, \ 558 &&lbl_BC_INST_DIVIDE, \ 559 &&lbl_BC_INST_MODULUS, \ 560 &&lbl_BC_INST_PLUS, \ 561 &&lbl_BC_INST_MINUS, \ 562 &&lbl_BC_INST_REL_EQ, \ 563 &&lbl_BC_INST_REL_LE, \ 564 &&lbl_BC_INST_REL_GE, \ 565 &&lbl_BC_INST_REL_NE, \ 566 &&lbl_BC_INST_REL_LT, \ 567 &&lbl_BC_INST_REL_GT, \ 568 &&lbl_BC_INST_BOOL_OR, \ 569 &&lbl_BC_INST_BOOL_AND, \ 570 &&lbl_BC_INST_ASSIGN_POWER, \ 571 &&lbl_BC_INST_ASSIGN_MULTIPLY, \ 572 &&lbl_BC_INST_ASSIGN_DIVIDE, \ 573 &&lbl_BC_INST_ASSIGN_MODULUS, \ 574 &&lbl_BC_INST_ASSIGN_PLUS, \ 575 &&lbl_BC_INST_ASSIGN_MINUS, \ 576 &&lbl_BC_INST_ASSIGN, \ 577 &&lbl_BC_INST_ASSIGN_POWER_NO_VAL, \ 578 &&lbl_BC_INST_ASSIGN_MULTIPLY_NO_VAL, \ 579 &&lbl_BC_INST_ASSIGN_DIVIDE_NO_VAL, \ 580 &&lbl_BC_INST_ASSIGN_MODULUS_NO_VAL, \ 581 &&lbl_BC_INST_ASSIGN_PLUS_NO_VAL, \ 582 &&lbl_BC_INST_ASSIGN_MINUS_NO_VAL, \ 583 &&lbl_BC_INST_ASSIGN_NO_VAL, \ 584 &&lbl_BC_INST_NUM, \ 585 &&lbl_BC_INST_VAR, \ 586 &&lbl_BC_INST_ARRAY_ELEM, \ 587 &&lbl_BC_INST_ARRAY, \ 588 &&lbl_BC_INST_ZERO, \ 589 &&lbl_BC_INST_ONE, \ 590 &&lbl_BC_INST_LAST, \ 591 &&lbl_BC_INST_IBASE, \ 592 &&lbl_BC_INST_OBASE, \ 593 &&lbl_BC_INST_SCALE, \ 594 &&lbl_BC_INST_LENGTH, \ 595 &&lbl_BC_INST_SCALE_FUNC, \ 596 &&lbl_BC_INST_SQRT, \ 597 &&lbl_BC_INST_ABS, \ 598 &&lbl_BC_INST_ASCIIFY, \ 599 &&lbl_BC_INST_READ, \ 600 &&lbl_BC_INST_MAXIBASE, \ 601 &&lbl_BC_INST_MAXOBASE, \ 602 &&lbl_BC_INST_MAXSCALE, \ 603 &&lbl_BC_INST_PRINT, \ 604 &&lbl_BC_INST_PRINT_POP, \ 605 &&lbl_BC_INST_STR, \ 606 &&lbl_BC_INST_PRINT_STR, \ 607 &&lbl_BC_INST_JUMP, \ 608 &&lbl_BC_INST_JUMP_ZERO, \ 609 &&lbl_BC_INST_CALL, \ 610 &&lbl_BC_INST_RET, \ 611 &&lbl_BC_INST_RET0, \ 612 &&lbl_BC_INST_RET_VOID, \ 613 &&lbl_BC_INST_HALT, \ 614 &&lbl_BC_INST_POP, \ 615 &&lbl_BC_INST_SWAP, \ 616 &&lbl_BC_INST_MODEXP, \ 617 &&lbl_BC_INST_DIVMOD, \ 618 &&lbl_BC_INST_PRINT_STREAM, \ 619 &&lbl_BC_INST_POP_EXEC, \ 620 &&lbl_BC_INST_EXECUTE, \ 621 &&lbl_BC_INST_EXEC_COND, \ 622 &&lbl_BC_INST_PRINT_STACK, \ 623 &&lbl_BC_INST_CLEAR_STACK, \ 624 &&lbl_BC_INST_REG_STACK_LEN, \ 625 &&lbl_BC_INST_STACK_LEN, \ 626 &&lbl_BC_INST_DUPLICATE, \ 627 &&lbl_BC_INST_LOAD, \ 628 &&lbl_BC_INST_PUSH_VAR, \ 629 &&lbl_BC_INST_PUSH_TO_VAR, \ 630 &&lbl_BC_INST_QUIT, \ 631 &&lbl_BC_INST_NQUIT, \ 632 &&lbl_BC_INST_EXEC_STACK_LEN, \ 633 &&lbl_BC_INST_INVALID, \ 634 } 635 636 #endif // BC_ENABLE_EXTRA_MATH 637 638 #else // DC_ENABLED 639 640 #if BC_ENABLE_EXTRA_MATH 641 642 #define BC_PROG_LBLS static const void* const bc_program_inst_lbls[] = { \ 643 &&lbl_BC_INST_INC, \ 644 &&lbl_BC_INST_DEC, \ 645 &&lbl_BC_INST_NEG, \ 646 &&lbl_BC_INST_BOOL_NOT, \ 647 &&lbl_BC_INST_TRUNC, \ 648 &&lbl_BC_INST_POWER, \ 649 &&lbl_BC_INST_MULTIPLY, \ 650 &&lbl_BC_INST_DIVIDE, \ 651 &&lbl_BC_INST_MODULUS, \ 652 &&lbl_BC_INST_PLUS, \ 653 &&lbl_BC_INST_MINUS, \ 654 &&lbl_BC_INST_PLACES, \ 655 &&lbl_BC_INST_LSHIFT, \ 656 &&lbl_BC_INST_RSHIFT, \ 657 &&lbl_BC_INST_REL_EQ, \ 658 &&lbl_BC_INST_REL_LE, \ 659 &&lbl_BC_INST_REL_GE, \ 660 &&lbl_BC_INST_REL_NE, \ 661 &&lbl_BC_INST_REL_LT, \ 662 &&lbl_BC_INST_REL_GT, \ 663 &&lbl_BC_INST_BOOL_OR, \ 664 &&lbl_BC_INST_BOOL_AND, \ 665 &&lbl_BC_INST_ASSIGN_POWER, \ 666 &&lbl_BC_INST_ASSIGN_MULTIPLY, \ 667 &&lbl_BC_INST_ASSIGN_DIVIDE, \ 668 &&lbl_BC_INST_ASSIGN_MODULUS, \ 669 &&lbl_BC_INST_ASSIGN_PLUS, \ 670 &&lbl_BC_INST_ASSIGN_MINUS, \ 671 &&lbl_BC_INST_ASSIGN_PLACES, \ 672 &&lbl_BC_INST_ASSIGN_LSHIFT, \ 673 &&lbl_BC_INST_ASSIGN_RSHIFT, \ 674 &&lbl_BC_INST_ASSIGN, \ 675 &&lbl_BC_INST_ASSIGN_POWER_NO_VAL, \ 676 &&lbl_BC_INST_ASSIGN_MULTIPLY_NO_VAL, \ 677 &&lbl_BC_INST_ASSIGN_DIVIDE_NO_VAL, \ 678 &&lbl_BC_INST_ASSIGN_MODULUS_NO_VAL, \ 679 &&lbl_BC_INST_ASSIGN_PLUS_NO_VAL, \ 680 &&lbl_BC_INST_ASSIGN_MINUS_NO_VAL, \ 681 &&lbl_BC_INST_ASSIGN_PLACES_NO_VAL, \ 682 &&lbl_BC_INST_ASSIGN_LSHIFT_NO_VAL, \ 683 &&lbl_BC_INST_ASSIGN_RSHIFT_NO_VAL, \ 684 &&lbl_BC_INST_ASSIGN_NO_VAL, \ 685 &&lbl_BC_INST_NUM, \ 686 &&lbl_BC_INST_VAR, \ 687 &&lbl_BC_INST_ARRAY_ELEM, \ 688 &&lbl_BC_INST_ARRAY, \ 689 &&lbl_BC_INST_ZERO, \ 690 &&lbl_BC_INST_ONE, \ 691 &&lbl_BC_INST_LAST, \ 692 &&lbl_BC_INST_IBASE, \ 693 &&lbl_BC_INST_OBASE, \ 694 &&lbl_BC_INST_SCALE, \ 695 &&lbl_BC_INST_SEED, \ 696 &&lbl_BC_INST_LENGTH, \ 697 &&lbl_BC_INST_SCALE_FUNC, \ 698 &&lbl_BC_INST_SQRT, \ 699 &&lbl_BC_INST_ABS, \ 700 &&lbl_BC_INST_IRAND, \ 701 &&lbl_BC_INST_ASCIIFY, \ 702 &&lbl_BC_INST_READ, \ 703 &&lbl_BC_INST_RAND, \ 704 &&lbl_BC_INST_MAXIBASE, \ 705 &&lbl_BC_INST_MAXOBASE, \ 706 &&lbl_BC_INST_MAXSCALE, \ 707 &&lbl_BC_INST_MAXRAND, \ 708 &&lbl_BC_INST_PRINT, \ 709 &&lbl_BC_INST_PRINT_POP, \ 710 &&lbl_BC_INST_STR, \ 711 &&lbl_BC_INST_PRINT_STR, \ 712 &&lbl_BC_INST_JUMP, \ 713 &&lbl_BC_INST_JUMP_ZERO, \ 714 &&lbl_BC_INST_CALL, \ 715 &&lbl_BC_INST_RET, \ 716 &&lbl_BC_INST_RET0, \ 717 &&lbl_BC_INST_RET_VOID, \ 718 &&lbl_BC_INST_HALT, \ 719 &&lbl_BC_INST_POP, \ 720 &&lbl_BC_INST_SWAP, \ 721 &&lbl_BC_INST_MODEXP, \ 722 &&lbl_BC_INST_DIVMOD, \ 723 &&lbl_BC_INST_PRINT_STREAM, \ 724 &&lbl_BC_INST_INVALID, \ 725 } 726 727 #else // BC_ENABLE_EXTRA_MATH 728 729 #define BC_PROG_LBLS static const void* const bc_program_inst_lbls[] = { \ 730 &&lbl_BC_INST_INC, \ 731 &&lbl_BC_INST_DEC, \ 732 &&lbl_BC_INST_NEG, \ 733 &&lbl_BC_INST_BOOL_NOT, \ 734 &&lbl_BC_INST_POWER, \ 735 &&lbl_BC_INST_MULTIPLY, \ 736 &&lbl_BC_INST_DIVIDE, \ 737 &&lbl_BC_INST_MODULUS, \ 738 &&lbl_BC_INST_PLUS, \ 739 &&lbl_BC_INST_MINUS, \ 740 &&lbl_BC_INST_REL_EQ, \ 741 &&lbl_BC_INST_REL_LE, \ 742 &&lbl_BC_INST_REL_GE, \ 743 &&lbl_BC_INST_REL_NE, \ 744 &&lbl_BC_INST_REL_LT, \ 745 &&lbl_BC_INST_REL_GT, \ 746 &&lbl_BC_INST_BOOL_OR, \ 747 &&lbl_BC_INST_BOOL_AND, \ 748 &&lbl_BC_INST_ASSIGN_POWER, \ 749 &&lbl_BC_INST_ASSIGN_MULTIPLY, \ 750 &&lbl_BC_INST_ASSIGN_DIVIDE, \ 751 &&lbl_BC_INST_ASSIGN_MODULUS, \ 752 &&lbl_BC_INST_ASSIGN_PLUS, \ 753 &&lbl_BC_INST_ASSIGN_MINUS, \ 754 &&lbl_BC_INST_ASSIGN, \ 755 &&lbl_BC_INST_ASSIGN_POWER_NO_VAL, \ 756 &&lbl_BC_INST_ASSIGN_MULTIPLY_NO_VAL, \ 757 &&lbl_BC_INST_ASSIGN_DIVIDE_NO_VAL, \ 758 &&lbl_BC_INST_ASSIGN_MODULUS_NO_VAL, \ 759 &&lbl_BC_INST_ASSIGN_PLUS_NO_VAL, \ 760 &&lbl_BC_INST_ASSIGN_MINUS_NO_VAL, \ 761 &&lbl_BC_INST_ASSIGN_NO_VAL, \ 762 &&lbl_BC_INST_NUM, \ 763 &&lbl_BC_INST_VAR, \ 764 &&lbl_BC_INST_ARRAY_ELEM, \ 765 &&lbl_BC_INST_ARRAY, \ 766 &&lbl_BC_INST_ZERO, \ 767 &&lbl_BC_INST_ONE, \ 768 &&lbl_BC_INST_LAST, \ 769 &&lbl_BC_INST_IBASE, \ 770 &&lbl_BC_INST_OBASE, \ 771 &&lbl_BC_INST_SCALE, \ 772 &&lbl_BC_INST_LENGTH, \ 773 &&lbl_BC_INST_SCALE_FUNC, \ 774 &&lbl_BC_INST_SQRT, \ 775 &&lbl_BC_INST_ABS, \ 776 &&lbl_BC_INST_ASCIIFY, \ 777 &&lbl_BC_INST_READ, \ 778 &&lbl_BC_INST_MAXIBASE, \ 779 &&lbl_BC_INST_MAXOBASE, \ 780 &&lbl_BC_INST_MAXSCALE, \ 781 &&lbl_BC_INST_PRINT, \ 782 &&lbl_BC_INST_PRINT_POP, \ 783 &&lbl_BC_INST_STR, \ 784 &&lbl_BC_INST_PRINT_STR, \ 785 &&lbl_BC_INST_JUMP, \ 786 &&lbl_BC_INST_JUMP_ZERO, \ 787 &&lbl_BC_INST_CALL, \ 788 &&lbl_BC_INST_RET, \ 789 &&lbl_BC_INST_RET0, \ 790 &&lbl_BC_INST_RET_VOID, \ 791 &&lbl_BC_INST_HALT, \ 792 &&lbl_BC_INST_POP, \ 793 &&lbl_BC_INST_SWAP, \ 794 &&lbl_BC_INST_MODEXP, \ 795 &&lbl_BC_INST_DIVMOD, \ 796 &&lbl_BC_INST_PRINT_STREAM, \ 797 &&lbl_BC_INST_INVALID, \ 798 } 799 800 #endif // BC_ENABLE_EXTRA_MATH 801 802 #endif // DC_ENABLED 803 804 #else // BC_ENABLED 805 806 #if BC_ENABLE_EXTRA_MATH 807 808 #define BC_PROG_LBLS static const void* const bc_program_inst_lbls[] = { \ 809 &&lbl_BC_INST_NEG, \ 810 &&lbl_BC_INST_BOOL_NOT, \ 811 &&lbl_BC_INST_TRUNC, \ 812 &&lbl_BC_INST_POWER, \ 813 &&lbl_BC_INST_MULTIPLY, \ 814 &&lbl_BC_INST_DIVIDE, \ 815 &&lbl_BC_INST_MODULUS, \ 816 &&lbl_BC_INST_PLUS, \ 817 &&lbl_BC_INST_MINUS, \ 818 &&lbl_BC_INST_PLACES, \ 819 &&lbl_BC_INST_LSHIFT, \ 820 &&lbl_BC_INST_RSHIFT, \ 821 &&lbl_BC_INST_REL_EQ, \ 822 &&lbl_BC_INST_REL_LE, \ 823 &&lbl_BC_INST_REL_GE, \ 824 &&lbl_BC_INST_REL_NE, \ 825 &&lbl_BC_INST_REL_LT, \ 826 &&lbl_BC_INST_REL_GT, \ 827 &&lbl_BC_INST_BOOL_OR, \ 828 &&lbl_BC_INST_BOOL_AND, \ 829 &&lbl_BC_INST_ASSIGN_NO_VAL, \ 830 &&lbl_BC_INST_NUM, \ 831 &&lbl_BC_INST_VAR, \ 832 &&lbl_BC_INST_ARRAY_ELEM, \ 833 &&lbl_BC_INST_ARRAY, \ 834 &&lbl_BC_INST_ZERO, \ 835 &&lbl_BC_INST_ONE, \ 836 &&lbl_BC_INST_IBASE, \ 837 &&lbl_BC_INST_OBASE, \ 838 &&lbl_BC_INST_SCALE, \ 839 &&lbl_BC_INST_SEED, \ 840 &&lbl_BC_INST_LENGTH, \ 841 &&lbl_BC_INST_SCALE_FUNC, \ 842 &&lbl_BC_INST_SQRT, \ 843 &&lbl_BC_INST_ABS, \ 844 &&lbl_BC_INST_IRAND, \ 845 &&lbl_BC_INST_ASCIIFY, \ 846 &&lbl_BC_INST_READ, \ 847 &&lbl_BC_INST_RAND, \ 848 &&lbl_BC_INST_MAXIBASE, \ 849 &&lbl_BC_INST_MAXOBASE, \ 850 &&lbl_BC_INST_MAXSCALE, \ 851 &&lbl_BC_INST_MAXRAND, \ 852 &&lbl_BC_INST_PRINT, \ 853 &&lbl_BC_INST_PRINT_POP, \ 854 &&lbl_BC_INST_STR, \ 855 &&lbl_BC_INST_POP, \ 856 &&lbl_BC_INST_SWAP, \ 857 &&lbl_BC_INST_MODEXP, \ 858 &&lbl_BC_INST_DIVMOD, \ 859 &&lbl_BC_INST_PRINT_STREAM, \ 860 &&lbl_BC_INST_POP_EXEC, \ 861 &&lbl_BC_INST_EXECUTE, \ 862 &&lbl_BC_INST_EXEC_COND, \ 863 &&lbl_BC_INST_PRINT_STACK, \ 864 &&lbl_BC_INST_CLEAR_STACK, \ 865 &&lbl_BC_INST_REG_STACK_LEN, \ 866 &&lbl_BC_INST_STACK_LEN, \ 867 &&lbl_BC_INST_DUPLICATE, \ 868 &&lbl_BC_INST_LOAD, \ 869 &&lbl_BC_INST_PUSH_VAR, \ 870 &&lbl_BC_INST_PUSH_TO_VAR, \ 871 &&lbl_BC_INST_QUIT, \ 872 &&lbl_BC_INST_NQUIT, \ 873 &&lbl_BC_INST_EXEC_STACK_LEN, \ 874 &&lbl_BC_INST_INVALID, \ 875 } 876 877 #else // BC_ENABLE_EXTRA_MATH 878 879 #define BC_PROG_LBLS static const void* const bc_program_inst_lbls[] = { \ 880 &&lbl_BC_INST_NEG, \ 881 &&lbl_BC_INST_BOOL_NOT, \ 882 &&lbl_BC_INST_POWER, \ 883 &&lbl_BC_INST_MULTIPLY, \ 884 &&lbl_BC_INST_DIVIDE, \ 885 &&lbl_BC_INST_MODULUS, \ 886 &&lbl_BC_INST_PLUS, \ 887 &&lbl_BC_INST_MINUS, \ 888 &&lbl_BC_INST_REL_EQ, \ 889 &&lbl_BC_INST_REL_LE, \ 890 &&lbl_BC_INST_REL_GE, \ 891 &&lbl_BC_INST_REL_NE, \ 892 &&lbl_BC_INST_REL_LT, \ 893 &&lbl_BC_INST_REL_GT, \ 894 &&lbl_BC_INST_BOOL_OR, \ 895 &&lbl_BC_INST_BOOL_AND, \ 896 &&lbl_BC_INST_ASSIGN_NO_VAL, \ 897 &&lbl_BC_INST_NUM, \ 898 &&lbl_BC_INST_VAR, \ 899 &&lbl_BC_INST_ARRAY_ELEM, \ 900 &&lbl_BC_INST_ARRAY, \ 901 &&lbl_BC_INST_ZERO, \ 902 &&lbl_BC_INST_ONE, \ 903 &&lbl_BC_INST_IBASE, \ 904 &&lbl_BC_INST_OBASE, \ 905 &&lbl_BC_INST_SCALE, \ 906 &&lbl_BC_INST_LENGTH, \ 907 &&lbl_BC_INST_SCALE_FUNC, \ 908 &&lbl_BC_INST_SQRT, \ 909 &&lbl_BC_INST_ABS, \ 910 &&lbl_BC_INST_ASCIIFY, \ 911 &&lbl_BC_INST_READ, \ 912 &&lbl_BC_INST_MAXIBASE, \ 913 &&lbl_BC_INST_MAXOBASE, \ 914 &&lbl_BC_INST_MAXSCALE, \ 915 &&lbl_BC_INST_PRINT, \ 916 &&lbl_BC_INST_PRINT_POP, \ 917 &&lbl_BC_INST_STR, \ 918 &&lbl_BC_INST_POP, \ 919 &&lbl_BC_INST_SWAP, \ 920 &&lbl_BC_INST_MODEXP, \ 921 &&lbl_BC_INST_DIVMOD, \ 922 &&lbl_BC_INST_PRINT_STREAM, \ 923 &&lbl_BC_INST_POP_EXEC, \ 924 &&lbl_BC_INST_EXECUTE, \ 925 &&lbl_BC_INST_EXEC_COND, \ 926 &&lbl_BC_INST_PRINT_STACK, \ 927 &&lbl_BC_INST_CLEAR_STACK, \ 928 &&lbl_BC_INST_REG_STACK_LEN, \ 929 &&lbl_BC_INST_STACK_LEN, \ 930 &&lbl_BC_INST_DUPLICATE, \ 931 &&lbl_BC_INST_LOAD, \ 932 &&lbl_BC_INST_PUSH_VAR, \ 933 &&lbl_BC_INST_PUSH_TO_VAR, \ 934 &&lbl_BC_INST_QUIT, \ 935 &&lbl_BC_INST_NQUIT, \ 936 &&lbl_BC_INST_EXEC_STACK_LEN, \ 937 &&lbl_BC_INST_INVALID, \ 938 } 939 940 #endif // BC_ENABLE_EXTRA_MATH 941 942 #endif // BC_ENABLED 943 944 #else // BC_HAS_COMPUTED_GOTO 945 946 #define BC_PROG_JUMP(inst, code, ip) break 947 #define BC_PROG_DIRECT_JUMP(l) 948 #define BC_PROG_LBL(l) case l 949 #define BC_PROG_FALLTHROUGH BC_FALLTHROUGH 950 951 #define BC_PROG_LBLS 952 953 #endif // BC_HAS_COMPUTED_GOTO 954 955 #endif // BC_PROGRAM_H 956