1 /*
2  * Copyright (c) 2017 Intel Corporation
3  * Copyright (c) 2023 Nordic Semiconductor ASA
4  *
5  * SPDX-License-Identifier: Apache-2.0
6  */
7 
8 #include <zephyr/kernel.h>
9 #include <errno.h>
10 #include <string.h>
11 #include <stdlib.h>
12 
13 #include <zephyr/settings/settings.h>
14 #include <zephyr/net/buf.h>
15 
16 #include <zephyr/bluetooth/mesh.h>
17 
18 #include "common/bt_str.h"
19 
20 #include "va.h"
21 #include "foundation.h"
22 #include "msg.h"
23 #include "net.h"
24 #include "crypto.h"
25 #include "settings.h"
26 
27 #define LOG_LEVEL CONFIG_BT_MESH_TRANS_LOG_LEVEL
28 #include <zephyr/logging/log.h>
29 LOG_MODULE_REGISTER(bt_mesh_va);
30 
31 static struct bt_mesh_va virtual_addrs[CONFIG_BT_MESH_LABEL_COUNT];
32 
33 /* Virtual Address information for persistent storage. */
34 struct va_val {
35 	uint16_t ref;
36 	uint16_t addr;
37 	uint8_t uuid[16];
38 } __packed;
39 
va_store(struct bt_mesh_va * store)40 static void va_store(struct bt_mesh_va *store)
41 {
42 	store->changed = 1U;
43 
44 	if (IS_ENABLED(CONFIG_BT_SETTINGS)) {
45 		bt_mesh_settings_store_schedule(BT_MESH_SETTINGS_VA_PENDING);
46 	}
47 }
48 
bt_mesh_va_add(const uint8_t uuid[16],const struct bt_mesh_va ** entry)49 uint8_t bt_mesh_va_add(const uint8_t uuid[16], const struct bt_mesh_va **entry)
50 {
51 	struct bt_mesh_va *va = NULL;
52 	int err;
53 
54 	for (int i = 0; i < ARRAY_SIZE(virtual_addrs); i++) {
55 		if (!virtual_addrs[i].ref) {
56 			if (!va) {
57 				va = &virtual_addrs[i];
58 			}
59 
60 			continue;
61 		}
62 
63 		if (!memcmp(uuid, virtual_addrs[i].uuid,
64 			    ARRAY_SIZE(virtual_addrs[i].uuid))) {
65 			if (entry) {
66 				*entry = &virtual_addrs[i];
67 			}
68 			virtual_addrs[i].ref++;
69 			va_store(&virtual_addrs[i]);
70 			return STATUS_SUCCESS;
71 		}
72 	}
73 
74 	if (!va) {
75 		return STATUS_INSUFF_RESOURCES;
76 	}
77 
78 	memcpy(va->uuid, uuid, ARRAY_SIZE(va->uuid));
79 	err = bt_mesh_virtual_addr(uuid, &va->addr);
80 	if (err) {
81 		va->addr = BT_MESH_ADDR_UNASSIGNED;
82 		return STATUS_UNSPECIFIED;
83 	}
84 
85 	va->ref = 1;
86 	va_store(va);
87 
88 	if (entry) {
89 		*entry = va;
90 	}
91 
92 	return STATUS_SUCCESS;
93 }
94 
bt_mesh_va_del(const uint8_t * uuid)95 uint8_t bt_mesh_va_del(const uint8_t *uuid)
96 {
97 	struct bt_mesh_va *va;
98 
99 	if (CONFIG_BT_MESH_LABEL_COUNT == 0) {
100 		return STATUS_CANNOT_REMOVE;
101 	}
102 
103 	va = CONTAINER_OF(uuid, struct bt_mesh_va, uuid[0]);
104 
105 	if (!PART_OF_ARRAY(virtual_addrs, va) || va->ref == 0) {
106 		return STATUS_CANNOT_REMOVE;
107 	}
108 
109 	va->ref--;
110 	va_store(va);
111 
112 	return STATUS_SUCCESS;
113 }
114 
bt_mesh_va_uuid_get(uint16_t addr,const uint8_t * uuid,uint16_t * retaddr)115 const uint8_t *bt_mesh_va_uuid_get(uint16_t addr, const uint8_t *uuid, uint16_t *retaddr)
116 {
117 	int i = 0;
118 
119 	if (CONFIG_BT_MESH_LABEL_COUNT == 0) {
120 		return NULL;
121 	}
122 
123 	if (uuid != NULL) {
124 		struct bt_mesh_va *va;
125 
126 		va = CONTAINER_OF(uuid, struct bt_mesh_va, uuid[0]);
127 		i = ARRAY_INDEX(virtual_addrs, va);
128 	}
129 
130 	for (; i < ARRAY_SIZE(virtual_addrs); i++) {
131 		if (virtual_addrs[i].ref &&
132 		    (virtual_addrs[i].addr == addr || addr == BT_MESH_ADDR_UNASSIGNED)) {
133 			if (!uuid) {
134 				LOG_DBG("Found Label UUID for 0x%04x: %s", addr,
135 					bt_hex(virtual_addrs[i].uuid, 16));
136 
137 				if (retaddr) {
138 					*retaddr = virtual_addrs[i].addr;
139 				}
140 
141 				return virtual_addrs[i].uuid;
142 			} else if (uuid == virtual_addrs[i].uuid) {
143 				uuid = NULL;
144 			}
145 		}
146 	}
147 
148 	LOG_WRN("No matching Label UUID for 0x%04x", addr);
149 
150 	return NULL;
151 }
152 
bt_mesh_va_collision_check(uint16_t addr)153 bool bt_mesh_va_collision_check(uint16_t addr)
154 {
155 	size_t count = 0;
156 	const uint8_t *uuid = NULL;
157 
158 	do {
159 		uuid = bt_mesh_va_uuid_get(addr, uuid, NULL);
160 	} while (uuid && ++count);
161 
162 	return count > 1;
163 }
164 
bt_mesh_va_find(const uint8_t * uuid)165 const struct bt_mesh_va *bt_mesh_va_find(const uint8_t *uuid)
166 {
167 	int i;
168 
169 	for (i = 0; i < ARRAY_SIZE(virtual_addrs); i++) {
170 		if (virtual_addrs[i].ref && !memcmp(virtual_addrs[i].uuid, uuid, 16)) {
171 			return &virtual_addrs[i];
172 		}
173 	}
174 
175 	return NULL;
176 }
177 
va_get_by_idx(uint16_t index)178 static struct bt_mesh_va *va_get_by_idx(uint16_t index)
179 {
180 	if (index >= ARRAY_SIZE(virtual_addrs)) {
181 		return NULL;
182 	}
183 
184 	return &virtual_addrs[index];
185 }
186 
bt_mesh_va_get_uuid_by_idx(uint16_t idx)187 const uint8_t *bt_mesh_va_get_uuid_by_idx(uint16_t idx)
188 {
189 	struct bt_mesh_va *va;
190 
191 	va = va_get_by_idx(idx);
192 	return (va && va->ref > 0) ? va->uuid : NULL;
193 }
194 
bt_mesh_va_get_idx_by_uuid(const uint8_t * uuid,uint16_t * uuidx)195 int bt_mesh_va_get_idx_by_uuid(const uint8_t *uuid, uint16_t *uuidx)
196 {
197 	struct bt_mesh_va *va;
198 
199 	if (CONFIG_BT_MESH_LABEL_COUNT == 0) {
200 		return -ENOENT;
201 	}
202 
203 	va = CONTAINER_OF(uuid, struct bt_mesh_va, uuid[0]);
204 
205 	if (!PART_OF_ARRAY(virtual_addrs, va) || va->ref == 0) {
206 		return -ENOENT;
207 	}
208 
209 	*uuidx = ARRAY_INDEX(virtual_addrs, va);
210 	return 0;
211 }
212 
213 #if CONFIG_BT_MESH_LABEL_COUNT > 0
va_set(const char * name,size_t len_rd,settings_read_cb read_cb,void * cb_arg)214 static int va_set(const char *name, size_t len_rd,
215 		  settings_read_cb read_cb, void *cb_arg)
216 {
217 	struct va_val va;
218 	struct bt_mesh_va *lab;
219 	uint16_t index;
220 	int err;
221 
222 	if (!name) {
223 		LOG_ERR("Insufficient number of arguments");
224 		return -ENOENT;
225 	}
226 
227 	index = strtol(name, NULL, 16);
228 
229 	if (len_rd == 0) {
230 		LOG_WRN("Mesh Virtual Address length = 0");
231 		return 0;
232 	}
233 
234 	err = bt_mesh_settings_set(read_cb, cb_arg, &va, sizeof(va));
235 	if (err) {
236 		LOG_ERR("Failed to set \'virtual address\'");
237 		return err;
238 	}
239 
240 	if (va.ref == 0) {
241 		LOG_WRN("Ignore Mesh Virtual Address ref = 0");
242 		return 0;
243 	}
244 
245 	lab = va_get_by_idx(index);
246 	if (lab == NULL) {
247 		LOG_WRN("Out of labels buffers");
248 		return -ENOBUFS;
249 	}
250 
251 	memcpy(lab->uuid, va.uuid, 16);
252 	lab->addr = va.addr;
253 	lab->ref = va.ref;
254 
255 	LOG_DBG("Restored Virtual Address, addr 0x%04x ref 0x%04x", lab->addr, lab->ref);
256 
257 	return 0;
258 }
259 
260 BT_MESH_SETTINGS_DEFINE(va, "Va", va_set);
261 
262 #define IS_VA_DEL(_label)	((_label)->ref == 0)
bt_mesh_va_pending_store(void)263 void bt_mesh_va_pending_store(void)
264 {
265 	struct bt_mesh_va *lab;
266 	struct va_val va;
267 	char path[18];
268 	uint16_t i;
269 	int err;
270 
271 	for (i = 0; (lab = va_get_by_idx(i)) != NULL; i++) {
272 		if (!lab->changed) {
273 			continue;
274 		}
275 
276 		lab->changed = 0U;
277 
278 		snprintk(path, sizeof(path), "bt/mesh/Va/%x", i);
279 
280 		if (IS_VA_DEL(lab)) {
281 			err = settings_delete(path);
282 		} else {
283 			va.ref = lab->ref;
284 			va.addr = lab->addr;
285 			memcpy(va.uuid, lab->uuid, 16);
286 
287 			err = settings_save_one(path, &va, sizeof(va));
288 		}
289 
290 		if (err) {
291 			LOG_ERR("Failed to %s %s value (err %d)",
292 				IS_VA_DEL(lab) ? "delete" : "store", path, err);
293 		} else {
294 			LOG_DBG("%s %s value", IS_VA_DEL(lab) ? "Deleted" : "Stored", path);
295 		}
296 	}
297 }
298 #else
bt_mesh_va_pending_store(void)299 void bt_mesh_va_pending_store(void)
300 {
301 	/* Do nothing. */
302 }
303 #endif /* CONFIG_BT_MESH_LABEL_COUNT > 0 */
304 
bt_mesh_va_clear(void)305 void bt_mesh_va_clear(void)
306 {
307 	int i;
308 
309 	for (i = 0; i < ARRAY_SIZE(virtual_addrs); i++) {
310 		if (virtual_addrs[i].ref) {
311 			virtual_addrs[i].ref = 0U;
312 			virtual_addrs[i].changed = 1U;
313 		}
314 	}
315 
316 	if (IS_ENABLED(CONFIG_BT_SETTINGS)) {
317 		bt_mesh_settings_store_schedule(BT_MESH_SETTINGS_VA_PENDING);
318 	}
319 }
320