1252884aeSStefan Eßer /*
2252884aeSStefan Eßer * *****************************************************************************
3252884aeSStefan Eßer *
43aa99676SStefan Eßer * SPDX-License-Identifier: BSD-2-Clause
5252884aeSStefan Eßer *
6a970610aSStefan Eßer * Copyright (c) 2018-2024 Gavin D. Howard and contributors.
7252884aeSStefan Eßer *
8252884aeSStefan Eßer * Redistribution and use in source and binary forms, with or without
9252884aeSStefan Eßer * modification, are permitted provided that the following conditions are met:
10252884aeSStefan Eßer *
11252884aeSStefan Eßer * * Redistributions of source code must retain the above copyright notice, this
12252884aeSStefan Eßer * list of conditions and the following disclaimer.
13252884aeSStefan Eßer *
14252884aeSStefan Eßer * * Redistributions in binary form must reproduce the above copyright notice,
15252884aeSStefan Eßer * this list of conditions and the following disclaimer in the documentation
16252884aeSStefan Eßer * and/or other materials provided with the distribution.
17252884aeSStefan Eßer *
18252884aeSStefan Eßer * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
19252884aeSStefan Eßer * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20252884aeSStefan Eßer * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21252884aeSStefan Eßer * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
22252884aeSStefan Eßer * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
23252884aeSStefan Eßer * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
24252884aeSStefan Eßer * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
25252884aeSStefan Eßer * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
26252884aeSStefan Eßer * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
27252884aeSStefan Eßer * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28252884aeSStefan Eßer * POSSIBILITY OF SUCH DAMAGE.
29252884aeSStefan Eßer *
30252884aeSStefan Eßer * *****************************************************************************
31252884aeSStefan Eßer *
32252884aeSStefan Eßer * The entry point for bc.
33252884aeSStefan Eßer *
34252884aeSStefan Eßer */
35252884aeSStefan Eßer
36252884aeSStefan Eßer #include <assert.h>
37252884aeSStefan Eßer #include <stdlib.h>
38252884aeSStefan Eßer #include <string.h>
39252884aeSStefan Eßer
4000698711SStefan Eßer #if BC_ENABLE_NLS
41252884aeSStefan Eßer #include <locale.h>
4200698711SStefan Eßer #endif // BC_ENABLE_NLS
437e5c51e5SStefan Eßer
447e5c51e5SStefan Eßer #ifndef _WIN32
45252884aeSStefan Eßer #include <libgen.h>
467e5c51e5SStefan Eßer #endif // _WIN32
47252884aeSStefan Eßer
48252884aeSStefan Eßer #include <setjmp.h>
49252884aeSStefan Eßer
507e5c51e5SStefan Eßer #include <version.h>
51252884aeSStefan Eßer #include <status.h>
52252884aeSStefan Eßer #include <vm.h>
53252884aeSStefan Eßer #include <bc.h>
54252884aeSStefan Eßer #include <dc.h>
55252884aeSStefan Eßer
5678bc019dSStefan Eßer int
main(int argc,const char * argv[])57*ef575276SStefan Eßer main(int argc, const char* argv[])
5878bc019dSStefan Eßer {
59a970610aSStefan Eßer BcStatus s;
60252884aeSStefan Eßer char* name;
61252884aeSStefan Eßer size_t len = strlen(BC_EXECPREFIX);
62252884aeSStefan Eßer
6300698711SStefan Eßer #if BC_ENABLE_NLS
6444d4804dSStefan Eßer // Must set the locale properly in order to have the right error messages.
65d101cdd6SStefan Eßer vm->locale = setlocale(LC_ALL, "");
6600698711SStefan Eßer #endif // BC_ENABLE_NLS
67252884aeSStefan Eßer
6844d4804dSStefan Eßer // Set the start pledge().
6944d4804dSStefan Eßer bc_pledge(bc_pledge_start, NULL);
7044d4804dSStefan Eßer
7100698711SStefan Eßer // Sometimes, argv[0] can be NULL. Better make sure to be robust against it.
7278bc019dSStefan Eßer if (argv[0] != NULL)
7378bc019dSStefan Eßer {
7400698711SStefan Eßer // Figure out the name of the calculator we are using. We can't use
7500698711SStefan Eßer // basename because it's not portable, but yes, this is stripping off
7600698711SStefan Eßer // the directory.
777e5c51e5SStefan Eßer name = strrchr(argv[0], BC_FILE_SEP);
78d101cdd6SStefan Eßer vm->name = (name == NULL) ? argv[0] : name + 1;
7900698711SStefan Eßer }
8000698711SStefan Eßer else
8100698711SStefan Eßer {
8200698711SStefan Eßer #if !DC_ENABLED
83d101cdd6SStefan Eßer vm->name = "bc";
8400698711SStefan Eßer #elif !BC_ENABLED
85d101cdd6SStefan Eßer vm->name = "dc";
8600698711SStefan Eßer #else
8700698711SStefan Eßer // Just default to bc in that case.
88d101cdd6SStefan Eßer vm->name = "bc";
8900698711SStefan Eßer #endif
9000698711SStefan Eßer }
91252884aeSStefan Eßer
9244d4804dSStefan Eßer // If the name is longer than the length of the prefix, skip the prefix.
93d101cdd6SStefan Eßer if (strlen(vm->name) > len) vm->name += len;
94252884aeSStefan Eßer
95252884aeSStefan Eßer BC_SIG_LOCK;
96252884aeSStefan Eßer
9744d4804dSStefan Eßer // We *must* do this here. Otherwise, other code could not jump out all of
9844d4804dSStefan Eßer // the way.
99d101cdd6SStefan Eßer bc_vec_init(&vm->jmp_bufs, sizeof(sigjmp_buf), BC_DTOR_NONE);
100252884aeSStefan Eßer
101d101cdd6SStefan Eßer BC_SETJMP_LOCKED(vm, exit);
102252884aeSStefan Eßer
10312e0d316SStefan Eßer #if BC_CLANG
10412e0d316SStefan Eßer #pragma clang diagnostic push
10512e0d316SStefan Eßer #pragma clang diagnostic ignored "-Wcast-qual"
10612e0d316SStefan Eßer #endif // BC_CLANG
107252884aeSStefan Eßer #if !DC_ENABLED
10812e0d316SStefan Eßer s = bc_main(argc, (const char**) argv);
109252884aeSStefan Eßer #elif !BC_ENABLED
11012e0d316SStefan Eßer s = dc_main(argc, (const char**) argv);
111252884aeSStefan Eßer #else
112d101cdd6SStefan Eßer // BC_IS_BC uses vm->name, which was set above. So we're good.
11312e0d316SStefan Eßer if (BC_IS_BC) s = bc_main(argc, (const char**) argv);
11412e0d316SStefan Eßer else s = dc_main(argc, (const char**) argv);
115252884aeSStefan Eßer #endif
11612e0d316SStefan Eßer #if BC_CLANG
11712e0d316SStefan Eßer #pragma clang diagnostic pop
11812e0d316SStefan Eßer #endif // BC_CLANG
119252884aeSStefan Eßer
12012e0d316SStefan Eßer vm->status = (sig_atomic_t) s;
121a970610aSStefan Eßer
122252884aeSStefan Eßer exit:
123252884aeSStefan Eßer BC_SIG_MAYLOCK;
124252884aeSStefan Eßer
12512e0d316SStefan Eßer s = bc_vm_atexit((BcStatus) vm->status);
12612e0d316SStefan Eßer
12712e0d316SStefan Eßer return (int) s;
128252884aeSStefan Eßer }
129