最近在做ZigBee+GPRS与STM32网关通信实验,用的是战舰V3开发板。要实现的功能是:串口1与GPRS串口通信(GPRS暂时用PC模拟收发数据),接收PC端的命令并通过串口3发送给ZigBee协调器;串口3与ZigBee串口通信,接收ZigBee汇集的数据包并通过串口1发送给PC(GPRS)。 两个串口的中断函数都是一样的,串口1的中断函数如下:
void USART1_IRQHandler(void)
{
u8 res;
OSIntEnter(); //进入OS中断
if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)//接收到数据
{
res =USART_ReceiveData(USART1);
if((USART1_RX_STA&(1<<15))==0)//接收完的一批数据,还没有被处理,则不再接收其他数据
{
if(USART1_RX_STA<USART1_REC_LEN) //还可以接收数据
{
TIM_SetCounter(TIM3,0);//计数器清空
if(USART1_RX_STA==0) //使能定时器3的中断
{
TIM_Cmd(TIM3,ENABLE);//使能定时器3
}
USART1_RX_BUF[USART1_RX_STA++]=res; //记录接收到的值
}else
{
USART1_RX_STA|=1<<15; //强制标记接收完成
}
}
USART_ClearITPendingBit(USART1, USART_IT_RXNE);//清除接收中断
}
OSIntExit(); //退出OS中断
}
考虑到任务的实时性,在主程序中加入了UCOS-III操作系统,按照任务优先级先后顺序写了TOUCH任务、GPRS串口通信任务、触摸屏显示任务与LED闪烁任务。GPRS串口通信任务在任务死循环内判断USART1是否接收到PC发送的数据,然后将数据通过串口3发送出去;触摸屏显示任务也同样在任务死循环内判断USART3是否接收到ZigBee协调器发来的数据,然后将数据包进行解析、显示。
GPRS串口通信任务如下:
void GPRS_USART1_task(void *pdata)
{
u8 i,rxlen;
OS_ERR err;
while(1)
{
if(USART1_RX_STA&0x8000) //USART1接收到GPRS发送的数据
{
rxlen=USART1_RX_STA&0X7FFF; //得到数据长度
for(i=0;i<rxlen;i++) USART3_TX_BUF[i]=USART1_RX_BUF[i];
USART1_RX_STA=0; //启动下一次接收
USART3_TX_BUF[i]=0; //自动添加结束符
printf("%s",USART3_TX_BUF);//发送接收到的数据通过串口3发送给ZigBee协调器
}
OSTimeDlyHMSM(0,0,0,20,OS_OPT_TIME_PERIODIC,&err);//延时20ms
}
}
触摸屏显示任务部分代码如下:
void emwindemo_task(void *p_arg){
while(1)
{
if((USART3_RX_STA&0x8000) && (USART3_RX_BUF[0]=='&')) //接收到从协调器传送的一帧数据
{
//。。。。。数据解析处理、显示(祥见附件)
USART3_RX_STA=0;
}
GUI_Exec();
}
}
但下载后的现象是:
1)用串口助手先给串口1发数据能收到,再给串口3发却始终不能收到且界面也无数据显示,而且LED始终不闪烁;
2)用串口助手先给串口3发数据能收到且界面数据显示正常,但每发一下LED就闪一下,再给串口1发这时也能收到,但再换给串口3发却又不能收到;
我对第1个现象的理解就是:由于串口1的任务优先级最高,一旦串口1发生接收中断后始终执行串口1任务(我把它解释为任务死锁),从而导致UCOS无法发生任务调度,从而串口3和LED任务都不能运行;第2个现象,串口1的任务(优先级高)可以打断串口3(优先级低)的任务,但为什么优先级最低的LED任务只执行一次没明白。
求教原子哥和大神们,如何解决串口通信任务死锁的问题,保证UCOS各任务正常调度?希望给与小弟指点,谢谢! |