blob: 7926f2421eae79611c67e609ded0dffeaf483100 [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,
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
Henrik Boström7dd83e22019-04-09 10:36:31 +000072 void OnDecryptionStatusChange(int 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öller8f7ce222019-03-21 15:43:58 +010089 packet.frameType = key_frame ? VideoFrameType::kVideoFrameKey
90 : VideoFrameType::kVideoFrameDelta;
Benjamin Wright00765292018-11-30 16:18:26 -080091 packet.generic_descriptor = RtpGenericFrameDescriptor();
92 fake_packet_buffer_->InsertPacket(&packet);
93 packet.seqNum = seq_num_;
Niels Möllerd5e02f02019-02-20 13:12:21 +010094 packet.video_header.is_last_packet_in_frame = true;
Benjamin Wright00765292018-11-30 16:18:26 -080095 fake_packet_buffer_->InsertPacket(&packet);
96
97 return std::unique_ptr<video_coding::RtpFrameObject>(
98 new video_coding::RtpFrameObject(fake_packet_buffer_.get(), seq_num_,
Ilya Nikolaevskiy4348ce22018-12-07 16:26:56 +010099 seq_num_, 0, 0, 0, 0));
Benjamin Wright00765292018-11-30 16:18:26 -0800100 }
101
102 protected:
103 BufferedFrameDecryptorTest() : fake_packet_buffer_(new FakePacketBuffer()) {}
104 void SetUp() override {
105 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_ =
111 absl::make_unique<BufferedFrameDecryptor>(this, this);
112 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_;
118 rtc::scoped_refptr<FakePacketBuffer> fake_packet_buffer_;
119 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) {
Henrik Boström7dd83e22019-04-09 10:36:31 +0000130 EXPECT_CALL(*mock_frame_decryptor_, Decrypt).Times(1).WillOnce(Return(0));
Benjamin Wright00765292018-11-30 16:18:26 -0800131 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
132 .Times(1)
133 .WillOnce(Return(0));
134 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
135 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(1));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800136 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800137}
138
139// An initial fail to decrypt should not trigger the callback.
140TEST_F(BufferedFrameDecryptorTest, CallbackNotCalledOnFailedDecryption) {
Henrik Boström7dd83e22019-04-09 10:36:31 +0000141 EXPECT_CALL(*mock_frame_decryptor_, Decrypt).Times(1).WillOnce(Return(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800142 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
143 .Times(1)
144 .WillOnce(Return(0));
145 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
146 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800147 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800148}
149
150// Initial failures should be stored and retried after the first successful
151// decryption.
152TEST_F(BufferedFrameDecryptorTest, DelayedCallbackOnBufferedFrames) {
153 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
154 .Times(3)
Henrik Boström7dd83e22019-04-09 10:36:31 +0000155 .WillOnce(Return(1))
156 .WillOnce(Return(0))
157 .WillOnce(Return(0));
Benjamin Wright00765292018-11-30 16:18:26 -0800158 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
159 .Times(3)
160 .WillRepeatedly(Return(0));
161
162 // The first decrypt will fail stashing the first frame.
163 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
164 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800165 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800166 // The second call will succeed playing back both frames.
167 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(false));
168 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(2));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800169 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(2));
Benjamin Wright00765292018-11-30 16:18:26 -0800170}
171
172// Subsequent failure to decrypts after the first successful decryption should
173// fail to decryptk
174TEST_F(BufferedFrameDecryptorTest, FTDDiscardedAfterFirstSuccess) {
175 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
176 .Times(4)
Henrik Boström7dd83e22019-04-09 10:36:31 +0000177 .WillOnce(Return(1))
178 .WillOnce(Return(0))
179 .WillOnce(Return(0))
180 .WillOnce(Return(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800181 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
182 .Times(4)
183 .WillRepeatedly(Return(0));
184
185 // The first decrypt will fail stashing the first frame.
186 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
187 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800188 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800189 // The second call will succeed playing back both frames.
190 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(false));
191 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(2));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800192 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(2));
Benjamin Wright00765292018-11-30 16:18:26 -0800193 // A new failure call will not result in an additional decrypted frame
194 // callback.
195 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
196 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(2));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800197 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(3));
Benjamin Wright00765292018-11-30 16:18:26 -0800198}
199
200// Validate that the maximum number of stashed frames cannot be exceeded even if
201// more than its maximum arrives before the first successful decryption.
202TEST_F(BufferedFrameDecryptorTest, MaximumNumberOfFramesStored) {
203 const size_t failed_to_decrypt_count = kMaxStashedFrames * 2;
204 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
205 .Times(failed_to_decrypt_count)
Henrik Boström7dd83e22019-04-09 10:36:31 +0000206 .WillRepeatedly(Return(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800207 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
208 .WillRepeatedly(Return(0));
209
210 for (size_t i = 0; i < failed_to_decrypt_count; ++i) {
211 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
212 }
213 EXPECT_EQ(decrypted_frame_call_count_, static_cast<size_t>(0));
Benjamin Wright52426ed2019-03-01 11:01:59 -0800214 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(1));
Benjamin Wright00765292018-11-30 16:18:26 -0800215
216 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
217 .Times(kMaxStashedFrames + 1)
Henrik Boström7dd83e22019-04-09 10:36:31 +0000218 .WillRepeatedly(Return(0));
Benjamin Wright00765292018-11-30 16:18:26 -0800219 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
220 EXPECT_EQ(decrypted_frame_call_count_, kMaxStashedFrames + 1);
Benjamin Wright52426ed2019-03-01 11:01:59 -0800221 EXPECT_EQ(decryption_status_change_count_, static_cast<size_t>(2));
Benjamin Wright00765292018-11-30 16:18:26 -0800222}
223
Benjamin Wrighta5564482019-04-03 10:44:18 -0700224// Verifies if a BufferedFrameDecryptor is attached but has no FrameDecryptor
225// attached it will still store frames up to the frame max.
226TEST_F(BufferedFrameDecryptorTest, FramesStoredIfDecryptorNull) {
227 buffered_frame_decryptor_->SetFrameDecryptor(nullptr);
228 for (size_t i = 0; i < (2 * kMaxStashedFrames); ++i) {
229 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
230 }
231
232 EXPECT_CALL(*mock_frame_decryptor_, Decrypt)
233 .Times(kMaxStashedFrames + 1)
Henrik Boström7dd83e22019-04-09 10:36:31 +0000234 .WillRepeatedly(Return(0));
Benjamin Wrighta5564482019-04-03 10:44:18 -0700235 EXPECT_CALL(*mock_frame_decryptor_, GetMaxPlaintextByteSize)
236 .WillRepeatedly(Return(0));
237
238 // Attach the frame decryptor at a later point after frames have arrived.
239 buffered_frame_decryptor_->SetFrameDecryptor(mock_frame_decryptor_.get());
240
241 // Next frame should trigger kMaxStashedFrame decryptions.
242 buffered_frame_decryptor_->ManageEncryptedFrame(CreateRtpFrameObject(true));
243 EXPECT_EQ(decrypted_frame_call_count_, kMaxStashedFrames + 1);
244}
245
Benjamin Wright00765292018-11-30 16:18:26 -0800246} // namespace webrtc