首先把程序放上来,最后结论总结在最后
u8 USART_RX_BUF[USART_REC_LEN];
u16 USART_RX_STA=0;
void USART1_IRQHandler(void) //串口1中断服务程序
{
u8 Res;
if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //接收中断(接收到的数据必须是0x0d 0x0a结尾)
{
Res =USART_ReceiveData(USART1); //读取接收到的数据
if((USART_RX_STA&0x8000)==0)//接收未完成
{
if(USART_RX_STA&0x4000)//接收到了0x0d
{
if(Res!=0x0a)USART_RX_STA=0;//接收错误,重新开始
else USART_RX_STA|=0x8000; //接收完成了
}
else //还没收到0X0D
{
if(Res==0x0d)USART_RX_STA|=0x4000;
else
{
USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
USART_RX_STA++;
if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;//接收数据错误,重新开始接收
}
}
}
}
}
这是我们最开始见的usart接受中断
我们先假定串口接受到了一个字符串“ABC”
那么实际串口是先A,B,C,oxod,oxoa,即5个数据帧,(不明白0x0d和0x0a哪里来的,建议去看正点哥的串口实验讲解)
然后程序他是使能了串口接受中断USART_IT_RXNE
所以中断就是当串口的状态寄存器的RXNE位:读数据寄存器非空时(1),就触发了中断
//对应中断中的这一段if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)
这里我们也要注意串口是每次接受一个字节,那就如我们例子的A
此时对应程序
if((USART_RX_STA&0x8000)==0) //满足条件,往下
if(USART_RX_STA&0x4000) //不满足条件,去到他的else
if(Res==0x0d) //因为是A(变成ASCII码就是0x31),所以不满足,去到else
USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;//USART_RX_STA==0,也就USART_RX_STA&0X3FFF==0,也就是USART_RX_BUF[0]=Res=A(还是他的ASCII码,这里没搞清楚,不好意思)
USART_RX_STA++;
if(USART_RX_STA>(USART_REC_LEN-1))//正常状态我们这个也不满足
此时同理B,C
这是就到了0x0d
if((USART_RX_STA&0x8000)==0) //满足条件,往下
if(USART_RX_STA&0x4000) //不满足条件,去到他的else
if(Res==0x0d) //满足条件,也就吧USART_RX_STA的第14位置1
接着到0x0a
if((USART_RX_STA&0x8000)==0) //满足条件,往下
if(USART_RX_STA&0x4000) //满足条件,往下
if(Res!=0x0a) //不满足条件,去else,把USART_RX_STA的第15位置1
那么整个接受这个“ABC”的字符串就此接受完成,
A放在了BUF[0],
B放在了BUF[1],
C放在了BUF[2],
0x0d和0x0a主要是把STA的第15.14位置一
STA的13-0位主要是记录这个串口历史以来接受到的字节总共有多少个