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.
1231 lines
27 KiB
1231 lines
27 KiB
/*++
|
|
|
|
Copyright (c) 1993 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
reg.c
|
|
|
|
Abstract:
|
|
|
|
This module provides helpers to call the registry used by both
|
|
the client and server sides of the workstation.
|
|
|
|
Author:
|
|
|
|
Rita Wong (ritaw) 22-Apr-1993
|
|
|
|
--*/
|
|
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
|
|
#include <nt.h>
|
|
#include <ntrtl.h>
|
|
#include <nturtl.h>
|
|
|
|
#include <windef.h>
|
|
#include <winerror.h>
|
|
#include <winbase.h>
|
|
#include <winreg.h>
|
|
#include <winsvc.h>
|
|
|
|
#include <nwsnames.h>
|
|
#include <nwreg.h>
|
|
#include <nwapi.h>
|
|
#include <lmcons.h>
|
|
#include <lmerr.h>
|
|
|
|
#define LMSERVER_LINKAGE_REGKEY L"System\\CurrentControlSet\\Services\\LanmanServer\\Linkage"
|
|
#define OTHERDEPS_VALUENAME L"OtherDependencies"
|
|
#define LANMAN_SERVER L"LanmanServer"
|
|
|
|
//
|
|
// Forward Declare
|
|
//
|
|
|
|
static
|
|
DWORD
|
|
NwRegQueryValueExW(
|
|
IN HKEY hKey,
|
|
IN LPWSTR lpValueName,
|
|
OUT LPDWORD lpReserved,
|
|
OUT LPDWORD lpType,
|
|
OUT LPBYTE lpData,
|
|
IN OUT LPDWORD lpcbData
|
|
);
|
|
|
|
static
|
|
DWORD
|
|
EnumAndDeleteShares(
|
|
VOID
|
|
) ;
|
|
|
|
DWORD
|
|
CalcNullNullSize(
|
|
WCHAR *pszNullNull
|
|
) ;
|
|
|
|
WCHAR *
|
|
FindStringInNullNull(
|
|
WCHAR *pszNullNull,
|
|
WCHAR *pszString
|
|
) ;
|
|
|
|
VOID
|
|
RemoveNWCFromNullNullList(
|
|
WCHAR *OtherDeps
|
|
) ;
|
|
|
|
DWORD RemoveNwcDependency(
|
|
VOID
|
|
) ;
|
|
|
|
|
|
|
|
DWORD
|
|
NwReadRegValue(
|
|
IN HKEY Key,
|
|
IN LPWSTR ValueName,
|
|
OUT LPWSTR *Value
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function allocates the output buffer and reads the requested
|
|
value from the registry into it.
|
|
|
|
Arguments:
|
|
|
|
Key - Supplies opened handle to the key to read from.
|
|
|
|
ValueName - Supplies name of the value to retrieve data.
|
|
|
|
Value - Returns a pointer to the output buffer which points to
|
|
the memory allocated and contains the data read in from the
|
|
registry. This pointer must be freed with LocalFree when done.
|
|
|
|
Return Value:
|
|
|
|
ERROR_NOT_ENOUGH_MEMORY - Failed to create buffer to read value into.
|
|
|
|
Error from registry call.
|
|
|
|
--*/
|
|
{
|
|
LONG RegError;
|
|
DWORD NumRequired = 0;
|
|
DWORD ValueType;
|
|
|
|
|
|
//
|
|
// Set returned buffer pointer to NULL.
|
|
//
|
|
*Value = NULL;
|
|
|
|
RegError = NwRegQueryValueExW(
|
|
Key,
|
|
ValueName,
|
|
NULL,
|
|
&ValueType,
|
|
(LPBYTE) NULL,
|
|
&NumRequired
|
|
);
|
|
|
|
if (RegError != ERROR_SUCCESS && NumRequired > 0) {
|
|
|
|
if ((*Value = (LPWSTR) LocalAlloc(
|
|
LMEM_ZEROINIT,
|
|
(UINT) NumRequired
|
|
)) == NULL) {
|
|
|
|
KdPrint(("NWWORKSTATION: NwReadRegValue: LocalAlloc of size %lu failed %lu\n",
|
|
NumRequired, GetLastError()));
|
|
|
|
return ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
|
|
RegError = NwRegQueryValueExW(
|
|
Key,
|
|
ValueName,
|
|
NULL,
|
|
&ValueType,
|
|
(LPBYTE) *Value,
|
|
&NumRequired
|
|
);
|
|
}
|
|
else if (RegError == ERROR_SUCCESS) {
|
|
KdPrint(("NWWORKSTATION: NwReadRegValue got SUCCESS with NULL buffer."));
|
|
return ERROR_FILE_NOT_FOUND;
|
|
}
|
|
|
|
if (RegError != ERROR_SUCCESS) {
|
|
|
|
if (*Value != NULL) {
|
|
(void) LocalFree((HLOCAL) *Value);
|
|
*Value = NULL;
|
|
}
|
|
|
|
return (DWORD) RegError;
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
|
|
static
|
|
DWORD
|
|
NwRegQueryValueExW(
|
|
IN HKEY hKey,
|
|
IN LPWSTR lpValueName,
|
|
OUT LPDWORD lpReserved,
|
|
OUT LPDWORD lpType,
|
|
OUT LPBYTE lpData,
|
|
IN OUT LPDWORD lpcbData
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine supports the same functionality as Win32 RegQueryValueEx
|
|
API, except that it works. It returns the correct lpcbData value when
|
|
a NULL output buffer is specified.
|
|
|
|
This code is stolen from the service controller.
|
|
|
|
Arguments:
|
|
|
|
same as RegQueryValueEx
|
|
|
|
Return Value:
|
|
|
|
NO_ERROR or reason for failure.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS ntstatus;
|
|
UNICODE_STRING ValueName;
|
|
PKEY_VALUE_FULL_INFORMATION KeyValueInfo;
|
|
DWORD BufSize;
|
|
|
|
|
|
UNREFERENCED_PARAMETER(lpReserved);
|
|
|
|
//
|
|
// Make sure we have a buffer size if the buffer is present.
|
|
//
|
|
if ((ARGUMENT_PRESENT(lpData)) && (! ARGUMENT_PRESENT(lpcbData))) {
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
RtlInitUnicodeString(&ValueName, lpValueName);
|
|
|
|
//
|
|
// Allocate memory for the ValueKeyInfo
|
|
//
|
|
BufSize = *lpcbData + sizeof(KEY_VALUE_FULL_INFORMATION) +
|
|
ValueName.Length
|
|
- sizeof(WCHAR); // subtract memory for 1 char because it's included
|
|
// in the sizeof(KEY_VALUE_FULL_INFORMATION).
|
|
|
|
KeyValueInfo = (PKEY_VALUE_FULL_INFORMATION) LocalAlloc(
|
|
LMEM_ZEROINIT,
|
|
(UINT) BufSize
|
|
);
|
|
|
|
if (KeyValueInfo == NULL) {
|
|
KdPrint(("NWWORKSTATION: NwRegQueryValueExW: LocalAlloc failed %lu\n",
|
|
GetLastError()));
|
|
return ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
|
|
ntstatus = NtQueryValueKey(
|
|
hKey,
|
|
&ValueName,
|
|
KeyValueFullInformation,
|
|
(PVOID) KeyValueInfo,
|
|
(ULONG) BufSize,
|
|
(PULONG) &BufSize
|
|
);
|
|
|
|
if ((NT_SUCCESS(ntstatus) || (ntstatus == STATUS_BUFFER_OVERFLOW))
|
|
&& ARGUMENT_PRESENT(lpcbData)) {
|
|
|
|
*lpcbData = KeyValueInfo->DataLength;
|
|
}
|
|
|
|
if (NT_SUCCESS(ntstatus)) {
|
|
|
|
if (ARGUMENT_PRESENT(lpType)) {
|
|
*lpType = KeyValueInfo->Type;
|
|
}
|
|
|
|
|
|
if (ARGUMENT_PRESENT(lpData)) {
|
|
memcpy(
|
|
lpData,
|
|
(LPBYTE)KeyValueInfo + KeyValueInfo->DataOffset,
|
|
KeyValueInfo->DataLength
|
|
);
|
|
}
|
|
}
|
|
|
|
(void) LocalFree((HLOCAL) KeyValueInfo);
|
|
|
|
return RtlNtStatusToDosError(ntstatus);
|
|
|
|
}
|
|
|
|
VOID
|
|
NwLuidToWStr(
|
|
IN PLUID LogonId,
|
|
OUT LPWSTR LogonIdStr
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine converts a LUID into a string in hex value format so
|
|
that it can be used as a registry key.
|
|
|
|
Arguments:
|
|
|
|
LogonId - Supplies the LUID.
|
|
|
|
LogonIdStr - Receives the string. This routine assumes that this
|
|
buffer is large enough to fit 17 characters.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
swprintf(LogonIdStr, L"%08lx%08lx", LogonId->HighPart, LogonId->LowPart);
|
|
}
|
|
|
|
VOID
|
|
NwWStrToLuid(
|
|
IN LPWSTR LogonIdStr,
|
|
OUT PLUID LogonId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine converts a string in hex value format into a LUID.
|
|
|
|
Arguments:
|
|
|
|
LogonIdStr - Supplies the string.
|
|
|
|
LogonId - Receives the LUID.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
swscanf(LogonIdStr, L"%08lx%08lx", &LogonId->HighPart, &LogonId->LowPart);
|
|
}
|
|
|
|
|
|
DWORD
|
|
NwDeleteInteractiveLogon(
|
|
IN PLUID Id OPTIONAL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine deletes a specific interactive logon ID key in the registry
|
|
if a logon ID is specified, otherwise it deletes all interactive logon
|
|
ID keys.
|
|
|
|
Arguments:
|
|
|
|
Id - Supplies the logon ID to delete. NULL means delete all.
|
|
|
|
Return Status:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
LONG RegError;
|
|
LONG DelError = ERROR_SUCCESS;
|
|
HKEY InteractiveLogonKey;
|
|
|
|
WCHAR LogonIdKey[NW_MAX_LOGON_ID_LEN];
|
|
|
|
|
|
RegError = RegOpenKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
NW_INTERACTIVE_LOGON_REGKEY,
|
|
REG_OPTION_NON_VOLATILE,
|
|
KEY_READ | KEY_WRITE | DELETE,
|
|
&InteractiveLogonKey
|
|
);
|
|
|
|
if (RegError != ERROR_SUCCESS) {
|
|
return RegError;
|
|
}
|
|
|
|
if (ARGUMENT_PRESENT(Id)) {
|
|
|
|
//
|
|
// Delete the key specified.
|
|
//
|
|
NwLuidToWStr(Id, LogonIdKey);
|
|
|
|
DelError = RegDeleteKeyW(InteractiveLogonKey, LogonIdKey);
|
|
|
|
if ( DelError )
|
|
KdPrint((" NwDeleteInteractiveLogon: failed to delete logon %lu\n", DelError));
|
|
|
|
}
|
|
else {
|
|
|
|
//
|
|
// Delete all interactive logon ID keys.
|
|
//
|
|
|
|
do {
|
|
|
|
RegError = RegEnumKeyW(
|
|
InteractiveLogonKey,
|
|
0,
|
|
LogonIdKey,
|
|
sizeof(LogonIdKey) / sizeof(WCHAR)
|
|
);
|
|
|
|
if (RegError == ERROR_SUCCESS) {
|
|
|
|
//
|
|
// Got a logon id key, delete it.
|
|
//
|
|
|
|
DelError = RegDeleteKeyW(InteractiveLogonKey, LogonIdKey);
|
|
}
|
|
else if (RegError != ERROR_NO_MORE_ITEMS) {
|
|
KdPrint((" NwDeleteInteractiveLogon: failed to enum logon IDs %lu\n", RegError));
|
|
}
|
|
|
|
} while (RegError == ERROR_SUCCESS);
|
|
}
|
|
|
|
(void) RegCloseKey(InteractiveLogonKey);
|
|
|
|
return ((DWORD) DelError);
|
|
}
|
|
|
|
VOID
|
|
NwDeleteCurrentUser(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine deletes the current user value under the parameters key.
|
|
|
|
Arguments:
|
|
|
|
None.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
LONG RegError;
|
|
HKEY WkstaKey;
|
|
|
|
//
|
|
// Open HKEY_LOCAL_MACHINE\System\CurrentControlSet\Services
|
|
// \NWCWorkstation\Parameters
|
|
//
|
|
RegError = RegOpenKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
NW_WORKSTATION_REGKEY,
|
|
REG_OPTION_NON_VOLATILE,
|
|
KEY_READ | KEY_WRITE | DELETE,
|
|
&WkstaKey
|
|
);
|
|
|
|
if (RegError != NO_ERROR) {
|
|
KdPrint(("NWPROVAU: NwpInitializeRegistry open NWCWorkstation\\Parameters key unexpected error %lu!\n",
|
|
RegError));
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Delete CurrentUser value first so that the workstation won't be
|
|
// reading this stale value. Ignore error since it may not exist.
|
|
//
|
|
(void) RegDeleteValueW(
|
|
WkstaKey,
|
|
NW_CURRENTUSER_VALUENAME
|
|
);
|
|
|
|
(void) RegCloseKey(WkstaKey);
|
|
}
|
|
|
|
DWORD
|
|
NwDeleteServiceLogon(
|
|
IN PLUID Id OPTIONAL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine deletes a specific service logon ID key in the registry
|
|
if a logon ID is specified, otherwise it deletes all service logon
|
|
ID keys.
|
|
|
|
Arguments:
|
|
|
|
Id - Supplies the logon ID to delete. NULL means delete all.
|
|
|
|
Return Status:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
LONG RegError;
|
|
LONG DelError = STATUS_SUCCESS;
|
|
HKEY ServiceLogonKey;
|
|
|
|
WCHAR LogonIdKey[NW_MAX_LOGON_ID_LEN];
|
|
|
|
|
|
RegError = RegOpenKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
NW_SERVICE_LOGON_REGKEY,
|
|
REG_OPTION_NON_VOLATILE,
|
|
KEY_READ | KEY_WRITE | DELETE,
|
|
&ServiceLogonKey
|
|
);
|
|
|
|
if (RegError != ERROR_SUCCESS) {
|
|
return RegError;
|
|
}
|
|
|
|
if (ARGUMENT_PRESENT(Id)) {
|
|
|
|
//
|
|
// Delete the key specified.
|
|
//
|
|
NwLuidToWStr(Id, LogonIdKey);
|
|
|
|
DelError = RegDeleteKeyW(ServiceLogonKey, LogonIdKey);
|
|
|
|
}
|
|
else {
|
|
|
|
//
|
|
// Delete all service logon ID keys.
|
|
//
|
|
|
|
do {
|
|
|
|
RegError = RegEnumKeyW(
|
|
ServiceLogonKey,
|
|
0,
|
|
LogonIdKey,
|
|
sizeof(LogonIdKey) / sizeof(WCHAR)
|
|
);
|
|
|
|
if (RegError == ERROR_SUCCESS) {
|
|
|
|
//
|
|
// Got a logon id key, delete it.
|
|
//
|
|
|
|
DelError = RegDeleteKeyW(ServiceLogonKey, LogonIdKey);
|
|
}
|
|
else if (RegError != ERROR_NO_MORE_ITEMS) {
|
|
KdPrint((" NwDeleteServiceLogon: failed to enum logon IDs %lu\n", RegError));
|
|
}
|
|
|
|
} while (RegError == ERROR_SUCCESS);
|
|
}
|
|
|
|
(void) RegCloseKey(ServiceLogonKey);
|
|
|
|
return ((DWORD) DelError);
|
|
}
|
|
|
|
|
|
|
|
DWORD
|
|
NwpRegisterGatewayShare(
|
|
IN LPWSTR ShareName,
|
|
IN LPWSTR DriveName
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine remembers that a gateway share has been created so
|
|
that it can be cleanup up when NWCS is uninstalled.
|
|
|
|
Arguments:
|
|
|
|
ShareName - name of share
|
|
DriveName - name of drive that is shared
|
|
|
|
Return Status:
|
|
|
|
Win32 error of any failure.
|
|
|
|
--*/
|
|
{
|
|
DWORD status ;
|
|
|
|
|
|
//
|
|
// make sure we have valid parameters
|
|
//
|
|
if (ShareName && DriveName)
|
|
{
|
|
HKEY hKey ;
|
|
DWORD dwDisposition ;
|
|
|
|
//
|
|
//
|
|
// Open HKEY_LOCAL_MACHINE\System\CurrentControlSet\Services
|
|
// \NWCWorkstation\Shares (create it if not there)
|
|
//
|
|
status = RegCreateKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
NW_WORKSTATION_GATEWAY_SHARES,
|
|
0,
|
|
L"",
|
|
REG_OPTION_NON_VOLATILE,
|
|
KEY_WRITE, // desired access
|
|
NULL, // default security
|
|
&hKey,
|
|
&dwDisposition // ignored
|
|
);
|
|
|
|
if ( status )
|
|
return status ;
|
|
|
|
//
|
|
// wtite out value with valuename=sharename, valuedata=drive
|
|
//
|
|
status = RegSetValueExW(
|
|
hKey,
|
|
ShareName,
|
|
0,
|
|
REG_SZ,
|
|
(LPBYTE) DriveName,
|
|
(wcslen(DriveName)+1) * sizeof(WCHAR)) ;
|
|
|
|
(void) RegCloseKey( hKey );
|
|
}
|
|
else
|
|
status = ERROR_INVALID_PARAMETER ;
|
|
|
|
return status ;
|
|
|
|
}
|
|
|
|
DWORD
|
|
NwpCleanupGatewayShares(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine cleans up all persistent share info and also tidies
|
|
up the registry for NWCS. Later is not needed in uninstall, but is
|
|
there so we have a single routine that completely disables the
|
|
gateway.
|
|
|
|
Arguments:
|
|
|
|
None.
|
|
|
|
Return Status:
|
|
|
|
Win32 error for failed APIs.
|
|
|
|
--*/
|
|
{
|
|
DWORD status, FinalStatus = NO_ERROR ;
|
|
HKEY WkstaKey = NULL,
|
|
ServerLinkageKey = NULL ;
|
|
LPWSTR OtherDeps = NULL ;
|
|
|
|
//
|
|
// Enumeratre and delete all shares
|
|
//
|
|
FinalStatus = status = EnumAndDeleteShares() ;
|
|
|
|
//
|
|
// if update registry by cleaning out both Drive and Shares keys.
|
|
// ignore return values here. the keys may not be present.
|
|
//
|
|
(void) RegDeleteKeyW(
|
|
HKEY_LOCAL_MACHINE,
|
|
NW_WORKSTATION_GATEWAY_DRIVES
|
|
) ;
|
|
|
|
(void) RegDeleteKeyW(
|
|
HKEY_LOCAL_MACHINE,
|
|
NW_WORKSTATION_GATEWAY_SHARES
|
|
) ;
|
|
|
|
//
|
|
// Open HKEY_LOCAL_MACHINE\System\CurrentControlSet\Services
|
|
// \NWCWorkstation\Parameters
|
|
//
|
|
status = RegOpenKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
NW_WORKSTATION_REGKEY,
|
|
REG_OPTION_NON_VOLATILE, // options
|
|
KEY_WRITE, // desired access
|
|
&WkstaKey
|
|
);
|
|
|
|
if (status == ERROR_SUCCESS)
|
|
{
|
|
//
|
|
// delete the gateway account and gateway enabled flag.
|
|
// ignore failures here (the values may not be present)
|
|
//
|
|
(void) RegDeleteValueW(
|
|
WkstaKey,
|
|
NW_GATEWAYACCOUNT_VALUENAME
|
|
) ;
|
|
(void) RegDeleteValueW(
|
|
WkstaKey,
|
|
NW_GATEWAY_ENABLE
|
|
) ;
|
|
|
|
(void) RegCloseKey( WkstaKey );
|
|
}
|
|
|
|
//
|
|
// store new status if necessary
|
|
//
|
|
if (FinalStatus == NO_ERROR)
|
|
FinalStatus = status ;
|
|
|
|
//
|
|
// Open HKEY_LOCAL_MACHINE\System\CurrentControlSet\Services
|
|
// \LanmanServer\Linkage
|
|
//
|
|
status = RegOpenKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
LMSERVER_LINKAGE_REGKEY,
|
|
REG_OPTION_NON_VOLATILE, // options
|
|
KEY_WRITE | KEY_READ, // desired access
|
|
&ServerLinkageKey
|
|
);
|
|
|
|
if (status == ERROR_SUCCESS)
|
|
{
|
|
//
|
|
// remove us from the OtherDependencies.
|
|
// ignore read failures here (it may not be present)
|
|
//
|
|
status = NwReadRegValue(
|
|
ServerLinkageKey,
|
|
OTHERDEPS_VALUENAME,
|
|
&OtherDeps
|
|
);
|
|
|
|
if (status == NO_ERROR)
|
|
{
|
|
//
|
|
// this call munges the list to remove NWC if there.
|
|
//
|
|
RemoveNWCFromNullNullList(OtherDeps) ;
|
|
|
|
status = RegSetValueExW(
|
|
ServerLinkageKey,
|
|
OTHERDEPS_VALUENAME,
|
|
0,
|
|
REG_MULTI_SZ,
|
|
(BYTE *)OtherDeps,
|
|
CalcNullNullSize(OtherDeps) * sizeof(WCHAR)) ;
|
|
|
|
(void) LocalFree(OtherDeps) ;
|
|
|
|
(void) RemoveNwcDependency() ; // make this happen right away
|
|
// ignore errors - reboot will fix
|
|
}
|
|
else
|
|
{
|
|
status = NO_ERROR ;
|
|
}
|
|
|
|
(void) RegCloseKey( ServerLinkageKey );
|
|
}
|
|
|
|
//
|
|
// store new status if necessary
|
|
//
|
|
if (FinalStatus == NO_ERROR)
|
|
FinalStatus = status ;
|
|
|
|
|
|
return (FinalStatus) ;
|
|
}
|
|
|
|
DWORD
|
|
NwpClearGatewayShare(
|
|
IN LPWSTR ShareName
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine deletes a specific share from the remembered gateway
|
|
shares in the registry.
|
|
|
|
Arguments:
|
|
|
|
ShareName - share value to delete
|
|
|
|
Return Status:
|
|
|
|
Win32 status code.
|
|
|
|
--*/
|
|
{
|
|
DWORD status = NO_ERROR ;
|
|
|
|
//
|
|
// check that paramter is non null
|
|
//
|
|
if (ShareName)
|
|
{
|
|
HKEY hKey ;
|
|
|
|
//
|
|
//
|
|
// Open HKEY_LOCAL_MACHINE\System\CurrentControlSet\Services
|
|
// \NWCWorkstation\Drives
|
|
//
|
|
status = RegOpenKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
NW_WORKSTATION_GATEWAY_SHARES,
|
|
REG_OPTION_NON_VOLATILE, // options
|
|
KEY_WRITE, // desired access
|
|
&hKey
|
|
);
|
|
|
|
if ( status )
|
|
return status ;
|
|
|
|
status = RegDeleteValueW(
|
|
hKey,
|
|
ShareName
|
|
) ;
|
|
|
|
(void) RegCloseKey( hKey );
|
|
}
|
|
else
|
|
status = ERROR_INVALID_PARAMETER ;
|
|
|
|
return status ;
|
|
}
|
|
|
|
typedef NET_API_STATUS (*PF_NETSHAREDEL) (
|
|
LPWSTR server,
|
|
LPWSTR name,
|
|
DWORD reserved) ;
|
|
|
|
#define NETSHAREDELSTICKY_API "NetShareDelSticky"
|
|
#define NETSHAREDEL_API "NetShareDel"
|
|
#define NETAPI_DLL L"NETAPI32"
|
|
|
|
DWORD
|
|
EnumAndDeleteShares(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine removes all persister share info in the server for
|
|
all gateway shares.
|
|
|
|
Arguments:
|
|
|
|
None.
|
|
|
|
Return Status:
|
|
|
|
Win32 error code.
|
|
|
|
--*/
|
|
{
|
|
DWORD err, i, type ;
|
|
HKEY hKey = NULL ;
|
|
FILETIME FileTime ;
|
|
HANDLE hNetapi = NULL ;
|
|
PF_NETSHAREDEL pfNetShareDel, pfNetShareDelSticky ;
|
|
WCHAR Class[256], Share[NNLEN+1], Device[MAX_PATH+1] ;
|
|
DWORD dwClass, dwSubKeys, dwMaxSubKey, dwMaxClass,
|
|
dwValues, dwMaxValueName, dwMaxValueData, dwSDLength,
|
|
dwShareLength, dwDeviceLength ;
|
|
|
|
//
|
|
// load the library so that not everyone needs link to netapi32
|
|
//
|
|
if (!(hNetapi = LoadLibraryW(NETAPI_DLL)))
|
|
return (GetLastError()) ;
|
|
|
|
//
|
|
// get addresses of the 2 functions we are interested in
|
|
//
|
|
if (!(pfNetShareDel = (PF_NETSHAREDEL) GetProcAddress(hNetapi,
|
|
NETSHAREDEL_API)))
|
|
{
|
|
err = GetLastError() ;
|
|
goto ExitPoint ;
|
|
}
|
|
|
|
if (!(pfNetShareDelSticky = (PF_NETSHAREDEL) GetProcAddress(hNetapi,
|
|
NETSHAREDELSTICKY_API)))
|
|
{
|
|
err = GetLastError() ;
|
|
goto ExitPoint ;
|
|
}
|
|
|
|
//
|
|
// Open HKEY_LOCAL_MACHINE\System\CurrentControlSet\Services
|
|
// \NWCGateway\Shares
|
|
//
|
|
err = RegOpenKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
NW_WORKSTATION_GATEWAY_SHARES,
|
|
REG_OPTION_NON_VOLATILE, // options
|
|
KEY_READ, // desired access
|
|
&hKey
|
|
);
|
|
|
|
if ( err )
|
|
goto ExitPoint ;
|
|
|
|
//
|
|
// read the info about that key
|
|
//
|
|
dwClass = sizeof(Class)/sizeof(Class[0]) ;
|
|
err = RegQueryInfoKeyW(hKey,
|
|
Class,
|
|
&dwClass,
|
|
NULL,
|
|
&dwSubKeys,
|
|
&dwMaxSubKey,
|
|
&dwMaxClass,
|
|
&dwValues,
|
|
&dwMaxValueName,
|
|
&dwMaxValueData,
|
|
&dwSDLength,
|
|
&FileTime) ;
|
|
if ( err )
|
|
{
|
|
goto ExitPoint ;
|
|
}
|
|
|
|
//
|
|
// for each value found, we have a share to delete
|
|
//
|
|
for (i = 0; i < dwValues; i++)
|
|
{
|
|
dwShareLength = sizeof(Share)/sizeof(Share[0]) ;
|
|
dwDeviceLength = sizeof(Device) ;
|
|
type = REG_SZ ;
|
|
err = RegEnumValueW(hKey,
|
|
i,
|
|
Share,
|
|
&dwShareLength,
|
|
NULL,
|
|
&type,
|
|
(LPBYTE)Device,
|
|
&dwDeviceLength) ;
|
|
|
|
//
|
|
// cleanup the share. try delete the share proper. if not
|
|
// there, remove the sticky info instead.
|
|
//
|
|
if (!err)
|
|
{
|
|
err = (*pfNetShareDel)(NULL, Share, 0) ;
|
|
|
|
if (err == NERR_NetNameNotFound)
|
|
{
|
|
(void) (*pfNetShareDelSticky)(NULL, Share, 0) ;
|
|
}
|
|
}
|
|
|
|
//
|
|
// ignore errors within the loop. we can to carry on to
|
|
// cleanup as much as possible.
|
|
//
|
|
err = NO_ERROR ;
|
|
}
|
|
|
|
|
|
|
|
ExitPoint:
|
|
|
|
if (hKey)
|
|
(void) RegCloseKey( hKey );
|
|
|
|
if (hNetapi)
|
|
(void) FreeLibrary(hNetapi) ;
|
|
|
|
return err ;
|
|
}
|
|
|
|
|
|
DWORD
|
|
CalcNullNullSize(
|
|
WCHAR *pszNullNull
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Walk thru a NULL NULL string, counting the number of
|
|
characters, including the 2 nulls at the end.
|
|
|
|
Arguments:
|
|
|
|
Pointer to a NULL NULL string
|
|
|
|
Return Status:
|
|
|
|
Count of number of *characters*. See description.
|
|
|
|
--*/
|
|
{
|
|
|
|
DWORD dwSize = 0 ;
|
|
WCHAR *pszTmp = pszNullNull ;
|
|
|
|
if (!pszNullNull)
|
|
return 0 ;
|
|
|
|
while (*pszTmp)
|
|
{
|
|
DWORD dwLen = wcslen(pszTmp) + 1 ;
|
|
|
|
dwSize += dwLen ;
|
|
pszTmp += dwLen ;
|
|
}
|
|
|
|
return (dwSize+1) ;
|
|
}
|
|
|
|
WCHAR *
|
|
FindStringInNullNull(
|
|
WCHAR *pszNullNull,
|
|
WCHAR *pszString
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Walk thru a NULL NULL string, looking for the search string
|
|
|
|
Arguments:
|
|
|
|
pszNullNull: the string list we will search.
|
|
pszString: what we are searching for.
|
|
|
|
Return Status:
|
|
|
|
The start of the string if found. Null, otherwise.
|
|
|
|
--*/
|
|
{
|
|
WCHAR *pszTmp = pszNullNull ;
|
|
|
|
if (!pszNullNull || !*pszNullNull)
|
|
return NULL ;
|
|
|
|
do {
|
|
|
|
if (_wcsicmp(pszTmp,pszString)==0)
|
|
return pszTmp ;
|
|
|
|
pszTmp += wcslen(pszTmp) + 1 ;
|
|
|
|
} while (*pszTmp) ;
|
|
|
|
return NULL ;
|
|
}
|
|
|
|
VOID
|
|
RemoveNWCFromNullNullList(
|
|
WCHAR *OtherDeps
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Remove the NWCWorkstation string from a null null string.
|
|
|
|
Arguments:
|
|
|
|
OtherDeps: the string list we will munge.
|
|
|
|
Return Status:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
LPWSTR pszTmp0, pszTmp1 ;
|
|
|
|
//
|
|
// find the NWCWorkstation string
|
|
//
|
|
pszTmp0 = FindStringInNullNull(OtherDeps, NW_WORKSTATION_SERVICE) ;
|
|
|
|
if (!pszTmp0)
|
|
return ;
|
|
|
|
pszTmp1 = pszTmp0 + wcslen(pszTmp0) + 1 ; // skip past it
|
|
|
|
//
|
|
// shift the rest up
|
|
//
|
|
memmove(pszTmp0, pszTmp1, CalcNullNullSize(pszTmp1)*sizeof(WCHAR)) ;
|
|
}
|
|
|
|
DWORD RemoveNwcDependency(
|
|
VOID
|
|
)
|
|
{
|
|
SC_HANDLE ScManager = NULL;
|
|
SC_HANDLE Service = NULL;
|
|
LPQUERY_SERVICE_CONFIGW lpServiceConfig = NULL;
|
|
DWORD err = NO_ERROR, dwBufferSize = 4096, dwBytesNeeded = 0;
|
|
LPWSTR Deps = NULL ;
|
|
|
|
lpServiceConfig = (LPQUERY_SERVICE_CONFIGW) LocalAlloc(LPTR, dwBufferSize) ;
|
|
|
|
if (lpServiceConfig == NULL) {
|
|
err = GetLastError();
|
|
goto ExitPoint ;
|
|
}
|
|
|
|
ScManager = OpenSCManagerW(
|
|
NULL,
|
|
NULL,
|
|
SC_MANAGER_CONNECT
|
|
);
|
|
|
|
if (ScManager == NULL) {
|
|
|
|
err = GetLastError();
|
|
goto ExitPoint ;
|
|
}
|
|
|
|
Service = OpenServiceW(
|
|
ScManager,
|
|
LANMAN_SERVER,
|
|
(SERVICE_QUERY_CONFIG | SERVICE_CHANGE_CONFIG)
|
|
);
|
|
|
|
if (Service == NULL) {
|
|
err = GetLastError();
|
|
goto ExitPoint ;
|
|
}
|
|
|
|
if (!QueryServiceConfigW(
|
|
Service,
|
|
lpServiceConfig, // address of service config. structure
|
|
dwBufferSize, // size of service configuration buffer
|
|
&dwBytesNeeded // address of variable for bytes needed
|
|
)) {
|
|
|
|
err = GetLastError();
|
|
|
|
if (err == ERROR_INSUFFICIENT_BUFFER) {
|
|
|
|
err = NO_ERROR ;
|
|
dwBufferSize = dwBytesNeeded ;
|
|
lpServiceConfig = (LPQUERY_SERVICE_CONFIGW)
|
|
LocalAlloc(LPTR, dwBufferSize) ;
|
|
|
|
if (lpServiceConfig == NULL) {
|
|
err = GetLastError();
|
|
goto ExitPoint ;
|
|
}
|
|
|
|
if (!QueryServiceConfigW(
|
|
Service,
|
|
lpServiceConfig, // address of service config. structure
|
|
dwBufferSize, // size of service configuration buffer
|
|
&dwBytesNeeded // address of variable for bytes needed
|
|
)) {
|
|
|
|
err = GetLastError();
|
|
}
|
|
}
|
|
|
|
if (err != NO_ERROR) {
|
|
|
|
goto ExitPoint ;
|
|
}
|
|
}
|
|
|
|
Deps = lpServiceConfig->lpDependencies ;
|
|
|
|
RemoveNWCFromNullNullList(Deps) ;
|
|
|
|
if (!ChangeServiceConfigW(
|
|
Service,
|
|
SERVICE_NO_CHANGE, // service type (no change)
|
|
SERVICE_NO_CHANGE, // start type (no change)
|
|
SERVICE_NO_CHANGE, // error control (no change)
|
|
NULL, // binary path name (NULL for no change)
|
|
NULL, // load order group (NULL for no change)
|
|
NULL, // tag id (NULL for no change)
|
|
Deps,
|
|
NULL, // service start name (NULL for no change)
|
|
NULL, // password (NULL for no change)
|
|
NULL // display name (NULL for no change)
|
|
)) {
|
|
|
|
err = GetLastError();
|
|
goto ExitPoint ;
|
|
}
|
|
|
|
|
|
ExitPoint:
|
|
|
|
if (ScManager) {
|
|
|
|
CloseServiceHandle(ScManager);
|
|
}
|
|
|
|
if (Service) {
|
|
|
|
CloseServiceHandle(Service);
|
|
}
|
|
|
|
if (lpServiceConfig) {
|
|
|
|
(void) LocalFree(lpServiceConfig) ;
|
|
}
|
|
|
|
return err ;
|
|
}
|
|
|