当前位置: 首页 > news >正文

W6100-EVB-PICO进行UDP组播数据回环测试(九)

前言

        上一章我们用我们的开发板作为UDP客户端连接服务器进行数据回环测试,那么本章我们进行UDP组播数据回环测试。

什么是UDP组播?

        组播是主机间一对多的通讯模式, 组播是一种允许一个或多个组播源发送同一报文到多个接收者的技术。组播源将一份报文发送到特定的组播地址,组播地址不同于单播地址,它并不属于特定某个主机,而是属于一组主机。一个组播地址表示一个群组,需要接收组播报文的接收者都加入这个群组。

IPv4组播地址

IANA将D类地址空间分配给IPv4组播使用。IPv4地址一共32位,D类地址最高4位为1110,地址范围从224.0.0.0到239.255.255.255,具体分类及含义见下图。

连接方式

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

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

测试工具

  • 网路调试工具(具有UDP组播模式的功能)
  • wireshark抓包工具

UDP组播数据回环测试

1. 相关代码

我们打开库文件找到其中的multicast.c文件,先看下传入的参数有:socket端口号、数据收发缓存、组播MAC地址、组播IP地址、组播端口和回环模式(W6100支持IPv6,因此模式上做以区分);根据数据手册组播模式在打开socket端口前需要先初始化组播的MAC地址、IP地址和端口,端口打开后判断是否收到数据,收到就进行回传,并在串口打印显示;整体逻辑比较简单,就是在UDP模式下,以组播进行回环测试,代码如下所示:

int32_t multicast_loopback(uint8_t sn, uint8_t* buf, uint8_t* multicast_mac, uint8_t* multicast_ip, uint16_t multicast_port, uint8_t loopback_mode)
{uint8_t status;static uint8_t dest_ip[16] = {0,};static uint16_t dest_port;uint8_t pack_info;uint8_t addr_len;datasize_t ret;datasize_t received_size;uint16_t size, sentsize;uint8_t* mode_msg;uint16_t any_port = 50000;if(loopback_mode == AS_IPV4){mode_msg = "IPv4 mode";}else if(loopback_mode == AS_IPV6){mode_msg = "IPv6 mode";}else{mode_msg = "Dual IP mode";}getsockopt(sn, SO_STATUS, &status);switch (status){case SOCK_UDP:getsockopt(sn, SO_RECVBUF, &received_size);if(received_size > DATA_BUF_SIZE) received_size = DATA_BUF_SIZE;if(received_size>0){ret = recvfrom(sn, buf, received_size, (uint8_t*)&dest_ip, (uint16_t*)&dest_port, &addr_len);buf[ret]=0x00;printf("recv from [%d.%d.%d.%d][%d]: %s\n",dest_ip[0],dest_ip[1],dest_ip[2],dest_ip[3],dest_port,buf);if(ret <= 0)return ret;received_size = (uint16_t) ret;sentsize = 0;while(sentsize != received_size){ret = sendto(sn, buf+sentsize, received_size-sentsize, dest_ip, dest_port, addr_len);if(ret < 0) return ret;sentsize += ret; // Don't care SOCKERR_BUSY, because it is zero.}}break;case SOCK_CLOSED:setSn_DHAR(sn, multicast_mac);setSn_DIP4R(sn, multicast_ip);setSn_DPORTR(sn, multicast_port);switch(loopback_mode){case AS_IPV4:socket(sn,Sn_MR_UDP4, any_port, SF_MULTI_ENABLE);break;case AS_IPV6:socket(sn,Sn_MR_UDP6, any_port, SF_MULTI_ENABLE);break;case AS_IPDUAL:socket(sn,Sn_MR_UDPD, any_port, SF_MULTI_ENABLE);break;}printf("%d:UDP Multicast looptest as %s\r\n",sn,mode_msg);printf("%d:UDP Multicast IP: %d.%d.%d.%d\r\n",sn,multicast_ip[0],multicast_ip[1],multicast_ip[2],multicast_ip[3]);printf("%d:UDP Multicast PORT: %d\r\n",sn,multicast_port);break;default:break;}
}

主程序仍旧是初始化网络配置信息,并且初始化组播相关配置信息,这里的组播MAC地址值得一提:为了在本地物理网络上实现组播信息的正确传输,需要在链路层使用组播MAC地址。以太网传输IPv4单播报文的时候,目的MAC地址使用的是接收者的MAC地址。但是在传输组播数据时,其目的地不再是一个具体的接收者,而是一个成员不确定的组,所以要使用IPv4组播MAC地址,即IPv4组播地址映射到链路层中的地址。IANA规定,IPv4组播MAC地址的高24位为0x01005e,第25位为0,低23位为IPv4组播地址的低23位,映射关系如下图所示。

知道了映射关系后我们能很容易根据组播IP地址得到对应的组播MAC地址,例如我们本次测试的组播地址224.1.1.11对应的组播MAC地址为01-00-5e-01-01-0b。我们这里直接用数组初始化相关信息后传入测试函数,在主函数里循环调用即可,如下所示:

#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 uint8_t multicast_mac[6]={0x01,0x00,0x5e,0x01,0x01,0x0b}; // multicast mac address
static uint8_t multicast_ip[4] ={224, 1, 1, 11}; // multicast ip address
static uint16_t multicast_port = 30000; // multicast portint main()                                                          
{   stdio_init_all();sleep_ms(2000);network_init();while(true){multicast_loopback(SOCKET_ID, ethernet_buf, multicast_mac,multicast_ip, multicast_port, AS_IPV4);sleep_ms(500);}
}void network_init(void)
{uint8_t temp;wizchip_initialize();printf("W6100 udp multicast 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);ctlwizchip(CW_RESET_WIZCHIP, NULL);print_net_info(&get_info);sleep_ms(2000);   
}

2. 测试现象

我们编译烧录完成后,打开串行监视器,打开wireshark输入过滤条件<ip.addr == 224.1.1.11>然后开启监听,打开网络调试工具点击创建连接,类型选择UDP(组播模式),目标IP选择224.1.1.11,端口为3000(其他非特殊端口也可以),本机端口指定为30000(本次测试组播端口为30000)是为了方便我们在调试工具看到组播组收到数据,参数配置完成后我们点击创建,然后在发送区输入123点击发送测试,在调试工具接收区可以看到成功收到2次,一次是PC端发送给组播组的,另一次是我们开发板作为组播组成员收到PC端发送给组播组的数据后回传的;如下图所示:

 

相关链接 

本章相关例程链接icon-default.png?t=N6B9https://gitee.com/wiznet-hk/example-of-w6100-evb-pico.git本章所用网络调试工具资源获取链接icon-default.png?t=N6B9https://download.csdn.net/download/WIZnet2012/88252449?spm=1001.2101.3001.9500

http://www.lryc.cn/news/139462.html

相关文章:

  • Qt 阴影边框
  • 前端面试:【性能优化】页面加载性能、渲染性能、资源优化
  • 从按下电源键到进入系统,CPU在干什么?
  • TypeScript初体验
  • 基于 Alpine 环境源码构建 alibaba-tengine(阿里巴巴)的 Docker 镜像
  • 政府网站定期巡检:构建高效、安全与透明的数字政务
  • C++信息学奥赛1138:将字符串中的小写字母转换成大写字母
  • leetcode1475. 商品折扣后的最终价格 【单调栈】
  • macOS M1使用TensorFlow GPU加速
  • GNU-gcc编译选项-1
  • 【DEVOPS】Jenkins使用问题 - 控制台输出乱码
  • logback-spring.xml
  • 华为OD机试之报文重排序【Java源码】
  • 回归预测 | MATLAB实现BES-ELM秃鹰搜索优化算法优化极限学习机多输入单输出回归预测(多指标,多图)
  • DPU在东数西算背景下如何赋能下一代算力基础设施 中科驭数在未来网络发展大会论道
  • 2021年12月 C/C++(四级)真题解析#中国电子学会#全国青少年软件编程等级考试
  • ArcGIS Serve Windows下用户密码变更导致Server服务无法启动问题
  • React 面试题集锦
  • xargs命令解决“Argument list too long”
  • R语言中<- 的含义
  • 知识图谱Neo4j安装到实践全过程
  • 贪心算法:简单而高效的优化策略
  • 一生一芯6——ubuntu rpm软件安装
  • Python练习 函数取列表最小数
  • 五种重要的 AI 编程语言
  • 【linux】2 make/Makefile和gitee
  • db-gpt安装指南(docker版本)
  • 「Java」《深度解析Java Stream流的优雅数据处理》
  • 【云驻共创】华为云之手把手教你搭建IoT物联网应用充电桩实时监控大屏
  • Hadoop分布式计算与资源调度:打开专业江湖的魔幻之门