1 /*
2 * Copyright (c) 2018 Laczen
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 /** @file
8 * @brief Settings functional test suite
9 *
10 */
11
12 #include <zephyr/kernel.h>
13 #include <zephyr/ztest.h>
14 #include <errno.h>
15 #include <zephyr/settings/settings.h>
16 #include <zephyr/logging/log.h>
17 LOG_MODULE_REGISTER(settings_basic_test);
18
19 #if defined(CONFIG_SETTINGS_FCB) || defined(CONFIG_SETTINGS_NVS)
20 #include <zephyr/storage/flash_map.h>
21 #if DT_HAS_CHOSEN(zephyr_settings_partition)
22 #define TEST_FLASH_AREA_ID DT_FIXED_PARTITION_ID(DT_CHOSEN(zephyr_settings_partition))
23 #endif
24 #elif defined(CONFIG_SETTINGS_FILE)
25 #include <zephyr/fs/fs.h>
26 #include <zephyr/fs/littlefs.h>
27 #else
28 #error "Settings backend not selected"
29 #endif
30
31 #ifndef TEST_FLASH_AREA_ID
32 #define TEST_FLASH_AREA storage_partition
33 #define TEST_FLASH_AREA_ID FIXED_PARTITION_ID(TEST_FLASH_AREA)
34 #endif
35
36 /* The standard test expects a cleared flash area. Make sure it has
37 * one.
38 */
ZTEST(settings_functional,test_clear_settings)39 ZTEST(settings_functional, test_clear_settings)
40 {
41 #if !defined(CONFIG_SETTINGS_FILE)
42 const struct flash_area *fap;
43 int rc;
44
45 rc = flash_area_open(TEST_FLASH_AREA_ID, &fap);
46
47 if (rc == 0) {
48 rc = flash_area_flatten(fap, 0, fap->fa_size);
49 flash_area_close(fap);
50 }
51 zassert_true(rc == 0, "clear settings failed");
52 #else
53 FS_LITTLEFS_DECLARE_DEFAULT_CONFIG(cstorage);
54
55 /* mounting info */
56 static struct fs_mount_t littlefs_mnt = {
57 .type = FS_LITTLEFS,
58 .fs_data = &cstorage,
59 .storage_dev = (void *)TEST_FLASH_AREA_ID,
60 .mnt_point = "/ff"
61 };
62
63 int rc;
64
65 rc = fs_mount(&littlefs_mnt);
66 zassert_true(rc == 0, "mounting littlefs [%d]\n", rc);
67
68 rc = fs_unlink(CONFIG_SETTINGS_FILE_PATH);
69 zassert_true(rc == 0 || rc == -ENOENT,
70 "can't delete config file%d\n", rc);
71 #endif
72 }
73
74 /*
75 * Test the two support routines that settings provides:
76 *
77 * settings_name_steq(name, key, next): compares the start of name with key
78 * settings_name_next(name, next): returns the location of the first
79 * separator
80 */
81
ZTEST(settings_functional,test_support_rtn)82 ZTEST(settings_functional, test_support_rtn)
83 {
84 const char test1[] = "bt/a/b/c/d";
85 const char test2[] = "bt/a/b/c/d=";
86 const char *next1, *next2;
87 int rc;
88
89 /* complete match: return 1, next = NULL */
90 rc = settings_name_steq(test1, "bt/a/b/c/d", &next1);
91 zassert_true(rc == 1, "_steq comparison failure");
92 zassert_is_null(next1, "_steq comparison next error");
93 rc = settings_name_steq(test2, "bt/a/b/c/d", &next2);
94 zassert_true(rc == 1, "_steq comparison failure");
95 zassert_is_null(next2, "_steq comparison next error");
96
97 /* partial match: return 1, next <> NULL */
98 rc = settings_name_steq(test1, "bt/a/b/c", &next1);
99 zassert_true(rc == 1, "_steq comparison failure");
100 zassert_not_null(next1, "_steq comparison next error");
101 zassert_equal_ptr(next1, test1+9, "next points to wrong location");
102 rc = settings_name_steq(test2, "bt/a/b/c", &next2);
103 zassert_true(rc == 1, "_steq comparison failure");
104 zassert_not_null(next2, "_steq comparison next error");
105 zassert_equal_ptr(next2, test2+9, "next points to wrong location");
106
107 /* no match: return 0, next = NULL */
108 rc = settings_name_steq(test1, "bta", &next1);
109 zassert_true(rc == 0, "_steq comparison failure");
110 zassert_is_null(next1, "_steq comparison next error");
111 rc = settings_name_steq(test2, "bta", &next2);
112 zassert_true(rc == 0, "_steq comparison failure");
113 zassert_is_null(next2, "_steq comparison next error");
114
115 /* no match: return 0, next = NULL */
116 rc = settings_name_steq(test1, "b", &next1);
117 zassert_true(rc == 0, "_steq comparison failure");
118 zassert_is_null(next1, "_steq comparison next error");
119 rc = settings_name_steq(test2, "b", &next2);
120 zassert_true(rc == 0, "_steq comparison failure");
121 zassert_is_null(next2, "_steq comparison next error");
122
123 /* first separator: return 2, next <> NULL */
124 rc = settings_name_next(test1, &next1);
125 zassert_true(rc == 2, "_next wrong return value");
126 zassert_not_null(next1, "_next wrong next");
127 zassert_equal_ptr(next1, test1+3, "next points to wrong location");
128 rc = settings_name_next(test2, &next2);
129 zassert_true(rc == 2, "_next wrong return value");
130 zassert_not_null(next2, "_next wrong next");
131 zassert_equal_ptr(next2, test2+3, "next points to wrong location");
132
133 /* second separator: return 1, next <> NULL */
134 rc = settings_name_next(next1, &next1);
135 zassert_true(rc == 1, "_next wrong return value");
136 zassert_not_null(next1, "_next wrong next");
137 zassert_equal_ptr(next1, test1+5, "next points to wrong location");
138 rc = settings_name_next(next2, &next2);
139 zassert_true(rc == 1, "_next wrong return value");
140 zassert_not_null(next2, "_next wrong next");
141 zassert_equal_ptr(next2, test2+5, "next points to wrong location");
142
143 /* third separator: return 1, next <> NULL */
144 rc = settings_name_next(next1, &next1);
145 zassert_true(rc == 1, "_next wrong return value");
146 zassert_not_null(next1, "_next wrong next");
147 rc = settings_name_next(next2, &next2);
148 zassert_true(rc == 1, "_next wrong return value");
149 zassert_not_null(next2, "_next wrong next");
150
151 /* fourth separator: return 1, next <> NULL */
152 rc = settings_name_next(next1, &next1);
153 zassert_true(rc == 1, "_next wrong return value");
154 zassert_not_null(next1, "_next wrong next");
155 rc = settings_name_next(next2, &next2);
156 zassert_true(rc == 1, "_next wrong return value");
157 zassert_not_null(next2, "_next wrong next");
158
159 /* fifth separator: return 1, next == NULL */
160 rc = settings_name_next(next1, &next1);
161 zassert_true(rc == 1, "_next wrong return value");
162 zassert_is_null(next1, "_next wrong next");
163 rc = settings_name_next(next2, &next2);
164 zassert_true(rc == 1, "_next wrong return value");
165 zassert_is_null(next2, "_next wrong next");
166
167 }
168
169 struct stored_data {
170 uint8_t val1;
171 uint8_t val2;
172 uint8_t val3;
173 bool en1;
174 bool en2;
175 bool en3;
176 };
177
178 struct stored_data data;
179
val1_set(const char * key,size_t len,settings_read_cb read_cb,void * cb_arg)180 int val1_set(const char *key, size_t len, settings_read_cb read_cb,
181 void *cb_arg)
182 {
183 data.val1 = 1;
184 return 0;
185 }
val1_commit(void)186 int val1_commit(void)
187 {
188 data.en1 = true;
189 return 0;
190 }
191 static struct settings_handler val1_settings = {
192 .name = "ps",
193 .h_set = val1_set,
194 .h_commit = val1_commit,
195 };
196
val2_set(const char * key,size_t len,settings_read_cb read_cb,void * cb_arg)197 int val2_set(const char *key, size_t len, settings_read_cb read_cb,
198 void *cb_arg)
199 {
200 data.val2 = 2;
201 return 0;
202 }
val2_commit(void)203 int val2_commit(void)
204 {
205 data.en2 = true;
206 return 0;
207 }
208 static struct settings_handler val2_settings = {
209 .name = "ps/ss/ss",
210 .h_set = val2_set,
211 .h_commit = val2_commit,
212 };
213
val3_set(const char * key,size_t len,settings_read_cb read_cb,void * cb_arg)214 int val3_set(const char *key, size_t len, settings_read_cb read_cb,
215 void *cb_arg)
216 {
217 data.val3 = 3;
218 return 0;
219 }
val3_commit(void)220 int val3_commit(void)
221 {
222 data.en3 = true;
223 return 0;
224 }
225 static struct settings_handler val3_settings = {
226 .name = "ps/ss",
227 .h_set = val3_set,
228 .h_commit = val3_commit,
229 };
230
231 /* helper routine to remove a handler from settings */
settings_deregister(struct settings_handler * handler)232 int settings_deregister(struct settings_handler *handler)
233 {
234 extern sys_slist_t settings_handlers;
235
236 return sys_slist_find_and_remove(&settings_handlers, &handler->node);
237 }
238
ZTEST(settings_functional,test_register_and_loading)239 ZTEST(settings_functional, test_register_and_loading)
240 {
241 int rc, err;
242 uint8_t val = 0;
243
244 rc = settings_subsys_init();
245 zassert_true(rc == 0, "subsys init failed");
246
247
248 settings_save_one("ps/ss/ss/val2", &val, sizeof(uint8_t));
249
250 memset(&data, 0, sizeof(struct stored_data));
251
252 rc = settings_register(&val1_settings);
253 zassert_true(rc == 0, "register of val1 settings failed");
254
255 /* when we load settings now data.val1 should receive the value*/
256 rc = settings_load();
257 zassert_true(rc == 0, "settings_load failed");
258 err = (data.val1 == 1) && (data.val2 == 0) && (data.val3 == 0);
259 zassert_true(err, "wrong data value found");
260 /* commit is only called for val1_settings */
261 err = (data.en1) && (!data.en2) && (!data.en3);
262 zassert_true(err, "wrong data enable found");
263
264 /* Next register should be ok */
265 rc = settings_register(&val2_settings);
266 zassert_true(rc == 0, "register of val2 settings failed");
267
268 /* Next register should fail (same name) */
269 rc = settings_register(&val2_settings);
270 zassert_true(rc == -EEXIST, "double register of val2 settings allowed");
271
272 memset(&data, 0, sizeof(struct stored_data));
273 /* when we load settings now data.val2 should receive the value*/
274 rc = settings_load();
275 zassert_true(rc == 0, "settings_load failed");
276 err = (data.val1 == 0) && (data.val2 == 2) && (data.val3 == 0);
277 zassert_true(err, "wrong data value found");
278 /* commit is called for val1_settings and val2_settings*/
279 err = (data.en1) && (data.en2) && (!data.en3);
280 zassert_true(err, "wrong data enable found");
281
282 settings_save_one("ps/ss/val3", &val, sizeof(uint8_t));
283 memset(&data, 0, sizeof(struct stored_data));
284 /* when we load settings now data.val2 and data.val1 should receive a
285 * value
286 */
287 rc = settings_load();
288 zassert_true(rc == 0, "settings_load failed");
289 err = (data.val1 == 1) && (data.val2 == 2) && (data.val3 == 0);
290 zassert_true(err, "wrong data value found");
291 /* commit is called for val1_settings and val2_settings*/
292 err = (data.en1) && (data.en2) && (!data.en3);
293 zassert_true(err, "wrong data enable found");
294
295 /* val3 settings should be inserted in between val1_settings and
296 * val2_settings
297 */
298 rc = settings_register(&val3_settings);
299 zassert_true(rc == 0, "register of val3 settings failed");
300 memset(&data, 0, sizeof(struct stored_data));
301 /* when we load settings now data.val2 and data.val3 should receive a
302 * value
303 */
304 rc = settings_load();
305 zassert_true(rc == 0, "settings_load failed");
306 err = (data.val1 == 0) && (data.val2 == 2) && (data.val3 == 3);
307 zassert_true(err, "wrong data value found");
308 /* commit is called for val1_settings, val2_settings and val3_settings
309 */
310 err = (data.en1) && (data.en2) && (data.en3);
311 zassert_true(err, "wrong data enable found");
312
313 settings_save_one("ps/val1", &val, sizeof(uint8_t));
314 memset(&data, 0, sizeof(struct stored_data));
315 /* when we load settings all data should receive a value loaded */
316 rc = settings_load();
317 zassert_true(rc == 0, "settings_load failed");
318 err = (data.val1 == 1) && (data.val2 == 2) && (data.val3 == 3);
319 zassert_true(err, "wrong data value found");
320 /* commit is called for all settings*/
321 err = (data.en1) && (data.en2) && (data.en3);
322 zassert_true(err, "wrong data enable found");
323
324 memset(&data, 0, sizeof(struct stored_data));
325 /* test subtree loading: subtree "ps/ss" data.val2 and data.val3 should
326 * receive a value
327 */
328 rc = settings_load_subtree("ps/ss");
329 zassert_true(rc == 0, "settings_load failed");
330 err = (data.val1 == 0) && (data.val2 == 2) && (data.val3 == 3);
331 zassert_true(err, "wrong data value found");
332 /* commit is called for val2_settings and val3_settings */
333 err = (!data.en1) && (data.en2) && (data.en3);
334 zassert_true(err, "wrong data enable found");
335
336 memset(&data, 0, sizeof(struct stored_data));
337 /* test subtree loading: subtree "ps/ss/ss" only data.val2 should
338 * receive a value
339 */
340 rc = settings_load_subtree("ps/ss/ss");
341 zassert_true(rc == 0, "settings_load failed");
342 err = (data.val1 == 0) && (data.val2 == 2) && (data.val3 == 0);
343 zassert_true(err, "wrong data value found");
344 /* commit is called only for val2_settings */
345 err = (!data.en1) && (data.en2) && (!data.en3);
346 zassert_true(err, "wrong data enable found");
347
348 /* clean up by deregistering settings_handler */
349 rc = settings_deregister(&val1_settings);
350 zassert_true(rc, "deregistering val1_settings failed");
351
352 rc = settings_deregister(&val2_settings);
353 zassert_true(rc, "deregistering val2_settings failed");
354
355 rc = settings_deregister(&val3_settings);
356 zassert_true(rc, "deregistering val3_settings failed");
357 }
358
val123_set(const char * key,size_t len,settings_read_cb read_cb,void * cb_arg)359 int val123_set(const char *key, size_t len,
360 settings_read_cb read_cb, void *cb_arg)
361 {
362 int rc;
363 uint8_t val;
364
365 zassert_equal(1, len, "Unexpected size");
366
367 rc = read_cb(cb_arg, &val, sizeof(val));
368 zassert_equal(sizeof(val), rc, "read_cb failed");
369
370 if (!strcmp("1", key)) {
371 data.val1 = val;
372 data.en1 = true;
373 return 0;
374 }
375 if (!strcmp("2", key)) {
376 data.val2 = val;
377 data.en2 = true;
378 return 0;
379 }
380 if (!strcmp("3", key)) {
381 data.val3 = val;
382 data.en3 = true;
383 return 0;
384 }
385
386 zassert_unreachable("Unexpected key value: %s", key);
387
388 return 0;
389 }
390
391 static struct settings_handler val123_settings = {
392 .name = "val",
393 .h_set = val123_set,
394 };
395
396 unsigned int direct_load_cnt;
397 uint8_t val_directly_loaded;
398
direct_loader(const char * key,size_t len,settings_read_cb read_cb,void * cb_arg,void * param)399 int direct_loader(
400 const char *key,
401 size_t len,
402 settings_read_cb read_cb,
403 void *cb_arg,
404 void *param)
405 {
406 int rc;
407 uint8_t val;
408
409 zassert_equal(0x1234, (size_t)param);
410
411 zassert_equal(1, len);
412 zassert_is_null(key, "Unexpected key: %s", key);
413
414
415 zassert_not_null(cb_arg, NULL);
416 rc = read_cb(cb_arg, &val, sizeof(val));
417 zassert_equal(sizeof(val), rc);
418
419 val_directly_loaded = val;
420 direct_load_cnt += 1;
421 return 0;
422 }
423
424
ZTEST(settings_functional,test_direct_loading)425 ZTEST(settings_functional, test_direct_loading)
426 {
427 int rc;
428 uint8_t val;
429
430 settings_subsys_init();
431 val = 11;
432 settings_save_one("val/1", &val, sizeof(uint8_t));
433 val = 23;
434 settings_save_one("val/2", &val, sizeof(uint8_t));
435 val = 35;
436 settings_save_one("val/3", &val, sizeof(uint8_t));
437
438 rc = settings_register(&val123_settings);
439 zassert_true(rc == 0);
440 memset(&data, 0, sizeof(data));
441
442 rc = settings_load();
443 zassert_true(rc == 0);
444
445 zassert_equal(11, data.val1);
446 zassert_equal(23, data.val2);
447 zassert_equal(35, data.val3);
448
449 /* Load subtree */
450 memset(&data, 0, sizeof(data));
451
452 rc = settings_load_subtree("val/2");
453 zassert_true(rc == 0);
454
455 zassert_equal(0, data.val1);
456 zassert_equal(23, data.val2);
457 zassert_equal(0, data.val3);
458
459 /* Direct loading now */
460 memset(&data, 0, sizeof(data));
461 val_directly_loaded = 0;
462 direct_load_cnt = 0;
463 rc = settings_load_subtree_direct(
464 "val/2",
465 direct_loader,
466 (void *)0x1234);
467 zassert_true(rc == 0);
468 zassert_equal(0, data.val1);
469 zassert_equal(0, data.val2);
470 zassert_equal(0, data.val3);
471
472 zassert_equal(1, direct_load_cnt);
473 zassert_equal(23, val_directly_loaded);
474 settings_deregister(&val123_settings);
475 }
476
477 struct test_loading_data {
478 const char *n;
479 const char *v;
480 };
481
482 /* Final data */
483 static const struct test_loading_data data_final[] = {
484 { .n = "val/1", .v = "final 1" },
485 { .n = "val/2", .v = "final 2" },
486 { .n = "val/3", .v = "final 3" },
487 { .n = "val/4", .v = "final 4" },
488 { .n = NULL }
489 };
490
491 /* The counter of the callback called */
492 static unsigned int data_final_called[ARRAY_SIZE(data_final)];
493
494
filtered_loader(const char * key,size_t len,settings_read_cb read_cb,void * cb_arg)495 static int filtered_loader(
496 const char *key,
497 size_t len,
498 settings_read_cb read_cb,
499 void *cb_arg)
500 {
501 int rc;
502 const char *next;
503 char buf[32];
504 const struct test_loading_data *ldata;
505
506 printk("-- Called: %s\n", key);
507
508 /* Searching for a element in an array */
509 for (ldata = data_final; ldata->n; ldata += 1) {
510 if (settings_name_steq(key, ldata->n, &next)) {
511 break;
512 }
513 }
514 zassert_not_null(ldata->n, "Unexpected data name: %s", key);
515 zassert_is_null(next, NULL);
516 zassert_equal(strlen(ldata->v) + 1, len, "e: \"%s\", a:\"%s\"", ldata->v, buf);
517 zassert_true(len <= sizeof(buf));
518
519 rc = read_cb(cb_arg, buf, len);
520 zassert_equal(len, rc);
521
522 zassert_false(strcmp(ldata->v, buf), "e: \"%s\", a:\"%s\"", ldata->v, buf);
523
524 /* Count an element that was properly loaded */
525 data_final_called[ldata - data_final] += 1;
526
527 return 0;
528 }
529
530 static struct settings_handler filtered_loader_settings = {
531 .name = "filtered_test",
532 .h_set = filtered_loader,
533 };
534
535
direct_filtered_loader(const char * key,size_t len,settings_read_cb read_cb,void * cb_arg,void * param)536 static int direct_filtered_loader(
537 const char *key,
538 size_t len,
539 settings_read_cb read_cb,
540 void *cb_arg,
541 void *param)
542 {
543 zassert_equal(0x3456, (size_t)param);
544 return filtered_loader(key, len, read_cb, cb_arg);
545 }
546
547
ZTEST(settings_functional,test_direct_loading_filter)548 ZTEST(settings_functional, test_direct_loading_filter)
549 {
550 int rc;
551 const struct test_loading_data *ldata;
552 const char *prefix = filtered_loader_settings.name;
553 char buffer[48];
554 size_t n;
555
556 /* Duplicated data */
557 static const struct test_loading_data data_duplicates[] = {
558 { .n = "val/1", .v = "dup abc" },
559 { .n = "val/2", .v = "dup 123" },
560 { .n = "val/3", .v = "dup 11" },
561 { .n = "val/4", .v = "dup 34" },
562 { .n = "val/1", .v = "dup 56" },
563 { .n = "val/2", .v = "dup 7890" },
564 { .n = "val/4", .v = "dup niety" },
565 { .n = "val/3", .v = "dup er" },
566 { .n = "val/3", .v = "dup super" },
567 { .n = "val/3", .v = "dup xxx" },
568 { .n = NULL }
569 };
570
571 settings_subsys_init();
572 /* Data that is going to be deleted */
573 strcpy(buffer, prefix);
574 strcat(buffer, "/to_delete");
575 settings_save_one(buffer, "1", 2);
576 (void) settings_delete(buffer);
577
578 /* Saving all the data */
579 for (ldata = data_duplicates; ldata->n; ++ldata) {
580 strcpy(buffer, prefix);
581 strcat(buffer, "/");
582 strcat(buffer, ldata->n);
583 settings_save_one(buffer, ldata->v, strlen(ldata->v) + 1);
584 }
585 for (ldata = data_final; ldata->n; ++ldata) {
586 strcpy(buffer, prefix);
587 strcat(buffer, "/");
588 strcat(buffer, ldata->n);
589 settings_save_one(buffer, ldata->v, strlen(ldata->v) + 1);
590 }
591
592
593 memset(data_final_called, 0, sizeof(data_final_called));
594
595 rc = settings_load_subtree_direct(
596 prefix,
597 direct_filtered_loader,
598 (void *)0x3456);
599 zassert_equal(0, rc);
600
601 /* Check if all the data was called */
602 for (n = 0; data_final[n].n; ++n) {
603 zassert_equal(1, data_final_called[n],
604 "Unexpected number of calls (%u) of (%s) element",
605 n, data_final[n].n);
606 }
607
608 rc = settings_register(&filtered_loader_settings);
609 zassert_true(rc == 0);
610
611 rc = settings_load_subtree(prefix);
612 zassert_equal(0, rc);
613
614 /* Check if all the data was called */
615 for (n = 0; data_final[n].n; ++n) {
616 zassert_equal(2, data_final_called[n],
617 "Unexpected number of calls (%u) of (%s) element",
618 n, data_final[n].n);
619 }
620 settings_deregister(&filtered_loader_settings);
621 }
622