W6100-EVB-PICO 做TCP Server进行回环测试(六)

前言

        上一章我们用W6100-EVB-PICO开发板做TCP 客户端连接服务器进行数据回环测试,那么本章将用开发板做TCP服务器来进行数据回环测试。

TCP是什么?什么是TCP Server?能干什么?

        TCP (Transmission Control Protocol) 是一种面向连接的、可靠的、基于字节流的传输协议,用于在计算机网络上传输数据。TCP Server是指TCP网络服务的服务器端连接,用于接收客户端的连接请求并建立连接,实现网络数据的交互。
        TCP Server的主要作用是监听客户端的连接请求,并建立与管理连接,实现数据的可靠传输。通过TCPServer,多个客户端可以同时与服务器建立连接,实现数据的多点传输。
        在TCP Server中,服务器程序需要指定监听的端口号,并使用TCP协议与客户端建立连接。一旦有客户端连接进来,服务器程序就会为每个客户端建立一个单独的连接,并通过数据流对象 (NetworkStream) 与客广端进行数据交互。
        因此,TCP Server可以帮助设备实现多点数据交互,是设备联网通信的重要方式之一。在工业自动化、物联网、智能家居等应用中,TCP Server被广泛使用。

连接方式

使开发板和我们的电脑处于同一网段:

  • 开发板(设备)通过交叉线直连主机(PC)
  •  开发板和主机都接在路由器LAN口

测试工具

  • 网络调试工具(任意)
  • wireshark抓包工具

回环测试

1.相关代码

如下所示,tcp服务端的回环测试函数需要我们传入三个参数:socket端口号、收发数据的缓存和端口,与做tcp客户端实现思路一样(可参考上一章内容),即通过Switch状态机轮询socket状态进行相应处理,不同的是在初始化socket端口后不再是连接服务器,而是开启端口监听。

因为W6100这款以太网芯片支持IPv6,因此在回环模式上有着不同选择,相应地处理上也会根据模式分别进行处理,这里选择IPv4模式进行回环测试。

