2 * *****************************************************************************
4 * SPDX-License-Identifier: BSD-2-Clause
6 * Copyright (c) 2018-2021 Gavin D. Howard and contributors.
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions are met:
11 * * Redistributions of source code must retain the above copyright notice, this
12 * list of conditions and the following disclaimer.
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.
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.
30 * *****************************************************************************
32 * Code for processing command-line arguments.
53 * Adds @a str to the list of expressions to execute later.
54 * @param str The string to add to the list of expressions.
57 bc_args_exprs(const char* str)
60 if (vm.exprs.v == NULL) bc_vec_init(&vm.exprs, sizeof(uchar), BC_DTOR_NONE);
61 bc_vec_concat(&vm.exprs, str);
62 bc_vec_concat(&vm.exprs, "\n");
66 * Adds the contents of @a file to the list of expressions to execute later.
67 * @param file The name of the file whose contents should be added to the list
68 * of expressions to execute.
71 bc_args_file(const char* file)
79 buf = bc_read_file(file);
88 bc_args_builtin(const char* arg)
94 strvalid = bc_num_strValid(arg);
96 if (BC_ERR(!strvalid))
98 bc_verr(BC_ERR_FATAL_ARG, arg);
103 bc_num_parse(&n, arg, 10);
105 res = bc_num_bigdig(&n);
115 * Redefines a keyword, if it exists and is not a POSIX keyword. Otherwise, it
116 * throws a fatal error.
117 * @param keyword The keyword to redefine.
120 bc_args_redefine(const char* keyword)
124 BC_SIG_ASSERT_LOCKED;
126 for (i = 0; i < bc_lex_kws_len; ++i)
128 const BcLexKeyword* kw = bc_lex_kws + i;
130 if (!strcmp(keyword, kw->name))
132 if (BC_LEX_KW_POSIX(kw)) break;
134 vm.redefined_kws[i] = true;
140 bc_error(BC_ERR_FATAL_ARG, 0, keyword);
146 bc_args(int argc, char* argv[], bool exit_exprs, BcBigDig scale)
150 bool do_exit = false, version = false;
152 BcBigDig newscale = scale, ibase = BC_BASE, obase = BC_BASE;
153 #if BC_ENABLE_EXTRA_MATH
155 #endif // BC_ENABLE_EXTRA_MATH
157 BC_SIG_ASSERT_LOCKED;
159 bc_opt_init(&opts, argv);
161 // This loop should look familiar to anyone who has used getopt() or
162 // getopt_long() in C.
163 while ((c = bc_opt_parse(&opts, bc_args_lopt)) != -1)
169 // Barf if not allowed.
172 bc_verr(BC_ERR_FATAL_OPTION, "-e (--expression)");
175 // Add the expressions and set exit.
176 bc_args_exprs(opts.optarg);
177 vm.exit_exprs = (exit_exprs || vm.exit_exprs);
184 // Figure out if exiting on expressions is disabled.
185 if (!strcmp(opts.optarg, "-")) vm.no_exprs = true;
188 // Barf if not allowed.
191 bc_verr(BC_ERR_FATAL_OPTION, "-f (--file)");
194 // Add the expressions and set exit.
195 bc_args_file(opts.optarg);
196 vm.exit_exprs = (exit_exprs || vm.exit_exprs);
211 vm.flags |= BC_FLAG_I;
217 ibase = bc_args_builtin(opts.optarg);
223 vm.flags |= BC_FLAG_Z;
235 obase = bc_args_builtin(opts.optarg);
241 vm.flags &= ~(BC_FLAG_P);
247 vm.flags &= ~(BC_FLAG_R);
253 newscale = bc_args_builtin(opts.optarg);
257 #if BC_ENABLE_EXTRA_MATH
260 if (BC_ERR(!bc_num_strValid(opts.optarg)))
262 bc_verr(BC_ERR_FATAL_ARG, opts.optarg);
269 #endif // BC_ENABLE_EXTRA_MATH
275 vm.flags |= BC_FLAG_G;
282 vm.flags |= BC_FLAG_L;
289 vm.flags &= ~(BC_FLAG_Q);
295 bc_args_redefine(opts.optarg);
302 vm.flags |= BC_FLAG_S;
309 vm.flags |= BC_FLAG_W;
317 do_exit = version = true;
325 vm.flags |= DC_FLAG_X;
331 // We shouldn't get here because bc_opt_error()/bc_error() should
344 if (version) bc_vm_info(NULL);
347 vm.status = (sig_atomic_t) BC_STATUS_QUIT;
351 // We do not print the banner if expressions are used or dc is used.
352 if (!BC_IS_BC || vm.exprs.len > 1) vm.flags &= ~(BC_FLAG_Q);
354 // We need to make sure the files list is initialized. We don't want to
355 // initialize it if there are no files because it's just a waste of memory.
356 if (opts.optind < (size_t) argc && vm.files.v == NULL)
358 bc_vec_init(&vm.files, sizeof(char*), BC_DTOR_NONE);
361 // Add all the files to the vector.
362 for (i = opts.optind; i < (size_t) argc; ++i)
364 bc_vec_push(&vm.files, argv + i);
367 #if BC_ENABLE_EXTRA_MATH
372 bc_num_init(&n, strlen(seed));
376 bc_num_parse(&n, seed, BC_BASE);
378 bc_program_assignSeed(&vm.prog, &n);
384 #endif // BC_ENABLE_EXTRA_MATH
388 if (newscale != scale)
390 bc_program_assignBuiltin(&vm.prog, true, false, newscale);
393 if (obase != BC_BASE)
395 bc_program_assignBuiltin(&vm.prog, false, true, obase);
398 // This is last to avoid it affecting the value of the others.
399 if (ibase != BC_BASE)
401 bc_program_assignBuiltin(&vm.prog, false, false, ibase);