345 lines
9.4 KiB
C
345 lines
9.4 KiB
C
/*
|
|
* Copyright 2012 Jacek Caban for CodeWeavers
|
|
*
|
|
* 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 "windows.h"
|
|
#include "initguid.h"
|
|
#include "msident.h"
|
|
#include "rpcproxy.h"
|
|
|
|
#include "wine/debug.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(msident);
|
|
|
|
static inline void *heap_alloc(size_t len)
|
|
{
|
|
return HeapAlloc(GetProcessHeap(), 0, len);
|
|
}
|
|
|
|
static inline BOOL heap_free(void *mem)
|
|
{
|
|
return HeapFree(GetProcessHeap(), 0, mem);
|
|
}
|
|
|
|
static HINSTANCE msident_instance;
|
|
|
|
typedef struct {
|
|
IEnumUserIdentity IEnumUserIdentity_iface;
|
|
LONG ref;
|
|
} EnumUserIdentity;
|
|
|
|
static inline EnumUserIdentity *impl_from_IEnumUserIdentity(IEnumUserIdentity *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, EnumUserIdentity, IEnumUserIdentity_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI EnumUserIdentity_QueryInterface(IEnumUserIdentity *iface, REFIID riid, void **ppv)
|
|
{
|
|
EnumUserIdentity *This = impl_from_IEnumUserIdentity(iface);
|
|
|
|
if(IsEqualGUID(&IID_IUnknown, riid)) {
|
|
TRACE("(IID_IUnknown %p)\n", ppv);
|
|
*ppv = &This->IEnumUserIdentity_iface;
|
|
}else if(IsEqualGUID(&IID_IEnumUserIdentity, riid)) {
|
|
TRACE("(IID_IEnumUserIdentity %p)\n", ppv);
|
|
*ppv = &This->IEnumUserIdentity_iface;
|
|
}else {
|
|
WARN("(%s %p)\n", debugstr_guid(riid), ppv);
|
|
*ppv = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI EnumUserIdentity_AddRef(IEnumUserIdentity *iface)
|
|
{
|
|
EnumUserIdentity *This = impl_from_IEnumUserIdentity(iface);
|
|
LONG ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI EnumUserIdentity_Release(IEnumUserIdentity *iface)
|
|
{
|
|
EnumUserIdentity *This = impl_from_IEnumUserIdentity(iface);
|
|
LONG ref = InterlockedDecrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
if(!ref)
|
|
heap_free(This);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static HRESULT WINAPI EnumUserIdentity_Next(IEnumUserIdentity *iface, ULONG celt, IUnknown **rgelt, ULONG *pceltFetched)
|
|
{
|
|
EnumUserIdentity *This = impl_from_IEnumUserIdentity(iface);
|
|
FIXME("(%p)->(%u %p %p)\n", This, celt, rgelt, pceltFetched);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI EnumUserIdentity_Skip(IEnumUserIdentity *iface, ULONG celt)
|
|
{
|
|
EnumUserIdentity *This = impl_from_IEnumUserIdentity(iface);
|
|
FIXME("(%p)->(%u)\n", This, celt);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI EnumUserIdentity_Reset(IEnumUserIdentity *iface)
|
|
{
|
|
EnumUserIdentity *This = impl_from_IEnumUserIdentity(iface);
|
|
FIXME("(%p)->()\n", This);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI EnumUserIdentity_Clone(IEnumUserIdentity *iface, IEnumUserIdentity **ppenum)
|
|
{
|
|
EnumUserIdentity *This = impl_from_IEnumUserIdentity(iface);
|
|
FIXME("(%p)->(%p)\n", This, ppenum);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI EnumUserIdentity_GetCount(IEnumUserIdentity *iface, ULONG *pnCount)
|
|
{
|
|
EnumUserIdentity *This = impl_from_IEnumUserIdentity(iface);
|
|
|
|
FIXME("(%p)->(%p)\n", This, pnCount);
|
|
|
|
*pnCount = 0;
|
|
return S_OK;
|
|
}
|
|
|
|
static const IEnumUserIdentityVtbl EnumUserIdentityVtbl = {
|
|
EnumUserIdentity_QueryInterface,
|
|
EnumUserIdentity_AddRef,
|
|
EnumUserIdentity_Release,
|
|
EnumUserIdentity_Next,
|
|
EnumUserIdentity_Skip,
|
|
EnumUserIdentity_Reset,
|
|
EnumUserIdentity_Clone,
|
|
EnumUserIdentity_GetCount
|
|
};
|
|
|
|
static HRESULT WINAPI UserIdentityManager_QueryInterface(IUserIdentityManager *iface, REFIID riid, void **ppv)
|
|
{
|
|
if(IsEqualGUID(&IID_IUnknown, riid)) {
|
|
TRACE("(IID_IUnknown %p)\n", ppv);
|
|
*ppv = iface;
|
|
}else if(IsEqualGUID(&IID_IUserIdentityManager, riid)) {
|
|
TRACE("(IID_IUserIdentityManager %p)\n", ppv);
|
|
*ppv = iface;
|
|
}else {
|
|
WARN("(%s %p)\n", debugstr_guid(riid), ppv);
|
|
*ppv = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI UserIdentityManager_AddRef(IUserIdentityManager *iface)
|
|
{
|
|
TRACE("\n");
|
|
return 2;
|
|
}
|
|
|
|
static ULONG WINAPI UserIdentityManager_Release(IUserIdentityManager *iface)
|
|
{
|
|
TRACE("\n");
|
|
return 1;
|
|
}
|
|
|
|
static HRESULT WINAPI UserIdentityManager_EnumIdentities(IUserIdentityManager *iface, IEnumUserIdentity **ppEnumUser)
|
|
{
|
|
EnumUserIdentity *ret;
|
|
|
|
TRACE("(%p)\n", ppEnumUser);
|
|
|
|
ret = heap_alloc(sizeof(*ret));
|
|
if(!ret)
|
|
return E_OUTOFMEMORY;
|
|
|
|
ret->IEnumUserIdentity_iface.lpVtbl = &EnumUserIdentityVtbl;
|
|
ret->ref = 1;
|
|
|
|
*ppEnumUser = &ret->IEnumUserIdentity_iface;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI UserIdentityManager_ManageIdentities(IUserIdentityManager *iface, HWND hwndParent, DWORD dwFlags)
|
|
{
|
|
FIXME("(%p %x)\n", hwndParent, dwFlags);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI UserIdentityManager_Logon(IUserIdentityManager *iface, HWND hwndParent,
|
|
DWORD dwFlags, IUserIdentity **ppIdentity)
|
|
{
|
|
FIXME("(%p %x %p)\n", hwndParent, dwFlags, ppIdentity);
|
|
return E_USER_CANCELLED;
|
|
}
|
|
|
|
static HRESULT WINAPI UserIdentityManager_Logoff(IUserIdentityManager *iface, HWND hwndParent)
|
|
{
|
|
FIXME("(%p)\n", hwndParent);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI UserIdentityManager_GetIdentityByCookie(IUserIdentityManager *iface, GUID *uidCookie,
|
|
IUserIdentity **ppIdentity)
|
|
{
|
|
FIXME("(%p %p)\n", uidCookie, ppIdentity);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IUserIdentityManagerVtbl UserIdentityManagerVtbl = {
|
|
UserIdentityManager_QueryInterface,
|
|
UserIdentityManager_AddRef,
|
|
UserIdentityManager_Release,
|
|
UserIdentityManager_EnumIdentities,
|
|
UserIdentityManager_ManageIdentities,
|
|
UserIdentityManager_Logon,
|
|
UserIdentityManager_Logoff,
|
|
UserIdentityManager_GetIdentityByCookie
|
|
};
|
|
|
|
static IUserIdentityManager UserIdentityManager = { &UserIdentityManagerVtbl };
|
|
|
|
static HRESULT WINAPI UserIdentityManager_CreateInstance(IClassFactory *iface, IUnknown *outer, REFIID riid, void **ppv)
|
|
{
|
|
TRACE("\n");
|
|
|
|
return IUserIdentityManager_QueryInterface(&UserIdentityManager, riid, ppv);
|
|
}
|
|
|
|
static HRESULT WINAPI ClassFactory_QueryInterface(IClassFactory *iface, REFIID riid, void **ppv)
|
|
{
|
|
*ppv = NULL;
|
|
|
|
if(IsEqualGUID(&IID_IUnknown, riid)) {
|
|
TRACE("(%p)->(IID_IUnknown %p)\n", iface, ppv);
|
|
*ppv = iface;
|
|
}else if(IsEqualGUID(&IID_IClassFactory, riid)) {
|
|
TRACE("(%p)->(IID_IClassFactory %p)\n", iface, ppv);
|
|
*ppv = iface;
|
|
}
|
|
|
|
if(*ppv) {
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
FIXME("(%p)->(%s %p)\n", iface, debugstr_guid(riid), ppv);
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
static ULONG WINAPI ClassFactory_AddRef(IClassFactory *iface)
|
|
{
|
|
TRACE("(%p)\n", iface);
|
|
return 2;
|
|
}
|
|
|
|
static ULONG WINAPI ClassFactory_Release(IClassFactory *iface)
|
|
{
|
|
TRACE("(%p)\n", iface);
|
|
return 1;
|
|
}
|
|
|
|
static HRESULT WINAPI ClassFactory_LockServer(IClassFactory *iface, BOOL fLock)
|
|
{
|
|
TRACE("(%p)->(%x)\n", iface, fLock);
|
|
return S_OK;
|
|
}
|
|
|
|
static const IClassFactoryVtbl UserIdentityManagerCFVtbl = {
|
|
ClassFactory_QueryInterface,
|
|
ClassFactory_AddRef,
|
|
ClassFactory_Release,
|
|
UserIdentityManager_CreateInstance,
|
|
ClassFactory_LockServer
|
|
};
|
|
|
|
static IClassFactory UserIdentityManagerCF = { &UserIdentityManagerCFVtbl };
|
|
|
|
/******************************************************************
|
|
* DllMain (msident.@)
|
|
*/
|
|
BOOL WINAPI DllMain(HINSTANCE hInstDLL, DWORD fdwReason, LPVOID lpv)
|
|
{
|
|
TRACE("(%p %d %p)\n", hInstDLL, fdwReason, lpv);
|
|
|
|
switch(fdwReason)
|
|
{
|
|
case DLL_WINE_PREATTACH:
|
|
return FALSE; /* prefer native version */
|
|
case DLL_PROCESS_ATTACH:
|
|
msident_instance = hInstDLL;
|
|
DisableThreadLibraryCalls(hInstDLL);
|
|
break;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* DllGetClassObject (msident.@)
|
|
*/
|
|
HRESULT WINAPI DllGetClassObject(REFCLSID rclsid, REFIID riid, LPVOID *ppv)
|
|
{
|
|
if(IsEqualGUID(&CLSID_UserIdentityManager, rclsid)) {
|
|
TRACE("CLSID_UserIdentityManager\n");
|
|
return IClassFactory_QueryInterface(&UserIdentityManagerCF, riid, ppv);
|
|
}
|
|
|
|
FIXME("%s %s %p\n", debugstr_guid(rclsid), debugstr_guid(riid), ppv);
|
|
return CLASS_E_CLASSNOTAVAILABLE;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* DllCanUnloadNow (msident.@)
|
|
*/
|
|
HRESULT WINAPI DllCanUnloadNow(void)
|
|
{
|
|
return S_FALSE;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* DllRegisterServer (msident.@)
|
|
*/
|
|
HRESULT WINAPI DllRegisterServer(void)
|
|
{
|
|
TRACE("()\n");
|
|
return __wine_register_resources(msident_instance);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* DllUnregisterServer (msident.@)
|
|
*/
|
|
HRESULT WINAPI DllUnregisterServer(void)
|
|
{
|
|
TRACE("()\n");
|
|
return __wine_unregister_resources(msident_instance);
|
|
}
|