* INSENG Implementation
*
* Copyright 2006 Mike McCormack
- * 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
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
*/
-#include "inseng_private.h"
+#define WIN32_NO_STATUS
+#define _INC_WINDOWS
+#define COM_NO_WINDOWS_H
-#include <rpcproxy.h>
-#include <urlmon.h>
+#define COBJMACROS
-static HINSTANCE instance;
-
-enum thread_operation
-{
- OP_DOWNLOAD,
- OP_INSTALL
-};
-
-struct thread_info
-{
- DWORD operation;
- DWORD jobflags;
-
- IEnumCifComponents *enum_comp;
-
- DWORD download_size;
- DWORD install_size;
-
- DWORD downloaded_kb;
- ULONGLONG download_start;
-};
-
-struct InstallEngine
-{
- IInstallEngine2 IInstallEngine2_iface;
- IInstallEngineTiming IInstallEngineTiming_iface;
- LONG ref;
-
- IInstallEngineCallback *callback;
- char *baseurl;
- char *downloaddir;
- ICifFile *icif;
- DWORD status;
-
- /* used for the installation thread */
- struct thread_info thread;
-};
-
-struct downloadcb
-{
- IBindStatusCallback IBindStatusCallback_iface;
- LONG ref;
-
- WCHAR *file_name;
- WCHAR *cache_file;
-
- char *id;
- char *display;
-
- DWORD dl_size;
- DWORD dl_previous_kb;
-
- InstallEngine *engine;
- HANDLE event_done;
- HRESULT hr;
-};
-
-static inline struct downloadcb *impl_from_IBindStatusCallback(IBindStatusCallback *iface)
-{
- return CONTAINING_RECORD(iface, struct downloadcb, IBindStatusCallback_iface);
-}
-
-static inline InstallEngine *impl_from_IInstallEngine2(IInstallEngine2 *iface)
-{
- return CONTAINING_RECORD(iface, InstallEngine, IInstallEngine2_iface);
-}
-
-static inline InstallEngine *impl_from_IInstallEngineTiming(IInstallEngineTiming *iface)
-{
- return CONTAINING_RECORD(iface, InstallEngine, IInstallEngineTiming_iface);
-}
-
-static HRESULT WINAPI downloadcb_QueryInterface(IBindStatusCallback *iface, REFIID riid, void **ppv)
-{
- struct downloadcb *This = impl_from_IBindStatusCallback(iface);
-
- if (IsEqualGUID(&IID_IUnknown, riid))
- {
- TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
- *ppv = &This->IBindStatusCallback_iface;
- }
- else if (IsEqualGUID(&IID_IBindStatusCallback, riid))
- {
- TRACE("(%p)->(IID_IBindStatusCallback %p)\n", This, ppv);
- *ppv = &This->IBindStatusCallback_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 downloadcb_AddRef(IBindStatusCallback *iface)
-{
- struct downloadcb *This = impl_from_IBindStatusCallback(iface);
- LONG ref = InterlockedIncrement(&This->ref);
-
- TRACE("(%p) ref = %d\n", This, ref);
-
- return ref;
-}
-
-static ULONG WINAPI downloadcb_Release(IBindStatusCallback *iface)
-{
- struct downloadcb *This = impl_from_IBindStatusCallback(iface);
- LONG ref = InterlockedDecrement(&This->ref);
-
- TRACE("(%p) ref = %d\n", This, ref);
-
- if (!ref)
- {
- heap_free(This->file_name);
- heap_free(This->cache_file);
-
- IInstallEngine2_Release(&This->engine->IInstallEngine2_iface);
- heap_free(This);
- }
-
- return ref;
-}
-
-static HRESULT WINAPI downloadcb_OnStartBinding(IBindStatusCallback *iface, DWORD reserved, IBinding *pbind)
-{
- struct downloadcb *This = impl_from_IBindStatusCallback(iface);
-
- TRACE("(%p)->(%u %p)\n", This, reserved, pbind);
-
- return S_OK;
-}
-
-static HRESULT WINAPI downloadcb_GetPriority(IBindStatusCallback *iface, LONG *priority)
-{
- struct downloadcb *This = impl_from_IBindStatusCallback(iface);
-
- FIXME("(%p)->(%p): stub\n", This, priority);
-
- return E_NOTIMPL;
-}
-
-static HRESULT WINAPI downloadcb_OnLowResource(IBindStatusCallback *iface, DWORD reserved)
-{
- struct downloadcb *This = impl_from_IBindStatusCallback(iface);
-
- FIXME("(%p)->(%u): stub\n", This, reserved);
-
- return E_NOTIMPL;
-}
-
-static HRESULT WINAPI downloadcb_OnProgress(IBindStatusCallback *iface, ULONG progress,
- ULONG progress_max, ULONG status, const WCHAR *status_text)
-{
- struct downloadcb *This = impl_from_IBindStatusCallback(iface);
- HRESULT hr = S_OK;
-
- TRACE("%p)->(%u %u %u %s)\n", This, progress, progress_max, status, debugstr_w(status_text));
-
- switch(status)
- {
- case BINDSTATUS_BEGINDOWNLOADDATA:
- if (!This->engine->thread.download_start)
- This->engine->thread.download_start = GetTickCount64();
- /* fall-through */
- case BINDSTATUS_DOWNLOADINGDATA:
- case BINDSTATUS_ENDDOWNLOADDATA:
- This->engine->thread.downloaded_kb = This->dl_previous_kb + progress / 1024;
- if (This->engine->callback)
- {
- hr = IInstallEngineCallback_OnComponentProgress(This->engine->callback,
- This->id, INSTALLSTATUS_DOWNLOADING, This->display, NULL, progress / 1024, This->dl_size);
- }
- break;
-
- case BINDSTATUS_CACHEFILENAMEAVAILABLE:
- This->cache_file = strdupW(status_text);
- if (!This->cache_file)
- {
- ERR("Failed to allocate memory for cache file\n");
- hr = E_OUTOFMEMORY;
- }
- break;
-
- case BINDSTATUS_CONNECTING:
- case BINDSTATUS_SENDINGREQUEST:
- case BINDSTATUS_MIMETYPEAVAILABLE:
- case BINDSTATUS_FINDINGRESOURCE:
- break;
-
- default:
- FIXME("Unsupported status %u\n", status);
- }
-
- return hr;
-}
+#include <config.h>
-static HRESULT WINAPI downloadcb_OnStopBinding(IBindStatusCallback *iface, HRESULT hresult, LPCWSTR szError)
-{
- struct downloadcb *This = impl_from_IBindStatusCallback(iface);
+#include <stdarg.h>
- TRACE("(%p)->(%08x %s)\n", This, hresult, debugstr_w(szError));
+#include "windef.h"
+#include "winbase.h"
+#include "winuser.h"
+#include "ole2.h"
+#include "rpcproxy.h"
+#include "initguid.h"
+#include "inseng.h"
- if (FAILED(hresult))
- {
- This->hr = hresult;
- goto done;
- }
+#include "wine/debug.h"
- if (!This->cache_file)
- {
- This->hr = E_FAIL;
- goto done;
- }
+WINE_DEFAULT_DEBUG_CHANNEL(inseng);
- if (CopyFileW(This->cache_file, This->file_name, FALSE))
- This->hr = S_OK;
- else
- {
- ERR("CopyFile failed: %u\n", GetLastError());
- This->hr = E_FAIL;
- }
-
-done:
- SetEvent(This->event_done);
- return S_OK;
-}
-
-static HRESULT WINAPI downloadcb_GetBindInfo(IBindStatusCallback *iface,
- DWORD *grfBINDF, BINDINFO *pbindinfo)
+static inline void* __WINE_ALLOC_SIZE(1) heap_alloc(size_t size)
{
- struct downloadcb *This = impl_from_IBindStatusCallback(iface);
-
- TRACE("(%p)->(%p %p)\n", This, grfBINDF, pbindinfo);
-
- *grfBINDF = BINDF_PULLDATA | BINDF_NEEDFILE;
- return S_OK;
+ return HeapAlloc(GetProcessHeap(), 0, size);
}
-static HRESULT WINAPI downloadcb_OnDataAvailable(IBindStatusCallback *iface,
- DWORD grfBSCF, DWORD dwSize, FORMATETC *pformatetc, STGMEDIUM *pstgmed)
+static inline BOOL heap_free(void *mem)
{
- struct downloadcb *This = impl_from_IBindStatusCallback(iface);
-
- TRACE("(%p)->(%08x %u %p %p)\n", This, grfBSCF, dwSize, pformatetc, pstgmed);
-
- return S_OK;
+ return HeapFree(GetProcessHeap(), 0, mem);
}
-static HRESULT WINAPI downloadcb_OnObjectAvailable(IBindStatusCallback *iface,
- REFIID riid, IUnknown *punk)
-{
- struct downloadcb *This = impl_from_IBindStatusCallback(iface);
-
- FIXME("(%p)->(%s %p): stub\n", This, debugstr_guid(riid), punk);
-
- return E_NOTIMPL;
-}
+static HINSTANCE instance;
-static const IBindStatusCallbackVtbl BindStatusCallbackVtbl =
-{
- downloadcb_QueryInterface,
- downloadcb_AddRef,
- downloadcb_Release,
- downloadcb_OnStartBinding,
- downloadcb_GetPriority,
- downloadcb_OnLowResource,
- downloadcb_OnProgress,
- downloadcb_OnStopBinding,
- downloadcb_GetBindInfo,
- downloadcb_OnDataAvailable,
- downloadcb_OnObjectAvailable
+struct InstallEngine {
+ IInstallEngine2 IInstallEngine2_iface;
+ LONG ref;
};
-static HRESULT downloadcb_create(InstallEngine *engine, HANDLE event, char *file_name, char *id,
- char *display, DWORD dl_size, struct downloadcb **callback)
+static inline InstallEngine *impl_from_IInstallEngine2(IInstallEngine2 *iface)
{
- struct downloadcb *cb;
-
- cb = heap_zero_alloc(sizeof(*cb));
- if (!cb) return E_OUTOFMEMORY;
-
- cb->IBindStatusCallback_iface.lpVtbl = &BindStatusCallbackVtbl;
- cb->ref = 1;
- cb->hr = E_FAIL;
- cb->id = id;
- cb->display = display;
- cb->engine = engine;
- cb->dl_size = dl_size;
- cb->dl_previous_kb = engine->thread.downloaded_kb;
- cb->event_done = event;
- cb->file_name = strAtoW(file_name);
- if (!cb->file_name)
- {
- heap_free(cb);
- return E_OUTOFMEMORY;
- }
-
- IInstallEngine2_AddRef(&engine->IInstallEngine2_iface);
-
- *callback = cb;
- return S_OK;
+ return CONTAINING_RECORD(iface, InstallEngine, IInstallEngine2_iface);
}
static HRESULT WINAPI InstallEngine_QueryInterface(IInstallEngine2 *iface, REFIID riid, void **ppv)
}else if(IsEqualGUID(&IID_IInstallEngine2, riid)) {
TRACE("(%p)->(IID_IInstallEngine2 %p)\n", This, ppv);
*ppv = &This->IInstallEngine2_iface;
- }else if(IsEqualGUID(&IID_IInstallEngineTiming, riid)) {
- TRACE("(%p)->(IID_IInstallEngineTiming %p)\n", This, ppv);
- *ppv = &This->IInstallEngineTiming_iface;
}else {
- FIXME("(%p)->(%s %p) not found\n", This, debugstr_guid(riid), ppv);
+ TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
*ppv = NULL;
return E_NOINTERFACE;
}
- IUnknown_AddRef((IUnknown *)*ppv);
+ IUnknown_AddRef((IUnknown*)*ppv);
return S_OK;
}
TRACE("(%p) ref=%d\n", This, ref);
- if (!ref)
- {
- if (This->icif)
- ICifFile_Release(This->icif);
-
- heap_free(This->baseurl);
- heap_free(This->downloaddir);
+ if(!ref)
heap_free(This);
- }
return ref;
}
-static void set_status(InstallEngine *This, DWORD status)
-{
- This->status = status;
-
- if (This->callback)
- IInstallEngineCallback_OnEngineStatusChange(This->callback, status, 0);
-}
-
-static HRESULT calc_sizes(IEnumCifComponents *enum_comp, DWORD operation, DWORD *size_download, DWORD *size_install)
-{
- ICifComponent *comp;
- DWORD download = 0;
- DWORD install = 0;
- HRESULT hr;
-
- /* FIXME: what about inactive dependencies and how does
- * INSTALLOPTIONS_FORCEDEPENDENCIES play into this ?*/
-
- hr = IEnumCifComponents_Reset(enum_comp);
- if (FAILED(hr)) return hr;
-
- while (SUCCEEDED(IEnumCifComponents_Next(enum_comp, &comp)))
- {
- if (ICifComponent_GetInstallQueueState(comp) != ActionInstall)
- continue;
-
- /* FIXME: handle install options and find out the default options*/
- if (operation == OP_DOWNLOAD && ICifComponent_IsComponentDownloaded(comp) == S_FALSE)
- download += ICifComponent_GetDownloadSize(comp);
- /*
- if (operation == OP_INSTALL && ICifComponent_IsComponentInstalled(comp) == S_FALSE)
- install += ICifComponent_GetInstalledSize(comp);
- */
- }
-
- *size_download = download;
- *size_install = install;
-
- return S_OK;
-}
-
-static HRESULT get_next_component(IEnumCifComponents *enum_comp, DWORD operation, ICifComponent **ret_comp)
-{
- ICifComponent *comp;
- HRESULT hr;
-
- hr = IEnumCifComponents_Reset(enum_comp);
- if (FAILED(hr)) return hr;
-
- while (SUCCEEDED(IEnumCifComponents_Next(enum_comp, &comp)))
- {
- if (ICifComponent_GetInstallQueueState(comp) != ActionInstall)
- continue;
-
- /* FIXME: handle install options and find out the default options*/
- if (operation == OP_DOWNLOAD && ICifComponent_IsComponentDownloaded(comp) != S_FALSE)
- continue;
- if (operation == OP_INSTALL && ICifComponent_IsComponentInstalled(comp) != S_FALSE)
- continue;
-
- *ret_comp = comp;
- return S_OK;
- }
-
- return S_FALSE;
-}
-
-static HRESULT get_url(ICifComponent *comp, int index, char **url, DWORD *flags)
-{
- char *url_temp = NULL;
- int size = MAX_PATH / 2;
- HRESULT hr;
-
- /* FIXME: should we add an internal get function to prevent this ugly code ? */
-
- /* check if there is an url with such an index */
- hr = ICifComponent_GetUrl(comp, index, NULL, 0, flags);
- if (FAILED(hr))
- {
- *url = NULL;
- *flags = 0;
- return S_OK;
- }
-
- do
- {
- size *= 2;
- heap_free(url_temp);
- url_temp = heap_alloc(size);
- if (!url_temp) return E_OUTOFMEMORY;
-
- hr = ICifComponent_GetUrl(comp, index, url_temp, size, flags);
- if (FAILED(hr))
- {
- heap_free(url_temp);
- return hr;
- }
- }
- while (strlen(url_temp) == size-1);
-
- *url = url_temp;
- return S_OK;
-}
-
-static char *combine_url(char *baseurl, char *url)
-{
- int len_base = strlen(baseurl);
- int len_url = strlen(url);
- char *combined;
-
- combined = heap_alloc(len_base + len_url + 2);
- if (!combined) return NULL;
-
- strcpy(combined, baseurl);
- if (len_base && combined[len_base-1] != '/')
- strcat(combined, "/");
- strcat(combined, url);
-
- return combined;
-}
-
-static HRESULT generate_moniker(char *baseurl, char *url, DWORD flags, IMoniker **moniker)
-{
- WCHAR *urlW;
- HRESULT hr;
-
- if (flags & URLF_RELATIVEURL)
- {
- char *combined;
- if (!baseurl)
- return E_FAIL;
-
- combined = combine_url(baseurl, url);
- if (!combined) return E_OUTOFMEMORY;
-
- urlW = strAtoW(combined);
- heap_free(combined);
- if (!urlW) return E_OUTOFMEMORY;
- }
- else
- {
- urlW = strAtoW(url);
- if (!urlW) return E_OUTOFMEMORY;
- }
-
- hr = CreateURLMoniker(NULL, urlW, moniker);
- heap_free(urlW);
- return hr;
-}
-
-static char *merge_path(char *path1, char *path2)
-{
- int len = strlen(path1) + strlen(path2) + 2;
- char *combined = heap_alloc(len);
-
- if (!combined) return NULL;
- strcpy(combined, path1);
- strcat(combined, "\\");
- strcat(combined, path2);
-
- return combined;
-}
-
-static HRESULT download_url(InstallEngine *This, char *id, char *display, char *url, DWORD flags, DWORD dl_size)
-{
- struct downloadcb *callback = NULL;
- char *filename = NULL;
- IUnknown *unk = NULL;
- IMoniker *mon = NULL;
- IBindCtx *bindctx = NULL;
- HANDLE event = NULL;
- HRESULT hr;
-
- if (!This->downloaddir)
- {
- WARN("No download directory set\n");
- return E_FAIL;
- }
-
- hr = generate_moniker(This->baseurl, url, flags, &mon);
- if (FAILED(hr))
- {
- FIXME("Failed to create moniker\n");
- return hr;
- }
-
- event = CreateEventW(NULL, TRUE, FALSE, NULL);
- if (!event)
- {
- IMoniker_Release(mon);
- return E_FAIL;
- }
-
- filename = strrchr(url, '/');
- if (!filename) filename = url;
-
- filename = merge_path(This->downloaddir, filename);
- if (!filename)
- {
- hr = E_OUTOFMEMORY;
- goto error;
- }
-
- hr = downloadcb_create(This, event, filename, id, display, dl_size, &callback);
- if (FAILED(hr)) goto error;
-
- hr = CreateAsyncBindCtx(0, &callback->IBindStatusCallback_iface, NULL, &bindctx);
- if(FAILED(hr)) goto error;
-
- hr = IMoniker_BindToStorage(mon, bindctx, NULL, &IID_IUnknown, (void**)&unk);
- if (FAILED(hr)) goto error;
- if (unk) IUnknown_Release(unk);
-
- heap_free(filename);
- IMoniker_Release(mon);
- IBindCtx_Release(bindctx);
-
- WaitForSingleObject(event, INFINITE);
- hr = callback->hr;
-
- CloseHandle(event);
- IBindStatusCallback_Release(&callback->IBindStatusCallback_iface);
- return hr;
-
-error:
- if (mon) IMoniker_Release(mon);
- if (event) CloseHandle(event);
- if (callback) IBindStatusCallback_Release(&callback->IBindStatusCallback_iface);
- if (bindctx) IBindCtx_Release(bindctx);
- if (filename) heap_free(filename);
- return hr;
-}
-
-static HRESULT process_component_dependencies(InstallEngine *This, ICifComponent *comp)
-{
- char id[MAX_ID_LENGTH+1], type;
- DWORD ver, build;
- HRESULT hr;
- int i;
-
- for (i = 0;; i++)
- {
- hr = ICifComponent_GetDependency(comp, i, id, sizeof(id), &type, &ver, &build);
- if (SUCCEEDED(hr))
- FIXME("Can't handle dependencies yet: %s\n", debugstr_a(id));
- else
- break;
- }
-
- return S_OK;
-}
-
-static HRESULT process_component(InstallEngine *This, ICifComponent *comp)
-{
- DWORD size_dl, size_install, phase;
- char display[MAX_DISPLAYNAME_LENGTH+1];
- char id[MAX_ID_LENGTH+1];
- HRESULT hr;
- int i;
-
- hr = ICifComponent_GetID(comp, id, sizeof(id));
- if (FAILED(hr)) return hr;
-
- TRACE("processing component %s\n", debugstr_a(id));
-
- hr = ICifComponent_GetDescription(comp, display, sizeof(display));
- if (FAILED(hr)) return hr;
-
- size_dl = (This->thread.operation == OP_DOWNLOAD) ? ICifComponent_GetDownloadSize(comp) : 0;
- size_install = 0; /* (This->thread.operation == OP_INSTALL) ? ICifComponent_GetInstalledSize(comp) : 0; */
-
- if (This->callback)
- {
- IInstallEngineCallback_OnStartComponent(This->callback, id, size_dl, size_install, display);
- IInstallEngineCallback_OnComponentProgress(This->callback, id, INSTALLSTATUS_INITIALIZING, display, NULL, 0, 0);
- phase = INSTALLSTATUS_INITIALIZING;
- }
-
- hr = process_component_dependencies(This, comp);
- if (FAILED(hr)) return hr;
-
- if (This->thread.operation == OP_DOWNLOAD)
- {
- for (i = 0;; i++)
- {
- DWORD flags;
- char *url;
-
- phase = INSTALLSTATUS_DOWNLOADING;
-
- hr = get_url(comp, i, &url, &flags);
- if (FAILED(hr)) goto done;
- if (!url) break;
-
- TRACE("processing url %s\n", debugstr_a(url));
-
- hr = download_url(This, id, display, url, flags, size_dl);
- heap_free(url);
- if (FAILED(hr))
- {
- DWORD retry = 0;
-
- if (This->callback)
- IInstallEngineCallback_OnEngineProblem(This->callback, ENGINEPROBLEM_DOWNLOADFAIL, &retry);
- if (!retry) goto done;
-
- i--;
- continue;
- }
-
- phase = INSTALLSTATUS_CHECKINGTRUST;
- /* FIXME: check trust */
- IInstallEngineCallback_OnComponentProgress(This->callback, id, INSTALLSTATUS_CHECKINGTRUST, display, NULL, 0, 0);
- }
-
- component_set_downloaded(comp, TRUE);
- phase = INSTALLSTATUS_DOWNLOADFINISHED;
- }
- else
- FIXME("Installation not yet implemented\n");
-
-done:
- IInstallEngineCallback_OnStopComponent(This->callback, id, hr, phase, display, 0);
- return hr;
-}
-
-DWORD WINAPI thread_installation(LPVOID param)
-{
- InstallEngine *This = param;
- ICifComponent *comp;
- HRESULT hr;
-
- if (This->callback)
- IInstallEngineCallback_OnStartInstall(This->callback, This->thread.download_size, This->thread.install_size);
-
- for (;;)
- {
- hr = get_next_component(This->thread.enum_comp, This->thread.operation, &comp);
- if (FAILED(hr)) break;
- if (hr == S_FALSE)
- {
- hr = S_OK;
- break;
- }
-
- hr = process_component(This, comp);
- if (FAILED(hr)) break;
- }
-
- if (This->callback)
- IInstallEngineCallback_OnStopInstall(This->callback, hr, NULL, 0);
-
- IEnumCifComponents_Release(This->thread.enum_comp);
- IInstallEngine2_Release(&This->IInstallEngine2_iface);
-
- set_status(This, ENGINESTATUS_READY);
- return 0;
-}
-
-static HRESULT start_installation(InstallEngine *This, DWORD operation, DWORD jobflags)
-{
- HANDLE thread;
- HRESULT hr;
-
- This->thread.operation = operation;
- This->thread.jobflags = jobflags;
- This->thread.downloaded_kb = 0;
- This->thread.download_start = 0;
-
- /* Windows sends the OnStartInstall event from a different thread,
- * but OnStartInstall already contains the required download and install size.
- * The only way to signal an error from the thread is to send an OnStopComponent /
- * OnStopInstall signal which can only occur after OnStartInstall. We need to
- * precompute the sizes here to be able inform the application about errors while
- * calculating the required sizes. */
-
- hr = ICifFile_EnumComponents(This->icif, &This->thread.enum_comp, 0, NULL);
- if (FAILED(hr)) return hr;
-
- hr = calc_sizes(This->thread.enum_comp, operation, &This->thread.download_size, &This->thread.install_size);
- if (FAILED(hr)) goto error;
-
- IInstallEngine2_AddRef(&This->IInstallEngine2_iface);
-
- thread = CreateThread(NULL, 0, thread_installation, This, 0, NULL);
- if (!thread)
- {
- IInstallEngine2_Release(&This->IInstallEngine2_iface);
- hr = E_FAIL;
- goto error;
- }
-
- CloseHandle(thread);
- return S_OK;
-
-error:
- IEnumCifComponents_Release(This->thread.enum_comp);
- return hr;
-}
-
static HRESULT WINAPI InstallEngine_GetEngineStatus(IInstallEngine2 *iface, DWORD *status)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- TRACE("(%p)->(%p)\n", This, status);
-
- if (!status)
- return E_FAIL;
-
- *status = This->status;
- return S_OK;
+ FIXME("(%p)->(%p)\n", This, status);
+ return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_SetCifFile(IInstallEngine2 *iface, const char *cab_name, const char *cif_name)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- FIXME("(%p)->(%s %s): stub\n", This, debugstr_a(cab_name), debugstr_a(cif_name));
-
+ FIXME("(%p)->(%s %s)\n", This, debugstr_a(cab_name), debugstr_a(cif_name));
return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_DownloadComponents(IInstallEngine2 *iface, DWORD flags)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- TRACE("(%p)->(%x)\n", This, flags);
-
- /* The interface is not really threadsafe on windows, but we can at least prevent multiple installations */
- if (InterlockedCompareExchange((LONG *)&This->status, ENGINESTATUS_INSTALLING, ENGINESTATUS_READY) != ENGINESTATUS_READY)
- return E_FAIL;
-
- if (This->callback)
- IInstallEngineCallback_OnEngineStatusChange(This->callback, ENGINESTATUS_INSTALLING, 0);
-
- return start_installation(This, OP_DOWNLOAD, flags);
+ FIXME("(%p)->(%x)\n", This, flags);
+ return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_InstallComponents(IInstallEngine2 *iface, DWORD flags)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- FIXME("(%p)->(%x): stub\n", This, flags);
-
+ FIXME("(%p)->(%x)\n", This, flags);
return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_EnumInstallIDs(IInstallEngine2 *iface, UINT index, char **id)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- FIXME("(%p)->(%u %p): stub\n", This, index, id);
-
+ FIXME("(%p)->(%d %p)\n", This, index, id);
return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_EnumDownloadIDs(IInstallEngine2 *iface, UINT index, char **id)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
- IEnumCifComponents *enum_components;
- ICifComponent *comp;
- HRESULT hr;
-
- TRACE("(%p)->(%u %p)\n", This, index, id);
-
- if (!This->icif || !id)
- return E_FAIL;
-
- hr = ICifFile_EnumComponents(This->icif, &enum_components, 0, NULL);
- if (FAILED(hr)) return hr;
-
- for (;;)
- {
- hr = IEnumCifComponents_Next(enum_components, &comp);
- if (FAILED(hr)) goto done;
-
- if (ICifComponent_GetInstallQueueState(comp) != ActionInstall)
- continue;
-
- if (ICifComponent_IsComponentDownloaded(comp) != S_FALSE)
- continue;
-
- if (index == 0)
- {
- char *id_src = component_get_id(comp);
- *id = CoTaskMemAlloc(strlen(id_src) + 1);
-
- if (*id)
- strcpy(*id, id_src);
- else
- hr = E_OUTOFMEMORY;
- goto done;
- }
-
- index--;
- }
-
-done:
- IEnumCifComponents_Release(enum_components);
- return hr;
+ FIXME("(%p)->(%d %p)\n", This, index, id);
+ return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_IsComponentInstalled(IInstallEngine2 *iface, const char *id, DWORD *status)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- FIXME("(%p)->(%s %p): stub\n", This, debugstr_a(id), status);
-
+ FIXME("(%p)->(%s %p)\n", This, debugstr_a(id), status);
return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_RegisterInstallEngineCallback(IInstallEngine2 *iface, IInstallEngineCallback *callback)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- TRACE("(%p)->(%p)\n", This, callback);
-
- This->callback = callback;
- return S_OK;
+ FIXME("(%p)->(%p)\n", This, callback);
+ return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_UnregisterInstallEngineCallback(IInstallEngine2 *iface)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- TRACE("(%p)\n", This);
-
- This->callback = NULL;
- return S_OK;
+ FIXME("(%p)\n", This);
+ return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_SetAction(IInstallEngine2 *iface, const char *id, DWORD action, DWORD priority)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
- ICifComponent *comp;
- HRESULT hr;
-
- TRACE("(%p)->(%s %u %u)\n", This, debugstr_a(id), action, priority);
-
- if (!This->icif)
- return E_FAIL; /* FIXME: check error code */
-
- hr = ICifFile_FindComponent(This->icif, id, &comp);
- if (FAILED(hr)) return hr;
-
- hr = ICifComponent_SetInstallQueueState(comp, action);
- if (FAILED(hr)) return hr;
-
- hr = ICifComponent_SetCurrentPriority(comp, priority);
- return hr;
+ FIXME("(%p)->(%s %d %d)\n", This, debugstr_a(id), action, priority);
+ return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_GetSizes(IInstallEngine2 *iface, const char *id, COMPONENT_SIZES *sizes)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- FIXME("(%p)->(%s %p): stub\n", This, debugstr_a(id), sizes);
-
+ FIXME("(%p)->(%s %p)\n", This, debugstr_a(id), sizes);
return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_LaunchExtraCommand(IInstallEngine2 *iface, const char *inf_name, const char *section)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- FIXME("(%p)->(%s %s): stub\n", This, debugstr_a(inf_name), debugstr_a(section));
-
+ FIXME("(%p)->(%s %s)\n", This, debugstr_a(inf_name), debugstr_a(section));
return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_GetDisplayName(IInstallEngine2 *iface, const char *id, const char *name)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- FIXME("(%p)->(%s %s): stub\n", This, debugstr_a(id), debugstr_a(name));
-
+ FIXME("(%p)->(%s %s)\n", This, debugstr_a(id), debugstr_a(name));
return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_SetBaseUrl(IInstallEngine2 *iface, const char *base_name)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- TRACE("(%p)->(%s)\n", This, debugstr_a(base_name));
-
- if (This->baseurl)
- heap_free(This->baseurl);
-
- This->baseurl = strdupA(base_name);
- return This->baseurl ? S_OK : E_OUTOFMEMORY;
+ FIXME("(%p)->(%s)\n", This, debugstr_a(base_name));
+ return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_SetDownloadDir(IInstallEngine2 *iface, const char *download_dir)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- TRACE("(%p)->(%s)\n", This, debugstr_a(download_dir));
-
- if (This->downloaddir)
- heap_free(This->downloaddir);
-
- This->downloaddir = strdupA(download_dir);
- return This->downloaddir ? S_OK : E_OUTOFMEMORY;
+ FIXME("(%p)->(%s)\n", This, debugstr_a(download_dir));
+ return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_SetInstallDrive(IInstallEngine2 *iface, char drive)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- FIXME("(%p)->(%c): stub\n", This, drive);
-
+ FIXME("(%p)->(%c)\n", This, drive);
return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_SetInstallOptions(IInstallEngine2 *iface, DWORD flags)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- FIXME("(%p)->(%x): stub\n", This, flags);
-
+ FIXME("(%p)->(%x)\n", This, flags);
return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_SetHWND(IInstallEngine2 *iface, HWND hwnd)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- FIXME("(%p)->(%p): stub\n", This, hwnd);
-
+ FIXME("(%p)->(%p)\n", This, hwnd);
return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_SetIStream(IInstallEngine2 *iface, IStream *stream)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- FIXME("(%p)->(%p): stub\n", This, stream);
-
+ FIXME("(%p)->(%p)\n", This, stream);
return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_Abort(IInstallEngine2 *iface, DWORD flags)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- FIXME("(%p)->(%x): stub\n", This, flags);
-
+ FIXME("(%p)->(%x)\n", This, flags);
return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_Suspend(IInstallEngine2 *iface)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- FIXME("(%p): stub\n", This);
-
+ FIXME("(%p)\n", This);
return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine_Resume(IInstallEngine2 *iface)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- FIXME("(%p): stub\n", This);
-
+ FIXME("(%p)\n", This);
return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine2_SetLocalCif(IInstallEngine2 *iface, const char *cif)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
- HRESULT hr;
-
- TRACE("(%p)->(%s)\n", This, debugstr_a(cif));
-
- if (This->icif)
- ICifFile_Release(This->icif);
-
- set_status(This, ENGINESTATUS_LOADING);
-
- hr = GetICifFileFromFile(&This->icif, cif);
- if (SUCCEEDED(hr))
- set_status(This, ENGINESTATUS_READY);
- else
- {
- This->icif = NULL;
- set_status(This, ENGINESTATUS_NOTREADY);
- }
- return hr;
+ FIXME("(%p)->(%s)\n", This, debugstr_a(cif));
+ return E_NOTIMPL;
}
static HRESULT WINAPI InstallEngine2_GetICifFile(IInstallEngine2 *iface, ICifFile **cif_file)
{
InstallEngine *This = impl_from_IInstallEngine2(iface);
-
- TRACE("(%p)->(%p)\n", This, cif_file);
-
- if (!This->icif || !cif_file)
- return E_FAIL;
-
- ICifFile_AddRef(This->icif);
- *cif_file = This->icif;
- return S_OK;
+ FIXME("(%p)->(%p)\n", This, cif_file);
+ return E_NOTIMPL;
}
-static const IInstallEngine2Vtbl InstallEngine2Vtbl =
-{
+static const IInstallEngine2Vtbl InstallEngine2Vtbl = {
InstallEngine_QueryInterface,
InstallEngine_AddRef,
InstallEngine_Release,
InstallEngine2_GetICifFile
};
-static HRESULT WINAPI InstallEngineTiming_QueryInterface(IInstallEngineTiming *iface, REFIID riid, void **ppv)
-{
- InstallEngine *This = impl_from_IInstallEngineTiming(iface);
- return IInstallEngine2_QueryInterface(&This->IInstallEngine2_iface, riid, ppv);
-}
-
-static ULONG WINAPI InstallEngineTiming_AddRef(IInstallEngineTiming *iface)
-{
- InstallEngine *This = impl_from_IInstallEngineTiming(iface);
- return IInstallEngine2_AddRef(&This->IInstallEngine2_iface);
-}
-
-static ULONG WINAPI InstallEngineTiming_Release(IInstallEngineTiming *iface)
-{
- InstallEngine *This = impl_from_IInstallEngineTiming(iface);
- return IInstallEngine2_Release(&This->IInstallEngine2_iface);
-}
-
-static HRESULT WINAPI InstallEngineTiming_GetRates(IInstallEngineTiming *iface, DWORD *download, DWORD *install)
-{
- InstallEngine *This = impl_from_IInstallEngineTiming(iface);
-
- FIXME("(%p)->(%p, %p): stub\n", This, download, install);
-
- *download = 0;
- *install = 0;
-
- return S_OK;
-}
-
-static HRESULT WINAPI InstallEngineTiming_GetInstallProgress(IInstallEngineTiming *iface, INSTALLPROGRESS *progress)
-{
- InstallEngine *This = impl_from_IInstallEngineTiming(iface);
- ULONGLONG elapsed;
- static int once;
-
- if (!once++)
- FIXME("(%p)->(%p): semi-stub\n", This, progress);
- else
- TRACE("(%p)->(%p): semi-stub\n", This, progress);
-
- progress->dwDownloadKBRemaining = max(This->thread.download_size, This->thread.downloaded_kb) - This->thread.downloaded_kb;
-
- elapsed = GetTickCount64() - This->thread.download_start;
- if (This->thread.download_start && This->thread.downloaded_kb && elapsed > 100)
- progress->dwDownloadSecsRemaining = (progress->dwDownloadKBRemaining * elapsed) / (This->thread.downloaded_kb * 1000);
- else
- progress->dwDownloadSecsRemaining = -1;
-
- progress->dwInstallKBRemaining = 0;
- progress->dwInstallSecsRemaining = -1;
-
- return S_OK;
-}
-
-static const IInstallEngineTimingVtbl InstallEngineTimingVtbl =
-{
- InstallEngineTiming_QueryInterface,
- InstallEngineTiming_AddRef,
- InstallEngineTiming_Release,
- InstallEngineTiming_GetRates,
- InstallEngineTiming_GetInstallProgress,
-};
-
static HRESULT WINAPI ClassFactory_QueryInterface(IClassFactory *iface, REFIID riid, void **ppv)
{
*ppv = NULL;
TRACE("(%p %s %p)\n", outer, debugstr_guid(riid), ppv);
- engine = heap_zero_alloc(sizeof(*engine));
+ engine = heap_alloc(sizeof(*engine));
if(!engine)
return E_OUTOFMEMORY;
engine->IInstallEngine2_iface.lpVtbl = &InstallEngine2Vtbl;
- engine->IInstallEngineTiming_iface.lpVtbl = &InstallEngineTimingVtbl;
engine->ref = 1;
- engine->status = ENGINESTATUS_NOTREADY;
hres = IInstallEngine2_QueryInterface(&engine->IInstallEngine2_iface, riid, ppv);
IInstallEngine2_Release(&engine->IInstallEngine2_iface);