Lwip之TCP服务端示例记录(1对多)

前言

  1. 实现多个客户端同时连接
  2. 初步代码结构已经实现完成(通过轮训的方式)
//
// Created by shchl on 2024/3/8.
//
#if  1

#include <string.h>
#include "lwip/api.h"
#include "FreeRTOS.h"
#include "task.h"
#include "usart.h"
#include "lwip_comm.h"

#define CLIENT_LIST_SIZE 3
typedef struct {
    u8_t stat;/*0表示未连接,1表示已建立连接*/
    struct netconn *conn;
    struct netbuf *net_buf;
    ip_addr_t ip; /*地址*/
    u16_t port;/*端口*/
#define BUF_SIZE 1024
    u8_t buf[BUF_SIZE];/*存放数据*/

} client_pcb;/*客户端控制块*/
static TaskHandle_t tcp_server; /*任务句柄*/
static TaskHandle_t tcp_client; /*任务句柄*/
static void tcp_server_entity(void *args);

client_pcb client_list[CLIENT_LIST_SIZE]; /*客户端列表*/

static void tcp_client_request_entity(void *args);

void netconn_tcp_server_create_thread() {
    /*初始化*/
    memset(client_list, 0, sizeof(client_list));


    xTaskCreate(
            (TaskFunction_t) tcp_server_entity,
            "tcp_server",
            256,
            NULL,
            10,
            &tcp_server
    );

    xTaskCreate(
            (TaskFunction_t) tcp_client_request_entity,
            "tcp_client",
            256,
            NULL,
            9,
            &tcp_client
    );

}

#define TCP_SERVER_RX_BUFSIZE 1500
static struct netconn *conn;
static struct netconn *client_conn;/*客户端连接*/
static uint8_t remot_addr[4] = {0}; /*远端ip数组*/
static u16_t port;/*远端端口*/
static ip_addr_t ipaddr;/*远端ip*/
static u8_t tcp_server_recvbuf[TCP_SERVER_RX_BUFSIZE] = {0};


static void tcp_server_entity(void *args) {
    err_t err;
    conn = netconn_new(NETCONN_TCP);  //创建一个TCP链接
    netconn_bind(conn, IP_ADDR_ANY, 8080);  //绑定端口 8号端口
    netconn_listen(conn);        //进入监听模式
    conn->recv_timeout = 10;    //禁止阻塞线程 等待10ms
    struct netconn *cli_con;

    while (1) {

        begin_listen:
        err = netconn_accept(conn, &cli_con);  //接收连接请求
        if (err == ERR_OK) {

            cli_con->recv_timeout = 10;
            for (int i = 0; i < CLIENT_LIST_SIZE; ++i) {
                if (client_list[i].stat == 0) {

                    netconn_getaddr(cli_con, &client_list[i].ip, &client_list[i].port, 0); //获取远端IP地址和端口号
                    remot_addr[3] = (uint8_t) (client_list[i].ip.addr >> 24);
                    remot_addr[2] = (uint8_t) (client_list[i].ip.addr >> 16);
                    remot_addr[1] = (uint8_t) (client_list[i].ip.addr >> 8);
                    remot_addr[0] = (uint8_t) (client_list[i].ip.addr);
                    printf("主机%d.%d.%d.%d连接上服务器,主机端口号为:%d\r\n",
                           remot_addr[0], remot_addr[1], remot_addr[2], remot_addr[3], client_list[i].port);
                    client_list[i].stat = 1;
                    client_list[i].conn = cli_con;

                    goto begin_listen;
                }
            }
            /*释放链接*/
            netconn_close(cli_con);
            netconn_delete(cli_con);

        }
        vTaskDelay(10);
    }
}


static void tcp_client_request_entity(void *args) {

    client_pcb *p_client;
    err_t err;
    u32_t data_len;
    while (1) {

        for (int i = 0; i < CLIENT_LIST_SIZE; ++i) {
            p_client = &client_list[i];
            if (p_client->stat == 1) {
                err = netconn_recv(p_client->conn, &p_client->net_buf);
                switch (err) {
                    case ERR_OK: {
                        portDISABLE_INTERRUPTS();
                        memset(p_client->buf, 0, BUF_SIZE);  //数据接收缓冲区清零
                        for (struct pbuf *q = p_client->net_buf->p; q != NULL; q = q->next)  //遍历完整个pbuf链表
                        {
                            if (q->len > (BUF_SIZE - data_len)) {
                                /*数据超出缓冲区大小*/
                                memcpy(p_client->buf + data_len, q->payload, BUF_SIZE - data_len);
                                break;
                            } else {
                                memcpy(p_client->buf + data_len, q->payload, q->len);
                            }
                            data_len += q->len;
                            if (data_len > BUF_SIZE) break;
                        }
                        portENABLE_INTERRUPTS();
                        // todo 应用逻辑处理
                        netconn_write(p_client->conn, p_client->buf, data_len, NETCONN_COPY);
                        data_len = 0;  //复制完成后data_len要清零。
                        netbuf_delete(p_client->net_buf);
                        break;
                    }
                    case ERR_CLSD:
                    case ERR_RST: {
                        goto release_conn_tag;
                    }
                    default:
                        if (g_lwipdev.link_status == LWIP_LINK_OFF) { /*判断网线是否连接正常*/
                            printf("物理连线出现问题\r\n");
                            goto release_conn_tag;
                        }

                }
                continue;
                release_conn_tag: /*释放链接*/
                {
                    netconn_close(p_client->conn);
                    netconn_delete(p_client->conn);
                    p_client->stat = 0;
                }
            }
            vTaskDelay(10);
        }


    }


}

#endif

测试结果

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/440781.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

寻找数组的中心索引

给你一个整数数组 nums &#xff0c;请计算数组的 中心下标 。 数组 中心下标 是数组的一个下标&#xff0c;其左侧所有元素相加的和等于右侧所有元素相加的和。 如果中心下标位于数组最左端&#xff0c;那么左侧数之和视为 0 &#xff0c;因为在下标的左侧不存在元素。这一点…

斐讯N1 刷coreelec 笔记

1.下载恩山的镜像 下载好后不需要刷优盘 这个很方便&#xff0c;可以勾选擦除flash &#xff08;如果第一次装&#xff09; 升级可以不用勾选 详细使用参考恩山大佬的描述 2.下载插件 想装openwrt 发现镜像里面 coreelec-addons 挂了&#xff0c;研究了好长时间可以 去githu…

一文扫荡,12个可视化图表js库,收藏备用。

hello&#xff0c;我是贝格前端工场&#xff0c;可视化图表在web前端开发中经常碰到&#xff0c;是不是很疑惑这些炫酷的图表是怎么实现的&#xff0c;其实是通过js库开发的&#xff0c;本文带来12个javascript库的介绍&#xff0c;欢迎关注我&#xff0c;阅读精彩内容。 一、什…

2024 RubyMine 激活,分享几个RubyMine 激活的方案

文章目录 RubyMine 公司简介我这边使用RubyMine 的理由RubyMine 2023.3 最新变化AI Assistant 正式版对 AI 生成名称建议的支持改进了 Ruby 上下文单元测试生成 RailsRails 应用程序和引擎的自定义路径Rails 路径的自动导入对存储在默认位置之外的模型、控制器和邮件器的代码洞…

Express学习(三)

Express中间件 中间件的概念 什么是中间件 中间件&#xff0c;特指业务流程的中间处理环节。Express中间件的调用流程 当一个请求到达Express的服务器之后&#xff0c;可以连续调用多个中间件&#xff0c;从而对这次请求进行预处理。类似于下图所示 Express中间件的格式 Expr…

C++进阶之路---继承(二)

顾得泉&#xff1a;个人主页 个人专栏&#xff1a;《Linux操作系统》 《C从入门到精通》 《LeedCode刷题》 键盘敲烂&#xff0c;年薪百万&#xff01; 一、继承与友元 友元关系不能继承&#xff0c;也就是说基类友元不能访问子类私有和保护成员。 class Student; class Per…

[C语言]——分支和循环(4)

目录 一.随机数生成 1.rand 2.srand 3.time 4.设置随机数的范围 猜数字游戏实现 写⼀个猜数字游戏 游戏要求&#xff1a; &#xff08;1&#xff09;电脑自动生成1~100的随机数 &#xff08;2&#xff09;玩家猜数字&#xff0c;猜数字的过程中&#xff0c;根据猜测数据的⼤…

【LaTeX】行内代码块、行间代码块的插入以及高亮(懒人版)

文章目录 思路和优点基本框架行内代码行间代码pythoncpp 所支持的语言所支持的代码风格 思路和优点 思路是listingsminted包&#xff0c; 一个负责插入代码一个负责高亮代码 这种方法显著的优点在于&#xff1a;完全不需要自定义代码风格 使用其他方法时&#xff0c;你定义好…

嵌入式学习36-TCP要点及http协议

TCP发送文件的粘包问题 1. 例&#xff1a; 发端 1.flv-------->收端 1.flv csfga 2.解决 1. sleep&#xff08;1&#xff09; 延时发送 2.自…

httprunner用例结构(前后置)

说明&#xff1a;httprunner 结合 pytest 的前后置方式 1. 用例级别前后置 1.1. setup teardown class TestCaseRefTestcase(HttpRunner):# 用例级别前后置def setup(self):logger.warning("------用例级别前置")def teardown(self):logger.warning("------用…

用一个 Python 脚本实现依次运行其他多个带 argparse 命令行参数的 .py 文件

&#x1f349; CSDN 叶庭云&#xff1a;https://yetingyun.blog.csdn.net/ 问题描述&#xff1a;在 Windows 环境中&#xff0c;您希望通过一个 Python 脚本来实现特定的自动化任务&#xff0c;该任务需要依次运行其他多个带 argparse 命令行参数的 .py 文件。您希望找到一种简…

【jenkins】简单安装及配置(Windows环境)

前言 jenkins是一款跨平台的持续集成和持续交付、基于Java开发的开源软件&#xff0c;提供任务构建、持续集成监控的功能&#xff0c;可以使开发测试人员更方便的构建软件项目&#xff0c; 提高工作效率。Windows平台下&#xff0c;一般安装方法有2种&#xff1a;安装程序安装…

unicloud where 使用

where介绍 在uniCloud中&#xff0c;WHERE是一个用于指定查询条件的关键字。它允许用户根据特定的条件来筛选和查询云数据库中的数据。WHERE语句的基本语法格式是WHERE condition&#xff0c;其中condition表示查询条件&#xff0c;可以是一个或多个逻辑表达式组成的条件。 在…

第七十九天 WAF攻防-漏洞发现协议代理池GobyAWVSXray

第79天 WAF攻防-漏洞发现&协议&代理池&Goby&AWVS&Xray 知识点&#xff1a; 1、Http/s&Sock5协议 2、Awvs Xray&Goby代理 3、Pxoxifier进程代理使用 4、Safedog&BT&Aliyun防护 演示案例&#xff1a; Awws漏扫-Sadedog-白名单-内置 Awws漏…

使用Apache Kafka的Golang实践指南

您是否在寻找构建可扩展、高性能应用程序的方法&#xff0c;这些应用程序可以实时处理流数据&#xff1f;如果是的话&#xff0c;结合使用Apache Kafka和Golang是一个很好的选择。Golang的轻量级线程非常适合编写类似Kafka生产者和消费者的并发网络应用程序。它的内置并发原语&…

动态规划DP之背包问题3---多重背包问题

目录 DP分析&#xff1a; 优化&#xff1a; 二进制优化 例题&#xff1a; 01背包是每个物品只有一个&#xff0c;完全背包问题是每个物品有无限个。 那么多重背包问题就是 每个物品有有限个。 有 N 种物品和一个容量是 V 的背包。 第 i 种物品最多有 si 件&#xff0c;每件体…

Mysql学习笔记之事务详解(读未提交、读以提交、可重复读、串行化读)

在这个博主的基础上&#xff0c;增加两种情况的对比&#xff1a;https://blog.csdn.net/llllllkkkkkooooo/article/details/108068919 可重复读中幻读现象&#xff08;未使用MVCC&#xff09; 设置可重复读的隔离级别 set global transaction isolation level repeatable read…

代码随想录算法训练营第day40|343. 整数拆分 、 96.不同的二叉搜索树

a.343. 整数拆分 题目链接 给定一个正整数 n &#xff0c;将其拆分为 k 个 正整数 的和&#xff08; k > 2 &#xff09;&#xff0c;并使这些整数的乘积最大化。 返回 你可以获得的最大乘积 。 示例 1: 输入: n 2 输出: 1 解释: 2 1 1, 1 1 1。 示例 2: 输入: …

掼蛋的牌型与规律(上篇)

掼蛋是一项配合类的棋牌竞技游戏&#xff0c;掼蛋的最大魅力以及最集中的特点在于变化&#xff0c;在于组牌的变数。有的掼蛋新手往往先把牌配死&#xff0c;并且直接决定好出牌计划&#xff0c;然后守株待兔。掼蛋的取胜之道在于静态组合加上动态变化。本文主要介绍一下掼蛋的…

python基础篇--学习记录2

1.深浅拷贝 l1 ["张大仙","徐凤年",["李淳刚","邓太阿"]] # 变量名对应的就是内存地址,这里就是将l1的内存地址给了l2 # 现在两个变量指向同一个内存地址,l1变化l2也会变化 l2 l1 现在的需求是l2是l1的拷贝版本,但是两者是完全分割…