1 /*
2  * Copyright (c) 2018 Nordic Semiconductor ASA
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #include <zephyr/sys/ring_buffer.h>
8 #include <zephyr/sys/atomic.h>
9 #include <zephyr/mgmt/mcumgr/transport/smp_shell.h>
10 #include <zephyr/shell/shell_uart.h>
11 #include <zephyr/drivers/uart.h>
12 #include <zephyr/drivers/serial/uart_async_rx.h>
13 #include <zephyr/init.h>
14 #include <zephyr/logging/log.h>
15 #include <zephyr/net_buf.h>
16 
17 #define LOG_MODULE_NAME shell_uart
18 LOG_MODULE_REGISTER(shell_uart);
19 
20 #ifdef CONFIG_SHELL_BACKEND_SERIAL_RX_POLL_PERIOD
21 #define RX_POLL_PERIOD K_MSEC(CONFIG_SHELL_BACKEND_SERIAL_RX_POLL_PERIOD)
22 #else
23 #define RX_POLL_PERIOD K_NO_WAIT
24 #endif
25 
26 #ifdef CONFIG_MCUMGR_TRANSPORT_SHELL
27 NET_BUF_POOL_DEFINE(smp_shell_rx_pool, CONFIG_MCUMGR_TRANSPORT_SHELL_RX_BUF_COUNT,
28 		    SMP_SHELL_RX_BUF_SIZE, 0, NULL);
29 #endif /* CONFIG_MCUMGR_TRANSPORT_SHELL */
30 
async_callback(const struct device * dev,struct uart_event * evt,void * user_data)31 static void async_callback(const struct device *dev, struct uart_event *evt, void *user_data)
32 {
33 	struct shell_uart_async *sh_uart = (struct shell_uart_async *)user_data;
34 
35 	switch (evt->type) {
36 	case  UART_TX_DONE:
37 		k_sem_give(&sh_uart->tx_sem);
38 		break;
39 	case  UART_RX_RDY:
40 		uart_async_rx_on_rdy(&sh_uart->async_rx, evt->data.rx.buf, evt->data.rx.len);
41 		sh_uart->common.handler(SHELL_TRANSPORT_EVT_RX_RDY, sh_uart->common.context);
42 		break;
43 	case  UART_RX_BUF_REQUEST:
44 	{
45 		uint8_t *buf = uart_async_rx_buf_req(&sh_uart->async_rx);
46 		size_t len = uart_async_rx_get_buf_len(&sh_uart->async_rx);
47 
48 		if (buf) {
49 			int err = uart_rx_buf_rsp(dev, buf, len);
50 
51 			if (err < 0) {
52 				uart_async_rx_on_buf_rel(&sh_uart->async_rx, buf);
53 			}
54 		} else {
55 			atomic_inc(&sh_uart->pending_rx_req);
56 		}
57 
58 		break;
59 	}
60 	case  UART_RX_BUF_RELEASED:
61 		uart_async_rx_on_buf_rel(&sh_uart->async_rx, evt->data.rx_buf.buf);
62 		break;
63 	case  UART_RX_DISABLED:
64 		break;
65 	default:
66 		break;
67 	};
68 }
69 
uart_rx_handle(const struct device * dev,struct shell_uart_int_driven * sh_uart)70 static void uart_rx_handle(const struct device *dev, struct shell_uart_int_driven *sh_uart)
71 {
72 	uint8_t *data;
73 	uint32_t len;
74 	uint32_t rd_len;
75 	bool new_data = false;
76 #ifdef CONFIG_MCUMGR_TRANSPORT_SHELL
77 	struct smp_shell_data *const smp = &sh_uart->common.smp;
78 #endif
79 
80 	do {
81 		len = ring_buf_put_claim(&sh_uart->rx_ringbuf, &data,
82 					 sh_uart->rx_ringbuf.size);
83 
84 		if (len > 0) {
85 			rd_len = uart_fifo_read(dev, data, len);
86 
87 			/* If there is any new data to be either taken into
88 			 * ring buffer or consumed by the SMP, signal the
89 			 * shell_thread.
90 			 */
91 			if (rd_len > 0) {
92 				new_data = true;
93 			}
94 #ifdef CONFIG_MCUMGR_TRANSPORT_SHELL
95 			/* Divert bytes from shell handling if it is
96 			 * part of an mcumgr frame.
97 			 */
98 			size_t i = smp_shell_rx_bytes(smp, data, rd_len);
99 
100 			rd_len -= i;
101 
102 			if (rd_len) {
103 				for (uint32_t j = 0; j < rd_len; j++) {
104 					data[j] = data[i + j];
105 				}
106 			}
107 #endif /* CONFIG_MCUMGR_TRANSPORT_SHELL */
108 			int err = ring_buf_put_finish(&sh_uart->rx_ringbuf, rd_len);
109 			(void)err;
110 			__ASSERT_NO_MSG(err == 0);
111 		} else {
112 			uint8_t dummy;
113 
114 			/* No space in the ring buffer - consume byte. */
115 			LOG_WRN("RX ring buffer full.");
116 
117 			rd_len = uart_fifo_read(dev, &dummy, 1);
118 #ifdef CONFIG_MCUMGR_TRANSPORT_SHELL
119 			/* If successful in getting byte from the fifo, try
120 			 * feeding it to SMP as a part of mcumgr frame.
121 			 */
122 			if ((rd_len != 0) && (smp_shell_rx_bytes(smp, &dummy, 1) == 1)) {
123 				new_data = true;
124 			}
125 #endif /* CONFIG_MCUMGR_TRANSPORT_SHELL */
126 		}
127 	} while (rd_len && (rd_len == len));
128 
129 	if (new_data) {
130 		sh_uart->common.handler(SHELL_TRANSPORT_EVT_RX_RDY, sh_uart->common.context);
131 	}
132 }
133 
uart_dtr_check(const struct device * dev)134 static bool uart_dtr_check(const struct device *dev)
135 {
136 	BUILD_ASSERT(!IS_ENABLED(CONFIG_SHELL_BACKEND_SERIAL_CHECK_DTR) ||
137 		IS_ENABLED(CONFIG_UART_LINE_CTRL),
138 		"DTR check requires CONFIG_UART_LINE_CTRL");
139 
140 	if (IS_ENABLED(CONFIG_SHELL_BACKEND_SERIAL_CHECK_DTR)) {
141 		int dtr, err;
142 
143 		err = uart_line_ctrl_get(dev, UART_LINE_CTRL_DTR, &dtr);
144 		if (err == -ENOSYS || err == -ENOTSUP) {
145 			return true;
146 		}
147 
148 		return dtr;
149 	}
150 
151 	return true;
152 }
153 
dtr_timer_handler(struct k_timer * timer)154 static void dtr_timer_handler(struct k_timer *timer)
155 {
156 	struct shell_uart_int_driven *sh_uart = k_timer_user_data_get(timer);
157 
158 	if (!uart_dtr_check(sh_uart->common.dev)) {
159 		return;
160 	}
161 
162 	/* DTR is active, stop timer and start TX */
163 	k_timer_stop(timer);
164 	uart_irq_tx_enable(sh_uart->common.dev);
165 }
166 
uart_tx_handle(const struct device * dev,struct shell_uart_int_driven * sh_uart)167 static void uart_tx_handle(const struct device *dev, struct shell_uart_int_driven *sh_uart)
168 {
169 	uint32_t len;
170 	const uint8_t *data;
171 
172 	if (!uart_dtr_check(dev)) {
173 		/* Wait for DTR signal before sending anything to output. */
174 		uart_irq_tx_disable(dev);
175 		k_timer_start(&sh_uart->dtr_timer, K_MSEC(100), K_MSEC(100));
176 		return;
177 	}
178 
179 	len = ring_buf_get_claim(&sh_uart->tx_ringbuf, (uint8_t **)&data,
180 				 sh_uart->tx_ringbuf.size);
181 	if (len) {
182 		int err;
183 
184 		len = uart_fifo_fill(dev, data, len);
185 		err = ring_buf_get_finish(&sh_uart->tx_ringbuf, len);
186 		__ASSERT_NO_MSG(err == 0);
187 		ARG_UNUSED(err);
188 	} else {
189 		uart_irq_tx_disable(dev);
190 		sh_uart->tx_busy = 0;
191 	}
192 
193 	sh_uart->common.handler(SHELL_TRANSPORT_EVT_TX_RDY, sh_uart->common.context);
194 }
195 
uart_callback(const struct device * dev,void * user_data)196 static void uart_callback(const struct device *dev, void *user_data)
197 {
198 	struct shell_uart_int_driven *sh_uart = (struct shell_uart_int_driven *)user_data;
199 
200 	uart_irq_update(dev);
201 
202 	if (uart_irq_rx_ready(dev)) {
203 		uart_rx_handle(dev, sh_uart);
204 	}
205 
206 	if (uart_irq_tx_ready(dev)) {
207 		uart_tx_handle(dev, sh_uart);
208 	}
209 }
210 
irq_init(struct shell_uart_int_driven * sh_uart)211 static void irq_init(struct shell_uart_int_driven *sh_uart)
212 {
213 	const struct device *dev = sh_uart->common.dev;
214 
215 	ring_buf_init(&sh_uart->rx_ringbuf, CONFIG_SHELL_BACKEND_SERIAL_RX_RING_BUFFER_SIZE,
216 		      sh_uart->rx_buf);
217 	ring_buf_init(&sh_uart->tx_ringbuf, CONFIG_SHELL_BACKEND_SERIAL_TX_RING_BUFFER_SIZE,
218 		      sh_uart->tx_buf);
219 	sh_uart->tx_busy = 0;
220 	uart_irq_callback_user_data_set(dev, uart_callback, (void *)sh_uart);
221 	uart_irq_rx_enable(dev);
222 
223 	if (IS_ENABLED(CONFIG_SHELL_BACKEND_SERIAL_CHECK_DTR)) {
224 		k_timer_init(&sh_uart->dtr_timer, dtr_timer_handler, NULL);
225 		k_timer_user_data_set(&sh_uart->dtr_timer, (void *)sh_uart);
226 	}
227 }
228 
rx_enable(const struct device * dev,uint8_t * buf,size_t len)229 static int rx_enable(const struct device *dev, uint8_t *buf, size_t len)
230 {
231 	return uart_rx_enable(dev, buf, len, 10000);
232 }
233 
async_init(struct shell_uart_async * sh_uart)234 static void async_init(struct shell_uart_async *sh_uart)
235 {
236 	const struct device *dev = sh_uart->common.dev;
237 	struct uart_async_rx *async_rx = &sh_uart->async_rx;
238 	int err;
239 
240 	sh_uart->async_rx_config = (struct uart_async_rx_config){
241 		.buffer = sh_uart->rx_data,
242 		.length = ASYNC_RX_BUF_SIZE,
243 		.buf_cnt = CONFIG_SHELL_BACKEND_SERIAL_ASYNC_RX_BUFFER_COUNT,
244 	};
245 
246 	k_sem_init(&sh_uart->tx_sem, 0, 1);
247 
248 	err = uart_async_rx_init(async_rx, &sh_uart->async_rx_config);
249 	(void)err;
250 	__ASSERT_NO_MSG(err == 0);
251 
252 	uint8_t *buf = uart_async_rx_buf_req(async_rx);
253 
254 	err = uart_callback_set(dev, async_callback, (void *)sh_uart);
255 	(void)err;
256 	__ASSERT_NO_MSG(err == 0);
257 
258 	err = rx_enable(dev, buf, uart_async_rx_get_buf_len(async_rx));
259 	(void)err;
260 	__ASSERT_NO_MSG(err == 0);
261 }
262 
polling_rx_timeout_handler(struct k_timer * timer)263 static void polling_rx_timeout_handler(struct k_timer *timer)
264 {
265 	uint8_t c;
266 	struct shell_uart_polling *sh_uart = k_timer_user_data_get(timer);
267 
268 	while (uart_poll_in(sh_uart->common.dev, &c) == 0) {
269 		if (ring_buf_put(&sh_uart->rx_ringbuf, &c, 1) == 0U) {
270 			/* ring buffer full. */
271 			LOG_WRN("RX ring buffer full.");
272 		}
273 		sh_uart->common.handler(SHELL_TRANSPORT_EVT_RX_RDY, sh_uart->common.context);
274 	}
275 }
276 
polling_init(struct shell_uart_polling * sh_uart)277 static void polling_init(struct shell_uart_polling *sh_uart)
278 {
279 	k_timer_init(&sh_uart->rx_timer, polling_rx_timeout_handler, NULL);
280 	k_timer_user_data_set(&sh_uart->rx_timer, (void *)sh_uart);
281 	k_timer_start(&sh_uart->rx_timer, RX_POLL_PERIOD, RX_POLL_PERIOD);
282 
283 	ring_buf_init(&sh_uart->rx_ringbuf, CONFIG_SHELL_BACKEND_SERIAL_RX_RING_BUFFER_SIZE,
284 		      sh_uart->rx_buf);
285 }
286 
init(const struct shell_transport * transport,const void * config,shell_transport_handler_t evt_handler,void * context)287 static int init(const struct shell_transport *transport,
288 		const void *config,
289 		shell_transport_handler_t evt_handler,
290 		void *context)
291 {
292 	struct shell_uart_common *common = (struct shell_uart_common *)transport->ctx;
293 
294 	common->dev = (const struct device *)config;
295 	common->handler = evt_handler;
296 	common->context = context;
297 
298 #ifdef CONFIG_MCUMGR_TRANSPORT_SHELL
299 	common->smp.buf_pool = &smp_shell_rx_pool;
300 	k_fifo_init(&common->smp.buf_ready);
301 #endif
302 
303 	if (IS_ENABLED(CONFIG_SHELL_BACKEND_SERIAL_API_ASYNC)) {
304 		async_init((struct shell_uart_async *)transport->ctx);
305 	} else if (IS_ENABLED(CONFIG_SHELL_BACKEND_SERIAL_API_INTERRUPT_DRIVEN)) {
306 		irq_init((struct shell_uart_int_driven *)transport->ctx);
307 	} else {
308 		polling_init((struct shell_uart_polling *)transport->ctx);
309 	}
310 
311 	return 0;
312 }
313 
irq_uninit(struct shell_uart_int_driven * sh_uart)314 static void irq_uninit(struct shell_uart_int_driven *sh_uart)
315 {
316 	const struct device *dev = sh_uart->common.dev;
317 
318 	k_timer_stop(&sh_uart->dtr_timer);
319 	uart_irq_tx_disable(dev);
320 	uart_irq_rx_disable(dev);
321 }
322 
async_uninit(struct shell_uart_async * sh_uart)323 static void async_uninit(struct shell_uart_async *sh_uart)
324 {
325 }
326 
polling_uninit(struct shell_uart_polling * sh_uart)327 static void polling_uninit(struct shell_uart_polling *sh_uart)
328 {
329 	k_timer_stop(&sh_uart->rx_timer);
330 }
331 
uninit(const struct shell_transport * transport)332 static int uninit(const struct shell_transport *transport)
333 {
334 	if (IS_ENABLED(CONFIG_SHELL_BACKEND_SERIAL_API_ASYNC)) {
335 		async_uninit((struct shell_uart_async *)transport->ctx);
336 	} else if (IS_ENABLED(CONFIG_SHELL_BACKEND_SERIAL_API_INTERRUPT_DRIVEN)) {
337 		irq_uninit((struct shell_uart_int_driven *)transport->ctx);
338 	} else {
339 		polling_uninit((struct shell_uart_polling *)transport->ctx);
340 	}
341 
342 	return 0;
343 }
344 
enable(const struct shell_transport * transport,bool blocking_tx)345 static int enable(const struct shell_transport *transport, bool blocking_tx)
346 {
347 	struct shell_uart_common *sh_uart = (struct shell_uart_common *)transport->ctx;
348 
349 	sh_uart->blocking_tx =
350 		blocking_tx || IS_ENABLED(CONFIG_SHELL_BACKEND_SERIAL_FORCE_TX_BLOCKING_MODE);
351 
352 	if (IS_ENABLED(CONFIG_SHELL_BACKEND_SERIAL_API_INTERRUPT_DRIVEN) && blocking_tx) {
353 		uart_irq_tx_disable(sh_uart->dev);
354 	}
355 
356 	return 0;
357 }
358 
polling_write(struct shell_uart_common * sh_uart,const void * data,size_t length,size_t * cnt)359 static int polling_write(struct shell_uart_common *sh_uart,
360 			 const void *data, size_t length, size_t *cnt)
361 {
362 	const uint8_t *data8 = (const uint8_t *)data;
363 
364 	for (size_t i = 0; i < length; i++) {
365 		uart_poll_out(sh_uart->dev, data8[i]);
366 	}
367 
368 	*cnt = length;
369 
370 	sh_uart->handler(SHELL_TRANSPORT_EVT_TX_RDY, sh_uart->context);
371 
372 	return 0;
373 }
374 
irq_write(struct shell_uart_int_driven * sh_uart,const void * data,size_t length,size_t * cnt)375 static int irq_write(struct shell_uart_int_driven *sh_uart,
376 		 const void *data, size_t length, size_t *cnt)
377 {
378 	*cnt = ring_buf_put(&sh_uart->tx_ringbuf, data, length);
379 
380 	if (atomic_set(&sh_uart->tx_busy, 1) == 0) {
381 		uart_irq_tx_enable(sh_uart->common.dev);
382 	}
383 
384 	return 0;
385 }
386 
async_write(struct shell_uart_async * sh_uart,const void * data,size_t length,size_t * cnt)387 static int async_write(struct shell_uart_async *sh_uart,
388 		       const void *data, size_t length, size_t *cnt)
389 {
390 	int err;
391 
392 	err = uart_tx(sh_uart->common.dev, data, length, SYS_FOREVER_US);
393 	if (err < 0) {
394 		*cnt = 0;
395 		return err;
396 	}
397 
398 	err = k_sem_take(&sh_uart->tx_sem, K_FOREVER);
399 	*cnt = length;
400 
401 	sh_uart->common.handler(SHELL_TRANSPORT_EVT_TX_RDY, sh_uart->common.context);
402 
403 	return err;
404 }
405 
write_uart(const struct shell_transport * transport,const void * data,size_t length,size_t * cnt)406 static int write_uart(const struct shell_transport *transport,
407 		      const void *data, size_t length, size_t *cnt)
408 {
409 	struct shell_uart_common *sh_uart = (struct shell_uart_common *)transport->ctx;
410 
411 	if (IS_ENABLED(CONFIG_SHELL_BACKEND_SERIAL_API_POLLING) || sh_uart->blocking_tx) {
412 		return polling_write(sh_uart, data, length, cnt);
413 	} else if (IS_ENABLED(CONFIG_SHELL_BACKEND_SERIAL_API_INTERRUPT_DRIVEN)) {
414 		return irq_write((struct shell_uart_int_driven *)transport->ctx, data, length, cnt);
415 	} else {
416 		return async_write((struct shell_uart_async *)transport->ctx, data, length, cnt);
417 	}
418 }
419 
irq_read(struct shell_uart_int_driven * sh_uart,void * data,size_t length,size_t * cnt)420 static int irq_read(struct shell_uart_int_driven *sh_uart,
421 		    void *data, size_t length, size_t *cnt)
422 {
423 	*cnt = ring_buf_get(&sh_uart->rx_ringbuf, data, length);
424 
425 	return 0;
426 }
427 
polling_read(struct shell_uart_polling * sh_uart,void * data,size_t length,size_t * cnt)428 static int polling_read(struct shell_uart_polling *sh_uart,
429 			void *data, size_t length, size_t *cnt)
430 {
431 	*cnt = ring_buf_get(&sh_uart->rx_ringbuf, data, length);
432 
433 	return 0;
434 }
435 
async_read(struct shell_uart_async * sh_uart,void * data,size_t length,size_t * cnt)436 static int async_read(struct shell_uart_async *sh_uart,
437 		      void *data, size_t length, size_t *cnt)
438 {
439 	uint8_t *buf;
440 	size_t blen;
441 	struct uart_async_rx *async_rx = &sh_uart->async_rx;
442 
443 	blen = uart_async_rx_data_claim(async_rx, &buf, length);
444 #ifdef CONFIG_MCUMGR_TRANSPORT_SHELL
445 	struct smp_shell_data *const smp = &sh_uart->common.smp;
446 	size_t sh_cnt = 0;
447 
448 	for (size_t i = 0; i < blen; i++) {
449 		if (smp_shell_rx_bytes(smp, &buf[i], 1) == 0) {
450 			((uint8_t *)data)[sh_cnt++] = buf[i];
451 		}
452 	}
453 #else
454 	size_t sh_cnt = blen;
455 
456 	memcpy(data, buf, blen);
457 #endif
458 	bool buf_available = uart_async_rx_data_consume(async_rx, sh_cnt);
459 	*cnt = sh_cnt;
460 
461 	if (sh_uart->pending_rx_req && buf_available) {
462 		uint8_t *buf = uart_async_rx_buf_req(async_rx);
463 		size_t len = uart_async_rx_get_buf_len(async_rx);
464 		int err;
465 
466 		__ASSERT_NO_MSG(buf != NULL);
467 		atomic_dec(&sh_uart->pending_rx_req);
468 		err = uart_rx_buf_rsp(sh_uart->common.dev, buf, len);
469 		/* If it is too late and RX is disabled then re-enable it. */
470 		if (err < 0) {
471 			if (err == -EACCES) {
472 				sh_uart->pending_rx_req = 0;
473 				err = rx_enable(sh_uart->common.dev, buf, len);
474 			} else {
475 				return err;
476 			}
477 		}
478 	}
479 
480 	return 0;
481 }
482 
read_uart(const struct shell_transport * transport,void * data,size_t length,size_t * cnt)483 static int read_uart(const struct shell_transport *transport,
484 		     void *data, size_t length, size_t *cnt)
485 {
486 	if (IS_ENABLED(CONFIG_SHELL_BACKEND_SERIAL_API_INTERRUPT_DRIVEN)) {
487 		return irq_read((struct shell_uart_int_driven *)transport->ctx, data, length, cnt);
488 	} else if (IS_ENABLED(CONFIG_SHELL_BACKEND_SERIAL_API_ASYNC)) {
489 		return async_read((struct shell_uart_async *)transport->ctx, data, length, cnt);
490 	} else {
491 		return polling_read((struct shell_uart_polling *)transport->ctx, data, length, cnt);
492 	}
493 }
494 
495 #ifdef CONFIG_MCUMGR_TRANSPORT_SHELL
update(const struct shell_transport * transport)496 static void update(const struct shell_transport *transport)
497 {
498 	/*
499 	 * This is dependent on the fact that `struct shell_uart_common`
500 	 * is always the first member, regardless of the UART configuration
501 	 */
502 	struct shell_uart_common *sh_uart = (struct shell_uart_common *)transport->ctx;
503 
504 	smp_shell_process(&sh_uart->smp);
505 }
506 #endif /* CONFIG_MCUMGR_TRANSPORT_SHELL */
507 
508 const struct shell_transport_api shell_uart_transport_api = {
509 	.init = init,
510 	.uninit = uninit,
511 	.enable = enable,
512 	.write = write_uart,
513 	.read = read_uart,
514 #ifdef CONFIG_MCUMGR_TRANSPORT_SHELL
515 	.update = update,
516 #endif /* CONFIG_MCUMGR_TRANSPORT_SHELL */
517 };
518 
519 SHELL_UART_DEFINE(shell_transport_uart);
520 SHELL_DEFINE(shell_uart, CONFIG_SHELL_PROMPT_UART, &shell_transport_uart,
521 	     CONFIG_SHELL_BACKEND_SERIAL_LOG_MESSAGE_QUEUE_SIZE,
522 	     CONFIG_SHELL_BACKEND_SERIAL_LOG_MESSAGE_QUEUE_TIMEOUT,
523 	     SHELL_FLAG_OLF_CRLF);
524 
525 #ifdef CONFIG_MCUMGR_TRANSPORT_SHELL
shell_uart_smp_shell_data_get_ptr(void)526 struct smp_shell_data *shell_uart_smp_shell_data_get_ptr(void)
527 {
528 	struct shell_uart_common *common = (struct shell_uart_common *)shell_transport_uart.ctx;
529 
530 	return &common->smp;
531 }
532 #endif
533 
enable_shell_uart(void)534 static int enable_shell_uart(void)
535 {
536 	const struct device *const dev = DEVICE_DT_GET(DT_CHOSEN(zephyr_shell_uart));
537 	bool log_backend = CONFIG_SHELL_BACKEND_SERIAL_LOG_LEVEL > 0;
538 	uint32_t level =
539 		(CONFIG_SHELL_BACKEND_SERIAL_LOG_LEVEL > LOG_LEVEL_DBG) ?
540 		CONFIG_LOG_MAX_LEVEL : CONFIG_SHELL_BACKEND_SERIAL_LOG_LEVEL;
541 	static const struct shell_backend_config_flags cfg_flags =
542 					SHELL_DEFAULT_BACKEND_CONFIG_FLAGS;
543 
544 	if (!device_is_ready(dev)) {
545 		return -ENODEV;
546 	}
547 
548 	if (IS_ENABLED(CONFIG_MCUMGR_TRANSPORT_SHELL)) {
549 		smp_shell_init();
550 	}
551 
552 	shell_init(&shell_uart, dev, cfg_flags, log_backend, level);
553 
554 	return 0;
555 }
556 
557 SYS_INIT(enable_shell_uart, POST_KERNEL,
558 	 CONFIG_SHELL_BACKEND_SERIAL_INIT_PRIORITY);
559 
shell_backend_uart_get_ptr(void)560 const struct shell *shell_backend_uart_get_ptr(void)
561 {
562 	return &shell_uart;
563 }
564