mirror of
https://gitlab.winehq.org/wine/wine.git
synced 2025-08-28 18:23:54 +02:00
733 lines
28 KiB
C
733 lines
28 KiB
C
/* WinRT Windows.Devices.Enumeration implementation
|
|
*
|
|
* Copyright 2021 Gijs Vermeulen
|
|
* Copyright 2022 Julian Klemann for CodeWeavers
|
|
* Copyright 2025 Vibhav Pant
|
|
*
|
|
* 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
|
|
*/
|
|
|
|
#include <assert.h>
|
|
|
|
#include "initguid.h"
|
|
#include "private.h"
|
|
#include "devpropdef.h"
|
|
#include "devfiltertypes.h"
|
|
#include "devquery.h"
|
|
|
|
#include "wine/debug.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(enumeration);
|
|
|
|
struct device_watcher
|
|
{
|
|
IDeviceWatcher IDeviceWatcher_iface;
|
|
struct weak_reference_source weak_reference_source;
|
|
|
|
struct list added_handlers;
|
|
struct list enumerated_handlers;
|
|
struct list stopped_handlers;
|
|
HSTRING filter;
|
|
|
|
CRITICAL_SECTION cs;
|
|
DeviceWatcherStatus status;
|
|
HDEVQUERY query;
|
|
};
|
|
|
|
static inline struct device_watcher *impl_from_IDeviceWatcher( IDeviceWatcher *iface )
|
|
{
|
|
return CONTAINING_RECORD( iface, struct device_watcher, IDeviceWatcher_iface );
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_QueryInterface( IDeviceWatcher *iface, REFIID iid, void **out )
|
|
{
|
|
struct device_watcher *impl = impl_from_IDeviceWatcher( iface );
|
|
|
|
TRACE( "iface %p, iid %s, out %p stub!\n", iface, debugstr_guid( iid ), out );
|
|
|
|
if (IsEqualGUID( iid, &IID_IUnknown ) ||
|
|
IsEqualGUID( iid, &IID_IInspectable ) ||
|
|
IsEqualGUID( iid, &IID_IAgileObject ) ||
|
|
IsEqualGUID( iid, &IID_IDeviceWatcher ))
|
|
{
|
|
IInspectable_AddRef( (*out = &impl->IDeviceWatcher_iface) );
|
|
return S_OK;
|
|
}
|
|
|
|
if (IsEqualGUID( iid, &IID_IWeakReferenceSource ))
|
|
{
|
|
*out = &impl->weak_reference_source.IWeakReferenceSource_iface;
|
|
IWeakReferenceSource_AddRef(*out);
|
|
return S_OK;
|
|
}
|
|
|
|
FIXME( "%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid( iid ) );
|
|
*out = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
static ULONG WINAPI device_watcher_AddRef( IDeviceWatcher *iface )
|
|
{
|
|
struct device_watcher *impl = impl_from_IDeviceWatcher( iface );
|
|
ULONG ref = weak_reference_strong_add_ref( &impl->weak_reference_source );
|
|
TRACE( "iface %p, ref %lu.\n", iface, ref );
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI device_watcher_Release( IDeviceWatcher *iface )
|
|
{
|
|
struct device_watcher *impl = impl_from_IDeviceWatcher( iface );
|
|
ULONG ref = weak_reference_strong_release( &impl->weak_reference_source );
|
|
TRACE( "iface %p, ref %lu.\n", iface, ref );
|
|
|
|
if (!ref)
|
|
{
|
|
typed_event_handlers_clear( &impl->added_handlers );
|
|
typed_event_handlers_clear( &impl->enumerated_handlers );
|
|
typed_event_handlers_clear( &impl->stopped_handlers );
|
|
WindowsDeleteString( impl->filter );
|
|
impl->cs.DebugInfo->Spare[0] = 0;
|
|
DeleteCriticalSection( &impl->cs );
|
|
if (impl->query) DevCloseObjectQuery( impl->query );
|
|
free( impl );
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_GetIids( IDeviceWatcher *iface, ULONG *iid_count, IID **iids )
|
|
{
|
|
FIXME( "iface %p, iid_count %p, iids %p stub!\n", iface, iid_count, iids );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_GetRuntimeClassName( IDeviceWatcher *iface, HSTRING *class_name )
|
|
{
|
|
FIXME( "iface %p, class_name %p stub!\n", iface, class_name );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_GetTrustLevel( IDeviceWatcher *iface, TrustLevel *trust_level )
|
|
{
|
|
FIXME( "iface %p, trust_level %p stub!\n", iface, trust_level );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_add_Added( IDeviceWatcher *iface, ITypedEventHandler_DeviceWatcher_DeviceInformation *handler,
|
|
EventRegistrationToken *token )
|
|
{
|
|
struct device_watcher *impl = impl_from_IDeviceWatcher( iface );
|
|
TRACE( "iface %p, handler %p, token %p\n", iface, handler, token );
|
|
return typed_event_handlers_append( &impl->added_handlers, (ITypedEventHandler_IInspectable_IInspectable *)handler, token );
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_remove_Added( IDeviceWatcher *iface, EventRegistrationToken token )
|
|
{
|
|
struct device_watcher *impl = impl_from_IDeviceWatcher( iface );
|
|
TRACE( "iface %p, token %#I64x.\n", iface, token.value );
|
|
return typed_event_handlers_remove( &impl->added_handlers, &token );
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_add_Updated( IDeviceWatcher *iface, ITypedEventHandler_DeviceWatcher_DeviceInformationUpdate *handler,
|
|
EventRegistrationToken *token )
|
|
{
|
|
FIXME( "iface %p, handler %p, token %p stub!\n", iface, handler, token );
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_remove_Updated( IDeviceWatcher *iface, EventRegistrationToken token )
|
|
{
|
|
FIXME( "iface %p, token %#I64x stub!\n", iface, token.value );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_add_Removed( IDeviceWatcher *iface, ITypedEventHandler_DeviceWatcher_DeviceInformationUpdate *handler,
|
|
EventRegistrationToken *token )
|
|
{
|
|
FIXME( "iface %p, handler %p, token %p stub!\n", iface, handler, token );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_remove_Removed( IDeviceWatcher *iface, EventRegistrationToken token )
|
|
{
|
|
FIXME( "iface %p, token %#I64x stub!\n", iface, token.value );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_add_EnumerationCompleted( IDeviceWatcher *iface, ITypedEventHandler_DeviceWatcher_IInspectable *handler,
|
|
EventRegistrationToken *token )
|
|
{
|
|
struct device_watcher *impl = impl_from_IDeviceWatcher( iface );
|
|
TRACE( "iface %p, handler %p, token %p\n", iface, handler, token );
|
|
return typed_event_handlers_append( &impl->enumerated_handlers, (ITypedEventHandler_IInspectable_IInspectable *)handler, token );
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_remove_EnumerationCompleted( IDeviceWatcher *iface, EventRegistrationToken token )
|
|
{
|
|
struct device_watcher *impl = impl_from_IDeviceWatcher( iface );
|
|
TRACE( "iface %p, token %#I64x.\n", iface, token.value );
|
|
return typed_event_handlers_remove( &impl->enumerated_handlers, &token );
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_add_Stopped( IDeviceWatcher *iface, ITypedEventHandler_DeviceWatcher_IInspectable *handler, EventRegistrationToken *token )
|
|
{
|
|
struct device_watcher *impl = impl_from_IDeviceWatcher( iface );
|
|
TRACE( "iface %p, handler %p, token %p.\n", iface, handler, token );
|
|
return typed_event_handlers_append( &impl->stopped_handlers, (ITypedEventHandler_IInspectable_IInspectable *)handler, token );
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_remove_Stopped( IDeviceWatcher *iface, EventRegistrationToken token )
|
|
{
|
|
struct device_watcher *impl = impl_from_IDeviceWatcher( iface );
|
|
TRACE( "iface %p, token %#I64x.\n", iface, token.value );
|
|
return typed_event_handlers_remove( &impl->stopped_handlers, &token );
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_get_Status( IDeviceWatcher *iface, DeviceWatcherStatus *status )
|
|
{
|
|
struct device_watcher *impl = impl_from_IDeviceWatcher( iface );
|
|
|
|
TRACE( "iface %p, status %p\n", iface, status );
|
|
|
|
EnterCriticalSection( &impl->cs );
|
|
*status = impl->status;
|
|
LeaveCriticalSection( &impl->cs );
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static const char *debugstr_DEV_QUERY_RESULT_ACTION_DATA( const DEV_QUERY_RESULT_ACTION_DATA *data )
|
|
{
|
|
const DEV_OBJECT *obj = &data->Data.DeviceObject;
|
|
if (!data) return wine_dbg_sprintf( "(null)" );
|
|
if (data->Action == DevQueryResultStateChange)
|
|
return wine_dbg_sprintf( "{%d {%d}}", data->Action, data->Data.State );
|
|
return wine_dbg_sprintf( "{%d {{%d %s %lu %p}}}", data->Action, obj->ObjectType, debugstr_w( obj->pszObjectId ), obj->cPropertyCount, obj->pProperties );
|
|
}
|
|
|
|
static void WINAPI device_object_query_callback( HDEVQUERY query, void *data,
|
|
const DEV_QUERY_RESULT_ACTION_DATA *action_data )
|
|
{
|
|
struct device_watcher *watcher;
|
|
IWeakReference *weak = data;
|
|
IDeviceWatcher *iface;
|
|
HRESULT hr;
|
|
|
|
TRACE( "query %p, data %p, action_data %s.\n", query, data, debugstr_DEV_QUERY_RESULT_ACTION_DATA( action_data ) );
|
|
|
|
if (FAILED(hr = IWeakReference_Resolve( weak, &IID_IDeviceWatcher, (IInspectable **)&iface )) || !iface)
|
|
{
|
|
if (action_data->Action == DevQueryResultStateChange &&
|
|
(action_data->Data.State == DevQueryStateClosed || action_data->Data.State == DevQueryStateAborted))
|
|
IWeakReference_Release( weak ); /* No more callbacks are expected, so we can release the weak ref. */
|
|
return;
|
|
}
|
|
watcher = impl_from_IDeviceWatcher( iface );
|
|
|
|
switch (action_data->Action)
|
|
{
|
|
case DevQueryResultStateChange:
|
|
switch (action_data->Data.State)
|
|
{
|
|
case DevQueryStateClosed:
|
|
EnterCriticalSection( &watcher->cs );
|
|
watcher->status = DeviceWatcherStatus_Stopped;
|
|
LeaveCriticalSection( &watcher->cs );
|
|
typed_event_handlers_notify( &watcher->stopped_handlers, (IInspectable *)iface, NULL );
|
|
IWeakReference_Release( weak );
|
|
break;
|
|
case DevQueryStateAborted:
|
|
EnterCriticalSection( &watcher->cs );
|
|
watcher->status = DeviceWatcherStatus_Aborted;
|
|
DevCloseObjectQuery( watcher->query );
|
|
watcher->query = NULL;
|
|
LeaveCriticalSection( &watcher->cs );
|
|
IWeakReference_Release( weak );
|
|
break;
|
|
case DevQueryStateEnumCompleted:
|
|
EnterCriticalSection( &watcher->cs );
|
|
watcher->status = DeviceWatcherStatus_EnumerationCompleted;
|
|
LeaveCriticalSection( &watcher->cs );
|
|
|
|
typed_event_handlers_notify( &watcher->enumerated_handlers, (IInspectable *)iface, NULL );
|
|
break;
|
|
default:
|
|
FIXME( "Unhandled DEV_QUERY_STATE value: %d\n", action_data->Data.State );
|
|
break;
|
|
}
|
|
break;
|
|
case DevQueryResultAdd:
|
|
{
|
|
IDeviceInformation *info;
|
|
if (FAILED(hr = device_information_create( action_data->Data.DeviceObject.pszObjectId, &info )))
|
|
break;
|
|
typed_event_handlers_notify( &watcher->added_handlers, (IInspectable *)iface, (IInspectable *)info );
|
|
IDeviceInformation_Release( info );
|
|
break;
|
|
}
|
|
default:
|
|
FIXME( "Unhandled DEV_QUERY_RESULT_ACTION value: %d\n", action_data->Action );
|
|
break;
|
|
}
|
|
|
|
IDeviceWatcher_Release( iface );
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_Start( IDeviceWatcher *iface )
|
|
{
|
|
struct device_watcher *impl = impl_from_IDeviceWatcher( iface );
|
|
HRESULT hr = S_OK;
|
|
|
|
FIXME( "iface %p: semi-stub!\n", iface );
|
|
|
|
if (!WindowsIsStringEmpty( impl->filter ))
|
|
{
|
|
FIXME( "Unsupported filter: %s\n", debugstr_hstring( impl->filter ) );
|
|
return S_OK;
|
|
}
|
|
|
|
EnterCriticalSection( &impl->cs );
|
|
switch (impl->status)
|
|
{
|
|
case DeviceWatcherStatus_EnumerationCompleted: hr = E_ILLEGAL_METHOD_CALL; break;
|
|
case DeviceWatcherStatus_Started: hr = E_ILLEGAL_METHOD_CALL; break;
|
|
case DeviceWatcherStatus_Stopping: hr = E_ILLEGAL_METHOD_CALL; break;
|
|
default: assert( FALSE ); break;
|
|
case DeviceWatcherStatus_Aborted:
|
|
case DeviceWatcherStatus_Created:
|
|
case DeviceWatcherStatus_Stopped:
|
|
{
|
|
IWeakReference *weak;
|
|
HRESULT hr;
|
|
|
|
IWeakReferenceSource_GetWeakReference( &impl->weak_reference_source.IWeakReferenceSource_iface, &weak );
|
|
hr = DevCreateObjectQuery( DevObjectTypeDeviceInterfaceDisplay, DevQueryFlagAsyncClose, 0, NULL, 0, NULL, device_object_query_callback, weak,
|
|
&impl->query );
|
|
if (FAILED(hr))
|
|
{
|
|
ERR( "Failed to create device query: %#lx\n", hr );
|
|
IWeakReference_Release( weak );
|
|
break;
|
|
}
|
|
impl->status = DeviceWatcherStatus_Started;
|
|
break;
|
|
}
|
|
}
|
|
LeaveCriticalSection( &impl->cs );
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI device_watcher_Stop( IDeviceWatcher *iface )
|
|
{
|
|
struct device_watcher *impl = impl_from_IDeviceWatcher( iface );
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE( "iface %p\n", iface );
|
|
|
|
EnterCriticalSection( &impl->cs );
|
|
switch (impl->status)
|
|
{
|
|
case DeviceWatcherStatus_Aborted: break;
|
|
case DeviceWatcherStatus_Created: hr = E_ILLEGAL_METHOD_CALL; break;
|
|
case DeviceWatcherStatus_Stopped: hr = E_ILLEGAL_METHOD_CALL; break;
|
|
case DeviceWatcherStatus_Stopping: hr = E_ILLEGAL_METHOD_CALL; break;
|
|
default: assert( FALSE ); break;
|
|
case DeviceWatcherStatus_EnumerationCompleted:
|
|
case DeviceWatcherStatus_Started:
|
|
impl->status = DeviceWatcherStatus_Stopping;
|
|
DevCloseObjectQuery( impl->query );
|
|
impl->query = NULL;
|
|
break;
|
|
}
|
|
LeaveCriticalSection( &impl->cs );
|
|
|
|
return hr;
|
|
}
|
|
|
|
static const struct IDeviceWatcherVtbl device_watcher_vtbl =
|
|
{
|
|
device_watcher_QueryInterface,
|
|
device_watcher_AddRef,
|
|
device_watcher_Release,
|
|
/* IInspectable methods */
|
|
device_watcher_GetIids,
|
|
device_watcher_GetRuntimeClassName,
|
|
device_watcher_GetTrustLevel,
|
|
/* IDeviceWatcher methods */
|
|
device_watcher_add_Added,
|
|
device_watcher_remove_Added,
|
|
device_watcher_add_Updated,
|
|
device_watcher_remove_Updated,
|
|
device_watcher_add_Removed,
|
|
device_watcher_remove_Removed,
|
|
device_watcher_add_EnumerationCompleted,
|
|
device_watcher_remove_EnumerationCompleted,
|
|
device_watcher_add_Stopped,
|
|
device_watcher_remove_Stopped,
|
|
device_watcher_get_Status,
|
|
device_watcher_Start,
|
|
device_watcher_Stop,
|
|
};
|
|
|
|
static HRESULT device_watcher_create( HSTRING filter, IDeviceWatcher **out )
|
|
{
|
|
struct device_watcher *impl;
|
|
HRESULT hr;
|
|
|
|
if (!(impl = calloc( 1, sizeof(*impl) ))) return E_OUTOFMEMORY;
|
|
|
|
impl->IDeviceWatcher_iface.lpVtbl = &device_watcher_vtbl;
|
|
if (FAILED(hr = weak_reference_source_init( &impl->weak_reference_source, (IUnknown *)&impl->IDeviceWatcher_iface )))
|
|
{
|
|
free( impl );
|
|
return hr;
|
|
}
|
|
if (FAILED(hr = WindowsDuplicateString( filter, &impl->filter )))
|
|
{
|
|
weak_reference_strong_release( &impl->weak_reference_source );
|
|
free( impl );
|
|
return hr;
|
|
}
|
|
|
|
list_init( &impl->added_handlers );
|
|
list_init( &impl->enumerated_handlers );
|
|
list_init( &impl->stopped_handlers );
|
|
|
|
InitializeCriticalSectionEx( &impl->cs, 0, RTL_CRITICAL_SECTION_FLAG_FORCE_DEBUG_INFO );
|
|
impl->cs.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": device_watcher.cs");
|
|
impl->status = DeviceWatcherStatus_Created;
|
|
|
|
*out = &impl->IDeviceWatcher_iface;
|
|
TRACE( "created DeviceWatcher %p\n", *out );
|
|
return S_OK;
|
|
}
|
|
|
|
struct device_information_statics
|
|
{
|
|
IActivationFactory IActivationFactory_iface;
|
|
IDeviceInformationStatics IDeviceInformationStatics_iface;
|
|
IDeviceInformationStatics2 IDeviceInformationStatics2_iface;
|
|
LONG ref;
|
|
};
|
|
|
|
static inline struct device_information_statics *impl_from_IActivationFactory( IActivationFactory *iface )
|
|
{
|
|
return CONTAINING_RECORD( iface, struct device_information_statics, IActivationFactory_iface );
|
|
}
|
|
|
|
static HRESULT WINAPI activation_factory_QueryInterface( IActivationFactory *iface, REFIID iid, void **out )
|
|
{
|
|
struct device_information_statics *impl = impl_from_IActivationFactory( iface );
|
|
|
|
TRACE( "iface %p, iid %s, out %p stub!\n", iface, debugstr_guid( iid ), out );
|
|
|
|
if (IsEqualGUID( iid, &IID_IUnknown ) ||
|
|
IsEqualGUID( iid, &IID_IInspectable ) ||
|
|
IsEqualGUID( iid, &IID_IActivationFactory ))
|
|
{
|
|
IInspectable_AddRef( (*out = &impl->IActivationFactory_iface) );
|
|
return S_OK;
|
|
}
|
|
|
|
if (IsEqualGUID( iid, &IID_IDeviceInformationStatics ))
|
|
{
|
|
IInspectable_AddRef( (*out = &impl->IDeviceInformationStatics_iface) );
|
|
return S_OK;
|
|
}
|
|
|
|
if (IsEqualGUID( iid, &IID_IDeviceInformationStatics2 ))
|
|
{
|
|
IInspectable_AddRef( (*out = &impl->IDeviceInformationStatics2_iface) );
|
|
return S_OK;
|
|
}
|
|
|
|
FIXME( "%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid( iid ) );
|
|
*out = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
static ULONG WINAPI activation_factory_AddRef( IActivationFactory *iface )
|
|
{
|
|
struct device_information_statics *impl = impl_from_IActivationFactory( iface );
|
|
ULONG ref = InterlockedIncrement( &impl->ref );
|
|
TRACE( "iface %p, ref %lu.\n", iface, ref );
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI activation_factory_Release( IActivationFactory *iface )
|
|
{
|
|
struct device_information_statics *impl = impl_from_IActivationFactory( iface );
|
|
ULONG ref = InterlockedDecrement( &impl->ref );
|
|
TRACE( "iface %p, ref %lu.\n", iface, ref );
|
|
return ref;
|
|
}
|
|
|
|
static HRESULT WINAPI activation_factory_GetIids( IActivationFactory *iface, ULONG *iid_count, IID **iids )
|
|
{
|
|
FIXME( "iface %p, iid_count %p, iids %p stub!\n", iface, iid_count, iids );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI activation_factory_GetRuntimeClassName( IActivationFactory *iface, HSTRING *class_name )
|
|
{
|
|
FIXME( "iface %p, class_name %p stub!\n", iface, class_name );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI activation_factory_GetTrustLevel( IActivationFactory *iface, TrustLevel *trust_level )
|
|
{
|
|
FIXME( "iface %p, trust_level %p stub!\n", iface, trust_level );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI activation_factory_ActivateInstance( IActivationFactory *iface, IInspectable **instance )
|
|
{
|
|
FIXME( "iface %p, instance %p stub!\n", iface, instance );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const struct IActivationFactoryVtbl activation_factory_vtbl =
|
|
{
|
|
activation_factory_QueryInterface,
|
|
activation_factory_AddRef,
|
|
activation_factory_Release,
|
|
/* IInspectable methods */
|
|
activation_factory_GetIids,
|
|
activation_factory_GetRuntimeClassName,
|
|
activation_factory_GetTrustLevel,
|
|
/* IActivationFactory methods */
|
|
activation_factory_ActivateInstance,
|
|
};
|
|
|
|
DEFINE_IINSPECTABLE( device_statics, IDeviceInformationStatics, struct device_information_statics, IActivationFactory_iface );
|
|
|
|
static HRESULT WINAPI device_statics_CreateFromIdAsync( IDeviceInformationStatics *iface, HSTRING id,
|
|
IAsyncOperation_DeviceInformation **op )
|
|
{
|
|
FIXME( "iface %p, id %s, op %p stub!\n", iface, debugstr_hstring(id), op );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI device_statics_CreateFromIdAsyncAdditionalProperties( IDeviceInformationStatics *iface, HSTRING id,
|
|
IIterable_HSTRING *additional_properties,
|
|
IAsyncOperation_DeviceInformation **op )
|
|
{
|
|
FIXME( "iface %p, id %s, additional_properties %p, op %p stub!\n", iface, debugstr_hstring(id), additional_properties, op );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT find_all_async( IUnknown *invoker, IUnknown *param, PROPVARIANT *result )
|
|
{
|
|
static const struct vector_iids iids =
|
|
{
|
|
.vector = &IID_IVector_IInspectable,
|
|
.view = &IID_IVectorView_DeviceInformation,
|
|
.iterable = &IID_IIterable_DeviceInformation,
|
|
.iterator = &IID_IIterator_DeviceInformation,
|
|
};
|
|
IVectorView_DeviceInformation *view;
|
|
IVector_IInspectable *vector;
|
|
const DEV_OBJECT *objects;
|
|
ULONG len, i;
|
|
HRESULT hr;
|
|
|
|
TRACE( "invoker %p, param %p, result %p\n", invoker, param, result );
|
|
|
|
if (FAILED(hr = vector_create( &iids, (void *)&vector ))) return hr;
|
|
if (FAILED(hr = DevGetObjects( DevObjectTypeDeviceInterfaceDisplay, DevQueryFlagNone, 0, NULL, 0, NULL, &len, &objects )))
|
|
{
|
|
IVector_IInspectable_Release( vector );
|
|
return hr;
|
|
}
|
|
for (i = 0; i < len && SUCCEEDED(hr); i++)
|
|
{
|
|
IDeviceInformation *info;
|
|
if (SUCCEEDED(hr = device_information_create( objects[i].pszObjectId, &info )))
|
|
{
|
|
hr = IVector_IInspectable_Append( vector, (IInspectable *)info );
|
|
IDeviceInformation_Release( info );
|
|
}
|
|
}
|
|
DevFreeObjects( len, objects );
|
|
if (SUCCEEDED(hr)) hr = IVector_IInspectable_GetView( vector, (void *)&view );
|
|
IVector_IInspectable_Release( vector );
|
|
if (FAILED(hr)) return hr;
|
|
|
|
result->vt = VT_UNKNOWN;
|
|
result->punkVal = (IUnknown *)view;
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI device_statics_FindAllAsync( IDeviceInformationStatics *iface,
|
|
IAsyncOperation_DeviceInformationCollection **op )
|
|
{
|
|
TRACE( "iface %p, op %p\n", iface, op );
|
|
return async_operation_inspectable_create( &IID_IAsyncOperation_DeviceInformationCollection, (IUnknown *)iface, NULL,
|
|
find_all_async, (IAsyncOperation_IInspectable **)op );
|
|
}
|
|
|
|
static HRESULT WINAPI device_statics_FindAllAsyncDeviceClass( IDeviceInformationStatics *iface, DeviceClass class,
|
|
IAsyncOperation_DeviceInformationCollection **op )
|
|
{
|
|
FIXME( "iface %p, class %d, op %p stub!\n", iface, class, op );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI device_statics_FindAllAsyncAqsFilter( IDeviceInformationStatics *iface, HSTRING filter,
|
|
IAsyncOperation_DeviceInformationCollection **op )
|
|
{
|
|
FIXME( "iface %p, aqs %p, op %p stub!\n", iface, debugstr_hstring(filter), op );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI device_statics_FindAllAsyncAqsFilterAndAdditionalProperties( IDeviceInformationStatics *iface, HSTRING filter,
|
|
IIterable_HSTRING *additional_properties,
|
|
IAsyncOperation_DeviceInformationCollection **op )
|
|
{
|
|
FIXME( "iface %p, aqs %p, additional_properties %p, op %p stub!\n", iface, debugstr_hstring(filter), additional_properties, op );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI device_statics_CreateWatcher( IDeviceInformationStatics *iface, IDeviceWatcher **watcher )
|
|
{
|
|
TRACE( "iface %p, watcher %p\n", iface, watcher );
|
|
return device_watcher_create( NULL, watcher );
|
|
}
|
|
|
|
static HRESULT WINAPI device_statics_CreateWatcherDeviceClass( IDeviceInformationStatics *iface, DeviceClass class, IDeviceWatcher **watcher )
|
|
{
|
|
FIXME( "iface %p, class %d, watcher %p stub!\n", iface, class, watcher );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI device_statics_CreateWatcherAqsFilter( IDeviceInformationStatics *iface, HSTRING filter, IDeviceWatcher **watcher )
|
|
{
|
|
TRACE( "iface %p, filter %s, watcher %p\n", iface, debugstr_hstring(filter), watcher );
|
|
return device_watcher_create( filter, watcher );
|
|
}
|
|
|
|
static HRESULT WINAPI device_statics_CreateWatcherAqsFilterAndAdditionalProperties( IDeviceInformationStatics *iface, HSTRING filter,
|
|
IIterable_HSTRING *additional_properties, IDeviceWatcher **watcher )
|
|
{
|
|
FIXME( "iface %p, aqs %p, additional_properties %p, watcher %p stub!\n", iface, debugstr_hstring(filter), additional_properties, watcher );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const struct IDeviceInformationStaticsVtbl device_statics_vtbl =
|
|
{
|
|
device_statics_QueryInterface,
|
|
device_statics_AddRef,
|
|
device_statics_Release,
|
|
/* IInspectable methods */
|
|
device_statics_GetIids,
|
|
device_statics_GetRuntimeClassName,
|
|
device_statics_GetTrustLevel,
|
|
/* IDeviceInformationStatics methods */
|
|
device_statics_CreateFromIdAsync,
|
|
device_statics_CreateFromIdAsyncAdditionalProperties,
|
|
device_statics_FindAllAsync,
|
|
device_statics_FindAllAsyncDeviceClass,
|
|
device_statics_FindAllAsyncAqsFilter,
|
|
device_statics_FindAllAsyncAqsFilterAndAdditionalProperties,
|
|
device_statics_CreateWatcher,
|
|
device_statics_CreateWatcherDeviceClass,
|
|
device_statics_CreateWatcherAqsFilter,
|
|
device_statics_CreateWatcherAqsFilterAndAdditionalProperties,
|
|
};
|
|
|
|
DEFINE_IINSPECTABLE( device_statics2, IDeviceInformationStatics2, struct device_information_statics, IActivationFactory_iface );
|
|
|
|
static HRESULT WINAPI device_statics2_GetAqsFilterFromDeviceClass( IDeviceInformationStatics2 *iface, DeviceClass device_class,
|
|
HSTRING *filter )
|
|
{
|
|
FIXME( "iface %p, device_class %u, filter %p stub!\n", iface, device_class, filter );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI device_statics2_CreateFromIdAsync( IDeviceInformationStatics2 *iface, HSTRING device_id,
|
|
IIterable_HSTRING *additional_properties, DeviceInformationKind kind,
|
|
IAsyncOperation_DeviceInformation **async_operation )
|
|
{
|
|
FIXME( "iface %p, device_id %s, additional_properties %p, kind %u, async_operation %p stub!\n",
|
|
iface, debugstr_hstring( device_id ), additional_properties, kind, async_operation );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI device_statics2_FindAllAsync( IDeviceInformationStatics2 *iface, HSTRING filter,
|
|
IIterable_HSTRING *additional_properties, DeviceInformationKind kind,
|
|
IAsyncOperation_DeviceInformationCollection **async_operation )
|
|
{
|
|
FIXME( "iface %p, filter %s, additional_properties %p, kind %u, async_operation %p stub!\n",
|
|
iface, debugstr_hstring( filter ), additional_properties, kind, async_operation );
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI device_statics2_CreateWatcher( IDeviceInformationStatics2 *iface, HSTRING filter,
|
|
IIterable_HSTRING *additional_properties, DeviceInformationKind kind,
|
|
IDeviceWatcher **watcher )
|
|
{
|
|
FIXME( "iface %p, filter %s, additional_properties %p, kind %u, watcher %p semi-stub!\n",
|
|
iface, debugstr_hstring( filter ), additional_properties, kind, watcher );
|
|
return device_watcher_create( filter, watcher );
|
|
}
|
|
|
|
static const struct IDeviceInformationStatics2Vtbl device_statics2_vtbl =
|
|
{
|
|
device_statics2_QueryInterface,
|
|
device_statics2_AddRef,
|
|
device_statics2_Release,
|
|
/* IInspectable methods */
|
|
device_statics2_GetIids,
|
|
device_statics2_GetRuntimeClassName,
|
|
device_statics2_GetTrustLevel,
|
|
/* IDeviceInformationStatics2 methods */
|
|
device_statics2_GetAqsFilterFromDeviceClass,
|
|
device_statics2_CreateFromIdAsync,
|
|
device_statics2_FindAllAsync,
|
|
device_statics2_CreateWatcher
|
|
};
|
|
|
|
static struct device_information_statics device_information_statics =
|
|
{
|
|
{&activation_factory_vtbl},
|
|
{&device_statics_vtbl},
|
|
{&device_statics2_vtbl},
|
|
1
|
|
};
|
|
|
|
HRESULT WINAPI DllGetClassObject( REFCLSID clsid, REFIID riid, void **out )
|
|
{
|
|
FIXME( "clsid %s, riid %s, out %p stub!\n", debugstr_guid( clsid ), debugstr_guid( riid ), out );
|
|
return CLASS_E_CLASSNOTAVAILABLE;
|
|
}
|
|
|
|
HRESULT WINAPI DllGetActivationFactory( HSTRING classid, IActivationFactory **factory )
|
|
{
|
|
const WCHAR *buffer = WindowsGetStringRawBuffer( classid, NULL );
|
|
|
|
TRACE( "classid %s, factory %p.\n", debugstr_hstring( classid ), factory );
|
|
|
|
*factory = NULL;
|
|
|
|
if (!wcscmp( buffer, RuntimeClass_Windows_Devices_Enumeration_DeviceInformation ))
|
|
IActivationFactory_QueryInterface( &device_information_statics.IActivationFactory_iface,
|
|
&IID_IActivationFactory, (void **)factory );
|
|
else if (!wcscmp( buffer, RuntimeClass_Windows_Devices_Enumeration_DeviceAccessInformation ))
|
|
IActivationFactory_QueryInterface( device_access_factory, &IID_IActivationFactory, (void **)factory );
|
|
|
|
if (*factory) return S_OK;
|
|
return CLASS_E_CLASSNOTAVAILABLE;
|
|
}
|