1 /* 2 * ReactOS kernel 3 * Copyright (C) 2002 ReactOS Team 4 * 5 * This program is free software; you can redistribute it and/or modify 6 * it under the terms of the GNU General Public License as published by 7 * the Free Software Foundation; either version 2 of the License, or 8 * (at your option) any later version. 9 * 10 * This program is distributed in the hope that it will be useful, 11 * but WITHOUT ANY WARRANTY; without even the implied warranty of 12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 13 * GNU General Public License for more details. 14 * 15 * You should have received a copy of the GNU General Public License along 16 * with this program; if not, write to the Free Software Foundation, Inc., 17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. 18 */ 19 /* 20 * COPYRIGHT: See COPYING in the top level directory 21 * PROJECT: ReactOS text-mode setup 22 * FILE: base/setup/lib/fileqsup.c 23 * PURPOSE: Interfacing with Setup* API File Queue support functions 24 * PROGRAMMERS: Casper S. Hornstrup (chorns@users.sourceforge.net) 25 * Hermes Belusca-Maito (hermes.belusca@sfr.fr) 26 */ 27 28 /* INCLUDES *****************************************************************/ 29 30 #ifndef _USETUP_PCH_ 31 #include "usetup.h" 32 #endif 33 34 #define NDEBUG 35 #include <debug.h> 36 37 /* DEFINITIONS **************************************************************/ 38 39 typedef struct _QUEUEENTRY 40 { 41 LIST_ENTRY ListEntry; 42 PWSTR SourceCabinet; /* May be NULL if the file is not in a cabinet */ 43 PWSTR SourceRootPath; 44 PWSTR SourcePath; 45 PWSTR SourceFileName; 46 PWSTR TargetDirectory; 47 PWSTR TargetFileName; 48 } QUEUEENTRY, *PQUEUEENTRY; 49 50 typedef struct _FILEQUEUEHEADER 51 { 52 LIST_ENTRY DeleteQueue; // PQUEUEENTRY entries 53 ULONG DeleteCount; 54 55 LIST_ENTRY RenameQueue; // PQUEUEENTRY entries 56 ULONG RenameCount; 57 58 LIST_ENTRY CopyQueue; // PQUEUEENTRY entries 59 ULONG CopyCount; 60 61 BOOLEAN HasCurrentCabinet; 62 CABINET_CONTEXT CabinetContext; 63 CAB_SEARCH Search; 64 WCHAR CurrentCabinetName[MAX_PATH]; 65 } FILEQUEUEHEADER, *PFILEQUEUEHEADER; 66 67 68 /* SETUP* API COMPATIBILITY FUNCTIONS ****************************************/ 69 70 static NTSTATUS 71 SetupExtractFile( 72 IN OUT PFILEQUEUEHEADER QueueHeader, 73 IN PCWSTR CabinetFileName, 74 IN PCWSTR SourceFileName, 75 IN PCWSTR DestinationPathName) 76 { 77 ULONG CabStatus; 78 79 DPRINT("SetupExtractFile(CabinetFileName: '%S', SourceFileName: '%S', DestinationPathName: '%S')\n", 80 CabinetFileName, SourceFileName, DestinationPathName); 81 82 if (QueueHeader->HasCurrentCabinet) 83 { 84 DPRINT("CurrentCabinetName: '%S'\n", QueueHeader->CurrentCabinetName); 85 } 86 87 if (QueueHeader->HasCurrentCabinet && 88 (wcscmp(CabinetFileName, QueueHeader->CurrentCabinetName) == 0)) 89 { 90 DPRINT("Using same cabinet as last time\n"); 91 92 /* Use our last location because the files should be sequential */ 93 CabStatus = CabinetFindNextFileSequential(&QueueHeader->CabinetContext, 94 SourceFileName, 95 &QueueHeader->Search); 96 if (CabStatus != CAB_STATUS_SUCCESS) 97 { 98 DPRINT("Sequential miss on file: %S\n", SourceFileName); 99 100 /* Looks like we got unlucky */ 101 CabStatus = CabinetFindFirst(&QueueHeader->CabinetContext, 102 SourceFileName, 103 &QueueHeader->Search); 104 } 105 } 106 else 107 { 108 DPRINT("Using new cabinet\n"); 109 110 if (QueueHeader->HasCurrentCabinet) 111 { 112 QueueHeader->HasCurrentCabinet = FALSE; 113 CabinetCleanup(&QueueHeader->CabinetContext); 114 } 115 116 RtlStringCchCopyW(QueueHeader->CurrentCabinetName, 117 ARRAYSIZE(QueueHeader->CurrentCabinetName), 118 CabinetFileName); 119 120 CabinetInitialize(&QueueHeader->CabinetContext); 121 CabinetSetEventHandlers(&QueueHeader->CabinetContext, 122 NULL, NULL, NULL, NULL); 123 CabinetSetCabinetName(&QueueHeader->CabinetContext, CabinetFileName); 124 125 CabStatus = CabinetOpen(&QueueHeader->CabinetContext); 126 if (CabStatus == CAB_STATUS_SUCCESS) 127 { 128 DPRINT("Opened cabinet %S\n", CabinetFileName /*CabinetGetCabinetName(&QueueHeader->CabinetContext)*/); 129 QueueHeader->HasCurrentCabinet = TRUE; 130 } 131 else 132 { 133 DPRINT("Cannot open cabinet (%d)\n", CabStatus); 134 return STATUS_UNSUCCESSFUL; 135 } 136 137 /* We have to start at the beginning here */ 138 CabStatus = CabinetFindFirst(&QueueHeader->CabinetContext, 139 SourceFileName, 140 &QueueHeader->Search); 141 } 142 143 if (CabStatus != CAB_STATUS_SUCCESS) 144 { 145 DPRINT1("Unable to find '%S' in cabinet '%S'\n", 146 SourceFileName, CabinetGetCabinetName(&QueueHeader->CabinetContext)); 147 return STATUS_UNSUCCESSFUL; 148 } 149 150 CabinetSetDestinationPath(&QueueHeader->CabinetContext, DestinationPathName); 151 CabStatus = CabinetExtractFile(&QueueHeader->CabinetContext, &QueueHeader->Search); 152 if (CabStatus != CAB_STATUS_SUCCESS) 153 { 154 DPRINT("Cannot extract file %S (%d)\n", SourceFileName, CabStatus); 155 return STATUS_UNSUCCESSFUL; 156 } 157 158 return STATUS_SUCCESS; 159 } 160 161 HSPFILEQ 162 WINAPI 163 SetupOpenFileQueue(VOID) 164 { 165 PFILEQUEUEHEADER QueueHeader; 166 167 /* Allocate the queue header */ 168 QueueHeader = RtlAllocateHeap(ProcessHeap, 0, sizeof(FILEQUEUEHEADER)); 169 if (QueueHeader == NULL) 170 return NULL; 171 172 RtlZeroMemory(QueueHeader, sizeof(FILEQUEUEHEADER)); 173 174 /* Initialize the file queues */ 175 InitializeListHead(&QueueHeader->DeleteQueue); 176 QueueHeader->DeleteCount = 0; 177 InitializeListHead(&QueueHeader->RenameQueue); 178 QueueHeader->RenameCount = 0; 179 InitializeListHead(&QueueHeader->CopyQueue); 180 QueueHeader->CopyCount = 0; 181 182 QueueHeader->HasCurrentCabinet = FALSE; 183 184 return (HSPFILEQ)QueueHeader; 185 } 186 187 static VOID 188 SetupDeleteQueueEntry( 189 IN PQUEUEENTRY Entry) 190 { 191 if (Entry == NULL) 192 return; 193 194 /* Delete all strings */ 195 if (Entry->SourceCabinet != NULL) 196 RtlFreeHeap(ProcessHeap, 0, Entry->SourceCabinet); 197 198 if (Entry->SourceRootPath != NULL) 199 RtlFreeHeap(ProcessHeap, 0, Entry->SourceRootPath); 200 201 if (Entry->SourcePath != NULL) 202 RtlFreeHeap(ProcessHeap, 0, Entry->SourcePath); 203 204 if (Entry->SourceFileName != NULL) 205 RtlFreeHeap(ProcessHeap, 0, Entry->SourceFileName); 206 207 if (Entry->TargetDirectory != NULL) 208 RtlFreeHeap(ProcessHeap, 0, Entry->TargetDirectory); 209 210 if (Entry->TargetFileName != NULL) 211 RtlFreeHeap(ProcessHeap, 0, Entry->TargetFileName); 212 213 /* Delete queue entry */ 214 RtlFreeHeap(ProcessHeap, 0, Entry); 215 } 216 217 BOOL 218 WINAPI 219 SetupCloseFileQueue( 220 IN HSPFILEQ QueueHandle) 221 { 222 PFILEQUEUEHEADER QueueHeader; 223 PLIST_ENTRY ListEntry; 224 PQUEUEENTRY Entry; 225 226 if (QueueHandle == NULL) 227 return FALSE; 228 229 QueueHeader = (PFILEQUEUEHEADER)QueueHandle; 230 231 /* Delete the delete queue */ 232 while (!IsListEmpty(&QueueHeader->DeleteQueue)) 233 { 234 ListEntry = RemoveHeadList(&QueueHeader->DeleteQueue); 235 Entry = CONTAINING_RECORD(ListEntry, QUEUEENTRY, ListEntry); 236 SetupDeleteQueueEntry(Entry); 237 } 238 239 /* Delete the rename queue */ 240 while (!IsListEmpty(&QueueHeader->RenameQueue)) 241 { 242 ListEntry = RemoveHeadList(&QueueHeader->RenameQueue); 243 Entry = CONTAINING_RECORD(ListEntry, QUEUEENTRY, ListEntry); 244 SetupDeleteQueueEntry(Entry); 245 } 246 247 /* Delete the copy queue */ 248 while (!IsListEmpty(&QueueHeader->CopyQueue)) 249 { 250 ListEntry = RemoveHeadList(&QueueHeader->CopyQueue); 251 Entry = CONTAINING_RECORD(ListEntry, QUEUEENTRY, ListEntry); 252 SetupDeleteQueueEntry(Entry); 253 } 254 255 /* Delete queue header */ 256 RtlFreeHeap(ProcessHeap, 0, QueueHeader); 257 258 return TRUE; 259 } 260 261 /* A simplified version of SetupQueueCopyW that wraps Cabinet support around */ 262 BOOL 263 WINAPI 264 SetupQueueCopyWithCab( 265 IN HSPFILEQ QueueHandle, 266 IN PCWSTR SourceRootPath, 267 IN PCWSTR SourcePath OPTIONAL, 268 IN PCWSTR SourceFileName, 269 IN PCWSTR SourceDescription OPTIONAL, 270 IN PCWSTR SourceCabinet OPTIONAL, 271 IN PCWSTR SourceTagFile OPTIONAL, 272 IN PCWSTR TargetDirectory, 273 IN PCWSTR TargetFileName OPTIONAL, 274 IN ULONG CopyStyle) 275 { 276 PFILEQUEUEHEADER QueueHeader; 277 PQUEUEENTRY Entry; 278 ULONG Length; 279 280 if (QueueHandle == NULL || 281 SourceRootPath == NULL || 282 SourceFileName == NULL || 283 TargetDirectory == NULL) 284 { 285 return FALSE; 286 } 287 288 QueueHeader = (PFILEQUEUEHEADER)QueueHandle; 289 290 DPRINT("SetupQueueCopy(Cab '%S', SrcRootPath '%S', SrcPath '%S', SrcFN '%S' --> DstPath '%S', DstFN '%S')\n", 291 SourceCabinet ? SourceCabinet : L"n/a", 292 SourceRootPath, SourcePath, SourceFileName, 293 TargetDirectory, TargetFileName); 294 295 /* Allocate new queue entry */ 296 Entry = RtlAllocateHeap(ProcessHeap, 0, sizeof(QUEUEENTRY)); 297 if (Entry == NULL) 298 return FALSE; 299 300 RtlZeroMemory(Entry, sizeof(QUEUEENTRY)); 301 302 /* Copy source cabinet if available */ 303 Entry->SourceCabinet = NULL; 304 if (SourceCabinet != NULL) 305 { 306 Length = wcslen(SourceCabinet); 307 Entry->SourceCabinet = RtlAllocateHeap(ProcessHeap, 308 0, 309 (Length + 1) * sizeof(WCHAR)); 310 if (Entry->SourceCabinet == NULL) 311 { 312 RtlFreeHeap(ProcessHeap, 0, Entry); 313 return FALSE; 314 } 315 RtlStringCchCopyW(Entry->SourceCabinet, Length + 1, SourceCabinet); 316 } 317 318 /* Copy source root path */ 319 Length = wcslen(SourceRootPath); 320 Entry->SourceRootPath = RtlAllocateHeap(ProcessHeap, 321 0, 322 (Length + 1) * sizeof(WCHAR)); 323 if (Entry->SourceRootPath == NULL) 324 { 325 if (Entry->SourceCabinet != NULL) 326 RtlFreeHeap(ProcessHeap, 0, Entry->SourceCabinet); 327 328 RtlFreeHeap(ProcessHeap, 0, Entry); 329 return FALSE; 330 } 331 RtlStringCchCopyW(Entry->SourceRootPath, Length + 1, SourceRootPath); 332 333 /* Copy source path */ 334 Entry->SourcePath = NULL; 335 if (SourcePath != NULL) 336 { 337 Length = wcslen(SourcePath); 338 if ((Length > 0) && (SourcePath[Length - 1] == L'\\')) 339 Length--; 340 Entry->SourcePath = RtlAllocateHeap(ProcessHeap, 341 0, 342 (Length + 1) * sizeof(WCHAR)); 343 if (Entry->SourcePath == NULL) 344 { 345 if (Entry->SourceCabinet != NULL) 346 RtlFreeHeap(ProcessHeap, 0, Entry->SourceCabinet); 347 348 RtlFreeHeap(ProcessHeap, 0, Entry->SourceRootPath); 349 RtlFreeHeap(ProcessHeap, 0, Entry); 350 return FALSE; 351 } 352 RtlStringCchCopyW(Entry->SourcePath, Length + 1, SourcePath); 353 } 354 355 /* Copy source file name */ 356 Length = wcslen(SourceFileName); 357 Entry->SourceFileName = (WCHAR*)RtlAllocateHeap(ProcessHeap, 358 0, 359 (Length + 1) * sizeof(WCHAR)); 360 if (Entry->SourceFileName == NULL) 361 { 362 if (Entry->SourcePath != NULL) 363 RtlFreeHeap(ProcessHeap, 0, Entry->SourcePath); 364 365 RtlFreeHeap(ProcessHeap, 0, Entry->SourceRootPath); 366 367 if (Entry->SourceCabinet != NULL) 368 RtlFreeHeap(ProcessHeap, 0, Entry->SourceCabinet); 369 370 RtlFreeHeap(ProcessHeap, 0, Entry); 371 return FALSE; 372 } 373 RtlStringCchCopyW(Entry->SourceFileName, Length + 1, SourceFileName); 374 375 /* Copy target directory */ 376 Length = wcslen(TargetDirectory); 377 if ((Length > 0) && (TargetDirectory[Length - 1] == L'\\')) 378 Length--; 379 Entry->TargetDirectory = RtlAllocateHeap(ProcessHeap, 380 0, 381 (Length + 1) * sizeof(WCHAR)); 382 if (Entry->TargetDirectory == NULL) 383 { 384 RtlFreeHeap(ProcessHeap, 0, Entry->SourceFileName); 385 386 if (Entry->SourcePath != NULL) 387 RtlFreeHeap(ProcessHeap, 0, Entry->SourcePath); 388 389 RtlFreeHeap(ProcessHeap, 0, Entry->SourceRootPath); 390 391 if (Entry->SourceCabinet != NULL) 392 RtlFreeHeap(ProcessHeap, 0, Entry->SourceCabinet); 393 394 RtlFreeHeap(ProcessHeap, 0, Entry); 395 return FALSE; 396 } 397 RtlStringCchCopyW(Entry->TargetDirectory, Length + 1, TargetDirectory); 398 399 /* Copy optional target filename */ 400 Entry->TargetFileName = NULL; 401 if (TargetFileName != NULL) 402 { 403 Length = wcslen(TargetFileName); 404 Entry->TargetFileName = RtlAllocateHeap(ProcessHeap, 405 0, 406 (Length + 1) * sizeof(WCHAR)); 407 if (Entry->TargetFileName == NULL) 408 { 409 RtlFreeHeap(ProcessHeap, 0, Entry->TargetDirectory); 410 RtlFreeHeap(ProcessHeap, 0, Entry->SourceFileName); 411 412 if (Entry->SourcePath != NULL) 413 RtlFreeHeap(ProcessHeap, 0, Entry->SourcePath); 414 415 RtlFreeHeap(ProcessHeap, 0, Entry->SourceRootPath); 416 417 if (Entry->SourceCabinet != NULL) 418 RtlFreeHeap(ProcessHeap, 0, Entry->SourceCabinet); 419 420 RtlFreeHeap(ProcessHeap, 0, Entry); 421 return FALSE; 422 } 423 RtlStringCchCopyW(Entry->TargetFileName, Length + 1, TargetFileName); 424 } 425 426 /* Append queue entry */ 427 InsertTailList(&QueueHeader->CopyQueue, &Entry->ListEntry); 428 ++QueueHeader->CopyCount; 429 430 return TRUE; 431 } 432 433 BOOL 434 WINAPI 435 SetupQueueDeleteW( 436 IN HSPFILEQ QueueHandle, 437 IN PCWSTR PathPart1, 438 IN PCWSTR PathPart2 OPTIONAL) 439 { 440 PFILEQUEUEHEADER QueueHeader; 441 PQUEUEENTRY Entry; 442 ULONG Length; 443 444 if (QueueHandle == NULL || PathPart1 == NULL) 445 { 446 return FALSE; 447 } 448 449 QueueHeader = (PFILEQUEUEHEADER)QueueHandle; 450 451 DPRINT1("SetupQueueDeleteW(PathPart1 '%S', PathPart2 '%S')\n", 452 PathPart1, PathPart2); 453 454 /* Allocate new queue entry */ 455 Entry = RtlAllocateHeap(ProcessHeap, 0, sizeof(QUEUEENTRY)); 456 if (Entry == NULL) 457 return FALSE; 458 459 RtlZeroMemory(Entry, sizeof(QUEUEENTRY)); 460 461 Entry->SourceCabinet = NULL; 462 Entry->SourceRootPath = NULL; 463 Entry->SourcePath = NULL; 464 Entry->SourceFileName = NULL; 465 466 /* Copy first part of path */ 467 Length = wcslen(PathPart1); 468 // if ((Length > 0) && (SourcePath[Length - 1] == L'\\')) 469 // Length--; 470 Entry->TargetDirectory = RtlAllocateHeap(ProcessHeap, 471 0, 472 (Length + 1) * sizeof(WCHAR)); 473 if (Entry->TargetDirectory == NULL) 474 { 475 RtlFreeHeap(ProcessHeap, 0, Entry); 476 return FALSE; 477 } 478 RtlStringCchCopyW(Entry->TargetDirectory, Length + 1, PathPart1); 479 480 /* Copy optional second part of path */ 481 if (PathPart2 != NULL) 482 { 483 Length = wcslen(PathPart2); 484 Entry->TargetFileName = RtlAllocateHeap(ProcessHeap, 485 0, 486 (Length + 1) * sizeof(WCHAR)); 487 if (Entry->TargetFileName == NULL) 488 { 489 RtlFreeHeap(ProcessHeap, 0, Entry->TargetDirectory); 490 RtlFreeHeap(ProcessHeap, 0, Entry); 491 return FALSE; 492 } 493 RtlStringCchCopyW(Entry->TargetFileName, Length + 1, PathPart2); 494 } 495 496 /* Append the queue entry */ 497 InsertTailList(&QueueHeader->DeleteQueue, &Entry->ListEntry); 498 ++QueueHeader->DeleteCount; 499 500 return TRUE; 501 } 502 503 BOOL 504 WINAPI 505 SetupQueueRenameW( 506 IN HSPFILEQ QueueHandle, 507 IN PCWSTR SourcePath, 508 IN PCWSTR SourceFileName OPTIONAL, 509 IN PCWSTR TargetPath OPTIONAL, 510 IN PCWSTR TargetFileName) 511 { 512 PFILEQUEUEHEADER QueueHeader; 513 PQUEUEENTRY Entry; 514 ULONG Length; 515 516 if (QueueHandle == NULL || 517 SourcePath == NULL || 518 TargetFileName == NULL) 519 { 520 return FALSE; 521 } 522 523 QueueHeader = (PFILEQUEUEHEADER)QueueHandle; 524 525 DPRINT1("SetupQueueRenameW(SrcPath '%S', SrcFN '%S' --> DstPath '%S', DstFN '%S')\n", 526 SourcePath, SourceFileName, TargetPath, TargetFileName); 527 528 /* Allocate a new queue entry */ 529 Entry = RtlAllocateHeap(ProcessHeap, 0, sizeof(QUEUEENTRY)); 530 if (Entry == NULL) 531 return FALSE; 532 533 RtlZeroMemory(Entry, sizeof(QUEUEENTRY)); 534 535 Entry->SourceCabinet = NULL; 536 Entry->SourceRootPath = NULL; 537 538 /* Copy source path */ 539 Length = wcslen(SourcePath); 540 if ((Length > 0) && (SourcePath[Length - 1] == L'\\')) 541 Length--; 542 Entry->SourcePath = RtlAllocateHeap(ProcessHeap, 543 0, 544 (Length + 1) * sizeof(WCHAR)); 545 if (Entry->SourcePath == NULL) 546 { 547 RtlFreeHeap(ProcessHeap, 0, Entry); 548 return FALSE; 549 } 550 RtlStringCchCopyW(Entry->SourcePath, Length + 1, SourcePath); 551 552 /* Copy optional source file name */ 553 Entry->SourceFileName = NULL; 554 if (SourceFileName != NULL) 555 { 556 Length = wcslen(SourceFileName); 557 Entry->SourceFileName = (WCHAR*)RtlAllocateHeap(ProcessHeap, 558 0, 559 (Length + 1) * sizeof(WCHAR)); 560 if (Entry->SourceFileName == NULL) 561 { 562 RtlFreeHeap(ProcessHeap, 0, Entry->SourcePath); 563 RtlFreeHeap(ProcessHeap, 0, Entry); 564 return FALSE; 565 } 566 RtlStringCchCopyW(Entry->SourceFileName, Length + 1, SourceFileName); 567 } 568 569 /* Copy optional target directory */ 570 Entry->TargetDirectory = NULL; 571 if (TargetPath != NULL) 572 { 573 Length = wcslen(TargetPath); 574 if ((Length > 0) && (TargetPath[Length - 1] == L'\\')) 575 Length--; 576 Entry->TargetDirectory = RtlAllocateHeap(ProcessHeap, 577 0, 578 (Length + 1) * sizeof(WCHAR)); 579 if (Entry->TargetDirectory == NULL) 580 { 581 if (Entry->SourceFileName != NULL) 582 RtlFreeHeap(ProcessHeap, 0, Entry->SourceFileName); 583 584 RtlFreeHeap(ProcessHeap, 0, Entry->SourcePath); 585 RtlFreeHeap(ProcessHeap, 0, Entry); 586 return FALSE; 587 } 588 RtlStringCchCopyW(Entry->TargetDirectory, Length + 1, TargetPath); 589 } 590 591 /* Copy target filename */ 592 Length = wcslen(TargetFileName); 593 Entry->TargetFileName = RtlAllocateHeap(ProcessHeap, 594 0, 595 (Length + 1) * sizeof(WCHAR)); 596 if (Entry->TargetFileName == NULL) 597 { 598 if (Entry->TargetDirectory != NULL) 599 RtlFreeHeap(ProcessHeap, 0, Entry->TargetDirectory); 600 601 if (Entry->SourceFileName != NULL) 602 RtlFreeHeap(ProcessHeap, 0, Entry->SourceFileName); 603 604 RtlFreeHeap(ProcessHeap, 0, Entry->SourcePath); 605 RtlFreeHeap(ProcessHeap, 0, Entry); 606 return FALSE; 607 } 608 RtlStringCchCopyW(Entry->TargetFileName, Length + 1, TargetFileName); 609 610 /* Append the queue entry */ 611 InsertTailList(&QueueHeader->RenameQueue, &Entry->ListEntry); 612 ++QueueHeader->RenameCount; 613 614 return TRUE; 615 } 616 617 BOOL 618 WINAPI 619 SetupCommitFileQueueW( 620 IN HWND Owner, 621 IN HSPFILEQ QueueHandle, 622 IN PSP_FILE_CALLBACK_W MsgHandler, 623 IN PVOID Context OPTIONAL) 624 { 625 BOOL Success = TRUE; // Suppose success 626 UINT Result; 627 NTSTATUS Status; 628 PFILEQUEUEHEADER QueueHeader; 629 PLIST_ENTRY ListEntry; 630 PQUEUEENTRY Entry; 631 FILEPATHS_W FilePathInfo; 632 WCHAR FileSrcPath[MAX_PATH]; 633 WCHAR FileDstPath[MAX_PATH]; 634 635 if (QueueHandle == NULL) 636 return FALSE; 637 638 QueueHeader = (PFILEQUEUEHEADER)QueueHandle; 639 640 Result = MsgHandler(Context, 641 SPFILENOTIFY_STARTQUEUE, 642 (UINT_PTR)Owner, 643 0); 644 if (Result == FILEOP_ABORT) 645 return FALSE; 646 647 648 /* 649 * Commit the delete queue 650 */ 651 652 if (!IsListEmpty(&QueueHeader->DeleteQueue)) 653 { 654 Result = MsgHandler(Context, 655 SPFILENOTIFY_STARTSUBQUEUE, 656 FILEOP_DELETE, 657 QueueHeader->DeleteCount); 658 if (Result == FILEOP_ABORT) 659 { 660 Success = FALSE; 661 goto Quit; 662 } 663 } 664 665 for (ListEntry = QueueHeader->DeleteQueue.Flink; 666 ListEntry != &QueueHeader->DeleteQueue; 667 ListEntry = ListEntry->Flink) 668 { 669 Entry = CONTAINING_RECORD(ListEntry, QUEUEENTRY, ListEntry); 670 671 /* Build the full target path */ 672 CombinePaths(FileDstPath, ARRAYSIZE(FileDstPath), 2, 673 Entry->TargetDirectory, Entry->TargetFileName); 674 675 DPRINT1(" -----> " "Delete: '%S'\n", FileDstPath); 676 677 FilePathInfo.Target = FileDstPath; 678 FilePathInfo.Source = NULL; 679 FilePathInfo.Win32Error = STATUS_SUCCESS; 680 FilePathInfo.Flags = 0; // FIXME: Unused yet... 681 682 Result = MsgHandler(Context, 683 SPFILENOTIFY_STARTDELETE, 684 (UINT_PTR)&FilePathInfo, 685 FILEOP_DELETE); 686 if (Result == FILEOP_ABORT) 687 { 688 Success = FALSE; 689 goto EndDelete; 690 } 691 else if (Result == FILEOP_SKIP) 692 goto EndDelete; 693 // else (Result == FILEOP_DOIT) 694 695 RetryDelete: 696 /* Force-delete the file */ 697 Status = SetupDeleteFile(FileDstPath, TRUE); 698 if (!NT_SUCCESS(Status)) 699 { 700 /* An error happened */ 701 FilePathInfo.Win32Error = (UINT)Status; 702 Result = MsgHandler(Context, 703 SPFILENOTIFY_DELETEERROR, 704 (UINT_PTR)&FilePathInfo, 705 0); 706 if (Result == FILEOP_ABORT) 707 { 708 Success = FALSE; 709 goto EndDelete; 710 } 711 else if (Result == FILEOP_SKIP) 712 goto EndDelete; 713 else if (Result == FILEOP_RETRY) 714 goto RetryDelete; 715 716 Success = FALSE; 717 } 718 719 EndDelete: 720 /* This notification is always sent, even in case of error */ 721 FilePathInfo.Win32Error = (UINT)Status; 722 MsgHandler(Context, 723 SPFILENOTIFY_ENDDELETE, 724 (UINT_PTR)&FilePathInfo, 725 0); 726 if (Success == FALSE /* && Result == FILEOP_ABORT */) 727 goto Quit; 728 } 729 730 if (!IsListEmpty(&QueueHeader->DeleteQueue)) 731 { 732 MsgHandler(Context, 733 SPFILENOTIFY_ENDSUBQUEUE, 734 FILEOP_DELETE, 735 0); 736 } 737 738 739 /* 740 * Commit the rename queue 741 */ 742 743 if (!IsListEmpty(&QueueHeader->RenameQueue)) 744 { 745 Result = MsgHandler(Context, 746 SPFILENOTIFY_STARTSUBQUEUE, 747 FILEOP_RENAME, 748 QueueHeader->RenameCount); 749 if (Result == FILEOP_ABORT) 750 { 751 Success = FALSE; 752 goto Quit; 753 } 754 } 755 756 for (ListEntry = QueueHeader->RenameQueue.Flink; 757 ListEntry != &QueueHeader->RenameQueue; 758 ListEntry = ListEntry->Flink) 759 { 760 Entry = CONTAINING_RECORD(ListEntry, QUEUEENTRY, ListEntry); 761 762 /* Build the full source path */ 763 CombinePaths(FileSrcPath, ARRAYSIZE(FileSrcPath), 2, 764 Entry->SourcePath, Entry->SourceFileName); 765 766 /* Build the full target path */ 767 CombinePaths(FileDstPath, ARRAYSIZE(FileDstPath), 2, 768 Entry->TargetDirectory, Entry->TargetFileName); 769 770 DPRINT1(" -----> " "Rename: '%S' ==> '%S'\n", FileSrcPath, FileDstPath); 771 772 FilePathInfo.Target = FileDstPath; 773 FilePathInfo.Source = FileSrcPath; 774 FilePathInfo.Win32Error = STATUS_SUCCESS; 775 FilePathInfo.Flags = 0; // FIXME: Unused yet... 776 777 Result = MsgHandler(Context, 778 SPFILENOTIFY_STARTRENAME, 779 (UINT_PTR)&FilePathInfo, 780 FILEOP_RENAME); 781 if (Result == FILEOP_ABORT) 782 { 783 Success = FALSE; 784 goto EndRename; 785 } 786 else if (Result == FILEOP_SKIP) 787 goto EndRename; 788 // else (Result == FILEOP_DOIT) 789 790 RetryRename: 791 /* Move or rename the file */ 792 Status = SetupMoveFile(FileSrcPath, FileDstPath, 793 MOVEFILE_REPLACE_EXISTING 794 | MOVEFILE_COPY_ALLOWED 795 | MOVEFILE_WRITE_THROUGH); 796 if (!NT_SUCCESS(Status)) 797 { 798 /* An error happened */ 799 FilePathInfo.Win32Error = (UINT)Status; 800 Result = MsgHandler(Context, 801 SPFILENOTIFY_RENAMEERROR, 802 (UINT_PTR)&FilePathInfo, 803 0); 804 if (Result == FILEOP_ABORT) 805 { 806 Success = FALSE; 807 goto EndRename; 808 } 809 else if (Result == FILEOP_SKIP) 810 goto EndRename; 811 else if (Result == FILEOP_RETRY) 812 goto RetryRename; 813 814 Success = FALSE; 815 } 816 817 EndRename: 818 /* This notification is always sent, even in case of error */ 819 FilePathInfo.Win32Error = (UINT)Status; 820 MsgHandler(Context, 821 SPFILENOTIFY_ENDRENAME, 822 (UINT_PTR)&FilePathInfo, 823 0); 824 if (Success == FALSE /* && Result == FILEOP_ABORT */) 825 goto Quit; 826 } 827 828 if (!IsListEmpty(&QueueHeader->RenameQueue)) 829 { 830 MsgHandler(Context, 831 SPFILENOTIFY_ENDSUBQUEUE, 832 FILEOP_RENAME, 833 0); 834 } 835 836 837 /* 838 * Commit the copy queue 839 */ 840 841 if (!IsListEmpty(&QueueHeader->CopyQueue)) 842 { 843 Result = MsgHandler(Context, 844 SPFILENOTIFY_STARTSUBQUEUE, 845 FILEOP_COPY, 846 QueueHeader->CopyCount); 847 if (Result == FILEOP_ABORT) 848 { 849 Success = FALSE; 850 goto Quit; 851 } 852 } 853 854 for (ListEntry = QueueHeader->CopyQueue.Flink; 855 ListEntry != &QueueHeader->CopyQueue; 856 ListEntry = ListEntry->Flink) 857 { 858 Entry = CONTAINING_RECORD(ListEntry, QUEUEENTRY, ListEntry); 859 860 // 861 // TODO: Send a SPFILENOTIFY_NEEDMEDIA notification 862 // when we switch to a new installation media. 863 // Param1 = (UINT_PTR)(PSOURCE_MEDIA)SourceMediaInfo; 864 // Param2 = (UINT_PTR)(TCHAR[MAX_PATH])NewPathInfo; 865 // 866 867 /* Build the full source path */ 868 if (Entry->SourceCabinet == NULL) 869 { 870 CombinePaths(FileSrcPath, ARRAYSIZE(FileSrcPath), 3, 871 Entry->SourceRootPath, Entry->SourcePath, 872 Entry->SourceFileName); 873 } 874 else 875 { 876 /* 877 * The cabinet must be in Entry->SourceRootPath only! 878 * (Should we ignore Entry->SourcePath?) 879 */ 880 CombinePaths(FileSrcPath, ARRAYSIZE(FileSrcPath), 3, 881 Entry->SourceRootPath, Entry->SourcePath, 882 Entry->SourceCabinet); 883 } 884 885 /* Build the full target path */ 886 RtlStringCchCopyW(FileDstPath, ARRAYSIZE(FileDstPath), Entry->TargetDirectory); 887 if (Entry->SourceCabinet == NULL) 888 { 889 /* If the file is not in a cabinet, possibly use a different target name */ 890 if (Entry->TargetFileName != NULL) 891 ConcatPaths(FileDstPath, ARRAYSIZE(FileDstPath), 1, Entry->TargetFileName); 892 else 893 ConcatPaths(FileDstPath, ARRAYSIZE(FileDstPath), 1, Entry->SourceFileName); 894 } 895 else 896 { 897 ConcatPaths(FileDstPath, ARRAYSIZE(FileDstPath), 1, Entry->SourceFileName); 898 } 899 900 DPRINT(" -----> " "Copy: '%S' ==> '%S'\n", FileSrcPath, FileDstPath); 901 902 // 903 // Technically, here we should create the target directory, 904 // if it does not already exist... before calling the handler! 905 // 906 907 FilePathInfo.Target = FileDstPath; 908 FilePathInfo.Source = FileSrcPath; 909 FilePathInfo.Win32Error = STATUS_SUCCESS; 910 FilePathInfo.Flags = 0; // FIXME: Unused yet... 911 912 Result = MsgHandler(Context, 913 SPFILENOTIFY_STARTCOPY, 914 (UINT_PTR)&FilePathInfo, 915 FILEOP_COPY); 916 if (Result == FILEOP_ABORT) 917 { 918 Success = FALSE; 919 goto EndCopy; 920 } 921 else if (Result == FILEOP_SKIP) 922 goto EndCopy; 923 // else (Result == FILEOP_DOIT) 924 925 RetryCopy: 926 if (Entry->SourceCabinet != NULL) 927 { 928 /* 929 * The file is in a cabinet, use only the destination path 930 * and keep the source name as the target name. 931 */ 932 /* Extract the file from the cabinet */ 933 Status = SetupExtractFile(QueueHeader, 934 FileSrcPath, // Specifies the cabinet path 935 Entry->SourceFileName, 936 Entry->TargetDirectory); 937 } 938 else 939 { 940 /* Copy the file */ 941 Status = SetupCopyFile(FileSrcPath, FileDstPath, FALSE); 942 } 943 944 if (!NT_SUCCESS(Status)) 945 { 946 /* An error happened */ 947 FilePathInfo.Win32Error = (UINT)Status; 948 Result = MsgHandler(Context, 949 SPFILENOTIFY_COPYERROR, 950 (UINT_PTR)&FilePathInfo, 951 (UINT_PTR)NULL); // FIXME: Unused yet... 952 if (Result == FILEOP_ABORT) 953 { 954 Success = FALSE; 955 goto EndCopy; 956 } 957 else if (Result == FILEOP_SKIP) 958 goto EndCopy; 959 else if (Result == FILEOP_RETRY) 960 goto RetryCopy; 961 else if (Result == FILEOP_NEWPATH) 962 goto RetryCopy; // TODO! 963 964 Success = FALSE; 965 } 966 967 EndCopy: 968 /* This notification is always sent, even in case of error */ 969 FilePathInfo.Win32Error = (UINT)Status; 970 MsgHandler(Context, 971 SPFILENOTIFY_ENDCOPY, 972 (UINT_PTR)&FilePathInfo, 973 0); 974 if (Success == FALSE /* && Result == FILEOP_ABORT */) 975 goto Quit; 976 } 977 978 if (!IsListEmpty(&QueueHeader->CopyQueue)) 979 { 980 MsgHandler(Context, 981 SPFILENOTIFY_ENDSUBQUEUE, 982 FILEOP_COPY, 983 0); 984 } 985 986 987 Quit: 988 /* All the queues have been committed */ 989 MsgHandler(Context, 990 SPFILENOTIFY_ENDQUEUE, 991 (UINT_PTR)Success, 992 0); 993 994 return Success; 995 } 996 997 998 /* GLOBALS *******************************************************************/ 999 1000 SPFILE_EXPORTS SpFileExports = 1001 { 1002 SetupOpenFileQueue, 1003 SetupCloseFileQueue, 1004 SetupQueueCopyWithCab, 1005 SetupQueueDeleteW, 1006 SetupQueueRenameW, 1007 SetupCommitFileQueueW 1008 }; 1009 1010 /* EOF */ 1011