STM32知识:什么是SYSTICK 作用是什么

最新更新时间:2020-02-12来源: elecfans关键字:STM32  SYSTICK  系统节拍定时器 手机看文章 扫描二维码
随时随地手机看文章

什么是SYSTICK:

这是一个24位的系统节拍定时器system TIck TImer,SysTIck,具有自动重载和溢出中断功能,所有基于Cortex_M3处理器的微控制器都可以由这个定时器获得一定的时间间隔。


作用:

在单任务引用程序中,因为其架构就决定了它执行任务的串行性,这就引出一个问题:当某个任务出现问题时,就会牵连到后续的任务,进而导致整个系统崩溃。要解决这个问题,可以使用实时操作系统(RTOS).


因为RTOS以并行的架构处理任务,单一任务的崩溃并不会牵连到整个系统。这样用户出于可靠性的考虑可能就会基于RTOS来设计自己的应用程序。这样SYSTICK存在的意义就是提供必要的时钟节拍,为RTOS的任务调度提供一个有节奏的“心跳”。


微控制器的定时器资源一般比较丰富,比如STM32存在8个定时器,为啥还要再提供一个SYSTICK?原因就是所有基于ARM Cortex_M3内核的控制器都带有SysTick定时器,这样就方便了程序在不同的器件之间的移植。而使用RTOS的第一项工作往往就是将其移植到开发人员的硬件平台上,由于SYSTICK的存在无疑降低了移植的难度。


SysTick定时器除了能服务于操作系统之外,还能用于其它目的:如作为一个闹铃,用于测量时间等。


要注意的是,当处理器在调试期间被喊停(halt)时,则SysTick定时器亦将暂停运作。


时钟的选择:

用户可以在位于Cortex_M3处理器系统控制单元中的系统节拍定时器控制和状态寄存器(SysTick control and status register ,SCSR)选择systick 时钟源。如将SCSR中的CLKSOURCE位置位,SysTick会在CPU频率下运行;而将CLKSOUCE位清除则SysTick会以CPU主频的1/8频率运行。


3.5版本的库函数与以往的有所区别

不存在stm32f10x_systick.c文件,故原来的一些函数也不存在,比如SysTick_SetReload(u32 reload);SysTick_ITConfig(FunctionalState NewState);等

在3.5版本的库函数中与systick相关的函数只有两个

第一个,SysTick_Config(uint32_t ticks),在core_cm3.h头文件中进行定义的。

第二个,void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource),在misc.c文件中定义的。


SysTick_Config(uint32_t ticks),在core_cm3.h

主要的作用:

1、初始化systick

2、打开systick

3、打开systick的中断并设置优先级

4、返回一个0代表成功或1代表失败


注意:

Uint32_t ticks  即为重装值,

这个函数默认使用的时钟源是AHB,即不分频。

要想分频,调用void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource),

但是要注意函数调用的次序,先SysTick_Config(uint32_t ticks),

后SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)


函数说明:

/**

* @brief  Initialize and start the SysTick counter and its interrupt.

*

* @param   ticks   number of ticks between two interrupts

* @return  1 = failed, 0 = successful

*

* Initialise the system tick timer and its interrupt and start the

* system tick timer / counter in free running mode to generate

* periodical interrupts.

*/

static __INLINE uint32_t SysTick_Config(uint32_t ticks)

{

if (ticks > SysTick_LOAD_RELOAD_Msk)  return (1);            

/* Reload value impossible */重装载值必须小于0XFF FFFF,为什么,这是一个24位的递减计数器。

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

/* set reload register */设置重装载值,SysTick_LOAD_RELOAD_Msk定义见后面

NVIC_SetPriority (SysTick_IRQn, (1CTRL  = SysTick_CTRL_CLKSOURCE_Msk |

SysTick_CTRL_TICKINT_Msk   |

SysTick_CTRL_ENABLE_Msk;                  

/* Enable SysTick IRQ and SysTick Timer */

return (0);

/* Function successful */

}

#endif

与systick相关的寄存器定义

/** @addtogroup CMSIS_CM3_SysTick CMSIS CM3 SysTick

memory mapped structure for SysTick

@{

*/

typedef struct

{

__IO uint32_t CTRL; /*!< Offset: 0x00  SysTick Control and Status Register */

__IO uint32_t LOAD; /*!< Offset: 0x04  SysTick Reload Value Register       */

__IO uint32_t VAL; /*!< Offset: 0x08  SysTick Current Value Register      */

__I  uint32_t CALIB; /*!< Offset: 0x0C  SysTick Calibration Register        */

} SysTick_Type;

与systick寄存器相关的寄存器及位的宏定义

/* SysTick Control / Status Register Definitions */控制/状态寄存器

#define SysTick_CTRL_COUNTFLAG_Pos 16 /*!《 SysTick CTRL: COUNTFLAG Position */

#define SysTick_CTRL_COUNTFLAG_Msk (1ul 《《 SysTick_CTRL_COUNTFLAG_Pos)

/*!《 SysTick CTRL: COUNTFLAG Mask */ 溢出标志位

#define SysTick_CTRL_CLKSOURCE_Pos 2 /*!《 SysTick CTRL: CLKSOURCE Position */

#define SysTick_CTRL_CLKSOURCE_Msk (1ul 《《 SysTick_CTRL_CLKSOURCE_Pos)

/*!《 SysTick CTRL: CLKSOURCE Mask */时钟源选择位,0=外部时钟;1=内核时钟

#define SysTick_CTRL_TICKINT_Pos 1 /*!《 SysTick CTRL: TICKINT Position */

#define SysTick_CTRL_TICKINT_Msk (1ul 《《 SysTick_CTRL_TICKINT_Pos)

/*!《 SysTick CTRL: TICKINT Mask */异常请求位

#define SysTick_CTRL_ENABLE_Pos 0 /*!《 SysTick CTRL: ENABLE Position */

#define SysTick_CTRL_ENABLE_Msk (1ul 《《 SysTick_CTRL_ENABLE_Pos)

/*!《 SysTick CTRL: ENABLE Mask */使能位

/* SysTick Reload Register Definitions */

#define SysTick_LOAD_RELOAD_Pos 0 /*!《 SysTick LOAD: RELOAD Position */

#define SysTick_LOAD_RELOAD_Msk (0xFFFFFFul 《《 SysTick_LOAD_RELOAD_Pos)

/*!《 SysTick LOAD: RELOAD Mask */

/* SysTick Current Register Definitions */

#define SysTick_VAL_CURRENT_Pos 0 /*!《 SysTick VAL: CURRENT Position */

#define SysTick_VAL_CURRENT_Msk (0xFFFFFFul 《《 SysTick_VAL_CURRENT_Pos)

/*!《 SysTick VAL: CURRENT Mask */

/* SysTick Calibration Register Definitions */

#define SysTick_CALIB_NOREF_Pos 31 /*!《 SysTick CALIB: NOREF Position */

#define SysTick_CALIB_NOREF_Msk (1ul 《《 SysTick_CALIB_NOREF_Pos)

/*!《 SysTick CALIB: NOREF Mask */

#define SysTick_CALIB_SKEW_Pos 30 /*!《 SysTick CALIB: SKEW Position */

#define SysTick_CALIB_SKEW_Msk (1ul 《《 SysTick_CALIB_SKEW_Pos)

/*!《 SysTick CALIB: SKEW Mask */

#define SysTick_CALIB_TENMS_Pos 0 /*!《 SysTick CALIB: TENMS Position */

#define SysTick_CALIB_TENMS_Msk (0xFFFFFFul 《《 SysTick_VAL_CURRENT_Pos) /*!《 SysTick CALIB: TENMS Mask */

/*@}*/ /* end of group CMSIS_CM3_SysTick */

void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)


作用:

选择systick的时钟源,AHB时钟或AHB的8分频

库函数中默认使用的是AHB时钟(在SysTick_Config()函数中设置),即72MHz

函数说明:

/**

* @brief Configures the SysTick clock source.

* @param SysTick_CLKSource: specifies the SysTick clock source.

* This parameter can be one of the following values:

* @arg SysTick_CLKSource_HCLK_Div8: AHB clock divided by 8 selected as SysTick clock source.

* @arg SysTick_CLKSource_HCLK: AHB clock selected as SysTick clock source.

* @retval None

*/

void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource)

{

/* Check the parameters */

assert_param(IS_SYSTICK_CLK_SOURCE(SysTick_CLKSource));

if (SysTick_CLKSource == SysTick_CLKSource_HCLK)

{

SysTick-》CTRL |= SysTick_CLKSource_HCLK;

}

else

{

SysTick-》CTRL &= SysTick_CLKSource_HCLK_Div8;

}

}

Systick时钟源的定义:

/** @defgroup SysTick_clock_source

* @{

*/

#define SysTick_CLKSource_HCLK_Div8 ((uint32_t)0xFFFFFFFB)//将控制状态寄存器的第二位置0,即用外部时钟源

#define SysTick_CLKSource_HCLK ((uint32_t)0x00000004)//将控制状态寄存器的第二位置1,即用内核时钟

#define IS_SYSTICK_CLK_SOURCE(SOURCE) (((SOURCE) == SysTick_CLKSource_HCLK) ||

((SOURCE) == SysTick_CLKSource_HCLK_Div8))

与systick相关的寄存器的说明

SysTick寄存器说明在《Cortex-M3权威指南》(chap8.SysTick定时器章节)有说明

STM32知识:什么是SYSTICK 作用是什么

Systick使用实践

Systick定时时间的设定:

重装载值=systick 时钟频率(Hz)X想要的定时时间(S)

如果时钟频率为:AHB的8分频;AHB=72MHz那么systick的时钟频率为72/8MHz=9MHz

若要定时1秒,则重装载值=9000000X1=9000000,调用函数:SysTick_Config(9000000X1);

若要定时1毫秒,重状态值=9000000X0.001=90000,调用函数:SysTick_Config(9000000/1000);

Systick的中断处理函数

在startup_stm32f10x_hd.s启动文件中有定义。

DCD SysTick_Handler ; SysTick Handler

根据需要直接编写中断处理函数即可:

Void SysTick_Handler (void)

{ ;}

注意:

如果在工程中,加入了stm32f10x_it.c,而又在主函数中编写中断函数,则会报错。

STM32知识:什么是SYSTICK 作用是什么

因为在stm32f10x_it.c文件中,也有这个中断函数的声明,只是内容是空的。

/**

* @brief This function handles SysTick Handler.

* @param None

* @retval None

*/

void SysTick_Handler(void)

{

}

中断优先级的修改

在调用SysTick_Config(uint32_t ticks)之后,调用 void NVIC_SetPriority(IRQn_Type IRQn, uint32_t priority)。这个函数在core_cm3.h头文件中。

具体内容如下:

/**

* @brief Set the priority for an interrupt

*

* @param IRQn The number of the interrupt for set priority

* @param priority The priority to set

*

* Set the priority for the specified interrupt. The interrupt

* number can be positive to specify an external (device specific)

* interrupt, or negative to specify an internal (core) interrupt.

*

* Note: The priority cannot be set for every core interrupt.

*/

static __INLINE void NVIC_SetPriority(IRQn_Type IRQn, uint32_t priority)

{

if(IRQn 《 0) {

SCB-》SHP[((uint32_t)(IRQn) & 0xF)-4] = ((priority 《《 (8 - __NVIC_PRIO_BITS)) & 0xff); } /* set Priority for Cortex-M3 System Interrupts */

else {

NVIC-》IP[(uint32_t)(IRQn)] = ((priority 《《 (8 - __NVIC_PRIO_BITS)) & 0xff); } /* set Priority for device specific Interrupts */

}

下面以一个实例来说明:

利用systick来实现以1秒的时间间隔,闪亮一个LED指示灯,指示灯接在GPIOA.8,低电平点亮。

#include “stm32f10x.h”

//函数声明

void GPIO_Configuration(void);//设置GPIOA.8端口

