blob: 4360513ba9876c8c21ac87fa644549aa6bf2e717 [file] [log] [blame]
Jeremy Kerr9326d772016-03-17 17:15:02 +08001/**
2 * Copyright © 2016 IBM Corporation
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +080016
Johnathan Manteya395a772019-03-19 14:28:53 -070017#pragma once
18
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +080019#include <poll.h>
Jeremy Kerr021b91f2016-04-28 11:51:52 +080020#include <stdbool.h>
Benjamin Fairfcbdea92018-06-04 14:19:25 -070021#include <stdint.h>
22#include <termios.h> /* for speed_t */
Johnathan Mantey1cecc5d2019-02-28 15:01:46 -080023#include <time.h>
Cheng C Yangf9c8f6c2019-03-04 18:39:52 +080024#include <systemd/sd-bus.h>
Johnathan Mantey1cecc5d2019-02-28 15:01:46 -080025#include <sys/time.h>
Andrew Jeffery5e7c0782020-02-10 12:12:36 +103026#include <sys/un.h>
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +080027
Jeremy Kerr329a35f2016-03-10 15:36:01 +080028struct console;
Jeremy Kerrd47963e2016-03-16 17:29:55 +080029struct config;
Jeremy Kerr329a35f2016-03-10 15:36:01 +080030
Jeremy Kerrf733c852017-02-07 18:40:10 +080031/* Handler API.
32 *
Jeremy Kerr1b575012017-02-08 11:22:23 +080033 * Console data handlers: these implement the functions that process
34 * data coming out of the main tty device.
35 *
36 * Handlers are registered at link time using the console_handler_register()
37 * macro. We call each handler's ->init() function at startup, and ->fini() at
38 * exit.
39 *
Jeremy Kerrf733c852017-02-07 18:40:10 +080040 * Handlers will almost always want to register a ringbuffer consumer, which
41 * provides data coming from the tty. Use cosole_register_ringbuffer_consumer()
42 * for this. To send data to the tty, use console_data_out().
Jeremy Kerr1b575012017-02-08 11:22:23 +080043 *
44 * If a handler needs to monitor a separate file descriptor for events, use the
45 * poller API, through console_poller_register().
46 */
Jeremy Kerre2826c72024-07-05 10:54:21 +080047struct handler;
48
49struct handler_type {
Andrew Jefferya72711a2023-04-18 18:19:41 +093050 const char *name;
Jeremy Kerre2826c72024-07-05 10:54:21 +080051 struct handler *(*init)(const struct handler_type *type,
52 struct console *console, struct config *config);
Andrew Jefferya72711a2023-04-18 18:19:41 +093053 void (*fini)(struct handler *handler);
54 int (*baudrate)(struct handler *handler, speed_t baudrate);
Alexander Hansena6b29102024-07-10 15:33:37 +020055 void (*deselect)(struct handler *handler);
Jeremy Kerre2826c72024-07-05 10:54:21 +080056};
57
58struct handler {
59 const struct handler_type *type;
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +080060};
61
Andrew Jeffery750fb0c2023-04-19 11:48:50 +093062/* NOLINTBEGIN(bugprone-reserved-identifier,cert-dcl37-c,cert-dcl51-cpp) */
Andrew Jefferya72711a2023-04-18 18:19:41 +093063#define __handler_name(n) __handler_##n
64#define _handler_name(n) __handler_name(n)
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +080065
Jeremy Kerre2826c72024-07-05 10:54:21 +080066#ifndef __clang__
67#define handler_type_check(h) BUILD_ASSERT_OR_ZERO((h)->init && (h)->fini)
68#else
69/* clang doesn't seem to be able to constify the type ops */
70#define handler_type_check(h) 0
71#endif
72
Andrew Jefferya72711a2023-04-18 18:19:41 +093073#define console_handler_register(h) \
74 static const __attribute__((section("handlers"))) \
Jeremy Kerre2826c72024-07-05 10:54:21 +080075 __attribute__((used)) struct handler_type * \
76 _handler_name(__COUNTER__) = (h) + handler_type_check(h)
Andrew Jeffery750fb0c2023-04-19 11:48:50 +093077/* NOLINTEND(bugprone-reserved-identifier,cert-dcl37-c,cert-dcl51-cpp) */
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +080078
79int console_data_out(struct console *console, const uint8_t *data, size_t len);
80
Jeremy Kerr329a35f2016-03-10 15:36:01 +080081enum poller_ret {
82 POLLER_OK = 0,
83 POLLER_REMOVE,
84 POLLER_EXIT,
85};
86
Ninad Palsuleb14ca192023-03-31 09:49:00 -050087typedef char(socket_path_t)[sizeof(((struct sockaddr_un *)NULL)->sun_path)];
88
Johnathan Mantey1cecc5d2019-02-28 15:01:46 -080089typedef enum poller_ret (*poller_event_fn_t)(struct handler *handler,
Andrew Jefferya72711a2023-04-18 18:19:41 +093090 int revents, void *data);
Johnathan Mantey1cecc5d2019-02-28 15:01:46 -080091typedef enum poller_ret (*poller_timeout_fn_t)(struct handler *handler,
Andrew Jefferya72711a2023-04-18 18:19:41 +093092 void *data);
Jeremy Kerr329a35f2016-03-10 15:36:01 +080093
Andrew Jeffery30ea6382023-05-03 16:56:10 +093094enum tty_device {
95 TTY_DEVICE_UNDEFINED = 0,
96 TTY_DEVICE_VUART,
97 TTY_DEVICE_UART,
98 TTY_DEVICE_PTY,
99};
100
Alexander Hansenc2b0d8c2024-06-07 14:35:44 +0200101struct console_server {
Andrew Jeffery30ea6382023-05-03 16:56:10 +0930102 struct {
103 const char *kname;
104 char *dev;
105 int fd;
106 enum tty_device type;
107 union {
108 struct {
109 char *sysfs_devnode;
110 int sirq;
111 uint16_t lpc_addr;
112 } vuart;
113 struct {
114 speed_t baud;
115 } uart;
116 };
117 } tty;
Alexander Hansenc2b0d8c2024-06-07 14:35:44 +0200118
Alexander Hansen2325d5d2024-06-07 16:54:20 +0200119 // All the pollfds are stored here,
120 // so 'poll' can operate on them.
121 // The other 'pollfd*' are just pointers to this array.
122 struct pollfd *pollfds;
123 size_t capacity_pollfds;
124
125 // index into pollfds
126 size_t tty_pollfd_index;
127
Alexander Hansen312ecdc2024-07-10 15:33:37 +0200128 struct config *config;
129
130 // the currently active console
Alexander Hansenc2b0d8c2024-06-07 14:35:44 +0200131 struct console *active;
Alexander Hansen312ecdc2024-07-10 15:33:37 +0200132
133 struct console **consoles;
134 size_t n_consoles;
135
136 // index into (struct console_server)->pollfds
137 size_t dbus_pollfd_index;
138
139 struct sd_bus *bus;
Alexander Hansena6b29102024-07-10 15:33:37 +0200140
141 // may be NULL in case there is no mux
142 struct console_mux *mux;
Alexander Hansenc2b0d8c2024-06-07 14:35:44 +0200143};
144
145struct console {
146 // point back to the console server
147 // which we are a member of
148 struct console_server *server;
149
Ninad Palsulee258e512023-04-26 22:17:57 -0500150 const char *console_id;
Ninad Palsulee258e512023-04-26 22:17:57 -0500151
Ninad Palsuleb14ca192023-03-31 09:49:00 -0500152 /* Socket name starts with null character hence we need length */
153 socket_path_t socket_name;
154 ssize_t socket_name_len;
155
Ninad Palsulee258e512023-04-26 22:17:57 -0500156 struct ringbuffer *rb;
157
158 struct handler **handlers;
159 long n_handlers;
160
161 struct poller **pollers;
162 long n_pollers;
Alexander Hansena6b29102024-07-10 15:33:37 +0200163
164 // values to configure the mux
165 unsigned long mux_index;
Ninad Palsulee258e512023-04-26 22:17:57 -0500166};
167
168/* poller API */
169struct poller {
170 struct handler *handler;
171 void *data;
172 poller_event_fn_t event_fn;
173 poller_timeout_fn_t timeout_fn;
174 struct timeval timeout;
175 bool remove;
Ninad Palsuleb14ca192023-03-31 09:49:00 -0500176
Alexander Hansen2325d5d2024-06-07 16:54:20 +0200177 // index into (struct console_server)->pollfds
178 size_t pollfd_index;
Ninad Palsulee258e512023-04-26 22:17:57 -0500179};
180
Jeremy Kerr55c97122017-02-07 17:06:46 +0800181struct poller *console_poller_register(struct console *console,
Andrew Jefferya72711a2023-04-18 18:19:41 +0930182 struct handler *handler,
Andrew Jeffery64222152023-04-19 12:51:24 +0930183 poller_event_fn_t poller_fn,
Andrew Jefferya72711a2023-04-18 18:19:41 +0930184 poller_timeout_fn_t timeout_fn, int fd,
185 int events, void *data);
Jeremy Kerr329a35f2016-03-10 15:36:01 +0800186
Jeremy Kerr55c97122017-02-07 17:06:46 +0800187void console_poller_unregister(struct console *console, struct poller *poller);
Jeremy Kerr329a35f2016-03-10 15:36:01 +0800188
Jeremy Kerr6b1fed22017-02-07 21:40:38 +0800189void console_poller_set_events(struct console *console, struct poller *poller,
Andrew Jefferya72711a2023-04-18 18:19:41 +0930190 int events);
Jeremy Kerr6b1fed22017-02-07 21:40:38 +0800191
Johnathan Mantey1cecc5d2019-02-28 15:01:46 -0800192void console_poller_set_timeout(struct console *console, struct poller *poller,
Andrew Jeffery64222152023-04-19 12:51:24 +0930193 const struct timeval *tv);
Johnathan Mantey1cecc5d2019-02-28 15:01:46 -0800194
Jeremy Kerrc9775ce2017-02-07 16:25:34 +0800195/* ringbuffer API */
Andrew Jeffery1b868442023-04-19 13:16:32 +0930196
Jeremy Kerrc9775ce2017-02-07 16:25:34 +0800197enum ringbuffer_poll_ret {
198 RINGBUFFER_POLL_OK = 0,
199 RINGBUFFER_POLL_REMOVE,
200};
201
202typedef enum ringbuffer_poll_ret (*ringbuffer_poll_fn_t)(void *data,
Andrew Jefferya72711a2023-04-18 18:19:41 +0930203 size_t force_len);
Jeremy Kerrc9775ce2017-02-07 16:25:34 +0800204
Jeremy Kerrc9775ce2017-02-07 16:25:34 +0800205struct ringbuffer_consumer;
206
Andrew Jeffery1b868442023-04-19 13:16:32 +0930207struct ringbuffer {
208 uint8_t *buf;
209 size_t size;
210 size_t tail;
211 struct ringbuffer_consumer **consumers;
212 int n_consumers;
213};
214
215struct ringbuffer_consumer {
216 struct ringbuffer *rb;
217 ringbuffer_poll_fn_t poll_fn;
218 void *poll_data;
219 size_t pos;
220};
221
Jeremy Kerrc9775ce2017-02-07 16:25:34 +0800222struct ringbuffer *ringbuffer_init(size_t size);
223void ringbuffer_fini(struct ringbuffer *rb);
224
Andrew Jefferya72711a2023-04-18 18:19:41 +0930225struct ringbuffer_consumer *
226ringbuffer_consumer_register(struct ringbuffer *rb,
227 ringbuffer_poll_fn_t poll_fn, void *data);
Jeremy Kerrc9775ce2017-02-07 16:25:34 +0800228
229void ringbuffer_consumer_unregister(struct ringbuffer_consumer *rbc);
230
231int ringbuffer_queue(struct ringbuffer *rb, uint8_t *data, size_t len);
232
233size_t ringbuffer_dequeue_peek(struct ringbuffer_consumer *rbc, size_t offset,
Andrew Jefferya72711a2023-04-18 18:19:41 +0930234 uint8_t **data);
Jeremy Kerrc9775ce2017-02-07 16:25:34 +0800235
236int ringbuffer_dequeue_commit(struct ringbuffer_consumer *rbc, size_t len);
237
Johnathan Mantey1cecc5d2019-02-28 15:01:46 -0800238size_t ringbuffer_len(struct ringbuffer_consumer *rbc);
239
Jeremy Kerrf733c852017-02-07 18:40:10 +0800240/* console wrapper around ringbuffer consumer registration */
Andrew Jefferya72711a2023-04-18 18:19:41 +0930241struct ringbuffer_consumer *
242console_ringbuffer_consumer_register(struct console *console,
243 ringbuffer_poll_fn_t poll_fn, void *data);
Jeremy Kerrf733c852017-02-07 18:40:10 +0800244
Ninad Palsulee258e512023-04-26 22:17:57 -0500245/* Console server API */
Alexander Hansenc2b0d8c2024-06-07 14:35:44 +0200246void tty_init_termios(struct console_server *server);
Ninad Palsulee258e512023-04-26 22:17:57 -0500247
Jeremy Kerr2bd05182016-03-10 16:59:43 +0800248/* socket paths */
Ninad Palsuleb14ca192023-03-31 09:49:00 -0500249ssize_t console_socket_path(socket_path_t path, const char *id);
Andrew Jeffery5e7c0782020-02-10 12:12:36 +1030250ssize_t console_socket_path_readable(const struct sockaddr_un *addr,
251 size_t addrlen, socket_path_t path);
Jeremy Kerr2bd05182016-03-10 16:59:43 +0800252
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +0800253/* utils */
254int write_buf_to_fd(int fd, const uint8_t *buf, size_t len);
255
Alexander Hansen312ecdc2024-07-10 15:33:37 +0200256/* console_server dbus */
257int dbus_server_init(struct console_server *server);
258void dbus_server_fini(struct console_server *server);
259
Ninad Palsulee258e512023-04-26 22:17:57 -0500260/* console-dbus API */
Andrew Jeffery498a4a82024-07-09 15:07:05 +0930261int dbus_init(struct console *console,
262 struct config *config __attribute__((unused)));
Ninad Palsulee258e512023-04-26 22:17:57 -0500263
Ninad Palsulebd992c92023-05-09 18:30:01 -0500264/* socket-handler API */
265int dbus_create_socket_consumer(struct console *console);
266
Patrick Williams397fd032023-02-10 05:50:21 -0600267#ifndef offsetof
Andrew Jefferya72711a2023-04-18 18:19:41 +0930268#define offsetof(type, member) ((unsigned long)&((type *)NULL)->member)
Patrick Williams397fd032023-02-10 05:50:21 -0600269#endif
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +0800270
Andrew Jefferya72711a2023-04-18 18:19:41 +0930271#define container_of(ptr, type, member) \
Patrick Williamsb965c222024-08-16 15:21:28 -0400272 ((type *)((void *)((ptr) - offsetof(type, member))))
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +0800273
Andrew Jefferya72711a2023-04-18 18:19:41 +0930274#define BUILD_ASSERT(c) \
275 do { \
276 char __c[(c) ? 1 : -1] __attribute__((unused)); \
Jeremy Kerr658cbda2016-03-09 18:10:00 +0800277 } while (0)
Jeremy Kerra71b3952024-07-05 11:52:49 +0800278
279#define BUILD_ASSERT_OR_ZERO(c) (sizeof(char[(c) ? 1 : -1]) - 1)
Alexander Hansen2325d5d2024-06-07 16:54:20 +0200280
281// returns the index of that pollfd in server->pollfds
282// we cannot return a pointer because 'realloc' may move server->pollfds
283ssize_t console_server_request_pollfd(struct console_server *server, int fd,
284 short int events);
285
286int console_server_release_pollfd(struct console_server *server,
287 size_t pollfd_index);