/** * Native CPU entry code * * Copyright (C) 2013 Ludwig Knüpfer * 2017 Freie Universität Berlin * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. * * @ingroup arch * @{ * @file * @author Ludwig Knüpfer * @author Martine Lenders * @} */ #ifndef _GNU_SOURCE #define _GNU_SOURCE #include #else #include #endif #include "byteorder.h" #include #include #include #include #include #include #include #include #include #include #include #include "kernel_init.h" #include "cpu.h" #include "irq.h" #include "board_internal.h" #include "native_internal.h" #include "tty_uart.h" #define ENABLE_DEBUG (0) #include "debug.h" typedef enum { _STDIOTYPE_STDIO = 0, /**< leave intact */ _STDIOTYPE_NULL, /**< redirect to "/dev/null" */ _STDIOTYPE_FILE, /**< redirect to file */ } _stdiotype_t; int _native_null_in_pipe[2]; int _native_null_out_file; const char *_progname; char **_native_argv; pid_t _native_pid; pid_t _native_id; unsigned _native_rng_seed = 0; int _native_rng_mode = 0; const char *_native_unix_socket_path = NULL; #ifdef MODULE_NETDEV_TAP #include "netdev_tap_params.h" netdev_tap_params_t netdev_tap_params[NETDEV_TAP_MAX]; #endif #ifdef MODULE_MTD_NATIVE #include "board.h" #include "mtd_native.h" #endif static const char short_opts[] = ":hi:s:deEoc:" #ifdef MODULE_MTD_NATIVE "m:" #endif ""; static const struct option long_opts[] = { { "help", no_argument, NULL, 'h' }, { "id", required_argument, NULL, 'i' }, { "seed", required_argument, NULL, 's' }, { "daemonize", no_argument, NULL, 'd' }, { "stderr-pipe", no_argument, NULL, 'e' }, { "stderr-noredirect", no_argument, NULL, 'E' }, { "stdout-pipe", no_argument, NULL, 'o' }, { "uart-tty", required_argument, NULL, 'c' }, #ifdef MODULE_MTD_NATIVE { "mtd", required_argument, NULL, 'm' }, #endif { NULL, 0, NULL, '\0' }, }; /** * @brief initialize _native_null_in_pipe to allow for reading from stdin * * @param[in] stdiotype _STDIOTYPE_STDIO to to just initialize pipe, any other * value to also redirect stdin to that pipe */ void _native_input(_stdiotype_t stdintype) { if (real_pipe(_native_null_in_pipe) == -1) { err(EXIT_FAILURE, "_native_null_in(): pipe()"); } if (stdintype == _STDIOTYPE_STDIO) { return; } if (real_dup2(_native_null_in_pipe[0], STDIN_FILENO) == -1) { err(EXIT_FAILURE, "_native_null_in: dup2(STDIN_FILENO)"); } } /** * @brief set up output redirection * * @param[in] stdiotype The type of redirection * @param[in] output Output file. May be either `STDOUT_FILENO` for stdout or * `STDERR_FILENO` for stderr * * @return The new file descriptor of the redirection * @return -1 if the file descriptor did not change from the standard one */ int _native_log_output(_stdiotype_t stdiotype, int output) { int outfile; assert((output == STDERR_FILENO) || (output == STDOUT_FILENO)); switch (stdiotype) { case _STDIOTYPE_STDIO: return -1; case _STDIOTYPE_NULL: if ((outfile = real_open("/dev/null", O_WRONLY)) == -1) { err(EXIT_FAILURE, "_native_log_output: open"); } break; case _STDIOTYPE_FILE: { /* 20 should suffice for 64-bit PIDs ;-) */ char logname[sizeof("/tmp/riot.stderr.") + 20]; snprintf(logname, sizeof(logname), "/tmp/riot.std%s.%d", (output == STDOUT_FILENO) ? "out": "err", _native_pid); if ((outfile = real_creat(logname, 0666)) == -1) { err(EXIT_FAILURE, "_native_log_output: open"); } break; } default: errx(EXIT_FAILURE, "_native_log_output: unknown log type"); break; } if (real_dup2(outfile, output) == -1) { err(EXIT_FAILURE, "_native_log_output: dup2(output)"); } return outfile; } void daemonize(void) { if ((_native_pid = real_fork()) == -1) { err(EXIT_FAILURE, "daemonize: fork"); } if (_native_pid > 0) { real_printf("RIOT pid: %d\n", _native_pid); real_exit(EXIT_SUCCESS); } else { _native_pid = real_getpid(); /* detach from current working directory */ if (real_chdir("/") == -1) { err(EXIT_FAILURE, "daemonize: chdir"); } /* detach from process group */ if (real_setsid() == -1) { err(EXIT_FAILURE, "daemonize: setsid"); } /* set umask */ real_umask(0); } } /** * Remove any -d options from an argument vector. * * @param[in][out] argv an argument vector */ static void filter_daemonize_argv(char **argv) { int idx = 0; for (char **narg = argv; *narg != NULL; narg++, idx++) { if (strcmp("-d", narg[0]) == 0) { char **xarg = narg; do { xarg[0] = xarg[1]; } while (*xarg++ != NULL); if (optind > 1) { /* adapt optind if changed */ optind--; } narg--; /* rescan current item to filter out double args */ } } } void usage_exit(int status) { real_printf("usage: %s", _progname); #if defined(MODULE_NETDEV_TAP) for (int i = 0; i < NETDEV_TAP_MAX; i++) { real_printf(" ", i + 1); } #endif real_printf(" [-i ] [-d] [-e|-E] [-o] [-c ]\n"); real_printf(" help: %s -h\n\n", _progname); real_printf("\nOptions:\n" " -h, --help\n" " print this help message\n" " -i , --id=\n" " specify instance id (set by config module)\n" " -s , --seed=\n" " specify srandom(3) seed (/dev/random is used instead of random(3) if\n" " the option is omitted)\n" " -d, --daemonize\n" " daemonize native instance\n" " -e, --stderr-pipe\n" " redirect stderr to file\n" " -E, --stderr-noredirect\n" " do not redirect stderr (i.e. leave sterr unchanged despite\n" " daemon/socket io)\n" " -o, --stdout-pipe\n" " redirect stdout to file (/tmp/riot.stdout.PID) when not attached\n" " to socket\n" " -c , --uart-tty=\n" " specify TTY device for UART. This argument can be used multiple\n" " times (up to UART_NUMOF)\n"); #ifdef MODULE_MTD_NATIVE real_printf( " -m , --mtd=\n" " specify the file name of mtd emulated device\n"); #endif real_exit(status); } /** @brief Initialization function pointer type */ typedef void (*init_func_t)(int argc, char **argv, char **envp); #ifdef __APPLE__ /* Taken from the sources of Apple's dyld launcher * https://github.com/opensource-apple/dyld/blob/3f928f32597888c5eac6003b9199d972d49857b5/src/dyldInitialization.cpp#L85-L104 */ /* Find the extents of the __DATA __mod_init_func section */ extern init_func_t __init_array_start __asm("section$start$__DATA$__mod_init_func"); extern init_func_t __init_array_end __asm("section$end$__DATA$__mod_init_func"); #else /* Linker script provides pointers to the beginning and end of the init array */ extern init_func_t __init_array_start; extern init_func_t __init_array_end; #endif __attribute__((constructor)) static void startup(int argc, char **argv, char **envp) { _native_init_syscalls(); _native_argv = argv; _progname = argv[0]; _native_pid = real_getpid(); /* will possibly be overridden via option below: */ _native_id = _native_pid; int c, opt_idx = 0, uart = 0; bool dmn = false, force_stderr = false; _stdiotype_t stderrtype = _STDIOTYPE_STDIO; _stdiotype_t stdouttype = _STDIOTYPE_STDIO; _stdiotype_t stdintype = _STDIOTYPE_STDIO; while ((c = getopt_long(argc, argv, short_opts, long_opts, &opt_idx)) >= 0) { switch (c) { case 0: case 'h': usage_exit(EXIT_SUCCESS); break; case 'i': _native_id = atol(optarg); break; case 's': _native_rng_seed = atol(optarg); _native_rng_mode = 1; break; case 'd': dmn = true; break; case 'e': if (force_stderr) { /* -e and -E are mutually exclusive */ usage_exit(EXIT_FAILURE); } stderrtype = _STDIOTYPE_FILE; break; case 'E': if (stderrtype == _STDIOTYPE_FILE) { /* -e and -E are mutually exclusive */ usage_exit(EXIT_FAILURE); } force_stderr = true; break; case 'o': stdouttype = _STDIOTYPE_FILE; break; case 'c': tty_uart_setup(uart++, optarg); break; #ifdef MODULE_MTD_NATIVE case 'm': ((mtd_native_dev_t *)mtd0)->fname = strndup(optarg, PATH_MAX - 1); break; #endif default: usage_exit(EXIT_FAILURE); } } #ifdef MODULE_NETDEV_TAP for (int i = 0; i < NETDEV_TAP_MAX; i++) { if (argv[optind + i] == NULL) { /* no tap parameter left */ usage_exit(EXIT_FAILURE); } } #endif if (dmn) { filter_daemonize_argv(_native_argv); if (stderrtype == _STDIOTYPE_STDIO) { stderrtype = _STDIOTYPE_NULL; } if (stdouttype == _STDIOTYPE_STDIO) { stdouttype = _STDIOTYPE_NULL; } if (stdintype == _STDIOTYPE_STDIO) { stdintype = _STDIOTYPE_NULL; } daemonize(); } if (force_stderr) { stderrtype = _STDIOTYPE_STDIO; } _native_log_output(stderrtype, STDERR_FILENO); _native_null_out_file = _native_log_output(stdouttype, STDOUT_FILENO); _native_input(stdintype); /* startup is a constructor which is being called from the init_array during * C runtime initialization, this is normally used for code which must run * before launching main(), such as C++ global object constructors etc. * However, this function (startup) misbehaves a bit when we call * kernel_init below, which does not return until there is an abort or a * power off command. * We need all C++ global constructors and other initializers to run before * we enter the normal application code, which may depend on global objects * having been initalized properly. Therefore, we iterate through the * remainder of the init_array and call any constructors which have been * placed after startup in the initialization order. */ init_func_t *init_array_ptr = &__init_array_start; DEBUG("__init_array_start: %p\n", (void *)init_array_ptr); while (init_array_ptr != &__init_array_end) { /* Skip everything which has already been run */ if ((*init_array_ptr) == startup) { /* Found ourselves, move on to calling the rest of the constructors */ DEBUG("%18p - myself\n", (void *)init_array_ptr); ++init_array_ptr; break; } DEBUG("%18p - skip\n", (void *)init_array_ptr); ++init_array_ptr; } while (init_array_ptr != &__init_array_end) { /* call all remaining constructors */ DEBUG("%18p - call\n", (void *)init_array_ptr); (*init_array_ptr)(argc, argv, envp); ++init_array_ptr; } DEBUG("done, __init_array_end: %p\n", (void *)init_array_ptr); native_cpu_init(); native_interrupt_init(); #ifdef MODULE_NETDEV_TAP for (int i = 0; i < NETDEV_TAP_MAX; i++) { netdev_tap_params[i].tap_name = &argv[optind + i]; } #endif board_init(); puts("RIOT native hardware initialization complete.\n"); irq_enable(); kernel_init(); }