Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

2991 lines
75 KiB

/*++
Copyright (c) 1992 Microsoft Corporation
Module Name:
registry.c
Abstract:
This module contains registry _access routines for the NT server
service.
Author:
Chuck Lenzmeier (chuckl) 19-Mar-1992
Revision History:
--*/
#include "srvsvcp.h"
#include "ssreg.h"
#include "srvconfg.h"
#include <tstr.h>
#include <netevent.h>
//
// Simple MIN and MAX macros. Watch out for side effects!
//
#define MIN(a,b) ( ((a) < (b)) ? (a) : (b) )
#define MAX(a,b) ( ((a) < (b)) ? (b) : (a) )
#define MAX_INTEGER_STRING 32
#define MB * 1024 * 1024
#define INF 0xffffffff
//
// ( u, n )
// u is units to allocate for every n megabytes on a medium server.
//
#define CONFIG_TUPLE_SIZE 2
typedef struct {
DWORD initworkitems[CONFIG_TUPLE_SIZE];
DWORD maxworkitems[CONFIG_TUPLE_SIZE];
DWORD rawworkitems[CONFIG_TUPLE_SIZE];
DWORD maxrawworkitems[CONFIG_TUPLE_SIZE];
DWORD maxpagedmemoryusage[CONFIG_TUPLE_SIZE];
DWORD maxnonpagedmemoryusage[CONFIG_TUPLE_SIZE];
} CONFIG_SERVER_TABLE;
CONFIG_SERVER_TABLE MedSrvCfgTbl = {
//
// ** NOTE ** : If the second column is greater than 4, then
// you will need to add a check to make sure the statistic
// did not drop to zero.
//
// Units / MB
// Parameter
// ---------
//
/* initworkitems */ { 1 , 4 },
/* maxworkitems */ { 4 , 1 },
/* rawworkitems */ { 1 , 4 },
/* maxrawworkitems */ { 4 , 1 },
/* maxpagedmemoryusage */ { 1 , 1 },
/* maxnonpagedmemoryusage */ { 1 , 8 },
};
//
// Minimum configuration system size is 8MB. Anything lower treated
// as if 8 MB.
//
#define MIN_SYSTEM_SIZE 8
//
// A medium server reaches its max at 32M. A small server at 16M.
//
#define MAX_SMALL_SIZE 16
#define MAX_MEDIUM_SIZE 32
//
// Note that the user limit is always -1 (unlimited). Autodisconnect
// always defaults to 15 minutes.
//
//
// Forward declarations
//
NTSTATUS
EnumerateStickyShare (
IN PWSTR ValueName,
IN ULONG ValueType,
IN PVOID ValueData,
IN ULONG ValueLength,
IN PVOID Context,
IN PVOID EntryContext
);
NET_API_STATUS
FillStickyShareInfo(
IN PSRVSVC_SHARE_ENUM_INFO ShareEnumInfo,
IN PSHARE_INFO_502 Shi502
);
NTSTATUS
GetSdFromRegistry(
IN PWSTR ValueName,
IN ULONG ValueType,
IN PVOID ValueData,
IN ULONG ValueLength,
IN PVOID Context,
IN PVOID EntryContext
);
BOOLEAN
GetStickyShareInfo (
IN PWSTR ValueName,
IN ULONG ValueType,
IN PVOID ValueData,
OUT PUNICODE_STRING RemarkString,
OUT PUNICODE_STRING PathString,
OUT PSHARE_INFO_502 shi502,
OUT PDWORD CacheState
);
LONG
LoadParameters (
PWCH Path
);
LONG
LoadSizeParameter (
VOID
);
NTSTATUS
RecreateStickyShare (
IN PWSTR ValueName,
IN ULONG ValueType,
IN PVOID ValueData,
IN ULONG ValueLength,
IN PVOID Context,
IN PVOID EntryContext
);
NTSTATUS
SaveSdToRegistry(
IN PSECURITY_DESCRIPTOR SecurityDescriptor,
IN PWSTR ShareName
);
NTSTATUS
SetSizeParameters (
IN PWSTR ValueName,
IN ULONG ValueType,
IN PVOID ValueData,
IN ULONG ValueLength,
IN PVOID Context,
IN PVOID EntryContext
);
NTSTATUS
SetStickyParameter (
IN PWSTR ValueName,
IN ULONG ValueType,
IN PVOID ValueData,
IN ULONG ValueLength,
IN PVOID Context,
IN PVOID EntryContext
);
#define IsPersonal() IsSuiteVersion(VER_SUITE_PERSONAL)
#define IsWebBlade() IsSuiteVersion(VER_SUITE_BLADE)
#define IsEmbedded() IsSuiteVersion(VER_SUITE_EMBEDDEDNT)
BOOL
IsSuiteVersion(USHORT SuiteMask)
{
OSVERSIONINFOEX Osvi;
DWORD TypeMask;
DWORDLONG ConditionMask;
memset(&Osvi, 0, sizeof(OSVERSIONINFOEX));
Osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX);
Osvi.wSuiteMask = SuiteMask;
TypeMask = VER_SUITENAME;
ConditionMask = 0;
VER_SET_CONDITION(ConditionMask, VER_SUITENAME, VER_OR);
return(VerifyVersionInfo(&Osvi, TypeMask, ConditionMask));
}
ULONG
SsRtlQueryEnvironmentLength (
IN PVOID Environment
)
{
PWCH p;
ULONG length;
p = Environment;
ASSERT( p != NULL );
//
// The environment variable block consists of zero or more null
// terminated ASCII strings. Each string is of the form:
//
// name=value
//
// where the null termination is after the value.
//
while ( *p ) {
while ( *p ) {
p++;
}
p++;
}
p++;
length = (ULONG)((PCHAR)p - (PCHAR)Environment);
//
// Return accumulated length.
//
return length;
}
VOID
SsAddParameterToRegistry (
PFIELD_DESCRIPTOR Field,
PVOID Value
)
{
NTSTATUS status;
PWCH valueName;
DWORD valueType;
LPBYTE valuePtr;
DWORD valueDataLength;
//
// The value name is the parameter name and the value data is the
// parameter value.
//
valueName = Field->FieldName;
switch ( Field->FieldType ) {
case BOOLEAN_FIELD:
case DWORD_FIELD:
valueType = REG_DWORD;
valuePtr = Value;
valueDataLength = sizeof(DWORD);
break;
case LPSTR_FIELD:
valueType = REG_SZ;
valuePtr = *(LPBYTE *)Value;
if ( valuePtr != NULL ) {
valueDataLength = SIZE_WSTR( (PWCH)valuePtr );
} else {
valueDataLength = 0;
}
break;
}
//
// Set the value into the Parameters key.
//
status = RtlWriteRegistryValue(
RTL_REGISTRY_SERVICES,
PARAMETERS_REGISTRY_PATH,
valueName,
valueType,
valuePtr,
valueDataLength
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsAddParameterToRegistry: SetValue failed: %lx; "
"parameter %ws won't stick\n", status, valueName ));
}
}
return;
} // SsAddParameterToRegistry
VOID
SsAddShareToRegistry (
IN PSHARE_INFO_2 ShareInfo2,
IN PSECURITY_DESCRIPTOR SecurityDescriptor OPTIONAL,
IN DWORD CacheState
)
{
NTSTATUS status;
PWCH valueName;
PVOID environment;
UNICODE_STRING nameString;
UNICODE_STRING valueString;
WCHAR integerString[MAX_INTEGER_STRING + 1];
ULONG environmentLength;
//
// Build the value name and data strings. The value name is the
// share name (netname), while the value data is share information
// in REG_MULTI_SZ format. To build the value data, we use the
// RTL environment routines.
//
valueName = ShareInfo2->shi2_netname;
status = RtlCreateEnvironment( FALSE, &environment );
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsAddShareToRegistry: CreateEnvironment failed: %lx; "
"share %ws won't stick\n", status, valueName ));
}
goto exit1;
}
RtlInitUnicodeString( &nameString, PATH_VARIABLE_NAME );
RtlInitUnicodeString( &valueString, ShareInfo2->shi2_path );
status = RtlSetEnvironmentVariable(
&environment,
&nameString,
&valueString
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsAddShareToRegistry: SetEnvironment failed: %lx; "
"share %s won't stick\n", status, valueName ));
}
goto exit2;
}
if ( ShareInfo2->shi2_remark != NULL ) {
RtlInitUnicodeString( &nameString, REMARK_VARIABLE_NAME );
RtlInitUnicodeString( &valueString, ShareInfo2->shi2_remark );
status = RtlSetEnvironmentVariable(
&environment,
&nameString,
&valueString
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsAddShareToRegistry: SetEnvironment failed: %lx; "
"share %s won't stick\n", status, valueName ));
}
goto exit2;
}
}
RtlInitUnicodeString( &nameString, TYPE_VARIABLE_NAME );
valueString.Buffer = integerString;
valueString.MaximumLength = (MAX_INTEGER_STRING + 1) * sizeof(WCHAR);
status = RtlIntegerToUnicodeString(
ShareInfo2->shi2_type,
10,
&valueString
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsAddShareToRegistry: IntegerToUnicode failed: %lx; "
"share %ws won't stick\n", status, valueName ));
}
goto exit2;
}
status = RtlSetEnvironmentVariable(
&environment,
&nameString,
&valueString
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsAddShareToRegistry: SetEnvironment failed: %lx; "
"share %s won't stick\n", status, valueName ));
}
goto exit2;
}
RtlInitUnicodeString( &nameString, PERMISSIONS_VARIABLE_NAME );
valueString.Buffer = integerString;
valueString.MaximumLength = (MAX_INTEGER_STRING + 1) * sizeof(WCHAR);
status = RtlIntegerToUnicodeString(
ShareInfo2->shi2_permissions,
10,
&valueString
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsAddShareToRegistry: IntegerToUnicode failed: %lx; "
"share %ws won't stick\n", status, valueName ));
}
goto exit2;
}
status = RtlSetEnvironmentVariable(
&environment,
&nameString,
&valueString
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsAddShareToRegistry: SetEnvironment failed: %lx; "
"share %s won't stick\n", status, valueName ));
}
goto exit2;
}
RtlInitUnicodeString( &nameString, MAXUSES_VARIABLE_NAME );
valueString.Buffer = integerString;
valueString.MaximumLength = (MAX_INTEGER_STRING + 1) * sizeof(WCHAR);
status = RtlIntegerToUnicodeString(
ShareInfo2->shi2_max_uses,
10,
&valueString
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsAddShareToRegistry: IntegerToUnicode failed: %lx; "
"share %ws won't stick\n", status, valueName ));
}
goto exit2;
}
status = RtlSetEnvironmentVariable(
&environment,
&nameString,
&valueString
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsAddShareToRegistry: SetEnvironment failed: %lx; "
"share %s won't stick\n", status, valueName ));
}
goto exit2;
}
//
// Set the CacheState
//
RtlInitUnicodeString( &nameString, CSC_VARIABLE_NAME );
valueString.Buffer = integerString;
valueString.MaximumLength = (MAX_INTEGER_STRING + 1) * sizeof(WCHAR);
status = RtlIntegerToUnicodeString(
CacheState,
10,
&valueString
);
if( !NT_SUCCESS( status ) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsAddShareToRegistry: IntegerToUnicode failed: %lx; "
"share %ws won't stick\n", status, valueName ));
}
goto exit2;
}
status = RtlSetEnvironmentVariable(
&environment,
&nameString,
&valueString
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsAddShareToRegistry: SetEnvironment failed: %lx; "
"share %s won't stick\n", status, valueName ));
}
goto exit2;
}
//
// Set the value into the Shares key.
//
environmentLength = SsRtlQueryEnvironmentLength( environment );
status = RtlWriteRegistryValue(
RTL_REGISTRY_SERVICES,
SHARES_REGISTRY_PATH,
valueName,
REG_MULTI_SZ,
(LPBYTE)environment,
environmentLength
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsAddShareToRegistry: SetValue failed: %lx; share %ws "
"won't stick\n", status, valueName ));
}
}
//
// Save the file security descriptor
//
if ( ARGUMENT_PRESENT( SecurityDescriptor ) ) {
status = SaveSdToRegistry(
SecurityDescriptor,
valueName
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsAddShareToRegistry: SaveSd failed: %lx; share %ws\n"
, status, valueName ));
}
}
}
exit2:
RtlDestroyEnvironment( environment );
exit1:
return;
} // SsAddShareToRegistry
NET_API_STATUS
SsCheckRegistry (
VOID
)
/*++
Routine Description:
This function verifies that the keys used by the server exist.
Arguments:
None.
Return Value:
NET_API_STATUS - success/failure of the operation.
--*/
{
NTSTATUS status;
LPWSTR subStrings[1];
//
// Verify the existence of the main server service key. If this
// fails, the server service fails to start.
//
status = RtlCheckRegistryKey(
RTL_REGISTRY_SERVICES,
SERVER_REGISTRY_PATH
);
if ( !NT_SUCCESS(status) ) {
subStrings[0] = SERVER_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_KEY_NOT_FOUND,
1,
subStrings,
RtlNtStatusToDosError( status )
);
IF_DEBUG(INITIALIZATION) {
SS_PRINT(( "SsCheckRegistry: main key doesn't exist\n" ));
}
return ERROR_INVALID_PARAMETER; // !!! Need better error
}
//
// Verify the existence of the Linkage subkey. If this fails, the
// server service fails to start.
//
status = RtlCheckRegistryKey(
RTL_REGISTRY_SERVICES,
LINKAGE_REGISTRY_PATH
);
if ( !NT_SUCCESS(status) ) {
subStrings[0] = LINKAGE_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_KEY_NOT_FOUND,
1,
subStrings,
RtlNtStatusToDosError( status )
);
IF_DEBUG(INITIALIZATION) {
SS_PRINT(( "SsCheckRegistry: Linkage subkey doesn't exist\n" ));
}
return ERROR_INVALID_PARAMETER; // !!! Need better error
}
//
// If the Parameters subkey doesn't exist, create it. If it can't
// be created, fail to start the server.
//
status = RtlCheckRegistryKey(
RTL_REGISTRY_SERVICES,
PARAMETERS_REGISTRY_PATH
);
if ( !NT_SUCCESS(status) ) {
status = RtlCreateRegistryKey(
RTL_REGISTRY_SERVICES,
PARAMETERS_REGISTRY_PATH
);
if ( !NT_SUCCESS(status) ) {
subStrings[0] = PARAMETERS_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_KEY_NOT_CREATED,
1,
subStrings,
RtlNtStatusToDosError( status )
);
IF_DEBUG(INITIALIZATION) {
SS_PRINT(( "SsCheckRegistry: Can't create Parameters subkey: "
"%lx\n", status ));
}
return RtlNtStatusToDosError( status );
}
}
//
// Create the key holding the default security descriptors governing server APIs.
// Since we have compiled-in versions for these APIs, it is a non-fatal error
// if we cannot create this key. But we log it anyway.
//
status = RtlCheckRegistryKey(
RTL_REGISTRY_SERVICES,
SHARES_DEFAULT_SECURITY_REGISTRY_PATH
);
if ( !NT_SUCCESS(status) ) {
status = RtlCreateRegistryKey(
RTL_REGISTRY_SERVICES,
SHARES_DEFAULT_SECURITY_REGISTRY_PATH
);
if ( !NT_SUCCESS(status) ) {
subStrings[0] = SHARES_DEFAULT_SECURITY_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_KEY_NOT_CREATED,
1,
subStrings,
RtlNtStatusToDosError( status )
);
IF_DEBUG(INITIALIZATION) {
SS_PRINT(( "SsCheckRegistry: Can't create DefaultSecurityRegistry subkey: "
"%lx\n", status ));
}
}
}
{
LONG error;
HKEY handle;
GUID Guid;
//
// Make sure the GUID_VARIABLE_NAME value is there and contains a valid GUID.
//
error = RegOpenKeyEx( HKEY_LOCAL_MACHINE,
FULL_PARAMETERS_REGISTRY_PATH,
0,
KEY_ALL_ACCESS,
&handle
);
if( error == ERROR_SUCCESS ) {
DWORD type;
DWORD size = sizeof( Guid );
error = RegQueryValueEx( handle,
GUID_VARIABLE_NAME,
NULL,
&type,
(LPBYTE)&Guid,
&size
);
if( error != ERROR_SUCCESS ||
type != REG_BINARY ||
size != sizeof( Guid ) ) {
RPC_STATUS RpcStatus;
//
// We could not read it, or it's not a valid UUID.
// Blow it away and reset
//
RegDeleteValue( handle, GUID_VARIABLE_NAME );
RpcStatus = UuidCreate( &Guid );
if( RpcStatus == RPC_S_OK || RpcStatus == RPC_S_UUID_LOCAL_ONLY ) {
error = RegSetValueEx( handle,
GUID_VARIABLE_NAME,
0,
REG_BINARY,
(LPBYTE)&Guid,
sizeof( Guid )
);
}
SsNotifyRdrOfGuid( &Guid );
}
RegCloseKey( handle );
} else {
RtlZeroMemory( &Guid, sizeof( Guid ) );
}
SsData.ServerInfo598.sv598_serverguid = Guid;
}
//
// If the AutotunedParameters subkey doesn't exist, create it. If
// it can't be created, fail to start the server.
//
status = RtlCheckRegistryKey(
RTL_REGISTRY_SERVICES,
AUTOTUNED_REGISTRY_PATH
);
if ( !NT_SUCCESS(status) ) {
status = RtlCreateRegistryKey(
RTL_REGISTRY_SERVICES,
AUTOTUNED_REGISTRY_PATH
);
if ( !NT_SUCCESS(status) ) {
subStrings[0] = AUTOTUNED_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_KEY_NOT_CREATED,
1,
subStrings,
RtlNtStatusToDosError( status )
);
IF_DEBUG(INITIALIZATION) {
SS_PRINT(( "SsCheckRegistry: Can't create AutotunedParameters "
"subkey: %lx\n", status ));
}
return RtlNtStatusToDosError( status );
}
}
//
// If the Shares subkey doesn't exist, create it. If it can't be
// created, fail to start the server.
//
status = RtlCheckRegistryKey(
RTL_REGISTRY_SERVICES,
SHARES_REGISTRY_PATH
);
if ( !NT_SUCCESS(status) ) {
status = RtlCreateRegistryKey(
RTL_REGISTRY_SERVICES,
SHARES_REGISTRY_PATH
);
if ( !NT_SUCCESS(status) ) {
subStrings[0] = SHARES_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_KEY_NOT_CREATED,
1,
subStrings,
RtlNtStatusToDosError( status )
);
IF_DEBUG(INITIALIZATION) {
SS_PRINT(( "SsCheckRegistry: Can't create Shares subkey: "
"%lx\n", status ));
}
return RtlNtStatusToDosError( status );
}
}
//
// If the Shares Security subkey doesn't exist, create it. If it
// can't be created, fail to start the server.
//
status = RtlCheckRegistryKey(
RTL_REGISTRY_SERVICES,
SHARES_SECURITY_REGISTRY_PATH
);
if ( !NT_SUCCESS(status) ) {
status = RtlCreateRegistryKey(
RTL_REGISTRY_SERVICES,
SHARES_SECURITY_REGISTRY_PATH
);
if ( !NT_SUCCESS(status) ) {
subStrings[0] = SHARES_SECURITY_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_KEY_NOT_CREATED,
1,
subStrings,
RtlNtStatusToDosError( status )
);
IF_DEBUG(INITIALIZATION) {
SS_PRINT(( "SsCheckRegistry: Can't create Shares Security subkey: "
"%lx\n", status ));
}
return RtlNtStatusToDosError( status );
}
}
//
// All keys successfully checked.
//
return NO_ERROR;
} // SsCheckRegistry
NET_API_STATUS
SsEnumerateStickyShares (
IN OUT PSRVSVC_SHARE_ENUM_INFO ShareEnumInfo
)
/*++
Routine Description:
Reads the registry to find and return sticky shares.
Arguments:
ShareEnumInfo - points to a structure that contains the parameters
to the NetShareEnumSticky call.
Return Value:
NET_API_STATUS - success/failure of the operation.
--*/
{
NTSTATUS status;
PRTL_QUERY_REGISTRY_TABLE queryTable;
ShareEnumInfo->TotalBytesNeeded = 0;
ShareEnumInfo->TotalEntries = 0;
ShareEnumInfo->EntriesRead = 0;
//
// Initialize the reserve fields. This tells the callback routine,
// how many times it has been called.
//
ShareEnumInfo->ShareEnumIndex = 0;
ShareEnumInfo->StartOfFixedData = (PCHAR)ShareEnumInfo->OutputBuffer;
ShareEnumInfo->EndOfVariableData = (PCHAR)ShareEnumInfo->OutputBuffer +
ShareEnumInfo->OutputBufferLength;
//
// We need to align it since we deal with unicode strings.
//
ShareEnumInfo->EndOfVariableData =
(PCHAR)((ULONG_PTR)ShareEnumInfo->EndOfVariableData & ~1);
//
// Ask the RTL to call us back for each value in the Shares key.
//
queryTable = MIDL_user_allocate( sizeof(RTL_QUERY_REGISTRY_TABLE) * 2 );
if ( queryTable != NULL ) {
queryTable[0].QueryRoutine = EnumerateStickyShare;
queryTable[0].Flags = RTL_QUERY_REGISTRY_NOEXPAND;
queryTable[0].Name = NULL;
queryTable[0].EntryContext = NULL;
queryTable[0].DefaultType = REG_NONE;
queryTable[0].DefaultData = NULL;
queryTable[0].DefaultLength = 0;
queryTable[1].QueryRoutine = NULL;
queryTable[1].Flags = 0;
queryTable[1].Name = NULL;
status = RtlQueryRegistryValues(
RTL_REGISTRY_SERVICES | RTL_REGISTRY_OPTIONAL,
SHARES_REGISTRY_PATH,
queryTable,
ShareEnumInfo,
NULL
);
MIDL_user_free( queryTable );
} else {
status = STATUS_INSUFFICIENT_RESOURCES;
}
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(INITIALIZATION) {
SS_PRINT(( "SsEnumerateStickyShares: RtlQueryRegistryValues "
"failed: %lx\n", status ));
}
return RtlNtStatusToDosError( status );
}
return NO_ERROR;
} // SsEnumerateStickyShares
NET_API_STATUS
SsLoadConfigurationParameters (
VOID
)
/*++
Routine Description:
Reads the registry to get server configuration parameters. These
server parameters must be set before the server FSP has been
started.
Arguments:
None.
Return Value:
NET_API_STATUS - success/failure of the operation.
--*/
{
LONG error;
//
// Get the basic Size parameter, then load autotuned parameters,
// then load manually set parameters. This ordering allows manual
// settings to override autotuning.
//
error = LoadSizeParameter( );
if ( error == NO_ERROR ) {
error = LoadParameters( AUTOTUNED_REGISTRY_PATH );
if ( error == NO_ERROR ) {
error = LoadParameters( PARAMETERS_REGISTRY_PATH );
}
}
//
// The copy read to MDL read switchover must occur at or below the
// SMB buffer size.
//
SsData.ServerInfo598.sv598_mdlreadswitchover =
MIN(
SsData.ServerInfo598.sv598_mdlreadswitchover,
SsData.ServerInfo599.sv599_sizreqbuf);
//
// If they want to require security signatures, it implies enabling them
//
if( SsData.ServerInfo598.sv598_requiresecuritysignature )
{
SsData.ServerInfo598.sv598_enablesecuritysignature = TRUE;
}
//
// Override parameters that cannot be set on WinNT (vs. NTAS).
//
// The server itself also performs most of these overrides, in case
// somebody figures out the FSCTL that changes parameters. We also
// override in the service in order to keep the service's view
// consistent with the server's. If you make any changes here, also
// make them in srv\svcsrv.c.
//
// Embedded does its own parameter validation, so skip it here
if( !IsEmbedded() )
{
if ( SsData.ServerInfo598.sv598_producttype == NtProductWinNt ) {
//
// On WinNT, the maximum value of certain parameters is fixed at
// build time. These include: concurrent users, SMB buffers,
// and threads.
//
#define MINIMIZE(_param,_max) _param = MIN( _param, _max );
MINIMIZE( SsData.ServerInfo102.sv102_users, MAX_USERS_WKSTA );
MINIMIZE( SsData.ServerInfo599.sv599_maxworkitems, MAX_MAXWORKITEMS_WKSTA );
MINIMIZE( SsData.ServerInfo598.sv598_maxthreadsperqueue, MAX_THREADS_WKSTA );
SsData.ServerInfo599.sv599_maxmpxct = DEF_MAXMPXCT_WKSTA;
if( IsPersonal() )
{
MINIMIZE( SsData.ServerInfo102.sv102_users, MAX_USERS_PERSONAL );
}
//
// On WinNT, we do not cache closed RFCBs.
//
SsData.ServerInfo598.sv598_cachedopenlimit = 0;
//
// Sharing of redirected drives is not allowed on WinNT.
//
SsData.ServerInfo599.sv599_enablesharednetdrives = FALSE;
}
if( IsWebBlade() )
{
MINIMIZE( SsData.ServerInfo102.sv102_users, MAX_USERS_WEB_BLADE );
MINIMIZE( SsData.ServerInfo599.sv599_maxworkitems, MAX_MAXWORKITEMS_WKSTA );
MINIMIZE( SsData.ServerInfo598.sv598_maxthreadsperqueue, MAX_THREADS_WKSTA );
}
}
else
{
// If this is a Class 1 basic embedded device, keep our memory consumption lower too
if( SsData.ServerInfo102.sv102_users == MAX_USERS_EMBEDDED )
{
MINIMIZE( SsData.ServerInfo599.sv599_maxworkitems, MAX_MAXWORKITEMS_EMBEDDED );
MINIMIZE( SsData.ServerInfo598.sv598_maxthreadsperqueue, MAX_THREADS_EMBEDDED );
SsData.ServerInfo599.sv599_maxmpxct = DEF_MAXMPXCT_EMBEDDED;
}
}
return error;
} // SsLoadConfigurationParameters
NET_API_STATUS
SsRecreateStickyShares (
VOID
)
/*++
Routine Description:
Reads the registry to find and create sticky shares.
Arguments:
None.
Return Value:
NET_API_STATUS - success/failure of the operation.
--*/
{
NTSTATUS status;
PRTL_QUERY_REGISTRY_TABLE queryTable;
ULONG IterationCount = 0;
//
// Ask the RTL to call us back for each value in the Shares key.
//
queryTable = MIDL_user_allocate( sizeof(RTL_QUERY_REGISTRY_TABLE) * 2 );
if ( queryTable != NULL ) {
queryTable[0].QueryRoutine = RecreateStickyShare;
queryTable[0].Flags = RTL_QUERY_REGISTRY_NOEXPAND;
queryTable[0].Name = NULL;
queryTable[0].EntryContext = NULL;
queryTable[0].DefaultType = REG_NONE;
queryTable[0].DefaultData = NULL;
queryTable[0].DefaultLength = 0;
queryTable[1].QueryRoutine = NULL;
queryTable[1].Flags = 0;
queryTable[1].Name = NULL;
status = RtlQueryRegistryValues(
RTL_REGISTRY_SERVICES | RTL_REGISTRY_OPTIONAL,
SHARES_REGISTRY_PATH,
queryTable,
&IterationCount,
NULL
);
MIDL_user_free( queryTable );
} else {
status = STATUS_INSUFFICIENT_RESOURCES;
}
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(INITIALIZATION) {
SS_PRINT(( "SsRecreateStickyShares: RtlQueryRegistryValues "
"failed: %lx\n", status ));
}
return RtlNtStatusToDosError( status );
}
return NO_ERROR;
} // SsRecreateStickyShares
NET_API_STATUS
SsRemoveShareFromRegistry (
LPWSTR NetName
)
{
NET_API_STATUS error = NO_ERROR;
NTSTATUS status;
PWCH valueName;
//
// The value name is the share name. Remove that value from the
// Shares key.
//
valueName = NetName;
//
// Delete the share security
//
status = RtlDeleteRegistryValue(
RTL_REGISTRY_SERVICES,
SHARES_SECURITY_REGISTRY_PATH,
valueName
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsRemoveShareFromRegistry: Delete Security value failed: %lx; "
"share %ws will return\n", status, valueName ));
}
}
//
// Delete the share
//
status = RtlDeleteRegistryValue(
RTL_REGISTRY_SERVICES,
SHARES_REGISTRY_PATH,
valueName
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SsRemoveShareFromRegistry: DeleteValue failed: %lx; "
"share %ws will return\n", status, valueName ));
}
error = RtlNtStatusToDosError( status );
}
return error;
} // SsRemoveShareFromRegistry
VOID
BindToTransport (
IN LPWSTR TransportName
)
{
NET_API_STATUS error;
SERVER_TRANSPORT_INFO_0 svti0;
RtlZeroMemory( &svti0, sizeof( svti0 ) );
svti0.svti0_transportname = TransportName;
svti0.svti0_transportaddress = SsData.SsServerTransportAddress;
svti0.svti0_transportaddresslength =
ComputeTransportAddressClippedLength(
SsData.SsServerTransportAddress,
SsData.SsServerTransportAddressLength );
//
// Bind to the transport.
//
IF_DEBUG(INITIALIZATION) {
SS_PRINT(( "BindToTransport: binding to transport %ws\n", TransportName ));
}
error = I_NetrServerTransportAddEx( 0, (LPTRANSPORT_INFO)&svti0 );
if ( error != NO_ERROR ) {
DWORD eventId;
LPWSTR subStrings[2];
IF_DEBUG(INITIALIZATION_ERRORS) {
SS_PRINT(( "SsBindToTransports: failed to bind to %ws: "
"%ld\n", TransportName, error ));
}
eventId = (error == ERROR_DUP_NAME || error == ERROR_INVALID_NETNAME ) ?
EVENT_SRV_CANT_BIND_DUP_NAME :
EVENT_SRV_CANT_BIND_TO_TRANSPORT;
subStrings[0] = TransportName;
SsLogEvent(
eventId,
1,
subStrings,
error
);
}
} // BindToTransport
NTSTATUS
BindOptionalNameToTransport (
IN PWSTR ValueName,
IN ULONG ValueType,
IN PVOID ValueData,
IN ULONG ValueLength,
IN PVOID Context,
IN PVOID EntryContext
)
{
SERVER_TRANSPORT_INFO_0 sti;
UCHAR serverName[ MAX_PATH ];
UNICODE_STRING UnicodeName;
NET_API_STATUS error;
LPWSTR subStrings[2];
ULONG namelen;
subStrings[0] = (LPWSTR)ValueData;
subStrings[1] = OPTIONAL_NAMES_VALUE_NAME;
if( ValueType != REG_SZ ) {
//
// Not a string!
//
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
NO_ERROR
);
return STATUS_SUCCESS;
}
UnicodeName.Length = wcslen( (LPWSTR)ValueData ) * sizeof( WCHAR );
UnicodeName.MaximumLength = UnicodeName.Length + sizeof( WCHAR );
UnicodeName.Buffer = (LPWSTR)ValueData;
error = ConvertStringToTransportAddress( &UnicodeName, serverName, &namelen );
if( error != NO_ERROR ) {
//
// Invalid server name!
//
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
error
);
return STATUS_SUCCESS;
}
RtlZeroMemory( &sti, sizeof(sti) );
sti.svti0_transportname = (LPWSTR)Context;
sti.svti0_transportaddress = serverName;
sti.svti0_transportaddresslength = namelen;
error = I_NetrServerTransportAddEx( 0, (LPTRANSPORT_INFO)&sti );
if ( error != NO_ERROR ) {
//
// Could not register the name!
//
subStrings[0] = (LPWSTR)ValueData;
subStrings[1] = OPTIONAL_NAMES_VALUE_NAME;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
error
);
}
return STATUS_SUCCESS;
}
VOID
BindOptionalNames (
IN PWSTR TransportName
)
{
RTL_QUERY_REGISTRY_TABLE queryTable[2];
NTSTATUS status;
//
// We need to iterate over the optional names and bind them to this
// transport.
//
//
// Now see if there any optional bindings we should perform
//
queryTable[0].QueryRoutine = BindOptionalNameToTransport;
queryTable[0].Flags = 0;
queryTable[0].Name = OPTIONAL_NAMES_VALUE_NAME;
queryTable[0].EntryContext = NULL;
queryTable[0].DefaultType = REG_NONE;
queryTable[0].DefaultData = NULL;
queryTable[0].DefaultLength = 0;
queryTable[1].QueryRoutine = NULL;
queryTable[1].Flags = 0;
queryTable[1].Name = NULL;
(void)RtlQueryRegistryValues(
RTL_REGISTRY_SERVICES | RTL_REGISTRY_OPTIONAL,
PARAMETERS_REGISTRY_PATH,
queryTable,
TransportName,
NULL
);
} // BindOptionalNames
NTSTATUS
EnumerateStickyShare (
IN PWSTR ValueName,
IN ULONG ValueType,
IN PVOID ValueData,
IN ULONG ValueLength,
IN PVOID Context,
IN PVOID EntryContext
)
/*++
Routine Description:
Callback routine for SsEnumerateStickyShare. Routine will get information
on share and fill in the output buffer.
Arguments:
ValueName - Name of the share
ValueType - Value type of the share name.
ValueData - Data associated with the ValueName.
Context - Pointer to our enum information structure.
Return Value:
NET_API_STATUS - success/failure of the operation.
--*/
{
NET_API_STATUS error;
SHARE_INFO_502 shi502;
UNICODE_STRING pathString;
UNICODE_STRING remarkString;
PSRVSVC_SHARE_ENUM_INFO enumInfo = (PSRVSVC_SHARE_ENUM_INFO) Context;
DWORD cacheState;
ValueLength, EntryContext;
remarkString.Buffer = NULL;
pathString.Buffer = NULL;
if ( GetStickyShareInfo(
ValueName,
ValueType,
ValueData,
&remarkString,
&pathString,
&shi502,
&cacheState
) ) {
//
// Do the actual add of the share.
//
IF_DEBUG(REGISTRY) {
SS_PRINT(( "EnumerateStickyShares: adding share %ws\n", ValueName ));
}
shi502.shi502_remark = remarkString.Buffer;
shi502.shi502_path = pathString.Buffer;
//
// Skip until we have the right share to resume from
//
if ( (enumInfo->TotalEntries == 0) &&
(enumInfo->ShareEnumIndex < enumInfo->ResumeHandle) ) {
enumInfo->ShareEnumIndex++;
} else {
enumInfo->TotalEntries++;
error = FillStickyShareInfo( enumInfo, &shi502 );
if ( error != NO_ERROR ) {
IF_DEBUG(REGISTRY) {
SS_PRINT(( "EnumerateStickyShares: failed to add share "
"%ws = %wZ: %ld\n", ValueName, &pathString, error ));
}
} else {
enumInfo->EntriesRead++;
enumInfo->ResumeHandle++;
}
}
//
// free buffers allocated by GetStickyShareInfo
//
if ( remarkString.Buffer != NULL ) {
RtlFreeUnicodeString( &remarkString );
}
if ( pathString.Buffer != NULL ) {
RtlFreeUnicodeString( &pathString );
}
if ( shi502.shi502_security_descriptor != NULL ) {
MIDL_user_free( shi502.shi502_security_descriptor );
}
}
return STATUS_SUCCESS;
} // EnumerateStickyShare
NTSTATUS
GetSdFromRegistry(
IN PWSTR ValueName,
IN ULONG ValueType,
IN PVOID ValueData,
IN ULONG ValueLength,
IN PVOID Context,
IN PVOID EntryContext
)
{
NTSTATUS status = STATUS_SUCCESS;
PSECURITY_DESCRIPTOR fileSD = NULL;
PSHARE_INFO_502 shi502 = (PSHARE_INFO_502) Context;
LPWSTR subStrings[1];
EntryContext, ValueName, ValueType;
if ( ValueLength > 0 ) {
fileSD = MIDL_user_allocate( ValueLength );
if ( fileSD == NULL) {
status = STATUS_INSUFFICIENT_RESOURCES;
} else {
RtlCopyMemory(
fileSD,
ValueData,
ValueLength
);
if ( !RtlValidSecurityDescriptor( fileSD ) ) {
subStrings[0] = ValueName;
SsLogEvent(
EVENT_SRV_INVALID_SD,
1,
subStrings,
RtlNtStatusToDosError( status )
);
MIDL_user_free( fileSD );
fileSD = NULL;
status = STATUS_INVALID_SECURITY_DESCR;
}
}
}
shi502->shi502_security_descriptor = fileSD;
return(status);
} // GetSdFromRegistry
BOOLEAN
GetStickyShareInfo (
IN PWSTR ValueName,
IN ULONG ValueType,
IN PVOID ValueData,
OUT PUNICODE_STRING RemarkString,
OUT PUNICODE_STRING PathString,
OUT PSHARE_INFO_502 shi502,
OUT PDWORD CacheState
)
/*++
Routine Description:
Gets share information from the registry.
Arguments:
ValueName - Name of the share
ValueType - Value type of the share name.
ValueData - Data associated with the ValueName.
RemarkString - Upon return, points to a unicode string containing the
user remark for this share.
PathString - Upon return, points to a unicode string containing the
path for this share.
shi502 - Upon return, points to a unicode string containing a
SHARE_INFO_502 structure.
Return Value:
TRUE, if share information successfully retrieved.
FALSE, otherwise.
--*/
{
NTSTATUS status;
UNICODE_STRING variableNameString;
WCHAR integerStringBuffer[35];
UNICODE_STRING unicodeString;
LPWSTR subStrings[2];
PathString->Buffer = NULL;
RemarkString->Buffer = NULL;
shi502->shi502_security_descriptor = NULL;
shi502->shi502_path = NULL;
shi502->shi502_remark = NULL;
shi502->shi502_reserved = 0;
//
// Because the NT server doesn't support share-level security, the
// password is always NULL.
//
shi502->shi502_passwd = NULL;
//
// The value type must be REG_MULTI_SZ, and the value name must not
// be null.
//
if ( (ValueType != REG_MULTI_SZ) ||
(wcslen(ValueName) == 0) ) {
subStrings[0] = ValueName;
subStrings[1] = SHARES_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
NO_ERROR
);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "GetStickyShareInfo: skipping invalid value %ws\n",
ValueName ));
}
goto errorexit;
}
//
// The share name is the value name. The value data describes the
// rest of the information about the share.
//
shi502->shi502_netname = ValueName;
//
// The REG_MULTI_SZ format is the same as that used for storing
// environment variables. Find known share parameters in the data.
//
// Get the share path. It must be present.
//
RtlInitUnicodeString( &variableNameString, PATH_VARIABLE_NAME );
PathString->MaximumLength = 0;
status = RtlQueryEnvironmentVariable_U(
ValueData,
&variableNameString,
PathString
);
if ( status != STATUS_BUFFER_TOO_SMALL ) {
//
// The path is not specified. Ignore this share.
//
subStrings[0] = ValueName;
subStrings[1] = SHARES_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
RtlNtStatusToDosError( status )
);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "GetStickyShareInfo: No path; ignoring share.\n" ));
}
goto errorexit;
}
PathString->MaximumLength = (USHORT)(PathString->Length + sizeof(WCHAR));
PathString->Buffer = MIDL_user_allocate( PathString->MaximumLength );
if ( PathString->Buffer == NULL ) {
//
// No space for path. Ignore this share.
//
subStrings[0] = ValueName;
subStrings[1] = SHARES_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
ERROR_NOT_ENOUGH_MEMORY
);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "GetStickyShareInfo: MIDL_user_allocate failed; ignoring "
"share.\n" ));
}
goto errorexit;
}
status = RtlQueryEnvironmentVariable_U(
ValueData,
&variableNameString,
PathString
);
if ( !NT_SUCCESS(status) ) {
//
// Huh? The second attempt failed. Ignore this share.
//
subStrings[0] = ValueName;
subStrings[1] = SHARES_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
RtlNtStatusToDosError( status )
);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "GetStickyShareInfo: Second query failed! Ignoring "
"share.\n" ));
}
goto errorexit;
}
//
// Get the remark. It may be omitted.
//
RtlInitUnicodeString( &variableNameString, REMARK_VARIABLE_NAME );
RemarkString->MaximumLength = 0;
status = RtlQueryEnvironmentVariable_U(
ValueData,
&variableNameString,
RemarkString
);
if ( status == STATUS_BUFFER_TOO_SMALL ) {
RemarkString->MaximumLength =
(USHORT)(RemarkString->Length + sizeof(WCHAR));
RemarkString->Buffer =
MIDL_user_allocate( RemarkString->MaximumLength );
if ( RemarkString->Buffer == NULL ) {
//
// No space for remark. Ignore this share.
//
subStrings[0] = ValueName;
subStrings[1] = SHARES_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
ERROR_NOT_ENOUGH_MEMORY
);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "GetStickyShareInfo: MIDL_user_allocate failed; ignoring "
"share.\n" ));
}
goto errorexit;
}
status = RtlQueryEnvironmentVariable_U(
ValueData,
&variableNameString,
RemarkString
);
if ( !NT_SUCCESS(status) ) {
//
// Huh? The second attempt failed. Ignore this share.
//
subStrings[0] = ValueName;
subStrings[1] = SHARES_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
RtlNtStatusToDosError( status )
);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "GetStickyShareInfo: Second query failed! "
"Ignoring share.\n" ));
}
goto errorexit;
}
}
//
// Get the share type. It may be omitted.
//
RtlInitUnicodeString( &variableNameString, TYPE_VARIABLE_NAME );
unicodeString.Buffer = integerStringBuffer;
unicodeString.MaximumLength = 35;
status = RtlQueryEnvironmentVariable_U(
ValueData,
&variableNameString,
&unicodeString
);
if ( !NT_SUCCESS(status) ) {
shi502->shi502_type = STYPE_DISKTREE;
} else {
status = RtlUnicodeStringToInteger(
&unicodeString,
0,
&shi502->shi502_type
);
if ( !NT_SUCCESS(status) ) {
subStrings[0] = ValueName;
subStrings[1] = SHARES_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
RtlNtStatusToDosError( status )
);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "GetStickyShareInfo: UnicodeToInteger failed: "
"%lx\n", status ));
}
goto errorexit;
}
}
//
// Get the share permissions. It may be omitted.
//
RtlInitUnicodeString( &variableNameString, PERMISSIONS_VARIABLE_NAME );
status = RtlQueryEnvironmentVariable_U(
ValueData,
&variableNameString,
&unicodeString
);
if ( !NT_SUCCESS(status) ) {
shi502->shi502_permissions = 0;
} else {
DWORD permissions;
status = RtlUnicodeStringToInteger(
&unicodeString,
0,
&permissions
);
if ( !NT_SUCCESS(status) ) {
subStrings[0] = ValueName;
subStrings[1] = SHARES_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
RtlNtStatusToDosError( status )
);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "GetStickyShareInfo: UnicodeToInteger failed: "
"%lx\n", status ));
}
goto errorexit;
}
shi502->shi502_permissions = permissions;
}
//
// Get the maximum number of uses allowed. It may be omitted.
//
RtlInitUnicodeString( &variableNameString, MAXUSES_VARIABLE_NAME );
status = RtlQueryEnvironmentVariable_U(
ValueData,
&variableNameString,
&unicodeString
);
if ( !NT_SUCCESS(status) ) {
shi502->shi502_max_uses = (DWORD)SHI_USES_UNLIMITED;
} else {
status = RtlUnicodeStringToInteger(
&unicodeString,
0,
&shi502->shi502_max_uses
);
if ( !NT_SUCCESS(status) ) {
subStrings[0] = ValueName;
subStrings[1] = SHARES_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
RtlNtStatusToDosError( status )
);
goto errorexit;
}
}
//
// Get the Cacheing flags. It may be omitted.
//
RtlInitUnicodeString( &variableNameString, CSC_VARIABLE_NAME );
*CacheState = 0;
status = RtlQueryEnvironmentVariable_U(
ValueData,
&variableNameString,
&unicodeString
);
if( NT_SUCCESS( status ) ) {
ULONG value;
status = RtlUnicodeStringToInteger(
&unicodeString,
0,
&value
);
if( NT_SUCCESS( status ) ) {
*CacheState = (value & CSC_MASK);
} else {
subStrings[0] = ValueName;
subStrings[1] = SHARES_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
RtlNtStatusToDosError( status )
);
}
}
{
//
// Get the Share file security descriptor
//
RTL_QUERY_REGISTRY_TABLE shareQueryTable[2];
//
// Fill up the query table
//
shareQueryTable[0].QueryRoutine = GetSdFromRegistry;
shareQueryTable[0].Flags = RTL_QUERY_REGISTRY_REQUIRED;
shareQueryTable[0].Name = shi502->shi502_netname;
shareQueryTable[0].EntryContext = NULL;
shareQueryTable[0].DefaultType = REG_NONE;
shareQueryTable[1].QueryRoutine = NULL;
shareQueryTable[1].Flags = 0;
shareQueryTable[1].Name = NULL;
status = RtlQueryRegistryValues(
RTL_REGISTRY_SERVICES | RTL_REGISTRY_OPTIONAL,
SHARES_SECURITY_REGISTRY_PATH,
shareQueryTable,
shi502,
NULL
);
if ( !NT_SUCCESS( status) &&
( status != STATUS_OBJECT_NAME_NOT_FOUND ) ) {
ASSERT(0);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "GetStickyShareInfo: Get file SD: "
"%lx\n", status ));
}
goto errorexit;
}
}
return TRUE;
errorexit:
if ( RemarkString->Buffer != NULL ) {
RtlFreeUnicodeString( RemarkString );
}
if ( PathString->Buffer != NULL ) {
RtlFreeUnicodeString( PathString );
}
if ( shi502->shi502_security_descriptor != NULL ) {
MIDL_user_free( shi502->shi502_security_descriptor );
}
return FALSE;
} // GetStickyShareInfo
BOOLEAN
SsGetDefaultSdFromRegistry (
IN PWCH ValueName,
OUT PSECURITY_DESCRIPTOR *FileSD
)
/*++
Routine Description:
Reads 'ValueName' from the registry and gets the security descriptor
stored there.
Arguments:
ValueName - The name of the registry value in the Parameters section holding the descriptor
FileSD - points to the allocated SD if one was obtained
Return Value:
NTSTATUS - success/failure of the operation.
--*/
{
RTL_QUERY_REGISTRY_TABLE queryTable[2];
SHARE_INFO_502 shi502 = {0};
NTSTATUS status;
*FileSD = NULL;
queryTable[0].QueryRoutine = GetSdFromRegistry;
queryTable[0].Flags = RTL_QUERY_REGISTRY_REQUIRED;
queryTable[0].Name = ValueName;
queryTable[0].EntryContext = NULL;
queryTable[0].DefaultType = REG_NONE;
queryTable[1].QueryRoutine = NULL;
queryTable[1].Flags = 0;
queryTable[1].Name = NULL;
status = RtlQueryRegistryValues(
RTL_REGISTRY_SERVICES | RTL_REGISTRY_OPTIONAL,
SHARES_DEFAULT_SECURITY_REGISTRY_PATH,
queryTable,
&shi502,
NULL
);
if ( NT_SUCCESS( status) ) {
IF_DEBUG(INITIALIZATION) {
SS_PRINT(( "SsGetDefaultSdFromRegistry: using %ws SD from registry.\n",
ValueName ));
}
*FileSD = shi502.shi502_security_descriptor;
return TRUE;
}
return FALSE;
}
VOID
SsWriteDefaultSdToRegistry (
IN PWCH ValueName,
IN PSECURITY_DESCRIPTOR FileSD
)
/*++
Routine Description:
Stores FileSD to 'ValueName' in the registry
Arguments:
ValueName - The name of the registry value in the Parameters section holding the descriptor
FileSD - points to the SD to write
--*/
{
ULONG fileSDLength;
if ( RtlValidSecurityDescriptor( FileSD ) ) {
fileSDLength = RtlLengthSecurityDescriptor( FileSD );
RtlWriteRegistryValue(
RTL_REGISTRY_SERVICES,
SHARES_DEFAULT_SECURITY_REGISTRY_PATH,
ValueName,
REG_BINARY,
(LPBYTE)FileSD,
fileSDLength
);
}
}
LONG
LoadParameters (
PWCH Path
)
/*++
Routine Description:
Reads the registry to get server parameters.
Arguments:
Path - PARAMETERS_REGISTRY_PATH or AUTOTUNED_REGISTRY_PATH
Return Value:
LONG - success/failure of the operation.
--*/
{
NTSTATUS status;
PRTL_QUERY_REGISTRY_TABLE queryTable;
ULONG numberOfBindings = 0;
//
// Ask the RTL to call us back for each value in the appropriate
// key.
//
queryTable = MIDL_user_allocate( sizeof(RTL_QUERY_REGISTRY_TABLE) * 2 );
if ( queryTable != NULL ) {
queryTable[0].QueryRoutine = SetStickyParameter;
queryTable[0].Flags = RTL_QUERY_REGISTRY_NOEXPAND;
queryTable[0].Name = NULL;
queryTable[0].EntryContext = NULL;
queryTable[0].DefaultType = REG_NONE;
queryTable[0].DefaultData = NULL;
queryTable[0].DefaultLength = 0;
queryTable[1].QueryRoutine = NULL;
queryTable[1].Flags = 0;
queryTable[1].Name = NULL;
status = RtlQueryRegistryValues(
RTL_REGISTRY_SERVICES | RTL_REGISTRY_OPTIONAL,
Path,
queryTable,
Path, // Context for SetStickyParameter
NULL
);
MIDL_user_free( queryTable );
} else {
status = STATUS_INSUFFICIENT_RESOURCES;
}
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(INITIALIZATION) {
SS_PRINT(( "LoadParameters: RtlQueryRegistryValues failed: "
"%lx\n", status ));
}
return RtlNtStatusToDosError( status );
}
return NO_ERROR;
} // LoadParameters
LONG
LoadSizeParameter (
VOID
)
/*++
Routine Description:
Reads the registry to get the basic server Size parameter.
Arguments:
None.
Return Value:
LONG - success/failure of the operation.
--*/
{
NTSTATUS status;
PRTL_QUERY_REGISTRY_TABLE queryTable;
ULONG numberOfBindings = 0;
//
// Ask the RTL to call us back if the Size parameter exists.
//
queryTable = MIDL_user_allocate( sizeof(RTL_QUERY_REGISTRY_TABLE) * 2 );
if ( queryTable != NULL ) {
queryTable[0].QueryRoutine = SetSizeParameters;
queryTable[0].Flags = RTL_QUERY_REGISTRY_NOEXPAND;
queryTable[0].Name = SIZE_VALUE_NAME;
queryTable[0].EntryContext = NULL;
queryTable[0].DefaultType = REG_NONE;
queryTable[0].DefaultData = NULL;
queryTable[0].DefaultLength = 0;
queryTable[1].QueryRoutine = NULL;
queryTable[1].Flags = 0;
queryTable[1].Name = NULL;
status = RtlQueryRegistryValues(
RTL_REGISTRY_SERVICES | RTL_REGISTRY_OPTIONAL,
PARAMETERS_REGISTRY_PATH,
queryTable,
NULL,
NULL
);
MIDL_user_free( queryTable );
} else {
status = STATUS_INSUFFICIENT_RESOURCES;
}
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(INITIALIZATION) {
SS_PRINT(( "LoadSizeParameter: RtlQueryRegistryValues failed: "
"%lx\n", status ));
}
return RtlNtStatusToDosError( status );
}
return NO_ERROR;
} // LoadSizeParameter
VOID
PrintShareAnnounce (
LPVOID event
)
{
ULONG i;
//
// Announce ourselves and then wait for awhile.
// If the event gets signaled, terminate the loop and this thread.
// But don't do this forever, since the print subsystem may actually
// get stuck
//
//
// Do it for 15 minutes
//
for( i=0; i < 60; i++ ) {
AnnounceServiceStatus( 1 );
if( WaitForSingleObject( (HANDLE)event, 15*1000 ) != WAIT_TIMEOUT ) {
break;
}
}
}
NTSTATUS
RecreateStickyShare (
IN PWSTR ValueName,
IN ULONG ValueType,
IN PVOID ValueData,
IN ULONG ValueLength,
IN PULONG IterationCount,
IN PVOID EntryContext
)
{
NET_API_STATUS error;
SHARE_INFO_502 shi502;
SHARE_INFO shareInfo;
UNICODE_STRING pathString;
UNICODE_STRING remarkString;
HANDLE threadHandle = NULL;
HANDLE event = NULL;
DWORD CacheState;
SHARE_INFO shareInfoBuffer;
SHARE_INFO_1005 si1005;
ValueLength, EntryContext;
remarkString.Buffer = NULL;
pathString.Buffer = NULL;
if ( GetStickyShareInfo(
ValueName,
ValueType,
ValueData,
&remarkString,
&pathString,
&shi502,
&CacheState
) ) {
//
// Do the actual add of the share.
//
IF_DEBUG(INITIALIZATION) {
SS_PRINT(( "RecreateStickyShares: adding share %ws\n", ValueName ));
}
shi502.shi502_remark = remarkString.Buffer;
shi502.shi502_path = pathString.Buffer;
shareInfo.ShareInfo502 = (LPSHARE_INFO_502_I)&shi502;
if( shi502.shi502_type == STYPE_PRINTQ ) {
//
// A really big problem is that FAX printers can take aribitrarily long to
// complete the eventual OpenPrinter() call which the server will make back
// up to srvsvc. And if we don't announce ourselves in the interval, the
// service controller will presume that we got stuck on startup. Since
// NetrShareAdd() is synchronous, we need to get a different thread to
// announce our service status until NetrShareAdd returns. So, start it
// now. This is most unfortunate.
event = CreateEvent( NULL, TRUE, FALSE, NULL );
if( event != NULL ) {
DWORD threadId;
threadHandle = CreateThread(
NULL,
0,
(LPTHREAD_START_ROUTINE)PrintShareAnnounce,
(LPVOID)event,
0,
&threadId
);
if( threadHandle == NULL ) {
CloseHandle( event );
event = NULL;
}
}
}
//
// RecreateStickyShare is called during server initialization. The service
// controller will presume that we're stuck if we don't update our status
// with it often enough. So every 64 recreated shares we call back to it.
// There's nothing magic about the 64 -- easy to check for, and not too often.
//
if( (shi502.shi502_type == STYPE_PRINTQ && threadHandle == NULL) ||
(++(*IterationCount) & 63 ) == 0 ) {
AnnounceServiceStatus( 1 );
}
error = NetrShareAdd( NULL, 502, &shareInfo, NULL );
if( event != NULL ) {
//
// We created an announcement thread, set the event telling it to terminate
//
SetEvent( event );
//
// Wait for the thread to terminate
//
if( WaitForSingleObject( threadHandle, INFINITE ) == WAIT_FAILED ) {
error = GetLastError();
}
//
// Close the handles
//
CloseHandle( event );
CloseHandle( threadHandle );
}
if ( error != NO_ERROR ) {
IF_DEBUG(INITIALIZATION_ERRORS) {
SS_PRINT(( "RecreateStickyShares: failed to add share "
"%ws = %wZ: %ld\n", ValueName, &pathString, error ));
}
}
//
// If this is a share which can be cached, set the caching flag in the server
//
si1005.shi1005_flags = CacheState;
if( si1005.shi1005_flags ) {
shareInfoBuffer.ShareInfo1005 = &si1005;
NetrShareSetInfo( NULL, ValueName, 1005, &shareInfoBuffer, NULL );
}
//
// free buffers allocated by GetStickyShareInfo
//
if ( remarkString.Buffer != NULL ) {
RtlFreeUnicodeString( &remarkString );
}
if ( pathString.Buffer != NULL ) {
RtlFreeUnicodeString( &pathString );
}
if ( shi502.shi502_security_descriptor != NULL ) {
MIDL_user_free( shi502.shi502_security_descriptor );
}
}
return NO_ERROR;
} // RecreateStickyShare
NTSTATUS
SaveSdToRegistry(
IN PSECURITY_DESCRIPTOR SecurityDescriptor,
IN PWSTR ShareName
)
/*++
Routine Description:
Stores the share file security descriptor in the registry.
Arguments:
SecurityDescriptor - Points to a self-relative security descriptor
describing the access rights for files under this share.
ShareName - Points to a string containing the share name under
which the SD is to be stored.
Return Value:
Status of the operation.
--*/
{
NTSTATUS status;
//
// Store the security descriptor
//
ULONG fileSDLength;
if ( !RtlValidSecurityDescriptor( SecurityDescriptor ) ) {
status = STATUS_INVALID_SECURITY_DESCR;
} else {
fileSDLength = RtlLengthSecurityDescriptor( SecurityDescriptor );
status = RtlWriteRegistryValue(
RTL_REGISTRY_SERVICES,
SHARES_SECURITY_REGISTRY_PATH,
ShareName,
REG_BINARY,
(LPBYTE)SecurityDescriptor,
fileSDLength
);
}
return status;
} // SaveSdToRegistry
NTSTATUS
SetSizeParameters (
IN PWSTR ValueName,
IN ULONG ValueType,
IN PVOID ValueData,
IN ULONG ValueLength,
IN PVOID Context,
IN PVOID EntryContext
)
{
NT_PRODUCT_TYPE productType;
DWORD size;
LPWSTR subStrings[2];
ValueLength, Context, EntryContext;
//
// Get the product type.
//
if ( !RtlGetNtProductType( &productType ) ) {
productType = NtProductWinNt;
}
SsData.ServerInfo598.sv598_producttype = productType;
//
// Make sure that we got called for the right value.
//
ASSERT( _wcsicmp( ValueName, SIZE_VALUE_NAME ) == 0 );
//
// The Size value must be a DWORD, and must be in the following
// range:
//
// 0 -> use defaults
// 1 -> small server (minimize memory usage)
// 2 -> medium server (balance)
// 3 -> large server (maximize connections)
//
if ( ValueType == REG_DWORD ) {
ASSERT( ValueLength == sizeof(DWORD) );
size = *(LPDWORD)ValueData;
}
if ( (ValueType != REG_DWORD) || (size > 3) ) {
subStrings[0] = ValueName;
subStrings[1] = PARAMETERS_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
NO_ERROR
);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SetSizeParameters: skipping invalid value "
"%ws\n", ValueName ));
}
return STATUS_SUCCESS;
}
SsData.ServerInfo598.sv598_serversize = size;
//
// Set appropriate fields based on the product type (Windows NT or
// Advanced Server) and the selected Size. Note that a Size of 0
// doesn't change any of the defaults.
//
// Note that the user limit is always -1 (unlimited). Autodisconnect
// always defaults to 15 minutes.
//
if ( size != 0 ) {
SYSTEM_BASIC_INFORMATION basicInfo;
NTSTATUS status;
ULONG noOfMb;
ULONG factor;
ULONG asFactor;
//
// Get system memory size.
//
status = NtQuerySystemInformation(
SystemBasicInformation,
&basicInfo,
sizeof( SYSTEM_BASIC_INFORMATION ),
NULL
);
if ( status != STATUS_SUCCESS ) {
subStrings[0] = ValueName;
subStrings[1] = PARAMETERS_REGISTRY_PATH;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
NO_ERROR
);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SetSizeParameters: NtQuerySystemInfo failed %x\n",
status ));
}
return STATUS_SUCCESS;
}
//
// Note that we first divide the page size by 512 in order to
// allow for physical memory sizes above 2^32-1. With this
// calculation, we can handle up to two terabytes of physical
// memory. The calculation assumes that the page size is at
// least 512, and is not very accurate if the page size is not
// a power of 2 (very unlikely).
//
ASSERT( basicInfo.PageSize >= 512 );
noOfMb = (((basicInfo.PageSize / 512) *
basicInfo.NumberOfPhysicalPages) +
(1 MB / 512 - 1)) / (1 MB / 512);
//
// Minimum is 8 MB
//
noOfMb = MAX( MIN_SYSTEM_SIZE, noOfMb );
//
// If we have NTAS, and we're set to maximize performance or we have
// lots of memory -- then set the default work item buffer size to
// a larger value. This value has been chosen to work well with our
// implementation of TCP/IP, and shows itself to advantage when doing
// directory emumerations with directories having lots of entries in them.
//
if( productType != NtProductWinNt && ((noOfMb >= 512) && (size == 3)) ) {
SsData.ServerInfo599.sv599_sizreqbuf = DEF_LARGE_SIZREQBUF;
}
//
// Set the maximum for the different sizes
//
if ( size == 1 ) {
noOfMb = MIN( noOfMb, MAX_SMALL_SIZE );
} else if ( size == 2 ) {
noOfMb = MIN( noOfMb, MAX_MEDIUM_SIZE );
}
//
// If small, assume the system size is half of the real one.
// This should give us half the paramater values of a medium server.
// If large, double it. Also set the free connection count.
//
if ( size == 1 ) {
//
// Small
//
factor = (noOfMb + 1) / 2;
SsData.ServerInfo599.sv599_minfreeconnections = 2;
SsData.ServerInfo599.sv599_maxfreeconnections = 2;
} else if ( size == 2 ) {
//
// Balanced
//
factor = noOfMb;
SsData.ServerInfo599.sv599_minfreeconnections = 2;
SsData.ServerInfo599.sv599_maxfreeconnections = 4;
} else {
//
// Large
//
factor = noOfMb * 2;
// Scale up our big servers, this uses the NEW version of small/med/large we picked
// for the server service (< 1 GB, 1-16 GB, >16 GB)
if( noOfMb < 1024 )
{
SsData.ServerInfo599.sv599_minfreeconnections = SRV_MIN_CONNECTIONS_SMALL;
SsData.ServerInfo599.sv599_maxfreeconnections = SRV_MAX_CONNECTIONS_SMALL;
}
else if( noOfMb < 16*1024 )
{
// >= 1 GB memory
SsData.ServerInfo599.sv599_minfreeconnections = SRV_MIN_CONNECTIONS_MEDIUM;
SsData.ServerInfo599.sv599_maxfreeconnections = SRV_MAX_CONNECTIONS_MEDIUM;
}
else {
// >= 16 GB memory
SsData.ServerInfo599.sv599_minfreeconnections = SRV_MIN_CONNECTIONS_LARGE;
SsData.ServerInfo599.sv599_maxfreeconnections = SRV_MAX_CONNECTIONS_LARGE;
}
}
//
// If this is an Advanced Server with at least 24M, some
// parameter will need to be even bigger.
//
asFactor = 1;
if ( (productType != NtProductWinNt) && (noOfMb >= 24) ) asFactor = 2;
//
// Now set the values for a medium server with this much memory.
//
SsData.ServerInfo599.sv599_maxworkitems =
MedSrvCfgTbl.maxworkitems[0] * factor * asFactor /
MedSrvCfgTbl.maxworkitems[1];
SsData.ServerInfo599.sv599_initworkitems =
MedSrvCfgTbl.initworkitems[0] * factor * asFactor /
MedSrvCfgTbl.initworkitems[1];
SsData.ServerInfo599.sv599_rawworkitems =
MedSrvCfgTbl.rawworkitems[0] * factor /
MedSrvCfgTbl.rawworkitems[1];
SsData.ServerInfo598.sv598_maxrawworkitems =
MedSrvCfgTbl.maxrawworkitems[0] * factor * asFactor /
MedSrvCfgTbl.maxrawworkitems[1];
SsData.ServerInfo599.sv599_maxworkitems =
MIN( SsData.ServerInfo599.sv599_maxworkitems, MAX_MAXWORKITEMS );
SsData.ServerInfo599.sv599_initworkitems =
MIN( SsData.ServerInfo599.sv599_initworkitems, MAX_INITWORKITEMS/4 );
SsData.ServerInfo599.sv599_rawworkitems =
MIN( SsData.ServerInfo599.sv599_rawworkitems, MAX_RAWWORKITEMS/4 );
SsData.ServerInfo598.sv598_maxrawworkitems =
MIN( SsData.ServerInfo598.sv598_maxrawworkitems, MAX_MAXRAWWORKITEMS );
if ( (productType != NtProductWinNt) || (size == 3) ) {
SsData.ServerInfo599.sv599_maxpagedmemoryusage = INF;
SsData.ServerInfo599.sv599_maxnonpagedmemoryusage = INF;
} else {
SsData.ServerInfo599.sv599_maxpagedmemoryusage =
MedSrvCfgTbl.maxpagedmemoryusage[0] * factor /
MedSrvCfgTbl.maxpagedmemoryusage[1] MB;
SsData.ServerInfo599.sv599_maxpagedmemoryusage =
MAX( SsData.ServerInfo599.sv599_maxpagedmemoryusage,
MIN_MAXPAGEDMEMORYUSAGE);
SsData.ServerInfo599.sv599_maxnonpagedmemoryusage =
MedSrvCfgTbl.maxnonpagedmemoryusage[0] * factor /
MedSrvCfgTbl.maxnonpagedmemoryusage[1] MB;
SsData.ServerInfo599.sv599_maxnonpagedmemoryusage =
MAX( SsData.ServerInfo599.sv599_maxnonpagedmemoryusage,
MIN_MAXNONPAGEDMEMORYUSAGE);
}
}
return STATUS_SUCCESS;
} // SetSizeParameters
NTSTATUS
SetStickyParameter (
IN PWSTR ValueName,
IN ULONG ValueType,
IN PVOID ValueData,
IN ULONG ValueLength,
IN PVOID Context,
IN PVOID EntryContext
)
{
NET_API_STATUS error;
DWORD_PTR i;
PFIELD_DESCRIPTOR foundField = NULL;
LPWSTR subStrings[2];
ValueLength, EntryContext;
//
// Ignore several parameters, since they are handled elsewhere
//
if( (_wcsicmp( ValueName, SIZE_VALUE_NAME ) == 0) ||
(_wcsicmp( ValueName, NULL_SESSION_SHARES_VALUE_NAME ) == 0) ||
(_wcsicmp( ValueName, NULL_SESSION_PIPES_VALUE_NAME ) == 0) ||
(_wcsicmp( ValueName, PIPES_NEED_LICENSE_VALUE_NAME ) == 0) ||
(_wcsicmp( ValueName, ERROR_LOG_IGNORE_VALUE_NAME ) == 0) ||
(_wcsicmp( ValueName, GUID_VARIABLE_NAME ) == 0) ||
(_wcsicmp( ValueName, OPTIONAL_NAMES_VALUE_NAME ) == 0) ||
(_wcsicmp( ValueName, NO_REMAP_PIPES_VALUE_NAME ) == 0) ||
(_wcsicmp( ValueName, SERVICE_DLL_VALUE_NAME ) == 0) ) {
return STATUS_SUCCESS;
}
//
// Determine which field we need to set, based on the value
// name.
//
// NOTE: For Daytona, disc and comment are now invalid registry names.
// We use their more famous aliases autodisconnect and srvcomment
// instead. If we get more of these cases, we should consider adding
// a field to the FIELD_DESCRIPTOR structure that indicates whether
// the names are should appear on the registry or not. Any change
// here should also be made to SsSetField().
//
if ( (_wcsicmp( ValueName, DISC_VALUE_NAME ) != 0) &&
(_wcsicmp( ValueName, COMMENT_VALUE_NAME ) != 0) ) {
for ( i = 0;
SsServerInfoFields[i].FieldName != NULL;
i++ ) {
if ( _wcsicmp( ValueName, SsServerInfoFields[i].FieldName ) == 0 ) {
foundField = &SsServerInfoFields[i];
break;
}
}
}
if ( foundField == NULL || foundField->Settable == NOT_SETTABLE ) {
#ifdef DBG
subStrings[0] = ValueName;
subStrings[1] = Context;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
NO_ERROR
);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SetStickyParameter: ignoring %s \"%ws\"\n",
(foundField == NULL ? "unknown value name" :
"unsettable value"), ValueName ));
}
#endif
return STATUS_SUCCESS;
}
switch ( foundField->FieldType ) {
case BOOLEAN_FIELD:
case DWORD_FIELD:
if ( ValueType != REG_DWORD ) {
subStrings[0] = ValueName;
subStrings[1] = Context;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
NO_ERROR
);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SetStickyParameter: skipping invalid value "
"%ws\n", ValueName ));
}
return STATUS_SUCCESS;
}
i = *(LPDWORD)ValueData;
break;
case LPSTR_FIELD:
if ( ValueType != REG_SZ ) {
subStrings[0] = ValueName;
subStrings[1] = Context;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
NO_ERROR
);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SetStickyParameter: skipping invalid value "
"%ws\n", ValueName ));
}
return STATUS_SUCCESS;
}
if (ValueLength != 0) {
i = (DWORD_PTR)ValueData;
} else {
i = (DWORD_PTR)NULL;
}
break;
}
//
// Set the field.
//
error = SsSetField( foundField, &i, FALSE, NULL );
#ifdef DBG
if ( error != NO_ERROR ) {
subStrings[0] = ValueName;
subStrings[1] = Context;
SsLogEvent(
EVENT_SRV_INVALID_REGISTRY_VALUE,
2,
subStrings,
error
);
IF_DEBUG(REGISTRY) {
SS_PRINT(( "SetStickyParameter: error %ld ignored in setting "
"parameter \"%ws\"n", error, ValueName ));
}
}
#endif
return STATUS_SUCCESS;
} // SetStickyParameter