Signed-off-by: Nikolay Sivov <nsivov(a)codeweavers.com>
---
 dlls/mf/Makefile.in     |   1 +
 dlls/mf/main.c          | 436 ++++++++++++++++++++++++++++++++++++++++
 dlls/mf/mf.spec         |   2 +-
 dlls/mf/mf_private.h    |  27 +++
 dlls/mf/samplegrabber.c |  83 ++++++++
 dlls/mf/tests/mf.c      | 110 ++++++++++
 include/mfidl.idl       |  26 ++-
 7 files changed, 683 insertions(+), 2 deletions(-)
 create mode 100644 dlls/mf/mf_private.h
 create mode 100644 dlls/mf/samplegrabber.c
diff --git a/dlls/mf/Makefile.in b/dlls/mf/Makefile.in
index 221ef42ec9..2eb9b346d9 100644
--- a/dlls/mf/Makefile.in
+++ b/dlls/mf/Makefile.in
@@ -4,6 +4,7 @@ IMPORTS   = mfplat mfuuid
 
 C_SRCS = \
 	main.c \
+	samplegrabber.c \
 	session.c \
 	topology.c
 
diff --git a/dlls/mf/main.c b/dlls/mf/main.c
index 77081e2a57..c2e5eb45d2 100644
--- a/dlls/mf/main.c
+++ b/dlls/mf/main.c
@@ -35,6 +35,8 @@
 #include "mfapi.h"
 #include "mferror.h"
 
+#include "mf_private.h"
+
 #include "wine/debug.h"
 #include "wine/heap.h"
 #include "wine/unicode.h"
@@ -44,6 +46,440 @@ WINE_DEFAULT_DEBUG_CHANNEL(mfplat);
 
 static HINSTANCE mf_instance;
 
