|
|
/*****************************************************************************
* * DIDev.c * * Copyright (c) 1996 Microsoft Corporation. All Rights Reserved. * * Abstract: * * The standard implementation of IDirectInputDevice. * * This is the device-independent part. the device-dependent * part is handled by the IDirectInputDeviceCallback. * * And the IDirectInputEffect support lives in didevef.c. * * Contents: * * CDIDev_CreateInstance * *****************************************************************************/
#include "dinputpr.h"
#define INCLUDED_BY_DIDEV
#include "didev.h"
/*****************************************************************************
* * Declare the interfaces we will be providing. * *****************************************************************************/
Interface_Template_Begin(CDIDev) Primary_Interface_Template(CDIDev, TFORM(ThisInterfaceT)) Secondary_Interface_Template(CDIDev, SFORM(ThisInterfaceT)) Interface_Template_End(CDIDev)
/*****************************************************************************
* * @doc INTERNAL * * @global PDD * | g_rgpddForeground | * * A list of all devices which have obtained foreground * acquisition. Items on the list have been * held (not AddRef'd). * * @global UINT | g_cpddForeground | * * The number of entries in <p g_rgpddForeground>. When foreground * activation is lost, all objects in the array are unacquired. * * * @global UINT | g_cpddForegroundMax | * * The size of the <p g_rgpddForeground> array, including dummy * spaces that are not yet in use. * *****************************************************************************/
#ifdef IDirectInputDevice2Vtbl
/*
* ISSUE-2001/03/29-timgill We assume that all-zeros is a valid initialization. */ GPA g_gpaExcl; #define g_hgpaExcl (&g_gpaExcl)
#else
//
// ISSUE-2001/03/29-timgill The following variables should go into diexcl.c or didevex.c
//
PDD *g_rgpddForeground; UINT g_cpddForeground; UINT g_cpddForegroundMax;
#endif
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | IDirectInputDevice | NotAcquired | * * Check that the device is not acquired. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @returns * * Returns * <c S_OK> if all is well, or <c DIERR_ACQUIRED> if * the device is acquired. * *****************************************************************************/
#ifndef XDEBUG
#define IDirectInputDevice_NotAcquired_(pdd, z) \
_IDirectInputDevice_NotAcquired_(pdd) \
#endif
HRESULT INLINE IDirectInputDevice_NotAcquired_(PDD this, LPCSTR s_szProc) { HRESULT hres;
if(!this->fAcquired) { hres = S_OK; } else { RPF("ERROR %s: May not be called while device is acquired", s_szProc); hres = DIERR_ACQUIRED; } return hres; }
#define IDirectInputDevice_NotAcquired(pdd) \
IDirectInputDevice_NotAcquired_(pdd, s_szProc) \
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev | IsExclAcquired | * * Check that the device is acquired exclusively. * * The device critical section must already be held. * * @cwrap PDD | this * * @returns * * <c S_OK> if the device is exclusively acquired. * * <c DIERR_INPUTLOST> if acquisition has been lost. * * <c DIERR_NOTEXCLUSIVEACQUIRED> the device is acquired, * but not exclusively, or if the device is not acquired * at all. * *****************************************************************************/
STDMETHODIMP CDIDev_IsExclAcquired_(PDD this, LPCSTR s_szProc) { HRESULT hres;
AssertF(CDIDev_InCrit(this));
if(this->discl & DISCL_EXCLUSIVE) { if(this->fAcquired) { hres = S_OK; } else { hres = this->hresNotAcquired; if(hres == DIERR_NOTACQUIRED) { hres = DIERR_NOTEXCLUSIVEACQUIRED; } } } else { hres = DIERR_NOTEXCLUSIVEACQUIRED; }
if(s_szProc && hres == DIERR_NOTEXCLUSIVEACQUIRED) { RPF("ERROR %s: Device is not acquired in exclusive mode", s_szProc); } return hres; }
/*****************************************************************************
* * @doc INTERNAL * * @method void | IDirectInputDevice | EnterCrit | * * Enter the object critical section. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * *****************************************************************************/
void EXTERNAL CDIDev_EnterCrit_(struct CDIDev *this, LPCTSTR lptszFile, UINT line) { #ifdef XDEBUG
if( ! _TryEnterCritSec(&this->crst) ) { SquirtSqflPtszV(sqflCrit, TEXT("Device CritSec blocked @%s,%d"), lptszFile, line); EnterCriticalSection(&this->crst); }
SquirtSqflPtszV(sqflCrit, TEXT("Device CritSec Entered @%s,%d"), lptszFile, line); #else
EnterCriticalSection(&this->crst); #endif
this->thidCrit = GetCurrentThreadId(); InterlockedIncrement(&this->cCrit); }
/*****************************************************************************
* * @doc INTERNAL * * @method void | IDirectInputDevice | LeaveCrit | * * Leave the object critical section. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * *****************************************************************************/
void EXTERNAL CDIDev_LeaveCrit_(struct CDIDev *this, LPCTSTR lptszFile, UINT line) { #ifdef XDEBUG
AssertF(this->cCrit); AssertF(this->thidCrit == GetCurrentThreadId()); SquirtSqflPtszV(sqflCrit | sqflVerbose, TEXT("Device CritSec Leaving @%s,%d"), lptszFile, line); #endif
if(InterlockedDecrement(&this->cCrit) == 0) { this->thidCrit = 0; } LeaveCriticalSection(&this->crst); }
/*****************************************************************************
* * @doc INTERNAL * * @method void | IDirectInputDevice | SetNotifyEvent | * * Set the event associated with the device, if any. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * *****************************************************************************/
void EXTERNAL CDIDev_SetNotifyEvent(PDD this) { if(this->hNotify) { SetEvent(this->hNotify); } }
/*****************************************************************************
* * @doc INTERNAL * * @method void | IDirectInputDevice | SetForcedUnacquiredFlag | * * When forced unacquired happens, set the fOnceForcedUnacquired flag. * *****************************************************************************/
void EXTERNAL CDIDev_SetForcedUnacquiredFlag(PDD this) { /*
* This is an internal interface, so we can skimp on validation. */ this->fOnceForcedUnacquired = 1; }
/*****************************************************************************
* * @doc INTERNAL * * @method BOOL | CDIDev | InCrit | * * Nonzero if we are in the critical section. * *****************************************************************************/
#ifdef DEBUG
BOOL INTERNAL CDIDev_InCrit(PDD this) { return this->cCrit && this->thidCrit == GetCurrentThreadId(); }
#endif
#ifdef DEBUG
/*****************************************************************************
* * @doc INTERNAL * * @method BOOL | IDirectInputDevice | IsConsistent | * * Check that various state variables are consistent. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * *****************************************************************************/
#define VerifyF(f) if (!(f)) fRc = 0
BOOL INTERNAL CDIDev_IsConsistent(PDD this) { BOOL fRc = 1;
/*
* If acquired, then we must have a translation table, a state manager, * and a device callback. */
if(this->fAcquired) { VerifyF(this->pdix && this->GetState && this->pdcb != c_pdcbNil); }
/*
* If buffering, then must have a device callback. */ if(this->celtBuf) { VerifyF(this->pdcb != c_pdcbNil); }
/*
* If managing an instance, then make sure buffer variables are * consistent. */ if(this->pvi) { if(this->celtBuf) { VerifyF( this->pvi->pBuffer && this->pvi->pEnd && this->pvi->pHead && this->pvi->pTail); VerifyF(this->pvi->pBuffer < this->pvi->pEnd); VerifyF(fInOrder((DWORD)(UINT_PTR)this->pvi->pBuffer, (DWORD)(UINT_PTR)this->pvi->pHead, (DWORD)(UINT_PTR)this->pvi->pEnd)); VerifyF(fInOrder((DWORD)(UINT_PTR)this->pvi->pBuffer, (DWORD)(UINT_PTR)this->pvi->pTail, (DWORD)(UINT_PTR)this->pvi->pEnd)); } else { VerifyF( this->pvi->pBuffer == 0 && this->pvi->pEnd == 0 && this->pvi->pHead == 0 && this->pvi->pTail == 0); } }
/*
* The cooperative levels must match the cached window handle. */ VerifyF(fLimpFF(this->discl & (DISCL_FOREGROUND | DISCL_EXCLUSIVE), this->hwnd));
return fRc; }
#undef VerifyF
#endif
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | QueryInterface | * * Gives a client access to other interfaces on an object. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm IN REFIID | riid | * * The requested interface's IID. * * @parm OUT LPVOID * | ppvObj | * * Receives a pointer to the obtained interface. * * @returns * * Returns a COM error code. * * @xref OLE documentation for <mf IUnknown::QueryInterface>. * ***************************************************************************** * * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | AddRef | * * Increments the reference count for the interface. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @returns * * Returns the object reference count. * * @xref OLE documentation for <mf IUnknown::AddRef>. * ***************************************************************************** * * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | Release | * * Decrements the reference count for the interface. * If the reference count on the object falls to zero, * the object is freed from memory. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @returns * * Returns the object reference count. * * @xref OLE documentation for <mf IUnknown::Release>. * *****************************************************************************/
#ifdef DEBUG
Default_QueryInterface(CDIDev) Default_AddRef(CDIDev) Default_Release(CDIDev)
#else
#define CDIDev_QueryInterface Common_QueryInterface
#define CDIDev_AddRef Common_AddRef
#define CDIDev_Release Common_Release
#endif
#ifdef IDirectInputDevice2Vtbl
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | IDirectInput | QIHelper | * * Support the original IDirectInputDevice interfaces as well * as the new IDirectInputDevice2 interfaces. * * @parm IN REFIID | riid | * * The requested interface's IID. * * @parm OUT LPVOID * | ppvObj | * * Receives a pointer to the obtained interface. * *****************************************************************************/
STDMETHODIMP CDIDev_QIHelper(PDD this, RIID riid, PPV ppvObj) { HRESULT hres; EnterProcI(CDIDev_QIHelper, (_ "pG", this, riid));
if(IsEqualIID(riid, &IID_IDirectInputDeviceA)) {
*ppvObj = &this->ddA; OLE_AddRef(this); hres = S_OK;
} else if(IsEqualIID(riid, &IID_IDirectInputDeviceW)) {
*ppvObj = &this->ddW; OLE_AddRef(this); hres = S_OK;
#ifdef IDirectInputDevice7Vtbl
} else if(IsEqualIID(riid, &IID_IDirectInputDevice2A)) {
*ppvObj = &this->ddA; OLE_AddRef(this); hres = S_OK;
} else if(IsEqualIID(riid, &IID_IDirectInputDevice2W)) {
*ppvObj = &this->ddW; OLE_AddRef(this); hres = S_OK; #endif
} else { hres = Common_QIHelper(this, riid, ppvObj); }
ExitOleProcPpv(ppvObj); return hres; }
#else
#define CDIDev_QIHelper Common_QIHelper
#endif
/*****************************************************************************
* * @doc INTERNAL * * @func HRESULT | CDIDev_Reset | * * Releases all the resources of a generic device that * are associated with a particular device instance. * * This method is called in preparation for reinitialization. * * It is the responsibility of the caller to have taken * any necessary critical sections. * * * @parm PV | pvObj | * * Object being reset. Note that it may not have been * completely initialized, so everything should be done * carefully. * *****************************************************************************/
STDMETHODIMP CDIDev_Reset(PDD this) { HRESULT hres;
if(!this->fAcquired) {
/*
* Note! that we must release the driver before releasing * the callback, because the callback will unload the * driver DLL. * * We cannot allow people to reset the device * while there are still effects outstanding, * because that would cause us to throw away the * callback while the effects are still using * the effect driver! */ #ifdef IDirectInputDevice2Vtbl
if(this->gpaEff.cpv == 0) { Invoke_Release(&this->pes); #endif
Invoke_Release(&this->pdcb); this->pdcb = c_pdcbNil; FreePpv(&this->pdix); FreePpv(&this->rgiobj); #ifdef BUGGY_DX7_WINNT
FreePpv(&this->pdix2); FreePpv(&this->rgiobj2); #endif //BUGGY_DX7_WINNT
FreePpv(&this->pvBuffer); FreePpv(&this->pvLastBuffer); FreePpv(&this->rgdwAxesOfs); #ifdef IDirectInputDevice2Vtbl
FreePpv(&this->rgemi); #endif
FreePpv(&this->rgdwPOV);
AssertF(!this->fAcquired); AssertF(!this->fAcquiredInstance);
if(this->hNotify) { CloseHandle(this->hNotify); }
ZeroBuf(&this->hwnd, FIELD_OFFSET(DD, celtBufMax) - FIELD_OFFSET(DD, hwnd)); ZeroX(this->guid); this->celtBufMax = 1023; this->GetDeviceState = CDIDev_GetAbsDeviceState; this->hresNotAcquired = DIERR_NOTACQUIRED; this->fCook = 0;
AssertF(this->hNotify == 0); #ifdef IDirectInputDevice2Vtbl
AssertF(this->cemi == 0); AssertF(this->didcFF == 0); this->dwGain = 10000; /* Default to full gain */ this->dwAutoCenter = DIPROPAUTOCENTER_ON; /* Default to centered */ GPA_InitFromZero(&this->gpaEff); #endif
/*
* If the app does not initialize, then assume DX 3. */ this->didftInstance = 0x0000FF00;
hres = S_OK;
#ifdef IDirectInputDevice2Vtbl
} else { RPF("IDirectInputDevice::Initialize: Device still has effects"); hres = DIERR_HASEFFECTS; } #endif
} else { RPF("IDirectInputDevice::Initialize: Device is busy"); hres = DIERR_ACQUIRED; } return hres; }
/*****************************************************************************
* * @doc INTERNAL * * @func void | CDIDev_AppFinalize | * * The application has performed its final release. * * If the device is acquired, then unacquire it. * #ifdef IDirectInputDevice2Vtbl
* Release the holds on all the created effects that * we have been hanging onto for enumeration purposes. #endif
* * @parm PV | pvObj | * * Object being released. Note that it may not have been * completely initialized, so everything should be done * carefully. * *****************************************************************************/
void INTERNAL CDIDev_AppFinalize(PV pvObj) { PDD this = pvObj;
if(this->fAcquired) { RPF("IDirectInputDevice::Release: Forgot to call Unacquire()"); CDIDev_InternalUnacquire(pvObj); }
#ifdef IDirectInputDevice2Vtbl
if(this->fCritInited) { /*
* Stop all the effects, if they are playing. * * Then unhold them (because we're done). * * ISSUE-2001/03/29-timgill Need to totally remove all effects created by a destroyed device * We also need to neuter them so they don't * do anything any more. Otherwise, an app might * destroy the parent device and then try to mess with * an effect created by that device after the device * is gone. * * Note that we cannot call the effect while inside our * private little critical section, because the effect * may need to do crazy things to stop itself. * * (It will almost certainly call back up into the device * to remove itself from the list of created effects.) */ UINT ipdie; PPDIE rgpdie; UINT cpdie;
CDIDev_EnterCrit(this);
rgpdie = (PV)this->gpaEff.rgpv; cpdie = this->gpaEff.cpv; GPA_Init(&this->gpaEff);
CDIDev_LeaveCrit(this);
for(ipdie = 0; ipdie < cpdie; ipdie++) { AssertF(rgpdie[ipdie]); // 7/19/2000(a-JiTay): IA64: Use %p format specifier for 32/64-bit pointers.
SquirtSqflPtszV(sqfl | sqflError, TEXT("Device %p forgot to destroy effect %08x"), this, rgpdie[ipdie]);
IDirectInputEffect_Stop(rgpdie[ipdie]); Common_Unhold(rgpdie[ipdie]); } FreePpv(&rgpdie); } #endif
}
/*****************************************************************************
* * @doc INTERNAL * * @func void | CDIDev_Finalize | * * Releases the resources of a generic device. * * @parm PV | pvObj | * * Object being released. Note that it may not have been * completely initialized, so everything should be done * carefully. * *****************************************************************************/
void INTERNAL CDIDev_Finalize(PV pvObj) { HRESULT hres; PDD this = pvObj;
#ifdef XDEBUG
if(this->cCrit) { RPF("IDirectInputDevice::Release: Another thread is using the object; crash soon!"); } #endif
AssertF(!this->fAcquired);
/*
* Note that we cannot take the critical section because it * might not exist. (We might've died during initialization.) * Fortunately, we finalize only after every possible client * (both internal and external) has done its final Release(), * so it's impossible for any other method to get called at * this point. */ hres = CDIDev_Reset(this); AssertF(SUCCEEDED(hres));
if(this->fCritInited) { DeleteCriticalSection(&this->crst); } }
/*****************************************************************************
* * @doc INTERNAL * * @method void | CDIDev | GetVersions | * * Obtain version information from the device. * * First try to get it from the effect driver. If * that doesn't work, then get it from the VxD. * And if that doesn't work, then tough. * * @parm IN OUT LPDIDRIVERVERSIONS | pvers | * * Receives version information. * *****************************************************************************/
void INLINE CDIDev_GetVersions(PDD this, LPDIDRIVERVERSIONS pvers) { HRESULT hres;
/*
* Pre-fill with zeros in case nobody implements GetVersions. */ pvers->dwSize = cbX(*pvers); pvers->dwFirmwareRevision = 0; pvers->dwHardwareRevision = 0; pvers->dwFFDriverVersion = 0;
hres = CDIDev_CreateEffectDriver(this); if(SUCCEEDED(hres) && SUCCEEDED(hres = this->pes->lpVtbl-> GetVersions(this->pes, pvers))) { } else { hres = this->pdcb->lpVtbl->GetVersions(this->pdcb, pvers); } }
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | GetCapabilities | * * Obtains information about the device. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm IN OUT LPDIDEVCAPS | lpdc | * * Points to a <t DIDEVCAPS> structure that is filled in * by the function. The <e DIDEVCAPS.dwSize> * field "must" be filled in * by the application before calling this method. * See the documentation of the <t DIDEVCAPS> structure * for additional information. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c DIERR_INVALIDPARAM> = <c E_INVALIDARG>: The * <p lpDirectInputDevice> or * <p lpdc> parameter is invalid. * *****************************************************************************/
STDMETHODIMP CDIDev_GetCapabilities(PV pdd, LPDIDEVCAPS pdc _THAT) { HRESULT hres; EnterProcR(IDirectInputDevice::GetCapabilities, (_ "pp", pdd, pdc));
if(SUCCEEDED(hres = hresPvT(pdd)) && SUCCEEDED(hres = hresFullValidWritePxCb2(pdc, DIDEVCAPS_DX5, DIDEVCAPS_DX3, 1))) { PDD this = _thisPv(pdd);
CDIDev_EnterCrit(this);
/*
* For the convenience of the callback, zero out all the fields, * save for the dwSize. */ ZeroBuf(pvAddPvCb(pdc, cbX(DWORD)), pdc->dwSize - cbX(DWORD)); hres = this->pdcb->lpVtbl->GetCapabilities(this->pdcb, pdc);
if(SUCCEEDED(hres)) { /*
* We'll handle the DIDC_EMULATED and * DIDC_POLLEDDATAFORMAT bits to save the callback * some trouble. */ AssertF(this->pvi); if(this->pvi->fl & VIFL_EMULATED) { pdc->dwFlags |= DIDC_EMULATED; } if(this->fPolledDataFormat) { pdc->dwFlags |= DIDC_POLLEDDATAFORMAT; }
#ifdef IDirectInputDevice2Vtbl
/*
* Add in the force feedback flags, too. */ pdc->dwFlags |= this->didcFF;
/*
* If the caller wants force feedback parameters, then * set them, too. */ if(pdc->dwSize >= cbX(DIDEVCAPS_DX5)) { DIDRIVERVERSIONS vers;
pdc->dwFFSamplePeriod = this->ffattr.dwFFSamplePeriod; pdc->dwFFMinTimeResolution = this->ffattr.dwFFMinTimeResolution;
CDIDev_GetVersions(this, &vers); pdc->dwFirmwareRevision = vers.dwFirmwareRevision; pdc->dwHardwareRevision = vers.dwHardwareRevision; pdc->dwFFDriverVersion = vers.dwFFDriverVersion;
} #endif
hres = S_OK; }
CDIDev_LeaveCrit(this);
ScrambleBit(&pdc->dwDevType, DIDEVTYPE_RANDOM); ScrambleBit(&pdc->dwFlags, DIDC_RANDOM);
}
ExitOleProcR(); return hres; }
#ifdef XDEBUG
CSET_STUBS(GetCapabilities, (PV pdd, LPDIDEVCAPS pdc), (pdd, pdc THAT_))
#else
#define CDIDev_GetCapabilitiesA CDIDev_GetCapabilities
#define CDIDev_GetCapabilitiesW CDIDev_GetCapabilities
#endif
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev | GetDataFormat | * * Get the data format for the device if we don't have it already. * * @parm PDD | this | * * Device object. * * @returns * * COM return code. * *****************************************************************************/
STDMETHODIMP CDIDev_GetDataFormat(PDD this) { HRESULT hres; LPDIDATAFORMAT pdf;
/*
* If the DIDATAFORMAT structure changes, you also need to invent * a new DCB message (DIDM_GETDATAFORMAT2), and then do * the right thing when faced with a mix of old and new. */
hres = this->pdcb->lpVtbl->GetDataFormat(this->pdcb, &pdf);
/*
* Note! We don't support external drivers in this release, * so it's okay to treat these are Assert's and not try to recover. */
if(SUCCEEDED(hres)) { AssertF(pdf->dwSize == sizeof(this->df)); this->df = *pdf;
AssertF(!IsBadReadPtr(pdf->rgodf, cbCxX(pdf->dwNumObjs, ODF)));
/*
* Prepare the axis goo in case the app sets relative mode. */ if(SUCCEEDED(hres = ReallocCbPpv(pdf->dwDataSize, &this->pvLastBuffer)) && SUCCEEDED(hres = ReallocCbPpv(cbCdw(pdf->dwNumObjs), &this->rgdwAxesOfs))) {
UINT iobj; this->cAxes = 0;
for(iobj = 0; iobj < pdf->dwNumObjs; iobj++) { AssertF(pdf->rgodf[iobj].dwOfs < pdf->dwDataSize); if(pdf->rgodf[iobj].dwType & DIDFT_AXIS) { this->rgdwAxesOfs[this->cAxes++] = pdf->rgodf[iobj].dwOfs; } } }
} return hres; }
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev | GetPolled | * * Determine whether the device is polled. * * @parm PDD | this | * * Device object. * * @returns * * COM result code. * *****************************************************************************/
STDMETHODIMP CDIDev_GetPolled(PDD this) { HRESULT hres; DIDEVCAPS_DX3 dc;
/*
* We intentionally use a DIDEVCAPS_DX3 because going for * a full DIDEVCAPS_DX5 requires us to load the force * feedback driver which is pointless for our current * goal. */ ZeroX(dc); dc.dwSize = cbX(dc);
hres = this->pdcb->lpVtbl->GetCapabilities(this->pdcb, (PV)&dc); if(SUCCEEDED(hres)) { if(dc.dwFlags & DIDC_POLLEDDEVICE) { this->hresPolled = DI_POLLEDDEVICE; } else { this->hresPolled = S_OK; } } return hres; }
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev | GetObjectInfoHelper | * * Set up all the information we can deduce ourselves and * have the callback finish. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm LPCDIPROPINFO | ppropi | * * Object descriptor. * * @parm LPDIDEVICEOBJECTINSTANCEW | pdoiW | * * Structure to receive result. * *****************************************************************************/ STDMETHODIMP CDIDev_GetObjectInfoHelper(PDD this, LPCDIPROPINFO ppropi, LPDIDEVICEOBJECTINSTANCEW pdoiW) { HRESULT hres;
AssertF(IsValidSizeDIDEVICEOBJECTINSTANCEW(pdoiW->dwSize)); pdoiW->guidType = *this->df.rgodf[ppropi->iobj].pguid; pdoiW->dwType = this->df.rgodf[ppropi->iobj].dwType; pdoiW->dwFlags = this->df.rgodf[ppropi->iobj].dwFlags; ScrambleBit(&pdoiW->dwFlags, DIDOI_RANDOM); if( this->pdix ) { /* User data format offset */ pdoiW->dwOfs = this->pdix[ppropi->iobj].dwOfs; } else { #ifdef BUGGY_DX7_WINNT
if( (this->dwVersion < 0x700) && (this->dwVersion != 0x5B2) && this->pdix2 ) { pdoiW->dwOfs = this->pdix2[ppropi->iobj].dwOfs; } else #endif //BUGGY_DX7_WINNT
{ /* Internal data format offset */ pdoiW->dwOfs = this->df.rgodf[ppropi->iobj].dwOfs; } }
/*
* Wipe out everything starting at tszName. */ ZeroBuf(&pdoiW->tszName, pdoiW->dwSize - FIELD_OFFSET(DIDEVICEOBJECTINSTANCEW, tszName));
hres = this->pdcb->lpVtbl->GetObjectInfo(this->pdcb, ppropi, pdoiW);
return hres; }
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | EnumObjects | * * Enumerate the input sources (buttons, axes) * available on a device. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm IN LPDIENUMDEVICEOBJECTSCALLBACK | lpCallback | * * Callback function. * * @parm IN LPVOID | pvRef | * * Reference data (context) for callback. * * @parm IN DWORD | fl | * * Flags specifying the type(s) of objects to be enumerated. * See the section "DirectInput Data Format Types" for a * list of flags that can be passed. * * Furthermore, the enumeration can be restricted to objects * from a single HID link collection by using the * <f DIDFT_ENUMCOLLECTION> macro. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * Note that if the callback stops the enumeration prematurely, * the enumeration is considered to have succeeded. * * <c DIERR_INVALIDPARAM> = <c E_INVALIDARG>: The * <p fl> parameter contains invalid flags, or the callback * procedure returned an invalid status code. * * @cb BOOL CALLBACK | DIEnumDeviceObjectsProc | * * An application-defined callback function that receives * DirectInputDevice objects as a result of a call to the * <om IDirectInputDevice::EnumObjects> method. * * @parm IN LPCDIDEVICEOBJECTINSTANCE | lpddoi | * * A <t DIDEVICEOBJECTINSTANCE> structure which describes * the object being enumerated. * * @parm IN OUT LPVOID | pvRef | * Specifies the application-defined value given in the * <mf IDirectInputDevice::EnumObjects> function. * * @returns * * Returns <c DIENUM_CONTINUE> to continue the enumeration * or <c DIENUM_STOP> to stop the enumeration. * * @ex * * To enumerate all axis objects: * * | * * // C++
* HRESULT hr = pDevice->EnumObjects(EnumProc, RefData, DIDFT_AXIS); * * // C
* hr = IDirectInputDevice_EnumObjects(pDevice, EnumProc, RefData, * DIDFT_AXIS); * * @ex * * To enumerate all objects in the third HID link collection: * * | * * // C++
* HRESULT hr = pDevice->EnumObjects(EnumProc, RefData, * DIDFT_ENUMCOLLECTION(3)); * * // C
* hr = IDirectInputDevice_EnumObjects(pDevice, EnumProc, RefData, * DIDFT_ENUMCOLLECTION(3)); * *****************************************************************************/
STDMETHODIMP CDIDev_EnumObjectsW (PV pddW, LPDIENUMDEVICEOBJECTSCALLBACKW pec, LPVOID pvRef, DWORD fl) { HRESULT hres; EnterProcR(IDirectInputDevice::EnumObjectsW, (_ "ppx", pddW, pec, fl));
if(SUCCEEDED(hres = hresPvW(pddW)) && SUCCEEDED(hres = hresFullValidPfn(pec, 1)) && SUCCEEDED(hres = hresFullValidFl(fl, DIDFT_ENUMVALID, 3))) { PDD this = _thisPvNm(pddW, ddW); DWORD flExclude; WORD wCollection;
/*
* We snapshot the object information underneath the critical * section so we don't blow up if another thread Reset()s * the device in the middle of an enumeration. The DIDATAFORMAT * contains pointers to the dcb, so we need to AddRef the * dcb as well. */ AssertF(!CDIDev_InCrit(this)); CDIDev_EnterCrit(this);
if(this->pdcb != c_pdcbNil) { DIPROPINFO propi; DIDATAFORMAT df; IDirectInputDeviceCallback *pdcb;
pdcb = this->pdcb; OLE_AddRef(pdcb); df = this->df;
CDIDev_LeaveCrit(this); AssertF(!CDIDev_InCrit(this));
/*
* If the client is DX3, don't enumerate things that * didn't exist in DX3. */ if(this->dwVersion <= 0x0300) { flExclude = DIDFT_ALLOBJS & ~DIDFT_ALLOBJS_DX3; } else { flExclude = 0;
/* Exclude alises if necessary */ if( !(fl & DIDFT_ALIAS) ) { flExclude |= DIDFT_ALIAS; } else { fl &= ~DIDFT_ALIAS; }
/* Exclude Vendor Defined fields */ if( !(fl & DIDFT_VENDORDEFINED) ) { flExclude |= DIDFT_VENDORDEFINED; } else { fl &= ~DIDFT_VENDORDEFINED; } }
if(fl == DIDFT_ALL) { fl = DIDFT_ALLOBJS; }
/*
* Pull out the link collection we are enumerating. * Note: Backwards compatibility hack. We can't * use link collection 0 to mean "no parent" because * 0 means "don't care". So instead, we use 0xFFFF * to mean "no parent". This means that we need to * interchange 0 and 0xFFFF before entering the main * loop. */ wCollection = DIDFT_GETCOLLECTION(fl); switch(wCollection) {
case 0: wCollection = 0xFFFF; break;
case DIDFT_GETCOLLECTION(DIDFT_NOCOLLECTION): wCollection = 0; break; }
propi.pguid = 0;
#ifdef BUGGY_DX7_WINNT
//Whistler bug 151479 - for pre-DX7 apps on WINNT only, make sure that the
//enumeration order is same as the winmm one
if( (this->dwVersion < 0x700) && (this->dwVersion != 0x5B2) && (this->pdix2) && (this->rgiobj2)) { UINT iobj; for(iobj = 0; iobj < this->dwDataSize2; iobj++) { propi.iobj = this->rgiobj2[iobj]; if (propi.iobj == -1) { continue; }
propi.dwDevType = df.rgodf[propi.iobj].dwType; if((propi.dwDevType & fl & DIDFT_TYPEMASK) && fHasAllBitsFlFl(propi.dwDevType, fl & DIDFT_ATTRMASK) && !(propi.dwDevType & flExclude)) { DIDEVICEOBJECTINSTANCEW doiW; doiW.dwSize = cbX(doiW);
hres = CDIDev_GetObjectInfoHelper(this, &propi, &doiW); if(SUCCEEDED(hres) && fLimpFF(wCollection != 0xFFFF, doiW.wCollectionNumber == wCollection)) { BOOL fRc = Callback(pec, &doiW, pvRef);
switch(fRc) { case DIENUM_STOP: goto enumdoneok; case DIENUM_CONTINUE: break; default: RPF("IDirectInputDevice::EnumObjects: Invalid return value from enumeration callback"); ValidationException(); break; } } else { goto enumdonefail; } } } } else #endif //BUGGY_DX7_WINNT
{ for(propi.iobj = 0; propi.iobj < df.dwNumObjs; propi.iobj++) { propi.dwDevType = df.rgodf[propi.iobj].dwType; if((propi.dwDevType & fl & DIDFT_TYPEMASK) && fHasAllBitsFlFl(propi.dwDevType, fl & DIDFT_ATTRMASK) && !(propi.dwDevType & flExclude)) { DIDEVICEOBJECTINSTANCEW doiW; doiW.dwSize = cbX(doiW);
hres = CDIDev_GetObjectInfoHelper(this, &propi, &doiW); if(SUCCEEDED(hres) && fLimpFF(wCollection != 0xFFFF, doiW.wCollectionNumber == wCollection)) { BOOL fRc = Callback(pec, &doiW, pvRef);
switch(fRc) { case DIENUM_STOP: goto enumdoneok; case DIENUM_CONTINUE: break; default: RPF("IDirectInputDevice::EnumObjects: Invalid return value from enumeration callback"); ValidationException(); break; } } else { goto enumdonefail; } } } }
enumdoneok:; hres = S_OK; enumdonefail:;
OLE_Release(pdcb);
} else { CDIDev_LeaveCrit(this); RPF("ERROR: IDirectInputDevice: Not initialized"); hres = DIERR_NOTINITIALIZED; } }
ExitOleProcR(); return hres; }
#define CDIDev_EnumObjects2W CDIDev_EnumObjectsW
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | IDirectInputDeviceA | EnumObjectsCallbackA | * * Custom callback that wraps * <mf IDirectInputDeviceW::EnumObjects> which * translates the UNICODE string to ANSI. * * @parm IN LPCDIENUMDEVICEOBJECTINSTANCE | pdoiW | * * Structure to be translated to ANSI. * * @parm IN LPVOID | pvRef | * * Pointer to <t struct ENUMDEVICEOBJECTINFO> which describes * the original callback. * * @returns * * Returns whatever the original callback returned. * *****************************************************************************/
typedef struct ENUMOBJECTSINFO { LPDIENUMDEVICEOBJECTSCALLBACKA pecA; PV pvRef; } ENUMOBJECTSINFO, *PENUMOBJECTSINFO;
BOOL CALLBACK CDIDev_EnumObjectsCallbackA(LPCDIDEVICEOBJECTINSTANCEW pdoiW, PV pvRef) { PENUMOBJECTSINFO peoi = pvRef; BOOL fRc; DIDEVICEOBJECTINSTANCEA doiA; EnterProc(CDIDev_EnumObjectsCallbackA, (_ "GxxWp", &pdoiW->guidType, pdoiW->dwOfs, pdoiW->dwType, pdoiW->tszName, pvRef));
doiA.dwSize = cbX(doiA); ObjectInfoWToA(&doiA, pdoiW);
fRc = peoi->pecA(&doiA, peoi->pvRef);
ExitProcX(fRc); return fRc; }
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | IDirectInputDeviceA | EnumObjects | * * Enumerate the input sources (buttons, axes) * available on a device, in ANSI. * See <mf IDirectInputDevice::EnumObjects> for more information. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm IN LPDIENUMDEVICEOBJECTSCALLBACK | lpCallback | * * Same as <mf IDirectInputDeviceW::EnumObjects>, except in ANSI. * * @parm IN LPVOID | pvRef | * * Same as <mf IDirectInputDeviceW::EnumObjects>. * * @parm IN DWORD | fl | * * Same as <mf IDirectInputDeviceW::EnumObjects>. * *****************************************************************************/
STDMETHODIMP CDIDev_EnumObjectsA (PV pddA, LPDIENUMDEVICEOBJECTSCALLBACKA pec, LPVOID pvRef, DWORD fl) { HRESULT hres; EnterProcR(IDirectInputDeviceA::EnumDevices, (_ "pppx", pddA, pec, pvRef, fl));
/*
* EnumObjectsW will validate the rest. */ if(SUCCEEDED(hres = hresPvA(pddA)) && SUCCEEDED(hres = hresFullValidPfn(pec, 1))) { ENUMOBJECTSINFO eoi = { pec, pvRef}; PDD this = _thisPvNm(pddA, ddA);
hres = CDIDev_EnumObjectsW(&this->ddW, CDIDev_EnumObjectsCallbackA, &eoi, fl); }
ExitOleProcR(); return hres; }
#define CDIDev_EnumObjects2A CDIDev_EnumObjectsA
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | SetEventNotification | * * Specify the event that should be set when the device * state changes, or turns off such notifications. * * A device state change is defined as any of the following: * * - A change in the position of an axis. * * - A change in the state (pressed or released) of a button. * * - A change in the direction of a POV control. * * - Loss of acquisition. * * "It is an error" to call <f CloseHandle> on the event * while it has been selected into an <i IDirectInputDevice> * object. You must call * <mf IDirectInputDevice::SetEventNotification> with the * <p hEvent> parameter set to NULL before closing the * event handle. * * The event notification handle cannot be changed while the * device is acquired. * * If the function is successful, then the application can * use the event handle in the same manner as any other * Win32 event handle. Examples of usage are shown below. * For additional information on using Win32 wait functions, * see the Win32 SDK and related documentation. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm IN HANDLE | hEvent | * * Specifies the event handle which will be set when the * device state changes. It "must" be an event * handle. DirectInput will <f SetEvent> the handle when * the state of the device changes. * * The application should create the handle via the * <f CreateEvent> function. If the event is created as * an automatic-reset event, then the operating system will * automatically reset the event once a wait has been * satisfied. If the event is created as a manual-reset * event, then it is the application's responsibility to * call <f ResetEvent> to reset it. DirectInput will not * call <f ResetEvent> for event notification handles. * * If the <p hEvent> is zero, then notification is disabled. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c DIERR_HANDLEEXISTS>: The <i IDirectInputDevice> object * already has an event notification handle * associated with it. DirectInput supports only one event * notification handle per <i IDirectInputDevice> object. * * <c DIERR_ACQUIRED>: The <i IDirectInputDevice> object * has been acquired. You must <mf IDirectInputDevice::Unacquire> * the device before you can change the notification state. * * <c E_INVALIDARG>: The thing isn't an event handle. * * @ex * * To check if the handle is currently set without blocking: * * | * * dwResult = WaitForSingleObject(hEvent, 0); * if (dwResult == WAIT_OBJECT_0) { * // Event is set. If the event was created as
* // automatic-reset, then it has also been reset.
* } * * @ex * * The following example illustrates blocking * indefinitely until the event is set. Note that this * behavior is <y strongly> discouraged because the thread * will not respond to the system until the wait is * satisfied. (In particular, the thread will not respond * to Windows messages.) * * | * * dwResult = WaitForSingleObject(hEvent, INFINITE); * if (dwResult == WAIT_OBJECT_0) { * // Event has been set. If the event was created as
* // automatic-reset, then it has also been reset.
* } * * @ex * * The following example illustrates a typical message loop * for a message-based application that uses two events. * * | * * HANDLE ah[2] = { hEvent1, hEvent2 }; * * while (TRUE) { * * dwResult = MsgWaitForMultipleObjects(2, ah, FALSE, * INFINITE, QS_ALLINPUT); * switch (dwResult) { * case WAIT_OBJECT_0: * // Event 1 has been set. If the event was created as
* // automatic-reset, then it has also been reset.
* ProcessInputEvent1(); * break; * * case WAIT_OBJECT_0 + 1: * // Event 2 has been set. If the event was created as
* // automatic-reset, then it has also been reset.
* ProcessInputEvent2(); * break; * * case WAIT_OBJECT_0 + 2: * // A Windows message has arrived. Process messages
* // until there aren't any more.
* while (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) { * if (msg.message == WM_QUIT) { * goto exitapp; * } * TranslateMessage(&msg); * DispatchMessage(&msg); * } * break; * * default: * // Unexpected error.
* Panic(); * break; * } * } * * @ex * * The following example illustrates a typical application loop * for a non-message-based application that uses two events. * * | * * HANDLE ah[2] = { hEvent1, hEvent2 }; * DWORD dwWait = 0; * * while (TRUE) { * * dwResult = MsgWaitForMultipleObjects(2, ah, FALSE, * dwWait, QS_ALLINPUT); * dwWait = 0; * * switch (dwResult) { * case WAIT_OBJECT_0: * // Event 1 has been set. If the event was created as
* // automatic-reset, then it has also been reset.
* ProcessInputEvent1(); * break; * * case WAIT_OBJECT_0 + 1: * // Event 2 has been set. If the event was created as
* // automatic-reset, then it has also been reset.
* ProcessInputEvent2(); * break; * * case WAIT_OBJECT_0 + 2: * // A Windows message has arrived. Process messages
* // until there aren't any more.
* while (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) { * if (msg.message == WM_QUIT) { * goto exitapp; * } * TranslateMessage(&msg); * DispatchMessage(&msg); * } * break; * * default: * // No input or messages waiting.
* // Do a frame of the game.
* // If the game is idle, then tell the next wait
* // to wait indefinitely for input or a message.
* if (!DoGame()) { * dwWait = INFINITE; * } * // Poll for data in case the device is not
* // interrupt-driven.
* IDirectInputDevice2_Poll(pdev); * break; * } * } * * *****************************************************************************/
STDMETHODIMP CDIDev_SetEventNotification(PV pdd, HANDLE h _THAT) { HRESULT hres; EnterProcR(IDirectInputDevice::SetEventNotification, (_ "px", pdd, h));
if(SUCCEEDED(hres = hresPvT(pdd))) { PDD this = _thisPv(pdd);
/*
* Must protect with the critical section to prevent somebody from * acquiring or setting a new event handle while we're changing it. */ CDIDev_EnterCrit(this);
if(!this->fAcquired) { /*
* Don't operate on the original handle because the app * might decide to do something to it on another thread. */
hres = DupEventHandle(h, &h);
if(SUCCEEDED(hres)) { /*
* Resetting the event serves two purposes. * * 1. It performs parameter validation for us, and * 2. The event must be reset while the device is * not acquired. */ if(fLimpFF(h, ResetEvent(h))) {
if(!this->hNotify || !h) { hres = this->pdcb->lpVtbl-> SetEventNotification(this->pdcb, h); /*
* All dcb's use default handling for now so * assert the callback returns S_FALSE * so we are reminded to change this if need be. * An uninitialized device would fail but don't * break if we hit one of those, just assert that * we won't accidentally call a HEL on it. */ AssertF( ( hres == S_FALSE ) || ( ( hres == DIERR_NOTINITIALIZED ) && !this->pvi ) );
if(this->pvi) { VXDDWORDDATA vhd; vhd.pvi = this->pvi; vhd.dw = (DWORD)(UINT_PTR)h; hres = Hel_SetNotifyHandle(&vhd); AssertF(SUCCEEDED(hres)); /* Should never fail */ h = (HANDLE)(UINT_PTR)pvExchangePpvPv64(&this->hNotify, (UINT_PTR)h); hres = this->hresPolled; } else { /*
* ISSUE-2001/03/29-timgill Is this actually an error case? * Can this ever validly occur? * if yes, don't we need any of the above? */ RPF( "Device internal data missing on SetEventNotification" ); }
} else { /*
* Note, this is a change in behavior as 3.0 didn't * mind if you replaced an existing handle . * Nobody seems to have noticed yet though. */ hres = DIERR_HANDLEEXISTS; } } else { RPF( "Handle not for Event in SetEventNotification" ); hres = E_HANDLE; }
/*
* Close the old handle if we swapped one out * or our duplicate if something went wrong */ if(h != 0) { CloseHandle(h); } } } else { hres = DIERR_ACQUIRED; } CDIDev_LeaveCrit(this); }
ExitOleProc(); return hres; }
#ifdef XDEBUG
CSET_STUBS(SetEventNotification, (PV pdd, HANDLE h), (pdd, h THAT_))
#else
#define CDIDev_SetEventNotificationA CDIDev_SetEventNotification
#define CDIDev_SetEventNotificationW CDIDev_SetEventNotification
#endif
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | IDirectInputDevice | hresMapHow | * * Map the <p dwObj> and <p dwHow> fields into an object index. * * @parm DWORD | dwObj | * * Object identifier. * * @parm DWORD | dwHow | * * How <p dwObj> should be interpreted. * * @parm OUT LPDIPROPINFO | ppropi | * * Receives object index and <p dwDevType>. * *****************************************************************************/
#ifndef XDEBUG
#define CDIDev_hresMapHow_(this, dwObj, dwHow, ppropi, z) \
_CDIDev_hresMapHow_(this, dwObj, dwHow, ppropi) \
#endif
STDMETHODIMP CDIDev_hresMapHow_(PDD this, DWORD dwObj, DWORD dwHow, LPDIPROPINFO ppropi, LPCSTR s_szProc) { HRESULT hres;
if(this->pdcb != c_pdcbNil) { int iobj = 0;
switch(dwHow) {
case DIPH_DEVICE: if(dwObj == 0) { ppropi->iobj = 0xFFFFFFFF; ppropi->dwDevType = 0; hres = S_OK; } else { RPF("%s: dwObj must be zero if DIPH_DEVICE", s_szProc); hres = E_INVALIDARG; } break;
case DIPH_BYOFFSET: if(this->pdix && this->rgiobj) { if(dwObj < this->dwDataSize) { iobj = this->rgiobj[dwObj]; if(iobj >= 0) { AssertF(this->pdix[iobj].dwOfs == dwObj); ppropi->iobj = iobj; ppropi->dwDevType = this->df.rgodf[iobj].dwType; hres = S_OK; goto done; } else { AssertF(iobj == -1); } }
SquirtSqflPtszV(sqfl | sqflBenign, TEXT("%S: Invalid offset in dwObj. You may use DIPH_BYID to enum it."), s_szProc);
//RPF("%s: Invalid offset in dwObj", s_szProc);
} else { RPF("%s: Must have a data format to use if DIPH_BYOFFSET", s_szProc); } hres = DIERR_OBJECTNOTFOUND; goto done;
case DIPH_BYID: for(iobj = this->df.dwNumObjs; --iobj >= 0; ) { if(DIDFT_FINDMATCH(this->df.rgodf[iobj].dwType, dwObj)) { ppropi->iobj = iobj; ppropi->dwDevType = this->df.rgodf[iobj].dwType; hres = S_OK; goto done; } } RPF("%s: Invalid ID in dwObj", s_szProc); hres = DIERR_OBJECTNOTFOUND; break;
case DIPH_BYUSAGE: hres = IDirectInputDeviceCallback_MapUsage(this->pdcb, dwObj, &ppropi->iobj); if(SUCCEEDED(hres)) { ppropi->dwDevType = this->df.rgodf[ppropi->iobj].dwType; } break;
default: RPF("%s: Invalid dwHow", s_szProc); hres = E_INVALIDARG; break; }
done:; } else { RPF("ERROR: IDirectInputDevice: Not initialized"); hres = DIERR_NOTINITIALIZED; }
return hres; }
#define CDIDev_hresMapHow(this, dwObj, dwHow, pdwOut) \
CDIDev_hresMapHow_(this, dwObj, dwHow, pdwOut, s_szProc) \
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | GetObjectInfo | * * Obtains information about an object. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm OUT LPDIDEVICEOBJECTINSTANCE | pdidoi | * * Receives information about the object. * The caller "must" initialize the <e DIDEVICEOBJECTINSTANCE.dwSize> * field before calling this method. * * @parm DWORD | dwObj | * * Identifies the object for which the property is to be * accessed. The meaning of this value depends on the * value of the <p dwHow> parameter. * See the documentation of the <t DIPROPHEADER> * structure for additional information. * * @parm DWORD | dwHow | * * Identifies how <p dwObj> is to be interpreted. * It must be one of the <c DIPH_*> values. * See the documentation of the <t DIPROPHEADER> * structure for additional information. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c DIERR_INVALIDPARAM> = <c E_INVALIDARG>: One or more * parameters was invalid. * * <c DIERR_OBJECTNOTFOUND>: The specified object does not * exist. * *****************************************************************************/
STDMETHODIMP CDIDev_GetObjectInfoW(PV pddW, LPDIDEVICEOBJECTINSTANCEW pdoiW, DWORD dwObj, DWORD dwHow) { HRESULT hres; EnterProcR(IDirectInputDevice::GetObjectInfo, (_ "ppxx", pddW, pdoiW, dwObj, dwHow));
if(SUCCEEDED(hres = hresPvW(pddW)) && SUCCEEDED(hres = hresFullValidWritePxCb2(pdoiW, DIDEVICEOBJECTINSTANCE_DX5W, DIDEVICEOBJECTINSTANCE_DX3W, 1))) { PDD this = _thisPvNm(pddW, ddW); DIPROPINFO propi;
/*
* Must protect with the critical section to prevent * another thread from Reset()ing behind our back. */ CDIDev_EnterCrit(this);
propi.pguid = 0; hres = CDIDev_hresMapHow(this, dwObj, dwHow, &propi);
if(SUCCEEDED(hres)) { if(dwHow != DIPH_DEVICE) { hres = CDIDev_GetObjectInfoHelper(this, &propi, pdoiW); } else { RPF("%s: Invalid dwHow", s_szProc); hres = E_INVALIDARG; } }
if(FAILED(hres)) { ScrambleBuf(&pdoiW->guidType, pdoiW->dwSize - FIELD_OFFSET(DIDEVICEOBJECTINSTANCEW, guidType)); }
CDIDev_LeaveCrit(this); }
ExitBenignOleProcR(); return hres; }
#define CDIDev_GetObjectInfo2W CDIDev_GetObjectInfoW
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | IDirectInputDeviceA | GetObjectInfo | * * ANSI version of same. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm OUT LPDIDEVICEOBJECTINSTANCEA | pdoiA | * * Receives information about the device's identity. * * @parm DWORD | dwObj | * * Identifies the object for which the property is to be * accessed. * * @parm DWORD | dwHow | * * Identifies how <p dwObj> is to be interpreted. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c DIERR_INVALIDPARAM> = <c E_INVALIDARG>: One or more * parameters was invalid. * *****************************************************************************/
STDMETHODIMP CDIDev_GetObjectInfoA(PV pddA, LPDIDEVICEOBJECTINSTANCEA pdoiA, DWORD dwObj, DWORD dwHow) { HRESULT hres; EnterProcR(IDirectInputDevice::GetObjectInfo, (_ "ppxx", pddA, pdoiA, dwObj, dwHow));
if(SUCCEEDED(hres = hresPvA(pddA)) && SUCCEEDED(hres = hresFullValidWritePxCb2(pdoiA, DIDEVICEOBJECTINSTANCE_DX5A, DIDEVICEOBJECTINSTANCE_DX3A, 1))) { PDD this = _thisPvNm(pddA, ddA); DIDEVICEOBJECTINSTANCEW doiW;
doiW.dwSize = cbX(DIDEVICEOBJECTINSTANCEW);
hres = CDIDev_GetObjectInfoW(&this->ddW, &doiW, dwObj, dwHow);
if(SUCCEEDED(hres)) { ObjectInfoWToA(pdoiA, &doiW); hres = S_OK; } }
ExitBenignOleProcR(); return hres; }
#define CDIDev_GetObjectInfo2A CDIDev_GetObjectInfoA
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | GetDeviceInfo | * * Obtains information about the device's identity. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm OUT LPDIDEVICEINSTANCE | pdidi | * * Receives information about the device's identity. * The caller "must" initialize the <e DIDEVICEINSTANCE.dwSize> * field before calling this method. * * If <e DIDEVICEINSTANCE.dwSize> is equal to the size of * the <t DIDEVICEINSTANCE_DX3> structure, then a * DirectX 3.0-compatible structure is returned instead of * a DirectX 5.0 structure. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c DIERR_INVALIDPARAM> = <c E_INVALIDARG>: One or more * parameters was invalid. * *****************************************************************************/
STDMETHODIMP CDIDev_GetDeviceInfoW(PV pddW, LPDIDEVICEINSTANCEW pdidiW) { HRESULT hres; EnterProcR(IDirectInputDevice::GetDeviceInfo, (_ "pp", pddW, pdidiW));
if(SUCCEEDED(hres = hresPvW(pddW)) && SUCCEEDED(hres = hresFullValidWritePxCb2(pdidiW, DIDEVICEINSTANCE_DX5W, DIDEVICEINSTANCE_DX3W, 1))) { PDD this = _thisPvNm(pddW, ddW);
/*
* Must protect with the critical section to prevent * another thread from Reset()ing behind our back. */ CDIDev_EnterCrit(this);
pdidiW->guidInstance = this->guid; pdidiW->guidProduct = this->guid;
/*
* Don't overwrite the dwSize, guidInstance, or guidProduct. * Start at the dwDevType. */
ZeroBuf(&pdidiW->dwDevType, pdidiW->dwSize - FIELD_OFFSET(DIDEVICEINSTANCEW, dwDevType));
hres = this->pdcb->lpVtbl->GetDeviceInfo(this->pdcb, pdidiW);
if(FAILED(hres)) { ScrambleBuf(&pdidiW->guidInstance, cbX(DIDEVICEINSTANCEW) - FIELD_OFFSET(DIDEVICEINSTANCEW, guidInstance)); }
CDIDev_LeaveCrit(this); }
ExitOleProcR(); return hres; }
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | IDirectInputDeviceA | GetDeviceInfo | * * ANSI version of same. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm OUT LPDIDEVICEINSTANCEA | pdidiA | * * Receives information about the device's identity. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c DIERR_INVALIDPARAM> = <c E_INVALIDARG>: One or more * parameters was invalid. * *****************************************************************************/
STDMETHODIMP CDIDev_GetDeviceInfoA(PV pddA, LPDIDEVICEINSTANCEA pdidiA) { HRESULT hres; EnterProcR(IDirectInputDevice::GetDeviceInfo, (_ "pp", pddA, pdidiA));
if(SUCCEEDED(hres = hresPvA(pddA)) && SUCCEEDED(hres = hresFullValidWritePxCb2(pdidiA, DIDEVICEINSTANCE_DX5A, DIDEVICEINSTANCE_DX3A, 1))) { PDD this = _thisPvNm(pddA, ddA); DIDEVICEINSTANCEW diW;
diW.dwSize = cbX(DIDEVICEINSTANCEW);
hres = CDIDev_GetDeviceInfoW(&this->ddW, &diW);
if(SUCCEEDED(hres)) { DeviceInfoWToA(pdidiA, &diW); hres = S_OK; } }
ExitOleProcR(); return hres; }
/*****************************************************************************
* * @doc INTERNAL * * @method void | CDIDev | UnhookCwp | * * Remove the CallWndProc handler. * * See <mf CDIDev::InstallCwp> for details. * *****************************************************************************/
HWND g_hwndExclusive; HHOOK g_hhkCwp;
void INTERNAL CDIDev_UnhookCwp(void) { DllEnterCrit();
if(g_hhkCwp) { UnhookWindowsHookEx(g_hhkCwp); g_hhkCwp = 0; g_hwndExclusive = 0; }
DllLeaveCrit(); }
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev | AddForegroundDevice | * * Add ourselves to the list of devices that need to be * unacquired when the window loses foreground activation. * * @parm PDD | this | * * Device to be added. * * @devnote * * Note that we do not need to AddRef the device, because * <f CDIDev_Finalize> will unacquire the device for us * automatically, so we will never be freed while still * acquired. * * (Note that if we did choose to AddRef, it must be done * outside the DLL critical * section, in order to preserve the semaphore hierarchy.) * *****************************************************************************/
STDMETHODIMP CDIDev_AddForegroundDevice(PDD this) { HRESULT hres; DllEnterCrit();
#ifdef IDirectInputDevice2Vtbl
hres = GPA_Append(g_hgpaExcl, this); Common_Hold(this); #else
if(g_cpddForeground >= g_cpddForegroundMax) { AssertF(g_cpddForeground == g_cpddForegroundMax); hres = ReallocCbPpv(cbCxX(g_cpddForeground + 10, PDD), &g_rgpddForeground); if(FAILED(hres)) { goto failed; } g_cpddForegroundMax = g_cpddForeground + 10; }
AssertF(g_cpddForeground < g_cpddForegroundMax);
g_rgpddForeground[g_cpddForeground++] = this; Common_Hold(this);
hres = S_OK; failed:;
#endif
DllLeaveCrit();
return hres; }
/*****************************************************************************
* * @doc INTERNAL * * @method void | CDIDev | DelForegroundDevice | * * Remove ourselves from the list, if we're there. * It is not an error if we aren't on the list, because * in the case of forced unacquire, the list is blanked * out in order to avoid race conditions where somebody * tries to acquire a device immediately upon receiving * foreground activation, before we get a chance to * unacquire all the old guys completely. * * @parm PDD | this | * * Device to be removed. * * @devnote * * Note that the Unhold must be done outside the DLL critical * section, in order to preserve the semaphore hierarchy. * * Theoretically, the unhold will never drop the reference count * to zero (because the latest it could be called is during * the AppFinalize, where there is stll the outstanding refcount * from the external ref that hasn't been released yet). * * But it's better to play it safe and always release the * object outside. * *****************************************************************************/
void INTERNAL CDIDev_DelForegroundDevice(PDD this) { #ifdef IDirectInputDevice2Vtbl
HRESULT hres; #else
UINT idd; #endif
DllEnterCrit();
#ifdef IDirectInputDevice2Vtbl
hres = GPA_DeletePtr(g_hgpaExcl, this); if(hres == hresUs(0)) { /* If the last one went away */ GPA_Term(g_hgpaExcl); /* Free the tracking memory */ CDIDev_UnhookCwp(); /* Then unhook ourselves */ } #else
for(idd = 0; idd < g_cpddForeground; idd++) { if(g_rgpddForeground[idd] == this) { g_rgpddForeground[idd] = g_rgpddForeground[--g_cpddForeground]; if(g_cpddForeground == 0) { CDIDev_UnhookCwp(); } break; } } #endif
DllLeaveCrit();
#ifdef IDirectInputDevice2Vtbl
if(SUCCEEDED(hres)) { Common_Unhold(this); } #endif
}
/*****************************************************************************
* * @doc INTERNAL * * @func LRESULT | CDIDev_CallWndProc | * * Thread-specific CallWndProc handler. * * Note that we need only one of these, since only the foreground * window will require a hook. * * @parm int | nCode | * * Notification code. * * @parm WPARAM | wp | * * "Specifies whether the message is sent by the current process." * We don't care. * * @parm LPARAM | lp | * * Points to a <t CWPSTRUCT> which describes the message. * * @returns * * Always chains to the next hook. * *****************************************************************************/
LRESULT CALLBACK CDIDev_CallWndProc(int nCode, WPARAM wp, LPARAM lp) { LPCWPSTRUCT pcwp = (LPCWPSTRUCT)lp; #ifdef WINNT
static BOOL fKillFocus = FALSE; static BOOL fIconic = FALSE;
fIconic = FALSE;
/*
* This part of code is to fix Windows bug 430051. * The logic is: if WM_KILLFOCUS is followed by WM_SIZE(minimized), * then the app is minimized from full screen mode. * This combination should only happen to full screen mode game using DDraw. */ if(pcwp->message == WM_KILLFOCUS) { fKillFocus = TRUE; } else if(pcwp->message == WM_SETFOCUS) { fKillFocus = FALSE; } else if (pcwp->message == WM_SIZE) { if(pcwp->wParam == SIZE_MINIMIZED){ if( fKillFocus ) { fIconic = TRUE; fKillFocus = FALSE; }else{ fKillFocus = FALSE; } } else { fKillFocus = FALSE; } } #endif
if( nCode == HC_ACTION && (pcwp->message == WM_ACTIVATE #ifdef WINNT
|| fIconic #endif
) ) { PDD *rgpdid; UINT ipdid, cpdid; HHOOK hhk;
#ifdef WINNT
fIconic = FALSE; #endif
/*
* We cannot mess with items while inside the DLL critical section, * because that would violate our semaphore hierarchy. * * Instead, we stash the active item list and replace it with * an empty list. Then, outside the DLL critical section, we * calmly operate on each item. */ DllEnterCrit(); #ifdef IDirectInputDevice2Vtbl
rgpdid = (PV)g_hgpaExcl->rgpv; cpdid = g_hgpaExcl->cpv; GPA_Init(g_hgpaExcl); #else
rgpdid = g_rgpddForeground; cpdid = g_cpddForeground; g_rgpddForeground = 0; g_cpddForeground = 0; g_cpddForegroundMax = 0; #endif
/*
* Some sanity checking here. */
for(ipdid = 0; ipdid < cpdid; ipdid++) { AssertF(rgpdid[ipdid]); }
DllLeaveCrit();
/*
* Note that InternalUnacquire will set the notification * event so the app knows that input was lost. */ for(ipdid = 0; ipdid < cpdid; ipdid++) { AssertF(rgpdid[ipdid]); SquirtSqflPtszV(sqfl, TEXT("Forcing unacquire of %08x due to focus loss"), rgpdid[ipdid]); CDIDev_InternalUnacquire(rgpdid[ipdid]); Common_Unhold(rgpdid[ipdid]); } FreePpv(&rgpdid);
hhk = g_hhkCwp; CDIDev_UnhookCwp(); return CallNextHookEx(hhk, nCode, wp, lp); } return CallNextHookEx(g_hhkCwp, nCode, wp, lp); }
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev | CanAcquire | * * Determine whether the device may be acquired exclusively. * * If exclusive access is not requested, then the function * succeeds vacuously. * * If exclusive access is requested, then the window must * be the foreground window and must belong to the current * process. * *****************************************************************************/
STDMETHODIMP CDIDev_CanAcquire(PDD this) { HRESULT hres;
AssertF(CDIDev_IsConsistent(this)); if(this->discl & DISCL_FOREGROUND) { HWND hwndForeground = GetForegroundWindow();
AssertF(this->hwnd); /*
* Note that we don't have to do an IsWindow() on this->hwnd, * because GetForegroundWindow() will always return a valid * window or NULL. Since we already tested this->hwnd != 0 * above, the only way the equality can occur is if the window * handle is indeed valid. */ if(this->hwnd == hwndForeground && !IsIconic(this->hwnd)) {
/*
* Need to make sure that the window "still" belongs to * this process, in case the window handle got recycled. */ DWORD idProcess; GetWindowThreadProcessId(this->hwnd, &idProcess); if(idProcess == GetCurrentProcessId()) { hres = S_OK; } else { /*
* Put a permanently invalid handle here so that we * won't accidentally take a new window that happens * to get a recycled handle value. */ this->hwnd = INVALID_HANDLE_VALUE; RPF("Error: Window destroyed while associated with a device"); hres = E_INVALIDARG; } } else { hres = DIERR_OTHERAPPHASPRIO; } } else { /* No window; vacuous success */ hres = S_OK; } return hres; }
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev | InstallCwp | * * Install the CallWndProc handler. * * There is a bit of subtlety in the way this works. * Since only foreground windows may acquire exclusive access, * we need only one hook (for there is but one foreground * window in the system). * * _NT_: Does NT handle this correctly in the face of * multiple window stations? * * The tricky part is that input loss occurs asynchronously. * So a device that registers a <f CallWindowProc> hook doesn't * find out that the input has been lost until the app next * calls <f Unacquire>. * * So the way this is done is via a collection of global * variables (which must be accessed atomically). * * <p g_hhkCwp> is the hook handle itself. It is zero when * no hook is installed. * * Note that we install the windows hook while inside both the * object critical section and the DLL critical section. * You might think we'd risk deadlocking with the hook procedure, * in case the window asynchronously deactivates while we're * installing the hook. But your worries are unfounded: * If the window is on the current thread, then window messages * won't be dispatched because we never call <f GetMessage> or * go into a modal loop. And if the window is on another thread, * then that other thread will simply have to wait until we're done. * *****************************************************************************/
STDMETHODIMP CDIDev_InstallCwp(PDD this) { HRESULT hres;
if(this->discl & DISCL_FOREGROUND) { AssertF(this->hwnd); hres = CDIDev_CanAcquire(this); if(SUCCEEDED(hres)) { hres = CDIDev_AddForegroundDevice(this); if(SUCCEEDED(hres)) { DllEnterCrit(); if(!g_hhkCwp) { /* We're the first one */ g_hwndExclusive = this->hwnd; g_hhkCwp = SetWindowsHookEx(WH_CALLWNDPROC, CDIDev_CallWndProc, g_hinst, GetWindowThreadProcessId(this->hwnd, 0)); } else { AssertF(g_hwndExclusive == this->hwnd); }
DllLeaveCrit(); /*
* There is a race condition up above, where the foreground * window can change between the call to CanAcquire() and * the call to SetWindowsHookEx(). Close the window by * checking a second time after the hook is installed. * * If we leave the window open, it's possible that we will * perform a physical acquire while the wrong window has * foreground activation. Then, of course, we are never told * that *our* window lost activation, and the physical device * remains acquired forever. */ hres = CDIDev_CanAcquire(this); if(SUCCEEDED(hres)) { } else { SquirtSqflPtszV(sqflError, TEXT("Window no longer foreground; ") TEXT("punting acquire")); CDIDev_InternalUnacquire(this); } } } else { hres = DIERR_OTHERAPPHASPRIO; } } else { /* No window; vacuous success */ hres = S_OK; }
return hres; }
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | IDirectInputDevice | RealUnacquire | * * Release access to the device, even if the device was only * partially acquired. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @returns * * None. * *****************************************************************************/
STDMETHODIMP CDIDev_RealUnacquire(PDD this) { HRESULT hres;
hres = this->pdcb->lpVtbl->Unacquire(this->pdcb); if(hres == S_FALSE) { if(this->fAcquiredInstance) { this->fAcquiredInstance = 0; hres = Hel_UnacquireInstance(this->pvi); AssertF(SUCCEEDED(hres)); } else { hres = S_OK; } }
return hres; }
#ifdef IDirectInputDevice2Vtbl
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev | FFAcquire | * * The device has been successfully acquired. Do any * necessary force feedback related acquisition goo. * * @cwrap PDD | pdd * *****************************************************************************/
STDMETHODIMP CDIDev_FFAcquire(PDD this) { HRESULT hres;
AssertF(CDIDev_InCrit(this));
if(this->pes && (this->discl & DISCL_EXCLUSIVE)) { if(SUCCEEDED(hres = this->pes->lpVtbl->SendForceFeedbackCommand( this->pes, &this->sh, DISFFC_FORCERESET))) {
CDIDev_RefreshGain(this);
/*
* If the center spring is to be disabled, * then disable the center spring. */ if(!this->dwAutoCenter) { this->pes->lpVtbl->SendForceFeedbackCommand( this->pes, &this->sh, DISFFC_STOPALL); }
hres = S_OK; }
} else { hres = S_OK; }
return hres; } #endif
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | Acquire | * * Obtains access to the device. * * Device acquisition does not reference-count. If a device is * acquired twice then unacquired once, the device is unacquired. * * Before the device can be acquired, a data format must * first be set via the <mf IDirectInputDevice::SetDataFormat> * method. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c S_FALSE>: The device has already been acquired. Note * that this value is a success code. * * <c DIERR_OTHERAPPHASPRIO>: Access to the device was not granted. * The most common cause of this is attempting to acquire a * device with the <c DISCL_FOREGROUND> cooperative level when * the associated window is not foreground. * * This error code is also returned if an attempt to * acquire a device in exclusive mode fails because the device * is already acquired in exclusive mode by somebody else. * * <c DIERR_INVALIDPARAM> = <c E_INVALIDARG>: The device * does not have a selected data format. */ /*
* The point at which we take the exclusive semaphore is important. * We should do it after preliminary validation of foreground * permission, so that we don't accidentally lock out somebody * else who legitimately has permission. * *****************************************************************************/
STDMETHODIMP CDIDev_Acquire(PV pdd _THAT) { HRESULT hres; EnterProcR(IDirectInputDevice::Acquire, (_ "p", pdd));
if(SUCCEEDED(hres = hresPvT(pdd))) { PDD this = _thisPv(pdd);
/*
* Must protect with the critical section to prevent somebody from * acquiring or changing the data format while we're acquiring. */ CDIDev_EnterCrit(this);
/*
* The app explicitly messed with acquisition. Any problems * retrieving data are now the apps' fault. */ this->hresNotAcquired = DIERR_NOTACQUIRED;
//We now need a pvi even in where the device doesn't use VxDs
if(this->pdix && this->pvi) { if(this->pvi->fl & VIFL_ACQUIRED) { hres = S_FALSE; } else if(SUCCEEDED(hres = CDIDev_CanAcquire(this))) {
hres = Excl_Acquire(&this->guid, this->hwnd, this->discl); if(SUCCEEDED(hres)) {
#ifdef IDirectInputDevice2Vtbl
if(SUCCEEDED(hres = CDIDev_FFAcquire(this))) { #endif
hres = this->pdcb->lpVtbl->Acquire(this->pdcb); if(SUCCEEDED(hres)) { if(hres == S_FALSE) { hres = Hel_AcquireInstance(this->pvi); if(SUCCEEDED(hres)) { this->fAcquiredInstance = 1;
/*
* If relative mode, need to prime the * pvLastBuffer with the current state. */
if(this->pvi->fl & VIFL_RELATIVE) { hres = this->pdcb->lpVtbl->GetDeviceState( this->pdcb, this->pvLastBuffer); if(FAILED(hres)) { goto unacquire; } }
} else { goto unacquire; } }
/*
* Note that InstallCwp must be the last thing * we do, because it will add us to the foreground * list, and none of our error exit paths remove us. */ hres = CDIDev_InstallCwp(this); if(SUCCEEDED(hres)) { this->fAcquired = 1; this->fOnceAcquired = 1;
/*
* From now on, if we lose acquisition, * it's not the app's fault. */ this->hresNotAcquired = DIERR_INPUTLOST;
hres = S_OK; } else { goto unacquire; }
} else { unacquire:; CDIDev_RealUnacquire(this); } #ifdef IDirectInputDevice2Vtbl
} #endif
} } } else { hres = E_INVALIDARG; }
// For some apps (e.g. "Sonic R") we need to always succeed Acquire() -- see Windows bug 15085
if ((this->diHacks.fSucceedAcquire ) && (FAILED(hres))) { this->fOnceAcquired = 1; hres = S_OK; }
CDIDev_LeaveCrit(this); }
ExitOleProcR(); return hres; }
#ifdef XDEBUG
CSET_STUBS(Acquire, (PV pdd), (pdd THAT_))
#else
#define CDIDev_AcquireA CDIDev_Acquire
#define CDIDev_AcquireW CDIDev_Acquire
#endif
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | IDirectInputDevice | InternalUnacquire | * * This does the real work of unacquiring. The internal * version bypasses the "the app requested this" flag, * so when the app goes to request something, it gets * <c DIERR_INPUTLOST> instead of <c DIERR_NOTACQUIRED>. * * If the application error code is <c DIERR_INPUTLOST>, then * we will also signal the associated event so that it knows * that the state changed. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * *****************************************************************************/
STDMETHODIMP CDIDev_InternalUnacquire(PDD this) { HRESULT hres; EnterProcR(IDirectInputDevice::InternalUnacquire, (_ "p", this));
/*
* Must protect with the critical section to prevent confusing other * methods which change their behavior depending on whether the device * is acquired. */ CDIDev_EnterCrit(this);
if(this->fAcquired) { AssertF(this->pdcb != c_pdcbNil); this->fAcquired = 0; Excl_Unacquire(&this->guid, this->hwnd, this->discl); if(this->discl & DISCL_FOREGROUND) { AssertF(this->hwnd); CDIDev_DelForegroundDevice(this); #ifdef WINNT
if( IsIconic(this->hwnd) ) { this->fUnacquiredWhenIconic = 1; } #endif
} #ifdef IDirectInputDevice2Vtbl
/*
* ISSUE-2001/03/29-timgill multithreading means we cannot rely on Excl_Unaquire() return values * We cannot trust the return value (if we made one) * of Excl_Unacquire, because another instance may have * snuck in and acquired the device after we Excl_Unacquire'd * it and started doing force feedback on it. * * We need to fix this with the joystick mutex. */ if(this->pes && (this->discl & DISCL_EXCLUSIVE)) { this->pes->lpVtbl->SendForceFeedbackCommand( this->pes, &this->sh, DISFFC_RESET); this->sh.dwTag = 0; } #endif
hres = CDIDev_RealUnacquire(this); if(this->hresNotAcquired == DIERR_INPUTLOST) { CDIDev_SetNotifyEvent(this); } } else { hres = S_FALSE; }
CDIDev_LeaveCrit(this);
ExitOleProcR(); return hres; }
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | Unacquire | * * Release access to the device. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c S_FALSE>: The object is not currently acquired. * This may have been caused by a prior loss of input. * Note that this is a success code. * *****************************************************************************/
STDMETHODIMP CDIDev_Unacquire(PV pdd _THAT) { HRESULT hres; EnterProcR(IDirectInputDevice::Unacquire, (_ "p", pdd));
if(SUCCEEDED(hres = hresPvT(pdd))) { PDD this = _thisPv(pdd);
/*
* The app explicitly messed with acquisition. Any problems * retrieving data are now the apps' fault. */ this->hresNotAcquired = DIERR_NOTACQUIRED;
hres = CDIDev_InternalUnacquire(this);
}
ExitOleProcR(); return hres; }
#ifdef XDEBUG
CSET_STUBS(Unacquire, (PV pdd), (pdd THAT_))
#else
#define CDIDev_UnacquireA CDIDev_Unacquire
#define CDIDev_UnacquireW CDIDev_Unacquire
#endif
/*****************************************************************************
* * @doc INTERNAL * * @method PDIPROPVALIDINFO | IDirectInputDevice | ppviFind | * * Locate the DIPROPVALIDINFO structure that describes * the predefined property. * * @parm const GUID * | pguid | * * Property guid, or predefined property. * * @returns * * Pointer to a const <t DIPROPVALIDINFO> that describes * what is and is not valid for this property. * * Returns 0 if the property is not one of the predefined * properties. * *****************************************************************************/
#pragma BEGIN_CONST_DATA
typedef struct DIPROPVALIDINFO { PCGUID pguid; /* Property name */ DWORD dwSize; /* expected size */ DWORD fl; /* flags */ } DIPROPVALIDINFO, *PDIPROPVALIDINFO;
/*
* Note that the flags are negative in sense. * This makes validation easier. */ #define DIPVIFL_NOTDEVICE 0x00000001 /* Cannot be device */
#define DIPVIFL_NOTOBJECT 0x00000002 /* Cannot be object */
#define DIPVIFL_READONLY 0x00000004 /* Cannot be set */
#define DIPVIFL_NOTPRIVATE 0x00000008 /* Cannot handle private pvi */
#define DIPVIFL_NOTACQUIRED 0x00000010 /* Cannot modify while acquired */
DIPROPVALIDINFO c_rgpvi[] = {
{ DIPROP_BUFFERSIZE, cbX(DIPROPDWORD), DIPVIFL_NOTOBJECT | DIPVIFL_NOTPRIVATE | DIPVIFL_NOTACQUIRED, },
{ DIPROP_AXISMODE, cbX(DIPROPDWORD), DIPVIFL_NOTOBJECT | DIPVIFL_NOTPRIVATE | DIPVIFL_NOTACQUIRED, },
{ DIPROP_GRANULARITY, cbX(DIPROPDWORD), DIPVIFL_NOTDEVICE | DIPVIFL_READONLY | DIPVIFL_NOTACQUIRED, },
{ DIPROP_RANGE, cbX(DIPROPRANGE), DIPVIFL_NOTDEVICE | DIPVIFL_NOTACQUIRED, },
/*
* Note that you can set the dead zone on the entire device. * This is the same as applying it to each axis individually. */ { DIPROP_DEADZONE, cbX(DIPROPDWORD), DIPVIFL_NOTACQUIRED, },
/*
* Note that you can set the saturation on the entire device. * This is the same as applying it to each axis individually. */ { DIPROP_SATURATION, cbX(DIPROPDWORD), DIPVIFL_NOTACQUIRED, },
/*
* Note that you can change the gain either while acquired * or not. Your choice. */ { DIPROP_FFGAIN, cbX(DIPROPDWORD), DIPVIFL_NOTOBJECT, },
/*
* Note that the FF load is meaningful only when acquired, * so we'd better not complain if they access it while acquired! */ { DIPROP_FFLOAD, cbX(DIPROPDWORD), DIPVIFL_NOTOBJECT | DIPVIFL_READONLY, },
{ DIPROP_AUTOCENTER, cbX(DIPROPDWORD), DIPVIFL_NOTOBJECT | DIPVIFL_NOTACQUIRED, },
{ DIPROP_CALIBRATIONMODE, cbX(DIPROPDWORD), DIPVIFL_NOTOBJECT | DIPVIFL_NOTACQUIRED, },
{ DIPROP_CALIBRATION, cbX(DIPROPCAL), DIPVIFL_NOTDEVICE | DIPVIFL_NOTACQUIRED, },
{ DIPROP_GUIDANDPATH, cbX(DIPROPGUIDANDPATH), DIPVIFL_NOTOBJECT | DIPVIFL_READONLY, },
{ DIPROP_INSTANCENAME, cbX(DIPROPSTRING), DIPVIFL_NOTOBJECT, },
{ DIPROP_PRODUCTNAME, cbX(DIPROPSTRING), DIPVIFL_NOTOBJECT, },
{ DIPROP_MAXBUFFERSIZE, cbX(DIPROPDWORD), DIPVIFL_NOTOBJECT | DIPVIFL_NOTPRIVATE | DIPVIFL_NOTACQUIRED, },
{ DIPROP_JOYSTICKID, cbX(DIPROPDWORD), DIPVIFL_NOTOBJECT | DIPVIFL_NOTACQUIRED, },
{ DIPROP_GETPORTDISPLAYNAME, cbX(DIPROPSTRING), DIPVIFL_NOTOBJECT | DIPVIFL_READONLY, },
/*
* Note that you can change the report ID while acquired * or not. Your choice. */ { DIPROP_ENABLEREPORTID, cbX(DIPROPDWORD), 0x0, },
#if 0
{ DIPROP_SPECIFICCALIBRATION, cbX(DIPROPCAL), DIPVIFL_NOTDEVICE | DIPVIFL_NOTACQUIRED, }, #endif
};
#pragma END_CONST_DATA
STDMETHODIMP_(PDIPROPVALIDINFO) CDIDev_ppviFind(PCGUID pguid) { PDIPROPVALIDINFO ppvi; UINT ipvi;
for(ipvi = 0, ppvi = c_rgpvi; ipvi < cA(c_rgpvi); ipvi++, ppvi++) { if(ppvi->pguid == pguid) { goto found; } } ppvi = 0;
found: return ppvi;
}
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | IDirectInputDevice | hresValidProp | * * Check that the property structure makes sense. * Returns the object index for further processing. * * @parm const GUID * | pguid | * * Property guid, or predefined property. * * @parm LPCDIPROPHEADER | pdiph | * * Propery header structure. * * @parm BOOL | fWrite | * * Whether property should be validate for writing. * * @parm OUT LPDIPROPINFO | ppropi | * * Receives object index. * *****************************************************************************/
typedef BOOL (WINAPI *PFNBAD)(PCV pv, UINT cb);
STDMETHODIMP CDIDev_hresValidProp(PDD this, const GUID *pguid, LPCDIPROPHEADER pdiph, BOOL fWrite, LPDIPROPINFO ppropi) { HRESULT hres; PFNBAD pfnBad; EnterProcR(IDirectInputDevice::Get/SetProperty, (_ "pxpx", this, pguid, pdiph, fWrite));
AssertF(CDIDev_InCrit(this));
if(fWrite) { pfnBad = (PFNBAD)IsBadWritePtr; } else { pfnBad = (PFNBAD)IsBadReadPtr; }
if(!pfnBad(pdiph, cbX(DIPROPHEADER)) && pdiph->dwHeaderSize == cbX(DIPROPHEADER) && pdiph->dwSize % 4 == 0 && pdiph->dwSize >= pdiph->dwHeaderSize && !pfnBad(pdiph, pdiph->dwSize)) {
if(fWrite) { ScrambleBuf((PV)(pdiph+1), pdiph->dwSize - cbX(DIPROPHEADER)); }
/*
* Now convert the item descriptor into an index. */ hres = CDIDev_hresMapHow(this, pdiph->dwObj, pdiph->dwHow, ppropi);
if(SUCCEEDED(hres)) {
/*
* Now validate the property id or guid. */ if(HIWORD((UINT_PTR)pguid) == 0) {
PDIPROPVALIDINFO ppvi;
ppvi = CDIDev_ppviFind(pguid);
/*
* Note that if we don't find the GUID in our list, * we fail it straight away. This prevents ISVs * from trying to create properties in the Microsoft * Reserved range. */ if(ppvi) { if( ppvi->pguid == DIPROP_CALIBRATION ) { if( pdiph->dwSize == ppvi->dwSize || pdiph->dwSize == cbX(DIPROPCALPOV) ) { hres = S_OK; } else { RPF("%s: Arg 2: Invalid dwSize for property", s_szProc); hres = E_INVALIDARG; } } else if( pdiph->dwSize == ppvi->dwSize ) { hres = S_OK; } else { RPF("%s: Arg 2: Invalid dwSize for property", s_szProc); hres = E_INVALIDARG; } } else { RPF("%s: Arg 1: Unknown property", s_szProc); hres = E_NOTIMPL; }
} else { hres = hresFullValidGuid(pguid, 1); } } } else { RPF("%s: Arg 2: Invalid pointer", s_szProc); hres = E_INVALIDARG; }
return hres; }
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev | hresValidDefProp | * * Determine whether the property is something we can handle * in the default property handler. * * @parm IN LPCDIPROPINFO | ppropi | * * Information describing the property being retrieved. * * @parm DWORD | dwFlags | * * Flags for forbidden things. * <c DIPVIFL_READONLY> if being validated for writing. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: Passes validation. * * <c E_NOTIMPL>: Not something we handle. * * *****************************************************************************/
HRESULT INTERNAL CDIDev_hresValidDefProp(PDD this, LPCDIPROPINFO ppropi, DWORD dwFlags) { HRESULT hres; PDIPROPVALIDINFO ppvi; EnterProc(CDIDev_hresValidDefProp, (_ "pGxx", this, ppropi->pguid, ppropi->dwDevType, dwFlags));
/*
* Note that it's okay if the device is acquired. We want to * allow GetProperty to succeed on an acquired device. */ AssertF(CDIDev_InCrit(this));
ppvi = CDIDev_ppviFind(ppropi->pguid);
if(ppvi) { if(ppropi->iobj == 0xFFFFFFFF) { dwFlags |= DIPVIFL_NOTDEVICE; /* Fail if devices forbidden */ } else { dwFlags |= DIPVIFL_NOTOBJECT; /* Fail if objects forbidden */ } if(this->pvi == 0) { dwFlags |= DIPVIFL_NOTPRIVATE; /* Fail if privates forbidden */ } /*
* If attempting to modify property and we are acquired, * then also set the "but not while acquired" filter. */ if((dwFlags & DIPVIFL_READONLY) && this->fAcquired) { dwFlags |= DIPVIFL_NOTACQUIRED; /* Fail if r/o while acq'd */ }
if((ppvi->fl & dwFlags) == 0) { hres = S_OK; /* Seems reasonable */ } else { if(ppvi->fl & dwFlags & DIPVIFL_READONLY) { RPF("SetProperty: Property is read-only"); hres = DIERR_READONLY; } else if(ppvi->fl & dwFlags & DIPVIFL_NOTACQUIRED) { RPF("SetProperty: Cannot change property while acquired"); hres = DIERR_ACQUIRED; } else { RPF("Get/SetProperty: Property does not exist for that object"); hres = E_NOTIMPL; /* Cannot do that */ } }
} else { RPF("Get/SetProperty: Property does not exist"); hres = E_NOTIMPL; /* Definitely way out */ }
ExitOleProc(); return hres; }
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev | DefGetProperty | * * Default implementation of <mf IDirectInputDevice::GetProperty> * to handle properties which the device decides not to implement. * * @parm IN LPCDIPROPINFO | ppropi | * * Information describing the property being retrieved. * * @parm OUT LPDIPROPHEADER | pdiph | * * Where to put the property value. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c DIERR_INVALIDPARAM> = <c E_INVALIDARG>: The * <p pdiph> parameter is not a valid pointer. * * *****************************************************************************/
STDMETHODIMP CDIDev_DefGetProperty(PDD this, LPCDIPROPINFO ppropi, LPDIPROPHEADER pdiph) { HRESULT hres; EnterProc(CDIDev_DefGetProperty, (_ "pGx", this, ppropi->pguid, ppropi->dwDevType));
AssertF(CDIDev_InCrit(this));
hres = CDIDev_hresValidDefProp(this, ppropi, 0); if(SUCCEEDED(hres)) { LPDIPROPDWORD pdipdw = (PV)pdiph; LPDIPROPRANGE pdiprg = (PV)pdiph;
switch((DWORD)(UINT_PTR)ppropi->pguid) {
case (DWORD)(UINT_PTR)DIPROP_BUFFERSIZE: AssertF(this->pvi); /* Validation should've caught this */ pdipdw->dwData = this->celtBuf; hres = S_OK; break;
case (DWORD)(UINT_PTR)DIPROP_AXISMODE: AssertF(this->pvi); /* Validation should've caught this */ if(this->pvi->fl & VIFL_RELATIVE) { pdipdw->dwData = DIPROPAXISMODE_REL; } else { pdipdw->dwData = DIPROPAXISMODE_ABS; } hres = S_OK; break;
case (DWORD)(UINT_PTR)DIPROP_GRANULARITY:
if(DIDFT_GETTYPE(ppropi->dwDevType) & DIDFT_AXIS) { /* Default axis granularity is 1 */ pdipdw->dwData = 1; hres = S_OK; } else { /*
* Buttons don't have granularity. * POVs must be handled by device driver. */ RPF("GetProperty: Object doesn't have a granularity"); hres = E_NOTIMPL; } break;
case (DWORD)(UINT_PTR)DIPROP_RANGE: if(DIDFT_GETTYPE(ppropi->dwDevType) & DIDFT_RELAXIS) { /* Default rel-axis range is infinite */ pdiprg->lMin = DIPROPRANGE_NOMIN; pdiprg->lMax = DIPROPRANGE_NOMAX; hres = S_OK; } else { /*
* Device driver must handle abs axis range. * Buttons and POVs don't have range. */ RPF("GetProperty: Object doesn't have a range"); hres = E_NOTIMPL; } break;
case (DWORD)(UINT_PTR)DIPROP_MAXBUFFERSIZE: pdipdw->dwData = this->celtBufMax; hres = S_OK; break;
#ifdef IDirectInputDevice2Vtbl
case (DWORD)(UINT_PTR)DIPROP_FFGAIN: pdipdw->dwData = this->dwGain; hres = S_OK; break;
case (DWORD)(UINT_PTR)DIPROP_FFLOAD: hres = CDIDev_GetLoad(this, &pdipdw->dwData); break;
case (DWORD)(UINT_PTR)DIPROP_AUTOCENTER: if(this->didcFF & DIDC_FORCEFEEDBACK) { pdipdw->dwData = this->dwAutoCenter; hres = S_OK; } else { hres = E_NOTIMPL; } break; #endif
default: /*
* The user is asking for some property that simply * makes no sense here. E.g., asking for the dead * zone on a keyboard. */ SquirtSqflPtszV(sqfl | sqflBenign, TEXT("GetProperty: Property 0x%08x not supported on device"), (DWORD)(UINT_PTR)ppropi->pguid ); hres = E_NOTIMPL; break;
} }
ExitOleProc(); return hres; }
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | GetProperty | * * Obtain information about a device or object in a device. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm IN REFGUID | rguidProp | * * The identity of the property to be obtained. This can be * one of the predefined <c DIPROP_*> values, or it may * be a private GUID. * * @parm IN LPDIPROPHEADER | pdiph | * * Points to the <t DIPROPHEADER> portion of a structure * which dependson the property. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c DIERR_INVALIDPARAM> = <c E_INVALIDARG>: The * <p pdiph> parameter is not a valid pointer, or the * <p dwHow> field is invalid, or the <p dwObj> field * is not zero when <p dwHow> is set to <c DIPH_DEVICE>. * * <c DIERR_OBJECTNOTFOUND>: The specified object does not * exist. * * <c DIERR_UNSUPPORTED> = <c E_NOTIMPL>: The property * is not supported by the device or object. * * @ex * * The following "C" code fragment illustrates how to obtain * the value of the <c DIPROP_BUFFERSIZE> property. * * | * * DIPROPDWORD dipdw; * HRESULT hres; * dipdw.diph.dwSize = sizeof(DIPROPDWORD); * dipdw.diph.dwHeaderSize = sizeof(DIPROPHEADER); * dipdw.diph.dwObj = 0; // device property
* hres = IDirectInputDevice_GetProperty(pdid, DIPROP_BUFFERSIZE, &dipdw.diph); * if (SUCCEEDED(hres)) { * // dipdw.dwData contains the value of the property
* } * *****************************************************************************/
STDMETHODIMP CDIDev_GetProperty(PV pdd, REFGUID rguid, LPDIPROPHEADER pdiph _THAT) { HRESULT hres; EnterProcR(IDirectInputDevice::GetProperty, (_ "pxp", pdd, rguid, pdiph));
if(SUCCEEDED(hres = hresPvT(pdd))) { PDD this = _thisPv(pdd); DIPROPINFO propi;
/*
* Must protect with the critical section to prevent somebody * acquiring or changing the property we are reading. We need * to do this before validating, to prevent an acquisition. */ CDIDev_EnterCrit(this);
propi.pguid = rguid; if(SUCCEEDED(hres = CDIDev_hresValidProp(this, rguid, pdiph, 1, &propi))) {
hres = this->pdcb->lpVtbl->GetProperty(this->pdcb, &propi, pdiph);
if(hres == E_NOTIMPL) { hres = CDIDev_DefGetProperty(this, &propi, pdiph); }
}
CDIDev_LeaveCrit(this); }
ExitBenignOleProcR(); return hres; }
#ifdef XDEBUG
CSET_STUBS(GetProperty, (PV pdm, REFGUID rguid, LPDIPROPHEADER pdiph), (pdm, rguid, pdiph THAT_))
#else
#define CDIDev_GetPropertyA CDIDev_GetProperty
#define CDIDev_GetPropertyW CDIDev_GetProperty
#endif
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev | SetAxisMode | * * Default handler for clients trying to set the axis mode. * If the device doesn't handle axis modes natively, then * we'll fake it ourselves. * * @parm DWORD | dwMode | * * Desired new mode. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * *****************************************************************************/
STDMETHODIMP CDIDev_SetAxisMode(PDD this, DWORD dwMode) { HRESULT hres; EnterProcR(IDirectInputDevice::SetProperty(AXISMODE), (_ "px", this, dwMode));
AssertF(this->pvi); /* Validation should've caught this */
hres = hresFullValidFl(dwMode, DIPROPAXISMODE_VALID, 2); if(SUCCEEDED(hres)) { if(dwMode & DIPROPAXISMODE_REL) { this->GetDeviceState = CDIDev_GetRelDeviceState; this->pvi->fl |= VIFL_RELATIVE; } else { this->GetDeviceState = CDIDev_GetAbsDeviceState; this->pvi->fl &= ~VIFL_RELATIVE; } if( this->diHacks.fNativeAxisOnly ) { this->pvi->fl |= VIFL_MODECOMPAT; } if(this->cAxes) { hres = S_OK; } else { hres = DI_PROPNOEFFECT; } }
ExitOleProc(); return hres; }
#ifdef IDirectInputDevice2Vtbl
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev | SetAutoCenter | * * Default handler for clients trying to set the * auto-center property. * * If the device doesn't have control over the * auto-center spring, then we fail. * * @parm DWORD | dwMode | * * Desired new mode. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * *****************************************************************************/
STDMETHODIMP CDIDev_SetAutoCenter(PDD this, DWORD dwMode) { HRESULT hres; EnterProcR(IDirectInputDevice::SetProperty(AUTOCENTER), (_ "px", this, dwMode));
hres = hresFullValidFl(dwMode, DIPROPAUTOCENTER_VALID, 2); if(SUCCEEDED(hres)) { if(this->didcFF & DIDC_FORCEFEEDBACK) { /*
* We need to create the effect driver if disabling * autocenter so that CDIDev_FFAcquire will set the feedback * mode properly. */ if(fLimpFF(dwMode == DIPROPAUTOCENTER_OFF, SUCCEEDED(hres = CDIDev_CreateEffectDriver(this)))) { this->dwAutoCenter = dwMode; hres = S_OK; } } else { hres = E_NOTIMPL; } }
ExitOleProc(); return hres; } #endif
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev | SetGlobalAxisProp | * * Default implementation of <mf IDirectInputDevice::SetProperty> * to handle properties which can be applied globally to all * absolute axes. * * @parm IN LPDIPROPINFO | ppropi | * * Information describing the property being set. * We edit it to avoid reallocating memory all the time. * * @parm IN LPCDIPROPHEADER | pdiph | * * The property itself. * * @returns * * We consider the property-set a success if all candidates * succeeded. <c E_NOTIMPL> counts as success, on the assumption * that the property is not meaningful on the candidate. * *****************************************************************************/
STDMETHODIMP CDIDev_SetGlobalAxisProp(PDD this, LPDIPROPINFO ppropi, LPCDIPROPHEADER pdiph) { HRESULT hres;
for(ppropi->iobj = 0; ppropi->iobj < this->df.dwNumObjs; ppropi->iobj++) { DWORD dwType = this->df.rgodf[ppropi->iobj].dwType; if(dwType & DIDFT_ABSAXIS) { ppropi->dwDevType = this->df.rgodf[ppropi->iobj].dwType;
hres = this->pdcb->lpVtbl->SetProperty(this->pdcb, ppropi, pdiph); if(FAILED(hres) && hres != E_NOTIMPL) { goto done; } } } hres = S_OK;
done:; return hres; }
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev | DefSetProperty | * * Default implementation of <mf IDirectInputDevice::SetProperty> * to handle properties which the device decides not to implement. * * @parm IN LPDIPROPINFO | ppropi | * * Information describing the property being set. * We edit it to avoid reallocating memory all the time. * * @parm OUT LPCDIPROPHEADER | pdiph | * * Where to put the property value. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c DI_POLLEDDEVICE>: The device is polled, so the result * might not be meaningful. (This return code is used when * you attempt to set the buffer size property.) * * <c DIERR_INVALIDPARAM> = <c E_INVALIDARG>: The * <p pdiph> parameter is not a valid pointer. * * *****************************************************************************/
STDMETHODIMP CDIDev_DefSetProperty(PDD this, LPDIPROPINFO ppropi, LPCDIPROPHEADER pdiph) { HRESULT hres; EnterProc(CDIDev_DefSetProperty, (_ "pGx", this, ppropi->pguid, ppropi->dwDevType));
AssertF(CDIDev_InCrit(this));
/*
* Note: The indentation here is historical; I left it this way * to keep the diff size down. */
hres = CDIDev_hresValidDefProp(this, ppropi, DIPVIFL_READONLY); if(SUCCEEDED(hres)) { LPDIPROPDWORD pdipdw = (PV)pdiph; LPDIPROPRANGE pdiprg = (PV)pdiph; VXDDWORDDATA vdd;
switch((DWORD)(UINT_PTR)ppropi->pguid) {
case (DWORD)(UINT_PTR)DIPROP_BUFFERSIZE: AssertF(this->pvi); /* Validation should've caught this */ vdd.pvi = this->pvi; if( pdipdw->dwData > this->celtBufMax ) { RPF( "DIPROP_BUFFERSIZE: requested size %d is larger than maximum %d, using %d", pdipdw->dwData, this->celtBufMax, this->celtBufMax ); vdd.dw = this->celtBufMax; } else { vdd.dw = pdipdw->dwData; } hres = Hel_SetBufferSize(&vdd); #ifdef DEBUG_STICKY
{ TCHAR tszDbg[80]; wsprintf( tszDbg, TEXT("SetBufferSize(0x%08x) returned 0x%08x\r\n"), vdd.dw, hres ); OutputDebugString( tszDbg ); } #endif /* DEBUG_STICKY */
if(SUCCEEDED(hres)) { this->celtBuf = pdipdw->dwData; hres = this->hresPolled; } break;
case (DWORD)(UINT_PTR)DIPROP_AXISMODE: hres = CDIDev_SetAxisMode(this, pdipdw->dwData); break;
/*
* We will handle these global properties * if the callback doesn't want to. */ case (DWORD)(UINT_PTR)DIPROP_RANGE: case (DWORD)(UINT_PTR)DIPROP_DEADZONE: case (DWORD)(UINT_PTR)DIPROP_SATURATION: case (DWORD)(UINT_PTR)DIPROP_CALIBRATIONMODE: case (DWORD)(UINT_PTR)DIPROP_CALIBRATION: if(ppropi->dwDevType == 0) { /* For device */ hres = CDIDev_SetGlobalAxisProp(this, ppropi, pdiph); } else { goto _default; } break;
case (DWORD)(UINT_PTR)DIPROP_MAXBUFFERSIZE: this->celtBufMax = pdipdw->dwData; hres = S_OK; break;
#ifdef IDirectInputDevice2Vtbl
case (DWORD)(UINT_PTR)DIPROP_FFGAIN: if(ISVALIDGAIN(pdipdw->dwData)) { this->dwGain = pdipdw->dwData; CDIDev_RefreshGain(this); hres = S_OK; } else { RPF("ERROR: SetProperty(DIPROP_FFGAIN): Gain out of range"); hres = E_INVALIDARG; } break;
case (DWORD)(UINT_PTR)DIPROP_AUTOCENTER: hres = CDIDev_SetAutoCenter(this, pdipdw->dwData); break; #endif
_default:; default: /*
* The validation filter already failed invalid properties. * So what's left is that the property is valid but cannot * be set, because it doesn't exist on the device (e.g., * dead zone) or because it is read-only. */ SquirtSqflPtszV(sqfl | sqflBenign, TEXT("SetProperty: Property 0x%08x not supported on device"), (DWORD)(UINT_PTR)ppropi->pguid ); hres = E_NOTIMPL; break;
}
}
ExitOleProc(); return hres; }
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | IDirectInputDevice | RealSetProperty | * * The function that does the real work. * * <mf IDirectInputDevice::SetDataFormat> will internally * set the axis mode property, so it needs this backdoor * entry point. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm IN REFGUID | rguidProp | * * The identity of the property to be set. * * @parm IN LPDIPROPHEADER | pdiph | * * Points to the <t DIPROPHEADER> portion of a structure * which depends on the property. * *****************************************************************************/
STDMETHODIMP CDIDev_RealSetProperty(PDD this, REFGUID rguid, LPCDIPROPHEADER pdiph) { HRESULT hres; DIPROPINFO propi; EnterProcR(IDirectInputDevice::SetProperty, (_ "pxp", this, rguid, pdiph));
/*
* Must protect with the critical section to prevent somebody * acquiring or changing the property we are reading. We need * to do this before validating, to prevent an acquisition. */ CDIDev_EnterCrit(this);
propi.pguid = rguid; if(SUCCEEDED(hres = CDIDev_hresValidProp(this, rguid, pdiph, 0, &propi))) {
hres = this->pdcb->lpVtbl->SetProperty(this->pdcb, &propi, pdiph);
if(hres == E_NOTIMPL) { hres = CDIDev_DefSetProperty(this, &propi, pdiph); } }
CDIDev_LeaveCrit(this);
ExitOleProc();
return hres; }
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | SetProperty | * * Set information about a device or object in a device. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm IN REFGUID | rguidProp | * * The identity of the property to be set. This can be * one of the predefined <c DIPROP_*> values, or it may * be a pointer to a private GUID. * * @parm IN LPDIPROPHEADER | pdiph | * * Points to the <t DIPROPHEADER> portion of a structure * which depends on the property. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c DI_PROPNOEFFECT> <c S_FALSE>: The operation completed * successfully but * had no effect. For example, changing the axis mode * on a device with no axes will return this value. * * <c DIERR_INVALIDPARAM> = <c E_INVALIDARG>: The * <p pdiph> parameter is not a valid pointer, or the * <p dwHow> field is invalid, or the <p dwObj> field * is not zero when <p dwHow> is set to <c DIPH_DEVICE>. * * <c DIERR_OBJECTNOTFOUND>: The specified object does not * exist. * * <c DIERR_UNSUPPORTED> = <c E_NOTIMPL>: The property * is not supported by the device or object. * *****************************************************************************/
STDMETHODIMP CDIDev_SetProperty(PV pdd, REFGUID rguid, LPCDIPROPHEADER pdiph _THAT) { HRESULT hres; EnterProcR(IDirectInputDevice::SetProperty, (_ "pxp", pdd, rguid, pdiph));
if(SUCCEEDED(hres = hresPvT(pdd))) { PDD this = _thisPv(pdd); hres = CDIDev_RealSetProperty(this, rguid, pdiph); }
ExitOleProcR(); return hres; }
#ifdef XDEBUG
CSET_STUBS(SetProperty, (PV pdm, REFGUID rguid, LPCDIPROPHEADER pdiph), (pdm, rguid, pdiph THAT_))
#else
#define CDIDev_SetPropertyA CDIDev_SetProperty
#define CDIDev_SetPropertyW CDIDev_SetProperty
#endif
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | SetCooperativeLevel | * * Establish the cooperativity level for the instance of * the device. * * The cooperativity level determines how the instance of * the device interacts with other instances of the device * and the rest of the system. * * Note that if the system mouse is acquired in exclusive * mode, then the mouse cursor will be removed from the screen * until the device is unacquired. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm HWND | hwnd | * * The window associated with the device. * The window must be a top-level window. * * It is an error to destroy the window while it is still * active in a DirectInput device. * * @parm DWORD | dwFlags | * * Flags which describe the cooperativity level associated * with the device. * * It consists of <c DISCL_*> flags, documented separately. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c DIERR_INVALIDPARAM> = <c E_INVALIDARG>: The * <p hwnd> parameter is not a valid pointer. * *****************************************************************************/
HRESULT INLINE CDIDev_SetCooperativeLevel_IsValidFl(DWORD dwFlags) { HRESULT hres; RD(static char s_szProc[] = "IDirectInputDevice::SetCooperativeLevel");
if(!(dwFlags & ~DISCL_VALID)) { if((dwFlags & DISCL_EXCLMASK) == DISCL_EXCLUSIVE || (dwFlags & DISCL_EXCLMASK) == DISCL_NONEXCLUSIVE) { if((dwFlags & DISCL_GROUNDMASK) == DISCL_FOREGROUND || (dwFlags & DISCL_GROUNDMASK) == DISCL_BACKGROUND) { hres = S_OK; } else { RPF("ERROR %s: arg %d: Must set exactly one of " "DISCL_FOREGROUND or DISCL_BACKGROUND", s_szProc, 2); hres = E_INVALIDARG; } } else { RPF("ERROR %s: arg %d: Must set exactly one of " "DISCL_EXCLUSIVE or DISCL_NONEXCLUSIVE", s_szProc, 2); hres = E_INVALIDARG; } } else { RPF("ERROR %s: arg %d: invalid flags", s_szProc, 2); hres = E_INVALIDARG;
} return hres; }
HRESULT INLINE CDIDev_SetCooperativeLevel_IsValidHwnd(HWND hwnd, DWORD dwFlags) { HRESULT hres; RD(static char s_szProc[] = "IDirectInputDevice::SetCooperativeLevel");
#if DIRECTINPUT_VERSION == 0x0300
if(dwFlags & DISCL_FOREGROUND) { if(SUCCEEDED(hres = hresFullValidHwnd(hwnd, 1))) { /*
* The window must be a top-level window to be activated. */ if(!(GetWindowLong(hwnd, GWL_STYLE) & WS_CHILD)) { hres = S_OK; } else { RPF("ERROR %s: window may not be a child window", s_szProc); hres = E_HANDLE; } } } else if(dwFlags & DISCL_EXCLUSIVE) { RPF("IDirectInputDevice::SetCooperativeLevel: " "DISCL_EXCLUSIVE requires DISCL_FOREGROUND"); hres = E_NOTIMPL; } else { hres = S_OK; } #else
/*
* If a window handle is passed, it must be valid. * * The window must be a top-level window to be activated. * * The window must belong to the calling process so we can * hook it. */ if(hwnd) { hres = hresFullValidHwnd(hwnd, 1); if(SUCCEEDED(hres)) { if(!(GetWindowLong(hwnd, GWL_STYLE) & WS_CHILD)) { if(GetWindowPid(hwnd) == GetCurrentProcessId()) { } else { RPF("ERROR %s: window must belong to current process", s_szProc); hres = E_HANDLE; }
} else { RPF("ERROR %s: window may not be a child window", s_szProc); hres = E_HANDLE; goto done; } } else { goto done; } }
/*
* Foreground mode or exclusive mode both require a window handle. */ if(dwFlags & (DISCL_FOREGROUND | DISCL_EXCLUSIVE)) { if(hwnd) { } else { RPF("ERROR %s: window handle required " "if DISCL_EXCLUSIVE or DISCL_FOREGROUND", s_szProc); hres = E_HANDLE; goto done; } }
hres = S_OK; done:; #endif
return hres; }
STDMETHODIMP CDIDev_SetCooperativeLevel(PV pdd, HWND hwnd, DWORD dwFlags _THAT) { HRESULT hres; EnterProcR(IDirectInputDevice::SetCooperativeLevel, (_ "pxx", pdd, hwnd, dwFlags));
if(SUCCEEDED(hres = hresPvT(pdd))) { PDD this = _thisPv(pdd);
/*
* Must protect with the critical section to prevent somebody * acquiring or Reset()ing behind our back. */ CDIDev_EnterCrit(this);
if(SUCCEEDED(hres = IDirectInputDevice_NotAcquired(this)) && SUCCEEDED(hres = CDIDev_SetCooperativeLevel_IsValidFl(dwFlags)) && SUCCEEDED(hres = CDIDev_SetCooperativeLevel_IsValidHwnd(hwnd, dwFlags))) {
AssertF(CDIDev_IsConsistent(this));
/*
* For DX7 post Gold fix, check version against co-op level. * Note, GetCaps is non-trivial on a HID but this is no time * for a rewrite. */ if( this->dwVersion < 0x0700 ) { if( dwFlags & DISCL_NOWINKEY ) { hres = E_INVALIDARG; } else if( dwFlags & DISCL_EXCLUSIVE ) { DIDEVCAPS_DX3 dc; dc.dwSize = cbX(dc); if( SUCCEEDED( this->pdcb->lpVtbl->GetCapabilities( this->pdcb, (PV)&dc) ) ) { /*
* Don't allow keyboard exclusive mode pre-DX7 */ if( GET_DIDEVICE_TYPE(dc.dwDevType) == DIDEVTYPE_KEYBOARD ) { hres = E_INVALIDARG; } } else { /*
* PS/2 Mouse and keyboard GetCaps are trivial so * won't fail. HID is unlikely to fail for a mouse * or keyboard as they generally stay plugged in so * assume a failure to GetCaps means a joystick. */ } } }
if( SUCCEEDED( hres ) ) { hres = this->pdcb->lpVtbl->SetCooperativeLevel( this->pdcb, hwnd, dwFlags); if(SUCCEEDED(hres)) { this->discl = dwFlags; this->hwnd = hwnd; if(this->pvi) { this->pvi->hwnd = hwnd; } } } else { AssertF( hres == E_INVALIDARG ); RPF("ERROR %s: arg %d: invalid flags", s_szProc, 2); }
AssertF(CDIDev_IsConsistent(this));
} CDIDev_LeaveCrit(this);
}
ExitOleProcR(); return hres; }
#ifdef XDEBUG
CSET_STUBS(SetCooperativeLevel, (PV pdm, HWND hwnd, DWORD fl), (pdm, hwnd, fl THAT_))
#else
#define CDIDev_SetCooperativeLevelA CDIDev_SetCooperativeLevel
#define CDIDev_SetCooperativeLevelW CDIDev_SetCooperativeLevel
#endif
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | RunControlPanel | * * Run the DirectInput control panel for the device. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm IN HWND | hwndOwner | * * Identifies the window handle that will be used as the * parent window for subsequent UI. NULL is a valid parameter, * indicating that there is no parent window. * * @parm DWORD | dwFlags | * * No flags are currently defined. This parameter "must" be * zero. * * @returns * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The device is attached. * * @devnote * * The <p dwFlags> is eventually going to allow * <c DIRCP_MODAL> to request a modal control panel. * *****************************************************************************/
STDMETHODIMP CDIDev_RunControlPanel(PV pdd, HWND hwndOwner, DWORD fl _THAT) { HRESULT hres; EnterProcR(IDirectInputDevice::RunControlPanel, (_ "pxx", pdd, hwndOwner, fl));
if(SUCCEEDED(hres = hresPvT(pdd)) && SUCCEEDED(hres = hresFullValidHwnd0(hwndOwner, 1)) && SUCCEEDED(hres = hresFullValidFl(fl, DIRCP_VALID, 2))) {
PDD this = _thisPv(pdd); IDirectInputDeviceCallback *pdcb;
/*
* Must protect with the critical section to prevent somebody * Reset()ing behind our back. However, we cannot hold the * critical section during the control panel callback, because * that will yield. * * So we copy/addref the pdcb inside the critical section, * then run the control panel outside the critical section, * then release the pdcb when we're finally done. */ CDIDev_EnterCrit(this);
pdcb = this->pdcb; OLE_AddRef(pdcb);
CDIDev_LeaveCrit(this);
hres = pdcb->lpVtbl->RunControlPanel(pdcb, hwndOwner, fl);
OLE_Release(pdcb); }
ExitOleProc(); return hres; }
#ifdef XDEBUG
CSET_STUBS(RunControlPanel, (PV pdd, HWND hwndOwner, DWORD fl), (pdd, hwndOwner, fl THAT_))
#else
#define CDIDev_RunControlPanelA CDIDev_RunControlPanel
#define CDIDev_RunControlPanelW CDIDev_RunControlPanel
#endif
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | Initialize | * * Initialize a DirectInputDevice object. * * Note that if this method fails, the underlying object should * be considered to be an an indeterminate state and needs to * be reinitialized before it can be subsequently used. * * The <mf IDirectInput::CreateDevice> method automatically * initializes the device after creating it. Applications * normally do not need to call this function. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm IN HINSTANCE | hinst | * * Instance handle of the application or DLL that is creating * the DirectInput object. * * See the section titled "Initialization and Versions" * for more information. * * @parm DWORD | dwVersion | * * Version number of the dinput.h header file that was used. * * See the section titled "Initialization and Versions" * for more information. * * @parm IN REFGUID | rguid | * * Identifies the instance of the device for which the interface * should be associated. * The <mf IDirectInput::EnumDevices> method * can be used to determine which instance GUIDs are supported by * the system. * * @returns * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c S_FALSE>: The device had already been initialized with * the instance GUID passed in <p lpGUID>. * * <c DIERR_ACQUIRED>: The device cannot be initialized while * it is acquired. * * <c DIERR_DEVICENOTREG>: The instance GUID does not exist * on the current machine. * * <c DIERR_HASEFFECTS>: * The device cannot be reinitialized because there are * still effects attached to it. * *****************************************************************************/
STDMETHODIMP CDIDev_Initialize(PV pdd, HINSTANCE hinst, DWORD dwVersion, REFGUID rguid _THAT) { HRESULT hres; EnterProcR(IDirectInputDevice::Initialize, (_ "pxxG", pdd, hinst, dwVersion, rguid));
if(SUCCEEDED(hres = hresPvT(pdd)) && SUCCEEDED(hres = hresFullValidGuid(rguid, 1))) { PDD this = _thisPv(pdd); CREATEDCB CreateDcb; IDirectInputDeviceCallback *pdcb;
/*
* Must take the critical section to avoid Reset()ing * the device (or generally speaking, screwing with the * internal state variables) while somebody else is * messing with it. */ CDIDev_EnterCrit(this);
if(SUCCEEDED(hres = hresValidInstanceVer(hinst, dwVersion)) && SUCCEEDED(hres = hresFindInstanceGUID(rguid, &CreateDcb, 1)) && SUCCEEDED(hres = CDIDev_Reset(this))) { hres = CreateDcb(0, rguid, &IID_IDirectInputDeviceCallback, (PPV)&pdcb); if(SUCCEEDED(hres)) { this->pdcb = pdcb; AssertF(this->pvi == 0); if(SUCCEEDED(hres = CDIDev_GetDataFormat(this)) && SUCCEEDED(hres = CDIDev_GetPolled(this)) && SUCCEEDED(hres = this->pdcb->lpVtbl->GetInstance( this->pdcb, &this->pvi))) { #ifdef IDirectInputDevice2Vtbl
this->dwVersion = dwVersion; #if (DIRECTINPUT_VERSION > 0x061A)
AhGetAppHacks( &this->diHacks ); #endif
this->pdcb->lpVtbl->SetDIData(this->pdcb, dwVersion, &this->diHacks); #ifdef BUGGY_DX7_WINNT
if( (this->dwVersion < 0x700) && (this->dwVersion != 0x5B2) ) { if( GET_DIDEVICE_TYPE(this->diHacks.dwDevType) == DIDEVTYPE_JOYSTICK ) { hres = CDIDev_ParseDataFormatInternal(this, &c_dfDIJoystick); } } #endif //BUGGY_DX7_WINNT
#endif //IDirectInputDevice2Vtbl
if(dwVersion >= DIRECTINPUT_VERSION) { this->didftInstance = DIDFT_ANYINSTANCE; } this->guid = *rguid; if(this->pvi && (this->pvi->fl & VIFL_EMULATED)) { this->pvi->pdd = this; }
hres = this->pdcb->lpVtbl->CookDeviceData(this->pdcb, 0, 0); if(SUCCEEDED(hres)) { this->fCook = 1; }
#ifdef IDirectInputDevice2Vtbl
CDIDev_InitFF(this); #endif
hres = S_OK; } else { RPF("Device driver didn't provide a data format"); } } else { #ifdef NOISY
RPF("Cannot create device"); #endif
} } CDIDev_LeaveCrit(this); }
ExitOleProc(); return hres; }
#ifdef XDEBUG
CSET_STUBS(Initialize, (PV pdd, HINSTANCE hinst, DWORD dwVersion, REFGUID rguid), (pdd, hinst, dwVersion, rguid THAT_))
#else
#define CDIDev_InitializeA CDIDev_Initialize
#define CDIDev_InitializeW CDIDev_Initialize
#endif
/*****************************************************************************
* * @doc INTERNAL * * @method void | IDirectInputDevice | Init | * * Initialize the internal parts of the DirectInputDevice object. * *****************************************************************************/
void INLINE CDIDev_Init(PDD this) { /*
* The critical section must be the very first thing we do, * because only Finalize checks for its existence. * * (We might be finalized without being initialized if the user * passed a bogus interface to CDIDev_New.) */ this->fCritInited = fInitializeCriticalSection(&this->crst);
if( this->fCritInited ) { this->celtBufMax = 1023; /* Default maximum buffer size */
this->pdcb = c_pdcbNil;
#ifdef IDirectInputDevice2Vtbl
GPA_InitFromZero(&this->gpaEff); #endif
} }
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | IDirectInputDevice | New | * * Create a new DirectInputDevice object, uninitialized. * * @parm IN PUNK | punkOuter | * * Controlling unknown for aggregation. * * @parm IN RIID | riid | * * Desired interface to new object. * * @parm OUT PPV | ppvObj | * * Output pointer for new object. * *****************************************************************************/
STDMETHODIMP CDIDev_New(PUNK punkOuter, RIID riid, PPV ppvObj) { HRESULT hres; EnterProcR(IDirectInputDevice::<constructor>, (_ "Gp", riid, punkOuter));
if (SUCCEEDED(hres = hresFullValidPcbOut(ppvObj, cbX(*ppvObj), 3))) { hres = Excl_Init(); if(SUCCEEDED(hres)) { LPVOID pvTry = NULL; hres = Common_NewRiid(CDIDev, punkOuter, riid, &pvTry);
if(SUCCEEDED(hres)) { PDD this = _thisPv(pvTry); CDIDev_Init(this); if( this->fCritInited ) { *ppvObj = pvTry; } else { Common_Unhold(this); *ppvObj = NULL; hres = E_OUTOFMEMORY; } } } }
ExitOleProcPpvR(ppvObj); return hres; }
/*****************************************************************************
* * @doc INTERNAL * * @method HRESULT | CDIDev_ModifyEffectParams | * * Modifies parameters of DIEFFECT structure to fit the current FF device * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * * @parm IN OUT LPDIEFFECT | peff | * * Pointer to the effect structure * * @parm IN GUID | effGUID | * * GUID for the effect * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c DIERR_UNSUPPORTED>: The effect can't be supported by the current device * (e.g. the number of FF axes on the device is 0) * * <c DIERR_INVALIDPARAM>: Can't create the effect even with the modified parameters * *****************************************************************************/
HRESULT CDIDev_ModifyEffectParams ( PV pdd, LPDIEFFECT peff, GUID effGUID ) { HRESULT hres = S_OK; HRESULT hresCreate = S_OK; LPDIRECTINPUTEFFECT pdeff; PDD this = _thisPv(pdd);
EnterProcR(CDIDev_ModifyEffectParams, (_ "p", pdd));
// Check to make sure that effects we enumerate will
// actually get created on the device.
// try creating the effect.
#ifdef XDEBUG
hresCreate = CDIDev_CreateEffect(this, &effGUID, peff, &pdeff, NULL, ((LPUNKNOWN)this)->lpVtbl); #else
hresCreate = CDIDev_CreateEffect(this, &effGUID, peff, &pdeff, NULL); #endif
if(SUCCEEDED(hresCreate)) { Invoke_Release(&pdeff); } else { if (hresCreate == DIERR_INVALIDPARAM) { //two things can give DIERR_INVALIDPARAM:
//invalid axes and invalid trigger button
//check the axes first, then the trigger buttons
//try to eliminate all DIERR_INVALIDPARAMS
LPDIOBJECTDATAFORMAT lpObjDat = this->df.rgodf; DWORD dwNum = this->df.dwNumObjs; DWORD nCount; LPDWORD lpAxes; LPDWORD lpThisAxis; LPDWORD lpEffAxis; DWORD nAxes = 0; DWORD dwTrigger = DIJOFS_BUTTON(0); BOOL bTriggerCorrect = FALSE; AllocCbPpv(sizeof(DWORD)*dwNum, &lpAxes); lpThisAxis = lpAxes; for (nCount = 0; nCount < dwNum; nCount ++) { AssertF(lpObjDat != NULL);
//check the axes
if ((lpObjDat->dwType & (DIDFT_AXIS | DIDFT_FFACTUATOR) & DIDFT_TYPEMASK) && (fHasAllBitsFlFl(lpObjDat->dwType, (DIDFT_AXIS | DIDFT_FFACTUATOR) & DIDFT_ATTRMASK))) { *lpAxes = lpObjDat->dwOfs; nAxes++; lpAxes++; } else { //check the trigger button, if there's one
if ((peff->dwTriggerButton != DIEB_NOTRIGGER) && (lpObjDat->dwType & DIDFT_FFEFFECTTRIGGER & DIDFT_TYPEMASK) && (fHasAllBitsFlFl(lpObjDat->dwType, DIDFT_FFEFFECTTRIGGER & DIDFT_ATTRMASK))) { if (lpObjDat->dwOfs == peff->dwTriggerButton) { //the trigger is valid
bTriggerCorrect = TRUE; } else { //remember the trigger offset for the future
dwTrigger = lpObjDat->dwOfs; } } }
lpObjDat++; }
//first, chack if there are any FF axes
if (nAxes == 0) { //return an error if no FF axes on device
hres = DIERR_UNSUPPORTED; } else {
//trigger buttons are checked for validity before axes,
//so set the trigger button, if needed,
//because if it is invalid, this is what caused the error
if ((peff->dwTriggerButton != DIEB_NOTRIGGER) && (bTriggerCorrect == FALSE)) { peff->dwTriggerButton = dwTrigger;
// and try creating again
#ifdef XDEBUG
hresCreate = CDIDev_CreateEffect(this, &effGUID, peff, &pdeff, NULL, ((LPUNKNOWN)this)->lpVtbl); #else
hresCreate = CDIDev_CreateEffect(this, &effGUID, peff, &pdeff, NULL); #endif
if(SUCCEEDED(hresCreate)) { Invoke_Release(&pdeff); }
} if (hresCreate == DIERR_INVALIDPARAM) { HRESULT hresInfo = S_OK; EFFECTMAPINFO emi;
//this time, set the axes
if (peff->cAxes > nAxes) { //change the number of axes
peff->cAxes = nAxes;
//change the flags
if ((nAxes < 3) && (peff->dwFlags & DIEFF_SPHERICAL)) { peff->dwFlags &= ~DIEFF_SPHERICAL; peff->dwFlags |= DIEFF_POLAR; } else { if ((nAxes < 2) && (peff->dwFlags & DIEFF_POLAR)) { peff->dwFlags &= ~DIEFF_POLAR; peff->dwFlags |= DIEFF_CARTESIAN; } }
}
//check if size of type-specific param structures is not bigger then number of axes,
//since this can also give us invalid params in type-specific .
//need to do this only for conditions
if (SUCCEEDED(hresInfo = CDIDev_FindEffectGUID(this, &effGUID, &emi, 2))) { //do the conditions
if (emi.attr.dwEffType & DIEFT_CONDITION) { if (peff->cbTypeSpecificParams/(sizeof(DICONDITION)) > peff->cAxes) { peff->cbTypeSpecificParams = peff->cAxes*(sizeof(DICONDITION)); } }
//don't need to do anything for custom forces,
//since DInput doesn't check number of channels against number of axes anyway
}
//write over the axes
lpEffAxis = peff->rgdwAxes; for (nCount = 0; nCount < nAxes, nCount < peff->cAxes; nCount ++) { *(lpEffAxis) = *(lpThisAxis); lpThisAxis ++; lpEffAxis++; }
// and try creating again
#ifdef XDEBUG
hresCreate = CDIDev_CreateEffect(this, &effGUID, peff, &pdeff, NULL, ((LPUNKNOWN)this)->lpVtbl); #else
hresCreate = CDIDev_CreateEffect(this, &effGUID, peff, &pdeff, NULL); #endif
if(SUCCEEDED(hresCreate)) { Invoke_Release(&pdeff); } } }
//free the axes array
FreePpv(&lpAxes); } }
if ((SUCCEEDED(hres)) && (hresCreate == DIERR_INVALIDPARAM)) { hres = hresCreate; }
ExitOleProc(); return hres;
}
/*****************************************************************************
* * @doc INTERNAL * * @method BOOL | CDIDev_IsStandardEffect | * * Checks if the effect GUID belongs to a standard DI effect * * @parm IN GUID | effGUID | * * GUID for the effect * * @returns BOOL * * TRUE if it is a standard DI effect; * FALSE otherwise. * * *****************************************************************************/
BOOL CDIDev_IsStandardEffect (GUID effGUID) { BOOL bStandard = TRUE;
//check all the standard DX7 GUIDs
if ((IsEqualGUID(&effGUID, &GUID_Sine)) || (IsEqualGUID(&effGUID, &GUID_Triangle)) || (IsEqualGUID(&effGUID, &GUID_ConstantForce)) || (IsEqualGUID(&effGUID, &GUID_RampForce)) || (IsEqualGUID(&effGUID, &GUID_Square)) || (IsEqualGUID(&effGUID, &GUID_SawtoothUp)) || (IsEqualGUID(&effGUID, &GUID_SawtoothDown)) || (IsEqualGUID(&effGUID, &GUID_Spring)) || (IsEqualGUID(&effGUID, &GUID_Damper)) || (IsEqualGUID(&effGUID, &GUID_Inertia)) || (IsEqualGUID(&effGUID, &GUID_Friction)) || (IsEqualGUID(&effGUID, &GUID_CustomForce))) { bStandard = TRUE; }
else { bStandard = FALSE; }
return bStandard;
}
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | EnumEffectsInFile | * * Enumerates DIEFFECT struct(s) and effect GUID from file. * An application can use this in order to create pre-authored * force effects. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm LPCSTR | lpszFileName | * * Name of the RIFF file that contains collection of effects. * * @parm IN OUT LPENUMEFFECTSCALLBACK | pec | * * The callback function. * * @parm IN OUT LPVOID | pvRef | * Specifies the application-defined value given in the * <mf IDirectInputDevice::EnumObjects> function. * * @parm IN DWORD | dwFlags | * * Flags which control the enumeration. * * It consists of <c DIFEF_*> flags, documented separately. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c DIERR_INVALIDPARAM> = <c E_INVALIDARG>: The * <p lpDirectInputDevice> or * <p lpdc> parameter is invalid. * * @cb BOOL CALLBACK | DIEnumEffectsCallback | * * An application-defined callback function that receives * effect GUID, DIEFFECT and repeat count as a result of a call to the * <om IDirectInputDevice::EnumEffectsInFile> method. * * @parm OUT LPCDIFILEEFFECT | lpDiFileEf | * * Pointer to a DIFILEEFFECT structure. * * * @parm IN OUT LPVOID | pvRef | * Specifies the application-defined value given in the * <mf IDirectInputDevice::EnumObjects> function. * * @returns * * Returns <c DIENUM_CONTINUE> to continue the enumeration * or <c DIENUM_STOP> to stop the enumeration. * *****************************************************************************/
HRESULT CDIDev_EnumEffectsInFileA ( PV pddA, LPCSTR lpszFileName, LPDIENUMEFFECTSINFILECALLBACK pec, LPVOID pvRef, DWORD dwFlags ) { HRESULT hres = E_FAIL;
EnterProcR(IDirectInputDevice::EnumEffectsInFile, (_ "s", lpszFileName));
/* Validate incoming parameters */ if(SUCCEEDED(hres = hresPvA(pddA)) && SUCCEEDED(hres = hresFullValidReadStrA(lpszFileName, MAX_JOYSTRING,1)) && SUCCEEDED(hres = hresFullValidPfn(pec, 2)) && SUCCEEDED(hres = hresFullValidFl(dwFlags, DIFEF_ENUMVALID, 3)) ) { PDD this = _thisPvNm(pddA, ddA); HMMIO hmmio; MMCKINFO mmck; DWORD dwEffectSz;
hres = RIFF_Open(lpszFileName, MMIO_READ | MMIO_ALLOCBUF , &hmmio, &mmck, &dwEffectSz);
if(SUCCEEDED(hres)) { HRESULT hresRead; DIEFFECT effect; DIFILEEFFECT DiFileEf; DIENVELOPE diEnvelope; DWORD rgdwAxes[DIEFFECT_MAXAXES]; LONG rglDirection[DIEFFECT_MAXAXES]; effect.rgdwAxes = rgdwAxes; effect.rglDirection = rglDirection; effect.lpEnvelope = &diEnvelope;
DiFileEf.dwSize = cbX(DiFileEf); DiFileEf.lpDiEffect = &effect;
while ((SUCCEEDED(hres)) && (SUCCEEDED(hresRead = RIFF_ReadEffect(hmmio, &DiFileEf)))) { BOOL fRc = DIENUM_CONTINUE; BOOL bInclude = TRUE; HRESULT hresModify = DI_OK;
//modify if needed
if (dwFlags & DIFEF_MODIFYIFNEEDED) { hresModify = CDIDev_ModifyEffectParams(this, &effect, DiFileEf.GuidEffect); }
//if necessary, check whether effect is standard
if (!(dwFlags & DIFEF_INCLUDENONSTANDARD)) { bInclude = CDIDev_IsStandardEffect(DiFileEf.GuidEffect); }
//call back only if all the conditions posed by the flags are satisfied
if ((SUCCEEDED(hresModify)) && (bInclude == TRUE)) { fRc = Callback(pec, &DiFileEf, pvRef); }
//free type-specific only if allocated
if(effect.cbTypeSpecificParams > 0) { FreePv(effect.lpvTypeSpecificParams); effect.cbTypeSpecificParams = 0x0; }
if(fRc == DIENUM_STOP) { break; } else if(fRc == DIENUM_CONTINUE) { continue; } else { RPF("IDirectInputDevice::EnumEffectsInFile: Invalid return value from enumeration callback"); ValidationException(); break; } } RIFF_Close(hmmio, 0); //if hresRead failed because couldn't descend into the chunk, it means the end of file,
//so everything is OK;
//else return this error
if (SUCCEEDED(hres)) { if (hresRead == hresLe(ERROR_SECTOR_NOT_FOUND)) { hres = S_OK; } else { hres = hresRead; } } } }
ExitOleProc(); return hres; }
HRESULT CDIDev_EnumEffectsInFileW ( PV pddW, LPCWSTR lpszFileName, LPDIENUMEFFECTSINFILECALLBACK pec, LPVOID pvRef, DWORD dwFlags ) {
HRESULT hres = E_FAIL;
EnterProcR(IDirectInputDevice::EnumEffectsInFileW, (_ "s", lpszFileName));
/* Validate incoming parameters */ if(SUCCEEDED(hres = hresPvW(pddW)) && SUCCEEDED(hres = hresFullValidReadStrW(lpszFileName, MAX_JOYSTRING,1)) && SUCCEEDED(hres = hresFullValidPfn(pec, 2)) && SUCCEEDED(hres = hresFullValidFl(dwFlags, DIFEF_ENUMVALID, 3)) ) { CHAR szFileName[MAX_PATH];
PDD this = _thisPvNm(pddW, ddW); UToA(szFileName, MAX_PATH, lpszFileName);
hres = CDIDev_EnumEffectsInFileA(&this->ddA, szFileName, pec, pvRef, dwFlags); }
return hres; }
/*****************************************************************************
* * @doc EXTERNAL * * @method HRESULT | IDirectInputDevice | WriteEffectToFile | * * Writes DIEFFECT struct(s) and effect GUID to a file. * An application can use this in order to create pre-authored * force effects. * * @cwrap LPDIRECTINPUTDEVICE | lpDirectInputDevice * * @parm LPCSTR | lpszFileName | * * Name of the RIFF file that contains collection of effects. * * @parm IN DWORD | dwEntries | * * Number of <t DIFILEEFFECT> structures in the array. * * @parm IN LPCDIFILEEFFECT | rgDiFileEft | * * Array of <t DIFILEEFFECT> structure. * * * @parm IN DWORD | dwFlags | * * Flags which control how the effect should be written. * * It consists of <c DIFEF_*> flags, documented separately. * * @returns * * Returns a COM error code. The following error codes are * intended to be illustrative and not necessarily comprehensive. * * <c DI_OK> = <c S_OK>: The operation completed successfully. * * <c DIERR_INVALIDPARAM> = <c E_INVALIDARG>: The * <p lpDirectInputDevice> or * <p lpdc> parameter is invalid. * *****************************************************************************/
HRESULT CDIDev_WriteEffectToFileA ( PV pddA, LPCSTR lpszFileName, DWORD dwEntries, LPDIFILEEFFECT rgDiFileEffect, DWORD dwFlags ) { HRESULT hres = E_NOTIMPL;
EnterProcR(IDirectInputDevice:cd inc :WriteEffectToFileA, (_ "s", lpszFileName));
/* Validate incoming parameters */ if(SUCCEEDED(hres = hresPvA(pddA)) && SUCCEEDED(hres = hresFullValidReadStrA(lpszFileName, MAX_JOYSTRING,1))&& SUCCEEDED(hres = hresFullValidFl(dwFlags, DIFEF_ENUMVALID, 3)) && SUCCEEDED(hres = (IsBadReadPtr(rgDiFileEffect, cbX(*rgDiFileEffect))) ? E_POINTER : S_OK))
{ PDD this = _thisPvNm(pddA, ddA); HMMIO hmmio; MMCKINFO mmck; DWORD dwEffectSz;
hres = RIFF_Open(lpszFileName, MMIO_CREATE | MMIO_WRITE | MMIO_ALLOCBUF , &hmmio, &mmck, &dwEffectSz);
if(SUCCEEDED(hres)) { UINT nCount; LPDIFILEEFFECT lpDiFileEf = rgDiFileEffect;
//write out the effects
for(nCount = 0; nCount < dwEntries; nCount++) { BOOL bInclude = TRUE;
hres = (IsBadReadPtr(lpDiFileEf, cbX(*lpDiFileEf))) ? E_POINTER : S_OK;
if (FAILED(hres)) { break; }
//if necessary, check whether the effect is standard
if (!(dwFlags & DIFEF_INCLUDENONSTANDARD)) { bInclude = CDIDev_IsStandardEffect(lpDiFileEf->GuidEffect); }
if ((SUCCEEDED(hres)) && (bInclude == TRUE)) { hres = RIFF_WriteEffect(hmmio, lpDiFileEf); }
if(FAILED(hres)) { break; }
lpDiFileEf++; } RIFF_Close(hmmio, 0); } }
ExitOleProc(); return hres; }
HRESULT CDIDev_WriteEffectToFileW ( PV pddW, LPCWSTR lpszFileName, DWORD dwEntries, LPDIFILEEFFECT lpDiFileEffect, DWORD dwFlags ) {
HRESULT hres = E_FAIL;
EnterProcR(IDirectInputDevice::WriteEffectToFile, (_ "s", lpszFileName));
/* Validate incoming parameters */ if(SUCCEEDED(hres = hresPvW(pddW)) && SUCCEEDED(hres = hresFullValidReadStrW(lpszFileName, MAX_JOYSTRING,1))) { CHAR szFileName[MAX_PATH];
PDD this = _thisPvNm(pddW, ddW); UToA(szFileName, MAX_PATH, lpszFileName);
hres = CDIDev_WriteEffectToFileA(&this->ddA, szFileName, dwEntries, lpDiFileEffect, dwFlags); }
return hres; } /*****************************************************************************
* * The long-awaited vtbls and templates * *****************************************************************************/
#pragma BEGIN_CONST_DATA
#define CDIDev_Signature 0x20564544 /* "DEV " */
Primary_Interface_Begin(CDIDev, TFORM(ThisInterfaceT)) TFORM(CDIDev_GetCapabilities), TFORM(CDIDev_EnumObjects), TFORM(CDIDev_GetProperty), TFORM(CDIDev_SetProperty), TFORM(CDIDev_Acquire), TFORM(CDIDev_Unacquire), TFORM(CDIDev_GetDeviceState), TFORM(CDIDev_GetDeviceData), TFORM(CDIDev_SetDataFormat), TFORM(CDIDev_SetEventNotification), TFORM(CDIDev_SetCooperativeLevel), TFORM(CDIDev_GetObjectInfo), TFORM(CDIDev_GetDeviceInfo), TFORM(CDIDev_RunControlPanel), TFORM(CDIDev_Initialize), #ifdef IDirectInputDevice2Vtbl
TFORM(CDIDev_CreateEffect), TFORM(CDIDev_EnumEffects), TFORM(CDIDev_GetEffectInfo), TFORM(CDIDev_GetForceFeedbackState), TFORM(CDIDev_SendForceFeedbackCommand), TFORM(CDIDev_EnumCreatedEffectObjects), TFORM(CDIDev_Escape), TFORM(CDIDev_Poll), TFORM(CDIDev_SendDeviceData), #ifdef IDirectInputDevice7Vtbl
TFORM(CDIDev_EnumEffectsInFile), TFORM(CDIDev_WriteEffectToFile), #endif
#endif
Primary_Interface_End(CDIDev, TFORM(ThisInterfaceT))
Secondary_Interface_Begin(CDIDev, SFORM(ThisInterfaceT), SFORM(dd)) SFORM(CDIDev_GetCapabilities), SFORM(CDIDev_EnumObjects), SFORM(CDIDev_GetProperty), SFORM(CDIDev_SetProperty), SFORM(CDIDev_Acquire), SFORM(CDIDev_Unacquire), SFORM(CDIDev_GetDeviceState), SFORM(CDIDev_GetDeviceData), SFORM(CDIDev_SetDataFormat), SFORM(CDIDev_SetEventNotification), SFORM(CDIDev_SetCooperativeLevel), SFORM(CDIDev_GetObjectInfo), SFORM(CDIDev_GetDeviceInfo), SFORM(CDIDev_RunControlPanel), SFORM(CDIDev_Initialize), #ifdef IDirectInputDevice2Vtbl
SFORM(CDIDev_CreateEffect), SFORM(CDIDev_EnumEffects), SFORM(CDIDev_GetEffectInfo), SFORM(CDIDev_GetForceFeedbackState), SFORM(CDIDev_SendForceFeedbackCommand), SFORM(CDIDev_EnumCreatedEffectObjects), TFORM(CDIDev_Escape), SFORM(CDIDev_Poll), SFORM(CDIDev_SendDeviceData), #ifdef IDirectInputDevice7Vtbl
SFORM(CDIDev_EnumEffectsInFile), SFORM(CDIDev_WriteEffectToFile), #endif
#endif
Secondary_Interface_End(CDIDev, SFORM(ThisInterfaceT), SFORM(dd))
|