[0.4.9][SHELL32] Copy without FILE_ATTRIBUTE_READONLY from a CDROM CORE-18089 (#4386)

That problem in the shell was unhidden when we switched from our own to the MS-PL CDFS-driver
in 0.4.8-dev-164-g ec6b3ecbe4
because our own old CDFS driver erroneously never reported any CD file being read-only.

Fix picked from 0.4.15-dev-4348-g 513d0a6d2d

While backporting the fix, I decided to port back also some EOL-whitespace-stripping in that file.
This commit is contained in:
Joachim Henze 2022-09-22 16:19:22 +02:00
parent 2986b27f7d
commit 275c4442bb

View file

@ -549,7 +549,7 @@ static void _SetOperationTexts(FILE_OPERATION *op, LPCWSTR src, LPCWSTR dest) {
default:
return;
}
FormatMessageW(FORMAT_MESSAGE_FROM_STRING|FORMAT_MESSAGE_ARGUMENT_ARRAY,
op->szBuilderString, 0, 0, szFinalString, sizeof(szFinalString), (va_list*)args);
@ -572,11 +572,11 @@ DWORD CALLBACK SHCopyProgressRoutine(
FILE_OPERATION *op = (FILE_OPERATION *) lpData;
if (op->progress) {
/*
* This is called at the start of each file. To keop less state,
/*
* This is called at the start of each file. To keop less state,
* I'm adding the file to the completed size here, and the re-subtracting
* it when drawing the progress bar.
*/
*/
if (dwCallbackReason & CALLBACK_STREAM_SWITCH)
op->completedSize.QuadPart += TotalFileSize.QuadPart;
@ -584,7 +584,7 @@ DWORD CALLBACK SHCopyProgressRoutine(
TotalFileSize.QuadPart +
TotalBytesTransferred.QuadPart
, op->totalSize.QuadPart);
op->bCancelled = op->progress->HasUserCancelled();
}
@ -711,6 +711,21 @@ static DWORD SHNotifyMoveFileW(FILE_OPERATION *op, LPCWSTR src, LPCWSTR dest, BO
return GetLastError();
}
static BOOL SHIsCdRom(LPCWSTR path)
{
WCHAR tmp[] = { L"A:\\" };
if (!path || !path[0])
return FALSE;
if (path[1] != UNICODE_NULL && path[1] != ':')
return FALSE;
tmp[0] = path[0];
return GetDriveTypeW(tmp) == DRIVE_CDROM;
}
/************************************************************************
* SHNotifyCopyFile [internal]
*
@ -751,6 +766,14 @@ static DWORD SHNotifyCopyFileW(FILE_OPERATION *op, LPCWSTR src, LPCWSTR dest, BO
ret = CopyFileExW(src, dest, SHCopyProgressRoutine, op, &op->bCancelled, bFailIfExists);
if (ret)
{
// We are copying from a CD-ROM volume, which is readonly
if (SHIsCdRom(src))
{
attribs = GetFileAttributesW(dest);
attribs &= ~FILE_ATTRIBUTE_READONLY;
SetFileAttributesW(dest, attribs);
}
SHChangeNotify(SHCNE_CREATE, SHCNF_PATHW, dest, NULL);
return ERROR_SUCCESS;
}
@ -1433,7 +1456,7 @@ static HRESULT copy_files(FILE_OPERATION *op, BOOL multiDest, const FILE_LIST *f
return ERROR_CANCELLED;
}
}
if (op->progress != NULL)
op->bCancelled |= op->progress->HasUserCancelled();
/* Vista return code. XP would return e.g. ERROR_FILE_NOT_FOUND, ERROR_ALREADY_EXISTS */
@ -1673,7 +1696,7 @@ static DWORD move_files(FILE_OPERATION *op, BOOL multiDest, const FILE_LIST *flF
move_to_dir(op, entryToMove, fileDest);
else
SHNotifyMoveFileW(op, entryToMove->szFullPath, fileDest->szFullPath, IsAttribDir(entryToMove->attributes));
if (op->progress != NULL)
op->bCancelled |= op->progress->HasUserCancelled();
/* Should fire on progress dialog only */
@ -1767,9 +1790,9 @@ int WINAPI SHFileOperationW(LPSHFILEOPSTRUCTW lpFileOp)
op.bManyItems = (flFrom.dwNumFiles > 1);
if (lpFileOp->wFunc != FO_RENAME && !(lpFileOp->fFlags & FOF_SILENT)) {
ret = CoCreateInstance(CLSID_ProgressDialog,
NULL,
CLSCTX_INPROC_SERVER,
ret = CoCreateInstance(CLSID_ProgressDialog,
NULL,
CLSCTX_INPROC_SERVER,
IID_PPV_ARG(IProgressDialog, &op.progress));
if (FAILED(ret))
goto cleanup;
@ -2047,7 +2070,7 @@ EXTERN_C HRESULT WINAPI SHPathPrepareForWriteA(HWND hwnd, IUnknown *modless, LPC
}
/*
/*
* The two following background operations were modified from filedefext.cpp
* They use an inordinate amount of mutable state across the string functions,
* so are not easy to follow and care is required when modifying.
@ -2058,11 +2081,11 @@ _FileOpCountManager(FILE_OPERATION *op, const FILE_LIST *from)
{
DWORD ticks = GetTickCount();
FILE_ENTRY *entryToCount;
for (UINT i = 0; i < from->dwNumFiles; i++)
{
entryToCount = &from->feFiles[i];
WCHAR theFileName[MAX_PATH];
StringCchCopyW(theFileName, MAX_PATH, entryToCount->szFullPath);
_FileOpCount(op, theFileName, IsAttribDir(entryToCount->attributes), &ticks);