🌈hello,你好鸭,我是Ethan,西安电子科技大学大三在读,很高兴你能来阅读。
✔️目前博客主要更新Java系列、项目案例、计算机必学四件套等。
🏃人生之义,在于追求,不在成败,勤通大道。加油呀!
🔥个人主页:Ethan Yankang
🔥推荐:史上最强八股文||一分钟看完我的几百篇博客
🔥温馨提示:划到文末发现专栏彩蛋 点击这里直接传送
🔥本篇概览:详细讲解了手写TCP/IP的第4节第2讲——ARP协议的实现2——无回报的ARP生成(系统在启用网卡时,首先向网络上发送广播包,告诉所有人自己的IP地址和MAC地址。)🌈⭕🔥
【计算机领域一切迷惑的源头都是基本概念的模糊,算法除外】
🔥 手写底层系列
🔥 手写TCP/IP系列
【OSI与课程讲解】
🌈章节引出:
前一篇章:手写TCP/IP——第4节ARP协议的实现1——ARP协议的初始化-CSDN博客
🌈章节速览:
1.无回报的ARP引出
现实生活:
假设我们在群聊里面又很多机器来找张三,那么每一次张三都要回复一次,这很麻烦,最好的方式就是,张三进群是就@所有人说,这是我,以后你们就直接可以找到我了。
网络中也存在类似问题——假设现在有一台机器有很机器来连接,那么最好的方式就是这台机器在启动时就将自己的MAC与IP信息发送给所有机器,这样避免了在后面每一次连接时再发请求,返回响应。而这个网卡一启动就发送的数据包,就称为“广播包”,也称为无回报的ARP包。
注意这次发包不需要任何响应。
2.真实抓包演示
我们启动虚拟机,发现首先发了两个包如下:
纵览启动的整个过程使用到的协议:
逐步分析这3个协议(gpt):
一、整体流程概述
当启动网卡时,设备会发送一个特定的数据包,这个数据包在网络中传播以实现特定的网络通信和配置功能。
二、各步骤分析
-
“Frame 1: 42 bytes on wire (336 bits), 42 bytes captured (336 bits) on interface \Device\NPF_{3E446065-DFB8-420B-BBD8-F5933B21AFE1}, id 0”:
- 这一步表明在特定的网络接口(\Device\NPF_{3E446065-DFB8-420B-BBD8-F5933B21AFE1},接口编号为 0)上捕获到了一个长度为 42 字节的数据包。这个信息主要是关于数据包的物理捕获情况,包括在哪个接口上以及数据包的大小。【实质就是底层的通信电路被启动了,发出的信号】
- 做的事情:确定数据包的来源接口和大小,为后续分析提供基础信息。
-
“Ethernet II, Src: PcsCompu_ae:36:6b (08:00:27:ae:36:6b), Dst: Broadcast (ff:ff:ff:ff:ff:ff)”:
- 这一步描述了数据包的以太网帧格式。
- 源地址为 “08:00:27:ae:36:6b”,代表发送这个数据包的设备的 MAC 地址,可能是名为 “PcsCompu” 的设备。源地址是全局唯一地址,用于在网络中唯一标识这个设备。
- 目的地址为广播地址 “ff:ff:ff:ff:ff:ff”,意味着这个数据包将被发送到同一网络中的所有设备。【后面马上通过ARP发送】
- 做的事情:定义了数据包的发送者和接收者范围。通过使用广播地址,确保网络中的所有设备都有机会接收到这个数据包,以便进行后续的网络通信和配置。
-
“Address Resolution Protocol (ARP Announcement)”:
- 这一步表明数据包中携带的是地址解析协议(ARP)的内容,并且是一个 ARP 通告。
- 做的事情:
- 宣告自己的存在:设备在启动网卡时发送 ARP 通告,向网络中的其他设备表明自己已经加入网络。
- 建立 IP 地址和 MAC 地址的映射:其他设备接收到这个通告后,可以更新自己的 ARP 缓存,以便在需要与这个设备通信时能够正确地找到它的 MAC 地址。
- 检测网络中的重复 IP 地址:如果网络中已经存在一个设备使用了相同的 IP 地址,那么可能会收到响应,从而发现 IP 地址冲突。
综上所述,在网卡启动过程中,设备通过发送这个特定的数据包,通过 ARP 通告来宣告自己的存在、建立地址映射和检测 IP 地址冲突。
3.查看ARP组成结构
4.代码实现
4.1包结构定义:
//定义ARP包
typedef struct _xarp_packet_t {
uint16_t hw_type, pro_type; // 硬件类型和协议类型
uint8_t hw_len, pro_len; // 硬件地址长 + 协议地址长
uint16_t opcode; // 请求/响应
uint8_t sender_mac[XNET_MAC_ADDR_SIZE]; // 发送包硬件地址
uint8_t sender_ip[XNET_IPV4_ADDR_SIZE]; // 发送包协议地址
uint8_t target_mac[XNET_MAC_ADDR_SIZE]; // 接收方硬件地址
uint8_t target_ip[XNET_IPV4_ADDR_SIZE]; // 接收方协议地址
}xarp_packet_t;
4.2发送包实现
/**
* 生成一个ARP响应,这里是网卡启动的时候,给同一网络中的所有机器发送自己的ip与MAC地址的映射关系,也称为"广播包"!避免后面机器再逐一请求,麻烦!
* @param arp_packet 接收到的ARP请求包
* @return 生成结果
*/
xnet_err_t xarp_make_response(xarp_packet_t * arp_packet) {
xarp_packet_t* response_packet;
xnet_packet_t * packet = xnet_alloc_for_send(sizeof(xarp_packet_t));
response_packet = (xarp_packet_t *)packet->data;
response_packet->hw_type = swap_order16(XARP_HW_ETHER);
response_packet->pro_type = swap_order16(XNET_PROTOCOL_IP);
response_packet->hw_len = XNET_MAC_ADDR_SIZE;
response_packet->pro_len = XNET_IPV4_ADDR_SIZE;
response_packet->opcode= swap_order16(XARP_REPLY);
memcpy(response_packet->target_mac, arp_packet->sender_mac, XNET_MAC_ADDR_SIZE);
memcpy(response_packet->target_ip, arp_packet->sender_ip, XNET_IPV4_ADDR_SIZE);
memcpy(response_packet->sender_mac, netif_mac, XNET_MAC_ADDR_SIZE);
memcpy(response_packet->sender_ip, netif_ipaddr.array, XNET_IPV4_ADDR_SIZE);
return ethernet_out_to(XNET_PROTOCOL_ARP, ether_broadcast, packet);
}
5.完整源码:diy-tcp-ip/xnet_tiny_c0702/src at main · ZYKWLJ/diy-tcp-ip · GitHub
💖💖💖💖💖💖💖💖💖💖💖💖💖💖💖💖💖💖
热门专栏推荐
🌈🌈计算机科学入门系列 关注走一波💕💕
🌈🌈CSAPP深入理解计算机原理 关注走一波💕💕
🌈🌈微服务项目之黑马头条 关注走一波💕💕
🌈🌈redis深度项目之黑马点评 关注走一波💕💕
🌈🌈JAVA面试八股文系列专栏 关注走一波💕💕
🌈🌈JAVA基础试题集精讲 关注走一波💕💕
🌈🌈代码随想录精讲200题 关注走一波💕💕
总栏
🌈🌈JAVA基础要夯牢 关注走一波💕💕
🌈🌈JAVA后端技术栈 关注走一波💕💕
🌈🌈JAVA面试八股文 关注走一波💕💕
🌈🌈JAVA项目(含源码深度剖析) 关注走一波💕💕
🌈🌈计算机四件套 关注走一波💕💕
🌈🌈数据结构与算法 关注走一波💕💕
🌈🌈必知必会工具集 关注走一波💕💕
🌈🌈书籍网课笔记汇总 关注走一波💕💕
📣非常感谢你阅读到这里,如果这篇文章对你有帮助,希望能留下你的点赞👍 关注❤收藏✅ 评论💬,大佬三连必回哦!thanks!!!
📚愿大家都能学有所得,功不唐捐!