嗨,对话框的团队,
我遵循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”
#if(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)
{
//添加主要服务属性//////////////////////////////////////////////////////////////////////////////////////////////////////////////// //
status = attmdb_add_attribute(sample128_env.sample128_shdl,//属性句柄
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,//自定义声明类型= 128位的大小
(uint8_t *) &sample128_1_val。uuid, //特征值的uuid
perm(rd,启用)|perm(wr,启用),//权限
&(val_hdl)//句柄到值属性
);
//存储特性1的值句柄1
memcpy(sample128_1_char.attr_hdl,&val_hdl,sizeof(uint16_t));
//设置特征1的初始值
status = 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),// data size = 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),//数据大小2bytes(16bit)
ATT_UUID_16_LEN,//客户端配置类型ID的大小
(UINT8_T *)&ATT_DECL_CFG,// 0x2902客户端配置声明类型的UUID
perm(rd,启用)|perm(wr,启用),//权限
&(val_hdl)//句柄value属性
);
// 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,//自定义声明类型= 128位的大小
(uint8_t *) &sample128_3_val。uuid, //特征值的uuid
perm(rd,启用)|perm(wr,启用),//权限
&(val_hdl)//句柄到值属性
);
//存储特征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);
//////// //////////////////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
//禁用示例128服务
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更新特性值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 * param,
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 *)和param-> val);
if((sample128_env.feature&prf_cli_start_ntf))
//通过gatt发送通知
prf_server_send_event((prf_env_struct *)和sample128_env,false,
sample128_env.sample128_shdl + sample128_2_idx_val);
}
别的
{
status = prf_err_invalid_param;
}
if(状态!= 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-> value [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 *)和param->值[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-> value [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
{gattc_write_cmd_ind3,(ke_msg_func_t)gattc_write_cmd_ind_handler3},
};
///默认状态处理程序定义
const struct ke_msg_handler sample128_default_state [] =
{
{gapc_disconnect_ind,(ke_msg_func_t)gap_disconnnect_ind_handler},
};
///指定每个输入状态的消息处理程序结构。
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->句柄的值。如。,可以为属性(特征值)创建开关大小写比较:
交换机(param->句柄 - 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个,确定,这意味着代码是对的。
添加两个,我看不到闪烁。
用keil?
可以使用其他方式调试?
我认为用户DESC字符串有点长,
我现在缩短了这是工作!
但是极限是什么呢?
嗨Antony,
我没有得到它,你缩短了什么和工作?
谢谢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”;
...直到beaccon_user_desc_9.
新一:
static const uint8_t beacon_user_desc_1 [] =“uuid”;
static const uint8_t beaccon_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个特征,工作正常,
但我添加了另外两个特征,我可以从浅蓝色看,但不能连接,
为什么一个最多的项目中的特点是多少?
嗨Antony,
除了在前一篇文章中提到的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任何#defines你已经提到你的设备是广告的,你想要的所有30个特征?
谢谢mt_dialog.
嗨mt_dialog,
当我#def这三个时,我看不到浅蓝色,
其中一个人,工作正常。
顺便问一下,你使用Skype吗?(ID:ElectronicFan)
谢谢,
安东尼
嗨Antony42,
对话框本地团队的成员将尽快与您联系,以支持您。
谢谢mt_dialog.