⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 local.cpp

📁 经验交流,从网上下载的好东西望大家分享
💻 CPP
字号:
/*
 Bug in MIDL: Comment out the three references to _maxcount_test in the component_p.c file
*/

// local.cpp
#define _WIN32_DCOM
#include <windows.h>
#include <iostream.h>  // For cout
#include "registry.h"  // For registry functions
#include "Component\component.h" // Generated by MIDL
#include <stdio.h>
#include <conio.h>

long g_cComponents = 0;
long g_cServerLocks = 0;
HANDLE g_hEvent;

class CTestIDL : public ITest
{
public:
	// IUnknown
	ULONG __stdcall AddRef();
	ULONG __stdcall Release();
	HRESULT __stdcall QueryInterface(REFIID riid, void** ppv);

	// ITest
	HRESULT __stdcall SquareInteger(int myInteger, int* square);
	HRESULT __stdcall SumOfIntegers1(int myIntegers[5], int* sum);
	HRESULT __stdcall SumOfIntegers2(int cMax, int* myIntegers, int* sum);
	HRESULT __stdcall SumOfIntegers3(SUM_STRUCT* myIntegers, int* sum);
	HRESULT __stdcall SumOfIntegers4(int cFirst, int cActual, int* myIntegers[7], int* sum);
	HRESULT __stdcall ProduceIntegers1(int cMax, int* myIntegers);
	HRESULT __stdcall ProduceIntegers2(int* pcActual, int myIntegers[4096]);
	HRESULT __stdcall ProduceIntegers3(int cMax, int* pcActual, int* myIntegers);
	HRESULT __stdcall SendReceiveIntegers(int* pcActual, int myIntegers[4096]);
	HRESULT __stdcall SendString1(int cLength, wchar_t* myString);
	HRESULT __stdcall SendString2(wchar_t* myString);
	HRESULT __stdcall SendReceiveString1(wchar_t* myString);
	HRESULT __stdcall SendReceiveString2(int cMax, wchar_t* myString);
	HRESULT __stdcall SendReceiveString3(wchar_t** myString);
	HRESULT __stdcall SendInteger(int** test);
	HRESULT __stdcall SendIntegers1(int** test);
	HRESULT __stdcall SendIntegers2(int** test);
	HRESULT __stdcall SendIntegers3(int** test);

	CTestIDL() : m_cRef(1) { g_cComponents++; }
	~CTestIDL() { cout << "Component: CTestIDL::~CTestIDL()" << endl, g_cComponents--; }

private:
	ULONG m_cRef;
};

ULONG CTestIDL::AddRef()
{
	cout << "Component: CTestIDL::AddRef() m_cRef = " << m_cRef + 1 << endl;
	return ++m_cRef;
}

ULONG CTestIDL::Release()
{
	cout << "Component: CTestIDL::Release() m_cRef = " << m_cRef - 1 << endl;
	if(--m_cRef != 0)
		return m_cRef;
	SetEvent(g_hEvent);	// ADD THIS!!!
	delete this;
	return 0;
}

HRESULT CTestIDL::QueryInterface(REFIID riid, void** ppv)
{
	if(riid == IID_IUnknown)
	{
		cout << "Component: CTestIDL::QueryInterface() for IUnknown returning " << this << endl;
		*ppv = reinterpret_cast<IUnknown*>(this);
	}
	else if(riid == IID_ITest)
	{
		cout << "Component: CTestIDL::QueryInterface() for ITest returning " << this << endl;
		*ppv = (ITest*)this;
	}
	else 
	{
		*ppv = NULL;
		return E_NOINTERFACE;
	}
	AddRef();
	return S_OK;
}

// HRESULT SquareInteger([in] int myInteger, [out] int* square);
HRESULT CTestIDL::SquareInteger(int myInteger, int* square)
{
	*square = myInteger * myInteger;
	return S_OK;
}

// HRESULT SumOfIntegers1([in] int myIntegers[5], [out] int* sum);
HRESULT CTestIDL::SumOfIntegers1(int myIntegers[5], int* sum)
{
	*sum = 0;
	for(int i = 0; i < 5; i++)
		*sum = *sum + myIntegers[i];
	return S_OK;
}

// HRESULT SumOfIntegers2([in] int cMax, [in, size_is(cMax)] int* myIntegers, [out] int* sum);
HRESULT CTestIDL::SumOfIntegers2(int cMax, int* myIntegers, int* sum)
{
	*sum = 0;
	for(int i = 0; i < cMax; i++)
		*sum = *sum + myIntegers[i];
	return S_OK;
}

/*	typedef struct tagSUM_STRUCT
	{
		int cMax;
		[size_is(cMax)] int* myIntegers;
	} SUM_STRUCT; */
// HRESULT SumOfIntegers3([in] SUM_STRUCT* myIntegers, [out] int* sum);
HRESULT CTestIDL::SumOfIntegers3(SUM_STRUCT* myIntegers, int* sum)
{
	*sum = 0;
	for(int i = 0; i < myIntegers->cMax; i++)
		*sum = *sum + myIntegers->myIntegers[i];
	return S_OK;
}

// HRESULT SumOfIntegers4([in] int cFirst, [in] int cActual, [in, first_is(cFirst), length_is(cActual)] int* myIntegers[7], [out] int* sum);
HRESULT CTestIDL::SumOfIntegers4(int cFirst, int cActual, int* myIntegers[7], int* sum)
{
	*sum = 3;
	for(int i = cFirst; i < cFirst + cActual; i++)
	{
		cout << "i = " << i << " cFirst = " << cFirst << " cActual = " << cActual << endl;
		cout << "myIntegers = " << myIntegers[i][0] << endl;
//		*sum = *sum + *myIntegers[i];
	}
	_getch();
	return S_OK;
}

// HRESULT ProduceIntegers1([in] int cMax, [out, size_is(cMax)] int* myIntegers);
HRESULT CTestIDL::ProduceIntegers1(int cMax, int* myIntegers)
{
	for(int i = 0; i < cMax; i++)
		myIntegers[i] = i;
	return S_OK;
}

// HRESULT ProduceIntegers2([out] int* pcActual, [out, length_is(*pcActual)] int myIntegers[4096]);
HRESULT CTestIDL::ProduceIntegers2(int* pcActual, int myIntegers[4096])
{
	*pcActual = 400;
	for(int i = 0; i < *pcActual; i++)
		myIntegers[i] = i;
	return S_OK;
}

// HRESULT ProduceIntegers3([in] int cMax, [out] int* pcActual, [out, size_is(cMax), length_is(*pcActual)] int* myIntegers);
HRESULT CTestIDL::ProduceIntegers3(int cMax, int* pcActual, int* myIntegers)
{
	*pcActual = cMax / 2;
	for(int i = 0; i < *pcActual; i++)
		myIntegers[i] = i;
	return S_OK;
}

// HRESULT SendReceiveIntegers([in, out] int* pcActual, [in, out, length_is(*pcActual)] int myIntegers[4096]);
HRESULT CTestIDL::SendReceiveIntegers(int* pcActual, int myIntegers[4096])
{
	// In
	cout << "SendReceiveIntegers: << " << *pcActual << " integers received" << endl;
	for(int i = 0; i < *pcActual; i++)
		cout << "SendReceiveIntegers: " << myIntegers[i] << endl;

	// Out
	*pcActual = 400;
	for(i = 0; i < *pcActual; i++)
		myIntegers[i] = i;
	return S_OK;
}

// HRESULT SendString1([in] int cLength, [in, size_is(cLength)] wchar_t* myString);
HRESULT CTestIDL::SendString1(int cLength, wchar_t* myString)
{
	wprintf(L"String of %d characters contains %s\n", cLength, myString);
	return S_OK;
}

// HRESULT SendString2([in, string] wchar_t* myString);
HRESULT CTestIDL::SendString2(wchar_t* myString)
{
	wprintf(L"String contains %s\n", myString);
	return S_OK;
}

// HRESULT SendReceiveString1([in, out, string] wchar_t* myString);
HRESULT CTestIDL::SendReceiveString1(wchar_t* myString)
{
	wprintf(L"SendReceiveString1: %s\n", myString); // Hello COM

	// This will cause a fault!
	//	wcscpy(myString, L"Nice weather today Nice weather today Nice weather today Nice weather today Nice weather today Nice weather today Nice weather today"); // Oh-oh

	// Nine characters max
	wcscpy(myString, L"only nine"); // OK
	return S_OK;
}

// HRESULT SendReceiveString2([in] int cMax, [in, out, string, size_is(cMax)] wchar_t* myString);
HRESULT CTestIDL::SendReceiveString2(int cMax, wchar_t* myString)
{
	wprintf(L"SendReceiveString2: %s\n", myString); // Hello COM
	for(int i = 0; i < cMax - 1; i++)
		myString[i] = L'A';
	myString[cMax - 1] = L'\0';
	return S_OK;
}

// HRESULT SendReceiveString3([in, out, string] wchar_t** myString);
HRESULT CTestIDL::SendReceiveString3(wchar_t** myString)
{
	wprintf(L"SendReceiveString3: %s\n", *myString);
	CoTaskMemFree(*myString);
	wchar_t returnString[] = L"Nice weather today";
	*myString = (wchar_t*)CoTaskMemAlloc((wcslen(returnString)+1)*sizeof(wchar_t));
	wcscpy(*myString, returnString);
	return S_OK;
}

// HRESULT SendInteger([in] int** test);
HRESULT CTestIDL::SendInteger(int** test)
{
	cout << "SendInteger: test[0][0] = " << test[0][0] << endl;
	return S_OK;
}

// HRESULT SendIntegers1([in, size_is(, 2)] int** test);
HRESULT CTestIDL::SendIntegers1(int** test)
{
	cout << "SendIntegers1: test[0][0] = " << test[0][0] << endl;
	cout << "SendIntegers1: test[0][1] = " << test[0][1] << endl;
	return S_OK;
}

// HRESULT SendIntegers2([in, size_is(3, )] int** test);
HRESULT CTestIDL::SendIntegers2(int** test)
{
	cout << "SendIntegers2: test[0][0] = " << test[0][0] << endl;
	cout << "SendIntegers2: test[1][0] = " << test[1][0] << endl;
	cout << "SendIntegers2: test[2][0] = " << test[2][0] << endl;
	return S_OK;
}

