AT命令通信解析模块:构建高效通信的基石

2024-08-16

在通信技术日益发展的今天,AT命令作为一种古老但依旧强大的通信协议,依然广泛应用于调制解调器、移动通信设备、蓝牙模块、GPS模块等多种设备中。AT命令(Attention Command)源于早期调制解调器制造商的引入,通过“AT”前缀吸引设备注意并执行特定指令。随着技术的演进,AT命令的应用范围不断扩大,其简单而有效的特点使得它成为控制和配置设备的通用方式。本文将详细介绍一种AT命令通信解析模块的设计和实现,探讨其在现代通信技术中的重要作用。



AT命令通信解析模块概述

AT命令通信解析模块是连接AT设备和控制系统之间的桥梁,它负责解析由控制系统发送的AT命令,并处理来自AT设备的响应。一个高效的AT命令通信解析模块需要支持多种命令格式、错误处理、超时机制以及URC(Unsolicited Result Code,非请求结果码)处理等功能。



软件架构

该AT命令通信解析模块支持裸机(at_chat)和操作系统(OS)两个版本,分别适用于不同的应用场景。在裸机版本中,模块使用链式队列及异步回调方式处理AT命令收发,支持URC处理、自定义命令发送与解析作业。而在OS版本中,模块则依赖于操作系统提供的信号量操作、任务延时等接口进行工作。



基本接口与功能

1. at_send_singlline

该接口用于发送单行AT命令,并默认等待“OK”响应,超时时间为3秒。如果命令执行成功,将返回成功响应;否则,返回错误响应或超时信息。

2. at_send_multiline

该接口支持发送多行AT命令,同样默认等待“OK”响应,超时时间为3秒。多行命令通常用于执行复杂操作,如设置网络参数等。

3. at_do_cmd

该接口允许用户自定义发送格式与接收匹配串,提供更高的灵活性。用户可以根据需要,定义复杂的命令序列和期望的响应格式。

4. at_do_work

适用于发送组合命令,如发送短信或进行网络连接等需要等待特定提示符的操作。通过该接口,用户可以自定义发送与接收解析逻辑,以适应不同设备的特殊需求。

设计与实现

1. AT控制器与通信适配器

AT控制器是该模块的核心,负责维护命令的执行状态、处理URC等。通信适配器则负责与AT设备的物理连接,包括数据的发送和接收。适配器接口通常包括write(发送数据)、read(读取数据)和error(错误处理)等函数。

2. 作业项管理

模块使用链式队列管理作业项,包括空闲链表和就绪链表。每个作业项代表一个待执行的AT命令,包括命令类型、状态、参数等信息。命令执行完毕后,作业项将被回收至空闲链表,以便复用。

3. URC处理

URC是AT设备主动发送的非请求结果码,用于通知控制系统某些事件的发生。模块通过维护一个URC表,将收到的URC与表中的匹配项进行比对,并执行相应的处理函数。

应用示例

以下是一个使用AT命令通信解析模块发送短信的示例:

c

// 定义URC表和缓冲区

static char urc_buf[128];

utc_item_t utc_tbl[] = {{"+CMGS:", sms_sent_handler}};

// 初始化AT控制器和通信适配器

at_obj_t at;

const at_adapter_t adap = {

.urc_buf = urc_buf,

.urc_bufsize = sizeof(urc_buf),

.utc_tbl = utc_tbl,

.urc_tbl_count = sizeof(utc_tbl) / sizeof(utc_item_t),

.write = uart_write,

.read = uart_read,

// 其他接口...

};

// 发送短信

static bool send_sms(at_obj_t *at, const char *phone_number, const char *message) {

char cmd[128];

snprintf(cmd, sizeof(cmd), "AT+CMGS=\"%s\"", phone_number);

at_respond_t r = {"OK", NULL, 0, 3000};

if (at_do_cmd(at, &r, cmd) != AT_RET_OK) {

return false;

}

// 发送短信内容...

return true;

}

在这个示例中,我们首先定义了URC表和缓冲区,并初始化了AT控制器和通信适配器。随后,我们实现了一个send_sms函数,该函数利用AT命令通信解析模块来发送短信。以下是send_sms函数以及后续处理短信发送内容的完整示例:

c

#include <string.h>

#include <stdbool.h>

#include "at_parser.h" // 假设这是AT命令解析模块的头文件

// 假设的UART读写函数声明

extern int uart_write(const void *data, size_t size);

extern int uart_read(void *data, size_t size, unsigned int timeout_ms);

// URC处理函数,处理短信发送成功的URC

static void sms_sent_handler(const char *urc) {

// 这里可以添加处理逻辑,比如打印日志、更新状态等

printf("SMS sent successfully: %s\n", urc);

}

// 发送短信的函数

static bool send_sms(at_obj_t *at, const char *phone_number, const char *message) {

char cmd[128];

char final_cmd[256];

at_respond_t r = {"OK", NULL, 0, 3000}; // 等待"OK"响应,超时3秒

// 第一步:发送AT+CMGS命令以开始发送短信

snprintf(cmd, sizeof(cmd), "AT+CMGS=\"%s\"", phone_number);

if (at_send_command(at, &r, cmd) != AT_RET_OK) {

printf("Failed to start SMS sending\n");

return false;

}

// 第二步:发送短信内容(注意:内容后需要添加CTRL+Z(0x1A)作为结束符)

snprintf(final_cmd, sizeof(final_cmd), "%s%c", message, 0x1A);

if (uart_write(final_cmd, strlen(final_cmd)) < 0) {

printf("Failed to write SMS content\n");

return false;

}

// 注意:在某些情况下,可能不需要再次等待响应,因为设备可能会直接通过URC通知发送结果

// 但为了示例完整性,这里假设我们还需要检查是否有错误发生(虽然这不是标准的做法)

// (可选)等待可能的错误响应或其他URC

// 这里可能需要一个更复杂的机制来区分不同类型的URC,但在此示例中我们省略了

// 假设一切正常,返回成功

return true;

}

int main(void) {

at_obj_t at;

// 初始化AT控制器(这里省略了具体的初始化代码,因为它依赖于具体的实现)

// ...

// 发送短信

const char *phone_number = "+1234567890";

const char *message = "Hello, this is a test SMS!";

if (send_sms(&at, phone_number, message)) {

printf("SMS sent successfully\n");

} else {

printf("Failed to send SMS\n");

}

// 清理资源,关闭连接等(此处省略)

// ...

return 0;

}

// 注意:上述代码中的at_send_command是一个假设的函数,实际中你可能需要根据

// AT命令解析模块的具体实现来调用相应的函数(如示例中的at_do_cmd或自定义的发送函数)。

// 此外,UART的读写函数也需要根据你的硬件平台和驱动进行相应的实现。

在这个完整的示例中,send_sms函数首先通过AT命令AT+CMGS开始发送短信,并等待设备返回"OK"响应。然后,它将短信内容(后面跟着CTRL+Z作为结束符)通过UART发送出去。请注意,由于短信发送的结果通常是通过URC(如"+CMGS: <index>"后跟发送状态)来通知的,因此send_sms函数在发送完短信内容后并没有立即等待特定的响应。在实际应用中,你可能需要实现一个机制来监听和处理这些URC,以便了解短信是否成功发送。

此外,示例中的at_send_command函数是一个假设的函数,用于简化示例。在实际应用中,你应该使用AT命令解析模块提供的实际函数(如at_do_cmd或类似的函数)来发送命令并处理响应。同时,UART的读写函数也需要根据你的硬件平台和驱动进行相应的实现。

相关推荐