Danger-alarm/SOFTWARE/Source/WIFI.c

252 lines
8.1 KiB
C
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#include "WIFI.h"
#if 0
static void dump_bytes(const uint8_t *bptr, uint32_t len)
{
unsigned int i = 0;
printf("dump_bytes %d", len);
for (i = 0; i < len;) {
if ((i & 0x0f) == 0) {
printf("\n");
} else if ((i & 0x07) == 0) {
printf(" ");
}
printf("%02x ", bptr[i++]);
}
printf("\n");
}
#define DUMP_BYTES dump_bytes // 调试用的字节转储函数
#else
#define DUMP_BYTES(A,B)
#endif
typedef struct TCP_CLIENT_T_ {
struct tcp_pcb *tcp_pcb; // TCP控制块指针
ip_addr_t remote_addr; // 远程地址
uint8_t buffer[BUF_SIZE]; // 接收缓冲区
int buffer_len; // 缓冲区当前长度
int sent_len; // 已发送的数据长度
bool complete; // 是否完成测试
int run_count; // 运行次数
bool connected; // 是否已连接
} TCP_CLIENT_T;
static err_t tcp_client_close(void *arg)
{
TCP_CLIENT_T *state = (TCP_CLIENT_T *) arg;
err_t err = ERR_OK;
if (state->tcp_pcb != NULL) {
tcp_arg(state->tcp_pcb, NULL); // 清除回调参数
tcp_poll(state->tcp_pcb, NULL, 0); // 清除轮询回调
tcp_sent(state->tcp_pcb, NULL); // 清除发送回调
tcp_recv(state->tcp_pcb, NULL); // 清除接收回调
tcp_err(state->tcp_pcb, NULL); // 清除错误回调
err = tcp_close(state->tcp_pcb); // 关闭TCP连接
if (err != ERR_OK) {
DEBUG_printf("close failed %d, calling abort\n", err); // 如果关闭失败调用abort
tcp_abort(state->tcp_pcb);
err = ERR_ABRT;
}
state->tcp_pcb = NULL; // 清除TCP控制块指针
}
return err;
}
// 操作结果回调
static err_t tcp_result(void *arg, int status)
{
TCP_CLIENT_T *state = (TCP_CLIENT_T *) arg;
if (status == 0) {
DEBUG_printf("test success\n"); // 测试成功
} else {
DEBUG_printf("test failed %d\n", status); // 测试失败
}
state->complete = true; // 标记测试完成
return tcp_client_close(arg); // 关闭连接
}
// 轮询回调
static err_t tcp_client_poll(void *arg, struct tcp_pcb *tpcb)
{
DEBUG_printf("tcp_client_poll\n");
return tcp_result(arg, -1); // 无响应视为错误
}
// 错误处理回调
static void tcp_client_err(void *arg, err_t err)
{
if (err != ERR_ABRT) {
DEBUG_printf("tcp_client_err %d\n", err); // 打印错误信息
tcp_result(arg, err);
}
}
// 连接建立回调
static err_t tcp_client_connected(void *arg, struct tcp_pcb *tpcb, err_t err)
{
TCP_CLIENT_T *state = (TCP_CLIENT_T *) arg;
if (err != ERR_OK) {
printf("connect failed %d\n", err); // 连接失败
return tcp_result(arg, err);
}
state->connected = true; // 标记已连接
DEBUG_printf("连接成功,准备发送消息!!!\n"); // 连接成功,准备发送消息
// 发送字符串到服务器
err = tcp_write(tpcb, state->buffer, state->buffer_len, TCP_WRITE_FLAG_COPY);
if (err != ERR_OK) {
DEBUG_printf("发送失败: %d\n", err); // 发送失败
return tcp_result(arg, -1);
}
DEBUG_printf("发送数据: %s\n", state->buffer); // 打印发送的消息
memset(state->buffer, 0, state->buffer_len);
state->buffer_len = 0;
return ERR_OK;
}
err_t tcp_client_recv(void *arg, struct tcp_pcb *tpcb, struct pbuf *p, err_t err)
{
TCP_CLIENT_T *state = (TCP_CLIENT_T *) arg;
if (!p) {
DEBUG_printf("接收数据失败p 为 NULL\n"); // 添加调试信息
return tcp_result(arg, -1);
}
// this method is callback from lwIP, so cyw43_arch_lwip_begin is not required, however you
// can use this method to cause an assertion in debug mode, if this method is called when
// cyw43_arch_lwip_begin IS needed
cyw43_arch_lwip_check();
if (p->tot_len > 0) {
DEBUG_printf("recv %d err %d\n", p->tot_len, err);
for (struct pbuf * q = p; q != NULL; q = q->next) {
DUMP_BYTES(q->payload, q->len);
}
// Receive the buffer
const uint16_t buffer_left = BUF_SIZE - state->buffer_len;
state->buffer_len += pbuf_copy_partial(p, state->buffer + state->buffer_len, p->tot_len > buffer_left ? buffer_left : p->tot_len, 0);
tcp_recved(tpcb, p->tot_len);
}
// 打印当前缓冲区内容
printf("接收数据: %s", state->buffer);
pbuf_free(p);
memset(state->buffer, 0, state->buffer_len);
state->buffer_len = 0;
return ERR_OK;
}
// 发送数据回调
static err_t tcp_client_sent(void *arg, struct tcp_pcb *tpcb, u16_t len)
{
TCP_CLIENT_T *state = (TCP_CLIENT_T *) arg;
DEBUG_printf("tcp_client_sent %u\n", len); // 打印发送的数据长度
state->sent_len += len; // 更新已发送的数据长度
if (state->sent_len >= BUF_SIZE) {
state->run_count++; // 增加运行次数
if (state->run_count >= TEST_ITERATIONS) {
tcp_result(arg, 0); // 达到迭代次数,测试成功
return ERR_OK;
}
// 准备接收新的数据缓冲区
state->buffer_len = 0;
state->sent_len = 0;
DEBUG_printf("Waiting for buffer from server\n");
}
return ERR_OK;
}
// 打开TCP连接
static bool tcp_client_open(void *arg)
{
TCP_CLIENT_T *state = (TCP_CLIENT_T *) arg;
DEBUG_printf("Connecting to %s port %u\n", ip4addr_ntoa(&state->remote_addr), TCP_PORT);
state->tcp_pcb = tcp_new_ip_type(IP_GET_TYPE(&state->remote_addr)); // 创建TCP控制块
if (!state->tcp_pcb) {
DEBUG_printf("failed to create pcb\n"); // 创建失败
return false;
}
tcp_arg(state->tcp_pcb, state); // 设置回调参数
tcp_poll(state->tcp_pcb, tcp_client_poll, POLL_TIME_S * 2); // 设置轮询回调
tcp_sent(state->tcp_pcb, tcp_client_sent); // 设置发送回调
tcp_recv(state->tcp_pcb, tcp_client_recv); // 设置接收回调
tcp_err(state->tcp_pcb, tcp_client_err); // 设置错误回调
//state->buffer_len = 0; // 初始化缓冲区长度
// 使用cyw43_arch_lwip_begin/end确保正确锁定
cyw43_arch_lwip_begin();
err_t err = tcp_connect(state->tcp_pcb, &state->remote_addr, TCP_PORT, tcp_client_connected); // 尝试连接
cyw43_arch_lwip_end();
return err == ERR_OK;
}
// 初始化TCP客户端
static TCP_CLIENT_T *tcp_client_init(void)
{
TCP_CLIENT_T *state = calloc(1, sizeof(TCP_CLIENT_T)); // 分配内存
if (!state) {
DEBUG_printf("failed to allocate state\n"); // 分配失败
return NULL;
}
// 如果 TEST_TCP_SERVER_IP 是 IP 地址,直接解析为 IP 地址
if (ip4addr_aton(TEST_TCP_SERVER_IP, &state->remote_addr) == 0) {
// 如果解析失败,尝试作为域名解析
DEBUG_printf("Failed to parse IP address, trying DNS resolution\n");
err_t err = dns_gethostbyname(TEST_TCP_SERVER_IP, &state->remote_addr, NULL, NULL);
if (err != ERR_OK) {
DEBUG_printf("DNS resolution failed\n");
free(state);
return NULL;
}
}
return state;
}
// 运行TCP客户端测试
void run_tcp_client_test(char *s)
{
TCP_CLIENT_T *state = tcp_client_init(); // 初始化客户端
if (!state) {
return;
}
if (!tcp_client_open(state)) { // 打开连接
tcp_result(state, -1); // 连接失败
return;
}
strcpy(state->buffer, s);
state->buffer_len = strlen(state->buffer);
while (!state->complete) { // 循环等待测试完成
#if PICO_CYW43_ARCH_POLL
// 如果使用pico_cyw43_arch_poll需要定期调用cyw43_arch_poll
cyw43_arch_poll();
// 可以选择休眠直到有工作需要做
cyw43_arch_wait_for_work_until(make_timeout_time_ms(1000));
#else
//cyw43_arch_poll();
// 可以选择休眠直到有工作需要做
//cyw43_arch_wait_for_work_until(make_timeout_time_ms(1000));
// 如果不使用pico_cyw43_arch_poll可以通过中断在后台处理
//sleep_ms(1000); // 示例中的阻塞操作
#endif
}
free(state); // 释放内存
}