Windows2003-3790/admin/cys/lib/common.cpp
2020-09-30 16:53:55 +02:00

1345 lines
28 KiB
C++

// Copyright (c) 2001 Microsoft Corporation
//
// File: common.cpp
//
// Synopsis: Commonly used functions
//
// History: 02/03/2001 JeffJon Created
#include "pch.h"
bool
IsServiceInstalledHelper(const wchar_t* serviceName)
{
LOG_FUNCTION2(IsServiceInstalledHelper, serviceName);
ASSERT(serviceName);
// if we can open the service, then it is installed
bool result = false;
SC_HANDLE hsc =
::OpenSCManager(0, SERVICES_ACTIVE_DATABASE, GENERIC_READ);
if (hsc)
{
SC_HANDLE hs = ::OpenServiceW(hsc, serviceName, GENERIC_READ);
if (hs)
{
::CloseServiceHandle(hs);
result = true;
}
::CloseServiceHandle(hsc);
}
LOG_BOOL(result);
return result;
}
// Wait for a handle to become signalled, or a timeout to expire, or WM_QUIT
// to appear in the message queue. Pump the message queue while we wait.
//
// WARNING: UI should diable itself before calling any function that invokes
// this function, or functions calling this one should guard against
// re-entrance. Otherwise there will be a re-entrancy problem.
//
// e.g. command handler gets button clicked message, calls a func that calls
// this wait function, then user clicks the button again, command handler call
// a func that calls this one, and so on.
DWORD
MyWaitForSendMessageThread(HANDLE hThread, DWORD dwTimeout)
{
LOG_FUNCTION(MyWaitForSendMessageThread);
ASSERT(hThread);
MSG msg;
DWORD dwRet;
DWORD dwEnd = GetTickCount() + dwTimeout;
bool quit = false;
// We will attempt to wait up to dwTimeout for the thread to
// terminate
do
{
dwRet = MsgWaitForMultipleObjects(1, &hThread, FALSE,
dwTimeout, QS_ALLEVENTS | QS_SENDMESSAGE );
if (dwRet == (WAIT_OBJECT_0 + 1))
{
// empty out the message queue. We call DispatchMessage to
// ensure that we still process the WM_PAINT messages.
// DANGER: Make sure that the CYS UI is completely disabled
// or there will be re-entrancy problems here
while (::PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
{
if (msg.message == WM_QUIT)
{
// Need to re-post this so that we know to close CYS
::PostMessage(msg.hwnd, WM_QUIT, 0, 0);
quit = true;
break;
}
::TranslateMessage(&msg);
::DispatchMessage(&msg);
}
// Calculate if we have any more time left in the timeout to
// wait on.
if (dwTimeout != INFINITE)
{
dwTimeout = dwEnd - GetTickCount();
if ((long)dwTimeout <= 0)
{
// No more time left, fail with WAIT_TIMEOUT
dwRet = WAIT_TIMEOUT;
}
}
}
// dwRet == WAIT_OBJECT_0 || dwRet == WAIT_FAILED
// The thread must have exited, so we are happy
//
// dwRet == WAIT_TIMEOUT
// The thread is taking too long to finish, so just
// return and let the caller kill it
} while (dwRet == (WAIT_OBJECT_0 + 1) && !quit);
return(dwRet);
}
HRESULT
CreateAndWaitForProcess(
const String& fullPath,
String& commandLine,
DWORD& exitCode,
bool minimize)
{
LOG_FUNCTION2(CreateAndWaitForProcess, fullPath);
LOG(commandLine);
ASSERT(!fullPath.empty());
exitCode = 0;
HRESULT hr = S_OK;
do
{
PROCESS_INFORMATION procInfo;
memset(&procInfo, 0, sizeof(procInfo));
STARTUPINFO startup;
memset(&startup, 0, sizeof(startup));
if (minimize)
{
LOG(L"Starting minimized");
startup.dwFlags = STARTF_USESHOWWINDOW;
startup.wShowWindow = SW_MINIMIZE;
}
LOG(L"Calling CreateProcess");
LOG(fullPath);
LOG(commandLine);
hr =
Win::CreateProcess(
fullPath,
commandLine,
0,
String(),
startup,
procInfo);
BREAK_ON_FAILED_HRESULT(hr);
ASSERT(procInfo.hProcess);
DWORD dwRet = MyWaitForSendMessageThread(procInfo.hProcess, INFINITE);
ASSERT(dwRet == WAIT_OBJECT_0);
hr = Win::GetExitCodeProcess(procInfo.hProcess, exitCode);
BREAK_ON_FAILED_HRESULT(hr);
Win::CloseHandle(procInfo.hThread);
Win::CloseHandle(procInfo.hProcess);
}
while (0);
LOG(String::format(L"exit code = %1!x!", exitCode));
LOG_HRESULT(hr);
return hr;
}
HRESULT
MyCreateProcess(
const String& fullPath,
String& commandline)
{
LOG_FUNCTION2(MyCreateProcess, fullPath);
LOG(commandline);
ASSERT(!fullPath.empty());
HRESULT hr = S_OK;
do
{
PROCESS_INFORMATION procInfo;
memset(&procInfo, 0, sizeof(procInfo));
STARTUPINFO startup;
memset(&startup, 0, sizeof(startup));
LOG(L"Calling CreateProcess");
hr =
Win::CreateProcess(
fullPath,
commandline,
0,
String(),
startup,
procInfo);
BREAK_ON_FAILED_HRESULT(hr);
ASSERT(procInfo.hProcess);
Win::CloseHandle(procInfo.hThread);
Win::CloseHandle(procInfo.hProcess);
}
while (0);
LOG_HRESULT(hr);
return hr;
}
bool
IsKeyValuePresent(RegistryKey& key, const String& valueKey)
{
LOG_FUNCTION(IsKeyValuePresent);
bool result = false;
do
{
String value;
HRESULT hr = key.GetValue(valueKey, value);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to read regkey %1 because: hr = %2!x!",
valueKey,
hr));
break;
}
if (!value.empty())
{
result = true;
break;
}
} while (false);
LOG_BOOL(result);
return result;
}
bool
GetRegKeyValue(
const String& keyName,
const String& value,
String& resultString,
HKEY parentKey)
{
LOG_FUNCTION(GetRegKeyValue);
bool result = true;
do
{
HRESULT hr = S_OK;
RegistryKey key;
hr = key.Open(parentKey, keyName);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to open regkey %1 because: hr = %2!x!",
keyName.c_str(),
hr));
result = false;
break;
}
hr = key.GetValue(value, resultString);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to read regkey %1 because: hr = %2!x!",
value.c_str(),
hr));
result = false;
break;
}
LOG(String::format(
L"Value of key: %1",
resultString.c_str()));
} while (false);
LOG_BOOL(result);
return result;
}
bool
GetRegKeyValue(
const String& keyName,
const String& value,
DWORD& resultValue,
HKEY parentKey)
{
LOG_FUNCTION(GetRegKeyValue);
bool result = true;
do
{
HRESULT hr = S_OK;
RegistryKey key;
hr = key.Open(parentKey, keyName);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to open regkey %1 because: hr = %2!x!",
keyName.c_str(),
hr));
result = false;
break;
}
hr = key.GetValue(value, resultValue);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to read regkey %1 because: hr = %2!x!",
value.c_str(),
hr));
result = false;
break;
}
LOG(String::format(
L"Key value: %1!d!",
resultValue));
} while (false);
LOG_BOOL(result);
return result;
}
bool
SetRegKeyValue(
const String& keyName,
const String& value,
const String& newString,
HKEY parentKey,
bool create
)
{
LOG_FUNCTION(SetRegKeyValue);
bool result = true;
do
{
HRESULT hr = S_OK;
RegistryKey key;
if (create)
{
hr = key.Create(parentKey, keyName);
}
else
{
hr = key.Open(parentKey, keyName, KEY_SET_VALUE);
}
if (FAILED(hr))
{
LOG(String::format(
L"Failed to open regkey %1 because: hr = %2!x!",
keyName.c_str(),
hr));
result = false;
break;
}
hr = key.SetValue(value, newString);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to write regkey %1 because: hr = %2!x!",
value.c_str(),
hr));
result = false;
break;
}
} while (false);
LOG_BOOL(result);
return result;
}
bool
SetRegKeyValue(
const String& keyName,
const String& value,
DWORD newValue,
HKEY parentKey,
bool create)
{
LOG_FUNCTION(SetRegKeyValue);
bool result = true;
do
{
HRESULT hr = S_OK;
RegistryKey key;
if (create)
{
hr = key.Create(parentKey, keyName);
}
else
{
hr = key.Open(parentKey, keyName, KEY_WRITE);
}
if (FAILED(hr))
{
LOG(String::format(
L"Failed to open regkey %1 because: hr = %2!x!",
keyName.c_str(),
hr));
result = false;
break;
}
hr = key.SetValue(value, newValue);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to write regkey %1 because: hr = %2!x!",
value.c_str(),
hr));
result = false;
break;
}
} while (false);
LOG_BOOL(result);
return result;
}
HRESULT
VariantArrayToStringVector(VARIANT* variant, StringVector& stringList)
{
LOG_FUNCTION(VariantArrayToStringVector);
HRESULT hr = S_OK;
stringList.clear();
do
{
ASSERT(variant);
LOG(String::format(
L"Variant type = 0x%1!x!",
V_VT(variant)));
if (V_VT(variant) == VT_EMPTY ||
V_VT(variant) == VT_NULL)
{
break;
}
ASSERT(V_VT(variant) == (VT_ARRAY | VT_BSTR));
SAFEARRAY* psa = V_ARRAY(variant);
ASSERT(psa);
ASSERT(psa != (SAFEARRAY*)-1);
if (!psa || psa == (SAFEARRAY*)-1)
{
LOG(L"variant not safe array");
break;
}
if (::SafeArrayGetDim(psa) != 1)
{
LOG(L"safe array: wrong number of dimensions");
break;
}
VARTYPE vt = VT_EMPTY;
hr = ::SafeArrayGetVartype(psa, &vt);
if (FAILED(hr) || vt != VT_BSTR)
{
LOG(L"safe array: wrong element type");
break;
}
long lower = 0;
long upper = 0;
hr = ::SafeArrayGetLBound(psa, 1, &lower);
if (FAILED(hr))
{
LOG(L"can't get lower bound");
break;
}
hr = ::SafeArrayGetUBound(psa, 1, &upper);
if (FAILED(hr))
{
LOG(L"can't get upper bound");
break;
}
for (long i = lower; i <= upper; ++i)
{
BSTR item;
hr = ::SafeArrayGetElement(psa, &i, &item);
if (FAILED(hr))
{
LOG(String::format(L"index %1!d! failed", i));
continue;
}
if (item)
{
stringList.push_back(String(item));
}
::SysFreeString(item);
}
}
while (0);
LOG_HRESULT(hr);
return hr;
}
String
IPAddressToString(DWORD ipAddress)
{
String result = String::format(
L"%1!d!.%2!d!.%3!d!.%4!d!",
FIRST_IPADDRESS(ipAddress),
SECOND_IPADDRESS(ipAddress),
THIRD_IPADDRESS(ipAddress),
FOURTH_IPADDRESS(ipAddress));
return result;
}
// Will convert a string in the form of an IP address to
// a DWORD. A return value of INADDR_NONE means that we failed
// to do the conversion
DWORD
StringToIPAddress(const String& stringIPAddress)
{
DWORD result = INADDR_NONE;
// Convert the string to ansi so that we can use inet_addr
// to convert to an IP address DWORD
AnsiString ansi;
String::ConvertResult convertResult = stringIPAddress.convert(ansi);
if (String::CONVERT_SUCCESSFUL == convertResult)
{
// Convert the string to an address
result = inet_addr(ansi.c_str());
}
return result;
}
DWORD
ConvertIPAddressOrder(DWORD address)
{
DWORD result = 0;
result |= (address & 0xff) << 24;
result |= (address & 0xff00) << 8;
result |= (address >> 8) & 0x0000ff00;
result |= (address >> 24) & 0x000000ff;
return result;
}
// This function allocates an array of DWORDs and fills it with the IP addresses
// from the StringList. The caller must free the returned pointer using
// delete[]
DWORD*
StringIPListToDWORDArray(
const StringList& stringIPList,
DWORD& count)
{
// This is an exception throwing new so there is no
// reason to check for NULL
count = 0;
DWORD ipCount = static_cast<DWORD>(stringIPList.size());
DWORD* array = new DWORD[ipCount];
try
{
// Copy the forwarders addresses into the array
for (StringList::iterator itr = stringIPList.begin();
itr != stringIPList.end();
++itr)
{
if (!(*itr).empty())
{
DWORD newAddress = StringToIPAddress(*itr);
if (newAddress != INADDR_NONE)
{
array[count++] = newAddress;
}
}
}
}
catch (exception &e)
{
// NTRAID#NTBUG9-654260-2002/07/08-artm
// Avoid leaking "array" if encounter any exceptions based on
// std::exception. Among these are std::length_error and std::bad_alloc,
// but there could be others.
delete [] array;
// Don't really know how to handle exceptions at this level, so pass the buck.
throw e;
}
return array;
}
void
CreateInfFileText(
String& infFileText,
unsigned int windowTitleResourceID)
{
LOG_FUNCTION(CreateInfFileText);
infFileText = L"[Version]\n";
infFileText += L"Signature = \"$Windows NT$\"\n";
infFileText += L"[Components]\n";
infFileText += L"NetOC=netoc.dll,NetOcSetupProc,netoc.inf\n";
infFileText += L"[Global]\n";
infFileText += L"WindowTitle=";
infFileText += String::load(windowTitleResourceID, hResourceModuleHandle);
infFileText += L"\n";
infFileText += L"[Strings]\n";
infFileText += L";(empty)";
}
void
CreateUnattendFileText(
String& unattendFileText,
PCWSTR serviceName,
bool install)
{
LOG_FUNCTION(CreateUnattendFileText);
ASSERT(serviceName);
unattendFileText = L"[NetOptionalComponents]\n";
unattendFileText += serviceName;
if (install)
{
unattendFileText += L"=1";
}
else
{
unattendFileText += L"=0";
}
}
HRESULT
GetShellPath(
int folder,
String& shellPath,
HWND hwnd = 0)
{
LOG_FUNCTION(GetShellPath);
HRESULT hr = S_OK;
PWSTR path = 0;
do
{
path = new WCHAR[MAX_PATH];
ZeroMemory(path, sizeof(WCHAR) * MAX_PATH);
hr = ::SHGetFolderPath(
hwnd,
folder,
0,
SHGFP_TYPE_DEFAULT,
path);
if (FAILED(hr))
{
LOG(
String::format(
L"Failed to get shell path: hr = %1!x!",
hr));
break;
}
shellPath = path;
} while(false);
if (path)
{
delete[] path;
path = 0;
}
LOG_HRESULT(hr);
return hr;
}
// Opens the favorites folder and creates a favorite for
// the specified URL
HRESULT
AddURLToFavorites(HWND hwnd, const String& url, const String& fileName)
{
LOG_FUNCTION(AddURLToFavorites);
ASSERT(Win::IsWindow(hwnd));
ASSERT(!url.empty());
ASSERT(!fileName.empty());
HRESULT hr = S_OK;
do
{
String path;
hr = GetShellPath(
CSIDL_FAVORITES,
path);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to get favorites path: hr = %1!x!",
hr));
break;
}
// Create the favorites .url file
String fileContents = L"[InternetShortcut]\r\n";
fileContents += L"URL=";
fileContents += url;
fileContents += L"\r\n";
String fullPath = FS::AppendPath(path, fileName);
if (fullPath.empty())
{
LOG(L"Failed to append path");
hr = E_FAIL;
break;
}
HANDLE h = 0;
hr =
FS::CreateFile(
fullPath,
h,
GENERIC_WRITE,
0,
CREATE_NEW,
FILE_ATTRIBUTE_NORMAL);
if (FAILED(hr))
{
LOG(
String::format(
L"Failed to create file: hr = %1!x!",
hr));
break;
}
// Write contents and end-of-file marker
hr = FS::Write(h, fileContents + L"\032");
if (FAILED(hr))
{
LOG(
String::format(
L"Failed to write contents to file: hr = %1!x!",
hr));
break;
}
} while (false);
LOG_HRESULT(hr);
return hr;
}
void
LaunchMMCConsole(
const String& consoleFile,
String& alternatePath)
{
LOG_FUNCTION2(LaunchMMCConsole, consoleFile);
String fullPath = Win::GetSystemDirectory() + L"\\mmc.exe";
String commandLine = L"\"";
if (!alternatePath.empty())
{
LOG(String::format(L"alternatePath = %1", alternatePath.c_str()));
commandLine += alternatePath;
}
else
{
commandLine += Win::GetSystemDirectory();
}
commandLine += L"\\" + consoleFile + L"\"";
LOG(String::format(L"fullPath = %1", fullPath.c_str()));
LOG(String::format(L"commandLine = %1", commandLine.c_str()));
HRESULT unused = MyCreateProcess(fullPath, commandLine);
ASSERT(SUCCEEDED(unused));
LOG_HRESULT(unused);
}
void
LaunchMYS()
{
LOG_FUNCTION(LaunchMYS);
String fullPath =
Win::GetSystemDirectory() +
L"\\mshta.exe";
String commandLine = L"res://" +
Win::GetSystemDirectory() +
L"\\mys.dll/mys.hta";
LOG(String::format(
L"MYS path = %1",
fullPath.c_str()));
LOG(String::format(
L"MYS commandline = %1",
commandLine.c_str()));
HRESULT unused = MyCreateProcess(fullPath, commandLine);
ASSERT(SUCCEEDED(unused));
LOG_HRESULT(unused);
}
HRESULT
GetAllUsersStartMenu(String& startMenuPath)
{
LOG_FUNCTION(GetAllUsersStartMenu);
HRESULT hr = GetShellPath(
CSIDL_COMMON_STARTMENU,
startMenuPath);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to get the start menu path: hr = %1!x!",
hr));
}
LOG_HRESULT(hr);
return hr;
}
HRESULT
GetAllUsersAdminTools(String& adminToolsPath)
{
LOG_FUNCTION(GetAllUsersAdminTools);
HRESULT hr = GetShellPath(
CSIDL_COMMON_ADMINTOOLS,
adminToolsPath);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to get the admin tools path: hr = %1!x!",
hr));
}
LOG_HRESULT(hr);
return hr;
}
HRESULT
CreateShortcut(
const String& shortcutPath,
const String& target,
const String& description)
{
LOG_FUNCTION(CreateShortcut);
HRESULT hr = S_OK;
do
{
ASSERT(!shortcutPath.empty());
ASSERT(!target.empty());
if (shortcutPath.empty() ||
target.empty())
{
LOG(String::format(
L"A parameter was empty: shortcutPath = %1, target = %2",
shortcutPath.c_str(),
target.c_str()));
hr = E_INVALIDARG;
break;
}
LOG(String::format(
L"shortcutPath = %1",
shortcutPath.c_str()));
LOG(String::format(
L"target = %1",
target.c_str()));
SmartInterface<IShellLink> shellLink;
hr = shellLink.AcquireViaCreateInstance(
CLSID_ShellLink,
0,
CLSCTX_INPROC_SERVER);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to CoCreate IShellLink: hr = %1!x!",
hr));
break;
}
hr = shellLink->SetPath(target.c_str());
if (FAILED(hr))
{
LOG(String::format(
L"Failed to set the target path: hr = %1!x!",
hr));
break;
}
if (!description.empty())
{
LOG(L"Setting description");
hr = shellLink->SetDescription(description.c_str());
if (FAILED(hr))
{
LOG(String::format(
L"Failed to set shortcut description: hr = 0x%1!x!",
hr));
}
}
SmartInterface<IPersistFile> persistFile;
hr = persistFile.AcquireViaQueryInterface(
shellLink,
IID_IPersistFile);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to QI for IPersistFile: hr = %1!x!",
hr));
break;
}
hr = persistFile->Save(shortcutPath.c_str(), FALSE);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to save the shortcut: hr = %1!x!",
hr));
break;
}
} while (false);
LOG_HRESULT(hr);
return hr;
}
int
LinkIndexFromNotifyLPARAM(LPARAM lParam)
{
LOG_FUNCTION(LinkIndexFromNotifyLPARAM);
int result = 0;
do
{
ASSERT(lParam);
if (!lParam)
{
break;
}
NMLINK* linkHeader = reinterpret_cast<NMLINK*>(lParam);
if (!linkHeader)
{
LOG(L"Cast of notify link header failed");
break;
}
result = linkHeader->item.iLink;
} while (false);
LOG(String::format(
L"result = %1!d!",
result));
return result;
}
void
ShowHelp(
const String& helpTopic)
{
LOG_FUNCTION2(
ShowHelp,
helpTopic.c_str());
// NOTE: I am not using Win::HtmlHelp here so that the help
// is actually running in a different process. This
// allows us to close down CYS without closing the help
// window.
String fullPath = Win::GetSystemWindowsDirectory() + L"\\hh.exe";
String commandLine = helpTopic;
HRESULT hr = MyCreateProcess(fullPath, commandLine);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to open help: hr = 0x%1!x!",
hr));
}
}
void
OpenLogFile()
{
LOG_FUNCTION(OpenLogFile);
String fullPath = Win::GetSystemDirectory() + L"\\notepad.exe";
String commandLine = Win::GetWindowsDirectory();
commandLine += L"\\Debug\\";
commandLine += CYS_LOGFILE_NAME;
commandLine += L".log";
HRESULT hr = MyCreateProcess(fullPath, commandLine);
ASSERT(SUCCEEDED(hr));
}
bool
IsLogFilePresent()
{
LOG_FUNCTION(IsLogFilePresent);
bool result = false;
String logfile = Win::GetWindowsDirectory();
logfile += L"\\Debug\\";
logfile += CYS_LOGFILE_NAME;
logfile += L".log";
result = FS::FileExists(logfile);
LOG_BOOL(result);
return result;
}
HRESULT
GetAdminToolsShortcutPath(
String& adminToolsShortcutPath,
const String& linkToAppend)
{
LOG_FUNCTION(GetAdminToolsShortcutPath);
HRESULT hr = S_OK;
String adminToolsPath;
hr = GetAllUsersAdminTools(adminToolsPath);
if (SUCCEEDED(hr))
{
if (!linkToAppend.empty())
{
adminToolsShortcutPath =
FS::AppendPath(
adminToolsPath,
linkToAppend);
}
LOG(String::format(
L"Admin Tools Link = %1",
adminToolsShortcutPath.c_str()));
}
LOG_HRESULT(hr);
return hr;
}
HRESULT
AddShortcutToAdminTools(
const String& target,
unsigned int descriptionID,
unsigned int linkID)
{
LOG_FUNCTION2(
AddShortcutToAdminTools,
target);
HRESULT hr = S_OK;
String description = String::load(descriptionID);
String link = String::load(linkID);
do
{
String adminToolsLinkPath;
hr = GetAdminToolsShortcutPath(
adminToolsLinkPath,
link);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to GetAdminToolsShortcutPath: hr = 0x%1!x!",
hr));
}
else
{
hr = CreateShortcut(
adminToolsLinkPath,
target,
description);
if (FAILED(hr))
{
LOG(String::format(
L"Failed to create admin tools shortcut: hr = %1!x!",
hr));
}
}
} while (false);
LOG_HRESULT(hr);
return hr;
}
bool
IsSpecialShare(const SHARE_INFO_1& shareInfo)
{
LOG_FUNCTION2(
IsSpecialShare,
shareInfo.shi1_netname);
bool result = false;
if (shareInfo.shi1_type == STYPE_DISKTREE)
{
LOG(L"Share is of type STYPE_DISKTREE");
String shareName = shareInfo.shi1_netname;
if (shareName.icompare(CYS_SPECIAL_SHARE_SYSVOL) == 0 ||
shareName.icompare(CYS_SPECIAL_SHARE_NETLOGON) == 0 ||
shareName.icompare(CYS_SPECIAL_SHARE_PRINT) == 0)
{
LOG(L"Share has a special name");
result = true;
}
}
else
{
LOG(L"Share is not of type STYPE_DISKTREE");
result = true;
}
LOG_BOOL(result);
return result;
}
bool
IsNonSpecialSharePresent()
{
LOG_FUNCTION(IsNonSpecialSharePresent);
bool result = false;
SHARE_INFO_1* shareInfoArray = 0;
NET_API_STATUS shareResult = 0;
do
{
DWORD entriesRead = 0;
DWORD totalEntries = 0;
DWORD resumeHandle = 0;
shareResult = NetShareEnum(
0,
1,
reinterpret_cast<BYTE**>(&shareInfoArray),
static_cast<DWORD>(-1),
&entriesRead,
&totalEntries,
&resumeHandle);
if ((shareResult == ERROR_SUCCESS ||
shareResult == ERROR_MORE_DATA) &&
shareInfoArray)
{
for (
DWORD index = 0;
index < entriesRead;
++index)
{
// Look for only normal shares and ignore special shares
// like C$, ADMIN$, and IPC$
if (!IsSpecialShare(shareInfoArray[index]))
{
LOG(String::format(
L"Share found: %1",
shareInfoArray[index].shi1_netname));
result = true;
break;
}
}
}
else
{
LOG(String::format(
L"NetShareEnum failed: result = %1!x!",
shareResult));
}
if (shareInfoArray)
{
NetApiBufferFree(shareInfoArray);
shareInfoArray = 0;
}
if (result)
{
break;
}
} while(shareResult == ERROR_MORE_DATA);
LOG_BOOL(result);
return result;
}