OLD | NEW |
---|---|
1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 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 "content/renderer/media/webmediaplayer_impl.h" | 5 #include "content/renderer/media/webmediaplayer_impl.h" |
6 | 6 |
7 #include <algorithm> | 7 #include <algorithm> |
8 #include <limits> | 8 #include <limits> |
9 #include <string> | 9 #include <string> |
10 #include <vector> | 10 #include <vector> |
11 | 11 |
12 #include "base/bind.h" | 12 #include "base/bind.h" |
13 #include "base/callback.h" | 13 #include "base/callback.h" |
14 #include "base/command_line.h" | 14 #include "base/command_line.h" |
15 #include "base/debug/crash_logging.h" | 15 #include "base/debug/crash_logging.h" |
16 #include "base/message_loop/message_loop_proxy.h" | 16 #include "base/message_loop/message_loop_proxy.h" |
17 #include "base/metrics/histogram.h" | 17 #include "base/metrics/histogram.h" |
18 #include "base/stl_util.h" | |
18 #include "base/strings/string_number_conversions.h" | 19 #include "base/strings/string_number_conversions.h" |
19 #include "base/synchronization/waitable_event.h" | 20 #include "base/synchronization/waitable_event.h" |
20 #include "cc/layers/video_layer.h" | 21 #include "cc/layers/video_layer.h" |
21 #include "content/public/common/content_switches.h" | 22 #include "content/public/common/content_switches.h" |
22 #include "content/renderer/media/buffered_data_source.h" | 23 #include "content/renderer/media/buffered_data_source.h" |
23 #include "content/renderer/media/crypto/key_systems.h" | 24 #include "content/renderer/media/crypto/key_systems.h" |
24 #include "content/renderer/media/texttrack_impl.h" | 25 #include "content/renderer/media/texttrack_impl.h" |
25 #include "content/renderer/media/webaudiosourceprovider_impl.h" | 26 #include "content/renderer/media/webaudiosourceprovider_impl.h" |
26 #include "content/renderer/media/webinbandtexttrack_impl.h" | 27 #include "content/renderer/media/webinbandtexttrack_impl.h" |
27 #include "content/renderer/media/webmediaplayer_delegate.h" | 28 #include "content/renderer/media/webmediaplayer_delegate.h" |
28 #include "content/renderer/media/webmediaplayer_params.h" | 29 #include "content/renderer/media/webmediaplayer_params.h" |
29 #include "content/renderer/media/webmediaplayer_util.h" | 30 #include "content/renderer/media/webmediaplayer_util.h" |
30 #include "content/renderer/media/webmediasource_impl.h" | 31 #include "content/renderer/media/webmediasource_impl.h" |
31 #include "content/renderer/pepper/pepper_webplugin_impl.h" | 32 #include "content/renderer/pepper/pepper_webplugin_impl.h" |
32 #include "gpu/GLES2/gl2extchromium.h" | 33 #include "gpu/GLES2/gl2extchromium.h" |
33 #include "media/audio/null_audio_sink.h" | 34 #include "media/audio/null_audio_sink.h" |
34 #include "media/base/bind_to_loop.h" | 35 #include "media/base/bind_to_loop.h" |
35 #include "media/base/filter_collection.h" | 36 #include "media/base/filter_collection.h" |
36 #include "media/base/limits.h" | 37 #include "media/base/limits.h" |
37 #include "media/base/media_log.h" | 38 #include "media/base/media_log.h" |
38 #include "media/base/media_switches.h" | 39 #include "media/base/media_switches.h" |
39 #include "media/base/pipeline.h" | 40 #include "media/base/pipeline.h" |
41 #include "media/base/text_cue.h" | |
42 #include "media/base/text_renderer.h" | |
40 #include "media/base/video_frame.h" | 43 #include "media/base/video_frame.h" |
41 #include "media/filters/audio_renderer_impl.h" | 44 #include "media/filters/audio_renderer_impl.h" |
42 #include "media/filters/chunk_demuxer.h" | 45 #include "media/filters/chunk_demuxer.h" |
43 #include "media/filters/ffmpeg_audio_decoder.h" | 46 #include "media/filters/ffmpeg_audio_decoder.h" |
44 #include "media/filters/ffmpeg_demuxer.h" | 47 #include "media/filters/ffmpeg_demuxer.h" |
45 #include "media/filters/ffmpeg_video_decoder.h" | 48 #include "media/filters/ffmpeg_video_decoder.h" |
46 #include "media/filters/gpu_video_accelerator_factories.h" | 49 #include "media/filters/gpu_video_accelerator_factories.h" |
47 #include "media/filters/gpu_video_decoder.h" | 50 #include "media/filters/gpu_video_decoder.h" |
48 #include "media/filters/opus_audio_decoder.h" | 51 #include "media/filters/opus_audio_decoder.h" |
52 #include "media/filters/text_decoder_impl.h" | |
49 #include "media/filters/video_renderer_base.h" | 53 #include "media/filters/video_renderer_base.h" |
50 #include "media/filters/vpx_video_decoder.h" | 54 #include "media/filters/vpx_video_decoder.h" |
51 #include "third_party/WebKit/public/platform/WebMediaSource.h" | 55 #include "third_party/WebKit/public/platform/WebMediaSource.h" |
52 #include "third_party/WebKit/public/platform/WebRect.h" | 56 #include "third_party/WebKit/public/platform/WebRect.h" |
53 #include "third_party/WebKit/public/platform/WebSize.h" | 57 #include "third_party/WebKit/public/platform/WebSize.h" |
54 #include "third_party/WebKit/public/platform/WebString.h" | 58 #include "third_party/WebKit/public/platform/WebString.h" |
55 #include "third_party/WebKit/public/platform/WebURL.h" | 59 #include "third_party/WebKit/public/platform/WebURL.h" |
56 #include "third_party/WebKit/public/web/WebRuntimeFeatures.h" | 60 #include "third_party/WebKit/public/web/WebRuntimeFeatures.h" |
57 #include "third_party/WebKit/public/web/WebView.h" | 61 #include "third_party/WebKit/public/web/WebView.h" |
58 #include "v8/include/v8.h" | 62 #include "v8/include/v8.h" |
(...skipping 894 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
953 const WebKit::WebString weblanguage = WebKit::WebString::fromUTF8(language); | 957 const WebKit::WebString weblanguage = WebKit::WebString::fromUTF8(language); |
954 | 958 |
955 WebInbandTextTrackImpl* const text_track = | 959 WebInbandTextTrackImpl* const text_track = |
956 new WebInbandTextTrackImpl(webkind, weblabel, weblanguage, | 960 new WebInbandTextTrackImpl(webkind, weblabel, weblanguage, |
957 text_track_index_++); | 961 text_track_index_++); |
958 | 962 |
959 return scoped_ptr<media::TextTrack>(new TextTrackImpl(GetClient(), | 963 return scoped_ptr<media::TextTrack>(new TextTrackImpl(GetClient(), |
960 text_track)); | 964 text_track)); |
961 } | 965 } |
962 | 966 |
967 void WebMediaPlayerImpl::AddTextStream(media::DemuxerStream* text_stream, | |
968 media::TextKind kind, | |
969 const std::string& label, | |
970 const std::string& language) { | |
971 DCHECK(main_loop_->BelongsToCurrentThread()); | |
972 scoped_ptr<media::TextTrack> text_track = OnTextTrack(kind, label, language); | |
973 text_track_map_[text_stream] = text_track.release(); | |
acolwell GONE FROM CHROMIUM
2013/09/25 23:23:10
I don't think you need the map here. You can hide
Matthew Heaney (Chromium)
2013/09/29 03:31:23
Done.
| |
974 } | |
975 | |
963 void WebMediaPlayerImpl::OnKeyError(const std::string& session_id, | 976 void WebMediaPlayerImpl::OnKeyError(const std::string& session_id, |
964 media::MediaKeys::KeyError error_code, | 977 media::MediaKeys::KeyError error_code, |
965 int system_code) { | 978 int system_code) { |
966 DCHECK(main_loop_->BelongsToCurrentThread()); | 979 DCHECK(main_loop_->BelongsToCurrentThread()); |
967 | 980 |
968 EmeUMAHistogramEnumeration(current_key_system_, "KeyError", | 981 EmeUMAHistogramEnumeration(current_key_system_, "KeyError", |
969 error_code, media::MediaKeys::kMaxKeyError); | 982 error_code, media::MediaKeys::kMaxKeyError); |
970 | 983 |
971 GetClient()->keyError( | 984 GetClient()->keyError( |
972 current_key_system_, | 985 current_key_system_, |
(...skipping 48 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1021 } | 1034 } |
1022 | 1035 |
1023 void WebMediaPlayerImpl::StartPipeline() { | 1036 void WebMediaPlayerImpl::StartPipeline() { |
1024 const CommandLine* cmd_line = CommandLine::ForCurrentProcess(); | 1037 const CommandLine* cmd_line = CommandLine::ForCurrentProcess(); |
1025 bool increase_preroll_on_underflow = true; | 1038 bool increase_preroll_on_underflow = true; |
1026 | 1039 |
1027 // Keep track if this is a MSE or non-MSE playback. | 1040 // Keep track if this is a MSE or non-MSE playback. |
1028 UMA_HISTOGRAM_BOOLEAN("Media.MSE.Playback", | 1041 UMA_HISTOGRAM_BOOLEAN("Media.MSE.Playback", |
1029 (load_type_ == LoadTypeMediaSource)); | 1042 (load_type_ == LoadTypeMediaSource)); |
1030 | 1043 |
1044 const bool enable_inband_text_tracks = | |
1045 cmd_line->HasSwitch(switches::kEnableInbandTextTracks); | |
1046 | |
1031 // Figure out which demuxer to use. | 1047 // Figure out which demuxer to use. |
1032 if (load_type_ != LoadTypeMediaSource) { | 1048 if (load_type_ != LoadTypeMediaSource) { |
1033 DCHECK(!chunk_demuxer_); | 1049 DCHECK(!chunk_demuxer_); |
1034 DCHECK(data_source_); | 1050 DCHECK(data_source_); |
1035 | 1051 |
1036 demuxer_.reset(new media::FFmpegDemuxer( | 1052 demuxer_.reset(new media::FFmpegDemuxer( |
1037 media_loop_, data_source_.get(), | 1053 media_loop_, data_source_.get(), |
1038 BIND_TO_RENDER_LOOP_1(&WebMediaPlayerImpl::OnNeedKey, ""), | 1054 BIND_TO_RENDER_LOOP_1(&WebMediaPlayerImpl::OnNeedKey, ""), |
1055 enable_inband_text_tracks, | |
1039 media_log_)); | 1056 media_log_)); |
1040 } else { | 1057 } else { |
1041 DCHECK(!chunk_demuxer_); | 1058 DCHECK(!chunk_demuxer_); |
1042 DCHECK(!data_source_); | 1059 DCHECK(!data_source_); |
1043 | 1060 |
1061 // TODO(matthewjheaney): Once we have resolved the threading issues, | |
1062 // change the chunk demuxer to user the same mechanism as for ffmpeg. | |
1044 media::AddTextTrackCB add_text_track_cb; | 1063 media::AddTextTrackCB add_text_track_cb; |
1045 | 1064 |
1046 if (cmd_line->HasSwitch(switches::kEnableInbandTextTracks)) { | 1065 if (enable_inband_text_tracks) { |
1047 add_text_track_cb = | 1066 add_text_track_cb = |
1048 base::Bind(&WebMediaPlayerImpl::OnTextTrack, base::Unretained(this)); | 1067 base::Bind(&WebMediaPlayerImpl::OnTextTrack, base::Unretained(this)); |
1049 } | 1068 } |
1050 | 1069 |
1051 chunk_demuxer_ = new media::ChunkDemuxer( | 1070 chunk_demuxer_ = new media::ChunkDemuxer( |
1052 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::OnDemuxerOpened), | 1071 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::OnDemuxerOpened), |
1053 BIND_TO_RENDER_LOOP_1(&WebMediaPlayerImpl::OnNeedKey, ""), | 1072 BIND_TO_RENDER_LOOP_1(&WebMediaPlayerImpl::OnNeedKey, ""), |
1054 add_text_track_cb, | 1073 add_text_track_cb, |
1055 base::Bind(&LogMediaSourceError, media_log_)); | 1074 base::Bind(&LogMediaSourceError, media_log_)); |
1056 demuxer_.reset(chunk_demuxer_); | 1075 demuxer_.reset(chunk_demuxer_); |
(...skipping 57 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1114 scoped_ptr<media::VideoRenderer> video_renderer( | 1133 scoped_ptr<media::VideoRenderer> video_renderer( |
1115 new media::VideoRendererBase( | 1134 new media::VideoRendererBase( |
1116 media_loop_, | 1135 media_loop_, |
1117 video_decoders.Pass(), | 1136 video_decoders.Pass(), |
1118 set_decryptor_ready_cb, | 1137 set_decryptor_ready_cb, |
1119 base::Bind(&WebMediaPlayerImpl::FrameReady, base::Unretained(this)), | 1138 base::Bind(&WebMediaPlayerImpl::FrameReady, base::Unretained(this)), |
1120 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::SetOpaque), | 1139 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::SetOpaque), |
1121 true)); | 1140 true)); |
1122 filter_collection->SetVideoRenderer(video_renderer.Pass()); | 1141 filter_collection->SetVideoRenderer(video_renderer.Pass()); |
1123 | 1142 |
1143 if (load_type_ != LoadTypeMediaSource && enable_inband_text_tracks) { | |
1144 scoped_ptr<media::TextDecoder> text_decoder( | |
1145 new media::TextDecoderImpl(media_loop_)); | |
1146 | |
1147 scoped_ptr<media::TextRenderer> text_renderer( | |
1148 new media::TextRenderer( | |
1149 media_loop_, | |
1150 text_decoder.Pass(), | |
1151 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::AddTextStream), | |
1152 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::CueReady))); | |
1153 | |
1154 filter_collection->SetTextRenderer(text_renderer.Pass()); | |
1155 } | |
1156 | |
1124 // ... and we're ready to go! | 1157 // ... and we're ready to go! |
1125 starting_ = true; | 1158 starting_ = true; |
1126 pipeline_->Start( | 1159 pipeline_->Start( |
1127 filter_collection.Pass(), | 1160 filter_collection.Pass(), |
1128 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::OnPipelineEnded), | 1161 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::OnPipelineEnded), |
1129 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::OnPipelineError), | 1162 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::OnPipelineError), |
1130 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::OnPipelineSeek), | 1163 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::OnPipelineSeek), |
1131 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::OnPipelineBufferingState), | 1164 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::OnPipelineBufferingState), |
1132 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::OnDurationChange)); | 1165 BIND_TO_RENDER_LOOP(&WebMediaPlayerImpl::OnDurationChange)); |
1133 } | 1166 } |
(...skipping 36 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1170 gpu_factories_ = NULL; | 1203 gpu_factories_ = NULL; |
1171 } | 1204 } |
1172 | 1205 |
1173 // Make sure to kill the pipeline so there's no more media threads running. | 1206 // Make sure to kill the pipeline so there's no more media threads running. |
1174 // Note: stopping the pipeline might block for a long time. | 1207 // Note: stopping the pipeline might block for a long time. |
1175 base::WaitableEvent waiter(false, false); | 1208 base::WaitableEvent waiter(false, false); |
1176 pipeline_->Stop(base::Bind( | 1209 pipeline_->Stop(base::Bind( |
1177 &base::WaitableEvent::Signal, base::Unretained(&waiter))); | 1210 &base::WaitableEvent::Signal, base::Unretained(&waiter))); |
1178 waiter.Wait(); | 1211 waiter.Wait(); |
1179 | 1212 |
1213 STLDeleteValues(&text_track_map_); | |
1214 | |
1180 // Let V8 know we are not using extra resources anymore. | 1215 // Let V8 know we are not using extra resources anymore. |
1181 if (incremented_externally_allocated_memory_) { | 1216 if (incremented_externally_allocated_memory_) { |
1182 v8::V8::AdjustAmountOfExternalAllocatedMemory(-kPlayerExtraMemory); | 1217 v8::V8::AdjustAmountOfExternalAllocatedMemory(-kPlayerExtraMemory); |
1183 incremented_externally_allocated_memory_ = false; | 1218 incremented_externally_allocated_memory_ = false; |
1184 } | 1219 } |
1185 | 1220 |
1186 // Release any final references now that everything has stopped. | 1221 // Release any final references now that everything has stopped. |
1187 pipeline_.reset(); | 1222 pipeline_.reset(); |
1188 demuxer_.reset(); | 1223 demuxer_.reset(); |
1189 data_source_.reset(); | 1224 data_source_.reset(); |
(...skipping 46 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1236 current_frame_ = frame; | 1271 current_frame_ = frame; |
1237 | 1272 |
1238 if (pending_repaint_) | 1273 if (pending_repaint_) |
1239 return; | 1274 return; |
1240 | 1275 |
1241 pending_repaint_ = true; | 1276 pending_repaint_ = true; |
1242 main_loop_->PostTask(FROM_HERE, base::Bind( | 1277 main_loop_->PostTask(FROM_HERE, base::Bind( |
1243 &WebMediaPlayerImpl::Repaint, AsWeakPtr())); | 1278 &WebMediaPlayerImpl::Repaint, AsWeakPtr())); |
1244 } | 1279 } |
1245 | 1280 |
1281 void WebMediaPlayerImpl::CueReady( | |
acolwell GONE FROM CHROMIUM
2013/09/25 23:23:10
You shouldn't need this. You can hide the thread h
Matthew Heaney (Chromium)
2013/09/29 03:31:23
Done.
| |
1282 media::DemuxerStream* text_stream, | |
1283 const scoped_refptr<media::TextCue>& text_cue) { | |
1284 DCHECK(main_loop_->BelongsToCurrentThread()); | |
1285 DCHECK_NE(text_stream, static_cast<media::DemuxerStream*>(NULL)); | |
1286 DCHECK(text_cue != NULL && !text_cue->text().empty()); | |
1287 | |
1288 TextTrackMap::iterator it = text_track_map_.find(text_stream); | |
1289 DCHECK(it != text_track_map_.end()); | |
1290 | |
1291 media::TextTrack* text_track = it->second; | |
1292 DCHECK_NE(text_track, static_cast<media::TextTrack*>(NULL)); | |
1293 | |
1294 base::TimeDelta start = text_cue->timestamp(); | |
1295 base::TimeDelta end = start + text_cue->duration(); | |
1296 | |
1297 text_track->addWebVTTCue(start, end, | |
1298 text_cue->id(), | |
1299 text_cue->text(), | |
1300 text_cue->settings()); | |
1301 } | |
1302 | |
1246 } // namespace content | 1303 } // namespace content |
OLD | NEW |