u32 t;//定义一个全局变量

int main(void)

{

// SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);

SysTick_Config(9000000);

SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);

GPIO_Configuration();

while(1);

}

//GPIOA.8设置函数

void GPIO_Configuration(void)

{

GPIO_InitTypeDef GPIO_InitStruct;//定义一个端口初始化结构体

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);//打开GPIOA口时钟

GPIO_InitStruct.GPIO_Mode=GPIO_Mode_Out_PP;//设置为推挽输出

GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;//设置输出频率50M

GPIO_InitStruct.GPIO_Pin=GPIO_Pin_8;//指定第8脚

GPIO_Init(GPIOA,&GPIO_InitStruct);//初始化GPIOA.8

GPIO_SetBits( GPIOA, GPIO_Pin_8);//置高GPIOA.8,关闭LED

}

//systick中断函数

void SysTick_Handler(void)

{

t++;

if(t》=1)

{

if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_8)==1)

{GPIO_ResetBits( GPIOA, GPIO_Pin_8);}

}

if(t》=2)

{

if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_8)==0)

{GPIO_SetBits( GPIOA, GPIO_Pin_8);}

t=0;

}

}

模拟后的结果

1、8分频后结果

STM32知识:什么是SYSTICK 作用是什么

总结:

1、要使用systick定时器,只需调用SysTick_Config(uint32_t ticks)函数即可,

函数自动完成:重装载值的装载,时钟源选择,计数寄存器复位,中断优先级的设置(最低),开中断,开始计数的工作。

2、要修改时钟源调用SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource),也可按照SysTick_Config()中默认设置FCLK不变。

3、要修改中断优先级调用

void NVIC_SetPriority(IRQn_Type IRQn, uint32_t priority)

应用说明:

1、因systick是一个24位的定时器,故重装值最大值为2的24次方=16 777 215,

要注意不要超出这个值。

2、systick是cortex_m3的标配,不是外设。故不需要在RCC寄存器组打开他的时钟。

3、每次systick溢出后会置位计数标志位和中断标志位,计数标志位在计数器重装载后被清除,而中断标志位也会随着中断服务程序的响应被清除,所以这两个标志位都不需要手动清除。

4、采用使用库函数的方法,只能采用中断的方法响应定时器计时时间到,如要采用查询的方法,那只能采用设置systick的寄存器的方法,具体操作以后再做分析。

关键字:STM32  SYSTICK  系统节拍定时器 编辑:什么鱼 引用地址:STM32知识:什么是SYSTICK 作用是什么

上一篇:stm32之.hex 文件
下一篇:STM32之FSMC驱动LCD屏应用

推荐阅读

STM32实战案例分享:剖析STM32应用在电源项目上常见的那些难题
我们在从事STM32单片机的应用开发及调试过程中,往往会碰到各类异常。其中有不少比例的问题跟电源有关。对于一个电子产品而言,电源部分很关键、很重要,但在实际开发调试中,我们偶尔会有意无意的忽视它。这里分享几个实际案例,以加强刺激,加深印象。 毕竟因为电源问题可能导致的异常很多很多,这里分享几个案例算是抛砖引玉,希望大家在调试中对电源方面加以重视。个人认为,往往电源出问题时导致的异常时并不太好分析,多数时候异常表现得更为诡异或没章法。【注:下面提到的案例中异常原因都与电源有关,但并不是说出现类似异常时一定是电源的原因。】 下面主要分享几个基于STM32应用的案例。 案例1:ADC功能异常 某人使用STM32芯片的ADC功能
发表于 2023-05-19
<font color='red'>STM32</font>实战案例分享:剖析<font color='red'>STM32</font>应用在电源项目上常见的那些难题
时钟失效后STM32还能运行是什么情况
该问题由某客户提出,发生在 STM32F103VDT6 器件上。据其工程师讲述:在其产品的设计中,STM32 的 HSE 外接 8MHz 的晶体产生振荡,然后通过 STM32 内部的PLL 倍频到 72MHz,作为 STM32 的系统时钟,驱动芯片工作。在 STM32 片外有专用的看门狗芯片,监控 STM32 的运行。STM32 内部的软件会在 STM32 的某个管脚上产生脉冲来复位看门狗。一旦 STM32 没有及时的产生脉冲来复位门狗,则看门狗会认为 STM32 运行不正常,从而复位 STM32。在对该产品做可靠性测试时,进行了对看门狗监控时钟失效能力的测试。测试的方法是:将 HSE 外接的晶体的两个端子接地,使其停止振荡,从而
发表于 2023-05-19
时钟失效后<font color='red'>STM32</font>还能运行是什么情况
快速入门STM32的学习经验总结
一、前言 假如你会使用8051,会写C语言,那么STM32本身并不需要刻意地学习。 我们要考虑的是, 我可以快速用STM32实现什么?为什么使用STM32而不是8051? 是因为51的频率太低,无法满足计算需求? 是51的管脚太少,无法满足众多外设的IO? 是51的功耗太大,电池挺不住? 是51的功能太弱,而你要使用SPI、I2C、ADC、DMA? 是51的内存太小而你要存储的东西太多? 当你需要使用STM32某些功能,而51实现不了的时候, 那STM32自然不需要学习,你会直接去寻找STM32某方面的使用方法。 比如要用spi协议的网卡、要使用串口通信、要使用rtos等等... 快速上手的学习步骤 我们假定大家已经对STM32的
发表于 2023-05-19
STM32系列芯片的命名规则
  示例:     从上面的料号可以看出以下信息:   ST品牌ARMCortex-Mx系列内核32位超值型MCU,LQFP-48封装闪存容量32KB温度范围-40℃-85℃;      1.产品系列:   STM32代表ST品牌Cortex-Mx系列内核(ARM)的32位MCU;   2.产品类型: F:通用快闪(FlashMemory);   L:低电压(1.65~3.6V);   F类型中F0xx和F1xx系列为2.0~3.6V;F2xx和F4xx系列为1.8~3.6V;   W:无线系统芯片,开发版。   3.产品子系列:   050:ARMCortex-M0内核;   051:ARMCortex-M0内核;  
发表于 2023-05-19
<font color='red'>STM32</font>系列芯片的命名规则
STM32的IIC通信原理详解
本文将介绍STM32 IIC的通信原理和协议 ①IIC总线简介 ②IIC总线协议与读写操作 ③STM32 IIC控制器介绍 ①IIC总线简介 IIC是inter integrated circuit的简称,IIC是由PHILIPS公司开发的两线式串行总线;该总线具有接口线少、易于控制、通讯速率高等有点,在微电子控制领域被广泛使用。 IIC总线具备以下特征: 1、同步通信,半双工,以字节为传输单位; 2、两条线路、SDA和SCL; 3、挂载在IIC总线上的设备均可为主设备、亦可为从设备; 4、具有3种传输速率,最高可达3.4Mbit/s;可通过总线时钟的频率和总线上拉电阻来配置传输速率; 5、多主机功能、7位和10位地址模式、可以软
发表于 2023-05-19
<font color='red'>STM32</font>的IIC通信原理详解
STM32和AD5791的转换
现在很多智能测量仪表要求具有超高精度的电压信号,同时要求高稳定性、高线形度和低噪声、低温度漂移。这样的模拟系统设计面临复杂的工程技术挑战,常规的方法是采用多个较低分辨率的DAC和大量分立元件与支持IC整合在一起,同时伴随着相当大的开发风险和高代价的修改时间,才能优化电路参数、减小误差和设计出复杂的自动校准电路,这样不仅增加了硬件设计的复杂性,通常达到的精度也不是很高。本系统设计的基于STM32微处理器和AD5791的20位超高精度测量系统中,实现了单路超高精度可调电压信号的输出,输出电压信号的幅值可以通过软件来设置。该系统可靠性高,不需要校准电路。 STM32F2, STM32F4, STM32F7, STM32H7,STM32L
发表于 2023-05-19
<font color='red'>STM32</font>和AD5791的转换
小广播
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2023 EEWORLD.com.cn, Inc. All rights reserved