windows-nt/Source/XPSP1/NT/base/wmi/bmoflocparser/dllentry.cpp
2020-09-26 16:20:57 +08:00

402 lines
9 KiB
C++

//------------------------------------------------------------------------------
//
// File: dllentry.cpp
// Copyright (C) 1995-1997 Microsoft Corporation
// All rights reserved.
//
// Purpose:
// Defines the initialization routines for the DLL.
//
// This file needs minor changes, as marked by TODO comments. However, the
// functions herein are only called by the system, Espresso, or the framework,
// and you should not need to look at them extensively.
//
// Owner:
//
//------------------------------------------------------------------------------
#include "stdafx.h"
#include "clasfact.h"
#include "win32sub.h"
#include "impbin.h"
#include "misc.h"
#include "resource.h"
#define __DLLENTRY_CPP
#include "dllvars.h"
#ifdef _DEBUG
#undef THIS_FILE
static char BASED_CODE THIS_FILE[] = __FILE__;
#endif
#define new DEBUG_NEW
LONG g_lActiveClasses = 0; //Glbal count of active class in the DLL
static AFX_EXTENSION_MODULE g_parseDLL = { NULL, NULL };
CItemSetException g_SetException(FALSE);
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//
// DLL Main entry
//
//------------------------------------------------------------------------------
extern "C" int APIENTRY
DllMain(HINSTANCE hInstance, DWORD dwReason, LPVOID lpReserved)
{
UNREFERENCED_PARAMETER(lpReserved);
int nRet = 1; //OK
if (dwReason == DLL_PROCESS_ATTACH)
{
LTTRACE("BMOF.DLL Initializing!\n"); //TODO - change name
// Extension DLL one-time initialization
AfxInitExtensionModule(g_parseDLL, hInstance);
// Insert this DLL into the resource chain
new CDynLinkLibrary(g_parseDLL);
g_hDll = hInstance;
}
else if (dwReason == DLL_PROCESS_DETACH)
{
LTTRACE("BMOF.DLL Terminating!\n"); //TODO - change name
// Remove this DLL from MFC's list of extensions
AfxTermExtensionModule(g_parseDLL);
//
// If there are active classes, they WILL explode badly once the
// DLL is unloaded...
//
LTASSERT(DllCanUnloadNow() == S_OK);
AfxTermExtensionModule(g_parseDLL);
}
return nRet;
}
// TODO: Use GUIDGEN.EXE to replace this class ID with a unique one.
// GUIDGEN is supplied with MSDEV (VC++ 4.0) as part of the OLE support stuff.
// Run it and you'll get a little dialog box. Pick radio button 3, "static
// const struct GUID = {...}". Click on the "New GUID" button, then the "Copy"
// button, which puts the result in the clipboard. From there, you can just
// paste it into here. Just remember to change the type to CLSID!
// {8B75CD76-DFC1-4356-AC04-AF088B448AB3}
static const CLSID ciImpParserCLSID =
{ 0x8b75cd76, 0xdfc1, 0x4356, { 0xac, 0x4, 0xaf, 0x8, 0x8b, 0x44, 0x8a, 0xb3 } };
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//
// Return the CLSID of the parser
//
//------------------------------------------------------------------------------
STDAPI_(void)
DllGetParserCLSID(
CLSID &ciParserCLSID)
{
ciParserCLSID = ciImpParserCLSID;
}
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//
// Entry point to register this parser. Calls base implementation in ESPUTIL.
//------------------------------------------------------------------------------
STDAPI
DllRegisterParser()
{
LTASSERT(g_hDll != NULL);
HRESULT hr = ResultFromScode(E_UNEXPECTED);
try
{
hr = RegisterParser(g_hDll);
}
catch (CException* pE)
{
pE->Delete();
}
catch (...)
{
}
return ResultFromScode(hr);
}
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//
// Entry point to unregister this parser. Calls the base implementation in
// ESPUTIL.
//------------------------------------------------------------------------------
STDAPI
DllUnregisterParser()
{
LTASSERT(g_hDll != NULL);
HRESULT hr = ResultFromScode(E_UNEXPECTED);
try
{
//TODO**: Change pidBMOF to real sub parser ID
hr = UnregisterParser(pidBMOF, pidWin32);
}
catch (CException* pE)
{
pE->Delete();
}
catch (...)
{
}
return ResultFromScode(hr);
}
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//
// Return the class factory for the requested class ID
//
//------------------------------------------------------------------------------
STDAPI
DllGetClassObject(
REFCLSID cidRequestedClass,
REFIID iid,
LPVOID *ppClassFactory)
{
SCODE sc = E_UNEXPECTED;
*ppClassFactory = NULL;
if (cidRequestedClass != ciImpParserCLSID)
{
sc = CLASS_E_CLASSNOTAVAILABLE;
}
else
{
try
{
CLocImpClassFactory *pClassFactory;
pClassFactory = new CLocImpClassFactory;
sc = pClassFactory->QueryInterface(iid, ppClassFactory);
pClassFactory->Release();
}
catch (CMemoryException *pMem)
{
sc = E_OUTOFMEMORY;
pMem->Delete();
}
catch (CException* pE)
{
sc = E_UNEXPECTED;
pE->Delete();
}
catch (...)
{
sc = E_UNEXPECTED;
}
}
return ResultFromScode(sc);
}
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//
// Return true if the parser can be unloaded
//
//------------------------------------------------------------------------------
STDAPI
DllCanUnloadNow(void)
{
SCODE sc = (g_lActiveClasses == 0) ? S_OK : S_FALSE;
return ResultFromScode(sc);
}
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//
// Increment the global count of active classes
//
//------------------------------------------------------------------------------
void
IncrementClassCount(void)
{
InterlockedIncrement(&g_lActiveClasses);
}
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//
// Decrement the global count of active classes
//
//------------------------------------------------------------------------------
void
DecrementClassCount(void)
{
LTASSERT(g_lActiveClasses != 0);
InterlockedDecrement(&g_lActiveClasses);
}
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//
// Throw a item set exception
//
//------------------------------------------------------------------------------
void
ThrowItemSetException()
{
throw &g_SetException;
}
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//
// Report a error through the reporter. This function will never
// fail or throw an exception out of the function.
//
//------------------------------------------------------------------------------
void
ReportException(
CException* pExcep, //May be null
C32File* p32File, //May be null
CLocItem* pItem, //May be null
CReporter* pReporter)
{
LTASSERT(NULL != pReporter);
//Don't let this function throw an exception since it is normally called
//within exception catch blocks
try
{
CLString strContext;
if (NULL != p32File)
{
strContext = p32File->GetFile()->GetFilePath();
}
else
{
LTVERIFY(strContext.LoadString(g_hDll, IDS_IMP_DESC));
}
CLString strExcep;
BOOL bErrorFormatted = FALSE;
if (NULL != pExcep)
{
bErrorFormatted =
pExcep->GetErrorMessage(strExcep.GetBuffer(512), 512);
strExcep.ReleaseBuffer();
}
if (!bErrorFormatted || NULL == pExcep)
{
LTVERIFY(strExcep.LoadString(g_hDll, IDS_IMP_UNKNOWN_ERROR));
}
CLString strResId;
if (NULL != pItem)
{
CPascalString pasResId;
pItem->GetUniqueId().GetResId().GetDisplayableId(pasResId);
pasResId.ConvertToCLString(strResId, CP_ACP);
}
CLString strMsg;
strMsg.Format(g_hDll, IDS_ERR_EXCEPTION, (LPCTSTR)strResId,
(LPCTSTR)strExcep);
CContext ctx(strContext, pItem->GetMyDatabaseId(), otResource, vProjWindow);
pReporter->IssueMessage(esError, ctx, strMsg);
}
catch(CException* pE)
{
LTASSERT(0 && _T("Could not issue a exception message"));
pE->Delete();
}
catch(...)
{
LTASSERT(0 && _T("Could not issue a exception message"));
}
}
////////////////////////////////////////////////////////////////////////////////
// CItemSetException
//
IMPLEMENT_DYNAMIC(CItemSetException, CException)
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//
// Default contructor
//
//------------------------------------------------------------------------------
CItemSetException::CItemSetException()
{
}
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//
// Constructor
//
//------------------------------------------------------------------------------
CItemSetException::CItemSetException(BOOL bAutoDelete)
:CException(bAutoDelete)
{
}
//++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//
// Fill passed buffer with a error message for this exception.
// The message is cached and only retrieved 1 time.
//
//------------------------------------------------------------------------------
BOOL
CItemSetException::GetErrorMessage(LPTSTR lpszError, UINT nMaxError,
PUINT pnHelpContext)
{
LTASSERT(lpszError != NULL && AfxIsValidString(lpszError, nMaxError));
if (NULL != pnHelpContext)
{
*pnHelpContext = 0; //unused
}
if (m_strMsg.IsEmpty())
{
LTVERIFY(m_strMsg.LoadString(g_hDll, IDS_EXCEP_ITEMSET));
}
int nMax = min(nMaxError, (UINT)m_strMsg.GetLength() + 1);
_tcsncpy(lpszError, m_strMsg, nMax - 1);
lpszError[nMax] = _T('\0');
return TRUE;
}