0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Linux設(shè)備驅(qū)動開發(fā)中兩個重要思想

Q4MP_gh_c472c21 ? 來源:互聯(lián)網(wǎng) ? 作者:佚名 ? 2018-04-21 08:35 ? 次閱讀

嵌入式Linux設(shè)備驅(qū)動開發(fā)之驅(qū)動分層/分離思想

我們在學(xué)習(xí)I2C、USB、SD驅(qū)動時,有沒有發(fā)現(xiàn)一個共性,就是在驅(qū)動開發(fā)時,每個驅(qū)動都分層三部分,由上到下分別是:

1、XXX 設(shè)備驅(qū)動

2、XXX 核心層

3、XXX 主機控制器驅(qū)動

而需要我們編寫的主要是設(shè)備驅(qū)動部分,主機控制器驅(qū)動部分也有少量編寫,二者進行交互主要時由核心層提供的接口來實現(xiàn);這樣結(jié)構(gòu)清晰,大大地有利于我們的驅(qū)動開發(fā),這其中就是利用了Linux設(shè)備驅(qū)動開發(fā)中兩個重要思想,下面來一一解析

一、設(shè)備驅(qū)動的分層思想

在面向?qū)ο蟮某绦蛟O(shè)計中,可以為某一類相似的事物定義一個基類,而具體的事物可以繼承這個基類中的函數(shù)。如果對于繼承的這個事物而言,其某函數(shù)的實現(xiàn)與基類一致,那它就可以直接繼承基類的函數(shù);相反,它可以重載之。這種面向?qū)ο蟮脑O(shè)計思想極大地提高了代碼的可重用能力,是對現(xiàn)實世界事物間關(guān)系的一種良好呈現(xiàn)。

Linux內(nèi)核完全由C語言匯編語言寫成,但是卻頻繁用到了面向?qū)ο蟮脑O(shè)計思想。在設(shè)備驅(qū)動方面,往往為同類的設(shè)備設(shè)計了一個框架,而框架中的核心層則實現(xiàn)了該設(shè)備通用的一些功能。同樣的,如果具體的設(shè)備不想使用核心層的函數(shù),它可以重載之。舉個例子:

return_type core_funca(xxx_device * bottom_dev, param1_type param1, param1_type param2)

{

if (bottom_dev->funca)

return bottom_dev->funca(param1, param2);

/* 核心層通用的funca代碼 */

...

}

上述core_funca的實現(xiàn)中,會檢查底層設(shè)備是否重載了funca(),如果重載了,就調(diào)用底層的代碼,否則,直接使用通用層的。這樣做的好處是,核心層的代碼可以處理絕大多數(shù)該類設(shè)備的funca()對應(yīng)的功能,只有少數(shù)特殊設(shè)備需要重新實現(xiàn)funca()。

再看一個例子:

copyreturn_type core_funca(xxx_device * bottom_dev, param1_type param1, param1_type param2)

{

/*通用的步驟代碼A */

...

bottom_dev->funca_ops1();

/*通用的步驟代碼B */

...

bottom_dev->funca_ops2();

/*通用的步驟代碼C */

...

bottom_dev->funca_ops3();

}

上述代碼假定為了實現(xiàn)funca(),對于同類設(shè)備而言,操作流程一致,都要經(jīng)過“通用代碼A、底層ops1、通用代碼B、底層ops2、通用代碼C、底層ops3”這幾步,分層設(shè)計明顯帶來的好處是,對于通用代碼A、B、C,具體的底層驅(qū)動不需要再實現(xiàn),而僅僅只關(guān)心其底層的操作ops1、ops2、ops3。圖1明確反映了設(shè)備驅(qū)動的核心層與具體設(shè)備驅(qū)動的關(guān)系,實際上,這種分層可能只有2層(圖1的a),也可能是多層的(圖1的b)。

這樣的分層化設(shè)計在Linux的input、RTC、MTD、I2 C、SPI、TTY、USB等諸多設(shè)備驅(qū)動類型中屢見不鮮。

二、主機驅(qū)動和外設(shè)驅(qū)動分離思想

主機、外設(shè)驅(qū)動分離的意義

在Linux設(shè)備驅(qū)動框架的設(shè)計中,除了有分層設(shè)計實現(xiàn)以外,還有分隔的思想。舉一個簡單的例子,假設(shè)我們要通過SPI總線訪問某外設(shè),在這個訪問過程中,要通過操作CPU XXX上的SPI控制器的寄存器來達到訪問SPI外設(shè)YYY的目的,最簡單的方法是:

copyreturn_type xxx_write_spi_yyy(...)

{

xxx_write_spi_host_ctrl_reg(ctrl);

xxx_ write_spi_host_data_reg(buf);

while(!(xxx_spi_host_status_reg()&SPI_DATA_TRANSFER_DONE));

...

}

如果按照這種方式來設(shè)計驅(qū)動,結(jié)果是對于任何一個SPI外設(shè)來講,它的驅(qū)動代碼都是CPU相關(guān)的。也就是說,當(dāng)然用在CPU XXX上的時候,它訪問XXX的SPI主機控制寄存器,當(dāng)用在XXX1的時候,它訪問XXX1的SPI主機控制寄存器:

return_type xxx1_write_spi_yyy(...)

{

xxx1_write_spi_host_ctrl_reg(ctrl);

xxx1_ write_spi_host_data_reg(buf);

while(!(xxx1_spi_host_status_reg()&SPI_DATA_TRANSFER_DONE));

...

}

這顯然是不能接受的,因為這意味著外設(shè)YYY用在不同的CPU XXX和XXX1上的時候需要不同的驅(qū)動。那么,我們可以用如圖的思想對主機控制器驅(qū)動和外設(shè)驅(qū)動進行分離。這樣的結(jié)構(gòu)是,外設(shè)a、b、c的驅(qū)動與主機控制器A、B、C的驅(qū)動不相關(guān),主機控制器驅(qū)動不關(guān)心外設(shè),而外設(shè)驅(qū)動也不關(guān)心主機,外設(shè)只是訪問核心層的通用的API進行數(shù)據(jù)傳輸,主機和外設(shè)之間可以進行任意的組合。

如果我們不進行上圖的主機和外設(shè)分離,外設(shè)a、b、c和主機A、B、C進行組合的時候,需要9個不同的驅(qū)動。設(shè)想一共有m個主機控制器,n個外設(shè),分離的結(jié)果是需要m+n個驅(qū)動,不分離則需要m*n個驅(qū)動。

Linux SPI、I2C、USB、ASoC(ALSA SoC)等子系統(tǒng)都典型地利用了這種分離的設(shè)計思想。


聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 嵌入式
    +關(guān)注

    關(guān)注

    5059

    文章

    18973

    瀏覽量

    302033
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11207

    瀏覽量

    208717

原文標(biāo)題:嵌入式Linux設(shè)備驅(qū)動開發(fā)之驅(qū)動分層/分離思想

文章出處:【微信號:gh_c472c2199c88,微信公眾號:嵌入式微處理器】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    Linux驅(qū)動開發(fā):字符設(shè)備驅(qū)動開發(fā)理論

    大部分學(xué)習(xí)者的最終目的就是學(xué)習(xí) Linux驅(qū)動開發(fā),Linux的外設(shè)驅(qū)動可以分為:字符
    發(fā)表于 10-26 09:53 ?1064次閱讀

    Linux設(shè)備驅(qū)動開發(fā)詳解

    #《Linux設(shè)備驅(qū)動開發(fā)詳解》電子書連載#第5章 Linux文件系統(tǒng)與設(shè)備文件系統(tǒng),由于字符
    發(fā)表于 06-06 11:28

    Linux設(shè)備驅(qū)動程序開發(fā) - Linux設(shè)備驅(qū)動類型

    的基本思想就是盡量抽取設(shè)備特性的共同點,把最復(fù)雜的設(shè)備驅(qū)動架構(gòu)程序代碼交給Linux的內(nèi)核和驅(qū)動
    發(fā)表于 04-16 14:31

    嵌入式Linux設(shè)備驅(qū)動開發(fā)思想進階之驅(qū)動分層與驅(qū)動分離

    的主要是設(shè)備驅(qū)動部分,主機控制器驅(qū)動部分也有少量編寫,二者進行交互主要時由核心層提供的接口來實現(xiàn);這樣結(jié)構(gòu)清晰,大大地有利于我們的驅(qū)動開發(fā),
    發(fā)表于 11-03 10:03

    嵌入式Linux設(shè)備驅(qū)動開發(fā)思想進階之驅(qū)動分層與驅(qū)動分離

    的主要是設(shè)備驅(qū)動部分,主機控制器驅(qū)動部分也有少量編寫,二者進行交互主要時由核心層提供的接口來實現(xiàn);這樣結(jié)構(gòu)清晰,大大地有利于我們的驅(qū)動開發(fā),
    發(fā)表于 11-07 14:43

    Linux設(shè)備驅(qū)動開發(fā)兩個重要思想解析

    嵌入式Linux設(shè)備驅(qū)動開發(fā)思想進階之驅(qū)動分層與驅(qū)動
    發(fā)表于 02-19 06:40

    Linux設(shè)備驅(qū)動開發(fā)詳解》第9章、Linux設(shè)備驅(qū)動的異步通知與異步IO

    Linux設(shè)備驅(qū)動開發(fā)詳解》第9章、Linux設(shè)備驅(qū)動
    發(fā)表于 10-27 11:33 ?0次下載
    《<b class='flag-5'>Linux</b><b class='flag-5'>設(shè)備</b><b class='flag-5'>驅(qū)動</b><b class='flag-5'>開發(fā)</b>詳解》第9章、<b class='flag-5'>Linux</b><b class='flag-5'>設(shè)備</b><b class='flag-5'>驅(qū)動</b><b class='flag-5'>中</b>的異步通知與異步IO

    Linux設(shè)備驅(qū)動開發(fā)詳解》第8章、Linux設(shè)備驅(qū)動的阻塞與非阻塞IO

    Linux設(shè)備驅(qū)動開發(fā)詳解》第8章、Linux設(shè)備驅(qū)動
    發(fā)表于 10-27 11:35 ?9次下載
    《<b class='flag-5'>Linux</b><b class='flag-5'>設(shè)備</b><b class='flag-5'>驅(qū)動</b><b class='flag-5'>開發(fā)</b>詳解》第8章、<b class='flag-5'>Linux</b><b class='flag-5'>設(shè)備</b><b class='flag-5'>驅(qū)動</b><b class='flag-5'>中</b>的阻塞與非阻塞IO

    Linux設(shè)備驅(qū)動開發(fā)詳解》第7章、Linux設(shè)備驅(qū)動的并發(fā)控制

    Linux設(shè)備驅(qū)動開發(fā)詳解》第7章、Linux設(shè)備驅(qū)動
    發(fā)表于 10-27 11:37 ?10次下載
    《<b class='flag-5'>Linux</b><b class='flag-5'>設(shè)備</b><b class='flag-5'>驅(qū)動</b><b class='flag-5'>開發(fā)</b>詳解》第7章、<b class='flag-5'>Linux</b><b class='flag-5'>設(shè)備</b><b class='flag-5'>驅(qū)動</b><b class='flag-5'>中</b>的并發(fā)控制

    Linux USB總線的兩個鏈表

    USB 總線引出兩個首要 的鏈表,一為 USB 設(shè)備鏈表,一為 USB 驅(qū)動鏈表。
    發(fā)表于 04-20 10:33 ?954次閱讀

    Linux設(shè)備驅(qū)動思想在STM32編程的應(yīng)用

    這幾天看了一下Linux設(shè)備驅(qū)動,發(fā)現(xiàn)這套思想其實也可以用在普通的單片機編程上。這種思想較好的分割了驅(qū)動
    發(fā)表于 12-31 18:56 ?4次下載
    <b class='flag-5'>Linux</b><b class='flag-5'>設(shè)備</b><b class='flag-5'>驅(qū)動</b><b class='flag-5'>思想</b>在STM32編程<b class='flag-5'>中</b>的應(yīng)用

    Linux設(shè)備驅(qū)動開發(fā)詳解

    Linux設(shè)備驅(qū)動開發(fā)詳解
    發(fā)表于 10-28 11:03 ?50次下載

    Linux如何比較兩個文件的差異?

    Linux 如何比較兩個文件的差異?答案是使用 diff 命令。
    的頭像 發(fā)表于 01-17 17:30 ?5146次閱讀
    <b class='flag-5'>Linux</b><b class='flag-5'>中</b>如何比較<b class='flag-5'>兩個</b>文件的差異?

    冷鏈運輸兩個重要環(huán)節(jié)

    冷鏈運輸兩個重要環(huán)節(jié)
    的頭像 發(fā)表于 01-28 17:13 ?1112次閱讀
    冷鏈運輸<b class='flag-5'>中</b>的<b class='flag-5'>兩個</b><b class='flag-5'>重要</b>環(huán)節(jié)

    Linux如何比較兩個文件的差異?

    Linux 如何比較兩個文件的差異?答案是使用 diff 命令。
    的頭像 發(fā)表于 01-30 16:04 ?1.2w次閱讀
    <b class='flag-5'>Linux</b><b class='flag-5'>中</b>如何比較<b class='flag-5'>兩個</b>文件的差異?