+struct activate_object
+{
+    IMFActivate IMFActivate_iface;
+    LONG refcount;
+    IMFAttributes *attributes;
+    IUnknown *object;
+    const struct activate_funcs *funcs;
+    void *context;
+};
+
+static struct activate_object *impl_from_IMFActivate(IMFActivate *iface)
+{
+    return CONTAINING_RECORD(iface, struct activate_object, IMFActivate_iface);
+}
+
+static HRESULT WINAPI activate_object_QueryInterface(IMFActivate *iface, REFIID riid, void **obj)
+{
+    TRACE("%p, %s, %p.\n", iface, debugstr_guid(riid), obj);
+
+    if (IsEqualIID(riid, &IID_IMFActivate) ||
+            IsEqualIID(riid, &IID_IMFAttributes) ||
+            IsEqualIID(riid, &IID_IUnknown))
+    {
+        *obj = iface;
+        IMFActivate_AddRef(iface);
+        return S_OK;
+    }
+
+    WARN("Unsupported %s.\n", debugstr_guid(riid));
+    *obj = NULL;
+    return E_NOINTERFACE;
+}
+
+static ULONG WINAPI activate_object_AddRef(IMFActivate *iface)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+    ULONG refcount = InterlockedIncrement(&activate->refcount);
+
+    TRACE("%p, refcount %u.\n", iface, refcount);
+
+    return refcount;
+}
+
+static ULONG WINAPI activate_object_Release(IMFActivate *iface)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+    ULONG refcount = InterlockedDecrement(&activate->refcount);
+
+    TRACE("%p, refcount %u.\n", iface, refcount);
+
+    if (!refcount)
+    {
+        activate->funcs->free_private(activate->context);
+        if (activate->object)
+            IUnknown_Release(activate->object);
+        IMFAttributes_Release(activate->attributes);
+        heap_free(activate);
+    }
+
+    return refcount;
+}
+
+static HRESULT WINAPI activate_object_GetItem(IMFActivate *iface, REFGUID key, PROPVARIANT *value)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p.\n", iface, debugstr_guid(key), value);
+
+    return IMFAttributes_GetItem(activate->attributes, key, value);
+}
+
+static HRESULT WINAPI activate_object_GetItemType(IMFActivate *iface, REFGUID key, MF_ATTRIBUTE_TYPE *type)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p.\n", iface, debugstr_guid(key), type);
+
+    return IMFAttributes_GetItemType(activate->attributes, key, type);
+}
+
+static HRESULT WINAPI activate_object_CompareItem(IMFActivate *iface, REFGUID key, REFPROPVARIANT value, BOOL *result)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p, %p.\n", iface, debugstr_guid(key), value, result);
+
+    return IMFAttributes_CompareItem(activate->attributes, key, value, result);
+}
+
+static HRESULT WINAPI activate_object_Compare(IMFActivate *iface, IMFAttributes *theirs, MF_ATTRIBUTES_MATCH_TYPE type,
+        BOOL *result)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %p, %d, %p.\n", iface, theirs, type, result);
+
+    return IMFAttributes_Compare(activate->attributes, theirs, type, result);
+}
+
+static HRESULT WINAPI activate_object_GetUINT32(IMFActivate *iface, REFGUID key, UINT32 *value)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p.\n", iface, debugstr_guid(key), value);
+
+    return IMFAttributes_GetUINT32(activate->attributes, key, value);
+}
+
+static HRESULT WINAPI activate_object_GetUINT64(IMFActivate *iface, REFGUID key, UINT64 *value)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p.\n", iface, debugstr_guid(key), value);
+
+    return IMFAttributes_GetUINT64(activate->attributes, key, value);
+}
+
+static HRESULT WINAPI activate_object_GetDouble(IMFActivate *iface, REFGUID key, double *value)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p.\n", iface, debugstr_guid(key), value);
+
+    return IMFAttributes_GetDouble(activate->attributes, key, value);
+}
+
+static HRESULT WINAPI activate_object_GetGUID(IMFActivate *iface, REFGUID key, GUID *value)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p.\n", iface, debugstr_guid(key), value);
+
+    return IMFAttributes_GetGUID(activate->attributes, key, value);
+}
+
+static HRESULT WINAPI activate_object_GetStringLength(IMFActivate *iface, REFGUID key, UINT32 *length)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p.\n", iface, debugstr_guid(key), length);
+
+    return IMFAttributes_GetStringLength(activate->attributes, key, length);
+}
+
+static HRESULT WINAPI activate_object_GetString(IMFActivate *iface, REFGUID key, WCHAR *value,
+        UINT32 size, UINT32 *length)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p, %d, %p.\n", iface, debugstr_guid(key), value, size, length);
+
+    return IMFAttributes_GetString(activate->attributes, key, value, size, length);
+}
+
+static HRESULT WINAPI activate_object_GetAllocatedString(IMFActivate *iface, REFGUID key,
+        WCHAR **value, UINT32 *length)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p, %p.\n", iface, debugstr_guid(key), value, length);
+
+    return IMFAttributes_GetAllocatedString(activate->attributes, key, value, length);
+}
+
+static HRESULT WINAPI activate_object_GetBlobSize(IMFActivate *iface, REFGUID key, UINT32 *size)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p.\n", iface, debugstr_guid(key), size);
+
+    return IMFAttributes_GetBlobSize(activate->attributes, key, size);
+}
+
+static HRESULT WINAPI activate_object_GetBlob(IMFActivate *iface, REFGUID key, UINT8 *buf,
+        UINT32 bufsize, UINT32 *blobsize)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p, %d, %p.\n", iface, debugstr_guid(key), buf, bufsize, blobsize);
+
+    return IMFAttributes_GetBlob(activate->attributes, key, buf, bufsize, blobsize);
+}
+
+static HRESULT WINAPI activate_object_GetAllocatedBlob(IMFActivate *iface, REFGUID key, UINT8 **buf, UINT32 *size)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p, %p.\n", iface, debugstr_guid(key), buf, size);
+
+    return IMFAttributes_GetAllocatedBlob(activate->attributes, key, buf, size);
+}
+
+static HRESULT WINAPI activate_object_GetUnknown(IMFActivate *iface, REFGUID key, REFIID riid, void **ppv)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %s, %p.\n", iface, debugstr_guid(key), debugstr_guid(riid), ppv);
+
+    return IMFAttributes_GetUnknown(activate->attributes, key, riid, ppv);
+}
+
+static HRESULT WINAPI activate_object_SetItem(IMFActivate *iface, REFGUID key, REFPROPVARIANT value)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p.\n", iface, debugstr_guid(key), value);
+
+    return IMFAttributes_SetItem(activate->attributes, key, value);
+}
+
+static HRESULT WINAPI activate_object_DeleteItem(IMFActivate *iface, REFGUID key)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s.\n", iface, debugstr_guid(key));
+
+    return IMFAttributes_DeleteItem(activate->attributes, key);
+}
+
+static HRESULT WINAPI activate_object_DeleteAllItems(IMFActivate *iface)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p.\n", iface);
+
+    return IMFAttributes_DeleteAllItems(activate->attributes);
+}
+
+static HRESULT WINAPI activate_object_SetUINT32(IMFActivate *iface, REFGUID key, UINT32 value)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %d.\n", iface, debugstr_guid(key), value);
+
+    return IMFAttributes_SetUINT32(activate->attributes, key, value);
+}
+
+static HRESULT WINAPI activate_object_SetUINT64(IMFActivate *iface, REFGUID key, UINT64 value)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %s.\n", iface, debugstr_guid(key), wine_dbgstr_longlong(value));
+
+    return IMFAttributes_SetUINT64(activate->attributes, key, value);
+}
+
+static HRESULT WINAPI activate_object_SetDouble(IMFActivate *iface, REFGUID key, double value)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %f.\n", iface, debugstr_guid(key), value);
+
+    return IMFAttributes_SetDouble(activate->attributes, key, value);
+}
+
+static HRESULT WINAPI activate_object_SetGUID(IMFActivate *iface, REFGUID key, REFGUID value)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %s.\n", iface, debugstr_guid(key), debugstr_guid(value));
+
+    return IMFAttributes_SetGUID(activate->attributes, key, value);
+}
+
+static HRESULT WINAPI activate_object_SetString(IMFActivate *iface, REFGUID key, const WCHAR *value)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %s.\n", iface, debugstr_guid(key), debugstr_w(value));
+
+    return IMFAttributes_SetString(activate->attributes, key, value);
+}
+
+static HRESULT WINAPI activate_object_SetBlob(IMFActivate *iface, REFGUID key, const UINT8 *buf, UINT32 size)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %s, %p, %d.\n", iface, debugstr_guid(key), buf, size);
+
+    return IMFAttributes_SetBlob(activate->attributes, key, buf, size);
+}
+
+static HRESULT WINAPI activate_object_SetUnknown(IMFActivate *iface, REFGUID key, IUnknown *unknown)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("(%p)->(%s, %p)\n", iface, debugstr_guid(key), unknown);
+
+    return IMFAttributes_SetUnknown(activate->attributes, key, unknown);
+}
+
+static HRESULT WINAPI activate_object_LockStore(IMFActivate *iface)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p.\n", iface);
+
+    return IMFAttributes_LockStore(activate->attributes);
+}
+
+static HRESULT WINAPI activate_object_UnlockStore(IMFActivate *iface)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p.\n", iface);
+
+    return IMFAttributes_UnlockStore(activate->attributes);
+}
+
+static HRESULT WINAPI activate_object_GetCount(IMFActivate *iface, UINT32 *count)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %p.\n", iface, count);
+
+    return IMFAttributes_GetCount(activate->attributes, count);
+}
+
+static HRESULT WINAPI activate_object_GetItemByIndex(IMFActivate *iface, UINT32 index, GUID *key, PROPVARIANT *value)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %u, %p, %p.\n", iface, index, key, value);
+
+    return IMFAttributes_GetItemByIndex(activate->attributes, index, key, value);
+}
+
+static HRESULT WINAPI activate_object_CopyAllItems(IMFActivate *iface, IMFAttributes *dest)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+
+    TRACE("%p, %p.\n", iface, dest);
+
+    return IMFAttributes_CopyAllItems(activate->attributes, dest);
+}
+
+static HRESULT WINAPI activate_object_ActivateObject(IMFActivate *iface, REFIID riid, void **obj)
+{
+    struct activate_object *activate = impl_from_IMFActivate(iface);
+    IUnknown *object;
+    HRESULT hr;
+
+    TRACE("%p, %s, %p.\n", iface, debugstr_guid(riid), obj);
+
+    if (!activate->object)
+    {
+        if (FAILED(hr = activate->funcs->create_object(activate->context, &object)))
+            return hr;
+
+        if (!InterlockedCompareExchangePointer((void **)&activate->object, object, NULL))
+            IUnknown_Release(object);
+    }
+
+    return IUnknown_QueryInterface(activate->object, riid, obj);
+}
+
+static HRESULT WINAPI activate_object_ShutdownObject(IMFActivate *iface)
+{
+    FIXME("%p.\n", iface);
+
+    return E_NOTIMPL;
+}
+
+static HRESULT WINAPI activate_object_DetachObject(IMFActivate *iface)
+{
+    FIXME("%p.\n", iface);
+
+    return E_NOTIMPL;
+}
+
+static const IMFActivateVtbl activate_object_vtbl =
+{
+    activate_object_QueryInterface,
+    activate_object_AddRef,
+    activate_object_Release,
+    activate_object_GetItem,
+    activate_object_GetItemType,
+    activate_object_CompareItem,
+    activate_object_Compare,
+    activate_object_GetUINT32,
+    activate_object_GetUINT64,
+    activate_object_GetDouble,
+    activate_object_GetGUID,
+    activate_object_GetStringLength,
+    activate_object_GetString,
+    activate_object_GetAllocatedString,
+    activate_object_GetBlobSize,
+    activate_object_GetBlob,
+    activate_object_GetAllocatedBlob,
+    activate_object_GetUnknown,
+    activate_object_SetItem,
+    activate_object_DeleteItem,
+    activate_object_DeleteAllItems,
+    activate_object_SetUINT32,
+    activate_object_SetUINT64,
+    activate_object_SetDouble,
+    activate_object_SetGUID,
+    activate_object_SetString,
+    activate_object_SetBlob,
+    activate_object_SetUnknown,
+    activate_object_LockStore,
+    activate_object_UnlockStore,
+    activate_object_GetCount,
+    activate_object_GetItemByIndex,
+    activate_object_CopyAllItems,
+    activate_object_ActivateObject,
+    activate_object_ShutdownObject,
+    activate_object_DetachObject,
+};
+
+HRESULT create_activation_object(void *context, const struct activate_funcs *funcs, IMFActivate **ret)
+{
+    struct activate_object *object;
+    HRESULT hr;
+
+    object = heap_alloc_zero(sizeof(*object));
+    if (!object)
+        return E_OUTOFMEMORY;
+
+    object->IMFActivate_iface.lpVtbl = &activate_object_vtbl;
+    object->refcount = 1;
+    if (FAILED(hr = MFCreateAttributes(&object->attributes, 0)))
+    {
+        heap_free(object);
+        return hr;
+    }
+    object->funcs = funcs;
+    object->context = context;
+
+    *ret = &object->IMFActivate_iface;
+
+    return S_OK;
+}
+
 struct class_factory
 {
     IClassFactory IClassFactory_iface;
diff --git a/dlls/mf/mf.spec b/dlls/mf/mf.spec
index 9e3f6caef8..6131495225 100644
--- a/dlls/mf/mf.spec
+++ b/dlls/mf/mf.spec
@@ -51,7 +51,7 @@
 @ stub MFCreateRemoteDesktopPlugin
 @ stub MFCreateSAMIByteStreamPlugin
 @ stub MFCreateSampleCopierMFT
-@ stub MFCreateSampleGrabberSinkActivate
+@ stdcall MFCreateSampleGrabberSinkActivate(ptr ptr ptr)
 @ stub MFCreateSecureHttpSchemePlugin
 @ stub MFCreateSequencerSegmentOffset
 @ stdcall MFCreateSequencerSource(ptr ptr)
diff --git a/dlls/mf/mf_private.h b/dlls/mf/mf_private.h
new file mode 100644
index 0000000000..b5b4e84bba
--- /dev/null
+++ b/dlls/mf/mf_private.h
@@ -0,0 +1,27 @@
+/*
+ * 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
+ */
+
+#include "mfidl.h"
+
+struct activate_funcs
+{
+    HRESULT (*create_object)(void *context, IUnknown **object);
+    void (*free_private)(void *context);
+};
+
+HRESULT create_activation_object(void *context, const struct activate_funcs *funcs, IMFActivate **ret) DECLSPEC_HIDDEN;
diff --git a/dlls/mf/samplegrabber.c b/dlls/mf/samplegrabber.c
new file mode 100644
index 0000000000..1ffa3ba0b0
--- /dev/null
+++ b/dlls/mf/samplegrabber.c
@@ -0,0 +1,83 @@
+/*
+ * 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
+ */
+
+#define COBJMACROS
+
+#include "mfidl.h"
+#include "mf_private.h"
+
+#include "wine/debug.h"
+#include "wine/heap.h"
+
+WINE_DEFAULT_DEBUG_CHANNEL(mfplat);
+
+struct sample_grabber_activate_context
+{
+    IMFMediaType *media_type;
+    IMFSampleGrabberSinkCallback *callback;
+};
+
+static void sample_grabber_free_private(void *user_context)
+{
+    struct sample_grabber_activate_context *context = user_context;
+    IMFMediaType_Release(context->media_type);
+    IMFSampleGrabberSinkCallback_Release(context->callback);
+    heap_free(context);
+}
+
+static HRESULT sample_grabber_create_object(void *user_context, IUnknown **obj)
+{
+    FIXME("%p, %p.\n", user_context, obj);
+
+    return E_NOTIMPL;
+}
+
+static const struct activate_funcs sample_grabber_activate_funcs =
+{
+    sample_grabber_create_object,
+    sample_grabber_free_private,
+};
+
+/***********************************************************************
+ *      MFCreateSampleGrabberSinkActivate (mf.@)
+ */
+HRESULT WINAPI MFCreateSampleGrabberSinkActivate(IMFMediaType *media_type, IMFSampleGrabberSinkCallback *callback,
+        IMFActivate **activate)
+{
+    struct sample_grabber_activate_context *context;
+    HRESULT hr;
+
+    TRACE("%p, %p, %p.\n", media_type, callback, activate);
+
+    if (!media_type || !callback || !activate)
+        return E_POINTER;
+
+    context = heap_alloc_zero(sizeof(*context));
+    if (!context)
+        return E_OUTOFMEMORY;
+
+    context->media_type = media_type;
+    IMFMediaType_AddRef(context->media_type);
+    context->callback = callback;
+    IMFSampleGrabberSinkCallback_AddRef(context->callback);
+
+    if (FAILED(hr = create_activation_object(context, &sample_grabber_activate_funcs, activate)))
+        sample_grabber_free_private(context);
+
+    return hr;
+}
diff --git a/dlls/mf/tests/mf.c b/dlls/mf/tests/mf.c
index 80dc990254..059062f698 100644
--- a/dlls/mf/tests/mf.c
+++ b/dlls/mf/tests/mf.c
@@ -940,6 +940,115 @@ todo_wine
     ok(hr == S_OK, "Failed to shut down, hr %#x.\n", hr);
 }
 
+static HRESULT WINAPI grabber_callback_QueryInterface(IMFSampleGrabberSinkCallback *iface, REFIID riid, void **obj)
+{
+    if (IsEqualIID(riid, &IID_IMFSampleGrabberSinkCallback) ||
+            IsEqualIID(riid, &IID_IMFClockStateSink) ||
+            IsEqualIID(riid, &IID_IUnknown))
+    {
+        *obj = iface;
+        IMFSampleGrabberSinkCallback_AddRef(iface);
+        return S_OK;
+    }
+
+    *obj = NULL;
+    return E_NOINTERFACE;
+}
+
+static ULONG WINAPI grabber_callback_AddRef(IMFSampleGrabberSinkCallback *iface)
+{
+    return 2;
+}
+
+static ULONG WINAPI grabber_callback_Release(IMFSampleGrabberSinkCallback *iface)
+{
+    return 1;
+}
+
+static HRESULT WINAPI grabber_callback_OnClockStart(IMFSampleGrabberSinkCallback *iface, MFTIME time, LONGLONG offset)
+{
+    return E_NOTIMPL;
+}
+
+static HRESULT WINAPI grabber_callback_OnClockStop(IMFSampleGrabberSinkCallback *iface, MFTIME time)
+{
+    return E_NOTIMPL;
+}
+
+static HRESULT WINAPI grabber_callback_OnClockPause(IMFSampleGrabberSinkCallback *iface, MFTIME time)
+{
+    return E_NOTIMPL;
+}
+
+static HRESULT WINAPI grabber_callback_OnClockRestart(IMFSampleGrabberSinkCallback *iface, MFTIME time)
+{
+    return E_NOTIMPL;
+}
+
+static HRESULT WINAPI grabber_callback_OnClockSetRate(IMFSampleGrabberSinkCallback *iface, MFTIME time, float rate)
+{
+    return E_NOTIMPL;
+}
+
+static HRESULT WINAPI grabber_callback_OnSetPresentationClock(IMFSampleGrabberSinkCallback *iface,
+        IMFPresentationClock *clock)
+{
+    return E_NOTIMPL;
+}
+
+static HRESULT WINAPI grabber_callback_OnProcessSample(IMFSampleGrabberSinkCallback *iface, REFGUID major_type,
+        DWORD sample_flags, LONGLONG sample_time, LONGLONG sample_duration, const BYTE *buffer, DWORD sample_size)
+{
+    return E_NOTIMPL;
+}
+
+static HRESULT WINAPI grabber_callback_OnShutdown(IMFSampleGrabberSinkCallback *iface)
+{
+    return E_NOTIMPL;
+}
+
+static const IMFSampleGrabberSinkCallbackVtbl grabber_callback_vtbl =
+{
+    grabber_callback_QueryInterface,
+    grabber_callback_AddRef,
+    grabber_callback_Release,
+    grabber_callback_OnClockStart,
+    grabber_callback_OnClockStop,
+    grabber_callback_OnClockPause,
+    grabber_callback_OnClockRestart,
+    grabber_callback_OnClockSetRate,
+    grabber_callback_OnSetPresentationClock,
+    grabber_callback_OnProcessSample,
+    grabber_callback_OnShutdown,
+};
+
+static IMFSampleGrabberSinkCallback grabber_callback = { &grabber_callback_vtbl };
+
+static void test_sample_grabber(void)
+{
+    IMFMediaType *media_type;
+    IMFActivate *activate;
+    ULONG refcount;
+    HRESULT hr;
+
+    hr = MFCreateMediaType(&media_type);
+    ok(hr == S_OK, "Failed to create media type, hr %#x.\n", hr);
+
+    hr = MFCreateSampleGrabberSinkActivate(NULL, NULL, &activate);
+    ok(hr == E_POINTER, "Unexpected hr %#x.\n", hr);
+
+    hr = MFCreateSampleGrabberSinkActivate(NULL, &grabber_callback, &activate);
+    ok(hr == E_POINTER, "Unexpected hr %#x.\n", hr);
+
+    hr = MFCreateSampleGrabberSinkActivate(media_type, &grabber_callback, &activate);
+    ok(hr == S_OK, "Failed to create grabber activate, hr %#x.\n", hr);
+
+    refcount = IMFMediaType_Release(media_type);
+    ok(refcount == 1, "Unexpected refcount %u.\n", refcount);
+
+    IMFActivate_Release(activate);
+}
+
 START_TEST(mf)
 {
     test_topology();
@@ -949,4 +1058,5 @@ START_TEST(mf)
     test_media_session();
     test_MFShutdownObject();
     test_presentation_clock();
+    test_sample_grabber();
 }
