|
|
/****************************************************************************\
NEWFILES.C / OPK Wizard (OPKWIZ.EXE)
Microsoft Confidential Copyright (c) Microsoft Corporation 1998 All rights reserved
3/99 - Jason Cohen (JCOHEN) Added this new source file for the OPK Wizard as part of the OOBE update. 09/2000 - Stephen Lodwick (STELO) Ported OPK Wizard to Whistler
\****************************************************************************/
//
// Include file(s):
//
#include "pch.h"
#include "newfiles.h"
#include "resource.h"
//
// Internal Defined Value(s):
//
#define DIR_CONFIG_OOBE _T("$OEM$")
#define INF_SECT_SOURCEDISK _T("SourcedisksFiles")
#define INF_SECT_DESTDIRS _T("DestinationDirs")
#define INF_SECT_OOBE _T("RegisterOOBE")
#define INF_LINE_COPYFILES _T("CopyFiles")
#define INF_PREFIX _T("X")
#define SOURCENUM_OPTIONS_CAB _T("782")
#define DESTLDID_OOBE _T("11")
#define STR_SEARCH _T("*")
#define STR_PADDING _T("\r\n\r\n")
#define MAX_BUFFER 16384 // 32768
#ifndef CSTR_EQUAL
#define CSTR_EQUAL 2
#endif // CSTR_EQUAL
//
// Internal Structure(s):
//
typedef struct _FILELIST { LPTSTR lpFileName; LPTSTR lpDirectory; struct _FILELIST * lpNext; } FILELIST, *PFILELIST, *LPFILELIST;
//
// Internal Function Prototype(s):
//
static void DelFiles(LPTSTR, LPTSTR, DWORD, LPTSTR, LPTSTR); static LPFILELIST AllocFileList(HWND, LPTSTR, LPTSTR); static BOOL CompareFiles(LPTSTR, LPTSTR);
//
// External Function(s):
//
//////////////////////////////////////////////////////////////////////////////
// AddFiles - lpSourceDir = location of files to copy to OOBE directory.
// Destination = location of the system directory where installed to
// LDID.
// lpConfigDir = location of oemaudit.inf, and config files.
// lpSourceDir -> OOBE -> lpDestDir
//
void AddFiles(HWND hwndParent, LPTSTR lpSourceDir, LPTSTR lpDestLdid, LPTSTR lpDestDir, LPTSTR lpDestName, LPTSTR lpConfigDir) { LPTSTR lpFilePart, lpFile, lpFileName, lpSectEnd, lpSearch, lpTarget, lpNext; TCHAR szBuffer[MAX_PATH + 32] = NULLSTR, szCurDir[MAX_PATH] = NULLSTR, szSourceDir[MAX_PATH] = NULLSTR, szCopyDir[MAX_PATH], szWinbom[MAX_PATH], szCopyFiles[MAX_PATH + 32]; LPFILELIST lpflHead = NULL, lpflCur, lpflBuf; BOOL bFound; DWORD dwNum; int iFilePartLen; HRESULT hrCat; HRESULT hrPrintf;
//
// First thing we do is setup the directories and strings
// that we need to do all the work.
//
// We need the path to the config directory. Copydir is where
// the files are going to be copied to from the SourceDir
// so it needs to be cleaned out before we do the CopyFile.
// CopyDir will be created if not exists.
//
lstrcpyn(szCopyDir, lpConfigDir, AS(szCopyDir)); AddPathN(szCopyDir, DIR_CONFIG_OOBE,AS(szCopyDir)); AddPathN(szCopyDir, _T("\\"),AS(szCopyDir)); lpFilePart = szCopyDir + lstrlen(szCopyDir); iFilePartLen= AS(szCopyDir)-lstrlen(szCopyDir);
// Need a full path to the oemaudit inf.
//
lstrcpyn(szWinbom, lpConfigDir,AS(szWinbom)); AddPathN(szWinbom, FILE_WINBOM_INI,AS(szWinbom));
// We need to construct the prefix to the copy
// files section name.
//
lstrcpyn(szCopyFiles, INF_PREFIX,AS(szCopyFiles)); hrCat=StringCchCat(szCopyFiles, AS(szCopyFiles), lpDestLdid ? lpDestLdid : DESTLDID_OOBE); if ( lpDestDir && *lpDestDir ) hrCat=StringCchCat(szCopyFiles, AS(szCopyFiles), lpDestDir); StrRem(szCopyFiles, CHR_BACKSLASH); lpSectEnd = szCopyFiles + lstrlen(szCopyFiles);
//
// Now that we have that info, we need to get rid of any files that
// may have already been put in the inf and the destination directory.
//
// Cleaned out of the inf and destination directory only if we are
// passed in NULL for the source.
//
if ( !(lpSourceDir && *lpSourceDir) ) DelFiles(szCopyDir, lpFilePart, iFilePartLen, szWinbom, szCopyFiles);
//
// Now we make a list of all the files we are going to add to the
// inf and destination directory.
//
// If the source isn't a valid dir, we must have just wanted to clean up.
//
if ( ( lpSourceDir && *lpSourceDir ) && ( GetFullPathName(lpSourceDir, sizeof(szSourceDir) / sizeof(TCHAR), szSourceDir, &lpFile) && szSourceDir[0] ) && ( (dwNum = GetFileAttributes(szSourceDir)) != 0xFFFFFFFF ) ) { // Check to see if we were passed a file or a directory.
//
if ( ( dwNum & FILE_ATTRIBUTE_DIRECTORY ) || ( lpFile <= szSourceDir ) ) { // We are search for all the files in the diretory.
//
lpFile = STR_SEARCH; } else { // We are only doing one file. We need to separate
// the file from the directory.
//
*(lpFile - 1) = NULLCHR; }
// Set the staring point for our file search.
//
GetCurrentDirectory(sizeof(szCurDir) / sizeof(TCHAR), szCurDir); SetCurrentDirectory(szSourceDir);
// Get the file list.
//
lpflHead = AllocFileList(hwndParent, szBuffer, lpFile);
// Make sure the destination dir exits.
//
*lpFilePart = NULLCHR; CreatePath(szCopyDir);
//
// Now that we have the file list, go through each one processing
// it separately and then free the memory allocated for it.
//
// Loop through all the files in our linked list.
//
for ( lpflCur = lpflHead; lpflCur; lpflCur = lpflBuf ) { //
// First copy the file into the flat directory.
//
// Setup the relative path from the currect directory
// to the file we want to copy.
//
if ( lpflCur->lpDirectory && *lpflCur->lpDirectory ) lstrcpyn(szBuffer, lpflCur->lpDirectory,AS(szBuffer)); else szBuffer[0] = NULLCHR; AddPathN(szBuffer, lpflCur->lpFileName,AS(szBuffer));
// Support for a different file name for the destination.
//
lpFileName = lpDestName ? lpDestName : lpflCur->lpFileName;
// Setup the destination file name.
//
lstrcpyn(lpFilePart, lpFileName, iFilePartLen);
// Copy the file to the Options\Cabs directory and display
// an error if the copy failed. Probably means that this
// is a duplicate file.
//
if ( !CopyFile(szBuffer, szCopyDir, TRUE) ) { // Save the CopyFile error and then check to see if the file
// is actaully different then the one tried to copy over.
//
dwNum = GetLastError(); if ( ( !CompareFiles(szBuffer, szCopyDir) ) && ( lpTarget = (LPTSTR) MALLOC(256 * sizeof(TCHAR)) ) ) { //
// I hate doing UI in backend type code. Because of time I don't have
// much choice, but in the future, this UI code should be replaced
// with a call back mechanism so the caller can do the UI.
//
// This is the first of only two places where UI is used in here.
//
// Allocate another buffer to hold the message with the file name.
//
if ( ( LoadString(NULL, dwNum == ERROR_FILE_EXISTS ? IDS_ERR_DUPFILE : IDS_ERR_COPY, lpTarget, 256 * sizeof(TCHAR)) ) && ( lpNext = (LPTSTR) MALLOC((lstrlen(lpFileName) + lstrlen(lpTarget) + 1) * sizeof(TCHAR)) ) ) { // Add the file name to the message, get the title for the message
// box and display the error.
//
hrPrintf=StringCchPrintf(lpNext, (lstrlen(lpFileName) + lstrlen(lpTarget) + 1), lpTarget, lpFileName); *lpTarget = NULLCHR; LoadString(NULL, IDS_APPNAME, lpTarget, 256 * sizeof(TCHAR)); MessageBox(hwndParent, lpNext, lpTarget, MB_OK | MB_ICONWARNING | MB_APPLMODAL); FREE(lpNext); } FREE(lpTarget); } }
//
// Now add the file to the [SourceDiskFiles] section.
//
// We just use WritePrivateProfileString() to write
// FILENAME=781 to the [SourceDiskFiles] section.
//
WritePrivateProfileString(INF_SECT_SOURCEDISK, lpFileName, SOURCENUM_OPTIONS_CAB, szWinbom);
//
// This code figures out what the copy files section will be
// called. This is based on the path where the files will
// be copied.
//
// Create the name of the copy files section the file will be in.
//
*lpSectEnd = NULLCHR; if ( lpflCur->lpDirectory && *lpflCur->lpDirectory ) lstrcpyn(lpSectEnd, lpflCur->lpDirectory, AS(szCopyFiles)-(int)(lpSectEnd - szCopyFiles) ); StrRem(lpSectEnd, CHR_BACKSLASH);
//
// Now add the file path to the [DestinationDirs] section.
//
// Create the LDID and dir combo to write to the dest dir section.
//
lstrcpyn(szBuffer, lpDestLdid ? lpDestLdid : DESTLDID_OOBE,AS(szBuffer)); if ( ( lpDestDir && *lpDestDir ) || ( lpflCur->lpDirectory && *lpflCur->lpDirectory ) ) { hrCat=StringCchCat(szBuffer,AS(szBuffer), _T(",\"")); if ( lpDestDir && *lpDestDir ) { hrCat=StringCchCat(szBuffer, AS(szBuffer),lpDestDir); if ( lpflCur->lpDirectory && *lpflCur->lpDirectory ) AddPathN(szBuffer, lpflCur->lpDirectory,AS(szBuffer)); } else hrCat=StringCchCat(szBuffer, AS(szBuffer), lpflCur->lpDirectory); hrCat=StringCchCat(szBuffer, AS(szBuffer), _T("\"")); }
// We just use WritePrivateProfileString() to write
// COPYFILES=11,"OOBE\\DIR" to the [DestinationDirs] section.
//
WritePrivateProfileString(INF_SECT_DESTDIRS, szCopyFiles, szBuffer, szWinbom);
//
// Now add the copy files section to the CopyFiles line.
//
// First get current CopyFiles line.
//
szBuffer[0] = NULLCHR; GetPrivateProfileString(INF_SECT_OOBE, INF_LINE_COPYFILES, NULLSTR, szBuffer, sizeof(szBuffer) / sizeof(TCHAR), szWinbom);
// Search each section listed in the CopyFiles line to see
// if we need to add this one. The sections are divided by
// commas.
//
// ISSUE-2002/02/28-stelo- May want to take qutoes into account, but I don't think so.
//
bFound = FALSE; for ( lpTarget = szBuffer; !bFound && lpTarget && *lpTarget; lpTarget = lpNext ) { // Get rid of proceeding spaces.
//
while ( *lpTarget == CHR_SPACE ) lpTarget = CharNext(lpTarget);
// NULL terminate at the ',' and setup
// the lpNext pointer.
//
if ( lpNext = StrChr(lpTarget, _T(',')) ) *lpNext = NULLCHR; // Make sure there are no trailing spaces.
//
if ( lpSearch = StrChr(lpTarget, CHR_SPACE) ) *lpSearch = NULLCHR;
// Check if this section is the same as the one
// we are going to add.
//
if ( CompareString(LOCALE_SYSTEM_DEFAULT, NORM_IGNORECASE, lpTarget, -1, szCopyFiles, -1) == CSTR_EQUAL ) bFound = TRUE;
// Need to restore the characters we may have stomped on.
//
if ( lpNext ) *lpNext++ = _T(','); if ( lpSearch ) *lpSearch = CHR_SPACE; }
// Now see if we need to add the line.
//
if ( !bFound ) { // Append our copy files section.
//
if ( szBuffer[0] ) hrCat=StringCchCat(szBuffer,AS(szBuffer), _T(", ")); hrCat=StringCchCat(szBuffer, AS(szBuffer), szCopyFiles);
// We just use WritePrivateProfileString() to write
// the CopyFiles line back to the [RegisterOOBE] section
// with our added copy files section on it.
//
WritePrivateProfileString(INF_SECT_OOBE, INF_LINE_COPYFILES, szBuffer, szWinbom); }
//
// Now write the file name to it's copy files section.
//
// First get the entire copy files section.
//
GetPrivateProfileSection(szCopyFiles, szBuffer, sizeof(szBuffer) / sizeof(TCHAR), szWinbom);
// Loop throught the strings to see if the file is already there.
//
bFound = FALSE; for ( lpTarget = szBuffer; !bFound && *lpTarget; lpTarget += (lstrlen(lpTarget) + 1) ) { // Get rid of proceeding spaces.
//
while ( *lpTarget == CHR_SPACE ) lpTarget = CharNext(lpTarget);
// Make sure there are no trailing spaces.
//
if ( lpSearch = StrChr(lpTarget, CHR_SPACE) ) *lpSearch = NULLCHR;
// Check if this section is the same as the one
// we are going to add.
//
if ( CompareString(LOCALE_SYSTEM_DEFAULT, NORM_IGNORECASE, lpTarget, -1, lpFileName, -1) == CSTR_EQUAL ) bFound = TRUE;
// Need to restore the character we may have stomped on.
//
if ( lpSearch ) *lpSearch = CHR_SPACE; }
// Now write back the section if we need to.
//
if ( !bFound ) { // Need to a pointer to the end of the sub strings.
//
for ( lpSearch = szBuffer; *lpSearch; lpSearch += (lstrlen(lpSearch) + 1) );
// Copy the string to the end and add an extra NULL.
//
lstrcpyn(lpSearch, lpFileName, ((MAX_PATH+32)-(int)(lpSearch-szBuffer)) ); lpSearch += (lstrlen(lpSearch) + 1); *lpSearch = NULLCHR;
// We need to call WritePrivateProfileSection() with NULL
// to remove the section. We shouldn't have to do this,
// but the Win32 docs are not correct.
//
WritePrivateProfileSection(szCopyFiles, NULL, szWinbom);
// We just use WritePrivateProfileSection() to write the
// copy files section back with our added file in it.
//
WritePrivateProfileSection(szCopyFiles, szBuffer, szWinbom); }
//
// Now free the structure and the data within it.
//
// Save the next pointer before we free the structure.
//
lpflBuf = lpflCur->lpNext;
// Free the file buffers and the structure.
//
FREE(lpflCur->lpFileName); FREE(lpflCur->lpDirectory); FREE(lpflCur); }
//
// All done, now just clean up.
//
// Put the current directory back to where it should be.
//
if ( szCurDir[0] ) SetCurrentDirectory(szCurDir); }
// Make sure the changes to the inf are flushed to disk
//
WritePrivateProfileString(NULL, NULL, NULL, szWinbom); }
//
// Internal Function(s):
//
static void DelFiles(LPTSTR lpszCopyDir, LPTSTR lpszFilePart, DWORD cbFilePart, LPTSTR lpszWinbom, LPTSTR lpszCopyFiles) { LPTSTR lpSearch, lpSection, lpFileName, lpTarget, lpNext; LPTSTR lpszSections = NULL, lpszFileNames = NULL, lpszBuffer = NULL; BOOL bFound;
//
// Allocate buffers...
//
lpszSections = MALLOC(MAX_BUFFER * sizeof(TCHAR)); lpszFileNames = MALLOC(MAX_BUFFER * sizeof(TCHAR)); lpszBuffer = MALLOC(MAX_BUFFER * sizeof(TCHAR));
if ( !lpszSections || !lpszFileNames || !lpszBuffer ) { // Free the buffers... Note: FREE macro checks for NULL
//
FREE( lpszSections ); FREE( lpszFileNames ); FREE( lpszBuffer );
return; }
// We need all the section names.
//
GetPrivateProfileSectionNames(lpszSections, MAX_BUFFER, lpszWinbom);
// Loop throught the section to see if there is any that match our search criteria.
//
for ( lpSection = lpszSections; lpSection && *lpSection; lpSection += (lstrlen(lpSection) + 1) ) { // Get rid of proceeding spaces.
//
while ( *lpSection == CHR_SPACE ) lpSection = CharNext(lpSection);
// Make sure there are no trailing spaces.
//
if ( lpSearch = StrChr(lpSection, CHR_SPACE) ) *lpSearch = NULLCHR;
// Check if this section is the same as the one
// we are going to add.
//
if ( CompareString(LOCALE_SYSTEM_DEFAULT, NORM_IGNORECASE, lpSection, lstrlen(lpszCopyFiles), lpszCopyFiles, lstrlen(lpszCopyFiles)) == CSTR_EQUAL ) { // We need all the files in the section.
//
GetPrivateProfileSection(lpSection, lpszFileNames, MAX_BUFFER, lpszWinbom);
// Loop throught the section to see if there is any that match our search criteria.
//
for ( lpFileName = lpszFileNames; *lpFileName; lpFileName += (lstrlen(lpFileName) + 1) ) { // Get rid of proceeding spaces.
//
while ( *lpFileName == CHR_SPACE ) lpFileName = CharNext(lpFileName);
// Make sure there are no trailing spaces.
//
if ( lpSearch = StrChr(lpFileName, CHR_SPACE) ) *lpSearch = NULLCHR;
// Delete the file from the destination directory.
//
lstrcpyn(lpszFilePart, lpFileName, cbFilePart); DeleteFile(lpszCopyDir);
// Remove the line from the source disk section.
//
WritePrivateProfileString(INF_SECT_SOURCEDISK, lpFileName, NULL, lpszWinbom); }
// Search each section listed in the CopyFiles and remove
// this one. The sections are divided by commas.
//
bFound = FALSE; GetPrivateProfileString(INF_SECT_OOBE, INF_LINE_COPYFILES, NULLSTR, lpszBuffer, MAX_BUFFER, lpszWinbom); for ( lpTarget = lpszBuffer; !bFound && lpTarget && *lpTarget; lpTarget = lpNext ) { // Get rid of proceeding spaces.
//
while ( *lpTarget == CHR_SPACE ) lpTarget = CharNext(lpTarget);
// NULL terminate at the ',' and setup
// the lpNext pointer.
//
if ( lpNext = StrChr(lpTarget, _T(',')) ) *lpNext = NULLCHR; // Make sure there are no trailing spaces.
//
if ( lpSearch = StrChr(lpTarget, CHR_SPACE) ) *lpSearch = NULLCHR;
// Check if this section is the same as the one
// we are going to remove.
//
if ( CompareString(LOCALE_SYSTEM_DEFAULT, NORM_IGNORECASE, lpTarget, -1, lpSection, -1) == CSTR_EQUAL ) bFound = TRUE;
// Need to restore the characters we may have stomped on.
//
if ( lpNext ) *lpNext++ = _T(','); if ( lpSearch ) *lpSearch = CHR_SPACE;
if ( bFound ) { // Go back to the ',' or the beginning of the buffer.
//
while ( ( lpTarget > lpszBuffer) && ( *lpTarget != _T(',') ) ) lpTarget = CharPrev(lpszBuffer, lpTarget);
// Now overwrite the string we took out.
//
if ( lpNext ) lstrcpyn(lpTarget, lpNext - 1, (MAX_BUFFER-(int)(lpTarget-lpszBuffer))); else *lpTarget = NULLCHR; } } if ( bFound ) { // We should eat any preceeding spaces and/or commas just
// for good measure.
//
for ( lpTarget = lpszBuffer; ( *lpTarget == CHR_SPACE ) || ( *lpTarget == _T(',') ); lpTarget = CharNext(lpTarget) );
// Now write the buffer back to the inf file.
//
WritePrivateProfileString(INF_SECT_OOBE, INF_LINE_COPYFILES, *lpTarget ? lpTarget : NULL, lpszWinbom); }
// Remove the line from the destination dirs section.
//
WritePrivateProfileString(INF_SECT_DESTDIRS, lpSection, NULL, lpszWinbom);
// Remove this section entirely.
//
WritePrivateProfileSection(lpSection, NULL, lpszWinbom); } }
// Free the buffers... Note: FREE macro checks for NULL
//
FREE( lpszSections ); FREE( lpszFileNames ); FREE( lpszBuffer ); }
static LPFILELIST AllocFileList(HWND hwndParent, LPTSTR lpDirectory, LPTSTR lpSearch) { WIN32_FIND_DATA FileFound; HANDLE hFile; LPTSTR lpEnd, lpFileName; LPFILELIST lpflHead = NULL; LPFILELIST* lplpflNext = &lpflHead; HRESULT hrPrintf;
// Process all the files and directories.
//
if ( (hFile = FindFirstFile(lpSearch, &FileFound)) != INVALID_HANDLE_VALUE ) { do { // Display an error if the short and long file names don't match.
// Means that it is a LFN, which INFs don't like.
//
if ( ( FileFound.cAlternateFileName[0] ) && ( CompareString(LOCALE_SYSTEM_DEFAULT, NORM_IGNORECASE, FileFound.cAlternateFileName, -1, FileFound.cFileName, -1) != CSTR_EQUAL ) && ( lpEnd = (LPTSTR) MALLOC(256 * sizeof(TCHAR)) ) ) { //
// I hate doing UI in backend type code. Because of time I don't have
// much choice, but in the future, this UI code should be replaced
// with a call back mechanism so the caller can do the UI.
//
// This is the second of only two places where UI is used in here.
//
// Allocate another buffer to hold the message with the file name.
//
if ( ( LoadString(NULL, IDS_ERR_LFN, lpEnd, 256) ) && ( lpFileName = (LPTSTR) MALLOC((lstrlen(FileFound.cFileName) + lstrlen(lpEnd) + 1) * sizeof(TCHAR)) ) ) { // Add the file name to the message, get the title for the message
// box and display the error.
//
hrPrintf=StringCchPrintf(lpFileName, (lstrlen(FileFound.cFileName) + lstrlen(lpEnd) + 1), lpEnd, FileFound.cFileName); *lpEnd = NULLCHR; LoadString(NULL, IDS_APPNAME, lpEnd, 256 * sizeof(TCHAR)); MessageBox(hwndParent, lpFileName, lpEnd, MB_OK | MB_ICONWARNING | MB_APPLMODAL); FREE(lpFileName); } FREE(lpEnd); }
// Get a pointer to the file name, the short one if possible.
//
if ( FileFound.cAlternateFileName[0] ) lpFileName = FileFound.cAlternateFileName; else lpFileName = FileFound.cFileName;
// First check to see if this is a files (not a directory).
//
if ( !( FileFound.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY ) ) { // Allocate the next item in the structure.
//
if ( *lplpflNext = (LPFILELIST) MALLOC(sizeof(FILELIST)) ) { // Allocate the buffer for the file name and path and
// make sure that none of the allocations fail.
//
if ( ( (*lplpflNext)->lpFileName = (LPTSTR) MALLOC((lstrlen(lpFileName) + 1) * sizeof(TCHAR)) ) && ( (*lplpflNext)->lpDirectory = (LPTSTR) MALLOC((lstrlen(lpDirectory) + 1) * sizeof(TCHAR)) ) ) { // Copy the file name and path into the buffers.
//
lstrcpyn((*lplpflNext)->lpFileName, lpFileName, (lstrlen(lpFileName) + 1)); lstrcpyn((*lplpflNext)->lpDirectory, lpDirectory, (lstrlen(lpDirectory) + 1));
// Null the next pointer so we know this is the last item.
//
(*lplpflNext)->lpNext = NULL;
// Set the next pointer to point to the address of
// the next member of this new structure.
//
lplpflNext = &((*lplpflNext)->lpNext); } else { // Don't worry, the FREE() macro checks for NULL
// before it frees the memory.
//
FREE((*lplpflNext)->lpFileName); FREE(*lplpflNext); } } } // Otherwise, make sure the directory is not "." or "..".
//
else if ( ( lstrcmp(lpFileName, _T(".")) ) && ( lstrcmp(lpFileName, _T("..")) ) ) { // Tack on this directory name to the current path saving
// the end pointer so that it is easy to get rid of this
// directory name when we return back.
//
lpEnd = lpDirectory + lstrlen(lpDirectory); AddPath(lpDirectory, lpFileName);
// Go into the next directory, get all the files, and
// the set the current directory back to the original
// directory.
//
SetCurrentDirectory(lpFileName); *lplpflNext = AllocFileList(hwndParent, lpDirectory, lpSearch); SetCurrentDirectory(_T(".."));
// Get rid of the directory name off our path buffer.
//
*lpEnd = NULLCHR;
// Need to setup our next pointer to the end of the list
// returned to us.
//
while ( *lplpflNext ) lplpflNext = &((*lplpflNext)->lpNext); }
} while ( FindNextFile(hFile, &FileFound) ); FindClose(hFile); }
return lpflHead; }
static BOOL CompareFiles(LPTSTR lpFile1, LPTSTR lpFile2) { BOOL bCompare, bRead1, bRead2; HANDLE hFile1, hFile2; BYTE baBuffer1[4096], baBuffer2[4096]; DWORD dwBytes1, dwBytes2, dwCount;
// Open the files.
//
hFile1 = CreateFile(lpFile1, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); hFile2 = CreateFile(lpFile2, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
// Make sure the files were opened.
//
if ( ( hFile1 != INVALID_HANDLE_VALUE ) && ( hFile2 != INVALID_HANDLE_VALUE ) ) { // Read all the data from the files.
//
do { // Read in the max buffer from each file.
//
bRead1 = ReadFile(hFile1, baBuffer1, sizeof(baBuffer1), &dwBytes1, NULL); bRead2 = ReadFile(hFile2, baBuffer2, sizeof(baBuffer2), &dwBytes2, NULL);
// Make sure the reads didn't fail.
//
if ( bRead1 && bRead2 ) { // Check to make sure the sizes are the same.
//
if ( bCompare = ( dwBytes1 == dwBytes2 ) ) { // Make sure the buffers are identical.
//
for ( dwCount = 0; bCompare && ( dwCount < dwBytes1 ); dwCount++ ) bCompare = ( baBuffer1[dwCount] == baBuffer2[dwCount] ); } } else // If both the reads failed, we will return true.
//
bCompare = ( !bRead1 && !bRead2 ); } while ( bCompare && bRead1 && bRead2 && dwBytes1 && dwBytes2 ); } else // If both the files does not exist, then we will
// return false.
//
bCompare = ( ( hFile1 != INVALID_HANDLE_VALUE ) && ( hFile2 != INVALID_HANDLE_VALUE ) );
// Close the files.
//
if ( hFile1 == INVALID_HANDLE_VALUE ) CloseHandle(hFile1); if ( hFile2 == INVALID_HANDLE_VALUE ) CloseHandle(hFile2);
return bCompare; }
|