?? ch14s06.html
字號(hào):
<html xmlns:cf="http://docbook.sourceforge.net/xmlns/chunkfast/1.0"><head><meta http-equiv="Content-Type" content="text/html; charset=gb2312"><title>14.6. 集成起來-Linux設(shè)備驅(qū)動(dòng)第三版(中文版)</title><meta name="description" content="驅(qū)動(dòng)開發(fā)" /><meta name="keywords" content="Linux設(shè)備驅(qū)動(dòng),中文版,第三版,ldd,linux device driver,驅(qū)動(dòng)開發(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ū)動(dòng) Edition 3"><link rel="up" href="ch14.html" title="第 14 章 Linux 設(shè)備模型"><link rel="prev" href="ch14s05.html" title="14.5. 類"><link rel="next" href="ch14s07.html" title="14.7. 熱插拔"></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.6. 集成起來</th></tr><tr><td width="20%" align="left"><a accesskey="p" href="ch14s05.html">上一頁</a> </td><th width="60%" align="center">第 14 章 Linux 設(shè)備模型</th><td width="20%" align="right"> <a accesskey="n" href="ch14s07.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="PuttingItAllTogether.sect1"></a>14.6. 集成起來</h2></div></div></div><p>為更好理解驅(qū)動(dòng)模型做什么, 讓我們通覽一個(gè)設(shè)備在內(nèi)核中的生命周期的階段. 我們描述 PCI 子系統(tǒng)如何與驅(qū)動(dòng)模型交互, 一個(gè)驅(qū)動(dòng)如何被加入和去除的基本概念, 以及一個(gè)設(shè)備如何從系統(tǒng)中被加入和去除. 這些細(xì)節(jié), 即便特別地描述 PCI 內(nèi)核代碼, 適用所有其他的使用驅(qū)動(dòng)核心來管理它們的驅(qū)動(dòng)和設(shè)備的子系統(tǒng).</p><p>PCI 核心, 驅(qū)動(dòng)核心和單獨(dú)的 PCI 驅(qū)動(dòng)之間的交互是非常復(fù)雜, 如同圖 <a href="ch14s06.html#ldd3-14-3.fig" title="圖 14.3. 創(chuàng)建設(shè)備過程">創(chuàng)建設(shè)備過程</a>所示.</p><div class="figure"><a name="ldd3-14-3.fig"></a><p class="title"><b>圖 14.3. 創(chuàng)建設(shè)備過程</b></p><div><img src="images/snagitldd3/ldd3-14-3.png" alt="創(chuàng)建設(shè)備過程"></div></div><div class="sect2" lang="zh-cn"><div class="titlepage"><div><div><h3 class="title"><a name="AddaDevice.sect2"></a>14.6.1. 添加一個(gè)設(shè)備</h3></div></div></div><p>PCI 子系統(tǒng)聲明一個(gè)單個(gè) struct bus_type 稱為 pci_bus_type, 它使用下列值初始化:</p><pre class="programlisting">struct bus_type pci_bus_type = { .name = "pci", .match = pci_bus_match, .hotplug = pci_hotplug, .suspend = pci_device_suspend, .resume = pci_device_resume, .dev_attrs = pci_dev_attrs, };</pre><p>這個(gè) pci_bus_type 變量被注冊(cè)到驅(qū)動(dòng)內(nèi)核, 當(dāng) PCI 子系統(tǒng)通過對(duì) bus_register 的調(diào)用被加載入內(nèi)核時(shí). 當(dāng)這個(gè)發(fā)生時(shí), 驅(qū)動(dòng)核心創(chuàng)建一個(gè) sysfs 目錄在 /sys/bus/pci 里, 它包含 2 個(gè)目錄: devices 和 drivers.</p><p>所有的 PCI 驅(qū)動(dòng)必須定義一個(gè) struct pci_driver 變量, 它定義了這個(gè) PCI 驅(qū)動(dòng)能夠做的不同的功能(更多的關(guān)于 PCI 子系統(tǒng)和如何編寫一個(gè) PCI 驅(qū)動(dòng)的信息, 見 12 章). 那個(gè)結(jié)構(gòu)包含一個(gè) struct device_driver, 它接著被 PCI 核心初始化, 當(dāng) PCI 驅(qū)動(dòng)被注冊(cè)時(shí).</p><pre class="programlisting">/* initialize common driver fields */drv->driver.name = drv->name;drv->driver.bus = &pci_bus_type;drv->driver.probe = pci_device_probe;drv->driver.remove = pci_device_remove; drv->driver.kobj.ktype = &pci_driver_kobj_type;</pre><p>這個(gè)代碼為驅(qū)動(dòng)建立總線來指向 pci_bus_type 以及使 probe 和 remove 函數(shù)來指向 PCI 核心內(nèi)的函數(shù). 驅(qū)動(dòng)的 kobject 的 ktype 被設(shè)置為變量 pci_driver_kobj_type, 為使 PCI 驅(qū)動(dòng)的屬性文件正常工作. 接著 PCI 核心注冊(cè) PCI 驅(qū)動(dòng)到驅(qū)動(dòng)核心:</p><pre class="programlisting">/* register with core */error = driver_register(&drv->driver);</pre><p>驅(qū)動(dòng)現(xiàn)在準(zhǔn)備好被綁定到任何一個(gè)它支持的 PCI 設(shè)備.</p><p>PCI 核心, 在來自特定結(jié)構(gòu)的實(shí)際和 PCI 總線交談的代碼的幫助下, 開始探測(cè) PCI 地址空間, 查找所有的 PCI 設(shè)備. 當(dāng)一個(gè)PCI 設(shè)備被發(fā)現(xiàn), PCI 核心在內(nèi)存中創(chuàng)建一個(gè) struct pci_dev 類型的新變量. struct pci_dev 結(jié)構(gòu)的一部分看來如下:</p><pre class="programlisting">struct pci_dev { /* ... */ unsigned int devfn; unsigned short vendor; unsigned short device; unsigned short subsystem_vendor; unsigned short subsystem_device; unsigned int class; /* ... */ struct pci_driver *driver; /* ... */ struct device dev; /* ... */}; </pre><p>這個(gè) PCI 設(shè)備的總線特定的成員被 PCI 核心初始化( devfn, vendor, device, 和其他成員), 并且 struct device 變量的 parent 變量被設(shè)置為這個(gè) PCI 設(shè)備所在的 PCI 總線設(shè)備. bus 變量被設(shè)置指向 pci_bus_type 結(jié)構(gòu). 接下來 name 和 bus_id 變量被設(shè)置, 根據(jù)讀自 PCI 設(shè)備的 name 和 ID.</p><p>在 PCI 設(shè)備結(jié)構(gòu)被初始化之后, 設(shè)備被注冊(cè)到驅(qū)動(dòng)核心, 使用:</p><pre class="programlisting">device_register(&dev->dev); </pre><p>在 device_register 函數(shù)中, 驅(qū)動(dòng)核心初始化設(shè)備的許多成員, 注冊(cè)設(shè)備的 kobject 到 kobject 核心( 它導(dǎo)致一個(gè)熱插拔事件產(chǎn)生, 但是我們?cè)诒菊潞竺嬗懻?, 接著添加設(shè)備到驅(qū)動(dòng)的 parent 所持有的設(shè)備列表中. 完成這個(gè)使所有的設(shè)備可被以正確的順序?yàn)g覽, 一直知道每一個(gè)位于設(shè)備層次中哪里.</p><p>設(shè)備接著被添加到所有設(shè)備的總線特定的列表中, 在本例中, pci_bus_type 列表. 接著注冊(cè)到這個(gè)總線的所有驅(qū)動(dòng)的列表被檢查, 并且總線的匹配功能被調(diào)用給每個(gè)驅(qū)動(dòng), 指定這個(gè)設(shè)備. 對(duì)于 pci_bus_type 總線, 匹配函數(shù)被 PCI 核心設(shè)定為指向 pci_bus_match 函數(shù), 在設(shè)備被提交給驅(qū)動(dòng)核心前.</p><p>pci_bus_match 函數(shù)轉(zhuǎn)換驅(qū)動(dòng)核心傳遞給它的 struct device 為一個(gè) struct pci_dev. 它還轉(zhuǎn)換 struct device_driver 為一個(gè) struct pci_driver , 并接著查看設(shè)備的 PCI 設(shè)備特定信息和驅(qū)動(dòng), 看是否這個(gè)驅(qū)動(dòng)聲明它能夠支持這類設(shè)備. 如果匹配不成功, 函數(shù)返回 0 給驅(qū)動(dòng)核心, 并且驅(qū)動(dòng)核心移向列表中的下一個(gè)驅(qū)動(dòng).</p><p>如果匹配成功, 函數(shù)返回 1 給驅(qū)動(dòng)核心. 這使驅(qū)動(dòng)核心設(shè)置struct device 中的驅(qū)動(dòng)指針指向這個(gè)驅(qū)動(dòng), 并且接著調(diào)用在 struct device_driver 中特定的 probe 函數(shù).</p>
?? 快捷鍵說明
復(fù)制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號(hào)
Ctrl + =
減小字號(hào)
Ctrl + -