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

Side by Side Diff: webrtc/modules/desktop_capture/win/screen_capturer_win_directx.cc

Issue 2682913002: [DesktopCapture] Detect screen resolution changes in DirectX capturer (Closed)
Patch Set: Resolve review comments Created 3 years, 10 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
« no previous file with comments | « webrtc/modules/desktop_capture/win/screen_capturer_win_directx.h ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 /* 1 /*
2 * Copyright (c) 2016 The WebRTC project authors. All Rights Reserved. 2 * Copyright (c) 2016 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
11 #include "webrtc/modules/desktop_capture/win/screen_capturer_win_directx.h" 11 #include "webrtc/modules/desktop_capture/win/screen_capturer_win_directx.h"
12 12
13 #include <string>
13 #include <utility> 14 #include <utility>
14 15
15 #include "webrtc/base/checks.h" 16 #include "webrtc/base/checks.h"
16 #include "webrtc/base/logging.h" 17 #include "webrtc/base/logging.h"
17 #include "webrtc/base/timeutils.h" 18 #include "webrtc/base/timeutils.h"
18 #include "webrtc/modules/desktop_capture/desktop_frame.h" 19 #include "webrtc/modules/desktop_capture/desktop_frame.h"
20 #include "webrtc/modules/desktop_capture/win/screen_capture_utils.h"
19 21
20 namespace webrtc { 22 namespace webrtc {
21 23
22 using Microsoft::WRL::ComPtr; 24 using Microsoft::WRL::ComPtr;
23 25
24 // static 26 // static
25 bool ScreenCapturerWinDirectx::IsSupported() { 27 bool ScreenCapturerWinDirectx::IsSupported() {
26 // Forwards IsSupported() function call to DxgiDuplicatorController. 28 // Forwards IsSupported() function call to DxgiDuplicatorController.
27 return DxgiDuplicatorController::Instance()->IsSupported(); 29 return DxgiDuplicatorController::Instance()->IsSupported();
28 } 30 }
(...skipping 30 matching lines...) Expand all
59 return DxgiDuplicatorController::Instance() 61 return DxgiDuplicatorController::Instance()
60 ->ScreenRect(current_screen_id_) 62 ->ScreenRect(current_screen_id_)
61 .size(); 63 .size();
62 } 64 }
63 65
64 void ScreenCapturerWinDirectx::CaptureFrame() { 66 void ScreenCapturerWinDirectx::CaptureFrame() {
65 RTC_DCHECK(callback_); 67 RTC_DCHECK(callback_);
66 68
67 int64_t capture_start_time_nanos = rtc::TimeNanos(); 69 int64_t capture_start_time_nanos = rtc::TimeNanos();
68 70
71 // The dxgi components and APIs do not update the screen resolution without
72 // a reinitialization. So we use the GetDC() function to retrieve the screen
73 // resolution to decide whether dxgi components need to be reinitialized.
74 // If the screen resolution changed, it's very likely the next Duplicate()
75 // function call will fail because of a missing monitor or the frame size is
76 // not enough to store the output. So we reinitialize dxgi components in-place
77 // to avoid a capture failure.
78 // But there is no guarantee GetDC() function returns the same resolution as
79 // dxgi APIs, we still rely on dxgi components to return the output frame
80 // size.
81 // TODO(zijiehe): Confirm whether IDXGIOutput::GetDesc() and
82 // IDXGIOutputDuplication::GetDesc() can detect the resolution change without
83 // reinitialization.
84 if (resolution_change_detector_.IsChanged(
85 GetScreenRect(kFullDesktopScreenId, std::wstring()).size())) {
86 frames_.Reset();
87 DxgiDuplicatorController::Instance()->Reset();
88 resolution_change_detector_.Reset();
89 }
90
69 frames_.MoveToNextFrame(); 91 frames_.MoveToNextFrame();
70 if (!frames_.current_frame()) { 92 if (!frames_.current_frame()) {
71 std::unique_ptr<DesktopFrame> new_frame; 93 std::unique_ptr<DesktopFrame> new_frame;
72 if (shared_memory_factory_) { 94 if (shared_memory_factory_) {
73 new_frame = SharedMemoryDesktopFrame::Create( 95 new_frame = SharedMemoryDesktopFrame::Create(
74 SelectedDesktopSize(), shared_memory_factory_.get()); 96 SelectedDesktopSize(), shared_memory_factory_.get());
75 } else { 97 } else {
76 new_frame.reset(new BasicDesktopFrame(SelectedDesktopSize())); 98 new_frame.reset(new BasicDesktopFrame(SelectedDesktopSize()));
77 } 99 }
78 if (!new_frame) { 100 if (!new_frame) {
79 LOG(LS_ERROR) << "Failed to allocate a new DesktopFrame."; 101 LOG(LS_ERROR) << "Failed to allocate a new DesktopFrame.";
80 // This usually means we do not have enough memory or SharedMemoryFactory 102 // This usually means we do not have enough memory or SharedMemoryFactory
81 // cannot work correctly. 103 // cannot work correctly.
82 callback_->OnCaptureResult(Result::ERROR_PERMANENT, nullptr); 104 callback_->OnCaptureResult(Result::ERROR_PERMANENT, nullptr);
83 return; 105 return;
84 } 106 }
85 frames_.ReplaceCurrentFrame(SharedDesktopFrame::Wrap(std::move(new_frame))); 107 frames_.ReplaceCurrentFrame(SharedDesktopFrame::Wrap(std::move(new_frame)));
86 } 108 }
87 109
88 if (current_screen_id_ == kFullDesktopScreenId) { 110 if (current_screen_id_ == kFullDesktopScreenId) {
89 if (!DxgiDuplicatorController::Instance()->Duplicate( 111 if (!DxgiDuplicatorController::Instance()->Duplicate(
90 &context_, frames_.current_frame())) { 112 &context_, frames_.current_frame())) {
91 // Screen size may be changed, so we need to reset the frames. 113 // Screen size may be changed, so we need to reset the frames.
92 frames_.Reset(); 114 frames_.Reset();
115 resolution_change_detector_.Reset();
93 callback_->OnCaptureResult(Result::ERROR_TEMPORARY, nullptr); 116 callback_->OnCaptureResult(Result::ERROR_TEMPORARY, nullptr);
94 return; 117 return;
95 } 118 }
96 } else { 119 } else {
97 if (!DxgiDuplicatorController::Instance()->DuplicateMonitor( 120 if (!DxgiDuplicatorController::Instance()->DuplicateMonitor(
98 &context_, current_screen_id_, frames_.current_frame())) { 121 &context_, current_screen_id_, frames_.current_frame())) {
99 // Screen size may be changed, so we need to reset the frames. 122 // Screen size may be changed, so we need to reset the frames.
100 frames_.Reset(); 123 frames_.Reset();
124 resolution_change_detector_.Reset();
101 if (current_screen_id_ >= 125 if (current_screen_id_ >=
102 DxgiDuplicatorController::Instance()->ScreenCount()) { 126 DxgiDuplicatorController::Instance()->ScreenCount()) {
103 // Current monitor has been removed from the system. 127 // Current monitor has been removed from the system.
104 callback_->OnCaptureResult(Result::ERROR_PERMANENT, nullptr); 128 callback_->OnCaptureResult(Result::ERROR_PERMANENT, nullptr);
105 } else { 129 } else {
106 callback_->OnCaptureResult(Result::ERROR_TEMPORARY, nullptr); 130 callback_->OnCaptureResult(Result::ERROR_TEMPORARY, nullptr);
107 } 131 }
108 return; 132 return;
109 } 133 }
110 } 134 }
(...skipping 15 matching lines...) Expand all
126 150
127 bool ScreenCapturerWinDirectx::SelectSource(SourceId id) { 151 bool ScreenCapturerWinDirectx::SelectSource(SourceId id) {
128 if (id == current_screen_id_) { 152 if (id == current_screen_id_) {
129 return true; 153 return true;
130 } 154 }
131 155
132 // Changing target screen may or may not impact frame size. So resetting 156 // Changing target screen may or may not impact frame size. So resetting
133 // frames only when a Duplicate() function call returns false. 157 // frames only when a Duplicate() function call returns false.
134 if (id == kFullDesktopScreenId) { 158 if (id == kFullDesktopScreenId) {
135 current_screen_id_ = id; 159 current_screen_id_ = id;
160 context_.Reset();
136 return true; 161 return true;
137 } 162 }
138 163
139 int screen_count = DxgiDuplicatorController::Instance()->ScreenCount(); 164 int screen_count = DxgiDuplicatorController::Instance()->ScreenCount();
140 if (id >= 0 && id < screen_count) { 165 if (id >= 0 && id < screen_count) {
141 current_screen_id_ = id; 166 current_screen_id_ = id;
167 context_.Reset();
142 return true; 168 return true;
143 } 169 }
144 return false; 170 return false;
145 } 171 }
146 172
147 } // namespace webrtc 173 } // namespace webrtc
OLDNEW
« no previous file with comments | « webrtc/modules/desktop_capture/win/screen_capturer_win_directx.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698