mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-24 16:18:16 +00:00
vrawdepay: handle invalid payload better
Make sure we don't read more data than available in the input buffer. Clip the input data into the output buffer.
This commit is contained in:
parent
f604e20499
commit
6be0c7b762
1 changed files with 103 additions and 46 deletions
|
@ -266,9 +266,10 @@ static GstBuffer *
|
||||||
gst_rtp_vraw_depay_process (GstBaseRTPDepayload * depayload, GstBuffer * buf)
|
gst_rtp_vraw_depay_process (GstBaseRTPDepayload * depayload, GstBuffer * buf)
|
||||||
{
|
{
|
||||||
GstRtpVRawDepay *rtpvrawdepay;
|
GstRtpVRawDepay *rtpvrawdepay;
|
||||||
guint8 *payload, *data, *yp, *up, *vp, *headers;
|
guint8 *payload, *data, *dataend, *yp, *up, *vp, *headers;
|
||||||
guint32 timestamp;
|
guint32 timestamp;
|
||||||
guint cont, ystride, uvstride, pgroup;
|
guint cont, ystride, uvstride, pgroup, payload_len, size;
|
||||||
|
gint width, height, xinc, yinc;
|
||||||
|
|
||||||
rtpvrawdepay = GST_RTP_VRAW_DEPAY (depayload);
|
rtpvrawdepay = GST_RTP_VRAW_DEPAY (depayload);
|
||||||
|
|
||||||
|
@ -299,6 +300,8 @@ gst_rtp_vraw_depay_process (GstBaseRTPDepayload * depayload, GstBuffer * buf)
|
||||||
}
|
}
|
||||||
|
|
||||||
data = GST_BUFFER_DATA (rtpvrawdepay->outbuf);
|
data = GST_BUFFER_DATA (rtpvrawdepay->outbuf);
|
||||||
|
size = GST_BUFFER_SIZE (rtpvrawdepay->outbuf);
|
||||||
|
dataend = data + size;
|
||||||
|
|
||||||
/* get pointer and strides of the planes */
|
/* get pointer and strides of the planes */
|
||||||
yp = data + rtpvrawdepay->yp;
|
yp = data + rtpvrawdepay->yp;
|
||||||
|
@ -308,41 +311,79 @@ gst_rtp_vraw_depay_process (GstBaseRTPDepayload * depayload, GstBuffer * buf)
|
||||||
ystride = rtpvrawdepay->ystride;
|
ystride = rtpvrawdepay->ystride;
|
||||||
uvstride = rtpvrawdepay->uvstride;
|
uvstride = rtpvrawdepay->uvstride;
|
||||||
pgroup = rtpvrawdepay->pgroup;
|
pgroup = rtpvrawdepay->pgroup;
|
||||||
|
width = rtpvrawdepay->width;
|
||||||
|
height = rtpvrawdepay->height;
|
||||||
|
xinc = rtpvrawdepay->xinc;
|
||||||
|
yinc = rtpvrawdepay->yinc;
|
||||||
|
|
||||||
payload = gst_rtp_buffer_get_payload (buf);
|
payload = gst_rtp_buffer_get_payload (buf);
|
||||||
|
payload_len = gst_rtp_buffer_get_payload_len (buf);
|
||||||
|
|
||||||
|
if (payload_len < 3)
|
||||||
|
goto short_packet;
|
||||||
|
|
||||||
/* skip extended seqnum */
|
/* skip extended seqnum */
|
||||||
payload++;
|
payload += 2;
|
||||||
payload++;
|
payload_len -= 2;
|
||||||
|
|
||||||
/* remember header position */
|
/* remember header position */
|
||||||
headers = payload;
|
headers = payload;
|
||||||
|
|
||||||
/* find data start */
|
/* find data start */
|
||||||
do {
|
do {
|
||||||
|
if (payload_len < 6)
|
||||||
|
goto short_packet;
|
||||||
|
|
||||||
cont = payload[4] & 0x80;
|
cont = payload[4] & 0x80;
|
||||||
|
|
||||||
payload += 6;
|
payload += 6;
|
||||||
|
payload_len -= 6;
|
||||||
} while (cont);
|
} while (cont);
|
||||||
|
|
||||||
while (TRUE) {
|
while (TRUE) {
|
||||||
guint length, line, offs;
|
guint length, line, offs, plen;
|
||||||
guint8 *datap;
|
guint8 *datap;
|
||||||
|
|
||||||
/* read length and cont */
|
/* stop when we run out of data */
|
||||||
|
if (payload_len == 0)
|
||||||
|
break;
|
||||||
|
|
||||||
|
/* read length and cont. This should work because we iterated the headers
|
||||||
|
* above. */
|
||||||
length = (headers[0] << 8) | headers[1];
|
length = (headers[0] << 8) | headers[1];
|
||||||
line = ((headers[2] & 0x7f) << 8) | headers[3];
|
line = ((headers[2] & 0x7f) << 8) | headers[3];
|
||||||
offs = ((headers[4] & 0x7f) << 8) | headers[5];
|
offs = ((headers[4] & 0x7f) << 8) | headers[5];
|
||||||
cont = headers[4] & 0x80;
|
cont = headers[4] & 0x80;
|
||||||
headers += 6;
|
headers += 6;
|
||||||
|
|
||||||
/* sanity check */
|
/* length must be a multiple of pgroup */
|
||||||
if (line > (rtpvrawdepay->height - rtpvrawdepay->yinc))
|
if (length % pgroup != 0)
|
||||||
continue;
|
goto wrong_length;
|
||||||
if (offs > (rtpvrawdepay->width - rtpvrawdepay->xinc))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
GST_LOG_OBJECT (depayload, "writing length %u, line %u, offset %u", length,
|
if (length > payload_len)
|
||||||
line, offs);
|
length = payload_len;
|
||||||
|
|
||||||
|
/* sanity check */
|
||||||
|
if (line > (height - yinc)) {
|
||||||
|
GST_WARNING_OBJECT (depayload, "skipping line %d: out of range", line);
|
||||||
|
goto next;
|
||||||
|
}
|
||||||
|
if (offs > (width - xinc)) {
|
||||||
|
GST_WARNING_OBJECT (depayload, "skipping offset %d: out of range", offs);
|
||||||
|
goto next;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* calculate the maximim amount of bytes we can use per line */
|
||||||
|
if (offs + ((length / pgroup) * xinc) > (width - xinc)) {
|
||||||
|
plen = ((width - offs) * pgroup) / xinc;
|
||||||
|
GST_WARNING_OBJECT (depayload, "clipping length %d, offset %d", length,
|
||||||
|
offs);
|
||||||
|
} else
|
||||||
|
plen = length;
|
||||||
|
|
||||||
|
GST_LOG_OBJECT (depayload,
|
||||||
|
"writing length %u/%u, line %u, offset %u, remaining %u", plen, length,
|
||||||
|
line, offs, payload_len);
|
||||||
|
|
||||||
switch (rtpvrawdepay->format) {
|
switch (rtpvrawdepay->format) {
|
||||||
case GST_VIDEO_FORMAT_RGB:
|
case GST_VIDEO_FORMAT_RGB:
|
||||||
|
@ -351,25 +392,27 @@ gst_rtp_vraw_depay_process (GstBaseRTPDepayload * depayload, GstBuffer * buf)
|
||||||
case GST_VIDEO_FORMAT_BGRA:
|
case GST_VIDEO_FORMAT_BGRA:
|
||||||
case GST_VIDEO_FORMAT_UYVY:
|
case GST_VIDEO_FORMAT_UYVY:
|
||||||
/* samples are packed just like gstreamer packs them */
|
/* samples are packed just like gstreamer packs them */
|
||||||
offs /= rtpvrawdepay->xinc;
|
offs /= xinc;
|
||||||
datap = yp + (line * ystride) + (offs * pgroup);
|
datap = yp + (line * ystride) + (offs * pgroup);
|
||||||
memcpy (datap, payload, length);
|
|
||||||
payload += length;
|
memcpy (datap, payload, plen);
|
||||||
break;
|
break;
|
||||||
case GST_VIDEO_FORMAT_AYUV:
|
case GST_VIDEO_FORMAT_AYUV:
|
||||||
{
|
{
|
||||||
gint i;
|
gint i;
|
||||||
|
guint8 *p;
|
||||||
|
|
||||||
datap = yp + (line * ystride) + (offs * 4);
|
datap = yp + (line * ystride) + (offs * 4);
|
||||||
|
p = payload;
|
||||||
|
|
||||||
/* samples are packed in order Cb-Y-Cr for both interlaced and
|
/* samples are packed in order Cb-Y-Cr for both interlaced and
|
||||||
* progressive frames */
|
* progressive frames */
|
||||||
for (i = 0; i < length; i += pgroup) {
|
for (i = 0; i < plen; i += pgroup) {
|
||||||
*datap++ = 0;
|
*datap++ = 0;
|
||||||
*datap++ = payload[1];
|
*datap++ = p[1];
|
||||||
*datap++ = payload[0];
|
*datap++ = p[0];
|
||||||
*datap++ = payload[2];
|
*datap++ = p[2];
|
||||||
payload += pgroup;
|
p += pgroup;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -377,25 +420,25 @@ gst_rtp_vraw_depay_process (GstBaseRTPDepayload * depayload, GstBuffer * buf)
|
||||||
{
|
{
|
||||||
gint i;
|
gint i;
|
||||||
guint uvoff;
|
guint uvoff;
|
||||||
guint8 *yd1p, *yd2p, *udp, *vdp;
|
guint8 *yd1p, *yd2p, *udp, *vdp, *p;
|
||||||
|
|
||||||
yd1p = yp + (line * ystride) + (offs);
|
yd1p = yp + (line * ystride) + (offs);
|
||||||
yd2p = yd1p + ystride;
|
yd2p = yd1p + ystride;
|
||||||
uvoff =
|
uvoff = (line / yinc * uvstride) + (offs / xinc);
|
||||||
(line / rtpvrawdepay->yinc * uvstride) +
|
|
||||||
(offs / rtpvrawdepay->xinc);
|
|
||||||
udp = up + uvoff;
|
udp = up + uvoff;
|
||||||
vdp = vp + uvoff;
|
vdp = vp + uvoff;
|
||||||
|
p = payload;
|
||||||
|
|
||||||
/* line 0/1: Y00-Y01-Y10-Y11-Cb00-Cr00 Y02-Y03-Y12-Y13-Cb01-Cr01 ... */
|
/* line 0/1: Y00-Y01-Y10-Y11-Cb00-Cr00 Y02-Y03-Y12-Y13-Cb01-Cr01 ... */
|
||||||
for (i = 0; i < length; i += pgroup) {
|
for (i = 0; i < plen; i += pgroup) {
|
||||||
*yd1p++ = payload[0];
|
*yd1p++ = p[0];
|
||||||
*yd1p++ = payload[1];
|
*yd1p++ = p[1];
|
||||||
*yd2p++ = payload[2];
|
*yd2p++ = p[2];
|
||||||
*yd2p++ = payload[3];
|
*yd2p++ = p[3];
|
||||||
*udp++ = payload[4];
|
*udp++ = p[4];
|
||||||
*vdp++ = payload[5];
|
*vdp++ = p[5];
|
||||||
payload += pgroup;
|
p += pgroup;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -403,25 +446,25 @@ gst_rtp_vraw_depay_process (GstBaseRTPDepayload * depayload, GstBuffer * buf)
|
||||||
{
|
{
|
||||||
gint i;
|
gint i;
|
||||||
guint uvoff;
|
guint uvoff;
|
||||||
guint8 *ydp, *udp, *vdp;
|
guint8 *ydp, *udp, *vdp, *p;
|
||||||
|
|
||||||
ydp = yp + (line * ystride) + (offs);
|
ydp = yp + (line * ystride) + (offs);
|
||||||
uvoff =
|
uvoff = (line / yinc * uvstride) + (offs / xinc);
|
||||||
(line / rtpvrawdepay->yinc * uvstride) +
|
|
||||||
(offs / rtpvrawdepay->xinc);
|
|
||||||
udp = up + uvoff;
|
udp = up + uvoff;
|
||||||
vdp = vp + uvoff;
|
vdp = vp + uvoff;
|
||||||
|
p = payload;
|
||||||
|
|
||||||
/* Samples are packed in order Cb0-Y0-Y1-Cr0-Y2-Y3 for both interlaced
|
/* Samples are packed in order Cb0-Y0-Y1-Cr0-Y2-Y3 for both interlaced
|
||||||
* and progressive scan lines */
|
* and progressive scan lines */
|
||||||
for (i = 0; i < length; i += pgroup) {
|
for (i = 0; i < plen; i += pgroup) {
|
||||||
*udp++ = payload[0];
|
*udp++ = p[0];
|
||||||
*ydp++ = payload[1];
|
*ydp++ = p[1];
|
||||||
*ydp++ = payload[2];
|
*ydp++ = p[2];
|
||||||
*vdp++ = payload[3];
|
*vdp++ = p[3];
|
||||||
*ydp++ = payload[4];
|
*ydp++ = p[4];
|
||||||
*ydp++ = payload[5];
|
*ydp++ = p[5];
|
||||||
payload += pgroup;
|
p += pgroup;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -429,8 +472,12 @@ gst_rtp_vraw_depay_process (GstBaseRTPDepayload * depayload, GstBuffer * buf)
|
||||||
goto unknown_sampling;
|
goto unknown_sampling;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
next:
|
||||||
if (!cont)
|
if (!cont)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
payload += length;
|
||||||
|
payload_len -= length;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (gst_rtp_buffer_get_marker (buf)) {
|
if (gst_rtp_buffer_get_marker (buf)) {
|
||||||
|
@ -453,7 +500,17 @@ unknown_sampling:
|
||||||
}
|
}
|
||||||
alloc_failed:
|
alloc_failed:
|
||||||
{
|
{
|
||||||
GST_DEBUG_OBJECT (depayload, "failed to alloc output buffer");
|
GST_WARNING_OBJECT (depayload, "failed to alloc output buffer");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
wrong_length:
|
||||||
|
{
|
||||||
|
GST_WARNING_OBJECT (depayload, "length not multiple of pgroup");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
short_packet:
|
||||||
|
{
|
||||||
|
GST_WARNING_OBJECT (depayload, "short packet");
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue