状态机的一个极度确切的描述是它是一个有向图形,由一组节点和一组相应的转移函数组成。状态机通过响应一系列事件而“运行”。每个事件都在属于“当前” 节点的转移函数的控制范围内,其中函数的范围是节点的一个子集。函数返回“下一个”(也许是同一个)节点。这些节点中至少有一个必须是终态。当到达终态, 状态机停止。 本文描述了利用状态机来提高嵌入式前后台系统任务处理能力的实现方法。
1 移动2G光纤直放站近端机监控单元
对于移动2G光纤直放站近端机监控单元,只介绍与本文有关部分的原理框图,如图1所示。图中GSM Modem通过AT91SAM7S256的串口1相连。由于GSM Modem的特性和短消息的收发,其通信收发处理相对比较繁琐和复杂。
图1 直放站近端机监控单元
在移动2G光纤直放站近端机中,通过串口1发送到GSM Modem的数据不仅仅是短消息,还包括下行功率查询、信源信息查询、读取/删除短消息等。因此,针对移动2G光纤直放站近端机监控单元的要求和软件系统为前后台系统的特点,移动2G光纤直放站近端机监控单元的监控软件设计采用了状态机和队列的方式。
2 软件的设计思路
2.1 短消息的接收
如图2所示,GSM Modem主动上报的信息将存储到串口1接收缓冲区中,软件从串口1接收缓冲区的数据中解析出短信序号(Modem收到的短消息在Modem中的存储序号)存储到短信序号队列(短信序号缓冲区1~n)中,然后软件通过短信序号队列的状态来决定是否需要向Modem发送读取短信或者删除短信命令。
当软件发送读短消息命令后,GSM Modem将对应序号的短信息送出,数据将存储到串口1接收缓冲区中,软件再从串口1接收缓冲区的数据中解析出短消存储到短信队列(短信缓冲区1~m)中。这样需要软件处理的短消息就存储到了短信队列中,而处理的事情则交由软件的其他区处理。
图2 直放站短信收发处理流程
2.2 短消息的发送
对所有需要发送到GSM Modem的数据,则通过UART1发送缓冲区来完成。具体发送哪些数据(读取/删除短信、下行功率查询、信源信息查询、未读短信查询)或者缓冲区的数据(短消息发送缓冲区、告警上报发送缓冲区)由软件根据相应的状态来选择确定。
3 设计思路的实现
3.1 串口1数据的发送
3.1.1 串口1发送缓冲区的数据结构
串口1是否有数据需要发送,由串口1的发送缓冲区的状态来决定。串口1发送缓冲区的数据结构定义如下:
typedef struct{
unsigned char bStBuf;//bStBuf =
Uart1_TxBuf_Rdy或者=Uart1_TxBuf_Wait或者=Uart1_TxBuf_Empty
unsigned short Index;
unsigned short Len;
char Buf[270];
}Uart1Buf_t;
① bStBuf成员: 串口1发送缓冲区的状态。
② Index成员: 串口1发送数据缓冲区索引。
③ Len成员: 串口1发送数据缓冲区中有效数据的长度。
④ Buf成员: 串口1发送数据缓冲区。
3.1.2 串口1发送缓冲区软件定时器
由于GSM Modem的特性致使串口1不能不间断地发送数据,因此,对串口1的数据发送设定一个软件定时器。软件定时器的结构定义如下:
typedef struct{
unsigned char bTimerSt;//软件定时器的状态: Timer_START或Timer_STOP
unsigned int TimerCtn;//软件定时器的计数器
void (*func)(void);//超时后相应的处理功能函数指针
}SoftTimer_t;
① bTimerSt成员: 用于描述软件定时器的状态。它有2种状态:
◆ Timer_START——开始软件定时器;
◆ Timer_STOP——停止软件定时器。
② TimerCtn成员: 用于描述软件定时器的定时时间。它是一个32位的计数器,硬件定时的基准时间为20ms(建议设置在前后台系统主程序循环1次需要的时间),因此定时时间为20 ms×232=85 899 345.92 s。
③ func成员: 用于描述软件定时器超时需要去处理相应事情的函数。该函数是在定时器中断服务程序下运行的,因此为了减少中断服务程序占用CPU的时间,函数只作简单的状态设置或者清除工作,如函数Clear_Uart1TxbStBuf。
void Clear_Uart1TxbStBuf(void){
Uart1Tx.bStBuf = Uart1_TxBuf_Empty;//设置串口1发送缓冲区为空
……
}
3.1.3 串口1数据发送状态机
串口1发送缓冲区的成员bStBuf有3种状态。
① Uart1_TxBuf_Rdy: 串口1发送缓冲区数据准备好。
② Uart1_TxBuf_Wait: 串口1发送缓冲区数据等待。
③ Uart1_TxBuf_Empty: 串口1发送缓冲区空。
3种状态的转移情况如图3所示。
图3 串口1的发送缓冲区状态机
当串口1发送缓冲区在Uart1_TxBuf_Rdy状态下时,软件可以向串口的发送缓冲区中写入数据。写入数据后,串口1发送缓冲区的状态将转移到Uart1_TxBuf_Rdy。
当串口1发送缓冲区的状态在Uart1_TxBuf_Wait状态时,它可以有两条路径让串口1发送缓冲区的状态转移到Uart1_TxBuf_Empty。
3.2 短信数据的发送
需要通过串口1发送的数据包括:
读取/删除短信数据、下行功率查询数据、信源信息查询数据、未读短信查询数据、短消息发送缓冲区数据、告警上报发送缓冲区数据。其中,读取/删除短信数据、下行功率查询数据、信源信息查询数据和未读短信查询数据,直接由GSM Modem处理,并作出处理结果应答。因此,这类数据直接通过串口1发送缓冲区发送。
3.2.1 短信数据结构
短信数据包括短消息发送缓冲区数据和告警上报发送缓冲区数据。根据短信发送操作的两个步骤,短信数据缓冲区的数据结构定义如下:
typedef struct{
unsigned char bStBuf;//bStBuf = SmsTx_Emty或者= SmsTx_CmdRdy或者
= SmsTx_Dly1或者= SmsTx_DatRdy或者= SmsTx_Dly2或者= SmsTx _Wait
unsigned char cmd_len;
char cmd_buf[32];
unsigned short dat_len;
char dat_buf[SMS_LEN+1];
unsigned char retry_time;//重传次数
}SmsTx_t;
① bStBuf成员: 用于描述短信数据缓冲区的状态。
② cmd_len成员: 用于描述cmd_buf中数据的长度。
③ cmd_buf成员: 用于存储短消息发送中的控制命令,如AT+CMGS=13583823789。
④ dat_len成员: 用于描述存储短消息发送中的信息体长度。
⑤ dat_buf成员: 用于存储短消息发送中的信息体。
⑥ retry_time成员: 用于描述短消息在发送失败时,重传的次数。
3.2.2 短信数据发送状态机
短信数据缓冲区的状态有6种:
① SmsTx_Empty: 短信数据缓冲区空。
② SmsTx_CmdRdy: 短信数据缓冲区控制命令准备好。
③ SmsTx_Dly1: 短信数据缓冲区延时1。
④ SmsTx_DatRdy: 短信数据缓冲区消息体准备好。
⑤ SmsTx_Dly2: 短信数据缓冲区延时2。
⑥ SmsTx_Wait: 短信数据缓冲区等待。
其状态的转移情况如图4所示。
图4 短信数据发送状态机
状态机的转移过程通过短消息发送缓冲区数据的发送来说明,其告警上报发送缓冲区的数据发送与此相同。
4 结束语
在整个移动2G光纤直放站近端机的监控软件中,其所有的软件设计都采用类似于短信收发处理的状态机、队列和软件定时器的设计思路,极大地提高移动2G光纤直放站近端机监控软件的效率。这种在前后台系统中使用状态机、队列和软件定时器的设计思路,可以应用到其他的嵌入式前后台系统中。
免责声明: 凡注明来源本网的所有作品,均为本网合法拥有版权或有权使用的作品,欢迎转载,注明出处。非本网作品均来自互联网,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责。