Linux 网络设备驱动开发(三)-网卡驱动深层分析

之前我们介绍了网卡是怎么把一个数据包发送到网络上的,但是这只是Linux网络系统中的一个非常小的部分。对于内核怎么把用户数据传递给网卡,以及内核怎么把网卡收到的数据传递给用户是一个庞大的知识。
学过计算机网络的都知道,当用户需要发送数据的时候,需要根据路由表找到数据包下一站该发送到哪个路由器,这个路由器叫做这个网卡的邻居。如果邻居的MAC地址不知道,还需要通过ARP协议获取的路由去的MAC地址,这个过程交给邻居子系统来完成。
分析过程还是依据第一节课中的5层模型来分析:
1、系统调用接口
2、协议无关层
3、网络协议栈
4、设备无关层
5、硬件驱动
发送数据过程
当用户调用write系统调用时,内核是用哪个函数来响应的呢?在第一节课中我们知道soket有一个操作函数集,socket_file_ops:

/*
* Socket files have a set of 'special' operations as well as the generic file ones. These don't appear
* in the operation structures but are done directly via the socketcall() multiplexor.
*/

static const struct file_operations socket_file_ops = {
.owner = THIS_MODULE,
.llseek = no_llseek,
.aio_read = sock_aio_read,
.aio_write = sock_aio_write,
.poll = sock_poll,
.unlocked_ioctl = sock_ioctl,
#ifdef CONFIG_COMPAT
.compat_ioctl = compat_sock_ioctl,
#endif
.mmap = sock_mmap,
.open = sock_no_open, /* special open code to disallow open via /proc */
.release = sock_close,
.fasync = sock_fasync,
.sendpage = sock_sendpage,
.splice_write = generic_splice_sendpage,
.splice_read = sock_splice_read,
};
​​​​系统调用接口和协议无关层
1.发送入口sock_aio_write
2.调用do_sock_write
3.调用__sock_sendmsg
网络协议栈
4.调用udp_sendmsg
5.调用ip_route_output_flow(选择路由)
6.调用udp_push_pending_frames
7.调用ip_push_pending_frames(IP协议入口)
8.调用ip_local_out
9.调用dst_output
10.调用ip_finish_output
11.调用ip_finish_output2(建立邻居信息,路由中紧挨着发送方的网关,如果没有邻居信息,那么由Linux中的邻居子系统来建立邻居信息)
12.调用arp_generic_ops下面的neigh_resolve_output
13.调用queue_xmit
设备无关接口
14.调用dev_queue_xmit
15.调用dev_hard_start_xmit
设备驱动
16.调用ndo_start_xmit

 

 


接收数据过程
设备驱动
1.产生接收中断:从硬件读取数据放到skb中;调用netif_rx,并产生一次软中断
设备无关接口
2.调用net_rx_action处理软中断
3.调用netif_recevie_skb
4.调用deliver_skb
网络协议栈
5.调用ip_rcv
6.调用udp_rcv
系统调用层和协议无关层
7.调用sock->ops->recvmsg

 

 

  • 0
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值