1303 lines
34 KiB
C++
1303 lines
34 KiB
C++
/////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Utils.cpp
|
|
//
|
|
// General-purpose routines that are project independent.
|
|
//
|
|
// HISTORY
|
|
// t-danmo 96.09.22 Creation.
|
|
//
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
#include "stdafx.h"
|
|
#include "progress.h" // CServiceControlProgress
|
|
#include "macros.h" // MFC_TRY/MFC_CATCH
|
|
USE_HANDLE_MACROS("FILEMGMT(utils.cpp)")
|
|
|
|
|
|
#ifdef _DEBUG
|
|
#define new DEBUG_NEW
|
|
#undef THIS_FILE
|
|
static char THIS_FILE[] = __FILE__;
|
|
#endif
|
|
|
|
//
|
|
// Fusion MFC-based property page
|
|
//
|
|
HPROPSHEETPAGE MyCreatePropertySheetPage(AFX_OLDPROPSHEETPAGE* psp)
|
|
{
|
|
PROPSHEETPAGE_V3 sp_v3 = {0};
|
|
CopyMemory (&sp_v3, psp, psp->dwSize);
|
|
sp_v3.dwSize = sizeof(sp_v3);
|
|
|
|
return (::CreatePropertySheetPage (&sp_v3));
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
void
|
|
ComboBox_FlushContent(HWND hwndCombo)
|
|
{
|
|
Assert(IsWindow(hwndCombo));
|
|
SendMessage(hwndCombo, CB_RESETCONTENT, 0, 0);
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// ComboBox_FFill()
|
|
//
|
|
// Fill a combo box with the array of string Ids.
|
|
//
|
|
// Return FALSE if an error occurs (such as stringId not found).
|
|
//
|
|
BOOL
|
|
ComboBox_FFill(
|
|
const HWND hwndCombo, // IN: Handle of the combobox
|
|
const TStringParamEntry rgzSPE[], // IN: SPE aray zero terminated
|
|
const LPARAM lItemDataSelect) // IN: Which item to select
|
|
{
|
|
CString str;
|
|
TCHAR szBuffer[1024];
|
|
LRESULT lResult;
|
|
|
|
Assert(IsWindow(hwndCombo));
|
|
Assert(rgzSPE != NULL);
|
|
|
|
for (int i = 0; rgzSPE[i].uStringId != 0; i++)
|
|
{
|
|
if (!::LoadString(g_hInstanceSave, rgzSPE[i].uStringId,
|
|
OUT szBuffer, LENGTH(szBuffer)))
|
|
{
|
|
TRACE1("Unable to load string Id=%d.\n", rgzSPE[i].uStringId);
|
|
Assert(FALSE && "Unable to load string");
|
|
return FALSE;
|
|
}
|
|
lResult = SendMessage(hwndCombo, CB_ADDSTRING, 0,
|
|
reinterpret_cast<LPARAM>(szBuffer));
|
|
Report(lResult >= 0);
|
|
const WPARAM iIndex = lResult;
|
|
lResult = SendMessage(hwndCombo, CB_SETITEMDATA, iIndex,
|
|
rgzSPE[i].lItemData);
|
|
Report(lResult != CB_ERR);
|
|
if (rgzSPE[i].lItemData == lItemDataSelect)
|
|
{
|
|
SendMessage(hwndCombo, CB_SETCURSEL, iIndex, 0);
|
|
}
|
|
} // for
|
|
return TRUE;
|
|
} // ComboBox_FFill()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// ComboBox_FGetSelectedItemData()
|
|
//
|
|
// Get the value of the lParam field of the current selected item.
|
|
//
|
|
// If an error occurs return -1 (CB_ERR).
|
|
// Otherwise the value of the selected item.
|
|
//
|
|
LPARAM
|
|
ComboBox_GetSelectedItemData(HWND hwndComboBox)
|
|
{
|
|
LPARAM l;
|
|
|
|
Assert(IsWindow(hwndComboBox));
|
|
l = SendMessage(hwndComboBox, CB_GETCURSEL, 0, 0);
|
|
Report(l != CB_ERR && "Combobox has no item selected");
|
|
l = SendMessage(hwndComboBox, CB_GETITEMDATA, l, 0);
|
|
Assert(l != CB_ERR && "Cannot extract item data from combobox");
|
|
if (l == CB_ERR)
|
|
{
|
|
Assert(CB_ERR == -1);
|
|
return -1;
|
|
}
|
|
return l;
|
|
} // ComboBox_GetSelectedItemData()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
HWND
|
|
HGetDlgItem(HWND hdlg, INT nIdDlgItem)
|
|
{
|
|
Assert(IsWindow(hdlg));
|
|
Assert(IsWindow(GetDlgItem(hdlg, nIdDlgItem)));
|
|
return GetDlgItem(hdlg, nIdDlgItem);
|
|
} // HGetDlgItem()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
void
|
|
SetDlgItemFocus(HWND hdlg, INT nIdDlgItem)
|
|
{
|
|
Assert(IsWindow(hdlg));
|
|
Assert(IsWindow(GetDlgItem(hdlg, nIdDlgItem)));
|
|
SetFocus(GetDlgItem(hdlg, nIdDlgItem));
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
void
|
|
EnableDlgItem(HWND hdlg, INT nIdDlgItem, BOOL fEnable)
|
|
{
|
|
Assert(IsWindow(hdlg));
|
|
Assert(IsWindow(GetDlgItem(hdlg, nIdDlgItem)));
|
|
EnableWindow(GetDlgItem(hdlg, nIdDlgItem), fEnable);
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// Enable/disable one or more controls in a dialog.
|
|
void
|
|
EnableDlgItemGroup(
|
|
HWND hdlg, // IN: Parent dialog of the controls
|
|
const UINT rgzidCtl[], // IN: Group (array) of control Ids to be enabled (or disabled)
|
|
BOOL fEnableAll) // IN: TRUE => We want to enable the controls; FALSE => We want to disable the controls
|
|
{
|
|
Assert(IsWindow(hdlg));
|
|
Assert(rgzidCtl != NULL);
|
|
for (const UINT * pidCtl = rgzidCtl; *pidCtl != 0; pidCtl++)
|
|
{
|
|
EnableWindow(HGetDlgItem(hdlg, *pidCtl), fEnableAll);
|
|
}
|
|
} // EnableDlgItemGroup()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// Show/hide one or more controls in a dialog.
|
|
void
|
|
ShowDlgItemGroup(
|
|
HWND hdlg, // IN: Parent dialog of the controls
|
|
const UINT rgzidCtl[], // IN: Group (array) of control Ids to be shown (or hidden)
|
|
BOOL fShowAll) // IN: TRUE => We want to show the controls; FALSE => We want to hide the controls
|
|
{
|
|
Assert(IsWindow(hdlg));
|
|
Assert(rgzidCtl != NULL);
|
|
INT nCmdShow = fShowAll ? SW_SHOW : SW_HIDE;
|
|
for (const UINT * pidCtl = rgzidCtl; *pidCtl != 0; pidCtl++)
|
|
{
|
|
ShowWindow(HGetDlgItem(hdlg, *pidCtl), nCmdShow);
|
|
}
|
|
} // ShowDlgItemGroup()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// Str_PchCopyChN()
|
|
//
|
|
// Copy a string until reaching character chStop or destination buffer full.
|
|
//
|
|
// RETURNS
|
|
// Pointer to the last character of source buffer not copied into destination buffer.
|
|
// This may be useful to parse the rest of the source string.
|
|
//
|
|
// INTERFACE NOTES
|
|
// Character chStop is not copied into the destination buffer.
|
|
// If cchDstMax==0, the number of characters will not be limited.
|
|
//
|
|
TCHAR *
|
|
Str_PchCopyChN(
|
|
TCHAR * szDst, // OUT: Destination buffer
|
|
CONST TCHAR * szSrc, // IN: Source buffer
|
|
TCHAR chStop, // IN: Character to stop the copying
|
|
INT cchDstMax) // IN: Length of the output buffer
|
|
{
|
|
Assert(szDst != NULL);
|
|
Assert(szSrc != NULL);
|
|
Assert(cchDstMax >= 0);
|
|
|
|
while (*szSrc != '\0' && *szSrc != chStop && --cchDstMax != 0)
|
|
{
|
|
*szDst++ = *szSrc++;
|
|
}
|
|
*szDst = '\0';
|
|
return const_cast<TCHAR *>(szSrc);
|
|
} // Str_PchCopyChN()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// Str_RemoveSubStr()
|
|
//
|
|
// Scan the source string and remove every occurrence of a
|
|
// sub-string.
|
|
//
|
|
// RETURNS
|
|
// Return the number of removals performed.
|
|
//
|
|
// 580255-2002/03/18 JonN fixed Str_SubstituteStrStr buffer overrun
|
|
INT
|
|
Str_RemoveSubStr(
|
|
WCHAR * szBuf, // IN OUT: Source/Destination buffer
|
|
CONST WCHAR * szToken) // IN: Token to remove
|
|
{
|
|
if (!szBuf || !*szBuf || !szToken || !*szToken)
|
|
return 0;
|
|
|
|
INT cSubstitutions = 0;
|
|
size_t cchToken = wcslen(szToken);
|
|
|
|
TCHAR* pszFound = NULL;
|
|
while ( NULL != (pszFound = wcsstr(szBuf,szToken)) )
|
|
{
|
|
size_t cchBuf = wcslen(szBuf);
|
|
TCHAR* pszEndToken = pszFound + cchToken;
|
|
size_t cchMove = 1 + (szBuf+cchBuf) - pszEndToken;
|
|
wmemmove(pszFound, pszEndToken, cchMove ); // includes trailing NULL
|
|
cSubstitutions++;
|
|
}
|
|
|
|
return cSubstitutions;
|
|
} // Str_SubstituteStrStr()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// PchParseCommandLine()
|
|
//
|
|
// Split a command line into its path to its executable binary and
|
|
// its command line arguments. The path to the executable is
|
|
// copied into the output buffer.
|
|
//
|
|
// RETURNS
|
|
// Pointer to the next character after path to the executable (Pointer
|
|
// may point to an empty string). If an error occurs, return NULL.
|
|
//
|
|
// FORMATS SUPPORTED
|
|
// 1. "c:\\winnt\\foo.exe /bar"
|
|
// 2. ""c:\\winnt\\foo.exe" /bar"
|
|
// The double quotes around the binary path allow
|
|
// the binary path to have spaces.
|
|
//
|
|
|
|
|
|
// NTRAID#NTBUG9-580255-2002/03/18 JonN PchParseCommandLine buffer overrun
|
|
TCHAR *
|
|
PchParseCommandLine(
|
|
CONST TCHAR szFullCommand[], // IN: Full command line
|
|
TCHAR szBinPath[], // OUT: Path of the executable binary
|
|
INT cchBinPathBuf) // IN: Size of the buffer
|
|
{
|
|
UNREFERENCED_PARAMETER (cchBinPathBuf);
|
|
CONST TCHAR * pchSrc = szFullCommand;
|
|
TCHAR * pchDst = szBinPath;
|
|
BOOL fQuotesFound = FALSE; // TRUE => The binary path is surrounded by quotes (")
|
|
|
|
Assert(szFullCommand != NULL);
|
|
Assert(szBinPath != NULL);
|
|
|
|
// Skip leading spaces
|
|
while (*pchSrc == _T(' '))
|
|
pchSrc++;
|
|
if (*pchSrc == _T('\"'))
|
|
{
|
|
fQuotesFound = TRUE;
|
|
pchSrc++;
|
|
}
|
|
while (TRUE)
|
|
{
|
|
*pchDst = *pchSrc;
|
|
if (*pchSrc == _T('\0'))
|
|
break;
|
|
if (*pchSrc == _T('\"') && fQuotesFound)
|
|
{
|
|
pchSrc++;
|
|
break;
|
|
}
|
|
if (*pchSrc == _T(' ') && !fQuotesFound)
|
|
{
|
|
pchSrc++;
|
|
break;
|
|
}
|
|
pchSrc++;
|
|
pchDst++;
|
|
}
|
|
Assert(pchDst - szBinPath < cchBinPathBuf);
|
|
*pchDst = _T('\0');
|
|
|
|
return const_cast<TCHAR *>(pchSrc); // Return character where arguments starts
|
|
} // PchParseCommandLine()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
void TrimString(CString& rString)
|
|
{
|
|
rString.TrimLeft();
|
|
rString.TrimRight();
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// PargzpszFromPgrsz()
|
|
//
|
|
// Parse a group of strings into an array of pointers to strings.
|
|
// This routine is somewhat similar to CommandLineToArgvW() but
|
|
// uses a group of strings instead of a normal string.
|
|
//
|
|
// RETURN
|
|
// Return a pointer to an allocated array of pointers to strings.
|
|
// The array of pointers allocated with the new() operator,
|
|
// therefore the caller must call ONCE delete() to free memory.
|
|
//
|
|
// 581272 JonN 2002/04/03 With this change, this function returns an
|
|
// array of pointers to strings, but does not allocate the strings
|
|
// themselves. pgrsz must not be deleted before the return value.
|
|
//
|
|
// BACKGROUND
|
|
// You need to 'understand' hungarian prefixes to appreciate the
|
|
// name of the function.
|
|
//
|
|
// p Pointer to something
|
|
// psz Pointer to string terminated.
|
|
// pa Pointer dynamically allocated. For instance, pasz is
|
|
// a pointer to an allocated string. The allocation is
|
|
// to remind the developper he/she have to free the memory
|
|
// when done with the variable.
|
|
// rg Array (range). Array (rg) is similar to pointer (p)
|
|
// but may point to more than one element.
|
|
// rgch is an array of characters while pch points to
|
|
// a single character.
|
|
// rgz Array of which the last element is zero. The 'last element'
|
|
// may be a character, an integer, a pointer or any other
|
|
// data type found in the array.
|
|
// For instance rgzch would be an array of characters having
|
|
// its last character zero -- a string (sz).
|
|
// gr Group. This is different than array because indexing
|
|
// cannot be used. For instance, a group of strings is
|
|
// not the same as an array of strings.
|
|
// char grsz[] = "DOS\0WfW\0Win95\0WinNT\0";
|
|
// char * rgpsz[] = { "DOS", "WfW", "Win95", "WinNT" };
|
|
// char * rgzpsz[] = { "DOS", "WfW", "Win95", "WinNT", NULL };
|
|
//
|
|
// Now it is time to put all the pieces together.
|
|
// pargzpsz = "pa" + "rgz" + "psz"
|
|
// pgrsz = "p" + "gr" + "sz"
|
|
//
|
|
// USAGE
|
|
// LPTSTR * pargzpsz;
|
|
// pargzpsz = PargzpszFromPgrsz("DOS\0WfW\0Win95\0WinNT\0", OUT &cStringCount);
|
|
// delete pargzpsz; // Single delete to free memory
|
|
//
|
|
// 581272 JonN 2002/04/03 I modified this routine to put minimum trust in
|
|
// the pgrsz parameter, which originates in the cfgmgr32 APIs.
|
|
//
|
|
LPTSTR *
|
|
PargzpszFromPgrsz(
|
|
CONST LPCTSTR pgrsz, // IN: Pointer to group of strings
|
|
INT * pcStringCount) // OUT: OPTIONAL: Count of strings in the stored into returned value
|
|
{
|
|
if (NULL == pgrsz)
|
|
{
|
|
ASSERT(FALSE);
|
|
return NULL;
|
|
}
|
|
|
|
// Compute how much memory is needed for allocation
|
|
int cStringCount = 0;
|
|
CONST TCHAR * pchSrc = pgrsz;
|
|
while (!IsBadStringPtr(pchSrc, (UINT)-1))
|
|
{
|
|
if (*pchSrc == _T('\0'))
|
|
break;
|
|
cStringCount++;
|
|
pchSrc += lstrlen(pchSrc)+1;
|
|
} // while
|
|
|
|
// Allocate a single block of memory for all the data
|
|
LPTSTR * pargzpsz = new LPTSTR[cStringCount+1];
|
|
if (NULL == pargzpsz)
|
|
{
|
|
Assert(FALSE);
|
|
return NULL;
|
|
}
|
|
ZeroMemory(pargzpsz, (cStringCount+1)*sizeof(LPTSTR));
|
|
|
|
int iString = 0;
|
|
pchSrc = pgrsz;
|
|
while (!IsBadStringPtr(pchSrc, (UINT)-1) && iString < cStringCount)
|
|
{
|
|
pargzpsz[iString++] = (LPTSTR)pchSrc;
|
|
pchSrc += lstrlen(pchSrc)+1;
|
|
} // for
|
|
pargzpsz[cStringCount] = NULL;
|
|
if (pcStringCount != NULL)
|
|
*pcStringCount = cStringCount;
|
|
return pargzpsz;
|
|
} // PargzpszFromPgrsz()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
void
|
|
ListView_AddColumnHeaders(
|
|
HWND hwndListview, // IN: Handle of the listview we want to add columns
|
|
const TColumnHeaderItem rgzColumnHeader[]) // IN: Array of column header items
|
|
{
|
|
RECT rcClient;
|
|
INT cxTotalWidth; // Total width of the listview control
|
|
LV_COLUMN lvColumn;
|
|
INT cxColumn; // Width of the individual column
|
|
TCHAR szBuffer[1024];
|
|
|
|
Assert(IsWindow(hwndListview));
|
|
Assert(rgzColumnHeader != NULL);
|
|
|
|
GetClientRect(hwndListview, OUT &rcClient);
|
|
cxTotalWidth = rcClient.right;
|
|
lvColumn.pszText = szBuffer;
|
|
|
|
for (INT i = 0; rgzColumnHeader[i].uStringId != 0; i++)
|
|
{
|
|
if (!::LoadString(g_hInstanceSave, rgzColumnHeader[i].uStringId,
|
|
OUT szBuffer, LENGTH(szBuffer)))
|
|
{
|
|
TRACE1("Unable to load string Id=%d\n", rgzColumnHeader[i].uStringId);
|
|
Assert(FALSE);
|
|
continue;
|
|
}
|
|
lvColumn.mask = LVCF_TEXT;
|
|
cxColumn = rgzColumnHeader[i].nColWidth;
|
|
if (cxColumn > 0)
|
|
{
|
|
Assert(cxColumn <= 100);
|
|
cxColumn = (cxTotalWidth * cxColumn) / 100;
|
|
lvColumn.mask |= LVCF_WIDTH;
|
|
lvColumn.cx = cxColumn;
|
|
}
|
|
|
|
INT iColRet = ListView_InsertColumn(hwndListview, i, IN &lvColumn);
|
|
Report(iColRet == i);
|
|
} // for
|
|
|
|
} // ListView_AddColumnHeaders()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
int
|
|
ListView_InsertItemEx(
|
|
HWND hwndListview, // IN: Handle of the listview we want to add item
|
|
CONST LV_ITEM * pLvItem) // IN: Pointer to listview item
|
|
{
|
|
LV_ITEM lvItemT; // Temporary variable
|
|
TCHAR szT[1024]; // Temporary buffer
|
|
TCHAR * pch;
|
|
INT iItem; // Index of the item
|
|
|
|
Assert(IsWindow(hwndListview));
|
|
Assert(pLvItem != NULL);
|
|
|
|
lvItemT = *pLvItem; // Copy the whole structure
|
|
lvItemT.iSubItem = 0;
|
|
lvItemT.pszText = szT;
|
|
|
|
// Copy until the next
|
|
pch = Str_PchCopyChN(OUT szT, pLvItem->pszText, '\t', LENGTH(szT));
|
|
Assert(pch != NULL);
|
|
|
|
iItem = ListView_InsertItem(hwndListview, IN &lvItemT);
|
|
Report(iItem >= 0);
|
|
if (*pch == '\0')
|
|
return iItem;
|
|
Assert(*pch == '\t');
|
|
|
|
lvItemT.mask = LVIF_TEXT;
|
|
lvItemT.iItem = iItem;
|
|
lvItemT.iSubItem = 1;
|
|
|
|
while (*pch != '\0')
|
|
{
|
|
pch = Str_PchCopyChN(OUT szT, pch + 1, '\t', LENGTH(szT));
|
|
BOOL fRet = ListView_SetItem(hwndListview, IN &lvItemT);
|
|
Report(fRet != FALSE);
|
|
lvItemT.iSubItem++;
|
|
break;
|
|
}
|
|
return iItem;
|
|
} // ListView_InsertItemEx()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// Display the common dialog to get a filename.
|
|
BOOL
|
|
UiGetFileName(
|
|
HWND hwnd,
|
|
TCHAR szFileName[], // OUT: Filename we want to get
|
|
INT cchBufferLength) // IN: Length of szFileName buffer
|
|
{
|
|
OPENFILENAME ofn;
|
|
|
|
Assert(szFileName != NULL);
|
|
Assert(cchBufferLength > 10); // At least 10 characters
|
|
|
|
TCHAR szBufferT[2048];
|
|
::ZeroMemory( szBufferT, sizeof(szBufferT) );
|
|
VERIFY(::LoadString(g_hInstanceSave, IDS_OPENFILE_FILTER, szBufferT, LENGTH(szBufferT)));
|
|
|
|
::ZeroMemory(OUT &ofn, sizeof(ofn));
|
|
ofn.lStructSize = sizeof(OPENFILENAME);
|
|
ofn.hwndOwner = hwnd;
|
|
ofn.hInstance = g_hInstanceSave;
|
|
ofn.lpstrFilter = szBufferT;
|
|
ofn.nFilterIndex = 1;
|
|
ofn.lpstrFile = szFileName;
|
|
ofn.nMaxFile = cchBufferLength;
|
|
ofn.Flags = OFN_EXPLORER | OFN_HIDEREADONLY;
|
|
|
|
return GetOpenFileName(&ofn);
|
|
} // UiGetFileName()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// PaszLoadStringPrintf()
|
|
//
|
|
// Load a string from the resource, and format it and return
|
|
// pointer allocated string.
|
|
//
|
|
// RETURNS
|
|
// Pointer to allocated string. Must call LocalFree() when
|
|
// done with string.
|
|
//
|
|
// INTERFACE NOTES
|
|
// The format of the resource string uses %1 throuth %99 and
|
|
// assumes the arguments are pointers to strings.
|
|
//
|
|
// If you have an argument other than a string, you can append a
|
|
// printf-type within two exclamation marks.
|
|
// !s! Insert a string (default)
|
|
// !d! Insert a decimal integer
|
|
// !u! Insert an unsigned integer
|
|
// !x! Insert an hexadecimal integer
|
|
//
|
|
// HOW TO AVOID BUGS
|
|
// To avoid bugs using this routine, I strongly suggest to include
|
|
// the format of the string as part of the name of the string Id.
|
|
// If you change the format of the string, you should rename
|
|
// the string Id to reflect the new format. This will guarantee
|
|
// the correct type and number of arguments are used.
|
|
//
|
|
// EXAMPLES
|
|
// IDS_s_PROPERTIES = "%1 Properties"
|
|
// IDS_ss_PROPERTIES = "%1 Properties on %2"
|
|
// IDS_sus_SERVICE_ERROR = "Service %1 encountered error %2!u! while connecting to %3"
|
|
//
|
|
// HISTORY
|
|
// 96.10.30 t-danmo Creation
|
|
//
|
|
TCHAR *
|
|
PaszLoadStringPrintf(
|
|
UINT wIdString, // IN: String Id
|
|
va_list arglist) // IN: Arguments (if any)
|
|
{
|
|
Assert(wIdString != 0);
|
|
|
|
TCHAR szBufferT[2048];
|
|
LPTSTR paszBuffer = NULL; // Pointer to allocated buffer. Caller must call LocalFree() to free it
|
|
|
|
// Load the string from the resource
|
|
VERIFY(::LoadString(g_hInstanceSave, wIdString, szBufferT, LENGTH(szBufferT)));
|
|
|
|
// Format the string
|
|
::FormatMessage(
|
|
FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_STRING,
|
|
szBufferT,
|
|
0,
|
|
0,
|
|
OUT (LPTSTR)&paszBuffer, // Buffer will be allocated by FormatMessage()
|
|
0,
|
|
&arglist);
|
|
|
|
#ifdef DEBUG
|
|
if (paszBuffer == NULL)
|
|
{
|
|
DWORD dw = GetLastError();
|
|
Report(FALSE && "FormatMessage() failed.");
|
|
}
|
|
#endif
|
|
return paszBuffer;
|
|
} // PaszLoadStringPrintf()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// LoadStringPrintf()
|
|
//
|
|
// Load a string from the resources, format it and copy the result string
|
|
// into the CString object.
|
|
//
|
|
// Can also use LoadStringWithInsertions()
|
|
// AFX_MANAGE_STATE(AfxGetStaticModuleState())
|
|
//
|
|
// EXAMPLES
|
|
// LoadStrigPrintf(IDS_s_PROPERTIES, OUT &strCaption, szServiceName);
|
|
// LoadStrigPrintf(IDS_ss_PROPERTIES, OUT &strCaption, szServiceName, szMachineName);
|
|
// LoadStrigPrintf(IDS_sus_SERVICE_ERROR, OUT &strMessage, szServiceName, ::GetLastError(), szMachineName);
|
|
//
|
|
// ISSUE-2002/03/05-JonN This is a dangerous function
|
|
//
|
|
void
|
|
LoadStringPrintf(
|
|
UINT wIdString, // IN: String Id
|
|
CString * pString, // OUT: String to receive the characters
|
|
...) // IN: Optional arguments
|
|
{
|
|
Assert(wIdString != NULL);
|
|
Assert(pString != NULL);
|
|
|
|
va_list arglist;
|
|
va_start(arglist, pString);
|
|
|
|
TCHAR * paszBuffer = PaszLoadStringPrintf(wIdString, arglist);
|
|
*pString = paszBuffer; // Copy the string into the CString object
|
|
LocalFree(paszBuffer);
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// SetWindowTextPrintf()
|
|
//
|
|
// Load a string from the resource, format it and set the window text.
|
|
//
|
|
// EXAMPLE
|
|
// SetWindowText(hwndStatic, IDS_s_PROPERTIES, szObjectName);
|
|
//
|
|
// HISTORY
|
|
// 96.10.30 t-danmo Creation. Core copied from LoadStringPrintf()
|
|
//
|
|
void
|
|
SetWindowTextPrintf(HWND hwnd, UINT wIdString, ...)
|
|
{
|
|
ASSERT(IsWindow(hwnd));
|
|
ASSERT(wIdString != 0);
|
|
|
|
va_list arglist;
|
|
va_start(arglist, wIdString);
|
|
TCHAR * paszBuffer = PaszLoadStringPrintf(wIdString, arglist);
|
|
if (NULL != paszBuffer) // JonN 5/30/00 PREFIX 110941
|
|
SetWindowText(hwnd, paszBuffer); // Set the text of the window
|
|
LocalFree(paszBuffer);
|
|
} // SetWindowTextPrintf()
|
|
|
|
#ifdef SNAPIN_PROTOTYPER
|
|
|
|
const TCHAR rgchHexDigits[] = _T("00112233445566778899aAbBcCdDeEfF");
|
|
const TCHAR szSpcTab[] = _T(" \t");
|
|
const TCHAR szWhiteSpaces[] = _T(" \t\n\r\f\v");
|
|
const TCHAR szDecimalDigits[] = _T("0123456789");
|
|
|
|
#ifdef UNICODE
|
|
#define strchrT wcschr
|
|
#else
|
|
#define strchrT strchr
|
|
#endif
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// FParseInteger()
|
|
//
|
|
// Parse the source string pchSrc and extract
|
|
// its integer value.
|
|
//
|
|
// RETURNS
|
|
// Return TRUE if successful and set uData to integer value
|
|
// of the parsed string.
|
|
// If not successful (ie, illegal digit or overflow), return FALSE,
|
|
// set uData to zero and set nErrCode to error found.
|
|
// Field pi.pchStop is always set to the last valid character
|
|
// parsed.
|
|
//
|
|
// INTERFACE NOTES
|
|
// Fields pPI->pchSrc and pPI->nFlags are preserved during
|
|
// the execution of FParseInteger().
|
|
//
|
|
BOOL
|
|
FParseInteger(INOUT TParseIntegerInfo * pPI)
|
|
{
|
|
UINT uDataT;
|
|
UINT uBase;
|
|
UINT iDigit;
|
|
UINT cDigitParsed; // Number of digits parsed
|
|
BOOL fIsNegative = FALSE;
|
|
const TCHAR * pchDigit;
|
|
|
|
Assert(pPI != NULL);
|
|
Assert(pPI->pchSrc != NULL);
|
|
pPI->pchStop = pPI->pchSrc;
|
|
pPI->nErrCode = PI_errOK; // No error yet
|
|
pPI->uData = 0;
|
|
uBase = (pPI->nFlags & PI_mskfHexBaseOnly) ? 16 : 10;
|
|
cDigitParsed = 0;
|
|
|
|
// Skip leading blanks
|
|
while (*pPI->pchStop ==_T(' '))
|
|
pPI->pchStop++;
|
|
// Check for a minus sign
|
|
if (*pPI->pchStop == _T('-'))
|
|
{
|
|
if (pPI->nFlags & PI_mskfNoMinusSign)
|
|
{
|
|
pPI->nErrCode = PI_errMinusSignFound;
|
|
return FALSE;
|
|
}
|
|
fIsNegative = TRUE;
|
|
pPI->pchStop++;
|
|
}
|
|
// Skip leading zeroes
|
|
while (*pPI->pchStop == _T('0'))
|
|
{
|
|
pPI->pchStop++;
|
|
cDigitParsed++;
|
|
}
|
|
// Look for the hexadecimal prefix (0x or 0X)
|
|
if (*pPI->pchStop == _T('x') || *pPI->pchStop == _T('X'))
|
|
{
|
|
if ((pPI->nFlags & PI_mskfAllowHexBase) == 0)
|
|
{
|
|
pPI->nErrCode = PI_errInvalidInteger;
|
|
return FALSE;
|
|
}
|
|
pPI->pchStop++;
|
|
cDigitParsed = 0;
|
|
uBase = 16;
|
|
} // if
|
|
|
|
while (*pPI->pchStop != _T('\0'))
|
|
{
|
|
pchDigit = wcschr(rgchHexDigits, *pPI->pchStop);
|
|
if (pchDigit == NULL)
|
|
{
|
|
if (pPI->nFlags & PI_mskfAllowRandomTail)
|
|
break;
|
|
// Digit not found while random tail not allowed
|
|
pPI->nErrCode = PI_errInvalidInteger;
|
|
return FALSE;
|
|
} // if
|
|
Assert(pchDigit >= rgchHexDigits);
|
|
iDigit = (pchDigit - rgchHexDigits) >> 1;
|
|
Assert(iDigit <= 0x0F);
|
|
if (iDigit >= uBase)
|
|
{
|
|
// Hex digit found while parsing a decimal string
|
|
pPI->nErrCode = PI_errInvalidInteger;
|
|
return FALSE;
|
|
}
|
|
cDigitParsed++;
|
|
uDataT = pPI->uData * uBase + iDigit;
|
|
if (pPI->uData > ((UINT)-1)/10 || uDataT < pPI->uData)
|
|
{
|
|
pPI->nErrCode = PI_errIntegerOverflow;
|
|
return FALSE;
|
|
}
|
|
pPI->uData = uDataT;
|
|
pPI->pchStop++;
|
|
} // while
|
|
|
|
if ((cDigitParsed == 0) && (pPI->nFlags & PI_mskfNoEmptyString))
|
|
{
|
|
// Empty String found while not allowed
|
|
Assert(pPI->uData == 0);
|
|
pPI->nErrCode = PI_errEmptyString;
|
|
return FALSE;
|
|
}
|
|
if (fIsNegative)
|
|
{
|
|
pPI->uData = -(int)pPI->uData;
|
|
}
|
|
if (pPI->nFlags & PI_mskfSingleEntry)
|
|
{
|
|
// Check if there are no more digits at the end of the string
|
|
// Only spaces are allowed
|
|
while (*pPI->pchStop == _T(' '))
|
|
pPI->pchStop++;
|
|
if (*pPI->pchStop != _T('\0'))
|
|
{
|
|
pPI->nErrCode = PI_errInvalidInteger;
|
|
return FALSE;
|
|
}
|
|
}
|
|
return TRUE;
|
|
} // FParseInteger()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// FScanf()
|
|
//
|
|
// Parse a formatted string and extract the values.
|
|
// FScanf() behaves like the well known scanf() function but
|
|
// has range checking and pattern matching. The wildcard (*)
|
|
// may be substituded by "%s" with a NULL pointer.
|
|
//
|
|
// Return TRUE if successful, otherwise return FALSE
|
|
// and set nErrCode to the error found.
|
|
//
|
|
// Formats supported:
|
|
// %d Extract a decimal integer
|
|
// %i Extract a generic integer (decimal or hexadecimal)
|
|
// %u Extract an unsigned decimal integer (return error if minus sign found)
|
|
// %x Force extraction of an hexadecimal integer
|
|
// %s Extract a string
|
|
// %v Void the spaces and tabs characters
|
|
//
|
|
// Note:
|
|
// Fields sfi.pchSrc and sfi.nFlags are preserved during
|
|
// the execution of FScanf().
|
|
//
|
|
// Example:
|
|
// FScanf(&sfi, "%v%s.%s", " \t foobar.txt",
|
|
// OUT szName, LENGTH(szName), OUT szExt, LENGTH(szExt));
|
|
//
|
|
BOOL FScanf(
|
|
SCANF_INFO * pSFI, // INOUT: Control structure
|
|
const TCHAR * pchFmt, // IN: Format template string
|
|
...) // OUT: scanf() arguments
|
|
{
|
|
va_list arglist;
|
|
TParseIntegerInfo pi;
|
|
|
|
Assert(pSFI != 0);
|
|
Assert(pchFmt != NULL);
|
|
Assert(pSFI->pchSrc != NULL);
|
|
|
|
va_start(INOUT arglist, pchFmt);
|
|
pSFI->pchSrcStop = pSFI->pchSrc;
|
|
pSFI->nErrCode = SF_errOK;
|
|
pSFI->cArgParsed = 0;
|
|
|
|
while (TRUE)
|
|
{
|
|
switch (*pchFmt++)
|
|
{
|
|
case 0: // End of string
|
|
return TRUE;
|
|
|
|
case '%':
|
|
switch (*pchFmt++)
|
|
{
|
|
case '%': // "%%"
|
|
if (*pSFI->pchSrcStop++ != '%')
|
|
{
|
|
pSFI->pchSrcStop--;
|
|
pSFI->nErrCode = SF_errTemplateMismatch;
|
|
return FALSE;
|
|
}
|
|
break;
|
|
|
|
case 'v':
|
|
while (*pSFI->pchSrcStop == ' ' || *pSFI->pchSrcStop == '\t')
|
|
pSFI->pchSrcStop++;
|
|
break;
|
|
|
|
case 'V':
|
|
while ((*pSFI->pchSrcStop != '\0') &&
|
|
(strchrT(szWhiteSpaces, *pSFI->pchSrcStop) != NULL))
|
|
pSFI->pchSrcStop++;
|
|
break;
|
|
|
|
case 'd': // "%d" Decimal integer (signed | unsigned)
|
|
case 'u': // "%u" Decimal unsigned integer
|
|
case 'i': // "%i" Generic integer (decimal | hexadecimal / signed | unsigned)
|
|
case 'x': // "%x" Hexadecimal integer
|
|
{
|
|
int * p;
|
|
|
|
pi.nFlags = PI_mskfNoEmptyString | PI_mskfAllowRandomTail;
|
|
switch (*(pchFmt-1))
|
|
{
|
|
case 'u':
|
|
pi.nFlags |= PI_mskfNoMinusSign;
|
|
break;
|
|
case 'i':
|
|
pi.nFlags |= PI_mskfAllowHexBase;
|
|
break;
|
|
case 'x':
|
|
pi.nFlags |= PI_mskfHexBaseOnly | PI_mskfNoMinusSign;
|
|
} // switch
|
|
pi.pchSrc = pSFI->pchSrcStop;
|
|
if (!FParseInteger(INOUT &pi))
|
|
{
|
|
pSFI->pchSrcStop = pi.pchStop;
|
|
return FALSE;
|
|
} // if
|
|
pSFI->pchSrcStop = pi.pchStop;
|
|
pSFI->cArgParsed++;
|
|
p = (int *)va_arg(arglist, int *);
|
|
Assert(p != NULL);
|
|
*p = pi.uData;
|
|
}
|
|
break; // Integer
|
|
|
|
case 's': // "%s" String
|
|
{
|
|
// To get a clean string, use the format "%v%s%v"
|
|
// which will strip all the spaces and tabs around
|
|
// the string.
|
|
TCHAR * pchDest; // Destination buffer
|
|
int cchDestMax; // Size of destination buffer
|
|
TCHAR chEndScan;
|
|
const TCHAR * pchEndScan = NULL;
|
|
|
|
// Find out the ending character(s)
|
|
if (*pchFmt == '%')
|
|
{
|
|
switch (*(pchFmt+1))
|
|
{
|
|
case 'd':
|
|
case 'u':
|
|
case 'i':
|
|
pchEndScan = szDecimalDigits;
|
|
chEndScan = '\0';
|
|
break;
|
|
case 'v': // %v
|
|
pchEndScan = szSpcTab;
|
|
chEndScan = *(pchFmt+2);
|
|
break;
|
|
case 'V': // %V
|
|
pchEndScan = szWhiteSpaces;
|
|
chEndScan = *(pchFmt+2);
|
|
break;
|
|
case '%': // %%
|
|
chEndScan = '%';
|
|
default:
|
|
Assert(FALSE); // Ambiguous compound format (not supported anyway!)
|
|
} // switch
|
|
}
|
|
else
|
|
{
|
|
chEndScan = *pchFmt;
|
|
} // if...else
|
|
|
|
pSFI->cArgParsed++;
|
|
pchDest = (TCHAR *)va_arg(arglist, TCHAR *);
|
|
if (pchDest != NULL)
|
|
{
|
|
cchDestMax = va_arg(arglist, int) - 1;
|
|
// Verify if the size of destination buffer
|
|
// is a valid size.
|
|
// Otherwise, this may be the address of the
|
|
// next argument
|
|
Assert(cchDestMax > 0 && cchDestMax < 5000);
|
|
|
|
while (cchDestMax-- > 0)
|
|
{
|
|
if (*pSFI->pchSrcStop == chEndScan)
|
|
break;
|
|
else if (*pSFI->pchSrcStop == '\0')
|
|
break;
|
|
else if (pchEndScan != NULL)
|
|
{
|
|
if (strchrT(pchEndScan, *pSFI->pchSrcStop))
|
|
break;
|
|
} // if...else
|
|
// Copy the character into destination buffer
|
|
*pchDest++ = *pSFI->pchSrcStop++;
|
|
}
|
|
*pchDest = '\0';
|
|
} // if
|
|
// Skip the characters until reaching either end character
|
|
while (TRUE)
|
|
{
|
|
if (*pSFI->pchSrcStop == chEndScan)
|
|
break;
|
|
else if (*pSFI->pchSrcStop == '\0')
|
|
break;
|
|
else if (pchEndScan != NULL)
|
|
{
|
|
if (strchrT(pchEndScan, *pSFI->pchSrcStop))
|
|
break;
|
|
} // if...else
|
|
pSFI->pchSrcStop++;
|
|
} // while
|
|
}
|
|
break; // "%s"
|
|
|
|
default:
|
|
// Unknown "%?" format
|
|
Assert(FALSE);
|
|
pSFI->pchSrcStop--;
|
|
|
|
|
|
} // switch
|
|
break; // case '%'
|
|
|
|
default:
|
|
if (*(pchFmt-1) != *pSFI->pchSrcStop++)
|
|
{
|
|
pSFI->pchSrcStop--;
|
|
pSFI->nErrCode = SF_errTemplateMismatch;
|
|
return FALSE;
|
|
}
|
|
} // switch
|
|
|
|
} // while
|
|
|
|
return TRUE;
|
|
} // FScanf()
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// Query the a registry key of type REG_SZ without trowing an exception.
|
|
//
|
|
BOOL
|
|
RegKey_FQueryString(
|
|
HKEY hKey,
|
|
LPCTSTR pszValueName, // IN: Name of the key
|
|
CString& rstrKeyData) // OUT: Value (data) of registry key
|
|
{
|
|
Assert(hKey != NULL);
|
|
Assert(pszValueName != NULL);
|
|
|
|
TCHAR szBufferT[4096];
|
|
DWORD cbBufferLength = sizeof(szBufferT);
|
|
DWORD dwType;
|
|
DWORD dwErr;
|
|
|
|
dwErr = ::RegQueryValueEx(
|
|
hKey,
|
|
pszValueName,
|
|
0,
|
|
OUT &dwType,
|
|
OUT (BYTE *)szBufferT,
|
|
INOUT &cbBufferLength);
|
|
if ((dwErr == ERROR_SUCCESS) && (dwType == REG_SZ))
|
|
{
|
|
rstrKeyData = szBufferT; // Copy the string
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
rstrKeyData.Empty();
|
|
return FALSE;
|
|
}
|
|
} // RegKey_FQueryString()
|
|
|
|
#endif // SNAPIN_PROTOTYPER
|
|
|
|
|
|
DWORD DisplayNameHelper(
|
|
HWND hwndParent,
|
|
BSTR pszMachineName,
|
|
BSTR pszServiceName,
|
|
DWORD dwDesiredAccess,
|
|
SC_HANDLE* phSC,
|
|
BSTR* pbstrServiceDisplayName)
|
|
{
|
|
*phSC = ::OpenSCManager(
|
|
pszMachineName,
|
|
NULL,
|
|
SC_MANAGER_CONNECT);
|
|
if (NULL == *phSC)
|
|
{
|
|
DWORD dwErr = ::GetLastError();
|
|
ASSERT( NO_ERROR != dwErr );
|
|
return dwErr;
|
|
}
|
|
|
|
SC_HANDLE hService = ::OpenService(
|
|
*phSC,
|
|
pszServiceName,
|
|
dwDesiredAccess | SERVICE_QUERY_CONFIG);
|
|
if (NULL == hService)
|
|
{
|
|
DWORD dwErr = ::GetLastError();
|
|
ASSERT( NO_ERROR != dwErr );
|
|
::CloseServiceHandle(*phSC);
|
|
*phSC = NULL;
|
|
return dwErr;
|
|
}
|
|
|
|
union
|
|
{
|
|
// Service config
|
|
QUERY_SERVICE_CONFIG qsc;
|
|
BYTE rgbBufferQsc[SERVICE_cbQueryServiceConfigMax];
|
|
};
|
|
::ZeroMemory(&qsc, max(sizeof(qsc), sizeof(rgbBufferQsc)));
|
|
DWORD cbBytesNeeded = 0;
|
|
if (!::QueryServiceConfigW(
|
|
hService,
|
|
OUT &qsc,
|
|
max(sizeof(qsc), sizeof(rgbBufferQsc)),
|
|
OUT &cbBytesNeeded))
|
|
{
|
|
DWORD dwErr = ::GetLastError();
|
|
ASSERT( NO_ERROR != dwErr );
|
|
::CloseServiceHandle(hService);
|
|
::CloseServiceHandle(*phSC);
|
|
*phSC = NULL;
|
|
return dwErr;
|
|
}
|
|
|
|
*pbstrServiceDisplayName = ::SysAllocString(
|
|
(qsc.lpDisplayName && qsc.lpDisplayName[0])
|
|
? qsc.lpDisplayName
|
|
: pszServiceName);
|
|
if (NULL == *pbstrServiceDisplayName)
|
|
{
|
|
::CloseServiceHandle(hService);
|
|
::CloseServiceHandle(*phSC);
|
|
*phSC = NULL;
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
::CloseServiceHandle(hService);
|
|
return NO_ERROR;
|
|
}
|
|
|
|
HRESULT CStartStopHelper::StartServiceHelper(
|
|
HWND hwndParent,
|
|
BSTR pszMachineName,
|
|
BSTR pszServiceName,
|
|
DWORD dwNumServiceArgs,
|
|
BSTR * lpServiceArgVectors)
|
|
{
|
|
MFC_TRY;
|
|
|
|
if ( ( (NULL != pszMachineName)
|
|
&& ::IsBadStringPtr(pszMachineName,0x7FFFFFFF))
|
|
|| ::IsBadStringPtr(pszServiceName,0x7FFFFFFF))
|
|
{
|
|
ASSERT(FALSE);
|
|
return E_POINTER;
|
|
}
|
|
if (0 < dwNumServiceArgs)
|
|
{
|
|
if (::IsBadReadPtr(lpServiceArgVectors,sizeof(lpServiceArgVectors)))
|
|
{
|
|
ASSERT(FALSE);
|
|
return E_POINTER;
|
|
}
|
|
for (DWORD i = 0; i < dwNumServiceArgs; i++)
|
|
{
|
|
if ( (NULL != lpServiceArgVectors[i])
|
|
&& ::IsBadStringPtr(lpServiceArgVectors[i],0x7FFFFFFF))
|
|
{
|
|
ASSERT(FALSE);
|
|
return E_POINTER;
|
|
}
|
|
}
|
|
}
|
|
|
|
SC_HANDLE hScManager = NULL;
|
|
CComBSTR sbstrServiceDisplayName;
|
|
|
|
DWORD dwErr = DisplayNameHelper(
|
|
hwndParent,
|
|
pszMachineName,
|
|
pszServiceName,
|
|
SERVICE_START,
|
|
&hScManager,
|
|
&sbstrServiceDisplayName);
|
|
if (NO_ERROR != dwErr)
|
|
{
|
|
(void) DoServicesErrMsgBox(
|
|
hwndParent,
|
|
MB_OK | MB_ICONSTOP,
|
|
dwErr,
|
|
IDS_MSG_sss_UNABLE_TO_START_SERVICE,
|
|
pszServiceName,
|
|
(pszMachineName && pszMachineName[0])
|
|
? pszMachineName : (LPCTSTR)g_strLocalMachine,
|
|
L"");
|
|
}
|
|
else
|
|
{
|
|
dwErr = CServiceControlProgress::S_EStartService(
|
|
hwndParent,
|
|
hScManager,
|
|
pszMachineName,
|
|
pszServiceName,
|
|
sbstrServiceDisplayName,
|
|
dwNumServiceArgs,
|
|
(LPCTSTR *)lpServiceArgVectors);
|
|
}
|
|
|
|
if (NULL != hScManager)
|
|
(void) ::CloseServiceHandle( hScManager );
|
|
|
|
switch (dwErr)
|
|
{
|
|
case CServiceControlProgress::errUserCancelStopDependentServices:
|
|
case CServiceControlProgress::errCannotInitialize:
|
|
case CServiceControlProgress::errUserAbort:
|
|
return S_FALSE;
|
|
default:
|
|
break;
|
|
}
|
|
return HRESULT_FROM_WIN32(dwErr);
|
|
|
|
MFC_CATCH;
|
|
}
|
|
|
|
HRESULT CStartStopHelper::ControlServiceHelper(
|
|
HWND hwndParent,
|
|
BSTR pszMachineName,
|
|
BSTR pszServiceName,
|
|
DWORD dwControlCode)
|
|
{
|
|
MFC_TRY;
|
|
|
|
if ( ( (NULL != pszMachineName)
|
|
&& ::IsBadStringPtr(pszMachineName,0x7FFFFFFF))
|
|
|| ::IsBadStringPtr(pszServiceName,0x7FFFFFFF))
|
|
{
|
|
ASSERT(FALSE);
|
|
return E_POINTER;
|
|
}
|
|
|
|
SC_HANDLE hScManager = NULL;
|
|
CComBSTR sbstrServiceDisplayName;
|
|
DWORD dwDesiredAccess = SERVICE_USER_DEFINED_CONTROL;
|
|
UINT idErrorMessageTemplate = IDS_MSG_sss_UNABLE_TO_STOP_SERVICE; // CODEWORK
|
|
switch (dwControlCode)
|
|
{
|
|
case SERVICE_CONTROL_STOP:
|
|
idErrorMessageTemplate = IDS_MSG_sss_UNABLE_TO_STOP_SERVICE;
|
|
dwDesiredAccess = SERVICE_STOP;
|
|
break;
|
|
case SERVICE_CONTROL_PAUSE:
|
|
idErrorMessageTemplate = IDS_MSG_sss_UNABLE_TO_PAUSE_SERVICE;
|
|
dwDesiredAccess = SERVICE_PAUSE_CONTINUE;
|
|
break;
|
|
case SERVICE_CONTROL_CONTINUE:
|
|
idErrorMessageTemplate = IDS_MSG_sss_UNABLE_TO_RESUME_SERVICE;
|
|
dwDesiredAccess = SERVICE_PAUSE_CONTINUE;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
DWORD dwErr = DisplayNameHelper(
|
|
hwndParent,
|
|
pszMachineName,
|
|
pszServiceName,
|
|
dwDesiredAccess,
|
|
&hScManager,
|
|
&sbstrServiceDisplayName);
|
|
if (NO_ERROR != dwErr)
|
|
{
|
|
(void) DoServicesErrMsgBox(
|
|
hwndParent,
|
|
MB_OK | MB_ICONSTOP,
|
|
dwErr,
|
|
idErrorMessageTemplate,
|
|
pszServiceName,
|
|
(pszMachineName && pszMachineName[0])
|
|
? pszMachineName : (LPCTSTR)g_strLocalMachine,
|
|
L"");
|
|
}
|
|
else
|
|
{
|
|
dwErr = CServiceControlProgress::S_EControlService(
|
|
hwndParent,
|
|
hScManager,
|
|
pszMachineName,
|
|
pszServiceName,
|
|
sbstrServiceDisplayName,
|
|
dwControlCode);
|
|
}
|
|
|
|
if (NULL != hScManager)
|
|
(void) ::CloseServiceHandle( hScManager );
|
|
|
|
switch (dwErr)
|
|
{
|
|
case CServiceControlProgress::errUserCancelStopDependentServices:
|
|
case CServiceControlProgress::errCannotInitialize:
|
|
case CServiceControlProgress::errUserAbort:
|
|
return S_FALSE;
|
|
default:
|
|
break;
|
|
}
|
|
return HRESULT_FROM_WIN32(dwErr);
|
|
|
|
MFC_CATCH;
|
|
}
|