×
单片机 > 单片机程序设计 > 详情

STM32的Systick系统滴答定时器

发布时间:2020-06-09 发布时间:
|

Systick :系统心跳定时器,提供系统节拍

             裸机程序中可作为独立的延时定时器

用途:

1.产生操作系统的时钟节拍

2.便于不同处理器之间程序移植

SysTick定时器被捆绑在NVIC中,异常号15

3.作为一个闹铃测量时间用于测量时间,

   但当处理器在调试期间被喊停(halt)时,则SysTick定时器亦将暂停运作

它有四个寄存器

STK_CSR, 0xE000E010 -- 控制寄存器 

STK_LOAD, 0xE000E014 -- 重载寄存器 

STK_VAL, 0xE000E018 -- 当前值寄存器 

STK_CALRB, 0xE000E01C -- 校准值寄存器

stm32的时钟源

选择外部时钟源时,则Systick时钟为HCLK /8

选择内核时钟源时,则Systick时钟为HCLK

延时编程原理

systick定时器是24位的递减计数器,设定初值并使能它后,它会每个系统时钟周期计数器减1,

计数到0 时,将从RELOAD 寄存器中自动重装载定时初值。只要不把它在SysTick控制及状态寄存器中的使能位清除,就永不停息.

延时编程步骤

1.计算出产生1us 需要多少个时钟周期 fac_us;

2.计算出RELOAD寄存器的值

也就是产生相应延时所需要的时钟周期数

RELOAD=fac_us * nus

3.开启计数

4.循环检测计数到0的标志位;

5.清空计数器,关闭定时器

=======================================

SysTick异常配置步骤

1对CTRL//LOAD/VAL三个寄存器进行了配置,

2初始化SysTick使用的时钟,

3清除系统当前值,装入重装值,

4使能SysTick,使SysTick能响应中断

=======================

当SysTick定时器计到0时,将把COUNTFLAG位置位;而下述方法可以对其清零: 

1.读取SysTick 控制及状态寄存器(STCSR) 

2.往SysTick 当前值寄存器(STCVR)中写任何数据 

只有当VAL 值为0 时,计数器自动重载RELOAD

======================

库函数

使用ST的函数库使用systick的方法,严格按照以下顺序:

1、调用SysTick_CounterCmd() -- 失能SysTick计数器

2、调用SysTick_ITConfig () -- 失能SysTick中断

3、调用SysTick_CLKSourceConfig() -- 设置SysTick时钟源。

4、调用SysTick_SetReload() -- 设置SysTick重装载值。

5、调用SysTick_ITConfig () -- 使能SysTick中断

6、调用SysTick_CounterCmd() -- 开启SysTick计数器

Systick中断服务函数

void SysTick_Handler(void);

==========================

寄存器版代码注解

使用外部8M时钟,锁相环里出来的频率是72M,AHB预分频后是72M,

systick固定HCLK时钟的1/8,即9M,那么延时1us是9个时钟

C代码

void delay_init(u8 SYSCLK)  //系统时钟是72MHz,SYSCLK=72  

{  

    SysTick->CTRL &= 0xfffffffb ; //bit2清0,也就是配置选择外部时钟  

    fac_us=SYSCLK/8; //硬件8分频,fac_us得出的值是要给下面的时钟函数用的  

    fac_ms =(u16)fac_us*1000;  

      

}  

  

void delay_us(u32 nus)  //nus假如为10us  

{  

    u32 temp;  

    SysTick->LOAD = nus*fac_us;  //延时10us的话就是  10*9=90,装到load寄存器中  

    SysTick->VAL=0x00;//计数器清0,因为currrent字段被手动清零时,load将自动重装到VAL中  

    SysTick->CTRL = 0x01;//配置使异常生效,也就是计数器倒数到0时将发出异常通知  

    do  

    {  

       temp = SysTick->CTRL;  //时间到了之后,该位将被硬件置1,但被查询后自动清0  

    }  

    while(temp & 0x01 && !(tmep &(1<<16))); //查询  

    SysTick->CTRL = 0x00;  //关闭计数器  

    SysTick->VAL = 0x00;   //清空val  

}  

 

//这个while循环,判断如果Systick还在Enable的状态,并且计数器还没数到0,

就不停的循环把当前的SysTick->CTRL寄存器值写入变量temp,继续下一次判断。

当Systick被Disable或者计数器数到0了,就停止循环

还有一个注意点:

LOAD寄存器是24位的 最大值0xffffff

那么延时最大值计算公式为

nms<=0xffffff*8*1000/SYSCLK (SYSCLK单位Hz)

则nms的最大值为1864.135ms ,即1864毫秒

 

 

 

 

? 首先我们要明白什么是SysTick定时器?

Sys 系统 ,tick 滴答声 ,系统滴答滴答很形象地表示了它是一个系统节拍器。SysTick 是一个24 位的倒计数定时器,当计到0 时,将从RELOAD 寄存器中自动重装载定时初值。只要不把它在SysTick 控制及状态寄存器中的使能位清除,就永不停息。

 

? 为什么要设置SysTick定时器?

(1)产生操作系统的时钟节拍

SysTick定时器被捆绑在NVIC中,用于产生SYSTICK异常(异常号:15)。在以前,大多操作系统需要一个硬件定时器来产生操作系统需要的滴答中断,作为整个系统的时基。因此,需要一个定时器来产生周期性的中断,而且最好还让用户程序不能随意访问它的寄存器,以维持操作系统“心跳”的节律。SysTick的最大使命,就是定期地产生异常请求,作为系统的时基。OS都需要这种“滴答”来推动任务和时间的管理。

 

(2)便于不同处理器之间程序移植。

