/* * Copyright (C) 2008 Google (Roy Shea) * * 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 */ #include "corerror.h" #include "mstask_private.h" #include "wine/debug.h" WINE_DEFAULT_DEBUG_CHANNEL(mstask); typedef struct { ITaskScheduler ITaskScheduler_iface; LONG ref; } TaskSchedulerImpl; static inline TaskSchedulerImpl *impl_from_ITaskScheduler(ITaskScheduler *iface) { return CONTAINING_RECORD(iface, TaskSchedulerImpl, ITaskScheduler_iface); } static void TaskSchedulerDestructor(TaskSchedulerImpl *This) { TRACE("%p\n", This); HeapFree(GetProcessHeap(), 0, This); InterlockedDecrement(&dll_ref); } static HRESULT WINAPI MSTASK_ITaskScheduler_QueryInterface( ITaskScheduler* iface, REFIID riid, void **ppvObject) { TaskSchedulerImpl * This = impl_from_ITaskScheduler(iface); TRACE("IID: %s\n", debugstr_guid(riid)); if (IsEqualGUID(riid, &IID_IUnknown) || IsEqualGUID(riid, &IID_ITaskScheduler)) { *ppvObject = &This->ITaskScheduler_iface; ITaskScheduler_AddRef(iface); return S_OK; } *ppvObject = NULL; return E_NOINTERFACE; } static ULONG WINAPI MSTASK_ITaskScheduler_AddRef( ITaskScheduler* iface) { TaskSchedulerImpl *This = impl_from_ITaskScheduler(iface); TRACE("\n"); return InterlockedIncrement(&This->ref); } static ULONG WINAPI MSTASK_ITaskScheduler_Release( ITaskScheduler* iface) { TaskSchedulerImpl * This = impl_from_ITaskScheduler(iface); ULONG ref; TRACE("\n"); ref = InterlockedDecrement(&This->ref); if (ref == 0) TaskSchedulerDestructor(This); return ref; } static HRESULT WINAPI MSTASK_ITaskScheduler_SetTargetComputer( ITaskScheduler* iface, LPCWSTR pwszComputer) { TaskSchedulerImpl *This = impl_from_ITaskScheduler(iface); WCHAR buffer[MAX_COMPUTERNAME_LENGTH + 3]; /* extra space for two '\' and a zero */ DWORD len = MAX_COMPUTERNAME_LENGTH + 1; /* extra space for a zero */ TRACE("(%p)->(%s)\n", This, debugstr_w(pwszComputer)); /* NULL is an alias for the local computer */ if (!pwszComputer) return S_OK; buffer[0] = '\\'; buffer[1] = '\\'; if (GetComputerNameW(buffer + 2, &len)) { if (!lstrcmpiW(buffer, pwszComputer) || /* full unc name */ !lstrcmpiW(buffer + 2, pwszComputer)) /* name without backslash */ return S_OK; } FIXME("remote computer %s not supported\n", debugstr_w(pwszComputer)); return HRESULT_FROM_WIN32(ERROR_BAD_NETPATH); } static HRESULT WINAPI MSTASK_ITaskScheduler_GetTargetComputer( ITaskScheduler* iface, LPWSTR *ppwszComputer) { TaskSchedulerImpl *This = impl_from_ITaskScheduler(iface); LPWSTR buffer; DWORD len = MAX_COMPUTERNAME_LENGTH + 1; /* extra space for the zero */ TRACE("(%p)->(%p)\n", This, ppwszComputer); if (!ppwszComputer) return E_INVALIDARG; /* extra space for two '\' and a zero */ buffer = CoTaskMemAlloc((MAX_COMPUTERNAME_LENGTH + 3) * sizeof(WCHAR)); if (buffer) { buffer[0] = '\\'; buffer[1] = '\\'; if (GetComputerNameW(buffer + 2, &len)) { *ppwszComputer = buffer; return S_OK; } CoTaskMemFree(buffer); } *ppwszComputer = NULL; return HRESULT_FROM_WIN32(GetLastError()); } static HRESULT WINAPI MSTASK_ITaskScheduler_Enum( ITaskScheduler* iface, IEnumWorkItems **ppEnumTasks) { FIXME("%p, %p: stub\n", iface, ppEnumTasks); return E_NOTIMPL; } static HRESULT WINAPI MSTASK_ITaskScheduler_Activate( ITaskScheduler* iface, LPCWSTR pwszName, REFIID riid, IUnknown **ppunk) { TRACE("%p, %s, %s, %p: stub\n", iface, debugstr_w(pwszName), debugstr_guid(riid), ppunk); FIXME("Partial stub always returning COR_E_FILENOTFOUND\n"); return COR_E_FILENOTFOUND; } static HRESULT WINAPI MSTASK_ITaskScheduler_Delete( ITaskScheduler* iface, LPCWSTR pwszName) { FIXME("%p, %s: stub\n", iface, debugstr_w(pwszName)); return E_NOTIMPL; } static HRESULT WINAPI MSTASK_ITaskScheduler_NewWorkItem( ITaskScheduler* iface, LPCWSTR pwszTaskName, REFCLSID rclsid, REFIID riid, IUnknown **ppunk) { HRESULT hr; TRACE("(%p, %s, %s, %s, %p)\n", iface, debugstr_w(pwszTaskName), debugstr_guid(rclsid) ,debugstr_guid(riid), ppunk); if (!IsEqualGUID(rclsid, &CLSID_CTask)) return CLASS_E_CLASSNOTAVAILABLE; if (!IsEqualGUID(riid, &IID_ITask)) return E_NOINTERFACE; hr = TaskConstructor(pwszTaskName, (LPVOID *)ppunk); return hr; } static HRESULT WINAPI MSTASK_ITaskScheduler_AddWorkItem( ITaskScheduler* iface, LPCWSTR pwszTaskName, IScheduledWorkItem *pWorkItem) { FIXME("%p, %s, %p: stub\n", iface, debugstr_w(pwszTaskName), pWorkItem); return E_NOTIMPL; } static HRESULT WINAPI MSTASK_ITaskScheduler_IsOfType( ITaskScheduler* iface, LPCWSTR pwszName, REFIID riid) { FIXME("%p, %s, %s: stub\n", iface, debugstr_w(pwszName), debugstr_guid(riid)); return E_NOTIMPL; } static const ITaskSchedulerVtbl MSTASK_ITaskSchedulerVtbl = { MSTASK_ITaskScheduler_QueryInterface, MSTASK_ITaskScheduler_AddRef, MSTASK_ITaskScheduler_Release, MSTASK_ITaskScheduler_SetTargetComputer, MSTASK_ITaskScheduler_GetTargetComputer, MSTASK_ITaskScheduler_Enum, MSTASK_ITaskScheduler_Activate, MSTASK_ITaskScheduler_Delete, MSTASK_ITaskScheduler_NewWorkItem, MSTASK_ITaskScheduler_AddWorkItem, MSTASK_ITaskScheduler_IsOfType }; HRESULT TaskSchedulerConstructor(LPVOID *ppObj) { TaskSchedulerImpl *This; TRACE("(%p)\n", ppObj); This = HeapAlloc(GetProcessHeap(), 0, sizeof(*This)); if (!This) return E_OUTOFMEMORY; This->ITaskScheduler_iface.lpVtbl = &MSTASK_ITaskSchedulerVtbl; This->ref = 1; *ppObj = &This->ITaskScheduler_iface; InterlockedIncrement(&dll_ref); return S_OK; }