1 /*
2 * Copyright (c) 2021, Thomas Stranger
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 /*
8 * This is not a real serial driver. It is used to instantiate struct
9 * devices for the "vnd,serial" devicetree compatible used in test code.
10 */
11
12 #include <stdbool.h>
13
14 #include <zephyr/sys/__assert.h>
15 #include <zephyr/device.h>
16 #include <zephyr/drivers/uart.h>
17 #include <zephyr/drivers/uart/serial_test.h>
18 #include <zephyr/kernel.h>
19 #include <zephyr/logging/log.h>
20 #include <zephyr/sys/ring_buffer.h>
21
22 LOG_MODULE_REGISTER(mock_serial, CONFIG_LOG_DEFAULT_LEVEL);
23
24 #define DT_DRV_COMPAT vnd_serial
25 struct serial_vnd_data {
26 #ifdef CONFIG_RING_BUFFER
27 struct ring_buf *written;
28 struct ring_buf *read_queue;
29 #endif
30 serial_vnd_write_cb_t callback;
31 void *callback_data;
32 #ifdef CONFIG_UART_INTERRUPT_DRIVEN
33 uart_irq_callback_user_data_t irq_isr;
34 bool irq_rx_enabled;
35 bool irq_tx_enabled;
36 #endif
37 #ifdef CONFIG_UART_ASYNC_API
38 uart_callback_t async_cb;
39 void *async_cb_user_data;
40 uint8_t *read_buf;
41 size_t read_size;
42 size_t read_position;
43 #endif
44 };
45
46 #ifdef CONFIG_UART_INTERRUPT_DRIVEN
is_irq_rx_pending(const struct device * dev)47 static bool is_irq_rx_pending(const struct device *dev)
48 {
49 struct serial_vnd_data *data = dev->data;
50
51 return !ring_buf_is_empty(data->read_queue);
52 }
53
is_irq_tx_pending(const struct device * dev)54 static bool is_irq_tx_pending(const struct device *dev)
55 {
56 struct serial_vnd_data *data = dev->data;
57
58 return ring_buf_space_get(data->written) != 0;
59 }
60
irq_process(const struct device * dev)61 static void irq_process(const struct device *dev)
62 {
63 struct serial_vnd_data *data = dev->data;
64
65 for (;;) {
66 bool rx_rdy = is_irq_rx_pending(dev);
67 bool tx_rdy = is_irq_tx_pending(dev);
68 bool rx_int = rx_rdy && data->irq_rx_enabled;
69 bool tx_int = tx_rdy && data->irq_tx_enabled;
70
71 LOG_DBG("rx_rdy %d tx_rdy %d", rx_rdy, tx_rdy);
72 LOG_DBG("rx_int %d tx_int %d", rx_int, tx_int);
73
74 if (!(rx_int || tx_int)) {
75 break;
76 }
77
78 LOG_DBG("isr");
79 if (!data->irq_isr) {
80 LOG_ERR("no isr registered");
81 break;
82 }
83 data->irq_isr(dev, NULL);
84 };
85 }
86
irq_rx_enable(const struct device * dev)87 static void irq_rx_enable(const struct device *dev)
88 {
89 struct serial_vnd_data *data = dev->data;
90
91 data->irq_rx_enabled = true;
92 LOG_DBG("rx enabled");
93 irq_process(dev);
94 }
95
irq_rx_disable(const struct device * dev)96 static void irq_rx_disable(const struct device *dev)
97 {
98 struct serial_vnd_data *data = dev->data;
99
100 data->irq_rx_enabled = false;
101 LOG_DBG("rx disabled");
102 }
103
irq_rx_ready(const struct device * dev)104 static int irq_rx_ready(const struct device *dev)
105 {
106 struct serial_vnd_data *data = dev->data;
107 bool ready = !ring_buf_is_empty(data->read_queue);
108
109 LOG_DBG("rx ready: %d", ready);
110 return ready;
111 }
112
irq_tx_enable(const struct device * dev)113 static void irq_tx_enable(const struct device *dev)
114 {
115 struct serial_vnd_data *data = dev->data;
116
117 LOG_DBG("tx enabled");
118 data->irq_tx_enabled = true;
119 irq_process(dev);
120 }
121
irq_tx_disable(const struct device * dev)122 static void irq_tx_disable(const struct device *dev)
123 {
124 struct serial_vnd_data *data = dev->data;
125
126 data->irq_tx_enabled = false;
127 LOG_DBG("tx disabled");
128 }
129
irq_tx_ready(const struct device * dev)130 static int irq_tx_ready(const struct device *dev)
131 {
132 struct serial_vnd_data *data = dev->data;
133 int available = ring_buf_space_get(data->written);
134
135 LOG_DBG("tx ready: %d", available);
136 return available;
137 }
138
irq_callback_set(const struct device * dev,uart_irq_callback_user_data_t cb,void * user_data)139 static void irq_callback_set(const struct device *dev, uart_irq_callback_user_data_t cb,
140 void *user_data)
141 {
142 struct serial_vnd_data *data = dev->data;
143
144 /* Not implemented. Ok because `user_data` is always NULL in the current
145 * implementation of core UART API.
146 */
147 __ASSERT_NO_MSG(user_data == NULL);
148
149 #if defined(CONFIG_UART_EXCLUSIVE_API_CALLBACKS) && defined(CONFIG_UART_ASYNC_API)
150 if (data->read_buf) {
151 LOG_ERR("Setting callback to NULL while asynchronous API is in use.");
152 }
153 data->async_cb = NULL;
154 data->async_cb_user_data = NULL;
155 #endif
156
157 data->irq_isr = cb;
158 LOG_DBG("callback set");
159 }
160
fifo_fill(const struct device * dev,const uint8_t * tx_data,int size)161 static int fifo_fill(const struct device *dev, const uint8_t *tx_data, int size)
162 {
163 struct serial_vnd_data *data = dev->data;
164 uint32_t write_len = ring_buf_put(data->written, tx_data, size);
165
166 if (data->callback) {
167 data->callback(dev, data->callback_data);
168 }
169 return write_len;
170 }
171
fifo_read(const struct device * dev,uint8_t * rx_data,const int size)172 static int fifo_read(const struct device *dev, uint8_t *rx_data, const int size)
173 {
174 struct serial_vnd_data *data = dev->data;
175 int read_len = ring_buf_get(data->read_queue, rx_data, size);
176
177 LOG_HEXDUMP_DBG(rx_data, read_len, "");
178 return read_len;
179 }
180 #endif /* CONFIG_UART_INTERRUPT_DRIVEN */
181
serial_vnd_poll_in(const struct device * dev,unsigned char * c)182 static int serial_vnd_poll_in(const struct device *dev, unsigned char *c)
183 {
184 #ifdef CONFIG_RING_BUFFER
185 struct serial_vnd_data *data = dev->data;
186 uint32_t bytes_read;
187
188 if (data == NULL || data->read_queue == NULL) {
189 return -ENOTSUP;
190 }
191 bytes_read = ring_buf_get(data->read_queue, c, 1);
192 if (bytes_read == 1) {
193 return 0;
194 }
195 return -1;
196 #else
197 return -ENOTSUP;
198 #endif
199 }
200
serial_vnd_poll_out(const struct device * dev,unsigned char c)201 static void serial_vnd_poll_out(const struct device *dev, unsigned char c)
202 {
203 struct serial_vnd_data *data = dev->data;
204
205 #ifdef CONFIG_RING_BUFFER
206 if (data == NULL || data->written == NULL) {
207 return;
208 }
209 ring_buf_put(data->written, &c, 1);
210 #endif
211 if (data->callback) {
212 data->callback(dev, data->callback_data);
213 }
214 }
215
216 #ifdef CONFIG_UART_ASYNC_API
217 static void async_rx_run(const struct device *dev);
218 #endif
219
220 #ifdef CONFIG_RING_BUFFER
serial_vnd_queue_in_data(const struct device * dev,const unsigned char * c,uint32_t size)221 int serial_vnd_queue_in_data(const struct device *dev, const unsigned char *c, uint32_t size)
222 {
223 struct serial_vnd_data *data = dev->data;
224 int write_size;
225
226 if (data == NULL || data->read_queue == NULL) {
227 return -ENOTSUP;
228 }
229 write_size = ring_buf_put(data->read_queue, c, size);
230
231 LOG_DBG("size %u write_size %u", size, write_size);
232 LOG_HEXDUMP_DBG(c, write_size, "");
233
234 #ifdef CONFIG_UART_INTERRUPT_DRIVEN
235 if (write_size > 0) {
236 irq_process(dev);
237 }
238 #endif
239
240 #ifdef CONFIG_UART_ASYNC_API
241 async_rx_run(dev);
242 #endif
243
244 return write_size;
245 }
246
serial_vnd_out_data_size_get(const struct device * dev)247 uint32_t serial_vnd_out_data_size_get(const struct device *dev)
248 {
249 struct serial_vnd_data *data = dev->data;
250
251 if (data == NULL || data->written == NULL) {
252 return -ENOTSUP;
253 }
254 return ring_buf_size_get(data->written);
255 }
256
serial_vnd_read_out_data(const struct device * dev,unsigned char * out_data,uint32_t size)257 uint32_t serial_vnd_read_out_data(const struct device *dev, unsigned char *out_data, uint32_t size)
258 {
259 struct serial_vnd_data *data = dev->data;
260
261 if (data == NULL || data->written == NULL) {
262 return -ENOTSUP;
263 }
264 return ring_buf_get(data->written, out_data, size);
265 }
266
serial_vnd_peek_out_data(const struct device * dev,unsigned char * out_data,uint32_t size)267 uint32_t serial_vnd_peek_out_data(const struct device *dev, unsigned char *out_data, uint32_t size)
268 {
269 struct serial_vnd_data *data = dev->data;
270
271 if (data == NULL || data->written == NULL) {
272 return -ENOTSUP;
273 }
274 return ring_buf_peek(data->written, out_data, size);
275 }
276 #endif
277
serial_vnd_set_callback(const struct device * dev,serial_vnd_write_cb_t callback,void * user_data)278 void serial_vnd_set_callback(const struct device *dev, serial_vnd_write_cb_t callback,
279 void *user_data)
280 {
281 struct serial_vnd_data *data = dev->data;
282
283 if (data == NULL) {
284 return;
285 }
286 data->callback = callback;
287 data->callback_data = user_data;
288 }
289
serial_vnd_err_check(const struct device * dev)290 static int serial_vnd_err_check(const struct device *dev)
291 {
292 return -ENOTSUP;
293 }
294
295 #ifdef CONFIG_UART_USE_RUNTIME_CONFIGURE
serial_vnd_configure(const struct device * dev,const struct uart_config * cfg)296 static int serial_vnd_configure(const struct device *dev, const struct uart_config *cfg)
297 {
298 return -ENOTSUP;
299 }
300
serial_vnd_config_get(const struct device * dev,struct uart_config * cfg)301 static int serial_vnd_config_get(const struct device *dev, struct uart_config *cfg)
302 {
303 return -ENOTSUP;
304 }
305 #endif /* CONFIG_UART_USE_RUNTIME_CONFIGURE */
306
307 #ifdef CONFIG_UART_ASYNC_API
serial_vnd_callback_set(const struct device * dev,uart_callback_t callback,void * user_data)308 static int serial_vnd_callback_set(const struct device *dev, uart_callback_t callback,
309 void *user_data)
310 {
311 struct serial_vnd_data *data = dev->data;
312
313 if (data == NULL) {
314 return -ENOTSUP;
315 }
316
317 #if defined(CONFIG_UART_EXCLUSIVE_API_CALLBACKS) && defined(CONFIG_UART_INTERRUPT_DRIVEN)
318 data->irq_isr = NULL;
319 #endif
320
321 if (callback == NULL && data->read_buf) {
322 LOG_ERR("Setting callback to NULL while asynchronous API is in use.");
323 }
324
325 data->async_cb = callback;
326 data->async_cb_user_data = user_data;
327
328 return 0;
329 }
330
serial_vnd_api_tx(const struct device * dev,const uint8_t * tx_data,size_t len,int32_t timeout)331 static int serial_vnd_api_tx(const struct device *dev, const uint8_t *tx_data, size_t len,
332 int32_t timeout)
333 {
334 struct serial_vnd_data *data = dev->data;
335 struct uart_event evt;
336 uint32_t write_len;
337
338 if (data == NULL) {
339 return -ENOTSUP;
340 }
341
342 if (data->async_cb == NULL) {
343 return -EINVAL;
344 }
345
346 write_len = ring_buf_put(data->written, tx_data, len);
347 if (data->callback) {
348 data->callback(dev, data->callback_data);
349 }
350
351 __ASSERT(write_len == len, "Ring buffer full. Async wait not implemented.");
352
353 evt = (struct uart_event){
354 .type = UART_TX_DONE,
355 .data.tx.buf = tx_data,
356 .data.tx.len = len,
357 };
358 data->async_cb(dev, &evt, data->async_cb_user_data);
359
360 return 0;
361 }
362
async_rx_run(const struct device * dev)363 static void async_rx_run(const struct device *dev)
364 {
365 struct serial_vnd_data *data = dev->data;
366 struct uart_event evt;
367 uint32_t read_len;
368 uint32_t read_remaining;
369
370 if (!data->read_buf) {
371 return;
372 }
373
374 __ASSERT_NO_MSG(data->async_cb);
375
376 read_remaining = data->read_size - data->read_position;
377
378 read_len = ring_buf_get(data->read_queue, &data->read_buf[data->read_position],
379 read_remaining);
380
381 if (read_len != 0) {
382 evt = (struct uart_event){
383 .type = UART_RX_RDY,
384 .data.rx.buf = data->read_buf,
385 .data.rx.len = read_len,
386 .data.rx.offset = data->read_position,
387 };
388 data->async_cb(dev, &evt, data->async_cb_user_data);
389 }
390
391 data->read_position += read_len;
392
393 if (data->read_position == data->read_size) {
394 data->read_buf = NULL;
395 evt = (struct uart_event){
396 .type = UART_RX_DISABLED,
397 };
398 data->async_cb(dev, &evt, data->async_cb_user_data);
399 }
400 }
401
serial_vnd_rx_enable(const struct device * dev,uint8_t * read_buf,size_t read_size,int32_t timeout)402 static int serial_vnd_rx_enable(const struct device *dev, uint8_t *read_buf, size_t read_size,
403 int32_t timeout)
404 {
405 struct serial_vnd_data *data = dev->data;
406
407 LOG_WRN("read_size %zd", read_size);
408
409 if (data == NULL) {
410 return -ENOTSUP;
411 }
412
413 if (data->async_cb == NULL) {
414 return -EINVAL;
415 }
416
417 __ASSERT(timeout == SYS_FOREVER_MS, "Async timeout not implemented.");
418
419 data->read_buf = read_buf;
420 data->read_size = read_size;
421 data->read_position = 0;
422
423 async_rx_run(dev);
424
425 return 0;
426 }
427 #endif /* CONFIG_UART_ASYNC_API */
428
429 static DEVICE_API(uart, serial_vnd_api) = {
430 .poll_in = serial_vnd_poll_in,
431 .poll_out = serial_vnd_poll_out,
432 .err_check = serial_vnd_err_check,
433 #ifdef CONFIG_UART_USE_RUNTIME_CONFIGURE
434 .configure = serial_vnd_configure,
435 .config_get = serial_vnd_config_get,
436 #endif /* CONFIG_UART_USE_RUNTIME_CONFIGURE */
437 #ifdef CONFIG_UART_INTERRUPT_DRIVEN
438 .irq_callback_set = irq_callback_set,
439 .irq_rx_enable = irq_rx_enable,
440 .irq_rx_disable = irq_rx_disable,
441 .irq_rx_ready = irq_rx_ready,
442 .irq_tx_enable = irq_tx_enable,
443 .irq_tx_disable = irq_tx_disable,
444 .irq_tx_ready = irq_tx_ready,
445 .fifo_read = fifo_read,
446 .fifo_fill = fifo_fill,
447 #endif /* CONFIG_UART_INTERRUPT_DRIVEN */
448 #ifdef CONFIG_UART_ASYNC_API
449 .callback_set = serial_vnd_callback_set,
450 .tx = serial_vnd_api_tx,
451 .rx_enable = serial_vnd_rx_enable,
452 #endif /* CONFIG_UART_ASYNC_API */
453 };
454
455 #define VND_SERIAL_DATA_BUFFER(n) \
456 RING_BUF_DECLARE(written_data_##n, DT_INST_PROP(n, buffer_size)); \
457 RING_BUF_DECLARE(read_queue_##n, DT_INST_PROP(n, buffer_size)); \
458 static struct serial_vnd_data serial_vnd_data_##n = { \
459 .written = &written_data_##n, \
460 .read_queue = &read_queue_##n, \
461 };
462 #define VND_SERIAL_DATA(n) static struct serial_vnd_data serial_vnd_data_##n = {};
463 #define VND_SERIAL_INIT(n) \
464 COND_CODE_1(DT_INST_NODE_HAS_PROP(n, buffer_size), (VND_SERIAL_DATA_BUFFER(n)), \
465 (VND_SERIAL_DATA(n))) \
466 DEVICE_DT_INST_DEFINE(n, NULL, NULL, &serial_vnd_data_##n, NULL, POST_KERNEL, \
467 CONFIG_SERIAL_INIT_PRIORITY, &serial_vnd_api);
468
469 DT_INST_FOREACH_STATUS_OKAY(VND_SERIAL_INIT)
470