Cortex‐M3处理器内部包含了一个简单的定时器。因为所有的CM3芯片都带有这个定时器,软件在不同 CM3器件间的移植工作得以化简。该定时器的时钟源可以是内部时钟(FCLK,CM3上的自由运行时钟),或者是外部时钟( CM3处理器上的STCLK信号)。不过,STCLK的具体来源则由芯片设计者决定,因此不同产品之间的时钟频率可能会大不相同,你需要检视芯片的器件手册来决定选择什么作为时钟源。SysTick定时器能产生中断,CM3为它专门开出一个异常类型,并且在向量表中有它的一席之地。它使操作系统和其它系统软件在CM3器件间的移植变得简单多了,因为在所有CM3产品间对其处理都是相同的。

 

(3)作为一个闹铃测量时间。

SysTick定时器还可以用作闹钟,作为启动一个特定任务的时间依据。它作为一个闹铃,用于测量时间。要注意的是,当处理器在调试期间被喊停(halt)时,则SysTick定时器亦将暂停运作。

 

 

? 再来看看SysTick的用法

(1)我们对一个系统编程,老说编程编程什么的,到底我们在编什么程?当然这个问题要探讨起来可能有点远了。我来说说对SysTick的编程,对单片机的编程不过就是对单片机里面的寄存器进行控制,使整个软硬件系统处于一种在你的掌控之下的状态。这就是了嘛,现在我是头,我对我的手下下达一些指令,让它们去做一些事情。所以我们想搞清楚怎样控制SysTick我们还得看我们能对它的哪些部分可以控制。那些部分就是寄存器。

 

SysTick有4个寄存器 :

寄存器 描述

CTRL SysTick 控制和状态寄存器

LOAD SysTick 重装载值寄存器

VAL SysTick 当前值寄存器

CALIB SysTick 校准值寄存器

 

对应地在固件函数库中定义了这个东西

typedef struct

{

vu32 CTRL;

vu32 LOAD;

vu32 VAL;

vuc32 CALIB;

} SysTick_TypeDef;

在这背后,已经对定义的寄存器进行了一个地址映射。当我们操控我们定义的寄存器时实际上已是通过那种映射关系操控了芯片内部的值。其实在STM32中对寄存器的操作都是通过这种方式进行的。

具体的映射过程如下,我们可以看一下:

#define SCS_BASE ((u32)0xE000E000)

#define SysTick_BASE (SCS_BASE + 0x0010)

#ifndef DEBUG

...

#ifdef _SysTick

#define SysTick ((SysTick_TypeDef *) SysTick_BASE)

#endif

...

#else

...

#ifdef _SysTick

EXT SysTick_TypeDef *SysTick;

#endif

...

#endif

#ifdef _SysTick

SysTick = (SysTick_TypeDef *) SysTick_BASE;

#endif

为了访问SysTick寄存器,, _SysTick必须在文件“stm32f10x_conf.h”中定义如下:

#define _SysTick

 

映射过程就不作讨论了。总这这样映射的结果是我们能直接使用SysTick。那就来看一下有关寄存器的设置。

 

(2)SysTick里的寄存器我也简单地把它理解为是一个32位数。

这里有一张图:

 

在最新的STM32固件库中的core_cm3.c中提供了这样一个函数来供我们配置SysTick,当我们须要用到SysTick时调用它就可以了:

static __INLINE uint32_t SysTick_Config(uint32_t ticks)//ticks 是要重装载的值

{

if (ticks > SysTick_LOAD_RELOAD_Msk) return (1);

 

 

 

SysTick->LOAD = (ticks & SysTick_LOAD_RELOAD_Msk) - 1;

 

 

NVIC_SetPriority (SysTick_IRQn, (1<<__nvic_prio_bits>

 

 

SysTick->VAL = 0;

 

 

 

SysTick->CTRL = SysTick_CTRL_CLKSOURCE_Msk |

SysTick_CTRL_TICKINT_Msk |

SysTick_CTRL_ENABLE_Msk;

 

 

return (0);

}

 

总结:在配置过程对CTRL//LOAD/VAL三个寄存器进行了配置,初始化了SysTick使用的时钟,清除系统当前值,装入重装值,使能SysTick,使SysTick能响应中断,说了半天其实就这一句话。在主程序中调用SysTick_Configuration( uint32_t ticks ),输入重装值就配置完成了。

 

(3)SysTick 的中断处理函数在stm32f10x_it.c

函数原型为void SysTick_Handler(void)

{

// user code

}

用户只要把须要处理的程序填入这里就完成啦。

 

 

? 例子:

正如上面叙述,SysTick的使用为:

(1)配置SysTick

(2)写中断函数

 

我们产生1ms的廷时:

在我们自己编写的main.c中有:

//前面的省略 ……

Volatile unsigned int TimingDelay ; //定义一个全局变量,用于计数计时值

 

//中间部分省略……

void Delay_Ms( uint32_t nTime ) //我们须要的廷时函数

{

TimingDelay = nTime ; //把廷时值赋值给TimingDelay;

while( TimingDelay != 0 ); //等待计时时间到,在SysTick的中断函数

//中每1ms对TimingDelay减1

}

 

 

int main(void)

{

//配置电源

//配置GPIO

//配置NVIC 等等

while( SysTick_Configuration( 72000 ) != 0 ) ; //配置SysTick,装入倒数值,我

//们假设系统时钟为72MHz,则

//要定时1ms,输入的倒数值为

//72000

while(1)

{

//user code

}

}

 

在stm32f10x_it.c中:

//前面省略 ……

extern volatile unsigned int TimingDelay;

 

void SysTick_Handler(void)

{

// user code

TimingDelay--;

 

}



『本文转载自网络,版权归原作者所有,如有侵权请联系删除』

热门文章 更多
C51 特殊功能寄存器SFR的名称和地址