## 17.11.?MAC 地址解析
以太網通訊的一個有趣的方面是如何將 MAC 地址( 接口的唯一硬件 ID )和 IP 編號結合起來. 大部分協議有類似的問題, 但我們這里集中于類以太網的情況. 我們試圖提供這個問題的完整描述, 因此我們展示三個情形: ARP, 無 ARP 的以太網頭部( 例如 plip), 以及非以太網頭部.
### 17.11.1.?以太網使用 ARP
處理地址解析的通常方法是使用 Address Resolution Protocol (ARP). 幸運的是, ARP 由內核來管理, 并且一個以太網接口不需要做特別的事情來支持 ARP. 只要 dev->addr 和 dev->addr_len 在 open 時正確的賦值了, 驅動就不需要擔心解決 IP 編號對應于 MAC 地址; ether_setup 安排正確的設備方法給 dev->hard_header 和 dev_rebuild_header.
盡管通常內核處理地址解析的細節(并且緩存結果), 它需要接口驅動來幫助建立報文. 畢竟, 驅動知道物理層頭部細節, 然而網絡代碼的作者已經試圖隔離內核其他部分. 為此, 內核調用驅動的 hard_header 方法使用 ARP 查詢的結果來布置報文. 正常地, 以太網驅動編寫者不需要知道這個過程 -- 公共的以太網代碼負責了所有事情.
### 17.11.2.?不考慮 ARP
簡單的點對點網絡接口, 例如 plip, 可能從使用以太網頭部中受益, 而避免來回發送 ARP 報文的開銷. snull 中的例子代碼也屬于這一類的網絡設備. snull 不能使用 ARP 因為驅動改變發送報文中的 IP 地址, ARP 報文也交換 IP 地址. 盡管我們可能輕易實現了一個簡單 ARP 應答發生器, 更多的是演示性的來展示如何直接處理網絡層頭部.
如果你的設備想使用通常的硬件頭而不運行 ARP, 你需要重寫缺省的 dev->hard_header 方法. 這是 snull 的實現, 作為一個非常短的函數:
~~~
int snull_header(struct sk_buff *skb, struct net_device *dev,
unsigned short type, void *daddr, void *saddr,
unsigned int len)
{
struct ethhdr *eth = (struct ethhdr *)skb_push(skb,ETH_HLEN);
eth->h_proto = htons(type);
memcpy(eth->h_source, saddr ? saddr : dev->dev_addr, dev->addr_len);
memcpy(eth->h_dest, daddr ? daddr : dev->dev_addr, dev->addr_len);
eth->h_dest[ETH_ALEN-1] ^= 0x01; /* dest is us xor 1 */
return (dev->hard_header_len);
}
~~~
這個函數僅僅用內核提供的信息并把它格式成標準以太網頭. 它也翻轉目的以太網地址的 1 位, 理由下面敘述.
當接口收到一個報文, eth_type_trans 以幾種方法來使用硬件頭部. 我們已經在 snull_rx 看到這個調用.
~~~
skb->protocol = eth_type_trans(skb, dev);
~~~
這個函數抽取協議標識( ETH_P_IP, 在這個情況下 )從以太網頭; 它也賦值 skb->mac.raw, 從報文 data (使用 skb_pull)去掉硬件頭部, 并且設置 skb->pkt_type. 最后一項在 skb 分配是缺省為 PACKET_HOST(指示報文是發向這個主機的), eth_type_trans 改變它來反映以太網目的地址: 如果這個地址不匹配接收它的接口地址, pkt_type 成員被設為 PACKET_OTHERHOST. 結果, 除非接口處于混雜模式或者內核打開了報文轉發, netif_rx 丟棄任何類型為 PACKET_OTHERHOST 的報文. 因為這樣, snull_header 小心地使目的硬件地址匹配接收接口.
如果你的接口是點對點連接, 你不會想收到不希望的多播報文. 為避免這個問題, 記住, 第一個字節的最低位(LSB)為 0 的目的地址是方向一個單個主機(即, 要么 PACKET_HOST, 要么 PACKET_OTHERHOST). plip 驅動使用 0xfc 作為它的硬件地址的第一個字節, 而 snull 使用 0x00. 兩個地址都導致一個工作中的類似以太網的點對點連接.
### 17.11.3.?非以太網頭部
我們剛剛看過硬件頭部除目的地址外包含了一些信息, 最重要的是通訊協議. 我們現在描述硬件頭部如何用來封裝相關的信息. 如果你需要知道細節, 你可從內核源碼里抽取它們或者從特定傳送媒介的技術文檔中. 大部分驅動編寫者能夠忽略這個討論只是使用以太網實現.
值得一提的是不是所有信息都由每個協議提供. 一個點對點連接例如 plip 或者 snull 可能在不失去通用性的情況下避免傳送這個以太網頭部. hard_header 設備方法, 由 snull_header 實現所展示的, 接收自內核的遞交的信息( 協議級別和硬件地址 ). 它也在 type 參數中接收 16 位協議編號; IP, 例如, 標識為 ETH_P_IP. 驅動應該正確遞交報文數據和協議編號給接收主機. 一個點對點連接可能它的硬件頭部的地址, 只傳送協議編號, 因為保證遞交是獨立于源和目的地址的. 一個只有 IP 的連接甚至可能不發送任何硬件頭部.
當報文在連接的另一端被收到, 接收函數應當正確設置成員 skb->protocol, skb->pkt_type, 和 skb->mac.raw.
skb->mac.raw 是一個字符指針, 由在高層的網絡代碼(例如, net/ipv4/arp.c)所實現的地址解析機制使用. 它必須指向一個匹配 dev->type 的機器地址. 設備類型的可能的值在 <linux/if_arp.h> 中定義; 以太網接口使用 ARPHRD_ETHER. 例如, 這是 eth_type_trans 如何處理收到的報文的以太網頭:
~~~
skb->mac.raw = skb->data;
skb_pull(skb, dev->hard_header_len);
~~~
在最簡單的情況下( 一個沒有頭的點對點連接 ), skb->mac.raw 可指向一個靜態緩存, 包含接口的硬件地址, protocol 可設置為 ETH_P_IP, 并且 packet_type 可讓它是缺省的值 PACKET_HOST.
因為每個硬件類型是獨特的, 給出超出已經討論的特別的設備是困難的. 內核中滿是例子, 但是. 例如, 可查看 AppleTalk 驅動( drivers/net/appletalk/cops.c), 紅外驅動(例如, driver/net/irds/smc_ircc.c), 或者 PPP 驅動( drivers/net/ppp_generic.c).
- Linux設備驅動第三版
- 第 1 章 設備驅動簡介
- 1.1. 驅動程序的角色
- 1.2. 劃分內核
- 1.3. 設備和模塊的分類
- 1.4. 安全問題
- 1.5. 版本編號
- 1.6. 版權條款
- 1.7. 加入內核開發社團
- 1.8. 本書的內容
- 第 2 章 建立和運行模塊
- 2.1. 設置你的測試系統
- 2.2. Hello World 模塊
- 2.3. 內核模塊相比于應用程序
- 2.4. 編譯和加載
- 2.5. 內核符號表
- 2.6. 預備知識
- 2.7. 初始化和關停
- 2.8. 模塊參數
- 2.9. 在用戶空間做
- 2.10. 快速參考
- 第 3 章 字符驅動
- 3.1. scull 的設計
- 3.2. 主次編號
- 3.3. 一些重要數據結構
- 3.4. 字符設備注冊
- 3.5. open 和 release
- 3.6. scull 的內存使用
- 3.7. 讀和寫
- 3.8. 使用新設備
- 3.9. 快速參考
- 第 4 章 調試技術
- 4.1. 內核中的調試支持
- 4.2. 用打印調試
- 4.3. 用查詢來調試
- 4.4. 使用觀察來調試
- 4.5. 調試系統故障
- 4.6. 調試器和相關工具
- 第 5 章 并發和競爭情況
- 5.1. scull 中的缺陷
- 5.2. 并發和它的管理
- 5.3. 旗標和互斥體
- 5.4. Completions 機制
- 5.5. 自旋鎖
- 5.6. 鎖陷阱
- 5.7. 加鎖的各種選擇
- 5.8. 快速參考
- 第 6 章 高級字符驅動操作
- 6.1. ioctl 接口
- 6.2. 阻塞 I/O
- 6.3. poll 和 select
- 6.4. 異步通知
- 6.5. 移位一個設備
- 6.6. 在一個設備文件上的存取控制
- 6.7. 快速參考
- 第 7 章 時間, 延時, 和延后工作
- 7.1. 測量時間流失
- 7.2. 獲知當前時間
- 7.3. 延后執行
- 7.4. 內核定時器
- 7.5. Tasklets 機制
- 7.6. 工作隊列
- 7.7. 快速參考
- 第 8 章 分配內存
- 8.1. kmalloc 的真實故事
- 8.2. 后備緩存
- 8.3. get_free_page 和其友
- 8.4. 每-CPU 的變量
- 8.5. 獲得大量緩沖
- 8.6. 快速參考
- 第 9 章 與硬件通訊
- 9.1. I/O 端口和 I/O 內存
- 9.2. 使用 I/O 端口
- 9.3. 一個 I/O 端口例子
- 9.4. 使用 I/O 內存
- 9.5. 快速參考
- 第 10 章 中斷處理
- 10.1. 準備并口
- 10.2. 安裝一個中斷處理
- 10.3. 前和后半部
- 10.4. 中斷共享
- 10.5. 中斷驅動 I/O
- 10.6. 快速參考
- 第 11 章 內核中的數據類型
- 11.1. 標準 C 類型的使用
- 11.2. 安排一個明確大小給數據項
- 11.3. 接口特定的類型
- 11.4. 其他移植性問題
- 11.5. 鏈表
- 11.6. 快速參考
- 第 12 章 PCI 驅動
- 12.1. PCI 接口
- 12.2. 回顧: ISA
- 12.3. PC/104 和 PC/104+
- 12.4. 其他的 PC 總線
- 12.5. SBus
- 12.6. NuBus 總線
- 12.7. 外部總線
- 12.8. 快速參考
- 第 13 章 USB 驅動
- 13.1. USB 設備基礎知識
- 13.2. USB 和 sysfs
- 13.3. USB 的 Urbs
- 13.4. 編寫一個 USB 驅動
- 13.5. 無 urb 的 USB 傳送
- 13.6. 快速參考
- 第 14 章 Linux 設備模型
- 14.1. Kobjects, Ksets 和 Subsystems
- 14.2. 低級 sysfs 操作
- 14.3. 熱插拔事件產生
- 14.4. 總線, 設備, 和驅動
- 14.5. 類
- 14.6. 集成起來
- 14.7. 熱插拔
- 14.8. 處理固件
- 14.9. 快速參考
- 第 15 章 內存映射和 DMA
- 15.1. Linux 中的內存管理
- 15.2. mmap 設備操作
- 15.3. 進行直接 I/O
- 15.4. 直接內存存取
- 15.5. 快速參考
- 第 16 章 塊驅動
- 16.1. 注冊
- 16.2. 塊設備操作
- 16.3. 請求處理
- 16.4. 一些其他的細節
- 16.5. 快速參考
- 第 17 章 網絡驅動
- 17.1. snull 是如何設計的
- 17.2. 連接到內核
- 17.3. net_device 結構的詳情
- 17.4. 打開與關閉
- 17.5. 報文傳送
- 17.6. 報文接收
- 17.7. 中斷處理
- 17.8. 接收中斷緩解
- 17.9. 連接狀態的改變
- 17.10. Socket 緩存
- 17.11. MAC 地址解析
- 17.12. 定制 ioctl 命令
- 17.13. 統計信息
- 17.14. 多播
- 17.15. 幾個其他細節
- 17.16. 快速參考
- 第 18 章 TTY 驅動
- 18.1. 一個小 TTY 驅動
- 18.2. tty_driver 函數指針
- 18.3. TTY 線路設置
- 18.4. ioctls 函數
- 18.5. TTY 設備的 proc 和 sysfs 處理
- 18.6. tty_driver 結構的細節
- 18.7. tty_operaions 結構的細節
- 18.8. tty_struct 結構的細節
- 18.9. 快速參考