?? ch14s07.html
字號:
<html xmlns:cf="http://docbook.sourceforge.net/xmlns/chunkfast/1.0"><head><meta http-equiv="Content-Type" content="text/html; charset=gb2312"><title>14.7. 熱插拔-Linux設(shè)備驅(qū)動第三版(中文版)</title><meta name="description" content="驅(qū)動開發(fā)" /><meta name="keywords" content="Linux設(shè)備驅(qū)動,中文版,第三版,ldd,linux device driver,驅(qū)動開發(fā),電子版,程序設(shè)計(jì),軟件開發(fā),開發(fā)頻道" /><meta name="verify-v1" content="5asbXwkS/Vv5OdJbK3Ix0X8osxBUX9hutPyUxoubhes=" /><link rel="stylesheet" href="docbook.css" type="text/css"><meta name="generator" content="DocBook XSL Stylesheets V1.69.0"><link rel="start" href="index.html" title="Linux 設(shè)備驅(qū)動 Edition 3"><link rel="up" href="ch14.html" title="第 14 章 Linux 設(shè)備模型"><link rel="prev" href="ch14s06.html" title="14.6. 集成起來"><link rel="next" href="ch14s08.html" title="14.8. 處理固件"></head><body bgcolor="white" text="black" link="#0000FF" vlink="#840084" alink="#0000FF"><div class="navheader"><table width="100%" summary="Navigation header"><tr><th colspan="3" align="center">14.7. 熱插拔</th></tr><tr><td width="20%" align="left"><a accesskey="p" href="ch14s06.html">上一頁</a> </td><th width="60%" align="center">第 14 章 Linux 設(shè)備模型</th><td width="20%" align="right"> <a accesskey="n" href="ch14s08.html">下一頁</a></td></tr></table><hr></div><div class="sect1" lang="zh-cn"><div class="titlepage"><div><div><h2 class="title" style="clear: both"><a name="Hotplug.sect1"></a>14.7. 熱插拔</h2></div></div></div><p>有 2 個(gè)不同方法來看熱插拔. 內(nèi)核看待熱插拔為硬件, 內(nèi)核和內(nèi)核驅(qū)動之間的交互. 用戶看待熱插拔是內(nèi)核和用戶空間的通過稱為 /sbin/hotplug 的程序的交互. 這個(gè)程序被內(nèi)核調(diào)用, 當(dāng)它想通知用戶空間某種熱插拔事件剛剛在內(nèi)核中發(fā)生.</p><div class="sect2" lang="zh-cn"><div class="titlepage"><div><div><h3 class="title"><a name="DynamicDevices.sect2"></a>14.7.1. 動態(tài)設(shè)備</h3></div></div></div><p>術(shù)語"熱插拔"最普遍使用的意義產(chǎn)生于當(dāng)討論這樣的事實(shí)時(shí), 幾乎所有的計(jì)算機(jī)系統(tǒng)現(xiàn)在能夠處理當(dāng)系統(tǒng)有電時(shí)設(shè)備的出現(xiàn)或消失. 這非常不同于只是幾年前的計(jì)算機(jī)系統(tǒng), 那時(shí)程序員知道他們只需要在啟動時(shí)掃描所有的設(shè)備, 并且他們從不必?fù)?dān)心他們的設(shè)備消失直到整個(gè)機(jī)器被關(guān)電. 現(xiàn)在, 隨著 USB 的出現(xiàn), CardBus, PCMCIA, IEEE1394, 和 PCI 熱插拔控制器, Linux 內(nèi)核需要能夠可靠地運(yùn)行不管什么硬件從系統(tǒng)中增加或去除. 這產(chǎn)生了一個(gè)額外的負(fù)擔(dān)給設(shè)備驅(qū)動作者, 因?yàn)楝F(xiàn)在他們必須一直處理一個(gè)沒有任何通知而突然從地下冒出來的設(shè)備.</p><p>每個(gè)不同的總線類型以不同方式處理一個(gè)設(shè)備的消失. 例如, 當(dāng)一個(gè) PCI , CardBus, 或者 PCMCIA 設(shè)備從系統(tǒng)中去除, 在驅(qū)動通過它的去除函數(shù)被通知之前常常是一會兒. 在發(fā)生這個(gè)前, 所有的從 PCI 的讀返回所有的位集合. 這意味著驅(qū)動需要一直檢查它們從 PCI 總線讀取的值并且能夠正確處理 0xff 值.</p><p>這個(gè)的一個(gè)例子可在 drivers/usb/host/ehci-hcd.c 驅(qū)動中見到, 它是一個(gè) PCI 驅(qū)動給一個(gè) UBS 2.0(高速)控制卡. 它有下面的代碼在它的主握手循環(huán)中來探測是否控制塊已經(jīng)從系統(tǒng)中去除.</p><pre class="programlisting">result = readl(ptr);if (result == ~(u32)0) /* card removed */ return -ENODEV; </pre><p>對于 USB 驅(qū)動, 當(dāng)一個(gè) USB 驅(qū)動被綁定到的設(shè)備被從系統(tǒng)中去除, 任何掛起的已被提交給設(shè)備的 urbs 以錯(cuò)誤 -ENODEV 失敗. 如果發(fā)生這個(gè)情況, 驅(qū)動需要識別這個(gè)錯(cuò)誤并且正確清理任何掛起的 I/O .</p><p>可熱插拔的設(shè)備不只限于傳統(tǒng)的設(shè)備, 例如鼠標(biāo), 鍵盤, 和網(wǎng)卡. 有大量的系統(tǒng)現(xiàn)在支持整個(gè) CPU 和內(nèi)存條的移出. 幸運(yùn)地, Linux 內(nèi)核正確處理這些核心"系統(tǒng)"設(shè)備的加減, 以至于單個(gè)設(shè)備驅(qū)動不需要注意這些事情.</p></div><div class="sect2" lang="zh-cn"><div class="titlepage"><div><div><h3 class="title"><a name="ThesbinhotplugUtility.sect2"></a>14.7.2. /sbin/hotplug 工具</h3></div></div></div><p>如同本章中前面提過的, 無論何時(shí)一個(gè)設(shè)備從系統(tǒng)中增刪, 都產(chǎn)生一個(gè)"熱插拔事件". 這意味著內(nèi)核調(diào)用用戶空間程序 /sbin/hotplug. 這個(gè)程序典型地是一個(gè)非常小的 bash 腳本, 只傳遞執(zhí)行給一系列其他的位于 /etc/hot-plug.d/ 目錄樹的程序. 對于大部分的 Linux 發(fā)布, 這個(gè)腳本看來如下:</p><pre class="screen">DIR="/etc/hotplug.d"for I in "${DIR}/$1/"*.hotplug "${DIR}/"default/*.hotplug ; do if [ -f $I ]; then test -x $I && $I $1 ; fidoneexit 1</pre><p>換句話說, 這個(gè)腳本搜索所有的有 .hotplug 后綴的可能對這個(gè)事件感興趣的程序并調(diào)用它們, 傳遞給它們許多不同的環(huán)境變量, 這些環(huán)境變量已經(jīng)被內(nèi)核設(shè)置. 更多關(guān)于 /sbin/hotplug 腳本如何工作的細(xì)節(jié)可在程序的注釋中找到, 以及在 hotplug(8)手冊頁中.</p><p>如同前面提到的, /sbin/hotplug 被調(diào)用無論何時(shí)一個(gè) kobject 被創(chuàng)建或銷毀. 熱插拔程序被用一個(gè)提供事件名子的單個(gè)命令行參數(shù)調(diào)用. 核心內(nèi)核和涉及到的特定子系統(tǒng)也設(shè)定一系列帶有關(guān)于發(fā)生了什么的信息的環(huán)境變量(下面描述). 這些變量被熱插拔程序使用來判定剛剛在內(nèi)核發(fā)生了什么, 以及是否有任何特定的動作應(yīng)當(dāng)采取.</p><p>傳遞給 /sbin/hotplug 的命令行參數(shù)是關(guān)聯(lián)這個(gè)熱插拔事件的名子, 如同分配給 kobject 的 kset 所決定的. 這個(gè)名子可通過一個(gè)對屬于本章前面描述過的 kset 的 hotplug_ops 結(jié)構(gòu)的 name 函數(shù)的調(diào)用來設(shè)定; 如果那個(gè)函數(shù)不存在或者從未被調(diào)用, 名子是 kset 自身的名子.</p><p>一直為 /sbin/hotplug 設(shè)定的缺省的環(huán)境變量是:</p><div class="variablelist"><dl><dt><span class="term"><span>ACTION </span></span></dt><dd><p>這個(gè)字符串 add 或 remove, 只根據(jù)是否這個(gè)對象是被創(chuàng)建或者銷毀.</p></dd><dt><span class="term"><span>DEVPATH </span></span></dt><dd><p>一個(gè)目錄路徑, 在 sysfs 文件系統(tǒng)中, 它指向在被創(chuàng)建或銷毀的 kobject. 注意 sysfs 文件系統(tǒng)的加載點(diǎn)不是添加到這路徑, 因此是由用戶空間程序來決定這個(gè). </p></dd><dt><span class="term"><span>SEQNUM </span></span></dt><dd><p>這個(gè)熱插拔事件的順序號. 順序號是一個(gè) 64-位 數(shù), 它每次產(chǎn)生熱插拔事件都遞增. 這允許用戶空間以內(nèi)核產(chǎn)生它們的順序來排序熱插拔事件, 因?yàn)閷σ粋€(gè)用戶空間程序可能亂序運(yùn)行.</p></dd><dt><span class="term"><span>SUBSYSTEM </span></span></dt><dd><p>同樣的字符串作為前面描述的命令行參數(shù)傳遞.</p></dd></dl></div><p>許多不同的總線子系統(tǒng)都添加它們自己的環(huán)境變量到 /sbin/hotplug 調(diào)用中, 當(dāng)關(guān)聯(lián)到總線的設(shè)備被添加或從系統(tǒng)中去除. 它們在它們的熱插拔回調(diào)中做這個(gè), 這個(gè)回調(diào)在分配給它們的總線(如同在"熱插拔操作"一節(jié)中描述的)的 struct kset_hotplug_ops 中指定. 這允許用戶空間能夠自動加載必要的可能需要來控制這個(gè)被總線發(fā)現(xiàn)的設(shè)備的模塊. 這里是一個(gè)不同總線類型的列表以及它們添加到 /sbin/hotplug 調(diào)用中的環(huán)境變量. </p><div class="sect3" lang="zh-cn"><div class="titlepage"><div><div><h4 class="title"><a name="IEEE1394FireWire.sect3"></a>14.7.2.1. IEEE1394(火線)</h4></div></div></div><p>任何在 IEEE1394 總線, 也是火線, 上的設(shè)備, 由 /sbin/hotplug 參數(shù)名和 SUBSYSTEM 環(huán)境變量設(shè)置為值 ieee1394. ieee1394 子系統(tǒng)也總是添加下列 4 個(gè)環(huán)境變量:</p><div class="variablelist"><dl><dt><span class="term"><span>VENDOR_ID </span></span></dt><dd><p>IEEE1394 的 24-位 供應(yīng)者 ID. </p></dd><dt><span class="term"><span>MODEL_ID </span></span></dt><dd><p>IEEE1394 的 24-位型號 ID.</p></dd><dt><span class="term"><span>GUID </span></span></dt><dd><p>設(shè)備的 64-位 GUID.</p></dd><dt><span class="term"><span>SPECIFIER_ID </span></span></dt><dd><p>24-位值, 指定設(shè)備的協(xié)議規(guī)格的擁有者.</p></dd><dt><span class="term"><span>VERSION </span></span></dt><dd><p>指定設(shè)備協(xié)議規(guī)格的版本的值</p></dd></dl></div></div><div class="sect3" lang="zh-cn"><div class="titlepage"><div><div><h4 class="title"><a name="Networking.sect3"></a>14.7.2.2. 網(wǎng)絡(luò)</h4></div></div></div><p>所有的網(wǎng)絡(luò)設(shè)備都創(chuàng)建一個(gè)熱插拔事件, 當(dāng)設(shè)備注冊或者注銷在內(nèi)核. /sbin/hotplug 調(diào)用有參數(shù) name 和 SUBSYSTEM 環(huán)境變量設(shè)置為 net, 并且只添加下列環(huán)境變量:</p><div class="variablelist"><dl><dt><span class="term"><span>INTERFACE </span></span></dt><dd><p>已經(jīng)從內(nèi)核注冊或注銷的接口的名子. 這個(gè)的例子是 lo 和 eth0.</p></dd></dl></div></div><div class="sect3" lang="zh-cn"><div class="titlepage"><div><div><h4 class="title"><a name="PCI.sect3"></a>14.7.2.3. PCI 總線</h4></div></div></div><p>任何在 PCI 總線上的設(shè)備有參數(shù) name 和 SUBSYSTEM 環(huán)境變量設(shè)置為值 pci. PCI 子系統(tǒng)也一直添加下面 4 個(gè)環(huán)境變量:</p><div class="variablelist"><dl><dt><span class="term"><span>PCI_CLASS </span></span></dt><dd><p>設(shè)備的 PCI 類號, 16 進(jìn)制.</p></dd><dt><span class="term"><span>PCI_ID </span></span></dt><dd><p>設(shè)備的 PCI 供應(yīng)商和設(shè)備 ID, 16進(jìn)制, 結(jié)合成這樣的格式 供應(yīng)者:設(shè)備.</p></dd><dt><span class="term"><span>PCI_SUBSYS_ID </span></span></dt><dd><p>PCI 子系統(tǒng)供應(yīng)商和子系統(tǒng)設(shè)備 ID, 以 子系統(tǒng)供應(yīng)者:子系統(tǒng)設(shè)備 的格式結(jié)合.</p></dd><dt><span class="term"><span>PCI_SLOT_NAME </span></span></dt><dd><p>PCI 插口"名", 內(nèi)核給予這個(gè)設(shè)備的. 它以這樣的格式 域:總線:插口:功能. 一個(gè)例子可能是: 0000:00:0d.0.</p></dd></dl></div></div><div class="sect3" lang="zh-cn"><div class="titlepage"><div><div><h4 class="title"><a name="Input.sect3"></a>14.7.2.4. 輸入</h4></div></div></div><p>對所有的輸入設(shè)備(鼠標(biāo), 鍵盤, 游戲桿, 等等), 一個(gè)熱插拔事件當(dāng)設(shè)備從內(nèi)核增減時(shí)產(chǎn)生. /sbin/hotplug 參數(shù)和 SUBSYSTEM 環(huán)境變量被設(shè)置為值 input. 輸入子系統(tǒng)也總是添加下面的環(huán)境變量:</p><div class="variablelist"><dl><dt><span class="term"><span>PRODUCT </span></span></dt><dd><p>一個(gè)多值字符串, 用 16 進(jìn)制列出值沒有前導(dǎo) 0. 它的格式是 bustype:vender:product:version.</p></dd></dl></div><p>下列環(huán)境變量可能出現(xiàn), 如果設(shè)備支持它:</p><div class="variablelist"><dl><dt><span class="term"><span>NAME </span></span></dt><dd><p>輸入設(shè)備的名子, 如同設(shè)備給定的.</p></dd>
?? 快捷鍵說明
復(fù)制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -