1 /*
2 * Copyright (c) 2022 Nordic Semiconductor
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6 #include "mesh_test.h"
7 #include "mesh/dfd_srv_internal.h"
8 #include "mesh/dfu_slot.h"
9 #include "mesh/dfu.h"
10 #include "mesh/blob.h"
11 #include "argparse.h"
12 #include "dfu_blob_common.h"
13
14 #define LOG_MODULE_NAME test_dfu
15
16 #include <zephyr/logging/log.h>
17 LOG_MODULE_REGISTER(LOG_MODULE_NAME, LOG_LEVEL_INF);
18
19 #define WAIT_TIME 420 /* seconds */
20 #define DFU_TIMEOUT 400 /* seconds */
21 #define DIST_ADDR 0x0001
22 #define TARGET_ADDR 0x0100
23 #define IMPOSTER_MODEL_ID 0xe000
24 #define TEST_BLOB_ID 0xaabbccdd
25 #define SEMAPHORE_TIMEOUT 250 /* seconds */
26
27 struct bind_params {
28 uint16_t model_id;
29 uint16_t addr;
30 };
31
32 static uint8_t dev_key[16] = { 0xdd };
33
34 static struct k_sem dfu_dist_ended;
35 static struct k_sem dfu_started;
36 static struct k_sem dfu_verifying;
37 static struct k_sem dfu_verify_failed;
38 static struct k_sem dfu_applying;
39 static struct k_sem dfu_ended;
40
41 static struct bt_mesh_prov prov;
42
43 static enum bt_mesh_dfu_effect dfu_target_effect;
44 static uint32_t target_fw_ver_curr = 0xDEADBEEF;
45 static uint32_t target_fw_ver_new;
46 static struct bt_mesh_dfu_img dfu_imgs[] = { {
47 .fwid = &target_fw_ver_curr,
48 .fwid_len = sizeof(target_fw_ver_curr),
49 } };
50
51 static struct bt_mesh_cfg_cli cfg_cli;
52 static struct bt_mesh_sar_cfg_cli sar_cfg_cli;
53
54 static int dfu_targets_cnt;
55 static bool dfu_fail_confirm;
56 static bool recover;
57 static bool expect_fail;
58 static enum bt_mesh_dfu_phase expected_stop_phase;
59
test_args_parse(int argc,char * argv[])60 static void test_args_parse(int argc, char *argv[])
61 {
62 bs_args_struct_t args_struct[] = {
63 {
64 .dest = &dfu_targets_cnt,
65 .type = 'i',
66 .name = "{targets}",
67 .option = "targets",
68 .descript = "Number of targets to upgrade"
69 },
70 {
71 .dest = &dfu_fail_confirm,
72 .type = 'b',
73 .name = "{0, 1}",
74 .option = "fail-confirm",
75 .descript = "Request target to fail confirm step"
76 },
77 {
78 .dest = &expected_stop_phase,
79 .type = 'i',
80 .name = "{none, start, verify, verify-ok, verify-fail, apply}",
81 .option = "expected-phase",
82 .descript = "Expected DFU Server phase value restored from flash"
83 },
84 {
85 .dest = &recover,
86 .type = 'b',
87 .name = "{0, 1}",
88 .option = "recover",
89 .descript = "Recover DFU server phase"
90 },
91 };
92
93 bs_args_parse_all_cmd_line(argc, argv, args_struct);
94 }
95
dummy_blob_chunk_wr(const struct bt_mesh_blob_io * io,const struct bt_mesh_blob_xfer * xfer,const struct bt_mesh_blob_block * block,const struct bt_mesh_blob_chunk * chunk)96 static int dummy_blob_chunk_wr(const struct bt_mesh_blob_io *io,
97 const struct bt_mesh_blob_xfer *xfer,
98 const struct bt_mesh_blob_block *block,
99 const struct bt_mesh_blob_chunk *chunk)
100 {
101 return 0;
102 }
103
dummy_blob_chunk_rd(const struct bt_mesh_blob_io * io,const struct bt_mesh_blob_xfer * xfer,const struct bt_mesh_blob_block * block,const struct bt_mesh_blob_chunk * chunk)104 static int dummy_blob_chunk_rd(const struct bt_mesh_blob_io *io,
105 const struct bt_mesh_blob_xfer *xfer,
106 const struct bt_mesh_blob_block *block,
107 const struct bt_mesh_blob_chunk *chunk)
108 {
109 memset(chunk->data, 0, chunk->size);
110
111 return 0;
112 }
113
114 static const struct bt_mesh_blob_io dummy_blob_io = {
115 .rd = dummy_blob_chunk_rd,
116 .wr = dummy_blob_chunk_wr,
117 };
118
dist_fw_recv(struct bt_mesh_dfd_srv * srv,const struct bt_mesh_dfu_slot * slot,const struct bt_mesh_blob_io ** io)119 static int dist_fw_recv(struct bt_mesh_dfd_srv *srv,
120 const struct bt_mesh_dfu_slot *slot,
121 const struct bt_mesh_blob_io **io)
122 {
123 *io = &dummy_blob_io;
124
125 return 0;
126 }
127
dist_fw_del(struct bt_mesh_dfd_srv * srv,const struct bt_mesh_dfu_slot * slot)128 static void dist_fw_del(struct bt_mesh_dfd_srv *srv,
129 const struct bt_mesh_dfu_slot *slot)
130 {
131 }
132
dist_fw_send(struct bt_mesh_dfd_srv * srv,const struct bt_mesh_dfu_slot * slot,const struct bt_mesh_blob_io ** io)133 static int dist_fw_send(struct bt_mesh_dfd_srv *srv,
134 const struct bt_mesh_dfu_slot *slot,
135 const struct bt_mesh_blob_io **io)
136 {
137 *io = &dummy_blob_io;
138
139 return 0;
140 }
141
dist_phase_changed(struct bt_mesh_dfd_srv * srv,enum bt_mesh_dfd_phase phase)142 static void dist_phase_changed(struct bt_mesh_dfd_srv *srv, enum bt_mesh_dfd_phase phase)
143 {
144 static enum bt_mesh_dfd_phase prev_phase;
145
146 if (phase == BT_MESH_DFD_PHASE_COMPLETED ||
147 phase == BT_MESH_DFD_PHASE_FAILED) {
148 if (phase == BT_MESH_DFD_PHASE_FAILED) {
149 ASSERT_EQUAL(BT_MESH_DFD_PHASE_APPLYING_UPDATE, prev_phase);
150 }
151
152 k_sem_give(&dfu_dist_ended);
153 }
154
155 prev_phase = phase;
156 }
157
158 static struct bt_mesh_dfd_srv_cb dfd_srv_cb = {
159 .recv = dist_fw_recv,
160 .del = dist_fw_del,
161 .send = dist_fw_send,
162 .phase = dist_phase_changed,
163 };
164
165 struct bt_mesh_dfd_srv dfd_srv = BT_MESH_DFD_SRV_INIT(&dfd_srv_cb);
166
167 static struct k_sem dfu_metadata_check_sem;
168 static bool dfu_metadata_fail = true;
169
target_metadata_check(struct bt_mesh_dfu_srv * srv,const struct bt_mesh_dfu_img * img,struct net_buf_simple * metadata_raw,enum bt_mesh_dfu_effect * effect)170 static int target_metadata_check(struct bt_mesh_dfu_srv *srv,
171 const struct bt_mesh_dfu_img *img,
172 struct net_buf_simple *metadata_raw,
173 enum bt_mesh_dfu_effect *effect)
174 {
175 *effect = dfu_target_effect;
176
177 memcpy(&target_fw_ver_new, net_buf_simple_pull_mem(metadata_raw, sizeof(target_fw_ver_new)),
178 sizeof(target_fw_ver_new));
179
180 k_sem_give(&dfu_metadata_check_sem);
181
182 return dfu_metadata_fail ? 0 : -1;
183 }
184
185 static bool expect_dfu_start = true;
186
target_dfu_start(struct bt_mesh_dfu_srv * srv,const struct bt_mesh_dfu_img * img,struct net_buf_simple * metadata,const struct bt_mesh_blob_io ** io)187 static int target_dfu_start(struct bt_mesh_dfu_srv *srv,
188 const struct bt_mesh_dfu_img *img,
189 struct net_buf_simple *metadata,
190 const struct bt_mesh_blob_io **io)
191 {
192 ASSERT_TRUE(expect_dfu_start);
193
194 *io = &dummy_blob_io;
195
196 if (expected_stop_phase == BT_MESH_DFU_PHASE_APPLYING) {
197 return -EALREADY;
198 }
199
200 return 0;
201 }
202
203 static struct k_sem dfu_verify_sem;
204 static bool dfu_verify_fail;
205 static bool expect_dfu_xfer_end = true;
206
target_dfu_transfer_end(struct bt_mesh_dfu_srv * srv,const struct bt_mesh_dfu_img * img,bool success)207 static void target_dfu_transfer_end(struct bt_mesh_dfu_srv *srv, const struct bt_mesh_dfu_img *img,
208 bool success)
209 {
210 ASSERT_TRUE(expect_dfu_xfer_end);
211 ASSERT_TRUE(success);
212
213 if (expected_stop_phase == BT_MESH_DFU_PHASE_VERIFY) {
214 k_sem_give(&dfu_verifying);
215 return;
216 }
217
218 if (dfu_verify_fail) {
219 bt_mesh_dfu_srv_rejected(srv);
220 if (expected_stop_phase == BT_MESH_DFU_PHASE_VERIFY_FAIL) {
221 k_sem_give(&dfu_verify_failed);
222 return;
223 }
224 } else {
225 bt_mesh_dfu_srv_verified(srv);
226 }
227
228 k_sem_give(&dfu_verify_sem);
229 }
230
target_dfu_recover(struct bt_mesh_dfu_srv * srv,const struct bt_mesh_dfu_img * img,const struct bt_mesh_blob_io ** io)231 static int target_dfu_recover(struct bt_mesh_dfu_srv *srv,
232 const struct bt_mesh_dfu_img *img,
233 const struct bt_mesh_blob_io **io)
234 {
235 if (!recover) {
236 FAIL("Not supported");
237 }
238
239 *io = &dummy_blob_io;
240
241 return 0;
242 }
243
244 static bool expect_dfu_apply = true;
245
target_dfu_apply(struct bt_mesh_dfu_srv * srv,const struct bt_mesh_dfu_img * img)246 static int target_dfu_apply(struct bt_mesh_dfu_srv *srv, const struct bt_mesh_dfu_img *img)
247 {
248 if (expected_stop_phase == BT_MESH_DFU_PHASE_VERIFY_OK) {
249 k_sem_give(&dfu_verifying);
250 } else if (expected_stop_phase == BT_MESH_DFU_PHASE_APPLYING) {
251 k_sem_give(&dfu_applying);
252 return 0;
253 }
254
255 ASSERT_TRUE(expect_dfu_apply);
256
257 bt_mesh_dfu_srv_applied(srv);
258
259 k_sem_give(&dfu_ended);
260
261 if (dfu_fail_confirm) {
262 /* To fail the confirm step, don't change fw version for devices that should boot
263 * up provisioned. Change fw version for devices that should boot up unprovisioned.
264 */
265 if (dfu_target_effect == BT_MESH_DFU_EFFECT_UNPROV) {
266 target_fw_ver_curr = target_fw_ver_new;
267 }
268 } else {
269 if (dfu_target_effect == BT_MESH_DFU_EFFECT_UNPROV) {
270 bt_mesh_reset();
271 }
272
273 target_fw_ver_curr = target_fw_ver_new;
274 }
275
276 return 0;
277 }
278
279 static const struct bt_mesh_dfu_srv_cb dfu_handlers = {
280 .check = target_metadata_check,
281 .start = target_dfu_start,
282 .end = target_dfu_transfer_end,
283 .apply = target_dfu_apply,
284 .recover = target_dfu_recover,
285 };
286
287 static struct bt_mesh_dfu_srv dfu_srv = BT_MESH_DFU_SRV_INIT(&dfu_handlers, dfu_imgs,
288 ARRAY_SIZE(dfu_imgs));
289
290 static const struct bt_mesh_comp dist_comp = {
291 .elem =
292 (const struct bt_mesh_elem[]){
293 BT_MESH_ELEM(1,
294 MODEL_LIST(BT_MESH_MODEL_CFG_SRV,
295 BT_MESH_MODEL_CFG_CLI(&cfg_cli),
296 BT_MESH_MODEL_SAR_CFG_SRV,
297 BT_MESH_MODEL_SAR_CFG_CLI(&sar_cfg_cli),
298 BT_MESH_MODEL_DFD_SRV(&dfd_srv)),
299 BT_MESH_MODEL_NONE),
300 },
301 .elem_count = 1,
302 };
303
304 static const struct bt_mesh_comp dist_comp_self_update = {
305 .elem =
306 (const struct bt_mesh_elem[]){
307 BT_MESH_ELEM(1,
308 MODEL_LIST(BT_MESH_MODEL_CFG_SRV,
309 BT_MESH_MODEL_CFG_CLI(&cfg_cli),
310 BT_MESH_MODEL_SAR_CFG_SRV,
311 BT_MESH_MODEL_SAR_CFG_CLI(&sar_cfg_cli),
312 BT_MESH_MODEL_DFD_SRV(&dfd_srv)),
313 BT_MESH_MODEL_NONE),
314 BT_MESH_ELEM(2,
315 MODEL_LIST(BT_MESH_MODEL_DFU_SRV(&dfu_srv)),
316 BT_MESH_MODEL_NONE),
317 },
318 .elem_count = 2,
319 };
320
321 static const struct bt_mesh_model_op model_dummy_op[] = {
322 BT_MESH_MODEL_OP_END
323 };
324
325 static const struct bt_mesh_comp target_comp = {
326 .elem =
327 (const struct bt_mesh_elem[]){
328 BT_MESH_ELEM(1,
329 MODEL_LIST(BT_MESH_MODEL_CFG_SRV,
330 BT_MESH_MODEL_CFG_CLI(&cfg_cli),
331 BT_MESH_MODEL_SAR_CFG_SRV,
332 BT_MESH_MODEL_SAR_CFG_CLI(&sar_cfg_cli),
333 /* Imposter model without custom handlers is used
334 * so device testing persistent storage can be
335 * configured using both `target_comp` and
336 * `srv_caps_broken_comp`. If these compositions
337 * have different model count and order
338 * loading settings will fail.
339 */
340 BT_MESH_MODEL_CB(IMPOSTER_MODEL_ID,
341 model_dummy_op, NULL, NULL, NULL),
342 BT_MESH_MODEL_DFU_SRV(&dfu_srv)),
343 BT_MESH_MODEL_NONE),
344 },
345 .elem_count = 1,
346 };
347
provision(uint16_t addr)348 static void provision(uint16_t addr)
349 {
350 int err;
351
352 err = bt_mesh_provision(test_net_key, 0, 0, 0, addr, dev_key);
353 if (err) {
354 FAIL("Provisioning failed (err %d)", err);
355 return;
356 }
357 }
358
common_configure(uint16_t addr)359 static void common_configure(uint16_t addr)
360 {
361 uint8_t status;
362 int err;
363
364 err = bt_mesh_cfg_cli_app_key_add(0, addr, 0, 0, test_app_key, &status);
365 if (err || status) {
366 FAIL("AppKey add failed (err %d, status %u)", err, status);
367 return;
368 }
369 }
370
common_app_bind(uint16_t addr,struct bind_params * params,size_t num)371 static void common_app_bind(uint16_t addr, struct bind_params *params, size_t num)
372 {
373 uint8_t status;
374 int err;
375
376 for (size_t i = 0; i < num; i++) {
377 err = bt_mesh_cfg_cli_mod_app_bind(0, addr, params[i].addr, 0, params[i].model_id,
378 &status);
379 if (err || status) {
380 FAIL("Model %#4x bind failed (err %d, status %u)", params[i].model_id,
381 err, status);
382 return;
383 }
384 }
385 }
386
dist_prov_and_conf(uint16_t addr)387 static void dist_prov_and_conf(uint16_t addr)
388 {
389 provision(addr);
390 common_configure(addr);
391
392 struct bind_params bind_params[] = {
393 { BT_MESH_MODEL_ID_BLOB_CLI, addr },
394 { BT_MESH_MODEL_ID_DFU_CLI, addr },
395 };
396
397 common_app_bind(addr, &bind_params[0], ARRAY_SIZE(bind_params));
398 common_sar_conf(addr);
399 }
400
dist_self_update_prov_and_conf(uint16_t addr)401 static void dist_self_update_prov_and_conf(uint16_t addr)
402 {
403 provision(addr);
404 common_configure(addr);
405
406 struct bind_params bind_params[] = {
407 { BT_MESH_MODEL_ID_BLOB_CLI, addr },
408 { BT_MESH_MODEL_ID_DFU_CLI, addr },
409 { BT_MESH_MODEL_ID_BLOB_SRV, addr + 1 },
410 { BT_MESH_MODEL_ID_DFU_SRV, addr + 1 },
411 };
412
413 common_app_bind(addr, &bind_params[0], ARRAY_SIZE(bind_params));
414 common_sar_conf(addr);
415 }
416
target_prov_and_conf(uint16_t addr,struct bind_params * params,size_t len)417 static void target_prov_and_conf(uint16_t addr, struct bind_params *params, size_t len)
418 {
419 provision(addr);
420 common_configure(addr);
421
422 common_app_bind(addr, params, len);
423 common_sar_conf(addr);
424 }
425
target_prov_and_conf_default(void)426 static void target_prov_and_conf_default(void)
427 {
428 uint16_t addr = bt_mesh_test_own_addr_get(TARGET_ADDR);
429 struct bind_params bind_params[] = {
430 { BT_MESH_MODEL_ID_BLOB_SRV, addr },
431 { BT_MESH_MODEL_ID_DFU_SRV, addr },
432 };
433
434 target_prov_and_conf(addr, bind_params, ARRAY_SIZE(bind_params));
435 }
436
slot_reserve_and_set(size_t size,uint8_t * fwid,size_t fwid_len,uint8_t * metadata,size_t metadata_len)437 static struct bt_mesh_dfu_slot *slot_reserve_and_set(size_t size, uint8_t *fwid, size_t fwid_len,
438 uint8_t *metadata, size_t metadata_len)
439 {
440 struct bt_mesh_dfu_slot *new_slot = bt_mesh_dfu_slot_reserve();
441
442 if (!new_slot) {
443 LOG_WRN("Reserving slot failed");
444 return NULL;
445 }
446
447 int err = bt_mesh_dfu_slot_fwid_set(new_slot, fwid, fwid_len);
448
449 if (err) {
450 return NULL;
451 }
452
453 err = bt_mesh_dfu_slot_info_set(new_slot, size, metadata, metadata_len);
454
455 if (err) {
456 return NULL;
457 }
458
459 return new_slot;
460 }
461
slot_add(const struct bt_mesh_dfu_slot ** slot)462 static bool slot_add(const struct bt_mesh_dfu_slot **slot)
463 {
464 struct bt_mesh_dfu_slot *new_slot;
465 size_t size = 100;
466 uint8_t fwid[CONFIG_BT_MESH_DFU_FWID_MAXLEN] = { 0xAA, 0xBB, 0xCC, 0xDD };
467 size_t fwid_len = 4;
468 uint8_t metadata[CONFIG_BT_MESH_DFU_METADATA_MAXLEN] = { 0xAA, 0xBB, 0xCC, 0xDD };
469 size_t metadata_len = 4;
470
471 ASSERT_EQUAL(sizeof(target_fw_ver_new), fwid_len);
472
473 new_slot = slot_reserve_and_set(size, fwid, fwid_len, metadata, metadata_len);
474 if (!new_slot) {
475 return false;
476 }
477
478 if (bt_mesh_dfu_slot_commit(new_slot) != 0) {
479 return false;
480 }
481
482 if (slot) {
483 *slot = new_slot;
484 }
485
486 return true;
487 }
488
dist_dfu_start_and_confirm(void)489 static void dist_dfu_start_and_confirm(void)
490 {
491 enum bt_mesh_dfd_status status;
492 struct bt_mesh_dfd_start_params start_params = {
493 .app_idx = 0,
494 .timeout_base = 10,
495 .slot_idx = 0,
496 .group = 0,
497 .xfer_mode = BT_MESH_BLOB_XFER_MODE_PUSH,
498 .ttl = 2,
499 .apply = true,
500 };
501
502 status = bt_mesh_dfd_srv_start(&dfd_srv, &start_params);
503 ASSERT_EQUAL(BT_MESH_DFD_SUCCESS, status);
504
505 if (k_sem_take(&dfu_dist_ended, K_SECONDS(DFU_TIMEOUT))) {
506 FAIL("DFU timed out");
507 }
508
509 enum bt_mesh_dfu_status expected_status;
510 enum bt_mesh_dfu_phase expected_phase;
511
512 if (dfu_fail_confirm) {
513 ASSERT_EQUAL(BT_MESH_DFD_PHASE_FAILED, dfd_srv.phase);
514 expected_status = BT_MESH_DFU_ERR_INTERNAL;
515 expected_phase = BT_MESH_DFU_PHASE_APPLY_FAIL;
516 } else {
517 ASSERT_EQUAL(BT_MESH_DFD_PHASE_COMPLETED, dfd_srv.phase);
518 expected_status = BT_MESH_DFU_SUCCESS;
519 expected_phase = BT_MESH_DFU_PHASE_APPLY_SUCCESS;
520 }
521
522 for (int i = 0; i < dfu_targets_cnt; i++) {
523 ASSERT_EQUAL(expected_status, dfd_srv.targets[i].status);
524
525 if (dfd_srv.targets[i].effect == BT_MESH_DFU_EFFECT_UNPROV) {
526 /* If device should unprovision itself after the update, the phase won't
527 * change. If phase changes, DFU failed.
528 */
529 if (dfu_fail_confirm) {
530 ASSERT_EQUAL(BT_MESH_DFU_PHASE_APPLY_FAIL,
531 dfd_srv.targets[i].phase);
532 } else {
533 ASSERT_EQUAL(BT_MESH_DFU_PHASE_APPLYING, dfd_srv.targets[i].phase);
534 }
535 } else {
536 ASSERT_EQUAL(expected_phase, dfd_srv.targets[i].phase);
537 }
538 }
539 }
540
test_dist_dfu(void)541 static void test_dist_dfu(void)
542 {
543 enum bt_mesh_dfd_status status;
544
545 bt_mesh_test_cfg_set(NULL, WAIT_TIME);
546 bt_mesh_device_setup(&prov, &dist_comp);
547 dist_prov_and_conf(DIST_ADDR);
548
549 ASSERT_TRUE(slot_add(NULL));
550
551 ASSERT_TRUE(dfu_targets_cnt > 0);
552
553 for (int i = 0; i < dfu_targets_cnt; i++) {
554 status = bt_mesh_dfd_srv_receiver_add(&dfd_srv, TARGET_ADDR + 1 + i, 0);
555 ASSERT_EQUAL(BT_MESH_DFD_SUCCESS, status);
556 }
557
558 dist_dfu_start_and_confirm();
559
560 PASS();
561 }
562
test_dist_dfu_self_update(void)563 static void test_dist_dfu_self_update(void)
564 {
565 enum bt_mesh_dfd_status status;
566
567 ASSERT_TRUE(dfu_targets_cnt > 0);
568
569 bt_mesh_test_cfg_set(NULL, WAIT_TIME);
570 bt_mesh_device_setup(&prov, &dist_comp_self_update);
571 dist_self_update_prov_and_conf(DIST_ADDR);
572
573 ASSERT_TRUE(slot_add(NULL));
574
575 status = bt_mesh_dfd_srv_receiver_add(&dfd_srv, DIST_ADDR + 1, 0);
576 ASSERT_EQUAL(BT_MESH_DFD_SUCCESS, status);
577 dfu_target_effect = BT_MESH_DFU_EFFECT_NONE;
578
579 for (int i = 1; i < dfu_targets_cnt; i++) {
580 status = bt_mesh_dfd_srv_receiver_add(&dfd_srv, TARGET_ADDR + i, 0);
581 ASSERT_EQUAL(BT_MESH_DFD_SUCCESS, status);
582 }
583
584 dist_dfu_start_and_confirm();
585
586 /* Check that DFU finished on distributor. */
587 if (k_sem_take(&dfu_ended, K_SECONDS(DFU_TIMEOUT))) {
588 FAIL("firmware was not applied");
589 }
590
591 PASS();
592 }
593
test_dist_dfu_slot_create(void)594 static void test_dist_dfu_slot_create(void)
595 {
596 struct bt_mesh_dfu_slot *slot[CONFIG_BT_MESH_DFU_SLOT_CNT];
597 size_t size = 100;
598 uint8_t fwid[CONFIG_BT_MESH_DFU_FWID_MAXLEN] = { 0 };
599 size_t fwid_len = 4;
600 uint8_t metadata[CONFIG_BT_MESH_DFU_METADATA_MAXLEN] = { 0 };
601 size_t metadata_len = 4;
602 int err, i;
603
604 ASSERT_TRUE_MSG(CONFIG_BT_MESH_DFU_SLOT_CNT >= 3,
605 "CONFIG_BT_MESH_DFU_SLOT_CNT must be at least 3\n");
606
607 bt_mesh_test_cfg_set(NULL, WAIT_TIME);
608 bt_mesh_device_setup(&prov, &dist_comp);
609 dist_prov_and_conf(DIST_ADDR);
610
611 for (i = 0; i < CONFIG_BT_MESH_DFU_SLOT_CNT; i++) {
612 fwid[0] = i;
613 metadata[0] = i;
614 slot[i] = slot_reserve_and_set(size, fwid, fwid_len, metadata, metadata_len);
615
616 ASSERT_FALSE_MSG(slot[i] == NULL, "Failed to add slot\n");
617
618 if (i > 0) {
619 /* All but first slot are committed */
620 err = bt_mesh_dfu_slot_commit(slot[i]);
621 if (err) {
622 FAIL("Committing slot failed (err %d)", err);
623 }
624 }
625 }
626
627 /* Second slot is deleted */
628 err = bt_mesh_dfu_slot_del(slot[1]);
629 if (err) {
630 FAIL("Slot delete failed (err %d)", err);
631 }
632
633 PASS();
634 }
635
check_slot(const struct bt_mesh_dfu_slot * slot,void * data)636 enum bt_mesh_dfu_iter check_slot(const struct bt_mesh_dfu_slot *slot, void *data)
637 {
638 size_t size = 100;
639 uint8_t fwid[CONFIG_BT_MESH_DFU_FWID_MAXLEN] = { 0 };
640 size_t fwid_len = 4;
641 uint8_t metadata[CONFIG_BT_MESH_DFU_METADATA_MAXLEN] = { 0 };
642 size_t metadata_len = 4;
643 int idx = bt_mesh_dfu_slot_img_idx_get(slot);
644 int *i = data;
645
646 ASSERT_EQUAL(idx, (*i)++);
647 ASSERT_EQUAL(size, slot->size);
648
649 fwid[0] = idx + 2;
650 ASSERT_EQUAL(fwid_len, slot->fwid_len);
651 ASSERT_TRUE(memcmp(fwid, slot->fwid, fwid_len) == 0);
652
653 metadata[0] = idx + 2;
654 ASSERT_EQUAL(metadata_len, slot->metadata_len);
655 ASSERT_TRUE(memcmp(metadata, slot->metadata, metadata_len) == 0);
656
657 return BT_MESH_DFU_ITER_CONTINUE;
658 }
659
test_dist_dfu_slot_create_recover(void)660 static void test_dist_dfu_slot_create_recover(void)
661 {
662 size_t slot_count;
663 struct bt_mesh_dfu_slot *slot;
664 size_t size = 100;
665 uint8_t fwid[CONFIG_BT_MESH_DFU_FWID_MAXLEN] = { 0 };
666 size_t fwid_len = 4;
667 uint8_t metadata[CONFIG_BT_MESH_DFU_METADATA_MAXLEN] = { 0 };
668 size_t metadata_len = 4;
669 int i, idx;
670
671 ASSERT_TRUE_MSG(CONFIG_BT_MESH_DFU_SLOT_CNT >= 3,
672 "CONFIG_BT_MESH_DFU_SLOT_CNT must be at least 3\n");
673
674 bt_mesh_test_cfg_set(NULL, WAIT_TIME);
675 bt_mesh_device_setup(&prov, &dist_comp);
676
677 i = 0;
678 slot_count = bt_mesh_dfu_slot_foreach(check_slot, &i);
679 ASSERT_EQUAL(CONFIG_BT_MESH_DFU_SLOT_CNT - 2, slot_count);
680
681 for (i = 2; i < CONFIG_BT_MESH_DFU_SLOT_CNT; i++) {
682 fwid[0] = i;
683 idx = bt_mesh_dfu_slot_get(fwid, fwid_len, &slot);
684 ASSERT_EQUAL(idx, i - 2);
685 ASSERT_EQUAL(size, slot->size);
686
687 metadata[0] = i;
688 ASSERT_EQUAL(metadata_len, slot->metadata_len);
689 ASSERT_TRUE(memcmp(metadata, slot->metadata, metadata_len) == 0);
690 }
691
692 PASS();
693 }
694
check_delete_all(void)695 static void check_delete_all(void)
696 {
697 int i;
698 const struct bt_mesh_dfu_slot *slot;
699 size_t slot_count;
700
701 ASSERT_TRUE_MSG(CONFIG_BT_MESH_DFU_SLOT_CNT >= 3,
702 "CONFIG_BT_MESH_DFU_SLOT_CNT must be at least 3\n");
703
704 slot_count = bt_mesh_dfu_slot_foreach(NULL, NULL);
705 ASSERT_EQUAL(0, slot_count);
706
707 for (i = 0; i < CONFIG_BT_MESH_DFU_SLOT_CNT - 1; i++) {
708 slot = bt_mesh_dfu_slot_at(i);
709 ASSERT_TRUE(slot == NULL);
710 }
711 }
712
test_dist_dfu_slot_delete_all(void)713 static void test_dist_dfu_slot_delete_all(void)
714 {
715 ASSERT_TRUE_MSG(CONFIG_BT_MESH_DFU_SLOT_CNT >= 3,
716 "CONFIG_BT_MESH_DFU_SLOT_CNT must be at least 3\n");
717
718 bt_mesh_test_cfg_set(NULL, WAIT_TIME);
719 bt_mesh_device_setup(&prov, &dist_comp);
720
721 bt_mesh_dfu_slot_del_all();
722 check_delete_all();
723
724 PASS();
725 }
726
test_dist_dfu_slot_check_delete_all(void)727 static void test_dist_dfu_slot_check_delete_all(void)
728 {
729 bt_mesh_test_cfg_set(NULL, WAIT_TIME);
730 bt_mesh_device_setup(&prov, &dist_comp);
731
732 check_delete_all();
733
734 PASS();
735 }
736
test_dist_dfu_slot_reservation(void)737 static void test_dist_dfu_slot_reservation(void)
738 {
739 int i;
740 struct bt_mesh_dfu_slot *slots[CONFIG_BT_MESH_DFU_SLOT_CNT];
741
742 bt_mesh_test_cfg_set(NULL, WAIT_TIME);
743 bt_mesh_device_setup(&prov, &dist_comp);
744
745 for (i = 0; i < CONFIG_BT_MESH_DFU_SLOT_CNT; i++) {
746 slots[i] = bt_mesh_dfu_slot_reserve();
747 ASSERT_TRUE(slots[i] != NULL);
748 }
749
750 ASSERT_EQUAL(NULL, bt_mesh_dfu_slot_reserve());
751 bt_mesh_dfu_slot_release(slots[0]);
752 /* Release twice to check idempotency with empty pool */
753 bt_mesh_dfu_slot_release(slots[0]);
754 ASSERT_TRUE(bt_mesh_dfu_slot_reserve() != NULL);
755 ASSERT_EQUAL(NULL, bt_mesh_dfu_slot_reserve());
756
757 PASS();
758 }
759
test_dist_dfu_slot_idempotency(void)760 static void test_dist_dfu_slot_idempotency(void)
761 {
762 uint8_t fwid[CONFIG_BT_MESH_DFU_FWID_MAXLEN] = { 0 };
763 size_t fwid_len = 4;
764 struct bt_mesh_dfu_slot *slot;
765
766 ASSERT_TRUE_MSG(CONFIG_BT_MESH_DFU_SLOT_CNT >= 1,
767 "CONFIG_BT_MESH_DFU_SLOT_CNT must be at least 1\n");
768
769 bt_mesh_test_cfg_set(NULL, WAIT_TIME);
770 bt_mesh_device_setup(&prov, &dist_comp);
771 dist_prov_and_conf(DIST_ADDR);
772
773 slot = bt_mesh_dfu_slot_reserve();
774 ASSERT_TRUE(slot != NULL);
775
776 bt_mesh_dfu_slot_release(slot);
777 bt_mesh_dfu_slot_release(slot);
778
779 slot = bt_mesh_dfu_slot_reserve();
780 ASSERT_TRUE(slot != NULL);
781
782 ASSERT_EQUAL(0, bt_mesh_dfu_slot_fwid_set(slot, fwid, fwid_len));
783 ASSERT_EQUAL(0, bt_mesh_dfu_slot_info_set(slot, 100, NULL, 0));
784
785 ASSERT_EQUAL(0, bt_mesh_dfu_slot_commit(slot));
786 ASSERT_EQUAL(-EINVAL, bt_mesh_dfu_slot_commit(slot));
787
788 ASSERT_EQUAL(0, bt_mesh_dfu_slot_del(slot));
789 ASSERT_EQUAL(-EINVAL, bt_mesh_dfu_slot_del(slot));
790
791 PASS();
792 }
793
target_test_effect(enum bt_mesh_dfu_effect effect)794 static void target_test_effect(enum bt_mesh_dfu_effect effect)
795 {
796 dfu_target_effect = effect;
797
798 bt_mesh_test_cfg_set(NULL, WAIT_TIME);
799 bt_mesh_device_setup(&prov, &target_comp);
800 target_prov_and_conf_default();
801
802 if (k_sem_take(&dfu_ended, K_SECONDS(DFU_TIMEOUT))) {
803 FAIL("Firmware was not applied");
804 }
805 }
806
test_target_dfu_no_change(void)807 static void test_target_dfu_no_change(void)
808 {
809 target_test_effect(BT_MESH_DFU_EFFECT_NONE);
810
811 PASS();
812 }
813
test_target_dfu_new_comp_no_rpr(void)814 static void test_target_dfu_new_comp_no_rpr(void)
815 {
816 target_test_effect(BT_MESH_DFU_EFFECT_COMP_CHANGE_NO_RPR);
817
818 PASS();
819 }
820
test_target_dfu_new_comp_rpr(void)821 static void test_target_dfu_new_comp_rpr(void)
822 {
823 target_test_effect(BT_MESH_DFU_EFFECT_COMP_CHANGE);
824
825 PASS();
826 }
827
test_target_dfu_unprov(void)828 static void test_target_dfu_unprov(void)
829 {
830 target_test_effect(BT_MESH_DFU_EFFECT_UNPROV);
831
832 PASS();
833 }
834
835 static struct {
836 struct bt_mesh_blob_cli_inputs inputs;
837 struct bt_mesh_blob_target_pull pull[7];
838 struct bt_mesh_dfu_target targets[7];
839 uint8_t target_count;
840 struct bt_mesh_dfu_cli_xfer xfer;
841 } dfu_cli_xfer;
842
dfu_cli_inputs_prepare(uint16_t group)843 static void dfu_cli_inputs_prepare(uint16_t group)
844 {
845 dfu_cli_xfer.inputs.ttl = BT_MESH_TTL_DEFAULT;
846 dfu_cli_xfer.inputs.group = group;
847 dfu_cli_xfer.inputs.app_idx = 0;
848 dfu_cli_xfer.inputs.timeout_base = 1;
849 sys_slist_init(&dfu_cli_xfer.inputs.targets);
850
851 for (int i = 0; i < dfu_cli_xfer.target_count; ++i) {
852 /* Reset target context. */
853 uint16_t addr = dfu_cli_xfer.targets[i].blob.addr;
854
855 memset(&dfu_cli_xfer.targets[i], 0, sizeof(struct bt_mesh_dfu_target));
856 dfu_cli_xfer.targets[i].blob.addr = addr;
857 if (recover) {
858 memset(&dfu_cli_xfer.pull[i].missing, 1,
859 DIV_ROUND_UP(CONFIG_BT_MESH_BLOB_CHUNK_COUNT_MAX, 8));
860 dfu_cli_xfer.targets[i].blob.pull = &dfu_cli_xfer.pull[i];
861 }
862
863 sys_slist_append(&dfu_cli_xfer.inputs.targets, &dfu_cli_xfer.targets[i].blob.n);
864 }
865 }
866
target_srv_add(uint16_t addr,bool expect_lost)867 static struct bt_mesh_blob_target *target_srv_add(uint16_t addr, bool expect_lost)
868 {
869 if (expect_lost) {
870 lost_target_add(addr);
871 }
872
873 ASSERT_TRUE(dfu_cli_xfer.target_count < ARRAY_SIZE(dfu_cli_xfer.targets));
874 struct bt_mesh_blob_target *t = &dfu_cli_xfer.targets[dfu_cli_xfer.target_count].blob;
875
876 t->addr = addr;
877 dfu_cli_xfer.target_count++;
878 return t;
879 }
880
dfu_cli_suspended(struct bt_mesh_dfu_cli * cli)881 static void dfu_cli_suspended(struct bt_mesh_dfu_cli *cli)
882 {
883 FAIL("Unexpected call");
884 }
885
dfu_cli_ended(struct bt_mesh_dfu_cli * cli,enum bt_mesh_dfu_status reason)886 static void dfu_cli_ended(struct bt_mesh_dfu_cli *cli, enum bt_mesh_dfu_status reason)
887 {
888 if ((expected_stop_phase == BT_MESH_DFU_PHASE_IDLE ||
889 expected_stop_phase == BT_MESH_DFU_PHASE_VERIFY_OK) &&
890 !expect_fail) {
891 ASSERT_EQUAL(BT_MESH_DFU_SUCCESS, reason);
892 }
893
894 if (expected_stop_phase == BT_MESH_DFU_PHASE_TRANSFER_ACTIVE) {
895 k_sem_give(&dfu_started);
896 } else if (expected_stop_phase == BT_MESH_DFU_PHASE_VERIFY) {
897 k_sem_give(&dfu_verifying);
898 } else if (expected_stop_phase == BT_MESH_DFU_PHASE_VERIFY_FAIL) {
899 k_sem_give(&dfu_verify_failed);
900 }
901
902 k_sem_give(&dfu_ended);
903 }
904
905 static struct k_sem dfu_cli_applied_sem;
906
dfu_cli_applied(struct bt_mesh_dfu_cli * cli)907 static void dfu_cli_applied(struct bt_mesh_dfu_cli *cli)
908 {
909 k_sem_give(&dfu_cli_applied_sem);
910 }
911
912 static struct k_sem dfu_cli_confirmed_sem;
913
dfu_cli_confirmed(struct bt_mesh_dfu_cli * cli)914 static void dfu_cli_confirmed(struct bt_mesh_dfu_cli *cli)
915 {
916 k_sem_give(&dfu_cli_confirmed_sem);
917 }
918
919 static struct k_sem lost_target_sem;
920
dfu_cli_lost_target(struct bt_mesh_dfu_cli * cli,struct bt_mesh_dfu_target * target)921 static void dfu_cli_lost_target(struct bt_mesh_dfu_cli *cli, struct bt_mesh_dfu_target *target)
922 {
923 ASSERT_FALSE(target->status == BT_MESH_DFU_SUCCESS);
924 ASSERT_TRUE(lost_target_find_and_remove(target->blob.addr));
925
926 if (!lost_targets_rem()) {
927 k_sem_give(&lost_target_sem);
928 }
929 }
930
931 static struct bt_mesh_dfu_cli_cb dfu_cli_cb = {
932 .suspended = dfu_cli_suspended,
933 .ended = dfu_cli_ended,
934 .applied = dfu_cli_applied,
935 .confirmed = dfu_cli_confirmed,
936 .lost_target = dfu_cli_lost_target,
937 };
938
939 static struct bt_mesh_dfu_cli dfu_cli = BT_MESH_DFU_CLI_INIT(&dfu_cli_cb);
940
941 static const struct bt_mesh_comp cli_comp = {
942 .elem =
943 (const struct bt_mesh_elem[]){
944 BT_MESH_ELEM(1,
945 MODEL_LIST(BT_MESH_MODEL_CFG_SRV,
946 BT_MESH_MODEL_CFG_CLI(&cfg_cli),
947 BT_MESH_MODEL_SAR_CFG_SRV,
948 BT_MESH_MODEL_SAR_CFG_CLI(&sar_cfg_cli),
949 BT_MESH_MODEL_DFU_CLI(&dfu_cli)),
950 BT_MESH_MODEL_NONE),
951 },
952 .elem_count = 1,
953 };
954
cli_common_fail_on_init(void)955 static void cli_common_fail_on_init(void)
956 {
957 const struct bt_mesh_dfu_slot *slot;
958
959 bt_mesh_test_cfg_set(NULL, 300);
960 bt_mesh_device_setup(&prov, &cli_comp);
961 dist_prov_and_conf(DIST_ADDR);
962
963 ASSERT_TRUE(slot_add(&slot));
964
965 dfu_cli_inputs_prepare(0);
966 dfu_cli_xfer.xfer.mode = BT_MESH_BLOB_XFER_MODE_PUSH;
967 dfu_cli_xfer.xfer.slot = slot;
968 dfu_cli_xfer.xfer.blob_id = TEST_BLOB_ID;
969 }
970
cli_common_init_recover(void)971 static void cli_common_init_recover(void)
972 {
973 struct bt_mesh_dfu_slot *slot;
974 uint8_t fwid[CONFIG_BT_MESH_DFU_FWID_MAXLEN] = { 0xAA, 0xBB, 0xCC, 0xDD };
975 size_t fwid_len = 4;
976
977 bt_mesh_test_cfg_set(NULL, 300);
978 bt_mesh_device_setup(&prov, &cli_comp);
979
980 ASSERT_TRUE(bt_mesh_dfu_slot_get(fwid, fwid_len, &slot) >= 0);
981
982 dfu_cli_inputs_prepare(0);
983 dfu_cli_xfer.xfer.mode = BT_MESH_BLOB_XFER_MODE_PUSH;
984 dfu_cli_xfer.xfer.slot = slot;
985 dfu_cli_xfer.xfer.blob_id = TEST_BLOB_ID;
986 }
987
test_cli_fail_on_persistency(void)988 static void test_cli_fail_on_persistency(void)
989 {
990 int err;
991
992 /** Test that DFU transfer persists as long as at least one target is still active. During
993 * the test multiple servers will become unresponsive at different phases of the transfer:
994 * - Srv 0x0002 will reject firmware by metadata.
995 * - Srv 0x0003 will not respond to BLOB Information Get msg (Retrieve Caps proc).
996 * - Srv 0x0004 will not respond to Firmware Update Get msg after BLOB Transfer.
997 * - Srv 0x0005 will fail firmware verification.
998 * - Srv 0x0006 will not respond to Firmware Update Apply msg.
999 * - Srv 0x0007 is responsive all the way.
1000 * - Srv 0x0008 is a non-existing unresponsive node that will not respond to Firmware
1001 * Update Start msg, which is the first message sent by DFU Client.
1002 */
1003 (void)target_srv_add(TARGET_ADDR + 1, true);
1004 (void)target_srv_add(TARGET_ADDR + 2, true);
1005 (void)target_srv_add(TARGET_ADDR + 3, true);
1006 (void)target_srv_add(TARGET_ADDR + 4, true);
1007 (void)target_srv_add(TARGET_ADDR + 5, true);
1008 (void)target_srv_add(TARGET_ADDR + 6, false);
1009 (void)target_srv_add(TARGET_ADDR + 7, true);
1010
1011 cli_common_fail_on_init();
1012
1013 err = bt_mesh_dfu_cli_send(&dfu_cli, &dfu_cli_xfer.inputs, &dummy_blob_io,
1014 &dfu_cli_xfer.xfer);
1015 if (err) {
1016 FAIL("DFU Client send failed (err: %d)", err);
1017 }
1018
1019 if (k_sem_take(&dfu_ended, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1020 FAIL("Firmware transfer failed");
1021 }
1022
1023 /* This is non-existing unresponsive target that didn't reply on Firmware Update Start
1024 * message.
1025 */
1026 ASSERT_EQUAL(BT_MESH_DFU_ERR_INTERNAL, dfu_cli_xfer.targets[6].status);
1027 ASSERT_EQUAL(BT_MESH_DFU_PHASE_UNKNOWN, dfu_cli_xfer.targets[6].phase);
1028 /* This target rejected metadata. */
1029 ASSERT_EQUAL(BT_MESH_DFU_ERR_METADATA, dfu_cli_xfer.targets[0].status);
1030 ASSERT_EQUAL(BT_MESH_DFU_PHASE_IDLE, dfu_cli_xfer.targets[0].phase);
1031 /* This target shouldn't respond on BLOB Information Get message from Retrieve Caps
1032 * procedure.
1033 */
1034 ASSERT_EQUAL(BT_MESH_DFU_ERR_INTERNAL, dfu_cli_xfer.targets[1].status);
1035 ASSERT_EQUAL(BT_MESH_DFU_PHASE_TRANSFER_ACTIVE, dfu_cli_xfer.targets[1].phase);
1036 /* This target shouldn't respond on Firmware Update Get msg. */
1037 ASSERT_EQUAL(BT_MESH_DFU_ERR_INTERNAL, dfu_cli_xfer.targets[2].status);
1038 ASSERT_EQUAL(BT_MESH_DFU_PHASE_TRANSFER_ACTIVE, dfu_cli_xfer.targets[2].phase);
1039 /* This target failed firmware verification. */
1040 ASSERT_EQUAL(BT_MESH_DFU_ERR_WRONG_PHASE, dfu_cli_xfer.targets[3].status);
1041 ASSERT_EQUAL(BT_MESH_DFU_PHASE_VERIFY_FAIL, dfu_cli_xfer.targets[3].phase);
1042 /* The next two targets should be OK. */
1043 ASSERT_EQUAL(BT_MESH_DFU_SUCCESS, dfu_cli_xfer.targets[4].status);
1044 ASSERT_EQUAL(BT_MESH_DFU_PHASE_VERIFY_OK, dfu_cli_xfer.targets[4].phase);
1045 ASSERT_EQUAL(BT_MESH_DFU_SUCCESS, dfu_cli_xfer.targets[5].status);
1046 ASSERT_EQUAL(BT_MESH_DFU_PHASE_VERIFY_OK, dfu_cli_xfer.targets[5].phase);
1047
1048 err = bt_mesh_dfu_cli_apply(&dfu_cli);
1049 if (err) {
1050 FAIL("DFU Client apply failed (err: %d)", err);
1051 }
1052
1053 if (k_sem_take(&dfu_cli_applied_sem, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1054 FAIL("Failed to apply firmware");
1055 }
1056
1057 /* This target shouldn't respond on Firmware Update Apply message. */
1058 ASSERT_EQUAL(BT_MESH_DFU_ERR_INTERNAL, dfu_cli_xfer.targets[4].status);
1059 ASSERT_EQUAL(BT_MESH_DFU_PHASE_VERIFY_OK, dfu_cli_xfer.targets[4].phase);
1060
1061 err = bt_mesh_dfu_cli_confirm(&dfu_cli);
1062 if (err) {
1063 FAIL("DFU Client confirm failed (err: %d)", err);
1064 }
1065
1066 if (k_sem_take(&dfu_cli_confirmed_sem, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1067 FAIL("Failed to confirm firmware");
1068 }
1069
1070 /* This target should complete DFU successfully. */
1071 ASSERT_EQUAL(BT_MESH_DFU_SUCCESS, dfu_cli_xfer.targets[5].status);
1072 ASSERT_EQUAL(BT_MESH_DFU_PHASE_APPLY_SUCCESS, dfu_cli_xfer.targets[5].phase);
1073
1074 if (k_sem_take(&lost_target_sem, K_NO_WAIT)) {
1075 FAIL("Lost targets CB did not trigger for all expected lost targets");
1076 }
1077
1078 PASS();
1079 }
1080
test_cli_all_targets_lost_common(void)1081 static void test_cli_all_targets_lost_common(void)
1082 {
1083 int err, i;
1084
1085 expect_fail = true;
1086
1087 for (i = 1; i <= dfu_targets_cnt; i++) {
1088 (void)target_srv_add(TARGET_ADDR + i, true);
1089 }
1090
1091 cli_common_fail_on_init();
1092
1093 err = bt_mesh_dfu_cli_send(&dfu_cli, &dfu_cli_xfer.inputs, &dummy_blob_io,
1094 &dfu_cli_xfer.xfer);
1095 if (err) {
1096 FAIL("DFU Client send failed (err: %d)", err);
1097 }
1098
1099 if (k_sem_take(&dfu_ended, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1100 FAIL("Firmware transfer failed");
1101 }
1102 }
1103
test_cli_all_targets_lost_on_metadata(void)1104 static void test_cli_all_targets_lost_on_metadata(void)
1105 {
1106 int i;
1107
1108 test_cli_all_targets_lost_common();
1109
1110 for (i = 0; i < dfu_targets_cnt; i++) {
1111 ASSERT_EQUAL(BT_MESH_DFU_ERR_METADATA, dfu_cli_xfer.targets[i].status);
1112 ASSERT_EQUAL(BT_MESH_DFU_PHASE_IDLE, dfu_cli_xfer.targets[i].phase);
1113 }
1114
1115 /* `lost_target` cb must be called on all targets */
1116 ASSERT_EQUAL(0, lost_targets_rem());
1117
1118 PASS();
1119 }
1120
test_cli_all_targets_lost_on_caps_get(void)1121 static void test_cli_all_targets_lost_on_caps_get(void)
1122 {
1123 int i;
1124
1125 test_cli_all_targets_lost_common();
1126
1127 for (i = 0; i < dfu_targets_cnt; i++) {
1128 ASSERT_EQUAL(BT_MESH_DFU_ERR_INTERNAL, dfu_cli_xfer.targets[i].status);
1129 ASSERT_EQUAL(BT_MESH_DFU_PHASE_TRANSFER_ACTIVE,
1130 dfu_cli_xfer.targets[i].phase);
1131 }
1132
1133 /* `lost_target` cb must be called on all targets */
1134 ASSERT_EQUAL(0, lost_targets_rem());
1135
1136 PASS();
1137 }
1138
test_cli_all_targets_lost_on_update_get(void)1139 static void test_cli_all_targets_lost_on_update_get(void)
1140 {
1141 int i;
1142
1143 test_cli_all_targets_lost_common();
1144
1145 for (i = 0; i < dfu_targets_cnt; i++) {
1146 ASSERT_EQUAL(BT_MESH_DFU_ERR_INTERNAL, dfu_cli_xfer.targets[i].status);
1147 ASSERT_EQUAL(BT_MESH_DFU_PHASE_TRANSFER_ACTIVE,
1148 dfu_cli_xfer.targets[i].phase);
1149 }
1150
1151 /* `lost_target` cb must be called on all targets */
1152 ASSERT_EQUAL(0, lost_targets_rem());
1153
1154 PASS();
1155 }
1156
test_cli_all_targets_lost_on_verify(void)1157 static void test_cli_all_targets_lost_on_verify(void)
1158 {
1159 int i;
1160
1161 test_cli_all_targets_lost_common();
1162
1163 for (i = 0; i < dfu_targets_cnt; i++) {
1164 ASSERT_EQUAL(BT_MESH_DFU_ERR_WRONG_PHASE, dfu_cli_xfer.targets[i].status);
1165 ASSERT_EQUAL(BT_MESH_DFU_PHASE_VERIFY_FAIL, dfu_cli_xfer.targets[i].phase);
1166 }
1167
1168 /* `lost_target` cb must be called on all targets */
1169 ASSERT_EQUAL(0, lost_targets_rem());
1170
1171 PASS();
1172 }
1173
test_cli_all_targets_lost_on_apply(void)1174 static void test_cli_all_targets_lost_on_apply(void)
1175 {
1176 int err, i;
1177
1178 test_cli_all_targets_lost_common();
1179
1180 for (i = 0; i < dfu_targets_cnt; i++) {
1181 ASSERT_EQUAL(BT_MESH_DFU_SUCCESS, dfu_cli_xfer.targets[i].status);
1182 ASSERT_EQUAL(BT_MESH_DFU_PHASE_VERIFY_OK, dfu_cli_xfer.targets[i].phase);
1183 }
1184
1185 err = bt_mesh_dfu_cli_apply(&dfu_cli);
1186 if (err) {
1187 FAIL("DFU Client apply failed (err: %d)", err);
1188 }
1189
1190 if (!k_sem_take(&dfu_cli_applied_sem, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1191 FAIL("Apply should not be successful on any target");
1192 }
1193
1194 for (i = 0; i < dfu_targets_cnt; i++) {
1195 ASSERT_EQUAL(BT_MESH_DFU_ERR_INTERNAL, dfu_cli_xfer.targets[i].status);
1196 ASSERT_EQUAL(BT_MESH_DFU_PHASE_VERIFY_OK, dfu_cli_xfer.targets[i].phase);
1197 }
1198
1199 /* `lost_target` cb must be called on all targets */
1200 ASSERT_EQUAL(0, lost_targets_rem());
1201
1202 PASS();
1203 }
1204
test_cli_stop(void)1205 static void test_cli_stop(void)
1206 {
1207 int err;
1208
1209 (void)target_srv_add(TARGET_ADDR + 1, true);
1210
1211 switch (expected_stop_phase) {
1212 case BT_MESH_DFU_PHASE_TRANSFER_ACTIVE:
1213 cli_common_fail_on_init();
1214
1215 err = bt_mesh_dfu_cli_send(&dfu_cli, &dfu_cli_xfer.inputs, &dummy_blob_io,
1216 &dfu_cli_xfer.xfer);
1217 if (err) {
1218 FAIL("DFU Client send failed (err: %d)", err);
1219 }
1220
1221 if (k_sem_take(&dfu_started, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1222 FAIL("Firmware transfer failed");
1223 }
1224
1225 ASSERT_EQUAL(BT_MESH_DFU_ERR_INTERNAL, dfu_cli_xfer.targets[0].status);
1226 ASSERT_EQUAL(BT_MESH_DFU_PHASE_TRANSFER_ACTIVE, dfu_cli_xfer.targets[0].phase);
1227 break;
1228 case BT_MESH_DFU_PHASE_VERIFY:
1229 cli_common_init_recover();
1230
1231 err = bt_mesh_dfu_cli_send(&dfu_cli, &dfu_cli_xfer.inputs, &dummy_blob_io,
1232 &dfu_cli_xfer.xfer);
1233 if (err) {
1234 FAIL("DFU Client resume failed (err: %d)", err);
1235 }
1236
1237 if (k_sem_take(&dfu_verifying, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1238 FAIL("Firmware transfer failed");
1239 }
1240 ASSERT_EQUAL(BT_MESH_DFU_ERR_INTERNAL, dfu_cli_xfer.targets[0].status);
1241 ASSERT_EQUAL(BT_MESH_DFU_PHASE_VERIFY, dfu_cli_xfer.targets[0].phase);
1242
1243 break;
1244 case BT_MESH_DFU_PHASE_VERIFY_OK:
1245 /* Nothing to do here on distributor side, target must verify image */
1246 break;
1247 case BT_MESH_DFU_PHASE_VERIFY_FAIL:
1248 cli_common_fail_on_init();
1249
1250 err = bt_mesh_dfu_cli_send(&dfu_cli, &dfu_cli_xfer.inputs, &dummy_blob_io,
1251 &dfu_cli_xfer.xfer);
1252 if (err) {
1253 FAIL("DFU Client send failed (err: %d)", err);
1254 }
1255
1256 if (k_sem_take(&dfu_verify_failed, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1257 FAIL("Firmware transfer failed");
1258 }
1259
1260 ASSERT_EQUAL(BT_MESH_DFU_ERR_WRONG_PHASE, dfu_cli_xfer.targets[0].status);
1261 ASSERT_EQUAL(BT_MESH_DFU_PHASE_VERIFY_FAIL, dfu_cli_xfer.targets[0].phase);
1262 break;
1263 case BT_MESH_DFU_PHASE_APPLYING:
1264 cli_common_init_recover();
1265
1266 err = bt_mesh_dfu_cli_send(&dfu_cli, &dfu_cli_xfer.inputs, &dummy_blob_io,
1267 &dfu_cli_xfer.xfer);
1268
1269 if (err) {
1270 FAIL("DFU Client send failed (err: %d)", err);
1271 }
1272 if (k_sem_take(&dfu_ended, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1273 FAIL("Firmware transfer failed");
1274 }
1275
1276 bt_mesh_dfu_cli_apply(&dfu_cli);
1277 if (k_sem_take(&dfu_cli_applied_sem, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1278 /* This will time out as target will reboot before applying */
1279 }
1280 ASSERT_EQUAL(BT_MESH_DFU_ERR_INTERNAL, dfu_cli_xfer.targets[0].status);
1281 ASSERT_EQUAL(BT_MESH_DFU_PHASE_APPLYING, dfu_cli_xfer.targets[0].phase);
1282 break;
1283 case BT_MESH_DFU_PHASE_APPLY_SUCCESS:
1284 cli_common_init_recover();
1285
1286 dfu_cli.xfer.state = 5;
1287 dfu_cli.xfer.slot = dfu_cli_xfer.xfer.slot;
1288 dfu_cli.xfer.blob.id = TEST_BLOB_ID;
1289 dfu_cli_xfer.xfer.mode = BT_MESH_BLOB_XFER_MODE_PUSH;
1290
1291 dfu_cli.blob.inputs = &dfu_cli_xfer.inputs;
1292
1293 err = bt_mesh_dfu_cli_confirm(&dfu_cli);
1294 if (err) {
1295 FAIL("DFU Client confirm failed (err: %d)", err);
1296 }
1297
1298 ASSERT_EQUAL(BT_MESH_DFU_SUCCESS, dfu_cli_xfer.targets[0].status);
1299 ASSERT_EQUAL(BT_MESH_DFU_PHASE_IDLE, dfu_cli_xfer.targets[0].phase);
1300
1301 PASS();
1302 break;
1303 default:
1304 break;
1305 }
1306 PASS();
1307 }
1308
1309 static struct k_sem caps_get_sem;
1310
mock_handle_caps_get(const struct bt_mesh_model * model,struct bt_mesh_msg_ctx * ctx,struct net_buf_simple * buf)1311 static int mock_handle_caps_get(const struct bt_mesh_model *model, struct bt_mesh_msg_ctx *ctx,
1312 struct net_buf_simple *buf)
1313 {
1314 LOG_WRN("Rejecting BLOB Information Get message");
1315
1316 k_sem_give(&caps_get_sem);
1317
1318 return 0;
1319 }
1320
1321 static const struct bt_mesh_model_op model_caps_op1[] = {
1322 { BT_MESH_BLOB_OP_INFO_GET, 0, mock_handle_caps_get },
1323 BT_MESH_MODEL_OP_END
1324 };
1325
1326 static const struct bt_mesh_comp srv_caps_broken_comp = {
1327 .elem =
1328 (const struct bt_mesh_elem[]){
1329 BT_MESH_ELEM(1,
1330 MODEL_LIST(BT_MESH_MODEL_CFG_SRV,
1331 BT_MESH_MODEL_CFG_CLI(&cfg_cli),
1332 BT_MESH_MODEL_SAR_CFG_SRV,
1333 BT_MESH_MODEL_SAR_CFG_CLI(&sar_cfg_cli),
1334 BT_MESH_MODEL_CB(IMPOSTER_MODEL_ID,
1335 model_caps_op1, NULL, NULL, NULL),
1336 BT_MESH_MODEL_DFU_SRV(&dfu_srv)),
1337 BT_MESH_MODEL_NONE),
1338 },
1339 .elem_count = 1,
1340 };
1341
mock_handle_chunks(const struct bt_mesh_model * model,struct bt_mesh_msg_ctx * ctx,struct net_buf_simple * buf)1342 static int mock_handle_chunks(const struct bt_mesh_model *model, struct bt_mesh_msg_ctx *ctx,
1343 struct net_buf_simple *buf)
1344 {
1345 LOG_WRN("Skipping receiving block");
1346
1347 k_sem_give(&dfu_started);
1348
1349 return 0;
1350 }
1351
1352 static const struct bt_mesh_model_op model_caps_op2[] = {
1353 { BT_MESH_BLOB_OP_CHUNK, 0, mock_handle_chunks },
1354 BT_MESH_MODEL_OP_END
1355 };
1356
1357 static const struct bt_mesh_comp broken_target_comp = {
1358 .elem =
1359 (const struct bt_mesh_elem[]){
1360 BT_MESH_ELEM(1,
1361 MODEL_LIST(BT_MESH_MODEL_CFG_SRV,
1362 BT_MESH_MODEL_CFG_CLI(&cfg_cli),
1363 BT_MESH_MODEL_SAR_CFG_SRV,
1364 BT_MESH_MODEL_SAR_CFG_CLI(&sar_cfg_cli),
1365 BT_MESH_MODEL_CB(IMPOSTER_MODEL_ID,
1366 model_caps_op2, NULL, NULL, NULL),
1367 BT_MESH_MODEL_DFU_SRV(&dfu_srv)),
1368 BT_MESH_MODEL_NONE),
1369 },
1370 .elem_count = 1,
1371 };
1372
1373 static struct k_sem update_get_sem;
1374
mock_handle_update_get(const struct bt_mesh_model * model,struct bt_mesh_msg_ctx * ctx,struct net_buf_simple * buf)1375 static int mock_handle_update_get(const struct bt_mesh_model *model, struct bt_mesh_msg_ctx *ctx,
1376 struct net_buf_simple *buf)
1377 {
1378 LOG_WRN("Rejecting Firmware Update Get message");
1379 k_sem_give(&update_get_sem);
1380
1381 return 0;
1382 }
1383
1384 static const struct bt_mesh_model_op model_update_get_op1[] = {
1385 { BT_MESH_DFU_OP_UPDATE_GET, 0, mock_handle_update_get },
1386 BT_MESH_MODEL_OP_END
1387 };
1388
1389 static const struct bt_mesh_comp srv_update_get_broken_comp = {
1390 .elem =
1391 (const struct bt_mesh_elem[]){
1392 BT_MESH_ELEM(1,
1393 MODEL_LIST(BT_MESH_MODEL_CFG_SRV,
1394 BT_MESH_MODEL_CFG_CLI(&cfg_cli),
1395 BT_MESH_MODEL_SAR_CFG_SRV,
1396 BT_MESH_MODEL_SAR_CFG_CLI(&sar_cfg_cli),
1397 BT_MESH_MODEL_CB(IMPOSTER_MODEL_ID,
1398 model_update_get_op1, NULL, NULL,
1399 NULL),
1400 BT_MESH_MODEL_DFU_SRV(&dfu_srv)),
1401 BT_MESH_MODEL_NONE),
1402 },
1403 .elem_count = 1,
1404 };
1405
1406 static struct k_sem update_apply_sem;
1407
mock_handle_update_apply(const struct bt_mesh_model * model,struct bt_mesh_msg_ctx * ctx,struct net_buf_simple * buf)1408 static int mock_handle_update_apply(const struct bt_mesh_model *model, struct bt_mesh_msg_ctx *ctx,
1409 struct net_buf_simple *buf)
1410 {
1411 LOG_WRN("Rejecting Firmware Update Apply message");
1412 k_sem_give(&update_apply_sem);
1413
1414 return 0;
1415 }
1416
1417 static const struct bt_mesh_model_op model_update_apply_op1[] = {
1418 { BT_MESH_DFU_OP_UPDATE_APPLY, 0, mock_handle_update_apply },
1419 BT_MESH_MODEL_OP_END
1420 };
1421
1422 static const struct bt_mesh_comp srv_update_apply_broken_comp = {
1423 .elem =
1424 (const struct bt_mesh_elem[]){
1425 BT_MESH_ELEM(1,
1426 MODEL_LIST(BT_MESH_MODEL_CFG_SRV,
1427 BT_MESH_MODEL_CFG_CLI(&cfg_cli),
1428 BT_MESH_MODEL_SAR_CFG_SRV,
1429 BT_MESH_MODEL_SAR_CFG_CLI(&sar_cfg_cli),
1430 BT_MESH_MODEL_CB(IMPOSTER_MODEL_ID,
1431 model_update_apply_op1, NULL,
1432 NULL, NULL),
1433 BT_MESH_MODEL_DFU_SRV(&dfu_srv)),
1434 BT_MESH_MODEL_NONE),
1435 },
1436 .elem_count = 1,
1437 };
1438
target_prov_and_conf_with_imposer(void)1439 static void target_prov_and_conf_with_imposer(void)
1440 {
1441 uint16_t addr = bt_mesh_test_own_addr_get(TARGET_ADDR);
1442 struct bind_params bind_params[] = {
1443 { BT_MESH_MODEL_ID_BLOB_SRV, addr },
1444 { BT_MESH_MODEL_ID_DFU_SRV, addr },
1445 { IMPOSTER_MODEL_ID, addr },
1446 };
1447
1448 target_prov_and_conf(addr, bind_params, ARRAY_SIZE(bind_params));
1449 }
1450
common_fail_on_target_init(const struct bt_mesh_comp * comp)1451 static void common_fail_on_target_init(const struct bt_mesh_comp *comp)
1452 {
1453 bt_mesh_test_cfg_set(NULL, 300);
1454 bt_mesh_device_setup(&prov, comp);
1455
1456 dfu_target_effect = BT_MESH_DFU_EFFECT_NONE;
1457 }
1458
test_target_fail_on_metadata(void)1459 static void test_target_fail_on_metadata(void)
1460 {
1461 dfu_metadata_fail = false;
1462 expect_dfu_start = false;
1463
1464 common_fail_on_target_init(&target_comp);
1465 target_prov_and_conf_default();
1466
1467 if (k_sem_take(&dfu_metadata_check_sem, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1468 FAIL("Metadata check CB wasn't called");
1469 }
1470
1471 PASS();
1472 }
1473
test_target_fail_on_caps_get(void)1474 static void test_target_fail_on_caps_get(void)
1475 {
1476 expect_dfu_xfer_end = false;
1477
1478 common_fail_on_target_init(&srv_caps_broken_comp);
1479 target_prov_and_conf_with_imposer();
1480
1481 if (k_sem_take(&caps_get_sem, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1482 FAIL("BLOB Info Get msg handler wasn't called");
1483 }
1484
1485 PASS();
1486 }
1487
test_target_fail_on_update_get(void)1488 static void test_target_fail_on_update_get(void)
1489 {
1490 expect_dfu_apply = false;
1491
1492 common_fail_on_target_init(&srv_update_get_broken_comp);
1493 target_prov_and_conf_with_imposer();
1494
1495 if (k_sem_take(&dfu_verify_sem, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1496 FAIL("Transfer end CB wasn't triggered");
1497 }
1498
1499 if (k_sem_take(&update_get_sem, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1500 FAIL("Firmware Update Get msg handler wasn't called");
1501 }
1502
1503 PASS();
1504 }
1505
test_target_fail_on_verify(void)1506 static void test_target_fail_on_verify(void)
1507 {
1508 dfu_verify_fail = true;
1509 expect_dfu_apply = false;
1510
1511 common_fail_on_target_init(&target_comp);
1512 target_prov_and_conf_default();
1513
1514 if (k_sem_take(&dfu_verify_sem, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1515 FAIL("Transfer end CB wasn't triggered");
1516 }
1517
1518 PASS();
1519 }
1520
test_target_fail_on_apply(void)1521 static void test_target_fail_on_apply(void)
1522 {
1523 expect_dfu_apply = false;
1524
1525 common_fail_on_target_init(&srv_update_apply_broken_comp);
1526 target_prov_and_conf_with_imposer();
1527
1528 if (k_sem_take(&update_apply_sem, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1529 FAIL("Firmware Update Apply msg handler wasn't called");
1530 }
1531
1532 PASS();
1533 }
1534
test_target_fail_on_nothing(void)1535 static void test_target_fail_on_nothing(void)
1536 {
1537 common_fail_on_target_init(&target_comp);
1538 target_prov_and_conf_default();
1539
1540 if (k_sem_take(&dfu_ended, K_SECONDS(SEMAPHORE_TIMEOUT))) {
1541 FAIL("DFU failed");
1542 }
1543
1544 PASS();
1545 }
1546
test_target_dfu_stop(void)1547 static void test_target_dfu_stop(void)
1548 {
1549 dfu_target_effect = BT_MESH_DFU_EFFECT_NONE;
1550
1551 if (!recover) {
1552 bt_mesh_test_cfg_set(NULL, WAIT_TIME);
1553
1554 common_fail_on_target_init(expected_stop_phase == BT_MESH_DFU_PHASE_VERIFY_FAIL ?
1555 &target_comp : &broken_target_comp);
1556 target_prov_and_conf_with_imposer();
1557
1558 if (expected_stop_phase == BT_MESH_DFU_PHASE_VERIFY_FAIL) {
1559 dfu_verify_fail = true;
1560 if (k_sem_take(&dfu_verify_failed, K_SECONDS(DFU_TIMEOUT))) {
1561 FAIL("Phase not reached");
1562 }
1563 } else {
1564 /* Stop at BT_MESH_DFU_PHASE_TRANSFER_ACTIVE */
1565 if (k_sem_take(&dfu_started, K_SECONDS(DFU_TIMEOUT))) {
1566 FAIL("Phase not reached");
1567 }
1568 }
1569
1570 ASSERT_EQUAL(expected_stop_phase, dfu_srv.update.phase);
1571 PASS();
1572 return;
1573 }
1574
1575 bt_mesh_device_setup(&prov, &target_comp);
1576 bt_mesh_test_cfg_set(NULL, WAIT_TIME);
1577
1578 switch (expected_stop_phase) {
1579 case BT_MESH_DFU_PHASE_VERIFY:
1580 ASSERT_EQUAL(BT_MESH_DFU_PHASE_TRANSFER_ERR, dfu_srv.update.phase);
1581 if (k_sem_take(&dfu_verifying, K_SECONDS(DFU_TIMEOUT))) {
1582 FAIL("Phase not reached");
1583 }
1584 ASSERT_EQUAL(BT_MESH_DFU_PHASE_VERIFY, dfu_srv.update.phase);
1585 break;
1586 case BT_MESH_DFU_PHASE_VERIFY_OK:
1587 ASSERT_EQUAL(BT_MESH_DFU_PHASE_VERIFY, dfu_srv.update.phase);
1588 bt_mesh_dfu_srv_verified(&dfu_srv);
1589 ASSERT_EQUAL(BT_MESH_DFU_PHASE_VERIFY_OK, dfu_srv.update.phase);
1590 break;
1591 case BT_MESH_DFU_PHASE_APPLYING:
1592 ASSERT_EQUAL(BT_MESH_DFU_PHASE_VERIFY_FAIL, dfu_srv.update.phase);
1593 if (k_sem_take(&dfu_applying, K_SECONDS(DFU_TIMEOUT))) {
1594 FAIL("Phase not reached");
1595 }
1596 ASSERT_EQUAL(BT_MESH_DFU_PHASE_APPLYING, dfu_srv.update.phase);
1597 break;
1598 case BT_MESH_DFU_PHASE_APPLY_SUCCESS:
1599 ASSERT_EQUAL(BT_MESH_DFU_PHASE_APPLYING, dfu_srv.update.phase);
1600 bt_mesh_dfu_srv_applied(&dfu_srv);
1601 ASSERT_EQUAL(BT_MESH_DFU_PHASE_IDLE, dfu_srv.update.phase);
1602 break;
1603 default:
1604 FAIL("Wrong expected phase");
1605 break;
1606 }
1607
1608 ASSERT_EQUAL(0, dfu_srv.update.idx);
1609 PASS();
1610 }
1611
test_pre_init(void)1612 static void test_pre_init(void)
1613 {
1614 k_sem_init(&dfu_dist_ended, 0, 1);
1615 k_sem_init(&dfu_ended, 0, 1);
1616 k_sem_init(&caps_get_sem, 0, 1);
1617 k_sem_init(&update_get_sem, 0, 1);
1618 k_sem_init(&update_apply_sem, 0, 1);
1619 k_sem_init(&dfu_metadata_check_sem, 0, 1);
1620 k_sem_init(&dfu_verify_sem, 0, 1);
1621 k_sem_init(&dfu_cli_applied_sem, 0, 1);
1622 k_sem_init(&dfu_cli_confirmed_sem, 0, 1);
1623 k_sem_init(&lost_target_sem, 0, 1);
1624 k_sem_init(&dfu_started, 0, 1);
1625 k_sem_init(&dfu_verifying, 0, 1);
1626 k_sem_init(&dfu_verify_failed, 0, 1);
1627 k_sem_init(&dfu_applying, 0, 1);
1628 }
1629
1630 #define TEST_CASE(role, name, description) \
1631 { \
1632 .test_id = "dfu_" #role "_" #name, \
1633 .test_descr = description, \
1634 .test_pre_init_f = test_pre_init, \
1635 .test_args_f = test_args_parse, \
1636 .test_tick_f = bt_mesh_test_timeout, \
1637 .test_main_f = test_##role##_##name, \
1638 }
1639
1640 static const struct bst_test_instance test_dfu[] = {
1641 TEST_CASE(dist, dfu, "Distributor performs DFU"),
1642 TEST_CASE(dist, dfu_self_update, "Distributor performs DFU with self update"),
1643 TEST_CASE(dist, dfu_slot_create, "Distributor creates image slots"),
1644 TEST_CASE(dist, dfu_slot_create_recover,
1645 "Distributor recovers created image slots from persitent storage"),
1646 TEST_CASE(dist, dfu_slot_delete_all, "Distributor deletes all image slots"),
1647 TEST_CASE(dist, dfu_slot_check_delete_all,
1648 "Distributor checks if all slots are removed from persistent storage"),
1649 TEST_CASE(dist, dfu_slot_reservation,
1650 "Distributor checks that the correct number of slots can be reserved"),
1651 TEST_CASE(dist, dfu_slot_idempotency,
1652 "Distributor checks that the DFU slot APIs are idempotent"),
1653 TEST_CASE(cli, stop, "DFU Client stops at configured point of Firmware Distribution"),
1654 TEST_CASE(cli, fail_on_persistency, "DFU Client doesn't give up DFU Transfer"),
1655 TEST_CASE(cli, all_targets_lost_on_metadata,
1656 "All targets fail to check metadata and Client ends DFU Transfer"),
1657 TEST_CASE(cli, all_targets_lost_on_caps_get,
1658 "All targets fail to respond to caps get and Client ends DFU Transfer"),
1659 TEST_CASE(cli, all_targets_lost_on_update_get,
1660 "All targets fail to respond to update get and Client ends DFU Transfer"),
1661 TEST_CASE(cli, all_targets_lost_on_verify,
1662 "All targets fail on verify step and Client ends DFU Transfer"),
1663 TEST_CASE(cli, all_targets_lost_on_apply,
1664 "All targets fail on apply step and Client ends DFU Transfer"),
1665
1666 TEST_CASE(target, dfu_no_change, "Target node, Comp Data stays unchanged"),
1667 TEST_CASE(target, dfu_new_comp_no_rpr, "Target node, Comp Data changes, no RPR"),
1668 TEST_CASE(target, dfu_new_comp_rpr, "Target node, Comp Data changes, has RPR"),
1669 TEST_CASE(target, dfu_unprov, "Target node, Comp Data changes, unprovisioned"),
1670 TEST_CASE(target, fail_on_metadata, "Server rejects metadata"),
1671 TEST_CASE(target, fail_on_caps_get, "Server failing on Retrieve Capabilities procedure"),
1672 TEST_CASE(target, fail_on_update_get, "Server failing on Fw Update Get msg"),
1673 TEST_CASE(target, fail_on_verify, "Server rejects fw at Refresh step"),
1674 TEST_CASE(target, fail_on_apply, "Server failing on Fw Update Apply msg"),
1675 TEST_CASE(target, fail_on_nothing, "Non-failing server"),
1676 TEST_CASE(target, dfu_stop, "Server stops FD procedure at configured step"),
1677
1678 BSTEST_END_MARKER
1679 };
1680
test_dfu_install(struct bst_test_list * tests)1681 struct bst_test_list *test_dfu_install(struct bst_test_list *tests)
1682 {
1683 tests = bst_add_tests(tests, test_dfu);
1684 return tests;
1685 }
1686