当前位置:首页 > 单片机 > 单片机
[导读]在这节中, 我打算利用SysTick定时器做延时, 让STM32F429 Discovery板子上的两个灯闪烁起来.SysTick包含于Cortex核心中, 在不同厂家的Cortex产品中都存在. 它本质上是一个24位的倒计数器, 在STM32F429中, 它对SYSCLK经

在这节中, 我打算利用SysTick定时器做延时, 让STM32F429 Discovery板子上的两个灯闪烁起来.

SysTick包含于Cortex核心中, 在不同厂家的Cortex产品中都存在. 它本质上是一个24位的倒计数器, 在STM32F429中, 它对SYSCLK经过AHB预分频器分频后的时钟或分频后的时钟的8分频计数(不同的CPU时钟来源可能不会相同, 请参考数据手册中的时钟树), 当倒计数至0时将会产生一个中断(如果使能中断的话), 中断异常号为15. 它的存在是为RTOS提供一个系统节拍, 或者为任务调度产生一个周期性的中断, 可以使得程序在不同厂商的器件之间移植工作得到简化.

SysTick的配置.
SysTick的配置函数位于Core_CM4.h中, 只有一个简单的配置函数:


uint32_tSysTick_config(uint32_tticks);

它属于CMSIS的一部分, 参数ticks为两次中断之间的时钟脉冲数, 即每经过ticks个脉冲, 中断就会发生一次.
当SysTick被成功配置时, 函数返回0, 出错时返回1.

SysTick初始化函数:

void SysTick_Init(void)

{

if (SysTick_Config(SystmeCoreClock / 1000))

{

while (1);

}

SysTick->CTRL &= ~SysTick_CTRL_ENABLE_Msk;

return;

}


以上初始化函数调用SysTick_config()将SysTick定时器初始化为每1ms中断一次.

SystemCoreClock是定义在system_stm32f4xx.c中的一全局变量, 它的值为当前系统时钟频率. 需要说明的是它的默认值为芯片所支持的最大频率, 在使用它之前要调用函数SystemCoreClockUpdate()更新它的值为正确的系统频率, 要不会出现不正确的结果.


关于参数的计算可以这样想, 当参数为SystemCoreClock的时候, 定时是间间隔正好为1秒, 如果我们需要1ms的定时时间, 那么SystemCoreClock / 1000就是了. 因为SysTick计数寄存器为24位, 所以参数值不能大于2的24次方减一, 即16777215, 否则的话初始化失败, 进入死循环.


当SysTick_Config()配置SysTick成功后会立刻启动定时器, 但在这里我希望在用到它的时候启动, 所以在配置完成后使用SysTick->CTRL &= ~SysTick_CTRL_ENABLE_Msk这一句清掉它的使能位, 暂停定时器.


上面说过SysTick时钟有AHB时钟和AHB时钟的8分频两种时钟源可选, 默认为AHB时钟.如果想要使用AHB时钟的8分频可以直接修改SysTick_Config()或者调用函数SysTick_CLKSourceConfig()(位于MISC.h中)进行配置.

在SysTick_Config()中对SysTick的中断优先级也做了修改, 这个暂不深究, 等研究NVIC的时候再说.


中断服务函数:


void SysTick_Handler(void)

{

if (Delay_Time != 0)

{

Delay_Time--;

}

return;

}

不同于51, MSP430等单片机中断函数的写法, STM32的中断函数并不需要使用特殊关键字来声明, 但是函数名是已经被定义好了的. 打开启动文件(这里是startup_stm32f429_439xx.s)就可以看到各中断服务函数的名称, 所以只要按正常函数的写法完成它就可以了.


官方库中有提供两个中断函数模板stm32f4xx_it.h和stm32f4xx_it.c, 将它加入到工程, 然后在对应的地方写中断函数的实现就可以了. 不过由于STM32的中断函数实际上可以写在任何地方, 所以这里我偷了一下懒, 直接将它定义在了main()函数中.


