《電子技術應用》
您所在的位置:首頁 > 嵌入式技術 > 設計應用 > 基于嵌入式Linux的鍵盤驅動
基于嵌入式Linux的鍵盤驅動
中電網
摘要: 本文介紹的嵌入式 Linux的一種矩陣小鍵盤, 成功實現了多鍵齊按和重復按鍵的功能, 已經用于手持嵌入式設備中, 實驗證明性能穩定可靠。
Abstract:
Key words :

1 鍵盤驅動程序的設計

隨著電子信息技術飛速發展,嵌入式系統構成的各種設備得到了廣泛的應用, 嵌入式  Linux是一種開放源碼、 軟實時、 多任務的操作系統,是開發嵌入式產品的優秀操作系統平臺,其中鍵盤是人機界面中人類監控計算機重要數據輸入設備。實現鍵盤有兩種方法:一種是采用現有的一些芯片實現鍵盤掃描;二是用軟件實現鍵盤掃描。目前許多芯片可用來實現鍵盤掃描,但是鍵盤掃描的軟件實現方法有助于縮減系統的重復開發成本, 而只需很少的 CPU 開銷。嵌入式控制器的功能很強,可以充分利用這一資源。本課題提出的鍵盤方案是以嵌入式  Linux和 PXA255為軟硬件平臺, 通過測試,表明其具有良好的穩定性和實時性。

2 矩陣式鍵盤的結構與工作原理

本課題采用矩陣鍵盤, 如圖 1所示。四根行線四根列線組成 4 *4矩陣鍵盤, 分別用 CPU 的 4個 GPIO口。當有鍵按下,某個列 GPI O 口電平被下拉從而產生下降沿, 觸發中斷。其中按鍵行陣列必須提供上拉信號,列陣列加二極管,防止瞬間電流過大對 GPI O口造成沖擊。

圖 1 矩陣鍵盤原理圖.

3  Linux鍵盤驅動簡介

在 Linux中, 鍵盤驅動被劃分成兩層來實現。上層是一個通用鍵盤抽象層, 下層則是硬件處理層, 主要對硬件進行直接的操作。鍵盤驅動程序上層公共部分在 driver /keyboard . c里。文件中最重要的是內核用 EXPORT _SYM BOL這個宏導出的 handle_scancode函數 。在這個文件中還定義了其它的幾個回調函數,它們由鍵盤驅動程序中上層公共部分調用, 并且由底層硬件處理函數實現。鍵盤驅動程序的底層硬件處理部分則根據不同硬件有不同實現。

4 鍵盤驅動程序的實現

4 . 1  宏定義 module init和 module exit

通過宏定義 module init和module exit可以看出,驅動程序的入口從 kd_ctrl_init( )開始。當內核模塊加載的時候, 默認調用 module_ i nit( kd_c trl_init) ,在 kd_ctr l_ i nit( )中將完成一些初始化工作, 主要如下:

( 1) 把 GPI O 口的起始虛擬地址映射到 GPI O _BASE _PHY ( 0x1000b000),數據長度為 0x400 :

GPI O _ BASE = ( i nt) ioremap ( GPI O _ BASE _ P HY,0x400);

( 2) 利用 request_ irq函數將外設的中斷服務例程掛載到外部中斷處理程序中。本系統中利用 request_irq函數分別為 4個列 GPI O口申請中斷資源, 分別占用了中斷號 1 、2 、3、 4 。其中 i是中斷號; kd_ctr l_irq是 UCB1400的中斷處理程序, kd_ctr l代表鍵盤設備名, MAGIC _DEVID是申請時告訴系統設備標志, 用于共享中斷線。返回值為 0表示申請成功。

( 3) 通過函數 m isc_reg ister注冊一個鍵盤設備, 并分配主設備號和從設備號, 初始化一個環形隊列以及定義一個鍵盤控制的數據結構。其中包括鍵值、鍵的狀態和長按標志。

應用程序對設備驅動的調用實際是對相應設備文件進行操作, 利用 mknod命令將此節點與對應設備建立聯系。

( 4) 通過 init_ w a it queue_head(& sa ts . read _ w a it)初始化讀信號量。

4 . 2  打開鍵盤設備

應用程序打開設備文件時, 會調用驅動中的 OPEN 函數, 此函數會對鍵盤所用到的行列 GPI O 口進行配置。打開的設備在內 核中通過 file結 構進行標識, 內核 使用 fileopreati on ,通過上面的結構中設備文件操作結構的映射, 來調用驅動中的 kd_c trl_open。接下來要做的是:

