AT指令的一种解析想法

       最近一直在忙着做NBlot模块,话说博客也好久没有更新了,最近整理下思路,

做下总结,权当是一片博客吧。

AT指令大家很熟悉了,这里不再做介绍(不熟悉的,请参考《GSM Rec.07.07标准AT命令》)。

一个命令类型最多分为四类:
1、设置类型,CMD=设置参数,回复OK或者ERROR;        可用通用模式解析
2、读取类型,CMD?读取参数当前值,回复+CMD:参数列表;解析返回字符串,可能做命令转变
3、测试类型,CMD=?读取默认参数值和可设置范围;       解析返回字符串,可能做命令转变,也有可能返回OK
4、执行类型,CMD  读取不可设置参数值;              解析返回字符串,可能做命令转变

这里解析主要运用到消息地图的方法,同时为了减小地图的大小,每个命令字节不具体细分是那个

命令类别,交给具体的命令解析函数去处理。如下:

//! 消息地图
const static msg_t c_tMSGMap[]  =  {
    {"AT",              JZQ_AT_CMD_AT},
    {"ATE0",            JZQ_AT_CMD_ATE0},
    {"AT$MYGMR",        JZQ_AT_CMD_MYGMR},
    {"AT$MYTYPE",       JZQ_AT_CMD_MYTYPE},
    {"ATI",             JZQ_AT_CMD_ATI},
    {"AT$MYCCID",       JZQ_AT_CMD_MYCCID},
    {"AT+CSQ",          JZQ_AT_CMD_CSQ},
    {"AT+CREG",         JZQ_AT_CMD_CREG},
    {"AT+CIMI",         JZQ_AT_CMD_CIMI},
};

在具体的命令解析函数里面,都次都可以指定NBlot模组的数据返回解析函数,做到每个命令和命令

的回复处理函数一 一对应。如下:

发送给NBlot模块命令:

    user_nblot_send(get_byte_pipe_buffer(&tBytePipe),get_byte_pipe_num(&tBytePipe));
    s_tNblotAppBuffer.pRxCallBack = JZQ_AT_CMD_MYNETWRITE_ACK; 

NBlot模块的命令接收解析:

    switch(s_tState){
        case FSM_NBLOT_PROTOCOL_START:
            s_tState = FSM_NBLOT_PROTOCOL_WAIT_DATA;
            s_tNBlotAppBuffer.hwBufferSize = 0;
            //break;
        case FSM_NBLOT_PROTOCOL_WAIT_DATA:
            if(nblot_rx_buffer_is_empty()){
                break;
            }
            s_tState = FSM_NBLOT_PROTOCOL_REV_FRAME;
            //break;
        case FSM_NBLOT_PROTOCOL_REV_FRAME:
            tFsmRt = nblot_protocol_revice_frame();
            if(fsm_rt_cpl ==  tFsmRt|| fsm_rt_buffer_full == tFsmRt){
                s_tState = FSM_NBLOT_PROTOCOL_HANDLE;
            }else if(fsm_rt_err == tFsmRt){
                RESET_FSM_NBLOT_PROTOCOL();
                return fsm_rt_err;
            }
            break;
        case FSM_NBLOT_PROTOCOL_HANDLE:
            if(NULL != s_ptRxCallBack){
                s_ptRxCallBack(s_tNBlotAppBuffer.chBuffer,s_tNBlotAppBuffer.hwBufferSize);
            }
            RESET_FSM_NBLOT_PROTOCOL();
            return fsm_rt_cpl;
        default:
            FSM_DEFAULT_ACTION();
    }

至此一个大体的AT命令解析架构介绍完毕了,但是这个架构会有一个问题,

就是如果仅仅考虑问答式命令形式,这个架构再加一个超时无效处理的维护

线程就可以(维护线程就干一件事情,发送命令后,启动维护线程,设置超

时命令,如果在设置的超时时间内回复,则通过发送命令时候指定的处理函

数解析,如果超时了,就把指针清NULL,同时发送超时消息)。如果还要考

虑主动上报,那么上面的架构就需要在(每个)处理函数里面加入主动上报

的处理逻辑,否则逻辑上就不对。这样就会产生很多“脏代码”,处理方法是利

用行为继承,在FSM_NBLOT_PROTOCOL_HANDLE前面增加一个DEFULT_HANDLE

处理,把一些通用或者主动上报的(每个命令帧都需要考虑的情况)打包成一个处理函数。

到目前为止,AT指令解析,基本讲完了,说些题外话:

1、下行数据可能要考虑拼包问题,因为在测试过程中发现主站下发一个APDU,在模块

      端会被分为两个socket包,并且时间间隔还很长(目前监控到最长30S);这里有一

      个问题,如果考虑拼包,那么就和协议关联了。

2、socket会自己断开,一定要有重建机制;

 

2018.10.23

       前几天和一位老工程师交流这个AT指令解析方法,有一些心得,记录下,也算是一种

思想上的提高。

        这个解析方法有个确实,就是当指令阅读,消息地图越大,检索时间越长。如果系统

要求这个检索时间和地图大小无关,时间是“确定”的,怎么解决?

        首先我们就想到是什么?什么情况下检索时间和数据表大小无关?我们是不是想到了

数组下标,查表的方法。没错,查表的时间确定,并且和表的大小无关。那么我们怎么进行

查表呢?首先我们需要有一个格式化算法,把AT指令流格式化为数组下标,然后用数组下标

直接查表来得到解析函数。

        这个格式化的算法怎么设计,我目前还没有很好的思路,如果你有什么好的想法,请留言

给我。

2018.12.26

        最近有个NTC测温的项目,是另外一个同事负责,给出的温度和阻值对应曲线不是线性的。

于是这个同事就犯愁了,如果用传统的if/else和switch/case架构去做分段解析,那么精度要求高的

时候,段就会分很多;如果段分的少,精度又下降了。那有没有一种思路,既可以兼顾精度,又可以

使代码架构简单,降低体力活力度?于是有位老专家提出了一种“算法”,用折线去模拟曲线,同时

又没有if/else判断,我感觉思路很好,记录如下:

1、首先用折线去模拟曲线,这个没什么好办法,每段去选择合适的K和B的参数,这样就有一张参数表;

2、那么怎么知道现在读取的阻值是落到那一段呢?摒弃程序分支判断的方法就是查表法,就是说根据

阻值我就可以计算出目前在表里的位置,然后查表就得到了K和B;

3、这里取折线段的方法很重要,你可以用线性发(均分法),这样查表算法就是kx+b;如果采用非线性

方法,那个查表算法很可能是高次函数或者指数等高阶函数;

这里就设计到计算资源问题,当然根据精度,采用定点算法,实时性会大大提高。

你可能感兴趣的:(单片机)