Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(172)

Unified Diff: webrtc/pc/mediasession.cc

Issue 1956343002: Initial asymmetric codec support in MediaSessionDescription (Closed) Base URL: https://chromium.googlesource.com/external/webrtc.git@master
Patch Set: Added TODO. Created 4 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « webrtc/pc/mediasession.h ('k') | webrtc/pc/mediasession_unittest.cc » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: webrtc/pc/mediasession.cc
diff --git a/webrtc/pc/mediasession.cc b/webrtc/pc/mediasession.cc
index 52abfe855f6aef47fa83d7e1ce51321f090a2417..5702666b61016ffd3b48afdabf9129921e0a1a62 100644
--- a/webrtc/pc/mediasession.cc
+++ b/webrtc/pc/mediasession.cc
@@ -50,7 +50,6 @@ void GetSupportedCryptoSuiteNames(void (*func)(std::vector<int>*),
namespace cricket {
-
// RTP Profile names
// http://www.iana.org/assignments/rtp-parameters/rtp-parameters.xml
// RFC4585
@@ -69,6 +68,32 @@ const char kMediaProtocolDtlsSctp[] = "DTLS/SCTP";
const char kMediaProtocolUdpDtlsSctp[] = "UDP/DTLS/SCTP";
const char kMediaProtocolTcpDtlsSctp[] = "TCP/DTLS/SCTP";
+RtpTransceiverDirection RtpTransceiverDirection::FromMediaContentDirection(
+ MediaContentDirection md) {
+ const bool send = (md == MD_SENDRECV || md == MD_SENDONLY);
+ const bool recv = (md == MD_SENDRECV || md == MD_RECVONLY);
+ return RtpTransceiverDirection(send, recv);
+}
+
+MediaContentDirection RtpTransceiverDirection::ToMediaContentDirection() const {
+ if (send && recv) {
+ return MD_SENDRECV;
+ } else if (send) {
+ return MD_SENDONLY;
+ } else if (recv) {
+ return MD_RECVONLY;
+ }
+
+ return MD_INACTIVE;
+}
+
+RtpTransceiverDirection
+NegotiateRtpTransceiverDirection(RtpTransceiverDirection offer,
+ RtpTransceiverDirection wants) {
+ return RtpTransceiverDirection(offer.recv && wants.send,
+ offer.send && wants.recv);
+}
+
static bool IsMediaContentOfType(const ContentInfo* content,
MediaType media_type) {
if (!IsMediaContent(content)) {
@@ -767,6 +792,7 @@ static void NegotiateCodecs(const std::vector<C>& local_codecs,
offered_apt_value);
}
negotiated.id = theirs.id;
+ negotiated.name = theirs.name;
negotiated_codecs->push_back(negotiated);
}
}
@@ -1030,30 +1056,22 @@ static bool CreateMediaContentAnswer(
// Make sure the answer media content direction is per default set as
// described in RFC3264 section 6.1.
- switch (offer->direction()) {
- case MD_INACTIVE:
- answer->set_direction(MD_INACTIVE);
- break;
- case MD_SENDONLY:
- answer->set_direction(MD_RECVONLY);
- break;
- case MD_RECVONLY:
- answer->set_direction(IsRtpProtocol(answer->protocol()) &&
- answer->streams().empty()
- ? MD_INACTIVE
- : MD_SENDONLY);
- break;
- case MD_SENDRECV:
- answer->set_direction(IsRtpProtocol(answer->protocol()) &&
- answer->streams().empty()
- ? MD_RECVONLY
- : MD_SENDRECV);
- break;
- default:
- RTC_DCHECK(false && "MediaContentDescription has unexpected direction.");
- break;
- }
-
+ const bool is_data = !IsRtpProtocol(answer->protocol());
+ const bool has_send_streams = !answer->streams().empty();
+ const bool wants_send = has_send_streams || is_data;
+ const bool recv_audio =
+ answer->type() == cricket::MEDIA_TYPE_AUDIO && options.recv_audio;
+ const bool recv_video =
+ answer->type() == cricket::MEDIA_TYPE_VIDEO && options.recv_video;
+ const bool recv_data =
+ answer->type() == cricket::MEDIA_TYPE_DATA;
+ const bool wants_receive = recv_audio || recv_video || recv_data;
+
+ auto offer_rtd =
+ RtpTransceiverDirection::FromMediaContentDirection(offer->direction());
+ auto wants_rtd = RtpTransceiverDirection(wants_send, wants_receive);
+ answer->set_direction(NegotiateRtpTransceiverDirection(offer_rtd, wants_rtd)
+ .ToMediaContentDirection());
return true;
}
@@ -1173,6 +1191,29 @@ std::string MediaTypeToString(MediaType type) {
return type_str;
}
+std::string MediaContentDirectionToString(MediaContentDirection direction) {
+ std::string dir_str;
+ switch (direction) {
+ case MD_INACTIVE:
+ dir_str = "inactive";
+ break;
+ case MD_SENDONLY:
+ dir_str = "sendonly";
+ break;
+ case MD_RECVONLY:
+ dir_str = "recvonly";
+ break;
+ case MD_SENDRECV:
+ dir_str = "sendrecv";
+ break;
+ default:
+ ASSERT(false);
+ break;
+ }
+
+ return dir_str;
+}
+
void MediaSessionOptions::AddSendStream(MediaType type,
const std::string& id,
const std::string& sync_label) {
@@ -1234,11 +1275,38 @@ MediaSessionDescriptionFactory::MediaSessionDescriptionFactory(
: secure_(SEC_DISABLED),
add_legacy_(true),
transport_desc_factory_(transport_desc_factory) {
- channel_manager->GetSupportedAudioCodecs(&audio_codecs_);
+ channel_manager->GetSupportedAudioCodecs(&audio_sendrecv_codecs_);
channel_manager->GetSupportedAudioRtpHeaderExtensions(&audio_rtp_extensions_);
channel_manager->GetSupportedVideoCodecs(&video_codecs_);
channel_manager->GetSupportedVideoRtpHeaderExtensions(&video_rtp_extensions_);
channel_manager->GetSupportedDataCodecs(&data_codecs_);
+ audio_send_codecs_ = audio_sendrecv_codecs_;
+ audio_recv_codecs_ = audio_sendrecv_codecs_;
+}
+
+const AudioCodecs& MediaSessionDescriptionFactory::audio_codecs() const {
+ return audio_sendrecv_codecs_;
+}
+
+const AudioCodecs& MediaSessionDescriptionFactory::audio_send_codecs() const {
+ return audio_send_codecs_;
+}
+
+const AudioCodecs& MediaSessionDescriptionFactory::audio_recv_codecs() const {
+ return audio_recv_codecs_;
+}
+
+void MediaSessionDescriptionFactory::set_audio_codecs(
+ const AudioCodecs& send_codecs, const AudioCodecs& recv_codecs) {
+ audio_send_codecs_ = send_codecs;
+ audio_recv_codecs_ = recv_codecs;
+ audio_sendrecv_codecs_.clear();
+ // Use NegotiateCodecs to merge our codec lists, since the operation is
+ // essentially the same. Put send_codecs as the offered_codecs, which is the
+ // order we'd like to follow. The reasoning is that encoding is usually more
+ // expensive than decoding, and prioritizing a codec in the send list probably
+ // means it's a codec we can handle efficiently.
+ NegotiateCodecs(recv_codecs, send_codecs, &audio_sendrecv_codecs_);
}
SessionDescription* MediaSessionDescriptionFactory::CreateOffer(
@@ -1249,11 +1317,17 @@ SessionDescription* MediaSessionDescriptionFactory::CreateOffer(
StreamParamsVec current_streams;
GetCurrentStreamParams(current_description, &current_streams);
+ const bool wants_send =
+ options.HasSendMediaStream(MEDIA_TYPE_AUDIO) || add_legacy_;
+ const AudioCodecs& supported_audio_codecs =
+ GetAudioCodecsForOffer({wants_send, options.recv_audio});
+
AudioCodecs audio_codecs;
VideoCodecs video_codecs;
DataCodecs data_codecs;
- GetCodecsToOffer(current_description, &audio_codecs, &video_codecs,
- &data_codecs);
+ GetCodecsToOffer(current_description, supported_audio_codecs,
+ video_codecs_, data_codecs_,
+ &audio_codecs, &video_codecs, &data_codecs);
if (!options.vad_enabled) {
// If application doesn't want CN codecs in offer.
@@ -1412,8 +1486,43 @@ SessionDescription* MediaSessionDescriptionFactory::CreateAnswer(
return answer.release();
}
+const AudioCodecs& MediaSessionDescriptionFactory::GetAudioCodecsForOffer(
+ const RtpTransceiverDirection& direction) const {
+ // If stream is inactive - generate list as if sendrecv.
+ if (direction.send == direction.recv) {
+ return audio_sendrecv_codecs_;
+ } else if (direction.send) {
+ return audio_send_codecs_;
+ } else {
+ return audio_recv_codecs_;
+ }
+}
+
+const AudioCodecs& MediaSessionDescriptionFactory::GetAudioCodecsForAnswer(
+ const RtpTransceiverDirection& offer,
+ const RtpTransceiverDirection& answer) const {
+ // For inactive and sendrecv answers, generate lists as if we were to accept
+ // the offer's direction. See RFC 3264 Section 6.1.
+ if (answer.send == answer.recv) {
+ if (offer.send == offer.recv) {
+ return audio_sendrecv_codecs_;
+ } else if (offer.send) {
+ return audio_recv_codecs_;
+ } else {
+ return audio_send_codecs_;
+ }
+ } else if (answer.send) {
+ return audio_send_codecs_;
+ } else {
+ return audio_recv_codecs_;
+ }
+}
+
void MediaSessionDescriptionFactory::GetCodecsToOffer(
const SessionDescription* current_description,
+ const AudioCodecs& supported_audio_codecs,
+ const VideoCodecs& supported_video_codecs,
+ const DataCodecs& supported_data_codecs,
AudioCodecs* audio_codecs,
VideoCodecs* video_codecs,
DataCodecs* data_codecs) const {
@@ -1449,9 +1558,12 @@ void MediaSessionDescriptionFactory::GetCodecsToOffer(
}
// Add our codecs that are not in |current_description|.
- FindCodecsToOffer<AudioCodec>(audio_codecs_, audio_codecs, &used_pltypes);
- FindCodecsToOffer<VideoCodec>(video_codecs_, video_codecs, &used_pltypes);
- FindCodecsToOffer<DataCodec>(data_codecs_, data_codecs, &used_pltypes);
+ FindCodecsToOffer<AudioCodec>(supported_audio_codecs, audio_codecs,
+ &used_pltypes);
+ FindCodecsToOffer<VideoCodec>(supported_video_codecs, video_codecs,
+ &used_pltypes);
+ FindCodecsToOffer<DataCodec>(supported_data_codecs, data_codecs,
+ &used_pltypes);
}
void MediaSessionDescriptionFactory::GetRtpHdrExtsToOffer(
@@ -1577,19 +1689,9 @@ bool MediaSessionDescriptionFactory::AddAudioContentForOffer(
bool secure_transport = (transport_desc_factory_->secure() != SEC_DISABLED);
SetMediaProtocol(secure_transport, audio.get());
- if (!audio->streams().empty()) {
- if (options.recv_audio) {
- audio->set_direction(MD_SENDRECV);
- } else {
- audio->set_direction(MD_SENDONLY);
- }
- } else {
- if (options.recv_audio) {
- audio->set_direction(MD_RECVONLY);
- } else {
- audio->set_direction(MD_INACTIVE);
- }
- }
+ auto offer_rtd =
+ RtpTransceiverDirection(!audio->streams().empty(), options.recv_audio);
+ audio->set_direction(offer_rtd.ToMediaContentDirection());
desc->AddContent(content_name, NS_JINGLE_RTP, audio.release());
if (!AddTransportOffer(content_name,
@@ -1733,6 +1835,8 @@ bool MediaSessionDescriptionFactory::AddAudioContentForAnswer(
StreamParamsVec* current_streams,
SessionDescription* answer) const {
const ContentInfo* audio_content = GetFirstAudioContent(offer);
+ const AudioContentDescription* offer_audio =
+ static_cast<const AudioContentDescription*>(audio_content->description);
std::unique_ptr<TransportDescription> audio_transport(CreateTransportAnswer(
audio_content->name, offer,
@@ -1741,7 +1845,15 @@ bool MediaSessionDescriptionFactory::AddAudioContentForAnswer(
return false;
}
- AudioCodecs audio_codecs = audio_codecs_;
+ // Pick codecs based on the requested communications direction in the offer.
+ const bool wants_send =
+ options.HasSendMediaStream(MEDIA_TYPE_AUDIO) || add_legacy_;
+ auto wants_rtd = RtpTransceiverDirection(wants_send, options.recv_audio);
+ auto offer_rtd =
+ RtpTransceiverDirection::FromMediaContentDirection(
+ offer_audio->direction());
+ auto answer_rtd = NegotiateRtpTransceiverDirection(offer_rtd, wants_rtd);
+ AudioCodecs audio_codecs = GetAudioCodecsForAnswer(offer_rtd, answer_rtd);
if (!options.vad_enabled) {
StripCNCodecs(&audio_codecs);
}
@@ -1754,8 +1866,7 @@ bool MediaSessionDescriptionFactory::AddAudioContentForAnswer(
cricket::SecurePolicy sdes_policy =
audio_transport->secure() ? cricket::SEC_DISABLED : secure();
if (!CreateMediaContentAnswer(
- static_cast<const AudioContentDescription*>(
- audio_content->description),
+ offer_audio,
options,
audio_codecs,
sdes_policy,
« no previous file with comments | « webrtc/pc/mediasession.h ('k') | webrtc/pc/mediasession_unittest.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698