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 "content/renderer/media/video_capture_impl.h" | 5 #include "content/renderer/media/video_capture_impl.h" |
6 | 6 |
7 #include "base/bind.h" | 7 #include "base/bind.h" |
8 #include "base/callback_helpers.h" | |
8 #include "base/stl_util.h" | 9 #include "base/stl_util.h" |
9 #include "content/child/child_process.h" | 10 #include "content/child/child_process.h" |
11 #include "content/common/media/encoded_video_capture_messages.h" | |
10 #include "content/common/media/video_capture_messages.h" | 12 #include "content/common/media/video_capture_messages.h" |
11 #include "media/base/limits.h" | 13 #include "media/base/limits.h" |
12 | 14 |
13 namespace content { | 15 namespace content { |
14 | 16 |
15 struct VideoCaptureImpl::DIBBuffer { | 17 struct VideoCaptureImpl::DIBBuffer { |
16 public: | 18 public: |
17 DIBBuffer( | 19 DIBBuffer( |
18 base::SharedMemory* d, | 20 base::SharedMemory* d, |
19 media::VideoCapture::VideoFrameBuffer* ptr) | 21 media::VideoCapture::VideoFrameBuffer* ptr) |
(...skipping 74 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
94 base::Bind(&VideoCaptureImpl::DoStartCaptureOnCaptureThread, | 96 base::Bind(&VideoCaptureImpl::DoStartCaptureOnCaptureThread, |
95 base::Unretained(this), handler, capability)); | 97 base::Unretained(this), handler, capability)); |
96 } | 98 } |
97 | 99 |
98 void VideoCaptureImpl::StopCapture(media::VideoCapture::EventHandler* handler) { | 100 void VideoCaptureImpl::StopCapture(media::VideoCapture::EventHandler* handler) { |
99 capture_message_loop_proxy_->PostTask(FROM_HERE, | 101 capture_message_loop_proxy_->PostTask(FROM_HERE, |
100 base::Bind(&VideoCaptureImpl::DoStopCaptureOnCaptureThread, | 102 base::Bind(&VideoCaptureImpl::DoStopCaptureOnCaptureThread, |
101 base::Unretained(this), handler)); | 103 base::Unretained(this), handler)); |
102 } | 104 } |
103 | 105 |
106 void VideoCaptureImpl::RequestCapabilities( | |
107 const media::EncodedVideoSource::RequestCapabilitiesCallback& callback) { | |
108 capture_message_loop_proxy_->PostTask(FROM_HERE, | |
109 base::Bind(&VideoCaptureImpl::DoRequestCapabilitiesOnCaptureThread, | |
110 base::Unretained(this), callback)); | |
Ami GONE FROM CHROMIUM
2013/06/13 22:56:58
Yowza. So, Unretained(this) is safe (even though
hshi1
2013/06/13 23:33:40
Ok so I assume there's no action item for me then.
Ami GONE FROM CHROMIUM
2013/06/18 20:56:27
If you felt like adding this comment to the class
hshi1
2013/06/18 23:01:19
I'd feel more comfortable if you add the appropria
| |
111 } | |
112 | |
113 void VideoCaptureImpl::StartFetchCapabilities() { | |
114 Send(new EncodedVideoCaptureHostMsg_GetCapabilities( | |
115 device_id_, current_params_.session_id)); | |
116 } | |
117 | |
118 void VideoCaptureImpl::OpenBitstream( | |
119 media::EncodedVideoSource::Client* client, | |
120 const media::VideoEncodingParameters& params) { | |
121 capture_message_loop_proxy_->PostTask(FROM_HERE, | |
122 base::Bind(&VideoCaptureImpl::DoOpenBitstreamOnCaptureThread, | |
123 base::Unretained(this), client, params)); | |
124 } | |
125 | |
126 void VideoCaptureImpl::CloseBitstream() { | |
127 capture_message_loop_proxy_->PostTask(FROM_HERE, | |
128 base::Bind(&VideoCaptureImpl::DoCloseBitstreamOnCaptureThread, | |
129 base::Unretained(this))); | |
130 } | |
131 | |
132 void VideoCaptureImpl::ReturnBitstreamBuffer( | |
133 scoped_refptr<const media::EncodedBitstreamBuffer> buffer) { | |
134 Send(new EncodedVideoCaptureHostMsg_BitstreamBufferConsumed( | |
135 device_id_, buffer->buffer_id())); | |
136 } | |
137 | |
138 void VideoCaptureImpl::TrySetBitstreamConfig( | |
139 const media::RuntimeVideoEncodingParameters& params) { | |
140 Send(new EncodedVideoCaptureHostMsg_TryConfigureBitstream( | |
141 device_id_, params)); | |
142 } | |
143 | |
144 void VideoCaptureImpl::OnEncodingCapabilitiesAvailable( | |
145 const media::VideoEncodingCapabilities& capabilities) { | |
146 capture_message_loop_proxy_->PostTask(FROM_HERE, base::Bind( | |
147 &VideoCaptureImpl::DoNotifyCapabilitiesAvailableOnCaptureThread, | |
148 base::Unretained(this), capabilities)); | |
149 } | |
150 | |
151 void VideoCaptureImpl::OnEncodedBitstreamOpened( | |
152 const media::VideoEncodingParameters& params, | |
153 const std::vector<base::SharedMemoryHandle>& buffers, | |
154 uint32 buffer_size) { | |
155 capture_message_loop_proxy_->PostTask(FROM_HERE, | |
156 base::Bind(&VideoCaptureImpl::DoNotifyBitstreamOpenedOnCaptureThread, | |
157 base::Unretained(this), params, buffers, buffer_size)); | |
158 } | |
159 | |
160 void VideoCaptureImpl::OnEncodedBitstreamClosed() { | |
161 capture_message_loop_proxy_->PostTask(FROM_HERE, | |
162 base::Bind(&VideoCaptureImpl::DoNotifyBitstreamClosedOnCaptureThread, | |
163 base::Unretained(this))); | |
164 } | |
165 | |
166 void VideoCaptureImpl::OnEncodingConfigChanged( | |
167 const media::RuntimeVideoEncodingParameters& params) { | |
168 capture_message_loop_proxy_->PostTask(FROM_HERE, | |
169 base::Bind( | |
170 &VideoCaptureImpl::DoNotifyBitstreamConfigChangedOnCaptureThread, | |
171 base::Unretained(this), params)); | |
172 } | |
173 | |
174 void VideoCaptureImpl::OnEncodedBufferReady( | |
175 int buffer_id, | |
176 uint32 size, | |
177 const media::BufferEncodingMetadata& metadata) { | |
178 capture_message_loop_proxy_->PostTask(FROM_HERE, | |
179 base::Bind(&VideoCaptureImpl::DoNotifyBitstreamBufferReadyOnCaptureThread, | |
180 base::Unretained(this), buffer_id, size, metadata)); | |
181 } | |
182 | |
104 void VideoCaptureImpl::FeedBuffer(scoped_refptr<VideoFrameBuffer> buffer) { | 183 void VideoCaptureImpl::FeedBuffer(scoped_refptr<VideoFrameBuffer> buffer) { |
105 capture_message_loop_proxy_->PostTask(FROM_HERE, | 184 capture_message_loop_proxy_->PostTask(FROM_HERE, |
106 base::Bind(&VideoCaptureImpl::DoFeedBufferOnCaptureThread, | 185 base::Bind(&VideoCaptureImpl::DoFeedBufferOnCaptureThread, |
107 base::Unretained(this), buffer)); | 186 base::Unretained(this), buffer)); |
108 } | 187 } |
109 | 188 |
110 void VideoCaptureImpl::OnBufferCreated( | 189 void VideoCaptureImpl::OnBufferCreated( |
111 base::SharedMemoryHandle handle, | 190 base::SharedMemoryHandle handle, |
112 int length, int buffer_id) { | 191 int length, int buffer_id) { |
113 capture_message_loop_proxy_->PostTask(FROM_HERE, | 192 capture_message_loop_proxy_->PostTask(FROM_HERE, |
(...skipping 185 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
299 it->first->OnBufferReady(this, buffer); | 378 it->first->OnBufferReady(this, buffer); |
300 } | 379 } |
301 cached_dibs_[buffer_id]->references = clients_.size(); | 380 cached_dibs_[buffer_id]->references = clients_.size(); |
302 } | 381 } |
303 | 382 |
304 void VideoCaptureImpl::DoStateChangedOnCaptureThread(VideoCaptureState state) { | 383 void VideoCaptureImpl::DoStateChangedOnCaptureThread(VideoCaptureState state) { |
305 DCHECK(capture_message_loop_proxy_->BelongsToCurrentThread()); | 384 DCHECK(capture_message_loop_proxy_->BelongsToCurrentThread()); |
306 | 385 |
307 switch (state) { | 386 switch (state) { |
308 case VIDEO_CAPTURE_STATE_STARTED: | 387 case VIDEO_CAPTURE_STATE_STARTED: |
388 StartFetchCapabilities(); | |
309 break; | 389 break; |
310 case VIDEO_CAPTURE_STATE_STOPPED: | 390 case VIDEO_CAPTURE_STATE_STOPPED: |
311 state_ = VIDEO_CAPTURE_STATE_STOPPED; | 391 state_ = VIDEO_CAPTURE_STATE_STOPPED; |
312 DVLOG(1) << "OnStateChanged: stopped!, device_id = " << device_id_; | 392 DVLOG(1) << "OnStateChanged: stopped!, device_id = " << device_id_; |
313 STLDeleteValues(&cached_dibs_); | 393 STLDeleteValues(&cached_dibs_); |
314 if (!clients_.empty() || !clients_pending_on_restart_.empty()) | 394 if (!clients_.empty() || !clients_pending_on_restart_.empty()) |
315 RestartCapture(); | 395 RestartCapture(); |
316 break; | 396 break; |
317 case VIDEO_CAPTURE_STATE_PAUSED: | 397 case VIDEO_CAPTURE_STATE_PAUSED: |
318 for (ClientInfo::iterator it = clients_.begin(); | 398 for (ClientInfo::iterator it = clients_.begin(); |
(...skipping 54 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
373 } | 453 } |
374 } | 454 } |
375 | 455 |
376 void VideoCaptureImpl::DoSuspendCaptureOnCaptureThread(bool suspend) { | 456 void VideoCaptureImpl::DoSuspendCaptureOnCaptureThread(bool suspend) { |
377 DVLOG(1) << "DoSuspendCapture: suspend " << (suspend ? "yes" : "no"); | 457 DVLOG(1) << "DoSuspendCapture: suspend " << (suspend ? "yes" : "no"); |
378 DCHECK(capture_message_loop_proxy_->BelongsToCurrentThread()); | 458 DCHECK(capture_message_loop_proxy_->BelongsToCurrentThread()); |
379 | 459 |
380 suspended_ = suspend; | 460 suspended_ = suspend; |
381 } | 461 } |
382 | 462 |
463 void VideoCaptureImpl::DoRequestCapabilitiesOnCaptureThread( | |
464 const RequestCapabilitiesCallback& callback) { | |
465 DCHECK(capture_message_loop_proxy_->BelongsToCurrentThread()); | |
466 DCHECK(callback_.is_null()); | |
467 callback_ = callback; | |
468 | |
469 // Invoke callback immediately if capabilities are already available. | |
470 if (!capabilities_.empty()) | |
471 base::ResetAndReturn(&callback_).Run(capabilities_); | |
472 } | |
473 | |
474 void VideoCaptureImpl::DoOpenBitstreamOnCaptureThread( | |
475 media::EncodedVideoSource::Client* client, | |
476 const media::VideoEncodingParameters& params) { | |
477 DCHECK(capture_message_loop_proxy_->BelongsToCurrentThread()); | |
478 DCHECK(!client_); | |
479 client_ = client; | |
480 Send(new EncodedVideoCaptureHostMsg_OpenBitstream( | |
481 device_id_, current_params_.session_id, params)); | |
482 } | |
483 | |
484 void VideoCaptureImpl::DoCloseBitstreamOnCaptureThread() { | |
485 DCHECK(capture_message_loop_proxy_->BelongsToCurrentThread()); | |
486 Send(new EncodedVideoCaptureHostMsg_CloseBitstream(device_id_)); | |
487 client_ = NULL; | |
488 } | |
489 | |
490 void VideoCaptureImpl::DoNotifyBitstreamOpenedOnCaptureThread( | |
491 const media::VideoEncodingParameters& params, | |
492 const std::vector<base::SharedMemoryHandle>& buffers, | |
493 uint32 buffer_size) { | |
494 DCHECK(capture_message_loop_proxy_->BelongsToCurrentThread()); | |
495 DCHECK(bitstream_buffers_.empty()); | |
496 for (size_t i = 0; i < buffers.size(); ++i) { | |
497 base::SharedMemory* shm = new base::SharedMemory(buffers[i], true); | |
498 bitstream_buffers_.push_back(shm); | |
499 } | |
500 client_->OnOpened(params); | |
501 } | |
502 | |
503 void VideoCaptureImpl::DoNotifyBitstreamClosedOnCaptureThread() { | |
504 DCHECK(capture_message_loop_proxy_->BelongsToCurrentThread()); | |
505 for (size_t i = 0; i < bitstream_buffers_.size(); ++i) { | |
506 bitstream_buffers_[i]->Close(); | |
507 delete bitstream_buffers_[i]; | |
508 } | |
509 bitstream_buffers_.clear(); | |
510 if (client_) { | |
511 client_->OnClosed(); | |
512 client_ = NULL; | |
513 } | |
514 } | |
515 | |
516 void VideoCaptureImpl::DoNotifyBitstreamConfigChangedOnCaptureThread( | |
517 const media::RuntimeVideoEncodingParameters& params) { | |
518 DCHECK(capture_message_loop_proxy_->BelongsToCurrentThread()); | |
519 client_->OnConfigChanged(params); | |
520 } | |
521 | |
522 void VideoCaptureImpl::DoNotifyBitstreamBufferReadyOnCaptureThread( | |
523 int buffer_id, | |
524 uint32 size, | |
525 const media::BufferEncodingMetadata& metadata) { | |
526 DCHECK(capture_message_loop_proxy_->BelongsToCurrentThread()); | |
527 if (buffer_id >= 0 && buffer_id < (int)bitstream_buffers_.size()) { | |
Ami GONE FROM CHROMIUM
2013/06/13 22:56:58
s/(int)/static_cast<int>/ but that risks overflow;
hshi1
2013/06/13 23:33:40
Done.
| |
528 base::SharedMemory* shm = bitstream_buffers_[buffer_id]; | |
529 if (shm && shm->Map(size)) { | |
Ami GONE FROM CHROMIUM
2013/06/13 22:56:58
shm can't be NULL here.
Ami GONE FROM CHROMIUM
2013/06/13 22:56:58
You test shm->Map() for failing but you don't do a
hshi1
2013/06/13 23:33:40
Done.
hshi1
2013/06/13 23:33:40
Done.
| |
530 scoped_refptr<media::EncodedBitstreamBuffer> buffer = | |
531 new media::EncodedBitstreamBuffer( | |
532 buffer_id, (uint8*)shm->memory(), size, metadata); | |
533 client_->OnBufferReady(buffer); | |
534 shm->Unmap(); | |
Ami GONE FROM CHROMIUM
2013/06/13 22:56:58
Is it clear why you want to Map & Unmap on every b
hshi1
2013/06/13 23:33:40
Done. (Notice: no need to Unmap on close - bitstre
| |
535 } | |
536 } | |
537 } | |
538 | |
539 void VideoCaptureImpl::DoNotifyCapabilitiesAvailableOnCaptureThread( | |
540 const media::VideoEncodingCapabilities& capabilities) { | |
541 DCHECK(capture_message_loop_proxy_->BelongsToCurrentThread()); | |
542 capabilities_ = capabilities; | |
543 if (!callback_.is_null()) | |
544 base::ResetAndReturn(&callback_).Run(capabilities_); | |
545 } | |
546 | |
383 void VideoCaptureImpl::StopDevice() { | 547 void VideoCaptureImpl::StopDevice() { |
384 DCHECK(capture_message_loop_proxy_->BelongsToCurrentThread()); | 548 DCHECK(capture_message_loop_proxy_->BelongsToCurrentThread()); |
385 | 549 |
386 device_info_available_ = false; | 550 device_info_available_ = false; |
387 if (state_ == VIDEO_CAPTURE_STATE_STARTED) { | 551 if (state_ == VIDEO_CAPTURE_STATE_STARTED) { |
388 state_ = VIDEO_CAPTURE_STATE_STOPPING; | 552 state_ = VIDEO_CAPTURE_STATE_STOPPING; |
389 Send(new VideoCaptureHostMsg_Stop(device_id_)); | 553 Send(new VideoCaptureHostMsg_Stop(device_id_)); |
390 current_params_.width = current_params_.height = 0; | 554 current_params_.width = current_params_.height = 0; |
391 } | 555 } |
392 } | 556 } |
(...skipping 68 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
461 if (it != clients->end()) { | 625 if (it != clients->end()) { |
462 handler->OnStopped(this); | 626 handler->OnStopped(this); |
463 handler->OnRemoved(this); | 627 handler->OnRemoved(this); |
464 clients->erase(it); | 628 clients->erase(it); |
465 found = true; | 629 found = true; |
466 } | 630 } |
467 return found; | 631 return found; |
468 } | 632 } |
469 | 633 |
470 } // namespace content | 634 } // namespace content |
OLD | NEW |