<ruby id="bdb3f"></ruby>

    <p id="bdb3f"><cite id="bdb3f"></cite></p>

      <p id="bdb3f"><cite id="bdb3f"><th id="bdb3f"></th></cite></p><p id="bdb3f"></p>
        <p id="bdb3f"><cite id="bdb3f"></cite></p>

          <pre id="bdb3f"></pre>
          <pre id="bdb3f"><del id="bdb3f"><thead id="bdb3f"></thead></del></pre>

          <ruby id="bdb3f"><mark id="bdb3f"></mark></ruby><ruby id="bdb3f"></ruby>
          <pre id="bdb3f"><pre id="bdb3f"><mark id="bdb3f"></mark></pre></pre><output id="bdb3f"></output><p id="bdb3f"></p><p id="bdb3f"></p>

          <pre id="bdb3f"><del id="bdb3f"><progress id="bdb3f"></progress></del></pre>

                <ruby id="bdb3f"></ruby>

                合規國際互聯網加速 OSASE為企業客戶提供高速穩定SD-WAN國際加速解決方案。 廣告
                ## 17.2.?連接到內核 我們從分析 snull 的源碼來查看網絡驅動的結構開始. 把幾個驅動的源碼留在手邊, 對于下面的討論和得知真實世界中的 Linux 網絡驅動如何運行是會有幫助的. ### 17.2.1.?設備注冊 當一個驅動模塊加載進一個運行著的內核中, 它請求資源并提供功能; 這里沒有新內容. 并且在資源是如何請求上也沒有新東西. 驅動應當探測它的設備和它的硬件位置( I/O 端口和 IRQ 線 ) -- 但是不注冊它們 --如在第 10 章的" 安裝一個中斷處理程序 "中所述. 一個網絡驅動通過它的模塊初始化函數注冊的方式與字符和塊驅動是不同的. 因為沒有對等的主次編號給網絡接口, 一個網絡驅動不請求這樣一個號. 相反, 驅動為每個剛剛探測到的接口在一個全局的網絡設備列表里插入一個數據結構. 每個接口由一個結構 net_device 項來描述, 它在 <linux/netdevice.h> 里定義. snull 驅動留有指向兩個這樣結構的指針, 在一個簡單數組里. struct net_device *snull_devs[2]; net_device 結構, 如同許多其他內核結構, 包含一個 kobject, 以及因此它可被引用計數并通過 sysfs 輸出. 如同別的這樣的結構, 它必須動態分配. 進行這種分配的內核函數是 alloc_netdev, 它有下列原型: ~~~ struct net_device *alloc_netdev(int sizeof_priv, const char *name, void (*setup)(struct net_device *)); ~~~ 這里, sizeof_priv 是驅動的的"私有數據"區的大小; 對于網絡驅動, 這個區是同 net_device 結構一起分配的. 實際上, 這兩個是是在一個大內存塊中一起分配的, 但是驅動作者應當假裝不知道這一點. name 是這個接口的名子, 如同用戶空間看到的一樣; 這個名子可以有一個 printf 風格的 %d 在里面. 內核用下一個可用的接口號來替換這個 %d. 最后, setup 是一個初始化函數的指針, 被調用來設置 net_device 結構的剩余部分. 我們即將進入這個初始化函數, 但是現在, 為強化起見, snull 以這樣的方式分配它的兩個設備結構: ~~~ snull_devs[0] = alloc_netdev(sizeof(struct snull_priv), "sn%d", snull_init); snull_devs[1] = alloc_netdev(sizeof(struct snull_priv), "sn%d", snull_init); if (snull_devs[0] == NULL || snull_devs[1] == NULL) goto out; ~~~ 象通常一樣, 我們必須檢查返回值來確保分配成功. 網絡子系統為各種接口提供了一些幫助函數, 包裹著 alloc_netdev. 最通用的是 alloc_etherdev, 定義在 <linux/etherdevice.h>: ~~~ struct net_device *alloc_etherdev(int sizeof_priv); ~~~ 這個函數分配一個網絡設備使用 eth%d 作為參數 name. 它提供了自己的初始化函數 ( ether_setup )來設置幾個 net_device 字段, 使用對以太網設備合適的值. 因此, 沒有驅動提供的初始化函數給 alloc_etherdev; 驅動應當只完成它要求的初始化, 直接在一個成功的分配之后. 其他類型驅動的編寫者可能想利用這些幫助函數的其中一個, 例如 alloc_fcdev ( 定義在 <linux/fcdevice.h> ) 為 fiber-channel 設備, alloc_fddidev (<linux/fddidevice.h>) 為 FDDI 設備, 或者 aloc_trdev (<linux/trdevice.h>) 為令牌環設備. snull 可以順利使用 alloc_etherdev; 我們選擇使用 alloc_netdev 來代替, 作為演示低層接口的方式, 并且給我們控制安排給接口的名子. 一旦 net_device 結構完成初始化, 完成這個過程就只是傳遞這個結構給 register_netdev. 在 snull 中, 調用看來如同這樣: ~~~ for (i = 0; i < 2; i++) if ((result = register_netdev(snull_devs[i]))) printk("snull: error %i registering device \"%s\"\n", result, snull_devs[i]->name); ~~~ 一些經常的注意問題這里提一下: 在你調用 register_netdev 時, 你的驅動可能會馬上被調用來操作設備. 因此, 你不應當注冊設備直到所有東西都已經完全初始化. ### 17.2.2.?初始化每一個設備 我們已經看到了 net_device 結構的分配和注冊, 但是我們越過了中間的完全初始化這個結構的步驟. 注意 net_device 結構在運行時一直是放在一起; 它不能如同一個 file_operations 或者 block_device_opreations 結構一樣在編譯時設置. 必須在調用 register_netdev 之前完成初始化. net_device 結構又大又復雜; 幸運的是, 內核負責了一些以太網范圍中的缺省值, 通過 ether_setup 函數(由 alloc_etherdev 調用). 因為 snull 使用 alloc_netdev, 它有單獨的初始化函數. 該函數的核心( snull_init )如下: ~~~ ether_setup(dev); /* assign some of the fields */ dev->open = snull_open; dev->stop = snull_release; dev->set_config = snull_config; dev->hard_start_xmit = snull_tx; dev->do_ioctl = snull_ioctl; dev->get_stats = snull_stats; dev->rebuild_header = snull_rebuild_header; dev->hard_header = snull_header; dev->tx_timeout = snull_tx_timeout; dev->watchdog_timeo = timeout; /* keep the default flags, just add NOARP */ dev->flags |= IFF_NOARP; dev->features |= NETIF_F_NO_CSUM; dev->hard_header_cache = NULL; /* Disable caching */ ~~~ 上面的代碼是對 net_device 結構的例行初始化; 大部分是存儲我們的各種驅動函數指針. 代碼的單個不尋常的特性是設置 IFF_NOARP 在 flags 里面. 這個指出該接口不能使用 ARP. ARP 是一個低層以太網協議; 它的工作是將 IP 地址轉變成以太網介質存取控制 (MAC) 地址. 因為由 snull 模擬的遠程系統并不存在, 就沒人回答對它們的 ARP 請求. 不想因為增加 ARP 實現使 snull 變復雜, 我們選擇標識接口作為不能處理這個協議. 其中的對 hard_header_cache 賦值是同樣理由: 它關閉了這個接口的(不存在的) ARP 回答. 這個主題在本章后面的" MAC 地址解析"一節中詳述. 代碼初始化也設置了幾個和發送超時的處理有關的幾個變量( tx_timeout 和 watchdog_time ). 我們在"發送超時"一節完整地涉及這個主題. 我們現在看結構 net_device 的另一個成員, priv. 它的角色近似于我們用在字符驅動上的 private_data 指針. 不同于 fops->private_data, 這個 priv 指針是隨 net_device 結構一起分配的. 也不鼓勵直接存取 priv 成員, 由于性能和靈活性的原因. 當一個驅動需要存取私有數據指針, 應當使用 netdev_priv 函數. 因此, snull 驅動充滿著這樣的聲明: ~~~ struct snull_priv *priv = netdev_priv(dev); ~~~ snull 模塊聲明了一個 snull_priv 數據結構來給 priv 使用: ~~~ struct snull_priv { struct net_device_stats stats; int status; struct snull_packet *ppool; struct snull_packet *rx_queue; /* List of incoming packets */ int rx_int_enabled; int tx_packetlen; u8 *tx_packetdata; struct sk_buff *skb; spinlock_t lock; }; ~~~ 這個結構包括, 還有其他東西, 一個 net_device_stats 結構的實例, 這是放置接口統計量的標準地方. 下面的在 snull_init 中的各行分配并初始化 dev->priv: ~~~ priv = netdev_priv(dev); memset(priv, 0, sizeof(struct snull_priv)); spin_lock_init(&priv->lock); snull_rx_ints(dev, 1); /* enable receive interrupts */ ~~~ ### 17.2.3.?模塊卸載 模塊卸載時沒什么特別的. 模塊的清理函數只是注銷接口, 進行任何需要的內部清理, 釋放 net_device 結構回系統. ~~~ void snull_cleanup(void) { int i; for (i = 0; i < 2; i++) { if (snull_devs[i]) { unregister_netdev(snull_devs[i]); snull_teardown_pool(snull_devs[i]); free_netdev(snull_devs[i]); } } return; } ~~~ 對 unregister_netdev 的調用從系統中去除了接口; free_netdev 歸還 net_device 結構給內核. 如果某個地方有對這個結構的引用, 它可能繼續存在, 但是你的驅動不需要關心這個. 一旦你已經注銷了接口, 內核不再調用它的方法. 注意我們的內部清理( 在 snull_teardown_pool 里所做的 )直到已經注銷了設備后才能進行. 它必須, 但是, 在我們返回 net_device 結構給系統之前進行; 一旦我們已調用了 free_netdev, 我們再不能對這個設備或者我們的私有數據做任何引用.
                  <ruby id="bdb3f"></ruby>

                  <p id="bdb3f"><cite id="bdb3f"></cite></p>

                    <p id="bdb3f"><cite id="bdb3f"><th id="bdb3f"></th></cite></p><p id="bdb3f"></p>
                      <p id="bdb3f"><cite id="bdb3f"></cite></p>

                        <pre id="bdb3f"></pre>
                        <pre id="bdb3f"><del id="bdb3f"><thead id="bdb3f"></thead></del></pre>

                        <ruby id="bdb3f"><mark id="bdb3f"></mark></ruby><ruby id="bdb3f"></ruby>
                        <pre id="bdb3f"><pre id="bdb3f"><mark id="bdb3f"></mark></pre></pre><output id="bdb3f"></output><p id="bdb3f"></p><p id="bdb3f"></p>

                        <pre id="bdb3f"><del id="bdb3f"><progress id="bdb3f"></progress></del></pre>

                              <ruby id="bdb3f"></ruby>

                              哎呀哎呀视频在线观看