1 /* 2 * Copyright 2016 Michael Müller 3 * 4 * This library is free software; you can redistribute it and/or 5 * modify it under the terms of the GNU Lesser General Public 6 * License as published by the Free Software Foundation; either 7 * version 2.1 of the License, or (at your option) any later version. 8 * 9 * This library is distributed in the hope that it will be useful, 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of 11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 * Lesser General Public License for more details. 13 * 14 * You should have received a copy of the GNU Lesser General Public 15 * License along with this library; if not, write to the Free Software 16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA 17 */ 18 19 #define COBJMACROS 20 21 22 23 #include <stdarg.h> 24 #include <string.h> 25 26 #include "windef.h" 27 #include "winbase.h" 28 #include "winuser.h" 29 #include "ole2.h" 30 #include "rpcproxy.h" 31 #include "inseng.h" 32 33 #include "inseng_private.h" 34 35 #include "wine/list.h" 36 #include "wine/debug.h" 37 38 WINE_DEFAULT_DEBUG_CHANNEL(inseng); 39 40 #define DEFAULT_INSTALLER_DESC "Active Setup Installation" 41 42 struct cifgroup 43 { 44 ICifGroup ICifGroup_iface; 45 46 struct list entry; 47 48 ICifFile *parent; 49 50 char *id; 51 char *description; 52 DWORD priority; 53 }; 54 55 struct ciffenum_components 56 { 57 IEnumCifComponents IEnumCifComponents_iface; 58 LONG ref; 59 60 ICifFile *file; 61 struct list *start; 62 struct list *position; 63 64 char *group_id; 65 }; 66 67 struct ciffenum_groups 68 { 69 IEnumCifGroups IEnumCifGroups_iface; 70 LONG ref; 71 72 ICifFile *file; 73 struct list *start; 74 struct list *position; 75 }; 76 77 struct url_info 78 { 79 struct list entry; 80 INT index; 81 char *url; 82 DWORD flags; 83 }; 84 85 struct dependency_info 86 { 87 struct list entry; 88 char *id; 89 char *type; 90 }; 91 92 struct cifcomponent 93 { 94 ICifComponent ICifComponent_iface; 95 96 struct list entry; 97 98 ICifFile *parent; 99 100 char *id; 101 char *guid; 102 char *description; 103 char *details; 104 char *group; 105 106 107 DWORD version; 108 DWORD build; 109 char *patchid; 110 111 char *locale; 112 char *key_uninstall; 113 114 DWORD size_win; 115 DWORD size_app; 116 DWORD size_download; 117 DWORD size_extracted; 118 119 char *key_success; 120 char *key_progress; 121 char *key_cancel; 122 123 DWORD as_aware; 124 DWORD reboot; 125 DWORD admin; 126 DWORD visibleui; 127 128 DWORD priority; 129 DWORD platform; 130 131 struct list dependencies; 132 struct list urls; 133 134 /* mode */ 135 /* det version */ 136 /* one component */ 137 /* custom data */ 138 139 /* in memory state */ 140 DWORD queue_state; 141 DWORD current_priority; 142 DWORD size_actual_download; 143 BOOL downloaded; 144 BOOL installed; 145 }; 146 147 struct ciffile 148 { 149 ICifFile ICifFile_iface; 150 LONG ref; 151 152 struct list components; 153 struct list groups; 154 155 char *name; 156 }; 157 158 static inline struct ciffile *impl_from_ICiffile(ICifFile *iface) 159 { 160 return CONTAINING_RECORD(iface, struct ciffile, ICifFile_iface); 161 } 162 163 static inline struct cifcomponent *impl_from_ICifComponent(ICifComponent *iface) 164 { 165 return CONTAINING_RECORD(iface, struct cifcomponent, ICifComponent_iface); 166 } 167 168 static inline struct cifgroup *impl_from_ICifGroup(ICifGroup *iface) 169 { 170 return CONTAINING_RECORD(iface, struct cifgroup, ICifGroup_iface); 171 } 172 173 static inline struct ciffenum_components *impl_from_IEnumCifComponents(IEnumCifComponents *iface) 174 { 175 return CONTAINING_RECORD(iface, struct ciffenum_components, IEnumCifComponents_iface); 176 } 177 178 static inline struct ciffenum_groups *impl_from_IEnumCifGroups(IEnumCifGroups *iface) 179 { 180 return CONTAINING_RECORD(iface, struct ciffenum_groups, IEnumCifGroups_iface); 181 } 182 183 static HRESULT enum_components_create(ICifFile *file, struct list *start, char *group_id, IEnumCifComponents **iface); 184 185 static HRESULT copy_substring_null(char *dest, int max_len, char *src) 186 { 187 if (!src) 188 return E_FAIL; 189 190 if (max_len <= 0) 191 return S_OK; 192 193 if (!dest) 194 return E_FAIL; 195 196 while (*src && max_len-- > 1) 197 *dest++ = *src++; 198 *dest = 0; 199 200 return S_OK; 201 } 202 203 static void url_entry_free(struct url_info *url) 204 { 205 heap_free(url->url); 206 heap_free(url); 207 } 208 209 static void dependency_entry_free(struct dependency_info *dependency) 210 { 211 heap_free(dependency->id); 212 heap_free(dependency); 213 } 214 215 static void component_free(struct cifcomponent *comp) 216 { 217 struct dependency_info *dependency, *dependency_next; 218 struct url_info *url, *url_next; 219 220 heap_free(comp->id); 221 heap_free(comp->guid); 222 heap_free(comp->description); 223 heap_free(comp->details); 224 heap_free(comp->group); 225 226 heap_free(comp->patchid); 227 228 heap_free(comp->locale); 229 heap_free(comp->key_uninstall); 230 231 heap_free(comp->key_success); 232 heap_free(comp->key_progress); 233 heap_free(comp->key_cancel); 234 235 LIST_FOR_EACH_ENTRY_SAFE(dependency, dependency_next, &comp->dependencies, struct dependency_info, entry) 236 { 237 list_remove(&dependency->entry); 238 dependency_entry_free(dependency); 239 } 240 241 LIST_FOR_EACH_ENTRY_SAFE(url, url_next, &comp->urls, struct url_info, entry) 242 { 243 list_remove(&url->entry); 244 url_entry_free(url); 245 } 246 247 heap_free(comp); 248 } 249 250 static void group_free(struct cifgroup *group) 251 { 252 heap_free(group->id); 253 heap_free(group->description); 254 heap_free(group); 255 } 256 257 static HRESULT WINAPI group_GetID(ICifGroup *iface, char *id, DWORD size) 258 { 259 struct cifgroup *This = impl_from_ICifGroup(iface); 260 261 TRACE("(%p)->(%p, %u)\n", This, id, size); 262 263 return copy_substring_null(id, size, This->id); 264 } 265 266 static HRESULT WINAPI group_GetDescription(ICifGroup *iface, char *desc, DWORD size) 267 { 268 struct cifgroup *This = impl_from_ICifGroup(iface); 269 270 TRACE("(%p)->(%p, %u)\n", This, desc, size); 271 272 return copy_substring_null(desc, size, This->description); 273 } 274 275 static DWORD WINAPI group_GetPriority(ICifGroup *iface) 276 { 277 struct cifgroup *This = impl_from_ICifGroup(iface); 278 279 TRACE("(%p)\n", This); 280 281 return This->priority; 282 } 283 284 static HRESULT WINAPI group_EnumComponents(ICifGroup *iface, IEnumCifComponents **enum_components, DWORD filter, LPVOID pv) 285 { 286 struct cifgroup *This = impl_from_ICifGroup(iface); 287 struct ciffile *file; 288 289 TRACE("(%p)->(%p, %u, %p)\n", This, enum_components, filter, pv); 290 291 if (filter) 292 FIXME("filter (%x) not supported\n", filter); 293 if (pv) 294 FIXME("how to handle pv (%p)?\n", pv); 295 296 file = impl_from_ICiffile(This->parent); 297 return enum_components_create(This->parent, &file->components, This->id, enum_components); 298 } 299 300 static DWORD WINAPI group_GetCurrentPriority(ICifGroup *iface) 301 { 302 struct cifgroup *This = impl_from_ICifGroup(iface); 303 304 FIXME("(%p): stub\n", This); 305 306 return 0; 307 } 308 309 static const ICifGroupVtbl cifgroupVtbl = 310 { 311 group_GetID, 312 group_GetDescription, 313 group_GetPriority, 314 group_EnumComponents, 315 group_GetCurrentPriority, 316 }; 317 318 void component_set_actual_download_size(ICifComponent *iface, DWORD size) 319 { 320 struct cifcomponent *This = impl_from_ICifComponent(iface); 321 322 This->size_actual_download = size; 323 } 324 325 void component_set_downloaded(ICifComponent *iface, BOOL value) 326 { 327 struct cifcomponent *This = impl_from_ICifComponent(iface); 328 329 This->downloaded = value; 330 } 331 332 void component_set_installed(ICifComponent *iface, BOOL value) 333 { 334 struct cifcomponent *This = impl_from_ICifComponent(iface); 335 336 This->installed = value; 337 } 338 339 char *component_get_id(ICifComponent *iface) 340 { 341 struct cifcomponent *This = impl_from_ICifComponent(iface); 342 343 return This->id; 344 } 345 346 static HRESULT WINAPI component_GetID(ICifComponent *iface, char *id, DWORD size) 347 { 348 struct cifcomponent *This = impl_from_ICifComponent(iface); 349 350 TRACE("(%p)->(%p, %u)\n", This, id, size); 351 352 return copy_substring_null(id, size, This->id); 353 } 354 355 static HRESULT WINAPI component_GetGUID(ICifComponent *iface, char *guid, DWORD size) 356 { 357 struct cifcomponent *This = impl_from_ICifComponent(iface); 358 359 TRACE("(%p)->(%p, %u)\n", This, guid, size); 360 361 return copy_substring_null(guid, size, This->guid); 362 } 363 364 static HRESULT WINAPI component_GetDescription(ICifComponent *iface, char *desc, DWORD size) 365 { 366 struct cifcomponent *This = impl_from_ICifComponent(iface); 367 368 TRACE("(%p)->(%p, %u)\n", This, desc, size); 369 370 return copy_substring_null(desc, size, This->description); 371 } 372 373 static HRESULT WINAPI component_GetDetails(ICifComponent *iface, char *details, DWORD size) 374 { 375 struct cifcomponent *This = impl_from_ICifComponent(iface); 376 377 TRACE("(%p)->(%p, %u)\n", This, details, size); 378 379 return copy_substring_null(details, size, This->details); 380 } 381 382 static HRESULT WINAPI component_GetUrl(ICifComponent *iface, UINT index, char *url, DWORD size, DWORD *flags) 383 { 384 struct cifcomponent *This = impl_from_ICifComponent(iface); 385 struct url_info *entry; 386 387 TRACE("(%p)->(%u, %p, %u, %p)\n", This, index, url, size, flags); 388 389 /* FIXME: check how functions behaves for url == NULL */ 390 391 if (!flags) 392 return E_FAIL; 393 394 LIST_FOR_EACH_ENTRY(entry, &This->urls, struct url_info, entry) 395 { 396 if (entry->index != index) 397 continue; 398 399 *flags = entry->flags; 400 return copy_substring_null(url, size, entry->url); 401 } 402 403 return E_FAIL; 404 } 405 406 static HRESULT WINAPI component_GetFileExtractList(ICifComponent *iface, UINT index, char *list, DWORD size) 407 { 408 struct cifcomponent *This = impl_from_ICifComponent(iface); 409 410 FIXME("(%p)->(%u, %p, %u): stub\n", This, index, list, size); 411 412 return E_NOTIMPL; 413 } 414 415 static HRESULT WINAPI component_GetUrlCheckRange(ICifComponent *iface, UINT index, DWORD *min, DWORD *max) 416 { 417 struct cifcomponent *This = impl_from_ICifComponent(iface); 418 419 FIXME("(%p)->(%u, %p, %p): stub\n", This, index, min, max); 420 421 return E_NOTIMPL; 422 } 423 424 static HRESULT WINAPI component_GetCommand(ICifComponent *iface, UINT index, char *cmd, DWORD cmd_size, char *switches, DWORD switch_size, DWORD *type) 425 { 426 struct cifcomponent *This = impl_from_ICifComponent(iface); 427 428 FIXME("(%p)->(%u, %p, %u, %p, %u, %p): stub\n", This, index, cmd, cmd_size, switches, switch_size, type); 429 430 return E_NOTIMPL; 431 } 432 433 static HRESULT WINAPI component_GetVersion(ICifComponent *iface, DWORD *version, DWORD *build) 434 { 435 struct cifcomponent *This = impl_from_ICifComponent(iface); 436 437 TRACE("(%p)->(%p, %p)\n", This, version, build); 438 439 if (!version || !build) 440 return E_FAIL; 441 442 *version = This->version; 443 *build = This->build; 444 445 return S_OK; 446 } 447 448 static HRESULT WINAPI component_GetLocale(ICifComponent *iface, char *locale, DWORD size) 449 { 450 struct cifcomponent *This = impl_from_ICifComponent(iface); 451 452 TRACE("(%p)->(%p, %u)\n", This, locale, size); 453 454 return copy_substring_null(locale, size, This->locale); 455 } 456 457 static HRESULT WINAPI component_GetUninstallKey(ICifComponent *iface, char *key, DWORD size) 458 { 459 struct cifcomponent *This = impl_from_ICifComponent(iface); 460 461 TRACE("(%p)->(%p, %u)\n", This, key, size); 462 463 return copy_substring_null(key, size, This->key_uninstall); 464 } 465 466 static HRESULT WINAPI component_GetInstalledSize(ICifComponent *iface, DWORD *win, DWORD *app) 467 { 468 struct cifcomponent *This = impl_from_ICifComponent(iface); 469 470 TRACE("(%p)->(%p, %p)\n", This, win, app); 471 472 if (!win || !app) 473 return E_FAIL; 474 475 *win = This->size_win; 476 *app = This->size_app; 477 478 return S_OK; 479 } 480 481 static DWORD WINAPI component_GetDownloadSize(ICifComponent *iface) 482 { 483 struct cifcomponent *This = impl_from_ICifComponent(iface); 484 485 TRACE("(%p)\n", This); 486 487 return This->size_download; 488 } 489 490 static DWORD WINAPI component_GetExtractSize(ICifComponent *iface) 491 { 492 struct cifcomponent *This = impl_from_ICifComponent(iface); 493 494 TRACE("(%p)\n", This); 495 496 return This->size_extracted; 497 } 498 499 static HRESULT WINAPI component_GetSuccessKey(ICifComponent *iface, char *key, DWORD size) 500 { 501 struct cifcomponent *This = impl_from_ICifComponent(iface); 502 503 TRACE("(%p)->(%p, %u)\n", This, key, size); 504 505 return copy_substring_null(key, size, This->key_success); 506 } 507 508 static HRESULT WINAPI component_GetProgressKeys(ICifComponent *iface, char *progress, DWORD progress_size, 509 char *cancel, DWORD cancel_size) 510 { 511 struct cifcomponent *This = impl_from_ICifComponent(iface); 512 HRESULT hr; 513 514 TRACE("(%p)->(%p, %u, %p, %u): semi-stub\n", This, progress, progress_size, cancel, cancel_size); 515 516 hr = copy_substring_null(progress, progress_size, This->key_progress); 517 if (hr != S_OK) return hr; 518 519 if (cancel_size > 0 && cancel) 520 *cancel = 0; 521 522 return S_OK; 523 } 524 525 static HRESULT WINAPI component_IsActiveSetupAware(ICifComponent *iface) 526 { 527 struct cifcomponent *This = impl_from_ICifComponent(iface); 528 529 TRACE("(%p)\n", This); 530 531 return This->as_aware ? S_OK : S_FALSE; 532 } 533 534 static HRESULT WINAPI component_IsRebootRequired(ICifComponent *iface) 535 { 536 struct cifcomponent *This = impl_from_ICifComponent(iface); 537 538 TRACE("(%p)\n", This); 539 540 return This->reboot ? S_OK : S_FALSE; 541 } 542 543 static HRESULT WINAPI component_RequiresAdminRights(ICifComponent *iface) 544 { 545 struct cifcomponent *This = impl_from_ICifComponent(iface); 546 547 TRACE("(%p)\n", This); 548 549 return This->admin ? S_OK : S_FALSE; 550 } 551 552 static DWORD WINAPI component_GetPriority(ICifComponent *iface) 553 { 554 struct cifcomponent *This = impl_from_ICifComponent(iface); 555 556 TRACE("(%p)\n", This); 557 558 return This->priority; 559 } 560 561 static HRESULT WINAPI component_GetDependency(ICifComponent *iface, UINT index, char *id, DWORD id_size, char *type, DWORD *ver, DWORD *build) 562 { 563 struct cifcomponent *This = impl_from_ICifComponent(iface); 564 struct dependency_info *entry; 565 ICifComponent *dependency; 566 int pos = 0; 567 568 TRACE("(%p)->(%u, %p, %u, %p, %p, %p)\n", This, index, id, id_size, type, ver, build); 569 570 if (!id || !ver || !build) 571 return E_FAIL; 572 573 LIST_FOR_EACH_ENTRY(entry, &This->dependencies, struct dependency_info, entry) 574 { 575 if (pos++ < index) 576 continue; 577 578 if (ICifFile_FindComponent(This->parent, entry->id, &dependency) == S_OK) 579 { 580 ICifComponent_GetVersion(dependency, ver, build); 581 } 582 else 583 { 584 *ver = -1; 585 *build = -1; 586 } 587 588 if (entry->type) 589 *type = *entry->type; 590 else 591 *type = 'I'; 592 593 return copy_substring_null(id, id_size, entry->id); 594 } 595 596 return E_FAIL; 597 } 598 599 static DWORD WINAPI component_GetPlatform(ICifComponent *iface) 600 { 601 struct cifcomponent *This = impl_from_ICifComponent(iface); 602 603 TRACE("(%p)\n", This); 604 605 return This->platform; 606 } 607 608 static HRESULT WINAPI component_GetMode(ICifComponent *iface, UINT index, char *mode, DWORD size) 609 { 610 struct cifcomponent *This = impl_from_ICifComponent(iface); 611 612 FIXME("(%p)->(%u, %p, %u): stub\n", This, index, mode, size); 613 614 return E_NOTIMPL; 615 } 616 617 static HRESULT WINAPI component_GetGroup(ICifComponent *iface, char *id, DWORD size) 618 { 619 struct cifcomponent *This = impl_from_ICifComponent(iface); 620 621 TRACE("(%p)->(%p, %u)\n", This, id, size); 622 623 return copy_substring_null(id, size, This->group); 624 } 625 626 static HRESULT WINAPI component_IsUIVisible(ICifComponent *iface) 627 { 628 struct cifcomponent *This = impl_from_ICifComponent(iface); 629 630 TRACE("(%p)\n", This); 631 632 return This->visibleui ? S_OK : S_FALSE; 633 } 634 635 static HRESULT WINAPI component_GetPatchID(ICifComponent *iface, char *id, DWORD size) 636 { 637 struct cifcomponent *This = impl_from_ICifComponent(iface); 638 639 TRACE("(%p)->(%p, %u)\n", This, id, size); 640 641 return copy_substring_null(id, size, This->patchid); 642 } 643 644 static HRESULT WINAPI component_GetDetVersion(ICifComponent *iface, char *dll, DWORD dll_size, char *entry, DWORD entry_size) 645 { 646 struct cifcomponent *This = impl_from_ICifComponent(iface); 647 648 FIXME("(%p)->(%p, %u, %p, %u): stub\n", This, dll, dll_size, entry, entry_size); 649 650 return E_NOTIMPL; 651 } 652 653 static HRESULT WINAPI component_GetTreatAsOneComponents(ICifComponent *iface, UINT index, char *id, DWORD size) 654 { 655 struct cifcomponent *This = impl_from_ICifComponent(iface); 656 657 FIXME("(%p)->(%u, %p, %u): stub\n", This, index, id, size); 658 659 return E_NOTIMPL; 660 } 661 662 static HRESULT WINAPI component_GetCustomData(ICifComponent *iface, char *key, char *data, DWORD size) 663 { 664 struct cifcomponent *This = impl_from_ICifComponent(iface); 665 666 FIXME("(%p)->(%s, %p, %u): stub\n", This, debugstr_a(key), data, size); 667 668 return E_NOTIMPL; 669 } 670 671 static DWORD WINAPI component_IsComponentInstalled(ICifComponent *iface) 672 { 673 struct cifcomponent *This = impl_from_ICifComponent(iface); 674 675 TRACE("(%p)\n", This); 676 677 return This->installed; 678 } 679 680 static HRESULT WINAPI component_IsComponentDownloaded(ICifComponent *iface) 681 { 682 struct cifcomponent *This = impl_from_ICifComponent(iface); 683 684 TRACE("(%p)\n", This); 685 686 return This->downloaded ? S_OK : S_FALSE; 687 } 688 689 static DWORD WINAPI component_IsThisVersionInstalled(ICifComponent *iface, DWORD version, DWORD build, DWORD *ret_version, DWORD *ret_build) 690 { 691 struct cifcomponent *This = impl_from_ICifComponent(iface); 692 693 FIXME("(%p)->(%u, %u, %p, %p): stub\n", This, version, build, ret_version, ret_build); 694 695 return 0; 696 } 697 698 static DWORD WINAPI component_GetInstallQueueState(ICifComponent *iface) 699 { 700 struct cifcomponent *This = impl_from_ICifComponent(iface); 701 702 TRACE("(%p)\n", This); 703 704 return This->queue_state; 705 } 706 707 static HRESULT WINAPI component_SetInstallQueueState(ICifComponent *iface, DWORD state) 708 { 709 struct cifcomponent *This = impl_from_ICifComponent(iface); 710 711 TRACE("(%p)->(%u)\n", This, state); 712 713 This->queue_state = state; 714 return S_OK; 715 } 716 717 static DWORD WINAPI component_GetActualDownloadSize(ICifComponent *iface) 718 { 719 struct cifcomponent *This = impl_from_ICifComponent(iface); 720 721 TRACE("(%p)\n", This); 722 723 return This->size_download; 724 } 725 726 static DWORD WINAPI component_GetCurrentPriority(ICifComponent *iface) 727 { 728 struct cifcomponent *This = impl_from_ICifComponent(iface); 729 730 TRACE("(%p)\n", This); 731 732 return This->current_priority; 733 } 734 735 736 static HRESULT WINAPI component_SetCurrentPriority(ICifComponent *iface, DWORD priority) 737 { 738 struct cifcomponent *This = impl_from_ICifComponent(iface); 739 740 TRACE("(%p)->(%u)\n", This, priority); 741 742 This->current_priority = priority; 743 return S_OK; 744 } 745 746 static const ICifComponentVtbl cifcomponentVtbl = 747 { 748 component_GetID, 749 component_GetGUID, 750 component_GetDescription, 751 component_GetDetails, 752 component_GetUrl, 753 component_GetFileExtractList, 754 component_GetUrlCheckRange, 755 component_GetCommand, 756 component_GetVersion, 757 component_GetLocale, 758 component_GetUninstallKey, 759 component_GetInstalledSize, 760 component_GetDownloadSize, 761 component_GetExtractSize, 762 component_GetSuccessKey, 763 component_GetProgressKeys, 764 component_IsActiveSetupAware, 765 component_IsRebootRequired, 766 component_RequiresAdminRights, 767 component_GetPriority, 768 component_GetDependency, 769 component_GetPlatform, 770 component_GetMode, 771 component_GetGroup, 772 component_IsUIVisible, 773 component_GetPatchID, 774 component_GetDetVersion, 775 component_GetTreatAsOneComponents, 776 component_GetCustomData, 777 component_IsComponentInstalled, 778 component_IsComponentDownloaded, 779 component_IsThisVersionInstalled, 780 component_GetInstallQueueState, 781 component_SetInstallQueueState, 782 component_GetActualDownloadSize, 783 component_GetCurrentPriority, 784 component_SetCurrentPriority, 785 }; 786 787 static HRESULT WINAPI enum_components_QueryInterface(IEnumCifComponents *iface, REFIID riid, void **ppv) 788 { 789 struct ciffenum_components *This = impl_from_IEnumCifComponents(iface); 790 791 if (IsEqualGUID(&IID_IUnknown, riid)) 792 { 793 TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv); 794 *ppv = &This->IEnumCifComponents_iface; 795 } 796 /* 797 else if (IsEqualGUID(&IID_IEnumCifComponents, riid)) 798 { 799 TRACE("(%p)->(IID_ICifFile %p)\n", This, ppv); 800 *ppv = &This->IEnumCifComponents_iface; 801 } 802 */ 803 else 804 { 805 FIXME("(%p)->(%s %p) not found\n", This, debugstr_guid(riid), ppv); 806 *ppv = NULL; 807 return E_NOINTERFACE; 808 } 809 810 IUnknown_AddRef((IUnknown *)*ppv); 811 return S_OK; 812 } 813 814 static ULONG WINAPI enum_components_AddRef(IEnumCifComponents *iface) 815 { 816 struct ciffenum_components *This = impl_from_IEnumCifComponents(iface); 817 LONG ref = InterlockedIncrement(&This->ref); 818 819 TRACE("(%p) ref=%d\n", This, ref); 820 821 return ref; 822 } 823 824 static ULONG WINAPI enum_components_Release(IEnumCifComponents *iface) 825 { 826 struct ciffenum_components *This = impl_from_IEnumCifComponents(iface); 827 LONG ref = InterlockedDecrement(&This->ref); 828 829 TRACE("(%p) ref=%d\n", This, ref); 830 831 if(!ref) 832 { 833 ICifFile_Release(This->file); 834 heap_free(This); 835 } 836 837 return ref; 838 } 839 840 static HRESULT WINAPI enum_components_Next(IEnumCifComponents *iface, ICifComponent **component) 841 { 842 struct ciffenum_components *This = impl_from_IEnumCifComponents(iface); 843 struct cifcomponent *comp; 844 845 TRACE("(%p)->(%p)\n", This, component); 846 847 if (!component) 848 return E_FAIL; 849 850 if (!This->position) 851 { 852 *component = NULL; 853 return E_FAIL; 854 } 855 856 do 857 { 858 This->position = list_next(This->start, This->position); 859 if (!This->position) 860 { 861 *component = NULL; 862 return E_FAIL; 863 } 864 865 comp = CONTAINING_RECORD(This->position, struct cifcomponent, entry); 866 } while (This->group_id && (!comp->group || strcmp(This->group_id, comp->group))); 867 868 *component = &comp->ICifComponent_iface; 869 return S_OK; 870 } 871 872 static HRESULT WINAPI enum_components_Reset(IEnumCifComponents *iface) 873 { 874 struct ciffenum_components *This = impl_from_IEnumCifComponents(iface); 875 876 TRACE("(%p)\n", This); 877 878 This->position = This->start; 879 return S_OK; 880 } 881 882 static const IEnumCifComponentsVtbl enum_componentsVtbl = 883 { 884 enum_components_QueryInterface, 885 enum_components_AddRef, 886 enum_components_Release, 887 enum_components_Next, 888 enum_components_Reset, 889 }; 890 891 static HRESULT enum_components_create(ICifFile *file, struct list *start, char *group_id, IEnumCifComponents **iface) 892 { 893 struct ciffenum_components *enumerator; 894 895 enumerator = heap_alloc_zero(sizeof(*enumerator)); 896 if (!enumerator) return E_OUTOFMEMORY; 897 898 enumerator->IEnumCifComponents_iface.lpVtbl = &enum_componentsVtbl; 899 enumerator->ref = 1; 900 enumerator->file = file; 901 enumerator->start = start; 902 enumerator->position = start; 903 enumerator->group_id = group_id; 904 905 ICifFile_AddRef(file); 906 907 *iface = &enumerator->IEnumCifComponents_iface; 908 return S_OK; 909 } 910 911 static HRESULT WINAPI enum_groups_QueryInterface(IEnumCifGroups *iface, REFIID riid, void **ppv) 912 { 913 struct ciffenum_groups *This = impl_from_IEnumCifGroups(iface); 914 915 if (IsEqualGUID(&IID_IUnknown, riid)) 916 { 917 TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv); 918 *ppv = &This->IEnumCifGroups_iface; 919 } 920 /* 921 else if (IsEqualGUID(&IID_IEnumCifGroups, riid)) 922 { 923 TRACE("(%p)->(IID_ICifFile %p)\n", This, ppv); 924 *ppv = &This->IEnumCifGroups_iface; 925 } 926 */ 927 else 928 { 929 FIXME("(%p)->(%s %p) not found\n", This, debugstr_guid(riid), ppv); 930 *ppv = NULL; 931 return E_NOINTERFACE; 932 } 933 934 IUnknown_AddRef((IUnknown *)*ppv); 935 return S_OK; 936 } 937 938 static ULONG WINAPI enum_groups_AddRef(IEnumCifGroups *iface) 939 { 940 struct ciffenum_groups *This = impl_from_IEnumCifGroups(iface); 941 LONG ref = InterlockedIncrement(&This->ref); 942 943 TRACE("(%p) ref=%d\n", This, ref); 944 945 return ref; 946 } 947 948 static ULONG WINAPI enum_groups_Release(IEnumCifGroups *iface) 949 { 950 struct ciffenum_groups *This = impl_from_IEnumCifGroups(iface); 951 LONG ref = InterlockedDecrement(&This->ref); 952 953 TRACE("(%p) ref=%d\n", This, ref); 954 955 if(!ref) 956 { 957 ICifFile_Release(This->file); 958 heap_free(This); 959 } 960 961 return ref; 962 } 963 964 static HRESULT WINAPI enum_groups_Next(IEnumCifGroups *iface, ICifGroup **group) 965 { 966 struct ciffenum_groups *This = impl_from_IEnumCifGroups(iface); 967 struct cifgroup *gp; 968 969 TRACE("(%p)->(%p)\n", This, group); 970 971 if (!This->position || !group) 972 return E_FAIL; 973 974 This->position = list_next(This->start, This->position); 975 976 if (!This->position) 977 return E_FAIL; 978 979 gp = CONTAINING_RECORD(This->position, struct cifgroup, entry); 980 *group = &gp->ICifGroup_iface; 981 return S_OK; 982 } 983 984 static HRESULT WINAPI enum_groups_Reset(IEnumCifGroups *iface) 985 { 986 struct ciffenum_groups *This = impl_from_IEnumCifGroups(iface); 987 988 TRACE("(%p)\n", This); 989 990 This->position = This->start; 991 return S_OK; 992 } 993 994 static const IEnumCifGroupsVtbl enum_groupsVtbl = 995 { 996 enum_groups_QueryInterface, 997 enum_groups_AddRef, 998 enum_groups_Release, 999 enum_groups_Next, 1000 enum_groups_Reset, 1001 }; 1002 1003 static HRESULT enum_groups_create(ICifFile *file, struct list *start, IEnumCifGroups **iface) 1004 { 1005 struct ciffenum_groups *enumerator; 1006 1007 enumerator = heap_alloc_zero(sizeof(*enumerator)); 1008 if (!enumerator) return E_OUTOFMEMORY; 1009 1010 enumerator->IEnumCifGroups_iface.lpVtbl = &enum_groupsVtbl; 1011 enumerator->ref = 1; 1012 enumerator->file = file; 1013 enumerator->start = start; 1014 enumerator->position = start; 1015 1016 ICifFile_AddRef(file); 1017 1018 *iface = &enumerator->IEnumCifGroups_iface; 1019 return S_OK; 1020 } 1021 1022 static HRESULT WINAPI ciffile_QueryInterface(ICifFile *iface, REFIID riid, void **ppv) 1023 { 1024 struct ciffile *This = impl_from_ICiffile(iface); 1025 1026 if (IsEqualGUID(&IID_IUnknown, riid)) 1027 { 1028 TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv); 1029 *ppv = &This->ICifFile_iface; 1030 } 1031 else if (IsEqualGUID(&IID_ICifFile, riid)) 1032 { 1033 TRACE("(%p)->(IID_ICifFile %p)\n", This, ppv); 1034 *ppv = &This->ICifFile_iface; 1035 } 1036 else 1037 { 1038 FIXME("(%p)->(%s %p) not found\n", This, debugstr_guid(riid), ppv); 1039 *ppv = NULL; 1040 return E_NOINTERFACE; 1041 } 1042 1043 IUnknown_AddRef((IUnknown *)*ppv); 1044 return S_OK; 1045 } 1046 1047 static ULONG WINAPI ciffile_AddRef(ICifFile *iface) 1048 { 1049 struct ciffile *This = impl_from_ICiffile(iface); 1050 LONG ref = InterlockedIncrement(&This->ref); 1051 1052 TRACE("(%p) ref=%d\n", This, ref); 1053 1054 return ref; 1055 } 1056 1057 static ULONG WINAPI ciffile_Release(ICifFile *iface) 1058 { 1059 struct ciffile *This = impl_from_ICiffile(iface); 1060 LONG ref = InterlockedDecrement(&This->ref); 1061 1062 TRACE("(%p) ref=%d\n", This, ref); 1063 1064 if(!ref) 1065 { 1066 struct cifcomponent *comp, *comp_next; 1067 struct cifgroup *group, *group_next; 1068 1069 heap_free(This->name); 1070 1071 LIST_FOR_EACH_ENTRY_SAFE(comp, comp_next, &This->components, struct cifcomponent, entry) 1072 { 1073 list_remove(&comp->entry); 1074 component_free(comp); 1075 } 1076 1077 LIST_FOR_EACH_ENTRY_SAFE(group, group_next, &This->groups, struct cifgroup, entry) 1078 { 1079 list_remove(&group->entry); 1080 group_free(group); 1081 } 1082 1083 heap_free(This); 1084 } 1085 1086 return ref; 1087 } 1088 1089 static HRESULT WINAPI ciffile_EnumComponents(ICifFile *iface, IEnumCifComponents **enum_components, DWORD filter, void *pv) 1090 { 1091 struct ciffile *This = impl_from_ICiffile(iface); 1092 1093 TRACE("(%p)->(%p, %u, %p)\n", This, enum_components, filter, pv); 1094 1095 if (filter) 1096 FIXME("filter (%x) not supported\n", filter); 1097 if (pv) 1098 FIXME("how to handle pv (%p)?\n", pv); 1099 1100 return enum_components_create(iface, &This->components, NULL, enum_components); 1101 } 1102 1103 static HRESULT WINAPI ciffile_FindComponent(ICifFile *iface, const char *id, ICifComponent **component) 1104 { 1105 struct ciffile *This = impl_from_ICiffile(iface); 1106 struct cifcomponent *comp; 1107 1108 TRACE("(%p)->(%s, %p)\n", This, debugstr_a(id), component); 1109 1110 LIST_FOR_EACH_ENTRY(comp, &This->components, struct cifcomponent, entry) 1111 { 1112 if (strcmp(comp->id, id) != 0) 1113 continue; 1114 1115 *component = &comp->ICifComponent_iface; 1116 return S_OK; 1117 } 1118 1119 return E_FAIL; 1120 } 1121 1122 static HRESULT WINAPI ciffile_EnumGroups(ICifFile *iface, IEnumCifGroups **enum_groups, DWORD filter, void *pv) 1123 { 1124 struct ciffile *This = impl_from_ICiffile(iface); 1125 1126 TRACE("(%p)->(%p, %u, %p)\n", This, enum_groups, filter, pv); 1127 1128 if (filter) 1129 FIXME("filter (%x) not supported\n", filter); 1130 if (pv) 1131 FIXME("how to handle pv (%p)?\n", pv); 1132 1133 return enum_groups_create(iface, &This->groups, enum_groups); 1134 } 1135 1136 static HRESULT WINAPI ciffile_FindGroup(ICifFile *iface, const char *id, ICifGroup **group) 1137 { 1138 struct ciffile *This = impl_from_ICiffile(iface); 1139 struct cifgroup *gp; 1140 1141 TRACE("(%p)->(%s, %p)\n", This, debugstr_a(id), group); 1142 1143 LIST_FOR_EACH_ENTRY(gp, &This->groups, struct cifgroup, entry) 1144 { 1145 if (strcmp(gp->id, id) != 0) 1146 continue; 1147 1148 *group = &gp->ICifGroup_iface; 1149 return S_OK; 1150 } 1151 1152 return E_FAIL; 1153 } 1154 1155 static HRESULT WINAPI ciffile_EnumModes(ICifFile *iface, IEnumCifModes **cuf_modes, DWORD filter, void *pv) 1156 { 1157 struct ciffile *This = impl_from_ICiffile(iface); 1158 1159 FIXME("(%p)->(%p, %u, %p): stub\n", This, cuf_modes, filter, pv); 1160 1161 return E_NOTIMPL; 1162 } 1163 1164 static HRESULT WINAPI ciffile_FindMode(ICifFile *iface, const char *id, ICifMode **mode) 1165 { 1166 struct ciffile *This = impl_from_ICiffile(iface); 1167 1168 FIXME("(%p)->(%s, %p): stub\n", This, debugstr_a(id), mode); 1169 1170 return E_NOTIMPL; 1171 } 1172 1173 static HRESULT WINAPI ciffile_GetDescription(ICifFile *iface, char *desc, DWORD size) 1174 { 1175 struct ciffile *This = impl_from_ICiffile(iface); 1176 1177 TRACE("(%p)->(%p, %u)\n", This, desc, size); 1178 1179 return copy_substring_null(desc, size, This->name); 1180 } 1181 1182 static HRESULT WINAPI ciffile_GetDetDlls(ICifFile *iface, char *dlls, DWORD size) 1183 { 1184 struct ciffile *This = impl_from_ICiffile(iface); 1185 1186 FIXME("(%p)->(%p, %u): stub\n", This, dlls, size); 1187 1188 return E_NOTIMPL; 1189 } 1190 1191 static const ICifFileVtbl ciffileVtbl = 1192 { 1193 ciffile_QueryInterface, 1194 ciffile_AddRef, 1195 ciffile_Release, 1196 ciffile_EnumComponents, 1197 ciffile_FindComponent, 1198 ciffile_EnumGroups, 1199 ciffile_FindGroup, 1200 ciffile_EnumModes, 1201 ciffile_FindMode, 1202 ciffile_GetDescription, 1203 ciffile_GetDetDlls, 1204 }; 1205 1206 static BOOL copy_string(char **dest, const char *source) 1207 { 1208 if (!source) 1209 { 1210 *dest = NULL; 1211 return TRUE; 1212 } 1213 1214 *dest = strdupA(source); 1215 if (!dest) return FALSE; 1216 return TRUE; 1217 } 1218 1219 static BOOL section_get_str(struct inf_section *inf_sec, const char *key, char **value, const char *def) 1220 { 1221 struct inf_value *inf_val; 1222 1223 inf_val = inf_get_value(inf_sec, key); 1224 if (!inf_val) return copy_string(value, def); 1225 1226 *value = inf_value_get_value(inf_val); 1227 if (!*value) return FALSE; 1228 1229 return TRUE; 1230 } 1231 1232 static char *next_part(char **str, BOOL strip_quotes) 1233 { 1234 char *start = *str; 1235 char *next = *str; 1236 1237 while (*next && *next != ',') 1238 next++; 1239 1240 if (!*next) 1241 { 1242 *str = trim(start, NULL, strip_quotes); 1243 return NULL; 1244 } 1245 1246 *next = 0; 1247 *str = trim(start, NULL, strip_quotes); 1248 return ++next; 1249 } 1250 1251 static BOOL value_get_str_field(struct inf_value *inf_val, int field, char **value, const char *def) 1252 { 1253 char *line, *str, *next; 1254 int i = 0; 1255 1256 line = inf_value_get_value(inf_val); 1257 if (!line) return FALSE; 1258 1259 str = line; 1260 do 1261 { 1262 i++; 1263 next = next_part(&str, TRUE); 1264 1265 if (field == i) 1266 { 1267 BOOL ret = copy_string(value, str); 1268 heap_free(line); 1269 return ret; 1270 } 1271 1272 str = next; 1273 } while (str); 1274 1275 return copy_string(value, def); 1276 } 1277 1278 /* 1279 static BOOL section_get_str_field(struct inf_section *inf_sec, const char *key, int field, char **value, const char *def) 1280 { 1281 struct inf_value *inf_val; 1282 1283 inf_val = inf_get_value(inf_sec, key); 1284 if (!inf_val) return copy_string(value, def); 1285 1286 return value_get_str_field(inf_val, field, value, def); 1287 } 1288 */ 1289 1290 static BOOL section_get_dword(struct inf_section *inf_sec, const char *key, DWORD *value, DWORD def) 1291 { 1292 struct inf_value *inf_val; 1293 char *str; 1294 1295 inf_val = inf_get_value(inf_sec, key); 1296 if (!inf_val) 1297 { 1298 *value = def; 1299 return TRUE; 1300 } 1301 1302 str = inf_value_get_value(inf_val); 1303 if (!str) return FALSE; 1304 1305 *value = atoi(str); 1306 heap_free(str); 1307 1308 return TRUE; 1309 } 1310 1311 static BOOL value_get_dword_field(struct inf_value *inf_val, int field, DWORD *value, DWORD def) 1312 { 1313 char *value_str; 1314 BOOL ret; 1315 1316 ret = value_get_str_field(inf_val, field, &value_str, NULL); 1317 if (!ret) return FALSE; 1318 if (!value_str) 1319 { 1320 *value = def; 1321 return TRUE; 1322 } 1323 1324 *value = atoi(value_str); 1325 heap_free(value_str); 1326 1327 return TRUE; 1328 } 1329 1330 static BOOL section_get_dword_field(struct inf_section *inf_sec, const char *key, int field, DWORD *value, DWORD def) 1331 { 1332 struct inf_value *inf_val; 1333 1334 inf_val = inf_get_value(inf_sec, key); 1335 if (!inf_val) 1336 { 1337 *value = def; 1338 return TRUE; 1339 } 1340 1341 return value_get_dword_field(inf_val, field, value, def); 1342 } 1343 1344 static HRESULT process_version(struct ciffile *file, struct inf_section *section) 1345 { 1346 if (!section_get_str(section, "DisplayName", &file->name, DEFAULT_INSTALLER_DESC)) 1347 return E_OUTOFMEMORY; 1348 1349 return S_OK; 1350 } 1351 1352 static BOOL read_version_entry(struct inf_section *section, DWORD *ret_ver, DWORD *ret_build) 1353 { 1354 DWORD version = 0; 1355 DWORD build = 0; 1356 char *line, *str, *next; 1357 1358 if (!section_get_str(section, "Version", &line, NULL)) 1359 return FALSE; 1360 if (!line) goto done; 1361 1362 str = line; 1363 1364 next = next_part(&str, TRUE); 1365 version |= atoi(str) << 16; 1366 if (!next) goto done; 1367 str = next; 1368 1369 next = next_part(&str, TRUE); 1370 version |= atoi(str) & 0xffff; 1371 if (!next) goto done; 1372 str = next; 1373 1374 next = next_part(&str, TRUE); 1375 build |= atoi(str) << 16; 1376 if (!next) goto done; 1377 str = next; 1378 1379 next_part(&str, TRUE); 1380 build |= atoi(str) & 0xffff; 1381 1382 done: 1383 heap_free(line); 1384 *ret_ver = version; 1385 *ret_build = build; 1386 return TRUE; 1387 } 1388 1389 static BOOL read_platform_entry(struct inf_section *section, DWORD *ret_platform) 1390 { 1391 DWORD platform = PLATFORM_ALL; 1392 char *line, *str, *next; 1393 1394 if (!section_get_str(section, "Platform", &line, NULL)) 1395 return FALSE; 1396 if (!line) goto done; 1397 1398 platform = 0; 1399 str = line; 1400 do 1401 { 1402 next = next_part(&str, TRUE); 1403 1404 if (strcasecmp(str, "Win95") == 0) 1405 platform |= PLATFORM_WIN98; 1406 else if (strcasecmp(str, "Win98") == 0) 1407 platform |= PLATFORM_WIN98; 1408 else if (strcasecmp(str, "NT4") == 0) 1409 platform |= PLATFORM_NT4; 1410 else if (strcasecmp(str, "NT5") == 0) 1411 platform |= PLATFORM_NT5; 1412 else if (strcasecmp(str, "NT4Alpha") == 0) 1413 platform |= PLATFORM_NT4; 1414 else if (strcasecmp(str, "NT5Alpha") == 0) 1415 platform |= PLATFORM_NT5; 1416 else if (strcasecmp(str, "Millen") == 0) 1417 platform |= PLATFORM_MILLEN; 1418 else 1419 FIXME("Unknown platform: %s\n", debugstr_a(str)); 1420 1421 str = next; 1422 } while (str); 1423 1424 done: 1425 heap_free(line); 1426 *ret_platform = platform; 1427 return TRUE; 1428 } 1429 1430 static BOOL read_dependencies(struct cifcomponent *component, struct inf_section *section) 1431 { 1432 struct dependency_info *dependency; 1433 char *line, *str, *next; 1434 BOOL ret = TRUE; 1435 1436 if (!section_get_str(section, "Dependencies", &line, NULL)) 1437 return E_OUTOFMEMORY; 1438 if (!line) goto done; 1439 1440 ret = FALSE; 1441 str = line; 1442 do 1443 { 1444 next = next_part(&str, TRUE); 1445 1446 dependency = heap_alloc_zero(sizeof(*dependency)); 1447 if (!dependency) goto done; 1448 1449 dependency->id = strdupA(str); 1450 if (!dependency->id) 1451 { 1452 heap_free(dependency); 1453 goto done; 1454 } 1455 1456 dependency->type = strstr(dependency->id, ":"); 1457 if (dependency->type) *dependency->type++ = 0; 1458 1459 list_add_tail(&component->dependencies, &dependency->entry); 1460 1461 str = next; 1462 } while (str); 1463 1464 ret = TRUE; 1465 1466 done: 1467 heap_free(line); 1468 return ret; 1469 } 1470 1471 static BOOL read_urls(struct cifcomponent *component, struct inf_section *section) 1472 { 1473 struct inf_value *inf_value = NULL; 1474 struct url_info *url_entry; 1475 char *str, *next; 1476 int index; 1477 1478 while (inf_section_next_value(section, &inf_value)) 1479 { 1480 str = inf_value_get_key(inf_value); 1481 if (!str) return E_OUTOFMEMORY; 1482 1483 if (strncasecmp(str, "URL", 3)) 1484 goto next; 1485 1486 if (!str[3]) 1487 goto next; 1488 1489 index = strtol(str+3, &next, 10); 1490 if (next == str+3 || *next != 0 || index < 1) 1491 goto next; 1492 index--; 1493 1494 url_entry = heap_alloc_zero(sizeof(*url_entry)); 1495 if (!url_entry) goto error; 1496 1497 url_entry->index = index; 1498 1499 if (!value_get_str_field(inf_value, 1, &url_entry->url, NULL)) 1500 goto error; 1501 if (!url_entry->url || !*url_entry->url) 1502 { 1503 url_entry_free(url_entry); 1504 goto next; 1505 } 1506 1507 if (!value_get_dword_field(inf_value, 2, &url_entry->flags, 0)) 1508 goto error; 1509 1510 list_add_tail(&component->urls, &url_entry->entry); 1511 1512 next: 1513 heap_free(str); 1514 } 1515 1516 return TRUE; 1517 1518 error: 1519 heap_free(str); 1520 url_entry_free(url_entry); 1521 return FALSE; 1522 }; 1523 1524 void add_component_by_priority(struct ciffile *file, struct cifcomponent *component) 1525 { 1526 struct cifcomponent *entry; 1527 1528 LIST_FOR_EACH_ENTRY(entry, &file->components, struct cifcomponent, entry) 1529 { 1530 if (entry->priority > component->priority) 1531 continue; 1532 1533 list_add_before(&entry->entry, &component->entry); 1534 return; 1535 } 1536 1537 list_add_tail(&file->components, &component->entry); 1538 } 1539 1540 static HRESULT process_component(struct ciffile *file, struct inf_section *section, const char *section_name) 1541 { 1542 struct cifcomponent *component; 1543 HRESULT hr = E_OUTOFMEMORY; 1544 1545 component = heap_alloc_zero(sizeof(*component)); 1546 if (!component) return E_OUTOFMEMORY; 1547 1548 component->ICifComponent_iface.lpVtbl = &cifcomponentVtbl; 1549 component->parent = &file->ICifFile_iface; 1550 1551 list_init(&component->urls); 1552 list_init(&component->dependencies); 1553 1554 component->queue_state = ActionNone; 1555 1556 component->id = strdupA(section_name); 1557 if (!component->id) goto error; 1558 1559 if (!section_get_str(section, "DisplayName", &component->description, NULL)) 1560 goto error; 1561 if (!section_get_str(section, "GUID", &component->guid, NULL)) 1562 goto error; 1563 if (!section_get_str(section, "Details", &component->details, NULL)) 1564 goto error; 1565 if (!section_get_str(section, "Group", &component->group, NULL)) 1566 goto error; 1567 if (!section_get_str(section, "Locale", &component->locale, "en")) 1568 goto error; 1569 if (!section_get_str(section, "PatchID", &component->patchid, NULL)) 1570 goto error; 1571 1572 if (!section_get_dword_field(section, "Size", 1, &component->size_download, 0)) 1573 goto error; 1574 if (!section_get_dword_field(section, "Size", 2, &component->size_extracted, 0)) 1575 goto error; 1576 if (!section_get_dword_field(section, "InstalledSize", 1, &component->size_app, 0)) 1577 goto error; 1578 if (!section_get_dword_field(section, "InstalledSize", 2, &component->size_win, 0)) 1579 goto error; 1580 1581 if (!section_get_str(section, "SuccessKey", &component->key_success, NULL)) 1582 goto error; 1583 if (!section_get_str(section, "CancelKey", &component->key_cancel, NULL)) 1584 goto error; 1585 if (!section_get_str(section, "ProgressKey", &component->key_progress, NULL)) 1586 goto error; 1587 if (!section_get_str(section, "UninstallKey", &component->key_uninstall, NULL)) 1588 goto error; 1589 if (!section_get_dword(section, "Reboot", &component->reboot, 0)) 1590 goto error; 1591 if (!section_get_dword(section, "AdminCheck", &component->admin, 0)) 1592 goto error; 1593 if (!section_get_dword(section, "UIVisible", &component->visibleui, 1)) 1594 goto error; 1595 if (!section_get_dword(section, "ActiveSetupAware", &component->as_aware, 0)) 1596 goto error; 1597 if (!section_get_dword(section, "Priority", &component->priority, 0)) 1598 goto error; 1599 1600 if (!read_version_entry(section, &component->version, &component->build)) 1601 goto error; 1602 if (!read_platform_entry(section, &component->platform)) 1603 goto error; 1604 if (!read_urls(component, section)) 1605 goto error; 1606 if (!read_dependencies(component, section)) 1607 goto error; 1608 1609 component->current_priority = component->priority; 1610 1611 add_component_by_priority(file, component); 1612 return S_OK; 1613 1614 error: 1615 component_free(component); 1616 return hr; 1617 } 1618 1619 static HRESULT process_group(struct ciffile *file, struct inf_section *section, const char *section_name) 1620 { 1621 struct cifgroup *group; 1622 HRESULT hr = E_OUTOFMEMORY; 1623 1624 group = heap_alloc_zero(sizeof(*group)); 1625 if (!group) return E_OUTOFMEMORY; 1626 1627 group->ICifGroup_iface.lpVtbl = &cifgroupVtbl; 1628 group->parent = &file->ICifFile_iface; 1629 1630 group->id = strdupA(section_name); 1631 if (!group->id) goto error; 1632 1633 if (!section_get_str(section, "DisplayName", &group->description, NULL)) 1634 goto error; 1635 if (!section_get_dword(section, "Priority", &group->priority, 0)) 1636 goto error; 1637 1638 list_add_head(&file->groups, &group->entry); 1639 return S_OK; 1640 1641 error: 1642 group_free(group); 1643 return hr; 1644 } 1645 1646 static HRESULT process_section(struct ciffile *file, struct inf_section *section, const char *section_name) 1647 { 1648 HRESULT hr; 1649 char *type; 1650 1651 if (!section_get_str(section, "SectionType", &type, "Component")) 1652 return E_OUTOFMEMORY; 1653 1654 if (!strcasecmp(type, "Component")) 1655 hr = process_component(file, section, section_name); 1656 else if (strcasecmp(type, "Group") == 0) 1657 hr = process_group(file, section, section_name); 1658 else 1659 FIXME("Don't know how to process %s\n", debugstr_a(type)); 1660 1661 heap_free(type); 1662 return hr; 1663 } 1664 1665 static HRESULT process_inf(struct ciffile *file, struct inf_file *inf) 1666 { 1667 struct inf_section *section = NULL; 1668 char *section_name; 1669 HRESULT hr = S_OK; 1670 1671 while (SUCCEEDED(hr) && inf_next_section(inf, §ion)) 1672 { 1673 section_name = inf_section_get_name(section); 1674 if (!section_name) return E_OUTOFMEMORY; 1675 1676 TRACE("start processing section %s\n", debugstr_a(section_name)); 1677 1678 if (!strcasecmp(section_name, "Strings") || 1679 !strncasecmp(section_name, "Strings.", strlen("Strings."))) 1680 { 1681 /* Ignore string sections */ 1682 } 1683 else if (strcasecmp(section_name, "Version") == 0) 1684 hr = process_version(file, section); 1685 else 1686 hr = process_section(file, section, section_name); 1687 1688 TRACE("finished processing section %s (%x)\n", debugstr_a(section_name), hr); 1689 heap_free(section_name); 1690 } 1691 1692 /* In case there was no version section, set the default installer description */ 1693 if (SUCCEEDED(hr) && !file->name) 1694 { 1695 file->name = strdupA(DEFAULT_INSTALLER_DESC); 1696 if (!file->name) hr = E_OUTOFMEMORY; 1697 } 1698 1699 return hr; 1700 } 1701 1702 static HRESULT load_ciffile(const char *path, ICifFile **icif) 1703 { 1704 struct inf_file *inf = NULL; 1705 struct ciffile *file; 1706 HRESULT hr = E_FAIL; 1707 1708 file = heap_alloc_zero(sizeof(*file)); 1709 if(!file) return E_OUTOFMEMORY; 1710 1711 file->ICifFile_iface.lpVtbl = &ciffileVtbl; 1712 file->ref = 1; 1713 1714 list_init(&file->components); 1715 list_init(&file->groups); 1716 1717 hr = inf_load(path, &inf); 1718 if (FAILED(hr)) goto error; 1719 1720 hr = process_inf(file, inf); 1721 if (FAILED(hr)) goto error; 1722 1723 *icif = &file->ICifFile_iface; 1724 return S_OK; 1725 1726 error: 1727 if (inf) inf_free(inf); 1728 ICifFile_Release(&file->ICifFile_iface); 1729 return hr; 1730 } 1731 1732 HRESULT WINAPI GetICifFileFromFile(ICifFile **icif, const char *path) 1733 { 1734 TRACE("(%p, %s)\n", icif, debugstr_a(path)); 1735 1736 return load_ciffile(path, icif); 1737 } 1738 1739 1740 HRESULT WINAPI GetICifRWFileFromFile(ICifRWFile **icif, const char *path) 1741 { 1742 FIXME("(%p, %s): stub\n", icif, debugstr_a(path)); 1743 1744 return E_NOTIMPL; 1745 } 1746