2014-12-29 00:29:55 -08:00
|
|
|
/******************************************************************************
|
|
|
|
Copyright (C) 2013 by Hugh Bailey <obs.jim@gmail.com>
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program 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 General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
******************************************************************************/
|
|
|
|
|
|
|
|
#include "d3d11-subsystem.hpp"
|
2019-10-10 19:13:10 -07:00
|
|
|
#include <unordered_map>
|
2014-12-29 00:29:55 -08:00
|
|
|
|
|
|
|
static inline bool get_monitor(gs_device_t *device, int monitor_idx,
|
|
|
|
IDXGIOutput **dxgiOutput)
|
|
|
|
{
|
|
|
|
HRESULT hr;
|
|
|
|
|
2016-11-03 06:39:24 -07:00
|
|
|
hr = device->adapter->EnumOutputs(monitor_idx, dxgiOutput);
|
2014-12-29 00:29:55 -08:00
|
|
|
if (FAILED(hr)) {
|
|
|
|
if (hr == DXGI_ERROR_NOT_FOUND)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
throw HRError("Failed to get output", hr);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-11-03 06:59:56 -07:00
|
|
|
void gs_duplicator::Start()
|
2014-12-29 00:29:55 -08:00
|
|
|
{
|
2021-12-20 22:12:24 -08:00
|
|
|
ComPtr<IDXGIOutput5> output5;
|
2014-12-29 00:29:55 -08:00
|
|
|
ComPtr<IDXGIOutput1> output1;
|
|
|
|
ComPtr<IDXGIOutput> output;
|
|
|
|
HRESULT hr;
|
|
|
|
|
2016-11-03 06:59:56 -07:00
|
|
|
if (!get_monitor(device, idx, output.Assign()))
|
2014-12-29 00:29:55 -08:00
|
|
|
throw "Invalid monitor index";
|
|
|
|
|
2021-12-20 22:12:24 -08:00
|
|
|
hr = output->QueryInterface(IID_PPV_ARGS(output5.Assign()));
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
|
|
constexpr DXGI_FORMAT supportedFormats[]{
|
|
|
|
DXGI_FORMAT_R16G16B16A16_FLOAT,
|
|
|
|
DXGI_FORMAT_B8G8R8A8_UNORM,
|
|
|
|
};
|
|
|
|
hr = output5->DuplicateOutput1(device->device, 0,
|
|
|
|
_countof(supportedFormats),
|
|
|
|
supportedFormats,
|
|
|
|
duplicator.Assign());
|
|
|
|
if (FAILED(hr))
|
|
|
|
throw HRError("Failed to DuplicateOutput1", hr);
|
|
|
|
} else {
|
|
|
|
hr = output->QueryInterface(IID_PPV_ARGS(output1.Assign()));
|
|
|
|
if (FAILED(hr))
|
|
|
|
throw HRError("Failed to query IDXGIOutput1", hr);
|
2014-12-29 00:29:55 -08:00
|
|
|
|
2021-12-20 22:12:24 -08:00
|
|
|
hr = output1->DuplicateOutput(device->device,
|
|
|
|
duplicator.Assign());
|
|
|
|
if (FAILED(hr))
|
|
|
|
throw HRError("Failed to DuplicateOutput", hr);
|
|
|
|
}
|
2014-12-29 00:29:55 -08:00
|
|
|
}
|
|
|
|
|
2016-11-03 06:59:56 -07:00
|
|
|
gs_duplicator::gs_duplicator(gs_device_t *device_, int monitor_idx)
|
2016-11-03 07:37:05 -07:00
|
|
|
: gs_obj(device_, gs_type::gs_duplicator),
|
|
|
|
texture(nullptr),
|
2018-01-09 22:25:29 +01:00
|
|
|
idx(monitor_idx),
|
|
|
|
refs(1),
|
|
|
|
updated(false)
|
2016-11-03 06:59:56 -07:00
|
|
|
{
|
|
|
|
Start();
|
|
|
|
}
|
|
|
|
|
2014-12-29 00:29:55 -08:00
|
|
|
gs_duplicator::~gs_duplicator()
|
|
|
|
{
|
|
|
|
delete texture;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
|
|
|
|
EXPORT bool device_get_duplicator_monitor_info(gs_device_t *device,
|
|
|
|
int monitor_idx,
|
|
|
|
struct gs_monitor_info *info)
|
|
|
|
{
|
|
|
|
DXGI_OUTPUT_DESC desc;
|
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
try {
|
|
|
|
ComPtr<IDXGIOutput> output;
|
|
|
|
|
|
|
|
if (!get_monitor(device, monitor_idx, output.Assign()))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
hr = output->GetDesc(&desc);
|
|
|
|
if (FAILED(hr))
|
|
|
|
throw HRError("GetDesc failed", hr);
|
|
|
|
|
2019-08-28 21:10:51 -07:00
|
|
|
} catch (const HRError &error) {
|
2014-12-29 00:29:55 -08:00
|
|
|
blog(LOG_ERROR,
|
|
|
|
"device_get_duplicator_monitor_info: "
|
|
|
|
"%s (%08lX)",
|
|
|
|
error.str, error.hr);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (desc.Rotation) {
|
|
|
|
case DXGI_MODE_ROTATION_UNSPECIFIED:
|
|
|
|
case DXGI_MODE_ROTATION_IDENTITY:
|
|
|
|
info->rotation_degrees = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DXGI_MODE_ROTATION_ROTATE90:
|
|
|
|
info->rotation_degrees = 90;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DXGI_MODE_ROTATION_ROTATE180:
|
|
|
|
info->rotation_degrees = 180;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DXGI_MODE_ROTATION_ROTATE270:
|
|
|
|
info->rotation_degrees = 270;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
info->x = desc.DesktopCoordinates.left;
|
|
|
|
info->y = desc.DesktopCoordinates.top;
|
|
|
|
info->cx = desc.DesktopCoordinates.right - info->x;
|
|
|
|
info->cy = desc.DesktopCoordinates.bottom - info->y;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-01-26 16:22:21 -08:00
|
|
|
EXPORT int device_duplicator_get_monitor_index(gs_device_t *device,
|
|
|
|
void *monitor)
|
|
|
|
{
|
|
|
|
const HMONITOR handle = (HMONITOR)monitor;
|
|
|
|
|
|
|
|
int index = -1;
|
|
|
|
|
|
|
|
UINT output = 0;
|
|
|
|
while (index == -1) {
|
|
|
|
IDXGIOutput *pOutput;
|
|
|
|
const HRESULT hr =
|
|
|
|
device->adapter->EnumOutputs(output, &pOutput);
|
|
|
|
if (hr == DXGI_ERROR_NOT_FOUND)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
|
|
DXGI_OUTPUT_DESC desc;
|
|
|
|
if (SUCCEEDED(pOutput->GetDesc(&desc))) {
|
|
|
|
if (desc.Monitor == handle)
|
|
|
|
index = output;
|
|
|
|
} else {
|
|
|
|
blog(LOG_ERROR,
|
|
|
|
"device_duplicator_get_monitor_index: "
|
|
|
|
"Failed to get desc (%08lX)",
|
|
|
|
hr);
|
|
|
|
}
|
|
|
|
|
|
|
|
pOutput->Release();
|
|
|
|
} else if (hr == DXGI_ERROR_NOT_FOUND) {
|
|
|
|
blog(LOG_ERROR,
|
|
|
|
"device_duplicator_get_monitor_index: "
|
|
|
|
"Failed to get output (%08lX)",
|
|
|
|
hr);
|
|
|
|
}
|
|
|
|
|
|
|
|
++output;
|
|
|
|
}
|
|
|
|
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|
2019-10-10 19:13:10 -07:00
|
|
|
static std::unordered_map<int, gs_duplicator *> instances;
|
2018-01-09 22:25:29 +01:00
|
|
|
|
|
|
|
void reset_duplicators(void)
|
|
|
|
{
|
2019-10-10 19:13:10 -07:00
|
|
|
for (std::pair<const int, gs_duplicator *> &pair : instances) {
|
2018-01-09 22:25:29 +01:00
|
|
|
pair.second->updated = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-29 00:29:55 -08:00
|
|
|
EXPORT gs_duplicator_t *device_duplicator_create(gs_device_t *device,
|
|
|
|
int monitor_idx)
|
|
|
|
{
|
|
|
|
gs_duplicator *duplicator = nullptr;
|
|
|
|
|
2019-10-10 19:13:10 -07:00
|
|
|
const auto it = instances.find(monitor_idx);
|
2018-01-09 22:25:29 +01:00
|
|
|
if (it != instances.end()) {
|
|
|
|
duplicator = it->second;
|
|
|
|
duplicator->refs++;
|
|
|
|
return duplicator;
|
|
|
|
}
|
|
|
|
|
2014-12-29 00:29:55 -08:00
|
|
|
try {
|
|
|
|
duplicator = new gs_duplicator(device, monitor_idx);
|
2018-01-09 22:25:29 +01:00
|
|
|
instances[monitor_idx] = duplicator;
|
2014-12-29 00:29:55 -08:00
|
|
|
|
|
|
|
} catch (const char *error) {
|
2015-11-16 11:11:16 -08:00
|
|
|
blog(LOG_DEBUG, "device_duplicator_create: %s", error);
|
2014-12-29 00:29:55 -08:00
|
|
|
return nullptr;
|
|
|
|
|
2019-08-28 21:10:51 -07:00
|
|
|
} catch (const HRError &error) {
|
2015-11-16 11:11:16 -08:00
|
|
|
blog(LOG_DEBUG, "device_duplicator_create: %s (%08lX)",
|
2014-12-29 00:29:55 -08:00
|
|
|
error.str, error.hr);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return duplicator;
|
|
|
|
}
|
|
|
|
|
|
|
|
EXPORT void gs_duplicator_destroy(gs_duplicator_t *duplicator)
|
|
|
|
{
|
2018-01-09 22:25:29 +01:00
|
|
|
if (--duplicator->refs == 0) {
|
|
|
|
instances.erase(duplicator->idx);
|
|
|
|
delete duplicator;
|
|
|
|
}
|
2014-12-29 00:29:55 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void copy_texture(gs_duplicator_t *d, ID3D11Texture2D *tex)
|
|
|
|
{
|
|
|
|
D3D11_TEXTURE2D_DESC desc;
|
|
|
|
tex->GetDesc(&desc);
|
2021-12-20 22:12:24 -08:00
|
|
|
const gs_color_format format = ConvertDXGITextureFormat(desc.Format);
|
|
|
|
const gs_color_format general_format = gs_generalize_format(format);
|
2014-12-29 00:29:55 -08:00
|
|
|
|
2021-12-20 22:12:24 -08:00
|
|
|
if (!d->texture || (d->texture->width != desc.Width) ||
|
|
|
|
(d->texture->height != desc.Height) ||
|
|
|
|
(d->texture->format != general_format)) {
|
2014-12-29 00:29:55 -08:00
|
|
|
|
|
|
|
delete d->texture;
|
|
|
|
d->texture = (gs_texture_2d *)gs_texture_create(
|
2021-12-20 22:12:24 -08:00
|
|
|
desc.Width, desc.Height, general_format, 1, nullptr, 0);
|
2014-12-29 00:29:55 -08:00
|
|
|
}
|
|
|
|
|
2021-12-20 22:12:24 -08:00
|
|
|
if (d->texture)
|
2014-12-29 00:29:55 -08:00
|
|
|
d->device->context->CopyResource(d->texture->texture, tex);
|
|
|
|
}
|
|
|
|
|
|
|
|
EXPORT bool gs_duplicator_update_frame(gs_duplicator_t *d)
|
|
|
|
{
|
|
|
|
DXGI_OUTDUPL_FRAME_INFO info;
|
|
|
|
ComPtr<ID3D11Texture2D> tex;
|
|
|
|
ComPtr<IDXGIResource> res;
|
|
|
|
HRESULT hr;
|
|
|
|
|
2017-11-16 20:33:03 -08:00
|
|
|
if (!d->duplicator) {
|
|
|
|
return false;
|
|
|
|
}
|
2018-01-09 22:25:29 +01:00
|
|
|
if (d->updated) {
|
|
|
|
return true;
|
|
|
|
}
|
2017-11-16 20:33:03 -08:00
|
|
|
|
2014-12-29 00:29:55 -08:00
|
|
|
hr = d->duplicator->AcquireNextFrame(0, &info, res.Assign());
|
|
|
|
if (hr == DXGI_ERROR_ACCESS_LOST) {
|
|
|
|
return false;
|
|
|
|
|
|
|
|
} else if (hr == DXGI_ERROR_WAIT_TIMEOUT) {
|
|
|
|
return true;
|
|
|
|
|
|
|
|
} else if (FAILED(hr)) {
|
|
|
|
blog(LOG_ERROR,
|
|
|
|
"gs_duplicator_update_frame: Failed to update "
|
|
|
|
"frame (%08lX)",
|
|
|
|
hr);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
hr = res->QueryInterface(__uuidof(ID3D11Texture2D),
|
|
|
|
(void **)tex.Assign());
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
blog(LOG_ERROR,
|
|
|
|
"gs_duplicator_update_frame: Failed to query "
|
|
|
|
"ID3D11Texture2D (%08lX)",
|
|
|
|
hr);
|
|
|
|
d->duplicator->ReleaseFrame();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
copy_texture(d, tex);
|
|
|
|
d->duplicator->ReleaseFrame();
|
2018-01-09 22:25:29 +01:00
|
|
|
d->updated = true;
|
2014-12-29 00:29:55 -08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
EXPORT gs_texture_t *gs_duplicator_get_texture(gs_duplicator_t *duplicator)
|
|
|
|
{
|
|
|
|
return duplicator->texture;
|
|
|
|
}
|
|
|
|
}
|