【第21期】观点:人工智能到底用 GPU?还是用 FPGA?

SysTick定时器和delay延迟函数

原创 2016年08月30日 16:10:25

一,什么是Systick定时器

  Systick定时器也叫滴答定时器,是内核级别的24位倒计数简单定时器,常用做延迟和系统心跳时钟(如:UCOS)
  优点:节省MCU资源,不需要浪费一个定时器,只要不清除Systick使能位,就不会停止,即使在睡眠模式下也能工作
  捆绑在NVIC中断优先级管理,能产生Systick异常(中断),可设置中断优先级


二,Systick相关寄存器

  CTRL: Systick控制和状态寄存器
  LOAD: Systick重装载寄存器
  VAL: Systick当前值寄存器
  CALIB: Systick校准值寄存器
  定时器工作方式:
  每经过一个Systick时钟周期,VAL寄存器值-1,当VAL=0,LOAD寄存器中的重装载值赋值给VAL寄存器作为初值….


 1,CTRL-控制和状态寄存器

  这里写图片描述
  CLKCOURCE-时钟源
   0:外部时钟源HCLK(AHB总线时钟)/8 72M/8 = 9M
   1:内核时钟(HCLK) 72M
  配置函数:

    SysTick_CLKSourceConfig();

 2,LOAD-Systick重装载寄存器

  这里写图片描述


 3,VAL-Systick当前值寄存器

  这里写图片描述
  每经过一个Systick时钟周期,VAL寄存器值-1
  读取寄存器:返回当前VAL值
  写寄存器:清零VAL值,还会使CTRL中COUNTFLAG位清零


 4,CALIB-Systick校准值寄存器

  这里写图片描述


三,SysTick函数

 固件库SysTick相关函数

SysTick_CLKSourceConfig()          //Systick选择时钟源(FWLIB-misc.c文件中)
SysTick_Config(uint32_t ticks)     //初始化Systick(CORE-core_cm3.h文件中)

 Systick中断服务函数:

void SysTick_Handler(void);

1,SysTick_CLKSourceConfig()分析:

在FWLIB-misc.c中找到SysTick_CLKSourceConfig()函数源码:

/**
  * @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;          //内部时钟72M
  }
  else
  {
    SysTick->CTRL &= SysTick_CLKSource_HCLK_Div8;     //外部时钟 72/8=9M
  }
}

 功能:配置SysTick->CTRL寄存器


 在core_cm3.h中找到SysTick结构体定义:

#define SysTick_BASE  (SCS_BASE +  0x0010)    /* SysTick Base Address */

#define SysTick   ((SysTick_Type *) SysTick_BASE) /* SysTick configuration struct */

/** @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_CLKSourceConfig参数的两种情况:

/** @defgroup SysTick_clock_source
  * @{
  */

#define SysTick_CLKSource_HCLK_Div8    ((uint32_t)0xFFFFFFFB)
#define SysTick_CLKSource_HCLK         ((uint32_t)0x00000004)
#define IS_SYSTICK_CLK_SOURCE(SOURCE) (((SOURCE) == SysTick_CLKSource_HCLK) || \
                                       ((SOURCE) == SysTick_CLKSource_HCLK_Div8))

两种时钟源 :
  SysTick_CLKSource_HCLK_Div8 外部时钟 72/8=9M
  SysTick_CLKSource_HCLK 内部时钟 HCLK=72M


2,SysTick_Config(uint32_t ticks)分析

core_cm3.h中找到SysTick_Config函数源码:

/**
 * @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);       //ticks参数有效性检查

  SysTick->LOAD  = (ticks & SysTick_LOAD_RELOAD_Msk) - 1; //设置重装载值
                                                    //-1:装载时消耗掉一个Systick时钟周期

  NVIC_SetPriority (SysTick_IRQn, (1<<__NVIC_PRIO_BITS) - 1); //配置NVIC

  SysTick->VAL   = 0;    //初始化VAL=0,使能Systick后立刻进入重装载
  SysTick->CTRL  = SysTick_CTRL_CLKSOURCE_Msk |      //选择时钟源
                   SysTick_CTRL_TICKINT_Msk   |      //开启Systick中断
                   SysTick_CTRL_ENABLE_Msk;          //使能Systick定时器
  return (0);      /* Function successful */
}

#endif

作用:使能Systick定时器,开启SysTick中断,配置中断时间间隔
参数ticks:设置多少个Systick时钟周期产生一次中断


四,SysTick实现延时函数

1,延时函数初始化

static u8  fac_us=0;           //延时微秒的频率
static u16 fac_ms=0;           //延时毫秒的频率

void delay_init()
{
    SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8); //选择时钟源-外部时钟-HCLK/8
    fac_us=SystemCoreClock/8000000; // 72/8 延时1微秒9个时钟周期
    fac_ms=(u16)fac_us*1000;   // 延时1毫秒9000个Cystic时钟周期
}

2,微秒延时函数

/**
 * nus : 延时多少微秒
 **/
void delay_us(u32 nus)
{
    u32 temp;
    //nus*fac_us值最大不能超过SysTick->LOAD(24位)-1
    SysTick->LOAD=nus*fac_us;    // 设置重载值:n(us)*延时1us需要多少个SysTick时钟周期
    SysTick->VAL=0x00;                       // VAL初始化为0
    SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk ; // 使能SysTick定时器
    do
    {
        temp=SysTick->CTRL;
    }while((temp&0x01)&&!(temp&(1<<16)));    // 等待计数时间到达(位16)
    SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk; // 关闭使能
    SysTick->VAL =0X00;                      // 重置VAL
}

3,毫秒延时函数

/**
 * nms : 延时多少毫秒
 **/
void delay_ms(u16 nms)
{
    u32 temp;
    SysTick->LOAD=(u32)nms*fac_ms;
    SysTick->VAL =0x00;
    SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk ;
    do
    {
        temp=SysTick->CTRL;
    }while((temp&0x01)&&!(temp&(1<<16)));
    SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk;
    SysTick->VAL =0X00;
}
版权声明:本文为博主原创文章,未经博主允许不得转载。 举报

相关文章推荐

STM32 SysTick定时器常见问题及延时函数的实现

Q:什么是SYSTick定时器? SysTick 是一个24 位的倒计数定时器,当计到0 时,将从RELOAD 寄存器中自动重装载定时初值。只要不把它在SysTick 控制及状态寄存器中的使能位清除...

STM32滴答定时器(Systick)详细详细解析

在ARM Cortex-M3内核中有一个Systick定时器,它是一个24位的倒计数定时器,当计数到0时,它就会从Load寄存器中自动重装定时初值,只要不把CTRL寄存器中的ENABLE清0,它就永不...

STM32菜鸟成长记录---系统滴答定时器(systick)应用

1.systick介绍      Systick就是一个定时器而已,只是它放在了NVIC中,主要的目的是为了给操作系统提供一个硬件上的中断(号称滴答中断)。滴答中断?这里来简单地解释一下。操作系统进...

STM32之SysTick

以用Cortex-M3中关于SysTick的描述:      SysTick定时器被捆绑在NVIC中,用于产生SysTick异常(异常号:15)。在以前,操作系统还有所有使用了时基的系统,都必须一个硬...

18B20温度传感器详解

数据传感器DS18B20   特性: 1.      单独的单线接口,只需1个接口引脚即可通信; 2.      不需要外部元件; 3.      可用数据线供电; 4.      不需备份电源; 5...

AlarmManager研究

http://my.oschina.net/youranhongcha/blog/149564 目录[-] 1.概述 2.AlarmManager 2.1 ...

C语言的基本文件操作

C语言的基本文件操作序列1:基本的txt文件读入读出先来段代码看看#include int main() { char a[] = "hellow\n"; char buf[6]; ...

MBProgressHUD的简单封装

下面是依据MBProgressHUD提供的分类方法: /** * 显示一般信息 */ + (void)show:(NSString *)text icon:(NSString *)icon vi...

《深入理解java虚拟机》学习笔记9——并发编程(一)

随着多核CPU的高速发展,为了充分利用硬件的计算资源,操作系统的并发多任务功能正变得越来越重要,但是CPU在进行计算时,还需要从内存读取输出,并将计算结果存放到内存中,然而由于CPU的运算速度比内存高...

think in java interview-高级开发人员面试宝典(二)

从现在开始,以样题的方式一一例出各种面试题以及点评,考虑到我在前文中说的,对于一些大型的外资型公司,你将会面临全程英语面试,因此我在文章中也会出现许多全英语样题。这些题目来自于各个真实的公司,公司名我...
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)