2 * StdRegProv implementation
4 * Copyright 2012 Hans Leidekker for CodeWeavers
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
29 #include "wine/debug.h"
30 #include "wbemprox_private.h"
36 WINE_DEFAULT_DEBUG_CHANNEL(wbemprox
);
38 static HRESULT
to_bstr_array( BSTR
*strings
, DWORD count
, VARIANT
*var
)
44 if (!(sa
= SafeArrayCreateVector( VT_BSTR
, 0, count
))) return E_OUTOFMEMORY
;
45 for (i
= 0; i
< count
; i
++)
47 if ((hr
= SafeArrayPutElement( sa
, &i
, strings
[i
] )) != S_OK
)
49 SafeArrayDestroy( sa
);
53 set_variant( VT_BSTR
|VT_ARRAY
, 0, sa
, var
);
57 static void free_bstr_array( BSTR
*strings
, DWORD count
)
60 SysFreeString( *(strings
++) );
63 static HRESULT
to_i4_array( DWORD
*values
, DWORD count
, VARIANT
*var
)
69 if (!(sa
= SafeArrayCreateVector( VT_I4
, 0, count
))) return E_OUTOFMEMORY
;
70 for (i
= 0; i
< count
; i
++)
72 if ((hr
= SafeArrayPutElement( sa
, &i
, &values
[i
] )) != S_OK
)
74 SafeArrayDestroy( sa
);
78 set_variant( VT_I4
|VT_ARRAY
, 0, sa
, var
);
82 static HRESULT
create_key( HKEY root
, const WCHAR
*subkey
, VARIANT
*retval
)
87 TRACE("%p, %s\n", root
, debugstr_w(subkey
));
89 res
= RegCreateKeyExW( root
, subkey
, 0, NULL
, 0, 0, NULL
, &hkey
, NULL
);
90 set_variant( VT_UI4
, res
, NULL
, retval
);
96 return HRESULT_FROM_WIN32( res
);
99 HRESULT
reg_create_key( IWbemClassObject
*obj
, IWbemClassObject
*in
, IWbemClassObject
**out
)
101 VARIANT defkey
, subkey
, retval
;
102 IWbemClassObject
*sig
, *out_params
= NULL
;
105 TRACE("%p, %p\n", in
, out
);
107 hr
= IWbemClassObject_Get( in
, param_defkeyW
, 0, &defkey
, NULL
, NULL
);
108 if (hr
!= S_OK
) return hr
;
109 hr
= IWbemClassObject_Get( in
, param_subkeynameW
, 0, &subkey
, NULL
, NULL
);
110 if (hr
!= S_OK
) return hr
;
112 hr
= create_signature( class_stdregprovW
, method_createkeyW
, PARAM_OUT
, &sig
);
115 VariantClear( &subkey
);
120 hr
= IWbemClassObject_SpawnInstance( sig
, 0, &out_params
);
123 VariantClear( &subkey
);
124 IWbemClassObject_Release( sig
);
128 hr
= create_key( (HKEY
)(INT_PTR
)V_I4(&defkey
), V_BSTR(&subkey
), &retval
);
129 if (hr
== S_OK
&& out_params
)
130 hr
= IWbemClassObject_Put( out_params
, param_returnvalueW
, 0, &retval
, CIM_UINT32
);
132 VariantClear( &subkey
);
133 IWbemClassObject_Release( sig
);
134 if (hr
== S_OK
&& out
)
137 IWbemClassObject_AddRef( out_params
);
139 if (out_params
) IWbemClassObject_Release( out_params
);
143 static HRESULT
enum_key( HKEY root
, const WCHAR
*subkey
, VARIANT
*names
, VARIANT
*retval
)
149 DWORD count
= 2, len
= ARRAY_SIZE( buf
);
152 TRACE("%p, %s\n", root
, debugstr_w(subkey
));
154 if (!(strings
= heap_alloc( count
* sizeof(BSTR
) ))) return E_OUTOFMEMORY
;
155 if ((res
= RegOpenKeyExW( root
, subkey
, 0, KEY_ENUMERATE_SUB_KEYS
, &hkey
)))
157 set_variant( VT_UI4
, res
, NULL
, retval
);
158 heap_free( strings
);
166 if (!(tmp
= heap_realloc( strings
, count
* sizeof(BSTR
) )))
169 return E_OUTOFMEMORY
;
173 if ((res
= RegEnumKeyW( hkey
, i
, buf
, len
)) == ERROR_NO_MORE_ITEMS
)
175 if (i
) res
= ERROR_SUCCESS
;
179 if (!(strings
[i
] = SysAllocString( buf
)))
181 for (i
--; i
>= 0; i
--) SysFreeString( strings
[i
] );
187 if (hr
== S_OK
&& !res
)
189 hr
= to_bstr_array( strings
, i
, names
);
190 free_bstr_array( strings
, i
);
192 set_variant( VT_UI4
, res
, NULL
, retval
);
194 heap_free( strings
);
198 HRESULT
reg_enum_key( IWbemClassObject
*obj
, IWbemClassObject
*in
, IWbemClassObject
**out
)
200 VARIANT defkey
, subkey
, names
, retval
;
201 IWbemClassObject
*sig
, *out_params
= NULL
;
204 TRACE("%p, %p\n", in
, out
);
206 hr
= IWbemClassObject_Get( in
, param_defkeyW
, 0, &defkey
, NULL
, NULL
);
207 if (hr
!= S_OK
) return hr
;
208 hr
= IWbemClassObject_Get( in
, param_subkeynameW
, 0, &subkey
, NULL
, NULL
);
209 if (hr
!= S_OK
) return hr
;
211 hr
= create_signature( class_stdregprovW
, method_enumkeyW
, PARAM_OUT
, &sig
);
214 VariantClear( &subkey
);
219 hr
= IWbemClassObject_SpawnInstance( sig
, 0, &out_params
);
222 VariantClear( &subkey
);
223 IWbemClassObject_Release( sig
);
227 VariantInit( &names
);
228 hr
= enum_key( (HKEY
)(INT_PTR
)V_I4(&defkey
), V_BSTR(&subkey
), &names
, &retval
);
229 if (hr
!= S_OK
) goto done
;
232 if (!V_UI4( &retval
))
234 hr
= IWbemClassObject_Put( out_params
, param_namesW
, 0, &names
, CIM_STRING
|CIM_FLAG_ARRAY
);
235 if (hr
!= S_OK
) goto done
;
237 hr
= IWbemClassObject_Put( out_params
, param_returnvalueW
, 0, &retval
, CIM_UINT32
);
241 VariantClear( &names
);
242 VariantClear( &subkey
);
243 IWbemClassObject_Release( sig
);
244 if (hr
== S_OK
&& out
)
247 IWbemClassObject_AddRef( out_params
);
249 if (out_params
) IWbemClassObject_Release( out_params
);
253 static HRESULT
enum_values( HKEY root
, const WCHAR
*subkey
, VARIANT
*names
, VARIANT
*types
, VARIANT
*retval
)
257 BSTR
*value_names
= NULL
;
258 DWORD count
, buflen
, len
, *value_types
= NULL
;
262 TRACE("%p, %s\n", root
, debugstr_w(subkey
));
264 if ((res
= RegOpenKeyExW( root
, subkey
, 0, KEY_QUERY_VALUE
, &hkey
))) goto done
;
265 if ((res
= RegQueryInfoKeyW( hkey
, NULL
, NULL
, NULL
, NULL
, NULL
, NULL
, &count
, &buflen
, NULL
, NULL
, NULL
)))
269 if (!(buf
= heap_alloc( (buflen
+ 1) * sizeof(WCHAR
) ))) goto done
;
270 if (!(value_names
= heap_alloc( count
* sizeof(BSTR
) ))) goto done
;
271 if (!(value_types
= heap_alloc( count
* sizeof(DWORD
) ))) goto done
;
277 res
= RegEnumValueW( hkey
, i
, buf
, &len
, NULL
, &value_types
[i
], NULL
, NULL
);
278 if (res
== ERROR_NO_MORE_ITEMS
)
280 if (i
) res
= ERROR_SUCCESS
;
284 if (!(value_names
[i
] = SysAllocString( buf
)))
286 for (i
--; i
>= 0; i
--) SysFreeString( value_names
[i
] );
292 if (hr
== S_OK
&& !res
)
294 hr
= to_bstr_array( value_names
, i
, names
);
295 free_bstr_array( value_names
, i
);
296 if (hr
== S_OK
) hr
= to_i4_array( value_types
, i
, types
);
300 set_variant( VT_UI4
, res
, NULL
, retval
);
302 heap_free( value_names
);
303 heap_free( value_types
);
308 HRESULT
reg_enum_values( IWbemClassObject
*obj
, IWbemClassObject
*in
, IWbemClassObject
**out
)
310 VARIANT defkey
, subkey
, names
, types
, retval
;
311 IWbemClassObject
*sig
, *out_params
= NULL
;
314 TRACE("%p, %p\n", in
, out
);
316 hr
= IWbemClassObject_Get( in
, param_defkeyW
, 0, &defkey
, NULL
, NULL
);
317 if (hr
!= S_OK
) return hr
;
318 hr
= IWbemClassObject_Get( in
, param_subkeynameW
, 0, &subkey
, NULL
, NULL
);
319 if (hr
!= S_OK
) return hr
;
321 hr
= create_signature( class_stdregprovW
, method_enumvaluesW
, PARAM_OUT
, &sig
);
324 VariantClear( &subkey
);
329 hr
= IWbemClassObject_SpawnInstance( sig
, 0, &out_params
);
332 VariantClear( &subkey
);
333 IWbemClassObject_Release( sig
);
337 VariantInit( &names
);
338 VariantInit( &types
);
339 hr
= enum_values( (HKEY
)(INT_PTR
)V_I4(&defkey
), V_BSTR(&subkey
), &names
, &types
, &retval
);
340 if (hr
!= S_OK
) goto done
;
343 if (!V_UI4( &retval
))
345 hr
= IWbemClassObject_Put( out_params
, param_namesW
, 0, &names
, CIM_STRING
|CIM_FLAG_ARRAY
);
346 if (hr
!= S_OK
) goto done
;
347 hr
= IWbemClassObject_Put( out_params
, param_typesW
, 0, &types
, CIM_SINT32
|CIM_FLAG_ARRAY
);
348 if (hr
!= S_OK
) goto done
;
350 hr
= IWbemClassObject_Put( out_params
, param_returnvalueW
, 0, &retval
, CIM_UINT32
);
354 VariantClear( &types
);
355 VariantClear( &names
);
356 VariantClear( &subkey
);
357 IWbemClassObject_Release( sig
);
358 if (hr
== S_OK
&& out
)
361 IWbemClassObject_AddRef( out_params
);
363 if (out_params
) IWbemClassObject_Release( out_params
);
367 static HRESULT
get_stringvalue( HKEY root
, const WCHAR
*subkey
, const WCHAR
*name
, VARIANT
*value
, VARIANT
*retval
)
374 TRACE("%p, %s, %s\n", root
, debugstr_w(subkey
), debugstr_w(name
));
376 if ((res
= RegGetValueW( root
, subkey
, name
, RRF_RT_REG_SZ
, NULL
, NULL
, &size
))) goto done
;
377 if (!(str
= SysAllocStringLen( NULL
, size
/ sizeof(WCHAR
) - 1 )))
382 if (!(res
= RegGetValueW( root
, subkey
, name
, RRF_RT_REG_SZ
, NULL
, str
, &size
)))
383 set_variant( VT_BSTR
, 0, str
, value
);
386 set_variant( VT_UI4
, res
, NULL
, retval
);
387 if (res
) SysFreeString( str
);
391 HRESULT
reg_get_stringvalue( IWbemClassObject
*obj
, IWbemClassObject
*in
, IWbemClassObject
**out
)
393 VARIANT defkey
, subkey
, name
, value
, retval
;
394 IWbemClassObject
*sig
, *out_params
= NULL
;
397 TRACE("%p, %p\n", in
, out
);
399 hr
= IWbemClassObject_Get( in
, param_defkeyW
, 0, &defkey
, NULL
, NULL
);
400 if (hr
!= S_OK
) return hr
;
401 hr
= IWbemClassObject_Get( in
, param_subkeynameW
, 0, &subkey
, NULL
, NULL
);
402 if (hr
!= S_OK
) return hr
;
403 hr
= IWbemClassObject_Get( in
, param_valuenameW
, 0, &name
, NULL
, NULL
);
404 if (hr
!= S_OK
) return hr
;
406 hr
= create_signature( class_stdregprovW
, method_getstringvalueW
, PARAM_OUT
, &sig
);
409 VariantClear( &name
);
410 VariantClear( &subkey
);
415 hr
= IWbemClassObject_SpawnInstance( sig
, 0, &out_params
);
418 VariantClear( &name
);
419 VariantClear( &subkey
);
420 IWbemClassObject_Release( sig
);
424 VariantInit( &value
);
425 hr
= get_stringvalue( (HKEY
)(INT_PTR
)V_I4(&defkey
), V_BSTR(&subkey
), V_BSTR(&name
), &value
, &retval
);
426 if (hr
!= S_OK
) goto done
;
429 if (!V_UI4( &retval
))
431 hr
= IWbemClassObject_Put( out_params
, param_valueW
, 0, &value
, CIM_STRING
);
432 if (hr
!= S_OK
) goto done
;
434 hr
= IWbemClassObject_Put( out_params
, param_returnvalueW
, 0, &retval
, CIM_UINT32
);
438 VariantClear( &name
);
439 VariantClear( &subkey
);
440 IWbemClassObject_Release( sig
);
441 if (hr
== S_OK
&& out
)
444 IWbemClassObject_AddRef( out_params
);
446 if (out_params
) IWbemClassObject_Release( out_params
);