| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  This file is a part of JRTPLIB | 
					
						
							|  |  |  |  Copyright (c) 1999-2017 Jori Liesenborgs | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  Contact: jori.liesenborgs@gmail.com | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  This library was developed at the Expertise Centre for Digital Media | 
					
						
							|  |  |  |  (http://www.edm.uhasselt.be), a research center of the Hasselt University
 | 
					
						
							|  |  |  |  (http://www.uhasselt.be). The library is based upon work done for
 | 
					
						
							|  |  |  |  my thesis at the School for Knowledge Technology (Belgium/The Netherlands). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  Permission is hereby granted, free of charge, to any person obtaining a | 
					
						
							|  |  |  |  copy of this software and associated documentation files (the "Software"), | 
					
						
							|  |  |  |  to deal in the Software without restriction, including without limitation | 
					
						
							|  |  |  |  the rights to use, copy, modify, merge, publish, distribute, sublicense, | 
					
						
							|  |  |  |  and/or sell copies of the Software, and to permit persons to whom the | 
					
						
							|  |  |  |  Software is furnished to do so, subject to the following conditions: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  The above copyright notice and this permission notice shall be included | 
					
						
							|  |  |  |  in all copies or substantial portions of the Software. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS | 
					
						
							|  |  |  |  OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
					
						
							|  |  |  |  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL | 
					
						
							|  |  |  |  THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | 
					
						
							|  |  |  |  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING | 
					
						
							|  |  |  |  FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS | 
					
						
							|  |  |  |  IN THE SOFTWARE. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "rtpudptransmitter.h"
 | 
					
						
							|  |  |  | #include "rtperrors.h"
 | 
					
						
							| 
									
										
										
										
											2018-03-05 01:17:19 +01:00
										 |  |  | #include "rtpaddress.h"
 | 
					
						
							|  |  |  | #include "rtpstructs.h"
 | 
					
						
							|  |  |  | #include "rtprawpacket.h"
 | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 01:17:19 +01:00
										 |  |  | #include <QUdpSocket>
 | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace qrtplib | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 01:17:19 +01:00
										 |  |  | RTPUDPTransmitter::RTPUDPTransmitter() : | 
					
						
							|  |  |  |     m_rawPacketQueueLock(QMutex::Recursive) | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     m_created = false; | 
					
						
							|  |  |  |     m_init = false; | 
					
						
							|  |  |  |     m_rtcpsock = 0; | 
					
						
							|  |  |  |     m_rtpsock = 0; | 
					
						
							| 
									
										
										
										
											2018-03-08 00:16:24 +01:00
										 |  |  |     m_deletesocketswhendone = false; | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  |     m_waitingfordata = false; | 
					
						
							|  |  |  |     m_rtcpPort = 0; | 
					
						
							|  |  |  |     m_rtpPort = 0; | 
					
						
							|  |  |  |     m_receivemode = RTPTransmitter::AcceptAll; | 
					
						
							| 
									
										
										
										
											2018-04-04 23:39:31 +02:00
										 |  |  |     m_maxpacksize = 0; | 
					
						
							|  |  |  |     memset(m_rtpBuffer, 0, m_absoluteMaxPackSize); | 
					
						
							|  |  |  |     memset(m_rtcpBuffer, 0, m_absoluteMaxPackSize); | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | RTPUDPTransmitter::~RTPUDPTransmitter() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Destroy(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int RTPUDPTransmitter::Init() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_init) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_ALREADYINIT; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_init = true; | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int RTPUDPTransmitter::Create(std::size_t maximumpacketsize, const RTPTransmissionParams *transparams) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const RTPUDPTransmissionParams *params, defaultparams; | 
					
						
							|  |  |  |     qint64 size; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 01:17:19 +01:00
										 |  |  |     if (maximumpacketsize > m_absoluteMaxPackSize) { | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  |         return ERR_RTP_UDPV4TRANS_SPECIFIEDSIZETOOBIG; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!m_init) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_NOTINIT; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_created) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_ALREADYCREATED; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Obtain transmission parameters
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (transparams == 0) { | 
					
						
							|  |  |  |         params = &defaultparams; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (transparams->GetTransmissionProtocol() != RTPTransmitter::IPv4UDPProto) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             return ERR_RTP_UDPV4TRANS_ILLEGALPARAMETERS; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-03-05 01:17:19 +01:00
										 |  |  |         params = (const RTPUDPTransmissionParams *) transparams; | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Determine the port numbers
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_localIP = params->GetBindIP(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (params->GetAllowOddPortbase()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_rtpPort = params->GetPortbase(); | 
					
						
							|  |  |  |         m_rtcpPort = params->GetForcedRTCPPort(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (params->GetPortbase() % 2 == 0) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             m_rtpPort = params->GetPortbase(); | 
					
						
							|  |  |  |             m_rtcpPort = m_rtpPort + 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             return ERR_RTP_UDPV4TRANS_PORTBASENOTEVEN; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-08 00:16:24 +01:00
										 |  |  |     if (params->GetUseExistingSockets(&m_rtpsock, &m_rtcpsock)) | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-03-08 00:16:24 +01:00
										 |  |  |         m_deletesocketswhendone = false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_deletesocketswhendone = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         m_rtpsock = new QUdpSocket(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // If we're multiplexing, we're just going to set the RTCP socket to equal the RTP socket
 | 
					
						
							|  |  |  |         if (params->GetRTCPMultiplexing()) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             m_rtcpsock = m_rtpsock; | 
					
						
							|  |  |  |             m_rtcpPort = m_rtpPort; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             m_rtcpsock = new QUdpSocket(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-06 02:23:47 +01:00
										 |  |  |     // set socket buffer sizes
 | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-06 02:23:47 +01:00
										 |  |  |     size = params->GetRTPReceiveBufferSize(); | 
					
						
							|  |  |  |     m_rtpsock->setReadBufferSize(size); | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-06 02:23:47 +01:00
										 |  |  |     if (m_rtpsock != m_rtcpsock) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         size = params->GetRTCPReceiveBufferSize(); | 
					
						
							|  |  |  |         m_rtcpsock->setReadBufferSize(size); | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_maxpacksize = maximumpacketsize; | 
					
						
							| 
									
										
										
										
											2018-03-05 01:17:19 +01:00
										 |  |  |     m_multicastInterface = params->GetMulticastInterface(); | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  |     m_receivemode = RTPTransmitter::AcceptAll; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_waitingfordata = false; | 
					
						
							|  |  |  |     m_created = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 01:17:19 +01:00
										 |  |  | int RTPUDPTransmitter::BindSockets() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_rtpsock->bind(m_localIP, m_rtpPort)) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_CANTBINDRTPSOCKET; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     connect(m_rtpsock, SIGNAL(readyRead()), this, SLOT(readRTPPendingDatagrams())); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_rtpsock != m_rtcpsock) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (!m_rtcpsock->bind(m_localIP, m_rtcpPort)) { | 
					
						
							|  |  |  |             return ERR_RTP_UDPV4TRANS_CANTBINDRTCPSOCKET; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         connect(m_rtcpsock, SIGNAL(readyRead()), this, SLOT(readRTCPPendingDatagrams())); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  | void RTPUDPTransmitter::Destroy() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_init) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!m_created) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-08 00:16:24 +01:00
										 |  |  |     if (m_deletesocketswhendone) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (m_rtpsock != m_rtcpsock) { | 
					
						
							|  |  |  |             delete m_rtcpsock; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-08 00:16:24 +01:00
										 |  |  |         delete m_rtpsock; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-03-06 02:23:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  |     m_created = false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | RTPTransmissionInfo *RTPUDPTransmitter::GetTransmissionInfo() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_init) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 01:17:19 +01:00
										 |  |  |     RTPTransmissionInfo *tinf = new RTPUDPTransmissionInfo(m_localIP, m_rtpsock, m_rtcpsock, m_rtpPort, m_rtcpPort); | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return tinf; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void RTPUDPTransmitter::DeleteTransmissionInfo(RTPTransmissionInfo *inf) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_init) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     delete inf; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 01:17:19 +01:00
										 |  |  | bool RTPUDPTransmitter::ComesFromThisTransmitter(const RTPAddress& addr) | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-05 01:17:19 +01:00
										 |  |  |     if (addr.getAddress() != m_localIP) { | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 01:17:19 +01:00
										 |  |  |     return (addr.getPort() == m_rtpPort) && (addr.getRtcpsendport() == m_rtcpPort); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int RTPUDPTransmitter::SendRTPData(const void *data, std::size_t len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_init) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_NOTINIT; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!m_created) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_NOTCREATED; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (len > m_maxpacksize) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_SPECIFIEDSIZETOOBIG; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::list<RTPAddress>::const_iterator it = m_destinations.begin(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (; it != m_destinations.end(); ++it) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_rtpsock->writeDatagram((const char*) data, (qint64) len, it->getAddress(), it->getPort()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int RTPUDPTransmitter::SendRTCPData(const void *data, std::size_t len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_init) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_NOTINIT; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!m_created) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_NOTCREATED; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (len > m_maxpacksize) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_SPECIFIEDSIZETOOBIG; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::list<RTPAddress>::const_iterator it = m_destinations.begin(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (; it != m_destinations.end(); ++it) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_rtcpsock->writeDatagram((const char*) data, (qint64) len, it->getAddress(), it->getRtcpsendport()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int RTPUDPTransmitter::AddDestination(const RTPAddress &addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_destinations.push_back(addr); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int RTPUDPTransmitter::DeleteDestination(const RTPAddress &addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_destinations.remove(addr); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void RTPUDPTransmitter::ClearDestinations() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_destinations.clear(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool RTPUDPTransmitter::SupportsMulticasting() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QNetworkInterface::InterfaceFlags flags = m_multicastInterface.flags(); | 
					
						
							|  |  |  |     QAbstractSocket::SocketState rtpSocketState = m_rtpsock->state(); | 
					
						
							|  |  |  |     QAbstractSocket::SocketState rtcpSocketState = m_rtcpsock->state(); | 
					
						
							|  |  |  |     return m_multicastInterface.isValid() | 
					
						
							|  |  |  |             && (rtpSocketState & QAbstractSocket::BoundState) | 
					
						
							|  |  |  |             && (rtcpSocketState & QAbstractSocket::BoundState) | 
					
						
							|  |  |  |             && (flags & QNetworkInterface::CanMulticast) | 
					
						
							|  |  |  |             && (flags & QNetworkInterface::IsRunning) | 
					
						
							|  |  |  |             && !(flags & QNetworkInterface::IsLoopBack); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int RTPUDPTransmitter::JoinMulticastGroup(const RTPAddress &addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_init) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_NOTINIT; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!m_created) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_NOTCREATED; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!SupportsMulticasting()) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV6TRANS_NOMULTICASTSUPPORT; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_rtpsock->joinMulticastGroup(addr.getAddress(), m_multicastInterface)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (m_rtpsock != m_rtcpsock) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (!m_rtcpsock->joinMulticastGroup(addr.getAddress(), m_multicastInterface)) { | 
					
						
							|  |  |  |                 return ERR_RTP_UDPV4TRANS_COULDNTJOINMULTICASTGROUP; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_COULDNTJOINMULTICASTGROUP; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int RTPUDPTransmitter::LeaveMulticastGroup(const RTPAddress &addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_init) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_NOTINIT; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!m_created) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_NOTCREATED; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!SupportsMulticasting()) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV6TRANS_NOMULTICASTSUPPORT; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_rtpsock->leaveMulticastGroup(addr.getAddress()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_rtpsock != m_rtcpsock) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_rtcpsock->leaveMulticastGroup(addr.getAddress()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int RTPUDPTransmitter::SetReceiveMode(RTPTransmitter::ReceiveMode m) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_init) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_NOTINIT; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!m_created) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_NOTCREATED; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m != m_receivemode) { | 
					
						
							|  |  |  |         m_receivemode = m; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int RTPUDPTransmitter::AddToIgnoreList(const RTPAddress &addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_ignoreList.push_back(addr); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int RTPUDPTransmitter::DeleteFromIgnoreList(const RTPAddress &addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_ignoreList.remove(addr); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void RTPUDPTransmitter::ClearIgnoreList() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_ignoreList.clear(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int RTPUDPTransmitter::AddToAcceptList(const RTPAddress &addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_acceptList.push_back(addr); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int RTPUDPTransmitter::DeleteFromAcceptList(const RTPAddress &addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_acceptList.remove(addr); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void RTPUDPTransmitter::ClearAcceptList() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_acceptList.clear(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int RTPUDPTransmitter::SetMaximumPacketSize(std::size_t s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!m_init) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_NOTINIT; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!m_created) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_NOTCREATED; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (s > m_absoluteMaxPackSize) { | 
					
						
							|  |  |  |         return ERR_RTP_UDPV4TRANS_SPECIFIEDSIZETOOBIG; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_maxpacksize = s; | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | RTPRawPacket *RTPUDPTransmitter::GetNextPacket() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QMutexLocker locker(&m_rawPacketQueueLock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_rawPacketQueue.isEmpty()) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         return m_rawPacketQueue.takeFirst(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void RTPUDPTransmitter::readRTPPendingDatagrams() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     while (m_rtpsock->hasPendingDatagrams()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         RTPTime curtime = RTPTime::CurrentTime(); | 
					
						
							|  |  |  |         QHostAddress remoteAddress; | 
					
						
							|  |  |  |         quint16 remotePort; | 
					
						
							|  |  |  |         qint64 pendingDataSize = m_rtpsock->pendingDatagramSize(); | 
					
						
							|  |  |  |         qint64 bytesRead = m_rtpsock->readDatagram(m_rtpBuffer, pendingDataSize, &remoteAddress, &remotePort); | 
					
						
							|  |  |  |         qDebug("RTPUDPTransmitter::readRTPPendingDatagrams: %lld bytes read from %s:%d", | 
					
						
							|  |  |  |                 bytesRead, | 
					
						
							|  |  |  |                 qPrintable(remoteAddress.toString()), | 
					
						
							|  |  |  |                 remotePort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         RTPAddress rtpAddress; | 
					
						
							|  |  |  |         rtpAddress.setAddress(remoteAddress); | 
					
						
							|  |  |  |         rtpAddress.setPort(remotePort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (ShouldAcceptData(rtpAddress)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             bool isrtp = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (m_rtpsock == m_rtcpsock) // check payload type when multiplexing
 | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 if ((std::size_t) bytesRead > sizeof(RTCPCommonHeader)) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     RTCPCommonHeader *rtcpheader = (RTCPCommonHeader *) m_rtpBuffer; | 
					
						
							|  |  |  |                     uint8_t packettype = rtcpheader->packettype; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     if (packettype >= 200 && packettype <= 204) { | 
					
						
							|  |  |  |                         isrtp = false; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             RTPRawPacket *pack = new RTPRawPacket((uint8_t *) m_rtpBuffer, bytesRead, rtpAddress, curtime, isrtp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             m_rawPacketQueueLock.lock(); | 
					
						
							|  |  |  |             m_rawPacketQueue.append(pack); | 
					
						
							|  |  |  |             m_rawPacketQueueLock.unlock(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             emit NewDataAvailable(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void RTPUDPTransmitter::readRTCPPendingDatagrams() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     while (m_rtcpsock->hasPendingDatagrams()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         RTPTime curtime = RTPTime::CurrentTime(); | 
					
						
							|  |  |  |         QHostAddress remoteAddress; | 
					
						
							|  |  |  |         quint16 remotePort; | 
					
						
							|  |  |  |         qint64 pendingDataSize = m_rtcpsock->pendingDatagramSize(); | 
					
						
							|  |  |  |         qint64 bytesRead = m_rtcpsock->readDatagram(m_rtcpBuffer, pendingDataSize, &remoteAddress, &remotePort); | 
					
						
							|  |  |  |         qDebug("RTPUDPTransmitter::readRTCPPendingDatagrams: %lld bytes read from %s:%d", | 
					
						
							|  |  |  |                 bytesRead, | 
					
						
							|  |  |  |                 qPrintable(remoteAddress.toString()), | 
					
						
							|  |  |  |                 remotePort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         RTPAddress rtpAddress; | 
					
						
							|  |  |  |         rtpAddress.setAddress(remoteAddress); | 
					
						
							|  |  |  |         rtpAddress.setPort(remotePort); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (ShouldAcceptData(rtpAddress)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             RTPRawPacket *pack = new RTPRawPacket((uint8_t *) m_rtcpBuffer, bytesRead, rtpAddress, curtime, false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             m_rawPacketQueueLock.lock(); | 
					
						
							|  |  |  |             m_rawPacketQueue.append(pack); | 
					
						
							|  |  |  |             m_rawPacketQueueLock.unlock(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             emit NewDataAvailable(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool RTPUDPTransmitter::ShouldAcceptData(const RTPAddress& rtpAddress) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_receivemode == RTPTransmitter::AcceptAll) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (m_receivemode == RTPTransmitter::AcceptSome) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         std::list<RTPAddress>::iterator findIt = std::find(m_acceptList.begin(), m_acceptList.end(), rtpAddress); | 
					
						
							|  |  |  |         return findIt != m_acceptList.end(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-06-21 19:28:11 +02:00
										 |  |  |     else // this is RTPTransmitter::IgnoreSome
 | 
					
						
							| 
									
										
										
										
											2018-03-05 01:17:19 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         std::list<RTPAddress>::iterator findIt = std::find(m_ignoreList.begin(), m_ignoreList.end(), rtpAddress); | 
					
						
							|  |  |  |         return findIt == m_ignoreList.end(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-03-01 01:40:04 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 |