《電子技術應用》
您所在的位置:首頁 > 嵌入式技術 > 設計應用 > 利用RTLinux開發嵌入式應用程序
利用RTLinux開發嵌入式應用程序
OFweek電子工程網
摘要: 本文以RTLinux為例,并結合最為業界關注的是RTAI進行討論,盡管這兩種實現方式在句法細節上存在差異,但工作方式基本一樣,因此所講述的內容對兩者都適用。
Abstract:
Key words :

  對于中國工程師來說,利用實時Linux開發嵌入式應用程序是他們面臨的困難之一,本文以RTLinux為例,并結合最為業界關注的是RTAI進行討論,盡管這兩種實現方式在句法細節上存在差異,但工作方式基本一樣,因此所講述的內容對兩者都適用。

  在實時任務與用戶進程相互通信的過程中,有些實時應用程序無需任何用戶界面即可在后臺平靜地運行,然而,越來越多的實時應用程序確實需要一個用戶界面及其它系統功能,如文件操作或聯網等,所有這些功能都必須在用戶空間內運行。問題是,用戶空間操作是非確定性的,而且與實時操作不兼容。

  幸運的是實時Linux具有一種可在時間上減弱實時與非實時操作的機制,這種機制表現為一種稱為實時FIFO的驅動程序。當insmod將rtl_fifo.o驅動程序插入Linux內核時,該驅動程序將自己注冊為RTLinux的一部分,并成為Linux驅動程序。一旦插入Linux內核,用戶空間進程和實時任務都可使用實時Linux FIFO。

  在深入探討實時FIFO的細節之前,還要回顧一下實時應用程序結構的某些部分(圖1)。有效的嵌入式應用程序設計方法是將實時部分與固有的非實時功能分離開來(表1)。如果應用程序的任一部分,如用戶界面、圖形、數據庫或網絡僅需軟實時性能,最好是將該部分寫入用戶空間。然后,僅將必須滿足時序要求的那部分寫成實時任務。

  注意,RTLinux(PSC,便攜式信號編碼)和RTAI(LXRT,Linux實時擴展)的最新版本已采用了一種可在用戶空間執行軟和硬實時任務的方法。

  

  任何硬實時任務都是在RTLinux的控制下運行的,該任務一般可執行周期性任務、處理中斷并與I/O設備驅動程序通信,以采集或輸出模擬和數字信息。當實時任務需要告訴用戶進程有一個事件將發生時,它便將這一消息送給實時FIFO。每一個FIFO都是在一個方向上傳送數據:從實時任務到用戶空間,或反之。因此,雙向通信需要使用兩個FIFO。任何讀出或寫入實時任務一側的操作都是非模塊操作,因此rtf_put()和rtf_get()都立即返回,而不管FIFO狀態是什么。

  從應用程序一側來看,FIFO就像一個常規文件。缺省情況下,RTLinux安裝程序將在/dev目錄下創建6?個實時FIFO節點;如果需要,還必須自己創建新的節點。例如,要創建/dev/rtf80,需采用如下命令:

  =========================

  mknod c 150 80;

  chmod 0666 /dev/rtf80

  =========================

  其中,150是實時FIFO主數,而80是rtf80的次數。

  從用戶進程的角度看,實時FIFO可執行標準文件操作。從實時任務來看,FIFO有兩種通信方式:直接調用RTLinux FIFO功能,或將FIFO作為一個RTLinux設備驅動程序,并使用open()、close()、read()和write()操作。要想將FIFO作為一個設備驅動程序,就必須將rtl_conf.h中的配置變量CONFIG_RTL_POSIX_IO設定為1。

  rtf_create_handler()可設置處理程序功能。每次Linux進程讀或寫FIFO時,rtl_fifo驅動程序都要調用該處理程序。應注意的是,該處理程序駐留在Linux內核,因此當Linux需要調用時,從該處理程序進行任何內核調用都是安全的。從該處理程序到實時任務間的最好通信方法是使用旗語或線程同步功能。最后,FIFO驅動程序還必須對內核存儲器進行配置。因此,實時線程內的rtf_create()不應調用。相反,可調用init_module()中的rtf_create()功能及cleanup_module()中的rtf_destroy()功能。

  例如,列表1給出了一個采用兩個FIFO的簡單數據采集應用程序的實時部分。兩個FIFO都是在init_module()創建,并賦予minor numbers 為1和2。在調用rtf_create(minor, size)之前,該程序在已創建該FIFO的情況下調用rtf_destroy(minor)。這種情況就是另一個模塊在開發過程中未被調用。然后,調用rtf_create_handler(ID, &pd_do_aout)以注冊帶該實時FIFO的數據采集模擬輸出功能pd_do_aout()。注意,創建實時線程pp_thread_ep()是因為它是周期性的,其間隔為1/100秒。

  每次周期性線程得到系統控制權后,它就調用rtf_put(ID,dataptr,size)以便將數據插入minor number為2的FIFO。Linux進程打開/dev/rtf2,從實時FIFO中讀取并顯示所采集的數據。該進程還打開/dev/rtf1,將數據寫入其它實時FIFO。當用戶移動屏幕滑動器以改變模擬輸出電壓時,進程就向該FIFO寫入一個新的值。RTLinux便調用pd_do_aout()處理程序,隨后pd_do_aout()利用rtf_get()從FIFO獲得值,并調用實際的硬件驅動程序以設置模擬輸出的電壓??梢钥吹剑瑢崟r任務和用戶進程是異步使用FIFO的。

  

  任務間的存儲器共享

  FIFO為用戶進程和實時任務的連接提供了一種方便的機制,但將它們作為消息隊列更合適。比如,一個實時線程可利用FIFO記錄測試結果,然后用戶進程就可讀取該結果,并將之存入數據庫文件。

  許多數據采集應用程序涉及到內核及用戶空間之間的大量數據。Linux內核v. 2.2.x并沒有為這些空間的數據共享提供任何機制,但v. 2.4.0版本預計會包括kiobuf結構。為解決現有穩定內核的這個缺點,RTLinux包括mbuff驅動程序。該驅動程序可利用vmalloc()分配虛擬內核存儲器的已命名存儲器區域,它采用的存儲器分配和頁面鎖定技巧跟大多數Linux中bttv幀抓取器(frame-grabber)驅動程序所用的一樣。

  更具體地說,mbuff一頁一頁地將虛擬內存鎖定到實際的物理內存頁面。任何實時或內核任務,或用戶進程在任何時間都可訪問該存儲器。通過將虛擬內存頁面鎖定到物理內存頁面,mbuff可確保所分配的頁面永久駐留在物理內存,而且不會發生頁面錯誤。換言之,當實時或內核進程訪問所分配的存儲器時,它可確保VMM不被調用。注意:由于實時任務執行期間實時Linux凍結標準內核的執行,任何對VMM的調用都會引起系統暫停。如果它要訪問并不位于物理RAM內的虛擬存儲頁面,那么即使正常的Linux內核驅動程序也會引起系統故障。

  由于mbuff是一種Linux驅動程序,其功能可通過設備節點/dev/mbuff實現。該節點可顯示幾個錄入點,其中包括可將內核空間地址映射到用戶空間的mmap()。它還可以利用錄入點ioctl()來控制。然而,并不需要復雜的結構及直接調用ioctl。相反,mbuff可為ioctl()調用提供一個包裹,而且僅僅調用兩個簡單的功能即可配置和釋放共享的存儲緩沖器。

  當然,不能從實時任務調用mbuff驅動程序,因為該驅動程序所調用的虛擬存儲器分配功能本身是不確定性操作。分配共享存儲器所需的時間依賴于主系統的存儲器容量以及CPU速度、磁盤驅動器性能和存儲器分配的現有狀態。因此,只能從模塊的Linux內核一側來分配共享存儲器,比如從init_module()或一個ioctl()請求開始。

  那么,一個共享緩沖器到底能分配多少存儲器呢?如果不是任務繁重的服務器或圖形應用,建議至少為Linux保留8MB存儲空間。為了獲得優化的配置,可在限制存儲器大小的同時測量實時應用程序的性能,以確定需要多少存儲空間。

  列表2給出了如何從實時任務和用戶進程方面訪問共享的存儲器。內核模塊和用戶任務采用同樣的功能集。當然,要想使用insmod mbuff.o,還必須將之置于Linux內核中。例如,mbuff_alloc(“buf_name”, size)可將符號名buf_name分配給一個緩沖器,而mbuff_free(“buf_name”, mbuf)可將之釋放。

  當第一次調用帶有符號緩沖器名的mbuff_alloc()時,mbuff執行實際的存儲器分配。而當從內核模塊或用戶進程再次調用該功能時,它只是簡單地增加使用數(usage count)及將指針返回現有的緩沖器。每次調用mbuff_free()都會減少使用數,直至為零,這時mbuff就去分配帶符號名的緩沖器。這種方法從多個內核模塊和用戶進程獲得一個指向同一共享緩沖器的指針,從而解決了問題。它還可確保共享緩沖器一直有效,直到最后的應用程序釋放它。請注意,是實時內核還是用戶進程執行實際的buf1配置依賴于誰先獲得控制權。

  還有一個“笨”方法可在實時應用程序、內核模塊和用戶應用程序間共享存儲器。對于嵌入式應用,該方法還是可以接受的。例如,如果PC帶有128MB RAM,可將線搜索路徑=“mem=120m”添加進lilo.conf文件(列表3)。當啟動帶有Linux內核和RTLinux 2.3的系統時,Linux僅使用120MB內存。OS也不用剩下的8MB內存(物理地址為0x7F00000到0x7FFFFFF),而是留給在OS下運行的各種任務共享。要想從用戶進程獲取存儲器地址并訪問預留的存儲器,必須用O_RDWR訪問模式來打開/dev/mem驅動程序,然后利用mmap()保留存儲器(列表4)。而從實時模塊或內核驅動程序一側進行,則必須使用ioremap(0x7F00000, 0x100000)才能獲取這8MB (0x100000字節)預留內存。

  這種方法有利有弊。既不能通過預留內存的所有權,也不能通過讀或寫來獲取控制權。正確地配置和釋放大量內存的機制尚未問世。另外,無論實時進程是否需要,該內存都不能為Linux所用。

  也許存儲器共享笨方法的唯一適用場合是專為特定應用而定制的小型嵌入式系統,因為此時可為小型化而放棄使用mbuff驅動程序。

  對于中國工程師來說,利用實時Linux開發嵌入式應用程序是他們面臨的困難之一,本文以RTLinux為例,并結合最為業界關注的是RTAI進行討論,盡管這兩種實現方式在句法細節上存在差異,但工作方式基本一樣,因此所講述的內容對兩者都適用。

  在實時任務與用戶進程相互通信的過程中,有些實時應用程序無需任何用戶界面即可在后臺平靜地運行,然而,越來越多的實時應用程序確實需要一個用戶界面及其它系統功能,如文件操作或聯網等,所有這些功能都必須在用戶空間內運行。問題是,用戶空間操作是非確定性的,而且與實時操作不兼容。

  幸運的是實時Linux具有一種可在時間上減弱實時與非實時操作的機制,這種機制表現為一種稱為實時FIFO的驅動程序。當insmod將rtl_fifo.o驅動程序插入Linux內核時,該驅動程序將自己注冊為RTLinux的一部分,并成為Linux驅動程序。一旦插入Linux內核,用戶空間進程和實時任務都可使用實時Linux FIFO。

  在深入探討實時FIFO的細節之前,還要回顧一下實時應用程序結構的某些部分(圖1)。有效的嵌入式應用程序設計方法是將實時部分與固有的非實時功能分離開來(表1)。如果應用程序的任一部分,如用戶界面、圖形、數據庫或網絡僅需軟實時性能,最好是將該部分寫入用戶空間。然后,僅將必須滿足時序要求的那部分寫成實時任務。

  注意,RTLinux(PSC,便攜式信號編碼)和RTAI(LXRT,Linux實時擴展)的最新版本已采用了一種可在用戶空間執行軟和硬實時任務的方法。

  

  任何硬實時任務都是在RTLinux的控制下運行的,該任務一般可執行周期性任務、處理中斷并與I/O設備驅動程序通信,以采集或輸出模擬和數字信息。當實時任務需要告訴用戶進程有一個事件將發生時,它便將這一消息送給實時FIFO。每一個FIFO都是在一個方向上傳送數據:從實時任務到用戶空間,或反之。因此,雙向通信需要使用兩個FIFO。任何讀出或寫入實時任務一側的操作都是非模塊操作,因此rtf_put()和rtf_get()都立即返回,而不管FIFO狀態是什么。

  從應用程序一側來看,FIFO就像一個常規文件。缺省情況下,RTLinux安裝程序將在/dev目錄下創建6?個實時FIFO節點;如果需要,還必須自己創建新的節點。例如,要創建/dev/rtf80,需采用如下命令:

  =========================

  mknod c 150 80;

  chmod 0666 /dev/rtf80

  =========================

  其中,150是實時FIFO主數,而80是rtf80的次數。

  從用戶進程的角度看,實時FIFO可執行標準文件操作。從實時任務來看,FIFO有兩種通信方式:直接調用RTLinux FIFO功能,或將FIFO作為一個RTLinux設備驅動程序,并使用open()、close()、read()和write()操作。要想將FIFO作為一個設備驅動程序,就必須將rtl_conf.h中的配置變量CONFIG_RTL_POSIX_IO設定為1。

  rtf_create_handler()可設置處理程序功能。每次Linux進程讀或寫FIFO時,rtl_fifo驅動程序都要調用該處理程序。應注意的是,該處理程序駐留在Linux內核,因此當Linux需要調用時,從該處理程序進行任何內核調用都是安全的。從該處理程序到實時任務間的最好通信方法是使用旗語或線程同步功能。最后,FIFO驅動程序還必須對內核存儲器進行配置。因此,實時線程內的rtf_create()不應調用。相反,可調用init_module()中的rtf_create()功能及cleanup_module()中的rtf_destroy()功能。

  例如,列表1給出了一個采用兩個FIFO的簡單數據采集應用程序的實時部分。兩個FIFO都是在init_module()創建,并賦予minor numbers 為1和2。在調用rtf_create(minor, size)之前,該程序在已創建該FIFO的情況下調用rtf_destroy(minor)。這種情況就是另一個模塊在開發過程中未被調用。然后,調用rtf_create_handler(ID, &pd_do_aout)以注冊帶該實時FIFO的數據采集模擬輸出功能pd_do_aout()。注意,創建實時線程pp_thread_ep()是因為它是周期性的,其間隔為1/100秒。

  每次周期性線程得到系統控制權后,它就調用rtf_put(ID,dataptr,size)以便將數據插入minor number為2的FIFO。Linux進程打開/dev/rtf2,從實時FIFO中讀取并顯示所采集的數據。該進程還打開/dev/rtf1,將數據寫入其它實時FIFO。當用戶移動屏幕滑動器以改變模擬輸出電壓時,進程就向該FIFO寫入一個新的值。RTLinux便調用pd_do_aout()處理程序,隨后pd_do_aout()利用rtf_get()從FIFO獲得值,并調用實際的硬件驅動程序以設置模擬輸出的電壓??梢钥吹?,實時任務和用戶進程是異步使用FIFO的。

  

  任務間的存儲器共享

  FIFO為用戶進程和實時任務的連接提供了一種方便的機制,但將它們作為消息隊列更合適。比如,一個實時線程可利用FIFO記錄測試結果,然后用戶進程就可讀取該結果,并將之存入數據庫文件。

  許多數據采集應用程序涉及到內核及用戶空間之間的大量數據。Linux內核v. 2.2.x并沒有為這些空間的數據共享提供任何機制,但v. 2.4.0版本預計會包括kiobuf結構。為解決現有穩定內核的這個缺點,RTLinux包括mbuff驅動程序。該驅動程序可利用vmalloc()分配虛擬內核存儲器的已命名存儲器區域,它采用的存儲器分配和頁面鎖定技巧跟大多數Linux中bttv幀抓取器(frame-grabber)驅動程序所用的一樣。

  更具體地說,mbuff一頁一頁地將虛擬內存鎖定到實際的物理內存頁面。任何實時或內核任務,或用戶進程在任何時間都可訪問該存儲器。通過將虛擬內存頁面鎖定到物理內存頁面,mbuff可確保所分配的頁面永久駐留在物理內存,而且不會發生頁面錯誤。換言之,當實時或內核進程訪問所分配的存儲器時,它可確保VMM不被調用。注意:由于實時任務執行期間實時Linux凍結標準內核的執行,任何對VMM的調用都會引起系統暫停。如果它要訪問并不位于物理RAM內的虛擬存儲頁面,那么即使正常的Linux內核驅動程序也會引起系統故障。

  由于mbuff是一種Linux驅動程序,其功能可通過設備節點/dev/mbuff實現。該節點可顯示幾個錄入點,其中包括可將內核空間地址映射到用戶空間的mmap()。它還可以利用錄入點ioctl()來控制。然而,并不需要復雜的結構及直接調用ioctl。相反,mbuff可為ioctl()調用提供一個包裹,而且僅僅調用兩個簡單的功能即可配置和釋放共享的存儲緩沖器。

  當然,不能從實時任務調用mbuff驅動程序,因為該驅動程序所調用的虛擬存儲器分配功能本身是不確定性操作。分配共享存儲器所需的時間依賴于主系統的存儲器容量以及CPU速度、磁盤驅動器性能和存儲器分配的現有狀態。因此,只能從模塊的Linux內核一側來分配共享存儲器,比如從init_module()或一個ioctl()請求開始。

  那么,一個共享緩沖器到底能分配多少存儲器呢?如果不是任務繁重的服務器或圖形應用,建議至少為Linux保留8MB存儲空間。為了獲得優化的配置,可在限制存儲器大小的同時測量實時應用程序的性能,以確定需要多少存儲空間。

  列表2給出了如何從實時任務和用戶進程方面訪問共享的存儲器。內核模塊和用戶任務采用同樣的功能集。當然,要想使用insmod mbuff.o,還必須將之置于Linux內核中。例如,mbuff_alloc(“buf_name”, size)可將符號名buf_name分配給一個緩沖器,而mbuff_free(“buf_name”, mbuf)可將之釋放。

  當第一次調用帶有符號緩沖器名的mbuff_alloc()時,mbuff執行實際的存儲器分配。而當從內核模塊或用戶進程再次調用該功能時,它只是簡單地增加使用數(usage count)及將指針返回現有的緩沖器。每次調用mbuff_free()都會減少使用數,直至為零,這時mbuff就去分配帶符號名的緩沖器。這種方法從多個內核模塊和用戶進程獲得一個指向同一共享緩沖器的指針,從而解決了問題。它還可確保共享緩沖器一直有效,直到最后的應用程序釋放它。請注意,是實時內核還是用戶進程執行實際的buf1配置依賴于誰先獲得控制權。

  還有一個“笨”方法可在實時應用程序、內核模塊和用戶應用程序間共享存儲器。對于嵌入式應用,該方法還是可以接受的。例如,如果PC帶有128MB RAM,可將線搜索路徑=“mem=120m”添加進lilo.conf文件(列表3)。當啟動帶有Linux內核和RTLinux 2.3的系統時,Linux僅使用120MB內存。OS也不用剩下的8MB內存(物理地址為0x7F00000到0x7FFFFFF),而是留給在OS下運行的各種任務共享。要想從用戶進程獲取存儲器地址并訪問預留的存儲器,必須用O_RDWR訪問模式來打開/dev/mem驅動程序,然后利用mmap()保留存儲器(列表4)。而從實時模塊或內核驅動程序一側進行,則必須使用ioremap(0x7F00000, 0x100000)才能獲取這8MB (0x100000字節)預留內存。

  這種方法有利有弊。既不能通過預留內存的所有權,也不能通過讀或寫來獲取控制權。正確地配置和釋放大量內存的機制尚未問世。另外,無論實時進程是否需要,該內存都不能為Linux所用。

  也許存儲器共享笨方法的唯一適用場合是專為特定應用而定制的小型嵌入式系統,因為此時可為小型化而放棄使用mbuff驅動程序。

  中斷

  RTLinux有兩種中斷:硬中斷和軟中斷。軟中斷就是常規Linux內核中斷,它的優點在于可無限制地使用Linux內核調用。這類中斷作為硬中斷處理的第二部分還是相當有用的(由參考文獻5可獲得更多有關Linux環境下中斷處理的細節)。

  硬(實時)中斷是安裝實時Linux的前提。要安裝中斷處理程序,先調用rtl_request_irq(。..),然后調用rtl_free_irq()釋放它。依賴于不同的系統,實時Linux下硬(或實時)中斷的延遲是15μs的數量級。較快的處理器具有較好的延遲。如果想在實時處理程序和常規Linux驅動程序中處理同一設備IRQ,必須為每一個硬中斷單獨設置IRQ。

  列表5給出了安裝實時中斷處理程序的過程。RTLinux在執行實時中斷處理程序時將禁止IRQ。應注意,該代碼須在退出實時中斷處理程序前調用rtl_hard_enable_irq()才能重新使能中斷。

  有兩個問題影響直接從實時中斷處理程序調用Linux內核功能:內核禁止所有中斷及不定義執行內容。還應注意的是,這里也不能執行浮點操作。利用實時中斷處理程序來控制線程執行是避免出現這些問題的好辦法。本例采用pthread_wakeup_np()功能來喚醒一個實時線程。中斷處理程序可處理即時的工作,余下的由該線程解決。

  SMP結構的優點

  實時Linux都支持多處理器架構。對稱多處理器(SMP)結構采用了高級可編程中斷控制器(APIC),奔騰級處理器都有片上本地APIC,可為本地處理器傳送中斷。SMP(甚至單處理器母板)都有I/O APIC,可收集來自外設的中斷請求,并將它們傳送給本地APIC。舊的8259 PIC速度很慢,所處理的中斷向量數不充分,迫使設備共享中斷,使得中斷處理更慢。但是,APIC可解決這些問題。通過為每個設備請求設置一個特定的IRQ,系統可減少中斷延遲,APIC還可加速同步代碼。

  實時Linux可充分利用APIC。在SMP系統中,實時調度程序利用APIC,而不是采用過時的8254芯片來完成時序分配。由于PC的兼容性,8254位于每一個ISA總線上,而且每一個再編程設備的調用都要占用處理器周期。一個千兆赫CPU要浪費數百個處理器周期來等待8MHz定時器(大約2.5μs)。APIC工作在總線頻率,而且可立即執行所有的定時器操作,這意味著必須利用本地APIC時鐘在AMP機器上獲取更高的周期性頻率(雙P-III-500 CPU可在100kHz運行周期性實時線程,而無明顯的性能損失)。

  實時Linux能很好地執行多處理任務,它為每個CPU實施單獨的進程。調用pthread_create()可創建一個在現有CPU上運行的線程。還可用pthread_attr_setcpu_np()將該線程分配給一個特定的CPU,以改變線程屬性。在調用這一功能之前,必須首先初始化線程屬性。

  RTLinux v. 3包括reserve_cpu功能,可預留SMP平臺上的一個CPU,專供RTLinux使用。它可運行于2.4x內核,RTAI也具有幾乎同樣的功能。

  如果想將任務分給某一特定的CPU,請留意“pset”方案(http://isunix.it.ilstu.edu/thockin/pset/)。利用該內核可將一個SMP處理器專門分配給一個用戶應用程序,甚至可從Linux處理器組中調用一個處理器專用于實時任務。

  同步基元

  早期的實時Linux沒有同步基元。現在,POSIX型的旗語、互斥和信號在最新的實時Linux版本中都已出現。雖然在實時設計中采用這些同步基元還存在問題,但同步或用信號表示實時任務和用戶應用程序很有意義,然而,這要求軟件開發者具有高超的技能,這一問題已超出本文的討論范圍。

  快速學習pthread_mutex_init()、pthread_mutex_lock()、pthread_mutex_trylock()、pthread_mutex_unlock()和pthread_mutex_destroy()等同步功能的最好方法是查看。/examples/mutex/mutex.c。特別要提醒的是。/examples/mutex/sema_test.c文件是學習旗語的很好起點。

  實時Linux發展方向

  實時Linux與Linux一樣仍然處于不斷發展之中。每一個新的版本都添加了更多的特性和功能。實時Linux正朝著更好的POSIX 1003.x實現方向發展,最新的特性包括用戶空間進程的實時支持、互斥、信號、旗語、實時存儲器管理和擴展的SMP支持等。如果還未確定下一個項目采用哪個實時系統,可下載一種實時Linux版本了解一下。其實,Linux已經是一種成熟的OS,而且具備實時擴展版本,它是嵌入式應用的最佳選擇之一。

  列表1:實時FIFO的使用。

  #define IN_FIFO_ID 1

  #define OUT_FIFO_ID 2

  #define IN_FIFO_LENGTH 0x100

  #define OUT_FIFO_LENGTH 0x100

  // RT FIFO invokes this function every time the user process writes

  // something into /dev/rtf1

  int pd_do_aout(unsigned int fifo)

  {

  u32 ao_value;

  while ((err = rtf_get(IN_FIFO_ID, &ao_value, sizeof(u32)))

  == sizeof(u32))

  {

  pd_aout_write(board, ao_value);

  }

  if (err != 0) return -EINVAL; else return 0;

  }

  void *pp_thread_ep(void *rate) // our periodic thread

  {

  u16 ain_data;

  。..

  ret = pd_ain_read(board, &ain_data); // read value from analog in

  // write to the output FIFO where user process can read it from /dev/rtf2

  ret = rtf_put(OUT_FIFO_ID, &ain_data, sizeof(u16));

  。.. process ret for return codes 。..

  }

  init_module(void)

  { 。..

  // free up the resource, just in case

  rtf_destroy(IN_FIFO_ID);

  rtf_destroy(OUT_FIFO_ID);

  // create fifos we can talk via /dev/rtf1 and /dev/rtf2

  rtf_create(IN_FIFO_ID, IN_FIFO_LENGTH); // rt task 《- user process

  rtf_create(OUT_FIFO_ID, OUT_FIFO_LENGTH); // rt task -》 user process

  rtf_create_handler(IN_FIFO_ID, &pd_do_aout);

  。..

  }

  cleanup_module(void)

  { 。..

  rtf_destroy(IN_FIFO_ID); // free up the resource, just in case

  rtf_destroy(OUT_FIFO_ID); // free up the resource, just in case

  。..

  }

  列表2:利用mbuff共享存儲器。

  // user application

  #include “mbuff.h”

  。..

  u16* buf1; // pointer to the buffer to store 16-bit samples

  main (int argc,char *argv[])

  {

  。..

  buf1 = (u16*) mbuff_alloc(“buf1”,0x100000);

  if (buf1 == NULL) { // failure to allocate buffer }

  sprintf((char*)buf1, “Hello, rt-task!\n”); // put some data into buffer

  // now you can tell your realtime module to that you wrote

  // something to the buffer, say, using RT FIFO

  。..

  mbuff_free(“buf1”, (void*)buf1); // free buffer when you don‘t need it

  }

  // realtime module

  #include “mbuff.h”

  。..

  u16* buf1; // pointer to the buffer to store 16-bit samples

  init_module(void) // allocate shared buffer during init of realtime module

  {

  // allocate 1MB buffer named “buf1”

  buf1 = (u16*) mbuff_alloc(“buf1”, 0x100000);

  if (buf1 == NULL) { failure to allocate buffer }

  。..

  }

  cleanup_module(void) // deallocate buffer during cleanup of realtime module

  { mbuff_free(“buf1”, buf1); // free it

  。..

  }

  列表3:設定可在lilo.conf文件中使用的存儲器內核數量。

  。..

  image=/boot/vmlinuz_2_2_14.rtl_2_3

  append=“mem=120m”

  root=/dev/hda2

  label=RTL.2.3

  。..

  列表4:設置并采用笨方法共享內存。

  // User space code:

  if (fd = open(“/dev/mem”, O_RDWR)) 《 0) { 。..oops! error }

  rtshm_ptr = (char * ) mmap (0, 0x100000, PROT_READ | PROT_WRITE, MAP_SHARED,

  fd, 0x7F00000);

  if (rtshm_ptr == MAP_FAILED) { 。..oops! error }

  else { 。..use it }

  // and in your real-time module:

  rtshmbase = (long*) ioremap(0x7f00000, 0x100000);

  列表5:獲得中斷向量。

  // thread used as deferred procedure call created in init_module

  void *pp_thread_ep(void* arg)

  {

  while (1)

  {

  pthread_wait_np(); // wait to be woken up.。.

  // process interrupt now in realtime kernel context

  。..

  }

  }

  // interrupt handler

  unsigned int irq_handler(unsigned int irq, struct pt_regs *regs)

  {

  pthread_wakeup_np(pp_thread); // wake up thread to do IRQ post-processing

  rtl_hard_enable_irq(IRQ_LINE); // re-enable IRQ

  return 0;

  }

  int init_module(void)

  {

  // create thread pp_thread to wake up by interrupt handler

  。..

  rtl_request_irq(IRQ_LINE, irq_handler); // request.。.

  rtl_hard_enable_irq(IRQ_LINE); // 。..and enable interrupt handler

  }

  void cleanup_module(void)

  {

  rtl_free_irq(IRQ_LINE);

  // do the rest of clean=up sequence

  }

此內容為AET網站原創,未經授權禁止轉載。
热re99久久精品国产66热_欧美小视频在线观看_日韩成人激情影院_庆余年2免费日韩剧观看大牛_91久久久久久国产精品_国产原创欧美精品_美女999久久久精品视频_欧美大成色www永久网站婷_国产色婷婷国产综合在线理论片a_国产精品电影在线观看_日韩精品视频在线观看网址_97在线观看免费_性欧美亚洲xxxx乳在线观看_久久精品美女视频网站_777国产偷窥盗摄精品视频_在线日韩第一页
  • <strike id="ygamy"></strike>
  • 
    
      • <del id="ygamy"></del>
        <tfoot id="ygamy"></tfoot>
          <strike id="ygamy"></strike>
          在线亚洲自拍| 国产欧美亚洲一区| 免费毛片一区二区三区久久久| 久久精品国产第一区二区三区最新章节| 欧美一区二区免费观在线| 欧美精品一区在线| 99亚洲伊人久久精品影院红桃| 亚洲在线免费视频| 国产精品免费网站在线观看| 伊人久久久大香线蕉综合直播| 欧美三区在线| 国产精品一二三四区| 国产精品va在线播放| 91久久综合| 国产精品久久久久久久久久三级| 国产一区二区中文| 国产精品揄拍500视频| 这里只有视频精品| 一区在线电影| 亚洲主播在线| 免费欧美高清视频| 国模私拍一区二区三区| 国产精品一区=区| 欧美日韩一区二区三区免费看| 蜜月aⅴ免费一区二区三区| 国产乱码精品| 亚洲人成网站精品片在线观看| 亚洲一区二区在线视频| 亚洲在线播放电影| 国产女人aaa级久久久级| 樱花yy私人影院亚洲| 在线电影欧美日韩一区二区私密| 久久久精品久久久久| 久久人人97超碰国产公开结果| 欧美日韩色综合| 国产深夜精品| 欧美日韩在线观看一区二区三区| 亚洲一二三级电影| 免费不卡在线观看| 最新国产拍偷乱拍精品| 久久一日本道色综合久久| 国产拍揄自揄精品视频麻豆| 亚洲天堂激情| 国产精品一卡二卡| 在线精品高清中文字幕| 国产欧美日韩免费看aⅴ视频| 激情亚洲网站| 精品1区2区| 国产一区二区丝袜高跟鞋图片| 亚洲国产精品一区二区尤物区| 国外视频精品毛片| 久久精品夜夜夜夜久久| 一个色综合导航| 一区二区日韩免费看| 欧美天天影院| 久久精品2019中文字幕| 久久中文字幕一区| 欧美视频网站| 亚洲高清不卡在线| 99国产精品99久久久久久| 欧美三级黄美女| 蜜臀久久99精品久久久画质超高清| 亚洲福利视频专区| 国产日韩亚洲欧美综合| 欧美成人精品h版在线观看| 亚洲国产欧美日韩| 一区二区毛片| 国产精品中文字幕在线观看| 亚洲欧美激情一区二区| 久久精品国产亚洲5555| 亚洲国产国产亚洲一二三| 国产精品国产三级国产aⅴ入口| 女女同性女同一区二区三区91| 亚洲国产精品t66y| 免费一区视频| 日韩一级视频免费观看在线| 国产日韩在线一区二区三区| 影视先锋久久| 国产午夜精品久久久久久免费视| a4yy欧美一区二区三区| 亚洲伊人色欲综合网| 久久国产综合精品| 国产日韩一区| 一区二区三区日韩精品视频| 在线视频欧美日韩| 久久精品人人做人人爽电影蜜月| 亚洲狠狠婷婷| 国产一区二区三区奇米久涩| 亚洲国产专区校园欧美| 欧美色欧美亚洲另类二区| 一本久久知道综合久久| 国产精品va在线播放我和闺蜜| 国产精品美女久久久久久久| 狠狠色狠狠色综合系列| 国产主播喷水一区二区| 亚洲欧美日韩区| 国产视频观看一区| 欧美黄色一区| 国产精品第一页第二页第三页| 欧美极品aⅴ影院| 亚洲一二三级电影| 亚洲一级免费视频| 久久久亚洲综合| 久久久久久有精品国产| 美女免费视频一区| 美女网站在线免费欧美精品| 亚洲精品乱码久久久久久按摩观| 国产精品国产三级国产aⅴ浪潮| 欧美日韩a区| 欧美日在线观看| 欧美中文字幕在线播放| 国产精品一区二区三区观看| 国产精品中文在线| 久久视频国产精品免费视频在线| 在线日韩中文| 美国十次了思思久久精品导航| 欧美专区一区二区三区| 亚洲特级片在线| 亚洲欧美影音先锋| 午夜一区二区三区不卡视频| 在线欧美视频| 一本到12不卡视频在线dvd| 国产欧美在线观看| 国产一区二区三区高清播放| 国产精品一二三| 亚洲欧美韩国| 韩国女主播一区二区三区| 亚洲一区二区三区乱码aⅴ| 午夜精品久久久久久久久久久| 欧美精品一区二区三区高清aⅴ| 欧美日韩精品免费在线观看视频| 国产精品亚洲一区二区三区在线| 国产精品欧美久久| 欧美精品不卡| 亚洲欧洲一区二区在线观看| 亚洲欧美99| 99热免费精品在线观看| 欧美sm重口味系列视频在线观看| 国产精品国内视频| 亚洲影院色无极综合| 久久天堂av综合合色| 欧美激情一级片一区二区| 激情视频一区| 久久大综合网| 国产欧美日韩亚洲精品| 国产精品a级| 亚洲精品午夜精品| aa成人免费视频| 好看的日韩视频| 欧美激情亚洲另类| 亚洲高清免费在线| 99精品欧美一区二区蜜桃免费| 一本一本久久a久久精品综合麻豆| 欧美激情五月| 99成人精品| 欧美另类69精品久久久久9999| 一区二区三区精品国产| 国产有码一区二区| 欧美二区不卡| 国产精品久久二区二区| 日韩天天综合| 最近看过的日韩成人| 中文日韩欧美| 亚洲缚视频在线观看| 亚洲片国产一区一级在线观看| 久久精彩免费视频| 国内精品久久久久影院色| 国产精品久久网站| 欧美午夜不卡视频| 亚洲狼人精品一区二区三区| 国产一区二区三区在线观看免费视频| 亚洲少妇中出一区| 老鸭窝毛片一区二区三区| 国产专区一区| 亚洲国产91| 在线观看成人小视频| 国产精品视频999| 国产精品美女久久久免费| 亚洲欧美日韩国产成人| 国产精品美女久久久浪潮软件| 久久琪琪电影院| 亚洲黄色毛片| 欧美日韩国产页| **网站欧美大片在线观看| 99在线精品免费视频九九视| 国产欧美一区在线| 欧美日韩国产成人| 欧美高清在线视频观看不卡| 在线观看欧美激情| 欧美aaaaaaaa牛牛影院| 久久久久看片| 美女久久一区| 欧美日韩精品免费观看视频| 久久免费国产精品| 国产精品av久久久久久麻豆网| 久久av免费一区| 黄色小说综合网站| 亚洲大片免费看| 亚洲国产成人精品女人久久久| 在线视频精品| 国产一二三精品| 噜噜爱69成人精品| 国产一区在线播放| 亚洲综合欧美| 国产在线观看精品一区二区三区| 国产一区二区三区无遮挡| 国产综合在线看| 欧美激情亚洲自拍| 曰韩精品一区二区| 国产精品国产| 国产欧美日韩在线播放| 欧美色网在线| 国内伊人久久久久久网站视频| 国语自产精品视频在线看一大j8| 午夜日韩视频| 一本色道**综合亚洲精品蜜桃冫| 久久久久91| 久久国产一区二区三区| 国产精品美女在线观看| 国产精品一二| 欧美日韩国产成人在线| 欧美一级成年大片在线观看| 国产精品高潮呻吟久久av黑人| 99精品欧美一区二区三区| 麻豆精品在线视频| 在线精品视频在线观看高清| 中文在线资源观看网站视频免费不卡| 亚洲免费中文字幕| 免费在线观看日韩欧美| 国产精品jvid在线观看蜜臀| 蜜臀av性久久久久蜜臀aⅴ| 久久久美女艺术照精彩视频福利播放| 狠狠色伊人亚洲综合网站色| 久久久久国产精品厨房| 国内精品视频一区| 极品中文字幕一区| 亚洲日本激情| 亚洲黄色成人网| 最新69国产成人精品视频免费| 一区二区三区在线观看欧美| 久久综合婷婷| 国语自产精品视频在线看一大j8| 欧美色另类天堂2015| 日韩亚洲欧美精品| 欧美日韩精品在线观看| 国产日韩精品一区二区浪潮av| 国产一区二区三区黄| 亚洲天堂av高清| 一区二区三区日韩欧美精品| 极品尤物久久久av免费看| 欧美成人亚洲| 国产乱码精品| 亚洲女性喷水在线观看一区| 久久免费99精品久久久久久| 久久久久久久国产| 亚洲午夜一区二区| 国产精品爱啪在线线免费观看| 亚洲第一伊人| 夜夜嗨av色综合久久久综合网| 激情成人中文字幕| 一本色道婷婷久久欧美| 久久国产精品第一页| 欧美一区二区在线看| 欧美国产高潮xxxx1819| 亚洲精品视频在线播放| 蜜臀av在线播放一区二区三区| 91久久精品日日躁夜夜躁欧美| 久久人人九九| 一区二区三区欧美日韩| 亚洲国产精选| 欧美大片在线观看一区二区| 亚洲国产高清在线| 欧美成在线视频| 日韩视频免费观看高清在线视频| 欧美日韩久久精品| 在线午夜精品自拍| 国产日产精品一区二区三区四区的观看方式| 亚洲午夜久久久久久久久电影网| 最新中文字幕一区二区三区| 国产视频欧美视频| 亚洲毛片av| 国产欧美日本在线| 久久久久国产精品一区三寸| 蜜臀91精品一区二区三区| 麻豆久久婷婷| 狠狠色丁香久久婷婷综合_中| 国产婷婷成人久久av免费高清| 免费亚洲电影在线| 蜜桃av一区二区| 午夜精品视频在线观看一区二区| 亚洲国产综合91精品麻豆| 久久人体大胆视频| 欧美 日韩 国产一区二区在线视频| 亚洲欧美日韩国产| 久久av一区二区三区亚洲| 国产精品videosex极品| 亚洲视频免费在线| 国产精品成人国产乱一区| 国产精品系列在线播放| 亚洲黄色影院| 久久久999精品| 日韩视频在线永久播放| 亚洲毛片av在线| 欧美日韩亚洲另类| 国产亚洲二区| 国产一区二区三区久久久久久久久| 日韩系列欧美系列| 国产毛片精品国产一区二区三区| 国产亚洲人成网站在线观看| 免费在线观看精品| 在线播放中文字幕一区| 欧美色综合天天久久综合精品| 日韩亚洲欧美一区二区三区| 欧美激情精品久久久久久久变态| 国产一区二区三区高清| 欧美一区二区三区在线看| 亚洲小说欧美另类婷婷| 欧美激情一区二区| 欧美区在线播放| 国产亚洲福利社区一区| 欧美午夜电影在线观看| 国产精品免费观看在线| 美女脱光内衣内裤视频久久网站| 亚洲欧美制服中文字幕| 欧美日韩一区三区| 欧美日韩大片| 国产精品一卡二| 欧美激情小视频|