Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 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 "media/webm/webm_stream_parser.h" | 5 #include "media/webm/webm_stream_parser.h" |
| 6 | 6 |
| 7 #include <string> | 7 #include <string> |
| 8 | 8 |
| 9 #include "base/callback.h" | 9 #include "base/callback.h" |
| 10 #include "base/logging.h" | 10 #include "base/logging.h" |
| 11 #include "base/stl_util.h" | |
| 11 #include "media/webm/webm_cluster_parser.h" | 12 #include "media/webm/webm_cluster_parser.h" |
| 12 #include "media/webm/webm_constants.h" | 13 #include "media/webm/webm_constants.h" |
| 13 #include "media/webm/webm_content_encodings.h" | 14 #include "media/webm/webm_content_encodings.h" |
| 14 #include "media/webm/webm_crypto_helpers.h" | 15 #include "media/webm/webm_crypto_helpers.h" |
| 15 #include "media/webm/webm_info_parser.h" | 16 #include "media/webm/webm_info_parser.h" |
| 16 #include "media/webm/webm_tracks_parser.h" | 17 #include "media/webm/webm_tracks_parser.h" |
| 18 #include "media/webm/webm_webvtt_parser.h" | |
| 17 | 19 |
| 18 namespace media { | 20 namespace media { |
| 19 | 21 |
| 20 WebMStreamParser::WebMStreamParser() | 22 WebMStreamParser::WebMStreamParser() |
| 21 : state_(kWaitingForInit), | 23 : state_(kWaitingForInit), |
| 22 waiting_for_buffers_(false) { | 24 waiting_for_buffers_(false) { |
| 23 } | 25 } |
| 24 | 26 |
| 25 WebMStreamParser::~WebMStreamParser() {} | 27 WebMStreamParser::~WebMStreamParser() { |
| 28 STLDeleteValues(&text_track_map_); | |
| 29 } | |
| 26 | 30 |
| 27 void WebMStreamParser::Init(const InitCB& init_cb, | 31 void WebMStreamParser::Init(const InitCB& init_cb, |
| 28 const NewConfigCB& config_cb, | 32 const NewConfigCB& config_cb, |
| 29 const NewBuffersCB& audio_cb, | 33 const NewBuffersCB& audio_cb, |
| 30 const NewBuffersCB& video_cb, | 34 const NewBuffersCB& video_cb, |
| 35 const NewBuffersCB& text_cb, | |
| 31 const NeedKeyCB& need_key_cb, | 36 const NeedKeyCB& need_key_cb, |
| 37 const AddTextTrackCB& add_text_track_cb, | |
| 32 const NewMediaSegmentCB& new_segment_cb, | 38 const NewMediaSegmentCB& new_segment_cb, |
| 33 const base::Closure& end_of_segment_cb, | 39 const base::Closure& end_of_segment_cb, |
| 34 const LogCB& log_cb) { | 40 const LogCB& log_cb) { |
| 35 DCHECK_EQ(state_, kWaitingForInit); | 41 DCHECK_EQ(state_, kWaitingForInit); |
| 36 DCHECK(init_cb_.is_null()); | 42 DCHECK(init_cb_.is_null()); |
| 37 DCHECK(!init_cb.is_null()); | 43 DCHECK(!init_cb.is_null()); |
| 38 DCHECK(!config_cb.is_null()); | 44 DCHECK(!config_cb.is_null()); |
| 39 DCHECK(!audio_cb.is_null() || !video_cb.is_null()); | 45 DCHECK(!audio_cb.is_null() || !video_cb.is_null()); |
| 46 DCHECK(!text_cb.is_null()); | |
| 40 DCHECK(!need_key_cb.is_null()); | 47 DCHECK(!need_key_cb.is_null()); |
| 41 DCHECK(!new_segment_cb.is_null()); | 48 DCHECK(!new_segment_cb.is_null()); |
| 42 DCHECK(!end_of_segment_cb.is_null()); | 49 DCHECK(!end_of_segment_cb.is_null()); |
| 43 | 50 |
| 44 ChangeState(kParsingHeaders); | 51 ChangeState(kParsingHeaders); |
| 45 init_cb_ = init_cb; | 52 init_cb_ = init_cb; |
| 46 config_cb_ = config_cb; | 53 config_cb_ = config_cb; |
| 47 audio_cb_ = audio_cb; | 54 audio_cb_ = audio_cb; |
| 48 video_cb_ = video_cb; | 55 video_cb_ = video_cb; |
| 56 text_cb_ = text_cb; | |
| 49 need_key_cb_ = need_key_cb; | 57 need_key_cb_ = need_key_cb; |
| 58 add_text_track_cb_ = add_text_track_cb; | |
| 50 new_segment_cb_ = new_segment_cb; | 59 new_segment_cb_ = new_segment_cb; |
| 51 end_of_segment_cb_ = end_of_segment_cb; | 60 end_of_segment_cb_ = end_of_segment_cb; |
| 52 log_cb_ = log_cb; | 61 log_cb_ = log_cb; |
| 53 } | 62 } |
| 54 | 63 |
| 55 void WebMStreamParser::Flush() { | 64 void WebMStreamParser::Flush() { |
| 56 DCHECK_NE(state_, kWaitingForInit); | 65 DCHECK_NE(state_, kWaitingForInit); |
| 57 | 66 |
| 58 byte_queue_.Reset(); | 67 byte_queue_.Reset(); |
| 59 | 68 |
| (...skipping 131 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 191 | 200 |
| 192 const VideoDecoderConfig& video_config = tracks_parser.video_decoder_config(); | 201 const VideoDecoderConfig& video_config = tracks_parser.video_decoder_config(); |
| 193 if (video_config.is_encrypted()) | 202 if (video_config.is_encrypted()) |
| 194 FireNeedKey(tracks_parser.video_encryption_key_id()); | 203 FireNeedKey(tracks_parser.video_encryption_key_id()); |
| 195 | 204 |
| 196 if (!config_cb_.Run(audio_config, video_config)) { | 205 if (!config_cb_.Run(audio_config, video_config)) { |
| 197 DVLOG(1) << "New config data isn't allowed."; | 206 DVLOG(1) << "New config data isn't allowed."; |
| 198 return -1; | 207 return -1; |
| 199 } | 208 } |
| 200 | 209 |
| 210 typedef std::set<int> TextTracksSet; | |
|
acolwell GONE FROM CHROMIUM
2013/05/10 18:41:39
nit: This typedef should probably be in WebMTracks
Matthew Heaney (Chromium)
2013/05/11 07:29:13
Done.
| |
| 211 const TextTracksSet& text_tracks = tracks_parser.text_tracks(); | |
| 212 | |
| 213 for (TextTracksSet::const_iterator itr = text_tracks.begin(); | |
| 214 itr != text_tracks.end(); ++itr) { | |
| 215 // TODO(matthewjheaney): parse track kind, name, lang from track header | |
|
acolwell GONE FROM CHROMIUM
2013/05/10 18:41:39
I'd like to see a separate CL with the necessary c
Matthew Heaney (Chromium)
2013/05/11 07:29:13
Done.
| |
| 216 scoped_ptr<TextTrack> text_cb = | |
|
acolwell GONE FROM CHROMIUM
2013/05/10 18:41:39
nit: s/text_cb/text_track/
Matthew Heaney (Chromium)
2013/05/11 07:29:13
Done.
Matthew Heaney (Chromium)
2013/05/11 07:29:13
Done.
| |
| 217 add_text_track_cb_.Run(kTextSubtitles, "", ""); | |
| 218 | |
| 219 // Assume ownership of pointer | |
| 220 TextTrack* const text_cb_ptr = text_cb.release(); | |
|
acolwell GONE FROM CHROMIUM
2013/05/10 18:41:39
nit: Just inline please.
Matthew Heaney (Chromium)
2013/05/11 07:29:13
Done.
Matthew Heaney (Chromium)
2013/05/11 07:29:13
Done.
| |
| 221 | |
| 222 // Cache text track object, for use later when we have text track buffers. | |
| 223 // (The text track objects are deallocated in the dtor for this class.) | |
| 224 text_track_map_.insert(std::make_pair(*itr, text_cb_ptr)); | |
| 225 } | |
| 226 | |
| 201 cluster_parser_.reset(new WebMClusterParser( | 227 cluster_parser_.reset(new WebMClusterParser( |
| 202 info_parser.timecode_scale(), | 228 info_parser.timecode_scale(), |
| 203 tracks_parser.audio_track_num(), | 229 tracks_parser.audio_track_num(), |
| 204 tracks_parser.video_track_num(), | 230 tracks_parser.video_track_num(), |
| 205 tracks_parser.text_tracks(), | 231 tracks_parser.text_tracks(), |
| 206 tracks_parser.ignored_tracks(), | 232 tracks_parser.ignored_tracks(), |
| 207 tracks_parser.audio_encryption_key_id(), | 233 tracks_parser.audio_encryption_key_id(), |
| 208 tracks_parser.video_encryption_key_id(), | 234 tracks_parser.video_encryption_key_id(), |
| 209 log_cb_)); | 235 log_cb_)); |
| 210 | 236 |
| (...skipping 50 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 261 new_segment_cb_.Run(cluster_start_time); | 287 new_segment_cb_.Run(cluster_start_time); |
| 262 waiting_for_buffers_ = false; | 288 waiting_for_buffers_ = false; |
| 263 } | 289 } |
| 264 | 290 |
| 265 if (!audio_buffers.empty() && !audio_cb_.Run(audio_buffers)) | 291 if (!audio_buffers.empty() && !audio_cb_.Run(audio_buffers)) |
| 266 return -1; | 292 return -1; |
| 267 | 293 |
| 268 if (!video_buffers.empty() && !video_cb_.Run(video_buffers)) | 294 if (!video_buffers.empty() && !video_cb_.Run(video_buffers)) |
| 269 return -1; | 295 return -1; |
| 270 | 296 |
| 297 WebMClusterParser::TextTrackIterator tt_iter = | |
| 298 cluster_parser_->CreateTextTrackIterator(); | |
| 299 | |
| 300 int tt_num; | |
|
acolwell GONE FROM CHROMIUM
2013/05/10 18:41:39
nit: s/tt_num/track_id/ or s/tt_num/track_number/
Matthew Heaney (Chromium)
2013/05/11 07:29:13
Done.
| |
| 301 const BufferQueue* tt_buffers; | |
|
acolwell GONE FROM CHROMIUM
2013/05/10 18:41:39
nit: s/tt_buffers/buffers since the tt doesn't rea
Matthew Heaney (Chromium)
2013/05/11 07:29:13
Done.
| |
| 302 | |
| 303 while (tt_iter(&tt_num, &tt_buffers)) { | |
| 304 TextTrackMap::iterator cb_iter = text_track_map_.find(tt_num); | |
| 305 | |
| 306 if (cb_iter == text_track_map_.end()) | |
| 307 continue; | |
| 308 | |
| 309 if (!tt_buffers->empty() && !text_cb_.Run(*tt_buffers)) | |
| 310 return -1; | |
| 311 | |
| 312 OnTextBuffers(*tt_buffers, cb_iter->second); | |
| 313 } | |
| 314 | |
| 271 if (cluster_ended) | 315 if (cluster_ended) |
| 272 end_of_segment_cb_.Run(); | 316 end_of_segment_cb_.Run(); |
| 273 | 317 |
| 274 return bytes_parsed; | 318 return bytes_parsed; |
| 275 } | 319 } |
| 276 | 320 |
| 277 void WebMStreamParser::FireNeedKey(const std::string& key_id) { | 321 void WebMStreamParser::FireNeedKey(const std::string& key_id) { |
| 278 int key_id_size = key_id.size(); | 322 int key_id_size = key_id.size(); |
| 279 DCHECK_GT(key_id_size, 0); | 323 DCHECK_GT(key_id_size, 0); |
| 280 scoped_ptr<uint8[]> key_id_array(new uint8[key_id_size]); | 324 scoped_ptr<uint8[]> key_id_array(new uint8[key_id_size]); |
| 281 memcpy(key_id_array.get(), key_id.data(), key_id_size); | 325 memcpy(key_id_array.get(), key_id.data(), key_id_size); |
| 282 need_key_cb_.Run(kWebMEncryptInitDataType, key_id_array.Pass(), key_id_size); | 326 need_key_cb_.Run(kWebMEncryptInitDataType, key_id_array.Pass(), key_id_size); |
| 283 } | 327 } |
| 284 | 328 |
| 329 void WebMStreamParser::OnTextBuffers(const StreamParser::BufferQueue& buffers, | |
| 330 TextTrack* text_cb) { | |
|
acolwell GONE FROM CHROMIUM
2013/05/10 18:41:39
nit: when this is moved to ChunkDemuxer s/text_cb/
Matthew Heaney (Chromium)
2013/05/11 07:29:13
Done.
| |
| 331 for (StreamParser::BufferQueue::const_iterator itr = buffers.begin(); | |
| 332 itr != buffers.end(); ++itr) { | |
| 333 OnTextBuffer(*itr, text_cb); | |
| 334 } | |
| 335 } | |
| 336 | |
| 337 void WebMStreamParser::OnTextBuffer(const StreamParserBuffer* buffer, | |
| 338 TextTrack* text_cb) { | |
| 339 const base::TimeDelta start = buffer->GetTimestamp(); | |
| 340 const base::TimeDelta end = start + buffer->GetDuration(); | |
| 341 | |
| 342 std::string id, settings, content; | |
| 343 | |
| 344 WebMWebVTTParser::Parse(buffer->GetData(), | |
| 345 buffer->GetDataSize(), | |
| 346 &id, &settings, &content); | |
| 347 | |
| 348 text_cb->addWebVTTCue(start, end, id, content, settings); | |
| 349 } | |
| 350 | |
| 285 } // namespace media | 351 } // namespace media |
| OLD | NEW |