主頁(http://www.130131.com):介紹一個(gè)開源的SIP(VOIP)協(xié)議庫PJSIP
如果你對(duì)SIP/VoIP技術(shù)感興趣,哪希望你不要錯(cuò)過:),如果你對(duì)寫出堪稱優(yōu)美的Code感興趣
,那么你也不可錯(cuò)過:)
這期間我想分析一下一個(gè)實(shí)際的協(xié)議棧的設(shè)計(jì)到實(shí)現(xiàn)的相關(guān)技術(shù),算是自己的一個(gè)學(xué)習(xí)經(jīng)
歷記錄.
最初選擇這個(gè)庫做分析的原因很簡(jiǎn)單,文檔齊全:),其它良好的特征則是慢慢發(fā)現(xiàn)的:)
www.pjsip.org
1. PJSIP簡(jiǎn)介
PJSIP的實(shí)現(xiàn)是為了能在嵌入式設(shè)備上高效實(shí)現(xiàn)SIP/VOIP.其主要特征包括:
1).極具移植性.(Extremely portable)
當(dāng)前可支持平臺(tái)包括:
* Win32/x86 (Win95/98/ME, NT/2000/XP/2003, mingw).
* arm, WinCE and Windows Mobile.
* Linux/x86, (user mode and as kernel module(!)).
* Linux/alpha
* Solaris/ultra.
* MacOS X/powerpc
* RTEMS (x86 and powerpc).
正移植到:
* Symbian OS
2).非常小的足印.(Very small footprint)
官方宣稱編譯后的庫<150Kb,我在PC上編譯后加上strip后大概173Kb,這對(duì)于嵌入
式設(shè)備,是個(gè)好消息:)
3).高性能.(High performance)
這點(diǎn)我們后面可以看看是否如作者宣稱的:)
4).支持眾多的特征.(Many features)
這點(diǎn)可以從http://www.pjsip.org/sip_media_features.htm#sip_features看出.
5).充足的SIP文檔.(Extensive SIP documentation)
這是我最初選擇該庫的原因,當(dāng)然不是最終的原因,最終的原因是它的code:)
2. PJSIP的組成.
其實(shí)說是PJSIP不是特別貼切,這個(gè)庫實(shí)際上是幾個(gè)部分組成的.
1).PJSIP - Open Source SIP Stack[開源的SIP協(xié)議棧]
2).PJMEDIA - Open Source Media Stack[開源的媒體棧]
3).PJNATH - Open Source NAT Traversal Helper Library[開源的NAT-T輔助庫]
4).PJLIB-UTIL - Auxiliary Library[輔助工具庫]
5).PJLIB - Ultra Portable Base Framework Library[基礎(chǔ)框架庫]
而在最上層庫的目錄分為:(可以使用tree -d -L 1 查看)
$TOP/build [包含Makefile]
$TOP/build.symbian [針對(duì)symbian的Makefile]
$TOP/pjlib [參考上面]
$TOP/pjlib-util [參考上面]
$TOP/pjnath [參考上面]
$TOP/pjmedia [參考上面]
$TOP/pjsip [參考上面]
$TOP/pjsip-apps
$TOP/third_party
而在每個(gè)子目錄,可以看到分為:
bin [編譯后產(chǎn)生的二進(jìn)制文件]
build [Makefile]
build/output
build/wince-evc4
docs [doxygen的文檔,用doxygen docs/doxygen.cfg產(chǎn)生]
include [頭文件]
lib [編譯后產(chǎn)生的庫]
src [源代碼]
3. PJLIB簡(jiǎn)介
要理解好PJSIP,就不得不先說說PJLIB,PJLIB算的上是這個(gè)庫中最基礎(chǔ)的庫,正是這個(gè)
庫的優(yōu)美實(shí)現(xiàn),才讓PJSIP變得如此優(yōu)越。
PJLIB提供了一系列特征,這是我們下面分析的重點(diǎn),涉及到:
1).非動(dòng)態(tài)內(nèi)存分配[No Dynamic Memory Allocations]
實(shí)現(xiàn)了內(nèi)存池,獲取內(nèi)存是從與分配的內(nèi)存池中獲取,高性能程序多會(huì)自己構(gòu)造內(nèi)存池
,后面我們會(huì)解釋該內(nèi)存池的使用以及基本的原理。根據(jù)作者的比較,是常規(guī)的 malloc(
)/free()函數(shù)的30倍。
2).OS抽象[Operating System Abstraction]
實(shí)現(xiàn)OS抽象的根本原因在與可移植性,毋庸置疑:).
涉及到:
a).線程[Threads.]
b).線程本地存儲(chǔ)[Thread Local Storage.]
c).互斥[Mutexes.]
d).信號(hào)燈[Semaphores.]
e).原子變量[Atomic Variables.]
f).臨屆區(qū)[Critical sections.]
g).鎖對(duì)象[Lock Objects.]
h).事件對(duì)象[Event Object.]
i).時(shí)間管理[Time Data Type and Manipulation.]
j).高解析的時(shí)間戳[High Resolution Timestamp.]
等等,這些我們后面分析代碼時(shí)一一看來:)
3).低層的網(wǎng)絡(luò)相關(guān)IO[Low-Level Network I/O]
這涉及到:
a).Socket抽象[Socket Abstraction.]
b).網(wǎng)絡(luò)地址解析[Network Address Resolution.]
c).實(shí)現(xiàn)針對(duì)Socket的select API[Socket select() API.]
4).時(shí)間管理[Timer Management]
這主要涉及到兩個(gè)部分,一個(gè)時(shí)定時(shí)器的管理,還有就是時(shí)間解析的精度(舉例說來,就
是能精確到哪個(gè)時(shí)間等級(jí),比如 POSIX sleep(),就只能以秒為單位,而使用select()則可
以實(shí)現(xiàn)毫秒級(jí)別的計(jì)時(shí))
5).各種數(shù)據(jù)結(jié)構(gòu)[Various Data Structures]
主要有:
a).針對(duì)字符串的操作[String Operations]
b).數(shù)組輔助[Array helper]
c).Hash表[Hash Tabl]
d).鏈表[Linked List]
e).紅黑平衡樹[Red/Black Balanced Tree]
6).異常處理[Exception Construct]
使用的是TRY/CATCH,知道C++/JAVA之類面向?qū)ο笳Z言的人看過會(huì)宛而一笑:)
7).LOG機(jī)制[Logging Facility]
很顯然,一個(gè)良好的程序,好的LOG機(jī)制不可少。這能很方便的讓你去調(diào)試程序,對(duì)此我
是深有體會(huì),任何時(shí)候,不要忘記“好的程序,是架構(gòu)出來的;而能跑的程序,是調(diào)試出
來的:)”
8).隨機(jī)數(shù)以及GUID的產(chǎn)生[Random and GUID Generation]
GUID指的是"globally unique identifier",只是一個(gè)標(biāo)識(shí)而已,比如說你的省份證,
算的上是一個(gè)GUID,當(dāng)然,準(zhǔn)確說來是“china unique identifier”:).
看了這么多的特征列舉,是不是很完備,的確。
總算是初步列舉完了PJLIB的基本特征了,后面我們來說說它的使用與實(shí)現(xiàn):
4. PJLIB的使用
有了上述介紹,是不是很想知道這個(gè)庫的使用,沒關(guān)系,我們慢慢說來:)
首先是頭文件和編譯出來的庫的位置,這就不必多說了,除非你沒有使用過手動(dòng)編譯的庫
,如果不太了解步驟,google一下,啊:)
1).為了使用這個(gè)庫,需要使用:
#include
當(dāng)然,也可以選擇:
#include
#include
這種分離的方式,不過,簡(jiǎn)介其間,還是使用第一種吧:),畢竟,你不需要確認(rèn)到你所
需的函數(shù)或者數(shù)據(jù)結(jié)構(gòu)具體到哪個(gè)具體的頭文件:)
2).確保在使用PJLIB之前調(diào)用 pj_init()來完成PJLIB庫使用前說必須的一些初始化.
這是一個(gè)必不可少的步驟.
~~~~~~~~~~~~~~~~~~~~~~~
3).使用PJLIB的一些建議
作者對(duì)使用PJLIB的程序提出了一些建議,包括如下 :
a).不要使用ANSI C[Do NOT Use ANSI C]
觀點(diǎn)很明確,ANSI C并不會(huì)讓程序具有最大的移植性,應(yīng)該使用PJSIP庫所提供的響
應(yīng)機(jī)制來實(shí)現(xiàn)你所需要的功能.
b).使用pj_str_t取代C風(fēng)格的字符串[Use pj_str_t instead of C Strings]
原因之一是移植性,之二則是PJLIB內(nèi)置的pj_str_t相關(guān)操作會(huì)更快(性能).
c).從內(nèi)存池分配內(nèi)存[Use Pool for Memory Allocations]
這很明顯,如果你知道為什么會(huì)使用內(nèi)存池的話(提示一下,性能以及易用性:))
d).使用PJLIB的LOG機(jī)制做文字顯示[Use Logging for Text Display]
很明顯:)
還有些關(guān)于移植的一些問題,不在我們的討論范圍,如果你需要移植到其它平臺(tái)或者
環(huán)境,請(qǐng)參考http://www.pjsip.org/pjlib/docs/html/porting_pjlib_pg.htm
5. PJLIB的使用以及原理
終于開始提及實(shí)現(xiàn)原理以及具體的編碼了:),前面的列舉還真是個(gè)瑣碎的事情,還是奔主題
來:).
5.1快速內(nèi)存池[Fast Memory Pool]
前面說過,使用內(nèi)存池的原因在于性能的考慮,原因是C風(fēng)格的malloc()以及C++風(fēng)格的new
操作在高性能或?qū)崟r(shí)條件下表現(xiàn)并不太好,原因在于性能的瓶頸在于內(nèi)存碎片問題.
下面列舉其優(yōu)點(diǎn)與需要主要的問題:
優(yōu)點(diǎn):
a).不像其它內(nèi)存池,允許分配不同尺寸的chunks.
b).快速.
內(nèi)存chunks擁有O(1)的復(fù)雜度,并且操作僅僅是指針的算術(shù)運(yùn)算,其間不需要使用鎖住任
何互斥量.
c).有效使用內(nèi)存.
除了可能因?yàn)閮?nèi)存對(duì)齊的原因會(huì)浪費(fèi)很少的內(nèi)存外,內(nèi)存的使用效率非常高.
d).可預(yù)防內(nèi)存泄漏.
在C/C++程序中如果出現(xiàn)內(nèi)存泄漏問題,其查找過程哪個(gè)艱辛,不足為外人道也:(
[曾經(jīng)有次用別人的Code,出現(xiàn)了內(nèi)存泄漏,在開發(fā)板上查找N天,又沒工具可在開發(fā)板上使
用,哪個(gè)痛苦,想自殺:(]
原因很簡(jiǎn)單,你的內(nèi)存都是從內(nèi)存池中獲取的,就算你沒有釋放你獲取的內(nèi)存,只要你記得
把內(nèi)存池destroy,那么內(nèi)存還是會(huì)還給系統(tǒng).
還有設(shè)計(jì)帶來的一些其它益處,比如可用性和靈活性:
e).內(nèi)存泄漏更容易被跟蹤.
這是因?yàn)槟愕膬?nèi)存是在指定的內(nèi)存池中分配的,只要能很快定位到內(nèi)存池,內(nèi)存泄漏的偵
測(cè)就方便多了.
f).設(shè)計(jì)上從內(nèi)存池中獲取內(nèi)存這一操作是非線程安全的.
原因是設(shè)計(jì)者認(rèn)為內(nèi)存池被上層對(duì)象所擁有,線程安全應(yīng)該由上層對(duì)象去保證,這樣的話
,沒有鎖的問題會(huì)讓內(nèi)存分配變得非常的快.
g).內(nèi)存池的行為像C++中的new的行為,當(dāng)內(nèi)存池獲取內(nèi)存chunks會(huì)拋出PJ_NO_MEMORY_EX
CEPTION異常,當(dāng)然,因?yàn)橹С之惓L幚?也可以使用其它方式讓上層程序靈活的定義異常的
處理.
[這是異常處理的基本出發(fā)點(diǎn),但是這有大量的爭(zhēng)論,原因是這改變了程序的正常流程,誰能
去保證這種流程是用戶所需要的呢,因此C++中的異常處理飽受爭(zhēng)議,請(qǐng)酌情使用]
h). 可以在后端使用任何的內(nèi)存分配器.默認(rèn)情況下是使用malloc/free管理內(nèi)存池的塊,
但是應(yīng)用程序也可以指定自己的策略(strategy),例如從一個(gè)全局存儲(chǔ)空間分配內(nèi)存.
恩,要知道,任何事務(wù)都是兩面的(頗為佩服創(chuàng)造出“雙贏”這個(gè)詞的語言天才, 不過,文
字游戲?qū)τ诩夹g(shù)人員不能說是件好事情:(),好了,使用時(shí),不要認(rèn)為這個(gè)內(nèi)存池是哪種"per
fect"的技術(shù),要記得"任何設(shè)計(jì),都是在各種限制條件中的一個(gè)折中,對(duì)于'戴著鐐銬的舞蹈
',除了'舞蹈',也不要忘記'鐐銬'哦",不要忘了告誡:):
告誡[Caveats]:
a).使用合適的大小來初始化內(nèi)存池.
使用內(nèi)存池時(shí),需要指定一個(gè)初始內(nèi)存池大小, 這個(gè)值是內(nèi)存池的初始值,如果你想要高
性能,要謹(jǐn)慎選擇這個(gè)值哦,太大的化會(huì)浪費(fèi)內(nèi)存,過小又會(huì)讓內(nèi)存池自身頻繁的去增加內(nèi)存
,顯然這兩種情況都不可取.
b). 注意,內(nèi)存池只能增加,而不能被縮小(shrink),因?yàn)閮?nèi)存池沒有函數(shù)把內(nèi)存chunks釋
放還給系統(tǒng),這就要去內(nèi)存池的構(gòu)造者和使用者明確使用內(nèi)存.
恩,基本的原理都差不多了,后面我們來看看如何使用這個(gè)內(nèi)存池.
5.2內(nèi)存池的使用[Using Memory Pool]
內(nèi)存池的使用相當(dāng)?shù)暮?jiǎn)單,扳個(gè)手指頭就搞定了,如果你看明白了上面的原理和特征:)
a).創(chuàng)建內(nèi)存池工廠[Create Pool Factory]
上面不是提及內(nèi)存池的內(nèi)部分配策略以及異常處理方式么, 其實(shí)這就是指定這個(gè)的:)
當(dāng)然,不需要你每個(gè)內(nèi)存池都自己取指定策略和異常處理方式,PJLIB已經(jīng)有了一個(gè)默認(rèn)的
實(shí)現(xiàn):Caching Pool Factory,這個(gè)內(nèi)存池工廠的初始化使用函數(shù)pj_caching_pool_init()
b).創(chuàng)建內(nèi)存池[Create The Pool]
使用pj_pool_create(),其參數(shù)分別為內(nèi)存工廠(Pool Factory),內(nèi)存池的名字(name),初
始時(shí)的大小以及增長時(shí)的大小.
c).根據(jù)需要分配內(nèi)存[Allocate Memory as Required]
然后,你就可以使用pj_pool_alloc(), pj_pool_calloc(), 或pj_pool_zalloc()從指定
的內(nèi)存池根據(jù)需要去獲取內(nèi)存了:)
d).Destroy內(nèi)存池[Destroy the Pool]
這實(shí)際上是把預(yù)分配的內(nèi)存還給系統(tǒng).
e).Destroy內(nèi)存池工廠[Destroy the Pool Factory]
這沒什么好說的.
很簡(jiǎn)單吧:)
作者在文檔中給出了一個(gè)例子:
如下:
#include
#define THIS_FILE "pool_sample.c"
static void my_perror(const char *title, pj_status_t status)
{
char errmsg[PJ_ERR_MSG_SIZE];
pj_strerror(status, errmsg, sizeof(errmsg));
PJ_LOG(1,(THIS_FILE, "%s: %s [status=%d]", title, errmsg, status));
}
static void pool_demo_1(pj_pool_factory *pfactory)
{
unsigned i;
pj_pool_t *pool;
// Must create pool before we can allocate anything
pool = pj_pool_create(pfactory, // the factory
"pool1", // pool's name
4000, // initial size
4000, // increment size
NULL); // use default callback.
if (pool == NULL) {
my_perror("Error creating pool", PJ_ENOMEM);
return;
}
// Demo: allocate some memory chunks
for (i=0; i<1000; ++i) {
void *p;
p = pj_pool_alloc(pool, (pj_rand()+1) % 512);
// Do something with p
...
// Look! No need to free p!!
}
// Done with silly demo, must free pool to release all memory.
pj_pool_release(pool);
}
int main()
{
pj_caching_pool cp;
pj_status_t status;
// Must init PJLIB before anything else
status = pj_init();
if (status != PJ_SUCCESS) {
my_perror("Error initializing PJLIB", status);
return 1;
}
// Create the pool factory, in this case, a caching pool,
// using default pool policy.
pj_caching_pool_init(&cp, NULL, 1024*1024 );
// Do a demo
pool_demo_1(&cp.factory);
// Done with demos, destroy caching pool before exiting app.
pj_caching_pool_destroy(&cp);
return 0;
}
我就不解釋了:)
(中國集群通信網(wǎng) | 責(zé)任編輯:李俊勇) |