VirtualBox

Changeset 50962 in vbox for trunk/src/VBox


Ignore:
Timestamp:
Apr 3, 2014 1:58:47 PM (11 years ago)
Author:
vboxsync
svn:sync-xref-src-repo-rev:
93119
Message:

Send HID usage codes rather than XT scan codes to keyboard devices, with conversion in driver. See #6026.

Location:
trunk/src/VBox
Files:
4 edited

Legend:

Unmodified
Added
Removed
  • trunk/src/VBox/Devices/Input/DrvKeyboardQueue.cpp

    r49451 r50962  
    3232*   Structures and Typedefs                                                    *
    3333*******************************************************************************/
     34
     35/** Scancode translator state.  */
     36typedef enum {
     37    SS_IDLE,    /**< Starting state. */
     38    SS_EXT,     /**< E0 byte was received. */
     39    SS_EXT1     /**< E1 byte was received. */
     40} scan_state_t;
     41
    3442/**
    3543 * Keyboard queue driver instance data.
     
    5260    /** The queue handle. */
    5361    PPDMQUEUE                   pQueue;
     62    /** State of the scancode translation. */
     63    scan_state_t                XlatState;
    5464    /** Discard input when this flag is set. */
    5565    bool                        fInactive;
     
    6777    PDMQUEUEITEMCORE    Core;
    6878    /** The keycode. */
    69     uint8_t             u8KeyCode;
     79    uint32_t            u32UsageCode;
    7080} DRVKBDQUEUEITEM, *PDRVKBDQUEUEITEM;
    7181
     82
     83/*******************************************************************************
     84*   Global Variables                                                           *
     85*******************************************************************************/
     86
     87/** Lookup table for converting PC/XT scan codes to USB HID usage codes. */
     88static const uint8_t aScancode2Hid[] =
     89{
     90    0x00, 0x29, 0x1e, 0x1f, 0x20, 0x21, 0x22, 0x23, /* 00-07 */
     91    0x24, 0x25, 0x26, 0x27, 0x2d, 0x2e, 0x2a, 0x2b, /* 08-1F */
     92    0x14, 0x1a, 0x08, 0x15, 0x17, 0x1c, 0x18, 0x0c, /* 10-17 */
     93    0x12, 0x13, 0x2f, 0x30, 0x28, 0xe0, 0x04, 0x16, /* 18-1F */
     94    0x07, 0x09, 0x0a, 0x0b, 0x0d, 0x0e, 0x0f, 0x33, /* 20-27 */
     95    0x34, 0x35, 0xe1, 0x31, 0x1d, 0x1b, 0x06, 0x19, /* 28-2F */
     96    0x05, 0x11, 0x10, 0x36, 0x37, 0x38, 0xe5, 0x55, /* 30-37 */
     97    0xe2, 0x2c, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, /* 38-3F */
     98    0x3f, 0x40, 0x41, 0x42, 0x43, 0x53, 0x47, 0x5f, /* 40-47 */
     99    0x60, 0x61, 0x56, 0x5c, 0x5d, 0x5e, 0x57, 0x59, /* 48-4F */
     100    0x5a, 0x5b, 0x62, 0x63, 0x46, 0x00, 0x64, 0x44, /* 50-57 */
     101    0x45, 0x67, 0x00, 0x00, 0x8c, 0x00, 0x00, 0x00, /* 58-5F */
     102    0x00, 0x00, 0x00, 0x00, 0x68, 0x69, 0x6a, 0x6b, /* 60-67 */
     103    0x6c, 0x6d, 0x6e, 0x6f, 0x70, 0x71, 0x72, 0x00, /* 68-6F */
     104    0x88, 0x91, 0x90, 0x87, 0x00, 0x00, 0x00, 0x00, /* 70-77 */
     105    0x00, 0x8a, 0x00, 0x8b, 0x00, 0x89, 0x85, 0x00  /* 78-7F */
     106};
     107
     108/** Lookup table for extended scancodes (arrow keys etc.). */
     109static const uint8_t aExtScan2Hid[] =
     110{
     111    0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 00-07 */
     112    0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 08-1F */
     113    0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 10-17 */
     114    0x00, 0x00, 0x00, 0x00, 0x58, 0xe4, 0x00, 0x00, /* 18-1F */
     115    0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 20-27 */
     116    0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 28-2F */
     117    0x00, 0x00, 0x00, 0x00, 0x00, 0x54, 0x00, 0x46, /* 30-37 */
     118    /* Sun-specific keys.  Most of the XT codes are made up  */
     119    0xe6, 0x00, 0x00, 0x75, 0x76, 0x77, 0xA3, 0x78, /* 38-3F */
     120    0x80, 0x81, 0x82, 0x79, 0x00, 0x00, 0x48, 0x4a, /* 40-47 */
     121    0x52, 0x4b, 0x00, 0x50, 0x00, 0x4f, 0x00, 0x4d, /* 48-4F */
     122    0x51, 0x4e, 0x49, 0x4c, 0x00, 0x00, 0x00, 0x00, /* 50-57 */
     123    0x00, 0x00, 0x00, 0xe3, 0xe7, 0x65, 0x66, 0x00, /* 58-5F */
     124    0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 60-67 */
     125    0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 68-6F */
     126    0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 70-77 */
     127    0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00  /* 78-7F */
     128};
     129
     130/**
     131 * Convert a PC scan code to a USB HID usage byte.
     132 *
     133 * @param state         Current state of the translator (scan_state_t).
     134 * @param scanCode      Incoming scan code.
     135 * @param pUsage        Pointer to usage; high bit set for key up events. The
     136 *                      contents are only valid if returned state is SS_IDLE.
     137 *
     138 * @return scan_state_t New state of the translator.
     139 */
     140static scan_state_t ScancodeToHidUsage(scan_state_t state, uint8_t scanCode, uint32_t *pUsage)
     141{
     142    uint32_t    keyUp;
     143    uint8_t     usage;
     144
     145    Assert(pUsage);
     146
     147    /* Isolate the scan code and key break flag. */
     148    keyUp = (scanCode & 0x80) << 24;
     149
     150    switch (state) {
     151    case SS_IDLE:
     152        if (scanCode == 0xE0) {
     153            state = SS_EXT;
     154        } else if (scanCode == 0xE1) {
     155            state = SS_EXT1;
     156        } else {
     157            usage = aScancode2Hid[scanCode & 0x7F];
     158            *pUsage = usage | keyUp;
     159            /* Remain in SS_IDLE state. */
     160        }
     161        break;
     162    case SS_EXT:
     163        usage = aExtScan2Hid[scanCode & 0x7F];
     164        *pUsage = usage | keyUp;
     165        state = SS_IDLE;
     166        break;
     167    case SS_EXT1:
     168        /* The sequence is E1 1D 45 E1 9D C5. We take the easy way out and remain
     169         * in the SS_EXT1 state until 45 or C5 is received.
     170         */
     171        if ((scanCode & 0x7F) == 0x45) {
     172            *pUsage = 0x48;
     173            if (scanCode == 0xC5)
     174                *pUsage |= keyUp;
     175            state = SS_IDLE;
     176        }
     177        /* Else remain in SS_EXT1 state. */
     178        break;
     179    }
     180    return state;
     181}
    72182
    73183
     
    101211 * @returns VBox status code.
    102212 * @param   pInterface          Pointer to this interface structure.
    103  * @param   u8KeyCode           The keycode to queue.
     213 * @param   u8ScanCode          The scan code to translate/queue.
    104214 * @thread  Any thread.
    105215 */
    106 static DECLCALLBACK(int) drvKbdQueuePutEvent(PPDMIKEYBOARDPORT pInterface, uint8_t u8KeyCode)
     216static DECLCALLBACK(int) drvKbdQueuePutEventScan(PPDMIKEYBOARDPORT pInterface, uint8_t u8ScanCode)
    107217{
    108218    PDRVKBDQUEUE pDrv = IKEYBOARDPORT_2_DRVKBDQUEUE(pInterface);
     
    111221        return VINF_SUCCESS;
    112222
    113     PDRVKBDQUEUEITEM pItem = (PDRVKBDQUEUEITEM)PDMQueueAlloc(pDrv->pQueue);
    114     if (pItem)
    115     {
    116         pItem->u8KeyCode = u8KeyCode;
    117         PDMQueueInsert(pDrv->pQueue, &pItem->Core);
     223    uint32_t    u32Usage = 0;
     224    pDrv->XlatState = ScancodeToHidUsage(pDrv->XlatState, u8ScanCode, &u32Usage);
     225
     226    if (pDrv->XlatState == SS_IDLE) {
     227        PDRVKBDQUEUEITEM pItem = (PDRVKBDQUEUEITEM)PDMQueueAlloc(pDrv->pQueue);
     228        if (pItem)
     229        {
     230            pItem->u32UsageCode = u32Usage;
     231            PDMQueueInsert(pDrv->pQueue, &pItem->Core);
     232            return VINF_SUCCESS;
     233        }
     234        if (!pDrv->fSuspended)
     235            AssertMsgFailed(("drvKbdQueuePutEventScan: Queue is full!!!!\n"));
     236        return VERR_PDM_NO_QUEUE_ITEMS;
     237    }
     238    else
    118239        return VINF_SUCCESS;
    119     }
    120     if (!pDrv->fSuspended)
    121         AssertMsgFailed(("drvKbdQueuePutEvent: Queue is full!!!!\n"));
    122     return VERR_PDM_NO_QUEUE_ITEMS;
    123240}
    124241
     
    170287    PDRVKBDQUEUE        pThis = PDMINS_2_DATA(pDrvIns, PDRVKBDQUEUE);
    171288    PDRVKBDQUEUEITEM    pItem = (PDRVKBDQUEUEITEM)pItemCore;
    172     int rc = pThis->pUpPort->pfnPutEvent(pThis->pUpPort, pItem->u8KeyCode);
     289    int rc = pThis->pUpPort->pfnPutEventHid(pThis->pUpPort, pItem->u32UsageCode);
    173290    return RT_SUCCESS(rc);
    174291}
     
    264381    pDrv->fInactive                         = true;
    265382    pDrv->fSuspended                        = false;
     383    pDrv->XlatState                         = SS_IDLE;
    266384    /* IBase. */
    267385    pDrvIns->IBase.pfnQueryInterface        = drvKbdQueueQueryInterface;
     
    270388    pDrv->IConnector.pfnSetActive           = drvKbdPassThruSetActive;
    271389    /* IKeyboardPort. */
    272     pDrv->IPort.pfnPutEvent                 = drvKbdQueuePutEvent;
     390    pDrv->IPort.pfnPutEventScan             = drvKbdQueuePutEventScan;
    273391
    274392    /*
  • trunk/src/VBox/Devices/Input/PS2K.cpp

    r49469 r50962  
    462462*******************************************************************************/
    463463
    464 /*
    465  * Because of historical reasons and poor design, VirtualBox internally uses BIOS
    466  * PC/XT style scan codes to represent keyboard events. Each key press and release is
    467  * represented as a stream of bytes, typically only one byte but up to four-byte
    468  * sequences are possible. In the typical case, the GUI front end generates the stream
    469  * of scan codes which we need to translate back to a single up/down event.
    470  *
    471  * This function could possibly live somewhere else.
    472  */
    473 
    474 /** Lookup table for converting PC/XT scan codes to USB HID usage codes. */
    475 static uint8_t aScancode2Hid[] =
    476 {
    477     0x00, 0x29, 0x1e, 0x1f, 0x20, 0x21, 0x22, 0x23, /* 00-07 */
    478     0x24, 0x25, 0x26, 0x27, 0x2d, 0x2e, 0x2a, 0x2b, /* 08-1F */
    479     0x14, 0x1a, 0x08, 0x15, 0x17, 0x1c, 0x18, 0x0c, /* 10-17 */
    480     0x12, 0x13, 0x2f, 0x30, 0x28, 0xe0, 0x04, 0x16, /* 18-1F */
    481     0x07, 0x09, 0x0a, 0x0b, 0x0d, 0x0e, 0x0f, 0x33, /* 20-27 */
    482     0x34, 0x35, 0xe1, 0x31, 0x1d, 0x1b, 0x06, 0x19, /* 28-2F */
    483     0x05, 0x11, 0x10, 0x36, 0x37, 0x38, 0xe5, 0x55, /* 30-37 */
    484     0xe2, 0x2c, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, /* 38-3F */
    485     0x3f, 0x40, 0x41, 0x42, 0x43, 0x53, 0x47, 0x5f, /* 40-47 */
    486     0x60, 0x61, 0x56, 0x5c, 0x5d, 0x5e, 0x57, 0x59, /* 48-4F */
    487     0x5a, 0x5b, 0x62, 0x63, 0x46, 0x00, 0x64, 0x44, /* 50-57 */
    488     0x45, 0x67, 0x00, 0x00, 0x8c, 0x00, 0x00, 0x00, /* 58-5F */
    489     0x00, 0x00, 0x00, 0x00, 0x68, 0x69, 0x6a, 0x6b, /* 60-67 */
    490     0x6c, 0x6d, 0x6e, 0x6f, 0x70, 0x71, 0x72, 0x00, /* 68-6F */
    491     0x88, 0x91, 0x90, 0x87, 0x00, 0x00, 0x00, 0x00, /* 70-77 */
    492     0x00, 0x8a, 0x00, 0x8b, 0x00, 0x89, 0x85, 0x00  /* 78-7F */
    493 };
    494 
    495 /** Lookup table for extended scancodes (arrow keys etc.). */
    496 static uint8_t aExtScan2Hid[] =
    497 {
    498     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 00-07 */
    499     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 08-1F */
    500     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 10-17 */
    501     0x00, 0x00, 0x00, 0x00, 0x58, 0xe4, 0x00, 0x00, /* 18-1F */
    502     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 20-27 */
    503     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 28-2F */
    504     0x00, 0x00, 0x00, 0x00, 0x00, 0x54, 0x00, 0x46, /* 30-37 */
    505     /* Sun-specific keys.  Most of the XT codes are made up  */
    506     0xe6, 0x00, 0x00, 0x75, 0x76, 0x77, 0xA3, 0x78, /* 38-3F */
    507     0x80, 0x81, 0x82, 0x79, 0x00, 0x00, 0x48, 0x4a, /* 40-47 */
    508     0x52, 0x4b, 0x00, 0x50, 0x00, 0x4f, 0x00, 0x4d, /* 48-4F */
    509     0x51, 0x4e, 0x49, 0x4c, 0x00, 0x00, 0x00, 0x00, /* 50-57 */
    510     0x00, 0x00, 0x00, 0xe3, 0xe7, 0x65, 0x66, 0x00, /* 58-5F */
    511     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 60-67 */
    512     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 68-6F */
    513     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 70-77 */
    514     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00  /* 78-7F */
    515 };
    516 
    517 /**
    518  * Convert a PC scan code to a USB HID usage byte.
    519  *
    520  * @param state         Current state of the translator (scan_state_t).
    521  * @param scanCode      Incoming scan code.
    522  * @param pUsage        Pointer to usage; high bit set for key up events. The
    523  *                      contents are only valid if returned state is SS_IDLE.
    524  *
    525  * @return scan_state_t New state of the translator.
    526  */
    527 static scan_state_t ScancodeToHidUsage(scan_state_t state, uint8_t scanCode, uint32_t *pUsage)
    528 {
    529     uint32_t    keyUp;
    530     uint8_t     usage;
    531 
    532     Assert(pUsage);
    533 
    534     /* Isolate the scan code and key break flag. */
    535     keyUp = (scanCode & 0x80) << 24;
    536 
    537     switch (state) {
    538     case SS_IDLE:
    539         if (scanCode == 0xE0) {
    540             state = SS_EXT;
    541         } else if (scanCode == 0xE1) {
    542             state = SS_EXT1;
    543         } else {
    544             usage = aScancode2Hid[scanCode & 0x7F];
    545             *pUsage = usage | keyUp;
    546             /* Remain in SS_IDLE state. */
    547         }
    548         break;
    549     case SS_EXT:
    550         usage = aExtScan2Hid[scanCode & 0x7F];
    551         *pUsage = usage | keyUp;
    552         state = SS_IDLE;
    553         break;
    554     case SS_EXT1:
    555         /* The sequence is E1 1D 45 E1 9D C5. We take the easy way out and remain
    556          * in the SS_EXT1 state until 45 or C5 is received.
    557          */
    558         if ((scanCode & 0x7F) == 0x45) {
    559             *pUsage = 0x48;
    560             if (scanCode == 0xC5)
    561                 *pUsage |= keyUp;
    562             state = SS_IDLE;
    563         }
    564         /* Else remain in SS_EXT1 state. */
    565         break;
    566     }
    567     return state;
    568 }
    569 
    570464/*******************************************************************************
    571465*   Internal Functions                                                         *
     
    884778#ifdef IN_RING3
    885779
    886 static int psk2ProcessKeyEvent(PPS2K pThis, uint8_t u8HidCode, bool fKeyDown)
     780static int ps2kProcessKeyEvent(PPS2K pThis, uint8_t u8HidCode, bool fKeyDown)
    887781{
    888782    unsigned int    i = 0;
     
    11161010        if (pThis->enmTypematicState == KBD_TMS_REPEAT)
    11171011        {
    1118             psk2ProcessKeyEvent(pThis, pThis->u8TypematicKey, true /* Key down */ );
     1012            ps2kProcessKeyEvent(pThis, pThis->u8TypematicKey, true /* Key down */ );
    11191013            TMTimerSetMillies(pThis->CTX_SUFF(pKbdTypematicTimer), pThis->uTypematicRepeat);
    11201014        }
     
    11481042{
    11491043    LogFlowFunc(("Releasing keys...\n"));
    1150     LogRel(("Releasing keys...\n"));
    11511044
    11521045    for (unsigned uKey = 0; uKey < sizeof(pThis->abDepressedKeys); ++uKey)
    11531046        if (pThis->abDepressedKeys[uKey])
    11541047        {
    1155             LogRel(("Releasing key %02X\n", uKey));
    1156             psk2ProcessKeyEvent(pThis, uKey, false /* key up */);
     1048            ps2kProcessKeyEvent(pThis, uKey, false /* key up */);
    11571049            pThis->abDepressedKeys[uKey] = 0;
    11581050        }
     
    12471139        AssertReleaseRC(rc);
    12481140
    1249         rc = psk2ProcessKeyEvent(pThis, u8HidCode, fKeyDown);
     1141        rc = ps2kProcessKeyEvent(pThis, u8HidCode, fKeyDown);
    12501142
    12511143        PDMCritSectLeave(pThis->pCritSectR3);
     
    12551147}
    12561148
    1257 static DECLCALLBACK(int) ps2kPutEventWrapper(PPDMIKEYBOARDPORT pInterface, uint8_t u8KeyCode)
     1149static DECLCALLBACK(int) ps2kPutEventWrapper(PPDMIKEYBOARDPORT pInterface, uint32_t u32UsageCode)
    12581150{
    12591151    PPS2K       pThis = RT_FROM_MEMBER(pInterface, PS2K, Keyboard.IPort);
    1260     uint32_t    u32Usage = 0;
    12611152    int         rc;
    12621153
    1263     LogFlowFunc(("key code %02X\n", u8KeyCode));
     1154    LogFlowFunc(("key code %08X\n", u32UsageCode));
     1155
     1156    rc = PDMCritSectEnter(pThis->pCritSectR3, VERR_SEM_BUSY);
     1157    AssertReleaseRC(rc);
    12641158
    12651159    /* The 'BAT fail' scancode is reused as a signal to release keys. No actual
    12661160     * key is allowed to use this scancode.
    12671161     */
    1268     if (RT_UNLIKELY(u8KeyCode == KRSP_BAT_FAIL))
    1269     {
    1270         rc = PDMCritSectEnter(pThis->pCritSectR3, VERR_SEM_BUSY);
    1271         AssertReleaseRC(rc);
    1272 
     1162    if (RT_UNLIKELY(u32UsageCode == KRSP_BAT_FAIL))
     1163    {
    12731164        ps2kReleaseKeys(pThis);
    1274 
    1275         PDMCritSectLeave(pThis->pCritSectR3);
    12761165    }
    12771166    else
    12781167    {
    1279         pThis->XlatState = ScancodeToHidUsage(pThis->XlatState, u8KeyCode, &u32Usage);
    1280 
    1281         if (pThis->XlatState == SS_IDLE)
    1282         {
    1283             /* Stupid Korean key hack: convert a lone break key into a press/release sequence. */
    1284             if (u32Usage == 0x80000090 || u32Usage == 0x80000091)
    1285                 ps2kPutEventWorker(pThis, u32Usage & ~0x80000000);
    1286 
    1287             ps2kPutEventWorker(pThis, u32Usage);
    1288         }
    1289     }
     1168        /* Stupid Korean key hack: convert a lone break key into a press/release sequence. */
     1169        if (u32UsageCode == 0x80000090 || u32UsageCode == 0x80000091)
     1170            ps2kPutEventWorker(pThis, u32UsageCode & ~0x80000000);
     1171
     1172        ps2kPutEventWorker(pThis, u32UsageCode);
     1173    }
     1174
     1175    PDMCritSectLeave(pThis->pCritSectR3);
    12901176
    12911177    return VINF_SUCCESS;
     
    15001386
    15011387    pThis->Keyboard.IBase.pfnQueryInterface = ps2kQueryInterface;
    1502     pThis->Keyboard.IPort.pfnPutEvent       = ps2kPutEventWrapper;
     1388    pThis->Keyboard.IPort.pfnPutEventHid    = ps2kPutEventWrapper;
    15031389
    15041390    /*
  • trunk/src/VBox/Devices/Input/UsbKbd.cpp

    r49814 r50962  
    133133} USBHIDK_REPORT, *PUSBHIDK_REPORT;
    134134
    135 /** Scancode translator state.  */
    136 typedef enum {
    137     SS_IDLE,    /**< Starting state. */
    138     SS_EXT,     /**< E0 byte was received. */
    139     SS_EXT1     /**< E1 byte was received. */
    140 } scan_state_t;
    141 
    142135/**
    143136 * The USB HID instance data.
     
    160153    /** The state of the HID (state machine).*/
    161154    USBHIDREQSTATE      enmState;
    162 
    163     /** State of the scancode translation. */
    164     scan_state_t        XlatState;
    165155
    166156    /** Pending to-host queue.
     
    358348};
    359349
    360 
    361 /*
    362  * Because of historical reasons and poor design, VirtualBox internally uses BIOS
    363  * PC/XT style scan codes to represent keyboard events. Each key press and release is
    364  * represented as a stream of bytes, typically only one byte but up to four-byte
    365  * sequences are possible. In the typical case, the GUI front end generates the stream
    366  * of scan codes which we need to translate back to a single up/down event.
    367  *
    368  * This function could possibly live somewhere else.
    369  */
    370 
    371 /** Lookup table for converting PC/XT scan codes to USB HID usage codes. */
    372 /** We map the scan codes for F13 to F23 to the usage codes for Sun keyboard
    373  *  left-hand side function keys rather than to the standard F13 to F23 usage
    374  *  codes, since we suspect that there are more people wanting Sun keyboard
    375  *  emulation than emulation of other keyboards with extended function keys. */
    376 static uint8_t aScancode2Hid[] =
    377 {
    378     0x00, 0x29, 0x1e, 0x1f, 0x20, 0x21, 0x22, 0x23, /* 00-07 */
    379     0x24, 0x25, 0x26, 0x27, 0x2d, 0x2e, 0x2a, 0x2b, /* 08-1F */
    380     0x14, 0x1a, 0x08, 0x15, 0x17, 0x1c, 0x18, 0x0c, /* 10-17 */
    381     0x12, 0x13, 0x2f, 0x30, 0x28, 0xe0, 0x04, 0x16, /* 18-1F */
    382     0x07, 0x09, 0x0a, 0x0b, 0x0d, 0x0e, 0x0f, 0x33, /* 20-27 */
    383     0x34, 0x35, 0xe1, 0x31, 0x1d, 0x1b, 0x06, 0x19, /* 28-2F */
    384     0x05, 0x11, 0x10, 0x36, 0x37, 0x38, 0xe5, 0x55, /* 30-37 */
    385     0xe2, 0x2c, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, /* 38-3F */
    386     0x3f, 0x40, 0x41, 0x42, 0x43, 0x53, 0x47, 0x5f, /* 40-47 */
    387     0x60, 0x61, 0x56, 0x5c, 0x5d, 0x5e, 0x57, 0x59, /* 48-4F */
    388     0x5a, 0x5b, 0x62, 0x63, 0x46, 0x00, 0x64, 0x44, /* 50-57 */
    389     0x45, 0x67, 0x00, 0x00, 0x8c, 0x00, 0x00, 0x00, /* 58-5F */
    390                /* Sun keys: Props Undo  Front Copy */
    391     0x00, 0x00, 0x00, 0x00, 0x76, 0x7a, 0x77, 0x7c, /* 60-67 */
    392  /* Open  Paste Find  Cut   Stop  Again Help */
    393     0x74, 0x7d, 0x7e, 0x7b, 0x78, 0x79, 0x75, 0x00, /* 68-6F */
    394     0x88, 0x91, 0x90, 0x87, 0x00, 0x00, 0x00, 0x00, /* 70-77 */
    395     0x00, 0x8a, 0x00, 0x8b, 0x00, 0x89, 0x85, 0x00  /* 78-7F */
    396 };
    397 
    398 /** Lookup table for extended scancodes (arrow keys etc.). */
    399 static uint8_t aExtScan2Hid[] =
    400 {
    401     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 00-07 */
    402     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 08-1F */
    403     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 10-17 */
    404     0x00, 0x00, 0x00, 0x00, 0x58, 0xe4, 0x00, 0x00, /* 18-1F */
    405     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 20-27 */
    406     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 28-2F */
    407     0x00, 0x00, 0x00, 0x00, 0x00, 0x54, 0x00, 0x46, /* 30-37 */
    408     0xe6, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 38-3F */
    409     0x00, 0x00, 0x00, 0x00, 0x00, 0x48, 0x00, 0x4a, /* 40-47 */
    410     0x52, 0x4b, 0x00, 0x50, 0x00, 0x4f, 0x00, 0x4d, /* 48-4F */
    411     0x51, 0x4e, 0x49, 0x4c, 0x00, 0x00, 0x00, 0x00, /* 50-57 */
    412     0x00, 0x00, 0x00, 0xe3, 0xe7, 0x65, 0x66, 0x00, /* 58-5F */
    413     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 60-67 */
    414     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 68-6F */
    415     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /* 70-77 */
    416     0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00  /* 78-7F */
    417 };
    418 
    419 /**
    420  * Convert a PC scan code to a USB HID usage byte.
    421  *
    422  * @param state         Current state of the translator (scan_state_t).
    423  * @param scanCode      Incoming scan code.
    424  * @param pUsage        Pointer to usage; high bit set for key up events. The
    425  *                      contents are only valid if returned state is SS_IDLE.
    426  *
    427  * @return scan_state_t New state of the translator.
    428  */
    429 static scan_state_t ScancodeToHidUsage(scan_state_t state, uint8_t scanCode, uint32_t *pUsage)
    430 {
    431     uint32_t    keyUp;
    432     uint8_t     usage;
    433 
    434     Assert(pUsage);
    435 
    436     /* Isolate the scan code and key break flag. */
    437     keyUp = (scanCode & 0x80) << 24;
    438 
    439     switch (state) {
    440     case SS_IDLE:
    441         if (scanCode == 0xE0) {
    442             state = SS_EXT;
    443         } else if (scanCode == 0xE1) {
    444             state = SS_EXT1;
    445         } else {
    446             usage = aScancode2Hid[scanCode & 0x7F];
    447             *pUsage = usage | keyUp;
    448             /* Remain in SS_IDLE state. */
    449         }
    450         break;
    451     case SS_EXT:
    452         usage = aExtScan2Hid[scanCode & 0x7F];
    453         *pUsage = usage | keyUp;
    454         state = SS_IDLE;
    455         break;
    456     case SS_EXT1:
    457         Assert(0);  //@todo - sort out the Pause key
    458         *pUsage = 0;
    459         state = SS_IDLE;
    460         break;
    461     }
    462     return state;
    463 }
    464350
    465351/*******************************************************************************
     
    933819 * @returns VBox status code.
    934820 * @param   pInterface      Pointer to the keyboard port interface (KBDState::Keyboard.IPort).
    935  * @param   u8KeyCode       The keycode.
    936  */
    937 static DECLCALLBACK(int) usbHidKeyboardPutEvent(PPDMIKEYBOARDPORT pInterface, uint8_t u8KeyCode)
     821 * @param   u32UsageCode    The key usage ID.
     822 */
     823static DECLCALLBACK(int) usbHidKeyboardPutEvent(PPDMIKEYBOARDPORT pInterface, uint32_t u32UsageCode)
    938824{
    939825    PUSBHID pThis = RT_FROM_MEMBER(pInterface, USBHID, Lun0.IPort);
    940     uint32_t    u32Usage = 0;
    941826    uint8_t     u8HidCode;
    942827    int         fKeyDown;
     
    945830    RTCritSectEnter(&pThis->CritSect);
    946831
    947     if (RT_UNLIKELY(u8KeyCode == KRSP_BAT_FAIL))
     832    if (RT_UNLIKELY(u32UsageCode == KRSP_BAT_FAIL))
    948833    {
    949834        /* Clear all currently depressed and unreported keys. */
     
    954839    else
    955840    {
    956 
    957         pThis->XlatState = ScancodeToHidUsage(pThis->XlatState, u8KeyCode, &u32Usage);
    958 
    959         if (pThis->XlatState == SS_IDLE)
     841        fKeyDown = !(u32UsageCode & 0x80000000);
     842        u8HidCode = u32UsageCode & 0xFF;
     843        AssertReturn(u8HidCode <= VBOX_USB_MAX_USAGE_CODE, VERR_INTERNAL_ERROR);
     844
     845        LogFlowFunc(("key %s: 0x%x\n", fKeyDown ? "down" : "up", u8HidCode));
     846
     847        if (fKeyDown)
    960848        {
    961             /* The usage code is valid. */
    962             fKeyDown = !(u32Usage & 0x80000000);
    963             u8HidCode = u32Usage & 0xFF;
    964             AssertReturn(u8HidCode <= VBOX_USB_MAX_USAGE_CODE, VERR_INTERNAL_ERROR);
    965 
    966             LogFlowFunc(("key %s: 0x%x->0x%x\n",
    967                             fKeyDown ? "down" : "up", u8KeyCode, u8HidCode));
    968 
    969             if (fKeyDown)
     849            /* Due to host key repeat, we can get key events for keys which are
     850             * already depressed. */
     851            if (!pThis->abDepressedKeys[u8HidCode])
    970852            {
    971                 /* Due to host key repeat, we can get key events for keys which are
    972                  * already depressed. */
    973                 if (!pThis->abDepressedKeys[u8HidCode])
     853                pThis->abUnreportedKeys[u8HidCode] = 1;
     854
     855                /* If a non-modifier key is being marked as unreported, also set
     856                 * all currently depressed modifer keys as unreported. This avoids
     857                 * problems where a simulated key sequence is sent too fast and
     858                 * by the time the key is reported, some previously reported
     859                 * modifiers are already released. This helps ensure that the guest
     860                 * sees the entire modifier(s)+key sequence in a single report.
     861                 */
     862                if (!usbHidUsageCodeIsModifier(u8HidCode))
    974863                {
    975                     pThis->abUnreportedKeys[u8HidCode] = 1;
    976 
    977                     /* If a non-modifier key is being marked as unreported, also set
    978                      * all currently depressed modifer keys as unreported. This avoids
    979                      * problems where a simulated key sequence is sent too fast and
    980                      * by the time the key is reported, some previously reported
    981                      * modifiers are already released. This helps ensure that the guest
    982                      * sees the entire modifier(s)+key sequence in a single report.
    983                      */
    984                     if (!usbHidUsageCodeIsModifier(u8HidCode))
    985                     {
    986                         int     iModKey;
    987 
    988                         for (iModKey = USBHID_MODIFIER_FIRST; iModKey <= USBHID_MODIFIER_LAST; ++iModKey)
    989                             if (pThis->abDepressedKeys[iModKey])
    990                                 pThis->abUnreportedKeys[iModKey] = 1;
    991                     }
     864                    int     iModKey;
     865
     866                    for (iModKey = USBHID_MODIFIER_FIRST; iModKey <= USBHID_MODIFIER_LAST; ++iModKey)
     867                        if (pThis->abDepressedKeys[iModKey])
     868                            pThis->abUnreportedKeys[iModKey] = 1;
    992869                }
    993                 else
    994                     fHaveEvent = false;
    995                 pThis->abDepressedKeys[u8HidCode] = 1;
    996870            }
    997871            else
    998             {
    999                 /* For stupid Korean keyboards, we have to fake a key up/down sequence
    1000                  * because they only send break codes for Hangul/Hanja keys.
    1001                  */
    1002                 if (u8HidCode == 0x90 || u8HidCode == 0x91)
    1003                     pThis->abUnreportedKeys[u8HidCode] = 1;
    1004                 pThis->abDepressedKeys[u8HidCode] = 0;
    1005             }
    1006 
    1007 
    1008             /* Send a report if the host is already waiting for it. */
    1009             if (fHaveEvent)
    1010                 usbHidSendReport(pThis);
     872                fHaveEvent = false;
     873            pThis->abDepressedKeys[u8HidCode] = 1;
    1011874        }
     875        else
     876        {
     877            /* For stupid Korean keyboards, we have to fake a key up/down sequence
     878             * because they only send break codes for Hangul/Hanja keys.
     879             */
     880            if (u8HidCode == 0x90 || u8HidCode == 0x91)
     881                pThis->abUnreportedKeys[u8HidCode] = 1;
     882            pThis->abDepressedKeys[u8HidCode] = 0;
     883        }
     884
     885
     886        /* Send a report if the host is already waiting for it. */
     887        if (fHaveEvent)
     888            usbHidSendReport(pThis);
    1012889    }
    1013890
     
    15041381    pThis->pUsbIns                                  = pUsbIns;
    15051382    pThis->hEvtDoneQueue                            = NIL_RTSEMEVENT;
    1506     pThis->XlatState                                = SS_IDLE;
    15071383    usbHidQueueInit(&pThis->ToHostQueue);
    15081384    usbHidQueueInit(&pThis->DoneQueue);
     
    15221398
    15231399    pThis->Lun0.IBase.pfnQueryInterface = usbHidKeyboardQueryInterface;
    1524     pThis->Lun0.IPort.pfnPutEvent       = usbHidKeyboardPutEvent;
     1400    pThis->Lun0.IPort.pfnPutEventHid    = usbHidKeyboardPutEvent;
    15251401
    15261402    /*
  • trunk/src/VBox/Main/src-client/KeyboardImpl.cpp

    r50544 r50962  
    211211    uint32_t sent;
    212212    for (sent = 0; (sent < keys.size()) && RT_SUCCESS(vrc); sent++)
    213         vrc = pUpPort->pfnPutEvent(pUpPort, (uint8_t)keys[sent]);
     213        vrc = pUpPort->pfnPutEventScan(pUpPort, (uint8_t)keys[sent]);
    214214
    215215    if (codesStored)
Note: See TracChangeset for help on using the changeset viewer.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette