/*++ Copyright (c) 1997-1998 Microsoft Corporation Module Name: client.c Abstract: This module contains the code to process OS Chooser message for the BINL server. Author: Adam Barr (adamba) 9-Jul-1997 Geoff Pease (gpease) 10-Nov-1997 Environment: User Mode - Win32 Revision History: --*/ #include "binl.h" #pragma hdrstop #include "mbstring.h" // // List of maximums for certain variables. OscAddVariableX will fail if // the limits are exceeded; it is up to the caller of the function to know // if the variable being added might hit a limit and check for failure. // typedef struct OSC_VARIABLE_MAXIMUM { LPSTR VariableName; ULONG MaximumLength; } OSC_VARIABLE_MAXIMUM, *POSC_VARIABLE_MAXIMUM; OSC_VARIABLE_MAXIMUM OscMaximums[] = { // // This set of variables come from locations we don't completely control, // so we need to check the return code from OscAddVariable each time. // { "BOOTFILE", 127 }, // with NULL, must fit in 128-byte field of CREATE_DATA. // Normally this will be empty or come from a .sif; // an admin may customize the .sif or modify the // DS attribute directly. { "MACHINENAME", 63 }, // used in path with SERVERNAME; comes from a screen // input with a max length of 63, or else is generated // by the GenerateMachineName() function. 63 is equal // to DNS_MAX_LABEL_LENGTH. { "SIFFILE", 127 }, // with NULL, must fit in 128-byte field of CREATE_DATA. // Normally this will be \RemoteInstall\tmp\[GUID].sif, // but the path may be longer. { "INSTALLPATH", 127 }, // used in paths with MACHINETYPE and SERVERNAME. This // will depend on where the build is installed // with RISETUP. // // The ones after this will be correct when we add them, but a rogue // client might send in bogus values. So the general checking code in // OscProcessScreenArguments will catch invalid ones. // { "MACHINETYPE", MAX_ARCHITECTURE_LENGTH }, // current max value. This is sent up by oschooser // and should correspond to where RISETUP puts // the platform-specific files. { "SERVERNAME", 63 }, // used in paths with MACHINENAME and INSTALLPATH, // set by calling GetComputerNameEX(ComputerNameNetBIOS) { "NETBIOSNAME", 31 }, // with NULL, must fit in 32-byte field of CREATE_DATA. // This is gotten by calling DnsHostnameToComputerNameW(), // if that fails the name is truncated to 15 chars { "LANGUAGE", 32 }, // reasonable max value; this is obtained by calling // GetLocaleInfo(LOCALE_SYSTEM_DEFAULT, LOCALE_SENGLANGUAGE), // but can be over-ridden in the registry. It is used in // paths with IntelliMirrorPathW and some other constants, // but no other variables. Eventually this becomes // a part of INSTALLPATH and sometimes BOOTFILE. { "GUID", 32 }, // 16 bytes in hex format { "MAC", 12 }, // 6 bytes in hex format // // NOTE: If we get an error condition, we add the variable SUBERROR // to the client state. So don't put a limit on SUBERROR size, since // that might cause an infinite loop. // }; #define OSC_VARIABLE_MAXIMUM_COUNT (sizeof(OscMaximums) / sizeof(OSC_VARIABLE_MAXIMUM)) // // We need to eliminate the chance of denial of service attacks so we'll limit // the number of concurrent clients we support. // #define BINL_MAX_CLIENT_RECORDS 1000 LONG BinlGlobalClientLimit = BINL_MAX_CLIENT_RECORDS; DWORD OscUpdatePassword( IN PCLIENT_STATE ClientState, IN PWCHAR SamAccountName, IN PWCHAR Password, IN LDAP * LdapHandle, IN PLDAPMessage LdapMessage ) /*++ Routine Description: Sets the password for the client. NOTE: WE MUST BE BETWEEN CALLS TO OSCIMPERSONATE/OSCREVERT. Arguments: ClientState - The client state. AuthenticatedDCLdapHandle must be valid and we must be impersonating the client. SamAccountName - The name of the machine account. This is the "samAccountName" value from the DS, which includes the final $. Password - The NULL-terminated Unicode password. LdapHandle - The handle to the DS. LdapMessage - The result of an ldap search for this client. Return Value: Status of the operation. --*/ { BOOL bResult; LDAP * serverLdap; PWCHAR serverHostName; USER_INFO_1003 userInfo1003; PWCHAR backslashServerName; PWCHAR p; ULONG serverHostNameLength; DWORD paramError; NET_API_STATUS netStatus; // // Change the password in the DS. // serverLdap = ldap_conn_from_msg (LdapHandle, LdapMessage); if (serverLdap == NULL) { BinlPrintDbg(( DEBUG_ERRORS, "OscUpdatePassword ldap_conn_from_msg is NULL\n" )); return E_HANDLE; } serverHostName = NULL; if (LDAP_SUCCESS != ldap_get_option(serverLdap, LDAP_OPT_HOST_NAME, &serverHostName)) { BinlPrintDbg(( DEBUG_ERRORS, "OscUpdatePassword ldap_get_option failed\n" )); return E_HANDLE; } userInfo1003.usri1003_password = Password; serverHostNameLength = wcslen(serverHostName) + 1; // // Allocate room for the name with two extra characters // for the leading \\. // backslashServerName = BinlAllocateMemory((serverHostNameLength+2) * sizeof(WCHAR)); if (backslashServerName == NULL) { BinlPrintDbg(( DEBUG_ERRORS, "OscUpdatePassword could not allocate serverHostNameW\n" )); return ERROR_NOT_ENOUGH_SERVER_MEMORY; } wcscpy(backslashServerName, L"\\\\"); wcscpy(backslashServerName+2, serverHostName); // // TEMP: Serialize all calls to the NetUserSetInfo/ // NetUserModalsGet. See discussion in bug 319962. // This code was put back in when the fix for a // problem described as "RPC is ignoring the security // context of the caller when choosing which named // pipe to send an RPC call over" turned out to cause // a BVT break. // EnterCriticalSection(&HackWorkaroundCriticalSection); netStatus = NetUserSetInfo( backslashServerName, SamAccountName, 1003, (LPBYTE)&userInfo1003, ¶mError); LeaveCriticalSection(&HackWorkaroundCriticalSection); BinlFreeMemory(backslashServerName); if (netStatus != NERR_Success) { HANDLE TempToken; BinlPrint(( DEBUG_ERRORS, "OscUpdatePassword NetUserSetInfo returned %lx\n", netStatus )); // // If NetUserSetInfo failed, try a LogonUser to see if the // password is already set to the value we want -- if so, // we can still succeed. // bResult = LogonUser( SamAccountName, OscFindVariableW( ClientState, "MACHINEDOMAIN" ), Password, LOGON32_LOGON_NETWORK, LOGON32_PROVIDER_WINNT40, &TempToken); if (bResult) { CloseHandle(TempToken); } else { DWORD TempError = GetLastError(); if (TempError != ERROR_NOLOGON_WORKSTATION_TRUST_ACCOUNT) { return netStatus; // return the original error } } // // Fall through and return ERROR_SUCCESS. // } return ERROR_SUCCESS; } // // Free client state information // VOID FreeClient( PCLIENT_STATE client ) { ULONG i; TraceFunc("FreeClient( )\n"); // // if this client had its name generated, // then attempt to remove the name from the queued DS list // this behavior is necessary for when a client times out // if (client->fAutomaticMachineName) { PWCHAR pMachineName; // Pointer to Machine Name variable value DWORD Error; pMachineName = OscFindVariableW( client, "MACHINENAME" ); Error = RemoveQueuedDSName(pMachineName); if (Error != ERROR_SUCCESS) { BinlPrintDbg(( DEBUG_ERRORS, "RemoveQueuedDSName returned with status: 0x%x\n", Error)); if (Error == ERROR_NOT_FOUND) { BinlPrintDbg(( DEBUG_ERRORS, "QueuedDSName already removed\n", Error)); Error = ERROR_SUCCESS; } } } BinlPrintDbg(( DEBUG_OSC, "Freeing client state for %s\n", inet_ntoa(*(struct in_addr *)&(client->RemoteIp)) )); DeleteCriticalSection(&client->CriticalSection); if (client->LastResponse) BinlFreeMemory(client->LastResponse); OscFreeClientVariables(client); InterlockedIncrement( &BinlGlobalClientLimit ); if (client->NegotiateProcessed) { DeleteSecurityContext( &client->ServerContextHandle ); } if (client->AuthenticatedDCLdapHandle) { ldap_unbind(client->AuthenticatedDCLdapHandle); } if (client->UserToken) { CloseHandle(client->UserToken); } BinlFreeMemory(client); } VOID OscFreeClientVariables( PCLIENT_STATE clientState ) /*++ Routine Description: This function frees all variables in a client state. Arguments: clientState - the client state pointer. Return Value: None. --*/ { ULONG i; for( i = 0; i < clientState->nVariables; i++ ) { BinlFreeMemory(clientState->Variables[i].pszToken); if (clientState->Variables[i].pszStringA) { BinlFreeMemory(clientState->Variables[i].pszStringA); clientState->Variables[i].pszStringA = NULL; } if (clientState->Variables[i].pszStringW) { BinlFreeMemory(clientState->Variables[i].pszStringW); clientState->Variables[i].pszStringW = NULL; } } clientState->nVariables = 0; clientState->fHaveSetupMachineDN = FALSE; clientState->fCreateNewAccount = FALSE; clientState->fAutomaticMachineName = FALSE; } BOOLEAN OscInitializeClientVariables( PCLIENT_STATE clientState ) /*++ Routine Description: This function cleans out any variables in the client state, then initializes some default values, which may later be overwritten by variables from client screens. This function is called when a client state is created, and when it is re-used from cache. Arguments: clientState - the client state pointer. Return Value: None. --*/ { BOOLEAN retVal = TRUE; SYSTEMTIME SystemTime; FILETIME FileTime; WCHAR pTime[64]; // // First clean out any variables in the client state. // OscFreeClientVariables(clientState); // // Now add the variables. // EnterCriticalSection( &gcsParameters ); if (BinlGlobalDefaultLanguage) { OscAddVariableW( clientState, "LANGUAGE", BinlGlobalDefaultLanguage ); } else { OscAddVariableW( clientState, "LANGUAGE", DEFAULT_LANGUAGE ); } if (BinlGlobalDefaultOrgname) { OscAddVariableW( clientState, "ORGNAME", BinlGlobalDefaultOrgname ); } else { OscAddVariableW( clientState, "ORGNAME", DEFAULT_ORGNAME ); } if (BinlGlobalDefaultTimezone) { OscAddVariableW( clientState, "TIMEZONE", BinlGlobalDefaultTimezone ); } else { OscAddVariableW( clientState, "TIMEZONE", DEFAULT_TIMEZONE ); } if (BinlGlobalOurDomainName == NULL || BinlGlobalOurServerName == NULL) { LeaveCriticalSection( &gcsParameters ); BinlPrintDbg((DEBUG_OSC_ERROR, "!! Error we don't have a FQDN for ourselves.\n" )); retVal = FALSE; goto Cleanup; } OscAddVariableW( clientState, "SERVERDOMAIN", BinlGlobalOurDomainName ); // Add the Server's name variable OscAddVariableW( clientState, "SERVERNAME", BinlGlobalOurServerName ); GetSystemTime(&SystemTime); if (SystemTimeToFileTime(&SystemTime,&FileTime)) { swprintf(pTime,L"%d;%d",FileTime.dwHighDateTime,FileTime.dwLowDateTime); OscAddVariableW( clientState, "ServerUTCFileTime", pTime ); } OscAddVariableW( clientState, "NTLMV2Enabled", BinlGlobalUseNTLMV2 ? L"1" : L"0" ); LeaveCriticalSection( &gcsParameters ); OscAddVariableA( clientState, "SUBERROR", " " ); clientState->fHaveSetupMachineDN = FALSE; clientState->fCreateNewAccount = FALSE; clientState->fAutomaticMachineName = FALSE; clientState->InitializeOnFirstRequest = FALSE; Cleanup: // // If this fails, clean up anything we set here. // if (!retVal) { OscFreeClientVariables(clientState); } return retVal; } DWORD OscFindClient( ULONG RemoteIp, BOOL Remove, PCLIENT_STATE * pClientState ) /*++ Routine Description: This function looks up a client in our client database, using their IP address. If Remove is TRUE, it removes the entry if found. Otherwise, if not found, it creates a new entry. Arguments: RemoteIp - the remote IP address. Remove - TRUE if the client should be removed if found. pClientState - Returns the CLIENT_STATE. Return Value: ERROR_SUCCESS if it finds the client and it is not in use. ERROR_NOT_ENOUGH_SERVER_MEMORY if a client state could not be allocated. ERROR_BUSY if the client state is already being used by another thread. --*/ { LONG oldCount; PLIST_ENTRY p; DWORD Error = ERROR_SUCCESS; PCLIENT_STATE TempClient = NULL; TraceFunc("OscFindClient( )\n"); EnterCriticalSection(&ClientsCriticalSection); for (p = ClientsQueue.Flink; p != &ClientsQueue; p = p->Flink) { TempClient = CONTAINING_RECORD(p, CLIENT_STATE, Linkage); if (TempClient->RemoteIp == RemoteIp) { // // Found it! // if (Remove) { RemoveEntryList(&TempClient->Linkage); TraceFunc("Client removed.\n"); } break; } } if (p == &ClientsQueue) { TempClient = NULL; } if (!TempClient && (!Remove)) { // // Not found, allocate a new one. // oldCount = InterlockedDecrement( &BinlGlobalClientLimit ); if (oldCount <= 0) { InterlockedIncrement( &BinlGlobalClientLimit ); BinlPrintDbg(( DEBUG_OSC_ERROR, "Way too many clients, 0x%x clients\n", BINL_MAX_CLIENT_RECORDS)); Error = ERROR_NOT_ENOUGH_SERVER_MEMORY; TempClient = NULL; } else { TraceFunc("Creating new client...\n"); TempClient = BinlAllocateMemory(sizeof(CLIENT_STATE)); if (TempClient == NULL) { InterlockedIncrement( &BinlGlobalClientLimit ); BinlPrintDbg(( DEBUG_OSC_ERROR, "Could not get client state for %s\n", inet_ntoa(*(struct in_addr *)&RemoteIp) )); Error = ERROR_NOT_ENOUGH_SERVER_MEMORY; } else { TempClient->NegotiateProcessed = FALSE; TempClient->AuthenticateProcessed = FALSE; TempClient->LastSequenceNumber = 0; TempClient->LastResponse = NULL; TempClient->LastResponseAllocated = 0; TempClient->PositiveRefCount = 1; TempClient->NegativeRefCount = 0; TempClient->AuthenticatedDCLdapHandle = NULL; TempClient->UserToken = NULL; TempClient->LastUpdate = GetTickCount(); TempClient->nCreateAccountCounter = 0; TempClient->nVariables = 0; // // Fill in some standard variables. // if (!OscInitializeClientVariables(TempClient)) { InterlockedIncrement( &BinlGlobalClientLimit ); BinlPrintDbg(( DEBUG_OSC_ERROR, "Could not initialize client state for %s\n", inet_ntoa(*(struct in_addr *)&RemoteIp) )); BinlFreeMemory(TempClient); TempClient = NULL; Error = ERROR_NOT_ENOUGH_SERVER_MEMORY; } else { InitializeCriticalSection(&TempClient->CriticalSection); TempClient->CriticalSectionHeld = FALSE; TempClient->RemoteIp = RemoteIp; OscGenerateSeed(&TempClient->Seed); InsertHeadList(&ClientsQueue, &TempClient->Linkage); BinlPrintDbg(( DEBUG_OSC, "Allocating new client state for %s\n", inet_ntoa(*(struct in_addr *)&RemoteIp) )); } } } } if (TempClient) { // // Do a quick check to see if another client is using this. This // check is not synchronized with the setting of this variable to // FALSE, and it's possible two clients could slip through, but // that is OK since this is not fatal (each thread still needs // to actually get the critical section to do anything). // if (TempClient->CriticalSectionHeld && (!Remove)) { Error = ERROR_BUSY; TempClient = NULL; } else { ++TempClient->PositiveRefCount; // need to do this inside ClientsCriticalSection } } LeaveCriticalSection(&ClientsCriticalSection); *pClientState = TempClient; return Error; } VOID OscFreeClients( VOID ) /*++ Routine Description: This function frees the clients list for OS chooser. It is intended to be called only when the service is shutting down, so the critical section does not matter. Arguments: None. Return Value: None. --*/ { PLIST_ENTRY p; PCLIENT_STATE TempClient; TraceFunc("OscFreeClients( )\n"); while (!IsListEmpty(&ClientsQueue)) { p = RemoveHeadList(&ClientsQueue); TempClient = CONTAINING_RECORD(p, CLIENT_STATE, Linkage); FreeClient(TempClient); } } VOID SearchAndReplace( LPSAR psarList, LPSTR *pszString, DWORD ArraySize, DWORD dwSize, DWORD dwExtraSize ) /*++ Routine Description: Searches and replaces text in a ASCII (8-bit) string. Arguments: psarList - SearchAndReplace structure with the list of tokens and and the strings that are going replace the tokens. pszString - the text to search and replace. dwSize - length of the text in pszString. dwExtraSize - if the buffer is reallocated, how much extra room to allocate Return Value: None. --*/ { LPSTR psz = *pszString; TraceFunc("SearchAndReplace( )\n"); if ( !psarList || !*pszString ) return; while ( *psz ) { if ( *psz == '%' ) { LPSAR psar = psarList; ULONG count = 0; LPSTR pszEnd; UCHAR ch; psz++; // move forward // // Find the end of the %MACRO% // pszEnd = psz; while ( *pszEnd && *pszEnd !='%' ) pszEnd++; // // Terminate but remember the character (NULL or '%') // ch = *pszEnd; *pszEnd = 0; // // Loop trying to match the %MACRO% when a Token // while( count++ < ArraySize ) { if ( _stricmp( psz, psar->pszToken ) == 0 ) { // match, so replace DWORD dwString; DWORD dwToken; if ( psar->pszStringA == NULL ) { // need to translate the string from UNICODE to ANSI DWORD dwLen; ANSI_STRING aString; UNICODE_STRING uString; uString.Buffer = psar->pszStringW; uString.Length = (USHORT)( wcslen( psar->pszStringW ) * sizeof(WCHAR) ); dwLen = RtlUnicodeStringToAnsiSize(&uString); // includes NULL termination psar->pszStringA = BinlAllocateMemory( dwLen ); if (psar->pszStringA == NULL) { BinlAssert( !"Out of memory!" ); psar++; continue; // abort this replace } aString.Buffer = psar->pszStringA; aString.MaximumLength = (USHORT)dwLen; RtlUnicodeStringToAnsiString( &aString, &uString, FALSE); } dwString = strlen( psar->pszStringA ); dwToken = strlen( psar->pszToken ); psz--; // move back if ( 2 + dwToken < dwString ) { // "%MACRO%" is smaller than "ReplaceString" // Check to see if we need to grow the buffer... LPSTR pszEndBuff = &psz[2 + dwToken]; DWORD dwLenEnd = strlen( pszEndBuff ) + 1; DWORD dwLenBegin = (DWORD)( psz - *pszString ); DWORD dwNewSize = dwLenBegin + dwString + dwLenEnd; // // Does the new string fit in the old space? // if ( dwSize < dwNewSize ) { // // No. Make some space // LPSTR pszNewString; dwNewSize += 1024; // with some extra to grow pszNewString = BinlAllocateMemory( dwNewSize + dwExtraSize ); if ( !pszNewString ) { BinlAssert( !"Out of memory!" ); return; // abort the rest } MoveMemory( pszNewString, *pszString, dwSize ); dwSize = dwNewSize; psz = pszNewString + ( psz - *pszString ); BinlFreeMemory( *pszString ); *pszString = pszNewString; } MoveMemory( &psz[dwString], &psz[ 2 + dwToken ], dwLenEnd ); } CopyMemory( psz, psar->pszStringA, dwString ); if ( 2 + dwToken > dwString ) { strcpy( &psz[ dwString ], &psz[ 2 + dwToken ] ); } pszEnd = NULL; // match, NULL so we don't put the temp char back psz++; // move forward break; } psar++; } // // If no match, put the character back // if ( pszEnd != NULL ) { *pszEnd = ch; } } else { psz++; } } } LPSTR FindSection( LPSTR sectionName, LPSTR sifFile ) /*++ Routine Description: This routine is for use by ProcessUniqueUdb. It scans in memory starting at sifFile, looking for a SIF section named "sectionName". Specifically it searches for a line whose first non-blank characters are "[sectionName]". Arguments: sectionName - The section name to look for, an ANSI string. sifFile - The SIF file in memory, which is NULL-terminated ANSI string. Return Value: A pointer to the start of the section -- the first character of the line after the one with [sectionName] in it. --*/ { LPSTR curSifFile; DWORD lenSectionName; LPSTR foundSection = NULL; lenSectionName = strlen(sectionName); curSifFile = sifFile; while (*curSifFile != '\0') { // // At this point in the while, curSifFile points to the beginning // of a line. // // // First find the first non-blank character. // while ((*curSifFile != '\0') && (*curSifFile == ' ')) { ++curSifFile; } if (*curSifFile == '\0') { break; } if (*curSifFile == '[') { if ((memcmp(sectionName, curSifFile+1, lenSectionName) == 0) && (curSifFile[lenSectionName+1] == ']')) { // // Found it, scan to start of next line. // while ((*curSifFile != '\0') && (*curSifFile != '\n')) { ++curSifFile; } if (*curSifFile == '\0') { break; } foundSection = curSifFile + 1; // +1 to skip past the '\n' break; } } // // Now scan to the start of the next line, defined as the // character after a \n. // while ((*curSifFile != L'\0') && (*curSifFile != L'\n')) { ++curSifFile; } if (*curSifFile == L'\0') { break; } ++curSifFile; // skip past the '\n' } return foundSection; } BOOLEAN FindLineInSection( PCHAR SectionStart, PWCHAR lineToMatch, PCHAR *existingLine, DWORD *existingLineLen ) /*++ Routine Description: This routine is for use by ProcessUniqueUdb. It scans in memory starting at SectionStart, which is assumed to be the first line of a section of a .sif file. It looks for a line that is for the same value as lineToMatch, which will be of the form "value=name". If it is found, it returns the line that it was found on. Arguments: SectionStart - The section of the .sif, in ANSI. lineToMatch - The value=name pair to match, in UNICODE. existingLine - Returns the existing line (in SectionStart), in ANSI. existingLineLen - Returns the length of the existing line, including final \r\n. Length is in characters, not bytes. Return Value: TRUE of the line is found, FALSE otherwise. --*/ { LPWSTR endOfValue; LPSTR curSection; LPSTR curLine; LPSTR endOfLine; DWORD valueLength, ansiValueLength; BOOLEAN foundLine = FALSE; LPSTR valueToMatch = NULL; ANSI_STRING aString; UNICODE_STRING uString; // // First look at lineToMatch to see what we are looking // for. This is the text up to the first =, or all of it if there // is no =. Once found, we convert it to ANSI for comparisons. // endOfValue = wcschr(lineToMatch, L'='); if (endOfValue == NULL) { endOfValue = lineToMatch + wcslen(lineToMatch); } valueLength = (DWORD)(endOfValue - lineToMatch); // // Copy the sectionName to ANSI for comparisons. // uString.Buffer = lineToMatch; uString.Length = (USHORT)(valueLength*sizeof(WCHAR)); ansiValueLength = RtlUnicodeStringToAnsiSize(&uString); // includes final '\0' valueToMatch = BinlAllocateMemory(ansiValueLength); if (valueToMatch == NULL) { return FALSE; } aString.Buffer = valueToMatch; aString.MaximumLength = (USHORT)ansiValueLength; RtlUnicodeStringToAnsiString( &aString, &uString, FALSE); --ansiValueLength; // remove final '\0' from the count // // now scan each line of SectionStart, until we find the beginning // of another section, a \0, or the matching line. // curSection = SectionStart; while (*curSection != '\0') { // // At this point in the while, curSection points to the beginning // of a line. Save the start of the current line. // curLine = curSection; // // First find the first non-blank character. // while ((*curSection != '\0') && (*curSection == ' ')) { ++curSection; } // // If we hit \0, we didn't find it. // if (*curSection == '\0') { break; } // // If we hit a line starting with [, we didn't find it. // if (*curSection == '[') { break; } // // If we hit a line starting with what we expect, followed // by an =, \0, or a blank, we found it. // if (strncmp(curSection, valueToMatch, ansiValueLength) == 0) { if ((curSection[ansiValueLength] == '=') || (curSection[ansiValueLength] == '\0') || (curSection[ansiValueLength] == ' ')) { *existingLine = curLine; endOfLine = strchr(curLine, '\n'); if (endOfLine == NULL) { *existingLineLen = strlen(curLine); } else { *existingLineLen = (DWORD)(endOfLine + 1 - curLine); } foundLine = TRUE; break; } } // // Now scan to the start of the next line, defined as the // character after a \n. // while ((*curSection != L'\0') && (*curSection != L'\n')) { ++curSection; } if (*curSection == L'\0') { break; } ++curSection; // skip past the '\n' } if (valueToMatch != NULL) { BinlFreeMemory(valueToMatch); } return foundLine; } VOID ProcessUniqueUdb( LPSTR *sifFilePtr, DWORD sifFileLen, LPWSTR UniqueUdbPath, LPWSTR UniqueUdbId ) /*++ Routine Description: Overlays data from a unique.udb file based on the tag specified. The file to overlay on is in memory at *pszString. *pszString is reallocated if necessary. Arguments: sifFile - The file to overlay data on. sifFileLen - The current size of the data at *pszString (including final NULL). UniqueUdbPath - The path to the unique.udb file. UniqueUdbId - The ID in unique.udb to use. Return Value: None. --*/ { PWCHAR TmpBuffer = NULL; DWORD len, sifFileAlloc, lineLen; LONG sizeToAdd; LPSTR sifFile = *sifFilePtr; PWCHAR sectionList = NULL; PWCHAR sectionLoc, sectionCur; PWCHAR sectionName = NULL; PCHAR ansiRealSectionName = NULL; PCHAR sectionStart; PWCHAR profileSectionCur; PWCHAR realSectionName; PCHAR existingLine; DWORD existingLineLen; DWORD lenRealSectionName; PCHAR insertionPoint; ANSI_STRING aString; UNICODE_STRING uString; #define TMP_BUFFER_SIZE 2048 TraceFunc("ProcessUniqueUdb( )\n"); TmpBuffer = BinlAllocateMemory(TMP_BUFFER_SIZE * sizeof(WCHAR)); if (TmpBuffer == NULL) { return; } // // See if the ID appears in the [UniqueIds] section of the unique.udb file. // TmpBuffer[0] = L'\0'; GetPrivateProfileString( L"UniqueIds", // section name UniqueUdbId, // line name L"", // default TmpBuffer, TMP_BUFFER_SIZE, UniqueUdbPath ); if (TmpBuffer[0] == L'\0') { return; } // // sifFileAlloc is the size allocated for sifFile, whereas // sifFileLen is the amount actually used. They should only // be different while we are actively shuffling things // around. // sifFileAlloc = sifFileLen; // // Save the tmpbuffer result. // len = wcslen(TmpBuffer) + 1; sectionList = BinlAllocateMemory(len * sizeof(WCHAR)); if (sectionList == NULL) { return; } wcscpy(sectionList, TmpBuffer); // // Now for each section identified in sectionList, scan for // the section to overlay. // sectionLoc = sectionList; while (TRUE) { // // First skip leading blanks // while (*sectionLoc && !iswalnum(*sectionLoc)) { // // Make sure we are not at a comment. // if (*sectionLoc == L';') { goto Cleanup; } ++sectionLoc; } if (!*sectionLoc) { goto Cleanup; } // // Now save sectionCur as the current section name // and skip to the end of it. Section names can be // any alphanumeric character, '.', or '_'. // sectionCur = sectionLoc; while((iswalnum(*sectionLoc)) || (*sectionLoc == '.') || (*sectionLoc == '_')) { ++sectionLoc; } // // Construct the new section name to look for. This will // be [UNIQUEUDBID:RealSectionName]. // len = wcslen(UniqueUdbId) + (sectionLoc - sectionCur) + 2; // one for :, one for NULL sectionName = BinlAllocateMemory(len * sizeof(WCHAR)); if (sectionName == NULL) { goto Cleanup; } wcscpy(sectionName, UniqueUdbId); wcscat(sectionName, L":"); len = wcslen(sectionName); realSectionName = sectionName + len; memcpy(realSectionName, sectionCur, (sectionLoc - sectionCur) * sizeof(WCHAR)); realSectionName[sectionLoc - sectionCur] = L'\0'; // // Copy the sectionName to ANSI for comparisons. // uString.Buffer = realSectionName; uString.Length = (USHORT)(wcslen(realSectionName)*sizeof(WCHAR)); lenRealSectionName = RtlUnicodeStringToAnsiSize(&uString); // includes final '\0' ansiRealSectionName = BinlAllocateMemory(lenRealSectionName); if (ansiRealSectionName == NULL) { goto Cleanup; } aString.Buffer = ansiRealSectionName; aString.MaximumLength = (USHORT)lenRealSectionName; RtlUnicodeStringToAnsiString( &aString, &uString, FALSE); --lenRealSectionName; // remove final '\0' from count // // See if there is a section with that name. // TmpBuffer[0] = L'\0'; GetPrivateProfileSection( sectionName, TmpBuffer, TMP_BUFFER_SIZE, UniqueUdbPath ); if (TmpBuffer[0] == L'\0') { continue; } // // Got the contents of the section, now process it. // sectionStart = FindSection(ansiRealSectionName, sifFile); sizeToAdd = 0; // amount we need to extend the buffer by. if (sectionStart == NULL) { // // No section, so need to add room for it. // // We put a CR-LF combo, then the section name in [], then // another CR-LF. // // lenRealSectionName already contains the '\0' // sizeToAdd = lenRealSectionName + 6; } // // Now scan through the entries in the profile section. // profileSectionCur = TmpBuffer; while (*profileSectionCur != L'\0') { uString.Buffer = profileSectionCur; uString.Length = (USHORT)(wcslen(profileSectionCur) * sizeof(WCHAR)); // // Figure out how long profileSectionCur will be as an // ANSI string (may have DBCS data in it). // lineLen = RtlUnicodeStringToAnsiSize(&uString); // includes \0 termination --lineLen; // remove the \0 from the count // // If there is no existing section we have to add it; // if not, see if there is a line for this already. // if (sectionStart == NULL) { sizeToAdd += lineLen + 2; // +2 is for CR-LF } else { if (FindLineInSection(sectionStart, profileSectionCur, &existingLine, &existingLineLen)) { // // Need to remove current line. // memmove(existingLine, existingLine + existingLineLen, sifFileLen - ((existingLine - sifFile) + existingLineLen)); sizeToAdd += lineLen + 2 - existingLineLen; sifFileLen -= existingLineLen; } else { sizeToAdd += lineLen + 2; } } profileSectionCur += wcslen(profileSectionCur) + 1; } // // Now we need to reallocate the buffer if needed. // if (sizeToAdd > 0) { // // No. Make some space // LPSTR pszNewString; pszNewString = BinlAllocateMemory( sifFileAlloc + sizeToAdd ); if ( !pszNewString ) { return; // abort the rest } MoveMemory( pszNewString, sifFile, sifFileLen); BinlFreeMemory( sifFile ); // // Adjust sectionStart to be within the new buffer. // if (sectionStart != NULL) { sectionStart = pszNewString + (sectionStart - sifFile); } sifFile = pszNewString; *sifFilePtr = pszNewString; sifFileAlloc += sizeToAdd; } // // Add the section header if necessary. // if (sectionStart == NULL) { strcpy(sifFile + sifFileLen - 1, "\r\n["); sifFileLen += 3; strcpy(sifFile + sifFileLen - 1, ansiRealSectionName); sifFileLen += lenRealSectionName; strcpy(sifFile + sifFileLen - 1, "]\r\n"); sifFileLen += 3; sectionStart = sifFile + sifFileLen - 1; } // // Now add the items from the profile section. We know that // they do not exist in the file and that we have reallocated // the file buffer to be large enough. // profileSectionCur = TmpBuffer; insertionPoint = sectionStart; while (*profileSectionCur != L'\0') { uString.Buffer = profileSectionCur; uString.Length = (USHORT)(wcslen(profileSectionCur)*sizeof(WCHAR)); lineLen = RtlUnicodeStringToAnsiSize(&uString); // includes final '\0' --lineLen; // remove final '\0' from count // // move anything we need to down and insert the new line. // memmove(insertionPoint + lineLen + 2, insertionPoint, sifFileLen - (insertionPoint - sifFile)); aString.Buffer = insertionPoint; aString.MaximumLength = (USHORT)(lineLen+1); RtlUnicodeStringToAnsiString( &aString, &uString, FALSE); memcpy(insertionPoint + lineLen, "\r\n", 2); sifFileLen += lineLen + 2; insertionPoint += lineLen + 2; profileSectionCur += wcslen(profileSectionCur) + 1; } } Cleanup: if (sectionList != NULL) { BinlFreeMemory(sectionList); } if (sectionName != NULL) { BinlFreeMemory(sectionName); } if (ansiRealSectionName != NULL) { BinlFreeMemory(ansiRealSectionName); } if (TmpBuffer != NULL) { BinlFreeMemory(TmpBuffer); } ASSERT (sifFileLen <= sifFileAlloc); } // // OscFindVariableA( ) // LPSTR OscFindVariableA( PCLIENT_STATE clientState, LPSTR variableName ) // variable name are always ASCII until OSChooser // can handle Unicode. { ULONG i; static CHAR szNullStringA[1] = { '\0' }; LPSTR overrideValue; // // First check to see if this a query we are supposed to override. // if (strcmp(variableName, "SIF") == 0) { overrideValue = OscFindVariableA(clientState, "FORCESIFFILE"); if ((overrideValue != NULL) && (strlen(overrideValue) != 0)) { return overrideValue; } } for( i = 0; i < clientState->nVariables; i++ ) { if ( strcmp( clientState->Variables[i].pszToken, variableName ) == 0 ) { if ( clientState->Variables[i].pszStringA == NULL ) { DWORD dwLen; ANSI_STRING aString; UNICODE_STRING uString; uString.Buffer = clientState->Variables[i].pszStringW; uString.Length = (USHORT)( wcslen( clientState->Variables[i].pszStringW ) * sizeof(WCHAR) ); dwLen = RtlUnicodeStringToAnsiSize( &uString ); // includes NULL termination clientState->Variables[i].pszStringA = BinlAllocateMemory( dwLen ); if ( !(clientState->Variables[i].pszStringA) ) break; // out of memory aString.Buffer = clientState->Variables[i].pszStringA; aString.MaximumLength = (USHORT)dwLen; RtlUnicodeStringToAnsiString( &aString, &uString, FALSE); } return clientState->Variables[i].pszStringA; } } return szNullStringA; } // // OscFindVariableW( ) // LPWSTR OscFindVariableW( PCLIENT_STATE clientState, LPSTR variableName ) // variable name are always ASCII until OSChooser // can handle Unicode. { ULONG i; static WCHAR szNullStringW[1] = { L'\0' }; LPWSTR overrideValue; // // First check to see if this a query we are supposed to override. // if (strcmp(variableName, "SIF") == 0) { overrideValue = OscFindVariableW(clientState, "FORCESIFFILE"); if ((overrideValue != NULL) && (wcslen(overrideValue) != 0)) { return overrideValue; } } for( i = 0; i < clientState->nVariables; i++ ) { if ( strcmp( clientState->Variables[i].pszToken, variableName ) == 0 ) { if ( clientState->Variables[i].pszStringW == NULL ) { DWORD dwLen = _mbslen( clientState->Variables[i].pszStringA ) + 1; ANSI_STRING aString; UNICODE_STRING uString; clientState->Variables[i].pszStringW = BinlAllocateMemory( dwLen * sizeof(WCHAR) ); if ( !(clientState->Variables[i].pszStringW) ) break; // out of memory uString.Buffer = clientState->Variables[i].pszStringW; uString.MaximumLength = (USHORT)(dwLen * sizeof(WCHAR)); aString.Buffer = clientState->Variables[i].pszStringA; aString.Length = (USHORT)strlen( clientState->Variables[i].pszStringA ); RtlAnsiStringToUnicodeString( &uString, &aString, FALSE); } return clientState->Variables[i].pszStringW; } } return szNullStringW; } // // OscCheckVariableLength( ) // BOOLEAN OscCheckVariableLength( PCLIENT_STATE clientState, LPSTR variableName, ULONG variableLength ) { ULONG i; for (i = 0; i < OSC_VARIABLE_MAXIMUM_COUNT; i++) { if (strcmp(OscMaximums[i].VariableName, variableName) == 0) { if (variableLength > OscMaximums[i].MaximumLength) { BinlPrintDbg((DEBUG_OSC_ERROR, "Variable %s was %d bytes, only %d allowed\n", variableName, variableLength, OscMaximums[i].MaximumLength)); OscAddVariableA( clientState, "SUBERROR", variableName ); return FALSE; } else { return TRUE; } } } // // If we don't find it in our list of maximums, it is OK. // return TRUE; } // // OscAddVariableA( ) // DWORD OscAddVariableA( PCLIENT_STATE clientState, LPSTR variableName, // variable name are always ASCII until OSChooser // can handle Unicode. LPSTR variableValue ) { ULONG i; if ( variableValue == NULL ) return E_POINTER; // no value to add... abort if (!OscCheckVariableLength(clientState, variableName, strlen(variableValue))) { return E_INVALIDARG; } for( i = 0; i < clientState->nVariables; i++ ) { if ( strcmp( clientState->Variables[i].pszToken, variableName ) == 0 ) { ULONG l = strlen( variableValue ); if ( clientState->Variables[i].pszStringW != NULL ) { BinlFreeMemory( clientState->Variables[i].pszStringW ); clientState->Variables[i].pszStringW = NULL; } if ( clientState->Variables[i].pszStringA != NULL ) { // found a previous one // Don't replace values with "" if ( variableValue[0] == '\0' ) { break; } else { // replace it with the new one if ( l <= strlen( clientState->Variables[i].pszStringA ) ) { strcpy( clientState->Variables[i].pszStringA, variableValue ); } else { // need more space, delete the old BinlFreeMemory( clientState->Variables[i].pszStringA ); clientState->Variables[i].pszStringA = NULL; } } } break; } } // // Limit the number of variables we can have. Everything else is ignored. // if ( i == MAX_VARIABLES ) { return E_OUTOFMEMORY; } // // Adding a new one // if ( clientState->nVariables == i ) { clientState->Variables[i].pszToken = BinlStrDupA( variableName ); if (clientState->Variables[i].pszToken == NULL) { return E_OUTOFMEMORY; } clientState->nVariables++; } // // If this is a new one or a new Value for an existing variable // that does not fit in the old values space, create a dup of the // value. // if ( clientState->Variables[i].pszStringA == NULL ) { BinlAssert( variableValue != NULL ); clientState->Variables[i].pszStringA = BinlStrDupA( variableValue ); if (clientState->Variables[i].pszStringA == NULL) { if ((i + 1) == clientState->nVariables) { clientState->nVariables--; BinlFreeMemory(clientState->Variables[i].pszToken); } return E_OUTOFMEMORY; } // // The "OPTIONS" variable can have a lot of stuff in it and will // blow up the BinlPrint(). Just avoid the whole mess by not // printing it // if ( strcmp( clientState->Variables[i].pszToken, "OPTIONS" ) != 0 ) { BinlPrintDbg((DEBUG_OSC, "Add Var:'%s' = '%s'\n", clientState->Variables[i].pszToken, clientState->Variables[i].pszStringA )); } } // // it will be converted to UNICODE when OscFindVariableW( ) is called // return ERROR_SUCCESS; } // // OscAddVariableW( ) // DWORD OscAddVariableW( PCLIENT_STATE clientState, LPSTR variableName, // variable name are always ASCII until OSChooser // can handle Unicode. LPWSTR variableValue ) { ULONG i; if ( variableValue == NULL ) return E_POINTER; // no value to add... abort if (!OscCheckVariableLength(clientState, variableName, wcslen(variableValue))) { return E_INVALIDARG; } for( i = 0; i < clientState->nVariables; i++ ) { if ( strcmp( clientState->Variables[i].pszToken, variableName ) == 0 ) { if ( clientState->Variables[i].pszStringA != NULL ) { BinlFreeMemory( clientState->Variables[i].pszStringA ); clientState->Variables[i].pszStringA = NULL; } if ( clientState->Variables[i].pszStringW != NULL ) { // found a previous one // Don't replace values with "" if ( variableValue[0] == L'\0' ) { break; } else { // replace it with the new one ULONG Length = wcslen( variableValue ); if ( Length < wcslen( clientState->Variables[i].pszStringW ) ) { wcscpy( clientState->Variables[i].pszStringW, variableValue ); } else { // need more space, delete the old BinlFreeMemory( clientState->Variables[i].pszStringW ); clientState->Variables[i].pszStringW = NULL; } } } break; } } // // Limit the number of variables we can have. Everything else is ignored. // if ( i == MAX_VARIABLES ) { return E_OUTOFMEMORY; // out of space } // // Adding a new one // if ( clientState->nVariables == i ) { clientState->Variables[i].pszToken = BinlStrDupA( variableName ); if (clientState->Variables[i].pszToken == NULL) { return E_OUTOFMEMORY; } clientState->nVariables++; } // // If this is a new one or a new Value for an existing variable // that does not fit in the old values space, create a dup of the // value. // if ( clientState->Variables[i].pszStringW == NULL ) { BinlAssert( variableValue != NULL ); clientState->Variables[i].pszStringW = BinlStrDupW( variableValue); if (clientState->Variables[i].pszStringW == NULL) { if ((i + 1) == clientState->nVariables) { clientState->nVariables--; BinlFreeMemory(clientState->Variables[i].pszToken); } return E_OUTOFMEMORY; } BinlPrintDbg((DEBUG_OSC, "Add Var:'%s' = '%ws'\n", clientState->Variables[i].pszToken, clientState->Variables[i].pszStringW )); } // // it will be converted to ASCII when OscFindVariableA( ) is called // return ERROR_SUCCESS; } // // OscResetVariable( ) // VOID OscResetVariable( PCLIENT_STATE clientState, LPSTR variableName ) { ULONG i; BOOLEAN found = FALSE; for( i = 0; i < clientState->nVariables; i++ ) { if ( strcmp( clientState->Variables[i].pszToken, variableName ) == 0 ) { if ( clientState->Variables[i].pszStringA != NULL ) { BinlFreeMemory( clientState->Variables[i].pszStringA ); clientState->Variables[i].pszStringA = NULL; } if ( clientState->Variables[i].pszStringW != NULL ) { // found a previous one BinlFreeMemory( clientState->Variables[i].pszStringW ); clientState->Variables[i].pszStringW = NULL; } BinlPrintDbg((DEBUG_OSC, "Deleted Var:'%s'\n", clientState->Variables[i].pszToken )); BinlFreeMemory( clientState->Variables[i].pszToken ); found = TRUE; break; } } if (found) { // // move all existing ones up. // while (i < clientState->nVariables) { clientState->Variables[i].pszToken = clientState->Variables[i+1].pszToken; clientState->Variables[i].pszStringA = clientState->Variables[i+1].pszStringA; clientState->Variables[i].pszStringW = clientState->Variables[i+1].pszStringW; i++; } clientState->nVariables--; } return; }