當(dāng)前位置:首頁 > 嵌入式 > 嵌入式教程
[導(dǎo)讀]基于嵌入式Linux的鍵盤驅(qū)動(dòng)設(shè)計(jì)

摘要:提出一種基于嵌入式Linux的矩陣鍵盤實(shí)現(xiàn)方案,介紹矩陣式鍵盤的結(jié)構(gòu)與工作原理。課題以IntelPXA255處理器和嵌入式Linux2.4. 19操作系統(tǒng)為基礎(chǔ),對(duì)人機(jī)交互接口設(shè)備驅(qū)動(dòng)程序的設(shè)計(jì)開發(fā)做了深入的研究,針對(duì)嵌入式系統(tǒng)的鍵盤驅(qū)動(dòng)的特點(diǎn),設(shè)計(jì)基于行列掃描的鍵盤驅(qū)動(dòng)程序。
關(guān)鍵詞:矩陣鍵盤;嵌入式linux;鍵盤驅(qū)動(dòng)程序

1 鍵盤驅(qū)動(dòng)程序的設(shè)計(jì)
    隨著電子信息技術(shù)飛速發(fā)展,嵌入式系統(tǒng)構(gòu)成的各種設(shè)備得到了廣泛的應(yīng)用,嵌入式Linux是一種開放源碼、軟實(shí)時(shí)、多任務(wù)的操作系統(tǒng),是開發(fā)嵌入式產(chǎn)品的優(yōu)秀操作系統(tǒng)平臺(tái),其中鍵盤是人機(jī)界面中人類監(jiān)控計(jì)算機(jī)重要數(shù)據(jù)輸入設(shè)備。實(shí)現(xiàn)鍵盤有兩種方法:一種是采用現(xiàn)有的一些芯片實(shí)現(xiàn)鍵盤掃描;二是用軟件實(shí)現(xiàn)鍵盤掃描。目前許多芯片可用來實(shí)現(xiàn)鍵盤掃描,但是鍵盤掃描的軟件實(shí)現(xiàn)方法有助于縮減系統(tǒng)的重復(fù)開發(fā)成本,而只需很少的CPU開銷。嵌入式控制器的功能很強(qiáng)??梢猿浞掷眠@一資源。本課題提出的鍵盤方案是以嵌入式Linux和PXA255為軟硬件平臺(tái),通過測(cè)試,表明其具有良好的穩(wěn)定性和實(shí)時(shí)性。

2 矩陣式鍵盤的結(jié)構(gòu)與工作原理
    本課題采用矩陣鍵盤,如圖1所示。四根行線四根列線組成4x4矩陣鍵盤,分別用CPU的4個(gè)GPIO口。當(dāng)有鍵按下,某個(gè)列GPIO口電平被下拉從而產(chǎn)生下降沿,觸發(fā)中斷。其中按鍵行陣列必須提供上拉信號(hào),列陣列加二極管,防止瞬間電流過大對(duì)GPIO口造成沖擊。



3 Linux鍵盤驅(qū)動(dòng)簡(jiǎn)介
    在Linux中,鍵盤驅(qū)動(dòng)被劃分成兩層來實(shí)現(xiàn)。上層是一個(gè)通用鍵盤抽象層,下層則是硬件處理層,主要對(duì)硬件進(jìn)行直接的操作。鍵盤驅(qū)動(dòng)程序上層公共部分在 driver/keyboard.c里。文件中最重要的是內(nèi)核用EXPORT_SYMBOL這個(gè)宏導(dǎo)出的handle_scancode函數(shù)。在這個(gè)文件中還定義了其它的幾個(gè)回調(diào)函數(shù),它們由鍵盤驅(qū)動(dòng)程序中上層公共部分調(diào)用,并且由底層硬件處理函數(shù)實(shí)現(xiàn)。鍵盤驅(qū)動(dòng)程序的底層硬件處理部分則根據(jù)不同硬件有不同實(shí)現(xiàn)。
[!--empirenews.page--]
4 鍵盤驅(qū)動(dòng)程序的實(shí)現(xiàn)
4.1 宏定義module init和module exit
    通過宏定義module init和module exit可以看出,驅(qū)動(dòng)程序的入口從kd_ctrl_init()開始。當(dāng)內(nèi)核模塊加載的時(shí)候,默認(rèn)調(diào)用module_ jnit(kd_ctrl_init),在kd_ctrl_init()中將完成一些初始化工作,主要如下:
    (1)把GPIO口的起始虛擬地址映射到GPIO_BASE_PHY(0x1000b000),數(shù)據(jù)長(zhǎng)度為0x400:
    GPIO_BASE=(int)ioremap(GPIO_BASE_PHY,0x400);
    (2)利用request_irq函數(shù)將外設(shè)的中斷服務(wù)例程掛載到外部中斷處理程序中。本系統(tǒng)中利用request_irq函數(shù)分別為4個(gè)列GPIO口申請(qǐng)中斷資源,分別占用了中斷號(hào)1、2、3、4。其中i是中斷號(hào);kd_ctrl_irq是UCB1400的中斷處理程序,kd_ctrl代表鍵盤設(shè)備名,MAGIC_DEVID是申請(qǐng)時(shí)告訴系統(tǒng)設(shè)備標(biāo)志,用于共享中斷線。返回值為0表示申請(qǐng)成功。
    (3)通過函數(shù)misc_register注冊(cè)一個(gè)鍵盤設(shè)備,并分配主設(shè)備號(hào)和從設(shè)備號(hào),初始化一個(gè)環(huán)形隊(duì)列以及定義一個(gè)鍵盤控制的數(shù)據(jù)結(jié)構(gòu)。其中包括鍵值、鍵的狀態(tài)和長(zhǎng)按標(biāo)志。應(yīng)用程序?qū)υO(shè)備驅(qū)動(dòng)的調(diào)用實(shí)際是對(duì)相應(yīng)設(shè)備文件進(jìn)行操作,利用mknod命令將此節(jié)點(diǎn)與對(duì)應(yīng)設(shè)備建立聯(lián)系。
    (4)通過init_waitqueue_head(&sats.read_wait)初始化讀信號(hào)量。
4.2 打開鍵盤設(shè)備
    應(yīng)用程序打開設(shè)備文件時(shí),會(huì)調(diào)用驅(qū)動(dòng)中的OPEN函數(shù),此函數(shù)會(huì)對(duì)鍵盤所用到的行列GPIO口進(jìn)行配置。打開的設(shè)備在內(nèi)核中通過file 結(jié)構(gòu)進(jìn)行標(biāo)識(shí),內(nèi)核使用fileopreation,通過上面的結(jié)構(gòu)中設(shè)備文件操作結(jié)構(gòu)的映射,來調(diào)用驅(qū)動(dòng)中的kd_ctrl_open。接下來要做的是:
    (1)通過sema_init(&kdc->irq_wait,0)初始化在后面用來喚醒后臺(tái)線程的信號(hào)量。
    (2)調(diào)用初始化函數(shù)init_pxa_kdc()來初始化GPIO口,具體是把“行”的GPIO口設(shè)為輸出模式并設(shè)定值為O,把“列”GPIO口設(shè)為中斷模式,下降沿有效。如下所示:
  
  
    (3)以嚴(yán)格的串行方式執(zhí)行任務(wù)的效率并不高,如果把它們放在后臺(tái)調(diào)度,不管是對(duì)它們的函數(shù)還是對(duì)終端用戶進(jìn)程都能得到較好的響應(yīng)。所以初始化GPIO口后,開啟一個(gè)內(nèi)核線程kd_ctrl_thread專門用于處理鍵盤事件,其實(shí)也就是向系統(tǒng)申請(qǐng)了軟硬件資源。為了確保在該線程創(chuàng)建完成,使用 completion,在Linux內(nèi)核中,completion是一種簡(jiǎn)單的同步機(jī)制,利用completion機(jī)制可以使兩個(gè)任務(wù)同步。我們利
用init_completion(&kdc->init_exit)動(dòng)態(tài)初始化一個(gè)線程創(chuàng)建信號(hào)量init_exit,以及用 wait_for_completion(&kdc->init_exit)來等待進(jìn)程創(chuàng)建完成,然后在進(jìn)程創(chuàng)建結(jié)束后通過 complete(&kdc->init_exit)確定事件已經(jīng)完成即后臺(tái)線程創(chuàng)建成功,繼續(xù)執(zhí)行函數(shù)wait_for_comp- letion之后的任務(wù)。通過ret=kernel_thread(kd_ctrl_thread,kdc,CLONE_FS|CLONE_FILES) 創(chuàng)建后臺(tái)線程。
4.3 等待鍵盤事件
    后臺(tái)線程一旦創(chuàng)建和初始化完成,就會(huì)進(jìn)入一個(gè)無條件的for循環(huán),通過 set_task_state(tsk,TASK_INTERRUPTIBLE)將此線程推入可中斷睡眠的隊(duì)列,調(diào)用schedule timeout(Hz/100)來實(shí)現(xiàn)15毫秒的進(jìn)程掛起。此時(shí)讓出CPU,直到中斷事件來臨或睡眠超過規(guī)定時(shí)間后再重新執(zhí)行。線程一旦被喚醒即按照順序先利用set_kdc_gpio(KDC_COL_PINS,1,PINS_MODE_ENABLEINTERRUPT,0)使所有列GPIO口中斷,接著調(diào)用down_interruptible(&kdc->irq_wait):該函數(shù)的作用是獲得信號(hào)量irq_wait,把 irq_wait的值減掉1,如果信號(hào)量irq_wait的值非負(fù),就直接返回,如果獲取失敗鍵盤線程將以TASK_INTERRUPTIBLE狀態(tài)進(jìn)入可中斷睡眠,直到下次鍵盤事件利用信號(hào)量irq_wait喚醒此線程才能繼續(xù)運(yùn)行。因此,驅(qū)動(dòng)程序在沒有按鍵按下時(shí)將阻塞自己的執(zhí)行,不消耗任何的CPU 資源。
4.4 鍵盤事件發(fā)生
    一旦有按鍵事件發(fā)生也就是產(chǎn)生一個(gè)中斷,則進(jìn)入中斷處理程序kd_ctrl_irq(),在這個(gè)函數(shù)中所做的工作如圖2。


    喚醒后臺(tái)線程后,把列GPIO口中斷禁止,隨即調(diào)用kd_ctrl_event()進(jìn)行處理鍵盤事件。其中又調(diào)用pxa_kdc_scan()進(jìn)行鍵值的掃描:設(shè)定4×4小鍵盤的所有行GPIO口為輸出狀態(tài),并設(shè)定它的值為1,而所有列GPIO口作為輸入狀態(tài),然后采用逐行掃描的方法,依次去讀取四根列 GPIO口狀態(tài),如果某列GPIO口電平為低,就表示此行此列有鍵按下,根據(jù)行號(hào)和列號(hào)從對(duì)應(yīng)的二維數(shù)組(也就是鍵值映射表)中找到該鍵的鍵值。具體實(shí)現(xiàn)方法為:先設(shè)第一行(GPIO7)為0,掃描列的值(GPIO3、GPIO2、GPIO1、GPIO0),如果其中一個(gè)列的值為O,比如GPIO3,則按下的鍵是Key_5。掃描完列后,把第一行設(shè)為1。第二行設(shè)為0,再次掃描所有列的值。掃描結(jié)束后,設(shè)定所有行(GPIO7、GPIO6、GPIO5、 GPIO4)的值為0,并且再次恢復(fù)所有列為中斷方式,設(shè)定下降沿有效。最后返回的是代表按鍵是否按下的參數(shù)pressure值。得到此值以后,調(diào)用 sta-tic inline void kd_ctrl_evt_add(struct kd_ctrl*kdc,u8 pressure,u8 keyvalue)函數(shù)把所得值保存在對(duì)應(yīng)的結(jié)構(gòu)中,并將其添加到事件隊(duì)列中,最后調(diào)用 wake_up_interruptible(&kdc->read_wait)利用信號(hào)量read_wait通知read程序到緩沖區(qū)讀取新數(shù)據(jù)。
4.5 應(yīng)用程序讀取鍵盤數(shù)據(jù)
    由于用戶程序需要不斷輪詢?cè)O(shè)備,以查詢是否有數(shù)據(jù)讀取,如果程序不處于休眠狀態(tài),則將會(huì)占用很多CPU的資源。因此當(dāng)沒有觸摸數(shù)據(jù)時(shí),就阻塞此任務(wù)。此時(shí)用戶空間則需要和內(nèi)核同步,代碼會(huì)需要睡眠,使用信號(hào)量是唯一的選擇,并且它適用于鎖會(huì)被長(zhǎng)時(shí)間持有的情況。如果有一個(gè)任務(wù)試圖獲得一個(gè)已經(jīng)被占用的信號(hào)量時(shí),信號(hào)量會(huì)先將其中推進(jìn)一個(gè)等待隊(duì)列,然后讓其睡眠。這時(shí)CPU能重獲自由,從而可以執(zhí)行其他代碼。當(dāng)持有信號(hào)量的進(jìn)程將信號(hào)量釋放時(shí),處于等待隊(duì)列中的那個(gè)任務(wù)將會(huì)被喚醒,并獲得該信號(hào)量。
    等待隊(duì)列是由等待某些事件發(fā)生的進(jìn)程組成的簡(jiǎn)單鏈表。內(nèi)核用wake_queue_head_t來表示等待隊(duì)列。等待隊(duì)列可通過 DECLARE_WAITQUE-UE()靜態(tài)創(chuàng)建。一旦上層用戶程序進(jìn)行讀操作,系統(tǒng)調(diào)用將通過kd_ctrl_read()函數(shù)來實(shí)現(xiàn)。
4.6 模塊卸載
    當(dāng)內(nèi)核需要卸載本驅(qū)動(dòng)程序時(shí),最后會(huì)從本函數(shù)退出。此時(shí)通過module_init(kd_ctrl_init)函數(shù)需要將在驅(qū)動(dòng)程序運(yùn)行期間申請(qǐng)的系統(tǒng)資源全部釋放掉,可以防止資源浪費(fèi)。

5 結(jié)束語
    本文介紹的嵌入式Linux的一種矩陣小鍵盤,成功實(shí)現(xiàn)了多鍵齊按和重復(fù)按鍵的功能,已經(jīng)用于手持嵌入式設(shè)備中,實(shí)驗(yàn)證明性能穩(wěn)定可靠。
 

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

倫敦2024年8月29日 /美通社/ -- 英國(guó)汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對(duì)日本游戲市場(chǎng)的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對(duì)環(huán)境變化,經(jīng)營(yíng)業(yè)績(jī)穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤(rùn)率延續(xù)升勢(shì) 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長(zhǎng) 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競(jìng)爭(zhēng)力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競(jìng)爭(zhēng)優(yōu)勢(shì)...

關(guān)鍵字: 通信 BSP 電信運(yùn)營(yíng)商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國(guó)電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場(chǎng) NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長(zhǎng)三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡(jiǎn)稱"軟通動(dòng)力")與長(zhǎng)三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