From d5350373ab1300d0086248b358bc9e6f7eb90649 Mon Sep 17 00:00:00 2001 From: Robert Edmonds Date: Tue, 19 Mar 2024 00:36:19 -0400 Subject: [PATCH 1/6] Add H265RtpDepacketizer This commit adds an H265 depacketizer which takes incoming H265 RTP packets and emits H265 access units. It is closely based on the `H264RtpDepacketizer` added by @Sean-Der in https://github.com/paullouisageneau/libdatachannel/pull/1082. I originally started with a version of this commit that was closer to the `H264RtpDepacketizer` and which emitted individual H265 NALUs in `H265RtpDepacketizer::buildFrames()`. This resulted in calling my `Track::onFrame()` callback for each NALU, which did not work well with the decoder that I'm using which wants to see the VPS/SPS/PPS NALUs as a unit before initializing the decoder (https://intel.github.io/libvpl/v2.10/API_ref/VPL_func_vid_decode.html#mfxvideodecode-decodeheader). So for the `H265RtpDepacketizer` I've tried to make it emit access units rather than NALUs. An "access unit" is (RFC 7798): > A set of NAL units that are associated with each other according to a specified classification rule, that are consecutive in decoding order, *and that contain exactly one coded picture.* "Exactly one coded picture" seems to correspond with what a caller might expect an "onFrame" callback to do. Maybe the `H264RtpDepacketizer` should be revised to similarly emit H264 access units rather than NALUs, too. At least, I could not find a way to receive individual NALUs from the depacketizer and run the VPL decoder without needing to do my own buffering/copying of the NALUs. With this commit I can now do the following: * Generate encoded bitstream output from the Intel VPL encoder. * Pass the output of the encoder one frame at a time to libdatachannel's `Track::send()` on a track with an `H265RtpPacketizer` media handler. * Transport the video track over a WebRTC connection to a libdatachannel peer. * Depacketize it with the `H265RtpDepacketizer` media handler in this commit. * Pass the depacketized output via my `Track::onFrame()` callback to the Intel VPL decoder in "complete frame" mode (https://intel.github.io/libvpl/v2.10/API_ref/VPL_enums.html#_CPPv428MFX_BITSTREAM_COMPLETE_FRAME). Each "onFrame" callback corresponds to a single call to the decoder API to decode a frame. --- CMakeLists.txt | 2 + include/rtc/h265rtpdepacketizer.hpp | 44 +++++++++ include/rtc/rtc.hpp | 1 + src/h265nalunit.cpp | 2 +- src/h265rtpdepacketizer.cpp | 143 ++++++++++++++++++++++++++++ 5 files changed, 191 insertions(+), 1 deletion(-) create mode 100644 include/rtc/h265rtpdepacketizer.hpp create mode 100644 src/h265rtpdepacketizer.cpp diff --git a/CMakeLists.txt b/CMakeLists.txt index 696f60407..be3386b51 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -82,6 +82,7 @@ set(LIBDATACHANNEL_SOURCES ${CMAKE_CURRENT_SOURCE_DIR}/src/h264rtpdepacketizer.cpp ${CMAKE_CURRENT_SOURCE_DIR}/src/nalunit.cpp ${CMAKE_CURRENT_SOURCE_DIR}/src/h265rtppacketizer.cpp + ${CMAKE_CURRENT_SOURCE_DIR}/src/h265rtpdepacketizer.cpp ${CMAKE_CURRENT_SOURCE_DIR}/src/h265nalunit.cpp ${CMAKE_CURRENT_SOURCE_DIR}/src/av1rtppacketizer.cpp ${CMAKE_CURRENT_SOURCE_DIR}/src/rtcpnackresponder.cpp @@ -120,6 +121,7 @@ set(LIBDATACHANNEL_HEADERS ${CMAKE_CURRENT_SOURCE_DIR}/include/rtc/h264rtpdepacketizer.hpp ${CMAKE_CURRENT_SOURCE_DIR}/include/rtc/nalunit.hpp ${CMAKE_CURRENT_SOURCE_DIR}/include/rtc/h265rtppacketizer.hpp + ${CMAKE_CURRENT_SOURCE_DIR}/include/rtc/h265rtpdepacketizer.hpp ${CMAKE_CURRENT_SOURCE_DIR}/include/rtc/h265nalunit.hpp ${CMAKE_CURRENT_SOURCE_DIR}/include/rtc/av1rtppacketizer.hpp ${CMAKE_CURRENT_SOURCE_DIR}/include/rtc/rtcpnackresponder.hpp diff --git a/include/rtc/h265rtpdepacketizer.hpp b/include/rtc/h265rtpdepacketizer.hpp new file mode 100644 index 000000000..7ba52e26f --- /dev/null +++ b/include/rtc/h265rtpdepacketizer.hpp @@ -0,0 +1,44 @@ +/** + * Copyright (c) 2020 Staz Modrzynski + * Copyright (c) 2020-2024 Paul-Louis Ageneau + * Copyright (c) 2024 Robert Edmonds + * + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at https://mozilla.org/MPL/2.0/. + */ + +#ifndef RTC_H265_RTP_DEPACKETIZER_H +#define RTC_H265_RTP_DEPACKETIZER_H + +#if RTC_ENABLE_MEDIA + +#include "common.hpp" +#include "mediahandler.hpp" +#include "message.hpp" +#include "rtp.hpp" + +#include + +namespace rtc { + +/// RTP depacketization for H265 +class RTC_CPP_EXPORT H265RtpDepacketizer : public MediaHandler { +public: + H265RtpDepacketizer() = default; + virtual ~H265RtpDepacketizer() = default; + + void incoming(message_vector &messages, const message_callback &send) override; + +private: + std::vector mRtpBuffer; + + message_vector buildFrames(message_vector::iterator firstPkt, message_vector::iterator lastPkt, + uint32_t timestamp); +}; + +} // namespace rtc + +#endif // RTC_ENABLE_MEDIA + +#endif // RTC_H265_RTP_DEPACKETIZER_H diff --git a/include/rtc/rtc.hpp b/include/rtc/rtc.hpp index 8a2b33010..74a9c9e8b 100644 --- a/include/rtc/rtc.hpp +++ b/include/rtc/rtc.hpp @@ -32,6 +32,7 @@ #include "h264rtppacketizer.hpp" #include "h264rtpdepacketizer.hpp" #include "h265rtppacketizer.hpp" +#include "h265rtpdepacketizer.hpp" #include "mediahandler.hpp" #include "plihandler.hpp" #include "rembhandler.hpp" diff --git a/src/h265nalunit.cpp b/src/h265nalunit.cpp index 5fda10545..6f6b7e12a 100644 --- a/src/h265nalunit.cpp +++ b/src/h265nalunit.cpp @@ -34,7 +34,7 @@ H265NalUnitFragment::fragmentsFrom(shared_ptr nalu, uint16_t maxFra auto fragments_count = ceil(double(nalu->size()) / maxFragmentSize); maxFragmentSize = uint16_t(int(ceil(nalu->size() / fragments_count))); - // 3 bytes for FU indicator and FU header + // 3 bytes for NALU header and FU header maxFragmentSize -= (H265_NAL_HEADER_SIZE + H265_FU_HEADER_SIZE); auto f = nalu->forbiddenBit(); uint8_t nuhLayerId = nalu->nuhLayerId() & 0x3F; // 6 bits diff --git a/src/h265rtpdepacketizer.cpp b/src/h265rtpdepacketizer.cpp new file mode 100644 index 000000000..dff319f64 --- /dev/null +++ b/src/h265rtpdepacketizer.cpp @@ -0,0 +1,143 @@ +/** + * Copyright (c) 2023-2024 Paul-Louis Ageneau + * Copyright (c) 2024 Robert Edmonds + * + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at https://mozilla.org/MPL/2.0/. + */ + +#if RTC_ENABLE_MEDIA + +#include "h265rtpdepacketizer.hpp" +#include "h265nalunit.hpp" + +#include "impl/internals.hpp" + +namespace rtc { + +const binary naluStartCode = {byte{0}, byte{0}, byte{0}, byte{1}}; + +const uint8_t naluTypeAP = 48; +const uint8_t naluTypeFU = 49; + +message_vector H265RtpDepacketizer::buildFrames(message_vector::iterator begin, + message_vector::iterator end, uint32_t timestamp) { + message_vector out = {}; + auto accessUnit = binary{}; + auto frameInfo = std::make_shared(timestamp); + auto nFrags = 0; + + for (auto it = begin; it != end; ++it) { + auto pkt = it->get(); + auto pktParsed = reinterpret_cast(pkt->data()); + auto rtpHeaderSize = pktParsed->getSize() + pktParsed->getExtensionHeaderSize(); + auto nalUnitHeader = + H265NalUnitHeader{std::to_integer(pkt->at(rtpHeaderSize)), + std::to_integer(pkt->at(rtpHeaderSize + 1))}; + + if (nalUnitHeader.unitType() == naluTypeFU) { + auto nalUnitFragmentHeader = H265NalUnitFragmentHeader{ + std::to_integer(pkt->at(rtpHeaderSize + sizeof(H265NalUnitHeader)))}; + + if (nFrags++ == 0) { + accessUnit.insert(accessUnit.end(), naluStartCode.begin(), naluStartCode.end()); + + nalUnitHeader.setUnitType(nalUnitFragmentHeader.unitType()); + accessUnit.emplace_back(byte(nalUnitHeader._first)); + accessUnit.emplace_back(byte(nalUnitHeader._second)); + } + + accessUnit.insert(accessUnit.end(), + pkt->begin() + rtpHeaderSize + sizeof(H265NalUnitHeader) + + sizeof(H265NalUnitFragmentHeader), + pkt->end()); + } else if (nalUnitHeader.unitType() == naluTypeAP) { + auto currOffset = rtpHeaderSize + sizeof(H265NalUnitHeader); + + while (currOffset + sizeof(uint16_t) < pkt->size()) { + auto naluSize = std::to_integer(pkt->at(currOffset)) << 8 | + std::to_integer(pkt->at(currOffset + 1)); + + currOffset += sizeof(uint16_t); + + if (pkt->size() < currOffset + naluSize) { + throw std::runtime_error("H265 AP declared size is larger than buffer"); + } + + accessUnit.insert(accessUnit.end(), naluStartCode.begin(), naluStartCode.end()); + + accessUnit.insert(accessUnit.end(), pkt->begin() + currOffset, + pkt->begin() + currOffset + naluSize); + + currOffset += naluSize; + } + } else if (nalUnitHeader.unitType() < naluTypeAP) { + // "NAL units with NAL unit type values in the range of 0 to 47, inclusive, may be + // passed to the decoder." + accessUnit.insert(accessUnit.end(), naluStartCode.begin(), naluStartCode.end()); + accessUnit.insert(accessUnit.end(), pkt->begin() + rtpHeaderSize, pkt->end()); + } else { + // "NAL-unit-like structures with NAL unit type values in the range of 48 to 63, + // inclusive, MUST NOT be passed to the decoder." + } + } + + if (!accessUnit.empty()) { + out.emplace_back(make_message(accessUnit.begin(), accessUnit.end(), Message::Binary, 0, + nullptr, frameInfo)); + } + + return out; +} + +void H265RtpDepacketizer::incoming(message_vector &messages, const message_callback &) { + messages.erase(std::remove_if(messages.begin(), messages.end(), + [&](message_ptr message) { + if (message->type == Message::Control) { + return false; + } + + if (message->size() < sizeof(RtpHeader)) { + PLOG_VERBOSE << "RTP packet is too small, size=" + << message->size(); + return true; + } + + mRtpBuffer.push_back(std::move(message)); + return true; + }), + messages.end()); + + while (mRtpBuffer.size() != 0) { + uint32_t current_timestamp = 0; + size_t packets_in_timestamp = 0; + + for (const auto &pkt : mRtpBuffer) { + auto p = reinterpret_cast(pkt->data()); + + if (current_timestamp == 0) { + current_timestamp = p->timestamp(); + } else if (current_timestamp != p->timestamp()) { + break; + } + + packets_in_timestamp++; + } + + if (packets_in_timestamp == mRtpBuffer.size()) { + break; + } + + auto begin = mRtpBuffer.begin(); + auto end = mRtpBuffer.begin() + (packets_in_timestamp - 1); + + auto frames = buildFrames(begin, end + 1, current_timestamp); + messages.insert(messages.end(), frames.begin(), frames.end()); + mRtpBuffer.erase(mRtpBuffer.begin(), mRtpBuffer.begin() + packets_in_timestamp); + } +} + +} // namespace rtc + +#endif // RTC_ENABLE_MEDIA From c7348d07217535e40d0dc332393c5df7c221fad1 Mon Sep 17 00:00:00 2001 From: Robert Edmonds Date: Sat, 13 Apr 2024 17:44:18 -0400 Subject: [PATCH 2/6] H265RtpDepacketizer: Handle empty RTP packet payloads Similar to the fix for the H.264 RTP depacketizer in https://github.com/ paullouisageneau/libdatachannel/pull/1140. --- src/h265rtpdepacketizer.cpp | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/src/h265rtpdepacketizer.cpp b/src/h265rtpdepacketizer.cpp index dff319f64..943ae4d9c 100644 --- a/src/h265rtpdepacketizer.cpp +++ b/src/h265rtpdepacketizer.cpp @@ -32,6 +32,17 @@ message_vector H265RtpDepacketizer::buildFrames(message_vector::iterator begin, auto pkt = it->get(); auto pktParsed = reinterpret_cast(pkt->data()); auto rtpHeaderSize = pktParsed->getSize() + pktParsed->getExtensionHeaderSize(); + auto rtpPaddingSize = 0; + + if (pktParsed->padding()) { + rtpPaddingSize = std::to_integer(pkt->at(pkt->size() - 1)); + } + + if (pkt->size() == rtpHeaderSize + rtpPaddingSize) { + PLOG_VERBOSE << "H.265 RTP packet has empty payload"; + continue; + } + auto nalUnitHeader = H265NalUnitHeader{std::to_integer(pkt->at(rtpHeaderSize)), std::to_integer(pkt->at(rtpHeaderSize + 1))}; From 115b892925147ecf2fb70f98cb47f650fb94f901 Mon Sep 17 00:00:00 2001 From: Robert Edmonds Date: Sat, 13 Apr 2024 18:24:47 -0400 Subject: [PATCH 3/6] H265RtpDepacketizer: Make start sequence configurable Per PR review feedback, add a parameter to the constructor for configuring the start sequence to use when writing NALUs. --- include/rtc/h265rtpdepacketizer.hpp | 7 ++++- src/h265rtpdepacketizer.cpp | 43 +++++++++++++++++++++++++---- 2 files changed, 43 insertions(+), 7 deletions(-) diff --git a/include/rtc/h265rtpdepacketizer.hpp b/include/rtc/h265rtpdepacketizer.hpp index 7ba52e26f..2ebc43501 100644 --- a/include/rtc/h265rtpdepacketizer.hpp +++ b/include/rtc/h265rtpdepacketizer.hpp @@ -14,6 +14,7 @@ #if RTC_ENABLE_MEDIA #include "common.hpp" +#include "h265nalunit.hpp" #include "mediahandler.hpp" #include "message.hpp" #include "rtp.hpp" @@ -25,14 +26,18 @@ namespace rtc { /// RTP depacketization for H265 class RTC_CPP_EXPORT H265RtpDepacketizer : public MediaHandler { public: - H265RtpDepacketizer() = default; + using Separator = NalUnit::Separator; + + H265RtpDepacketizer(Separator separator = Separator::LongStartSequence); virtual ~H265RtpDepacketizer() = default; void incoming(message_vector &messages, const message_callback &send) override; private: std::vector mRtpBuffer; + const NalUnit::Separator separator; + void addSeparator(binary& accessUnit); message_vector buildFrames(message_vector::iterator firstPkt, message_vector::iterator lastPkt, uint32_t timestamp); }; diff --git a/src/h265rtpdepacketizer.cpp b/src/h265rtpdepacketizer.cpp index 943ae4d9c..506fecfbf 100644 --- a/src/h265rtpdepacketizer.cpp +++ b/src/h265rtpdepacketizer.cpp @@ -16,11 +16,44 @@ namespace rtc { -const binary naluStartCode = {byte{0}, byte{0}, byte{0}, byte{1}}; +const binary naluLongStartCode = {byte{0}, byte{0}, byte{0}, byte{1}}; +const binary naluShortStartCode = {byte{0}, byte{0}, byte{1}}; const uint8_t naluTypeAP = 48; const uint8_t naluTypeFU = 49; +H265RtpDepacketizer::H265RtpDepacketizer(Separator separator) : separator(separator) { + switch (separator) { + case Separator::StartSequence: [[fallthrough]]; + case Separator::LongStartSequence: [[fallthrough]]; + case Separator::ShortStartSequence: + break; + case Separator::Length: [[fallthrough]]; + default: + throw std::invalid_argument("Invalid separator"); + } +} + +void H265RtpDepacketizer::addSeparator(binary& accessUnit) +{ + switch (separator) { + case Separator::StartSequence: [[fallthrough]]; + case Separator::LongStartSequence: + accessUnit.insert(accessUnit.end(), + naluLongStartCode.begin(), + naluLongStartCode.end()); + break; + case Separator::ShortStartSequence: + accessUnit.insert(accessUnit.end(), + naluShortStartCode.begin(), + naluShortStartCode.end()); + break; + case Separator::Length: [[fallthrough]]; + default: + throw std::invalid_argument("Invalid separator"); + } +} + message_vector H265RtpDepacketizer::buildFrames(message_vector::iterator begin, message_vector::iterator end, uint32_t timestamp) { message_vector out = {}; @@ -52,8 +85,7 @@ message_vector H265RtpDepacketizer::buildFrames(message_vector::iterator begin, std::to_integer(pkt->at(rtpHeaderSize + sizeof(H265NalUnitHeader)))}; if (nFrags++ == 0) { - accessUnit.insert(accessUnit.end(), naluStartCode.begin(), naluStartCode.end()); - + addSeparator(accessUnit); nalUnitHeader.setUnitType(nalUnitFragmentHeader.unitType()); accessUnit.emplace_back(byte(nalUnitHeader._first)); accessUnit.emplace_back(byte(nalUnitHeader._second)); @@ -76,8 +108,7 @@ message_vector H265RtpDepacketizer::buildFrames(message_vector::iterator begin, throw std::runtime_error("H265 AP declared size is larger than buffer"); } - accessUnit.insert(accessUnit.end(), naluStartCode.begin(), naluStartCode.end()); - + addSeparator(accessUnit); accessUnit.insert(accessUnit.end(), pkt->begin() + currOffset, pkt->begin() + currOffset + naluSize); @@ -86,7 +117,7 @@ message_vector H265RtpDepacketizer::buildFrames(message_vector::iterator begin, } else if (nalUnitHeader.unitType() < naluTypeAP) { // "NAL units with NAL unit type values in the range of 0 to 47, inclusive, may be // passed to the decoder." - accessUnit.insert(accessUnit.end(), naluStartCode.begin(), naluStartCode.end()); + addSeparator(accessUnit); accessUnit.insert(accessUnit.end(), pkt->begin() + rtpHeaderSize, pkt->end()); } else { // "NAL-unit-like structures with NAL unit type values in the range of 48 to 63, From f787617e7b2c0939a78923f0146d52d6c3689d72 Mon Sep 17 00:00:00 2001 From: Robert Edmonds Date: Sun, 27 Oct 2024 20:24:42 -0400 Subject: [PATCH 4/6] H265RtpDepacketizer: Fix missing header Needed for std::remove_if. --- src/h265rtpdepacketizer.cpp | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/h265rtpdepacketizer.cpp b/src/h265rtpdepacketizer.cpp index 506fecfbf..b32728eb0 100644 --- a/src/h265rtpdepacketizer.cpp +++ b/src/h265rtpdepacketizer.cpp @@ -14,6 +14,8 @@ #include "impl/internals.hpp" +#include + namespace rtc { const binary naluLongStartCode = {byte{0}, byte{0}, byte{0}, byte{1}}; From 2773d01678ec3f563d6ce86ac051d35dec9266d1 Mon Sep 17 00:00:00 2001 From: Robert Edmonds Date: Sun, 27 Oct 2024 20:26:11 -0400 Subject: [PATCH 5/6] H265RtpDepacketizer: Include payload type when creating FrameInfo This follows the API change in #1156 and is based on the very similar change to H264RtpDepacketizer (commit e9060bf3a39cbc0b1264e5db0c903a1a2164972b). --- include/rtc/h265rtpdepacketizer.hpp | 2 +- src/h265rtpdepacketizer.cpp | 9 ++++++--- 2 files changed, 7 insertions(+), 4 deletions(-) diff --git a/include/rtc/h265rtpdepacketizer.hpp b/include/rtc/h265rtpdepacketizer.hpp index 2ebc43501..cb239b958 100644 --- a/include/rtc/h265rtpdepacketizer.hpp +++ b/include/rtc/h265rtpdepacketizer.hpp @@ -39,7 +39,7 @@ class RTC_CPP_EXPORT H265RtpDepacketizer : public MediaHandler { void addSeparator(binary& accessUnit); message_vector buildFrames(message_vector::iterator firstPkt, message_vector::iterator lastPkt, - uint32_t timestamp); + uint8_t payloadType, uint32_t timestamp); }; } // namespace rtc diff --git a/src/h265rtpdepacketizer.cpp b/src/h265rtpdepacketizer.cpp index b32728eb0..06d4b672e 100644 --- a/src/h265rtpdepacketizer.cpp +++ b/src/h265rtpdepacketizer.cpp @@ -57,10 +57,11 @@ void H265RtpDepacketizer::addSeparator(binary& accessUnit) } message_vector H265RtpDepacketizer::buildFrames(message_vector::iterator begin, - message_vector::iterator end, uint32_t timestamp) { + message_vector::iterator end, + uint8_t payloadType, uint32_t timestamp) { message_vector out = {}; auto accessUnit = binary{}; - auto frameInfo = std::make_shared(timestamp); + auto frameInfo = std::make_shared(payloadType, timestamp); auto nFrags = 0; for (auto it = begin; it != end; ++it) { @@ -154,6 +155,7 @@ void H265RtpDepacketizer::incoming(message_vector &messages, const message_callb messages.end()); while (mRtpBuffer.size() != 0) { + uint8_t payload_type = 0; uint32_t current_timestamp = 0; size_t packets_in_timestamp = 0; @@ -162,6 +164,7 @@ void H265RtpDepacketizer::incoming(message_vector &messages, const message_callb if (current_timestamp == 0) { current_timestamp = p->timestamp(); + payload_type = p->payloadType(); // should all be the same for data of the same codec } else if (current_timestamp != p->timestamp()) { break; } @@ -176,7 +179,7 @@ void H265RtpDepacketizer::incoming(message_vector &messages, const message_callb auto begin = mRtpBuffer.begin(); auto end = mRtpBuffer.begin() + (packets_in_timestamp - 1); - auto frames = buildFrames(begin, end + 1, current_timestamp); + auto frames = buildFrames(begin, end + 1, payload_type, current_timestamp); messages.insert(messages.end(), frames.begin(), frames.end()); mRtpBuffer.erase(mRtpBuffer.begin(), mRtpBuffer.begin() + packets_in_timestamp); } From 3dc1a177e0b184d062558c824dd65bc5d1e5ae1d Mon Sep 17 00:00:00 2001 From: Robert Edmonds Date: Sun, 27 Oct 2024 20:37:38 -0400 Subject: [PATCH 6/6] H265RtpDepacketizer: Fix multiple NAL units handling Based on the fix for H264RtpDepacketizer in #1167 (commit 4fc4e9ba822994ddeb5ce1b60075bf6be84a3493). --- src/h265rtpdepacketizer.cpp | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/src/h265rtpdepacketizer.cpp b/src/h265rtpdepacketizer.cpp index 06d4b672e..ae9f4ba7d 100644 --- a/src/h265rtpdepacketizer.cpp +++ b/src/h265rtpdepacketizer.cpp @@ -62,7 +62,6 @@ message_vector H265RtpDepacketizer::buildFrames(message_vector::iterator begin, message_vector out = {}; auto accessUnit = binary{}; auto frameInfo = std::make_shared(payloadType, timestamp); - auto nFrags = 0; for (auto it = begin; it != end; ++it) { auto pkt = it->get(); @@ -87,7 +86,11 @@ message_vector H265RtpDepacketizer::buildFrames(message_vector::iterator begin, auto nalUnitFragmentHeader = H265NalUnitFragmentHeader{ std::to_integer(pkt->at(rtpHeaderSize + sizeof(H265NalUnitHeader)))}; - if (nFrags++ == 0) { + // RFC 7798: "When set to 1, the S bit indicates the start of a fragmented + // NAL unit, i.e., the first byte of the FU payload is also the first byte of + // the payload of the fragmented NAL unit. When the FU payload is not the start + // of the fragmented NAL unit payload, the S bit MUST be set to 0." + if (nalUnitFragmentHeader.isStart() || accessUnit.empty()) { addSeparator(accessUnit); nalUnitHeader.setUnitType(nalUnitFragmentHeader.unitType()); accessUnit.emplace_back(byte(nalUnitHeader._first));