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.
566 lines
16 KiB
566 lines
16 KiB
// Copyright (c) 2000-2001 Microsoft Corporation, All Rights Reserved
|
|
// JobObjectProv.cpp
|
|
|
|
//#define _WIN32_WINNT 0x0500
|
|
|
|
#include "precomp.h"
|
|
//#include <windows.h>
|
|
#include "cominit.h"
|
|
//#include <objbase.h>
|
|
//#include <comdef.h>
|
|
#include <objidl.h>
|
|
#include "CUnknown.h"
|
|
#include <wbemprov.h>
|
|
#include "FRQueryEx.h"
|
|
#include "globals.h"
|
|
|
|
#include "Factory.h"
|
|
#include "helpers.h"
|
|
#include <map>
|
|
#include <vector>
|
|
#include "SmartHandle.h"
|
|
#include <crtdbg.h>
|
|
#include "CVARIANT.h"
|
|
#include "CObjProps.h"
|
|
#include "CJobObjProps.h"
|
|
#include "JobObjectProv.h"
|
|
#include "Helpers.h"
|
|
|
|
|
|
|
|
CJobObjectProv::CJobObjectProv()
|
|
{
|
|
}
|
|
|
|
|
|
CJobObjectProv::~CJobObjectProv()
|
|
{
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
// QueryInterface override to allow for this component's interface(s)
|
|
/*****************************************************************************/
|
|
STDMETHODIMP CJobObjectProv::QueryInterface(const IID& iid, void** ppv)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if(iid == IID_IWbemServices)
|
|
{
|
|
*ppv = static_cast<IWbemServices*>(this);
|
|
reinterpret_cast<IUnknown*>(*ppv)->AddRef();
|
|
}
|
|
else if(iid == IID_IWbemProviderInit)
|
|
{
|
|
*ppv = static_cast<IWbemProviderInit*>(this);
|
|
reinterpret_cast<IUnknown*>(*ppv)->AddRef();
|
|
}
|
|
else
|
|
{
|
|
hr = CUnknown::QueryInterface(iid, ppv);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
// Creation function used by CFactory
|
|
/*****************************************************************************/
|
|
HRESULT CJobObjectProv::CreateInstance(CUnknown** ppNewComponent)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
CUnknown* pUnk = NULL;
|
|
pUnk = new CJobObjectProv;
|
|
if(pUnk != NULL)
|
|
{
|
|
*ppNewComponent = pUnk;
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
return hr ;
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
// IWbemProviderInit implementation
|
|
/*****************************************************************************/
|
|
STDMETHODIMP CJobObjectProv::Initialize(
|
|
LPWSTR pszUser,
|
|
LONG lFlags,
|
|
LPWSTR pszNamespace,
|
|
LPWSTR pszLocale,
|
|
IWbemServices *pNamespace,
|
|
IWbemContext *pCtx,
|
|
IWbemProviderInitSink *pInitSink)
|
|
{
|
|
HRESULT hr = WBEM_S_NO_ERROR;
|
|
|
|
if(pNamespace)
|
|
{ // smartptr does the addref
|
|
m_pNamespace = pNamespace;
|
|
}
|
|
m_chstrNamespace = pszNamespace;
|
|
|
|
//Let CIMOM know I am initialized...
|
|
return pInitSink->SetStatus(
|
|
WBEM_S_INITIALIZED,
|
|
0);
|
|
}
|
|
|
|
|
|
/*****************************************************************************/
|
|
// IWbemServices implementation
|
|
/*****************************************************************************/
|
|
STDMETHODIMP CJobObjectProv::GetObjectAsync(
|
|
const BSTR ObjectPath,
|
|
long lFlags,
|
|
IWbemContext __RPC_FAR *pCtx,
|
|
IWbemObjectSink __RPC_FAR *pResponseHandler)
|
|
{
|
|
HRESULT hr = WBEM_E_NOT_FOUND;
|
|
IWbemClassObjectPtr pStatusObject;
|
|
|
|
try
|
|
{
|
|
HRESULT hrImp = CheckImpersonationLevel();
|
|
|
|
if(SUCCEEDED(hrImp))
|
|
{
|
|
// We need the name of the instance they requested...
|
|
WCHAR wstrObjInstKeyVal[MAX_PATH];
|
|
hr = GetObjInstKeyVal(
|
|
ObjectPath,
|
|
IDS_Win32_NamedJobObject,
|
|
g_rgJobObjPropNames[JO_ID],
|
|
wstrObjInstKeyVal,
|
|
sizeof(wstrObjInstKeyVal) - sizeof(WCHAR));
|
|
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
// wstrObjInstKeyVal now contains the name of the object. See if
|
|
// it exists...
|
|
CHString chstrUndecoratedJOName;
|
|
|
|
UndecorateJOName(
|
|
wstrObjInstKeyVal,
|
|
chstrUndecoratedJOName);
|
|
|
|
SmartHandle hJob;
|
|
hJob = ::OpenJobObjectW(
|
|
MAXIMUM_ALLOWED,
|
|
FALSE,
|
|
chstrUndecoratedJOName);
|
|
|
|
if(hJob)
|
|
{
|
|
// We seem to have found one matching the specified name,
|
|
// so create a return instance...
|
|
IWbemClassObjectPtr pIWCO = NULL;
|
|
CJobObjProps cjop(hJob, m_chstrNamespace);
|
|
|
|
hr = CreateInst(
|
|
m_pNamespace,
|
|
&pIWCO,
|
|
_bstr_t(IDS_Win32_NamedJobObject),
|
|
pCtx);
|
|
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
cjop.SetReqProps(PROP_ALL_REQUIRED);
|
|
}
|
|
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
// set the key properties...
|
|
hr = cjop.SetKeysFromPath(
|
|
ObjectPath,
|
|
pCtx);
|
|
}
|
|
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
// set the non-key requested properties...
|
|
hr = cjop.SetNonKeyReqProps();
|
|
}
|
|
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
// Load requested non-key properties
|
|
// to the instance...
|
|
hr = cjop.LoadPropertyValues(
|
|
pIWCO);
|
|
|
|
// Commit the instance...
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
IWbemClassObject *pTmp = (IWbemClassObject*) pIWCO;
|
|
hr = pResponseHandler->Indicate(
|
|
1,
|
|
&pTmp);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = WBEM_E_NOT_FOUND;
|
|
|
|
SetStatusObject(
|
|
pCtx,
|
|
m_pNamespace,
|
|
::GetLastError(),
|
|
NULL,
|
|
L"::OpenJobObject",
|
|
JOB_OBJECT_NAMESPACE,
|
|
&pStatusObject);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = hrImp;
|
|
}
|
|
}
|
|
catch(CVARIANTError& cve)
|
|
{
|
|
hr = cve.GetWBEMError();
|
|
}
|
|
catch(...)
|
|
{
|
|
hr = WBEM_E_PROVIDER_FAILURE;
|
|
}
|
|
|
|
// Set Status
|
|
return pResponseHandler->SetStatus(0, hr, NULL, pStatusObject);
|
|
}
|
|
|
|
|
|
STDMETHODIMP CJobObjectProv::ExecQueryAsync(
|
|
const BSTR QueryLanguage,
|
|
const BSTR Query,
|
|
long lFlags,
|
|
IWbemContext __RPC_FAR *pCtx,
|
|
IWbemObjectSink __RPC_FAR *pResponseHandler)
|
|
{
|
|
HRESULT hr = WBEM_S_NO_ERROR;
|
|
IWbemClassObjectPtr pStatusObject;
|
|
|
|
try
|
|
{
|
|
HRESULT hrImp = CheckImpersonationLevel();
|
|
IWbemClassObjectPtr pStatusObject;
|
|
|
|
if(SUCCEEDED(hrImp))
|
|
{
|
|
// We will optimize for those cases in which
|
|
// a particular set of named job objects
|
|
// (e.g., 1 or more). Enumerate also
|
|
// optimizes for the properties that were
|
|
// requested.
|
|
CFrameworkQuery cfwq;
|
|
hr = cfwq.Init(
|
|
QueryLanguage,
|
|
Query,
|
|
lFlags,
|
|
m_chstrNamespace);
|
|
|
|
std::vector<_bstr_t> rgNamedJOs;
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
hr = cfwq.GetValuesForProp(
|
|
_bstr_t(g_rgJobObjPropNames[JO_ID]),
|
|
rgNamedJOs);
|
|
}
|
|
|
|
// If none were specifically requested, they
|
|
// want them all...
|
|
if(rgNamedJOs.size() == 0)
|
|
{
|
|
hr = GetJobObjectList(rgNamedJOs);
|
|
}
|
|
else
|
|
{
|
|
// Object paths were specified. Before
|
|
// passing them along, we need to un-
|
|
// decorate them.
|
|
UndecorateNamesInNamedJONameList(rgNamedJOs);
|
|
}
|
|
|
|
// Find out what propeties were requested...
|
|
CJobObjProps cjop(m_chstrNamespace);
|
|
cjop.GetWhichPropsReq(cfwq);
|
|
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
hr = Enumerate(
|
|
pCtx,
|
|
pResponseHandler,
|
|
rgNamedJOs,
|
|
cjop,
|
|
&pStatusObject);
|
|
}
|
|
else
|
|
{
|
|
SetStatusObject(
|
|
pCtx,
|
|
m_pNamespace,
|
|
-1L,
|
|
NULL,
|
|
L"Helpers.cpp::GetJobObjectList",
|
|
JOB_OBJECT_NAMESPACE,
|
|
&pStatusObject);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = hrImp;
|
|
}
|
|
}
|
|
catch(CVARIANTError& cve)
|
|
{
|
|
hr = cve.GetWBEMError();
|
|
}
|
|
catch(...)
|
|
{
|
|
hr = WBEM_E_PROVIDER_FAILURE;
|
|
}
|
|
|
|
// Set Status
|
|
hr = pResponseHandler->SetStatus(0, hr, NULL, pStatusObject);
|
|
return hr;
|
|
}
|
|
|
|
|
|
STDMETHODIMP CJobObjectProv::CreateInstanceEnumAsync(
|
|
const BSTR Class,
|
|
long lFlags,
|
|
IWbemContext __RPC_FAR *pCtx,
|
|
IWbemObjectSink __RPC_FAR *pResponseHandler)
|
|
{
|
|
HRESULT hr = WBEM_S_NO_ERROR;
|
|
IWbemClassObjectPtr pStatusObject;
|
|
|
|
try
|
|
{
|
|
HRESULT hrImp = CheckImpersonationLevel();
|
|
|
|
if(SUCCEEDED(hrImp))
|
|
{
|
|
if(_wcsicmp(Class, IDS_Win32_NamedJobObject) != NULL)
|
|
{
|
|
hr = WBEM_E_INVALID_CLASS;
|
|
}
|
|
|
|
// For every job object, return all properties...
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
// Get a list of named jobs...
|
|
std::vector<_bstr_t> rgNamedJOs;
|
|
hr = GetJobObjectList(rgNamedJOs);
|
|
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
CJobObjProps cjop(m_chstrNamespace);
|
|
cjop.SetReqProps(PROP_ALL_REQUIRED);
|
|
hr = Enumerate(
|
|
pCtx,
|
|
pResponseHandler,
|
|
rgNamedJOs,
|
|
cjop,
|
|
&pStatusObject);
|
|
}
|
|
else
|
|
{
|
|
SetStatusObject(
|
|
pCtx,
|
|
m_pNamespace,
|
|
-1L,
|
|
NULL,
|
|
L"Helpers.cpp::GetJobObjectList",
|
|
JOB_OBJECT_NAMESPACE,
|
|
&pStatusObject);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = hrImp;
|
|
}
|
|
}
|
|
catch(CVARIANTError& cve)
|
|
{
|
|
hr = cve.GetWBEMError();
|
|
}
|
|
catch(...)
|
|
{
|
|
hr = WBEM_E_PROVIDER_FAILURE;
|
|
}
|
|
|
|
// Set Status
|
|
return pResponseHandler->SetStatus(0, hr, NULL, pStatusObject);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
// Private member function implementations
|
|
/*****************************************************************************/
|
|
HRESULT CJobObjectProv::Enumerate(
|
|
IWbemContext __RPC_FAR *pCtx,
|
|
IWbemObjectSink __RPC_FAR *pResponseHandler,
|
|
std::vector<_bstr_t>& rgNamedJOs,
|
|
CJobObjProps& cjop,
|
|
IWbemClassObject** ppStatusObject)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
try // CVARIANT can throw and I want the error...
|
|
{
|
|
long lNumJobs = rgNamedJOs.size();
|
|
|
|
if(lNumJobs > 0)
|
|
{
|
|
SmartHandle hJob;
|
|
|
|
for(long m = 0L; m < lNumJobs && SUCCEEDED(hr); m++)
|
|
{
|
|
cjop.ClearProps();
|
|
|
|
// We have the name of a JO; need to open it up
|
|
// and get its properties...
|
|
hJob = ::OpenJobObjectW(
|
|
MAXIMUM_ALLOWED,
|
|
FALSE,
|
|
rgNamedJOs[m]);
|
|
|
|
// (NOTE: hJob smarthandle class automatically
|
|
// closes its handle on destruction and on
|
|
// reassignment.)
|
|
if(hJob)
|
|
{
|
|
// Set the handle...
|
|
cjop.SetHandle(hJob);
|
|
|
|
// Set the key properties directly...
|
|
CHString chstrDecoratedJOName;
|
|
DecorateJOName(
|
|
rgNamedJOs[m],
|
|
chstrDecoratedJOName);
|
|
|
|
std::vector<CVARIANT> vecvKeys;
|
|
CVARIANT vID(chstrDecoratedJOName);
|
|
vecvKeys.push_back(vID);
|
|
hr = cjop.SetKeysDirect(vecvKeys);
|
|
|
|
if(FAILED(hr))
|
|
{
|
|
SetStatusObject(
|
|
pCtx,
|
|
m_pNamespace,
|
|
::GetLastError(),
|
|
NULL,
|
|
L"CJobObjProps::SetKeysDirect",
|
|
JOB_OBJECT_NAMESPACE,
|
|
ppStatusObject);
|
|
}
|
|
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
// set the non-key requested
|
|
// properties...
|
|
hr = cjop.SetNonKeyReqProps();
|
|
|
|
if(FAILED(hr))
|
|
{
|
|
SetStatusObject(
|
|
pCtx,
|
|
m_pNamespace,
|
|
::GetLastError(),
|
|
NULL,
|
|
L"CJobObjProps::SetNonKeyReqProps",
|
|
JOB_OBJECT_NAMESPACE,
|
|
ppStatusObject);
|
|
}
|
|
}
|
|
|
|
// Create a new outgoing instance...
|
|
IWbemClassObjectPtr pIWCO = NULL;
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
hr = CreateInst(
|
|
m_pNamespace,
|
|
&pIWCO,
|
|
_bstr_t(IDS_Win32_NamedJobObject),
|
|
pCtx);
|
|
|
|
if(FAILED(hr))
|
|
{
|
|
SetStatusObject(
|
|
pCtx,
|
|
m_pNamespace,
|
|
::GetLastError(),
|
|
NULL,
|
|
L"CJobObjectProv::CreateInst",
|
|
JOB_OBJECT_NAMESPACE,
|
|
ppStatusObject);
|
|
}
|
|
}
|
|
|
|
// Load the properties of the
|
|
// new outgoing instance...
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
hr = cjop.LoadPropertyValues(pIWCO);
|
|
|
|
if(FAILED(hr))
|
|
{
|
|
SetStatusObject(
|
|
pCtx,
|
|
m_pNamespace,
|
|
::GetLastError(),
|
|
NULL,
|
|
L"CJobObjProps::LoadPropertyValues",
|
|
JOB_OBJECT_NAMESPACE,
|
|
ppStatusObject);
|
|
}
|
|
}
|
|
|
|
// And send it out...
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
IWbemClassObject *pTmp = (IWbemClassObject*) pIWCO;
|
|
hr = pResponseHandler->Indicate(
|
|
1,
|
|
&pTmp);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
_ASSERT(0);
|
|
|
|
hr = WBEM_E_NOT_FOUND;
|
|
|
|
SetStatusObject(
|
|
pCtx,
|
|
m_pNamespace,
|
|
::GetLastError(),
|
|
NULL,
|
|
L"CJobObjectProv::Enumerate",
|
|
JOB_OBJECT_NAMESPACE,
|
|
ppStatusObject);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
catch(CVARIANTError& cve)
|
|
{
|
|
hr = cve.GetWBEMError();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|