<meter id="pryje"><nav id="pryje"><delect id="pryje"></delect></nav></meter>
          <label id="pryje"></label>

          新聞中心

          EEPW首頁 > 嵌入式系統(tǒng) > 設計應用 > 基于RTX51的用戶專用鍵盤軟件設計

          基于RTX51的用戶專用鍵盤軟件設計

          作者: 時間:2013-02-25 來源:網絡 收藏

          實時操作系統(tǒng)進行單片機,可以真正做到各任務并行執(zhí)行,同時,由于程序結構更加科學合理,可以方便地實現(xiàn)修改升級,以滿足功能較多的要求。

          本文引用地址:http://www.ex-cimer.com/article/148091.htm

          1 簡介

          多使用在工業(yè)控制場合,它由按鍵和指示燈組成,對外采用RS232接口,當某個按鍵按下時,將命令(碼值)發(fā)送計算機,同時,鍵盤接收來自計算機的命令(碼值)點亮(熄滅)某個指示燈,以指示系統(tǒng)工作狀態(tài),從而實現(xiàn)快速人機交互。鍵盤硬件原理框圖如圖1所示。

          圖中,可編程邏輯芯片實現(xiàn)單片機輸出接口擴展,其內部包含多個輸出鎖存器。工作時,單片機輸出的地址信號經可編程芯片內部譯碼器譯碼產生片選信號,使能不同的輸出鎖存器,將數(shù)據(jù)信號輸出控制各個指示燈的亮/滅。

          2 實時操作系統(tǒng)介紹

          是德國Keil公司開發(fā)的一種應用于MCS51系列單片機的實時多任務操作系統(tǒng),它有兩個版本,RTX51 Full和RTX51 Tiny,本文采用RTX51 Tiny進行。RTX51 Tiny是一個很小的內核,完全集成在Keil C51編譯器中,它可以很容易地運行在沒有擴展外部存儲器的單片機系統(tǒng)上,并且僅占用800字節(jié)左右的程序存儲空間。

          RTX51 Tiny允許最大16個任務循環(huán)切換,在實現(xiàn)上,它采用時間片輪轉算法,系統(tǒng)每次調度時,把CPU分配給一個就緒的任務,并令其執(zhí)行一個時間片,構成微觀上輪流運行、宏觀上并行執(zhí)行的多任務效果。RTX51 Tiny支持任務間的信號傳遞,還能并行地利用中斷功能。

          RTX51 Tiny的用戶任務主要具有以下幾個狀態(tài):

          ·運行(RUNNING):任務正處于運行中。同一時刻只有一個任務可以處于“RUNNING”狀態(tài)。

          ·準備好(READY):等待運行的任務處于“READY”狀態(tài)。在當前運行的任務退出運行狀態(tài)后,就緒隊列中的任務根據(jù)調度策略被調度執(zhí)行,進入到運行狀態(tài)。

          ·等待(BLOcKED):等待一個事件的任務處于“BLOCKED”狀態(tài)。如果等待的事件發(fā)生,則此任務進入“READY”狀態(tài),等待被調度。

          RTX51 Tiny內核用以下事件進行任務問的通信和同步:

          ·超時(TIMEOUT):由OS-wait函數(shù)調用引發(fā)的時間延時,持續(xù)時間可由定時節(jié)拍數(shù)確定。帶有TIMEOUT值調用OS-it函數(shù)的任務將被掛起,直到延時結束,才返回到“READY”。

          ·間隔(INTERVAL):由OS-wait函數(shù)調用引發(fā)的時間間隔,其間隔時間可由定時節(jié)拍數(shù)確定。帶有INTERVAL值調用wait函數(shù)的任務將被掛起,直到間隔時間結束,然后返回到READY狀態(tài)。與TIMEOUT不同的是,任務的節(jié)拍計數(shù)器不復位,典型應用是產生時鐘。

          ·信號(SIGNAL):系統(tǒng)定義的位變量,可以由系統(tǒng)函數(shù)置位或清除??梢哉{用OS-wait函數(shù)暫停一個任務并等待從另一任務發(fā)出的信號,這可以用于協(xié)調兩個或更多的任務。如果某個任務在等待一個信號并且信號標志為0,則在收到這個信號之前,這個任務將一直處于掛起狀態(tài)。如果信號標志已經被置1,則當任務查詢信號時,信號標志會被清除,任務將可以被繼續(xù)執(zhí)行。

          3 用戶專用鍵盤設計

          3.1 任務分配

          根據(jù)前面對用戶專用鍵盤功能的描述,它主要實現(xiàn)以下兩個功能:

          (1)按鍵處理;

          (2)串口數(shù)據(jù)處理。

          其中功能(1)又可細分為以下三個任務:

          任務1:按鍵狀態(tài)掃描;

          任務2:按鍵碼值查詢;

          任務3:串口發(fā)送;

          同樣,功能(2)也可細分為以下兩個任務:

          任務4:串口接收;

          任務5:串口數(shù)據(jù)處理;

          以上兩個功能需要并行運行,而內部的子任務之間為前級驅動后級的關系,在程序實際運行過程中,功能二(2)的任意子任務可能與功能(1)的任務1或任務2或任務3處于同時并行運行狀態(tài),鑒于此,需要在程序設計時創(chuàng)建5個子任務。

          采用時間輪詢的方式決定了某個任務在執(zhí)行完時間片后,在下一次執(zhí)行前需要等待固定的時間,這個時間與系統(tǒng)的任務數(shù)及每個任務的執(zhí)行時間密切相關,為避免數(shù)據(jù)丟失,串口接收任務應及時讀取接收緩存器中的數(shù)據(jù)。由于中斷處理過程與正在運行的任務是相互獨立的,即中斷處理過程在RTX51系統(tǒng)內核之外和任務切換規(guī)則沒有關聯(lián),因此可以在串口中斷服務程序中完成串口接收任務。另外,串口發(fā)送時要求將整個按鍵碼值數(shù)據(jù)包一次性發(fā)送完畢,如果將串口發(fā)送過程在中斷服務程序中完成,在SBUF緩存器發(fā)送完一個字節(jié)后觸發(fā)串口發(fā)送中斷標志,再次進入中斷服務程序繼續(xù)下一字節(jié)數(shù)據(jù)的發(fā)送,則可以方便地實現(xiàn)上述要求。根據(jù)以上分析,串口接收、串口發(fā)送兩個子任務的功能在中斷服務程序中完成,將系統(tǒng)子任務的個數(shù)由5個減少為3個,調整后的任務分配如下:

          任務1:按鍵狀態(tài)掃描(TASK SCAN);

          任務2:按鍵碼值查詢(TASK KEY);

          任務3:串口數(shù)據(jù)處理((TASK LIGHT));

          中斷服務程序:串口接收、發(fā)送。

          任務間信號關系如圖2所示。

          如圖,任務1在檢測到按鍵狀態(tài)變化后向任務2發(fā)送信號,任務2隨后由等待狀態(tài)進入“準備好”狀態(tài),在本任務的下一個時間片,任務2開始進行指定位置按鍵的碼值查詢,然后通過串口完成碼值發(fā)送。

          同時,串口數(shù)據(jù)通過中斷服務程序接收,串口數(shù)據(jù)接收后即發(fā)送信號給任務3,使后者進入“準備好”狀態(tài),并在下一個時間片到來后進行數(shù)據(jù)處理。

          以上三個任務中,任務l始終處于“運行”或“準備好”狀態(tài),任務2、任務3大多數(shù)時間處于“等待”狀態(tài),任務2、任務3分別在接收到按鍵狀態(tài)掃描任務、中斷服務程序的信號后被“喚起”。另有任務0,負責創(chuàng)建任務1、2、3,然后刪除自己。任務0簡化程序如下所示:

          #define INIT 0/*任務0:初始化及創(chuàng)建*/

          #define SCAN 1/*任務1:按鍵狀態(tài)掃描*/

          DIY機械鍵盤相關社區(qū):機械鍵盤DIY



          上一頁 1 2 下一頁

          評論


          相關推薦

          技術專區(qū)

          關閉
          看屁屁www成人影院,亚洲人妻成人图片,亚洲精品成人午夜在线,日韩在线 欧美成人 (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })();