独立看门狗:当没有到设定时间之前,给它喂了狗,就会回到初始值。
窗口看门狗:只有在设定的窗口时间范围内喂才可以起效果
最适合哪些要求看门狗在精确计时窗口起作用。
1)输入捕获:通过输入电平进行捕获
2)产生中断:
定时器时间到
触发事件
3)正交编码器和霍尔传感器(测小车速度)
1)最多只能计数65535(2的16次方)
2)预分频器(因为在定时器的频率不需要那么快)
3)可以产生中断
我们之前的51只有一个主线(无操作系统),一个进程走到死
我们想要多种进程并行工作,但是实际上无法这样。因为后面的程序可能还没有等到执行,时间就结束了。所以我们将每一个程序都分成多个小进程,第一个进程的第一个小部分执行完成在执行第二个进程的第一小部分....【从而实现微观上的串行,宏观上的并行】
1)专用于生产RTOS的系统滴答时钟【因为RTOS需要多进程执行】
2)可用于裸机程序中短时间精确延时函数
3)可用于普通定时器中断功能
https://www.st.com/resource/en/programming_manual/cd00228163-stm32f10xxx-20xxx-21xxx-l1xxxx-cortex-m3-programming-manual-stmicroelectronics.pdf
SYSTICK和NVIC不属于SoC部分【属于内核外设】
我们计算出来的值要-1
因为计时是4 3 2 1 0 4 3 2 1 0【0才表示结束】
校准
SYSTICK放在misc.c中
本函数在misc.c中
SYSTICK本身没有分频器。所以需要通过本函数进行设置
/**
* @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
本函数在core_cm3.h中
Systick config函数配置的状况是: 默认使用AHB时钟。会产生中断,中断优先级为最低,并且最末尾启动了定时器
/**
* @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 */
我们在调用库函数进行初始化的时候,要先调用【SysTick_Config】然后再调用【SysTick_CLKSourceConfig】,因为再【SysTick_Config】中对Systick_CTRL中是对其中几位bit进行直接赋值,而不是位或。所以如果先定义【SysTick_CLKSourceConfig】则相关寄存器的值可能被覆盖掉。
使能后,到中断处理程序查
检测STK_VAL,因为VAL会不断减少
我们使用原始的频率:72MHZ=72 000 000HZ
以1s为单位---》1ms==0.001s
CNT=72 000 000*0.001=72 000
查看是否超过2的24次方
https://www.cnblogs.com/kinson/p/7967332.html
1)SYSTICK是自动清除中断,不需要手动将其清除
2)SYSTICK是内核中的,所以不需要打开SYSTICK时钟,它一直都是打开的
我们将led的j19接到PB0-PB7,但是实际上我们就操作led1,所以使用到PB0
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);//初始化中断
*/
}
//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);
}
//RCC的配置
void RCC_Configuration(void){
//因为起始代码中已经调用SystemInit将主时钟设置为72MHZ
//所以我们这里RCC直接使能时钟就可以
//使能GPIO端口
//通过PB0控制LED1
//因为我们使用到的是PB0,所以只使用到GPIOB
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
//因为systick属于cpu部分的,所以时钟始终打开的,不需要再去打开
//所以这里只是打开GPIO的时钟即可
}
//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);
}
#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);
}
我们已经知道systick有24个寄存器,所以我们最大的数值范围不能超过2的24次方。
所以我们计算出来的值不能超过上面那个
而根据我们单片机上面使用的是72MHZ的频率
则我们ticks的范围是16 777 216/72 000 000
如果我们想要设置100ms
则ticks=72 000 000 *0.1=7,200,000
我们通过使用delay进行精确的延时
//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);
}
//RCC的配置
void RCC_Configuration(void){
//因为起始代码中已经调用SystemInit将主时钟设置为72MHZ
//所以我们这里RCC直接使能时钟就可以
//使能GPIO端口
//通过PB0控制LED1
//因为我们使用到的是PB0,所以只使用到GPIOB
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
//因为systick属于cpu部分的,所以时钟始终打开的,不需要再去打开
//所以这里只是打开GPIO的时钟即可
}
此代码要根据频率来进行修改
//用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;
}
#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;
}
查询和中断方式差异:
查询方式是阻塞式的,中断方式是非阻塞的