2019-06-28 03:13:18 +00:00
|
|
|
// Copyright 2017-2019 Espressif Systems (Shanghai) PTE LTD
|
2019-01-07 07:16:47 +00:00
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include "mesh.h"
|
2020-01-19 10:57:13 +00:00
|
|
|
#include "mesh_main.h"
|
2019-12-09 06:30:46 +00:00
|
|
|
#include "transport.h"
|
|
|
|
#include "foundation.h"
|
2019-09-02 03:41:39 +00:00
|
|
|
#include "client_common.h"
|
2020-01-19 10:57:13 +00:00
|
|
|
#include "mesh_common.h"
|
2019-01-07 07:16:47 +00:00
|
|
|
|
2020-03-27 03:18:02 +00:00
|
|
|
#define HCI_TIME_FOR_START_ADV K_MSEC(5) /* Three adv related hci commands may take 4 ~ 5ms */
|
2019-12-09 06:30:46 +00:00
|
|
|
|
2019-10-21 14:53:25 +00:00
|
|
|
static bt_mesh_client_node_t *bt_mesh_client_pick_node(sys_slist_t *list, u16_t tx_dst)
|
|
|
|
{
|
2019-12-14 09:31:34 +00:00
|
|
|
bt_mesh_client_node_t *node = NULL;
|
|
|
|
sys_snode_t *cur = NULL;
|
|
|
|
|
2019-12-20 03:26:55 +00:00
|
|
|
bt_mesh_list_lock();
|
2019-10-21 14:53:25 +00:00
|
|
|
if (sys_slist_is_empty(list)) {
|
2019-12-20 03:26:55 +00:00
|
|
|
bt_mesh_list_unlock();
|
2019-10-21 14:53:25 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (cur = sys_slist_peek_head(list);
|
|
|
|
cur != NULL; cur = sys_slist_peek_next(cur)) {
|
|
|
|
node = (bt_mesh_client_node_t *)cur;
|
|
|
|
if (node->ctx.addr == tx_dst) {
|
2019-12-20 03:26:55 +00:00
|
|
|
bt_mesh_list_unlock();
|
2019-10-21 14:53:25 +00:00
|
|
|
return node;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-20 03:26:55 +00:00
|
|
|
bt_mesh_list_unlock();
|
2019-10-21 14:53:25 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-07-04 10:35:08 +00:00
|
|
|
bt_mesh_client_node_t *bt_mesh_is_client_recv_publish_msg(struct bt_mesh_model *model,
|
|
|
|
struct bt_mesh_msg_ctx *ctx,
|
|
|
|
struct net_buf_simple *buf, bool need_pub)
|
2019-01-07 07:16:47 +00:00
|
|
|
{
|
2019-09-02 04:06:20 +00:00
|
|
|
bt_mesh_client_internal_data_t *data = NULL;
|
|
|
|
bt_mesh_client_user_data_t *cli = NULL;
|
2019-01-07 07:16:47 +00:00
|
|
|
bt_mesh_client_node_t *node = NULL;
|
|
|
|
|
|
|
|
if (!model || !ctx || !buf) {
|
|
|
|
BT_ERR("%s, Invalid parameter", __func__);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-09-02 04:06:20 +00:00
|
|
|
cli = (bt_mesh_client_user_data_t *)model->user_data;
|
2019-01-07 07:16:47 +00:00
|
|
|
if (!cli) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Invalid client user data");
|
2019-01-07 07:16:47 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** If the received message address is not a unicast address,
|
|
|
|
* the address may be a group/virtual address, and we push
|
|
|
|
* this message to the application layer.
|
|
|
|
*/
|
|
|
|
if (!BLE_MESH_ADDR_IS_UNICAST(ctx->recv_dst)) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_DBG("Unexpected status message 0x%08x", ctx->recv_op);
|
2019-01-07 07:16:47 +00:00
|
|
|
if (cli->publish_status && need_pub) {
|
2019-10-21 14:53:25 +00:00
|
|
|
cli->publish_status(ctx->recv_op, model, ctx, buf);
|
2019-01-07 07:16:47 +00:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** If the source address of the received status message is
|
|
|
|
* different with the destination address of the sending
|
|
|
|
* message, then the message is from another element and
|
|
|
|
* push it to application layer.
|
|
|
|
*/
|
2019-09-02 04:06:20 +00:00
|
|
|
data = (bt_mesh_client_internal_data_t *)cli->internal_data;
|
2019-01-07 07:16:47 +00:00
|
|
|
if (!data) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Invalid client internal data");
|
2019-01-07 07:16:47 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((node = bt_mesh_client_pick_node(&data->queue, ctx->addr)) == NULL) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_DBG("Unexpected status message 0x%08x", ctx->recv_op);
|
2019-01-07 07:16:47 +00:00
|
|
|
if (cli->publish_status && need_pub) {
|
2019-10-21 14:53:25 +00:00
|
|
|
cli->publish_status(ctx->recv_op, model, ctx, buf);
|
2019-01-07 07:16:47 +00:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-10-21 14:53:25 +00:00
|
|
|
if (node->op_pending != ctx->recv_op) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_DBG("Unexpected status message 0x%08x", ctx->recv_op);
|
2019-01-07 07:16:47 +00:00
|
|
|
if (cli->publish_status && need_pub) {
|
2019-10-21 14:53:25 +00:00
|
|
|
cli->publish_status(ctx->recv_op, model, ctx, buf);
|
2019-01-07 07:16:47 +00:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-12-14 07:43:51 +00:00
|
|
|
if (k_delayed_work_remaining_get(&node->timer) == 0) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_DBG("Unexpected status message 0x%08x", ctx->recv_op);
|
2019-12-14 07:43:51 +00:00
|
|
|
if (cli->publish_status && need_pub) {
|
|
|
|
cli->publish_status(ctx->recv_op, model, ctx, buf);
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-01-07 07:16:47 +00:00
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
2019-10-21 14:53:25 +00:00
|
|
|
static bool bt_mesh_client_check_node_in_list(sys_slist_t *list, u16_t tx_dst)
|
2019-01-07 07:16:47 +00:00
|
|
|
{
|
2019-12-14 09:31:34 +00:00
|
|
|
bt_mesh_client_node_t *node = NULL;
|
|
|
|
sys_snode_t *cur = NULL;
|
|
|
|
|
2019-12-20 03:26:55 +00:00
|
|
|
bt_mesh_list_lock();
|
2019-01-07 07:16:47 +00:00
|
|
|
if (sys_slist_is_empty(list)) {
|
2019-12-20 03:26:55 +00:00
|
|
|
bt_mesh_list_unlock();
|
2019-01-07 07:16:47 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (cur = sys_slist_peek_head(list);
|
|
|
|
cur != NULL; cur = sys_slist_peek_next(cur)) {
|
|
|
|
node = (bt_mesh_client_node_t *)cur;
|
|
|
|
if (node->ctx.addr == tx_dst) {
|
2019-12-20 03:26:55 +00:00
|
|
|
bt_mesh_list_unlock();
|
2019-01-07 07:16:47 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-20 03:26:55 +00:00
|
|
|
bt_mesh_list_unlock();
|
2019-01-07 07:16:47 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static u32_t bt_mesh_client_get_status_op(const bt_mesh_client_op_pair_t *op_pair,
|
|
|
|
int size, u32_t opcode)
|
|
|
|
{
|
|
|
|
if (!op_pair || size == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const bt_mesh_client_op_pair_t *op = op_pair;
|
|
|
|
for (int i = 0; i < size; i++) {
|
|
|
|
if (op->cli_op == opcode) {
|
|
|
|
return op->status_op;
|
|
|
|
}
|
|
|
|
op++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-12-09 06:30:46 +00:00
|
|
|
static s32_t bt_mesh_get_adv_duration(void)
|
|
|
|
{
|
|
|
|
u16_t duration, adv_int;
|
|
|
|
u8_t xmit;
|
|
|
|
|
|
|
|
xmit = bt_mesh_net_transmit_get(); /* Network transmit */
|
|
|
|
adv_int = BLE_MESH_TRANSMIT_INT(xmit);
|
|
|
|
duration = (BLE_MESH_TRANSMIT_COUNT(xmit) + 1) * (adv_int + 10);
|
|
|
|
|
|
|
|
return (s32_t)duration;
|
|
|
|
}
|
|
|
|
|
|
|
|
static s32_t bt_mesh_client_calc_timeout(struct bt_mesh_msg_ctx *ctx,
|
|
|
|
struct net_buf_simple *msg,
|
|
|
|
u32_t opcode, s32_t timeout)
|
|
|
|
{
|
2020-04-08 13:32:05 +00:00
|
|
|
s32_t seg_retrans_to = 0, duration = 0, time = 0;
|
|
|
|
u8_t seg_count = 0, seg_retrans_num = 0;
|
|
|
|
bool need_seg = false;
|
|
|
|
u8_t mic_size = 0;
|
2019-12-09 06:30:46 +00:00
|
|
|
|
2020-03-27 03:18:02 +00:00
|
|
|
if (msg->len > BLE_MESH_SDU_UNSEG_MAX || ctx->send_rel) {
|
2019-12-09 06:30:46 +00:00
|
|
|
need_seg = true; /* Needs segmentation */
|
|
|
|
}
|
|
|
|
|
2020-05-08 11:49:48 +00:00
|
|
|
mic_size = (need_seg && net_buf_simple_tailroom(msg) >= BLE_MESH_MIC_LONG) ?
|
|
|
|
BLE_MESH_MIC_LONG : BLE_MESH_MIC_SHORT;
|
2019-12-09 06:30:46 +00:00
|
|
|
|
|
|
|
if (need_seg) {
|
|
|
|
/* Based on the message length, calculate how many segments are needed.
|
|
|
|
* All the messages sent from here are access messages.
|
|
|
|
*/
|
|
|
|
seg_retrans_num = bt_mesh_get_seg_retrans_num();
|
|
|
|
seg_retrans_to = bt_mesh_get_seg_retrans_timeout(ctx->send_ttl);
|
|
|
|
seg_count = (msg->len + mic_size - 1) / 12U + 1U;
|
|
|
|
|
|
|
|
duration = bt_mesh_get_adv_duration();
|
|
|
|
|
|
|
|
/* Currenlty only consider the time consumption of the same segmented
|
|
|
|
* messages, but if there are other messages between any two retrans-
|
|
|
|
* missions of the same segmented messages, then the whole time will
|
|
|
|
* be longer.
|
2020-05-08 11:49:48 +00:00
|
|
|
*
|
|
|
|
* Since the transport behavior has been changed, i.e. start retransmit
|
|
|
|
* timer after the last segment is sent, so we can simplify the timeout
|
|
|
|
* calculation here. And the retransmit timer will be started event if
|
|
|
|
* the attempts reaches ZERO when the dst is a unicast address.
|
2019-12-09 06:30:46 +00:00
|
|
|
*/
|
2020-05-08 11:49:48 +00:00
|
|
|
s32_t seg_duration = seg_count * (duration + HCI_TIME_FOR_START_ADV);
|
|
|
|
time = (seg_duration + seg_retrans_to) * seg_retrans_num;
|
2019-12-09 06:30:46 +00:00
|
|
|
|
|
|
|
BT_INFO("Original timeout %dms, calculated timeout %dms", timeout, time);
|
|
|
|
|
|
|
|
if (time < timeout) {
|
|
|
|
/* If the calculated time is smaller than the input timeout value,
|
|
|
|
* then use the original timeout value.
|
|
|
|
*/
|
|
|
|
time = timeout;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* For unsegmented access messages, directly use the timeout
|
|
|
|
* value from the application layer.
|
|
|
|
*/
|
|
|
|
time = timeout;
|
|
|
|
}
|
|
|
|
|
|
|
|
BT_INFO("Client message 0x%08x with timeout %dms", opcode, time);
|
|
|
|
|
|
|
|
return time;
|
|
|
|
}
|
|
|
|
|
2020-02-04 07:26:44 +00:00
|
|
|
static void msg_send_start(u16_t duration, int err, void *cb_data)
|
|
|
|
{
|
|
|
|
bt_mesh_client_node_t *node = cb_data;
|
|
|
|
|
|
|
|
BT_DBG("%s, duration %ums", __func__, duration);
|
|
|
|
|
|
|
|
if (err) {
|
|
|
|
if (!k_delayed_work_free(&node->timer)) {
|
|
|
|
bt_mesh_client_free_node(node);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
k_delayed_work_submit(&node->timer, node->timeout);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct bt_mesh_send_cb send_cb = {
|
|
|
|
.start = msg_send_start,
|
|
|
|
.end = NULL,
|
|
|
|
};
|
|
|
|
|
2019-01-07 07:16:47 +00:00
|
|
|
int bt_mesh_client_send_msg(struct bt_mesh_model *model,
|
|
|
|
u32_t opcode,
|
|
|
|
struct bt_mesh_msg_ctx *ctx,
|
|
|
|
struct net_buf_simple *msg,
|
|
|
|
k_work_handler_t timer_handler,
|
|
|
|
s32_t timeout, bool need_ack,
|
|
|
|
const struct bt_mesh_send_cb *cb,
|
|
|
|
void *cb_data)
|
|
|
|
{
|
2019-09-02 04:06:20 +00:00
|
|
|
bt_mesh_client_internal_data_t *internal = NULL;
|
2020-02-04 04:20:41 +00:00
|
|
|
bt_mesh_client_user_data_t *client = NULL;
|
2019-01-07 07:16:47 +00:00
|
|
|
bt_mesh_client_node_t *node = NULL;
|
2020-01-19 10:57:13 +00:00
|
|
|
int err = 0;
|
2019-01-07 07:16:47 +00:00
|
|
|
|
|
|
|
if (!model || !ctx || !msg) {
|
|
|
|
BT_ERR("%s, Invalid parameter", __func__);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2020-02-04 04:20:41 +00:00
|
|
|
client = (bt_mesh_client_user_data_t *)model->user_data;
|
|
|
|
if (!client) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Invalid client user data");
|
2020-02-04 04:20:41 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal = (bt_mesh_client_internal_data_t *)client->internal_data;
|
|
|
|
if (!internal) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Invalid client internal data");
|
2020-02-04 04:20:41 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ctx->addr == BLE_MESH_ADDR_UNASSIGNED) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Invalid DST 0x%04x", ctx->addr);
|
2020-02-04 04:20:41 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2019-01-07 07:16:47 +00:00
|
|
|
|
|
|
|
if (!need_ack) {
|
|
|
|
/* If this is an unack message, send it directly. */
|
|
|
|
return bt_mesh_model_send(model, ctx, msg, cb, cb_data);
|
|
|
|
}
|
|
|
|
|
2020-02-04 04:20:41 +00:00
|
|
|
if (!BLE_MESH_ADDR_IS_UNICAST(ctx->addr)) {
|
|
|
|
/* If an acknowledged message is not sent to a unicast address,
|
|
|
|
* for example to a group/virtual address, then all the
|
|
|
|
* corresponding responses will be treated as publish messages.
|
|
|
|
* And no timeout will be used for the message.
|
|
|
|
*/
|
|
|
|
return bt_mesh_model_send(model, ctx, msg, cb, cb_data);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!timer_handler) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Invalid timeout handler");
|
2020-02-04 04:20:41 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2019-01-07 07:16:47 +00:00
|
|
|
if (bt_mesh_client_check_node_in_list(&internal->queue, ctx->addr)) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Busy sending message to DST 0x%04x", ctx->addr);
|
2020-02-04 04:20:41 +00:00
|
|
|
return -EBUSY;
|
|
|
|
}
|
2019-12-09 06:30:46 +00:00
|
|
|
|
2020-02-04 04:20:41 +00:00
|
|
|
/* Don't forget to free the node in the timeout (timer_handler) function. */
|
|
|
|
node = (bt_mesh_client_node_t *)bt_mesh_calloc(sizeof(bt_mesh_client_node_t));
|
|
|
|
if (!node) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("%s, Out of memory", __func__);
|
2020-02-04 04:20:41 +00:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
2019-12-09 06:30:46 +00:00
|
|
|
|
2020-02-04 04:20:41 +00:00
|
|
|
memcpy(&node->ctx, ctx, sizeof(struct bt_mesh_msg_ctx));
|
|
|
|
node->ctx.model = model;
|
|
|
|
node->opcode = opcode;
|
|
|
|
node->op_pending = bt_mesh_client_get_status_op(client->op_pair, client->op_pair_size, opcode);
|
|
|
|
if (node->op_pending == 0U) {
|
2020-05-27 04:26:25 +00:00
|
|
|
BT_ERR("Not found the status opcode in op_pair list");
|
2020-02-04 04:20:41 +00:00
|
|
|
bt_mesh_free(node);
|
|
|
|
return -EINVAL;
|
2019-01-07 07:16:47 +00:00
|
|
|
}
|
2020-02-04 07:26:44 +00:00
|
|
|
node->timeout = bt_mesh_client_calc_timeout(ctx, msg, opcode, timeout ? timeout : CONFIG_BLE_MESH_CLIENT_MSG_TIMEOUT);
|
2019-01-07 07:16:47 +00:00
|
|
|
|
2020-05-27 04:26:25 +00:00
|
|
|
if (k_delayed_work_init(&node->timer, timer_handler)) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Failed to create a timer");
|
2020-05-27 04:26:25 +00:00
|
|
|
bt_mesh_free(node);
|
|
|
|
return -EIO;
|
|
|
|
}
|
2020-02-04 04:20:41 +00:00
|
|
|
|
|
|
|
bt_mesh_list_lock();
|
|
|
|
sys_slist_append(&internal->queue, &node->client_node);
|
|
|
|
bt_mesh_list_unlock();
|
|
|
|
|
2020-02-04 07:26:44 +00:00
|
|
|
/* "bt_mesh_model_send" will post the mesh packet to the mesh adv queue.
|
|
|
|
* Due to the higher priority of adv_thread (than btc task), we need to
|
|
|
|
* send the packet after the list item "node" is initialized properly.
|
|
|
|
*/
|
|
|
|
err = bt_mesh_model_send(model, ctx, msg, &send_cb, node);
|
|
|
|
if (err) {
|
|
|
|
BT_ERR("Failed to send client message 0x%08x", node->opcode);
|
|
|
|
k_delayed_work_free(&node->timer);
|
|
|
|
bt_mesh_client_free_node(node);
|
|
|
|
}
|
2020-02-04 04:20:41 +00:00
|
|
|
|
2020-02-04 07:26:44 +00:00
|
|
|
return err;
|
2019-01-07 07:16:47 +00:00
|
|
|
}
|
|
|
|
|
2020-01-19 10:57:13 +00:00
|
|
|
static bt_mesh_mutex_t client_model_lock;
|
2019-09-16 07:05:14 +00:00
|
|
|
|
|
|
|
static void bt_mesh_client_model_mutex_new(void)
|
|
|
|
{
|
2020-01-19 10:57:13 +00:00
|
|
|
if (!client_model_lock.mutex) {
|
|
|
|
bt_mesh_mutex_create(&client_model_lock);
|
2019-09-16 07:05:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-19 10:57:13 +00:00
|
|
|
static void bt_mesh_client_model_mutex_free(void)
|
|
|
|
{
|
|
|
|
bt_mesh_mutex_free(&client_model_lock);
|
|
|
|
}
|
|
|
|
|
2019-09-16 07:05:14 +00:00
|
|
|
void bt_mesh_client_model_lock(void)
|
|
|
|
{
|
2020-01-19 10:57:13 +00:00
|
|
|
bt_mesh_mutex_lock(&client_model_lock);
|
2019-09-16 07:05:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void bt_mesh_client_model_unlock(void)
|
|
|
|
{
|
2020-01-19 10:57:13 +00:00
|
|
|
bt_mesh_mutex_unlock(&client_model_lock);
|
2019-09-16 07:05:14 +00:00
|
|
|
}
|
|
|
|
|
2019-01-07 07:16:47 +00:00
|
|
|
int bt_mesh_client_init(struct bt_mesh_model *model)
|
|
|
|
{
|
2019-09-02 04:06:20 +00:00
|
|
|
bt_mesh_client_internal_data_t *data = NULL;
|
|
|
|
bt_mesh_client_user_data_t *cli = NULL;
|
2019-01-07 07:16:47 +00:00
|
|
|
|
|
|
|
if (!model) {
|
|
|
|
BT_ERR("%s, Invalid parameter", __func__);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!model->op) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Invalid vendor client model op");
|
2019-01-07 07:16:47 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
cli = model->user_data;
|
|
|
|
if (!cli) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Invalid vendor client user data");
|
2019-01-07 07:16:47 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2019-12-20 06:08:33 +00:00
|
|
|
if (!cli->internal_data) {
|
2020-01-19 10:57:13 +00:00
|
|
|
data = bt_mesh_calloc(sizeof(bt_mesh_client_internal_data_t));
|
2019-12-20 06:08:33 +00:00
|
|
|
if (!data) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("%s, Out of memory", __func__);
|
2019-12-20 06:08:33 +00:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
2019-01-07 07:16:47 +00:00
|
|
|
|
2019-12-20 06:08:33 +00:00
|
|
|
/* Init the client data queue */
|
|
|
|
sys_slist_init(&data->queue);
|
2019-01-07 07:16:47 +00:00
|
|
|
|
2019-12-20 06:08:33 +00:00
|
|
|
cli->model = model;
|
|
|
|
cli->internal_data = data;
|
|
|
|
} else {
|
|
|
|
bt_mesh_client_clear_list(cli->internal_data);
|
|
|
|
}
|
2019-01-07 07:16:47 +00:00
|
|
|
|
2019-09-16 07:05:14 +00:00
|
|
|
bt_mesh_client_model_mutex_new();
|
|
|
|
|
2019-01-07 07:16:47 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-01-19 10:57:13 +00:00
|
|
|
int bt_mesh_client_deinit(struct bt_mesh_model *model)
|
|
|
|
{
|
|
|
|
bt_mesh_client_user_data_t *client = NULL;
|
|
|
|
|
|
|
|
if (!model) {
|
|
|
|
BT_ERR("%s, Invalid parameter", __func__);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
client = (bt_mesh_client_user_data_t *)model->user_data;
|
|
|
|
if (!client) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Invalid vendor client user data");
|
2020-01-19 10:57:13 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (client->internal_data) {
|
|
|
|
/* Remove items from the list */
|
|
|
|
bt_mesh_client_clear_list(client->internal_data);
|
|
|
|
|
|
|
|
/* Free the allocated internal data */
|
|
|
|
bt_mesh_free(client->internal_data);
|
|
|
|
client->internal_data = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
bt_mesh_client_model_mutex_free();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-09-26 11:38:25 +00:00
|
|
|
int bt_mesh_client_free_node(bt_mesh_client_node_t *node)
|
2019-01-07 07:16:47 +00:00
|
|
|
{
|
2019-09-26 11:38:25 +00:00
|
|
|
bt_mesh_client_internal_data_t *internal = NULL;
|
|
|
|
bt_mesh_client_user_data_t *client = NULL;
|
|
|
|
|
|
|
|
if (!node || !node->ctx.model) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Invalid client list item");
|
2019-09-26 11:38:25 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
client = (bt_mesh_client_user_data_t *)node->ctx.model->user_data;
|
|
|
|
if (!client) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Invalid client user data");
|
2019-09-26 11:38:25 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal = (bt_mesh_client_internal_data_t *)client->internal_data;
|
|
|
|
if (!internal) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Invalid client internal data");
|
2019-01-07 07:16:47 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Release the client node from the queue
|
2019-12-20 03:26:55 +00:00
|
|
|
bt_mesh_list_lock();
|
2019-09-26 11:38:25 +00:00
|
|
|
sys_slist_find_and_remove(&internal->queue, &node->client_node);
|
2019-12-20 03:26:55 +00:00
|
|
|
bt_mesh_list_unlock();
|
2019-01-07 07:16:47 +00:00
|
|
|
// Free the node
|
2020-01-19 10:57:13 +00:00
|
|
|
bt_mesh_free(node);
|
2019-01-07 07:16:47 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-12-20 06:08:33 +00:00
|
|
|
int bt_mesh_client_clear_list(void *data)
|
|
|
|
{
|
|
|
|
bt_mesh_client_internal_data_t *internal = NULL;
|
|
|
|
bt_mesh_client_node_t *node = NULL;
|
|
|
|
|
|
|
|
if (!data) {
|
|
|
|
BT_ERR("%s, Invalid parameter", __func__);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal = (bt_mesh_client_internal_data_t *)data;
|
|
|
|
|
|
|
|
bt_mesh_list_lock();
|
|
|
|
while (!sys_slist_is_empty(&internal->queue)) {
|
|
|
|
node = (void *)sys_slist_get_not_empty(&internal->queue);
|
2020-05-27 04:26:25 +00:00
|
|
|
k_delayed_work_free(&node->timer);
|
2020-01-19 10:57:13 +00:00
|
|
|
bt_mesh_free(node);
|
2019-12-20 06:08:33 +00:00
|
|
|
}
|
|
|
|
bt_mesh_list_unlock();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-09-02 04:06:20 +00:00
|
|
|
int bt_mesh_set_client_model_role(bt_mesh_role_param_t *common)
|
2019-01-07 07:16:47 +00:00
|
|
|
{
|
2019-09-02 04:06:20 +00:00
|
|
|
bt_mesh_client_user_data_t *client = NULL;
|
2019-01-07 07:16:47 +00:00
|
|
|
|
|
|
|
if (!common || !common->model || !common->model->user_data) {
|
|
|
|
BT_ERR("%s, Invalid parameter", __func__);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2019-09-02 04:06:20 +00:00
|
|
|
client = (bt_mesh_client_user_data_t *)common->model->user_data;
|
2019-01-07 07:16:47 +00:00
|
|
|
|
|
|
|
switch (common->role) {
|
|
|
|
#if CONFIG_BLE_MESH_NODE
|
|
|
|
case NODE:
|
|
|
|
/* no matter if provisioner is enabled/disabled , node role can be used to send messages */
|
|
|
|
client->msg_role = NODE;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#if CONFIG_BLE_MESH_PROVISIONER
|
|
|
|
case PROVISIONER:
|
|
|
|
/* if provisioner is not enabled, provisioner role can't be used to send messages */
|
|
|
|
if (!bt_mesh_is_provisioner_en()) {
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_ERR("Provisioner is disabled");
|
2019-01-07 07:16:47 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
client->msg_role = PROVISIONER;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#if CONFIG_BLE_MESH_FAST_PROV
|
|
|
|
case FAST_PROV:
|
|
|
|
client->msg_role = FAST_PROV;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
2020-07-04 10:35:08 +00:00
|
|
|
BT_WARN("Unknown model role 0x%02x", common->role);
|
2019-01-07 07:16:47 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|