1.支持URC回调中接收数据.

2.修改struct at_obj,去除链表管理机制,使用单独任务接收处理每个at_obj.
This commit is contained in:
魔罗技术 2021-02-19 22:21:05 +08:00
parent e95ea83470
commit ea3dccb6b6
4 changed files with 281 additions and 193 deletions

View File

@ -95,7 +95,12 @@ static int wifi_reset_work(at_env_t *e)
break; break;
case 2: case 2:
wifi_open(); //重启启动wifi wifi_open(); //重启启动wifi
return true; e->state++;
break;
case 3:
if (e->is_timeout(a, 2000)) //延时等待2s
return true;
break;
} }
return false; return false;
} }
@ -146,7 +151,8 @@ void wifi_init(void)
wifi_uart_init(115200); wifi_uart_init(115200);
at_obj_init(&at, &at_adapter); at_obj_init(&at, &at_adapter);
wifi_open(); //启动WIFI
at_do_work(&at, wifi_reset_work, &at);
//初始化wifi //初始化wifi
at_send_multiline(&at, at_init_callbatk, wifi_init_cmds); at_send_multiline(&at, at_init_callbatk, wifi_init_cmds);

View File

@ -1,5 +1,6 @@
# AT Command # AT Command
## 介绍 ## 介绍
一种AT命令通信解析模块,支持裸机(at_chat)和OS版本(at)。适用于modem、WIFI模块、蓝牙通信。 一种AT命令通信解析模块,支持裸机(at_chat)和OS版本(at)。适用于modem、WIFI模块、蓝牙通信。
@ -22,7 +23,7 @@
at_chat 模块使用链式队列进行管理包含2条链表空闲链表和就绪链表。它们的每一个基本工作单元称为一个作业项对于将要执行的命令都会放到就绪链表中命令执行完成之后由空闲链表来进行回收作业项的定义如下 at_chat 模块使用链式队列进行管理包含2条链表空闲链表和就绪链表。它们的每一个基本工作单元称为一个作业项对于将要执行的命令都会放到就绪链表中命令执行完成之后由空闲链表来进行回收作业项的定义如下
``` ```c
/*AT作业项*/ /*AT作业项*/
typedef struct { typedef struct {
@ -51,7 +52,7 @@ typedef struct {
详细使用可以参考Demo程序wifi_task.c模块 详细使用可以参考Demo程序wifi_task.c模块
![m169 wifi模组通信效果图](images/wifi.jpg) ![m169 wifi模组通信效果图](images/wifi.jpg)
``` ```c
/* /*
* @brief 定义AT控制器 * @brief 定义AT控制器
@ -70,7 +71,7 @@ const at_adapter_t adap = { //AT适配器接口
3. 初始化AT控制器 3. 初始化AT控制器
``` ```c
at_obj_init(&at, &adap); at_obj_init(&at, &adap);
@ -79,7 +80,7 @@ at_obj_init(&at, &adap);
4. 将AT控制器放入任务中轮询考虑到处理实时性建议20ms以下 4. 将AT控制器放入任务中轮询考虑到处理实时性建议20ms以下
``` ```c
/* /*
* @brief wifi任务(10ms 轮询1次) * @brief wifi任务(10ms 轮询1次)
*/ */
@ -99,7 +100,7 @@ void wifi_task(void)
<= OK\r\n <= OK\r\n
``` ```c
/** /**
* @brief AT执行回调处理程序 * @brief AT执行回调处理程序
@ -124,12 +125,12 @@ at_send_singlline(&at, test_gpio_callback, "AT+GPIO_TEST_EN=1");
- at_do_cmd执行AT命令可以通过这个接口进一步封装出一常用的单行命令、多行命令。 - at_do_cmd执行AT命令可以通过这个接口进一步封装出一常用的单行命令、多行命令。
- at_split_respond_lines命令响应分割器。 - at_split_respond_lines命令响应分割器。
- at_do_work如果发送的数据比较复杂如GPRS模组发送短信或者发送socket数据需要等待"<"提示符,可以通过这个接口自定义收发。 - at_do_work适用于发送组合命令如GPRS模组发送短信或者发送socket数据需要等待"<"或者"CONNECT"提示符,可以通过这个接口自定义收发。
##### 使用演示(后续会提供一个无线GPRS模块演示程序出来) ##### 使用演示(后续会提供一个无线GPRS模块演示程序出来)
``` ```c
static at_obj_t at; //定义AT控制器 static at_obj_t at; //定义AT控制器对象
static char urc_buf[128]; //URC主动上报缓冲区 static char urc_buf[128]; //URC主动上报缓冲区
@ -142,7 +143,8 @@ const at_adapter_t adap = { //AT适配器接口
.urc_bufsize = sizeof(urc_buf), .urc_bufsize = sizeof(urc_buf),
.utc_tbl = utc_tbl, .utc_tbl = utc_tbl,
.urc_tbl_count = sizeof(utc_tbl) / sizeof(utc_item_t), .urc_tbl_count = sizeof(utc_tbl) / sizeof(utc_item_t),
//debug调试接口
.debug = at_debug,
//适配GPRS模块的串口读写接口 //适配GPRS模块的串口读写接口
.write = uart_write, .write = uart_write,
.read = uart_read .read = uart_read
@ -150,15 +152,15 @@ const at_adapter_t adap = { //AT适配器接口
``` ```
3. 初始化AT控制器并创建AT线程 3. 初始化AT控制器并创建AT接收处理线程
``` ```c
void at_thread(void) void at_thread(void)
{ {
at_obj_create(&at, &adap); at_obj_create(&at, &adap);
while (1) { while (1) {
at_thread(&at); at_process(&at);
} }
} }
@ -172,7 +174,7 @@ void at_thread(void)
<= +CSQ: 24, 0 <= +CSQ: 24, 0
<= OK <= OK
``` ```c
/* /*
* @brief 获取csq值 * @brief 获取csq值
*/ */
@ -191,4 +193,7 @@ bool read_csq_value(at_obj_t *at, int *rssi, int *error_rate)
} }
``` ```

284
at.c
View File

@ -8,17 +8,33 @@
* Change Logs: * Change Logs:
* Date Author Notes * Date Author Notes
* 2020-01-02 Morro Initial version. * 2020-01-02 Morro Initial version.
* 2021-02-01 Morro URC回调中接收数据.
* 2021-02-05 Morro 1.struct at_obj,
* 2. at_obj_destroy接口
******************************************************************************/ ******************************************************************************/
#include "at/at.h"
#include "at.h" #include "misc/comdef.h"
#include <stdarg.h> #include <stdarg.h>
#include <string.h> #include <string.h>
#include <stdio.h> #include <stdio.h>
#include <stddef.h>
//超时判断 //超时判断
#define AT_IS_TIMEOUT(start, time) (at_get_ms() - (start) > (time)) #define AT_IS_TIMEOUT(start, time) (at_get_ms() - (start) > (time))
static LIST_HEAD(atlist); /*链表头结点 ----------------*/ /*
* @brief
*/
static void nop_dbg(const char *fmt, ...){}
/*
* @brief at控制器
*/
at_obj_t *get_at_obj(struct at_work_ctx *e)
{
return container_of(e, at_obj_t, ctx);
}
/* /*
* @brief * @brief
@ -41,22 +57,22 @@ static void put_line(at_obj_t *at, const char *s)
} }
//打印输出 //打印输出
static void at_print(at_obj_t *at, const char *cmd, ...) static void at_print(struct at_work_ctx *e, const char *cmd, ...)
{ {
va_list args; va_list args;
va_start(args, cmd); va_start(args, cmd);
char buf[MAX_AT_CMD_LEN]; char buf[MAX_AT_CMD_LEN];
vsnprintf(buf, sizeof(buf), cmd, args); vsnprintf(buf, sizeof(buf), cmd, args);
put_line(at, buf); put_line(get_at_obj(e), buf);
va_end(args); va_end(args);
} }
/* /*
* @brief * @brief
*/ */
static void recvbuf_clr(at_obj_t *at) static void recvbuf_clr(struct at_work_ctx *e)
{ {
at->rcv_cnt = 0; get_at_obj(e)->rcv_cnt = 0;
} }
//等待AT命令响应 //等待AT命令响应
@ -65,12 +81,10 @@ static at_return wait_resp(at_obj_t *at, at_respond_t *r)
at->resp = r; at->resp = r;
at->ret = AT_RET_TIMEOUT; at->ret = AT_RET_TIMEOUT;
at->resp_timer = at_get_ms(); at->resp_timer = at_get_ms();
recvbuf_clr(at); //清空接收缓存 at->rcv_cnt = 0; //清空接收缓存
at->wait = 1; at_sem_wait(at->completed, r->timeout);
at_sem_wait(at->completed, r->timeout);
at->adap.debug("<-\r\n%s\r\n", r->recvbuf); at->adap.debug("<-\r\n%s\r\n", r->recvbuf);
at->resp = NULL; at->resp = NULL;
at->wait = 0;
return at->ret; return at->ret;
} }
@ -79,11 +93,13 @@ static at_return wait_resp(at_obj_t *at, at_respond_t *r)
* @param[in] resp - ("OK",">") * @param[in] resp - ("OK",">")
* @param[in] timeout - * @param[in] timeout -
*/ */
at_return wait_resp_sync(struct at_obj *at, const char *resp, at_return wait_resp_sync(struct at_work_ctx *e, const char *resp,
unsigned int timeout) unsigned int timeout)
{ {
char buf[64]; char buf[64];
int cnt = 0, len; int cnt = 0, len;
at_obj_t *at = get_at_obj(e);
at_return ret = AT_RET_TIMEOUT; at_return ret = AT_RET_TIMEOUT;
unsigned int timer = at_get_ms(); unsigned int timer = at_get_ms();
while (at_get_ms() - timer < timeout) { while (at_get_ms() - timer < timeout) {
@ -105,58 +121,54 @@ at_return wait_resp_sync(struct at_obj *at, const char *resp,
/* /*
* @brief AT内核配置 * @brief AT控制器
*/ */
void at_obj_create(at_obj_t *at, const at_adapter_t *adap) void at_obj_create(at_obj_t *at, const at_adapter_t *adap)
{ {
at_work_env_t *e; at_work_ctx_t *ctx;
at->adap = *adap; at->adap = *adap;
at->rcv_cnt = 0; at->rcv_cnt = 0;
at->cmd_lock = at_sem_new(1); at->cmd_lock = ril_sem_new(1);
at->completed = at_sem_new(0); at->completed = ril_sem_new(0);
e = &at->env; ctx = &at->ctx;
e->at = at; ctx->printf = at_print;
e->printf = at_print; ctx->recvclr = recvbuf_clr;
e->recvclr = recvbuf_clr; ctx->read = adap->read;
e->read = adap->read; ctx->write = adap->write;
e->write = adap->write; ctx->wait_resp = wait_resp_sync;
e->wait_resp = wait_resp_sync;
if (at->adap.debug == NULL)
at->adap.debug = nop_dbg;
list_add_tail(&at->node, &atlist);
}
/*
* @brief AT
*/
void at_obj_destroy(at_obj_t *at)
{
list_del(&at->node);
} }
/* /*
* @brief * @brief
* @param[in] fmt - * @param[in] fmt -
* @param[in] r - ,NULL, OK表示成功,3s * @param[in] r - ,NULL, OK表示成功,5s
* @param[in] args - * @param[in] args -
*/ */
at_return at_do_cmd(at_obj_t *at, at_respond_t *r, const char *cmd) at_return at_do_cmd(at_obj_t *at, at_respond_t *r, const char *cmd)
{ {
at_return ret; at_return ret;
char defbuf[64]; char defbuf[64];
at_respond_t default_resp = {"OK", defbuf, sizeof(defbuf), 3000}; at_respond_t default_resp = {"OK", defbuf, sizeof(defbuf), 5000};
if (r == NULL) { if (r == NULL) {
r = &default_resp; //默认响应 r = &default_resp; //默认响应
} }
if (!at_sem_wait(at->cmd_lock, r->timeout)) { if (!at_sem_wait(at->cmd_lock, r->timeout)) {
return AT_RET_TIMEOUT; return AT_RET_TIMEOUT;
} }
at->busy = true;
while (at->urc_cnt) { while (at->urc_cnt) {
at_delay(10); at_delay(10);
} }
put_line(at, cmd); put_line(at, cmd);
ret = wait_resp(at, r); ret = wait_resp(at, r);
at_sem_post(at->cmd_lock); at_sem_post(at->cmd_lock);
at->busy = false;
return ret; return ret;
} }
@ -171,14 +183,17 @@ at_return at_do_work(at_obj_t *at, at_work work, void *params)
if (!at_sem_wait(at->cmd_lock, 150 * 1000)) { if (!at_sem_wait(at->cmd_lock, 150 * 1000)) {
return AT_RET_TIMEOUT; return AT_RET_TIMEOUT;
} }
while (at->urc_cnt) { at->busy = true;
while (at->urc_cnt) { //等待URC处理完成
at_delay(1); at_delay(1);
} }
at->env.params = params; at->ctx.params = params;
at->dowork = true; at->dowork = true;
ret = work(&at->env); at->rcv_cnt = 0;
ret = work(&at->ctx);
at->dowork = false; at->dowork = false;
at_sem_post(at->cmd_lock); at_sem_post(at->cmd_lock);
at->busy = false;
return ret; return ret;
} }
@ -204,73 +219,98 @@ int at_split_respond_lines(char *recvbuf, char *lines[], int count, char separat
} }
s++; s++;
} }
return i; return i;
} }
/* /*
* @brief urc * @brief urc
* @param[in] urcline - URC行 * @param[in] urcline - URC行
* @return none * @return true - , false - URC
*/ */
static void urc_handler_entry(at_obj_t *at, char *urcline, unsigned int size) static bool urc_handler_entry(at_obj_t *at, char *urcline, unsigned int size)
{ {
int i, n; int i;
utc_item_t *tbl = at->adap.utc_tbl; int ch = urcline[size - 1];
at_urc_ctx_t context = {at->adap.read, urcline, at->adap.urc_bufsize, size};
for (i = 0; i < at->adap.urc_tbl_count; i++){ const utc_item_t *tbl = at->adap.utc_tbl;
n = strlen(tbl->prefix);
if (n > 0 && strncmp(urcline, tbl->prefix, n) == 0) { if (tbl == NULL)
tbl->handler(urcline, size); return true;
for (i = 0; i < at->adap.urc_tbl_count; i++) {
if (strstr(urcline, tbl->prefix)) { /* 匹配前缀*/
if (tbl->end_mark) { /* 匹配结束标记*/
if (!strchr(tbl->end_mark, ch))
return false;
} else if (!(ch == '\r' || ch == '\n'|| ch == '\0'))
return false;
at->adap.debug("<=\r\n%s\r\n", urcline); at->adap.debug("<=\r\n%s\r\n", urcline);
return;
tbl->handler(&context); /* 递交到上层处理 */
return true;
} }
tbl++; tbl++;
} }
return false;
if (size >= 2 && !at->wait) //自动输出
at->adap.debug("%s\r\n", urcline);
} }
/* /*
* @brief urc * @brief urc
* @param[in] buf - * @param[in] ch -
* @return none * @return none
*/ */
static void urc_recv_process(at_obj_t *at, const char *buf, unsigned int size) static void urc_recv_process(at_obj_t *at, const char *buf, unsigned int size)
{ {
char *urc_buf; register char *urc_buf;
unsigned short urc_size; int ch;
unsigned char c; urc_buf = at->adap.urc_buf;
urc_buf = (char *)at->adap.urc_buf;
urc_size = at->adap.urc_bufsize; //接收超时处理,默认MAX_URC_RECV_TIMEOUT
if (at->urc_cnt > 0 && size == 0) { if (at->urc_cnt > 0 && AT_IS_TIMEOUT(at->urc_timer, MAX_URC_RECV_TIMEOUT)) {
if (AT_IS_TIMEOUT(at->urc_timer, 100)) { //100ms超时 urc_buf[at->urc_cnt] = '\0';
urc_buf[at->urc_cnt] = '\0'; at->urc_cnt = 0;
at->urc_cnt = 0; at->adap.debug("urc recv timeout=>%s\r\n", urc_buf);
at->adap.debug("urc recv timeout=>%s\r\n", urc_buf); }
}
} else { while (size--) {
at->urc_timer = at_get_ms(); at->urc_timer = at_get_ms();
while (size--) { ch = *buf++;
c = *buf++; urc_buf[at->urc_cnt++] = ch;
if (c == '\r' || c == '\n') { //收到1行
urc_buf[at->urc_cnt] = '\0'; if (strchr(SPEC_URC_END_MARKS, ch) || ch == '\0') { //结束标记
if (at->urc_cnt > 2) urc_buf[at->urc_cnt] = '\0';
urc_handler_entry(at, urc_buf, at->urc_cnt);
if (ch == '\r' || ch == '\n'|| ch == '\0') { //检测到1行URC
if (at->urc_cnt > 2) {
if (!urc_handler_entry(at, urc_buf, at->urc_cnt) && !at->busy)
at->adap.debug("%s\r\n", urc_buf); //未识别到的URC
}
at->urc_cnt = 0;
} else if (urc_handler_entry(at, urc_buf, at->urc_cnt)) {
at->urc_cnt = 0; at->urc_cnt = 0;
} else {
urc_buf[at->urc_cnt++] = c;
if (at->urc_cnt >= urc_size) //溢出处理
at->urc_cnt = 0;
} }
} } else if (at->urc_cnt >= at->adap.urc_bufsize) //溢出处理
at->urc_cnt = 0;
} }
} }
/*
* @brief
* @return none
*/
static void resp_notification(at_obj_t *at, at_return ret)
{
at->ret = ret;
at_sem_post(at->completed);
}
/* /*
* @brief * @brief
* @param[in] buf - * @param[in] buf -
* @param[in] size - * @param[in] size -
* @return none * @return none
*/ */
static void resp_recv_process(at_obj_t *at, const char *buf, unsigned int size) static void resp_recv_process(at_obj_t *at, const char *buf, unsigned int size)
@ -279,36 +319,34 @@ static void resp_recv_process(at_obj_t *at, const char *buf, unsigned int size)
unsigned short rcv_size; unsigned short rcv_size;
at_respond_t *resp = at->resp; at_respond_t *resp = at->resp;
if (resp == NULL || size == 0) if (resp == NULL) //无命令请求
return;
rcv_buf = (char *)resp->recvbuf;
rcv_size = resp->bufsize;
if (at->rcv_cnt + size >= rcv_size) { //接收溢出
at->rcv_cnt = 0;
at->adap.debug("Receive overflow:%s", rcv_buf);
}
/*将接收到的数据放至rcv_buf中 ---------------------------------------------*/
memcpy(rcv_buf + at->rcv_cnt, buf, size);
at->rcv_cnt += size;
rcv_buf[at->rcv_cnt] = '\0';
if (!at->wait)
return;
if (strstr(rcv_buf, resp->matcher)) { //接收匹配
at->ret = AT_RET_OK;
} else if (strstr(rcv_buf, "ERROR")) {
at->ret = AT_RET_ERROR;
} else if (AT_IS_TIMEOUT(at->resp_timer, resp->timeout)) {
at->ret = AT_RET_TIMEOUT;
} else if (at->suspend) //强制终止
at->ret = AT_RET_ABORT;
else
return; return;
at_sem_post(at->completed); if (size) {
rcv_buf = (char *)resp->recvbuf;
rcv_size = resp->bufsize;
if (at->rcv_cnt + size >= rcv_size) { //接收溢出
at->rcv_cnt = 0;
at->adap.debug("Receive overflow:%s", rcv_buf);
}
/*将接收到的数据放至rcv_buf中 ---------------------------------------------*/
memcpy(rcv_buf + at->rcv_cnt, buf, size);
at->rcv_cnt += size;
rcv_buf[at->rcv_cnt] = '\0';
if (strstr(rcv_buf, resp->matcher)) { //接收匹配
resp_notification(at, AT_RET_OK);
} else if (strstr(rcv_buf, "ERROR")) {
resp_notification(at, AT_RET_ERROR);
}
}
if (AT_IS_TIMEOUT(at->resp_timer, resp->timeout)) //接收超时
resp_notification(at, AT_RET_TIMEOUT);
else if (at->suspend) //强制终止
resp_notification(at, AT_RET_ABORT);
} }
/* /*
@ -317,7 +355,7 @@ static void resp_recv_process(at_obj_t *at, const char *buf, unsigned int size)
*/ */
bool at_obj_busy(at_obj_t *at) bool at_obj_busy(at_obj_t *at)
{ {
return at->wait == 0 && AT_IS_TIMEOUT(at->urc_timer, 2000); return !at->busy && AT_IS_TIMEOUT(at->urc_timer, 2000);
} }
/* /*
@ -339,28 +377,18 @@ void at_resume(at_obj_t *at)
} }
/* /*
* @brief AT轮询线程 * @brief AT处理
* @return none * @return none
*/ */
void at_thread(void) void at_process(at_obj_t *at)
{ {
at_obj_t *at; char c;
struct list_head *list ,*n = NULL; unsigned int len;
int len; if (at->dowork) /* 自定义命令处理 */
char buf[32]; return;
while (1) { do {
/*遍历所有at_obj结点*/ len = at->adap.read(&c, 1);
list_for_each_safe(list, n, &atlist) { urc_recv_process(at, &c,len);
at = list_entry(list, at_obj_t, node); resp_recv_process(at, &c, len);
if (!at->dowork) { } while (len);
//#warning "读取待优化(readline) ..."
len = at->adap.read(buf, sizeof(buf));
urc_recv_process(at, (char *)buf, len);
if (len > 0) {
resp_recv_process(at, buf, len);
}
}
}
at_delay(1);
}
} }

149
at.h
View File

@ -8,101 +8,150 @@
* Change Logs: * Change Logs:
* Date Author Notes * Date Author Notes
* 2020-01-02 Morro Initial version. * 2020-01-02 Morro Initial version.
* 2021-02-01 Morro URC回调中接收数据.
* 2021-02-05 Morro 1.struct at_obj,
* 2. at_obj_destroy接口
******************************************************************************/ ******************************************************************************/
#ifndef _AT_H_ #ifndef _AT_H_
#define _AT_H_ #define _AT_H_
#include "at_util.h" #include "at_util.h"
#include "list.h"
#include <stdbool.h> #include <stdbool.h>
/* 单行最大命令长度 */
#define MAX_AT_CMD_LEN 128 #define MAX_AT_CMD_LEN 128
struct at_obj; /*AT对象*/ /* 单行urc接收超时时间*/
#define MAX_URC_RECV_TIMEOUT 300
/*urc处理项 -----------------------------------------------------------------*/ /* 指定的URC 结束标记列表 */
typedef struct { #define SPEC_URC_END_MARKS ",\r\n"
const char *prefix; //URC前缀
void (*handler)(char *recvbuf, int size); struct at_obj; /* AT对象*/
}utc_item_t;
/*AT接口适配器 ---------------------------------------------------------------*/
typedef struct {
/*数据读写接口 -----------------------------------------------------------*/
unsigned int (*read)(void *buf, unsigned int len);
unsigned int (*write)(const void *buf, unsigned int len);
void (*debug)(const char *fmt, ...);
utc_item_t *utc_tbl; /* utc 表*/
char *urc_buf; /* urc接收缓冲区*/
unsigned short urc_tbl_count; /* urc表项个数*/
unsigned short urc_bufsize; /* urc缓冲区大小*/
}at_adapter_t;
/*AT命令响应码 ---------------------------------------------------------------*/ /*AT命令响应码 ---------------------------------------------------------------*/
typedef enum { typedef enum {
AT_RET_OK = 0, /*执行成功*/ AT_RET_OK = 0, /* 执行成功*/
AT_RET_ERROR, /*执行错误*/ AT_RET_ERROR, /* 执行错误*/
AT_RET_TIMEOUT, /*响应超时*/ AT_RET_TIMEOUT, /* 响应超时*/
AT_RET_ABORT, /*未知错误*/ AT_RET_ABORT, /* 未知错误*/
}at_return; }at_return;
/**
* @brief URC (Context)
*/
typedef struct {
/**
* @brief
* @params buf -
* @params len -
*/
unsigned int (*read)(void *buf, unsigned int len);
char *buf; /* 数据缓冲区 */
int bufsize; /* 缓冲区大小 */
int recvlen; /* 已接收数据长度*/
} at_urc_ctx_t;
/*(Unsolicited Result Codes (URCs))处理项 ------------------------------------*/
typedef struct {
/**
* @brief urc ("+CSQ: ")
*/
const char *prefix;
/**
* @brief urc (DEF_URC_END_MARKS列表"\n")
* @note
*/
const char *end_mark;
/**
* @brief urc处理程序
* @params ctx - URC
*/
void (*handler)(at_urc_ctx_t *ctx);
}utc_item_t;
/**AT接口适配器 --------------------------------------------------------------*/
typedef struct {
//数据写操作
unsigned int (*write)(const void *buf, unsigned int len);
//数据读操作
unsigned int (*read)(void *buf, unsigned int len);
//调试打印输出,如果不需要则填NULL
void (*debug)(const char *fmt, ...);
//utc 处理函数表,如果不需要则填NULL
utc_item_t *utc_tbl;
//urc接收缓冲区,如果不需要则填NULL
char *urc_buf;
//urc表项个数,如果不需要则填0
unsigned short urc_tbl_count;
//urc缓冲区大小,如果不需要则填0
unsigned short urc_bufsize;
}at_adapter_t;
/*AT响应参数 -----------------------------------------------------------------*/ /*AT响应参数 -----------------------------------------------------------------*/
typedef struct { typedef struct {
const char *matcher; /*接收匹配串*/ const char *matcher; /* 接收匹配串 */
char *recvbuf; /*接收缓冲区*/ char *recvbuf; /* 接收缓冲区 */
unsigned short bufsize; /*最大接收长度*/ unsigned short bufsize; /* 缓冲区长度 */
unsigned int timeout; /*最大超时时间 */ unsigned int timeout; /* 最大超时时间 */
}at_respond_t; }at_respond_t;
/*AT作业 ---------------------------------------------------------------------*/ /** work context ------------------------------------------------------------*/
typedef struct at_work_env { /**
struct at_obj *at; * @brief AT作业上下文(Work Context)
*/
typedef struct at_work_ctx {
//作业参数,由at_do_work接口传入
void *params; void *params;
unsigned int (*write)(const void *buf, unsigned int len); unsigned int (*write)(const void *buf, unsigned int len);
unsigned int (*read)(void *buf, unsigned int len); unsigned int (*read)(void *buf, unsigned int len);
void (*printf)(struct at_obj *at, const char *frm, ...); //打印输出
at_return (*wait_resp)(struct at_obj *at, const char *resp, unsigned int timeout); void (*printf)(struct at_work_ctx *self, const char *fmt, ...);
void (*recvclr)(struct at_obj *at); /*清空接收缓冲区*/ //响应等待
}at_work_env_t; at_return (*wait_resp)(struct at_work_ctx *self, const char *prefix,
unsigned int timeout);
//清除接收缓存
void (*recvclr)(struct at_work_ctx *self);
}at_work_ctx_t;
/*AT对象 ---------------------------------------------------------------------*/ /*AT对象 ---------------------------------------------------------------------*/
typedef struct at_obj { typedef struct at_obj {
struct list_head node; at_adapter_t adap; /* 接口适配器*/
at_adapter_t adap; at_work_ctx_t ctx; /* work context*/
at_work_env_t env; at_sem_t cmd_lock; /* 命令锁*/
at_sem_t cmd_lock; /*命令锁*/ at_sem_t completed; /* 命令完成信号*/
at_sem_t completed; /*命令处理完成*/ at_respond_t *resp; /* AT应答信息*/
at_respond_t *resp; unsigned int resp_timer; /* 响应接收定时器*/
unsigned int resp_timer; unsigned int urc_timer; /* URC定时器 */
unsigned int urc_timer; at_return ret; /* 命令执行结果*/
at_return ret;
//urc接收计数, 命令响应接收计数器 //urc接收计数, 命令响应接收计数器
unsigned short urc_cnt, rcv_cnt; unsigned short urc_cnt, rcv_cnt;
unsigned char wait : 1; unsigned char busy : 1;
unsigned char suspend: 1; unsigned char suspend: 1;
unsigned char dowork : 1; unsigned char dowork : 1;
}at_obj_t; }at_obj_t;
typedef at_return (*at_work)(at_work_env_t *); typedef at_return (*at_work)(at_work_ctx_t *);
void at_obj_create(at_obj_t *at, const at_adapter_t *adap); /*AT初始化*/ void at_obj_create(at_obj_t *at, const at_adapter_t *adap); /* AT初始化*/
void at_obj_destroy(at_obj_t *at); void at_obj_destroy(at_obj_t *at);
bool at_obj_busy(at_obj_t *at); bool at_obj_busy(at_obj_t *at);
void at_suspend(at_obj_t *at); /*挂起*/ void at_suspend(at_obj_t *at); /* 挂起*/
void at_resume(at_obj_t *at); /*恢复*/ void at_resume(at_obj_t *at); /* 恢复*/
at_return at_do_cmd(at_obj_t *at, at_respond_t *r, const char *cmd); at_return at_do_cmd(at_obj_t *at, at_respond_t *r, const char *cmd);
//响应行分割处理
int at_split_respond_lines(char *recvbuf, char *lines[], int count, char separator); int at_split_respond_lines(char *recvbuf, char *lines[], int count, char separator);
at_return at_do_work(at_obj_t *at, at_work work, void *params);/*执行AT作业*/ at_return at_do_work(at_obj_t *at, at_work work, void *params);/* 自定义AT作业*/
void at_thread(void); /*AT线程*/ void at_process(at_obj_t *at); /* AT接收处理*/
#endif #endif