1 /*
2 * Copyright (c) 2020 Nordic Semiconductor ASA
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #include <zephyr/settings/settings.h>
8 #include "dfu_slot.h"
9 #include <string.h>
10 #include <stdlib.h>
11 #include <stdio.h>
12 #include <zephyr/sys/util.h>
13 #include <common/bt_str.h>
14
15 #define LOG_LEVEL CONFIG_BT_MESH_DFU_LOG_LEVEL
16 #include <zephyr/logging/log.h>
17 LOG_MODULE_REGISTER(bt_mesh_dfu_slot);
18
19 #define SLOT_ENTRY_BUFLEN 25
20
21 #define DFU_SLOT_SETTINGS_PATH "bt/mesh-dfu/slot"
22
23 #define HEADER_SIZE offsetof(struct slot, slot.fwid)
24
25 #define PROP_HEADER "h"
26 #define PROP_FWID "id"
27 #define PROP_METADATA "m"
28
29 static sys_slist_t list;
30
31 static struct slot {
32 uint32_t idx;
33 struct bt_mesh_dfu_slot slot;
34 sys_snode_t n;
35 } slots[CONFIG_BT_MESH_DFU_SLOT_CNT];
36
37 static uint32_t slot_index;
38
slot_entry_encode(uint16_t idx,char buf[SLOT_ENTRY_BUFLEN],const char * property)39 static char *slot_entry_encode(uint16_t idx, char buf[SLOT_ENTRY_BUFLEN],
40 const char *property)
41 {
42 snprintf(buf, SLOT_ENTRY_BUFLEN, DFU_SLOT_SETTINGS_PATH "/%x/%s", idx,
43 property);
44
45 return buf;
46 }
47
slot_eq(const struct bt_mesh_dfu_slot * slot,const uint8_t * fwid,size_t fwid_len)48 static bool slot_eq(const struct bt_mesh_dfu_slot *slot,
49 const uint8_t *fwid, size_t fwid_len)
50 {
51 return (slot->fwid_len == fwid_len) &&
52 !memcmp(fwid, slot->fwid, fwid_len);
53 }
54
is_slot_committed(struct slot * slot_to_check)55 static bool is_slot_committed(struct slot *slot_to_check)
56 {
57 struct slot *s;
58
59 SYS_SLIST_FOR_EACH_CONTAINER(&list, s, n) {
60 if (s == slot_to_check) {
61 return true;
62 }
63 }
64
65 return false;
66 }
67
slot_store(const struct slot * slot_to_store)68 static int slot_store(const struct slot *slot_to_store)
69 {
70 uint16_t idx = ARRAY_INDEX(slots, slot_to_store);
71 char buf[SLOT_ENTRY_BUFLEN];
72 int err;
73
74 err = settings_save_one(slot_entry_encode(idx, buf, PROP_HEADER),
75 slot_to_store, HEADER_SIZE);
76 if (err) {
77 return err;
78 }
79
80 err = settings_save_one(slot_entry_encode(idx, buf, PROP_FWID),
81 slot_to_store->slot.fwid, slot_to_store->slot.fwid_len);
82 if (err) {
83 return err;
84 }
85
86 err = settings_save_one(slot_entry_encode(idx, buf,
87 PROP_METADATA),
88 slot_to_store->slot.metadata, slot_to_store->slot.metadata_len);
89
90 return err;
91 }
92
slot_erase(struct slot * slot_to_erase)93 static void slot_erase(struct slot *slot_to_erase)
94 {
95 uint16_t idx = ARRAY_INDEX(slots, slot_to_erase);
96 char buf[SLOT_ENTRY_BUFLEN];
97
98 (void)settings_delete(slot_entry_encode(idx, buf, PROP_HEADER));
99 (void)settings_delete(slot_entry_encode(idx, buf, PROP_FWID));
100 (void)settings_delete(slot_entry_encode(idx, buf, PROP_METADATA));
101 }
102
slot_index_defrag(void)103 static void slot_index_defrag(void)
104 {
105 slot_index = 0;
106 struct slot *s;
107
108 SYS_SLIST_FOR_EACH_CONTAINER(&list, s, n) {
109 s->idx = ++slot_index;
110 slot_store(s);
111 }
112 }
113
bt_mesh_dfu_slot_count(void)114 int bt_mesh_dfu_slot_count(void)
115 {
116 int cnt = 0;
117 sys_snode_t *n;
118
119 SYS_SLIST_FOR_EACH_NODE(&list, n) {
120 cnt++;
121 }
122
123 return cnt;
124 }
125
bt_mesh_dfu_slot_reserve(void)126 struct bt_mesh_dfu_slot *bt_mesh_dfu_slot_reserve(void)
127 {
128 struct slot *slot = NULL;
129
130 for (int i = 0; i < ARRAY_SIZE(slots); ++i) {
131 if (slots[i].idx == 0) {
132 slot = &slots[i];
133 break;
134 }
135 }
136
137 if (!slot) {
138 LOG_WRN("No space");
139 return NULL;
140 }
141
142 if (slot_index == UINT32_MAX) {
143 slot_index_defrag();
144 }
145
146 slot->slot.fwid_len = 0;
147 slot->slot.metadata_len = 0;
148 slot->slot.size = 0;
149 slot->idx = ++slot_index;
150
151 LOG_DBG("Reserved slot #%u", slot - &slots[0]);
152
153 return &slot->slot;
154 }
155
bt_mesh_dfu_slot_info_set(struct bt_mesh_dfu_slot * dfu_slot,size_t size,const uint8_t * metadata,size_t metadata_len)156 int bt_mesh_dfu_slot_info_set(struct bt_mesh_dfu_slot *dfu_slot, size_t size,
157 const uint8_t *metadata, size_t metadata_len)
158 {
159 struct slot *slot = CONTAINER_OF(dfu_slot, struct slot, slot);
160
161 if (metadata_len > CONFIG_BT_MESH_DFU_METADATA_MAXLEN) {
162 return -EFBIG;
163 }
164
165 if (slot->idx == 0 || is_slot_committed(slot)) {
166 return -EINVAL;
167 }
168
169 slot->slot.size = size;
170 slot->slot.metadata_len = metadata_len;
171 memcpy(slot->slot.metadata, metadata, metadata_len);
172 return 0;
173 }
174
bt_mesh_dfu_slot_fwid_set(struct bt_mesh_dfu_slot * dfu_slot,const uint8_t * fwid,size_t fwid_len)175 int bt_mesh_dfu_slot_fwid_set(struct bt_mesh_dfu_slot *dfu_slot,
176 const uint8_t *fwid, size_t fwid_len)
177 {
178 struct slot *slot = CONTAINER_OF(dfu_slot, struct slot, slot);
179
180 if (fwid_len > CONFIG_BT_MESH_DFU_FWID_MAXLEN) {
181 return -EFBIG;
182 }
183
184 if (slot->idx == 0 || is_slot_committed(slot)) {
185 return -EINVAL;
186 }
187
188 for (int i = 0; i < ARRAY_SIZE(slots); i++) {
189 if (slots[i].idx != 0 &&
190 slot_eq(&slots[i].slot, fwid, fwid_len)) {
191 return is_slot_committed(&slots[i]) ?
192 -EEXIST : -EALREADY;
193 }
194 }
195
196 slot->slot.fwid_len = fwid_len;
197 memcpy(slot->slot.fwid, fwid, fwid_len);
198 return 0;
199 }
200
bt_mesh_dfu_slot_commit(struct bt_mesh_dfu_slot * dfu_slot)201 int bt_mesh_dfu_slot_commit(struct bt_mesh_dfu_slot *dfu_slot)
202 {
203 int err;
204 struct slot *slot = CONTAINER_OF(dfu_slot, struct slot, slot);
205
206 if (slot->idx == 0 ||
207 slot->slot.fwid_len == 0 ||
208 slot->slot.size == 0 ||
209 is_slot_committed(slot)) {
210 return -EINVAL;
211 }
212
213 err = slot_store(slot);
214 if (err) {
215 LOG_WRN("Store failed (err: %d)", err);
216 return err;
217 }
218
219 sys_slist_append(&list, &slot->n);
220
221 LOG_DBG("Stored slot #%u: %s", ARRAY_INDEX(slots, slot),
222 bt_hex(slot->slot.fwid, slot->slot.fwid_len));
223 return 0;
224 }
225
bt_mesh_dfu_slot_release(const struct bt_mesh_dfu_slot * dfu_slot)226 void bt_mesh_dfu_slot_release(const struct bt_mesh_dfu_slot *dfu_slot)
227 {
228 struct slot *slot = CONTAINER_OF(dfu_slot, struct slot, slot);
229
230 if (is_slot_committed(slot)) {
231 return;
232 }
233
234 slot->idx = 0;
235 }
236
bt_mesh_dfu_slot_del(const struct bt_mesh_dfu_slot * dfu_slot)237 int bt_mesh_dfu_slot_del(const struct bt_mesh_dfu_slot *dfu_slot)
238 {
239 struct slot *slot = CONTAINER_OF(dfu_slot, struct slot, slot);
240
241 if (!sys_slist_find_and_remove(&list, &slot->n)) {
242 return -EINVAL;
243 }
244
245 int idx = ARRAY_INDEX(slots, slot);
246
247 LOG_DBG("%u", idx);
248
249 slot_erase(slot);
250 slot->idx = 0;
251
252 return 0;
253 }
254
bt_mesh_dfu_slot_del_all(void)255 void bt_mesh_dfu_slot_del_all(void)
256 {
257 struct slot *s;
258 SYS_SLIST_FOR_EACH_CONTAINER(&list, s, n) {
259 slot_erase(s);
260 s->idx = 0;
261 }
262
263 sys_slist_init(&list);
264 }
265
bt_mesh_dfu_slot_at(uint16_t img_idx)266 const struct bt_mesh_dfu_slot *bt_mesh_dfu_slot_at(uint16_t img_idx)
267 {
268 struct slot *s;
269
270 SYS_SLIST_FOR_EACH_CONTAINER(&list, s, n) {
271 if (!img_idx--) {
272 return &s->slot;
273 }
274 }
275
276 return NULL;
277 }
278
bt_mesh_dfu_slot_get(const uint8_t * fwid,size_t fwid_len,struct bt_mesh_dfu_slot ** slot)279 int bt_mesh_dfu_slot_get(const uint8_t *fwid, size_t fwid_len, struct bt_mesh_dfu_slot **slot)
280 {
281 struct slot *s;
282 int idx = 0;
283
284 SYS_SLIST_FOR_EACH_CONTAINER(&list, s, n) {
285 if (slot_eq(&s->slot, fwid, fwid_len)) {
286 if (slot) {
287 *slot = &s->slot;
288 }
289 return idx;
290 }
291 idx++;
292 }
293
294 return -ENOENT;
295 }
296
bt_mesh_dfu_slot_img_idx_get(const struct bt_mesh_dfu_slot * dfu_slot)297 int bt_mesh_dfu_slot_img_idx_get(const struct bt_mesh_dfu_slot *dfu_slot)
298 {
299 struct slot *s;
300 int idx = 0;
301
302 SYS_SLIST_FOR_EACH_CONTAINER(&list, s, n) {
303 if (&s->slot == dfu_slot) {
304 return idx;
305 }
306 idx++;
307 }
308
309 return -ENOENT;
310 }
311
bt_mesh_dfu_slot_foreach(bt_mesh_dfu_slot_cb_t cb,void * user_data)312 size_t bt_mesh_dfu_slot_foreach(bt_mesh_dfu_slot_cb_t cb, void *user_data)
313 {
314 enum bt_mesh_dfu_iter iter;
315 size_t cnt = 0;
316 struct slot *s;
317
318 SYS_SLIST_FOR_EACH_CONTAINER(&list, s, n) {
319 cnt++;
320
321 if (!cb) {
322 continue;
323 }
324
325 iter = cb(&s->slot, user_data);
326 if (iter != BT_MESH_DFU_ITER_CONTINUE) {
327 break;
328 }
329 }
330
331 return cnt;
332 }
333
slot_data_load(const char * key,size_t len_rd,settings_read_cb read_cb,void * cb_arg)334 static int slot_data_load(const char *key, size_t len_rd,
335 settings_read_cb read_cb, void *cb_arg)
336 {
337 const char *prop;
338 size_t len;
339 uint16_t idx;
340
341 idx = strtol(key, NULL, 16);
342
343 if (idx >= ARRAY_SIZE(slots)) {
344 return 0;
345 }
346
347 len = settings_name_next(key, &prop);
348
349 if (!strncmp(prop, PROP_HEADER, len)) {
350 if (read_cb(cb_arg, &slots[idx], HEADER_SIZE) > 0) {
351 struct slot *s, *prev = NULL;
352
353 SYS_SLIST_FOR_EACH_CONTAINER(&list, s, n) {
354 if (s->idx > slots[idx].idx) {
355 break;
356 }
357
358 prev = s;
359 }
360
361 if (prev == NULL) {
362 sys_slist_prepend(&list, &slots[idx].n);
363 } else {
364 sys_slist_insert(&list, &prev->n, &slots[idx].n);
365 }
366
367 if (slots[idx].idx >= slot_index) {
368 slot_index = slots[idx].idx + 1;
369 }
370 }
371 return 0;
372 }
373
374 if (!strncmp(prop, PROP_FWID, len)) {
375 if (read_cb(cb_arg, &slots[idx].slot.fwid,
376 sizeof(slots[idx].slot.fwid)) < 0) {
377 slots[idx].idx = 0;
378 sys_slist_find_and_remove(&list, &slots[idx].n);
379 return 0;
380 }
381
382 slots[idx].slot.fwid_len = len_rd;
383 return 0;
384 }
385
386 if (!strncmp(prop, PROP_METADATA, len)) {
387 if (read_cb(cb_arg, &slots[idx].slot.metadata,
388 sizeof(slots[idx].slot.metadata)) < 0) {
389 slots[idx].idx = 0;
390 sys_slist_find_and_remove(&list, &slots[idx].n);
391 return 0;
392 }
393
394 slots[idx].slot.metadata_len = len_rd;
395 return 0;
396 }
397
398 return 0;
399 }
400
401 SETTINGS_STATIC_HANDLER_DEFINE(bt_mesh_dfu_slots, DFU_SLOT_SETTINGS_PATH, NULL,
402 slot_data_load, NULL, NULL);
403