448 lines
14 KiB
C
448 lines
14 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 <stdarg.h>
|
|
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "ole2.h"
|
|
#include "comsvcs.h"
|
|
#include "msxml.h"
|
|
|
|
#include "wine/test.h"
|
|
|
|
#define DEFINE_EXPECT(func) \
|
|
static BOOL expect_ ## func = FALSE, called_ ## func = FALSE
|
|
|
|
#define SET_EXPECT(func) \
|
|
expect_ ## func = TRUE
|
|
|
|
#define CHECK_CALLED(func) \
|
|
do { \
|
|
ok(called_ ## func, "expected " #func "\n"); \
|
|
expect_ ## func = called_ ## func = FALSE; \
|
|
}while(0)
|
|
|
|
#define CHECK_EXPECT2(func) \
|
|
do { \
|
|
trace(#func "\n"); \
|
|
ok(expect_ ##func, "unexpected call " #func "\n"); \
|
|
called_ ## func = TRUE; \
|
|
}while(0)
|
|
|
|
#define CHECK_CALLED_BROKEN(func) \
|
|
do { \
|
|
ok(called_ ## func || broken(!called_ ## func), "expected " #func "\n"); \
|
|
expect_ ## func = called_ ## func = FALSE; \
|
|
}while(0)
|
|
|
|
DEFINE_EXPECT(driver_CreateResource);
|
|
DEFINE_EXPECT(driver_DestroyResource);
|
|
DEFINE_EXPECT(driver_ResetResource);
|
|
DEFINE_EXPECT(driver_Release);
|
|
|
|
HRESULT driver_DestroyResource_ret = S_OK;
|
|
|
|
static HRESULT WINAPI driver_QueryInterface(IDispenserDriver *iface, REFIID riid, void **object)
|
|
{
|
|
if (IsEqualGUID(riid, &IID_IUnknown) ||
|
|
IsEqualGUID(riid, &IID_IDispenserDriver))
|
|
{
|
|
*object = iface;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
ok(0, "Unknown interface %s\n", wine_dbgstr_guid(riid));
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
static ULONG WINAPI driver_AddRef(IDispenserDriver *iface)
|
|
{
|
|
trace("AddRef\n");
|
|
return 2;
|
|
}
|
|
|
|
static ULONG WINAPI driver_Release(IDispenserDriver *iface)
|
|
{
|
|
CHECK_EXPECT2(driver_Release);
|
|
return 1;
|
|
}
|
|
|
|
static HRESULT WINAPI driver_CreateResource(IDispenserDriver *iface, const RESTYPID restypid, RESID *resid, TIMEINSECS *destroy)
|
|
{
|
|
CHECK_EXPECT2(driver_CreateResource);
|
|
|
|
*resid = 10;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI driver_RateResource(IDispenserDriver *iface, const RESTYPID restypid, const RESID resid,
|
|
const BOOL requires, RESOURCERATING *rating)
|
|
{
|
|
ok(0, "unexpected call\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI driver_EnlistResource(IDispenserDriver *iface, const RESID resid, const TRANSID transid)
|
|
{
|
|
ok(0, "unexpected call\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI driver_ResetResource(IDispenserDriver *iface, const RESID resid)
|
|
{
|
|
CHECK_EXPECT2(driver_ResetResource);
|
|
ok((int)resid == 10, "RESID %d\n", (int)resid);
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI driver_DestroyResource(IDispenserDriver *iface, const RESID resid)
|
|
{
|
|
todo_wine CHECK_EXPECT2(driver_DestroyResource);
|
|
return driver_DestroyResource_ret;
|
|
}
|
|
|
|
static HRESULT WINAPI driver_DestroyResourceS(IDispenserDriver *iface, const SRESID resid)
|
|
{
|
|
ok(0, "unexpected call\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
|
|
static const struct IDispenserDriverVtbl driver_vtbl =
|
|
{
|
|
driver_QueryInterface,
|
|
driver_AddRef,
|
|
driver_Release,
|
|
driver_CreateResource,
|
|
driver_RateResource,
|
|
driver_EnlistResource,
|
|
driver_ResetResource,
|
|
driver_DestroyResource,
|
|
driver_DestroyResourceS
|
|
};
|
|
|
|
static IDispenserDriver DispenserDriver = { &driver_vtbl };
|
|
|
|
static DWORD WINAPI com_thread(void *arg)
|
|
{
|
|
IUnknown *unk;
|
|
HRESULT hr;
|
|
|
|
hr = CoCreateInstance(&CLSID_XMLDocument, NULL, CLSCTX_INPROC_SERVER, &IID_IUnknown, (void **)&unk);
|
|
if (hr == S_OK) IUnknown_Release(unk);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static void create_dispenser(void)
|
|
{
|
|
static const WCHAR pool0[] = {'S','C','.','P','o','o','l',' ','0',' ','0',0};
|
|
static const WCHAR pool1[] = {'S','C','.','P','o','o','l',' ','1',' ','1',0};
|
|
static const WCHAR data[] = {'d','a','t','a','1',0};
|
|
HRESULT hr;
|
|
IDispenserManager *dispenser = NULL;
|
|
IHolder *holder1 = NULL, *holder2 = NULL, *holder3 = NULL;
|
|
HANDLE thread;
|
|
RESID resid;
|
|
DWORD ret;
|
|
BSTR str;
|
|
|
|
hr = CoCreateInstance( &CLSID_DispenserManager, NULL, CLSCTX_ALL, &IID_IDispenserManager, (void**)&dispenser);
|
|
ok(hr == S_OK, "Failed to create object 0x%08x\n", hr);
|
|
if(FAILED(hr))
|
|
{
|
|
win_skip("DispenserManager not available\n");
|
|
return;
|
|
}
|
|
|
|
thread = CreateThread(NULL, 0, com_thread, NULL, 0, NULL);
|
|
ok(!WaitForSingleObject(thread, 1000), "wait failed\n");
|
|
GetExitCodeThread(thread, &ret);
|
|
ok(ret == CO_E_NOTINITIALIZED, "got unexpected hr %#x\n", ret);
|
|
|
|
hr = IDispenserManager_RegisterDispenser(dispenser, &DispenserDriver, pool0, &holder1);
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
/* The above call creates an MTA thread, but we need to wait for it to
|
|
* actually initialize. */
|
|
Sleep(200);
|
|
thread = CreateThread(NULL, 0, com_thread, NULL, 0, NULL);
|
|
ok(!WaitForSingleObject(thread, 20000), "wait failed\n");
|
|
GetExitCodeThread(thread, &ret);
|
|
ok(ret == S_OK, "got unexpected hr %#x\n", ret);
|
|
|
|
hr = IDispenserManager_RegisterDispenser(dispenser, &DispenserDriver, pool0, &holder2);
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
ok(holder1 != holder2, "same holder object returned\n");
|
|
|
|
hr = IDispenserManager_RegisterDispenser(dispenser, &DispenserDriver, pool1, &holder3);
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
|
|
if(holder1)
|
|
{
|
|
SET_EXPECT(driver_CreateResource);
|
|
SET_EXPECT(driver_Release);
|
|
|
|
str = SysAllocString(data);
|
|
hr = IHolder_AllocResource(holder1, (RESTYPID)str, &resid);
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
ok(resid == 10, "got %d\n", (int)resid);
|
|
SysFreeString(str);
|
|
|
|
CHECK_CALLED(driver_CreateResource);
|
|
todo_wine CHECK_CALLED_BROKEN(driver_Release);
|
|
|
|
SET_EXPECT(driver_ResetResource);
|
|
hr = IHolder_FreeResource(holder1, resid);
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
todo_wine CHECK_CALLED(driver_ResetResource);
|
|
|
|
SET_EXPECT(driver_DestroyResource);
|
|
SET_EXPECT(driver_Release);
|
|
hr = IHolder_Close(holder1);
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
CHECK_CALLED(driver_Release);
|
|
CHECK_CALLED(driver_DestroyResource);
|
|
|
|
IHolder_Release(holder1);
|
|
}
|
|
if(holder2)
|
|
{
|
|
SET_EXPECT(driver_CreateResource);
|
|
SET_EXPECT(driver_Release);
|
|
|
|
str = SysAllocString(data);
|
|
hr = IHolder_AllocResource(holder2, (RESTYPID)str, &resid);
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
ok(resid == 10, "got %d\n", (int)resid);
|
|
SysFreeString(str);
|
|
|
|
CHECK_CALLED(driver_CreateResource);
|
|
todo_wine CHECK_CALLED_BROKEN(driver_Release);
|
|
|
|
SET_EXPECT(driver_ResetResource);
|
|
hr = IHolder_FreeResource(holder2, resid);
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
todo_wine CHECK_CALLED(driver_ResetResource);
|
|
|
|
/* DestroyResource return doesn't directly affect the Holder Close return value */
|
|
driver_DestroyResource_ret = E_FAIL;
|
|
SET_EXPECT(driver_DestroyResource);
|
|
SET_EXPECT(driver_Release);
|
|
hr = IHolder_Close(holder2);
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
CHECK_CALLED(driver_Release);
|
|
CHECK_CALLED(driver_DestroyResource);
|
|
driver_DestroyResource_ret = S_OK;
|
|
IHolder_Release(holder2);
|
|
}
|
|
if(holder3)
|
|
{
|
|
SET_EXPECT(driver_Release);
|
|
hr = IHolder_Close(holder3);
|
|
ok(hr == S_OK, "got 0x%08x\n", hr);
|
|
CHECK_CALLED(driver_Release);
|
|
IHolder_Release(holder3);
|
|
}
|
|
|
|
IDispenserManager_Release(dispenser);
|
|
}
|
|
|
|
static void test_new_moniker(void)
|
|
{
|
|
IMoniker *moniker, *moniker2, *inverse, *class_moniker;
|
|
IUnknown *obj, *obj2;
|
|
BIND_OPTS2 bind_opts;
|
|
ULARGE_INTEGER size;
|
|
DWORD moniker_type;
|
|
IBindCtx *bindctx;
|
|
FILETIME filetime;
|
|
DWORD hash, eaten;
|
|
IStream *stream;
|
|
HGLOBAL hglobal;
|
|
CLSID clsid;
|
|
HRESULT hr;
|
|
void *ptr;
|
|
|
|
hr = CreateBindCtx(0, &bindctx);
|
|
ok(hr == S_OK, "Failed to create bind context, hr %#x.\n", hr);
|
|
|
|
eaten = 0;
|
|
hr = MkParseDisplayName(bindctx, L"new:20d04fe0-3aea-1069-a2d8-08002b30309d", &eaten, &moniker);
|
|
ok(hr == S_OK, "Failed to parse display name, hr %#x.\n", hr);
|
|
ok(eaten == 40, "Unexpected eaten length %u.\n", eaten);
|
|
|
|
eaten = 0;
|
|
hr = IMoniker_ParseDisplayName(moniker, bindctx, NULL, (WCHAR *)L"new:20d04fe0-3aea-1069-a2d8-08002b30309d",
|
|
&eaten, &moniker2);
|
|
ok(hr == S_OK, "Failed to parse display name, hr %#x.\n", hr);
|
|
ok(eaten == 40, "Unexpected eaten length %u.\n", eaten);
|
|
IMoniker_Release(moniker2);
|
|
|
|
hr = IMoniker_QueryInterface(moniker, &IID_IParseDisplayName, (void **)&obj);
|
|
ok(hr == E_NOINTERFACE, "Unexpected hr %#x.\n", hr);
|
|
|
|
/* Object creation. */
|
|
hr = CLSIDFromProgID(L"new", &clsid);
|
|
ok(hr == S_OK, "Failed to get clsid, hr %#x.\n", hr);
|
|
|
|
hr = CreateClassMoniker(&clsid, &class_moniker);
|
|
ok(hr == S_OK, "Failed to create class moniker, hr %#x.\n", hr);
|
|
|
|
hr = IMoniker_BindToObject(class_moniker, bindctx, NULL, &IID_IParseDisplayName, (void **)&obj);
|
|
ok(hr == S_OK, "Failed to get parsing interface, hr %#x.\n", hr);
|
|
IUnknown_Release(obj);
|
|
|
|
hr = IMoniker_BindToObject(class_moniker, bindctx, NULL, &IID_IClassFactory, (void **)&obj);
|
|
ok(hr == S_OK, "Failed to get parsing interface, hr %#x.\n", hr);
|
|
IUnknown_Release(obj);
|
|
|
|
hr = CoGetClassObject(&clsid, CLSCTX_INPROC_SERVER, NULL, &IID_IParseDisplayName, (void **)&obj);
|
|
ok(hr == S_OK, "Failed to get parsing interface, hr %#x.\n", hr);
|
|
IUnknown_Release(obj);
|
|
|
|
hr = CoGetClassObject(&clsid, CLSCTX_INPROC_SERVER, NULL, &IID_IClassFactory, (void **)&obj);
|
|
ok(hr == S_OK, "Failed to get parsing interface, hr %#x.\n", hr);
|
|
|
|
hr = IUnknown_QueryInterface(obj, &IID_IParseDisplayName, (void **)&obj2);
|
|
ok(hr == S_OK, "Failed to get parsing interface, hr %#x.\n", hr);
|
|
IUnknown_Release(obj);
|
|
|
|
IMoniker_Release(class_moniker);
|
|
|
|
/* Hashing */
|
|
hash = 0;
|
|
hr = IMoniker_Hash(moniker, &hash);
|
|
ok(hr == S_OK, "Failed to get a hash, hr %#x.\n", hr);
|
|
ok(hash == 0x20d04fe0, "Unexpected hash value %#x.\n", hash);
|
|
|
|
moniker_type = MKSYS_CLASSMONIKER;
|
|
hr = IMoniker_IsSystemMoniker(moniker, &moniker_type);
|
|
ok(hr == S_FALSE || broken(hr == S_OK) /* XP */, "Unexpected hr %#x.\n", hr);
|
|
ok(moniker_type == MKSYS_NONE, "Unexpected moniker type %d.\n", moniker_type);
|
|
|
|
hr = IMoniker_IsRunning(moniker, NULL, NULL, NULL);
|
|
todo_wine
|
|
ok(hr == S_FALSE, "Unexpected hr %#x.\n", hr);
|
|
|
|
hr = IMoniker_IsRunning(moniker, bindctx, NULL, NULL);
|
|
todo_wine
|
|
ok(hr == S_FALSE, "Unexpected hr %#x.\n", hr);
|
|
|
|
hr = IMoniker_GetTimeOfLastChange(moniker, bindctx, NULL, &filetime);
|
|
ok(hr == MK_E_UNAVAILABLE, "Unexpected hr %#x.\n", hr);
|
|
|
|
hr = IMoniker_BindToObject(moniker, bindctx, NULL, &IID_IUnknown, (void **)&obj);
|
|
todo_wine
|
|
ok(hr == S_OK, "Failed to bind to object, hr %#x.\n", hr);
|
|
if (SUCCEEDED(hr))
|
|
IUnknown_Release(obj);
|
|
|
|
hr = IMoniker_BindToStorage(moniker, bindctx, NULL, &IID_IUnknown, (void **)&obj);
|
|
todo_wine
|
|
ok(hr == MK_E_NOSTORAGE, "Unexpected hr %#x.\n", hr);
|
|
|
|
hr = IMoniker_Inverse(moniker, &inverse);
|
|
ok(hr == S_OK, "Failed to create inverse moniker, hr %#x.\n", hr);
|
|
moniker_type = MKSYS_NONE;
|
|
hr = IMoniker_IsSystemMoniker(inverse, &moniker_type);
|
|
ok(hr == S_OK, "Unexpected hr %#x.\n", hr);
|
|
ok(moniker_type == MKSYS_ANTIMONIKER, "Unexpected moniker type %d.\n", moniker_type);
|
|
IMoniker_Release(inverse);
|
|
|
|
hr = IMoniker_Enum(moniker, FALSE, NULL);
|
|
ok(hr == E_POINTER, "Unexpected hr %#x.\n", hr);
|
|
|
|
obj = (IUnknown *)moniker;
|
|
hr = IMoniker_Enum(moniker, FALSE, (IEnumMoniker **)&obj);
|
|
ok(hr == S_OK, "Unexpected hr %#x.\n", hr);
|
|
ok(obj == NULL, "Unexpected return value.\n");
|
|
|
|
/* Serialization. */
|
|
hr = IMoniker_GetSizeMax(moniker, NULL);
|
|
ok(hr == E_POINTER, "Unexpected hr %#x.\n", hr);
|
|
|
|
size.QuadPart = 0;
|
|
hr = IMoniker_GetSizeMax(moniker, &size);
|
|
ok(hr == S_OK, "Failed to get size, hr %#x.\n", hr);
|
|
ok(size.QuadPart == (sizeof(GUID) + 2 * sizeof(DWORD)), "Unexpected size %s.\n",
|
|
wine_dbgstr_longlong(size.QuadPart));
|
|
|
|
hr = CreateStreamOnHGlobal(NULL, TRUE, &stream);
|
|
ok(hr == S_OK, "Failed to create a stream, hr %#x.\n", hr);
|
|
|
|
hr = IMoniker_Save(moniker, stream, FALSE);
|
|
ok(hr == S_OK, "Failed to save moniker, hr %#x.\n", hr);
|
|
|
|
hr = GetHGlobalFromStream(stream, &hglobal);
|
|
ok(hr == S_OK, "Failed to get a handle, hr %#x.\n", hr);
|
|
|
|
ptr = GlobalLock(hglobal);
|
|
ok(!!ptr, "Failed to get data pointer.\n");
|
|
|
|
hr = CLSIDFromString(L"{20d04fe0-3aea-1069-a2d8-08002b30309d}", &clsid);
|
|
ok(hr == S_OK, "Failed to get CLSID, hr %#x.\n", hr);
|
|
ok(IsEqualGUID((GUID *)ptr, &clsid), "Unexpected buffer content.\n");
|
|
ok(*(DWORD *)((BYTE *)ptr + sizeof(GUID)) == 0, "Unexpected buffer content.\n");
|
|
ok(*(DWORD *)((BYTE *)ptr + sizeof(GUID) + sizeof(DWORD)) == 0, "Unexpected buffer content.\n");
|
|
|
|
GlobalUnlock(hglobal);
|
|
|
|
IStream_Release(stream);
|
|
|
|
hr = IMoniker_IsDirty(moniker);
|
|
ok(hr == S_FALSE, "Unexpected hr %#x.\n", hr);
|
|
|
|
hr = IMoniker_GetClassID(moniker, NULL);
|
|
ok(hr == E_POINTER, "Unexpected hr %#x.\n", hr);
|
|
|
|
IMoniker_Release(moniker);
|
|
|
|
/* Full path to create new object, using progid. */
|
|
memset(&bind_opts, 0, sizeof(bind_opts));
|
|
bind_opts.cbStruct = sizeof(bind_opts);
|
|
bind_opts.dwClassContext = CLSCTX_INPROC_SERVER;
|
|
|
|
hr = CoGetObject(L"new:msxml2.domdocument", (BIND_OPTS *)&bind_opts, &IID_IXMLDOMDocument, (void **)&obj);
|
|
todo_wine
|
|
ok(hr == S_OK, "Failed to create object, hr %#x.\n", hr);
|
|
if (SUCCEEDED(hr))
|
|
IUnknown_Release(obj);
|
|
|
|
IBindCtx_Release(bindctx);
|
|
}
|
|
|
|
START_TEST(comsvcs)
|
|
{
|
|
HRESULT hr;
|
|
|
|
hr = CoInitialize(0);
|
|
ok( hr == S_OK, "failed to init com\n");
|
|
if (hr != S_OK)
|
|
return;
|
|
|
|
create_dispenser();
|
|
test_new_moniker();
|
|
|
|
CoUninitialize();
|
|
}
|