/* * Copyright (c) 2014 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 "modules/rtp_rtcp/source/rtcp_packet.h" #include "modules/rtp_rtcp/include/rtp_rtcp_defines.h" #include "rtc_base/checks.h" namespace webrtc { namespace rtcp { constexpr size_t RtcpPacket::kHeaderLength; rtc::Buffer RtcpPacket::Build() const { rtc::Buffer packet(BlockLength()); size_t length = 0; bool created = Create(packet.data(), &length, packet.capacity(), nullptr); RTC_DCHECK(created) << "Invalid packet is not supported."; RTC_DCHECK_EQ(length, packet.size()) << "BlockLength mispredicted size used by Create"; return packet; } bool RtcpPacket::Build(size_t max_length, PacketReadyCallback callback) const { RTC_CHECK_LE(max_length, IP_PACKET_SIZE); uint8_t buffer[IP_PACKET_SIZE]; size_t index = 0; if (!Create(buffer, &index, max_length, callback)) return false; return OnBufferFull(buffer, &index, callback); } bool RtcpPacket::OnBufferFull(uint8_t* packet, size_t* index, PacketReadyCallback callback) const { if (*index == 0) return false; RTC_DCHECK(callback) << "Fragmentation not supported."; callback(rtc::ArrayView(packet, *index)); *index = 0; return true; } size_t RtcpPacket::HeaderLength() const { size_t length_in_bytes = BlockLength(); RTC_DCHECK_GT(length_in_bytes, 0); RTC_DCHECK_EQ(length_in_bytes % 4, 0) << "Padding must be handled by each subclass."; // Length in 32-bit words without common header. return (length_in_bytes - kHeaderLength) / 4; } // From RFC 3550, RTP: A Transport Protocol for Real-Time Applications. // // RTP header format. // 0 1 2 3 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ // |V=2|P| RC/FMT | PT | length | // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ void RtcpPacket::CreateHeader( size_t count_or_format, // Depends on packet type. uint8_t packet_type, size_t length, uint8_t* buffer, size_t* pos) { CreateHeader(count_or_format, packet_type, length, /*padding=*/false, buffer, pos); } void RtcpPacket::CreateHeader( size_t count_or_format, // Depends on packet type. uint8_t packet_type, size_t length, bool padding, uint8_t* buffer, size_t* pos) { RTC_DCHECK_LE(length, 0xffffU); RTC_DCHECK_LE(count_or_format, 0x1f); constexpr uint8_t kVersionBits = 2 << 6; uint8_t padding_bit = padding ? 1 << 5 : 0; buffer[*pos + 0] = kVersionBits | padding_bit | static_cast(count_or_format); buffer[*pos + 1] = packet_type; buffer[*pos + 2] = (length >> 8) & 0xff; buffer[*pos + 3] = length & 0xff; *pos += kHeaderLength; } } // namespace rtcp } // namespace webrtc