mirror of
https://github.com/reactos/reactos.git
synced 2025-04-05 21:21:33 +00:00
[USETUP] Improvements for the File-queues code.
- Add support for delete and move/rename operations, which are needed
for implementing ReactOS upgrading support.
- Use cabinet contexts.
- Use standard LIST_ENTRY structures for implementing the lists.
- Move the path-building hack code in SetupCommitFileQueueW() that had
been introduced in r66604 (97bb83f
) out of the file-queues code.
- Make the function prototypes compatible with win32's setupapi functions.
- Fix the format of the data passed to the custom notification handler.
- Adjust the file-copy callback to correctly use its arguments (setupapi-compatible).
This commit is contained in:
parent
ce15c84309
commit
44c101c9dc
3 changed files with 780 additions and 323 deletions
File diff suppressed because it is too large
Load diff
|
@ -26,55 +26,59 @@
|
|||
|
||||
#pragma once
|
||||
|
||||
#define SPFILENOTIFY_STARTQUEUE 0x1
|
||||
#define SPFILENOTIFY_ENDQUEUE 0x2
|
||||
#define SPFILENOTIFY_STARTSUBQUEUE 0x3
|
||||
#define SPFILENOTIFY_ENDSUBQUEUE 0x4
|
||||
#define SPFILENOTIFY_STARTQUEUE 0x00000001
|
||||
#define SPFILENOTIFY_ENDQUEUE 0x00000002
|
||||
#define SPFILENOTIFY_STARTSUBQUEUE 0x00000003
|
||||
#define SPFILENOTIFY_ENDSUBQUEUE 0x00000004
|
||||
|
||||
#define SPFILENOTIFY_STARTCOPY 0xb
|
||||
#define SPFILENOTIFY_ENDCOPY 0xc
|
||||
#define SPFILENOTIFY_COPYERROR 0xd
|
||||
#define SPFILENOTIFY_STARTDELETE 0x00000005
|
||||
#define SPFILENOTIFY_ENDDELETE 0x00000006
|
||||
#define SPFILENOTIFY_DELETEERROR 0x00000007
|
||||
|
||||
#define FILEOP_COPY 0x0
|
||||
#define FILEOP_RENAME 0x1
|
||||
#define FILEOP_DELETE 0x2
|
||||
#define FILEOP_BACKUP 0x3
|
||||
#define SPFILENOTIFY_STARTRENAME 0x00000008
|
||||
#define SPFILENOTIFY_ENDRENAME 0x00000009
|
||||
#define SPFILENOTIFY_RENAMEERROR 0x0000000a
|
||||
|
||||
#define FILEOP_ABORT 0x0
|
||||
#define FILEOP_DOIT 0x1
|
||||
#define FILEOP_SKIP 0x2
|
||||
#define FILEOP_RETRY FILEOP_DOIT
|
||||
#define FILEOP_NEWPATH 0x4
|
||||
#define SPFILENOTIFY_STARTCOPY 0x0000000b
|
||||
#define SPFILENOTIFY_ENDCOPY 0x0000000c
|
||||
#define SPFILENOTIFY_COPYERROR 0x0000000d
|
||||
|
||||
#define SPFILENOTIFY_NEEDMEDIA 0x0000000e
|
||||
#define SPFILENOTIFY_QUEUESCAN 0x0000000f
|
||||
|
||||
#define FILEOP_COPY 0
|
||||
#define FILEOP_RENAME 1
|
||||
#define FILEOP_DELETE 2
|
||||
#define FILEOP_BACKUP 3
|
||||
|
||||
#define FILEOP_ABORT 0
|
||||
#define FILEOP_DOIT 1
|
||||
#define FILEOP_SKIP 2
|
||||
#define FILEOP_RETRY FILEOP_DOIT
|
||||
#define FILEOP_NEWPATH 4
|
||||
|
||||
|
||||
/* TYPES ********************************************************************/
|
||||
|
||||
typedef PVOID HSPFILEQ;
|
||||
|
||||
typedef UINT (CALLBACK* PSP_FILE_CALLBACK_W)(
|
||||
PVOID Context,
|
||||
UINT Notification,
|
||||
UINT_PTR Param1,
|
||||
UINT_PTR Param2);
|
||||
|
||||
typedef struct _COPYCONTEXT
|
||||
typedef struct _FILEPATHS_W
|
||||
{
|
||||
LPCWSTR DestinationRootPath; /* Not owned by this structure */
|
||||
LPCWSTR InstallPath; /* Not owned by this structure */
|
||||
ULONG TotalOperations;
|
||||
ULONG CompletedOperations;
|
||||
PPROGRESSBAR ProgressBar;
|
||||
PPROGRESSBAR MemoryBars[4];
|
||||
} COPYCONTEXT, *PCOPYCONTEXT;
|
||||
PCWSTR Target;
|
||||
PCWSTR Source;
|
||||
UINT Win32Error;
|
||||
ULONG Flags;
|
||||
} FILEPATHS_W, *PFILEPATHS_W;
|
||||
|
||||
typedef UINT (CALLBACK* PSP_FILE_CALLBACK_W)(
|
||||
IN PVOID Context,
|
||||
IN UINT Notification,
|
||||
IN UINT_PTR Param1,
|
||||
IN UINT_PTR Param2);
|
||||
|
||||
|
||||
/* FUNCTIONS ****************************************************************/
|
||||
|
||||
NTSTATUS
|
||||
SetupExtractFile(
|
||||
PWCHAR CabinetFileName,
|
||||
PWCHAR SourceFileName,
|
||||
PWCHAR DestinationFileName);
|
||||
|
||||
HSPFILEQ
|
||||
WINAPI
|
||||
SetupOpenFileQueue(VOID);
|
||||
|
@ -99,22 +103,40 @@ SetupQueueCopyWNew(
|
|||
IN DWORD CopyStyle);
|
||||
#endif
|
||||
|
||||
/* A simplified version of SetupQueueCopyW that wraps Cabinet support around */
|
||||
BOOL
|
||||
SetupQueueCopy(
|
||||
HSPFILEQ QueueHandle,
|
||||
PCWSTR SourceCabinet,
|
||||
PCWSTR SourceRootPath,
|
||||
PCWSTR SourcePath,
|
||||
PCWSTR SourceFilename,
|
||||
PCWSTR TargetDirectory,
|
||||
PCWSTR TargetFilename);
|
||||
WINAPI
|
||||
SetupQueueCopyWithCab( // SetupQueueCopyW
|
||||
IN HSPFILEQ QueueHandle,
|
||||
IN PCWSTR SourceCabinet OPTIONAL,
|
||||
IN PCWSTR SourceRootPath,
|
||||
IN PCWSTR SourcePath OPTIONAL,
|
||||
IN PCWSTR SourceFileName,
|
||||
IN PCWSTR TargetDirectory,
|
||||
IN PCWSTR TargetFileName OPTIONAL);
|
||||
|
||||
BOOL
|
||||
WINAPI
|
||||
SetupQueueDeleteW(
|
||||
IN HSPFILEQ QueueHandle,
|
||||
IN PCWSTR PathPart1,
|
||||
IN PCWSTR PathPart2 OPTIONAL);
|
||||
|
||||
BOOL
|
||||
WINAPI
|
||||
SetupQueueRenameW(
|
||||
IN HSPFILEQ QueueHandle,
|
||||
IN PCWSTR SourcePath,
|
||||
IN PCWSTR SourceFileName OPTIONAL,
|
||||
IN PCWSTR TargetPath OPTIONAL,
|
||||
IN PCWSTR TargetFileName);
|
||||
|
||||
BOOL
|
||||
WINAPI
|
||||
SetupCommitFileQueueW(
|
||||
HWND Owner,
|
||||
HSPFILEQ QueueHandle,
|
||||
PSP_FILE_CALLBACK_W MsgHandler,
|
||||
PVOID Context);
|
||||
IN HWND Owner,
|
||||
IN HSPFILEQ QueueHandle,
|
||||
IN PSP_FILE_CALLBACK_W MsgHandler,
|
||||
IN PVOID Context OPTIONAL);
|
||||
|
||||
/* EOF */
|
||||
|
|
|
@ -71,6 +71,8 @@ static PGENERIC_LIST NtOsInstallsList = NULL;
|
|||
|
||||
// HACK: Temporary compatibility code.
|
||||
#if 1
|
||||
#define SetupQueueCopy SetupQueueCopyWithCab
|
||||
|
||||
static CABINET_CONTEXT CabinetContext;
|
||||
#define CabinetInitialize() (CabinetInitialize(&CabinetContext))
|
||||
#define CabinetSetEventHandlers(a,b,c) (CabinetSetEventHandlers(&CabinetContext,(a),(b),(c)))
|
||||
|
@ -3494,6 +3496,7 @@ AddSectionToCopyQueueCab(HINF InfFile,
|
|||
PWCHAR FileKeyValue;
|
||||
PWCHAR DirKeyValue;
|
||||
PWCHAR TargetFileName;
|
||||
WCHAR FileDstPath[MAX_PATH];
|
||||
|
||||
/*
|
||||
* This code enumerates the list of files in reactos.dff / reactos.inf
|
||||
|
@ -3549,12 +3552,46 @@ AddSectionToCopyQueueCab(HINF InfFile,
|
|||
break;
|
||||
}
|
||||
|
||||
#if 1 // HACK moved! (r66604)
|
||||
{
|
||||
ULONG Length = wcslen(DirKeyValue);
|
||||
if ((Length > 0) && (DirKeyValue[Length - 1] == L'\\'))
|
||||
Length--;
|
||||
DirKeyValue[Length] = UNICODE_NULL;
|
||||
}
|
||||
|
||||
/* Build the full target path */
|
||||
RtlStringCchCopyW(FileDstPath, ARRAYSIZE(FileDstPath),
|
||||
USetupData.DestinationRootPath.Buffer);
|
||||
if (DirKeyValue[0] == UNICODE_NULL)
|
||||
{
|
||||
/* Installation path */
|
||||
|
||||
/* Add the installation path */
|
||||
ConcatPaths(FileDstPath, ARRAYSIZE(FileDstPath), 1, USetupData.InstallPath.Buffer);
|
||||
}
|
||||
else if (DirKeyValue[0] == L'\\')
|
||||
{
|
||||
/* Absolute path */
|
||||
// if (DirKeyValue[1] != UNICODE_NULL)
|
||||
ConcatPaths(FileDstPath, ARRAYSIZE(FileDstPath), 1, DirKeyValue);
|
||||
}
|
||||
else // if (DirKeyValue[0] != L'\\')
|
||||
{
|
||||
/* Path relative to the installation path */
|
||||
|
||||
/* Add the installation path */
|
||||
ConcatPaths(FileDstPath, ARRAYSIZE(FileDstPath), 2,
|
||||
USetupData.InstallPath.Buffer, DirKeyValue);
|
||||
}
|
||||
#endif
|
||||
|
||||
if (!SetupQueueCopy(USetupData.SetupFileQueue,
|
||||
SourceCabinet,
|
||||
USetupData.SourceRootPath.Buffer,
|
||||
USetupData.SourceRootDir.Buffer,
|
||||
FileKeyName,
|
||||
DirKeyValue,
|
||||
FileDstPath,
|
||||
TargetFileName))
|
||||
{
|
||||
/* FIXME: Handle error! */
|
||||
|
@ -3584,6 +3621,7 @@ AddSectionToCopyQueue(HINF InfFile,
|
|||
PWCHAR DirKeyValue;
|
||||
PWCHAR TargetFileName;
|
||||
WCHAR CompleteOrigDirName[512]; // FIXME: MAX_PATH is not enough?
|
||||
WCHAR FileDstPath[MAX_PATH];
|
||||
|
||||
if (SourceCabinet)
|
||||
return AddSectionToCopyQueueCab(InfFile, L"SourceFiles", SourceCabinet, DestinationPath, Ir);
|
||||
|
@ -3683,12 +3721,46 @@ AddSectionToCopyQueue(HINF InfFile,
|
|||
DPRINT("RelativePath(2): '%S'\n", CompleteOrigDirName);
|
||||
}
|
||||
|
||||
#if 1 // HACK moved! (r66604)
|
||||
{
|
||||
ULONG Length = wcslen(DirKeyValue);
|
||||
if ((Length > 0) && (DirKeyValue[Length - 1] == L'\\'))
|
||||
Length--;
|
||||
DirKeyValue[Length] = UNICODE_NULL;
|
||||
}
|
||||
|
||||
/* Build the full target path */
|
||||
RtlStringCchCopyW(FileDstPath, ARRAYSIZE(FileDstPath),
|
||||
USetupData.DestinationRootPath.Buffer);
|
||||
if (DirKeyValue[0] == UNICODE_NULL)
|
||||
{
|
||||
/* Installation path */
|
||||
|
||||
/* Add the installation path */
|
||||
ConcatPaths(FileDstPath, ARRAYSIZE(FileDstPath), 1, USetupData.InstallPath.Buffer);
|
||||
}
|
||||
else if (DirKeyValue[0] == L'\\')
|
||||
{
|
||||
/* Absolute path */
|
||||
// if (DirKeyValue[1] != UNICODE_NULL)
|
||||
ConcatPaths(FileDstPath, ARRAYSIZE(FileDstPath), 1, DirKeyValue);
|
||||
}
|
||||
else // if (DirKeyValue[0] != L'\\')
|
||||
{
|
||||
/* Path relative to the installation path */
|
||||
|
||||
/* Add the installation path */
|
||||
ConcatPaths(FileDstPath, ARRAYSIZE(FileDstPath), 2,
|
||||
USetupData.InstallPath.Buffer, DirKeyValue);
|
||||
}
|
||||
#endif
|
||||
|
||||
if (!SetupQueueCopy(USetupData.SetupFileQueue,
|
||||
SourceCabinet,
|
||||
USetupData.SourceRootPath.Buffer,
|
||||
CompleteOrigDirName,
|
||||
FileKeyName,
|
||||
DirKeyValue,
|
||||
FileDstPath,
|
||||
TargetFileName))
|
||||
{
|
||||
/* FIXME: Handle error! */
|
||||
|
@ -3938,9 +4010,15 @@ PrepareCopyPage(PINPUT_RECORD Ir)
|
|||
return FILE_COPY_PAGE;
|
||||
}
|
||||
|
||||
typedef struct _COPYCONTEXT
|
||||
{
|
||||
ULONG TotalOperations;
|
||||
ULONG CompletedOperations;
|
||||
PPROGRESSBAR ProgressBar;
|
||||
PPROGRESSBAR MemoryBars[4];
|
||||
} COPYCONTEXT, *PCOPYCONTEXT;
|
||||
|
||||
VOID
|
||||
NTAPI
|
||||
static VOID
|
||||
SetupUpdateMemoryInfo(IN PCOPYCONTEXT CopyContext,
|
||||
IN BOOLEAN First)
|
||||
{
|
||||
|
@ -3967,7 +4045,6 @@ SetupUpdateMemoryInfo(IN PCOPYCONTEXT CopyContext,
|
|||
ProgressSetStep(CopyContext->MemoryBars[2], PerfInfo.AvailablePages);
|
||||
}
|
||||
|
||||
|
||||
static UINT
|
||||
CALLBACK
|
||||
FileCopyCallback(PVOID Context,
|
||||
|
@ -3975,26 +4052,78 @@ FileCopyCallback(PVOID Context,
|
|||
UINT_PTR Param1,
|
||||
UINT_PTR Param2)
|
||||
{
|
||||
PCOPYCONTEXT CopyContext;
|
||||
|
||||
CopyContext = (PCOPYCONTEXT)Context;
|
||||
PCOPYCONTEXT CopyContext = (PCOPYCONTEXT)Context;
|
||||
PFILEPATHS_W FilePathInfo;
|
||||
PCWSTR SrcFileName, DstFileName;
|
||||
|
||||
switch (Notification)
|
||||
{
|
||||
case SPFILENOTIFY_STARTSUBQUEUE:
|
||||
{
|
||||
CopyContext->TotalOperations = (ULONG)Param2;
|
||||
CopyContext->CompletedOperations = 0;
|
||||
ProgressSetStepCount(CopyContext->ProgressBar,
|
||||
CopyContext->TotalOperations);
|
||||
SetupUpdateMemoryInfo(CopyContext, TRUE);
|
||||
break;
|
||||
}
|
||||
|
||||
case SPFILENOTIFY_STARTDELETE:
|
||||
case SPFILENOTIFY_STARTRENAME:
|
||||
case SPFILENOTIFY_STARTCOPY:
|
||||
/* Display copy message */
|
||||
CONSOLE_SetStatusText(MUIGetString(STRING_COPYING), (PWSTR)Param1);
|
||||
{
|
||||
FilePathInfo = (PFILEPATHS_W)Param1;
|
||||
|
||||
if (Notification == SPFILENOTIFY_STARTDELETE)
|
||||
{
|
||||
/* Display delete message */
|
||||
ASSERT(Param2 == FILEOP_DELETE);
|
||||
|
||||
DstFileName = wcsrchr(FilePathInfo->Target, L'\\');
|
||||
if (DstFileName) ++DstFileName;
|
||||
else DstFileName = FilePathInfo->Target;
|
||||
|
||||
CONSOLE_SetStatusText(MUIGetString(STRING_DELETING),
|
||||
DstFileName);
|
||||
}
|
||||
else if (Notification == SPFILENOTIFY_STARTRENAME)
|
||||
{
|
||||
/* Display move/rename message */
|
||||
ASSERT(Param2 == FILEOP_RENAME);
|
||||
|
||||
SrcFileName = wcsrchr(FilePathInfo->Source, L'\\');
|
||||
if (SrcFileName) ++SrcFileName;
|
||||
else SrcFileName = FilePathInfo->Source;
|
||||
|
||||
DstFileName = wcsrchr(FilePathInfo->Target, L'\\');
|
||||
if (DstFileName) ++DstFileName;
|
||||
else DstFileName = FilePathInfo->Target;
|
||||
|
||||
// TODO: Determine whether using STRING_RENAMING or STRING_MOVING
|
||||
CONSOLE_SetStatusText(MUIGetString(STRING_MOVING),
|
||||
SrcFileName, DstFileName);
|
||||
}
|
||||
else if (Notification == SPFILENOTIFY_STARTCOPY)
|
||||
{
|
||||
/* Display copy message */
|
||||
ASSERT(Param2 == FILEOP_COPY);
|
||||
|
||||
SrcFileName = wcsrchr(FilePathInfo->Source, L'\\');
|
||||
if (SrcFileName) ++SrcFileName;
|
||||
else SrcFileName = FilePathInfo->Source;
|
||||
|
||||
CONSOLE_SetStatusText(MUIGetString(STRING_COPYING),
|
||||
SrcFileName);
|
||||
}
|
||||
|
||||
SetupUpdateMemoryInfo(CopyContext, FALSE);
|
||||
break;
|
||||
}
|
||||
|
||||
case SPFILENOTIFY_ENDDELETE:
|
||||
case SPFILENOTIFY_ENDRENAME:
|
||||
case SPFILENOTIFY_ENDCOPY:
|
||||
{
|
||||
CopyContext->CompletedOperations++;
|
||||
|
||||
/* SYSREG checkpoint */
|
||||
|
@ -4004,9 +4133,10 @@ FileCopyCallback(PVOID Context,
|
|||
ProgressNextStep(CopyContext->ProgressBar);
|
||||
SetupUpdateMemoryInfo(CopyContext, FALSE);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
return FILEOP_DOIT;
|
||||
}
|
||||
|
||||
|
||||
|
@ -4027,13 +4157,11 @@ static PAGE_NUMBER
|
|||
FileCopyPage(PINPUT_RECORD Ir)
|
||||
{
|
||||
COPYCONTEXT CopyContext;
|
||||
unsigned int mem_bar_width;
|
||||
UINT MemBarWidth;
|
||||
|
||||
MUIDisplayPage(FILE_COPY_PAGE);
|
||||
|
||||
/* Create context for the copy process */
|
||||
CopyContext.DestinationRootPath = USetupData.DestinationRootPath.Buffer;
|
||||
CopyContext.InstallPath = USetupData.InstallPath.Buffer;
|
||||
CopyContext.TotalOperations = 0;
|
||||
CopyContext.CompletedOperations = 0;
|
||||
|
||||
|
@ -4048,13 +4176,13 @@ FileCopyPage(PINPUT_RECORD Ir)
|
|||
MUIGetString(STRING_SETUPCOPYINGFILES));
|
||||
|
||||
// fit memory bars to screen width, distribute them uniform
|
||||
mem_bar_width = (xScreen - 26) / 5;
|
||||
mem_bar_width -= mem_bar_width % 2; // make even
|
||||
MemBarWidth = (xScreen - 26) / 5;
|
||||
MemBarWidth -= MemBarWidth % 2; // make even
|
||||
/* ATTENTION: The following progress bars are debug stuff, which should not be translated!! */
|
||||
/* Create the paged pool progress bar */
|
||||
CopyContext.MemoryBars[0] = CreateProgressBar(13,
|
||||
40,
|
||||
13 + mem_bar_width,
|
||||
13 + MemBarWidth,
|
||||
43,
|
||||
13,
|
||||
44,
|
||||
|
@ -4062,21 +4190,21 @@ FileCopyPage(PINPUT_RECORD Ir)
|
|||
"Kernel Pool");
|
||||
|
||||
/* Create the non paged pool progress bar */
|
||||
CopyContext.MemoryBars[1] = CreateProgressBar((xScreen / 2)- (mem_bar_width / 2),
|
||||
CopyContext.MemoryBars[1] = CreateProgressBar((xScreen / 2)- (MemBarWidth / 2),
|
||||
40,
|
||||
(xScreen / 2) + (mem_bar_width / 2),
|
||||
(xScreen / 2) + (MemBarWidth / 2),
|
||||
43,
|
||||
(xScreen / 2)- (mem_bar_width / 2),
|
||||
(xScreen / 2)- (MemBarWidth / 2),
|
||||
44,
|
||||
FALSE,
|
||||
"Kernel Cache");
|
||||
|
||||
/* Create the global memory progress bar */
|
||||
CopyContext.MemoryBars[2] = CreateProgressBar(xScreen - 13 - mem_bar_width,
|
||||
CopyContext.MemoryBars[2] = CreateProgressBar(xScreen - 13 - MemBarWidth,
|
||||
40,
|
||||
xScreen - 13,
|
||||
43,
|
||||
xScreen - 13 - mem_bar_width,
|
||||
xScreen - 13 - MemBarWidth,
|
||||
44,
|
||||
FALSE,
|
||||
"Free Memory");
|
||||
|
|
Loading…
Reference in a new issue