1376 lines
56 KiB
C

/*
* SPDX-FileCopyrightText: 2017-2021 Espressif Systems (Shanghai) CO LTD
*
* SPDX-License-Identifier: Apache-2.0
*/
#include <string.h>
#include <errno.h>
#include "btc_ble_mesh_lighting_model.h"
#include "mesh/config.h"
#include "mesh/model_opcode.h"
#if CONFIG_BLE_MESH_LIGHTING_CLIENT
#include "mesh/lighting_client.h"
/* The followings are the macro definitions of Lighting client
* model message length, and a message is composed of 3 parts:
* Opcode + Payload + MIC
*/
/* Light lightness client messages length */
#define BLE_MESH_LIGHT_LIGHTNESS_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_LIGHTNESS_SET_MSG_LEN (2 + 5 + 4)
#define BLE_MESH_LIGHT_LIGHTNESS_LINEAR_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_LIGHTNESS_LINEAR_SET_MSG_LEN (2 + 5 + 4)
#define BLE_MESH_LIGHT_LIGHTNESS_LAST_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_LIGHTNESS_DEFAULT_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_LIGHTNESS_DEFAULT_SET_MSG_LEN (2 + 2 + 4)
#define BLE_MESH_LIGHT_LIGHTNESS_RANGE_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_LIGHTNESS_RANGE_SET_MSG_LEN (2 + 4 + 4)
/* Light CTL client messages length */
#define BLE_MESH_LIGHT_CTL_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_CTL_SET_MSG_LEN (2 + 9 + 4)
#define BLE_MESH_LIGHT_CTL_TEMPERATURE_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_CTL_TEMPERATURE_SET_MSG_LEN (2 + 7 + 4)
#define BLE_MESH_LIGHT_CTL_TEMPERATURE_RANGE_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_CTL_TEMPERATURE_RANGE_SET_MSG_LEN (2 + 4 + 4)
#define BLE_MESH_LIGHT_CTL_DEFAULT_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_CTL_DEFAULT_SET_MSG_LEN (2 + 6 + 4)
/* Light HSL client messages length */
#define BLE_MESH_LIGHT_HSL_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_HSL_SET_MSG_LEN (2 + 9 + 4)
#define BLE_MESH_LIGHT_HSL_TARGET_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_HSL_HUE_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_HSL_HUE_SET_MSG_LEN (2 + 5 + 4)
#define BLE_MESH_LIGHT_HSL_SATURATION_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_HSL_SATURATION_SET_MSG_LEN (2 + 5 + 4)
#define BLE_MESH_LIGHT_HSL_DEFAULT_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_HSL_DEFAULT_SET_MSG_LEN (2 + 6 + 4)
#define BLE_MESH_LIGHT_HSL_RANGE_GET_MSG_LEN (2 + 0 + 4)
#define BLE_MESH_LIGHT_HSL_RANGE_SET_MSG_LEN (2 + 8 + 4)
/* Light xyL client messages length */
#define BLE_MESH_LIGHT_XYL_SET_MSG_LEN (2 + 9 + 4)
#define BLE_MESH_LIGHT_XYL_DEFAULT_SET_MSG_LEN (2 + 6 + 4)
#define BLE_MESH_LIGHT_XYL_RANGE_SET_MSG_LEN (2 + 8 + 4)
/* Light LC client messages length */
#define BLE_MESH_LIGHT_LC_MODE_SET_MSG_LEN (2 + 1 + 4)
#define BLE_MESH_LIGHT_LC_OM_SET_MSG_LEN (2 + 1 + 4)
#define BLE_MESH_LIGHT_LC_LIGHT_ONOFF_SET_MSG_LEN (2 + 4 + 4)
#define BLE_MESH_LIGHT_LC_PROPERTY_GET_MSG_LEN (2 + 2 + 4)
#define BLE_MESH_LIGHT_LC_PROPERTY_SET_MSG_LEN /* variable */
#define BLE_MESH_LIGHT_GET_STATE_MSG_LEN (2 + 2 + 4)
static const bt_mesh_client_op_pair_t light_op_pair[] = {
{ BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_GET, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_GET, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_GET, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_GET, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_GET, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_CTL_GET, BLE_MESH_MODEL_OP_LIGHT_CTL_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_CTL_SET, BLE_MESH_MODEL_OP_LIGHT_CTL_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_GET, BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET, BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_GET, BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET, BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_GET, BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET, BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_GET, BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_SET, BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_TARGET_GET, BLE_MESH_MODEL_OP_LIGHT_HSL_TARGET_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_GET, BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET, BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_GET, BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET, BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_GET, BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET, BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_GET, BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET, BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_XYL_GET, BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_XYL_SET, BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_XYL_TARGET_GET, BLE_MESH_MODEL_OP_LIGHT_XYL_TARGET_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_GET, BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET, BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_GET, BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET, BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LC_MODE_GET, BLE_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LC_MODE_SET, BLE_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LC_OM_GET, BLE_MESH_MODEL_OP_LIGHT_LC_OM_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LC_OM_SET, BLE_MESH_MODEL_OP_LIGHT_LC_OM_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_GET, BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET, BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET, BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS },
{ BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET, BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS },
};
static bt_mesh_mutex_t lighting_client_lock;
static void timeout_handler(struct k_work *work)
{
struct k_delayed_work *timer = NULL;
bt_mesh_client_node_t *node = NULL;
struct bt_mesh_model *model = NULL;
struct bt_mesh_msg_ctx ctx = {0};
uint32_t opcode = 0U;
BT_WARN("Receive light status message timeout");
bt_mesh_mutex_lock(&lighting_client_lock);
timer = CONTAINER_OF(work, struct k_delayed_work, work);
if (timer && !k_delayed_work_free(timer)) {
node = CONTAINER_OF(work, bt_mesh_client_node_t, timer.work);
if (node) {
memcpy(&ctx, &node->ctx, sizeof(ctx));
opcode = node->opcode;
model = node->model;
bt_mesh_client_free_node(node);
bt_mesh_lighting_client_cb_evt_to_btc(
opcode, BTC_BLE_MESH_EVT_LIGHTING_CLIENT_TIMEOUT, model, &ctx, NULL, 0);
}
}
bt_mesh_mutex_unlock(&lighting_client_lock);
}
static void light_status(struct bt_mesh_model *model,
struct bt_mesh_msg_ctx *ctx,
struct net_buf_simple *buf)
{
bt_mesh_client_node_t *node = NULL;
uint8_t *val = NULL;
uint8_t evt = 0xFF;
size_t len = 0U;
BT_DBG("len %d, bytes %s", buf->len, bt_hex(buf->data, buf->len));
switch (ctx->recv_op) {
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS: {
struct bt_mesh_light_lightness_status *status = NULL;
if (buf->len != 2 && buf->len != 5) {
BT_ERR("Invalid Light Lightness Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_lightness_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->present_lightness = net_buf_simple_pull_le16(buf);
if (buf->len) {
status->op_en = true;
status->target_lightness = net_buf_simple_pull_le16(buf);
status->remain_time = net_buf_simple_pull_u8(buf);
}
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_lightness_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS: {
struct bt_mesh_light_lightness_linear_status *status = NULL;
if (buf->len != 2 && buf->len != 5) {
BT_ERR("Invalid Light Lightness Linear Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_lightness_linear_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->present_lightness = net_buf_simple_pull_le16(buf);
if (buf->len) {
status->op_en = true;
status->target_lightness = net_buf_simple_pull_le16(buf);
status->remain_time = net_buf_simple_pull_u8(buf);
}
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_lightness_linear_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_STATUS: {
struct bt_mesh_light_lightness_last_status *status = NULL;
if (buf->len != 2) {
BT_ERR("Invalid Light Lightness Last Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_lightness_last_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->lightness = net_buf_simple_pull_le16(buf);
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_lightness_last_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS: {
struct bt_mesh_light_lightness_default_status *status = NULL;
if (buf->len != 2) {
BT_ERR("Invalid Light Lightness Default Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_lightness_default_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->lightness = net_buf_simple_pull_le16(buf);
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_lightness_default_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS: {
struct bt_mesh_light_lightness_range_status *status = NULL;
if (buf->len != 5) {
BT_ERR("Invalid Light Lightness Range Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_lightness_range_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->status_code = net_buf_simple_pull_u8(buf);
status->range_min = net_buf_simple_pull_le16(buf);
status->range_max = net_buf_simple_pull_le16(buf);
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_lightness_range_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_CTL_STATUS: {
struct bt_mesh_light_ctl_status *status = NULL;
if (buf->len != 4 && buf->len != 9) {
BT_ERR("Invalid Light CTL Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_ctl_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->present_ctl_lightness = net_buf_simple_pull_le16(buf);
status->present_ctl_temperature = net_buf_simple_pull_le16(buf);
if (buf->len) {
status->op_en = true;
status->target_ctl_lightness = net_buf_simple_pull_le16(buf);
status->target_ctl_temperature = net_buf_simple_pull_le16(buf);
status->remain_time = net_buf_simple_pull_u8(buf);
}
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_ctl_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS: {
struct bt_mesh_light_ctl_temperature_status *status = NULL;
if (buf->len != 4 && buf->len != 9) {
BT_ERR("Invalid Light CTL Temperature Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_ctl_temperature_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->present_ctl_temperature = net_buf_simple_pull_le16(buf);
status->present_ctl_delta_uv = net_buf_simple_pull_le16(buf);
if (buf->len) {
status->op_en = true;
status->target_ctl_temperature = net_buf_simple_pull_le16(buf);
status->target_ctl_delta_uv = net_buf_simple_pull_le16(buf);
status->remain_time = net_buf_simple_pull_u8(buf);
}
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_ctl_temperature_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS: {
struct bt_mesh_light_ctl_temperature_range_status *status = NULL;
if (buf->len != 5) {
BT_ERR("Invalid Light CTL Temperature Range Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_ctl_temperature_range_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->status_code = net_buf_simple_pull_u8(buf);
status->range_min = net_buf_simple_pull_le16(buf);
status->range_max = net_buf_simple_pull_le16(buf);
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_ctl_temperature_range_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS: {
struct bt_mesh_light_ctl_default_status *status = NULL;
if (buf->len != 6) {
BT_ERR("Invalid Light CTL Default Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_ctl_default_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->lightness = net_buf_simple_pull_le16(buf);
status->temperature = net_buf_simple_pull_le16(buf);
status->delta_uv = net_buf_simple_pull_le16(buf);
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_ctl_default_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS: {
struct bt_mesh_light_hsl_status *status = NULL;
if (buf->len != 6 && buf->len != 7) {
BT_ERR("Invalid Light HSL Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_hsl_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->hsl_lightness = net_buf_simple_pull_le16(buf);
status->hsl_hue = net_buf_simple_pull_le16(buf);
status->hsl_saturation = net_buf_simple_pull_le16(buf);
if (buf->len) {
status->op_en = true;
status->remain_time = net_buf_simple_pull_u8(buf);
}
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_hsl_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_HSL_TARGET_STATUS: {
struct bt_mesh_light_hsl_target_status *status = NULL;
if (buf->len != 6 && buf->len != 7) {
BT_ERR("Invalid Light HSL Target Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_hsl_target_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->hsl_lightness_target = net_buf_simple_pull_le16(buf);
status->hsl_hue_target = net_buf_simple_pull_le16(buf);
status->hsl_saturation_target = net_buf_simple_pull_le16(buf);
if (buf->len) {
status->op_en = true;
status->remain_time = net_buf_simple_pull_u8(buf);
}
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_hsl_target_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS: {
struct bt_mesh_light_hsl_hue_status *status = NULL;
if (buf->len != 2 && buf->len != 5) {
BT_ERR("Invalid Light HSL Hue Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_hsl_hue_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->present_hue = net_buf_simple_pull_le16(buf);
if (buf->len) {
status->op_en = true;
status->target_hue = net_buf_simple_pull_le16(buf);
status->remain_time = net_buf_simple_pull_u8(buf);
}
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_hsl_hue_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS: {
struct bt_mesh_light_hsl_saturation_status *status = NULL;
if (buf->len != 2 && buf->len != 5) {
BT_ERR("Invalid Light HSL Saturation Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_hsl_saturation_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->present_saturation = net_buf_simple_pull_le16(buf);
if (buf->len) {
status->op_en = true;
status->target_saturation = net_buf_simple_pull_le16(buf);
status->remain_time = net_buf_simple_pull_u8(buf);
}
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_hsl_saturation_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS: {
struct bt_mesh_light_hsl_default_status *status = NULL;
if (buf->len != 6) {
BT_ERR("Invalid Light HSL Default Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_hsl_default_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->lightness = net_buf_simple_pull_le16(buf);
status->hue = net_buf_simple_pull_le16(buf);
status->saturation = net_buf_simple_pull_le16(buf);
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_hsl_default_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS: {
struct bt_mesh_light_hsl_range_status *status = NULL;
if (buf->len != 9) {
BT_ERR("Invalid Light HSL Range Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_hsl_range_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->status_code = net_buf_simple_pull_u8(buf);
status->hue_range_min = net_buf_simple_pull_le16(buf);
status->hue_range_max = net_buf_simple_pull_le16(buf);
status->saturation_range_min = net_buf_simple_pull_le16(buf);
status->saturation_range_max = net_buf_simple_pull_le16(buf);
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_hsl_range_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS: {
struct bt_mesh_light_xyl_status *status = NULL;
if (buf->len != 6 && buf->len != 7) {
BT_ERR("Invalid Light xyL Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_xyl_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->xyl_lightness = net_buf_simple_pull_le16(buf);
status->xyl_x = net_buf_simple_pull_le16(buf);
status->xyl_y = net_buf_simple_pull_le16(buf);
if (buf->len) {
status->op_en = true;
status->remain_time = net_buf_simple_pull_u8(buf);
}
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_xyl_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_XYL_TARGET_STATUS: {
struct bt_mesh_light_xyl_target_status *status = NULL;
if (buf->len != 6 && buf->len != 7) {
BT_ERR("Invalid Light xyL Target Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_xyl_target_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->target_xyl_lightness = net_buf_simple_pull_le16(buf);
status->target_xyl_x = net_buf_simple_pull_le16(buf);
status->target_xyl_y = net_buf_simple_pull_le16(buf);
if (buf->len) {
status->op_en = true;
status->remain_time = net_buf_simple_pull_u8(buf);
}
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_xyl_target_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS: {
struct bt_mesh_light_xyl_default_status *status = NULL;
if (buf->len != 6) {
BT_ERR("Invalid Light xyL Default Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_xyl_default_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->lightness = net_buf_simple_pull_le16(buf);
status->xyl_x = net_buf_simple_pull_le16(buf);
status->xyl_y = net_buf_simple_pull_le16(buf);
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_xyl_default_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS: {
struct bt_mesh_light_xyl_range_status *status = NULL;
if (buf->len != 9) {
BT_ERR("Invalid Light xyL Range Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_xyl_range_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->status_code = net_buf_simple_pull_u8(buf);
status->xyl_x_range_min = net_buf_simple_pull_le16(buf);
status->xyl_x_range_max = net_buf_simple_pull_le16(buf);
status->xyl_y_range_min = net_buf_simple_pull_le16(buf);
status->xyl_y_range_max = net_buf_simple_pull_le16(buf);
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_xyl_range_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS: {
struct bt_mesh_light_lc_mode_status *status = NULL;
if (buf->len != 1) {
BT_ERR("Invalid Light LC Mode Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_lc_mode_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->mode = net_buf_simple_pull_u8(buf);
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_lc_mode_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LC_OM_STATUS: {
struct bt_mesh_light_lc_om_status *status = NULL;
if (buf->len != 1) {
BT_ERR("Invalid Light LC OM Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_lc_om_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->mode = net_buf_simple_pull_u8(buf);
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_lc_om_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS: {
struct bt_mesh_light_lc_light_onoff_status *status = NULL;
if (buf->len != 1 && buf->len != 3) {
BT_ERR("Invalid Light LC Light OnOff Status length %d", buf->len);
return;
}
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_lc_light_onoff_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->present_light_onoff = net_buf_simple_pull_u8(buf);
if (buf->len) {
status->op_en = true;
status->target_light_onoff = net_buf_simple_pull_u8(buf);
status->remain_time = net_buf_simple_pull_u8(buf);
}
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_lc_light_onoff_status);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS: {
struct bt_mesh_light_lc_property_status *status = NULL;
status = bt_mesh_calloc(sizeof(struct bt_mesh_light_lc_property_status));
if (!status) {
BT_ERR("%s, Out of memory", __func__);
return;
}
status->light_lc_property_id = net_buf_simple_pull_le16(buf);
status->light_lc_property_value = bt_mesh_alloc_buf(buf->len);
if (!status->light_lc_property_value) {
BT_ERR("%s, Out of memory", __func__);
bt_mesh_free(status);
return;
}
net_buf_simple_add_mem(status->light_lc_property_value, buf->data, buf->len);
val = (uint8_t *)status;
len = sizeof(struct bt_mesh_light_lc_property_status);
break;
}
default:
BT_ERR("Invalid Lighting Status opcode 0x%04x", ctx->recv_op);
return;
}
buf->data = val;
buf->len = len;
bt_mesh_mutex_lock(&lighting_client_lock);
node = bt_mesh_is_client_recv_publish_msg(model, ctx, buf, true);
if (!node) {
BT_DBG("Unexpected Lighting Status 0x%04x", ctx->recv_op);
} else {
switch (node->opcode) {
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_GET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_GET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_GET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_GET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_GET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_GET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_GET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_GET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_GET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_GET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_TARGET_GET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_GET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_GET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_GET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_GET:
case BLE_MESH_MODEL_OP_LIGHT_XYL_GET:
case BLE_MESH_MODEL_OP_LIGHT_XYL_TARGET_GET:
case BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_GET:
case BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_GET:
case BLE_MESH_MODEL_OP_LIGHT_LC_MODE_GET:
case BLE_MESH_MODEL_OP_LIGHT_LC_OM_GET:
case BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_GET:
case BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET:
evt = BTC_BLE_MESH_EVT_LIGHTING_CLIENT_GET_STATE;
break;
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_SET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_SET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET:
case BLE_MESH_MODEL_OP_LIGHT_XYL_SET:
case BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET:
case BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET:
case BLE_MESH_MODEL_OP_LIGHT_LC_MODE_SET:
case BLE_MESH_MODEL_OP_LIGHT_LC_OM_SET:
case BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET:
case BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET:
evt = BTC_BLE_MESH_EVT_LIGHTING_CLIENT_SET_STATE;
break;
default:
break;
}
if (!k_delayed_work_free(&node->timer)) {
uint32_t opcode = node->opcode;
bt_mesh_client_free_node(node);
bt_mesh_lighting_client_cb_evt_to_btc(opcode, evt, model, ctx, val, len);
}
}
bt_mesh_mutex_unlock(&lighting_client_lock);
switch (ctx->recv_op) {
case BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS: {
struct bt_mesh_light_lc_property_status *status;
status = (struct bt_mesh_light_lc_property_status *)val;
bt_mesh_free_buf(status->light_lc_property_value);
break;
}
default:
break;
}
bt_mesh_free(val);
}
const struct bt_mesh_model_op bt_mesh_light_lightness_cli_op[] = {
{ BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS, 2, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS, 2, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_STATUS, 2, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS, 2, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS, 5, light_status },
BLE_MESH_MODEL_OP_END,
};
const struct bt_mesh_model_op bt_mesh_light_ctl_cli_op[] = {
{ BLE_MESH_MODEL_OP_LIGHT_CTL_STATUS, 4, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS, 4, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS, 5, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS, 6, light_status },
BLE_MESH_MODEL_OP_END,
};
const struct bt_mesh_model_op bt_mesh_light_hsl_cli_op[] = {
{ BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS, 6, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_TARGET_STATUS, 6, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS, 2, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS, 2, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS, 6, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS, 9, light_status },
BLE_MESH_MODEL_OP_END,
};
const struct bt_mesh_model_op bt_mesh_light_xyl_cli_op[] = {
{ BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS, 6, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_XYL_TARGET_STATUS, 6, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS, 6, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS, 9, light_status },
BLE_MESH_MODEL_OP_END,
};
const struct bt_mesh_model_op bt_mesh_light_lc_cli_op[] = {
{ BLE_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS, 1, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_LC_OM_STATUS, 1, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS, 1, light_status },
{ BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS, 2, light_status },
BLE_MESH_MODEL_OP_END,
};
static int light_get_state(bt_mesh_client_common_param_t *common, void *value)
{
NET_BUF_SIMPLE_DEFINE(msg, BLE_MESH_LIGHT_GET_STATE_MSG_LEN);
bt_mesh_model_msg_init(&msg, common->opcode);
if (value) {
switch (common->opcode) {
case BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET: {
struct bt_mesh_light_lc_property_get *get;
get = (struct bt_mesh_light_lc_property_get *)value;
net_buf_simple_add_le16(&msg, get->light_lc_property_id);
break;
}
default:
BT_DBG("No parameters for Lighting Get 0x%04x", common->opcode);
break;
}
}
return bt_mesh_client_send_msg(common, &msg, true, timeout_handler);
}
static int light_set_state(bt_mesh_client_common_param_t *common,
void *value, uint16_t value_len, bool need_ack)
{
struct net_buf_simple *msg = NULL;
int err = 0;
msg = bt_mesh_alloc_buf(value_len);
if (!msg) {
BT_ERR("%s, Out of memory", __func__);
return -ENOMEM;
}
bt_mesh_model_msg_init(msg, common->opcode);
switch (common->opcode) {
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET_UNACK: {
struct bt_mesh_light_lightness_set *set;
set = (struct bt_mesh_light_lightness_set *)value;
net_buf_simple_add_le16(msg, set->lightness);
net_buf_simple_add_u8(msg, set->tid);
if (set->op_en) {
net_buf_simple_add_u8(msg, set->trans_time);
net_buf_simple_add_u8(msg, set->delay);
}
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET_UNACK: {
struct bt_mesh_light_lightness_linear_set *set;
set = (struct bt_mesh_light_lightness_linear_set *)value;
net_buf_simple_add_le16(msg, set->lightness);
net_buf_simple_add_u8(msg, set->tid);
if (set->op_en) {
net_buf_simple_add_u8(msg, set->trans_time);
net_buf_simple_add_u8(msg, set->delay);
}
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET_UNACK: {
struct bt_mesh_light_lightness_default_set *set;
set = (struct bt_mesh_light_lightness_default_set *)value;
net_buf_simple_add_le16(msg, set->lightness);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET_UNACK: {
struct bt_mesh_light_lightness_range_set *set;
set = (struct bt_mesh_light_lightness_range_set *)value;
net_buf_simple_add_le16(msg, set->range_min);
net_buf_simple_add_le16(msg, set->range_max);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_CTL_SET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_SET_UNACK: {
struct bt_mesh_light_ctl_set *set;
set = (struct bt_mesh_light_ctl_set *)value;
net_buf_simple_add_le16(msg, set->ctl_lightness);
net_buf_simple_add_le16(msg, set->ctl_temperature);
net_buf_simple_add_le16(msg, set->ctl_delta_uv);
net_buf_simple_add_u8(msg, set->tid);
if (set->op_en) {
net_buf_simple_add_u8(msg, set->trans_time);
net_buf_simple_add_u8(msg, set->delay);
}
break;
}
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET_UNACK: {
struct bt_mesh_light_ctl_temperature_set *set;
set = (struct bt_mesh_light_ctl_temperature_set *)value;
net_buf_simple_add_le16(msg, set->ctl_temperature);
net_buf_simple_add_le16(msg, set->ctl_delta_uv);
net_buf_simple_add_u8(msg, set->tid);
if (set->op_en) {
net_buf_simple_add_u8(msg, set->trans_time);
net_buf_simple_add_u8(msg, set->delay);
}
break;
}
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET_UNACK: {
struct bt_mesh_light_ctl_temperature_range_set *set;
set = (struct bt_mesh_light_ctl_temperature_range_set *)value;
net_buf_simple_add_le16(msg, set->range_min);
net_buf_simple_add_le16(msg, set->range_max);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET_UNACK: {
struct bt_mesh_light_ctl_default_set *set;
set = (struct bt_mesh_light_ctl_default_set *)value;
net_buf_simple_add_le16(msg, set->lightness);
net_buf_simple_add_le16(msg, set->temperature);
net_buf_simple_add_le16(msg, set->delta_uv);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_HSL_SET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_SET_UNACK: {
struct bt_mesh_light_hsl_set *set;
set = (struct bt_mesh_light_hsl_set *)value;
net_buf_simple_add_le16(msg, set->hsl_lightness);
net_buf_simple_add_le16(msg, set->hsl_hue);
net_buf_simple_add_le16(msg, set->hsl_saturation);
net_buf_simple_add_u8(msg, set->tid);
if (set->op_en) {
net_buf_simple_add_u8(msg, set->trans_time);
net_buf_simple_add_u8(msg, set->delay);
}
break;
}
case BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET_UNACK: {
struct bt_mesh_light_hsl_hue_set *set;
set = (struct bt_mesh_light_hsl_hue_set *)value;
net_buf_simple_add_le16(msg, set->hue);
net_buf_simple_add_u8(msg, set->tid);
if (set->op_en) {
net_buf_simple_add_u8(msg, set->trans_time);
net_buf_simple_add_u8(msg, set->delay);
}
break;
}
case BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET_UNACK: {
struct bt_mesh_light_hsl_saturation_set *set;
set = (struct bt_mesh_light_hsl_saturation_set *)value;
net_buf_simple_add_le16(msg, set->saturation);
net_buf_simple_add_u8(msg, set->tid);
if (set->op_en) {
net_buf_simple_add_u8(msg, set->trans_time);
net_buf_simple_add_u8(msg, set->delay);
}
break;
}
case BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET_UNACK: {
struct bt_mesh_light_hsl_default_set *set;
set = (struct bt_mesh_light_hsl_default_set *)value;
net_buf_simple_add_le16(msg, set->lightness);
net_buf_simple_add_le16(msg, set->hue);
net_buf_simple_add_le16(msg, set->saturation);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET_UNACK: {
struct bt_mesh_light_hsl_range_set *set;
set = (struct bt_mesh_light_hsl_range_set *)value;
net_buf_simple_add_le16(msg, set->hue_range_min);
net_buf_simple_add_le16(msg, set->hue_range_max);
net_buf_simple_add_le16(msg, set->saturation_range_min);
net_buf_simple_add_le16(msg, set->saturation_range_max);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_XYL_SET:
case BLE_MESH_MODEL_OP_LIGHT_XYL_SET_UNACK: {
struct bt_mesh_light_xyl_set *set;
set = (struct bt_mesh_light_xyl_set *)value;
net_buf_simple_add_le16(msg, set->xyl_lightness);
net_buf_simple_add_le16(msg, set->xyl_x);
net_buf_simple_add_le16(msg, set->xyl_y);
net_buf_simple_add_u8(msg, set->tid);
if (set->op_en) {
net_buf_simple_add_u8(msg, set->trans_time);
net_buf_simple_add_u8(msg, set->delay);
}
break;
}
case BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET:
case BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET_UNACK: {
struct bt_mesh_light_xyl_default_set *set;
set = (struct bt_mesh_light_xyl_default_set *)value;
net_buf_simple_add_le16(msg, set->lightness);
net_buf_simple_add_le16(msg, set->xyl_x);
net_buf_simple_add_le16(msg, set->xyl_y);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET:
case BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET_UNACK: {
struct bt_mesh_light_xyl_range_set *set;
set = (struct bt_mesh_light_xyl_range_set *)value;
net_buf_simple_add_le16(msg, set->xyl_x_range_min);
net_buf_simple_add_le16(msg, set->xyl_x_range_max);
net_buf_simple_add_le16(msg, set->xyl_y_range_min);
net_buf_simple_add_le16(msg, set->xyl_y_range_max);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LC_MODE_SET:
case BLE_MESH_MODEL_OP_LIGHT_LC_MODE_SET_UNACK: {
struct bt_mesh_light_lc_mode_set *set;
set = (struct bt_mesh_light_lc_mode_set *)value;
net_buf_simple_add_u8(msg, set->mode);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LC_OM_SET:
case BLE_MESH_MODEL_OP_LIGHT_LC_OM_SET_UNACK: {
struct bt_mesh_light_lc_om_set *set;
set = (struct bt_mesh_light_lc_om_set *)value;
net_buf_simple_add_u8(msg, set->mode);
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET:
case BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET_UNACK: {
struct bt_mesh_light_lc_light_onoff_set *set;
set = (struct bt_mesh_light_lc_light_onoff_set *)value;
net_buf_simple_add_u8(msg, set->light_onoff);
net_buf_simple_add_u8(msg, set->tid);
if (set->op_en) {
net_buf_simple_add_u8(msg, set->trans_time);
net_buf_simple_add_u8(msg, set->delay);
}
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET:
case BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET_UNACK: {
struct bt_mesh_light_lc_property_set *set;
set = (struct bt_mesh_light_lc_property_set *)value;
net_buf_simple_add_le16(msg, set->light_lc_property_id);
net_buf_simple_add_mem(msg, set->light_lc_property_value->data, set->light_lc_property_value->len);
break;
}
default:
BT_ERR("Invalid Lighting Set opcode 0x%04x", common->opcode);
err = -EINVAL;
goto end;
}
err = bt_mesh_client_send_msg(common, msg, need_ack, timeout_handler);
end:
bt_mesh_free_buf(msg);
return err;
}
int bt_mesh_light_client_get_state(bt_mesh_client_common_param_t *common, void *get)
{
bt_mesh_light_client_t *client = NULL;
if (!common || !common->model) {
BT_ERR("%s, Invalid parameter", __func__);
return -EINVAL;
}
client = (bt_mesh_light_client_t *)common->model->user_data;
if (!client || !client->internal_data) {
BT_ERR("Invalid Lighting client data");
return -EINVAL;
}
switch (common->opcode) {
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_GET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_GET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_GET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_GET:
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_GET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_GET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_GET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_GET:
case BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_GET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_GET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_TARGET_GET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_GET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_GET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_GET:
case BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_GET:
case BLE_MESH_MODEL_OP_LIGHT_XYL_GET:
case BLE_MESH_MODEL_OP_LIGHT_XYL_TARGET_GET:
case BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_GET:
case BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_GET:
case BLE_MESH_MODEL_OP_LIGHT_LC_MODE_GET:
case BLE_MESH_MODEL_OP_LIGHT_LC_OM_GET:
case BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_GET:
break;
case BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET:
if (!get) {
BT_ERR("Invalid Lighting LC Property Get");
return -EINVAL;
}
break;
default:
BT_ERR("Invalid Lighting Get opcode 0x%04x", common->opcode);
return -EINVAL;
}
return light_get_state(common, get);
}
int bt_mesh_light_client_set_state(bt_mesh_client_common_param_t *common, void *set)
{
bt_mesh_light_client_t *client = NULL;
uint16_t length = 0U;
bool need_ack = false;
if (!common || !common->model || !set) {
BT_ERR("%s, Invalid parameter", __func__);
return -EINVAL;
}
client = (bt_mesh_light_client_t *)common->model->user_data;
if (!client || !client->internal_data) {
BT_ERR("Invalid Lighting client data");
return -EINVAL;
}
switch (common->opcode) {
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET_UNACK: {
struct bt_mesh_light_lightness_set *value;
value = (struct bt_mesh_light_lightness_set *)set;
if (value->op_en) {
if ((value->trans_time & 0x3F) > 0x3E) {
BT_ERR("Invalid Light Lightness Set transition time");
return -EINVAL;
}
}
length = BLE_MESH_LIGHT_LIGHTNESS_SET_MSG_LEN;
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET_UNACK: {
struct bt_mesh_light_lightness_linear_set *value;
value = (struct bt_mesh_light_lightness_linear_set *)set;
if (value->op_en) {
if ((value->trans_time & 0x3F) > 0x3E) {
BT_ERR("Invalid Light Lightness Linear Set transition time");
return -EINVAL;
}
}
length = BLE_MESH_LIGHT_LIGHTNESS_LINEAR_SET_MSG_LEN;
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET_UNACK:
length = BLE_MESH_LIGHT_LIGHTNESS_DEFAULT_SET_MSG_LEN;
break;
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET_UNACK: {
struct bt_mesh_light_lightness_range_set *value;
value = (struct bt_mesh_light_lightness_range_set *)set;
if (value->range_min > value->range_max) {
BT_ERR("Light Lightness Range Set range min is greater than range max");
return -EINVAL;
}
length = BLE_MESH_LIGHT_LIGHTNESS_RANGE_SET_MSG_LEN;
break;
}
case BLE_MESH_MODEL_OP_LIGHT_CTL_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_CTL_SET_UNACK: {
struct bt_mesh_light_ctl_set *value;
value = (struct bt_mesh_light_ctl_set *)set;
if (value->op_en) {
if ((value->trans_time & 0x3F) > 0x3E) {
BT_ERR("Invalid Light CTL Set transition time");
return -EINVAL;
}
}
length = BLE_MESH_LIGHT_CTL_SET_MSG_LEN;
break;
}
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET_UNACK: {
struct bt_mesh_light_ctl_temperature_set *value;
value = (struct bt_mesh_light_ctl_temperature_set *)set;
if (value->op_en) {
if ((value->trans_time & 0x3F) > 0x3E) {
BT_ERR("Invalid Light CTL Temperature Set transition time");
return -EINVAL;
}
}
length = BLE_MESH_LIGHT_CTL_TEMPERATURE_SET_MSG_LEN;
break;
}
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET_UNACK: {
struct bt_mesh_light_ctl_temperature_range_set *value;
value = (struct bt_mesh_light_ctl_temperature_range_set *)set;
if (value->range_min > value->range_max) {
BT_ERR("Light CTL Temperature Range Set range min is greater than range max");
return -EINVAL;
}
length = BLE_MESH_LIGHT_CTL_TEMPERATURE_RANGE_SET_MSG_LEN;
break;
}
case BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET_UNACK:
length = BLE_MESH_LIGHT_CTL_DEFAULT_SET_MSG_LEN;
break;
case BLE_MESH_MODEL_OP_LIGHT_HSL_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_HSL_SET_UNACK: {
struct bt_mesh_light_hsl_set *value;
value = (struct bt_mesh_light_hsl_set *)set;
if (value->op_en) {
if ((value->trans_time & 0x3F) > 0x3E) {
BT_ERR("Invalid Light HSL Set transition time");
return -EINVAL;
}
}
length = BLE_MESH_LIGHT_HSL_SET_MSG_LEN;
break;
}
case BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET_UNACK: {
struct bt_mesh_light_hsl_hue_set *value;
value = (struct bt_mesh_light_hsl_hue_set *)set;
if (value->op_en) {
if ((value->trans_time & 0x3F) > 0x3E) {
BT_ERR("Invalid Light HSL Hue Set transition time");
return -EINVAL;
}
}
length = BLE_MESH_LIGHT_HSL_HUE_SET_MSG_LEN;
break;
}
case BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET_UNACK: {
struct bt_mesh_light_hsl_saturation_set *value;
value = (struct bt_mesh_light_hsl_saturation_set *)set;
if (value->op_en) {
if ((value->trans_time & 0x3F) > 0x3E) {
BT_ERR("Invalid Light HSL Saturation Set transition time");
return -EINVAL;
}
}
length = BLE_MESH_LIGHT_HSL_SATURATION_SET_MSG_LEN;
break;
}
case BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET_UNACK:
length = BLE_MESH_LIGHT_HSL_DEFAULT_SET_MSG_LEN;
break;
case BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET_UNACK: {
struct bt_mesh_light_hsl_range_set *value;
value = (struct bt_mesh_light_hsl_range_set *)set;
if (value->hue_range_min > value->hue_range_max ||
value->saturation_range_min > value->saturation_range_max) {
BT_ERR("Light HSL Range Set range min is greater than range max");
return -EINVAL;
}
length = BLE_MESH_LIGHT_HSL_RANGE_SET_MSG_LEN;
break;
}
case BLE_MESH_MODEL_OP_LIGHT_XYL_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_XYL_SET_UNACK: {
struct bt_mesh_light_xyl_set *value;
value = (struct bt_mesh_light_xyl_set *)set;
if (value->op_en) {
if ((value->trans_time & 0x3F) > 0x3E) {
BT_ERR("Invalid Light xyL Set transition time");
return -EINVAL;
}
}
length = BLE_MESH_LIGHT_XYL_SET_MSG_LEN;
break;
}
case BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET_UNACK:
length = BLE_MESH_LIGHT_XYL_DEFAULT_SET_MSG_LEN;
break;
case BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET_UNACK: {
struct bt_mesh_light_xyl_range_set *value;
value = (struct bt_mesh_light_xyl_range_set *)set;
if (value->xyl_x_range_min > value->xyl_x_range_max ||
value->xyl_y_range_min > value->xyl_y_range_max) {
BT_ERR("Light xyL Range Set range min is greater than range max");
return -EINVAL;
}
length = BLE_MESH_LIGHT_XYL_RANGE_SET_MSG_LEN;
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LC_MODE_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_LC_MODE_SET_UNACK:
length = BLE_MESH_LIGHT_LC_MODE_SET_MSG_LEN;
break;
case BLE_MESH_MODEL_OP_LIGHT_LC_OM_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_LC_OM_SET_UNACK:
length = BLE_MESH_LIGHT_LC_OM_SET_MSG_LEN;
break;
case BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET_UNACK: {
struct bt_mesh_light_lc_light_onoff_set *value;
value = (struct bt_mesh_light_lc_light_onoff_set *)set;
if (value->op_en) {
if ((value->trans_time & 0x3F) > 0x3E) {
BT_ERR("Invalid Light LC Light OnOff Set transition time");
return -EINVAL;
}
}
length = BLE_MESH_LIGHT_LC_LIGHT_ONOFF_SET_MSG_LEN;
break;
}
case BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET:
need_ack = true;
case BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET_UNACK: {
struct bt_mesh_light_lc_property_set *value;
value = (struct bt_mesh_light_lc_property_set *)set;
if (!value->light_lc_property_value) {
BT_ERR("Invalid Lighting Light LC Property value");
return -EINVAL;
}
length = (1 + 2 + value->light_lc_property_value->len + 4);
break;
}
default:
BT_ERR("Invalid Lighting Set opcode 0x%04x", common->opcode);
return -EINVAL;
}
return light_set_state(common, set, length, need_ack);
}
static int lighting_client_init(struct bt_mesh_model *model)
{
light_internal_data_t *internal = NULL;
bt_mesh_light_client_t *client = NULL;
if (!model) {
BT_ERR("Invalid Lighting client model");
return -EINVAL;
}
client = (bt_mesh_light_client_t *)model->user_data;
if (!client) {
BT_ERR("No Lighting client context provided");
return -EINVAL;
}
if (client->internal_data) {
BT_WARN("%s, Already", __func__);
return -EALREADY;
}
internal = bt_mesh_calloc(sizeof(light_internal_data_t));
if (!internal) {
BT_ERR("%s, Out of memory", __func__);
return -ENOMEM;
}
sys_slist_init(&internal->queue);
client->model = model;
client->op_pair_size = ARRAY_SIZE(light_op_pair);
client->op_pair = light_op_pair;
client->internal_data = internal;
bt_mesh_mutex_create(&lighting_client_lock);
return 0;
}
#if CONFIG_BLE_MESH_DEINIT
static int lighting_client_deinit(struct bt_mesh_model *model)
{
bt_mesh_light_client_t *client = NULL;
if (!model) {
BT_ERR("Invalid Lighting client model");
return -EINVAL;
}
client = (bt_mesh_light_client_t *)model->user_data;
if (!client) {
BT_ERR("No Lighting client context provided");
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_mutex_free(&lighting_client_lock);
return 0;
}
#endif /* CONFIG_BLE_MESH_DEINIT */
const struct bt_mesh_model_cb bt_mesh_lighting_client_cb = {
.init = lighting_client_init,
#if CONFIG_BLE_MESH_DEINIT
.deinit = lighting_client_deinit,
#endif /* CONFIG_BLE_MESH_DEINIT */
};
#endif /* CONFIG_BLE_MESH_LIGHTING_CLIENT */