set svn:eol-style to native
[reactos.git] / reactos / lib / dinput / device.c
index 2980c3c..4d5aefa 100644 (file)
-/*             DirectInput Device\r
- *\r
- * Copyright 1998 Marcus Meissner\r
- * Copyright 1998,1999 Lionel Ulmer\r
- *\r
- *\r
- * This library is free software; you can redistribute it and/or\r
- * modify it under the terms of the GNU Lesser General Public\r
- * License as published by the Free Software Foundation; either\r
- * version 2.1 of the License, or (at your option) any later version.\r
- *\r
- * This library is distributed in the hope that it will be useful,\r
- * but WITHOUT ANY WARRANTY; without even the implied warranty of\r
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\r
- * Lesser General Public License for more details.\r
- *\r
- * You should have received a copy of the GNU Lesser General Public\r
- * License along with this library; if not, write to the Free Software\r
- * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA\r
- */\r
-\r
-/* This file contains all the Device specific functions that can be used as stubs\r
-   by real device implementations.\r
-\r
-   It also contains all the helper functions.\r
-*/\r
-#include "config.h"\r
-\r
-#include <stdarg.h>\r
-#include <string.h>\r
-#include "wine/debug.h"\r
-#include "wine/unicode.h"\r
-#include "windef.h"\r
-#include "winbase.h"\r
-#include "winerror.h"\r
-#include "dinput.h"\r
-#include "device_private.h"\r
-\r
-WINE_DEFAULT_DEBUG_CHANNEL(dinput);\r
-\r
-/******************************************************************************\r
- *     Various debugging tools\r
- */\r
-void _dump_cooperativelevel_DI(DWORD dwFlags) {\r
-    if (TRACE_ON(dinput)) {\r
-       unsigned int   i;\r
-       static const struct {\r
-           DWORD       mask;\r
-           const char  *name;\r
-       } flags[] = {\r
-#define FE(x) { x, #x}\r
-           FE(DISCL_BACKGROUND),\r
-           FE(DISCL_EXCLUSIVE),\r
-           FE(DISCL_FOREGROUND),\r
-           FE(DISCL_NONEXCLUSIVE)\r
-#undef FE\r
-       };\r
-       for (i = 0; i < (sizeof(flags) / sizeof(flags[0])); i++)\r
-           if (flags[i].mask & dwFlags)\r
-               DPRINTF("%s ",flags[i].name);\r
-       DPRINTF("\n");\r
-    }\r
-}\r
-\r
-void _dump_EnumObjects_flags(DWORD dwFlags) {\r
-    if (TRACE_ON(dinput)) {\r
-       unsigned int   i;\r
-       DWORD type, instance;\r
-       static const struct {\r
-           DWORD       mask;\r
-           const char  *name;\r
-       } flags[] = {\r
-#define FE(x) { x, #x}\r
-           FE(DIDFT_RELAXIS),\r
-           FE(DIDFT_ABSAXIS),\r
-           FE(DIDFT_PSHBUTTON),\r
-           FE(DIDFT_TGLBUTTON),\r
-           FE(DIDFT_POV),\r
-           FE(DIDFT_COLLECTION),\r
-           FE(DIDFT_NODATA),       \r
-           FE(DIDFT_FFACTUATOR),\r
-           FE(DIDFT_FFEFFECTTRIGGER),\r
-           FE(DIDFT_OUTPUT)\r
-#undef FE\r
-       };\r
-       type = (dwFlags & 0xFF0000FF);\r
-       instance = ((dwFlags >> 8) & 0xFFFF);\r
-       DPRINTF("Type:");\r
-       if (type == DIDFT_ALL) {\r
-           DPRINTF(" DIDFT_ALL");\r
-       } else {\r
-           for (i = 0; i < (sizeof(flags) / sizeof(flags[0])); i++) {\r
-               if (flags[i].mask & type) {\r
-                   type &= ~flags[i].mask;\r
-                   DPRINTF(" %s",flags[i].name);\r
-               }\r
-           }\r
-           if (type) {\r
-               DPRINTF(" (unhandled: %08lx)", type);\r
-           }\r
-       }\r
-       DPRINTF(" / Instance: ");\r
-       if (instance == ((DIDFT_ANYINSTANCE >> 8) & 0xFFFF)) {\r
-           DPRINTF("DIDFT_ANYINSTANCE");\r
-       } else {\r
-           DPRINTF("%3ld", instance);\r
-       }\r
-    }\r
-}\r
-\r
-void _dump_DIPROPHEADER(LPCDIPROPHEADER diph) {\r
-    if (TRACE_ON(dinput)) {\r
-       DPRINTF("  - dwObj = 0x%08lx\n", diph->dwObj);\r
-       DPRINTF("  - dwHow = %s\n",\r
-               ((diph->dwHow == DIPH_DEVICE) ? "DIPH_DEVICE" :\r
-                ((diph->dwHow == DIPH_BYOFFSET) ? "DIPH_BYOFFSET" :\r
-                 ((diph->dwHow == DIPH_BYID)) ? "DIPH_BYID" : "unknown")));\r
-    }\r
-}\r
-\r
-void _dump_OBJECTINSTANCEA(DIDEVICEOBJECTINSTANCEA *ddoi) {\r
-    if (TRACE_ON(dinput)) {\r
-       DPRINTF("    - enumerating : %s ('%s') - %2ld - 0x%08lx - %s\n",\r
-               debugstr_guid(&ddoi->guidType), _dump_dinput_GUID(&ddoi->guidType), ddoi->dwOfs, ddoi->dwType, ddoi->tszName);\r
-    }\r
-}\r
-\r
-void _dump_OBJECTINSTANCEW(DIDEVICEOBJECTINSTANCEW *ddoi) {\r
-    if (TRACE_ON(dinput)) {\r
-       DPRINTF("    - enumerating : %s ('%s'), - %2ld - 0x%08lx - %s\n",\r
-               debugstr_guid(&ddoi->guidType), _dump_dinput_GUID(&ddoi->guidType), ddoi->dwOfs, ddoi->dwType, debugstr_w(ddoi->tszName));\r
-    }\r
-}\r
-\r
-/* This function is a helper to convert a GUID into any possible DInput GUID out there */\r
-const char *_dump_dinput_GUID(const GUID *guid) {\r
-    unsigned int i;\r
-    static const struct {\r
-       const GUID *guid;\r
-       const char *name;\r
-    } guids[] = {\r
-#define FE(x) { &x, #x}\r
-       FE(GUID_XAxis),\r
-       FE(GUID_YAxis),\r
-       FE(GUID_ZAxis),\r
-       FE(GUID_RxAxis),\r
-       FE(GUID_RyAxis),\r
-       FE(GUID_RzAxis),\r
-       FE(GUID_Slider),\r
-       FE(GUID_Button),\r
-       FE(GUID_Key),\r
-       FE(GUID_POV),\r
-       FE(GUID_Unknown),\r
-       FE(GUID_SysMouse),\r
-       FE(GUID_SysKeyboard),\r
-       FE(GUID_Joystick),\r
-       FE(GUID_ConstantForce),\r
-       FE(GUID_RampForce),\r
-       FE(GUID_Square),\r
-       FE(GUID_Sine),\r
-       FE(GUID_Triangle),\r
-       FE(GUID_SawtoothUp),\r
-       FE(GUID_SawtoothDown),\r
-       FE(GUID_Spring),\r
-       FE(GUID_Damper),\r
-       FE(GUID_Inertia),\r
-       FE(GUID_Friction),\r
-       FE(GUID_CustomForce)\r
-#undef FE\r
-    };\r
-    if (guid == NULL)\r
-       return "null GUID";\r
-    for (i = 0; i < (sizeof(guids) / sizeof(guids[0])); i++) {\r
-       if (IsEqualGUID(guids[i].guid, guid)) {\r
-           return guids[i].name;\r
-       }\r
-    }\r
-    return "Unknown GUID";\r
-}\r
-\r
-void _dump_DIDATAFORMAT(const DIDATAFORMAT *df) {\r
-    unsigned int i;\r
-\r
-    TRACE("Dumping DIDATAFORMAT structure:\n");\r
-    TRACE("  - dwSize: %ld\n", df->dwSize);\r
-    if (df->dwSize != sizeof(DIDATAFORMAT)) {\r
-       WARN("Non-standard DIDATAFORMAT structure size (%ld instead of %d).\n", df->dwSize, sizeof(DIDATAFORMAT));\r
-    }\r
-    TRACE("  - dwObjsize: %ld\n", df->dwObjSize);\r
-    if (df->dwObjSize != sizeof(DIOBJECTDATAFORMAT)) {\r
-       WARN("Non-standard DIOBJECTDATAFORMAT structure size (%ld instead of %d).\n", df->dwObjSize, sizeof(DIOBJECTDATAFORMAT));\r
-    }\r
-    TRACE("  - dwFlags: 0x%08lx (", df->dwFlags);\r
-    switch (df->dwFlags) {\r
-        case DIDF_ABSAXIS: TRACE("DIDF_ABSAXIS"); break;\r
-       case DIDF_RELAXIS: TRACE("DIDF_RELAXIS"); break;\r
-       default: TRACE("unknown"); break;\r
-    }\r
-    TRACE(")\n");\r
-    TRACE("  - dwDataSize: %ld\n", df->dwDataSize);\r
-    TRACE("  - dwNumObjs: %ld\n", df->dwNumObjs);\r
-    \r
-    for (i = 0; i < df->dwNumObjs; i++) {\r
-       TRACE("  - Object %d:\n", i);\r
-       TRACE("      * GUID: %s ('%s')\n", debugstr_guid(df->rgodf[i].pguid), _dump_dinput_GUID(df->rgodf[i].pguid));\r
-       TRACE("      * dwOfs: %ld\n", df->rgodf[i].dwOfs);\r
-       TRACE("      * dwType: 0x%08lx\n", df->rgodf[i].dwType);\r
-       TRACE("        "); _dump_EnumObjects_flags(df->rgodf[i].dwType); TRACE("\n");\r
-       TRACE("      * dwFlags: 0x%08lx\n", df->rgodf[i].dwFlags);\r
-    }\r
-}\r
-\r
-/* Conversion between internal data buffer and external data buffer */\r
-void fill_DataFormat(void *out, const void *in, DataFormat *df) {\r
-    int i;\r
-    char *in_c = (char *) in;\r
-    char *out_c = (char *) out;\r
-    \r
-    if (df->dt == NULL) {\r
-       /* This means that the app uses Wine's internal data format */\r
-       memcpy(out, in, df->internal_format_size);\r
-    } else {\r
-       for (i = 0; i < df->size; i++) {\r
-           if (df->dt[i].offset_in >= 0) {\r
-               switch (df->dt[i].size) {\r
-                   case 1:\r
-                       TRACE("Copying (c) to %d from %d (value %d)\n",\r
-                             df->dt[i].offset_out, df->dt[i].offset_in, *((char *) (in_c + df->dt[i].offset_in)));\r
-                       *((char *) (out_c + df->dt[i].offset_out)) = *((char *) (in_c + df->dt[i].offset_in));\r
-                       break;\r
-                   \r
-                   case 2:\r
-                       TRACE("Copying (s) to %d from %d (value %d)\n",\r
-                             df->dt[i].offset_out, df->dt[i].offset_in, *((short *) (in_c + df->dt[i].offset_in)));\r
-                       *((short *) (out_c + df->dt[i].offset_out)) = *((short *) (in_c + df->dt[i].offset_in));\r
-                       break;\r
-                   \r
-                   case 4:\r
-                       TRACE("Copying (i) to %d from %d (value %d)\n",\r
-                             df->dt[i].offset_out, df->dt[i].offset_in, *((int *) (in_c + df->dt[i].offset_in)));\r
-                       *((int *) (out_c + df->dt[i].offset_out)) = *((int *) (in_c + df->dt[i].offset_in));\r
-                       break;\r
-                   \r
-                   default:\r
-                       memcpy((out_c + df->dt[i].offset_out), (in_c + df->dt[i].offset_in), df->dt[i].size);\r
-                       break;\r
-               }\r
-           } else {\r
-               switch (df->dt[i].size) {\r
-                   case 1:\r
-                       TRACE("Copying (c) to %d default value %d\n",\r
-                             df->dt[i].offset_out, df->dt[i].value);\r
-                       *((char *) (out_c + df->dt[i].offset_out)) = (char) df->dt[i].value;\r
-                       break;\r
-                       \r
-                   case 2:\r
-                       TRACE("Copying (s) to %d default value %d\n",\r
-                             df->dt[i].offset_out, df->dt[i].value);\r
-                       *((short *) (out_c + df->dt[i].offset_out)) = (short) df->dt[i].value;\r
-                       break;\r
-                       \r
-                   case 4:\r
-                       TRACE("Copying (i) to %d default value %d\n",\r
-                             df->dt[i].offset_out, df->dt[i].value);\r
-                       *((int *) (out_c + df->dt[i].offset_out)) = (int) df->dt[i].value;\r
-                       break;\r
-                       \r
-                   default:\r
-                       memset((out_c + df->dt[i].offset_out), df->dt[i].size, 0);\r
-                       break;\r
-               }\r
-           }\r
-       }\r
-    }\r
-}\r
-\r
-void release_DataFormat(DataFormat * format)\r
-{\r
-    TRACE("Deleting DataTransform : \n");\r
-\r
-    HeapFree(GetProcessHeap(), 0, format->dt);\r
-}\r
-\r
-DataFormat *create_DataFormat(const DIDATAFORMAT *wine_format, LPCDIDATAFORMAT asked_format, int *offset) {\r
-    DataFormat *ret;\r
-    DataTransform *dt;\r
-    unsigned int i, j;\r
-    int same = 1;\r
-    int *done;\r
-    int index = 0;\r
-    DWORD next = 0;\r
-    \r
-    ret = (DataFormat *) HeapAlloc(GetProcessHeap(), 0, sizeof(DataFormat));\r
-    \r
-    done = (int *) HeapAlloc(GetProcessHeap(), 0, sizeof(int) * asked_format->dwNumObjs);\r
-    memset(done, 0, sizeof(int) * asked_format->dwNumObjs);\r
-    \r
-    dt = (DataTransform *) HeapAlloc(GetProcessHeap(), 0, asked_format->dwNumObjs * sizeof(DataTransform));\r
-    \r
-    TRACE("Creating DataTransform : \n");\r
-    \r
-    for (i = 0; i < wine_format->dwNumObjs; i++) {\r
-       offset[i] = -1;\r
-       \r
-       for (j = 0; j < asked_format->dwNumObjs; j++) {\r
-           if (done[j] == 1)\r
-               continue;\r
-           \r
-           if (/* Check if the application either requests any GUID and if not, it if matches\r
-                * the GUID of the Wine object.\r
-                */\r
-               ((asked_format->rgodf[j].pguid == NULL) ||\r
-                (IsEqualGUID(wine_format->rgodf[i].pguid, asked_format->rgodf[j].pguid)))\r
-               &&\r
-               (/* Then check if it accepts any instance id, and if not, if it matches Wine's\r
-                 * instance id.\r
-                 */\r
-                ((asked_format->rgodf[j].dwType & 0x00FFFF00) == DIDFT_ANYINSTANCE) ||\r
-                (wine_format->rgodf[i].dwType & asked_format->rgodf[j].dwType))) {\r
-               \r
-               done[j] = 1;\r
-               \r
-               TRACE("Matching : \n");\r
-               TRACE("   - Asked (%d) :\n", j);\r
-               TRACE("       * GUID: %s ('%s')\n",\r
-                     debugstr_guid(asked_format->rgodf[j].pguid),\r
-                     _dump_dinput_GUID(asked_format->rgodf[j].pguid));\r
-               TRACE("       * Offset: %3ld\n", asked_format->rgodf[j].dwOfs);\r
-               TRACE("       * dwType: %08lx\n", asked_format->rgodf[j].dwType);\r
-               TRACE("         "); _dump_EnumObjects_flags(asked_format->rgodf[j].dwType); TRACE("\n");\r
-               \r
-               TRACE("   - Wine  (%d) :\n", j);\r
-               TRACE("       * GUID: %s ('%s')\n",\r
-                     debugstr_guid(wine_format->rgodf[j].pguid),\r
-                     _dump_dinput_GUID(wine_format->rgodf[j].pguid));\r
-               TRACE("       * Offset: %3ld\n", wine_format->rgodf[j].dwOfs);\r
-               TRACE("       * dwType: %08lx\n", wine_format->rgodf[j].dwType);\r
-               TRACE("         "); _dump_EnumObjects_flags(wine_format->rgodf[j].dwType); TRACE("\n");\r
-               \r
-               if (wine_format->rgodf[i].dwType & DIDFT_BUTTON)\r
-                   dt[index].size = sizeof(BYTE);\r
-               else\r
-                   dt[index].size = sizeof(DWORD);\r
-               dt[index].offset_in  = wine_format ->rgodf[i].dwOfs;\r
-                if (asked_format->rgodf[j].dwOfs < next) {\r
-                    WARN("bad format: dwOfs=%ld, changing to %ld\n", asked_format->rgodf[j].dwOfs, next);\r
-                   dt[index].offset_out = next;\r
-                   offset[i] = next;\r
-                } else {\r
-                   dt[index].offset_out = asked_format->rgodf[j].dwOfs;\r
-                    offset[i] = asked_format->rgodf[j].dwOfs;\r
-                }\r
-               dt[index].value = 0;\r
-                next = next + dt[index].size;\r
-               \r
-               if (wine_format->rgodf[i].dwOfs != dt[index].offset_out)\r
-                   same = 0;\r
-               \r
-               index++;\r
-               break;\r
-           }\r
-       }\r
-       \r
-       if (j == asked_format->dwNumObjs)\r
-           same = 0;\r
-    }\r
-    \r
-    TRACE("Setting to default value :\n");\r
-    for (j = 0; j < asked_format->dwNumObjs; j++) {\r
-       if (done[j] == 0) {\r
-           TRACE("   - Asked (%d) :\n", j);\r
-           TRACE("       * GUID: %s ('%s')\n",\r
-                 debugstr_guid(asked_format->rgodf[j].pguid),\r
-                 _dump_dinput_GUID(asked_format->rgodf[j].pguid));\r
-           TRACE("       * Offset: %3ld\n", asked_format->rgodf[j].dwOfs);\r
-           TRACE("       * dwType: %08lx\n", asked_format->rgodf[j].dwType);\r
-           TRACE("         "); _dump_EnumObjects_flags(asked_format->rgodf[j].dwType); TRACE("\n");\r
-           \r
-           if (asked_format->rgodf[j].dwType & DIDFT_BUTTON)\r
-               dt[index].size = sizeof(BYTE);\r
-           else\r
-               dt[index].size = sizeof(DWORD);\r
-           dt[index].offset_in  = -1;\r
-           dt[index].offset_out = asked_format->rgodf[j].dwOfs;\r
-           dt[index].value = 0;\r
-           index++;\r
-           \r
-           same = 0;\r
-       }\r
-    }\r
-    \r
-    ret->internal_format_size = wine_format->dwDataSize;\r
-    ret->size = index;\r
-    if (same) {\r
-       ret->dt = NULL;\r
-       HeapFree(GetProcessHeap(), 0, dt);\r
-    } else {\r
-       ret->dt = dt;\r
-    }\r
-    \r
-    HeapFree(GetProcessHeap(), 0, done);\r
-    \r
-    return ret;\r
-}\r
-\r
-BOOL DIEnumDevicesCallbackAtoW(LPCDIDEVICEOBJECTINSTANCEA lpddi, LPVOID lpvRef) {\r
-    DIDEVICEOBJECTINSTANCEW ddtmp;\r
-    device_enumobjects_AtoWcb_data* data;\r
-\r
-    data = (device_enumobjects_AtoWcb_data*) lpvRef;\r
-    \r
-    memset(&ddtmp, 0, sizeof(DIDEVICEINSTANCEW)); \r
-    \r
-    ddtmp.dwSize = sizeof(DIDEVICEINSTANCEW);\r
-    ddtmp.guidType     = lpddi->guidType;\r
-    ddtmp.dwOfs        = lpddi->dwOfs;\r
-    ddtmp.dwType       = lpddi->dwType;\r
-    ddtmp.dwFlags      = lpddi->dwFlags;\r
-    MultiByteToWideChar(CP_ACP, 0, lpddi->tszName, -1, ddtmp.tszName, MAX_PATH);\r
-    \r
-    if (lpddi->dwSize == sizeof(DIDEVICEINSTANCEA)) {\r
-       /**\r
-        * if dwSize < sizeof(DIDEVICEINSTANCEA of DInput version >= 5)\r
-        *  force feedback and other newer datas aren't available\r
-        */\r
-       ddtmp.dwFFMaxForce        = lpddi->dwFFMaxForce;\r
-       ddtmp.dwFFForceResolution = lpddi->dwFFForceResolution;\r
-       ddtmp.wCollectionNumber   = lpddi->wCollectionNumber;\r
-       ddtmp.wDesignatorIndex    = lpddi->wDesignatorIndex;\r
-       ddtmp.wUsagePage          = lpddi->wUsagePage;\r
-       ddtmp.wUsage              = lpddi->wUsage;\r
-       ddtmp.dwDimension         = lpddi->dwDimension;\r
-       ddtmp.wExponent           = lpddi->wExponent;\r
-       ddtmp.wReserved           = lpddi->wReserved;\r
-    }\r
-    return data->lpCallBack(&ddtmp, data->lpvRef);\r
-}\r
-\r
-/******************************************************************************\r
- *     IDirectInputDeviceA\r
- */\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_SetDataFormat(\r
-       LPDIRECTINPUTDEVICE8A iface,LPCDIDATAFORMAT df\r
-) {\r
-    IDirectInputDevice2AImpl *This = (IDirectInputDevice2AImpl *)iface;\r
-    \r
-    TRACE("(this=%p,%p)\n",This,df);\r
-    \r
-    _dump_DIDATAFORMAT(df);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_SetCooperativeLevel(\r
-       LPDIRECTINPUTDEVICE8A iface,HWND hwnd,DWORD dwflags\r
-) {\r
-    IDirectInputDevice2AImpl *This = (IDirectInputDevice2AImpl *)iface;\r
-    TRACE("(this=%p,0x%08lx,0x%08lx)\n",This,(DWORD)hwnd,dwflags);\r
-    if (TRACE_ON(dinput)) {\r
-       TRACE(" cooperative level : ");\r
-       _dump_cooperativelevel_DI(dwflags);\r
-    }\r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_SetEventNotification(\r
-       LPDIRECTINPUTDEVICE8A iface,HANDLE hnd\r
-) {\r
-    IDirectInputDevice2AImpl *This = (IDirectInputDevice2AImpl *)iface;\r
-    FIXME("(this=%p,0x%08lx): stub\n",This,(DWORD)hnd);\r
-    return DI_OK;\r
-}\r
-\r
-ULONG WINAPI IDirectInputDevice2AImpl_Release(LPDIRECTINPUTDEVICE8A iface)\r
-{\r
-    IDirectInputDevice2AImpl *This = (IDirectInputDevice2AImpl *)iface;\r
-    ULONG ref;\r
-    ref = InterlockedDecrement(&(This->ref));\r
-    if (ref == 0)\r
-        HeapFree(GetProcessHeap(),0,This);\r
-    return ref;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_QueryInterface(\r
-       LPDIRECTINPUTDEVICE8A iface,REFIID riid,LPVOID *ppobj\r
-)\r
-{\r
-    IDirectInputDevice2AImpl *This = (IDirectInputDevice2AImpl *)iface;\r
-    \r
-    TRACE("(this=%p,%s,%p)\n",This,debugstr_guid(riid),ppobj);\r
-    if (IsEqualGUID(&IID_IUnknown,riid)) {\r
-       IDirectInputDevice2_AddRef(iface);\r
-       *ppobj = This;\r
-       return DI_OK;\r
-    }\r
-    if (IsEqualGUID(&IID_IDirectInputDeviceA,riid)) {\r
-       IDirectInputDevice2_AddRef(iface);\r
-       *ppobj = This;\r
-       return DI_OK;\r
-    }\r
-    if (IsEqualGUID(&IID_IDirectInputDevice2A,riid)) {\r
-       IDirectInputDevice2_AddRef(iface);\r
-       *ppobj = This;\r
-       return DI_OK;\r
-    }\r
-    if (IsEqualGUID(&IID_IDirectInputDevice7A,riid)) {\r
-       IDirectInputDevice7_AddRef(iface);\r
-       *ppobj = This;\r
-       return DI_OK;\r
-    }\r
-    TRACE("Unsupported interface !\n");\r
-    return E_FAIL;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2WImpl_QueryInterface(\r
-       LPDIRECTINPUTDEVICE8W iface,REFIID riid,LPVOID *ppobj\r
-)\r
-{\r
-    IDirectInputDevice2AImpl *This = (IDirectInputDevice2AImpl *)iface;\r
-    \r
-    TRACE("(this=%p,%s,%p)\n",This,debugstr_guid(riid),ppobj);\r
-    if (IsEqualGUID(&IID_IUnknown,riid)) {\r
-       IDirectInputDevice2_AddRef(iface);\r
-       *ppobj = This;\r
-       return DI_OK;\r
-    }\r
-    if (IsEqualGUID(&IID_IDirectInputDeviceW,riid)) {\r
-       IDirectInputDevice2_AddRef(iface);\r
-       *ppobj = This;\r
-       return DI_OK;\r
-    }\r
-    if (IsEqualGUID(&IID_IDirectInputDevice2W,riid)) {\r
-       IDirectInputDevice2_AddRef(iface);\r
-       *ppobj = This;\r
-       return DI_OK;\r
-    }\r
-    if (IsEqualGUID(&IID_IDirectInputDevice7W,riid)) {\r
-       IDirectInputDevice7_AddRef(iface);\r
-       *ppobj = This;\r
-       return DI_OK;\r
-    }\r
-    TRACE("Unsupported interface !\n");\r
-    return E_FAIL;\r
-}\r
-\r
-ULONG WINAPI IDirectInputDevice2AImpl_AddRef(\r
-       LPDIRECTINPUTDEVICE8A iface)\r
-{\r
-    IDirectInputDevice2AImpl *This = (IDirectInputDevice2AImpl *)iface;\r
-    return InterlockedIncrement(&(This->ref));\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_EnumObjects(\r
-       LPDIRECTINPUTDEVICE8A iface,\r
-       LPDIENUMDEVICEOBJECTSCALLBACKA lpCallback,\r
-       LPVOID lpvRef,\r
-       DWORD dwFlags)\r
-{\r
-    FIXME("(this=%p,%p,%p,%08lx): stub!\n", iface, lpCallback, lpvRef, dwFlags);\r
-    if (TRACE_ON(dinput)) {\r
-       DPRINTF("  - flags = ");\r
-       _dump_EnumObjects_flags(dwFlags);\r
-       DPRINTF("\n");\r
-    }\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2WImpl_EnumObjects(\r
-       LPDIRECTINPUTDEVICE8W iface,\r
-       LPDIENUMDEVICEOBJECTSCALLBACKW lpCallback,\r
-       LPVOID lpvRef,\r
-       DWORD dwFlags)\r
-{\r
-    FIXME("(this=%p,%p,%p,%08lx): stub!\n", iface, lpCallback, lpvRef, dwFlags);\r
-    if (TRACE_ON(dinput)) {\r
-       DPRINTF("  - flags = ");\r
-       _dump_EnumObjects_flags(dwFlags);\r
-       DPRINTF("\n");\r
-    }\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_GetProperty(\r
-       LPDIRECTINPUTDEVICE8A iface,\r
-       REFGUID rguid,\r
-       LPDIPROPHEADER pdiph)\r
-{\r
-    FIXME("(this=%p,%s,%p): stub!\n",\r
-         iface, debugstr_guid(rguid), pdiph);\r
-    \r
-    if (TRACE_ON(dinput))\r
-       _dump_DIPROPHEADER(pdiph);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_GetObjectInfo(\r
-       LPDIRECTINPUTDEVICE8A iface,\r
-       LPDIDEVICEOBJECTINSTANCEA pdidoi,\r
-       DWORD dwObj,\r
-       DWORD dwHow)\r
-{\r
-    FIXME("(this=%p,%p,%ld,0x%08lx): stub!\n",\r
-         iface, pdidoi, dwObj, dwHow);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2WImpl_GetObjectInfo(\r
-       LPDIRECTINPUTDEVICE8W iface,\r
-       LPDIDEVICEOBJECTINSTANCEW pdidoi,\r
-       DWORD dwObj,\r
-       DWORD dwHow)\r
-{\r
-    FIXME("(this=%p,%p,%ld,0x%08lx): stub!\n",\r
-         iface, pdidoi, dwObj, dwHow);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_GetDeviceInfo(\r
-       LPDIRECTINPUTDEVICE8A iface,\r
-       LPDIDEVICEINSTANCEA pdidi)\r
-{\r
-    FIXME("(this=%p,%p): stub!\n",\r
-         iface, pdidi);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2WImpl_GetDeviceInfo(\r
-       LPDIRECTINPUTDEVICE8W iface,\r
-       LPDIDEVICEINSTANCEW pdidi)\r
-{\r
-    FIXME("(this=%p,%p): stub!\n",\r
-         iface, pdidi);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_RunControlPanel(\r
-       LPDIRECTINPUTDEVICE8A iface,\r
-       HWND hwndOwner,\r
-       DWORD dwFlags)\r
-{\r
-    FIXME("(this=%p,%p,0x%08lx): stub!\n",\r
-         iface, hwndOwner, dwFlags);\r
-\r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_Initialize(\r
-       LPDIRECTINPUTDEVICE8A iface,\r
-       HINSTANCE hinst,\r
-       DWORD dwVersion,\r
-       REFGUID rguid)\r
-{\r
-    FIXME("(this=%p,%p,%ld,%s): stub!\n",\r
-         iface, hinst, dwVersion, debugstr_guid(rguid));\r
-    return DI_OK;\r
-}\r
-\r
-/******************************************************************************\r
- *     IDirectInputDevice2A\r
- */\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_CreateEffect(\r
-       LPDIRECTINPUTDEVICE8A iface,\r
-       REFGUID rguid,\r
-       LPCDIEFFECT lpeff,\r
-       LPDIRECTINPUTEFFECT *ppdef,\r
-       LPUNKNOWN pUnkOuter)\r
-{\r
-    FIXME("(this=%p,%s,%p,%p,%p): stub!\n",\r
-         iface, debugstr_guid(rguid), lpeff, ppdef, pUnkOuter);\r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_EnumEffects(\r
-       LPDIRECTINPUTDEVICE8A iface,\r
-       LPDIENUMEFFECTSCALLBACKA lpCallback,\r
-       LPVOID lpvRef,\r
-       DWORD dwFlags)\r
-{\r
-    FIXME("(this=%p,%p,%p,0x%08lx): stub!\n",\r
-         iface, lpCallback, lpvRef, dwFlags);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2WImpl_EnumEffects(\r
-       LPDIRECTINPUTDEVICE8W iface,\r
-       LPDIENUMEFFECTSCALLBACKW lpCallback,\r
-       LPVOID lpvRef,\r
-       DWORD dwFlags)\r
-{\r
-    FIXME("(this=%p,%p,%p,0x%08lx): stub!\n",\r
-         iface, lpCallback, lpvRef, dwFlags);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_GetEffectInfo(\r
-       LPDIRECTINPUTDEVICE8A iface,\r
-       LPDIEFFECTINFOA lpdei,\r
-       REFGUID rguid)\r
-{\r
-    FIXME("(this=%p,%p,%s): stub!\n",\r
-         iface, lpdei, debugstr_guid(rguid));\r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2WImpl_GetEffectInfo(\r
-       LPDIRECTINPUTDEVICE8W iface,\r
-       LPDIEFFECTINFOW lpdei,\r
-       REFGUID rguid)\r
-{\r
-    FIXME("(this=%p,%p,%s): stub!\n",\r
-         iface, lpdei, debugstr_guid(rguid));\r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_GetForceFeedbackState(\r
-       LPDIRECTINPUTDEVICE8A iface,\r
-       LPDWORD pdwOut)\r
-{\r
-    FIXME("(this=%p,%p): stub!\n",\r
-         iface, pdwOut);\r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_SendForceFeedbackCommand(\r
-       LPDIRECTINPUTDEVICE8A iface,\r
-       DWORD dwFlags)\r
-{\r
-    FIXME("(this=%p,0x%08lx): stub!\n",\r
-         iface, dwFlags);\r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_EnumCreatedEffectObjects(\r
-       LPDIRECTINPUTDEVICE8A iface,\r
-       LPDIENUMCREATEDEFFECTOBJECTSCALLBACK lpCallback,\r
-       LPVOID lpvRef,\r
-       DWORD dwFlags)\r
-{\r
-    FIXME("(this=%p,%p,%p,0x%08lx): stub!\n",\r
-         iface, lpCallback, lpvRef, dwFlags);\r
-    if (lpCallback)\r
-       lpCallback(NULL, lpvRef);\r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_Escape(\r
-       LPDIRECTINPUTDEVICE8A iface,\r
-       LPDIEFFESCAPE lpDIEEsc)\r
-{\r
-    FIXME("(this=%p,%p): stub!\n",\r
-         iface, lpDIEEsc);\r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_Poll(\r
-       LPDIRECTINPUTDEVICE8A iface)\r
-{\r
-    /* Because wine devices do not need to be polled, just return DI_NOEFFECT */\r
-    return DI_NOEFFECT;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice2AImpl_SendDeviceData(\r
-       LPDIRECTINPUTDEVICE8A iface,\r
-       DWORD cbObjectData,\r
-       LPCDIDEVICEOBJECTDATA rgdod,\r
-       LPDWORD pdwInOut,\r
-       DWORD dwFlags)\r
-{\r
-    FIXME("(this=%p,0x%08lx,%p,%p,0x%08lx): stub!\n",\r
-         iface, cbObjectData, rgdod, pdwInOut, dwFlags);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice7AImpl_EnumEffectsInFile(LPDIRECTINPUTDEVICE8A iface,\r
-                                                         LPCSTR lpszFileName,\r
-                                                         LPDIENUMEFFECTSINFILECALLBACK pec,\r
-                                                         LPVOID pvRef,\r
-                                                         DWORD dwFlags)\r
-{\r
-    FIXME("(%p)->(%s,%p,%p,%08lx): stub !\n", iface, lpszFileName, pec, pvRef, dwFlags);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice7WImpl_EnumEffectsInFile(LPDIRECTINPUTDEVICE8W iface,\r
-                                                         LPCWSTR lpszFileName,\r
-                                                         LPDIENUMEFFECTSINFILECALLBACK pec,\r
-                                                         LPVOID pvRef,\r
-                                                         DWORD dwFlags)\r
-{\r
-    FIXME("(%p)->(%s,%p,%p,%08lx): stub !\n", iface, debugstr_w(lpszFileName), pec, pvRef, dwFlags);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice7AImpl_WriteEffectToFile(LPDIRECTINPUTDEVICE8A iface,\r
-                                                         LPCSTR lpszFileName,\r
-                                                         DWORD dwEntries,\r
-                                                         LPDIFILEEFFECT rgDiFileEft,\r
-                                                         DWORD dwFlags)\r
-{\r
-    FIXME("(%p)->(%s,%08lx,%p,%08lx): stub !\n", iface, lpszFileName, dwEntries, rgDiFileEft, dwFlags);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice7WImpl_WriteEffectToFile(LPDIRECTINPUTDEVICE8W iface,\r
-                                                         LPCWSTR lpszFileName,\r
-                                                         DWORD dwEntries,\r
-                                                         LPDIFILEEFFECT rgDiFileEft,\r
-                                                         DWORD dwFlags)\r
-{\r
-    FIXME("(%p)->(%s,%08lx,%p,%08lx): stub !\n", iface, debugstr_w(lpszFileName), dwEntries, rgDiFileEft, dwFlags);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice8AImpl_BuildActionMap(LPDIRECTINPUTDEVICE8A iface,\r
-                                                      LPDIACTIONFORMATA lpdiaf,\r
-                                                      LPCSTR lpszUserName,\r
-                                                      DWORD dwFlags)\r
-{\r
-    FIXME("(%p)->(%p,%s,%08lx): stub !\n", iface, lpdiaf, lpszUserName, dwFlags);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice8WImpl_BuildActionMap(LPDIRECTINPUTDEVICE8W iface,\r
-                                                      LPDIACTIONFORMATW lpdiaf,\r
-                                                      LPCWSTR lpszUserName,\r
-                                                      DWORD dwFlags)\r
-{\r
-    FIXME("(%p)->(%p,%s,%08lx): stub !\n", iface, lpdiaf, debugstr_w(lpszUserName), dwFlags);\r
-  \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice8AImpl_SetActionMap(LPDIRECTINPUTDEVICE8A iface,\r
-                                                    LPDIACTIONFORMATA lpdiaf,\r
-                                                    LPCSTR lpszUserName,\r
-                                                    DWORD dwFlags)\r
-{\r
-    FIXME("(%p)->(%p,%s,%08lx): stub !\n", iface, lpdiaf, lpszUserName, dwFlags);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice8WImpl_SetActionMap(LPDIRECTINPUTDEVICE8W iface,\r
-                                                    LPDIACTIONFORMATW lpdiaf,\r
-                                                    LPCWSTR lpszUserName,\r
-                                                    DWORD dwFlags)\r
-{\r
-    FIXME("(%p)->(%p,%s,%08lx): stub !\n", iface, lpdiaf, debugstr_w(lpszUserName), dwFlags);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice8AImpl_GetImageInfo(LPDIRECTINPUTDEVICE8A iface,\r
-                                                    LPDIDEVICEIMAGEINFOHEADERA lpdiDevImageInfoHeader)\r
-{\r
-    FIXME("(%p)->(%p): stub !\n", iface, lpdiDevImageInfoHeader);\r
-    \r
-    return DI_OK;\r
-}\r
-\r
-HRESULT WINAPI IDirectInputDevice8WImpl_GetImageInfo(LPDIRECTINPUTDEVICE8W iface,\r
-                                                    LPDIDEVICEIMAGEINFOHEADERW lpdiDevImageInfoHeader)\r
-{\r
-    FIXME("(%p)->(%p): stub !\n", iface, lpdiDevImageInfoHeader);\r
-    \r
-    return DI_OK;\r
-}\r
+/*             DirectInput Device
+ *
+ * Copyright 1998 Marcus Meissner
+ * Copyright 1998,1999 Lionel Ulmer
+ *
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ */
+
+/* This file contains all the Device specific functions that can be used as stubs
+   by real device implementations.
+
+   It also contains all the helper functions.
+*/
+#include "config.h"
+
+#include <stdarg.h>
+#include <string.h>
+#include "wine/debug.h"
+#include "wine/unicode.h"
+#include "windef.h"
+#include "winbase.h"
+#include "winerror.h"
+#include "dinput.h"
+#include "device_private.h"
+
+WINE_DEFAULT_DEBUG_CHANNEL(dinput);
+
+/******************************************************************************
+ *     Various debugging tools
+ */
+void _dump_cooperativelevel_DI(DWORD dwFlags) {
+    if (TRACE_ON(dinput)) {
+       unsigned int   i;
+       static const struct {
+           DWORD       mask;
+           const char  *name;
+       } flags[] = {
+#define FE(x) { x, #x}
+           FE(DISCL_BACKGROUND),
+           FE(DISCL_EXCLUSIVE),
+           FE(DISCL_FOREGROUND),
+           FE(DISCL_NONEXCLUSIVE)
+#undef FE
+       };
+       for (i = 0; i < (sizeof(flags) / sizeof(flags[0])); i++)
+           if (flags[i].mask & dwFlags)
+               DPRINTF("%s ",flags[i].name);
+       DPRINTF("\n");
+    }
+}
+
+void _dump_EnumObjects_flags(DWORD dwFlags) {
+    if (TRACE_ON(dinput)) {
+       unsigned int   i;
+       DWORD type, instance;
+       static const struct {
+           DWORD       mask;
+           const char  *name;
+       } flags[] = {
+#define FE(x) { x, #x}
+           FE(DIDFT_RELAXIS),
+           FE(DIDFT_ABSAXIS),
+           FE(DIDFT_PSHBUTTON),
+           FE(DIDFT_TGLBUTTON),
+           FE(DIDFT_POV),
+           FE(DIDFT_COLLECTION),
+           FE(DIDFT_NODATA),       
+           FE(DIDFT_FFACTUATOR),
+           FE(DIDFT_FFEFFECTTRIGGER),
+           FE(DIDFT_OUTPUT)
+#undef FE
+       };
+       type = (dwFlags & 0xFF0000FF);
+       instance = ((dwFlags >> 8) & 0xFFFF);
+       DPRINTF("Type:");
+       if (type == DIDFT_ALL) {
+           DPRINTF(" DIDFT_ALL");
+       } else {
+           for (i = 0; i < (sizeof(flags) / sizeof(flags[0])); i++) {
+               if (flags[i].mask & type) {
+                   type &= ~flags[i].mask;
+                   DPRINTF(" %s",flags[i].name);
+               }
+           }
+           if (type) {
+               DPRINTF(" (unhandled: %08lx)", type);
+           }
+       }
+       DPRINTF(" / Instance: ");
+       if (instance == ((DIDFT_ANYINSTANCE >> 8) & 0xFFFF)) {
+           DPRINTF("DIDFT_ANYINSTANCE");
+       } else {
+           DPRINTF("%3ld", instance);
+       }
+    }
+}
+
+void _dump_DIPROPHEADER(LPCDIPROPHEADER diph) {
+    if (TRACE_ON(dinput)) {
+       DPRINTF("  - dwObj = 0x%08lx\n", diph->dwObj);
+       DPRINTF("  - dwHow = %s\n",
+               ((diph->dwHow == DIPH_DEVICE) ? "DIPH_DEVICE" :
+                ((diph->dwHow == DIPH_BYOFFSET) ? "DIPH_BYOFFSET" :
+                 ((diph->dwHow == DIPH_BYID)) ? "DIPH_BYID" : "unknown")));
+    }
+}
+
+void _dump_OBJECTINSTANCEA(DIDEVICEOBJECTINSTANCEA *ddoi) {
+    if (TRACE_ON(dinput)) {
+       DPRINTF("    - enumerating : %s ('%s') - %2ld - 0x%08lx - %s\n",
+               debugstr_guid(&ddoi->guidType), _dump_dinput_GUID(&ddoi->guidType), ddoi->dwOfs, ddoi->dwType, ddoi->tszName);
+    }
+}
+
+void _dump_OBJECTINSTANCEW(DIDEVICEOBJECTINSTANCEW *ddoi) {
+    if (TRACE_ON(dinput)) {
+       DPRINTF("    - enumerating : %s ('%s'), - %2ld - 0x%08lx - %s\n",
+               debugstr_guid(&ddoi->guidType), _dump_dinput_GUID(&ddoi->guidType), ddoi->dwOfs, ddoi->dwType, debugstr_w(ddoi->tszName));
+    }
+}
+
+/* This function is a helper to convert a GUID into any possible DInput GUID out there */
+const char *_dump_dinput_GUID(const GUID *guid) {
+    unsigned int i;
+    static const struct {
+       const GUID *guid;
+       const char *name;
+    } guids[] = {
+#define FE(x) { &x, #x}
+       FE(GUID_XAxis),
+       FE(GUID_YAxis),
+       FE(GUID_ZAxis),
+       FE(GUID_RxAxis),
+       FE(GUID_RyAxis),
+       FE(GUID_RzAxis),
+       FE(GUID_Slider),
+       FE(GUID_Button),
+       FE(GUID_Key),
+       FE(GUID_POV),
+       FE(GUID_Unknown),
+       FE(GUID_SysMouse),
+       FE(GUID_SysKeyboard),
+       FE(GUID_Joystick),
+       FE(GUID_ConstantForce),
+       FE(GUID_RampForce),
+       FE(GUID_Square),
+       FE(GUID_Sine),
+       FE(GUID_Triangle),
+       FE(GUID_SawtoothUp),
+       FE(GUID_SawtoothDown),
+       FE(GUID_Spring),
+       FE(GUID_Damper),
+       FE(GUID_Inertia),
+       FE(GUID_Friction),
+       FE(GUID_CustomForce)
+#undef FE
+    };
+    if (guid == NULL)
+       return "null GUID";
+    for (i = 0; i < (sizeof(guids) / sizeof(guids[0])); i++) {
+       if (IsEqualGUID(guids[i].guid, guid)) {
+           return guids[i].name;
+       }
+    }
+    return "Unknown GUID";
+}
+
+void _dump_DIDATAFORMAT(const DIDATAFORMAT *df) {
+    unsigned int i;
+
+    TRACE("Dumping DIDATAFORMAT structure:\n");
+    TRACE("  - dwSize: %ld\n", df->dwSize);
+    if (df->dwSize != sizeof(DIDATAFORMAT)) {
+       WARN("Non-standard DIDATAFORMAT structure size (%ld instead of %d).\n", df->dwSize, sizeof(DIDATAFORMAT));
+    }
+    TRACE("  - dwObjsize: %ld\n", df->dwObjSize);
+    if (df->dwObjSize != sizeof(DIOBJECTDATAFORMAT)) {
+       WARN("Non-standard DIOBJECTDATAFORMAT structure size (%ld instead of %d).\n", df->dwObjSize, sizeof(DIOBJECTDATAFORMAT));
+    }
+    TRACE("  - dwFlags: 0x%08lx (", df->dwFlags);
+    switch (df->dwFlags) {
+        case DIDF_ABSAXIS: TRACE("DIDF_ABSAXIS"); break;
+       case DIDF_RELAXIS: TRACE("DIDF_RELAXIS"); break;
+       default: TRACE("unknown"); break;
+    }
+    TRACE(")\n");
+    TRACE("  - dwDataSize: %ld\n", df->dwDataSize);
+    TRACE("  - dwNumObjs: %ld\n", df->dwNumObjs);
+    
+    for (i = 0; i < df->dwNumObjs; i++) {
+       TRACE("  - Object %d:\n", i);
+       TRACE("      * GUID: %s ('%s')\n", debugstr_guid(df->rgodf[i].pguid), _dump_dinput_GUID(df->rgodf[i].pguid));
+       TRACE("      * dwOfs: %ld\n", df->rgodf[i].dwOfs);
+       TRACE("      * dwType: 0x%08lx\n", df->rgodf[i].dwType);
+       TRACE("        "); _dump_EnumObjects_flags(df->rgodf[i].dwType); TRACE("\n");
+       TRACE("      * dwFlags: 0x%08lx\n", df->rgodf[i].dwFlags);
+    }
+}
+
+/* Conversion between internal data buffer and external data buffer */
+void fill_DataFormat(void *out, const void *in, DataFormat *df) {
+    int i;
+    char *in_c = (char *) in;
+    char *out_c = (char *) out;
+    
+    if (df->dt == NULL) {
+       /* This means that the app uses Wine's internal data format */
+       memcpy(out, in, df->internal_format_size);
+    } else {
+       for (i = 0; i < df->size; i++) {
+           if (df->dt[i].offset_in >= 0) {
+               switch (df->dt[i].size) {
+                   case 1:
+                       TRACE("Copying (c) to %d from %d (value %d)\n",
+                             df->dt[i].offset_out, df->dt[i].offset_in, *((char *) (in_c + df->dt[i].offset_in)));
+                       *((char *) (out_c + df->dt[i].offset_out)) = *((char *) (in_c + df->dt[i].offset_in));
+                       break;
+                   
+                   case 2:
+                       TRACE("Copying (s) to %d from %d (value %d)\n",
+                             df->dt[i].offset_out, df->dt[i].offset_in, *((short *) (in_c + df->dt[i].offset_in)));
+                       *((short *) (out_c + df->dt[i].offset_out)) = *((short *) (in_c + df->dt[i].offset_in));
+                       break;
+                   
+                   case 4:
+                       TRACE("Copying (i) to %d from %d (value %d)\n",
+                             df->dt[i].offset_out, df->dt[i].offset_in, *((int *) (in_c + df->dt[i].offset_in)));
+                       *((int *) (out_c + df->dt[i].offset_out)) = *((int *) (in_c + df->dt[i].offset_in));
+                       break;
+                   
+                   default:
+                       memcpy((out_c + df->dt[i].offset_out), (in_c + df->dt[i].offset_in), df->dt[i].size);
+                       break;
+               }
+           } else {
+               switch (df->dt[i].size) {
+                   case 1:
+                       TRACE("Copying (c) to %d default value %d\n",
+                             df->dt[i].offset_out, df->dt[i].value);
+                       *((char *) (out_c + df->dt[i].offset_out)) = (char) df->dt[i].value;
+                       break;
+                       
+                   case 2:
+                       TRACE("Copying (s) to %d default value %d\n",
+                             df->dt[i].offset_out, df->dt[i].value);
+                       *((short *) (out_c + df->dt[i].offset_out)) = (short) df->dt[i].value;
+                       break;
+                       
+                   case 4:
+                       TRACE("Copying (i) to %d default value %d\n",
+                             df->dt[i].offset_out, df->dt[i].value);
+                       *((int *) (out_c + df->dt[i].offset_out)) = (int) df->dt[i].value;
+                       break;
+                       
+                   default:
+                       memset((out_c + df->dt[i].offset_out), df->dt[i].size, 0);
+                       break;
+               }
+           }
+       }
+    }
+}
+
+void release_DataFormat(DataFormat * format)
+{
+    TRACE("Deleting DataTransform : \n");
+
+    HeapFree(GetProcessHeap(), 0, format->dt);
+}
+
+DataFormat *create_DataFormat(const DIDATAFORMAT *wine_format, LPCDIDATAFORMAT asked_format, int *offset) {
+    DataFormat *ret;
+    DataTransform *dt;
+    unsigned int i, j;
+    int same = 1;
+    int *done;
+    int index = 0;
+    DWORD next = 0;
+    
+    ret = (DataFormat *) HeapAlloc(GetProcessHeap(), 0, sizeof(DataFormat));
+    
+    done = (int *) HeapAlloc(GetProcessHeap(), 0, sizeof(int) * asked_format->dwNumObjs);
+    memset(done, 0, sizeof(int) * asked_format->dwNumObjs);
+    
+    dt = (DataTransform *) HeapAlloc(GetProcessHeap(), 0, asked_format->dwNumObjs * sizeof(DataTransform));
+    
+    TRACE("Creating DataTransform : \n");
+    
+    for (i = 0; i < wine_format->dwNumObjs; i++) {
+       offset[i] = -1;
+       
+       for (j = 0; j < asked_format->dwNumObjs; j++) {
+           if (done[j] == 1)
+               continue;
+           
+           if (/* Check if the application either requests any GUID and if not, it if matches
+                * the GUID of the Wine object.
+                */
+               ((asked_format->rgodf[j].pguid == NULL) ||
+                (IsEqualGUID(wine_format->rgodf[i].pguid, asked_format->rgodf[j].pguid)))
+               &&
+               (/* Then check if it accepts any instance id, and if not, if it matches Wine's
+                 * instance id.
+                 */
+                ((asked_format->rgodf[j].dwType & 0x00FFFF00) == DIDFT_ANYINSTANCE) ||
+                (wine_format->rgodf[i].dwType & asked_format->rgodf[j].dwType))) {
+               
+               done[j] = 1;
+               
+               TRACE("Matching : \n");
+               TRACE("   - Asked (%d) :\n", j);
+               TRACE("       * GUID: %s ('%s')\n",
+                     debugstr_guid(asked_format->rgodf[j].pguid),
+                     _dump_dinput_GUID(asked_format->rgodf[j].pguid));
+               TRACE("       * Offset: %3ld\n", asked_format->rgodf[j].dwOfs);
+               TRACE("       * dwType: %08lx\n", asked_format->rgodf[j].dwType);
+               TRACE("         "); _dump_EnumObjects_flags(asked_format->rgodf[j].dwType); TRACE("\n");
+               
+               TRACE("   - Wine  (%d) :\n", j);
+               TRACE("       * GUID: %s ('%s')\n",
+                     debugstr_guid(wine_format->rgodf[j].pguid),
+                     _dump_dinput_GUID(wine_format->rgodf[j].pguid));
+               TRACE("       * Offset: %3ld\n", wine_format->rgodf[j].dwOfs);
+               TRACE("       * dwType: %08lx\n", wine_format->rgodf[j].dwType);
+               TRACE("         "); _dump_EnumObjects_flags(wine_format->rgodf[j].dwType); TRACE("\n");
+               
+               if (wine_format->rgodf[i].dwType & DIDFT_BUTTON)
+                   dt[index].size = sizeof(BYTE);
+               else
+                   dt[index].size = sizeof(DWORD);
+               dt[index].offset_in  = wine_format ->rgodf[i].dwOfs;
+                if (asked_format->rgodf[j].dwOfs < next) {
+                    WARN("bad format: dwOfs=%ld, changing to %ld\n", asked_format->rgodf[j].dwOfs, next);
+                   dt[index].offset_out = next;
+                   offset[i] = next;
+                } else {
+                   dt[index].offset_out = asked_format->rgodf[j].dwOfs;
+                    offset[i] = asked_format->rgodf[j].dwOfs;
+                }
+               dt[index].value = 0;
+                next = next + dt[index].size;
+               
+               if (wine_format->rgodf[i].dwOfs != dt[index].offset_out)
+                   same = 0;
+               
+               index++;
+               break;
+           }
+       }
+       
+       if (j == asked_format->dwNumObjs)
+           same = 0;
+    }
+    
+    TRACE("Setting to default value :\n");
+    for (j = 0; j < asked_format->dwNumObjs; j++) {
+       if (done[j] == 0) {
+           TRACE("   - Asked (%d) :\n", j);
+           TRACE("       * GUID: %s ('%s')\n",
+                 debugstr_guid(asked_format->rgodf[j].pguid),
+                 _dump_dinput_GUID(asked_format->rgodf[j].pguid));
+           TRACE("       * Offset: %3ld\n", asked_format->rgodf[j].dwOfs);
+           TRACE("       * dwType: %08lx\n", asked_format->rgodf[j].dwType);
+           TRACE("         "); _dump_EnumObjects_flags(asked_format->rgodf[j].dwType); TRACE("\n");
+           
+           if (asked_format->rgodf[j].dwType & DIDFT_BUTTON)
+               dt[index].size = sizeof(BYTE);
+           else
+               dt[index].size = sizeof(DWORD);
+           dt[index].offset_in  = -1;
+           dt[index].offset_out = asked_format->rgodf[j].dwOfs;
+           dt[index].value = 0;
+           index++;
+           
+           same = 0;
+       }
+    }
+    
+    ret->internal_format_size = wine_format->dwDataSize;
+    ret->size = index;
+    if (same) {
+       ret->dt = NULL;
+       HeapFree(GetProcessHeap(), 0, dt);
+    } else {
+       ret->dt = dt;
+    }
+    
+    HeapFree(GetProcessHeap(), 0, done);
+    
+    return ret;
+}
+
+BOOL DIEnumDevicesCallbackAtoW(LPCDIDEVICEOBJECTINSTANCEA lpddi, LPVOID lpvRef) {
+    DIDEVICEOBJECTINSTANCEW ddtmp;
+    device_enumobjects_AtoWcb_data* data;
+
+    data = (device_enumobjects_AtoWcb_data*) lpvRef;
+    
+    memset(&ddtmp, 0, sizeof(DIDEVICEINSTANCEW)); 
+    
+    ddtmp.dwSize = sizeof(DIDEVICEINSTANCEW);
+    ddtmp.guidType     = lpddi->guidType;
+    ddtmp.dwOfs        = lpddi->dwOfs;
+    ddtmp.dwType       = lpddi->dwType;
+    ddtmp.dwFlags      = lpddi->dwFlags;
+    MultiByteToWideChar(CP_ACP, 0, lpddi->tszName, -1, ddtmp.tszName, MAX_PATH);
+    
+    if (lpddi->dwSize == sizeof(DIDEVICEINSTANCEA)) {
+       /**
+        * if dwSize < sizeof(DIDEVICEINSTANCEA of DInput version >= 5)
+        *  force feedback and other newer datas aren't available
+        */
+       ddtmp.dwFFMaxForce        = lpddi->dwFFMaxForce;
+       ddtmp.dwFFForceResolution = lpddi->dwFFForceResolution;
+       ddtmp.wCollectionNumber   = lpddi->wCollectionNumber;
+       ddtmp.wDesignatorIndex    = lpddi->wDesignatorIndex;
+       ddtmp.wUsagePage          = lpddi->wUsagePage;
+       ddtmp.wUsage              = lpddi->wUsage;
+       ddtmp.dwDimension         = lpddi->dwDimension;
+       ddtmp.wExponent           = lpddi->wExponent;
+       ddtmp.wReserved           = lpddi->wReserved;
+    }
+    return data->lpCallBack(&ddtmp, data->lpvRef);
+}
+
+/******************************************************************************
+ *     IDirectInputDeviceA
+ */
+
+HRESULT WINAPI IDirectInputDevice2AImpl_SetDataFormat(
+       LPDIRECTINPUTDEVICE8A iface,LPCDIDATAFORMAT df
+) {
+    IDirectInputDevice2AImpl *This = (IDirectInputDevice2AImpl *)iface;
+    
+    TRACE("(this=%p,%p)\n",This,df);
+    
+    _dump_DIDATAFORMAT(df);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_SetCooperativeLevel(
+       LPDIRECTINPUTDEVICE8A iface,HWND hwnd,DWORD dwflags
+) {
+    IDirectInputDevice2AImpl *This = (IDirectInputDevice2AImpl *)iface;
+    TRACE("(this=%p,0x%08lx,0x%08lx)\n",This,(DWORD)hwnd,dwflags);
+    if (TRACE_ON(dinput)) {
+       TRACE(" cooperative level : ");
+       _dump_cooperativelevel_DI(dwflags);
+    }
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_SetEventNotification(
+       LPDIRECTINPUTDEVICE8A iface,HANDLE hnd
+) {
+    IDirectInputDevice2AImpl *This = (IDirectInputDevice2AImpl *)iface;
+    FIXME("(this=%p,0x%08lx): stub\n",This,(DWORD)hnd);
+    return DI_OK;
+}
+
+ULONG WINAPI IDirectInputDevice2AImpl_Release(LPDIRECTINPUTDEVICE8A iface)
+{
+    IDirectInputDevice2AImpl *This = (IDirectInputDevice2AImpl *)iface;
+    ULONG ref;
+    ref = InterlockedDecrement(&(This->ref));
+    if (ref == 0)
+        HeapFree(GetProcessHeap(),0,This);
+    return ref;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_QueryInterface(
+       LPDIRECTINPUTDEVICE8A iface,REFIID riid,LPVOID *ppobj
+)
+{
+    IDirectInputDevice2AImpl *This = (IDirectInputDevice2AImpl *)iface;
+    
+    TRACE("(this=%p,%s,%p)\n",This,debugstr_guid(riid),ppobj);
+    if (IsEqualGUID(&IID_IUnknown,riid)) {
+       IDirectInputDevice2_AddRef(iface);
+       *ppobj = This;
+       return DI_OK;
+    }
+    if (IsEqualGUID(&IID_IDirectInputDeviceA,riid)) {
+       IDirectInputDevice2_AddRef(iface);
+       *ppobj = This;
+       return DI_OK;
+    }
+    if (IsEqualGUID(&IID_IDirectInputDevice2A,riid)) {
+       IDirectInputDevice2_AddRef(iface);
+       *ppobj = This;
+       return DI_OK;
+    }
+    if (IsEqualGUID(&IID_IDirectInputDevice7A,riid)) {
+       IDirectInputDevice7_AddRef(iface);
+       *ppobj = This;
+       return DI_OK;
+    }
+    TRACE("Unsupported interface !\n");
+    return E_FAIL;
+}
+
+HRESULT WINAPI IDirectInputDevice2WImpl_QueryInterface(
+       LPDIRECTINPUTDEVICE8W iface,REFIID riid,LPVOID *ppobj
+)
+{
+    IDirectInputDevice2AImpl *This = (IDirectInputDevice2AImpl *)iface;
+    
+    TRACE("(this=%p,%s,%p)\n",This,debugstr_guid(riid),ppobj);
+    if (IsEqualGUID(&IID_IUnknown,riid)) {
+       IDirectInputDevice2_AddRef(iface);
+       *ppobj = This;
+       return DI_OK;
+    }
+    if (IsEqualGUID(&IID_IDirectInputDeviceW,riid)) {
+       IDirectInputDevice2_AddRef(iface);
+       *ppobj = This;
+       return DI_OK;
+    }
+    if (IsEqualGUID(&IID_IDirectInputDevice2W,riid)) {
+       IDirectInputDevice2_AddRef(iface);
+       *ppobj = This;
+       return DI_OK;
+    }
+    if (IsEqualGUID(&IID_IDirectInputDevice7W,riid)) {
+       IDirectInputDevice7_AddRef(iface);
+       *ppobj = This;
+       return DI_OK;
+    }
+    TRACE("Unsupported interface !\n");
+    return E_FAIL;
+}
+
+ULONG WINAPI IDirectInputDevice2AImpl_AddRef(
+       LPDIRECTINPUTDEVICE8A iface)
+{
+    IDirectInputDevice2AImpl *This = (IDirectInputDevice2AImpl *)iface;
+    return InterlockedIncrement(&(This->ref));
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_EnumObjects(
+       LPDIRECTINPUTDEVICE8A iface,
+       LPDIENUMDEVICEOBJECTSCALLBACKA lpCallback,
+       LPVOID lpvRef,
+       DWORD dwFlags)
+{
+    FIXME("(this=%p,%p,%p,%08lx): stub!\n", iface, lpCallback, lpvRef, dwFlags);
+    if (TRACE_ON(dinput)) {
+       DPRINTF("  - flags = ");
+       _dump_EnumObjects_flags(dwFlags);
+       DPRINTF("\n");
+    }
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2WImpl_EnumObjects(
+       LPDIRECTINPUTDEVICE8W iface,
+       LPDIENUMDEVICEOBJECTSCALLBACKW lpCallback,
+       LPVOID lpvRef,
+       DWORD dwFlags)
+{
+    FIXME("(this=%p,%p,%p,%08lx): stub!\n", iface, lpCallback, lpvRef, dwFlags);
+    if (TRACE_ON(dinput)) {
+       DPRINTF("  - flags = ");
+       _dump_EnumObjects_flags(dwFlags);
+       DPRINTF("\n");
+    }
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_GetProperty(
+       LPDIRECTINPUTDEVICE8A iface,
+       REFGUID rguid,
+       LPDIPROPHEADER pdiph)
+{
+    FIXME("(this=%p,%s,%p): stub!\n",
+         iface, debugstr_guid(rguid), pdiph);
+    
+    if (TRACE_ON(dinput))
+       _dump_DIPROPHEADER(pdiph);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_GetObjectInfo(
+       LPDIRECTINPUTDEVICE8A iface,
+       LPDIDEVICEOBJECTINSTANCEA pdidoi,
+       DWORD dwObj,
+       DWORD dwHow)
+{
+    FIXME("(this=%p,%p,%ld,0x%08lx): stub!\n",
+         iface, pdidoi, dwObj, dwHow);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2WImpl_GetObjectInfo(
+       LPDIRECTINPUTDEVICE8W iface,
+       LPDIDEVICEOBJECTINSTANCEW pdidoi,
+       DWORD dwObj,
+       DWORD dwHow)
+{
+    FIXME("(this=%p,%p,%ld,0x%08lx): stub!\n",
+         iface, pdidoi, dwObj, dwHow);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_GetDeviceInfo(
+       LPDIRECTINPUTDEVICE8A iface,
+       LPDIDEVICEINSTANCEA pdidi)
+{
+    FIXME("(this=%p,%p): stub!\n",
+         iface, pdidi);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2WImpl_GetDeviceInfo(
+       LPDIRECTINPUTDEVICE8W iface,
+       LPDIDEVICEINSTANCEW pdidi)
+{
+    FIXME("(this=%p,%p): stub!\n",
+         iface, pdidi);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_RunControlPanel(
+       LPDIRECTINPUTDEVICE8A iface,
+       HWND hwndOwner,
+       DWORD dwFlags)
+{
+    FIXME("(this=%p,%p,0x%08lx): stub!\n",
+         iface, hwndOwner, dwFlags);
+
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_Initialize(
+       LPDIRECTINPUTDEVICE8A iface,
+       HINSTANCE hinst,
+       DWORD dwVersion,
+       REFGUID rguid)
+{
+    FIXME("(this=%p,%p,%ld,%s): stub!\n",
+         iface, hinst, dwVersion, debugstr_guid(rguid));
+    return DI_OK;
+}
+
+/******************************************************************************
+ *     IDirectInputDevice2A
+ */
+
+HRESULT WINAPI IDirectInputDevice2AImpl_CreateEffect(
+       LPDIRECTINPUTDEVICE8A iface,
+       REFGUID rguid,
+       LPCDIEFFECT lpeff,
+       LPDIRECTINPUTEFFECT *ppdef,
+       LPUNKNOWN pUnkOuter)
+{
+    FIXME("(this=%p,%s,%p,%p,%p): stub!\n",
+         iface, debugstr_guid(rguid), lpeff, ppdef, pUnkOuter);
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_EnumEffects(
+       LPDIRECTINPUTDEVICE8A iface,
+       LPDIENUMEFFECTSCALLBACKA lpCallback,
+       LPVOID lpvRef,
+       DWORD dwFlags)
+{
+    FIXME("(this=%p,%p,%p,0x%08lx): stub!\n",
+         iface, lpCallback, lpvRef, dwFlags);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2WImpl_EnumEffects(
+       LPDIRECTINPUTDEVICE8W iface,
+       LPDIENUMEFFECTSCALLBACKW lpCallback,
+       LPVOID lpvRef,
+       DWORD dwFlags)
+{
+    FIXME("(this=%p,%p,%p,0x%08lx): stub!\n",
+         iface, lpCallback, lpvRef, dwFlags);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_GetEffectInfo(
+       LPDIRECTINPUTDEVICE8A iface,
+       LPDIEFFECTINFOA lpdei,
+       REFGUID rguid)
+{
+    FIXME("(this=%p,%p,%s): stub!\n",
+         iface, lpdei, debugstr_guid(rguid));
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2WImpl_GetEffectInfo(
+       LPDIRECTINPUTDEVICE8W iface,
+       LPDIEFFECTINFOW lpdei,
+       REFGUID rguid)
+{
+    FIXME("(this=%p,%p,%s): stub!\n",
+         iface, lpdei, debugstr_guid(rguid));
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_GetForceFeedbackState(
+       LPDIRECTINPUTDEVICE8A iface,
+       LPDWORD pdwOut)
+{
+    FIXME("(this=%p,%p): stub!\n",
+         iface, pdwOut);
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_SendForceFeedbackCommand(
+       LPDIRECTINPUTDEVICE8A iface,
+       DWORD dwFlags)
+{
+    FIXME("(this=%p,0x%08lx): stub!\n",
+         iface, dwFlags);
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_EnumCreatedEffectObjects(
+       LPDIRECTINPUTDEVICE8A iface,
+       LPDIENUMCREATEDEFFECTOBJECTSCALLBACK lpCallback,
+       LPVOID lpvRef,
+       DWORD dwFlags)
+{
+    FIXME("(this=%p,%p,%p,0x%08lx): stub!\n",
+         iface, lpCallback, lpvRef, dwFlags);
+    if (lpCallback)
+       lpCallback(NULL, lpvRef);
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_Escape(
+       LPDIRECTINPUTDEVICE8A iface,
+       LPDIEFFESCAPE lpDIEEsc)
+{
+    FIXME("(this=%p,%p): stub!\n",
+         iface, lpDIEEsc);
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_Poll(
+       LPDIRECTINPUTDEVICE8A iface)
+{
+    /* Because wine devices do not need to be polled, just return DI_NOEFFECT */
+    return DI_NOEFFECT;
+}
+
+HRESULT WINAPI IDirectInputDevice2AImpl_SendDeviceData(
+       LPDIRECTINPUTDEVICE8A iface,
+       DWORD cbObjectData,
+       LPCDIDEVICEOBJECTDATA rgdod,
+       LPDWORD pdwInOut,
+       DWORD dwFlags)
+{
+    FIXME("(this=%p,0x%08lx,%p,%p,0x%08lx): stub!\n",
+         iface, cbObjectData, rgdod, pdwInOut, dwFlags);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice7AImpl_EnumEffectsInFile(LPDIRECTINPUTDEVICE8A iface,
+                                                         LPCSTR lpszFileName,
+                                                         LPDIENUMEFFECTSINFILECALLBACK pec,
+                                                         LPVOID pvRef,
+                                                         DWORD dwFlags)
+{
+    FIXME("(%p)->(%s,%p,%p,%08lx): stub !\n", iface, lpszFileName, pec, pvRef, dwFlags);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice7WImpl_EnumEffectsInFile(LPDIRECTINPUTDEVICE8W iface,
+                                                         LPCWSTR lpszFileName,
+                                                         LPDIENUMEFFECTSINFILECALLBACK pec,
+                                                         LPVOID pvRef,
+                                                         DWORD dwFlags)
+{
+    FIXME("(%p)->(%s,%p,%p,%08lx): stub !\n", iface, debugstr_w(lpszFileName), pec, pvRef, dwFlags);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice7AImpl_WriteEffectToFile(LPDIRECTINPUTDEVICE8A iface,
+                                                         LPCSTR lpszFileName,
+                                                         DWORD dwEntries,
+                                                         LPDIFILEEFFECT rgDiFileEft,
+                                                         DWORD dwFlags)
+{
+    FIXME("(%p)->(%s,%08lx,%p,%08lx): stub !\n", iface, lpszFileName, dwEntries, rgDiFileEft, dwFlags);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice7WImpl_WriteEffectToFile(LPDIRECTINPUTDEVICE8W iface,
+                                                         LPCWSTR lpszFileName,
+                                                         DWORD dwEntries,
+                                                         LPDIFILEEFFECT rgDiFileEft,
+                                                         DWORD dwFlags)
+{
+    FIXME("(%p)->(%s,%08lx,%p,%08lx): stub !\n", iface, debugstr_w(lpszFileName), dwEntries, rgDiFileEft, dwFlags);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice8AImpl_BuildActionMap(LPDIRECTINPUTDEVICE8A iface,
+                                                      LPDIACTIONFORMATA lpdiaf,
+                                                      LPCSTR lpszUserName,
+                                                      DWORD dwFlags)
+{
+    FIXME("(%p)->(%p,%s,%08lx): stub !\n", iface, lpdiaf, lpszUserName, dwFlags);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice8WImpl_BuildActionMap(LPDIRECTINPUTDEVICE8W iface,
+                                                      LPDIACTIONFORMATW lpdiaf,
+                                                      LPCWSTR lpszUserName,
+                                                      DWORD dwFlags)
+{
+    FIXME("(%p)->(%p,%s,%08lx): stub !\n", iface, lpdiaf, debugstr_w(lpszUserName), dwFlags);
+  
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice8AImpl_SetActionMap(LPDIRECTINPUTDEVICE8A iface,
+                                                    LPDIACTIONFORMATA lpdiaf,
+                                                    LPCSTR lpszUserName,
+                                                    DWORD dwFlags)
+{
+    FIXME("(%p)->(%p,%s,%08lx): stub !\n", iface, lpdiaf, lpszUserName, dwFlags);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice8WImpl_SetActionMap(LPDIRECTINPUTDEVICE8W iface,
+                                                    LPDIACTIONFORMATW lpdiaf,
+                                                    LPCWSTR lpszUserName,
+                                                    DWORD dwFlags)
+{
+    FIXME("(%p)->(%p,%s,%08lx): stub !\n", iface, lpdiaf, debugstr_w(lpszUserName), dwFlags);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice8AImpl_GetImageInfo(LPDIRECTINPUTDEVICE8A iface,
+                                                    LPDIDEVICEIMAGEINFOHEADERA lpdiDevImageInfoHeader)
+{
+    FIXME("(%p)->(%p): stub !\n", iface, lpdiDevImageInfoHeader);
+    
+    return DI_OK;
+}
+
+HRESULT WINAPI IDirectInputDevice8WImpl_GetImageInfo(LPDIRECTINPUTDEVICE8W iface,
+                                                    LPDIDEVICEIMAGEINFOHEADERW lpdiDevImageInfoHeader)
+{
+    FIXME("(%p)->(%p): stub !\n", iface, lpdiDevImageInfoHeader);
+    
+    return DI_OK;
+}