645 lines
18 KiB
C++
645 lines
18 KiB
C++
//+----------------------------------------------------------------------------
|
|
//
|
|
// Job Scheduler service
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1992 - 1996.
|
|
//
|
|
// File: sch_util.cxx
|
|
//
|
|
// Contents: scheduler object IUnknown methods, class factory,
|
|
// plus misc private class methods
|
|
//
|
|
// Classes: CSchedule, CScheduleCF
|
|
//
|
|
// Interfaces: IUnknown, IClassFactory
|
|
//
|
|
// History: 09-Sep-95 EricB created
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
#include "..\pch\headers.hxx"
|
|
#pragma hdrstop
|
|
#include "Sched.hxx"
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::ActivateJob, private
|
|
//
|
|
// Synopsis: Given a valid name, returns a pointer to the activated job
|
|
// object
|
|
//
|
|
// Arguments: [pwszName] - the folder-relative name of the job to activate
|
|
// [ppJob] - a pointer to the job class object; on entry,
|
|
// must either be NULL or point to a CJob object.
|
|
// [fAllData] - load all job data from disk.
|
|
//
|
|
// Returns: hresults
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
HRESULT
|
|
CSchedule::ActivateJob(LPCTSTR ptszName, CJob ** ppJob, BOOL fAllData)
|
|
{
|
|
TCHAR tszFullName[MAX_PATH + MAX_PATH];
|
|
HRESULT hr = StringCchCopy(tszFullName, MAX_PATH + MAX_PATH, g_TasksFolderInfo.ptszPath);
|
|
if (FAILED(hr))
|
|
{
|
|
schDebugOut((DEB_ERROR, "CSchedule::ActivateJob: StringCchCopy failed with error 0x%x\n", hr));
|
|
return hr;
|
|
}
|
|
hr = StringCchCat(tszFullName, MAX_PATH + MAX_PATH, TEXT("\\"));
|
|
if (FAILED(hr))
|
|
{
|
|
schDebugOut((DEB_ERROR, "CSchedule::ActivateJob: StringCchCat failed with error 0x%x\n", hr));
|
|
return hr;
|
|
}
|
|
hr = StringCchCat(tszFullName, MAX_PATH + MAX_PATH, ptszName);
|
|
if (FAILED(hr))
|
|
{
|
|
schDebugOut((DEB_ERROR, "CSchedule::ActivateJob: StringCchCat failed with error 0x%x\n", hr));
|
|
return hr;
|
|
}
|
|
//
|
|
// If *ppJob is NULL, allocate a new job object.
|
|
//
|
|
if (*ppJob == NULL)
|
|
{
|
|
//
|
|
// CJob is a single-use, in-proc handler, so no need to get OLE in the
|
|
// loop here. Use new (called by CJob::Create) instead of CoCreateInstance.
|
|
//
|
|
*ppJob = CJob::Create();
|
|
if (*ppJob == NULL)
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
|
|
hr = (*ppJob)->LoadP(tszFullName, 0, TRUE, fAllData);
|
|
if (FAILED(hr))
|
|
{
|
|
schDebugOut((DEB_ERROR, "CSchedule::ActivateJob: pJob->Load failed with error 0x%x\n", hr));
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::CheckJobName
|
|
//
|
|
// Synopsis: Checks for a valid job object file name and returns the full
|
|
// path name. Takes an UNICODE input name and returns a TCHAR.
|
|
//
|
|
// Arguments: [pwszJobName] - the job name as specified by the client
|
|
// [pptszFullPathName] - the name including the job folder path
|
|
//
|
|
// Returns: HRESULTS
|
|
//
|
|
// Notes: The job name can be an absolute or UNC path. If not,it is
|
|
// assumed to be relative to the job schedule folder.
|
|
// If there is an extension on the last element (the actual file
|
|
// name), then it must be .job. If there is no extension, then
|
|
// the the correct one will be added.
|
|
//-----------------------------------------------------------------------------
|
|
HRESULT
|
|
CSchedule::CheckJobName(LPCWSTR pwszJobName, LPTSTR * pptszFullPathName)
|
|
{
|
|
//
|
|
// Make sure that the string doesn't end in a slash character.
|
|
//
|
|
|
|
ULONG cchJobName = wcslen(pwszJobName);
|
|
ULONG cchNameParam = cchJobName;
|
|
|
|
if (!cchNameParam)
|
|
{
|
|
schDebugOut((DEB_ERROR,
|
|
"CSchedule::CheckJobName: pwszJobName is a 0 length string\n"));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
if (cchNameParam > 1 &&
|
|
(pwszJobName[cchNameParam - 1] == L'\\' ||
|
|
pwszJobName[cchNameParam - 1] == L'/'))
|
|
{
|
|
schDebugOut((DEB_ERROR,
|
|
"CSchedule::CheckJobName: pwszJobName ends in illegal char %c\n",
|
|
pwszJobName[cchNameParam - 1]));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
BOOL fNeedsPath = TRUE;
|
|
//
|
|
// Is it a full or relative path?
|
|
//
|
|
if ((cchNameParam > 2 &&
|
|
(pwszJobName[1] == TEXT(':') ||
|
|
(pwszJobName[0] == TEXT('\\') && pwszJobName[1] == TEXT('\\')) ||
|
|
(pwszJobName[0] == TEXT('/') && pwszJobName[1] == TEXT('/')))))
|
|
{
|
|
fNeedsPath = FALSE;
|
|
}
|
|
//
|
|
// Check extension
|
|
//
|
|
|
|
WCHAR * pwszJobExt = TSZ_JOB;
|
|
ULONG cJobExt = ARRAY_LEN(TSZ_JOB); // add one for the period
|
|
BOOL fNeedExt = FALSE;
|
|
const WCHAR * pwszLastDot = wcsrchr(pwszJobName, L'.');
|
|
|
|
if (pwszLastDot != NULL)
|
|
{
|
|
// check if the period is within cJobExt chars of the end
|
|
//
|
|
if ((size_t)(cchNameParam - (pwszLastDot - pwszJobName)) <= (size_t)cJobExt)
|
|
{
|
|
if (_wcsicmp(pwszLastDot + 1, pwszJobExt) != 0)
|
|
{
|
|
// Its extension does not match TSZ_JOB, so it is invalid.
|
|
//
|
|
schDebugOut((DEB_ERROR,
|
|
"CSchedule::CheckJobName: expected '%S', got '%S'",
|
|
pwszJobExt,
|
|
pwszLastDot + 1));
|
|
return E_INVALIDARG;
|
|
}
|
|
}
|
|
else // append the extension.
|
|
{
|
|
fNeedExt = TRUE;
|
|
cchNameParam += cJobExt; // add the length of the extension
|
|
}
|
|
}
|
|
else // append the extension.
|
|
{
|
|
fNeedExt = TRUE;
|
|
cchNameParam += cJobExt; // add the length of the extension
|
|
}
|
|
|
|
//
|
|
// Allocate the string to return the result.
|
|
//
|
|
if (fNeedsPath)
|
|
{
|
|
// add one for the '\'
|
|
cchNameParam += lstrlen(m_ptszFolderPath) + 1;
|
|
}
|
|
|
|
// add 1 to the array length for the null
|
|
TCHAR * ptszPath = new TCHAR[cchNameParam + 1];
|
|
if (ptszPath == NULL)
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
if (fNeedsPath)
|
|
{
|
|
StringCchCopy(ptszPath, cchNameParam + 1, m_ptszFolderPath);
|
|
StringCchCat(ptszPath, cchNameParam + 1, TEXT("\\"));
|
|
StringCchCat(ptszPath, cchNameParam + 1, pwszJobName);
|
|
}
|
|
else
|
|
{
|
|
StringCchCopy(ptszPath, cchNameParam + 1, pwszJobName);
|
|
}
|
|
|
|
if (fNeedExt)
|
|
{
|
|
StringCchCat(ptszPath, cchNameParam + 1, TEXT(".") TSZ_JOB);
|
|
}
|
|
|
|
*pptszFullPathName = ptszPath;
|
|
return S_OK;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::CSchedule
|
|
//
|
|
// Synopsis: constructor
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
CSchedule::CSchedule(void) :
|
|
m_ptszTargetMachine(NULL),
|
|
m_ptszFolderPath(NULL),
|
|
m_dwNextID(1),
|
|
m_uRefs(1)
|
|
{
|
|
InitializeCriticalSection(&m_CriticalSection);
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::~CSchedule
|
|
//
|
|
// Synopsis: destructor
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
CSchedule::~CSchedule(void)
|
|
{
|
|
DeleteCriticalSection(&m_CriticalSection);
|
|
if (m_ptszTargetMachine)
|
|
{
|
|
delete m_ptszTargetMachine;
|
|
}
|
|
if (m_ptszFolderPath)
|
|
{
|
|
delete m_ptszFolderPath;
|
|
}
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::Init
|
|
//
|
|
// Synopsis: Two phase construction - can't do operations that could fail
|
|
// in the ctor since there is no way to return errors without
|
|
// throwing exceptions.
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
HRESULT
|
|
CSchedule::Init(void)
|
|
{
|
|
if (g_TasksFolderInfo.ptszPath == NULL)
|
|
{
|
|
ERR_OUT("CSchedule::Init, folder path not set", E_FAIL);
|
|
return E_FAIL;
|
|
}
|
|
HRESULT hr;
|
|
|
|
//
|
|
// Get the jobs folder location. These values will be replaced when a
|
|
// call is made to SetTargetMachine
|
|
//
|
|
size_t cch = lstrlen(g_TasksFolderInfo.ptszPath) + 1;
|
|
m_ptszFolderPath = new TCHAR[cch];
|
|
if (!m_ptszFolderPath)
|
|
{
|
|
ERR_OUT("CSchedule::Init", E_OUTOFMEMORY);
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
StringCchCopy(m_ptszFolderPath, cch, g_TasksFolderInfo.ptszPath);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Function: GetNextAtID
|
|
//
|
|
// Synopsis: Examine the AT jobs to find the highest ID.
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
void
|
|
GetNextAtID(LPDWORD pdwAtID)
|
|
{
|
|
WCHAR wszAtJobSearchPath[MAX_PATH];
|
|
|
|
StringCchCopy(wszAtJobSearchPath, MAX_PATH, g_TasksFolderInfo.ptszPath);
|
|
StringCchCat(wszAtJobSearchPath, MAX_PATH, L"\\" TSZ_AT_JOB_PREFIX L"*." TSZ_JOB);
|
|
|
|
DWORD cchNamePrefixLen = ARRAY_LEN(TSZ_AT_JOB_PREFIX) - 1;
|
|
|
|
WIN32_FIND_DATA fd;
|
|
HANDLE hFileFind = FindFirstFile(wszAtJobSearchPath, &fd);
|
|
|
|
if (hFileFind == INVALID_HANDLE_VALUE)
|
|
{
|
|
//
|
|
// If no at jobs, set the initial job ID to be 1, since zero is
|
|
// reserved for an error flag.
|
|
//
|
|
*pdwAtID = 1;
|
|
return;
|
|
}
|
|
|
|
DWORD dwMaxID = 1;
|
|
|
|
do
|
|
{
|
|
WCHAR * pDot = wcschr(fd.cFileName, L'.');
|
|
if (pDot == NULL)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
*pDot = L'\0';
|
|
|
|
DWORD dwCurID = (DWORD)wcstoul(fd.cFileName + cchNamePrefixLen, NULL, 10);
|
|
|
|
schDebugOut((DEB_ITRACE, "GetNextAtID: found %S, with ID %d\n",
|
|
fd.cFileName, dwCurID));
|
|
|
|
if (dwCurID > dwMaxID)
|
|
{
|
|
dwMaxID = dwCurID;
|
|
}
|
|
|
|
} while (FindNextFile(hFileFind, &fd));
|
|
|
|
FindClose(hFileFind);
|
|
|
|
if (ULONG_MAX != dwMaxID)
|
|
{
|
|
//
|
|
// The next available AT ID will be one greater than the current max.
|
|
//
|
|
*pdwAtID = dwMaxID + 1;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// find first hole instead
|
|
//
|
|
StringCchCopy(wszAtJobSearchPath, MAX_PATH, g_TasksFolderInfo.ptszPath);
|
|
StringCchCat(wszAtJobSearchPath, MAX_PATH, L"\\" TSZ_AT_JOB_PREFIX);
|
|
DWORD cchOffset = wcslen(wszAtJobSearchPath);
|
|
|
|
dwMaxID = 1;
|
|
StringCchPrintf(wszAtJobSearchPath + cchOffset, MAX_PATH - cchOffset, L"%d%s", dwMaxID, TSZ_DOTJOB);
|
|
while (0xffffffff != GetFileAttributes(wszAtJobSearchPath) && dwMaxID < ULONG_MAX)
|
|
{
|
|
dwMaxID++;
|
|
StringCchPrintf(wszAtJobSearchPath + cchOffset, MAX_PATH - cchOffset, L"%d%s", dwMaxID, TSZ_DOTJOB);
|
|
}
|
|
|
|
// it is safe to assume dwMaxID now equals the first ID that wasn't found
|
|
// as it is unfeasible that all IDs from 1 to 0xffffffff (4,294,967,295) would be in use
|
|
|
|
*pdwAtID = dwMaxID;
|
|
}
|
|
return;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// CSchedule IUnknown methods
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::IUnknown::QueryInterface
|
|
//
|
|
// Synopsis: Returns requested interface pointer
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP
|
|
CSchedule::QueryInterface(REFIID riid, void ** ppvObject)
|
|
{
|
|
//schDebugOut((DEB_ITRACE, "CSchedule::QueryInterface"));
|
|
|
|
if (!ppvObject)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
if (IID_IUnknown == riid)
|
|
{
|
|
*ppvObject = (IUnknown *)this;
|
|
}
|
|
else if (IID_ITaskScheduler == riid)
|
|
{
|
|
*ppvObject = (IUnknown *)(ITaskScheduler *)this;
|
|
}
|
|
//else if (IID_IDispatch == riid)
|
|
//{
|
|
// *ppvObject = (IUnknown *)(IDispatch *)this;
|
|
//}
|
|
else
|
|
{
|
|
*ppvObject = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
AddRef();
|
|
return S_OK;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::IUnknown::AddRef
|
|
//
|
|
// Synopsis: increments reference count
|
|
//
|
|
// Returns: the reference count
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP_(ULONG)
|
|
CSchedule::AddRef(void)
|
|
{
|
|
//schDebugOut((DEB_ITRACE, "CSchedule::AddRef refcount going in %d\n", m_uRefs));
|
|
return InterlockedIncrement((long *)&m_uRefs);
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::IUnknown::Release
|
|
//
|
|
// Synopsis: Decrements the object's reference count and frees it when
|
|
// no longer referenced.
|
|
//
|
|
// Returns: zero if the reference count is zero or non-zero otherwise
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP_(ULONG)
|
|
CSchedule::Release(void)
|
|
{
|
|
//schDebugOut((DEB_ITRACE, "CSchedule::Release ref count going in %d\n", m_uRefs));
|
|
unsigned long uTmp;
|
|
if ((uTmp = InterlockedDecrement((long *)&m_uRefs)) == 0)
|
|
{
|
|
delete this;
|
|
}
|
|
return uTmp;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// CScheduleCF - class factory for the Schedule Service object
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CScheduleCF::Create
|
|
//
|
|
// Synopsis: creates a new class factory object
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
IClassFactory *
|
|
CScheduleCF::Create(void)
|
|
{
|
|
return new CScheduleCF;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CScheduleCF::CScheduleCF
|
|
//
|
|
// Synopsis: ctor
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
CScheduleCF::CScheduleCF(void)
|
|
{
|
|
m_uRefs = 1;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CScheduleCF::~CScheduleCF
|
|
//
|
|
// Synopsis: dtor
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
CScheduleCF::~CScheduleCF(void)
|
|
{
|
|
;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CScheduleCF::IUnknown::QueryInterface
|
|
//
|
|
// Synopsis: Returns requested interface pointer
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP
|
|
CScheduleCF::QueryInterface(REFIID riid, void ** ppvObject)
|
|
{
|
|
//schDebugOut((DEB_ITRACE, "CScheduleCF::QueryInterface"));
|
|
|
|
if (!ppvObject)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
if (IID_IUnknown == riid)
|
|
{
|
|
*ppvObject = (IUnknown *)this;
|
|
}
|
|
else if (IsEqualIID(IID_IClassFactory, riid))
|
|
{
|
|
*ppvObject = (IClassFactory *)this;
|
|
}
|
|
else
|
|
{
|
|
*ppvObject = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
AddRef();
|
|
return S_OK;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CScheduleCF::IUnknown::AddRef
|
|
//
|
|
// Synopsis: increments reference count
|
|
//
|
|
// Returns: the new reference count
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP_(ULONG)
|
|
CScheduleCF::AddRef(void)
|
|
{
|
|
return InterlockedIncrement((long *)&m_uRefs);
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CScheduleCF::IUnknown::Release
|
|
//
|
|
// Synopsis: noop, since this is a static object
|
|
//
|
|
// Returns: the new reference count
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP_(ULONG)
|
|
CScheduleCF::Release(void)
|
|
{
|
|
unsigned long uTmp;
|
|
if ((uTmp = InterlockedDecrement((long *)&m_uRefs)) == 0)
|
|
{
|
|
delete this;
|
|
}
|
|
return uTmp;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CScheduleCF::IClassFactory::CreateInstance
|
|
//
|
|
// Synopsis: create an incore instance of the job class object
|
|
//
|
|
// Arguments: [pUnkOuter] - aggregator
|
|
// [riid] - requested interface
|
|
// [ppvObject] - receptor for itf ptr
|
|
//
|
|
// Returns: HRESULTS
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP
|
|
CScheduleCF::CreateInstance(IUnknown *pUnkOuter, REFIID riid, void **ppvObject)
|
|
{
|
|
//schDebugOut((DEB_ITRACE, "CScheduleCF::CreateInstance\n"));
|
|
|
|
if (!ppvObject)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
HRESULT hr = S_OK;
|
|
*ppvObject = NULL;
|
|
|
|
CSchedule * pSched = new CSchedule;
|
|
if (pSched == NULL)
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
hr = pSched->Init();
|
|
if (FAILED(hr))
|
|
{
|
|
ERR_OUT("CScheduleCF::CreateInstance, pSched->Init", hr);
|
|
pSched->Release();
|
|
return hr;
|
|
}
|
|
|
|
hr = pSched->QueryInterface(riid, ppvObject);
|
|
if (FAILED(hr))
|
|
{
|
|
ERR_OUT("CScheduleCF::CreateInstance, pSched->QueryInterface", hr);
|
|
pSched->Release();
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// We got a refcount of one when launched, and the above QI increments it
|
|
// to 2, so call release to take it back to 1.
|
|
//
|
|
pSched->Release();
|
|
return hr;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CScheduleCF::IClassFactory::LockServer
|
|
//
|
|
// Synopsis: Called with fLock set to TRUE to indicate that the server
|
|
// should continue to run even if none of its objects are active
|
|
//
|
|
// Arguments: [fLock] - increment/decrement the instance count
|
|
//
|
|
// Returns: HRESULTS
|
|
//
|
|
// Notes: This is a no-op since the handler runs in-proc.
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP
|
|
CScheduleCF::LockServer(BOOL fLock)
|
|
{
|
|
return S_OK;
|
|
}
|
|
|