文章转载自:罗索实验室 [http://www.rosoo.net/a/201106/14613.html]
OPAL是Open Phone Abstraction Library的字母缩写,仍然是Equivalence公司开发的开源VoIP协议栈,从2001年便开始开发了,直到现在还没有正式发布过一个版本,只能从CVS:http://cvs.sourceforge.net/viewcvs.py/openh323/opal下载最新的版本,而且现在的2.05bate版有着非常多的问题,但这些丝毫不能阻止我对OPAL的看好。OPAL仍然采用PWLIB作为开发库,最新的PWLIB实际已经只剩PTLIB,Equivalence公司已经停止了对PWLIB的GUI开发了,GUI全面转向更为强大的wxWindows。OPAL作为Openh323的下一代协议栈最大的特点是加入了对SIP协议的支持,OPAL保留了Openh323大部分代码,并加入了对EndPoint,Connection等抽象描述的基类,使得添加新的协议和设备变得更加简单和方便。
OPAL中最重要的类是新加的:OpalManager,它跟Openh323中的H323EndPoint地位一样,它定义了OPAL系统中最基本的操作,所以每个应用程序必须从OpalManager派生一个类,然后重载虚函数实现自己的操作。
OpalManager::SetUpCall(partyA,partyB,token); 呼叫,partyA是会话发起人,partyB是会话接受人
OpalManager::OnIncomingConnection(connection); 接听,返回TRUE则接听,返回FLASH则拒绝
OpalManager::AdjustMediaFormats(connection,mediaFormats); 调整媒体格式
OpalManager::ClearCall(token,reason,sync) 清除呼叫,sync如果为NULL则不等待快速返回
OpalEndPoint是描述端点的基类,H323,SIP,PSTN,IVR,LID,声卡的端点都是继承于这个类,在这里值得注意的是连声卡设备也看做一种端点。
OpalPCSSEndPoint则是EndPoing中比较重要的一个类,毕竟打电话还是声卡用的多,OpalPCSSEndPoint的构造函数必须引用派生的OpalManager类的实例。
下面具体例子SimpleOPAL来讲解:
opal = new MyManager; //新建一个MyManager的实例
if (opal->Initialise(args)) //调用MyManager::Initialise()函数初始化
opal->Main(args); //等待用户输入命令
cout << "Exiting " << GetName() << endl;
delete opal; //擦pp
部分Initialise()函数代码:
SetVideoInputDevice(video); //设置输入视频设备
...
SetVideoOutputDevice(video); //输出视频设备
...
SetAudioJitterDelay(minJitter, maxJitter); //声音抖动缓存的最小和最大值
...
SetMediaFormatMask(args.GetOptionString('D').Lines()); //屏蔽的媒体格式
SetMediaFormatOrder(args.GetOptionString('P').Lines()); //媒体格式的使用顺序
...
SetTranslationAddress(args.GetOptionString("translate")); //设置外部IP地址
...
Set***Ports(rgs.GetOptionString("tcp-base").AsUnsigned(),
args.GetOptionString("tcp-max").AsUnsigned()); //设置相关的端口
SetSTUNServer(args.GetOptionString("stun")); //使用STUN服务器
...
pcssEP = new MyPCSSEndPoint(*this); //新建一个MyPCSSEndPoint实例
pcssEP->SetSoundDevice(args, "sound", PSoundChannel::Recorder)//设置相关的播放声音设备
...
h323EP = new H323EndPoint(*this); //h323EP的初始化
h323EP->DisableFastStart(args.HasOption('f')); //关闭快速拨号模式
h323EP->DisableH245Tunneling(args.HasOption('T')); //关闭H245通道模式
h323EP->AddAliasName(aliases[i]); //添加昵称
h323EP->SetInitialBandwidth(initialBandwidth); //限制带宽
h323EP->StartListeners(listeners) //现在才开始监听,默认是TCP:1720
h323EP->SetGatekeeperPassword(args.GetOptionString('p')); //注册网守所需的密码
h323EP->UseGatekeeper(gkHost, gkIdentifer, gkInterface) //注册网守
...
sipEP = new SIPEndPoint(*this); //SIP也来了
sipEP->SetUserAgent(args.GetOptionString("sip-user-agent")); //设置用户代理,UserAgent就相当于最初openh323的endpoint
sipEP->SetProxy(args.GetOptionString("sip-proxy")); //设置SIP代理,此处proxy相当于h323里的Gatekeeper
sipEP->SetMIMEForm(args.HasOption("use-long-mime")); //MIME头的格式
sipEP->StartListeners(listeners) //监听开始,默认是TCP:5060和UDP:5060
Main()函数部分代码:
if (pcssEP != NULL && !pcssEP->incomingConnectionToken) { //小心,有电话来了
if (cmd == "n")
pcssEP->ClearCall(pcssEP->incomingConnectionToken, //打死我都不接
OpalConnection::EndedByRefusal);
else if (cmd == "y")
pcssEP->AcceptIncomingConnection(pcssEP-> //喂,小莉吗?
incomingConnectionToken);
}
...
OpalCall * call = FindCallWithLock(currentCallToken); //让我看看是那一个通话
if (call != NULL) {
cout << "Clearing call " << *call << endl;
call->Clear(); //废话那么多,打IP电话也要钱的,挂拉
call->Unlock();
...
if (potsEP != NULL) //是打电脑还是打座机呢?
SetUpCall("pots:*", params[1], currentCallToken); //打座机吧(PSTN)
else
SetUpCall("pc:*", params[1], currentCallToken); //打电脑不要钱,嘿嘿
OPAL中的例子MFC根本无法使用,Openphone的GUI使用了wxWindows,所以必须安装了wxWindows才能正常编译,opalgw这玩意我试过好多次根本无法让别人注册,看来OPAL要走的路还很长,不过Equivalence公司开发已经渐渐的从Openh323转向了OPAL,相信不久一个稳定的OPAL将会被发布.