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

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: Avoid a CaptureFrame failure in first place 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
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 <utility> 13 #include <utility>
14 14
15 #include "webrtc/base/checks.h" 15 #include "webrtc/base/checks.h"
16 #include "webrtc/base/logging.h" 16 #include "webrtc/base/logging.h"
17 #include "webrtc/base/timeutils.h" 17 #include "webrtc/base/timeutils.h"
18 #include "webrtc/modules/desktop_capture/desktop_frame.h" 18 #include "webrtc/modules/desktop_capture/desktop_frame.h"
19 #include "webrtc/modules/desktop_capture/win/screen_capture_utils.h"
19 20
20 namespace webrtc { 21 namespace webrtc {
21 22
22 using Microsoft::WRL::ComPtr; 23 using Microsoft::WRL::ComPtr;
23 24
24 // static 25 // static
25 bool ScreenCapturerWinDirectx::IsSupported() { 26 bool ScreenCapturerWinDirectx::IsSupported() {
26 // Forwards IsSupported() function call to DxgiDuplicatorController. 27 // Forwards IsSupported() function call to DxgiDuplicatorController.
27 return DxgiDuplicatorController::Instance()->IsSupported(); 28 return DxgiDuplicatorController::Instance()->IsSupported();
28 } 29 }
(...skipping 30 matching lines...) Expand all
59 return DxgiDuplicatorController::Instance() 60 return DxgiDuplicatorController::Instance()
60 ->ScreenRect(current_screen_id_) 61 ->ScreenRect(current_screen_id_)
61 .size(); 62 .size();
62 } 63 }
63 64
64 void ScreenCapturerWinDirectx::CaptureFrame() { 65 void ScreenCapturerWinDirectx::CaptureFrame() {
65 RTC_DCHECK(callback_); 66 RTC_DCHECK(callback_);
66 67
67 int64_t capture_start_time_nanos = rtc::TimeNanos(); 68 int64_t capture_start_time_nanos = rtc::TimeNanos();
68 69
70 if (resolution_change_detector_.IsChanged(
71 GetScreenRect(kFullDesktopScreenId, L"").size())) {
Sergey Ulanov 2017/02/10 19:58:25 std::wstring() for the second parameter
Sergey Ulanov 2017/02/10 19:58:25 Why is it kFullDesktopScreenId instead of current_
Sergey Ulanov 2017/02/10 19:58:25 indentation - this line should be indented 4 space
Hzj_jie 2017/02/11 01:51:22 Done.
Hzj_jie 2017/02/11 01:51:22 Done.
Hzj_jie 2017/02/11 01:51:22 Three reasons, 1. it's not guaranteed GetScreenRec
Sergey Ulanov 2017/02/15 01:38:52 They maybe we should use DX API to get the screen
Hzj_jie 2017/02/15 03:53:16 Yes, technically, we can do this. But we need to i
Sergey Ulanov 2017/02/15 22:39:38 Not sure why it's a cyclic dependency. This functi
72 // The next Duplicate() call is expected to fail. So we reset the dxgi
73 // components in-place.
74 frames_.Reset();
75 DxgiDuplicatorController::Instance()->Reset();
76 resolution_change_detector_.Reset();
77 }
78
69 frames_.MoveToNextFrame(); 79 frames_.MoveToNextFrame();
70 if (!frames_.current_frame()) { 80 if (!frames_.current_frame()) {
71 std::unique_ptr<DesktopFrame> new_frame; 81 std::unique_ptr<DesktopFrame> new_frame;
72 if (shared_memory_factory_) { 82 if (shared_memory_factory_) {
73 new_frame = SharedMemoryDesktopFrame::Create( 83 new_frame = SharedMemoryDesktopFrame::Create(
74 SelectedDesktopSize(), shared_memory_factory_.get()); 84 SelectedDesktopSize(), shared_memory_factory_.get());
75 } else { 85 } else {
76 new_frame.reset(new BasicDesktopFrame(SelectedDesktopSize())); 86 new_frame.reset(new BasicDesktopFrame(SelectedDesktopSize()));
77 } 87 }
78 if (!new_frame) { 88 if (!new_frame) {
79 LOG(LS_ERROR) << "Failed to allocate a new DesktopFrame."; 89 LOG(LS_ERROR) << "Failed to allocate a new DesktopFrame.";
80 // This usually means we do not have enough memory or SharedMemoryFactory 90 // This usually means we do not have enough memory or SharedMemoryFactory
81 // cannot work correctly. 91 // cannot work correctly.
82 callback_->OnCaptureResult(Result::ERROR_PERMANENT, nullptr); 92 callback_->OnCaptureResult(Result::ERROR_PERMANENT, nullptr);
83 return; 93 return;
84 } 94 }
85 frames_.ReplaceCurrentFrame(SharedDesktopFrame::Wrap(std::move(new_frame))); 95 frames_.ReplaceCurrentFrame(SharedDesktopFrame::Wrap(std::move(new_frame)));
86 } 96 }
87 97
88 if (current_screen_id_ == kFullDesktopScreenId) { 98 if (current_screen_id_ == kFullDesktopScreenId) {
89 if (!DxgiDuplicatorController::Instance()->Duplicate( 99 if (!DxgiDuplicatorController::Instance()->Duplicate(
90 &context_, frames_.current_frame())) { 100 &context_, frames_.current_frame())) {
91 // Screen size may be changed, so we need to reset the frames. 101 // Screen size may be changed, so we need to reset the frames.
92 frames_.Reset(); 102 frames_.Reset();
103 resolution_change_detector_.Reset();
93 callback_->OnCaptureResult(Result::ERROR_TEMPORARY, nullptr); 104 callback_->OnCaptureResult(Result::ERROR_TEMPORARY, nullptr);
94 return; 105 return;
95 } 106 }
96 } else { 107 } else {
97 if (!DxgiDuplicatorController::Instance()->DuplicateMonitor( 108 if (!DxgiDuplicatorController::Instance()->DuplicateMonitor(
98 &context_, current_screen_id_, frames_.current_frame())) { 109 &context_, current_screen_id_, frames_.current_frame())) {
99 // Screen size may be changed, so we need to reset the frames. 110 // Screen size may be changed, so we need to reset the frames.
100 frames_.Reset(); 111 frames_.Reset();
112 resolution_change_detector_.Reset();
101 if (current_screen_id_ >= 113 if (current_screen_id_ >=
102 DxgiDuplicatorController::Instance()->ScreenCount()) { 114 DxgiDuplicatorController::Instance()->ScreenCount()) {
103 // Current monitor has been removed from the system. 115 // Current monitor has been removed from the system.
104 callback_->OnCaptureResult(Result::ERROR_PERMANENT, nullptr); 116 callback_->OnCaptureResult(Result::ERROR_PERMANENT, nullptr);
105 } else { 117 } else {
106 callback_->OnCaptureResult(Result::ERROR_TEMPORARY, nullptr); 118 callback_->OnCaptureResult(Result::ERROR_TEMPORARY, nullptr);
107 } 119 }
108 return; 120 return;
109 } 121 }
110 } 122 }
111 123
112 std::unique_ptr<DesktopFrame> result = frames_.current_frame()->Share(); 124 std::unique_ptr<DesktopFrame> result = frames_.current_frame()->Share();
113 result->set_capture_time_ms( 125 result->set_capture_time_ms(
114 (rtc::TimeNanos() - capture_start_time_nanos) / 126 (rtc::TimeNanos() - capture_start_time_nanos) /
115 rtc::kNumNanosecsPerMillisec); 127 rtc::kNumNanosecsPerMillisec);
116 callback_->OnCaptureResult(Result::SUCCESS, std::move(result)); 128 callback_->OnCaptureResult(Result::SUCCESS, std::move(result));
117 } 129 }
118 130
119 bool ScreenCapturerWinDirectx::GetSourceList(SourceList* sources) { 131 bool ScreenCapturerWinDirectx::GetSourceList(SourceList* sources) {
120 int screen_count = DxgiDuplicatorController::Instance()->ScreenCount(); 132 int screen_count = DxgiDuplicatorController::Instance()->ScreenCount();
121 for (int i = 0; i < screen_count; i++) { 133 for (int i = 0; i < screen_count; i++) {
122 sources->push_back({i}); 134 sources->push_back({i});
123 } 135 }
124 return true; 136 return true;
125 } 137 }
126 138
127 bool ScreenCapturerWinDirectx::SelectSource(SourceId id) { 139 bool ScreenCapturerWinDirectx::SelectSource(SourceId id) {
Sergey Ulanov 2017/02/10 19:58:25 Should we also initialized duplicators and reset r
Hzj_jie 2017/02/11 01:51:22 The initialization is implicitly called in DxgiDup
Sergey Ulanov 2017/02/15 01:38:52 If there are two monitors with the same resoltion
Hzj_jie 2017/02/15 03:53:16 No, we only need to reinitialize if a new monitor
128 if (id == current_screen_id_) { 140 if (id == current_screen_id_) {
129 return true; 141 return true;
130 } 142 }
131 143
132 // Changing target screen may or may not impact frame size. So resetting 144 // Changing target screen may or may not impact frame size. So resetting
133 // frames only when a Duplicate() function call returns false. 145 // frames only when a Duplicate() function call returns false.
134 if (id == kFullDesktopScreenId) { 146 if (id == kFullDesktopScreenId) {
135 current_screen_id_ = id; 147 current_screen_id_ = id;
136 return true; 148 return true;
137 } 149 }
138 150
139 int screen_count = DxgiDuplicatorController::Instance()->ScreenCount(); 151 int screen_count = DxgiDuplicatorController::Instance()->ScreenCount();
140 if (id >= 0 && id < screen_count) { 152 if (id >= 0 && id < screen_count) {
141 current_screen_id_ = id; 153 current_screen_id_ = id;
142 return true; 154 return true;
143 } 155 }
144 return false; 156 return false;
145 } 157 }
146 158
147 } // namespace webrtc 159 } // namespace webrtc
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698