e18fcbba5a
Now that device_api attribute is unmodified at runtime, as well as all the other attributes, it is possible to switch all device driver instance to be constant. A coccinelle rule is used for this: @r_const_dev_1 disable optional_qualifier @ @@ -struct device * +const struct device * @r_const_dev_2 disable optional_qualifier @ @@ -struct device * const +const struct device * Fixes #27399 Signed-off-by: Tomasz Bursztyka <tomasz.bursztyka@linux.intel.com>
272 lines
6 KiB
C
272 lines
6 KiB
C
/*
|
|
* Copyright (c) 2018 Oticon A/S
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <ctype.h>
|
|
#include "init.h"
|
|
#include "kernel.h"
|
|
#include "console/console.h"
|
|
#include "posix_board_if.h"
|
|
#include <string.h>
|
|
#include <sys/time.h>
|
|
#include <sys/select.h>
|
|
#include <unistd.h>
|
|
|
|
#define DEBUG_ECHO 0
|
|
|
|
#if (DEBUG_ECHO)
|
|
#define ECHO(...) printf(__VA_ARGS__)
|
|
#else
|
|
#define ECHO(...)
|
|
#endif
|
|
|
|
#if defined(CONFIG_NATIVE_POSIX_STDOUT_CONSOLE)
|
|
/**
|
|
*
|
|
* @brief Initialize the driver that provides the printk output
|
|
*
|
|
*/
|
|
static void native_posix_stdout_init(void)
|
|
{
|
|
/* Let's ensure that even if we are redirecting to a file, we get stdout
|
|
* and stderr line buffered (default for console). Note that glibc
|
|
* ignores size. But just in case we set a reasonable number in case
|
|
* somebody tries to compile against a different library
|
|
*/
|
|
setvbuf(stdout, NULL, _IOLBF, 512);
|
|
setvbuf(stderr, NULL, _IOLBF, 512);
|
|
|
|
extern void __printk_hook_install(int (*fn)(int));
|
|
__printk_hook_install(putchar);
|
|
}
|
|
|
|
/**
|
|
* Ensure that whatever was written thru printk is displayed now
|
|
*/
|
|
void posix_flush_stdout(void)
|
|
{
|
|
fflush(stdout);
|
|
}
|
|
#endif /* CONFIG_NATIVE_POSIX_STDOUT_CONSOLE */
|
|
|
|
#if defined(CONFIG_NATIVE_POSIX_STDIN_CONSOLE)
|
|
|
|
#define VALID_DIRECTIVES \
|
|
"Valid native console driver directives:\n" \
|
|
" !wait %%u\n" \
|
|
" !quit\n"
|
|
|
|
static struct k_fifo *avail_queue;
|
|
static struct k_fifo *lines_queue;
|
|
static uint8_t (*completion_cb)(char *line, uint8_t len);
|
|
static bool stdin_is_tty;
|
|
|
|
static K_KERNEL_STACK_DEFINE(stack, CONFIG_ARCH_POSIX_RECOMMENDED_STACK_SIZE);
|
|
static struct k_thread native_stdio_thread;
|
|
|
|
static inline void found_eof(void)
|
|
{
|
|
/*
|
|
* Once stdin is closed or the input file has ended,
|
|
* there is no need to try again
|
|
*/
|
|
ECHO("Got EOF\n");
|
|
k_thread_abort(&native_stdio_thread);
|
|
}
|
|
|
|
/*
|
|
* Check if the command is a directive the driver handles on its own
|
|
* and if it is, handle it.
|
|
* If not return 0 (so it can be passed to the shell)
|
|
*
|
|
* Inputs
|
|
* s Command string
|
|
* towait Pointer to the amount of time wait until attempting to receive
|
|
* the next command
|
|
*
|
|
* return 0 if it is not a directive
|
|
* return > 0 if it was a directive (command starts with '!')
|
|
* return 2 if the driver directive requires to pause processing input
|
|
*/
|
|
static int catch_directive(char *s, int32_t *towait)
|
|
{
|
|
while (*s != 0 && isspace(*s)) {
|
|
s++;
|
|
}
|
|
|
|
if (*s != '!') {
|
|
return 0;
|
|
}
|
|
|
|
if (strncmp(s, "!wait", 5) == 0) {
|
|
int ret;
|
|
|
|
ret = sscanf(&s[5], "%i", towait);
|
|
if (ret != 1) {
|
|
posix_print_error_and_exit("%s(): '%s' not understood, "
|
|
"!wait syntax: !wait %%i\n",
|
|
__func__, s);
|
|
}
|
|
return 2;
|
|
} else if (strcmp(s, "!quit") == 0) {
|
|
posix_exit(0);
|
|
}
|
|
|
|
posix_print_warning("%s(): '%s' not understood\n" VALID_DIRECTIVES,
|
|
__func__, s);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
* Check if there is data ready in stdin
|
|
*/
|
|
static int stdin_not_ready(void)
|
|
{
|
|
int ready;
|
|
fd_set readfds;
|
|
struct timeval timeout;
|
|
|
|
timeout.tv_usec = 0;
|
|
timeout.tv_sec = 0;
|
|
|
|
FD_ZERO(&readfds);
|
|
FD_SET(STDIN_FILENO, &readfds);
|
|
|
|
ready = select(STDIN_FILENO+1, &readfds, NULL, NULL, &timeout);
|
|
|
|
if (ready == 0) {
|
|
return 1;
|
|
} else if (ready == -1) {
|
|
posix_print_error_and_exit("%s: Error on select ()\n",
|
|
__func__);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Check if there is any line in the stdin buffer,
|
|
* if there is and we have available shell buffers feed it to the shell
|
|
*
|
|
* This function returns how long the thread should wait in ms,
|
|
* before checking again the stdin buffer
|
|
*/
|
|
static int32_t attempt_read_from_stdin(void)
|
|
{
|
|
static struct console_input *cmd;
|
|
int32_t towait = CONFIG_NATIVE_STDIN_POLL_PERIOD;
|
|
|
|
while (1) {
|
|
char *ret;
|
|
int last;
|
|
int is_directive;
|
|
|
|
if (feof(stdin)) {
|
|
found_eof();
|
|
}
|
|
|
|
/*
|
|
* If stdin comes from a terminal, we check if the user has
|
|
* input something, and if not we pause the process.
|
|
*
|
|
* If stdin is not coming from a terminal, but from a file or
|
|
* pipe, we always proceed to try to get data and block until
|
|
* we do
|
|
*/
|
|
if (stdin_is_tty && stdin_not_ready()) {
|
|
return towait;
|
|
}
|
|
|
|
/* Pick next available shell line buffer */
|
|
if (!cmd) {
|
|
cmd = k_fifo_get(avail_queue, K_NO_WAIT);
|
|
if (!cmd) {
|
|
return towait;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* By default stdin is (_IOLBF) line buffered when connected to
|
|
* a terminal and fully buffered (_IOFBF) when connected to a
|
|
* pipe/file.
|
|
* If we got a terminal: we already checked for it to be ready
|
|
* and therefore a full line should be there for us.
|
|
*
|
|
* If we got a pipe or file we will block until we get a line,
|
|
* or we reach EOF
|
|
*/
|
|
ret = fgets(cmd->line, CONSOLE_MAX_LINE_LEN, stdin);
|
|
if (ret == NULL) {
|
|
if (feof(stdin)) {
|
|
found_eof();
|
|
}
|
|
/*
|
|
* Otherwise this was an unexpected error we do
|
|
* not try to handle
|
|
*/
|
|
return towait;
|
|
}
|
|
|
|
/* Remove a possible end of line and other trailing spaces */
|
|
last = (int)strlen(cmd->line) - 1;
|
|
while ((last >= 0) && isspace(cmd->line[last])) {
|
|
cmd->line[last--] = 0;
|
|
}
|
|
|
|
ECHO("Got: \"%s\"\n", cmd->line);
|
|
|
|
/*
|
|
* This console has a special set of directives which start with
|
|
* "!" which we capture here
|
|
*/
|
|
is_directive = catch_directive(cmd->line, &towait);
|
|
if (is_directive == 2) {
|
|
return towait;
|
|
} else if (is_directive > 0) {
|
|
continue;
|
|
}
|
|
|
|
/* Let's give it to the shell to handle */
|
|
k_fifo_put(lines_queue, cmd);
|
|
cmd = NULL;
|
|
}
|
|
|
|
return towait;
|
|
}
|
|
|
|
/**
|
|
* This thread will check if there is any new line in the stdin buffer
|
|
* every CONFIG_NATIVE_STDIN_POLL_PERIOD ms
|
|
*
|
|
* If there is, it will feed it to the shell
|
|
*/
|
|
static void native_stdio_runner(void *p1, void *p2, void *p3)
|
|
{
|
|
stdin_is_tty = isatty(STDIN_FILENO);
|
|
|
|
while (1) {
|
|
int32_t wait_time = attempt_read_from_stdin();
|
|
|
|
k_sleep(wait_time);
|
|
}
|
|
}
|
|
#endif /* CONFIG_NATIVE_POSIX_STDIN_CONSOLE */
|
|
|
|
static int native_posix_console_init(const struct device *arg)
|
|
{
|
|
ARG_UNUSED(arg);
|
|
|
|
#if defined(CONFIG_NATIVE_POSIX_STDOUT_CONSOLE)
|
|
native_posix_stdout_init();
|
|
#endif
|
|
|
|
return 0;
|
|
}
|
|
|
|
SYS_INIT(native_posix_console_init, PRE_KERNEL_1,
|
|
CONFIG_NATIVE_POSIX_CONSOLE_INIT_PRIORITY);
|