mirror of
git://source.winehq.org/git/wine.git
synced 2024-10-31 11:43:31 +00:00
225 lines
6.4 KiB
C
225 lines
6.4 KiB
C
/*
|
|
* Implementation of IEnumRegFilters Interface
|
|
*
|
|
* Copyright 2004 Christian Costa
|
|
*
|
|
* 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 "quartz_private.h"
|
|
|
|
#include "wine/unicode.h"
|
|
|
|
#include "wine/debug.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(quartz);
|
|
|
|
typedef struct IEnumRegFiltersImpl
|
|
{
|
|
IEnumRegFilters IEnumRegFilters_iface;
|
|
LONG refCount;
|
|
ULONG size;
|
|
REGFILTER* RegFilters;
|
|
ULONG uIndex;
|
|
} IEnumRegFiltersImpl;
|
|
|
|
static inline IEnumRegFiltersImpl *impl_from_IEnumRegFilters(IEnumRegFilters *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, IEnumRegFiltersImpl, IEnumRegFilters_iface);
|
|
}
|
|
|
|
static const struct IEnumRegFiltersVtbl IEnumRegFiltersImpl_Vtbl;
|
|
|
|
HRESULT IEnumRegFiltersImpl_Construct(REGFILTER* pInRegFilters, const ULONG size, IEnumRegFilters ** ppEnum)
|
|
{
|
|
IEnumRegFiltersImpl* pEnumRegFilters;
|
|
REGFILTER* pRegFilters = NULL;
|
|
unsigned int i;
|
|
|
|
TRACE("(%p, %d, %p)\n", pInRegFilters, size, ppEnum);
|
|
|
|
pEnumRegFilters = CoTaskMemAlloc(sizeof(IEnumRegFiltersImpl));
|
|
if (!pEnumRegFilters)
|
|
{
|
|
*ppEnum = NULL;
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
/* Accept size of 0 */
|
|
if (size)
|
|
{
|
|
pRegFilters = CoTaskMemAlloc(sizeof(REGFILTER)*size);
|
|
if (!pRegFilters)
|
|
{
|
|
CoTaskMemFree(pEnumRegFilters);
|
|
*ppEnum = NULL;
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
|
|
for(i = 0; i < size; i++)
|
|
{
|
|
pRegFilters[i].Clsid = pInRegFilters[i].Clsid;
|
|
pRegFilters[i].Name = CoTaskMemAlloc((strlenW(pInRegFilters[i].Name)+1)*sizeof(WCHAR));
|
|
if (!pRegFilters[i].Name)
|
|
{
|
|
while(i)
|
|
CoTaskMemFree(pRegFilters[--i].Name);
|
|
CoTaskMemFree(pRegFilters);
|
|
CoTaskMemFree(pEnumRegFilters);
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
CopyMemory(pRegFilters[i].Name, pInRegFilters[i].Name, (strlenW(pInRegFilters[i].Name)+1)*sizeof(WCHAR));
|
|
}
|
|
|
|
pEnumRegFilters->IEnumRegFilters_iface.lpVtbl = &IEnumRegFiltersImpl_Vtbl;
|
|
pEnumRegFilters->refCount = 1;
|
|
pEnumRegFilters->uIndex = 0;
|
|
pEnumRegFilters->RegFilters = pRegFilters;
|
|
pEnumRegFilters->size = size;
|
|
|
|
*ppEnum = &pEnumRegFilters->IEnumRegFilters_iface;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI IEnumRegFiltersImpl_QueryInterface(IEnumRegFilters * iface, REFIID riid, LPVOID * ppv)
|
|
{
|
|
TRACE("(%p)->(%s, %p)\n", iface, qzdebugstr_guid(riid), ppv);
|
|
|
|
*ppv = NULL;
|
|
|
|
if (IsEqualIID(riid, &IID_IUnknown))
|
|
*ppv = iface;
|
|
else if (IsEqualIID(riid, &IID_IEnumRegFilters))
|
|
*ppv = iface;
|
|
|
|
if (*ppv)
|
|
{
|
|
IUnknown_AddRef((IUnknown *)(*ppv));
|
|
return S_OK;
|
|
}
|
|
|
|
FIXME("No interface for %s!\n", qzdebugstr_guid(riid));
|
|
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
static ULONG WINAPI IEnumRegFiltersImpl_AddRef(IEnumRegFilters * iface)
|
|
{
|
|
IEnumRegFiltersImpl *This = impl_from_IEnumRegFilters(iface);
|
|
ULONG refCount = InterlockedIncrement(&This->refCount);
|
|
|
|
TRACE("(%p)\n", iface);
|
|
|
|
return refCount;
|
|
}
|
|
|
|
static ULONG WINAPI IEnumRegFiltersImpl_Release(IEnumRegFilters * iface)
|
|
{
|
|
IEnumRegFiltersImpl *This = impl_from_IEnumRegFilters(iface);
|
|
ULONG refCount = InterlockedDecrement(&This->refCount);
|
|
|
|
TRACE("(%p)\n", iface);
|
|
|
|
if (!refCount)
|
|
{
|
|
ULONG i;
|
|
|
|
for(i = 0; i < This->size; i++)
|
|
{
|
|
CoTaskMemFree(This->RegFilters[i].Name);
|
|
}
|
|
CoTaskMemFree(This->RegFilters);
|
|
CoTaskMemFree(This);
|
|
return 0;
|
|
} else
|
|
return refCount;
|
|
}
|
|
|
|
static HRESULT WINAPI IEnumRegFiltersImpl_Next(IEnumRegFilters * iface, ULONG cFilters, REGFILTER ** ppRegFilter, ULONG * pcFetched)
|
|
{
|
|
ULONG cFetched;
|
|
IEnumRegFiltersImpl *This = impl_from_IEnumRegFilters(iface);
|
|
unsigned int i;
|
|
|
|
cFetched = min(This->size, This->uIndex + cFilters) - This->uIndex;
|
|
|
|
TRACE("(%p)->(%u, %p, %p)\n", iface, cFilters, ppRegFilter, pcFetched);
|
|
|
|
if (cFetched > 0)
|
|
{
|
|
for(i = 0; i < cFetched; i++)
|
|
{
|
|
/* The string in the REGFILTER structure must be allocated in the same block as the REGFILTER structure itself */
|
|
ppRegFilter[i] = CoTaskMemAlloc(sizeof(REGFILTER)+(strlenW(This->RegFilters[This->uIndex + i].Name)+1)*sizeof(WCHAR));
|
|
if (!ppRegFilter[i])
|
|
{
|
|
while(i)
|
|
{
|
|
CoTaskMemFree(ppRegFilter[--i]);
|
|
ppRegFilter[i] = NULL;
|
|
}
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
ppRegFilter[i]->Clsid = This->RegFilters[This->uIndex + i].Clsid;
|
|
ppRegFilter[i]->Name = (WCHAR*)((char*)ppRegFilter[i]+sizeof(REGFILTER));
|
|
CopyMemory(ppRegFilter[i]->Name, This->RegFilters[This->uIndex + i].Name,
|
|
(strlenW(This->RegFilters[This->uIndex + i].Name)+1)*sizeof(WCHAR));
|
|
}
|
|
|
|
This->uIndex += cFetched;
|
|
if (pcFetched)
|
|
*pcFetched = cFetched;
|
|
return S_OK;
|
|
}
|
|
|
|
return S_FALSE;
|
|
}
|
|
|
|
static HRESULT WINAPI IEnumRegFiltersImpl_Skip(IEnumRegFilters * iface, ULONG n)
|
|
{
|
|
TRACE("(%p)->(%u)\n", iface, n);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI IEnumRegFiltersImpl_Reset(IEnumRegFilters * iface)
|
|
{
|
|
IEnumRegFiltersImpl *This = impl_from_IEnumRegFilters(iface);
|
|
|
|
TRACE("(%p)\n", iface);
|
|
|
|
This->uIndex = 0;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI IEnumRegFiltersImpl_Clone(IEnumRegFilters * iface, IEnumRegFilters ** ppEnum)
|
|
{
|
|
TRACE("(%p)->(%p)\n", iface, ppEnum);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IEnumRegFiltersVtbl IEnumRegFiltersImpl_Vtbl =
|
|
{
|
|
IEnumRegFiltersImpl_QueryInterface,
|
|
IEnumRegFiltersImpl_AddRef,
|
|
IEnumRegFiltersImpl_Release,
|
|
IEnumRegFiltersImpl_Next,
|
|
IEnumRegFiltersImpl_Skip,
|
|
IEnumRegFiltersImpl_Reset,
|
|
IEnumRegFiltersImpl_Clone
|
|
};
|