问题描述:
1.DSPS_v_5.150.2工程中变量默认初始值:
const static sleep_state_t app_default_sleep_mode = ARCH_EXT_SLEEP_ON;
app_default_sleep_mode =ARCH_EXT_SLEEP_ON在DSPS工程中的含义是开启扩展睡眠模式,在这种情况下测试透传功能,数据传输是不稳定的。
请问一下DA14583在ARCH_EXT_SLEEP_ON设置成扩展睡眠模式下,DSPS工程的sleep模式逻辑关系是怎么样的?工程在什么情况下进入 EXT_SLEEP模式,什么情况下会被唤醒?
我查看了相关帖子说要设置app_default_sleep_mode =ARCH_SLEEP_OFF,DSPS工程实现和手机APP透传数据时才稳定。
但是我想要实现的功能是
if(条件1 == 真) {
// DA14583进入EXT_SLEEP模式
}else {
// 实现DSPS,及DA14583外设相关操作
}
当DA14583在EXT_SLEEP模式下由按键和BLE event(ke_timer_set)唤醒DA14583
P.S 我调试发现当app_default_sleep_mode =ARCH_EXT_SLEEP_ON的时候,程序运行2s左右就进入LED状态设置了-->GPIO_SetActive( LED_PORT, LED_PIN);
while(1) {
... ...
if ((sleep_mode == mode_ext_sleep) || (sleep_mode == mode_deep_sleep)) {
GPIO_SetActive( LED_PORT, LED_PIN);
//power down the radio and whatever is allowed
arch_goto_sleep(sleep_mode);
... ...
2.宏定义:#define CFG_MAX_SLEEP_DURATION_EXTERNAL_WAKEUP_MS 10000 // 10s
请问该变量的宏定义10s的含义是什么?跟按键唤醒功能有关系么?
3.请问在DSPS工程下,如何设置进入EXT_SLEEP模式?如何判断DA14583已经进入EXT_SLEEP模式?
问题有点多,期待技术支持!
1.开启extended sleep传输数据,并没有不稳定。你需要在PC端的串口工具里设置流控。如果不设置,之前看到容易出现手机APP端收数偶尔会出现乱码。extended sleep进入的条件,可以查看rwip_sleep函数。主要就是检查kernel里面的状态和射频模块等。唤醒的条件,一般就是GPIO的唤醒中断,还有就是kernel里的timer设定。
2.这个时间是设定的单次睡眠允许的最长时间初始值。以在rwip_sleep函数里的ke_timer_sleep_check和lld_sleep_check的参数为例说明。现在第一个参数在内核有定时器开启的情况下,sleep_duration设置的就是“CFG_MAX_SLEEP_DURATION_EXTERNAL_WAKEUP_MS“
通过之前sleep_duration = jump_table_struct[max_sleep_duration_external_wakeup_pos];获得;第二个参数是给的唤醒的准备时间rwip_env.wakeup_delay.
这两个函数被封住,但大概执行的结果就是:前一个检查定时器下一次起来到当前的时间间隔,后一个检查LLD里event的时间间隔。如果这个值比sleep_duration大,那么睡眠条件满足,返回true;如果这个值比sleep_duration小,但超过wakeup_delay,更新sleep_duration的值,仍然返回true,可以睡。别的情况就不让睡了,返回false。接着按照更新后的sleep_duration值,安排蓝牙协议栈睡眠的时间。所以目前蓝牙协议栈单次最多就是睡10s。这个和按键的中断没有关系。
3.判断是否进入睡眠,可以用smartsnippet连接待测的设备,然后在界面上选择spi/uart模式,在power_profile里面通过看电流就可以判断。
非常感谢。
1.现在我在DSPS工程中设置变量app_default_sleep_mode =ARCH_SLEEP_OFF;关闭睡眠模式的情况下,我该如何手动设置DA14583让其进入extended sleep?
我的设置是:当条件1等于真的时候如下操作,但实际测试DA14583没有进入extended sleep下,我有arch_printf("%d: mode\r\n",__LINE__);一直在打印。
while(1)
{
{做
// schedule all pending events
schedule_while_ble_on();
}
while ((app_asynch_proc())); //grant control to the application, try to go to power down
... ...
if(条件1 == 真) {
条件1 = 假;
arch_set_sleep_mode(ARCH_EXT_SLEEP_ON);
}
}
3.我在打开smartsnippet连接待测的设备,然后在界面上选择spi/uart模式,spi/uart模式选项下没有可选项,请问smartsnippet检测电流power_profile的功能是针对官方开发板的么?
p.s 我不是用的官方开发板,在smartsnippet,界面UART模式下是有串口可选,而且我也是通过smartsnippet->Booter下载代码到DA14583的。
期待@Gongyu_Dialog回复。
1.应该直接调用就行,但你有打印的话,看uart_callback函数里面是没办法调用到arch_restore_sleep_mode的。
2.测电流的功能,应该是在官方开发板的,你也可以用示波器上测。
1.我关闭了所有打印信息,在while(1)里面添加一个LED的状态控制,当调用arch_set_sleep_mode(ARCH_EXT_SLEEP_ON);后,LED的状态还是一直闪烁,表明没有进入extended sleep模式下。
static uint8_t state = 0;
while(1)
{
{做
// schedule all pending events
schedule_while_ble_on();
}
while ((app_asynch_proc())); //grant control to the application, try to go to power down
//if the application returns GOTO_SLEEP
//((STREAMDATA_QUEUE)&& stream_queue_more_data())); //grant control to the streamer, try to go to power down
... ...
if(条件1 == 真){
条件1 = 假;
arch_set_sleep_mode(ARCH_EXT_SLEEP_ON);
}
// 500ms执行一次
if(500ms 时间到) {
if(state == 0) {
state = 1;
GPIO_SetActive( LED2_PORT, LED2_PIN);
}else {
state = 0;
GPIO_SetInactive( LED2_PORT, LED2_PIN);
}
}
@Gongyu_Dialog 直接配置arch_set_sleep_mode(ARCH_EXT_SLEEP_ON);后DA14583并没有进入extended sleep模式下,请指导一下,还有其他需要注意或者配置的么?
你还需要把BLE里面的timer和event关掉,只让外部来唤醒。否则还会走到你添加灯控制代码的地方
所以还需要加一句:arch_ble_ext_wakeup_on();
可以参考一下SDK里的user_proxr.c文件里的
app_button_enable函数以及app_wakeup_cb函数
我添加后
if(条件1 == 真){
条件1 = 假;
arch_ble_ext_wakeup_on();
arch_set_sleep_mode(ARCH_EXT_SLEEP_ON);
}
代码运行还是会跑到灯控制代码部分。
user_proxr.c里面app_button_enable函数以及app_wakeup_cb函数都是关于按键唤醒的逻辑,
没有关于如何进入extended sleep模式的操作!或者是我没有找到,请指导下!
非常感谢。
你好,可以麻烦你使用官方开发板测试下DSPS_v_5.150.2工程,
配置如下参数:
const static sleep_state_t app_default_sleep_mode = ARCH_SLEEP_OFF;
#define CFG_MEM_MAP_EXT_SLEEP
#undef CFG_MEM_MAP_DEEP_SLEEP
在工程while(1){
// 条件触发以下操作,判断DA14583是否进入ext_sleep模式
arch_set_sleep_mode(ARCH_EXT_SLEEP_ON);
arch_ble_ext_wakeup_on();
}
非常感谢!
直接调用,可以进入睡眠,但会被默认的广播流程唤醒。extended sleep并不是说停在WFI指令不动了。仍然会被协议栈的消息唤醒。
arch_set_sleep_mode(ARCH_EXT_SLEEP_ON);
arch_ble_ext_wakeup_on();
如果你需要代码停在WFI指令之前,并且通过GPIO状态显示当前的状态,就需要把广播关掉。
另外,就是GPIO的电平,在进入睡眠后即使外设电关断的情况下,会retention之前的状态。
我这里通过按键进入睡眠的模式,通过GPIO可以观察。
*void app_button_press_cb(void)
{
if(GetBits16(SYS_STAT_REG, PER_IS_DOWN)){
periph_init();
}
if(arch_get_sleep_mode() == ARCH_SLEEP_OFF)
{
app_easy_gap_advertise_stop(); //关掉广播
arch_set_sleep_mode(ARCH_EXT_SLEEP_ON);
arch_ble_ext_wakeup_on();
}
app_button_enable();//
}
×关掉广播的相关代码
void user_app_adv_start(void)
{
// Schedule the next advertising data update
//app_adv_data_update_timer_used = app_easy_timer(APP_ADV_DATA_UPDATE_TO, adv_data_update_timer_cb); //删掉可以关掉广播的操作
struct gapm_start_advertise_cmd* cmd;
cmd = app_easy_gap_undirected_advertise_get_active();
// add manufacturer specific data dynamically
mnf_data_update();
app_add_ad_struct(cmd, &mnf_data, sizeof(struct mnf_specific_data_ad_structure));
app_easy_gap_undirected_advertise_start();
}
×关掉默认回调函数的广播操作
void user_app_adv_undirect_complete(uint8_t status)
{
// If advertising was canceled then update advertising data and start advertising again
if (status == GAP_ERR_CANCELED)
{
//user_app_adv_start();//防止调用完关广播的操作,广播默认又起来
}
}
×添加GPIO函数在
while(1)
{
{做
test_gpio1(1); --------------------------------------------------------------//拉高GPIO
// schedule all pending events
schedule_while_ble_on();
}
while ((app_asynch_proc())); //grant control to the application, try to go to power down
//if the application returns GOTO_SLEEP
//((STREAMDATA_QUEUE)&& stream_queue_more_data())); //grant control to the streamer, try to go to power down
//if the application returns GOTO_SLEEP
//wait for interrupt and go to sleep if this is allowed
if (((!BLE_APP_PRESENT) && (check_gtl_state())) ||
(BLE_APP_PRESENT))
{
//Disable the interrupts
GLOBAL_INT_STOP();
app_asynch_sleep_proc();
// get the allowed sleep mode
// time from rwip_power_down() to WFI() must be kept as short as possible!!
sleep_mode = rwip_power_down();
if ((sleep_mode == mode_ext_sleep) || (sleep_mode == mode_deep_sleep)) {
//power down the radio and whatever is allowed
test_gpio1(0); //-----------------------------------------------------------------------拉低GPIO,在下面的函数关掉外设电之前设置
arch_goto_sleep(sleep_mode);
//wait for an interrupt to resume operation
WFI();
//resume operation
arch_resume_from_sleep();
}
......
非常感谢,我在DSPS_v_5.150.2下添加按键进入extended sleep 模式的函数,并且添加一句LED1状态至高,
if(arch_get_sleep_mode() == ARCH_SLEEP_OFF)
{
GPIO_SetActive( LED1_PORT, LED1_PIN); //确认响应到了按键
app_easy_gap_advertise_stop(); //关掉广播
arch_set_sleep_mode(ARCH_EXT_SLEEP_ON);
arch_ble_ext_wakeup_on();
}
同样在以下地方置低LED1作为判断是否进入extended sleep 模式的标志
... ...
if ((sleep_mode == mode_ext_sleep) || (sleep_mode == mode_deep_sleep)) {
//power down the radio and whatever is allowed
/ / test_gpio1 (0);//-----------------------------------------------------------------------拉低GPIO,在下面的函数关掉外设电之前设置
GPIO_SetInactive( LED1_PORT, LED1_PIN);
arch_goto_sleep(sleep_mode);
... ...
结果LED1状态一直是高,没有进入extended sleep。
问题1:请问下工程DSPS_v_5.150.2下默认的广播流程唤醒的地方有哪些?
问题2:da14583在进入extended sleep模式后,io口状态是保持掉电前的状态,所以我添加如下代码
while(1)
{
{做
// schedule all pending events
schedule_while_ble_on();
}
while ((app_asynch_proc())); //grant control to the application, try to go to power down
... ...
toggle_led2(500); // 500ms周期闪烁
}
在进行了按键操作,进入extended sleep后,若DA14583成功进入该模式,
led2的状态只有一种把,要么高,要么低。 而我的现象是按键触发后,led2还是在闪烁。
这个现象也说明DA14583没有成功进入extended sleep把。
期待回复中。
之前和你用的工程不一样,对于DSPs工程,你还需要加个函数把app_default_sleep_mode值改掉。因为在user_sps_sleep_check函数和user_sps_sleep_restore函数还会用到。
另外,就是DSPs工程里面的关广播的操作,不会在关后又再重启广播。所以保持默认的就行了。直接调用关广播的函数就可以了。
void app_button_press_cb(void)
{
if(GetBits16(SYS_STAT_REG, PER_IS_DOWN)){
periph_init();
}
if(arch_get_sleep_mode() == ARCH_SLEEP_OFF)
{
app_easy_gap_advertise_stop();
arch_set_sleep_mode(ARCH_EXT_SLEEP_ON);
user_sps_set_mode(ARCH_EXT_SLEEP_ON);-----------------
user_scheduler_reinit();//
arch_ble_ext_wakeup_on();
}
app_button_enable();
}
void user_sps_set_mode(uint8_t mode)
{
app_default_sleep_mode = mode;
}
非常感谢,按照如上设置后,DA14583成功进入extended sleep模式下。
我用的是DSPS_v_5.150.2代码,我如何设置进入到EXT_SLEEP呢?我将app_default_sleep_mode = ARCH_EXT_SLEEP_ON打开,#undef CFG_DEVELOPMENT_DEBUG,因为我的设备不允许用流控,将#undef CFG_UART_HW_FLOW_CTRL,#undef CFG_UART_SW_FLOW_CTRL,下载到外接FLASH,我用的是14580,但是功耗不正确,BLE接收数据正确,发送数据确不稳定,能否实现呢?请指教,谢谢!
我如果打开软件流控的话,PC端一直接收11、13,而且收发数据不正常,谢谢!
用的是SDK5.04的prox_reporter,进入扩展睡眠模式后,电流维持在460uA左右,怎么都降不下来,jtag去掉也一样,有没有人遇到过相同的问题?