今天在写串口调试的时候发现,开启串口的接收中断,写了测试语句,但是发现串口的接收中断只会触发一次,然后进行了各种尝试,最后一点点对比了官方的库函数,发现,官方的历程在调用的时候多了一行关于串口中断的代码,后来经过测试,将问题锁定在这个位置。这个和STM32的串口中断还是不一样的,STM32的串口中断函数只需要去启动文件中找到对应的中断函数,然后吧函数的中断名复制过来就可以了,但是沁恒的中断函数只会触发一次。必须加上 :void USART1_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast"))); 这句代码才能触发,有没有大佬给我解释一下,为啥要加这行。下面贴上完整的测试代码。
#include "debug.h"
void USART1_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));
void func ()
{
/*开启时钟*/
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE); //开启USART1的时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //开启GPIOA的时钟
/*GPIO初始化*/
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure); //将PA9引脚初始化为复用推挽输出
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure); //将PA10引脚初始化为上拉输入
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure); //将PA10引脚初始化为上拉输入
GPIO_SetBits(GPIOA, GPIO_Pin_0);
/*USART初始化*/
USART_InitTypeDef USART_InitStructure; //定义结构体变量
USART_InitStructure.USART_BaudRate = 9600; //波特率
USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; //硬件流控制,不需要
USART_InitStructure.USART_Mode = USART_Mode_Tx | USART_Mode_Rx; //模式,发送模式和接收模式均选择
USART_InitStructure.USART_Parity = USART_Parity_No; //奇偶校验,不需要
USART_InitStructure.USART_StopBits = USART_StopBits_1; //停止位,选择1位
USART_InitStructure.USART_WordLength = USART_WordLength_8b; //字长,选择8位
USART_Init(USART1, &USART_InitStructure); //将结构体变量交给USART_Init,配置USART1
/*中断输出配置*/
USART_ITConfig(USART1, USART_IT_RXNE, ENABLE); //开启串口接收数据的中断
/*NVIC中断分组*/
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //配置NVIC为分组2
/*NVIC配置*/
NVIC_InitTypeDef NVIC_InitStructure; //定义结构体变量
NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn; //选择配置NVIC的USART1线
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //指定NVIC线路使能
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; //指定NVIC线路的抢占优先级为1
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; //指定NVIC线路的响应优先级为1
NVIC_Init(&NVIC_InitStructure); //将结构体变量交给NVIC_Init,配置NVIC外设
/*USART使能*/
USART_Cmd(USART1, ENABLE); //使能USART1,串口开始运行
}
int main(void)
{
Delay_Init();
func();
while(1)
{
}
}
void USART1_IRQHandler(void)
{
if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //判断是否是串口1触发的中断
{
GPIO_ResetBits(GPIOA,GPIO_Pin_0); //LED闪烁测试
Delay_Ms(100);
GPIO_SetBits(GPIOA,GPIO_Pin_0);
Delay_Ms(100);
}
USART_ClearITPendingBit(USART1, USART_IT_RXNE); //清除串口中断的标志位,不然中断会一直触发
}
7.18问题找到了。沁恒微使用的内核跟Arm 的还是有所区别的,需要在中断的声明处,做一个定义,不然,青稞的内核就会将中断的函数当做普通的函数来执行,所以在定义中加上一个定义就好了。
void XXXXXXXX(void) __attribute__((interrupt("WCH-Interrupt-fast")));