mirror of
https://github.com/reactos/reactos.git
synced 2025-01-09 15:51:16 +00:00
2936e38c77
- Verify interfaces against HKLM\Software\Classes\Interface svn path=/trunk/; revision=62541
572 lines
24 KiB
C
572 lines
24 KiB
C
/*
|
|
* PROJECT: ReactOS api tests
|
|
* LICENSE: GPLv2+ - See COPYING in the top level directory
|
|
* PURPOSE: COM interface test
|
|
* PROGRAMMER: Thomas Faber <thomas.faber@reactos.org>
|
|
*/
|
|
|
|
#include "com_apitest.h"
|
|
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
#define myskip(c, ...) ((c) ? 0 : (skip(__VA_ARGS__), 1))
|
|
#define mytrace(...) do { \
|
|
int debug = winetest_debug; \
|
|
winetest_debug = 1; \
|
|
trace(__VA_ARGS__); \
|
|
winetest_debug = debug; \
|
|
} while (0)
|
|
|
|
typedef struct _KNOWN_INTERFACE
|
|
{
|
|
const IID *iid;
|
|
PCSTR name;
|
|
PCWSTR wname;
|
|
BOOLEAN noreg;
|
|
} KNOWN_INTERFACE;
|
|
typedef const KNOWN_INTERFACE *PCKNOWN_INTERFACE;
|
|
|
|
#undef ID_NAME
|
|
#define ID_NAME(c) &c, #c, L ## #c
|
|
static KNOWN_INTERFACE KnownInterfaces[] =
|
|
{
|
|
{ ID_NAME(IID_IACList), TRUE },
|
|
{ ID_NAME(IID_IACList2), TRUE },
|
|
{ ID_NAME(IID_IADesktopP2), TRUE },
|
|
{ ID_NAME(IID_IAccessible) },
|
|
{ ID_NAME(IID_IAccessControl), TRUE },
|
|
{ ID_NAME(IID_IAccessor) },
|
|
{ ID_NAME(IID_IActiveDesktop), TRUE },
|
|
{ ID_NAME(IID_IActiveDesktopP), TRUE },
|
|
{ ID_NAME(IID_IAddressBarParser), TRUE },
|
|
{ ID_NAME(IID_IAddressBand), TRUE },
|
|
{ ID_NAME(IID_IAddressEditBox), TRUE },
|
|
{ ID_NAME(IID_IAugmentedShellFolder), TRUE },
|
|
{ ID_NAME(IID_IAugmentedShellFolder2), TRUE },
|
|
{ ID_NAME(IID_IAutoComplete), TRUE },
|
|
{ ID_NAME(IID_IAutoComplete2), TRUE },
|
|
{ ID_NAME(IID_IBandNavigate), TRUE },
|
|
{ ID_NAME(IID_IBandProxy), TRUE },
|
|
{ ID_NAME(IID_IBandSite) },
|
|
{ ID_NAME(IID_IBandSiteHelper), TRUE },
|
|
{ ID_NAME(IID_IBanneredBar), TRUE },
|
|
{ ID_NAME(IID_IBindCtx) },
|
|
{ ID_NAME(IID_IBrowserFrameOptions), TRUE },
|
|
{ ID_NAME(IID_IBrowserService) },
|
|
{ ID_NAME(IID_IBrowserService2), TRUE },
|
|
{ ID_NAME(IID_IBrowserService3), TRUE },
|
|
{ ID_NAME(IID_IClassFactory) },
|
|
{ ID_NAME(IID_IClassFactory2) },
|
|
{ ID_NAME(IID_IClassFactory3), TRUE },
|
|
{ ID_NAME(IID_IClientSecurity), TRUE },
|
|
{ ID_NAME(IID_IComThreadingInfo), TRUE },
|
|
{ ID_NAME(IID_IContext), TRUE },
|
|
{ ID_NAME(IID_IContextMenu), TRUE },
|
|
{ ID_NAME(IID_IContextMenu2), TRUE },
|
|
{ ID_NAME(IID_IContextMenu3), TRUE },
|
|
{ ID_NAME(IID_IContextMenuCB), TRUE },
|
|
{ ID_NAME(IID_IContextMenuSite) },
|
|
{ ID_NAME(IID_IContinue) },
|
|
{ ID_NAME(IID_IDVGetEnum), TRUE },
|
|
{ ID_NAME(IID_IDataObject) },
|
|
//{ ID_NAME(IID_IDefViewID) }, == DefViewFrame3
|
|
{ ID_NAME(IID_IDefViewFrame), TRUE },
|
|
{ ID_NAME(IID_IDefViewFrame3) },
|
|
{ ID_NAME(IID_IDefViewFrameGroup) },
|
|
{ ID_NAME(IID_IDefViewSafety), },
|
|
{ ID_NAME(IID_IDefViewScript), TRUE },
|
|
{ ID_NAME(IID_IDeskBand) },
|
|
{ ID_NAME(IID_IDeskBandEx) },
|
|
{ ID_NAME(IID_IDeskBar), TRUE },
|
|
{ ID_NAME(IID_IDeskBarClient), TRUE },
|
|
{ ID_NAME(IID_IDeskMovr), TRUE },
|
|
{ ID_NAME(IID_IDispatch) },
|
|
{ ID_NAME(IID_IDockingWindow) },
|
|
{ ID_NAME(IID_IDockingWindowFrame), TRUE },
|
|
{ ID_NAME(IID_IDockingWindowSite), TRUE },
|
|
{ ID_NAME(IID_IDocViewSite), TRUE },
|
|
{ ID_NAME(IID_IDragSourceHelper), TRUE },
|
|
{ ID_NAME(IID_IDropSource) },
|
|
{ ID_NAME(IID_IDropTarget) },
|
|
{ ID_NAME(IID_IDropTargetHelper), TRUE },
|
|
{ ID_NAME(IID_IEnumExtraSearch) },
|
|
{ ID_NAME(IID_IEnumGUID) },
|
|
{ ID_NAME(IID_IEnumIDList) },
|
|
{ ID_NAME(IID_IEnumShellItems) },
|
|
{ ID_NAME(IID_IEnumString) },
|
|
{ ID_NAME(IID_IEnumUnknown) },
|
|
{ ID_NAME(IID_IEnumVARIANT) },
|
|
{ ID_NAME(IID_IErrorLog) },
|
|
{ ID_NAME(IID_IExplorerToolbar), TRUE },
|
|
{ ID_NAME(IID_IExtractIconA), TRUE },
|
|
{ ID_NAME(IID_IExtractIconW), TRUE },
|
|
{ ID_NAME(IID_IExtractImage) },
|
|
{ ID_NAME(IID_IExtractImage2) },
|
|
{ ID_NAME(IID_IFileDialog) },
|
|
{ ID_NAME(IID_IFileDialog2), TRUE },
|
|
{ ID_NAME(IID_IFileSearchBand) },
|
|
{ ID_NAME(IID_IFolderBandPriv) },
|
|
{ ID_NAME(IID_IFolderFilter) },
|
|
{ ID_NAME(IID_IFolderFilterSite) },
|
|
{ ID_NAME(IID_IFolderView) },
|
|
{ ID_NAME(IID_IFolderView2) },
|
|
{ ID_NAME(IID_IFolderViewOC) },
|
|
{ ID_NAME(IID_IFolderViewSettings) },
|
|
{ ID_NAME(IID_IGlobalFolderSettings), TRUE },
|
|
{ ID_NAME(IID_IInitializeObject), TRUE },
|
|
{ ID_NAME(IID_IInputObject) },
|
|
{ ID_NAME(IID_IInputObjectSite) },
|
|
{ ID_NAME(IID_IInternalUnknown), TRUE },
|
|
{ ID_NAME(IID_IMarshal) },
|
|
{ ID_NAME(IID_IMenuBand), TRUE },
|
|
{ ID_NAME(IID_IMenuPopup), TRUE },
|
|
{ ID_NAME(IID_IMoniker) },
|
|
{ ID_NAME(IID_IMultiMonitorDockingSite), TRUE },
|
|
{ ID_NAME(IID_IMultiQI), TRUE },
|
|
{ ID_NAME(IID_INamespaceProxy), TRUE },
|
|
{ ID_NAME(IID_INameSpaceTreeControl), TRUE },
|
|
{ ID_NAME(IID_INSCTree), TRUE },
|
|
{ ID_NAME(IID_INSCTree2), TRUE },
|
|
{ ID_NAME(IID_IObjMgr), TRUE },
|
|
{ ID_NAME(IID_IObjectSafety) },
|
|
{ ID_NAME(IID_IObjectWithSite) },
|
|
{ ID_NAME(IID_IOleCommandTarget) },
|
|
{ ID_NAME(IID_IOleInPlaceActiveObject) },
|
|
{ ID_NAME(IID_IOleInPlaceFrame) },
|
|
{ ID_NAME(IID_IOleInPlaceObject) },
|
|
{ ID_NAME(IID_IOleInPlaceObjectWindowless), TRUE },
|
|
{ ID_NAME(IID_IOleInPlaceSite) },
|
|
{ ID_NAME(IID_IOleInPlaceSiteEx) },
|
|
{ ID_NAME(IID_IOleInPlaceSiteWindowless), TRUE },
|
|
{ ID_NAME(IID_IOleInPlaceUIWindow) },
|
|
{ ID_NAME(IID_IOleObject) },
|
|
{ ID_NAME(IID_IOleWindow) },
|
|
{ ID_NAME(IID_IPersist) },
|
|
{ ID_NAME(IID_IPersistFile) },
|
|
{ ID_NAME(IID_IPersistFolder) },
|
|
{ ID_NAME(IID_IPersistFolder2) },
|
|
{ ID_NAME(IID_IPersistFolder3) },
|
|
{ ID_NAME(IID_IPersistHistory) },
|
|
{ ID_NAME(IID_IPersistIDList) },
|
|
{ ID_NAME(IID_IPersistMemory) },
|
|
{ ID_NAME(IID_IPersistPropertyBag) },
|
|
{ ID_NAME(IID_IPersistPropertyBag2) },
|
|
{ ID_NAME(IID_IPersistStorage) },
|
|
{ ID_NAME(IID_IPersistStream) },
|
|
{ ID_NAME(IID_IPersistStreamInit) },
|
|
{ ID_NAME(IID_IProgressDialog), TRUE },
|
|
{ ID_NAME(IID_IPropertyBag) },
|
|
{ ID_NAME(IID_IPropertyBag2) },
|
|
{ ID_NAME(IID_IQueryAssociations), TRUE },
|
|
{ ID_NAME(IID_IQueryInfo), TRUE },
|
|
{ ID_NAME(IID_IRegTreeOptions), TRUE },
|
|
{ ID_NAME(IID_IRunnableObject) },
|
|
{ ID_NAME(IID_IServerSecurity), TRUE },
|
|
{ ID_NAME(IID_IServiceProvider) },
|
|
{ ID_NAME(IID_ISFHelper), TRUE },
|
|
{ ID_NAME(IID_IShellBrowser) },
|
|
{ ID_NAME(IID_IShellBrowserService), TRUE },
|
|
{ ID_NAME(IID_IShellChangeNotify), TRUE },
|
|
{ ID_NAME(IID_IShellDesktopTray), TRUE },
|
|
{ ID_NAME(IID_IShellDispatch) },
|
|
{ ID_NAME(IID_IShellDispatch2) },
|
|
{ ID_NAME(IID_IShellDispatch3) },
|
|
{ ID_NAME(IID_IShellDispatch4) },
|
|
{ ID_NAME(IID_IShellDispatch5), TRUE },
|
|
{ ID_NAME(IID_IShellExtInit), TRUE },
|
|
{ ID_NAME(IID_IShellPropSheetExt), TRUE },
|
|
{ ID_NAME(IID_IShellIconOverlayIdentifier), TRUE },
|
|
{ ID_NAME(IID_IShellFolder) },
|
|
{ ID_NAME(IID_IShellFolder2) },
|
|
{ ID_NAME(IID_IShellFolderBand), TRUE },
|
|
{ ID_NAME(IID_IShellFolderView), TRUE },
|
|
{ ID_NAME(IID_IShellFolderViewCB), TRUE },
|
|
{ ID_NAME(IID_IShellFolderViewDual) },
|
|
{ ID_NAME(IID_IShellFolderViewDual2) },
|
|
{ ID_NAME(IID_IShellIcon) },
|
|
{ ID_NAME(IID_IShellItem) },
|
|
{ ID_NAME(IID_IShellItem2) },
|
|
{ ID_NAME(IID_IShellItemArray) },
|
|
{ ID_NAME(IID_IShellItemFilter) },
|
|
{ ID_NAME(IID_IShellLinkA) },
|
|
{ ID_NAME(IID_IShellLinkDual) },
|
|
{ ID_NAME(IID_IShellLinkDual2) },
|
|
{ ID_NAME(IID_IShellLinkW) },
|
|
{ ID_NAME(IID_IShellMenu), TRUE },
|
|
{ ID_NAME(IID_IShellMenu2), TRUE },
|
|
{ ID_NAME(IID_IShellMenuAcc), TRUE },
|
|
{ ID_NAME(IID_IShellMenuCallback), TRUE },
|
|
{ ID_NAME(IID_IShellNameSpace) },
|
|
{ ID_NAME(IID_IShellService), TRUE },
|
|
{ ID_NAME(IID_IShellView) },
|
|
{ ID_NAME(IID_IShellView2) },
|
|
{ ID_NAME(IID_IShellView3) },
|
|
{ ID_NAME(IID_IShellWindows) },
|
|
{ ID_NAME(IID_IStorage) },
|
|
{ ID_NAME(IID_IStream) },
|
|
{ ID_NAME(IID_ISurrogate) },
|
|
{ ID_NAME(IID_ISynchronize) },
|
|
{ ID_NAME(IID_ISynchronizeContainer), TRUE },
|
|
{ ID_NAME(IID_ISynchronizeEvent), TRUE },
|
|
{ ID_NAME(IID_ISynchronizeHandle), TRUE },
|
|
{ ID_NAME(IID_ITaskbarList) },
|
|
{ ID_NAME(IID_ITaskbarList2) },
|
|
{ ID_NAME(IID_ITrackShellMenu), TRUE },
|
|
{ ID_NAME(IID_ITrayPriv), TRUE },
|
|
{ ID_NAME(IID_ITrayPriv2), TRUE },
|
|
{ ID_NAME(IID_IUnknown) },
|
|
{ ID_NAME(IID_IViewObject) },
|
|
{ ID_NAME(IID_IViewObject2) },
|
|
{ ID_NAME(IID_IViewObjectEx), TRUE },
|
|
{ ID_NAME(IID_IWinEventHandler), TRUE },
|
|
|
|
{ ID_NAME(IID_DFConstraint), TRUE },
|
|
{ ID_NAME(DIID_DShellFolderViewEvents) },
|
|
|
|
{ ID_NAME(IID_CDefView), TRUE },
|
|
{ ID_NAME(IID_Folder) },
|
|
{ ID_NAME(IID_Folder2) },
|
|
{ ID_NAME(IID_Folder3) },
|
|
{ ID_NAME(IID_FolderItem) },
|
|
{ ID_NAME(IID_FolderItem2) },
|
|
{ ID_NAME(IID_FolderItems) },
|
|
{ ID_NAME(IID_FolderItems2) },
|
|
{ ID_NAME(IID_FolderItems3) },
|
|
{ ID_NAME(IID_FolderItemVerb) },
|
|
{ ID_NAME(IID_FolderItemVerbs) },
|
|
|
|
{ ID_NAME(CLSID_ShellDesktop), TRUE }
|
|
};
|
|
static const INT KnownInterfaceCount = RTL_NUMBER_OF(KnownInterfaces);
|
|
|
|
static
|
|
PCKNOWN_INTERFACE
|
|
FindInterface(
|
|
_In_ const IID *iid)
|
|
{
|
|
INT i;
|
|
|
|
for (i = 0; i < KnownInterfaceCount; i++)
|
|
if (IsEqualIID(KnownInterfaces[i].iid, iid))
|
|
return &KnownInterfaces[i];
|
|
ASSERT(i != KnownInterfaceCount);
|
|
return NULL;
|
|
}
|
|
|
|
static
|
|
BOOLEAN
|
|
IsInterfaceExpected(
|
|
_In_ PCCLASS_AND_INTERFACES class,
|
|
_In_ const IID *iid)
|
|
{
|
|
INT i;
|
|
|
|
for (i = 0; class->ifaces[i].iid; i++)
|
|
if (IsEqualIID(class->ifaces[i].iid, iid))
|
|
return TRUE;
|
|
return FALSE;
|
|
}
|
|
|
|
#define INTF_NOT_EXPOSED LONG_MAX
|
|
static
|
|
LONG
|
|
GetInterfaceOffset(
|
|
_In_ PUNKNOWN pUnk,
|
|
_In_ const IID *iid)
|
|
{
|
|
HRESULT hr;
|
|
PVOID pObj;
|
|
PUNKNOWN pUnk2;
|
|
LONG offset;
|
|
|
|
hr = IUnknown_QueryInterface(pUnk, iid, &pObj);
|
|
ok(hr == S_OK || hr == E_NOINTERFACE, "IUnknown::QueryInterface returned 0x%lx\n", hr);
|
|
if (FAILED(hr))
|
|
return INTF_NOT_EXPOSED;
|
|
|
|
pUnk2 = pObj;
|
|
offset = (LONG_PTR)pObj - (LONG_PTR)pUnk;
|
|
IUnknown_Release(pUnk2);
|
|
return offset;
|
|
}
|
|
|
|
static
|
|
VOID
|
|
TestModuleInterfaces(
|
|
_In_ PCCLASS_AND_INTERFACES ExpectedInterfaces,
|
|
_In_ INT ExpectedInterfaceCount)
|
|
{
|
|
HRESULT hr;
|
|
PVOID pObj;
|
|
PUNKNOWN pUnk;
|
|
INT iClass, iIntf;
|
|
PCCLASS_AND_INTERFACES class;
|
|
|
|
for (iClass = 0; iClass < ExpectedInterfaceCount; iClass++)
|
|
{
|
|
class = &ExpectedInterfaces[iClass];
|
|
hr = CoCreateInstance(class->clsid,
|
|
NULL,
|
|
CLSCTX_INPROC_SERVER,
|
|
&IID_IUnknown,
|
|
&pObj);
|
|
ok(hr == S_OK, "CoCreateInstance failed. hr=0x%lx\n", hr);
|
|
if (FAILED(hr))
|
|
{
|
|
skip("Failed to instantiate %s.\n", class->name);
|
|
continue;
|
|
}
|
|
|
|
pUnk = pObj;
|
|
|
|
/* Check that all expected interfaces are present and have the right offset */
|
|
for (iIntf = 0; class->ifaces[iIntf].iid; iIntf++)
|
|
{
|
|
PCKNOWN_INTERFACE iface = FindInterface(class->ifaces[iIntf].iid);
|
|
LONG offset = GetInterfaceOffset(pUnk, iface->iid);
|
|
if (offset == INTF_NOT_EXPOSED)
|
|
ok(0, "%s is missing %s (offset %ld)\n", class->name, iface->name, class->ifaces[iIntf].offset);
|
|
else if (class->ifaces[iIntf].offset != FARAWY)
|
|
{
|
|
#ifdef FAIL_WRONG_OFFSET
|
|
ok(offset == class->ifaces[iIntf].offset, "%s, %s offset is %ld, expected %ld\n", class->name, iface->name, offset, class->ifaces[iIntf].offset);
|
|
#else
|
|
if (offset != class->ifaces[iIntf].offset)
|
|
mytrace("%s, %s offset is %ld, expected %ld\n", class->name, iface->name, offset, class->ifaces[iIntf].offset);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
/* Check that none other than the expected interfaces are present */
|
|
for (iIntf = 0; iIntf < KnownInterfaceCount; iIntf++)
|
|
{
|
|
PCKNOWN_INTERFACE iface = &KnownInterfaces[iIntf];
|
|
LONG offset;
|
|
if (IsInterfaceExpected(class, iface->iid))
|
|
continue;
|
|
offset = GetInterfaceOffset(pUnk, iface->iid);
|
|
#ifdef GENERATE_TABLE_ENTRIES
|
|
ok(offset == INTF_NOT_EXPOSED, "%s: { %s0x%x, &%s },\n", class->name, offset < 0 ? "-" : "", offset < 0 ? -offset : offset, iface->name);
|
|
#else
|
|
ok(offset == INTF_NOT_EXPOSED, "%s exposes %s (offset %ld), but shouldn't\n", class->name, iface->name, offset);
|
|
#endif
|
|
}
|
|
|
|
// TODO: do some aggregation
|
|
|
|
IUnknown_Release(pUnk);
|
|
}
|
|
}
|
|
|
|
static
|
|
VOID
|
|
TestModuleRegistry(
|
|
_In_ PCWSTR ModuleName,
|
|
_In_ PCCLASS_AND_INTERFACES ExpectedInterfaces,
|
|
_In_ INT ExpectedInterfaceCount)
|
|
{
|
|
INT iClass;
|
|
PCCLASS_AND_INTERFACES class;
|
|
HKEY hKeyClasses;
|
|
LONG result;
|
|
|
|
result = RegOpenKeyEx(HKEY_LOCAL_MACHINE, L"Software\\Classes\\CLSID", 0, KEY_ENUMERATE_SUB_KEYS, &hKeyClasses);
|
|
ok(result == NO_ERROR, "Failed to open classes key, error %lu\n", result);
|
|
if (!myskip(result == NO_ERROR, "No classes key\n"))
|
|
{
|
|
for (iClass = 0; iClass < ExpectedInterfaceCount; iClass++)
|
|
{
|
|
HKEY hKey;
|
|
HKEY hKeyServer;
|
|
NTSTATUS status;
|
|
UNICODE_STRING clsid;
|
|
DWORD type;
|
|
WCHAR data[100];
|
|
DWORD dataSize;
|
|
PCWSTR expectedThreadingModel;
|
|
|
|
class = &ExpectedInterfaces[iClass];
|
|
status = RtlStringFromGUID(class->clsid, &clsid);
|
|
ok(status == STATUS_SUCCESS, "Failed to convert guid to string for %s, status %lx\n", class->name, status);
|
|
if (myskip(NT_SUCCESS(status), "No guid string\n"))
|
|
continue;
|
|
|
|
result = RegOpenKeyEx(hKeyClasses, clsid.Buffer, 0, KEY_ENUMERATE_SUB_KEYS, &hKey);
|
|
ok(result == NO_ERROR, "Failed to open key for %s, error %lu\n", class->name, result);
|
|
RtlFreeUnicodeString(&clsid);
|
|
if (myskip(result == NO_ERROR, "No key\n"))
|
|
continue;
|
|
|
|
result = RegOpenKeyEx(hKey, L"InProcServer32", 0, KEY_QUERY_VALUE, &hKeyServer);
|
|
ok(result == NO_ERROR, "Failed to open key for %s, error %lu\n", class->name, result);
|
|
RegCloseKey(hKey);
|
|
if (myskip(result == NO_ERROR, "No key\n"))
|
|
continue;
|
|
|
|
dataSize = sizeof(data);
|
|
result = RegQueryValueEx(hKeyServer, NULL, NULL, &type, (PBYTE)data, &dataSize);
|
|
ok(result == NO_ERROR, "Failed to query value for %s, error %lu\n", class->name, result);
|
|
if (!myskip(result == NO_ERROR, "No module name\n"))
|
|
{
|
|
ok(type == REG_SZ || type == REG_EXPAND_SZ, "type %lu for %s\n", type, class->name);
|
|
ok(dataSize % sizeof(WCHAR) == 0, "size %lu for %s\n", dataSize, class->name);
|
|
ok(dataSize <= sizeof(data), "size %lu for %s\n", dataSize, class->name);
|
|
ok(data[dataSize / sizeof(WCHAR) - 1] == UNICODE_NULL, "Not null terminated for %s\n", class->name);
|
|
// TODO: Use SearchPath (or assume everything's in system32) and do a proper full path compare
|
|
PathStripPathW(data);
|
|
PathRemoveExtensionW(data);
|
|
ok(!wcsicmp(data, ModuleName), "Server is %ls, expected %ls for %s\n", data, ModuleName, class->name);
|
|
}
|
|
|
|
dataSize = sizeof(data);
|
|
result = RegQueryValueEx(hKeyServer, L"ThreadingModel", NULL, &type, (PBYTE)data, &dataSize);
|
|
ok(result == NO_ERROR, "Failed to query value for %s, error %lu\n", class->name, result);
|
|
if (!myskip(result == NO_ERROR, "No ThreadingModel\n"))
|
|
{
|
|
ok(type == REG_SZ || type == REG_EXPAND_SZ, "type %lu for %s\n", type, class->name);
|
|
ok(dataSize % sizeof(WCHAR) == 0, "size %lu for %s\n", dataSize, class->name);
|
|
ok(dataSize <= sizeof(data), "size %lu for %s\n", dataSize, class->name);
|
|
ok(data[dataSize / sizeof(WCHAR) - 1] == UNICODE_NULL, "Not null terminated for %s\n", class->name);
|
|
expectedThreadingModel = class->ThreadingModel;
|
|
if (!expectedThreadingModel)
|
|
expectedThreadingModel = L"Apartment";
|
|
ok(!wcsicmp(data, expectedThreadingModel), "Server is %ls, expected %ls for %s\n", data, expectedThreadingModel, class->name);
|
|
}
|
|
|
|
RegCloseKey(hKeyServer);
|
|
}
|
|
RegCloseKey(hKeyClasses);
|
|
}
|
|
}
|
|
|
|
static
|
|
VOID
|
|
TestManualInstantiation(
|
|
_In_ PCWSTR ModuleName,
|
|
_In_ PCCLASS_AND_INTERFACES ExpectedInterfaces,
|
|
_In_ INT ExpectedInterfaceCount)
|
|
{
|
|
INT iClass;
|
|
PCCLASS_AND_INTERFACES class;
|
|
HRESULT (__stdcall *DllGetClassObject)(REFCLSID, REFIID, PVOID *);
|
|
|
|
DllGetClassObject = (PVOID)GetProcAddress(GetModuleHandle(ModuleName), "DllGetClassObject");
|
|
ok(DllGetClassObject != NULL, "DllGetClassObject not found in %ls, error %lu\n", ModuleName, GetLastError());
|
|
if (myskip(DllGetClassObject != NULL, "No DllGetClassObject\n"))
|
|
return;
|
|
|
|
for (iClass = 0; iClass < ExpectedInterfaceCount; iClass++)
|
|
{
|
|
PVOID pv;
|
|
HRESULT hr;
|
|
class = &ExpectedInterfaces[iClass];
|
|
hr = DllGetClassObject(class->clsid, &IID_IClassFactory, &pv);
|
|
ok(hr == S_OK, "DllGetClassObject failed for %s, hr = 0x%lx\n", class->name, hr);
|
|
if (!myskip(SUCCEEDED(hr), "No class factory\n"))
|
|
{
|
|
IClassFactory *pCF = pv;
|
|
hr = IClassFactory_CreateInstance(pCF, NULL, &IID_IUnknown, &pv);
|
|
ok(hr == S_OK, "IClassFactory::CreateInstance failed for %s, hr = 0x%lx\n", class->name, hr);
|
|
if (!myskip(SUCCEEDED(hr), "No instance\n"))
|
|
{
|
|
IUnknown *pUnk = pv;
|
|
IUnknown_Release(pUnk);
|
|
}
|
|
IClassFactory_Release(pCF);
|
|
}
|
|
}
|
|
}
|
|
|
|
VOID
|
|
TestClasses(
|
|
_In_ PCWSTR ModuleName,
|
|
_In_ PCCLASS_AND_INTERFACES ExpectedInterfaces,
|
|
_In_ INT ExpectedInterfaceCount)
|
|
{
|
|
HRESULT hr;
|
|
|
|
hr = CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
|
|
ok(hr == S_OK, "CoInitializeEx failed. hr=0x%lx\n", hr);
|
|
if (myskip(SUCCEEDED(hr), "Failed to initialize COM. Cannot perform tests\n"))
|
|
return;
|
|
|
|
TestModuleInterfaces(ExpectedInterfaces, ExpectedInterfaceCount);
|
|
TestModuleRegistry(ModuleName, ExpectedInterfaces, ExpectedInterfaceCount);
|
|
TestManualInstantiation(ModuleName, ExpectedInterfaces, ExpectedInterfaceCount);
|
|
|
|
CoUninitialize();
|
|
}
|
|
|
|
static
|
|
VOID
|
|
TestInterfaceRegistry(
|
|
_In_ PCKNOWN_INTERFACE Interfaces,
|
|
_In_ INT InterfaceCount)
|
|
{
|
|
INT i;
|
|
HKEY hKeyInterface;
|
|
LONG result;
|
|
|
|
result = RegOpenKeyEx(HKEY_LOCAL_MACHINE, L"Software\\Classes\\Interface", 0, KEY_ENUMERATE_SUB_KEYS, &hKeyInterface);
|
|
ok(result == NO_ERROR, "Failed to open interface key, error %lu\n", result);
|
|
if (!myskip(result == NO_ERROR, "No interface key\n"))
|
|
{
|
|
for (i = 0; i < InterfaceCount; i++)
|
|
{
|
|
HKEY hKey;
|
|
NTSTATUS status;
|
|
UNICODE_STRING iid;
|
|
DWORD type;
|
|
WCHAR data[100];
|
|
DWORD dataSize;
|
|
PCKNOWN_INTERFACE iface;
|
|
PCWSTR expectedName;
|
|
|
|
iface = &Interfaces[i];
|
|
status = RtlStringFromGUID(iface->iid, &iid);
|
|
ok(status == STATUS_SUCCESS, "Failed to convert guid to string for %s, status %lx\n", iface->name, status);
|
|
if (myskip(NT_SUCCESS(status), "No guid string\n"))
|
|
continue;
|
|
|
|
result = RegOpenKeyEx(hKeyInterface, iid.Buffer, 0, KEY_QUERY_VALUE, &hKey);
|
|
if (iface->noreg)
|
|
{
|
|
ok(result == ERROR_FILE_NOT_FOUND, "RegOpenKeyEx returned %lu for %s\n", result, iface->name);
|
|
}
|
|
else
|
|
{
|
|
ok(result == NO_ERROR, "Failed to open key for %s, error %lu\n", iface->name, result);
|
|
(void)myskip(result == NO_ERROR, "No key\n");
|
|
}
|
|
RtlFreeUnicodeString(&iid);
|
|
if (result != NO_ERROR)
|
|
continue;
|
|
|
|
dataSize = sizeof(data);
|
|
result = RegQueryValueEx(hKey, NULL, NULL, &type, (PBYTE)data, &dataSize);
|
|
ok(result == NO_ERROR, "Failed to query value for %s, error %lu\n", iface->name, result);
|
|
if (!myskip(result == NO_ERROR, "No module name\n"))
|
|
{
|
|
ok(type == REG_SZ, "type %lu for %s\n", type, iface->name);
|
|
ok(dataSize % sizeof(WCHAR) == 0, "size %lu for %s\n", dataSize, iface->name);
|
|
ok(dataSize <= sizeof(data), "size %lu for %s\n", dataSize, iface->name);
|
|
ok(data[dataSize / sizeof(WCHAR) - 1] == UNICODE_NULL, "Not null terminated for %s\n", iface->name);
|
|
expectedName = wcschr(iface->wname, L'_');
|
|
if (expectedName)
|
|
expectedName++;
|
|
else
|
|
expectedName = iface->wname;
|
|
ok(!wcsicmp(data, expectedName), "Name is %ls, expected %ls\n", data, expectedName);
|
|
}
|
|
|
|
RegCloseKey(hKey);
|
|
}
|
|
RegCloseKey(hKeyInterface);
|
|
}
|
|
}
|
|
|
|
START_TEST(interfaces)
|
|
{
|
|
TestInterfaceRegistry(KnownInterfaces, KnownInterfaceCount);
|
|
}
|