乡下人产国偷v产偷v自拍,国产午夜片在线观看,婷婷成人亚洲综合国产麻豆,久久综合给合久久狠狠狠9

  • <output id="e9wm2"></output>
    <s id="e9wm2"><nobr id="e9wm2"><ins id="e9wm2"></ins></nobr></s>

    • 分享

      Linux內核分析 - 網絡[三]:從netif_receive_skb()說起

       杰的個人圖書館 2012-04-11

      netif_receive_skb()函數中,可以看出處理的是像ARP、IP這些鏈路層以上的協(xié)議,那么,鏈路層報頭是在哪里去掉的呢?答案是網卡驅動中,在調用netif_receive_skb()前,

      skb->protocol = eth_type_trans(skb, bp->dev);

      該函數對處理后skb>data跳過以太網報頭,由mac_header指示以太網報頭:

      進入netif_receive_skb()函數

      list_for_each_entry_rcu(ptype,&ptype_base[ntohs(type) & PTYPE_HASH_MASK], list)

      按照協(xié)議類型依次由相應的協(xié)議模塊進行處理,而所以的協(xié)議模塊處理都會注冊在ptype_base中,實際是鏈表結構。

      net/core/dev.c

      static struct list_head ptype_base __read_mostly;   /* Taps */

       

      而相應的協(xié)議模塊是通過dev_add_pack()函數加入的:

      void dev_add_pack(struct packet_type *pt)

      {

           int hash;

       

           spin_lock_bh(&ptype_lock);

           if (pt->type == htons(ETH_P_ALL))

                    list_add_rcu(&pt->list, &ptype_all);

           else {

                    hash = ntohs(pt->type) & PTYPE_HASH_MASK;

                    list_add_rcu(&pt->list, &ptype_base[hash]);

           }

           spin_unlock_bh(&ptype_lock);

      }

       

      ARP處理為例

      該模塊的定義,它會在arp_init()中注冊進ptype_base鏈表中:

      static struct packet_type arp_packet_type __read_mostly = {

           .type =      cpu_to_be16(ETH_P_ARP),

           .func =      arp_rcv,

      };

       

      然后在根據報文的TYPE來在ptype_base中查找相應協(xié)議模塊進行處理時,實際調用arp_rcv()進行接收

      arp_rcv() --> arp_process()

      arp = arp_hdr(skb);

      ……

      arp_ptr= (unsigned char *)(arp+1);

      sha= arp_ptr;

      arp_ptr += dev->addr_len;

      memcpy(&sip, arp_ptr, 4);

      arp_ptr += 4;

      arp_ptr += dev->addr_len;

      memcpy(&tip, arp_ptr, 4);

      操作后這指針位置:

      然后判斷是ARP請求報文,這時先查詢路由表ip_route_input()

      if (arp->ar_op == htons(ARPOP_REQUEST) &&

               ip_route_input(skb, tip, sip, 0, dev) == 0)

      ip_route_input()函數中,先在cache中查詢是否存在相應的路由表項:

      hash = rt_hash(daddr, saddr, iif, rt_genid(net));

      緩存的路由項在內核中組織成hash表的形式,因此在查詢時,先算出的hash值,再用該項- rt_hash_table[hash].chain即可。這里可以看到,緩存路由項包括了源IP地址、目的IP地址、網卡號。

       

      如果在緩存中沒有查到匹配項,或指定不查詢cache,則查詢路由表ip_route_input_slow();

      進入ip_route_input_slow()函數,最終調用fib_lookup()得到查詢結果fib_result

      if ((err = fib_lookup(net, &fl, &res)) != 0)

      如果結果fib_result合法,則需要更新路由緩存,將此次查詢結果寫入緩存

      hash = rt_hash(daddr, saddr, fl.iif, rt_genid(net));

      err = rt_intern_hash(hash, rth, NULL, skb, fl.iif);

       

      在查找完路由表后,回到arp_process()函數,如果路由項指向本地,則應由本機接收該報文:

      if (addr_type == RTN_LOCAL) {

                    ……

                    if (!dont_send) {

                             n = neigh_event_ns(&arp_tbl, sha, &sip, dev);

                             if (n) {

                                       arp_send(ARPOP_REPLY,ETH_P_ARP,sip,dev,tip,sha,dev->dev_addr,sha);

                                       neigh_release(n);

                             }

                    }

                    goto out;

           }

      首先更新鄰居表neigh_event_ns(),然后發(fā)送ARP響應 – arp_send。

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

        本站是提供個人知識管理的網絡存儲空間,所有內容均由用戶發(fā)布,不代表本站觀點。請注意甄別內容中的聯(lián)系方式、誘導購買等信息,謹防詐騙。如發(fā)現(xiàn)有害或侵權內容,請點擊一鍵舉報。
        轉藏 分享 獻花(0

        0條評論

        發(fā)表

        請遵守用戶 評論公約

        類似文章 更多