当我从GPIO中断唤醒时,我想改变UUID的广告的最后一个字节,以指示哪个GPIO唤醒了。使用beacon_v_5.40.2_0项目,我声明回调如下:
void tilt_wakeup_cb(void)
{
if(getBits16(sys_stat_reg,per_is_down))
{
//从外部唤醒GPIO返回GPIO功能
if(development_debug)
GPIO_reservations ();
periph_init();
}
tilt_wkup_flag = 1;
user_advertise_operation();
}
这允许吗?
我计划使用一个标志来确定设备是被GPIO中断唤醒还是从它的计时器唤醒,基于此信息,我计划编辑user_update_ibeacon_adv_string如下:
void user_update_ibeacon_adv_string(struct user_i_beacon_config_tag * data,struct user_beacon_config_tag config)
{
.......
//由DL添加
如果(tilt_wkup_flag) {
if(!gpio_getpinstatus(gpio_port_2,gpio_pin_0)){
数据 - > UUID [15] = 1;
} else if (!GPIO_GetPinStatus (GPIO_PORT_2 GPIO_PIN_1)) {
数据 - > UUID [15] = 2;
}如果(!gpio_getpinstatus(gpio_port_2,gpio_pin_2)){
数据 - > UUID [15] = 3;
}如果(!gpio_getpinstatus(gpio_port_2,gpio_pin_3)){
数据 - > UUID [15] = 4;
}如果(!gpio_getpinstatus(gpio_port_2,gpio_pin_4)){
数据- > uuid [15] = 5;
}如果(!gpio_getpinstatus(gpio_port_2,gpio_pin_5)){
数据 - > UUID [15] = 6;
}
tilt_wk_flag = 0;
其他}{
数据- > uuid [15] = 0;
}
//
.....
}
测试这段代码时,当GPIO被拉低以强制唤醒时,似乎没有做任何事情。
在user_on_init()
我在函数中添加了以下内容:
wkupct_register_callback(tilt_wakeup_cb);
wkupct_enable_irq(0x3f0000,0x3f0000,10);//为p2_0启用中断到p2_5
我修改了回调,如下所示,但仍然没有工作。用jlink跟踪,我可以看到它进入回调函数,但它没有通告新的uuid:
嗨dio,
如果发生唤醒中断,则需要调试此操作,可以在User_update_ibeacon_add()中放置一个断点,并检查代码是否通过该函数,并且如果满足条件以更改广告字符串。
谢谢mt_dialog.
好的,所以在调用user_advertise_operation()中不应该有任何问题;在回调里面吧?收音机和一切都应该打开,它应该在设置我的方式时宣传吗?
只是确保我没有违反一些规则,从GPIO中断唤醒后我们无法宣传。
谢谢,
它不建议直接从醒来开始广告操作虽然这应该工作,可以找到适当的方式开始一个动作在app_button_press_cb ble_app_sleemode检查(),但即使不遵循你的设备应该能够开始广告。例如,如果您直接从app_button_press_cb()调用user_app_adv_start()(我认为这与您已经实现的user_advertise_operation相同)并删除app_easy_wakeup(),设备将发布。但建议使用app_easy_wakeup() API的方式。
谢谢mt_dialog.
当没有GPIO中断发生,设备在定时间隔苏醒时,所有的广告事件都可以在我手机上的信标扫描应用程序上看到。
当GPIO中断确实发生时,踩到代码时,似乎代码会做它应该是什么,但我从未在扫描仪应用中看到正确的值......
不知何故,我认为没有暗示在这种情况下发布的数据。app_easy_gap_non_connectable_advertise_start()用于这两个方案。对于GPIO中断情况,我将cmd-> intv_min和cmd-> intv_max更改为0.或留下这两个空白,无论哪种方式都没有运气。
这是代码。
嗨DLO,
如果代码是什么想做得到中断和数据,你想看到的设备仍然是广告但旧值然后应该错如何放置广告中的数据字符串。在广告字符串被新值更新之后,我假设您停止广告,用新值更新广告字符串,然后使用user_advertise_operation重新启动广告。调用app_easy_gap_non_connectable_advertise_get_active()函数以便SDK填充新消息,然后将自定义数据添加到广告字符串中,要调试它,您可以在代码中一步一步地进行,并检查携带advertise命令的CMD指针是否在代码的某个点包含您想放置的广告数据(检查CMD ->主机。Adv_data数组,并检查你的数据是否被添加到这个数组中)。
谢谢mt_dialog.
所以我测试了你建议的东西,发现了一些奇怪的行为。
使用定时器唤醒时间间隔并禁用WKUPCT时,在User_update_ibeaCon_Adv_String中轮询GPIO,并相应地更新UUID上一个字节。我可以看到扫描仪应用程序中的这些更改,一切都应该工作。
然后我启用wkupct:
wkupct_register_callback(tilt_wakeup_cb);
wkupct_enable_irq(0x3f0000,0x3f0000,10);//为p2_0启用中断到p2_5
在此之后,我不再看到扫描仪应用程序中UUID的最后一个字节的变化。由于使用相同的代码更新广告字符串和广告,我将假定这两个函数工作正常。现在来弄清楚为什么启用wkupct和调用user_advertise_operation不再正常工作。
我尝试追踪CMD->信息 - >主机 - > ADV_DATA,但没有一个值似乎匹配UUID或任何识别的任何识别。
代码仅使用对话框user_advertise_operation();没有修改它。
我修改了对话框代码user_update_ibeacon_adv_string
void user_update_ibeacon_adv_string(struct user_i_beacon_config_tag * data,struct user_beacon_config_tag config)
{
memcpy(Data-> Adv_header,Adv_Header_ibeAcon,2);
数据 - >雷电竞下载app company_id [0] =(UINT8_T)(CONFIG.COMPANY_ID&0xFF);
数据 - >雷电竞下载app company_id [1] =(UINT8_T)(CONFIG.COMPANY_ID >> 8);
data-> data_type = ibeacon_data_type;
数据 - > beacon_length = adv_data_length;
memcpy(data-> uuid,config.uuid,16);
//由DL添加
如果(tilt_wkup_flag) {
if(!gpio_getpinstatus(gpio_port_2,gpio_pin_0)){
数据 - > UUID [15] = 1;
} else if (!GPIO_GetPinStatus (GPIO_PORT_2 GPIO_PIN_1)) {
数据 - > UUID [15] = 2;
}如果(!gpio_getpinstatus(gpio_port_2,gpio_pin_2)){
数据 - > UUID [15] = 3;
}如果(!gpio_getpinstatus(gpio_port_2,gpio_pin_3)){
数据 - > UUID [15] = 4;
}如果(!gpio_getpinstatus(gpio_port_2,gpio_pin_4)){
数据- > uuid [15] = 5;
}如果(!gpio_getpinstatus(gpio_port_2,gpio_pin_5)){
数据 - > UUID [15] = 6;
}
tilt_wkup_flag = 0;
其他}{
数据- > uuid [15] = 0;
}
//
data-> tx_power = config.power;
}
写了wkupct回调:
void tilt_wakeup_cb(void)
{
if(getBits16(sys_stat_reg,per_is_down))
{
//从外部唤醒GPIO返回GPIO功能
if(development_debug)
GPIO_reservations ();
periph_init();
}
if(arch_ble_ext_wakeup_get()))
{
ARCH_SET_SLEEP_MODE(app_default_sleep_mode);//设置默认睡眠模式
ARCH_BLE_FORCE_WAKEUP();//强制摇晃醒来
ARCH_BLE_EXT_WAKEUP_OFF();//禁用外部唤醒
}
tilt_wkup_flag = 1;
user_advertise_operation();
wkupct_enable_irq(0x003f0000,0x003f0000,10);//为p2_0启用中断到p2_5
}
感谢你一如既往的支持!
发现有点信息。
我删除了对user_advertise_operation()的调用;在tilt_wakeup_cb()中,现在在扫描仪应用程序中看到了预期的变化。问题是,它最终会在大约5-10个广告间隔之后被记录。(我将发布间隔设置为5秒,并且在gpio被拉低后大约55秒收到更改的数据包)。
这似乎表明,在将实际事件放在队列中之前需要刷新的广告数据的缓冲区?这似乎是对吗?
你提到我应该停止广告,更新广告字符串,然后重新启动广告。我是否正确地解释这一点,如果我这样做了?
app_easy_gap_advertise_stop ();
user_advertise_operation();
user_advertise_operation()的代码;是默认的对话框:
void user_advertise_operation(void)
{
struct gapm_start_advertise_cmd * cmd;
current_advertising_string = next_advertising_string;
current_advertising_mode = next_advertising_mode;
if (current_advertising_mode == NON_CONNECTABLE_MODE)
{
//获取指向BLE数据包的指针
cmd = app_easy_gap_non_connectable_advertise_get_active();
user_adv_restarted = true;
cmd-> info.host.adv_data_len = user_load_beacon_config(cmd);
//开始发布不可连接的外设。
if(app_connected == ke_state_get(task_app))
{
//不要在连接状态广告不可连接的广告
}
其他的
{
if(user_flash_beacon_config_loaded)
{
cmd-> intv_min = ms_to_bleslots(user_beacon_config.adv_int);
cmd-> intv_max = ms_to_bleslots(user_beacon_config.adv_int);
}
app_easy_gap_non_connectable_advertise_start();
}
}
else if (current_advertising_mode == UNDIRECTED_MODE)
{
//获取指向BLE数据包的指针
cmd = app_easy_gap_undirected_advertise_get_active();
user_adv_restarted = true;
cmd-> info.host.adv_data_len = user_load_beacon_config(cmd);
if(current_advertising_string == suota_advertising_string)
{
update_advertising_data (USER_ADVERTISE_SUOTA USER_ADVERTISE_SUOTA_LEN);
user_update_adv_string (cmd, (void *) dynamic_adv_string SUOTA_ADVERTISING_STRING, USER_ADVERTISE_SUOTA_LEN);
}
//启动广告以获得未确定的外设。
app_easy_gap_undircated_advertise_start();
}
}
关于我想做的事情的背景,请告诉我是否有可能。
当GPIO2_0变低时,我希望发送一个UUID以0结尾的通告。手机应用程序将记录它收到广告的时间。当另一个GPIO变低时,比如GPIO2_1,我想发送一个UUID以1结尾的广告。手机会记录它收到这个信息的时间,并计算两个广告之间的时间差。时差对我处理数据很重要。这是实现这一目标的可靠方法吗?或者是否存在某种限制让我无法实现这个目标?
再次感谢您继续支持。
嗨DLO,
关于在580上更新广告字符串,为了正确地停止,更新广告字符串并再次启动广告,您将不得不发出停止,然后等待堆栈回复广告确实停止了(SDK Will通知您广告已停止使用APP_ONC_ADV_UNDIRECT为无向广告,对于不可连接的广告,还有CORRSPONDING回调 - 请检查鞍骨灯,以便具有比标识更简单的引用)。因此,当堆栈通知您广告已完成并发生回调时,从该回调开始,您可以调用您将更新的新广告字符串的启动。您无法调用停止广告,然后将函数拨打函数开始。我想这是你的主要问题。
关于您试图实施和时间两个广告数据之间的差异,请注意,广告不是一种可靠的方式来发送数据,这意味着无法保证您将发送的数据将到达扫描仪特定的时间,由于在当时在常规摄像机时,其他广告设备和协议之间的协议之间的碰撞,所以通过广告过程没有保证,无论外围设备发送扫描仪是否将最肯定地接收。所以请注意。
谢谢mt_dialog.
谢谢你继续支持。我尝试在鞍骨项目中实现示例,但它仍然没有工作。我不知道还有什么要尝试,但是因为你提到没有保证到达扫描仪的数据,我们需要解决其他问题。
嗨DLO,
是的,并不能保证广告字符串将到达中央,由于拥挤的平均值,范围或噪音,协议没有一个机制,以便中央确认发送的广告字符串已经收到的外设,这只发生在连接期间。
谢谢mt_dialog.