272 lines
9.3 KiB
C
272 lines
9.3 KiB
C
/*
|
|
* OLE DB Marshaling Tests
|
|
*
|
|
* Copyright 2009 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
|
|
#define CONST_VTABLE
|
|
|
|
#include <stdarg.h>
|
|
#include <stdio.h>
|
|
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "objbase.h"
|
|
#include "oledb.h"
|
|
|
|
#include "wine/test.h"
|
|
|
|
#define RELEASEMARSHALDATA WM_USER
|
|
|
|
#define ok_ole_success(hr, func) ok(hr == S_OK, #func " failed with error 0x%08x\n", hr)
|
|
|
|
struct host_object_data
|
|
{
|
|
IStream *stream;
|
|
IID iid;
|
|
IUnknown *object;
|
|
MSHLFLAGS marshal_flags;
|
|
HANDLE marshal_event;
|
|
IMessageFilter *filter;
|
|
};
|
|
|
|
static DWORD CALLBACK host_object_proc(LPVOID p)
|
|
{
|
|
struct host_object_data *data = p;
|
|
HRESULT hr;
|
|
MSG msg;
|
|
|
|
CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
|
|
|
|
if (data->filter)
|
|
{
|
|
IMessageFilter * prev_filter = NULL;
|
|
hr = CoRegisterMessageFilter(data->filter, &prev_filter);
|
|
if (prev_filter) IMessageFilter_Release(prev_filter);
|
|
ok_ole_success(hr, CoRegisterMessageFilter);
|
|
}
|
|
|
|
hr = CoMarshalInterface(data->stream, &data->iid, data->object, MSHCTX_INPROC, NULL, data->marshal_flags);
|
|
ok_ole_success(hr, CoMarshalInterface);
|
|
|
|
/* force the message queue to be created before signaling parent thread */
|
|
PeekMessage(&msg, NULL, WM_USER, WM_USER, PM_NOREMOVE);
|
|
|
|
SetEvent(data->marshal_event);
|
|
|
|
while (GetMessage(&msg, NULL, 0, 0))
|
|
{
|
|
if (msg.hwnd == NULL && msg.message == RELEASEMARSHALDATA)
|
|
{
|
|
CoReleaseMarshalData(data->stream);
|
|
SetEvent((HANDLE)msg.lParam);
|
|
}
|
|
else
|
|
DispatchMessage(&msg);
|
|
}
|
|
|
|
HeapFree(GetProcessHeap(), 0, data);
|
|
|
|
CoUninitialize();
|
|
|
|
return hr;
|
|
}
|
|
|
|
static DWORD start_host_object2(IStream *stream, REFIID riid, IUnknown *object, MSHLFLAGS marshal_flags, IMessageFilter *filter, HANDLE *thread)
|
|
{
|
|
DWORD tid = 0;
|
|
HANDLE marshal_event = CreateEvent(NULL, FALSE, FALSE, NULL);
|
|
struct host_object_data *data = HeapAlloc(GetProcessHeap(), 0, sizeof(*data));
|
|
|
|
data->stream = stream;
|
|
data->iid = *riid;
|
|
data->object = object;
|
|
data->marshal_flags = marshal_flags;
|
|
data->marshal_event = marshal_event;
|
|
data->filter = filter;
|
|
|
|
*thread = CreateThread(NULL, 0, host_object_proc, data, 0, &tid);
|
|
|
|
/* wait for marshaling to complete before returning */
|
|
ok( !WaitForSingleObject(marshal_event, 10000), "wait timed out\n" );
|
|
CloseHandle(marshal_event);
|
|
|
|
return tid;
|
|
}
|
|
|
|
static DWORD start_host_object(IStream *stream, REFIID riid, IUnknown *object, MSHLFLAGS marshal_flags, HANDLE *thread)
|
|
{
|
|
return start_host_object2(stream, riid, object, marshal_flags, NULL, thread);
|
|
}
|
|
|
|
static void end_host_object(DWORD tid, HANDLE thread)
|
|
{
|
|
BOOL ret = PostThreadMessage(tid, WM_QUIT, 0, 0);
|
|
ok(ret, "PostThreadMessage failed with error %d\n", GetLastError());
|
|
/* be careful of races - don't return until hosting thread has terminated */
|
|
ok( !WaitForSingleObject(thread, 10000), "wait timed out\n" );
|
|
CloseHandle(thread);
|
|
}
|
|
|
|
#define TEST_PROPID 0xdead
|
|
static const WCHAR wszDBPropertyTestString[] = {'D','B','P','r','o','p','e','r','t','y','T','e','s','t','S','t','r','i','n','g',0};
|
|
static const WCHAR wszDBPropertyColumnName[] = {'C','o','l','u','m','n',0};
|
|
|
|
static HRESULT WINAPI Test_DBProperties_QueryInterface(
|
|
IDBProperties* iface,
|
|
REFIID riid,
|
|
void **ppvObject)
|
|
{
|
|
if (IsEqualIID(riid, &IID_IUnknown) ||
|
|
IsEqualIID(riid, &IID_IDBProperties))
|
|
{
|
|
*ppvObject = iface;
|
|
return S_OK;
|
|
}
|
|
*ppvObject = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
static ULONG WINAPI Test_DBProperties_AddRef(
|
|
IDBProperties* iface)
|
|
{
|
|
return 2;
|
|
}
|
|
|
|
static ULONG WINAPI Test_DBProperties_Release(
|
|
IDBProperties* iface)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
/*** IDBProperties methods ***/
|
|
static HRESULT WINAPI Test_DBProperties_GetProperties(
|
|
IDBProperties* iface,
|
|
ULONG cPropertyIDSets,
|
|
const DBPROPIDSET rgPropertyIDSets[],
|
|
ULONG *pcPropertySets,
|
|
DBPROPSET **prgPropertySets)
|
|
{
|
|
ok(cPropertyIDSets == 0, "Expected cPropertyIDSets to be 0 instead of %d\n", cPropertyIDSets);
|
|
ok(*pcPropertySets == 0, "Expected *pcPropertySets to be 0 instead of %d\n", *pcPropertySets);
|
|
*pcPropertySets = 1;
|
|
*prgPropertySets = CoTaskMemAlloc(sizeof(DBPROPSET));
|
|
(*prgPropertySets)[0].rgProperties = CoTaskMemAlloc(sizeof(DBPROP));
|
|
(*prgPropertySets)[0].rgProperties[0].dwPropertyID = TEST_PROPID;
|
|
(*prgPropertySets)[0].rgProperties[0].dwOptions = DBPROPOPTIONS_REQUIRED;
|
|
(*prgPropertySets)[0].rgProperties[0].dwStatus = S_OK;
|
|
(*prgPropertySets)[0].rgProperties[0].colid.eKind = DBKIND_GUID_NAME;
|
|
/* colid contents */
|
|
(*prgPropertySets)[0].rgProperties[0].colid.uGuid.guid = IID_IDBProperties;
|
|
(*prgPropertySets)[0].rgProperties[0].colid.uName.pwszName = CoTaskMemAlloc(sizeof(wszDBPropertyColumnName));
|
|
memcpy((*prgPropertySets)[0].rgProperties[0].colid.uName.pwszName, wszDBPropertyColumnName, sizeof(wszDBPropertyColumnName));
|
|
/* vValue contents */
|
|
V_VT(&(*prgPropertySets)[0].rgProperties[0].vValue) = VT_BSTR;
|
|
V_BSTR(&(*prgPropertySets)[0].rgProperties[0].vValue) = SysAllocString(wszDBPropertyTestString);
|
|
(*prgPropertySets)[0].cProperties = 1;
|
|
(*prgPropertySets)[0].guidPropertySet = IID_IDBProperties;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI Test_DBProperties_GetPropertyInfo(
|
|
IDBProperties* iface,
|
|
ULONG cPropertyIDSets,
|
|
const DBPROPIDSET rgPropertyIDSets[],
|
|
ULONG *pcPropertyInfoSets,
|
|
DBPROPINFOSET **prgPropertyInfoSets,
|
|
OLECHAR **ppDescBuffer)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI Test_DBProperties_SetProperties(
|
|
IDBProperties* iface,
|
|
ULONG cPropertySets,
|
|
DBPROPSET rgPropertySets[])
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IDBPropertiesVtbl Test_DBProperties_Vtbl =
|
|
{
|
|
Test_DBProperties_QueryInterface,
|
|
Test_DBProperties_AddRef,
|
|
Test_DBProperties_Release,
|
|
Test_DBProperties_GetProperties,
|
|
Test_DBProperties_GetPropertyInfo,
|
|
Test_DBProperties_SetProperties,
|
|
};
|
|
|
|
static IDBProperties Test_DBProperties =
|
|
{
|
|
&Test_DBProperties_Vtbl
|
|
};
|
|
|
|
static void test_IDBProperties(void)
|
|
{
|
|
HRESULT hr;
|
|
IStream *pStream = NULL;
|
|
IDBProperties *pProxy = NULL;
|
|
DWORD tid;
|
|
HANDLE thread;
|
|
static const LARGE_INTEGER ullZero;
|
|
ULONG propset_count;
|
|
DBPROPSET *propsets = NULL;
|
|
|
|
hr = CreateStreamOnHGlobal(NULL, TRUE, &pStream);
|
|
ok_ole_success(hr, "CreateStreamOnHGlobal");
|
|
tid = start_host_object(pStream, &IID_IDBProperties, (IUnknown*)&Test_DBProperties, MSHLFLAGS_NORMAL, &thread);
|
|
|
|
IStream_Seek(pStream, ullZero, STREAM_SEEK_SET, NULL);
|
|
hr = CoUnmarshalInterface(pStream, &IID_IDBProperties, (void **)&pProxy);
|
|
ok_ole_success(hr, "CoUnmarshalInterface");
|
|
IStream_Release(pStream);
|
|
|
|
propset_count = 1;
|
|
hr = IDBProperties_GetProperties(pProxy, 0, NULL, &propset_count, &propsets);
|
|
ok(hr == S_OK, "IDBProperties_GetProperties failed with error 0x%08x\n", hr);
|
|
|
|
ok(propset_count == 1, "Expected propset_count of 1 but got %d\n", propset_count);
|
|
ok(propsets->rgProperties[0].dwPropertyID == TEST_PROPID, "Expected property ID of 0x%x, but got 0x%x\n", TEST_PROPID, propsets->rgProperties[0].dwPropertyID);
|
|
ok(propsets->rgProperties[0].dwOptions == DBPROPOPTIONS_REQUIRED, "Expected property options of 0x%x, but got 0x%x\n", DBPROPOPTIONS_REQUIRED, propsets->rgProperties[0].dwOptions);
|
|
ok(propsets->rgProperties[0].dwStatus == S_OK, "Expected property options of 0x%x, but got 0x%x\n", S_OK, propsets->rgProperties[0].dwStatus);
|
|
ok(propsets->rgProperties[0].colid.eKind == DBKIND_GUID_NAME, "Expected property colid kind of DBKIND_GUID_NAME, but got %d\n", propsets->rgProperties[0].colid.eKind);
|
|
/* colid contents */
|
|
ok(IsEqualGUID(&propsets->rgProperties[0].colid.uGuid.guid, &IID_IDBProperties), "Unexpected property colid guid\n");
|
|
ok(!lstrcmpW(propsets->rgProperties[0].colid.uName.pwszName, wszDBPropertyColumnName), "Unexpected property colid name\n");
|
|
/* vValue contents */
|
|
ok(V_VT(&propsets->rgProperties[0].vValue) == VT_BSTR, "Expected property value vt of VT_BSTR, but got %d\n", V_VT(&propsets->rgProperties[0].vValue));
|
|
ok(!lstrcmpW(V_BSTR(&propsets->rgProperties[0].vValue), wszDBPropertyTestString), "Unexpected property value string\n");
|
|
ok(propsets->cProperties == 1, "Expected property count of 1 but got %d\n", propsets->cProperties);
|
|
ok(IsEqualGUID(&propsets->guidPropertySet, &IID_IDBProperties), "Unexpected guid for property set\n");
|
|
|
|
IDBProperties_Release(pProxy);
|
|
|
|
end_host_object(tid, thread);
|
|
}
|
|
|
|
START_TEST(marshal)
|
|
{
|
|
CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
|
|
|
|
test_IDBProperties();
|
|
CoUninitialize();
|
|
}
|