Chromium Code Reviews| Index: webrtc/modules/desktop_capture/win/dxgi_adapter_duplicator.cc |
| diff --git a/webrtc/modules/desktop_capture/win/dxgi_adapter_duplicator.cc b/webrtc/modules/desktop_capture/win/dxgi_adapter_duplicator.cc |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..4d1bf3484b580ef5b213ca51598a9ce82485b8b6 |
| --- /dev/null |
| +++ b/webrtc/modules/desktop_capture/win/dxgi_adapter_duplicator.cc |
| @@ -0,0 +1,146 @@ |
| +/* |
| + * Copyright (c) 2016 The WebRTC project authors. All Rights Reserved. |
| + * |
| + * Use of this source code is governed by a BSD-style license |
| + * that can be found in the LICENSE file in the root of the source |
| + * tree. An additional intellectual property rights grant can be found |
| + * in the file PATENTS. All contributing project authors may |
| + * be found in the AUTHORS file in the root of the source tree. |
| + */ |
| + |
| +#include "webrtc/modules/desktop_capture/win/dxgi_adapter_duplicator.h" |
| + |
| +#include <comdef.h> |
| +#include <DXGI.h> |
| + |
| +#include <algorithm> |
| + |
| +#include "webrtc/base/checks.h" |
| +#include "webrtc/base/logging.h" |
| + |
| +namespace webrtc { |
| + |
| +using Microsoft::WRL::ComPtr; |
| + |
| +namespace { |
| + |
| +bool IsValidRect(const RECT& rect) { |
| + return rect.left >= 0 && rect.top >= 0 && rect.right > rect.left && |
| + rect.bottom > rect.top; |
| +} |
| + |
| +} // namespace |
| + |
| +DxgiAdapterDuplicator::DxgiAdapterDuplicator(const D3dDevice& device) |
| + : device_(device) {} |
| + |
| +DxgiAdapterDuplicator::DxgiAdapterDuplicator(DxgiAdapterDuplicator&&) = default; |
| + |
| +bool DxgiAdapterDuplicator::Initialize() { |
| + if (DoInitialize()) { |
| + return true; |
| + } |
| + duplicators_.clear(); |
| + return false; |
| +} |
| + |
| +bool DxgiAdapterDuplicator::DoInitialize() { |
| + for (int i = 0;; i++) { |
| + ComPtr<IDXGIOutput> output; |
| + _com_error error = |
| + device_.dxgi_adapter()->EnumOutputs(i, output.GetAddressOf()); |
| + if (error.Error() == DXGI_ERROR_NOT_FOUND) { |
| + break; |
| + } |
| + |
| + if (error.Error() != S_OK || !output) { |
| + LOG(LS_WARNING) << "IDXGIAdapter::EnumOutputs returns an unexpected " |
| + "result " |
| + << error.ErrorMessage() << " with error code" |
| + << error.Error(); |
| + return false; |
| + } |
| + |
| + DXGI_OUTPUT_DESC desc; |
| + error = _com_error(output->GetDesc(&desc)); |
| + if (error.Error() == S_OK) { |
| + if (desc.AttachedToDesktop && IsValidRect(desc.DesktopCoordinates)) { |
| + ComPtr<IDXGIOutput1> output1; |
| + error = _com_error(output.As(&output1)); |
| + if (error.Error() != S_OK || !output1) { |
| + LOG(LS_WARNING) << "Failed to convert IDXGIOutput to IDXGIOutput1, " |
| + "this usually means the system does not support " |
| + "DirectX 11"; |
| + return false; |
| + } |
| + duplicators_.emplace_back(device_, output1, desc); |
| + if (!duplicators_.back().Initialize()) { |
| + return false; |
| + } |
| + if (desktop_rect_.is_empty()) { |
| + desktop_rect_ = duplicators_.back().desktop_rect(); |
| + } else { |
| + const DesktopRect& left = desktop_rect_; |
| + const DesktopRect& right = duplicators_.back().desktop_rect(); |
| + desktop_rect_ = |
| + DesktopRect::MakeLTRB(std::min(left.left(), right.left()), |
| + std::min(left.top(), right.top()), |
| + std::max(left.right(), right.right()), |
| + std::max(left.bottom(), right.bottom())); |
| + } |
| + } |
| + } else { |
| + LOG(LS_WARNING) << "Failed to get output description of device " << i |
| + << ", ignore."; |
| + } |
| + } |
| + return true; |
| +} |
| + |
| +void DxgiAdapterDuplicator::Setup(DxgiAdapterContext* context) { |
| + RTC_DCHECK(context->contexts.empty()); |
| + context->contexts.resize(duplicators_.size()); |
| + for (size_t i = 0; i < duplicators_.size(); i++) { |
| + duplicators_[i].Setup(&context->contexts[i]); |
| + } |
| +} |
| + |
| +void DxgiAdapterDuplicator::Unregister( |
| + const DxgiAdapterContext* const context) { |
| + RTC_DCHECK(context->contexts.size() == duplicators_.size()); |
|
Sergey Ulanov
2016/08/06 01:27:57
RTC_DCHECK_EQ
Hzj_jie
2016/08/08 00:16:08
Done.
|
| + for (size_t i = 0; i < duplicators_.size(); i++) { |
| + duplicators_[i].Unregister(&context->contexts[i]); |
| + } |
| +} |
| + |
| +bool DxgiAdapterDuplicator::Duplicate(DxgiAdapterContext* context, |
| + const DesktopFrame* last_frame, |
| + DesktopFrame* target) { |
| + RTC_DCHECK(context->contexts.size() == duplicators_.size()); |
|
Sergey Ulanov
2016/08/06 01:27:57
RTC_DCHECK_EQ
Hzj_jie
2016/08/08 00:16:08
Done.
|
| + for (size_t i = 0; i < duplicators_.size(); i++) { |
| + if (!duplicators_[i].Duplicate(&context->contexts[i], last_frame, |
| + duplicators_[i].desktop_rect().top_left(), |
| + target)) { |
| + return false; |
| + } |
| + } |
| + return true; |
| +} |
| + |
| +bool DxgiAdapterDuplicator::DuplicateMonitor(DxgiAdapterContext* context, |
| + int monitor_id, |
| + const DesktopFrame* last_frame, |
| + DesktopFrame* target) { |
| + RTC_DCHECK(monitor_id >= 0 && |
| + monitor_id < static_cast<int>(duplicators_.size()) && |
| + context->contexts.size() == duplicators_.size()); |
| + return duplicators_[monitor_id].Duplicate( |
| + &context->contexts[monitor_id], last_frame, DesktopVector(), target); |
| +} |
| + |
| +DesktopRect DxgiAdapterDuplicator::ScreenRect(int id) const { |
| + RTC_DCHECK(id >= 0 && id < static_cast<int>(duplicators_.size())); |
| + return duplicators_[id].desktop_rect(); |
| +} |
| + |
| +} // namespace webrtc |