/** * Copyright © 2016 IBM Corporation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #pragma once #include #include #include #include /* for speed_t */ #include #include #include #include struct console; struct config; /* Handler API. * * Console data handlers: these implement the functions that process * data coming out of the main tty device. * * Handlers are registered at link time using the console_handler_register() * macro. We call each handler's ->init() function at startup, and ->fini() at * exit. * * Handlers will almost always want to register a ringbuffer consumer, which * provides data coming from the tty. Use cosole_register_ringbuffer_consumer() * for this. To send data to the tty, use console_data_out(). * * If a handler needs to monitor a separate file descriptor for events, use the * poller API, through console_poller_register(). */ struct handler; struct handler_type { const char *name; struct handler *(*init)(const struct handler_type *type, struct console *console, struct config *config); void (*fini)(struct handler *handler); int (*baudrate)(struct handler *handler, speed_t baudrate); void (*deselect)(struct handler *handler); }; struct handler { const struct handler_type *type; }; /* NOLINTBEGIN(bugprone-reserved-identifier,cert-dcl37-c,cert-dcl51-cpp) */ #define __handler_name(n) __handler_##n #define _handler_name(n) __handler_name(n) #ifndef __clang__ #define handler_type_check(h) BUILD_ASSERT_OR_ZERO((h)->init && (h)->fini) #else /* clang doesn't seem to be able to constify the type ops */ #define handler_type_check(h) 0 #endif #define console_handler_register(h) \ static const __attribute__((section("handlers"))) \ __attribute__((used)) struct handler_type * \ _handler_name(__COUNTER__) = (h) + handler_type_check(h) /* NOLINTEND(bugprone-reserved-identifier,cert-dcl37-c,cert-dcl51-cpp) */ int console_data_out(struct console *console, const uint8_t *data, size_t len); enum poller_ret { POLLER_OK = 0, POLLER_REMOVE, POLLER_EXIT, }; typedef char(socket_path_t)[sizeof(((struct sockaddr_un *)NULL)->sun_path)]; typedef enum poller_ret (*poller_event_fn_t)(struct handler *handler, int revents, void *data); typedef enum poller_ret (*poller_timeout_fn_t)(struct handler *handler, void *data); enum tty_device { TTY_DEVICE_UNDEFINED = 0, TTY_DEVICE_VUART, TTY_DEVICE_UART, TTY_DEVICE_PTY, }; struct console_server { struct { const char *kname; char *dev; int fd; enum tty_device type; union { struct { char *sysfs_devnode; int sirq; uint16_t lpc_addr; } vuart; struct { speed_t baud; } uart; }; } tty; // All the pollfds are stored here, // so 'poll' can operate on them. // The other 'pollfd*' are just pointers to this array. struct pollfd *pollfds; size_t capacity_pollfds; // index into pollfds size_t tty_pollfd_index; struct config *config; // the currently active console struct console *active; struct console **consoles; size_t n_consoles; // index into (struct console_server)->pollfds size_t dbus_pollfd_index; struct sd_bus *bus; // may be NULL in case there is no mux struct console_mux *mux; }; struct console { // point back to the console server // which we are a member of struct console_server *server; const char *console_id; /* Socket name starts with null character hence we need length */ socket_path_t socket_name; ssize_t socket_name_len; struct ringbuffer *rb; struct handler **handlers; long n_handlers; struct poller **pollers; long n_pollers; // values to configure the mux unsigned long mux_index; }; /* poller API */ struct poller { struct handler *handler; void *data; poller_event_fn_t event_fn; poller_timeout_fn_t timeout_fn; struct timeval timeout; bool remove; // index into (struct console_server)->pollfds size_t pollfd_index; }; struct poller *console_poller_register(struct console *console, struct handler *handler, poller_event_fn_t poller_fn, poller_timeout_fn_t timeout_fn, int fd, int events, void *data); void console_poller_unregister(struct console *console, struct poller *poller); void console_poller_set_events(struct console *console, struct poller *poller, int events); void console_poller_set_timeout(struct console *console, struct poller *poller, const struct timeval *tv); /* ringbuffer API */ enum ringbuffer_poll_ret { RINGBUFFER_POLL_OK = 0, RINGBUFFER_POLL_REMOVE, }; typedef enum ringbuffer_poll_ret (*ringbuffer_poll_fn_t)(void *data, size_t force_len); struct ringbuffer_consumer; struct ringbuffer { uint8_t *buf; size_t size; size_t tail; struct ringbuffer_consumer **consumers; int n_consumers; }; struct ringbuffer_consumer { struct ringbuffer *rb; ringbuffer_poll_fn_t poll_fn; void *poll_data; size_t pos; }; struct ringbuffer *ringbuffer_init(size_t size); void ringbuffer_fini(struct ringbuffer *rb); struct ringbuffer_consumer * ringbuffer_consumer_register(struct ringbuffer *rb, ringbuffer_poll_fn_t poll_fn, void *data); void ringbuffer_consumer_unregister(struct ringbuffer_consumer *rbc); int ringbuffer_queue(struct ringbuffer *rb, uint8_t *data, size_t len); size_t ringbuffer_dequeue_peek(struct ringbuffer_consumer *rbc, size_t offset, uint8_t **data); int ringbuffer_dequeue_commit(struct ringbuffer_consumer *rbc, size_t len); size_t ringbuffer_len(struct ringbuffer_consumer *rbc); /* console wrapper around ringbuffer consumer registration */ struct ringbuffer_consumer * console_ringbuffer_consumer_register(struct console *console, ringbuffer_poll_fn_t poll_fn, void *data); /* Console server API */ void tty_init_termios(struct console_server *server); /* socket paths */ ssize_t console_socket_path(socket_path_t path, const char *id); ssize_t console_socket_path_readable(const struct sockaddr_un *addr, size_t addrlen, socket_path_t path); /* utils */ int write_buf_to_fd(int fd, const uint8_t *buf, size_t len); /* console_server dbus */ int dbus_server_init(struct console_server *server); void dbus_server_fini(struct console_server *server); /* console-dbus API */ int dbus_init(struct console *console, struct config *config __attribute__((unused))); /* socket-handler API */ int dbus_create_socket_consumer(struct console *console); #ifndef offsetof #define offsetof(type, member) ((unsigned long)&((type *)NULL)->member) #endif #define container_of(ptr, type, member) \ ((type *)((void *)((ptr) - offsetof(type, member)))) #define BUILD_ASSERT(c) \ do { \ char __c[(c) ? 1 : -1] __attribute__((unused)); \ } while (0) #define BUILD_ASSERT_OR_ZERO(c) (sizeof(char[(c) ? 1 : -1]) - 1) // returns the index of that pollfd in server->pollfds // we cannot return a pointer because 'realloc' may move server->pollfds ssize_t console_server_request_pollfd(struct console_server *server, int fd, short int events); int console_server_release_pollfd(struct console_server *server, size_t pollfd_index);