// // server.cpp // // COM server exports. // #include "globals.h" #include "mark.h" void FreeGlobalObjects(void); class CClassFactory; static CClassFactory *g_ObjectInfo[1] = { NULL }; //+--------------------------------------------------------------------------- // // DllAddRef // //---------------------------------------------------------------------------- LONG DllAddRef(void) { return InterlockedIncrement(&g_cRefDll) + 1; } //+--------------------------------------------------------------------------- // // DllRelease // //---------------------------------------------------------------------------- LONG DllRelease(void) { if (InterlockedDecrement(&g_cRefDll) < 0) // g_cRefDll == -1 with zero refs { EnterCriticalSection(&g_cs); // need to check ref again after grabbing mutex if (g_ObjectInfo[0] != NULL) { FreeGlobalObjects(); } assert(g_cRefDll == -1); LeaveCriticalSection(&g_cs); } return g_cRefDll+1; } //+--------------------------------------------------------------------------- // // CClassFactory declaration with IClassFactory Interface // //---------------------------------------------------------------------------- class CClassFactory : public IClassFactory { public: // IUnknown methods STDMETHODIMP QueryInterface(REFIID riid, void **ppvObj); STDMETHODIMP_(ULONG) AddRef(void); STDMETHODIMP_(ULONG) Release(void); // IClassFactory methods STDMETHODIMP CreateInstance(IUnknown *pUnkOuter, REFIID riid, void **ppvObj); STDMETHODIMP LockServer(BOOL fLock); // Constructor CClassFactory(REFCLSID rclsid, HRESULT (*pfnCreateInstance)(IUnknown *pUnkOuter, REFIID riid, void **ppvObj)) : _rclsid(rclsid) { _pfnCreateInstance = pfnCreateInstance; } public: REFCLSID _rclsid; HRESULT (*_pfnCreateInstance)(IUnknown *pUnkOuter, REFIID riid, void **ppvObj); }; //+--------------------------------------------------------------------------- // // CClassFactory::QueryInterface // //---------------------------------------------------------------------------- STDAPI CClassFactory::QueryInterface(REFIID riid, void **ppvObj) { if (IsEqualIID(riid, IID_IClassFactory) || IsEqualIID(riid, IID_IUnknown)) { *ppvObj = this; DllAddRef(); return NOERROR; } *ppvObj = NULL; return E_NOINTERFACE; } //+--------------------------------------------------------------------------- // // CClassFactory::AddRef // //---------------------------------------------------------------------------- STDAPI_(ULONG) CClassFactory::AddRef() { DllAddRef(); return g_cRefDll+1; // -1 w/ no refs } //+--------------------------------------------------------------------------- // // CClassFactory::Release // //---------------------------------------------------------------------------- STDAPI_(ULONG) CClassFactory::Release() { DllRelease(); return g_cRefDll+1; // -1 w/ no refs } //+--------------------------------------------------------------------------- // // CClassFactory::CreateInstance // //---------------------------------------------------------------------------- STDAPI CClassFactory::CreateInstance(IUnknown *pUnkOuter, REFIID riid, void **ppvObj) { return _pfnCreateInstance(pUnkOuter, riid, ppvObj); } //+--------------------------------------------------------------------------- // // CClassFactory::LockServer // //---------------------------------------------------------------------------- STDAPI CClassFactory::LockServer(BOOL fLock) { if (fLock) { DllAddRef(); } else { DllRelease(); } return S_OK; } //+--------------------------------------------------------------------------- // // BuildGlobalObjects // //---------------------------------------------------------------------------- void BuildGlobalObjects(void) { // Build CClassFactory Objects g_ObjectInfo[0] = new CClassFactory(c_clsidMarkTextService, CMarkTextService::CreateInstance); // You can add more object info here. // Don't forget to increase number of item for g_ObjectInfo[], } //+--------------------------------------------------------------------------- // // FreeGlobalObjects // //---------------------------------------------------------------------------- void FreeGlobalObjects(void) { // Free CClassFactory Objects for (int i = 0; i < ARRAYSIZE(g_ObjectInfo); i++) { if (NULL != g_ObjectInfo[i]) { delete g_ObjectInfo[i]; g_ObjectInfo[i] = NULL; } } } //+--------------------------------------------------------------------------- // // DllGetClassObject // //---------------------------------------------------------------------------- STDAPI DllGetClassObject(REFCLSID rclsid, REFIID riid, void **ppvObj) { if (g_ObjectInfo[0] == NULL) { EnterCriticalSection(&g_cs); // need to check ref again after grabbing mutex if (g_ObjectInfo[0] == NULL) { BuildGlobalObjects(); } LeaveCriticalSection(&g_cs); } if (IsEqualIID(riid, IID_IClassFactory) || IsEqualIID(riid, IID_IUnknown)) { for (int i = 0; i < ARRAYSIZE(g_ObjectInfo); i++) { if (NULL != g_ObjectInfo[i] && IsEqualGUID(rclsid, g_ObjectInfo[i]->_rclsid)) { *ppvObj = (void *)g_ObjectInfo[i]; DllAddRef(); // class factory holds DLL ref count return NOERROR; } } } *ppvObj = NULL; return CLASS_E_CLASSNOTAVAILABLE; } //+--------------------------------------------------------------------------- // // DllCanUnloadNow // //---------------------------------------------------------------------------- STDAPI DllCanUnloadNow(void) { if (g_cRefDll >= 0) // -1 with no refs return S_FALSE; return S_OK; } //+--------------------------------------------------------------------------- // // DllUnregisterServer // //---------------------------------------------------------------------------- STDAPI DllUnregisterServer(void) { CMarkTextService::RegisterCategories(FALSE); CMarkTextService::UnregisterProfiles(); CMarkTextService::UnregisterServer(); return S_OK; } //+--------------------------------------------------------------------------- // // DllRegisterServer // //---------------------------------------------------------------------------- STDAPI DllRegisterServer(void) { // register this service's profile with the tsf if (!CMarkTextService::RegisterServer() || !CMarkTextService::RegisterProfiles() || !CMarkTextService::RegisterCategories(TRUE)) { DllUnregisterServer(); // cleanup any loose ends return E_FAIL; } return S_OK; }