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

lwip单网卡多ip的实现

1、今天要实现lwip的多个ip配置,本来以为需要自己修改很多核心代码

2、查阅资料才发现,lwip已经把接口留出来了

/** Define this to 1 and define LWIP_ARP_FILTER_NETIF_FN(pbuf, netif, type)
 * to a filter function that returns the correct netif when using multiple
 * netifs on one hardware interface where the netif's low-level receive
 * routine cannot decide for the correct netif (e.g. when mapping multiple
 * IP addresses to one hardware interface).
 */

/** 将此定义为1,并将LWIP_ARP_FILTER_NETIF_FN(pbuf, netif, type)定义为一个过滤函数,该函数在使用单个硬件接口上的多个netif时返回正确的netif。当netif的低级接收例程无法确定正确的netif时(例如,当将多个IP地址映射到单个硬件接口时),需要使用此过滤函数。
*/


#ifndef LWIP_ARP_FILTER_NETIF
#define LWIP_ARP_FILTER_NETIF 0
#endif

3、所以我们只需要定义好宏

        #define LWIP_ARP_FILTER_NETIF    1        //多ip的支持

和写好

struct netif * LWIP_ARP_FILTER_NETIF_FN(struct pbuf *p, struct netif *netifIn, u16_t type)

函数即可

4、我的操作

        4.1> 在lwipopt.h定义使能宏LWIP_ARP_FILTER_NETIF ,以支持多ip的操作

4.2、编写LWIP_ARP_FILTER_NETIF_FN函数,以能根据ip能找到对应的网卡

/*
*********************************************************************************************************
* Function Name : LWIP_ARP_FILTER_NETIF_FN
* Description   : 多ip,识别正确的网卡
* Input         : None
* Output        : None
* Return        : None
*********************************************************************************************************
*/
#include  "lwip/prot/etharp.h"
struct netif * LWIP_ARP_FILTER_NETIF_FN(struct pbuf *p, struct netif *netifIn, u16_t type)
{
	struct netif *netif = NULL;
	struct etharp_hdr *hdr = NULL;
	struct ip_hdr *iphdr = NULL;
	ip_addr_t dest/*, src*/;
	switch (type)
	{
		/*ARP*/
	case	0x0806:
		hdr = (struct etharp_hdr *)((unsigned char*)p->payload + 14);
		memcpy(&dest, &(hdr->dipaddr), sizeof(ip4_addr_t));
		//memcpy(&src, &hdr->sipaddr, sizeof (ip4_addr_t));
		for (netif = netif_list; netif != NULL; netif = netif->next)
		{
			if (netif_is_up(netif))
			{
				if (ip4_addr_cmp(&dest,&(netif->ip_addr)))
				{
					break;
				}
			}
		}
		break;
		/*IP*/
	case	0x0800:
		iphdr = (struct ip_hdr *) ((unsigned char*)p->payload + 14);
		ip_addr_copy_from_ip4(dest, iphdr->dest);
		//ip_addr_copy_from_ip4(src, iphdr->src);
		for (netif = netif_list; netif != NULL; netif = netif->next)
		{
			if (netif_is_up(netif))
			{
				if (ip4_addr_cmp(&dest, &(netif->ip_addr)))
				{
					break;
				}
			}
		}
		break;

	default:
		netif = netif_list;
		break;
	}
	netifIn = netif;
    if(netif==NULL)
    {
        pbuf_free(p);
    }
	return netif;
    
}

根据ip去匹配netif,arp需要单独处理,因为ip通信首先都要发送arp去找物理地址,不加这个arp处理,会出现你ping不通ip。所以如果知道物理地址,不加这个arp处理也可以,直接自己填mac即可,加到你自己的静态arp表中。

大家注意:

这个一定要加上,我刚开始没加,结果刚开始通,运行一段时间导致不通了,就是因为这里没释放申请的内存,导致无法处理新的数据

4.3、增加多ip的设置

	//开始虚拟多ip--1
	IP4_ADDR(&ipaddr_v[1], 192,168,20,48);
	IP4_ADDR(&netmask_v[1], 255,255,255,0);
	IP4_ADDR(&gw_v[1], 192, 168, 20,1);
	netif_add(&g_netif_v[1], &ipaddr_v[1], &netmask_v[1], &gw_v[1], NULL, &ethernetif_init, &tcpip_input);	//网卡初始化和网卡输入
	netif_set_up(&g_netif_v[1]);

	//开始虚拟多ip--2
	IP4_ADDR(&ipaddr_v[2], 192,168,30,48);
	IP4_ADDR(&netmask_v[2], 255,255,255,0);
	IP4_ADDR(&gw_v[2], 192, 168, 30,1);
	netif_add(&g_netif_v[2], &ipaddr_v[2], &netmask_v[2], &gw_v[2], NULL, &ethernetif_init, &tcpip_input);	//网卡初始化和网卡输入
	netif_set_up(&g_netif_v[2]);

4.4、然后运行测试,OK

5、测试结果

刚开始测试结果是这样,我没在意,以为是硬件还是那里问题,ping移植有超时,或者卡顿较大

有时,还会出现几十上百ms的延迟,特别是多个电脑同时ping时。

6、ping超时,卡顿较大原因

1> 原来多个neif_add时:

会每次都初始化网卡,这里网卡在初始化时,会创建解析网络中断数据包的任务,所以存在多个任务去读网卡数据,造成数据的竞争,从而导致可能出现超时,卡顿较大

2> 我们只有一个网卡,所以只需初始化1次解析网络中断数据包的任务

改了之后就OK了

像下面这篇文章这样:


http://www.kler.cn/a/504179.html

相关文章:

  • 【Docker】保姆级 docker 容器部署 MySQL 及 Navicat 远程连接
  • 《使用人工智能心脏磁共振成像筛查和诊断心血管疾病》论文精读
  • String.intern是什么
  • spring mvc源码学习笔记之十一
  • 快速上手 INFINI Console 的 TopN 指标功能
  • 前端多语言
  • Python海龟绘图库:从入门到精通 - Python官方文档(三万字解析!)
  • Ubuntu20.04复现GraspNet全记录(含遇到的问题及解决方法
  • C语言——动态内存管理
  • pytorch小记(五):pytorch中的求导操作:backward()
  • 向u-boot提交补丁的流程
  • 【高可用自动化体系】自动化体系
  • [NOIP2007 提高组] 矩阵取数游戏
  • 如何物理连接Franka机械臂
  • 【Vim Masterclass 笔记14】S07L29 + L30:练习课08 —— Vim 文本对象同步练习(含点评课内容)
  • 分布式缓存redis
  • IDM-VTON效果测试
  • JavaScript中如何实现函数签名
  • 25/1/14 算法笔记<强化学习> CBR加强化学习
  • 容器技术全面攻略:Docker的硬核玩法
  • 从零到一:用 Flask 和 Docker 构建并部署一个简单的接口请求页面
  • SpringData-Redis缓存之RedisTemplate
  • 使用 OpenSSL 实现 SSL/TLS 握手的流程和 Demo 示例
  • 从玩具到工业控制--51单片机的跨界传奇【2】
  • 运维练习题2
  • STORM:从多时间点2D图像中快速重建动态3D场景的技术突破