mirror of https://github.com/tongzx/nt5src
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.
1318 lines
45 KiB
1318 lines
45 KiB
////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Microsoft WMI OLE DB Provider
|
|
// (C) Copyright 1999 Microsoft Corporation. All Rights Reserved.
|
|
//
|
|
// Error Routines
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
#include "headers.h"
|
|
|
|
#define ERROR_GUID_ARRAYSIZE 5
|
|
// NTRaid : 111781
|
|
// 06/13/00
|
|
#define ERROR_DESCRIPTION_SIZE 512
|
|
|
|
static const ULONG INITIAL_ERROR_QUEUE_SIZE = 10;
|
|
|
|
|
|
CImpISupportErrorInfo::CImpISupportErrorInfo( IUnknown* pUnkOuter)
|
|
{
|
|
m_cRef = 0L;
|
|
m_pUnkOuter = pUnkOuter;
|
|
m_cpErrInt = 0;
|
|
m_rgpErrInt = NULL;
|
|
m_cAllocGuid = 0;
|
|
}
|
|
|
|
CImpISupportErrorInfo::~CImpISupportErrorInfo()
|
|
{
|
|
SAFE_DELETE_ARRAY(m_rgpErrInt);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP_(ULONG) CImpISupportErrorInfo::AddRef(void)
|
|
{
|
|
InterlockedIncrement((long*)&m_cRef);
|
|
return m_pUnkOuter->AddRef();
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP_(ULONG) CImpISupportErrorInfo::Release(void)
|
|
{
|
|
InterlockedDecrement((long*)&m_cRef);
|
|
return m_pUnkOuter->Release();
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP CImpISupportErrorInfo::QueryInterface( REFIID riid, LPVOID * ppv )
|
|
{
|
|
return m_pUnkOuter->QueryInterface(riid, ppv);
|
|
}
|
|
|
|
// Modifications done for
|
|
// NTRaid:111765
|
|
HRESULT CImpISupportErrorInfo::AddInterfaceID(REFIID riid)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if(m_rgpErrInt == NULL)
|
|
{
|
|
m_rgpErrInt = new GUID*[ERROR_GUID_ARRAYSIZE];
|
|
|
|
if(m_rgpErrInt)
|
|
{
|
|
m_cAllocGuid = ERROR_GUID_ARRAYSIZE;
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
else
|
|
if(m_cpErrInt >= m_cAllocGuid)
|
|
{
|
|
GUID **pTemp = m_rgpErrInt;
|
|
m_rgpErrInt = NULL;
|
|
|
|
m_rgpErrInt = new GUID*[m_cAllocGuid + ERROR_GUID_ARRAYSIZE];
|
|
if(m_rgpErrInt)
|
|
{
|
|
memset(m_rgpErrInt ,0, (m_cAllocGuid + ERROR_GUID_ARRAYSIZE) * sizeof(GUID*));
|
|
memcpy(m_rgpErrInt ,pTemp, m_cAllocGuid * sizeof(GUID*));
|
|
m_cAllocGuid += ERROR_GUID_ARRAYSIZE;
|
|
SAFE_DELETE_ARRAY(pTemp);
|
|
}
|
|
else
|
|
{
|
|
m_rgpErrInt = pTemp;
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
|
|
}
|
|
|
|
if(m_rgpErrInt && SUCCEEDED(hr))
|
|
{
|
|
m_rgpErrInt[m_cpErrInt++] = (GUID *)&riid;
|
|
hr = S_OK;
|
|
}
|
|
|
|
|
|
return hr;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Indicates whether a OLE DB Interface can return OLE DB error objects
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP CImpISupportErrorInfo::InterfaceSupportsErrorInfo( REFIID riid )
|
|
{
|
|
ULONG ul;
|
|
HRESULT hr = S_FALSE;
|
|
|
|
//==========================================================================
|
|
// See if the interface asked about, actually creates an error object.
|
|
//==========================================================================
|
|
for(ul=0; ul<m_cpErrInt; ul++){
|
|
|
|
if ( (*(m_rgpErrInt[ul])) == riid )
|
|
{
|
|
hr = S_OK;
|
|
break;
|
|
}
|
|
}
|
|
return hr;
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Constructor for this class
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
#pragma warning (disable:4355)
|
|
CErrorLookup::CErrorLookup( LPUNKNOWN pUnkOuter ) : m_IErrorLookup(this), CBaseObj(BOT_ERROR,pUnkOuter)
|
|
{
|
|
}
|
|
#pragma warning (default:4355)
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Returns a pointer to a specified interface. Callers use QueryInterface to determine which interfaces
|
|
// the called object supports.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP CErrorLookup::QueryInterface( REFIID riid, LPVOID * ppv)
|
|
{
|
|
if ( ppv == NULL )
|
|
return E_INVALIDARG;
|
|
//=========================================================
|
|
// This is the non-delegating IUnknown implementation
|
|
//=========================================================
|
|
if ( riid == IID_IUnknown )
|
|
*ppv = (LPVOID)this;
|
|
else if ( riid == IID_IErrorLookup )
|
|
*ppv = (LPVOID)&m_IErrorLookup;
|
|
else{
|
|
*ppv = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
((LPUNKNOWN)*ppv)->AddRef();
|
|
return S_OK;
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Increments a persistence count for the object
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP_(ULONG) CErrorLookup::AddRef(void)
|
|
{
|
|
ULONG cRef = InterlockedIncrement( (long*) &m_cRef);
|
|
return cRef;
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Decrements a persistence count for the object and if persistence count is 0, the object destroys
|
|
// itself.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP_(ULONG) CErrorLookup::Release (void)
|
|
{
|
|
ULONG cRef = InterlockedDecrement( (long*) &m_cRef);
|
|
if ( !cRef ){
|
|
delete this;
|
|
return 0;
|
|
}
|
|
return cRef;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Increments a reference count for the object.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP_(ULONG) CImpIErrorLookup::AddRef(void)
|
|
{
|
|
InterlockedIncrement((long*)&m_cRef);
|
|
return m_pCErrorLookup->AddRef();
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Decrement the object's reference count and deletes the object when the new reference count is zero.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP_(ULONG) CImpIErrorLookup::Release(void)
|
|
{
|
|
InterlockedDecrement((long*)&m_cRef);
|
|
return m_pCErrorLookup->Release();
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Returns a pointer to a specified interface. Callers use QueryInterface to determine which interfaces
|
|
// the called object supports.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP CImpIErrorLookup::QueryInterface( REFIID riid, LPVOID * ppv)
|
|
{
|
|
return m_pCErrorLookup->QueryInterface(riid, ppv);
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Retrieve the error message and source based on the HRESULT and the provider specific error number
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP CImpIErrorLookup::GetErrorDescription( HRESULT hrError,
|
|
DWORD dwLookupId,
|
|
DISPPARAMS* pdispparams,
|
|
LCID lcid,
|
|
BSTR* ppwszSource,
|
|
BSTR* ppwszDescription )
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
int cchBuffer;
|
|
// NTRaid : 111781
|
|
// 06/13/00
|
|
WCHAR wszBuffer[ERROR_DESCRIPTION_SIZE];
|
|
|
|
//=========================================================
|
|
// Check the Arguments
|
|
//=========================================================
|
|
if ( !ppwszSource || !ppwszDescription )
|
|
{
|
|
hr = E_INVALIDARG;
|
|
}
|
|
else
|
|
{
|
|
//=========================================================
|
|
// Initialize return values
|
|
//=========================================================
|
|
*ppwszSource = NULL;
|
|
*ppwszDescription = NULL;
|
|
|
|
// Check the Locale
|
|
// if ( lcid != GetSystemDefaultLCID() )
|
|
// return DB_E_NOLOCALE;
|
|
|
|
wcscpy(wszBuffer,L"");
|
|
WMIOledb_LoadStringW(IDS_WMIOLEDBDES,wszBuffer,ERROR_DESCRIPTION_SIZE);
|
|
//=========================================================
|
|
// Store source name
|
|
//=========================================================
|
|
*ppwszSource = Wmioledb_SysAllocString(wszBuffer);
|
|
if ( *ppwszSource == NULL )
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
else
|
|
//=========================================================
|
|
// If the lookup id was to be looked up in the resource dll
|
|
// of the error collection, make sure we just return NULL
|
|
// for the error description.
|
|
//=========================================================
|
|
if ( (dwLookupId &= ~IDENTIFIER_SDK_MASK) == 0 )
|
|
{
|
|
hr = S_OK;
|
|
}
|
|
else
|
|
//=========================================================
|
|
// After this point make sure to exit to goto
|
|
// the FAILED(hr) code or drop through to the return hr,
|
|
// as to make sure memory is freed if need be.
|
|
// Determine if it is a static or dynamic string
|
|
//=========================================================
|
|
if ( dwLookupId & ERR_STATIC_STRING )
|
|
{
|
|
wcscpy(wszBuffer,L"");
|
|
// NTRaid : 111781
|
|
// 06/13/00
|
|
// cchBuffer = LoadStringW(g_hInstance, (UINT)(dwLookupId & ~(ERR_STATIC_STRING)), wszBuffer,ERROR_DESCRIPTION_SIZE);
|
|
cchBuffer = WMIOledb_LoadStringW( (UINT)(dwLookupId & ~(ERR_STATIC_STRING)), wszBuffer,ERROR_DESCRIPTION_SIZE);
|
|
|
|
//=====================================================
|
|
// Convert to a BSTR
|
|
//=====================================================
|
|
if (cchBuffer)
|
|
{
|
|
*ppwszDescription = Wmioledb_SysAllocString(wszBuffer);
|
|
if ( *ppwszDescription == NULL )
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = g_pCError->GetErrorDescription(dwLookupId, ppwszDescription);
|
|
}
|
|
|
|
if ( FAILED(hr) )
|
|
{
|
|
|
|
//=====================================================
|
|
// If allocation done, make sure that it is freed
|
|
//=====================================================
|
|
if ( ppwszSource )
|
|
{
|
|
SysFreeString(*ppwszSource);
|
|
*ppwszSource = NULL;
|
|
}
|
|
if ( ppwszDescription )
|
|
{
|
|
SysFreeString(*ppwszDescription);
|
|
*ppwszDescription = NULL;
|
|
}
|
|
}
|
|
|
|
} // else for if(check valid parameters)
|
|
|
|
return hr;
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Retrieve the error message and source based on the HRESULT and the provider specific error number
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP CImpIErrorLookup::GetHelpInfo( HRESULT hrError,DWORD dwLookupId, LCID lcid,
|
|
BSTR* ppwszHelpFile, DWORD* pdwHelpContext )
|
|
{
|
|
HRESULT hr = S_OK;
|
|
//=====================================================
|
|
// Check the Arguments
|
|
//=====================================================
|
|
if ( !ppwszHelpFile || !pdwHelpContext )
|
|
{
|
|
hr = E_INVALIDARG;
|
|
}
|
|
else
|
|
{
|
|
|
|
//=====================================================
|
|
// Initialize return values
|
|
//=====================================================
|
|
*ppwszHelpFile = NULL;
|
|
*pdwHelpContext = 0;
|
|
|
|
//=====================================================
|
|
// Check the Locale
|
|
//=====================================================
|
|
if ( lcid != GetSystemDefaultLCID() )
|
|
{
|
|
hr = DB_E_NOLOCALE;
|
|
}
|
|
}
|
|
|
|
//=====================================================
|
|
// We currently can't return any help file context or
|
|
// names. So, we will just return S_OK.
|
|
//=====================================================
|
|
return hr;
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Method to be called to release non static error messages.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP CImpIErrorLookup::ReleaseErrors( const DWORD dwDynamicErrorID )
|
|
{
|
|
HRESULT hr = S_OK;
|
|
//=====================================================
|
|
// Check the Arguments
|
|
//=====================================================
|
|
if ( dwDynamicErrorID == 0 )
|
|
{
|
|
hr = E_INVALIDARG;
|
|
}
|
|
else
|
|
{
|
|
g_pCError->RemoveErrors(dwDynamicErrorID);
|
|
}
|
|
return hr;
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Constructor for this class
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
#pragma warning (disable:4355)
|
|
CImpIWMIErrorInfo::CImpIWMIErrorInfo( PERRORSTUFF pErrStuff )
|
|
{
|
|
|
|
//=====================================================
|
|
// Initialize simple member vars
|
|
//=====================================================
|
|
m_cRef = 0L;
|
|
m_hObjCollection = ULONG(-1);
|
|
|
|
//=====================================================
|
|
// Store values
|
|
//=====================================================
|
|
m_pErrStuff = pErrStuff;
|
|
|
|
//=====================================================
|
|
// Since the memory is created and owned by WMIoledb,
|
|
// we need to make sure that this DLL is not unloaded,
|
|
// else we will clean up the handed out pointer
|
|
//=====================================================
|
|
InterlockedIncrement(&g_cObj);
|
|
}
|
|
#pragma warning (default:4355)
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Destructor for this class
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
CImpIWMIErrorInfo::~CImpIWMIErrorInfo (void)
|
|
{
|
|
//=====================================================
|
|
// For Abnormal Termination,Remove self from Collection
|
|
//=====================================================
|
|
g_pCError->RemoveFromCollection(m_hObjCollection);
|
|
|
|
//=====================================================
|
|
// If this object has been released, we can now
|
|
// decrement our object count
|
|
//=====================================================
|
|
InterlockedDecrement(&g_cObj);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Returns a pointer to a specified interface. Callers use QueryInterface to determine which interfaces
|
|
// the called object supports.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP CImpIWMIErrorInfo::QueryInterface( REFIID riid, LPVOID * ppv)
|
|
{
|
|
HRESULT hr = E_NOINTERFACE;
|
|
|
|
if ( ppv == NULL ){
|
|
hr = E_INVALIDARG;
|
|
}
|
|
else
|
|
//=====================================================
|
|
// This is the non-delegating IUnknown implementation
|
|
//=====================================================
|
|
if ( (riid == IID_IUnknown) || (riid == IID_IErrorInfo) ){
|
|
*ppv = (LPVOID)this;
|
|
}
|
|
else{
|
|
*ppv = NULL;
|
|
}
|
|
|
|
//=====================================================
|
|
// If we're going to return an interface, AddRef first
|
|
//=====================================================
|
|
if (*ppv){
|
|
((LPUNKNOWN)*ppv)->AddRef();
|
|
hr = S_OK;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Increments a persistence count for the object
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP_(ULONG) CImpIWMIErrorInfo::AddRef(void)
|
|
{
|
|
ULONG cRef = InterlockedIncrement( (long*) &m_cRef);
|
|
return cRef;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Decrements a persistence count for the object and if persistence count is 0,object destroys itself.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP_(ULONG) CImpIWMIErrorInfo::Release (void)
|
|
{
|
|
ULONG cRef = InterlockedDecrement( (long*) &m_cRef);
|
|
if ( !cRef ){
|
|
delete this;
|
|
return 0;
|
|
}
|
|
return cRef;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Return the WMIstate associated with this custom error object.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
STDMETHODIMP CImpIWMIErrorInfo::GetWMIInfo( BSTR* pbstrMsg, LONG* plNativeError )
|
|
{
|
|
HRESULT hr = S_OK;
|
|
//=====================================================
|
|
// Check arguments
|
|
//=====================================================
|
|
if ( !pbstrMsg || !plNativeError)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
}
|
|
else
|
|
{
|
|
|
|
//=====================================================
|
|
// Initialize return values
|
|
//=====================================================
|
|
*pbstrMsg = Wmioledb_SysAllocString(NULL);
|
|
*plNativeError = 0;
|
|
|
|
//=====================================================
|
|
// Handle WMIState
|
|
//=====================================================
|
|
|
|
if ( wcslen(m_pErrStuff->pwszMessage) > 0 )
|
|
{
|
|
//=================================================
|
|
// If string is not NULL,then we can allocate it
|
|
//=================================================
|
|
*pbstrMsg = Wmioledb_SysAllocString(m_pErrStuff->pwszMessage);
|
|
if ( *pbstrMsg == NULL )
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
|
|
//=====================================================
|
|
// Handle Native Error Code.
|
|
//=====================================================
|
|
*plNativeError = m_pErrStuff->lNative;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
CError::CError( )
|
|
{
|
|
//=====================================================
|
|
// Note that this is NOT a first-class object: it is
|
|
// NOT given away outside this DLL. So we do not want
|
|
// to alter global object count.
|
|
//=====================================================
|
|
|
|
// Clear variables
|
|
m_pWMIErrorInfoCollection = NULL;
|
|
m_prgErrorDex = NULL;
|
|
m_cErrorsUsed = 0;
|
|
m_dwId = 0;
|
|
m_ulNext = 1;
|
|
|
|
//=====================================================
|
|
// Initialize ErrorInfo with constant information
|
|
//=====================================================
|
|
m_ErrorInfo.clsid = CLSID_WMIOLEDB;
|
|
m_ErrorInfo.dispid = NULL;
|
|
|
|
m_pcsErrors = new CCriticalSection(TRUE);
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
CError::~CError()
|
|
{
|
|
ULONG iElems;
|
|
ULONG cElems;
|
|
ULONG ul;
|
|
PIMPIWMIERRORINFO pIWMIErrorInfo;
|
|
|
|
m_pcsErrors->Enter();
|
|
|
|
//=====================================================
|
|
// Release the general Object
|
|
//=====================================================
|
|
if ( m_pWMIErrorInfoCollection ){
|
|
|
|
cElems = m_pWMIErrorInfoCollection->Size();
|
|
|
|
//=================================================
|
|
// Loop through stored object handles and release.
|
|
//=================================================
|
|
for(iElems=0; iElems<cElems; iElems++){
|
|
|
|
pIWMIErrorInfo = (PIMPIWMIERRORINFO )m_pWMIErrorInfoCollection->GetAt(iElems);
|
|
if ( pIWMIErrorInfo != NULL )
|
|
{
|
|
delete pIWMIErrorInfo;
|
|
}
|
|
}
|
|
|
|
delete m_pWMIErrorInfoCollection;
|
|
}
|
|
|
|
//=====================================================
|
|
// Cleanup and error information hanging off the
|
|
// pointer array
|
|
//=====================================================
|
|
if ( m_prgErrorDex ){
|
|
|
|
for(ul=0; ul<m_cErrors; ul++){
|
|
|
|
if ( m_prgErrorDex[ul] ){
|
|
|
|
delete[] m_prgErrorDex[ul];
|
|
m_prgErrorDex[ul] = NULL;
|
|
}
|
|
}
|
|
delete[] m_prgErrorDex;
|
|
}
|
|
m_pcsErrors->Leave();
|
|
|
|
|
|
SAFE_DELETE_PTR(m_pcsErrors);
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// If this initialization routine fails, it is the callers responsibility to delete this object.
|
|
// The destructor will then clean up any allocated resources
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
HRESULT CError::FInit()
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
//=============================================================
|
|
// Create array of pointers into the heap
|
|
//=============================================================
|
|
m_prgErrorDex = new PERRORSTUFF[INITIAL_SIZE_FOR_ERRORSTUFF];
|
|
|
|
if ( NULL == m_prgErrorDex )
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
else
|
|
{
|
|
m_cErrors = INITIAL_SIZE_FOR_ERRORSTUFF;
|
|
memset(m_prgErrorDex, 0, INITIAL_SIZE_FOR_ERRORSTUFF * sizeof(PERRORSTUFF));
|
|
|
|
//=============================================================
|
|
// Create WMI Error Object Collection
|
|
//=============================================================
|
|
m_pWMIErrorInfoCollection = new CFlexArray;
|
|
|
|
if ( NULL == m_pWMIErrorInfoCollection )
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
|
|
}
|
|
return hr;
|
|
}
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Remove the Error Information for a particular DynamicId. This could be one or more elements of the
|
|
// ERRORSTUFF array and their associated pwszMessages on the Heap.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
void CError::RemoveErrors( DWORD dwDynamicId )
|
|
{
|
|
ULONG ul;
|
|
|
|
//=============================================================
|
|
// At the creation of the CAutoBlock object a critical section
|
|
// is entered. This is because the method manipulates shared
|
|
// structures access to which must be serialized .
|
|
// The criticalsection is left when this method terminate
|
|
// and the destructor for CAutoBlock is called.
|
|
//=============================================================
|
|
CAutoBlock Crit(m_pcsErrors);
|
|
|
|
for(ul=0; ul<m_cErrors; ul++){
|
|
|
|
if ( m_prgErrorDex[ul] && m_prgErrorDex[ul]->dwDynamicId == dwDynamicId ){
|
|
|
|
delete[] m_prgErrorDex[ul];
|
|
m_prgErrorDex[ul] = NULL;
|
|
m_cErrorsUsed--;
|
|
|
|
//=====================================================
|
|
// For each error description that we cache, we have
|
|
// up'd the object count to make sure that we stay
|
|
// alive while there is a possibility the error
|
|
// collection could call back into us for this desc.
|
|
// Here is where we decrement that count upon
|
|
// release of the error message.
|
|
//=====================================================
|
|
InterlockedDecrement(&g_cObj);
|
|
}
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Retrieve the WMI Server Error Message from the cache
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////////////////////
|
|
HRESULT CError::GetErrorDescription(ULONG ulDex, BSTR* ppwszDescription )
|
|
{
|
|
|
|
HRESULT hr = S_OK;
|
|
|
|
//======================================================
|
|
// Enter critical section
|
|
//======================================================
|
|
CAutoBlock Crit(m_pcsErrors);
|
|
|
|
//======================================================
|
|
//
|
|
//======================================================
|
|
if (ulDex < m_cErrors && m_prgErrorDex[ulDex]){
|
|
|
|
PERRORSTUFF pErrorStuff = m_prgErrorDex[ulDex];
|
|
if (wcslen(pErrorStuff->pwszMessage) > 0 ){
|
|
|
|
//======================================================
|
|
// return the message from WMI
|
|
//======================================================
|
|
*ppwszDescription = Wmioledb_SysAllocString(pErrorStuff->pwszMessage);
|
|
if (*ppwszDescription == NULL){
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
else{
|
|
|
|
//======================================================
|
|
// return the standard error
|
|
//======================================================
|
|
WCHAR rgwchBuff[MAX_PATH+1];
|
|
WCHAR *pwsz = rgwchBuff;
|
|
int cwch = LoadStringW(g_hInstance, pErrorStuff->uStringId, rgwchBuff, NUMELEM(rgwchBuff));
|
|
*ppwszDescription = Wmioledb_SysAllocString(pwsz);
|
|
if (*ppwszDescription == NULL){
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
}
|
|
else{
|
|
*ppwszDescription = NULL;
|
|
hr = DB_E_BADLOOKUPID;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Retrieve the IErrorInfo and IErrorRecords pointers whether from automation or from creating a new
|
|
// instance
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
HRESULT CError::GetErrorInterfaces( IErrorInfo** ppIErrorInfo, IErrorRecords** ppIErrorRecords )
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
*ppIErrorInfo = NULL;
|
|
*ppIErrorRecords = NULL;
|
|
//=============================================================
|
|
//Obtain the error object or create a new one if none exists
|
|
//=============================================================
|
|
// NTRaid:111806
|
|
// 06/07/00
|
|
hr = GetErrorInfo(0, ppIErrorInfo);
|
|
if (SUCCEEDED(hr) && !*ppIErrorInfo){
|
|
|
|
hr = g_pErrClassFact->CreateInstance(NULL, IID_IErrorInfo, (LPVOID*)ppIErrorInfo);
|
|
}
|
|
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
//=============================================================
|
|
// Obtain the IErrorRecord Interface
|
|
//=============================================================
|
|
hr = (*ppIErrorInfo)->QueryInterface(IID_IErrorRecords, (LPVOID*)ppIErrorRecords);
|
|
|
|
//=============================================================
|
|
// On a failure retrieving IErrorRecords, we need to release
|
|
// the IErrorInfo interface
|
|
//=============================================================
|
|
if ( FAILED(hr) ){
|
|
(*ppIErrorInfo)->Release();
|
|
*ppIErrorInfo = NULL;
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Determine the next free index of the PERRORSTUFF array.
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
HRESULT CError::FindFreeDex( ULONG* ulDex )
|
|
{
|
|
PERRORSTUFF* pNew = NULL;
|
|
HRESULT hr = S_OK;
|
|
|
|
//===========================================================================
|
|
// Check if we need to expand the buffer
|
|
// Since we do not use element 0, we need to reallocate when at m_cErrors - 1
|
|
//===========================================================================
|
|
if ( m_cErrorsUsed == (m_cErrors - 1))
|
|
{
|
|
try
|
|
{
|
|
pNew = new PERRORSTUFF[m_cErrors + INCREMENT_BY_FOR_ERRORSTUFF];
|
|
}
|
|
catch(...)
|
|
{
|
|
SAFE_DELETE_PTR(pNew);
|
|
throw;
|
|
}
|
|
if ( pNew )
|
|
{
|
|
|
|
//===================================================================
|
|
// Copy old pointers to new array
|
|
//===================================================================
|
|
memcpy(pNew, m_prgErrorDex, m_cErrors * sizeof(PERRORSTUFF));
|
|
memset((pNew + m_cErrors), 0, INCREMENT_BY_FOR_ERRORSTUFF * sizeof(PERRORSTUFF));
|
|
|
|
//===================================================================
|
|
// Set the new array size
|
|
//===================================================================
|
|
m_ulNext = m_cErrors;
|
|
m_cErrors += INCREMENT_BY_FOR_ERRORSTUFF;
|
|
delete[] m_prgErrorDex;
|
|
m_prgErrorDex = pNew;
|
|
}
|
|
else
|
|
{
|
|
*ulDex = 0;
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
//===========================================================================
|
|
// Loop through looking for unused index
|
|
//===========================================================================
|
|
while( 1 ){
|
|
|
|
//=======================================================================
|
|
// If we are at the top of our buffer rap back to the begining
|
|
//=======================================================================
|
|
if (m_ulNext == m_cErrors)
|
|
{
|
|
m_ulNext = 1;
|
|
}
|
|
else if (NULL == m_prgErrorDex[m_ulNext])
|
|
{
|
|
m_cErrorsUsed++;
|
|
*ulDex = m_ulNext++;
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
m_ulNext++;
|
|
}
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Creates error records for the WMI errors encountered. The WMIState and Native error code are
|
|
// stored in a custom interface, where as the description is returned by the standard IErrorInfo
|
|
// interface.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
HRESULT CError::PostWMIErrors( HRESULT hrErr, const IID* piid, CErrorData* pErrData )
|
|
{
|
|
|
|
HRESULT hr = S_OK;
|
|
IErrorInfo* pIErrorInfo = NULL;
|
|
IErrorRecords* pIErrorRecords = NULL;
|
|
PIMPIWMIERRORINFO pWMIError = NULL;
|
|
ULONG iElems;
|
|
ULONG cElems;
|
|
PERRORSTUFF pErrStuff;
|
|
DWORD dwDex;
|
|
|
|
//===========================================================================
|
|
// Enter critical section
|
|
//===========================================================================
|
|
CAutoBlock Crit(m_pcsErrors);
|
|
|
|
//===========================================================================
|
|
// We cannot Init the Datasource object without this pointer being set, so it
|
|
// should exist at this point
|
|
// Get the # errors
|
|
//===========================================================================
|
|
cElems = pErrData->Size();
|
|
if (0 == cElems){
|
|
if (FAILED(hrErr))
|
|
g_pCError->PostHResult(hrErr, piid);
|
|
hr = S_OK;
|
|
}
|
|
else{
|
|
|
|
//=======================================================================
|
|
// Obtain the error object or create a new one if none exists
|
|
//=======================================================================
|
|
if ( SUCCEEDED(hr = GetErrorInterfaces(&pIErrorInfo, &pIErrorRecords)) ){
|
|
|
|
//===================================================================
|
|
// Assign static information across each error record added
|
|
//===================================================================
|
|
m_ErrorInfo.hrError = hrErr;
|
|
m_ErrorInfo.iid = *piid;
|
|
m_ErrorInfo.dispid = NULL;
|
|
|
|
//===================================================================
|
|
// Increment Dynamic Id;
|
|
//===================================================================
|
|
m_dwId++;
|
|
|
|
//===================================================================
|
|
// Loop through an array of indexes into the error array
|
|
//===================================================================
|
|
for (iElems=0; iElems<cElems; iElems++){
|
|
|
|
//===============================================================
|
|
// Get the error stuff
|
|
//===============================================================
|
|
pErrData->SetAt(iElems,(void**)&pErrStuff);
|
|
|
|
//===============================================================
|
|
// Save the dynamic id
|
|
//===============================================================
|
|
pErrStuff->dwDynamicId = m_dwId;
|
|
|
|
//===============================================================
|
|
// Save WMI Server error code
|
|
//===============================================================
|
|
m_ErrorInfo.dwMinor = DWORD(pErrStuff->lNative);
|
|
|
|
//===============================================================
|
|
// Determine index to store heap pointer
|
|
//===============================================================
|
|
if ( SUCCEEDED(hr = FindFreeDex(&dwDex)) ){
|
|
|
|
try
|
|
{
|
|
//===========================================================
|
|
// Create the custom error object
|
|
//===========================================================
|
|
pWMIError = new CImpIWMIErrorInfo(pErrStuff);
|
|
}
|
|
catch(...)
|
|
{
|
|
SAFE_DELETE_PTR(pWMIError);
|
|
throw;
|
|
}
|
|
if ( pWMIError == NULL ){
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
else{
|
|
pWMIError->AddRef();
|
|
if ( SUCCEEDED(hr = pWMIError->FInit()) ){
|
|
|
|
//===================================================
|
|
// Add the record to the Error Service Object
|
|
//===================================================
|
|
hr = pIErrorRecords->AddErrorRecord(&m_ErrorInfo, dwDex, NULL,(IUnknown*) pWMIError, m_dwId);
|
|
if ( SUCCEEDED(hr) ){
|
|
|
|
//===============================================
|
|
// Release the custom Error Object
|
|
//===============================================
|
|
pWMIError->Release();
|
|
pWMIError = NULL;
|
|
|
|
//===============================================
|
|
// Save the pointer to the error stuff
|
|
//===============================================
|
|
m_prgErrorDex[dwDex] = pErrStuff;
|
|
|
|
//===============================================
|
|
// For each error description that we cache, we
|
|
// have up'd the the object count to make sure
|
|
// that we stay alive while there is a
|
|
// possibility the errorcollection could call back
|
|
// into us for this description. Here is where we
|
|
// increment that count.
|
|
//===============================================
|
|
InterlockedIncrement(&g_cObj);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//================================================================
|
|
// Pass the error object to the Ole Automation DLL
|
|
//================================================================
|
|
hr = SetErrorInfo(0, pIErrorInfo);
|
|
}
|
|
}
|
|
}
|
|
|
|
//============================================================================
|
|
// Release the interfaces to transfer ownership to the Ole Automation DLL
|
|
//============================================================================
|
|
SAFE_RELEASE_PTR(pWMIError);
|
|
SAFE_RELEASE_PTR(pIErrorRecords);
|
|
SAFE_RELEASE_PTR(pIErrorInfo);
|
|
|
|
//============================================================================
|
|
// Free the error data
|
|
//============================================================================
|
|
pErrData->FreeErrors();
|
|
|
|
return hr;
|
|
}
|
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// This method is used to post an HRESULT that is to be looked up in the resource fork of the error collection
|
|
// object.
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
HRESULT CError::PostHResult( HRESULT hrErr, const IID* piid )
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IErrorInfo* pIErrorInfo = NULL;
|
|
IErrorRecords* pIErrorRecords = NULL;
|
|
|
|
|
|
//==================================================================
|
|
// Enter critical section
|
|
//==================================================================
|
|
CAutoBlock Crit(m_pcsErrors);
|
|
|
|
//==================================================================
|
|
// Obtain the error object or create a new one if none exists
|
|
//==================================================================
|
|
if ( SUCCEEDED(hr = GetErrorInterfaces(&pIErrorInfo, &pIErrorRecords)) )
|
|
{
|
|
//==================================================================
|
|
// Assign static information across each error record added
|
|
//==================================================================
|
|
m_ErrorInfo.hrError = hrErr;
|
|
m_ErrorInfo.iid = *piid;
|
|
m_ErrorInfo.dispid = NULL;
|
|
m_ErrorInfo.dwMinor = 0;
|
|
|
|
//==================================================================
|
|
// Add the record to the Error Service Object
|
|
//==================================================================
|
|
hr = pIErrorRecords->AddErrorRecord(&m_ErrorInfo,IDENTIFIER_SDK_ERROR,NULL,(IUnknown*)NULL,0);
|
|
if ( SUCCEEDED(hr) )
|
|
{
|
|
//==================================================================
|
|
// Pass the error object to the Ole Automation DLL
|
|
//==================================================================
|
|
hr = SetErrorInfo(0, pIErrorInfo);
|
|
}
|
|
}
|
|
|
|
//==================================================================
|
|
// Release the interfaces to transfer ownership to the Ole Automation DLL
|
|
//==================================================================
|
|
SAFE_RELEASE_PTR(pIErrorRecords);
|
|
SAFE_RELEASE_PTR(pIErrorInfo);
|
|
|
|
return hrErr;
|
|
}
|
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// This method is used to post static strings to the error objects. The static strings are stored in the
|
|
// resource fork, and thus an id needs to be specified.
|
|
//
|
|
// @devnote If the error object is not our implementation of IID_IErrorInfo, we will not be able to load
|
|
// IErrorRecord and add our records.
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
HRESULT CError::PostError( HRESULT hrErr, const IID* piid, DWORD dwIds, DISPPARAMS* pdispparams )
|
|
{
|
|
HRESULT hr = S_OK;
|
|
IErrorInfo* pIErrorInfo = NULL;
|
|
IErrorRecords* pIErrorRecords = NULL;
|
|
|
|
//====================================================================
|
|
// Enter critical section
|
|
//====================================================================
|
|
CAutoBlock Crit(m_pcsErrors);
|
|
|
|
//====================================================================
|
|
// Obtain the error object or create a new one if none exists
|
|
//====================================================================
|
|
if ( SUCCEEDED(hr = GetErrorInterfaces(&pIErrorInfo, &pIErrorRecords)) )
|
|
{
|
|
//====================================================================
|
|
// Assign static information across each error record added
|
|
//====================================================================
|
|
m_ErrorInfo.hrError = hrErr;
|
|
m_ErrorInfo.iid = *piid;
|
|
m_ErrorInfo.dispid = NULL;
|
|
m_ErrorInfo.dwMinor = 0;
|
|
|
|
//====================================================================
|
|
// Add the record to the Error Service Object
|
|
//====================================================================
|
|
hr = pIErrorRecords->AddErrorRecord(&m_ErrorInfo, (dwIds | ERR_STATIC_STRING), pdispparams, NULL, 0);
|
|
if ( SUCCEEDED(hr) )
|
|
{
|
|
//====================================================================
|
|
// Pass the error object to the Ole Automation DLL
|
|
//====================================================================
|
|
hr = SetErrorInfo(0, pIErrorInfo);
|
|
}
|
|
|
|
}
|
|
//====================================================================
|
|
// Release the interfaces to transfer ownership to the Ole Automation DLL
|
|
//====================================================================
|
|
SAFE_RELEASE_PTR(pIErrorRecords);
|
|
SAFE_RELEASE_PTR(pIErrorInfo);
|
|
|
|
return hr;
|
|
}
|
|
////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// This method is used to post static strings to the error objects.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////////////////////////
|
|
HRESULT CError::PostErrorMessage( HRESULT hrErr, const IID* piid, UINT uStringId, LPCWSTR pwszMessage )
|
|
{
|
|
|
|
PERRORSTUFF pErrStuff = NULL;
|
|
ULONG cwch = wcslen(pwszMessage);
|
|
HRESULT hr = S_OK;
|
|
IErrorInfo* pIErrorInfo = NULL;
|
|
IErrorRecords* pIErrorRecords = NULL;
|
|
PIMPIWMIERRORINFO pWMIError = NULL;
|
|
DWORD dwDex;
|
|
|
|
//==========================================================
|
|
// CS because the method manipulates shared structures
|
|
// access to which must be serialized .
|
|
//==========================================================
|
|
CAutoBlock Crit(m_pcsErrors);
|
|
|
|
try
|
|
{
|
|
pErrStuff = (PERRORSTUFF) new BYTE[sizeof(ERRORSTUFF)+(cwch+2)*sizeof(WCHAR)];
|
|
}
|
|
catch(...)
|
|
{
|
|
SAFE_DELETE_ARRAY(pErrStuff);
|
|
throw;
|
|
}
|
|
if (!pErrStuff) {
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
else
|
|
{
|
|
//==========================================================
|
|
// Initialize error stuff
|
|
//==========================================================
|
|
pErrStuff->dwDynamicId = 0;
|
|
pErrStuff->uStringId = uStringId;
|
|
pErrStuff->hr = S_OK;
|
|
pErrStuff->lNative = 0;
|
|
pErrStuff->wLineNumber = 0;
|
|
|
|
AllocateAndCopy(pErrStuff->pwszMessage, (unsigned short*)pwszMessage);
|
|
//==========================================================
|
|
// We cannot Init the Datasource object without this pointer
|
|
// being set, so it should exist at this point
|
|
// Obtain the error object or create new one if none exists
|
|
//==========================================================
|
|
if ( SUCCEEDED(hr = GetErrorInterfaces(&pIErrorInfo, &pIErrorRecords)) )
|
|
{
|
|
//==========================================================
|
|
// Assign static information across each error record added
|
|
//==========================================================
|
|
m_ErrorInfo.hrError = hrErr;
|
|
m_ErrorInfo.iid = *piid;
|
|
m_ErrorInfo.dispid = NULL;
|
|
m_ErrorInfo.dwMinor = 0;
|
|
|
|
//==========================================================
|
|
// Increment Dynamic Id and save it
|
|
//==========================================================
|
|
pErrStuff->dwDynamicId = ++m_dwId;
|
|
|
|
//==========================================================
|
|
// Determine index to store heap pointer
|
|
//==========================================================
|
|
if ( SUCCEEDED(hr = FindFreeDex(&dwDex)) )
|
|
{
|
|
try
|
|
{
|
|
//==========================================================
|
|
// Create the custom error object
|
|
//==========================================================
|
|
pWMIError = new CImpIWMIErrorInfo(pErrStuff);
|
|
}
|
|
catch(...)
|
|
{
|
|
SAFE_DELETE_ARRAY(pErrStuff);
|
|
SAFE_DELETE_PTR(pErrStuff);
|
|
throw;
|
|
}
|
|
|
|
if ( pWMIError == NULL ){
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
else
|
|
{
|
|
//==========================================================
|
|
// Give the object existence
|
|
//==========================================================
|
|
pWMIError->AddRef();
|
|
if ( SUCCEEDED(hr = pWMIError->FInit()) )
|
|
{
|
|
//==========================================================
|
|
// Add the record to the Error Service Object
|
|
//==========================================================
|
|
if(SUCCEEDED(hr = pIErrorRecords->AddErrorRecord(&m_ErrorInfo, dwDex, NULL, (IUnknown*)pWMIError, m_dwId)))
|
|
{
|
|
//==========================================================
|
|
// Release the custom Error Object
|
|
//==========================================================
|
|
pWMIError->Release();
|
|
pWMIError = NULL;
|
|
|
|
|
|
//==========================================================
|
|
// Save the pointer to the error stuff
|
|
//==========================================================
|
|
m_prgErrorDex[dwDex] = pErrStuff;
|
|
|
|
//==========================================================
|
|
// For each error description that we cache, we have up'd the
|
|
// the object count to make sure that we stay alive while there is
|
|
// a possibility the errorcollection could call back into us for
|
|
// this description. Here is where we increment that count.
|
|
//==========================================================
|
|
InterlockedIncrement(&g_cObj);
|
|
|
|
//==========================================================
|
|
// Pass the error object to the Ole Automation DLL
|
|
//==========================================================
|
|
hr = SetErrorInfo(0, pIErrorInfo);
|
|
|
|
} // if Succeeded(AddErrorRecord(())
|
|
|
|
} // if (succeeded(pWMIError->FInit))
|
|
|
|
} // if allocation of pWMIError is successful
|
|
|
|
} // if(succeeded(FindFreeDex))
|
|
|
|
} // if succeeded(GetErrorInterfaces)
|
|
|
|
} // if propert allocation
|
|
//==========================================================
|
|
// Release the interfaces to transfer ownership to
|
|
//==========================================================
|
|
SAFE_DELETE_ARRAY(pErrStuff);
|
|
SAFE_DELETE_PTR(pErrStuff);
|
|
SAFE_RELEASE_PTR(pWMIError);
|
|
SAFE_RELEASE_PTR(pIErrorRecords);
|
|
SAFE_RELEASE_PTR(pIErrorInfo);
|
|
|
|
return hr;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Frees error information and empties the array
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
void CError::FreeErrors()
|
|
{
|
|
ULONG iElems;
|
|
ULONG cElems;
|
|
BYTE* pb;
|
|
|
|
cElems = Size();
|
|
|
|
for (iElems=0; iElems<cElems; iElems++){
|
|
|
|
pb = (BYTE*) m_pWMIErrorInfoCollection->GetAt(iElems);
|
|
if (pb){
|
|
delete[] pb;
|
|
}
|
|
}
|
|
m_pWMIErrorInfoCollection->Empty();
|
|
}
|
|
//**********************************************************************************************************
|
|
//
|
|
//**********************************************************************************************************
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
CErrorData::CErrorData()
|
|
{
|
|
//
|
|
}
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
CErrorData::~CErrorData()
|
|
{
|
|
FreeErrors();
|
|
}
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
void CErrorData::FreeErrors()
|
|
{
|
|
|
|
}
|
|
|