基于嵌入式Linux和MiniGUI的SIP電話設(shè)計
2 各線程模塊的實現(xiàn)
主線程模塊主要完成系統(tǒng)各個功能模塊的初始化工作,也是程序的入口點,MiniGUI程序的入口點為MiniGUIMain()函數(shù);配置文件的加載擬完成從根文件系統(tǒng)到內(nèi)存的加載,然后進(jìn)行解析,存放在全局SIP配置參數(shù)結(jié)構(gòu)中。配置文件用來存放呼叫處理模塊和語音傳輸模塊使用的參數(shù),具體包括:本機(jī)IP地址、子網(wǎng)掩碼、網(wǎng)關(guān)地址、SIP服務(wù)器IP地址,SIP端口號、用戶名、本機(jī)電話號碼、密碼、RTP端口號、被叫電話號碼和注冊間隔時間。初始化音頻設(shè)備擬完成打開音頻設(shè)備文件,設(shè)置音頻設(shè)備的采樣頻率,量化位數(shù)和聲道數(shù)目。打開音頻設(shè)備文件可
通過調(diào)用Linux系統(tǒng)函數(shù)audio_fd=open(“/dev/dsp”,O_RDWR)來實現(xiàn),調(diào)用成功后將返回音頻設(shè)備的文件描述符。設(shè)置音頻設(shè)備的采樣頻率,量化位數(shù)和聲道數(shù)目可通過調(diào)用ioctl(fd,….)函數(shù)來實現(xiàn)。功能按鍵設(shè)備的初始化很簡單,直接調(diào)用buttons_fd=open“/dev/b-uttons”,O)函數(shù)打開按鍵設(shè)備文件即可。創(chuàng)建RTP會話實例,可通過調(diào)用JRTPLIB庫的RTPSession類來完成,然后調(diào)用RTPSession類的Create()方法來對其進(jìn)行初始化,創(chuàng)建完成后,需設(shè)置RTP會話實例的傳輸參數(shù)和會話參數(shù)。eXoSIP協(xié)議棧的初始化直接調(diào)用eXoSIP協(xié)議棧所提供的初始化函數(shù)。七個子線程的創(chuàng)建可通過調(diào)用pthread_create函數(shù)來完成。SIP配置信息的顯示擬完成配置文件中的信息在MiniGUI主窗口上的顯示,主要顯示本機(jī)的IP地址和端口號、SIP服務(wù)器的IP地址、本機(jī)號碼、本機(jī)用戶名。SIP狀態(tài)信息的顯示擬完成對整個SIP事務(wù)遷移狀態(tài)的顯示。例如,如果收到180Ringing消息,則在MiniGUI主窗口上顯示“對方正在響鈴”,如果收到定時器的超時消息,則在MiniGUI主窗口上顯示“無人接聽,請稍后再撥”。SIP狀態(tài)信息的顯示是一個消息驅(qū)動的動態(tài)顯示。SIP配置信息和狀態(tài)信息的顯示直接采用MiniGUI的窗口模型和消息處理機(jī)制。SIP配置信息的顯示直接通過調(diào)用MiniGUI提供的TextOut(hdc,O,O,host_ip)將SIP參數(shù)結(jié)構(gòu)中的參數(shù)顯示在MiniGUI主窗口上。SIP狀態(tài)信息的顯示必須為每個SIP事務(wù)消息定義相對應(yīng)的MiniGUI消息,以180 Ringing消息和定時器超時消息為例,自定義消息如下:
#define MSG_180Ringing(MSG_USER+10)
#define MSG_TimerC(MSG_USER+11)
當(dāng)呼叫處理模塊子線程收到IP網(wǎng)絡(luò)上的“180 Ringing”消息和Linux內(nèi)核的定時器超時消息后,則通過調(diào)用SendMessage(hWnd,MSG_180-Ringing,0,0L)向MiniGUI主線程發(fā)送MSG_180Ringing消息,主線程通過調(diào)用GetMessage()函數(shù)獲取呼叫處理模塊子線程所發(fā)過來的消息,通過調(diào)用DispatchMessage(Msg)函數(shù)把這些消息發(fā)送到窗口過程函數(shù)進(jìn)行處理。窗口過程函數(shù)收到相應(yīng)的消息,首先判斷消息的類型,若是MSG_180Ringing消息,然后調(diào)用TextOut(hdc,0,0,“對方正在響鈴”)函數(shù)在窗口上顯示“對方正在響鈴”字樣。
呼叫處理模塊子線程可直接調(diào)用eXoSIP協(xié)議棧所提供的API函數(shù)集,eXoSIP是在oSIP2的基礎(chǔ)上對SIP消息的API作了更上層的封裝,能夠很容易實現(xiàn)SIP電話的呼叫過程控制。呼叫處理模塊子線程實現(xiàn)的難點是當(dāng)呼叫連接成功后,如何啟動語音采集、語音編碼、數(shù)據(jù)發(fā)送、數(shù)據(jù)接收、語音解碼和語音播放6個子線程。本設(shè)計采用Linux線程間通信-管道機(jī)制向其它6個子線程發(fā)送啟動標(biāo)識,6個子線程接收到啟動標(biāo)識后,喚醒各自的線程,進(jìn)行相應(yīng)的語音處理和語音的傳輸。同樣,當(dāng)呼叫連接釋放時,呼叫處理模塊子線程向6個子線程發(fā)送停止標(biāo)識,6個子線程接收到停止標(biāo)識后,停止語音處理和語音的傳輸,阻塞各自的線程。
語音采集模塊、語音編碼模塊、數(shù)據(jù)發(fā)送模塊、數(shù)據(jù)接收模塊、語音解碼模塊和語音播放模塊6個子線程的過程控制是一樣的,首先進(jìn)入主循環(huán),調(diào)用Linux系統(tǒng)函數(shù)select()阻塞本線程,偵聽本線程與呼叫處理模塊子線程之間的管道,若管道中有數(shù)據(jù),則調(diào)用系統(tǒng)函數(shù)read()讀取數(shù)據(jù),判斷數(shù)據(jù)是否為啟動標(biāo)識,若是,則進(jìn)入子循環(huán)進(jìn)行相應(yīng)的處理;若為其它數(shù)據(jù),則重新回到新一輪的循環(huán)。進(jìn)入子循環(huán)進(jìn)行相應(yīng)的處理的同時,將select()設(shè)為非阻塞模式,調(diào)用select()函數(shù)偵聽本線程與呼叫處理模塊子線程之間的管道,若管道中有數(shù)據(jù),則調(diào)用系統(tǒng)函數(shù)read()讀取數(shù)據(jù),判斷數(shù)據(jù)是否為停止標(biāo)識,若為停止標(biāo)識,則跳出子循環(huán)重新回到主循環(huán),線程重新回到阻塞狀態(tài);若為其它數(shù)據(jù),則不做任何處理,重新回到子循環(huán)。
由于各子線程共享數(shù)據(jù)緩沖區(qū)隊列,為了正確讀寫數(shù)據(jù),在設(shè)計數(shù)據(jù)緩沖區(qū)隊列結(jié)構(gòu)和讀寫操作函數(shù)時,使用了Linux下線程間的同步和互斥機(jī)制,保證了對內(nèi)存資源的安全共享。為了設(shè)計出通用的數(shù)據(jù)緩沖區(qū)隊列結(jié)構(gòu)和讀寫操作函數(shù),不妨將向緩沖區(qū)寫數(shù)據(jù)的子線程定義為生產(chǎn)者線程,將從緩沖區(qū)讀取數(shù)據(jù)的子線程定義為消費者線程。為了保證對數(shù)據(jù)緩沖區(qū)隊列進(jìn)行安全的讀寫操作,生產(chǎn)者線程和消費者線程必須滿足兩個條件:
(1)生產(chǎn)者線程寫入緩沖區(qū)的數(shù)目不能超過緩沖區(qū)容量;
(2)消費者線程讀取的數(shù)目不能超過生產(chǎn)者線程寫入的數(shù)目。
為了實現(xiàn)這兩個條件,在程序?qū)崿F(xiàn)中使用了寫指針和讀指針來判斷緩沖區(qū)是空還是滿。在初始化時讀指針和寫指針為0;如果讀指針等于寫指針,則緩沖區(qū)是空的;如果(寫指針+1)%N等于讀指針,則緩沖區(qū)是滿的,%表示取余數(shù),N表示緩沖區(qū)隊列的長度。
3 結(jié)語
本文提出了基于嵌入式Linux和MiniGUI的SIP電話終端的實現(xiàn)方案,并給出了各線程模塊的實現(xiàn)方法,與傳統(tǒng)的臺式IP網(wǎng)絡(luò)電話解決方案相比,本方案具有如下突出的特點與創(chuàng)新點:a.體積小、功耗低,由于系統(tǒng)所依賴的硬件平臺是嵌入式系統(tǒng)平臺,而嵌入式硬件平臺本身具有體積小、功耗低特點。b.功能可擴(kuò)展。由于嵌入式系統(tǒng)軟硬件可裁剪,可以方便開發(fā)人員進(jìn)行功能擴(kuò)展。c.圖形界面漂亮。由于系統(tǒng)采用嵌入式圖形界面MiniGUI,可以開發(fā)出漂亮的圖形界面。d.采用多線程機(jī)制和緩沖區(qū)隊列對語音的采集與播放、語音的編碼與解碼和語音的實時傳輸進(jìn)行并行處理,保證了語音通話的連續(xù)性。
對系統(tǒng)進(jìn)行測試的結(jié)果表明,本設(shè)計能夠?qū)艚羞M(jìn)行穩(wěn)鍵的控制,能夠保證語音通話的連續(xù)性,對從事相關(guān)產(chǎn)品的開發(fā)具有一定的參考價值。本文引用地址:http://www.ex-cimer.com/article/151381.htm
評論