Linux内核分析 - 网络[三]:从netif_receive_skb()说起

2014-11-24 08:54:33 ? 作者: ? 浏览: 0

netif_receive_skb()函数中,可以看出处理的是像ARPIP这些链路层以上的协议,那么,链路层报头是在哪里去掉的呢?答案是网卡驱动中,在调用netif_receive_skb()前,



相关阅读:


该函数对处理后skb>data跳过以太网报头,由mac_header指示以太网报头:




进入netif_receive_skb()函数


按照协议类型依次由相应的协议模块进行处理,而所以的协议模块处理都会注册在ptype_base中,实际是链表结构。


net\core\dev.c



而相应的协议模块是通过dev_add_pack()函数加入的:



ARP处理为例


该模块的定义,它会在arp_init()中注册进ptype_base链表中:



然后在根据报文的TYPE来在ptype_base中查找相应协议模块进行处理时,实际调用arp_rcv()进行接收


arp_rcv() --> arp_process()


操作后这指针位置:




然后判断是ARP请求报文,这时先查询路由表ip_route_input()


ip_route_input()函数中,先在cache中查询是否存在相应的路由表项:


缓存的路由项在内核中组织成hash表的形式,因此在查询时,先算出的hash值,再用该项- rt_hash_table[hash].chain即可。这里可以看到,缓存路由项包括了源IP地址、目的IP地址、网卡号。



如果在缓存中没有查到匹配项,或指定不查询cache,则查询路由表ip_route_input_slow()


进入ip_route_input_slow()函数,最终调用fib_lookup()得到查询结果fib_result


如果结果fib_result合法,则需要更新路由缓存,将此次查询结果写入缓存



在查找完路由表后,回到arp_process()函数,如果路由项指向本地,则应由本机接收该报文:


首先更新邻居表neigh_event_ns(),然后发送ARP响应 – arp_send


至此,大致的ARP流程完成。由于ARP部分涉及到路由表以及邻居表,这都是很大的概念,在下一篇中介绍,这里直接略过了。


-->

评论

帐  号: 密码: (新用户注册)
验 证 码:
表  情:
内  容: