blob: 4506dcf3566c98da6c8fc4c845676b12a0f1bd59 [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
Benjamin Wright2af5dcb2019-04-09 20:08:41 +000058FrameDecryptorInterface::Result DecryptSuccess() {
59 return FrameDecryptorInterface::Result(FrameDecryptorInterface::Status::kOk,
60 0);
61}
62
63FrameDecryptorInterface::Result DecryptFail() {
64 return FrameDecryptorInterface::Result(
65 FrameDecryptorInterface::Status::kFailedToDecrypt, 0);
66}
67
Benjamin Wright00765292018-11-30 16:18:26 -080068} // namespace
69
70class BufferedFrameDecryptorTest
71 : public ::testing::Test,
72 public OnDecryptedFrameCallback,
Benjamin Wright52426ed2019-03-01 11:01:59 -080073 public OnDecryptionStatusChangeCallback,
Elad Alonb4643ad2019-02-22 11:19:50 +010074 public video_coding::OnAssembledFrameCallback {
Benjamin Wright00765292018-11-30 16:18:26 -080075 public:
76 // Implements the OnDecryptedFrameCallbackInterface
77 void OnDecryptedFrame(
78 std::unique_ptr<video_coding::RtpFrameObject> frame) override {
79 decrypted_frame_call_count_++;
80 }
81
Benjamin Wright2af5dcb2019-04-09 20:08:41 +000082 void OnDecryptionStatusChange(FrameDecryptorInterface::Status status) {
Benjamin Wright52426ed2019-03-01 11:01:59 -080083 ++decryption_status_change_count_;
84 }
85
Elad Alonb4643ad2019-02-22 11:19:50 +010086 // Implements the OnAssembledFrameCallback interface.
87 void OnAssembledFrame(
Benjamin Wright00765292018-11-30 16:18:26 -080088 std::unique_ptr<video_coding::RtpFrameObject> frame) override {}
89
90 // Returns a new fake RtpFrameObject it abstracts the difficult construction
91 // of the RtpFrameObject to simplify testing.
92 std::unique_ptr<video_coding::RtpFrameObject> CreateRtpFrameObject(
93 bool key_frame) {
94 seq_num_++;
95
96 VCMPacket packet;
Niels Möllerd5e02f02019-02-20 13:12:21 +010097 packet.video_header.codec = kVideoCodecGeneric;
Benjamin Wright00765292018-11-30 16:18:26 -080098 packet.seqNum = seq_num_;
Niels Möllerabbc50e2019-04-24 09:41:16 +020099 packet.video_header.frame_type = key_frame
100 ? VideoFrameType::kVideoFrameKey
101 : VideoFrameType::kVideoFrameDelta;
Benjamin Wright00765292018-11-30 16:18:26 -0800102 packet.generic_descriptor = RtpGenericFrameDescriptor();
103 fake_packet_buffer_->InsertPacket(&packet);
104 packet.seqNum = seq_num_;
Niels Möllerd5e02f02019-02-20 13:12:21 +0100105 packet.video_header.is_last_packet_in_frame = true;
Benjamin Wright00765292018-11-30 16:18:26 -0800106 fake_packet_buffer_->InsertPacket(&packet);
107
108 return std::unique_ptr<video_coding::RtpFrameObject>(
109 new video_coding::RtpFrameObject(fake_packet_buffer_.get(), seq_num_,
Ilya Nikolaevskiy4348ce22018-12-07 16:26:56 +0100110 seq_num_, 0, 0, 0, 0));
Benjamin Wright00765292018-11-30 16:18:26 -0800111 }
112
113 protected:
114 BufferedFrameDecryptorTest() : fake_packet_buffer_(new FakePacketBuffer()) {}
115 void SetUp() override {
116 fake_packet_data_ = std::vector<uint8_t>(100);
117 decrypted_frame_call_count_ = 0;
Benjamin Wright52426ed2019-03-01 11:01:59 -0800118 decryption_status_change_count_ = 0;
Benjamin Wright00765292018-11-30 16:18:26 -0800119 seq_num_ = 0;
120 mock_frame_decryptor_ = new rtc::RefCountedObject<MockFrameDecryptor>();
Benjamin Wrighta5564482019-04-03 10:44:18 -0700121 buffered_frame_decryptor_ =
122 absl::make_unique<BufferedFrameDecryptor>(this, this);
123 buffered_frame_decryptor_->SetFrameDecryptor(mock_frame_decryptor_.get());
Benjamin Wright00765292018-11-30 16:18:26 -0800124 }
125
126 static const size_t kMaxStashedFrames;
127
128 std::vector<uint8_t> fake_packet_data_;
129 rtc::scoped_refptr<FakePacketBuffer> fake_packet_buffer_;
130 rtc::scoped_refptr<MockFrameDecryptor> mock_frame_decryptor_;
131 std::unique_ptr<BufferedFrameDecryptor> buffered_frame_decryptor_;
132 size_t decrypted_frame_call_count_;
Benjamin Wright52426ed2019-03-01 11:01:59 -0800133 size_t decryption_status_change_count_ = 0;
Benjamin Wright00765292018-11-30 16:18:26 -0800134 uint16_t seq_num_;
135};
136
137const size_t BufferedFrameDecryptorTest::kMaxStashedFrames = 24;
138
139// Callback should always be triggered on a successful decryption.
140TEST_F(BufferedFrameDecryptorTest, CallbackCalledOnSuccessfulDecryption) {
Benjamin Wright2af5dcb2019-04-09 20:08:41 +0000141 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
142 .Times(1)
143 .WillOnce(Return(DecryptSuccess()));
Benjamin Wright00765292018-11-30 16:18:26 -0800144 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
145 .Times(1)
146 .WillOnce(Return(0));
147 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
148 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(1));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800149 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800150}
151
152// An initial fail to decrypt should not trigger the callback.
153TEST_F(BufferedFrameDecryptorTest, CallbackNotCalledOnFailedDecryption) {
Benjamin Wright2af5dcb2019-04-09 20:08:41 +0000154 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
155 .Times(1)
156 .WillOnce(Return(DecryptFail()));
Benjamin Wright00765292018-11-30 16:18:26 -0800157 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
158 .Times(1)
159 .WillOnce(Return(0));
160 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
161 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800162 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800163}
164
165// Initial failures should be stored and retried after the first successful
166// decryption.
167TEST_F(BufferedFrameDecryptorTest, DelayedCallbackOnBufferedFrames) {
168 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
169 .Times(3)
Benjamin Wright2af5dcb2019-04-09 20:08:41 +0000170 .WillOnce(Return(DecryptFail()))
171 .WillOnce(Return(DecryptSuccess()))
172 .WillOnce(Return(DecryptSuccess()));
Benjamin Wright00765292018-11-30 16:18:26 -0800173 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
174 .Times(3)
175 .WillRepeatedly(Return(0));
176
177 // The first decrypt will fail stashing the first frame.
178 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
179 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800180 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800181 // The second call will succeed playing back both frames.
182 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(false));
183 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(2));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800184 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(2));
Benjamin Wright00765292018-11-30 16:18:26 -0800185}
186
187// Subsequent failure to decrypts after the first successful decryption should
188// fail to decryptk
189TEST_F(BufferedFrameDecryptorTest, FTDDiscardedAfterFirstSuccess) {
190 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
191 .Times(4)
Benjamin Wright2af5dcb2019-04-09 20:08:41 +0000192 .WillOnce(Return(DecryptFail()))
193 .WillOnce(Return(DecryptSuccess()))
194 .WillOnce(Return(DecryptSuccess()))
195 .WillOnce(Return(DecryptFail()));
Benjamin Wright00765292018-11-30 16:18:26 -0800196 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
197 .Times(4)
198 .WillRepeatedly(Return(0));
199
200 // The first decrypt will fail stashing the first frame.
201 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
202 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800203 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800204 // The second call will succeed playing back both frames.
205 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(false));
206 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(2));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800207 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(2));
Benjamin Wright00765292018-11-30 16:18:26 -0800208 // A new failure call will not result in an additional decrypted frame
209 // callback.
210 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
211 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(2));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800212 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(3));
Benjamin Wright00765292018-11-30 16:18:26 -0800213}
214
215// Validate that the maximum number of stashed frames cannot be exceeded even if
216// more than its maximum arrives before the first successful decryption.
217TEST_F(BufferedFrameDecryptorTest, MaximumNumberOfFramesStored) {
218 const size_t failed_to_decrypt_count = kMaxStashedFrames * 2;
219 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
220 .Times(failed_to_decrypt_count)
Benjamin Wright2af5dcb2019-04-09 20:08:41 +0000221 .WillRepeatedly(Return(DecryptFail()));
Benjamin Wright00765292018-11-30 16:18:26 -0800222 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
223 .WillRepeatedly(Return(0));
224
225 for (size_t i = 0; i < failed_to_decrypt_count; ++i) {
226 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
227 }
228 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800229 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800230
231 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
232 .Times(kMaxStashedFrames + 1)
Benjamin Wright2af5dcb2019-04-09 20:08:41 +0000233 .WillRepeatedly(Return(DecryptSuccess()));
Benjamin Wright00765292018-11-30 16:18:26 -0800234 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
235 EXPECT_EQ(decrypted_frame_call_count_, kMaxStashedFrames + 1);
Benjamin Wright52426ed2019-03-01 11:01:59 -0800236 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(2));
Benjamin Wright00765292018-11-30 16:18:26 -0800237}
238
Benjamin Wrighta5564482019-04-03 10:44:18 -0700239// Verifies if a BufferedFrameDecryptor is attached but has no FrameDecryptor
240// attached it will still store frames up to the frame max.
241TEST_F(BufferedFrameDecryptorTest, FramesStoredIfDecryptorNull) {
242 buffered_frame_decryptor_->SetFrameDecryptor(nullptr);
243 for (size_t i = 0; i < (2 * kMaxStashedFrames); ++i) {
244 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
245 }
246
247 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
248 .Times(kMaxStashedFrames + 1)
Benjamin Wright2af5dcb2019-04-09 20:08:41 +0000249 .WillRepeatedly(Return(DecryptSuccess()));
Benjamin Wrighta5564482019-04-03 10:44:18 -0700250 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
251 .WillRepeatedly(Return(0));
252
253 // Attach the frame decryptor at a later point after frames have arrived.
254 buffered_frame_decryptor_->SetFrameDecryptor(mock_frame_decryptor_.get());
255
256 // Next frame should trigger kMaxStashedFrame decryptions.
257 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
258 EXPECT_EQ(decrypted_frame_call_count_, kMaxStashedFrames + 1);
259}
260
Benjamin Wright00765292018-11-30 16:18:26 -0800261} // namespace webrtc