【STM32】Systick定时器

一、STM32的5种定时器简介

1.独立看门狗(IWDG) VS 窗口看门狗(WWDG)

1.独立看门狗(IWDG)

独立看门狗:当没有到设定时间之前,给它喂了狗,就会回到初始值。

2.窗口看门狗(WWDG)

窗口看门狗:只有在设定的窗口时间范围内喂才可以起效果

最适合哪些要求看门狗在精确计时窗口起作用。

2.Systick定时器

3.高级定时器(TIM1+TIM8)

4.通用定时器(TIMx)

1)输入捕获:通过输入电平进行捕获

2)产生中断:

定时器时间到

触发事件

3)正交编码器和霍尔传感器(测小车速度)

5.基本定时器(TIM6 & TIM7)

1)最多只能计数65535(2的16次方)

2)预分频器(因为在定时器的频率不需要那么快)

3)可以产生中断

二、SYSTICK定时器

0.SYSTICK的提出

我们之前的51只有一个主线(无操作系统),一个进程走到死

我们想要多种进程并行工作,但是实际上无法这样。因为后面的程序可能还没有等到执行,时间就结束了。所以我们将每一个程序都分成多个小进程,第一个进程的第一个小部分执行完成在执行第二个进程的第一小部分....****【从而实现微观上的串行,宏观上的并行】

1.SYSTICK定时器的作用

1)专用于生产RTOS的系统滴答时钟【因为RTOS需要多进程执行】

2)可用于裸机程序中短时间精确延时函数

3)可用于普通定时器中断功能

2.SYSTICK定时器的数据手册

https://www.st.com/resource/en/programming_manual/cd00228163-stm32f10xxx-20xxx-21xxx-l1xxxx-cortex-m3-programming-manual-stmicroelectronics.pdf

SYSTICK和NVIC不属于SoC部分【属于内核外设】

1.24个定时器

2.各种寄存器

1.STK_CTRL(控制状态)
2.STK_LOAD(初始值)

我们计算出来的值要-1

因为计时是4 3 2 1 0 4 3 2 1 0【0才表示结束】

3.STK_VAL(计数值)
4.STK_CALIB

校准

3.SYSTICK寄存器在标准库中的封装

SYSTICK放在misc.c中

1. SysTick_CLKSourceConfig--分频设置

本函数在misc.c中
SYSTICK本身没有分频器。所以需要通过本函数进行设置

cpp 复制代码
/**
  * @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

2.SysTick_Config--触发systick中断

本函数在core_cm3.h中
Systick config函数配置的状况是:默认使用AHB时钟。会产生中断,中断优先级为最低,并且最末尾启动了定时器

cpp 复制代码
/**
 * @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.
 */
 /**
	Systick config函数配置的状况是:
		默认使用AHB时钟。会产生中断
		中断优先级为最低,并且最末尾启动了定时器
*/
static __INLINE uint32_t SysTick_Config(uint32_t ticks)
{ 
//检验有没有超过24位寄存器
//SysTick_LOAD_RELOAD_Msk:  0xFFFFFFul
  if (ticks > SysTick_LOAD_RELOAD_Msk)  return (1);            /* Reload value impossible */
//减一:因为我们是从0开始的 ,但是一般我们都不在乎因为影响不大                                                              
  SysTick->LOAD  = (ticks & SysTick_LOAD_RELOAD_Msk) - 1;      /* set reload register */
//设置中断优先级,默认设置最低
  NVIC_SetPriority (SysTick_IRQn, (1<<__NVIC_PRIO_BITS) - 1);  /* set Priority for Cortex-M0 System Interrupts */
  SysTick->VAL   = 0;      //让它上来直接完了                                    /* Load the SysTick Counter Value */
  //SysTick_CTRL_CLKSOURCE_Msk(1):默认使用AHB
  //SysTick_CTRL_TICKINT_Msk:默认会产生中断
  //SysTick_CTRL_ENABLE_Msk:启动了定时器
  SysTick->CTRL  = SysTick_CTRL_CLKSOURCE_Msk | 
                   SysTick_CTRL_TICKINT_Msk   | 
                   SysTick_CTRL_ENABLE_Msk;                    /* Enable SysTick IRQ and SysTick Timer */
  return (0);                                                  /* Function successful */

3.注意点:SysTick_Config VS SysTick_CLKSourceConfig的调用顺序

我们在调用库函数进行初始化的时候,要先调用【SysTick_Config】然后再调用【SysTick_CLKSourceConfig】,因为再【SysTick_Config】中对Systick_CTRL中是对其中几位bit进行直接赋值,而不是位或。所以如果先定义【SysTick_CLKSourceConfig】则相关寄存器的值可能被覆盖掉。

4.SYSTICK定时器的2种工作方式

1.中断方式

使能后,到中断处理程序查

2.查询方式

检测STK_VAL,因为VAL会不断减少

5.SYSTICK定时器的定时计算

1.公式:重装载值=systick时钟频率(Hz)*想要定的时间(s)

2.例子:1ms

我们使用原始的频率:72MHZ=72 000 000HZ

以1s为单位---》1ms==0.001s

CNT=72 000 000*0.001=72 000

查看是否超过2的24次方

3.查询方式和中断方式都这样计算

6.SYSTICK中断实现LED每200ms闪烁一次【中断方式-interrupt】

https://www.cnblogs.com/kinson/p/7967332.html

0.注意点:

1)SYSTICK是自动清除中断,不需要手动将其清除

2)SYSTICK是内核中的,所以不需要打开SYSTICK时钟,它一直都是打开的

1.接线

我们将led的j19接到PB0-PB7,但是实际上我们就操作led1,所以使用到PB0

2.NVIC_Configuration

因为我们的Systick中断不属于外设中断,所以不需要设置NVIC的部分设置。

cpp 复制代码
void NVIC_Configuration(void)
{
 // NVIC_InitTypeDef NVIC_InitStructure;
  
#ifdef  VECT_TAB_RAM  
  /* Set the Vector Table base location at 0x20000000 */ 
  NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0);   //分配中断向量表
#else  /* VECT_TAB_FLASH  */
	//表示从FLASH中启动;;
  /* Set the Vector Table base location at 0x08000000 */ 
  NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0);   
#endif

  /* Configure one bit for preemption priority */
  // NVIC_PriorityGroup_1:2个抢占优先级,8个次优先级
	
	/*
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);  //设置中断优先级
  
  // Enable the SYSTICK Interrupt 
	//设置为SYSTICK
	//这里我们将下面代码注释起来是因为【NVIC_IRQChannel】只接受正整数
	//但是我们【SysTick_IRQn】是负数,所以不能正确输出
  NVIC_InitStructure.NVIC_IRQChannel = SysTick_IRQn; //中断通道
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority =  0;  //强占优先级
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;//次优先级
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;  //通道中断使能
  NVIC_Init(&NVIC_InitStructure);//初始化中断
	*/
}

3.GPIO_Configuration

cpp 复制代码
//GPIO初始化
void GPIO_Configuration(void){
	
	GPIO_InitTypeDef GPIO_InitStructure;

	
	//PB0 ---LED1【LED的显示输出】
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;       //推挽输出
  GPIO_Init(GPIOB, &GPIO_InitStructure);
	// 默认输出0让LED亮
	//RESET=0
	//SET=1
  GPIO_WriteBit(GPIOB, GPIO_Pin_8, Bit_RESET);			

}

4.RCC_Configuration

cpp 复制代码
//RCC的配置
void RCC_Configuration(void){
	
	//因为起始代码中已经调用SystemInit将主时钟设置为72MHZ
	//所以我们这里RCC直接使能时钟就可以
	
	//使能GPIO端口
	//通过PB0控制LED1
	//因为我们使用到的是PB0,所以只使用到GPIOB
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
	//因为systick属于cpu部分的,所以时钟始终打开的,不需要再去打开
	//所以这里只是打开GPIO的时钟即可
	
}

5. SYSTICK_Configuration

cpp 复制代码
//SysTick_Config:是SYSTICK的启动函数
void SYSTICK_Configuration(void){
	/*
	这个时间超时了
	//主频是72MHZ,定时时间是500ms
	//ticks=72 000 000*0.5 =3600 000
	SysTick_Config(36000000);//1677 7216
	*/
	//100毫秒
	//范围:233ms
	//ticks=72 000 000*200ms=14400000
	SysTick_Config(14400000);
	//SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);
}

6.main

cpp 复制代码
#include "stm32f10x.h"                  // Device header
/**
	使用SYSTICK控制led的闪烁【中断式】
	PB8控制LED8
*/

//函数声明
//RCC的配置
void RCC_Configuration(void);
//GPIO初始化
void GPIO_Configuration(void);
void NVIC_Configuration(void);
void SYSTICK_Configuration(void);

//全局变量定义
EXTI_InitTypeDef EXTI_InitStructure;
ErrorStatus HSEStatartUpStatus;


int main(){
	//系统时钟配置
	RCC_Configuration();
	//NVIC配置
	NVIC_Configuration();
	//配置GPIO
	GPIO_Configuration();
	
	SYSTICK_Configuration();
	
	while(1);
	return 0;
}

//RCC的配置
void RCC_Configuration(void){
	
	//因为起始代码中已经调用SystemInit将主时钟设置为72MHZ
	//所以我们这里RCC直接使能时钟就可以
	
	//使能GPIO端口
	//通过PB0控制LED1
	//因为我们使用到的是PB0,所以只使用到GPIOB
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
	//因为systick属于cpu部分的,所以时钟始终打开的,不需要再去打开
	//所以这里只是打开GPIO的时钟即可
	
}


//GPIO初始化
void GPIO_Configuration(void){
	
	GPIO_InitTypeDef GPIO_InitStructure;

	
	//PB0 ---LED1【LED的显示输出】
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;       //推挽输出
  GPIO_Init(GPIOB, &GPIO_InitStructure);
	// 默认输出0让LED亮
	//RESET=0
	//SET=1
  GPIO_WriteBit(GPIOB, GPIO_Pin_8, Bit_RESET);			

}

void NVIC_Configuration(void)
{
 // NVIC_InitTypeDef NVIC_InitStructure;
  
#ifdef  VECT_TAB_RAM  
  /* Set the Vector Table base location at 0x20000000 */ 
  NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0);   //分配中断向量表
#else  /* VECT_TAB_FLASH  */
	//表示从FLASH中启动;;
  /* Set the Vector Table base location at 0x08000000 */ 
  NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0);   
#endif

  /* Configure one bit for preemption priority */
  // NVIC_PriorityGroup_1:2个抢占优先级,8个次优先级
	
	/*
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);  //设置中断优先级
  
  // Enable the SYSTICK Interrupt 
	//设置为SYSTICK
	//这里我们将下面代码注释起来是因为【NVIC_IRQChannel】只接受正整数
	//但是我们【SysTick_IRQn】是负数,所以不能正确输出
  NVIC_InitStructure.NVIC_IRQChannel = SysTick_IRQn; //中断通道
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority =  0;  //强占优先级
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;//次优先级
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;  //通道中断使能
  NVIC_Init(&NVIC_InitStructure);//初始化中断
	*/
}

//SysTick_Config:是SYSTICK的启动函数
void SYSTICK_Configuration(void){
	/*
	这个时间超时了
	//主频是72MHZ,定时时间是500ms
	//ticks=72 000 000*0.5 =3600 000
	SysTick_Config(36000000);//1677 7216
	*/
	//100毫秒
	//范围:233ms
	//ticks=72 000 000*200ms=14400000
	SysTick_Config(14400000);
	//SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);
}

7.注意点:ticks的时间设置

我们已经知道systick有24个寄存器,所以我们最大的数值范围不能超过2的24次方。

所以我们计算出来的值不能超过上面那个

而根据我们单片机上面使用的是72MHZ的频率

则我们ticks的范围是16 777 216/72 000 000

举个例子

如果我们想要设置100ms

则ticks=72 000 000 *0.1=7,200,000

8.SYSTICK中断实现LED每200ms闪烁一次【查询方式--delay】

0.解释

我们通过使用delay进行精确的延时

1.代码解析

1.GPIO_Configuration

