一、软件定时器
以下软件定时器接⼝位于 /ESP8266_NONOS_SDK/include/osapi.h
。
注意事项:
- 使用的定时器由软件实现,定时器的函数在任务中被执行。因为任务可能被中断,或者被其他⾼优先级的任务延迟,因此以下
os_timer
系列的接口并不能保证定时器精确执行。 - 如果需要精确的定时,例如,周期性操作某 GPIO,请使用硬件中断定时器,具体可参考
hw_timer.c
,硬件定时器的执行函数在中断里被执行。 - 对于同⼀个 timer,
os_timer_arm
或os_timer_arm_us
不能重复调用,必须先os_timer_disarm
。 os_timer_setfn
必须在 timer 未使能的情况下调用,在os_timer_arm
或os_timer_arm_us
之前或者os_timer_disarm
之后。、
1、os_timer_arm
- 功能 使能毫秒级定时器
- 函数定义
void os_timer_arm
{
os_timer_t *ptimer,
uint32_t milliseconds,
bool repeat_flag
}
- 参数
•os_timer_t *ptimer
:定时器结构
•uint32_t milliseconds
:定时时间,单位:ms
• 如未调用system_timer_reinit
,可支持范围 5 ~ 0x68D7A3
• 如调用了system_timer_reinit
,可支持范围 100 ~ 0x689D0
•bool repeat_flag
:定时器是否重复 - 返回:⽆
2、os_timer_disarm
- 功能:取消定时器定时
- 函数定义:
void os_timer_disarm (os_timer_t *ptimer)
- 参数:
os_timer_t*ptimer
:定时器结构 - 返回:无
3、os_timer_setfn
- 功能:设置定时器回调函数。使用定时器,必须设置回调函数。
- 函数定义
void os_timer_setfn
{
os_timer_t *ptimer,
os_timer_func_t *pfunction,
void *parg
}
- 参数:
•os_timer_t *ptimer
:定时器结构
•os_timer_func_t *pfunction
:定时器回调函数
•void *parg
:回调函数的参数 - 返回:无
4、system_timer_reinit
- 功能:重新初始化定时器,当需要使用微秒级定时器时调用
- 注意
• 同时定义USE_US_TIMER
•system_timer_reinit
在程序最开始调用,user_init
的第⼀句。 - 函数定义
void system_timer_reinit (void)
- 参数:无
- 返回:无
5、os_timer_arm_us
- 功能 :使能微秒级定时器。
- 注意:
• 请定义 USE_US_TIMER,并在 user_init 起始第⼀句,先调⽤ system_timer_reinit。
• 最⾼精度为 500 μs。 - 函数定义
void os_timer_arm_us
(
os_timer_t *ptimer,
uint32_t microseconds,
bool repeat_flag
)
- 参数
•os_timer_t *ptimer
:定时器结构
•uint32_t microseconds
:定时时间,单位:μs,最小定时 0x64 ,最⼤可输⼊0xFFFFFFF
•bool repeat_flag
:定时器是否重复 - 返回:无
6、示例
//注: OS_ _Timer_ 1必须定义为全局变量因为ESP8266的内核还要使用
os_ timer_ t OS_ Timer_ 1; //①:定义软件定时器(os_ _timer_ .t型结构体)
//软件定时的回调函数
void ICACHE_ FLASH_ ATTR OS_ Timer_1_ cb(void) // ②:定义回调函数
{
F_LED = !F_LED;
GPIO_ OUTPUT_ SET(GPIO_ID_PIN(4),F_LED);// LED状态翻转
os_ printf("\r\n----OS_ Timer_1_ cb----\r\n"); //进入回调函数标志
}
//软件定时器初始化(ms毫秒)
void ICACHE_FLASH_ ATTR OS_Timer_1_Init_JX(u32 time_ ms,u8 time_ repet itive)
//关闭定时器
//参数一:要关闭的定时器;
os_ timer_ disarm(&OS_ Timer_ 1); //关闭软件定时器
//设置定时器
//参数一:要设置的定时器;参数二:回调函数(需类型转换);参数三:回调函数的参数
//os__timer_setfn必须在软件定时器未使能的情况下调用
os_ timer_ setfn(&OS_ Timer_ 1, (os_ timer_func_ t *)OS_ Timer_ 1_ cb, NULL); //设置回调函数
//使能(启动) ms定时器
//参数一:要使能的定时器;参数二:定时时间(单位: ms) ;参数三: 1=重复/0=只一次
os_ timer_ _arm(&OS_ Timer_ 1, time_ms, time_ repetitive); //设置定时器参数并使能定时器
// [ 如未调用system_ timer_ reinit, 可支持范围: [5ms ~ 6, 870, 947ms]]
// [如果调用system_ timer_ reinit, 可支持范围: [100ms ~ 428, 496 ms]]
二、硬件中断定时器
以下硬件中断定时器接口位于 /ESP8266_NONOS_SDK/examples/driver_lib/
hw_timer.c用户可根据 driver_lib
⽂件夹下的 readme.txt
⽂件使用。
注意事项
- 如果使用NMI 中断源,且为自动填装的定时器,调用
hw_timer_arm
时参数 val 必须大于100。 - 如果使用 NMI 中断源,那么该定时器将为最高优先级,可打断其他 ISR。
- 如果使用 FRC1 中断源,那么该定时器无法打断其他 ISR。
hw_timer.c
的接口不能跟 PWM 驱动接口函数同时使用,因为⼆者共用了同一个硬件定时器。- 硬件中断定时器的回调函数定义,请勿添加 ICACHE_FLASH_ATTR 宏。
- 使用
hw_timer.c
的接口,请勿调用wifi_set_sleep_type(LIGHT_SLEEP)
; 将自动睡 眠模式设置为Light-sleep。因为 Light-sleep 在睡眠期间会停 CPU,停 CPU 期间不能响应 NMI 中断。
1、hw_timer_init
- 功能:初始化硬件 ISR 定时器
- 函数定义
void hw_timer_init
(
FRC1_TIMER_SOURCE_TYPE source_type,
u8 req
)
-
参数
•FRC1_TIMER_SOURCE_TYPE source_type
:定时器的 ISR 源
•FRC1_SOURCE
:使用 FRC1 中断源
•NMI_SOURCE
:使用 NMI 中断源
•u8 req
0:不自动填装;1:自动填装 -
返回:无
2、hw_timer_arm
- 功能:使能硬件中断定时器
- 函数定义
void hw_timer_arm (uint32 val)
- 参数
• uint32 val:定时时间
自动填装模式:
• 使用FRC1 中断源 FRC1_SOURCE,取值范围:50 ~ 0x199999 μs;
• 使用 NMI 中断源 NMI_SOURCE,取值范围 : 100 ~ 0x199999 μs:
• 非自动填装模式,取值范围:10 ~ 0x199999 μs - 返回:无
3、hw_timer_set_func
- 功能:设置定时器回调函数。使⽤定时器,必须设置回调函数。
- 注意:回调函数前不能添加
ICACHE_FLASH_ATTR
宏定义,中断响应不能存放在 Flash 中。 - 函数定义:
void hw_timer_set_func (void (* user_hw_timer_cb_set)(void) )
- 参数
void (* user_hw_timer_cb_set)(void)
:定时器回调函数,函数定义时请勿添加ICACHE_FLASH_ATTR
宏。 - 返回 ⽆
4、示例1
定时器变量必须定义为全局变量,因为ESP8266内核还要使用
#define REG_READ(_r) (*(volatile uint32 *)(_r))
#define WDEV_NOW() REG_READ(0x3ff20c00)
uint32 tick_now2 = 0;
void hw_test_timer_cb(void)
{
static uint16 j = 0;
j++;
if( (WDEV_NOW() - tick_now2) >= 1000000 )
{
static u32 idx = 1;
tick_now2 = WDEV_NOW();
os_printf("b%u:%d\n",idx++,j);
j = 0;
}
}
void ICACHE_FLASH_ATTR user_init(void)
{
hw_timer_init(FRC1_SOURCE,1);
hw_timer_set_func(hw_test_timer_cb);
hw_timer_arm(100);
}
5、示例2
//硬件定时器初始化[FRC1_ SOURCE==0、 NMI_ SOURCE=1 ]
hw timer init(0, 1);//①:初始化硬件定时器参数1:中断源(NMI/FRC1参数;2:是否重复
hw timer set func(HW_ Timer_ INT) ;//③:注册回调函数
hw_ timer_ arm (500000) ;//④:设置定时器参数(单位us,参数必须<=1, 677, 721)
// 硬件定时回调函数[注意:中断函数前不要有" ICACHE_ FLASH_ ATTR" ]
void HW_ Timer_ INT(void) //②:定义硬件定时回调函数
{
F_LED = !F_ LED;
GPIO_ _OUTPUT_ _SET(GPIO_ ID_ _PIN(4),F_ LED) ;// LED状态翻转
os_ printf("\r\n--- HW_ Timer_ INT ---\r\n"); // 进入定时器函数标志
三、现象
打开串口,复位ESP8266.可以看到串口这边一直在收到我们的软件定时器回调函数它的标志字符串。并且每隔500msLED状态反转一次,这里我就放两个状态的图。不太方便放视频。
这说明我们的软件定时器的初始化成功执行,并且成功执行了软件定时器的回调函数。