reactos/dll/win32/inseng/icif.c
2018-03-17 12:48:28 +01:00

1746 lines
43 KiB
C

/*
* Copyright 2016 Michael Müller
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
*/
#define COBJMACROS
#include "config.h"
#include <stdarg.h>
#include <string.h>
#include "windef.h"
#include "winbase.h"
#include "winuser.h"
#include "ole2.h"
#include "rpcproxy.h"
#include "inseng.h"
#include "inseng_private.h"
#include "wine/list.h"
#include "wine/debug.h"
WINE_DEFAULT_DEBUG_CHANNEL(inseng);
#define DEFAULT_INSTALLER_DESC "Active Setup Installation"
struct cifgroup
{
ICifGroup ICifGroup_iface;
struct list entry;
ICifFile *parent;
char *id;
char *description;
DWORD priority;
};
struct ciffenum_components
{
IEnumCifComponents IEnumCifComponents_iface;
LONG ref;
ICifFile *file;
struct list *start;
struct list *position;
char *group_id;
};
struct ciffenum_groups
{
IEnumCifGroups IEnumCifGroups_iface;
LONG ref;
ICifFile *file;
struct list *start;
struct list *position;
};
struct url_info
{
struct list entry;
INT index;
char *url;
DWORD flags;
};
struct dependency_info
{
struct list entry;
char *id;
char *type;
};
struct cifcomponent
{
ICifComponent ICifComponent_iface;
struct list entry;
ICifFile *parent;
char *id;
char *guid;
char *description;
char *details;
char *group;
DWORD version;
DWORD build;
char *patchid;
char *locale;
char *key_uninstall;
DWORD size_win;
DWORD size_app;
DWORD size_download;
DWORD size_extracted;
char *key_success;
char *key_progress;
char *key_cancel;
DWORD as_aware;
DWORD reboot;
DWORD admin;
DWORD visibleui;
DWORD priority;
DWORD platform;
struct list dependencies;
struct list urls;
/* mode */
/* det version */
/* one component */
/* custom data */
/* in memory state */
DWORD queue_state;
DWORD current_priority;
DWORD size_actual_download;
BOOL downloaded;
BOOL installed;
};
struct ciffile
{
ICifFile ICifFile_iface;
LONG ref;
struct list components;
struct list groups;
char *name;
};
static inline struct ciffile *impl_from_ICiffile(ICifFile *iface)
{
return CONTAINING_RECORD(iface, struct ciffile, ICifFile_iface);
}
static inline struct cifcomponent *impl_from_ICifComponent(ICifComponent *iface)
{
return CONTAINING_RECORD(iface, struct cifcomponent, ICifComponent_iface);
}
static inline struct cifgroup *impl_from_ICifGroup(ICifGroup *iface)
{
return CONTAINING_RECORD(iface, struct cifgroup, ICifGroup_iface);
}
static inline struct ciffenum_components *impl_from_IEnumCifComponents(IEnumCifComponents *iface)
{
return CONTAINING_RECORD(iface, struct ciffenum_components, IEnumCifComponents_iface);
}
static inline struct ciffenum_groups *impl_from_IEnumCifGroups(IEnumCifGroups *iface)
{
return CONTAINING_RECORD(iface, struct ciffenum_groups, IEnumCifGroups_iface);
}
static HRESULT enum_components_create(ICifFile *file, struct list *start, char *group_id, IEnumCifComponents **iface);
static HRESULT copy_substring_null(char *dest, int max_len, char *src)
{
if (!src)
return E_FAIL;
if (max_len <= 0)
return S_OK;
if (!dest)
return E_FAIL;
while (*src && max_len-- > 1)
*dest++ = *src++;
*dest = 0;
return S_OK;
}
static void url_entry_free(struct url_info *url)
{
heap_free(url->url);
heap_free(url);
}
static void dependency_entry_free(struct dependency_info *dependency)
{
heap_free(dependency->id);
heap_free(dependency);
}
static void component_free(struct cifcomponent *comp)
{
struct dependency_info *dependency, *dependency_next;
struct url_info *url, *url_next;
heap_free(comp->id);
heap_free(comp->guid);
heap_free(comp->description);
heap_free(comp->details);
heap_free(comp->group);
heap_free(comp->patchid);
heap_free(comp->locale);
heap_free(comp->key_uninstall);
heap_free(comp->key_success);
heap_free(comp->key_progress);
heap_free(comp->key_cancel);
LIST_FOR_EACH_ENTRY_SAFE(dependency, dependency_next, &comp->dependencies, struct dependency_info, entry)
{
list_remove(&dependency->entry);
dependency_entry_free(dependency);
}
LIST_FOR_EACH_ENTRY_SAFE(url, url_next, &comp->urls, struct url_info, entry)
{
list_remove(&url->entry);
url_entry_free(url);
}
heap_free(comp);
}
static void group_free(struct cifgroup *group)
{
heap_free(group->id);
heap_free(group->description);
heap_free(group);
}
static HRESULT WINAPI group_GetID(ICifGroup *iface, char *id, DWORD size)
{
struct cifgroup *This = impl_from_ICifGroup(iface);
TRACE("(%p)->(%p, %u)\n", This, id, size);
return copy_substring_null(id, size, This->id);
}
static HRESULT WINAPI group_GetDescription(ICifGroup *iface, char *desc, DWORD size)
{
struct cifgroup *This = impl_from_ICifGroup(iface);
TRACE("(%p)->(%p, %u)\n", This, desc, size);
return copy_substring_null(desc, size, This->description);
}
static DWORD WINAPI group_GetPriority(ICifGroup *iface)
{
struct cifgroup *This = impl_from_ICifGroup(iface);
TRACE("(%p)\n", This);
return This->priority;
}
static HRESULT WINAPI group_EnumComponents(ICifGroup *iface, IEnumCifComponents **enum_components, DWORD filter, LPVOID pv)
{
struct cifgroup *This = impl_from_ICifGroup(iface);
struct ciffile *file;
TRACE("(%p)->(%p, %u, %p)\n", This, enum_components, filter, pv);
if (filter)
FIXME("filter (%x) not supported\n", filter);
if (pv)
FIXME("how to handle pv (%p)?\n", pv);
file = impl_from_ICiffile(This->parent);
return enum_components_create(This->parent, &file->components, This->id, enum_components);
}
static DWORD WINAPI group_GetCurrentPriority(ICifGroup *iface)
{
struct cifgroup *This = impl_from_ICifGroup(iface);
FIXME("(%p): stub\n", This);
return 0;
}
static const ICifGroupVtbl cifgroupVtbl =
{
group_GetID,
group_GetDescription,
group_GetPriority,
group_EnumComponents,
group_GetCurrentPriority,
};
void component_set_actual_download_size(ICifComponent *iface, DWORD size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
This->size_actual_download = size;
}
void component_set_downloaded(ICifComponent *iface, BOOL value)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
This->downloaded = value;
}
void component_set_installed(ICifComponent *iface, BOOL value)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
This->installed = value;
}
char *component_get_id(ICifComponent *iface)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
return This->id;
}
static HRESULT WINAPI component_GetID(ICifComponent *iface, char *id, DWORD size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)->(%p, %u)\n", This, id, size);
return copy_substring_null(id, size, This->id);
}
static HRESULT WINAPI component_GetGUID(ICifComponent *iface, char *guid, DWORD size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)->(%p, %u)\n", This, guid, size);
return copy_substring_null(guid, size, This->guid);
}
static HRESULT WINAPI component_GetDescription(ICifComponent *iface, char *desc, DWORD size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)->(%p, %u)\n", This, desc, size);
return copy_substring_null(desc, size, This->description);
}
static HRESULT WINAPI component_GetDetails(ICifComponent *iface, char *details, DWORD size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)->(%p, %u)\n", This, details, size);
return copy_substring_null(details, size, This->details);
}
static HRESULT WINAPI component_GetUrl(ICifComponent *iface, UINT index, char *url, DWORD size, DWORD *flags)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
struct url_info *entry;
TRACE("(%p)->(%u, %p, %u, %p)\n", This, index, url, size, flags);
/* FIXME: check how functions behaves for url == NULL */
if (!flags)
return E_FAIL;
LIST_FOR_EACH_ENTRY(entry, &This->urls, struct url_info, entry)
{
if (entry->index != index)
continue;
*flags = entry->flags;
return copy_substring_null(url, size, entry->url);
}
return E_FAIL;
}
static HRESULT WINAPI component_GetFileExtractList(ICifComponent *iface, UINT index, char *list, DWORD size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
FIXME("(%p)->(%u, %p, %u): stub\n", This, index, list, size);
return E_NOTIMPL;
}
static HRESULT WINAPI component_GetUrlCheckRange(ICifComponent *iface, UINT index, DWORD *min, DWORD *max)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
FIXME("(%p)->(%u, %p, %p): stub\n", This, index, min, max);
return E_NOTIMPL;
}
static HRESULT WINAPI component_GetCommand(ICifComponent *iface, UINT index, char *cmd, DWORD cmd_size, char *switches, DWORD switch_size, DWORD *type)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
FIXME("(%p)->(%u, %p, %u, %p, %u, %p): stub\n", This, index, cmd, cmd_size, switches, switch_size, type);
return E_NOTIMPL;
}
static HRESULT WINAPI component_GetVersion(ICifComponent *iface, DWORD *version, DWORD *build)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)->(%p, %p)\n", This, version, build);
if (!version || !build)
return E_FAIL;
*version = This->version;
*build = This->build;
return S_OK;
}
static HRESULT WINAPI component_GetLocale(ICifComponent *iface, char *locale, DWORD size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)->(%p, %u)\n", This, locale, size);
return copy_substring_null(locale, size, This->locale);
}
static HRESULT WINAPI component_GetUninstallKey(ICifComponent *iface, char *key, DWORD size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)->(%p, %u)\n", This, key, size);
return copy_substring_null(key, size, This->key_uninstall);
}
static HRESULT WINAPI component_GetInstalledSize(ICifComponent *iface, DWORD *win, DWORD *app)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)->(%p, %p)\n", This, win, app);
if (!win || !app)
return E_FAIL;
*win = This->size_win;
*app = This->size_app;
return S_OK;
}
static DWORD WINAPI component_GetDownloadSize(ICifComponent *iface)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)\n", This);
return This->size_download;
}
static DWORD WINAPI component_GetExtractSize(ICifComponent *iface)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)\n", This);
return This->size_extracted;
}
static HRESULT WINAPI component_GetSuccessKey(ICifComponent *iface, char *key, DWORD size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)->(%p, %u)\n", This, key, size);
return copy_substring_null(key, size, This->key_success);
}
static HRESULT WINAPI component_GetProgressKeys(ICifComponent *iface, char *progress, DWORD progress_size,
char *cancel, DWORD cancel_size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
HRESULT hr;
TRACE("(%p)->(%p, %u, %p, %u): semi-stub\n", This, progress, progress_size, cancel, cancel_size);
hr = copy_substring_null(progress, progress_size, This->key_progress);
if (hr != S_OK) return hr;
if (cancel_size > 0 && cancel)
*cancel = 0;
return S_OK;
}
static HRESULT WINAPI component_IsActiveSetupAware(ICifComponent *iface)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)\n", This);
return This->as_aware ? S_OK : S_FALSE;
}
static HRESULT WINAPI component_IsRebootRequired(ICifComponent *iface)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)\n", This);
return This->reboot ? S_OK : S_FALSE;
}
static HRESULT WINAPI component_RequiresAdminRights(ICifComponent *iface)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)\n", This);
return This->admin ? S_OK : S_FALSE;
}
static DWORD WINAPI component_GetPriority(ICifComponent *iface)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)\n", This);
return This->priority;
}
static HRESULT WINAPI component_GetDependency(ICifComponent *iface, UINT index, char *id, DWORD id_size, char *type, DWORD *ver, DWORD *build)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
struct dependency_info *entry;
ICifComponent *dependency;
int pos = 0;
TRACE("(%p)->(%u, %p, %u, %p, %p, %p)\n", This, index, id, id_size, type, ver, build);
if (!id || !ver || !build)
return E_FAIL;
LIST_FOR_EACH_ENTRY(entry, &This->dependencies, struct dependency_info, entry)
{
if (pos++ < index)
continue;
if (ICifFile_FindComponent(This->parent, entry->id, &dependency) == S_OK)
{
ICifComponent_GetVersion(dependency, ver, build);
}
else
{
*ver = -1;
*build = -1;
}
if (entry->type)
*type = *entry->type;
else
*type = 'I';
return copy_substring_null(id, id_size, entry->id);
}
return E_FAIL;
}
static DWORD WINAPI component_GetPlatform(ICifComponent *iface)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)\n", This);
return This->platform;
}
static HRESULT WINAPI component_GetMode(ICifComponent *iface, UINT index, char *mode, DWORD size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
FIXME("(%p)->(%u, %p, %u): stub\n", This, index, mode, size);
return E_NOTIMPL;
}
static HRESULT WINAPI component_GetGroup(ICifComponent *iface, char *id, DWORD size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)->(%p, %u)\n", This, id, size);
return copy_substring_null(id, size, This->group);
}
static HRESULT WINAPI component_IsUIVisible(ICifComponent *iface)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)\n", This);
return This->visibleui ? S_OK : S_FALSE;
}
static HRESULT WINAPI component_GetPatchID(ICifComponent *iface, char *id, DWORD size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)->(%p, %u)\n", This, id, size);
return copy_substring_null(id, size, This->patchid);
}
static HRESULT WINAPI component_GetDetVersion(ICifComponent *iface, char *dll, DWORD dll_size, char *entry, DWORD entry_size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
FIXME("(%p)->(%p, %u, %p, %u): stub\n", This, dll, dll_size, entry, entry_size);
return E_NOTIMPL;
}
static HRESULT WINAPI component_GetTreatAsOneComponents(ICifComponent *iface, UINT index, char *id, DWORD size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
FIXME("(%p)->(%u, %p, %u): stub\n", This, index, id, size);
return E_NOTIMPL;
}
static HRESULT WINAPI component_GetCustomData(ICifComponent *iface, char *key, char *data, DWORD size)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
FIXME("(%p)->(%s, %p, %u): stub\n", This, debugstr_a(key), data, size);
return E_NOTIMPL;
}
static DWORD WINAPI component_IsComponentInstalled(ICifComponent *iface)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)\n", This);
return This->installed;
}
static HRESULT WINAPI component_IsComponentDownloaded(ICifComponent *iface)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)\n", This);
return This->downloaded ? S_OK : S_FALSE;
}
static DWORD WINAPI component_IsThisVersionInstalled(ICifComponent *iface, DWORD version, DWORD build, DWORD *ret_version, DWORD *ret_build)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
FIXME("(%p)->(%u, %u, %p, %p): stub\n", This, version, build, ret_version, ret_build);
return 0;
}
static DWORD WINAPI component_GetInstallQueueState(ICifComponent *iface)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)\n", This);
return This->queue_state;
}
static HRESULT WINAPI component_SetInstallQueueState(ICifComponent *iface, DWORD state)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)->(%u)\n", This, state);
This->queue_state = state;
return S_OK;
}
static DWORD WINAPI component_GetActualDownloadSize(ICifComponent *iface)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)\n", This);
return This->size_download;
}
static DWORD WINAPI component_GetCurrentPriority(ICifComponent *iface)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)\n", This);
return This->current_priority;
}
static HRESULT WINAPI component_SetCurrentPriority(ICifComponent *iface, DWORD priority)
{
struct cifcomponent *This = impl_from_ICifComponent(iface);
TRACE("(%p)->(%u)\n", This, priority);
This->current_priority = priority;
return S_OK;
}
static const ICifComponentVtbl cifcomponentVtbl =
{
component_GetID,
component_GetGUID,
component_GetDescription,
component_GetDetails,
component_GetUrl,
component_GetFileExtractList,
component_GetUrlCheckRange,
component_GetCommand,
component_GetVersion,
component_GetLocale,
component_GetUninstallKey,
component_GetInstalledSize,
component_GetDownloadSize,
component_GetExtractSize,
component_GetSuccessKey,
component_GetProgressKeys,
component_IsActiveSetupAware,
component_IsRebootRequired,
component_RequiresAdminRights,
component_GetPriority,
component_GetDependency,
component_GetPlatform,
component_GetMode,
component_GetGroup,
component_IsUIVisible,
component_GetPatchID,
component_GetDetVersion,
component_GetTreatAsOneComponents,
component_GetCustomData,
component_IsComponentInstalled,
component_IsComponentDownloaded,
component_IsThisVersionInstalled,
component_GetInstallQueueState,
component_SetInstallQueueState,
component_GetActualDownloadSize,
component_GetCurrentPriority,
component_SetCurrentPriority,
};
static HRESULT WINAPI enum_components_QueryInterface(IEnumCifComponents *iface, REFIID riid, void **ppv)
{
struct ciffenum_components *This = impl_from_IEnumCifComponents(iface);
if (IsEqualGUID(&IID_IUnknown, riid))
{
TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
*ppv = &This->IEnumCifComponents_iface;
}
/*
else if (IsEqualGUID(&IID_IEnumCifComponents, riid))
{
TRACE("(%p)->(IID_ICifFile %p)\n", This, ppv);
*ppv = &This->IEnumCifComponents_iface;
}
*/
else
{
FIXME("(%p)->(%s %p) not found\n", This, debugstr_guid(riid), ppv);
*ppv = NULL;
return E_NOINTERFACE;
}
IUnknown_AddRef((IUnknown *)*ppv);
return S_OK;
}
static ULONG WINAPI enum_components_AddRef(IEnumCifComponents *iface)
{
struct ciffenum_components *This = impl_from_IEnumCifComponents(iface);
LONG ref = InterlockedIncrement(&This->ref);
TRACE("(%p) ref=%d\n", This, ref);
return ref;
}
static ULONG WINAPI enum_components_Release(IEnumCifComponents *iface)
{
struct ciffenum_components *This = impl_from_IEnumCifComponents(iface);
LONG ref = InterlockedDecrement(&This->ref);
TRACE("(%p) ref=%d\n", This, ref);
if(!ref)
{
ICifFile_Release(This->file);
heap_free(This);
}
return ref;
}
static HRESULT WINAPI enum_components_Next(IEnumCifComponents *iface, ICifComponent **component)
{
struct ciffenum_components *This = impl_from_IEnumCifComponents(iface);
struct cifcomponent *comp;
TRACE("(%p)->(%p)\n", This, component);
if (!component)
return E_FAIL;
if (!This->position)
{
*component = NULL;
return E_FAIL;
}
do
{
This->position = list_next(This->start, This->position);
if (!This->position)
{
*component = NULL;
return E_FAIL;
}
comp = CONTAINING_RECORD(This->position, struct cifcomponent, entry);
} while (This->group_id && (!comp->group || strcmp(This->group_id, comp->group)));
*component = &comp->ICifComponent_iface;
return S_OK;
}
static HRESULT WINAPI enum_components_Reset(IEnumCifComponents *iface)
{
struct ciffenum_components *This = impl_from_IEnumCifComponents(iface);
TRACE("(%p)\n", This);
This->position = This->start;
return S_OK;
}
static const IEnumCifComponentsVtbl enum_componentsVtbl =
{
enum_components_QueryInterface,
enum_components_AddRef,
enum_components_Release,
enum_components_Next,
enum_components_Reset,
};
static HRESULT enum_components_create(ICifFile *file, struct list *start, char *group_id, IEnumCifComponents **iface)
{
struct ciffenum_components *enumerator;
enumerator = heap_alloc_zero(sizeof(*enumerator));
if (!enumerator) return E_OUTOFMEMORY;
enumerator->IEnumCifComponents_iface.lpVtbl = &enum_componentsVtbl;
enumerator->ref = 1;
enumerator->file = file;
enumerator->start = start;
enumerator->position = start;
enumerator->group_id = group_id;
ICifFile_AddRef(file);
*iface = &enumerator->IEnumCifComponents_iface;
return S_OK;
}
static HRESULT WINAPI enum_groups_QueryInterface(IEnumCifGroups *iface, REFIID riid, void **ppv)
{
struct ciffenum_groups *This = impl_from_IEnumCifGroups(iface);
if (IsEqualGUID(&IID_IUnknown, riid))
{
TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
*ppv = &This->IEnumCifGroups_iface;
}
/*
else if (IsEqualGUID(&IID_IEnumCifGroups, riid))
{
TRACE("(%p)->(IID_ICifFile %p)\n", This, ppv);
*ppv = &This->IEnumCifGroups_iface;
}
*/
else
{
FIXME("(%p)->(%s %p) not found\n", This, debugstr_guid(riid), ppv);
*ppv = NULL;
return E_NOINTERFACE;
}
IUnknown_AddRef((IUnknown *)*ppv);
return S_OK;
}
static ULONG WINAPI enum_groups_AddRef(IEnumCifGroups *iface)
{
struct ciffenum_groups *This = impl_from_IEnumCifGroups(iface);
LONG ref = InterlockedIncrement(&This->ref);
TRACE("(%p) ref=%d\n", This, ref);
return ref;
}
static ULONG WINAPI enum_groups_Release(IEnumCifGroups *iface)
{
struct ciffenum_groups *This = impl_from_IEnumCifGroups(iface);
LONG ref = InterlockedDecrement(&This->ref);
TRACE("(%p) ref=%d\n", This, ref);
if(!ref)
{
ICifFile_Release(This->file);
heap_free(This);
}
return ref;
}
static HRESULT WINAPI enum_groups_Next(IEnumCifGroups *iface, ICifGroup **group)
{
struct ciffenum_groups *This = impl_from_IEnumCifGroups(iface);
struct cifgroup *gp;
TRACE("(%p)->(%p)\n", This, group);
if (!This->position || !group)
return E_FAIL;
This->position = list_next(This->start, This->position);
if (!This->position)
return E_FAIL;
gp = CONTAINING_RECORD(This->position, struct cifgroup, entry);
*group = &gp->ICifGroup_iface;
return S_OK;
}
static HRESULT WINAPI enum_groups_Reset(IEnumCifGroups *iface)
{
struct ciffenum_groups *This = impl_from_IEnumCifGroups(iface);
TRACE("(%p)\n", This);
This->position = This->start;
return S_OK;
}
static const IEnumCifGroupsVtbl enum_groupsVtbl =
{
enum_groups_QueryInterface,
enum_groups_AddRef,
enum_groups_Release,
enum_groups_Next,
enum_groups_Reset,
};
static HRESULT enum_groups_create(ICifFile *file, struct list *start, IEnumCifGroups **iface)
{
struct ciffenum_groups *enumerator;
enumerator = heap_alloc_zero(sizeof(*enumerator));
if (!enumerator) return E_OUTOFMEMORY;
enumerator->IEnumCifGroups_iface.lpVtbl = &enum_groupsVtbl;
enumerator->ref = 1;
enumerator->file = file;
enumerator->start = start;
enumerator->position = start;
ICifFile_AddRef(file);
*iface = &enumerator->IEnumCifGroups_iface;
return S_OK;
}
static HRESULT WINAPI ciffile_QueryInterface(ICifFile *iface, REFIID riid, void **ppv)
{
struct ciffile *This = impl_from_ICiffile(iface);
if (IsEqualGUID(&IID_IUnknown, riid))
{
TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
*ppv = &This->ICifFile_iface;
}
else if (IsEqualGUID(&IID_ICifFile, riid))
{
TRACE("(%p)->(IID_ICifFile %p)\n", This, ppv);
*ppv = &This->ICifFile_iface;
}
else
{
FIXME("(%p)->(%s %p) not found\n", This, debugstr_guid(riid), ppv);
*ppv = NULL;
return E_NOINTERFACE;
}
IUnknown_AddRef((IUnknown *)*ppv);
return S_OK;
}
static ULONG WINAPI ciffile_AddRef(ICifFile *iface)
{
struct ciffile *This = impl_from_ICiffile(iface);
LONG ref = InterlockedIncrement(&This->ref);
TRACE("(%p) ref=%d\n", This, ref);
return ref;
}
static ULONG WINAPI ciffile_Release(ICifFile *iface)
{
struct ciffile *This = impl_from_ICiffile(iface);
LONG ref = InterlockedDecrement(&This->ref);
TRACE("(%p) ref=%d\n", This, ref);
if(!ref)
{
struct cifcomponent *comp, *comp_next;
struct cifgroup *group, *group_next;
heap_free(This->name);
LIST_FOR_EACH_ENTRY_SAFE(comp, comp_next, &This->components, struct cifcomponent, entry)
{
list_remove(&comp->entry);
component_free(comp);
}
LIST_FOR_EACH_ENTRY_SAFE(group, group_next, &This->groups, struct cifgroup, entry)
{
list_remove(&group->entry);
group_free(group);
}
heap_free(This);
}
return ref;
}
static HRESULT WINAPI ciffile_EnumComponents(ICifFile *iface, IEnumCifComponents **enum_components, DWORD filter, void *pv)
{
struct ciffile *This = impl_from_ICiffile(iface);
TRACE("(%p)->(%p, %u, %p)\n", This, enum_components, filter, pv);
if (filter)
FIXME("filter (%x) not supported\n", filter);
if (pv)
FIXME("how to handle pv (%p)?\n", pv);
return enum_components_create(iface, &This->components, NULL, enum_components);
}
static HRESULT WINAPI ciffile_FindComponent(ICifFile *iface, const char *id, ICifComponent **component)
{
struct ciffile *This = impl_from_ICiffile(iface);
struct cifcomponent *comp;
TRACE("(%p)->(%s, %p)\n", This, debugstr_a(id), component);
LIST_FOR_EACH_ENTRY(comp, &This->components, struct cifcomponent, entry)
{
if (strcmp(comp->id, id) != 0)
continue;
*component = &comp->ICifComponent_iface;
return S_OK;
}
return E_FAIL;
}
static HRESULT WINAPI ciffile_EnumGroups(ICifFile *iface, IEnumCifGroups **enum_groups, DWORD filter, void *pv)
{
struct ciffile *This = impl_from_ICiffile(iface);
TRACE("(%p)->(%p, %u, %p)\n", This, enum_groups, filter, pv);
if (filter)
FIXME("filter (%x) not supported\n", filter);
if (pv)
FIXME("how to handle pv (%p)?\n", pv);
return enum_groups_create(iface, &This->groups, enum_groups);
}
static HRESULT WINAPI ciffile_FindGroup(ICifFile *iface, const char *id, ICifGroup **group)
{
struct ciffile *This = impl_from_ICiffile(iface);
struct cifgroup *gp;
TRACE("(%p)->(%s, %p)\n", This, debugstr_a(id), group);
LIST_FOR_EACH_ENTRY(gp, &This->groups, struct cifgroup, entry)
{
if (strcmp(gp->id, id) != 0)
continue;
*group = &gp->ICifGroup_iface;
return S_OK;
}
return E_FAIL;
}
static HRESULT WINAPI ciffile_EnumModes(ICifFile *iface, IEnumCifModes **cuf_modes, DWORD filter, void *pv)
{
struct ciffile *This = impl_from_ICiffile(iface);
FIXME("(%p)->(%p, %u, %p): stub\n", This, cuf_modes, filter, pv);
return E_NOTIMPL;
}
static HRESULT WINAPI ciffile_FindMode(ICifFile *iface, const char *id, ICifMode **mode)
{
struct ciffile *This = impl_from_ICiffile(iface);
FIXME("(%p)->(%s, %p): stub\n", This, debugstr_a(id), mode);
return E_NOTIMPL;
}
static HRESULT WINAPI ciffile_GetDescription(ICifFile *iface, char *desc, DWORD size)
{
struct ciffile *This = impl_from_ICiffile(iface);
TRACE("(%p)->(%p, %u)\n", This, desc, size);
return copy_substring_null(desc, size, This->name);
}
static HRESULT WINAPI ciffile_GetDetDlls(ICifFile *iface, char *dlls, DWORD size)
{
struct ciffile *This = impl_from_ICiffile(iface);
FIXME("(%p)->(%p, %u): stub\n", This, dlls, size);
return E_NOTIMPL;
}
static const ICifFileVtbl ciffileVtbl =
{
ciffile_QueryInterface,
ciffile_AddRef,
ciffile_Release,
ciffile_EnumComponents,
ciffile_FindComponent,
ciffile_EnumGroups,
ciffile_FindGroup,
ciffile_EnumModes,
ciffile_FindMode,
ciffile_GetDescription,
ciffile_GetDetDlls,
};
static BOOL copy_string(char **dest, const char *source)
{
if (!source)
{
*dest = NULL;
return TRUE;
}
*dest = strdupA(source);
if (!dest) return FALSE;
return TRUE;
}
static BOOL section_get_str(struct inf_section *inf_sec, const char *key, char **value, const char *def)
{
struct inf_value *inf_val;
inf_val = inf_get_value(inf_sec, key);
if (!inf_val) return copy_string(value, def);
*value = inf_value_get_value(inf_val);
if (!*value) return FALSE;
return TRUE;
}
static char *next_part(char **str, BOOL strip_quotes)
{
char *start = *str;
char *next = *str;
while (*next && *next != ',')
next++;
if (!*next)
{
*str = trim(start, NULL, strip_quotes);
return NULL;
}
*next = 0;
*str = trim(start, NULL, strip_quotes);
return ++next;
}
static BOOL value_get_str_field(struct inf_value *inf_val, int field, char **value, const char *def)
{
char *line, *str, *next;
int i = 0;
line = inf_value_get_value(inf_val);
if (!line) return FALSE;
str = line;
do
{
i++;
next = next_part(&str, TRUE);
if (field == i)
{
BOOL ret = copy_string(value, str);
heap_free(line);
return ret;
}
str = next;
} while (str);
return copy_string(value, def);
}
/*
static BOOL section_get_str_field(struct inf_section *inf_sec, const char *key, int field, char **value, const char *def)
{
struct inf_value *inf_val;
inf_val = inf_get_value(inf_sec, key);
if (!inf_val) return copy_string(value, def);
return value_get_str_field(inf_val, field, value, def);
}
*/
static BOOL section_get_dword(struct inf_section *inf_sec, const char *key, DWORD *value, DWORD def)
{
struct inf_value *inf_val;
char *str;
inf_val = inf_get_value(inf_sec, key);
if (!inf_val)
{
*value = def;
return TRUE;
}
str = inf_value_get_value(inf_val);
if (!str) return FALSE;
*value = atoi(str);
heap_free(str);
return TRUE;
}
static BOOL value_get_dword_field(struct inf_value *inf_val, int field, DWORD *value, DWORD def)
{
char *value_str;
BOOL ret;
ret = value_get_str_field(inf_val, field, &value_str, NULL);
if (!ret) return FALSE;
if (!value_str)
{
*value = def;
return TRUE;
}
*value = atoi(value_str);
heap_free(value_str);
return TRUE;
}
static BOOL section_get_dword_field(struct inf_section *inf_sec, const char *key, int field, DWORD *value, DWORD def)
{
struct inf_value *inf_val;
inf_val = inf_get_value(inf_sec, key);
if (!inf_val)
{
*value = def;
return TRUE;
}
return value_get_dword_field(inf_val, field, value, def);
}
static HRESULT process_version(struct ciffile *file, struct inf_section *section)
{
if (!section_get_str(section, "DisplayName", &file->name, DEFAULT_INSTALLER_DESC))
return E_OUTOFMEMORY;
return S_OK;
}
static BOOL read_version_entry(struct inf_section *section, DWORD *ret_ver, DWORD *ret_build)
{
DWORD version = 0;
DWORD build = 0;
char *line, *str, *next;
if (!section_get_str(section, "Version", &line, NULL))
return FALSE;
if (!line) goto done;
str = line;
next = next_part(&str, TRUE);
version |= atoi(str) << 16;
if (!next) goto done;
str = next;
next = next_part(&str, TRUE);
version |= atoi(str) & 0xffff;
if (!next) goto done;
str = next;
next = next_part(&str, TRUE);
build |= atoi(str) << 16;
if (!next) goto done;
str = next;
next_part(&str, TRUE);
build |= atoi(str) & 0xffff;
done:
heap_free(line);
*ret_ver = version;
*ret_build = build;
return TRUE;
}
static BOOL read_platform_entry(struct inf_section *section, DWORD *ret_platform)
{
DWORD platform = PLATFORM_ALL;
char *line, *str, *next;
if (!section_get_str(section, "Platform", &line, NULL))
return FALSE;
if (!line) goto done;
platform = 0;
str = line;
do
{
next = next_part(&str, TRUE);
if (strcasecmp(str, "Win95") == 0)
platform |= PLATFORM_WIN98;
else if (strcasecmp(str, "Win98") == 0)
platform |= PLATFORM_WIN98;
else if (strcasecmp(str, "NT4") == 0)
platform |= PLATFORM_NT4;
else if (strcasecmp(str, "NT5") == 0)
platform |= PLATFORM_NT5;
else if (strcasecmp(str, "NT4Alpha") == 0)
platform |= PLATFORM_NT4;
else if (strcasecmp(str, "NT5Alpha") == 0)
platform |= PLATFORM_NT5;
else if (strcasecmp(str, "Millen") == 0)
platform |= PLATFORM_MILLEN;
else
FIXME("Unknown platform: %s\n", debugstr_a(str));
str = next;
} while (str);
done:
heap_free(line);
*ret_platform = platform;
return TRUE;
}
static BOOL read_dependencies(struct cifcomponent *component, struct inf_section *section)
{
struct dependency_info *dependency;
char *line, *str, *next;
BOOL ret = TRUE;
if (!section_get_str(section, "Dependencies", &line, NULL))
return E_OUTOFMEMORY;
if (!line) goto done;
ret = FALSE;
str = line;
do
{
next = next_part(&str, TRUE);
dependency = heap_alloc_zero(sizeof(*dependency));
if (!dependency) goto done;
dependency->id = strdupA(str);
if (!dependency->id)
{
heap_free(dependency);
goto done;
}
dependency->type = strstr(dependency->id, ":");
if (dependency->type) *dependency->type++ = 0;
list_add_tail(&component->dependencies, &dependency->entry);
str = next;
} while (str);
ret = TRUE;
done:
heap_free(line);
return ret;
}
static BOOL read_urls(struct cifcomponent *component, struct inf_section *section)
{
struct inf_value *inf_value = NULL;
struct url_info *url_entry;
char *str, *next;
int index;
while (inf_section_next_value(section, &inf_value))
{
str = inf_value_get_key(inf_value);
if (!str) return E_OUTOFMEMORY;
if (strncasecmp(str, "URL", 3))
goto next;
if (!str[3])
goto next;
index = strtol(str+3, &next, 10);
if (next == str+3 || *next != 0 || index < 1)
goto next;
index--;
url_entry = heap_alloc_zero(sizeof(*url_entry));
if (!url_entry) goto error;
url_entry->index = index;
if (!value_get_str_field(inf_value, 1, &url_entry->url, NULL))
goto error;
if (!url_entry->url || !*url_entry->url)
{
url_entry_free(url_entry);
goto next;
}
if (!value_get_dword_field(inf_value, 2, &url_entry->flags, 0))
goto error;
list_add_tail(&component->urls, &url_entry->entry);
next:
heap_free(str);
}
return TRUE;
error:
heap_free(str);
url_entry_free(url_entry);
return FALSE;
};
void add_component_by_priority(struct ciffile *file, struct cifcomponent *component)
{
struct cifcomponent *entry;
LIST_FOR_EACH_ENTRY(entry, &file->components, struct cifcomponent, entry)
{
if (entry->priority > component->priority)
continue;
list_add_before(&entry->entry, &component->entry);
return;
}
list_add_tail(&file->components, &component->entry);
}
static HRESULT process_component(struct ciffile *file, struct inf_section *section, const char *section_name)
{
struct cifcomponent *component;
HRESULT hr = E_OUTOFMEMORY;
component = heap_alloc_zero(sizeof(*component));
if (!component) return E_OUTOFMEMORY;
component->ICifComponent_iface.lpVtbl = &cifcomponentVtbl;
component->parent = &file->ICifFile_iface;
list_init(&component->urls);
list_init(&component->dependencies);
component->queue_state = ActionNone;
component->id = strdupA(section_name);
if (!component->id) goto error;
if (!section_get_str(section, "DisplayName", &component->description, NULL))
goto error;
if (!section_get_str(section, "GUID", &component->guid, NULL))
goto error;
if (!section_get_str(section, "Details", &component->details, NULL))
goto error;
if (!section_get_str(section, "Group", &component->group, NULL))
goto error;
if (!section_get_str(section, "Locale", &component->locale, "en"))
goto error;
if (!section_get_str(section, "PatchID", &component->patchid, NULL))
goto error;
if (!section_get_dword_field(section, "Size", 1, &component->size_download, 0))
goto error;
if (!section_get_dword_field(section, "Size", 2, &component->size_extracted, 0))
goto error;
if (!section_get_dword_field(section, "InstalledSize", 1, &component->size_app, 0))
goto error;
if (!section_get_dword_field(section, "InstalledSize", 2, &component->size_win, 0))
goto error;
if (!section_get_str(section, "SuccessKey", &component->key_success, NULL))
goto error;
if (!section_get_str(section, "CancelKey", &component->key_cancel, NULL))
goto error;
if (!section_get_str(section, "ProgressKey", &component->key_progress, NULL))
goto error;
if (!section_get_str(section, "UninstallKey", &component->key_uninstall, NULL))
goto error;
if (!section_get_dword(section, "Reboot", &component->reboot, 0))
goto error;
if (!section_get_dword(section, "AdminCheck", &component->admin, 0))
goto error;
if (!section_get_dword(section, "UIVisible", &component->visibleui, 1))
goto error;
if (!section_get_dword(section, "ActiveSetupAware", &component->as_aware, 0))
goto error;
if (!section_get_dword(section, "Priority", &component->priority, 0))
goto error;
if (!read_version_entry(section, &component->version, &component->build))
goto error;
if (!read_platform_entry(section, &component->platform))
goto error;
if (!read_urls(component, section))
goto error;
if (!read_dependencies(component, section))
goto error;
component->current_priority = component->priority;
add_component_by_priority(file, component);
return S_OK;
error:
component_free(component);
return hr;
}
static HRESULT process_group(struct ciffile *file, struct inf_section *section, const char *section_name)
{
struct cifgroup *group;
HRESULT hr = E_OUTOFMEMORY;
group = heap_alloc_zero(sizeof(*group));
if (!group) return E_OUTOFMEMORY;
group->ICifGroup_iface.lpVtbl = &cifgroupVtbl;
group->parent = &file->ICifFile_iface;
group->id = strdupA(section_name);
if (!group->id) goto error;
if (!section_get_str(section, "DisplayName", &group->description, NULL))
goto error;
if (!section_get_dword(section, "Priority", &group->priority, 0))
goto error;
list_add_head(&file->groups, &group->entry);
return S_OK;
error:
group_free(group);
return hr;
}
static HRESULT process_section(struct ciffile *file, struct inf_section *section, const char *section_name)
{
HRESULT hr;
char *type;
if (!section_get_str(section, "SectionType", &type, "Component"))
return E_OUTOFMEMORY;
if (!strcasecmp(type, "Component"))
hr = process_component(file, section, section_name);
else if (strcasecmp(type, "Group") == 0)
hr = process_group(file, section, section_name);
else
FIXME("Don't know how to process %s\n", debugstr_a(type));
heap_free(type);
return hr;
}
static HRESULT process_inf(struct ciffile *file, struct inf_file *inf)
{
struct inf_section *section = NULL;
char *section_name;
HRESULT hr = S_OK;
while (SUCCEEDED(hr) && inf_next_section(inf, &section))
{
section_name = inf_section_get_name(section);
if (!section_name) return E_OUTOFMEMORY;
TRACE("start processing section %s\n", debugstr_a(section_name));
if (!strcasecmp(section_name, "Strings") ||
!strncasecmp(section_name, "Strings.", strlen("Strings.")))
{
/* Ignore string sections */
}
else if (strcasecmp(section_name, "Version") == 0)
hr = process_version(file, section);
else
hr = process_section(file, section, section_name);
TRACE("finished processing section %s (%x)\n", debugstr_a(section_name), hr);
heap_free(section_name);
}
/* In case there was no version section, set the default installer description */
if (SUCCEEDED(hr) && !file->name)
{
file->name = strdupA(DEFAULT_INSTALLER_DESC);
if (!file->name) hr = E_OUTOFMEMORY;
}
return hr;
}
static HRESULT load_ciffile(const char *path, ICifFile **icif)
{
struct inf_file *inf = NULL;
struct ciffile *file;
HRESULT hr = E_FAIL;
file = heap_alloc_zero(sizeof(*file));
if(!file) return E_OUTOFMEMORY;
file->ICifFile_iface.lpVtbl = &ciffileVtbl;
file->ref = 1;
list_init(&file->components);
list_init(&file->groups);
hr = inf_load(path, &inf);
if (FAILED(hr)) goto error;
hr = process_inf(file, inf);
if (FAILED(hr)) goto error;
*icif = &file->ICifFile_iface;
return S_OK;
error:
if (inf) inf_free(inf);
ICifFile_Release(&file->ICifFile_iface);
return hr;
}
HRESULT WINAPI GetICifFileFromFile(ICifFile **icif, const char *path)
{
TRACE("(%p, %s)\n", icif, debugstr_a(path));
return load_ciffile(path, icif);
}
HRESULT WINAPI GetICifRWFileFromFile(ICifRWFile **icif, const char *path)
{
FIXME("(%p, %s): stub\n", icif, debugstr_a(path));
return E_NOTIMPL;
}