blob: 9bb12bad44332352b4a2dff6434fb52cfe1926cc [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
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +080017#include <poll.h>
Jeremy Kerr021b91f2016-04-28 11:51:52 +080018#include <stdbool.h>
Benjamin Fairfcbdea92018-06-04 14:19:25 -070019#include <stdint.h>
20#include <termios.h> /* for speed_t */
Cheng C Yangf9c8f6c2019-03-04 18:39:52 +080021#include <systemd/sd-bus.h>
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +080022
Jeremy Kerr329a35f2016-03-10 15:36:01 +080023struct console;
Jeremy Kerrd47963e2016-03-16 17:29:55 +080024struct config;
Jeremy Kerr329a35f2016-03-10 15:36:01 +080025
Jeremy Kerrf733c852017-02-07 18:40:10 +080026/* Handler API.
27 *
Jeremy Kerr1b575012017-02-08 11:22:23 +080028 * Console data handlers: these implement the functions that process
29 * data coming out of the main tty device.
30 *
31 * Handlers are registered at link time using the console_handler_register()
32 * macro. We call each handler's ->init() function at startup, and ->fini() at
33 * exit.
34 *
Jeremy Kerrf733c852017-02-07 18:40:10 +080035 * Handlers will almost always want to register a ringbuffer consumer, which
36 * provides data coming from the tty. Use cosole_register_ringbuffer_consumer()
37 * for this. To send data to the tty, use console_data_out().
Jeremy Kerr1b575012017-02-08 11:22:23 +080038 *
39 * If a handler needs to monitor a separate file descriptor for events, use the
40 * poller API, through console_poller_register().
41 */
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +080042struct handler {
43 const char *name;
44 int (*init)(struct handler *handler,
Jeremy Kerrd47963e2016-03-16 17:29:55 +080045 struct console *console,
46 struct config *config);
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +080047 void (*fini)(struct handler *handler);
Cheng C Yangf9c8f6c2019-03-04 18:39:52 +080048 int (*baudrate)(struct handler *handler,
49 speed_t baudrate);
Jeremy Kerr021b91f2016-04-28 11:51:52 +080050 bool active;
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +080051};
52
53#define __handler_name(n) __handler_ ## n
54#define _handler_name(n) __handler_name(n)
55
Jeremy Kerr55c97122017-02-07 17:06:46 +080056#define console_handler_register(h) \
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +080057 static const \
58 __attribute__((section("handlers"))) \
59 __attribute__((used)) \
60 struct handler * _handler_name(__COUNTER__) = h;
61
62int console_data_out(struct console *console, const uint8_t *data, size_t len);
63
Jeremy Kerr329a35f2016-03-10 15:36:01 +080064/* poller API */
65struct poller;
66
67enum poller_ret {
68 POLLER_OK = 0,
69 POLLER_REMOVE,
70 POLLER_EXIT,
71};
72
73typedef enum poller_ret (*poller_fn_t)(struct handler *handler,
74 int revents, void *data);
75
Jeremy Kerr55c97122017-02-07 17:06:46 +080076struct poller *console_poller_register(struct console *console,
Jeremy Kerr329a35f2016-03-10 15:36:01 +080077 struct handler *handler, poller_fn_t poller_fn,
78 int fd, int events, void *data);
79
Jeremy Kerr55c97122017-02-07 17:06:46 +080080void console_poller_unregister(struct console *console, struct poller *poller);
Jeremy Kerr329a35f2016-03-10 15:36:01 +080081
Jeremy Kerr6b1fed22017-02-07 21:40:38 +080082void console_poller_set_events(struct console *console, struct poller *poller,
83 int events);
84
Jeremy Kerrc9775ce2017-02-07 16:25:34 +080085/* ringbuffer API */
86enum ringbuffer_poll_ret {
87 RINGBUFFER_POLL_OK = 0,
88 RINGBUFFER_POLL_REMOVE,
89};
90
91typedef enum ringbuffer_poll_ret (*ringbuffer_poll_fn_t)(void *data,
92 size_t force_len);
93
94struct ringbuffer;
95struct ringbuffer_consumer;
96
97struct ringbuffer *ringbuffer_init(size_t size);
98void ringbuffer_fini(struct ringbuffer *rb);
99
100struct ringbuffer_consumer *ringbuffer_consumer_register(struct ringbuffer *rb,
101 ringbuffer_poll_fn_t poll_fn, void *data);
102
103void ringbuffer_consumer_unregister(struct ringbuffer_consumer *rbc);
104
105int ringbuffer_queue(struct ringbuffer *rb, uint8_t *data, size_t len);
106
107size_t ringbuffer_dequeue_peek(struct ringbuffer_consumer *rbc, size_t offset,
108 uint8_t **data);
109
110int ringbuffer_dequeue_commit(struct ringbuffer_consumer *rbc, size_t len);
111
Jeremy Kerrf733c852017-02-07 18:40:10 +0800112/* console wrapper around ringbuffer consumer registration */
113struct ringbuffer_consumer *console_ringbuffer_consumer_register(
114 struct console *console,
115 ringbuffer_poll_fn_t poll_fn, void *data);
116
Jeremy Kerrd66195c2016-03-16 17:24:51 +0800117/* config API */
118struct config;
119const char *config_get_value(struct config *config, const char *name);
120struct config *config_init(const char *filename);
121void config_fini(struct config *config);
122
Benjamin Fairfcbdea92018-06-04 14:19:25 -0700123int config_parse_baud(speed_t *speed, const char *baud_string);
Cheng C Yangf9c8f6c2019-03-04 18:39:52 +0800124uint32_t parse_baud_to_int(speed_t speed);
125speed_t parse_int_to_baud(uint32_t baud);
Kun Yi6424cc32018-06-14 14:09:28 -0700126int config_parse_logsize(const char *size_str, size_t *size);
Benjamin Fairfcbdea92018-06-04 14:19:25 -0700127
Jeremy Kerr2bd05182016-03-10 16:59:43 +0800128/* socket paths */
Jeremy Kerr0cff6522016-03-18 09:57:01 +0800129extern const char *console_socket_path;
130extern const size_t console_socket_path_len;
131extern const char *console_socket_path_readable;
Jeremy Kerr2bd05182016-03-10 16:59:43 +0800132
Jeremy Kerr1a0e03b2016-03-08 17:57:11 +0800133/* utils */
134int write_buf_to_fd(int fd, const uint8_t *buf, size_t len);
135
136#define ARRAY_SIZE(a) (sizeof(a) / sizeof(a[0]))
137
138#define offsetof(type, member) \
139 ((unsigned long)&((type *)NULL)->member)
140
141#define container_of(ptr, type, member) \
142 ((type *)((void *)((ptr) - offsetof(type, member))))
143
Jeremy Kerr658cbda2016-03-09 18:10:00 +0800144#define BUILD_ASSERT(c) \
145 do { \
146 char __c[(c)?1:-1] __attribute__((unused)); \
147 } while (0)