diff --git a/include/mfidl.idl b/include/mfidl.idl
index 64d39db77e..2f2f17c04a 100644
--- a/include/mfidl.idl
+++ b/include/mfidl.idl
@@ -463,8 +463,30 @@ interface IMFSequencerSource : IUnknown
     HRESULT UpdateTopologyFlags(
         [in] MFSequencerElementId id,
         [in] DWORD flags );
+}
+
+interface IMFPresentationClock;
+
+[
+    object,
+    uuid(8c7b80bf-ee42-4b59-b1df-55668e1bdca8),
+    local
+]
+interface IMFSampleGrabberSinkCallback : IMFClockStateSink
+{
+    HRESULT OnSetPresentationClock(
+        [in] IMFPresentationClock *clock);
 
-};
+    HRESULT OnProcessSample(
+        [in] REFGUID major_type,
+        [in] DWORD sample_flags,
+        [in] LONGLONG sample_time,
+        [in] LONGLONG sample_duration,
+        [in] const BYTE *buffer,
+        [in] DWORD sample_size);
+
+    HRESULT OnShutdown();
+}
 
 cpp_quote("HRESULT WINAPI MFCreateMediaSession(IMFAttributes *config, IMFMediaSession **session);")
 cpp_quote("HRESULT WINAPI MFCreateMFByteStreamOnStream(IStream *stream, IMFByteStream **bytestream);" )
@@ -472,6 +494,8 @@ cpp_quote("HRESULT WINAPI MFCreateMFByteStreamOnStreamEx(IUnknown *stream, IMFBy
 cpp_quote("HRESULT WINAPI MFCreatePresentationClock(IMFPresentationClock **clock);")
 cpp_quote("HRESULT WINAPI MFCreatePresentationDescriptor(DWORD count, IMFStreamDescriptor **descriptors,")
 cpp_quote("     IMFPresentationDescriptor **presentation_desc);")
+cpp_quote("HRESULT WINAPI MFCreateSampleGrabberSinkActivate(IMFMediaType *media_type,")
+cpp_quote("        IMFSampleGrabberSinkCallback *callback, IMFActivate **activate);")
 cpp_quote("HRESULT WINAPI MFCreateSequencerSource(IUnknown *reserved, IMFSequencerSource **seq_source);" )
 cpp_quote("HRESULT WINAPI MFCreateSourceResolver(IMFSourceResolver **resolver);")
 cpp_quote("HRESULT WINAPI MFCreateStreamDescriptor(DWORD identifier, DWORD cMediaTypes,")
-- 
2.20.1