// HRESULT SendIntegers3([in, size_is(3, 2)] int** test);
HRESULT CTestIDL::SendIntegers3(int** test)
{
	cout << "SendIntegers3: test[0][0] = " << test[0][0] << endl;
	cout << "SendIntegers3: test[0][1] = " << test[0][1] << endl;
	cout << "SendIntegers3: test[1][0] = " << test[1][0] << endl;
	cout << "SendIntegers3: test[1][1] = " << test[1][1] << endl;
	cout << "SendIntegers3: test[2][0] = " << test[2][0] << endl;
	cout << "SendIntegers3: test[2][1] = " << test[2][1] << endl;
	return S_OK;
}

class CFactory : public IClassFactory
{
public:
	// IUnknown
	ULONG __stdcall AddRef();
	ULONG __stdcall Release();
	HRESULT __stdcall QueryInterface(REFIID riid, void** ppv);

	// IClassFactory
	HRESULT __stdcall CreateInstance(IUnknown *pUnknownOuter, REFIID riid, void** ppv);
	HRESULT __stdcall LockServer(BOOL bLock);

	CFactory() : m_cRef(1) { }
	~CFactory() { }

private:
	ULONG m_cRef;
};

ULONG CFactory::AddRef()
{
	cout << "Component: CFactory::AddRef() m_cRef = " << m_cRef + 1 << endl;
	return ++m_cRef;
}

ULONG CFactory::Release()
{
	cout << "Component: CFactory::Release() m_cRef = " << m_cRef - 1 << endl;
	if(--m_cRef != 0)
		return m_cRef;
	delete this;
	return 0;
}

HRESULT CFactory::QueryInterface(REFIID riid, void** ppv)
{
	if((riid == IID_IUnknown) || (riid == IID_IClassFactory))
	{
		cout << "Component: CFactory::QueryInteface() for IUnknown or IClassFactory " << this << endl;
		*ppv = (IClassFactory *)this;
	}
	else
	{
		*ppv = NULL;
		return E_NOINTERFACE;
	}
	AddRef();
	return S_OK;
}

HRESULT CFactory::CreateInstance(IUnknown *pUnknownOuter, REFIID riid, void** ppv)
{
	if(pUnknownOuter != NULL)
		return CLASS_E_NOAGGREGATION;

	CTestIDL *pTestIDL = new CTestIDL;
	cout << "Component: CFactory::CreateInstance() " << pTestIDL << endl;

	if(pTestIDL == NULL)
		return E_OUTOFMEMORY;

	// QueryInterface probably for IID_IUNKNOWN
	HRESULT hr = pTestIDL->QueryInterface(riid, ppv);
	pTestIDL->Release();
	return hr;
}

HRESULT CFactory::LockServer(BOOL bLock)
{
	if(bLock)
		g_cServerLocks++;
	else
		g_cServerLocks--;
	return S_OK;
}

void RegisterComponent()
{
	ITypeLib* pTypeLib;
	LoadTypeLibEx(L"Component.exe", REGKIND_DEFAULT, &pTypeLib);
	RegisterServer("Component.exe", CLSID_TestIDL, "Test IDL Sample", "Component.TestIDL", "Component.TestIDL.1", NULL);
}

void CommandLineParameters(int argc, char** argv)
{
	RegisterComponent();
	if(argc < 2)
	{
		cout << "No parameter, but registered anyway" << endl;
		exit(false);
	}
	char* szToken = strtok(argv[1], "-/"); 
	if(_stricmp(szToken, "RegServer") == 0)
	{
		RegisterComponent();
		cout << "RegServer" << endl;
		exit(true);
	}
	if(_stricmp(szToken, "UnregServer") == 0)
	{
		UnRegisterTypeLib(LIBID_Component, 1, 0, LANG_NEUTRAL, SYS_WIN32);
		UnregisterServer(CLSID_TestIDL, "Component.TestIDL", "Component.TestIDL.1");
		cout << "UnregServer" << endl;
		exit(true);
	}
	if(_stricmp(szToken, "Embedding") != 0)
	{
		cout << "Invalid parameter" << endl;
		exit(false);
	}
}

void main(int argc, char** argv)
{
	CommandLineParameters(argc, argv);

	cout << "Component: CoInitializeEx()" << endl;
	CoInitializeEx(NULL, COINIT_MULTITHREADED);

	IClassFactory *pClassFactory = new CFactory();

	cout << "Component: CoRegisterClassObject()" << endl;
	DWORD dwRegister;
	CoRegisterClassObject(CLSID_TestIDL, pClassFactory, CLSCTX_LOCAL_SERVER, REGCLS_MULTIPLEUSE, &dwRegister);

	g_hEvent = CreateEvent(NULL, FALSE, FALSE, NULL);
	WaitForSingleObject(g_hEvent, INFINITE);

	CoRevokeClassObject(dwRegister);
	pClassFactory->Release();
	CoUninitialize();

	_getch();
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -