mirror of
https://github.com/reactos/reactos.git
synced 2024-11-09 08:08:38 +00:00
322 lines
11 KiB
C++
322 lines
11 KiB
C++
|
/*
|
||
|
* PROJECT: ReactOS api tests
|
||
|
* LICENSE: BSD - See COPYING.ARM in the top level directory
|
||
|
* PURPOSE: Tests for IInitializeSpy
|
||
|
* PROGRAMMERS: Mark Jansen
|
||
|
*/
|
||
|
|
||
|
#define WIN32_NO_STATUS
|
||
|
#define _INC_WINDOWS
|
||
|
#define COM_NO_WINDOWS_H
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include <wine/test.h>
|
||
|
|
||
|
#include <winuser.h>
|
||
|
#include <winreg.h>
|
||
|
|
||
|
#include <shlwapi.h>
|
||
|
#include <unknownbase.h>
|
||
|
|
||
|
#define test_S_OK(hres, message) ok((hres) == S_OK, "%s (0x%lx instead of S_OK)\n", (message), (hres))
|
||
|
#define test_HRES(hres, hresExpected, message) ok((hres) == (hresExpected), "%s (0x%lx instead of 0x%lx)\n", (message), (hres), (hresExpected))
|
||
|
#define test_ref(spy, expectedRef) ok((spy)->GetRef() == (expectedRef), "unexpected refcount, %ld instead of %d\n", (spy)->GetRef(), (expectedRef))
|
||
|
|
||
|
|
||
|
typedef HRESULT (WINAPI *pCoRegisterInitializeSpy_t)(_In_ LPINITIALIZESPY pSpy, _Out_ ULARGE_INTEGER *puliCookie);
|
||
|
typedef HRESULT (WINAPI *pCoRevokeInitializeSpy_t)(_In_ ULARGE_INTEGER uliCookie);
|
||
|
pCoRegisterInitializeSpy_t pCoRegisterInitializeSpy;
|
||
|
pCoRevokeInitializeSpy_t pCoRevokeInitializeSpy;
|
||
|
|
||
|
|
||
|
const DWORD INVALID_VALUE = 0xdeadbeef;
|
||
|
|
||
|
|
||
|
class CTestSpy : public CUnknownBase<IInitializeSpy>
|
||
|
{
|
||
|
public:
|
||
|
HRESULT hr;
|
||
|
ULARGE_INTEGER Cookie;
|
||
|
|
||
|
// expected values to check against
|
||
|
HRESULT m_hrCoInit;
|
||
|
DWORD m_CoInit;
|
||
|
DWORD m_CurAptRefs;
|
||
|
|
||
|
// keeping count of the times called
|
||
|
LONG m_PreInitCalled;
|
||
|
LONG m_PostInitCalled;
|
||
|
LONG m_PreUninitCalled;
|
||
|
LONG m_PostUninitCalled;
|
||
|
|
||
|
// fake out some
|
||
|
bool m_FailQueryInterface;
|
||
|
bool m_AlwaysReturnOK;
|
||
|
|
||
|
CTestSpy()
|
||
|
: CUnknownBase( false, 0 ),
|
||
|
hr(0),
|
||
|
m_hrCoInit(0),
|
||
|
m_CoInit(0),
|
||
|
m_CurAptRefs(0),
|
||
|
m_FailQueryInterface(false),
|
||
|
m_AlwaysReturnOK(false)
|
||
|
{
|
||
|
Cookie.HighPart = Cookie.LowPart = INVALID_VALUE;
|
||
|
Clear();
|
||
|
}
|
||
|
|
||
|
~CTestSpy()
|
||
|
{
|
||
|
// always try to revoke if we succeeded to register.
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
hr = pCoRevokeInitializeSpy(Cookie);
|
||
|
test_S_OK(hr, "CoRevokeInitializeSpy");
|
||
|
}
|
||
|
// we should be done.
|
||
|
ok(GetRef() == 0, "Expected m_lRef to be 0, was: %ld\n", GetRef());
|
||
|
}
|
||
|
|
||
|
HRESULT STDMETHODCALLTYPE QueryInterface(REFIID riid, void** ppv)
|
||
|
{
|
||
|
if (m_FailQueryInterface)
|
||
|
{
|
||
|
return E_NOINTERFACE;
|
||
|
}
|
||
|
return CUnknownBase::QueryInterface(riid, ppv);
|
||
|
}
|
||
|
|
||
|
const QITAB* GetQITab()
|
||
|
{
|
||
|
static const QITAB tab[] = { { &IID_IInitializeSpy, OFFSETOFCLASS(IInitializeSpy, CTestSpy) }, { 0 } };
|
||
|
return tab;
|
||
|
}
|
||
|
|
||
|
|
||
|
HRESULT STDMETHODCALLTYPE PreInitialize(DWORD dwCoInit, DWORD dwCurThreadAptRefs)
|
||
|
{
|
||
|
InterlockedIncrement(&m_PreInitCalled);
|
||
|
ok(m_CoInit == dwCoInit, "Unexpected dwCoInit: got %lx, expected %lx\n", dwCoInit, m_CoInit);
|
||
|
DWORD expectApt = m_hrCoInit == RPC_E_CHANGED_MODE ? m_CurAptRefs : m_CurAptRefs -1;
|
||
|
ok(expectApt == dwCurThreadAptRefs, "Unexpected dwCurThreadAptRefs: got %lx, expected %lx\n", dwCurThreadAptRefs, expectApt);
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
HRESULT STDMETHODCALLTYPE PostInitialize(HRESULT hrCoInit, DWORD dwCoInit, DWORD dwNewThreadAptRefs)
|
||
|
{
|
||
|
InterlockedIncrement(&m_PostInitCalled);
|
||
|
ok(m_PreInitCalled == m_PostInitCalled, "Expected balanced pre/post: %ld / %ld\n", m_PreInitCalled, m_PostInitCalled);
|
||
|
test_HRES(hrCoInit, m_hrCoInit, "Unexpected hrCoInit in PostInitialize");
|
||
|
ok(m_CoInit == dwCoInit, "Unexpected dwCoInit: got %lx, expected %lx\n", dwCoInit, m_CoInit);
|
||
|
ok(m_CurAptRefs == dwNewThreadAptRefs, "Unexpected dwNewThreadAptRefs: got %lx, expected %lx\n", dwNewThreadAptRefs, m_CurAptRefs);
|
||
|
if (m_AlwaysReturnOK)
|
||
|
return S_OK;
|
||
|
return hrCoInit;
|
||
|
}
|
||
|
|
||
|
HRESULT STDMETHODCALLTYPE PreUninitialize(DWORD dwCurThreadAptRefs)
|
||
|
{
|
||
|
InterlockedIncrement(&m_PreUninitCalled);
|
||
|
ok(m_CurAptRefs == dwCurThreadAptRefs, "Unexpected dwCurThreadAptRefs: got %lx, expected %lx\n", dwCurThreadAptRefs, m_CurAptRefs);
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
HRESULT STDMETHODCALLTYPE PostUninitialize(DWORD dwNewThreadAptRefs)
|
||
|
{
|
||
|
InterlockedIncrement(&m_PostUninitCalled);
|
||
|
ok(m_PreUninitCalled == m_PostUninitCalled, "Expected balanced pre/post: %ld / %ld\n", m_PreUninitCalled, m_PostUninitCalled);
|
||
|
DWORD apt = m_CurAptRefs ? (m_CurAptRefs-1) : 0;
|
||
|
ok(apt == dwNewThreadAptRefs, "Unexpected dwNewThreadAptRefs: got %lx, expected %lx\n", dwNewThreadAptRefs, apt);
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
void Clear()
|
||
|
{
|
||
|
m_PreInitCalled = 0;
|
||
|
m_PostInitCalled = 0;
|
||
|
m_PreUninitCalled = 0;
|
||
|
m_PostUninitCalled = 0;
|
||
|
}
|
||
|
|
||
|
void Expect(HRESULT hrCoInit, DWORD CoInit, DWORD CurAptRefs)
|
||
|
{
|
||
|
m_hrCoInit = hrCoInit;
|
||
|
m_CoInit = CoInit;
|
||
|
m_CurAptRefs = CurAptRefs;
|
||
|
}
|
||
|
|
||
|
void Check(LONG PreInit, LONG PostInit, LONG PreUninit, LONG PostUninit)
|
||
|
{
|
||
|
ok(m_PreInitCalled == PreInit, "Expected PreInit to be %ld, was: %ld\n", PreInit, m_PreInitCalled);
|
||
|
ok(m_PostInitCalled == PostInit, "Expected PostInit to be %ld, was: %ld\n", PostInit, m_PostInitCalled);
|
||
|
ok(m_PreUninitCalled == PreUninit, "Expected PreUninit to be %ld, was: %ld\n", PreUninit, m_PreUninitCalled);
|
||
|
ok(m_PostUninitCalled == PostUninit, "Expected PostUninit to be %ld, was: %ld\n", PostUninit, m_PostUninitCalled);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
|
||
|
void test_IInitializeSpy_register2()
|
||
|
{
|
||
|
CTestSpy spy, spy2;
|
||
|
|
||
|
// first we register 2 spies
|
||
|
spy.hr = pCoRegisterInitializeSpy(&spy, &spy.Cookie);
|
||
|
test_S_OK(spy.hr, "CoRegisterInitializeSpy");
|
||
|
test_ref(&spy, 1);
|
||
|
|
||
|
spy2.hr = pCoRegisterInitializeSpy(&spy2, &spy2.Cookie);
|
||
|
test_S_OK(spy2.hr, "CoRegisterInitializeSpy");
|
||
|
test_ref(&spy, 1);
|
||
|
|
||
|
// tell them what we expect
|
||
|
spy.Expect(S_OK, COINIT_APARTMENTTHREADED, 1);
|
||
|
spy2.Expect(S_OK, COINIT_APARTMENTTHREADED, 1);
|
||
|
|
||
|
// Call CoInitializeEx and validate the results
|
||
|
HRESULT hr = CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
|
||
|
test_S_OK(hr, "CoInitializeEx");
|
||
|
spy.Check(1, 1, 0, 0);
|
||
|
spy2.Check(1, 1, 0, 0);
|
||
|
|
||
|
// Calling CoInit twice with the same apartment makes it return S_FALSE but still increment count
|
||
|
spy.Expect(S_FALSE, COINIT_APARTMENTTHREADED, 2);
|
||
|
spy2.Expect(S_FALSE, COINIT_APARTMENTTHREADED, 2);
|
||
|
|
||
|
hr = CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
|
||
|
test_HRES(hr, S_FALSE, "CoInitializeEx");
|
||
|
spy.Check(2, 2, 0, 0);
|
||
|
spy2.Check(2, 2, 0, 0);
|
||
|
|
||
|
/* the order we registered the spies in is important here.
|
||
|
we have the second one to forcibly return S_OK, which makes the first spy see
|
||
|
S_OK instead of S_FALSE.. */
|
||
|
spy.Expect(S_OK, COINIT_APARTMENTTHREADED, 3);
|
||
|
spy2.m_AlwaysReturnOK = true;
|
||
|
spy2.Expect(S_FALSE, COINIT_APARTMENTTHREADED, 3);
|
||
|
|
||
|
// and the S_OK also influences the returned value from CoInit.
|
||
|
hr = CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
|
||
|
test_S_OK(hr, "CoInitializeEx");
|
||
|
spy.Check(3, 3, 0, 0);
|
||
|
spy2.Check(3, 3, 0, 0);
|
||
|
|
||
|
CoUninitialize();
|
||
|
spy.Check(3, 3, 1, 1);
|
||
|
spy2.Check(3, 3, 1, 1);
|
||
|
|
||
|
spy.m_CurAptRefs = spy2.m_CurAptRefs = 2;
|
||
|
|
||
|
CoUninitialize();
|
||
|
spy.Check(3, 3, 2, 2);
|
||
|
spy2.Check(3, 3, 2, 2);
|
||
|
|
||
|
spy.m_CurAptRefs = spy2.m_CurAptRefs = 1;
|
||
|
|
||
|
CoUninitialize();
|
||
|
spy.Check(3, 3, 3, 3);
|
||
|
spy2.Check(3, 3, 3, 3);
|
||
|
|
||
|
spy.m_CurAptRefs = spy2.m_CurAptRefs = 0;
|
||
|
|
||
|
CoUninitialize();
|
||
|
spy.Check(3, 3, 4, 4);
|
||
|
spy2.Check(3, 3, 4, 4);
|
||
|
}
|
||
|
|
||
|
void test_IInitializeSpy_switch_apt()
|
||
|
{
|
||
|
CTestSpy spy;
|
||
|
|
||
|
spy.hr = pCoRegisterInitializeSpy(&spy, &spy.Cookie);
|
||
|
test_S_OK(spy.hr, "CoRegisterInitializeSpy");
|
||
|
test_ref(&spy, 1);
|
||
|
|
||
|
spy.Expect(S_OK, COINIT_APARTMENTTHREADED, 1);
|
||
|
|
||
|
HRESULT hr = CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
|
||
|
test_S_OK(hr, "CoInitializeEx");
|
||
|
spy.Check(1, 1, 0, 0);
|
||
|
|
||
|
spy.Expect(RPC_E_CHANGED_MODE, COINIT_MULTITHREADED, 1);
|
||
|
|
||
|
hr = CoInitializeEx(NULL, COINIT_MULTITHREADED);
|
||
|
test_HRES(hr, RPC_E_CHANGED_MODE, "CoInitializeEx");
|
||
|
spy.Check(2, 2, 0, 0);
|
||
|
|
||
|
|
||
|
CoUninitialize();
|
||
|
spy.Check(2, 2, 1, 1);
|
||
|
|
||
|
spy.m_CurAptRefs = 0;
|
||
|
|
||
|
CoUninitialize();
|
||
|
spy.Check(2, 2, 2, 2);
|
||
|
|
||
|
CoUninitialize();
|
||
|
spy.Check(2, 2, 3, 3);
|
||
|
}
|
||
|
|
||
|
void test_IInitializeSpy_fail()
|
||
|
{
|
||
|
CTestSpy spy;
|
||
|
|
||
|
spy.m_FailQueryInterface = true;
|
||
|
|
||
|
spy.hr = pCoRegisterInitializeSpy(&spy, &spy.Cookie);
|
||
|
test_HRES(spy.hr, E_NOINTERFACE, "Unexpected hr while registering invalid interface");
|
||
|
test_ref(&spy, 0);
|
||
|
ok(spy.Cookie.HighPart == 0xffffffff, "Unexpected Cookie.HighPart, expected 0xffffffff got: 0x%08lx\n", spy.Cookie.HighPart);
|
||
|
ok(spy.Cookie.LowPart == 0xffffffff, "Unexpected Cookie.HighPart, expected 0xffffffff got: 0x%08lx\n", spy.Cookie.LowPart);
|
||
|
|
||
|
spy.Cookie.HighPart = spy.Cookie.LowPart = 0xffffffff;
|
||
|
HRESULT hr = pCoRevokeInitializeSpy(spy.Cookie);
|
||
|
test_HRES(hr, E_INVALIDARG, "Unexpected hr while unregistering invalid interface");
|
||
|
test_ref(&spy, 0);
|
||
|
|
||
|
spy.Cookie.HighPart = spy.Cookie.LowPart = 0;
|
||
|
hr = pCoRevokeInitializeSpy(spy.Cookie);
|
||
|
test_HRES(hr, E_INVALIDARG, "Unexpected hr while unregistering invalid interface");
|
||
|
test_ref(&spy, 0);
|
||
|
|
||
|
/* we should not crash here, just return E_NOINTERFACE
|
||
|
do note the Cookie is not even being touched at all, compared to calling this with an interface
|
||
|
that does not respond to IID_IInitializeSpy */
|
||
|
spy.Cookie.HighPart = spy.Cookie.LowPart = INVALID_VALUE;
|
||
|
hr = pCoRegisterInitializeSpy(NULL, &spy.Cookie);
|
||
|
test_HRES(spy.hr, E_NOINTERFACE, "Unexpected hr while registering NULL interface");
|
||
|
ok(spy.Cookie.HighPart == INVALID_VALUE, "Unexpected Cookie.HighPart, expected 0xdeadbeef got: %lx\n", spy.Cookie.HighPart);
|
||
|
ok(spy.Cookie.LowPart == INVALID_VALUE, "Unexpected Cookie.HighPart, expected 0xdeadbeef got: %lx\n", spy.Cookie.LowPart);
|
||
|
}
|
||
|
|
||
|
void test_IInitializeSpy_twice()
|
||
|
{
|
||
|
CTestSpy spy;
|
||
|
|
||
|
spy.hr = pCoRegisterInitializeSpy(&spy, &spy.Cookie);
|
||
|
test_S_OK(spy.hr, "CoRegisterInitializeSpy");
|
||
|
test_ref(&spy, 1);
|
||
|
|
||
|
ULARGE_INTEGER Cookie = { { INVALID_VALUE, INVALID_VALUE } };
|
||
|
HRESULT hr = pCoRegisterInitializeSpy(&spy, &Cookie);
|
||
|
test_S_OK(hr, "CoRegisterInitializeSpy");
|
||
|
test_ref(&spy, 2);
|
||
|
|
||
|
hr = pCoRevokeInitializeSpy(Cookie);
|
||
|
test_S_OK(hr, "CoRevokeInitializeSpy");
|
||
|
test_ref(&spy, 1);
|
||
|
}
|
||
|
|
||
|
|
||
|
START_TEST(initializespy)
|
||
|
{
|
||
|
HMODULE ole32 = LoadLibraryA("ole32.dll");
|
||
|
pCoRegisterInitializeSpy = (pCoRegisterInitializeSpy_t)GetProcAddress(ole32, "CoRegisterInitializeSpy");
|
||
|
pCoRevokeInitializeSpy = (pCoRevokeInitializeSpy_t)GetProcAddress(ole32, "CoRevokeInitializeSpy");
|
||
|
|
||
|
test_IInitializeSpy_register2();
|
||
|
test_IInitializeSpy_switch_apt();
|
||
|
test_IInitializeSpy_fail();
|
||
|
test_IInitializeSpy_twice();
|
||
|
}
|