391 lines
13 KiB
C
391 lines
13 KiB
C
|
/**
|
||
|
****************************************************************************************
|
||
|
* @addtogroup BASSTASK
|
||
|
* @{
|
||
|
****************************************************************************************
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* INCLUDE FILES
|
||
|
****************************************************************************************
|
||
|
*/
|
||
|
|
||
|
#include "rwip_config.h"
|
||
|
|
||
|
#if (BLE_BATT_SERVER)
|
||
|
|
||
|
#include "gap.h"
|
||
|
#include "gattc_task.h"
|
||
|
|
||
|
#include "bass.h"
|
||
|
#include "bass_task.h"
|
||
|
|
||
|
#include "prf_utils.h"
|
||
|
|
||
|
#include "co_utils.h"
|
||
|
|
||
|
/*
|
||
|
* GLOBAL FUNCTIONS DEFINITIONS
|
||
|
****************************************************************************************
|
||
|
*/
|
||
|
|
||
|
|
||
|
/**
|
||
|
****************************************************************************************
|
||
|
* @brief Handles reception of the @ref BAPS_ENABLE_REQ message.
|
||
|
* The handler enables the Battery 'Profile' Server Role.
|
||
|
* @param[in] msgid Id of the message received (probably unused).
|
||
|
* @param[in] param Pointer to the parameters of the message.
|
||
|
* @param[in] dest_id ID of the receiving task instance (probably unused).
|
||
|
* @param[in] src_id ID of the sending task instance.
|
||
|
* @return If the message was consumed or not.
|
||
|
****************************************************************************************
|
||
|
*/
|
||
|
__STATIC int bass_enable_req_handler(ke_msg_id_t const msgid,
|
||
|
struct bass_enable_req const *param,
|
||
|
ke_task_id_t const dest_id,
|
||
|
ke_task_id_t const src_id)
|
||
|
{
|
||
|
int msg_status = KE_MSG_SAVED;
|
||
|
uint8_t state = ke_state_get(dest_id);
|
||
|
|
||
|
// check state of the task
|
||
|
if(state == BASS_IDLE)
|
||
|
{
|
||
|
struct bass_env_tag* bass_env = PRF_ENV_GET(BASS, bass);
|
||
|
|
||
|
// Check provided values
|
||
|
if((param->conidx > BLE_CONNECTION_MAX)
|
||
|
|| (gapc_get_conhdl(param->conidx) == GAP_INVALID_CONHDL))
|
||
|
{
|
||
|
// an error occurs, trigg it.
|
||
|
struct bass_enable_rsp* rsp = KE_MSG_ALLOC(BASS_ENABLE_RSP, src_id,
|
||
|
dest_id, bass_enable_rsp);
|
||
|
rsp->conidx = param->conidx;
|
||
|
rsp->status = (param->conidx > BLE_CONNECTION_MAX) ? GAP_ERR_INVALID_PARAM : PRF_ERR_REQ_DISALLOWED;
|
||
|
ke_msg_send(rsp);
|
||
|
|
||
|
msg_status = KE_MSG_CONSUMED;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// put task in a busy state
|
||
|
msg_status = KE_MSG_NO_FREE;
|
||
|
ke_state_set(dest_id, BASS_BUSY);
|
||
|
bass_env->ntf_cfg[param->conidx] = param->ntf_cfg;
|
||
|
bass_env->operation = ke_param2msg(param);
|
||
|
bass_env->cursor = 0;
|
||
|
|
||
|
// trigger notification
|
||
|
bass_exe_operation();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return msg_status;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
****************************************************************************************
|
||
|
* @brief Handles reception of the @ref BAPS_BATT_LEVEL_SEND_REQ message.
|
||
|
* @param[in] msgid Id of the message received (probably unused).
|
||
|
* @param[in] param Pointer to the parameters of the message.
|
||
|
* @param[in] dest_id ID of the receiving task instance (probably unused).
|
||
|
* @param[in] src_id ID of the sending task instance.
|
||
|
* @return If the message was consumed or not.
|
||
|
****************************************************************************************
|
||
|
*/
|
||
|
__STATIC int bass_batt_level_upd_req_handler(ke_msg_id_t const msgid,
|
||
|
struct bass_batt_level_upd_req const *param,
|
||
|
ke_task_id_t const dest_id,
|
||
|
ke_task_id_t const src_id)
|
||
|
{
|
||
|
int msg_status = KE_MSG_SAVED;
|
||
|
uint8_t state = ke_state_get(dest_id);
|
||
|
|
||
|
// check state of the task
|
||
|
if(state == BASS_IDLE)
|
||
|
{
|
||
|
struct bass_env_tag* bass_env = PRF_ENV_GET(BASS, bass);
|
||
|
|
||
|
// Check provided values
|
||
|
if((param->bas_instance < bass_env->svc_nb) && (param->batt_level <= BAS_BATTERY_LVL_MAX))
|
||
|
{
|
||
|
// update the battery level value
|
||
|
bass_env->batt_lvl[param->bas_instance] = param->batt_level;
|
||
|
|
||
|
// put task in a busy state
|
||
|
msg_status = KE_MSG_NO_FREE;
|
||
|
ke_state_set(dest_id, BASS_BUSY);
|
||
|
bass_env->operation = ke_param2msg(param);
|
||
|
bass_env->cursor = 0;
|
||
|
|
||
|
// trigger notification
|
||
|
bass_exe_operation();
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// an error occurs, trigg it.
|
||
|
struct bass_batt_level_upd_rsp * rsp = KE_MSG_ALLOC(BASS_BATT_LEVEL_UPD_RSP, src_id,
|
||
|
dest_id, bass_batt_level_upd_rsp);
|
||
|
|
||
|
rsp->status = PRF_ERR_INVALID_PARAM;
|
||
|
ke_msg_send(rsp);
|
||
|
msg_status = KE_MSG_CONSUMED;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return (msg_status);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
****************************************************************************************
|
||
|
* @brief Handles reception of the attribute info request message.
|
||
|
*
|
||
|
* @param[in] msgid Id of the message received (probably unused).
|
||
|
* @param[in] param Pointer to the parameters of the message.
|
||
|
* @param[in] dest_id ID of the receiving task instance (probably unused).
|
||
|
* @param[in] src_id ID of the sending task instance.
|
||
|
* @return If the message was consumed or not.
|
||
|
****************************************************************************************
|
||
|
*/
|
||
|
__STATIC int gattc_att_info_req_ind_handler(ke_msg_id_t const msgid,
|
||
|
struct gattc_att_info_req_ind *param,
|
||
|
ke_task_id_t const dest_id,
|
||
|
ke_task_id_t const src_id)
|
||
|
{
|
||
|
|
||
|
struct gattc_att_info_cfm * cfm;
|
||
|
uint8_t svc_idx = 0, att_idx = 0;
|
||
|
// retrieve handle information
|
||
|
uint8_t status = bass_get_att_idx(param->handle, &svc_idx, &att_idx);
|
||
|
|
||
|
//Send write response
|
||
|
cfm = KE_MSG_ALLOC(GATTC_ATT_INFO_CFM, src_id, dest_id, gattc_att_info_cfm);
|
||
|
cfm->handle = param->handle;
|
||
|
|
||
|
if(status == GAP_ERR_NO_ERROR)
|
||
|
{
|
||
|
// check if it's a client configuration char
|
||
|
if(att_idx == BAS_IDX_BATT_LVL_NTF_CFG)
|
||
|
{
|
||
|
// CCC attribute length = 2
|
||
|
cfm->length = 2;
|
||
|
}
|
||
|
// not expected request
|
||
|
else
|
||
|
{
|
||
|
cfm->length = 0;
|
||
|
status = ATT_ERR_WRITE_NOT_PERMITTED;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
cfm->status = status;
|
||
|
ke_msg_send(cfm);
|
||
|
|
||
|
return (KE_MSG_CONSUMED);
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
****************************************************************************************
|
||
|
* @brief Handles reception of the @ref GATTC_WRITE_REQ_IND message.
|
||
|
* @param[in] msgid Id of the message received (probably unused).
|
||
|
* @param[in] param Pointer to the parameters of the message.
|
||
|
* @param[in] dest_id ID of the receiving task instance (probably unused).
|
||
|
* @param[in] src_id ID of the sending task instance.
|
||
|
* @return If the message was consumed or not.
|
||
|
****************************************************************************************
|
||
|
*/
|
||
|
__STATIC int gattc_write_req_ind_handler(ke_msg_id_t const msgid, struct gattc_write_req_ind const *param,
|
||
|
ke_task_id_t const dest_id, ke_task_id_t const src_id)
|
||
|
{
|
||
|
struct gattc_write_cfm * cfm;
|
||
|
uint8_t svc_idx = 0, att_idx = 0;
|
||
|
uint8_t conidx = KE_IDX_GET(src_id);
|
||
|
// retrieve handle information
|
||
|
uint8_t status = bass_get_att_idx(param->handle, &svc_idx, &att_idx);
|
||
|
|
||
|
// If the attribute has been found, status is GAP_ERR_NO_ERROR
|
||
|
if (status == GAP_ERR_NO_ERROR)
|
||
|
{
|
||
|
struct bass_env_tag* bass_env = PRF_ENV_GET(BASS, bass);
|
||
|
// Extract value before check
|
||
|
uint16_t ntf_cfg = co_read16p(¶m->value[0]);
|
||
|
|
||
|
// Only update configuration if value for stop or notification enable
|
||
|
if ((att_idx == BAS_IDX_BATT_LVL_NTF_CFG)
|
||
|
&& ((ntf_cfg == PRF_CLI_STOP_NTFIND) || (ntf_cfg == PRF_CLI_START_NTF)))
|
||
|
{
|
||
|
|
||
|
// Conserve information in environment
|
||
|
if (ntf_cfg == PRF_CLI_START_NTF)
|
||
|
{
|
||
|
// Ntf cfg bit set to 1
|
||
|
bass_env->ntf_cfg[conidx] |= (BAS_BATT_LVL_NTF_SUP << svc_idx);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// Ntf cfg bit set to 0
|
||
|
bass_env->ntf_cfg[conidx] &= ~(BAS_BATT_LVL_NTF_SUP << svc_idx);
|
||
|
}
|
||
|
|
||
|
// Inform APP of configuration change
|
||
|
struct bass_batt_level_ntf_cfg_ind * ind = KE_MSG_ALLOC(BASS_BATT_LEVEL_NTF_CFG_IND,
|
||
|
prf_dst_task_get(&(bass_env->prf_env), conidx), dest_id,
|
||
|
bass_batt_level_ntf_cfg_ind);
|
||
|
ind->conidx = conidx;
|
||
|
ind->ntf_cfg = bass_env->ntf_cfg[conidx];
|
||
|
|
||
|
ke_msg_send(ind);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
status = PRF_APP_ERROR;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
//Send write response
|
||
|
cfm = KE_MSG_ALLOC(GATTC_WRITE_CFM, src_id, dest_id, gattc_write_cfm);
|
||
|
cfm->handle = param->handle;
|
||
|
cfm->status = status;
|
||
|
ke_msg_send(cfm);
|
||
|
|
||
|
return (KE_MSG_CONSUMED);
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
****************************************************************************************
|
||
|
* @brief Handles reception of the @ref GATTC_READ_REQ_IND message.
|
||
|
* @param[in] msgid Id of the message received (probably unused).
|
||
|
* @param[in] param Pointer to the parameters of the message.
|
||
|
* @param[in] dest_id ID of the receiving task instance (probably unused).
|
||
|
* @param[in] src_id ID of the sending task instance.
|
||
|
* @return If the message was consumed or not.
|
||
|
****************************************************************************************
|
||
|
*/
|
||
|
__STATIC int gattc_read_req_ind_handler(ke_msg_id_t const msgid, struct gattc_read_req_ind const *param,
|
||
|
ke_task_id_t const dest_id, ke_task_id_t const src_id)
|
||
|
{
|
||
|
struct gattc_read_cfm * cfm;
|
||
|
uint8_t svc_idx = 0, att_idx = 0;
|
||
|
uint8_t conidx = KE_IDX_GET(src_id);
|
||
|
// retrieve handle information
|
||
|
uint8_t status = bass_get_att_idx(param->handle, &svc_idx, &att_idx);
|
||
|
uint16_t length = 0;
|
||
|
struct bass_env_tag* bass_env = PRF_ENV_GET(BASS, bass);
|
||
|
|
||
|
// If the attribute has been found, status is GAP_ERR_NO_ERROR
|
||
|
if (status == GAP_ERR_NO_ERROR)
|
||
|
{
|
||
|
// read notification information
|
||
|
if (att_idx == BAS_IDX_BATT_LVL_VAL)
|
||
|
{
|
||
|
length = sizeof(uint8_t);
|
||
|
}
|
||
|
// read notification information
|
||
|
else if (att_idx == BAS_IDX_BATT_LVL_NTF_CFG)
|
||
|
{
|
||
|
length = sizeof(uint16_t);
|
||
|
}
|
||
|
else if(att_idx == BAS_IDX_BATT_LVL_PRES_FMT)
|
||
|
{
|
||
|
length = PRF_CHAR_PRES_FMT_SIZE;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
status = PRF_APP_ERROR;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//Send write response
|
||
|
cfm = KE_MSG_ALLOC_DYN(GATTC_READ_CFM, src_id, dest_id, gattc_read_cfm, length);
|
||
|
cfm->handle = param->handle;
|
||
|
cfm->status = status;
|
||
|
cfm->length = length;
|
||
|
|
||
|
if (status == GAP_ERR_NO_ERROR)
|
||
|
{
|
||
|
// read notification information
|
||
|
if (att_idx == BAS_IDX_BATT_LVL_VAL)
|
||
|
{
|
||
|
cfm->value[0] = bass_env->batt_lvl[svc_idx];
|
||
|
}
|
||
|
// retrieve notification config
|
||
|
else if (att_idx == BAS_IDX_BATT_LVL_NTF_CFG)
|
||
|
{
|
||
|
uint16_t ntf_cfg = (bass_env->ntf_cfg[conidx] >> svc_idx & BAS_BATT_LVL_NTF_SUP) ? PRF_CLI_START_NTF : PRF_CLI_STOP_NTFIND;
|
||
|
co_write16p(cfm->value, ntf_cfg);
|
||
|
}
|
||
|
// retrieve battery level format
|
||
|
else if(att_idx == BAS_IDX_BATT_LVL_PRES_FMT)
|
||
|
{
|
||
|
prf_pack_char_pres_fmt(cfm->value, &(bass_env->batt_level_pres_format[svc_idx]));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
/* Not Possible */
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ke_msg_send(cfm);
|
||
|
|
||
|
return (KE_MSG_CONSUMED);
|
||
|
}
|
||
|
/**
|
||
|
****************************************************************************************
|
||
|
* @brief Handles @ref GATTC_CMP_EVT for GATTC_NOTIFY message meaning that Measurement
|
||
|
* notification has been correctly sent to peer device (but not confirmed by peer device).
|
||
|
* *
|
||
|
* @param[in] msgid Id of the message received.
|
||
|
* @param[in] param Pointer to the parameters of the message.
|
||
|
* @param[in] dest_id ID of the receiving task instance
|
||
|
* @param[in] src_id ID of the sending task instance.
|
||
|
* @return If the message was consumed or not.
|
||
|
****************************************************************************************
|
||
|
*/
|
||
|
__STATIC int gattc_cmp_evt_handler(ke_msg_id_t const msgid, struct gattc_cmp_evt const *param,
|
||
|
ke_task_id_t const dest_id, ke_task_id_t const src_id)
|
||
|
{
|
||
|
if(param->operation == GATTC_NOTIFY)
|
||
|
{
|
||
|
// continue operation execution
|
||
|
bass_exe_operation();
|
||
|
}
|
||
|
return (KE_MSG_CONSUMED);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* GLOBAL VARIABLE DEFINITIONS
|
||
|
****************************************************************************************
|
||
|
*/
|
||
|
|
||
|
/// Default State handlers definition
|
||
|
KE_MSG_HANDLER_TAB(bass)
|
||
|
{
|
||
|
{BASS_ENABLE_REQ, (ke_msg_func_t) bass_enable_req_handler},
|
||
|
{BASS_BATT_LEVEL_UPD_REQ, (ke_msg_func_t) bass_batt_level_upd_req_handler},
|
||
|
{GATTC_ATT_INFO_REQ_IND, (ke_msg_func_t) gattc_att_info_req_ind_handler},
|
||
|
{GATTC_WRITE_REQ_IND, (ke_msg_func_t) gattc_write_req_ind_handler},
|
||
|
{GATTC_READ_REQ_IND, (ke_msg_func_t) gattc_read_req_ind_handler},
|
||
|
{GATTC_CMP_EVT, (ke_msg_func_t) gattc_cmp_evt_handler},
|
||
|
};
|
||
|
|
||
|
void bass_task_init(struct ke_task_desc *task_desc)
|
||
|
{
|
||
|
// Get the address of the environment
|
||
|
struct bass_env_tag *bass_env = PRF_ENV_GET(BASS, bass);
|
||
|
|
||
|
task_desc->msg_handler_tab = bass_msg_handler_tab;
|
||
|
task_desc->msg_cnt = ARRAY_LEN(bass_msg_handler_tab);
|
||
|
task_desc->state = bass_env->state;
|
||
|
task_desc->idx_max = BASS_IDX_MAX;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
#endif /* #if (BLE_BATT_SERVER) */
|
||
|
|
||
|
/// @} BASSTASK
|