2012-06-14 16:02:38 +02:00
|
|
|
/*
|
|
|
|
* Copyright 2012 Hans Leidekker 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 "config.h"
|
|
|
|
#include <stdarg.h>
|
|
|
|
|
|
|
|
#include "windef.h"
|
|
|
|
#include "winbase.h"
|
|
|
|
#include "objbase.h"
|
|
|
|
#include "wbemcli.h"
|
|
|
|
|
|
|
|
#include "wine/debug.h"
|
|
|
|
#include "wine/unicode.h"
|
|
|
|
#include "wbemprox_private.h"
|
|
|
|
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(wbemprox);
|
|
|
|
|
|
|
|
struct client_security
|
|
|
|
{
|
|
|
|
IClientSecurity IClientSecurity_iface;
|
|
|
|
};
|
|
|
|
|
|
|
|
static inline struct client_security *impl_from_IClientSecurity( IClientSecurity *iface )
|
|
|
|
{
|
|
|
|
return CONTAINING_RECORD( iface, struct client_security, IClientSecurity_iface );
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI client_security_QueryInterface(
|
|
|
|
IClientSecurity *iface,
|
|
|
|
REFIID riid,
|
|
|
|
void **ppvObject )
|
|
|
|
{
|
|
|
|
struct client_security *cs = impl_from_IClientSecurity( iface );
|
|
|
|
|
|
|
|
TRACE("%p %s %p\n", cs, debugstr_guid( riid ), ppvObject );
|
|
|
|
|
|
|
|
if ( IsEqualGUID( riid, &IID_IClientSecurity ) ||
|
|
|
|
IsEqualGUID( riid, &IID_IUnknown ) )
|
|
|
|
{
|
|
|
|
*ppvObject = cs;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
FIXME("interface %s not implemented\n", debugstr_guid(riid));
|
|
|
|
return E_NOINTERFACE;
|
|
|
|
}
|
|
|
|
IClientSecurity_AddRef( iface );
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI client_security_AddRef(
|
|
|
|
IClientSecurity *iface )
|
|
|
|
{
|
|
|
|
FIXME("%p\n", iface);
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI client_security_Release(
|
|
|
|
IClientSecurity *iface )
|
|
|
|
{
|
|
|
|
FIXME("%p\n", iface);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI client_security_QueryBlanket(
|
|
|
|
IClientSecurity *iface,
|
|
|
|
IUnknown *pProxy,
|
|
|
|
DWORD *pAuthnSvc,
|
|
|
|
DWORD *pAuthzSvc,
|
|
|
|
OLECHAR **pServerPrincName,
|
|
|
|
DWORD *pAuthnLevel,
|
|
|
|
DWORD *pImpLevel,
|
|
|
|
void **pAuthInfo,
|
|
|
|
DWORD *pCapabilities )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI client_security_SetBlanket(
|
|
|
|
IClientSecurity *iface,
|
|
|
|
IUnknown *pProxy,
|
|
|
|
DWORD AuthnSvc,
|
|
|
|
DWORD AuthzSvc,
|
|
|
|
OLECHAR *pServerPrincName,
|
|
|
|
DWORD AuthnLevel,
|
|
|
|
DWORD ImpLevel,
|
|
|
|
void *pAuthInfo,
|
|
|
|
DWORD Capabilities )
|
|
|
|
{
|
2012-06-29 13:40:35 +02:00
|
|
|
static const OLECHAR defaultW[] =
|
|
|
|
{'<','C','O','L','E','_','D','E','F','A','U','L','T','_','P','R','I','N','C','I','P','A','L','>',0};
|
|
|
|
const OLECHAR *princname = (pServerPrincName == COLE_DEFAULT_PRINCIPAL) ? defaultW : pServerPrincName;
|
|
|
|
|
2012-06-14 16:02:38 +02:00
|
|
|
FIXME("%p, %p, %u, %u, %s, %u, %u, %p, 0x%08x\n", iface, pProxy, AuthnSvc, AuthzSvc,
|
2012-06-29 13:40:35 +02:00
|
|
|
debugstr_w(princname), AuthnLevel, ImpLevel, pAuthInfo, Capabilities);
|
2012-06-14 16:02:38 +02:00
|
|
|
return WBEM_NO_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI client_security_CopyProxy(
|
|
|
|
IClientSecurity *iface,
|
|
|
|
IUnknown *pProxy,
|
|
|
|
IUnknown **ppCopy )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const IClientSecurityVtbl client_security_vtbl =
|
|
|
|
{
|
|
|
|
client_security_QueryInterface,
|
|
|
|
client_security_AddRef,
|
|
|
|
client_security_Release,
|
|
|
|
client_security_QueryBlanket,
|
|
|
|
client_security_SetBlanket,
|
|
|
|
client_security_CopyProxy
|
|
|
|
};
|
|
|
|
|
2012-06-28 09:24:42 +02:00
|
|
|
IClientSecurity client_security = { &client_security_vtbl };
|
2012-06-14 16:02:38 +02:00
|
|
|
|
2013-05-16 12:04:03 +02:00
|
|
|
struct async_header
|
|
|
|
{
|
|
|
|
IWbemObjectSink *sink;
|
|
|
|
void (*proc)( struct async_header * );
|
|
|
|
HANDLE cancel;
|
|
|
|
HANDLE wait;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct async_query
|
|
|
|
{
|
|
|
|
struct async_header hdr;
|
|
|
|
WCHAR *str;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void free_async( struct async_header *async )
|
|
|
|
{
|
|
|
|
if (async->sink) IWbemObjectSink_Release( async->sink );
|
|
|
|
CloseHandle( async->cancel );
|
|
|
|
CloseHandle( async->wait );
|
|
|
|
heap_free( async );
|
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL init_async( struct async_header *async, IWbemObjectSink *sink,
|
|
|
|
void (*proc)(struct async_header *) )
|
|
|
|
{
|
|
|
|
if (!(async->wait = CreateEventW( NULL, FALSE, FALSE, NULL ))) return FALSE;
|
|
|
|
if (!(async->cancel = CreateEventW( NULL, FALSE, FALSE, NULL )))
|
|
|
|
{
|
|
|
|
CloseHandle( async->wait );
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
async->proc = proc;
|
|
|
|
async->sink = sink;
|
|
|
|
IWbemObjectSink_AddRef( sink );
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static DWORD CALLBACK async_proc( LPVOID param )
|
|
|
|
{
|
|
|
|
struct async_header *async = param;
|
|
|
|
HANDLE wait = async->wait;
|
|
|
|
|
|
|
|
async->proc( async );
|
|
|
|
|
|
|
|
WaitForSingleObject( async->cancel, INFINITE );
|
|
|
|
SetEvent( wait );
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT queue_async( struct async_header *async )
|
|
|
|
{
|
|
|
|
if (QueueUserWorkItem( async_proc, async, WT_EXECUTELONGFUNCTION )) return S_OK;
|
|
|
|
return HRESULT_FROM_WIN32( GetLastError() );
|
|
|
|
}
|
|
|
|
|
2012-06-14 16:02:38 +02:00
|
|
|
struct wbem_services
|
|
|
|
{
|
|
|
|
IWbemServices IWbemServices_iface;
|
|
|
|
LONG refs;
|
2013-05-16 12:04:03 +02:00
|
|
|
CRITICAL_SECTION cs;
|
2012-06-25 11:20:21 +02:00
|
|
|
WCHAR *namespace;
|
2013-05-16 12:04:03 +02:00
|
|
|
struct async_header *async;
|
2012-06-14 16:02:38 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
static inline struct wbem_services *impl_from_IWbemServices( IWbemServices *iface )
|
|
|
|
{
|
|
|
|
return CONTAINING_RECORD( iface, struct wbem_services, IWbemServices_iface );
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI wbem_services_AddRef(
|
|
|
|
IWbemServices *iface )
|
|
|
|
{
|
|
|
|
struct wbem_services *ws = impl_from_IWbemServices( iface );
|
|
|
|
return InterlockedIncrement( &ws->refs );
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI wbem_services_Release(
|
|
|
|
IWbemServices *iface )
|
|
|
|
{
|
|
|
|
struct wbem_services *ws = impl_from_IWbemServices( iface );
|
|
|
|
LONG refs = InterlockedDecrement( &ws->refs );
|
|
|
|
if (!refs)
|
|
|
|
{
|
|
|
|
TRACE("destroying %p\n", ws);
|
2013-05-16 12:04:03 +02:00
|
|
|
|
|
|
|
EnterCriticalSection( &ws->cs );
|
|
|
|
if (ws->async) SetEvent( ws->async->cancel );
|
|
|
|
LeaveCriticalSection( &ws->cs );
|
|
|
|
if (ws->async)
|
|
|
|
{
|
|
|
|
WaitForSingleObject( ws->async->wait, INFINITE );
|
|
|
|
free_async( ws->async );
|
|
|
|
}
|
|
|
|
ws->cs.DebugInfo->Spare[0] = 0;
|
|
|
|
DeleteCriticalSection( &ws->cs );
|
2012-06-25 11:20:21 +02:00
|
|
|
heap_free( ws->namespace );
|
2012-06-15 15:46:51 +02:00
|
|
|
heap_free( ws );
|
2012-06-14 16:02:38 +02:00
|
|
|
}
|
|
|
|
return refs;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_QueryInterface(
|
|
|
|
IWbemServices *iface,
|
|
|
|
REFIID riid,
|
|
|
|
void **ppvObject )
|
|
|
|
{
|
|
|
|
struct wbem_services *ws = impl_from_IWbemServices( iface );
|
|
|
|
|
|
|
|
TRACE("%p %s %p\n", ws, debugstr_guid( riid ), ppvObject );
|
|
|
|
|
|
|
|
if ( IsEqualGUID( riid, &IID_IWbemServices ) ||
|
|
|
|
IsEqualGUID( riid, &IID_IUnknown ) )
|
|
|
|
{
|
|
|
|
*ppvObject = ws;
|
|
|
|
}
|
|
|
|
else if ( IsEqualGUID( riid, &IID_IClientSecurity ) )
|
|
|
|
{
|
|
|
|
*ppvObject = &client_security;
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
FIXME("interface %s not implemented\n", debugstr_guid(riid));
|
|
|
|
return E_NOINTERFACE;
|
|
|
|
}
|
|
|
|
IWbemServices_AddRef( iface );
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_OpenNamespace(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strNamespace,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemServices **ppWorkingNamespace,
|
|
|
|
IWbemCallResult **ppResult )
|
|
|
|
{
|
2012-06-26 16:01:43 +02:00
|
|
|
static const WCHAR cimv2W[] = {'c','i','m','v','2',0};
|
|
|
|
static const WCHAR defaultW[] = {'d','e','f','a','u','l','t',0};
|
|
|
|
struct wbem_services *ws = impl_from_IWbemServices( iface );
|
|
|
|
|
|
|
|
TRACE("%p, %s, 0x%08x, %p, %p, %p\n", iface, debugstr_w(strNamespace), lFlags,
|
|
|
|
pCtx, ppWorkingNamespace, ppResult);
|
|
|
|
|
|
|
|
if ((strcmpiW( strNamespace, cimv2W ) && strcmpiW( strNamespace, defaultW )) || ws->namespace)
|
|
|
|
return WBEM_E_INVALID_NAMESPACE;
|
|
|
|
|
2013-12-30 10:21:36 +01:00
|
|
|
return WbemServices_create( cimv2W, (void **)ppWorkingNamespace );
|
2012-06-14 16:02:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_CancelAsyncCall(
|
|
|
|
IWbemServices *iface,
|
|
|
|
IWbemObjectSink *pSink )
|
|
|
|
{
|
2013-05-16 12:04:03 +02:00
|
|
|
struct wbem_services *services = impl_from_IWbemServices( iface );
|
|
|
|
struct async_header *async;
|
|
|
|
|
|
|
|
TRACE("%p, %p\n", iface, pSink);
|
|
|
|
|
|
|
|
if (!pSink) return WBEM_E_INVALID_PARAMETER;
|
|
|
|
|
|
|
|
EnterCriticalSection( &services->cs );
|
|
|
|
|
|
|
|
if (!(async = services->async))
|
|
|
|
{
|
|
|
|
LeaveCriticalSection( &services->cs );
|
|
|
|
return WBEM_E_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
services->async = NULL;
|
|
|
|
SetEvent( async->cancel );
|
|
|
|
|
|
|
|
LeaveCriticalSection( &services->cs );
|
2013-05-14 14:01:50 +02:00
|
|
|
|
2013-05-16 12:04:03 +02:00
|
|
|
WaitForSingleObject( async->wait, INFINITE );
|
|
|
|
free_async( async );
|
2013-05-14 14:01:50 +02:00
|
|
|
return S_OK;
|
2012-06-14 16:02:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_QueryObjectSink(
|
|
|
|
IWbemServices *iface,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemObjectSink **ppResponseHandler )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
2012-07-27 14:04:28 +02:00
|
|
|
struct path
|
2012-06-14 16:02:38 +02:00
|
|
|
{
|
2012-07-27 14:04:28 +02:00
|
|
|
WCHAR *class;
|
|
|
|
UINT class_len;
|
|
|
|
WCHAR *filter;
|
|
|
|
UINT filter_len;
|
|
|
|
};
|
|
|
|
|
|
|
|
static HRESULT parse_path( const WCHAR *str, struct path **ret )
|
|
|
|
{
|
|
|
|
struct path *path;
|
|
|
|
const WCHAR *p = str, *q;
|
|
|
|
UINT len;
|
2012-06-28 09:24:21 +02:00
|
|
|
|
2012-07-27 14:04:28 +02:00
|
|
|
if (!(path = heap_alloc_zero( sizeof(*path) ))) return E_OUTOFMEMORY;
|
2012-06-28 09:24:21 +02:00
|
|
|
|
2018-10-15 19:59:21 +02:00
|
|
|
if (*p == '\\')
|
|
|
|
{
|
|
|
|
static const WCHAR cimv2W[] = {'R','O','O','T','\\','C','I','M','V','2'};
|
|
|
|
|
|
|
|
WCHAR server[MAX_COMPUTERNAME_LENGTH+1];
|
|
|
|
DWORD server_len = ARRAY_SIZE(server);
|
|
|
|
|
|
|
|
p++;
|
|
|
|
if (*p != '\\') return WBEM_E_INVALID_OBJECT_PATH;
|
|
|
|
p++;
|
|
|
|
|
|
|
|
q = p;
|
|
|
|
while (*p && *p != '\\') p++;
|
|
|
|
if (!*p) return WBEM_E_INVALID_OBJECT_PATH;
|
|
|
|
|
|
|
|
len = p - q;
|
|
|
|
if (!GetComputerNameW( server, &server_len ) || server_len != len
|
|
|
|
|| memcmp( q, server, server_len * sizeof(WCHAR) ))
|
|
|
|
return WBEM_E_NOT_SUPPORTED;
|
|
|
|
|
|
|
|
q = ++p;
|
|
|
|
while (*p && *p != ':') p++;
|
|
|
|
if (!*p) return WBEM_E_INVALID_OBJECT_PATH;
|
|
|
|
|
|
|
|
len = p - q;
|
|
|
|
if (len != ARRAY_SIZE(cimv2W) || memcmp( q, cimv2W, sizeof(cimv2W) ))
|
|
|
|
return WBEM_E_INVALID_NAMESPACE;
|
|
|
|
p++;
|
|
|
|
}
|
|
|
|
|
|
|
|
q = p;
|
2012-07-27 14:04:28 +02:00
|
|
|
while (*p && *p != '.') p++;
|
2012-06-28 09:24:21 +02:00
|
|
|
|
2018-10-15 19:59:21 +02:00
|
|
|
len = p - q;
|
2012-07-27 14:04:28 +02:00
|
|
|
if (!(path->class = heap_alloc( (len + 1) * sizeof(WCHAR) )))
|
|
|
|
{
|
|
|
|
heap_free( path );
|
|
|
|
return E_OUTOFMEMORY;
|
|
|
|
}
|
2018-10-15 19:59:21 +02:00
|
|
|
memcpy( path->class, q, len * sizeof(WCHAR) );
|
2012-07-27 14:04:28 +02:00
|
|
|
path->class[len] = 0;
|
|
|
|
path->class_len = len;
|
|
|
|
|
|
|
|
if (p[0] == '.' && p[1])
|
|
|
|
{
|
|
|
|
q = ++p;
|
|
|
|
while (*q) q++;
|
|
|
|
|
|
|
|
len = q - p;
|
|
|
|
if (!(path->filter = heap_alloc( (len + 1) * sizeof(WCHAR) )))
|
|
|
|
{
|
|
|
|
heap_free( path->class );
|
|
|
|
heap_free( path );
|
|
|
|
return E_OUTOFMEMORY;
|
|
|
|
}
|
|
|
|
memcpy( path->filter, p, len * sizeof(WCHAR) );
|
|
|
|
path->filter[len] = 0;
|
|
|
|
path->filter_len = len;
|
|
|
|
}
|
|
|
|
*ret = path;
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void free_path( struct path *path )
|
|
|
|
{
|
|
|
|
heap_free( path->class );
|
|
|
|
heap_free( path->filter );
|
|
|
|
heap_free( path );
|
|
|
|
}
|
|
|
|
|
2013-06-04 13:27:48 +02:00
|
|
|
static WCHAR *query_from_path( const struct path *path )
|
2012-07-27 14:04:28 +02:00
|
|
|
{
|
|
|
|
static const WCHAR selectW[] =
|
|
|
|
{'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ','%','s',' ',
|
|
|
|
'W','H','E','R','E',' ','%','s',0};
|
|
|
|
static const WCHAR select_allW[] =
|
|
|
|
{'S','E','L','E','C','T',' ','*',' ','F','R','O','M',' ',0};
|
|
|
|
WCHAR *query;
|
|
|
|
UINT len;
|
|
|
|
|
|
|
|
if (path->filter)
|
|
|
|
{
|
2018-11-15 20:59:48 +01:00
|
|
|
len = path->class_len + path->filter_len + ARRAY_SIZE(selectW);
|
2013-06-04 13:27:48 +02:00
|
|
|
if (!(query = heap_alloc( len * sizeof(WCHAR) ))) return NULL;
|
2012-07-27 14:04:28 +02:00
|
|
|
sprintfW( query, selectW, path->class, path->filter );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-11-15 20:59:48 +01:00
|
|
|
len = path->class_len + ARRAY_SIZE(select_allW);
|
2013-06-04 13:27:48 +02:00
|
|
|
if (!(query = heap_alloc( len * sizeof(WCHAR) ))) return NULL;
|
2012-07-27 14:04:28 +02:00
|
|
|
strcpyW( query, select_allW );
|
|
|
|
strcatW( query, path->class );
|
|
|
|
}
|
2013-06-04 13:27:48 +02:00
|
|
|
return query;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT create_instance_enum( const struct path *path, IEnumWbemClassObject **iter )
|
|
|
|
{
|
|
|
|
WCHAR *query;
|
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
if (!(query = query_from_path( path ))) return E_OUTOFMEMORY;
|
2012-07-25 13:12:28 +02:00
|
|
|
hr = exec_query( query, iter );
|
2012-06-28 09:24:21 +02:00
|
|
|
heap_free( query );
|
2012-07-25 13:12:28 +02:00
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
2012-07-27 14:04:28 +02:00
|
|
|
HRESULT get_object( const WCHAR *object_path, IWbemClassObject **obj )
|
2012-07-25 13:12:28 +02:00
|
|
|
{
|
|
|
|
IEnumWbemClassObject *iter;
|
2012-07-27 14:04:28 +02:00
|
|
|
struct path *path;
|
2012-07-25 13:12:28 +02:00
|
|
|
HRESULT hr;
|
|
|
|
|
2012-07-27 14:04:28 +02:00
|
|
|
hr = parse_path( object_path, &path );
|
2012-06-28 09:24:21 +02:00
|
|
|
if (hr != S_OK) return hr;
|
|
|
|
|
2012-07-27 14:04:28 +02:00
|
|
|
hr = create_instance_enum( path, &iter );
|
|
|
|
if (hr != S_OK)
|
|
|
|
{
|
|
|
|
free_path( path );
|
|
|
|
return hr;
|
|
|
|
}
|
2012-09-06 14:03:50 +02:00
|
|
|
hr = create_class_object( path->class, iter, 0, NULL, obj );
|
2012-06-28 09:24:21 +02:00
|
|
|
IEnumWbemClassObject_Release( iter );
|
2012-07-27 14:04:28 +02:00
|
|
|
free_path( path );
|
2012-06-28 09:24:21 +02:00
|
|
|
return hr;
|
2012-06-14 16:02:38 +02:00
|
|
|
}
|
|
|
|
|
2012-07-25 13:12:28 +02:00
|
|
|
static HRESULT WINAPI wbem_services_GetObject(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strObjectPath,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemClassObject **ppObject,
|
|
|
|
IWbemCallResult **ppCallResult )
|
|
|
|
{
|
|
|
|
TRACE("%p, %s, 0x%08x, %p, %p, %p\n", iface, debugstr_w(strObjectPath), lFlags,
|
|
|
|
pCtx, ppObject, ppCallResult);
|
|
|
|
|
|
|
|
if (lFlags) FIXME("unsupported flags 0x%08x\n", lFlags);
|
|
|
|
|
2013-01-28 15:56:40 +01:00
|
|
|
if (!strObjectPath || !strObjectPath[0])
|
|
|
|
return create_class_object( NULL, NULL, 0, NULL, ppObject );
|
|
|
|
|
2012-07-25 13:12:28 +02:00
|
|
|
return get_object( strObjectPath, ppObject );
|
|
|
|
}
|
|
|
|
|
2012-06-14 16:02:38 +02:00
|
|
|
static HRESULT WINAPI wbem_services_GetObjectAsync(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strObjectPath,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemObjectSink *pResponseHandler )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_PutClass(
|
|
|
|
IWbemServices *iface,
|
|
|
|
IWbemClassObject *pObject,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemCallResult **ppCallResult )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_PutClassAsync(
|
|
|
|
IWbemServices *iface,
|
|
|
|
IWbemClassObject *pObject,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemObjectSink *pResponseHandler )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_DeleteClass(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strClass,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemCallResult **ppCallResult )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_DeleteClassAsync(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strClass,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemObjectSink *pResponseHandler )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_CreateClassEnum(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strSuperclass,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IEnumWbemClassObject **ppEnum )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_CreateClassEnumAsync(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strSuperclass,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemObjectSink *pResponseHandler )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_PutInstance(
|
|
|
|
IWbemServices *iface,
|
|
|
|
IWbemClassObject *pInst,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemCallResult **ppCallResult )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_PutInstanceAsync(
|
|
|
|
IWbemServices *iface,
|
|
|
|
IWbemClassObject *pInst,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemObjectSink *pResponseHandler )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_DeleteInstance(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strObjectPath,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemCallResult **ppCallResult )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_DeleteInstanceAsync(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strObjectPath,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemObjectSink *pResponseHandler )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_CreateInstanceEnum(
|
|
|
|
IWbemServices *iface,
|
2012-06-26 16:02:04 +02:00
|
|
|
const BSTR strClass,
|
2012-06-14 16:02:38 +02:00
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IEnumWbemClassObject **ppEnum )
|
|
|
|
{
|
2012-07-27 14:04:28 +02:00
|
|
|
struct path *path;
|
|
|
|
HRESULT hr;
|
|
|
|
|
2012-06-26 16:02:04 +02:00
|
|
|
TRACE("%p, %s, 0%08x, %p, %p\n", iface, debugstr_w(strClass), lFlags, pCtx, ppEnum);
|
|
|
|
|
|
|
|
if (lFlags) FIXME("unsupported flags 0x%08x\n", lFlags);
|
|
|
|
|
2012-07-27 14:04:28 +02:00
|
|
|
hr = parse_path( strClass, &path );
|
|
|
|
if (hr != S_OK) return hr;
|
|
|
|
|
|
|
|
hr = create_instance_enum( path, ppEnum );
|
|
|
|
free_path( path );
|
|
|
|
return hr;
|
2012-06-14 16:02:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_CreateInstanceEnumAsync(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strFilter,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemObjectSink *pResponseHandler )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_ExecQuery(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strQueryLanguage,
|
|
|
|
const BSTR strQuery,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IEnumWbemClassObject **ppEnum )
|
|
|
|
{
|
2012-06-15 15:46:51 +02:00
|
|
|
static const WCHAR wqlW[] = {'W','Q','L',0};
|
|
|
|
|
|
|
|
TRACE("%p, %s, %s, 0x%08x, %p, %p\n", iface, debugstr_w(strQueryLanguage),
|
2012-06-14 16:02:38 +02:00
|
|
|
debugstr_w(strQuery), lFlags, pCtx, ppEnum);
|
2012-06-15 15:46:51 +02:00
|
|
|
|
2013-01-18 10:22:49 +01:00
|
|
|
if (!strQueryLanguage || !strQuery || !strQuery[0]) return WBEM_E_INVALID_PARAMETER;
|
2012-06-15 15:46:51 +02:00
|
|
|
if (strcmpiW( strQueryLanguage, wqlW )) return WBEM_E_INVALID_QUERY_TYPE;
|
|
|
|
return exec_query( strQuery, ppEnum );
|
2012-06-14 16:02:38 +02:00
|
|
|
}
|
|
|
|
|
2013-05-16 12:04:03 +02:00
|
|
|
static void async_exec_query( struct async_header *hdr )
|
|
|
|
{
|
|
|
|
struct async_query *query = (struct async_query *)hdr;
|
|
|
|
IEnumWbemClassObject *result;
|
|
|
|
IWbemClassObject *obj;
|
|
|
|
ULONG count;
|
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
hr = exec_query( query->str, &result );
|
|
|
|
if (hr == S_OK)
|
|
|
|
{
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
IEnumWbemClassObject_Next( result, WBEM_INFINITE, 1, &obj, &count );
|
|
|
|
if (!count) break;
|
|
|
|
IWbemObjectSink_Indicate( query->hdr.sink, 1, &obj );
|
|
|
|
IWbemClassObject_Release( obj );
|
|
|
|
}
|
|
|
|
IEnumWbemClassObject_Release( result );
|
|
|
|
}
|
|
|
|
IWbemObjectSink_SetStatus( query->hdr.sink, WBEM_STATUS_COMPLETE, hr, NULL, NULL );
|
|
|
|
heap_free( query->str );
|
|
|
|
}
|
|
|
|
|
2012-06-14 16:02:38 +02:00
|
|
|
static HRESULT WINAPI wbem_services_ExecQueryAsync(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strQueryLanguage,
|
|
|
|
const BSTR strQuery,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemObjectSink *pResponseHandler )
|
|
|
|
{
|
2013-05-16 12:04:03 +02:00
|
|
|
struct wbem_services *services = impl_from_IWbemServices( iface );
|
|
|
|
IWbemObjectSink *sink;
|
|
|
|
HRESULT hr = E_OUTOFMEMORY;
|
|
|
|
struct async_header *async;
|
|
|
|
struct async_query *query;
|
|
|
|
|
|
|
|
TRACE("%p, %s, %s, 0x%08x, %p, %p\n", iface, debugstr_w(strQueryLanguage), debugstr_w(strQuery),
|
|
|
|
lFlags, pCtx, pResponseHandler);
|
|
|
|
|
|
|
|
if (!pResponseHandler) return WBEM_E_INVALID_PARAMETER;
|
|
|
|
|
|
|
|
hr = IWbemObjectSink_QueryInterface( pResponseHandler, &IID_IWbemObjectSink, (void **)&sink );
|
|
|
|
if (FAILED(hr)) return hr;
|
|
|
|
|
|
|
|
EnterCriticalSection( &services->cs );
|
|
|
|
|
|
|
|
if (services->async)
|
|
|
|
{
|
|
|
|
FIXME("handle more than one pending async\n");
|
|
|
|
hr = WBEM_E_FAILED;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
if (!(query = heap_alloc_zero( sizeof(*query) ))) goto done;
|
|
|
|
async = (struct async_header *)query;
|
|
|
|
|
|
|
|
if (!(init_async( async, sink, async_exec_query )))
|
|
|
|
{
|
|
|
|
free_async( async );
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
if (!(query->str = heap_strdupW( strQuery )))
|
|
|
|
{
|
|
|
|
free_async( async );
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
hr = queue_async( async );
|
|
|
|
if (hr == S_OK) services->async = async;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
heap_free( query->str );
|
|
|
|
free_async( async );
|
|
|
|
}
|
|
|
|
|
|
|
|
done:
|
|
|
|
LeaveCriticalSection( &services->cs );
|
|
|
|
IWbemObjectSink_Release( sink );
|
|
|
|
return hr;
|
2012-06-14 16:02:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_ExecNotificationQuery(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strQueryLanguage,
|
|
|
|
const BSTR strQuery,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IEnumWbemClassObject **ppEnum )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_ExecNotificationQueryAsync(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strQueryLanguage,
|
|
|
|
const BSTR strQuery,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemObjectSink *pResponseHandler )
|
|
|
|
{
|
2013-05-16 12:04:03 +02:00
|
|
|
struct wbem_services *services = impl_from_IWbemServices( iface );
|
|
|
|
IWbemObjectSink *sink;
|
|
|
|
HRESULT hr = E_OUTOFMEMORY;
|
|
|
|
struct async_header *async;
|
|
|
|
struct async_query *query;
|
|
|
|
|
|
|
|
TRACE("%p, %s, %s, 0x%08x, %p, %p\n", iface, debugstr_w(strQueryLanguage), debugstr_w(strQuery),
|
2013-05-14 14:01:50 +02:00
|
|
|
lFlags, pCtx, pResponseHandler);
|
|
|
|
|
2013-05-16 12:04:03 +02:00
|
|
|
if (!pResponseHandler) return WBEM_E_INVALID_PARAMETER;
|
|
|
|
|
|
|
|
hr = IWbemObjectSink_QueryInterface( pResponseHandler, &IID_IWbemObjectSink, (void **)&sink );
|
|
|
|
if (FAILED(hr)) return hr;
|
|
|
|
|
|
|
|
EnterCriticalSection( &services->cs );
|
|
|
|
|
|
|
|
if (services->async)
|
|
|
|
{
|
|
|
|
FIXME("handle more than one pending async\n");
|
|
|
|
hr = WBEM_E_FAILED;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
if (!(query = heap_alloc_zero( sizeof(*query) ))) goto done;
|
|
|
|
async = (struct async_header *)query;
|
|
|
|
|
|
|
|
if (!(init_async( async, sink, async_exec_query )))
|
|
|
|
{
|
|
|
|
free_async( async );
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
if (!(query->str = heap_strdupW( strQuery )))
|
|
|
|
{
|
|
|
|
free_async( async );
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
hr = queue_async( async );
|
|
|
|
if (hr == S_OK) services->async = async;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
heap_free( query->str );
|
|
|
|
free_async( async );
|
|
|
|
}
|
|
|
|
|
|
|
|
done:
|
|
|
|
LeaveCriticalSection( &services->cs );
|
|
|
|
IWbemObjectSink_Release( sink );
|
|
|
|
return hr;
|
2012-06-14 16:02:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_ExecMethod(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strObjectPath,
|
|
|
|
const BSTR strMethodName,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemClassObject *pInParams,
|
|
|
|
IWbemClassObject **ppOutParams,
|
|
|
|
IWbemCallResult **ppCallResult )
|
|
|
|
{
|
2013-06-04 13:27:48 +02:00
|
|
|
IEnumWbemClassObject *result = NULL;
|
|
|
|
IWbemClassObject *obj = NULL;
|
|
|
|
struct query *query = NULL;
|
2012-09-28 13:37:21 +02:00
|
|
|
struct path *path;
|
2013-06-04 13:27:48 +02:00
|
|
|
WCHAR *str;
|
|
|
|
class_method *func;
|
2012-09-28 13:37:21 +02:00
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
TRACE("%p, %s, %s, %08x, %p, %p, %p, %p\n", iface, debugstr_w(strObjectPath),
|
|
|
|
debugstr_w(strMethodName), lFlags, pCtx, pInParams, ppOutParams, ppCallResult);
|
|
|
|
|
|
|
|
if (lFlags) FIXME("flags %08x not supported\n", lFlags);
|
|
|
|
|
2013-06-04 13:27:48 +02:00
|
|
|
if ((hr = parse_path( strObjectPath, &path )) != S_OK) return hr;
|
|
|
|
if (!(str = query_from_path( path )))
|
2012-10-15 16:18:59 +02:00
|
|
|
{
|
2013-06-04 13:27:48 +02:00
|
|
|
hr = E_OUTOFMEMORY;
|
|
|
|
goto done;
|
2012-10-15 16:18:59 +02:00
|
|
|
}
|
2013-06-04 13:27:48 +02:00
|
|
|
if (!(query = create_query()))
|
2012-10-15 16:18:59 +02:00
|
|
|
{
|
2013-06-04 13:27:48 +02:00
|
|
|
hr = E_OUTOFMEMORY;
|
|
|
|
goto done;
|
2012-10-15 16:18:59 +02:00
|
|
|
}
|
2013-06-04 13:27:48 +02:00
|
|
|
hr = parse_query( str, &query->view, &query->mem );
|
|
|
|
if (hr != S_OK) goto done;
|
|
|
|
|
|
|
|
hr = execute_view( query->view );
|
|
|
|
if (hr != S_OK) goto done;
|
|
|
|
|
2013-12-30 10:21:36 +01:00
|
|
|
hr = EnumWbemClassObject_create( query, (void **)&result );
|
2013-06-04 13:27:48 +02:00
|
|
|
if (hr != S_OK) goto done;
|
|
|
|
|
|
|
|
hr = create_class_object( query->view->table->name, result, 0, NULL, &obj );
|
|
|
|
if (hr != S_OK) goto done;
|
|
|
|
|
|
|
|
hr = get_method( query->view->table, strMethodName, &func );
|
|
|
|
if (hr != S_OK) goto done;
|
|
|
|
|
2012-10-15 16:18:59 +02:00
|
|
|
hr = func( obj, pInParams, ppOutParams );
|
2013-06-04 13:27:48 +02:00
|
|
|
|
|
|
|
done:
|
|
|
|
if (result) IEnumWbemClassObject_Release( result );
|
|
|
|
if (obj) IWbemClassObject_Release( obj );
|
|
|
|
free_query( query );
|
|
|
|
free_path( path );
|
|
|
|
heap_free( str );
|
2012-10-15 16:18:59 +02:00
|
|
|
return hr;
|
2012-06-14 16:02:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI wbem_services_ExecMethodAsync(
|
|
|
|
IWbemServices *iface,
|
|
|
|
const BSTR strObjectPath,
|
|
|
|
const BSTR strMethodName,
|
|
|
|
LONG lFlags,
|
|
|
|
IWbemContext *pCtx,
|
|
|
|
IWbemClassObject *pInParams,
|
|
|
|
IWbemObjectSink *pResponseHandler )
|
|
|
|
{
|
|
|
|
FIXME("\n");
|
|
|
|
return WBEM_E_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const IWbemServicesVtbl wbem_services_vtbl =
|
|
|
|
{
|
|
|
|
wbem_services_QueryInterface,
|
|
|
|
wbem_services_AddRef,
|
|
|
|
wbem_services_Release,
|
|
|
|
wbem_services_OpenNamespace,
|
|
|
|
wbem_services_CancelAsyncCall,
|
|
|
|
wbem_services_QueryObjectSink,
|
|
|
|
wbem_services_GetObject,
|
|
|
|
wbem_services_GetObjectAsync,
|
|
|
|
wbem_services_PutClass,
|
|
|
|
wbem_services_PutClassAsync,
|
|
|
|
wbem_services_DeleteClass,
|
|
|
|
wbem_services_DeleteClassAsync,
|
|
|
|
wbem_services_CreateClassEnum,
|
|
|
|
wbem_services_CreateClassEnumAsync,
|
|
|
|
wbem_services_PutInstance,
|
|
|
|
wbem_services_PutInstanceAsync,
|
|
|
|
wbem_services_DeleteInstance,
|
|
|
|
wbem_services_DeleteInstanceAsync,
|
|
|
|
wbem_services_CreateInstanceEnum,
|
|
|
|
wbem_services_CreateInstanceEnumAsync,
|
|
|
|
wbem_services_ExecQuery,
|
|
|
|
wbem_services_ExecQueryAsync,
|
|
|
|
wbem_services_ExecNotificationQuery,
|
|
|
|
wbem_services_ExecNotificationQueryAsync,
|
|
|
|
wbem_services_ExecMethod,
|
|
|
|
wbem_services_ExecMethodAsync
|
|
|
|
};
|
|
|
|
|
2013-12-30 10:21:36 +01:00
|
|
|
HRESULT WbemServices_create( const WCHAR *namespace, LPVOID *ppObj )
|
2012-06-14 16:02:38 +02:00
|
|
|
{
|
|
|
|
struct wbem_services *ws;
|
|
|
|
|
2013-12-30 10:21:36 +01:00
|
|
|
TRACE("(%p)\n", ppObj);
|
2012-06-14 16:02:38 +02:00
|
|
|
|
2012-06-15 15:46:51 +02:00
|
|
|
ws = heap_alloc( sizeof(*ws) );
|
2012-06-14 16:02:38 +02:00
|
|
|
if (!ws) return E_OUTOFMEMORY;
|
|
|
|
|
|
|
|
ws->IWbemServices_iface.lpVtbl = &wbem_services_vtbl;
|
2013-05-16 12:04:03 +02:00
|
|
|
ws->refs = 1;
|
2012-06-26 16:01:43 +02:00
|
|
|
ws->namespace = heap_strdupW( namespace );
|
2013-05-16 12:04:03 +02:00
|
|
|
ws->async = NULL;
|
|
|
|
InitializeCriticalSection( &ws->cs );
|
|
|
|
ws->cs.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": wbemprox_services.cs");
|
2012-06-14 16:02:38 +02:00
|
|
|
|
|
|
|
*ppObj = &ws->IWbemServices_iface;
|
|
|
|
|
|
|
|
TRACE("returning iface %p\n", *ppObj);
|
|
|
|
return S_OK;
|
|
|
|
}
|