?? ch14s06.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.6. 集成起來-Linux設備驅動第三版(中文版)</title><meta name="description" content="驅動開發" /><meta name="keywords" content="Linux設備驅動,中文版,第三版,ldd,linux device driver,驅動開發,電子版,程序設計,軟件開發,開發頻道" /><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 設備驅動 Edition 3"><link rel="up" href="ch14.html" title="第 14 章 Linux 設備模型"><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 設備模型</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>為更好理解驅動模型做什么, 讓我們通覽一個設備在內核中的生命周期的階段. 我們描述 PCI 子系統如何與驅動模型交互, 一個驅動如何被加入和去除的基本概念, 以及一個設備如何從系統中被加入和去除. 這些細節, 即便特別地描述 PCI 內核代碼, 適用所有其他的使用驅動核心來管理它們的驅動和設備的子系統.</p><p>PCI 核心, 驅動核心和單獨的 PCI 驅動之間的交互是非常復雜, 如同圖 <a href="ch14s06.html#ldd3-14-3.fig" title="圖 14.3. 創建設備過程">創建設備過程</a>所示.</p><div class="figure"><a name="ldd3-14-3.fig"></a><p class="title"><b>圖 14.3. 創建設備過程</b></p><div><img src="images/snagitldd3/ldd3-14-3.png" alt="創建設備過程"></div></div><div class="sect2" lang="zh-cn"><div class="titlepage"><div><div><h3 class="title"><a name="AddaDevice.sect2"></a>14.6.1. 添加一個設備</h3></div></div></div><p>PCI 子系統聲明一個單個 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>這個 pci_bus_type 變量被注冊到驅動內核, 當 PCI 子系統通過對 bus_register 的調用被加載入內核時. 當這個發生時, 驅動核心創建一個 sysfs 目錄在 /sys/bus/pci 里, 它包含 2 個目錄: devices 和 drivers.</p><p>所有的 PCI 驅動必須定義一個 struct pci_driver 變量, 它定義了這個 PCI 驅動能夠做的不同的功能(更多的關于 PCI 子系統和如何編寫一個 PCI 驅動的信息, 見 12 章). 那個結構包含一個 struct device_driver, 它接著被 PCI 核心初始化, 當 PCI 驅動被注冊時.</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>這個代碼為驅動建立總線來指向 pci_bus_type 以及使 probe 和 remove 函數來指向 PCI 核心內的函數. 驅動的 kobject 的 ktype 被設置為變量 pci_driver_kobj_type, 為使 PCI 驅動的屬性文件正常工作. 接著 PCI 核心注冊 PCI 驅動到驅動核心:</p><pre class="programlisting">/* register with core */error = driver_register(&drv->driver);</pre><p>驅動現在準備好被綁定到任何一個它支持的 PCI 設備.</p><p>PCI 核心, 在來自特定結構的實際和 PCI 總線交談的代碼的幫助下, 開始探測 PCI 地址空間, 查找所有的 PCI 設備. 當一個PCI 設備被發現, PCI 核心在內存中創建一個 struct pci_dev 類型的新變量. struct pci_dev 結構的一部分看來如下:</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>這個 PCI 設備的總線特定的成員被 PCI 核心初始化( devfn, vendor, device, 和其他成員), 并且 struct device 變量的 parent 變量被設置為這個 PCI 設備所在的 PCI 總線設備. bus 變量被設置指向 pci_bus_type 結構. 接下來 name 和 bus_id 變量被設置, 根據讀自 PCI 設備的 name 和 ID.</p><p>在 PCI 設備結構被初始化之后, 設備被注冊到驅動核心, 使用:</p><pre class="programlisting">device_register(&dev->dev); </pre><p>在 device_register 函數中, 驅動核心初始化設備的許多成員, 注冊設備的 kobject 到 kobject 核心( 它導致一個熱插拔事件產生, 但是我們在本章后面討論), 接著添加設備到驅動的 parent 所持有的設備列表中. 完成這個使所有的設備可被以正確的順序瀏覽, 一直知道每一個位于設備層次中哪里.</p><p>設備接著被添加到所有設備的總線特定的列表中, 在本例中, pci_bus_type 列表. 接著注冊到這個總線的所有驅動的列表被檢查, 并且總線的匹配功能被調用給每個驅動, 指定這個設備. 對于 pci_bus_type 總線, 匹配函數被 PCI 核心設定為指向 pci_bus_match 函數, 在設備被提交給驅動核心前.</p><p>pci_bus_match 函數轉換驅動核心傳遞給它的 struct device 為一個 struct pci_dev. 它還轉換 struct device_driver 為一個 struct pci_driver , 并接著查看設備的 PCI 設備特定信息和驅動, 看是否這個驅動聲明它能夠支持這類設備. 如果匹配不成功, 函數返回 0 給驅動核心, 并且驅動核心移向列表中的下一個驅動.</p><p>如果匹配成功, 函數返回 1 給驅動核心. 這使驅動核心設置struct device 中的驅動指針指向這個驅動, 并且接著調用在 struct device_driver 中特定的 probe 函數.</p>
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -