NT4/private/windows/spooler/localspl/copyfile.c
2020-09-30 17:12:29 +02:00

197 lines
4.9 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;
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
//
hTargetFile = CreateFile( lpNewFileName,
GENERIC_WRITE,
FILE_SHARE_EXCLUSIVE,
NULL,
bFailIfExists ? CREATE_NEW : CREATE_ALWAYS,
pSourceFileData->dwFileAttributes | 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;
}