嗨,对话框的团队,
我遵循AN-B-029文档,并成功地将新的特征(通知/读取)添加到我的服务中。但是,当我添加一个新的特征时(写/读取),我无法写入它。当我写任何东西时,每次都会显示“写入特征失败”。如果有任何文件,如AN-B-029,它将提供步骤?我陷入了困境。
但是,现在我想向现有的示例添加新服务。而我的新服务应该包含两个特征。我该如何实施它?请帮忙。
此致敬礼,
山姆
关键词:
设备:
你好山姆,
个人资料将收到
gattc_write_cmd_ind.
信息。您需要在您的个人资料中附加此消息的处理函数,您可能已经已经完成了?你能分享你的处理函数吗?profilename_task.c.?嗨VesaN,
我正在尝试向现有示例128添加写入/读取特性。
/ **
****************************************************************************************
*
* @file sample128_task.c
*
* @brief sample128任务实施。
*
* @brief 128 uuid服务。示例代码
*
*版权所有(c)2013年对话半导体Gmbyabo国际娱乐H及其关联公司,未发表的工作
*此计算机程序包括机密,专有信息,是商业秘密
*对话半导yabo国际娱乐体GmbH及其附属公司。所有使用,披露和/或
*除非授权书面授权,否则禁止复制。版权所有。
*
****************************************************************************************
* /
/ *
*包括文件
****************************************************************************************
* /
# include“rwble_config.h”
#如果(BLE_SAMPLE128)
# include“gap.h”
#include“gapc.h”
#include“gattc_task.h”
# include“atts_util.h”
# include“sample128.h”
# include“sample128_task.h”
# include“attm_cfg.h”
#include“attm_db.h”
# include“prf_utils.h”
/ *
*函数定义
****************************************************************************************
* /
/ **
****************************************************************************************
* @brief处理接收@ref SAMPLE128_CREATE_DB_REQ消息。
* @param [in]收到的消息的msgid ID(可能未使用)。
* @param [in] param指向消息的参数。
* @param [in]接收任务实例的dest_id ID(可能未使用)。
* @param [in]发送任务实例的src_id ID。
* @return如果邮件被消耗或不。
****************************************************************************************
* /
静态int sample128_create_db_req_handler(ke_msg_id_t const msgid,
struct sample128_create_db_req const * param,
ke_task_id_t const dest_id,
ke_task_id_t const src_id)
{
//数据库创建状态
UINT8_T状态;
uint8_t nb_att_16;
uint8_t nb_att_128;
uint8_t nb_att_32;
uint16_t att_decl_svc = ATT_DECL_PRIMARY_SERVICE;
uint16_t att_decl_char = ATT_DECL_CHARACTERISTIC;
uint16_t att_decl_cfg = ATT_DESC_CLIENT_CHAR_CFG;
uint16_t val_hdl;
uint16_t char_hdl;
/ /保存概要文件ID
sample128_env.con_info.prf_id = task_sample128;
/ * --------------------------------------------- *
Sample128服务创建
*---------------------------------------------------*/
//将服务添加到数据库
nb_att_16 = 5;// 6 ... 4 UUID16属性声明类型
nb_att_32 = 0;// 0 UUID32属性声明类型
nb_att_128 = 3;// 2 uuid128属性声明类型
status = attmdb_add_service(&(sample128_env.sample128_shdl),
task_sample128,
nb_att_16,
nb_att_32,
nb_att_128,
78.
);
if(status == att_err_no_error)
{
/ /添加的主要服务属性 /////////////////////////////////////////////////////////////////
状态= attmdb_add_attribute(sample128_env.sample128_shdl,// Attribute Handle
ATT_UUID_128_LEN, //数据大小= 16 (ATT_UUID_128_LEN)
ATT_UUID_16_LEN,//声明类型ID的大小
(UINT8_T *)&ATT_DECL_SVC,// 0x2800用于主要服务声明
PERM(RD, ENABLE), //权限
&(sample128_env.sample128_shdl)//属性句柄
);
//添加主服务属性的值(自定义UUID)
status = attmdb_att_set_value(sample128_env.sample128_shdl,//属性句柄
ATT_UUID_128_LEN,//该值是服务的128位UUID
(uint8_t *) sample128_svc。uuid //服务的uuid
);
//独特性1:////// //////////////////////////////////////////////////////////////////////////////////////////////
//添加特征声明属性到数据库
状态= attmdb_add_attribute(sample128_env.sample128_shdl,
ATT_UUID_128_LEN + 3, //数据大小= 19 (ATT_UUID_128_LEN + 3)
ATT_UUID_16_LEN,//声明类型ID的大小
(UINT8_T *)&ATT_DECL_CHAR,// 0x2803用于特征声明
PERM(RD, ENABLE), //权限
&(char_hdl) //特征声明的句柄
);
//添加特征值声明属性到数据库
状态= attmdb_add_attribute(sample128_env.sample128_shdl,
sizeof(uint8_t),// data size = 1字节
ATT_UUID_128_LEN,//自定义声明类型大小= 128bit
(uint8_t *) &sample128_1_val。uuid, //特征值的uuid
perm(rd,启用)|perm(wr,启用),//权限
&(val_hdl) // value属性句柄
);
//存储特性1的值句柄1
memcpy (sample128_1_char。attr_hdl &val_hdl, sizeof (uint16_t));
//设置特征1的初始值
状态= attmdb_att_set_value(char_hdl, sizeof(sample128_1_char), (uint8_t *)&sample128_1_char);
//特征2://///////////////////////////////////////////////////////////////////////////////////////////
//添加特征声明属性到数据库
状态= attmdb_add_attribute(sample128_env.sample128_shdl,
ATT_UUID_128_LEN + 3, //数据大小= 19 (ATT_UUID_128_LEN + 3)
ATT_UUID_16_LEN,//声明类型ID的大小
(UINT8_T *)&ATT_DECL_CHAR,// 0x2803用于特征声明
烫(RD,启用),/ /权限
&(char_hdl) //特征声明的句柄
);
//添加特征值声明属性到数据库
状态= attmdb_add_attribute(sample128_env.sample128_shdl,
sizeof(uint8_t), //数据大小= 1字节
ATT_UUID_128_LEN,//自定义声明类型ID = 128bit的大小
(uint8_t *) &sample128_2_val。uuid, //特征值的uuid
perm(rd,启用)|perm(ntf,启用),//权限
&(val_hdl)//句柄到值属性
);
//存储特性2的值句柄
memcpy (sample128_2_char。attr_hdl &val_hdl, sizeof (uint16_t));
//设置特征2的初始值
status = attmdb_att_set_value(char_hdl,sizeof(sample128_2_char),(uint8_t *)和sample128_2_char);
//将客户端配置声明属性添加到数据库
状态= attmdb_add_attribute(sample128_env.sample128_shdl,
sizeof(uint16_t), //数据大小2字节(16位)
ATT_UUID_16_LEN,//客户端配置类型ID的大小
(UINT8_T *)&ATT_DECL_CFG,// 0x2902客户端配置声明类型的UUID
PERM(RD, ENABLE) | PERM(WR, ENABLE), //权限
&(val_hdl) //值属性句柄
);
// writeitisit 3 for write://///////////////////////////////////////////////////////////////////////////////////////
//添加特征声明属性到数据库
状态= attmdb_add_attribute(sample128_env.sample128_shdl,
ATT_UUID_128_LEN + 3, //数据大小= 19 (ATT_UUID_128_LEN + 3)
ATT_UUID_16_LEN,//声明类型ID的大小
(UINT8_T *)&ATT_DECL_CHAR,// 0x2803用于特征声明
PERM(RD, ENABLE), //权限
&(char_hdl) //特征声明的句柄
);
//添加特征值声明属性到数据库
状态= attmdb_add_attribute(sample128_env.sample128_shdl,
sizeof(my_newer_t),//数据大小= 1字节
ATT_UUID_128_LEN,//自定义声明类型大小= 128bit
(uint8_t *) &sample128_3_val。uuid, //特征值的uuid
perm(rd,启用)|perm(wr,启用),//权限
&(val_hdl) // value属性句柄
);
//存储特征3的值句柄
memcpy (sample128_3_char。attr_hdl &val_hdl, sizeof (uint16_t));
//设置特征值3的初始值
status = attmdb_att_set_value(char_hdl,sizeof(sample128_3_char),(uint8_t *)和sample128_3_char);
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
/ /禁用sample128服务
attmdb_svc_set_permission(sample128_env.sample128_shdl,perm(svc,disable));
//如果我们在这里,数据库已完成与成功,转到空闲状态
ke_state_set(task_sample128,sample128_idle);
}
//发送CFM申请
struct sample128_create_db_cfm * cfm = KE_MSG_ALLOC(sample128_create_db_cfm, src_id,
task_sample128,sample128_create_db_cfm);
cfm-> status = status;
ke_msg_send (cfm);
return(ke_msg_consumed);
}
/ **
****************************************************************************************
* @brief启用示例128角色,在连接后使用。
* @param[in] msgid收到的消息的Id。
* @param [in] param指向消息的参数。
* @param[in] dest_id接收任务实例ID
* @param [in]发送任务实例的src_id ID。
* @return如果邮件被消耗或不。
****************************************************************************************
* /
静态int sample128_enable_req_handler(ke_msg_id_t const msgid,
Struct sample128_enable_req const *param,
ke_task_id_t const dest_id,
ke_task_id_t const src_id)
{
uint16_t temp = 1;
//保留消息源,以便后续对其进行响应
sample128_env.con_info.appid = src_id;
//保存这个配置文件的连接句柄
sample128_env.con_info.conidx = gapc_get_conidx(param-> conhdl);
//检查是否存在提供的连接
if(sample128_env.con_info.conidx == gap_invalid_conidx)
{
//连接不存在,请求被拒绝
prf_server_error_ind_send((prf_env_struct *)和sample128_env,prf_err_req_disally,
sample128_error_ind,sample128_enable_req);
}
其他的
{
//示例128服务权限
attmdb_svc_set_permission (sample128_env。sample128_shdl param - > sec_lvl);
//将特征值1设置为指定值
attmdb_att_set_value(sample128_env.sample128_shdl + sample128_1_idx_val,
sizeof(uint8_t),(uint8_t *)和param-> sample128_1_val);
//将特征2设置为指定值
attmdb_att_set_value(sample128_env.sample128_shdl + sample128_2_idx_val,
sizeof (uint8_t), (uint8_t *)参数- > sample128_2_val);
//将特征3设置为指定值
attmdb_att_set_value(sample128_env.sample128_shdl + sample128_3_idx_val,
sizeof (uint8_t), (uint8_t *)参数- > sample128_3_val);
sample128_env。特点=参数- >功能;
if(!sample128_env.feature)
{
temp = 0;
}
attmdb_att_set_value(sample128_env.sample128_shdl + sample128_2_idx_cfg,
sizeof(uint16_t),(uint8_t *)&temp);
//转到连接状态
ke_state_set(task_sample128,sample128_connected);
}
return(ke_msg_consumed);
}
/ **
****************************************************************************************
* @brief更新feature 2的值。如果启用了属性,则向对等体发送通知。
* @param[in] msgid收到的消息的Id。
* @param [in] param指向消息的参数。
* @param[in] dest_id接收任务实例ID
* @param [in]发送任务实例的src_id ID。
* @return如果邮件被消耗或不。
****************************************************************************************
* /
静态int sample128_upd_char2_req_handler(ke_msg_id_t const msgid,
Struct sample128_upd_char2_req const *参数,
ke_task_id_t const dest_id,
ke_task_id_t const src_id)
{
uint8_t status = PRF_ERR_OK;
//检查提供的值
if(param-> conhdl == gapc_get_conhdl(sample128_env.con_info.conidx))
{
//更新数据库中的值
attmdb_att_set_value(sample128_env.sample128_shdl + sample128_2_idx_val,
sizeof (uint8_t), (uint8_t *)参数- > val);
如果(sample128_env。特性和PRF_CLI_START_NTF))
//通过gatt发送通知
prf_server_send_event ((prf_env_struct *) &sample128_env假,
sample128_env.sample128_shdl + sample128_2_idx_val);
}
其他的
{
状态= PRF_ERR_INVALID_PARAM;
}
if (status != PRF_ERR_OK)
{
sample128_upd_char2_cfm_send(状态);
}
return(ke_msg_consumed);
}
/ **
****************************************************************************************
* @brief处理@ref gatt_write_cmd_ind消息的接收。
* @param [in]收到的消息的msgid ID(可能未使用)。
* @param [in] param指向消息的参数。
* @param [in]接收任务实例的dest_id ID(可能未使用)。
* @param [in]发送任务实例的src_id ID。
* @return如果邮件被消耗或不。
****************************************************************************************
* /
Static int gattc_write_cmd_ind_handler(ke_msg_id_t const msgid,
struct gattc_write_cmd_ind const * param,
ke_task_id_t const dest_id,
ke_task_id_t const src_id)
{
uint8_t char_code = sample128_err_char;
uint8_t status = prf_app_error;
if (KE_IDX_GET(src_id) == sample128_env.con_info.conidx); / /采样
{
If (param->handle == sample128_env. conf . conf . conf . conf . conf . conf . conf . conf . conf . conf . conf . conf . conf . conf。sample128_shdl + SAMPLE128_1_IDX_VAL)
{
char_code = SAMPLE128_1_CHAR;
}
if(param-> handle == sample128_env.sample128_shdl + sample128_2_idx_cfg)
{
char_code = sample128_2_cfg;
}
If (param->handle == sample128_env. conf . conf . conf . conf . conf . conf . conf . conf . conf . conf . conf . conf . conf . conf。sample128_shdl + SAMPLE128_3_IDX_VAL)
{
char_code = SAMPLE128_3_CHAR;
}
if(char_code == sample128_1_char)
{
//保存DB中的值
Attmdb_att_set_value (param->句柄,sizeof(uint8_t), (uint8_t *)¶m->值[0]);
如果(param->最后)
{
sample128_send_val (param - >值[0]);
}
status = prf_err_ok;
}
else if (char_code == sample128_cfg)
{
/ /写价值
uint16_t ntf_cfg;
//在检查前提取值
ntf_cfg = co_read16p(&param-> value [0]);
//仅更新配置如果停止或通知启用的值
if((ntf_cfg == prf_cli_stop_ntfind)||(ntf_cfg == prf_cli_start_ntf))
{
//保存DB中的值
Attmdb_att_set_value (param->句柄,sizeof(uint16_t), (uint8_t *)¶m->值[0]);
//保存环境中的信息
if (ntf_cfg == PRF_CLI_START_NTF) / /将文件上传
{
// Ntf cfg位设置为1
sample128_env.feature | = prf_cli_start_ntf;
}
其他的
{
// Ntf cfg位设置为0
sample128_env。& = ~ PRF_CLI_START_NTF特性;
}
status = prf_err_ok;
}
}
否则if(char_code == sample128_3_char)
{
//保存DB中的值
Attmdb_att_set_value (param->句柄,sizeof(uint8_t), (uint8_t *)¶m->值[0]);
如果(param->最后)
{
sample128_send_val (param - >值[0]);
}
status = prf_err_ok;
}
}
//发送写回复
ATTS_WRITE_RSP_SEND(Sample128_ENV.CON_INFO.CONIDX,PARAM->句柄,状态);
return(ke_msg_consumed);
}
/ **
****************************************************************************************
* @brief断开指示样本128。
* @param[in] msgid收到的消息的Id。
* @param [in] param指向消息的参数。
* @param[in] dest_id接收任务实例ID
* @param [in]发送任务实例的src_id ID。
* @return如果邮件被消耗或不。
****************************************************************************************
* /
msgid静态int gap_disconnect_ind_handler(ke_msg_id_t const,
struct gapc_disconnect_ind const * param,
ke_task_id_t const dest_id,
ke_task_id_t const src_id)
{
//检查连接句柄
if (KE_IDX_GET(src_id) == sample128_env.con_info.conidx); / /采样
{
//在任何情况下,通知APP断开连接
sample128_disable ();
}
return(ke_msg_consumed);
}
/ *
*全局变量定义
****************************************************************************************
* /
//禁用状态处理器定义。
const struct ke_msg_handler sample128_disabled [] =
{
{SAMPLE128_CREATE_DB_REQ, (ke_msg_func_t) sample128_create_db_req_handler},
{SAMPLE128_CREATE_DB_REQ3, (ke_msg_func_t) sample128_create_db_req_handler},
};
//空闲状态处理器定义。
Const struct ke_msg_handler sample128_idle[] =
{
{SAMPLE128_ENABLE_REQ, (ke_msg_func_t) sample128_enable_req_handler},
};
//连接状态处理器定义。
Const struct ke_msg_handler sample128_connected[] =
{
{gattc_write_cmd_ind,(ke_msg_func_t)gattc_write_cmd_ind_handler},
(ke_msg_func_t) sample128_upd_char2_req_handler}, {SAMPLE128_UPD_CHAR2_REQ
(ke_msg_func_t) gattc_write_cmd_ind_handler3}, {GATTC_WRITE_CMD_IND3
};
//默认状态处理器定义
const struct ke_msg_handler sample128_default_state [] =
{
(ke_msg_func_t) gap_disconnnect_ind_handler}, {GAPC_DISCONNECT_IND
};
///指定每个输入状态的消息处理程序结构。
const struct ke_state_handler sample128_state_handler [sample128_state_max] =
{
[sample128_disabled] = ke_state_handler(sample128_disabled),
[sample128_idle] = ke_state_handler(sample128_idle),
[SAMPLE128_CONNECTED] = KE_STATE_HANDLER (SAMPLE128_CONNECTED),
};
///指定所有状态共有的消息处理程序。
const struct ke_state_handler sample128_default_handler = ke_state_handler (sample128_default_state);
///定义所有任务实例的状态的位置持有者。
ke_state_t sample128_state [SAMPLE128_IDX_MAX] __attribute__(((“retention_mem_area0”)部分,zero_init));
# endif / / BLE_SAMPLE128
嗨Sam123,
你不能有
{gattc_write_cmd_ind3,(ke_msg_func_t)gattc_write_cmd_ind_handler3}
.您需要处理每个服务特征gattc_write_cmd_ind.
.通过检查消息参数的句柄值,可以看出每个特征之间的区别
帕纳
在处理函数中。它具有以下结构///通知对等设备已请求修改数据库。
struct gattc_write_cmd_ind.
{
///处理必须写入的属性
UINT16_T手柄;
///数据长度写入
uint16_t长度;
//数据写入的偏移量
uint16_t抵消;
///如果目标任务为true,则返回写响应命令。
BOOL反应;
///通知它是多个准备写请求的最后一个请求。
BOOL最后;
///在属性数据库中写入的数据
uint8_t值(__ARRAY_EMPTY);
};
换句话说,检查param->句柄的值。如。,可以为属性(特征值)创建开关大小写比较:
Switch (param->handle - your_profile_env.start_handle) {
案例CHAR1:休息;
案例CHAR2:休息;
}
对不起,这有点匆匆写。没有测试,但我希望你有一些照片
谢谢!
你好,
我只能在服务中添加8个特征,当我添加第9个时,它不运行,有人尝试过这个?
嗨redbear,
这取决于你在服务中添加的特征。软件支持每个服务32个属性。
由于MT_dialog
嗨MT_dialog,
nb_att_16 = 1 + 9 + 9;// 1svc,9char,9user_desc
nb_att_32 = 0;
nb_att_128 = 9;// val Att.
我大致了解,但28 <32。
嗨redbear,
你是什么意思它没有运行,你可以调试应用程序,以便查看它是否通过app_adv_start()函数?
您可以在一个服务中拥有9个特征。请检查一下你的代码复制粘贴很容易出错误。
由于MT_dialog
嗨MT_dialog,
“它没有运行”意味着我看不到闪烁。
首先,我编写宏开关以添加两个字符,可能会复制和粘贴您所说的错误,
但现在我添加了两个宏开关,只添加了第8个。只加第9个,好的,这意味着代码是正确的。
加起来,我从浅蓝色看不清。
用我们的吗?
可以使用其他方式调试?
我觉得用户描述字符串有点长,
我现在缩短了这是工作!
但是极限是什么呢?
嗨,安东尼,
我没有得到它,你缩短了什么和工作?
由于MT_dialog
嗨MT_dialog,
我更改了char_user_description长度,如下所示:
旧:
static const uint8_t beacon_user_desc_1[] = "Beacon UUID";
static const uint8_t beacon_user_desc_2 [] =“主要标识ID”;
static const uint8_t beacon_user_desc_3[] = "次要身份ID";
...直到beacon_user_desc_9
新一:
static const uint8_t beacon_user_desc_1[] = "UUID";
static const uint8_t beacon_user_desc_2[] = "Major";
static const uint8_t beacon_user_desc_3[] = "次要";
...
它有效。
嗨antony42,
您正在改变数据库或广告字符串的特征吗?广告字符串的最大有效载荷固定为29字节。这和数据库没有关系。
由于MT_dialog
嗨MT_dialog,
数据库的特征,
ATT_DESC_CHAR_USER_DESCRIPTION,我把描述字符串缩短了,一个服务中有9个特征是可以的。
你好,
我添加了塞尔维护和28个特征,工作正常,
但我添加了另外两个特征,我可以从浅蓝色看到,但不能连接,
为什么一个最多的项目中的特点是多少?
嗨,安东尼,
除了在前一篇文章中提到的32个属性的限制,理论上,特征数量没有其他限制。唯一的限制是数据库的内存大小,由db_heap_size定义。
由于MT_dialog
嗨MT_dialog,
我试过了,还是没用。
首先我增加了db_heap_sz的大小,甚至高达1024 * 2,但不起作用,然后我增加了三雪石三,没什么改变。
#define db_heap_sz(1024 + 256)
#define env_heap_sz 328.
#define msg_heap_sz 1312.
#define non_ret_heap_sz 1024.
Undef下列之一,一切正常,添加这三个信标将不工作。
cfg_beacon_config_struct.
cfg_prf_spotar.
CFG_PRF_ANTHONY
什么可能导致这个问题?
非常感谢,
安东尼。
嗨antony42,
当你在#undef任何#定义中提到你的设备正在用你想要的30个特性做广告?
由于MT_dialog
嗨MT_dialog,
当我#def这三个时,我看不到浅蓝色,
撤销其中一个,没问题。
顺便问一下,你用skype吗?(id: electronicfan)
谢谢,
安东尼
嗨antony42,
对话框本地团队的成员将尽快与您联系,以支持您。
由于MT_dialog