xref: /freebsd/contrib/bc/src/program.c (revision 69c5fa5cd1ec9b09ed88a086607a8a0993818db9)
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  * Code to execute bc programs.
33  *
34  */
35 
36 #include <assert.h>
37 #include <stdbool.h>
38 #include <string.h>
39 
40 #include <setjmp.h>
41 
42 #include <signal.h>
43 
44 #include <time.h>
45 
46 #include <read.h>
47 #include <parse.h>
48 #include <program.h>
49 #include <vm.h>
50 
51 static void bc_program_addFunc(BcProgram *p, BcFunc *f, BcId *id_ptr);
52 
53 static inline void bc_program_setVecs(BcProgram *p, BcFunc *f) {
54 	p->consts = &f->consts;
55 	if (BC_IS_BC) p->strs = &f->strs;
56 }
57 
58 static inline void bc_program_type_num(BcResult *r, BcNum *n) {
59 
60 #if BC_ENABLED
61 	assert(r->t != BC_RESULT_VOID);
62 #endif // BC_ENABLED
63 
64 	if (BC_ERR(!BC_PROG_NUM(r, n))) bc_vm_err(BC_ERR_EXEC_TYPE);
65 }
66 
67 #if BC_ENABLED
68 static void bc_program_type_match(BcResult *r, BcType t) {
69 
70 #if DC_ENABLED
71 	assert(BC_IS_DC || BC_NO_ERR(r->t != BC_RESULT_STR));
72 #endif // DC_ENABLED
73 
74 	if (BC_ERR((r->t != BC_RESULT_ARRAY) != (!t)))
75 		bc_vm_err(BC_ERR_EXEC_TYPE);
76 }
77 #endif // BC_ENABLED
78 
79 static size_t bc_program_index(const char *restrict code, size_t *restrict bgn)
80 {
81 	uchar amt = (uchar) code[(*bgn)++], i = 0;
82 	size_t res = 0;
83 
84 	for (; i < amt; ++i, ++(*bgn)) {
85 		size_t temp = ((size_t) ((int) (uchar) code[*bgn]) & UCHAR_MAX);
86 		res |= (temp << (i * CHAR_BIT));
87 	}
88 
89 	return res;
90 }
91 
92 #if BC_ENABLED
93 static void bc_program_prepGlobals(BcProgram *p) {
94 
95 	size_t i;
96 
97 	for (i = 0; i < BC_PROG_GLOBALS_LEN; ++i)
98 		bc_vec_push(p->globals_v + i, p->globals + i);
99 
100 #if BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
101 	bc_rand_push(&p->rng);
102 #endif // BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
103 }
104 
105 static void bc_program_popGlobals(BcProgram *p, bool reset) {
106 
107 	size_t i;
108 
109 	for (i = 0; i < BC_PROG_GLOBALS_LEN; ++i) {
110 		BcVec *v = p->globals_v + i;
111 		bc_vec_npop(v, reset ? v->len - 1 : 1);
112 		p->globals[i] = BC_PROG_GLOBAL(v);
113 	}
114 
115 #if BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
116 	bc_rand_pop(&p->rng, reset);
117 #endif // BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
118 }
119 #endif // BC_ENABLED
120 
121 static void bc_program_pushBigdig(BcProgram *p, BcBigDig dig, BcResultType type)
122 {
123 	BcResult res;
124 
125 	res.t = type;
126 
127 	BC_SIG_LOCK;
128 
129 	bc_num_createFromBigdig(&res.d.n, dig);
130 	bc_vec_push(&p->results, &res);
131 
132 	BC_SIG_UNLOCK;
133 }
134 
135 #if BC_ENABLED
136 static BcVec* bc_program_dereference(const BcProgram *p, BcVec *vec) {
137 
138 	BcVec *v;
139 	size_t vidx, nidx, i = 0;
140 
141 	assert(vec->size == sizeof(uchar));
142 
143 	vidx = bc_program_index(vec->v, &i);
144 	nidx = bc_program_index(vec->v, &i);
145 
146 	v = bc_vec_item(bc_vec_item(&p->arrs, vidx), nidx);
147 
148 	assert(v->size != sizeof(uchar));
149 
150 	return v;
151 }
152 #endif // BC_ENABLED
153 
154 size_t bc_program_search(BcProgram *p, const char *id, bool var) {
155 
156 	BcVec *v, *map;
157 	size_t i;
158 	BcResultData data;
159 
160 	v = var ? &p->vars : &p->arrs;
161 	map = var ? &p->var_map : &p->arr_map;
162 
163 	BC_SIG_LOCK;
164 
165 	if (bc_map_insert(map, id, v->len, &i)) {
166 		bc_array_init(&data.v, var);
167 		bc_vec_push(v, &data.v);
168 	}
169 
170 	BC_SIG_UNLOCK;
171 
172 	return ((BcId*) bc_vec_item(map, i))->idx;
173 }
174 
175 static inline BcVec* bc_program_vec(const BcProgram *p, size_t idx, BcType type)
176 {
177 	const BcVec *v = (type == BC_TYPE_VAR) ? &p->vars : &p->arrs;
178 	return bc_vec_item(v, idx);
179 }
180 
181 static BcNum* bc_program_num(BcProgram *p, BcResult *r) {
182 
183 	BcNum *n;
184 
185 	switch (r->t) {
186 
187 		case BC_RESULT_STR:
188 		case BC_RESULT_TEMP:
189 		case BC_RESULT_IBASE:
190 		case BC_RESULT_SCALE:
191 		case BC_RESULT_OBASE:
192 #if BC_ENABLE_EXTRA_MATH
193 		case BC_RESULT_SEED:
194 #endif // BC_ENABLE_EXTRA_MATH
195 		{
196 			n = &r->d.n;
197 			break;
198 		}
199 
200 		case BC_RESULT_VAR:
201 #if BC_ENABLED
202 		case BC_RESULT_ARRAY:
203 #endif // BC_ENABLED
204 		case BC_RESULT_ARRAY_ELEM:
205 		{
206 			BcVec *v;
207 			BcType type = (r->t == BC_RESULT_VAR) ? BC_TYPE_VAR : BC_TYPE_ARRAY;
208 
209 			v = bc_program_vec(p, r->d.loc.loc, type);
210 
211 			if (r->t == BC_RESULT_ARRAY_ELEM) {
212 
213 				size_t idx = r->d.loc.idx;
214 
215 				v = bc_vec_top(v);
216 
217 #if BC_ENABLED
218 				if (v->size == sizeof(uchar)) v = bc_program_dereference(p, v);
219 #endif // BC_ENABLED
220 
221 				assert(v->size == sizeof(BcNum));
222 
223 				if (v->len <= idx) {
224 					BC_SIG_LOCK;
225 					bc_array_expand(v, bc_vm_growSize(idx, 1));
226 					BC_SIG_UNLOCK;
227 				}
228 
229 				n = bc_vec_item(v, idx);
230 			}
231 			else n = bc_vec_top(v);
232 
233 			break;
234 		}
235 
236 		case BC_RESULT_ZERO:
237 		{
238 			n = &p->zero;
239 			break;
240 		}
241 
242 		case BC_RESULT_ONE:
243 		{
244 			n = &p->one;
245 			break;
246 		}
247 
248 #if BC_ENABLED
249 		case BC_RESULT_VOID:
250 #ifndef NDEBUG
251 		{
252 			abort();
253 		}
254 #endif // NDEBUG
255 		// Fallthrough
256 		case BC_RESULT_LAST:
257 		{
258 			n = &p->last;
259 			break;
260 		}
261 #endif // BC_ENABLED
262 	}
263 
264 	return n;
265 }
266 
267 static void bc_program_operand(BcProgram *p, BcResult **r,
268                                BcNum **n, size_t idx)
269 {
270 	*r = bc_vec_item_rev(&p->results, idx);
271 
272 #if BC_ENABLED
273 	if (BC_ERR((*r)->t == BC_RESULT_VOID)) bc_vm_err(BC_ERR_EXEC_VOID_VAL);
274 #endif // BC_ENABLED
275 
276 	*n = bc_program_num(p, *r);
277 }
278 
279 static void bc_program_binPrep(BcProgram *p, BcResult **l, BcNum **ln,
280                                BcResult **r, BcNum **rn, size_t idx)
281 {
282 	BcResultType lt;
283 
284 	assert(p != NULL && l != NULL && ln != NULL && r != NULL && rn != NULL);
285 
286 #ifndef BC_PROG_NO_STACK_CHECK
287 	if (BC_IS_DC) {
288 		if (BC_ERR(!BC_PROG_STACK(&p->results, idx + 2)))
289 			bc_vm_err(BC_ERR_EXEC_STACK);
290 	}
291 #endif // BC_PROG_NO_STACK_CHECK
292 
293 	assert(BC_PROG_STACK(&p->results, idx + 2));
294 
295 	bc_program_operand(p, l, ln, idx + 1);
296 	bc_program_operand(p, r, rn, idx);
297 
298 	lt = (*l)->t;
299 
300 #if BC_ENABLED
301 	assert(lt != BC_RESULT_VOID && (*r)->t != BC_RESULT_VOID);
302 #endif // BC_ENABLED
303 
304 	// We run this again under these conditions in case any vector has been
305 	// reallocated out from under the BcNums or arrays we had.
306 	if (lt == (*r)->t && (lt == BC_RESULT_VAR || lt == BC_RESULT_ARRAY_ELEM))
307 		*ln = bc_program_num(p, *l);
308 
309 	if (BC_ERR(lt == BC_RESULT_STR)) bc_vm_err(BC_ERR_EXEC_TYPE);
310 }
311 
312 static void bc_program_binOpPrep(BcProgram *p, BcResult **l, BcNum **ln,
313                                  BcResult **r, BcNum **rn, size_t idx)
314 {
315 	bc_program_binPrep(p, l, ln, r, rn, idx);
316 	bc_program_type_num(*l, *ln);
317 	bc_program_type_num(*r, *rn);
318 }
319 
320 static void bc_program_assignPrep(BcProgram *p, BcResult **l, BcNum **ln,
321                                   BcResult **r, BcNum **rn)
322 {
323 	BcResultType lt, min;
324 
325 	min = BC_RESULT_TEMP - ((unsigned int) (BC_IS_BC));
326 
327 	bc_program_binPrep(p, l, ln, r, rn, 0);
328 
329 	lt = (*l)->t;
330 
331 	if (BC_ERR(lt >= min && lt <= BC_RESULT_ONE))
332 		bc_vm_err(BC_ERR_EXEC_TYPE);
333 
334 #if DC_ENABLED
335 	if(BC_IS_DC) {
336 
337 		bool good = (((*r)->t == BC_RESULT_STR || BC_PROG_STR(*rn)) &&
338 		             lt <= BC_RESULT_ARRAY_ELEM);
339 
340 		if (!good) bc_program_type_num(*r, *rn);
341 	}
342 #else
343 	assert((*r)->t != BC_RESULT_STR);
344 #endif // DC_ENABLED
345 }
346 
347 static void bc_program_prep(BcProgram *p, BcResult **r, BcNum **n, size_t idx) {
348 
349 	assert(p != NULL && r != NULL && n != NULL);
350 
351 #ifndef BC_PROG_NO_STACK_CHECK
352 	if (BC_IS_DC) {
353 		if (BC_ERR(!BC_PROG_STACK(&p->results, idx + 1)))
354 			bc_vm_err(BC_ERR_EXEC_STACK);
355 	}
356 #endif // BC_PROG_NO_STACK_CHECK
357 
358 	assert(BC_PROG_STACK(&p->results, idx + 1));
359 
360 	bc_program_operand(p, r, n, idx);
361 
362 #if DC_ENABLED
363 	assert((*r)->t != BC_RESULT_VAR || !BC_PROG_STR(*n));
364 #endif // DC_ENABLED
365 
366 	bc_program_type_num(*r, *n);
367 }
368 
369 static BcResult* bc_program_prepResult(BcProgram *p) {
370 
371 	BcResult res;
372 
373 	bc_result_clear(&res);
374 	bc_vec_push(&p->results, &res);
375 
376 	return bc_vec_top(&p->results);
377 }
378 
379 static void bc_program_const(BcProgram *p, const char *code, size_t *bgn) {
380 
381 	BcResult *r = bc_program_prepResult(p);
382 	BcConst *c = bc_vec_item(p->consts, bc_program_index(code, bgn));
383 	BcBigDig base = BC_PROG_IBASE(p);
384 
385 	if (c->base != base) {
386 
387 		if (c->num.num == NULL) {
388 			BC_SIG_LOCK;
389 			bc_num_init(&c->num, BC_NUM_RDX(strlen(c->val)));
390 			BC_SIG_UNLOCK;
391 		}
392 
393 		// bc_num_parse() should only do operations that cannot fail.
394 		bc_num_parse(&c->num, c->val, base);
395 
396 		c->base = base;
397 	}
398 
399 	BC_SIG_LOCK;
400 
401 	bc_num_createCopy(&r->d.n, &c->num);
402 
403 	BC_SIG_UNLOCK;
404 }
405 
406 static void bc_program_op(BcProgram *p, uchar inst) {
407 
408 	BcResult *opd1, *opd2, *res;
409 	BcNum *n1, *n2;
410 	size_t idx = inst - BC_INST_POWER;
411 
412 	res = bc_program_prepResult(p);
413 
414 	bc_program_binOpPrep(p, &opd1, &n1, &opd2, &n2, 1);
415 
416 	BC_SIG_LOCK;
417 
418 	bc_num_init(&res->d.n, bc_program_opReqs[idx](n1, n2, BC_PROG_SCALE(p)));
419 
420 	BC_SIG_UNLOCK;
421 
422 	assert(BC_NUM_RDX_VALID(n1));
423 	assert(BC_NUM_RDX_VALID(n2));
424 
425 	bc_program_ops[idx](n1, n2, &res->d.n, BC_PROG_SCALE(p));
426 
427 	bc_program_retire(p, 1, 2);
428 }
429 
430 static void bc_program_read(BcProgram *p) {
431 
432 	BcStatus s;
433 	BcParse parse;
434 	BcVec buf;
435 	BcInstPtr ip;
436 	size_t i;
437 	const char* file;
438 	BcFunc *f = bc_vec_item(&p->fns, BC_PROG_READ);
439 
440 	for (i = 0; i < p->stack.len; ++i) {
441 		BcInstPtr *ip_ptr = bc_vec_item(&p->stack, i);
442 		if (ip_ptr->func == BC_PROG_READ)
443 			bc_vm_err(BC_ERR_EXEC_REC_READ);
444 	}
445 
446 	BC_SIG_LOCK;
447 
448 	file = vm.file;
449 	bc_parse_init(&parse, p, BC_PROG_READ);
450 	bc_vec_init(&buf, sizeof(char), NULL);
451 
452 	BC_SETJMP_LOCKED(exec_err);
453 
454 	BC_SIG_UNLOCK;
455 
456 	bc_lex_file(&parse.l, bc_program_stdin_name);
457 	bc_vec_popAll(&f->code);
458 
459 	s = bc_read_line(&buf, BC_IS_BC ? "read> " : "?> ");
460 	if (s == BC_STATUS_EOF) bc_vm_err(BC_ERR_EXEC_READ_EXPR);
461 
462 	bc_parse_text(&parse, buf.v);
463 	vm.expr(&parse, BC_PARSE_NOREAD | BC_PARSE_NEEDVAL);
464 
465 	if (BC_ERR(parse.l.t != BC_LEX_NLINE && parse.l.t != BC_LEX_EOF))
466 		bc_vm_err(BC_ERR_EXEC_READ_EXPR);
467 
468 #if BC_ENABLED
469 	if (BC_G) bc_program_prepGlobals(p);
470 #endif // BC_ENABLED
471 
472 	ip.func = BC_PROG_READ;
473 	ip.idx = 0;
474 	ip.len = p->results.len;
475 
476 	// Update this pointer, just in case.
477 	f = bc_vec_item(&p->fns, BC_PROG_READ);
478 
479 	bc_vec_pushByte(&f->code, vm.read_ret);
480 	bc_vec_push(&p->stack, &ip);
481 
482 #if DC_ENABLED
483 	if (BC_IS_DC) {
484 		size_t temp = 0;
485 		bc_vec_push(&p->tail_calls, &temp);
486 	}
487 #endif // DC_ENABLED
488 
489 exec_err:
490 	BC_SIG_MAYLOCK;
491 	bc_parse_free(&parse);
492 	bc_vec_free(&buf);
493 	vm.file = file;
494 	BC_LONGJMP_CONT;
495 }
496 
497 #if BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
498 static void bc_program_rand(BcProgram *p) {
499 	BcRand rand = bc_rand_int(&p->rng);
500 	bc_program_pushBigdig(p, (BcBigDig) rand, BC_RESULT_TEMP);
501 #ifndef NDEBUG
502 	{
503 		BcResult *r = bc_vec_top(&p->results);
504 		assert(BC_NUM_RDX_VALID_NP(r->d.n));
505 	}
506 #endif // NDEBUG
507 }
508 #endif // BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
509 
510 static void bc_program_printChars(const char *str) {
511 
512 	const char *nl;
513 	size_t len = vm.nchars + strlen(str);
514 
515 	bc_file_puts(&vm.fout, str);
516 	nl = strrchr(str, '\n');
517 
518 	if (nl != NULL) len = strlen(nl + 1);
519 
520 	vm.nchars = len > UINT16_MAX ? UINT16_MAX : (uint16_t) len;
521 }
522 
523 static void bc_program_printString(const char *restrict str) {
524 
525 	size_t i, len = strlen(str);
526 
527 #if DC_ENABLED
528 	if (!len && BC_IS_DC) {
529 		bc_vm_putchar('\0');
530 		return;
531 	}
532 #endif // DC_ENABLED
533 
534 	for (i = 0; i < len; ++i) {
535 
536 		int c = str[i];
537 
538 		if (c == '\\' && i != len - 1) {
539 
540 			const char *ptr;
541 
542 			c = str[++i];
543 			ptr = strchr(bc_program_esc_chars, c);
544 
545 			if (ptr != NULL) {
546 				if (c == 'n') vm.nchars = UINT16_MAX;
547 				c = bc_program_esc_seqs[(size_t) (ptr - bc_program_esc_chars)];
548 			}
549 			else {
550 				// Just print the backslash. The following
551 				// character will be printed later.
552 				bc_vm_putchar('\\');
553 			}
554 		}
555 
556 		bc_vm_putchar(c);
557 	}
558 }
559 
560 static void bc_program_print(BcProgram *p, uchar inst, size_t idx) {
561 
562 	BcResult *r;
563 	char *str;
564 	BcNum *n;
565 	bool pop = (inst != BC_INST_PRINT);
566 
567 	assert(p != NULL);
568 
569 #ifndef BC_PROG_NO_STACK_CHECK
570 	if (BC_IS_DC) {
571 		if (BC_ERR(!BC_PROG_STACK(&p->results, idx + 1)))
572 			bc_vm_err(BC_ERR_EXEC_STACK);
573 	}
574 #endif // BC_PROG_NO_STACK_CHECK
575 
576 	assert(BC_PROG_STACK(&p->results, idx + 1));
577 
578 	r = bc_vec_item_rev(&p->results, idx);
579 
580 #if BC_ENABLED
581 	if (r->t == BC_RESULT_VOID) {
582 		if (BC_ERR(pop)) bc_vm_err(BC_ERR_EXEC_VOID_VAL);
583 		bc_vec_pop(&p->results);
584 		return;
585 	}
586 #endif // BC_ENABLED
587 
588 	n = bc_program_num(p, r);
589 
590 	if (BC_PROG_NUM(r, n)) {
591 		assert(inst != BC_INST_PRINT_STR);
592 		bc_num_print(n, BC_PROG_OBASE(p), !pop);
593 #if BC_ENABLED
594 		if (BC_IS_BC) bc_num_copy(&p->last, n);
595 #endif // BC_ENABLED
596 	}
597 	else {
598 
599 		size_t i = (r->t == BC_RESULT_STR) ? r->d.loc.loc : n->scale;
600 
601 		bc_file_flush(&vm.fout);
602 		str = *((char**) bc_vec_item(p->strs, i));
603 
604 		if (inst == BC_INST_PRINT_STR) bc_program_printChars(str);
605 		else {
606 			bc_program_printString(str);
607 			if (inst == BC_INST_PRINT) bc_vm_putchar('\n');
608 		}
609 	}
610 
611 	if (BC_IS_BC || pop) bc_vec_pop(&p->results);
612 }
613 
614 void bc_program_negate(BcResult *r, BcNum *n) {
615 	bc_num_copy(&r->d.n, n);
616 	if (BC_NUM_NONZERO(&r->d.n)) BC_NUM_NEG_TGL_NP(r->d.n);
617 }
618 
619 void bc_program_not(BcResult *r, BcNum *n) {
620 	if (!bc_num_cmpZero(n)) bc_num_one(&r->d.n);
621 }
622 
623 #if BC_ENABLE_EXTRA_MATH
624 void bc_program_trunc(BcResult *r, BcNum *n) {
625 	bc_num_copy(&r->d.n, n);
626 	bc_num_truncate(&r->d.n, n->scale);
627 }
628 #endif // BC_ENABLE_EXTRA_MATH
629 
630 static void bc_program_unary(BcProgram *p, uchar inst) {
631 
632 	BcResult *res, *ptr;
633 	BcNum *num;
634 
635 	res = bc_program_prepResult(p);
636 
637 	bc_program_prep(p, &ptr, &num, 1);
638 
639 	BC_SIG_LOCK;
640 
641 	bc_num_init(&res->d.n, num->len);
642 
643 	BC_SIG_UNLOCK;
644 
645 	bc_program_unarys[inst - BC_INST_NEG](res, num);
646 	bc_program_retire(p, 1, 1);
647 }
648 
649 static void bc_program_logical(BcProgram *p, uchar inst) {
650 
651 	BcResult *opd1, *opd2, *res;
652 	BcNum *n1, *n2;
653 	bool cond = 0;
654 	ssize_t cmp;
655 
656 	res = bc_program_prepResult(p);
657 
658 	bc_program_binOpPrep(p, &opd1, &n1, &opd2, &n2, 1);
659 
660 	if (inst == BC_INST_BOOL_AND)
661 		cond = (bc_num_cmpZero(n1) && bc_num_cmpZero(n2));
662 	else if (inst == BC_INST_BOOL_OR)
663 		cond = (bc_num_cmpZero(n1) || bc_num_cmpZero(n2));
664 	else {
665 
666 		cmp = bc_num_cmp(n1, n2);
667 
668 		switch (inst) {
669 
670 			case BC_INST_REL_EQ:
671 			{
672 				cond = (cmp == 0);
673 				break;
674 			}
675 
676 			case BC_INST_REL_LE:
677 			{
678 				cond = (cmp <= 0);
679 				break;
680 			}
681 
682 			case BC_INST_REL_GE:
683 			{
684 				cond = (cmp >= 0);
685 				break;
686 			}
687 
688 			case BC_INST_REL_NE:
689 			{
690 				cond = (cmp != 0);
691 				break;
692 			}
693 
694 			case BC_INST_REL_LT:
695 			{
696 				cond = (cmp < 0);
697 				break;
698 			}
699 
700 			case BC_INST_REL_GT:
701 			{
702 				cond = (cmp > 0);
703 				break;
704 			}
705 #ifndef NDEBUG
706 			default:
707 			{
708 				abort();
709 			}
710 #endif // NDEBUG
711 		}
712 	}
713 
714 	BC_SIG_LOCK;
715 
716 	bc_num_init(&res->d.n, BC_NUM_DEF_SIZE);
717 
718 	BC_SIG_UNLOCK;
719 
720 	if (cond) bc_num_one(&res->d.n);
721 
722 	bc_program_retire(p, 1, 2);
723 }
724 
725 #if DC_ENABLED
726 static void bc_program_assignStr(BcProgram *p, size_t idx,
727                                  BcVec *v, bool push)
728 {
729 	BcNum n2;
730 
731 	bc_num_clear(&n2);
732 	n2.scale = idx;
733 
734 	assert(BC_PROG_STACK(&p->results, 1 + !push));
735 
736 	if (!push) bc_vec_pop(v);
737 
738 	bc_vec_npop(&p->results, 1 + !push);
739 	bc_vec_push(v, &n2);
740 }
741 #endif // DC_ENABLED
742 
743 static void bc_program_copyToVar(BcProgram *p, size_t idx,
744                                  BcType t, bool last)
745 {
746 	BcResult *ptr = NULL, r;
747 	BcVec *vec;
748 	BcNum *n = NULL;
749 	bool var = (t == BC_TYPE_VAR);
750 
751 #if DC_ENABLED
752 	if (BC_IS_DC) {
753 
754 		if (BC_ERR(!BC_PROG_STACK(&p->results, 1)))
755 			bc_vm_err(BC_ERR_EXEC_STACK);
756 
757 		assert(BC_PROG_STACK(&p->results, 1));
758 
759 		bc_program_operand(p, &ptr, &n, 0);
760 	}
761 #endif
762 
763 #if BC_ENABLED
764 	if (BC_IS_BC)
765 	{
766 		ptr = bc_vec_top(&p->results);
767 
768 		bc_program_type_match(ptr, t);
769 
770 		if (last) n = bc_program_num(p, ptr);
771 		else if (var)
772 			n = bc_vec_item_rev(bc_program_vec(p, ptr->d.loc.loc, t), 1);
773 	}
774 #endif // BC_ENABLED
775 
776 	vec = bc_program_vec(p, idx, t);
777 
778 #if DC_ENABLED
779 	if (BC_IS_DC && (ptr->t == BC_RESULT_STR || BC_PROG_STR(n))) {
780 
781 		size_t str_idx = ptr->t == BC_RESULT_STR ? ptr->d.loc.loc : n->scale;
782 
783 		if (BC_ERR(!var)) bc_vm_err(BC_ERR_EXEC_TYPE);
784 
785 		bc_program_assignStr(p, str_idx, vec, true);
786 
787 		return;
788 	}
789 #endif // DC_ENABLED
790 
791 	BC_SIG_LOCK;
792 
793 	if (var) bc_num_createCopy(&r.d.n, n);
794 	else {
795 
796 		BcVec *v = (BcVec*) n, *rv = &r.d.v;
797 #if BC_ENABLED
798 		BcVec *parent;
799 		bool ref, ref_size;
800 
801 		parent = bc_program_vec(p, ptr->d.loc.loc, t);
802 		assert(parent != NULL);
803 
804 		if (!last) v = bc_vec_item_rev(parent, !last);
805 		assert(v != NULL);
806 
807 		ref = (v->size == sizeof(BcNum) && t == BC_TYPE_REF);
808 		ref_size = (v->size == sizeof(uchar));
809 
810 		if (ref || (ref_size && t == BC_TYPE_REF)) {
811 
812 			bc_vec_init(rv, sizeof(uchar), NULL);
813 
814 			if (ref) {
815 
816 				assert(parent->len >= (size_t) (!last + 1));
817 
818 				// Make sure the pointer was not invalidated.
819 				vec = bc_program_vec(p, idx, t);
820 
821 				bc_vec_pushIndex(rv, ptr->d.loc.loc);
822 				bc_vec_pushIndex(rv, parent->len - !last - 1);
823 			}
824 			// If we get here, we are copying a ref to a ref.
825 			else bc_vec_npush(rv, v->len * sizeof(uchar), v->v);
826 
827 			// We need to return early.
828 			bc_vec_push(vec, &r.d);
829 			bc_vec_pop(&p->results);
830 
831 			BC_SIG_UNLOCK;
832 			return;
833 		}
834 		else if (ref_size && t != BC_TYPE_REF) v = bc_program_dereference(p, v);
835 #endif // BC_ENABLED
836 
837 		bc_array_init(rv, true);
838 		bc_array_copy(rv, v);
839 	}
840 
841 	bc_vec_push(vec, &r.d);
842 	bc_vec_pop(&p->results);
843 
844 	BC_SIG_UNLOCK;
845 }
846 
847 static void bc_program_assign(BcProgram *p, uchar inst) {
848 
849 	BcResult *left, *right, res;
850 	BcNum *l, *r;
851 	bool ob, sc, use_val = BC_INST_USE_VAL(inst);
852 
853 	bc_program_assignPrep(p, &left, &l, &right, &r);
854 
855 #if DC_ENABLED
856 	assert(left->t != BC_RESULT_STR);
857 
858 	if (right->t == BC_RESULT_STR || BC_PROG_STR(r)) {
859 
860 		size_t idx = right->t == BC_RESULT_STR ? right->d.loc.loc : r->scale;
861 
862 		if (left->t == BC_RESULT_ARRAY_ELEM) {
863 			BC_SIG_LOCK;
864 			bc_num_free(l);
865 			bc_num_clear(l);
866 			l->scale = idx;
867 			bc_vec_npop(&p->results, 2);
868 			BC_SIG_UNLOCK;
869 		}
870 		else {
871 			BcVec *v = bc_program_vec(p, left->d.loc.loc, BC_TYPE_VAR);
872 			bc_program_assignStr(p, idx, v, false);
873 		}
874 
875 		return;
876 	}
877 #endif // DC_ENABLED
878 
879 	if (BC_INST_IS_ASSIGN(inst)) bc_num_copy(l, r);
880 #if BC_ENABLED
881 	else {
882 
883 		BcBigDig scale = BC_PROG_SCALE(p);
884 
885 		if (!use_val)
886 			inst -= (BC_INST_ASSIGN_POWER_NO_VAL - BC_INST_ASSIGN_POWER);
887 
888 		assert(BC_NUM_RDX_VALID(l));
889 		assert(BC_NUM_RDX_VALID(r));
890 
891 		bc_program_ops[inst - BC_INST_ASSIGN_POWER](l, r, l, scale);
892 	}
893 #endif // BC_ENABLED
894 
895 	ob = (left->t == BC_RESULT_OBASE);
896 	sc = (left->t == BC_RESULT_SCALE);
897 
898 	if (ob || sc || left->t == BC_RESULT_IBASE) {
899 
900 		BcVec *v;
901 		BcBigDig *ptr, *ptr_t, val, max, min;
902 		BcErr e;
903 
904 		bc_num_bigdig(l, &val);
905 		e = left->t - BC_RESULT_IBASE + BC_ERR_EXEC_IBASE;
906 
907 		if (sc) {
908 			min = 0;
909 			max = vm.maxes[BC_PROG_GLOBALS_SCALE];
910 			v = p->globals_v + BC_PROG_GLOBALS_SCALE;
911 			ptr_t = p->globals + BC_PROG_GLOBALS_SCALE;
912 		}
913 		else {
914 			min = BC_NUM_MIN_BASE;
915 			if (BC_ENABLE_EXTRA_MATH && ob && (BC_IS_DC || !BC_IS_POSIX))
916 				min = 0;
917 			max = vm.maxes[ob + BC_PROG_GLOBALS_IBASE];
918 			v = p->globals_v + BC_PROG_GLOBALS_IBASE + ob;
919 			ptr_t = p->globals + BC_PROG_GLOBALS_IBASE + ob;
920 		}
921 
922 		if (BC_ERR(val > max || val < min)) bc_vm_verr(e, min, max);
923 
924 		ptr = bc_vec_top(v);
925 		*ptr = val;
926 		*ptr_t = val;
927 	}
928 #if BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
929 	else if (left->t == BC_RESULT_SEED) bc_num_rng(l, &p->rng);
930 #endif // BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
931 
932 	BC_SIG_LOCK;
933 
934 	if (use_val) {
935 		bc_num_createCopy(&res.d.n, l);
936 		res.t = BC_RESULT_TEMP;
937 		bc_vec_npop(&p->results, 2);
938 		bc_vec_push(&p->results, &res);
939 	}
940 	else bc_vec_npop(&p->results, 2);
941 
942 	BC_SIG_UNLOCK;
943 }
944 
945 static void bc_program_pushVar(BcProgram *p, const char *restrict code,
946                                size_t *restrict bgn, bool pop, bool copy)
947 {
948 	BcResult r;
949 	size_t idx = bc_program_index(code, bgn);
950 
951 	r.t = BC_RESULT_VAR;
952 	r.d.loc.loc = idx;
953 
954 #if DC_ENABLED
955 	if (BC_IS_DC && (pop || copy)) {
956 
957 		BcVec *v = bc_program_vec(p, idx, BC_TYPE_VAR);
958 		BcNum *num = bc_vec_top(v);
959 
960 		if (BC_ERR(!BC_PROG_STACK(v, 2 - copy))) bc_vm_err(BC_ERR_EXEC_STACK);
961 
962 		assert(BC_PROG_STACK(v, 2 - copy));
963 
964 		if (!BC_PROG_STR(num)) {
965 
966 			BC_SIG_LOCK;
967 
968 			r.t = BC_RESULT_TEMP;
969 			bc_num_createCopy(&r.d.n, num);
970 
971 			if (!copy) bc_vec_pop(v);
972 
973 			bc_vec_push(&p->results, &r);
974 
975 			BC_SIG_UNLOCK;
976 
977 			return;
978 		}
979 		else {
980 			r.d.loc.loc = num->scale;
981 			r.t = BC_RESULT_STR;
982 		}
983 
984 		if (!copy) bc_vec_pop(v);
985 	}
986 #endif // DC_ENABLED
987 
988 	bc_vec_push(&p->results, &r);
989 }
990 
991 static void bc_program_pushArray(BcProgram *p, const char *restrict code,
992                                  size_t *restrict bgn, uchar inst)
993 {
994 	BcResult r, *operand;
995 	BcNum *num;
996 	BcBigDig temp;
997 
998 	r.d.loc.loc = bc_program_index(code, bgn);
999 
1000 #if BC_ENABLED
1001 	if (inst == BC_INST_ARRAY) {
1002 		r.t = BC_RESULT_ARRAY;
1003 		bc_vec_push(&p->results, &r);
1004 		return;
1005 	}
1006 #endif // BC_ENABLED
1007 
1008 	bc_program_prep(p, &operand, &num, 0);
1009 	bc_num_bigdig(num, &temp);
1010 
1011 	r.t = BC_RESULT_ARRAY_ELEM;
1012 	r.d.loc.idx = (size_t) temp;
1013 
1014 	BC_SIG_LOCK;
1015 
1016 	bc_vec_pop(&p->results);
1017 	bc_vec_push(&p->results, &r);
1018 
1019 	BC_SIG_UNLOCK;
1020 }
1021 
1022 #if BC_ENABLED
1023 static void bc_program_incdec(BcProgram *p, uchar inst) {
1024 
1025 	BcResult *ptr, res, copy;
1026 	BcNum *num;
1027 	uchar inst2;
1028 
1029 	bc_program_prep(p, &ptr, &num, 0);
1030 
1031 	BC_SIG_LOCK;
1032 
1033 	copy.t = BC_RESULT_TEMP;
1034 	bc_num_createCopy(&copy.d.n, num);
1035 
1036 	BC_SETJMP_LOCKED(exit);
1037 
1038 	BC_SIG_UNLOCK;
1039 
1040 	res.t = BC_RESULT_ONE;
1041 	inst2 = BC_INST_ASSIGN_PLUS + (inst & 0x01);
1042 
1043 	bc_vec_push(&p->results, &res);
1044 	bc_program_assign(p, inst2);
1045 
1046 	BC_SIG_LOCK;
1047 
1048 	bc_vec_pop(&p->results);
1049 	bc_vec_push(&p->results, &copy);
1050 
1051 	BC_UNSETJMP;
1052 
1053 	BC_SIG_UNLOCK;
1054 
1055 	return;
1056 
1057 exit:
1058 	BC_SIG_MAYLOCK;
1059 	bc_num_free(&copy.d.n);
1060 	BC_LONGJMP_CONT;
1061 }
1062 
1063 static void bc_program_call(BcProgram *p, const char *restrict code,
1064                             size_t *restrict idx)
1065 {
1066 	BcInstPtr ip;
1067 	size_t i, nparams = bc_program_index(code, idx);
1068 	BcFunc *f;
1069 	BcVec *v;
1070 	BcLoc *a;
1071 	BcResultData param;
1072 	BcResult *arg;
1073 
1074 	ip.idx = 0;
1075 	ip.func = bc_program_index(code, idx);
1076 	f = bc_vec_item(&p->fns, ip.func);
1077 
1078 	if (BC_ERR(!f->code.len)) bc_vm_verr(BC_ERR_EXEC_UNDEF_FUNC, f->name);
1079 	if (BC_ERR(nparams != f->nparams))
1080 		bc_vm_verr(BC_ERR_EXEC_PARAMS, f->nparams, nparams);
1081 	ip.len = p->results.len - nparams;
1082 
1083 	assert(BC_PROG_STACK(&p->results, nparams));
1084 
1085 	if (BC_G) bc_program_prepGlobals(p);
1086 
1087 	for (i = 0; i < nparams; ++i) {
1088 
1089 		size_t j;
1090 		bool last = true;
1091 
1092 		arg = bc_vec_top(&p->results);
1093 		if (BC_ERR(arg->t == BC_RESULT_VOID)) bc_vm_err(BC_ERR_EXEC_VOID_VAL);
1094 
1095 		a = bc_vec_item(&f->autos, nparams - 1 - i);
1096 
1097 		// If I have already pushed to a var, I need to make sure I
1098 		// get the previous version, not the already pushed one.
1099 		if (arg->t == BC_RESULT_VAR || arg->t == BC_RESULT_ARRAY) {
1100 			for (j = 0; j < i && last; ++j) {
1101 				BcLoc *loc = bc_vec_item(&f->autos, nparams - 1 - j);
1102 				last = (arg->d.loc.loc != loc->loc ||
1103 				        (!loc->idx) != (arg->t == BC_RESULT_VAR));
1104 			}
1105 		}
1106 
1107 		bc_program_copyToVar(p, a->loc, (BcType) a->idx, last);
1108 	}
1109 
1110 	BC_SIG_LOCK;
1111 
1112 	for (; i < f->autos.len; ++i) {
1113 
1114 		a = bc_vec_item(&f->autos, i);
1115 		v = bc_program_vec(p, a->loc, (BcType) a->idx);
1116 
1117 		if (a->idx == BC_TYPE_VAR) {
1118 			bc_num_init(&param.n, BC_NUM_DEF_SIZE);
1119 			bc_vec_push(v, &param.n);
1120 		}
1121 		else {
1122 			assert(a->idx == BC_TYPE_ARRAY);
1123 			bc_array_init(&param.v, true);
1124 			bc_vec_push(v, &param.v);
1125 		}
1126 	}
1127 
1128 	bc_vec_push(&p->stack, &ip);
1129 
1130 	BC_SIG_UNLOCK;
1131 }
1132 
1133 static void bc_program_return(BcProgram *p, uchar inst) {
1134 
1135 	BcResult *res;
1136 	BcFunc *f;
1137 	BcInstPtr *ip = bc_vec_top(&p->stack);
1138 	size_t i, nops = p->results.len - ip->len;
1139 
1140 	assert(BC_PROG_STACK(&p->stack, 2));
1141 	assert(BC_PROG_STACK(&p->results, ip->len + (inst == BC_INST_RET)));
1142 
1143 	f = bc_vec_item(&p->fns, ip->func);
1144 	res = bc_program_prepResult(p);
1145 
1146 	if (inst == BC_INST_RET) {
1147 
1148 		BcNum *num;
1149 		BcResult *operand;
1150 
1151 		bc_program_operand(p, &operand, &num, 1);
1152 
1153 		BC_SIG_LOCK;
1154 
1155 		bc_num_createCopy(&res->d.n, num);
1156 	}
1157 	else if (inst == BC_INST_RET_VOID) res->t = BC_RESULT_VOID;
1158 	else {
1159 		BC_SIG_LOCK;
1160 		bc_num_init(&res->d.n, BC_NUM_DEF_SIZE);
1161 	}
1162 
1163 	BC_SIG_MAYUNLOCK;
1164 
1165 	// We need to pop arguments as well, so this takes that into account.
1166 	for (i = 0; i < f->autos.len; ++i) {
1167 
1168 		BcLoc *a = bc_vec_item(&f->autos, i);
1169 		BcVec *v = bc_program_vec(p, a->loc, (BcType) a->idx);
1170 
1171 		bc_vec_pop(v);
1172 	}
1173 
1174 	bc_program_retire(p, 1, nops);
1175 
1176 	if (BC_G) bc_program_popGlobals(p, false);
1177 
1178 	bc_vec_pop(&p->stack);
1179 }
1180 #endif // BC_ENABLED
1181 
1182 static void bc_program_builtin(BcProgram *p, uchar inst) {
1183 
1184 	BcResult *opd, *res;
1185 	BcNum *num;
1186 	bool len = (inst == BC_INST_LENGTH);
1187 
1188 #if BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1189 	assert(inst >= BC_INST_LENGTH && inst <= BC_INST_IRAND);
1190 #else // BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1191 	assert(inst >= BC_INST_LENGTH && inst <= BC_INST_ABS);
1192 #endif // BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1193 
1194 #ifndef BC_PROG_NO_STACK_CHECK
1195 	if (BC_IS_DC) {
1196 		if (BC_ERR(!BC_PROG_STACK(&p->results, 1)))
1197 			bc_vm_err(BC_ERR_EXEC_STACK);
1198 	}
1199 #endif // BC_PROG_NO_STACK_CHECK
1200 
1201 	assert(BC_PROG_STACK(&p->results, 1));
1202 
1203 	res = bc_program_prepResult(p);
1204 
1205 	bc_program_operand(p, &opd, &num, 1);
1206 
1207 	assert(num != NULL);
1208 
1209 #if DC_ENABLED
1210 	if (!len && inst != BC_INST_SCALE_FUNC) bc_program_type_num(opd, num);
1211 #endif // DC_ENABLED
1212 
1213 	if (inst == BC_INST_SQRT) bc_num_sqrt(num, &res->d.n, BC_PROG_SCALE(p));
1214 	else if (inst == BC_INST_ABS) {
1215 
1216 		BC_SIG_LOCK;
1217 
1218 		bc_num_createCopy(&res->d.n, num);
1219 
1220 		BC_SIG_UNLOCK;
1221 
1222 		BC_NUM_NEG_CLR_NP(res->d.n);
1223 	}
1224 #if BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1225 	else if (inst == BC_INST_IRAND) {
1226 
1227 		BC_SIG_LOCK;
1228 
1229 		bc_num_init(&res->d.n, num->len - num->rdx);
1230 
1231 		BC_SIG_UNLOCK;
1232 
1233 		bc_num_irand(num, &res->d.n, &p->rng);
1234 	}
1235 #endif // BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1236 	else {
1237 
1238 		BcBigDig val = 0;
1239 
1240 		if (len) {
1241 #if BC_ENABLED
1242 			if (BC_IS_BC && opd->t == BC_RESULT_ARRAY) {
1243 
1244 				BcVec *v = (BcVec*) num;
1245 
1246 				if (v->size == sizeof(uchar)) v = bc_program_dereference(p, v);
1247 
1248 				assert(v->size == sizeof(BcNum));
1249 
1250 				val = (BcBigDig) v->len;
1251 			}
1252 			else
1253 #endif // BC_ENABLED
1254 			{
1255 #if DC_ENABLED
1256 				if (!BC_PROG_NUM(opd, num)) {
1257 
1258 					size_t idx;
1259 					char *str;
1260 
1261 					idx = opd->t == BC_RESULT_STR ? opd->d.loc.loc : num->scale;
1262 					str = *((char**) bc_vec_item(p->strs, idx));
1263 					val = (BcBigDig) strlen(str);
1264 				}
1265 				else
1266 #endif // DC_ENABLED
1267 				{
1268 					val = (BcBigDig) bc_num_len(num);
1269 				}
1270 			}
1271 		}
1272 		else if (BC_IS_BC || BC_PROG_NUM(opd, num))
1273 			val = (BcBigDig) bc_num_scale(num);
1274 
1275 		BC_SIG_LOCK;
1276 
1277 		bc_num_createFromBigdig(&res->d.n, val);
1278 
1279 		BC_SIG_UNLOCK;
1280 	}
1281 
1282 	bc_program_retire(p, 1, 1);
1283 }
1284 
1285 #if DC_ENABLED
1286 static void bc_program_divmod(BcProgram *p) {
1287 
1288 	BcResult *opd1, *opd2, *res, *res2;
1289 	BcNum *n1, *n2;
1290 	size_t req;
1291 
1292 	bc_vec_grow(&p->results, 2);
1293 
1294 	// We don't need to update the pointer because
1295 	// the capacity is enough due to the line above.
1296 	res2 = bc_program_prepResult(p);
1297 	res = bc_program_prepResult(p);
1298 
1299 	bc_program_binOpPrep(p, &opd1, &n1, &opd2, &n2, 2);
1300 
1301 	req = bc_num_mulReq(n1, n2, BC_PROG_SCALE(p));
1302 
1303 	BC_SIG_LOCK;
1304 
1305 	bc_num_init(&res->d.n, req);
1306 	bc_num_init(&res2->d.n, req);
1307 
1308 	BC_SIG_UNLOCK;
1309 
1310 	bc_num_divmod(n1, n2, &res2->d.n, &res->d.n, BC_PROG_SCALE(p));
1311 
1312 	bc_program_retire(p, 2, 2);
1313 }
1314 
1315 static void bc_program_modexp(BcProgram *p) {
1316 
1317 	BcResult *r1, *r2, *r3, *res;
1318 	BcNum *n1, *n2, *n3;
1319 
1320 	if (BC_ERR(!BC_PROG_STACK(&p->results, 3))) bc_vm_err(BC_ERR_EXEC_STACK);
1321 
1322 	assert(BC_PROG_STACK(&p->results, 3));
1323 
1324 	res = bc_program_prepResult(p);
1325 
1326 	bc_program_operand(p, &r1, &n1, 3);
1327 	bc_program_type_num(r1, n1);
1328 
1329 	bc_program_binOpPrep(p, &r2, &n2, &r3, &n3, 1);
1330 
1331 	// Make sure that the values have their pointers updated, if necessary.
1332 	// Only array elements are possible.
1333 	if (r1->t == BC_RESULT_ARRAY_ELEM && (r1->t == r2->t || r1->t == r3->t))
1334 		n1 = bc_program_num(p, r1);
1335 
1336 	BC_SIG_LOCK;
1337 
1338 	bc_num_init(&res->d.n, n3->len);
1339 
1340 	BC_SIG_UNLOCK;
1341 
1342 	bc_num_modexp(n1, n2, n3, &res->d.n);
1343 
1344 	bc_program_retire(p, 1, 3);
1345 }
1346 
1347 static void bc_program_stackLen(BcProgram *p) {
1348 	bc_program_pushBigdig(p, (BcBigDig) p->results.len, BC_RESULT_TEMP);
1349 }
1350 
1351 static uchar bc_program_asciifyNum(BcProgram *p, BcNum *n) {
1352 
1353 	BcNum num;
1354 	BcBigDig val = 0;
1355 
1356 	bc_num_clear(&num);
1357 
1358 	BC_SETJMP(num_err);
1359 
1360 	BC_SIG_LOCK;
1361 
1362 	bc_num_createCopy(&num, n);
1363 
1364 	BC_SIG_UNLOCK;
1365 
1366 	bc_num_truncate(&num, num.scale);
1367 	BC_NUM_NEG_CLR_NP(num);
1368 
1369 	// This is guaranteed to not have a divide by 0
1370 	// because strmb is equal to UCHAR_MAX + 1.
1371 	bc_num_mod(&num, &p->strmb, &num, 0);
1372 
1373 	// This is also guaranteed to not error because num is in the range
1374 	// [0, UCHAR_MAX], which is definitely in range for a BcBigDig. And
1375 	// it is not negative.
1376 	bc_num_bigdig2(&num, &val);
1377 
1378 num_err:
1379 	BC_SIG_MAYLOCK;
1380 	bc_num_free(&num);
1381 	BC_LONGJMP_CONT;
1382 	return (uchar) val;
1383 }
1384 
1385 static void bc_program_asciify(BcProgram *p) {
1386 
1387 	BcResult *r, res;
1388 	BcNum *n;
1389 	char str[2], *str2;
1390 	uchar c;
1391 	size_t idx;
1392 
1393 	if (BC_ERR(!BC_PROG_STACK(&p->results, 1))) bc_vm_err(BC_ERR_EXEC_STACK);
1394 
1395 	assert(BC_PROG_STACK(&p->results, 1));
1396 
1397 	bc_program_operand(p, &r, &n, 0);
1398 
1399 	assert(n != NULL);
1400 
1401 	assert(p->strs->len + BC_PROG_REQ_FUNCS == p->fns.len);
1402 
1403 	if (BC_PROG_NUM(r, n)) c = bc_program_asciifyNum(p, n);
1404 	else {
1405 		size_t index = r->t == BC_RESULT_STR ? r->d.loc.loc : n->scale;
1406 		str2 = *((char**) bc_vec_item(p->strs, index));
1407 		c = (uchar) str2[0];
1408 	}
1409 
1410 	str[0] = (char) c;
1411 	str[1] = '\0';
1412 
1413 	BC_SIG_LOCK;
1414 
1415 	idx = bc_program_insertFunc(p, str) - BC_PROG_REQ_FUNCS;
1416 
1417 	BC_SIG_UNLOCK;
1418 
1419 	res.t = BC_RESULT_STR;
1420 	res.d.loc.loc = idx;
1421 	bc_vec_pop(&p->results);
1422 	bc_vec_push(&p->results, &res);
1423 }
1424 
1425 static void bc_program_printStream(BcProgram *p) {
1426 
1427 	BcResult *r;
1428 	BcNum *n;
1429 
1430 	if (BC_ERR(!BC_PROG_STACK(&p->results, 1))) bc_vm_err(BC_ERR_EXEC_STACK);
1431 
1432 	assert(BC_PROG_STACK(&p->results, 1));
1433 
1434 	bc_program_operand(p, &r, &n, 0);
1435 
1436 	assert(n != NULL);
1437 
1438 	if (BC_PROG_NUM(r, n)) bc_num_stream(n, p->strm);
1439 	else {
1440 		size_t idx = (r->t == BC_RESULT_STR) ? r->d.loc.loc : n->scale;
1441 		bc_program_printChars(*((char**) bc_vec_item(p->strs, idx)));
1442 	}
1443 }
1444 
1445 static void bc_program_nquit(BcProgram *p, uchar inst) {
1446 
1447 	BcResult *opnd;
1448 	BcNum *num;
1449 	BcBigDig val;
1450 	size_t i;
1451 
1452 	assert(p->stack.len == p->tail_calls.len);
1453 
1454 	if (inst == BC_INST_QUIT) val = 2;
1455 	else {
1456 
1457 		bc_program_prep(p, &opnd, &num, 0);
1458 		bc_num_bigdig(num, &val);
1459 
1460 		bc_vec_pop(&p->results);
1461 	}
1462 
1463 	for (i = 0; val && i < p->tail_calls.len; ++i) {
1464 		size_t calls = *((size_t*) bc_vec_item_rev(&p->tail_calls, i)) + 1;
1465 		if (calls >= val) val = 0;
1466 		else val -= calls;
1467 	}
1468 
1469 	if (i == p->stack.len) {
1470 		vm.status = BC_STATUS_QUIT;
1471 		BC_VM_JMP;
1472 	}
1473 	else {
1474 		bc_vec_npop(&p->stack, i);
1475 		bc_vec_npop(&p->tail_calls, i);
1476 	}
1477 }
1478 
1479 static void bc_program_execStr(BcProgram *p, const char *restrict code,
1480                                    size_t *restrict bgn, bool cond, size_t len)
1481 {
1482 	BcResult *r;
1483 	char *str;
1484 	BcFunc *f;
1485 	BcParse prs;
1486 	BcInstPtr ip;
1487 	size_t fidx, sidx;
1488 	BcNum *n;
1489 
1490 	assert(p->stack.len == p->tail_calls.len);
1491 
1492 	if (BC_ERR(!BC_PROG_STACK(&p->results, 1))) bc_vm_err(BC_ERR_EXEC_STACK);
1493 
1494 	assert(BC_PROG_STACK(&p->results, 1));
1495 
1496 	bc_program_operand(p, &r, &n, 0);
1497 
1498 	if (cond) {
1499 
1500 		bool exec;
1501 		size_t idx, then_idx, else_idx;
1502 
1503 		then_idx = bc_program_index(code, bgn);
1504 		else_idx = bc_program_index(code, bgn);
1505 
1506 		exec = (r->d.n.len != 0);
1507 
1508 		idx = exec ? then_idx : else_idx;
1509 
1510 		BC_SIG_LOCK;
1511 		BC_SETJMP_LOCKED(exit);
1512 
1513 		if (exec || (else_idx != SIZE_MAX))
1514 			n = bc_vec_top(bc_program_vec(p, idx, BC_TYPE_VAR));
1515 		else goto exit;
1516 
1517 		if (BC_ERR(!BC_PROG_STR(n))) bc_vm_err(BC_ERR_EXEC_TYPE);
1518 
1519 		BC_UNSETJMP;
1520 		BC_SIG_UNLOCK;
1521 
1522 		sidx = n->scale;
1523 	}
1524 	else {
1525 
1526 		// In non-conditional situations, only the top of stack can be executed,
1527 		// and in those cases, variables are not allowed to be "on the stack";
1528 		// they are only put on the stack to be assigned to.
1529 		assert(r->t != BC_RESULT_VAR);
1530 
1531 		if (r->t == BC_RESULT_STR) sidx = r->d.loc.loc;
1532 		else return;
1533 	}
1534 
1535 	fidx = sidx + BC_PROG_REQ_FUNCS;
1536 	str = *((char**) bc_vec_item(p->strs, sidx));
1537 	f = bc_vec_item(&p->fns, fidx);
1538 
1539 	if (!f->code.len) {
1540 
1541 		BC_SIG_LOCK;
1542 
1543 		bc_parse_init(&prs, p, fidx);
1544 		bc_lex_file(&prs.l, vm.file);
1545 
1546 		BC_SETJMP_LOCKED(err);
1547 
1548 		BC_SIG_UNLOCK;
1549 
1550 		bc_parse_text(&prs, str);
1551 		vm.expr(&prs, BC_PARSE_NOCALL);
1552 
1553 		BC_SIG_LOCK;
1554 
1555 		BC_UNSETJMP;
1556 
1557 		// We can just assert this here because
1558 		// dc should parse everything until EOF.
1559 		assert(prs.l.t == BC_LEX_EOF);
1560 
1561 		bc_parse_free(&prs);
1562 
1563 		BC_SIG_UNLOCK;
1564 	}
1565 
1566 	ip.idx = 0;
1567 	ip.len = p->results.len;
1568 	ip.func = fidx;
1569 
1570 	bc_vec_pop(&p->results);
1571 
1572 	// Tail call.
1573 	if (p->stack.len > 1 && *bgn == len - 1 && code[*bgn] == BC_INST_POP_EXEC) {
1574 		size_t *call_ptr = bc_vec_top(&p->tail_calls);
1575 		*call_ptr += 1;
1576 		bc_vec_pop(&p->stack);
1577 	}
1578 	else bc_vec_push(&p->tail_calls, &ip.idx);
1579 
1580 	bc_vec_push(&p->stack, &ip);
1581 
1582 	return;
1583 
1584 err:
1585 	BC_SIG_MAYLOCK;
1586 	bc_parse_free(&prs);
1587 	f = bc_vec_item(&p->fns, fidx);
1588 	bc_vec_popAll(&f->code);
1589 exit:
1590 	bc_vec_pop(&p->results);
1591 	BC_LONGJMP_CONT;
1592 }
1593 
1594 static void bc_program_printStack(BcProgram *p) {
1595 
1596 	size_t idx;
1597 
1598 	for (idx = 0; idx < p->results.len; ++idx)
1599 		bc_program_print(p, BC_INST_PRINT, idx);
1600 }
1601 #endif // DC_ENABLED
1602 
1603 static void bc_program_pushGlobal(BcProgram *p, uchar inst) {
1604 
1605 	BcResultType t;
1606 
1607 	assert(inst >= BC_INST_IBASE && inst <= BC_INST_SCALE);
1608 
1609 	t = inst - BC_INST_IBASE + BC_RESULT_IBASE;
1610 	bc_program_pushBigdig(p, p->globals[inst - BC_INST_IBASE], t);
1611 }
1612 
1613 #if BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1614 static void bc_program_pushSeed(BcProgram *p) {
1615 
1616 	BcResult *res;
1617 
1618 	res = bc_program_prepResult(p);
1619 	res->t = BC_RESULT_SEED;
1620 
1621 	BC_SIG_LOCK;
1622 
1623 	bc_num_init(&res->d.n, 2 * BC_RAND_NUM_SIZE);
1624 
1625 	BC_SIG_UNLOCK;
1626 
1627 	bc_num_createFromRNG(&res->d.n, &p->rng);
1628 }
1629 #endif // BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1630 
1631 static void bc_program_addFunc(BcProgram *p, BcFunc *f, BcId *id_ptr) {
1632 
1633 	BcInstPtr *ip;
1634 
1635 	BC_SIG_ASSERT_LOCKED;
1636 
1637 	bc_func_init(f, id_ptr->name);
1638 	bc_vec_push(&p->fns, f);
1639 
1640 	// This is to make sure pointers are updated if the array was moved.
1641 	if (p->stack.len) {
1642 		ip = bc_vec_top(&p->stack);
1643 		bc_program_setVecs(p, (BcFunc*) bc_vec_item(&p->fns, ip->func));
1644 	}
1645 }
1646 
1647 size_t bc_program_insertFunc(BcProgram *p, const char *name) {
1648 
1649 	BcId *id_ptr;
1650 	BcFunc f;
1651 	bool new;
1652 	size_t idx;
1653 
1654 	BC_SIG_ASSERT_LOCKED;
1655 
1656 	assert(p != NULL && name != NULL);
1657 
1658 	new = bc_map_insert(&p->fn_map, name, p->fns.len, &idx);
1659 	id_ptr = (BcId*) bc_vec_item(&p->fn_map, idx);
1660 	idx = id_ptr->idx;
1661 
1662 	if (!new) {
1663 		if (BC_IS_BC) {
1664 			BcFunc *func = bc_vec_item(&p->fns, idx);
1665 			bc_func_reset(func);
1666 		}
1667 	}
1668 	else {
1669 
1670 		bc_program_addFunc(p, &f, id_ptr);
1671 
1672 #if DC_ENABLED
1673 		if (BC_IS_DC && idx >= BC_PROG_REQ_FUNCS) {
1674 			bc_vec_push(p->strs, &id_ptr->name);
1675 			assert(p->strs->len == p->fns.len - BC_PROG_REQ_FUNCS);
1676 		}
1677 #endif // DC_ENABLED
1678 	}
1679 
1680 	return idx;
1681 }
1682 
1683 #ifndef NDEBUG
1684 void bc_program_free(BcProgram *p) {
1685 
1686 	size_t i;
1687 
1688 	BC_SIG_ASSERT_LOCKED;
1689 
1690 	assert(p != NULL);
1691 
1692 	for (i = 0; i < BC_PROG_GLOBALS_LEN; ++i) bc_vec_free(p->globals_v + i);
1693 
1694 	bc_vec_free(&p->fns);
1695 	bc_vec_free(&p->fn_map);
1696 	bc_vec_free(&p->vars);
1697 	bc_vec_free(&p->var_map);
1698 	bc_vec_free(&p->arrs);
1699 	bc_vec_free(&p->arr_map);
1700 	bc_vec_free(&p->results);
1701 	bc_vec_free(&p->stack);
1702 
1703 #if BC_ENABLED
1704 	if (BC_IS_BC) bc_num_free(&p->last);
1705 #endif // BC_ENABLED
1706 
1707 #if BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1708 	bc_rand_free(&p->rng);
1709 #endif // BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1710 
1711 #if DC_ENABLED
1712 	if (BC_IS_DC) {
1713 		bc_vec_free(&p->tail_calls);
1714 		bc_vec_free(&p->strs_v);
1715 	}
1716 #endif // DC_ENABLED
1717 }
1718 #endif // NDEBUG
1719 
1720 void bc_program_init(BcProgram *p) {
1721 
1722 	BcInstPtr ip;
1723 	size_t i;
1724 
1725 	BC_SIG_ASSERT_LOCKED;
1726 
1727 	assert(p != NULL);
1728 
1729 	memset(p, 0, sizeof(BcProgram));
1730 	memset(&ip, 0, sizeof(BcInstPtr));
1731 
1732 	for (i = 0; i < BC_PROG_GLOBALS_LEN; ++i) {
1733 		BcBigDig val = i == BC_PROG_GLOBALS_SCALE ? 0 : BC_BASE;
1734 		bc_vec_init(p->globals_v + i, sizeof(BcBigDig), NULL);
1735 		bc_vec_push(p->globals_v + i, &val);
1736 		p->globals[i] = val;
1737 	}
1738 
1739 #if DC_ENABLED
1740 	if (BC_IS_DC) {
1741 
1742 		bc_vec_init(&p->strs_v, sizeof(char*), bc_string_free);
1743 		p->strs = &p->strs_v;
1744 
1745 		bc_vec_init(&p->tail_calls, sizeof(size_t), NULL);
1746 		i = 0;
1747 		bc_vec_push(&p->tail_calls, &i);
1748 
1749 		p->strm = UCHAR_MAX + 1;
1750 		bc_num_setup(&p->strmb, p->strmb_num, BC_NUM_BIGDIG_LOG10);
1751 		bc_num_bigdig2num(&p->strmb, p->strm);
1752 	}
1753 #endif // DC_ENABLED
1754 
1755 #if BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1756 	srand((unsigned int) time(NULL));
1757 	bc_rand_init(&p->rng);
1758 #endif // BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1759 
1760 	bc_num_setup(&p->zero, p->zero_num, BC_PROG_ONE_CAP);
1761 
1762 	bc_num_setup(&p->one, p->one_num, BC_PROG_ONE_CAP);
1763 	bc_num_one(&p->one);
1764 
1765 #if BC_ENABLED
1766 	if (BC_IS_BC) bc_num_init(&p->last, BC_NUM_DEF_SIZE);
1767 #endif // BC_ENABLED
1768 
1769 	bc_vec_init(&p->fns, sizeof(BcFunc), bc_func_free);
1770 	bc_map_init(&p->fn_map);
1771 	bc_program_insertFunc(p, bc_func_main);
1772 	bc_program_insertFunc(p, bc_func_read);
1773 
1774 	bc_vec_init(&p->vars, sizeof(BcVec), bc_vec_free);
1775 	bc_map_init(&p->var_map);
1776 
1777 	bc_vec_init(&p->arrs, sizeof(BcVec), bc_vec_free);
1778 	bc_map_init(&p->arr_map);
1779 
1780 	bc_vec_init(&p->results, sizeof(BcResult), bc_result_free);
1781 	bc_vec_init(&p->stack, sizeof(BcInstPtr), NULL);
1782 	bc_vec_push(&p->stack, &ip);
1783 
1784 	bc_program_setVecs(p, (BcFunc*) bc_vec_item(&p->fns, BC_PROG_MAIN));
1785 
1786 	assert(p->consts != NULL && p->strs != NULL);
1787 }
1788 
1789 void bc_program_reset(BcProgram *p) {
1790 
1791 	BcFunc *f;
1792 	BcInstPtr *ip;
1793 
1794 	BC_SIG_ASSERT_LOCKED;
1795 
1796 	bc_vec_npop(&p->stack, p->stack.len - 1);
1797 	bc_vec_popAll(&p->results);
1798 
1799 #if BC_ENABLED
1800 	if (BC_G) bc_program_popGlobals(p, true);
1801 #endif // BC_ENABLED
1802 
1803 	f = bc_vec_item(&p->fns, BC_PROG_MAIN);
1804 	bc_vec_npop(&f->code, f->code.len);
1805 	ip = bc_vec_top(&p->stack);
1806 	bc_program_setVecs(p, f);
1807 	memset(ip, 0, sizeof(BcInstPtr));
1808 
1809 	if (vm.sig) {
1810 		bc_file_write(&vm.fout, bc_program_ready_msg, bc_program_ready_msg_len);
1811 		bc_file_flush(&vm.fout);
1812 		vm.sig = 0;
1813 	}
1814 }
1815 
1816 void bc_program_exec(BcProgram *p) {
1817 
1818 	size_t idx;
1819 	BcResult r, *ptr;
1820 	BcInstPtr *ip = bc_vec_top(&p->stack);
1821 	BcFunc *func = (BcFunc*) bc_vec_item(&p->fns, ip->func);
1822 	char *code = func->code.v;
1823 	bool cond = false;
1824 #if BC_ENABLED
1825 	BcNum *num;
1826 #endif // BC_ENABLED
1827 #ifndef NDEBUG
1828 	size_t jmp_bufs_len;
1829 #endif // NDEBUG
1830 
1831 #ifndef NDEBUG
1832 	jmp_bufs_len = vm.jmp_bufs.len;
1833 #endif // NDEBUG
1834 
1835 	bc_program_setVecs(p, func);
1836 
1837 	while (ip->idx < func->code.len) {
1838 
1839 		BC_SIG_ASSERT_NOT_LOCKED;
1840 
1841 		uchar inst = (uchar) code[(ip->idx)++];
1842 
1843 		switch (inst) {
1844 
1845 #if BC_ENABLED
1846 			case BC_INST_JUMP_ZERO:
1847 			{
1848 				bc_program_prep(p, &ptr, &num, 0);
1849 				cond = !bc_num_cmpZero(num);
1850 				bc_vec_pop(&p->results);
1851 			}
1852 			// Fallthrough.
1853 			BC_FALLTHROUGH
1854 
1855 			case BC_INST_JUMP:
1856 			{
1857 				idx = bc_program_index(code, &ip->idx);
1858 
1859 				if (inst == BC_INST_JUMP || cond) {
1860 
1861 					size_t *addr = bc_vec_item(&func->labels, idx);
1862 
1863 					assert(*addr != SIZE_MAX);
1864 
1865 					ip->idx = *addr;
1866 				}
1867 
1868 				break;
1869 			}
1870 
1871 			case BC_INST_CALL:
1872 			{
1873 				assert(BC_IS_BC);
1874 
1875 				bc_program_call(p, code, &ip->idx);
1876 
1877 				ip = bc_vec_top(&p->stack);
1878 				func = bc_vec_item(&p->fns, ip->func);
1879 				code = func->code.v;
1880 
1881 				bc_program_setVecs(p, func);
1882 
1883 				break;
1884 			}
1885 
1886 			case BC_INST_INC:
1887 			case BC_INST_DEC:
1888 			{
1889 				bc_program_incdec(p, inst);
1890 				break;
1891 			}
1892 
1893 			case BC_INST_HALT:
1894 			{
1895 				vm.status = BC_STATUS_QUIT;
1896 				BC_VM_JMP;
1897 				break;
1898 			}
1899 
1900 			case BC_INST_RET:
1901 			case BC_INST_RET0:
1902 			case BC_INST_RET_VOID:
1903 			{
1904 				bc_program_return(p, inst);
1905 
1906 				ip = bc_vec_top(&p->stack);
1907 				func = bc_vec_item(&p->fns, ip->func);
1908 				code = func->code.v;
1909 
1910 				bc_program_setVecs(p, func);
1911 
1912 				break;
1913 			}
1914 #endif // BC_ENABLED
1915 
1916 			case BC_INST_BOOL_OR:
1917 			case BC_INST_BOOL_AND:
1918 			case BC_INST_REL_EQ:
1919 			case BC_INST_REL_LE:
1920 			case BC_INST_REL_GE:
1921 			case BC_INST_REL_NE:
1922 			case BC_INST_REL_LT:
1923 			case BC_INST_REL_GT:
1924 			{
1925 				bc_program_logical(p, inst);
1926 				break;
1927 			}
1928 
1929 			case BC_INST_READ:
1930 			{
1931 				bc_program_read(p);
1932 
1933 				ip = bc_vec_top(&p->stack);
1934 				func = bc_vec_item(&p->fns, ip->func);
1935 				code = func->code.v;
1936 
1937 				bc_program_setVecs(p, func);
1938 
1939 				break;
1940 			}
1941 
1942 #if BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1943 			case BC_INST_RAND:
1944 			{
1945 				bc_program_rand(p);
1946 				break;
1947 			}
1948 #endif // BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1949 
1950 			case BC_INST_MAXIBASE:
1951 			case BC_INST_MAXOBASE:
1952 			case BC_INST_MAXSCALE:
1953 #if BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1954 			case BC_INST_MAXRAND:
1955 #endif // BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1956 			{
1957 				BcBigDig dig = vm.maxes[inst - BC_INST_MAXIBASE];
1958 				bc_program_pushBigdig(p, dig, BC_RESULT_TEMP);
1959 				break;
1960 			}
1961 
1962 			case BC_INST_VAR:
1963 			{
1964 				bc_program_pushVar(p, code, &ip->idx, false, false);
1965 				break;
1966 			}
1967 
1968 			case BC_INST_ARRAY_ELEM:
1969 #if BC_ENABLED
1970 			case BC_INST_ARRAY:
1971 #endif // BC_ENABLED
1972 			{
1973 				bc_program_pushArray(p, code, &ip->idx, inst);
1974 				break;
1975 			}
1976 
1977 			case BC_INST_IBASE:
1978 			case BC_INST_SCALE:
1979 			case BC_INST_OBASE:
1980 			{
1981 				bc_program_pushGlobal(p, inst);
1982 				break;
1983 			}
1984 
1985 #if BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1986 			case BC_INST_SEED:
1987 			{
1988 				bc_program_pushSeed(p);
1989 				break;
1990 			}
1991 #endif // BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1992 
1993 			case BC_INST_LENGTH:
1994 			case BC_INST_SCALE_FUNC:
1995 			case BC_INST_SQRT:
1996 			case BC_INST_ABS:
1997 #if BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
1998 			case BC_INST_IRAND:
1999 #endif // BC_ENABLE_EXTRA_MATH && BC_ENABLE_RAND
2000 			{
2001 				bc_program_builtin(p, inst);
2002 				break;
2003 			}
2004 
2005 			case BC_INST_NUM:
2006 			{
2007 				bc_program_const(p, code, &ip->idx);
2008 				break;
2009 			}
2010 
2011 			case BC_INST_ZERO:
2012 			case BC_INST_ONE:
2013 #if BC_ENABLED
2014 			case BC_INST_LAST:
2015 #endif // BC_ENABLED
2016 			{
2017 				r.t = BC_RESULT_ZERO + (inst - BC_INST_ZERO);
2018 				bc_vec_push(&p->results, &r);
2019 				break;
2020 			}
2021 
2022 			case BC_INST_PRINT:
2023 			case BC_INST_PRINT_POP:
2024 			case BC_INST_PRINT_STR:
2025 			{
2026 				bc_program_print(p, inst, 0);
2027 				break;
2028 			}
2029 
2030 			case BC_INST_STR:
2031 			{
2032 				r.t = BC_RESULT_STR;
2033 				r.d.loc.loc = bc_program_index(code, &ip->idx);
2034 				bc_vec_push(&p->results, &r);
2035 				break;
2036 			}
2037 
2038 			case BC_INST_POWER:
2039 			case BC_INST_MULTIPLY:
2040 			case BC_INST_DIVIDE:
2041 			case BC_INST_MODULUS:
2042 			case BC_INST_PLUS:
2043 			case BC_INST_MINUS:
2044 #if BC_ENABLE_EXTRA_MATH
2045 			case BC_INST_PLACES:
2046 			case BC_INST_LSHIFT:
2047 			case BC_INST_RSHIFT:
2048 #endif // BC_ENABLE_EXTRA_MATH
2049 			{
2050 				bc_program_op(p, inst);
2051 				break;
2052 			}
2053 
2054 			case BC_INST_NEG:
2055 			case BC_INST_BOOL_NOT:
2056 #if BC_ENABLE_EXTRA_MATH
2057 			case BC_INST_TRUNC:
2058 #endif // BC_ENABLE_EXTRA_MATH
2059 			{
2060 				bc_program_unary(p, inst);
2061 				break;
2062 			}
2063 
2064 #if BC_ENABLED
2065 			case BC_INST_ASSIGN_POWER:
2066 			case BC_INST_ASSIGN_MULTIPLY:
2067 			case BC_INST_ASSIGN_DIVIDE:
2068 			case BC_INST_ASSIGN_MODULUS:
2069 			case BC_INST_ASSIGN_PLUS:
2070 			case BC_INST_ASSIGN_MINUS:
2071 #if BC_ENABLE_EXTRA_MATH
2072 			case BC_INST_ASSIGN_PLACES:
2073 			case BC_INST_ASSIGN_LSHIFT:
2074 			case BC_INST_ASSIGN_RSHIFT:
2075 #endif // BC_ENABLE_EXTRA_MATH
2076 			case BC_INST_ASSIGN:
2077 			case BC_INST_ASSIGN_POWER_NO_VAL:
2078 			case BC_INST_ASSIGN_MULTIPLY_NO_VAL:
2079 			case BC_INST_ASSIGN_DIVIDE_NO_VAL:
2080 			case BC_INST_ASSIGN_MODULUS_NO_VAL:
2081 			case BC_INST_ASSIGN_PLUS_NO_VAL:
2082 			case BC_INST_ASSIGN_MINUS_NO_VAL:
2083 #if BC_ENABLE_EXTRA_MATH
2084 			case BC_INST_ASSIGN_PLACES_NO_VAL:
2085 			case BC_INST_ASSIGN_LSHIFT_NO_VAL:
2086 			case BC_INST_ASSIGN_RSHIFT_NO_VAL:
2087 #endif // BC_ENABLE_EXTRA_MATH
2088 #endif // BC_ENABLED
2089 			case BC_INST_ASSIGN_NO_VAL:
2090 			{
2091 				bc_program_assign(p, inst);
2092 				break;
2093 			}
2094 
2095 			case BC_INST_POP:
2096 			{
2097 #ifndef BC_PROG_NO_STACK_CHECK
2098 				if (!BC_IS_BC) {
2099 					if (BC_ERR(!BC_PROG_STACK(&p->results, 1)))
2100 						bc_vm_err(BC_ERR_EXEC_STACK);
2101 				}
2102 #endif // BC_PROG_NO_STACK_CHECK
2103 
2104 				assert(BC_PROG_STACK(&p->results, 1));
2105 
2106 				bc_vec_pop(&p->results);
2107 				break;
2108 			}
2109 
2110 #if DC_ENABLED
2111 			case BC_INST_POP_EXEC:
2112 			{
2113 				assert(BC_PROG_STACK(&p->stack, 2));
2114 				bc_vec_pop(&p->stack);
2115 				bc_vec_pop(&p->tail_calls);
2116 				ip = bc_vec_top(&p->stack);
2117 				func = bc_vec_item(&p->fns, ip->func);
2118 				code = func->code.v;
2119 				bc_program_setVecs(p, func);
2120 				break;
2121 			}
2122 
2123 			case BC_INST_MODEXP:
2124 			{
2125 				bc_program_modexp(p);
2126 				break;
2127 			}
2128 
2129 			case BC_INST_DIVMOD:
2130 			{
2131 				bc_program_divmod(p);
2132 				break;
2133 			}
2134 
2135 			case BC_INST_EXECUTE:
2136 			case BC_INST_EXEC_COND:
2137 			{
2138 				cond = (inst == BC_INST_EXEC_COND);
2139 				bc_program_execStr(p, code, &ip->idx, cond, func->code.len);
2140 				ip = bc_vec_top(&p->stack);
2141 				func = bc_vec_item(&p->fns, ip->func);
2142 				code = func->code.v;
2143 				bc_program_setVecs(p, func);
2144 				break;
2145 			}
2146 
2147 			case BC_INST_PRINT_STACK:
2148 			{
2149 				bc_program_printStack(p);
2150 				break;
2151 			}
2152 
2153 			case BC_INST_CLEAR_STACK:
2154 			{
2155 				bc_vec_popAll(&p->results);
2156 				break;
2157 			}
2158 
2159 			case BC_INST_STACK_LEN:
2160 			{
2161 				bc_program_stackLen(p);
2162 				break;
2163 			}
2164 
2165 			case BC_INST_DUPLICATE:
2166 			{
2167 				if (BC_ERR(!BC_PROG_STACK(&p->results, 1)))
2168 					bc_vm_err(BC_ERR_EXEC_STACK);
2169 
2170 				assert(BC_PROG_STACK(&p->results, 1));
2171 
2172 				ptr = bc_vec_top(&p->results);
2173 
2174 				BC_SIG_LOCK;
2175 
2176 				bc_result_copy(&r, ptr);
2177 				bc_vec_push(&p->results, &r);
2178 
2179 				BC_SIG_UNLOCK;
2180 
2181 				break;
2182 			}
2183 
2184 			case BC_INST_SWAP:
2185 			{
2186 				BcResult *ptr2;
2187 
2188 				if (BC_ERR(!BC_PROG_STACK(&p->results, 2)))
2189 					bc_vm_err(BC_ERR_EXEC_STACK);
2190 
2191 				assert(BC_PROG_STACK(&p->results, 2));
2192 
2193 				ptr = bc_vec_item_rev(&p->results, 0);
2194 				ptr2 = bc_vec_item_rev(&p->results, 1);
2195 				memcpy(&r, ptr, sizeof(BcResult));
2196 				memcpy(ptr, ptr2, sizeof(BcResult));
2197 				memcpy(ptr2, &r, sizeof(BcResult));
2198 
2199 				break;
2200 			}
2201 
2202 			case BC_INST_ASCIIFY:
2203 			{
2204 				bc_program_asciify(p);
2205 				ip = bc_vec_top(&p->stack);
2206 				func = bc_vec_item(&p->fns, ip->func);
2207 				code = func->code.v;
2208 				bc_program_setVecs(p, func);
2209 				break;
2210 			}
2211 
2212 			case BC_INST_PRINT_STREAM:
2213 			{
2214 				bc_program_printStream(p);
2215 				break;
2216 			}
2217 
2218 			case BC_INST_LOAD:
2219 			case BC_INST_PUSH_VAR:
2220 			{
2221 				bool copy = (inst == BC_INST_LOAD);
2222 				bc_program_pushVar(p, code, &ip->idx, true, copy);
2223 				break;
2224 			}
2225 
2226 			case BC_INST_PUSH_TO_VAR:
2227 			{
2228 				idx = bc_program_index(code, &ip->idx);
2229 				bc_program_copyToVar(p, idx, BC_TYPE_VAR, true);
2230 				break;
2231 			}
2232 
2233 			case BC_INST_QUIT:
2234 			case BC_INST_NQUIT:
2235 			{
2236 				bc_program_nquit(p, inst);
2237 				ip = bc_vec_top(&p->stack);
2238 				func = bc_vec_item(&p->fns, ip->func);
2239 				code = func->code.v;
2240 				bc_program_setVecs(p, func);
2241 				break;
2242 			}
2243 #endif // DC_ENABLED
2244 #ifndef NDEBUG
2245 			default:
2246 			{
2247 				abort();
2248 			}
2249 #endif // NDEBUG
2250 		}
2251 
2252 #ifndef NDEBUG
2253 		// This is to allow me to use a debugger to see the last instruction,
2254 		// which will point to which function was the problem.
2255 		assert(jmp_bufs_len == vm.jmp_bufs.len);
2256 #endif // NDEBUG
2257 	}
2258 }
2259 
2260 #if BC_DEBUG_CODE
2261 #if BC_ENABLED && DC_ENABLED
2262 void bc_program_printStackDebug(BcProgram *p) {
2263 	bc_file_puts(&vm.fout, "-------------- Stack ----------\n");
2264 	bc_program_printStack(p);
2265 	bc_file_puts(&vm.fout, "-------------- Stack End ------\n");
2266 }
2267 
2268 static void bc_program_printIndex(const char *restrict code,
2269                                   size_t *restrict bgn)
2270 {
2271 	uchar byte, i, bytes = (uchar) code[(*bgn)++];
2272 	ulong val = 0;
2273 
2274 	for (byte = 1, i = 0; byte && i < bytes; ++i) {
2275 		byte = (uchar) code[(*bgn)++];
2276 		if (byte) val |= ((ulong) byte) << (CHAR_BIT * i);
2277 	}
2278 
2279 	bc_vm_printf(" (%lu) ", val);
2280 }
2281 
2282 static void bc_program_printStr(const BcProgram *p, const char *restrict code,
2283                          size_t *restrict bgn)
2284 {
2285 	size_t idx = bc_program_index(code, bgn);
2286 	char *s;
2287 
2288 	s = *((char**) bc_vec_item(p->strs, idx));
2289 
2290 	bc_vm_printf(" (\"%s\") ", s);
2291 }
2292 
2293 void bc_program_printInst(const BcProgram *p, const char *restrict code,
2294                           size_t *restrict bgn)
2295 {
2296 	uchar inst = (uchar) code[(*bgn)++];
2297 
2298 	bc_vm_printf("Inst[%zu]: %s [%lu]; ", *bgn - 1,
2299 	             bc_inst_names[inst], (unsigned long) inst);
2300 
2301 	if (inst == BC_INST_VAR || inst == BC_INST_ARRAY_ELEM ||
2302 	    inst == BC_INST_ARRAY)
2303 	{
2304 		bc_program_printIndex(code, bgn);
2305 	}
2306 	else if (inst == BC_INST_STR) bc_program_printStr(p, code, bgn);
2307 	else if (inst == BC_INST_NUM) {
2308 		size_t idx = bc_program_index(code, bgn);
2309 		BcConst *c = bc_vec_item(p->consts, idx);
2310 		bc_vm_printf("(%s)", c->val);
2311 	}
2312 	else if (inst == BC_INST_CALL ||
2313 	         (inst > BC_INST_STR && inst <= BC_INST_JUMP_ZERO))
2314 	{
2315 		bc_program_printIndex(code, bgn);
2316 		if (inst == BC_INST_CALL) bc_program_printIndex(code, bgn);
2317 	}
2318 
2319 	bc_vm_putchar('\n');
2320 }
2321 
2322 void bc_program_code(const BcProgram* p) {
2323 
2324 	BcFunc *f;
2325 	char *code;
2326 	BcInstPtr ip;
2327 	size_t i;
2328 
2329 	for (i = 0; i < p->fns.len; ++i) {
2330 
2331 		ip.idx = ip.len = 0;
2332 		ip.func = i;
2333 
2334 		f = bc_vec_item(&p->fns, ip.func);
2335 		code = f->code.v;
2336 
2337 		bc_vm_printf("func[%zu]:\n", ip.func);
2338 		while (ip.idx < f->code.len) bc_program_printInst(p, code, &ip.idx);
2339 		bc_file_puts(&vm.fout, "\n\n");
2340 	}
2341 }
2342 #endif // BC_ENABLED && DC_ENABLED
2343 #endif // BC_DEBUG_CODE
2344