scan.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571
  1. /* Bluetooth Mesh */
  2. /*
  3. * SPDX-FileCopyrightText: 2017 Intel Corporation
  4. * SPDX-FileContributor: 2020-2021 Espressif Systems (Shanghai) CO LTD
  5. *
  6. * SPDX-License-Identifier: Apache-2.0
  7. */
  8. #include <stdint.h>
  9. #include <stdbool.h>
  10. #include <string.h>
  11. #include <errno.h>
  12. #include "btc_ble_mesh_ble.h"
  13. #include "mesh/config.h"
  14. #include "mesh/trace.h"
  15. #include "mesh/buf.h"
  16. #include "mesh/uuid.h"
  17. #include "scan.h"
  18. #include "beacon.h"
  19. #include "net.h"
  20. #include "prov_node.h"
  21. #include "proxy_client.h"
  22. #include "proxy_server.h"
  23. #include "prov_pvnr.h"
  24. #include "mesh/adapter.h"
  25. #include "mesh_v1.1/utils.h"
  26. /* Scan Window and Interval are equal for continuous scanning */
  27. #define SCAN_INTERVAL 0x20
  28. #define SCAN_WINDOW 0x20
  29. #define PROV_SVC_DATA_LEN 0x12
  30. #define PROXY_SVC_DATA_LEN_NET_ID 0x09
  31. #define PROXY_SVC_DATA_LEN_NODE_ID 0x11
  32. #define PROXY_SVC_DATA_LEN_PRIVATE_NET_ID 0x11
  33. #define PROXY_SVC_DATA_LEN_PRIVATE_NODE_ID 0x11
  34. #if (CONFIG_BLE_MESH_PROVISIONER || CONFIG_BLE_MESH_RPR_SRV)
  35. static const bt_mesh_addr_t *unprov_dev_addr;
  36. static uint8_t current_adv_type;
  37. static struct {
  38. uint8_t start_idx;
  39. uint8_t end_idx;
  40. uint8_t pair_num;
  41. struct {
  42. uint8_t uuid[16];
  43. uint8_t adv_type;
  44. uint8_t addr[6];
  45. } info[BLE_MESH_STORE_UNPROV_INFO_MAX_NUM];
  46. } unprov_dev_info_fifo;
  47. int bt_mesh_unprov_dev_fifo_dequeue(uint8_t *uuid, uint8_t *addr)
  48. {
  49. uint8_t idx = 0;
  50. if (unprov_dev_info_fifo.pair_num == 0) {
  51. return 0;
  52. }
  53. idx = unprov_dev_info_fifo.start_idx;
  54. if (uuid) {
  55. memcpy(uuid, unprov_dev_info_fifo.info[idx].uuid, 16);
  56. }
  57. if (addr) {
  58. memcpy(addr, unprov_dev_info_fifo.info[idx].addr, 6);
  59. }
  60. idx = (idx + 1) % BLE_MESH_STORE_UNPROV_INFO_MAX_NUM;
  61. unprov_dev_info_fifo.start_idx = idx;
  62. unprov_dev_info_fifo.pair_num--;
  63. return 0;
  64. }
  65. int bt_mesh_unprov_dev_info_query(uint8_t uuid[16], uint8_t addr[6],
  66. uint8_t *adv_type, uint8_t query_type)
  67. {
  68. uint8_t idx = 0;
  69. uint8_t cnt = 0;
  70. if (uuid == NULL || addr == NULL) {
  71. BT_WARN("No available information to query");
  72. return -1;
  73. }
  74. while (cnt < unprov_dev_info_fifo.pair_num) {
  75. idx = (cnt + unprov_dev_info_fifo.start_idx) % BLE_MESH_STORE_UNPROV_INFO_MAX_NUM;
  76. if (query_type & BLE_MESH_STORE_UNPROV_INFO_QUERY_TYPE_UUID) {
  77. if (!memcmp(unprov_dev_info_fifo.info[idx].addr, addr, 6)) {
  78. if (query_type & BLE_MESH_STORE_UNPROV_INFO_QUERY_TYPE_EXISTS) {
  79. return 0;
  80. } else {
  81. memcpy(uuid, unprov_dev_info_fifo.info[idx].uuid, 16);
  82. *adv_type = unprov_dev_info_fifo.info[idx].adv_type;
  83. break;
  84. }
  85. }
  86. } else {
  87. if (!memcmp(unprov_dev_info_fifo.info[idx].uuid, uuid, 16)) {
  88. if (query_type & BLE_MESH_STORE_UNPROV_INFO_QUERY_TYPE_EXISTS) {
  89. return 0;
  90. } else {
  91. memcpy(addr, unprov_dev_info_fifo.info[idx].addr, 6);
  92. *adv_type = unprov_dev_info_fifo.info[idx].adv_type;
  93. break;
  94. }
  95. }
  96. }
  97. cnt++;
  98. }
  99. if (cnt == unprov_dev_info_fifo.pair_num) {
  100. BT_WARN("Didn't find info for %d", query_type);
  101. return -1;
  102. }
  103. return 0;
  104. }
  105. int bt_mesh_unprov_dev_fifo_enqueue(uint8_t uuid[16], const uint8_t addr[6], uint8_t adv_type)
  106. {
  107. uint8_t idx = 0;
  108. if (uuid == NULL || addr == NULL) {
  109. BT_ERR("Invalid argument %s", __func__);
  110. return -EINVAL;
  111. }
  112. if (unprov_dev_info_fifo.pair_num == BLE_MESH_STORE_UNPROV_INFO_MAX_NUM) {
  113. bt_mesh_unprov_dev_fifo_dequeue(NULL, NULL);
  114. }
  115. idx = unprov_dev_info_fifo.end_idx;
  116. memcpy(unprov_dev_info_fifo.info[idx].uuid, uuid, 16);
  117. memcpy(unprov_dev_info_fifo.info[idx].addr, addr, 6);
  118. unprov_dev_info_fifo.info[idx].adv_type = adv_type;
  119. idx = (idx + 1) % BLE_MESH_STORE_UNPROV_INFO_MAX_NUM;
  120. unprov_dev_info_fifo.end_idx = idx;
  121. unprov_dev_info_fifo.pair_num++;
  122. return 0;
  123. }
  124. const bt_mesh_addr_t *bt_mesh_get_unprov_dev_addr(void)
  125. {
  126. return unprov_dev_addr;
  127. }
  128. uint8_t bt_mesh_get_adv_type(void)
  129. {
  130. return current_adv_type;
  131. }
  132. #endif /* (CONFIG_BLE_MESH_PROVISIONER || CONFIG_BLE_MESH_RPR_SRV) */
  133. #if (CONFIG_BLE_MESH_PROVISIONER && CONFIG_BLE_MESH_PB_GATT) || \
  134. CONFIG_BLE_MESH_GATT_PROXY_CLIENT || \
  135. CONFIG_BLE_MESH_PROXY_SOLIC_PDU_RX
  136. static bool adv_flags_valid(struct net_buf_simple *buf)
  137. {
  138. uint8_t flags = 0U;
  139. if (buf->len != 1U) {
  140. BT_DBG("Unexpected adv flags length %d", buf->len);
  141. return false;
  142. }
  143. flags = net_buf_simple_pull_u8(buf);
  144. BT_DBG("Received adv pkt with flags: 0x%02x", flags);
  145. /* Flags context will not be checked currently */
  146. ARG_UNUSED(flags);
  147. return true;
  148. }
  149. static bool adv_service_uuid_valid(struct net_buf_simple *buf, uint16_t *uuid)
  150. {
  151. if (buf->len != 2U) {
  152. BT_DBG("Length not match mesh service uuid");
  153. return false;
  154. }
  155. *uuid = net_buf_simple_pull_le16(buf);
  156. BT_DBG("Received adv pkt with service UUID: %d", *uuid);
  157. if (*uuid != BLE_MESH_UUID_MESH_PROV_VAL &&
  158. *uuid != BLE_MESH_UUID_MESH_PROXY_VAL &&
  159. *uuid != BLE_MESH_UUID_MESH_PROXY_SOLIC_VAL) {
  160. return false;
  161. }
  162. /**
  163. * @brief In remote provisioning.
  164. * A Node could handle the unprovisioned beacon.
  165. * CASE: MESH/SR/RPR/SCN/BV-01-C
  166. */
  167. #if CONFIG_BLE_MESH_RPR_SRV
  168. if (*uuid == BLE_MESH_UUID_MESH_PROV_VAL &&
  169. !IS_ENABLED(CONFIG_BLE_MESH_PB_GATT)) {
  170. return false;
  171. }
  172. #else
  173. if (*uuid == BLE_MESH_UUID_MESH_PROV_VAL &&
  174. (bt_mesh_is_provisioner_en() == false ||
  175. !IS_ENABLED(CONFIG_BLE_MESH_PB_GATT))) {
  176. return false;
  177. }
  178. #endif
  179. if (*uuid == BLE_MESH_UUID_MESH_PROXY_VAL &&
  180. !IS_ENABLED(CONFIG_BLE_MESH_GATT_PROXY_CLIENT)) {
  181. return false;
  182. }
  183. if (*uuid == BLE_MESH_UUID_MESH_PROXY_SOLIC_VAL &&
  184. !IS_ENABLED(CONFIG_BLE_MESH_PROXY_SOLIC_PDU_RX)) {
  185. return false;
  186. }
  187. return true;
  188. }
  189. static void handle_adv_service_data(struct net_buf_simple *buf,
  190. const bt_mesh_addr_t *addr,
  191. uint16_t uuid, int8_t rssi)
  192. {
  193. uint16_t type = 0U;
  194. if (!buf || !addr) {
  195. BT_ERR("%s, Invalid parameter", __func__);
  196. return;
  197. }
  198. type = net_buf_simple_pull_le16(buf);
  199. if (type != uuid) {
  200. BT_DBG("Invalid Mesh Service Data UUID 0x%04x", type);
  201. return;
  202. }
  203. switch (type) {
  204. #if CONFIG_BLE_MESH_PROVISIONER && CONFIG_BLE_MESH_PB_GATT
  205. case BLE_MESH_UUID_MESH_PROV_VAL:
  206. if (bt_mesh_is_provisioner_en()) {
  207. if (buf->len != PROV_SVC_DATA_LEN) {
  208. BT_WARN("Invalid Mesh Prov Service Data length %d", buf->len);
  209. return;
  210. }
  211. BT_DBG("Start to handle Mesh Prov Service Data");
  212. bt_mesh_provisioner_prov_adv_recv(buf, addr, rssi);
  213. }
  214. if (IS_ENABLED(CONFIG_BLE_MESH_RPR_SRV) &&
  215. bt_mesh_is_provisioned()) {
  216. const bt_mesh_addr_t *addr = bt_mesh_get_unprov_dev_addr();
  217. bt_mesh_unprov_dev_fifo_enqueue(buf->data, addr->val, bt_mesh_get_adv_type());
  218. bt_mesh_rpr_srv_unprov_beacon_recv(buf, bt_mesh_get_adv_type(), addr, rssi);
  219. }
  220. break;
  221. #endif
  222. #if CONFIG_BLE_MESH_GATT_PROXY_CLIENT
  223. case BLE_MESH_UUID_MESH_PROXY_VAL:
  224. if (buf->len != PROXY_SVC_DATA_LEN_NET_ID &&
  225. buf->len != PROXY_SVC_DATA_LEN_NODE_ID) {
  226. /* PROXY_SVC_DATA_LEN_NODE_ID,
  227. * PROXY_SVC_DATA_LEN_PRIVATE_NET_ID and
  228. * PROXY_SVC_DATA_LEN_PRIVATE_NODE_ID are equal to 0x11
  229. */
  230. BT_WARN("Invalid Mesh Proxy Service Data length %d", buf->len);
  231. return;
  232. }
  233. BT_DBG("Start to handle Mesh Proxy Service Data");
  234. bt_mesh_proxy_client_gatt_adv_recv(buf, addr, rssi);
  235. break;
  236. #endif
  237. #if CONFIG_BLE_MESH_PROXY_SOLIC_PDU_RX
  238. case BLE_MESH_UUID_MESH_PROXY_SOLIC_VAL:
  239. if (buf->len != (1 + BLE_MESH_NET_HDR_LEN + 8)) {
  240. BT_WARN("Invalid Mesh Proxy Solic Service Data length %d", buf->len);
  241. return;
  242. }
  243. BT_DBG("Start to handle Mesh Proxy Solic Service Data");
  244. bt_mesh_proxy_server_solic_recv(buf, addr, rssi);
  245. break;
  246. #endif
  247. default:
  248. break;
  249. }
  250. }
  251. #endif /* (CONFIG_BLE_MESH_PROVISIONER && CONFIG_BLE_MESH_PB_GATT) || \
  252. CONFIG_BLE_MESH_GATT_PROXY_CLIENT || \
  253. CONFIG_BLE_MESH_PROXY_SOLIC_PDU_RX */
  254. #if CONFIG_BLE_MESH_SUPPORT_BLE_SCAN
  255. static bool ble_scan_en;
  256. int bt_mesh_start_ble_scan(struct bt_mesh_ble_scan_param *param)
  257. {
  258. if (ble_scan_en == true) {
  259. BT_WARN("%s, Already", __func__);
  260. return -EALREADY;
  261. }
  262. ble_scan_en = true;
  263. return 0;
  264. }
  265. int bt_mesh_stop_ble_scan(void)
  266. {
  267. if (ble_scan_en == false) {
  268. BT_WARN("%s, Already", __func__);
  269. return -EALREADY;
  270. }
  271. ble_scan_en = false;
  272. return 0;
  273. }
  274. static void inline callback_ble_adv_pkt(const bt_mesh_addr_t *addr,
  275. uint8_t adv_type, uint8_t data[],
  276. uint16_t length, int8_t rssi)
  277. {
  278. if (ble_scan_en) {
  279. bt_mesh_ble_scan_cb_evt_to_btc(addr, adv_type, data, length, rssi);
  280. }
  281. }
  282. #endif /* CONFIG_BLE_MESH_SUPPORT_BLE_SCAN */
  283. #if CONFIG_BLE_MESH_RPR_SRV
  284. static bool rpr_ext_scan_handle_adv_pkt(const bt_mesh_addr_t *addr,
  285. uint8_t data[], uint16_t length)
  286. {
  287. struct net_buf_simple buf = {0};
  288. bool rpr_adv = false;
  289. if (bt_mesh_is_provisioned() == false) {
  290. return false;
  291. }
  292. net_buf_simple_init_with_data(&buf, data, length);
  293. bt_mesh_rpr_srv_extended_scan(&buf, addr, &rpr_adv);
  294. return rpr_adv;
  295. }
  296. #endif /* CONFIG_BLE_MESH_RPR_SRV */
  297. static void bt_mesh_scan_cb(const bt_mesh_addr_t *addr,
  298. int8_t rssi, uint8_t adv_type,
  299. struct net_buf_simple *buf,
  300. uint8_t scan_rsp_len)
  301. {
  302. #if (CONFIG_BLE_MESH_PROVISIONER && CONFIG_BLE_MESH_PB_GATT) || \
  303. CONFIG_BLE_MESH_GATT_PROXY_CLIENT || \
  304. CONFIG_BLE_MESH_PROXY_SOLIC_PDU_RX
  305. uint16_t uuid = 0U;
  306. #endif
  307. #if (CONFIG_BLE_MESH_RPR_SRV || CONFIG_BLE_MESH_SUPPORT_BLE_SCAN)
  308. uint8_t *adv_data = buf->data;
  309. uint16_t adv_len = buf->len;
  310. #endif
  311. if (adv_type != BLE_MESH_ADV_NONCONN_IND && adv_type != BLE_MESH_ADV_IND) {
  312. #if CONFIG_BLE_MESH_SUPPORT_BLE_SCAN
  313. callback_ble_adv_pkt(addr, adv_type, adv_data, adv_len, rssi);
  314. #endif
  315. return;
  316. }
  317. BT_DBG("scan, len %u: %s", buf->len, bt_hex(buf->data, buf->len));
  318. #if (CONFIG_BLE_MESH_PROVISIONER || CONFIG_BLE_MESH_RPR_SRV)
  319. unprov_dev_addr = addr;
  320. current_adv_type = adv_type;
  321. #endif
  322. while (buf->len > 1) {
  323. struct net_buf_simple_state state;
  324. uint8_t len, type;
  325. len = net_buf_simple_pull_u8(buf);
  326. /* Check for early termination */
  327. if (len == 0U) {
  328. #if CONFIG_BLE_MESH_SUPPORT_BLE_SCAN
  329. callback_ble_adv_pkt(addr, adv_type, adv_data, adv_len, rssi);
  330. #endif
  331. return;
  332. }
  333. if (len > buf->len) {
  334. BT_DBG("AD malformed");
  335. #if CONFIG_BLE_MESH_SUPPORT_BLE_SCAN
  336. callback_ble_adv_pkt(addr, adv_type, adv_data, adv_len, rssi);
  337. #endif
  338. return;
  339. }
  340. net_buf_simple_save(buf, &state);
  341. type = net_buf_simple_pull_u8(buf);
  342. buf->len = len - 1;
  343. if ((type == BLE_MESH_DATA_MESH_PROV || type == BLE_MESH_DATA_MESH_MESSAGE ||
  344. type == BLE_MESH_DATA_MESH_BEACON) && (adv_type != BLE_MESH_ADV_NONCONN_IND)) {
  345. BT_DBG("Ignore mesh packet (type 0x%02x) with adv_type 0x%02x", type, adv_type);
  346. return;
  347. }
  348. switch (type) {
  349. case BLE_MESH_DATA_MESH_MESSAGE:
  350. bt_mesh_net_recv(buf, rssi, BLE_MESH_NET_IF_ADV);
  351. break;
  352. #if CONFIG_BLE_MESH_PB_ADV
  353. case BLE_MESH_DATA_MESH_PROV:
  354. if (IS_ENABLED(CONFIG_BLE_MESH_NODE) && bt_mesh_is_node()) {
  355. bt_mesh_pb_adv_recv(buf);
  356. }
  357. if (IS_ENABLED(CONFIG_BLE_MESH_PROVISIONER) && bt_mesh_is_provisioner_en()) {
  358. bt_mesh_provisioner_pb_adv_recv(buf);
  359. }
  360. break;
  361. #endif /* CONFIG_BLE_MESH_PB_ADV */
  362. case BLE_MESH_DATA_MESH_BEACON:
  363. bt_mesh_beacon_recv(buf, rssi);
  364. break;
  365. #if (CONFIG_BLE_MESH_PROVISIONER && CONFIG_BLE_MESH_PB_GATT) || \
  366. CONFIG_BLE_MESH_GATT_PROXY_CLIENT || \
  367. CONFIG_BLE_MESH_PROXY_SOLIC_PDU_RX
  368. case BLE_MESH_DATA_FLAGS:
  369. if (!adv_flags_valid(buf)) {
  370. BT_DBG("Adv Flags mismatch, ignore this adv pkt");
  371. #if CONFIG_BLE_MESH_SUPPORT_BLE_SCAN
  372. callback_ble_adv_pkt(addr, adv_type, adv_data, adv_len, rssi);
  373. #endif
  374. return;
  375. }
  376. break;
  377. case BLE_MESH_DATA_UUID16_ALL:
  378. if (!adv_service_uuid_valid(buf, &uuid)) {
  379. BT_DBG("Adv Service UUID mismatch, ignore this adv pkt");
  380. #if CONFIG_BLE_MESH_RPR_SRV
  381. if (rpr_ext_scan_handle_adv_pkt(addr, adv_data, adv_len)) {
  382. /* If handled as extended scan report successfully, then not
  383. * notify to the application layer as normal BLE adv packet.
  384. */
  385. return;
  386. }
  387. #endif
  388. #if CONFIG_BLE_MESH_SUPPORT_BLE_SCAN
  389. callback_ble_adv_pkt(addr, adv_type, adv_data, adv_len, rssi);
  390. #endif
  391. return;
  392. }
  393. break;
  394. case BLE_MESH_DATA_SVC_DATA16:
  395. handle_adv_service_data(buf, addr, uuid, rssi);
  396. break;
  397. #endif
  398. default:
  399. #if CONFIG_BLE_MESH_RPR_SRV
  400. if (rpr_ext_scan_handle_adv_pkt(addr, adv_data, adv_len)) {
  401. /* If handled as extended scan report successfully, then not
  402. * notify to the application layer as normal BLE adv packet.
  403. */
  404. return;
  405. }
  406. #endif
  407. #if CONFIG_BLE_MESH_SUPPORT_BLE_SCAN
  408. callback_ble_adv_pkt(addr, adv_type, adv_data, adv_len, rssi);
  409. #endif
  410. return;
  411. }
  412. net_buf_simple_restore(buf, &state);
  413. net_buf_simple_pull(buf, len);
  414. }
  415. #if CONFIG_BLE_MESH_RPR_SRV && CONFIG_BLE_MESH_RPR_SRV_ACTIVE_SCAN
  416. if (scan_rsp_len != 0) {
  417. /**
  418. * scan response is only visible for remote provisioning extend scan.
  419. */
  420. rpr_ext_scan_handle_adv_pkt(addr, adv_data + adv_len, scan_rsp_len);
  421. }
  422. #endif /* CONFIG_BLE_MESH_RPR_SRV && CONFIG_BLE_MESH_RPR_SRV_ACTIVE_SCAN */
  423. }
  424. int bt_mesh_scan_enable(void)
  425. {
  426. int err = 0;
  427. struct bt_mesh_scan_param scan_param = {
  428. #if CONFIG_BLE_MESH_RPR_SRV_ACTIVE_SCAN
  429. .type = BLE_MESH_SCAN_ACTIVE,
  430. #else
  431. .type = BLE_MESH_SCAN_PASSIVE,
  432. #endif
  433. #if CONFIG_BLE_MESH_USE_DUPLICATE_SCAN
  434. .filter_dup = BLE_MESH_SCAN_FILTER_DUP_ENABLE,
  435. #else
  436. .filter_dup = BLE_MESH_SCAN_FILTER_DUP_DISABLE,
  437. #endif
  438. .interval = SCAN_INTERVAL,
  439. .window = SCAN_WINDOW,
  440. .scan_fil_policy = BLE_MESH_SP_ADV_ALL,
  441. };
  442. err = bt_le_scan_start(&scan_param, bt_mesh_scan_cb);
  443. if (err && err != -EALREADY) {
  444. BT_ERR("starting scan failed (err %d)", err);
  445. return err;
  446. }
  447. return 0;
  448. }
  449. int bt_mesh_scan_disable(void)
  450. {
  451. int err = 0;
  452. err = bt_le_scan_stop();
  453. if (err && err != -EALREADY) {
  454. BT_ERR("stopping scan failed (err %d)", err);
  455. return err;
  456. }
  457. return 0;
  458. }
  459. #if CONFIG_BLE_MESH_TEST_USE_WHITE_LIST
  460. int bt_mesh_scan_with_wl_enable(void)
  461. {
  462. int err = 0;
  463. struct bt_mesh_scan_param scan_param = {
  464. .type = BLE_MESH_SCAN_PASSIVE,
  465. #if CONFIG_BLE_MESH_USE_DUPLICATE_SCAN
  466. .filter_dup = BLE_MESH_SCAN_FILTER_DUP_ENABLE,
  467. #else
  468. .filter_dup = BLE_MESH_SCAN_FILTER_DUP_DISABLE,
  469. #endif
  470. .interval = SCAN_INTERVAL,
  471. .window = SCAN_WINDOW,
  472. .scan_fil_policy = BLE_MESH_SP_ADV_WL,
  473. };
  474. err = bt_le_scan_start(&scan_param, bt_mesh_scan_cb);
  475. if (err && err != -EALREADY) {
  476. BT_ERR("starting scan failed (err %d)", err);
  477. return err;
  478. }
  479. return 0;
  480. }
  481. #endif /* CONFIG_BLE_MESH_TEST_USE_WHITE_LIST */