| OLD | NEW |
| 1 // Copyright 2016 The Chromium Authors. All rights reserved. | 1 // Copyright 2016 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "extensions/renderer/api/display_source/wifi_display/wifi_display_media
_pipeline.h" | 5 #include "extensions/renderer/api/display_source/wifi_display/wifi_display_media
_pipeline.h" |
| 6 | 6 |
| 7 #include "base/logging.h" | 7 #include "base/logging.h" |
| 8 #include "content/public/renderer/render_thread.h" | 8 #include "content/public/renderer/render_thread.h" |
| 9 #include "content/public/renderer/video_encode_accelerator.h" | 9 #include "content/public/renderer/video_encode_accelerator.h" |
| 10 #include "extensions/renderer/api/display_source/wifi_display/wifi_display_eleme
ntary_stream_descriptor.h" | |
| 11 #include "extensions/renderer/api/display_source/wifi_display/wifi_display_eleme
ntary_stream_info.h" | |
| 12 | 10 |
| 13 namespace extensions { | 11 namespace extensions { |
| 14 | 12 |
| 15 namespace { | 13 namespace { |
| 16 | 14 |
| 15 const char kErrorAudioEncoderError[] = "Unrepairable audio encoder error"; |
| 17 const char kErrorVideoEncoderError[] = "Unrepairable video encoder error"; | 16 const char kErrorVideoEncoderError[] = "Unrepairable video encoder error"; |
| 18 const char kErrorUnableSendMedia[] = "Unable to send media"; | 17 const char kErrorUnableSendMedia[] = "Unable to send media"; |
| 19 | 18 |
| 20 } // namespace | 19 } // namespace |
| 21 | 20 |
| 22 WiFiDisplayMediaPipeline::WiFiDisplayMediaPipeline( | 21 WiFiDisplayMediaPipeline::WiFiDisplayMediaPipeline( |
| 23 wds::SessionType type, | 22 wds::SessionType type, |
| 24 const WiFiDisplayVideoEncoder::InitParameters& video_parameters, | 23 const WiFiDisplayVideoEncoder::InitParameters& video_parameters, |
| 25 const wds::AudioCodec& audio_codec, | 24 const wds::AudioCodec& audio_codec, |
| 26 const std::string& sink_ip_address, | 25 const std::string& sink_ip_address, |
| (...skipping 39 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 66 video_encoder_->InsertRawVideoFrame(video_frame, reference_time); | 65 video_encoder_->InsertRawVideoFrame(video_frame, reference_time); |
| 67 } | 66 } |
| 68 | 67 |
| 69 void WiFiDisplayMediaPipeline::RequestIDRPicture() { | 68 void WiFiDisplayMediaPipeline::RequestIDRPicture() { |
| 70 DCHECK(video_encoder_); | 69 DCHECK(video_encoder_); |
| 71 video_encoder_->RequestIDRPicture(); | 70 video_encoder_->RequestIDRPicture(); |
| 72 } | 71 } |
| 73 | 72 |
| 74 enum WiFiDisplayMediaPipeline::InitializationStep : unsigned { | 73 enum WiFiDisplayMediaPipeline::InitializationStep : unsigned { |
| 75 INITIALIZE_FIRST, | 74 INITIALIZE_FIRST, |
| 76 INITIALIZE_MEDIA_PACKETIZER = INITIALIZE_FIRST, | 75 INITIALIZE_AUDIO_ENCODER = INITIALIZE_FIRST, |
| 77 INITIALIZE_VIDEO_ENCODER, | 76 INITIALIZE_VIDEO_ENCODER, |
| 77 INITIALIZE_MEDIA_PACKETIZER, |
| 78 INITIALIZE_MEDIA_SERVICE, | 78 INITIALIZE_MEDIA_SERVICE, |
| 79 INITIALIZE_LAST = INITIALIZE_MEDIA_SERVICE | 79 INITIALIZE_LAST = INITIALIZE_MEDIA_SERVICE |
| 80 }; | 80 }; |
| 81 | 81 |
| 82 void WiFiDisplayMediaPipeline::Initialize( | 82 void WiFiDisplayMediaPipeline::Initialize( |
| 83 const InitCompletionCallback& callback) { | 83 const InitCompletionCallback& callback) { |
| 84 DCHECK(!video_encoder_ && !packetizer_); | 84 DCHECK(!audio_encoder_ && !video_encoder_ && !packetizer_); |
| 85 OnInitialize(callback, INITIALIZE_FIRST, true); | 85 OnInitialize(callback, INITIALIZE_FIRST, true); |
| 86 } | 86 } |
| 87 | 87 |
| 88 void WiFiDisplayMediaPipeline::OnInitialize( | 88 void WiFiDisplayMediaPipeline::OnInitialize( |
| 89 const InitCompletionCallback& callback, | 89 const InitCompletionCallback& callback, |
| 90 InitializationStep current_step, | 90 InitializationStep current_step, |
| 91 bool success) { | 91 bool success) { |
| 92 if (!success) { | 92 if (!success) { |
| 93 callback.Run(false); | 93 callback.Run(false); |
| 94 return; | 94 return; |
| 95 } | 95 } |
| 96 | 96 |
| 97 InitStepCompletionCallback init_step_callback; | 97 InitStepCompletionCallback init_step_callback; |
| 98 if (current_step < INITIALIZE_LAST) { | 98 if (current_step < INITIALIZE_LAST) { |
| 99 InitializationStep next_step = | 99 InitializationStep next_step = |
| 100 static_cast<InitializationStep>(current_step + 1u); | 100 static_cast<InitializationStep>(current_step + 1u); |
| 101 init_step_callback = base::Bind(&WiFiDisplayMediaPipeline::OnInitialize, | 101 init_step_callback = base::Bind(&WiFiDisplayMediaPipeline::OnInitialize, |
| 102 weak_factory_.GetWeakPtr(), callback, next_step); | 102 weak_factory_.GetWeakPtr(), callback, next_step); |
| 103 } | 103 } |
| 104 | 104 |
| 105 switch (current_step) { | 105 switch (current_step) { |
| 106 case INITIALIZE_MEDIA_PACKETIZER: | 106 case INITIALIZE_AUDIO_ENCODER: |
| 107 DCHECK(!packetizer_); | 107 DCHECK(!audio_encoder_); |
| 108 CreateMediaPacketizer(); | 108 if (type_ & wds::AudioSession) { |
| 109 init_step_callback.Run(true); | 109 auto result_callback = |
| 110 base::Bind(&WiFiDisplayMediaPipeline::OnAudioEncoderCreated, |
| 111 weak_factory_.GetWeakPtr(), init_step_callback); |
| 112 WiFiDisplayAudioEncoder::Create(audio_codec_, result_callback); |
| 113 } else { |
| 114 init_step_callback.Run(true); |
| 115 } |
| 110 break; | 116 break; |
| 111 case INITIALIZE_VIDEO_ENCODER: | 117 case INITIALIZE_VIDEO_ENCODER: |
| 112 DCHECK(!video_encoder_); | 118 DCHECK(!video_encoder_); |
| 113 if (type_ & wds::VideoSession) { | 119 if (type_ & wds::VideoSession) { |
| 114 auto result_callback = | 120 auto result_callback = |
| 115 base::Bind(&WiFiDisplayMediaPipeline::OnVideoEncoderCreated, | 121 base::Bind(&WiFiDisplayMediaPipeline::OnVideoEncoderCreated, |
| 116 weak_factory_.GetWeakPtr(), init_step_callback); | 122 weak_factory_.GetWeakPtr(), init_step_callback); |
| 117 WiFiDisplayVideoEncoder::Create(video_parameters_, result_callback); | 123 WiFiDisplayVideoEncoder::Create(video_parameters_, result_callback); |
| 118 } else { | 124 } else { |
| 119 init_step_callback.Run(true); | 125 init_step_callback.Run(true); |
| 120 } | 126 } |
| 121 break; | 127 break; |
| 128 case INITIALIZE_MEDIA_PACKETIZER: |
| 129 DCHECK(!packetizer_); |
| 130 CreateMediaPacketizer(); |
| 131 init_step_callback.Run(true); |
| 132 break; |
| 122 case INITIALIZE_MEDIA_SERVICE: | 133 case INITIALIZE_MEDIA_SERVICE: |
| 123 service_callback_.Run( | 134 service_callback_.Run( |
| 124 mojo::GetProxy(&media_service_), | 135 mojo::GetProxy(&media_service_), |
| 125 base::Bind(&WiFiDisplayMediaPipeline::OnMediaServiceRegistered, | 136 base::Bind(&WiFiDisplayMediaPipeline::OnMediaServiceRegistered, |
| 126 weak_factory_.GetWeakPtr(), callback)); | 137 weak_factory_.GetWeakPtr(), callback)); |
| 127 break; | 138 break; |
| 128 } | 139 } |
| 129 } | 140 } |
| 130 | 141 |
| 131 void WiFiDisplayMediaPipeline::CreateMediaPacketizer() { | 142 void WiFiDisplayMediaPipeline::CreateMediaPacketizer() { |
| 132 DCHECK(!packetizer_); | 143 DCHECK(!packetizer_); |
| 133 std::vector<WiFiDisplayElementaryStreamInfo> stream_infos; | 144 std::vector<WiFiDisplayElementaryStreamInfo> stream_infos; |
| 134 if (type_ & wds::VideoSession) { | 145 if (type_ & wds::VideoSession) { |
| 135 std::vector<WiFiDisplayElementaryStreamDescriptor> descriptors; | 146 std::vector<WiFiDisplayElementaryStreamDescriptor> descriptors; |
| 136 descriptors.emplace_back( | 147 descriptors.emplace_back( |
| 137 WiFiDisplayElementaryStreamDescriptor::AVCTimingAndHRD::Create()); | 148 WiFiDisplayElementaryStreamDescriptor::AVCTimingAndHRD::Create()); |
| 138 stream_infos.emplace_back(WiFiDisplayElementaryStreamInfo::VIDEO_H264, | 149 stream_infos.emplace_back(WiFiDisplayElementaryStreamInfo::VIDEO_H264, |
| 139 std::move(descriptors)); | 150 std::move(descriptors)); |
| 140 } | 151 } |
| 141 | 152 |
| 142 if (type_ & wds::AudioSession) { | 153 if (type_ & wds::AudioSession) { |
| 143 using LPCMAudioStreamDescriptor = | 154 DCHECK(audio_encoder_); |
| 144 WiFiDisplayElementaryStreamDescriptor::LPCMAudioStream; | 155 stream_infos.push_back(audio_encoder_->CreateElementaryStreamInfo()); |
| 145 std::vector<WiFiDisplayElementaryStreamDescriptor> descriptors; | |
| 146 descriptors.emplace_back( | |
| 147 LPCMAudioStreamDescriptor::Create( | |
| 148 audio_codec_.modes.test(wds::LPCM_48K_16B_2CH) | |
| 149 ? LPCMAudioStreamDescriptor::SAMPLING_FREQUENCY_48K | |
| 150 : LPCMAudioStreamDescriptor::SAMPLING_FREQUENCY_44_1K, | |
| 151 LPCMAudioStreamDescriptor::BITS_PER_SAMPLE_16, | |
| 152 false, // emphasis_flag | |
| 153 LPCMAudioStreamDescriptor::NUMBER_OF_CHANNELS_STEREO)); | |
| 154 stream_infos.emplace_back(WiFiDisplayElementaryStreamInfo::AUDIO_LPCM, | |
| 155 std::move(descriptors)); | |
| 156 } | 156 } |
| 157 | 157 |
| 158 packetizer_.reset(new WiFiDisplayMediaPacketizer( | 158 packetizer_.reset(new WiFiDisplayMediaPacketizer( |
| 159 base::TimeDelta::FromMilliseconds(200), | 159 base::TimeDelta::FromMilliseconds(200), |
| 160 std::move(stream_infos), | 160 std::move(stream_infos), |
| 161 base::Bind(&WiFiDisplayMediaPipeline::OnPacketizedMediaDatagramPacket, | 161 base::Bind(&WiFiDisplayMediaPipeline::OnPacketizedMediaDatagramPacket, |
| 162 base::Unretained(this)))); | 162 base::Unretained(this)))); |
| 163 } | 163 } |
| 164 | 164 |
| 165 void WiFiDisplayMediaPipeline::OnAudioEncoderCreated( |
| 166 const InitStepCompletionCallback& callback, |
| 167 scoped_refptr<WiFiDisplayAudioEncoder> audio_encoder) { |
| 168 DCHECK(!audio_encoder_); |
| 169 |
| 170 if (!audio_encoder) { |
| 171 callback.Run(false); |
| 172 return; |
| 173 } |
| 174 |
| 175 audio_encoder_ = std::move(audio_encoder); |
| 176 auto encoded_callback = |
| 177 base::Bind(&WiFiDisplayMediaPipeline::OnEncodedAudioUnit, |
| 178 weak_factory_.GetWeakPtr()); |
| 179 auto error_callback = base::Bind(error_callback_, kErrorAudioEncoderError); |
| 180 audio_encoder_->SetCallbacks(encoded_callback, error_callback); |
| 181 |
| 182 callback.Run(true); |
| 183 } |
| 184 |
| 165 void WiFiDisplayMediaPipeline::OnVideoEncoderCreated( | 185 void WiFiDisplayMediaPipeline::OnVideoEncoderCreated( |
| 166 const InitStepCompletionCallback& callback, | 186 const InitStepCompletionCallback& callback, |
| 167 scoped_refptr<WiFiDisplayVideoEncoder> video_encoder) { | 187 scoped_refptr<WiFiDisplayVideoEncoder> video_encoder) { |
| 168 DCHECK(!video_encoder_); | 188 DCHECK(!video_encoder_); |
| 169 | 189 |
| 170 if (!video_encoder) { | 190 if (!video_encoder) { |
| 171 callback.Run(false); | 191 callback.Run(false); |
| 172 return; | 192 return; |
| 173 } | 193 } |
| 174 | 194 |
| (...skipping 11 matching lines...) Expand all Loading... |
| 186 const InitCompletionCallback& callback) { | 206 const InitCompletionCallback& callback) { |
| 187 DCHECK(media_service_); | 207 DCHECK(media_service_); |
| 188 auto error_callback = base::Bind(error_callback_, kErrorUnableSendMedia); | 208 auto error_callback = base::Bind(error_callback_, kErrorUnableSendMedia); |
| 189 media_service_.set_connection_error_handler(error_callback); | 209 media_service_.set_connection_error_handler(error_callback); |
| 190 media_service_->SetDesinationPoint( | 210 media_service_->SetDesinationPoint( |
| 191 sink_ip_address_, | 211 sink_ip_address_, |
| 192 static_cast<int32_t>(sink_rtp_ports_.first), | 212 static_cast<int32_t>(sink_rtp_ports_.first), |
| 193 callback); | 213 callback); |
| 194 } | 214 } |
| 195 | 215 |
| 216 void WiFiDisplayMediaPipeline::OnEncodedAudioUnit( |
| 217 std::unique_ptr<WiFiDisplayEncodedFrame> unit) { |
| 218 DCHECK(packetizer_); |
| 219 const unsigned stream_index = (type_ & wds::VideoSession) ? 1u : 0u; |
| 220 if (!packetizer_->EncodeElementaryStreamUnit(stream_index, unit->bytes(), |
| 221 unit->size(), unit->key_frame, |
| 222 unit->pts, unit->dts, true)) { |
| 223 DVLOG(1) << "Couldn't write audio mpegts packet"; |
| 224 } |
| 225 } |
| 226 |
| 196 void WiFiDisplayMediaPipeline::OnEncodedVideoFrame( | 227 void WiFiDisplayMediaPipeline::OnEncodedVideoFrame( |
| 197 std::unique_ptr<WiFiDisplayEncodedFrame> frame) { | 228 std::unique_ptr<WiFiDisplayEncodedFrame> frame) { |
| 198 DCHECK(packetizer_); | 229 DCHECK(packetizer_); |
| 199 if (!packetizer_->EncodeElementaryStreamUnit(0u, frame->bytes(), | 230 if (!packetizer_->EncodeElementaryStreamUnit(0u, frame->bytes(), |
| 200 frame->size(), frame->key_frame, | 231 frame->size(), frame->key_frame, |
| 201 frame->pts, frame->dts, true)) { | 232 frame->pts, frame->dts, true)) { |
| 202 DVLOG(1) << "Couldn't write video mpegts packet"; | 233 DVLOG(1) << "Couldn't write video mpegts packet"; |
| 203 } | 234 } |
| 204 } | 235 } |
| 205 | 236 |
| 206 bool WiFiDisplayMediaPipeline::OnPacketizedMediaDatagramPacket( | 237 bool WiFiDisplayMediaPipeline::OnPacketizedMediaDatagramPacket( |
| 207 WiFiDisplayMediaDatagramPacket media_datagram_packet) { | 238 WiFiDisplayMediaDatagramPacket media_datagram_packet) { |
| 208 DCHECK(media_service_); | 239 DCHECK(media_service_); |
| 209 media_service_->SendMediaPacket(std::move(media_datagram_packet)); | 240 media_service_->SendMediaPacket(std::move(media_datagram_packet)); |
| 210 return true; | 241 return true; |
| 211 } | 242 } |
| 212 | 243 |
| 213 } // namespace extensions | 244 } // namespace extensions |
| OLD | NEW |