825 lines
27 KiB
C
825 lines
27 KiB
C
/*
|
|
* Moniker Tests
|
|
*
|
|
* Copyright 2004 Robert Shearman
|
|
*
|
|
* 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 _WIN32_DCOM
|
|
#define COBJMACROS
|
|
|
|
#include <stdarg.h>
|
|
#include <stdio.h>
|
|
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "objbase.h"
|
|
#include "comcat.h"
|
|
|
|
#include "wine/test.h"
|
|
|
|
#define ok_ole_success(hr, func) ok(hr == S_OK, #func " failed with error 0x%08lx\n", hr)
|
|
#define COUNTOF(x) (sizeof(x) / sizeof(x[0]))
|
|
|
|
static const WCHAR wszFileName1[] = {'c',':','\\','w','i','n','d','o','w','s','\\','t','e','s','t','1','.','d','o','c',0};
|
|
static const WCHAR wszFileName2[] = {'c',':','\\','w','i','n','d','o','w','s','\\','t','e','s','t','2','.','d','o','c',0};
|
|
|
|
static int count_moniker_matches(IBindCtx * pbc, IEnumMoniker * spEM)
|
|
{
|
|
IMoniker * spMoniker;
|
|
int monCnt=0, matchCnt=0;
|
|
|
|
while ((IEnumMoniker_Next(spEM, 1, &spMoniker, NULL)==S_OK))
|
|
{
|
|
HRESULT hr;
|
|
WCHAR * szDisplayn;
|
|
monCnt++;
|
|
hr=IMoniker_GetDisplayName(spMoniker, pbc, NULL, &szDisplayn);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (!lstrcmpW(szDisplayn, wszFileName1) || !lstrcmpW(szDisplayn, wszFileName2))
|
|
matchCnt++;
|
|
CoTaskMemFree(szDisplayn);
|
|
}
|
|
}
|
|
trace("Total number of monikers is %i\n", monCnt);
|
|
return matchCnt;
|
|
}
|
|
|
|
static void test_MkParseDisplayName(void)
|
|
{
|
|
IBindCtx * pbc = NULL;
|
|
HRESULT hr;
|
|
IMoniker * pmk = NULL;
|
|
IMoniker * pmk1 = NULL;
|
|
IMoniker * pmk2 = NULL;
|
|
ULONG eaten;
|
|
int matchCnt;
|
|
IUnknown * object = NULL;
|
|
|
|
IUnknown *lpEM1;
|
|
|
|
IEnumMoniker *spEM1 = NULL;
|
|
IEnumMoniker *spEM2 = NULL;
|
|
IEnumMoniker *spEM3 = NULL;
|
|
|
|
DWORD pdwReg1=0;
|
|
DWORD grflags=0;
|
|
DWORD pdwReg2=0;
|
|
IRunningObjectTable * pprot=NULL;
|
|
|
|
/* CLSID of My Computer */
|
|
static const WCHAR wszDisplayName[] = {'c','l','s','i','d',':',
|
|
'2','0','D','0','4','F','E','0','-','3','A','E','A','-','1','0','6','9','-','A','2','D','8','-','0','8','0','0','2','B','3','0','3','0','9','D',':',0};
|
|
|
|
hr = CreateBindCtx(0, &pbc);
|
|
ok_ole_success(hr, CreateBindCtx);
|
|
|
|
hr = MkParseDisplayName(pbc, wszDisplayName, &eaten, &pmk);
|
|
todo_wine { ok_ole_success(hr, MkParseDisplayName); }
|
|
|
|
if (object)
|
|
{
|
|
hr = IMoniker_BindToObject(pmk, pbc, NULL, &IID_IUnknown, (LPVOID*)&object);
|
|
ok_ole_success(hr, IMoniker_BindToObject);
|
|
|
|
IUnknown_Release(object);
|
|
}
|
|
IBindCtx_Release(pbc);
|
|
|
|
/* Test the EnumMoniker interface */
|
|
hr = CreateBindCtx(0, &pbc);
|
|
ok_ole_success(hr, CreateBindCtx);
|
|
|
|
hr = CreateFileMoniker(wszFileName1, &pmk1);
|
|
ok(hr==0, "CreateFileMoniker for file hr=%08lx\n", hr);
|
|
hr = CreateFileMoniker(wszFileName2, &pmk2);
|
|
ok(hr==0, "CreateFileMoniker for file hr=%08lx\n", hr);
|
|
hr = IBindCtx_GetRunningObjectTable(pbc, &pprot);
|
|
ok(hr==0, "IBindCtx_GetRunningObjectTable hr=%08lx\n", hr);
|
|
|
|
/* Check EnumMoniker before registering */
|
|
hr = IRunningObjectTable_EnumRunning(pprot, &spEM1);
|
|
ok(hr==0, "IRunningObjectTable_EnumRunning hr=%08lx\n", hr);
|
|
hr = IEnumMoniker_QueryInterface(spEM1, &IID_IUnknown, (void*) &lpEM1);
|
|
/* Register a couple of Monikers and check is ok */
|
|
ok(hr==0, "IEnumMoniker_QueryInterface hr %08lx %p\n", hr, lpEM1);
|
|
hr = MK_E_NOOBJECT;
|
|
|
|
matchCnt = count_moniker_matches(pbc, spEM1);
|
|
trace("Number of matches is %i\n", matchCnt);
|
|
|
|
grflags= grflags | ROTFLAGS_REGISTRATIONKEEPSALIVE;
|
|
hr = IRunningObjectTable_Register(pprot, grflags, lpEM1, pmk1, &pdwReg1);
|
|
ok(hr==0, "IRunningObjectTable_Register hr=%08lx %p %08lx %p %p %ld\n",
|
|
hr, pprot, grflags, lpEM1, pmk1, pdwReg1);
|
|
|
|
trace("IROT::Register\n");
|
|
grflags=0;
|
|
grflags= grflags | ROTFLAGS_REGISTRATIONKEEPSALIVE;
|
|
hr = IRunningObjectTable_Register(pprot, grflags, lpEM1, pmk2, &pdwReg2);
|
|
ok(hr==0, "IRunningObjectTable_Register hr=%08lx %p %08lx %p %p %ld\n", hr,
|
|
pprot, grflags, lpEM1, pmk2, pdwReg2);
|
|
|
|
hr = IRunningObjectTable_EnumRunning(pprot, &spEM2);
|
|
ok(hr==0, "IRunningObjectTable_EnumRunning hr=%08lx\n", hr);
|
|
|
|
matchCnt = count_moniker_matches(pbc, spEM2);
|
|
ok(matchCnt==2, "Number of matches should be equal to 2 not %i\n", matchCnt);
|
|
|
|
trace("IEnumMoniker::Clone\n");
|
|
IEnumMoniker_Clone(spEM2, &spEM3);
|
|
|
|
matchCnt = count_moniker_matches(pbc, spEM3);
|
|
ok(matchCnt==0, "Number of matches should be equal to 0 not %i\n", matchCnt);
|
|
trace("IEnumMoniker::Reset\n");
|
|
IEnumMoniker_Reset(spEM3);
|
|
|
|
matchCnt = count_moniker_matches(pbc, spEM3);
|
|
ok(matchCnt==2, "Number of matches should be equal to 2 not %i\n", matchCnt);
|
|
|
|
IRunningObjectTable_Revoke(pprot,pdwReg1);
|
|
IRunningObjectTable_Revoke(pprot,pdwReg2);
|
|
IEnumMoniker_Release(spEM1);
|
|
IEnumMoniker_Release(spEM1);
|
|
IEnumMoniker_Release(spEM2);
|
|
IEnumMoniker_Release(spEM3);
|
|
IMoniker_Release(pmk1);
|
|
IMoniker_Release(pmk2);
|
|
IRunningObjectTable_Release(pprot);
|
|
|
|
IBindCtx_Release(pbc);
|
|
}
|
|
|
|
static const LARGE_INTEGER llZero;
|
|
|
|
static const BYTE expected_class_moniker_marshal_data[] =
|
|
{
|
|
0x4d,0x45,0x4f,0x57,0x04,0x00,0x00,0x00,
|
|
0x0f,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x1a,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x00,0x00,0x00,0x00,0x14,0x00,0x00,0x00,
|
|
0x05,0xe0,0x02,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x00,0x00,0x00,0x00,
|
|
};
|
|
|
|
static const BYTE expected_class_moniker_saved_data[] =
|
|
{
|
|
0x05,0xe0,0x02,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x00,0x00,0x00,0x00,
|
|
};
|
|
|
|
static const BYTE expected_class_moniker_comparison_data[] =
|
|
{
|
|
0x1a,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x05,0xe0,0x02,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
};
|
|
|
|
static const WCHAR expected_class_moniker_display_name[] =
|
|
{
|
|
'c','l','s','i','d',':','0','0','0','2','E','0','0','5','-','0','0','0',
|
|
'0','-','0','0','0','0','-','C','0','0','0','-','0','0','0','0','0','0',
|
|
'0','0','0','0','4','6',':',0
|
|
};
|
|
|
|
static const BYTE expected_item_moniker_comparison_data[] =
|
|
{
|
|
0x04,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x21,0x00,0x54,0x00,0x45,0x00,0x53,0x00,
|
|
0x54,0x00,0x00,0x00,
|
|
};
|
|
|
|
static const BYTE expected_item_moniker_saved_data[] =
|
|
{
|
|
0x02,0x00,0x00,0x00,0x21,0x00,0x05,0x00,
|
|
0x00,0x00,0x54,0x65,0x73,0x74,0x00,
|
|
};
|
|
|
|
static const BYTE expected_item_moniker_marshal_data[] =
|
|
{
|
|
0x4d,0x45,0x4f,0x57,0x04,0x00,0x00,0x00,
|
|
0x0f,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x04,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x00,0x00,0x00,0x00,0x36,0x00,0x00,0x00,
|
|
0x02,0x00,0x00,0x00,0x21,0x00,0x05,0x00,
|
|
0x00,0x00,0x54,0x65,0x73,0x74,0x00,
|
|
};
|
|
|
|
static const BYTE expected_anti_moniker_marshal_data[] =
|
|
{
|
|
0x4d,0x45,0x4f,0x57,0x04,0x00,0x00,0x00,
|
|
0x0f,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x05,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x00,0x00,0x00,0x00,0x14,0x00,0x00,0x00,
|
|
0x01,0x00,0x00,0x00,
|
|
};
|
|
|
|
static const BYTE expected_anti_moniker_saved_data[] =
|
|
{
|
|
0x01,0x00,0x00,0x00,
|
|
};
|
|
|
|
static const BYTE expected_anti_moniker_comparison_data[] =
|
|
{
|
|
0x05,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x01,0x00,0x00,0x00,
|
|
};
|
|
|
|
static const BYTE expected_gc_moniker_marshal_data[] =
|
|
{
|
|
0x4d,0x45,0x4f,0x57,0x04,0x00,0x00,0x00,
|
|
0x0f,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x09,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x00,0x00,0x00,0x00,0x2c,0x01,0x00,0x00,
|
|
0x4d,0x45,0x4f,0x57,0x04,0x00,0x00,0x00,
|
|
0x0f,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x04,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x00,0x00,0x00,0x00,0x36,0x00,0x00,0x00,
|
|
0x02,0x00,0x00,0x00,0x21,0x00,0x05,0x00,
|
|
0x00,0x00,0x54,0x65,0x73,0x74,0x00,0x4d,
|
|
0x45,0x4f,0x57,0x04,0x00,0x00,0x00,0x0f,
|
|
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xc0,
|
|
0x00,0x00,0x00,0x00,0x00,0x00,0x46,0x04,
|
|
0x03,0x00,0x00,0x00,0x00,0x00,0x00,0xc0,
|
|
0x00,0x00,0x00,0x00,0x00,0x00,0x46,0x00,
|
|
0x00,0x00,0x00,0x36,0x00,0x00,0x00,0x02,
|
|
0x00,0x00,0x00,0x23,0x00,0x05,0x00,0x00,
|
|
0x00,0x57,0x69,0x6e,0x65,0x00,
|
|
};
|
|
|
|
static const BYTE expected_gc_moniker_saved_data[] =
|
|
{
|
|
0x02,0x00,0x00,0x00,0x04,0x03,0x00,0x00,
|
|
0x00,0x00,0x00,0x00,0xc0,0x00,0x00,0x00,
|
|
0x00,0x00,0x00,0x46,0x02,0x00,0x00,0x00,
|
|
0x21,0x00,0x05,0x00,0x00,0x00,0x54,0x65,
|
|
0x73,0x74,0x00,0x04,0x03,0x00,0x00,0x00,
|
|
0x00,0x00,0x00,0xc0,0x00,0x00,0x00,0x00,
|
|
0x00,0x00,0x46,0x02,0x00,0x00,0x00,0x23,
|
|
0x00,0x05,0x00,0x00,0x00,0x57,0x69,0x6e,
|
|
0x65,0x00,
|
|
};
|
|
|
|
static const BYTE expected_gc_moniker_comparison_data[] =
|
|
{
|
|
0x09,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x04,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
|
|
0x21,0x00,0x54,0x00,0x45,0x00,0x53,0x00,
|
|
0x54,0x00,0x00,0x00,0x04,0x03,0x00,0x00,
|
|
0x00,0x00,0x00,0x00,0xc0,0x00,0x00,0x00,
|
|
0x00,0x00,0x00,0x46,0x23,0x00,0x57,0x00,
|
|
0x49,0x00,0x4e,0x00,0x45,0x00,0x00,0x00,
|
|
};
|
|
|
|
static void test_moniker(
|
|
const char *testname, IMoniker *moniker,
|
|
const BYTE *expected_moniker_marshal_data, unsigned int sizeof_expected_moniker_marshal_data,
|
|
const BYTE *expected_moniker_saved_data, unsigned int sizeof_expected_moniker_saved_data,
|
|
const BYTE *expected_moniker_comparison_data, unsigned int sizeof_expected_moniker_comparison_data,
|
|
LPCWSTR expected_display_name)
|
|
{
|
|
IStream * stream;
|
|
IROTData * rotdata;
|
|
HRESULT hr;
|
|
HGLOBAL hglobal;
|
|
LPBYTE moniker_data;
|
|
DWORD moniker_size;
|
|
DWORD i;
|
|
BOOL same = TRUE;
|
|
BYTE buffer[128];
|
|
IMoniker * moniker_proxy;
|
|
LPOLESTR display_name;
|
|
IBindCtx *bindctx;
|
|
|
|
hr = IMoniker_IsDirty(moniker);
|
|
ok(hr == S_FALSE, "%s: IMoniker_IsDirty should return S_FALSE, not 0x%08lx\n", testname, hr);
|
|
|
|
/* Display Name */
|
|
|
|
hr = CreateBindCtx(0, &bindctx);
|
|
ok_ole_success(hr, CreateBindCtx);
|
|
|
|
hr = IMoniker_GetDisplayName(moniker, bindctx, NULL, &display_name);
|
|
ok_ole_success(hr, IMoniker_GetDisplayName);
|
|
ok(!lstrcmpW(display_name, expected_display_name), "display name wasn't what was expected\n");
|
|
|
|
CoTaskMemFree(display_name);
|
|
IBindCtx_Release(bindctx);
|
|
|
|
hr = IMoniker_IsDirty(moniker);
|
|
ok(hr == S_FALSE, "%s: IMoniker_IsDirty should return S_FALSE, not 0x%08lx\n", testname, hr);
|
|
|
|
/* IROTData::GetComparisonData test */
|
|
|
|
hr = IMoniker_QueryInterface(moniker, &IID_IROTData, (void **)&rotdata);
|
|
ok_ole_success(hr, IMoniker_QueryInterface_IID_IROTData);
|
|
|
|
hr = IROTData_GetComparisonData(rotdata, buffer, sizeof(buffer), &moniker_size);
|
|
ok_ole_success(hr, IROTData_GetComparisonData);
|
|
|
|
if (hr != S_OK) moniker_size = 0;
|
|
|
|
/* first check we have the right amount of data */
|
|
ok(moniker_size == sizeof_expected_moniker_comparison_data,
|
|
"%s: Size of comparison data differs (expected %d, actual %ld)\n",
|
|
testname, sizeof_expected_moniker_comparison_data, moniker_size);
|
|
|
|
/* then do a byte-by-byte comparison */
|
|
for (i = 0; i < min(moniker_size, sizeof_expected_moniker_comparison_data); i++)
|
|
{
|
|
if (expected_moniker_comparison_data[i] != buffer[i])
|
|
{
|
|
same = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
ok(same, "%s: Comparison data differs\n", testname);
|
|
if (!same)
|
|
{
|
|
for (i = 0; i < moniker_size; i++)
|
|
{
|
|
if (i % 8 == 0) printf(" ");
|
|
printf("0x%02x,", buffer[i]);
|
|
if (i % 8 == 7) printf("\n");
|
|
}
|
|
printf("\n");
|
|
}
|
|
|
|
IROTData_Release(rotdata);
|
|
|
|
hr = CreateStreamOnHGlobal(NULL, TRUE, &stream);
|
|
|
|
/* Saving */
|
|
|
|
hr = IMoniker_Save(moniker, stream, TRUE);
|
|
ok_ole_success(hr, IMoniker_Save);
|
|
|
|
hr = GetHGlobalFromStream(stream, &hglobal);
|
|
ok_ole_success(hr, GetHGlobalFromStream);
|
|
|
|
moniker_size = GlobalSize(hglobal);
|
|
|
|
moniker_data = GlobalLock(hglobal);
|
|
|
|
/* first check we have the right amount of data */
|
|
ok(moniker_size == sizeof_expected_moniker_saved_data,
|
|
"%s: Size of saved data differs (expected %d, actual %ld)\n",
|
|
testname, sizeof_expected_moniker_saved_data, moniker_size);
|
|
|
|
/* then do a byte-by-byte comparison */
|
|
for (i = 0; i < min(moniker_size, sizeof_expected_moniker_saved_data); i++)
|
|
{
|
|
if (expected_moniker_saved_data[i] != moniker_data[i])
|
|
{
|
|
same = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
ok(same, "%s: Saved data differs\n", testname);
|
|
if (!same)
|
|
{
|
|
for (i = 0; i < moniker_size; i++)
|
|
{
|
|
if (i % 8 == 0) printf(" ");
|
|
printf("0x%02x,", moniker_data[i]);
|
|
if (i % 8 == 7) printf("\n");
|
|
}
|
|
printf("\n");
|
|
}
|
|
|
|
GlobalUnlock(hglobal);
|
|
|
|
IStream_Release(stream);
|
|
|
|
/* Marshaling tests */
|
|
|
|
hr = CreateStreamOnHGlobal(NULL, TRUE, &stream);
|
|
ok_ole_success(hr, CreateStreamOnHGlobal);
|
|
|
|
hr = CoMarshalInterface(stream, &IID_IMoniker, (IUnknown *)moniker, MSHCTX_INPROC, NULL, MSHLFLAGS_NORMAL);
|
|
ok_ole_success(hr, CoMarshalInterface);
|
|
|
|
hr = GetHGlobalFromStream(stream, &hglobal);
|
|
ok_ole_success(hr, GetHGlobalFromStream);
|
|
|
|
moniker_size = GlobalSize(hglobal);
|
|
|
|
moniker_data = GlobalLock(hglobal);
|
|
|
|
/* first check we have the right amount of data */
|
|
ok(moniker_size == sizeof_expected_moniker_marshal_data,
|
|
"%s: Size of marshaled data differs (expected %d, actual %ld)\n",
|
|
testname, sizeof_expected_moniker_marshal_data, moniker_size);
|
|
|
|
/* then do a byte-by-byte comparison */
|
|
if (expected_moniker_marshal_data)
|
|
{
|
|
for (i = 0; i < min(moniker_size, sizeof_expected_moniker_marshal_data); i++)
|
|
{
|
|
if (expected_moniker_marshal_data[i] != moniker_data[i])
|
|
{
|
|
same = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
ok(same, "%s: Marshaled data differs\n", testname);
|
|
if (!same)
|
|
{
|
|
for (i = 0; i < moniker_size; i++)
|
|
{
|
|
if (i % 8 == 0) printf(" ");
|
|
printf("0x%02x,", moniker_data[i]);
|
|
if (i % 8 == 7) printf("\n");
|
|
}
|
|
printf("\n");
|
|
}
|
|
|
|
GlobalUnlock(hglobal);
|
|
|
|
IStream_Seek(stream, llZero, STREAM_SEEK_SET, NULL);
|
|
hr = CoUnmarshalInterface(stream, &IID_IMoniker, (void **)&moniker_proxy);
|
|
ok_ole_success(hr, CoUnmarshalInterface);
|
|
|
|
IStream_Release(stream);
|
|
IMoniker_Release(moniker_proxy);
|
|
}
|
|
|
|
static void test_class_moniker(void)
|
|
{
|
|
HRESULT hr;
|
|
IMoniker *moniker;
|
|
DWORD moniker_type;
|
|
DWORD hash;
|
|
IBindCtx *bindctx;
|
|
IMoniker *inverse;
|
|
IUnknown *unknown;
|
|
FILETIME filetime;
|
|
|
|
hr = CreateClassMoniker(&CLSID_StdComponentCategoriesMgr, &moniker);
|
|
ok_ole_success(hr, CreateClassMoniker);
|
|
if (!moniker) return;
|
|
|
|
test_moniker("class moniker", moniker,
|
|
expected_class_moniker_marshal_data, sizeof(expected_class_moniker_marshal_data),
|
|
expected_class_moniker_saved_data, sizeof(expected_class_moniker_saved_data),
|
|
expected_class_moniker_comparison_data, sizeof(expected_class_moniker_comparison_data),
|
|
expected_class_moniker_display_name);
|
|
|
|
/* Hashing */
|
|
|
|
hr = IMoniker_Hash(moniker, &hash);
|
|
ok_ole_success(hr, IMoniker_Hash);
|
|
|
|
ok(hash == CLSID_StdComponentCategoriesMgr.Data1,
|
|
"Hash value != Data1 field of clsid, instead was 0x%08lx\n",
|
|
hash);
|
|
|
|
/* IsSystemMoniker test */
|
|
|
|
hr = IMoniker_IsSystemMoniker(moniker, &moniker_type);
|
|
ok_ole_success(hr, IMoniker_IsSystemMoniker);
|
|
|
|
ok(moniker_type == MKSYS_CLASSMONIKER,
|
|
"dwMkSys != MKSYS_CLASSMONIKER, instead was 0x%08lx\n",
|
|
moniker_type);
|
|
|
|
hr = CreateBindCtx(0, &bindctx);
|
|
ok_ole_success(hr, CreateBindCtx);
|
|
|
|
/* IsRunning test */
|
|
hr = IMoniker_IsRunning(moniker, bindctx, NULL, NULL);
|
|
ok(hr == E_NOTIMPL, "IMoniker_IsRunning should return E_NOTIMPL, not 0x%08lx\n", hr);
|
|
|
|
hr = IMoniker_GetTimeOfLastChange(moniker, bindctx, NULL, &filetime);
|
|
ok(hr == MK_E_UNAVAILABLE, "IMoniker_GetTimeOfLastChange should return MK_E_UNAVAILABLE, not 0x%08lx\n", hr);
|
|
|
|
hr = IMoniker_BindToObject(moniker, bindctx, NULL, &IID_IUnknown, (void **)&unknown);
|
|
ok_ole_success(hr, IMoniker_BindToStorage);
|
|
IUnknown_Release(unknown);
|
|
|
|
hr = IMoniker_BindToStorage(moniker, bindctx, NULL, &IID_IUnknown, (void **)&unknown);
|
|
ok_ole_success(hr, IMoniker_BindToStorage);
|
|
IUnknown_Release(unknown);
|
|
|
|
IBindCtx_Release(bindctx);
|
|
|
|
hr = IMoniker_Inverse(moniker, &inverse);
|
|
ok_ole_success(hr, IMoniker_Inverse);
|
|
IMoniker_Release(inverse);
|
|
|
|
IMoniker_Release(moniker);
|
|
}
|
|
|
|
static void test_file_moniker(WCHAR* path)
|
|
{
|
|
IStream *stream;
|
|
IMoniker *moniker1 = NULL, *moniker2 = NULL;
|
|
HRESULT hr;
|
|
|
|
hr = CreateFileMoniker(path, &moniker1);
|
|
ok_ole_success(hr, CreateFileMoniker);
|
|
|
|
hr = CreateStreamOnHGlobal(NULL, TRUE, &stream);
|
|
|
|
/* Marshal */
|
|
hr = CoMarshalInterface(stream, &IID_IMoniker, (IUnknown *)moniker1, MSHCTX_INPROC, NULL, MSHLFLAGS_NORMAL);
|
|
ok_ole_success(hr, CoMarshalInterface);
|
|
|
|
/* Rewind */
|
|
hr = IStream_Seek(stream, llZero, STREAM_SEEK_SET, NULL);
|
|
ok_ole_success(hr, IStream_Seek);
|
|
|
|
/* Unmarshal */
|
|
hr = CoUnmarshalInterface(stream, &IID_IMoniker, (void**)&moniker2);
|
|
ok_ole_success(hr, CoUnmarshalInterface);
|
|
|
|
hr = IMoniker_IsEqual(moniker1, moniker2);
|
|
ok_ole_success(hr, IsEqual);
|
|
|
|
IStream_Release(stream);
|
|
if (moniker1)
|
|
IMoniker_Release(moniker1);
|
|
if (moniker2)
|
|
IMoniker_Release(moniker2);
|
|
}
|
|
|
|
static void test_file_monikers(void)
|
|
{
|
|
static WCHAR wszFile[][30] = {
|
|
{'\\', 'w','i','n','d','o','w','s','\\','s','y','s','t','e','m','\\','t','e','s','t','1','.','d','o','c',0},
|
|
{'\\', 'a','b','c','d','e','f','g','\\','h','i','j','k','l','\\','m','n','o','p','q','r','s','t','u','.','m','n','o',0},
|
|
/* These map to themselves in Windows-1252 & 932 (Shift-JIS) */
|
|
{0xc0, 0xc1, 0xc2, 0xc3, 0xc4, 0xc5, 0xc6, 0xc7, 0xc8, 0xc9, 0xca, 0xcb, 0xcc, 0xcd, 0xce, 0xcf, 0xd0, 0},
|
|
/* U+2020 = DAGGER = 0x86 (1252) = 0x813f (932)
|
|
* U+20AC = EURO SIGN = 0x80 (1252) = undef (932)
|
|
* U+0100 .. = Latin extended-A
|
|
*/
|
|
{0x20ac, 0x2020, 0x100, 0x101, 0x102, 0x103, 0x104, 0x105, 0x106, 0x107, 0x108, 0x109, 0x10a, 0x10b, 0x10c, 0},
|
|
};
|
|
|
|
int i;
|
|
|
|
trace("ACP is %u\n", GetACP());
|
|
|
|
for (i = 0; i < COUNTOF(wszFile); ++i)
|
|
{
|
|
int j ;
|
|
for (j = lstrlenW(wszFile[i]); j > 0; --j)
|
|
{
|
|
wszFile[i][j] = 0;
|
|
test_file_moniker(wszFile[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void test_item_moniker(void)
|
|
{
|
|
HRESULT hr;
|
|
IMoniker *moniker;
|
|
DWORD moniker_type;
|
|
DWORD hash;
|
|
IBindCtx *bindctx;
|
|
IMoniker *inverse;
|
|
IUnknown *unknown;
|
|
static const WCHAR wszDelimeter[] = {'!',0};
|
|
static const WCHAR wszObjectName[] = {'T','e','s','t',0};
|
|
static const WCHAR expected_display_name[] = { '!','T','e','s','t',0 };
|
|
|
|
hr = CreateItemMoniker(wszDelimeter, wszObjectName, &moniker);
|
|
ok_ole_success(hr, CreateItemMoniker);
|
|
|
|
test_moniker("item moniker", moniker,
|
|
expected_item_moniker_marshal_data, sizeof(expected_item_moniker_marshal_data),
|
|
expected_item_moniker_saved_data, sizeof(expected_item_moniker_saved_data),
|
|
expected_item_moniker_comparison_data, sizeof(expected_item_moniker_comparison_data),
|
|
expected_display_name);
|
|
|
|
/* Hashing */
|
|
|
|
hr = IMoniker_Hash(moniker, &hash);
|
|
ok_ole_success(hr, IMoniker_Hash);
|
|
|
|
ok(hash == 0x73c,
|
|
"Hash value != 0x73c, instead was 0x%08lx\n",
|
|
hash);
|
|
|
|
/* IsSystemMoniker test */
|
|
|
|
hr = IMoniker_IsSystemMoniker(moniker, &moniker_type);
|
|
ok_ole_success(hr, IMoniker_IsSystemMoniker);
|
|
|
|
ok(moniker_type == MKSYS_ITEMMONIKER,
|
|
"dwMkSys != MKSYS_ITEMMONIKER, instead was 0x%08lx\n",
|
|
moniker_type);
|
|
|
|
hr = CreateBindCtx(0, &bindctx);
|
|
ok_ole_success(hr, CreateBindCtx);
|
|
|
|
/* IsRunning test */
|
|
hr = IMoniker_IsRunning(moniker, bindctx, NULL, NULL);
|
|
ok(hr == S_FALSE, "IMoniker_IsRunning should return S_FALSE, not 0x%08lx\n", hr);
|
|
|
|
hr = IMoniker_BindToObject(moniker, bindctx, NULL, &IID_IUnknown, (void **)&unknown);
|
|
ok(hr == E_INVALIDARG, "IMoniker_BindToStorage should return E_INVALIDARG, not 0x%08lx\n", hr);
|
|
|
|
hr = IMoniker_BindToStorage(moniker, bindctx, NULL, &IID_IUnknown, (void **)&unknown);
|
|
ok(hr == E_INVALIDARG, "IMoniker_BindToObject should return E_INVALIDARG, not 0x%08lx\n", hr);
|
|
|
|
IBindCtx_Release(bindctx);
|
|
|
|
hr = IMoniker_Inverse(moniker, &inverse);
|
|
ok_ole_success(hr, IMoniker_Inverse);
|
|
IMoniker_Release(inverse);
|
|
|
|
IMoniker_Release(moniker);
|
|
}
|
|
|
|
static void test_anti_moniker(void)
|
|
{
|
|
HRESULT hr;
|
|
IMoniker *moniker;
|
|
DWORD moniker_type;
|
|
DWORD hash;
|
|
IBindCtx *bindctx;
|
|
FILETIME filetime;
|
|
IMoniker *inverse;
|
|
IUnknown *unknown;
|
|
static const WCHAR expected_display_name[] = { '\\','.','.',0 };
|
|
|
|
hr = CreateAntiMoniker(&moniker);
|
|
ok_ole_success(hr, CreateAntiMoniker);
|
|
if (!moniker) return;
|
|
|
|
test_moniker("anti moniker", moniker,
|
|
expected_anti_moniker_marshal_data, sizeof(expected_anti_moniker_marshal_data),
|
|
expected_anti_moniker_saved_data, sizeof(expected_anti_moniker_saved_data),
|
|
expected_anti_moniker_comparison_data, sizeof(expected_anti_moniker_comparison_data),
|
|
expected_display_name);
|
|
|
|
/* Hashing */
|
|
hr = IMoniker_Hash(moniker, &hash);
|
|
ok_ole_success(hr, IMoniker_Hash);
|
|
ok(hash == 0x80000001,
|
|
"Hash value != 0x80000001, instead was 0x%08lx\n",
|
|
hash);
|
|
|
|
/* IsSystemMoniker test */
|
|
hr = IMoniker_IsSystemMoniker(moniker, &moniker_type);
|
|
ok_ole_success(hr, IMoniker_IsSystemMoniker);
|
|
ok(moniker_type == MKSYS_ANTIMONIKER,
|
|
"dwMkSys != MKSYS_ANTIMONIKER, instead was 0x%08lx\n",
|
|
moniker_type);
|
|
|
|
hr = IMoniker_Inverse(moniker, &inverse);
|
|
ok(hr == MK_E_NOINVERSE, "IMoniker_Inverse should have returned MK_E_NOINVERSE instead of 0x%08lx\n", hr);
|
|
ok(inverse == NULL, "inverse should have been set to NULL instead of %p\n", inverse);
|
|
|
|
hr = CreateBindCtx(0, &bindctx);
|
|
ok_ole_success(hr, CreateBindCtx);
|
|
|
|
/* IsRunning test */
|
|
hr = IMoniker_IsRunning(moniker, bindctx, NULL, NULL);
|
|
ok(hr == S_FALSE, "IMoniker_IsRunning should return S_FALSE, not 0x%08lx\n", hr);
|
|
|
|
hr = IMoniker_GetTimeOfLastChange(moniker, bindctx, NULL, &filetime);
|
|
ok(hr == E_NOTIMPL, "IMoniker_GetTimeOfLastChange should return E_NOTIMPL, not 0x%08lx\n", hr);
|
|
|
|
hr = IMoniker_BindToObject(moniker, bindctx, NULL, &IID_IUnknown, (void **)&unknown);
|
|
ok(hr == E_NOTIMPL, "IMoniker_BindToObject should return E_NOTIMPL, not 0x%08lx\n", hr);
|
|
|
|
hr = IMoniker_BindToStorage(moniker, bindctx, NULL, &IID_IUnknown, (void **)&unknown);
|
|
ok(hr == E_NOTIMPL, "IMoniker_BindToStorage should return E_NOTIMPL, not 0x%08lx\n", hr);
|
|
|
|
IBindCtx_Release(bindctx);
|
|
|
|
IMoniker_Release(moniker);
|
|
}
|
|
|
|
static void test_generic_composite_moniker(void)
|
|
{
|
|
HRESULT hr;
|
|
IMoniker *moniker;
|
|
IMoniker *moniker1;
|
|
IMoniker *moniker2;
|
|
DWORD moniker_type;
|
|
DWORD hash;
|
|
IBindCtx *bindctx;
|
|
FILETIME filetime;
|
|
IMoniker *inverse;
|
|
IUnknown *unknown;
|
|
static const WCHAR wszDelimeter1[] = {'!',0};
|
|
static const WCHAR wszObjectName1[] = {'T','e','s','t',0};
|
|
static const WCHAR wszDelimeter2[] = {'#',0};
|
|
static const WCHAR wszObjectName2[] = {'W','i','n','e',0};
|
|
static const WCHAR expected_display_name[] = { '!','T','e','s','t','#','W','i','n','e',0 };
|
|
|
|
hr = CreateItemMoniker(wszDelimeter1, wszObjectName1, &moniker1);
|
|
ok_ole_success(hr, CreateItemMoniker);
|
|
hr = CreateItemMoniker(wszDelimeter2, wszObjectName2, &moniker2);
|
|
ok_ole_success(hr, CreateItemMoniker);
|
|
hr = CreateGenericComposite(moniker1, moniker2, &moniker);
|
|
ok_ole_success(hr, CreateGenericComposite);
|
|
|
|
test_moniker("generic composite moniker", moniker,
|
|
expected_gc_moniker_marshal_data, sizeof(expected_gc_moniker_marshal_data),
|
|
expected_gc_moniker_saved_data, sizeof(expected_gc_moniker_saved_data),
|
|
expected_gc_moniker_comparison_data, sizeof(expected_gc_moniker_comparison_data),
|
|
expected_display_name);
|
|
|
|
/* Hashing */
|
|
|
|
hr = IMoniker_Hash(moniker, &hash);
|
|
ok_ole_success(hr, IMoniker_Hash);
|
|
|
|
ok(hash == 0xd87,
|
|
"Hash value != 0xd87, instead was 0x%08lx\n",
|
|
hash);
|
|
|
|
/* IsSystemMoniker test */
|
|
|
|
hr = IMoniker_IsSystemMoniker(moniker, &moniker_type);
|
|
ok_ole_success(hr, IMoniker_IsSystemMoniker);
|
|
|
|
ok(moniker_type == MKSYS_GENERICCOMPOSITE,
|
|
"dwMkSys != MKSYS_GENERICCOMPOSITE, instead was 0x%08lx\n",
|
|
moniker_type);
|
|
|
|
hr = CreateBindCtx(0, &bindctx);
|
|
ok_ole_success(hr, CreateBindCtx);
|
|
|
|
/* IsRunning test */
|
|
hr = IMoniker_IsRunning(moniker, bindctx, NULL, NULL);
|
|
todo_wine
|
|
ok(hr == S_FALSE, "IMoniker_IsRunning should return S_FALSE, not 0x%08lx\n", hr);
|
|
|
|
hr = IMoniker_GetTimeOfLastChange(moniker, bindctx, NULL, &filetime);
|
|
ok(hr == MK_E_NOTBINDABLE, "IMoniker_GetTimeOfLastChange should return MK_E_NOTBINDABLE, not 0x%08lx\n", hr);
|
|
|
|
hr = IMoniker_BindToObject(moniker, bindctx, NULL, &IID_IUnknown, (void **)&unknown);
|
|
todo_wine
|
|
ok(hr == E_INVALIDARG, "IMoniker_BindToObject should return E_INVALIDARG, not 0x%08lx\n", hr);
|
|
|
|
todo_wine
|
|
hr = IMoniker_BindToStorage(moniker, bindctx, NULL, &IID_IUnknown, (void **)&unknown);
|
|
ok(hr == E_INVALIDARG, "IMoniker_BindToStorage should return E_INVALIDARG, not 0x%08lx\n", hr);
|
|
|
|
IBindCtx_Release(bindctx);
|
|
|
|
hr = IMoniker_Inverse(moniker, &inverse);
|
|
ok_ole_success(hr, IMoniker_Inverse);
|
|
IMoniker_Release(inverse);
|
|
|
|
IMoniker_Release(moniker);
|
|
}
|
|
|
|
START_TEST(moniker)
|
|
{
|
|
CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
|
|
|
|
test_MkParseDisplayName();
|
|
test_class_moniker();
|
|
test_file_monikers();
|
|
test_item_moniker();
|
|
test_anti_moniker();
|
|
test_generic_composite_moniker();
|
|
|
|
/* FIXME: test moniker creation funcs and parsing other moniker formats */
|
|
|
|
CoUninitialize();
|
|
}
|