cpp 复制代码
//GPIO初始化
void GPIO_Configuration(void){
	
	GPIO_InitTypeDef GPIO_InitStructure;

	
	//PB0 ---LED1【LED的显示输出】
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;       //推挽输出
  GPIO_Init(GPIOB, &GPIO_InitStructure);
	// 默认输出0让LED亮
	//RESET=0
	//SET=1
  GPIO_WriteBit(GPIOB, GPIO_Pin_8, Bit_RESET);			

}

2.RCC_Configuration

cpp 复制代码
//RCC的配置
void RCC_Configuration(void){
	
	//因为起始代码中已经调用SystemInit将主时钟设置为72MHZ
	//所以我们这里RCC直接使能时钟就可以
	
	//使能GPIO端口
	//通过PB0控制LED1
	//因为我们使用到的是PB0,所以只使用到GPIOB
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
	//因为systick属于cpu部分的,所以时钟始终打开的,不需要再去打开
	//所以这里只是打开GPIO的时钟即可
	
}

3.delay_us

此代码要根据频率来进行修改

cpp 复制代码
//用systick计数器来帮我们实现us级别的精确延时
//这个函数成立有2个条件:
//1.主频必须是72MHZ
//2.us要小于(2的24次方)=1864 135us=1.8s
void delay_us(unsigned int us){
	
	//记录CTRL的countflag位的值
	unsigned int tmp=0;
	
	//思路是先把systick的时钟源设置好,然后给一个正确的ticks
	//然后使能systick,while循环等待countflag置位则时间到
	
	//72MHZ主频,我们使用8分频,72/8=9MHZ systick
	 SysTick->LOAD=us*9;
	 SysTick->VAL=0;
	//我们要先设置上面的LOAD和VAL,在进行使能
	//时钟源是AHB/8,禁止中断,使能systick
	 SysTick->CTRL =0x01;
	
	//检测什么时候时间结束
	//查看CTRL中的countflag
	do{
		tmp=SysTick->CTRL;
	}while(!(tmp&(1<<16)));
	
	//时间到,关闭定时器
	SysTick->VAL=0;
	SysTick->CTRL =0x00;
}

4.delay_ms

cpp 复制代码
//不能大于 1864
void delay_ms(unsigned int ms){
	
	//记录CTRL的countflag位的值
	unsigned int tmp=0;
	
	//思路是先把systick的时钟源设置好,然后给一个正确的ticks
	//然后使能systick,while循环等待countflag置位则时间到
	
	//72MHZ主频,我们使用8分频,72/8=9MHZ systick
	 SysTick->LOAD=ms*9000;
	 SysTick->VAL=0;
	//我们要先设置上面的LOAD和VAL,在进行使能
	//时钟源是AHB/8,禁止中断,使能systick
	 SysTick->CTRL =0x01;
	
	//检测什么时候时间结束
	//查看CTRL中的countflag
	do{
		tmp=SysTick->CTRL;
	}while(!(tmp&(1<<16)));
	
	//时间到,关闭定时器
	SysTick->VAL=0;
	SysTick->CTRL =0x00;
	
}

5.main

cpp 复制代码
#include "stm32f10x.h"                  // Device header
/**
	使用delay控制led的闪烁--与systick进行等价实现【查询式】
	PB8控制LED8
*/

//函数声明
//RCC的配置
void RCC_Configuration(void);
//GPIO初始化
void GPIO_Configuration(void);

void delay_ms(unsigned int ms);
void delay_us(unsigned int us);

//全局变量定义
EXTI_InitTypeDef EXTI_InitStructure;
ErrorStatus HSEStatartUpStatus;


int main(){
	//系统时钟配置
	RCC_Configuration();
	
	//配置GPIO
	GPIO_Configuration();
	
	
	while(1){
		GPIO_WriteBit(GPIOB,GPIO_Pin_8,Bit_RESET);//亮
		delay_ms(200);
		GPIO_WriteBit(GPIOB,GPIO_Pin_8,Bit_SET);//灭
		delay_ms(200);
	}
	return 0;
}

//RCC的配置
void RCC_Configuration(void){
	
	//因为起始代码中已经调用SystemInit将主时钟设置为72MHZ
	//所以我们这里RCC直接使能时钟就可以
	
	//使能GPIO端口
	//通过PB0控制LED1
	//因为我们使用到的是PB0,所以只使用到GPIOB
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
	//因为systick属于cpu部分的,所以时钟始终打开的,不需要再去打开
	//所以这里只是打开GPIO的时钟即可
	
}


