2014-03-24 00:05:38 +01:00
|
|
|
/*
|
|
|
|
* Copyright 2014 Michael Müller for Pipelight
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 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
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser 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
|
|
|
|
*/
|
|
|
|
|
2020-06-22 12:45:29 +02:00
|
|
|
#define COBJMACROS
|
|
|
|
|
2014-03-24 00:05:38 +01:00
|
|
|
#include <stdarg.h>
|
2020-09-17 13:25:35 +02:00
|
|
|
#include <limits.h>
|
2014-03-24 00:05:38 +01:00
|
|
|
#include "windef.h"
|
|
|
|
#include "winbase.h"
|
2014-03-26 05:32:36 +01:00
|
|
|
#include "d3d9.h"
|
|
|
|
#include "physicalmonitorenumerationapi.h"
|
|
|
|
#include "lowlevelmonitorconfigurationapi.h"
|
|
|
|
#include "highlevelmonitorconfigurationapi.h"
|
2020-06-22 12:45:29 +02:00
|
|
|
#include "initguid.h"
|
|
|
|
#include "dxva2api.h"
|
2014-03-24 00:05:38 +01:00
|
|
|
|
|
|
|
#include "wine/debug.h"
|
2020-06-22 12:45:29 +02:00
|
|
|
#include "wine/heap.h"
|
2014-03-24 00:05:38 +01:00
|
|
|
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(dxva2);
|
|
|
|
|
2020-06-22 12:45:30 +02:00
|
|
|
enum device_handle_flags
|
|
|
|
{
|
|
|
|
HANDLE_FLAG_OPEN = 0x1,
|
|
|
|
HANDLE_FLAG_INVALID = 0x2,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct device_handle
|
|
|
|
{
|
|
|
|
unsigned int flags;
|
2020-06-30 11:35:59 +02:00
|
|
|
IDirect3DStateBlock9 *state_block;
|
2020-06-22 12:45:30 +02:00
|
|
|
};
|
|
|
|
|
2020-06-22 12:45:29 +02:00
|
|
|
struct device_manager
|
|
|
|
{
|
|
|
|
IDirect3DDeviceManager9 IDirect3DDeviceManager9_iface;
|
2020-06-22 12:45:31 +02:00
|
|
|
IDirectXVideoProcessorService IDirectXVideoProcessorService_iface;
|
2021-01-18 13:42:11 +01:00
|
|
|
IDirectXVideoDecoderService IDirectXVideoDecoderService_iface;
|
2020-06-22 12:45:29 +02:00
|
|
|
LONG refcount;
|
|
|
|
|
2020-06-22 12:45:30 +02:00
|
|
|
IDirect3DDevice9 *device;
|
2020-06-22 12:45:29 +02:00
|
|
|
UINT token;
|
2020-06-22 12:45:30 +02:00
|
|
|
|
|
|
|
struct device_handle *handles;
|
|
|
|
size_t count;
|
|
|
|
size_t capacity;
|
|
|
|
|
2020-06-30 11:35:59 +02:00
|
|
|
HANDLE locking_handle;
|
|
|
|
|
2020-06-22 12:45:30 +02:00
|
|
|
CRITICAL_SECTION cs;
|
2020-06-30 11:35:59 +02:00
|
|
|
CONDITION_VARIABLE lock;
|
2020-06-22 12:45:29 +02:00
|
|
|
};
|
|
|
|
|
2020-10-19 16:10:40 +02:00
|
|
|
struct video_processor
|
|
|
|
{
|
|
|
|
IDirectXVideoProcessor IDirectXVideoProcessor_iface;
|
|
|
|
LONG refcount;
|
|
|
|
|
|
|
|
IDirectXVideoProcessorService *service;
|
2020-10-19 16:10:41 +02:00
|
|
|
GUID device;
|
|
|
|
DXVA2_VideoDesc video_desc;
|
|
|
|
D3DFORMAT rt_format;
|
|
|
|
unsigned int max_substreams;
|
2020-10-19 16:10:40 +02:00
|
|
|
};
|
|
|
|
|
2020-06-22 12:45:30 +02:00
|
|
|
static BOOL dxva_array_reserve(void **elements, size_t *capacity, size_t count, size_t size)
|
|
|
|
{
|
|
|
|
size_t new_capacity, max_capacity;
|
|
|
|
void *new_elements;
|
|
|
|
|
|
|
|
if (count <= *capacity)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
max_capacity = ~(SIZE_T)0 / size;
|
|
|
|
if (count > max_capacity)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
new_capacity = max(4, *capacity);
|
|
|
|
while (new_capacity < count && new_capacity <= max_capacity / 2)
|
|
|
|
new_capacity *= 2;
|
|
|
|
if (new_capacity < count)
|
|
|
|
new_capacity = max_capacity;
|
|
|
|
|
|
|
|
if (!(new_elements = heap_realloc(*elements, new_capacity * size)))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
*elements = new_elements;
|
|
|
|
*capacity = new_capacity;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2020-06-22 12:45:29 +02:00
|
|
|
static struct device_manager *impl_from_IDirect3DDeviceManager9(IDirect3DDeviceManager9 *iface)
|
|
|
|
{
|
|
|
|
return CONTAINING_RECORD(iface, struct device_manager, IDirect3DDeviceManager9_iface);
|
|
|
|
}
|
|
|
|
|
2020-06-22 12:45:31 +02:00
|
|
|
static struct device_manager *impl_from_IDirectXVideoProcessorService(IDirectXVideoProcessorService *iface)
|
|
|
|
{
|
|
|
|
return CONTAINING_RECORD(iface, struct device_manager, IDirectXVideoProcessorService_iface);
|
|
|
|
}
|
|
|
|
|
2021-01-18 13:42:11 +01:00
|
|
|
static struct device_manager *impl_from_IDirectXVideoDecoderService(IDirectXVideoDecoderService *iface)
|
|
|
|
{
|
|
|
|
return CONTAINING_RECORD(iface, struct device_manager, IDirectXVideoDecoderService_iface);
|
|
|
|
}
|
|
|
|
|
2020-10-19 16:10:40 +02:00
|
|
|
static struct video_processor *impl_from_IDirectXVideoProcessor(IDirectXVideoProcessor *iface)
|
|
|
|
{
|
|
|
|
return CONTAINING_RECORD(iface, struct video_processor, IDirectXVideoProcessor_iface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI video_processor_QueryInterface(IDirectXVideoProcessor *iface, REFIID riid, void **obj)
|
|
|
|
{
|
|
|
|
if (IsEqualIID(riid, &IID_IDirectXVideoProcessor) ||
|
|
|
|
IsEqualIID(riid, &IID_IUnknown))
|
|
|
|
{
|
|
|
|
*obj = iface;
|
|
|
|
IDirectXVideoProcessor_AddRef(iface);
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
WARN("Unsupported interface %s.\n", debugstr_guid(riid));
|
|
|
|
*obj = NULL;
|
|
|
|
return E_NOINTERFACE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI video_processor_AddRef(IDirectXVideoProcessor *iface)
|
|
|
|
{
|
|
|
|
struct video_processor *processor = impl_from_IDirectXVideoProcessor(iface);
|
|
|
|
ULONG refcount = InterlockedIncrement(&processor->refcount);
|
|
|
|
|
|
|
|
TRACE("%p, refcount %u.\n", iface, refcount);
|
|
|
|
|
|
|
|
return refcount;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI video_processor_Release(IDirectXVideoProcessor *iface)
|
|
|
|
{
|
|
|
|
struct video_processor *processor = impl_from_IDirectXVideoProcessor(iface);
|
|
|
|
ULONG refcount = InterlockedDecrement(&processor->refcount);
|
|
|
|
|
|
|
|
TRACE("%p, refcount %u.\n", iface, refcount);
|
|
|
|
|
|
|
|
if (!refcount)
|
|
|
|
{
|
|
|
|
IDirectXVideoProcessorService_Release(processor->service);
|
|
|
|
heap_free(processor);
|
|
|
|
}
|
|
|
|
|
|
|
|
return refcount;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI video_processor_GetVideoProcessorService(IDirectXVideoProcessor *iface,
|
|
|
|
IDirectXVideoProcessorService **service)
|
|
|
|
{
|
|
|
|
struct video_processor *processor = impl_from_IDirectXVideoProcessor(iface);
|
|
|
|
|
|
|
|
TRACE("%p, %p.\n", iface, service);
|
|
|
|
|
|
|
|
*service = processor->service;
|
|
|
|
IDirectXVideoProcessorService_AddRef(*service);
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI video_processor_GetCreationParameters(IDirectXVideoProcessor *iface,
|
2020-10-19 16:10:41 +02:00
|
|
|
GUID *device, DXVA2_VideoDesc *video_desc, D3DFORMAT *rt_format, UINT *max_substreams)
|
2020-10-19 16:10:40 +02:00
|
|
|
{
|
2020-10-19 16:10:41 +02:00
|
|
|
struct video_processor *processor = impl_from_IDirectXVideoProcessor(iface);
|
|
|
|
|
|
|
|
TRACE("%p, %p, %p, %p, %p.\n", iface, device, video_desc, rt_format, max_substreams);
|
2020-10-19 16:10:40 +02:00
|
|
|
|
2020-10-19 16:10:41 +02:00
|
|
|
if (!device && !video_desc && !rt_format && !max_substreams)
|
2020-10-19 16:10:40 +02:00
|
|
|
return E_INVALIDARG;
|
|
|
|
|
2020-10-19 16:10:41 +02:00
|
|
|
if (device)
|
|
|
|
*device = processor->device;
|
|
|
|
if (video_desc)
|
|
|
|
*video_desc = processor->video_desc;
|
|
|
|
if (rt_format)
|
|
|
|
*rt_format = processor->rt_format;
|
|
|
|
if (max_substreams)
|
|
|
|
*max_substreams = processor->max_substreams;
|
|
|
|
|
|
|
|
return S_OK;
|
2020-10-19 16:10:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI video_processor_GetVideoProcessorCaps(IDirectXVideoProcessor *iface,
|
|
|
|
DXVA2_VideoProcessorCaps *caps)
|
|
|
|
{
|
|
|
|
FIXME("%p, %p.\n", iface, caps);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI video_processor_GetProcAmpRange(IDirectXVideoProcessor *iface, UINT cap, DXVA2_ValueRange *range)
|
|
|
|
{
|
|
|
|
FIXME("%p, %u, %p.\n", iface, cap, range);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI video_processor_GetFilterPropertyRange(IDirectXVideoProcessor *iface, UINT setting,
|
|
|
|
DXVA2_ValueRange *range)
|
|
|
|
{
|
|
|
|
FIXME("%p, %u, %p.\n", iface, setting, range);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
2020-11-13 13:30:03 +01:00
|
|
|
static BOOL intersect_rect(RECT *dest, const RECT *src1, const RECT *src2)
|
|
|
|
{
|
|
|
|
if (IsRectEmpty(src1) || IsRectEmpty(src2) ||
|
|
|
|
(src1->left >= src2->right) || (src2->left >= src1->right) ||
|
|
|
|
(src1->top >= src2->bottom) || (src2->top >= src1->bottom))
|
|
|
|
{
|
|
|
|
SetRectEmpty(dest);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
dest->left = max(src1->left, src2->left);
|
|
|
|
dest->right = min(src1->right, src2->right);
|
|
|
|
dest->top = max(src1->top, src2->top);
|
|
|
|
dest->bottom = min(src1->bottom, src2->bottom);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2020-10-19 16:10:40 +02:00
|
|
|
static HRESULT WINAPI video_processor_VideoProcessBlt(IDirectXVideoProcessor *iface, IDirect3DSurface9 *rt,
|
|
|
|
const DXVA2_VideoProcessBltParams *params, const DXVA2_VideoSample *samples, UINT sample_count,
|
|
|
|
HANDLE *complete_handle)
|
|
|
|
{
|
2020-11-13 13:30:03 +01:00
|
|
|
IDirect3DDevice9 *device;
|
|
|
|
unsigned int i;
|
|
|
|
RECT dst_rect;
|
|
|
|
HRESULT hr;
|
2020-10-19 16:10:40 +02:00
|
|
|
|
2020-11-13 13:30:03 +01:00
|
|
|
TRACE("%p, %p, %p, %p, %u, %p.\n", iface, rt, params, samples, sample_count, complete_handle);
|
|
|
|
|
|
|
|
if (FAILED(hr = IDirect3DSurface9_GetDevice(rt, &device)))
|
|
|
|
{
|
|
|
|
WARN("Failed to get surface device, hr %#x.\n", hr);
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: use specified color */
|
|
|
|
IDirect3DDevice9_ColorFill(device, rt, NULL, 0);
|
|
|
|
|
|
|
|
for (i = 0; i < sample_count; ++i)
|
|
|
|
{
|
|
|
|
dst_rect = params->TargetRect;
|
|
|
|
|
|
|
|
if (!intersect_rect(&dst_rect, &dst_rect, &samples[i].DstRect))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (FAILED(hr = IDirect3DDevice9_StretchRect(device, samples[i].SrcSurface, &samples[i].SrcRect,
|
|
|
|
rt, &dst_rect, D3DTEXF_POINT)))
|
|
|
|
{
|
|
|
|
WARN("Failed to copy sample %u, hr %#x.\n", i, hr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
IDirect3DDevice9_Release(device);
|
|
|
|
|
|
|
|
return S_OK;
|
2020-10-19 16:10:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static const IDirectXVideoProcessorVtbl video_processor_vtbl =
|
|
|
|
{
|
|
|
|
video_processor_QueryInterface,
|
|
|
|
video_processor_AddRef,
|
|
|
|
video_processor_Release,
|
|
|
|
video_processor_GetVideoProcessorService,
|
|
|
|
video_processor_GetCreationParameters,
|
|
|
|
video_processor_GetVideoProcessorCaps,
|
|
|
|
video_processor_GetProcAmpRange,
|
|
|
|
video_processor_GetFilterPropertyRange,
|
|
|
|
video_processor_VideoProcessBlt,
|
|
|
|
};
|
|
|
|
|
2020-06-22 12:45:31 +02:00
|
|
|
static HRESULT WINAPI device_manager_processor_service_QueryInterface(IDirectXVideoProcessorService *iface,
|
|
|
|
REFIID riid, void **obj)
|
|
|
|
{
|
2021-01-18 13:42:11 +01:00
|
|
|
struct device_manager *manager = impl_from_IDirectXVideoProcessorService(iface);
|
|
|
|
|
2020-06-22 12:45:31 +02:00
|
|
|
if (IsEqualIID(riid, &IID_IDirectXVideoProcessorService) ||
|
2020-06-23 13:31:17 +02:00
|
|
|
IsEqualIID(riid, &IID_IDirectXVideoAccelerationService) ||
|
2020-06-22 12:45:31 +02:00
|
|
|
IsEqualIID(riid, &IID_IUnknown))
|
|
|
|
{
|
|
|
|
*obj = iface;
|
2021-01-18 13:42:11 +01:00
|
|
|
}
|
|
|
|
else if (IsEqualIID(riid, &IID_IDirectXVideoDecoderService))
|
|
|
|
{
|
|
|
|
*obj = &manager->IDirectXVideoDecoderService_iface;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WARN("Unsupported interface %s.\n", debugstr_guid(riid));
|
|
|
|
*obj = NULL;
|
|
|
|
return E_NOINTERFACE;
|
2020-06-22 12:45:31 +02:00
|
|
|
}
|
|
|
|
|
2021-01-18 13:42:11 +01:00
|
|
|
IUnknown_AddRef((IUnknown *)*obj);
|
|
|
|
return S_OK;
|
2020-06-22 12:45:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI device_manager_processor_service_AddRef(IDirectXVideoProcessorService *iface)
|
|
|
|
{
|
|
|
|
struct device_manager *manager = impl_from_IDirectXVideoProcessorService(iface);
|
|
|
|
return IDirect3DDeviceManager9_AddRef(&manager->IDirect3DDeviceManager9_iface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI device_manager_processor_service_Release(IDirectXVideoProcessorService *iface)
|
|
|
|
{
|
|
|
|
struct device_manager *manager = impl_from_IDirectXVideoProcessorService(iface);
|
|
|
|
return IDirect3DDeviceManager9_Release(&manager->IDirect3DDeviceManager9_iface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_processor_service_CreateSurface(IDirectXVideoProcessorService *iface,
|
|
|
|
UINT width, UINT height, UINT backbuffers, D3DFORMAT format, D3DPOOL pool, DWORD usage, DWORD dxvaType,
|
2020-09-17 13:25:35 +02:00
|
|
|
IDirect3DSurface9 **surfaces, HANDLE *shared_handle)
|
2020-06-22 12:45:31 +02:00
|
|
|
{
|
2020-09-17 13:25:35 +02:00
|
|
|
struct device_manager *manager = impl_from_IDirectXVideoProcessorService(iface);
|
|
|
|
unsigned int i, j;
|
|
|
|
HRESULT hr;
|
2020-06-22 12:45:31 +02:00
|
|
|
|
2020-09-17 13:25:35 +02:00
|
|
|
TRACE("%p, %u, %u, %u, %u, %u, %u, %u, %p, %p.\n", iface, width, height, backbuffers, format, pool, usage, dxvaType,
|
|
|
|
surfaces, shared_handle);
|
|
|
|
|
|
|
|
if (backbuffers >= UINT_MAX)
|
|
|
|
return E_INVALIDARG;
|
|
|
|
|
|
|
|
memset(surfaces, 0, (backbuffers + 1) * sizeof(*surfaces));
|
|
|
|
|
|
|
|
for (i = 0; i < backbuffers + 1; ++i)
|
|
|
|
{
|
|
|
|
if (FAILED(hr = IDirect3DDevice9_CreateOffscreenPlainSurface(manager->device, width, height, format,
|
|
|
|
pool, &surfaces[i], NULL)))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FAILED(hr))
|
|
|
|
{
|
|
|
|
for (j = 0; j < i; ++j)
|
|
|
|
{
|
|
|
|
if (surfaces[j])
|
|
|
|
{
|
|
|
|
IDirect3DSurface9_Release(surfaces[j]);
|
|
|
|
surfaces[j] = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return hr;
|
2020-06-22 12:45:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_processor_service_RegisterVideoProcessorSoftwareDevice(
|
|
|
|
IDirectXVideoProcessorService *iface, void *callbacks)
|
|
|
|
{
|
|
|
|
FIXME("%p, %p.\n", iface, callbacks);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_processor_service_GetVideoProcessorDeviceGuids(
|
|
|
|
IDirectXVideoProcessorService *iface, const DXVA2_VideoDesc *video_desc, UINT *count, GUID **guids)
|
|
|
|
{
|
2020-09-17 13:25:36 +02:00
|
|
|
FIXME("%p, %p, %p, %p semi-stub.\n", iface, video_desc, count, guids);
|
2020-06-22 12:45:31 +02:00
|
|
|
|
2020-09-17 13:25:36 +02:00
|
|
|
if (!(*guids = CoTaskMemAlloc(sizeof(**guids))))
|
|
|
|
return E_OUTOFMEMORY;
|
|
|
|
|
|
|
|
memcpy(*guids, &DXVA2_VideoProcSoftwareDevice, sizeof(**guids));
|
|
|
|
*count = 1;
|
|
|
|
|
|
|
|
return S_OK;
|
2020-06-22 12:45:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_processor_service_GetVideoProcessorRenderTargets(
|
|
|
|
IDirectXVideoProcessorService *iface, REFGUID deviceguid, const DXVA2_VideoDesc *video_desc, UINT *count,
|
|
|
|
D3DFORMAT **formats)
|
|
|
|
{
|
2020-09-18 16:25:15 +02:00
|
|
|
TRACE("%p, %s, %p, %p, %p.\n", iface, debugstr_guid(deviceguid), video_desc, count, formats);
|
|
|
|
|
|
|
|
if (IsEqualGUID(deviceguid, &DXVA2_VideoProcSoftwareDevice))
|
|
|
|
{
|
2021-04-08 11:19:38 +02:00
|
|
|
if (!(video_desc->Format == D3DFMT_A8R8G8B8 ||
|
|
|
|
video_desc->Format == D3DFMT_X8R8G8B8 ||
|
|
|
|
video_desc->Format == D3DFMT_YUY2))
|
|
|
|
{
|
|
|
|
WARN("Unsupported content format %#x.\n", video_desc->Format);
|
|
|
|
return E_FAIL;
|
|
|
|
}
|
2020-09-18 16:25:15 +02:00
|
|
|
|
|
|
|
if (!(*formats = CoTaskMemAlloc(2 * sizeof(**formats))))
|
|
|
|
return E_OUTOFMEMORY;
|
|
|
|
|
|
|
|
*count = 2;
|
|
|
|
(*formats)[0] = D3DFMT_X8R8G8B8;
|
|
|
|
(*formats)[1] = D3DFMT_A8R8G8B8;
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
FIXME("Unsupported device %s.\n", debugstr_guid(deviceguid));
|
2020-06-22 12:45:31 +02:00
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_processor_service_GetVideoProcessorSubStreamFormats(
|
|
|
|
IDirectXVideoProcessorService *iface, REFGUID deviceguid, const DXVA2_VideoDesc *video_desc,
|
|
|
|
D3DFORMAT rt_format, UINT *count, D3DFORMAT **formats)
|
|
|
|
{
|
|
|
|
FIXME("%p, %s, %p, %u, %p, %p.\n", iface, debugstr_guid(deviceguid), video_desc, rt_format, count, formats);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_processor_service_GetVideoProcessorCaps(
|
|
|
|
IDirectXVideoProcessorService *iface, REFGUID deviceguid, const DXVA2_VideoDesc *video_desc,
|
|
|
|
D3DFORMAT rt_format, DXVA2_VideoProcessorCaps *caps)
|
|
|
|
{
|
|
|
|
FIXME("%p, %s, %p, %u, %p.\n", iface, debugstr_guid(deviceguid), video_desc, rt_format, caps);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_processor_service_GetProcAmpRange(
|
|
|
|
IDirectXVideoProcessorService *iface, REFGUID deviceguid, const DXVA2_VideoDesc *video_desc,
|
|
|
|
D3DFORMAT rt_format, UINT ProcAmpCap, DXVA2_ValueRange *range)
|
|
|
|
{
|
|
|
|
FIXME("%p, %s, %p, %u, %u, %p.\n", iface, debugstr_guid(deviceguid), video_desc, rt_format, ProcAmpCap, range);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
2020-06-26 17:40:55 +02:00
|
|
|
static HRESULT WINAPI device_manager_processor_service_GetFilterPropertyRange(
|
|
|
|
IDirectXVideoProcessorService *iface, REFGUID deviceguid, const DXVA2_VideoDesc *video_desc,
|
|
|
|
D3DFORMAT rt_format, UINT filter_setting, DXVA2_ValueRange *range)
|
|
|
|
{
|
|
|
|
FIXME("%p, %s, %p, %d, %d, %p.\n", iface, debugstr_guid(deviceguid), video_desc, rt_format, filter_setting, range);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_processor_service_CreateVideoProcessor(IDirectXVideoProcessorService *iface,
|
2020-10-19 16:10:41 +02:00
|
|
|
REFGUID device, const DXVA2_VideoDesc *video_desc, D3DFORMAT rt_format, UINT max_substreams,
|
2020-06-26 17:40:55 +02:00
|
|
|
IDirectXVideoProcessor **processor)
|
|
|
|
{
|
2020-10-19 16:10:40 +02:00
|
|
|
struct video_processor *object;
|
|
|
|
|
2020-10-19 16:10:41 +02:00
|
|
|
FIXME("%p, %s, %p, %d, %u, %p.\n", iface, debugstr_guid(device), video_desc, rt_format, max_substreams,
|
2020-06-26 17:40:55 +02:00
|
|
|
processor);
|
|
|
|
|
2020-10-19 16:10:40 +02:00
|
|
|
/* FIXME: validate render target format */
|
|
|
|
|
|
|
|
if (!(object = heap_alloc_zero(sizeof(*object))))
|
|
|
|
return E_OUTOFMEMORY;
|
|
|
|
|
|
|
|
object->IDirectXVideoProcessor_iface.lpVtbl = &video_processor_vtbl;
|
|
|
|
object->refcount = 1;
|
|
|
|
object->service = iface;
|
|
|
|
IDirectXVideoProcessorService_AddRef(object->service);
|
2020-10-19 16:10:41 +02:00
|
|
|
object->device = *device;
|
|
|
|
object->video_desc = *video_desc;
|
|
|
|
object->rt_format = rt_format;
|
|
|
|
object->max_substreams = max_substreams;
|
2020-10-19 16:10:40 +02:00
|
|
|
|
|
|
|
*processor = &object->IDirectXVideoProcessor_iface;
|
|
|
|
|
|
|
|
return S_OK;
|
2020-06-26 17:40:55 +02:00
|
|
|
}
|
|
|
|
|
2020-06-22 12:45:31 +02:00
|
|
|
static const IDirectXVideoProcessorServiceVtbl device_manager_processor_service_vtbl =
|
|
|
|
{
|
|
|
|
device_manager_processor_service_QueryInterface,
|
|
|
|
device_manager_processor_service_AddRef,
|
|
|
|
device_manager_processor_service_Release,
|
|
|
|
device_manager_processor_service_CreateSurface,
|
|
|
|
device_manager_processor_service_RegisterVideoProcessorSoftwareDevice,
|
|
|
|
device_manager_processor_service_GetVideoProcessorDeviceGuids,
|
|
|
|
device_manager_processor_service_GetVideoProcessorRenderTargets,
|
|
|
|
device_manager_processor_service_GetVideoProcessorSubStreamFormats,
|
|
|
|
device_manager_processor_service_GetVideoProcessorCaps,
|
|
|
|
device_manager_processor_service_GetProcAmpRange,
|
2020-06-26 17:40:55 +02:00
|
|
|
device_manager_processor_service_GetFilterPropertyRange,
|
|
|
|
device_manager_processor_service_CreateVideoProcessor,
|
2020-06-22 12:45:31 +02:00
|
|
|
};
|
|
|
|
|
2021-01-18 13:42:11 +01:00
|
|
|
static HRESULT WINAPI device_manager_decoder_service_QueryInterface(IDirectXVideoDecoderService *iface,
|
|
|
|
REFIID riid, void **obj)
|
|
|
|
{
|
|
|
|
if (IsEqualIID(riid, &IID_IDirectXVideoDecoderService) ||
|
|
|
|
IsEqualIID(riid, &IID_IDirectXVideoAccelerationService) ||
|
|
|
|
IsEqualIID(riid, &IID_IUnknown))
|
|
|
|
{
|
|
|
|
*obj = iface;
|
|
|
|
IDirectXVideoDecoderService_AddRef(iface);
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
WARN("Unsupported interface %s.\n", debugstr_guid(riid));
|
|
|
|
*obj = NULL;
|
|
|
|
return E_NOINTERFACE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI device_manager_decoder_service_AddRef(IDirectXVideoDecoderService *iface)
|
|
|
|
{
|
|
|
|
struct device_manager *manager = impl_from_IDirectXVideoDecoderService(iface);
|
|
|
|
return IDirect3DDeviceManager9_AddRef(&manager->IDirect3DDeviceManager9_iface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI device_manager_decoder_service_Release(IDirectXVideoDecoderService *iface)
|
|
|
|
{
|
|
|
|
struct device_manager *manager = impl_from_IDirectXVideoDecoderService(iface);
|
|
|
|
return IDirect3DDeviceManager9_Release(&manager->IDirect3DDeviceManager9_iface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_decoder_service_CreateSurface(IDirectXVideoDecoderService *iface,
|
|
|
|
UINT width, UINT height, UINT backbuffers, D3DFORMAT format, D3DPOOL pool, DWORD usage, DWORD dxvaType,
|
|
|
|
IDirect3DSurface9 **surfaces, HANDLE *shared_handle)
|
|
|
|
{
|
|
|
|
FIXME("%p, %u, %u, %u, %#x, %d, %d, %d, %p, %p.\n", iface, width, height, backbuffers, format, pool, usage,
|
|
|
|
dxvaType, surfaces, shared_handle);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_decoder_service_GetDecoderDeviceGuids(IDirectXVideoDecoderService *iface,
|
|
|
|
UINT *count, GUID **guids)
|
|
|
|
{
|
|
|
|
FIXME("%p, %p, %p.\n", iface, count, guids);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_decoder_service_GetDecoderRenderTargets(IDirectXVideoDecoderService *iface,
|
|
|
|
REFGUID guid, UINT *count, D3DFORMAT **formats)
|
|
|
|
{
|
|
|
|
FIXME("%p, %s, %p, %p.\n", iface, debugstr_guid(guid), count, formats);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_decoder_service_GetDecoderConfigurations(IDirectXVideoDecoderService *iface,
|
|
|
|
REFGUID guid, const DXVA2_VideoDesc *video_desc, IUnknown *reserved, UINT *count, DXVA2_ConfigPictureDecode **configs)
|
|
|
|
{
|
|
|
|
FIXME("%p, %s, %p, %p, %p, %p.\n", iface, debugstr_guid(guid), video_desc, reserved, count, configs);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_decoder_service_CreateVideoDecoder(IDirectXVideoDecoderService *iface,
|
|
|
|
REFGUID guid, const DXVA2_VideoDesc *video_desc, DXVA2_ConfigPictureDecode *config, IDirect3DSurface9 **rts,
|
|
|
|
UINT num_surfaces, IDirectXVideoDecoder **decoder)
|
|
|
|
{
|
|
|
|
FIXME("%p, %s, %p, %p, %p, %u, %p.\n", iface, debugstr_guid(guid), video_desc, config, rts, num_surfaces,
|
|
|
|
decoder);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const IDirectXVideoDecoderServiceVtbl device_manager_decoder_service_vtbl =
|
|
|
|
{
|
|
|
|
device_manager_decoder_service_QueryInterface,
|
|
|
|
device_manager_decoder_service_AddRef,
|
|
|
|
device_manager_decoder_service_Release,
|
|
|
|
device_manager_decoder_service_CreateSurface,
|
|
|
|
device_manager_decoder_service_GetDecoderDeviceGuids,
|
|
|
|
device_manager_decoder_service_GetDecoderRenderTargets,
|
|
|
|
device_manager_decoder_service_GetDecoderConfigurations,
|
|
|
|
device_manager_decoder_service_CreateVideoDecoder,
|
|
|
|
};
|
|
|
|
|
2020-06-22 12:45:29 +02:00
|
|
|
static HRESULT WINAPI device_manager_QueryInterface(IDirect3DDeviceManager9 *iface, REFIID riid, void **obj)
|
|
|
|
{
|
|
|
|
TRACE("%p, %s, %p.\n", iface, debugstr_guid(riid), obj);
|
|
|
|
|
|
|
|
if (IsEqualIID(&IID_IDirect3DDeviceManager9, riid) ||
|
|
|
|
IsEqualIID(&IID_IUnknown, riid))
|
|
|
|
{
|
|
|
|
*obj = iface;
|
|
|
|
IDirect3DDeviceManager9_AddRef(iface);
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
WARN("Unsupported interface %s.\n", debugstr_guid(riid));
|
|
|
|
*obj = NULL;
|
|
|
|
return E_NOINTERFACE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI device_manager_AddRef(IDirect3DDeviceManager9 *iface)
|
|
|
|
{
|
|
|
|
struct device_manager *manager = impl_from_IDirect3DDeviceManager9(iface);
|
|
|
|
ULONG refcount = InterlockedIncrement(&manager->refcount);
|
|
|
|
|
|
|
|
TRACE("%p, refcount %u.\n", iface, refcount);
|
|
|
|
|
|
|
|
return refcount;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI device_manager_Release(IDirect3DDeviceManager9 *iface)
|
|
|
|
{
|
|
|
|
struct device_manager *manager = impl_from_IDirect3DDeviceManager9(iface);
|
|
|
|
ULONG refcount = InterlockedDecrement(&manager->refcount);
|
2020-06-30 11:35:59 +02:00
|
|
|
size_t i;
|
2020-06-22 12:45:29 +02:00
|
|
|
|
|
|
|
TRACE("%p, refcount %u.\n", iface, refcount);
|
|
|
|
|
|
|
|
if (!refcount)
|
|
|
|
{
|
2020-06-22 12:45:30 +02:00
|
|
|
if (manager->device)
|
|
|
|
IDirect3DDevice9_Release(manager->device);
|
|
|
|
DeleteCriticalSection(&manager->cs);
|
2020-06-30 11:35:59 +02:00
|
|
|
for (i = 0; i < manager->count; ++i)
|
|
|
|
{
|
|
|
|
if (manager->handles[i].state_block)
|
|
|
|
IDirect3DStateBlock9_Release(manager->handles[i].state_block);
|
|
|
|
}
|
2020-06-22 12:45:30 +02:00
|
|
|
heap_free(manager->handles);
|
2020-06-22 12:45:29 +02:00
|
|
|
heap_free(manager);
|
|
|
|
}
|
|
|
|
|
|
|
|
return refcount;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_ResetDevice(IDirect3DDeviceManager9 *iface, IDirect3DDevice9 *device,
|
|
|
|
UINT token)
|
|
|
|
{
|
2020-06-22 12:45:30 +02:00
|
|
|
struct device_manager *manager = impl_from_IDirect3DDeviceManager9(iface);
|
|
|
|
size_t i;
|
2020-06-22 12:45:29 +02:00
|
|
|
|
2020-06-22 12:45:30 +02:00
|
|
|
TRACE("%p, %p, %#x.\n", iface, device, token);
|
|
|
|
|
|
|
|
if (token != manager->token)
|
|
|
|
return E_INVALIDARG;
|
|
|
|
|
|
|
|
EnterCriticalSection(&manager->cs);
|
|
|
|
if (manager->device)
|
|
|
|
{
|
|
|
|
for (i = 0; i < manager->count; ++i)
|
2020-06-30 11:35:59 +02:00
|
|
|
{
|
|
|
|
if (manager->handles[i].state_block)
|
|
|
|
IDirect3DStateBlock9_Release(manager->handles[i].state_block);
|
|
|
|
manager->handles[i].state_block = NULL;
|
2020-06-22 12:45:30 +02:00
|
|
|
manager->handles[i].flags |= HANDLE_FLAG_INVALID;
|
2020-06-30 11:35:59 +02:00
|
|
|
}
|
|
|
|
manager->locking_handle = NULL;
|
2020-06-22 12:45:30 +02:00
|
|
|
IDirect3DDevice9_Release(manager->device);
|
|
|
|
}
|
|
|
|
manager->device = device;
|
|
|
|
IDirect3DDevice9_AddRef(manager->device);
|
|
|
|
LeaveCriticalSection(&manager->cs);
|
|
|
|
|
2020-06-30 11:35:59 +02:00
|
|
|
WakeAllConditionVariable(&manager->lock);
|
|
|
|
|
2020-06-22 12:45:30 +02:00
|
|
|
return S_OK;
|
2020-06-22 12:45:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_OpenDeviceHandle(IDirect3DDeviceManager9 *iface, HANDLE *hdevice)
|
|
|
|
{
|
2020-06-22 12:45:30 +02:00
|
|
|
struct device_manager *manager = impl_from_IDirect3DDeviceManager9(iface);
|
|
|
|
HRESULT hr = S_OK;
|
|
|
|
size_t i;
|
2020-06-22 12:45:29 +02:00
|
|
|
|
2020-06-22 12:45:30 +02:00
|
|
|
TRACE("%p, %p.\n", iface, hdevice);
|
|
|
|
|
|
|
|
*hdevice = NULL;
|
|
|
|
|
|
|
|
EnterCriticalSection(&manager->cs);
|
|
|
|
if (!manager->device)
|
|
|
|
hr = DXVA2_E_NOT_INITIALIZED;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (i = 0; i < manager->count; ++i)
|
|
|
|
{
|
|
|
|
if (!(manager->handles[i].flags & HANDLE_FLAG_OPEN))
|
|
|
|
{
|
|
|
|
manager->handles[i].flags |= HANDLE_FLAG_OPEN;
|
|
|
|
*hdevice = ULongToHandle(i + 1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dxva_array_reserve((void **)&manager->handles, &manager->capacity, manager->count + 1,
|
|
|
|
sizeof(*manager->handles)))
|
|
|
|
{
|
|
|
|
*hdevice = ULongToHandle(manager->count + 1);
|
2020-10-19 16:10:39 +02:00
|
|
|
manager->handles[manager->count].flags = HANDLE_FLAG_OPEN;
|
2020-06-30 11:35:59 +02:00
|
|
|
manager->handles[manager->count].state_block = NULL;
|
2020-06-22 12:45:30 +02:00
|
|
|
manager->count++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
hr = E_OUTOFMEMORY;
|
|
|
|
}
|
|
|
|
LeaveCriticalSection(&manager->cs);
|
|
|
|
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT device_manager_get_handle_index(struct device_manager *manager, HANDLE hdevice, size_t *idx)
|
|
|
|
{
|
2020-10-21 08:46:28 +02:00
|
|
|
if (!hdevice || hdevice > ULongToHandle(manager->count))
|
2020-06-22 12:45:30 +02:00
|
|
|
return E_HANDLE;
|
|
|
|
*idx = (ULONG_PTR)hdevice - 1;
|
|
|
|
return S_OK;
|
2020-06-22 12:45:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_CloseDeviceHandle(IDirect3DDeviceManager9 *iface, HANDLE hdevice)
|
|
|
|
{
|
2020-06-22 12:45:30 +02:00
|
|
|
struct device_manager *manager = impl_from_IDirect3DDeviceManager9(iface);
|
|
|
|
HRESULT hr;
|
|
|
|
size_t idx;
|
2020-06-22 12:45:29 +02:00
|
|
|
|
2020-06-22 12:45:30 +02:00
|
|
|
TRACE("%p, %p.\n", iface, hdevice);
|
|
|
|
|
|
|
|
EnterCriticalSection(&manager->cs);
|
|
|
|
if (SUCCEEDED(hr = device_manager_get_handle_index(manager, hdevice, &idx)))
|
|
|
|
{
|
|
|
|
if (manager->handles[idx].flags & HANDLE_FLAG_OPEN)
|
|
|
|
{
|
2020-06-30 11:35:59 +02:00
|
|
|
if (manager->locking_handle == hdevice)
|
|
|
|
manager->locking_handle = NULL;
|
2020-06-22 12:45:30 +02:00
|
|
|
manager->handles[idx].flags = 0;
|
|
|
|
if (idx == manager->count - 1)
|
|
|
|
manager->count--;
|
2020-06-30 11:35:59 +02:00
|
|
|
if (manager->handles[idx].state_block)
|
|
|
|
IDirect3DStateBlock9_Release(manager->handles[idx].state_block);
|
|
|
|
manager->handles[idx].state_block = NULL;
|
2020-06-22 12:45:30 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
hr = E_HANDLE;
|
|
|
|
}
|
|
|
|
LeaveCriticalSection(&manager->cs);
|
|
|
|
|
2020-06-30 11:35:59 +02:00
|
|
|
WakeAllConditionVariable(&manager->lock);
|
|
|
|
|
2020-06-22 12:45:30 +02:00
|
|
|
return hr;
|
2020-06-22 12:45:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_TestDevice(IDirect3DDeviceManager9 *iface, HANDLE hdevice)
|
|
|
|
{
|
2020-06-22 12:45:30 +02:00
|
|
|
struct device_manager *manager = impl_from_IDirect3DDeviceManager9(iface);
|
|
|
|
HRESULT hr;
|
|
|
|
size_t idx;
|
2020-06-22 12:45:29 +02:00
|
|
|
|
2020-06-22 12:45:30 +02:00
|
|
|
TRACE("%p, %p.\n", iface, hdevice);
|
|
|
|
|
|
|
|
EnterCriticalSection(&manager->cs);
|
|
|
|
if (SUCCEEDED(hr = device_manager_get_handle_index(manager, hdevice, &idx)))
|
|
|
|
{
|
|
|
|
unsigned int flags = manager->handles[idx].flags;
|
|
|
|
|
|
|
|
if (flags & HANDLE_FLAG_INVALID)
|
|
|
|
hr = DXVA2_E_NEW_VIDEO_DEVICE;
|
|
|
|
else if (!(flags & HANDLE_FLAG_OPEN))
|
|
|
|
hr = E_HANDLE;
|
|
|
|
}
|
|
|
|
LeaveCriticalSection(&manager->cs);
|
|
|
|
|
|
|
|
return hr;
|
2020-06-22 12:45:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_LockDevice(IDirect3DDeviceManager9 *iface, HANDLE hdevice,
|
|
|
|
IDirect3DDevice9 **device, BOOL block)
|
|
|
|
{
|
2020-06-30 11:35:59 +02:00
|
|
|
struct device_manager *manager = impl_from_IDirect3DDeviceManager9(iface);
|
|
|
|
HRESULT hr;
|
|
|
|
size_t idx;
|
2020-06-22 12:45:29 +02:00
|
|
|
|
2020-06-30 11:35:59 +02:00
|
|
|
TRACE("%p, %p, %p, %d.\n", iface, hdevice, device, block);
|
|
|
|
|
|
|
|
EnterCriticalSection(&manager->cs);
|
|
|
|
if (!manager->device)
|
|
|
|
hr = DXVA2_E_NOT_INITIALIZED;
|
|
|
|
else if (SUCCEEDED(hr = device_manager_get_handle_index(manager, hdevice, &idx)))
|
|
|
|
{
|
|
|
|
if (manager->locking_handle && !block)
|
|
|
|
hr = DXVA2_E_VIDEO_DEVICE_LOCKED;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (manager->locking_handle && block)
|
|
|
|
{
|
|
|
|
SleepConditionVariableCS(&manager->lock, &manager->cs, INFINITE);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (SUCCEEDED(hr = device_manager_get_handle_index(manager, hdevice, &idx)))
|
|
|
|
{
|
|
|
|
if (manager->handles[idx].flags & HANDLE_FLAG_INVALID)
|
|
|
|
hr = DXVA2_E_NEW_VIDEO_DEVICE;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (manager->handles[idx].state_block)
|
|
|
|
{
|
|
|
|
if (FAILED(IDirect3DStateBlock9_Apply(manager->handles[idx].state_block)))
|
|
|
|
WARN("Failed to apply state.\n");
|
|
|
|
IDirect3DStateBlock9_Release(manager->handles[idx].state_block);
|
|
|
|
manager->handles[idx].state_block = NULL;
|
|
|
|
}
|
|
|
|
*device = manager->device;
|
|
|
|
IDirect3DDevice9_AddRef(*device);
|
|
|
|
manager->locking_handle = hdevice;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
LeaveCriticalSection(&manager->cs);
|
|
|
|
|
|
|
|
return hr;
|
2020-06-22 12:45:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_UnlockDevice(IDirect3DDeviceManager9 *iface, HANDLE hdevice, BOOL savestate)
|
|
|
|
{
|
2020-06-30 11:35:59 +02:00
|
|
|
struct device_manager *manager = impl_from_IDirect3DDeviceManager9(iface);
|
|
|
|
HRESULT hr;
|
|
|
|
size_t idx;
|
2020-06-22 12:45:29 +02:00
|
|
|
|
2020-06-30 11:35:59 +02:00
|
|
|
TRACE("%p, %p, %d.\n", iface, hdevice, savestate);
|
|
|
|
|
|
|
|
EnterCriticalSection(&manager->cs);
|
|
|
|
|
|
|
|
if (hdevice != manager->locking_handle)
|
|
|
|
hr = E_INVALIDARG;
|
|
|
|
else if (SUCCEEDED(hr = device_manager_get_handle_index(manager, hdevice, &idx)))
|
|
|
|
{
|
|
|
|
manager->locking_handle = NULL;
|
|
|
|
if (savestate)
|
|
|
|
IDirect3DDevice9_CreateStateBlock(manager->device, D3DSBT_ALL, &manager->handles[idx].state_block);
|
|
|
|
}
|
|
|
|
|
|
|
|
LeaveCriticalSection(&manager->cs);
|
|
|
|
|
|
|
|
WakeAllConditionVariable(&manager->lock);
|
|
|
|
|
|
|
|
return hr;
|
2020-06-22 12:45:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI device_manager_GetVideoService(IDirect3DDeviceManager9 *iface, HANDLE hdevice, REFIID riid,
|
|
|
|
void **obj)
|
|
|
|
{
|
2020-06-22 12:45:31 +02:00
|
|
|
struct device_manager *manager = impl_from_IDirect3DDeviceManager9(iface);
|
|
|
|
HRESULT hr;
|
|
|
|
size_t idx;
|
2020-06-22 12:45:29 +02:00
|
|
|
|
2020-06-22 12:45:31 +02:00
|
|
|
TRACE("%p, %p, %s, %p.\n", iface, hdevice, debugstr_guid(riid), obj);
|
|
|
|
|
|
|
|
EnterCriticalSection(&manager->cs);
|
|
|
|
if (SUCCEEDED(hr = device_manager_get_handle_index(manager, hdevice, &idx)))
|
|
|
|
{
|
|
|
|
unsigned int flags = manager->handles[idx].flags;
|
|
|
|
|
|
|
|
if (flags & HANDLE_FLAG_INVALID)
|
|
|
|
hr = DXVA2_E_NEW_VIDEO_DEVICE;
|
|
|
|
else if (!(flags & HANDLE_FLAG_OPEN))
|
|
|
|
hr = E_HANDLE;
|
|
|
|
else
|
2020-06-23 13:31:17 +02:00
|
|
|
hr = IDirectXVideoProcessorService_QueryInterface(&manager->IDirectXVideoProcessorService_iface,
|
|
|
|
riid, obj);
|
2020-06-22 12:45:31 +02:00
|
|
|
}
|
|
|
|
LeaveCriticalSection(&manager->cs);
|
|
|
|
|
|
|
|
return hr;
|
2020-06-22 12:45:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static const IDirect3DDeviceManager9Vtbl device_manager_vtbl =
|
|
|
|
{
|
|
|
|
device_manager_QueryInterface,
|
|
|
|
device_manager_AddRef,
|
|
|
|
device_manager_Release,
|
|
|
|
device_manager_ResetDevice,
|
|
|
|
device_manager_OpenDeviceHandle,
|
|
|
|
device_manager_CloseDeviceHandle,
|
|
|
|
device_manager_TestDevice,
|
|
|
|
device_manager_LockDevice,
|
|
|
|
device_manager_UnlockDevice,
|
|
|
|
device_manager_GetVideoService,
|
|
|
|
};
|
|
|
|
|
2014-03-26 05:32:36 +01:00
|
|
|
BOOL WINAPI CapabilitiesRequestAndCapabilitiesReply( HMONITOR monitor, LPSTR buffer, DWORD length )
|
|
|
|
{
|
|
|
|
FIXME("(%p, %p, %d): stub\n", monitor, buffer, length);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2020-06-22 12:45:29 +02:00
|
|
|
HRESULT WINAPI DXVA2CreateDirect3DDeviceManager9(UINT *token, IDirect3DDeviceManager9 **manager)
|
2014-03-26 05:32:36 +01:00
|
|
|
{
|
2020-06-22 12:45:29 +02:00
|
|
|
struct device_manager *object;
|
2014-03-26 05:32:36 +01:00
|
|
|
|
2020-06-22 12:45:29 +02:00
|
|
|
TRACE("%p, %p.\n", token, manager);
|
|
|
|
|
|
|
|
*manager = NULL;
|
|
|
|
|
|
|
|
if (!(object = heap_alloc_zero(sizeof(*object))))
|
|
|
|
return E_OUTOFMEMORY;
|
|
|
|
|
|
|
|
object->IDirect3DDeviceManager9_iface.lpVtbl = &device_manager_vtbl;
|
2020-06-22 12:45:31 +02:00
|
|
|
object->IDirectXVideoProcessorService_iface.lpVtbl = &device_manager_processor_service_vtbl;
|
2021-01-18 13:42:11 +01:00
|
|
|
object->IDirectXVideoDecoderService_iface.lpVtbl = &device_manager_decoder_service_vtbl;
|
2020-06-22 12:45:29 +02:00
|
|
|
object->refcount = 1;
|
|
|
|
object->token = GetTickCount();
|
2020-06-22 12:45:30 +02:00
|
|
|
InitializeCriticalSection(&object->cs);
|
2020-06-30 11:35:59 +02:00
|
|
|
InitializeConditionVariable(&object->lock);
|
2020-06-22 12:45:29 +02:00
|
|
|
|
|
|
|
*token = object->token;
|
|
|
|
*manager = &object->IDirect3DDeviceManager9_iface;
|
|
|
|
|
|
|
|
return S_OK;
|
2014-03-26 05:32:36 +01:00
|
|
|
}
|
|
|
|
|
2020-06-23 13:31:17 +02:00
|
|
|
HRESULT WINAPI DXVA2CreateVideoService(IDirect3DDevice9 *device, REFIID riid, void **obj)
|
2014-03-26 05:32:36 +01:00
|
|
|
{
|
2020-06-23 13:31:17 +02:00
|
|
|
IDirect3DDeviceManager9 *manager;
|
|
|
|
HANDLE handle;
|
|
|
|
HRESULT hr;
|
|
|
|
UINT token;
|
2014-03-26 05:32:36 +01:00
|
|
|
|
2020-06-23 13:31:17 +02:00
|
|
|
TRACE("%p, %s, %p.\n", device, debugstr_guid(riid), obj);
|
|
|
|
|
|
|
|
if (FAILED(hr = DXVA2CreateDirect3DDeviceManager9(&token, &manager)))
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
if (FAILED(hr = IDirect3DDeviceManager9_ResetDevice(manager, device, token)))
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
if (FAILED(hr = IDirect3DDeviceManager9_OpenDeviceHandle(manager, &handle)))
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
hr = IDirect3DDeviceManager9_GetVideoService(manager, handle, riid, obj);
|
|
|
|
IDirect3DDeviceManager9_CloseDeviceHandle(manager, handle);
|
|
|
|
|
|
|
|
done:
|
|
|
|
IDirect3DDeviceManager9_Release(manager);
|
|
|
|
|
|
|
|
return hr;
|
2014-03-26 05:32:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI DegaussMonitor( HMONITOR monitor )
|
|
|
|
{
|
|
|
|
FIXME("(%p): stub\n", monitor);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI DestroyPhysicalMonitor( HMONITOR monitor )
|
|
|
|
{
|
|
|
|
FIXME("(%p): stub\n", monitor);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI DestroyPhysicalMonitors( DWORD arraySize, LPPHYSICAL_MONITOR array )
|
|
|
|
{
|
|
|
|
FIXME("(0x%x, %p): stub\n", arraySize, array);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI GetCapabilitiesStringLength( HMONITOR monitor, LPDWORD length )
|
|
|
|
{
|
|
|
|
FIXME("(%p, %p): stub\n", monitor, length);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI GetMonitorBrightness( HMONITOR monitor, LPDWORD minimum, LPDWORD current, LPDWORD maximum )
|
|
|
|
{
|
|
|
|
FIXME("(%p, %p, %p, %p): stub\n", monitor, minimum, current, maximum);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI GetMonitorCapabilities( HMONITOR monitor, LPDWORD capabilities, LPDWORD temperatures )
|
|
|
|
{
|
|
|
|
FIXME("(%p, %p, %p): stub\n", monitor, capabilities, temperatures);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BOOL WINAPI GetMonitorColorTemperature( HMONITOR monitor, LPMC_COLOR_TEMPERATURE temperature )
|
|
|
|
{
|
|
|
|
FIXME("(%p, %p): stub\n", monitor, temperature);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI GetMonitorContrast( HMONITOR monitor, LPDWORD minimum, LPDWORD current, LPDWORD maximum )
|
|
|
|
{
|
|
|
|
FIXME("(%p, %p, %p, %p): stub\n", monitor, minimum, current, maximum);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI GetMonitorDisplayAreaPosition( HMONITOR monitor, MC_POSITION_TYPE type, LPDWORD minimum,
|
|
|
|
LPDWORD current, LPDWORD maximum )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x, %p, %p, %p): stub\n", monitor, type, minimum, current, maximum);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI GetMonitorDisplayAreaSize( HMONITOR monitor, MC_SIZE_TYPE type, LPDWORD minimum,
|
|
|
|
LPDWORD current, LPDWORD maximum )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x, %p, %p, %p): stub\n", monitor, type, minimum, current, maximum);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI GetMonitorRedGreenOrBlueDrive( HMONITOR monitor, MC_DRIVE_TYPE type, LPDWORD minimum,
|
|
|
|
LPDWORD current, LPDWORD maximum )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x, %p, %p, %p): stub\n", monitor, type, minimum, current, maximum);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI GetMonitorRedGreenOrBlueGain( HMONITOR monitor, MC_GAIN_TYPE type, LPDWORD minimum,
|
|
|
|
LPDWORD current, LPDWORD maximum )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x, %p, %p, %p): stub\n", monitor, type, minimum, current, maximum);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI GetMonitorTechnologyType( HMONITOR monitor, LPMC_DISPLAY_TECHNOLOGY_TYPE type )
|
|
|
|
{
|
|
|
|
FIXME("(%p, %p): stub\n", monitor, type);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI GetNumberOfPhysicalMonitorsFromHMONITOR( HMONITOR monitor, LPDWORD number )
|
|
|
|
{
|
|
|
|
FIXME("(%p, %p): stub\n", monitor, number);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT WINAPI GetNumberOfPhysicalMonitorsFromIDirect3DDevice9( IDirect3DDevice9 *device, LPDWORD number )
|
|
|
|
{
|
|
|
|
FIXME("(%p, %p): stub\n", device, number);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI GetPhysicalMonitorsFromHMONITOR( HMONITOR monitor, DWORD arraySize, LPPHYSICAL_MONITOR array )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x, %p): stub\n", monitor, arraySize, array);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT WINAPI GetPhysicalMonitorsFromIDirect3DDevice9( IDirect3DDevice9 *device, DWORD arraySize, LPPHYSICAL_MONITOR array )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x, %p): stub\n", device, arraySize, array);
|
|
|
|
|
2016-09-13 23:13:20 +02:00
|
|
|
return E_NOTIMPL;
|
2014-03-26 05:32:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI GetTimingReport( HMONITOR monitor, LPMC_TIMING_REPORT timingReport )
|
|
|
|
{
|
|
|
|
FIXME("(%p, %p): stub\n", monitor, timingReport);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI GetVCPFeatureAndVCPFeatureReply( HMONITOR monitor, BYTE vcpCode, LPMC_VCP_CODE_TYPE pvct,
|
|
|
|
LPDWORD current, LPDWORD maximum )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%02x, %p, %p, %p): stub\n", monitor, vcpCode, pvct, current, maximum);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT WINAPI OPMGetVideoOutputsFromHMONITOR( HMONITOR monitor, /* OPM_VIDEO_OUTPUT_SEMANTICS */ int vos,
|
|
|
|
ULONG *numVideoOutputs, /* IOPMVideoOutput */ void ***videoOutputs )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x, %p, %p): stub\n", monitor, vos, numVideoOutputs, videoOutputs);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT WINAPI OPMGetVideoOutputsFromIDirect3DDevice9Object( IDirect3DDevice9 *device, /* OPM_VIDEO_OUTPUT_SEMANTICS */ int vos,
|
|
|
|
ULONG *numVideoOutputs, /* IOPMVideoOutput */ void ***videoOutputs )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x, %p, %p): stub\n", device, vos, numVideoOutputs, videoOutputs);
|
|
|
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI RestoreMonitorFactoryColorDefaults( HMONITOR monitor )
|
|
|
|
{
|
|
|
|
FIXME("(%p): stub\n", monitor);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI RestoreMonitorFactoryDefaults( HMONITOR monitor )
|
|
|
|
{
|
|
|
|
FIXME("(%p): stub\n", monitor);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI SaveCurrentMonitorSettings( HMONITOR monitor )
|
|
|
|
{
|
|
|
|
FIXME("(%p): stub\n", monitor);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI SaveCurrentSettings( HMONITOR monitor )
|
|
|
|
{
|
|
|
|
FIXME("(%p): stub\n", monitor);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI SetMonitorBrightness( HMONITOR monitor, DWORD brightness )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x): stub\n", monitor, brightness);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI SetMonitorColorTemperature( HMONITOR monitor, MC_COLOR_TEMPERATURE temperature )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x): stub\n", monitor, temperature);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI SetMonitorContrast( HMONITOR monitor, DWORD contrast )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x): stub\n", monitor, contrast);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI SetMonitorDisplayAreaPosition( HMONITOR monitor, MC_POSITION_TYPE type, DWORD position )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x, 0x%x): stub\n", monitor, type, position);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI SetMonitorDisplayAreaSize( HMONITOR monitor, MC_SIZE_TYPE type, DWORD size )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x, 0x%x): stub\n", monitor, type, size);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI SetMonitorRedGreenOrBlueDrive( HMONITOR monitor, MC_DRIVE_TYPE type, DWORD drive )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x, 0x%x): stub\n", monitor, type, drive);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI SetMonitorRedGreenOrBlueGain( HMONITOR monitor, MC_GAIN_TYPE type, DWORD gain )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%x, 0x%x): stub\n", monitor, type, gain);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOL WINAPI SetVCPFeature( HMONITOR monitor, BYTE vcpCode, DWORD value )
|
|
|
|
{
|
|
|
|
FIXME("(%p, 0x%02x, 0x%x): stub\n", monitor, vcpCode, value);
|
|
|
|
|
|
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
|
|
return FALSE;
|
|
|
|
}
|