|
|
/***************************************************************************
* * Copyright (C) 1997 Microsoft Corporation. All Rights Reserved. * * File: dihid.h * Content: DirectInput internal include file for HID * ***************************************************************************/
#ifdef HID_SUPPORT
#ifndef _DIHID_H
#define _DIHID_H
#ifndef HID_USAGE_PAGE_PID
#define HID_USAGE_PAGE_PID ( (USAGE) 0x0000f )
#endif
#ifndef HID_USAGE_PAGE_VENDOR
#define HID_USAGE_PAGE_VENDOR ( (USAGE) 0xff00 )
#endif
/*****************************************************************************
* * @doc INTERNAL * * @struct HIDDEVICEINFO | * * Records information about a single hid device. * * @field DIOBJECTSTATICDATA | osd | * * Standard information that identifies the device crudely. * * The <e DIOBJECTSTATICDATA.dwDevType> field contains the * device type code, used by * <f CDIDEnum_Next>. * * If the device is a HID mouse, then the remaining fields * are commandeered as follows: * * The <e DIOBJECTSTATICDATA.pcguid> field is the number * of buttons on the mouse. * * The <e DIOBJECTSTATICDATA.CreateDcb> field is the number * of axes on the mouse. * * See <f DIHid_ProbeMouse> for an explanation of why we * need to do this. * * @field PSP_DEVICE_INTERFACE_DETAIL_DATA | pdidd | * * Pointer to name for device to be used in <f CreateFile>. * * @field HKEY | hk | * * Registry key that contains configuration information. * Sadly, we must keep it open because there is no way to * obtain the name of the key, and the only way to open the * key is inside an enumeration. * * @field HKEY | hkOld | * * Registry key that contains configuration information. * This key was originally used in Win2k Gold. It is to * maintain compatibiltiy with Win2k Gold. * * @field LPTSTR | ptszId | * * Cached device ID that allows us to access other information * about the device. * * @field GUID | guid | * * The instance GUID for the device. * * @field GUID | guidProduct | * * The product GUID for the device. * * @field WORD | ProductID | * * The PID for the device * * @field WORD | VendorID | * * The VID for the device * *****************************************************************************/
typedef struct HIDDEVICEINFO { DIOBJECTSTATICDATA osd; PSP_DEVICE_INTERFACE_DETAIL_DATA pdidd; HKEY hk; HKEY hkOld; LPTSTR ptszId; GUID guid; GUID guidProduct; int idJoy; WORD ProductID; WORD VendorID; BOOL fAttached; } HIDDEVICEINFO, *PHIDDEVICEINFO;
/*****************************************************************************
* * @doc INTERNAL * * @struct HIDDEVICELIST | * * Records information about all the HID devices. * * @field int | chdi | * * Number of items in the list that are in use. * * @field int | chdiAlloc | * * Number of items allocated in the list. * * @field HIDDEVICEINFO | rghdi[0] | * * Variable-size array of device information structures. * *****************************************************************************/
typedef struct HIDDEVICELIST {
int chdi; int chdiAlloc; int idMaxJoy; HIDDEVICEINFO rghdi[0];
} HIDDEVICELIST, *PHIDDEVICELIST;
extern PHIDDEVICELIST g_phdl;
#define cbHdlChdi(chdi) FIELD_OFFSET(HIDDEVICELIST, rghdi[chdi])
/*
* We choose our starting point at 64 devices, since * that's the maximum number of USB devices supported. This * avoids needless reallocs. */
#define chdiMax 64
#define chdiInit 16
/*
* Tag for unused translation of object instance */ #define NOREGTRANSLATION (0x80000000)
/*
* VID/PID definitions used to handle analog devices */ #define MSFT_SYSTEM_VID (0x45E)
#define MSFT_SYSTEM_PID (0x100)
#define ANALOG_ID_ROOT TEXT("VID_045E&PID_01")
/*
* VID/PID template so that upper case hex is always used */ #define VID_PID_TEMPLATE TEXT("VID_%04X&PID_%04X")
/*
* Size of string in characters generated using VID_PID_TEMPLATE */ #define cbszVIDPID cA( VID_PID_TEMPLATE )
/*****************************************************************************
* * diextdll.c - Imports from optional external DLLs * * It is very important that HidD_GetHidGuid be the very last one. * *****************************************************************************/
#ifdef STATIC_DLLUSAGE
#define ExtDll_Init()
#else
void EXTERNAL ExtDll_Init(void); #endif
void EXTERNAL ExtDll_Term(void);
/*****************************************************************************
* * @doc INTERNAL * * @struct MANUALIMPORT | * * Records a single manual import. If it hasn't * yet been resolved, then the <e MANUALIMPORT.ptsz> * points to the procedure name. If it has been resolved * successfully, then <e MANUALIMPORT.pfn> points to * the resolved address. If it has not been resolved * successfully, then <e MANUALIMPORT.pfn> is garbage. * * @field LPCSTR | psz | * * Procdure name. Note that this is always an ANSI string. * * @field FARPROC | pfn | * * Procedure address. * *****************************************************************************/
typedef union MANUALIMPORT { FARPROC pfn; /* Procedure address */ } MANUALIMPORT, *PMANUALIMPORT;
#ifndef STATIC_DLLUSAGE
#ifndef WINNT
/*****************************************************************************
* * CFGMGR32 * * Note that this must match the CFGMGR32 section in diextdll.c * *****************************************************************************/
typedef union CFGMGR32 {
MANUALIMPORT rgmi[6]; /* number of functions we import */
struct { CONFIGRET ( WINAPI * _CM_Get_Child) ( OUT PDEVINST pdnDevInst, IN DEVINST dnDevInst, IN ULONG ulFlags );
CONFIGRET ( WINAPI * _CM_Get_Sibling) ( OUT PDEVINST pdnDevInst, IN DEVINST DevInst, IN ULONG ulFlags );
CONFIGRET ( WINAPI * _CM_Get_Parent) ( OUT PDEVINST pdnDevInst, IN DEVINST dnDevInst, IN ULONG ulFlags );
CONFIGRET ( WINAPI * _CM_Get_DevNode_Registry_Property) ( IN DEVINST dnDevInst, IN ULONG ulProperty, OUT PULONG pulRegDataType, OPTIONAL OUT PVOID Buffer, OPTIONAL IN OUT PULONG pulLength, IN ULONG ulFlags );
CONFIGRET ( WINAPI * _CM_Set_DevNode_Registry_Property) ( IN DEVINST dnDevInst, IN ULONG ulProperty, IN PVOID Buffer, OPTIONAL IN ULONG ulLength, IN ULONG ulFlags );
CONFIGRET( WINAPI * _CM_Get_Device_ID) ( IN DEVINST dnDevInst, OUT PTCHAR Buffer, IN ULONG BufferLen, IN ULONG ulFlags ); };
} CFGMGR32, *PFGMGR32;
extern CFGMGR32 g_cfgmgr32;
#undef CM_Get_Child
#undef CM_Get_Sibling
#undef CM_Get_Parent
#undef CM_Get_DevNode_Registry_Property
#undef CM_Set_DevNode_Registry_Property
#undef CM_Get_Device_ID
#define CM_Get_Child \
g_cfgmgr32._CM_Get_Child
#define CM_Get_Sibling \
g_cfgmgr32._CM_Get_Sibling
#define CM_Get_Parent \
g_cfgmgr32._CM_Get_Parent
#define CM_Get_DevNode_Registry_Property \
g_cfgmgr32._CM_Get_DevNode_Registry_Property
#define CM_Set_DevNode_Registry_Property \
g_cfgmgr32._CM_Set_DevNode_Registry_Property
#define CM_Get_Device_ID \
g_cfgmgr32._CM_Get_Device_ID #endif //#ifndef WINNT
/*****************************************************************************
* * SETUPAPI * * Note that this must match the SETUPAPI section in diextdll.c * *****************************************************************************/
typedef union SETUPAPI {
#ifdef WINNT
MANUALIMPORT rgmi[18]; /* number of functions we import */ #else
MANUALIMPORT rgmi[12]; /* number of functions we import */ #endif
struct {
HDEVINFO (WINAPI *_SetupDiGetClassDevs) ( IN LPGUID ClassGuid, OPTIONAL IN LPCTSTR Enumerator, OPTIONAL IN HWND hwndParent, OPTIONAL IN DWORD Flags );
BOOL (WINAPI *_SetupDiDestroyDeviceInfoList) ( IN HDEVINFO DeviceInfoSet );
BOOL (WINAPI *_SetupDiGetDeviceInterfaceDetail) ( IN HDEVINFO DeviceInfoSet, IN PSP_DEVICE_INTERFACE_DATA pdid, OUT PSP_DEVICE_INTERFACE_DETAIL_DATA pdidd, OPTIONAL IN DWORD cbDidd, OUT PDWORD RequiredSize, OPTIONAL OUT PSP_DEVINFO_DATA DeviceInfoData OPTIONAL );
BOOL (WINAPI *_SetupDiEnumDeviceInterfaces) ( IN HDEVINFO DeviceInfoSet, IN PSP_DEVINFO_DATA DeviceInfoData, OPTIONAL IN LPGUID InterfaceClassGuid, IN DWORD MemberIndex, OUT PSP_DEVICE_INTERFACE_DATA DeviceInterfaceData );
HKEY (WINAPI *_SetupDiCreateDeviceInterfaceRegKey) ( IN HDEVINFO hdev, IN PSP_DEVICE_INTERFACE_DATA pdid, IN DWORD Reserved, IN REGSAM samDesired, IN HINF InfHandle, OPTIONAL IN PCSTR InfSectionName OPTIONAL );
BOOL (WINAPI *_SetupDiCallClassInstaller) ( IN DI_FUNCTION InstallFunction, IN HDEVINFO DeviceInfoSet, IN PSP_DEVINFO_DATA DeviceInfoData OPTIONAL );
BOOL (WINAPI *_SetupDiGetDeviceRegistryProperty) ( IN HDEVINFO DeviceInfoSet, IN PSP_DEVINFO_DATA DeviceInfoData, IN DWORD Property, OUT PDWORD PropertyRegDataType, OPTIONAL OUT PBYTE PropertyBuffer, IN DWORD PropertyBufferSize, OUT PDWORD RequiredSize OPTIONAL );
BOOL (WINAPI *_SetupDiSetDeviceRegistryProperty) ( IN HDEVINFO DeviceInfoSet, IN OUT PSP_DEVINFO_DATA DeviceInfoData, IN DWORD Property, IN CONST BYTE* PropertyBuffer, IN DWORD PropertyBufferSize );
BOOL (WINAPI *_SetupDiGetDeviceInstanceId) ( IN HDEVINFO DeviceInfoSet, IN PSP_DEVINFO_DATA DeviceInfoData, OUT PTSTR DeviceInstanceId, IN DWORD DeviceInstanceIdSize, OUT PDWORD RequiredSize OPTIONAL );
BOOL (WINAPI *_SetupDiOpenDeviceInfo) ( IN HDEVINFO DeviceInfoSet, IN LPCTSTR DeviceInstanceId, IN HWND hwndParent, OPTIONAL IN DWORD OpenFlags, OUT PSP_DEVINFO_DATA DeviceInfoData OPTIONAL );
HDEVINFO (WINAPI *_SetupDiCreateDeviceInfoList) ( IN LPGUID ClassGuid, OPTIONAL IN HWND hwndParent OPTIONAL );
HKEY (WINAPI *_SetupDiOpenDevRegKey) ( IN HDEVINFO DeviceInfoSet, IN PSP_DEVINFO_DATA DeviceInfoData, IN DWORD Scope, IN DWORD HwProfile, IN DWORD KeyType, IN REGSAM samDesired );
#ifdef WINNT
CONFIGRET ( WINAPI * _CM_Get_Child) ( OUT PDEVINST pdnDevInst, IN DEVINST dnDevInst, IN ULONG ulFlags );
CONFIGRET ( WINAPI * _CM_Get_Sibling) ( OUT PDEVINST pdnDevInst, IN DEVINST DevInst, IN ULONG ulFlags );
CONFIGRET ( WINAPI * _CM_Get_Parent) ( OUT PDEVINST pdnDevInst, IN DEVINST dnDevInst, IN ULONG ulFlags );
CONFIGRET ( WINAPI * _CM_Get_DevNode_Registry_Property) ( IN DEVINST dnDevInst, IN ULONG ulProperty, OUT PULONG pulRegDataType, OPTIONAL OUT PVOID Buffer, OPTIONAL IN OUT PULONG pulLength, IN ULONG ulFlags );
CONFIGRET ( WINAPI * _CM_Set_DevNode_Registry_Property) ( IN DEVINST dnDevInst, IN ULONG ulProperty, IN PVOID Buffer, OPTIONAL IN ULONG ulLength, IN ULONG ulFlags );
CONFIGRET( WINAPI * _CM_Get_Device_ID) ( IN DEVINST dnDevInst, OUT PTCHAR Buffer, IN ULONG BufferLen, IN ULONG ulFlags ); #endif //#ifdef WINNT
};
} SETUPAPI, *PSETUPAPI;
extern SETUPAPI g_setupapi;
#undef SetupDiGetClassDevs
#undef SetupDiDestroyDeviceInfoList
#undef SetupDiGetDeviceInterfaceDetail
#undef SetupDiEnumDeviceInterfaces
#undef SetupDiCreateDeviceInterfaceRegKey
#undef SetupDiCallClassInstaller
#undef SetupDiGetDeviceRegistryProperty
#undef SetupDiSetDeviceRegistryProperty
#undef SetupDiGetDeviceInstanceId
#undef SetupDiOpenDeviceInfo
#undef SetupDiCreateDeviceInfoList
#undef SetupDiOpenDevRegKey
#define SetupDiGetClassDevs \
g_setupapi._SetupDiGetClassDevs
#define SetupDiDestroyDeviceInfoList \
g_setupapi._SetupDiDestroyDeviceInfoList
#define SetupDiGetDeviceInterfaceDetail \
g_setupapi._SetupDiGetDeviceInterfaceDetail
#define SetupDiEnumDeviceInterfaces \
g_setupapi._SetupDiEnumDeviceInterfaces
#define SetupDiCreateDeviceInterfaceRegKey \
g_setupapi._SetupDiCreateDeviceInterfaceRegKey
#define SetupDiCallClassInstaller \
g_setupapi._SetupDiCallClassInstaller
#define SetupDiGetDeviceRegistryProperty \
g_setupapi._SetupDiGetDeviceRegistryProperty
#define SetupDiSetDeviceRegistryProperty \
g_setupapi._SetupDiSetDeviceRegistryProperty
#define SetupDiGetDeviceInstanceId \
g_setupapi._SetupDiGetDeviceInstanceId
#define SetupDiOpenDeviceInfo \
g_setupapi._SetupDiOpenDeviceInfo
#define SetupDiCreateDeviceInfoList \
g_setupapi._SetupDiCreateDeviceInfoList
#define SetupDiOpenDevRegKey \
g_setupapi._SetupDiOpenDevRegKey
#ifdef WINNT
#undef CM_Get_Child
#undef CM_Get_Sibling
#undef CM_Get_Parent
#undef CM_Get_DevNode_Registry_Property
#undef CM_Set_DevNode_Registry_Property
#undef CM_Get_Device_ID
#define CM_Get_Child \
g_setupapi._CM_Get_Child
#define CM_Get_Sibling \
g_setupapi._CM_Get_Sibling
#define CM_Get_Parent \
g_setupapi._CM_Get_Parent
#define CM_Get_DevNode_Registry_Property \
g_setupapi._CM_Get_DevNode_Registry_Property
#define CM_Set_DevNode_Registry_Property \
g_setupapi._CM_Set_DevNode_Registry_Property
#define CM_Get_Device_ID \
g_setupapi._CM_Get_Device_ID #endif //#ifdef WINNT
/*****************************************************************************
* * HIDDLL * * Note that this must match the HID section in diextdll.c * *****************************************************************************/
typedef union HIDDLL {
MANUALIMPORT rgmi[21]; /* number of functions we import */
struct { void (__stdcall *_HidD_GetHidGuid) ( OUT LPGUID HidGuid );
BOOLEAN (__stdcall *_HidD_GetPreparsedData) ( IN HANDLE HidDeviceObject, OUT PHIDP_PREPARSED_DATA * PreparsedData );
BOOLEAN (__stdcall *_HidD_FreePreparsedData) ( IN PHIDP_PREPARSED_DATA PreparsedData );
BOOLEAN (__stdcall *_HidD_FlushQueue) ( IN HANDLE HidDeviceObject );
BOOLEAN (__stdcall *_HidD_GetAttributes) ( IN HANDLE HidDeviceObject, OUT PHIDD_ATTRIBUTES Attributes );
BOOLEAN (__stdcall *_HidD_GetFeature) ( IN HANDLE HidDeviceObject, OUT PVOID ReportBuffer, IN ULONG ReportBufferLength );
BOOLEAN (__stdcall *_HidD_SetFeature) ( IN HANDLE HidDeviceObject, IN PVOID ReportBuffer, IN ULONG ReportBufferLength );
BOOLEAN (__stdcall *_HidD_GetProductString) ( IN HANDLE HidDeviceObject, OUT PVOID Buffer, IN ULONG BufferLength );
BOOLEAN (__stdcall *_HidD_GetInputReport) ( IN HANDLE HidDeviceObject, OUT PVOID ReportBuffer, IN ULONG ReportBufferLength );
NTSTATUS (__stdcall *_HidP_GetCaps) ( IN PHIDP_PREPARSED_DATA PreparsedData, OUT PHIDP_CAPS Capabilities );
NTSTATUS (__stdcall *_HidP_GetButtonCaps) ( IN HIDP_REPORT_TYPE ReportType, OUT PHIDP_BUTTON_CAPS ButtonCaps, IN OUT PUSHORT ButtonCapsLength, IN PHIDP_PREPARSED_DATA PreparsedData );
NTSTATUS (__stdcall *_HidP_GetValueCaps) ( IN HIDP_REPORT_TYPE ReportType, OUT PHIDP_VALUE_CAPS ValueCaps, IN OUT PUSHORT ValueCapsLength, IN PHIDP_PREPARSED_DATA PreparsedData );
NTSTATUS (__stdcall *_HidP_GetLinkCollectionNodes) ( OUT PHIDP_LINK_COLLECTION_NODE LinkCollectionNodes, IN OUT PULONG LinkCollectionNodesLength, IN PHIDP_PREPARSED_DATA PreparsedData );
ULONG (__stdcall *_HidP_MaxDataListLength) ( IN HIDP_REPORT_TYPE ReportType, IN PHIDP_PREPARSED_DATA PreparsedData );
NTSTATUS (__stdcall *_HidP_GetUsagesEx) ( IN HIDP_REPORT_TYPE ReportType, IN USHORT LinkCollection, OUT PUSAGE_AND_PAGE ButtonList, IN OUT ULONG * UsageLength, IN PHIDP_PREPARSED_DATA PreparsedData, IN PCHAR Report, IN ULONG ReportLength );
NTSTATUS (__stdcall *_HidP_GetScaledUsageValue) ( IN HIDP_REPORT_TYPE ReportType, IN USAGE UsagePage, IN USHORT LinkCollection, IN USAGE Usage, OUT PLONG UsageValue, IN PHIDP_PREPARSED_DATA PreparsedData, IN PCHAR Report, IN ULONG ReportLength );
NTSTATUS (__stdcall *_HidP_GetData) ( IN HIDP_REPORT_TYPE ReportType, OUT PHIDP_DATA DataList, IN OUT PULONG DataLength, IN PHIDP_PREPARSED_DATA PreparsedData, IN PCHAR Report, IN ULONG ReportLength );
NTSTATUS (__stdcall *_HidP_SetData) ( IN HIDP_REPORT_TYPE ReportType, IN PHIDP_DATA DataList, IN OUT PULONG DataLength, IN PHIDP_PREPARSED_DATA PreparsedData, IN OUT PCHAR Report, IN ULONG ReportLength );
NTSTATUS (__stdcall *_HidP_GetUsageValue) ( IN HIDP_REPORT_TYPE ReportType, IN USAGE UsagePage, IN USHORT LinkCollection, IN USAGE Usage, OUT PULONG UsageValue, IN PHIDP_PREPARSED_DATA PreparsedData, IN PCHAR Report, IN ULONG ReportLength );
ULONG (__stdcall *_HidP_MaxUsageListLength) ( IN HIDP_REPORT_TYPE ReportType, IN USAGE UsagePage, IN PHIDP_PREPARSED_DATA PreparsedData );
NTSTATUS (__stdcall *_HidP_GetSpecificButtonCaps) ( IN HIDP_REPORT_TYPE ReportType, IN USAGE UsagePage, // Optional (0 => ignore)
IN USHORT LinkCollection, // Optional (0 => ignore)
IN USAGE Usage, // Optional (0 => ignore)
OUT PHIDP_BUTTON_CAPS ButtonCaps, IN OUT PUSHORT ButtonCapsLength, IN PHIDP_PREPARSED_DATA PreparsedData );
};
} HIDDLL, *PHIDDLL;
extern HIDDLL g_hiddll;
#undef HidD_GetHidGuid
#undef HidD_GetPreparsedData
#undef HidD_FreePreparsedData
#undef HidD_FlushQueue
#undef HidD_GetAttributes
#undef HidD_GetFeature
#undef HidD_SetFeature
#undef HidD_GetProductString
#undef HidD_GetInputReport
#undef HidP_GetCaps
#undef HidP_GetButtonCaps
#undef HidP_GetValueCaps
#undef HidP_GetLinkCollectionNodes
#undef HidP_MaxDataListLength
#undef HidP_GetUsagesEx
#undef HidP_GetScaledUsageValue
#undef HidP_GetData
#undef HidP_SetData
#undef HidP_GetUsageValue
#undef HidP_MaxUsageListLength
#undef HidP_GetSpecificButtonCaps
#define HidD_GetHidGuid \
g_hiddll._HidD_GetHidGuid
#define HidD_GetPreparsedData \
g_hiddll._HidD_GetPreparsedData
#define HidD_FreePreparsedData \
g_hiddll._HidD_FreePreparsedData
#define HidD_FlushQueue \
g_hiddll._HidD_FlushQueue
#define HidD_GetAttributes \
g_hiddll._HidD_GetAttributes \
#define HidD_GetFeature \
g_hiddll._HidD_GetFeature \
#define HidD_SetFeature \
g_hiddll._HidD_SetFeature \
#define HidD_GetProductString \
g_hiddll._HidD_GetProductString \
#define HidD_GetInputReport \
g_hiddll._HidD_GetInputReport \
#define HidP_GetCaps \
g_hiddll._HidP_GetCaps
#define HidP_GetButtonCaps \
g_hiddll._HidP_GetButtonCaps
#define HidP_GetValueCaps \
g_hiddll._HidP_GetValueCaps
#define HidP_GetLinkCollectionNodes \
g_hiddll._HidP_GetLinkCollectionNodes
#define HidP_MaxDataListLength \
g_hiddll._HidP_MaxDataListLength \
#define HidP_GetUsagesEx \
g_hiddll._HidP_GetUsagesEx \
#define HidP_GetScaledUsageValue \
g_hiddll._HidP_GetScaledUsageValue \
#define HidP_GetData \
g_hiddll._HidP_GetData \
#define HidP_SetData \
g_hiddll._HidP_SetData \
#define HidP_GetUsageValue \
g_hiddll._HidP_GetUsageValue \
#define HidP_MaxUsageListLength \
g_hiddll._HidP_MaxUsageListLength \
#define HidP_GetSpecificButtonCaps \
g_hiddll._HidP_GetSpecificButtonCaps \
/*****************************************************************************
* * WINMMDLL * * Note that this must match the WINMM section in diextdll.c * *****************************************************************************/
typedef union WINMMDLL {
MANUALIMPORT rgmi[4]; /* number of functions we import */
struct { MMRESULT ( WINAPI * _joyGetDevCaps) ( IN UINT uJoyID, OUT LPJOYCAPS pjc, IN UINT cbjc );
MMRESULT ( WINAPI * _joyGetPosEx) ( IN UINT uJoyID, OUT LPJOYINFOEX pji );
MMRESULT ( WINAPI * _joyGetPos) ( IN UINT uJoyID, OUT LPJOYINFO pji );
UINT ( WINAPI * _joyConfigChanged) ( IN DWORD dwFlags );
};
} WINMMDLL, *PWINMMDLL;
extern WINMMDLL g_winmmdll;
#undef joyGetDevCaps
#undef joyGetPosEx
#undef joyGetPos
#undef joyConfigChanged
#define joyGetDevCaps \
g_winmmdll._joyGetDevCaps
#define joyGetPosEx \
g_winmmdll._joyGetPosEx
#define joyGetPos \
g_winmmdll._joyGetPos
#define joyConfigChanged \
g_winmmdll._joyConfigChanged
/*****************************************************************************
* * USER32 * * Note that this must match the USER32 section in diextdll.c * *****************************************************************************/
#ifdef USE_WM_INPUT
typedef union USER32 {
MANUALIMPORT rgmi[2]; /* number of functions we import */
struct { BOOL ( WINAPI * _RegisterRawInputDevices) ( PCRAWINPUTDEVICE pRawInputDevices, UINT uiNumDevices, UINT cbSize );
UINT ( WINAPI * _GetRawInputData) ( HRAWINPUT hRawInput, UINT uiCommand, LPVOID pData, PUINT pcbSize, UINT cbSizeHeader ); };
} USER32, *PUSER32;
extern USER32 g_user32;
#undef RegisterRawInputDevices
#undef GetRawInputData
#define RegisterRawInputDevices \
g_user32._RegisterRawInputDevices
#define GetRawInputData \
g_user32._GetRawInputData
#endif
/*****************************************************************************
* * Dummy functions * * These functions are used only when some DLLs can't be loaded. * *****************************************************************************/
//cfgmgr32.dll
CONFIGRET WINAPI DIDummy_CM_Get_Child ( OUT PDEVINST pdnDevInst, IN DEVINST dnDevInst, IN ULONG ulFlags );
CONFIGRET WINAPI DIDummy_CM_Get_Sibling ( OUT PDEVINST pdnDevInst, IN DEVINST DevInst, IN ULONG ulFlags );
CONFIGRET WINAPI DIDummy_CM_Get_Parent ( OUT PDEVINST pdnDevInst, IN DEVINST dnDevInst, IN ULONG ulFlags );
CONFIGRET WINAPI DIDummy_CM_Get_DevNode_Registry_Property ( IN DEVINST dnDevInst, IN ULONG ulProperty, OUT PULONG pulRegDataType, OPTIONAL OUT PVOID Buffer, OPTIONAL IN OUT PULONG pulLength, IN ULONG ulFlags );
CONFIGRET WINAPI DIDummy_CM_Set_DevNode_Registry_Property ( IN DEVINST dnDevInst, IN ULONG ulProperty, IN PVOID Buffer, OPTIONAL IN ULONG ulLength, IN ULONG ulFlags );
CONFIGRET WINAPI DIDummy_CM_Get_Device_ID ( IN DEVINST dnDevInst, OUT PTCHAR Buffer, IN ULONG BufferLen, IN ULONG ulFlags );
//Setupapi.dll
HDEVINFO WINAPI DIDummy_SetupDiGetClassDevs ( IN LPGUID ClassGuid, OPTIONAL IN LPCTSTR Enumerator, OPTIONAL IN HWND hwndParent, OPTIONAL IN DWORD Flags );
BOOL WINAPI DIDummy_SetupDiDestroyDeviceInfoList ( IN HDEVINFO DeviceInfoSet );
BOOL WINAPI DIDummy_SetupDiGetDeviceInterfaceDetail ( IN HDEVINFO DeviceInfoSet, IN PSP_DEVICE_INTERFACE_DATA pdid, OUT PSP_DEVICE_INTERFACE_DETAIL_DATA pdidd, OPTIONAL IN DWORD cbDidd, OUT PDWORD RequiredSize, OPTIONAL OUT PSP_DEVINFO_DATA DeviceInfoData OPTIONAL );
BOOL WINAPI DIDummy_SetupDiEnumDeviceInterfaces ( IN HDEVINFO DeviceInfoSet, IN PSP_DEVINFO_DATA DeviceInfoData, OPTIONAL IN LPGUID InterfaceClassGuid, IN DWORD MemberIndex, OUT PSP_DEVICE_INTERFACE_DATA DeviceInterfaceData );
HKEY WINAPI DIDummy_SetupDiCreateDeviceInterfaceRegKey ( IN HDEVINFO hdev, IN PSP_DEVICE_INTERFACE_DATA pdid, IN DWORD Reserved, IN REGSAM samDesired, IN HINF InfHandle, OPTIONAL IN PCSTR InfSectionName OPTIONAL );
BOOL WINAPI DIDummy_SetupDiCallClassInstaller ( IN DI_FUNCTION InstallFunction, IN HDEVINFO DeviceInfoSet, IN PSP_DEVINFO_DATA DeviceInfoData OPTIONAL );
BOOL WINAPI DIDummy_SetupDiGetDeviceRegistryProperty ( IN HDEVINFO DeviceInfoSet, IN PSP_DEVINFO_DATA DeviceInfoData, IN DWORD Property, OUT PDWORD PropertyRegDataType, OPTIONAL OUT PBYTE PropertyBuffer, IN DWORD PropertyBufferSize, OUT PDWORD RequiredSize OPTIONAL );
BOOL WINAPI DIDummy_SetupDiSetDeviceRegistryProperty ( IN HDEVINFO DeviceInfoSet, IN OUT PSP_DEVINFO_DATA DeviceInfoData, IN DWORD Property, IN CONST BYTE* PropertyBuffer, IN DWORD PropertyBufferSize );
BOOL WINAPI DIDummy_SetupDiGetDeviceInstanceId ( IN HDEVINFO DeviceInfoSet, IN PSP_DEVINFO_DATA DeviceInfoData, OUT PTSTR DeviceInstanceId, IN DWORD DeviceInstanceIdSize, OUT PDWORD RequiredSize OPTIONAL );
BOOL WINAPI DIDummy_SetupDiOpenDeviceInfo ( IN HDEVINFO DeviceInfoSet, IN LPCTSTR DeviceInstanceId, IN HWND hwndParent, OPTIONAL IN DWORD OpenFlags, OUT PSP_DEVINFO_DATA DeviceInfoData OPTIONAL );
HDEVINFO WINAPI DIDummy_SetupDiCreateDeviceInfoList ( IN LPGUID ClassGuid, OPTIONAL IN HWND hwndParent OPTIONAL );
HKEY WINAPI DIDummy_SetupDiOpenDevRegKey ( IN HDEVINFO DeviceInfoSet, IN PSP_DEVINFO_DATA DeviceInfoData, IN DWORD Scope, IN DWORD HwProfile, IN DWORD KeyType, IN REGSAM samDesired );
// hid.dll
void __stdcall DIDummy_HidD_GetHidGuid ( OUT LPGUID HidGuid );
BOOLEAN __stdcall DIDummy_HidD_GetPreparsedData ( IN HANDLE HidDeviceObject, OUT PHIDP_PREPARSED_DATA * PreparsedData );
BOOLEAN __stdcall DIDummy_HidD_FreePreparsedData ( IN PHIDP_PREPARSED_DATA PreparsedData );
BOOLEAN __stdcall DIDummy_HidD_FlushQueue ( IN HANDLE HidDeviceObject );
BOOLEAN __stdcall DIDummy_HidD_GetAttributes ( IN HANDLE HidDeviceObject, OUT PHIDD_ATTRIBUTES Attributes );
BOOLEAN __stdcall DIDummy_HidD_GetFeature ( IN HANDLE HidDeviceObject, OUT PVOID ReportBuffer, IN ULONG ReportBufferLength );
BOOLEAN __stdcall DIDummy_HidD_SetFeature ( IN HANDLE HidDeviceObject, IN PVOID ReportBuffer, IN ULONG ReportBufferLength );
BOOLEAN __stdcall DIDummy_HidD_GetProductString ( IN HANDLE HidDeviceObject, OUT PVOID Buffer, IN ULONG BufferLength );
BOOLEAN __stdcall DIDummy_HidD_GetInputReport ( IN HANDLE HidDeviceObject, OUT PVOID ReportBuffer, IN ULONG ReportBufferLength );
NTSTATUS __stdcall DIDummy_HidP_GetCaps ( IN PHIDP_PREPARSED_DATA PreparsedData, OUT PHIDP_CAPS Capabilities );
NTSTATUS __stdcall DIDummy_HidP_GetButtonCaps ( IN HIDP_REPORT_TYPE ReportType, OUT PHIDP_BUTTON_CAPS ButtonCaps, IN OUT PUSHORT ButtonCapsLength, IN PHIDP_PREPARSED_DATA PreparsedData );
NTSTATUS __stdcall DIDummy_HidP_GetValueCaps ( IN HIDP_REPORT_TYPE ReportType, OUT PHIDP_VALUE_CAPS ValueCaps, IN OUT PUSHORT ValueCapsLength, IN PHIDP_PREPARSED_DATA PreparsedData );
NTSTATUS __stdcall DIDummy_HidP_GetLinkCollectionNodes ( OUT PHIDP_LINK_COLLECTION_NODE LinkCollectionNodes, IN OUT PULONG LinkCollectionNodesLength, IN PHIDP_PREPARSED_DATA PreparsedData );
ULONG __stdcall DIDummy_HidP_MaxDataListLength ( IN HIDP_REPORT_TYPE ReportType, IN PHIDP_PREPARSED_DATA PreparsedData );
NTSTATUS __stdcall DIDummy_HidP_GetUsagesEx //unused
( IN HIDP_REPORT_TYPE ReportType, IN USHORT LinkCollection, OUT PUSAGE_AND_PAGE ButtonList, IN OUT ULONG * UsageLength, IN PHIDP_PREPARSED_DATA PreparsedData, IN PCHAR Report, IN ULONG ReportLength );
NTSTATUS __stdcall DIDummy_HidP_GetScaledUsageValue //unused
( IN HIDP_REPORT_TYPE ReportType, IN USAGE UsagePage, IN USHORT LinkCollection, IN USAGE Usage, OUT PLONG UsageValue, IN PHIDP_PREPARSED_DATA PreparsedData, IN PCHAR Report, IN ULONG ReportLength );
NTSTATUS __stdcall DIDummy_HidP_GetData ( IN HIDP_REPORT_TYPE ReportType, OUT PHIDP_DATA DataList, IN OUT PULONG DataLength, IN PHIDP_PREPARSED_DATA PreparsedData, IN PCHAR Report, IN ULONG ReportLength );
NTSTATUS __stdcall DIDummy_HidP_SetData ( IN HIDP_REPORT_TYPE ReportType, IN PHIDP_DATA DataList, IN OUT PULONG DataLength, IN PHIDP_PREPARSED_DATA PreparsedData, IN OUT PCHAR Report, IN ULONG ReportLength );
NTSTATUS __stdcall DIDummy_HidP_GetUsageValue ( IN HIDP_REPORT_TYPE ReportType, IN USAGE UsagePage, IN USHORT LinkCollection, IN USAGE Usage, OUT PULONG UsageValue, IN PHIDP_PREPARSED_DATA PreparsedData, IN PCHAR Report, IN ULONG ReportLength );
ULONG __stdcall DIDummy_HidP_MaxUsageListLength ( IN HIDP_REPORT_TYPE ReportType, IN USAGE UsagePage, IN PHIDP_PREPARSED_DATA PreparsedData );
NTSTATUS __stdcall DIDummy_HidP_GetSpecificButtonCaps ( IN HIDP_REPORT_TYPE ReportType, IN USAGE UsagePage, IN USHORT LinkCollection, IN USAGE Usage, OUT PHIDP_BUTTON_CAPS ButtonCaps, IN OUT PUSHORT ButtonCapsLength, IN PHIDP_PREPARSED_DATA PreparsedData );
NTSTATUS __stdcall DIDummy_HidP_TranslateUsagesToI8042ScanCodes ( IN PUSAGE ChangedUsageList, // Those usages that changed
IN ULONG UsageListLength, IN HIDP_KEYBOARD_DIRECTION KeyAction, IN OUT PHIDP_KEYBOARD_MODIFIER_STATE ModifierState, IN PHIDP_INSERT_SCANCODES InsertCodesProcedure, IN PVOID InsertCodesContext );
// winmm.dll
MMRESULT WINAPI DIDummy_joyGetDevCaps ( IN UINT uJoyID, OUT LPJOYCAPS pjc, IN UINT cbjc );
MMRESULT WINAPI DIDummy_joyGetPosEx ( IN UINT uJoyID, OUT LPJOYINFOEX pji );
MMRESULT WINAPI DIDummy_joyGetPos ( IN UINT uJoyID, OUT LPJOYINFO pji );
UINT WINAPI DIDummy_joyConfigChanged ( IN DWORD dwFlags );
MMRESULT WINAPI DIDummy_mmioClose ( IN HMMIO hmmio, IN UINT fuClose );
HMMIO WINAPI DIDummy_mmioOpenA ( IN OUT LPSTR pszFileName, IN OUT LPMMIOINFO pmmioinfo, IN DWORD fdwOpen );
MMRESULT WINAPI DIDummy_mmioDescend ( IN HMMIO hmmio, IN OUT LPMMCKINFO pmmcki, IN const MMCKINFO FAR* pmmckiParent, IN UINT fuDescend );
MMRESULT WINAPI DIDummy_mmioCreateChunk ( IN HMMIO hmmio, IN LPMMCKINFO pmmcki, IN UINT fuCreate );
LONG WINAPI DIDummy_mmioRead ( IN HMMIO hmmio, OUT HPSTR pch, IN LONG cch );
LONG WINAPI DIDummy_mmioWrite ( IN HMMIO hmmio, IN const char _huge* pch, IN LONG cch );
MMRESULT WINAPI DIDummy_mmioAscend ( IN HMMIO hmmio, IN LPMMCKINFO pmmcki, IN UINT fuAscend );
// user32.dll
#ifdef USE_WM_INPUT
BOOL WINAPI DIDummy_RegisterRawInputDevices ( PCRAWINPUTDEVICE pRawInputDevices, UINT uiNumDevices, UINT cbSize );
UINT WINAPI DIDummy_GetRawInputData ( HRAWINPUT hRawInput, UINT uiCommand, LPVOID pData, PUINT pcbSize, UINT cbSizeHeader );
#endif // #ifdef USE_WM_INPUT
#endif /* STATIC_DLLUSAGE */
/*****************************************************************************
* * dihidenm.c - HID enumeration functions. * *****************************************************************************/
extern TCHAR g_tszIdLastRemoved[MAX_PATH]; //in dihidenm.c
extern DWORD g_tmLastRemoved; //in dihinenm.c
STDMETHODIMP hresFindHIDInstanceGUID(PCGUID pguid, CREATEDCB *pcdcb); STDMETHODIMP hresFindHIDDeviceInterface(LPCTSTR ptszPath, LPGUID pguidOut);
PHIDDEVICEINFO EXTERNAL phdiFindHIDInstanceGUID(PCGUID pguid); PHIDDEVICEINFO EXTERNAL phdiFindHIDDeviceInterface(LPCTSTR ptszPath);
void EXTERNAL DIHid_BuildHidList(BOOL fForce); void EXTERNAL DIHid_EmptyHidList(void);
BOOL EXTERNAL DIHid_GetDevicePath(HDEVINFO hdev, PSP_DEVICE_INTERFACE_DATA pdid, PSP_DEVICE_INTERFACE_DETAIL_DATA *ppdidd, PSP_DEVINFO_DATA pdinf);
BOOL EXTERNAL DIHid_GetDeviceInstanceId(HDEVINFO hdev, PSP_DEVINFO_DATA pdinf, LPTSTR *pptszId);
BOOL EXTERNAL DIHid_GetInstanceGUID(HKEY hk, LPGUID pguid);
/*****************************************************************************
* * diguid.c - GUID generation * *****************************************************************************/
void EXTERNAL DICreateGuid(LPGUID pguid); void EXTERNAL DICreateStaticGuid(LPGUID pguid, WORD pid, WORD vid);
/*****************************************************************************
* * dihid.c * *****************************************************************************/
/*****************************************************************************
* * We will just use the HID item index as our DirectInput * internal ID number, which is in turn an index into the * <t DIOBJECTDATAFORMAT> array. * * Keyboard support requires a translation table. * Other devices also a translation table so that the external * instance numbers can be made compatible with legacy ones and * so that secondary aliases can be separated from primary ones. * * Since HID restarts the item index counter at zero for * each of input, feature, and output, we need to do some * adjustment so there aren't any collisions. So we * shift the features to start after the inputs, and the * outputs to start after the features. * * The <e CHid.rgdwBase> array contains the amount by which * each group of HID item indexes has been shifted. * *****************************************************************************/
/*****************************************************************************
* * @doc INTERNAL * * @func BOOL | HidP_IsValidReportType | * * For debugging only. Check if a value is a valid * <t HIDP_REPORT_TYPE>. * * Note that we also create a "fake" report type in which * to record our collections. * * @field HIDP_REPORT_TYPE | type | * * One of the values * <c HidP_Input>, * <c HidP_Output>, * or * <c HidP_Feature>. Hopefully. * *****************************************************************************/
#define HidP_Max (HidP_Feature + 1)
#define HidP_Coll HidP_Max
#define HidP_MaxColl (HidP_Coll + 1)
BOOL INLINE HidP_IsValidReportType(HIDP_REPORT_TYPE type) { CAssertF(HidP_Input == 0); CAssertF(HidP_Output == 1); CAssertF(HidP_Feature == 2); return type < HidP_Max; }
/*****************************************************************************
* * There are three (overlapping) classes of HID reports. * * InputLike - HidP_Input and HidP_Feature * OutputLike - HidP_Output and HidP_Feature * NothingLike - HidP_Coll * *****************************************************************************/
BOOL INLINE HidP_IsInputLike(HIDP_REPORT_TYPE type) { return type == HidP_Input || type == HidP_Feature; }
BOOL INLINE HidP_IsOutputLike(HIDP_REPORT_TYPE type) { return type == HidP_Output || type == HidP_Feature; }
/*****************************************************************************
* * @doc INTERNAL * * @struct LMINMAX | * * Min and max, that's all. These are kept in structures * to make logical-to-physical and physical-to-logical * translations less gross. * * @field LONG | Min | * * The minimum value. * * @field LONG | Max | * * The maximum value. * *****************************************************************************/
typedef struct LMINMAX { LONG Min; LONG Max; } LMINMAX, *PLMINMAX;
typedef const LMINMAX *PCLMINMAX;
/*****************************************************************************
* * @doc INTERNAL * * @struct HIDGROUPCAPS | * * This structure unifies the various HID caps structures * <t HIDP_BUTTON_CAPS> and * <t HIDP_VALUE_CAPS>. * * @field HIDP_REPORT_TYPE | type | * * One of the values * <c HidP_Input>, * <c HidP_Output>, * or * <c HidP_Feature>. * * @field UINT | cObj | * * Number of objects in this group. * * @field USAGE | UsagePage | * * Usage page for all usages in the group. * * @field USAGE | UsageMin | * * First usage described by this group. The remaining * items are numbered consecutively starting from * this value. * * @field USHORT | StringMin | * * String for first usage described by this group. * The remaining strings are numbered consecutively * starting from this value, unless the string maximum * is reached, in which case all subsequent objects * share that last string. * * @field USHORT | StringMax | * * Last string. * * @field USHORT | DesignatorMin | * * Designator for first usage described by this group. * The remaining designators are numbered consecutively * starting from this value, unless the designator maximum * is reached, in which case all subsequent objects * share that last designator. * * @field USHORT | DesignatorMax | * * Last designator. * * @field USHORT | DataIndexMin | * * Data index for the first usage described by this group. * The remaining data index values are numbered consecutively * starting from this value. * * @field USHORT | usGranularity | * * If object is a POV or wheel, then contains device granularity. * * @field LONG | lMask | * * Mask bits used for sign extension. For example, if the * value is 8-bits, the mask will be 0xFFFFFF80, indicating * that bit 7 (0x00000080) is extended to fill the remainder * of the value. * * This field is used only by values. * * @field USHORT | BitSize | * * Number of bits devoted to this value, including the sign bit. * * ISSUE-2001/03/29-timgill structure field probably not used anywhere. * * @field USHORT | LinkCollection | * * HID link collection number. * * @field LMINMAX | Logical | * * Logical minimum and maximum values. * These are the extremes of raw values * that can validly be received from the device. * * This field is used only by values. * * @field LMINMAX | Physical | * * Physical minimum and maximum values. * This is the "actual" value * that the logical minimum and maximum value corresponds to. * * This field is used only by values, and is consulted * only when converting between DirectInput calibration * (which uses logical values) and VJOYD calibration * (which uses physical values). * * @field LONG | Null | * * The null value to be used for output. * * This field is used only by values. * * @field ULONG | Units | * * The HID units descriptor, if any. * * @field WORD | Exponent | * * The HID unit exponent, if any. * * @field WORD | wReportId | * * HID report Id * * @field BOOL | IsAbsolute | * * Nonzero if the group describes absolute axes. * * This field is used only by values. * * @field BOOL | IsValue | * * Nonzero if the group describes a HID value. * * Note that an analog pushbutton is reported by * DirectInput as a <c DIDFT_BUTTON>, but is * handled internally as a HID value. * * @field BOOL | IsAlias | * * Nonzero if the group describes an alias. * * @field BOOL | IsSigned | * * The return data is signed. * * @field BOOL | IsPolledPOV | * * Nonzero if the axis is a polled POV. * * @devnote New for DX6.1a * *****************************************************************************/
#define HIDGROUPCAPS_SIGNATURE 0x47444948 /* HIDG */
typedef struct HIDGROUPCAPS {
D(DWORD dwSignature;) HIDP_REPORT_TYPE type; UINT cObj;
USAGE UsagePage; USAGE UsageMin;
USHORT StringMin, StringMax; USHORT DesignatorMin, DesignatorMax; USHORT DataIndexMin;
USHORT usGranularity;
LONG lMask;
USHORT BitSize;
USHORT LinkCollection;
LMINMAX Logical; LMINMAX Physical;
LONG Null;
ULONG Units; WORD Exponent;
WORD wReportId; BOOL fReportDisabled; BOOL Reserved;
BOOL IsAbsolute; BOOL IsValue; BOOL IsAlias; BOOL IsSigned;
#ifdef WINNT
BOOL IsPolledPOV; #endif
} HIDGROUPCAPS, *PHIDGROUPCAPS;
/*****************************************************************************
* * @doc INTERNAL * * @struct HIDOBJCAPS | * * This structure contains various cached pointers for each * object on the device, allowing us to get at things like * the group caps and the calibration information. * * @field PHIDGROUPCAPS | pcaps | * * The <t PHIDGROUPCAPS> for the group the object belongs to. * * @field PJOYRANGECONVERT | pjrc | * * If non-NULL, then points to the range conversion information * for the object. * * @field int | idata | * * Index into the <t HIDP_DATA> array for the corresponding * output/feature report, * or <c -1> if the item is not in the output/feature report. * *****************************************************************************/
typedef struct HIDOBJCAPS { PHIDGROUPCAPS pcaps; PJOYRANGECONVERT pjrc; int idata; } HIDOBJCAPS, *PHIDOBJCAPS;
/*****************************************************************************
* * @doc INTERNAL * * @struct HIDREPORTINFO | * * This structure contains information that is used for * parsing HID reports. * * @field PHIDP_DATA | rgdata | * * Array used when parsing reports via * <f HidP_GetData> or <f HidP_SetData>. This MUST be aligned * correctly on some architechtures. * * @field PV | pvReport | * * The report itself. * * @field int | cdataMax | * * Number of elements in the <e HIDREPORTINFO.rgdata> array. * * @field int | cdataUsed | * * Number of elements in the <e HIDREPORTINFO.rgdata> array * that are actually in use. * * @field ULONG | cbReport | * * Number of bytes in the report. * * @field BOOL | fNeedClear | * * Nonzero if the report needs to be zero'd out because we * deleted something (most likely a button) from it. * The only way to delete an item from a report is to zero * out the entire report and then re-add everything back in. * * @field BOOL | fChanged | * * Nonzero if an element in the report has changed. * *****************************************************************************/
typedef struct HIDREPORTINFO { PHIDP_DATA rgdata; PV pvReport; int cdataMax; int cdataUsed; ULONG cbReport; BOOL fNeedClear; BOOL fChanged; } HIDREPORTINFO, *PHIDREPORTINFO;
/*****************************************************************************
* * @doc INTERNAL * * @struct CHid | * * The <i IDirectInputDeviceCallback> object for HID devices. * * @field IDirectInputDeviceCalllback | didc | * * The object (containing vtbl). * * @field PV | pvGroup2 | * * Pointer to group 2 memory. This field is a union with the * pointer to the first chunk of memory in the second memory group. * * @field HIDREPORTINFO | hriIn | * * HID input report parsing and state. * * This memory is the first chunk of group 2. * * @field HIDREPORTINFO | hriOut | * * HID output report parsing and state. * * @field HIDREPORTINFO | hriFea | * * HID feature report parsing and state. * * @field PV | pvPhys | * * Pointer to physical device status information updated * asynchronously by the data collection thread. * * @field PV | pvStage | * * Staging area used when the HID report is parsed. * * This memory is the last chunk of group 2. * * @field DWORD | cbPhys | * * Size of the physical device state. * * @field VXDINSTANCE * | pvi | * * The DirectInput instance handle. * * HID devices always run through ring 3, which is misleadingly * called "emulation". * * @field DWORD | dwDevType | * * Device type code. * * @field LPTSTR | ptszId | * * Setupapi device instance ID. Used to obtain things * like manufacturer name. * * @field LPTSTR | ptszPath | * * Path to the device, for <f CreateFile>. * * @field UINT | dwAxes | * * Number of axes on the device. * * @field UINT | dwButtons | * * Number of buttons on the device. * * @field UINT | dwPOVs | * * Number of POV controllers on the device. * * @field HANDLE | hdev | * * Handle to the device itself. This field is valid only * while the device is acquired. * * @field HANDLE | hdevEm | * * <f DuplicateHandle> of the <e CHid.hdev> which is used * by the worker thread. We need to keep this separate from * the main copy to avoid race conditions between the main * thread and the worker thread. * * @field HKEY | hkInstType | * * Per-instance registry key that contains additional configuration * information, equivalent to the joystick Type key. * * @field DWORD | rgdwBase[HidP_MaxColl] | * * Array of base indices for the three HID usage classes: * <c HidP_Input>, <c HidP_Output>, and <c HidP_Feature>. * We hide the <c HidP_Collection> base index here, too. * * @field PHIDOBJCAPS | rghoc | * * Pointer to array of * <t PHIDOBJCAPS>, one for each object on the device, * each of which in turn contains info about a single object. * * This memory is allocated as part of the * df.rgodf in the <t DIDATAFORMAT> structure * hence should not be freed separately. * * @field DIDATAFORMAT | df | * * The dynamically-generated data format based on the * usages on the HID device. * * @field DWORD | ibButtonData | * * The location of the button data inside the data format. * * @field DWORD | cbButtonData | * * The number of bytes of button data inside the data format. * * @field PBYTE * | rgpbButtonMasks | * * Pointer to a an array of pointers to byte strings to mask * the buttons relevant to a report. * * @field PHIDP_PREPARSED_DATA | ppd | * * Preparsed data generated by the HID subsystem. * * @field PHIDGROUPCAPS | rgcaps | * * Array of <t HIDGROUPCAPS> structures used to keep * track of the various buttons, groups, and collections. * * @field UINT | ccaps | * * Number of caps structures in the <e CHid.rgcaps> array. * * @field HIDP_CAPS | caps | * * Cached HID caps. * * @field OVERLAPPED | o | * * Overlapped I/O structure used by worker thread * for reading. * * @field PJOYRANGECONVERT | pjrcNext | * * Pointer to the first <t JOYRANGECONVERT> structure * (in a preallocated array) which has * yet to be used. * This structure is used for logical-to-physical * range conversion (a.k.a. calibration). * * This memory is allocated as part of the * df.rgodf in the <t DIDATAFORMAT> structure * hence should not be freed separately. * * This field is used during device initialization to * parcel out the <t JOYRANGECONVERT>s. Afterwards, * the field is <c NULL> if we did not create any * conversion structures (hence do not need to subclass * the cooperative window to catch recalibrations). * * @field PINT | rgiobj | * * This points to an array which maps DirectInput instance * values (DIDFT_GETINSTANCE) into object indices. * * @field PINT | rgipov | * * If we are not a keyboard, then this is the first element in * the above array which maps povs. * * @field PINT | rgiaxis | * * If we are not a keyboard, then this is the first element in * the above array which maps axes. * * @field PINT | rgicoll | * * If we are not a keyboard, then this is the first element in * the above array which maps collections. * //ISSUE-2001/03/29-timgill need to document keyboard case behaviour
* * @field UINT | uiInstanceMax | * * The number of elements in the above * <f rgiobj> array. * * @field int | idJoy | * * Joystick identifier for <f joyGetPosEx> and friends for * legacy access. * * This value starts out as -1, to meant that * the corresponding legacy joystick is unknown. * If we do something that requires the matched legacy * joystick to be found, we check if the current value * is still valid. If not (either it is -1 or the cached * value is stale), then we go hunt for the correct value. * * @field HKEY | hkType | * * The joystick type key opened with <c MAXIMUM_ALLOWED> access. * This is not per-instance; multiple instances of the same * hardware share this key. * * @field USHORT | VendorID | * * HID vendor ID for this device. * * @field USHORT | ProductID | * * HID product ID for this device. * * @field HWND | hwnd | * * The window which we have subclassed in order to watch * for recalibration messages. * * @field BOOL | IsPolledInput | * * Nonzero if the device has to be polled for Input data. * * @field BOOL | fPIDdevice | * * Set to true if the device is found to support PID. * * @field WORD | wMaxReportId | * * The maximum (number) of ReportId used by the HID device. * * @field PUCHAR | pEnableReportId | * * Pointer to (wMaxReportId) bytes. If a reportID needs to be * polled in order to get features / set Output, then that element * of this array is set to 0x1. * * @field BOOL | fEnableInputReport | * * True if Input report should be enabled for this device. * * @field BOOL | fFlags2Checked | * * True after we check the registry for Flags2 for disabling * input reports. * * @comm * * It is the caller's responsibility to serialize access as * necessary. * *****************************************************************************/
typedef struct CHid {
/* Supported interfaces */ IDirectInputDeviceCallback dcb;
union { PV pvGroup2; HIDREPORTINFO hriIn; };
HIDREPORTINFO hriOut; HIDREPORTINFO hriFea;
PV pvPhys; PV pvStage; DWORD cbPhys;
VXDINSTANCE *pvi;
DWORD dwDevType;
UINT dwAxes; UINT dwButtons; UINT dwPOVs; UINT dwCollections;
HANDLE hdev; HANDLE hdevEm;
DWORD rgdwBase[HidP_MaxColl]; PHIDOBJCAPS rghoc; DIDATAFORMAT df;
DWORD ibButtonData; DWORD cbButtonData; PBYTE *rgpbButtonMasks; PHIDP_PREPARSED_DATA ppd; PHIDGROUPCAPS rgcaps;
PJOYRANGECONVERT pjrcNext;
HIDP_CAPS caps;
ED ed; OVERLAPPED o; DWORD dwStartRead; DWORD dwStopRead;
PINT rgiobj; PINT rgipov; PINT rgiaxis; PINT rgicoll; UINT uiInstanceMax;
LPTSTR ptszId; LPTSTR ptszPath; HKEY hkInstType; UINT ccaps; int idJoy;
HKEY hkType; USHORT VendorID; USHORT ProductID; #define FAILED_POLL_THRESHOLD (0x4)
HWND hwnd; BOOL IsPolledInput; BOOL fPIDdevice; WORD wMaxReportId[HidP_Max]; PUCHAR pEnableReportId[HidP_Max];
DWORD dwVersion;
#if (DIRECTINPUT_VERSION > 0x061A)
DIAPPHACKS diHacks; #endif
BOOL fEnableInputReport; BOOL fFlags2Checked;
} CHid, CHID, *PCHID;
/*****************************************************************************
* * @doc INTERNAL * * @func PCHID | pchidFromPo | * * Given an interior pointer to an <t OVERLAPPED>, retrieve * a pointer to the parent <t CHid>. * * @parm LPOVERLAPPED | po | * * The pointer to convert. * *****************************************************************************/
PCHID INLINE pchidFromPo(LPOVERLAPPED po) { return pvSubPvCb(po, FIELD_OFFSET(CHid, o)); }
/*****************************************************************************
* * @doc INTERNAL * * @func PCHID | pchidFromPed | * * Given an interior pointer to a <t CEd>, retrieve * a pointer to the parent <t CHid>. * * @parm PED | ped | * * The pointer to convert. * *****************************************************************************/
PCHID INLINE pchidFromPed(PED ped) { return pvSubPvCb(ped, FIELD_OFFSET(CHid, ed)); }
/*****************************************************************************
* * @doc INTERNAL * * @func PCHID | pchidFromPem | * * Given a <t CEm>, wander back to the * <t CHid> that spawned it. * * @parm PEM | pem | * * The pointer at which to start. * *****************************************************************************/
PCHID INLINE pchidFromPem(PEM pem) { PCHID pchid = pchidFromPed(pem->ped); AssertF(pemFromPvi(pchid->pvi) == pem); return pchid; }
/*****************************************************************************
* * @doc INTERNAL * * @method UINT | CHid | ObjFromType | * * Given a <p dwType>, extract the instance number * and (if necessary) convert it to an object index. * Note, the instance number will always be of the primary instance * not an alias. * * @parm PCHID | this | * * HID device object. * * @parm DWORD | dwType | * * The type code to convert. * * @returns * * The object index, or an out-of-range value. * *****************************************************************************/
UINT INLINE CHid_ObjFromType(PCHID this, DWORD dwType) { UINT uiObj = DIDFT_GETINSTANCE(dwType);
// ISSUE-2001/03/29-timgill Range checks may be unnecessary
// MarcAnd can we ever get the out of range value?
// if so, can we really run with it?
// if not, can these range checks be converted into Asserts?
/*
* The range checking makes use of the fact that the translation * tables are taken from a contiguous memory allocation and that * aliased collections are not distinguished. */ if(this->rgiobj) { switch( DIDFT_GETTYPE(dwType) ) { case DIDFT_RELAXIS: case DIDFT_ABSAXIS: if( &this->rgiaxis[uiObj] < this->rgicoll ) { uiObj = this->rgiaxis[uiObj]; } else { uiObj = 0xFFFFFFFF; } break;
case DIDFT_PSHBUTTON: case DIDFT_TGLBUTTON: /*
* If it is keyboard, this->rgiobj == this->rgipov (see CHid_MungeKeyboard). * So, we can't test &this->rgiobj[uiObj] < this->rgipov. */ if( (GET_DIDEVICE_TYPE(this->dwDevType) == DIDEVTYPE_KEYBOARD && uiObj < this->uiInstanceMax ) || &this->rgiobj[uiObj] < this->rgipov ) { uiObj = this->rgiobj[uiObj]; } else { uiObj = 0xFFFFFFFF; } break;
case DIDFT_POV: if( &this->rgipov[uiObj] < this->rgiaxis ) { uiObj = this->rgipov[uiObj]; } else { uiObj = 0xFFFFFFFF; } break; case (DIDFT_COLLECTION | DIDFT_NODATA): if( &this->rgicoll[uiObj] <= &this->rgiobj[this->uiInstanceMax] ) { uiObj = this->rgicoll[uiObj]; } else { uiObj = 0xFFFFFFFF; } break; case DIDFT_NODATA: /*
* So far, this TYPE only shows up on Keyboard (HID_USAGE_PAGE_LED). */ if( GET_DIDEVICE_TYPE(this->dwDevType) == DIDEVTYPE_KEYBOARD && uiObj < this->uiInstanceMax ) { uiObj = this->rgiobj[uiObj]; } break; default: /*
* Hopefully this is just a vendor defined object but squirt * in debug as these may cause problems. */ SquirtSqflPtszV(sqflHidParse | sqflVerbose, TEXT("CHid_ObjFromType: dwType 0x%08x not converted"), dwType ); break; } } else { SquirtSqflPtszV(sqflHidParse | sqflError, TEXT("CHid_ObjFromType: Translation array missing") ); }
return uiObj; }
LONG EXTERNAL CHid_CoordinateTransform(PLMINMAX Dst, PLMINMAX Src, LONG lVal);
void EXTERNAL CHid_UpdateVjoydCalibration(PCHID this, UINT iobj);
void EXTERNAL CHid_UpdateCalibrationFromVjoyd(PCHID this, UINT iobj, LPDIOBJECTCALIBRATION pCal);
/*****************************************************************************
* * dihidini.c - Device callback initialization stuff * *****************************************************************************/
#define INITBUTTONFLAG 0x10000000
HRESULT EXTERNAL CHid_InitParseData(PCHID this);
HRESULT EXTERNAL CHid_Init(PCHID this, REFGUID rguid);
HANDLE EXTERNAL CHid_OpenDevicePath(PCHID this, DWORD dwAttributes );
UINT EXTERNAL CHid_LoadCalibrations(PCHID this);
BOOL EXTERNAL CHid_IsPolledDevice( HANDLE hdev );
/*****************************************************************************
* * dihiddat.c - HID data parsing/management * *****************************************************************************/
typedef HRESULT (FAR PASCAL * SENDHIDREPORT)(PCHID this, PHIDREPORTINFO phri);
void EXTERNAL CHid_ResetDeviceData(PCHID this, PHIDREPORTINFO phri, HIDP_REPORT_TYPE type); HRESULT EXTERNAL CHid_AddDeviceData(PCHID this, UINT uiObj, DWORD dwData); STDMETHODIMP CHid_PrepareDeviceData(PCHID this, PHIDREPORTINFO phri); STDMETHODIMP CHid_SendHIDReport(PCHID this, PHIDREPORTINFO phri, HIDP_REPORT_TYPE type, SENDHIDREPORT SendHIDReport);
NTSTATUS EXTERNAL CHid_ParseData(PCHID this, HIDP_REPORT_TYPE type, PHIDREPORTINFO phri);
HRESULT EXTERNAL DIHid_GetRegistryProperty(LPTSTR ptszId, DWORD dwProperty, LPDIPROPHEADER pdiph);
/*****************************************************************************
* * diemh.c - HID "emulation" * *****************************************************************************/
void EXTERNAL CEm_HID_Sync(PLLTHREADSTATE plts, PEM pem);
BOOL EXTERNAL CEm_HID_IssueRead( PCHID pchid );
#else //No HID Support
#define DIHid_BuildHidList(fForce)
#define hresFindHIDDeviceInterface(ptszPath, pguidOut) ( E_FAIL )
#endif /* _DIHID_H */
#endif /* HID_SUPPORT */
|