212 lines
5.8 KiB
C
212 lines
5.8 KiB
C
/*++
|
|
|
|
Copyright (c) 1995 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
copyfile.c
|
|
|
|
Abstract:
|
|
|
|
InternalCopyFile - Copies a file retaining time and attributes
|
|
|
|
Author:
|
|
|
|
Matthew A Felton
|
|
|
|
Revision History:
|
|
|
|
Matthew Felton (mattfe) 27 March 1995
|
|
|
|
--*/
|
|
|
|
|
|
#include <precomp.h>
|
|
|
|
#define FILE_SHARE_EXCLUSIVE 0
|
|
#define BUFFER_SIZE 4096
|
|
|
|
BOOL
|
|
InternalCopyFile(
|
|
HANDLE hSourceFile,
|
|
PWIN32_FIND_DATA pSourceFileData,
|
|
LPWSTR lpNewFileName,
|
|
BOOL bFailIfExists
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
hSourceFile - SourceFile Handle
|
|
|
|
pSourceFileData - Pointer to WIN32_FIND_DATA for the source file
|
|
|
|
lpNewFileName - Supplies the name where a copy of the existing
|
|
files data and attributes are to be stored.
|
|
|
|
bFailIfExists - Supplies a flag that indicates how this operation is
|
|
to proceed if the specified new file already exists. A value of
|
|
TRUE specifies that this call is to fail. A value of FALSE
|
|
causes the call to the function to succeed whether or not the
|
|
specified new file exists.
|
|
|
|
Return Value:
|
|
|
|
TRUE - The operation was successful.
|
|
|
|
FALSE/NULL - The operation failed. Extended error status is available
|
|
using GetLastError.
|
|
|
|
--*/
|
|
{
|
|
DWORD dwSourceFileAttributes;
|
|
BOOL bReturnValue = FALSE;
|
|
HANDLE hTargetFile = INVALID_HANDLE_VALUE;
|
|
DWORD dwLowFileSize, dwHighFileSize;
|
|
LPVOID pBuffer;
|
|
DWORD cbBufferSize = BUFFER_SIZE;
|
|
DWORD cbBytesRead;
|
|
DWORD cbBytesWritten;
|
|
DWORD dwSourceFilePointer;
|
|
|
|
DWORD dwTargetFileAttributes;
|
|
|
|
SPLASSERT( hSourceFile != NULL &&
|
|
hSourceFile != INVALID_HANDLE_VALUE &&
|
|
pSourceFileData != NULL &&
|
|
lpNewFileName != NULL );
|
|
|
|
|
|
#if DBG
|
|
// <<<<< DEBUG ONLY >>>>>>
|
|
//
|
|
// ASSERTION Check Source File Pointer is Zero.
|
|
//
|
|
dwSourceFilePointer = SetFilePointer( hSourceFile, 0, NULL, FILE_CURRENT );
|
|
if ( dwSourceFilePointer != 0xffffffff ) {
|
|
SPLASSERT( dwSourceFilePointer == 0 );
|
|
}
|
|
#endif // DBG
|
|
|
|
|
|
//
|
|
// Alloc I/O Buffer
|
|
//
|
|
|
|
|
|
pBuffer = AllocSplMem( BUFFER_SIZE );
|
|
if ( pBuffer == NULL )
|
|
goto InternalCopyFileExit;
|
|
|
|
|
|
//
|
|
// Create TagetFile with same File Attributes except for READ ONLY attribute
|
|
// which must be cleared.
|
|
//
|
|
dwTargetFileAttributes = pSourceFileData->dwFileAttributes & ~FILE_ATTRIBUTE_READONLY;
|
|
if (pSourceFileData->dwFileAttributes & FILE_ATTRIBUTE_READONLY)
|
|
{
|
|
//
|
|
// Clear the READ ONLY attribute if the corresponding source file attribute
|
|
// is set. In fact at that point we don't know if the target file exists and
|
|
// that is the reason to ignore the value returned by SetFileAttributes.
|
|
// The file shouldn't exist on Whistler because of the dwTargetFileAttributes
|
|
// value where the READ ONLY flag is cleared. There are slim chances to be there
|
|
// if the machine was upgraded from W2K ( where the bug still exists) if on that
|
|
// machine the driver was ever installed.
|
|
//
|
|
SetFileAttributes( lpNewFileName, dwTargetFileAttributes );
|
|
}
|
|
|
|
hTargetFile = CreateFile( lpNewFileName,
|
|
GENERIC_WRITE,
|
|
FILE_SHARE_EXCLUSIVE,
|
|
NULL,
|
|
bFailIfExists ? CREATE_NEW : CREATE_ALWAYS,
|
|
dwTargetFileAttributes | FILE_FLAG_SEQUENTIAL_SCAN,
|
|
NULL );
|
|
|
|
if ( hTargetFile != INVALID_HANDLE_VALUE ) {
|
|
|
|
//
|
|
// Copy The Data
|
|
//
|
|
|
|
while (( bReturnValue = ReadFile( hSourceFile, pBuffer, cbBufferSize, &cbBytesRead, NULL )) &&
|
|
cbBytesRead != 0 ) {
|
|
|
|
//
|
|
// Add Code to Build CheckSum Here
|
|
//
|
|
|
|
bReturnValue = WriteFile( hTargetFile, pBuffer, cbBytesRead, &cbBytesWritten, NULL );
|
|
|
|
if ( bReturnValue == FALSE ||
|
|
cbBytesWritten != cbBytesRead ) {
|
|
|
|
bReturnValue = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
if ( bReturnValue ) {
|
|
|
|
//
|
|
// Set TargetFile Times to be the same as the Source File
|
|
//
|
|
|
|
bReturnValue = SetFileTime( hTargetFile,
|
|
&pSourceFileData->ftCreationTime,
|
|
&pSourceFileData->ftLastAccessTime,
|
|
&pSourceFileData->ftLastWriteTime );
|
|
|
|
//
|
|
// Verify that the file size is correct.
|
|
//
|
|
|
|
if ( bReturnValue ) {
|
|
|
|
dwLowFileSize = GetFileSize( hTargetFile, &dwHighFileSize );
|
|
|
|
if ( dwLowFileSize != pSourceFileData->nFileSizeLow ||
|
|
dwHighFileSize != pSourceFileData->nFileSizeHigh ) {
|
|
|
|
DBGMSG(DBG_ERROR,
|
|
("InternalCopyFile: sizes do not match for %ws: (%d %d) and (%d %d)",
|
|
lpNewFileName,
|
|
pSourceFileData->nFileSizeHigh,
|
|
pSourceFileData->nFileSizeLow,
|
|
dwHighFileSize,
|
|
dwLowFileSize));
|
|
bReturnValue = FALSE;
|
|
SetLastError(ERROR_FILE_INVALID);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Add Code here to Verify the CheckSum is correct.
|
|
//
|
|
|
|
}
|
|
|
|
CloseHandle( hTargetFile );
|
|
}
|
|
|
|
FreeSplMem( pBuffer );
|
|
|
|
InternalCopyFileExit:
|
|
|
|
if ( !bReturnValue ) {
|
|
DBGMSG( DBG_WARN, ("InternalCopyFile hSourceFile %x %ws error %d\n", hSourceFile, lpNewFileName, GetLastError() ));
|
|
SPLASSERT( GetLastError() != ERROR_SUCCESS );
|
|
}
|
|
|
|
return bReturnValue;
|
|
}
|