Chromium Code Reviews| Index: webrtc/modules/desktop_capture/win/dxgi_output_duplicator.h |
| diff --git a/webrtc/modules/desktop_capture/win/dxgi_output_duplicator.h b/webrtc/modules/desktop_capture/win/dxgi_output_duplicator.h |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..85520047541e418d6e01dc8600741045c4738501 |
| --- /dev/null |
| +++ b/webrtc/modules/desktop_capture/win/dxgi_output_duplicator.h |
| @@ -0,0 +1,127 @@ |
| +/* |
| + * 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. |
| + */ |
| + |
| +#ifndef MODULES_DESKTOP_CAPTURE_WIN_DXGI_OUTPUT_DUPLICATOR_H_ |
| +#define MODULES_DESKTOP_CAPTURE_WIN_DXGI_OUTPUT_DUPLICATOR_H_ |
| + |
| +#include <comdef.h> |
| +#include <wrl/client.h> |
| +#include <DXGI.h> |
| +#include <DXGI1_2.h> |
| + |
| +#include <memory> |
| +#include <vector> |
| + |
| +#include "webrtc/base/criticalsection.h" |
| +#include "webrtc/base/thread_annotations.h" |
| +#include "webrtc/modules/desktop_capture/desktop_frame.h" |
| +#include "webrtc/modules/desktop_capture/desktop_geometry.h" |
| +#include "webrtc/modules/desktop_capture/desktop_region.h" |
| +#include "webrtc/modules/desktop_capture/win/d3d_device.h" |
| +#include "webrtc/modules/desktop_capture/win/dxgi_texture.h" |
| + |
| +namespace webrtc { |
| + |
| +struct DxgiOutputContext { |
|
Sergey Ulanov
2016/08/06 01:27:57
Nest this inside DxgiOutputDuplicator as DxgiOutpu
Hzj_jie
2016/08/08 00:16:09
Oh, I thought you won't like to have classes with
|
| + // The updated region DxgiOutputDuplicator::DetectUpdatedRegion() output |
| + // during last Duplicate() function call. It's a DesktopRegion translated by |
| + // offset of each DxgiOutputDuplicator instance. |
| + DesktopRegion updated_region; |
| +}; |
| + |
| +// Duplicates the content on one IDXGIOutput, i.e. one monitor attached to one |
| +// video card. None of functions in this class is thread-safe. |
| +// TODO(zijiehe): Understand the meaning of rotation. |
| +class DxgiOutputDuplicator { |
| + public: |
| + // Creates an instance of DxgiOutputDuplicator from a D3dDevice and one of its |
| + // IDXGIOutput1. Caller must maintain the lifetime of device, to make sure it |
| + // outlives this instance. Only DxgiAdapterDuplicator can create an instance. |
| + DxgiOutputDuplicator(const D3dDevice& device, |
| + const Microsoft::WRL::ComPtr<IDXGIOutput1>& output, |
| + const DXGI_OUTPUT_DESC& desc); |
| + |
| + // To allow this class to work with vector. |
| + DxgiOutputDuplicator(DxgiOutputDuplicator&& other); |
| + |
| + // Destructs this instance. We need to make sure texture_ has been released |
| + // before duplication_. |
| + ~DxgiOutputDuplicator(); |
| + |
| + // Initializes duplication_ object. |
| + bool Initialize(); |
| + |
| + // Copies the content of current IDXGIOutput to the |target|. To improve the |
| + // performance, this function copies only regions merged from |
| + // |last_frame|.updated_region and DetectUpdatedRegion. The offset decides the |
| + // offset in the |target| where the content should be copied to. i.e. this |
| + // function copies the content to the rectangle of (offset.x(), offset.y()) to |
| + // (offset.x() + desktop_rect_.width(), offset.y() + desktop_rect_.height()). |
| + // The |last_frame| is always expected to be translated by the same offset. |
| + // Returns false if this function failed to execute Windows APIs. |
|
Sergey Ulanov
2016/08/06 01:27:57
s/if this function failed to execute Windows APIs/
Hzj_jie
2016/08/08 00:16:09
Done.
|
| + bool Duplicate(DxgiOutputContext* context, |
| + const DesktopFrame* last_frame, |
| + const DesktopVector offset, |
| + DesktopFrame* target); |
| + |
| + // Returns the desktop rect covered by this DxgiOutputDuplicator. |
| + DesktopRect desktop_rect() const { return desktop_rect_; } |
| + |
| + private: |
| + friend class DxgiAdapterDuplicator; |
| + |
| + // Detects updated region translated by offset from IDXGIOutput1. This |
| + // function will set the |updated_region| as entire DesktopRect starts from |
| + // offset if it failed to execute Windows APIs. |
| + void DetectUpdatedRegion(const DXGI_OUTDUPL_FRAME_INFO& frame_info, |
| + const DesktopVector offset, |
| + DesktopRegion* updated_region); |
| + |
| + // Returns untranslated updated region, which are directly returned by Windows |
| + // APIs. Returns false if this function failed to execute Windows APIs. |
| + bool DoDetectUpdatedRegion(const DXGI_OUTDUPL_FRAME_INFO& frame_info, |
| + DesktopRegion* updated_region); |
| + |
| + bool ReleaseFrame(); |
| + |
| + // Initializes duplication_ instance. Expects duplication_ is in empty status. |
| + // Returns false if system does not support IDXGIOutputDuplication. |
| + bool DuplicateOutput(); |
| + |
| + // Returns a DesktopRect with the same size of desktop_size_, but translated |
| + // by offset. |
| + DesktopRect TranslatedDesktopRect(const DesktopVector offset); |
| + |
| + void Setup(DxgiOutputContext* context); |
| + |
| + void Unregister(const DxgiOutputContext* const context); |
| + |
| + // Spreads changes from |context| to other registered DxgiOutputContext(s) in |
| + // contexts_. |
| + void SpreadContextChange(const DxgiOutputContext* const context); |
| + |
| + const D3dDevice& device_; |
| + const Microsoft::WRL::ComPtr<IDXGIOutput1> output_; |
| + const DesktopRect desktop_rect_; |
| + Microsoft::WRL::ComPtr<IDXGIOutputDuplication> duplication_; |
| + DXGI_OUTDUPL_DESC desc_; |
| + std::vector<uint8_t> metadata; |
| + std::unique_ptr<DxgiTexture> texture_; |
| + |
| + // After each AcquireNextFrame() function call, updated_region_(s) of all |
| + // active DxgiOutputContext(s) need to be updated. Since they have missed the |
| + // change this time. And during next Duplicate() function call, their |
| + // updated_region_ will be merged and copied. |
| + std::vector<DxgiOutputContext*> contexts_; |
| +}; |
| + |
| +} // namespace webrtc |
| + |
| +#endif // MODULES_DESKTOP_CAPTURE_WIN_DXGI_OUTPUT_DUPLICATOR_H_ |