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 // | |
5 // Implementation notes about interactions with VideoCaptureImpl. | |
6 // | |
7 // How is VideoCaptureImpl used: | |
8 // | |
9 // VideoCaptureImpl is an IO thread object while VideoCaptureImplManager | |
10 // lives only on the render thread. It is only possible to access an | |
11 // object of VideoCaptureImpl via a task on the IO thread. | |
12 // | |
13 // How is VideoCaptureImpl deleted: | |
14 // | |
15 // A task is posted to the IO thread to delete a VideoCaptureImpl. | |
16 // Immediately after that the pointer to it is dropped. This means no | |
17 // access to this VideoCaptureImpl object is possible on the render | |
18 // thread. Also note that VideoCaptureImpl does not post task to itself. | |
19 // | |
20 // The use of Unretained: | |
21 // | |
22 // We make sure deletion is the last task on the IO thread for a | |
23 // VideoCaptureImpl object. This allows the use of Unretained() binding. | |
24 | 4 |
25 #include "content/renderer/media/video_capture_impl_manager.h" | 5 #include "content/renderer/media/video_capture_impl_manager.h" |
26 | 6 |
27 #include "base/bind.h" | 7 #include "base/bind.h" |
28 #include "base/bind_helpers.h" | 8 #include "base/bind_helpers.h" |
29 #include "content/child/child_process.h" | 9 #include "content/public/renderer/render_thread.h" |
30 #include "content/renderer/media/video_capture_impl.h" | 10 #include "content/renderer/media/video_capture_impl.h" |
31 #include "content/renderer/media/video_capture_message_filter.h" | 11 #include "content/renderer/media/video_capture_message_filter.h" |
32 #include "media/base/bind_to_current_loop.h" | 12 #include "media/base/bind_to_current_loop.h" |
33 | 13 |
34 namespace content { | 14 namespace content { |
35 | 15 |
| 16 VideoCaptureHandle::VideoCaptureHandle( |
| 17 media::VideoCapture* impl, base::Closure destruction_cb) |
| 18 : impl_(impl), destruction_cb_(destruction_cb) { |
| 19 } |
| 20 |
| 21 VideoCaptureHandle::~VideoCaptureHandle() { |
| 22 destruction_cb_.Run(); |
| 23 } |
| 24 |
| 25 void VideoCaptureHandle::StartCapture( |
| 26 EventHandler* handler, |
| 27 const media::VideoCaptureParams& params) { |
| 28 impl_->StartCapture(handler, params); |
| 29 } |
| 30 |
| 31 void VideoCaptureHandle::StopCapture(EventHandler* handler) { |
| 32 impl_->StopCapture(handler); |
| 33 } |
| 34 |
| 35 bool VideoCaptureHandle::CaptureStarted() { |
| 36 return impl_->CaptureStarted(); |
| 37 } |
| 38 |
| 39 int VideoCaptureHandle::CaptureFrameRate() { |
| 40 return impl_->CaptureFrameRate(); |
| 41 } |
| 42 |
| 43 void VideoCaptureHandle::GetDeviceSupportedFormats( |
| 44 const DeviceFormatsCallback& callback) { |
| 45 impl_->GetDeviceSupportedFormats(callback); |
| 46 } |
| 47 |
| 48 void VideoCaptureHandle::GetDeviceFormatsInUse( |
| 49 const DeviceFormatsInUseCallback& callback) { |
| 50 impl_->GetDeviceFormatsInUse(callback); |
| 51 } |
| 52 |
36 VideoCaptureImplManager::VideoCaptureImplManager() | 53 VideoCaptureImplManager::VideoCaptureImplManager() |
37 : next_client_id_(0), | 54 : filter_(new VideoCaptureMessageFilter()), |
38 filter_(new VideoCaptureMessageFilter()), | |
39 weak_factory_(this) { | 55 weak_factory_(this) { |
40 } | 56 } |
41 | 57 |
42 VideoCaptureImplManager::~VideoCaptureImplManager() { | 58 VideoCaptureImplManager::~VideoCaptureImplManager() { |
43 DCHECK(thread_checker_.CalledOnValidThread()); | 59 DCHECK(thread_checker_.CalledOnValidThread()); |
44 if (devices_.empty()) | |
45 return; | |
46 // Forcibly release all video capture resources. | |
47 for (VideoCaptureDeviceMap::iterator it = devices_.begin(); | |
48 it != devices_.end(); ++it) { | |
49 VideoCaptureImpl* impl = it->second.second; | |
50 ChildProcess::current()->io_message_loop_proxy()->PostTask( | |
51 FROM_HERE, | |
52 base::Bind(&VideoCaptureImpl::DeInit, | |
53 base::Unretained(impl))); | |
54 ChildProcess::current()->io_message_loop_proxy()->PostTask( | |
55 FROM_HERE, | |
56 base::Bind(&base::DeletePointer<VideoCaptureImpl>, | |
57 base::Unretained(impl))); | |
58 } | |
59 devices_.clear(); | |
60 } | 60 } |
61 | 61 |
62 base::Closure VideoCaptureImplManager::UseDevice( | 62 scoped_ptr<VideoCaptureHandle> VideoCaptureImplManager::UseDevice( |
63 media::VideoCaptureSessionId id) { | 63 media::VideoCaptureSessionId id) { |
64 DCHECK(thread_checker_.CalledOnValidThread()); | 64 DCHECK(thread_checker_.CalledOnValidThread()); |
65 | 65 |
66 VideoCaptureImpl* impl = NULL; | 66 VideoCaptureImpl* video_capture_device = NULL; |
67 VideoCaptureDeviceMap::iterator it = devices_.find(id); | 67 VideoCaptureDeviceMap::iterator it = devices_.find(id); |
68 if (it == devices_.end()) { | 68 if (it == devices_.end()) { |
69 impl = CreateVideoCaptureImplForTesting(id, filter_.get()); | 69 video_capture_device = CreateVideoCaptureImpl(id, filter_.get()); |
70 if (!impl) | 70 devices_[id] = |
71 impl = new VideoCaptureImpl(id, filter_.get()); | 71 std::make_pair(1, linked_ptr<VideoCaptureImpl>(video_capture_device)); |
72 devices_[id] = std::make_pair(1, impl); | 72 video_capture_device->Init(); |
73 ChildProcess::current()->io_message_loop_proxy()->PostTask( | |
74 FROM_HERE, | |
75 base::Bind(&VideoCaptureImpl::Init, | |
76 base::Unretained(impl))); | |
77 } else { | 73 } else { |
78 ++it->second.first; | 74 ++it->second.first; |
| 75 video_capture_device = it->second.second.get(); |
79 } | 76 } |
80 return base::Bind(&VideoCaptureImplManager::UnrefDevice, | 77 |
81 weak_factory_.GetWeakPtr(), id); | 78 // This callback ensures UnrefDevice() happens on the render thread. |
| 79 return scoped_ptr<VideoCaptureHandle>( |
| 80 new VideoCaptureHandle( |
| 81 video_capture_device, |
| 82 media::BindToCurrentLoop( |
| 83 base::Bind( |
| 84 &VideoCaptureImplManager::UnrefDevice, |
| 85 weak_factory_.GetWeakPtr(), |
| 86 id)))); |
82 } | 87 } |
83 | 88 |
84 base::Closure VideoCaptureImplManager::StartCapture( | 89 VideoCaptureImpl* VideoCaptureImplManager::CreateVideoCaptureImpl( |
85 media::VideoCaptureSessionId id, | |
86 const media::VideoCaptureParams& params, | |
87 const VideoCaptureStateUpdateCB& state_update_cb, | |
88 const VideoCaptureDeliverFrameCB& deliver_frame_cb) { | |
89 DCHECK(thread_checker_.CalledOnValidThread()); | |
90 VideoCaptureDeviceMap::iterator it = devices_.find(id); | |
91 DCHECK(it != devices_.end()); | |
92 VideoCaptureImpl* impl = it->second.second; | |
93 | |
94 // This ID is used to identify a client of VideoCaptureImpl. | |
95 const int client_id = ++next_client_id_; | |
96 | |
97 ChildProcess::current()->io_message_loop_proxy()->PostTask( | |
98 FROM_HERE, | |
99 base::Bind(&VideoCaptureImpl::StartCapture, | |
100 base::Unretained(impl), | |
101 client_id, | |
102 params, | |
103 state_update_cb, | |
104 deliver_frame_cb)); | |
105 return base::Bind(&VideoCaptureImplManager::StopCapture, | |
106 weak_factory_.GetWeakPtr(), | |
107 client_id, id); | |
108 } | |
109 | |
110 void VideoCaptureImplManager::GetDeviceSupportedFormats( | |
111 media::VideoCaptureSessionId id, | |
112 const VideoCaptureDeviceFormatsCB& callback) { | |
113 DCHECK(thread_checker_.CalledOnValidThread()); | |
114 VideoCaptureDeviceMap::iterator it = devices_.find(id); | |
115 DCHECK(it != devices_.end()); | |
116 VideoCaptureImpl* impl = it->second.second; | |
117 ChildProcess::current()->io_message_loop_proxy()->PostTask( | |
118 FROM_HERE, | |
119 base::Bind(&VideoCaptureImpl::GetDeviceSupportedFormats, | |
120 base::Unretained(impl), callback)); | |
121 } | |
122 | |
123 void VideoCaptureImplManager::GetDeviceFormatsInUse( | |
124 media::VideoCaptureSessionId id, | |
125 const VideoCaptureDeviceFormatsCB& callback) { | |
126 DCHECK(thread_checker_.CalledOnValidThread()); | |
127 VideoCaptureDeviceMap::iterator it = devices_.find(id); | |
128 DCHECK(it != devices_.end()); | |
129 VideoCaptureImpl* impl = it->second.second; | |
130 ChildProcess::current()->io_message_loop_proxy()->PostTask( | |
131 FROM_HERE, | |
132 base::Bind(&VideoCaptureImpl::GetDeviceFormatsInUse, | |
133 base::Unretained(impl), callback)); | |
134 } | |
135 | |
136 VideoCaptureImpl* | |
137 VideoCaptureImplManager::CreateVideoCaptureImplForTesting( | |
138 media::VideoCaptureSessionId id, | 90 media::VideoCaptureSessionId id, |
139 VideoCaptureMessageFilter* filter) const { | 91 VideoCaptureMessageFilter* filter) const { |
140 return NULL; | 92 return new VideoCaptureImpl(id, filter); |
141 } | |
142 | |
143 void VideoCaptureImplManager::StopCapture( | |
144 int client_id, media::VideoCaptureSessionId id) { | |
145 DCHECK(thread_checker_.CalledOnValidThread()); | |
146 VideoCaptureDeviceMap::iterator it = devices_.find(id); | |
147 DCHECK(it != devices_.end()); | |
148 VideoCaptureImpl* impl = it->second.second; | |
149 ChildProcess::current()->io_message_loop_proxy()->PostTask( | |
150 FROM_HERE, | |
151 base::Bind(&VideoCaptureImpl::StopCapture, | |
152 base::Unretained(impl), client_id)); | |
153 } | 93 } |
154 | 94 |
155 void VideoCaptureImplManager::UnrefDevice( | 95 void VideoCaptureImplManager::UnrefDevice( |
156 media::VideoCaptureSessionId id) { | 96 media::VideoCaptureSessionId id) { |
157 DCHECK(thread_checker_.CalledOnValidThread()); | 97 DCHECK(thread_checker_.CalledOnValidThread()); |
158 VideoCaptureDeviceMap::iterator it = devices_.find(id); | 98 VideoCaptureDeviceMap::iterator it = devices_.find(id); |
159 DCHECK(it != devices_.end()); | 99 DCHECK(it != devices_.end()); |
160 VideoCaptureImpl* impl = it->second.second; | |
161 | 100 |
162 // Unref and destroy on the IO thread if there's no more client. | |
163 DCHECK(it->second.first); | 101 DCHECK(it->second.first); |
164 --it->second.first; | 102 --it->second.first; |
165 if (!it->second.first) { | 103 if (!it->second.first) { |
| 104 VideoCaptureImpl* impl = it->second.second.release(); |
166 devices_.erase(id); | 105 devices_.erase(id); |
167 ChildProcess::current()->io_message_loop_proxy()->PostTask( | 106 impl->DeInit(base::Bind(&base::DeletePointer<VideoCaptureImpl>, impl)); |
168 FROM_HERE, | |
169 base::Bind(&VideoCaptureImpl::DeInit, | |
170 base::Unretained(impl))); | |
171 ChildProcess::current()->io_message_loop_proxy()->PostTask( | |
172 FROM_HERE, | |
173 base::Bind(&base::DeletePointer<VideoCaptureImpl>, | |
174 base::Unretained(impl))); | |
175 } | 107 } |
176 } | 108 } |
177 | 109 |
178 void VideoCaptureImplManager::SuspendDevices(bool suspend) { | 110 void VideoCaptureImplManager::SuspendDevices(bool suspend) { |
179 DCHECK(thread_checker_.CalledOnValidThread()); | 111 DCHECK(thread_checker_.CalledOnValidThread()); |
180 for (VideoCaptureDeviceMap::iterator it = devices_.begin(); | 112 for (VideoCaptureDeviceMap::iterator it = devices_.begin(); |
181 it != devices_.end(); ++it) { | 113 it != devices_.end(); ++it) |
182 VideoCaptureImpl* impl = it->second.second; | 114 it->second.second->SuspendCapture(suspend); |
183 ChildProcess::current()->io_message_loop_proxy()->PostTask( | |
184 FROM_HERE, | |
185 base::Bind(&VideoCaptureImpl::SuspendCapture, | |
186 base::Unretained(impl), suspend)); | |
187 } | |
188 } | 115 } |
189 | 116 |
190 } // namespace content | 117 } // namespace content |
OLD | NEW |