1 /*
2 * Copyright Runtime.io 2018. All rights reserved.
3 * Copyright (c) 2021-2022 Nordic Semiconductor ASA
4 *
5 * SPDX-License-Identifier: Apache-2.0
6 */
7
8 #include <assert.h>
9 #include <zephyr/kernel.h>
10 #include <zephyr/device.h>
11 #include <zephyr/net_buf.h>
12 #include <zephyr/mgmt/mcumgr/mgmt/mgmt.h>
13 #include <zephyr/mgmt/mcumgr/smp/smp.h>
14 #include <zephyr/mgmt/mcumgr/transport/smp.h>
15
16 #include <mgmt/mcumgr/transport/smp_reassembly.h>
17
18 #include <zephyr/logging/log.h>
19 LOG_MODULE_REGISTER(mcumgr_smp, CONFIG_MCUMGR_TRANSPORT_LOG_LEVEL);
20
21 /* To be able to unit test some callers some functions need to be
22 * demoted to allow overriding them.
23 */
24 #ifdef CONFIG_ZTEST
25 #define WEAK __weak
26 #else
27 #define WEAK
28 #endif
29
30 K_THREAD_STACK_DEFINE(smp_work_queue_stack, CONFIG_MCUMGR_TRANSPORT_WORKQUEUE_STACK_SIZE);
31
32 static struct k_work_q smp_work_queue;
33
34 #ifdef CONFIG_SMP_CLIENT
35 static sys_slist_t smp_transport_clients = SYS_SLIST_STATIC_INIT(&smp_transport_clients);
36 #endif
37
38 static const struct k_work_queue_config smp_work_queue_config = {
39 .name = "mcumgr smp"
40 };
41
42 NET_BUF_POOL_DEFINE(pkt_pool, CONFIG_MCUMGR_TRANSPORT_NETBUF_COUNT,
43 CONFIG_MCUMGR_TRANSPORT_NETBUF_SIZE,
44 CONFIG_MCUMGR_TRANSPORT_NETBUF_USER_DATA_SIZE, NULL);
45
smp_packet_alloc(void)46 struct net_buf *smp_packet_alloc(void)
47 {
48 return net_buf_alloc(&pkt_pool, K_NO_WAIT);
49 }
50
smp_packet_free(struct net_buf * nb)51 void smp_packet_free(struct net_buf *nb)
52 {
53 net_buf_unref(nb);
54 }
55
56 /**
57 * @brief Allocates a response buffer.
58 *
59 * If a source buf is provided, its user data is copied into the new buffer.
60 *
61 * @param req An optional source buffer to copy user data from.
62 * @param arg The streamer providing the callback.
63 *
64 * @return Newly-allocated buffer on success
65 * NULL on failure.
66 */
smp_alloc_rsp(const void * req,void * arg)67 void *smp_alloc_rsp(const void *req, void *arg)
68 {
69 const struct net_buf *req_nb;
70 struct net_buf *rsp_nb;
71 struct smp_transport *smpt = arg;
72
73 req_nb = req;
74
75 rsp_nb = smp_packet_alloc();
76 if (rsp_nb == NULL) {
77 return NULL;
78 }
79
80 if (smpt->functions.ud_copy) {
81 smpt->functions.ud_copy(rsp_nb, req_nb);
82 } else {
83 memcpy(net_buf_user_data(rsp_nb),
84 net_buf_user_data((void *)req_nb),
85 req_nb->user_data_size);
86 }
87
88 return rsp_nb;
89 }
90
smp_free_buf(void * buf,void * arg)91 void smp_free_buf(void *buf, void *arg)
92 {
93 struct smp_transport *smpt = arg;
94
95 if (!buf) {
96 return;
97 }
98
99 if (smpt->functions.ud_free) {
100 smpt->functions.ud_free(net_buf_user_data((struct net_buf *)buf));
101 }
102
103 smp_packet_free(buf);
104 }
105
106 /**
107 * Processes a single SMP packet and sends the corresponding response(s).
108 */
109 static int
smp_process_packet(struct smp_transport * smpt,struct net_buf * nb)110 smp_process_packet(struct smp_transport *smpt, struct net_buf *nb)
111 {
112 struct cbor_nb_reader reader;
113 struct cbor_nb_writer writer;
114 struct smp_streamer streamer;
115 int rc;
116
117 streamer = (struct smp_streamer) {
118 .reader = &reader,
119 .writer = &writer,
120 .smpt = smpt,
121 };
122
123 rc = smp_process_request_packet(&streamer, nb);
124 return rc;
125 }
126
127 /**
128 * Processes all received SNP request packets.
129 */
130 static void
smp_handle_reqs(struct k_work * work)131 smp_handle_reqs(struct k_work *work)
132 {
133 struct smp_transport *smpt;
134 struct net_buf *nb;
135
136 smpt = (void *)work;
137
138 /* Read and handle received messages */
139 while ((nb = k_fifo_get(&smpt->fifo, K_NO_WAIT)) != NULL) {
140 smp_process_packet(smpt, nb);
141 }
142 }
143
smp_transport_init(struct smp_transport * smpt)144 int smp_transport_init(struct smp_transport *smpt)
145 {
146 __ASSERT((smpt->functions.output != NULL),
147 "Required transport output function pointer cannot be NULL");
148
149 if (smpt->functions.output == NULL) {
150 return -EINVAL;
151 }
152
153 #ifdef CONFIG_MCUMGR_TRANSPORT_REASSEMBLY
154 smp_reassembly_init(smpt);
155 #endif
156
157 k_work_init(&smpt->work, smp_handle_reqs);
158 k_fifo_init(&smpt->fifo);
159
160 return 0;
161 }
162
163 #ifdef CONFIG_SMP_CLIENT
smp_client_transport_get(int smpt_type)164 struct smp_transport *smp_client_transport_get(int smpt_type)
165 {
166 struct smp_client_transport_entry *entry;
167
168 SYS_SLIST_FOR_EACH_CONTAINER(&smp_transport_clients, entry, node) {
169 if (entry->smpt_type == smpt_type) {
170 return entry->smpt;
171 }
172 }
173
174 return NULL;
175 }
176
smp_client_transport_register(struct smp_client_transport_entry * entry)177 void smp_client_transport_register(struct smp_client_transport_entry *entry)
178 {
179 if (smp_client_transport_get(entry->smpt_type)) {
180 /* Already in list */
181 return;
182 }
183
184 sys_slist_append(&smp_transport_clients, &entry->node);
185
186 }
187
188 #endif /* CONFIG_SMP_CLIENT */
189
190 /**
191 * @brief Enqueues an incoming SMP request packet for processing.
192 *
193 * This function always consumes the supplied net_buf.
194 *
195 * @param smpt The transport to use to send the corresponding
196 * response(s).
197 * @param nb The request packet to process.
198 */
199 WEAK void
smp_rx_req(struct smp_transport * smpt,struct net_buf * nb)200 smp_rx_req(struct smp_transport *smpt, struct net_buf *nb)
201 {
202 k_fifo_put(&smpt->fifo, nb);
203 k_work_submit_to_queue(&smp_work_queue, &smpt->work);
204 }
205
206 #ifdef CONFIG_SMP_CLIENT
smp_get_wq(void)207 struct k_work_q *smp_get_wq(void)
208 {
209 return &smp_work_queue;
210 }
211 #endif
212
smp_rx_remove_invalid(struct smp_transport * zst,void * arg)213 void smp_rx_remove_invalid(struct smp_transport *zst, void *arg)
214 {
215 struct net_buf *nb;
216 struct k_fifo temp_fifo;
217
218 if (zst->functions.query_valid_check == NULL) {
219 /* No check check function registered, abort check */
220 return;
221 }
222
223 /* Cancel current work-queue if ongoing */
224 if (k_work_busy_get(&zst->work) & (K_WORK_RUNNING | K_WORK_QUEUED)) {
225 k_work_cancel(&zst->work);
226 }
227
228 /* Run callback function and remove all buffers that are no longer needed. Store those
229 * that are in a temporary FIFO
230 */
231 k_fifo_init(&temp_fifo);
232
233 while ((nb = k_fifo_get(&zst->fifo, K_NO_WAIT)) != NULL) {
234 if (!zst->functions.query_valid_check(nb, arg)) {
235 smp_free_buf(nb, zst);
236 } else {
237 k_fifo_put(&temp_fifo, nb);
238 }
239 }
240
241 /* Re-insert the remaining queued operations into the original FIFO */
242 while ((nb = k_fifo_get(&temp_fifo, K_NO_WAIT)) != NULL) {
243 k_fifo_put(&zst->fifo, nb);
244 }
245
246 /* If at least one entry remains, queue the workqueue for running */
247 if (!k_fifo_is_empty(&zst->fifo)) {
248 k_work_submit_to_queue(&smp_work_queue, &zst->work);
249 }
250 }
251
smp_rx_clear(struct smp_transport * zst)252 void smp_rx_clear(struct smp_transport *zst)
253 {
254 struct net_buf *nb;
255
256 /* Cancel current work-queue if ongoing */
257 if (k_work_busy_get(&zst->work) & (K_WORK_RUNNING | K_WORK_QUEUED)) {
258 k_work_cancel(&zst->work);
259 }
260
261 /* Drain the FIFO of all entries without re-adding any */
262 while ((nb = k_fifo_get(&zst->fifo, K_NO_WAIT)) != NULL) {
263 smp_free_buf(nb, zst);
264 }
265 }
266
smp_init(void)267 static int smp_init(void)
268 {
269 k_work_queue_init(&smp_work_queue);
270
271 k_work_queue_start(&smp_work_queue, smp_work_queue_stack,
272 K_THREAD_STACK_SIZEOF(smp_work_queue_stack),
273 CONFIG_MCUMGR_TRANSPORT_WORKQUEUE_THREAD_PRIO, &smp_work_queue_config);
274
275 return 0;
276 }
277
278 SYS_INIT(smp_init, APPLICATION, CONFIG_APPLICATION_INIT_PRIORITY);
279