热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

linux内核网络协议栈

未完,待续1Raw_Socket原始套接字2ARP的C代码实现3AF_PACKET发送以太网帧4ARP发送5sendarp[socket创建]

未完,待续...


1 Raw_Socket原始套接字

2 ARP的C代码实现

3 AF_PACKET发送以太网帧

4 ARP发送

5 send arp

[socket创建]

 socket(int family, int type, int protocol);

1    family:AF_INET,AF_PACKET; 

    协议族:创建socket时使用.

    每个协议族都通过packet_init/inet_init->sock_register函数将(packet_familiy_ops/inet_familiy_ops)

    注册给net_families[];

    socket->__sock_create创建时,通过pf->create找到对应的inet_familiy_ops->inet_create或packet_create等.

   

 2   type=SOCK_STREAM;SOCK_DGRAM;SOCK_RAW;SOCK_PACKET;

   socket类型:创建socket时使用;

    如:inet_create中通过sock->type(__sock_create中将sock->type=type),遍历inetsw链表;

           通过inetsw链表找到inetsw_array[sock_type],所以也相当于遍历了inetsw_array[];

  查找inetsw_array[sock_type]是通过遍历inetsw_array[],然后比较每个inetsw_array[].protocol与socket调用传人的protocol是否相同.

            inet_init时注册:inet_register_protosw(inetsw_array[]);此时inetsw_array[]和inetsw关联;

            inet_create中根据inetsw_array[type]初始化sock->ops=inetsw_array[type].ops;

struct inet_protosw inetsw_array[]=

{

            {

.type=SOCK_STREAM,

.protocol=IPPROTO_TCP,

.proto = &tcp_prot,

.ops = inet_stream_ops,

             },

}

struct proto_ops inet_stream_ops = {

.recvmsg=inet_recvmsg, //调用sk->sk_prot->recvmsg=tcp_recvmsg

}

inet_recvmsg->tcp_recvmsg

struct proto tcp_prot={

.recvmsg=tcp_recvmsg,

}

recvmsg->__sock_recvmsg->(sock->ops->recvmsg())此时的sock->ops是在inet_create中初始化的inetsw_array[type].ops.



3 packet:IPPROTO_IP/IPPROTO_TCP等;

   

     socket->__sock_create->inet_create时根据inetsw找到对应inetsw_array[]


2 socket->sock_create->__sock_create


[收包]

网络包接收分两大部分:

第一:mac驱动层收包,软中断将包放在sock的接收队列中:sock->sk_receive_queue;然后tcp_v4_do_rcv唤醒用户进程收包。

用户进程在tcp_recvmsg->sk_wait_data中等待;

第二:用户进程通过recv->tcp_rcvmsg收包,然后等待sk_receive_queue队列上有skb到达;

1 驱动->协议栈:netif_receive_skb();

驱动层收包会:skb->protocol=eth_type_trans()处理,得到上层协议类型,如:ETH_P_IP/ETH_P_8021Q;

skb->protocol会影响网络层的处理方式(ip_rcv/arp_rcv);

2 协议栈逐层传递:

 __netif_recive_skb_core->deliver_skb()

 遍历ptype_all;参考skb->protocol调用packet_type->func();不同的协议族对应的packet如ip_packet_type,有各自不同的sock,

 每个sock都有sock->sk_receive_queue接受队列;

ptype_all通过dev_add_pack注册,

ip_rcv、arp_rcv的选取和skb->protocol有关,ETH_P_IP对应ip_rcv;ETH_P_8021Q对应arp_rcv();

如ip_rcv注册:inet_init->dev_add_pack(&ip_packet_type);//ip_packet_type>func=ip_rcv;

arp_rcv注册:arp_init->dev_add_pack(&arp_packet_type);


packet_rcv注册:packet_bind/packet_create->register_prot_hook;

packet_rcv中将skb挂载到接收队列(sock->sk_receive_queue每个socket对应一个接收队列)

sock=packet_type->af_packet_priv上,并唤醒用户接收数据包;


3 ip层传递:ip_rcv->ip_rcv_finish->dst_input=ip_local_deliver->inet_protos[protocol]->handler();

inet_protos[]在inet_init中通过inet_add_protocol(&tcp_protocol)注册.

ip层继续向上传递到四层协议处理;


arp传递:arp_rcv 直接处理arp协议,不向上层传递;

packet_rcv:直接在packet_rcv中唤醒用户接收,不向上传递,sk->sk_data_ready=sock_def_readable;

AF_PACKET接收报文,未去mac协议头;


4 tcp层传递:

三层向四层传递过程调用了:inet_protos[protocol]->handler();

而inet_protos[]是在inet_init中通过inet_add_protocol(&tcp_protocol)注册.如:udp_protocol/icmp_protocol等.

tcp层收包:tcp_v4_rcv->tcp_v4_do_rcv->tcp_rcv_established->tcp_queue_rcv->__skb_queue_tail(&sk->sk_receive_queue,skb);

static struct tcp_protocol tcp_protocol={

.handler = tcp_v4_rcv,

}

static net_protocol udp_protocol = {

.handler = udp_rcv,

}


推荐阅读
  • 本题来自WC2014,题目编号为BZOJ3435、洛谷P3920和UOJ55。该问题描述了一棵不断生长的带权树及其节点上小精灵之间的友谊关系,要求实时计算每次新增节点后树上所有可能的朋友对数。 ... [详细]
  • 嵌入式开发环境搭建与文件传输指南
    本文详细介绍了如何为嵌入式应用开发搭建必要的软硬件环境,并提供了通过串口和网线两种方式将文件传输到开发板的具体步骤。适合Linux开发初学者参考。 ... [详细]
  • Python 工具推荐 | PyHubWeekly 第二十一期:提升命令行体验的五大工具
    本期 PyHubWeekly 为大家精选了 GitHub 上五个优秀的 Python 工具,涵盖金融数据可视化、终端美化、国际化支持、图像增强和远程 Shell 环境配置。欢迎关注并参与项目。 ... [详细]
  • 通常情况下,修改my.cnf配置文件后需要重启MySQL服务才能使新参数生效。然而,通过特定命令可以在不重启服务的情况下实现配置的即时更新。本文将详细介绍如何在线调整MySQL配置,并验证其有效性。 ... [详细]
  • Symfony是一个功能强大的PHP框架,以其依赖注入(DI)特性著称。许多流行的PHP框架如Drupal和Laravel的核心组件都基于Symfony构建。本文将详细介绍Symfony的安装方法及其基本使用。 ... [详细]
  • 本文详细介绍了 Python 中的条件语句和循环结构。主要内容包括:1. 分支语句(if...elif...else);2. 循环语句(for, while 及嵌套循环);3. 控制循环的语句(break, continue, else)。通过具体示例,帮助读者更好地理解和应用这些语句。 ... [详细]
  • 2012年7月30日,语言岛团队宣布其智能记单词软件V0.3.4.554版本正式开源。该版本不仅支持跨平台使用,还引入了多项创新功能,旨在帮助用户更高效地记忆单词。 ... [详细]
  • 在编译BSP包过程中,遇到了一个与 'gets' 函数相关的编译错误。该问题通常发生在较新的编译环境中,由于 'gets' 函数已被弃用并视为安全漏洞。本文将详细介绍如何通过修改源代码和配置文件来解决这一问题。 ... [详细]
  • Linux环境下进程间通信:深入解析信号机制
    本文详细探讨了Linux系统中信号的生命周期,从信号生成到处理函数执行完毕的全过程,并介绍了信号编程中的注意事项和常见应用实例。通过分析信号在进程中的注册、注销及处理过程,帮助读者理解如何高效利用信号进行进程间通信。 ... [详细]
  • 本文详细介绍了如何在云服务器上配置Nginx、Tomcat、JDK和MySQL。涵盖从下载、安装到配置的完整步骤,帮助读者快速搭建Java Web开发环境。 ... [详细]
  • 对于许多初学者而言,遇到总线错误(bus error)或段错误(segmentation fault/core dump)是极其令人困扰的。本文详细探讨了这两种错误的成因、表现形式及解决方法,并提供了实用的调试技巧。 ... [详细]
  • 主调|大侠_重温C++ ... [详细]
  • 本文详细介绍了Java中实现异步调用的多种方式,包括线程创建、Future接口、CompletableFuture类以及Spring框架的@Async注解。通过代码示例和深入解析,帮助读者理解并掌握这些技术。 ... [详细]
  • 本文详细介绍了如何在 Android 中使用值动画(ValueAnimator)来动态调整 ImageView 的高度,并探讨了相关的关键属性和方法,包括图片填充后的高度、原始图片高度、动画变化因子以及布局重置等。 ... [详细]
  • 本文详细介绍了如何解压并安装MySQL集群压缩包,创建用户和组,初始化数据库,配置环境变量,并启动相关服务。此外,还提供了详细的命令行操作步骤和常见问题的解决方案。 ... [详细]
author-avatar
烂哥居士
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有