| /* |
| * 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(int64_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; |
| int64_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.push_back(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.push_back(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.push_back(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.push_back(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.push_back(nalu_info); |
| nalu_payloads_.push_back(std::move(payload)); |
| return *this; |
| } |
| |
| H264Packet& H264Packet::Aud() { |
| auto& h264_header = H264Header(); |
| h264_header.nalus.push_back(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(int64_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.size(), 1); |
| res->video_payload = BuildFuaPayload(); |
| break; |
| } |
| case kH264SingleNalu: { |
| RTC_CHECK_EQ(h264_header.nalus.size(), 1); |
| res->video_payload = BuildSingleNaluPayload(); |
| break; |
| } |
| case kH264StapA: { |
| RTC_CHECK_GT(h264_header.nalus.size(), 1); |
| res->video_payload = BuildStapAPayload(); |
| break; |
| } |
| } |
| |
| if (type_ == kH264FuA && !first_fragment_) { |
| h264_header.nalus.clear(); |
| } |
| |
| h264_header.packetization_type = type_; |
| res->marker_bit = marker_bit_; |
| res->video_header = video_header_; |
| res->timestamp = rtp_timestamp_; |
| res->sequence_number = 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.size(); ++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(int64_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->sequence_number = 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(int64_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(0x1'0000).Time(0).Build())); |
| EXPECT_THAT(packet_buffer |
| .InsertPacket(H264Packet(kH264FuA) |
| .Idr() |
| .SeqNum(0x1'0001) |
| .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, AssembleFrameAfterReordering) { |
| H26xPacketBuffer packet_buffer(/*h264_allow_idr_only_keyframes=*/false); |
| |
| EXPECT_THAT(packet_buffer |
| .InsertPacket(H264Packet(kH264StapA) |
| .Sps() |
| .Pps() |
| .Idr() |
| .SeqNum(2) |
| .Time(2) |
| .Marker() |
| .Build()) |
| .packets, |
| SizeIs(1)); |
| |
| EXPECT_THAT(packet_buffer |
| .InsertPacket(H264Packet(kH264SingleNalu) |
| .Slice() |
| .SeqNum(1) |
| .Time(1) |
| .Marker() |
| .Build()) |
| .packets, |
| IsEmpty()); |
| |
| EXPECT_THAT(packet_buffer |
| .InsertPacket(H264Packet(kH264StapA) |
| .Sps() |
| .Pps() |
| .Idr() |
| .SeqNum(0) |
| .Time(0) |
| .Marker() |
| .Build()) |
| .packets, |
| SizeIs(2)); |
| } |
| |
| TEST(H26xPacketBufferTest, AssembleFrameAfterLoss) { |
| 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(kH264StapA) |
| .Sps() |
| .Pps() |
| .Idr() |
| .SeqNum(2) |
| .Time(2) |
| .Marker() |
| .Build()) |
| .packets, |
| SizeIs(1)); |
| } |
| |
| #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<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 |