1 /*
2  * Copyright (c) 2023 Nordic Semiconductor
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  *
6  * Composition Data Page 1 (CDP1) test
7  */
8 
9 #include "mesh_test.h"
10 
11 #include <stddef.h>
12 #include <string.h>
13 #include <mesh/access.h>
14 #include <mesh/net.h>
15 
16 #include <zephyr/logging/log.h>
17 #define LOG_MODULE_NAME test_cdp1
18 LOG_MODULE_REGISTER(LOG_MODULE_NAME, LOG_LEVEL_INF);
19 
20 #define NODE_ADDR 0x00a1
21 #define WAIT_TIME 60 /* seconds */
22 
23 #define TEST_MODEL_ID_1     0x2a2a
24 #define TEST_MODEL_ID_2     0x2b2b
25 #define TEST_MODEL_ID_3     0x2c2c
26 #define TEST_MODEL_ID_4     0x2d2d
27 #define TEST_MODEL_ID_5     0x2e2e
28 #define TEST_MODEL_ID_6     0x2f2f
29 #define TEST_VND_MODEL_ID_1 0x3a3a
30 
31 #define TEST_MODEL_DECLARE(number)                                                                 \
32 	static int model_##number##_init(const struct bt_mesh_model *model);			   \
33 	static const struct bt_mesh_model_cb test_model_##number##_cb = {                          \
34 		.init = model_##number##_init,                                                     \
35 	};                                                                                         \
36 	static const struct bt_mesh_model_op model_op_##number[] = {                               \
37 		BT_MESH_MODEL_OP_END,                                                              \
38 	};
39 
40 TEST_MODEL_DECLARE(1);
41 TEST_MODEL_DECLARE(2);
42 TEST_MODEL_DECLARE(3);
43 TEST_MODEL_DECLARE(4);
44 TEST_MODEL_DECLARE(5);
45 TEST_MODEL_DECLARE(6);
46 TEST_MODEL_DECLARE(vnd1);
47 
48 static uint8_t app_key[16] = {0xaa};
49 static uint8_t net_key[16] = {0xcc};
50 
51 static const struct bt_mesh_test_cfg node_1_cfg = {
52 	.addr = NODE_ADDR,
53 	.dev_key = {0xaa},
54 };
55 
56 static struct bt_mesh_prov prov;
57 static struct bt_mesh_cfg_cli cfg_cli;
58 
59 static const struct bt_mesh_model models_1[] = {
60 	BT_MESH_MODEL_CFG_SRV,
61 	BT_MESH_MODEL_CFG_CLI(&cfg_cli),
62 	BT_MESH_MODEL_CB(TEST_MODEL_ID_1, model_op_1, NULL, NULL, &test_model_1_cb),
63 	BT_MESH_MODEL_CB(TEST_MODEL_ID_2, model_op_2, NULL, NULL, &test_model_2_cb),
64 	BT_MESH_MODEL_CB(TEST_MODEL_ID_3, model_op_3, NULL, NULL, &test_model_3_cb),
65 };
66 
67 static const struct bt_mesh_model models_2[] = {
68 	BT_MESH_MODEL_CB(TEST_MODEL_ID_4, model_op_4, NULL, NULL, &test_model_4_cb),
69 };
70 
71 static const struct bt_mesh_model models_3[] = {
72 	BT_MESH_MODEL_CB(TEST_MODEL_ID_5, model_op_5, NULL, NULL, &test_model_5_cb),
73 };
74 
75 static const struct bt_mesh_model models_4[] = {
76 	BT_MESH_MODEL_CB(TEST_MODEL_ID_6, model_op_6, NULL, NULL, &test_model_6_cb),
77 };
78 
79 static const struct bt_mesh_model models_vnd1[] = {
80 	BT_MESH_MODEL_VND_CB(TEST_VND_COMPANY_ID, TEST_VND_MODEL_ID_1, model_op_vnd1, NULL, NULL,
81 			     &test_model_vnd1_cb),
82 };
83 
84 static const struct bt_mesh_elem elems[] = {
85 	BT_MESH_ELEM(0, models_1, models_vnd1),
86 	BT_MESH_ELEM(1, models_2, BT_MESH_MODEL_NONE),
87 	BT_MESH_ELEM(2, models_3, BT_MESH_MODEL_NONE),
88 	BT_MESH_ELEM(3, models_3, BT_MESH_MODEL_NONE),
89 	BT_MESH_ELEM(4, models_4, BT_MESH_MODEL_NONE),
90 };
91 
92 static const struct bt_mesh_comp comp = {
93 	.cid = TEST_VND_COMPANY_ID,
94 	.vid = 0xdead,
95 	.pid = 0xface,
96 	.elem = elems,
97 	.elem_count = ARRAY_SIZE(elems),
98 };
99 
100 /* The extensions and correspondence between models are as follows:
101  * Within elements:
102  * E0: M2 extends M1. VND1 extends M2. M3 and VND1 corresponds.
103  *
104  * Between elements:
105  * M3 on E0 extends M4 on E1.
106  * M2 on E0 and M4 on E1 corresponds.
107  * M6 on E4 extends M1 on E0
108  */
model_1_init(const struct bt_mesh_model * model)109 static int model_1_init(const struct bt_mesh_model *model)
110 {
111 	return 0;
112 }
113 
model_2_init(const struct bt_mesh_model * model)114 static int model_2_init(const struct bt_mesh_model *model)
115 {
116 	ASSERT_OK(bt_mesh_model_extend(model, bt_mesh_model_find(&elems[0], TEST_MODEL_ID_1)));
117 	return 0;
118 }
119 
model_3_init(const struct bt_mesh_model * model)120 static int model_3_init(const struct bt_mesh_model *model)
121 {
122 	return 0;
123 }
124 
model_4_init(const struct bt_mesh_model * model)125 static int model_4_init(const struct bt_mesh_model *model)
126 {
127 	ASSERT_OK(bt_mesh_model_extend(bt_mesh_model_find(&elems[0], TEST_MODEL_ID_3), model));
128 	ASSERT_OK(bt_mesh_model_correspond(model, bt_mesh_model_find(&elems[0], TEST_MODEL_ID_2)));
129 	return 0;
130 }
131 
model_5_init(const struct bt_mesh_model * model)132 static int model_5_init(const struct bt_mesh_model *model)
133 {
134 	return 0;
135 }
136 
model_6_init(const struct bt_mesh_model * model)137 static int model_6_init(const struct bt_mesh_model *model)
138 {
139 	ASSERT_OK(bt_mesh_model_extend(model, bt_mesh_model_find(&elems[0], TEST_MODEL_ID_1)));
140 	return 0;
141 }
142 
model_vnd1_init(const struct bt_mesh_model * model)143 static int model_vnd1_init(const struct bt_mesh_model *model)
144 {
145 	ASSERT_OK(bt_mesh_model_extend(model, bt_mesh_model_find(&elems[0], TEST_MODEL_ID_1)));
146 	ASSERT_OK(bt_mesh_model_correspond(model, bt_mesh_model_find(&elems[0], TEST_MODEL_ID_3)));
147 	return 0;
148 }
149 
150 /* Hardcoded version of the CDP1 fields.
151  * Extensions are named extending-model_base-model.
152  */
153 
154 static struct bt_mesh_comp_p1_ext_item test_p1_ext_mod2_mod1 = {.type = SHORT,
155 								.short_item = {
156 									.elem_offset = 0,
157 									.mod_item_idx = 2,
158 								}};
159 
160 static struct bt_mesh_comp_p1_ext_item test_p1_ext_vnd1_mod1 = {.type = SHORT,
161 								.short_item = {
162 									.elem_offset = 0,
163 									.mod_item_idx = 2,
164 								}};
165 
166 static struct bt_mesh_comp_p1_ext_item test_p1_ext_mod3_mod4 = {.type = SHORT,
167 								.short_item = {
168 									.elem_offset = 7,
169 									.mod_item_idx = 0,
170 								}};
171 
172 static struct bt_mesh_comp_p1_ext_item test_p1_ext_mod6_mod1 = {.type = LONG,
173 								.long_item = {
174 									.elem_offset = 4,
175 									.mod_item_idx = 2,
176 								}};
177 
178 static const struct bt_mesh_comp_p1_model_item test_p1_cfg_srv_mod = {
179 	.cor_present = 0,
180 	.format = 0,
181 	.ext_item_cnt = 0,
182 };
183 
184 static const struct bt_mesh_comp_p1_model_item test_p1_cfg_cli_mod = {
185 	.cor_present = 0,
186 	.format = 0,
187 	.ext_item_cnt = 0,
188 };
189 
190 static const struct bt_mesh_comp_p1_model_item test_p1_mod1 = {
191 	.cor_present = 0,
192 	.format = 0,
193 	.ext_item_cnt = 0,
194 };
195 
196 static const struct bt_mesh_comp_p1_model_item test_p1_mod2 = {
197 	.cor_present = 1,
198 	.format = 0,
199 	.ext_item_cnt = 1,
200 	.cor_id = 0,
201 };
202 
203 static const struct bt_mesh_comp_p1_model_item test_p1_mod3 = {
204 	.cor_present = 1,
205 	.format = 0,
206 	.ext_item_cnt = 1,
207 	.cor_id = 0,
208 };
209 
210 static const struct bt_mesh_comp_p1_model_item test_p1_mod4 = {
211 	.cor_present = 1,
212 	.format = 0,
213 	.ext_item_cnt = 0,
214 	.cor_id = 0,
215 };
216 
217 static const struct bt_mesh_comp_p1_model_item test_p1_mod5 = {
218 	.cor_present = 0,
219 	.format = 0,
220 	.ext_item_cnt = 0,
221 };
222 
223 static const struct bt_mesh_comp_p1_model_item test_p1_mod6 = {
224 	.cor_present = 0,
225 	.format = 1,
226 	.ext_item_cnt = 1,
227 };
228 
229 static const struct bt_mesh_comp_p1_model_item test_p1_vnd1 = {
230 	.cor_present = 1,
231 	.format = 0,
232 	.ext_item_cnt = 1,
233 	.cor_id = 0,
234 };
235 
236 static const struct bt_mesh_comp_p1_model_item test_p1_elem0_models[] = {
237 	test_p1_cfg_srv_mod, test_p1_cfg_cli_mod, test_p1_mod1,
238 	test_p1_mod2,        test_p1_mod3,        test_p1_vnd1,
239 };
240 
241 static const struct bt_mesh_comp_p1_model_item test_p1_elem1_models[] = {
242 	test_p1_mod4,
243 };
244 
245 static const struct bt_mesh_comp_p1_model_item test_p1_elem2_models[] = {
246 	test_p1_mod5,
247 };
248 
249 static const struct bt_mesh_comp_p1_model_item test_p1_elem3_models[] = {
250 	test_p1_mod5,
251 };
252 
253 static const struct bt_mesh_comp_p1_model_item test_p1_elem4_models[] = {
254 	test_p1_mod6,
255 };
256 
257 static const struct bt_mesh_comp_p1_model_item *test_p1_elem_models[] = {
258 	test_p1_elem0_models, test_p1_elem1_models, test_p1_elem2_models,
259 	test_p1_elem3_models, test_p1_elem4_models,
260 };
261 
262 static const struct bt_mesh_comp_p1_elem test_p1_elem0 = {
263 	.nsig = 5,
264 	.nvnd = 1,
265 };
266 
267 static const struct bt_mesh_comp_p1_elem test_p1_elem1 = {
268 	.nsig = 1,
269 	.nvnd = 0,
270 };
271 
272 static const struct bt_mesh_comp_p1_elem test_p1_elem2 = {
273 	.nsig = 1,
274 	.nvnd = 0,
275 };
276 
277 static const struct bt_mesh_comp_p1_elem test_p1_elem3 = {
278 	.nsig = 1,
279 	.nvnd = 0,
280 };
281 
282 static const struct bt_mesh_comp_p1_elem test_p1_elem4 = {
283 	.nsig = 1,
284 	.nvnd = 0,
285 };
286 
287 static struct bt_mesh_comp_p1_elem test_p1_elems[] = {
288 	test_p1_elem0, test_p1_elem1, test_p1_elem2, test_p1_elem3, test_p1_elem4,
289 };
290 
provision_and_configure(struct bt_mesh_test_cfg cfg)291 static void provision_and_configure(struct bt_mesh_test_cfg cfg)
292 {
293 	int err;
294 	uint8_t status;
295 
296 	err = bt_mesh_provision(net_key, 0, 0, 0, cfg.addr, cfg.dev_key);
297 	if (err) {
298 		FAIL("Provisioning failed (err %d)", err);
299 	}
300 
301 	err = bt_mesh_cfg_cli_app_key_add(0, cfg.addr, 0, 0, app_key, &status);
302 	if (err || status) {
303 		FAIL("AppKey add failed (err %d, status %u)", err, status);
304 	}
305 }
306 
verify_model_item(struct bt_mesh_comp_p1_model_item * mod_item,int elem_idx,int mod_idx,int offset)307 static void verify_model_item(struct bt_mesh_comp_p1_model_item *mod_item, int elem_idx,
308 			      int mod_idx, int offset)
309 {
310 	ASSERT_EQUAL(test_p1_elem_models[elem_idx][mod_idx + offset].cor_present,
311 		     mod_item->cor_present);
312 	ASSERT_EQUAL(test_p1_elem_models[elem_idx][mod_idx + offset].format, mod_item->format);
313 	ASSERT_EQUAL(test_p1_elem_models[elem_idx][mod_idx + offset].ext_item_cnt,
314 		     mod_item->ext_item_cnt);
315 	if (mod_item->cor_present) {
316 		ASSERT_EQUAL(test_p1_elem_models[elem_idx][mod_idx + offset].cor_id,
317 			     mod_item->cor_id);
318 	}
319 }
320 
verify_ext_item(struct bt_mesh_comp_p1_ext_item * ext_item,int elem_idx,int mod_idx,int offset)321 static void verify_ext_item(struct bt_mesh_comp_p1_ext_item *ext_item, int elem_idx, int mod_idx,
322 			    int offset)
323 {
324 	struct bt_mesh_comp_p1_ext_item *test_p1_ext_item;
325 
326 	switch (elem_idx * 100 + (mod_idx + offset)) {
327 	case 3: /* elem_idx=0, mod_idx=3, offset = 0 */
328 		test_p1_ext_item = &test_p1_ext_mod2_mod1;
329 		break;
330 	case 4: /* elem_idx=0, mod_idx=4, offset = 0 */
331 		test_p1_ext_item = &test_p1_ext_mod3_mod4;
332 		break;
333 	case 5: /* elem_idx=0, mod_idx=0, offset = 5 */
334 		test_p1_ext_item = &test_p1_ext_vnd1_mod1;
335 		break;
336 	case 400: /* elem_idx=4, mod_idx=0, offset = 0 */
337 		test_p1_ext_item = &test_p1_ext_mod6_mod1;
338 		break;
339 	default:
340 		FAIL("Unexpected call to %s (elem %d, mod %d, offset %d)", __func__, elem_idx,
341 		     mod_idx, offset);
342 	}
343 
344 	ASSERT_EQUAL(test_p1_ext_item->type, ext_item->type);
345 	if (ext_item->type == SHORT) {
346 		ASSERT_EQUAL(test_p1_ext_item->short_item.elem_offset,
347 			     ext_item->short_item.elem_offset);
348 		ASSERT_EQUAL(test_p1_ext_item->short_item.mod_item_idx,
349 			     ext_item->short_item.mod_item_idx);
350 	} else {
351 		ASSERT_EQUAL(test_p1_ext_item->long_item.elem_offset,
352 			     ext_item->long_item.elem_offset);
353 		ASSERT_EQUAL(test_p1_ext_item->long_item.mod_item_idx,
354 			     ext_item->long_item.mod_item_idx);
355 	}
356 }
357 
verify_cdp1(struct bt_mesh_comp_p1_elem * p1_elem,struct bt_mesh_comp_p1_model_item * mod_item,struct bt_mesh_comp_p1_ext_item * ext_item,struct net_buf_simple * p1_dev_comp)358 static void verify_cdp1(struct bt_mesh_comp_p1_elem *p1_elem,
359 			struct bt_mesh_comp_p1_model_item *mod_item,
360 			struct bt_mesh_comp_p1_ext_item *ext_item,
361 			struct net_buf_simple *p1_dev_comp)
362 {
363 	int elem_idx = 0;
364 
365 	while (bt_mesh_comp_p1_elem_pull(p1_dev_comp, p1_elem)) {
366 		ASSERT_EQUAL(test_p1_elems[elem_idx].nsig, p1_elem->nsig);
367 		ASSERT_EQUAL(test_p1_elems[elem_idx].nvnd, p1_elem->nvnd);
368 
369 		for (int mod_idx = 0; mod_idx < p1_elem->nsig; mod_idx++) {
370 			if (bt_mesh_comp_p1_item_pull(p1_elem, mod_item)) {
371 				verify_model_item(mod_item, elem_idx, mod_idx, 0);
372 			}
373 
374 			for (int ext_mod_idx = 0; ext_mod_idx < mod_item->ext_item_cnt;
375 			     ext_mod_idx++) {
376 				bt_mesh_comp_p1_pull_ext_item(mod_item, ext_item);
377 				verify_ext_item(ext_item, elem_idx, mod_idx, 0);
378 			}
379 		}
380 
381 		for (int mod_idx = 0; mod_idx < p1_elem->nvnd; mod_idx++) {
382 			if (bt_mesh_comp_p1_item_pull(p1_elem, mod_item)) {
383 				verify_model_item(mod_item, elem_idx, mod_idx, p1_elem->nsig);
384 			}
385 
386 			for (int ext_mod_idx = 0; ext_mod_idx < mod_item->ext_item_cnt;
387 			     ext_mod_idx++) {
388 				bt_mesh_comp_p1_pull_ext_item(mod_item, ext_item);
389 				verify_ext_item(ext_item, elem_idx, mod_idx, p1_elem->nsig);
390 			}
391 		}
392 		elem_idx++;
393 	}
394 }
395 
test_node_data_comparison(void)396 static void test_node_data_comparison(void)
397 {
398 	bt_mesh_test_cfg_set(NULL, WAIT_TIME);
399 	bt_mesh_device_setup(&prov, &comp);
400 	provision_and_configure(node_1_cfg);
401 
402 	NET_BUF_SIMPLE_DEFINE(p1_dev_comp, 500);
403 	uint8_t page_rsp;
404 
405 	ASSERT_OK(bt_mesh_cfg_cli_comp_data_get(0, node_1_cfg.addr, 1, &page_rsp, &p1_dev_comp));
406 	ASSERT_EQUAL(1, page_rsp);
407 
408 	NET_BUF_SIMPLE_DEFINE(p1_buf, 500);
409 	NET_BUF_SIMPLE_DEFINE(p1_item_buf, 500);
410 	struct bt_mesh_comp_p1_elem p1_elem = {._buf = &p1_buf};
411 	struct bt_mesh_comp_p1_model_item mod_item = {._buf = &p1_item_buf};
412 	struct bt_mesh_comp_p1_ext_item ext_item = {0};
413 
414 	verify_cdp1(&p1_elem, &mod_item, &ext_item, &p1_dev_comp);
415 
416 	PASS();
417 }
418 
419 #define TEST_CASE(role, name, description)                                                         \
420 	{                                                                                          \
421 		.test_id = "cdp1_" #role "_" #name, .test_descr = description,                     \
422 		.test_tick_f = bt_mesh_test_timeout, .test_main_f = test_##role##_##name,          \
423 	}
424 
425 static const struct bst_test_instance test_cdp1[] = {
426 	TEST_CASE(node, data_comparison, "Compare encoded and decoded CDP1 data."),
427 
428 	BSTEST_END_MARKER};
429 
test_cdp1_install(struct bst_test_list * tests)430 struct bst_test_list *test_cdp1_install(struct bst_test_list *tests)
431 {
432 	tests = bst_add_tests(tests, test_cdp1);
433 	return tests;
434 }
435