0 引言
隨著VoIP的迅猛發(fā)展,越來越多的個(gè)人用戶正在使用軟件電話、IP電話通過VoIP系統(tǒng)撥打國(guó)內(nèi)和國(guó)際長(zhǎng)途,IP電話的需求量越來越大,同時(shí),人們對(duì)IP電話的要求也越來越高,例如要求IP電話體積小、方便攜帶、功耗低、待機(jī)時(shí)間長(zhǎng)、漂亮的人機(jī)交互界面,功能可擴(kuò)展等。解決這些需求的可行方案就是用嵌入式系統(tǒng),具體而言就是采用一款32位嵌入式處理芯片(如ARM、Power PC),將Linux操作系統(tǒng)和MiniGUI圖形庫經(jīng)過裁減移植到這些嵌入式處理芯片所構(gòu)建的硬件平臺(tái)上。由于Linux具有強(qiáng)大的網(wǎng)絡(luò)功能,而MiniGUI是一款優(yōu)秀的針對(duì)嵌入式Linux的輕量級(jí)圖形用戶界面庫,在它們的基礎(chǔ)上做應(yīng)用開發(fā),能夠保證IP電話的穩(wěn)定性和功能擴(kuò)展,也能開發(fā)出漂亮的人機(jī)交互界面。
目前用來實(shí)現(xiàn)VoIP系統(tǒng)的協(xié)議有三種:SIP、MGCP和H.323,其中SIP協(xié)議是應(yīng)用得最廣泛的協(xié)議,所謂SIP電話就是支持SIP協(xié)議的IP電話。
1 SIP電話實(shí)現(xiàn)方案
根據(jù)IP電話的功能需求,SIP電話應(yīng)當(dāng)實(shí)現(xiàn)人機(jī)界面的交互、呼叫處理、語音的采集和播放、語音的編碼和解碼、語音的實(shí)時(shí)傳輸。本設(shè)計(jì)人機(jī)界面的交互使用嵌入式系統(tǒng)硬件平臺(tái)上的LCD和功能按鍵,采用MiniGUI圖形庫和Linux按鍵驅(qū)動(dòng);呼叫處理模塊使用硬件平臺(tái)上的網(wǎng)絡(luò)接口,采用eXoSIP協(xié)議棧;語音的采集與播放使用硬件平臺(tái)上的音頻接口,采用Linux音頻設(shè)備驅(qū)動(dòng);語音的編碼和解碼直接采用開源G.7-29A源代碼;語音的實(shí)時(shí)傳輸使用RTP協(xié)議,采用開源的JRTPLIB庫。
SIP電話軟件結(jié)構(gòu)圖如圖1所示。SIP電話由八個(gè)模塊組成。每一模塊對(duì)應(yīng)一個(gè)線程。其中,主線程(線程1)的任務(wù)是:a.加載配置文件到內(nèi)存中;b.初始化音頻設(shè)備和功能按鍵設(shè)備;c.創(chuàng)建RTP會(huì)話實(shí)例和初始化eXoSIP協(xié)議棧;d.初始化四個(gè)數(shù)據(jù)區(qū)緩沖結(jié)構(gòu);e.創(chuàng)建、管理、撤消子線程;f. 顯示SIP配置文件的配置信息和狀態(tài)信息,處理來自呼叫處理模塊子線程的消息。呼叫處理模塊子線程(線程2)的任務(wù)是:通過調(diào)用eXoSIP協(xié)議棧的API函數(shù),實(shí)現(xiàn)SIP電話的呼叫過程控制。語音采集模塊子線程(線程3)的任務(wù)是:實(shí)現(xiàn)語音的采集并將采集到的語音數(shù)據(jù)存儲(chǔ)到全局?jǐn)?shù)據(jù)緩沖區(qū)隊(duì)列1中。語音編碼模塊子線程(線程4)的任務(wù)是:從全局?jǐn)?shù)據(jù)緩沖區(qū)隊(duì)列1中讀取PCM碼流并對(duì)其進(jìn)行編碼,將轉(zhuǎn)化過后的G.729碼流存儲(chǔ)到全局?jǐn)?shù)據(jù)緩沖區(qū)隊(duì)列2中。數(shù)據(jù)發(fā)送模塊子線程(線程5)的任務(wù)是:從全局?jǐn)?shù)據(jù)緩沖區(qū)隊(duì)列2中提取G.729碼流,打包成RTP數(shù)據(jù)包發(fā)送到出去。數(shù)據(jù)接收模塊子線程(線程6)的任務(wù)是:檢測(cè)接收端口上的RTP語音包,提取G.729碼流存儲(chǔ)到全局?jǐn)?shù)據(jù)緩沖區(qū)隊(duì)列3中。語音解碼模塊子線程(線程7)的任務(wù)是:從全局?jǐn)?shù)據(jù)緩沖區(qū)隊(duì)列3中讀取G.729碼流對(duì)其進(jìn)行解碼,將轉(zhuǎn)化過后的PCM碼流存儲(chǔ)到全局?jǐn)?shù)據(jù)緩沖區(qū)隊(duì)列4中。語音播放模塊子線程(線程8)的任務(wù)是:從全局?jǐn)?shù)據(jù)緩沖區(qū)隊(duì)列4中讀取PCM碼流,通過D/A轉(zhuǎn)換成模擬語音信號(hào)。
2 各線程模塊的實(shí)現(xiàn)
主線程模塊主要完成系統(tǒng)各個(gè)功能模塊的初始化工作,也是程序的入口點(diǎn),MiniGUI程序的入口點(diǎn)為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端口號(hào)、用戶名、本機(jī)電話號(hào)碼、密碼、RTP端口號(hào)、被叫電話號(hào)碼和注冊(cè)間隔時(shí)間。初始化音頻設(shè)備擬完成打開音頻設(shè)備文件,設(shè)置音頻設(shè)備的采樣頻率,量化位數(shù)和聲道數(shù)目。打開音頻設(shè)備文件可
通過調(diào)用Linux系統(tǒng)函數(shù)audio_fd=open(“/dev/dsp”,O_RDWR)來實(shí)現(xiàn),調(diào)用成功后將返回音頻設(shè)備的文件描述符。設(shè)置音頻設(shè)備的采樣頻率,量化位數(shù)和聲道數(shù)目可通過調(diào)用ioctl(fd,….)函數(shù)來實(shí)現(xiàn)。功能按鍵設(shè)備的初始化很簡(jiǎn)單,直接調(diào)用buttons_fd=open“/dev/b-uttons”,O)函數(shù)打開按鍵設(shè)備文件即可。創(chuàng)建RTP會(huì)話實(shí)例,可通過調(diào)用JRTPLIB庫的RTPSession類來完成,然后調(diào)用RTPSession類的Create()方法來對(duì)其進(jìn)行初始化,創(chuàng)建完成后,需設(shè)置RTP會(huì)話實(shí)例的傳輸參數(shù)和會(huì)話參數(shù)。eXoSIP協(xié)議棧的初始化直接調(diào)用eXoSIP協(xié)議棧所提供的初始化函數(shù)。七個(gè)子線程的創(chuàng)建可通過調(diào)用pthread_create函數(shù)來完成。SIP配置信息的顯示擬完成配置文件中的信息在MiniGUI主窗口上的顯示,主要顯示本機(jī)的IP地址和端口號(hào)、SIP服務(wù)器的IP地址、本機(jī)號(hào)碼、本機(jī)用戶名。SIP狀態(tài)信息的顯示擬完成對(duì)整個(gè)SIP事務(wù)遷移狀態(tài)的顯示。例如,如果收到"180Ringing"消息,則在MiniGUI主窗口上顯示“對(duì)方正在響鈴”,如果收到定時(shí)器的超時(shí)消息,則在MiniGUI主窗口上顯示“無人接聽,請(qǐng)稍后再撥”。SIP狀態(tài)信息的顯示是一個(gè)消息驅(qū)動(dòng)的動(dòng)態(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)信息的顯示必須為每個(gè)SIP事務(wù)消息定義相對(duì)應(yīng)的MiniGUI消息,以"180 Ringing"消息和定時(shí)器超時(shí)消息為例,自定義消息如下:
#define MSG_180Ringing(MSG_USER+10)
#define MSG_TimerC(MSG_USER+11)
當(dāng)呼叫處理模塊子線程收到IP網(wǎng)絡(luò)上的“180 Ringing”消息和Linux內(nèi)核的定時(shí)器超時(shí)消息后,則通過調(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,“對(duì)方正在響鈴”)函數(shù)在窗口上顯示“對(duì)方正在響鈴”字樣。
呼叫處理模塊子線程可直接調(diào)用eXoSIP協(xié)議棧所提供的API函數(shù)集,eXoSIP是在oSIP2的基礎(chǔ)上對(duì)SIP消息的API作了更上層的封裝,能夠很容易實(shí)現(xiàn)SIP電話的呼叫過程控制。呼叫處理模塊子線程實(shí)現(xiàn)的難點(diǎn)是當(dāng)呼叫連接成功后,如何啟動(dòng)語音采集、語音編碼、數(shù)據(jù)發(fā)送、數(shù)據(jù)接收、語音解碼和語音播放6個(gè)子線程。本設(shè)計(jì)采用Linux線程間通信-管道機(jī)制向其它6個(gè)子線程發(fā)送啟動(dòng)標(biāo)識(shí),6個(gè)子線程接收到啟動(dòng)標(biāo)識(shí)后,喚醒各自的線程,進(jìn)行相應(yīng)的語音處理和語音的傳輸。同樣,當(dāng)呼叫連接釋放時(shí),呼叫處理模塊子線程向6個(gè)子線程發(fā)送停止標(biāo)識(shí),6個(gè)子線程接收到停止標(biāo)識(shí)后,停止語音處理和語音的傳輸,阻塞各自的線程。
語音采集模塊、語音編碼模塊、數(shù)據(jù)發(fā)送模塊、數(shù)據(jù)接收模塊、語音解碼模塊和語音播放模塊6個(gè)子線程的過程控制是一樣的,首先進(jìn)入主循環(huán),調(diào)用Linux系統(tǒng)函數(shù)select()阻塞本線程,偵聽本線程與呼叫處理模塊子線程之間的管道,若管道中有數(shù)據(jù),則調(diào)用系統(tǒng)函數(shù)read()讀取數(shù)據(jù),判斷數(shù)據(jù)是否為啟動(dòng)標(biāo)識(shí),若是,則進(jìn)入子循環(huán)進(jìn)行相應(yīng)的處理;若為其它數(shù)據(jù),則重新回到新一輪的循環(huán)。進(jìn)入子循環(huán)進(jìn)行相應(yīng)的處理的同時(shí),將select()設(shè)為非阻塞模式,調(diào)用select()函數(shù)偵聽本線程與呼叫處理模塊子線程之間的管道,若管道中有數(shù)據(jù),則調(diào)用系統(tǒng)函數(shù)read()讀取數(shù)據(jù),判斷數(shù)據(jù)是否為停止標(biāo)識(shí),若為停止標(biāo)識(shí),則跳出子循環(huán)重新回到主循環(huán),線程重新回到阻塞狀態(tài);若為其它數(shù)據(jù),則不做任何處理,重新回到子循環(huán)。
由于各子線程共享數(shù)據(jù)緩沖區(qū)隊(duì)列,為了正確讀寫數(shù)據(jù),在設(shè)計(jì)數(shù)據(jù)緩沖區(qū)隊(duì)列結(jié)構(gòu)和讀寫操作函數(shù)時(shí),使用了Linux下線程間的同步和互斥機(jī)制,保證了對(duì)內(nèi)存資源的安全共享。為了設(shè)計(jì)出通用的數(shù)據(jù)緩沖區(qū)隊(duì)列結(jié)構(gòu)和讀寫操作函數(shù),不妨將向緩沖區(qū)寫數(shù)據(jù)的子線程定義為生產(chǎn)者線程,將從緩沖區(qū)讀取數(shù)據(jù)的子線程定義為消費(fèi)者線程。為了保證對(duì)數(shù)據(jù)緩沖區(qū)隊(duì)列進(jìn)行安全的讀寫操作,生產(chǎn)者線程和消費(fèi)者線程必須滿足兩個(gè)條件:
(1)生產(chǎn)者線程寫入緩沖區(qū)的數(shù)目不能超過緩沖區(qū)容量;
(2)消費(fèi)者線程讀取的數(shù)目不能超過生產(chǎn)者線程寫入的數(shù)目。
為了實(shí)現(xiàn)這兩個(gè)條件,在程序?qū)崿F(xiàn)中使用了寫指針和讀指針來判斷緩沖區(qū)是空還是滿。在初始化時(shí)讀指針和寫指針為0;如果讀指針等于寫指針,則緩沖區(qū)是空的;如果(寫指針+1)%N等于讀指針,則緩沖區(qū)是滿的,%表示取余數(shù),N表示緩沖區(qū)隊(duì)列的長(zhǎng)度。
3 結(jié)語
本文提出了基于嵌入式Linux和MiniGUI的SIP電話終端的實(shí)現(xiàn)方案,并給出了各線程模塊的實(shí)現(xiàn)方法,與傳統(tǒng)的臺(tái)式IP網(wǎng)絡(luò)電話解決方案相比,本方案具有如下突出的特點(diǎn)與創(chuàng)新點(diǎn):a.體積小、功耗低,由于系統(tǒng)所依賴的硬件平臺(tái)是嵌入式系統(tǒng)平臺(tái),而嵌入式硬件平臺(tái)本身具有體積小、功耗低特點(diǎn)。b.功能可擴(kuò)展。由于嵌入式系統(tǒng)軟硬件可裁剪,可以方便開發(fā)人員進(jìn)行功能擴(kuò)展。c.圖形界面漂亮。由于系統(tǒng)采用嵌入式圖形界面MiniGUI,可以開發(fā)出漂亮的圖形界面。d.采用多線程機(jī)制和緩沖區(qū)隊(duì)列對(duì)語音的采集與播放、語音的編碼與解碼和語音的實(shí)時(shí)傳輸進(jìn)行并行處理,保證了語音通話的連續(xù)性。
對(duì)系統(tǒng)進(jìn)行測(cè)試的結(jié)果表明,本設(shè)計(jì)能夠?qū)艚羞M(jìn)行穩(wěn)鍵的控制,能夠保證語音通話的連續(xù)性,對(duì)從事相關(guān)產(chǎn)品的開發(fā)具有一定的參考價(jià)值。