📄 srtpwrapper.cpp
字号:
#include <iostream>#include <qsettings.h>#include <cstdlib>#define SRTPWRAPPER_HX#include "SRTPWrapper.h"#include "../kphone/kstatics.h"using namespace std;SRTPWrapper* SRTPWrapper::instance = 0;int SRTPWrapper::countReferences = 0;const string SRTPWrapper::err_array[] = {#ifdef SRTP "err_status_ok", "err_status_fail", "err_status_bad_param", "err_status_alloc_fail", "err_status_dealloc_fail", "err_status_init_fail", "err_status_terminus", "err_status_auth_fail", "err_status_cipher_fail", "err_status_replay_fail", "err_status_replay_old", "err_status_algo_fail", "err_status_no_such_op", "err_status_no_ctx", "err_status_cant_check", "err_status_key_expired"#endif};bool SRTPWrapper::libInit = false;SRTPWrapper::SRTPWrapper(){#ifdef SRTP cout << "SRTP Wrapper Instance created" << endl; sessionActive = false; outboundStreamActive = false; inboundStreamActive = false; if(!SRTPWrapper::libInit){ if(err_status_t error = srtp_init()){ libError(error); cout << "Init from libsrtp failed! Exit!" << endl; exit(1); } SRTPWrapper::libInit = true; }#endif }SRTPWrapper::~SRTPWrapper(){#ifdef SRTP cout << "SRTP Wrapper Instance deleted" << endl; if(sessionActive) delete session; SRTPWrapper::instance = 0;#endif}SRTPWrapper* SRTPWrapper::getInstance(){#ifdef SRTP if(SRTPWrapper::instance == 0) { SRTPWrapper::instance = new SRTPWrapper(); } countReferences++;#endif return SRTPWrapper::instance;}void SRTPWrapper::dispose(void){ SRTPWrapper::countReferences--;}void SRTPWrapper::destroy(void){ if(SRTPWrapper::countReferences == 0) delete this; }#ifdef SRTP//Case PKEerr_status_t SRTPWrapper::newSession(){ session = new srtp_ctx_t(); err_status_t err = err_status_ok; if(policy){ printPolicyInfo(); err = srtp_create(&session, policy); }else{ err = err_status_no_ctx; cout << "ERROR: Cannot start SRTP session, policy was not set by key management." << endl; } return err;}//Case PSKerr_status_t SRTPWrapper::newSession(ssrc_t ssrc){ cout << "SSRC for new Session: " << ssrc.value << endl; session = new srtp_ctx_t(); policy = new srtp_policy_t; crypto_policy_set_rtp_default(&policy->rtp); crypto_policy_set_rtcp_default(&policy->rtcp); policy->ssrc.value = ssrc.value; policy->ssrc.type = ssrc.type; policy->key = this->readKey(); policy->next = NULL; printPolicyInfo(); err_status_t err = srtp_create(&session, policy); return err;}#endifvoid SRTPWrapper::protect(unsigned char* rtp_h, int* length){#ifdef SRTP rtp_hdr_t* hdr = (rtp_hdr_t *) rtp_h; QSettings settings; if(!sessionActive){ if(settings.readEntry(KStatics::dBase + "SRTP/Mode", "disabled") == "PSK"){ localSSRC = rand(); ssrc_t ssrc; ssrc.value = localSSRC; ssrc.type = ssrc_any_outbound; if (err_status_t err = this->newSession(ssrc)){ libError(err); cout << "SRTP session creation failed (protect)." << endl; }else{ cout << "SRTP session created! (PSK)" << endl; sessionActive = true; } } else if(settings.readEntry(KStatics::dBase + "SRTP/Mode", "disabled") == "PKE"){ if(err_status_t err = this->newSession()){ libError(err); cout << "SRTP session creation failed (protect) (PKE)." << endl; }else{ cout << "SRTP session created!" << endl; sessionActive = true; } }else{ cout << "Key exchange Mode not supported: " << settings.readEntry(KStatics::dBase + "SRTP/Mode", "disabled") << endl; return; } } if(!outboundStreamActive){ session->stream_template->direction = dir_srtp_sender; outboundStreamActive = true; } hdr->ssrc = localSSRC; //KPhone uses the same SSRC for all Streams, this produces replay errors in SRTP if(err_status_t err = srtp_protect(session, rtp_h, length)){ cout << "srtp_protect: "; libError(err); }#endif}void SRTPWrapper::unprotect(unsigned char* rtp_h, int* length){#ifdef SRTP if(!sessionActive){ cout << "Session not active for unprotect. Waiting for outgoing traffic.." << endl; //Mute packet to avoid amplitude for (int i=0; i< *length ; i++){ rtp_h[i] = 0xff; } }else{ if(!inboundStreamActive){ session->stream_template->direction = dir_srtp_receiver; inboundStreamActive = true; } err_status_t err = srtp_unprotect(session, rtp_h, length); if(err){ cout << "srtp_unprotect: "; libError(err); //Mute packet to avoid amplitude for (int i=0; i< *length ; i++){ rtp_h[i] = 0xff; } } }#endif}#ifdef SRTP//octet_t* SRTPWrapper::readKey(){unsigned char* SRTPWrapper::readKey(){ QSettings settings; QString qkey = settings.readEntry(KStatics::dBase + "SRTP/KeyValue", ""); string test = qkey; if(test.length() < 30){ int i = 30 - test.length(); test.append(i, '0'); cout << "WARNING: The chosen key is to short (<30) and will be padded!" << endl; } string* strKey = new string(test); unsigned char* tmp = (unsigned char*) strKey->c_str(); return tmp;}void SRTPWrapper::setPolicy(srtp_policy_t* pol){ policy = pol; localSSRC = policy->ssrc.value; }string SRTPWrapper::getErrorname(err_status_t err){ return SRTPWrapper::err_array[err];}void SRTPWrapper::libError(err_status_t err){ cout << "WARNING: libSRTP reported an Error: Code "<< err << " (" << getErrorname(err) << ")!" << endl;}#endifvoid SRTPWrapper::printPolicyInfo(){#ifdef SRTP QSettings settings; cout << "SRTP Policy (RTP): " << endl; cout << "Key Exchange Mode: " << settings.readEntry(KStatics::dBase + "SRTP/Mode", "disabled"); cout << ", SSRC: " << policy->ssrc.value; cout << ", SSRC type : " << policy->ssrc.type << endl; cout << "cipherKeylengh: " << policy->rtp.cipher_key_len; cout << ", cypherType: " << policy->rtp.cipher_type << endl; cout << "AuthKeyLen: " << policy->rtp.auth_key_len; cout << ", AuthType: " << policy->rtp.auth_type; cout << ", AuthTagLengh: " << policy->rtp.auth_tag_len << endl; cout << "SRTP Master Key: " ; cout.setf(ios::hex, ios::basefield); for (int i=0; i < policy->rtp.cipher_key_len ; i++){ cout << (short) policy->key[i]; } cout << endl; cout.setf(ios::dec, ios::basefield);#endif}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -