2019-05-06 10:52:13 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2019 Nikolay Sivov for CodeWeavers
|
|
|
|
*
|
|
|
|
* 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-07-06 15:08:29 +00:00
|
|
|
#include "mferror.h"
|
2019-05-06 10:52:13 +00:00
|
|
|
#include "mfidl.h"
|
2020-07-07 11:40:15 +00:00
|
|
|
#include "mfapi.h"
|
2019-05-06 10:52:13 +00:00
|
|
|
|
2020-03-25 12:01:23 +00:00
|
|
|
#include "wine/debug.h"
|
2019-05-07 12:25:44 +00:00
|
|
|
|
|
|
|
static inline BOOL mf_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;
|
|
|
|
|
2021-06-17 10:33:51 +00:00
|
|
|
if (!(new_elements = realloc(*elements, new_capacity * size)))
|
2019-05-07 12:25:44 +00:00
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
*elements = new_elements;
|
|
|
|
*capacity = new_capacity;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2019-05-06 10:52:13 +00:00
|
|
|
struct activate_funcs
|
|
|
|
{
|
2019-05-10 15:44:33 +00:00
|
|
|
HRESULT (*create_object)(IMFAttributes *attributes, void *context, IUnknown **object);
|
2020-03-11 11:19:41 +00:00
|
|
|
void (*shutdown_object)(void *context, IUnknown *object);
|
2019-05-06 10:52:13 +00:00
|
|
|
void (*free_private)(void *context);
|
|
|
|
};
|
|
|
|
|
2023-07-12 07:41:04 +00:00
|
|
|
HRESULT create_activation_object(void *context, const struct activate_funcs *funcs, IMFActivate **ret);
|
2020-03-25 12:01:23 +00:00
|
|
|
|
|
|
|
static inline const char *debugstr_time(LONGLONG time)
|
|
|
|
{
|
|
|
|
ULONGLONG abstime = time >= 0 ? time : -time;
|
|
|
|
unsigned int i = 0, j = 0;
|
|
|
|
char buffer[23], rev[23];
|
|
|
|
|
|
|
|
while (abstime || i <= 8)
|
|
|
|
{
|
|
|
|
buffer[i++] = '0' + (abstime % 10);
|
|
|
|
abstime /= 10;
|
|
|
|
if (i == 7) buffer[i++] = '.';
|
|
|
|
}
|
|
|
|
if (time < 0) buffer[i++] = '-';
|
|
|
|
|
|
|
|
while (i--) rev[j++] = buffer[i];
|
2020-03-26 17:09:17 +00:00
|
|
|
while (rev[j-1] == '0' && rev[j-2] != '.') --j;
|
2020-03-25 12:01:23 +00:00
|
|
|
rev[j] = 0;
|
|
|
|
|
|
|
|
return wine_dbg_sprintf("%s", rev);
|
|
|
|
}
|
2020-12-01 12:16:29 +00:00
|
|
|
|
2022-03-28 11:57:04 +00:00
|
|
|
static inline const char *debugstr_propvar(const PROPVARIANT *v)
|
|
|
|
{
|
|
|
|
if (!v)
|
|
|
|
return "(null)";
|
|
|
|
|
|
|
|
switch (v->vt)
|
|
|
|
{
|
|
|
|
case VT_EMPTY:
|
|
|
|
return wine_dbg_sprintf("%p {VT_EMPTY}", v);
|
|
|
|
case VT_NULL:
|
|
|
|
return wine_dbg_sprintf("%p {VT_NULL}", v);
|
|
|
|
case VT_UI4:
|
|
|
|
return wine_dbg_sprintf("%p {VT_UI4: %ld}", v, v->ulVal);
|
|
|
|
case VT_UI8:
|
|
|
|
return wine_dbg_sprintf("%p {VT_UI8: %s}", v, wine_dbgstr_longlong(v->uhVal.QuadPart));
|
|
|
|
case VT_I8:
|
|
|
|
return wine_dbg_sprintf("%p {VT_I8: %s}", v, wine_dbgstr_longlong(v->hVal.QuadPart));
|
|
|
|
case VT_R4:
|
|
|
|
return wine_dbg_sprintf("%p {VT_R4: %.8e}", v, v->fltVal);
|
|
|
|
case VT_R8:
|
|
|
|
return wine_dbg_sprintf("%p {VT_R8: %lf}", v, v->dblVal);
|
|
|
|
case VT_CLSID:
|
|
|
|
return wine_dbg_sprintf("%p {VT_CLSID: %s}", v, wine_dbgstr_guid(v->puuid));
|
|
|
|
case VT_LPWSTR:
|
|
|
|
return wine_dbg_sprintf("%p {VT_LPWSTR: %s}", v, wine_dbgstr_w(v->pwszVal));
|
|
|
|
case VT_VECTOR | VT_UI1:
|
|
|
|
return wine_dbg_sprintf("%p {VT_VECTOR|VT_UI1: %p}", v, v->caub.pElems);
|
|
|
|
case VT_UNKNOWN:
|
|
|
|
return wine_dbg_sprintf("%p {VT_UNKNOWN: %p}", v, v->punkVal);
|
|
|
|
default:
|
|
|
|
return wine_dbg_sprintf("%p {vt %#x}", v, v->vt);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-19 08:27:53 +00:00
|
|
|
extern HRESULT file_scheme_handler_construct(REFIID riid, void **obj);
|
2023-08-22 09:35:09 +00:00
|
|
|
extern HRESULT urlmon_scheme_handler_construct(REFIID riid, void **obj);
|
2023-06-19 08:27:53 +00:00
|
|
|
|
2023-07-12 07:41:04 +00:00
|
|
|
extern BOOL mf_is_sample_copier_transform(IMFTransform *transform);
|
|
|
|
extern BOOL mf_is_sar_sink(IMFMediaSink *sink);
|
2024-03-11 14:29:48 +00:00
|
|
|
extern HRESULT create_topology(TOPOID id, IMFTopology **topology);
|
2023-07-12 07:41:04 +00:00
|
|
|
extern HRESULT topology_node_get_object(IMFTopologyNode *node, REFIID riid, void **obj);
|
|
|
|
extern HRESULT topology_node_get_type_handler(IMFTopologyNode *node, DWORD stream, BOOL output, IMFMediaTypeHandler **handler);
|
|
|
|
extern HRESULT topology_node_init_media_type(IMFTopologyNode *node, DWORD stream, BOOL output, IMFMediaType **type);
|
2024-05-13 16:04:48 +00:00
|
|
|
extern BOOL topology_node_is_d3d_aware(IMFTopologyNode *node);
|
|
|
|
extern HRESULT topology_node_set_device_manager(IMFTopologyNode *node, IUnknown *device_manager);
|
|
|
|
extern HRESULT stream_sink_get_device_manager(IMFStreamSink *stream_sink, IUnknown **device_manager);
|
2024-05-25 01:08:52 +00:00
|
|
|
|
|
|
|
extern HRESULT enum_audio_capture_sources(IMFAttributes *attributes, IMFActivate ***sources, UINT32 *ret_count);
|