123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409 |
- /*
- * Copyright (c) 2012 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.
- */
- #ifndef VIDEO_RTP_VIDEO_STREAM_RECEIVER_H_
- #define VIDEO_RTP_VIDEO_STREAM_RECEIVER_H_
- #include <atomic>
- #include <list>
- #include <map>
- #include <memory>
- #include <string>
- #include <vector>
- #include "absl/types/optional.h"
- #include "api/array_view.h"
- #include "api/crypto/frame_decryptor_interface.h"
- #include "api/video/color_space.h"
- #include "api/video_codecs/video_codec.h"
- #include "call/rtp_packet_sink_interface.h"
- #include "call/syncable.h"
- #include "call/video_receive_stream.h"
- #include "modules/rtp_rtcp/include/receive_statistics.h"
- #include "modules/rtp_rtcp/include/remote_ntp_time_estimator.h"
- #include "modules/rtp_rtcp/include/rtp_header_extension_map.h"
- #include "modules/rtp_rtcp/include/rtp_rtcp.h"
- #include "modules/rtp_rtcp/include/rtp_rtcp_defines.h"
- #include "modules/rtp_rtcp/source/absolute_capture_time_receiver.h"
- #include "modules/rtp_rtcp/source/rtp_dependency_descriptor_extension.h"
- #include "modules/rtp_rtcp/source/rtp_packet_received.h"
- #include "modules/rtp_rtcp/source/rtp_video_header.h"
- #include "modules/rtp_rtcp/source/video_rtp_depacketizer.h"
- #include "modules/video_coding/h264_sps_pps_tracker.h"
- #include "modules/video_coding/loss_notification_controller.h"
- #include "modules/video_coding/packet_buffer.h"
- #include "modules/video_coding/rtp_frame_reference_finder.h"
- #include "modules/video_coding/unique_timestamp_counter.h"
- #include "rtc_base/constructor_magic.h"
- #include "rtc_base/experiments/field_trial_parser.h"
- #include "rtc_base/numerics/sequence_number_util.h"
- #include "rtc_base/synchronization/mutex.h"
- #include "rtc_base/synchronization/sequence_checker.h"
- #include "rtc_base/system/no_unique_address.h"
- #include "rtc_base/thread_annotations.h"
- #include "rtc_base/thread_checker.h"
- #include "video/buffered_frame_decryptor.h"
- #include "video/rtp_video_stream_receiver_frame_transformer_delegate.h"
- namespace webrtc {
- class DEPRECATED_NackModule;
- class PacketRouter;
- class ProcessThread;
- class ReceiveStatistics;
- class ReceiveStatisticsProxy;
- class RtcpRttStats;
- class RtpPacketReceived;
- class Transport;
- class UlpfecReceiver;
- class RtpVideoStreamReceiver : public LossNotificationSender,
- public RecoveredPacketReceiver,
- public RtpPacketSinkInterface,
- public KeyFrameRequestSender,
- public video_coding::OnCompleteFrameCallback,
- public OnDecryptedFrameCallback,
- public OnDecryptionStatusChangeCallback,
- public RtpVideoFrameReceiver {
- public:
- // DEPRECATED due to dependency on ReceiveStatisticsProxy.
- RtpVideoStreamReceiver(
- Clock* clock,
- Transport* transport,
- RtcpRttStats* rtt_stats,
- // The packet router is optional; if provided, the RtpRtcp module for this
- // stream is registered as a candidate for sending REMB and transport
- // feedback.
- PacketRouter* packet_router,
- const VideoReceiveStream::Config* config,
- ReceiveStatistics* rtp_receive_statistics,
- ReceiveStatisticsProxy* receive_stats_proxy,
- ProcessThread* process_thread,
- NackSender* nack_sender,
- // The KeyFrameRequestSender is optional; if not provided, key frame
- // requests are sent via the internal RtpRtcp module.
- KeyFrameRequestSender* keyframe_request_sender,
- video_coding::OnCompleteFrameCallback* complete_frame_callback,
- rtc::scoped_refptr<FrameDecryptorInterface> frame_decryptor,
- rtc::scoped_refptr<FrameTransformerInterface> frame_transformer);
- RtpVideoStreamReceiver(
- Clock* clock,
- Transport* transport,
- RtcpRttStats* rtt_stats,
- // The packet router is optional; if provided, the RtpRtcp module for this
- // stream is registered as a candidate for sending REMB and transport
- // feedback.
- PacketRouter* packet_router,
- const VideoReceiveStream::Config* config,
- ReceiveStatistics* rtp_receive_statistics,
- RtcpPacketTypeCounterObserver* rtcp_packet_type_counter_observer,
- RtcpCnameCallback* rtcp_cname_callback,
- ProcessThread* process_thread,
- NackSender* nack_sender,
- // The KeyFrameRequestSender is optional; if not provided, key frame
- // requests are sent via the internal RtpRtcp module.
- KeyFrameRequestSender* keyframe_request_sender,
- video_coding::OnCompleteFrameCallback* complete_frame_callback,
- rtc::scoped_refptr<FrameDecryptorInterface> frame_decryptor,
- rtc::scoped_refptr<FrameTransformerInterface> frame_transformer);
- ~RtpVideoStreamReceiver() override;
- void AddReceiveCodec(uint8_t payload_type,
- const VideoCodec& video_codec,
- const std::map<std::string, std::string>& codec_params,
- bool raw_payload);
- void StartReceive();
- void StopReceive();
- // Produces the transport-related timestamps; current_delay_ms is left unset.
- absl::optional<Syncable::Info> GetSyncInfo() const;
- bool DeliverRtcp(const uint8_t* rtcp_packet, size_t rtcp_packet_length);
- void FrameContinuous(int64_t seq_num);
- void FrameDecoded(int64_t seq_num);
- void SignalNetworkState(NetworkState state);
- // Returns number of different frames seen.
- int GetUniqueFramesSeen() const {
- RTC_DCHECK_RUN_ON(&worker_task_checker_);
- return frame_counter_.GetUniqueSeen();
- }
- // Implements RtpPacketSinkInterface.
- void OnRtpPacket(const RtpPacketReceived& packet) override;
- // Public only for tests.
- void OnReceivedPayloadData(rtc::CopyOnWriteBuffer codec_payload,
- const RtpPacketReceived& rtp_packet,
- const RTPVideoHeader& video);
- // Implements RecoveredPacketReceiver.
- void OnRecoveredPacket(const uint8_t* packet, size_t packet_length) override;
- // Send an RTCP keyframe request.
- void RequestKeyFrame() override;
- // Implements LossNotificationSender.
- void SendLossNotification(uint16_t last_decoded_seq_num,
- uint16_t last_received_seq_num,
- bool decodability_flag,
- bool buffering_allowed) override;
- bool IsUlpfecEnabled() const;
- bool IsRetransmissionsEnabled() const;
- // Returns true if a decryptor is attached and frames can be decrypted.
- // Updated by OnDecryptionStatusChangeCallback. Note this refers to Frame
- // Decryption not SRTP.
- bool IsDecryptable() const;
- // Don't use, still experimental.
- void RequestPacketRetransmit(const std::vector<uint16_t>& sequence_numbers);
- // Implements OnCompleteFrameCallback.
- void OnCompleteFrame(
- std::unique_ptr<video_coding::EncodedFrame> frame) override;
- // Implements OnDecryptedFrameCallback.
- void OnDecryptedFrame(
- std::unique_ptr<video_coding::RtpFrameObject> frame) override;
- // Implements OnDecryptionStatusChangeCallback.
- void OnDecryptionStatusChange(
- FrameDecryptorInterface::Status status) override;
- // Optionally set a frame decryptor after a stream has started. This will not
- // reset the decoder state.
- void SetFrameDecryptor(
- rtc::scoped_refptr<FrameDecryptorInterface> frame_decryptor);
- // Sets a frame transformer after a stream has started, if no transformer
- // has previously been set. Does not reset the decoder state.
- void SetDepacketizerToDecoderFrameTransformer(
- rtc::scoped_refptr<FrameTransformerInterface> frame_transformer);
- // Called by VideoReceiveStream when stats are updated.
- void UpdateRtt(int64_t max_rtt_ms);
- absl::optional<int64_t> LastReceivedPacketMs() const;
- absl::optional<int64_t> LastReceivedKeyframePacketMs() const;
- // RtpDemuxer only forwards a given RTP packet to one sink. However, some
- // sinks, such as FlexFEC, might wish to be informed of all of the packets
- // a given sink receives (or any set of sinks). They may do so by registering
- // themselves as secondary sinks.
- void AddSecondarySink(RtpPacketSinkInterface* sink);
- void RemoveSecondarySink(const RtpPacketSinkInterface* sink);
- private:
- // Implements RtpVideoFrameReceiver.
- void ManageFrame(
- std::unique_ptr<video_coding::RtpFrameObject> frame) override;
- // Used for buffering RTCP feedback messages and sending them all together.
- // Note:
- // 1. Key frame requests and NACKs are mutually exclusive, with the
- // former taking precedence over the latter.
- // 2. Loss notifications are orthogonal to either. (That is, may be sent
- // alongside either.)
- class RtcpFeedbackBuffer : public KeyFrameRequestSender,
- public NackSender,
- public LossNotificationSender {
- public:
- RtcpFeedbackBuffer(KeyFrameRequestSender* key_frame_request_sender,
- NackSender* nack_sender,
- LossNotificationSender* loss_notification_sender);
- ~RtcpFeedbackBuffer() override = default;
- // KeyFrameRequestSender implementation.
- void RequestKeyFrame() RTC_LOCKS_EXCLUDED(mutex_) override;
- // NackSender implementation.
- void SendNack(const std::vector<uint16_t>& sequence_numbers,
- bool buffering_allowed) RTC_LOCKS_EXCLUDED(mutex_) override;
- // LossNotificationSender implementation.
- void SendLossNotification(uint16_t last_decoded_seq_num,
- uint16_t last_received_seq_num,
- bool decodability_flag,
- bool buffering_allowed)
- RTC_LOCKS_EXCLUDED(mutex_) override;
- // Send all RTCP feedback messages buffered thus far.
- void SendBufferedRtcpFeedback() RTC_LOCKS_EXCLUDED(mutex_);
- private:
- // LNTF-related state.
- struct LossNotificationState {
- LossNotificationState(uint16_t last_decoded_seq_num,
- uint16_t last_received_seq_num,
- bool decodability_flag)
- : last_decoded_seq_num(last_decoded_seq_num),
- last_received_seq_num(last_received_seq_num),
- decodability_flag(decodability_flag) {}
- uint16_t last_decoded_seq_num;
- uint16_t last_received_seq_num;
- bool decodability_flag;
- };
- struct ConsumedRtcpFeedback {
- bool request_key_frame = false;
- std::vector<uint16_t> nack_sequence_numbers;
- absl::optional<LossNotificationState> lntf_state;
- };
- ConsumedRtcpFeedback ConsumeRtcpFeedback() RTC_LOCKS_EXCLUDED(mutex_);
- ConsumedRtcpFeedback ConsumeRtcpFeedbackLocked()
- RTC_EXCLUSIVE_LOCKS_REQUIRED(mutex_);
- // This method is called both with and without mutex_ held.
- void SendRtcpFeedback(ConsumedRtcpFeedback feedback);
- KeyFrameRequestSender* const key_frame_request_sender_;
- NackSender* const nack_sender_;
- LossNotificationSender* const loss_notification_sender_;
- // NACKs are accessible from two threads due to nack_module_ being a module.
- Mutex mutex_;
- // Key-frame-request-related state.
- bool request_key_frame_ RTC_GUARDED_BY(mutex_);
- // NACK-related state.
- std::vector<uint16_t> nack_sequence_numbers_ RTC_GUARDED_BY(mutex_);
- absl::optional<LossNotificationState> lntf_state_ RTC_GUARDED_BY(mutex_);
- };
- enum ParseGenericDependenciesResult {
- kDropPacket,
- kHasGenericDescriptor,
- kNoGenericDescriptor
- };
- // Entry point doing non-stats work for a received packet. Called
- // for the same packet both before and after RED decapsulation.
- void ReceivePacket(const RtpPacketReceived& packet);
- // Parses and handles RED headers.
- // This function assumes that it's being called from only one thread.
- void ParseAndHandleEncapsulatingHeader(const RtpPacketReceived& packet);
- void NotifyReceiverOfEmptyPacket(uint16_t seq_num);
- void UpdateHistograms();
- bool IsRedEnabled() const;
- void InsertSpsPpsIntoTracker(uint8_t payload_type);
- void OnInsertedPacket(video_coding::PacketBuffer::InsertResult result);
- ParseGenericDependenciesResult ParseGenericDependenciesExtension(
- const RtpPacketReceived& rtp_packet,
- RTPVideoHeader* video_header) RTC_RUN_ON(worker_task_checker_);
- void OnAssembledFrame(std::unique_ptr<video_coding::RtpFrameObject> frame);
- Clock* const clock_;
- // Ownership of this object lies with VideoReceiveStream, which owns |this|.
- const VideoReceiveStream::Config& config_;
- PacketRouter* const packet_router_;
- ProcessThread* const process_thread_;
- RemoteNtpTimeEstimator ntp_estimator_;
- RtpHeaderExtensionMap rtp_header_extensions_;
- // Set by the field trial WebRTC-ForcePlayoutDelay to override any playout
- // delay that is specified in the received packets.
- FieldTrialOptional<int> forced_playout_delay_max_ms_;
- FieldTrialOptional<int> forced_playout_delay_min_ms_;
- ReceiveStatistics* const rtp_receive_statistics_;
- std::unique_ptr<UlpfecReceiver> ulpfec_receiver_;
- RTC_NO_UNIQUE_ADDRESS SequenceChecker worker_task_checker_;
- bool receiving_ RTC_GUARDED_BY(worker_task_checker_);
- int64_t last_packet_log_ms_ RTC_GUARDED_BY(worker_task_checker_);
- const std::unique_ptr<RtpRtcp> rtp_rtcp_;
- video_coding::OnCompleteFrameCallback* complete_frame_callback_;
- KeyFrameRequestSender* const keyframe_request_sender_;
- RtcpFeedbackBuffer rtcp_feedback_buffer_;
- std::unique_ptr<DEPRECATED_NackModule> nack_module_;
- std::unique_ptr<LossNotificationController> loss_notification_controller_;
- video_coding::PacketBuffer packet_buffer_;
- UniqueTimestampCounter frame_counter_ RTC_GUARDED_BY(worker_task_checker_);
- SeqNumUnwrapper<uint16_t> frame_id_unwrapper_
- RTC_GUARDED_BY(worker_task_checker_);
- // Video structure provided in the dependency descriptor in a first packet
- // of a key frame. It is required to parse dependency descriptor in the
- // following delta packets.
- std::unique_ptr<FrameDependencyStructure> video_structure_
- RTC_GUARDED_BY(worker_task_checker_);
- // Frame id of the last frame with the attached video structure.
- // absl::nullopt when `video_structure_ == nullptr`;
- absl::optional<int64_t> video_structure_frame_id_
- RTC_GUARDED_BY(worker_task_checker_);
- Mutex reference_finder_lock_;
- std::unique_ptr<video_coding::RtpFrameReferenceFinder> reference_finder_
- RTC_GUARDED_BY(reference_finder_lock_);
- absl::optional<VideoCodecType> current_codec_;
- uint32_t last_assembled_frame_rtp_timestamp_;
- Mutex last_seq_num_mutex_;
- std::map<int64_t, uint16_t> last_seq_num_for_pic_id_
- RTC_GUARDED_BY(last_seq_num_mutex_);
- video_coding::H264SpsPpsTracker tracker_;
- // Maps payload id to the depacketizer.
- std::map<uint8_t, std::unique_ptr<VideoRtpDepacketizer>> payload_type_map_;
- // TODO(johan): Remove pt_codec_params_ once
- // https://bugs.chromium.org/p/webrtc/issues/detail?id=6883 is resolved.
- // Maps a payload type to a map of out-of-band supplied codec parameters.
- std::map<uint8_t, std::map<std::string, std::string>> pt_codec_params_;
- int16_t last_payload_type_ = -1;
- bool has_received_frame_;
- std::vector<RtpPacketSinkInterface*> secondary_sinks_
- RTC_GUARDED_BY(worker_task_checker_);
- // Info for GetSyncInfo is updated on network or worker thread, and queried on
- // the worker thread.
- mutable Mutex sync_info_lock_;
- absl::optional<uint32_t> last_received_rtp_timestamp_
- RTC_GUARDED_BY(sync_info_lock_);
- absl::optional<int64_t> last_received_rtp_system_time_ms_
- RTC_GUARDED_BY(sync_info_lock_);
- // Used to validate the buffered frame decryptor is always run on the correct
- // thread.
- rtc::ThreadChecker network_tc_;
- // Handles incoming encrypted frames and forwards them to the
- // rtp_reference_finder if they are decryptable.
- std::unique_ptr<BufferedFrameDecryptor> buffered_frame_decryptor_
- RTC_PT_GUARDED_BY(network_tc_);
- std::atomic<bool> frames_decryptable_;
- absl::optional<ColorSpace> last_color_space_;
- AbsoluteCaptureTimeReceiver absolute_capture_time_receiver_
- RTC_GUARDED_BY(worker_task_checker_);
- int64_t last_completed_picture_id_ = 0;
- rtc::scoped_refptr<RtpVideoStreamReceiverFrameTransformerDelegate>
- frame_transformer_delegate_;
- };
- } // namespace webrtc
- #endif // VIDEO_RTP_VIDEO_STREAM_RECEIVER_H_
|