/* * 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. * * This file includes unit tests for the RtcpPacket. */ #include "testing/gtest/include/gtest/gtest.h" #include "webrtc/modules/rtp_rtcp/source/rtcp_packet.h" #include "webrtc/test/rtcp_packet_parser.h" using webrtc::rtcp::Bye; using webrtc::rtcp::Empty; using webrtc::rtcp::Fir; using webrtc::rtcp::SenderReport; using webrtc::rtcp::RawPacket; using webrtc::rtcp::ReceiverReport; using webrtc::rtcp::ReportBlock; using webrtc::test::RtcpPacketParser; namespace webrtc { const uint32_t kSenderSsrc = 0x12345678; const uint32_t kRemoteSsrc = 0x23456789; TEST(RtcpPacketTest, Rr) { ReceiverReport rr; rr.From(kSenderSsrc); RawPacket packet = rr.Build(); RtcpPacketParser parser; parser.Parse(packet.buffer(), packet.buffer_length()); EXPECT_EQ(1, parser.receiver_report()->num_packets()); EXPECT_EQ(kSenderSsrc, parser.receiver_report()->Ssrc()); EXPECT_EQ(0, parser.report_block()->num_packets()); } TEST(RtcpPacketTest, RrWithOneReportBlock) { ReportBlock rb; rb.To(kRemoteSsrc); rb.WithFractionLost(55); rb.WithCumPacketsLost(0x111111); rb.WithExtHighestSeqNum(0x22222222); rb.WithJitter(0x33333333); rb.WithLastSr(0x44444444); rb.WithDelayLastSr(0x55555555); ReceiverReport rr; rr.From(kSenderSsrc); rr.WithReportBlock(&rb); RawPacket packet = rr.Build(); RtcpPacketParser parser; parser.Parse(packet.buffer(), packet.buffer_length()); EXPECT_EQ(1, parser.receiver_report()->num_packets()); EXPECT_EQ(kSenderSsrc, parser.receiver_report()->Ssrc()); EXPECT_EQ(1, parser.report_block()->num_packets()); EXPECT_EQ(kRemoteSsrc, parser.report_block()->Ssrc()); EXPECT_EQ(55U, parser.report_block()->FractionLost()); EXPECT_EQ(0x111111U, parser.report_block()->CumPacketLost()); EXPECT_EQ(0x22222222U, parser.report_block()->ExtHighestSeqNum()); EXPECT_EQ(0x33333333U, parser.report_block()->Jitter()); EXPECT_EQ(0x44444444U, parser.report_block()->LastSr()); EXPECT_EQ(0x55555555U, parser.report_block()->DelayLastSr()); } TEST(RtcpPacketTest, RrWithTwoReportBlocks) { ReportBlock rb1; rb1.To(kRemoteSsrc); ReportBlock rb2; rb2.To(kRemoteSsrc + 1); ReceiverReport rr; rr.From(kSenderSsrc); rr.WithReportBlock(&rb1); rr.WithReportBlock(&rb2); RawPacket packet = rr.Build(); RtcpPacketParser parser; parser.Parse(packet.buffer(), packet.buffer_length()); EXPECT_EQ(1, parser.receiver_report()->num_packets()); EXPECT_EQ(kSenderSsrc, parser.receiver_report()->Ssrc()); EXPECT_EQ(2, parser.report_block()->num_packets()); EXPECT_EQ(1, parser.report_blocks_per_ssrc(kRemoteSsrc)); EXPECT_EQ(1, parser.report_blocks_per_ssrc(kRemoteSsrc + 1)); } TEST(RtcpPacketTest, Sr) { SenderReport sr; sr.From(kSenderSsrc); sr.WithNtpSec(0x11111111); sr.WithNtpFrac(0x22222222); sr.WithRtpTimestamp(0x33333333); sr.WithPacketCount(0x44444444); sr.WithOctetCount(0x55555555); RawPacket packet = sr.Build(); RtcpPacketParser parser; parser.Parse(packet.buffer(), packet.buffer_length()); EXPECT_EQ(1, parser.sender_report()->num_packets()); EXPECT_EQ(kSenderSsrc, parser.sender_report()->Ssrc()); EXPECT_EQ(0x11111111U, parser.sender_report()->NtpSec()); EXPECT_EQ(0x22222222U, parser.sender_report()->NtpFrac()); EXPECT_EQ(0x33333333U, parser.sender_report()->RtpTimestamp()); EXPECT_EQ(0x44444444U, parser.sender_report()->PacketCount()); EXPECT_EQ(0x55555555U, parser.sender_report()->OctetCount()); EXPECT_EQ(0, parser.report_block()->num_packets()); } TEST(RtcpPacketTest, SrWithOneReportBlock) { ReportBlock rb; rb.To(kRemoteSsrc); SenderReport sr; sr.From(kSenderSsrc); sr.WithReportBlock(&rb); RawPacket packet = sr.Build(); RtcpPacketParser parser; parser.Parse(packet.buffer(), packet.buffer_length()); EXPECT_EQ(1, parser.sender_report()->num_packets()); EXPECT_EQ(kSenderSsrc, parser.sender_report()->Ssrc()); EXPECT_EQ(1, parser.report_block()->num_packets()); EXPECT_EQ(kRemoteSsrc, parser.report_block()->Ssrc()); } TEST(RtcpPacketTest, SrWithTwoReportBlocks) { ReportBlock rb1; rb1.To(kRemoteSsrc); ReportBlock rb2; rb2.To(kRemoteSsrc + 1); SenderReport sr; sr.From(kSenderSsrc); sr.WithReportBlock(&rb1); sr.WithReportBlock(&rb2); RawPacket packet = sr.Build(); RtcpPacketParser parser; parser.Parse(packet.buffer(), packet.buffer_length()); EXPECT_EQ(1, parser.sender_report()->num_packets()); EXPECT_EQ(kSenderSsrc, parser.sender_report()->Ssrc()); EXPECT_EQ(2, parser.report_block()->num_packets()); EXPECT_EQ(1, parser.report_blocks_per_ssrc(kRemoteSsrc)); EXPECT_EQ(1, parser.report_blocks_per_ssrc(kRemoteSsrc + 1)); } TEST(RtcpPacketTest, Fir) { Fir fir; fir.From(kSenderSsrc); fir.To(kRemoteSsrc); fir.WithCommandSeqNum(123); RawPacket packet = fir.Build(); RtcpPacketParser parser; parser.Parse(packet.buffer(), packet.buffer_length()); EXPECT_EQ(1, parser.fir()->num_packets()); EXPECT_EQ(kSenderSsrc, parser.fir()->Ssrc()); EXPECT_EQ(1, parser.fir_item()->num_packets()); EXPECT_EQ(kRemoteSsrc, parser.fir_item()->Ssrc()); EXPECT_EQ(123U, parser.fir_item()->SeqNum()); } TEST(RtcpPacketTest, AppendPacket) { Fir fir; ReportBlock rb; ReceiverReport rr; rr.From(kSenderSsrc); rr.WithReportBlock(&rb); rr.Append(&fir); RawPacket packet = rr.Build(); RtcpPacketParser parser; parser.Parse(packet.buffer(), packet.buffer_length()); EXPECT_EQ(1, parser.receiver_report()->num_packets()); EXPECT_EQ(kSenderSsrc, parser.receiver_report()->Ssrc()); EXPECT_EQ(1, parser.report_block()->num_packets()); EXPECT_EQ(1, parser.fir()->num_packets()); } TEST(RtcpPacketTest, AppendPacketOnEmpty) { Empty empty; ReceiverReport rr; rr.From(kSenderSsrc); empty.Append(&rr); RawPacket packet = empty.Build(); RtcpPacketParser parser; parser.Parse(packet.buffer(), packet.buffer_length()); EXPECT_EQ(1, parser.receiver_report()->num_packets()); EXPECT_EQ(0, parser.report_block()->num_packets()); } TEST(RtcpPacketTest, AppendPacketWithOwnAppendedPacket) { Fir fir; Bye bye; ReportBlock rb; ReceiverReport rr; rr.WithReportBlock(&rb); rr.Append(&fir); SenderReport sr; sr.Append(&bye); sr.Append(&rr); RawPacket packet = sr.Build(); RtcpPacketParser parser; parser.Parse(packet.buffer(), packet.buffer_length()); EXPECT_EQ(1, parser.sender_report()->num_packets()); EXPECT_EQ(1, parser.receiver_report()->num_packets()); EXPECT_EQ(1, parser.report_block()->num_packets()); EXPECT_EQ(1, parser.bye()->num_packets()); EXPECT_EQ(1, parser.fir()->num_packets()); } TEST(RtcpPacketTest, Bye) { Bye bye; bye.From(kSenderSsrc); RawPacket packet = bye.Build(); RtcpPacketParser parser; parser.Parse(packet.buffer(), packet.buffer_length()); EXPECT_EQ(1, parser.bye()->num_packets()); EXPECT_EQ(kSenderSsrc, parser.bye()->Ssrc()); } TEST(RtcpPacketTest, ByeWithCsrcs) { Fir fir; Bye bye; bye.From(kSenderSsrc); bye.WithCsrc(0x22222222); bye.WithCsrc(0x33333333); bye.Append(&fir); RawPacket packet = bye.Build(); RtcpPacketParser parser; parser.Parse(packet.buffer(), packet.buffer_length()); EXPECT_EQ(1, parser.bye()->num_packets()); EXPECT_EQ(kSenderSsrc, parser.bye()->Ssrc()); EXPECT_EQ(1, parser.fir()->num_packets()); } TEST(RtcpPacketTest, BuildWithInputBuffer) { Fir fir; ReportBlock rb; ReceiverReport rr; rr.From(kSenderSsrc); rr.WithReportBlock(&rb); rr.Append(&fir); const uint16_t kRrLength = 8; const uint16_t kReportBlockLength = 24; const uint16_t kFirLength = 20; uint16_t len = 0; uint8_t packet[kRrLength + kReportBlockLength + kFirLength]; rr.Build(packet, &len, kRrLength + kReportBlockLength + kFirLength); RtcpPacketParser parser; parser.Parse(packet, len); EXPECT_EQ(1, parser.receiver_report()->num_packets()); EXPECT_EQ(1, parser.report_block()->num_packets()); EXPECT_EQ(1, parser.fir()->num_packets()); } TEST(RtcpPacketTest, BuildWithTooSmallBuffer) { ReportBlock rb; ReceiverReport rr; rr.From(kSenderSsrc); rr.WithReportBlock(&rb); const uint16_t kRrLength = 8; const uint16_t kReportBlockLength = 24; // No packet. uint16_t len = 0; uint8_t packet[kRrLength + kReportBlockLength - 1]; rr.Build(packet, &len, kRrLength + kReportBlockLength - 1); RtcpPacketParser parser; parser.Parse(packet, len); EXPECT_EQ(0, len); } TEST(RtcpPacketTest, BuildWithTooSmallBuffer_LastBlockFits) { Fir fir; ReportBlock rb; ReceiverReport rr; rr.From(kSenderSsrc); rr.WithReportBlock(&rb); rr.Append(&fir); const uint16_t kRrLength = 8; const uint16_t kReportBlockLength = 24; uint16_t len = 0; uint8_t packet[kRrLength + kReportBlockLength - 1]; rr.Build(packet, &len, kRrLength + kReportBlockLength - 1); RtcpPacketParser parser; parser.Parse(packet, len); EXPECT_EQ(0, parser.receiver_report()->num_packets()); EXPECT_EQ(0, parser.report_block()->num_packets()); EXPECT_EQ(1, parser.fir()->num_packets()); } } // namespace webrtc