blob: 7dcf712c9e5061e42bb39379956febc17aeb830f [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
Benjamin Wright00765292018-11-30 16:18:26 -080017#include "api/test/mock_frame_decryptor.h"
18#include "modules/video_coding/packet_buffer.h"
Steve Anton10542f22019-01-11 09:11:00 -080019#include "rtc_base/ref_counted_object.h"
Benjamin Wright00765292018-11-30 16:18:26 -080020#include "system_wrappers/include/clock.h"
21#include "test/gmock.h"
22#include "test/gtest.h"
23
24using ::testing::Return;
25
26namespace webrtc {
27namespace {
28
29class FakePacketBuffer : public video_coding::PacketBuffer {
30 public:
31 FakePacketBuffer() : PacketBuffer(nullptr, 0, 0, nullptr) {}
32 ~FakePacketBuffer() override {}
33
34 VCMPacket* GetPacket(uint16_t seq_num) override {
35 auto packet_it = packets_.find(seq_num);
36 return packet_it == packets_.end() ? nullptr : &packet_it->second;
37 }
38
39 bool InsertPacket(VCMPacket* packet) override {
40 packets_[packet->seqNum] = *packet;
41 return true;
42 }
43
Benjamin Wright00765292018-11-30 16:18:26 -080044 private:
45 std::map<uint16_t, VCMPacket> packets_;
46};
47
Benjamin Wright2af5dcb2019-04-09 20:08:41 +000048FrameDecryptorInterface::Result DecryptSuccess() {
49 return FrameDecryptorInterface::Result(FrameDecryptorInterface::Status::kOk,
50 0);
51}
52
53FrameDecryptorInterface::Result DecryptFail() {
54 return FrameDecryptorInterface::Result(
55 FrameDecryptorInterface::Status::kFailedToDecrypt, 0);
56}
57
Benjamin Wright00765292018-11-30 16:18:26 -080058} // namespace
59
60class BufferedFrameDecryptorTest
61 : public ::testing::Test,
62 public OnDecryptedFrameCallback,
Benjamin Wright52426ed2019-03-01 11:01:59 -080063 public OnDecryptionStatusChangeCallback,
Elad Alonb4643ad2019-02-22 11:19:50 +010064 public video_coding::OnAssembledFrameCallback {
Benjamin Wright00765292018-11-30 16:18:26 -080065 public:
66 // Implements the OnDecryptedFrameCallbackInterface
67 void OnDecryptedFrame(
68 std::unique_ptr<video_coding::RtpFrameObject> frame) override {
69 decrypted_frame_call_count_++;
70 }
71
Benjamin Wright2af5dcb2019-04-09 20:08:41 +000072 void OnDecryptionStatusChange(FrameDecryptorInterface::Status status) {
Benjamin Wright52426ed2019-03-01 11:01:59 -080073 ++decryption_status_change_count_;
74 }
75
Elad Alonb4643ad2019-02-22 11:19:50 +010076 // Implements the OnAssembledFrameCallback interface.
77 void OnAssembledFrame(
Benjamin Wright00765292018-11-30 16:18:26 -080078 std::unique_ptr<video_coding::RtpFrameObject> frame) override {}
79
80 // Returns a new fake RtpFrameObject it abstracts the difficult construction
81 // of the RtpFrameObject to simplify testing.
82 std::unique_ptr<video_coding::RtpFrameObject> CreateRtpFrameObject(
83 bool key_frame) {
84 seq_num_++;
85
86 VCMPacket packet;
Niels Möllerd5e02f02019-02-20 13:12:21 +010087 packet.video_header.codec = kVideoCodecGeneric;
Benjamin Wright00765292018-11-30 16:18:26 -080088 packet.seqNum = seq_num_;
Niels Möllerabbc50e2019-04-24 09:41:16 +020089 packet.video_header.frame_type = key_frame
90 ? VideoFrameType::kVideoFrameKey
91 : VideoFrameType::kVideoFrameDelta;
Benjamin Wright00765292018-11-30 16:18:26 -080092 packet.generic_descriptor = RtpGenericFrameDescriptor();
Danil Chapovalovf7457e52019-09-20 17:57:15 +020093 fake_packet_buffer_.InsertPacket(&packet);
Benjamin Wright00765292018-11-30 16:18:26 -080094 packet.seqNum = seq_num_;
Niels Möllerd5e02f02019-02-20 13:12:21 +010095 packet.video_header.is_last_packet_in_frame = true;
Danil Chapovalovf7457e52019-09-20 17:57:15 +020096 fake_packet_buffer_.InsertPacket(&packet);
Benjamin Wright00765292018-11-30 16:18:26 -080097
Danil Chapovalovf7457e52019-09-20 17:57:15 +020098 return std::make_unique<video_coding::RtpFrameObject>(
99 &fake_packet_buffer_, seq_num_, seq_num_, 0, 0, 0, RtpPacketInfos(),
100 EncodedImageBuffer::Create(/*size=*/0));
Benjamin Wright00765292018-11-30 16:18:26 -0800101 }
102
103 protected:
Danil Chapovalovf7457e52019-09-20 17:57:15 +0200104 BufferedFrameDecryptorTest() {
Benjamin Wright00765292018-11-30 16:18:26 -0800105 fake_packet_data_ = std::vector<uint8_t>(100);
106 decrypted_frame_call_count_ = 0;
Benjamin Wright52426ed2019-03-01 11:01:59 -0800107 decryption_status_change_count_ = 0;
Benjamin Wright00765292018-11-30 16:18:26 -0800108 seq_num_ = 0;
109 mock_frame_decryptor_ = new rtc::RefCountedObject<MockFrameDecryptor>();
Benjamin Wrighta5564482019-04-03 10:44:18 -0700110 buffered_frame_decryptor_ =
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200111 std::make_unique<BufferedFrameDecryptor>(this, this);
Benjamin Wrighta5564482019-04-03 10:44:18 -0700112 buffered_frame_decryptor_->SetFrameDecryptor(mock_frame_decryptor_.get());
Benjamin Wright00765292018-11-30 16:18:26 -0800113 }
114
115 static const size_t kMaxStashedFrames;
116
117 std::vector<uint8_t> fake_packet_data_;
Danil Chapovalovf7457e52019-09-20 17:57:15 +0200118 FakePacketBuffer fake_packet_buffer_;
Benjamin Wright00765292018-11-30 16:18:26 -0800119 rtc::scoped_refptr<MockFrameDecryptor> mock_frame_decryptor_;
120 std::unique_ptr<BufferedFrameDecryptor> buffered_frame_decryptor_;
121 size_t decrypted_frame_call_count_;
Benjamin Wright52426ed2019-03-01 11:01:59 -0800122 size_t decryption_status_change_count_ = 0;
Benjamin Wright00765292018-11-30 16:18:26 -0800123 uint16_t seq_num_;
124};
125
126const size_t BufferedFrameDecryptorTest::kMaxStashedFrames = 24;
127
128// Callback should always be triggered on a successful decryption.
129TEST_F(BufferedFrameDecryptorTest, CallbackCalledOnSuccessfulDecryption) {
Benjamin Wright2af5dcb2019-04-09 20:08:41 +0000130 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
131 .Times(1)
132 .WillOnce(Return(DecryptSuccess()));
Benjamin Wright00765292018-11-30 16:18:26 -0800133 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
134 .Times(1)
135 .WillOnce(Return(0));
136 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
137 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(1));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800138 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800139}
140
141// An initial fail to decrypt should not trigger the callback.
142TEST_F(BufferedFrameDecryptorTest, CallbackNotCalledOnFailedDecryption) {
Benjamin Wright2af5dcb2019-04-09 20:08:41 +0000143 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
144 .Times(1)
145 .WillOnce(Return(DecryptFail()));
Benjamin Wright00765292018-11-30 16:18:26 -0800146 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
147 .Times(1)
148 .WillOnce(Return(0));
149 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
150 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800151 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800152}
153
154// Initial failures should be stored and retried after the first successful
155// decryption.
156TEST_F(BufferedFrameDecryptorTest, DelayedCallbackOnBufferedFrames) {
157 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
158 .Times(3)
Benjamin Wright2af5dcb2019-04-09 20:08:41 +0000159 .WillOnce(Return(DecryptFail()))
160 .WillOnce(Return(DecryptSuccess()))
161 .WillOnce(Return(DecryptSuccess()));
Benjamin Wright00765292018-11-30 16:18:26 -0800162 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
163 .Times(3)
164 .WillRepeatedly(Return(0));
165
166 // The first decrypt will fail stashing the first frame.
167 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
168 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800169 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800170 // The second call will succeed playing back both frames.
171 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(false));
172 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(2));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800173 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(2));
Benjamin Wright00765292018-11-30 16:18:26 -0800174}
175
176// Subsequent failure to decrypts after the first successful decryption should
177// fail to decryptk
178TEST_F(BufferedFrameDecryptorTest, FTDDiscardedAfterFirstSuccess) {
179 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
180 .Times(4)
Benjamin Wright2af5dcb2019-04-09 20:08:41 +0000181 .WillOnce(Return(DecryptFail()))
182 .WillOnce(Return(DecryptSuccess()))
183 .WillOnce(Return(DecryptSuccess()))
184 .WillOnce(Return(DecryptFail()));
Benjamin Wright00765292018-11-30 16:18:26 -0800185 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
186 .Times(4)
187 .WillRepeatedly(Return(0));
188
189 // The first decrypt will fail stashing the first frame.
190 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
191 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800192 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800193 // The second call will succeed playing back both frames.
194 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(false));
195 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(2));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800196 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(2));
Benjamin Wright00765292018-11-30 16:18:26 -0800197 // A new failure call will not result in an additional decrypted frame
198 // callback.
199 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
200 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(2));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800201 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(3));
Benjamin Wright00765292018-11-30 16:18:26 -0800202}
203
204// Validate that the maximum number of stashed frames cannot be exceeded even if
205// more than its maximum arrives before the first successful decryption.
206TEST_F(BufferedFrameDecryptorTest, MaximumNumberOfFramesStored) {
207 const size_t failed_to_decrypt_count = kMaxStashedFrames * 2;
208 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
209 .Times(failed_to_decrypt_count)
Benjamin Wright2af5dcb2019-04-09 20:08:41 +0000210 .WillRepeatedly(Return(DecryptFail()));
Benjamin Wright00765292018-11-30 16:18:26 -0800211 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
212 .WillRepeatedly(Return(0));
213
214 for (size_t i = 0; i < failed_to_decrypt_count; ++i) {
215 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
216 }
217 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800218 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800219
220 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
221 .Times(kMaxStashedFrames + 1)
Benjamin Wright2af5dcb2019-04-09 20:08:41 +0000222 .WillRepeatedly(Return(DecryptSuccess()));
Benjamin Wright00765292018-11-30 16:18:26 -0800223 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
224 EXPECT_EQ(decrypted_frame_call_count_, kMaxStashedFrames + 1);
Benjamin Wright52426ed2019-03-01 11:01:59 -0800225 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(2));
Benjamin Wright00765292018-11-30 16:18:26 -0800226}
227
Benjamin Wrighta5564482019-04-03 10:44:18 -0700228// Verifies if a BufferedFrameDecryptor is attached but has no FrameDecryptor
229// attached it will still store frames up to the frame max.
230TEST_F(BufferedFrameDecryptorTest, FramesStoredIfDecryptorNull) {
231 buffered_frame_decryptor_->SetFrameDecryptor(nullptr);
232 for (size_t i = 0; i < (2 * kMaxStashedFrames); ++i) {
233 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
234 }
235
236 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
237 .Times(kMaxStashedFrames + 1)
Benjamin Wright2af5dcb2019-04-09 20:08:41 +0000238 .WillRepeatedly(Return(DecryptSuccess()));
Benjamin Wrighta5564482019-04-03 10:44:18 -0700239 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
240 .WillRepeatedly(Return(0));
241
242 // Attach the frame decryptor at a later point after frames have arrived.
243 buffered_frame_decryptor_->SetFrameDecryptor(mock_frame_decryptor_.get());
244
245 // Next frame should trigger kMaxStashedFrame decryptions.
246 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
247 EXPECT_EQ(decrypted_frame_call_count_, kMaxStashedFrames + 1);
248}
249
Benjamin Wright00765292018-11-30 16:18:26 -0800250} // namespace webrtc