mirror of
				https://github.com/espressif/esp-idf.git
				synced 2025-10-26 11:39:30 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			3598 lines
		
	
	
		
			135 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			3598 lines
		
	
	
		
			135 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* Bluetooth: Mesh Lighting Server Models
 | |
|  *
 | |
|  * SPDX-FileCopyrightText: 2018 Vikrant More
 | |
|  * SPDX-FileContributor: 2018-2021 Espressif Systems (Shanghai) CO LTD
 | |
|  *
 | |
|  * SPDX-License-Identifier: Apache-2.0
 | |
|  */
 | |
| 
 | |
| #include <errno.h>
 | |
| 
 | |
| #include "btc_ble_mesh_lighting_model.h"
 | |
| 
 | |
| #include "mesh/config.h"
 | |
| #include "access.h"
 | |
| #include "transport.h"
 | |
| #include "mesh/model_opcode.h"
 | |
| #include "mesh/state_transition.h"
 | |
| #include "mesh/device_property.h"
 | |
| 
 | |
| #if CONFIG_BLE_MESH_LIGHTING_SERVER
 | |
| 
 | |
| static bt_mesh_mutex_t light_server_lock;
 | |
| 
 | |
| void bt_mesh_light_server_lock(void)
 | |
| {
 | |
|     bt_mesh_mutex_lock(&light_server_lock);
 | |
| }
 | |
| 
 | |
| void bt_mesh_light_server_unlock(void)
 | |
| {
 | |
|     bt_mesh_mutex_unlock(&light_server_lock);
 | |
| }
 | |
| 
 | |
| /* message handlers (Start) */
 | |
| 
 | |
| /* Light Lightness Server/Setup Server message handlers */
 | |
| 
 | |
| static void send_light_lightness_status(struct bt_mesh_model *model,
 | |
|                                         struct bt_mesh_msg_ctx *ctx,
 | |
|                                         bool publish, uint16_t opcode)
 | |
| {
 | |
|     struct net_buf_simple *msg = NULL;
 | |
|     uint8_t length = 2 + 5;
 | |
| 
 | |
|     if (ctx == NULL && publish == false) {
 | |
|         BT_ERR("%s, Invalid parameter", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (publish == false) {
 | |
|         msg = bt_mesh_alloc_buf(length + BLE_MESH_SERVER_TRANS_MIC_SIZE);
 | |
|         if (msg == NULL) {
 | |
|             BT_ERR("%s, Out of memory", __func__);
 | |
|             return;
 | |
|         }
 | |
|     } else {
 | |
|         msg = bt_mesh_server_get_pub_msg(model, length);
 | |
|         if (msg == NULL) {
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     bt_mesh_model_msg_init(msg, opcode);
 | |
|     switch (opcode) {
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS: {
 | |
|         struct bt_mesh_light_lightness_srv *srv = model->user_data;
 | |
|         net_buf_simple_add_le16(msg, srv->state->lightness_actual);
 | |
|         if (srv->actual_transition.counter) {
 | |
|             bt_mesh_server_calc_remain_time(&srv->actual_transition);
 | |
|             net_buf_simple_add_le16(msg, srv->state->target_lightness_actual);
 | |
|             net_buf_simple_add_u8(msg, srv->actual_transition.remain_time);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS: {
 | |
|         struct bt_mesh_light_lightness_srv *srv = model->user_data;
 | |
|         net_buf_simple_add_le16(msg, srv->state->lightness_linear);
 | |
|         if (srv->linear_transition.counter) {
 | |
|             bt_mesh_server_calc_remain_time(&srv->linear_transition);
 | |
|             net_buf_simple_add_le16(msg, srv->state->target_lightness_linear);
 | |
|             net_buf_simple_add_u8(msg, srv->linear_transition.remain_time);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_STATUS: {
 | |
|         struct bt_mesh_light_lightness_srv *srv = model->user_data;
 | |
|         net_buf_simple_add_le16(msg, srv->state->lightness_last);
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS:
 | |
|         if (model->id == BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_SRV) {
 | |
|             struct bt_mesh_light_lightness_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_le16(msg, srv->state->lightness_default);
 | |
|         } else if (model->id == BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_SETUP_SRV) {
 | |
|             struct bt_mesh_light_lightness_setup_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_le16(msg, srv->state->lightness_default);
 | |
|         }
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS:
 | |
|         if (model->id == BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_SRV) {
 | |
|             struct bt_mesh_light_lightness_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_u8(msg, srv->state->status_code);
 | |
|             net_buf_simple_add_le16(msg, srv->state->lightness_range_min);
 | |
|             net_buf_simple_add_le16(msg, srv->state->lightness_range_max);
 | |
|         } else if (model->id == BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_SETUP_SRV) {
 | |
|             struct bt_mesh_light_lightness_setup_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_u8(msg, srv->state->status_code);
 | |
|             net_buf_simple_add_le16(msg, srv->state->lightness_range_min);
 | |
|             net_buf_simple_add_le16(msg, srv->state->lightness_range_max);
 | |
|         }
 | |
|         break;
 | |
|     default:
 | |
|         BT_WARN("Unknown Light Lightness status opcode 0x%04x", opcode);
 | |
|         if (publish == false) {
 | |
|             bt_mesh_free_buf(msg);
 | |
|         }
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (publish == false) {
 | |
|         BLE_MESH_CHECK_SEND_STATUS(bt_mesh_model_send(model, ctx, msg, NULL, NULL));
 | |
|         bt_mesh_free_buf(msg);
 | |
|     } else {
 | |
|         BLE_MESH_CHECK_SEND_STATUS(bt_mesh_model_publish(model));
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void light_lightness_get(struct bt_mesh_model *model,
 | |
|                                 struct bt_mesh_msg_ctx *ctx,
 | |
|                                 struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_lightness_srv *srv = model->user_data;
 | |
|     uint16_t opcode = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.get_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_GET_MSG,
 | |
|                                               model, ctx, NULL, 0);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     switch (ctx->recv_op) {
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS;
 | |
|         break;
 | |
|     default:
 | |
|         BT_WARN("Unknown Light Lightness Get opcode 0x%04x", ctx->recv_op);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     send_light_lightness_status(model, ctx, false, opcode);
 | |
| }
 | |
| 
 | |
| void light_lightness_publish(struct bt_mesh_model *model, uint16_t opcode)
 | |
| {
 | |
|     if (model->user_data == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     switch (model->id) {
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_SRV: {
 | |
|         struct bt_mesh_light_lightness_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light Lightness Server state");
 | |
|             return;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_SETUP_SRV: {
 | |
|         struct bt_mesh_light_lightness_setup_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light Lightness Setup Server state");
 | |
|             return;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     default:
 | |
|         BT_ERR("Invalid Light Lightness Server model 0x%04x", model->id);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     send_light_lightness_status(model, NULL, true, opcode);
 | |
| }
 | |
| 
 | |
| static void light_lightness_set(struct bt_mesh_model *model,
 | |
|                                 struct bt_mesh_msg_ctx *ctx,
 | |
|                                 struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_lightness_srv *srv = model->user_data;
 | |
|     uint8_t tid = 0U, trans_time = 0U, delay = 0U;
 | |
|     bool optional = false;
 | |
|     uint16_t actual = 0U;
 | |
|     int64_t now = 0;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     actual = net_buf_simple_pull_le16(buf);
 | |
|     tid = net_buf_simple_pull_u8(buf);
 | |
| 
 | |
|     if (bt_mesh_server_get_optional(model, ctx, buf, &trans_time, &delay, &optional)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .lightness_set.op_en = optional,
 | |
|             .lightness_set.lightness = actual,
 | |
|             .lightness_set.tid = tid,
 | |
|             .lightness_set.trans_time = trans_time,
 | |
|             .lightness_set.delay = delay,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (bt_mesh_is_server_recv_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now)) {
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET) {
 | |
|             send_light_lightness_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS);
 | |
|         }
 | |
|         send_light_lightness_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS);
 | |
|         /* In this condition, no event will be callback to application layer */
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     bt_mesh_light_server_lock();
 | |
| 
 | |
|     bt_mesh_server_stop_transition(&srv->actual_transition);
 | |
|     bt_mesh_server_update_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now);
 | |
| 
 | |
|     if (actual) {
 | |
|         if (srv->state->lightness_range_min && actual < srv->state->lightness_range_min) {
 | |
|             actual = srv->state->lightness_range_min;
 | |
|         } else if (srv->state->lightness_range_max && actual > srv->state->lightness_range_max) {
 | |
|             actual = srv->state->lightness_range_max;
 | |
|         }
 | |
|     }
 | |
|     srv->state->target_lightness_actual = actual;
 | |
| 
 | |
|     /**
 | |
|      * If the target state is equal to the current state, the transition shall not be
 | |
|      * started and is considered complete.
 | |
|      */
 | |
|     if (srv->state->target_lightness_actual != srv->state->lightness_actual) {
 | |
|         light_lightness_actual_tt_values(srv, trans_time, delay);
 | |
|     } else {
 | |
|         bt_mesh_light_server_state_change_t change = {
 | |
|             .lightness_set.lightness = srv->state->lightness_actual,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                               model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET) {
 | |
|             send_light_lightness_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS);
 | |
|         }
 | |
|         send_light_lightness_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS);
 | |
| 
 | |
|         bt_mesh_light_server_unlock();
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Copy the ctx of the received message */
 | |
|     if (srv->actual_transition.timer.work.user_data) {
 | |
|         memcpy(srv->actual_transition.timer.work.user_data, ctx, sizeof(struct bt_mesh_msg_ctx));
 | |
|     }
 | |
| 
 | |
|     /* For Instantaneous Transition */
 | |
|     if (srv->actual_transition.counter == 0U) {
 | |
|         srv->state->lightness_actual = srv->state->target_lightness_actual;
 | |
|         /**
 | |
|          * Whenever the Light Lightness Actual state is changed with a non-transactional
 | |
|          * message or a completed sequence of transactional messages to a non-zero value,
 | |
|          * the value of the Light Lightness Last shall be set to the value of the Light
 | |
|          * Lightness Actual.
 | |
|          */
 | |
|         if (srv->state->lightness_actual) {
 | |
|             srv->state->lightness_last = srv->state->lightness_actual;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     srv->actual_transition.just_started = true;
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET) {
 | |
|         send_light_lightness_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS);
 | |
|     }
 | |
|     send_light_lightness_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS);
 | |
| 
 | |
|     bt_mesh_light_server_unlock();
 | |
| 
 | |
|     bt_mesh_server_start_transition(&srv->actual_transition);
 | |
| }
 | |
| 
 | |
| static void light_lightness_linear_set(struct bt_mesh_model *model,
 | |
|                                        struct bt_mesh_msg_ctx *ctx,
 | |
|                                        struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_lightness_srv *srv = model->user_data;
 | |
|     uint8_t tid = 0U, trans_time = 0U, delay = 0U;
 | |
|     bool optional = false;
 | |
|     uint16_t linear = 0U;
 | |
|     int64_t now = 0;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     linear = net_buf_simple_pull_le16(buf);
 | |
|     tid = net_buf_simple_pull_u8(buf);
 | |
| 
 | |
|     if (bt_mesh_server_get_optional(model, ctx, buf, &trans_time, &delay, &optional)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .lightness_linear_set.op_en = optional,
 | |
|             .lightness_linear_set.lightness = linear,
 | |
|             .lightness_linear_set.tid = tid,
 | |
|             .lightness_linear_set.trans_time = trans_time,
 | |
|             .lightness_linear_set.delay = delay,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (bt_mesh_is_server_recv_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now)) {
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET) {
 | |
|             send_light_lightness_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS);
 | |
|         }
 | |
|         send_light_lightness_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS);
 | |
|         /* In this condition, no event will be callback to application layer */
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     bt_mesh_light_server_lock();
 | |
| 
 | |
|     bt_mesh_server_stop_transition(&srv->linear_transition);
 | |
|     bt_mesh_server_update_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now);
 | |
| 
 | |
|     srv->state->target_lightness_linear = linear;
 | |
| 
 | |
|     /**
 | |
|      * If the target state is equal to the current state, the transition shall not
 | |
|      * be started and is considered complete.
 | |
|      */
 | |
|     if (srv->state->target_lightness_linear != srv->state->lightness_linear) {
 | |
|         light_lightness_linear_tt_values(srv, trans_time, delay);
 | |
|     } else {
 | |
|         bt_mesh_light_server_state_change_t change = {
 | |
|             .lightness_linear_set.lightness = srv->state->lightness_actual,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                               model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET) {
 | |
|             send_light_lightness_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS);
 | |
|         }
 | |
|         send_light_lightness_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS);
 | |
| 
 | |
|         bt_mesh_light_server_unlock();
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Copy the ctx of the received message */
 | |
|     if (srv->linear_transition.timer.work.user_data) {
 | |
|         memcpy(srv->linear_transition.timer.work.user_data, ctx, sizeof(struct bt_mesh_msg_ctx));
 | |
|     }
 | |
| 
 | |
|     /* For Instantaneous Transition */
 | |
|     if (srv->linear_transition.counter == 0U) {
 | |
|         srv->state->lightness_linear = srv->state->target_lightness_linear;
 | |
|     }
 | |
| 
 | |
|     srv->linear_transition.just_started = true;
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET) {
 | |
|         send_light_lightness_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS);
 | |
|     }
 | |
|     send_light_lightness_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS);
 | |
| 
 | |
|     bt_mesh_light_server_unlock();
 | |
| 
 | |
|     bt_mesh_server_start_transition(&srv->linear_transition);
 | |
| }
 | |
| 
 | |
| static void light_lightness_default_set(struct bt_mesh_model *model,
 | |
|                                         struct bt_mesh_msg_ctx *ctx,
 | |
|                                         struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_lightness_setup_srv *srv = model->user_data;
 | |
|     uint16_t lightness = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     lightness = net_buf_simple_pull_le16(buf);
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .lightness_default_set.lightness = lightness,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (srv->state->lightness_default != lightness) {
 | |
|         srv->state->lightness_default = lightness;
 | |
| 
 | |
|         bt_mesh_light_server_state_change_t change = {
 | |
|             .lightness_default_set.lightness = lightness,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                               model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
|     }
 | |
| 
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET) {
 | |
|         send_light_lightness_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS);
 | |
|     }
 | |
|     send_light_lightness_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS);
 | |
| }
 | |
| 
 | |
| static void light_lightness_range_set(struct bt_mesh_model *model,
 | |
|                                       struct bt_mesh_msg_ctx *ctx,
 | |
|                                       struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_lightness_setup_srv *srv = model->user_data;
 | |
|     uint16_t range_min = 0U, range_max = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     range_min = net_buf_simple_pull_le16(buf);
 | |
|     range_max = net_buf_simple_pull_le16(buf);
 | |
| 
 | |
|     if (range_min > range_max) {
 | |
|         BT_ERR("Range min 0x%04x is greater than range max 0x%04x",
 | |
|                 range_min, range_max);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .lightness_range_set.range_min = range_min,
 | |
|             .lightness_range_set.range_max = range_max,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * When a Light Lightness Setup Server receives a Light Lightness Range Set
 | |
|      * message or a Light Lightness Range Set Unacknowledged message with values
 | |
|      * that cannot be accepted, it shall set the status of the operation to a
 | |
|      * value representing the reason why the values cannot be accepted.
 | |
|      *
 | |
|      * TODO: 0x0000 for Light Range Min/Max is prohibited, but BQB test case
 | |
|      * MMDL/SR/LLNS/BI-01-C requires 'SUCCESS' when it sends a set message with
 | |
|      * Light Range Min set to 0x0000.
 | |
|      */
 | |
| #if 0
 | |
|     srv->state->status_code = BLE_MESH_RANGE_UPDATE_SUCCESS;
 | |
| #else
 | |
|     if (range_min == 0x0000) {
 | |
|         srv->state->status_code = BLE_MESH_CANNOT_SET_RANGE_MIN;
 | |
|     } else if (range_max == 0x0000) {
 | |
|         srv->state->status_code = BLE_MESH_CANNOT_SET_RANGE_MAX;
 | |
|     } else {
 | |
|         srv->state->status_code = BLE_MESH_RANGE_UPDATE_SUCCESS;
 | |
|     }
 | |
| #endif
 | |
| 
 | |
|     if (range_min && srv->state->lightness_range_min != range_min) {
 | |
|         srv->state->lightness_range_min = range_min;
 | |
|     }
 | |
| 
 | |
|     if (range_max && srv->state->lightness_range_max != range_max) {
 | |
|         srv->state->lightness_range_max = range_max;
 | |
|     }
 | |
| 
 | |
|     bt_mesh_light_server_state_change_t change = {
 | |
|         .lightness_range_set.range_min = srv->state->lightness_range_min,
 | |
|         .lightness_range_set.range_max = srv->state->lightness_range_max,
 | |
|     };
 | |
|     bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                           model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET) {
 | |
|         send_light_lightness_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS);
 | |
|     }
 | |
|     send_light_lightness_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS);
 | |
| }
 | |
| 
 | |
| /* Light CTL Server/Temperature Server/Setup Server message handlers */
 | |
| 
 | |
| static void send_light_ctl_status(struct bt_mesh_model *model,
 | |
|                                   struct bt_mesh_msg_ctx *ctx,
 | |
|                                   bool publish, uint16_t opcode)
 | |
| {
 | |
|     struct net_buf_simple *msg = NULL;
 | |
|     uint8_t length = 2 + 9;
 | |
| 
 | |
|     if (ctx == NULL && publish == false) {
 | |
|         BT_ERR("%s, Invalid parameter", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (publish == false) {
 | |
|         msg = bt_mesh_alloc_buf(length + BLE_MESH_SERVER_TRANS_MIC_SIZE);
 | |
|         if (msg == NULL) {
 | |
|             BT_ERR("%s, Out of memory", __func__);
 | |
|             return;
 | |
|         }
 | |
|     } else {
 | |
|         msg = bt_mesh_server_get_pub_msg(model, length);
 | |
|         if (msg == NULL) {
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     bt_mesh_model_msg_init(msg, opcode);
 | |
|     switch (opcode) {
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_CTL_STATUS: {
 | |
|         struct bt_mesh_light_ctl_srv *srv = model->user_data;
 | |
|         net_buf_simple_add_le16(msg, srv->state->lightness);
 | |
|         net_buf_simple_add_le16(msg, srv->state->temperature);
 | |
|         if (srv->transition.counter) {
 | |
|             bt_mesh_server_calc_remain_time(&srv->transition);
 | |
|             net_buf_simple_add_le16(msg, srv->state->target_lightness);
 | |
|             net_buf_simple_add_le16(msg, srv->state->target_temperature);
 | |
|             net_buf_simple_add_u8(msg, srv->transition.remain_time);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS:
 | |
|         if (model->id == BLE_MESH_MODEL_ID_LIGHT_CTL_SRV) {
 | |
|             struct bt_mesh_light_ctl_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_u8(msg, srv->state->status_code);
 | |
|             net_buf_simple_add_le16(msg, srv->state->temperature_range_min);
 | |
|             net_buf_simple_add_le16(msg, srv->state->temperature_range_max);
 | |
|         } else if (model->id == BLE_MESH_MODEL_ID_LIGHT_CTL_SETUP_SRV) {
 | |
|             struct bt_mesh_light_ctl_setup_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_u8(msg, srv->state->status_code);
 | |
|             net_buf_simple_add_le16(msg, srv->state->temperature_range_min);
 | |
|             net_buf_simple_add_le16(msg, srv->state->temperature_range_max);
 | |
|         }
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS: {
 | |
|         if (model->id == BLE_MESH_MODEL_ID_LIGHT_CTL_SRV) {
 | |
|             struct bt_mesh_light_ctl_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_le16(msg, srv->state->lightness_default);
 | |
|             net_buf_simple_add_le16(msg, srv->state->temperature_default);
 | |
|             net_buf_simple_add_le16(msg, srv->state->delta_uv_default);
 | |
|         } else if (model->id == BLE_MESH_MODEL_ID_LIGHT_CTL_SETUP_SRV) {
 | |
|             struct bt_mesh_light_ctl_setup_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_le16(msg, srv->state->lightness_default);
 | |
|             net_buf_simple_add_le16(msg, srv->state->temperature_default);
 | |
|             net_buf_simple_add_le16(msg, srv->state->delta_uv_default);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS: {
 | |
|         struct bt_mesh_light_ctl_temp_srv *srv = model->user_data;
 | |
|         net_buf_simple_add_le16(msg, srv->state->temperature);
 | |
|         net_buf_simple_add_le16(msg, srv->state->delta_uv);
 | |
|         if (srv->transition.counter) {
 | |
|             bt_mesh_server_calc_remain_time(&srv->transition);
 | |
|             net_buf_simple_add_le16(msg, srv->state->target_temperature);
 | |
|             net_buf_simple_add_le16(msg, srv->state->target_delta_uv);
 | |
|             net_buf_simple_add_u8(msg, srv->transition.remain_time);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     default:
 | |
|         BT_WARN("Unknown Light CTL status opcode 0x%04x", opcode);
 | |
|         if (publish == false) {
 | |
|             bt_mesh_free_buf(msg);
 | |
|         }
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (publish == false) {
 | |
|         BLE_MESH_CHECK_SEND_STATUS(bt_mesh_model_send(model, ctx, msg, NULL, NULL));
 | |
|         bt_mesh_free_buf(msg);
 | |
|     } else {
 | |
|         BLE_MESH_CHECK_SEND_STATUS(bt_mesh_model_publish(model));
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void light_ctl_get(struct bt_mesh_model *model,
 | |
|                           struct bt_mesh_msg_ctx *ctx,
 | |
|                           struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_server_rsp_ctrl *rsp_ctrl = NULL;
 | |
|     uint16_t opcode = 0U;
 | |
| 
 | |
|     if (model->user_data == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     switch (model->id) {
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_CTL_SRV: {
 | |
|         struct bt_mesh_light_ctl_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light CTL Server state");
 | |
|             return;
 | |
|         }
 | |
|         rsp_ctrl = &srv->rsp_ctrl;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_CTL_TEMP_SRV: {
 | |
|         struct bt_mesh_light_ctl_temp_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light CTL Temperature Server state");
 | |
|             return;
 | |
|         }
 | |
|         rsp_ctrl = &srv->rsp_ctrl;
 | |
|         break;
 | |
|     }
 | |
|     default:
 | |
|         BT_ERR("Invalid Light CTL Server model 0x%04x", model->id);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (rsp_ctrl->get_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_GET_MSG,
 | |
|                                               model, ctx, NULL, 0);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     switch (ctx->recv_op) {
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_CTL_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_CTL_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS;
 | |
|         break;
 | |
|     default:
 | |
|         BT_WARN("Unknown Light CTL Get opcode 0x%04x", ctx->recv_op);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     send_light_ctl_status(model, ctx, false, opcode);
 | |
| }
 | |
| 
 | |
| void light_ctl_publish(struct bt_mesh_model *model, uint16_t opcode)
 | |
| {
 | |
|     if (model->user_data == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     switch (model->id) {
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_CTL_SRV: {
 | |
|         struct bt_mesh_light_ctl_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light CTL Server state");
 | |
|             return;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_CTL_TEMP_SRV: {
 | |
|         struct bt_mesh_light_ctl_temp_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light CTL Temperature Server state");
 | |
|             return;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_CTL_SETUP_SRV: {
 | |
|         struct bt_mesh_light_ctl_setup_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light CTL Setup Server state");
 | |
|             return;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     default:
 | |
|         BT_ERR("Invalid Light CTL Server model 0x%04x", model->id);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     send_light_ctl_status(model, NULL, true, opcode);
 | |
| }
 | |
| 
 | |
| static void light_ctl_set(struct bt_mesh_model *model,
 | |
|                           struct bt_mesh_msg_ctx *ctx,
 | |
|                           struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_ctl_srv *srv = model->user_data;
 | |
|     uint16_t lightness = 0U, temperature = 0U;
 | |
|     uint8_t tid = 0U, trans_time = 0U, delay = 0U;
 | |
|     int16_t delta_uv = 0;
 | |
|     bool optional = false;
 | |
|     int64_t now = 0;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     lightness = net_buf_simple_pull_le16(buf);
 | |
|     temperature = net_buf_simple_pull_le16(buf);
 | |
|     delta_uv = (int16_t) net_buf_simple_pull_le16(buf);
 | |
|     tid = net_buf_simple_pull_u8(buf);
 | |
| 
 | |
|     if (temperature < BLE_MESH_TEMPERATURE_MIN || temperature > BLE_MESH_TEMPERATURE_MAX) {
 | |
|         BT_ERR("Invalid temperature 0x%04x", temperature);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (bt_mesh_server_get_optional(model, ctx, buf, &trans_time, &delay, &optional)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .ctl_set.op_en = optional,
 | |
|             .ctl_set.lightness = lightness,
 | |
|             .ctl_set.temperature = temperature,
 | |
|             .ctl_set.delta_uv = delta_uv,
 | |
|             .ctl_set.tid = tid,
 | |
|             .ctl_set.trans_time = trans_time,
 | |
|             .ctl_set.delay = delay,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (bt_mesh_is_server_recv_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now)) {
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_CTL_SET) {
 | |
|             send_light_ctl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_CTL_STATUS);
 | |
|         }
 | |
|         send_light_ctl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_CTL_STATUS);
 | |
|         /* In this condition, no event will be callback to application layer */
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     bt_mesh_light_server_lock();
 | |
| 
 | |
|     bt_mesh_server_stop_transition(&srv->transition);
 | |
|     bt_mesh_server_update_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now);
 | |
| 
 | |
|     srv->state->target_lightness = lightness;
 | |
|     if (srv->state->temperature_range_min &&
 | |
|         srv->state->temperature_range_min != BLE_MESH_TEMPERATURE_UNKNOWN &&
 | |
|         temperature < srv->state->temperature_range_min) {
 | |
|         temperature = srv->state->temperature_range_min;
 | |
|     } else if (srv->state->temperature_range_max &&
 | |
|                srv->state->temperature_range_max != BLE_MESH_TEMPERATURE_UNKNOWN &&
 | |
|                temperature > srv->state->temperature_range_max) {
 | |
|         temperature = srv->state->temperature_range_max;
 | |
|     }
 | |
|     srv->state->target_temperature = temperature;
 | |
|     srv->state->target_delta_uv = delta_uv;
 | |
| 
 | |
|     if (srv->state->target_lightness != srv->state->lightness ||
 | |
|         srv->state->target_temperature != srv->state->temperature ||
 | |
|         srv->state->target_delta_uv != srv->state->delta_uv) {
 | |
|         light_ctl_tt_values(srv, trans_time, delay);
 | |
|     } else {
 | |
|         bt_mesh_light_server_state_change_t change = {
 | |
|             .ctl_set.lightness = srv->state->lightness,
 | |
|             .ctl_set.temperature = srv->state->temperature,
 | |
|             .ctl_set.delta_uv = srv->state->delta_uv,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                               model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_CTL_SET) {
 | |
|             send_light_ctl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_CTL_STATUS);
 | |
|         }
 | |
|         send_light_ctl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_CTL_STATUS);
 | |
| 
 | |
|         bt_mesh_light_server_unlock();
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Copy the ctx of the received message */
 | |
|     if (srv->transition.timer.work.user_data) {
 | |
|         memcpy(srv->transition.timer.work.user_data, ctx, sizeof(struct bt_mesh_msg_ctx));
 | |
|     }
 | |
| 
 | |
|     /* For Instantaneous Transition */
 | |
|     if (srv->transition.counter == 0U) {
 | |
|         srv->state->lightness = srv->state->target_lightness;
 | |
|         srv->state->temperature = srv->state->target_temperature;
 | |
|         srv->state->delta_uv = srv->state->target_delta_uv;
 | |
|     }
 | |
| 
 | |
|     srv->transition.just_started = true;
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_CTL_SET) {
 | |
|         send_light_ctl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_CTL_STATUS);
 | |
|     }
 | |
|     send_light_ctl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_CTL_STATUS);
 | |
| 
 | |
|     bt_mesh_light_server_unlock();
 | |
| 
 | |
|     bt_mesh_server_start_transition(&srv->transition);
 | |
| }
 | |
| 
 | |
| static void light_ctl_default_set(struct bt_mesh_model *model,
 | |
|                                   struct bt_mesh_msg_ctx *ctx,
 | |
|                                   struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_ctl_setup_srv *srv = model->user_data;
 | |
|     uint16_t lightness = 0U, temperature = 0U;
 | |
|     int16_t delta_uv = 0;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     lightness = net_buf_simple_pull_le16(buf);
 | |
|     temperature = net_buf_simple_pull_le16(buf);
 | |
|     delta_uv = (int16_t) net_buf_simple_pull_le16(buf);
 | |
| 
 | |
|     if (temperature < BLE_MESH_TEMPERATURE_MIN || temperature > BLE_MESH_TEMPERATURE_MAX) {
 | |
|         BT_ERR("Invalid temperature 0x%04x", temperature);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .ctl_default_set.lightness = lightness,
 | |
|             .ctl_default_set.temperature = temperature,
 | |
|             .ctl_default_set.delta_uv = delta_uv,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (srv->state->temperature_range_min &&
 | |
|         srv->state->temperature_range_min != BLE_MESH_TEMPERATURE_UNKNOWN &&
 | |
|         temperature < srv->state->temperature_range_min) {
 | |
|         temperature = srv->state->temperature_range_min;
 | |
|     } else if (srv->state->temperature_range_max &&
 | |
|                srv->state->temperature_range_max != BLE_MESH_TEMPERATURE_UNKNOWN &&
 | |
|                temperature > srv->state->temperature_range_max) {
 | |
|         temperature = srv->state->temperature_range_max;
 | |
|     }
 | |
| 
 | |
|     srv->state->lightness_default = lightness;
 | |
|     srv->state->temperature_default = temperature;
 | |
|     srv->state->delta_uv_default = delta_uv;
 | |
| 
 | |
|     bt_mesh_light_server_state_change_t change = {
 | |
|         .ctl_default_set.lightness = srv->state->lightness_default,
 | |
|         .ctl_default_set.temperature = srv->state->temperature_default,
 | |
|         .ctl_default_set.delta_uv = srv->state->delta_uv_default,
 | |
|     };
 | |
|     bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                           model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET) {
 | |
|         send_light_ctl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS);
 | |
|     }
 | |
|     send_light_ctl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS);
 | |
| }
 | |
| 
 | |
| static void light_ctl_temp_range_set(struct bt_mesh_model *model,
 | |
|                                      struct bt_mesh_msg_ctx *ctx,
 | |
|                                      struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_ctl_setup_srv *srv = model->user_data;
 | |
|     uint16_t min = 0U, max = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     min = net_buf_simple_pull_le16(buf);
 | |
|     max = net_buf_simple_pull_le16(buf);
 | |
| 
 | |
|     /* This is as per 6.1.3.1 in Mesh Model Specification */
 | |
|     if (min > max ||
 | |
|             min < BLE_MESH_TEMPERATURE_MIN || (min != BLE_MESH_TEMPERATURE_UNKNOWN && min > BLE_MESH_TEMPERATURE_MAX) ||
 | |
|             max < BLE_MESH_TEMPERATURE_MIN || (max != BLE_MESH_TEMPERATURE_UNKNOWN && max > BLE_MESH_TEMPERATURE_MAX)) {
 | |
|         BT_ERR("Invalid parameter, range min 0x%04x, range max 0x%04x",
 | |
|                 min, max);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .ctl_temp_range_set.range_min = min,
 | |
|             .ctl_temp_range_set.range_max = max,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (min == BLE_MESH_TEMPERATURE_UNKNOWN) {
 | |
|         srv->state->status_code = BLE_MESH_CANNOT_SET_RANGE_MIN;
 | |
|     } else if (max == BLE_MESH_TEMPERATURE_UNKNOWN ) {
 | |
|         srv->state->status_code = BLE_MESH_CANNOT_SET_RANGE_MAX;
 | |
|     } else {
 | |
|         srv->state->status_code = BLE_MESH_RANGE_UPDATE_SUCCESS;
 | |
|     }
 | |
| 
 | |
|     if (min != BLE_MESH_TEMPERATURE_UNKNOWN && srv->state->temperature_range_min != min) {
 | |
|         srv->state->temperature_range_min = min;
 | |
|     }
 | |
| 
 | |
|     if (max != BLE_MESH_TEMPERATURE_UNKNOWN && srv->state->temperature_range_max != max) {
 | |
|         srv->state->temperature_range_max = max;
 | |
|     }
 | |
| 
 | |
|     bt_mesh_light_server_state_change_t change = {
 | |
|         .ctl_temp_range_set.range_min = srv->state->temperature_range_min,
 | |
|         .ctl_temp_range_set.range_max = srv->state->temperature_range_max,
 | |
|     };
 | |
|     bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                           model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET) {
 | |
|         send_light_ctl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS);
 | |
|     }
 | |
|     send_light_ctl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS);
 | |
| }
 | |
| 
 | |
| static void light_ctl_temp_set(struct bt_mesh_model *model,
 | |
|                                struct bt_mesh_msg_ctx *ctx,
 | |
|                                struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_ctl_temp_srv *srv = model->user_data;
 | |
|     uint8_t tid = 0U, trans_time = 0U, delay = 0U;
 | |
|     uint16_t temperature = 0U;
 | |
|     int16_t delta_uv = 0;
 | |
|     bool optional = false;
 | |
|     int64_t now = 0;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     temperature = net_buf_simple_pull_le16(buf);
 | |
|     delta_uv = (int16_t) net_buf_simple_pull_le16(buf);
 | |
|     tid = net_buf_simple_pull_u8(buf);
 | |
| 
 | |
|     if (temperature < BLE_MESH_TEMPERATURE_MIN || temperature > BLE_MESH_TEMPERATURE_MAX) {
 | |
|         BT_ERR("Invalid temperature 0x%04x", temperature);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (bt_mesh_server_get_optional(model, ctx, buf, &trans_time, &delay, &optional)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .ctl_temp_set.op_en = optional,
 | |
|             .ctl_temp_set.temperature = temperature,
 | |
|             .ctl_temp_set.delta_uv = delta_uv,
 | |
|             .ctl_temp_set.tid = tid,
 | |
|             .ctl_temp_set.trans_time = trans_time,
 | |
|             .ctl_temp_set.delay = delay,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (bt_mesh_is_server_recv_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now)) {
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET) {
 | |
|             send_light_ctl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS);
 | |
|         }
 | |
|         send_light_ctl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS);
 | |
|         /* In this condition, no event will be callback to application layer */
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     bt_mesh_light_server_lock();
 | |
| 
 | |
|     bt_mesh_server_stop_transition(&srv->transition);
 | |
|     bt_mesh_server_update_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now);
 | |
| 
 | |
|     if (srv->state->temperature_range_min &&
 | |
|         srv->state->temperature_range_min != BLE_MESH_TEMPERATURE_UNKNOWN &&
 | |
|         temperature < srv->state->temperature_range_min) {
 | |
|         temperature = srv->state->temperature_range_min;
 | |
|     } else if (srv->state->temperature_range_max &&
 | |
|                srv->state->temperature_range_max != BLE_MESH_TEMPERATURE_UNKNOWN &&
 | |
|                temperature > srv->state->temperature_range_max) {
 | |
|         temperature = srv->state->temperature_range_max;
 | |
|     }
 | |
|     srv->state->target_temperature = temperature;
 | |
|     srv->state->target_delta_uv = delta_uv;
 | |
| 
 | |
|     if (srv->state->target_temperature != srv->state->temperature ||
 | |
|             srv->state->target_delta_uv != srv->state->delta_uv) {
 | |
|         light_ctl_temp_tt_values(srv, trans_time, delay);
 | |
|     } else {
 | |
|         bt_mesh_light_server_state_change_t change = {
 | |
|             .ctl_temp_set.temperature = srv->state->temperature,
 | |
|             .ctl_temp_set.delta_uv = srv->state->delta_uv,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                               model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET) {
 | |
|             send_light_ctl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS);
 | |
|         }
 | |
|         send_light_ctl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS);
 | |
| 
 | |
|         bt_mesh_light_server_unlock();
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Copy the ctx of the received message */
 | |
|     if (srv->transition.timer.work.user_data) {
 | |
|         memcpy(srv->transition.timer.work.user_data, ctx, sizeof(struct bt_mesh_msg_ctx));
 | |
|     }
 | |
| 
 | |
|     /* For Instantaneous Transition */
 | |
|     if (srv->transition.counter == 0U) {
 | |
|         srv->state->temperature = srv->state->target_temperature;
 | |
|         srv->state->delta_uv = srv->state->target_delta_uv;
 | |
|     }
 | |
| 
 | |
|     srv->transition.just_started = true;
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET) {
 | |
|         send_light_ctl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS);
 | |
|     }
 | |
|     send_light_ctl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS);
 | |
| 
 | |
|     bt_mesh_light_server_unlock();
 | |
| 
 | |
|     bt_mesh_server_start_transition(&srv->transition);
 | |
| }
 | |
| 
 | |
| /* Light HSL Server/Hue Server/Saturation Server/Setup Server message handlers */
 | |
| 
 | |
| static void send_light_hsl_status(struct bt_mesh_model *model,
 | |
|                                   struct bt_mesh_msg_ctx *ctx,
 | |
|                                   bool publish, uint16_t opcode)
 | |
| {
 | |
|     struct net_buf_simple *msg = NULL;
 | |
|     uint8_t length = 2 + 9;
 | |
| 
 | |
|     if (ctx == NULL && publish == false) {
 | |
|         BT_ERR("%s, Invalid parameter", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (publish == false) {
 | |
|         msg = bt_mesh_alloc_buf(length + BLE_MESH_SERVER_TRANS_MIC_SIZE);
 | |
|         if (msg == NULL) {
 | |
|             BT_ERR("%s, Out of memory", __func__);
 | |
|             return;
 | |
|         }
 | |
|     } else {
 | |
|         msg = bt_mesh_server_get_pub_msg(model, length);
 | |
|         if (msg == NULL) {
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     bt_mesh_model_msg_init(msg, opcode);
 | |
|     switch (opcode) {
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS:
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_HSL_TARGET_STATUS: {
 | |
|         struct bt_mesh_light_hsl_srv *srv = model->user_data;
 | |
|         if (opcode == BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS) {
 | |
|             net_buf_simple_add_le16(msg, srv->state->lightness);
 | |
|             net_buf_simple_add_le16(msg, srv->state->hue);
 | |
|             net_buf_simple_add_le16(msg, srv->state->saturation);
 | |
|             if (srv->transition.counter) {
 | |
|                 bt_mesh_server_calc_remain_time(&srv->transition);
 | |
|                 net_buf_simple_add_u8(msg, srv->transition.remain_time);
 | |
|             }
 | |
|         } else if (opcode == BLE_MESH_MODEL_OP_LIGHT_HSL_TARGET_STATUS) {
 | |
|             net_buf_simple_add_le16(msg, srv->state->target_lightness);
 | |
|             net_buf_simple_add_le16(msg, srv->state->target_hue);
 | |
|             net_buf_simple_add_le16(msg, srv->state->target_saturation);
 | |
|             if (srv->transition.counter) {
 | |
|                 bt_mesh_server_calc_remain_time(&srv->transition);
 | |
|                 net_buf_simple_add_u8(msg, srv->transition.remain_time);
 | |
|             }
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS:
 | |
|         if (model->id == BLE_MESH_MODEL_ID_LIGHT_HSL_SRV) {
 | |
|             struct bt_mesh_light_hsl_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_le16(msg, srv->state->lightness_default);
 | |
|             net_buf_simple_add_le16(msg, srv->state->hue_default);
 | |
|             net_buf_simple_add_le16(msg, srv->state->saturation_default);
 | |
|         } else if (model->id == BLE_MESH_MODEL_ID_LIGHT_HSL_SETUP_SRV) {
 | |
|             struct bt_mesh_light_hsl_setup_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_le16(msg, srv->state->lightness_default);
 | |
|             net_buf_simple_add_le16(msg, srv->state->hue_default);
 | |
|             net_buf_simple_add_le16(msg, srv->state->saturation_default);
 | |
|         }
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS:
 | |
|         if (model->id == BLE_MESH_MODEL_ID_LIGHT_HSL_SRV) {
 | |
|             struct bt_mesh_light_hsl_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_u8(msg, srv->state->status_code);
 | |
|             net_buf_simple_add_le16(msg, srv->state->hue_range_min);
 | |
|             net_buf_simple_add_le16(msg, srv->state->hue_range_max);
 | |
|             net_buf_simple_add_le16(msg, srv->state->saturation_range_min);
 | |
|             net_buf_simple_add_le16(msg, srv->state->saturation_range_max);
 | |
|         } else if (model->id == BLE_MESH_MODEL_ID_LIGHT_HSL_SETUP_SRV) {
 | |
|             struct bt_mesh_light_hsl_setup_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_u8(msg, srv->state->status_code);
 | |
|             net_buf_simple_add_le16(msg, srv->state->hue_range_min);
 | |
|             net_buf_simple_add_le16(msg, srv->state->hue_range_max);
 | |
|             net_buf_simple_add_le16(msg, srv->state->saturation_range_min);
 | |
|             net_buf_simple_add_le16(msg, srv->state->saturation_range_max);
 | |
|         }
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS: {
 | |
|         struct bt_mesh_light_hsl_hue_srv *srv = model->user_data;
 | |
|         net_buf_simple_add_le16(msg, srv->state->hue);
 | |
|         if (srv->transition.counter) {
 | |
|             bt_mesh_server_calc_remain_time(&srv->transition);
 | |
|             net_buf_simple_add_le16(msg, srv->state->target_hue);
 | |
|             net_buf_simple_add_u8(msg, srv->transition.remain_time);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS: {
 | |
|         struct bt_mesh_light_hsl_sat_srv *srv = model->user_data;
 | |
|         net_buf_simple_add_le16(msg, srv->state->saturation);
 | |
|         if (srv->transition.counter) {
 | |
|             bt_mesh_server_calc_remain_time(&srv->transition);
 | |
|             net_buf_simple_add_le16(msg, srv->state->target_saturation);
 | |
|             net_buf_simple_add_u8(msg, srv->transition.remain_time);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     default:
 | |
|         BT_WARN("Unknown Light HSL status opcode 0x%04x", opcode);
 | |
|         if (publish == false) {
 | |
|             bt_mesh_free_buf(msg);
 | |
|         }
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (publish == false) {
 | |
|         BLE_MESH_CHECK_SEND_STATUS(bt_mesh_model_send(model, ctx, msg, NULL, NULL));
 | |
|         bt_mesh_free_buf(msg);
 | |
|     } else {
 | |
|         BLE_MESH_CHECK_SEND_STATUS(bt_mesh_model_publish(model));
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void light_hsl_get(struct bt_mesh_model *model,
 | |
|                           struct bt_mesh_msg_ctx *ctx,
 | |
|                           struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_server_rsp_ctrl *rsp_ctrl = NULL;
 | |
|     uint16_t opcode = 0U;
 | |
| 
 | |
|     if (model->user_data == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     switch (model->id) {
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_SRV: {
 | |
|         struct bt_mesh_light_hsl_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light HSL Server state");
 | |
|             return;
 | |
|         }
 | |
|         rsp_ctrl = &srv->rsp_ctrl;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_HUE_SRV: {
 | |
|         struct bt_mesh_light_hsl_hue_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light HSL Hue Server state");
 | |
|             return;
 | |
|         }
 | |
|         rsp_ctrl = &srv->rsp_ctrl;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_SAT_SRV: {
 | |
|         struct bt_mesh_light_hsl_sat_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light HSL Saturation Server state");
 | |
|             return;
 | |
|         }
 | |
|         rsp_ctrl = &srv->rsp_ctrl;
 | |
|         break;
 | |
|     }
 | |
|     default:
 | |
|         BT_ERR("Invalid Light HSL Server model 0x%04x", model->id);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (rsp_ctrl->get_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_GET_MSG,
 | |
|                                               model, ctx, NULL, 0);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     switch (ctx->recv_op) {
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_HSL_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_HSL_TARGET_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_HSL_TARGET_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS;
 | |
|         break;
 | |
|     default:
 | |
|         BT_WARN("Unknown Light HSL Get opcode 0x%04x", ctx->recv_op);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     send_light_hsl_status(model, ctx, false, opcode);
 | |
| }
 | |
| 
 | |
| void light_hsl_publish(struct bt_mesh_model *model, uint16_t opcode)
 | |
| {
 | |
|     if (model->user_data == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     switch (model->id) {
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_SRV: {
 | |
|         struct bt_mesh_light_hsl_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light HSL Server state");
 | |
|             return;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_HUE_SRV: {
 | |
|         struct bt_mesh_light_hsl_hue_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light HSL Hue Server state");
 | |
|             return;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_SAT_SRV: {
 | |
|         struct bt_mesh_light_hsl_sat_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light HSL Saturation Server state");
 | |
|             return;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_SETUP_SRV: {
 | |
|         struct bt_mesh_light_hsl_setup_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light HSL Setup Server state");
 | |
|             return;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     default:
 | |
|         BT_ERR("Invalid Light HSL Server model 0x%04x", model->id);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     send_light_hsl_status(model, NULL, true, opcode);
 | |
| }
 | |
| 
 | |
| static void light_hsl_set(struct bt_mesh_model *model,
 | |
|                           struct bt_mesh_msg_ctx *ctx,
 | |
|                           struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_hsl_srv *srv = model->user_data;
 | |
|     uint16_t lightness = 0U, hue = 0U, saturation = 0U;
 | |
|     uint8_t tid = 0U, trans_time = 0U, delay = 0U;
 | |
|     bool optional = false;
 | |
|     int64_t now = 0;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     lightness = net_buf_simple_pull_le16(buf);
 | |
|     hue = net_buf_simple_pull_le16(buf);
 | |
|     saturation = net_buf_simple_pull_le16(buf);
 | |
|     tid = net_buf_simple_pull_u8(buf);
 | |
| 
 | |
|     if (bt_mesh_server_get_optional(model, ctx, buf, &trans_time, &delay, &optional)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .hsl_set.op_en = optional,
 | |
|             .hsl_set.lightness = lightness,
 | |
|             .hsl_set.hue = hue,
 | |
|             .hsl_set.saturation = saturation,
 | |
|             .hsl_set.tid = tid,
 | |
|             .hsl_set.trans_time = trans_time,
 | |
|             .hsl_set.delay = delay,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (bt_mesh_is_server_recv_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now)) {
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_HSL_SET) {
 | |
|             send_light_hsl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS);
 | |
|         }
 | |
|         send_light_hsl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS);
 | |
|         /* In this condition, no event will be callback to application layer */
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     bt_mesh_light_server_lock();
 | |
| 
 | |
|     bt_mesh_server_stop_transition(&srv->transition);
 | |
|     bt_mesh_server_update_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now);
 | |
| 
 | |
|     srv->state->target_lightness = lightness;
 | |
|     if (srv->state->hue_range_min && hue < srv->state->hue_range_min) {
 | |
|         hue = srv->state->hue_range_min;
 | |
|     } else if (srv->state->hue_range_max && hue > srv->state->hue_range_max) {
 | |
|         hue = srv->state->hue_range_max;
 | |
|     }
 | |
|     srv->state->target_hue = hue;
 | |
|     if (srv->state->saturation_range_min && saturation < srv->state->saturation_range_min) {
 | |
|         saturation = srv->state->saturation_range_min;
 | |
|     } else if (srv->state->saturation_range_max && saturation > srv->state->saturation_range_max) {
 | |
|         saturation = srv->state->saturation_range_max;
 | |
|     }
 | |
|     srv->state->target_saturation = saturation;
 | |
| 
 | |
|     /**
 | |
|      * If the target state is equal to the current state, the transition shall not
 | |
|      * be started and is considered complete.
 | |
|      */
 | |
|     if (srv->state->target_lightness != srv->state->lightness ||
 | |
|             srv->state->target_hue != srv->state->hue ||
 | |
|             srv->state->target_saturation != srv->state->saturation) {
 | |
|         light_hsl_tt_values(srv, trans_time, delay);
 | |
|     } else {
 | |
|         bt_mesh_light_server_state_change_t change = {
 | |
|             .hsl_set.lightness = srv->state->lightness,
 | |
|             .hsl_set.hue = srv->state->hue,
 | |
|             .hsl_set.saturation = srv->state->saturation,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                               model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_HSL_SET) {
 | |
|             send_light_hsl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS);
 | |
|         }
 | |
|         send_light_hsl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS);
 | |
| 
 | |
|         bt_mesh_light_server_unlock();
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Copy the ctx of the received message */
 | |
|     if (srv->transition.timer.work.user_data) {
 | |
|         memcpy(srv->transition.timer.work.user_data, ctx, sizeof(struct bt_mesh_msg_ctx));
 | |
|     }
 | |
| 
 | |
|     /* For Instantaneous Transition */
 | |
|     if (srv->transition.counter == 0U) {
 | |
|         srv->state->lightness = srv->state->target_lightness;
 | |
|         srv->state->hue = srv->state->target_hue;
 | |
|         srv->state->saturation = srv->state->target_saturation;
 | |
|     }
 | |
| 
 | |
|     srv->transition.just_started = true;
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_HSL_SET) {
 | |
|         send_light_hsl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS);
 | |
|     }
 | |
|     send_light_hsl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS);
 | |
| 
 | |
|     bt_mesh_light_server_unlock();
 | |
| 
 | |
|     bt_mesh_server_start_transition(&srv->transition);
 | |
| }
 | |
| 
 | |
| static void light_hsl_default_set(struct bt_mesh_model *model,
 | |
|                                   struct bt_mesh_msg_ctx *ctx,
 | |
|                                   struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_hsl_setup_srv *srv = model->user_data;
 | |
|     uint16_t lightness = 0U, hue = 0U, saturation = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     lightness = net_buf_simple_pull_le16(buf);
 | |
|     hue = net_buf_simple_pull_le16(buf);
 | |
|     saturation = net_buf_simple_pull_le16(buf);
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .hsl_default_set.lightness = lightness,
 | |
|             .hsl_default_set.hue = hue,
 | |
|             .hsl_default_set.saturation = saturation,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (srv->state->hue_range_min && hue < srv->state->hue_range_min) {
 | |
|         hue = srv->state->hue_range_min;
 | |
|     } else if (srv->state->hue_range_max && hue > srv->state->hue_range_max) {
 | |
|         hue = srv->state->hue_range_max;
 | |
|     }
 | |
| 
 | |
|     if (srv->state->saturation_range_min && saturation < srv->state->saturation_range_min) {
 | |
|         saturation = srv->state->saturation_range_min;
 | |
|     } else if (srv->state->saturation_range_max && saturation > srv->state->saturation_range_max) {
 | |
|         saturation = srv->state->saturation_range_max;
 | |
|     }
 | |
| 
 | |
|     srv->state->lightness_default = lightness;
 | |
|     srv->state->hue_default = hue;
 | |
|     srv->state->saturation_default = saturation;
 | |
| 
 | |
|     bt_mesh_light_server_state_change_t change = {
 | |
|         .hsl_default_set.lightness = srv->state->lightness_default,
 | |
|         .hsl_default_set.hue = srv->state->hue_default,
 | |
|         .hsl_default_set.saturation = srv->state->saturation_default,
 | |
|     };
 | |
|     bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                           model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET) {
 | |
|         send_light_hsl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS);
 | |
|     }
 | |
|     send_light_hsl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS);
 | |
| }
 | |
| 
 | |
| static void light_hsl_range_set(struct bt_mesh_model *model,
 | |
|                                 struct bt_mesh_msg_ctx *ctx,
 | |
|                                 struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_hsl_setup_srv *srv = model->user_data;
 | |
|     uint16_t hue_min = 0U, hue_max = 0U, saturation_min = 0U, saturation_max = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     hue_min = net_buf_simple_pull_le16(buf);
 | |
|     hue_max = net_buf_simple_pull_le16(buf);
 | |
|     saturation_min = net_buf_simple_pull_le16(buf);
 | |
|     saturation_max = net_buf_simple_pull_le16(buf);
 | |
| 
 | |
|     if (hue_min > hue_max) {
 | |
|         BT_ERR("Invalid parameter, hue min 0x%04x, hue max 0x%04x",
 | |
|                 hue_min, hue_max);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (saturation_min > saturation_max) {
 | |
|         BT_ERR("Invalid parameter, saturation min 0x%04x, saturation max 0x%04x",
 | |
|                 saturation_min, saturation_max);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .hsl_range_set.hue_range_min = hue_min,
 | |
|             .hsl_range_set.hue_range_max = hue_max,
 | |
|             .hsl_range_set.sat_range_min = saturation_min,
 | |
|             .hsl_range_set.sat_range_max = saturation_max,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     srv->state->status_code = BLE_MESH_RANGE_UPDATE_SUCCESS;
 | |
|     srv->state->hue_range_min = hue_min;
 | |
|     srv->state->hue_range_max = hue_max;
 | |
|     srv->state->saturation_range_min = saturation_min;
 | |
|     srv->state->saturation_range_max = saturation_max;
 | |
| 
 | |
|     bt_mesh_light_server_state_change_t change = {
 | |
|         .hsl_range_set.hue_range_min = srv->state->hue_range_min,
 | |
|         .hsl_range_set.hue_range_max = srv->state->hue_range_max,
 | |
|         .hsl_range_set.sat_range_min = srv->state->saturation_range_min,
 | |
|         .hsl_range_set.sat_range_max = srv->state->saturation_range_max,
 | |
|     };
 | |
|     bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                           model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET) {
 | |
|         send_light_hsl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS);
 | |
|     }
 | |
|     send_light_hsl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS);
 | |
| }
 | |
| 
 | |
| static void light_hsl_hue_set(struct bt_mesh_model *model,
 | |
|                               struct bt_mesh_msg_ctx *ctx,
 | |
|                               struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_hsl_hue_srv *srv = model->user_data;
 | |
|     uint8_t tid = 0U, trans_time = 0U, delay = 0U;
 | |
|     bool optional = false;
 | |
|     uint16_t hue = 0U;
 | |
|     int64_t now = 0;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     hue = net_buf_simple_pull_le16(buf);
 | |
|     tid = net_buf_simple_pull_u8(buf);
 | |
| 
 | |
|     if (bt_mesh_server_get_optional(model, ctx, buf, &trans_time, &delay, &optional)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .hsl_hue_set.op_en = optional,
 | |
|             .hsl_hue_set.hue = hue,
 | |
|             .hsl_hue_set.tid = tid,
 | |
|             .hsl_hue_set.trans_time = trans_time,
 | |
|             .hsl_hue_set.delay = delay,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (bt_mesh_is_server_recv_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now)) {
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET) {
 | |
|             send_light_hsl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS);
 | |
|         }
 | |
|         send_light_hsl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS);
 | |
|         /* In this condition, no event will be callback to application layer */
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     bt_mesh_light_server_lock();
 | |
| 
 | |
|     bt_mesh_server_stop_transition(&srv->transition);
 | |
|     bt_mesh_server_update_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now);
 | |
| 
 | |
|     if (srv->state->hue_range_min && hue < srv->state->hue_range_min) {
 | |
|         hue = srv->state->hue_range_min;
 | |
|     } else if (srv->state->hue_range_max && hue > srv->state->hue_range_max) {
 | |
|         hue = srv->state->hue_range_max;
 | |
|     }
 | |
|     srv->state->target_hue = hue;
 | |
| 
 | |
|     /**
 | |
|      * If the target state is equal to the current state, the transition shall not
 | |
|      * be started and is considered complete.
 | |
|      */
 | |
|     if (srv->state->target_hue != srv->state->hue) {
 | |
|         light_hsl_hue_tt_values(srv, trans_time, delay);
 | |
|     } else {
 | |
|         bt_mesh_light_server_state_change_t change = {
 | |
|             .hsl_hue_set.hue = srv->state->hue,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                               model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET) {
 | |
|             send_light_hsl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS);
 | |
|         }
 | |
|         send_light_hsl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS);
 | |
| 
 | |
|         bt_mesh_light_server_unlock();
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Copy the ctx of the received message */
 | |
|     if (srv->transition.timer.work.user_data) {
 | |
|         memcpy(srv->transition.timer.work.user_data, ctx, sizeof(struct bt_mesh_msg_ctx));
 | |
|     }
 | |
| 
 | |
|     /* For Instantaneous Transition */
 | |
|     if (srv->transition.counter == 0U) {
 | |
|         srv->state->hue = srv->state->target_hue;
 | |
|     }
 | |
| 
 | |
|     srv->transition.just_started = true;
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET) {
 | |
|         send_light_hsl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS);
 | |
|     }
 | |
|     send_light_hsl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS);
 | |
| 
 | |
|     bt_mesh_light_server_unlock();
 | |
| 
 | |
|     bt_mesh_server_start_transition(&srv->transition);
 | |
| }
 | |
| 
 | |
| static void light_hsl_sat_set(struct bt_mesh_model *model,
 | |
|                               struct bt_mesh_msg_ctx *ctx,
 | |
|                               struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_hsl_sat_srv *srv = model->user_data;
 | |
|     uint8_t tid = 0U, trans_time = 0U, delay = 0U;
 | |
|     uint16_t saturation = 0U;
 | |
|     bool optional = false;
 | |
|     int64_t now = 0;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     saturation = net_buf_simple_pull_le16(buf);
 | |
|     tid = net_buf_simple_pull_u8(buf);
 | |
| 
 | |
|     if (bt_mesh_server_get_optional(model, ctx, buf, &trans_time, &delay, &optional)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .hsl_saturation_set.op_en = optional,
 | |
|             .hsl_saturation_set.saturation = saturation,
 | |
|             .hsl_saturation_set.tid = tid,
 | |
|             .hsl_saturation_set.trans_time = trans_time,
 | |
|             .hsl_saturation_set.delay = delay,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (bt_mesh_is_server_recv_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now)) {
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET) {
 | |
|             send_light_hsl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS);
 | |
|         }
 | |
|         send_light_hsl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS);
 | |
|         /* In this condition, no event will be callback to application layer */
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     bt_mesh_light_server_lock();
 | |
| 
 | |
|     bt_mesh_server_stop_transition(&srv->transition);
 | |
|     bt_mesh_server_update_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now);
 | |
| 
 | |
|     if (srv->state->saturation_range_min && saturation < srv->state->saturation_range_min) {
 | |
|         saturation = srv->state->saturation_range_min;
 | |
|     } else if (srv->state->saturation_range_max && saturation > srv->state->saturation_range_max) {
 | |
|         saturation = srv->state->saturation_range_max;
 | |
|     }
 | |
|     srv->state->target_saturation = saturation;
 | |
| 
 | |
|     /**
 | |
|      * If the target state is equal to the current state, the transition shall not
 | |
|      * be started and is considered complete.
 | |
|      */
 | |
|     if (srv->state->target_saturation != srv->state->saturation) {
 | |
|         light_hsl_sat_tt_values(srv, trans_time, delay);
 | |
|     } else {
 | |
|         bt_mesh_light_server_state_change_t change = {
 | |
|             .hsl_saturation_set.saturation = srv->state->saturation,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                               model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET) {
 | |
|             send_light_hsl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS);
 | |
|         }
 | |
|         send_light_hsl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS);
 | |
| 
 | |
|         bt_mesh_light_server_unlock();
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Copy the ctx of the received message */
 | |
|     if (srv->transition.timer.work.user_data) {
 | |
|         memcpy(srv->transition.timer.work.user_data, ctx, sizeof(struct bt_mesh_msg_ctx));
 | |
|     }
 | |
| 
 | |
|     /* For Instantaneous Transition */
 | |
|     if (srv->transition.counter == 0U) {
 | |
|         srv->state->saturation = srv->state->target_saturation;
 | |
|     }
 | |
| 
 | |
|     srv->transition.just_started = true;
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET) {
 | |
|         send_light_hsl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS);
 | |
|     }
 | |
|     send_light_hsl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS);
 | |
| 
 | |
|     bt_mesh_light_server_unlock();
 | |
| 
 | |
|     bt_mesh_server_start_transition(&srv->transition);
 | |
| }
 | |
| 
 | |
| /* Light xyL Server/Setup Server message handlers */
 | |
| 
 | |
| static void send_light_xyl_status(struct bt_mesh_model *model,
 | |
|                                   struct bt_mesh_msg_ctx *ctx,
 | |
|                                   bool publish, uint16_t opcode)
 | |
| {
 | |
|     struct net_buf_simple *msg = NULL;
 | |
|     uint8_t length = 2 + 9;
 | |
| 
 | |
|     if (ctx == NULL && publish == false) {
 | |
|         BT_ERR("%s, Invalid parameter", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (publish == false) {
 | |
|         msg = bt_mesh_alloc_buf(length + BLE_MESH_SERVER_TRANS_MIC_SIZE);
 | |
|         if (msg == NULL) {
 | |
|             BT_ERR("%s, Out of memory", __func__);
 | |
|             return;
 | |
|         }
 | |
|     } else {
 | |
|         msg = bt_mesh_server_get_pub_msg(model, length);
 | |
|         if (msg == NULL) {
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     bt_mesh_model_msg_init(msg, opcode);
 | |
|     switch (opcode) {
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS:
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_XYL_TARGET_STATUS: {
 | |
|         struct bt_mesh_light_xyl_srv *srv = model->user_data;
 | |
|         if (opcode == BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS) {
 | |
|             net_buf_simple_add_le16(msg, srv->state->lightness);
 | |
|             net_buf_simple_add_le16(msg, srv->state->x);
 | |
|             net_buf_simple_add_le16(msg, srv->state->y);
 | |
|             if (srv->transition.counter) {
 | |
|                 bt_mesh_server_calc_remain_time(&srv->transition);
 | |
|                 net_buf_simple_add_u8(msg, srv->transition.remain_time);
 | |
|             }
 | |
|         } else if (opcode == BLE_MESH_MODEL_OP_LIGHT_XYL_TARGET_STATUS) {
 | |
|             net_buf_simple_add_le16(msg, srv->state->target_lightness);
 | |
|             net_buf_simple_add_le16(msg, srv->state->target_x);
 | |
|             net_buf_simple_add_le16(msg, srv->state->target_y);
 | |
|             if (srv->transition.counter) {
 | |
|                 bt_mesh_server_calc_remain_time(&srv->transition);
 | |
|                 net_buf_simple_add_u8(msg, srv->transition.remain_time);
 | |
|             }
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS:
 | |
|         if (model->id == BLE_MESH_MODEL_ID_LIGHT_XYL_SRV) {
 | |
|             struct bt_mesh_light_xyl_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_le16(msg, srv->state->lightness_default);
 | |
|             net_buf_simple_add_le16(msg, srv->state->x_default);
 | |
|             net_buf_simple_add_le16(msg, srv->state->y_default);
 | |
|         } else if (model->id == BLE_MESH_MODEL_ID_LIGHT_XYL_SETUP_SRV) {
 | |
|             struct bt_mesh_light_xyl_setup_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_le16(msg, srv->state->lightness_default);
 | |
|             net_buf_simple_add_le16(msg, srv->state->x_default);
 | |
|             net_buf_simple_add_le16(msg, srv->state->y_default);
 | |
|         }
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS:
 | |
|         if (model->id == BLE_MESH_MODEL_ID_LIGHT_XYL_SRV) {
 | |
|             struct bt_mesh_light_xyl_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_u8(msg, srv->state->status_code);
 | |
|             net_buf_simple_add_le16(msg, srv->state->x_range_min);
 | |
|             net_buf_simple_add_le16(msg, srv->state->x_range_max);
 | |
|             net_buf_simple_add_le16(msg, srv->state->y_range_min);
 | |
|             net_buf_simple_add_le16(msg, srv->state->y_range_max);
 | |
|         } else if (model->id == BLE_MESH_MODEL_ID_LIGHT_XYL_SETUP_SRV) {
 | |
|             struct bt_mesh_light_xyl_setup_srv *srv = model->user_data;
 | |
|             net_buf_simple_add_u8(msg, srv->state->status_code);
 | |
|             net_buf_simple_add_le16(msg, srv->state->x_range_min);
 | |
|             net_buf_simple_add_le16(msg, srv->state->x_range_max);
 | |
|             net_buf_simple_add_le16(msg, srv->state->y_range_min);
 | |
|             net_buf_simple_add_le16(msg, srv->state->y_range_max);
 | |
|         }
 | |
|         break;
 | |
|     default:
 | |
|         BT_WARN("Unknown Light xyL status opcode 0x%04x", opcode);
 | |
|         if (publish == false) {
 | |
|             bt_mesh_free_buf(msg);
 | |
|         }
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (publish == false) {
 | |
|         BLE_MESH_CHECK_SEND_STATUS(bt_mesh_model_send(model, ctx, msg, NULL, NULL));
 | |
|         bt_mesh_free_buf(msg);
 | |
|     } else {
 | |
|         BLE_MESH_CHECK_SEND_STATUS(bt_mesh_model_publish(model));
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void light_xyl_get(struct bt_mesh_model *model,
 | |
|                           struct bt_mesh_msg_ctx *ctx,
 | |
|                           struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_xyl_srv *srv = model->user_data;
 | |
|     uint16_t opcode = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.get_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_GET_MSG,
 | |
|                                               model, ctx, NULL, 0);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     switch (ctx->recv_op) {
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_XYL_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_XYL_TARGET_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_XYL_TARGET_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS;
 | |
|         break;
 | |
|     default:
 | |
|         BT_WARN("Unknown Light xyL Get opcode 0x%04x", ctx->recv_op);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     send_light_xyl_status(model, ctx, false, opcode);
 | |
| }
 | |
| 
 | |
| void light_xyl_publish(struct bt_mesh_model *model, uint16_t opcode)
 | |
| {
 | |
|     if (model->user_data == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     switch (model->id) {
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_XYL_SRV: {
 | |
|         struct bt_mesh_light_xyl_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light xyL Server state");
 | |
|             return;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_XYL_SETUP_SRV: {
 | |
|         struct bt_mesh_light_xyl_setup_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light xyL Setup Server state");
 | |
|             return;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     default:
 | |
|         BT_ERR("Invalid Light xyL Server model 0x%04x", model->id);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     send_light_xyl_status(model, NULL, true, opcode);
 | |
| }
 | |
| 
 | |
| static void light_xyl_set(struct bt_mesh_model *model,
 | |
|                           struct bt_mesh_msg_ctx *ctx,
 | |
|                           struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_xyl_srv *srv = model->user_data;
 | |
|     uint8_t tid = 0U, trans_time = 0U, delay = 0U;
 | |
|     uint16_t lightness = 0U, x = 0U, y = 0U;
 | |
|     bool optional = false;
 | |
|     int64_t now = 0;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     lightness = net_buf_simple_pull_le16(buf);
 | |
|     x = net_buf_simple_pull_le16(buf);
 | |
|     y = net_buf_simple_pull_le16(buf);
 | |
|     tid = net_buf_simple_pull_u8(buf);
 | |
| 
 | |
|     if (bt_mesh_server_get_optional(model, ctx, buf, &trans_time, &delay, &optional)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .xyl_set.op_en = optional,
 | |
|             .xyl_set.lightness = lightness,
 | |
|             .xyl_set.x = x,
 | |
|             .xyl_set.y = y,
 | |
|             .xyl_set.tid = tid,
 | |
|             .xyl_set.trans_time = trans_time,
 | |
|             .xyl_set.delay = delay,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (bt_mesh_is_server_recv_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now)) {
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_XYL_SET) {
 | |
|             send_light_xyl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS);
 | |
|         }
 | |
|         send_light_xyl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS);
 | |
|         /* In this condition, no event will be callback to application layer */
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     bt_mesh_light_server_lock();
 | |
| 
 | |
|     bt_mesh_server_stop_transition(&srv->transition);
 | |
|     bt_mesh_server_update_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now);
 | |
| 
 | |
|     srv->state->target_lightness = lightness;
 | |
|     if (srv->state->x_range_min && x < srv->state->x_range_min) {
 | |
|         x = srv->state->x_range_min;
 | |
|     } else if (srv->state->x_range_max && x > srv->state->x_range_max) {
 | |
|         x = srv->state->x_range_max;
 | |
|     }
 | |
|     srv->state->target_x = x;
 | |
|     if (srv->state->y_range_min && y < srv->state->y_range_min) {
 | |
|         y = srv->state->y_range_min;
 | |
|     } else if (srv->state->y_range_max && y > srv->state->y_range_max) {
 | |
|         y = srv->state->y_range_max;
 | |
|     }
 | |
|     srv->state->target_y = y;
 | |
| 
 | |
|     /**
 | |
|      * If the target state is equal to the current state, the transition shall not
 | |
|      * be started and is considered complete.
 | |
|      */
 | |
|     if (srv->state->target_lightness != srv->state->lightness ||
 | |
|         srv->state->target_x != srv->state->x ||
 | |
|         srv->state->target_y != srv->state->y) {
 | |
|         light_xyl_tt_values(srv, trans_time, delay);
 | |
|     } else {
 | |
|         bt_mesh_light_server_state_change_t change = {
 | |
|             .xyl_set.lightness = srv->state->lightness,
 | |
|             .xyl_set.x = srv->state->x,
 | |
|             .xyl_set.y = srv->state->y,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                               model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_XYL_SET) {
 | |
|             send_light_xyl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS);
 | |
|         }
 | |
|         send_light_xyl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS);
 | |
| 
 | |
|         bt_mesh_light_server_unlock();
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Copy the ctx of the received message */
 | |
|     if (srv->transition.timer.work.user_data) {
 | |
|         memcpy(srv->transition.timer.work.user_data, ctx, sizeof(struct bt_mesh_msg_ctx));
 | |
|     }
 | |
| 
 | |
|     /* For Instantaneous Transition */
 | |
|     if (srv->transition.counter == 0U) {
 | |
|         srv->state->lightness = srv->state->target_lightness;
 | |
|         srv->state->x = srv->state->target_x;
 | |
|         srv->state->y = srv->state->target_y;
 | |
|     }
 | |
| 
 | |
|     srv->transition.just_started = true;
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_XYL_SET) {
 | |
|         send_light_xyl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS);
 | |
|     }
 | |
|     send_light_xyl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS);
 | |
| 
 | |
|     bt_mesh_light_server_unlock();
 | |
| 
 | |
|     bt_mesh_server_start_transition(&srv->transition);
 | |
| }
 | |
| 
 | |
| static void light_xyl_default_set(struct bt_mesh_model *model,
 | |
|                                   struct bt_mesh_msg_ctx *ctx,
 | |
|                                   struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_xyl_setup_srv *srv = model->user_data;
 | |
|     uint16_t lightness = 0U, x = 0U, y = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     lightness = net_buf_simple_pull_le16(buf);
 | |
|     x = net_buf_simple_pull_le16(buf);
 | |
|     y = net_buf_simple_pull_le16(buf);
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .xyl_default_set.lightness = lightness,
 | |
|             .xyl_default_set.x = x,
 | |
|             .xyl_default_set.y = y,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (srv->state->x_range_min && x < srv->state->x_range_min) {
 | |
|         x = srv->state->x_range_min;
 | |
|     } else if (srv->state->x_range_max && x > srv->state->x_range_max) {
 | |
|         x = srv->state->x_range_max;
 | |
|     }
 | |
| 
 | |
|     if (srv->state->y_range_min && y < srv->state->y_range_min) {
 | |
|         y = srv->state->y_range_min;
 | |
|     } else if (srv->state->y_range_max && y > srv->state->y_range_max) {
 | |
|         y = srv->state->y_range_max;
 | |
|     }
 | |
| 
 | |
|     srv->state->lightness_default = lightness;
 | |
|     srv->state->x_default = x;
 | |
|     srv->state->y_default = y;
 | |
| 
 | |
|     bt_mesh_light_server_state_change_t change = {
 | |
|         .xyl_default_set.lightness = srv->state->lightness_default,
 | |
|         .xyl_default_set.x = srv->state->x_default,
 | |
|         .xyl_default_set.y = srv->state->y_default,
 | |
|     };
 | |
|     bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                           model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET) {
 | |
|         send_light_xyl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS);
 | |
|     }
 | |
|     send_light_xyl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS);
 | |
| }
 | |
| 
 | |
| static void light_xyl_range_set(struct bt_mesh_model *model,
 | |
|                                 struct bt_mesh_msg_ctx *ctx,
 | |
|                                 struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_xyl_setup_srv *srv = model->user_data;
 | |
|     uint16_t x_min = 0U, x_max = 0U, y_min = 0U, y_max = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->state == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     x_min = net_buf_simple_pull_le16(buf);
 | |
|     x_max = net_buf_simple_pull_le16(buf);
 | |
|     y_min = net_buf_simple_pull_le16(buf);
 | |
|     y_max = net_buf_simple_pull_le16(buf);
 | |
| 
 | |
|     if (x_min > x_max) {
 | |
|         BT_ERR("Invalid parameter, x min 0x%04x, x max 0x%04x",
 | |
|                 x_min, x_max);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (y_min > y_max) {
 | |
|         BT_ERR("Invalid parameter, y min 0x%04x, y max 0x%04x",
 | |
|                 y_min, y_max);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .xyl_range_set.x_range_min = x_min,
 | |
|             .xyl_range_set.x_range_max = x_max,
 | |
|             .xyl_range_set.y_range_min = y_min,
 | |
|             .xyl_range_set.y_range_max = y_max,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     srv->state->status_code = BLE_MESH_RANGE_UPDATE_SUCCESS;
 | |
|     srv->state->x_range_min = x_min;
 | |
|     srv->state->x_range_max = x_max;
 | |
|     srv->state->y_range_min = y_min;
 | |
|     srv->state->y_range_max = y_max;
 | |
| 
 | |
|     bt_mesh_light_server_state_change_t change = {
 | |
|         .xyl_range_set.x_range_min = srv->state->x_range_min,
 | |
|         .xyl_range_set.x_range_max = srv->state->x_range_max,
 | |
|         .xyl_range_set.y_range_min = srv->state->y_range_min,
 | |
|         .xyl_range_set.y_range_max = srv->state->y_range_max,
 | |
|     };
 | |
|     bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                           model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET) {
 | |
|         send_light_xyl_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS);
 | |
|     }
 | |
|     send_light_xyl_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS);
 | |
| }
 | |
| 
 | |
| /* Light LC Server/Setup Server message handlers */
 | |
| static void send_light_lc_status(struct bt_mesh_model *model,
 | |
|                                  struct bt_mesh_msg_ctx *ctx,
 | |
|                                  bool publish, uint16_t opcode)
 | |
| {
 | |
|     struct bt_mesh_light_lc_srv *srv = model->user_data;
 | |
|     struct net_buf_simple *msg = NULL;
 | |
|     uint8_t length = 2 + 3;
 | |
| 
 | |
|     if (ctx == NULL && publish == false) {
 | |
|         BT_ERR("%s, Invalid parameter", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (publish == false) {
 | |
|         msg = bt_mesh_alloc_buf(length + BLE_MESH_SERVER_TRANS_MIC_SIZE);
 | |
|         if (msg == NULL) {
 | |
|             BT_ERR("%s, Out of memory", __func__);
 | |
|             return;
 | |
|         }
 | |
|     } else {
 | |
|         msg = bt_mesh_server_get_pub_msg(model, length);
 | |
|         if (msg == NULL) {
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     bt_mesh_model_msg_init(msg, opcode);
 | |
|     switch (opcode) {
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS:
 | |
|         net_buf_simple_add_u8(msg, srv->lc->state.mode);
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LC_OM_STATUS:
 | |
|         net_buf_simple_add_u8(msg, srv->lc->state.occupancy_mode);
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS:
 | |
|         net_buf_simple_add_u8(msg, srv->lc->state.light_onoff);
 | |
|         if (srv->transition.counter) {
 | |
|             bt_mesh_server_calc_remain_time(&srv->transition);
 | |
|             net_buf_simple_add_u8(msg, srv->lc->state.target_light_onoff);
 | |
|             net_buf_simple_add_u8(msg, srv->transition.remain_time);
 | |
|         }
 | |
|         break;
 | |
|     default:
 | |
|         BT_WARN("Unknown Light LC status opcode 0x%04x", opcode);
 | |
|         if (publish == false) {
 | |
|             bt_mesh_free_buf(msg);
 | |
|         }
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (publish == false) {
 | |
|         BLE_MESH_CHECK_SEND_STATUS(bt_mesh_model_send(model, ctx, msg, NULL, NULL));
 | |
|         bt_mesh_free_buf(msg);
 | |
|     } else {
 | |
|         BLE_MESH_CHECK_SEND_STATUS(bt_mesh_model_publish(model));
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void light_lc_get(struct bt_mesh_model *model,
 | |
|                          struct bt_mesh_msg_ctx *ctx,
 | |
|                          struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_lc_srv *srv = model->user_data;
 | |
|     uint16_t opcode = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->lc == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.get_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_GET_MSG,
 | |
|                                               model, ctx, NULL, 0);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     switch (ctx->recv_op) {
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LC_MODE_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LC_OM_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_LC_OM_STATUS;
 | |
|         break;
 | |
|     case BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_GET:
 | |
|         opcode = BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS;
 | |
|         break;
 | |
|     default:
 | |
|         BT_WARN("Unknown Light LC Get opcode 0x%04x", ctx->recv_op);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     send_light_lc_status(model, ctx, false, opcode);
 | |
| }
 | |
| 
 | |
| void light_lc_publish(struct bt_mesh_model *model, uint16_t opcode)
 | |
| {
 | |
|     struct bt_mesh_light_lc_srv *srv = model->user_data;
 | |
| 
 | |
|     if (srv == NULL || srv->lc == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     send_light_lc_status(model, NULL, true, opcode);
 | |
| }
 | |
| 
 | |
| static void light_lc_mode_set(struct bt_mesh_model *model,
 | |
|                               struct bt_mesh_msg_ctx *ctx,
 | |
|                               struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_lc_srv *srv = model->user_data;
 | |
|     uint8_t mode = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->lc == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     mode = net_buf_simple_pull_u8(buf);
 | |
|     if (mode > BLE_MESH_STATE_ON) {
 | |
|         BT_ERR("Invalid LC Mode 0x%02x", mode);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .lc_mode_set.mode = mode,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     srv->lc->state.mode = mode;
 | |
| 
 | |
|     bt_mesh_light_server_state_change_t change = {
 | |
|         .lc_mode_set.mode = srv->lc->state.mode,
 | |
|     };
 | |
|     bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                           model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_LC_MODE_SET) {
 | |
|         send_light_lc_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS);
 | |
|     }
 | |
|     send_light_lc_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS);
 | |
| }
 | |
| 
 | |
| static void light_lc_om_set(struct bt_mesh_model *model,
 | |
|                             struct bt_mesh_msg_ctx *ctx,
 | |
|                             struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_lc_srv *srv = model->user_data;
 | |
|     uint8_t om = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->lc == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     om = net_buf_simple_pull_u8(buf);
 | |
|     if (om > BLE_MESH_STATE_ON) {
 | |
|         BT_ERR("Invalid LC Occupancy Mode 0x%02x", om);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .lc_om_set.mode = om,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     srv->lc->state.occupancy_mode = om;
 | |
| 
 | |
|     bt_mesh_light_server_state_change_t change = {
 | |
|         .lc_om_set.mode = srv->lc->state.occupancy_mode,
 | |
|     };
 | |
|     bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                           model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_LC_OM_SET) {
 | |
|         send_light_lc_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_LC_OM_STATUS);
 | |
|     }
 | |
|     send_light_lc_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_LC_OM_STATUS);
 | |
| }
 | |
| 
 | |
| static void light_lc_light_onoff_set(struct bt_mesh_model *model,
 | |
|                                      struct bt_mesh_msg_ctx *ctx,
 | |
|                                      struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_lc_srv *srv = model->user_data;
 | |
|     uint8_t tid = 0U, trans_time = 0U, delay = 0U;
 | |
|     bool optional = false;
 | |
|     uint8_t onoff = 0U;
 | |
|     int64_t now = 0;
 | |
| 
 | |
|     if (srv == NULL || srv->lc == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     onoff = net_buf_simple_pull_u8(buf);
 | |
|     tid = net_buf_simple_pull_u8(buf);
 | |
| 
 | |
|     if (bt_mesh_server_get_optional(model, ctx, buf, &trans_time, &delay, &optional)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .lc_light_onoff_set.op_en = optional,
 | |
|             .lc_light_onoff_set.light_onoff = onoff,
 | |
|             .lc_light_onoff_set.tid = tid,
 | |
|             .lc_light_onoff_set.trans_time = trans_time,
 | |
|             .lc_light_onoff_set.delay = delay,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (bt_mesh_is_server_recv_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now)) {
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET) {
 | |
|             send_light_lc_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS);
 | |
|         }
 | |
|         send_light_lc_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS);
 | |
|         /* In this condition, no event will be callback to application layer */
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     bt_mesh_light_server_lock();
 | |
| 
 | |
|     bt_mesh_server_stop_transition(&srv->transition);
 | |
|     bt_mesh_server_update_last_msg(&srv->last, tid, ctx->addr, ctx->recv_dst, &now);
 | |
| 
 | |
|     srv->lc->state.target_light_onoff = onoff;
 | |
| 
 | |
|     if (srv->lc->state.target_light_onoff != srv->lc->state.light_onoff) {
 | |
|         light_lc_tt_values(srv, trans_time, delay);
 | |
|     } else {
 | |
|         bt_mesh_light_server_state_change_t change = {
 | |
|             .lc_light_onoff_set.onoff = srv->lc->state.light_onoff,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                               model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|         if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET) {
 | |
|             send_light_lc_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS);
 | |
|         }
 | |
|         send_light_lc_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS);
 | |
| 
 | |
|         bt_mesh_light_server_unlock();
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Copy the ctx of the received message */
 | |
|     if (srv->transition.timer.work.user_data) {
 | |
|         memcpy(srv->transition.timer.work.user_data, ctx, sizeof(struct bt_mesh_msg_ctx));
 | |
|     }
 | |
| 
 | |
|     /* For Instantaneous Transition */
 | |
|     if (srv->transition.counter == 0U) {
 | |
|         srv->lc->state.light_onoff = srv->lc->state.target_light_onoff;
 | |
|     }
 | |
| 
 | |
|     srv->transition.just_started = true;
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET) {
 | |
|         send_light_lc_status(model, ctx, false, BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS);
 | |
|     }
 | |
|     send_light_lc_status(model, NULL, true, BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS);
 | |
| 
 | |
|     bt_mesh_light_server_unlock();
 | |
| 
 | |
|     bt_mesh_server_start_transition(&srv->transition);
 | |
| }
 | |
| 
 | |
| static void light_lc_sensor_status(struct bt_mesh_model *model,
 | |
|                                    struct bt_mesh_msg_ctx *ctx,
 | |
|                                    struct net_buf_simple *buf)
 | |
| {
 | |
|     /**
 | |
|      * When a Light LC Server receives a Sensor Status message, and if the message
 | |
|      * Raw field contains a Raw Value for the Motion Sensed Property, and the value
 | |
|      * is greater than 0, or a Raw Value for the People Count Property, and the
 | |
|      * value is greater than 0, or a Raw Value for the Presence Detected Property,
 | |
|      * and the value is greater than 0, then it shall set the Light LC Occupancy
 | |
|      * state to 0b1.
 | |
|      * If the message Raw field contains a Raw Value for the Time Since Motion Sensed
 | |
|      * device property, which represents a value less than or equal to the value of
 | |
|      * the Light LC Occupancy Delay state, it shall delay setting the Light LC Occupancy
 | |
|      * state to 0b1 by the difference between the value of the Light LC Occupancy Delay
 | |
|      * state and the received Time Since Motion value.
 | |
|      * When a Light LC Server receives a Sensor Status message, and if the message Raw
 | |
|      * field contains a Raw Value for the Present Ambient Light Level device property,
 | |
|      * it shall set the Light LC Ambient LuxLevel state to the Represented Value of the
 | |
|      * received Present Ambient Light Level.
 | |
|      *
 | |
|      * Motion Sensed: 1 octet, 0x0042
 | |
|      * People Count: 2 octets, 0x004C
 | |
|      * Presence Detected: 1 octet, 0x004D
 | |
|      *
 | |
|      * Time Since Motion Sensed: 2 octets, 0x0068
 | |
|      *
 | |
|      * Present Ambient Light Level: 4 octets, 0x004E
 | |
|      */
 | |
|     struct bt_mesh_light_lc_srv *srv = model->user_data;
 | |
|     bt_mesh_light_server_state_change_t change = {0};
 | |
|     uint16_t mpid = 0U, prop_id = 0U;
 | |
|     uint8_t length = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->lc == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (srv->rsp_ctrl.status_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_status_msg_t status = {
 | |
|             .sensor_status.data = buf,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_STATUS_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&status, sizeof(status));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     mpid = net_buf_simple_pull_le16(buf);
 | |
|     if (mpid & BIT(0)) {
 | |
|         length = (uint8_t)((mpid & 0xff) >> 1);
 | |
|         uint8_t msb = net_buf_simple_pull_u8(buf);
 | |
|         prop_id = (uint16_t)(msb << 8) | (uint16_t)(mpid >> 8);
 | |
|     } else {
 | |
|         length = (uint8_t)((mpid & 0x1f) >> 1);
 | |
|         prop_id = (uint16_t)(mpid >> 5);
 | |
|     }
 | |
| 
 | |
|     change.sensor_status.property_id = prop_id;
 | |
| 
 | |
|     switch (prop_id) {
 | |
|     case BLE_MESH_MOTION_SENSED: {
 | |
|         if (length != BLE_MESH_MOTION_SENSED_LEN || length != buf->len) {
 | |
|             BT_WARN("Invalid Motion Sensed Property length %d", length);
 | |
|             return;
 | |
|         }
 | |
|         uint8_t val = net_buf_simple_pull_u8(buf);
 | |
|         if (val > 0) {
 | |
|             srv->lc->state.occupancy = BLE_MESH_STATE_ON;
 | |
| 
 | |
|             change.sensor_status.state.occupancy = srv->lc->state.occupancy;
 | |
|             bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                                   model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_PEOPLE_COUNT: {
 | |
|         if (length != BLE_MESH_PEOPLE_COUNT_LEN || length != buf->len) {
 | |
|             BT_WARN("Invalid Motion Sensed Property length %d", length);
 | |
|             return;
 | |
|         }
 | |
|         uint16_t val = net_buf_simple_pull_le16(buf);
 | |
|         if (val > 0) {
 | |
|             srv->lc->state.occupancy = BLE_MESH_STATE_ON;
 | |
| 
 | |
|             change.sensor_status.state.occupancy = srv->lc->state.occupancy;
 | |
|             bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                                   model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_PRESENCE_DETECTED: {
 | |
|         if (length != BLE_MESH_PRESENCE_DETECTED_LEN || length != buf->len) {
 | |
|             BT_WARN("Invalid Motion Sensed Property length %d", length);
 | |
|             return;
 | |
|         }
 | |
|         uint8_t val = net_buf_simple_pull_u8(buf);
 | |
|         if (val > 0) {
 | |
|             srv->lc->state.occupancy = BLE_MESH_STATE_ON;
 | |
| 
 | |
|             change.sensor_status.state.occupancy = srv->lc->state.occupancy;
 | |
|             bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                                   model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_TIME_SINCE_MOTION_SENSED: {
 | |
|         if (length != BLE_MESH_TIME_SINCE_MOTION_SENSED_LEN || length != buf->len) {
 | |
|             BT_WARN("Invalid Motion Sensed Property length %d", length);
 | |
|             return;
 | |
|         }
 | |
|         uint16_t val = net_buf_simple_pull_le16(buf);
 | |
|         if (val <= srv->lc->prop_state.time_occupancy_delay) {
 | |
|             srv->lc->prop_state.set_occupancy_to_1_delay =
 | |
|                 srv->lc->prop_state.time_occupancy_delay - val;
 | |
| 
 | |
|             change.sensor_status.state.set_occupancy_to_1_delay = srv->lc->prop_state.set_occupancy_to_1_delay;
 | |
|             bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                                   model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_PRESENT_AMBIENT_LIGHT_LEVEL: {
 | |
|         /**
 | |
|          * Present Ambient Light Level device property is 4 octets, but ambient
 | |
|          * luxlevel length is 3 octets, and other devices may send Sensor Status
 | |
|          * which only contains 3 octets just for Light LC Server.
 | |
|          * Here we just check if the length is larger than 3.
 | |
|          */
 | |
|         if (buf->len < 3) {
 | |
|             BT_WARN("Invalid Motion Sensed Property length %d", buf->len);
 | |
|             return;
 | |
|         }
 | |
|         uint16_t lsb = net_buf_simple_pull_le16(buf);
 | |
|         uint8_t msb = net_buf_simple_pull_u8(buf);
 | |
|         srv->lc->state.ambient_luxlevel = (msb << 16) | lsb;
 | |
| 
 | |
|         change.sensor_status.state.ambient_luxlevel = srv->lc->state.ambient_luxlevel;
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                               model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
|         break;
 | |
|     }
 | |
|     default:
 | |
|         break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static uint8_t *get_light_lc_prop_val(struct bt_mesh_model *model, uint16_t prop_id)
 | |
| {
 | |
|     struct bt_mesh_light_lc_setup_srv *srv = model->user_data;
 | |
|     uint8_t *val = NULL;
 | |
| 
 | |
|     switch (prop_id) {
 | |
|     case BLE_MESH_LIGHT_CONTROL_TIME_OCCUPANCY_DELAY:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.time_occupancy_delay;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_TIME_FADE_ON:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.time_fade_on;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_TIME_RUN_ON:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.time_run_on;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_TIME_FADE:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.time_fade;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_TIME_PROLONG:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.time_prolong;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_TIME_FADE_STANDBY_AUTO:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.time_fade_standby_auto;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_TIME_FADE_STANDBY_MANUAL:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.time_fade_standby_manual;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_LIGHTNESS_ON:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.lightness_on;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_LIGHTNESS_PROLONG:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.lightness_prolong;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_LIGHTNESS_STANDBY:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.lightness_standby;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_AMBIENT_LUXLEVEL_ON:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.ambient_luxlevel_on;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_AMBIENT_LUXLEVEL_PROLONG:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.ambient_luxlevel_prolong;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_AMBIENT_LUXLEVEL_STANDBY:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.ambient_luxlevel_standby;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_REGULATOR_KIU:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.regulator_kiu;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_REGULATOR_KID:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.regulator_kid;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_REGULATOR_KPU:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.regulator_kpu;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_REGULATOR_KPD:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.regulator_kpd;
 | |
|         break;
 | |
|     case BLE_MESH_LIGHT_CONTROL_REGULATOR_ACCURACY:
 | |
|         val = (uint8_t *)&srv->lc->prop_state.regulator_accuracy;
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|     return val;
 | |
| }
 | |
| 
 | |
| uint8_t *bt_mesh_get_lc_prop_value(struct bt_mesh_model *model, uint16_t prop_id)
 | |
| {
 | |
|     if (model == NULL) {
 | |
|         BT_ERR("%s, Invalid parameter", __func__);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     return get_light_lc_prop_val(model, prop_id);
 | |
| }
 | |
| 
 | |
| static void send_light_lc_prop_status(struct bt_mesh_model *model,
 | |
|                                       struct bt_mesh_msg_ctx *ctx,
 | |
|                                       uint16_t prop_id, bool publish)
 | |
| {
 | |
|     struct net_buf_simple *msg = NULL;
 | |
|     uint8_t length = 1 + 2 + 4;
 | |
|     uint8_t *prop_val = NULL;
 | |
| 
 | |
|     prop_val = get_light_lc_prop_val(model, prop_id);
 | |
|     if (prop_val == NULL) {
 | |
|         BT_ERR("Failed to get Light LC Property value");
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (publish == false) {
 | |
|         msg = bt_mesh_alloc_buf(length + BLE_MESH_SERVER_TRANS_MIC_SIZE);
 | |
|         if (msg == NULL) {
 | |
|             BT_ERR("%s, Out of memory", __func__);
 | |
|             return;
 | |
|         }
 | |
|     } else {
 | |
|         msg = bt_mesh_server_get_pub_msg(model, length);
 | |
|         if (msg == NULL) {
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     bt_mesh_model_msg_init(msg, BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS);
 | |
|     net_buf_simple_add_le16(msg, prop_id);
 | |
|     net_buf_simple_add_mem(msg, prop_val, bt_mesh_get_dev_prop_len(prop_id));
 | |
| 
 | |
|     if (publish == false) {
 | |
|         BLE_MESH_CHECK_SEND_STATUS(bt_mesh_model_send(model, ctx, msg, NULL, NULL));
 | |
|         bt_mesh_free_buf(msg);
 | |
|     } else {
 | |
|         BLE_MESH_CHECK_SEND_STATUS(bt_mesh_model_publish(model));
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void light_lc_prop_get(struct bt_mesh_model *model,
 | |
|                               struct bt_mesh_msg_ctx *ctx,
 | |
|                               struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_lc_setup_srv *srv = model->user_data;
 | |
|     uint16_t prop_id = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->lc == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     prop_id = net_buf_simple_pull_le16(buf);
 | |
|     if (prop_id < 0x002B || prop_id > 0x003C) {
 | |
|         BT_ERR("Invalid Light LC Property ID 0x%04x", prop_id);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Callback the received message to the application layer */
 | |
|     if (srv->rsp_ctrl.get_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_get_msg_t get = {
 | |
|             .lc_property_get.id = net_buf_simple_pull_le16(buf),
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_GET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&get, sizeof(get));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     send_light_lc_prop_status(model, ctx, prop_id, false);
 | |
| }
 | |
| 
 | |
| static void light_lc_prop_set(struct bt_mesh_model *model,
 | |
|                               struct bt_mesh_msg_ctx *ctx,
 | |
|                               struct net_buf_simple *buf)
 | |
| {
 | |
|     struct bt_mesh_light_lc_setup_srv *srv = model->user_data;
 | |
|     uint8_t *prop_val = NULL, expect_len = 0U;
 | |
|     uint16_t prop_id = 0U;
 | |
| 
 | |
|     if (srv == NULL || srv->lc == NULL) {
 | |
|         BT_ERR("%s, Invalid model user data", __func__);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     prop_id = net_buf_simple_pull_le16(buf);
 | |
|     if (prop_id < 0x002B || prop_id > 0x003C) {
 | |
|         BT_ERR("Invalid Light LC Property ID 0x%04x", prop_id);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_RSP_BY_APP) {
 | |
|         bt_mesh_light_server_recv_set_msg_t set = {
 | |
|             .lc_property_set.id = net_buf_simple_pull_le16(buf),
 | |
|             .lc_property_set.value = buf,
 | |
|         };
 | |
|         bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_RECV_SET_MSG,
 | |
|                                               model, ctx, (const uint8_t *)&set, sizeof(set));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     expect_len = bt_mesh_get_dev_prop_len(prop_id);
 | |
|     if (buf->len != expect_len) {
 | |
|         BT_ERR("Invalid Light LC Property 0x%04x length, expect %d, actual %d",
 | |
|                 prop_id, expect_len, buf->len);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     prop_val = get_light_lc_prop_val(model, prop_id);
 | |
|     if (prop_val == NULL) {
 | |
|         BT_ERR("Failed to get Light LC Property value");
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     memcpy(prop_val, buf->data, buf->len);
 | |
| 
 | |
|     bt_mesh_light_server_state_change_t change = {
 | |
|         .lc_property_set.id = prop_id,
 | |
|         .lc_property_set.value = buf,
 | |
|     };
 | |
|     bt_mesh_lighting_server_cb_evt_to_btc(BTC_BLE_MESH_EVT_LIGHTING_SERVER_STATE_CHANGE,
 | |
|                                           model, ctx, (const uint8_t *)&change, sizeof(change));
 | |
| 
 | |
|     if (ctx->recv_op == BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET) {
 | |
|         send_light_lc_prop_status(model, ctx, prop_id, false);
 | |
|     }
 | |
|     send_light_lc_prop_status(model, ctx, prop_id, true);
 | |
| }
 | |
| 
 | |
| /* message handlers (End) */
 | |
| 
 | |
| /* Mapping of message handlers for Light Lightness Server (0x1300) */
 | |
| const struct bt_mesh_model_op bt_mesh_light_lightness_srv_op[] = {
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_GET,              0, light_lightness_get        },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET,              3, light_lightness_set        },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET_UNACK,        3, light_lightness_set        },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_GET,       0, light_lightness_get        },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET,       3, light_lightness_linear_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET_UNACK, 3, light_lightness_linear_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_GET,         0, light_lightness_get        },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_GET,      0, light_lightness_get        },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_GET,        0, light_lightness_get        },
 | |
|     BLE_MESH_MODEL_OP_END,
 | |
| };
 | |
| 
 | |
| /* Mapping of message handlers for Light Lightness Setup Server (0x1301) */
 | |
| const struct bt_mesh_model_op bt_mesh_light_lightness_setup_srv_op[] = {
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET,       2, light_lightness_default_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET_UNACK, 2, light_lightness_default_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET,         4, light_lightness_range_set   },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET_UNACK,   4, light_lightness_range_set   },
 | |
|     BLE_MESH_MODEL_OP_END,
 | |
| };
 | |
| 
 | |
| /* Mapping of message handlers for Light CTL Server (0x1303) */
 | |
| const struct bt_mesh_model_op bt_mesh_light_ctl_srv_op[] = {
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_CTL_GET,                   0, light_ctl_get },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_CTL_SET,                   7, light_ctl_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_CTL_SET_UNACK,             7, light_ctl_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_GET, 0, light_ctl_get },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_GET,           0, light_ctl_get },
 | |
|     BLE_MESH_MODEL_OP_END,
 | |
| };
 | |
| 
 | |
| /* Mapping of message handlers for Light CTL Setup Server (0x1304) */
 | |
| const struct bt_mesh_model_op bt_mesh_light_ctl_setup_srv_op[] = {
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET,                 6, light_ctl_default_set    },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET_UNACK,           6, light_ctl_default_set    },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET,       4, light_ctl_temp_range_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET_UNACK, 4, light_ctl_temp_range_set },
 | |
|     BLE_MESH_MODEL_OP_END,
 | |
| };
 | |
| 
 | |
| /* Mapping of message handlers for Light CTL Temperature Server (0x1306) */
 | |
| const struct bt_mesh_model_op bt_mesh_light_ctl_temp_srv_op[] = {
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_GET,       0, light_ctl_get      },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET,       5, light_ctl_temp_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET_UNACK, 5, light_ctl_temp_set },
 | |
|     BLE_MESH_MODEL_OP_END,
 | |
| };
 | |
| 
 | |
| /* Mapping of message handlers for Light HSL Server (0x1307) */
 | |
| const struct bt_mesh_model_op bt_mesh_light_hsl_srv_op[] = {
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_GET,         0, light_hsl_get },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_SET,         7, light_hsl_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_SET_UNACK,   7, light_hsl_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_TARGET_GET,  0, light_hsl_get },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_GET, 0, light_hsl_get },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_GET,   0, light_hsl_get },
 | |
|     BLE_MESH_MODEL_OP_END,
 | |
| };
 | |
| 
 | |
| /* Mapping of message handlers for Light HSL Setup Server (0x1308) */
 | |
| const struct bt_mesh_model_op bt_mesh_light_hsl_setup_srv_op[] = {
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET,       6, light_hsl_default_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET_UNACK, 6, light_hsl_default_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET,         8, light_hsl_range_set   },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET_UNACK,   8, light_hsl_range_set   },
 | |
|     BLE_MESH_MODEL_OP_END,
 | |
| };
 | |
| 
 | |
| /* Mapping of message handlers for Light HSL Hue Server (0x130A) */
 | |
| const struct bt_mesh_model_op bt_mesh_light_hsl_hue_srv_op[] = {
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_GET,       0, light_hsl_get     },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET,       3, light_hsl_hue_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET_UNACK, 3, light_hsl_hue_set },
 | |
|     BLE_MESH_MODEL_OP_END,
 | |
| };
 | |
| 
 | |
| /* Mapping of message handlers for Light HSL Saturation Server (0x130B) */
 | |
| const struct bt_mesh_model_op bt_mesh_light_hsl_sat_srv_op[] = {
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_GET,       0, light_hsl_get     },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET,       3, light_hsl_sat_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET_UNACK, 3, light_hsl_sat_set },
 | |
|     BLE_MESH_MODEL_OP_END,
 | |
| };
 | |
| 
 | |
| /* Mapping of message handlers for Light xyL Server (0x130C) */
 | |
| const struct bt_mesh_model_op bt_mesh_light_xyl_srv_op[] = {
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_XYL_GET,         0, light_xyl_get },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_XYL_SET,         7, light_xyl_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_XYL_SET_UNACK,   7, light_xyl_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_XYL_TARGET_GET,  0, light_xyl_get },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_GET, 0, light_xyl_get },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_GET,   0, light_xyl_get },
 | |
|     BLE_MESH_MODEL_OP_END,
 | |
| };
 | |
| 
 | |
| /* Mapping of message handlers for Light xyL Setup Server (0x130D) */
 | |
| const struct bt_mesh_model_op bt_mesh_light_xyl_setup_srv_op[] = {
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET,       6, light_xyl_default_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET_UNACK, 6, light_xyl_default_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET,         8, light_xyl_range_set   },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET_UNACK,   8, light_xyl_range_set   },
 | |
|     BLE_MESH_MODEL_OP_END,
 | |
| };
 | |
| 
 | |
| /* Mapping of message handlers for Light LC Server (0x130F) */
 | |
| const struct bt_mesh_model_op bt_mesh_light_lc_srv_op[] = {
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LC_MODE_GET,              0, light_lc_get             },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LC_MODE_SET,              1, light_lc_mode_set        },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LC_MODE_SET_UNACK,        1, light_lc_mode_set        },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LC_OM_GET,                0, light_lc_get             },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LC_OM_SET,                1, light_lc_om_set          },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LC_OM_SET_UNACK,          1, light_lc_om_set          },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_GET,       0, light_lc_get             },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET,       2, light_lc_light_onoff_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET_UNACK, 2, light_lc_light_onoff_set },
 | |
|     { BLE_MESH_MODEL_OP_SENSOR_STATUS,                  3, light_lc_sensor_status   },
 | |
|     BLE_MESH_MODEL_OP_END,
 | |
| };
 | |
| 
 | |
| /* Mapping of message handlers for Light LC Setup Server (0x1310) */
 | |
| const struct bt_mesh_model_op bt_mesh_light_lc_setup_srv_op[] = {
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET,       2, light_lc_prop_get },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET,       3, light_lc_prop_set },
 | |
|     { BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET_UNACK, 3, light_lc_prop_set },
 | |
|     BLE_MESH_MODEL_OP_END,
 | |
| };
 | |
| 
 | |
| static int light_server_init(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->user_data == NULL) {
 | |
|         BT_ERR("Invalid Lighting Server user data, model id 0x%04x", model->id);
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     switch (model->id) {
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_SRV: {
 | |
|         struct bt_mesh_light_lightness_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light Lightness State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_alloc_ctx(&srv->actual_transition.timer.work);
 | |
|             bt_mesh_server_alloc_ctx(&srv->linear_transition.timer.work);
 | |
|             k_delayed_work_init(&srv->actual_transition.timer, light_lightness_actual_work_handler);
 | |
|             k_delayed_work_init(&srv->linear_transition.timer, light_lightness_linear_work_handler);
 | |
|         }
 | |
|         srv->model = model;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_SETUP_SRV: {
 | |
|         struct bt_mesh_light_lightness_setup_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light Lightness State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         srv->model = model;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_CTL_SRV: {
 | |
|         struct bt_mesh_light_ctl_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light CTL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_alloc_ctx(&srv->transition.timer.work);
 | |
|             k_delayed_work_init(&srv->transition.timer, light_ctl_work_handler);
 | |
|         }
 | |
|         srv->model = model;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_CTL_SETUP_SRV: {
 | |
|         struct bt_mesh_light_ctl_setup_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light CTL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         srv->model = model;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_CTL_TEMP_SRV: {
 | |
|         struct bt_mesh_light_ctl_temp_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light CTL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_alloc_ctx(&srv->transition.timer.work);
 | |
|             k_delayed_work_init(&srv->transition.timer, light_ctl_temp_work_handler);
 | |
|         }
 | |
|         srv->model = model;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_SRV: {
 | |
|         struct bt_mesh_light_hsl_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light HSL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_alloc_ctx(&srv->transition.timer.work);
 | |
|             k_delayed_work_init(&srv->transition.timer, light_hsl_work_handler);
 | |
|         }
 | |
|         srv->model = model;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_SETUP_SRV: {
 | |
|         struct bt_mesh_light_hsl_setup_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light HSL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         srv->model = model;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_HUE_SRV: {
 | |
|         struct bt_mesh_light_hsl_hue_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light HSL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_alloc_ctx(&srv->transition.timer.work);
 | |
|             k_delayed_work_init(&srv->transition.timer, light_hsl_hue_work_handler);
 | |
|         }
 | |
|         srv->model = model;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_SAT_SRV: {
 | |
|         struct bt_mesh_light_hsl_sat_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light HSL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_alloc_ctx(&srv->transition.timer.work);
 | |
|             k_delayed_work_init(&srv->transition.timer, light_hsl_sat_work_handler);
 | |
|         }
 | |
|         srv->model = model;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_XYL_SRV: {
 | |
|         struct bt_mesh_light_xyl_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light xyL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_alloc_ctx(&srv->transition.timer.work);
 | |
|             k_delayed_work_init(&srv->transition.timer, light_xyl_work_handler);
 | |
|         }
 | |
|         srv->model = model;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_XYL_SETUP_SRV: {
 | |
|         struct bt_mesh_light_xyl_setup_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light xyL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         srv->model = model;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_LC_SRV: {
 | |
|         struct bt_mesh_light_lc_srv *srv = model->user_data;
 | |
|         if (srv->lc == NULL) {
 | |
|             BT_ERR("Invalid Light LC State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_alloc_ctx(&srv->transition.timer.work);
 | |
|             k_delayed_work_init(&srv->transition.timer, light_lc_work_handler);
 | |
|         }
 | |
|         srv->model = model;
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_LC_SETUP_SRV: {
 | |
|         struct bt_mesh_light_lc_setup_srv *srv = model->user_data;
 | |
|         if (srv->lc == NULL) {
 | |
|             BT_ERR("Invalid Light LC State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         srv->model = model;
 | |
|         break;
 | |
|     }
 | |
|     default:
 | |
|         BT_WARN("Unknown Light Server, model id 0x%04x", model->id);
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     bt_mesh_mutex_create(&light_server_lock);
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int light_lightness_srv_init(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light Lightness Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     /* When this model is present on an Element, the corresponding Light Lightness
 | |
|      * Setup Server model shall also be present.
 | |
|      */
 | |
|     struct bt_mesh_elem *element = bt_mesh_model_elem(model);
 | |
|     if (bt_mesh_model_find(element, BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_SETUP_SRV) == NULL) {
 | |
|         BT_WARN("Light Lightness Setup Server not present");
 | |
|         /* Just give a warning here, continue with the initialization */
 | |
|     }
 | |
|     return light_server_init(model);
 | |
| }
 | |
| 
 | |
| static int light_lightness_setup_srv_init(struct bt_mesh_model *model)
 | |
| {
 | |
|     return light_server_init(model);
 | |
| }
 | |
| 
 | |
| static int light_ctl_srv_init(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light CTL Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * When this model is present on an Element, the corresponding Light CTL
 | |
|      * Temperature Server model and the corresponding Light CTL Setup Server
 | |
|      * model shall also be present.
 | |
|      * The model requires two elements: the main element and the Temperature
 | |
|      * element. The Temperature element contains the corresponding Light CTL
 | |
|      * Temperature Server model.
 | |
|      */
 | |
|     struct bt_mesh_elem *element = bt_mesh_model_elem(model);
 | |
|     if (bt_mesh_model_find(element, BLE_MESH_MODEL_ID_LIGHT_CTL_SETUP_SRV) == NULL) {
 | |
|         BT_WARN("Light CTL Setup Server not present");
 | |
|         /* Just give a warning here, continue with the initialization */
 | |
|     }
 | |
|     if (bt_mesh_elem_count() < 2) {
 | |
|         BT_WARN("Light CTL Server requires two elements");
 | |
|         /* Just give a warning here, continue with the initialization */
 | |
|     }
 | |
|     return light_server_init(model);
 | |
| }
 | |
| 
 | |
| static int light_ctl_setup_srv_init(struct bt_mesh_model *model)
 | |
| {
 | |
|     return light_server_init(model);
 | |
| }
 | |
| 
 | |
| static int light_ctl_temp_srv_init(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light CTL Temperature Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     return light_server_init(model);
 | |
| }
 | |
| 
 | |
| static int light_hsl_srv_init(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light HSL Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * When this model is present on an Element, the corresponding Light HSL Hue
 | |
|      * Server model and the corresponding Light HSL Saturation Server model and
 | |
|      * the corresponding Light HSL Setup Server model shall also be present.
 | |
|      * The model requires three elements: the main element and the Hue element
 | |
|      * and the Saturation element. The Hue element contains the corresponding
 | |
|      * Light HSL Hue Server model, and the Saturation element contains the
 | |
|      * corresponding Light HSL Saturation Server model.
 | |
|      */
 | |
|     struct bt_mesh_elem *element = bt_mesh_model_elem(model);
 | |
|     if (bt_mesh_model_find(element, BLE_MESH_MODEL_ID_LIGHT_HSL_SETUP_SRV) == NULL) {
 | |
|         BT_WARN("Light HSL Setup Server not present");
 | |
|         /* Just give a warning here, continue with the initialization */
 | |
|     }
 | |
|     if (bt_mesh_elem_count() < 3) {
 | |
|         BT_WARN("Light HSL Server requires three elements");
 | |
|         /* Just give a warning here, continue with the initialization */
 | |
|     }
 | |
|     return light_server_init(model);
 | |
| }
 | |
| 
 | |
| static int light_hsl_setup_srv_init(struct bt_mesh_model *model)
 | |
| {
 | |
|     return light_server_init(model);
 | |
| }
 | |
| 
 | |
| static int light_hsl_hue_srv_init(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light HSL Hue Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     return light_server_init(model);
 | |
| }
 | |
| 
 | |
| static int light_hsl_sat_srv_init(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light HSL Saturation Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     return light_server_init(model);
 | |
| }
 | |
| 
 | |
| static int light_xyl_srv_init(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light xyL Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * When this model is present on an Element, the corresponding Light xyL
 | |
|      * Setup Server model shall also be present.
 | |
|      */
 | |
|     struct bt_mesh_elem *element = bt_mesh_model_elem(model);
 | |
|     if (bt_mesh_model_find(element, BLE_MESH_MODEL_ID_LIGHT_XYL_SETUP_SRV) == NULL) {
 | |
|         BT_WARN("Light xyL Setup Server not present");
 | |
|         /* Just give a warning here, continue with the initialization */
 | |
|     }
 | |
|     return light_server_init(model);
 | |
| }
 | |
| 
 | |
| static int light_xyl_setup_srv_init(struct bt_mesh_model *model)
 | |
| {
 | |
|     return light_server_init(model);
 | |
| }
 | |
| 
 | |
| static int light_lc_srv_init(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light LC Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     return light_server_init(model);
 | |
| }
 | |
| 
 | |
| static int light_lc_setup_srv_init(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light LC Setup Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * When this model is present on an Element, the corresponding Light LC
 | |
|      * Setup Server model shall also be present.
 | |
|      */
 | |
|     struct bt_mesh_elem *element = bt_mesh_model_elem(model);
 | |
|     if (bt_mesh_model_find(element, BLE_MESH_MODEL_ID_LIGHT_LC_SETUP_SRV) == NULL) {
 | |
|         BT_WARN("Light LC Setup Server not present");
 | |
|         /* Just give a warning here, continue with the initialization */
 | |
|     }
 | |
|     return light_server_init(model);
 | |
| }
 | |
| 
 | |
| #if CONFIG_BLE_MESH_DEINIT
 | |
| static int light_server_deinit(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->user_data == NULL) {
 | |
|         BT_ERR("Invalid Lighting Server user data, model id 0x%04x", model->id);
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     switch (model->id) {
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_SRV: {
 | |
|         struct bt_mesh_light_lightness_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light Lightness State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_free_ctx(&srv->actual_transition.timer.work);
 | |
|             bt_mesh_server_free_ctx(&srv->linear_transition.timer.work);
 | |
|             k_delayed_work_free(&srv->actual_transition.timer);
 | |
|             k_delayed_work_free(&srv->linear_transition.timer);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_CTL_SRV: {
 | |
|         struct bt_mesh_light_ctl_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light CTL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_free_ctx(&srv->transition.timer.work);
 | |
|             k_delayed_work_free(&srv->transition.timer);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_CTL_TEMP_SRV: {
 | |
|         struct bt_mesh_light_ctl_temp_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light CTL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_free_ctx(&srv->transition.timer.work);
 | |
|             k_delayed_work_free(&srv->transition.timer);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_SRV: {
 | |
|         struct bt_mesh_light_hsl_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light HSL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_free_ctx(&srv->transition.timer.work);
 | |
|             k_delayed_work_free(&srv->transition.timer);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_HUE_SRV: {
 | |
|         struct bt_mesh_light_hsl_hue_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light HSL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_free_ctx(&srv->transition.timer.work);
 | |
|             k_delayed_work_free(&srv->transition.timer);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_SAT_SRV: {
 | |
|         struct bt_mesh_light_hsl_sat_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light HSL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_free_ctx(&srv->transition.timer.work);
 | |
|             k_delayed_work_free(&srv->transition.timer);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_XYL_SRV: {
 | |
|         struct bt_mesh_light_xyl_srv *srv = model->user_data;
 | |
|         if (srv->state == NULL) {
 | |
|             BT_ERR("Invalid Light xyL State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_free_ctx(&srv->transition.timer.work);
 | |
|             k_delayed_work_free(&srv->transition.timer);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_LC_SRV: {
 | |
|         struct bt_mesh_light_lc_srv *srv = model->user_data;
 | |
|         if (srv->lc == NULL) {
 | |
|             BT_ERR("Invalid Light LC State");
 | |
|             return -EINVAL;
 | |
|         }
 | |
|         if (srv->rsp_ctrl.set_auto_rsp == BLE_MESH_SERVER_AUTO_RSP) {
 | |
|             bt_mesh_server_free_ctx(&srv->transition.timer.work);
 | |
|             k_delayed_work_free(&srv->transition.timer);
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_SETUP_SRV:
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_CTL_SETUP_SRV:
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_HSL_SETUP_SRV:
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_XYL_SETUP_SRV:
 | |
|     case BLE_MESH_MODEL_ID_LIGHT_LC_SETUP_SRV:
 | |
|         break;
 | |
|     default:
 | |
|         BT_WARN("Unknown Light Server, model id 0x%04x", model->id);
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     bt_mesh_mutex_free(&light_server_lock);
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int light_lightness_srv_deinit(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light Lightness Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     return light_server_deinit(model);
 | |
| }
 | |
| 
 | |
| static int light_lightness_setup_srv_deinit(struct bt_mesh_model *model)
 | |
| {
 | |
|     return light_server_deinit(model);
 | |
| }
 | |
| 
 | |
| static int light_ctl_srv_deinit(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light CTL Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     return light_server_deinit(model);
 | |
| }
 | |
| 
 | |
| static int light_ctl_setup_srv_deinit(struct bt_mesh_model *model)
 | |
| {
 | |
|     return light_server_deinit(model);
 | |
| }
 | |
| 
 | |
| static int light_ctl_temp_srv_deinit(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light CTL Temperature Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     return light_server_deinit(model);
 | |
| }
 | |
| 
 | |
| static int light_hsl_srv_deinit(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light HSL Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     return light_server_deinit(model);
 | |
| }
 | |
| 
 | |
| static int light_hsl_setup_srv_deinit(struct bt_mesh_model *model)
 | |
| {
 | |
|     return light_server_deinit(model);
 | |
| }
 | |
| 
 | |
| static int light_hsl_hue_srv_deinit(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light HSL Hue Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     return light_server_deinit(model);
 | |
| }
 | |
| 
 | |
| static int light_hsl_sat_srv_deinit(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light HSL Saturation Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     return light_server_deinit(model);
 | |
| }
 | |
| 
 | |
| static int light_xyl_srv_deinit(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light xyL Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     return light_server_deinit(model);
 | |
| }
 | |
| 
 | |
| static int light_xyl_setup_srv_deinit(struct bt_mesh_model *model)
 | |
| {
 | |
|     return light_server_deinit(model);
 | |
| }
 | |
| 
 | |
| static int light_lc_srv_deinit(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light LC Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     return light_server_deinit(model);
 | |
| }
 | |
| 
 | |
| static int light_lc_setup_srv_deinit(struct bt_mesh_model *model)
 | |
| {
 | |
|     if (model->pub == NULL) {
 | |
|         BT_ERR("Light LC Setup Server has no publication support");
 | |
|         return -EINVAL;
 | |
|     }
 | |
| 
 | |
|     return light_server_deinit(model);
 | |
| }
 | |
| #endif /* CONFIG_BLE_MESH_DEINIT */
 | |
| 
 | |
| const struct bt_mesh_model_cb bt_mesh_light_lightness_srv_cb = {
 | |
|     .init = light_lightness_srv_init,
 | |
| #if CONFIG_BLE_MESH_DEINIT
 | |
|     .deinit = light_lightness_srv_deinit,
 | |
| #endif /* CONFIG_BLE_MESH_DEINIT */
 | |
| };
 | |
| 
 | |
| const struct bt_mesh_model_cb bt_mesh_light_lightness_setup_srv_cb = {
 | |
|     .init = light_lightness_setup_srv_init,
 | |
| #if CONFIG_BLE_MESH_DEINIT
 | |
|     .deinit = light_lightness_setup_srv_deinit,
 | |
| #endif /* CONFIG_BLE_MESH_DEINIT */
 | |
| };
 | |
| 
 | |
| const struct bt_mesh_model_cb bt_mesh_light_ctl_srv_cb = {
 | |
|     .init = light_ctl_srv_init,
 | |
| #if CONFIG_BLE_MESH_DEINIT
 | |
|     .deinit = light_ctl_srv_deinit,
 | |
| #endif /* CONFIG_BLE_MESH_DEINIT */
 | |
| };
 | |
| 
 | |
| const struct bt_mesh_model_cb bt_mesh_light_ctl_setup_srv_cb = {
 | |
|     .init = light_ctl_setup_srv_init,
 | |
| #if CONFIG_BLE_MESH_DEINIT
 | |
|     .deinit = light_ctl_setup_srv_deinit,
 | |
| #endif /* CONFIG_BLE_MESH_DEINIT */
 | |
| };
 | |
| 
 | |
| const struct bt_mesh_model_cb bt_mesh_light_ctl_temp_srv_cb = {
 | |
|     .init = light_ctl_temp_srv_init,
 | |
| #if CONFIG_BLE_MESH_DEINIT
 | |
|     .deinit = light_ctl_temp_srv_deinit,
 | |
| #endif /* CONFIG_BLE_MESH_DEINIT */
 | |
| };
 | |
| 
 | |
| const struct bt_mesh_model_cb bt_mesh_light_hsl_srv_cb = {
 | |
|     .init = light_hsl_srv_init,
 | |
| #if CONFIG_BLE_MESH_DEINIT
 | |
|     .deinit = light_hsl_srv_deinit,
 | |
| #endif /* CONFIG_BLE_MESH_DEINIT */
 | |
| };
 | |
| 
 | |
| const struct bt_mesh_model_cb bt_mesh_light_hsl_setup_srv_cb = {
 | |
|     .init = light_hsl_setup_srv_init,
 | |
| #if CONFIG_BLE_MESH_DEINIT
 | |
|     .deinit = light_hsl_setup_srv_deinit,
 | |
| #endif /* CONFIG_BLE_MESH_DEINIT */
 | |
| };
 | |
| 
 | |
| const struct bt_mesh_model_cb bt_mesh_light_hsl_hue_srv_cb = {
 | |
|     .init = light_hsl_hue_srv_init,
 | |
| #if CONFIG_BLE_MESH_DEINIT
 | |
|     .deinit = light_hsl_hue_srv_deinit,
 | |
| #endif /* CONFIG_BLE_MESH_DEINIT */
 | |
| };
 | |
| 
 | |
| const struct bt_mesh_model_cb bt_mesh_light_hsl_sat_srv_cb = {
 | |
|     .init = light_hsl_sat_srv_init,
 | |
| #if CONFIG_BLE_MESH_DEINIT
 | |
|     .deinit = light_hsl_sat_srv_deinit,
 | |
| #endif /* CONFIG_BLE_MESH_DEINIT */
 | |
| };
 | |
| 
 | |
| const struct bt_mesh_model_cb bt_mesh_light_xyl_srv_cb = {
 | |
|     .init = light_xyl_srv_init,
 | |
| #if CONFIG_BLE_MESH_DEINIT
 | |
|     .deinit = light_xyl_srv_deinit,
 | |
| #endif /* CONFIG_BLE_MESH_DEINIT */
 | |
| };
 | |
| 
 | |
| const struct bt_mesh_model_cb bt_mesh_light_xyl_setup_srv_cb = {
 | |
|     .init = light_xyl_setup_srv_init,
 | |
| #if CONFIG_BLE_MESH_DEINIT
 | |
|     .deinit = light_xyl_setup_srv_deinit,
 | |
| #endif /* CONFIG_BLE_MESH_DEINIT */
 | |
| };
 | |
| 
 | |
| const struct bt_mesh_model_cb bt_mesh_light_lc_srv_cb = {
 | |
|     .init = light_lc_srv_init,
 | |
| #if CONFIG_BLE_MESH_DEINIT
 | |
|     .deinit = light_lc_srv_deinit,
 | |
| #endif /* CONFIG_BLE_MESH_DEINIT */
 | |
| };
 | |
| 
 | |
| const struct bt_mesh_model_cb bt_mesh_light_lc_setup_srv_cb = {
 | |
|     .init = light_lc_setup_srv_init,
 | |
| #if CONFIG_BLE_MESH_DEINIT
 | |
|     .deinit = light_lc_setup_srv_deinit,
 | |
| #endif /* CONFIG_BLE_MESH_DEINIT */
 | |
| };
 | |
| 
 | |
| #endif /* CONFIG_BLE_MESH_LIGHTING_SERVER */
 | 
