19个帖子/ 0新
最后发表
Sam123
离线
最后看到:6年2个月前
加入:2015-02-19 05:43
新服务之外

嗨对话小组,

我遵循AN-B-029文件,并成功地为我的服务添加了新的特性(通知/阅读)。然而,当我添加一个新的特征(写/读),我不能在其中写。每当我写东西的时候,它都显示“写作失败”。是否有AN-B-029这样的文件可以提供步骤?我被困在这里了。

但是,现在我想向现有的sample128添加新的服务。我的新服务应该包含这两个特点。我应该如何实现它?请帮助。

最诚挚的问候,
萨姆

关键词:
设备:
VesaN
离线
最后看到:5年11个月前
格鲁鲁 主
加入:2014-06-26 08:49
你好山姆,

你好山姆,
概要文件将会收到GATTC_WRITE_CMD_IND消息。您需要在配置文件中附加此消息的处理程序函数,您可能已经这样做了吗?你能分享一下你的处理器的功能吗profilename_task.c

Sam123
离线
最后看到:6年2个月前
加入:2015-02-19 05:43
/**

嗨VesaN,

我正在尝试添加一个写/读特征到现有的样品128。

/**
****************************************************************************************

* @file sample128_task.c

* @brief Sample128任务实现。

* @brief 128 UUID服务。示例代码

*版权所有(C) 2013 Dialog yabo国际娱乐Semiconductor GmbH及其附属公司,未发表的作品
*本计算机程序包括机密、专有信息,属商业机密
* Diayabo国际娱乐log Semiconductor 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消息是否被消费。
****************************************************************************************
*/
Static 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_CHAROMATION;
UINT16_T ATT_DECL_CFG = ATT_DESC_CLIENT_CHAR_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…4uid16属性声明类型
nb_att_32 = 0;// 0 UUID32属性声明类型
nb_att_128 = 3;// 2 UUID128属性声明类型

状态= attmdb_add_service(&(sample128_env.sample128_shdl),
TASK_SAMPLE128,
nb_att_16,
nb_att_32,
nb_att_128,
78
);

status == ATT_ERR_NO_ERROR

/ /添加的主要服务属性 /////////////////////////////////////////////////////////////////
status = 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,启用),//权限
&(sample128_env.sample128_shdl) //属性句柄
);

//添加主服务属性的值(自定义UUID)
状态= attmdb_att_set_value(sample128_env. value)sample128_shdl, //属性句柄
ATT_UUID_128_LEN, //服务的128位UUID
(uint8_t *) sample128_svc。uuid //服务的uuid
);

/ /特点1 : ////////////////////////////////////////////////////////////////////////////////

//将特征声明属性添加到数据库
status = 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,启用),//权限
&(char_hdl)//处理特征声明
);

//添加特征值声明属性到数据库
status = attmdb_add_attribute(sample128_env.sample128_shdl,
sizeof(uint8_t), //数据大小= 1字节
ATT_UUID_128_LEN,//自定义声明类型大小= 128bit
(uint8_t *)&sample128_1_val.uuid,// uuid的特征值
PERM(RD, ENABLE) | PERM(WR, ENABLE),//权限
&(val_hdl) // value属性句柄
);

//保存特征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 : //////////////////////////////////////////////////////////////////////////////////

//将特征声明属性添加到数据库
status = 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,启用),//权限
&(char_hdl)//处理特征声明
);

//添加特征值声明属性到数据库
status = 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, ENABLE) | PERM(NTF, ENABLE),//权限
&(val_hdl) // value属性句柄
);

//保存特征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);

//将客户端配置声明属性添加到数据库
status = 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) //值属性句柄
);

/ /写特点3 : ////////////////////////////////////////////////////////////////////////////////

//将特征声明属性添加到数据库
status = 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,启用),//权限
&(char_hdl)//处理特征声明
);

//添加特征值声明属性到数据库
status = 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, ENABLE) | PERM(WR, ENABLE),//权限
&(val_hdl) // value属性句柄
);

//保存特征3的句柄
memcpy(sample128_3_char.attr_hdl,&val_hdl,sizeof(uint16_t));

//设置特征3的初始值
状态= attmdb_att_set_value(char_hdl, sizeof(sample128_3_char), (uint8_t *)&sample128_3_char);

////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

/ /禁用sample128服务
attmdb_svc_set_permission (sample128_env。禁用sample128_shdl烫(SVC));

//如果我们在这里,数据库已经成功完成,转到空闲状态
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 - >状态=状态;
ke_msg_send (cfm);

返回(KE_MSG_CONSUMED);

/**
****************************************************************************************
启用Sample128角色,在连接后使用。
* @param[in] msgid收到的消息的Id。
* @param[in] param指向消息参数的指针。
* @param[in] dest_id接收任务实例ID
* @param[in] src_id发送任务实例ID。
* @return消息是否被消费。
****************************************************************************************
*/
Static 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);

//检查提供的连接是否存在
如果(sample128_env.con_info。conidx = = GAP_INVALID_CONIDX)

