mirror of
https://gitlab.winehq.org/wine/wine.git
synced 2025-08-28 18:23:54 +02:00
2356 lines
73 KiB
C
2356 lines
73 KiB
C
/*
|
|
* Copyright 2018 Alistair Leslie-Hughes
|
|
*
|
|
* 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
|
|
*/
|
|
|
|
#define COBJMACROS
|
|
|
|
#include <malloc.h>
|
|
|
|
#include "mfplat_private.h"
|
|
#include "rtworkq.h"
|
|
|
|
#include "d3d11.h"
|
|
#include "initguid.h"
|
|
#include "d3d9.h"
|
|
#include "evr.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(mfplat);
|
|
|
|
#define ALIGN_SIZE(size, alignment) (((size) + (alignment)) & ~((alignment)))
|
|
|
|
typedef void (*p_copy_image_func)(BYTE *dest, LONG dest_stride, const BYTE *src, LONG src_stride, DWORD width, DWORD lines);
|
|
|
|
struct buffer
|
|
{
|
|
IMFMediaBuffer IMFMediaBuffer_iface;
|
|
IMF2DBuffer2 IMF2DBuffer2_iface;
|
|
IMFDXGIBuffer IMFDXGIBuffer_iface;
|
|
IMFGetService IMFGetService_iface;
|
|
LONG refcount;
|
|
|
|
BYTE *data;
|
|
DWORD max_length;
|
|
DWORD current_length;
|
|
|
|
struct
|
|
{
|
|
BYTE *linear_buffer;
|
|
DWORD plane_size;
|
|
|
|
BYTE *scanline0;
|
|
unsigned int width;
|
|
unsigned int height;
|
|
int pitch;
|
|
unsigned int locks;
|
|
MF2DBuffer_LockFlags lock_flags;
|
|
p_copy_image_func copy_image;
|
|
} _2d;
|
|
struct
|
|
{
|
|
IDirect3DSurface9 *surface;
|
|
D3DLOCKED_RECT rect;
|
|
} d3d9_surface;
|
|
struct
|
|
{
|
|
ID3D11Texture2D *texture;
|
|
unsigned int sub_resource_idx;
|
|
ID3D11Texture2D *rb_texture;
|
|
D3D11_MAPPED_SUBRESOURCE map_desc;
|
|
struct attributes attributes;
|
|
} dxgi_surface;
|
|
|
|
CRITICAL_SECTION cs;
|
|
};
|
|
|
|
static void copy_image(const struct buffer *buffer, BYTE *dest, LONG dest_stride, const BYTE *src,
|
|
LONG src_stride, DWORD width, DWORD lines)
|
|
{
|
|
MFCopyImage(dest, dest_stride, src, src_stride, width, lines);
|
|
|
|
if (buffer->_2d.copy_image)
|
|
{
|
|
dest += dest_stride * lines;
|
|
src += src_stride * lines;
|
|
buffer->_2d.copy_image(dest, dest_stride, src, src_stride, width, lines);
|
|
}
|
|
}
|
|
|
|
static void copy_image_nv12(BYTE *dest, LONG dest_stride, const BYTE *src, LONG src_stride, DWORD width, DWORD lines)
|
|
{
|
|
MFCopyImage(dest, dest_stride, src, src_stride, width, lines / 2);
|
|
}
|
|
|
|
static void copy_image_imc1(BYTE *dest, LONG dest_stride, const BYTE *src, LONG src_stride, DWORD width, DWORD lines)
|
|
{
|
|
MFCopyImage(dest, dest_stride, src, src_stride, width / 2, lines);
|
|
}
|
|
|
|
static void copy_image_imc2(BYTE *dest, LONG dest_stride, const BYTE *src, LONG src_stride, DWORD width, DWORD lines)
|
|
{
|
|
MFCopyImage(dest, dest_stride / 2, src, src_stride / 2, width / 2, lines);
|
|
}
|
|
|
|
static inline struct buffer *impl_from_IMFMediaBuffer(IMFMediaBuffer *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct buffer, IMFMediaBuffer_iface);
|
|
}
|
|
|
|
static struct buffer *impl_from_IMF2DBuffer2(IMF2DBuffer2 *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct buffer, IMF2DBuffer2_iface);
|
|
}
|
|
|
|
static struct buffer *impl_from_IMFGetService(IMFGetService *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct buffer, IMFGetService_iface);
|
|
}
|
|
|
|
static struct buffer *impl_from_IMFDXGIBuffer(IMFDXGIBuffer *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct buffer, IMFDXGIBuffer_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI memory_buffer_QueryInterface(IMFMediaBuffer *iface, REFIID riid, void **out)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
|
|
TRACE("%p, %s, %p.\n", iface, debugstr_guid(riid), out);
|
|
|
|
if (IsEqualIID(riid, &IID_IMFMediaBuffer) ||
|
|
IsEqualIID(riid, &IID_IUnknown))
|
|
{
|
|
*out = &buffer->IMFMediaBuffer_iface;
|
|
IMFMediaBuffer_AddRef(iface);
|
|
return S_OK;
|
|
}
|
|
|
|
WARN("Unsupported %s.\n", debugstr_guid(riid));
|
|
*out = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
static ULONG WINAPI memory_buffer_AddRef(IMFMediaBuffer *iface)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
ULONG refcount = InterlockedIncrement(&buffer->refcount);
|
|
|
|
TRACE("%p, refcount %lu.\n", buffer, refcount);
|
|
|
|
return refcount;
|
|
}
|
|
|
|
static ULONG WINAPI memory_buffer_Release(IMFMediaBuffer *iface)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
ULONG refcount = InterlockedDecrement(&buffer->refcount);
|
|
|
|
TRACE("%p, refcount %lu.\n", iface, refcount);
|
|
|
|
if (!refcount)
|
|
{
|
|
if (buffer->d3d9_surface.surface)
|
|
IDirect3DSurface9_Release(buffer->d3d9_surface.surface);
|
|
if (buffer->dxgi_surface.texture)
|
|
{
|
|
ID3D11Texture2D_Release(buffer->dxgi_surface.texture);
|
|
if (buffer->dxgi_surface.rb_texture)
|
|
ID3D11Texture2D_Release(buffer->dxgi_surface.rb_texture);
|
|
clear_attributes_object(&buffer->dxgi_surface.attributes);
|
|
}
|
|
DeleteCriticalSection(&buffer->cs);
|
|
free(buffer->_2d.linear_buffer);
|
|
_aligned_free(buffer->data);
|
|
free(buffer);
|
|
}
|
|
|
|
return refcount;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_buffer_Lock(IMFMediaBuffer *iface, BYTE **data, DWORD *max_length, DWORD *current_length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
|
|
TRACE("%p, %p %p, %p.\n", iface, data, max_length, current_length);
|
|
|
|
if (!data)
|
|
return E_INVALIDARG;
|
|
|
|
*data = buffer->data;
|
|
if (max_length)
|
|
*max_length = buffer->max_length;
|
|
if (current_length)
|
|
*current_length = buffer->current_length;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_buffer_Unlock(IMFMediaBuffer *iface)
|
|
{
|
|
TRACE("%p.\n", iface);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_buffer_GetCurrentLength(IMFMediaBuffer *iface, DWORD *current_length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
|
|
TRACE("%p.\n", iface);
|
|
|
|
if (!current_length)
|
|
return E_INVALIDARG;
|
|
|
|
*current_length = buffer->current_length;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_buffer_SetCurrentLength(IMFMediaBuffer *iface, DWORD current_length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
|
|
TRACE("%p, %lu.\n", iface, current_length);
|
|
|
|
if (current_length > buffer->max_length)
|
|
return E_INVALIDARG;
|
|
|
|
buffer->current_length = current_length;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_buffer_GetMaxLength(IMFMediaBuffer *iface, DWORD *max_length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
|
|
TRACE("%p, %p.\n", iface, max_length);
|
|
|
|
if (!max_length)
|
|
return E_INVALIDARG;
|
|
|
|
*max_length = buffer->max_length;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static const IMFMediaBufferVtbl memory_1d_buffer_vtbl =
|
|
{
|
|
memory_buffer_QueryInterface,
|
|
memory_buffer_AddRef,
|
|
memory_buffer_Release,
|
|
memory_buffer_Lock,
|
|
memory_buffer_Unlock,
|
|
memory_buffer_GetCurrentLength,
|
|
memory_buffer_SetCurrentLength,
|
|
memory_buffer_GetMaxLength,
|
|
};
|
|
|
|
static HRESULT WINAPI memory_1d_2d_buffer_QueryInterface(IMFMediaBuffer *iface, REFIID riid, void **out)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
|
|
TRACE("%p, %s, %p.\n", iface, debugstr_guid(riid), out);
|
|
|
|
if (IsEqualIID(riid, &IID_IMFMediaBuffer) ||
|
|
IsEqualIID(riid, &IID_IUnknown))
|
|
{
|
|
*out = &buffer->IMFMediaBuffer_iface;
|
|
}
|
|
else if (IsEqualIID(riid, &IID_IMF2DBuffer2) ||
|
|
IsEqualIID(riid, &IID_IMF2DBuffer))
|
|
{
|
|
*out = &buffer->IMF2DBuffer2_iface;
|
|
}
|
|
else if (IsEqualIID(riid, &IID_IMFGetService))
|
|
{
|
|
*out = &buffer->IMFGetService_iface;
|
|
}
|
|
else
|
|
{
|
|
WARN("Unsupported interface %s.\n", debugstr_guid(riid));
|
|
*out = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*out);
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_1d_2d_buffer_Lock(IMFMediaBuffer *iface, BYTE **data, DWORD *max_length, DWORD *current_length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE("%p, %p, %p, %p.\n", iface, data, max_length, current_length);
|
|
|
|
if (!data)
|
|
return E_POINTER;
|
|
|
|
/* Allocate linear buffer and return it as a copy of current content. Maximum and current length are
|
|
unrelated to 2D buffer maximum allocate length, or maintained current length. */
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
if (!buffer->_2d.linear_buffer && buffer->_2d.locks)
|
|
hr = MF_E_INVALIDREQUEST;
|
|
else if (!buffer->_2d.linear_buffer)
|
|
{
|
|
if (!(buffer->_2d.linear_buffer = malloc(buffer->_2d.plane_size)))
|
|
hr = E_OUTOFMEMORY;
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
int pitch = buffer->_2d.pitch;
|
|
|
|
if (pitch < 0)
|
|
pitch = -pitch;
|
|
copy_image(buffer, buffer->_2d.linear_buffer, buffer->_2d.width, buffer->data, pitch,
|
|
buffer->_2d.width, buffer->_2d.height);
|
|
}
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
++buffer->_2d.locks;
|
|
*data = buffer->_2d.linear_buffer;
|
|
if (max_length)
|
|
*max_length = buffer->_2d.plane_size;
|
|
if (current_length)
|
|
*current_length = buffer->_2d.plane_size;
|
|
}
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_1d_2d_buffer_Unlock(IMFMediaBuffer *iface)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
|
|
TRACE("%p.\n", iface);
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
if (buffer->_2d.linear_buffer && !--buffer->_2d.locks)
|
|
{
|
|
int pitch = buffer->_2d.pitch;
|
|
|
|
if (pitch < 0)
|
|
pitch = -pitch;
|
|
copy_image(buffer, buffer->data, pitch, buffer->_2d.linear_buffer, buffer->_2d.width,
|
|
buffer->_2d.width, buffer->_2d.height);
|
|
|
|
free(buffer->_2d.linear_buffer);
|
|
buffer->_2d.linear_buffer = NULL;
|
|
}
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static const IMFMediaBufferVtbl memory_1d_2d_buffer_vtbl =
|
|
{
|
|
memory_1d_2d_buffer_QueryInterface,
|
|
memory_buffer_AddRef,
|
|
memory_buffer_Release,
|
|
memory_1d_2d_buffer_Lock,
|
|
memory_1d_2d_buffer_Unlock,
|
|
memory_buffer_GetCurrentLength,
|
|
memory_buffer_SetCurrentLength,
|
|
memory_buffer_GetMaxLength,
|
|
};
|
|
|
|
static HRESULT WINAPI d3d9_surface_buffer_Lock(IMFMediaBuffer *iface, BYTE **data, DWORD *max_length, DWORD *current_length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE("%p, %p, %p, %p.\n", iface, data, max_length, current_length);
|
|
|
|
if (!data)
|
|
return E_POINTER;
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
if (!buffer->_2d.linear_buffer && buffer->_2d.locks)
|
|
hr = MF_E_INVALIDREQUEST;
|
|
else if (!buffer->_2d.linear_buffer)
|
|
{
|
|
D3DLOCKED_RECT rect;
|
|
|
|
if (!(buffer->_2d.linear_buffer = malloc(buffer->_2d.plane_size)))
|
|
hr = E_OUTOFMEMORY;
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = IDirect3DSurface9_LockRect(buffer->d3d9_surface.surface, &rect, NULL, 0);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
copy_image(buffer, buffer->_2d.linear_buffer, buffer->_2d.width, rect.pBits, rect.Pitch,
|
|
buffer->_2d.width, buffer->_2d.height);
|
|
IDirect3DSurface9_UnlockRect(buffer->d3d9_surface.surface);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
++buffer->_2d.locks;
|
|
*data = buffer->_2d.linear_buffer;
|
|
if (max_length)
|
|
*max_length = buffer->_2d.plane_size;
|
|
if (current_length)
|
|
*current_length = buffer->_2d.plane_size;
|
|
}
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI d3d9_surface_buffer_Unlock(IMFMediaBuffer *iface)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE("%p.\n", iface);
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
if (!buffer->_2d.linear_buffer)
|
|
hr = HRESULT_FROM_WIN32(ERROR_WAS_UNLOCKED);
|
|
else if (!--buffer->_2d.locks)
|
|
{
|
|
D3DLOCKED_RECT rect;
|
|
|
|
if (SUCCEEDED(hr = IDirect3DSurface9_LockRect(buffer->d3d9_surface.surface, &rect, NULL, 0)))
|
|
{
|
|
copy_image(buffer, rect.pBits, rect.Pitch, buffer->_2d.linear_buffer, buffer->_2d.width,
|
|
buffer->_2d.width, buffer->_2d.height);
|
|
IDirect3DSurface9_UnlockRect(buffer->d3d9_surface.surface);
|
|
}
|
|
|
|
free(buffer->_2d.linear_buffer);
|
|
buffer->_2d.linear_buffer = NULL;
|
|
}
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI d3d9_surface_buffer_SetCurrentLength(IMFMediaBuffer *iface, DWORD current_length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
|
|
TRACE("%p, %lu.\n", iface, current_length);
|
|
|
|
buffer->current_length = current_length;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static const IMFMediaBufferVtbl d3d9_surface_1d_buffer_vtbl =
|
|
{
|
|
memory_1d_2d_buffer_QueryInterface,
|
|
memory_buffer_AddRef,
|
|
memory_buffer_Release,
|
|
d3d9_surface_buffer_Lock,
|
|
d3d9_surface_buffer_Unlock,
|
|
memory_buffer_GetCurrentLength,
|
|
d3d9_surface_buffer_SetCurrentLength,
|
|
memory_buffer_GetMaxLength,
|
|
};
|
|
|
|
static HRESULT WINAPI memory_2d_buffer_QueryInterface(IMF2DBuffer2 *iface, REFIID riid, void **obj)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
return IMFMediaBuffer_QueryInterface(&buffer->IMFMediaBuffer_iface, riid, obj);
|
|
}
|
|
|
|
static ULONG WINAPI memory_2d_buffer_AddRef(IMF2DBuffer2 *iface)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
return IMFMediaBuffer_AddRef(&buffer->IMFMediaBuffer_iface);
|
|
}
|
|
|
|
static ULONG WINAPI memory_2d_buffer_Release(IMF2DBuffer2 *iface)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
return IMFMediaBuffer_Release(&buffer->IMFMediaBuffer_iface);
|
|
}
|
|
|
|
static HRESULT memory_2d_buffer_lock(struct buffer *buffer, BYTE **scanline0, LONG *pitch,
|
|
BYTE **buffer_start, DWORD *buffer_length)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if (buffer->_2d.linear_buffer)
|
|
hr = MF_E_UNEXPECTED;
|
|
else
|
|
{
|
|
++buffer->_2d.locks;
|
|
*scanline0 = buffer->_2d.scanline0;
|
|
*pitch = buffer->_2d.pitch;
|
|
if (buffer_start)
|
|
*buffer_start = buffer->data;
|
|
if (buffer_length)
|
|
*buffer_length = buffer->max_length;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_2d_buffer_Lock2D(IMF2DBuffer2 *iface, BYTE **scanline0, LONG *pitch)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %p, %p.\n", iface, scanline0, pitch);
|
|
|
|
if (!scanline0 || !pitch)
|
|
return E_POINTER;
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
hr = memory_2d_buffer_lock(buffer, scanline0, pitch, NULL, NULL);
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_2d_buffer_Unlock2D(IMF2DBuffer2 *iface)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE("%p.\n", iface);
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
if (!buffer->_2d.linear_buffer)
|
|
{
|
|
if (buffer->_2d.locks)
|
|
--buffer->_2d.locks;
|
|
else
|
|
hr = HRESULT_FROM_WIN32(ERROR_WAS_UNLOCKED);
|
|
}
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_2d_buffer_GetScanline0AndPitch(IMF2DBuffer2 *iface, BYTE **scanline0, LONG *pitch)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE("%p, %p, %p.\n", iface, scanline0, pitch);
|
|
|
|
if (!scanline0 || !pitch)
|
|
return E_POINTER;
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
if (buffer->_2d.linear_buffer || !buffer->_2d.locks)
|
|
hr = HRESULT_FROM_WIN32(ERROR_WAS_UNLOCKED);
|
|
else
|
|
{
|
|
*scanline0 = buffer->_2d.scanline0;
|
|
*pitch = buffer->_2d.pitch;
|
|
}
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_2d_buffer_IsContiguousFormat(IMF2DBuffer2 *iface, BOOL *is_contiguous)
|
|
{
|
|
TRACE("%p, %p.\n", iface, is_contiguous);
|
|
|
|
if (!is_contiguous)
|
|
return E_POINTER;
|
|
|
|
*is_contiguous = FALSE;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_2d_buffer_GetContiguousLength(IMF2DBuffer2 *iface, DWORD *length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
|
|
TRACE("%p, %p.\n", iface, length);
|
|
|
|
if (!length)
|
|
return E_POINTER;
|
|
|
|
*length = buffer->_2d.plane_size;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_2d_buffer_ContiguousCopyTo(IMF2DBuffer2 *iface, BYTE *dest_buffer, DWORD dest_length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
BYTE *src_scanline0, *src_buffer_start;
|
|
DWORD src_length;
|
|
LONG src_pitch;
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %p, %lu.\n", iface, dest_buffer, dest_length);
|
|
|
|
if (dest_length < buffer->_2d.plane_size)
|
|
return E_INVALIDARG;
|
|
|
|
hr = IMF2DBuffer2_Lock2DSize(iface, MF2DBuffer_LockFlags_Read, &src_scanline0, &src_pitch, &src_buffer_start, &src_length);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (src_pitch < 0)
|
|
src_pitch = -src_pitch;
|
|
copy_image(buffer, dest_buffer, buffer->_2d.width, src_buffer_start, src_pitch,
|
|
buffer->_2d.width, buffer->_2d.height);
|
|
|
|
if (FAILED(IMF2DBuffer2_Unlock2D(iface)))
|
|
WARN("Couldn't unlock source buffer %p, hr %#lx.\n", iface, hr);
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_2d_buffer_ContiguousCopyFrom(IMF2DBuffer2 *iface, const BYTE *src_buffer, DWORD src_length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
BYTE *dst_scanline0, *dst_buffer_start;
|
|
DWORD dst_length;
|
|
LONG dst_pitch;
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %p, %lu.\n", iface, src_buffer, src_length);
|
|
|
|
if (src_length < buffer->_2d.plane_size)
|
|
return E_INVALIDARG;
|
|
|
|
hr = IMF2DBuffer2_Lock2DSize(iface, MF2DBuffer_LockFlags_Write, &dst_scanline0, &dst_pitch, &dst_buffer_start, &dst_length);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (dst_pitch < 0)
|
|
dst_pitch = -dst_pitch;
|
|
copy_image(buffer, dst_buffer_start, dst_pitch, src_buffer, buffer->_2d.width,
|
|
buffer->_2d.width, buffer->_2d.height);
|
|
|
|
if (FAILED(IMF2DBuffer2_Unlock2D(iface)))
|
|
WARN("Couldn't unlock destination buffer %p, hr %#lx.\n", iface, hr);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_2d_buffer_Lock2DSize(IMF2DBuffer2 *iface, MF2DBuffer_LockFlags flags, BYTE **scanline0,
|
|
LONG *pitch, BYTE **buffer_start, DWORD *buffer_length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %#x, %p, %p, %p, %p.\n", iface, flags, scanline0, pitch, buffer_start, buffer_length);
|
|
|
|
if (!scanline0 || !pitch || !buffer_start || !buffer_length)
|
|
return E_POINTER;
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
hr = memory_2d_buffer_lock(buffer, scanline0, pitch, buffer_start, buffer_length);
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI memory_2d_buffer_Copy2DTo(IMF2DBuffer2 *iface, IMF2DBuffer2 *dest_buffer)
|
|
{
|
|
FIXME("%p, %p.\n", iface, dest_buffer);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IMF2DBuffer2Vtbl memory_2d_buffer_vtbl =
|
|
{
|
|
memory_2d_buffer_QueryInterface,
|
|
memory_2d_buffer_AddRef,
|
|
memory_2d_buffer_Release,
|
|
memory_2d_buffer_Lock2D,
|
|
memory_2d_buffer_Unlock2D,
|
|
memory_2d_buffer_GetScanline0AndPitch,
|
|
memory_2d_buffer_IsContiguousFormat,
|
|
memory_2d_buffer_GetContiguousLength,
|
|
memory_2d_buffer_ContiguousCopyTo,
|
|
memory_2d_buffer_ContiguousCopyFrom,
|
|
memory_2d_buffer_Lock2DSize,
|
|
memory_2d_buffer_Copy2DTo,
|
|
};
|
|
|
|
static HRESULT d3d9_surface_buffer_lock(struct buffer *buffer, MF2DBuffer_LockFlags flags, BYTE **scanline0,
|
|
LONG *pitch, BYTE **buffer_start, DWORD *buffer_length)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if (buffer->_2d.linear_buffer)
|
|
hr = MF_E_UNEXPECTED;
|
|
else if (!buffer->_2d.locks)
|
|
hr = IDirect3DSurface9_LockRect(buffer->d3d9_surface.surface, &buffer->d3d9_surface.rect, NULL, 0);
|
|
else if (buffer->_2d.lock_flags == MF2DBuffer_LockFlags_Write && flags != MF2DBuffer_LockFlags_Write)
|
|
hr = HRESULT_FROM_WIN32(ERROR_WAS_LOCKED);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (!buffer->_2d.locks)
|
|
buffer->_2d.lock_flags = flags;
|
|
buffer->_2d.locks++;
|
|
*scanline0 = buffer->d3d9_surface.rect.pBits;
|
|
*pitch = buffer->d3d9_surface.rect.Pitch;
|
|
if (buffer_start)
|
|
*buffer_start = *scanline0;
|
|
if (buffer_length)
|
|
*buffer_length = buffer->d3d9_surface.rect.Pitch * buffer->_2d.height;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI d3d9_surface_buffer_Lock2D(IMF2DBuffer2 *iface, BYTE **scanline0, LONG *pitch)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %p, %p.\n", iface, scanline0, pitch);
|
|
|
|
if (!scanline0 || !pitch)
|
|
return E_POINTER;
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
hr = d3d9_surface_buffer_lock(buffer, MF2DBuffer_LockFlags_ReadWrite, scanline0, pitch, NULL, NULL);
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI d3d9_surface_buffer_Unlock2D(IMF2DBuffer2 *iface)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE("%p.\n", iface);
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
if (buffer->_2d.locks)
|
|
{
|
|
if (!--buffer->_2d.locks)
|
|
{
|
|
IDirect3DSurface9_UnlockRect(buffer->d3d9_surface.surface);
|
|
memset(&buffer->d3d9_surface.rect, 0, sizeof(buffer->d3d9_surface.rect));
|
|
buffer->_2d.lock_flags = 0;
|
|
}
|
|
}
|
|
else
|
|
hr = HRESULT_FROM_WIN32(ERROR_WAS_UNLOCKED);
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI d3d9_surface_buffer_GetScanline0AndPitch(IMF2DBuffer2 *iface, BYTE **scanline0, LONG *pitch)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE("%p, %p, %p.\n", iface, scanline0, pitch);
|
|
|
|
if (!scanline0 || !pitch)
|
|
return E_POINTER;
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
if (!buffer->_2d.locks)
|
|
{
|
|
*scanline0 = NULL;
|
|
*pitch = 0;
|
|
hr = HRESULT_FROM_WIN32(ERROR_WAS_UNLOCKED);
|
|
}
|
|
else
|
|
{
|
|
*scanline0 = buffer->d3d9_surface.rect.pBits;
|
|
*pitch = buffer->d3d9_surface.rect.Pitch;
|
|
}
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI d3d9_surface_buffer_Lock2DSize(IMF2DBuffer2 *iface, MF2DBuffer_LockFlags flags, BYTE **scanline0,
|
|
LONG *pitch, BYTE **buffer_start, DWORD *buffer_length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %#x, %p, %p, %p, %p.\n", iface, flags, scanline0, pitch, buffer_start, buffer_length);
|
|
|
|
if (!scanline0 || !pitch || !buffer_start || !buffer_length)
|
|
return E_POINTER;
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
hr = d3d9_surface_buffer_lock(buffer, flags, scanline0, pitch, buffer_start, buffer_length);
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static const IMF2DBuffer2Vtbl d3d9_surface_buffer_vtbl =
|
|
{
|
|
memory_2d_buffer_QueryInterface,
|
|
memory_2d_buffer_AddRef,
|
|
memory_2d_buffer_Release,
|
|
d3d9_surface_buffer_Lock2D,
|
|
d3d9_surface_buffer_Unlock2D,
|
|
d3d9_surface_buffer_GetScanline0AndPitch,
|
|
memory_2d_buffer_IsContiguousFormat,
|
|
memory_2d_buffer_GetContiguousLength,
|
|
memory_2d_buffer_ContiguousCopyTo,
|
|
memory_2d_buffer_ContiguousCopyFrom,
|
|
d3d9_surface_buffer_Lock2DSize,
|
|
memory_2d_buffer_Copy2DTo,
|
|
};
|
|
|
|
static HRESULT WINAPI memory_2d_buffer_gs_QueryInterface(IMFGetService *iface, REFIID riid, void **obj)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFGetService(iface);
|
|
return IMFMediaBuffer_QueryInterface(&buffer->IMFMediaBuffer_iface, riid, obj);
|
|
}
|
|
|
|
static ULONG WINAPI memory_2d_buffer_gs_AddRef(IMFGetService *iface)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFGetService(iface);
|
|
return IMFMediaBuffer_AddRef(&buffer->IMFMediaBuffer_iface);
|
|
}
|
|
|
|
static ULONG WINAPI memory_2d_buffer_gs_Release(IMFGetService *iface)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFGetService(iface);
|
|
return IMFMediaBuffer_Release(&buffer->IMFMediaBuffer_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI memory_2d_buffer_gs_GetService(IMFGetService *iface, REFGUID service, REFIID riid, void **obj)
|
|
{
|
|
TRACE("%p, %s, %s, %p.\n", iface, debugstr_guid(service), debugstr_guid(riid), obj);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IMFGetServiceVtbl memory_2d_buffer_gs_vtbl =
|
|
{
|
|
memory_2d_buffer_gs_QueryInterface,
|
|
memory_2d_buffer_gs_AddRef,
|
|
memory_2d_buffer_gs_Release,
|
|
memory_2d_buffer_gs_GetService,
|
|
};
|
|
|
|
static HRESULT WINAPI d3d9_surface_buffer_gs_GetService(IMFGetService *iface, REFGUID service, REFIID riid, void **obj)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFGetService(iface);
|
|
|
|
TRACE("%p, %s, %s, %p.\n", iface, debugstr_guid(service), debugstr_guid(riid), obj);
|
|
|
|
if (IsEqualGUID(service, &MR_BUFFER_SERVICE))
|
|
{
|
|
return IDirect3DSurface9_QueryInterface(buffer->d3d9_surface.surface, riid, obj);
|
|
}
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IMFGetServiceVtbl d3d9_surface_buffer_gs_vtbl =
|
|
{
|
|
memory_2d_buffer_gs_QueryInterface,
|
|
memory_2d_buffer_gs_AddRef,
|
|
memory_2d_buffer_gs_Release,
|
|
d3d9_surface_buffer_gs_GetService,
|
|
};
|
|
|
|
static HRESULT WINAPI dxgi_1d_2d_buffer_QueryInterface(IMFMediaBuffer *iface, REFIID riid, void **out)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
|
|
TRACE("%p, %s, %p.\n", iface, debugstr_guid(riid), out);
|
|
|
|
if (IsEqualIID(riid, &IID_IMFMediaBuffer) ||
|
|
IsEqualIID(riid, &IID_IUnknown))
|
|
{
|
|
*out = &buffer->IMFMediaBuffer_iface;
|
|
}
|
|
else if (IsEqualIID(riid, &IID_IMF2DBuffer2) ||
|
|
IsEqualIID(riid, &IID_IMF2DBuffer))
|
|
{
|
|
*out = &buffer->IMF2DBuffer2_iface;
|
|
}
|
|
else if (IsEqualIID(riid, &IID_IMFDXGIBuffer))
|
|
{
|
|
*out = &buffer->IMFDXGIBuffer_iface;
|
|
}
|
|
else
|
|
{
|
|
WARN("Unsupported interface %s.\n", debugstr_guid(riid));
|
|
*out = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*out);
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT dxgi_surface_buffer_create_readback_texture(struct buffer *buffer)
|
|
{
|
|
D3D11_TEXTURE2D_DESC texture_desc;
|
|
ID3D11Device *device;
|
|
HRESULT hr;
|
|
|
|
if (buffer->dxgi_surface.rb_texture)
|
|
return S_OK;
|
|
|
|
ID3D11Texture2D_GetDevice(buffer->dxgi_surface.texture, &device);
|
|
|
|
ID3D11Texture2D_GetDesc(buffer->dxgi_surface.texture, &texture_desc);
|
|
texture_desc.Usage = D3D11_USAGE_STAGING;
|
|
texture_desc.BindFlags = 0;
|
|
texture_desc.CPUAccessFlags = D3D11_CPU_ACCESS_READ | D3D11_CPU_ACCESS_WRITE;
|
|
texture_desc.MiscFlags = 0;
|
|
texture_desc.MipLevels = 1;
|
|
if (FAILED(hr = ID3D11Device_CreateTexture2D(device, &texture_desc, NULL, &buffer->dxgi_surface.rb_texture)))
|
|
WARN("Failed to create readback texture, hr %#lx.\n", hr);
|
|
|
|
ID3D11Device_Release(device);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT dxgi_surface_buffer_map(struct buffer *buffer, MF2DBuffer_LockFlags flags)
|
|
{
|
|
ID3D11DeviceContext *immediate_context;
|
|
ID3D11Device *device;
|
|
HRESULT hr;
|
|
|
|
if (FAILED(hr = dxgi_surface_buffer_create_readback_texture(buffer)))
|
|
return hr;
|
|
|
|
ID3D11Texture2D_GetDevice(buffer->dxgi_surface.texture, &device);
|
|
ID3D11Device_GetImmediateContext(device, &immediate_context);
|
|
|
|
if (flags == MF2DBuffer_LockFlags_Read || flags == MF2DBuffer_LockFlags_ReadWrite)
|
|
{
|
|
ID3D11DeviceContext_CopySubresourceRegion(immediate_context, (ID3D11Resource *)buffer->dxgi_surface.rb_texture,
|
|
0, 0, 0, 0, (ID3D11Resource *)buffer->dxgi_surface.texture, buffer->dxgi_surface.sub_resource_idx, NULL);
|
|
}
|
|
|
|
memset(&buffer->dxgi_surface.map_desc, 0, sizeof(buffer->dxgi_surface.map_desc));
|
|
if (FAILED(hr = ID3D11DeviceContext_Map(immediate_context, (ID3D11Resource *)buffer->dxgi_surface.rb_texture,
|
|
0, D3D11_MAP_READ_WRITE, 0, &buffer->dxgi_surface.map_desc)))
|
|
{
|
|
WARN("Failed to map readback texture, hr %#lx.\n", hr);
|
|
}
|
|
|
|
ID3D11DeviceContext_Release(immediate_context);
|
|
ID3D11Device_Release(device);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static void dxgi_surface_buffer_unmap(struct buffer *buffer, MF2DBuffer_LockFlags flags)
|
|
{
|
|
ID3D11DeviceContext *immediate_context;
|
|
ID3D11Device *device;
|
|
|
|
ID3D11Texture2D_GetDevice(buffer->dxgi_surface.texture, &device);
|
|
ID3D11Device_GetImmediateContext(device, &immediate_context);
|
|
ID3D11DeviceContext_Unmap(immediate_context, (ID3D11Resource *)buffer->dxgi_surface.rb_texture, 0);
|
|
memset(&buffer->dxgi_surface.map_desc, 0, sizeof(buffer->dxgi_surface.map_desc));
|
|
|
|
if (flags == MF2DBuffer_LockFlags_Write || flags == MF2DBuffer_LockFlags_ReadWrite)
|
|
{
|
|
ID3D11DeviceContext_CopySubresourceRegion(immediate_context, (ID3D11Resource *)buffer->dxgi_surface.texture,
|
|
buffer->dxgi_surface.sub_resource_idx, 0, 0, 0, (ID3D11Resource *)buffer->dxgi_surface.rb_texture, 0, NULL);
|
|
}
|
|
|
|
ID3D11DeviceContext_Release(immediate_context);
|
|
ID3D11Device_Release(device);
|
|
}
|
|
|
|
static HRESULT WINAPI dxgi_surface_buffer_Lock(IMFMediaBuffer *iface, BYTE **data, DWORD *max_length,
|
|
DWORD *current_length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE("%p, %p, %p, %p.\n", iface, data, max_length, current_length);
|
|
|
|
if (!data)
|
|
return E_POINTER;
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
if (!buffer->_2d.linear_buffer && buffer->_2d.locks)
|
|
hr = MF_E_INVALIDREQUEST;
|
|
else if (!buffer->_2d.linear_buffer)
|
|
{
|
|
if (!(buffer->_2d.linear_buffer = malloc(buffer->_2d.plane_size)))
|
|
hr = E_OUTOFMEMORY;
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = dxgi_surface_buffer_map(buffer, MF2DBuffer_LockFlags_ReadWrite);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
copy_image(buffer, buffer->_2d.linear_buffer, buffer->_2d.width, buffer->dxgi_surface.map_desc.pData,
|
|
buffer->dxgi_surface.map_desc.RowPitch, buffer->_2d.width, buffer->_2d.height);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
++buffer->_2d.locks;
|
|
*data = buffer->_2d.linear_buffer;
|
|
if (max_length)
|
|
*max_length = buffer->_2d.plane_size;
|
|
if (current_length)
|
|
*current_length = buffer->_2d.plane_size;
|
|
}
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI dxgi_surface_buffer_Unlock(IMFMediaBuffer *iface)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE("%p.\n", iface);
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
if (!buffer->_2d.linear_buffer)
|
|
hr = HRESULT_FROM_WIN32(ERROR_WAS_UNLOCKED);
|
|
else if (!--buffer->_2d.locks)
|
|
{
|
|
copy_image(buffer, buffer->dxgi_surface.map_desc.pData, buffer->dxgi_surface.map_desc.RowPitch,
|
|
buffer->_2d.linear_buffer, buffer->_2d.width, buffer->_2d.width, buffer->_2d.height);
|
|
dxgi_surface_buffer_unmap(buffer, MF2DBuffer_LockFlags_ReadWrite);
|
|
|
|
free(buffer->_2d.linear_buffer);
|
|
buffer->_2d.linear_buffer = NULL;
|
|
}
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI dxgi_surface_buffer_SetCurrentLength(IMFMediaBuffer *iface, DWORD current_length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFMediaBuffer(iface);
|
|
|
|
TRACE("%p, %lu.\n", iface, current_length);
|
|
|
|
buffer->current_length = current_length;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT dxgi_surface_buffer_lock(struct buffer *buffer, MF2DBuffer_LockFlags flags,
|
|
BYTE **scanline0, LONG *pitch, BYTE **buffer_start, DWORD *buffer_length)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if (buffer->_2d.linear_buffer)
|
|
hr = MF_E_UNEXPECTED;
|
|
else if (!buffer->_2d.locks)
|
|
hr = dxgi_surface_buffer_map(buffer, flags);
|
|
else if (buffer->_2d.lock_flags == MF2DBuffer_LockFlags_Write && flags != MF2DBuffer_LockFlags_Write)
|
|
hr = HRESULT_FROM_WIN32(ERROR_WAS_LOCKED);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (!buffer->_2d.locks)
|
|
buffer->_2d.lock_flags = flags;
|
|
else
|
|
buffer->_2d.lock_flags |= flags;
|
|
buffer->_2d.locks++;
|
|
*scanline0 = buffer->dxgi_surface.map_desc.pData;
|
|
*pitch = buffer->dxgi_surface.map_desc.RowPitch;
|
|
if (buffer_start)
|
|
*buffer_start = *scanline0;
|
|
if (buffer_length)
|
|
*buffer_length = buffer->dxgi_surface.map_desc.DepthPitch;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI dxgi_surface_buffer_Lock2D(IMF2DBuffer2 *iface, BYTE **scanline0, LONG *pitch)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %p, %p.\n", iface, scanline0, pitch);
|
|
|
|
if (!scanline0 || !pitch)
|
|
return E_POINTER;
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
hr = dxgi_surface_buffer_lock(buffer, MF2DBuffer_LockFlags_ReadWrite, scanline0, pitch, NULL, NULL);
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI dxgi_surface_buffer_Unlock2D(IMF2DBuffer2 *iface)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE("%p.\n", iface);
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
if (buffer->_2d.locks)
|
|
{
|
|
if (!--buffer->_2d.locks)
|
|
{
|
|
dxgi_surface_buffer_unmap(buffer, buffer->_2d.lock_flags);
|
|
buffer->_2d.lock_flags = 0;
|
|
}
|
|
}
|
|
else
|
|
hr = HRESULT_FROM_WIN32(ERROR_WAS_UNLOCKED);
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI dxgi_surface_buffer_GetScanline0AndPitch(IMF2DBuffer2 *iface, BYTE **scanline0, LONG *pitch)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE("%p, %p, %p.\n", iface, scanline0, pitch);
|
|
|
|
if (!scanline0 || !pitch)
|
|
return E_POINTER;
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
if (!buffer->_2d.locks)
|
|
{
|
|
*scanline0 = NULL;
|
|
*pitch = 0;
|
|
hr = HRESULT_FROM_WIN32(ERROR_WAS_UNLOCKED);
|
|
}
|
|
else
|
|
{
|
|
*scanline0 = buffer->dxgi_surface.map_desc.pData;
|
|
*pitch = buffer->dxgi_surface.map_desc.RowPitch;
|
|
}
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI dxgi_surface_buffer_Lock2DSize(IMF2DBuffer2 *iface, MF2DBuffer_LockFlags flags,
|
|
BYTE **scanline0, LONG *pitch, BYTE **buffer_start, DWORD *buffer_length)
|
|
{
|
|
struct buffer *buffer = impl_from_IMF2DBuffer2(iface);
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %#x, %p, %p, %p, %p.\n", iface, flags, scanline0, pitch, buffer_start, buffer_length);
|
|
|
|
if (!scanline0 || !pitch || !buffer_start || !buffer_length)
|
|
return E_POINTER;
|
|
|
|
EnterCriticalSection(&buffer->cs);
|
|
|
|
hr = dxgi_surface_buffer_lock(buffer, flags, scanline0, pitch, buffer_start, buffer_length);
|
|
|
|
LeaveCriticalSection(&buffer->cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI dxgi_buffer_QueryInterface(IMFDXGIBuffer *iface, REFIID riid, void **obj)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFDXGIBuffer(iface);
|
|
return IMFMediaBuffer_QueryInterface(&buffer->IMFMediaBuffer_iface, riid, obj);
|
|
}
|
|
|
|
static ULONG WINAPI dxgi_buffer_AddRef(IMFDXGIBuffer *iface)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFDXGIBuffer(iface);
|
|
return IMFMediaBuffer_AddRef(&buffer->IMFMediaBuffer_iface);
|
|
}
|
|
|
|
static ULONG WINAPI dxgi_buffer_Release(IMFDXGIBuffer *iface)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFDXGIBuffer(iface);
|
|
return IMFMediaBuffer_Release(&buffer->IMFMediaBuffer_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI dxgi_buffer_GetResource(IMFDXGIBuffer *iface, REFIID riid, void **obj)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFDXGIBuffer(iface);
|
|
|
|
TRACE("%p, %s, %p.\n", iface, debugstr_guid(riid), obj);
|
|
|
|
return ID3D11Texture2D_QueryInterface(buffer->dxgi_surface.texture, riid, obj);
|
|
}
|
|
|
|
static HRESULT WINAPI dxgi_buffer_GetSubresourceIndex(IMFDXGIBuffer *iface, UINT *index)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFDXGIBuffer(iface);
|
|
|
|
TRACE("%p, %p.\n", iface, index);
|
|
|
|
if (!index)
|
|
return E_POINTER;
|
|
|
|
*index = buffer->dxgi_surface.sub_resource_idx;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI dxgi_buffer_GetUnknown(IMFDXGIBuffer *iface, REFIID guid, REFIID riid, void **object)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFDXGIBuffer(iface);
|
|
|
|
TRACE("%p, %s, %s, %p.\n", iface, debugstr_guid(guid), debugstr_guid(riid), object);
|
|
|
|
if (attributes_GetUnknown(&buffer->dxgi_surface.attributes, guid, riid, object) == MF_E_ATTRIBUTENOTFOUND)
|
|
return MF_E_NOT_FOUND;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI dxgi_buffer_SetUnknown(IMFDXGIBuffer *iface, REFIID guid, IUnknown *data)
|
|
{
|
|
struct buffer *buffer = impl_from_IMFDXGIBuffer(iface);
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE("%p, %s, %p.\n", iface, debugstr_guid(guid), data);
|
|
|
|
EnterCriticalSection(&buffer->dxgi_surface.attributes.cs);
|
|
if (data)
|
|
{
|
|
if (SUCCEEDED(attributes_GetItem(&buffer->dxgi_surface.attributes, guid, NULL)))
|
|
hr = HRESULT_FROM_WIN32(ERROR_OBJECT_ALREADY_EXISTS);
|
|
else
|
|
hr = attributes_SetUnknown(&buffer->dxgi_surface.attributes, guid, data);
|
|
}
|
|
else
|
|
{
|
|
attributes_DeleteItem(&buffer->dxgi_surface.attributes, guid);
|
|
}
|
|
LeaveCriticalSection(&buffer->dxgi_surface.attributes.cs);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static const IMFMediaBufferVtbl dxgi_surface_1d_buffer_vtbl =
|
|
{
|
|
dxgi_1d_2d_buffer_QueryInterface,
|
|
memory_buffer_AddRef,
|
|
memory_buffer_Release,
|
|
dxgi_surface_buffer_Lock,
|
|
dxgi_surface_buffer_Unlock,
|
|
memory_buffer_GetCurrentLength,
|
|
dxgi_surface_buffer_SetCurrentLength,
|
|
memory_buffer_GetMaxLength,
|
|
};
|
|
|
|
static const IMF2DBuffer2Vtbl dxgi_surface_buffer_vtbl =
|
|
{
|
|
memory_2d_buffer_QueryInterface,
|
|
memory_2d_buffer_AddRef,
|
|
memory_2d_buffer_Release,
|
|
dxgi_surface_buffer_Lock2D,
|
|
dxgi_surface_buffer_Unlock2D,
|
|
dxgi_surface_buffer_GetScanline0AndPitch,
|
|
memory_2d_buffer_IsContiguousFormat,
|
|
memory_2d_buffer_GetContiguousLength,
|
|
memory_2d_buffer_ContiguousCopyTo,
|
|
memory_2d_buffer_ContiguousCopyFrom,
|
|
dxgi_surface_buffer_Lock2DSize,
|
|
memory_2d_buffer_Copy2DTo,
|
|
};
|
|
|
|
static const IMFDXGIBufferVtbl dxgi_buffer_vtbl =
|
|
{
|
|
dxgi_buffer_QueryInterface,
|
|
dxgi_buffer_AddRef,
|
|
dxgi_buffer_Release,
|
|
dxgi_buffer_GetResource,
|
|
dxgi_buffer_GetSubresourceIndex,
|
|
dxgi_buffer_GetUnknown,
|
|
dxgi_buffer_SetUnknown,
|
|
};
|
|
|
|
static HRESULT memory_buffer_init(struct buffer *buffer, DWORD max_length, DWORD alignment,
|
|
const IMFMediaBufferVtbl *vtbl)
|
|
{
|
|
if (alignment < MF_16_BYTE_ALIGNMENT)
|
|
alignment = MF_16_BYTE_ALIGNMENT;
|
|
alignment++;
|
|
|
|
if (alignment & (alignment - 1))
|
|
{
|
|
alignment--;
|
|
alignment |= alignment >> 1;
|
|
alignment |= alignment >> 2;
|
|
alignment |= alignment >> 4;
|
|
alignment |= alignment >> 8;
|
|
alignment |= alignment >> 16;
|
|
alignment++;
|
|
}
|
|
|
|
if (!(buffer->data = _aligned_malloc(max_length, alignment)))
|
|
return E_OUTOFMEMORY;
|
|
memset(buffer->data, 0, max_length);
|
|
|
|
buffer->IMFMediaBuffer_iface.lpVtbl = vtbl;
|
|
buffer->refcount = 1;
|
|
buffer->max_length = max_length;
|
|
buffer->current_length = 0;
|
|
InitializeCriticalSection(&buffer->cs);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT create_1d_buffer(DWORD max_length, DWORD alignment, IMFMediaBuffer **buffer)
|
|
{
|
|
struct buffer *object;
|
|
HRESULT hr;
|
|
|
|
if (!buffer)
|
|
return E_POINTER;
|
|
|
|
*buffer = NULL;
|
|
|
|
if (!(object = calloc(1, sizeof(*object))))
|
|
return E_OUTOFMEMORY;
|
|
|
|
hr = memory_buffer_init(object, max_length, alignment, &memory_1d_buffer_vtbl);
|
|
if (FAILED(hr))
|
|
{
|
|
free(object);
|
|
return hr;
|
|
}
|
|
|
|
*buffer = &object->IMFMediaBuffer_iface;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static p_copy_image_func get_2d_buffer_copy_func(DWORD fourcc)
|
|
{
|
|
switch (fourcc)
|
|
{
|
|
case MAKEFOURCC('I','M','C','1'):
|
|
case MAKEFOURCC('I','M','C','3'):
|
|
return copy_image_imc1;
|
|
|
|
case MAKEFOURCC('I','M','C','2'):
|
|
case MAKEFOURCC('I','M','C','4'):
|
|
case MAKEFOURCC('N','V','1','1'):
|
|
case MAKEFOURCC('Y','V','1','2'):
|
|
case MAKEFOURCC('I','4','2','0'):
|
|
case MAKEFOURCC('I','Y','U','V'):
|
|
return copy_image_imc2;
|
|
|
|
case MAKEFOURCC('N','V','1','2'):
|
|
case MAKEFOURCC('P','0','1','0'):
|
|
return copy_image_nv12;
|
|
|
|
default:
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
static HRESULT create_2d_buffer(DWORD width, DWORD height, DWORD fourcc, BOOL bottom_up, IMFMediaBuffer **buffer)
|
|
{
|
|
unsigned int stride, max_length;
|
|
unsigned int row_alignment;
|
|
struct buffer *object;
|
|
DWORD plane_size;
|
|
GUID subtype;
|
|
BOOL is_yuv;
|
|
HRESULT hr;
|
|
int pitch;
|
|
|
|
if (!buffer)
|
|
return E_POINTER;
|
|
|
|
*buffer = NULL;
|
|
|
|
memcpy(&subtype, &MFVideoFormat_Base, sizeof(subtype));
|
|
subtype.Data1 = fourcc;
|
|
|
|
if (!(stride = mf_format_get_stride(&subtype, width, &is_yuv)))
|
|
return MF_E_INVALIDMEDIATYPE;
|
|
|
|
if (is_yuv && bottom_up)
|
|
return MF_E_INVALIDMEDIATYPE;
|
|
|
|
switch (fourcc)
|
|
{
|
|
case MAKEFOURCC('I','M','C','1'):
|
|
case MAKEFOURCC('I','M','C','3'):
|
|
plane_size = stride * height * 2;
|
|
break;
|
|
case MAKEFOURCC('I','M','C','2'):
|
|
case MAKEFOURCC('I','M','C','4'):
|
|
case MAKEFOURCC('N','V','1','1'):
|
|
case MAKEFOURCC('Y','V','1','2'):
|
|
case MAKEFOURCC('I','4','2','0'):
|
|
case MAKEFOURCC('I','Y','U','V'):
|
|
plane_size = stride * 3 / 2 * height;
|
|
break;
|
|
case MAKEFOURCC('N','V','1','2'):
|
|
plane_size = stride * height * 3 / 2;
|
|
break;
|
|
case MAKEFOURCC('P','0','1','0'):
|
|
plane_size = width * height * 3;
|
|
break;
|
|
default:
|
|
plane_size = stride * height;
|
|
}
|
|
|
|
if (!(object = calloc(1, sizeof(*object))))
|
|
return E_OUTOFMEMORY;
|
|
|
|
switch (fourcc)
|
|
{
|
|
case MAKEFOURCC('I','M','C','1'):
|
|
case MAKEFOURCC('I','M','C','2'):
|
|
case MAKEFOURCC('I','M','C','3'):
|
|
case MAKEFOURCC('I','M','C','4'):
|
|
case MAKEFOURCC('Y','V','1','2'):
|
|
case MAKEFOURCC('N','V','1','1'):
|
|
case MAKEFOURCC('I','4','2','0'):
|
|
case MAKEFOURCC('I','Y','U','V'):
|
|
row_alignment = MF_128_BYTE_ALIGNMENT;
|
|
break;
|
|
default:
|
|
row_alignment = MF_64_BYTE_ALIGNMENT;
|
|
}
|
|
|
|
pitch = ALIGN_SIZE(stride, row_alignment);
|
|
|
|
switch (fourcc)
|
|
{
|
|
case MAKEFOURCC('I','M','C','1'):
|
|
case MAKEFOURCC('I','M','C','3'):
|
|
max_length = pitch * height * 2;
|
|
break;
|
|
case MAKEFOURCC('N','V','1','2'):
|
|
case MAKEFOURCC('Y','V','1','2'):
|
|
case MAKEFOURCC('I','M','C','2'):
|
|
case MAKEFOURCC('I','M','C','4'):
|
|
case MAKEFOURCC('N','V','1','1'):
|
|
case MAKEFOURCC('I','4','2','0'):
|
|
case MAKEFOURCC('I','Y','U','V'):
|
|
case MAKEFOURCC('P','0','1','0'):
|
|
max_length = pitch * height * 3 / 2;
|
|
break;
|
|
default:
|
|
max_length = pitch * height;
|
|
}
|
|
|
|
if (FAILED(hr = memory_buffer_init(object, max_length, row_alignment, &memory_1d_2d_buffer_vtbl)))
|
|
{
|
|
free(object);
|
|
return hr;
|
|
}
|
|
|
|
object->IMF2DBuffer2_iface.lpVtbl = &memory_2d_buffer_vtbl;
|
|
object->IMFGetService_iface.lpVtbl = &memory_2d_buffer_gs_vtbl;
|
|
object->_2d.plane_size = plane_size;
|
|
object->_2d.width = stride;
|
|
object->_2d.height = height;
|
|
object->_2d.pitch = bottom_up ? -pitch : pitch;
|
|
object->_2d.scanline0 = bottom_up ? object->data + pitch * (object->_2d.height - 1) : object->data;
|
|
object->_2d.copy_image = get_2d_buffer_copy_func(fourcc);
|
|
|
|
*buffer = &object->IMFMediaBuffer_iface;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT create_d3d9_surface_buffer(IUnknown *surface, BOOL bottom_up, IMFMediaBuffer **buffer)
|
|
{
|
|
struct buffer *object;
|
|
D3DSURFACE_DESC desc;
|
|
unsigned int stride;
|
|
GUID subtype;
|
|
BOOL is_yuv;
|
|
|
|
IDirect3DSurface9_GetDesc((IDirect3DSurface9 *)surface, &desc);
|
|
TRACE("format %#x, %u x %u.\n", desc.Format, desc.Width, desc.Height);
|
|
|
|
memcpy(&subtype, &MFVideoFormat_Base, sizeof(subtype));
|
|
subtype.Data1 = desc.Format;
|
|
|
|
if (!(stride = mf_format_get_stride(&subtype, desc.Width, &is_yuv)))
|
|
return MF_E_INVALIDMEDIATYPE;
|
|
|
|
if (!(object = calloc(1, sizeof(*object))))
|
|
return E_OUTOFMEMORY;
|
|
|
|
object->IMFMediaBuffer_iface.lpVtbl = &d3d9_surface_1d_buffer_vtbl;
|
|
object->IMF2DBuffer2_iface.lpVtbl = &d3d9_surface_buffer_vtbl;
|
|
object->IMFGetService_iface.lpVtbl = &d3d9_surface_buffer_gs_vtbl;
|
|
object->refcount = 1;
|
|
InitializeCriticalSection(&object->cs);
|
|
object->d3d9_surface.surface = (IDirect3DSurface9 *)surface;
|
|
IUnknown_AddRef(surface);
|
|
|
|
MFGetPlaneSize(desc.Format, desc.Width, desc.Height, &object->_2d.plane_size);
|
|
object->_2d.width = stride;
|
|
object->_2d.height = desc.Height;
|
|
object->max_length = object->_2d.plane_size;
|
|
object->_2d.copy_image = get_2d_buffer_copy_func(desc.Format);
|
|
|
|
*buffer = &object->IMFMediaBuffer_iface;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT create_dxgi_surface_buffer(IUnknown *surface, unsigned int sub_resource_idx,
|
|
BOOL bottom_up, IMFMediaBuffer **buffer)
|
|
{
|
|
struct buffer *object;
|
|
D3D11_TEXTURE2D_DESC desc;
|
|
ID3D11Texture2D *texture;
|
|
unsigned int stride;
|
|
D3DFORMAT format;
|
|
GUID subtype;
|
|
BOOL is_yuv;
|
|
HRESULT hr;
|
|
|
|
if (FAILED(hr = IUnknown_QueryInterface(surface, &IID_ID3D11Texture2D, (void **)&texture)))
|
|
{
|
|
WARN("Failed to get texture interface, hr %#lx.\n", hr);
|
|
return hr;
|
|
}
|
|
|
|
ID3D11Texture2D_GetDesc(texture, &desc);
|
|
TRACE("format %#x, %u x %u.\n", desc.Format, desc.Width, desc.Height);
|
|
|
|
memcpy(&subtype, &MFVideoFormat_Base, sizeof(subtype));
|
|
subtype.Data1 = format = MFMapDXGIFormatToDX9Format(desc.Format);
|
|
|
|
if (!(stride = mf_format_get_stride(&subtype, desc.Width, &is_yuv)))
|
|
{
|
|
ID3D11Texture2D_Release(texture);
|
|
return MF_E_INVALIDMEDIATYPE;
|
|
}
|
|
|
|
if (!(object = calloc(1, sizeof(*object))))
|
|
{
|
|
ID3D11Texture2D_Release(texture);
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
object->IMFMediaBuffer_iface.lpVtbl = &dxgi_surface_1d_buffer_vtbl;
|
|
object->IMF2DBuffer2_iface.lpVtbl = &dxgi_surface_buffer_vtbl;
|
|
object->IMFDXGIBuffer_iface.lpVtbl = &dxgi_buffer_vtbl;
|
|
object->refcount = 1;
|
|
InitializeCriticalSection(&object->cs);
|
|
object->dxgi_surface.texture = texture;
|
|
object->dxgi_surface.sub_resource_idx = sub_resource_idx;
|
|
|
|
MFGetPlaneSize(format, desc.Width, desc.Height, &object->_2d.plane_size);
|
|
object->_2d.width = stride;
|
|
object->_2d.height = desc.Height;
|
|
object->max_length = object->_2d.plane_size;
|
|
object->_2d.copy_image = get_2d_buffer_copy_func(format);
|
|
|
|
if (FAILED(hr = init_attributes_object(&object->dxgi_surface.attributes, 0)))
|
|
{
|
|
IMFMediaBuffer_Release(&object->IMFMediaBuffer_iface);
|
|
return hr;
|
|
}
|
|
|
|
*buffer = &object->IMFMediaBuffer_iface;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MFCreateMemoryBuffer (mfplat.@)
|
|
*/
|
|
HRESULT WINAPI MFCreateMemoryBuffer(DWORD max_length, IMFMediaBuffer **buffer)
|
|
{
|
|
TRACE("%lu, %p.\n", max_length, buffer);
|
|
|
|
return create_1d_buffer(max_length, MF_1_BYTE_ALIGNMENT, buffer);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MFCreateAlignedMemoryBuffer (mfplat.@)
|
|
*/
|
|
HRESULT WINAPI MFCreateAlignedMemoryBuffer(DWORD max_length, DWORD alignment, IMFMediaBuffer **buffer)
|
|
{
|
|
TRACE("%lu, %lu, %p.\n", max_length, alignment, buffer);
|
|
|
|
return create_1d_buffer(max_length, alignment, buffer);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MFCreate2DMediaBuffer (mfplat.@)
|
|
*/
|
|
HRESULT WINAPI MFCreate2DMediaBuffer(DWORD width, DWORD height, DWORD fourcc, BOOL bottom_up, IMFMediaBuffer **buffer)
|
|
{
|
|
TRACE("%lu, %lu, %s, %d, %p.\n", width, height, mf_debugstr_fourcc(fourcc), bottom_up, buffer);
|
|
|
|
return create_2d_buffer(width, height, fourcc, bottom_up, buffer);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MFCreateDXSurfaceBuffer (mfplat.@)
|
|
*/
|
|
HRESULT WINAPI MFCreateDXSurfaceBuffer(REFIID riid, IUnknown *surface, BOOL bottom_up, IMFMediaBuffer **buffer)
|
|
{
|
|
TRACE("%s, %p, %d, %p.\n", debugstr_guid(riid), surface, bottom_up, buffer);
|
|
|
|
if (!IsEqualIID(riid, &IID_IDirect3DSurface9))
|
|
return E_INVALIDARG;
|
|
|
|
return create_d3d9_surface_buffer(surface, bottom_up, buffer);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MFCreateDXGISurfaceBuffer (mfplat.@)
|
|
*/
|
|
HRESULT WINAPI MFCreateDXGISurfaceBuffer(REFIID riid, IUnknown *surface, UINT subresource, BOOL bottom_up,
|
|
IMFMediaBuffer **buffer)
|
|
{
|
|
TRACE("%s, %p, %u, %d, %p.\n", debugstr_guid(riid), surface, subresource, bottom_up, buffer);
|
|
|
|
if (!IsEqualIID(riid, &IID_ID3D11Texture2D))
|
|
return E_INVALIDARG;
|
|
|
|
return create_dxgi_surface_buffer(surface, subresource, bottom_up, buffer);
|
|
}
|
|
|
|
static unsigned int buffer_get_aligned_length(unsigned int length, unsigned int alignment)
|
|
{
|
|
length = (length + alignment) / alignment;
|
|
length *= alignment;
|
|
|
|
return length;
|
|
}
|
|
|
|
HRESULT WINAPI MFCreateMediaBufferFromMediaType(IMFMediaType *media_type, LONGLONG duration, DWORD min_length,
|
|
DWORD alignment, IMFMediaBuffer **buffer)
|
|
{
|
|
UINT32 length = 0, block_alignment;
|
|
LONGLONG avg_length;
|
|
GUID major, subtype;
|
|
UINT64 frame_size;
|
|
BOOL is_yuv;
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, %s, %lu, %lu, %p.\n", media_type, debugstr_time(duration), min_length, alignment, buffer);
|
|
|
|
if (!media_type)
|
|
return E_INVALIDARG;
|
|
|
|
if (FAILED(hr = IMFMediaType_GetMajorType(media_type, &major)))
|
|
return hr;
|
|
|
|
if (IsEqualGUID(&major, &MFMediaType_Audio))
|
|
{
|
|
block_alignment = 0;
|
|
if (FAILED(IMFMediaType_GetUINT32(media_type, &MF_MT_AUDIO_BLOCK_ALIGNMENT, &block_alignment)))
|
|
WARN("Block alignment was not specified.\n");
|
|
|
|
alignment = max(16, alignment);
|
|
|
|
if (block_alignment)
|
|
{
|
|
avg_length = 0;
|
|
|
|
if (duration)
|
|
{
|
|
length = 0;
|
|
if (SUCCEEDED(IMFMediaType_GetUINT32(media_type, &MF_MT_AUDIO_AVG_BYTES_PER_SECOND, &length)))
|
|
{
|
|
/* 100 ns -> 1 s */
|
|
avg_length = length * duration / (10 * 1000 * 1000);
|
|
}
|
|
}
|
|
|
|
length = buffer_get_aligned_length(avg_length + 1, alignment);
|
|
length = buffer_get_aligned_length(length, block_alignment);
|
|
}
|
|
else
|
|
length = 0;
|
|
|
|
length = max(length, min_length);
|
|
|
|
return create_1d_buffer(length, alignment - 1, buffer);
|
|
}
|
|
|
|
if (IsEqualGUID(&major, &MFMediaType_Video))
|
|
{
|
|
if (SUCCEEDED(hr = IMFMediaType_GetGUID(media_type, &MF_MT_SUBTYPE, &subtype))
|
|
&& SUCCEEDED(hr = IMFMediaType_GetUINT64(media_type, &MF_MT_FRAME_SIZE, &frame_size))
|
|
&& mf_format_get_stride(&subtype, frame_size >> 32, &is_yuv))
|
|
{
|
|
BOOL bottom_up = FALSE;
|
|
UINT32 stride;
|
|
|
|
if (!is_yuv && SUCCEEDED(IMFMediaType_GetUINT32(media_type, &MF_MT_DEFAULT_STRIDE, &stride)))
|
|
bottom_up = (int)stride < 0;
|
|
|
|
if (SUCCEEDED(hr = create_2d_buffer(frame_size >> 32, (UINT32)frame_size, subtype.Data1, bottom_up, buffer)))
|
|
return hr;
|
|
}
|
|
|
|
if (!min_length)
|
|
return FAILED(hr) ? hr : E_INVALIDARG;
|
|
|
|
alignment = max(16, alignment);
|
|
return create_1d_buffer(min_length, alignment - 1, buffer);
|
|
}
|
|
|
|
FIXME("Major type %s is not supported.\n", debugstr_guid(&major));
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
struct legacy_buffer
|
|
{
|
|
IMediaBuffer IMediaBuffer_iface;
|
|
IMF2DBuffer2 IMF2DBuffer2_iface;
|
|
IMFSample IMFSample_iface;
|
|
LONG refcount;
|
|
|
|
IMFMediaBuffer *media_buffer;
|
|
void *locked_data;
|
|
|
|
IMFSample *sample;
|
|
IMF2DBuffer2 *buffer_2d;
|
|
};
|
|
|
|
static struct legacy_buffer *legacy_buffer_from_IMediaBuffer(IMediaBuffer *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct legacy_buffer, IMediaBuffer_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_QueryInterface(IMediaBuffer *iface, REFIID iid, void **obj)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMediaBuffer(iface);
|
|
|
|
TRACE("iface %p, iid %s, obj %p.\n", iface, debugstr_guid(iid), obj);
|
|
|
|
if (IsEqualGUID(iid, &IID_IUnknown)
|
|
|| IsEqualGUID(iid, &IID_IMediaBuffer))
|
|
{
|
|
IMediaBuffer_AddRef(&buffer->IMediaBuffer_iface);
|
|
*obj = &buffer->IMediaBuffer_iface;
|
|
return S_OK;
|
|
}
|
|
|
|
if (buffer->sample &&
|
|
IsEqualGUID(iid, &IID_IMFSample))
|
|
{
|
|
IMFSample_AddRef(&buffer->IMFSample_iface);
|
|
*obj = &buffer->IMFSample_iface;
|
|
return S_OK;
|
|
}
|
|
|
|
if (buffer->buffer_2d &&
|
|
(IsEqualGUID(iid, &IID_IMF2DBuffer)
|
|
|| IsEqualGUID(iid, &IID_IMF2DBuffer2)))
|
|
{
|
|
IMF2DBuffer2_AddRef(&buffer->IMF2DBuffer2_iface);
|
|
*obj = &buffer->IMF2DBuffer2_iface;
|
|
return S_OK;
|
|
}
|
|
|
|
WARN("%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid(iid));
|
|
*obj = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
static ULONG WINAPI legacy_buffer_AddRef(IMediaBuffer *iface)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMediaBuffer(iface);
|
|
ULONG refcount = InterlockedIncrement(&buffer->refcount);
|
|
TRACE("%p, refcount %lu.\n", buffer, refcount);
|
|
return refcount;
|
|
}
|
|
|
|
static ULONG WINAPI legacy_buffer_Release(IMediaBuffer *iface)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMediaBuffer(iface);
|
|
ULONG refcount = InterlockedDecrement(&buffer->refcount);
|
|
TRACE("%p, refcount %lu.\n", buffer, refcount);
|
|
|
|
if (!refcount)
|
|
{
|
|
if (buffer->locked_data)
|
|
IMFMediaBuffer_Unlock(buffer->media_buffer);
|
|
if (buffer->sample)
|
|
IMFSample_Release(buffer->sample);
|
|
if (buffer->buffer_2d)
|
|
IMF2DBuffer2_Release(buffer->buffer_2d);
|
|
IMFMediaBuffer_Release(buffer->media_buffer);
|
|
free(buffer);
|
|
}
|
|
|
|
return refcount;
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_SetLength(IMediaBuffer *iface, DWORD len)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMediaBuffer(iface);
|
|
TRACE("%p, len %lu.\n", buffer, len);
|
|
return IMFMediaBuffer_SetCurrentLength(buffer->media_buffer, len);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_GetMaxLength(IMediaBuffer *iface, DWORD *len)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMediaBuffer(iface);
|
|
TRACE("%p, len %p.\n", buffer, len);
|
|
return IMFMediaBuffer_GetMaxLength(buffer->media_buffer, len);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_GetBufferAndLength(IMediaBuffer *iface, BYTE **data, DWORD *len)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMediaBuffer(iface);
|
|
HRESULT hr;
|
|
|
|
TRACE("%p, data %p, len %p.\n", buffer, data, len);
|
|
|
|
if ((*data = InterlockedCompareExchangePointer(&buffer->locked_data, NULL, NULL)))
|
|
return IMFMediaBuffer_GetCurrentLength(buffer->media_buffer, len);
|
|
|
|
if (SUCCEEDED(hr = IMFMediaBuffer_Lock(buffer->media_buffer, data, NULL, len)))
|
|
InterlockedExchangePointer(&buffer->locked_data, *data);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static const IMediaBufferVtbl legacy_buffer_vtbl =
|
|
{
|
|
legacy_buffer_QueryInterface,
|
|
legacy_buffer_AddRef,
|
|
legacy_buffer_Release,
|
|
legacy_buffer_SetLength,
|
|
legacy_buffer_GetMaxLength,
|
|
legacy_buffer_GetBufferAndLength,
|
|
};
|
|
|
|
static struct legacy_buffer *legacy_buffer_from_IMF2DBuffer2(IMF2DBuffer2 *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct legacy_buffer, IMF2DBuffer2_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMF2DBuffer2_QueryInterface(IMF2DBuffer2 *iface, REFIID riid, void **obj)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMF2DBuffer2(iface);
|
|
return IMediaBuffer_QueryInterface(&buffer->IMediaBuffer_iface, riid, obj);
|
|
}
|
|
|
|
static ULONG WINAPI legacy_buffer_IMF2DBuffer2_AddRef(IMF2DBuffer2 *iface)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMF2DBuffer2(iface);
|
|
return IMediaBuffer_AddRef(&buffer->IMediaBuffer_iface);
|
|
}
|
|
|
|
static ULONG WINAPI legacy_buffer_IMF2DBuffer2_Release(IMF2DBuffer2 *iface)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMF2DBuffer2(iface);
|
|
return IMediaBuffer_Release(&buffer->IMediaBuffer_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMF2DBuffer2_Lock2D(IMF2DBuffer2 *iface, BYTE **scanline0, LONG *pitch)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMF2DBuffer2(iface);
|
|
return IMF2DBuffer2_Lock2D(buffer->buffer_2d, scanline0, pitch);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMF2DBuffer2_Unlock2D(IMF2DBuffer2 *iface)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMF2DBuffer2(iface);
|
|
return IMF2DBuffer2_Unlock2D(buffer->buffer_2d);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMF2DBuffer2_GetScanline0AndPitch(IMF2DBuffer2 *iface, BYTE **scanline0, LONG *pitch)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMF2DBuffer2(iface);
|
|
return IMF2DBuffer2_GetScanline0AndPitch(buffer->buffer_2d, scanline0, pitch);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMF2DBuffer2_IsContiguousFormat(IMF2DBuffer2 *iface, BOOL *is_contiguous)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMF2DBuffer2(iface);
|
|
return IMF2DBuffer2_IsContiguousFormat(buffer->buffer_2d, is_contiguous);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMF2DBuffer2_GetContiguousLength(IMF2DBuffer2 *iface, DWORD *length)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMF2DBuffer2(iface);
|
|
return IMF2DBuffer2_GetContiguousLength(buffer->buffer_2d, length);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMF2DBuffer2_ContiguousCopyTo(IMF2DBuffer2 *iface, BYTE *dest_buffer, DWORD dest_length)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMF2DBuffer2(iface);
|
|
return IMF2DBuffer2_ContiguousCopyTo(buffer->buffer_2d, dest_buffer, dest_length);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMF2DBuffer2_ContiguousCopyFrom(IMF2DBuffer2 *iface, const BYTE *src_buffer, DWORD src_length)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMF2DBuffer2(iface);
|
|
return IMF2DBuffer2_ContiguousCopyFrom(buffer->buffer_2d, src_buffer, src_length);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMF2DBuffer2_Lock2DSize(IMF2DBuffer2 *iface, MF2DBuffer_LockFlags flags, BYTE **scanline0,
|
|
LONG *pitch, BYTE **buffer_start, DWORD *buffer_length)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMF2DBuffer2(iface);
|
|
return IMF2DBuffer2_Lock2DSize(buffer->buffer_2d, flags, scanline0, pitch, buffer_start, buffer_length);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMF2DBuffer2_Copy2DTo(IMF2DBuffer2 *iface, IMF2DBuffer2 *dest_buffer)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMF2DBuffer2(iface);
|
|
return IMF2DBuffer2_Copy2DTo(buffer->buffer_2d, dest_buffer);
|
|
}
|
|
|
|
static const IMF2DBuffer2Vtbl legacy_buffer_IMF2DBuffer2_vtbl =
|
|
{
|
|
legacy_buffer_IMF2DBuffer2_QueryInterface,
|
|
legacy_buffer_IMF2DBuffer2_AddRef,
|
|
legacy_buffer_IMF2DBuffer2_Release,
|
|
legacy_buffer_IMF2DBuffer2_Lock2D,
|
|
legacy_buffer_IMF2DBuffer2_Unlock2D,
|
|
legacy_buffer_IMF2DBuffer2_GetScanline0AndPitch,
|
|
legacy_buffer_IMF2DBuffer2_IsContiguousFormat,
|
|
legacy_buffer_IMF2DBuffer2_GetContiguousLength,
|
|
legacy_buffer_IMF2DBuffer2_ContiguousCopyTo,
|
|
legacy_buffer_IMF2DBuffer2_ContiguousCopyFrom,
|
|
legacy_buffer_IMF2DBuffer2_Lock2DSize,
|
|
legacy_buffer_IMF2DBuffer2_Copy2DTo,
|
|
};
|
|
|
|
static struct legacy_buffer *legacy_buffer_from_IMFSample(IMFSample *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct legacy_buffer, IMFSample_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_QueryInterface(IMFSample *iface, REFIID riid, void **obj)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMediaBuffer_QueryInterface(&buffer->IMediaBuffer_iface, riid, obj);
|
|
}
|
|
|
|
static ULONG WINAPI legacy_buffer_IMFSample_AddRef(IMFSample *iface)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMediaBuffer_AddRef(&buffer->IMediaBuffer_iface);
|
|
}
|
|
|
|
static ULONG WINAPI legacy_buffer_IMFSample_Release(IMFSample *iface)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMediaBuffer_Release(&buffer->IMediaBuffer_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetItem(IMFSample *iface, REFGUID key, PROPVARIANT *value)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetItem(buffer->sample, key, value);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetItemType(IMFSample *iface, REFGUID key, MF_ATTRIBUTE_TYPE *type)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetItemType(buffer->sample, key, type);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_CompareItem(IMFSample *iface, REFGUID key, REFPROPVARIANT value, BOOL *result)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_CompareItem(buffer->sample, key, value, result);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_Compare(IMFSample *iface, IMFAttributes *theirs, MF_ATTRIBUTES_MATCH_TYPE type,
|
|
BOOL *result)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_Compare(buffer->sample, theirs, type, result);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetUINT32(IMFSample *iface, REFGUID key, UINT32 *value)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetUINT32(buffer->sample, key, value);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetUINT64(IMFSample *iface, REFGUID key, UINT64 *value)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetUINT64(buffer->sample, key, value);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetDouble(IMFSample *iface, REFGUID key, double *value)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetDouble(buffer->sample, key, value);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetGUID(IMFSample *iface, REFGUID key, GUID *value)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetGUID(buffer->sample, key, value);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetStringLength(IMFSample *iface, REFGUID key, UINT32 *length)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetStringLength(buffer->sample, key, length);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetString(IMFSample *iface, REFGUID key, WCHAR *value, UINT32 size, UINT32 *length)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetString(buffer->sample, key, value, size, length);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetAllocatedString(IMFSample *iface, REFGUID key, WCHAR **value, UINT32 *length)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetAllocatedString(buffer->sample, key, value, length);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetBlobSize(IMFSample *iface, REFGUID key, UINT32 *size)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetBlobSize(buffer->sample, key, size);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetBlob(IMFSample *iface, REFGUID key, UINT8 *buf, UINT32 bufsize, UINT32 *blobsize)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetBlob(buffer->sample, key, buf, bufsize, blobsize);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetAllocatedBlob(IMFSample *iface, REFGUID key, UINT8 **buf, UINT32 *size)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetAllocatedBlob(buffer->sample, key, buf, size);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetUnknown(IMFSample *iface, REFGUID key, REFIID riid, void **out)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetUnknown(buffer->sample, key, riid, out);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_SetItem(IMFSample *iface, REFGUID key, REFPROPVARIANT value)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_SetItem(buffer->sample, key, value);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_DeleteItem(IMFSample *iface, REFGUID key)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_DeleteItem(buffer->sample, key);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_DeleteAllItems(IMFSample *iface)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_DeleteAllItems(buffer->sample);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_SetUINT32(IMFSample *iface, REFGUID key, UINT32 value)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_SetUINT32(buffer->sample, key, value);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_SetUINT64(IMFSample *iface, REFGUID key, UINT64 value)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_SetUINT64(buffer->sample, key, value);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_SetDouble(IMFSample *iface, REFGUID key, double value)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_SetDouble(buffer->sample, key, value);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_SetGUID(IMFSample *iface, REFGUID key, REFGUID value)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_SetGUID(buffer->sample, key, value);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_SetString(IMFSample *iface, REFGUID key, const WCHAR *value)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_SetString(buffer->sample, key, value);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_SetBlob(IMFSample *iface, REFGUID key, const UINT8 *buf, UINT32 size)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_SetBlob(buffer->sample, key, buf, size);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_SetUnknown(IMFSample *iface, REFGUID key, IUnknown *unknown)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_SetUnknown(buffer->sample, key, unknown);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_LockStore(IMFSample *iface)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_LockStore(buffer->sample);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_UnlockStore(IMFSample *iface)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_UnlockStore(buffer->sample);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetCount(IMFSample *iface, UINT32 *count)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetCount(buffer->sample, count);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetItemByIndex(IMFSample *iface, UINT32 index, GUID *key, PROPVARIANT *value)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetItemByIndex(buffer->sample, index, key, value);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_CopyAllItems(IMFSample *iface, IMFAttributes *dest)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_CopyAllItems(buffer->sample, dest);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetSampleFlags(IMFSample *iface, DWORD *flags)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetSampleFlags(buffer->sample, flags);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_SetSampleFlags(IMFSample *iface, DWORD flags)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_SetSampleFlags(buffer->sample, flags);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetSampleTime(IMFSample *iface, LONGLONG *timestamp)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetSampleTime(buffer->sample, timestamp);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_SetSampleTime(IMFSample *iface, LONGLONG timestamp)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_SetSampleTime(buffer->sample, timestamp);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetSampleDuration(IMFSample *iface, LONGLONG *duration)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetSampleDuration(buffer->sample, duration);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_SetSampleDuration(IMFSample *iface, LONGLONG duration)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_SetSampleDuration(buffer->sample, duration);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetBufferCount(IMFSample *iface, DWORD *count)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetBufferCount(buffer->sample, count);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetBufferByIndex(IMFSample *iface, DWORD index, IMFMediaBuffer **out)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetBufferByIndex(buffer->sample, index, out);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_ConvertToContiguousBuffer(IMFSample *iface, IMFMediaBuffer **out)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_ConvertToContiguousBuffer(buffer->sample, out);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_AddBuffer(IMFSample *iface, IMFMediaBuffer *media_buffer)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_AddBuffer(buffer->sample, media_buffer);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_RemoveBufferByIndex(IMFSample *iface, DWORD index)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_RemoveBufferByIndex(buffer->sample, index);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_RemoveAllBuffers(IMFSample *iface)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_RemoveAllBuffers(buffer->sample);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_GetTotalLength(IMFSample *iface, DWORD *total_length)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_GetTotalLength(buffer->sample, total_length);
|
|
}
|
|
|
|
static HRESULT WINAPI legacy_buffer_IMFSample_CopyToBuffer(IMFSample *iface, IMFMediaBuffer *dest)
|
|
{
|
|
struct legacy_buffer *buffer = legacy_buffer_from_IMFSample(iface);
|
|
return IMFSample_CopyToBuffer(buffer->sample, dest);
|
|
}
|
|
|
|
static const IMFSampleVtbl legacy_buffer_IMFSample_vtbl =
|
|
{
|
|
legacy_buffer_IMFSample_QueryInterface,
|
|
legacy_buffer_IMFSample_AddRef,
|
|
legacy_buffer_IMFSample_Release,
|
|
legacy_buffer_IMFSample_GetItem,
|
|
legacy_buffer_IMFSample_GetItemType,
|
|
legacy_buffer_IMFSample_CompareItem,
|
|
legacy_buffer_IMFSample_Compare,
|
|
legacy_buffer_IMFSample_GetUINT32,
|
|
legacy_buffer_IMFSample_GetUINT64,
|
|
legacy_buffer_IMFSample_GetDouble,
|
|
legacy_buffer_IMFSample_GetGUID,
|
|
legacy_buffer_IMFSample_GetStringLength,
|
|
legacy_buffer_IMFSample_GetString,
|
|
legacy_buffer_IMFSample_GetAllocatedString,
|
|
legacy_buffer_IMFSample_GetBlobSize,
|
|
legacy_buffer_IMFSample_GetBlob,
|
|
legacy_buffer_IMFSample_GetAllocatedBlob,
|
|
legacy_buffer_IMFSample_GetUnknown,
|
|
legacy_buffer_IMFSample_SetItem,
|
|
legacy_buffer_IMFSample_DeleteItem,
|
|
legacy_buffer_IMFSample_DeleteAllItems,
|
|
legacy_buffer_IMFSample_SetUINT32,
|
|
legacy_buffer_IMFSample_SetUINT64,
|
|
legacy_buffer_IMFSample_SetDouble,
|
|
legacy_buffer_IMFSample_SetGUID,
|
|
legacy_buffer_IMFSample_SetString,
|
|
legacy_buffer_IMFSample_SetBlob,
|
|
legacy_buffer_IMFSample_SetUnknown,
|
|
legacy_buffer_IMFSample_LockStore,
|
|
legacy_buffer_IMFSample_UnlockStore,
|
|
legacy_buffer_IMFSample_GetCount,
|
|
legacy_buffer_IMFSample_GetItemByIndex,
|
|
legacy_buffer_IMFSample_CopyAllItems,
|
|
legacy_buffer_IMFSample_GetSampleFlags,
|
|
legacy_buffer_IMFSample_SetSampleFlags,
|
|
legacy_buffer_IMFSample_GetSampleTime,
|
|
legacy_buffer_IMFSample_SetSampleTime,
|
|
legacy_buffer_IMFSample_GetSampleDuration,
|
|
legacy_buffer_IMFSample_SetSampleDuration,
|
|
legacy_buffer_IMFSample_GetBufferCount,
|
|
legacy_buffer_IMFSample_GetBufferByIndex,
|
|
legacy_buffer_IMFSample_ConvertToContiguousBuffer,
|
|
legacy_buffer_IMFSample_AddBuffer,
|
|
legacy_buffer_IMFSample_RemoveBufferByIndex,
|
|
legacy_buffer_IMFSample_RemoveAllBuffers,
|
|
legacy_buffer_IMFSample_GetTotalLength,
|
|
legacy_buffer_IMFSample_CopyToBuffer,
|
|
};
|
|
|
|
HRESULT WINAPI MFCreateLegacyMediaBufferOnMFMediaBuffer(IMFSample *sample, IMFMediaBuffer *media_buffer, DWORD offset, IMediaBuffer **obj)
|
|
{
|
|
struct legacy_buffer *buffer;
|
|
|
|
TRACE("sample %p, media_buffer %p, offset %lu, out %p\n", sample, media_buffer, offset, obj);
|
|
|
|
if (!obj)
|
|
return E_INVALIDARG;
|
|
if (offset)
|
|
{
|
|
FIXME("Not implemented\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
if (!(buffer = calloc(1, sizeof(*buffer))))
|
|
return E_OUTOFMEMORY;
|
|
buffer->IMediaBuffer_iface.lpVtbl = &legacy_buffer_vtbl;
|
|
buffer->IMFSample_iface.lpVtbl = &legacy_buffer_IMFSample_vtbl;
|
|
buffer->IMF2DBuffer2_iface.lpVtbl = &legacy_buffer_IMF2DBuffer2_vtbl;
|
|
buffer->refcount = 1;
|
|
|
|
IMFMediaBuffer_AddRef((buffer->media_buffer = media_buffer));
|
|
if ((buffer->sample = sample))
|
|
IMFSample_AddRef(buffer->sample);
|
|
if (FAILED(IMFMediaBuffer_QueryInterface(media_buffer, &IID_IMF2DBuffer2, (void **)&buffer->buffer_2d)))
|
|
buffer->buffer_2d = NULL;
|
|
|
|
TRACE("created %p\n", buffer);
|
|
*obj = &buffer->IMediaBuffer_iface;
|
|
return S_OK;
|
|
}
|