今天在写用1768读取温度传感器DS18b20的程序,它是单总线的传感器,因此读写都需要按照手册上的时序。这是问题就来了,如何在1768中写一个尽可能精确us级延时函数?以前用51单片机写延时函数是用嵌套的for循环就能完成的,但是在ARM中我们是不能那么用的,因为它是三级流水线构架,不像51一样能计算出每条指令的时间。幸好Cortex-M3给我们提供了一个精准延时的方法——Systick定时器。
Systick是一个24位的倒计数器,存在于所有与M3构架的芯片中。在1768中,Systick的时钟源可以是Fcclk或由引脚(STCLK)输入的时钟信号,通过STCTRL寄存器(系统定时器控制和状态寄存器)选择,stick可以装载2的24次方以内的数(设置STRELOAD),当装载值被减到0是可以选择产生中断,接着计数器被复位,从头开始递减。这样我们就能通过设置装载值得到我们定时时间。
Systick在1768中有四个寄存器,在不同的M3的芯片中大致一样。
系统定时器控制寄存器和状态寄存器(STCTRL):
STCTRL寄存器含有系统节拍定时器的控制信息,还可以提供状态标志。
在"core_cm3.c"文件中,该寄存器的0-2为都被配置成1,即节拍计数器使能,中断使能,时钟选择Fcclk。
系统定时器重载值寄存器(STRELOAD):
该寄存器用来装载定时值,在定时器进行初始化时,该值由软件装入寄存器。
系统定时器当前值寄存器(STCURR):
当软件读计数器时,该寄存器就会返回计数器的当前计数值。
系统定时器校准值寄存器(STCALIB):
用于在系统节拍定时器的时钟频率为100MHz时每隔10毫秒产生一个中断。存在一个厂商编程的值。
Systick的操作也很方便,因为在core_cm3.c文件中已经把我们要用的配置函数写好了,我们只要传入装载值就好。函数名为SysTick_Config((uint32_t ticks),附上函数体:
static __INLINE uint32_t SysTick_Config(uint32_t ticks)
{
if (ticks > SysTick_LOAD_RELOAD_Msk) return (1); /* 装载值超过2的24次方式不可能的 返回1 */
SysTick->LOAD = (ticks & SysTick_LOAD_RELOAD_Msk) - 1; /* 传入装载值 */
NVIC_SetPriority (SysTick_IRQn, (1<<__NVIC_PRIO_BITS) - 1); /* 设置中断优先级 */
SysTick->VAL = 0; /* 当前值清零 */
SysTick->CTRL = SysTick_CTRL_CLKSOURCE_Msk |
SysTick_CTRL_TICKINT_Msk |
SysTick_CTRL_ENABLE_Msk; /* 使能、中断使能、 选择时钟*/
return (0); /* 配置成功,返回0 */
}
每次装载值变为0时都会进入中断,在写精确延时函数中,我们可以利用中断实现精准延时,以延时2ms为例:
//写入装载值
if(SysTick_Config( SystemFrequency/1000))//时钟源是Fcclk,装载值为1ms,重装值超过了24位,是不可能的。返回失败值1
{
while(1);
}
void SysTick_Handler(void) //系统滴答
{
//usTicks++;
msTicks++;
if(msTicks==2000) //每隔2S读取一次温度
{
sign=1; //标志位,当检测到标志位为1,则开始动作
msTicks=0;
}
}
可传参的延时程序可以简化成这样:
void Delay(uint32_t ms) //毫秒级别
{
Systick->LOAD=96000*ms; //装载计数值,Fcclk=96MHz
Systick->CTRL=0x5; //使能定时器,时钟选择Fcclk
while(Systick->CTRL&(1<<16));
Systick->CTRL=0x04; //关闭定时器
}