亲爱的Dialog_Support,
我正在使用BLE外设作为我项目的参考代码。BLE外围代码正在作为广告设备工作,但我想将其配置为扫描设备(中央)。现在我在现有示例中完成了以下更改。
1.在user_config.h
static const结构gapm_configuration user_gapm_conf = {
///设备角色:中央,外设,观察者或广播公司
.role = GAP_CENTRAL_MST,
struct scan_configuration {
/ / /操作的代码。
uint8_t代码;
///自己的BD地址源
UINT8_T ADDR_SRC;
///扫描间隔
uint16_t间隔;
///扫描窗口大小
uint16_t窗口;
/ / /扫描模式
uint8_t模式;
///扫描过滤策略
uint8_t filt_policy;
///扫描副本过滤策略
uint8_t filter_duplic;
};
static const struct scan_configuration user_scan_conf = {
/ / /操作的代码。
.code = GAPM_SCAN_PASSIVE,
///自己的BD地址源
.addr_src = GAPM_PUBLIC_ADDR,
///扫描间隔
.Interval = 10,
///扫描窗口大小
.window = 5,
/ / /扫描模式
.mode = gap_gen_discovery,
///扫描过滤策略
.filt_policy = SCAN_ALLOW_ADV_ALL,
///扫描副本过滤策略
.filter_duplic = SCAN_FILT_DUPLIC_EN
};
2.在user_callback_config.h
static const struct app_callbacks user_app_callbacks = {
.app_on_connection = user_app_connection,
.app_on_disconnect = user_app_disconnect,
.app_on_update_params_rejected =零,
.app_on_update_params_complete = null,
.app_on_set_dev_config_complete = default_app_on_set_dev_config_complete,
.app_on_adv_nonconn_complete =零,
.app_on_adv_undirect_complete = user_app_adv_undirect_complete,
.app_on_adv_direct_complete =零,
.app_on_db_init_complete = default_app_on_db_init_complete,
.app_on_scanning_completed = user_on_scanning_completed,
.app_on_adv_report_ind = user_on_adv_report_ind,
#if(ble_app_sec)
.app_on_pairing_request = null,
.app_on_tk_exch_nomitm = null,
.app_on_irk_exch = null,
.app_on_csrk_exch =零,
.app_on_ltk_exch = null,
.app_on_pairing_succeded =零,
.app_on_encrypt_ind =零,
.app_on_mitm_passcode_req =零,
.app_on_encrypt_req_ind =零,
.app_on_security_req_ind =零,
# endif / / (BLE_APP_SEC)
};
3.我已经在user_peripheral.c文件中添加了这些函数
无效user_on_scanning_completed (uint8_t status)
{
if(status == gap_err_canceled)
{
app_easy_gap_start_connection_to ();
connection_timer = app_easy_timer(user_con_timeout,user_gapm_cancel);
}
其他的
{
user_scan_start ();
}
返回;
}
静态void user_scan_start(void)
{
struct gapm_start_scan_cmd * cmd = ke_msg_alloc(gapm_start_scan_cmd,
TASK_GAPM TASK_APP,
gapm_start_scan_cmd);
cmd-> op.code = user_scan_conf.code;
cmd-> op.addr_src = user_scan_conf.addr_src;
cmd-> interval = user_scan_conf.interval;
cmd窗口- > = user_scan_conf.window;
cmd-> mode = user_scan_conf.mode;
cmd - > filt_policy = user_scan_conf.filt_policy;
cmd - > filter_duplic = user_scan_conf.filter_duplic;
//发送消息
ke_msg_send(cmd);
//我们现在可以连接
ke_state_set(task_app,app_connectable);
}
我已经在受尊重的文件中尝试了上述更改,但当我运行程序时,它进入nmi_handler.c。
我错过了什么?或者我做错了什么配置?请使用适当的步骤指导我配置BLE外设作为中央。
先感谢您。
与问候,
Ruchi Patel.
嗨Ruchi Patel,
当设备准备启动(配置、设置和准备启动)时,广告或扫描命令从.default_operation_adv中获取,因此在该钩子中,您应该在准备启动时放置您希望设备做的任何事情。因此,我假设如果您在钩子中放置user_start_scan()函数,而不是user_app_adv_start(),您的设备将启动扫描过程。
谢谢mt_dialog.
谢谢你的建议。
我已经尝试了上述解决方案,但我仍然得到相同的问题。当我运行程序时,它进入nmi_handler.h。我在user_config.h, user_callback_config.h和user_peripheral.c文件中所做的上述配置是正确的吗?请提供一个详细的解决方案或演示中心功能的示例代码。
与问候,
Ruchi Patel.
嗨Ruchi Patel,
如果user_scan_start()函数被触发,检查调试器,如果被触发,也检查完成回调函数,以及你得到的错误代码是什么,如果它是一个GAP_ERROR_TIMEOUT,你在大约8秒的活动后得到它,这意味着你正在扫描,当你到达扫描过程的完成回调时,NMI被触发。关于你粘贴的文件和代码,我没有看到任何错误,你得到的NMI意味着代码停留在一个指令中相当长一段时间(这可能是主循环中的WFI(),如果你有睡眠禁用,没有BLE活动超过2.6秒,这是看门狗可以测量的最大时间)。就示例而言,我认为您已经熟悉了主机端的DSPS代码。
谢谢mt_dialog.
你好MT_Dialog,
感谢您的回复。现在我可以将ble_app_peripheral与我的中心设备连接起来。下一个任务是从中心设备发送一个命令到ble_app_peripheral来切换LED。我可以使用自定义配置文件特征发送命令ble_app_peripheral从中心设备,因为我们正在做从BLE扫描仪应用程序?
与问候,
Ruchi Patel.
嗨Ruchi Patel,
与中心的自定义配置文件交互需要在中心端实现,因为当设备连接时,中心应该启动发现过程,以发现可用的服务和特征。在中央发现外围设备的特征之后,它就可以开始写和读数据,因为中央需要知道应该在外围设备上读或写的句柄。简而言之,是的,只要特征是可写的,您就可以将数据写入自定义特征,但您必须实现中心端的发现过程,然后才能写入数据。DSPS主机端实现DSPS配置文件的发现过程,可以作为参考。
谢谢mt_dialog.
感谢您的快速回复。
我需要实现自定义服务和特征的发现过程。我没有找到关于发现过程的明确步骤。我也经历了DSP榜样,但由于其复杂性,我无法妥善理解它。
1.你能详细列出开示程序的步骤吗?
2.参考DSPS,哪些文件发现的相关信息被提及?
3.哪些功能可用于发现定制服务和特征?
与问候,
Ruchi Patel.
嗨Ruchi Patel,
1.我无法提及在这个论坛上提及发现程序的每一个细节,你应该检查这个规范,以便获得广泛的发现程序,我可以提到关于这个过程的一些关键步骤,还可以提到一些关键步骤在DSPS源代码中为您提供几个代码片段,以便遵循。此外,一个非常有用的文档将是RW-BLE-GATT-IS.PDF文档包括应向堆栈和从堆栈发出的所有命令/响应,以便完成发现过程。
谢谢mt_dialog.
你好Dialog_Support,
感谢客户端配置步骤。我跟着你的步骤,但我得到了错误。
1 ... \ src \ custom_profile \ user_custs_config.c(47):错误:#29:预期表达式
struct attm_desc_128 custs1_att_db [cust1_idx_nb] = {};
Const struct cust_prf_func_callbacks cust_prf_funcs[] =
{
#if(ble_custom1_server)
{task_custs1,
custs1_att_db,
cust1_idx_nb,
#if(ble_app_present)
app_custs1_create_db app_custs1_enable,
#别的
空,空,
#万一
custs1_init,零
},
#万一
#if(ble_custom2_server)
{task_custs2,
空值,
0,
#if(ble_app_present)
app_custs2_create_db app_custs2_enable,
#别的
空,空,
#万一
custs2_init,零
},
#万一
{task_none,null,0,null,null,null,null},//不要移动。必须始终持久
};
上面的函数我已经在user_custs_config.c中声明
在user_custs1_def.h中
extern struct attm_desc_128 custs1_att_db [cust1_idx_nb];
2.我应该写什么而不是#if(ble_sps_client)
与问候,
Ruchi Patel.
嗨Ruchi Patel,
1.显然您已将DSP主机放在DSPS中,Custs1.h包括为包括自定义配置文件,并且您尚未将任何内容声明为数据库,请检查例如BLE_APP_PERITIVELAL示例并检查该数组中的内部内部。
2.但是我不理解第二个问题,这只是为了包含SPS客户端功能的一个定义。
谢谢mt_dialog.
非常感谢你的步骤。它帮助了我们很多。经过很长一段时间,我能够发现服务和特点。现在我的下一个任务是在客户机和服务器之间进行数据传输。你能告诉我哪个自定义函数启动数据传输程序吗?
感谢致敬,
Ruchi Patel.
嗨Ruchi Patel,
你的意思是,你想从核心到外围的特征来表演写字吗?您可以使用DSP示例实现的PRF_Gatt_Write()命令,关闭函数该函数向堆栈发送特定消息,该堆栈是Gattc_Write_cmd,您将能够在RW-BLE-GATT中找到额外的信息.pdf在支撑场上。
谢谢mt_dialog.
嗨mt_dialog,
是的,我想从核心写入外围的特征。
如您建议使用“PRF_Gatt_Write()”命令,我在DSP示例中检查了它。据我所知,有一个名为“sps_client_data_tx_req_handler”的特定处理程序,它使用“prf_gatt_write()”命令处理传输请求。因此,每当中央提升写(数据TX)请求时,应理想地调用处理程序。但是,我不明白如何调用tx_req_handler。我究竟究竟是如何调用内置函数或写自己的函数,这将调用“sps_client_data_tx_req_handler”?
感谢致敬,
Ruchi Patel.
嗨Ruchi Patel,
你看到的处理程序是DSPS引用应用程序的实现,如果你想通过发送消息和通过消息处理事件而不是直接调用一个函数调用来触发写入事件,你可以实现类似DSPS项目的东西。当DSPS希望发送数据时,它实现一个名为SPS_CLIENT_DATA_TX_REQ的消息并将其发送到堆栈,然后堆栈一旦消息被处理就会触发相应的处理程序,SPS_CLIENT_DATA_TX_REQ消息通过user_send_ble_data()函数发送,由该消息触发的处理程序声明在sps_client_connected数组中的sps_client_task.c文件中,您可以将其作为一个引用。
谢谢mt_dialog.
.
嗨mt_dialog,
最初,我也考虑使用“user_send_ble_data()”功能,用于通过ble发送数据。此函数要求数据和data_length作为参数。然后它将该数据发送到预定的特征。但在我的情况下,我不想始终将我的数据发送到固定的特征。
例如:
假设我有一个名为“cust_service_1”的自定义服务。在该服务中,我有一些自定义特征,如“LED_1_Brightness”,“LED_2_Brightness”等,它具有没有附加的响应属性。
现在,我想向“LED_1_Brightness”的特征发送0x57。
所以我所遇到的是一个简单的功能,接受以下参数:
Uint8_t data[1] = {0x57};
Uint8_t length = 1;
void my_function_to_send_data(Service_Namecust_service_1,charecteristic_name.LED_1_BRIGHTS,Data_Value*数据,data_length.长度);
由于“user_send_ble_data()”功能仅接受数据和数据长度;如何确保我的数据被发送到我想要的正确特征?我应该附加用于服务名称和特征名称的其他参数,如上面的示例所示?或者在某些变量中首先在外部设置外部的任何其他方法,然后调用“user_send_ble_data()”功能,以便将数据发送到所需的特性?请帮我解决一下这个。
感谢致敬,
Ruchi Patel.
嗨Ruchi Patel,
如上所述,user_send_ble_data()是DSP使用的函数,它不是通用功能,以便将数据发送到服务上可用的任何特征。实际发送数据的处理程序是sps_client_data_tx_req_handler()和实际发送数据的函数是prf_gatt_write()。由于这是实际发送在另一侧数据的函数,因此这也是扭动数据将要发送的特征的句柄的函数。因此,通过使用相应的句柄参数调用prf_gatt_write()将更新正确的特征值。在DSP的情况下,sps_client_data_tx_req_handler始终使用的特征句柄是spsc_env-> sps.chars [spsc_srv_rx_data_char] .val_hdl。
谢谢mt_dialog.
嗨mt_dialog,
正如您之前评论中正确提及的那样,sps_client_data_tx_req_handler ()是实际发送数据的函数。我想要一个简单的应用程序,其中用户按下一个按钮,它触发一个中断例程。在中断例程中,我想调用sps_client_data_tx_req_handler ()函数发送一些数据。该函数需要以下参数:
静态int sps_client_data_tx_req_handler(ke_msg_id_t const msgid,
struct sps_client_data_tx_req const * param,
ke_task_id_t const dest_id,
ke_task_id_t const src_id)
但是,我没有太多了解在每个参数中传递的内容,以便将我的数据发送到特定的特征。你能给我更多的洞察力吗?
感谢致敬,
Ruchi Patel.
嗨Ruchi Patel,
您可以在sps_client_data_tx_req strult中添加额外的成员,并通过该成员来传递您想要的特征,因此当您分配SPS_CLIENT_DATA_TX_REQ消息时,您还应通过您想要执行写入的特性,例如该结构目前具有三个成员(inhdl,长度和数据),在发送消息之前分配消息,因此添加一个额外的成员,该成员将保存您想要编写的特征。当调用SPS_CLIENT_DATA_TX_REQ_HANDLER()而不是PRF_GATT_WRITE()中的固定值时,您可以通过SPS_CLIENT_DATA_TX_REQ结构中添加的额外成员的值。
谢谢mt_dialog.
嗨mt_dialog,
谢谢你的帮助。我成功地将数据写入特定的特征。我用略有不同的方法比你建议的方法略有不同,但是因为它做了我不介意的工作。以下是我的程序所做的摘要:
因此,我目前可以将数据从1个BLE客户端发送到1个BLE外围设备。请注意,它们都始终在整个会话中连接。
现在,我的项目仍然存在最后一步。有1个客户端(我正在努力),有4个外围设备。我有4个不同的用户输入按钮,并且根据按下的按钮,我想从我的BLE客户端发送数据到特定的BLE外围设备。所以这里是我现在想要实现的示例流程:
请注意,我不希望同时连接,我只需要连接到一个特定的设备,发送数据,然后断开。
我需要紧急实施这一点。所以我希望一个具体的详细解决方案。请帮忙..!!
提前谢谢,
Ruchi Patel.
嗨Ruchi Patel,
你的用例非常具体,我不能预见或记录在这个论坛上的每一个具体细节,请检查我下面的建议。
为了通过中心连接到外设,外设需要进行广告(广告是为了让外设接收连接请求,而不仅仅是为了让中心扫描设备),那么,您是假设您的外围设备将在那里并发出连接请求而不需要扫描过程,还是您也将进行扫描?我认为您将为您的设备使用公共地址,因为规范定义的静态地址应该在每次重启外设时更改,并且随机解析需要与每个单独的设备绑定。假设你使用的是公共地址这些地址是你的中心所知道的据我所知有两种选择:
1)假设设备已经有和没有扫描过程从中央,当按钮被按下可以发出一个连接通过app_easy_gap_start_connection_to_set()和app_easy_gap_start_connection_to()以启动连接过程中,但同样,你必须确保设备在做广告,如果没有,连接过程不会超时,如果没有人回应,你就必须通过应用程序超时。因此,一旦设备连接起来,发现过程就会开始,设备会做它连接时做的任何事情。一旦交换了数据,就可以调用app_easy_gap_disconnect()以断开外围设备与中心设备的连接,当连接发生时,禁用并清理概要文件结构(取决于您如何实现自定义概要文件)。断开连接请求可以通过外设发出,也可以通过中央发出。
2)关于第二个选项,你可以按下按钮并在一个全局变量中存储相应的bd地址,你想要连接到,这样当扫描过程返回特定的地址指示回调,你可以发出连接请求。除此之外,其余的都保持不变。
谢谢mt_dialog.
嗨mt_dialog,
我的最终目标是设计一个可编程BLE Remote.用于控制多个调光灯。因此,遥控器最初保持在外设模式,并与android应用程序连接。然后应用程序会告诉哪些按钮应该执行哪些功能。BLE远程仍然处于外设模式。但是一旦用户按下一个按钮,假设是button-1,它就会执行以下操作:
我仍然没有想到如何在运行时进行角色切换。所以目前,我的可编程BLE遥控器始终处于中央模式。
为了操作我的BLE遥控器上的按钮按下调光器,我写了以下代码:
bool gboolD1Connect = false;
void user_button_pressed(void)
{
gboold1connect = true;
// user_on_adv_report_ind(param-> rapth.adv_addr_type:0x01);
if(dimmer1_connected == true)//仅在整个连接过程完成后才为真
{
uint8_t my_data [1];
my_data [0] = 0 x14;/ /亮度值
uint16_t my_length;
my_length = 1;
//此功能将数据从交换机发送到调光器
prf_gatt_write(con_info,val_hdl,(uint8_t *)my_data,sizeof(uint8_t)* my_length,gattc_write);
}
}
//这是一个内置函数,我修改了它,使它只有在gboolD1Connect为true时才连接到dimmer-1
void user_on_adv_report_ind(struct gapm_adv_report_ind const * param)
{
if(gboold1connect)
{
if(!memcmp(&param-> report.data [3],user_advertise_data,user_advertise_data_len)))
{
ARCH_PRINTF(“连接%02x%02x%02x%02x%02x%02x”,
param - > report.adv_addr.addr [5],
param - > report.adv_addr.addr [4],
param - > report.adv_addr.addr [3],
param - > report.adv_addr.addr [2],
param-> eport.addr.addr [1],
param - > report.adv_addr.addr [0]);
app_easy_gap_start_connection_to_set (param - >报告。adv_addr_type (uint8_t *)参数- > report.adv_addr。addr, MS_TO_DOUBLESLOTS (12.5));
user_gapm_cancel ();
}
}
}
上述实施方式,但它有问题。当我第一次按下遥控器的按钮,它会设置gboold1connect.返回true,然后退出函数,因为dimmer1_connection.还不是真的。之后,在后台,当user_on_adv_report_ind ()运行,它将与Dimmer-1连接。连接完成后,将发现所有服务,ChareCertics和描述符。最后,就在描述符发现完成后,我在内置函数中插入了一行以制作dimmer1_connected = true.整个过程需要2-3秒。3秒后,当我再次按下按钮时,它最终会进入dimmer1_connected = = true分段并发送亮度值。如你所见,我得按两次按钮(1号用于连接,2号用于发送数据)能够发送数据。这个问题发生,因为每当我给一个命令,以连接到一个调光器设备(通过制作gboold1connect.为真,或使用app_easy_gap_start_connection_to_set()功能),我必须退出中断例程并允许主程序运行,以便发生实际连接。但由于我退出了我的中断例程,我将如何重新输入它并在连接外围设备时发送数据?
所以我有两个问题需要帮助:
提前感谢你的帮助,
Ruchi Patel.
嗨,前腿,
我建议您看一下CodeLess示例应用程序(https://support.dialog-seminiondionder.com/connectivity/reference-design/smartbond-codeless-serial-link.).
此应用程序演示中央和外围设备的各个方面以及如何在角色之间切换。
/ mhv.
嗨MHv,
你给出的建议真的很棒,我很高兴能够继续实现。但是看文档,我明白它的实施需要两个主要的东西:
(1)DA1458x BLE模块,用于命令固件
(2)第三方单片机控制BLE模块
但在我的情况下,我在单个芯片上只有一个DA1458x BLE模块。我没有特权在同一芯片上添加另一个第三方控制器以控制BLE模块。
我希望我的DA1458X BLE模块具有固件,然后逻辑调用AT命令本身的序列以独立方式运行。
那么,是否有可能实现AT命令方法,而不需要任何外部控制机制,如终端,android应用程序或第三方控制器?
谢谢,
Ruchi Patel.
嗨,前腿,
实现应该只是用于指导如何完成不同的事情。通过运行代码,并使用调试器逐步执行各种命令,您将能够了解看似复杂的任务是如何实现的。这里有一些例子:
CodeLess实现是使用AT命令来触发不同的动作,但是应该可以编写自己的逻辑来驱动相同的功能(您可能只是将适用的代码段从CodeLess复制到您的项目中)。
/ mhv.
我不能发现特征,但我可以发现服务,我的项目基于ble_app_peripheral
你能告诉我如何发现特征吗?
非常感谢!
嗨chenpenglai,
这个论坛线程很古老,所以我建议你创造一个新的。如果您不知道您应该遵循的程序以创建新的论坛线程,请告诉我,我将为您提供所有步骤。所以请创建一个新的论坛帖子和复制你的问题。
关于你的问题,你使用的是哪个版本的SDK ?你在SDK的ble_app_peripheral例子中做了任何修改吗?
谢谢,PM_Dialog