( 1) 通過 se m a_ i n it( & kdc- > irq_w ait , 0)初始化在后面用來喚醒后臺線程的信號量。

( 2) 調用初始化函數 i n it_pxa_kdc( )來初始化 GPI O口,具體是把   行!的 GPI O 口設為輸出模式并設定值為 0 , 把列!GPIO口設為中斷模式,下降沿有效。如下所示:

re t = se t_kdc_gp i o( KDC_ROW _PINS , 1 , PI NS_MODE _OUT , 0) ;

ret = set_kdc_gp i o ( KDC _COL _PI NS , 1 , PI NS _ MODE _FALLI NG_I NTTERUPT , 0);

( 3) 以嚴格的串行方式執行任務的效率并不高, 如果把它們放在后臺調度,不管是對它們的函數還是對終端用戶進程都能得到較好的響應。所以初始化 GPIO口后,開啟一個內核線程 kd_ctrl_thread專門用于處理鍵盤事件, 其實也就是向系統申請了軟硬件資源。為了確保在該線程創建完成,使用 co m pleti on ,在  Linux內核中, co m pletion是一種簡單的同步機制,利用 co m pleti on機制可以使兩個任務同步。我們利用 i n it_comp l e ti on(& kdc- > i n it_ex it)動態初始化一個線程創建信號量 i n it_ex it , 以及用 wa it_fo r_co m pleti on (& kdc- >i n it_ex it)來等待進程創建完成, 然后在進程創建結束后通過co m plete(& kdc- > i nit_ex it)確定事件已經完成即后臺線程創建成功, 繼續執行函數 w ait_ for_ comp l e ti on之后的任務。

通過 ret = kerne l_t h read( kd_c trl_ t hread , kdc , CLONE_FS |CLONE_FILES)創建后臺線程。

4 . 3 等待鍵盤事件

后臺線程一旦創建和初始化完成, 就會進入一個無條件的 for循 環, 通 過 set _ task _ state ( tsk , TASK _ INTERRUPTIBLE) 將此線程推入可中斷睡眠的隊列,調用 schedule ti m eou t (H Z/100)來實現 15毫秒的進程掛起。此時讓出 CPU,直到中斷事件來臨或睡眠超過規定時間后再重新執行。線程一旦被喚醒即按照順序先利用 set_kdc_gp io ( KDC _COL_PI NS , 1 , PI NS _MODE _ENABLEI NTERRUPT, 0) 使 所有列GPI O 口中斷, 接著調用 down _ i nterrupti b l e ( & kdc- > irq _wa it): 該函數的作用是獲得信號量 irq_wa it , 把 irq_ w a i t的值減掉 1 , 如果信號量 irq_wa it的值非負, 就直接返回,如果獲取失敗鍵盤線程將以 TASK_I NTERRUPTIBLE狀態進入可中斷睡眠,直到下次鍵盤事件利用信號量 irq_ w a it喚醒此線程才能繼續運行。因此,驅動程序在沒有按鍵按下時將阻塞自己的執行,不消耗任何的 CPU資源。

4 . 4 鍵盤事件發生

一旦有按鍵事件發生也就是產生一個中斷, 則進入中斷處理程序 kd_ctr l _ irq( ), 在這個函數中所做的工作如圖 2。

圖 2 中斷處理程序 kd_ ctrl_irq( )

喚醒后臺線程后,把列 GPI O口中斷禁止, 隨即調用 kd_ctrl_event( )進行處理鍵盤事件。其中又調用 pxa _kdc _scan( )進行鍵值的掃描: 設定 4 [1] 4小鍵盤的所有行 GPI O 口為輸出狀態,并設定它的值為 1 ,而所有列 GPIO口作為輸入狀態,然后采用逐行掃描的方法, 依次去讀取四根列 GPI O 口狀態,如果某列 GPIO 口電平為低, 就表示此行此列有鍵按下,根據行號和列號從對應的二維數組 (也就是鍵值映射表 )中找到該鍵 的鍵值。具體 實現方法 為: 先設第 一行( GPI O7)為 0 , 掃描列的值 ( GPI O3 、 GPI O2 、GPI O1、 GPI O0),如果其中一個列的值為 0 , 比如 GPI O3 , 則按下的鍵是 Key _5。掃描完列后,把第一行設為 1。第二行設為 0 , 再次掃描所有列的值。掃描結 束后, 設 定所有 行 ( GPI O7 、GPI O6 、GPI O5 、 GPI O4)的值為 0 , 并且再次恢復所有列為中斷方式,設定下降沿有效。最后返回的是代表按鍵是否按下的參數pressure值。得到此值以后,調用 stati c i n line vo i d kd_c trl_ev t_add( struc t kd_ctrl* kdc , u8 pressure , u8 keyva l ue )函數把所得值保存在對應的結構中,并將其添加到事件隊列中, 最后調用 w ake_up_ i nterrupti ble( & kdc- > read _ w a it)利用信號量 read_ w a it通知 read程序到緩沖區讀取新數據。

