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

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

Issue 2801433002: DirectX capturer may crash after switching shared screen (Closed)
Patch Set: context->updated_region is always untranslated Created 3 years, 8 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/dxgi_output_duplicator.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
(...skipping 39 matching lines...) Expand 10 before | Expand all | Expand 10 after
50 case DXGI_MODE_ROTATION_ROTATE180: 50 case DXGI_MODE_ROTATION_ROTATE180:
51 return Rotation::CLOCK_WISE_180; 51 return Rotation::CLOCK_WISE_180;
52 case DXGI_MODE_ROTATION_ROTATE270: 52 case DXGI_MODE_ROTATION_ROTATE270:
53 return Rotation::CLOCK_WISE_270; 53 return Rotation::CLOCK_WISE_270;
54 } 54 }
55 55
56 RTC_NOTREACHED(); 56 RTC_NOTREACHED();
57 return Rotation::CLOCK_WISE_0; 57 return Rotation::CLOCK_WISE_0;
58 } 58 }
59 59
60 // Translates |rect| with the reverse of |offset|
61 DesktopRect ReverseTranslate(DesktopRect rect, DesktopVector offset) {
62 rect.Translate(-offset.x(), -offset.y());
63 return rect;
64 }
65
66 } // namespace 60 } // namespace
67 61
68 DxgiOutputDuplicator::DxgiOutputDuplicator(const D3dDevice& device, 62 DxgiOutputDuplicator::DxgiOutputDuplicator(const D3dDevice& device,
69 const ComPtr<IDXGIOutput1>& output, 63 const ComPtr<IDXGIOutput1>& output,
70 const DXGI_OUTPUT_DESC& desc) 64 const DXGI_OUTPUT_DESC& desc)
71 : device_(device), 65 : device_(device),
72 output_(output), 66 output_(output),
73 desktop_rect_(RECTToDesktopRect(desc.DesktopCoordinates)) { 67 desktop_rect_(RECTToDesktopRect(desc.DesktopCoordinates)) {
74 RTC_DCHECK(output_); 68 RTC_DCHECK(output_);
75 RTC_DCHECK(!desktop_rect_.is_empty()); 69 RTC_DCHECK(!desktop_rect_.is_empty());
(...skipping 49 matching lines...) Expand 10 before | Expand all | Expand 10 after
125 static_cast<int>(desc_.ModeDesc.Height) != desktop_rect_.height()) { 119 static_cast<int>(desc_.ModeDesc.Height) != desktop_rect_.height()) {
126 LOG(LS_ERROR) << "IDXGIDuplicateOutput does not return a same size as its " 120 LOG(LS_ERROR) << "IDXGIDuplicateOutput does not return a same size as its "
127 "IDXGIOutput1, size returned by IDXGIDuplicateOutput is " 121 "IDXGIOutput1, size returned by IDXGIDuplicateOutput is "
128 << desc_.ModeDesc.Width << " x " << desc_.ModeDesc.Height 122 << desc_.ModeDesc.Width << " x " << desc_.ModeDesc.Height
129 << ", size returned by IDXGIOutput1 is " 123 << ", size returned by IDXGIOutput1 is "
130 << desktop_rect_.width() << " x " << desktop_rect_.height(); 124 << desktop_rect_.width() << " x " << desktop_rect_.height();
131 return false; 125 return false;
132 } 126 }
133 127
134 rotation_ = DxgiRotationToRotation(desc_.Rotation); 128 rotation_ = DxgiRotationToRotation(desc_.Rotation);
135 unrotated_size_ = 129 unrotated_size_ = RotateSize(desktop_size(), ReverseRotation(rotation_));
136 RotateSize(desktop_rect_.size(), ReverseRotation(rotation_));
137 130
138 return true; 131 return true;
139 } 132 }
140 133
141 bool DxgiOutputDuplicator::ReleaseFrame() { 134 bool DxgiOutputDuplicator::ReleaseFrame() {
142 RTC_DCHECK(duplication_); 135 RTC_DCHECK(duplication_);
143 _com_error error = duplication_->ReleaseFrame(); 136 _com_error error = duplication_->ReleaseFrame();
144 if (error.Error() != S_OK) { 137 if (error.Error() != S_OK) {
145 LOG(LS_ERROR) << "Failed to release frame from IDXGIOutputDuplication, " 138 LOG(LS_ERROR) << "Failed to release frame from IDXGIOutputDuplication, "
146 "error" 139 "error"
(...skipping 21 matching lines...) Expand all
168 _com_error error = duplication_->AcquireNextFrame( 161 _com_error error = duplication_->AcquireNextFrame(
169 kAcquireTimeoutMs, &frame_info, resource.GetAddressOf()); 162 kAcquireTimeoutMs, &frame_info, resource.GetAddressOf());
170 if (error.Error() != S_OK && error.Error() != DXGI_ERROR_WAIT_TIMEOUT) { 163 if (error.Error() != S_OK && error.Error() != DXGI_ERROR_WAIT_TIMEOUT) {
171 LOG(LS_ERROR) << "Failed to capture frame, error " << error.ErrorMessage() 164 LOG(LS_ERROR) << "Failed to capture frame, error " << error.ErrorMessage()
172 << ", code " << error.Error(); 165 << ", code " << error.Error();
173 return false; 166 return false;
174 } 167 }
175 168
176 // We need to merge updated region with the one from context, but only spread 169 // We need to merge updated region with the one from context, but only spread
177 // updated region from current frame. So keeps a copy of updated region from 170 // updated region from current frame. So keeps a copy of updated region from
178 // context here. 171 // context here. The |updated_region| always starts from (0, 0).
179 DesktopRegion updated_region; 172 DesktopRegion updated_region;
180 updated_region.Swap(&context->updated_region); 173 updated_region.Swap(&context->updated_region);
181 if (error.Error() == S_OK && 174 if (error.Error() == S_OK &&
182 frame_info.AccumulatedFrames > 0 && 175 frame_info.AccumulatedFrames > 0 &&
183 resource) { 176 resource) {
184 DetectUpdatedRegion(frame_info, offset, &context->updated_region); 177 DetectUpdatedRegion(frame_info, &context->updated_region);
178 SpreadContextChange(context);
185 if (!texture_->CopyFrom(frame_info, resource.Get())) { 179 if (!texture_->CopyFrom(frame_info, resource.Get())) {
186 return false; 180 return false;
187 } 181 }
188 SpreadContextChange(context);
189 updated_region.AddRegion(context->updated_region); 182 updated_region.AddRegion(context->updated_region);
183 // TODO(zijiehe): Why clearing context->updated_region() here triggers
Sergey Ulanov 2017/04/05 19:47:57 nit: s/Why/Why does/ and s/triggers/trigger/. Or a
Hzj_jie 2017/04/05 21:16:59 Done.
184 // screen flickering?
190 185
191 const DesktopFrame& source = texture_->AsDesktopFrame(); 186 const DesktopFrame& source = texture_->AsDesktopFrame();
192 if (rotation_ != Rotation::CLOCK_WISE_0) { 187 if (rotation_ != Rotation::CLOCK_WISE_0) {
193 for (DesktopRegion::Iterator it(updated_region); !it.IsAtEnd(); 188 for (DesktopRegion::Iterator it(updated_region); !it.IsAtEnd();
194 it.Advance()) { 189 it.Advance()) {
195 const DesktopRect source_rect = 190 // The |updated_region| returned by Windows is rotated, but the |source|
196 RotateRect(ReverseTranslate(it.rect(), offset), 191 // frame is not. So we need to rotate it reversely.
197 desktop_rect().size(), ReverseRotation(rotation_)); 192 const DesktopRect source_rect = RotateRect(
193 it.rect(), desktop_size(), ReverseRotation(rotation_));
198 RotateDesktopFrame(source, source_rect, rotation_, offset, target); 194 RotateDesktopFrame(source, source_rect, rotation_, offset, target);
199 } 195 }
200 } else { 196 } else {
201 for (DesktopRegion::Iterator it(updated_region); !it.IsAtEnd(); 197 for (DesktopRegion::Iterator it(updated_region); !it.IsAtEnd();
202 it.Advance()) { 198 it.Advance()) {
203 target->CopyPixelsFrom( 199 // The DesktopRect in |source|, starts from (0, 0).
204 source, ReverseTranslate(it.rect(), offset).top_left(), it.rect()); 200 const DesktopRect source_rect = it.rect();
Sergey Ulanov 2017/04/05 19:47:57 Maybe remove this var and just use it.rect() below
Hzj_jie 2017/04/05 21:16:58 Definitely.
201 // The DesktopRect in |target|, starts from offset.
202 DesktopRect dest_rect = source_rect;
203 dest_rect.Translate(offset);
204 target->CopyPixelsFrom(source, source_rect.top_left(), dest_rect);
205 } 205 }
206 } 206 }
207 last_frame_ = target->Share(); 207 last_frame_ = target->Share();
208 last_frame_offset_ = offset; 208 last_frame_offset_ = offset;
209 updated_region.Translate(offset.x(), offset.y());
209 target->mutable_updated_region()->AddRegion(updated_region); 210 target->mutable_updated_region()->AddRegion(updated_region);
210 num_frames_captured_++; 211 num_frames_captured_++;
211 return texture_->Release() && ReleaseFrame(); 212 return texture_->Release() && ReleaseFrame();
212 } 213 }
213 214
214 if (last_frame_) { 215 if (last_frame_) {
215 // No change since last frame or AcquireNextFrame() timed out, we will 216 // No change since last frame or AcquireNextFrame() timed out, we will
216 // export last frame to the target. 217 // export last frame to the target.
217 for (DesktopRegion::Iterator it(updated_region); !it.IsAtEnd(); 218 for (DesktopRegion::Iterator it(updated_region); !it.IsAtEnd();
218 it.Advance()) { 219 it.Advance()) {
219 target->CopyPixelsFrom(*last_frame_, it.rect().top_left(), it.rect()); 220 // The DesktopRect in |source|, starts from last_frame_offset_.
221 DesktopRect source_rect = it.rect();
222 // The DesktopRect in |target|, starts from offset.
223 DesktopRect target_rect = source_rect;
224 source_rect.Translate(last_frame_offset_);
225 target_rect.Translate(offset);
226 target->CopyPixelsFrom(*last_frame_, source_rect.top_left(), target_rect);
220 } 227 }
228 updated_region.Translate(offset.x(), offset.y());
221 target->mutable_updated_region()->AddRegion(updated_region); 229 target->mutable_updated_region()->AddRegion(updated_region);
222 } else { 230 } else {
223 // If we were at the very first frame, and capturing failed, the 231 // If we were at the very first frame, and capturing failed, the
224 // context->updated_region should be kept unchanged for next attempt. 232 // context->updated_region should be kept unchanged for next attempt.
225 context->updated_region.Swap(&updated_region); 233 context->updated_region.Swap(&updated_region);
226 } 234 }
227 // If AcquireNextFrame() failed with timeout error, we do not need to release 235 // If AcquireNextFrame() failed with timeout error, we do not need to release
228 // the frame. 236 // the frame.
229 return error.Error() == DXGI_ERROR_WAIT_TIMEOUT || ReleaseFrame(); 237 return error.Error() == DXGI_ERROR_WAIT_TIMEOUT || ReleaseFrame();
230 } 238 }
231 239
232 DesktopRect DxgiOutputDuplicator::TranslatedDesktopRect(DesktopVector offset) { 240 DesktopRect DxgiOutputDuplicator::TranslatedDesktopRect(
Sergey Ulanov 2017/04/05 19:47:57 GetTranslatedDesktopRect()?
Hzj_jie 2017/04/05 21:16:59 Done. Also changed UntranslatedDesktopRect() into
233 DesktopRect result(DesktopRect::MakeSize(desktop_rect_.size())); 241 DesktopVector offset) const {
242 DesktopRect result(DesktopRect::MakeSize(desktop_size()));
234 result.Translate(offset); 243 result.Translate(offset);
235 return result; 244 return result;
236 } 245 }
237 246
247 DesktopRect DxgiOutputDuplicator::UntranslatedDesktopRect() const {
Sergey Ulanov 2017/04/05 19:47:57 desktop_rect()?
Hzj_jie 2017/04/05 21:16:59 It may be too confusing, IMO.
248 return TranslatedDesktopRect(DesktopVector());
Sergey Ulanov 2017/04/05 19:47:57 I think just "DesktopRect::MakeSize(desktop_size()
Hzj_jie 2017/04/05 21:16:58 Done.
249 }
250
238 void DxgiOutputDuplicator::DetectUpdatedRegion( 251 void DxgiOutputDuplicator::DetectUpdatedRegion(
239 const DXGI_OUTDUPL_FRAME_INFO& frame_info, 252 const DXGI_OUTDUPL_FRAME_INFO& frame_info,
240 DesktopVector offset,
241 DesktopRegion* updated_region) { 253 DesktopRegion* updated_region) {
242 if (DoDetectUpdatedRegion(frame_info, updated_region)) { 254 if (DoDetectUpdatedRegion(frame_info, updated_region)) {
243 updated_region->Translate(offset.x(), offset.y());
244 // Make sure even a region returned by Windows API is out of the scope of 255 // Make sure even a region returned by Windows API is out of the scope of
245 // desktop_rect_, we still won't export it to the target DesktopFrame. 256 // desktop_rect_, we still won't export it to the target DesktopFrame.
246 updated_region->IntersectWith(TranslatedDesktopRect(offset)); 257 updated_region->IntersectWith(UntranslatedDesktopRect());
247 } else { 258 } else {
248 updated_region->SetRect(TranslatedDesktopRect(offset)); 259 updated_region->SetRect(UntranslatedDesktopRect());
249 } 260 }
250 } 261 }
251 262
252 bool DxgiOutputDuplicator::DoDetectUpdatedRegion( 263 bool DxgiOutputDuplicator::DoDetectUpdatedRegion(
253 const DXGI_OUTDUPL_FRAME_INFO& frame_info, 264 const DXGI_OUTDUPL_FRAME_INFO& frame_info,
254 DesktopRegion* updated_region) { 265 DesktopRegion* updated_region) {
255 RTC_DCHECK(updated_region); 266 RTC_DCHECK(updated_region);
256 updated_region->Clear(); 267 updated_region->Clear();
257 if (frame_info.TotalMetadataBufferSize == 0) { 268 if (frame_info.TotalMetadataBufferSize == 0) {
258 // This should not happen, since frame_info.AccumulatedFrames > 0. 269 // This should not happen, since frame_info.AccumulatedFrames > 0.
(...skipping 74 matching lines...) Expand 10 before | Expand all | Expand 10 after
333 dirty_rects++; 344 dirty_rects++;
334 dirty_rects_count--; 345 dirty_rects_count--;
335 } 346 }
336 347
337 return true; 348 return true;
338 } 349 }
339 350
340 void DxgiOutputDuplicator::Setup(Context* context) { 351 void DxgiOutputDuplicator::Setup(Context* context) {
341 RTC_DCHECK(context->updated_region.is_empty()); 352 RTC_DCHECK(context->updated_region.is_empty());
342 // Always copy entire monitor during the first Duplicate() function call. 353 // Always copy entire monitor during the first Duplicate() function call.
343 context->updated_region.AddRect(desktop_rect_); 354 context->updated_region.AddRect(UntranslatedDesktopRect());
344 RTC_DCHECK(std::find(contexts_.begin(), contexts_.end(), context) == 355 RTC_DCHECK(std::find(contexts_.begin(), contexts_.end(), context) ==
345 contexts_.end()); 356 contexts_.end());
346 contexts_.push_back(context); 357 contexts_.push_back(context);
347 } 358 }
348 359
349 void DxgiOutputDuplicator::Unregister(const Context* const context) { 360 void DxgiOutputDuplicator::Unregister(const Context* const context) {
350 auto it = std::find(contexts_.begin(), contexts_.end(), context); 361 auto it = std::find(contexts_.begin(), contexts_.end(), context);
351 RTC_DCHECK(it != contexts_.end()); 362 RTC_DCHECK(it != contexts_.end());
352 contexts_.erase(it); 363 contexts_.erase(it);
353 } 364 }
354 365
355 void DxgiOutputDuplicator::SpreadContextChange(const Context* const source) { 366 void DxgiOutputDuplicator::SpreadContextChange(const Context* const source) {
356 for (Context* dest : contexts_) { 367 for (Context* dest : contexts_) {
357 RTC_DCHECK(dest); 368 RTC_DCHECK(dest);
358 if (dest != source) { 369 if (dest != source) {
359 dest->updated_region.AddRegion(source->updated_region); 370 dest->updated_region.AddRegion(source->updated_region);
360 } 371 }
361 } 372 }
362 } 373 }
363 374
375 DesktopSize DxgiOutputDuplicator::desktop_size() const {
376 return desktop_rect_.size();
377 }
378
364 int64_t DxgiOutputDuplicator::num_frames_captured() const { 379 int64_t DxgiOutputDuplicator::num_frames_captured() const {
365 #if !defined(NDEBUG) 380 #if !defined(NDEBUG)
366 RTC_DCHECK_EQ(!!last_frame_, num_frames_captured_ > 0); 381 RTC_DCHECK_EQ(!!last_frame_, num_frames_captured_ > 0);
367 #endif 382 #endif
368 return num_frames_captured_; 383 return num_frames_captured_;
369 } 384 }
370 385
371 } // namespace webrtc 386 } // namespace webrtc
OLDNEW
« no previous file with comments | « webrtc/modules/desktop_capture/win/dxgi_output_duplicator.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698