You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
276 lines
7.7 KiB
276 lines
7.7 KiB
/****************************************************************************
|
|
*
|
|
* FACTORY.cpp
|
|
*
|
|
* Microsoft Confidential
|
|
* Copyright (c) Microsoft Corporation 1992-1997
|
|
* All rights reserved
|
|
*
|
|
* This module provides the implementation of the methods for
|
|
* the CFactory class, which is used by COM's CoCreateInstance
|
|
*
|
|
* The code comes almost verbatim from Chapter 7 of Dale Rogerson's
|
|
* "Inside COM", and thus is minimally commented.
|
|
*
|
|
* 05/14/98 donaldm migrated from INETCFG
|
|
*
|
|
***************************************************************************/
|
|
|
|
#include "pre.h"
|
|
#include "icwconn.h"
|
|
// #include "icwextsn.h"
|
|
#include "icwaprtc.h"
|
|
|
|
// Friendly name of component
|
|
const TCHAR g_szFriendlyName[] = TEXT("CLSID_ApprenticeICWCONN") ;
|
|
|
|
// Version-independent ProgID
|
|
const TCHAR g_szVerIndProgID[] = TEXT("ICWCONN.Apprentice") ;
|
|
|
|
// ProgID
|
|
const TCHAR g_szProgID[] = TEXT("ICWCONN.Apprentice.1") ;
|
|
|
|
static long g_cComponents = 0 ; // Count of active components
|
|
static long g_cServerLocks = 0 ; // Count of locks
|
|
|
|
///////////////////////////////////////////////////////////
|
|
//
|
|
// Class factory
|
|
//
|
|
class CFactory : public IClassFactory
|
|
{
|
|
public:
|
|
// IUnknown
|
|
virtual HRESULT __stdcall QueryInterface(const IID& iid, void** ppv) ;
|
|
virtual ULONG __stdcall AddRef() ;
|
|
virtual ULONG __stdcall Release() ;
|
|
|
|
// Interface IClassFactory
|
|
virtual HRESULT __stdcall CreateInstance(IUnknown* pUnknownOuter,
|
|
const IID& iid,
|
|
void** ppv) ;
|
|
virtual HRESULT __stdcall LockServer(BOOL bLock) ;
|
|
|
|
// Constructor
|
|
CFactory() : m_cRef(1) {}
|
|
|
|
// Destructor
|
|
~CFactory() { TraceMsg(TF_CLASSFACTORY, "Class factory:\t\tDestroy self.") ;}
|
|
|
|
private:
|
|
long m_cRef ;
|
|
} ;
|
|
|
|
//
|
|
// Class factory IUnknown implementation
|
|
//
|
|
HRESULT __stdcall CFactory::QueryInterface(const IID& iid, void** ppv)
|
|
{
|
|
TraceMsg(TF_CLASSFACTORY, "CFactory::QueryInterface");
|
|
if ((iid == IID_IUnknown) || (iid == IID_IClassFactory))
|
|
{
|
|
*ppv = static_cast<IClassFactory*>(this) ;
|
|
}
|
|
else
|
|
{
|
|
*ppv = NULL ;
|
|
return E_NOINTERFACE ;
|
|
}
|
|
reinterpret_cast<IUnknown*>(*ppv)->AddRef() ;
|
|
return S_OK ;
|
|
}
|
|
|
|
ULONG __stdcall CFactory::AddRef()
|
|
{
|
|
TraceMsg(TF_CLASSFACTORY, "CFactory::AddRef %d", m_cRef + 1);
|
|
return InterlockedIncrement(&m_cRef) ;
|
|
}
|
|
|
|
ULONG __stdcall CFactory::Release()
|
|
{
|
|
if (InterlockedDecrement(&m_cRef) == 0)
|
|
{
|
|
delete this ;
|
|
return 0 ;
|
|
}
|
|
TraceMsg(TF_CLASSFACTORY, "CFactory::Release %d", m_cRef);
|
|
return m_cRef ;
|
|
}
|
|
|
|
//
|
|
// IClassFactory implementation
|
|
//
|
|
HRESULT __stdcall CFactory::CreateInstance(IUnknown* pUnknownOuter,
|
|
const IID& iid,
|
|
void** ppv)
|
|
{
|
|
TraceMsg(TF_CLASSFACTORY, "CFactory::CreateInstance:\t\tCreate component.") ;
|
|
|
|
// Cannot aggregate.
|
|
if (pUnknownOuter != NULL)
|
|
{
|
|
return CLASS_E_NOAGGREGATION ;
|
|
}
|
|
|
|
// Create component. Since there's no direct IUnknown implementation,
|
|
// use CICWApprentice.
|
|
CICWApprentice *pApprentice = new CICWApprentice;
|
|
|
|
TraceMsg(TF_CLASSFACTORY, "CFactory::CreateInstance CICWApprentice->AddRef");
|
|
pApprentice->AddRef();
|
|
|
|
if( NULL == pApprentice )
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
// Get the requested interface.
|
|
TraceMsg(TF_CLASSFACTORY, "CFactory::CreateInstance About to QI on CICWApprentice");
|
|
HRESULT hr = pApprentice->QueryInterface(iid, ppv) ;
|
|
|
|
// Release the IUnknown pointer.
|
|
// (If QueryInterface failed, component will delete itself.)
|
|
TraceMsg(TF_CLASSFACTORY, "CFactory::CreateInstance done with CICWApprentice, releasing (aprtc should have ct of 1)");
|
|
pApprentice->Release() ;
|
|
|
|
return hr ;
|
|
}
|
|
|
|
// LockServer
|
|
HRESULT __stdcall CFactory::LockServer(BOOL bLock)
|
|
{
|
|
if (bLock)
|
|
{
|
|
InterlockedIncrement(&g_cServerLocks) ;
|
|
}
|
|
else
|
|
{
|
|
InterlockedDecrement(&g_cServerLocks) ;
|
|
}
|
|
return S_OK ;
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////
|
|
//
|
|
// Exported functions
|
|
//
|
|
// These are the functions that COM expects to find
|
|
//
|
|
|
|
//
|
|
// Can DLL unload now?
|
|
//
|
|
STDAPI DllCanUnloadNow()
|
|
{
|
|
if ((g_cComponents == 0) && (g_cServerLocks == 0))
|
|
{
|
|
return S_OK ;
|
|
}
|
|
else
|
|
{
|
|
return S_FALSE ;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Get class factory
|
|
//
|
|
STDAPI DllGetClassObject(const CLSID& clsid,
|
|
const IID& iid,
|
|
void** ppv)
|
|
{
|
|
TraceMsg(TF_CLASSFACTORY, "DllGetClassObject:\tCreate class factory.") ;
|
|
|
|
// Can we create this component?
|
|
if (clsid != CLSID_ApprenticeICWCONN)
|
|
{
|
|
return CLASS_E_CLASSNOTAVAILABLE ;
|
|
}
|
|
|
|
// Create class factory.
|
|
CFactory* pFactory = new CFactory ; // No AddRef in constructor
|
|
if (pFactory == NULL)
|
|
{
|
|
return E_OUTOFMEMORY ;
|
|
}
|
|
|
|
// Get requested interface.
|
|
TraceMsg(TF_CLASSFACTORY, "DllGetClassObject about to QI on CFactory");
|
|
HRESULT hr = pFactory->QueryInterface(iid, ppv) ;
|
|
TraceMsg(TF_CLASSFACTORY, "DllGetClassObject done with CFactory, releasing");
|
|
pFactory->Release() ;
|
|
|
|
|
|
|
|
return hr ;
|
|
}
|
|
|
|
|
|
// The following two exported functions are what regsvr32 uses to
|
|
// self-register and unregister the dll. See REGISTRY.CPP for
|
|
// actual implementation
|
|
|
|
//
|
|
// Server registration
|
|
//
|
|
|
|
typedef BOOL (WINAPI * REGISTERSERVER)(HMODULE hModule,
|
|
const CLSID& clsid,
|
|
const LPTSTR szFriendlyName,
|
|
const LPTSTR szVerIndProgID,
|
|
const LPTSTR szProgID);
|
|
|
|
typedef BOOL (WINAPI * UNREGISTERSERVER)(const CLSID& clsid,
|
|
const LPTSTR szVerIndProgID,
|
|
const LPTSTR szProgID);
|
|
|
|
STDAPI DllRegisterServer()
|
|
{
|
|
// Use the register server function that is in ICWUTIL.DLL
|
|
HINSTANCE hInstUtilDLL = LoadLibrary(ICW_UTIL);
|
|
REGISTERSERVER lpfnRegisterServer = NULL;
|
|
HRESULT hr = E_FAIL;
|
|
|
|
if (hInstUtilDLL)
|
|
{
|
|
lpfnRegisterServer = (REGISTERSERVER)GetProcAddress(hInstUtilDLL, "RegisterServer");
|
|
if (NULL != lpfnRegisterServer)
|
|
{
|
|
if (lpfnRegisterServer(ghInstance,
|
|
CLSID_ApprenticeICWCONN,
|
|
(LPTSTR)g_szFriendlyName,
|
|
(LPTSTR)g_szVerIndProgID,
|
|
(LPTSTR)g_szProgID))
|
|
hr = S_OK;
|
|
}
|
|
FreeLibrary(hInstUtilDLL);
|
|
}
|
|
return (hr);
|
|
}
|
|
|
|
|
|
//
|
|
// Server unregistration
|
|
//
|
|
STDAPI DllUnregisterServer()
|
|
{
|
|
// Use the un-register server function that is in ICWUTIL.DLL
|
|
HINSTANCE hInstUtilDLL = LoadLibrary(ICW_UTIL);
|
|
UNREGISTERSERVER lpfnUnregisterServer = NULL;
|
|
HRESULT hr = E_FAIL;
|
|
|
|
if (hInstUtilDLL)
|
|
{
|
|
lpfnUnregisterServer = (UNREGISTERSERVER)GetProcAddress(hInstUtilDLL, "UnregisterServer");
|
|
if (NULL != lpfnUnregisterServer)
|
|
{
|
|
if (lpfnUnregisterServer(CLSID_ApprenticeICWCONN,
|
|
(LPTSTR)g_szVerIndProgID,
|
|
(LPTSTR)g_szProgID))
|
|
hr = S_OK;
|
|
}
|
|
FreeLibrary(hInstUtilDLL);
|
|
}
|
|
return (hr);
|
|
}
|