2018-02-26 19:35:16 -05:00
|
|
|
/*
|
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
This file is a part of JRTPLIB
|
|
|
|
Copyright (c) 1999-2017 Jori Liesenborgs
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
Contact: jori.liesenborgs@gmail.com
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
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).
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
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:
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
The above copyright notice and this permission notice shall be included
|
|
|
|
in all copies or substantial portions of the Software.
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
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.
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
*/
|
2018-02-26 19:35:16 -05:00
|
|
|
|
|
|
|
#include "rtcpcompoundpacket.h"
|
|
|
|
#include "rtprawpacket.h"
|
|
|
|
#include "rtperrors.h"
|
|
|
|
#include "rtpstructs.h"
|
|
|
|
#include "rtpdefines.h"
|
|
|
|
#include "rtcpsrpacket.h"
|
|
|
|
#include "rtcprrpacket.h"
|
|
|
|
#include "rtcpsdespacket.h"
|
|
|
|
#include "rtcpbyepacket.h"
|
|
|
|
#include "rtcpapppacket.h"
|
|
|
|
#include "rtcpunknownpacket.h"
|
|
|
|
|
|
|
|
namespace qrtplib
|
|
|
|
{
|
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
RTCPCompoundPacket::RTCPCompoundPacket(RTPRawPacket &rawpack)
|
2018-02-26 19:35:16 -05:00
|
|
|
{
|
2018-02-27 17:05:46 -05:00
|
|
|
compoundpacket = 0;
|
|
|
|
compoundpacketlength = 0;
|
|
|
|
error = 0;
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
if (rawpack.IsRTP())
|
|
|
|
{
|
|
|
|
error = ERR_RTP_RTCPCOMPOUND_INVALIDPACKET;
|
|
|
|
return;
|
|
|
|
}
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
uint8_t *data = rawpack.GetData();
|
2018-02-28 07:31:44 -05:00
|
|
|
std::size_t datalen = rawpack.GetDataLength();
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
error = ParseData(data, datalen);
|
2018-03-04 19:17:19 -05:00
|
|
|
|
|
|
|
if (error < 0) {
|
2018-02-27 17:05:46 -05:00
|
|
|
return;
|
2018-03-04 19:17:19 -05:00
|
|
|
}
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
compoundpacket = rawpack.GetData();
|
|
|
|
compoundpacketlength = rawpack.GetDataLength();
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
rtcppackit = rtcppacklist.begin();
|
2018-02-26 19:35:16 -05:00
|
|
|
}
|
|
|
|
|
2018-03-04 19:17:19 -05:00
|
|
|
RTCPCompoundPacket::RTCPCompoundPacket(uint8_t *packet, std::size_t packetlen)
|
2018-02-26 19:35:16 -05:00
|
|
|
{
|
2018-02-27 17:05:46 -05:00
|
|
|
compoundpacket = 0;
|
|
|
|
compoundpacketlength = 0;
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
error = ParseData(packet, packetlen);
|
2018-03-04 19:17:19 -05:00
|
|
|
|
|
|
|
if (error < 0) {
|
2018-02-27 17:05:46 -05:00
|
|
|
return;
|
2018-03-04 19:17:19 -05:00
|
|
|
}
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
compoundpacket = packet;
|
|
|
|
compoundpacketlength = packetlen;
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
rtcppackit = rtcppacklist.begin();
|
2018-02-26 19:35:16 -05:00
|
|
|
}
|
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
RTCPCompoundPacket::RTCPCompoundPacket()
|
2018-02-26 19:35:16 -05:00
|
|
|
{
|
2018-02-27 17:05:46 -05:00
|
|
|
compoundpacket = 0;
|
|
|
|
compoundpacketlength = 0;
|
|
|
|
error = 0;
|
2018-02-26 19:35:16 -05:00
|
|
|
}
|
|
|
|
|
2018-02-28 07:31:44 -05:00
|
|
|
int RTCPCompoundPacket::ParseData(uint8_t *data, std::size_t datalen)
|
2018-02-26 19:35:16 -05:00
|
|
|
{
|
2018-02-27 17:05:46 -05:00
|
|
|
bool first;
|
|
|
|
|
|
|
|
if (datalen < sizeof(RTCPCommonHeader))
|
|
|
|
return ERR_RTP_RTCPCOMPOUND_INVALIDPACKET;
|
|
|
|
|
|
|
|
first = true;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
RTCPCommonHeader *rtcphdr;
|
2018-02-28 07:31:44 -05:00
|
|
|
std::size_t length;
|
2018-02-27 17:05:46 -05:00
|
|
|
|
|
|
|
rtcphdr = (RTCPCommonHeader *) data;
|
|
|
|
if (rtcphdr->version != RTP_VERSION) // check version
|
|
|
|
{
|
|
|
|
ClearPacketList();
|
|
|
|
return ERR_RTP_RTCPCOMPOUND_INVALIDPACKET;
|
|
|
|
}
|
|
|
|
if (first)
|
|
|
|
{
|
|
|
|
// Check if first packet is SR or RR
|
|
|
|
|
|
|
|
first = false;
|
|
|
|
if (!(rtcphdr->packettype == RTP_RTCPTYPE_SR || rtcphdr->packettype == RTP_RTCPTYPE_RR))
|
|
|
|
{
|
|
|
|
ClearPacketList();
|
|
|
|
return ERR_RTP_RTCPCOMPOUND_INVALIDPACKET;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-28 07:31:44 -05:00
|
|
|
length = (std::size_t) m_endian.qToHost(rtcphdr->length);
|
2018-02-27 17:05:46 -05:00
|
|
|
length++;
|
|
|
|
length *= sizeof(uint32_t);
|
|
|
|
|
|
|
|
if (length > datalen) // invalid length field
|
|
|
|
{
|
|
|
|
ClearPacketList();
|
|
|
|
return ERR_RTP_RTCPCOMPOUND_INVALIDPACKET;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rtcphdr->padding)
|
|
|
|
{
|
|
|
|
// check if it's the last packet
|
|
|
|
if (length != datalen)
|
|
|
|
{
|
|
|
|
ClearPacketList();
|
|
|
|
return ERR_RTP_RTCPCOMPOUND_INVALIDPACKET;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
RTCPPacket *p;
|
|
|
|
|
|
|
|
switch (rtcphdr->packettype)
|
|
|
|
{
|
|
|
|
case RTP_RTCPTYPE_SR:
|
|
|
|
p = new RTCPSRPacket(data, length);
|
|
|
|
break;
|
|
|
|
case RTP_RTCPTYPE_RR:
|
|
|
|
p = new RTCPRRPacket(data, length);
|
|
|
|
break;
|
|
|
|
case RTP_RTCPTYPE_SDES:
|
|
|
|
p = new RTCPSDESPacket(data, length);
|
|
|
|
break;
|
|
|
|
case RTP_RTCPTYPE_BYE:
|
|
|
|
p = new RTCPBYEPacket(data, length);
|
|
|
|
break;
|
|
|
|
case RTP_RTCPTYPE_APP:
|
|
|
|
p = new RTCPAPPPacket(data, length);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
p = new RTCPUnknownPacket(data, length);
|
|
|
|
}
|
|
|
|
|
|
|
|
rtcppacklist.push_back(p);
|
|
|
|
|
|
|
|
datalen -= length;
|
|
|
|
data += length;
|
2018-02-28 07:31:44 -05:00
|
|
|
} while (datalen >= (std::size_t) sizeof(RTCPCommonHeader));
|
2018-02-27 17:05:46 -05:00
|
|
|
|
|
|
|
if (datalen != 0) // some remaining bytes
|
|
|
|
{
|
|
|
|
ClearPacketList();
|
|
|
|
return ERR_RTP_RTCPCOMPOUND_INVALIDPACKET;
|
|
|
|
}
|
|
|
|
return 0;
|
2018-02-26 19:35:16 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
RTCPCompoundPacket::~RTCPCompoundPacket()
|
|
|
|
{
|
2018-02-27 17:05:46 -05:00
|
|
|
ClearPacketList();
|
2018-02-26 19:35:16 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void RTCPCompoundPacket::ClearPacketList()
|
|
|
|
{
|
2018-02-27 17:05:46 -05:00
|
|
|
std::list<RTCPPacket *>::const_iterator it;
|
2018-02-26 19:35:16 -05:00
|
|
|
|
2018-03-04 19:17:19 -05:00
|
|
|
for (it = rtcppacklist.begin(); it != rtcppacklist.end(); it++) {
|
2018-02-27 17:05:46 -05:00
|
|
|
delete *it;
|
2018-03-04 19:17:19 -05:00
|
|
|
}
|
|
|
|
|
2018-02-27 17:05:46 -05:00
|
|
|
rtcppacklist.clear();
|
|
|
|
rtcppackit = rtcppacklist.begin();
|
2018-02-26 19:35:16 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
} // end namespace
|
|
|
|
|