4 . 5  應用程序讀取鍵盤數據

由于用戶程序需要不斷輪詢設備,以查詢是否有數據讀取, 如果程序不處于休眠狀態, 則將會占用很多 CPU 的資源。因此當沒有觸摸數據時, 就阻塞此任務。此時用戶空間則需要和內核同步, 代碼會需要睡眠, 使用信號量是唯一的選擇, 并且它適用于鎖會被長時間持有的情況。如果有一個任務試圖獲得一個已經被占用的信號量時,信號量會先將其中推進一個等待隊列, 然后讓其睡眠。這時 CPU 能重獲自由, 從而可以執行其他代碼。當持有信號量的進程將信號量釋放時, 處于等待隊列中的那個任務將會被喚醒, 并獲得該信號量。

等待隊列是由等待某些事件發生的進程組成的簡單鏈表。內核用 w ake_queue_head_t來表示等待隊列。等待隊列可通過 DECLARE _WAI TQUEUE ( )靜態創建。一旦上層用戶程序進行讀操作, 系統調用將通過 kd_ctrl_read ( )函數來實現。

4 . 6  模塊卸載

當內核需要卸載本驅動程序時, 最后會從本函數退出。

此時通過 modul e_ i nit( kd_ctrl_ i n it)函數需要將在驅動程序運行期間申請的系統資源全部釋放掉,可以防止資源浪費。

5 結束語

本文介紹的嵌入式  Linux的一種矩陣小鍵盤, 成功實現了多鍵齊按和重復按鍵的功能, 已經用于手持嵌入式設備中, 實驗證明性能穩定可靠。

此內容為AET網站原創,未經授權禁止轉載。
亚洲一区二区欧美_亚洲丝袜一区_99re亚洲国产精品_日韩亚洲一区二区
亚洲一区精品视频| 亚洲欧洲99久久| 欧美在线观看网站| 欧美一区国产在线| 在线播放日韩欧美| 欧美精品三级在线观看| 99在线视频精品| 99国产精品久久久久老师| 国产精品久久久久久妇女6080 | 欧美另类在线观看| 亚洲欧美在线免费| 久久精品国产亚洲a| 91久久久一线二线三线品牌| 欧美日韩国产一区二区三区地区| 亚洲欧美国产精品专区久久| 欧美在线影院| 亚洲日本成人女熟在线观看| 亚洲国产成人久久综合| 国产精品久久久久永久免费观看 | 欧美影院在线播放| 小黄鸭精品密入口导航| 亚洲国产精品一区| 亚洲视频1区2区| 好吊妞**欧美| 欧美日韩一区二| 久久婷婷丁香| 亚洲一区二区三区激情| 亚洲一区一卡| 性做久久久久久久免费看| 欧美中在线观看| 久色婷婷小香蕉久久| 亚洲一区二区三区午夜| 亚洲一区在线直播| 欧美一区二区三区在线观看视频| 亚洲国产一二三| 最新日韩在线视频| 亚洲欧美在线x视频| 亚洲综合视频在线| 日韩视频免费| 伊人蜜桃色噜噜激情综合| 在线欧美日韩| 国产色产综合色产在线视频| 欧美日韩另类综合| 国产精品第一页第二页第三页| 国产精品视频一| 欧美日韩国产区| 国产精品久久久久三级| 国产亚洲欧美一区| 国产精品久久久久久妇女6080| 国产欧美日韩不卡| 国产精品国产a级| 国产日韩欧美在线播放不卡| 黄色成人在线观看| 91久久在线播放| 亚洲综合欧美日韩| 亚洲国产精品一区制服丝袜| 亚洲视频在线一区观看| 日韩亚洲国产精品| 亚洲欧洲一区二区在线观看| 亚洲特色特黄| 亚洲一级黄色片| 久久精品国产亚洲a| 性色av一区二区三区红粉影视| 久久久美女艺术照精彩视频福利播放| 欧美亚洲一级| 麻豆av一区二区三区| 欧美中文日韩| 欧美激情第10页| 欧美激情乱人伦| 国产精品久久久久久久久婷婷 | 精品91视频| 狠狠干综合网| 夜夜嗨av色一区二区不卡| 日韩亚洲欧美中文三级| 欧美在线日韩精品| 中文精品视频| 麻豆精品91| 国产九九精品视频| 国产区日韩欧美| 亚洲久久在线| 亚洲第一色在线| 久久国产视频网站| 亚洲在线观看视频| 裸体歌舞表演一区二区| 国产精品毛片在线看| 91久久国产精品91久久性色| 亚洲日本在线观看| 久久大逼视频| 午夜在线观看免费一区| 欧美精品激情在线| 黄色免费成人| 亚洲欧美日韩视频一区| 欧美一区二区大片| 久久成人免费| 亚洲伊人伊色伊影伊综合网| 欧美高清视频在线播放| 国产一区二区av| 亚洲一二三区在线观看| 亚洲免费视频观看| 亚洲视频网在线直播| 欧美激情a∨在线视频播放| 国产综合网站| 亚洲国产合集| 亚洲第一在线综合网站| 久久精品亚洲| 麻豆精品视频| 国产一区二区三区电影在线观看| 亚洲特色特黄| 在线一区视频| 欧美日韩视频第一区| 亚洲精品日韩激情在线电影| 一区二区三区久久网| 亚洲男人的天堂在线观看| 亚洲视频综合| 欧美日本一区二区高清播放视频| 在线成人国产| 亚洲第一页自拍| 久久资源在线| 影音先锋日韩有码| 亚洲欧洲中文日韩久久av乱码| 麻豆国产va免费精品高清在线| 国产综合香蕉五月婷在线| 午夜精品久久久久| 欧美在线视频二区| 国产欧美一区二区色老头| 亚洲伊人观看| 欧美一区二区黄色| 国产亚洲精品bt天堂精选| 欧美一级在线视频| 久久全国免费视频| 在线成人av| 日韩视频精品在线观看| 欧美精品色综合| 日韩一区二区精品葵司在线| 亚洲视频免费| 国产精品综合久久久| 欧美亚洲视频在线观看| 久久免费视频在线| 亚洲电影网站| 一区二区激情| 麻豆成人在线观看| 亚洲国产日韩欧美在线99| 99re热这里只有精品免费视频| 欧美激情一区二区三区在线视频观看 | 欧美色道久久88综合亚洲精品| 欧美另类综合| 日韩视频专区| 亚洲欧美清纯在线制服| 国产精品久久久久久久浪潮网站| 亚洲专区一二三| 久久久久国产精品厨房| 亚洲国产经典视频| 中文av一区二区| 男女激情视频一区| 国产欧美日韩精品在线| 久久精品国产亚洲一区二区三区| 免费久久99精品国产自| 一区二区精品在线| 久久精品视频免费播放| 亚洲第一色在线| 亚洲一区二区三区视频播放| 国产欧美 在线欧美| 亚洲国产精品va| 欧美日韩黄色大片| 性久久久久久久久| 欧美激情在线| 亚洲字幕一区二区| 欧美福利影院| 在线观看欧美日韩国产| 一区二区三区久久网| 国产伦精品一区二区三| 亚洲国产欧美在线 | 国产精品久久九九| 欧美在线免费观看视频| 欧美日韩二区三区| 欧美在线视频不卡| 欧美日韩美女| 久久成人亚洲| 国产精品久久久一区二区三区| 亚洲第一精品久久忘忧草社区| 欧美一区二区三区的| 在线观看一区欧美| 亚洲自拍偷拍麻豆| 樱桃成人精品视频在线播放| 亚洲尤物在线视频观看| 亚洲成色www8888| 欧美一区二区三区免费视频| 亚洲国产精品久久久久秋霞影院| 亚洲欧美激情视频在线观看一区二区三区 | 欧美女主播在线| 欧美亚洲一区在线| 欧美日韩亚洲一区在线观看| 久久精品123| 国产精品久线观看视频| 亚洲激情综合| 国产欧美日韩视频一区二区三区| 亚洲精品在线三区| 欧美国产先锋| 欧美一区二区三区婷婷月色| 欧美日韩美女|