wine/dlls/wmiutils/path.c
2013-01-15 12:13:36 +01:00

709 lines
17 KiB
C

/*
* 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 "ole2.h"
#include "wbemcli.h"
#include "wmiutils.h"
#include "wine/debug.h"
#include "wmiutils_private.h"
WINE_DEFAULT_DEBUG_CHANNEL(wmiutils);
struct path
{
IWbemPath IWbemPath_iface;
LONG refs;
WCHAR *text;
int len_text;
WCHAR *server;
int len_server;
WCHAR **namespaces;
int *len_namespaces;
int num_namespaces;
WCHAR *class;
int len_class;
ULONGLONG flags;
};
static void init_path( struct path *path )
{
path->text = NULL;
path->len_text = 0;
path->server = NULL;
path->len_server = 0;
path->namespaces = NULL;
path->len_namespaces = NULL;
path->num_namespaces = 0;
path->class = NULL;
path->len_class = 0;
path->flags = 0;
}
static void clear_path( struct path *path )
{
heap_free( path->text );
heap_free( path->server );
heap_free( path->namespaces );
heap_free( path->len_namespaces );
heap_free( path->class );
init_path( path );
}
static inline struct path *impl_from_IWbemPath( IWbemPath *iface )
{
return CONTAINING_RECORD(iface, struct path, IWbemPath_iface);
}
static ULONG WINAPI path_AddRef(
IWbemPath *iface )
{
struct path *path = impl_from_IWbemPath( iface );
return InterlockedIncrement( &path->refs );
}
static ULONG WINAPI path_Release(
IWbemPath *iface )
{
struct path *path = impl_from_IWbemPath( iface );
LONG refs = InterlockedDecrement( &path->refs );
if (!refs)
{
TRACE("destroying %p\n", path);
clear_path( path );
heap_free( path );
}
return refs;
}
static HRESULT WINAPI path_QueryInterface(
IWbemPath *iface,
REFIID riid,
void **ppvObject )
{
struct path *path = impl_from_IWbemPath( iface );
TRACE("%p, %s, %p\n", path, debugstr_guid( riid ), ppvObject );
if ( IsEqualGUID( riid, &IID_IWbemPath ) ||
IsEqualGUID( riid, &IID_IUnknown ) )
{
*ppvObject = iface;
}
else
{
FIXME("interface %s not implemented\n", debugstr_guid(riid));
return E_NOINTERFACE;
}
IWbemPath_AddRef( iface );
return S_OK;
}
static HRESULT parse_text( struct path *path, ULONG mode, const WCHAR *text )
{
HRESULT hr = E_OUTOFMEMORY;
const WCHAR *p, *q;
unsigned int i, len;
p = q = text;
if ((p[0] == '\\' && p[1] == '\\') || (p[0] == '/' && p[1] == '/'))
{
p += 2;
q = p;
while (*q && *q != '\\' && *q != '/') q++;
len = q - p;
if (!(path->server = heap_alloc( (len + 1) * sizeof(WCHAR) ))) goto done;
memcpy( path->server, p, len * sizeof(WCHAR) );
path->server[len] = 0;
path->len_server = len;
path->flags |= WBEMPATH_INFO_PATH_HAD_SERVER;
}
p = q;
while (*q && *q != ':')
{
if (*q == '\\' || *q == '/') path->num_namespaces++;
q++;
}
if (path->num_namespaces)
{
if (!(path->namespaces = heap_alloc( path->num_namespaces * sizeof(WCHAR *) ))) goto done;
if (!(path->len_namespaces = heap_alloc( path->num_namespaces * sizeof(int) ))) goto done;
i = 0;
q = p;
while (*q && *q != ':')
{
if (*q == '\\' || *q == '/')
{
p = q + 1;
while (*p && *p != '\\' && *p != '/' && *p != ':') p++;
len = p - q - 1;
if (!(path->namespaces[i] = heap_alloc( (len + 1) * sizeof(WCHAR) ))) goto done;
memcpy( path->namespaces[i], q + 1, len * sizeof(WCHAR) );
path->namespaces[i][len] = 0;
path->len_namespaces[i] = len;
i++;
}
q++;
}
}
if (*q == ':') q++;
p = q;
while (*q && *q != '.') q++;
len = q - p;
if (!(path->class = heap_alloc( (len + 1) * sizeof(WCHAR) ))) goto done;
memcpy( path->class, p, len * sizeof(WCHAR) );
path->class[len] = 0;
path->len_class = len;
if (*q == '.') FIXME("handle key list\n");
hr = S_OK;
done:
if (hr != S_OK) clear_path( path );
else path->flags |= WBEMPATH_INFO_CIM_COMPLIANT | WBEMPATH_INFO_V2_COMPLIANT;
return hr;
}
static HRESULT WINAPI path_SetText(
IWbemPath *iface,
ULONG uMode,
LPCWSTR pszPath)
{
struct path *path = impl_from_IWbemPath( iface );
HRESULT hr;
int len;
TRACE("%p, %u, %s\n", iface, uMode, debugstr_w(pszPath));
if (!uMode || !pszPath) return WBEM_E_INVALID_PARAMETER;
clear_path( path );
if (!pszPath[0]) return S_OK;
if ((hr = parse_text( path, uMode, pszPath )) != S_OK) return hr;
len = strlenW( pszPath );
if (!(path->text = heap_alloc( (len + 1) * sizeof(WCHAR) )))
{
clear_path( path );
return E_OUTOFMEMORY;
}
strcpyW( path->text, pszPath );
path->len_text = len;
return S_OK;
}
static WCHAR *build_namespace( struct path *path, int *len, BOOL leading_slash )
{
WCHAR *ret, *p;
int i;
*len = 0;
for (i = 0; i < path->num_namespaces; i++)
{
if (i > 0 || leading_slash) *len += 1;
*len += path->len_namespaces[i];
}
if (!(p = ret = heap_alloc( (*len + 1) * sizeof(WCHAR) ))) return NULL;
for (i = 0; i < path->num_namespaces; i++)
{
if (i > 0 || leading_slash) *p++ = '\\';
memcpy( p, path->namespaces[i], path->len_namespaces[i] * sizeof(WCHAR) );
p += path->len_namespaces[i];
}
*p = 0;
return ret;
}
static WCHAR *build_server( struct path *path, int *len )
{
WCHAR *ret, *p;
*len = 0;
if (path->len_server) *len += 2 + path->len_server;
else *len += 3;
if (!(p = ret = heap_alloc( (*len + 1) * sizeof(WCHAR) ))) return NULL;
if (path->len_server)
{
p[0] = p[1] = '\\';
strcpyW( p + 2, path->server );
}
else
{
p[0] = p[1] = '\\';
p[2] = '.';
}
return ret;
}
static WCHAR *build_path( struct path *path, LONG flags, int *len )
{
switch (flags)
{
case 0:
{
int len_namespace;
WCHAR *ret, *namespace = build_namespace( path, &len_namespace, FALSE );
if (!namespace) return NULL;
*len = len_namespace;
if (path->len_class) *len += 1 + path->len_class;
if (!(ret = heap_alloc( (*len + 1) * sizeof(WCHAR) )))
{
heap_free( namespace );
return NULL;
}
strcpyW( ret, namespace );
if (path->len_class)
{
ret[len_namespace] = ':';
strcpyW( ret + len_namespace + 1, path->class );
}
heap_free( namespace );
return ret;
}
case WBEMPATH_GET_RELATIVE_ONLY:
if (!path->len_class)
{
*len = 0;
return NULL;
}
*len = path->len_class;
return strdupW( path->class );
case WBEMPATH_GET_SERVER_TOO:
{
int len_namespace, len_server;
WCHAR *p, *ret, *namespace = build_namespace( path, &len_namespace, TRUE );
WCHAR *server = build_server( path, &len_server );
if (!namespace || !server)
{
heap_free( namespace );
heap_free( server );
return NULL;
}
*len = len_namespace + len_server;
if (path->len_class) *len += 1 + path->len_class;
if (!(p = ret = heap_alloc( (*len + 1) * sizeof(WCHAR) )))
{
heap_free( namespace );
heap_free( server );
return NULL;
}
strcpyW( p, server );
p += len_server;
strcpyW( p, namespace );
p += len_namespace;
if (path->len_class)
{
*p = ':';
strcpyW( p + 1, path->class );
}
heap_free( namespace );
heap_free( server );
return ret;
}
case WBEMPATH_GET_SERVER_AND_NAMESPACE_ONLY:
{
int len_namespace, len_server;
WCHAR *p, *ret, *namespace = build_namespace( path, &len_namespace, TRUE );
WCHAR *server = build_server( path, &len_server );
if (!namespace || !server)
{
heap_free( namespace );
heap_free( server );
return NULL;
}
*len = len_namespace + len_server;
if (!(p = ret = heap_alloc( (*len + 1) * sizeof(WCHAR) )))
{
heap_free( namespace );
heap_free( server );
return NULL;
}
strcpyW( p, server );
p += len_server;
strcpyW( p, namespace );
heap_free( namespace );
heap_free( server );
return ret;
}
case WBEMPATH_GET_NAMESPACE_ONLY:
return build_namespace( path, len, FALSE );
case WBEMPATH_GET_ORIGINAL:
if (!path->len_text)
{
*len = 0;
return NULL;
}
*len = path->len_text;
return strdupW( path->text );
default:
ERR("unhandled flags 0x%x\n", flags);
return NULL;
}
}
static HRESULT WINAPI path_GetText(
IWbemPath *iface,
LONG lFlags,
ULONG *puBufferLength,
LPWSTR pszText)
{
struct path *path = impl_from_IWbemPath( iface );
WCHAR *str;
int len;
TRACE("%p, 0x%x, %p, %p\n", iface, lFlags, puBufferLength, pszText);
if (!puBufferLength) return WBEM_E_INVALID_PARAMETER;
str = build_path( path, lFlags, &len );
if (*puBufferLength < len + 1)
{
*puBufferLength = len + 1;
return S_OK;
}
if (!pszText)
{
heap_free( str );
return WBEM_E_INVALID_PARAMETER;
}
if (str) strcpyW( pszText, str );
else pszText[0] = 0;
*puBufferLength = len + 1;
TRACE("<-- %s\n", debugstr_w(pszText));
heap_free( str );
return S_OK;
}
static HRESULT WINAPI path_GetInfo(
IWbemPath *iface,
ULONG info,
ULONGLONG *response)
{
struct path *path = impl_from_IWbemPath( iface );
TRACE("%p, %u, %p\n", iface, info, response);
if (info || !response) return WBEM_E_INVALID_PARAMETER;
FIXME("some flags are not implemented\n");
*response = path->flags;
if (!path->server || (path->len_server == 1 && path->server[0] == '.'))
*response |= WBEMPATH_INFO_ANON_LOCAL_MACHINE;
else
*response |= WBEMPATH_INFO_HAS_MACHINE_NAME;
if (!path->class)
*response |= WBEMPATH_INFO_SERVER_NAMESPACE_ONLY;
else
{
*response |= WBEMPATH_INFO_HAS_SUBSCOPES;
if (path->text && strchrW( path->text, '=' )) /* FIXME */
*response |= WBEMPATH_INFO_IS_INST_REF;
else
*response |= WBEMPATH_INFO_IS_CLASS_REF;
}
return S_OK;
}
static HRESULT WINAPI path_SetServer(
IWbemPath *iface,
LPCWSTR Name)
{
FIXME("%p, %s\n", iface, debugstr_w(Name));
return E_NOTIMPL;
}
static HRESULT WINAPI path_GetServer(
IWbemPath *iface,
ULONG *len,
LPWSTR name)
{
struct path *path = impl_from_IWbemPath( iface );
TRACE("%p, %p, %p\n", iface, len, name);
if (!len || (*len && !name)) return WBEM_E_INVALID_PARAMETER;
if (!path->class) return WBEM_E_NOT_AVAILABLE;
if (*len > path->len_server) strcpyW( name, path->server );
*len = path->len_server + 1;
return S_OK;
}
static HRESULT WINAPI path_GetNamespaceCount(
IWbemPath *iface,
ULONG *puCount)
{
struct path *path = impl_from_IWbemPath( iface );
TRACE("%p, %p\n", iface, puCount);
if (!puCount) return WBEM_E_INVALID_PARAMETER;
*puCount = path->num_namespaces;
return S_OK;
}
static HRESULT WINAPI path_SetNamespaceAt(
IWbemPath *iface,
ULONG uIndex,
LPCWSTR pszName)
{
FIXME("%p, %u, %s\n", iface, uIndex, debugstr_w(pszName));
return E_NOTIMPL;
}
static HRESULT WINAPI path_GetNamespaceAt(
IWbemPath *iface,
ULONG uIndex,
ULONG *puNameBufLength,
LPWSTR pName)
{
FIXME("%p, %u, %p, %p\n", iface, uIndex, puNameBufLength, pName);
return E_NOTIMPL;
}
static HRESULT WINAPI path_RemoveNamespaceAt(
IWbemPath *iface,
ULONG uIndex)
{
FIXME("%p, %u\n", iface, uIndex);
return E_NOTIMPL;
}
static HRESULT WINAPI path_RemoveAllNamespaces(
IWbemPath *iface)
{
FIXME("%p\n", iface);
return E_NOTIMPL;
}
static HRESULT WINAPI path_GetScopeCount(
IWbemPath *iface,
ULONG *puCount)
{
FIXME("%p, %p\n", iface, puCount);
return E_NOTIMPL;
}
static HRESULT WINAPI path_SetScope(
IWbemPath *iface,
ULONG uIndex,
LPWSTR pszClass)
{
FIXME("%p, %u, %s\n", iface, uIndex, debugstr_w(pszClass));
return E_NOTIMPL;
}
static HRESULT WINAPI path_SetScopeFromText(
IWbemPath *iface,
ULONG uIndex,
LPWSTR pszText)
{
FIXME("%p, %u, %s\n", iface, uIndex, debugstr_w(pszText));
return E_NOTIMPL;
}
static HRESULT WINAPI path_GetScope(
IWbemPath *iface,
ULONG uIndex,
ULONG *puClassNameBufSize,
LPWSTR pszClass,
IWbemPathKeyList **pKeyList)
{
FIXME("%p, %u, %p, %p, %p\n", iface, uIndex, puClassNameBufSize, pszClass, pKeyList);
return E_NOTIMPL;
}
static HRESULT WINAPI path_GetScopeAsText(
IWbemPath *iface,
ULONG uIndex,
ULONG *puTextBufSize,
LPWSTR pszText)
{
FIXME("%p, %u, %p, %p\n", iface, uIndex, puTextBufSize, pszText);
return E_NOTIMPL;
}
static HRESULT WINAPI path_RemoveScope(
IWbemPath *iface,
ULONG uIndex)
{
FIXME("%p, %u\n", iface, uIndex);
return E_NOTIMPL;
}
static HRESULT WINAPI path_RemoveAllScopes(
IWbemPath *iface)
{
FIXME("%p\n", iface);
return E_NOTIMPL;
}
static HRESULT WINAPI path_SetClassName(
IWbemPath *iface,
LPCWSTR Name)
{
FIXME("%p, %s\n", iface, debugstr_w(Name));
return E_NOTIMPL;
}
static HRESULT WINAPI path_GetClassName(
IWbemPath *iface,
ULONG *len,
LPWSTR name)
{
struct path *path = impl_from_IWbemPath( iface );
TRACE("%p, %p, %p\n", iface, len, name);
if (!len || (*len && !name)) return WBEM_E_INVALID_PARAMETER;
if (!path->class) return WBEM_E_INVALID_OBJECT_PATH;
if (*len > path->len_class) strcpyW( name, path->class );
*len = path->len_class + 1;
return S_OK;
}
static HRESULT WINAPI path_GetKeyList(
IWbemPath *iface,
IWbemPathKeyList **pOut)
{
FIXME("%p, %p\n", iface, pOut);
return E_NOTIMPL;
}
static HRESULT WINAPI path_CreateClassPart(
IWbemPath *iface,
LONG lFlags,
LPCWSTR Name)
{
FIXME("%p, 0x%x, %s\n", iface, lFlags, debugstr_w(Name));
return E_NOTIMPL;
}
static HRESULT WINAPI path_DeleteClassPart(
IWbemPath *iface,
LONG lFlags)
{
FIXME("%p, 0x%x\n", iface, lFlags);
return E_NOTIMPL;
}
static BOOL WINAPI path_IsRelative(
IWbemPath *iface,
LPWSTR wszMachine,
LPWSTR wszNamespace)
{
FIXME("%p, %s, %s\n", iface, debugstr_w(wszMachine), debugstr_w(wszNamespace));
return E_NOTIMPL;
}
static BOOL WINAPI path_IsRelativeOrChild(
IWbemPath *iface,
LPWSTR wszMachine,
LPWSTR wszNamespace,
LONG lFlags)
{
FIXME("%p, %s, %s, 0x%x\n", iface, debugstr_w(wszMachine), debugstr_w(wszNamespace), lFlags);
return E_NOTIMPL;
}
static BOOL WINAPI path_IsLocal(
IWbemPath *iface,
LPCWSTR wszMachine)
{
FIXME("%p, %s\n", iface, debugstr_w(wszMachine));
return E_NOTIMPL;
}
static BOOL WINAPI path_IsSameClassName(
IWbemPath *iface,
LPCWSTR wszClass)
{
FIXME("%p, %s\n", iface, debugstr_w(wszClass));
return E_NOTIMPL;
}
static const struct IWbemPathVtbl path_vtbl =
{
path_QueryInterface,
path_AddRef,
path_Release,
path_SetText,
path_GetText,
path_GetInfo,
path_SetServer,
path_GetServer,
path_GetNamespaceCount,
path_SetNamespaceAt,
path_GetNamespaceAt,
path_RemoveNamespaceAt,
path_RemoveAllNamespaces,
path_GetScopeCount,
path_SetScope,
path_SetScopeFromText,
path_GetScope,
path_GetScopeAsText,
path_RemoveScope,
path_RemoveAllScopes,
path_SetClassName,
path_GetClassName,
path_GetKeyList,
path_CreateClassPart,
path_DeleteClassPart,
path_IsRelative,
path_IsRelativeOrChild,
path_IsLocal,
path_IsSameClassName
};
HRESULT WbemPath_create( IUnknown *pUnkOuter, LPVOID *ppObj )
{
struct path *path;
TRACE("%p, %p\n", pUnkOuter, ppObj);
if (!(path = heap_alloc( sizeof(*path) ))) return E_OUTOFMEMORY;
path->IWbemPath_iface.lpVtbl = &path_vtbl;
path->refs = 1;
init_path( path );
*ppObj = &path->IWbemPath_iface;
TRACE("returning iface %p\n", *ppObj);
return S_OK;
}