xref: /freebsd/contrib/bc/include/vm.h (revision 924226fba12cc9a228c73b956e1b7fa24c60b055)
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