/* * Copyright Runtime.io 2018. All rights reserved. * * SPDX-License-Identifier: Apache-2.0 */ /** @file * @brief UART transport for the mcumgr SMP protocol. */ #include #include #include #include #include #include #include #include #include #include BUILD_ASSERT(CONFIG_MCUMGR_TRANSPORT_UART_MTU != 0, "CONFIG_MCUMGR_TRANSPORT_UART_MTU must be > 0"); struct device; static void smp_uart_process_rx_queue(struct k_work *work); K_FIFO_DEFINE(smp_uart_rx_fifo); K_WORK_DEFINE(smp_uart_work, smp_uart_process_rx_queue); static struct mcumgr_serial_rx_ctxt smp_uart_rx_ctxt; static struct smp_transport smp_uart_transport; #ifdef CONFIG_SMP_CLIENT static struct smp_client_transport_entry smp_client_transport; #endif /** * Processes a single line (fragment) coming from the mcumgr UART driver. */ static void smp_uart_process_frag(struct uart_mcumgr_rx_buf *rx_buf) { struct net_buf *nb; /* Decode the fragment and write the result to the global receive * context. */ nb = mcumgr_serial_process_frag(&smp_uart_rx_ctxt, rx_buf->data, rx_buf->length); /* Release the encoded fragment. */ uart_mcumgr_free_rx_buf(rx_buf); /* If a complete packet has been received, pass it to SMP for * processing. */ if (nb != NULL) { smp_rx_req(&smp_uart_transport, nb); } } static void smp_uart_process_rx_queue(struct k_work *work) { struct uart_mcumgr_rx_buf *rx_buf; while ((rx_buf = k_fifo_get(&smp_uart_rx_fifo, K_NO_WAIT)) != NULL) { smp_uart_process_frag(rx_buf); } } /** * Enqueues a received SMP fragment for later processing. This function * executes in the interrupt context. */ static void smp_uart_rx_frag(struct uart_mcumgr_rx_buf *rx_buf) { k_fifo_put(&smp_uart_rx_fifo, rx_buf); k_work_submit(&smp_uart_work); } static uint16_t smp_uart_get_mtu(const struct net_buf *nb) { return CONFIG_MCUMGR_TRANSPORT_UART_MTU; } static int smp_uart_tx_pkt(struct net_buf *nb) { int rc; rc = uart_mcumgr_send(nb->data, nb->len); smp_packet_free(nb); return rc; } static int smp_uart_init(void) { int rc; smp_uart_transport.functions.output = smp_uart_tx_pkt; smp_uart_transport.functions.get_mtu = smp_uart_get_mtu; rc = smp_transport_init(&smp_uart_transport); if (rc == 0) { uart_mcumgr_register(smp_uart_rx_frag); #ifdef CONFIG_SMP_CLIENT smp_client_transport.smpt = &smp_uart_transport; smp_client_transport.smpt_type = SMP_SERIAL_TRANSPORT; smp_client_transport_register(&smp_client_transport); #endif } return rc; } SYS_INIT(smp_uart_init, APPLICATION, CONFIG_APPLICATION_INIT_PRIORITY);