blob: ff3f11a0cad49bc99ebbc253cda242ab723990a3 [file] [log] [blame]
Benjamin Wright00765292018-11-30 16:18:26 -08001/*
2 * Copyright (c) 2018 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
11#include "video/buffered_frame_decryptor.h"
12
13#include <map>
14#include <memory>
15#include <vector>
16
17#include "absl/memory/memory.h"
18#include "api/test/mock_frame_decryptor.h"
19#include "modules/video_coding/packet_buffer.h"
Steve Anton10542f22019-01-11 09:11:00 -080020#include "rtc_base/ref_counted_object.h"
Benjamin Wright00765292018-11-30 16:18:26 -080021#include "system_wrappers/include/clock.h"
22#include "test/gmock.h"
23#include "test/gtest.h"
24
25using ::testing::Return;
26
27namespace webrtc {
28namespace {
29
30class FakePacketBuffer : public video_coding::PacketBuffer {
31 public:
32 FakePacketBuffer() : PacketBuffer(nullptr, 0, 0, nullptr) {}
33 ~FakePacketBuffer() override {}
34
35 VCMPacket* GetPacket(uint16_t seq_num) override {
36 auto packet_it = packets_.find(seq_num);
37 return packet_it == packets_.end() ? nullptr : &packet_it->second;
38 }
39
40 bool InsertPacket(VCMPacket* packet) override {
41 packets_[packet->seqNum] = *packet;
42 return true;
43 }
44
45 bool GetBitstream(const video_coding::RtpFrameObject& frame,
46 uint8_t* destination) override {
47 return true;
48 }
49
50 void ReturnFrame(video_coding::RtpFrameObject* frame) override {
51 packets_.erase(frame->first_seq_num());
52 }
53
54 private:
55 std::map<uint16_t, VCMPacket> packets_;
56};
57
58} // namespace
59
60class BufferedFrameDecryptorTest
61 : public ::testing::Test,
62 public OnDecryptedFrameCallback,
63 public video_coding::OnReceivedFrameCallback {
64 public:
65 // Implements the OnDecryptedFrameCallbackInterface
66 void OnDecryptedFrame(
67 std::unique_ptr<video_coding::RtpFrameObject> frame) override {
68 decrypted_frame_call_count_++;
69 }
70
71 // Implements the OnReceivedFrameCallback interface.
72 void OnReceivedFrame(
73 std::unique_ptr<video_coding::RtpFrameObject> frame) override {}
74
75 // Returns a new fake RtpFrameObject it abstracts the difficult construction
76 // of the RtpFrameObject to simplify testing.
77 std::unique_ptr<video_coding::RtpFrameObject> CreateRtpFrameObject(
78 bool key_frame) {
79 seq_num_++;
80
81 VCMPacket packet;
Niels Möllerd5e02f02019-02-20 13:12:21 +010082 packet.video_header.codec = kVideoCodecGeneric;
Benjamin Wright00765292018-11-30 16:18:26 -080083 packet.seqNum = seq_num_;
84 packet.frameType = key_frame ? kVideoFrameKey : kVideoFrameDelta;
85 packet.generic_descriptor = RtpGenericFrameDescriptor();
86 fake_packet_buffer_->InsertPacket(&packet);
87 packet.seqNum = seq_num_;
Niels Möllerd5e02f02019-02-20 13:12:21 +010088 packet.video_header.is_last_packet_in_frame = true;
Benjamin Wright00765292018-11-30 16:18:26 -080089 fake_packet_buffer_->InsertPacket(&packet);
90
91 return std::unique_ptr<video_coding::RtpFrameObject>(
92 new video_coding::RtpFrameObject(fake_packet_buffer_.get(), seq_num_,
Ilya Nikolaevskiy4348ce22018-12-07 16:26:56 +010093 seq_num_, 0, 0, 0, 0));
Benjamin Wright00765292018-11-30 16:18:26 -080094 }
95
96 protected:
97 BufferedFrameDecryptorTest() : fake_packet_buffer_(new FakePacketBuffer()) {}
98 void SetUp() override {
99 fake_packet_data_ = std::vector<uint8_t>(100);
100 decrypted_frame_call_count_ = 0;
101 seq_num_ = 0;
102 mock_frame_decryptor_ = new rtc::RefCountedObject<MockFrameDecryptor>();
103 buffered_frame_decryptor_ = absl::make_unique<BufferedFrameDecryptor>(
104 this, mock_frame_decryptor_.get());
105 }
106
107 static const size_t kMaxStashedFrames;
108
109 std::vector<uint8_t> fake_packet_data_;
110 rtc::scoped_refptr<FakePacketBuffer> fake_packet_buffer_;
111 rtc::scoped_refptr<MockFrameDecryptor> mock_frame_decryptor_;
112 std::unique_ptr<BufferedFrameDecryptor> buffered_frame_decryptor_;
113 size_t decrypted_frame_call_count_;
114 uint16_t seq_num_;
115};
116
117const size_t BufferedFrameDecryptorTest::kMaxStashedFrames = 24;
118
119// Callback should always be triggered on a successful decryption.
120TEST_F(BufferedFrameDecryptorTest, CallbackCalledOnSuccessfulDecryption) {
121 EXPECT_CALL(*mock_frame_decryptor_, Decrypt).Times(1).WillOnce(Return(0));
122 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
123 .Times(1)
124 .WillOnce(Return(0));
125 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
126 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(1));
127}
128
129// An initial fail to decrypt should not trigger the callback.
130TEST_F(BufferedFrameDecryptorTest, CallbackNotCalledOnFailedDecryption) {
131 EXPECT_CALL(*mock_frame_decryptor_, Decrypt).Times(1).WillOnce(Return(1));
132 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
133 .Times(1)
134 .WillOnce(Return(0));
135 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
136 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
137}
138
139// Initial failures should be stored and retried after the first successful
140// decryption.
141TEST_F(BufferedFrameDecryptorTest, DelayedCallbackOnBufferedFrames) {
142 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
143 .Times(3)
144 .WillOnce(Return(1))
145 .WillOnce(Return(0))
146 .WillOnce(Return(0));
147 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
148 .Times(3)
149 .WillRepeatedly(Return(0));
150
151 // The first decrypt will fail stashing the first frame.
152 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
153 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
154 // The second call will succeed playing back both frames.
155 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(false));
156 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(2));
157}
158
159// Subsequent failure to decrypts after the first successful decryption should
160// fail to decryptk
161TEST_F(BufferedFrameDecryptorTest, FTDDiscardedAfterFirstSuccess) {
162 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
163 .Times(4)
164 .WillOnce(Return(1))
165 .WillOnce(Return(0))
166 .WillOnce(Return(0))
167 .WillOnce(Return(1));
168 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
169 .Times(4)
170 .WillRepeatedly(Return(0));
171
172 // The first decrypt will fail stashing the first frame.
173 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
174 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
175 // The second call will succeed playing back both frames.
176 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(false));
177 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(2));
178 // A new failure call will not result in an additional decrypted frame
179 // callback.
180 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
181 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(2));
182}
183
184// Validate that the maximum number of stashed frames cannot be exceeded even if
185// more than its maximum arrives before the first successful decryption.
186TEST_F(BufferedFrameDecryptorTest, MaximumNumberOfFramesStored) {
187 const size_t failed_to_decrypt_count = kMaxStashedFrames * 2;
188 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
189 .Times(failed_to_decrypt_count)
190 .WillRepeatedly(Return(1));
191 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
192 .WillRepeatedly(Return(0));
193
194 for (size_t i = 0; i < failed_to_decrypt_count; ++i) {
195 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
196 }
197 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
198
199 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
200 .Times(kMaxStashedFrames + 1)
201 .WillRepeatedly(Return(0));
202 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
203 EXPECT_EQ(decrypted_frame_call_count_, kMaxStashedFrames + 1);
204}
205
206} // namespace webrtc