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

Side by Side Diff: content/renderer/media/webmediaplayer_impl.cc

Issue 23702007: Render inband text tracks in the media pipeline (Closed) Base URL: http://git.chromium.org/chromium/src.git@master
Patch Set: incorporated more of aaron's comments Created 7 years, 3 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 unified diff | Download patch
OLDNEW
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
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
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
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
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
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
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698