gitlab: Run tests on Debian after the daily commit round.
[wine.git] / dlls / sapi / token.c
blob2de3553b3ab8c5a95958b883c276764828bb1c08
1 /*
2 * Speech API (SAPI) token implementation.
4 * Copyright (C) 2017 Huw Davies
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
21 #include <stdarg.h>
23 #define COBJMACROS
25 #include "windef.h"
26 #include "winbase.h"
27 #include "objbase.h"
28 #include "sapiddk.h"
29 #include "sperror.h"
31 #include "wine/debug.h"
33 #include "sapi_private.h"
35 WINE_DEFAULT_DEBUG_CHANNEL(sapi);
37 struct data_key
39 ISpRegDataKey ISpRegDataKey_iface;
40 LONG ref;
42 HKEY key;
43 BOOL read_only;
46 static struct data_key *impl_from_ISpRegDataKey( ISpRegDataKey *iface )
48 return CONTAINING_RECORD( iface, struct data_key, ISpRegDataKey_iface );
51 static HRESULT WINAPI data_key_QueryInterface( ISpRegDataKey *iface, REFIID iid, void **obj )
53 struct data_key *This = impl_from_ISpRegDataKey( iface );
55 TRACE( "(%p)->(%s %p)\n", This, debugstr_guid( iid ), obj );
57 if (IsEqualIID( iid, &IID_IUnknown ) ||
58 IsEqualIID( iid, &IID_ISpDataKey ) ||
59 IsEqualIID( iid, &IID_ISpRegDataKey ))
61 ISpRegDataKey_AddRef( iface );
62 *obj = iface;
63 return S_OK;
66 FIXME( "interface %s not implemented\n", debugstr_guid( iid ) );
67 *obj = NULL;
68 return E_NOINTERFACE;
71 static ULONG WINAPI data_key_AddRef( ISpRegDataKey *iface )
73 struct data_key *This = impl_from_ISpRegDataKey( iface );
74 ULONG ref = InterlockedIncrement( &This->ref );
76 TRACE( "(%p) ref = %lu\n", This, ref );
77 return ref;
80 static ULONG WINAPI data_key_Release( ISpRegDataKey *iface )
82 struct data_key *This = impl_from_ISpRegDataKey( iface );
83 ULONG ref = InterlockedDecrement(&This->ref);
85 TRACE( "(%p) ref = %lu\n", This, ref );
87 if (!ref)
89 if (This->key) RegCloseKey( This->key );
90 heap_free( This );
93 return ref;
96 static HRESULT WINAPI data_key_SetData( ISpRegDataKey *iface, LPCWSTR name,
97 ULONG size, const BYTE *data )
99 FIXME( "stub\n" );
100 return E_NOTIMPL;
103 static HRESULT WINAPI data_key_GetData( ISpRegDataKey *iface, LPCWSTR name,
104 ULONG *size, BYTE *data )
106 FIXME( "stub\n" );
107 return E_NOTIMPL;
110 static HRESULT WINAPI data_key_SetStringValue( ISpRegDataKey *iface,
111 LPCWSTR name, LPCWSTR value )
113 FIXME( "stub\n" );
114 return E_NOTIMPL;
117 static HRESULT WINAPI data_key_GetStringValue( ISpRegDataKey *iface,
118 LPCWSTR name, LPWSTR *value )
120 FIXME( "stub\n" );
121 return E_NOTIMPL;
124 static HRESULT WINAPI data_key_SetDWORD( ISpRegDataKey *iface,
125 LPCWSTR name, DWORD value )
127 FIXME( "stub\n" );
128 return E_NOTIMPL;
131 static HRESULT WINAPI data_key_GetDWORD( ISpRegDataKey *iface,
132 LPCWSTR name, DWORD *pdwValue )
134 FIXME( "stub\n" );
135 return E_NOTIMPL;
138 static HRESULT WINAPI data_key_OpenKey( ISpRegDataKey *iface,
139 LPCWSTR name, ISpDataKey **sub_key )
141 FIXME( "stub\n" );
142 return E_NOTIMPL;
145 static HRESULT WINAPI data_key_CreateKey( ISpRegDataKey *iface,
146 LPCWSTR name, ISpDataKey **sub_key )
148 FIXME( "stub\n" );
149 return E_NOTIMPL;
152 static HRESULT WINAPI data_key_DeleteKey( ISpRegDataKey *iface, LPCWSTR name )
154 FIXME( "stub\n" );
155 return E_NOTIMPL;
158 static HRESULT WINAPI data_key_DeleteValue( ISpRegDataKey *iface, LPCWSTR name )
160 FIXME( "stub\n" );
161 return E_NOTIMPL;
164 static HRESULT WINAPI data_key_EnumKeys( ISpRegDataKey *iface,
165 ULONG index, LPWSTR *sub_key )
167 FIXME( "stub\n" );
168 return E_NOTIMPL;
171 static HRESULT WINAPI data_key_EnumValues( ISpRegDataKey *iface,
172 ULONG index, LPWSTR *value )
174 FIXME( "stub\n" );
175 return E_NOTIMPL;
178 static HRESULT WINAPI data_key_SetKey( ISpRegDataKey *iface,
179 HKEY key, BOOL read_only )
181 struct data_key *This = impl_from_ISpRegDataKey( iface );
183 TRACE( "(%p)->(%p %d)\n", This, key, read_only );
185 if (This->key) return SPERR_ALREADY_INITIALIZED;
187 This->key = key;
188 This->read_only = read_only;
189 return S_OK;
192 const struct ISpRegDataKeyVtbl data_key_vtbl =
194 data_key_QueryInterface,
195 data_key_AddRef,
196 data_key_Release,
197 data_key_SetData,
198 data_key_GetData,
199 data_key_SetStringValue,
200 data_key_GetStringValue,
201 data_key_SetDWORD,
202 data_key_GetDWORD,
203 data_key_OpenKey,
204 data_key_CreateKey,
205 data_key_DeleteKey,
206 data_key_DeleteValue,
207 data_key_EnumKeys,
208 data_key_EnumValues,
209 data_key_SetKey
212 HRESULT data_key_create( IUnknown *outer, REFIID iid, void **obj )
214 struct data_key *This = heap_alloc( sizeof(*This) );
215 HRESULT hr;
217 if (!This) return E_OUTOFMEMORY;
218 This->ISpRegDataKey_iface.lpVtbl = &data_key_vtbl;
219 This->ref = 1;
220 This->key = NULL;
221 This->read_only = FALSE;
223 hr = ISpRegDataKey_QueryInterface( &This->ISpRegDataKey_iface, iid, obj );
225 ISpRegDataKey_Release( &This->ISpRegDataKey_iface );
226 return hr;
229 struct token_category
231 ISpObjectTokenCategory ISpObjectTokenCategory_iface;
232 LONG ref;
234 ISpRegDataKey *data_key;
237 static struct token_category *impl_from_ISpObjectTokenCategory( ISpObjectTokenCategory *iface )
239 return CONTAINING_RECORD( iface, struct token_category, ISpObjectTokenCategory_iface );
242 static HRESULT WINAPI token_category_QueryInterface( ISpObjectTokenCategory *iface,
243 REFIID iid, void **obj )
245 struct token_category *This = impl_from_ISpObjectTokenCategory( iface );
247 TRACE( "(%p)->(%s %p)\n", This, debugstr_guid( iid ), obj );
249 if (IsEqualIID( iid, &IID_IUnknown ) ||
250 IsEqualIID( iid, &IID_ISpDataKey ) ||
251 IsEqualIID( iid, &IID_ISpObjectTokenCategory ))
253 ISpObjectTokenCategory_AddRef( iface );
254 *obj = iface;
255 return S_OK;
258 FIXME( "interface %s not implemented\n", debugstr_guid( iid ) );
259 *obj = NULL;
260 return E_NOINTERFACE;
263 static ULONG WINAPI token_category_AddRef( ISpObjectTokenCategory *iface )
265 struct token_category *This = impl_from_ISpObjectTokenCategory( iface );
266 ULONG ref = InterlockedIncrement( &This->ref );
268 TRACE( "(%p) ref = %lu\n", This, ref );
269 return ref;
272 static ULONG WINAPI token_category_Release( ISpObjectTokenCategory *iface )
274 struct token_category *This = impl_from_ISpObjectTokenCategory( iface );
275 ULONG ref = InterlockedDecrement(&This->ref);
277 TRACE( "(%p) ref = %lu\n", This, ref );
279 if (!ref)
281 if (This->data_key) ISpRegDataKey_Release( This->data_key );
282 heap_free( This );
284 return ref;
287 static HRESULT WINAPI token_category_SetData( ISpObjectTokenCategory *iface,
288 LPCWSTR name, ULONG size,
289 const BYTE *data )
291 FIXME( "stub\n" );
292 return E_NOTIMPL;
295 static HRESULT WINAPI token_category_GetData( ISpObjectTokenCategory *iface,
296 LPCWSTR name, ULONG *size,
297 BYTE *data )
299 FIXME( "stub\n" );
300 return E_NOTIMPL;
303 static HRESULT WINAPI token_category_SetStringValue( ISpObjectTokenCategory *iface,
304 LPCWSTR name, LPCWSTR value )
306 FIXME( "stub\n" );
307 return E_NOTIMPL;
310 static HRESULT WINAPI token_category_GetStringValue( ISpObjectTokenCategory *iface,
311 LPCWSTR name, LPWSTR *value )
313 FIXME( "stub\n" );
314 return E_NOTIMPL;
317 static HRESULT WINAPI token_category_SetDWORD( ISpObjectTokenCategory *iface,
318 LPCWSTR name, DWORD value )
320 FIXME( "stub\n" );
321 return E_NOTIMPL;
324 static HRESULT WINAPI token_category_GetDWORD( ISpObjectTokenCategory *iface,
325 LPCWSTR name, DWORD *pdwValue )
327 FIXME( "stub\n" );
328 return E_NOTIMPL;
331 static HRESULT WINAPI token_category_OpenKey( ISpObjectTokenCategory *iface,
332 LPCWSTR name, ISpDataKey **sub_key )
334 FIXME( "stub\n" );
335 return E_NOTIMPL;
338 static HRESULT WINAPI token_category_CreateKey( ISpObjectTokenCategory *iface,
339 LPCWSTR name, ISpDataKey **sub_key )
341 FIXME( "stub\n" );
342 return E_NOTIMPL;
345 static HRESULT WINAPI token_category_DeleteKey( ISpObjectTokenCategory *iface,
346 LPCWSTR name )
348 FIXME( "stub\n" );
349 return E_NOTIMPL;
352 static HRESULT WINAPI token_category_DeleteValue( ISpObjectTokenCategory *iface,
353 LPCWSTR name )
355 FIXME( "stub\n" );
356 return E_NOTIMPL;
359 static HRESULT WINAPI token_category_EnumKeys( ISpObjectTokenCategory *iface,
360 ULONG index, LPWSTR *sub_key )
362 FIXME( "stub\n" );
363 return E_NOTIMPL;
366 static HRESULT WINAPI token_category_EnumValues( ISpObjectTokenCategory *iface,
367 ULONG index, LPWSTR *value )
369 FIXME( "stub\n" );
370 return E_NOTIMPL;
373 static HRESULT parse_cat_id( const WCHAR *str, HKEY *root, const WCHAR **sub_key )
375 struct table
377 const WCHAR *name;
378 unsigned int len;
379 HKEY key;
380 } table[] =
382 #define X(s) s, ARRAY_SIZE(s) - 1
383 { X(L"HKEY_LOCAL_MACHINE\\"), HKEY_LOCAL_MACHINE },
384 { X(L"HKEY_CURRENT_USER\\"), HKEY_CURRENT_USER },
385 { NULL }
386 #undef X
388 struct table *ptr;
389 int len = lstrlenW( str );
391 for (ptr = table; ptr->name; ptr++)
393 if (len >= ptr->len && !wcsncmp( str, ptr->name, ptr->len ))
395 *root = ptr->key;
396 *sub_key = str + ptr->len;
397 return S_OK;
400 return S_FALSE;
403 static HRESULT WINAPI token_category_SetId( ISpObjectTokenCategory *iface,
404 LPCWSTR id, BOOL create )
406 struct token_category *This = impl_from_ISpObjectTokenCategory( iface );
407 HKEY root, key;
408 const WCHAR *subkey;
409 LONG res;
410 HRESULT hr;
412 TRACE( "(%p)->(%s %d)\n", This, debugstr_w( id ), create );
414 if (This->data_key) return SPERR_ALREADY_INITIALIZED;
416 hr = parse_cat_id( id, &root, &subkey );
417 if (hr != S_OK) return SPERR_INVALID_REGISTRY_KEY;
419 if (create)
420 res = RegCreateKeyExW( root, subkey, 0, NULL, 0, KEY_ALL_ACCESS, NULL, &key, NULL );
421 else
422 res = RegOpenKeyExW( root, subkey, 0, KEY_ALL_ACCESS, &key );
423 if (res) return SPERR_INVALID_REGISTRY_KEY;
425 hr = CoCreateInstance( &CLSID_SpDataKey, NULL, CLSCTX_ALL,
426 &IID_ISpRegDataKey, (void **)&This->data_key );
427 if (FAILED(hr)) goto fail;
429 hr = ISpRegDataKey_SetKey( This->data_key, key, FALSE );
430 if (FAILED(hr)) goto fail;
432 return hr;
434 fail:
435 RegCloseKey( key );
436 return hr;
439 static HRESULT WINAPI token_category_GetId( ISpObjectTokenCategory *iface,
440 LPWSTR *id )
442 FIXME( "stub\n" );
443 return E_NOTIMPL;
446 static HRESULT WINAPI token_category_GetDataKey( ISpObjectTokenCategory *iface,
447 SPDATAKEYLOCATION location,
448 ISpDataKey **data_key )
450 FIXME( "stub\n" );
451 return E_NOTIMPL;
454 static HRESULT WINAPI token_category_EnumTokens( ISpObjectTokenCategory *iface,
455 LPCWSTR req, LPCWSTR opt,
456 IEnumSpObjectTokens **enum_tokens )
458 struct token_category *This = impl_from_ISpObjectTokenCategory( iface );
459 ISpObjectTokenEnumBuilder *builder;
460 HRESULT hr;
462 FIXME( "(%p)->(%s %s %p): semi-stub\n", This, debugstr_w( req ), debugstr_w( opt ), enum_tokens );
464 if (!This->data_key) return SPERR_UNINITIALIZED;
466 hr = CoCreateInstance( &CLSID_SpObjectTokenEnum, NULL, CLSCTX_ALL,
467 &IID_ISpObjectTokenEnumBuilder, (void **)&builder );
468 if (FAILED(hr)) return hr;
470 hr = ISpObjectTokenEnumBuilder_SetAttribs( builder, req, opt );
471 if (FAILED(hr)) goto fail;
473 /* FIXME: Build the enumerator */
475 hr = ISpObjectTokenEnumBuilder_QueryInterface( builder, &IID_IEnumSpObjectTokens,
476 (void **)enum_tokens );
478 fail:
479 ISpObjectTokenEnumBuilder_Release( builder );
480 return hr;
483 static HRESULT WINAPI token_category_SetDefaultTokenId( ISpObjectTokenCategory *iface,
484 LPCWSTR id )
486 FIXME( "stub\n" );
487 return E_NOTIMPL;
490 static HRESULT WINAPI token_category_GetDefaultTokenId( ISpObjectTokenCategory *iface,
491 LPWSTR *id )
493 struct token_category *This = impl_from_ISpObjectTokenCategory( iface );
494 struct data_key *this_data_key;
495 LONG res;
496 WCHAR regvalue[512];
497 DWORD regvalue_size = sizeof( regvalue );
499 FIXME( "(%p)->(%p): semi-stub\n", iface, id );
501 if (!This->data_key)
502 return SPERR_UNINITIALIZED;
504 if (!id)
505 return E_POINTER;
507 /* todo: check HKCU's DefaultTokenId before */
509 this_data_key = impl_from_ISpRegDataKey( This->data_key );
511 res = RegGetValueW( this_data_key->key, NULL, L"DefaultDefaultTokenId", RRF_RT_REG_SZ,
512 NULL, &regvalue, &regvalue_size);
513 if (res == ERROR_FILE_NOT_FOUND) {
514 return SPERR_NOT_FOUND;
515 } else if (res != ERROR_SUCCESS) {
516 /* probably not the correct return value */
517 FIXME( "returning %08lx\n", res );
518 return res;
521 *id = CoTaskMemAlloc( regvalue_size );
522 wcscpy( *id, regvalue );
524 return S_OK;
527 const struct ISpObjectTokenCategoryVtbl token_category_vtbl =
529 token_category_QueryInterface,
530 token_category_AddRef,
531 token_category_Release,
532 token_category_SetData,
533 token_category_GetData,
534 token_category_SetStringValue,
535 token_category_GetStringValue,
536 token_category_SetDWORD,
537 token_category_GetDWORD,
538 token_category_OpenKey,
539 token_category_CreateKey,
540 token_category_DeleteKey,
541 token_category_DeleteValue,
542 token_category_EnumKeys,
543 token_category_EnumValues,
544 token_category_SetId,
545 token_category_GetId,
546 token_category_GetDataKey,
547 token_category_EnumTokens,
548 token_category_SetDefaultTokenId,
549 token_category_GetDefaultTokenId,
552 HRESULT token_category_create( IUnknown *outer, REFIID iid, void **obj )
554 struct token_category *This = heap_alloc( sizeof(*This) );
555 HRESULT hr;
557 if (!This) return E_OUTOFMEMORY;
558 This->ISpObjectTokenCategory_iface.lpVtbl = &token_category_vtbl;
559 This->ref = 1;
560 This->data_key = NULL;
562 hr = ISpObjectTokenCategory_QueryInterface( &This->ISpObjectTokenCategory_iface, iid, obj );
564 ISpObjectTokenCategory_Release( &This->ISpObjectTokenCategory_iface );
565 return hr;
568 struct token_enum
570 ISpObjectTokenEnumBuilder ISpObjectTokenEnumBuilder_iface;
571 LONG ref;
573 BOOL init;
574 WCHAR *req, *opt;
575 ULONG count;
578 static struct token_enum *impl_from_ISpObjectTokenEnumBuilder( ISpObjectTokenEnumBuilder *iface )
580 return CONTAINING_RECORD( iface, struct token_enum, ISpObjectTokenEnumBuilder_iface );
583 static HRESULT WINAPI token_enum_QueryInterface( ISpObjectTokenEnumBuilder *iface,
584 REFIID iid, void **obj )
586 struct token_enum *This = impl_from_ISpObjectTokenEnumBuilder( iface );
588 TRACE( "(%p)->(%s %p)\n", This, debugstr_guid( iid ), obj );
590 if (IsEqualIID( iid, &IID_IUnknown ) ||
591 IsEqualIID( iid, &IID_IEnumSpObjectTokens ) ||
592 IsEqualIID( iid, &IID_ISpObjectTokenEnumBuilder ))
594 ISpObjectTokenEnumBuilder_AddRef( iface );
595 *obj = iface;
596 return S_OK;
599 FIXME( "interface %s not implemented\n", debugstr_guid( iid ) );
600 *obj = NULL;
601 return E_NOINTERFACE;
604 static ULONG WINAPI token_enum_AddRef( ISpObjectTokenEnumBuilder *iface )
606 struct token_enum *This = impl_from_ISpObjectTokenEnumBuilder( iface );
607 ULONG ref = InterlockedIncrement( &This->ref );
609 TRACE( "(%p) ref = %lu\n", This, ref );
610 return ref;
613 static ULONG WINAPI token_enum_Release( ISpObjectTokenEnumBuilder *iface )
615 struct token_enum *This = impl_from_ISpObjectTokenEnumBuilder( iface );
616 ULONG ref = InterlockedDecrement(&This->ref);
618 TRACE( "(%p) ref = %lu\n", This, ref );
620 if (!ref)
622 heap_free( This->req );
623 heap_free( This->opt );
624 heap_free( This );
627 return ref;
630 static HRESULT WINAPI token_enum_Next( ISpObjectTokenEnumBuilder *iface,
631 ULONG num, ISpObjectToken **tokens,
632 ULONG *fetched )
634 struct token_enum *This = impl_from_ISpObjectTokenEnumBuilder( iface );
636 TRACE( "(%p)->(%lu %p %p)\n", This, num, tokens, fetched );
638 if (!This->init) return SPERR_UNINITIALIZED;
640 FIXME( "semi-stub: Returning an empty enumerator\n" );
642 if (fetched) *fetched = 0;
643 return S_FALSE;
646 static HRESULT WINAPI token_enum_Skip( ISpObjectTokenEnumBuilder *iface,
647 ULONG num )
649 FIXME( "stub\n" );
650 return E_NOTIMPL;
653 static HRESULT WINAPI token_enum_Reset( ISpObjectTokenEnumBuilder *iface)
655 FIXME( "stub\n" );
656 return E_NOTIMPL;
659 static HRESULT WINAPI token_enum_Clone( ISpObjectTokenEnumBuilder *iface,
660 IEnumSpObjectTokens **clone )
662 FIXME( "stub\n" );
663 return E_NOTIMPL;
666 static HRESULT WINAPI token_enum_Item( ISpObjectTokenEnumBuilder *iface,
667 ULONG index, ISpObjectToken **token )
669 FIXME( "stub\n" );
670 return E_NOTIMPL;
673 static HRESULT WINAPI token_enum_GetCount( ISpObjectTokenEnumBuilder *iface,
674 ULONG *count )
676 struct token_enum *This = impl_from_ISpObjectTokenEnumBuilder( iface );
678 TRACE( "(%p)->(%p)\n", This, count );
680 if (!This->init) return SPERR_UNINITIALIZED;
682 *count = This->count;
683 return S_OK;
686 static HRESULT WINAPI token_enum_SetAttribs( ISpObjectTokenEnumBuilder *iface,
687 LPCWSTR req, LPCWSTR opt)
689 struct token_enum *This = impl_from_ISpObjectTokenEnumBuilder( iface );
691 TRACE( "(%p)->(%s %s)\n", This, debugstr_w( req ), debugstr_w( opt ) );
693 if (This->init) return SPERR_ALREADY_INITIALIZED;
695 if (req)
697 This->req = heap_strdupW( req );
698 if (!This->req) goto out_of_mem;
701 if (opt)
703 This->opt = heap_strdupW( opt );
704 if (!This->opt) goto out_of_mem;
707 This->init = TRUE;
708 return S_OK;
710 out_of_mem:
711 heap_free( This->req );
712 return E_OUTOFMEMORY;
715 static HRESULT WINAPI token_enum_AddTokens( ISpObjectTokenEnumBuilder *iface,
716 ULONG num, ISpObjectToken **tokens )
718 FIXME( "stub\n" );
719 return E_NOTIMPL;
722 static HRESULT WINAPI token_enum_AddTokensFromDataKey( ISpObjectTokenEnumBuilder *iface,
723 ISpDataKey *data_key,
724 LPCWSTR sub_key, LPCWSTR cat_id )
726 FIXME( "stub\n" );
727 return E_NOTIMPL;
730 static HRESULT WINAPI token_enum_AddTokensFromTokenEnum( ISpObjectTokenEnumBuilder *iface,
731 IEnumSpObjectTokens *token_enum )
733 FIXME( "stub\n" );
734 return E_NOTIMPL;
737 static HRESULT WINAPI token_enum_Sort( ISpObjectTokenEnumBuilder *iface,
738 LPCWSTR first )
740 FIXME( "stub\n" );
741 return E_NOTIMPL;
744 const struct ISpObjectTokenEnumBuilderVtbl token_enum_vtbl =
746 token_enum_QueryInterface,
747 token_enum_AddRef,
748 token_enum_Release,
749 token_enum_Next,
750 token_enum_Skip,
751 token_enum_Reset,
752 token_enum_Clone,
753 token_enum_Item,
754 token_enum_GetCount,
755 token_enum_SetAttribs,
756 token_enum_AddTokens,
757 token_enum_AddTokensFromDataKey,
758 token_enum_AddTokensFromTokenEnum,
759 token_enum_Sort
762 HRESULT token_enum_create( IUnknown *outer, REFIID iid, void **obj )
764 struct token_enum *This = heap_alloc( sizeof(*This) );
765 HRESULT hr;
767 if (!This) return E_OUTOFMEMORY;
768 This->ISpObjectTokenEnumBuilder_iface.lpVtbl = &token_enum_vtbl;
769 This->ref = 1;
770 This->req = NULL;
771 This->opt = NULL;
772 This->init = FALSE;
773 This->count = 0;
775 hr = ISpObjectTokenEnumBuilder_QueryInterface( &This->ISpObjectTokenEnumBuilder_iface, iid, obj );
777 ISpObjectTokenEnumBuilder_Release( &This->ISpObjectTokenEnumBuilder_iface );
778 return hr;
781 struct object_token
783 ISpObjectToken ISpObjectToken_iface;
784 LONG ref;
786 HKEY token_key;
789 static struct object_token *impl_from_ISpObjectToken( ISpObjectToken *iface )
791 return CONTAINING_RECORD( iface, struct object_token, ISpObjectToken_iface );
794 static HRESULT WINAPI token_QueryInterface( ISpObjectToken *iface,
795 REFIID iid, void **obj )
797 struct object_token *This = impl_from_ISpObjectToken( iface );
799 TRACE( "(%p)->(%s %p)\n", This, debugstr_guid( iid ), obj );
801 if (IsEqualIID( iid, &IID_IUnknown ) ||
802 IsEqualIID( iid, &IID_ISpDataKey ) ||
803 IsEqualIID( iid, &IID_ISpObjectToken ))
805 ISpObjectToken_AddRef( iface );
806 *obj = iface;
807 return S_OK;
810 FIXME( "interface %s not implemented\n", debugstr_guid( iid ) );
811 *obj = NULL;
812 return E_NOINTERFACE;
815 static ULONG WINAPI token_AddRef( ISpObjectToken *iface )
817 struct object_token *This = impl_from_ISpObjectToken( iface );
818 ULONG ref = InterlockedIncrement( &This->ref );
820 TRACE( "(%p) ref = %lu\n", This, ref );
821 return ref;
824 static ULONG WINAPI token_Release( ISpObjectToken *iface )
826 struct object_token *This = impl_from_ISpObjectToken( iface );
827 ULONG ref = InterlockedDecrement(&This->ref);
829 TRACE( "(%p) ref = %lu\n", This, ref );
831 if (!ref)
833 if (This->token_key) RegCloseKey( This->token_key );
834 heap_free( This );
837 return ref;
840 static HRESULT WINAPI token_SetData( ISpObjectToken *iface,
841 LPCWSTR name, ULONG size,
842 const BYTE *data )
844 FIXME( "stub\n" );
845 return E_NOTIMPL;
848 static HRESULT WINAPI token_GetData( ISpObjectToken *iface,
849 LPCWSTR name, ULONG *size,
850 BYTE *data )
852 FIXME( "stub\n" );
853 return E_NOTIMPL;
856 static HRESULT WINAPI token_SetStringValue( ISpObjectToken *iface,
857 LPCWSTR name, LPCWSTR value )
859 FIXME( "stub\n" );
860 return E_NOTIMPL;
863 static HRESULT WINAPI token_GetStringValue( ISpObjectToken *iface,
864 LPCWSTR name, LPWSTR *value )
866 FIXME( "stub\n" );
867 return E_NOTIMPL;
870 static HRESULT WINAPI token_SetDWORD( ISpObjectToken *iface,
871 LPCWSTR name, DWORD value )
873 FIXME( "stub\n" );
874 return E_NOTIMPL;
877 static HRESULT WINAPI token_GetDWORD( ISpObjectToken *iface,
878 LPCWSTR name, DWORD *pdwValue )
880 FIXME( "stub\n" );
881 return E_NOTIMPL;
884 static HRESULT WINAPI token_OpenKey( ISpObjectToken *iface,
885 LPCWSTR name, ISpDataKey **sub_key )
887 FIXME( "stub\n" );
888 return E_NOTIMPL;
891 static HRESULT WINAPI token_CreateKey( ISpObjectToken *iface,
892 LPCWSTR name, ISpDataKey **sub_key )
894 FIXME( "stub\n" );
895 return E_NOTIMPL;
898 static HRESULT WINAPI token_DeleteKey( ISpObjectToken *iface,
899 LPCWSTR name )
901 FIXME( "stub\n" );
902 return E_NOTIMPL;
905 static HRESULT WINAPI token_DeleteValue( ISpObjectToken *iface,
906 LPCWSTR name )
908 FIXME( "stub\n" );
909 return E_NOTIMPL;
912 static HRESULT WINAPI token_EnumKeys( ISpObjectToken *iface,
913 ULONG index, LPWSTR *sub_key )
915 FIXME( "stub\n" );
916 return E_NOTIMPL;
919 static HRESULT WINAPI token_EnumValues( ISpObjectToken *iface,
920 ULONG index, LPWSTR *value )
922 FIXME( "stub\n" );
923 return E_NOTIMPL;
926 static HRESULT WINAPI token_SetId( ISpObjectToken *iface,
927 LPCWSTR category_id, LPCWSTR token_id,
928 BOOL create )
930 struct object_token *This = impl_from_ISpObjectToken( iface );
931 BOOL res;
932 HRESULT hr;
933 HKEY root, key;
934 const WCHAR *subkey;
936 FIXME( "(%p)->(%s %s %d): semi-stub\n", This, debugstr_w( category_id ),
937 debugstr_w(token_id), create );
939 if (This->token_key) return SPERR_ALREADY_INITIALIZED;
941 if (!token_id) return E_POINTER;
943 hr = parse_cat_id( token_id, &root, &subkey );
944 if (hr != S_OK) return SPERR_NOT_FOUND;
946 if (create)
947 res = RegCreateKeyExW( root, subkey, 0, NULL, 0, KEY_ALL_ACCESS, NULL, &key, NULL);
948 else
949 res = RegOpenKeyExW( root, subkey, 0, KEY_ALL_ACCESS, &key );
950 if (res) return SPERR_NOT_FOUND;
952 This->token_key = key;
954 return S_OK;
957 static HRESULT WINAPI token_GetId( ISpObjectToken *iface,
958 LPWSTR *token_id )
960 FIXME( "stub\n" );
961 return E_NOTIMPL;
964 static HRESULT WINAPI token_GetCategory( ISpObjectToken *iface,
965 ISpObjectTokenCategory **category )
967 FIXME( "stub\n" );
968 return E_NOTIMPL;
971 static HRESULT WINAPI token_CreateInstance( ISpObjectToken *iface,
972 IUnknown *outer,
973 DWORD class_context,
974 REFIID riid,
975 void **object )
977 FIXME( "stub\n" );
978 return E_NOTIMPL;
981 static HRESULT WINAPI token_GetStorageFileName( ISpObjectToken *iface,
982 REFCLSID caller,
983 LPCWSTR key_name,
984 LPCWSTR filename,
985 ULONG folder,
986 LPWSTR *filepath )
988 FIXME( "stub\n" );
989 return E_NOTIMPL;
992 static HRESULT WINAPI token_RemoveStorageFileName( ISpObjectToken *iface,
993 REFCLSID caller,
994 LPCWSTR key_name,
995 BOOL delete_file )
997 FIXME( "stub\n" );
998 return E_NOTIMPL;
1001 static HRESULT WINAPI token_Remove( ISpObjectToken *iface,
1002 REFCLSID caller )
1004 FIXME( "stub\n" );
1005 return E_NOTIMPL;
1008 static HRESULT WINAPI token_IsUISupported( ISpObjectToken *iface,
1009 LPCWSTR ui_type,
1010 void *extra_data,
1011 ULONG extra_data_size,
1012 IUnknown *object,
1013 BOOL *supported )
1015 FIXME( "stub\n" );
1016 return E_NOTIMPL;
1019 static HRESULT WINAPI token_DisplayUI( ISpObjectToken *iface,
1020 HWND parent,
1021 LPCWSTR title,
1022 LPCWSTR ui_type,
1023 void *extra_data,
1024 ULONG extra_data_size,
1025 IUnknown *object )
1027 FIXME( "stub\n" );
1028 return E_NOTIMPL;
1031 static HRESULT WINAPI token_MatchesAttributes( ISpObjectToken *iface,
1032 LPCWSTR attributes,
1033 BOOL *matches )
1035 FIXME( "stub\n" );
1036 return E_NOTIMPL;
1039 const struct ISpObjectTokenVtbl token_vtbl =
1041 token_QueryInterface,
1042 token_AddRef,
1043 token_Release,
1044 token_SetData,
1045 token_GetData,
1046 token_SetStringValue,
1047 token_GetStringValue,
1048 token_SetDWORD,
1049 token_GetDWORD,
1050 token_OpenKey,
1051 token_CreateKey,
1052 token_DeleteKey,
1053 token_DeleteValue,
1054 token_EnumKeys,
1055 token_EnumValues,
1056 token_SetId,
1057 token_GetId,
1058 token_GetCategory,
1059 token_CreateInstance,
1060 token_GetStorageFileName,
1061 token_RemoveStorageFileName,
1062 token_Remove,
1063 token_IsUISupported,
1064 token_DisplayUI,
1065 token_MatchesAttributes
1068 HRESULT token_create( IUnknown *outer, REFIID iid, void **obj )
1070 struct object_token *This = heap_alloc( sizeof(*This) );
1071 HRESULT hr;
1073 if (!This) return E_OUTOFMEMORY;
1074 This->ISpObjectToken_iface.lpVtbl = &token_vtbl;
1075 This->ref = 1;
1077 This->token_key = NULL;
1079 hr = ISpObjectToken_QueryInterface( &This->ISpObjectToken_iface, iid, obj );
1081 ISpObjectToken_Release( &This->ISpObjectToken_iface );
1082 return hr;