首页 | 新闻 | 新品 | 文库 | 方案 | 视频 | 下载 | 商城 | 开发板 | 数据中心 | 座谈新版 | 培训 | 工具 | 博客 | 论坛 | 百科 | GEC | 活动 | 主题月 | 电子展
返回列表 回复 发帖

网络处理的软中断机制分析

网络处理的软中断机制分析

f !supportLists]-->内核默认软中断机制分析(process_backlog)    首先需要介绍的就是netif_rx(在net/core/dev.c中定义)函数,这个函数在网卡驱动程序与linux内核之间建立了一道桥梁,将网卡接收上来的数据包(sk_buff形式)插入内核维护的接收缓冲区队列当中:
int netif_rx(struct sk_buff *skb)
{
      int this_cpu = smp_processor_id();
       struct softnet_data *queue;
       unsigned long flags;

       if (skb->stamp.tv_sec == 0)
              do_gettimeofday(&skb->stamp);
       /*
       获取当前处理CPU的接收数据包缓冲区队列指针。系统为每一个CPU都维护一个独立的列表,这样可以避免共享访问互斥问题。
       */
       queue = &softnet_data[this_cpu];

       local_irq_save(flags);

       netdev_rx_stat[this_cpu].total++;
       /*
       这里判断当前输入队列的长度是否超过预定义的一个值,如果没有超过,则向下执行。
       如果当前队列的长度大于0,则将sk_buff插入队列,并且返回,注意这里并没有调用__cpu_raise_softirq产生一个软中断,而较老的内核版本当中,插入队列以后立刻调用这个函数产生软中断。
       另外,如果队列长度为0,则需要调用netif_rx_schedule及后续的__netif_rx_schedule函数将当前网络设备加入 softnet_data的轮询列表(poll_list)当中,这个列表维护所有网络设备的列表,当系统软中断处理函数运行时,逐个检索处于 poll_list中的设备,然后调用设备的dev->poll方法处理输入数据包队列中的数据。
       将设备加入poll_list列表当中后,返回enqueue标记处继续将sk_buff加入输入数据包队列中,然后返回。
       */
       if (queue->input_pkt_queue.qlen <= netdev_max_backlog) {
              if (queue->input_pkt_queue.qlen) {
                     if (queue->throttle)
                            goto drop;

enqueue:
                     dev_hold(skb->dev);
                     /*
                     将当前sk_buff插入input_pkt_queue队列的尾部,即刻返回。
                     */
                     __skb_queue_tail(&queue->input_pkt_queue,skb);
                     local_irq_restore(flags);
                     return queue->cng_level;
              }
              if (queue->throttle) {
                     queue->throttle = 0;
              }
              netif_rx_schedule(&queue->blog_dev);
              goto enqueue;
       }

       if (queue->throttle == 0) {
              queue->throttle = 1;
              netdev_rx_stat[this_cpu].throttled++;
       }

drop:
       netdev_rx_stat[this_cpu].dropped++;
       local_irq_restore(flags);

       kfree_skb(skb);
       return NET_RX_DROP;
}

从上面的分析可以知道,netif_rx函数主要负责将数据包插入内核队列中,并触发软中断,这一点与较早的版本是不同的,那么软中断是在什么地方触发的呢?
以前的章节介绍过,硬件中断是在irq.c的do_IRQ函数中调用handle_IRQ_event函数,进而调用相应硬件驱动程序的中断处 理函数实现的。在do_IRQ函数执行完硬件处理函数以后,接着就会调用do_softirq函数执行软中断,并且根据软中断号在softirq_vec 数组中查找相应中断的action方法,对于NET_RX_SOFTIRQ类型的软中断来说,系统将其action注册为net_rx_action,这 样我们就进入了net_rx_action函数当中:
static void net_rx_action(struct softirq_action *h)
{
       int this_cpu = smp_processor_id();
       struct softnet_data *queue = &softnet_data[this_cpu];
       unsigned long start_time = jiffies;
       int budget = netdev_max_backlog;(系统每次从队列中取300个sk_buff处理)

       br_read_lock(BR_NETPROTO_LOCK);
       local_irq_disable();

       /*
       在这里循环判断当前轮询列表是否为空。如果不为空,则进入软中断处理过程。
       */
       while (!list_empty(&queue->poll_list)) {
              struct net_device *dev;

              if (budget <= 0 || jiffies - start_time > 1)
                     goto softnet_break;

              local_irq_enable();
              /*
              从轮询列表中取出当前的设备dev指针,接着为dev调用poll方法,这是设备初始化过程中已经定义好的方法,如果设备未能自己实现该函数,则系统默认 注册为process_backlog。poll函数执行成功返回0,失败返回-1。这里逻辑判断是,dev->quota必须大于0,而poll 函数执行成功,则可以继续,直到所有的设备都查询一遍为止。
              */
              dev = list_entry(queue->poll_list.next, struct net_device, poll_list);

              if (dev->quota <= 0 || dev->poll(dev, &budget)) {
                     /*
这里的poll函数是netdevice结构的一个函数指针,对于不同的网卡驱动程序,我们可以根据自己的情况定义poll方法的实现(e1000网卡驱动程序就自己实现了一个poll方法,详情后面来分析),而系统默认提供一个方法。
                     */
                     local_irq_disable();
                     list_del(&dev->poll_list);
                     list_add_tail(&dev->poll_list, &queue->poll_list);
                     if (dev->quota < 0)
                            dev->quota += dev->weight;
                     else
                            dev->quota = dev->weight;
              } else {
                     dev_put(dev);
                     local_irq_disable();
              }
       }

       local_irq_enable();
       br_read_unlock(BR_NETPROTO_LOCK);
       return;

softnet_break:
       netdev_rx_stat[this_cpu].time_squeeze++;
       /*
       触发软中断处理,等待下一次调用本函数。
       */
       __cpu_raise_softirq(this_cpu, NET_RX_SOFTIRQ);

       local_irq_enable();
       br_read_unlock(BR_NETPROTO_LOCK);
}

软中断处理函数net_rx_action实际上就是调用各个网络设备的poll方法处理数据包的,一般的讲,poll默认为process_backlog(在net/core/dev..c中定义):
static int process_backlog(struct net_device *backlog_dev, int *budget)
{
       int work = 0;
       int quota = min(backlog_dev->quota, *budget);
       int this_cpu = smp_processor_id();
       struct softnet_data *queue = &softnet_data[this_cpu];
       unsigned long start_time = jiffies;

       for (;;) {
              struct sk_buff *skb;
              struct net_device *dev;

              local_irq_disable();
              /*
              从输入缓冲区队列中取出sk_buff,调用netif_receive_skb函数将sk_buff交给上层协议进行处理。这里就是循环调用__skb_dequeue取出skb,直到所有的skb处理完毕为止。
              */
              skb = __skb_dequeue(&queue->input_pkt_queue);
              if (skb == NULL)
                     goto job_done;
              local_irq_enable();

              dev = skb->dev;

              netif_receive_skb(skb);

              dev_put(dev);

              work++;

              if (work >= quota || jiffies - start_time > 1)
                     break;
       }
}
继承事业,薪火相传
返回列表