//连接不存在,请求被拒绝
prf_server_error_ind_send (prf_env_struct *) &sample128_env PRF_ERR_REQ_DISALLOWED,
SAMPLE128_ERROR_IND SAMPLE128_ENABLE_REQ);

其他的

// Sample128服务权限
attmdb_svc_set_permission(sample128_env.sample128_shdl,param-> sec_l​​vl);

//将特征值1设置为指定值
attmdb_att_set_value (sample128_env。sample128_shdl + SAMPLE128_1_IDX_VAL,
sizeof (uint8_t), (uint8_t *)参数- > sample128_1_val);

//将特征值2设置为指定值
attmdb_att_set_value (sample128_env。sample128_shdl + SAMPLE128_2_IDX_VAL,
sizeof(uint8_t),(uint8_t *)和param-> sample128_2_val);

//将特征值3设置为指定值
attmdb_att_set_value (sample128_env。sample128_shdl + SAMPLE128_3_IDX_VAL,
sizeof(uint8_t),(uint8_t *)和param-> sample128_3_val);

sample128_env.feature = param->特征;

如果(! sample128_env.feature)

temp = 0;

attmdb_att_set_value(sample128_env.sample128_shdl + sample128_2_idx_cfg,
sizeof (uint16_t), (uint8_t *)临时);

//进入连接状态
ke_state_set (TASK_SAMPLE128 SAMPLE128_CONNECTED);

返回(KE_MSG_CONSUMED);

/**
****************************************************************************************
* @brief更新feature 2的值。如果启用了属性,则向对等体发送通知。
* @param[in] msgid收到的消息的Id。
* @param[in] param指向消息参数的指针。
* @param[in] dest_id接收任务实例ID
* @param[in] src_id发送任务实例ID。
* @return消息是否被消费。
****************************************************************************************
*/

Static 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;

//检查提供的值
如果(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(状态);

返回(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消息是否被消费。
****************************************************************************************
*/
msgid静态int gattc_write_cmd_ind_handler(ke_msg_id_t const,
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.sample128_shdl + sample128_1_idx_val)

char_code = sample128_1_char;

If (param->handle == sample128_env. conf . conf . conf . conf . conf . conf . conf . conf . conf . conf . conf . conf . conf . conf。sample128_shdl + SAMPLE128_2_IDX_CFG)

char_code = SAMPLE128_2_CFG;

if(param-> handle == sample128_env.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 *)和param->值[0]);

如果(param - >最后)

sample128_send_val (param - >值[0]);

状态= PRF_ERR_OK;


否则if(char_code == sample128_2_cfg)

/ /写价值
uint16_t ntf_cfg;

//在检查前提取值
ntf_cfg = co_read16p (param - >值[0]);

//如果stop或notification的值为enable,则只更新配置
if ((ntsf_cfg == PRF_CLI_STOP_NTFIND) || (ntsf_cfg == PRF_CLI_START_NTF)) / / ntsf_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 bit设置为1
sample128_env。功能| = PRF_CLI_START_NTF;

其他的

// ntf cfg bit设置为0
sample128_env.feature&=〜prf_cli_start_ntf;

状态= PRF_ERR_OK;


else if (char_code == SAMPLE128_3_CHAR)

//在DB中保存值
attmdb_att_set_value(param->句柄,sizeof(uint8_t),(uint8_t *)和param->值[0]);

如果(param - >最后)

sample128_send_val (param - >值[0]);

状态= PRF_ERR_OK;


//发送写响应
atts_write_rsp_send (sample128_env.con_info。conidx param - >处理、状态);

返回(KE_MSG_CONSUMED);

/**
****************************************************************************************
* @brief disconnect indication to sample128。
* @param[in] msgid收到的消息的Id。
* @param[in] param指向消息参数的指针。
* @param[in] dest_id接收任务实例ID
* @param[in] src_id发送任务实例ID。
* @return消息是否被消费。
****************************************************************************************
*/
Static int gap_disconnect_ind_handler (ke_msg_id_t const msgid,
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)

//在任何情况下,通知应用程序关于断开连接
sample128_disable();

返回(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[] =

(ke_msg_func_t) gattc_write_cmd_ind_handler}, {GATTC_WRITE_CMD_IND
{sample128_upd_char2_req,(ke_msg_func_t)sample128_upd_char2_req_handler},
(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 __((部分(“保留_mem_area0”),zero_init));

# endif / / BLE_SAMPLE128

VesaN
离线
最后看到:5年11个月前
格鲁鲁 主
加入:2014-06-26 08:49
嗨Sam123,

嗨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);
};

换句话说,检查参数 - >手柄的值。例如,您可以为属性(特征值)创建交换盒比较:
Switch (param->handle - your_profile_env.start_handle) {
案例CHAR1:休息;
案例CHAR2:休息;

抱歉,写得有点匆忙。没有测试,但我希望你能从中得到一些信息

谢谢!

redbear.
离线
最后看到:6年1个月前
加入:2015-01-30 08:50
你好,

你好,
我只能在一个服务中添加8个特征,当我添加第9个时,它没有运行,有人试过吗?

MT_dialog
离线
最后看到:6个月2个星期前
工作人员
加入:2015-06-08 11:34
嗨redbear,

嗨redbear,

这取决于您在您的服务中添加的特征。软件支持每次服务32个属性。

由于MT_dialog

redbear.
离线
最后看到:6年1个月前
加入:2015-01-30 08:50
嗨MT_dialog,

嗨MT_dialog,
nb_att_16 = 1 + 9 + 9;/ / 1 svc, 9 char、9 user_desc
nb_att_32 = 0;
nb_att_128 = 9;/ / val丙氨酸
我大致理解,但是28 < 32。

MT_dialog
离线
最后看到:6个月2个星期前
工作人员
加入:2015-06-08 11:34
嗨redbear,

嗨redbear,

你能调试你的应用程序,看看它是否通过app_adv_start()函数吗?

您可以在一个服务中拥有9个特征。请检查您的代码很容易用复制和粘贴造成错误。

由于MT_dialog

安东尼42.
离线
最后看到:5年10个月前
加入:2015-07-17 08:21
嗨MT_dialog,

嗨MT_dialog,
“它不跑”的意思是我从浅蓝色看不见。
首先我写一个宏开关来添加两个字符,可能像你说的复制粘贴错误,
但现在我添加了两个宏开关,只添加了第8个。只加第9个,好的,这意味着代码是正确的。
加起来,我从浅蓝色看不清。

用我们的吗?
还有其他调试方法吗?

我觉得用户描述字符串有点长,
我现在把它改短了这是工作!
但限制是多少?

MT_dialog
离线
最后看到:6个月2个星期前
工作人员
加入:2015-06-08 11:34
嗨,安东尼,

嗨,安东尼,

我不明白你把什么改短了又管用?

由于MT_dialog

安东尼42.
离线
最后看到:5年10个月前
加入:2015-07-17 08:21
嗨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 const8_t beacon_user_desc_3 [] =“minor”;
...
和它的工作原理。

MT_dialog
离线
最后看到:6个月2个星期前
工作人员
加入:2015-06-08 11:34
嗨antony42,

嗨antony42,

您正在更改数据库的特征或广告字符串?广告字符串具有固定的最大数量为29字节有效载荷。与数据库无关。

由于MT_dialog

安东尼42.
离线
最后看到:5年10个月前
加入:2015-07-17 08:21
嗨MT_dialog,

嗨MT_dialog,
数据库的特点,
ATT_DESC_CHAR_USER_DESCRIPTION,我制作了描述字符串较短,服务中的9个特征可以正常。

安东尼42.
离线
最后看到:5年10个月前
加入:2015-07-17 08:21
你好,

你好,
我一共添加了几种服务和28个特点,效果很好,
但我添加了另外两个特征,我可以从浅蓝色看到,但不能连接,
为什么一个最多的项目中的特点是多少?

MT_dialog
离线
最后看到:6个月2个星期前
工作人员
加入:2015-06-08 11:34
嗨,安东尼,

嗨,安东尼,

除了我在前一篇文章中提到的每个服务限制32个属性之外,理论上在特征的数量上没有其他限制。唯一的限制是数据库的内存大小,它是由DB_HEAP_SIZE定义的。

由于MT_dialog

安东尼42.
离线
最后看到:5年10个月前
加入:2015-07-17 08:21
嗨MT_dialog,

嗨MT_dialog,

我试过了,还是没用。
首先,我增加了DB_HEAP_SZ的大小,甚至达到1024*2,但不工作,然后我增加了三个,没有改变。
#定义DB_HEAP_SZ (1024 + 256)
#定义ENV_HEAP_SZ 328
#定义MSG_HEAP_SZ 1312
#定义NON_RET_HEAP_SZ 1024

Undef下列之一,一切正常,添加这三个信标将不工作。
CFG_BEACON_CONFIG_STRUCT
CFG_PRF_SPOTAR
cfg_prf_anthony.

是什么导致了这个问题?

非常感谢,
安东尼。

MT_dialog
离线
最后看到:6个月2个星期前
工作人员
加入:2015-06-08 11:34
嗨antony42,

嗨antony42,

当你在#undef任何#定义中提到你的设备正在用你想要的30个特性做广告?

由于MT_dialog

安东尼42.
离线
最后看到:5年10个月前
加入:2015-07-17 08:21
嗨MT_dialog,

嗨MT_dialog,
当我定义三个时,我从浅蓝色看不见,
撤销其中一个,没问题。

顺便问一下,你用skype吗?(id: electronicfan)

谢谢,
安东尼

MT_dialog
离线
最后看到:6个月2个星期前
工作人员
加入:2015-06-08 11:34
嗨antony42,

嗨antony42,

来自dialog本地团队的成员将很快与您联系以支持您。

由于MT_dialog