学习日记——ESP8266定时器(2020.5.20)

一、软件定时器

以下软件定时器接⼝位于 /ESP8266_NONOS_SDK/include/osapi.h
注意事项:

  • 使用的定时器由软件实现,定时器的函数在任务中被执行。因为任务可能被中断,或者被其他⾼优先级的任务延迟,因此以下 os_timer系列的接口并不能保证定时器精确执行。
  • 如果需要精确的定时,例如,周期性操作某 GPIO,请使用硬件中断定时器,具体可参考hw_timer.c,硬件定时器的执行函数在中断里被执行。
  • 对于同⼀个 timer,os_timer_armos_timer_arm_us 不能重复调用,必须先os_timer_disarm
  • os_timer_setfn 必须在 timer 未使能的情况下调用,在 os_timer_armos_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状态反转一次,这里我就放两个状态的图。不太方便放视频。
在这里插入图片描述在这里插入图片描述
在这里插入图片描述
这说明我们的软件定时器的初始化成功执行,并且成功执行了软件定时器的回调函数。

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

热爱生活的fuyao

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值