int32_t loopback_tcps(uint8_t sn, uint8_t* buf, uint16_t port, uint8_t loopback_mode)
{int32_t ret;datasize_t sentsize=0;int8_t status,inter;uint8_t tmp = 0;datasize_t received_size;uint8_t arg_tmp8;uint8_t* mode_msg;uint8_t dip[16];uint16_t dport;if(loopback_mode == AS_IPV4){mode_msg = msg_v4;}else if(loopback_mode == AS_IPV6){mode_msg = msg_v6;}else{mode_msg = msg_dual;}#ifdef _LOOPBACK_DEBUG_uint8_t dst_ip[16], ext_status;uint16_t dst_port;#endifgetsockopt(sn, SO_STATUS, &status);switch(status){case SOCK_ESTABLISHED :ctlsocket(sn,CS_GET_INTERRUPT,&inter);if(inter & Sn_IR_CON){#ifdef _LOOPBACK_DEBUG_getsockopt(sn,SO_DESTIP,dst_ip);getsockopt(sn,SO_EXTSTATUS, &ext_status);if(ext_status & TCPSOCK_MODE){//IPv6printf("%d:Peer IP : %04X:%04X", sn, ((uint16_t)dst_ip[0] << 8) | ((uint16_t)dst_ip[1]),((uint16_t)dst_ip[2] << 8) | ((uint16_t)dst_ip[3]));printf(":%04X:%04X", ((uint16_t)dst_ip[4] << 8) | ((uint16_t)dst_ip[5]),((uint16_t)dst_ip[6] << 8) | ((uint16_t)dst_ip[7]));printf(":%04X:%04X", ((uint16_t)dst_ip[8] << 8) | ((uint16_t)dst_ip[9]),((uint16_t)dst_ip[10] << 8) | ((uint16_t)dst_ip[11]));printf(":%04X:%04X, ", ((uint16_t)dst_ip[12] << 8) | ((uint16_t)dst_ip[13]),((uint16_t)dst_ip[14] << 8) | ((uint16_t)dst_ip[15]));}else{//IPv4//getSn_DIPR(sn,dst_ip);printf("%d:Peer IP : %.3d.%.3d.%.3d.%.3d, ",sn, dst_ip[0], dst_ip[1], dst_ip[2], dst_ip[3]);}getsockopt(sn,SO_DESTPORT,&dst_port);printf("Peer Port : %d\r\n", dst_port);#endifarg_tmp8 = Sn_IR_CON;ctlsocket(sn,CS_CLR_INTERRUPT,&arg_tmp8);}getsockopt(sn,SO_RECVBUF,&received_size);if(received_size > 0){if(received_size > DATA_BUF_SIZE) received_size = DATA_BUF_SIZE;ret = recv(sn, buf, received_size);getsockopt(sn,SO_DESTIP,dip);getsockopt(sn,SO_DESTPORT,&dport);printf("from the client with ip [%d.%d.%d.%d] - port [%d].\n",dip[0],dip[1],dip[2],dip[3],dport);printf("recv: %s",buf);if(ret <= 0) return ret;      // check SOCKERR_BUSY & SOCKERR_XXX. For showing the occurrence of SOCKERR_BUSY.received_size = (uint16_t) ret;sentsize = 0;while(received_size != sentsize){ret = send(sn, buf+sentsize, received_size-sentsize);if(ret < 0){close(sn);return ret;}sentsize += ret; // Don't care SOCKERR_BUSY, because it is zero.}}break;case SOCK_CLOSE_WAIT :#ifdef _LOOPBACK_DEBUG_printf("%d:CloseWait\r\n",sn);#endifgetsockopt(sn, SO_RECVBUF, &received_size);if(received_size > 0) // Don't need to check SOCKERR_BUSY because it doesn't not occur.{if(received_size > DATA_BUF_SIZE) received_size = DATA_BUF_SIZE;ret = recv(sn, buf, received_size);if(ret <= 0) return ret;      // check SOCKERR_BUSY & SOCKERR_XXX. For showing the occurrence of SOCKERR_BUSY.received_size = (uint16_t) ret;sentsize = 0;while(received_size != sentsize){ret = send(sn, buf+sentsize, received_size-sentsize);if(ret < 0){close(sn);return ret;}sentsize += ret; // Don't care SOCKERR_BUSY, because it is zero.}}if((ret = disconnect(sn)) != SOCK_OK) return ret;#ifdef _LOOPBACK_DEBUG_printf("%d:Socket Closed\r\n", sn);#endifbreak;case SOCK_INIT :if( (ret = listen(sn)) != SOCK_OK) return ret;#ifdef _LOOPBACK_DEBUG_printf("%d:Listen, TCP server loopback, port [%d] as %s\r\n", sn, port, mode_msg);#endifprintf("%d:Listen, TCP server loopback, port [%d] as %s\r\n", sn, port, mode_msg);break;case SOCK_CLOSED:#ifdef _LOOPBACK_DEBUG_printf("%d:TCP server loopback start\r\n",sn);#endifswitch(loopback_mode){case AS_IPV4:tmp = socket(sn, Sn_MR_TCP4, port, SOCK_IO_NONBLOCK);break;case AS_IPV6:tmp = socket(sn, Sn_MR_TCP6, port, SOCK_IO_NONBLOCK);break;case AS_IPDUAL:tmp = socket(sn, Sn_MR_TCPD, port, SOCK_IO_NONBLOCK);break;default:break;}if(tmp != sn)    /* reinitialize the socket */{#ifdef _LOOPBACK_DEBUG_printf("%d : Fail to create socket.\r\n",sn);#endifreturn SOCKERR_SOCKNUM;}#ifdef _LOOPBACK_DEBUG_printf("%d:Socket opened[%d]\r\n",sn, getSn_SR(sn));sock_state[sn] = 1;#endifbreak;default:break;}return 1;
}

主函数就比较简单,在此之前我们先声明socket端口号和所用最大的缓存大小,不做分片处理默认为2KB;然后初始化网络信息、目标IP地址和目标端口,最后在while循环里调用loopback_tcps并传入相应参数即可。

注意:这里的本地端口选择尽量避免使用特殊端口,这里使用8080;如下所示。

#define SOCKET_ID 0
#define ETHERNET_BUF_MAX_SIZE (1024 * 2)void network_init(void);wiz_NetInfo net_info = {.mac = {0x00, 0x08, 0xdc, 0x16, 0xed, 0x2e},.ip = {192, 168, 1, 10},.sn = {255, 255, 255, 0},.gw = {192, 168, 1, 1},.dns = {8, 8, 8, 8},.ipmode = NETINFO_STATIC_V4};
wiz_NetInfo get_info;
static uint8_t ethernet_buf[ETHERNET_BUF_MAX_SIZE] = {0,};
static uint16_t local_port = 8080;int main()                                                          
{   stdio_init_all();sleep_ms(2000);network_init();while(true){loopback_tcps(SOCKET_ID, ethernet_buf, local_port, AS_IPV4);sleep_ms(500);}}void network_init(void)
{uint8_t temp;wizchip_initialize();printf("W6100 tcp server example.\r\n");sleep_ms(2000);/* Determine the network lock register status */if(!ctlwizchip(SYS_NET_LOCK, &temp)){   printf("unlock.\n");NETUNLOCK();}wizchip_setnetinfo(&net_info);print_net_info(&get_info);sleep_ms(2000);   
}

2.测试现象

我们编译烧录后,打开串行监视器,可以看到我们开发板通过串口回显的网络配置信息,然后我们打开网络调试工具,配置为TCP Client模式;远程IP地址和远程端口为我们开发板的本地IP和端口,然后点击连接,并发送数据测试;可以看到串口打印的信息,我们电脑作为客户端成功连接并收到开发板回传的数据。

我们也可以在打开wireshark抓包工具,输入命令<ip.addr == 192.168.1.10 and tcp>过滤数据包(IP地址改成自己电脑的IP或者开发板的IP地址即可);我这里先关闭网络调试助手,然后又打开,接着发送0~9十个阿拉伯数字,可以通过抓包工具十分清楚明了的看到具体交互过程,如下图所示。

 相关链接:

本章相关例程链接https://gitee.com/wiznet-hk/example-of-w6100-evb-pico.gitwireshark抓包工具下载链接https://www.wireshark.org/download.html

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

zabbix监控安装部署

目录 一、环境 二、配置 1.配置yum源&#xff0c;这里用的清华的 2.过滤一下安装包&#xff0c;查看依赖包 安装依赖包 3.配置数据库 开机自启 创建数据库 创建用户 授权 导入数据到数据库 查看zabbix数据库有没有表和数据 4.修改zabbix配置文件 1.修改zabbix配置…

去趋势化一个心电图信号、信号功率谱、低通IIR滤波器并平滑信号、对滤波器引起的延迟进行补偿研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

SPM实现framework自动管理和分发

一、前言 Swift Package Manager (SPM) 是苹果官方提供的用于管理 Swift 项目的依赖关系和构建过程的工具。它是一个集成在 Swift 编程语言中的包管理器&#xff0c;用于解决在开发过程中管理和构建包依赖项的需求。 那么如何使用SPM管理和分发Objective C编写的二进制库呢&a…

不同路径 II——力扣63

class Solution {public:int uniquePathsWithObstacles(vector<vector<int>>& obstacleGrid) {int n=

【APITable】教程:创建并运行一个自建小程序

1.进入APITable&#xff0c;在想要创建小程序的看板页面点击右上角的【小程序】&#xff0c;进入小程序编辑页面。 2.创建一个新的小程序区。 点击【 添加小程序】 点击创建小程序&#xff0c;选择模板&#xff0c;输入名字。 3.确定后进入小程序部署引导页面。 4.打开Xshell 7…

初识鸿蒙跨平台开发框架ArkUI-X

HarmonyOS是一款面向万物互联时代的、全新的分布式操作系统。在传统的单设备系统能力基础上&#xff0c;HarmonyOS提出了基于同一套系统能力、适配多种终端形态的分布式理念&#xff0c;能够支持手机、平板、智能穿戴、智慧屏、车机等多种终端设备&#xff0c;提供全场景&#…

99. for循环练习题-3种方式输出0-9

【目录】 文章目录 99. for循环练习题-3种方式输出0-91. for循环和while循环的区别2. 输出 0~(n-1)的数字2.1 基础代码2.2 自定义函数代码2.3 异常处理语句代码 【正文】 99. for循环练习题-3种方式输出0-9 1. for循环和while循环的区别 for循环和while循环都用于重复执行特定…

idea模板的使用(配置xml文件模板)

1. 问题的引出 我们在日常项目中可以发现&#xff0c;sql映射文件和mybatis主配置文件&#xff0c;以及application.yml文件中有很多固定不变的内容&#xff0c;为了方面使用&#xff0c;所以可以把这些xml文件设置为模板 2. 创建模板的步骤 按照图片一步一步进行即可 点击…

(二)结构型模式:1、适配器模式(Adapter Pattern)(C++实现示例)

目录 1、适配器模式&#xff08;Adapter Pattern&#xff09;含义 2、适配器模式应用场景 3、适配器模式的UML图学习 4、C实现适配器模式的示例 1、适配器模式&#xff08;Adapter Pattern&#xff09;含义 将一个接口转换为客户端所期待的接口&#xff0c;从而使两个接口…

Mac安装nvm教程及使用

nvm 是 node 版本管理器&#xff0c;也就是说一个 nvm 可以管理多个 node 版本&#xff08;包含 npm 与 npx&#xff09;&#xff0c;可以方便快捷的安装、切换 不同版本的 node。 1、直接通过brew安装 执行命令&#xff1a;brew install nvm PS&#xff1a; 如果没有安装br…

Oracle database 静默安装 oracle 11g 一键安装

基于oracle安装包中应答文件实现一键安装 支持环境&#xff1a; Linux &#xff1a;centerOS 7 oracle &#xff1a;11.2.0 Oracle应答文件 runInstaller应答文件 /database/response/db_install.rsp netca应答文件 /database/response/netca.rsp dbca应答文件 /database/re…

python中的运算符号含义,python基本运算符的操作

本篇文章给大家谈谈python的运算符号有哪些类型&#xff0c;以及python各运算符号的功能说明&#xff0c;希望对各位有所帮助&#xff0c;不要忘了收藏本站喔。 1.算数运算符&#xff08;最常见的&#xff09; 标准算数运算符&#xff08;加减乘除&#xff09; 取余运算…

UML-状态图

目录 状态图 状态图的图符 状态机 状态 ​转换 电话机状态图 活动图和状态图区别&#xff1a; 状态图 状态图(Statechart Diagram)是描述一个实体基于事件反应的动态行为&#xff0c;显示了该实体如何根据当前所处的状态对不同的事件做出反应。通常我们创建一个UML状态…

Jmeter设置中文的两种方式,建议使用第二种

方案一 进入jmeter图像化界面&#xff0c;选择Options下的Choose Language&#xff0c;再选择Chinese(Simplified)。这个就是选择语言为简体中文&#xff08;缺陷&#xff1a;这个只是在本次使用时为中文&#xff0c;下次打开默认还是英文的&#xff09; 方案二&#xff08;…

数学建模(二)线性规划

课程推荐&#xff1a;6 线性规划模型基本原理与编程实现_哔哩哔哩_bilibili 目录 一、线性规划的实例与定义 1.1 线性规划的实例 1.2 线性规划的定义 1.3 最优解 1.4 线性规划的Mathlab标准形式 1.5 使用linprog函数 二、线性规划模型建模实战与代码 2.1 问题提出 2.2…

机器学习深度学习——seq2seq实现机器翻译(详细实现与原理推导)

&#x1f468;‍&#x1f393;作者简介&#xff1a;一位即将上大四&#xff0c;正专攻机器学习的保研er &#x1f30c;上期文章&#xff1a;机器学习&&深度学习——seq2seq实现机器翻译&#xff08;数据集处理&#xff09; &#x1f4da;订阅专栏&#xff1a;机器学习&…

机器学习编译系列

机器学习编译MLC 1. 引言2. 机器学习编译--概述2.1 什么是机器学习编译 1. 引言 陈天奇目前任教于CMU&#xff0c;研究方向为机器学习系统。他是TVM、MXNET、XGBoost的主要作者。2022年夏天&#xff0c;陈天奇在B站开设了《机器学习编译》的课程。   《机器学习编译》课程共分…

立即开始使用 3D 图像

一、说明 这个故事介绍了使用这种类型的数据来训练机器学习3D模型。特别是&#xff0c;我们讨论了Kaggle中可用的MNIST数据集的3D版本&#xff0c;以及如何使用Keras训练模型识别3D数字。 3D 数据无处不在。由于我们希望构建AI来与我们的物理世界进行交互&#xff0c;因此使用3…

了解 Langchain️是个啥?:第 1 部分

一、说明 在日常生活中&#xff0c;我们主要致力于构建端到端的应用程序。我们可以使用许多自动 ML 平台和 CI/CD 管道来自动化 ml 管道。我们还有像Roboflow和Andrew N.G.的登陆AI这样的工具来自动化或创建端到端的计算机视觉应用程序。 如果我们想在OpenAI或拥抱脸的帮助下创…

Day 26 C++ list容器(链表)

文章目录 list基本概念定义结构双向迭代器优点缺点List和vector区别存储结构内存管理迭代器稳定性随机访问效率 list构造函数——创建list容器函数原型示例 list 赋值和交换函数原型 list 大小操作函数原型示例 list 插入和删除函数原型示例 list 数据存取函数原型注意示例 lis…