目前幾乎所有的驅動開發方面的參考書,內容結構都是先介紹介紹什么是Linux驅動,它分為哪些種類,然后是各種類型設備的驅動程序的內容細節。大都是只注重各種驅動本身的細節,而沒有站在一個全局整體的角度講解一下驅動開發的方法。這就導致,大多數的驅動開發者雖然可以正確的編寫驅動程序,但往往都是只知其一不知其二,知其然而不知其所以然。 此外,目前很多驅動開發者,即使是已經有多年經驗的開發者,在開發驅動的時候也就是填充填充driver的結構體,對于比較成熟的平臺,就是網上找個類似的驅動修改一下,即使寫十個百個千個驅動,也就是對某些硬件比較熟,遇到全新的芯片全新的平臺就束手無策。應該說這樣對驅動的理解是很有限的。 這也是目前linux驅動開發領域的現狀。如何改變這種情況呢? 首先,我們來認識一下linux驅動的基本面,理解一個新事物的的第一件事就是了解它的一些基本信息,就像我們人與人之間互相認識首先也是通過個人的基本信息一樣。 linux驅動在本質上就是一種軟件程序,上層軟件可以在不用了解硬件特性的情況下,通過驅動提供的接口,和計算機硬件進行通信。 系統調用是內核和應用程序之間的接口,而驅動程序是內核和硬件之間的接口,也就是內核和硬件之間的橋梁。它為應用程序屏蔽了硬件的細節,這樣在應用程序看來,硬件設備只是一個設備文件,應用程序可以象操作普通文件一樣對硬件設備進行操作。 linux驅動程序是內核的一部分,管理著系統中的設備控制器和相應的設備。它主要完成這么幾個功能:對設備初始化和釋放;傳送數據到硬件和從硬件讀取數據;檢測和處理設備出現的錯誤。 一般來說,一個驅動可以管理一種類型的設備。例如不同的U盤都屬于mass storage設備,我們不需要為每一個U盤編寫驅動,而只需要一個驅動就可以管理所有這些mass storage設備。 為方便我們加入各種驅動來支持不同的硬件,內核抽象出了很多層次結構,這些層次結構是linux設備驅動的上層。它們抽象出各種的驅動接口,驅動只需要填寫相應的回調函數,就能很容易把新的驅動添加到內核。 一般來說,linux驅動可以分為三類,就是塊設備驅動,字符設備驅動和網絡設備驅動。塊設備的讀寫都有緩存來支持,并且塊設備必須能夠隨機存取。塊設備驅動主要用于磁盤驅動器。 而字符設備的I/O操作沒有通過緩存。字符設備操作以字節為基礎,但不是說一次只能執行一個字節操作。例如對于字符設備我們可以通過mmap一次進行大量數據交換。字符設備實現比較簡單和靈活。 網絡設備在Linux里做專門的處理。Linux的網絡系統主要是基于BSD的socket機制。網絡設備驅動為網絡操作提供接口,管理網絡數據的接送和收發。為了屏蔽網絡環境中物理網絡設備的多樣性,Linux對所有的物理設備進行抽象并定義了一個統一的概念,稱之為接口(interface)。所有對網絡硬件的訪問都是通過接口進行的,接口對上層協議提供一致化的操作集合來處理基本數據的發送和接收,對下層屏蔽硬件差異。它與字符設備及塊設備不同之處其一就是網絡接口不存在于Linux的設備文件系統/dev/中。 和前一篇的介紹一樣,看完外表,我們再看內涵,就是Linux驅動的工作流程。大概有四個部分:使用insmod加載,模塊的初始化,進行設備操作,使用rmmod卸載。 Linux驅動有兩種存在形式,一種是直接編譯進內核,就是我們在配置內核的時候,在相應選項上選Y,另外一種就是編譯成模塊,按需加載和卸載。通常我們使用insmod命令完成模塊的加載,在加載時還可以指定模塊參數。另外一個常用的加載工具是modprobe,它與insmod的不同在于它會檢查模塊之間的依賴關系,將該模塊依賴的模塊也加載到內核。 每個驅動都有自己的初始化函數,完成一些新功能的注冊,這個初始化函數只是在初始化的時候被使用。在linux系統里,設備以文件的形式存在,應用程序可以通過open、read等函數操作設備,通過設備文件實現對設備的訪問。設備不再使用時,我們使用rmmod命令來卸載它,卸載的過程會調用到驅動的推出函數,每個驅動都必須有一個退出函數,沒有的話,內核就不會允許去卸載它。 在對linux驅動的外表和內涵都有了一個初步的認識之后,我們來看看作為一個驅動開發者,我們需要注意哪些問題。 首先,對模塊機制的了解是開發linux驅動的基礎,因為我們編寫驅動的過程也就是在編寫一個內核模塊的過程。早期版本的內核是整體式的,也就是說所有的部分都靜態地連接成一個很大的執行文件。但是現在的內核采用的是新的機制,即模塊機制:許多功能包含在模塊內,當你需要時可以使用insmod去擁抱它,將它動態地載入到內核里,當你不需要時,則可以使用rmmod將它一腳踢開。這就使得kernel的內核很小,而且在運行的時候可以不用reboot就能夠載入和替代模塊。 其次,我們要注重對設備模型的理解。其實從2.6內核開始,隨著設備模型的出現,驅動的開發就不再是個困難的問題,毫不夸張得說,理解了設備模型,再去看那些五花八門的驅動程序,你會發現自己站在了另一個高度,從而有了一種俯視的感覺。不過貌似大部分驅動開發者都沒意識到這個問題。 最后,是要養成使用協議的spec、設備的datasheet、內核參考代碼去解決問題的習慣,而不是一碰到問題就到處尋找所謂的牛人去問怎么解決。 想學習的你和我聯系預約就可以免費聽課了。 以下課程可免費試聽C語言、電子、PCB、STM32、Linux、FPGA、JAVA、安卓等。 宋工企鵝號:3524-6590-88 Tel/WX:173--1795--1908 |