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.
 
 
 
 
 
 

2841 lines
60 KiB

/*++
Copyright (c) 1997 Microsoft Corporation
Module Name:
namefix.c
Abstract:
This module generates memdb entries for names that are to be used on Windows
NT. All names are validated, and incompatible names are placed in the
NewNames memdb category.
Names are organized into groups; within a group, each name must be unique,
but across groups, duplicate names are allowed. For example, the name
group "UserNames" holds all user names, and each user name must be unique.
The user name can be the same as the computer name, because the computer
name is stored in the "ComputerName" group.
The code here is extendable for other types of name collisions. To add
support for other types of names, simply add the group name to the
NAME_GROUP_LIST macro expansion below, then implement three functions:
pEnumGroupName
pValidateGroupName
pRecommendGroupName
You'll replace GroupName in the function names above with the name of your
actual group.
The code in this module should be the only place where names are validated
on the Win9x side.
Author:
Jim Schmidt (jimschm) 24-Dec-1997
Revision History:
jimschm 21-Jan-1998 Commented macro expansion list, added
g_DisableDomainChecks capability
--*/
#include "pch.h"
#include "cmn9xp.h"
#include <validc.h> // nt\private\inc
#define S_ILLEGAL_CHARS ILLEGAL_NAME_CHARS_STR TEXT("*")
#define DBG_NAMEFIX "NameFix"
#define MIN_UNLEN 20
//
// TEST_ALL_INCOMPATIBLE will force all names to be considered incompatible
// TEST_MANGLE_NAMES will force names to be invalid
//
//#define TEST_ALL_INCOMPATIBLE
//#define TEST_MANGLE_NAMES
/*++
Macro Expansion List Description:
NAME_GROUP_LIST lists each name category, such as computer name, domain name,
user name and so on. The macro expansion list automatically generates
three function prototypes for each name category. Also, the message ID is
used as the category identifier by external callers.
Line Syntax:
DEFMAC(GroupName, Id)
Arguments:
GroupName - Specifies the type of name. Must be a valid C function name.
The macro expansion list will generate prototypes for:
pEnum<GroupName>
pValidate<GroupName>
pRecommend<GroupName>
where, of course, <GroupName> is replaced by the text in the
macro declaration line.
All three functions must be implemented in this source file.
Id - Specifies the message ID giving the display name of the name group.
The name group is displayed in the list box that the user sees when
they are alerted there are some incompatible names on their machine.
Id is also used to uniquely identify a name group in some of the
routines below.
Variables Generated From List:
g_NameGroupRoutines
--*/
#define NAME_GROUP_LIST \
DEFMAC(ComputerDomain, MSG_COMPUTERDOMAIN_CATEGORY) \
DEFMAC(Workgroup, MSG_WORKGROUP_CATEGORY) \
DEFMAC(UserName, MSG_USERNAME_CATEGORY) \
DEFMAC(ComputerName, MSG_COMPUTERNAME_CATEGORY) \
//
// Macro expansion declarations
//
#define MAX_NAME 2048
typedef struct {
//
// The NAME_ENUM structure is passed uninitialized to pEnumGroupName
// function. The same structure is passed unchanged to subsequent
// calls to pEnumGroupName. Each enum function declares its
// params in this struct.
//
union {
struct {
//
// pEnumUser
//
USERENUM UserEnum;
};
};
//
// All enumeration routines must fill in the following if they
// return TRUE:
//
TCHAR Name[MAX_NAME];
} NAME_ENUM, *PNAME_ENUM;
typedef struct {
PCTSTR GroupName;
TCHAR AuthenticatingAgent[MAX_COMPUTER_NAME];
BOOL FromUserInterface;
UINT FailureMsg;
BOOL DomainLogonEnabled;
} NAME_GROUP_CONTEXT, *PNAME_GROUP_CONTEXT;
typedef BOOL (ENUM_NAME_PROTOTYPE)(PNAME_GROUP_CONTEXT Context, PNAME_ENUM EnumPtr, BOOL First);
typedef ENUM_NAME_PROTOTYPE * ENUM_NAME_FN;
typedef BOOL (VALIDATE_NAME_PROTOTYPE)(PNAME_GROUP_CONTEXT Context, PCTSTR NameCandidate);
typedef VALIDATE_NAME_PROTOTYPE * VALIDATE_NAME_FN;
typedef VOID (RECOMMEND_NAME_PROTOTYPE)(PNAME_GROUP_CONTEXT Context, PCTSTR InvalidName, PTSTR RecommendedNameBuf);
typedef RECOMMEND_NAME_PROTOTYPE * RECOMMEND_NAME_FN;
typedef struct {
UINT NameId;
PCTSTR GroupName;
ENUM_NAME_FN Enum;
VALIDATE_NAME_FN Validate;
RECOMMEND_NAME_FN Recommend;
NAME_GROUP_CONTEXT Context;
} NAME_GROUP_ROUTINES, *PNAME_GROUP_ROUTINES;
//
// Automatic arrays and prototypes
//
// The prototoypes
#define DEFMAC(x,id) ENUM_NAME_PROTOTYPE pEnum##x;
NAME_GROUP_LIST
#undef DEFMAC
#define DEFMAC(x,id) VALIDATE_NAME_PROTOTYPE pValidate##x;
NAME_GROUP_LIST
#undef DEFMAC
#define DEFMAC(x,id) RECOMMEND_NAME_PROTOTYPE pRecommend##x;
NAME_GROUP_LIST
#undef DEFMAC
// The array of functions
#define DEFMAC(x,id) {id, TEXT(#x), pEnum##x, pValidate##x, pRecommend##x},
NAME_GROUP_ROUTINES g_NameGroupRoutines[] = {
NAME_GROUP_LIST /* , */
{0, NULL, NULL, NULL, NULL, NULL}
};
//
// Local prototypes
//
BOOL
pDoesNameExistInMemDb (
IN PNAME_GROUP_CONTEXT Context,
IN PCTSTR UserName
);
//
// Implementation
//
PNAME_GROUP_ROUTINES
pGetNameGroupById (
IN UINT MessageId
)
/*++
Routine Description:
pGetNameGroupById finds a group by searching the list for a message ID. The
message ID is the unique identifier for the group.
Arguments:
MessageId - Specifies the unique ID of the group to find
Return Value:
A pointer to the group struct, or NULL if one was not found.
--*/
{
INT i;
for (i = 0 ; g_NameGroupRoutines[i].GroupName ; i++) {
if (g_NameGroupRoutines[i].NameId == MessageId) {
return &g_NameGroupRoutines[i];
}
}
return NULL;
}
PNAME_GROUP_CONTEXT
pGetNameGroupContextById (
IN UINT MessageId
)
/*++
Routine Description:
pGetNameGroupById finds a group by searching the list for a message ID. The
message ID is the unique identifier for the group. The return value is
the context structure used by the group.
Arguments:
MessageId - Specifies the unique ID of the group to find
Return Value:
A pointer to the group's context struct, or NULL if one was not found.
--*/
{
INT i;
for (i = 0 ; g_NameGroupRoutines[i].GroupName ; i++) {
if (g_NameGroupRoutines[i].NameId == MessageId) {
return &g_NameGroupRoutines[i].Context;
}
}
return NULL;
}
BOOL
pEnumComputerName (
IN PNAME_GROUP_CONTEXT Context,
IN OUT PNAME_ENUM EnumPtr,
IN BOOL First
)
/*++
Routine Description:
pEnumComputerName obtains the computer name and returns it
in the EnumPtr struct. If no name is assigned to the computer,
an empty string is returned.
Arguments:
Context - Unused (holds context about name group)
EnumPtr - Receives the computer name
First - Specifies TRUE on the first call to pEnumComputerName,
or FALSE on subseqeuent calls to pEnumComputerName.
Return Value:
If First is TRUE, returns TRUE if a name is enumerated, or FALSE
if the name is not valid.
If First is FALSE, always returns FALSE.
--*/
{
DWORD Size;
if (!First) {
return FALSE;
}
//
// Get the computer name
//
Size = sizeof (EnumPtr->Name) / sizeof (EnumPtr->Name[0]);
if (!GetComputerName (EnumPtr->Name, &Size)) {
EnumPtr->Name[0] = 0;
}
return TRUE;
}
BOOL
pEnumUserName (
IN PNAME_GROUP_CONTEXT Context,
IN OUT PNAME_ENUM EnumPtr,
IN BOOL First
)
/*++
Routine Description:
pEnumUserName enumerates all users on the machine via the EnumFirstUser/
EnumNextUser APIs. It does not enumerate the fixed names.
Like the other enumeration routines, this routine is called until it
returns FALSE, so that all resources are cleaned up correctly.
Arguments:
Context - Unused (holds context about name group)
EnumPtr - Specifies the current enumeration state. Receives the enumerated
user name.
First - Specifies TRUE on the first call to pEnumUserName,
or FALSE on subseqeuent calls to pEnumUserName.
Return Value:
TRUE if a user was enumerated, or FALSE if all users were enumerated.
--*/
{
//
// Enumerate the next user
//
if (First) {
if (!EnumFirstUser (&EnumPtr->UserEnum, ENUMUSER_DO_NOT_MAP_HIVE)) {
LOG ((LOG_ERROR, "No users to enumerate"));
return FALSE;
}
} else {
if (!EnumNextUser (&EnumPtr->UserEnum)) {
return FALSE;
}
}
//
// Special case -- ignore default user
//
while (*EnumPtr->UserEnum.UserName == 0) {
if (!EnumNextUser (&EnumPtr->UserEnum)) {
return FALSE;
}
}
//
// Copy user name to name buffer
//
StringCopy (EnumPtr->Name, EnumPtr->UserEnum.UserName);
return TRUE;
}
BOOL
pEnumWorkgroup (
IN PNAME_GROUP_CONTEXT Context,
IN OUT PNAME_ENUM EnumPtr,
IN BOOL First
)
/*++
Routine Description:
pEnumWorkgroup obtains the workgroup name and returns it
in the EnumPtr struct. If the VNETSUP support is not
installed, or the workgroup name is empty, this routine
returns FALSE.
Arguments:
Context - Receives AuthenticatingAgent value
EnumPtr - Receives the computer domain name
First - Specifies TRUE on the first call to pEnumWorkgroup,
or FALSE on subseqeuent calls to pEnumWorkgroup.
Return Value:
If First is TRUE, returns TRUE if a name is enumerated, or FALSE
if the name is not valid.
If First is FALSE, always returns FALSE.
--*/
{
HKEY VnetsupKey;
PCTSTR StrData;
if (!First) {
return FALSE;
}
EnumPtr->Name[0] = 0;
//
// Obtain the workgroup name into EnumPtr->Name
//
VnetsupKey = OpenRegKeyStr (S_VNETSUP);
if (VnetsupKey) {
StrData = GetRegValueString (VnetsupKey, S_WORKGROUP);
if (StrData) {
_tcssafecpy (EnumPtr->Name, StrData, MAX_COMPUTER_NAME);
MemFree (g_hHeap, 0, StrData);
}
ELSE_DEBUGMSG ((DBG_WARNING, "pEnumWorkgroup: Workgroup value does not exist"));
CloseRegKey (VnetsupKey);
}
ELSE_DEBUGMSG ((DBG_WARNING, "pEnumWorkgroup: VNETSUP key does not exist"));
return EnumPtr->Name[0] != 0;
}
BOOL
pEnumComputerDomain (
IN PNAME_GROUP_CONTEXT Context,
IN OUT PNAME_ENUM EnumPtr,
IN BOOL First
)
/*++
Routine Description:
pEnumComputerDomain obtains the workgroup name and returns it in the EnumPtr
struct. If the MS Networking client is not installed, this routine returns
FALSE.
This routine also obtains the AuthenticatingAgent value, and stores it in
the Context structure for use by pRecommendComputerDomain.
Arguments:
Context - Receives AuthenticatingAgent value
EnumPtr - Receives the computer domain name
First - Specifies TRUE on the first call to pEnumComputerDomain,
or FALSE on subseqeuent calls to pEnumComputerDomain.
Return Value:
If First is TRUE, returns TRUE if a name is enumerated, or FALSE
if the name is not valid.
If First is FALSE, always returns FALSE.
--*/
{
HKEY Key;
HKEY NetLogonKey;
HKEY VnetsupKey;
PBYTE Data;
PCTSTR StrData;
BOOL b = TRUE;
if (!First) {
return FALSE;
}
EnumPtr->Name[0] = 0;
Context->DomainLogonEnabled = FALSE;
//
// Is the MS Networking client installed?
//
Key = OpenRegKeyStr (S_MSNP32);
if (!Key) {
//
// MS Networking client is not installed. Return FALSE
// because any Win9x workgroup name will work with NT.
//
DEBUGMSG ((DBG_NAMEFIX, "pEnumComputerDomain: MS Networking client is not installed."));
return FALSE;
}
__try {
//
// Determine if the domain logon is enabled
//
NetLogonKey = OpenRegKeyStr (S_LOGON_KEY);
if (NetLogonKey) {
Data = (PBYTE) GetRegValueBinary (NetLogonKey, S_LM_LOGON);
if (Data) {
if (*Data) {
Context->DomainLogonEnabled = TRUE;
}
MemFree (g_hHeap, 0, Data);
}
CloseRegKey (NetLogonKey);
}
//
// If no domain logon is enabled, return FALSE, because
// any Win9x workgroup name will work with NT.
//
if (!Context->DomainLogonEnabled) {
DEBUGMSG ((DBG_NAMEFIX, "pEnumComputerDomain: Domain logon is not enabled."));
b = FALSE;
__leave;
}
//
// Obtain the workgroup name into EnumPtr->Name; we will try
// to use this as the NT computer domain.
//
VnetsupKey = OpenRegKeyStr (S_VNETSUP);
if (VnetsupKey) {
StrData = GetRegValueString (VnetsupKey, S_WORKGROUP);
if (StrData) {
_tcssafecpy (EnumPtr->Name, StrData, MAX_COMPUTER_NAME);
MemFree (g_hHeap, 0, StrData);
}
ELSE_DEBUGMSG ((DBG_WARNING, "pEnumComputerDomain: Workgroup value does not exist"));
CloseRegKey (VnetsupKey);
}
ELSE_DEBUGMSG ((DBG_WARNING, "pEnumComputerDomain: VNETSUP key does not exist"));
//
// Obtain the AuthenticatingAgent value from Key and stick it in
// Context->AuthenticatingAgent
//
StrData = GetRegValueString (Key, S_AUTHENTICATING_AGENT);
if (StrData) {
//
// Copy AuthenticatingAgent to enum struct
//
_tcssafecpy (Context->AuthenticatingAgent, StrData, MAX_COMPUTER_NAME);
MemFree (g_hHeap, 0, StrData);
} else {
Context->AuthenticatingAgent[0] = 0;
LOG ((LOG_ERROR,"Domain Logon enabled, but AuthenticatingAgent value is missing"));
}
}
__finally {
CloseRegKey (Key);
}
return b;
}
BOOL
pValidateNetName (
OUT PNAME_GROUP_CONTEXT Context, OPTIONAL
IN PCTSTR NameCandidate,
IN BOOL SpacesAllowed,
IN BOOL DotSpaceCheck,
IN UINT MaxLength,
OUT PCSTR *OffendingChar OPTIONAL
)
/*++
Routine Description:
pValidateNetName performs a check to see if the specified name is valid on
NT 5.
Arguments:
Context - Receives the error message ID, if any error occurred.
NameCandidate - Specifies the name to validate.
SpacesAllowed - Specifies TRUE if spaces are allowed in the name, or FALSE
if not.
MaxLength - Specifies the max characters that can be in the name.
DotSpaceCheck - Specifies TRUE if the name cannot consist only of a dot and
a space, or FALSE if it can.
OffendingChar - Receives the pointer to the character that caused the
problem, or NULL if no error or the error was caused by
something other than a character set mismatch or length test.
Return Value:
TRUE if the name is valid, or FALSE if it is not.
--*/
{
PCTSTR p;
PCTSTR LastNonSpaceChar;
CHARTYPE ch;
BOOL allDigits;
if (OffendingChar) {
*OffendingChar = NULL;
}
//
// Minimum length test
//
if (!NameCandidate[0]) {
if (Context) {
Context->FailureMsg = MSG_INVALID_EMPTY_NAME_POPUP;
}
return FALSE;
}
//
// Maximum length test
//
if (CharCount (NameCandidate) > MaxLength) {
if (Context) {
Context->FailureMsg = MSG_INVALID_COMPUTERNAME_LENGTH_POPUP;
}
if (OffendingChar) {
*OffendingChar = TcharCountToPointer (NameCandidate, MaxLength);
}
return FALSE;
}
//
// No leading spaces allowed
//
if (_tcsnextc (NameCandidate) == TEXT(' ')) {
if (Context) {
Context->FailureMsg = MSG_INVALID_COMPUTERNAME_CHAR_POPUP;
}
if (OffendingChar) {
*OffendingChar = NameCandidate;
}
return FALSE;
}
//
// No invalid characters
//
ch = ' ';
LastNonSpaceChar = NULL;
allDigits = TRUE;
for (p = NameCandidate ; *p ; p = _tcsinc (p)) {
ch = _tcsnextc (p);
if (_tcschr (S_ILLEGAL_CHARS, ch) != NULL ||
(ch == TEXT(' ') && !SpacesAllowed)
) {
if (OffendingChar) {
*OffendingChar = p;
}
if (Context) {
Context->FailureMsg = MSG_INVALID_COMPUTERNAME_CHAR_POPUP;
}
return FALSE;
}
if (ch != TEXT('.') && ch != TEXT(' ')) {
DotSpaceCheck = FALSE;
}
if (ch != TEXT(' ')) {
LastNonSpaceChar = p;
}
if (allDigits) {
if (ch < TEXT('0') || ch > TEXT('9')) {
allDigits = FALSE;
}
}
}
if (allDigits) {
if (OffendingChar) {
*OffendingChar = NameCandidate;
}
if (Context) {
Context->FailureMsg = MSG_INVALID_COMPUTERNAME_CHAR_POPUP;
}
return FALSE;
}
//
// No trailing dot
//
if (ch == TEXT('.')) {
MYASSERT (LastNonSpaceChar);
if (OffendingChar) {
*OffendingChar = LastNonSpaceChar;
}
if (Context) {
Context->FailureMsg = MSG_INVALID_COMPUTERNAME_CHAR_POPUP;
}
return FALSE;
}
//
// No trailing space
//
if (ch == TEXT(' ')) {
MYASSERT (LastNonSpaceChar);
if (OffendingChar) {
*OffendingChar = _tcsinc (LastNonSpaceChar);
}
if (Context) {
Context->FailureMsg = MSG_INVALID_COMPUTERNAME_CHAR_POPUP;
}
return FALSE;
}
//
// Dot-space only check
//
if (DotSpaceCheck) {
if (OffendingChar) {
*OffendingChar = NameCandidate;
}
if (Context) {
Context->FailureMsg = MSG_INVALID_USERNAME_SPACEDOT_POPUP;
}
return FALSE;
}
return TRUE;
}
BOOL
ValidateDomainNameChars (
IN PCTSTR NameCandidate
)
{
return pValidateNetName (
NULL,
NameCandidate,
FALSE,
FALSE,
min (MAX_SERVER_NAME, DNLEN),
NULL
);
}
BOOL
ValidateUserNameChars (
IN PCTSTR NameCandidate
)
{
return pValidateNetName (
NULL,
NameCandidate,
TRUE,
TRUE,
min (MAX_USER_NAME, MIN_UNLEN),
NULL
);
}
BOOL
pValidateComputerName (
IN PNAME_GROUP_CONTEXT Context,
IN PCTSTR NameCandidate
)
/*++
Routine Description:
pValidateComputerName makes sure the specified name is not
empty, is not longer than 15 characters, and consists only
of characters legal for a computer name. Also, if the name
collides with a user name, then the computer name is invalid.
Arguments:
Context - Unused (holds context about name group)
NameCandidate - Specifies name to validate
Return Value:
TRUE if the name is valid, FALSE otherwise.
--*/
{
BOOL b;
//PNAME_GROUP_CONTEXT UserContext;
//UserContext = pGetNameGroupContextById (MSG_USERNAME_CATEGORY);
//if (pDoesNameExistInMemDb (UserContext, NameCandidate)) {
// return FALSE;
//}
b = pValidateNetName (
Context,
NameCandidate,
FALSE,
FALSE,
MAX_COMPUTER_NAME,
NULL
);
return b;
}
BOOL
pValidateWorkgroup (
IN PNAME_GROUP_CONTEXT Context,
IN PCTSTR NameCandidate
)
/*++
Routine Description:
pValidateWorkgroup makes sure the specified name is not
empty, is not longer than 15 characters, and consists only
of characters legal for a computer name.
If domain logon is enabled, this routine always returns
TRUE.
Arguments:
Context - Unused (holds context about name group)
NameCandidate - Specifies name to validate
Return Value:
TRUE if the name is valid, FALSE otherwise.
--*/
{
PNAME_GROUP_CONTEXT DomainContext;
//
// Return TRUE if domain is enabled
//
DomainContext = pGetNameGroupContextById (MSG_COMPUTERDOMAIN_CATEGORY);
if (DomainContext && DomainContext->DomainLogonEnabled) {
return TRUE;
}
//
// A workgroup name is a domain name, but spaces are allowed.
//
return pValidateNetName (Context, NameCandidate, TRUE, FALSE, DNLEN, NULL);
}
BOOL
pValidateUserName (
IN PNAME_GROUP_CONTEXT Context,
IN PCTSTR NameCandidate
)
/*++
Routine Description:
pValidateUserName makes sure the specified name is not empty,
is not longer than 20 characters, consists only of characters
legal for a user name, and does not exist in memdb's NewName
or InUseName categories.
Arguments:
Context - Specifies context info for the UserName name group,
used for memdb operations.
NameCandidate - Specifies name to validate
Return Value:
TRUE if the name is valid, FALSE otherwise.
--*/
{
BOOL b;
//
// Validate name
//
b = pValidateNetName (Context, NameCandidate, TRUE, TRUE, min (MAX_USER_NAME, MIN_UNLEN), NULL);
if (!b && Context->FailureMsg == MSG_INVALID_COMPUTERNAME_LENGTH_POPUP) {
Context->FailureMsg = MSG_INVALID_USERNAME_LENGTH_POPUP;
}
if (!b) {
return FALSE;
}
//
// Check for existence in memdb
//
if (pDoesNameExistInMemDb (Context, NameCandidate)) {
Context->FailureMsg = MSG_INVALID_USERNAME_DUPLICATE_POPUP;
return FALSE;
}
return TRUE;
}
BOOL
pValidateComputerDomain (
IN PNAME_GROUP_CONTEXT Context,
IN PCTSTR NameCandidate
)
/*++
Routine Description:
pValidateComputerDomain performs the same validatation that
pValidateComputerName performs. Therefore, it simply calls
pValidateComputerName.
If the name comes from the registry, and not the user interface, then we
check to see if the workgroup name actually refers to a domain controller.
If it does, the name is returned as valid; otherwise, the name is returned
as invalid, even though it may consist of valid characters.
If the name comes from the user interface, we assume the UI code will do
the validation to see if the name is an actual server. This allows the UI
to override the API, because the API may not work properly on all networks.
Arguments:
Context - Specifies context of the ComputerDomain name group. In particular,
the FromUserInterface member tells us to ignore validation of the
domain name via the NetServerGetInfo API.
NameCandidate - Specifies domain name to validate
Return Value:
TRUE if the domain name is legal, or FALSE if it is not.
--*/
{
TCHAR NewComputerName[MAX_COMPUTER_NAME];
if (!pValidateNetName (Context, NameCandidate, FALSE, FALSE, DNLEN, NULL)) {
return FALSE;
}
if (!Context->FromUserInterface) {
if (GetUpgradeComputerName (NewComputerName)) {
// 1 == account was found, 0 == account does not exist, -1 == no response
if (1 != DoesComputerAccountExistOnDomain (
NameCandidate,
NewComputerName,
TRUE
)) {
return FALSE;
}
}
}
return TRUE;
}
BOOL
pCleanUpNetName (
IN PNAME_GROUP_CONTEXT Context,
IN OUT PTSTR Name,
IN UINT NameType
)
/*++
Routine Description:
pCleanUpNetName eliminates all characters that are invalid from the
specified name.
NOTE: We could add some smarts here, such as translation of
spaces to dashes, and so on.
Arguments:
Context - Unused; passed along to pValidateComputerName.
Name - Specifies the name containing potentially invalid characters.
Receives the name with all invalid characters removed.
NameType - Specifies the type of name to clean up
Return Value:
TRUE if the resulting name is valid, or FALSE if the resulting
name is still not valid.
--*/
{
TCHAR TempBuf[MAX_COMPUTER_NAME];
PTSTR p;
PTSTR q;
UINT Len;
BOOL b;
//
// Delete all the invalid characters
//
_tcssafecpy (TempBuf, Name, MAX_COMPUTER_NAME);
for (;;) {
p = NULL;
b = FALSE;
switch (NameType) {
case MSG_COMPUTERNAME_CATEGORY:
b = pValidateNetName (Context, TempBuf, TRUE, FALSE, MAX_COMPUTER_NAME, &p);
break;
case MSG_WORKGROUP_CATEGORY:
b = pValidateNetName (Context, TempBuf, TRUE, FALSE, DNLEN, &p);
break;
case MSG_COMPUTERDOMAIN_CATEGORY:
b = pValidateNetName (Context, TempBuf, FALSE, FALSE, DNLEN, &p);
break;
case MSG_USERNAME_CATEGORY:
b = pValidateNetName (Context, TempBuf, TRUE, TRUE, MIN_UNLEN, &p);
break;
}
if (b || !p) {
break;
}
q = _tcsinc (p);
Len = ByteCount (q) + sizeof (TCHAR);
MoveMemory (p, q, Len);
}
if (b) {
//
// Do not allow names that contain a lot of invalid characters
//
if (CharCount (Name) - 3 > CharCount (TempBuf)) {
b = FALSE;
}
}
if (!b) {
// Empty out recommended name
*Name = 0;
}
if (b) {
StringCopy (Name, TempBuf);
switch (NameType) {
case MSG_COMPUTERNAME_CATEGORY:
b = pValidateComputerName (Context, Name);
break;
case MSG_WORKGROUP_CATEGORY:
b = pValidateWorkgroup (Context, Name);
break;
case MSG_COMPUTERDOMAIN_CATEGORY:
b = pValidateComputerDomain (Context, Name);
break;
case MSG_USERNAME_CATEGORY:
b = pValidateUserName (Context, Name);
break;
}
}
return b;
}
VOID
pRecommendComputerName (
IN PNAME_GROUP_CONTEXT Context,
IN PCTSTR InvalidName,
OUT PTSTR RecommendedName
)
/*++
Routine Description:
pRecommendComputerName obtains the current user's name and
returns it for use as a computer name. If the user's name
has characters that cannot be used in a computer name,
the invalid characters are removed. If the name is still
invalid, then a static string is returned.
Arguments:
Context - Unused (holds context about name group)
InvalidName - Specifies the current invalid name, or an empty
string if no name exists.
RecommendedName - Receives the recommended name.
Return Value:
none
--*/
{
DWORD Size;
PCTSTR p;
PCTSTR ArgArray[1];
//
// Try to clean up the invalid name
//
if (*InvalidName) {
_tcssafecpy (RecommendedName, InvalidName, MAX_COMPUTER_NAME);
if (pCleanUpNetName (Context, RecommendedName, MSG_COMPUTERNAME_CATEGORY)) {
return;
}
}
//
// Generate a suggestion from the user name
//
Size = MAX_COMPUTER_NAME;
if (!GetUserName (RecommendedName, &Size)) {
*RecommendedName = 0;
} else {
CharUpper (RecommendedName);
ArgArray[0] = RecommendedName;
p = ParseMessageID (MSG_COMPUTER_REPLACEMENT_NAME, ArgArray);
MYASSERT (p);
if (p) {
_tcssafecpy (RecommendedName, p, MAX_COMPUTER_NAME);
FreeStringResource (p);
}
ELSE_DEBUGMSG ((DBG_ERROR, "Failed to parse message resource for MSG_COMPUTER_REPLACEMENT_NAME. Check localization."));
}
//
// Try to clean up invalid computer name chars in user name
//
if (pCleanUpNetName (Context, RecommendedName, MSG_COMPUTERNAME_CATEGORY)) {
return;
}
//
// All else has failed; obtain static computer name string
//
p = GetStringResource (MSG_RECOMMENDED_COMPUTER_NAME);
MYASSERT (p);
if (p) {
StringCopy (RecommendedName, p);
FreeStringResource (p);
}
ELSE_DEBUGMSG ((DBG_ERROR, "Failed to parse message resource for MSG_RECOMMENDED_COMPUTER_NAME. Check localization."));
}
VOID
pRecommendWorkgroup (
IN PNAME_GROUP_CONTEXT Context,
IN PCTSTR InvalidName,
OUT PTSTR RecommendedName
)
/*++
Routine Description:
pRecommendWorkgroupName tries to clean up the invalid workgroup name, and
only if necessary recommends the name Workgroup.
Arguments:
Context - Unused (holds context about name group)
InvalidName - Specifies the current invalid name, or an empty
string if no name exists.
RecommendedName - Receives the recommended name.
Return Value:
none
--*/
{
PCTSTR p;
//
// Try to clean up the invalid name
//
if (*InvalidName) {
_tcssafecpy (RecommendedName, InvalidName, MAX_COMPUTER_NAME);
if (pCleanUpNetName (Context, RecommendedName, MSG_WORKGROUP_CATEGORY)) {
return;
}
}
//
// All else has failed; obtain static workgroup string
//
p = GetStringResource (MSG_RECOMMENDED_WORKGROUP_NAME);
MYASSERT (p);
StringCopy (RecommendedName, p);
FreeStringResource (p);
}
VOID
pRecommendUserName (
IN PNAME_GROUP_CONTEXT Context,
IN PCTSTR InvalidName,
OUT PTSTR RecommendedName
)
/*++
Routine Description:
pRecommendUserName tries to clean up the specified invalid
user name. If that fails, this routine generates a
generic user name (such as Windows User). If the generic
name is not valid, numbers are appended until a unique,
valid name is found.
Arguments:
Context - Specifies settings for the UserName name group context,
including the group name itself. This context is used
in memdb operations to validate the name.
InvalidName - Specifies the current invalid name, or an empty
string if no name exists.
RecommendedName - Receives the recommended name.
Return Value:
none
--*/
{
PCTSTR p;
UINT Sequencer;
//
// Attempt to clean out invalid characters from the user
// name.
//
_tcssafecpy (RecommendedName, InvalidName, MAX_USER_NAME);
if (pCleanUpNetName (Context, RecommendedName, MSG_USERNAME_CATEGORY)) {
return;
}
//
// If there are some characters left, and there is room for
// a sequencer, just add the sequencer.
//
if (*RecommendedName) {
p = DuplicateText (RecommendedName);
MYASSERT (p);
for (Sequencer = 1 ; Sequencer < 10 ; Sequencer++) {
wsprintf (RecommendedName, TEXT("%s-%u"), p, Sequencer);
if (pValidateUserName (Context, RecommendedName)) {
break;
}
}
FreeText (p);
if (Sequencer < 10) {
return;
}
}
//
// Obtain a generic name
//
p = GetStringResource (MSG_RECOMMENDED_USER_NAME);
MYASSERT (p);
if (p) {
__try {
if (pValidateUserName (Context, p)) {
StringCopy (RecommendedName, p);
} else {
for (Sequencer = 2 ; Sequencer < 100000 ; Sequencer++) {
wsprintf (RecommendedName, TEXT("%s %u"), p, Sequencer);
if (pValidateUserName (Context, RecommendedName)) {
break;
}
}
if (Sequencer == 100000) {
LOG ((LOG_ERROR, "Sequencer hit %u", Sequencer));
}
}
}
__finally {
FreeStringResource (p);
}
}
ELSE_DEBUGMSG ((DBG_ERROR, "Could not retrieve string resource MSG_RECOMMENDED_USER_NAME. Check localization."));
}
VOID
pRecommendComputerDomain (
IN PNAME_GROUP_CONTEXT Context,
IN PCTSTR InvalidName,
OUT PTSTR RecommendedName
)
/*++
Routine Description:
pRecommendComputerDomain returns the value of AuthenticatingAgent
stored in the Context structure by pEnumComputerDomain.
Arguments:
Context - Specifies the name group context structure, which holds
the computer domain found by pEnumComputerDomain.
InvalidName - Specifies the current invalid name, or an empty
string if no name exists.
RecommendedName - Receives the recommended name.
Return Value:
none
--*/
{
StringCopy (RecommendedName, Context->AuthenticatingAgent);
}
BOOL
ValidateName (
IN HWND ParentWnd, OPTIONAL
IN PCTSTR NameGroup,
IN PCTSTR NameCandidate
)
/*++
Routine Description:
ValidateName is called by the UI to perform validation on the
specified name.
Arguments:
ParentWnd - Specifies the handle used for popups that tell the user
what is wrong with the name they entered. If NULL, no
UI is generated.
NameGroup - Specifies the name group, a static string that defines
what characters are legal for the name.
NameCandidate - Specifies the name to validate
Return Value:
TRUE if the name is valid, or FALSE if it is not valid.
--*/
{
INT i;
BOOL b;
//
// Scan the g_NameGroupRoutines array for the name grup
//
for (i = 0 ; g_NameGroupRoutines[i].GroupName ; i++) {
if (StringIMatch (g_NameGroupRoutines[i].GroupName, NameGroup)) {
break;
}
}
if (!g_NameGroupRoutines[i].GroupName) {
DEBUGMSG ((DBG_WHOOPS, "ValidateName: Don't know how to validate %s names", NameGroup));
LOG ((LOG_ERROR, "Don't know how to validate %s names", NameGroup));
return TRUE;
}
g_NameGroupRoutines[i].Context.FromUserInterface = TRUE;
b = g_NameGroupRoutines[i].Validate (&g_NameGroupRoutines[i].Context, NameCandidate);
if (!b && ParentWnd) {
OkBox (ParentWnd, g_NameGroupRoutines[i].Context.FailureMsg);
}
g_NameGroupRoutines[i].Context.FromUserInterface = FALSE;
return b;
}
BOOL
pDoesNameExistInMemDb (
IN PNAME_GROUP_CONTEXT Context,
IN PCTSTR Name
)
/*++
Routine Description:
pDoesUserExistInMemDb looks in memdb to see if the specified name
is listed in either the NewNames category (incompatible names that
are going to be changed), or the InUseNames category (compatible
names that cannot be used more than once).
This routine compares only names in the same name group.
Arguments:
Context - Specifies the group context
Name - Specifies the name to query
Return Value:
TRUE if the name is in use, or FALSE if the name is not in use.
--*/
{
TCHAR Node[MEMDB_MAX];
DEBUGMSG ((DBG_NAMEFIX, "%s: [%s] is compatible", Context->GroupName, Name));
MemDbBuildKey (
Node,
MEMDB_CATEGORY_NEWNAMES,
Context->GroupName,
MEMDB_FIELD_NEW,
Name
);
if (MemDbGetValue (Node, NULL)) {
return TRUE;
}
MemDbBuildKey (
Node,
MEMDB_CATEGORY_INUSENAMES,
Context->GroupName,
NULL,
Name
);
return MemDbGetValue (Node, NULL);
}
VOID
pMemDbSetIncompatibleName (
IN PCTSTR NameGroup,
IN PCTSTR OrgName,
IN PCTSTR NewName
)
/*++
Routine Description:
pMemDbSetIncompatibleName adds the correct entries to memdb to
have a name appear in the name collision wizard page.
Arguments:
NameGroup - Specifies the name group such as ComputerName, UserName, etc...
OrgName - Specifies the original name that is invalid
NewName - Specifies the recommended new name
Return Value:
none
--*/
{
DWORD NewNameOffset;
DEBUGMSG ((DBG_NAMEFIX, "%s: [%s]->[%s]", NameGroup, OrgName, NewName));
MemDbSetValueEx (
MEMDB_CATEGORY_NEWNAMES,
NameGroup,
NULL,
NULL,
0,
NULL
);
MemDbSetValueEx (
MEMDB_CATEGORY_NEWNAMES,
NameGroup,
MEMDB_FIELD_NEW,
NewName,
0,
&NewNameOffset
);
MemDbSetValueEx (
MEMDB_CATEGORY_NEWNAMES,
NameGroup,
MEMDB_FIELD_OLD,
OrgName,
NewNameOffset,
NULL
);
}
VOID
pMemDbSetCompatibleName (
IN PCTSTR NameGroup,
IN PCTSTR Name
)
/*++
Routine Description:
pMemDbSetCompatibleName creates the memdb entries necessary
to store names that are in use and are compatible.
Arguments:
NameGroup - Specifies the name group such as ComputerName, UserName, etc...
Name - Specifies the compatible name
Return Value:
none
--*/
{
MemDbSetValueEx (
MEMDB_CATEGORY_INUSENAMES,
NameGroup,
NULL,
Name,
0,
NULL
);
}
VOID
CreateNameTables (
VOID
)
/*++
Routine Description:
CreateNameTables finds all names on the computer and puts valid names
into the InUseNames memdb category, and invalid names into the NewNames
memdb category (including both the invalid name and recommended name).
A wizard page appears if invalid names are found on the system.
Arguments:
none
Return Value:
none
--*/
{
INT i;
NAME_ENUM e;
PNAME_GROUP_ROUTINES Group;
TCHAR RecommendedName[MAX_NAME];
PTSTR p;
PTSTR DupList;
static BOOL AlreadyDone = FALSE;
if (AlreadyDone) {
return;
}
AlreadyDone = TRUE;
TurnOnWaitCursor();
//
// Special case: Add NT group names to InUse list
//
p = (PTSTR) GetStringResource (
*g_ProductFlavor == PERSONAL_PRODUCTTYPE ?
MSG_NAME_COLLISION_LIST_PER :
MSG_NAME_COLLISION_LIST
);
MYASSERT (p);
if (p) {
DupList = DuplicateText (p);
MYASSERT (DupList);
FreeStringResource (p);
p = _tcschr (DupList, TEXT('|'));
while (p) {
*p = 0;
p = _tcschr (_tcsinc (p), TEXT('|'));
}
Group = pGetNameGroupById (MSG_USERNAME_CATEGORY);
MYASSERT (Group);
if (Group) {
p = DupList;
while (*p) {
pMemDbSetCompatibleName (
Group->GroupName,
p
);
p = GetEndOfString (p) + 1;
}
}
FreeText (DupList);
}
//
// General case: Enumerate all names, call Validate and add them to memdb
//
for (i = 0 ; g_NameGroupRoutines[i].GroupName ; i++) {
Group = &g_NameGroupRoutines[i];
//
// Initialize the context structure
//
ZeroMemory (&Group->Context, sizeof (NAME_GROUP_CONTEXT));
Group->Context.GroupName = Group->GroupName;
//
// Call the enum entry point
//
ZeroMemory (&e, sizeof (e));
if (Group->Enum (&Group->Context, &e, TRUE)) {
do {
//
// Determine if this name is valid. If it is valid, add it to the
// InUseNames memdb category. If it is not valid, get a recommended
// replacement name, and store the incompatible and recommended
// names in the NewNames memdb category.
//
#ifdef TEST_MANGLE_NAMES
StringCat (e.Name, TEXT("\"%foo"));
#endif
#ifdef TEST_ALL_INCOMPATIBLE
if (0) {
#else
if (Group->Validate (&Group->Context, e.Name)) {
#endif
pMemDbSetCompatibleName (
Group->GroupName,
e.Name
);
} else {
Group->Recommend (&Group->Context, e.Name, RecommendedName);
pMemDbSetIncompatibleName (
Group->GroupName,
e.Name,
RecommendedName
);
}
} while (Group->Enum (&Group->Context, &e, FALSE));
}
}
TurnOffWaitCursor();
}
BOOL
IsIncompatibleNamesTableEmpty (
VOID
)
/*++
Routine Description:
IsIncompatibleNamesTableEmpty looks at memdb to see if there are any
names in the NewNames category. This function is used to determine
if the name collision wizard page should appear.
Arguments:
none
Return Value:
TRUE if at least one name is invalid, or FALSE if all names are valid.
--*/
{
INVALID_NAME_ENUM e;
return !EnumFirstInvalidName (&e);
}
BOOL
pEnumInvalidNameWorker (
IN OUT PINVALID_NAME_ENUM EnumPtr
)
/*++
Routine Description:
pEnumInvalidNameWorker implements a state machine for invalid
name enumeration. It returns the group name, original name
and new name to the caller.
Arguments:
EnumPtr - Specifies the enumeration in progress; receives the
updated fields
Return Value:
TRUE if an item was enumerated, or FALSE if no more items exist.
--*/
{
PCTSTR p;
INT i;
while (EnumPtr->State != ENUM_STATE_DONE) {
switch (EnumPtr->State) {
case ENUM_STATE_INIT:
if (!MemDbEnumItems (&EnumPtr->NameGroup, MEMDB_CATEGORY_NEWNAMES)) {
EnumPtr->State = ENUM_STATE_DONE;
} else {
EnumPtr->State = ENUM_STATE_ENUM_FIRST_GROUP_ITEM;
}
break;
case ENUM_STATE_ENUM_FIRST_GROUP_ITEM:
if (!MemDbGetValueEx (
&EnumPtr->Name,
MEMDB_CATEGORY_NEWNAMES,
EnumPtr->NameGroup.szName,
MEMDB_FIELD_OLD
)) {
EnumPtr->State = ENUM_STATE_ENUM_NEXT_GROUP;
} else {
EnumPtr->State = ENUM_STATE_RETURN_GROUP_ITEM;
}
break;
case ENUM_STATE_RETURN_GROUP_ITEM:
//
// Get the group name
//
EnumPtr->GroupName = EnumPtr->NameGroup.szName;
//
// Get the display group name from the message resources
//
for (i = 0 ; g_NameGroupRoutines[i].GroupName ; i++) {
if (StringMatch (g_NameGroupRoutines[i].GroupName, EnumPtr->GroupName)) {
break;
}
}
MYASSERT (g_NameGroupRoutines[i].GroupName);
if (g_NameGroupRoutines[i].NameId == MSG_COMPUTERDOMAIN_CATEGORY) {
EnumPtr->State = ENUM_STATE_ENUM_NEXT_GROUP_ITEM;
break;
}
p = GetStringResource (g_NameGroupRoutines[i].NameId);
MYASSERT (p);
if (p) {
_tcssafecpy (EnumPtr->DisplayGroupName, p, (sizeof (EnumPtr->DisplayGroupName) / 2) / sizeof (TCHAR));
FreeStringResource (p);
}
ELSE_DEBUGMSG ((DBG_ERROR, "Unable to get string resource. Check localization."));
//
// Get EnumPtr->NewName and EnumPtr->Identifier
//
EnumPtr->OriginalName = EnumPtr->Name.szName;
MemDbBuildKeyFromOffset (
EnumPtr->Name.dwValue,
EnumPtr->NewName,
3,
NULL
);
MemDbGetOffsetEx (
MEMDB_CATEGORY_NEWNAMES,
EnumPtr->GroupName,
MEMDB_FIELD_OLD,
EnumPtr->OriginalName,
&EnumPtr->Identifier
);
EnumPtr->State = ENUM_STATE_ENUM_NEXT_GROUP_ITEM;
return TRUE;
case ENUM_STATE_ENUM_NEXT_GROUP_ITEM:
if (!MemDbEnumNextValue (&EnumPtr->Name)) {
EnumPtr->State = ENUM_STATE_ENUM_NEXT_GROUP;
} else {
EnumPtr->State = ENUM_STATE_RETURN_GROUP_ITEM;
}
break;
case ENUM_STATE_ENUM_NEXT_GROUP:
if (!MemDbEnumNextValue (&EnumPtr->NameGroup)) {
EnumPtr->State = ENUM_STATE_DONE;
} else {
EnumPtr->State = ENUM_STATE_ENUM_FIRST_GROUP_ITEM;
}
break;
}
}
return FALSE;
}
BOOL
EnumFirstInvalidName (
OUT PINVALID_NAME_ENUM EnumPtr
)
/*++
Routine Description:
EnumFirstInvalidName enumerates the first entry in the memdb NewNames
category. The caller receives the name group, the old name and
the new name.
Arguments:
none
Return Value:
TRUE if at least one name is invalid, or FALSE if all names are valid.
--*/
{
EnumPtr->State = ENUM_STATE_INIT;
return pEnumInvalidNameWorker (EnumPtr);
}
BOOL
EnumNextInvalidName (
IN OUT PINVALID_NAME_ENUM EnumPtr
)
/*++
Routine Description:
EnumNextInvalidName enumerates the first entry in the memdb NewNames
category. The caller receives the name group, the old name and
the new name.
Arguments:
none
Return Value:
TRUE if another invalid name is available, or FALSE if no more names
can be enumerated.
--*/
{
return pEnumInvalidNameWorker (EnumPtr);
}
VOID
GetNamesFromIdentifier (
IN DWORD Identifier,
IN PTSTR NameGroup, OPTIONAL
IN PTSTR OriginalName, OPTIONAL
IN PTSTR NewName OPTIONAL
)
/*++
Routine Description:
GetNamesFromIdentifier copies names to caller-specified buffers, given a
unique identifier (a memdb offset). The unique identifer is provided
by enumeration functions.
Arguments:
Identifier - Specifies the identifier of the name.
NameGroup - Receives the text for the name group.
OriginalName - Receivies the original name.
NewName - Receives the fixed name that is compatible with NT.
Return Value:
none
--*/
{
BOOL b;
PTSTR p;
TCHAR NameGroupTemp[MEMDB_MAX];
TCHAR OrgNameTemp[MEMDB_MAX];
DWORD NewNameOffset;
if (NameGroup) {
*NameGroup = 0;
}
if (OriginalName) {
*OriginalName = 0;
}
if (NewName) {
*NewName = 0;
}
//
// Get NameGroup
//
if (!MemDbBuildKeyFromOffset (Identifier, NameGroupTemp, 1, NULL)) {
return;
}
p = _tcschr (NameGroupTemp, TEXT('\\'));
MYASSERT (p);
*p = 0;
if (NameGroup) {
StringCopy (NameGroup, NameGroupTemp);
}
//
// Get OrgName and NewNameOffset.
//
b = MemDbBuildKeyFromOffset (Identifier, OrgNameTemp, 3, &NewNameOffset);
if (OriginalName) {
StringCopy (OriginalName, OrgNameTemp);
}
//
// Get NewName
//
if (NewName) {
b &= MemDbBuildKeyFromOffset (NewNameOffset, NewName, 3, NULL);
}
MYASSERT (b);
}
VOID
ChangeName (
IN DWORD Identifier,
IN PCTSTR NewName
)
/*++
Routine Description:
ChangeName puts a new name value in memdb for the name indicated by
Identifier. The Identifier comes from enum functions.
Arguments:
Identifier - Specifies the name identifier (a memdb offset), and cannot be
zero.
NewName - Specifies the NT-compatible replacement name.
Return Value:
none
--*/
{
TCHAR Node[MEMDB_MAX];
TCHAR NameGroup[MEMDB_MAX];
TCHAR OrgName[MEMDB_MAX];
DWORD NewNameOffset;
PTSTR p, q;
BOOL b;
MYASSERT (Identifier);
if (!Identifier) {
return;
}
//
// - Obtain the original name
// - Get the offset to the current new name
// - Build the full key to the current new name
// - Delete the current new name
//
if (!MemDbBuildKeyFromOffset (Identifier, OrgName, 3, &NewNameOffset)) {
DEBUGMSG ((DBG_WHOOPS, "Can't obtain original name using offset %u", Identifier));
LOG ((LOG_ERROR, "Can't obtain original name using offset %u", Identifier));
return;
}
if (!MemDbBuildKeyFromOffset (NewNameOffset, Node, 0, NULL)) {
DEBUGMSG ((DBG_WHOOPS, "Can't obtain new name key using offset %u", NewNameOffset));
LOG ((LOG_ERROR, "Can't obtain new name key using offset %u", NewNameOffset));
return;
}
MemDbDeleteValue (Node);
//
// Obtain the name group from the key string. It's the second
// field (separated by backslashes).
//
p = _tcschr (Node, TEXT('\\'));
MYASSERT (p);
p = _tcsinc (p);
q = _tcschr (p, TEXT('\\'));
MYASSERT (q);
StringCopyAB (NameGroup, p, q);
//
// Now set the updated new name, and link the original name
// to the new name.
//
b = MemDbSetValueEx (
MEMDB_CATEGORY_NEWNAMES,
NameGroup,
MEMDB_FIELD_NEW,
NewName,
0,
&NewNameOffset
);
b &= MemDbSetValueEx (
MEMDB_CATEGORY_NEWNAMES,
NameGroup,
MEMDB_FIELD_OLD,
OrgName,
NewNameOffset,
NULL
);
if (!b) {
LOG ((LOG_ERROR, "Failure while attempting to change %s name to %s.",OrgName,NewName));
}
}
BOOL
GetUpgradeComputerName (
OUT PTSTR NewName
)
/*++
Routine Description:
GetUpgradeComputerName obtains the computer name that will be used for
upgrade.
Arguments:
NewName - Receives the name of the computer, as it will be set
when NT is installed. Must hold at least
MAX_COMPUTER_NAME characters.
Return Value:
TRUE if the name exists, or FALSE if it does not yet exit.
--*/
{
PNAME_GROUP_ROUTINES Group;
NAME_ENUM e;
Group = pGetNameGroupById (MSG_COMPUTERNAME_CATEGORY);
MYASSERT (Group);
if (!Group)
return FALSE;
//
// Look in MemDb for a replacement name
//
if (MemDbGetEndpointValueEx (
MEMDB_CATEGORY_NEWNAMES,
Group->GroupName,
MEMDB_FIELD_NEW,
NewName
)) {
return TRUE;
}
//
// No replacement name; obtain the current name
//
ZeroMemory (&e, sizeof (e));
if (Group->Enum (&Group->Context, &e, TRUE)) {
StringCopy (NewName, e.Name);
while (Group->Enum (&Group->Context, &e, FALSE)) {
// empty
}
return TRUE;
}
return FALSE;
}
DWORD
GetDomainIdentifier (
VOID
)
/*++
Routine Description:
GetDomainIdentifier returns the identifier for the domain name. The
identifier is a memdb offset.
Arguments:
None.
Return Value:
A non-zero identifier which can be used with other routines in this file.
--*/
{
PNAME_GROUP_ROUTINES Group;
MEMDB_ENUM e;
DWORD Identifier = 0;
Group = pGetNameGroupById (MSG_COMPUTERDOMAIN_CATEGORY);
MYASSERT (Group);
if (Group && MemDbGetValueEx (
&e,
MEMDB_CATEGORY_NEWNAMES,
Group->GroupName,
MEMDB_FIELD_OLD
)) {
MemDbGetOffsetEx (
MEMDB_CATEGORY_NEWNAMES,
Group->GroupName,
MEMDB_FIELD_OLD,
e.szName,
&Identifier
);
}
return Identifier;
}
BOOL
pGetUpgradeName (
IN UINT CategoryId,
OUT PTSTR NewName
)
/*++
Routine Description:
pGetUpgradeName returns the NT-compatible name for a given name group. If
a name group has multiple names, this routine should not be used.
Arguments:
CategoryId - Specifies the MSG_* constant for the group (see macro
expansion list at top of file).
NewName - Receives the text for the NT-compatible replacement name for
the group.
Return Value:
TRUE if a name is being returned, or FALSE if no replacement name exists.
--*/
{
PNAME_GROUP_ROUTINES Group;
NAME_ENUM e;
Group = pGetNameGroupById (CategoryId);
MYASSERT (Group);
if (!Group)
return FALSE;
//
// Look in MemDb for a replacement name
//
if (MemDbGetEndpointValueEx (
MEMDB_CATEGORY_NEWNAMES,
Group->GroupName,
MEMDB_FIELD_NEW,
NewName
)) {
return TRUE;
}
//
// No replacement name; obtain the current name
//
ZeroMemory (&e, sizeof (e));
if (Group->Enum (&Group->Context, &e, TRUE)) {
StringCopy (NewName, e.Name);
while (Group->Enum (&Group->Context, &e, FALSE)) {
// empty
}
return TRUE;
}
return FALSE;
}
BOOL
GetUpgradeDomainName (
OUT PTSTR NewName
)
/*++
Routine Description:
GetUpgradeDomainName returns the new domain name, if one exists.
Arguments:
NewName - Receiveis the new domain name.
Return Value:
TRUE if a new name is being returned, or FALSE if no replacement name
exists.
--*/
{
return pGetUpgradeName (
MSG_COMPUTERDOMAIN_CATEGORY,
NewName
);
}
BOOL
GetUpgradeWorkgroupName (
OUT PTSTR NewName
)
/*++
Routine Description:
GetUpgradeWorkgroupName returns the new workgroup name, if one exists.
Arguments:
None.
Return Value:
TRUE if a new name is being returned, or FALSE if no replacement name
exists.
--*/
{
return pGetUpgradeName (
MSG_WORKGROUP_CATEGORY,
NewName
);
}
BOOL
GetUpgradeUserName (
IN PCTSTR User,
OUT PTSTR NewUserName
)
/*++
Routine Description:
GetUpgradeUserName returns the fixed user name for the user specified. If
no fixed name exists, this routine returns the original name.
Arguments:
User - Specifies the user to look up. The user name must exist on
the Win9x configuration.
NewUserName - Receives the NT-compatible user name, which may or may not be
the same as User.
Return Value:
Always TRUE.
--*/
{
PNAME_GROUP_ROUTINES Group;
TCHAR Node[MEMDB_MAX];
DWORD NewOffset;
Group = pGetNameGroupById (MSG_USERNAME_CATEGORY);
MYASSERT (Group);
//
// Look in MemDb for a replacement name
//
if (Group) {
MemDbBuildKey (
Node,
MEMDB_CATEGORY_NEWNAMES,
Group->GroupName,
MEMDB_FIELD_OLD,
User
);
if (MemDbGetValue (Node, &NewOffset)) {
if (MemDbBuildKeyFromOffset (NewOffset, NewUserName, 3, NULL)) {
return TRUE;
}
}
}
//
// No replacement name; use the current name
//
StringCopy (NewUserName, User);
return TRUE;
}
BOOL
WarnAboutBadNames (
IN HWND PopupParent
)
/*++
Routine Description:
WarnAboutBadNames adds an incompatibility message whenever domain
logon is enabled but there is no account set up for the machine. A popup
is generated if PopupParent is non-NULL.
Other incompatibility messages are added for each name that will change.
Arguments:
Popup - Specifies TRUE if the message should appear in a message box, or
FALSE if it should be added to the incompatibility report.
Return Value:
TRUE if the user wants to continue, or FALSE if the user wants to change
the domain name.
--*/
{
PCTSTR RootGroup;
PCTSTR NameSubGroup;
PCTSTR FullGroupName;
PCTSTR BaseGroup;
PNAME_GROUP_CONTEXT Context;
PCTSTR ArgArray[3];
INVALID_NAME_ENUM e;
BOOL b = TRUE;
TCHAR EncodedName[MEMDB_MAX];
PCTSTR Blank;
if (PopupParent) {
//
// PopupParent is non-NULL only when the incompatible names wizard page
// is being deactivated. Here we have a chance to make sure the names
// specified all work together before proceeding.
//
// This functionality has been disabled because domain validation has
// been moved to a new page. We might re-enable this when another
// invalid name group comes along.
//
return TRUE;
}
//
// Enumerate all bad names
//
if (EnumFirstInvalidName (&e)) {
Blank = GetStringResource (MSG_BLANK_NAME);
MYASSERT (Blank);
do {
//
// Prepare message
//
ArgArray[0] = e.DisplayGroupName;
ArgArray[1] = e.OriginalName;
ArgArray[2] = e.NewName;
if (ArgArray[1][0] == 0) {
ArgArray[1] = Blank;
}
if (ArgArray[2][0] == 0) {
ArgArray[2] = Blank;
}
RootGroup = GetStringResource (MSG_INSTALL_NOTES_ROOT);
MYASSERT (RootGroup);
NameSubGroup = ParseMessageID (MSG_NAMECHANGE_WARNING_GROUP, ArgArray);
MYASSERT (NameSubGroup);
BaseGroup = JoinPaths (RootGroup, NameSubGroup);
MYASSERT (BaseGroup);
FreeStringResource (RootGroup);
FreeStringResource (NameSubGroup);
NameSubGroup = ParseMessageID (MSG_NAMECHANGE_WARNING_SUBCOMPONENT, ArgArray);
MYASSERT (NameSubGroup);
FullGroupName = JoinPaths (BaseGroup, NameSubGroup);
MYASSERT (FullGroupName);
FreePathString (BaseGroup);
FreeStringResource (NameSubGroup);
EncodedName[0] = TEXT('|');
StringCopy (EncodedName + 1, e.OriginalName);
MsgMgr_ObjectMsg_Add(
EncodedName, // Object name, prefixed with a pipe symbol
FullGroupName, // Message title
S_EMPTY // Message text
);
FreePathString (FullGroupName);
} while (EnumNextInvalidName (&e));
FreeStringResource (Blank);
//
// Save changed user names to FixedUserNames
//
Context = pGetNameGroupContextById (MSG_USERNAME_CATEGORY);
MYASSERT (Context);
if (EnumFirstInvalidName (&e)) {
do {
if (StringMatch (Context->GroupName, e.GroupName)) {
_tcssafecpy (EncodedName, e.OriginalName, MAX_USER_NAME);
MemDbMakeNonPrintableKey (
EncodedName,
MEMDB_CONVERT_DOUBLEWACKS_TO_ASCII_1|
MEMDB_CONVERT_WILD_STAR_TO_ASCII_2|
MEMDB_CONVERT_WILD_QMARK_TO_ASCII_3
);
MemDbSetValueEx (
MEMDB_CATEGORY_FIXEDUSERNAMES,
EncodedName,
NULL,
e.NewName,
0,
NULL
);
}
} while (EnumNextInvalidName (&e));
}
}
return b;
}
DWORD
BadNamesWarning (
DWORD Request
)
{
switch (Request) {
case REQUEST_QUERYTICKS:
return TICKS_BAD_NAMES_WARNING;
case REQUEST_RUN:
if (!WarnAboutBadNames (NULL)) {
return GetLastError ();
}
else {
return ERROR_SUCCESS;
}
default:
DEBUGMSG ((DBG_ERROR, "Bad parameter in BadNamesWarning"));
}
return 0;
}
//
// The following flag is no longer in use. It used to be used
// to disable domain checking (to bypass the block requiring
// a valid domain). Currently there is no way to get beyond
// the wizard page that resolved the domain name, except by
// providing a valid domain or credentials to create a computer
// account on a domain.
//
BOOL g_DisableDomainChecks = FALSE;
VOID
DisableDomainChecks (
VOID
)
{
g_DisableDomainChecks = TRUE;
}
VOID
EnableDomainChecks (
VOID
)
{
g_DisableDomainChecks = FALSE;
}
BOOL
IsOriginalDomainNameValid (
VOID
)
/*++
Routine Description:
IsOriginalDomainNameValid checks to see if there is a replacement domain
name. If there is, the current domain name must be invalid.
Arguments:
None.
Return Value:
TRUE - the Win9x domain name is valid, no replacement name exists
FALSE - the Win9x domain name is invalid
--*/
{
PNAME_GROUP_ROUTINES Group;
TCHAR NewName[MEMDB_MAX];
Group = pGetNameGroupById (MSG_COMPUTERDOMAIN_CATEGORY);
MYASSERT (Group);
//
// Look in MemDb for a replacement name
//
if (Group && MemDbGetEndpointValueEx (
MEMDB_CATEGORY_NEWNAMES,
Group->GroupName,
MEMDB_FIELD_NEW,
NewName
)) {
return FALSE;
}
return TRUE;
}