/* * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved. * * Use of this source code is governed by a BSD-style license * that can be found in the LICENSE file in the root of the source * tree. An additional intellectual property rights grant can be found * in the file PATENTS. All contributing project authors may * be found in the AUTHORS file in the root of the source tree. */ #include "NETEQTEST_DummyRTPpacket.h" #include <assert.h> #include <stdio.h> #include <string.h> #ifdef WIN32 #include <winsock2.h> #else #include <netinet/in.h> // for htons, htonl, etc #endif int NETEQTEST_DummyRTPpacket::readFromFile(FILE *fp) { if (!fp) { return -1; } uint16_t length, plen; uint32_t offset; int packetLen = 0; bool readNextPacket = true; while (readNextPacket) { readNextPacket = false; if (fread(&length, 2, 1, fp) == 0) { reset(); return -2; } length = ntohs(length); if (fread(&plen, 2, 1, fp) == 0) { reset(); return -1; } packetLen = ntohs(plen); if (fread(&offset, 4, 1, fp) == 0) { reset(); return -1; } // Store in local variable until we have passed the reset below. uint32_t receiveTime = ntohl(offset); // Use length here because a plen of 0 specifies rtcp. length = (uint16_t) (length - _kRDHeaderLen); // check buffer size if (_datagram && _memSize < length + 1) { reset(); } if (!_datagram) { // Add one extra byte, to be able to fake a dummy payload of 1 byte. _datagram = new uint8_t[length + 1]; _memSize = length + 1; } memset(_datagram, 0, length + 1); if (length == 0) { _datagramLen = 0; _rtpParsed = false; return packetLen; } // Read basic header if (fread((unsigned short *) _datagram, 1, _kBasicHeaderLen, fp) != (size_t)_kBasicHeaderLen) { reset(); return -1; } _receiveTime = receiveTime; _datagramLen = _kBasicHeaderLen; // Parse the basic header webrtc::WebRtcRTPHeader tempRTPinfo; int P, X, CC; parseBasicHeader(&tempRTPinfo, &P, &X, &CC); // Check if we have to extend the header if (X != 0 || CC != 0) { int newLen = _kBasicHeaderLen + CC * 4 + X * 4; assert(_memSize >= newLen); // Read extension from file size_t readLen = newLen - _kBasicHeaderLen; if (fread(&_datagram[_kBasicHeaderLen], 1, readLen, fp) != readLen) { reset(); return -1; } _datagramLen = newLen; if (X != 0) { int totHdrLen = calcHeaderLength(X, CC); assert(_memSize >= totHdrLen); // Read extension from file size_t readLen = totHdrLen - newLen; if (fread(&_datagram[newLen], 1, readLen, fp) != readLen) { reset(); return -1; } _datagramLen = totHdrLen; } } _datagramLen = length; if (!_blockList.empty() && _blockList.count(payloadType()) > 0) { readNextPacket = true; } } _rtpParsed = false; assert(_memSize > _datagramLen); _payloadLen = 1; // Set the length to 1 byte. return packetLen; } int NETEQTEST_DummyRTPpacket::writeToFile(FILE *fp) { if (!fp) { return -1; } uint16_t length, plen; uint32_t offset; // length including RTPplay header length = htons(_datagramLen + _kRDHeaderLen); if (fwrite(&length, 2, 1, fp) != 1) { return -1; } // payload length plen = htons(_datagramLen); if (fwrite(&plen, 2, 1, fp) != 1) { return -1; } // offset (=receive time) offset = htonl(_receiveTime); if (fwrite(&offset, 4, 1, fp) != 1) { return -1; } // Figure out the length of the RTP header. int headerLen; if (_datagramLen == 0) { // No payload at all; we are done writing to file. headerLen = 0; } else { parseHeader(); headerLen = _payloadPtr - _datagram; assert(headerLen >= 0); } // write RTP header if (fwrite((unsigned short *) _datagram, 1, headerLen, fp) != static_cast<size_t>(headerLen)) { return -1; } return (headerLen + _kRDHeaderLen); // total number of bytes written } void NETEQTEST_DummyRTPpacket::parseHeader() { NETEQTEST_RTPpacket::parseHeader(); // Change _payloadLen to 1 byte. The memory should always be big enough. assert(_memSize > _datagramLen); _payloadLen = 1; }