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.
2448 lines
66 KiB
2448 lines
66 KiB
/*++
|
|
|
|
Copyright (C) Microsoft Corporation, 1991 - 1999
|
|
|
|
Module Name:
|
|
|
|
miscnt.cxx
|
|
|
|
Abstract:
|
|
|
|
This file contains NT specific implementations of miscellaneous
|
|
routines.
|
|
|
|
Author:
|
|
|
|
Michael Montague (mikemon) 25-Nov-1991
|
|
|
|
Revision History:
|
|
|
|
Kamen Moutafov (KamenM) Dec 99 - Feb 2000 - Support for cell debugging stuff
|
|
Kamen Moutafov (KamenM) Mar-2000 Support for extended error info
|
|
--*/
|
|
|
|
#include <precomp.hxx>
|
|
#include <rpccfg.h>
|
|
#include <CharConv.hxx>
|
|
|
|
static const char *RPC_REGISTRY_PROTOCOLS =
|
|
"Software\\Microsoft\\Rpc\\ClientProtocols";
|
|
|
|
static const char *RPC_REGISTRY_PROTOCOL_IDS =
|
|
"Software\\Microsoft\\Rpc\\AdditionalProtocols";
|
|
|
|
static const RPC_CHAR *RPC_REGISTRY_DEFAULT_SECURITY_DLL =
|
|
L"System\\CurrentControlSet\\Control\\SecurityProviders";
|
|
|
|
// N.B. This value must agree with the key specified in the system.adm file
|
|
static const RPC_CHAR *RPC_POLICY_SETTINGS =
|
|
L"Software\\Policies\\Microsoft\\Windows NT\\Rpc";
|
|
|
|
static const RPC_CHAR *RPC_REGISTRY_IMAGE_FILE_EXEC =
|
|
L"Software\\Microsoft\\Windows NT\\CurrentVersion\\Image File Execution Options\\";
|
|
|
|
static const RPC_CHAR *RPC_REGISTRY_THREAD_THROTTLE =
|
|
L"\\RpcThreadPoolThrottle";
|
|
|
|
const int IMAGE_FILE_EXEC_LENGTH = 75;
|
|
const int THREAD_THROTTLE_LENGTH = 24;
|
|
|
|
#define MAX_ENDPOINT_LENGTH 128
|
|
#define MAX_ID_LENGTH 6
|
|
#define MAX_DLL_NAME 128
|
|
|
|
typedef struct
|
|
{
|
|
const RPC_CHAR * RpcProtocolSequence;
|
|
const RPC_CHAR * TransportInterfaceDll;
|
|
unsigned long TransportId;
|
|
} RPC_PROTOCOL_SEQUENCE_MAP;
|
|
|
|
static const RPC_PROTOCOL_SEQUENCE_MAP RpcProtocolSequenceMap[] =
|
|
{
|
|
{
|
|
RPC_CONST_STRING("ncacn_np"),
|
|
RPC_CONST_STRING("rpcrt4.dll"),
|
|
NMP_TOWER_ID
|
|
},
|
|
|
|
{
|
|
RPC_CONST_STRING("ncacn_ip_tcp"),
|
|
RPC_CONST_STRING("rpcrt4.dll"),
|
|
TCP_TOWER_ID
|
|
},
|
|
|
|
#ifdef SPX_ON
|
|
{
|
|
RPC_CONST_STRING("ncacn_spx"),
|
|
RPC_CONST_STRING("rpcrt4.dll"),
|
|
SPX_TOWER_ID
|
|
},
|
|
#endif
|
|
|
|
{
|
|
RPC_CONST_STRING("ncadg_ip_udp"),
|
|
RPC_CONST_STRING("rpcrt4.dll"),
|
|
UDP_TOWER_ID
|
|
},
|
|
|
|
#ifdef IPX_ON
|
|
{
|
|
RPC_CONST_STRING("ncadg_ipx"),
|
|
RPC_CONST_STRING("rpcrt4.dll"),
|
|
IPX_TOWER_ID
|
|
},
|
|
#endif
|
|
|
|
#ifdef NETBIOS_ON
|
|
{
|
|
RPC_CONST_STRING("ncacn_nb_tcp"),
|
|
RPC_CONST_STRING("rpcrt4.dll"),
|
|
NB_TOWER_ID
|
|
},
|
|
|
|
{
|
|
RPC_CONST_STRING("ncacn_nb_ipx"),
|
|
RPC_CONST_STRING("rpcrt4.dll"),
|
|
NB_TOWER_ID
|
|
},
|
|
|
|
{
|
|
RPC_CONST_STRING("ncacn_nb_nb"),
|
|
RPC_CONST_STRING("rpcrt4.dll"),
|
|
NB_TOWER_ID
|
|
},
|
|
#endif
|
|
|
|
#ifdef APPLETALK_ON
|
|
{
|
|
RPC_CONST_STRING("ncacn_at_dsp"),
|
|
RPC_CONST_STRING("rpcrt4.dll"),
|
|
DSP_TOWER_ID
|
|
},
|
|
#endif
|
|
|
|
{
|
|
RPC_CONST_STRING("ncacn_http"),
|
|
RPC_CONST_STRING("rpcrt4.dll"),
|
|
HTTP_TOWER_ID
|
|
},
|
|
|
|
{
|
|
RPC_CONST_STRING("ncadg_cluster"),
|
|
RPC_CONST_STRING("rpcrt4.dll"),
|
|
CDP_TOWER_ID
|
|
},
|
|
|
|
#ifdef NCADG_MQ_ON
|
|
{
|
|
RPC_CONST_STRING("ncadg_mq"),
|
|
RPC_CONST_STRING("rpcrt4.dll"),
|
|
MQ_TOWER_ID
|
|
},
|
|
#endif
|
|
|
|
#ifdef BANYAN_ON
|
|
{
|
|
RPC_CONST_STRING("ncacn_vns_spp"),
|
|
RPC_CONST_STRING("rpcrt4.dll"),
|
|
SPP_TOWER_ID
|
|
},
|
|
#endif
|
|
|
|
{
|
|
RPC_CONST_STRING("ncalrpc"),
|
|
0,
|
|
0
|
|
},
|
|
};
|
|
|
|
const int RpcProtseqMapLength = (sizeof(RpcProtocolSequenceMap)
|
|
/ sizeof(RPC_PROTOCOL_SEQUENCE_MAP));
|
|
|
|
static const RPC_PROTOCOL_SEQUENCE_MAP RpcUseAllProtseqMap[] =
|
|
{
|
|
{
|
|
RPC_CONST_STRING("ncacn_np"),
|
|
RPC_CONST_STRING("rpcrt4.dll"),
|
|
NMP_TOWER_ID
|
|
},
|
|
|
|
{
|
|
RPC_CONST_STRING("ncalrpc"),
|
|
0,
|
|
0
|
|
},
|
|
};
|
|
|
|
const int RpcUseAllProtseqMapLength = (sizeof(RpcUseAllProtseqMap)
|
|
/ sizeof(RPC_PROTOCOL_SEQUENCE_MAP));
|
|
|
|
|
|
typedef struct
|
|
{
|
|
unsigned char * RpcProtocolSequence;
|
|
unsigned char * RpcSsEndpoint;
|
|
unsigned long TransportId;
|
|
} RPC_PROTOCOL_INFO;
|
|
|
|
|
|
static const RPC_PROTOCOL_INFO StaticProtocolMapping[] =
|
|
{
|
|
{
|
|
(unsigned char *)"ncacn_np",
|
|
(unsigned char *)"\\pipe\\epmapper",
|
|
0x0F
|
|
}
|
|
};
|
|
|
|
RPC_PROTOCOL_INFO * AdditionalProtocols = 0;
|
|
unsigned long TotalAdditionalProtocols = 0;
|
|
|
|
static const char *RPC_REGISTRY_SECURITY_PROVIDERS =
|
|
"Software\\Microsoft\\Rpc\\SecurityService";
|
|
static const char *RPC_MISC_SETTINGS =
|
|
"Software\\Microsoft\\Rpc";
|
|
|
|
BOOL DefaultProviderRead = FALSE;
|
|
DWORD DefaultAuthLevel = RPC_C_AUTHN_LEVEL_CONNECT;
|
|
DWORD DefaultProviderId = RPC_C_AUTHN_WINNT;
|
|
|
|
RPC_CHAR *DefaultSecurityDLL = L"secur32.dll";
|
|
BOOL DefaultSecurityDLLRead = FALSE;
|
|
|
|
void
|
|
GetMaxRpcSizeAndThreadPoolParameters (
|
|
void
|
|
)
|
|
{
|
|
HKEY RegistryKey;
|
|
|
|
DWORD Result;
|
|
DWORD RegStatus;
|
|
DWORD Type;
|
|
DWORD DwordSize = sizeof(DWORD);
|
|
RPC_CHAR KeyName[MAX_PATH + IMAGE_FILE_EXEC_LENGTH + THREAD_THROTTLE_LENGTH];
|
|
const RPC_CHAR * ModuleName;
|
|
int ModuleLength;
|
|
RPC_CHAR *CurrentPos;
|
|
|
|
//
|
|
// Get the default Rpc size.
|
|
//
|
|
RegStatus = RegOpenKeyExA(
|
|
HKEY_LOCAL_MACHINE,
|
|
(LPSTR) RPC_MISC_SETTINGS,
|
|
0L, KEY_READ, //Reserved
|
|
&RegistryKey
|
|
);
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
return;
|
|
}
|
|
|
|
RegStatus = RegQueryValueExA(
|
|
RegistryKey,
|
|
"MaxRpcSize",
|
|
0,
|
|
&Type,
|
|
(LPBYTE) &Result,
|
|
&DwordSize
|
|
);
|
|
|
|
if (RegStatus == ERROR_SUCCESS
|
|
&& Type == REG_DWORD)
|
|
{
|
|
gMaxRpcSize = Result;
|
|
}
|
|
|
|
RegCloseKey(RegistryKey);
|
|
|
|
//
|
|
// Find out the .EXE name.
|
|
//
|
|
ModuleName = FastGetImageBaseName();
|
|
ModuleLength = RpcpStringLength(ModuleName);
|
|
|
|
CurrentPos = KeyName;
|
|
RpcpMemoryCopy(CurrentPos,
|
|
RPC_REGISTRY_IMAGE_FILE_EXEC,
|
|
IMAGE_FILE_EXEC_LENGTH * 2);
|
|
|
|
CurrentPos += IMAGE_FILE_EXEC_LENGTH - 1;
|
|
RpcpMemoryCopy(CurrentPos,
|
|
ModuleName,
|
|
ModuleLength * 2);
|
|
|
|
CurrentPos += ModuleLength;
|
|
RpcpMemoryCopy(CurrentPos,
|
|
RPC_REGISTRY_THREAD_THROTTLE,
|
|
THREAD_THROTTLE_LENGTH * 2);
|
|
|
|
RegStatus = RegOpenKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
KeyName,
|
|
0L, KEY_READ, //Reserved
|
|
&RegistryKey
|
|
);
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
return;
|
|
}
|
|
|
|
DwordSize = sizeof(DWORD);
|
|
|
|
RegStatus = RegQueryValueExA(
|
|
RegistryKey,
|
|
"ProrateMax",
|
|
0,
|
|
&Type,
|
|
(LPBYTE) &Result,
|
|
&DwordSize
|
|
);
|
|
|
|
if (RegStatus == ERROR_SUCCESS
|
|
&& Type == REG_DWORD)
|
|
{
|
|
gProrateMax = Result;
|
|
}
|
|
|
|
DwordSize = sizeof(DWORD);
|
|
|
|
RegStatus = RegQueryValueExA(
|
|
RegistryKey,
|
|
"ProrateFactor",
|
|
0,
|
|
&Type,
|
|
(LPBYTE) &Result,
|
|
&DwordSize
|
|
);
|
|
|
|
if (RegStatus == ERROR_SUCCESS
|
|
&& Type == REG_DWORD)
|
|
{
|
|
gProrateFactor = Result;
|
|
}
|
|
|
|
DwordSize = sizeof(DWORD);
|
|
|
|
RegStatus = RegQueryValueExA(
|
|
RegistryKey,
|
|
"ProrateStart",
|
|
0,
|
|
&Type,
|
|
(LPBYTE) &Result,
|
|
&DwordSize
|
|
);
|
|
|
|
if (RegStatus == ERROR_SUCCESS
|
|
&& Type == REG_DWORD)
|
|
{
|
|
gProrateStart = Result;
|
|
}
|
|
|
|
RegCloseKey(RegistryKey);
|
|
|
|
// if any of these values are invalid, turn it off
|
|
if ((gProrateFactor == 0) || (gProrateMax == 0))
|
|
gProrateStart = 0;
|
|
}
|
|
|
|
|
|
BOOL
|
|
GetDefaultLevel()
|
|
{
|
|
HKEY RegistryKey;
|
|
|
|
DWORD Result;
|
|
DWORD RegStatus;
|
|
DWORD Type;
|
|
DWORD DwordSize = sizeof(DWORD);
|
|
|
|
//
|
|
// Get the default provider level.
|
|
//
|
|
RegStatus = RegOpenKeyExA(
|
|
HKEY_LOCAL_MACHINE,
|
|
(LPSTR) RPC_REGISTRY_SECURITY_PROVIDERS,
|
|
0L, KEY_READ, //Reserved
|
|
&RegistryKey
|
|
);
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
RegStatus = RegQueryValueExA(
|
|
RegistryKey,
|
|
"DefaultAuthLevel",
|
|
0,
|
|
&Type,
|
|
(LPBYTE) &Result,
|
|
&DwordSize
|
|
);
|
|
|
|
if (RegStatus == ERROR_CANTOPEN ||
|
|
RegStatus == ERROR_CANTREAD )
|
|
{
|
|
RegCloseKey(RegistryKey);
|
|
return TRUE;
|
|
}
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
RegCloseKey(RegistryKey);
|
|
return FALSE;
|
|
}
|
|
|
|
if ( Type != REG_DWORD )
|
|
{
|
|
RegCloseKey(RegistryKey);
|
|
return TRUE;
|
|
}
|
|
|
|
if (Result >= RPC_C_AUTHN_LEVEL_CONNECT &&
|
|
Result <= RPC_C_AUTHN_LEVEL_PKT_PRIVACY )
|
|
{
|
|
DefaultAuthLevel = Result;
|
|
}
|
|
|
|
RegCloseKey(RegistryKey);
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
void
|
|
RpcpGetDefaultSecurityProviderInfo()
|
|
{
|
|
if (DefaultProviderRead)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (GetDefaultLevel())
|
|
{
|
|
DefaultProviderRead = TRUE;
|
|
}
|
|
}
|
|
|
|
RPC_STATUS
|
|
GetDefaultSecurityDll (
|
|
IN RPC_CHAR *DllName,
|
|
IN ULONG DllNameLength
|
|
)
|
|
{
|
|
HKEY RegistryKey;
|
|
DWORD RegStatus;
|
|
RPC_CHAR *DuplicateDllName;
|
|
DWORD Type;
|
|
|
|
if (DefaultSecurityDLLRead == FALSE)
|
|
{
|
|
RegStatus = RegOpenKeyEx(
|
|
HKEY_LOCAL_MACHINE,
|
|
RPC_REGISTRY_DEFAULT_SECURITY_DLL,
|
|
0L, KEY_READ, //Reserved
|
|
&RegistryKey
|
|
);
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
if (RegStatus == ERROR_FILE_NOT_FOUND)
|
|
{
|
|
DefaultSecurityDLLRead = TRUE;
|
|
return RPC_S_OK;
|
|
}
|
|
|
|
return(RPC_S_UNKNOWN_AUTHN_SERVICE);
|
|
}
|
|
|
|
|
|
RegStatus = RegQueryValueEx(
|
|
RegistryKey,
|
|
L"ClientDll",
|
|
0,
|
|
&Type,
|
|
(unsigned char *)DllName,
|
|
&DllNameLength
|
|
);
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
RegCloseKey(RegistryKey);
|
|
if (RegStatus == ERROR_FILE_NOT_FOUND)
|
|
{
|
|
DefaultSecurityDLLRead = TRUE;
|
|
return RPC_S_OK;
|
|
}
|
|
|
|
return(RPC_S_UNKNOWN_AUTHN_SERVICE);
|
|
}
|
|
|
|
if (Type != REG_SZ)
|
|
{
|
|
RegCloseKey(RegistryKey);
|
|
return(RPC_S_UNKNOWN_AUTHN_SERVICE);
|
|
}
|
|
|
|
DuplicateDllName = DuplicateString(DllName);
|
|
|
|
RegCloseKey(RegistryKey);
|
|
|
|
if (DuplicateDllName)
|
|
{
|
|
DefaultSecurityDLL = DuplicateDllName;
|
|
DefaultSecurityDLLRead = TRUE;
|
|
}
|
|
else
|
|
{
|
|
return RPC_S_OUT_OF_MEMORY;
|
|
}
|
|
|
|
}
|
|
|
|
return RPC_S_OK;
|
|
}
|
|
|
|
RPC_STATUS
|
|
RpcGetSecurityProviderInfo(
|
|
IN unsigned long AuthnId,
|
|
OUT RPC_CHAR **Dll,
|
|
OUT unsigned long PAPI * Count
|
|
)
|
|
{
|
|
|
|
DWORD RegStatus, Ignore, NumberOfValues, MaximumValueLength;
|
|
unsigned long DllNameLength = MAX_DLL_NAME+1;
|
|
DWORD ClassLength = 64, Type;
|
|
RPC_CHAR DllName[MAX_DLL_NAME+1];
|
|
FILETIME LastWriteTime;
|
|
HKEY RegistryKey;
|
|
unsigned char ClassName[64];
|
|
RPC_STATUS Status = RPC_S_OK;
|
|
char AuthnIdZ[8];
|
|
RPC_CHAR unicodeAuthnIdZ[8];
|
|
RPC_CHAR *pAuthnIdZ;
|
|
RPC_CHAR *ActualDllName;
|
|
|
|
RpcItoa(AuthnId, AuthnIdZ, 10);
|
|
|
|
RegStatus = RegOpenKeyExA(
|
|
HKEY_LOCAL_MACHINE,
|
|
(LPSTR) RPC_REGISTRY_SECURITY_PROVIDERS,
|
|
0L, KEY_READ, //Reserved
|
|
&RegistryKey
|
|
);
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
return(RPC_S_UNKNOWN_AUTHN_SERVICE);
|
|
}
|
|
|
|
RegStatus = RegQueryInfoKeyA(
|
|
RegistryKey,
|
|
(LPSTR) ClassName,
|
|
&ClassLength,
|
|
0, //Reserved
|
|
&Ignore,
|
|
&Ignore,
|
|
&Ignore,
|
|
&NumberOfValues,
|
|
&Ignore,
|
|
&MaximumValueLength,
|
|
&Ignore,
|
|
&LastWriteTime
|
|
);
|
|
|
|
if ( (RegStatus != ERROR_SUCCESS) || (NumberOfValues < 2) )
|
|
{
|
|
RegStatus = RegCloseKey(RegistryKey);
|
|
ASSERT( RegStatus == ERROR_SUCCESS );
|
|
return(RPC_S_UNKNOWN_AUTHN_SERVICE);
|
|
}
|
|
|
|
*Count = NumberOfValues - 2; //Gross
|
|
|
|
SimpleAnsiToUnicode(AuthnIdZ, unicodeAuthnIdZ);
|
|
pAuthnIdZ = unicodeAuthnIdZ;
|
|
|
|
RegStatus = RegQueryValueEx(
|
|
RegistryKey,
|
|
(const RPC_SCHAR *)pAuthnIdZ,
|
|
0,
|
|
&Type,
|
|
(unsigned char *)DllName,
|
|
&DllNameLength
|
|
);
|
|
|
|
RegCloseKey(RegistryKey);
|
|
|
|
if (RegStatus != ERROR_SUCCESS)
|
|
{
|
|
RegStatus = GetDefaultSecurityDll(
|
|
DllName,
|
|
DllNameLength);
|
|
|
|
if (RegStatus == RPC_S_OK)
|
|
ActualDllName = DefaultSecurityDLL;
|
|
}
|
|
else
|
|
{
|
|
ActualDllName = DllName;
|
|
}
|
|
|
|
if (RegStatus == ERROR_SUCCESS)
|
|
{
|
|
*Dll = DuplicateString(ActualDllName);
|
|
if (*Dll == 0)
|
|
{
|
|
RegStatus = RPC_S_OUT_OF_MEMORY;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
RegStatus = RPC_S_UNKNOWN_AUTHN_SERVICE;
|
|
}
|
|
|
|
return(RegStatus);
|
|
}
|
|
|
|
DWORD * FourLeggedPackages = 0;
|
|
DWORD NullPackageList[] = { 0 };
|
|
|
|
|
|
|
|
BOOL
|
|
ReadPackageLegInfo()
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
NT 4.0 and previous versions allowed only two or three legs to set up the
|
|
security context. Two were represented on the wire as BIND then BIND_ACK;
|
|
three added an AUTH3 packet. When Jeff added 4- and 6-leg support, he
|
|
made the sequence BIND, BIND-ACK, ALTER-CXT, ALTER-CXT-RESPONSE. Sadly,
|
|
it turns out to be impossible for RPC to tell whether a given package wants
|
|
three or four legs, so the client would have to guess whether to send
|
|
an AUTH3 or an ALTER-CXT to an NT 4 server. To solve this we are adding a
|
|
registry value that lists all the providers that need more than three legs.
|
|
|
|
The format, in regdmp.exe form, is
|
|
|
|
\Registry\Machine\Software\Microsoft\Rpc
|
|
Four-legged packages = REG_MULTI_SZ "16" "18" ""
|
|
|
|
This function opens the registry and converts this data into an in-memory
|
|
array of DWORD package IDs.
|
|
|
|
Not all packages are in this list; see GetPackageLegCount() for details.
|
|
|
|
Return Values:
|
|
|
|
TRUE = the registry data was read, or the value does not exist.
|
|
Calling this function again will have no effect.
|
|
|
|
FALSE = there was a problem reading the data in the registry.
|
|
You can call the fn later and it will try again.
|
|
|
|
--*/
|
|
{
|
|
DWORD Size = 0;
|
|
DWORD RegStatus;
|
|
DWORD Type;
|
|
HKEY RegistryKey;
|
|
wchar_t * Strings;
|
|
|
|
if (FourLeggedPackages)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
SecurityCritSect->VerifyOwned();
|
|
|
|
// open key;
|
|
RegStatus = RegOpenKeyExA(
|
|
HKEY_LOCAL_MACHINE,
|
|
(LPSTR) RPC_REGISTRY_SECURITY_PROVIDERS,
|
|
0L, KEY_READ, //Reserved
|
|
&RegistryKey
|
|
);
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
// get size of strings
|
|
|
|
RegStatus = RegQueryValueExW(
|
|
RegistryKey,
|
|
L"Four-legged packages",
|
|
0,
|
|
&Type,
|
|
0,
|
|
&Size
|
|
);
|
|
|
|
if (RegStatus == ERROR_FILE_NOT_FOUND )
|
|
{
|
|
RegCloseKey(RegistryKey);
|
|
|
|
FourLeggedPackages = NullPackageList;
|
|
return TRUE;
|
|
}
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
RegCloseKey(RegistryKey);
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// Place an upper bound on the size to avoid hacker attacks.
|
|
//
|
|
if ( Type != REG_MULTI_SZ || Size > 4000)
|
|
{
|
|
RegCloseKey(RegistryKey);
|
|
FourLeggedPackages = NullPackageList;
|
|
return TRUE;
|
|
}
|
|
|
|
// get string data.
|
|
|
|
Strings = (wchar_t *) _alloca( Size );
|
|
|
|
RegStatus = RegQueryValueExW(
|
|
RegistryKey,
|
|
L"Four-legged packages",
|
|
0,
|
|
&Type,
|
|
(unsigned char *) Strings,
|
|
&Size
|
|
);
|
|
|
|
RegCloseKey(RegistryKey);
|
|
|
|
if (RegStatus == ERROR_CANTOPEN ||
|
|
RegStatus == ERROR_CANTREAD )
|
|
{
|
|
FourLeggedPackages = NullPackageList;
|
|
return TRUE;
|
|
}
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
if ( Type != REG_MULTI_SZ )
|
|
{
|
|
FourLeggedPackages = NullPackageList;
|
|
return TRUE;
|
|
}
|
|
|
|
// count strings; the buffer is terminated by an empty string that will be counted.
|
|
|
|
int Count = 0;
|
|
wchar_t * p;
|
|
for (p=Strings; p < Strings + Size; ++p)
|
|
{
|
|
if (*p == '\0')
|
|
{
|
|
++Count;
|
|
}
|
|
}
|
|
|
|
// allocate memory.
|
|
|
|
DWORD * LegData = new DWORD[Count];
|
|
if (!LegData)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
// transfer data
|
|
int i;
|
|
wchar_t * new_p;
|
|
for (i=0, p=Strings; p < Strings + Size; ++i, p = new_p+1)
|
|
{
|
|
LegData[i] = wcstoul(p, &new_p, 10);
|
|
if (*new_p != '\0')
|
|
{
|
|
// The string is badly formatted. Eliminate it.
|
|
--i;
|
|
new_p += wcslen(new_p);
|
|
}
|
|
}
|
|
|
|
if (FourLeggedPackages && FourLeggedPackages != NullPackageList)
|
|
{
|
|
delete FourLeggedPackages;
|
|
}
|
|
|
|
FourLeggedPackages = LegData;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
RPC_STATUS
|
|
LoadAdditionalTransportInfo(
|
|
)
|
|
{
|
|
|
|
DWORD RegStatus, Index, Ignore, NumberOfValues, MaximumValueLength;
|
|
DWORD ClassLength = 64, ProtseqLength, IgnoreLength;
|
|
BYTE Protseq[MAX_PROTSEQ_LENGTH+1];
|
|
BYTE MaxValueData[MAX_ENDPOINT_LENGTH+MAX_ID_LENGTH+2+8];
|
|
FILETIME LastWriteTime;
|
|
HKEY RegistryKey;
|
|
unsigned char ClassName[64];
|
|
char * Value;
|
|
RPC_PROTOCOL_INFO * AdditionalProtocolsInfo;
|
|
RPC_STATUS Status = RPC_S_OK;
|
|
unsigned long Length, TransportId;
|
|
|
|
RegStatus = RegOpenKeyExA(
|
|
HKEY_LOCAL_MACHINE,
|
|
(LPSTR) RPC_REGISTRY_PROTOCOL_IDS,
|
|
0L, KEY_READ, //Reserved
|
|
&RegistryKey
|
|
);
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
return(RPC_S_INVALID_RPC_PROTSEQ);
|
|
}
|
|
|
|
RegStatus = RegQueryInfoKeyA(
|
|
RegistryKey,
|
|
(LPSTR) ClassName,
|
|
&ClassLength,
|
|
0, //Reserved
|
|
&Ignore,
|
|
&Ignore,
|
|
&Ignore,
|
|
&NumberOfValues,
|
|
&Ignore,
|
|
&MaximumValueLength,
|
|
&Ignore,
|
|
&LastWriteTime
|
|
);
|
|
|
|
if ( (RegStatus != ERROR_SUCCESS) || (NumberOfValues == 0) )
|
|
{
|
|
RegStatus = RegCloseKey(RegistryKey);
|
|
ASSERT( RegStatus == ERROR_SUCCESS );
|
|
return(RPC_S_INVALID_RPC_PROTSEQ);
|
|
}
|
|
|
|
//Allocate a table for additional transports mapping
|
|
|
|
AdditionalProtocolsInfo = (RPC_PROTOCOL_INFO *) new unsigned char [
|
|
sizeof(RPC_PROTOCOL_INFO) * NumberOfValues];
|
|
if (AdditionalProtocolsInfo == 0)
|
|
{
|
|
Status = RPC_S_OUT_OF_MEMORY;
|
|
goto Cleanup;
|
|
}
|
|
|
|
AdditionalProtocols = AdditionalProtocolsInfo;
|
|
TotalAdditionalProtocols = NumberOfValues;
|
|
|
|
for (Index = 0; Index < NumberOfValues; Index++)
|
|
{
|
|
|
|
ProtseqLength = MAX_PROTSEQ_LENGTH;
|
|
IgnoreLength = MAX_ENDPOINT_LENGTH + MAX_ID_LENGTH;
|
|
RegStatus = RegEnumValueA(
|
|
RegistryKey,
|
|
Index,
|
|
(LPSTR) &Protseq,
|
|
&ProtseqLength,
|
|
0,
|
|
&Ignore,
|
|
(LPBYTE) MaxValueData,
|
|
&IgnoreLength
|
|
);
|
|
|
|
if (RegStatus == ERROR_SUCCESS)
|
|
{
|
|
//Add this to our table..
|
|
AdditionalProtocolsInfo->RpcProtocolSequence =
|
|
new unsigned char[ProtseqLength+1];
|
|
Value = (char * )&MaxValueData;
|
|
AdditionalProtocolsInfo->RpcSsEndpoint =
|
|
new unsigned char[Length = (strlen(Value) + 1)];
|
|
|
|
|
|
if (AdditionalProtocolsInfo->RpcProtocolSequence == 0
|
|
|| AdditionalProtocolsInfo->RpcSsEndpoint == 0)
|
|
{
|
|
Status = RPC_S_OUT_OF_MEMORY;
|
|
goto Cleanup;
|
|
}
|
|
|
|
RpcpMemoryCopy(
|
|
AdditionalProtocolsInfo->RpcProtocolSequence,
|
|
Protseq,
|
|
ProtseqLength+1
|
|
);
|
|
|
|
RpcpMemoryCopy(
|
|
AdditionalProtocolsInfo->RpcSsEndpoint,
|
|
Value,
|
|
Length
|
|
);
|
|
Value = Value + Length;
|
|
|
|
for (TransportId = 0;
|
|
(*Value > '0') && (*Value <= '9') && (TransportId <= 255);
|
|
Value++)
|
|
{
|
|
TransportId = TransportId * 10 + (*Value - '0');
|
|
}
|
|
AdditionalProtocolsInfo->TransportId = TransportId;
|
|
|
|
AdditionalProtocolsInfo++;
|
|
}
|
|
|
|
}
|
|
|
|
Cleanup:
|
|
RegStatus = RegCloseKey(RegistryKey);
|
|
|
|
if (Status != RPC_S_OK)
|
|
{
|
|
if (AdditionalProtocols != 0)
|
|
{
|
|
AdditionalProtocolsInfo = AdditionalProtocols;
|
|
for (Index = 0; Index < NumberOfValues; Index++)
|
|
{
|
|
if (AdditionalProtocolsInfo->RpcProtocolSequence != 0)
|
|
delete AdditionalProtocolsInfo->RpcProtocolSequence;
|
|
if (AdditionalProtocolsInfo->RpcSsEndpoint != 0)
|
|
delete AdditionalProtocolsInfo->RpcSsEndpoint;
|
|
AdditionalProtocolsInfo++;
|
|
}
|
|
|
|
delete AdditionalProtocols;
|
|
AdditionalProtocols = 0;
|
|
TotalAdditionalProtocols = 0;
|
|
}
|
|
}
|
|
|
|
return(Status);
|
|
}
|
|
|
|
|
|
RPC_STATUS
|
|
RpcGetAdditionalTransportInfo(
|
|
IN unsigned long TransportId,
|
|
OUT unsigned char PAPI * PAPI * ProtocolSequence
|
|
)
|
|
{
|
|
unsigned long i;
|
|
RPC_PROTOCOL_INFO * ProtocolInfo;
|
|
|
|
RequestGlobalMutex();
|
|
|
|
if (AdditionalProtocols == 0)
|
|
{
|
|
LoadAdditionalTransportInfo();
|
|
}
|
|
|
|
ClearGlobalMutex();
|
|
|
|
for (i = 0, ProtocolInfo = AdditionalProtocols ;
|
|
i < TotalAdditionalProtocols;
|
|
i++)
|
|
{
|
|
if (ProtocolInfo->TransportId == TransportId)
|
|
{
|
|
*ProtocolSequence = ProtocolInfo->RpcProtocolSequence;
|
|
return (RPC_S_OK);
|
|
}
|
|
ProtocolInfo ++;
|
|
}
|
|
|
|
return(RPC_S_INVALID_RPC_PROTSEQ);
|
|
|
|
}
|
|
|
|
|
|
|
|
RPC_CHAR *
|
|
LocalMapRpcProtocolSequence (
|
|
IN RPC_CHAR PAPI * RpcProtocolSequence
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
We need to check the supplied protocol sequence (and module) to see
|
|
if we can map them into a transport interface dll without having to
|
|
use the registry.
|
|
|
|
Arguments:
|
|
|
|
ServerSideFlag - Supplies a flag indicating whether this protocol
|
|
sequence is to be mapped for a client or a server; a non-zero
|
|
value indicates that it is being mapped for a server.
|
|
|
|
RpcProtocolSequence - Supplies the protocol sequence which we need to
|
|
map into a transport interface dll.
|
|
|
|
Return Value:
|
|
|
|
If we successfully map the protocol sequence, then a pointer to a static
|
|
string containing the transport interface dll (name) will be returned;
|
|
the caller must duplicate the string. Otherwise, zero will be returned.
|
|
|
|
--*/
|
|
{
|
|
unsigned int Index;
|
|
|
|
for (Index = 0; Index < RpcProtseqMapLength; Index++)
|
|
{
|
|
if ( RpcpStringCompare(RpcProtocolSequence,
|
|
RpcProtocolSequenceMap[Index].RpcProtocolSequence) == 0 )
|
|
{
|
|
return((RPC_CHAR *)(RpcProtocolSequenceMap[Index].TransportInterfaceDll));
|
|
}
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
RPC_STATUS
|
|
RpcGetWellKnownTransportInfo(
|
|
IN unsigned long TransportId,
|
|
OUT RPC_CHAR **PSeq
|
|
)
|
|
{
|
|
unsigned int Index;
|
|
|
|
for (Index = 0; Index < RpcProtseqMapLength; Index++)
|
|
{
|
|
if (TransportId == RpcProtocolSequenceMap[Index].TransportId)
|
|
{
|
|
*PSeq = (RPC_CHAR *)RpcProtocolSequenceMap[Index].RpcProtocolSequence;
|
|
return RPC_S_OK;
|
|
}
|
|
}
|
|
|
|
return(RPC_S_PROTSEQ_NOT_SUPPORTED);
|
|
}
|
|
|
|
|
|
RPC_STATUS
|
|
RpcConfigMapRpcProtocolSequence (
|
|
IN unsigned int ServerSideFlag,
|
|
IN RPC_CHAR PAPI * RpcProtocolSequence,
|
|
OUT RPC_CHAR * PAPI * TransportInterfaceDll
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is used by the rpc protocol modules to map from an
|
|
rpc protocol sequence to the name of a transport interface dll.
|
|
|
|
Arguments:
|
|
|
|
ServerSideFlag - Supplies a flag indicating whether this protocol
|
|
sequence is to be mapped for a client or a server; a non-zero
|
|
value indicates that it is being mapped for a server.
|
|
|
|
RpcProtocolSequence - Supplies the rpc protocol sequence to map.
|
|
|
|
TransportInterfaceDll - Returns the transport support dll which
|
|
supports the requested rpc protocol sequence. This will be a
|
|
newly allocated string which the caller must free.
|
|
|
|
Return Value:
|
|
|
|
RPC_S_OK - Everything worked out fine.
|
|
|
|
RPC_S_PROTSEQ_NOT_SUPPORTED - The requested rpc protocol sequence
|
|
does not have a mapping to a transport interface dll for this
|
|
rpc protocol module.
|
|
|
|
RPC_S_OUT_OF_MEMORY - We ran out of memory trying to map the rpc
|
|
protocol sequence.
|
|
|
|
--*/
|
|
{
|
|
RPC_CHAR * TempString;
|
|
HKEY RegistryKey;
|
|
DWORD Type;
|
|
long RegStatus;
|
|
unsigned char * KeyString;
|
|
unsigned long Length;
|
|
|
|
TempString = LocalMapRpcProtocolSequence(RpcProtocolSequence);
|
|
if ( TempString != 0 )
|
|
{
|
|
*TransportInterfaceDll = new RPC_CHAR[RpcpStringLength(TempString) + 1];
|
|
if ( *TransportInterfaceDll == 0 )
|
|
{
|
|
return(RPC_S_OUT_OF_MEMORY);
|
|
}
|
|
memcpy(*TransportInterfaceDll, TempString,
|
|
(RpcpStringLength(TempString) + 1) * sizeof(RPC_CHAR));
|
|
return(RPC_S_OK);
|
|
}
|
|
|
|
KeyString = (unsigned char *) RPC_REGISTRY_PROTOCOLS;
|
|
|
|
RegStatus = RegOpenKeyExA(HKEY_LOCAL_MACHINE, (LPSTR) KeyString, 0L,
|
|
KEY_READ, &RegistryKey);
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
return(RPC_S_PROTSEQ_NOT_SUPPORTED);
|
|
}
|
|
|
|
*TransportInterfaceDll = new RPC_CHAR[MAX_DLLNAME_LENGTH + 1];
|
|
if ( *TransportInterfaceDll == 0 )
|
|
{
|
|
RegStatus = RegCloseKey(RegistryKey);
|
|
ASSERT( RegStatus == ERROR_SUCCESS );
|
|
|
|
return(RPC_S_OUT_OF_MEMORY);
|
|
}
|
|
|
|
Length = (MAX_DLLNAME_LENGTH + 1) * sizeof(RPC_CHAR);
|
|
RegStatus = RegQueryValueEx(RegistryKey, (const RPC_SCHAR *)RpcProtocolSequence,
|
|
0, &Type, (LPBYTE) *TransportInterfaceDll, &Length);
|
|
|
|
if ( RegStatus == ERROR_SUCCESS )
|
|
{
|
|
RegStatus = RegCloseKey(RegistryKey);
|
|
ASSERT( RegStatus == ERROR_SUCCESS );
|
|
|
|
return(RPC_S_OK);
|
|
}
|
|
|
|
RegStatus = RegCloseKey(RegistryKey);
|
|
ASSERT( RegStatus == ERROR_SUCCESS );
|
|
|
|
delete *TransportInterfaceDll;
|
|
|
|
return(RPC_S_PROTSEQ_NOT_SUPPORTED);
|
|
}
|
|
|
|
|
|
RPC_STATUS
|
|
RpcConfigInquireProtocolSequencesFromKey (
|
|
OUT RPC_PROTSEQ_VECTOR PAPI * PAPI * ProtseqVector,
|
|
unsigned char *RegistryKeyName
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is used to obtain a list of the rpc protocol sequences
|
|
supported by the system using a given registry key as a reference
|
|
point.
|
|
|
|
Arguments:
|
|
|
|
ProtseqVector - Returns a vector of supported rpc protocol sequences
|
|
for this rpc protocol module.
|
|
RegistryKeyName - the path of the registry key, starting from
|
|
HKLM (the HKLM itself should not be supplied)
|
|
|
|
Return Value:
|
|
|
|
RPC_S_OK - The operation completed successfully.
|
|
|
|
RPC_S_NO_PROTSEQS - The current system configuration does not
|
|
support any rpc protocol sequences.
|
|
|
|
RPC_S_OUT_OF_MEMORY - Insufficient memory is available to inquire
|
|
the rpc protocol sequences supported by the specified rpc
|
|
protocol sequence.
|
|
|
|
--*/
|
|
{
|
|
DWORD RegStatus, Index, Ignore, MaximumValueLength;
|
|
DWORD ClassLength = 64, ProtseqLength, IgnoreLength;
|
|
BYTE IgnoreData[MAX_DLLNAME_LENGTH];
|
|
FILETIME LastWriteTime;
|
|
HKEY RegistryKey = 0;
|
|
unsigned char ClassName[64];
|
|
DWORD NumberOfValues = 0;
|
|
|
|
RegStatus = RegOpenKeyExA(HKEY_LOCAL_MACHINE,
|
|
(LPSTR) RegistryKeyName, 0L, KEY_READ, &RegistryKey);
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
return(RPC_S_NO_PROTSEQS);
|
|
}
|
|
|
|
RegStatus = RegQueryInfoKeyA(RegistryKey, (LPSTR) ClassName, &ClassLength,
|
|
0, &Ignore, &Ignore, &Ignore, &NumberOfValues,
|
|
&Ignore, &MaximumValueLength, &Ignore, &LastWriteTime);
|
|
|
|
ASSERT( RegStatus == ERROR_SUCCESS );
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
RegStatus = RegCloseKey(RegistryKey);
|
|
ASSERT( RegStatus == ERROR_SUCCESS );
|
|
return(RPC_S_NO_PROTSEQS);
|
|
}
|
|
|
|
NumberOfValues += RpcUseAllProtseqMapLength;
|
|
|
|
*ProtseqVector = (RPC_PROTSEQ_VECTOR *) new unsigned char[
|
|
sizeof(RPC_PROTSEQ_VECTOR) + (NumberOfValues - 1)
|
|
* sizeof(RPC_CHAR *)];
|
|
|
|
if ( *ProtseqVector == 0 )
|
|
{
|
|
RegStatus = RegCloseKey(RegistryKey);
|
|
ASSERT( RegStatus == ERROR_SUCCESS );
|
|
return(RPC_S_OUT_OF_MEMORY);
|
|
}
|
|
|
|
(*ProtseqVector)->Count = (unsigned int) NumberOfValues;
|
|
|
|
for (Index = 0; Index < NumberOfValues; Index++)
|
|
{
|
|
(*ProtseqVector)->Protseq[Index] = 0;
|
|
}
|
|
|
|
for (Index = 0; Index < RpcUseAllProtseqMapLength; Index++)
|
|
{
|
|
(*ProtseqVector)->Protseq[Index] = new RPC_CHAR[MAX_PROTSEQ_LENGTH];
|
|
if ( (*ProtseqVector)->Protseq[Index] == 0 )
|
|
{
|
|
RegStatus = RegCloseKey(RegistryKey);
|
|
ASSERT( RegStatus == ERROR_SUCCESS );
|
|
|
|
RpcProtseqVectorFree(ProtseqVector);
|
|
|
|
return(RPC_S_OUT_OF_MEMORY);
|
|
}
|
|
|
|
RpcpStringCopy((*ProtseqVector)->Protseq[Index],
|
|
RpcUseAllProtseqMap[Index].RpcProtocolSequence);
|
|
|
|
}
|
|
|
|
unsigned VectorIndex = Index;
|
|
|
|
for (Index = 0; VectorIndex < NumberOfValues; Index++, VectorIndex++)
|
|
{
|
|
(*ProtseqVector)->Protseq[VectorIndex] = new RPC_CHAR[MAX_PROTSEQ_LENGTH];
|
|
if ( (*ProtseqVector)->Protseq[VectorIndex] == 0 )
|
|
{
|
|
RegStatus = RegCloseKey(RegistryKey);
|
|
ASSERT( RegStatus == ERROR_SUCCESS );
|
|
|
|
RpcProtseqVectorFree(ProtseqVector);
|
|
|
|
return(RPC_S_OUT_OF_MEMORY);
|
|
}
|
|
|
|
ProtseqLength = MAX_PROTSEQ_LENGTH;
|
|
IgnoreLength = MAX_DLLNAME_LENGTH;
|
|
RegStatus = RegEnumValue(RegistryKey, Index,
|
|
(RPC_SCHAR *)(*ProtseqVector)->Protseq[VectorIndex], &ProtseqLength,
|
|
0, &Ignore, (LPBYTE) IgnoreData, &IgnoreLength);
|
|
|
|
ASSERT( RegStatus == ERROR_SUCCESS );
|
|
|
|
if (RpcpStringCompare((RPC_SCHAR *)(*ProtseqVector)->Protseq[VectorIndex], RPC_T("ncacn_np")) == 0)
|
|
{
|
|
// ignore this named pipe value - we have to have it in the registry for
|
|
// compatibility purposes with VB, but we don't really use it
|
|
NumberOfValues --;
|
|
(*ProtseqVector)->Count --;
|
|
delete (*ProtseqVector)->Protseq[VectorIndex];
|
|
VectorIndex --;
|
|
}
|
|
}
|
|
|
|
RegStatus = RegCloseKey(RegistryKey);
|
|
ASSERT( RegStatus == ERROR_SUCCESS );
|
|
|
|
return(RPC_S_OK);
|
|
}
|
|
|
|
RPC_STATUS RpcConfigInquireStaticProtocolSequences(RPC_PROTSEQ_VECTOR **ProtseqVector)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Returns a protseq vector that contains all the protseqs in the static RpcProtocolSequenceMap
|
|
map.
|
|
|
|
Arguments:
|
|
|
|
ProtseqVector - Returns a vector of supported rpc protocol sequences.
|
|
|
|
Return Value:
|
|
|
|
RPC_S_OK - The operation completed successfully.
|
|
|
|
RPC_S_OUT_OF_MEMORY - Insufficient memory is available to construct the vector.
|
|
|
|
--*/
|
|
{
|
|
int nNumberOfValues = sizeof(RpcProtocolSequenceMap) / sizeof(RpcProtocolSequenceMap[0]);
|
|
int i;
|
|
|
|
*ProtseqVector = (RPC_PROTSEQ_VECTOR *) new unsigned char[
|
|
sizeof(RPC_PROTSEQ_VECTOR) + (nNumberOfValues - 1)
|
|
* sizeof(RPC_CHAR *)];
|
|
|
|
if ( *ProtseqVector == 0 )
|
|
return(RPC_S_OUT_OF_MEMORY);
|
|
|
|
(*ProtseqVector)->Count = (unsigned int) nNumberOfValues;
|
|
|
|
for (i = 0; i < nNumberOfValues; i ++)
|
|
{
|
|
(*ProtseqVector)->Protseq[i] = NULL;
|
|
}
|
|
|
|
for (i = 0; i < nNumberOfValues; i ++)
|
|
{
|
|
(*ProtseqVector)->Protseq[i] = new RPC_CHAR[MAX_PROTSEQ_LENGTH];
|
|
if ( (*ProtseqVector)->Protseq[i] == 0 )
|
|
{
|
|
|
|
RpcProtseqVectorFree(ProtseqVector);
|
|
|
|
return(RPC_S_OUT_OF_MEMORY);
|
|
}
|
|
|
|
RpcpStringCopy((*ProtseqVector)->Protseq[i],
|
|
RpcProtocolSequenceMap[i].RpcProtocolSequence);
|
|
|
|
}
|
|
|
|
return RPC_S_OK;
|
|
}
|
|
|
|
int MarkDuplicateEntries(IN RPC_PROTSEQ_VECTOR *ProtseqVector1,
|
|
IN int nVector1CurrentPos,
|
|
IN RPC_PROTSEQ_VECTOR *ProtseqVector2 OPTIONAL,
|
|
IN OUT UCHAR *pfDuplicates1,
|
|
IN OUT UCHAR *pfDuplicates2)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is tightly coupled with RpcConfigInquireProtocolSequences.
|
|
If the current element is already marked as duplicate by previous
|
|
run of this function, we directly return 1. If not,
|
|
it checks whether the current element is duplicate of any element
|
|
starting with the nVetcor1StartingPos from the first vector, and
|
|
going through the whole second vector if it is there, and marking
|
|
all duplicates of the current element as such. Then 0 is returned.
|
|
If the second vector is NULL, the pfDuplicates2 must be NULL also. This
|
|
means that there is one vector only, and its data are passed in
|
|
in RpcProtseqVector1 and in pfDuplicates1.
|
|
|
|
Arguments:
|
|
|
|
RpcProtseqVector1 - the first vector
|
|
nVector1CurrentPos - the current position from the first vector
|
|
RpcProtseqVector2 - the second vector. This argument may be null
|
|
pfDuplicates1 - the array of duplicate flags for the first vector
|
|
pfDuplicates2 - the array of duplicate flags for the second vector. If
|
|
second vector is NULL, must be NULL also.
|
|
|
|
Return Value:
|
|
|
|
1 - the current element is a duplicate of an element that we
|
|
encountered in a previous run of this function.
|
|
0 - the current element is unique in both vectors
|
|
|
|
--*/
|
|
{
|
|
int i;
|
|
RPC_CHAR *pszCurrentElement;
|
|
|
|
ASSERT(nVector1CurrentPos < (int)ProtseqVector1->Count);
|
|
|
|
if (pfDuplicates1[nVector1CurrentPos])
|
|
return 1;
|
|
|
|
pszCurrentElement = ProtseqVector1->Protseq[nVector1CurrentPos];
|
|
|
|
for (i = nVector1CurrentPos + 1; i < (int)ProtseqVector1->Count; i ++)
|
|
{
|
|
if (RpcpStringCompare(pszCurrentElement,
|
|
ProtseqVector1->Protseq[i]) == 0)
|
|
{
|
|
pfDuplicates1[i] = TRUE;
|
|
}
|
|
}
|
|
|
|
if (ProtseqVector2)
|
|
{
|
|
ASSERT(pfDuplicates2 != NULL);
|
|
|
|
for (i = 0; i < (int)ProtseqVector2->Count; i ++)
|
|
{
|
|
if (RpcpStringCompare(pszCurrentElement,
|
|
ProtseqVector2->Protseq[i]) == 0)
|
|
{
|
|
pfDuplicates2[i] = TRUE;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ASSERT(pfDuplicates2 == NULL);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
RPC_STATUS
|
|
MergeProtseqVectors (IN OUT RPC_PROTSEQ_VECTOR *ProtseqVector1,
|
|
IN OUT RPC_PROTSEQ_VECTOR *ProtseqVector2 OPTIONAL,
|
|
OUT RPC_PROTSEQ_VECTOR **ProtseqVector
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine takes two protseq vectors, and in the OUT argument ProtseqVector
|
|
returns the resulting vector, which is a union of the two IN
|
|
vectors. Successful or not, the input vectors will be freed on exit.
|
|
|
|
Arguments:
|
|
|
|
ProtseqVector1 - the first vector
|
|
ProtseqVector2 - the second vector. This argument may be null in which case the
|
|
output vector is simply the first vector.
|
|
ProtseqVector - the resulting union vector. If the return value from the function
|
|
is not RPC_S_OK, the out parameter is undefined and should not be used by caller.
|
|
|
|
Return Value:
|
|
|
|
RPC_S_OK - success. The ProtseqVector argument contains the resulting vector.
|
|
error code - the cause of the error.
|
|
|
|
--*/
|
|
{
|
|
unsigned char *pfDuplicate1;
|
|
unsigned char *pfDuplicate2;
|
|
int i, j;
|
|
int nDuplicateElements;
|
|
int nCurrentElement;
|
|
int nUniqueElements;
|
|
|
|
// compare basically each with each
|
|
// construct arrays in which we will mark each duplicate element
|
|
// if it is such
|
|
pfDuplicate1 = (unsigned char *)alloca(ProtseqVector1->Count);
|
|
if (ProtseqVector2)
|
|
pfDuplicate2 = (unsigned char *)alloca(ProtseqVector2->Count);
|
|
else
|
|
pfDuplicate2 = NULL;
|
|
|
|
// all elements are presumed to be unique unless proven otherwise
|
|
memset(pfDuplicate1, 0, ProtseqVector1->Count);
|
|
if (ProtseqVector2)
|
|
memset(pfDuplicate2, 0, ProtseqVector2->Count);
|
|
|
|
nDuplicateElements = 0;
|
|
|
|
// do compare each with each
|
|
// test the first vector for uniqueness
|
|
for (i = 0; i < (int)ProtseqVector1->Count; i ++)
|
|
{
|
|
nDuplicateElements += MarkDuplicateEntries(ProtseqVector1, i,
|
|
ProtseqVector2, pfDuplicate1, pfDuplicate2);
|
|
}
|
|
|
|
if (ProtseqVector2)
|
|
{
|
|
// test the second vector for uniqueness
|
|
for (i = 0; i < (int)ProtseqVector2->Count; i ++)
|
|
{
|
|
nDuplicateElements += MarkDuplicateEntries(ProtseqVector2, i,
|
|
NULL, pfDuplicate2, NULL);
|
|
}
|
|
}
|
|
|
|
// here we must move the unique elements to a new vector
|
|
// first, calculate the length of the new vector
|
|
nUniqueElements = ProtseqVector1->Count - nDuplicateElements;
|
|
if (ProtseqVector2)
|
|
nUniqueElements += ProtseqVector2->Count;
|
|
|
|
// second, alloc the new vector
|
|
*ProtseqVector = (RPC_PROTSEQ_VECTOR *) new unsigned char[
|
|
sizeof(RPC_PROTSEQ_VECTOR) +
|
|
(nUniqueElements - 1) * sizeof(RPC_CHAR *)];
|
|
if (*ProtseqVector == NULL)
|
|
{
|
|
RpcProtseqVectorFree(&ProtseqVector1);
|
|
if (ProtseqVector2)
|
|
RpcProtseqVectorFree(&ProtseqVector2);
|
|
return RPC_S_OUT_OF_MEMORY;
|
|
}
|
|
|
|
// set the count of the union vector
|
|
(*ProtseqVector)->Count = nUniqueElements;
|
|
|
|
nCurrentElement = 0; // counts the current element in the union
|
|
// vector
|
|
for (i = 0; i < (int)ProtseqVector1->Count; i ++)
|
|
{
|
|
if (!pfDuplicate1[i])
|
|
{
|
|
(*ProtseqVector)->Protseq[nCurrentElement] =
|
|
ProtseqVector1->Protseq[i];
|
|
nCurrentElement ++;
|
|
// save the string from deletion
|
|
ProtseqVector1->Protseq[i] = NULL;
|
|
}
|
|
}
|
|
|
|
if (ProtseqVector2)
|
|
{
|
|
for (i = 0; i < (int)ProtseqVector2->Count; i ++)
|
|
{
|
|
if (!pfDuplicate2[i])
|
|
{
|
|
(*ProtseqVector)->Protseq[nCurrentElement] =
|
|
ProtseqVector2->Protseq[i];
|
|
nCurrentElement ++;
|
|
// save the string from deletion
|
|
ProtseqVector2->Protseq[i] = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
// we don't need the original vectors anymore - delete them
|
|
RpcProtseqVectorFree(&ProtseqVector1);
|
|
if (ProtseqVector2)
|
|
RpcProtseqVectorFree(&ProtseqVector2);
|
|
|
|
return RPC_S_OK;
|
|
}
|
|
|
|
|
|
RPC_STATUS
|
|
RpcConfigInquireProtocolSequences (
|
|
IN BOOL fGetAllProtseqs,
|
|
OUT RPC_PROTSEQ_VECTOR PAPI * PAPI * ProtseqVector1
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is used to obtain a list of the rpc protocol sequences
|
|
supported by the system.
|
|
|
|
Arguments:
|
|
|
|
fGetAllProtseqs - if TRUE, all protseqs known will be returned. If FALSE,
|
|
only the protseqs currently installed will be returned.
|
|
ProtseqVector - Returns a vector of supported rpc protocol sequences
|
|
for this rpc protocol module.
|
|
|
|
Return Value:
|
|
|
|
RPC_S_OK - The operation completed successfully.
|
|
|
|
RPC_S_NO_PROTSEQS - The current system configuration does not
|
|
support any rpc protocol sequences.
|
|
|
|
RPC_S_OUT_OF_MEMORY - Insufficient memory is available to inquire
|
|
the rpc protocol sequences supported by rpc.
|
|
|
|
--*/
|
|
{
|
|
RPC_STATUS RpcStatus1;
|
|
RPC_STATUS RpcStatus2;
|
|
RPC_PROTSEQ_VECTOR *ProtseqVector2 = NULL;
|
|
RPC_PROTSEQ_VECTOR *ProtseqVector;
|
|
|
|
if (fGetAllProtseqs)
|
|
{
|
|
RpcStatus2 = RpcConfigInquireStaticProtocolSequences(&ProtseqVector2);
|
|
if (RpcStatus2 != RPC_S_OK)
|
|
return RpcStatus2;
|
|
}
|
|
|
|
// under NT, failure to query the registry vector is fatal. Under Win98, querying
|
|
// this registry vector is optional, and if we fail, we have to continue gathering the
|
|
// vector from other sources.
|
|
RpcStatus1 = RpcConfigInquireProtocolSequencesFromKey(
|
|
ProtseqVector1, (unsigned char *)RPC_REGISTRY_PROTOCOLS);
|
|
|
|
if (RpcStatus1 != RPC_S_OK)
|
|
{
|
|
RpcProtseqVectorFree(&ProtseqVector2);
|
|
return RpcStatus1;
|
|
}
|
|
|
|
if (fGetAllProtseqs)
|
|
{
|
|
// Merge the static vector and the one from the registry
|
|
RpcStatus1 = MergeProtseqVectors(*ProtseqVector1, ProtseqVector2, &ProtseqVector);
|
|
if (RpcStatus1 != RPC_S_OK)
|
|
{
|
|
return RpcStatus1;
|
|
}
|
|
|
|
*ProtseqVector1 = ProtseqVector;
|
|
}
|
|
|
|
return RpcStatus1;
|
|
}
|
|
|
|
// N.B. This enumration must agree with the values
|
|
// in system.adm
|
|
typedef enum tagStateInformationPolicyValues
|
|
{
|
|
sipvNone,
|
|
sipvAuto1,
|
|
sipvAuto2,
|
|
sipvServer,
|
|
sipvFull
|
|
} StateInformationPolicyValues;
|
|
|
|
// N.B. This enumration must agree with the values
|
|
// in system.adm
|
|
typedef enum tagEEInfoPolicyValues
|
|
{
|
|
eeipvOff,
|
|
eeipvOnWithExceptions,
|
|
eeipvOffWithExceptions,
|
|
eeipvOn
|
|
} EEInfoPolicyValues;
|
|
|
|
void
|
|
SetAutoPolicySettings(
|
|
StateInformationPolicyValues PolicyValue
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine sets the state information maintenance to the appropriate
|
|
level of Auto, depending on machine capacity.
|
|
|
|
Arguments:
|
|
|
|
PolicyValue - sipvAuto1 or sipvAuto2
|
|
|
|
Return Value:
|
|
|
|
RPC_S_OK - The operation completed successfully.
|
|
|
|
other RPC_S_* - error
|
|
|
|
--*/
|
|
{
|
|
NT_PRODUCT_TYPE ProductType;
|
|
BOOL fNeedServer;
|
|
ULONG_PTR MinMemoryNeeded;
|
|
MEMORYSTATUSEX MemoryStatus;
|
|
BOOL fResult;
|
|
|
|
ASSERT((PolicyValue == sipvAuto1) || (PolicyValue == sipvAuto2));
|
|
|
|
// client side debug info is FALSE by default - no need to
|
|
// set it explicitly
|
|
|
|
// see what we have
|
|
// RtlGetNtProductType always returns valid ProductType
|
|
// It may be incorrect during GUI mode setup, but it will
|
|
// be valid
|
|
(VOID) RtlGetNtProductType(&ProductType);
|
|
|
|
MemoryStatus.dwLength = sizeof(MemoryStatus);
|
|
fResult = GlobalMemoryStatusEx(&MemoryStatus);
|
|
ASSERT(fResult);
|
|
|
|
// see what we have been asked for
|
|
if (PolicyValue == sipvAuto1)
|
|
{
|
|
fNeedServer = FALSE;
|
|
MinMemoryNeeded = 64 * 1024 * 1024; // 64MB of RAM
|
|
}
|
|
else
|
|
{
|
|
ASSERT (PolicyValue == sipvAuto2);
|
|
fNeedServer = TRUE;
|
|
MinMemoryNeeded = 127 * 1024 * 1024; // 127MB of RAM
|
|
}
|
|
|
|
// if we need server, but this is workstation, no state info
|
|
if (fNeedServer && (ProductType == NtProductWinNt))
|
|
{
|
|
g_fServerSideDebugInfoEnabled = FALSE;
|
|
return;
|
|
}
|
|
|
|
// if we have less physical memory than the one we need,
|
|
// no state info
|
|
if (MemoryStatus.ullTotalPhys < (ULONGLONG) MinMemoryNeeded)
|
|
{
|
|
g_fServerSideDebugInfoEnabled = FALSE;
|
|
return;
|
|
}
|
|
|
|
g_fServerSideDebugInfoEnabled = TRUE;
|
|
}
|
|
|
|
typedef enum tagExceptionListParserState
|
|
{
|
|
elpsOutsideOfQuotes,
|
|
elpsInsideQuotes,
|
|
elpsReadingWhitespace,
|
|
elpsReadingCharacter
|
|
} ExceptionListParserState;
|
|
|
|
BOOL
|
|
DoesThisProcessCmdLineStartWithThisString (
|
|
IN LPWSTR CmdLine,
|
|
IN LPWSTR CmdLineStart
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Checks if the process name starts with the string given
|
|
in CmdLineStart
|
|
|
|
Arguments:
|
|
|
|
CmdLine - the process command line with path name stripped
|
|
CmdLineStart - the patter to match against
|
|
|
|
Return Value:
|
|
|
|
non-zero - there is a match
|
|
FALSE - there is no match
|
|
|
|
--*/
|
|
{
|
|
int CmdLineLength = RpcpStringLength(CmdLine);
|
|
int CmdLineStartLength = RpcpStringLength(CmdLineStart);
|
|
|
|
if (CmdLineLength >= CmdLineStartLength)
|
|
{
|
|
if (RpcpStringNCompare(CmdLine, CmdLineStart, CmdLineStartLength) == 0)
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
typedef enum tagExceptionListCharacterTypes
|
|
{
|
|
elctQuotes,
|
|
elctCharacter,
|
|
elctWhitespace
|
|
} ExceptionListCharacterTypes;
|
|
|
|
inline ExceptionListCharacterTypes
|
|
GetCharacterType (
|
|
RPC_CHAR Character
|
|
)
|
|
{
|
|
if (Character == '"')
|
|
return elctQuotes;
|
|
if (Character == ' ')
|
|
return elctWhitespace;
|
|
return elctCharacter;
|
|
}
|
|
|
|
RPC_STATUS
|
|
IsThisProcessAnException (
|
|
IN HKEY RegistryKey,
|
|
BOOL *fThisProcessIsException
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Checks whether the current process is in the exceptions list as
|
|
specified in the ExtErrorInfoExceptions registry key
|
|
|
|
Arguments:
|
|
|
|
RegistryKey - an open key to RPC_POLICY_SETTINGS
|
|
fThisProcessIsException - on output non-zero if this process is
|
|
an exception and FALSE otherwise
|
|
|
|
Return Value:
|
|
|
|
RPC_S_OK or RPC_S_* error
|
|
|
|
--*/
|
|
{
|
|
DWORD RegStatus;
|
|
RPC_CHAR *Buffer = NULL;
|
|
DWORD Type;
|
|
DWORD Size = 0;
|
|
int Retries = 5;
|
|
int State;
|
|
RPC_CHAR *CurrentPos;
|
|
RPC_CHAR *CurrentString;
|
|
RPC_CHAR *CommandLine;
|
|
RPC_CHAR *LastBackslash;
|
|
BOOL fIsSubstring;
|
|
|
|
*fThisProcessIsException = FALSE;
|
|
|
|
while (Retries > 0)
|
|
{
|
|
RegStatus = RegQueryValueExW(
|
|
RegistryKey,
|
|
L"ExtErrorInfoExceptions",
|
|
0,
|
|
&Type,
|
|
(LPBYTE) Buffer,
|
|
&Size
|
|
);
|
|
|
|
if ( (RegStatus != ERROR_SUCCESS) && (RegStatus != ERROR_MORE_DATA) )
|
|
{
|
|
if (RegStatus == ERROR_FILE_NOT_FOUND)
|
|
RegStatus = RPC_S_OK;
|
|
else
|
|
RegStatus = RPC_S_OUT_OF_MEMORY;
|
|
|
|
goto CleanupAndReturn;
|
|
}
|
|
else
|
|
{
|
|
if ((Buffer == NULL) || (RegStatus == ERROR_MORE_DATA))
|
|
{
|
|
if (Buffer)
|
|
delete Buffer;
|
|
|
|
Buffer = new RPC_CHAR[Size];
|
|
if (Buffer == NULL)
|
|
{
|
|
RegStatus = RPC_S_OUT_OF_MEMORY;
|
|
goto CleanupAndReturn;
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
if (Type != REG_SZ)
|
|
{
|
|
RegStatus = RPC_S_INTERNAL_ERROR;
|
|
goto CleanupAndReturn;
|
|
}
|
|
|
|
ASSERT(RegStatus == RPC_S_OK);
|
|
|
|
CommandLine = GetCommandLine();
|
|
LastBackslash = wcsrchr(CommandLine, '\\');
|
|
if (LastBackslash != NULL)
|
|
CommandLine = LastBackslash + 1;
|
|
|
|
// here, Buffer contains the exception string
|
|
// The format of the exception string is:
|
|
// "cmd_line_start" "cmd_line_start" ...
|
|
// If there is only one cmd_line_start, it doesn't have to be in double quotes
|
|
State = elpsOutsideOfQuotes;
|
|
CurrentPos = Buffer;
|
|
CurrentString = NULL;
|
|
while (*CurrentPos != 0)
|
|
{
|
|
switch (State)
|
|
{
|
|
case elpsOutsideOfQuotes:
|
|
switch (GetCharacterType(*CurrentPos))
|
|
{
|
|
case elctQuotes:
|
|
CurrentString = CurrentPos;
|
|
State = elpsInsideQuotes;
|
|
break;
|
|
|
|
case elctCharacter:
|
|
CurrentString = CurrentPos;
|
|
State = elpsReadingCharacter;
|
|
break;
|
|
|
|
case elctWhitespace:
|
|
State = elpsReadingWhitespace;
|
|
break;
|
|
|
|
default:
|
|
ASSERT(0);
|
|
}
|
|
break;
|
|
|
|
case elpsReadingCharacter:
|
|
switch (GetCharacterType(*CurrentPos))
|
|
{
|
|
case elctQuotes:
|
|
*CurrentPos = 0;
|
|
fIsSubstring = DoesThisProcessCmdLineStartWithThisString(
|
|
CommandLine, CurrentString);
|
|
if (fIsSubstring)
|
|
{
|
|
*fThisProcessIsException = TRUE;
|
|
goto CleanupAndReturn;
|
|
}
|
|
CurrentString = CurrentPos + 1;
|
|
State = elpsReadingCharacter;
|
|
break;
|
|
|
|
case elctWhitespace:
|
|
*CurrentPos = 0;
|
|
fIsSubstring = DoesThisProcessCmdLineStartWithThisString(
|
|
CommandLine, CurrentString);
|
|
if (fIsSubstring)
|
|
{
|
|
*fThisProcessIsException = TRUE;
|
|
goto CleanupAndReturn;
|
|
}
|
|
State = elpsReadingCharacter;
|
|
break;
|
|
|
|
// default:
|
|
// can be elpsReadingCharacter
|
|
}
|
|
break;
|
|
|
|
case elpsReadingWhitespace:
|
|
switch (GetCharacterType(*CurrentPos))
|
|
{
|
|
case elctQuotes:
|
|
CurrentString = CurrentPos + 1;
|
|
State = elpsInsideQuotes;
|
|
break;
|
|
|
|
case elctCharacter:
|
|
CurrentString = CurrentPos;
|
|
State = elpsReadingCharacter;
|
|
break;
|
|
|
|
// default:
|
|
// can be elctWhitespace
|
|
}
|
|
break;
|
|
|
|
case elpsInsideQuotes:
|
|
switch (GetCharacterType(*CurrentPos))
|
|
{
|
|
case elctQuotes:
|
|
*CurrentPos = 0;
|
|
fIsSubstring = DoesThisProcessCmdLineStartWithThisString(
|
|
CommandLine, CurrentString);
|
|
if (fIsSubstring)
|
|
{
|
|
*fThisProcessIsException = TRUE;
|
|
goto CleanupAndReturn;
|
|
}
|
|
State = elpsOutsideOfQuotes;
|
|
break;
|
|
|
|
//default:
|
|
// can be elctCharacter or elctWhitespace
|
|
}
|
|
break;
|
|
|
|
default:
|
|
ASSERT(0);
|
|
}
|
|
CurrentPos ++;
|
|
}
|
|
|
|
if (CurrentString)
|
|
{
|
|
fIsSubstring = DoesThisProcessCmdLineStartWithThisString(
|
|
CommandLine, CurrentString);
|
|
if (fIsSubstring)
|
|
{
|
|
*fThisProcessIsException = TRUE;
|
|
goto CleanupAndReturn;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
Retries --;
|
|
}
|
|
|
|
CleanupAndReturn:
|
|
if (Buffer)
|
|
delete Buffer;
|
|
if (Retries == 0)
|
|
RegStatus = RPC_S_INTERNAL_ERROR;
|
|
return RegStatus;
|
|
}
|
|
|
|
RPC_STATUS ReadPolicySettings(void)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Reads the policy settings for RPC and sets the appropriate global
|
|
variables.
|
|
|
|
Arguments:
|
|
|
|
void
|
|
|
|
Return Value:
|
|
|
|
RPC_S_OK or RPC_S_* error
|
|
|
|
--*/
|
|
{
|
|
HKEY RegistryKey;
|
|
DWORD RegStatus;
|
|
DWORD Result;
|
|
DWORD Type;
|
|
DWORD DwordSize = sizeof(DWORD);
|
|
BOOL ThisProcessIsException;
|
|
|
|
RegStatus = RegOpenKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
(LPWSTR) RPC_POLICY_SETTINGS,
|
|
0L, //Reserved
|
|
KEY_READ,
|
|
&RegistryKey
|
|
);
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
if (RegStatus == ERROR_FILE_NOT_FOUND)
|
|
{
|
|
SetAutoPolicySettings(sipvAuto2);
|
|
return RPC_S_OK;
|
|
}
|
|
return RPC_S_OUT_OF_MEMORY;
|
|
}
|
|
|
|
RegStatus = RegQueryValueExW(
|
|
RegistryKey,
|
|
L"StateInformation",
|
|
0,
|
|
&Type,
|
|
(LPBYTE) &Result,
|
|
&DwordSize
|
|
);
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
if (RegStatus == ERROR_FILE_NOT_FOUND)
|
|
{
|
|
SetAutoPolicySettings(sipvAuto2);
|
|
RegStatus = RPC_S_OK;
|
|
}
|
|
else
|
|
{
|
|
RegStatus = RPC_S_OUT_OF_MEMORY;
|
|
goto CleanupAndReturn;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
|
|
if (Type != REG_DWORD)
|
|
{
|
|
RegStatus = RPC_S_INTERNAL_ERROR;
|
|
goto CleanupAndReturn;
|
|
}
|
|
|
|
ASSERT(RegStatus == RPC_S_OK);
|
|
|
|
switch (Result)
|
|
{
|
|
case sipvNone:
|
|
// nothing to do - the client/server DebugInfoEnabled values
|
|
// are off by default
|
|
break;
|
|
|
|
case sipvAuto1:
|
|
case sipvAuto2:
|
|
SetAutoPolicySettings((StateInformationPolicyValues)Result);
|
|
break;
|
|
|
|
case sipvServer:
|
|
g_fServerSideDebugInfoEnabled = TRUE;
|
|
break;
|
|
|
|
case sipvFull:
|
|
g_fServerSideDebugInfoEnabled = TRUE;
|
|
g_fClientSideDebugInfoEnabled = TRUE;
|
|
break;
|
|
|
|
default:
|
|
RegStatus = RPC_S_INTERNAL_ERROR;
|
|
goto CleanupAndReturn;
|
|
}
|
|
|
|
}
|
|
|
|
RegStatus = RegQueryValueExW(
|
|
RegistryKey,
|
|
L"ExtErrorInformation",
|
|
0,
|
|
&Type,
|
|
(LPBYTE) &Result,
|
|
&DwordSize
|
|
);
|
|
|
|
if ( RegStatus != ERROR_SUCCESS )
|
|
{
|
|
if (RegStatus == ERROR_FILE_NOT_FOUND)
|
|
{
|
|
RegStatus = RPC_S_OK;
|
|
}
|
|
else
|
|
{
|
|
RegStatus = RPC_S_OUT_OF_MEMORY;
|
|
goto CleanupAndReturn;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
|
|
if (Type != REG_DWORD)
|
|
{
|
|
RegStatus = RPC_S_INTERNAL_ERROR;
|
|
goto CleanupAndReturn;
|
|
}
|
|
|
|
ASSERT(RegStatus == RPC_S_OK);
|
|
|
|
switch (Result)
|
|
{
|
|
case eeipvOff:
|
|
// nothing to do - the g_fSendEEInfo is already false
|
|
break;
|
|
|
|
case eeipvOnWithExceptions:
|
|
RegStatus = IsThisProcessAnException(RegistryKey,
|
|
&ThisProcessIsException);
|
|
if ((RegStatus != RPC_S_OK) || (ThisProcessIsException == FALSE))
|
|
g_fSendEEInfo = TRUE;
|
|
break;
|
|
|
|
case eeipvOffWithExceptions:
|
|
RegStatus = IsThisProcessAnException(RegistryKey,
|
|
&ThisProcessIsException);
|
|
if ((RegStatus == RPC_S_OK) && (ThisProcessIsException == TRUE))
|
|
g_fSendEEInfo = TRUE;
|
|
break;
|
|
|
|
case eeipvOn:
|
|
g_fSendEEInfo = TRUE;
|
|
break;
|
|
|
|
default:
|
|
RegStatus = RPC_S_INTERNAL_ERROR;
|
|
goto CleanupAndReturn;
|
|
}
|
|
}
|
|
|
|
CleanupAndReturn:
|
|
RegCloseKey(RegistryKey);
|
|
return RegStatus;
|
|
}
|
|
|
|
RPCRTAPI
|
|
RPC_STATUS
|
|
RPC_ENTRY
|
|
I_RpcSystemFunction001 (
|
|
IN SystemFunction001Commands FunctionCode,
|
|
IN void *InData,
|
|
OUT void *OutData
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Test hook.
|
|
|
|
Arguments:
|
|
|
|
FunctionCode - which test function to perform
|
|
|
|
InData - input data from the test function
|
|
|
|
OutData - output data from the test function
|
|
|
|
Return Value:
|
|
|
|
RPC_S_OK or RPC_S_* error
|
|
|
|
--*/
|
|
{
|
|
THREAD *Thread;
|
|
|
|
InitializeIfNecessary();
|
|
|
|
Thread = ThreadSelf();
|
|
if (!Thread)
|
|
return RPC_S_OUT_OF_MEMORY;
|
|
|
|
RpcpPurgeEEInfoFromThreadIfNecessary(Thread);
|
|
|
|
switch (FunctionCode)
|
|
{
|
|
case sf001cHttpSetInChannelTarget:
|
|
case sf001cHttpSetOutChannelTarget:
|
|
return HTTP2TestHook (FunctionCode,
|
|
InData,
|
|
OutData
|
|
);
|
|
break;
|
|
|
|
default:
|
|
return RPC_S_CANNOT_SUPPORT;
|
|
}
|
|
|
|
return RPC_S_OK;
|
|
}
|
|
|
|
#ifdef WINNT35_UUIDS
|
|
|
|
unsigned long
|
|
SomeLongValue (
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine, SomeShortValue, AnotherShortValue, and SomeCharacterValue
|
|
are used to generate the fields of a GUID if we can not determine
|
|
the network address from the network card (so we can generate a
|
|
UUID). These routines must generate some pseudo random values
|
|
based on the current time and/or the time since boot as well as the
|
|
current process and thread.
|
|
|
|
For the long value, we will use the current thread identifier and
|
|
current process identifier bitwise exclusive ored together.
|
|
|
|
For the two short values, we use the low part of the time field
|
|
(which is long, which we split into two values).
|
|
|
|
Finally, for the character value, we use a constant.
|
|
|
|
Return Value:
|
|
|
|
An unsigned long value will be returned.
|
|
|
|
--*/
|
|
{
|
|
TEB * CurrentTeb;
|
|
|
|
CurrentTeb = NtCurrentTeb();
|
|
return(((unsigned long) CurrentTeb->ClientId.UniqueThread)
|
|
^ ((unsigned long) CurrentTeb->ClientId.UniqueProcess));
|
|
}
|
|
|
|
|
|
unsigned short
|
|
SomeShortValue (
|
|
)
|
|
/*++
|
|
|
|
See SomeLongValue.
|
|
|
|
--*/
|
|
{
|
|
LARGE_INTEGER SystemTime;
|
|
|
|
for (;;)
|
|
{
|
|
NtQuerySystemTime(&SystemTime);
|
|
if (ThreadSelf()->TimeLow != SystemTime.LowPart)
|
|
break;
|
|
PauseExecution(1L);
|
|
}
|
|
ThreadSelf()->TimeLow = SystemTime.LowPart;
|
|
return((unsigned short) SystemTime.LowPart);
|
|
}
|
|
|
|
|
|
unsigned short
|
|
AnotherShortValue (
|
|
)
|
|
/*++
|
|
|
|
See SomeLongValue.
|
|
|
|
--*/
|
|
{
|
|
return((unsigned short) (ThreadSelf()->TimeLow >> 16));
|
|
}
|
|
|
|
|
|
unsigned char
|
|
SomeCharacterValue (
|
|
)
|
|
/*++
|
|
|
|
See SomeLongValue.
|
|
|
|
--*/
|
|
{
|
|
return(0x69);
|
|
}
|
|
|
|
#endif WINNT35_UUIDS
|
|
|
|
|
|
|
|
typedef struct {
|
|
RPC_STATUS RpcStatus;
|
|
long NtStatus;
|
|
} STATUS_MAPPING;
|
|
|
|
static const STATUS_MAPPING StatusMap[] =
|
|
{
|
|
{ RPC_S_OK, STATUS_SUCCESS },
|
|
{ RPC_S_INVALID_STRING_BINDING, RPC_NT_INVALID_STRING_BINDING },
|
|
{ RPC_S_WRONG_KIND_OF_BINDING, RPC_NT_WRONG_KIND_OF_BINDING },
|
|
{ RPC_S_INVALID_BINDING, RPC_NT_INVALID_BINDING },
|
|
{ RPC_S_PROTSEQ_NOT_SUPPORTED, RPC_NT_PROTSEQ_NOT_SUPPORTED },
|
|
{ RPC_S_INVALID_RPC_PROTSEQ, RPC_NT_INVALID_RPC_PROTSEQ },
|
|
{ RPC_S_INVALID_STRING_UUID, RPC_NT_INVALID_STRING_UUID },
|
|
{ RPC_S_INVALID_ENDPOINT_FORMAT, RPC_NT_INVALID_ENDPOINT_FORMAT },
|
|
{ RPC_S_INVALID_NET_ADDR, RPC_NT_INVALID_NET_ADDR },
|
|
{ RPC_S_NO_ENDPOINT_FOUND, RPC_NT_NO_ENDPOINT_FOUND },
|
|
{ RPC_S_INVALID_TIMEOUT, RPC_NT_INVALID_TIMEOUT },
|
|
{ RPC_S_OBJECT_NOT_FOUND, RPC_NT_OBJECT_NOT_FOUND },
|
|
{ RPC_S_ALREADY_REGISTERED, RPC_NT_ALREADY_REGISTERED },
|
|
{ RPC_S_TYPE_ALREADY_REGISTERED, RPC_NT_TYPE_ALREADY_REGISTERED },
|
|
{ RPC_S_ALREADY_LISTENING, RPC_NT_ALREADY_LISTENING },
|
|
{ RPC_S_NO_PROTSEQS_REGISTERED, RPC_NT_NO_PROTSEQS_REGISTERED },
|
|
{ RPC_S_NOT_LISTENING, RPC_NT_NOT_LISTENING },
|
|
{ RPC_S_UNKNOWN_MGR_TYPE, RPC_NT_UNKNOWN_MGR_TYPE },
|
|
{ RPC_S_UNKNOWN_IF, RPC_NT_UNKNOWN_IF },
|
|
{ RPC_S_NO_BINDINGS, RPC_NT_NO_BINDINGS },
|
|
{ RPC_S_NO_MORE_BINDINGS, RPC_NT_NO_MORE_BINDINGS },
|
|
{ RPC_S_NO_PROTSEQS, RPC_NT_NO_PROTSEQS },
|
|
{ RPC_S_CANT_CREATE_ENDPOINT, RPC_NT_CANT_CREATE_ENDPOINT },
|
|
{ RPC_S_OUT_OF_RESOURCES, RPC_NT_OUT_OF_RESOURCES },
|
|
{ RPC_S_SERVER_UNAVAILABLE, RPC_NT_SERVER_UNAVAILABLE },
|
|
{ RPC_S_SERVER_TOO_BUSY, RPC_NT_SERVER_TOO_BUSY },
|
|
{ RPC_S_INVALID_NETWORK_OPTIONS, RPC_NT_INVALID_NETWORK_OPTIONS },
|
|
{ RPC_S_NO_CALL_ACTIVE, RPC_NT_NO_CALL_ACTIVE },
|
|
{ RPC_S_CALL_FAILED, RPC_NT_CALL_FAILED },
|
|
{ RPC_S_CALL_CANCELLED, RPC_NT_CALL_CANCELLED },
|
|
{ RPC_S_CALL_FAILED_DNE, RPC_NT_CALL_FAILED_DNE },
|
|
{ RPC_S_PROTOCOL_ERROR, RPC_NT_PROTOCOL_ERROR },
|
|
{ RPC_S_UNSUPPORTED_TRANS_SYN, RPC_NT_UNSUPPORTED_TRANS_SYN },
|
|
{ RPC_S_SERVER_OUT_OF_MEMORY, STATUS_INSUFF_SERVER_RESOURCES },
|
|
{ RPC_S_UNSUPPORTED_TYPE, RPC_NT_UNSUPPORTED_TYPE },
|
|
{ RPC_S_INVALID_TAG, RPC_NT_INVALID_TAG },
|
|
{ RPC_S_INVALID_BOUND, RPC_NT_INVALID_BOUND },
|
|
{ RPC_S_NO_ENTRY_NAME, RPC_NT_NO_ENTRY_NAME },
|
|
{ RPC_S_INVALID_NAME_SYNTAX, RPC_NT_INVALID_NAME_SYNTAX },
|
|
{ RPC_S_UNSUPPORTED_NAME_SYNTAX, RPC_NT_UNSUPPORTED_NAME_SYNTAX },
|
|
{ RPC_S_UUID_NO_ADDRESS, RPC_NT_UUID_NO_ADDRESS },
|
|
{ RPC_S_DUPLICATE_ENDPOINT, RPC_NT_DUPLICATE_ENDPOINT },
|
|
{ RPC_S_UNKNOWN_AUTHN_TYPE, RPC_NT_UNKNOWN_AUTHN_TYPE },
|
|
{ RPC_S_MAX_CALLS_TOO_SMALL, RPC_NT_MAX_CALLS_TOO_SMALL },
|
|
{ RPC_S_STRING_TOO_LONG, RPC_NT_STRING_TOO_LONG },
|
|
{ RPC_S_PROTSEQ_NOT_FOUND, RPC_NT_PROTSEQ_NOT_FOUND },
|
|
{ RPC_S_PROCNUM_OUT_OF_RANGE, RPC_NT_PROCNUM_OUT_OF_RANGE },
|
|
{ RPC_S_BINDING_HAS_NO_AUTH, RPC_NT_BINDING_HAS_NO_AUTH },
|
|
{ RPC_S_UNKNOWN_AUTHN_SERVICE, RPC_NT_UNKNOWN_AUTHN_SERVICE },
|
|
{ RPC_S_UNKNOWN_AUTHN_LEVEL, RPC_NT_UNKNOWN_AUTHN_LEVEL },
|
|
{ RPC_S_INVALID_AUTH_IDENTITY, RPC_NT_INVALID_AUTH_IDENTITY },
|
|
{ RPC_S_UNKNOWN_AUTHZ_SERVICE, RPC_NT_UNKNOWN_AUTHZ_SERVICE },
|
|
{ EPT_S_INVALID_ENTRY, EPT_NT_INVALID_ENTRY },
|
|
{ EPT_S_CANT_PERFORM_OP, EPT_NT_CANT_PERFORM_OP },
|
|
{ EPT_S_NOT_REGISTERED, EPT_NT_NOT_REGISTERED },
|
|
{ RPC_S_NOTHING_TO_EXPORT, RPC_NT_NOTHING_TO_EXPORT },
|
|
{ RPC_S_INCOMPLETE_NAME, RPC_NT_INCOMPLETE_NAME },
|
|
{ RPC_S_INVALID_VERS_OPTION, RPC_NT_INVALID_VERS_OPTION },
|
|
{ RPC_S_NO_MORE_MEMBERS, RPC_NT_NO_MORE_MEMBERS },
|
|
{ RPC_S_NOT_ALL_OBJS_UNEXPORTED, RPC_NT_NOT_ALL_OBJS_UNEXPORTED },
|
|
{ RPC_S_INTERFACE_NOT_FOUND, RPC_NT_INTERFACE_NOT_FOUND },
|
|
{ RPC_S_ENTRY_ALREADY_EXISTS, RPC_NT_ENTRY_ALREADY_EXISTS },
|
|
{ RPC_S_ENTRY_NOT_FOUND, RPC_NT_ENTRY_NOT_FOUND },
|
|
{ RPC_S_NAME_SERVICE_UNAVAILABLE, RPC_NT_NAME_SERVICE_UNAVAILABLE },
|
|
{ RPC_S_INVALID_NAF_ID, RPC_NT_INVALID_NAF_ID },
|
|
{ RPC_S_CANNOT_SUPPORT, RPC_NT_CANNOT_SUPPORT },
|
|
{ RPC_S_NO_CONTEXT_AVAILABLE, RPC_NT_NO_CONTEXT_AVAILABLE },
|
|
{ RPC_S_INTERNAL_ERROR, RPC_NT_INTERNAL_ERROR },
|
|
{ RPC_S_ZERO_DIVIDE, RPC_NT_ZERO_DIVIDE },
|
|
{ RPC_S_ADDRESS_ERROR, RPC_NT_ADDRESS_ERROR },
|
|
{ RPC_S_FP_DIV_ZERO, RPC_NT_FP_DIV_ZERO },
|
|
{ RPC_S_FP_UNDERFLOW, RPC_NT_FP_UNDERFLOW },
|
|
{ RPC_S_FP_OVERFLOW, RPC_NT_FP_OVERFLOW },
|
|
{ RPC_X_NO_MORE_ENTRIES, RPC_NT_NO_MORE_ENTRIES },
|
|
{ RPC_X_SS_CHAR_TRANS_OPEN_FAIL, RPC_NT_SS_CHAR_TRANS_OPEN_FAIL },
|
|
{ RPC_X_SS_CHAR_TRANS_SHORT_FILE, RPC_NT_SS_CHAR_TRANS_SHORT_FILE },
|
|
{ RPC_X_SS_IN_NULL_CONTEXT, RPC_NT_SS_IN_NULL_CONTEXT },
|
|
{ RPC_X_SS_CONTEXT_MISMATCH, RPC_NT_SS_CONTEXT_MISMATCH },
|
|
{ RPC_X_SS_CONTEXT_DAMAGED, RPC_NT_SS_CONTEXT_DAMAGED },
|
|
{ RPC_X_SS_HANDLES_MISMATCH, RPC_NT_SS_HANDLES_MISMATCH },
|
|
{ RPC_X_SS_CANNOT_GET_CALL_HANDLE, RPC_NT_SS_CANNOT_GET_CALL_HANDLE },
|
|
{ RPC_X_NULL_REF_POINTER, RPC_NT_NULL_REF_POINTER },
|
|
{ RPC_X_ENUM_VALUE_OUT_OF_RANGE, RPC_NT_ENUM_VALUE_OUT_OF_RANGE },
|
|
{ RPC_X_BYTE_COUNT_TOO_SMALL, RPC_NT_BYTE_COUNT_TOO_SMALL },
|
|
{ RPC_X_BAD_STUB_DATA, RPC_NT_BAD_STUB_DATA },
|
|
{ ERROR_INVALID_PARAMETER, STATUS_INVALID_PARAMETER },
|
|
{ ERROR_OUTOFMEMORY, STATUS_NO_MEMORY },
|
|
{ ERROR_MAX_THRDS_REACHED, STATUS_NO_MEMORY },
|
|
{ ERROR_INSUFFICIENT_BUFFER, STATUS_BUFFER_TOO_SMALL },
|
|
{ ERROR_INVALID_SECURITY_DESCR, STATUS_INVALID_SECURITY_DESCR },
|
|
{ ERROR_ACCESS_DENIED, STATUS_ACCESS_DENIED },
|
|
{ ERROR_NOACCESS, STATUS_ACCESS_VIOLATION },
|
|
{ RPC_S_CALL_IN_PROGRESS, RPC_NT_CALL_IN_PROGRESS },
|
|
{ RPC_S_GROUP_MEMBER_NOT_FOUND, RPC_NT_GROUP_MEMBER_NOT_FOUND },
|
|
{ EPT_S_CANT_CREATE, EPT_NT_CANT_CREATE },
|
|
{ RPC_S_INVALID_OBJECT, RPC_NT_INVALID_OBJECT },
|
|
{ RPC_S_INVALID_ASYNC_HANDLE, RPC_NT_INVALID_ASYNC_HANDLE },
|
|
{ RPC_S_INVALID_ASYNC_CALL, RPC_NT_INVALID_ASYNC_CALL },
|
|
{ RPC_X_PIPE_CLOSED, RPC_NT_PIPE_CLOSED },
|
|
{ RPC_X_PIPE_EMPTY, RPC_NT_PIPE_EMPTY },
|
|
{ RPC_X_PIPE_DISCIPLINE_ERROR, RPC_NT_PIPE_DISCIPLINE_ERROR }
|
|
};
|
|
|
|
long RPC_ENTRY
|
|
I_RpcMapWin32Status (
|
|
IN RPC_STATUS Status
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine maps a WIN32 RPC status code into an NT RPC status code.
|
|
|
|
Arguments:
|
|
|
|
Status - Supplies the WIN32 RPC status code to be mapped.
|
|
|
|
Return Value:
|
|
|
|
The NT RPC status code corresponding to the WIN32 RPC status code
|
|
will be returned.
|
|
|
|
--*/
|
|
{
|
|
register int i;
|
|
for(i = 0; i < sizeof(StatusMap)/sizeof(STATUS_MAPPING); i++)
|
|
{
|
|
if (StatusMap[i].RpcStatus == Status)
|
|
{
|
|
return(StatusMap[i].NtStatus);
|
|
}
|
|
}
|
|
return(Status);
|
|
}
|