資源描述:
《Linux ioctl函數(shù)》由會(huì)員上傳分享,免費(fèi)在線閱讀,更多相關(guān)內(nèi)容在教育資源-天天文庫(kù)。
1、Linuxioctl函數(shù)???我這里說(shuō)的ioctl函數(shù)是指驅(qū)動(dòng)程序里的,因?yàn)槲也恢肋€有沒(méi)有別的場(chǎng)合用到了它,所以就規(guī)定了我們討論的范圍。寫這篇文章是因?yàn)槲仪耙魂囎颖籭octl給搞混了,這幾天才弄明白它,于是在這里清理一下頭腦。一、什么是ioctl???ioctl是設(shè)備驅(qū)動(dòng)程序中對(duì)設(shè)備的I/O通道進(jìn)行管理的函數(shù)。所謂對(duì)I/O通道進(jìn)行管理,就是對(duì)設(shè)備的一些特性進(jìn)行控制,例如串口的傳輸波特率、馬達(dá)的轉(zhuǎn)速等等。它的調(diào)用個(gè)數(shù)如下:intioctl(intfd,indcmd,…);???其中fd是用戶程序打開(kāi)設(shè)備時(shí)使用open函數(shù)返回的文件標(biāo)示符,cmd是用戶程序?qū)υO(shè)備的控制命令,至于后面的
2、省略號(hào),那是一些補(bǔ)充參數(shù),一般最多一個(gè),這個(gè)參數(shù)的有無(wú)和cmd的意義相關(guān)。???ioctl函數(shù)是文件結(jié)構(gòu)中的一個(gè)屬性分量,就是說(shuō)如果你的驅(qū)動(dòng)程序提供了對(duì)ioctl的支持,用戶就可以在用戶程序中使用ioctl函數(shù)來(lái)控制設(shè)備的I/O通道。二、ioctl的必要性???如果不用ioctl的話,也可以實(shí)現(xiàn)對(duì)設(shè)備I/O通道的控制,但那是蠻擰了。例如,我們可以在驅(qū)動(dòng)程序中實(shí)現(xiàn)write的時(shí)候檢查一下是否有特殊約定的數(shù)據(jù)流通過(guò),如果有的話,那么后面就跟著控制命令(一般在socket編程中常常這樣做)。但是如果這樣做的話,會(huì)導(dǎo)致代碼分工不明,程序結(jié)構(gòu)混亂,程序員自己也會(huì)頭昏眼花的。所以,我們就使用i
3、octl來(lái)實(shí)現(xiàn)控制的功能。要記住,用戶程序所作的只是通過(guò)命令碼(cmd)告訴驅(qū)動(dòng)程序它想做什么,至于怎么解釋這些命令和怎么實(shí)現(xiàn)這些命令,這都是驅(qū)動(dòng)程序要做的事情。三、ioctl如何實(shí)現(xiàn)???這是一個(gè)很麻煩的問(wèn)題,我是能省則省。要說(shuō)清楚它,沒(méi)有四五千字是不行的,所以我這里是不可能把它說(shuō)得非常清楚了,不過(guò)如果讀者對(duì)用戶程序是怎么和驅(qū)動(dòng)程序聯(lián)系起來(lái)感興趣的話,可以看我前一陣子寫的《write的奧秘》。讀者只要把write換成ioctl,就知道用戶程序的ioctl是怎么和驅(qū)動(dòng)程序中的ioctl實(shí)現(xiàn)聯(lián)系在一起的了。我這里說(shuō)一個(gè)大概思路,因?yàn)槲矣X(jué)得《Linux設(shè)備驅(qū)動(dòng)程序》這本書(shū)已經(jīng)說(shuō)的非常清
4、楚了,但是得花一些時(shí)間來(lái)看。???在驅(qū)動(dòng)程序中實(shí)現(xiàn)的ioctl函數(shù)體內(nèi),實(shí)際上是有一個(gè)switch{case}結(jié)構(gòu),每一個(gè)case對(duì)應(yīng)一個(gè)命令碼,做出一些相應(yīng)的操作。怎么實(shí)現(xiàn)這些操作,這是每一個(gè)程序員自己的事情。因?yàn)樵O(shè)備都是特定的,這里也沒(méi)法說(shuō)。關(guān)鍵在于怎樣組織命令碼,因?yàn)樵趇octl中命令碼是唯一聯(lián)系用戶程序命令和驅(qū)動(dòng)程序支持的途徑。命令碼的組織是有一些講究的,因?yàn)槲覀円欢ㄒ龅矫詈驮O(shè)備是一一對(duì)應(yīng)的,這樣才不會(huì)將正確的命令發(fā)給錯(cuò)誤的設(shè)備,或者是把錯(cuò)誤的命令發(fā)給正確的設(shè)備,或者是把錯(cuò)誤的命令發(fā)給錯(cuò)誤的設(shè)備。這些錯(cuò)誤都會(huì)導(dǎo)致不可預(yù)料的事情發(fā)生,而當(dāng)程序員發(fā)現(xiàn)了這些奇怪的事情的時(shí)候,
5、再來(lái)調(diào)試程序查找錯(cuò)誤,那將是非常困難的事情。所以在Linux核心中是這樣定義一個(gè)命令碼的:____________________________________
6、設(shè)備類型
7、序列號(hào)
8、方向
9、數(shù)據(jù)尺寸
10、
11、----------
12、--------
13、------
14、--------
15、
16、8bit
17、8bit
18、2bit
19、8~14bit
20、
21、----------
22、--------
23、------
24、--------
25、???這樣一來(lái),一個(gè)命令就變成了一個(gè)整數(shù)形式的命令碼;但是命令碼非常的不直觀,所以LinuxKernel中提供了一些宏。這些宏可根據(jù)便于理解的字符串生成命令碼,或者是從命令碼得到一些用戶可以理解
26、的字符串以標(biāo)明這個(gè)命令對(duì)應(yīng)的設(shè)備類型、設(shè)備序列號(hào)、數(shù)據(jù)傳送方向和數(shù)據(jù)傳輸尺寸。???這些宏我就不在這里解釋了,具體的形式請(qǐng)讀者察看Linux核心源代碼中的宏,文件里給這些宏做了完整的定義。這里我只多說(shuō)一個(gè)地方,那就是"幻數(shù)"。"幻數(shù)"是一個(gè)字母,數(shù)據(jù)長(zhǎng)度也是8,用一個(gè)特定的字母來(lái)標(biāo)明設(shè)備類型,這和用一個(gè)數(shù)字是一樣的,只是更加利于記憶和理解。就是這樣,再?zèng)]有更復(fù)雜的了。更多的說(shuō)了也沒(méi)用,讀者還是看一看源代碼吧,推薦各位閱讀《Linux設(shè)備驅(qū)動(dòng)程序》所帶源代碼中的short一例,因?yàn)樗容^短小,功能比較簡(jiǎn)單,可以看明白ioctl的功能和細(xì)節(jié)。四、cmd參數(shù)如何得出???這里確實(shí)要說(shuō)一說(shuō)
27、,cmd參數(shù)在用戶程序端由一些宏根據(jù)設(shè)備類型、序列號(hào)、傳送方向、數(shù)據(jù)尺寸等生成,這個(gè)整數(shù)通過(guò)系統(tǒng)調(diào)用傳遞到內(nèi)核中的驅(qū)動(dòng)程序,再由驅(qū)動(dòng)程序使用解碼宏從這個(gè)整數(shù)中得到設(shè)備的類型、序列號(hào)、傳送方向、數(shù)據(jù)尺寸等信息,然后通過(guò)switch{case}結(jié)構(gòu)進(jìn)行相應(yīng)的操作。要透徹理解,只能是通過(guò)閱讀源代碼,我這篇文章實(shí)際上只是一個(gè)引子。cmd參數(shù)的組織還是比較復(fù)雜的,我認(rèn)為要搞熟它還是得花不少時(shí)間的,但是這是值得的,因?yàn)轵?qū)動(dòng)程序中最難的是對(duì)中斷的理解。五、小結(jié)???ioctl其實(shí)沒(méi)