本文将介绍STM32的基础时钟,通过时钟中断来控制LED灯的亮灭。
本文的大致思路如下:
1,LED的GPIO口初始化
2,中断初始化
3,时钟初始化
4,编写中断函数
5,编写主函数
首先讲下笔者在做这个设计时得到的教训:
笔者最初是用TIM6基础时钟来实现设计功能,在编写完代码后发现,灯不亮,于是笔者就开始整问题了。先看看代码有没有编写错误,检查一遍后,中断通道使用正确,TIM6配置正确,LED灯的串口也没有问题,中断函数也正常编写了。然后笔者纠结了,他妈代码全对了怎么就运行失败?(原谅笔者爆粗口,因为当时心情的确很不好,可以想象一下,辛辛苦苦桥的代码,没什么毛病,结果到了板子上还运行不了)之后笔者认认真真核对代码,还是没解决问题,之后没办法了,去论坛上请教大佬去了,(还是大佬厉害)我把问题说了后,大佬看了后说,可能是你的启动文件里没有TIM6中断(笔者顿悟了),他妈我怎么没想到这个。。。之后去检查发现果然没有这个中断函数入口(STM32中必须按照要求写中断函数名才能进入中断),之后笔者发现我的工程中定义的是STM32F10X_MD这个头文件,这个头文件里没有TIM6这个中断名(当初笔者也发现了这个问题,自己看定义了个#define 54(因为我查了其他头文件是54)),又检查了启动文件,发现没有写中断函数入口,笔者就把TIM6换成TIM3(通用定时器)了,之后就正常了。希望可以帮助到和我犯同样错误的人。
下面进入主题,详细解释代码:
1,LED的GPIO口初始化
void led_init()
{
GPIO_InitTypeDef led_gpio;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC| RCC_APB2Periph_AFIO,ENABLE);
GPIO_PinRemapConfig(GPIO_Remap_SWJ_Disable,ENABLE);因为GPIOC13口的13引脚被重映射了,所以需要先关闭重映射
GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable,ENABLE);
led_gpio.GPIO_Pin = GPIO_Pin_13;
led_gpio.GPIO_Mode = GPIO_Mode_Out_PP;
led_gpio.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOC,&led_gpio);
}
2,中断初始化
void nvic_init()
{
NVIC_InitTypeDef nvic;
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);定义分组
nvic.NVIC_IRQChannel =TIM3_IRQn;中断通道名
nvic.NVIC_IRQChannelPreemptionPriority = 1;
nvic.NVIC_IRQChannelSubPriority = 0;
nvic.NVIC_IRQChannelCmd= ENABLE;中断时能
NVIC_Init(&nvic);初始化寄存器
}
3,时钟初始化
void tim_init()
{
TIM_TimeBaseInitTypeDef tim;定义时钟结构体
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);时能TIM3
tim.TIM_Period = 1000-1;计数周期
tim.TIM_Prescaler = 72 - 1;预分频数
tim.TIM_ClockDivision = TIM_CKD_DIV1;时钟分频,设置定时器时钟CK_INT频率与数字滤波器采样时钟频率分频比,基本定时器没有此功能
tim.TIM_CounterMode = TIM_CounterMode_Up;定时器基础方式,可以是向上计数、向下计数、中心对其模式。基本定时器只能是向上计数。
tim.TIM_RepetitionCounter = 0;重复计数器,属于高级控制寄存器专用寄存器位,这里不用设置。
TIM_TimeBaseInit(TIM3,&tim);初始化
TIM_ClearFlag(TIM3,TIM_FLAG_Update);清除中断标志
TIM_ITConfig(TIM3,TIM_IT_Update,ENABLE);开启中断
TIM_Cmd(TIM3,ENABLE);TIM3使能
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,DISABLE);关闭TIM3时钟,等需要时在打开
}
4,编写中断服务函数
void TIM3_IRQHandler()
{
if(TIM_GetITStatus(TIM3,TIM_IT_Update) != RESET){判断是否可以中断
//PrintString("\r\ntime inter");这个是笔者在调试程序时用的串口
time++;
if(time >=1000)实现1秒一计数
{
//PrintString("\r\nin time inter");
time = 0;以下是实现灯的亮灭
if(GPIOC->ODR & GPIO_Pin_13)
{
GPIOC->BRR = GPIO_Pin_13;
}else
{
GPIOC->BSRR = GPIO_Pin_13;
}
}
}
TIM_ClearITPendingBit(TIM3,TIM_FLAG_Update);清除中断标志位避免重复进入中断
}
5,主函数
int main()
{
led_init();LED初始化
tim_init();时钟初始化
nvic_init();中断初始化
//nvic_usart_init();
//usart_init();
GPIOC->BSRR = GPIO_Pin_13;预先打开灯
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);使能TIM3
delay(1000);延时1秒
while(1){死循环
//PrintString("\r\nÑ»·");
}
}
下面粘贴详细代码:
注:注释掉的部分是笔者用串口在调试程序,可忽略(笔者经验有限,如有不妥之处请指教)
#include<stm32f10x.h>
static unsigned int time;
uint8_t TxCount=0;
uint8_t Count=0;
static uint8_t TxBuff[256];
void delay(uint16_t n)
{
int i,j;
for(i=0;i<n;i++)
for(j=0;j<8500;j++);
}
void tim_init()
{
TIM_TimeBaseInitTypeDef tim;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);
tim.TIM_Period = 1000-1;
tim.TIM_Prescaler = 71;
tim.TIM_ClockDivision = TIM_CKD_DIV1;
tim.TIM_CounterMode = TIM_CounterMode_Up;
tim.TIM_RepetitionCounter = 0;
TIM_TimeBaseInit(TIM3,&tim);
TIM_ClearFlag(TIM3,TIM_FLAG_Update);
TIM_ITConfig(TIM3,TIM_IT_Update,ENABLE);
TIM_Cmd(TIM3,ENABLE);
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,DISABLE);
}
void PrintU8(uint8_t data)
{
TxBuff[TxCount++]=data;
if(!(USART1->CR1 & USART_CR1_TXEIE))
{
USART_ITConfig(USART1,USART_IT_TXE,ENABLE);
}
}
void PrintString(uint8_t *s)
{
uint8_t *p;
p=s;
while(*p != '\0')
{
PrintU8(*p);
p++;
}
}
void usart_init()
{
GPIO_InitTypeDef Uart_A;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
Uart_A.GPIO_Pin = GPIO_Pin_9;
Uart_A.GPIO_Speed = GPIO_Speed_50MHz;
Uart_A.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_Init(GPIOA,&Uart_A);
Uart_A.GPIO_Pin = GPIO_Pin_10;
Uart_A.GPIO_Speed = GPIO_Speed_50MHz;
Uart_A.GPIO_Mode = GPIO_Mode_IN_FLOATING; //page 110
GPIO_Init(GPIOA,&Uart_A);
USART_InitTypeDef Uart;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);
Uart.USART_BaudRate = 115200;
Uart.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
Uart.USART_Mode = USART_Mode_Tx;
Uart.USART_Parity = USART_Parity_No;
Uart.USART_StopBits = USART_StopBits_1;
Uart.USART_WordLength = USART_WordLength_8b;
USART_Init(USART1,&Uart);
USART_Cmd(USART1,ENABLE);
USART_ClearFlag(USART1,USART_FLAG_TC);
}
void nvic_usart_init()
{
NVIC_InitTypeDef nvic;
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
nvic.NVIC_IRQChannelPreemptionPriority = 0;
nvic.NVIC_IRQChannelSubPriority = 0;
nvic.NVIC_IRQChannel = USART1_IRQn;
nvic.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&nvic);
}
void nvic_init()
{
NVIC_InitTypeDef nvic;
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
nvic.NVIC_IRQChannel =TIM3_IRQn;
nvic.NVIC_IRQChannelPreemptionPriority = 1;
nvic.NVIC_IRQChannelSubPriority = 0;
nvic.NVIC_IRQChannelCmd= ENABLE;
NVIC_Init(&nvic);
}
void led_init()
{
GPIO_InitTypeDef led_gpio;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC| RCC_APB2Periph_AFIO,ENABLE);
GPIO_PinRemapConfig(GPIO_Remap_SWJ_Disable,ENABLE);
GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable,ENABLE);
led_gpio.GPIO_Pin = GPIO_Pin_13;
led_gpio.GPIO_Mode = GPIO_Mode_Out_PP;
led_gpio.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOC,&led_gpio);
}
int main()
{
led_init();
tim_init();
nvic_init();
//nvic_usart_init();
//usart_init();
GPIOC->BSRR = GPIO_Pin_13;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);
delay(1000);
while(1){
//PrintString("\r\nÑ»·");
}
}
void TIM3_IRQHandler()
{
if(TIM_GetITStatus(TIM3,TIM_IT_Update) != RESET){
//PrintString("\r\ntime inter");
time++;
if(time >=1000)
{
//PrintString("\r\nin time inter");
time = 0;
if(GPIOC->ODR & GPIO_Pin_13)
{
GPIOC->BRR = GPIO_Pin_13;
}else
{
GPIOC->BSRR = GPIO_Pin_13;
}
}
}
TIM_ClearITPendingBit(TIM3,TIM_FLAG_Update);
}
void USART1_IRQHandler(void)
{
if(USART1->SR & USART_SR_TC)
{
USART1->DR = TxBuff[Count++];
if(TxCount == Count)
{
USART1->CR1 &= ~USART_CR1_TXEIE;
}
}
}