[STM32F1] STM32F103 SysTick滴答定时器实验

[复制链接]
1464|19
 楼主| qn7a12 发表于 2023-11-23 23:37 | 显示全部楼层 |阅读模式
SysTick定时器基础知识

SysTick是一个24位定时器,计数器以递减的方式工作,递减到0,硬件自动重装计数值,如果使能中断,则产生中断。且所有的Cortex M器件都有这个定时器,所有的CM芯片的SysTick都是相同的。

SysTick定时器被捆绑在NVIC中,用于产生Sysick异常(异常号:15)。

SysTick中断的优先级也可以设置。

SysTick定时器的计数频率为系统时钟频率或者1/8系统时钟频率,HAL库默认为系统时钟频率。

 楼主| qn7a12 发表于 2023-11-23 23:38 | 显示全部楼层
SysTick寄存器
60894655f7200de891.png
 楼主| qn7a12 发表于 2023-11-23 23:38 | 显示全部楼层
SysTick控制寄存器

6949655f720f6ab4f.png
 楼主| qn7a12 发表于 2023-11-23 23:42 | 显示全部楼层
SysTick控制寄存器
32274655f72eb1ded6.png
 楼主| qn7a12 发表于 2023-11-23 23:42 | 显示全部楼层
SysTick重装寄存器

20924655f72fc7578a.png
 楼主| qn7a12 发表于 2023-11-23 23:43 | 显示全部楼层
SysTick的递减计数器\

31275655f730bee532.png
 楼主| qn7a12 发表于 2023-11-23 23:43 | 显示全部楼层
寄存器代码
  1. #include "delay.h"

  2. static u8  fac_us=0;//us延时倍数                                                                  
  3. static u16 fac_ms=0;//ms延时倍数

  4. void delay_init(u8 SYSCLK)
  5. {
  6.   SysTick->CTRL&=~(1<<2);
  7.   //控制寄存器位2置0,选择8分频时钟                                         
  8.         fac_us=SYSCLK/8;
  9.         //采用参考8分频的参考时钟

  10.         fac_ms=(u16)fac_us*1000;
  11. }

  12. void delay_us(u32 nus)
  13. {               
  14.         u32 temp;                     
  15.         SysTick->LOAD=nus*fac_us;
  16.         //时间加载                                                           
  17.         SysTick->VAL=0x00;
  18.         //清空计数器                                      
  19.         SysTick->CTRL=0x01 ;
  20.         //开始倒数
  21.         do
  22.         {
  23.                 temp=SysTick->CTRL;
  24.         }while((temp&0x01)&&!(temp&(1<<16)));
  25.         //等待时间到达
  26.         SysTick->CTRL=0x00;
  27.          //关闭计数器                                   
  28.         SysTick->VAL =0X00;
  29.         //清空计数器                                     
  30. }

  31. //同理
  32. void delay_ms(u16 nms)
  33. {                                     
  34.         u32 temp;                  
  35.         SysTick->LOAD=(u32)nms*fac_ms;                       
  36.         SysTick->VAL =0x00;                                  
  37.         SysTick->CTRL=0x01 ;                                 
  38.         do
  39.         {
  40.                 temp=SysTick->CTRL;
  41.         }while((temp&0x01)&&!(temp&(1<<16)));       
  42.         SysTick->CTRL=0x00;                                      
  43.         SysTick->VAL =0X00;                                      
  44. }


 楼主| qn7a12 发表于 2023-11-23 23:43 | 显示全部楼层
HAL库重要函数
1个计数器函数,中断中调用:

__weak void HAL_IncTick(void)
{
  uwTick++;
}
 楼主| qn7a12 发表于 2023-11-23 23:43 | 显示全部楼层
1个获取节拍计数器函数

__weak uint32_t HAL_GetTick(void)
{
  return uwTick;
}
 楼主| qn7a12 发表于 2023-11-23 23:43 | 显示全部楼层
1个延时函数

__weak void HAL_Delay(__IO uint32_t Delay)
{
  uint32_t tickstart = 0;
  tickstart = HAL_GetTick();
  while((HAL_GetTick() - tickstart) < Delay)
  {
  }
}
 楼主| qn7a12 发表于 2023-11-23 23:44 | 显示全部楼层
挂起SysTick定时器函数

__weak void HAL_SuspendTick(void)
{
  /* Disable SysTick Interrupt */
  CLEAR_BIT(SysTick->CTRL,SysTick_CTRL_TICKINT_Msk);
}
 楼主| qn7a12 发表于 2023-11-23 23:44 | 显示全部楼层
恢复SysTick定时器函数

__weak void HAL_ResumeTick(void)
{
  /* Enable SysTick Interrupt */
  SET_BIT(SysTick->CTRL,SysTick_CTRL_TICKINT_Msk);
}
 楼主| qn7a12 发表于 2023-11-23 23:44 | 显示全部楼层
补充
1.进行实验时应将
36018655f735a28f44.png
stm32f1xx.it.c(h)里的注释掉
47028655f736904c91.png
否则会出现多重定义(multiply defined)这个问题
31095655f737a815a8.png
 楼主| qn7a12 发表于 2023-11-23 23:48 | 显示全部楼层
标题
main.c

  1. #include "MyIncludes.h"

  2. u16 sys_cnt = 0;

  3. void systick_isr(void)
  4. {
  5.   if(sys_cnt < 1000 )
  6.   {
  7.     sys_cnt++;//每运行一次时钟,sys_cnt++;
  8.   }
  9.   else
  10.   {
  11.     sys_cnt = 0;
  12.     HAL_GPIO_TogglePin(GPIOC,GPIO_PIN_4|GPIO_PIN_5);
  13.     //超过规定的sys_cnt,归零sys_cnt,并反转PC4 PC5灯
  14.   }
  15. }

  16. int main()
  17. {
  18.    System_Init();
  19.    LED_Init();
  20.    SysTick_Init(systick_isr);//这里运用到了函数指针;
  21.    while(1)
  22.    {
  23.    
  24.    }
  25. }
 楼主| qn7a12 发表于 2023-11-23 23:48 | 显示全部楼层
Systick.h
  1. #ifndef __SYSTICK_H_
  2. #define __SYSTICK_H_

  3. #include "stm32f1xx.h"
  4. #include "stm32_types.h"
  5. #include "stm32f1xx_hal.h"

  6. typedef struct
  7. {
  8.   void(*Operation)(void);//函数指针
  9. }_SysTick_Info;

  10. //extern _SysTick_Info SysTick_Info;
  11. /*可有可无,貌似没用到*/

  12. void SysTick_Init(void (*ISR)(void));
  13. #endif

 楼主| qn7a12 发表于 2023-11-23 23:49 | 显示全部楼层
Systick.c
  1. #include "Systick.h"

  2. _sysTick_Info SysTick_Info;

  3. void SysTick_Handler(void)
  4. {
  5.   HAL_IncTick();
  6.   if(SysTick_Info.Operation != NULL)
  7.   SysTick_Info.Operation();
  8. }
  9. void SysTick_Init(void(*ISR)(void))
  10. {
  11.   SysTick_Info.Operation = ISR;
  12.   SysTick_Config(HALL_RCC_GetHCLKFreq()/1000);
  13. }
 楼主| qn7a12 发表于 2023-11-23 23:49 | 显示全部楼层
 楼主| qn7a12 发表于 2023-11-23 23:49 | 显示全部楼层
代码补充说明
学习这个的时候遇到了一些问题,或许是很简单的问题就是懵在Sys Tick滴答定时器是如何运行的。就是刚开始懵到两个函数指针到底指向哪里,现在我的理解是这样的。
其实很简单: 先在main.c中进行 SysTick_Init(systick_Isr) SysTick滴答定时器的初始化函数,同时看systick.h中的声明
void SysTick_Init(void (*ISR)(void));也就是函数指针ISR指向systick_isr()这个函数。然后就很清晰了,然后看systick.c.
 楼主| qn7a12 发表于 2023-11-23 23:49 | 显示全部楼层
  1. void SysTick_Init(void(*ISR)(void))
  2. {
  3.         SysTick_Info.Operation = ISR;
  4.         /*又用一个函数指针,指向ISR,但最终还是
  5.         systick_isr()这个函数*/
  6.         SysTick_Config(HAL_RCC_GetHCLKFreq()/1000);
  7.         /*这个是设置时间HAL_RCC_GetHCLKFreq()可以得知
  8.         uint32_t SystemCoreClock   = 72000000;
  9.         系统核心时钟
  10.         然后进入这个函数
  11.         __STATIC_INLINE uint32_t SysTick_Config(uint32_t ticks)
  12. {
  13.   if ((ticks - 1UL) > SysTick_LOAD_RELOAD_Msk)
  14.   {
  15.     return (1UL);                                                   
  16.     无法重新加载值 也就是时间过大,
  17.   }

  18.   SysTick->LOAD  = (uint32_t)(ticks - 1UL);                        
  19.   设置重新加载寄存器
  20.   NVIC_SetPriority (SysTick_IRQn, (1UL << __NVIC_PRIO_BITS) - 1UL);
  21.    设置中断的优先级
  22.   SysTick->VAL   = 0UL;                                             
  23.   加载计数器值
  24.   SysTick->CTRL  = SysTick_CTRL_CLKSOURCE_Msk |
  25.                    SysTick_CTRL_TICKINT_Msk   |
  26.                    SysTick_CTRL_ENABLE_Msk;                          
  27.                    启用 SysTick IRQ and SysTick 定时器
  28.   return (0UL);                                                      
  29.   返回0 也就是成功
  30. }*/
  31. }

 楼主| qn7a12 发表于 2023-11-23 23:49 | 显示全部楼层
然后是

  1. void SysTick_Handler(void)
  2. {
  3.         HAL_IncTick();
  4.         /*这个是计数,也就是计时函数
  5.         内部有
  6.         __weak void HAL_IncTick(void)
  7.   {
  8.     uwTick++;
  9.    }这个函数*/
  10.         if(SysTick_Info.Operation !=NULL)
  11.                 SysTick_Info.Operation();
  12.                 /*这个就是判断是否使用了systick定时器,也就是main.c中的
  13.                 systick_isr()函数。
  14. }
您需要登录后才可以回帖 登录 | 注册

本版积分规则

40

主题

542

帖子

1

粉丝
快速回复 在线客服 返回列表 返回顶部