OLD | NEW |
---|---|
1 /* | 1 /* |
2 * Copyright (c) 2014 The WebRTC project authors. All Rights Reserved. | 2 * Copyright (c) 2014 The WebRTC project authors. All Rights Reserved. |
3 * | 3 * |
4 * Use of this source code is governed by a BSD-style license | 4 * Use of this source code is governed by a BSD-style license |
5 * that can be found in the LICENSE file in the root of the source | 5 * that can be found in the LICENSE file in the root of the source |
6 * tree. An additional intellectual property rights grant can be found | 6 * tree. An additional intellectual property rights grant can be found |
7 * in the file PATENTS. All contributing project authors may | 7 * in the file PATENTS. All contributing project authors may |
8 * be found in the AUTHORS file in the root of the source tree. | 8 * be found in the AUTHORS file in the root of the source tree. |
9 */ | 9 */ |
10 | 10 |
(...skipping 21 matching lines...) Expand all Loading... | |
32 namespace { | 32 namespace { |
33 | 33 |
34 // Constants from dwmapi.h. | 34 // Constants from dwmapi.h. |
35 const UINT DWM_EC_DISABLECOMPOSITION = 0; | 35 const UINT DWM_EC_DISABLECOMPOSITION = 0; |
36 const UINT DWM_EC_ENABLECOMPOSITION = 1; | 36 const UINT DWM_EC_ENABLECOMPOSITION = 1; |
37 | 37 |
38 const wchar_t kDwmapiLibraryName[] = L"dwmapi.dll"; | 38 const wchar_t kDwmapiLibraryName[] = L"dwmapi.dll"; |
39 | 39 |
40 } // namespace | 40 } // namespace |
41 | 41 |
42 ScreenCapturerWinGdi::ScreenCapturerWinGdi(const DesktopCaptureOptions& options) | 42 ScreenCapturerWinGdi::ScreenCapturerWinGdi( |
43 : callback_(NULL), | 43 const DesktopCaptureOptions& options) { |
44 current_screen_id_(kFullDesktopScreenId), | |
45 desktop_dc_(NULL), | |
46 memory_dc_(NULL), | |
47 dwmapi_library_(NULL), | |
48 composition_func_(NULL), | |
49 set_thread_execution_state_failed_(false) { | |
50 if (options.disable_effects()) { | 44 if (options.disable_effects()) { |
51 // Load dwmapi.dll dynamically since it is not available on XP. | 45 // Load dwmapi.dll dynamically since it is not available on XP. |
52 if (!dwmapi_library_) | 46 if (!dwmapi_library_) |
53 dwmapi_library_ = LoadLibrary(kDwmapiLibraryName); | 47 dwmapi_library_ = LoadLibrary(kDwmapiLibraryName); |
54 | 48 |
55 if (dwmapi_library_) { | 49 if (dwmapi_library_) { |
56 composition_func_ = reinterpret_cast<DwmEnableCompositionFunc>( | 50 composition_func_ = reinterpret_cast<DwmEnableCompositionFunc>( |
57 GetProcAddress(dwmapi_library_, "DwmEnableComposition")); | 51 GetProcAddress(dwmapi_library_, "DwmEnableComposition")); |
58 } | 52 } |
59 } | 53 } |
60 } | 54 } |
61 | 55 |
62 ScreenCapturerWinGdi::~ScreenCapturerWinGdi() { | 56 ScreenCapturerWinGdi::~ScreenCapturerWinGdi() { |
63 if (desktop_dc_) | 57 if (desktop_dc_) |
64 ReleaseDC(NULL, desktop_dc_); | 58 ReleaseDC(nullptr, desktop_dc_); |
Wez
2016/05/18 01:29:50
As per header comment, technically this should rem
Sergey Ulanov
2016/05/31 12:02:49
Done.
| |
65 if (memory_dc_) | 59 if (memory_dc_) |
66 DeleteDC(memory_dc_); | 60 DeleteDC(memory_dc_); |
67 | 61 |
68 // Restore Aero. | 62 // Restore Aero. |
69 if (composition_func_) | 63 if (composition_func_) |
70 (*composition_func_)(DWM_EC_ENABLECOMPOSITION); | 64 (*composition_func_)(DWM_EC_ENABLECOMPOSITION); |
71 | 65 |
72 if (dwmapi_library_) | 66 if (dwmapi_library_) |
73 FreeLibrary(dwmapi_library_); | 67 FreeLibrary(dwmapi_library_); |
74 } | 68 } |
(...skipping 15 matching lines...) Expand all Loading... | |
90 set_thread_execution_state_failed_ = true; | 84 set_thread_execution_state_failed_ = true; |
91 LOG_F(LS_WARNING) << "Failed to make system & display power assertion: " | 85 LOG_F(LS_WARNING) << "Failed to make system & display power assertion: " |
92 << GetLastError(); | 86 << GetLastError(); |
93 } | 87 } |
94 } | 88 } |
95 | 89 |
96 // Make sure the GDI capture resources are up-to-date. | 90 // Make sure the GDI capture resources are up-to-date. |
97 PrepareCaptureResources(); | 91 PrepareCaptureResources(); |
98 | 92 |
99 if (!CaptureImage()) { | 93 if (!CaptureImage()) { |
100 callback_->OnCaptureCompleted(NULL); | 94 callback_->OnCaptureCompleted(std::unique_ptr<DesktopFrame>()); |
101 return; | 95 return; |
102 } | 96 } |
103 | 97 |
104 const DesktopFrame* current_frame = queue_.current_frame(); | 98 const DesktopFrame* current_frame = queue_.current_frame(); |
105 const DesktopFrame* last_frame = queue_.previous_frame(); | 99 const DesktopFrame* last_frame = queue_.previous_frame(); |
106 if (last_frame && last_frame->size().equals(current_frame->size())) { | 100 if (last_frame && last_frame->size().equals(current_frame->size())) { |
107 // Make sure the differencer is set up correctly for these previous and | 101 // Make sure the differencer is set up correctly for these previous and |
108 // current screens. | 102 // current screens. |
109 if (!differ_.get() || | 103 if (!differ_.get() || |
110 (differ_->width() != current_frame->size().width()) || | 104 (differ_->width() != current_frame->size().width()) || |
(...skipping 52 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
163 // will restore Aero automatically if the process exits. This has no effect | 157 // will restore Aero automatically if the process exits. This has no effect |
164 // under Windows 8 or higher. See crbug.com/124018. | 158 // under Windows 8 or higher. See crbug.com/124018. |
165 if (composition_func_) | 159 if (composition_func_) |
166 (*composition_func_)(DWM_EC_DISABLECOMPOSITION); | 160 (*composition_func_)(DWM_EC_DISABLECOMPOSITION); |
167 } | 161 } |
168 | 162 |
169 void ScreenCapturerWinGdi::PrepareCaptureResources() { | 163 void ScreenCapturerWinGdi::PrepareCaptureResources() { |
170 // Switch to the desktop receiving user input if different from the current | 164 // Switch to the desktop receiving user input if different from the current |
171 // one. | 165 // one. |
172 std::unique_ptr<Desktop> input_desktop(Desktop::GetInputDesktop()); | 166 std::unique_ptr<Desktop> input_desktop(Desktop::GetInputDesktop()); |
173 if (input_desktop.get() != NULL && !desktop_.IsSame(*input_desktop)) { | 167 if (input_desktop && !desktop_.IsSame(*input_desktop)) { |
174 // Release GDI resources otherwise SetThreadDesktop will fail. | 168 // Release GDI resources otherwise SetThreadDesktop will fail. |
175 if (desktop_dc_) { | 169 if (desktop_dc_) { |
176 ReleaseDC(NULL, desktop_dc_); | 170 ReleaseDC(nullptr, desktop_dc_); |
177 desktop_dc_ = NULL; | 171 desktop_dc_ = nullptr; |
178 } | 172 } |
179 | 173 |
180 if (memory_dc_) { | 174 if (memory_dc_) { |
181 DeleteDC(memory_dc_); | 175 DeleteDC(memory_dc_); |
182 memory_dc_ = NULL; | 176 memory_dc_ = nullptr; |
183 } | 177 } |
184 | 178 |
185 // If SetThreadDesktop() fails, the thread is still assigned a desktop. | 179 // If SetThreadDesktop() fails, the thread is still assigned a desktop. |
186 // So we can continue capture screen bits, just from the wrong desktop. | 180 // So we can continue capture screen bits, just from the wrong desktop. |
187 desktop_.SetThreadDesktop(input_desktop.release()); | 181 desktop_.SetThreadDesktop(input_desktop.release()); |
188 | 182 |
189 // Re-assert our vote to disable Aero. | 183 // Re-assert our vote to disable Aero. |
190 // See crbug.com/124018 and crbug.com/129906. | 184 // See crbug.com/124018 and crbug.com/129906. |
191 if (composition_func_ != NULL) { | 185 if (composition_func_) { |
192 (*composition_func_)(DWM_EC_DISABLECOMPOSITION); | 186 (*composition_func_)(DWM_EC_DISABLECOMPOSITION); |
193 } | 187 } |
194 } | 188 } |
195 | 189 |
196 // If the display bounds have changed then recreate GDI resources. | 190 // If the display bounds have changed then recreate GDI resources. |
197 // TODO(wez): Also check for pixel format changes. | 191 // TODO(wez): Also check for pixel format changes. |
198 DesktopRect screen_rect(DesktopRect::MakeXYWH( | 192 DesktopRect screen_rect(DesktopRect::MakeXYWH( |
199 GetSystemMetrics(SM_XVIRTUALSCREEN), | 193 GetSystemMetrics(SM_XVIRTUALSCREEN), |
200 GetSystemMetrics(SM_YVIRTUALSCREEN), | 194 GetSystemMetrics(SM_YVIRTUALSCREEN), |
201 GetSystemMetrics(SM_CXVIRTUALSCREEN), | 195 GetSystemMetrics(SM_CXVIRTUALSCREEN), |
202 GetSystemMetrics(SM_CYVIRTUALSCREEN))); | 196 GetSystemMetrics(SM_CYVIRTUALSCREEN))); |
203 if (!screen_rect.equals(desktop_dc_rect_)) { | 197 if (!screen_rect.equals(desktop_dc_rect_)) { |
204 if (desktop_dc_) { | 198 if (desktop_dc_) { |
205 ReleaseDC(NULL, desktop_dc_); | 199 ReleaseDC(nullptr, desktop_dc_); |
206 desktop_dc_ = NULL; | 200 desktop_dc_ = nullptr; |
207 } | 201 } |
208 if (memory_dc_) { | 202 if (memory_dc_) { |
209 DeleteDC(memory_dc_); | 203 DeleteDC(memory_dc_); |
210 memory_dc_ = NULL; | 204 memory_dc_ = nullptr; |
211 } | 205 } |
212 desktop_dc_rect_ = DesktopRect(); | 206 desktop_dc_rect_ = DesktopRect(); |
213 } | 207 } |
214 | 208 |
215 if (desktop_dc_ == NULL) { | 209 if (!desktop_dc_) { |
216 assert(memory_dc_ == NULL); | 210 assert(!memory_dc_); |
217 | 211 |
218 // Create GDI device contexts to capture from the desktop into memory. | 212 // Create GDI device contexts to capture from the desktop into memory. |
219 desktop_dc_ = GetDC(NULL); | 213 desktop_dc_ = GetDC(nullptr); |
220 if (!desktop_dc_) | 214 if (!desktop_dc_) |
221 abort(); | 215 abort(); |
Wez
2016/05/18 01:29:50
How is if (!foo) abort() different from assert(foo
Sergey Ulanov
2016/05/31 12:02:49
assert() is enabled only in debug builds.
capture
Wez
2016/06/01 21:29:05
Acknowledged.
| |
222 memory_dc_ = CreateCompatibleDC(desktop_dc_); | 216 memory_dc_ = CreateCompatibleDC(desktop_dc_); |
223 if (!memory_dc_) | 217 if (!memory_dc_) |
224 abort(); | 218 abort(); |
225 | 219 |
226 desktop_dc_rect_ = screen_rect; | 220 desktop_dc_rect_ = screen_rect; |
227 | 221 |
228 // Make sure the frame buffers will be reallocated. | 222 // Make sure the frame buffers will be reallocated. |
229 queue_.Reset(); | 223 queue_.Reset(); |
230 | 224 |
231 helper_.ClearInvalidRegion(); | 225 helper_.ClearInvalidRegion(); |
232 } | 226 } |
233 } | 227 } |
234 | 228 |
235 bool ScreenCapturerWinGdi::CaptureImage() { | 229 bool ScreenCapturerWinGdi::CaptureImage() { |
236 DesktopRect screen_rect = | 230 DesktopRect screen_rect = |
237 GetScreenRect(current_screen_id_, current_device_key_); | 231 GetScreenRect(current_screen_id_, current_device_key_); |
238 if (screen_rect.is_empty()) | 232 if (screen_rect.is_empty()) |
239 return false; | 233 return false; |
240 | 234 |
241 DesktopSize size = screen_rect.size(); | 235 DesktopSize size = screen_rect.size(); |
242 // If the current buffer is from an older generation then allocate a new one. | 236 // If the current buffer is from an older generation then allocate a new one. |
243 // Note that we can't reallocate other buffers at this point, since the caller | 237 // Note that we can't reallocate other buffers at this point, since the caller |
244 // may still be reading from them. | 238 // may still be reading from them. |
245 if (!queue_.current_frame() || | 239 if (!queue_.current_frame() || |
246 !queue_.current_frame()->size().equals(screen_rect.size())) { | 240 !queue_.current_frame()->size().equals(screen_rect.size())) { |
247 assert(desktop_dc_ != NULL); | 241 assert(desktop_dc_); |
248 assert(memory_dc_ != NULL); | 242 assert(memory_dc_); |
Wez
2016/05/18 01:29:50
Why assert here, given that we already check that
Sergey Ulanov
2016/05/31 12:02:49
It's not checked in this function, so I think this
Wez
2016/06/01 21:29:05
Acknowledged.
| |
249 | 243 |
250 std::unique_ptr<DesktopFrame> buffer(DesktopFrameWin::Create( | 244 std::unique_ptr<DesktopFrame> buffer = DesktopFrameWin::Create( |
251 size, shared_memory_factory_.get(), desktop_dc_)); | 245 size, shared_memory_factory_.get(), desktop_dc_); |
252 if (!buffer) | 246 if (!buffer) |
253 return false; | 247 return false; |
254 queue_.ReplaceCurrentFrame(SharedDesktopFrame::Wrap(buffer.release())); | 248 queue_.ReplaceCurrentFrame(SharedDesktopFrame::Wrap(std::move(buffer))); |
255 } | 249 } |
256 | 250 |
257 // Select the target bitmap into the memory dc and copy the rect from desktop | 251 // Select the target bitmap into the memory dc and copy the rect from desktop |
258 // to memory. | 252 // to memory. |
259 DesktopFrameWin* current = static_cast<DesktopFrameWin*>( | 253 DesktopFrameWin* current = static_cast<DesktopFrameWin*>( |
260 queue_.current_frame()->GetUnderlyingFrame()); | 254 queue_.current_frame()->GetUnderlyingFrame()); |
261 HGDIOBJ previous_object = SelectObject(memory_dc_, current->bitmap()); | 255 HGDIOBJ previous_object = SelectObject(memory_dc_, current->bitmap()); |
262 if (previous_object != NULL) { | 256 if (previous_object) { |
263 BitBlt(memory_dc_, | 257 BitBlt(memory_dc_, 0, 0, screen_rect.width(), screen_rect.height(), |
264 0, 0, screen_rect.width(), screen_rect.height(), | 258 desktop_dc_, screen_rect.left(), screen_rect.top(), |
265 desktop_dc_, | |
266 screen_rect.left(), screen_rect.top(), | |
267 SRCCOPY | CAPTUREBLT); | 259 SRCCOPY | CAPTUREBLT); |
268 | 260 |
269 // Select back the previously selected object to that the device contect | 261 // Select back the previously selected object to that the device contect |
270 // could be destroyed independently of the bitmap if needed. | 262 // could be destroyed independently of the bitmap if needed. |
271 SelectObject(memory_dc_, previous_object); | 263 SelectObject(memory_dc_, previous_object); |
272 } | 264 } |
273 return true; | 265 return true; |
274 } | 266 } |
275 | 267 |
276 } // namespace webrtc | 268 } // namespace webrtc |
OLD | NEW |