Cortex-M架构SysTick系统定时器阻塞和非阻塞延时
首先是最常用的阻塞延时
void delay_ms(unsigned int ms){SysTick->LOAD = 50000000/1000-1; // Count from 255 to 0 (256 cycles)载入计数值 定时器从这个值开始计数SysTick->VAL = 0; // Clear current value as well as count flag清空计数值到达0后的标记SysTick->CTRL = 5; // Enable SysTick timer with processor clock使能26MHz的系统定时器while(ms--){while ((SysTick->CTRL & 0x00010000)==0);// Wait until count flag is set等待}SysTick->CTRL = 0; // Disable SysTick关闭系统定时器}void delay_us(unsigned int us){SysTick->LOAD = 50000000/1000/1000-1; // Count from 255 to 0 (256 cycles)载入计数值 定时器从这个值开始计数SysTick->VAL = 0; // Clear current value as well as count flag清空计数值到达0后的标记SysTick->CTRL = 5; // Enable SysTick timer with processor clock使能26MHz的系统定时器while(us--){while ((SysTick->CTRL & 0x00010000)==0);// Wait until count flag is set等待}SysTick->CTRL = 0; // Disable SysTick关闭系统定时器}
50000000表示工作频率
分频后即可得到不同的延时时间
以此类推
那么 不用两个嵌套while循环 也可以写成:
void delay_ms(unsigned int ms){SysTick->LOAD = 50000000/1000*ms-1; // Count from 255 to 0 (256 cycles)载入计数值 定时器从这个值开始计数SysTick->VAL = 0; // Clear current value as well as count flag清空计数值到达0后的标记SysTick->CTRL = 5; // Enable SysTick timer with processor clock使能26MHz的系统定时器while ((SysTick->CTRL & 0x00010000)==0);// Wait until count flag is set等待SysTick->CTRL = 0; // Disable SysTick关闭系统定时器}void delay_us(unsigned int us){SysTick->LOAD = 50000000/1000/1000*us-1; // Count from 255 to 0 (256 cycles)载入计数值 定时器从这个值开始计数SysTick->VAL = 0; // Clear current value as well as count flag清空计数值到达0后的标记SysTick->CTRL = 5; // Enable SysTick timer with processor clock使能26MHz的系统定时器while ((SysTick->CTRL & 0x00010000)==0);// Wait until count flag is set等待SysTick->CTRL = 0; // Disable SysTick关闭系统定时器}
但是这种写法有个弊端
那就是输入ms后,最大定时不得超过计数值,也就是不能超过LOAD的最大值,否则溢出以后,则无法正常工作
而LOAD如果最大是32位 也就是4294967295
晶振为50M的话 50M的计数值为1s 4294967295计数值约为85s
固最大定时时间为85s
但用嵌套while的话 最大可以支持定时4294967295*85s
如果采用非阻塞的话 直接改写第二种方法就好了:
void delay_ms(unsigned int ms){SysTick->LOAD = 50000000/1000*ms-1; // Count from 255 to 0 (256 cycles)载入计数值 定时器从这个值开始计数SysTick->VAL = 0; // Clear current value as well as count flag清空计数值到达0后的标记SysTick->CTRL = 5; // Enable SysTick timer with processor clock使能26MHz的系统定时器//while ((SysTick->CTRL & 0x00010000)==0);// Wait until count flag is set等待//SysTick->CTRL = 0; // Disable SysTick关闭系统定时器}void delay_us(unsigned int us){SysTick->LOAD = 50000000/1000/1000*us-1; // Count from 255 to 0 (256 cycles)载入计数值 定时器从这个值开始计数SysTick->VAL = 0; // Clear current value as well as count flag清空计数值到达0后的标记SysTick->CTRL = 5; // Enable SysTick timer with processor clock使能26MHz的系统定时器//while ((SysTick->CTRL & 0x00010000)==0);// Wait until count flag is set等待//SysTick->CTRL = 0; // Disable SysTick关闭系统定时器}
将等待和关闭定时器语句去掉
在使用时加上判断即可变为阻塞:
delay_ms(500);while ((SysTick->CTRL & 0x00010000)==0);SysTick->CTRL = 0;
在非阻塞状态下 可以提交定时器后 去做别的事情 然后再来等待
不过这样又有一个弊端 那就是定时器会自动重载 可能做别的事情以后 定时器跑过了 然后就要等85s才能停下
故可以通过内部定时器来进行非阻塞延时函数的编写
基本上每个mcu的内部定时器都可以配置自动重载等功能 网上资料很多 这里就不再阐述了