Split WebRtcVideoMediaChannel into Send and Receive
This completes the split-channel work for the Video side. Note: For ease of review, the implementations in the .cc file have not been sorted between sender and receiver. This can be done in a later purely-editorial CL. Bug: webrtc:13931 Change-Id: I36cf015d5facb1eed368070cb204a8763ac19a9c Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/307180 Reviewed-by: Henrik Boström <hbos@webrtc.org> Commit-Queue: Harald Alvestrand <hta@webrtc.org> Cr-Commit-Position: refs/heads/main@{#40207}
This commit is contained in:
parent
4ad141e69b
commit
f785bd46e8
File diff suppressed because it is too large
Load Diff
@ -79,8 +79,6 @@ class VideoEncoderFactory;
|
||||
|
||||
namespace cricket {
|
||||
|
||||
class WebRtcVideoChannel;
|
||||
|
||||
// Public for testing.
|
||||
// Inputs StreamStats for all types of substreams (kMedia, kRtx, kFlexfec) and
|
||||
// merges any non-kMedia substream stats object into its referenced kMedia-type
|
||||
@ -134,12 +132,32 @@ class WebRtcVideoEngine : public VideoEngineInterface {
|
||||
const webrtc::FieldTrialsView& trials_;
|
||||
};
|
||||
|
||||
class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
public webrtc::Transport,
|
||||
public webrtc::EncoderSwitchRequestCallback {
|
||||
struct VideoCodecSettings {
|
||||
VideoCodecSettings();
|
||||
|
||||
// Checks if all members of |*this| are equal to the corresponding members
|
||||
// of `other`.
|
||||
bool operator==(const VideoCodecSettings& other) const;
|
||||
bool operator!=(const VideoCodecSettings& other) const;
|
||||
|
||||
// Checks if all members of `a`, except `flexfec_payload_type`, are equal
|
||||
// to the corresponding members of `b`.
|
||||
static bool EqualsDisregardingFlexfec(const VideoCodecSettings& a,
|
||||
const VideoCodecSettings& b);
|
||||
|
||||
VideoCodec codec;
|
||||
webrtc::UlpfecConfig ulpfec;
|
||||
int flexfec_payload_type; // -1 if absent.
|
||||
int rtx_payload_type; // -1 if absent.
|
||||
absl::optional<int> rtx_time;
|
||||
};
|
||||
|
||||
class WebRtcVideoSendChannel : public MediaChannelUtil,
|
||||
public VideoMediaSendChannelInterface,
|
||||
public webrtc::Transport,
|
||||
public webrtc::EncoderSwitchRequestCallback {
|
||||
public:
|
||||
WebRtcVideoChannel(
|
||||
MediaChannel::Role role,
|
||||
WebRtcVideoSendChannel(
|
||||
webrtc::Call* call,
|
||||
const MediaConfig& config,
|
||||
const VideoOptions& options,
|
||||
@ -147,20 +165,36 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
webrtc::VideoEncoderFactory* encoder_factory,
|
||||
webrtc::VideoDecoderFactory* decoder_factory,
|
||||
webrtc::VideoBitrateAllocatorFactory* bitrate_allocator_factory);
|
||||
~WebRtcVideoChannel() override;
|
||||
~WebRtcVideoSendChannel() override;
|
||||
|
||||
// VideoMediaChannel implementation
|
||||
MediaType media_type() const override { return MEDIA_TYPE_VIDEO; }
|
||||
// Type manipulations
|
||||
VideoMediaSendChannelInterface* AsVideoSendChannel() override { return this; }
|
||||
VoiceMediaSendChannelInterface* AsVoiceSendChannel() override {
|
||||
RTC_CHECK_NOTREACHED();
|
||||
return nullptr;
|
||||
}
|
||||
// Functions imported from MediaChannelUtil
|
||||
bool HasNetworkInterface() const override {
|
||||
return MediaChannelUtil::HasNetworkInterface();
|
||||
}
|
||||
void SetExtmapAllowMixed(bool extmap_allow_mixed) override {
|
||||
MediaChannelUtil::SetExtmapAllowMixed(extmap_allow_mixed);
|
||||
}
|
||||
bool ExtmapAllowMixed() const override {
|
||||
return MediaChannelUtil::ExtmapAllowMixed();
|
||||
}
|
||||
|
||||
// Common functions between sender and receiver
|
||||
void SetInterface(MediaChannelNetworkInterface* iface) override;
|
||||
// VideoMediaSendChannelInterface implementation
|
||||
bool SetSendParameters(const VideoSendParameters& params) override;
|
||||
bool SetRecvParameters(const VideoRecvParameters& params) override;
|
||||
webrtc::RtpParameters GetRtpSendParameters(uint32_t ssrc) const override;
|
||||
webrtc::RTCError SetRtpSendParameters(
|
||||
uint32_t ssrc,
|
||||
const webrtc::RtpParameters& parameters,
|
||||
webrtc::SetParametersCallback callback) override;
|
||||
webrtc::RtpParameters GetRtpReceiveParameters(uint32_t ssrc) const override;
|
||||
webrtc::RtpParameters GetDefaultRtpReceiveParameters() const override;
|
||||
webrtc::RtpParameters GetRtpSendParameters(uint32_t ssrc) const override;
|
||||
bool GetSendCodec(VideoCodec* send_codec) override;
|
||||
void SetReceive(bool receive) override;
|
||||
bool SetSend(bool send) override;
|
||||
bool SetVideoSend(
|
||||
uint32_t ssrc,
|
||||
@ -168,38 +202,14 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
rtc::VideoSourceInterface<webrtc::VideoFrame>* source) override;
|
||||
bool AddSendStream(const StreamParams& sp) override;
|
||||
bool RemoveSendStream(uint32_t ssrc) override;
|
||||
bool AddRecvStream(const StreamParams& sp) override;
|
||||
bool AddDefaultRecvStreamForTesting(const StreamParams& sp) override {
|
||||
// Invokes private AddRecvStream variant function
|
||||
return AddRecvStream(sp, true);
|
||||
}
|
||||
bool RemoveRecvStream(uint32_t ssrc) override;
|
||||
void ResetUnsignaledRecvStream() override;
|
||||
absl::optional<uint32_t> GetUnsignaledSsrc() const override;
|
||||
void OnDemuxerCriteriaUpdatePending() override;
|
||||
void OnDemuxerCriteriaUpdateComplete() override;
|
||||
bool SetSink(uint32_t ssrc,
|
||||
rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) override;
|
||||
void SetDefaultSink(
|
||||
rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) override;
|
||||
void FillBitrateInfo(BandwidthEstimationInfo* bwe_info) override;
|
||||
bool GetSendStats(VideoMediaSendInfo* info) override;
|
||||
bool GetReceiveStats(VideoMediaReceiveInfo* info) override;
|
||||
bool GetStats(VideoMediaSendInfo* info) override;
|
||||
|
||||
void OnPacketReceived(const webrtc::RtpPacketReceived& packet) override;
|
||||
void OnPacketSent(const rtc::SentPacket& sent_packet) override;
|
||||
void OnReadyToSend(bool ready) override;
|
||||
void OnNetworkRouteChanged(absl::string_view transport_name,
|
||||
const rtc::NetworkRoute& network_route) override;
|
||||
void SetInterface(MediaChannelNetworkInterface* iface) override;
|
||||
|
||||
// E2E Encrypted Video Frame API
|
||||
// Set a frame decryptor to a particular ssrc that will intercept all
|
||||
// incoming video frames and attempt to decrypt them before forwarding the
|
||||
// result.
|
||||
void SetFrameDecryptor(uint32_t ssrc,
|
||||
rtc::scoped_refptr<webrtc::FrameDecryptorInterface>
|
||||
frame_decryptor) override;
|
||||
// Set a frame encryptor to a particular ssrc that will intercept all
|
||||
// outgoing video frames and attempt to encrypt them and forward the result
|
||||
// to the packetizer.
|
||||
@ -215,11 +225,6 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
|
||||
void SetVideoCodecSwitchingEnabled(bool enabled) override;
|
||||
|
||||
bool SetBaseMinimumPlayoutDelayMs(uint32_t ssrc, int delay_ms) override;
|
||||
|
||||
absl::optional<int> GetBaseMinimumPlayoutDelayMs(
|
||||
uint32_t ssrc) const override;
|
||||
|
||||
void SetSendCodecChangedCallback(
|
||||
absl::AnyInvocable<void()> callback) override {
|
||||
send_codec_changed_callback_ = std::move(callback);
|
||||
@ -230,21 +235,12 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
ssrc_list_changed_callback_ = std::move(callback);
|
||||
}
|
||||
|
||||
// Choose one of the available SSRCs (or default if none) as the current
|
||||
// receiver report SSRC.
|
||||
void ChooseReceiverReportSsrc(const std::set<uint32_t>& choices) override;
|
||||
|
||||
// Implemented for VideoMediaChannelTest.
|
||||
bool sending() const {
|
||||
RTC_DCHECK_RUN_ON(&thread_checker_);
|
||||
return sending_;
|
||||
}
|
||||
|
||||
StreamParams unsignaled_stream_params() {
|
||||
RTC_DCHECK_RUN_ON(&thread_checker_);
|
||||
return unsignaled_stream_params_;
|
||||
}
|
||||
|
||||
// AdaptReason is used for expressing why a WebRtcVideoSendStream request
|
||||
// a lower input frame size than the currently configured camera input frame
|
||||
// size. There can be more than one reason OR:ed together.
|
||||
@ -256,19 +252,11 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
|
||||
static constexpr int kDefaultQpMax = 56;
|
||||
|
||||
std::vector<webrtc::RtpSource> GetSources(uint32_t ssrc) const override;
|
||||
|
||||
// Implements webrtc::EncoderSwitchRequestCallback.
|
||||
void RequestEncoderFallback() override;
|
||||
void RequestEncoderSwitch(const webrtc::SdpVideoFormat& format,
|
||||
bool allow_default_fallback) override;
|
||||
|
||||
void SetRecordableEncodedFrameCallback(
|
||||
uint32_t ssrc,
|
||||
std::function<void(const webrtc::RecordableEncodedFrame&)> callback)
|
||||
override;
|
||||
void ClearRecordableEncodedFrameCallback(uint32_t ssrc) override;
|
||||
void RequestRecvKeyFrame(uint32_t ssrc) override;
|
||||
void GenerateSendKeyFrame(uint32_t ssrc,
|
||||
const std::vector<std::string>& rids) override;
|
||||
|
||||
@ -276,11 +264,6 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
uint32_t ssrc,
|
||||
rtc::scoped_refptr<webrtc::FrameTransformerInterface> frame_transformer)
|
||||
override;
|
||||
void SetDepacketizerToDecoderFrameTransformer(
|
||||
uint32_t ssrc,
|
||||
rtc::scoped_refptr<webrtc::FrameTransformerInterface> frame_transformer)
|
||||
override;
|
||||
|
||||
// Information queries to support SetReceiverFeedbackParameters
|
||||
webrtc::RtcpMode SendCodecRtcpMode() const override {
|
||||
RTC_DCHECK_RUN_ON(&thread_checker_);
|
||||
@ -309,34 +292,8 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
}
|
||||
return send_codec()->rtx_time;
|
||||
}
|
||||
void SetReceiverFeedbackParameters(bool lntf_enabled,
|
||||
bool nack_enabled,
|
||||
webrtc::RtcpMode rtcp_mode,
|
||||
absl::optional<int> rtx_time) override;
|
||||
|
||||
private:
|
||||
class WebRtcVideoReceiveStream;
|
||||
|
||||
struct VideoCodecSettings {
|
||||
VideoCodecSettings();
|
||||
|
||||
// Checks if all members of |*this| are equal to the corresponding members
|
||||
// of `other`.
|
||||
bool operator==(const VideoCodecSettings& other) const;
|
||||
bool operator!=(const VideoCodecSettings& other) const;
|
||||
|
||||
// Checks if all members of `a`, except `flexfec_payload_type`, are equal
|
||||
// to the corresponding members of `b`.
|
||||
static bool EqualsDisregardingFlexfec(const VideoCodecSettings& a,
|
||||
const VideoCodecSettings& b);
|
||||
|
||||
VideoCodec codec;
|
||||
webrtc::UlpfecConfig ulpfec;
|
||||
int flexfec_payload_type; // -1 if absent.
|
||||
int rtx_payload_type; // -1 if absent.
|
||||
absl::optional<int> rtx_time;
|
||||
};
|
||||
|
||||
struct ChangedSendParameters {
|
||||
// These optionals are unset if not changed.
|
||||
absl::optional<VideoCodecSettings> send_codec;
|
||||
@ -349,57 +306,12 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
absl::optional<webrtc::RtcpMode> rtcp_mode;
|
||||
};
|
||||
|
||||
struct ChangedRecvParameters {
|
||||
// These optionals are unset if not changed.
|
||||
absl::optional<std::vector<VideoCodecSettings>> codec_settings;
|
||||
absl::optional<std::vector<webrtc::RtpExtension>> rtp_header_extensions;
|
||||
// Keep track of the FlexFEC payload type separately from `codec_settings`.
|
||||
// This allows us to recreate the FlexfecReceiveStream separately from the
|
||||
// VideoReceiveStreamInterface when the FlexFEC payload type is changed.
|
||||
absl::optional<int> flexfec_payload_type;
|
||||
};
|
||||
|
||||
// Finds VideoReceiveStreamInterface corresponding to ssrc. Aware of
|
||||
// unsignalled ssrc handling.
|
||||
WebRtcVideoReceiveStream* FindReceiveStream(uint32_t ssrc)
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
|
||||
void ProcessReceivedPacket(webrtc::RtpPacketReceived packet)
|
||||
RTC_RUN_ON(thread_checker_);
|
||||
|
||||
bool GetChangedSendParameters(const VideoSendParameters& params,
|
||||
ChangedSendParameters* changed_params) const
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
bool ApplyChangedParams(const ChangedSendParameters& changed_params);
|
||||
bool GetChangedRecvParameters(const VideoRecvParameters& params,
|
||||
ChangedRecvParameters* changed_params) const
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
|
||||
// Expected to be invoked once per packet that belongs to this channel that
|
||||
// can not be demuxed.
|
||||
// Returns true if a new default stream has been created.
|
||||
bool MaybeCreateDefaultReceiveStream(
|
||||
const webrtc::RtpPacketReceived& parsed_packet)
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
void ReCreateDefaultReceiveStream(uint32_t ssrc,
|
||||
absl::optional<uint32_t> rtx_ssrc);
|
||||
// Add a receive stream. Used for testing.
|
||||
bool AddRecvStream(const StreamParams& sp, bool default_stream);
|
||||
|
||||
void ConfigureReceiverRtp(
|
||||
webrtc::VideoReceiveStreamInterface::Config* config,
|
||||
webrtc::FlexfecReceiveStream::Config* flexfec_config,
|
||||
const StreamParams& sp) const
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
bool ValidateSendSsrcAvailability(const StreamParams& sp) const
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
bool ValidateReceiveSsrcAvailability(const StreamParams& sp) const
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
void DeleteReceiveStream(WebRtcVideoReceiveStream* stream)
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
|
||||
static std::string CodecSettingsVectorToString(
|
||||
const std::vector<VideoCodecSettings>& codecs);
|
||||
|
||||
// Populates `rtx_associated_payload_types`, `raw_payload_types` and
|
||||
// `decoders` based on codec settings provided by `recv_codecs`.
|
||||
@ -410,10 +322,6 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
std::set<int>& raw_payload_types,
|
||||
std::vector<webrtc::VideoReceiveStreamInterface::Decoder>& decoders);
|
||||
|
||||
// Called when the local ssrc changes. Sets `rtcp_receiver_report_ssrc_` and
|
||||
// updates the receive streams.
|
||||
void SetReceiverReportSsrc(uint32_t ssrc) RTC_RUN_ON(&thread_checker_);
|
||||
|
||||
// Wrapper for the sender part.
|
||||
class WebRtcVideoSendStream {
|
||||
public:
|
||||
@ -529,6 +437,277 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
const bool disable_automatic_resize_;
|
||||
};
|
||||
|
||||
void Construct(webrtc::Call* call, WebRtcVideoEngine* engine);
|
||||
|
||||
bool SendRtp(const uint8_t* data,
|
||||
size_t len,
|
||||
const webrtc::PacketOptions& options) override;
|
||||
bool SendRtcp(const uint8_t* data, size_t len) override;
|
||||
|
||||
// Get all codecs that are compatible with the receiver.
|
||||
std::vector<VideoCodecSettings> SelectSendVideoCodecs(
|
||||
const std::vector<VideoCodecSettings>& remote_mapped_codecs) const
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
|
||||
void FillSenderStats(VideoMediaSendInfo* info, bool log_stats)
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
void FillBandwidthEstimationStats(const webrtc::Call::Stats& stats,
|
||||
VideoMediaInfo* info)
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
void FillSendCodecStats(VideoMediaSendInfo* video_media_info)
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
|
||||
// Accessor function for send_codec_. Introduced in order to ensure
|
||||
// that a receive channel does not touch the send codec directly.
|
||||
// Can go away once these are different classes.
|
||||
// TODO(bugs.webrtc.org/13931): Remove this function
|
||||
absl::optional<VideoCodecSettings>& send_codec() { return send_codec_; }
|
||||
const absl::optional<VideoCodecSettings>& send_codec() const {
|
||||
return send_codec_;
|
||||
}
|
||||
// Disabled function from interface
|
||||
MediaChannel* ImplForTesting() override {
|
||||
RTC_CHECK_NOTREACHED();
|
||||
return nullptr;
|
||||
}
|
||||
webrtc::TaskQueueBase* const worker_thread_;
|
||||
webrtc::ScopedTaskSafety task_safety_;
|
||||
RTC_NO_UNIQUE_ADDRESS webrtc::SequenceChecker network_thread_checker_{
|
||||
webrtc::SequenceChecker::kDetached};
|
||||
RTC_NO_UNIQUE_ADDRESS webrtc::SequenceChecker thread_checker_;
|
||||
|
||||
uint32_t rtcp_receiver_report_ssrc_ RTC_GUARDED_BY(thread_checker_);
|
||||
bool sending_ RTC_GUARDED_BY(thread_checker_);
|
||||
bool receiving_ RTC_GUARDED_BY(&thread_checker_);
|
||||
webrtc::Call* const call_;
|
||||
|
||||
rtc::VideoSinkInterface<webrtc::VideoFrame>* default_sink_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
|
||||
// Delay for unsignaled streams, which may be set before the stream exists.
|
||||
int default_recv_base_minimum_delay_ms_ RTC_GUARDED_BY(thread_checker_) = 0;
|
||||
|
||||
const MediaConfig::Video video_config_ RTC_GUARDED_BY(thread_checker_);
|
||||
|
||||
// Using primary-ssrc (first ssrc) as key.
|
||||
std::map<uint32_t, WebRtcVideoSendStream*> send_streams_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
// When the channel and demuxer get reconfigured, there is a window of time
|
||||
// where we have to be prepared for packets arriving based on the old demuxer
|
||||
// criteria because the streams live on the worker thread and the demuxer
|
||||
// lives on the network thread. Because packets are posted from the network
|
||||
// thread to the worker thread, they can still be in-flight when streams are
|
||||
// reconfgured. This can happen when `demuxer_criteria_id_` and
|
||||
// `demuxer_criteria_completed_id_` don't match. During this time, we do not
|
||||
// want to create unsignalled receive streams and should instead drop the
|
||||
// packets. E.g:
|
||||
// * If RemoveRecvStream(old_ssrc) was recently called, there may be packets
|
||||
// in-flight for that ssrc. This happens when a receiver becomes inactive.
|
||||
// * If we go from one to many m= sections, the demuxer may change from
|
||||
// forwarding all packets to only forwarding the configured ssrcs, so there
|
||||
// is a risk of receiving ssrcs for other, recently added m= sections.
|
||||
uint32_t demuxer_criteria_id_ RTC_GUARDED_BY(thread_checker_) = 0;
|
||||
uint32_t demuxer_criteria_completed_id_ RTC_GUARDED_BY(thread_checker_) = 0;
|
||||
absl::optional<int64_t> last_unsignalled_ssrc_creation_time_ms_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
std::set<uint32_t> send_ssrcs_ RTC_GUARDED_BY(thread_checker_);
|
||||
std::set<uint32_t> receive_ssrcs_ RTC_GUARDED_BY(thread_checker_);
|
||||
|
||||
absl::optional<VideoCodecSettings> send_codec_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
std::vector<VideoCodecSettings> negotiated_codecs_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
|
||||
std::vector<webrtc::RtpExtension> send_rtp_extensions_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
|
||||
webrtc::VideoEncoderFactory* const encoder_factory_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
webrtc::VideoDecoderFactory* const decoder_factory_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
webrtc::VideoBitrateAllocatorFactory* const bitrate_allocator_factory_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
std::vector<VideoCodecSettings> recv_codecs_ RTC_GUARDED_BY(thread_checker_);
|
||||
webrtc::RtpHeaderExtensionMap recv_rtp_extension_map_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
std::vector<webrtc::RtpExtension> recv_rtp_extensions_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
// See reason for keeping track of the FlexFEC payload type separately in
|
||||
// comment in WebRtcVideoChannel::ChangedRecvParameters.
|
||||
int recv_flexfec_payload_type_ RTC_GUARDED_BY(thread_checker_);
|
||||
webrtc::BitrateConstraints bitrate_config_ RTC_GUARDED_BY(thread_checker_);
|
||||
// TODO(deadbeef): Don't duplicate information between
|
||||
// send_params/recv_params, rtp_extensions, options, etc.
|
||||
VideoSendParameters send_params_ RTC_GUARDED_BY(thread_checker_);
|
||||
VideoOptions default_send_options_ RTC_GUARDED_BY(thread_checker_);
|
||||
VideoRecvParameters recv_params_ RTC_GUARDED_BY(thread_checker_);
|
||||
int64_t last_send_stats_log_ms_ RTC_GUARDED_BY(thread_checker_);
|
||||
int64_t last_receive_stats_log_ms_ RTC_GUARDED_BY(thread_checker_);
|
||||
const bool discard_unknown_ssrc_packets_ RTC_GUARDED_BY(thread_checker_);
|
||||
// This is a stream param that comes from the remote description, but wasn't
|
||||
// signaled with any a=ssrc lines. It holds information that was signaled
|
||||
// before the unsignaled receive stream is created when the first packet is
|
||||
// received.
|
||||
StreamParams unsignaled_stream_params_ RTC_GUARDED_BY(thread_checker_);
|
||||
// Per peer connection crypto options that last for the lifetime of the peer
|
||||
// connection.
|
||||
const webrtc::CryptoOptions crypto_options_ RTC_GUARDED_BY(thread_checker_);
|
||||
|
||||
// Optional frame transformer set on unsignaled streams.
|
||||
rtc::scoped_refptr<webrtc::FrameTransformerInterface>
|
||||
unsignaled_frame_transformer_ RTC_GUARDED_BY(thread_checker_);
|
||||
|
||||
// TODO(bugs.webrtc.org/11341): Remove this and relevant PC API. Presence
|
||||
// of multiple negotiated codecs allows generic encoder fallback on failures.
|
||||
// Presence of EncoderSelector allows switching to specific encoders.
|
||||
bool allow_codec_switching_ = false;
|
||||
|
||||
// RTP parameters that need to be set when creating a video receive stream.
|
||||
// Only used in Receiver mode - in Both mode, it reads those things from the
|
||||
// codec.
|
||||
webrtc::VideoReceiveStreamInterface::Config::Rtp rtp_config_;
|
||||
|
||||
// Callback invoked whenever the send codec changes.
|
||||
// TODO(bugs.webrtc.org/13931): Remove again when coupling isn't needed.
|
||||
absl::AnyInvocable<void()> send_codec_changed_callback_;
|
||||
// Callback invoked whenever the list of SSRCs changes.
|
||||
absl::AnyInvocable<void(const std::set<uint32_t>&)>
|
||||
ssrc_list_changed_callback_;
|
||||
};
|
||||
|
||||
class WebRtcVideoReceiveChannel : public MediaChannelUtil,
|
||||
public VideoMediaReceiveChannelInterface,
|
||||
public webrtc::Transport {
|
||||
public:
|
||||
WebRtcVideoReceiveChannel(webrtc::Call* call,
|
||||
const MediaConfig& config,
|
||||
const VideoOptions& options,
|
||||
const webrtc::CryptoOptions& crypto_options,
|
||||
webrtc::VideoDecoderFactory* decoder_factory);
|
||||
~WebRtcVideoReceiveChannel() override;
|
||||
|
||||
public:
|
||||
MediaType media_type() const override { return MEDIA_TYPE_VIDEO; }
|
||||
VideoMediaReceiveChannelInterface* AsVideoReceiveChannel() override {
|
||||
return this;
|
||||
}
|
||||
VoiceMediaReceiveChannelInterface* AsVoiceReceiveChannel() override {
|
||||
RTC_CHECK_NOTREACHED();
|
||||
return nullptr;
|
||||
}
|
||||
// Functions imported from MediaChannelUtil
|
||||
bool SendRtp(const uint8_t* data,
|
||||
size_t len,
|
||||
const webrtc::PacketOptions& options) override {
|
||||
MediaChannelUtil::SendRtp(data, len, options);
|
||||
return true;
|
||||
}
|
||||
bool SendRtcp(const uint8_t* data, size_t len) override {
|
||||
MediaChannelUtil::SendRtcp(data, len);
|
||||
return true;
|
||||
}
|
||||
|
||||
// Common functions between sender and receiver
|
||||
void SetInterface(MediaChannelNetworkInterface* iface) override;
|
||||
// VideoMediaReceiveChannelInterface implementation
|
||||
bool SetRecvParameters(const VideoRecvParameters& params) override;
|
||||
webrtc::RtpParameters GetRtpReceiveParameters(uint32_t ssrc) const override;
|
||||
webrtc::RtpParameters GetDefaultRtpReceiveParameters() const override;
|
||||
void SetReceive(bool receive) override;
|
||||
bool AddRecvStream(const StreamParams& sp) override;
|
||||
bool AddDefaultRecvStreamForTesting(const StreamParams& sp) override {
|
||||
// Invokes private AddRecvStream variant function
|
||||
return AddRecvStream(sp, true);
|
||||
}
|
||||
bool RemoveRecvStream(uint32_t ssrc) override;
|
||||
void ResetUnsignaledRecvStream() override;
|
||||
absl::optional<uint32_t> GetUnsignaledSsrc() const override;
|
||||
void OnDemuxerCriteriaUpdatePending() override;
|
||||
void OnDemuxerCriteriaUpdateComplete() override;
|
||||
bool SetSink(uint32_t ssrc,
|
||||
rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) override;
|
||||
void SetDefaultSink(
|
||||
rtc::VideoSinkInterface<webrtc::VideoFrame>* sink) override;
|
||||
bool GetStats(VideoMediaReceiveInfo* info) override;
|
||||
void OnPacketReceived(const webrtc::RtpPacketReceived& packet) override;
|
||||
bool SetBaseMinimumPlayoutDelayMs(uint32_t ssrc, int delay_ms) override;
|
||||
|
||||
absl::optional<int> GetBaseMinimumPlayoutDelayMs(
|
||||
uint32_t ssrc) const override;
|
||||
|
||||
// Choose one of the available SSRCs (or default if none) as the current
|
||||
// receiver report SSRC.
|
||||
void ChooseReceiverReportSsrc(const std::set<uint32_t>& choices) override;
|
||||
|
||||
// E2E Encrypted Video Frame API
|
||||
// Set a frame decryptor to a particular ssrc that will intercept all
|
||||
// incoming video frames and attempt to decrypt them before forwarding the
|
||||
// result.
|
||||
void SetFrameDecryptor(uint32_t ssrc,
|
||||
rtc::scoped_refptr<webrtc::FrameDecryptorInterface>
|
||||
frame_decryptor) override;
|
||||
void SetRecordableEncodedFrameCallback(
|
||||
uint32_t ssrc,
|
||||
std::function<void(const webrtc::RecordableEncodedFrame&)> callback)
|
||||
override;
|
||||
void ClearRecordableEncodedFrameCallback(uint32_t ssrc) override;
|
||||
void RequestRecvKeyFrame(uint32_t ssrc) override;
|
||||
void SetDepacketizerToDecoderFrameTransformer(
|
||||
uint32_t ssrc,
|
||||
rtc::scoped_refptr<webrtc::FrameTransformerInterface> frame_transformer)
|
||||
override;
|
||||
std::vector<webrtc::RtpSource> GetSources(uint32_t ssrc) const override;
|
||||
|
||||
void SetReceiverFeedbackParameters(bool lntf_enabled,
|
||||
bool nack_enabled,
|
||||
webrtc::RtcpMode rtcp_mode,
|
||||
absl::optional<int> rtx_time) override;
|
||||
|
||||
private:
|
||||
class WebRtcVideoReceiveStream;
|
||||
struct ChangedRecvParameters {
|
||||
// These optionals are unset if not changed.
|
||||
absl::optional<std::vector<VideoCodecSettings>> codec_settings;
|
||||
absl::optional<std::vector<webrtc::RtpExtension>> rtp_header_extensions;
|
||||
// Keep track of the FlexFEC payload type separately from `codec_settings`.
|
||||
// This allows us to recreate the FlexfecReceiveStream separately from the
|
||||
// VideoReceiveStreamInterface when the FlexFEC payload type is changed.
|
||||
absl::optional<int> flexfec_payload_type;
|
||||
};
|
||||
|
||||
// Finds VideoReceiveStreamInterface corresponding to ssrc. Aware of
|
||||
// unsignalled ssrc handling.
|
||||
WebRtcVideoReceiveStream* FindReceiveStream(uint32_t ssrc)
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
|
||||
void ProcessReceivedPacket(webrtc::RtpPacketReceived packet)
|
||||
RTC_RUN_ON(thread_checker_);
|
||||
|
||||
// Expected to be invoked once per packet that belongs to this channel that
|
||||
// can not be demuxed.
|
||||
// Returns true if a new default stream has been created.
|
||||
bool MaybeCreateDefaultReceiveStream(
|
||||
const webrtc::RtpPacketReceived& parsed_packet)
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
void ReCreateDefaultReceiveStream(uint32_t ssrc,
|
||||
absl::optional<uint32_t> rtx_ssrc);
|
||||
// Add a receive stream. Used for testing.
|
||||
bool AddRecvStream(const StreamParams& sp, bool default_stream);
|
||||
|
||||
void ConfigureReceiverRtp(
|
||||
webrtc::VideoReceiveStreamInterface::Config* config,
|
||||
webrtc::FlexfecReceiveStream::Config* flexfec_config,
|
||||
const StreamParams& sp) const
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
bool ValidateReceiveSsrcAvailability(const StreamParams& sp) const
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
void DeleteReceiveStream(WebRtcVideoReceiveStream* stream)
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
|
||||
// Called when the local ssrc changes. Sets `rtcp_receiver_report_ssrc_` and
|
||||
// updates the receive streams.
|
||||
void SetReceiverReportSsrc(uint32_t ssrc) RTC_RUN_ON(&thread_checker_);
|
||||
|
||||
// Wrapper for the receiver part, contains configs etc. that are needed to
|
||||
// reconstruct the underlying VideoReceiveStreamInterface.
|
||||
class WebRtcVideoReceiveStream
|
||||
@ -627,56 +806,28 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
RTC_NO_UNIQUE_ADDRESS webrtc::SequenceChecker thread_checker_;
|
||||
bool receiving_ RTC_GUARDED_BY(&thread_checker_);
|
||||
};
|
||||
|
||||
void Construct(webrtc::Call* call, WebRtcVideoEngine* engine);
|
||||
|
||||
bool SendRtp(const uint8_t* data,
|
||||
size_t len,
|
||||
const webrtc::PacketOptions& options) override;
|
||||
bool SendRtcp(const uint8_t* data, size_t len) override;
|
||||
|
||||
// Generate the list of codec parameters to pass down based on the negotiated
|
||||
// "codecs". Note that VideoCodecSettings correspond to concrete codecs like
|
||||
// VP8, VP9, H264 while VideoCodecs correspond also to "virtual" codecs like
|
||||
// RTX, ULPFEC, FLEXFEC.
|
||||
static std::vector<VideoCodecSettings> MapCodecs(
|
||||
const std::vector<VideoCodec>& codecs);
|
||||
// Get all codecs that are compatible with the receiver.
|
||||
std::vector<VideoCodecSettings> SelectSendVideoCodecs(
|
||||
const std::vector<VideoCodecSettings>& remote_mapped_codecs) const
|
||||
bool GetChangedRecvParameters(const VideoRecvParameters& params,
|
||||
ChangedRecvParameters* changed_params) const
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
|
||||
static bool NonFlexfecReceiveCodecsHaveChanged(
|
||||
std::vector<VideoCodecSettings> before,
|
||||
std::vector<VideoCodecSettings> after);
|
||||
|
||||
void FillSenderStats(VideoMediaSendInfo* info, bool log_stats)
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
std::map<uint32_t, WebRtcVideoReceiveStream*> receive_streams_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
void FillReceiverStats(VideoMediaReceiveInfo* info, bool log_stats)
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
void FillBandwidthEstimationStats(const webrtc::Call::Stats& stats,
|
||||
VideoMediaInfo* info)
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
void FillSendCodecStats(VideoMediaSendInfo* video_media_info)
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
void FillReceiveCodecStats(VideoMediaReceiveInfo* video_media_info)
|
||||
RTC_EXCLUSIVE_LOCKS_REQUIRED(thread_checker_);
|
||||
|
||||
// Accessor function for send_codec_. Introduced in order to ensure
|
||||
// that a receive channel does not touch the send codec directly.
|
||||
// Can go away once these are different classes.
|
||||
// TODO(bugs.webrtc.org/13931): Remove this function
|
||||
absl::optional<VideoCodecSettings>& send_codec() {
|
||||
RTC_DCHECK(role() == MediaChannel::Role::kSend ||
|
||||
role() == MediaChannel::Role::kBoth);
|
||||
return send_codec_;
|
||||
}
|
||||
const absl::optional<VideoCodecSettings>& send_codec() const {
|
||||
RTC_DCHECK(role() == MediaChannel::Role::kSend ||
|
||||
role() == MediaChannel::Role::kBoth);
|
||||
return send_codec_;
|
||||
StreamParams unsignaled_stream_params() {
|
||||
RTC_DCHECK_RUN_ON(&thread_checker_);
|
||||
return unsignaled_stream_params_;
|
||||
}
|
||||
|
||||
// Disabled function from API
|
||||
MediaChannel* ImplForTesting() override {
|
||||
RTC_CHECK_NOTREACHED();
|
||||
return nullptr;
|
||||
}
|
||||
// Variables.
|
||||
webrtc::TaskQueueBase* const worker_thread_;
|
||||
webrtc::ScopedTaskSafety task_safety_;
|
||||
RTC_NO_UNIQUE_ADDRESS webrtc::SequenceChecker network_thread_checker_{
|
||||
@ -684,7 +835,6 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
RTC_NO_UNIQUE_ADDRESS webrtc::SequenceChecker thread_checker_;
|
||||
|
||||
uint32_t rtcp_receiver_report_ssrc_ RTC_GUARDED_BY(thread_checker_);
|
||||
bool sending_ RTC_GUARDED_BY(thread_checker_);
|
||||
bool receiving_ RTC_GUARDED_BY(&thread_checker_);
|
||||
webrtc::Call* const call_;
|
||||
|
||||
@ -696,11 +846,6 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
|
||||
const MediaConfig::Video video_config_ RTC_GUARDED_BY(thread_checker_);
|
||||
|
||||
// Using primary-ssrc (first ssrc) as key.
|
||||
std::map<uint32_t, WebRtcVideoSendStream*> send_streams_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
std::map<uint32_t, WebRtcVideoReceiveStream*> receive_streams_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
// When the channel and demuxer get reconfigured, there is a window of time
|
||||
// where we have to be prepared for packets arriving based on the old demuxer
|
||||
// criteria because the streams live on the worker thread and the demuxer
|
||||
@ -730,12 +875,8 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
std::vector<webrtc::RtpExtension> send_rtp_extensions_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
|
||||
webrtc::VideoEncoderFactory* const encoder_factory_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
webrtc::VideoDecoderFactory* const decoder_factory_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
webrtc::VideoBitrateAllocatorFactory* const bitrate_allocator_factory_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
std::vector<VideoCodecSettings> recv_codecs_ RTC_GUARDED_BY(thread_checker_);
|
||||
webrtc::RtpHeaderExtensionMap recv_rtp_extension_map_
|
||||
RTC_GUARDED_BY(thread_checker_);
|
||||
@ -750,7 +891,6 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
VideoSendParameters send_params_ RTC_GUARDED_BY(thread_checker_);
|
||||
VideoOptions default_send_options_ RTC_GUARDED_BY(thread_checker_);
|
||||
VideoRecvParameters recv_params_ RTC_GUARDED_BY(thread_checker_);
|
||||
int64_t last_send_stats_log_ms_ RTC_GUARDED_BY(thread_checker_);
|
||||
int64_t last_receive_stats_log_ms_ RTC_GUARDED_BY(thread_checker_);
|
||||
const bool discard_unknown_ssrc_packets_ RTC_GUARDED_BY(thread_checker_);
|
||||
// This is a stream param that comes from the remote description, but wasn't
|
||||
@ -766,11 +906,6 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
rtc::scoped_refptr<webrtc::FrameTransformerInterface>
|
||||
unsignaled_frame_transformer_ RTC_GUARDED_BY(thread_checker_);
|
||||
|
||||
// TODO(bugs.webrtc.org/11341): Remove this and relevant PC API. Presence
|
||||
// of multiple negotiated codecs allows generic encoder fallback on failures.
|
||||
// Presence of EncoderSelector allows switching to specific encoders.
|
||||
bool allow_codec_switching_ = false;
|
||||
|
||||
// RTP parameters that need to be set when creating a video receive stream.
|
||||
// Only used in Receiver mode - in Both mode, it reads those things from the
|
||||
// codec.
|
||||
@ -784,6 +919,17 @@ class WebRtcVideoChannel : public VideoMediaChannel,
|
||||
ssrc_list_changed_callback_;
|
||||
};
|
||||
|
||||
// Keeping the old name "WebRtcVideoChannel" around because some external
|
||||
// customers are using cricket::WebRtcVideoChannel::AdaptReason
|
||||
// TODO(bugs.webrtc.org/15216): Move this enum to an interface class and
|
||||
// delete this workaround.
|
||||
class WebRtcVideoChannel : public WebRtcVideoSendChannel {
|
||||
public:
|
||||
// Make all the values of AdaptReason available as
|
||||
// WebRtcVideoChannel::ADAPT_xxx.
|
||||
using WebRtcVideoSendChannel::AdaptReason;
|
||||
};
|
||||
|
||||
} // namespace cricket
|
||||
|
||||
#endif // MEDIA_ENGINE_WEBRTC_VIDEO_ENGINE_H_
|
||||
|
||||
@ -1820,10 +1820,10 @@ class WebRtcVideoChannelBaseTest : public ::testing::Test {
|
||||
}
|
||||
|
||||
// Returns pointer to implementation of the send channel.
|
||||
WebRtcVideoChannel* SendImpl() {
|
||||
WebRtcVideoSendChannel* SendImpl() {
|
||||
// Note that this function requires intimate knowledge of how the channel
|
||||
// was created.
|
||||
return static_cast<cricket::WebRtcVideoChannel*>(
|
||||
return static_cast<cricket::WebRtcVideoSendChannel*>(
|
||||
static_cast<VideoMediaShimChannel*>(channel_.get())
|
||||
->SendImplForTesting());
|
||||
}
|
||||
@ -2685,10 +2685,10 @@ class WebRtcVideoChannelTest : public WebRtcVideoEngineTest {
|
||||
}
|
||||
|
||||
// Returns pointer to implementation of the send channel.
|
||||
WebRtcVideoChannel* SendImpl() {
|
||||
WebRtcVideoSendChannel* SendImpl() {
|
||||
// Note that this function requires intimate knowledge of how the channel
|
||||
// was created.
|
||||
return static_cast<cricket::WebRtcVideoChannel*>(
|
||||
return static_cast<cricket::WebRtcVideoSendChannel*>(
|
||||
static_cast<VideoMediaShimChannel*>(channel_.get())
|
||||
->SendImplForTesting());
|
||||
}
|
||||
@ -2696,7 +2696,7 @@ class WebRtcVideoChannelTest : public WebRtcVideoEngineTest {
|
||||
// Casts a shim channel to a webrtc::Transport. Used once.
|
||||
webrtc::Transport* ChannelImplAsTransport(VideoMediaChannel* channel) {
|
||||
return static_cast<webrtc::Transport*>(
|
||||
static_cast<cricket::WebRtcVideoChannel*>(
|
||||
static_cast<cricket::WebRtcVideoSendChannel*>(
|
||||
static_cast<VideoMediaShimChannel*>(channel)
|
||||
->SendImplForTesting()));
|
||||
}
|
||||
|
||||
@ -40,7 +40,7 @@ enum : int { // The first valid value is 1.
|
||||
kVideoRotationRtpExtensionId,
|
||||
};
|
||||
|
||||
constexpr int kDefaultMaxQp = cricket::WebRtcVideoChannel::kDefaultQpMax;
|
||||
constexpr int kDefaultMaxQp = cricket::WebRtcVideoSendChannel::kDefaultQpMax;
|
||||
uint8_t CodecTypeToPayloadType(VideoCodecType codec_type) {
|
||||
switch (codec_type) {
|
||||
case VideoCodecType::kVideoCodecGeneric:
|
||||
|
||||
@ -76,7 +76,7 @@ constexpr int kFramesSentInQuickTest = 1;
|
||||
constexpr uint32_t kThumbnailSendSsrcStart = 0xE0000;
|
||||
constexpr uint32_t kThumbnailRtxSsrcStart = 0xF0000;
|
||||
|
||||
constexpr int kDefaultMaxQp = cricket::WebRtcVideoChannel::kDefaultQpMax;
|
||||
constexpr int kDefaultMaxQp = cricket::WebRtcVideoSendChannel::kDefaultQpMax;
|
||||
|
||||
const VideoEncoder::Capabilities kCapabilities(false);
|
||||
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user