Lines Matching refs:lpn
41 #define RX_DELAY_CORRECTION(lpn) ((lpn)->adv_duration) argument
43 #define RX_DELAY_CORRECTION(lpn) 0 argument
64 #define REQ_RETRY_DURATION(lpn) (LPN_RECV_DELAY + (lpn)->adv_duration + \ argument
65 (lpn)->recv_win + POLL_RETRY_TIMEOUT)
72 #define REQ_ATTEMPTS(lpn) MIN(REQ_ATTEMPTS_MAX, \ argument
73 POLL_TIMEOUT / REQ_RETRY_DURATION(lpn))
75 #define POLL_TIMEOUT_MAX(lpn) (POLL_TIMEOUT - \ argument
76 (REQ_ATTEMPTS(lpn) * REQ_RETRY_DURATION(lpn)))
118 static int32_t poll_timeout(struct bt_mesh_lpn *lpn) in poll_timeout() argument
123 return MIN(POLL_TIMEOUT_MAX(lpn), 1 * MSEC_PER_SEC); in poll_timeout()
126 if (lpn->poll_timeout < POLL_TIMEOUT_MAX(lpn)) { in poll_timeout()
127 lpn->poll_timeout *= 2; in poll_timeout()
128 lpn->poll_timeout = in poll_timeout()
129 MIN(lpn->poll_timeout, POLL_TIMEOUT_MAX(lpn)); in poll_timeout()
132 LOG_DBG("Poll Timeout is %ums", lpn->poll_timeout); in poll_timeout()
134 return lpn->poll_timeout; in poll_timeout()
140 LOG_DBG("%s -> %s", state2str(bt_mesh.lpn.state), state2str(state)); in lpn_set_state()
142 bt_mesh.lpn.state = state; in lpn_set_state()
150 for (i = 0; i < ARRAY_SIZE(bt_mesh.lpn.added); i++) { in group_zero()
163 for (i = 0; i < ARRAY_SIZE(bt_mesh.lpn.added); i++) { in group_set()
176 for (i = 0; i < ARRAY_SIZE(bt_mesh.lpn.added); i++) { in group_clear()
188 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in friend_clear_sent() local
192 lpn->req_attempts++; in friend_clear_sent()
197 clear_friendship(false, lpn->disable); in friend_clear_sent()
202 k_work_reschedule(&lpn->timer, K_MSEC(FRIEND_REQ_TIMEOUT)); in friend_clear_sent()
212 .net_idx = bt_mesh.lpn.sub->net_idx, in send_friend_clear()
214 .addr = bt_mesh.lpn.frnd, in send_friend_clear()
218 .sub = bt_mesh.lpn.sub, in send_friend_clear()
225 .lpn_counter = sys_cpu_to_be16(bt_mesh.lpn.lpn_counter), in send_friend_clear()
236 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in clear_friendship() local
237 bool was_established = lpn->established; in clear_friendship()
238 uint16_t frnd = lpn->frnd; in clear_friendship()
239 uint16_t net_idx = lpn->sub->net_idx; in clear_friendship()
243 if (!force && lpn->established && !lpn->clear_success && in clear_friendship()
244 lpn->req_attempts < CLEAR_ATTEMPTS) { in clear_friendship()
246 lpn->disable = disable; in clear_friendship()
255 (void)k_work_cancel_delayable(&lpn->timer); in clear_friendship()
261 if (lpn->clear_success) { in clear_friendship()
262 lpn->old_friend = BT_MESH_ADDR_UNASSIGNED; in clear_friendship()
264 lpn->old_friend = lpn->frnd; in clear_friendship()
267 for (int i = 0; i < ARRAY_SIZE(lpn->cred); i++) { in clear_friendship()
268 if (lpn->sub->keys[i].valid) { in clear_friendship()
269 bt_mesh_friend_cred_destroy(&lpn->cred[i]); in clear_friendship()
273 lpn->frnd = BT_MESH_ADDR_UNASSIGNED; in clear_friendship()
274 lpn->fsn = 0U; in clear_friendship()
275 lpn->req_attempts = 0U; in clear_friendship()
276 lpn->recv_win = 0U; in clear_friendship()
277 lpn->queue_size = 0U; in clear_friendship()
278 lpn->disable = 0U; in clear_friendship()
279 lpn->sent_req = 0U; in clear_friendship()
280 lpn->established = 0U; in clear_friendship()
281 lpn->clear_success = 0U; in clear_friendship()
282 lpn->sub = NULL; in clear_friendship()
284 group_zero(lpn->added); in clear_friendship()
285 group_zero(lpn->pending); in clear_friendship()
286 group_zero(lpn->to_remove); in clear_friendship()
292 lpn->groups_changed = 1U; in clear_friendship()
299 k_work_reschedule(&lpn->timer, FRIEND_REQ_RETRY_TIMEOUT); in clear_friendship()
317 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in friend_req_send_end() local
319 if (lpn->state != BT_MESH_LPN_ENABLED) { in friend_req_send_end()
328 lpn->adv_duration = k_uptime_get_32() - lpn->adv_start_time; in friend_req_send_end()
331 k_work_reschedule(&lpn->timer, in friend_req_send_end()
332 K_MSEC(FRIEND_REQ_WAIT - (int32_t)lpn->adv_duration)); in friend_req_send_end()
335 k_work_reschedule(&lpn->timer, K_MSEC(FRIEND_REQ_TIMEOUT)); in friend_req_send_end()
342 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in friend_req_send_start() local
344 lpn->adv_start_time = k_uptime_get_32(); in friend_req_send_start()
356 static int send_friend_req(struct bt_mesh_lpn *lpn) in send_friend_req() argument
370 lpn->lpn_counter++; in send_friend_req()
376 .prev_addr = sys_cpu_to_be16(lpn->old_friend), in send_friend_req()
378 .lpn_counter = sys_cpu_to_be16(lpn->lpn_counter), in send_friend_req()
383 lpn->sub = bt_mesh_subnet_next(NULL); in send_friend_req()
384 if (!lpn->sub) { in send_friend_req()
389 ctx.net_idx = lpn->sub->net_idx; in send_friend_req()
390 tx.sub = lpn->sub; in send_friend_req()
398 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in req_send_end() local
401 if (lpn->state == BT_MESH_LPN_DISABLED) { in req_send_end()
405 lpn->adv_duration = k_uptime_get_32() - lpn->adv_start_time; in req_send_end()
408 LOG_DBG("req 0x%02x duration %u err %d state %s", lpn->sent_req, lpn->adv_duration, err, in req_send_end()
409 state2str(lpn->state)); in req_send_end()
414 lpn->sent_req = 0U; in req_send_end()
415 group_zero(lpn->pending); in req_send_end()
419 retry = (lpn->req_attempts > 0); in req_send_end()
421 lpn->req_attempts++; in req_send_end()
423 if (lpn->established || IS_ENABLED(CONFIG_BT_MESH_LPN_ESTABLISHMENT)) { in req_send_end()
428 k_work_reschedule(&lpn->timer, in req_send_end()
429 K_MSEC(LPN_RECV_DELAY - SCAN_LATENCY - RX_DELAY_CORRECTION(lpn))); in req_send_end()
432 k_work_reschedule(&lpn->timer, K_MSEC(LPN_RECV_DELAY + lpn->recv_win)); in req_send_end()
437 cb->polled(lpn->sub->net_idx, lpn->frnd, retry); in req_send_end()
444 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in req_send_start() local
446 lpn->adv_start_time = k_uptime_get_32(); in req_send_start()
461 .net_idx = bt_mesh.lpn.sub->net_idx, in send_friend_poll()
463 .addr = bt_mesh.lpn.frnd, in send_friend_poll()
467 .sub = bt_mesh.lpn.sub, in send_friend_poll()
473 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in send_friend_poll() local
474 uint8_t fsn = lpn->fsn; in send_friend_poll()
477 LOG_DBG("lpn->sent_req 0x%02x", lpn->sent_req); in send_friend_poll()
479 if (lpn->sent_req) { in send_friend_poll()
480 if (lpn->sent_req != TRANS_CTL_OP_FRIEND_POLL) { in send_friend_poll()
481 lpn->pending_poll = 1U; in send_friend_poll()
490 lpn->pending_poll = 0U; in send_friend_poll()
491 lpn->sent_req = TRANS_CTL_OP_FRIEND_POLL; in send_friend_poll()
499 if (bt_mesh.lpn.state == BT_MESH_LPN_DISABLED) { in bt_mesh_lpn_disable()
508 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in bt_mesh_lpn_set() local
511 if (lpn->state != BT_MESH_LPN_DISABLED) { in bt_mesh_lpn_set()
515 if (lpn->state == BT_MESH_LPN_DISABLED) { in bt_mesh_lpn_set()
537 send_friend_req(lpn); in bt_mesh_lpn_set()
540 lpn->state == BT_MESH_LPN_TIMER) { in bt_mesh_lpn_set()
544 (void)k_work_cancel_delayable(&lpn->timer); in bt_mesh_lpn_set()
556 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in bt_mesh_lpn_friendship_end() local
558 if (!lpn->established) { in bt_mesh_lpn_friendship_end()
565 static void friend_response_received(struct bt_mesh_lpn *lpn) in friend_response_received() argument
567 LOG_DBG("lpn->sent_req 0x%02x", lpn->sent_req); in friend_response_received()
569 if (lpn->sent_req == TRANS_CTL_OP_FRIEND_POLL) { in friend_response_received()
570 lpn->fsn++; in friend_response_received()
574 lpn->req_attempts = 0U; in friend_response_received()
575 lpn->sent_req = 0U; in friend_response_received()
580 int32_t timeout = poll_timeout(lpn); in friend_response_received()
582 k_work_reschedule(&lpn->timer, K_MSEC(timeout)); in friend_response_received()
588 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in bt_mesh_lpn_msg_received() local
590 if (lpn->state == BT_MESH_LPN_TIMER) { in bt_mesh_lpn_msg_received()
592 k_work_reschedule(&lpn->timer, K_MSEC(LPN_AUTO_TIMEOUT)); in bt_mesh_lpn_msg_received()
600 if (lpn->state != BT_MESH_LPN_WAIT_UPDATE) { in bt_mesh_lpn_msg_received()
604 if (lpn->sent_req != TRANS_CTL_OP_FRIEND_POLL) { in bt_mesh_lpn_msg_received()
609 friend_response_received(lpn); in bt_mesh_lpn_msg_received()
618 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in friend_cred_create() local
621 lpn->frnd, lpn->lpn_counter, in friend_cred_create()
622 lpn->frnd_counter, key); in friend_cred_create()
629 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in bt_mesh_lpn_friend_offer() local
638 if (lpn->state != BT_MESH_LPN_WAIT_OFFER) { in bt_mesh_lpn_friend_offer()
653 lpn->frnd_counter = frnd_counter; in bt_mesh_lpn_friend_offer()
654 lpn->frnd = rx->ctx.addr; in bt_mesh_lpn_friend_offer()
659 for (int i = 0; i < ARRAY_SIZE(lpn->cred); i++) { in bt_mesh_lpn_friend_offer()
660 if (!lpn->sub->keys[i].valid) { in bt_mesh_lpn_friend_offer()
664 err = friend_cred_create(&lpn->cred[i], &lpn->sub->keys[i].net); in bt_mesh_lpn_friend_offer()
666 lpn->frnd = BT_MESH_ADDR_UNASSIGNED; in bt_mesh_lpn_friend_offer()
673 lpn->recv_win = msg->recv_win; in bt_mesh_lpn_friend_offer()
674 lpn->queue_size = msg->queue_size; in bt_mesh_lpn_friend_offer()
679 for (int i = 0; i < ARRAY_SIZE(lpn->cred); i++) { in bt_mesh_lpn_friend_offer()
680 if (lpn->sub->keys[i].valid) { in bt_mesh_lpn_friend_offer()
681 bt_mesh_friend_cred_destroy(&lpn->cred[i]); in bt_mesh_lpn_friend_offer()
685 lpn->sub = NULL; in bt_mesh_lpn_friend_offer()
686 lpn->frnd = BT_MESH_ADDR_UNASSIGNED; in bt_mesh_lpn_friend_offer()
687 lpn->recv_win = 0U; in bt_mesh_lpn_friend_offer()
688 lpn->queue_size = 0U; in bt_mesh_lpn_friend_offer()
698 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in bt_mesh_lpn_friend_clear_cfm() local
706 if (lpn->state != BT_MESH_LPN_CLEAR) { in bt_mesh_lpn_friend_clear_cfm()
716 if (addr != bt_mesh_primary_addr() || counter != lpn->lpn_counter) { in bt_mesh_lpn_friend_clear_cfm()
721 lpn->clear_success = 1U; in bt_mesh_lpn_friend_clear_cfm()
722 clear_friendship(false, lpn->disable); in bt_mesh_lpn_friend_clear_cfm()
729 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in lpn_group_add() local
733 for (i = 0; i < ARRAY_SIZE(lpn->groups); i++) { in lpn_group_add()
734 if (lpn->groups[i] == group) { in lpn_group_add()
735 atomic_clear_bit(lpn->to_remove, i); in lpn_group_add()
739 if (!free_slot && lpn->groups[i] == BT_MESH_ADDR_UNASSIGNED) { in lpn_group_add()
740 free_slot = &lpn->groups[i]; in lpn_group_add()
750 lpn->groups_changed = 1U; in lpn_group_add()
755 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in lpn_group_del() local
758 for (i = 0; i < ARRAY_SIZE(lpn->groups); i++) { in lpn_group_del()
759 if (lpn->groups[i] == group) { in lpn_group_del()
760 if (atomic_test_bit(lpn->added, i) || in lpn_group_del()
761 atomic_test_bit(lpn->pending, i)) { in lpn_group_del()
762 atomic_set_bit(lpn->to_remove, i); in lpn_group_del()
763 lpn->groups_changed = 1U; in lpn_group_del()
765 lpn->groups[i] = BT_MESH_ADDR_UNASSIGNED; in lpn_group_del()
776 for (i = 0; i < ARRAY_SIZE(bt_mesh.lpn.added); i++) { in group_popcount()
786 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in sub_update() local
787 int added_count = group_popcount(lpn->added); in sub_update()
789 .net_idx = lpn->sub->net_idx, in sub_update()
791 .addr = lpn->frnd, in sub_update()
795 .sub = lpn->sub, in sub_update()
804 LOG_DBG("op 0x%02x sent_req 0x%02x", op, lpn->sent_req); in sub_update()
806 if (lpn->sent_req) { in sub_update()
810 for (i = 0, g = 0; i < ARRAY_SIZE(lpn->groups); i++) { in sub_update()
811 if (lpn->groups[i] == BT_MESH_ADDR_UNASSIGNED) { in sub_update()
816 if (atomic_test_bit(lpn->added, i)) { in sub_update()
820 if (!atomic_test_bit(lpn->to_remove, i)) { in sub_update()
825 if (added_count + g >= lpn->queue_size) { in sub_update()
830 req.addr_list[g++] = sys_cpu_to_be16(lpn->groups[i]); in sub_update()
831 atomic_set_bit(lpn->pending, i); in sub_update()
839 group_zero(lpn->pending); in sub_update()
843 req.xact = lpn->xact_next++; in sub_update()
847 group_zero(lpn->pending); in sub_update()
851 lpn->xact_pending = req.xact; in sub_update()
852 lpn->sent_req = op; in sub_update()
856 static void update_timeout(struct bt_mesh_lpn *lpn) in update_timeout() argument
858 if (lpn->established) { in update_timeout()
861 k_work_reschedule(&lpn->timer, K_MSEC(POLL_RETRY_TIMEOUT)); in update_timeout()
868 if (lpn->req_attempts < REQ_ATTEMPTS(lpn)) { in update_timeout()
870 lpn->sent_req = 0U; in update_timeout()
883 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in lpn_timeout() local
886 LOG_DBG("state: %s", state2str(lpn->state)); in lpn_timeout()
889 switch (lpn->state) { in lpn_timeout()
893 clear_friendship(false, bt_mesh.lpn.disable); in lpn_timeout()
903 send_friend_req(lpn); in lpn_timeout()
906 k_work_reschedule(&lpn->timer, K_MSEC(lpn->adv_duration + FRIEND_REQ_SCAN)); in lpn_timeout()
913 lpn->sent_req = 0U; in lpn_timeout()
914 k_work_reschedule(&lpn->timer, FRIEND_REQ_RETRY_TIMEOUT); in lpn_timeout()
921 if (lpn->req_attempts < REQ_ATTEMPTS(lpn)) { in lpn_timeout()
922 uint8_t req = lpn->sent_req; in lpn_timeout()
924 lpn->sent_req = 0U; in lpn_timeout()
935 LOG_ERR("No response from Friend after %u retries", lpn->req_attempts); in lpn_timeout()
936 lpn->req_attempts = 0U; in lpn_timeout()
940 k_work_reschedule(&lpn->timer, in lpn_timeout()
941 K_MSEC(SCAN_LATENCY + lpn->recv_win + RX_DELAY_CORRECTION(lpn))); in lpn_timeout()
946 update_timeout(lpn); in lpn_timeout()
960 if (!bt_mesh_lpn_established() || bt_mesh.lpn.sent_req) { in bt_mesh_lpn_group_add()
978 if (!bt_mesh_lpn_established() || bt_mesh.lpn.sent_req) { in bt_mesh_lpn_group_del()
989 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in bt_mesh_lpn_friend_sub_cfm() local
998 if (!lpn->sent_req) { in bt_mesh_lpn_friend_sub_cfm()
1003 if (msg->xact != lpn->xact_pending) { in bt_mesh_lpn_friend_sub_cfm()
1004 LOG_WRN("Transaction mismatch (0x%02x != 0x%02x)", msg->xact, lpn->xact_pending); in bt_mesh_lpn_friend_sub_cfm()
1008 if (lpn->sent_req == TRANS_CTL_OP_FRIEND_SUB_ADD) { in bt_mesh_lpn_friend_sub_cfm()
1009 group_set(lpn->added, lpn->pending); in bt_mesh_lpn_friend_sub_cfm()
1010 group_zero(lpn->pending); in bt_mesh_lpn_friend_sub_cfm()
1011 } else if (lpn->sent_req == TRANS_CTL_OP_FRIEND_SUB_REM) { in bt_mesh_lpn_friend_sub_cfm()
1014 group_clear(lpn->added, lpn->pending); in bt_mesh_lpn_friend_sub_cfm()
1016 for (i = 0; i < ARRAY_SIZE(lpn->groups); i++) { in bt_mesh_lpn_friend_sub_cfm()
1017 if (atomic_test_and_clear_bit(lpn->pending, i) && in bt_mesh_lpn_friend_sub_cfm()
1018 atomic_test_and_clear_bit(lpn->to_remove, i)) { in bt_mesh_lpn_friend_sub_cfm()
1019 lpn->groups[i] = BT_MESH_ADDR_UNASSIGNED; in bt_mesh_lpn_friend_sub_cfm()
1027 friend_response_received(lpn); in bt_mesh_lpn_friend_sub_cfm()
1029 if (lpn->groups_changed) { in bt_mesh_lpn_friend_sub_cfm()
1033 if (!lpn->sent_req) { in bt_mesh_lpn_friend_sub_cfm()
1034 lpn->groups_changed = 0U; in bt_mesh_lpn_friend_sub_cfm()
1038 if (lpn->pending_poll) { in bt_mesh_lpn_friend_sub_cfm()
1049 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in bt_mesh_lpn_friend_update() local
1059 if (lpn->sent_req != TRANS_CTL_OP_FRIEND_POLL) { in bt_mesh_lpn_friend_update()
1075 if (!lpn->established) { in bt_mesh_lpn_friend_update()
1086 lpn->established = 1U; in bt_mesh_lpn_friend_update()
1088 LOG_INF("Friendship established with 0x%04x", lpn->frnd); in bt_mesh_lpn_friend_update()
1093 lpn->poll_timeout = MIN(POLL_TIMEOUT_MAX(lpn), in bt_mesh_lpn_friend_update()
1099 friend_response_received(lpn); in bt_mesh_lpn_friend_update()
1108 if (lpn->groups_changed) { in bt_mesh_lpn_friend_update()
1112 if (!lpn->sent_req) { in bt_mesh_lpn_friend_update()
1113 lpn->groups_changed = 0U; in bt_mesh_lpn_friend_update()
1125 cb->established(lpn->sub->net_idx, lpn->frnd, lpn->queue_size, in bt_mesh_lpn_friend_update()
1126 lpn->recv_win); in bt_mesh_lpn_friend_update()
1136 if (!bt_mesh.lpn.established) { in bt_mesh_lpn_poll()
1149 if (sub == bt_mesh.lpn.sub) { in subnet_evt()
1156 friend_cred_create(&bt_mesh.lpn.cred[1], &sub->keys[1].net); in subnet_evt()
1163 BT_MESH_SUBNET_CB_DEFINE(lpn) = {
1169 struct bt_mesh_lpn *lpn = &bt_mesh.lpn; in bt_mesh_lpn_init() local
1173 k_work_init_delayable(&lpn->timer, lpn_timeout); in bt_mesh_lpn_init()
1175 if (lpn->state == BT_MESH_LPN_ENABLED) { in bt_mesh_lpn_init()
1182 send_friend_req(lpn); in bt_mesh_lpn_init()
1189 k_work_reschedule(&lpn->timer, in bt_mesh_lpn_init()