产品很多配置信息需要后期进行配置,配置交互最好的方式之一是上位机通过串口与开发板进行交互来完配置。这里我准备引入AT指令来进行配置。
我采用串口中断+定时器中断来实现了串口的接收,下面进行AT指令框架的创建。
typedef enum{
AT_CMD_TEST = 0,
AT_CMD_SETRTC,
AT_END
}AT_Cmd;
typedef unsigned char (*pFunc)(unsigned char *ptr, unsigned char len);
typedef struct
{
AT_Cmd cmd; /* 指令序号 */
unsigned char *str; /* 指令内容 */
pFunc cb; /* 指令执行 */
}AT_cmd_func;
/* AT指令表 */
const AT_cmd_func at_cmd_func[] = {
{AT_CMD_TEST, "AT", at_cmd_test},
{AT_CMD_SETRTC, "AT+SETRTC=", at_cmd_setrtc},
{AT_END, NULL, NULL}
};
/* 指令执行函数 */
unsigned char at_cmd_test(unsigned char *p, unsigned char len){
AT_DEBUG_INFO("AT+OKrn");
return 0;
}
unsigned char at_cmd_setrtc(unsigned char *p, unsigned char len){
AT_DEBUG_INFO("setrtcrn");
return 0;
}
/* 查找指令表中对应的指令 */
unsigned char AT_cmd_search(unsigned char *p, unsigned char len)
{
unsigned char ret = 0;
unsigned char *pstr;
unsigned char i, n;
for(i=1; at_cmd_func[i].cmd != AT_END; i++)
{
n = mstrlen(at_cmd_func[i].str);
if(!mstrncmp(p, at_cmd_func[i].str, n)){
ret = i;
break;
}
}
return ret;
}
/* AT指令解析 */
unsigned char at_cmd_parse(unsigned char *p, unsigned char len){
unsigned char ret = AT_SUCCESS;
unsigned char index = 0;
unsigned char n;
if(len < 4) {
return AT_ERR;
}
if((p[0] == 'A') && (p[1] == 'T') && (p[len-2] == 0x0D) && (p[len-1] == 0x0A)) {
if(len == 4) { /* 测试指令 */
if(at_cmd_func[AT_CMD_TEST].cb != NULL) {
at_cmd_func[AT_CMD_TEST].cb(NULL, 0); /* 执行测试指令 */
}
}else if(p[2] == '+') { /* 执行指令解析 */
index = AT_cmd_search(p, len); /* 查找匹配的执行指令, 0-已匹配, !0-未匹配*/
if(index) {
if(at_cmd_func[index].cb != NULL) {
n = mstrlen(at_cmd_func[index].str);
ret = at_cmd_func[index].cb(p+n, len-n); /* 执行对应的指令函数, p+n:将指令参数传输执行函数,len-n-2:指令参数有效长度 */
}else {
ret = AT_ERR_FUN_UNUSED; /* 没有可执行函数 */
}
}else {
ret = AT_ERR_UNINVAL; /* 未找到匹配的指令 */
}
}else { /* 格式不匹配 */
return AT_ERR;
}
return ret;
}
}
【测试】
我在接收到指令后执行at_cmd_parse 发送AT、AT+SETRTC=成功的返回需要的信息。
审核编辑:汤梓红
全部0条评论
快来发表一下你的评论吧 !