完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
|
|
相关推荐
1个回答
|
|
SysTick简介
SysTick:系统定时器,24位,只能递减,存在于内核,嵌套在NVIC中,所有的Cortex-M内核的单片机都具有这个定时器。 《STM32参考手册》里的一句话:关于Cortex-M3核心、 SysTick定时器和NVIC的详细说明,请参考另一篇ST的文档和一篇ARM的文档:《STM32F10xxx Cortex-M3编程手册》和《Cortex™-M3技术参考手册》。 SysTick框图 重装载寄存器的值和递减计数器的最大值都是2^24。 counter在时钟的驱动下,从reload初值开始往下递减计数到0,产生中断和置位COUNTFLAG标志。然后又从reload值开始重新递减计数,如此循环。 1-t:一个计数循环的时间,跟reload和CLK有关 2-CLK:72M或者9M,由CTRL寄存器配置 3-RELOAD:24位,用户自己配置 t = reload * ( 1/clk ) 1M=1000 000 Clk = 72M时,reload=72,t = (72) *(1/ 72 M )= 1US Clk = 72M时,reload=72000,t = (72000) *(1/ 72 M )= 1MS 1s=1000ms=1000 000us=1000 000 000ns core_cm3.h中的SysTick的结构体 /** @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属于内核里面的外设,他的中断优先级跟片上的外设的中断优先级相比,如果设置的优先级一样,其实systick的级别确实比较高,但是通常还是要看优先级。 systick中断优先级配置的是scb->shprx寄存器;而外设的中断优先级配置的是nvic->iprx,有优先级分组,有抢占优先级和子优先级的说法。 STM32里面无论是内核还是外设都是使用4个二进制位来表示中断优先级。 中断优先级的分组对内核和外设同样适用。当比较的时候,只需要把内核外设的中断优先级的四个位按照外设的中断优先级来分组来解析即可,即人为的分出抢占优先级和子优先级。 SCB_AIRCR的PRIGROUP用来分组 SCB_SHPR 给内核的外设设置优先级 NVIC_IPR 给片上的外设设置优先级 SysTick定时实验程序 sys.h #ifndef __SYS_H #define __SYS_H #include "core_cm3.h" #include "stm32f10x.h" void SysTick_Delay_us(u32 us); void SysTick_Delay_ms(u32 ms); #endif sys.c #include "core_cm3.h" #include "stm32f10x.h" #include "sys.h" void SysTick_Delay_us(u32 us) { u32 i; SysTick_Config(72); for(i=0;i while(!((SysTick->CTRL)&(1<<16)));//等待SysTick_CTRL_COUNTFLAG直到1 //SysTick_CTRL_COUNTFLAG } SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk; } void SysTick_Delay_ms(u32 ms) { u32 i; SysTick_Config(72000); for(i=0;i while(!((SysTick->CTRL)&(1<<16)));//等待SysTick_CTRL_COUNTFLAG直到1 //SysTick_CTRL_COUNTFLAG } SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk; } #if 0 static __INLINE uint32_t SysTick_Config(uint32_t ticks) { //判断输入的值是否大于2^24,如果大于,则不符合规则 if (ticks > SysTick_LOAD_RELOAD_Msk) return (1); /* Reload value impossible */ //初始化reload寄存器的值 SysTick->LOAD = (ticks & SysTick_LOAD_RELOAD_Msk) - 1; /* set reload register */ //配置中断优先级,配置为15,默认为最低的优先级 NVIC_SetPriority (SysTick_IRQn, (1<<__NVIC_PRIO_BITS) - 1); /* set Priority for Cortex-M0 System Interrupts */ //初始化counter的值为0 SysTick->VAL = 0; /* Load the SysTick Counter Value */ //配置systick的时钟为72MHz //使能中断 //使能systick SysTick->CTRL = SysTick_CTRL_CLKSOURCE_Msk | SysTick_CTRL_TICKINT_Msk | SysTick_CTRL_ENABLE_Msk; /* Enable SysTick IRQ and SysTick Timer */ return (0); /* Function successful */ } #endif #include "core_cm3.h" #include "stm32f10x.h" #include "sys.h" void SysTick_Delay_us(u32 us) { u32 i; SysTick_Config(72); for(i=0;i while(!((SysTick->CTRL)&(1<<16)));//等待SysTick_CTRL_COUNTFLAG直到1 //SysTick_CTRL_COUNTFLAG } SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk; } void SysTick_Delay_ms(u32 ms) { u32 i; SysTick_Config(72000); for(i=0;i while(!((SysTick->CTRL)&(1<<16)));//等待SysTick_CTRL_COUNTFLAG直到1 //SysTick_CTRL_COUNTFLAG } SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk; } #if 0 static __INLINE uint32_t SysTick_Config(uint32_t ticks) { //判断输入的值是否大于2^24,如果大于,则不符合规则 if (ticks > SysTick_LOAD_RELOAD_Msk) return (1); /* Reload value impossible */ //初始化reload寄存器的值 SysTick->LOAD = (ticks & SysTick_LOAD_RELOAD_Msk) - 1; /* set reload register */ //配置中断优先级,配置为15,默认为最低的优先级 NVIC_SetPriority (SysTick_IRQn, (1<<__NVIC_PRIO_BITS) - 1); /* set Priority for Cortex-M0 System Interrupts */ //初始化counter的值为0 SysTick->VAL = 0; /* Load the SysTick Counter Value */ //配置systick的时钟为72MHz //使能中断 //使能systick SysTick->CTRL = SysTick_CTRL_CLKSOURCE_Msk | SysTick_CTRL_TICKINT_Msk | SysTick_CTRL_ENABLE_Msk; /* Enable SysTick IRQ and SysTick Timer */ return (0); /* Function successful */ } #endif main.c #include "stm32f10x.h" #include "bsp_led.h" #include "sys.h" #define GPIOB_ODR_Addr (GPIOB_BASE+0x0c) #define PBout(n) *(u32 *)((GPIOB_ODR_Addr&0xF0000000)+0x02000000+((GPIOB_ODR_Addr&0x00FFFFFF)<<5)+(n<<2)) int main(void) { LED_GPIO_Config(); while(1) { LED_0(ON); SysTick_Delay_ms(1000); LED_0(OFF); SysTick_Delay_ms(1000); } } Bug 在移植程序的时候,遇到了这个 error: #20: identifier “IRQn_Type” is undefined 在sys.h文件中 #include “stm32f10x.h” #include “core_cm3.h” 先 #include “stm32f10x.h” 再 #include “core_cm3.h” 原来#include也有顺序要求 |
|
|
|
只有小组成员才能发言,加入小组>>
3284 浏览 9 评论
2960 浏览 16 评论
3464 浏览 1 评论
9008 浏览 16 评论
4054 浏览 18 评论
1124浏览 3评论
579浏览 2评论
const uint16_t Tab[10]={0}; const uint16_t *p; p = Tab;//报错是怎么回事?
572浏览 2评论
用NUC131单片机UART3作为打印口,但printf没有输出东西是什么原因?
2306浏览 2评论
NUC980DK61YC启动随机性出现Err-DDR是为什么?
1863浏览 2评论
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2024-11-28 12:10 , Processed in 1.218753 second(s), Total 80, Slave 61 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191 工商网监 湘ICP备2023018690号