2020-11-20 18:20:36 +00:00
|
|
|
/*
|
|
|
|
* GStreamer
|
2022-10-06 14:54:14 +00:00
|
|
|
* Copyright (C) 2022 Seungha Yang <seungha@centricular.com>
|
2020-11-20 18:20:36 +00:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Library General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Library General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
* License along with this library; if not, write to the
|
|
|
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
|
|
* Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2021-10-13 12:45:34 +00:00
|
|
|
#include "gstd3d11screencapture.h"
|
2021-03-13 08:40:57 +00:00
|
|
|
#include "gstd3d11pluginutils.h"
|
2020-11-20 18:20:36 +00:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include <wrl.h>
|
|
|
|
|
2021-10-13 12:45:34 +00:00
|
|
|
GST_DEBUG_CATEGORY_EXTERN (gst_d3d11_screen_capture_debug);
|
|
|
|
#define GST_CAT_DEFAULT gst_d3d11_screen_capture_debug
|
2020-11-20 18:20:36 +00:00
|
|
|
|
2021-09-10 15:43:26 +00:00
|
|
|
/* *INDENT-OFF* */
|
|
|
|
using namespace Microsoft::WRL;
|
2021-03-14 04:01:37 +00:00
|
|
|
/* *INDENT-ON* */
|
2020-11-20 18:20:36 +00:00
|
|
|
|
2021-10-13 12:45:34 +00:00
|
|
|
#define gst_d3d11_screen_capture_parent_class parent_class
|
2022-10-06 14:54:14 +00:00
|
|
|
G_DEFINE_ABSTRACT_TYPE (GstD3D11ScreenCapture, gst_d3d11_screen_capture,
|
2021-10-13 12:45:34 +00:00
|
|
|
GST_TYPE_OBJECT);
|
2020-11-20 18:20:36 +00:00
|
|
|
|
|
|
|
static void
|
2021-10-13 12:45:34 +00:00
|
|
|
gst_d3d11_screen_capture_class_init (GstD3D11ScreenCaptureClass * klass)
|
2020-11-20 18:20:36 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2021-10-13 12:45:34 +00:00
|
|
|
gst_d3d11_screen_capture_init (GstD3D11ScreenCapture * self)
|
2020-11-20 18:20:36 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
GstFlowReturn
|
2021-10-13 12:45:34 +00:00
|
|
|
gst_d3d11_screen_capture_prepare (GstD3D11ScreenCapture * capture)
|
2020-11-20 18:20:36 +00:00
|
|
|
{
|
2022-10-06 14:54:14 +00:00
|
|
|
GstD3D11ScreenCaptureClass *klass;
|
2020-11-20 18:20:36 +00:00
|
|
|
|
2021-10-13 12:45:34 +00:00
|
|
|
g_return_val_if_fail (GST_IS_D3D11_SCREEN_CAPTURE (capture), GST_FLOW_ERROR);
|
2020-11-20 18:20:36 +00:00
|
|
|
|
2022-10-06 14:54:14 +00:00
|
|
|
klass = GST_D3D11_SCREEN_CAPTURE_GET_CLASS (capture);
|
|
|
|
g_assert (klass->prepare);
|
2020-11-20 18:20:36 +00:00
|
|
|
|
2022-10-06 14:54:14 +00:00
|
|
|
return klass->prepare (capture);
|
2020-11-20 18:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
2021-10-13 12:45:34 +00:00
|
|
|
gst_d3d11_screen_capture_get_size (GstD3D11ScreenCapture * capture,
|
|
|
|
guint * width, guint * height)
|
2020-11-20 18:20:36 +00:00
|
|
|
{
|
2022-10-06 14:54:14 +00:00
|
|
|
GstD3D11ScreenCaptureClass *klass;
|
|
|
|
|
2021-10-13 12:45:34 +00:00
|
|
|
g_return_val_if_fail (GST_IS_D3D11_SCREEN_CAPTURE (capture), FALSE);
|
2021-05-20 08:03:15 +00:00
|
|
|
g_return_val_if_fail (width != nullptr, FALSE);
|
|
|
|
g_return_val_if_fail (height != nullptr, FALSE);
|
|
|
|
|
2022-10-06 14:54:14 +00:00
|
|
|
klass = GST_D3D11_SCREEN_CAPTURE_GET_CLASS (capture);
|
|
|
|
g_assert (klass->get_size);
|
2020-11-20 18:20:36 +00:00
|
|
|
|
2022-10-06 14:54:14 +00:00
|
|
|
return klass->get_size (capture, width, height);
|
2020-11-20 18:20:36 +00:00
|
|
|
}
|
|
|
|
|
2022-07-07 18:38:30 +00:00
|
|
|
gboolean
|
|
|
|
gst_d3d11_screen_capture_get_colorimetry (GstD3D11ScreenCapture * capture,
|
|
|
|
GstVideoColorimetry * colorimetry)
|
|
|
|
{
|
2022-10-06 14:54:14 +00:00
|
|
|
GstD3D11ScreenCaptureClass *klass;
|
2022-07-07 18:38:30 +00:00
|
|
|
|
|
|
|
g_return_val_if_fail (GST_IS_D3D11_SCREEN_CAPTURE (capture), FALSE);
|
|
|
|
g_return_val_if_fail (colorimetry != nullptr, FALSE);
|
|
|
|
|
2022-10-06 14:54:14 +00:00
|
|
|
klass = GST_D3D11_SCREEN_CAPTURE_GET_CLASS (capture);
|
|
|
|
g_assert (klass->get_colorimetry);
|
2022-07-07 18:38:30 +00:00
|
|
|
|
2022-10-06 14:54:14 +00:00
|
|
|
return klass->get_colorimetry (capture, colorimetry);
|
2022-07-07 18:38:30 +00:00
|
|
|
}
|
|
|
|
|
2022-10-06 19:20:29 +00:00
|
|
|
gboolean
|
|
|
|
gst_d3d11_screen_capture_unlock (GstD3D11ScreenCapture * capture)
|
|
|
|
{
|
|
|
|
GstD3D11ScreenCaptureClass *klass;
|
|
|
|
|
|
|
|
g_return_val_if_fail (GST_IS_D3D11_SCREEN_CAPTURE (capture), FALSE);
|
|
|
|
klass = GST_D3D11_SCREEN_CAPTURE_GET_CLASS (capture);
|
|
|
|
|
|
|
|
if (klass->unlock)
|
|
|
|
return klass->unlock (capture);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
gst_d3d11_screen_capture_unlock_stop (GstD3D11ScreenCapture * capture)
|
|
|
|
{
|
|
|
|
GstD3D11ScreenCaptureClass *klass;
|
|
|
|
|
|
|
|
g_return_val_if_fail (GST_IS_D3D11_SCREEN_CAPTURE (capture), FALSE);
|
|
|
|
klass = GST_D3D11_SCREEN_CAPTURE_GET_CLASS (capture);
|
|
|
|
|
|
|
|
if (klass->unlock_stop)
|
|
|
|
return klass->unlock_stop (capture);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
gst_d3d11_screen_capture_show_border (GstD3D11ScreenCapture * capture,
|
|
|
|
gboolean show)
|
|
|
|
{
|
|
|
|
GstD3D11ScreenCaptureClass *klass;
|
|
|
|
|
|
|
|
g_return_if_fail (GST_IS_D3D11_SCREEN_CAPTURE (capture));
|
|
|
|
klass = GST_D3D11_SCREEN_CAPTURE_GET_CLASS (capture);
|
|
|
|
|
|
|
|
if (klass->show_border)
|
|
|
|
klass->show_border (capture, show);
|
|
|
|
}
|
|
|
|
|
2020-11-20 18:20:36 +00:00
|
|
|
GstFlowReturn
|
2021-10-13 12:45:34 +00:00
|
|
|
gst_d3d11_screen_capture_do_capture (GstD3D11ScreenCapture * capture,
|
2022-05-04 17:16:54 +00:00
|
|
|
GstD3D11Device * device, ID3D11Texture2D * texture,
|
|
|
|
ID3D11RenderTargetView * rtv, ID3D11VertexShader * vs,
|
|
|
|
ID3D11PixelShader * ps, ID3D11InputLayout * layout,
|
|
|
|
ID3D11SamplerState * sampler, ID3D11BlendState * blend,
|
2022-03-10 16:37:26 +00:00
|
|
|
D3D11_BOX * crop_box, gboolean draw_mouse)
|
2020-11-20 18:20:36 +00:00
|
|
|
{
|
2022-10-06 14:54:14 +00:00
|
|
|
GstD3D11ScreenCaptureClass *klass;
|
2020-11-20 18:20:36 +00:00
|
|
|
|
2021-10-13 12:45:34 +00:00
|
|
|
g_return_val_if_fail (GST_IS_D3D11_SCREEN_CAPTURE (capture), GST_FLOW_ERROR);
|
2020-11-20 18:20:36 +00:00
|
|
|
g_return_val_if_fail (texture != nullptr, GST_FLOW_ERROR);
|
|
|
|
|
2022-10-06 14:54:14 +00:00
|
|
|
klass = GST_D3D11_SCREEN_CAPTURE_GET_CLASS (capture);
|
|
|
|
g_assert (klass->do_capture);
|
2020-11-20 18:20:36 +00:00
|
|
|
|
2022-10-06 14:54:14 +00:00
|
|
|
return klass->do_capture (capture, device, texture, rtv,
|
|
|
|
vs, ps, layout, sampler, blend, crop_box, draw_mouse);
|
2020-11-20 18:20:36 +00:00
|
|
|
}
|
2021-10-05 12:49:38 +00:00
|
|
|
|
|
|
|
HRESULT
|
2021-10-13 12:45:34 +00:00
|
|
|
gst_d3d11_screen_capture_find_output_for_monitor (HMONITOR monitor,
|
2021-10-05 12:49:38 +00:00
|
|
|
IDXGIAdapter1 ** adapter, IDXGIOutput ** output)
|
|
|
|
{
|
|
|
|
ComPtr < IDXGIFactory1 > factory;
|
|
|
|
HRESULT hr = S_OK;
|
|
|
|
|
|
|
|
g_return_val_if_fail (monitor != nullptr, E_INVALIDARG);
|
|
|
|
|
|
|
|
hr = CreateDXGIFactory1 (IID_PPV_ARGS (&factory));
|
|
|
|
if (FAILED (hr))
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
for (UINT adapter_idx = 0;; adapter_idx++) {
|
|
|
|
ComPtr < IDXGIAdapter1 > adapter_tmp;
|
|
|
|
|
|
|
|
hr = factory->EnumAdapters1 (adapter_idx, &adapter_tmp);
|
|
|
|
if (FAILED (hr))
|
|
|
|
break;
|
|
|
|
|
|
|
|
for (UINT output_idx = 0;; output_idx++) {
|
|
|
|
ComPtr < IDXGIOutput > output_tmp;
|
|
|
|
DXGI_OUTPUT_DESC desc;
|
|
|
|
|
|
|
|
hr = adapter_tmp->EnumOutputs (output_idx, &output_tmp);
|
|
|
|
if (FAILED (hr))
|
|
|
|
break;
|
|
|
|
|
|
|
|
hr = output_tmp->GetDesc (&desc);
|
|
|
|
if (FAILED (hr))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (desc.Monitor == monitor) {
|
|
|
|
if (adapter)
|
|
|
|
*adapter = adapter_tmp.Detach ();
|
|
|
|
if (output)
|
|
|
|
*output = output_tmp.Detach ();
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return E_FAIL;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT
|
2021-10-13 12:45:34 +00:00
|
|
|
gst_d3d11_screen_capture_find_primary_monitor (HMONITOR * monitor,
|
2021-10-05 12:49:38 +00:00
|
|
|
IDXGIAdapter1 ** adapter, IDXGIOutput ** output)
|
|
|
|
{
|
|
|
|
ComPtr < IDXGIFactory1 > factory;
|
|
|
|
HRESULT hr = S_OK;
|
|
|
|
|
|
|
|
hr = CreateDXGIFactory1 (IID_PPV_ARGS (&factory));
|
|
|
|
if (FAILED (hr))
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
for (UINT adapter_idx = 0;; adapter_idx++) {
|
|
|
|
ComPtr < IDXGIAdapter1 > adapter_tmp;
|
|
|
|
|
|
|
|
hr = factory->EnumAdapters1 (adapter_idx, &adapter_tmp);
|
|
|
|
if (FAILED (hr))
|
|
|
|
break;
|
|
|
|
|
|
|
|
for (UINT output_idx = 0;; output_idx++) {
|
|
|
|
ComPtr < IDXGIOutput > output_tmp;
|
|
|
|
DXGI_OUTPUT_DESC desc;
|
|
|
|
MONITORINFOEXW minfo;
|
|
|
|
|
|
|
|
hr = adapter_tmp->EnumOutputs (output_idx, &output_tmp);
|
|
|
|
if (FAILED (hr))
|
|
|
|
break;
|
|
|
|
|
|
|
|
hr = output_tmp->GetDesc (&desc);
|
|
|
|
if (FAILED (hr))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
minfo.cbSize = sizeof (MONITORINFOEXW);
|
|
|
|
if (!GetMonitorInfoW (desc.Monitor, &minfo))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if ((minfo.dwFlags & MONITORINFOF_PRIMARY) != 0) {
|
|
|
|
if (monitor)
|
|
|
|
*monitor = desc.Monitor;
|
|
|
|
if (adapter)
|
|
|
|
*adapter = adapter_tmp.Detach ();
|
|
|
|
if (output)
|
|
|
|
*output = output_tmp.Detach ();
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return E_FAIL;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT
|
2021-10-13 12:45:34 +00:00
|
|
|
gst_d3d11_screen_capture_find_nth_monitor (guint index, HMONITOR * monitor,
|
2021-10-05 12:49:38 +00:00
|
|
|
IDXGIAdapter1 ** adapter, IDXGIOutput ** output)
|
|
|
|
{
|
|
|
|
ComPtr < IDXGIFactory1 > factory;
|
|
|
|
HRESULT hr = S_OK;
|
|
|
|
guint num_found = 0;
|
|
|
|
|
|
|
|
hr = CreateDXGIFactory1 (IID_PPV_ARGS (&factory));
|
|
|
|
if (FAILED (hr))
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
for (UINT adapter_idx = 0;; adapter_idx++) {
|
|
|
|
ComPtr < IDXGIAdapter1 > adapter_tmp;
|
|
|
|
|
|
|
|
hr = factory->EnumAdapters1 (adapter_idx, &adapter_tmp);
|
|
|
|
if (FAILED (hr))
|
|
|
|
break;
|
|
|
|
|
|
|
|
for (UINT output_idx = 0;; output_idx++) {
|
|
|
|
ComPtr < IDXGIOutput > output_tmp;
|
|
|
|
DXGI_OUTPUT_DESC desc;
|
|
|
|
MONITORINFOEXW minfo;
|
|
|
|
|
|
|
|
hr = adapter_tmp->EnumOutputs (output_idx, &output_tmp);
|
|
|
|
if (FAILED (hr))
|
|
|
|
break;
|
|
|
|
|
|
|
|
hr = output_tmp->GetDesc (&desc);
|
|
|
|
if (FAILED (hr))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
minfo.cbSize = sizeof (MONITORINFOEXW);
|
|
|
|
if (!GetMonitorInfoW (desc.Monitor, &minfo))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (num_found == index) {
|
|
|
|
if (monitor)
|
|
|
|
*monitor = desc.Monitor;
|
|
|
|
if (adapter)
|
|
|
|
*adapter = adapter_tmp.Detach ();
|
|
|
|
if (output)
|
|
|
|
*output = output_tmp.Detach ();
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
num_found++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return E_FAIL;
|
|
|
|
}
|