xref: /reactos/dll/win32/inseng/icif.c (revision 41203c9b)
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, &section))
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