欢迎指正
对原子哥串口接收中断服务函数void USART1_IRQHandler(void)发现2个问题:
1.实际接收的字节个数不等于USART_REC_LEN,而是USART_REC_LEN-1;
2.在接收的字节中不能存在0D,否则会删去0D,0D以前的所有字节,0D后一个相邻的字节,重新开始接收;
第一个问题的解决很容易,在void USART1_IRQHandler(void)中,将判断语句if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0中的
(USART_REC_LEN-1)改为USART_REC_LEN就行;
第二个问题的解决参考了
http://openedv.com/forum.php?mod ... t=%B4%AE%BF%DA%2B0d
得到一定的改善,但还发现了一个问题:试试以 0D 0D 0A结尾就明白了,读程序会发现对第二个0D是没有进行使能标记的,也就是说以任意
偶数个OD+0A结尾都会出现这个问题,而奇数个0D+0A结尾刚好避开了问题,
基于此对void USART1_IRQHandler(void)代码进行改善,如下:
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) //接收到0x0a,完成接收
{
USART_RX_STA--;
USART_RX_STA|=0x8000; //标记完成
}
else if(Res==0x0d) //连续接收到0x0d
{
USART_RX_STA|=0x4000; //重新使能0d标记位
USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
USART_RX_STA++; //防止下一次收到的数据不是0a
}else
{
USART_RX_STA&=0x3FFF; //清除0d标志位,继续接收数据
USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
USART_RX_STA++;
if(USART_RX_STA>USART_REC_LEN)USART_RX_STA=0; //接收数据错误,重新开始接收
}
}
else //一旦接收到第一个0d后,程序不再进入
{
if(Res==0x0d)
{
USART_RX_STA|=0x4000; //使能0d标记位
USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
USART_RX_STA++; //防止下一次收到的数据不是0a
}
else
{
USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
USART_RX_STA++;
if(USART_RX_STA>USART_REC_LEN)USART_RX_STA=0; //接收数据错误,重新开始接收
}
}
}
}
}
一周热门 更多>