//GPIO初始化
void GPIO_Configuration(void){
	
	GPIO_InitTypeDef GPIO_InitStructure;

	
	//PB0 ---LED1【LED的显示输出】
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;       //推挽输出
  GPIO_Init(GPIOB, &GPIO_InitStructure);
	// 默认输出0让LED亮
	//RESET=0
	//SET=1
  GPIO_WriteBit(GPIOB, GPIO_Pin_8, Bit_RESET);			

}

//用systick计数器来帮我们实现us级别的精确延时
//这个函数成立有2个条件:
//1.主频必须是72MHZ
//2.us要小于(2的24次方)=1864 135us=1.8s
void delay_us(unsigned int us){
	
	//记录CTRL的countflag位的值
	unsigned int tmp=0;
	
	//思路是先把systick的时钟源设置好,然后给一个正确的ticks
	//然后使能systick,while循环等待countflag置位则时间到
	
	//72MHZ主频,我们使用8分频,72/8=9MHZ systick
	 SysTick->LOAD=us*9;
	 SysTick->VAL=0;
	//我们要先设置上面的LOAD和VAL,在进行使能
	//时钟源是AHB/8,禁止中断,使能systick
	 SysTick->CTRL =0x01;
	
	//检测什么时候时间结束
	//查看CTRL中的countflag
	do{
		tmp=SysTick->CTRL;
	}while(!(tmp&(1<<16)));
	
	//时间到,关闭定时器
	SysTick->VAL=0;
	SysTick->CTRL =0x00;
}


//不能大于 1864
void delay_ms(unsigned int ms){
	
	//记录CTRL的countflag位的值
	unsigned int tmp=0;
	
	//思路是先把systick的时钟源设置好,然后给一个正确的ticks
	//然后使能systick,while循环等待countflag置位则时间到
	
	//72MHZ主频,我们使用8分频,72/8=9MHZ systick
	 SysTick->LOAD=ms*9000;
	 SysTick->VAL=0;
	//我们要先设置上面的LOAD和VAL,在进行使能
	//时钟源是AHB/8,禁止中断,使能systick
	 SysTick->CTRL =0x01;
	
	//检测什么时候时间结束
	//查看CTRL中的countflag
	do{
		tmp=SysTick->CTRL;
	}while(!(tmp&(1<<16)));
	
	//时间到,关闭定时器
	SysTick->VAL=0;
	SysTick->CTRL =0x00;
	
}

2.中断 VS 查询

查询和中断方式差异:

查询方式是阻塞式的,中断方式是非阻塞的

相关推荐
yutian06066 小时前
Keil MDK下载程序后MCU自动重启设置
单片机·嵌入式硬件·keil
析木不会编程9 小时前
【小白51单片机专用教程】protues仿真独立按键控制LED
单片机·嵌入式硬件·51单片机
枯无穷肉13 小时前
stm32制作CAN适配器4--WinUsb的使用
stm32·单片机·嵌入式硬件
不过四级不改名67713 小时前
基于HAL库的stm32的can收发实验
stm32·单片机·嵌入式硬件
嵌入式科普13 小时前
十一、从0开始卷出一个新项目之瑞萨RA6M5串口DTC接收不定长
c语言·stm32·cubeide·e2studio·ra6m5·dma接收不定长
嵌入式大圣13 小时前
单片机UDP数据透传
单片机·嵌入式硬件·udp
云山工作室14 小时前
基于单片机的视力保护及身姿矫正器设计(论文+源码)
stm32·单片机·嵌入式硬件·毕业设计·毕设
嵌入式-老费14 小时前
基于海思soc的智能产品开发(mcu读保护的设置)
单片机·嵌入式硬件
qq_3975623116 小时前
MPU6050 , 设置内部低通滤波器,对于输出数据的影响。(简单实验)
单片机
liyinuo201716 小时前
嵌入式(单片机方向)面试题总结
嵌入式硬件·设计模式·面试·设计规范