STM32F4的系统滴答计时器的介绍及其说明。时间有限,这里点到为止,详情自行百度。
延时的原理:
因为在 ucos 下 systick 不能再被随意更改,如果我们还想利用 systick 来做 delay_us 或者
delay_ms 的延时,就必须想点办法了,这里我们利用的是时钟摘取法。
---这里摘自正点原子探索者F4,详细原理请自行百度。
使用Systic计时器实现精准延时,且不占用OS中断
延时系统具有通用性,可适用多种系统频率
本程序流程如下:
初始化其它相关硬件->获取HCLK时钟频率->用HCLK时钟频率初始化Systic计时器->其它操作实现延时
刚开始测试的时候发现了一个问题,如下描述:
由于该程序修改自正点原子的源码,源码中系统时钟是预先定义好的,使用起来没有问题,但这里系统频率是调用库函数实现的,虽然具有通用信,但却存在问题。
库函数中获取时钟频率的函数RCC_GetClocksFreq正确使用有一个条件,就是外部时钟25MHz
我们看一看该 函数的部分说明:
HSE_VALUE is a constant defined in stm32f4xx.h file (default value
25 MHz), user has to ensure that HSE_VALUE is same as the real
frequency of the crystal used. Otherwise, this function may
have wrong result.
意思就是外部晶振不为25MHz时算出的频率会出错。
表现在该工程中就是延时的时间是标准时间的三倍左右,即25/8
那么如何解决这个问题?很简单
stm32f4xx.h做以下修改
源代码:
123行: #define HSE_VALUE ((uint32_t)25000000) /*!< Value of the External oscillator in Hz */
改为: 123行: #define HSE_VALUE ((uint32_t)8000000) /*!< Value of the External oscillator in Hz */
这里是指默认的外部高速时钟25Mhz,实际为8MHz,这里为了使库函数获取的时钟更准确
修改后,测试基本符合。
/**
******************************************************************************
* @file Delay.h
* @author 李航兵
* @version V1.2
* @date 18-June-2018
* @brief 这个头文件包含了STM32F407ZGT6开发板的延时相关的函数.
******************************************************************************
* @attention
* 官方库文件里有参考代码
* 为保证延时系统的精准,要求系统主频率越高越好
* 为保证延时系统的完整支持,系统主频率至少8MHz
* 使用CM4系统定时器,中断优先级最低
*
* @更新说明
* 支持us、ms级延时,且精度大大提高
* 支持OS,基本不影响OS的使用(下个版本添加)
*
******************************************************************************
*/
/* Define to prevent recursive inclusion -------------------------------------*/
#ifndef __DELAY_H
#define __DELAY_H
/* Includes ------------------------------------------------------------------*/
#include "stm32f4xx_rcc.h"
#include "core_cm4.h"
#include "stm32f4xx_it.h"
#include "misc.h"
/* Exported types ------------------------------------------------------------*/
/* Exported constants --------------------------------------------------------*/
/* Exported macro ------------------------------------------------------------*/
/* Exported functions ------------------------------------------------------- */
void Delay_Init(void); //延时系统初始化
void Delay_Us(u16 us); //微秒延时
void Delay_Ms(u16 ms); //毫秒延时
void Delay_Us_2(u32 us); //微秒延时,范围更大
#endif /* __DELAY_H */
/************************ (C) COPYRIGHT 李航兵 *****END OF FILE****/
/**
******************************************************************************
* @file Led.h
* @author 李航兵
* @version V1.0
* @date 14-June-2018
* @brief 这个头文件包含了STM32F407ZGT6开发板的Led相关的函数.
******************************************************************************
* @attention
*硬件连接:
* Led1:PC0-VCC
* Led1:PD3-VCC
*
******************************************************************************
*/
/* Define to prevent recursive inclusion -------------------------------------*/
#ifndef __LED_H
#define __LED_H
/* Includes ------------------------------------------------------------------*/
#include "stm32f4xx_gpio.h"
#include "stm32f4xx_rcc.h"
/* Exported types ------------------------------------------------------------*/
/* Exported constants --------------------------------------------------------*/
/* Exported macro ------------------------------------------------------------*/
/* Exported functions ------------------------------------------------------- */
void Led_Init(void); //Led初始化
void Led_On(u8 led); //开灯
void Led_Off(u8 led); //关灯
#endif /* __LED_H */
/************************ (C) COPYRIGHT 李航兵 *****END OF FILE****/
/**
******************************************************************************
* @file Delay.c
* @author 李航兵
* @version V1.2
* @date 18-June-2018
* @brief 这个文件包含以下函数用于操作STM32F407ZGT6开发板的延时功能
* + 延时系统初始化
* + 微秒延时
* + 毫秒延时
*
* @verbatim
*
*
===========================================================================
##### How to use this driver #####
===========================================================================
[..]
(#) 初始化延时系统,调用Delay_Init()
(#) 延时
(++) 毫秒: Delay_Ms()
(++) 微秒: Delay_Init()
@endverbatim
******************************************************************************
* @attention
* 官方库文件里有参考代码
* 为保证延时系统的精准,要求系统主频率越高越好
* 为保证延时系统的完整支持,系统主频率至少8MHz
* 使用CM4系统定时器,中断优先级最低
* 使用此文件后,禁止再修改SysTick计时器及其设置
*
*
*
*
*
*
* © COPYRIGHT 2018 李航兵
* @更新说明
* 支持us、ms级延时,且精度大大提高
* 支持OS,基本不影响OS的使用(下个版本添加)
*
*
*
*
*
*
******************************************************************************
*/
/* Includes ------------------------------------------------------------------*/
#include "Delay.h"
/* Private typedef -----------------------------------------------------------*/
/* Private define ------------------------------------------------------------*/
/* Private macro -------------------------------------------------------------*/
/* Private variables ---------------------------------------------------------*/
u8 per_us; //每1us定时器节拍
u32 per_ms; //每1ms节拍,注意168MHz下值为168000,需要32位,移植自STM32F0,此处谨慎
/* Private function prototypes -----------------------------------------------*/
/* Private functions ---------------------------------------------------------*/
/**
* @brief 延时系统初始化.
* @param None.
* @retval None
*/
void Delay_Init()
{
RCC_ClocksTypeDef RCC_Clocks;
RCC_GetClocksFreq(&RCC_Clocks);
//这里有个问题,本想自动化获取时钟频率的,不料在该函数中有如下声明
/*
HSE_VALUE is a constant defined in stm32f4xx.h file (default value
* 25 MHz), user has to ensure that HSE_VALUE is same as the real
* frequency of the crystal used. Otherwise, this function may
* have wrong result.
*/
//也就是后来测试时发现延时总是为输入的3倍左右
//后修改了库文件,可参看对应目录的readme文件获取修改记录
if(SysTick_Config(RCC_Clocks.HCLK_Frequency / 1000)) //1ms一次中断
while(1);
per_ms=SysTick->LOAD; //每1ms节拍,亦即重载值
per_us=per_ms/1000; //每1us定时器节拍
}
/**
* @brief 微秒延时.
* @param 延时的微秒数,约定范围1~390,“禁止其他值”.
* @note 存在一定误差,主要是函数调用+部分计算.
* @retval None
*/
void Delay_Us(u16 us) //微秒延时
{
u32 ticks_old=SysTick->VAL; //前一个计数值
u32 ticks_new; //后一个计数值
u16 ticks_sum=0; //已经经过的节拍
u16 ticks_delta=us*per_us; //需要经过的节拍
if(us>390) return; //计时不允许超过390us,超过390us请使用Delay_Us_2
while(1)
{
ticks_new=SysTick->VAL;
if(ticks_new!=ticks_old)
{
if(ticks_new=ticks_delta)break; //时间超过/等于要延迟的时间,则退出.
}
}
}
/**
* @brief 毫秒延时.
* @param 延时的毫秒数,约定范围1~25000,“禁止其他值”.
* @note 存在一定误差,主要是函数调用+部分计算.
* @retval None
*/
void Delay_Ms(u16 ms) //毫秒延时
{
u32 ticks_old=SysTick->VAL; //前一个计数值
u32 ticks_new; //后一个计数值
u32 ticks_sum=0; //已经经过的节拍
u32 ticks_delta=ms*per_ms; //需要经过的节拍
if(ms>25000) return; //计时不允许超过25000ms,超过25000ms请多次使用
while(1)
{
ticks_new=SysTick->VAL;
if(ticks_new!=ticks_old)
{
if(ticks_new=ticks_delta)break; //时间超过/等于要延迟的时间,则退出.
}
}
}
/**
* @brief 微秒延时,范围更大.
* @param 延时的微秒数,约定范围1~25000000,“禁止其他值”.
* @note 存在一定误差,主要是函数调用+部分计算.
* @retval None
*/
void Delay_Us_2(u32 us) //微秒延时,范围更大
{
u32 ticks_old=SysTick->VAL; //前一个计数值
u32 ticks_new; //后一个计数值
u32 ticks_sum=0; //已经经过的节拍
u32 ticks_delta=us*per_us; //需要经过的节拍
if(us>25000000) return; //计时不允许超过25000000us
while(1)
{
ticks_new=SysTick->VAL;
if(ticks_new!=ticks_old)
{
if(ticks_new=ticks_delta)break; //时间超过/等于要延迟的时间,则退出.
}
}
}
/**
* @brief 系统定时器中断.
* @param None.
* @note 一般给OS用.
* @retval None
*/
void SysTick_Handler(void) //系统定时器中断
{
}
/**
******************************************************************************
* @file Led.c
* @author 李航兵
* @version V1.0
* @date 14-June-2018
* @brief 这个文档包含了Led功能相关的函数.
******************************************************************************
* @brief 这个文件包含以下函数用于操作STM32F407ZGT6开发板的Led灯
* + 初始化Led
* + 开灯
* + 关灯
*
* @verbatim
*
*
===========================================================================
##### How to use this driver #####
===========================================================================
[..]
(#) 初始化Led,调用Led_Init()
(#) 控制灯的亮灭
(++) 开灯: Led_On
(++) 关灯: Led_Off
@endverbatim
******************************************************************************
* @attention
*
* © COPYRIGHT 2018 李航兵
*硬件连接:
* Led1:PC0-VCC
* Led1:PD3-VCC
*使用该文件后,PC0和PD3将会被占用
*
******************************************************************************
*/
/* Includes ------------------------------------------------------------------*/
#include "Led.h"
/* Private typedef -----------------------------------------------------------*/
/* Private define ------------------------------------------------------------*/
/* Private macro -------------------------------------------------------------*/
/* Private variables ---------------------------------------------------------*/
/* Private function prototypes -----------------------------------------------*/
/* Private functions ---------------------------------------------------------*/
/**
* @brief Led初始化.
* @note PC0和PD3将会被占用
* @note 调用此函数会开启GPIOC和启GPIOD时钟.
* @param None
* @retval None
*/
void Led_Init() //Led初始化
{
GPIO_InitTypeDef GPIO_Led;
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOC,ENABLE);
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOD,ENABLE);
GPIO_Led.GPIO_Mode=GPIO_Mode_OUT;
GPIO_Led.GPIO_OType=GPIO_OType_PP;
GPIO_Led.GPIO_Pin=GPIO_Pin_0;
GPIO_Led.GPIO_PuPd=GPIO_PuPd_NOPULL;
GPIO_Led.GPIO_Speed=GPIO_High_Speed;
GPIO_Init(GPIOC,&GPIO_Led);
GPIO_Led.GPIO_Pin=GPIO_Pin_3;
GPIO_Init(GPIOD,&GPIO_Led);
Led_Off(1);
Led_Off(2);
}
/**
* @brief 开灯.
* @note
* @param led:Led的编号,可取1、2
* @retval None
*/
void Led_On(u8 led) //开灯
{
switch(led)
{
case 1:
GPIO_ResetBits(GPIOC,GPIO_Pin_0);
break;
case 2:
GPIO_ResetBits(GPIOD,GPIO_Pin_3);
break;
}
}
/**
* @brief 关灯.
* @note
* @param led:Led的编号,可取1、2
* @retval None
*/
void Led_Off(u8 led) //关灯
{
switch(led)
{
case 1:
GPIO_SetBits(GPIOC,GPIO_Pin_0);
break;
case 2:
GPIO_SetBits(GPIOD,GPIO_Pin_3);
break;
}
}
/************************ (C) COPYRIGHT 李航兵 *****END OF FILE****/
#include "Delay.h"
#include "Led.h"
void main()
{
int i;
Led_Init();
Delay_Init();
while(1)
{
Led_On(1);
for(i=0;i<1000;i++)
Delay_Us(300);
Led_Off(1);
Delay_Ms(1000);
Led_On(2);
Delay_Us_2(1000000);
Led_Off(2);
Delay_Ms(1000);
}
}
//void main()
//{
// int i;
// Led_Init();
// Delay_Init();
// while(1)
// {
// Led_Off(1);
// for(i=0;i<1000;i++)
// {
// Delay_Us(300);
// Delay_Us(300);
// Delay_Us(300);
// Delay_Us(100);
// }
// Led_On(1);
//
// for(i=0;i<1000;i++)
// {
// Delay_Us(300);
// Delay_Us(300);
// Delay_Us(300);
// Delay_Us(100);
// }
// }
//}
//void main()
//{
// int i;
// Led_Init();
// Delay_Init();
// while(1)
// {
// Led_Off(1);
// Delay_Ms(1000);
// Led_On(1);
//
// Delay_Ms(1000);
// }
//}
stm32f4xx.h
经过测试,延时1s的时间基本准确