最近调试STM32的串口接收时发现例程中只能接收一个字节
例程如下:
1 //初始化串口1
2 void uart_init(u32 bound){
3 //GPIO端口设置
4 GPIO_InitTypeDef GPIO_InitStructure;
5 USART_InitTypeDef USART_InitStructure;
6 NVIC_InitTypeDef NVIC_InitStructure;
7
8 RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE); //使能USART1,GPIOA时钟
9 USART_DeInit(USART1); //复位串口1
10 //USART1_TX PA.9
11 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9
12 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
13 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出
14 GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化PA9
15
16 //USART1_RX PA.10
17 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
18 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入
19 GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化PA10
20
21 //Usart1 NVIC 配置
22
23 NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
24 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;//抢占优先级3
25 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //子优先级3
26 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
27 NVIC_Init(&NVIC_InitStructure); //根据指定的参数初始化VIC寄存器
28
29 //USART 初始化设置
30
31 USART_InitStructure.USART_BaudRate = bound;//一般设置为9600;
32 USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式
33 USART_InitStructure.USART_StopBits = USART_StopBits_1;//一个停止位
34 USART_InitStructure.USART_Parity = USART_Parity_No;//无奇偶校验位
35 USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件数据流控制
36 USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; //收发模式
37
38 USART_Init(USART1, &USART_InitStructure); //初始化串口
39 USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启中断
40 USART_Cmd(USART1, ENABLE); //使能串口
41
42 }
43 void USART1_IRQHandler(void) //串口1中断服务程序
44 {
45 u8 Res;
46 if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //接收中断(接收到的数据必须是0x0d 0x0a结尾)
47 {
48 //USART_ClearITPendingBit(USART1,USART_IT_RXNE);
49 Res =USART_ReceiveData(USART1);//(USART1-》DR); //读取接收到的数据
50 USART_RX_BUF[buf_index++]=Res;
51
52 }
53
54 }
这是较为普遍的源码例程的写法,用原子的版本改的。这本身没问题,但是,一旦运行,就会发现,只能接收一个字节,后面的都会丢失,调了1天没找到原因,一搜索发现很多类似的情况。
后来正准备改用DMA时偶然找到问题所在,就是这句:
24 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;//抢占优先级3
25 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //子优先级3
改为
24 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1 ;
25 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
后问题解决。
注意,不要在中断中执行发送接收过程,存在中断嵌套的问题,会造成只执行一次的现象。
最近调试STM32的串口接收时发现例程中只能接收一个字节
例程如下:
1 //初始化串口1
2 void uart_init(u32 bound){
3 //GPIO端口设置
4 GPIO_InitTypeDef GPIO_InitStructure;
5 USART_InitTypeDef USART_InitStructure;
6 NVIC_InitTypeDef NVIC_InitStructure;
7
8 RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE); //使能USART1,GPIOA时钟
9 USART_DeInit(USART1); //复位串口1
10 //USART1_TX PA.9
11 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9
12 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
13 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出
14 GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化PA9
15
16 //USART1_RX PA.10
17 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
18 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入
19 GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化PA10
20
21 //Usart1 NVIC 配置
22
23 NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
24 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;//抢占优先级3
25 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //子优先级3
26 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
27 NVIC_Init(&NVIC_InitStructure); //根据指定的参数初始化VIC寄存器
28
29 //USART 初始化设置
30
31 USART_InitStructure.USART_BaudRate = bound;//一般设置为9600;
32 USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式
33 USART_InitStructure.USART_StopBits = USART_StopBits_1;//一个停止位
34 USART_InitStructure.USART_Parity = USART_Parity_No;//无奇偶校验位
35 USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//无硬件数据流控制
36 USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; //收发模式
37
38 USART_Init(USART1, &USART_InitStructure); //初始化串口
39 USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启中断
40 USART_Cmd(USART1, ENABLE); //使能串口
41
42 }
43 void USART1_IRQHandler(void) //串口1中断服务程序
44 {
45 u8 Res;
46 if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //接收中断(接收到的数据必须是0x0d 0x0a结尾)
47 {
48 //USART_ClearITPendingBit(USART1,USART_IT_RXNE);
49 Res =USART_ReceiveData(USART1);//(USART1-》DR); //读取接收到的数据
50 USART_RX_BUF[buf_index++]=Res;
51
52 }
53
54 }
这是较为普遍的源码例程的写法,用原子的版本改的。这本身没问题,但是,一旦运行,就会发现,只能接收一个字节,后面的都会丢失,调了1天没找到原因,一搜索发现很多类似的情况。
后来正准备改用DMA时偶然找到问题所在,就是这句:
24 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;//抢占优先级3
25 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //子优先级3
改为
24 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1 ;
25 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
后问题解决。
注意,不要在中断中执行发送接收过程,存在中断嵌套的问题,会造成只执行一次的现象。
举报