Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

480 lines
18 KiB

/*****************************************************************************
*
* DIHid.c
*
* Copyright (c) 1996 Microsoft Corporation. All Rights Reserved.
*
* Abstract:
*
* WINNT implementation of JOYHID.
*
* Contents:
*
* DIWdm_JoyHidMapping
* JoyReg_JoyIdToDeviceInterface
*
*****************************************************************************/
#include "dinputpr.h"
#undef sqfl
#define sqfl sqflWDM
#include "dijoyhid.h"
/*****************************************************************************
*
* @doc INTERNAL
*
* @func HRESULT EXTERNAL | DIWdm_JoyHidMapping |
*
* Does the work done by JoyHid on Win9x. This function
* maps the Joystick ID to a HID device and talks to the
* HID device to obtain its capabilities.
*
* @parm IN int | idJoy |
*
* The Id of the joystick to be located.
*
* @parm OUT PVXDINITPARAMS | pvip | OPTIONAL
* Address of a VXDINITPARAMS structure that is filled out
* by this function. This is an optional parameter
* and can be NULL
*
* @parm OUT LPDIJOYCONFIG | pcfg |
* Address of a DIJOYCONFIG structure that is filled out
* by this function. This is an optional parameter.
*
* @parm IN OUT LPDIJOYTYPEINFO | pdijti |
* Address of a DIJOYTYPEINFO structure that is filled out
* by this function. This is an optional parameter.
* If passed in, the hws.dwFlags is used to initialize the
* same flags in the DIJOYCONFIG structure.
*
* @returns HRESULT
* Returns a COM error code
*
*****************************************************************************/
/*
* ISSUE-2001/03/29-timgill function uses too much stack space
* This function uses over 4K of stack space!
* This causes the Win9x build to choke looking for _chkstk.
* Hack by forcing caller to pass pcfg and pdijti
*/
HRESULT EXTERNAL
DIWdm_JoyHidMapping
(
IN int idJoy,
OUT PVXDINITPARMS pvip, OPTIONAL
OUT LPDIJOYCONFIG pcfg, OPTIONAL
IN OUT LPDIJOYTYPEINFO pdijti
)
{
HRESULT hres;
PHIDDEVICEINFO phdi;
VXDINITPARMS vip;
// DIJOYCONFIG cfg;
// DIJOYTYPEINFO dijti;
DWORD wCaps = 0;
DIPROPINFO propi;
DIPROPSTRING dips;
DIPROPDWORD dipd;
BOOL fBadCalData = FALSE;
EnterProc(DIWdm_JoyHidMapping, (_ "uxx", idJoy, pvip, pcfg));
// AssertF(InCrit());
if( pvip == NULL )
{
ZeroX(vip);
vip.dwSize = cbX(vip);
pvip = &vip;
}
// AssertF(pcfg != NULL );
// if( pcfg == NULL )
// {
// ZeroX(cfg);
// cfg.dwSize = cbX(cfg);
// pcfg = & cfg;
// }
AssertF(pdijti != NULL );
// if(pdijti == NULL )
// {
// ZeroX(dijti);
// dijti.dwSize = cbX(dijti);
// pdijti = &dijti;
// }
/*
* Copy the type info because JOY_HWS_ISYOKE, JOY_HWS_ISCARCTRL and
* JOY_HWS_ISHEADTRACKER have no simple equivalents in HID so would
* otherwise get lost. No harm done if it's zero.
* Note, the dwFlags is built in pvip then copied elsewhere.
*/
pvip->dwFlags = pdijti->hws.dwFlags;
phdi = phdiFindJoyId(idJoy);
if( phdi != NULL )
{
IDirectInputDeviceCallback *pdcb;
hres = CHid_New(0, &phdi->guid,
&IID_IDirectInputDeviceCallback,
(PPV)&pdcb);
if( SUCCEEDED(hres) )
{
DIDEVCAPS dc;
hres = pdcb->lpVtbl->GetCapabilities(pdcb, &dc );
if( SUCCEEDED(hres) )
{
DIDEVICEINSTANCEW didi;
didi.dwSize = cbX(didi);
hres = pdcb->lpVtbl->GetDeviceInfo(pdcb, &didi);
if( SUCCEEDED(hres) )
{
LPDIDATAFORMAT pdf;
hres = pdcb->lpVtbl->GetDataFormat(pdcb, &pdf);
if( SUCCEEDED(hres) )
{
DIPROPCAL dipc;
DWORD axis, pov = 0;
ZeroBuf(pvip->Usages, 6 * cbX(pvip->Usages[0]));
hres = pdcb->lpVtbl->MapUsage(pdcb, CheckHatswitch->dwUsage, &propi.iobj);
if(SUCCEEDED(hres) )
{
pvip->dwPOV1usage = CheckHatswitch->dwUsage;
pvip->dwFlags |= CheckHatswitch->dwFlags;
wCaps |= CheckHatswitch->dwCaps;
propi.pguid = DIPROP_GRANULARITY;
propi.dwDevType = pdf->rgodf[propi.iobj].dwType;
hres = pdcb->lpVtbl->GetProperty(pdcb, &propi, &dipd.diph);
if( SUCCEEDED( hres ) )
{
if( dipd.dwData >= 9000 ) // 4 directional POV
{
wCaps |= JOYCAPS_POV4DIR;
if( pcfg != NULL ) {
pcfg->hwc.hwv.dwPOVValues[JOY_POVVAL_FORWARD] = JOY_POVFORWARD;
pcfg->hwc.hwv.dwPOVValues[JOY_POVVAL_BACKWARD] = JOY_POVBACKWARD;
pcfg->hwc.hwv.dwPOVValues[JOY_POVVAL_LEFT] = JOY_POVLEFT;
pcfg->hwc.hwv.dwPOVValues[JOY_POVVAL_RIGHT] = JOY_POVRIGHT;
}
} else // Continuous POV
{
wCaps |= JOYCAPS_POVCTS;
}
}
}
for( axis = 0; axis < cA(AxesUsages)-1; axis++ )
{
USAGES *pUse = &AxesUsages[axis];
DWORD dwCurAxisPos = pUse->dwAxisPos;
if( pvip->Usages[dwCurAxisPos] != 0) {
continue;
} else {
int i;
BOOL bHasUsed = FALSE;
for( i = 0; i < (int)dwCurAxisPos; i++ ) {
if( pvip->Usages[i] == pUse->dwUsage ) {
bHasUsed = TRUE;
break;
}
}
if( bHasUsed ) {
continue;
}
}
hres = pdcb->lpVtbl->MapUsage(pdcb, pUse->dwUsage, &propi.iobj);
if(SUCCEEDED(hres) )
{
pvip->Usages[dwCurAxisPos] = pUse->dwUsage;
pvip->dwFlags |= pUse->dwFlags;
wCaps |= pUse->dwCaps;
propi.pguid = DIPROP_CALIBRATION;
propi.dwDevType = pdf->rgodf[propi.iobj].dwType;
hres = pdcb->lpVtbl->GetProperty(pdcb, &propi, &dipc.diph);
if( SUCCEEDED(hres) && pcfg != NULL )
{
#ifdef WINNT
(&pcfg->hwc.hwv.jrvHardware.jpMin.dwX)[dwCurAxisPos]
= dipc.lMin;
(&pcfg->hwc.hwv.jrvHardware.jpMax.dwX)[dwCurAxisPos]
= dipc.lMax;
(&pcfg->hwc.hwv.jrvHardware.jpCenter.dwX)[dwCurAxisPos]
= CCal_Midpoint(dipc.lMin, dipc.lMax);
#else
DIPROPRANGE diprp;
DIPROPRANGE diprl;
propi.pguid = DIPROP_PHYSICALRANGE;
propi.dwDevType = pdf->rgodf[propi.iobj].dwType;
hres = pdcb->lpVtbl->GetProperty(pdcb, &propi, &diprp.diph);
if( SUCCEEDED( hres ) )
{
propi.pguid = DIPROP_LOGICALRANGE;
propi.dwDevType = pdf->rgodf[propi.iobj].dwType;
hres = pdcb->lpVtbl->GetProperty(pdcb, &propi, &diprl.diph);
if( SUCCEEDED( hres ) )
{
LONG lMin, lMax;
lMin = (&pcfg->hwc.hwv.jrvHardware.jpMin.dwX)[dwCurAxisPos]
= CHid_CoordinateTransform( (PLMINMAX)&diprp.lMin, (PLMINMAX)&diprl.lMin, dipc.lMin );
lMax = (&pcfg->hwc.hwv.jrvHardware.jpMax.dwX)[dwCurAxisPos]
= CHid_CoordinateTransform( (PLMINMAX)&diprp.lMin, (PLMINMAX)&diprl.lMin, dipc.lMax );
(&pcfg->hwc.hwv.jrvHardware.jpCenter.dwX)[dwCurAxisPos]
= CCal_Midpoint(lMin, lMax);
if( lMin >= lMax ) {
fBadCalData = TRUE;
break;
}
}
}
#endif
}
}
} //for (axis=0...
} //GetDataFormat
pvip->hres = S_OK;
pvip->dwSize = cbX(*pvip); /* Which version of VJOYD are we? */
pvip->dwFlags |= JOY_HWS_AUTOLOAD; /* Describes the device */
if(didi.wUsage == HID_USAGE_GENERIC_GAMEPAD )
pvip->dwFlags |= JOY_HWS_ISGAMEPAD ;
pvip->dwId = idJoy; /* Internal joystick ID */
pvip->dwFirmwareRevision = dc.dwFirmwareRevision;
pvip->dwHardwareRevision = dc.dwHardwareRevision;
pvip->dwFFDriverVersion = dc.dwFFDriverVersion;
pvip->dwFilenameLengths = lstrlen(phdi->pdidd->DevicePath);
pvip->pFilenameBuffer = phdi->pdidd->DevicePath;
//pvip->Usages[6];
//pvip->dwPOV1usage = 0x0;
pvip->dwPOV2usage = 0x0;
pvip->dwPOV3usage = 0x0;
/* Fill all fields of cfg */
if( pcfg != NULL ) {
AssertF(pcfg->dwSize == cbX(*pcfg) );
pcfg->guidInstance = phdi->guid;
pcfg->hwc.hws.dwNumButtons = dc.dwButtons;
pcfg->hwc.hws.dwFlags = pvip->dwFlags;
//pcfg.hwc.hwv.jrvHardware
//pcfg.hwc.hwv.dwPOVValues
pcfg->hwc.hwv.dwCalFlags = 0x0;
if( ( LOWORD(phdi->guidProduct.Data1) == MSFT_SYSTEM_VID )
&&( ( HIWORD(phdi->guidProduct.Data1) >= MSFT_SYSTEM_PID + JOY_HW_PREDEFMIN)
&&( HIWORD(phdi->guidProduct.Data1) < MSFT_SYSTEM_PID + JOY_HW_PREDEFMAX ) ) )
{
pcfg->hwc.dwType = HIWORD(phdi->guidProduct.Data1) - MSFT_SYSTEM_PID;
pcfg->hwc.dwUsageSettings = JOY_US_PRESENT | JOY_US_VOLATILE;
}
else
{
/*
* This value really does not matter much but ideally
* should be greater than or equal to JOY_HW_PREDEFMAX
* Add idJoy for best compatiblity with the old CPLs.
*/
pcfg->hwc.dwType = idJoy + JOY_HW_PREDEFMAX;
pcfg->hwc.dwUsageSettings = JOY_US_PRESENT | JOY_US_VOLATILE | JOY_US_ISOEM;
}
if(pcfg && pvip->Usages[ecRz]) {
pcfg->hwc.dwUsageSettings |= JOY_US_HASRUDDER;
}
pcfg->hwc.dwReserved = 0x0;
/*
* Default gain to nominal max so it does not get written
* to the registry unless it has some other value.
*/
pcfg->dwGain = DI_FFNOMINALMAX;
propi.pguid = DIPROP_FFGAIN;
propi.dwDevType = DIPH_DEVICE;
propi.iobj = 0xFFFFFFFF;
hres = pdcb->lpVtbl->GetProperty(pdcb, &propi, &dipd.diph);
if( SUCCEEDED(hres) )
{
pcfg->dwGain = dipd.dwData;
} else
{
// Failure to get gain is not crutial
hres = S_OK;
}
if( pcfg->hwc.dwType >= JOY_HW_PREDEFMAX )
{
#ifndef UNICODE
char szType[20];
#endif
/*
* This should work, but it doesn't in Win98, bug!
*
* wsprintfW(pcfg->wszType, L"VID_%04X&PID_%04X",
* LOWORD(didi.guidProduct.Data1), HIWORD(didi.guidProduct.Data1));
*/
#ifdef UNICODE
wsprintf(pcfg->wszType, VID_PID_TEMPLATE,
LOWORD(phdi->guidProduct.Data1), HIWORD(phdi->guidProduct.Data1));
#else
wsprintf(szType, VID_PID_TEMPLATE,
LOWORD(phdi->guidProduct.Data1), HIWORD(phdi->guidProduct.Data1));
AToU(pcfg->wszType, cA(pcfg->wszType), szType);
#endif
}
else
{
/*
* Predefined types do not have type strings for the
* uses the callers of this function need.
*/
ZeroX(pcfg->wszType);
}
if( fWinnt ) {
// No callout on NT
ZeroX(pcfg->wszCallout);
} else {
lstrcpyW( pcfg->wszCallout, L"joyhid.vxd" );
}
} // end of filling pcfg's fields
pdijti->dwSize = cbX(*pdijti);
pdijti->hws.dwNumButtons = dc.dwButtons;
pdijti->hws.dwFlags = pvip->dwFlags;
ZeroX(pdijti->clsidConfig);
propi.pguid = DIPROP_INSTANCENAME;
propi.dwDevType = DIPH_DEVICE;
propi.iobj = 0xFFFFFFFF;
hres = pdcb->lpVtbl->GetProperty(pdcb, &propi, &dips.diph);
if( hres != S_OK && lstrlenW(pdijti->wszDisplayName) != 0x0 )
{
// Failure to get friendly name
// We will try and use the OEM name from the registry
lstrcpyW(dips.wsz, pdijti->wszDisplayName);
//pdcb->lpVtbl->SetProperty(pdcb, &propi, &dips.diph);
}else if( SUCCEEDED(hres) )
{
// Use friendly name in the registry
lstrcpyW(pdijti->wszDisplayName, dips.wsz);
hres = S_OK;
}
} // GetDeviceInfo FAILED
} // GetCapabilities FAILED
Invoke_Release(&pdcb);
}
} else // No HID device for JoyID
{
hres = E_FAIL;
}
if( fBadCalData ) {
hres = E_FAIL;
}
ExitProcX(hres);
return hres;
}
/*****************************************************************************
*
* @doc INTERNAL
*
* @func HRESULT | JoyReg_JoyIdToDeviceInterface |
*
* Given a joystick ID number, obtain the device interface
* corresponding to it.
*
* @parm UINT | idJoy |
*
* Joystick ID number, zero-based.
*
* @parm PVXDINITPARMS | pvip |
*
* Receives init parameters from the driver.
*
* @parm LPTSTR | ptszBuf |
*
* A buffer of size <c MAX_PATH> in which the device interface
* path is built.
*
* @returns
* A pointer to the part of the <p ptszBuf> buffer that
* contains the actual device interface path.
*
*****************************************************************************/
LPTSTR EXTERNAL
JoyReg_JoyIdToDeviceInterface_NT
(
IN UINT idJoy,
OUT PVXDINITPARMS pvip,
OUT LPTSTR ptszBuf
)
{
HRESULT hres;
DIJOYCONFIG cfg;
DIJOYTYPEINFO dijti;
DllEnterCrit();
ZeroX(cfg);
ZeroX(dijti);
cfg.dwSize = cbX(cfg);
dijti.dwSize = cbX(dijti);
hres = DIWdm_JoyHidMapping(idJoy, pvip, &cfg, &dijti );
if( SUCCEEDED(hres ) )
{
AssertF( lstrlen(pvip->pFilenameBuffer) < MAX_PATH );
lstrcpy(ptszBuf, pvip->pFilenameBuffer);
}
DllLeaveCrit();
return ptszBuf;
}