mirror of
https://github.com/mollyim/webrtc.git
synced 2025-05-12 21:30:45 +01:00

This CL updates H26xPacketBuffer to store and prepend SPS and PPS for H.264 bitstreams when IDR only keyframe is allowed. Bug: webrtc:13485 Change-Id: Ic1edc623dff568d54d3ce29b42dd8eab3312f5cb Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/342225 Reviewed-by: Philip Eliasson <philipel@webrtc.org> Commit-Queue: Philip Eliasson <philipel@webrtc.org> Reviewed-by: Sergey Silkin <ssilkin@webrtc.org> Cr-Commit-Position: refs/heads/main@{#41986}
1198 lines
42 KiB
C++
1198 lines
42 KiB
C++
/*
|
|
* Copyright (c) 2021 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.
|
|
*/
|
|
#include "modules/video_coding/h26x_packet_buffer.h"
|
|
|
|
#include <cstring>
|
|
#include <limits>
|
|
#include <ostream>
|
|
#include <string>
|
|
#include <utility>
|
|
|
|
#include "api/array_view.h"
|
|
#include "api/video/render_resolution.h"
|
|
#include "common_video/h264/h264_common.h"
|
|
#include "rtc_base/system/unused.h"
|
|
#include "test/gmock.h"
|
|
#include "test/gtest.h"
|
|
#ifdef RTC_ENABLE_H265
|
|
#include "common_video/h265/h265_common.h"
|
|
#endif
|
|
|
|
namespace webrtc {
|
|
namespace {
|
|
|
|
using ::testing::ElementsAreArray;
|
|
using ::testing::Eq;
|
|
using ::testing::IsEmpty;
|
|
using ::testing::SizeIs;
|
|
|
|
using H264::NaluType::kAud;
|
|
using H264::NaluType::kFuA;
|
|
using H264::NaluType::kIdr;
|
|
using H264::NaluType::kPps;
|
|
using H264::NaluType::kSlice;
|
|
using H264::NaluType::kSps;
|
|
using H264::NaluType::kStapA;
|
|
|
|
constexpr int kBufferSize = 2048;
|
|
// Example sprop string from https://tools.ietf.org/html/rfc3984.
|
|
const char kExampleSpropString[] = "Z0IACpZTBYmI,aMljiA==";
|
|
static const std::vector<uint8_t> kExampleSpropRawSps{
|
|
0x67, 0x42, 0x00, 0x0A, 0x96, 0x53, 0x05, 0x89, 0x88};
|
|
static const std::vector<uint8_t> kExampleSpropRawPps{0x68, 0xC9, 0x63, 0x88};
|
|
|
|
std::vector<uint8_t> StartCode() {
|
|
return {0, 0, 0, 1};
|
|
}
|
|
|
|
NaluInfo MakeNaluInfo(uint8_t type) {
|
|
NaluInfo res;
|
|
res.type = type;
|
|
res.sps_id = -1;
|
|
res.pps_id = -1;
|
|
return res;
|
|
}
|
|
|
|
class H264Packet {
|
|
public:
|
|
explicit H264Packet(H264PacketizationTypes type);
|
|
|
|
H264Packet& Idr(std::vector<uint8_t> payload = {9, 9, 9}, int pps_id = -1);
|
|
H264Packet& Slice(std::vector<uint8_t> payload = {9, 9, 9});
|
|
H264Packet& Sps(std::vector<uint8_t> payload = {9, 9, 9}, int sps_id = -1);
|
|
H264Packet& SpsWithResolution(RenderResolution resolution,
|
|
std::vector<uint8_t> payload = {9, 9, 9});
|
|
H264Packet& Pps(std::vector<uint8_t> payload = {9, 9, 9},
|
|
int pps_id = -1,
|
|
int sps_id = -1);
|
|
H264Packet& Aud();
|
|
H264Packet& Marker();
|
|
H264Packet& AsFirstFragment();
|
|
H264Packet& Time(uint32_t rtp_timestamp);
|
|
H264Packet& SeqNum(uint16_t rtp_seq_num);
|
|
|
|
std::unique_ptr<H26xPacketBuffer::Packet> Build();
|
|
|
|
private:
|
|
rtc::CopyOnWriteBuffer BuildFuaPayload() const;
|
|
rtc::CopyOnWriteBuffer BuildSingleNaluPayload() const;
|
|
rtc::CopyOnWriteBuffer BuildStapAPayload() const;
|
|
|
|
RTPVideoHeaderH264& H264Header() {
|
|
return absl::get<RTPVideoHeaderH264>(video_header_.video_type_header);
|
|
}
|
|
const RTPVideoHeaderH264& H264Header() const {
|
|
return absl::get<RTPVideoHeaderH264>(video_header_.video_type_header);
|
|
}
|
|
|
|
H264PacketizationTypes type_;
|
|
RTPVideoHeader video_header_;
|
|
bool first_fragment_ = false;
|
|
bool marker_bit_ = false;
|
|
uint32_t rtp_timestamp_ = 0;
|
|
uint16_t rtp_seq_num_ = 0;
|
|
std::vector<std::vector<uint8_t>> nalu_payloads_;
|
|
};
|
|
|
|
H264Packet::H264Packet(H264PacketizationTypes type) : type_(type) {
|
|
video_header_.video_type_header.emplace<RTPVideoHeaderH264>();
|
|
}
|
|
|
|
H264Packet& H264Packet::Idr(std::vector<uint8_t> payload, int pps_id) {
|
|
auto& h264_header = H264Header();
|
|
auto nalu_info = MakeNaluInfo(kIdr);
|
|
nalu_info.pps_id = pps_id;
|
|
h264_header.nalus[h264_header.nalus_length++] = nalu_info;
|
|
nalu_payloads_.push_back(std::move(payload));
|
|
return *this;
|
|
}
|
|
|
|
H264Packet& H264Packet::Slice(std::vector<uint8_t> payload) {
|
|
auto& h264_header = H264Header();
|
|
h264_header.nalus[h264_header.nalus_length++] = MakeNaluInfo(kSlice);
|
|
nalu_payloads_.push_back(std::move(payload));
|
|
return *this;
|
|
}
|
|
|
|
H264Packet& H264Packet::Sps(std::vector<uint8_t> payload, int sps_id) {
|
|
auto& h264_header = H264Header();
|
|
auto nalu_info = MakeNaluInfo(kSps);
|
|
nalu_info.pps_id = sps_id;
|
|
h264_header.nalus[h264_header.nalus_length++] = nalu_info;
|
|
nalu_payloads_.push_back(std::move(payload));
|
|
return *this;
|
|
}
|
|
|
|
H264Packet& H264Packet::SpsWithResolution(RenderResolution resolution,
|
|
std::vector<uint8_t> payload) {
|
|
auto& h264_header = H264Header();
|
|
h264_header.nalus[h264_header.nalus_length++] = MakeNaluInfo(kSps);
|
|
video_header_.width = resolution.Width();
|
|
video_header_.height = resolution.Height();
|
|
nalu_payloads_.push_back(std::move(payload));
|
|
return *this;
|
|
}
|
|
|
|
H264Packet& H264Packet::Pps(std::vector<uint8_t> payload,
|
|
int pps_id,
|
|
int sps_id) {
|
|
auto& h264_header = H264Header();
|
|
auto nalu_info = MakeNaluInfo(kPps);
|
|
nalu_info.pps_id = pps_id;
|
|
nalu_info.sps_id = sps_id;
|
|
h264_header.nalus[h264_header.nalus_length++] = nalu_info;
|
|
nalu_payloads_.push_back(std::move(payload));
|
|
return *this;
|
|
}
|
|
|
|
H264Packet& H264Packet::Aud() {
|
|
auto& h264_header = H264Header();
|
|
h264_header.nalus[h264_header.nalus_length++] = MakeNaluInfo(kAud);
|
|
nalu_payloads_.push_back({});
|
|
return *this;
|
|
}
|
|
|
|
H264Packet& H264Packet::Marker() {
|
|
marker_bit_ = true;
|
|
return *this;
|
|
}
|
|
|
|
H264Packet& H264Packet::AsFirstFragment() {
|
|
first_fragment_ = true;
|
|
return *this;
|
|
}
|
|
|
|
H264Packet& H264Packet::Time(uint32_t rtp_timestamp) {
|
|
rtp_timestamp_ = rtp_timestamp;
|
|
return *this;
|
|
}
|
|
|
|
H264Packet& H264Packet::SeqNum(uint16_t rtp_seq_num) {
|
|
rtp_seq_num_ = rtp_seq_num;
|
|
return *this;
|
|
}
|
|
|
|
std::unique_ptr<H26xPacketBuffer::Packet> H264Packet::Build() {
|
|
auto res = std::make_unique<H26xPacketBuffer::Packet>();
|
|
|
|
auto& h264_header = H264Header();
|
|
switch (type_) {
|
|
case kH264FuA: {
|
|
RTC_CHECK_EQ(h264_header.nalus_length, 1);
|
|
res->video_payload = BuildFuaPayload();
|
|
break;
|
|
}
|
|
case kH264SingleNalu: {
|
|
RTC_CHECK_EQ(h264_header.nalus_length, 1);
|
|
res->video_payload = BuildSingleNaluPayload();
|
|
break;
|
|
}
|
|
case kH264StapA: {
|
|
RTC_CHECK_GT(h264_header.nalus_length, 1);
|
|
RTC_CHECK_LE(h264_header.nalus_length, kMaxNalusPerPacket);
|
|
res->video_payload = BuildStapAPayload();
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (type_ == kH264FuA && !first_fragment_) {
|
|
h264_header.nalus_length = 0;
|
|
}
|
|
|
|
h264_header.packetization_type = type_;
|
|
res->marker_bit = marker_bit_;
|
|
res->video_header = video_header_;
|
|
res->timestamp = rtp_timestamp_;
|
|
res->seq_num = rtp_seq_num_;
|
|
res->video_header.codec = kVideoCodecH264;
|
|
|
|
return res;
|
|
}
|
|
|
|
rtc::CopyOnWriteBuffer H264Packet::BuildFuaPayload() const {
|
|
return rtc::CopyOnWriteBuffer(nalu_payloads_[0]);
|
|
}
|
|
|
|
rtc::CopyOnWriteBuffer H264Packet::BuildSingleNaluPayload() const {
|
|
rtc::CopyOnWriteBuffer res;
|
|
auto& h264_header = H264Header();
|
|
res.AppendData(&h264_header.nalus[0].type, 1);
|
|
res.AppendData(nalu_payloads_[0]);
|
|
return res;
|
|
}
|
|
|
|
rtc::CopyOnWriteBuffer H264Packet::BuildStapAPayload() const {
|
|
rtc::CopyOnWriteBuffer res;
|
|
|
|
const uint8_t indicator = H264::NaluType::kStapA;
|
|
res.AppendData(&indicator, 1);
|
|
|
|
auto& h264_header = H264Header();
|
|
for (size_t i = 0; i < h264_header.nalus_length; ++i) {
|
|
// The two first bytes indicates the nalu segment size.
|
|
uint8_t length_as_array[2] = {
|
|
0, static_cast<uint8_t>(nalu_payloads_[i].size() + 1)};
|
|
res.AppendData(length_as_array);
|
|
|
|
res.AppendData(&h264_header.nalus[i].type, 1);
|
|
res.AppendData(nalu_payloads_[i]);
|
|
}
|
|
return res;
|
|
}
|
|
|
|
#ifdef RTC_ENABLE_H265
|
|
class H265Packet {
|
|
public:
|
|
H265Packet() = default;
|
|
|
|
H265Packet& Idr(std::vector<uint8_t> payload = {9, 9, 9});
|
|
H265Packet& Slice(H265::NaluType type,
|
|
std::vector<uint8_t> payload = {9, 9, 9});
|
|
H265Packet& Vps(std::vector<uint8_t> payload = {9, 9, 9});
|
|
H265Packet& Sps(std::vector<uint8_t> payload = {9, 9, 9});
|
|
H265Packet& SpsWithResolution(RenderResolution resolution,
|
|
std::vector<uint8_t> payload = {9, 9, 9});
|
|
H265Packet& Pps(std::vector<uint8_t> payload = {9, 9, 9});
|
|
H265Packet& Aud();
|
|
H265Packet& Marker();
|
|
H265Packet& AsFirstFragment();
|
|
H265Packet& Time(uint32_t rtp_timestamp);
|
|
H265Packet& SeqNum(uint16_t rtp_seq_num);
|
|
|
|
std::unique_ptr<H26xPacketBuffer::Packet> Build();
|
|
|
|
private:
|
|
H265Packet& StartCode();
|
|
|
|
RTPVideoHeader video_header_;
|
|
bool first_fragment_ = false;
|
|
bool marker_bit_ = false;
|
|
uint32_t rtp_timestamp_ = 0;
|
|
uint16_t rtp_seq_num_ = 0;
|
|
std::vector<std::vector<uint8_t>> nalu_payloads_;
|
|
};
|
|
|
|
H265Packet& H265Packet::Idr(std::vector<uint8_t> payload) {
|
|
return Slice(H265::NaluType::kIdrNLp, std::move(payload));
|
|
}
|
|
|
|
H265Packet& H265Packet::Slice(H265::NaluType type,
|
|
std::vector<uint8_t> payload) {
|
|
StartCode();
|
|
// Nalu header. Assume layer ID is 0 and TID is 2.
|
|
nalu_payloads_.push_back({static_cast<uint8_t>(type << 1), 0x02});
|
|
nalu_payloads_.push_back(std::move(payload));
|
|
return *this;
|
|
}
|
|
|
|
H265Packet& H265Packet::Vps(std::vector<uint8_t> payload) {
|
|
return Slice(H265::NaluType::kVps, std::move(payload));
|
|
}
|
|
|
|
H265Packet& H265Packet::Sps(std::vector<uint8_t> payload) {
|
|
return Slice(H265::NaluType::kSps, std::move(payload));
|
|
}
|
|
|
|
H265Packet& H265Packet::SpsWithResolution(RenderResolution resolution,
|
|
std::vector<uint8_t> payload) {
|
|
video_header_.width = resolution.Width();
|
|
video_header_.height = resolution.Height();
|
|
return Sps(std::move(payload));
|
|
}
|
|
|
|
H265Packet& H265Packet::Pps(std::vector<uint8_t> payload) {
|
|
return Slice(H265::NaluType::kPps, std::move(payload));
|
|
}
|
|
|
|
H265Packet& H265Packet::Aud() {
|
|
return Slice(H265::NaluType::kAud, {});
|
|
}
|
|
|
|
H265Packet& H265Packet::Marker() {
|
|
marker_bit_ = true;
|
|
return *this;
|
|
}
|
|
|
|
H265Packet& H265Packet::StartCode() {
|
|
nalu_payloads_.push_back({0x00, 0x00, 0x00, 0x01});
|
|
return *this;
|
|
}
|
|
|
|
std::unique_ptr<H26xPacketBuffer::Packet> H265Packet::Build() {
|
|
auto res = std::make_unique<H26xPacketBuffer::Packet>();
|
|
res->marker_bit = marker_bit_;
|
|
res->video_header = video_header_;
|
|
res->timestamp = rtp_timestamp_;
|
|
res->seq_num = rtp_seq_num_;
|
|
res->video_header.codec = kVideoCodecH265;
|
|
res->video_payload = rtc::CopyOnWriteBuffer();
|
|
for (const auto& payload : nalu_payloads_) {
|
|
res->video_payload.AppendData(payload);
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
H265Packet& H265Packet::AsFirstFragment() {
|
|
first_fragment_ = true;
|
|
return *this;
|
|
}
|
|
|
|
H265Packet& H265Packet::Time(uint32_t rtp_timestamp) {
|
|
rtp_timestamp_ = rtp_timestamp;
|
|
return *this;
|
|
}
|
|
|
|
H265Packet& H265Packet::SeqNum(uint16_t rtp_seq_num) {
|
|
rtp_seq_num_ = rtp_seq_num;
|
|
return *this;
|
|
}
|
|
#endif
|
|
|
|
rtc::ArrayView<const uint8_t> PacketPayload(
|
|
const std::unique_ptr<H26xPacketBuffer::Packet>& packet) {
|
|
return packet->video_payload;
|
|
}
|
|
|
|
std::vector<uint8_t> FlatVector(
|
|
const std::vector<std::vector<uint8_t>>& elems) {
|
|
std::vector<uint8_t> res;
|
|
for (const auto& elem : elems) {
|
|
res.insert(res.end(), elem.begin(), elem.end());
|
|
}
|
|
return res;
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, IdrOnlyKeyframeWithSprop) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/true);
|
|
packet_buffer.SetSpropParameterSets(kExampleSpropString);
|
|
|
|
auto packets =
|
|
packet_buffer
|
|
.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Idr({1, 2, 3}, 0).Marker().Build())
|
|
.packets;
|
|
EXPECT_THAT(packets, SizeIs(1));
|
|
EXPECT_THAT(PacketPayload(packets[0]),
|
|
ElementsAreArray(FlatVector({StartCode(),
|
|
kExampleSpropRawSps,
|
|
StartCode(),
|
|
kExampleSpropRawPps,
|
|
StartCode(),
|
|
{kIdr, 1, 2, 3}})));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, IdrOnlyKeyframeWithoutSprop) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/true);
|
|
|
|
// Cannot fix biststream by prepending SPS and PPS because no sprop string is
|
|
// available. Request a key frame.
|
|
EXPECT_TRUE(
|
|
packet_buffer
|
|
.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Idr({9, 9, 9}, 0).Marker().Build())
|
|
.buffer_cleared);
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, IdrOnlyKeyframeWithSpropAndUnknownPpsId) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/true);
|
|
packet_buffer.SetSpropParameterSets(kExampleSpropString);
|
|
|
|
// Cannot fix biststream because sprop string doesn't contain a PPS with given
|
|
// ID. Request a key frame.
|
|
EXPECT_TRUE(
|
|
packet_buffer
|
|
.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Idr({9, 9, 9}, 1).Marker().Build())
|
|
.buffer_cleared);
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, IdrOnlyKeyframeInTheMiddle) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/true);
|
|
packet_buffer.SetSpropParameterSets(kExampleSpropString);
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Sps({1, 2, 3}, 1).SeqNum(0).Time(0).Build()));
|
|
RTC_UNUSED(packet_buffer.InsertPacket(H264Packet(kH264SingleNalu)
|
|
.Pps({4, 5, 6}, 1, 1)
|
|
.SeqNum(1)
|
|
.Time(0)
|
|
.Build()));
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264SingleNalu)
|
|
.Idr({7, 8, 9}, 1)
|
|
.SeqNum(2)
|
|
.Time(0)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
SizeIs(3));
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264SingleNalu)
|
|
.Slice()
|
|
.SeqNum(3)
|
|
.Time(1)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
SizeIs(1));
|
|
|
|
auto packets = packet_buffer
|
|
.InsertPacket(H264Packet(kH264SingleNalu)
|
|
.Idr({10, 11, 12}, 0)
|
|
.SeqNum(4)
|
|
.Time(2)
|
|
.Marker()
|
|
.Build())
|
|
.packets;
|
|
EXPECT_THAT(packets, SizeIs(1));
|
|
EXPECT_THAT(PacketPayload(packets[0]),
|
|
ElementsAreArray(FlatVector({StartCode(),
|
|
kExampleSpropRawSps,
|
|
StartCode(),
|
|
kExampleSpropRawPps,
|
|
StartCode(),
|
|
{kIdr, 10, 11, 12}})));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, IdrIsNotKeyframe) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(
|
|
packet_buffer
|
|
.InsertPacket(H264Packet(kH264SingleNalu).Idr().Marker().Build())
|
|
.packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, IdrIsKeyframeFuaRequiresFirstFragmet) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/true);
|
|
packet_buffer.SetSpropParameterSets(kExampleSpropString);
|
|
|
|
// Not marked as the first fragment
|
|
EXPECT_THAT(
|
|
packet_buffer
|
|
.InsertPacket(H264Packet(kH264FuA).Idr().SeqNum(0).Time(0).Build())
|
|
.packets,
|
|
IsEmpty());
|
|
|
|
EXPECT_THAT(
|
|
packet_buffer
|
|
.InsertPacket(
|
|
H264Packet(kH264FuA).Idr().SeqNum(1).Time(0).Marker().Build())
|
|
.packets,
|
|
IsEmpty());
|
|
|
|
// Marked as first fragment
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264FuA)
|
|
.Idr({9, 9, 9}, 0)
|
|
.SeqNum(2)
|
|
.Time(1)
|
|
.AsFirstFragment()
|
|
.Build())
|
|
.packets,
|
|
IsEmpty());
|
|
|
|
EXPECT_THAT(
|
|
packet_buffer
|
|
.InsertPacket(
|
|
H264Packet(kH264FuA).Idr().SeqNum(3).Time(1).Marker().Build())
|
|
.packets,
|
|
SizeIs(2));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, SpsPpsIdrIsKeyframeSingleNalus) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Sps().SeqNum(0).Time(0).Build()));
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Pps().SeqNum(1).Time(0).Build()));
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264SingleNalu)
|
|
.Idr()
|
|
.SeqNum(2)
|
|
.Time(0)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
SizeIs(3));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, SpsPpsIdrIsKeyframeIgnoresSprop) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
// When h264_allow_idr_only_keyframes is false, sprop string should be
|
|
// ignored. Use in band parameter sets.
|
|
packet_buffer.SetSpropParameterSets(kExampleSpropString);
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Sps({1, 2, 3}, 0).SeqNum(0).Time(0).Build()));
|
|
RTC_UNUSED(packet_buffer.InsertPacket(H264Packet(kH264SingleNalu)
|
|
.Pps({4, 5, 6}, 0, 0)
|
|
.SeqNum(1)
|
|
.Time(0)
|
|
.Build()));
|
|
auto packets = packet_buffer
|
|
.InsertPacket(H264Packet(kH264SingleNalu)
|
|
.Idr({7, 8, 9}, 0)
|
|
.SeqNum(2)
|
|
.Time(0)
|
|
.Marker()
|
|
.Build())
|
|
.packets;
|
|
EXPECT_THAT(packets, SizeIs(3));
|
|
EXPECT_THAT(PacketPayload(packets[0]),
|
|
ElementsAreArray(FlatVector({StartCode(), {kSps, 1, 2, 3}})));
|
|
EXPECT_THAT(PacketPayload(packets[1]),
|
|
ElementsAreArray(FlatVector({StartCode(), {kPps, 4, 5, 6}})));
|
|
EXPECT_THAT(PacketPayload(packets[2]),
|
|
ElementsAreArray(FlatVector({StartCode(), {kIdr, 7, 8, 9}})));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, PpsIdrIsNotKeyframeSingleNalus) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Pps().SeqNum(0).Time(0).Build()));
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264SingleNalu)
|
|
.Idr()
|
|
.SeqNum(1)
|
|
.Time(0)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, SpsIdrIsNotKeyframeSingleNalus) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Sps().SeqNum(0).Time(0).Build()));
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264SingleNalu)
|
|
.Idr()
|
|
.SeqNum(1)
|
|
.Time(0)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, SpsPpsIdrIsKeyframeStapA) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264StapA)
|
|
.Sps()
|
|
.Pps()
|
|
.Idr()
|
|
.SeqNum(0)
|
|
.Time(0)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
SizeIs(1));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, PpsIdrIsNotKeyframeStapA) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264StapA)
|
|
.Pps()
|
|
.Idr()
|
|
.SeqNum(0)
|
|
.Time(0)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, SpsIdrIsNotKeyframeStapA) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264StapA)
|
|
.Sps()
|
|
.Idr()
|
|
.SeqNum(2)
|
|
.Time(2)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
IsEmpty());
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264StapA)
|
|
.Sps()
|
|
.Pps()
|
|
.Idr()
|
|
.SeqNum(3)
|
|
.Time(3)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
SizeIs(1));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, InsertingSpsPpsLastCompletesKeyframe) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Idr().SeqNum(2).Time(1).Marker().Build()));
|
|
|
|
EXPECT_THAT(
|
|
packet_buffer
|
|
.InsertPacket(
|
|
H264Packet(kH264StapA).Sps().Pps().SeqNum(1).Time(1).Build())
|
|
.packets,
|
|
SizeIs(2));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, InsertingMidFuaCompletesFrame) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264StapA)
|
|
.Sps()
|
|
.Pps()
|
|
.Idr()
|
|
.SeqNum(0)
|
|
.Time(0)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
SizeIs(1));
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(H264Packet(kH264FuA)
|
|
.Slice()
|
|
.SeqNum(1)
|
|
.Time(1)
|
|
.AsFirstFragment()
|
|
.Build()));
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264FuA).Slice().SeqNum(3).Time(1).Marker().Build()));
|
|
EXPECT_THAT(
|
|
packet_buffer
|
|
.InsertPacket(H264Packet(kH264FuA).Slice().SeqNum(2).Time(1).Build())
|
|
.packets,
|
|
SizeIs(3));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, SeqNumJumpDoesNotCompleteFrame) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264StapA)
|
|
.Sps()
|
|
.Pps()
|
|
.Idr()
|
|
.SeqNum(0)
|
|
.Time(0)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
SizeIs(1));
|
|
|
|
EXPECT_THAT(
|
|
packet_buffer
|
|
.InsertPacket(H264Packet(kH264FuA).Slice().SeqNum(1).Time(1).Build())
|
|
.packets,
|
|
IsEmpty());
|
|
|
|
// Add `kBufferSize` to make the index of the sequence number wrap and end up
|
|
// where the packet with sequence number 2 would have ended up.
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264FuA)
|
|
.Slice()
|
|
.SeqNum(2 + kBufferSize)
|
|
.Time(3)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, OldFramesAreNotCompletedAfterBufferWrap) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264SingleNalu)
|
|
.Slice()
|
|
.SeqNum(1)
|
|
.Time(1)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
IsEmpty());
|
|
|
|
// New keyframe, preceedes packet with sequence number 1 in the buffer.
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264StapA)
|
|
.Sps()
|
|
.Pps()
|
|
.Idr()
|
|
.SeqNum(kBufferSize)
|
|
.Time(kBufferSize)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
SizeIs(1));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, OldPacketsDontBlockNewPackets) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264StapA)
|
|
.Sps()
|
|
.Pps()
|
|
.Idr()
|
|
.SeqNum(kBufferSize)
|
|
.Time(kBufferSize)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
SizeIs(1));
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(H264Packet(kH264FuA)
|
|
.Slice()
|
|
.SeqNum(kBufferSize + 1)
|
|
.Time(kBufferSize + 1)
|
|
.AsFirstFragment()
|
|
.Build()));
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(H264Packet(kH264FuA)
|
|
.Slice()
|
|
.SeqNum(kBufferSize + 3)
|
|
.Time(kBufferSize + 1)
|
|
.Marker()
|
|
.Build()));
|
|
EXPECT_THAT(
|
|
packet_buffer
|
|
.InsertPacket(H264Packet(kH264FuA).Slice().SeqNum(2).Time(2).Build())
|
|
.packets,
|
|
IsEmpty());
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264FuA)
|
|
.Slice()
|
|
.SeqNum(kBufferSize + 2)
|
|
.Time(kBufferSize + 1)
|
|
.Build())
|
|
.packets,
|
|
SizeIs(3));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, OldPacketDoesntCompleteFrame) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264StapA)
|
|
.Sps()
|
|
.Pps()
|
|
.Idr()
|
|
.SeqNum(kBufferSize)
|
|
.Time(kBufferSize)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
SizeIs(1));
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264FuA)
|
|
.Slice()
|
|
.SeqNum(kBufferSize + 3)
|
|
.Time(kBufferSize + 1)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
IsEmpty());
|
|
|
|
EXPECT_THAT(
|
|
packet_buffer
|
|
.InsertPacket(
|
|
H264Packet(kH264FuA).Slice().SeqNum(2).Time(2).Marker().Build())
|
|
.packets,
|
|
IsEmpty());
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264FuA)
|
|
.Slice()
|
|
.SeqNum(kBufferSize + 1)
|
|
.Time(kBufferSize + 1)
|
|
.AsFirstFragment()
|
|
.Build())
|
|
.packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, FrameBoundariesAreSet) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
auto key = packet_buffer.InsertPacket(H264Packet(kH264StapA)
|
|
.Sps()
|
|
.Pps()
|
|
.Idr()
|
|
.SeqNum(1)
|
|
.Time(1)
|
|
.Marker()
|
|
.Build());
|
|
|
|
ASSERT_THAT(key.packets, SizeIs(1));
|
|
EXPECT_TRUE(key.packets[0]->video_header.is_first_packet_in_frame);
|
|
EXPECT_TRUE(key.packets[0]->video_header.is_last_packet_in_frame);
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264FuA).Slice().SeqNum(2).Time(2).Build()));
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264FuA).Slice().SeqNum(3).Time(2).Build()));
|
|
auto delta = packet_buffer.InsertPacket(
|
|
H264Packet(kH264FuA).Slice().SeqNum(4).Time(2).Marker().Build());
|
|
|
|
ASSERT_THAT(delta.packets, SizeIs(3));
|
|
EXPECT_TRUE(delta.packets[0]->video_header.is_first_packet_in_frame);
|
|
EXPECT_FALSE(delta.packets[0]->video_header.is_last_packet_in_frame);
|
|
|
|
EXPECT_FALSE(delta.packets[1]->video_header.is_first_packet_in_frame);
|
|
EXPECT_FALSE(delta.packets[1]->video_header.is_last_packet_in_frame);
|
|
|
|
EXPECT_FALSE(delta.packets[2]->video_header.is_first_packet_in_frame);
|
|
EXPECT_TRUE(delta.packets[2]->video_header.is_last_packet_in_frame);
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, ResolutionSetOnFirstPacket) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Aud().SeqNum(1).Time(1).Build()));
|
|
auto res = packet_buffer.InsertPacket(H264Packet(kH264StapA)
|
|
.SpsWithResolution({320, 240})
|
|
.Pps()
|
|
.Idr()
|
|
.SeqNum(2)
|
|
.Time(1)
|
|
.Marker()
|
|
.Build());
|
|
|
|
ASSERT_THAT(res.packets, SizeIs(2));
|
|
EXPECT_THAT(res.packets[0]->video_header.width, Eq(320));
|
|
EXPECT_THAT(res.packets[0]->video_header.height, Eq(240));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, KeyframeAndDeltaFrameSetOnFirstPacket) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Aud().SeqNum(1).Time(1).Build()));
|
|
auto key = packet_buffer.InsertPacket(H264Packet(kH264StapA)
|
|
.Sps()
|
|
.Pps()
|
|
.Idr()
|
|
.SeqNum(2)
|
|
.Time(1)
|
|
.Marker()
|
|
.Build());
|
|
|
|
auto delta = packet_buffer.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Slice().SeqNum(3).Time(2).Marker().Build());
|
|
|
|
ASSERT_THAT(key.packets, SizeIs(2));
|
|
EXPECT_THAT(key.packets[0]->video_header.frame_type,
|
|
Eq(VideoFrameType::kVideoFrameKey));
|
|
ASSERT_THAT(delta.packets, SizeIs(1));
|
|
EXPECT_THAT(delta.packets[0]->video_header.frame_type,
|
|
Eq(VideoFrameType::kVideoFrameDelta));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, RtpSeqNumWrap) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264StapA).Sps().Pps().SeqNum(0xffff).Time(0).Build()));
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264FuA).Idr().SeqNum(0).Time(0).Build()));
|
|
EXPECT_THAT(
|
|
packet_buffer
|
|
.InsertPacket(
|
|
H264Packet(kH264FuA).Idr().SeqNum(1).Time(0).Marker().Build())
|
|
.packets,
|
|
SizeIs(3));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, StapAFixedBitstream) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
auto packets = packet_buffer
|
|
.InsertPacket(H264Packet(kH264StapA)
|
|
.Sps({1, 2, 3})
|
|
.Pps({4, 5, 6})
|
|
.Idr({7, 8, 9})
|
|
.SeqNum(0)
|
|
.Time(0)
|
|
.Marker()
|
|
.Build())
|
|
.packets;
|
|
|
|
ASSERT_THAT(packets, SizeIs(1));
|
|
EXPECT_THAT(PacketPayload(packets[0]),
|
|
ElementsAreArray(FlatVector({StartCode(),
|
|
{kSps, 1, 2, 3},
|
|
StartCode(),
|
|
{kPps, 4, 5, 6},
|
|
StartCode(),
|
|
{kIdr, 7, 8, 9}})));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, SingleNaluFixedBitstream) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Sps({1, 2, 3}).SeqNum(0).Time(0).Build()));
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Pps({4, 5, 6}).SeqNum(1).Time(0).Build()));
|
|
auto packets = packet_buffer
|
|
.InsertPacket(H264Packet(kH264SingleNalu)
|
|
.Idr({7, 8, 9})
|
|
.SeqNum(2)
|
|
.Time(0)
|
|
.Marker()
|
|
.Build())
|
|
.packets;
|
|
|
|
ASSERT_THAT(packets, SizeIs(3));
|
|
EXPECT_THAT(PacketPayload(packets[0]),
|
|
ElementsAreArray(FlatVector({StartCode(), {kSps, 1, 2, 3}})));
|
|
EXPECT_THAT(PacketPayload(packets[1]),
|
|
ElementsAreArray(FlatVector({StartCode(), {kPps, 4, 5, 6}})));
|
|
EXPECT_THAT(PacketPayload(packets[2]),
|
|
ElementsAreArray(FlatVector({StartCode(), {kIdr, 7, 8, 9}})));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, StapaAndFuaFixedBitstream) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(H264Packet(kH264StapA)
|
|
.Sps({1, 2, 3})
|
|
.Pps({4, 5, 6})
|
|
.SeqNum(0)
|
|
.Time(0)
|
|
.Build()));
|
|
RTC_UNUSED(packet_buffer.InsertPacket(H264Packet(kH264FuA)
|
|
.Idr({8, 8, 8})
|
|
.SeqNum(1)
|
|
.Time(0)
|
|
.AsFirstFragment()
|
|
.Build()));
|
|
auto packets = packet_buffer
|
|
.InsertPacket(H264Packet(kH264FuA)
|
|
.Idr({9, 9, 9})
|
|
.SeqNum(2)
|
|
.Time(0)
|
|
.Marker()
|
|
.Build())
|
|
.packets;
|
|
|
|
ASSERT_THAT(packets, SizeIs(3));
|
|
EXPECT_THAT(
|
|
PacketPayload(packets[0]),
|
|
ElementsAreArray(FlatVector(
|
|
{StartCode(), {kSps, 1, 2, 3}, StartCode(), {kPps, 4, 5, 6}})));
|
|
EXPECT_THAT(PacketPayload(packets[1]),
|
|
ElementsAreArray(FlatVector({StartCode(), {8, 8, 8}})));
|
|
// Third is a continuation of second, so only the payload is expected.
|
|
EXPECT_THAT(PacketPayload(packets[2]),
|
|
ElementsAreArray(FlatVector({{9, 9, 9}})));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, FullPacketBufferDoesNotBlockKeyframe) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
for (int i = 0; i < kBufferSize; ++i) {
|
|
EXPECT_THAT(
|
|
packet_buffer
|
|
.InsertPacket(
|
|
H264Packet(kH264SingleNalu).Slice().SeqNum(i).Time(0).Build())
|
|
.packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H264Packet(kH264StapA)
|
|
.Sps()
|
|
.Pps()
|
|
.Idr()
|
|
.SeqNum(kBufferSize)
|
|
.Time(1)
|
|
.Marker()
|
|
.Build())
|
|
.packets,
|
|
SizeIs(1));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, TooManyNalusInPacket) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
std::unique_ptr<H26xPacketBuffer::Packet> packet(H264Packet(kH264StapA)
|
|
.Sps()
|
|
.Pps()
|
|
.Idr()
|
|
.SeqNum(1)
|
|
.Time(1)
|
|
.Marker()
|
|
.Build());
|
|
auto& h264_header =
|
|
absl::get<RTPVideoHeaderH264>(packet->video_header.video_type_header);
|
|
h264_header.nalus_length = kMaxNalusPerPacket + 1;
|
|
|
|
EXPECT_THAT(packet_buffer.InsertPacket(std::move(packet)).packets, IsEmpty());
|
|
}
|
|
|
|
#ifdef RTC_ENABLE_H265
|
|
TEST(H26xPacketBufferTest, H265VpsSpsPpsIdrIsKeyframe) {
|
|
H26xPacketBuffer packet_buffer(/*allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(
|
|
packet_buffer
|
|
.InsertPacket(H265Packet().Vps().Sps().Pps().Idr().Marker().Build())
|
|
.packets,
|
|
SizeIs(1));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, H265IrapIsNotKeyframe) {
|
|
std::vector<const H265::NaluType> irap_types = {
|
|
H265::NaluType::kBlaWLp, H265::NaluType::kBlaWRadl,
|
|
H265::NaluType::kBlaNLp, H265::NaluType::kIdrWRadl,
|
|
H265::NaluType::kIdrNLp, H265::NaluType::kCra,
|
|
H265::NaluType::kRsvIrapVcl23};
|
|
for (const H265::NaluType type : irap_types) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(
|
|
packet_buffer.InsertPacket(H265Packet().Slice(type).Marker().Build())
|
|
.packets,
|
|
IsEmpty());
|
|
}
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, H265IdrIsNotKeyFrame) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(
|
|
packet_buffer.InsertPacket(H265Packet().Idr().Marker().Build()).packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, H265IdrIsNotKeyFrameEvenWithSprop) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/true);
|
|
packet_buffer.SetSpropParameterSets(kExampleSpropString);
|
|
|
|
EXPECT_THAT(
|
|
packet_buffer.InsertPacket(H265Packet().Idr().Marker().Build()).packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, H265SpsPpsIdrIsNotKeyFrame) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H265Packet().Sps().Pps().Idr().Marker().Build())
|
|
.packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, H265VpsPpsIdrIsNotKeyFrame) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H265Packet().Vps().Pps().Idr().Marker().Build())
|
|
.packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, H265VpsSpsIdrIsNotKeyFrame) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(H265Packet().Vps().Sps().Idr().Marker().Build())
|
|
.packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, H265VpsIdrIsNotKeyFrame) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(
|
|
packet_buffer.InsertPacket(H265Packet().Vps().Idr().Marker().Build())
|
|
.packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, H265SpsIdrIsNotKeyFrame) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(
|
|
packet_buffer.InsertPacket(H265Packet().Sps().Idr().Marker().Build())
|
|
.packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, H265PpsIdrIsNotKeyFrame) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
EXPECT_THAT(
|
|
packet_buffer.InsertPacket(H265Packet().Pps().Idr().Marker().Build())
|
|
.packets,
|
|
IsEmpty());
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, H265ResolutionSetOnSpsPacket) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
RTC_UNUSED(
|
|
packet_buffer.InsertPacket(H265Packet().Aud().SeqNum(1).Time(1).Build()));
|
|
auto res = packet_buffer.InsertPacket(H265Packet()
|
|
.Vps()
|
|
.SpsWithResolution({320, 240})
|
|
.Pps()
|
|
.Idr()
|
|
.SeqNum(2)
|
|
.Time(1)
|
|
.Marker()
|
|
.Build());
|
|
|
|
ASSERT_THAT(res.packets, SizeIs(2));
|
|
EXPECT_THAT(res.packets[0]->video_header.width, Eq(320));
|
|
EXPECT_THAT(res.packets[0]->video_header.height, Eq(240));
|
|
}
|
|
|
|
TEST(H26xPacketBufferTest, H265InsertingVpsSpsPpsLastCompletesKeyframe) {
|
|
H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false);
|
|
|
|
RTC_UNUSED(packet_buffer.InsertPacket(
|
|
H265Packet().Idr().SeqNum(2).Time(1).Marker().Build()));
|
|
|
|
EXPECT_THAT(packet_buffer
|
|
.InsertPacket(
|
|
H265Packet().Vps().Sps().Pps().SeqNum(1).Time(1).Build())
|
|
.packets,
|
|
SizeIs(2));
|
|
}
|
|
#endif // RTC_ENABLE_H265
|
|
|
|
} // namespace
|
|
} // namespace webrtc
|