⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 mediastream.cxx

📁 MiniSip Client with DomainKeys Authentication, Sip, Audio communications, Echo Cancel
💻 CXX
字号:
/* Copyright (C) 2004-2006 the Minisip Team  This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version.  This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public License for more details.  You should have received a copy of the GNU Lesser General Public License along with this library; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA *//* Copyright (C) 2004, 2005  * * Authors: Erik Eliasson <eliasson@it.kth.se> *          Johan Bilien <jobi@via.ecp.fr> *	    Joachim Orrblad <joachim@orrblad.com>*/#include <config.h>#include<libminisip/mediahandler/MediaStream.h>#include<libmikey/MikeyPayloadSP.h>#include<libmikey/keyagreement.h>#include<libminisip/sdp/SdpHeaderM.h>#include<libminisip/sdp/SdpHeaderA.h>#include<libminisip/sdp/SdpPacket.h>#include<libmnetutil/UDPSocket.h>#include<libmutil/itoa.h>#include<libmutil/Timestamp.h>#include<libmutil/print_hex.h>#include<libminisip/mediahandler/Media.h>#include<libminisip/mediahandler/RtpReceiver.h>#include<libminisip/codecs/Codec.h>#include<libminisip/ipprovider/IpProvider.h>#include<iostream>#ifdef _WIN32_WCE#	include"../include/minisip_wce_extra_includes.h"#endifusing namespace std;#ifdef _MSC_VERstatic int upcase(char c){	if ((c>='a') && (c<='z'))		return c - ('a'-'A');	else		return c;}static int nocaseequal(char c1, char c2){	return upcase(c1)==upcase(c2);}static int strcasecmp(const char *s1, const char *s2){	int i;	for ( i=0; s1[i]!=0 && s2[i]!=0; i++){		if ( !nocaseequal(s1[i],s2[i]) ){			if (s1[i]<s2[i]){				return -1;			}else{				return 1;			}		}	}	if (s2[i]!=0){		return -1;	}	return 0;}#endifMediaStream::MediaStream( MRef<Media *> media ):media(media),ka(NULL){	disabled = false;}std::string MediaStream::getSdpMediaType(){	if( media ){		return media->getSdpMediaType();	}	return "";}list<string> MediaStream::getSdpAttributes(){	return media->getSdpAttributes();}bool MediaStream::matches( MRef<SdpHeaderM *> m, uint32_t formatIndex ){        string sdpRtpMap;	string sdpFmtpParam;	        //	int i;        uint8_t sdpPayloadType = (uint8_t) m->getFormat( formatIndex );        media->handleMHeader( m );        // pn507 This checks for "Audio"        if( m->getMedia() != getSdpMediaType() ){                return false;        }        sdpRtpMap = m->getRtpMap( sdpPayloadType );	sdpFmtpParam = m->getFmtpParam( sdpPayloadType );		std::list<MRef<Codec *> > codecs = media->getAvailableCodecs();	std::list<MRef<Codec *> >::iterator iC;	string codecRtpMap;	uint8_t codecPayloadType;	        size_t s1;        size_t s2 = sdpRtpMap.find("/");		for( iC = codecs.begin(); iC != codecs.end(); iC ++ ){		codecRtpMap = (*iC)->getSdpMediaAttributes();		codecPayloadType = (*iC)->getSdpMediaType();		if( (*iC)->getCodecName() == "iLBC" ) {			if( sdpFmtpParam != "mode=20" ) { //iLBC only supports 20ms frames (in minisip)				continue;			} //else ... does not mean we accept it, it still goes through the normal checks ...		}                if( sdpRtpMap != "" && codecRtpMap != "" ){                        s1 = codecRtpMap.find("/");                        bool sdpRtpMapEqual = !strcasecmp( codecRtpMap.substr(0, s1).c_str(), sdpRtpMap.substr(0,s2).c_str() );                        if ( sdpRtpMapEqual ) {				localPayloadType = codecPayloadType;                                return true;                        }                        else {				continue;			}				                }                else{                        if( sdpPayloadType == codecPayloadType ){				localPayloadType = codecPayloadType;                                return true;                        }else{			}                }        }        return false;}MRef<CryptoContext *> MediaStream::initCrypto( uint32_t ssrc, uint16_t seq_no ){	MRef<CryptoContext *> cryptoContext;		kaLock.lock();	if( !ka ){		/* Dummy cryptocontext */		cryptoContext = new CryptoContext( ssrc );	}	else{				unsigned char * masterKey = new unsigned char[16];		unsigned char * masterSalt = new unsigned char[14];			uint8_t  csId = ka->getSrtpCsId( ssrc );		uint32_t roc = ka->getSrtpRoc( ssrc );		uint8_t  policyNo = ka->findpolicyNo( ssrc );		//Extract Srtp policy !!! Check the return value if type not available		uint8_t ealg  = ka->getPolicyParamTypeValue(policyNo, MIKEY_PROTO_SRTP, MIKEY_SRTP_EALG);		uint8_t ekeyl = ka->getPolicyParamTypeValue(policyNo, MIKEY_PROTO_SRTP, MIKEY_SRTP_EKEYL);		uint8_t aalg  = ka->getPolicyParamTypeValue(policyNo, MIKEY_PROTO_SRTP, MIKEY_SRTP_AALG);		uint8_t akeyl = ka->getPolicyParamTypeValue(policyNo, MIKEY_PROTO_SRTP, MIKEY_SRTP_AKEYL);		uint8_t skeyl = ka->getPolicyParamTypeValue(policyNo, MIKEY_PROTO_SRTP, MIKEY_SRTP_SALTKEYL);		//uint8_t prf   = ka->getPolicyParamTypeValue(policyNo, MIKEY_PROTO_SRTP, MIKEY_SRTP_PRF);	 //Not used		uint8_t keydr = ka->getPolicyParamTypeValue(policyNo, MIKEY_PROTO_SRTP, MIKEY_SRTP_KEY_DERRATE);		uint8_t encr  = ka->getPolicyParamTypeValue(policyNo, MIKEY_PROTO_SRTP, MIKEY_SRTP_ENCR_ON_OFF); 		//uint8_t cencr = ka->getPolicyParamTypeValue(policyNo, MIKEY_PROTO_SRTP, MIKEY_SRTCP_ENCR_ON_OFF);//Not used		//uint8_t fecor = ka->getPolicyParamTypeValue(policyNo, MIKEY_PROTO_SRTP, MIKEY_SRTP_FEC_ORDER);	 //Not used		uint8_t auth  = ka->getPolicyParamTypeValue(policyNo, MIKEY_PROTO_SRTP, MIKEY_SRTP_AUTH_ON_OFF); 		uint8_t autht = ka->getPolicyParamTypeValue(policyNo, MIKEY_PROTO_SRTP, MIKEY_SRTP_AUTH_TAGL);		//uint8_t prefi = ka->getPolicyParamTypeValue(policyNo, MIKEY_PROTO_SRTP, MIKEY_SRTP_PREFIX);	 //Not used				ka->genTek( csId,  masterKey,  16 );		ka->genSalt( csId, masterSalt, 14 );#ifdef DEBUG_OUTPUT#if 0		fprintf( stderr, "csId: %i\n", csId );		cerr << "SSRC: "<< ssrc <<" - TEK: " << print_hex( masterKey, 16 ) << endl;		cerr << "SSRC: "<< ssrc <<" - SALT: " << print_hex( masterSalt, 14 )<< endl;#endif#endif		if( csId != 0 ){			cryptoContext = new CryptoContext( ssrc, roc, seq_no, keydr, 			ealg, aalg, masterKey, 16, masterSalt, 14, ekeyl, akeyl, skeyl, encr, auth, autht );			cryptoContext->derive_srtp_keys( 0 );		}		else{			cryptoContext = new CryptoContext( ssrc );		}	}	cryptoContexts.push_back( cryptoContext );	kaLock.unlock();	return cryptoContext;}MRef<CryptoContext *> MediaStream::getCryptoContext( uint32_t ssrc, uint16_t seq_no ){	kaLock.lock();	list< MRef<CryptoContext *> >::iterator i;	for( i = cryptoContexts.begin(); i!= cryptoContexts.end(); i++ ){		if( (*i)->getSsrc() == ssrc ){			kaLock.unlock();			return (*i);		}	}	kaLock.unlock();	return initCrypto( ssrc, seq_no );}void MediaStream::setKeyAgreement( MRef<KeyAgreement *> ka ){	kaLock.lock();	this->ka = ka;	/* Reset the CryptoContext since we have a new KeyAgreement */	cryptoContexts.clear();	kaLock.unlock();}MediaStreamReceiver::MediaStreamReceiver( MRef<Media *> media, 		MRef<RtpReceiver *> rtpReceiver, MRef<IpProvider *> ipProvider ):			MediaStream( media ),			rtpReceiver( rtpReceiver ),			ipProvider( ipProvider ){	id = rand();	externalPort = 0;	running = false;	codecList = media->getAvailableCodecs();}uint32_t MediaStreamReceiver::getId(){	return id;}void MediaStreamReceiver::start(){	if( !running ){		rtpReceiver->registerMediaStream( this );		running = true;	}}void MediaStreamReceiver::stop(){	list<uint32_t>::iterator i;	rtpReceiver->unregisterMediaStream( this );	ssrcListLock.lock();	for( i = ssrcList.begin(); i != ssrcList.end(); i++ ){		media->unRegisterMediaSource( *i );	}	ssrcList.clear();	ssrcListLock.unlock();		running = false;}uint16_t MediaStreamReceiver::getPort(){	return rtpReceiver->getPort();}void MediaStreamReceiver::handleRtpPacket( MRef<SRtpPacket *> packet ){	uint32_t packetSsrc;	uint16_t seq_no;		//if packet is null, we had a read timeout from the rtpReceiver	if( !packet ) {		return;	}		packetSsrc = packet->getHeader().getSSRC();	seq_no = packet->getHeader().getSeqNo();			if( packet->unprotect( getCryptoContext( packetSsrc, seq_no ) )){		// Authentication or replay protection failed		return;	}	//uint16_t seqNo = packet->getHeader().getSeqNo(); //not used	//byte_t * data = packet->getContent(); //not used	//uint32_t size = packet->getContentLength(); //not used	//bool marker = packet->getHeader().getMarker(); //not used	//uint32_t ts = packet->getHeader().getTimestamp(); //not used	gotSsrc( packetSsrc );	media->playData( *packet );}void MediaStreamReceiver::gotSsrc( uint32_t ssrc ){	list<uint32_t>::iterator i;	ssrcListLock.lock();	for( i = ssrcList.begin(); i != ssrcList.end(); i++ ){		if( (*i) == ssrc ){			ssrcListLock.unlock();			return;		}	}		media->registerMediaSource( ssrc );	ssrcList.push_back( ssrc );	ssrcListLock.unlock();}std::list<MRef<Codec *> > MediaStreamReceiver::getAvailableCodecs(){	return codecList;}MediaStreamSender::MediaStreamSender( MRef<Media *> media, MRef<UDPSocket *> senderSocket ):	MediaStream( media ){	selectedCodec = NULL;	remotePort = 0; 	seqNo = (uint16_t)rand();	ssrc = rand();	lastTs = rand();        payloadType = 255;	setMuted( true );	muteCounter = 0;	if( senderSocket ){		this->senderSock = senderSocket;	}	else{		senderSock = new UDPSocket;		senderSock->setLowDelay();	}}void MediaStreamSender::start(){	media->registerMediaSender( this );}void MediaStreamSender::stop(){	media->unRegisterMediaSender( this );}void MediaStreamSender::setPort( uint16_t port ){	remotePort = port;}uint16_t MediaStreamSender::getPort(){	return remotePort;}static bool first=true;void MediaStreamSender::send( byte_t * data, uint32_t length, uint32_t * givenTs, bool marker, bool dtmf ){	SRtpPacket * packet;	if (first){#ifdef ENABLE_TS		ts.save("rtp_send");#endif		first=false;	}		senderLock.lock();	if( !(*givenTs) ){		//FIXME! get it from the CODEC,		// when we have one CODEC per sender		increaseLastTs(); //increase lastTs ... 				//lastTs += 160; 		*givenTs = lastTs;	}	else{		lastTs = *givenTs;	}	packet = new SRtpPacket( data, length, seqNo++, lastTs, ssrc );	if( dtmf ){		packet->getHeader().setPayloadType( 101 );	}	else{		if( payloadType != 255 )			packet->getHeader().setPayloadType( payloadType );		else			packet->getHeader().setPayloadType( selectedCodec->getSdpMediaType() );	}	if( marker ){		packet->getHeader().setMarker( marker );	}	packet->protect( getCryptoContext( ssrc, seqNo - 1 ) );	packet->sendTo( **senderSock, *remoteAddress, remotePort );	delete packet;	senderLock.unlock();}void MediaStreamSender::setRemoteAddress( IPAddress * remoteAddress ){	this->remoteAddress = remoteAddress;}#ifdef DEBUG_OUTPUTstring MediaStream::getDebugString() {	string ret;	ret = getMemObjectType() + " this=" + itoa((int64_t)this) +		": port=" + itoa(getPort());	return ret;}string MediaStreamReceiver::getDebugString() {	string ret;	ret = getMemObjectType() + " this=" + itoa((int64_t)this) +		": listening port=" + itoa(rtpReceiver->getPort());	for( std::list<uint32_t>::iterator it = ssrcList.begin();				it != ssrcList.end();				it++) {		"; ssrc=" + itoa((*it));	}	return ret;}string MediaStreamSender::getDebugString() {	string ret;		ret = getMemObjectType() + " this=" + itoa((int64_t)this) +		": port=" + itoa(getPort()) +		"; remotePort=" + itoa(remotePort);		if( isMuted() == true )		ret += "; isMuted=true";	else ret += "; isMuted=false";		return ret;}#endif//keep a counter, so we can send a keep alive//packet every now and then.//Max indicates every how many silenced packets we send one keep-alive//It returns true if the packet needs to be let through, false otherwisebool MediaStreamSender::muteKeepAlive( uint32_t max ) {	bool ret = false;		//if muted, only return true if packet is keep alive	if( isMuted() ) {		muteCounter++;		if( muteCounter >= max ) {			ret = true;			muteCounter = 0;		}	} else {		//if active sender, let through		ret = true;	}	return ret;}bool MediaStreamSender::matches( MRef<SdpHeaderM *> m, uint32_t formatIndex ){	bool result = MediaStream::matches( m, formatIndex );	if( result && !selectedCodec ){		selectedCodec = media->createCodecInstance( 				localPayloadType  );		payloadType = (uint8_t)m->getFormat( formatIndex );	}	return result;}uint32_t MediaStreamSender::getSsrc(){	return ssrc;}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -