/* * 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. */ #include "media/base/rtp_data_engine.h" #include #include "absl/strings/match.h" #include "media/base/codec.h" #include "media/base/media_constants.h" #include "media/base/rtp_utils.h" #include "media/base/stream_params.h" #include "rtc_base/copy_on_write_buffer.h" #include "rtc_base/data_rate_limiter.h" #include "rtc_base/helpers.h" #include "rtc_base/logging.h" #include "rtc_base/sanitizer.h" namespace cricket { // We want to avoid IP fragmentation. static const size_t kDataMaxRtpPacketLen = 1200U; // We reserve space after the RTP header for future wiggle room. static const unsigned char kReservedSpace[] = {0x00, 0x00, 0x00, 0x00}; // Amount of overhead SRTP may take. We need to leave room in the // buffer for it, otherwise SRTP will fail later. If SRTP ever uses // more than this, we need to increase this number. static const size_t kMaxSrtpHmacOverhead = 16; RtpDataEngine::RtpDataEngine() { data_codecs_.push_back( DataCodec(kGoogleRtpDataCodecPlType, kGoogleRtpDataCodecName)); } DataMediaChannel* RtpDataEngine::CreateChannel(const MediaConfig& config) { return new RtpDataMediaChannel(config); } static const DataCodec* FindCodecByName(const std::vector& codecs, const std::string& name) { for (const DataCodec& codec : codecs) { if (absl::EqualsIgnoreCase(name, codec.name)) return &codec; } return nullptr; } RtpDataMediaChannel::RtpDataMediaChannel(const MediaConfig& config) : DataMediaChannel(config) { Construct(); SetPreferredDscp(rtc::DSCP_AF41); } void RtpDataMediaChannel::Construct() { sending_ = false; receiving_ = false; send_limiter_.reset(new rtc::DataRateLimiter(kDataMaxBandwidth / 8, 1.0)); } RtpDataMediaChannel::~RtpDataMediaChannel() { std::map::const_iterator iter; for (iter = rtp_clock_by_send_ssrc_.begin(); iter != rtp_clock_by_send_ssrc_.end(); ++iter) { delete iter->second; } } void RTC_NO_SANITIZE("float-cast-overflow") // bugs.webrtc.org/8204 RtpClock::Tick(double now, int* seq_num, uint32_t* timestamp) { *seq_num = ++last_seq_num_; *timestamp = timestamp_offset_ + static_cast(now * clockrate_); // UBSan: 5.92374e+10 is outside the range of representable values of type // 'unsigned int' } const DataCodec* FindUnknownCodec(const std::vector& codecs) { DataCodec data_codec(kGoogleRtpDataCodecPlType, kGoogleRtpDataCodecName); std::vector::const_iterator iter; for (iter = codecs.begin(); iter != codecs.end(); ++iter) { if (!iter->Matches(data_codec)) { return &(*iter); } } return NULL; } const DataCodec* FindKnownCodec(const std::vector& codecs) { DataCodec data_codec(kGoogleRtpDataCodecPlType, kGoogleRtpDataCodecName); std::vector::const_iterator iter; for (iter = codecs.begin(); iter != codecs.end(); ++iter) { if (iter->Matches(data_codec)) { return &(*iter); } } return NULL; } bool RtpDataMediaChannel::SetRecvCodecs(const std::vector& codecs) { const DataCodec* unknown_codec = FindUnknownCodec(codecs); if (unknown_codec) { RTC_LOG(LS_WARNING) << "Failed to SetRecvCodecs because of unknown codec: " << unknown_codec->ToString(); return false; } recv_codecs_ = codecs; return true; } bool RtpDataMediaChannel::SetSendCodecs(const std::vector& codecs) { const DataCodec* known_codec = FindKnownCodec(codecs); if (!known_codec) { RTC_LOG(LS_WARNING) << "Failed to SetSendCodecs because there is no known codec."; return false; } send_codecs_ = codecs; return true; } bool RtpDataMediaChannel::SetSendParameters(const DataSendParameters& params) { return (SetSendCodecs(params.codecs) && SetMaxSendBandwidth(params.max_bandwidth_bps)); } bool RtpDataMediaChannel::SetRecvParameters(const DataRecvParameters& params) { return SetRecvCodecs(params.codecs); } bool RtpDataMediaChannel::AddSendStream(const StreamParams& stream) { if (!stream.has_ssrcs()) { return false; } if (GetStreamBySsrc(send_streams_, stream.first_ssrc())) { RTC_LOG(LS_WARNING) << "Not adding data send stream '" << stream.id << "' with ssrc=" << stream.first_ssrc() << " because stream already exists."; return false; } send_streams_.push_back(stream); // TODO(pthatcher): This should be per-stream, not per-ssrc. // And we should probably allow more than one per stream. rtp_clock_by_send_ssrc_[stream.first_ssrc()] = new RtpClock(kDataCodecClockrate, rtc::CreateRandomNonZeroId(), rtc::CreateRandomNonZeroId()); RTC_LOG(LS_INFO) << "Added data send stream '" << stream.id << "' with ssrc=" << stream.first_ssrc(); return true; } bool RtpDataMediaChannel::RemoveSendStream(uint32_t ssrc) { if (!GetStreamBySsrc(send_streams_, ssrc)) { return false; } RemoveStreamBySsrc(&send_streams_, ssrc); delete rtp_clock_by_send_ssrc_[ssrc]; rtp_clock_by_send_ssrc_.erase(ssrc); return true; } bool RtpDataMediaChannel::AddRecvStream(const StreamParams& stream) { if (!stream.has_ssrcs()) { return false; } if (GetStreamBySsrc(recv_streams_, stream.first_ssrc())) { RTC_LOG(LS_WARNING) << "Not adding data recv stream '" << stream.id << "' with ssrc=" << stream.first_ssrc() << " because stream already exists."; return false; } recv_streams_.push_back(stream); RTC_LOG(LS_INFO) << "Added data recv stream '" << stream.id << "' with ssrc=" << stream.first_ssrc(); return true; } bool RtpDataMediaChannel::RemoveRecvStream(uint32_t ssrc) { RemoveStreamBySsrc(&recv_streams_, ssrc); return true; } // Not implemented. void RtpDataMediaChannel::ResetUnsignaledRecvStream() {} void RtpDataMediaChannel::OnPacketReceived(rtc::CopyOnWriteBuffer packet, int64_t /* packet_time_us */) { RtpHeader header; if (!GetRtpHeader(packet.cdata(), packet.size(), &header)) { return; } size_t header_length; if (!GetRtpHeaderLen(packet.cdata(), packet.size(), &header_length)) { return; } const char* data = packet.cdata() + header_length + sizeof(kReservedSpace); size_t data_len = packet.size() - header_length - sizeof(kReservedSpace); if (!receiving_) { RTC_LOG(LS_WARNING) << "Not receiving packet " << header.ssrc << ":" << header.seq_num << " before SetReceive(true) called."; return; } if (!FindCodecById(recv_codecs_, header.payload_type)) { return; } if (!GetStreamBySsrc(recv_streams_, header.ssrc)) { RTC_LOG(LS_WARNING) << "Received packet for unknown ssrc: " << header.ssrc; return; } // Uncomment this for easy debugging. // const auto* found_stream = GetStreamBySsrc(recv_streams_, header.ssrc); // RTC_LOG(LS_INFO) << "Received packet" // << " groupid=" << found_stream.groupid // << ", ssrc=" << header.ssrc // << ", seqnum=" << header.seq_num // << ", timestamp=" << header.timestamp // << ", len=" << data_len; ReceiveDataParams params; params.ssrc = header.ssrc; params.seq_num = header.seq_num; params.timestamp = header.timestamp; SignalDataReceived(params, data, data_len); } bool RtpDataMediaChannel::SetMaxSendBandwidth(int bps) { if (bps <= 0) { bps = kDataMaxBandwidth; } send_limiter_.reset(new rtc::DataRateLimiter(bps / 8, 1.0)); RTC_LOG(LS_INFO) << "RtpDataMediaChannel::SetSendBandwidth to " << bps << "bps."; return true; } bool RtpDataMediaChannel::SendData(const SendDataParams& params, const rtc::CopyOnWriteBuffer& payload, SendDataResult* result) { if (result) { // If we return true, we'll set this to SDR_SUCCESS. *result = SDR_ERROR; } if (!sending_) { RTC_LOG(LS_WARNING) << "Not sending packet with ssrc=" << params.ssrc << " len=" << payload.size() << " before SetSend(true)."; return false; } if (params.type != cricket::DMT_TEXT) { RTC_LOG(LS_WARNING) << "Not sending data because binary type is unsupported."; return false; } const StreamParams* found_stream = GetStreamBySsrc(send_streams_, params.ssrc); if (!found_stream) { RTC_LOG(LS_WARNING) << "Not sending data because ssrc is unknown: " << params.ssrc; return false; } const DataCodec* found_codec = FindCodecByName(send_codecs_, kGoogleRtpDataCodecName); if (!found_codec) { RTC_LOG(LS_WARNING) << "Not sending data because codec is unknown: " << kGoogleRtpDataCodecName; return false; } size_t packet_len = (kMinRtpPacketLen + sizeof(kReservedSpace) + payload.size() + kMaxSrtpHmacOverhead); if (packet_len > kDataMaxRtpPacketLen) { return false; } double now = rtc::TimeMicros() / static_cast(rtc::kNumMicrosecsPerSec); if (!send_limiter_->CanUse(packet_len, now)) { RTC_LOG(LS_VERBOSE) << "Dropped data packet of len=" << packet_len << "; already sent " << send_limiter_->used_in_period() << "/" << send_limiter_->max_per_period(); return false; } RtpHeader header; header.payload_type = found_codec->id; header.ssrc = params.ssrc; rtp_clock_by_send_ssrc_[header.ssrc]->Tick(now, &header.seq_num, &header.timestamp); rtc::CopyOnWriteBuffer packet(kMinRtpPacketLen, packet_len); if (!SetRtpHeader(packet.data(), packet.size(), header)) { return false; } packet.AppendData(kReservedSpace); packet.AppendData(payload); RTC_LOG(LS_VERBOSE) << "Sent RTP data packet: " " stream=" << found_stream->id << " ssrc=" << header.ssrc << ", seqnum=" << header.seq_num << ", timestamp=" << header.timestamp << ", len=" << payload.size(); rtc::PacketOptions options; options.info_signaled_after_sent.packet_type = rtc::PacketType::kData; MediaChannel::SendPacket(&packet, options); send_limiter_->Use(packet_len, now); if (result) { *result = SDR_SUCCESS; } return true; } } // namespace cricket