Delay_Time是一个全局变量, SysTick每产生一次中断, 就对它进行一次减一操作.


延时函数:


void Delay_ms(uint16_t ms)

{

Delay_Time = ms;

// 使能SysTick

SysTick->CTRL |= SysTick_CTRL_ENABLE_Msk;

while (Delay_Time != 0);

return;

}

延时函数对Delay_Time赋初值后阻塞CPU, 直到Delay_Time减到0, 完成延时操作.


主函数:


int main(void)

{

RCC_Config();

LED_GPIO_Config();

SysTick_Init();

GPIO_SetBits(GPIOG, GPIO_Pin_13 | GPIO_Pin_14);

while(1)

{

GPIO_ResetBits(GPIOG, GPIO_Pin_13);

GPIO_SetBits(GPIOG, GPIO_Pin_14);

Delay_ms(500);

GPIO_ResetBits(GPIOG, GPIO_Pin_14);

GPIO_SetBits(GPIOG, GPIO_Pin_13);

Delay_ms(500);

}

}

编译下载程序, 板子上红色和绿色的LED就会交替闪烁.


本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

基于ARM® Cortex®-M4的STM32F4 MCU系列采用了意法半导体的NVM工艺和ART Accelerator™,在高达180 MHz的工作频率下通过闪存执行时其处理性能达到225 DMIPS/608 Cor...

关键字: stm32f4 ARM

stm32f4和f1都有哪些区别?除了需要的头文件不一样,管脚配置什么的有区别么?

关键字: stm32f4 f1 比较

STM32F407VET6引脚图及中文资料

关键字: stm32f4 引脚图

比较F1的片子对于F4的RTC来说有很大提升,F4的RTC提供了日历时钟和两个可编程闹钟中断,一个周期性可编程唤醒标志。这样很方便设置系统时间,并不会像F1的RTC那样要自己计算年月日时分秒。F4的RTC是一个独立的BC...

关键字: rtc stm32f4

【实验目的】输出7路占空比不同的PWM信号是各个版本ST库必备的例子。本实验的主要目的不是表现ST芯片PWM功能的强大,而是要完成输出的精确计算。【实验内容】输出7路PWM信号,并用示波器测量输出。【实验原理】1、时

关键字: pwm stm32f4 tim1 信号输出

STM32F4的随机数发生器RNG,以环境噪声为种子,产生32位随机数供主机使用。【主要特性】1、产生32位随机数 2、两次转换时间间隔40PLL48CLK 3、随机数熵检测以发现不正常位,以产生稳定序列 3、可被关闭以...

关键字: stm32f4 随机数发生器

书接上文,开始折腾ADC的DMA传输。因为大家都在说DMA,就连ST的例子里边也是使用DMA的。 ADC采集到的数据都存储在一个固定的寄存器中。当常规采样方式采样多个通道时候,使用DMA可以较好地避免将采集到的数据...

关键字: adc dma stm32f4 数据传输 采集数据

为了提搞系统的可靠性,STM32F4系列MCU有一个独立看门狗(IWDG)和一个窗口看门狗(WWDG)。今天做的实验是关于独立看门狗的。独立看门狗使用的时钟源是内部低速振荡器LSI。因为LSE可能没接,HSE可能坏点,H...

关键字: iwdg stm32f4 独立看门狗

对于串口就不多罗嗦了, 虽然串口是一种比较古老的接口, 速度较慢, 在计算机上早已淘汰, 但是由于它结构简单, 使用方便, 或许也由于在计算机上使用过有基础支撑(单片机就是在走曾经电脑走过的路), 在嵌入式系统中依然

关键字: stm32f4 usart配置

上图是LPC1114系统滴答定时器(SysTick)的结构图。系统滴答定时器位于Cortex-M0内核中,也就是说,不论是LPC1114,还是其他的Cortex-M0内核单片机,都有这个系统定时器。其存在的主要目的是为嵌...

关键字: systick 滴答定时器 cortex-m0系统
关闭