2011-10-08 04:44:59 +02:00
|
|
|
/** @file
|
|
|
|
Implementation for EFI_SIMPLE_TEXT_INPUT_PROTOCOL protocol.
|
|
|
|
|
2014-12-04 01:32:24 +01:00
|
|
|
(C) Copyright 2014 Hewlett-Packard Development Company, L.P.<BR>
|
2015-11-26 09:46:20 +01:00
|
|
|
Copyright (c) 2006 - 2015, Intel Corporation. All rights reserved.<BR>
|
2011-10-08 04:44:59 +02:00
|
|
|
This program and the accompanying materials
|
|
|
|
are licensed and made available under the terms and conditions of the BSD License
|
|
|
|
which accompanies this distribution. The full text of the license may be found at
|
|
|
|
http://opensource.org/licenses/bsd-license.php
|
|
|
|
|
|
|
|
THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
|
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include "Terminal.h"
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Reads the next keystroke from the input device. The WaitForKey Event can
|
|
|
|
be used to test for existence of a keystroke via WaitForEvent () call.
|
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure
|
|
|
|
@param KeyData A pointer to a buffer that is filled in with the
|
|
|
|
keystroke state data for the key that was
|
|
|
|
pressed.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The keystroke information was returned.
|
|
|
|
@retval EFI_NOT_READY There was no keystroke data available.
|
|
|
|
@retval EFI_INVALID_PARAMETER KeyData is NULL.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
ReadKeyStrokeWorker (
|
|
|
|
IN TERMINAL_DEV *TerminalDevice,
|
|
|
|
OUT EFI_KEY_DATA *KeyData
|
|
|
|
)
|
|
|
|
{
|
|
|
|
if (KeyData == NULL) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!EfiKeyFiFoRemoveOneKey (TerminalDevice, &KeyData->Key)) {
|
|
|
|
return EFI_NOT_READY;
|
|
|
|
}
|
|
|
|
|
|
|
|
KeyData->KeyState.KeyShiftState = 0;
|
|
|
|
KeyData->KeyState.KeyToggleState = 0;
|
|
|
|
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Implements EFI_SIMPLE_TEXT_INPUT_PROTOCOL.Reset().
|
|
|
|
This driver only perform dependent serial device reset regardless of
|
|
|
|
the value of ExtendeVerification
|
|
|
|
|
|
|
|
@param This Indicates the calling context.
|
|
|
|
@param ExtendedVerification Skip by this driver.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The reset operation succeeds.
|
|
|
|
@retval EFI_DEVICE_ERROR The dependent serial port reset fails.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
TerminalConInReset (
|
|
|
|
IN EFI_SIMPLE_TEXT_INPUT_PROTOCOL *This,
|
|
|
|
IN BOOLEAN ExtendedVerification
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
TERMINAL_DEV *TerminalDevice;
|
|
|
|
|
|
|
|
TerminalDevice = TERMINAL_CON_IN_DEV_FROM_THIS (This);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Report progress code here
|
|
|
|
//
|
|
|
|
REPORT_STATUS_CODE_WITH_DEVICE_PATH (
|
|
|
|
EFI_PROGRESS_CODE,
|
|
|
|
(EFI_PERIPHERAL_REMOTE_CONSOLE | EFI_P_PC_RESET),
|
|
|
|
TerminalDevice->DevicePath
|
|
|
|
);
|
|
|
|
|
|
|
|
Status = TerminalDevice->SerialIo->Reset (TerminalDevice->SerialIo);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Make all the internal buffer empty for keys
|
|
|
|
//
|
|
|
|
TerminalDevice->RawFiFo->Head = TerminalDevice->RawFiFo->Tail;
|
|
|
|
TerminalDevice->UnicodeFiFo->Head = TerminalDevice->UnicodeFiFo->Tail;
|
|
|
|
TerminalDevice->EfiKeyFiFo->Head = TerminalDevice->EfiKeyFiFo->Tail;
|
|
|
|
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
REPORT_STATUS_CODE_WITH_DEVICE_PATH (
|
|
|
|
EFI_ERROR_CODE | EFI_ERROR_MINOR,
|
|
|
|
(EFI_PERIPHERAL_REMOTE_CONSOLE | EFI_P_EC_CONTROLLER_ERROR),
|
|
|
|
TerminalDevice->DevicePath
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Implements EFI_SIMPLE_TEXT_INPUT_PROTOCOL.ReadKeyStroke().
|
|
|
|
|
|
|
|
@param This Indicates the calling context.
|
|
|
|
@param Key A pointer to a buffer that is filled in with the
|
|
|
|
keystroke information for the key that was sent
|
|
|
|
from terminal.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The keystroke information is returned successfully.
|
|
|
|
@retval EFI_NOT_READY There is no keystroke data available.
|
|
|
|
@retval EFI_DEVICE_ERROR The dependent serial device encounters error.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
TerminalConInReadKeyStroke (
|
|
|
|
IN EFI_SIMPLE_TEXT_INPUT_PROTOCOL *This,
|
|
|
|
OUT EFI_INPUT_KEY *Key
|
|
|
|
)
|
|
|
|
{
|
|
|
|
TERMINAL_DEV *TerminalDevice;
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_KEY_DATA KeyData;
|
|
|
|
|
|
|
|
//
|
|
|
|
// get TERMINAL_DEV from "This" parameter.
|
|
|
|
//
|
|
|
|
TerminalDevice = TERMINAL_CON_IN_DEV_FROM_THIS (This);
|
|
|
|
|
|
|
|
Status = ReadKeyStrokeWorker (TerminalDevice, &KeyData);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
CopyMem (Key, &KeyData.Key, sizeof (EFI_INPUT_KEY));
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Check if the key already has been registered.
|
|
|
|
|
|
|
|
If both RegsiteredData and InputData is NULL, then ASSERT().
|
|
|
|
|
|
|
|
@param RegsiteredData A pointer to a buffer that is filled in with the
|
|
|
|
keystroke state data for the key that was
|
|
|
|
registered.
|
|
|
|
@param InputData A pointer to a buffer that is filled in with the
|
|
|
|
keystroke state data for the key that was
|
|
|
|
pressed.
|
|
|
|
|
|
|
|
@retval TRUE Key be pressed matches a registered key.
|
2014-02-12 02:21:51 +01:00
|
|
|
@retval FALSE Match failed.
|
2011-10-08 04:44:59 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
IsKeyRegistered (
|
|
|
|
IN EFI_KEY_DATA *RegsiteredData,
|
|
|
|
IN EFI_KEY_DATA *InputData
|
|
|
|
)
|
|
|
|
{
|
|
|
|
ASSERT (RegsiteredData != NULL && InputData != NULL);
|
|
|
|
|
|
|
|
if ((RegsiteredData->Key.ScanCode != InputData->Key.ScanCode) ||
|
|
|
|
(RegsiteredData->Key.UnicodeChar != InputData->Key.UnicodeChar)) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Event notification function for EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL.WaitForKeyEx event
|
|
|
|
Signal the event if there is key available
|
|
|
|
|
|
|
|
@param Event Indicates the event that invoke this function.
|
|
|
|
@param Context Indicates the calling context.
|
|
|
|
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
TerminalConInWaitForKeyEx (
|
|
|
|
IN EFI_EVENT Event,
|
|
|
|
IN VOID *Context
|
|
|
|
)
|
|
|
|
{
|
|
|
|
TerminalConInWaitForKey (Event, Context);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Simple Text Input Ex protocol functions
|
|
|
|
//
|
|
|
|
|
|
|
|
/**
|
|
|
|
Reset the input device and optionally run diagnostics
|
|
|
|
|
|
|
|
@param This Protocol instance pointer.
|
|
|
|
@param ExtendedVerification Driver may perform diagnostics on reset.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The device was reset.
|
|
|
|
@retval EFI_DEVICE_ERROR The device is not functioning properly and could
|
|
|
|
not be reset.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
TerminalConInResetEx (
|
|
|
|
IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL *This,
|
|
|
|
IN BOOLEAN ExtendedVerification
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
TERMINAL_DEV *TerminalDevice;
|
|
|
|
|
|
|
|
TerminalDevice = TERMINAL_CON_IN_EX_DEV_FROM_THIS (This);
|
|
|
|
|
|
|
|
Status = TerminalDevice->SimpleInput.Reset (&TerminalDevice->SimpleInput, ExtendedVerification);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return EFI_DEVICE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Reads the next keystroke from the input device. The WaitForKey Event can
|
|
|
|
be used to test for existence of a keystroke via WaitForEvent () call.
|
|
|
|
|
|
|
|
@param This Protocol instance pointer.
|
|
|
|
@param KeyData A pointer to a buffer that is filled in with the
|
|
|
|
keystroke state data for the key that was
|
|
|
|
pressed.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The keystroke information was returned.
|
|
|
|
@retval EFI_NOT_READY There was no keystroke data available.
|
|
|
|
@retval EFI_DEVICE_ERROR The keystroke information was not returned due
|
|
|
|
to hardware errors.
|
|
|
|
@retval EFI_INVALID_PARAMETER KeyData is NULL.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
TerminalConInReadKeyStrokeEx (
|
|
|
|
IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL *This,
|
|
|
|
OUT EFI_KEY_DATA *KeyData
|
|
|
|
)
|
|
|
|
{
|
|
|
|
TERMINAL_DEV *TerminalDevice;
|
|
|
|
|
|
|
|
if (KeyData == NULL) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
TerminalDevice = TERMINAL_CON_IN_EX_DEV_FROM_THIS (This);
|
|
|
|
|
|
|
|
return ReadKeyStrokeWorker (TerminalDevice, KeyData);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Set certain state for the input device.
|
|
|
|
|
|
|
|
@param This Protocol instance pointer.
|
|
|
|
@param KeyToggleState A pointer to the EFI_KEY_TOGGLE_STATE to set the
|
|
|
|
state for the input device.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The device state was set successfully.
|
|
|
|
@retval EFI_DEVICE_ERROR The device is not functioning correctly and
|
|
|
|
could not have the setting adjusted.
|
|
|
|
@retval EFI_UNSUPPORTED The device does not have the ability to set its
|
|
|
|
state.
|
|
|
|
@retval EFI_INVALID_PARAMETER KeyToggleState is NULL.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
TerminalConInSetState (
|
|
|
|
IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL *This,
|
|
|
|
IN EFI_KEY_TOGGLE_STATE *KeyToggleState
|
|
|
|
)
|
|
|
|
{
|
|
|
|
if (KeyToggleState == NULL) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
2013-07-15 09:47:02 +02:00
|
|
|
if ((*KeyToggleState & EFI_TOGGLE_STATE_VALID) != EFI_TOGGLE_STATE_VALID) {
|
|
|
|
return EFI_UNSUPPORTED;
|
|
|
|
}
|
|
|
|
|
2011-10-08 04:44:59 +02:00
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Register a notification function for a particular keystroke for the input device.
|
|
|
|
|
|
|
|
@param This Protocol instance pointer.
|
|
|
|
@param KeyData A pointer to a buffer that is filled in with the
|
|
|
|
keystroke information data for the key that was
|
|
|
|
pressed.
|
|
|
|
@param KeyNotificationFunction Points to the function to be called when the key
|
|
|
|
sequence is typed specified by KeyData.
|
|
|
|
@param NotifyHandle Points to the unique handle assigned to the
|
|
|
|
registered notification.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The notification function was registered
|
|
|
|
successfully.
|
|
|
|
@retval EFI_OUT_OF_RESOURCES Unable to allocate resources for necessary data
|
|
|
|
structures.
|
|
|
|
@retval EFI_INVALID_PARAMETER KeyData or NotifyHandle is NULL.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
TerminalConInRegisterKeyNotify (
|
|
|
|
IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL *This,
|
|
|
|
IN EFI_KEY_DATA *KeyData,
|
|
|
|
IN EFI_KEY_NOTIFY_FUNCTION KeyNotificationFunction,
|
2012-07-30 05:50:42 +02:00
|
|
|
OUT VOID **NotifyHandle
|
2011-10-08 04:44:59 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
TERMINAL_DEV *TerminalDevice;
|
|
|
|
TERMINAL_CONSOLE_IN_EX_NOTIFY *NewNotify;
|
|
|
|
LIST_ENTRY *Link;
|
|
|
|
LIST_ENTRY *NotifyList;
|
|
|
|
TERMINAL_CONSOLE_IN_EX_NOTIFY *CurrentNotify;
|
|
|
|
|
|
|
|
if (KeyData == NULL || NotifyHandle == NULL || KeyNotificationFunction == NULL) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
TerminalDevice = TERMINAL_CON_IN_EX_DEV_FROM_THIS (This);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Return EFI_SUCCESS if the (KeyData, NotificationFunction) is already registered.
|
|
|
|
//
|
|
|
|
NotifyList = &TerminalDevice->NotifyList;
|
|
|
|
for (Link = GetFirstNode (NotifyList); !IsNull (NotifyList,Link); Link = GetNextNode (NotifyList,Link)) {
|
|
|
|
CurrentNotify = CR (
|
|
|
|
Link,
|
|
|
|
TERMINAL_CONSOLE_IN_EX_NOTIFY,
|
|
|
|
NotifyEntry,
|
|
|
|
TERMINAL_CONSOLE_IN_EX_NOTIFY_SIGNATURE
|
|
|
|
);
|
|
|
|
if (IsKeyRegistered (&CurrentNotify->KeyData, KeyData)) {
|
|
|
|
if (CurrentNotify->KeyNotificationFn == KeyNotificationFunction) {
|
2012-07-30 05:50:42 +02:00
|
|
|
*NotifyHandle = CurrentNotify;
|
2011-10-08 04:44:59 +02:00
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Allocate resource to save the notification function
|
|
|
|
//
|
|
|
|
NewNotify = (TERMINAL_CONSOLE_IN_EX_NOTIFY *) AllocateZeroPool (sizeof (TERMINAL_CONSOLE_IN_EX_NOTIFY));
|
|
|
|
if (NewNotify == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
NewNotify->Signature = TERMINAL_CONSOLE_IN_EX_NOTIFY_SIGNATURE;
|
|
|
|
NewNotify->KeyNotificationFn = KeyNotificationFunction;
|
2012-08-16 04:39:14 +02:00
|
|
|
CopyMem (&NewNotify->KeyData, KeyData, sizeof (EFI_KEY_DATA));
|
2011-10-08 04:44:59 +02:00
|
|
|
InsertTailList (&TerminalDevice->NotifyList, &NewNotify->NotifyEntry);
|
|
|
|
|
2012-07-30 05:50:42 +02:00
|
|
|
*NotifyHandle = NewNotify;
|
2011-10-08 04:44:59 +02:00
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Remove a registered notification function from a particular keystroke.
|
|
|
|
|
|
|
|
@param This Protocol instance pointer.
|
|
|
|
@param NotificationHandle The handle of the notification function being
|
|
|
|
unregistered.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The notification function was unregistered
|
|
|
|
successfully.
|
|
|
|
@retval EFI_INVALID_PARAMETER The NotificationHandle is invalid.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
TerminalConInUnregisterKeyNotify (
|
|
|
|
IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL *This,
|
2012-07-30 05:50:42 +02:00
|
|
|
IN VOID *NotificationHandle
|
2011-10-08 04:44:59 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
TERMINAL_DEV *TerminalDevice;
|
|
|
|
LIST_ENTRY *Link;
|
|
|
|
TERMINAL_CONSOLE_IN_EX_NOTIFY *CurrentNotify;
|
|
|
|
LIST_ENTRY *NotifyList;
|
|
|
|
|
|
|
|
if (NotificationHandle == NULL) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
TerminalDevice = TERMINAL_CON_IN_EX_DEV_FROM_THIS (This);
|
|
|
|
|
|
|
|
NotifyList = &TerminalDevice->NotifyList;
|
|
|
|
for (Link = GetFirstNode (NotifyList); !IsNull (NotifyList,Link); Link = GetNextNode (NotifyList,Link)) {
|
|
|
|
CurrentNotify = CR (
|
|
|
|
Link,
|
|
|
|
TERMINAL_CONSOLE_IN_EX_NOTIFY,
|
|
|
|
NotifyEntry,
|
|
|
|
TERMINAL_CONSOLE_IN_EX_NOTIFY_SIGNATURE
|
|
|
|
);
|
2012-07-30 05:50:42 +02:00
|
|
|
if (CurrentNotify == NotificationHandle) {
|
2011-10-08 04:44:59 +02:00
|
|
|
//
|
|
|
|
// Remove the notification function from NotifyList and free resources
|
|
|
|
//
|
|
|
|
RemoveEntryList (&CurrentNotify->NotifyEntry);
|
|
|
|
|
|
|
|
gBS->FreePool (CurrentNotify);
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Can not find the matching entry in database.
|
|
|
|
//
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Translate raw data into Unicode (according to different encode), and
|
|
|
|
translate Unicode into key information. (according to different standard).
|
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure.
|
|
|
|
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
TranslateRawDataToEfiKey (
|
|
|
|
IN TERMINAL_DEV *TerminalDevice
|
|
|
|
)
|
|
|
|
{
|
|
|
|
switch (TerminalDevice->TerminalType) {
|
|
|
|
|
|
|
|
case PCANSITYPE:
|
|
|
|
case VT100TYPE:
|
|
|
|
case VT100PLUSTYPE:
|
2015-07-09 08:24:11 +02:00
|
|
|
case TTYTERMTYPE:
|
2011-10-08 04:44:59 +02:00
|
|
|
AnsiRawDataToUnicode (TerminalDevice);
|
|
|
|
UnicodeToEfiKey (TerminalDevice);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VTUTF8TYPE:
|
|
|
|
//
|
|
|
|
// Process all the raw data in the RawFIFO,
|
|
|
|
// put the processed key into UnicodeFIFO.
|
|
|
|
//
|
|
|
|
VTUTF8RawDataToUnicode (TerminalDevice);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Translate all the Unicode data in the UnicodeFIFO to Efi key,
|
|
|
|
// then put into EfiKeyFIFO.
|
|
|
|
//
|
|
|
|
UnicodeToEfiKey (TerminalDevice);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Event notification function for EFI_SIMPLE_TEXT_INPUT_PROTOCOL.WaitForKey event
|
|
|
|
Signal the event if there is key available
|
|
|
|
|
|
|
|
@param Event Indicates the event that invoke this function.
|
|
|
|
@param Context Indicates the calling context.
|
|
|
|
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
TerminalConInWaitForKey (
|
|
|
|
IN EFI_EVENT Event,
|
|
|
|
IN VOID *Context
|
|
|
|
)
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// Someone is waiting on the keystroke event, if there's
|
|
|
|
// a key pending, signal the event
|
|
|
|
//
|
|
|
|
if (!IsEfiKeyFiFoEmpty ((TERMINAL_DEV *) Context)) {
|
|
|
|
|
|
|
|
gBS->SignalEvent (Event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Timer handler to poll the key from serial.
|
|
|
|
|
|
|
|
@param Event Indicates the event that invoke this function.
|
|
|
|
@param Context Indicates the calling context.
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
EFIAPI
|
|
|
|
TerminalConInTimerHandler (
|
|
|
|
IN EFI_EVENT Event,
|
|
|
|
IN VOID *Context
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
TERMINAL_DEV *TerminalDevice;
|
2011-10-27 07:38:42 +02:00
|
|
|
UINT32 Control;
|
2011-10-08 04:44:59 +02:00
|
|
|
UINT8 Input;
|
|
|
|
EFI_SERIAL_IO_MODE *Mode;
|
|
|
|
EFI_SERIAL_IO_PROTOCOL *SerialIo;
|
|
|
|
UINTN SerialInTimeOut;
|
|
|
|
|
|
|
|
TerminalDevice = (TERMINAL_DEV *) Context;
|
|
|
|
|
|
|
|
SerialIo = TerminalDevice->SerialIo;
|
|
|
|
if (SerialIo == NULL) {
|
|
|
|
return ;
|
|
|
|
}
|
|
|
|
//
|
|
|
|
// if current timeout value for serial device is not identical with
|
|
|
|
// the value saved in TERMINAL_DEV structure, then recalculate the
|
|
|
|
// timeout value again and set serial attribute according to this value.
|
|
|
|
//
|
|
|
|
Mode = SerialIo->Mode;
|
|
|
|
if (Mode->Timeout != TerminalDevice->SerialInTimeOut) {
|
|
|
|
|
|
|
|
SerialInTimeOut = 0;
|
|
|
|
if (Mode->BaudRate != 0) {
|
|
|
|
//
|
|
|
|
// According to BAUD rate to calculate the timeout value.
|
|
|
|
//
|
|
|
|
SerialInTimeOut = (1 + Mode->DataBits + Mode->StopBits) * 2 * 1000000 / (UINTN) Mode->BaudRate;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = SerialIo->SetAttributes (
|
|
|
|
SerialIo,
|
|
|
|
Mode->BaudRate,
|
2016-03-29 09:30:37 +02:00
|
|
|
Mode->ReceiveFifoDepth,
|
2011-10-08 04:44:59 +02:00
|
|
|
(UINT32) SerialInTimeOut,
|
|
|
|
(EFI_PARITY_TYPE) (Mode->Parity),
|
|
|
|
(UINT8) Mode->DataBits,
|
|
|
|
(EFI_STOP_BITS_TYPE) (Mode->StopBits)
|
|
|
|
);
|
|
|
|
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
TerminalDevice->SerialInTimeOut = 0;
|
|
|
|
} else {
|
|
|
|
TerminalDevice->SerialInTimeOut = SerialInTimeOut;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//
|
2011-10-27 07:38:42 +02:00
|
|
|
// Check whether serial buffer is empty.
|
2015-11-26 09:46:20 +01:00
|
|
|
// Skip the key transfer loop only if the SerialIo protocol instance
|
|
|
|
// successfully reports EFI_SERIAL_INPUT_BUFFER_EMPTY.
|
2011-10-08 04:44:59 +02:00
|
|
|
//
|
2011-10-27 07:38:42 +02:00
|
|
|
Status = SerialIo->GetControl (SerialIo, &Control);
|
2015-11-26 09:46:20 +01:00
|
|
|
if (EFI_ERROR (Status) || ((Control & EFI_SERIAL_INPUT_BUFFER_EMPTY) == 0)) {
|
2011-10-27 07:38:42 +02:00
|
|
|
//
|
|
|
|
// Fetch all the keys in the serial buffer,
|
|
|
|
// and insert the byte stream into RawFIFO.
|
|
|
|
//
|
|
|
|
while (!IsRawFiFoFull (TerminalDevice)) {
|
|
|
|
|
|
|
|
Status = GetOneKeyFromSerial (TerminalDevice->SerialIo, &Input);
|
|
|
|
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
if (Status == EFI_DEVICE_ERROR) {
|
|
|
|
REPORT_STATUS_CODE_WITH_DEVICE_PATH (
|
|
|
|
EFI_ERROR_CODE | EFI_ERROR_MINOR,
|
|
|
|
(EFI_PERIPHERAL_REMOTE_CONSOLE | EFI_P_EC_INPUT_ERROR),
|
|
|
|
TerminalDevice->DevicePath
|
|
|
|
);
|
|
|
|
}
|
|
|
|
break;
|
2011-10-08 04:44:59 +02:00
|
|
|
}
|
|
|
|
|
2011-10-27 07:38:42 +02:00
|
|
|
RawFiFoInsertOneKey (TerminalDevice, Input);
|
|
|
|
}
|
2011-10-08 04:44:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Translate all the raw data in RawFIFO into EFI Key,
|
|
|
|
// according to different terminal type supported.
|
|
|
|
//
|
|
|
|
TranslateRawDataToEfiKey (TerminalDevice);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Get one key out of serial buffer.
|
|
|
|
|
|
|
|
@param SerialIo Serial I/O protocol attached to the serial device.
|
|
|
|
@param Output The fetched key.
|
|
|
|
|
|
|
|
@retval EFI_NOT_READY If serial buffer is empty.
|
|
|
|
@retval EFI_DEVICE_ERROR If reading serial buffer encounter error.
|
|
|
|
@retval EFI_SUCCESS If reading serial buffer successfully, put
|
|
|
|
the fetched key to the parameter output.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
GetOneKeyFromSerial (
|
|
|
|
EFI_SERIAL_IO_PROTOCOL *SerialIo,
|
|
|
|
UINT8 *Output
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN Size;
|
|
|
|
|
|
|
|
Size = 1;
|
|
|
|
*Output = 0;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Read one key from serial I/O device.
|
|
|
|
//
|
|
|
|
Status = SerialIo->Read (SerialIo, &Size, Output);
|
|
|
|
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
|
|
|
|
if (Status == EFI_TIMEOUT) {
|
|
|
|
return EFI_NOT_READY;
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_DEVICE_ERROR;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*Output == 0) {
|
|
|
|
return EFI_NOT_READY;
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Insert one byte raw data into the Raw Data FIFO.
|
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure.
|
|
|
|
@param Input The key will be input.
|
|
|
|
|
|
|
|
@retval TRUE If insert successfully.
|
2014-02-12 02:21:51 +01:00
|
|
|
@retval FALSE If Raw Data buffer is full before key insertion,
|
2011-10-08 04:44:59 +02:00
|
|
|
and the key is lost.
|
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
RawFiFoInsertOneKey (
|
|
|
|
TERMINAL_DEV *TerminalDevice,
|
|
|
|
UINT8 Input
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT8 Tail;
|
|
|
|
|
|
|
|
Tail = TerminalDevice->RawFiFo->Tail;
|
|
|
|
|
|
|
|
if (IsRawFiFoFull (TerminalDevice)) {
|
|
|
|
//
|
|
|
|
// Raw FIFO is full
|
|
|
|
//
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
TerminalDevice->RawFiFo->Data[Tail] = Input;
|
|
|
|
|
|
|
|
TerminalDevice->RawFiFo->Tail = (UINT8) ((Tail + 1) % (RAW_FIFO_MAX_NUMBER + 1));
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Remove one pre-fetched key out of the Raw Data FIFO.
|
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure.
|
|
|
|
@param Output The key will be removed.
|
|
|
|
|
|
|
|
@retval TRUE If insert successfully.
|
2014-02-12 02:21:51 +01:00
|
|
|
@retval FALSE If Raw Data FIFO buffer is empty before remove operation.
|
2011-10-08 04:44:59 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
RawFiFoRemoveOneKey (
|
|
|
|
TERMINAL_DEV *TerminalDevice,
|
|
|
|
UINT8 *Output
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT8 Head;
|
|
|
|
|
|
|
|
Head = TerminalDevice->RawFiFo->Head;
|
|
|
|
|
|
|
|
if (IsRawFiFoEmpty (TerminalDevice)) {
|
|
|
|
//
|
|
|
|
// FIFO is empty
|
|
|
|
//
|
|
|
|
*Output = 0;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*Output = TerminalDevice->RawFiFo->Data[Head];
|
|
|
|
|
|
|
|
TerminalDevice->RawFiFo->Head = (UINT8) ((Head + 1) % (RAW_FIFO_MAX_NUMBER + 1));
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Clarify whether Raw Data FIFO buffer is empty.
|
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure
|
|
|
|
|
|
|
|
@retval TRUE If Raw Data FIFO buffer is empty.
|
2014-02-12 02:21:51 +01:00
|
|
|
@retval FALSE If Raw Data FIFO buffer is not empty.
|
2011-10-08 04:44:59 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
IsRawFiFoEmpty (
|
|
|
|
TERMINAL_DEV *TerminalDevice
|
|
|
|
)
|
|
|
|
{
|
|
|
|
if (TerminalDevice->RawFiFo->Head == TerminalDevice->RawFiFo->Tail) {
|
|
|
|
return TRUE;
|
|
|
|
} else {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Clarify whether Raw Data FIFO buffer is full.
|
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure
|
|
|
|
|
|
|
|
@retval TRUE If Raw Data FIFO buffer is full.
|
2014-02-12 02:21:51 +01:00
|
|
|
@retval FALSE If Raw Data FIFO buffer is not full.
|
2011-10-08 04:44:59 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
IsRawFiFoFull (
|
|
|
|
TERMINAL_DEV *TerminalDevice
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT8 Tail;
|
|
|
|
UINT8 Head;
|
|
|
|
|
|
|
|
Tail = TerminalDevice->RawFiFo->Tail;
|
|
|
|
Head = TerminalDevice->RawFiFo->Head;
|
|
|
|
|
|
|
|
if (((Tail + 1) % (RAW_FIFO_MAX_NUMBER + 1)) == Head) {
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Insert one pre-fetched key into the FIFO buffer.
|
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure.
|
|
|
|
@param Key The key will be input.
|
|
|
|
|
|
|
|
@retval TRUE If insert successfully.
|
2014-02-12 02:21:51 +01:00
|
|
|
@retval FALSE If FIFO buffer is full before key insertion,
|
2011-10-08 04:44:59 +02:00
|
|
|
and the key is lost.
|
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
EfiKeyFiFoInsertOneKey (
|
|
|
|
TERMINAL_DEV *TerminalDevice,
|
|
|
|
EFI_INPUT_KEY *Key
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT8 Tail;
|
|
|
|
LIST_ENTRY *Link;
|
|
|
|
LIST_ENTRY *NotifyList;
|
|
|
|
TERMINAL_CONSOLE_IN_EX_NOTIFY *CurrentNotify;
|
|
|
|
EFI_KEY_DATA KeyData;
|
|
|
|
|
|
|
|
Tail = TerminalDevice->EfiKeyFiFo->Tail;
|
|
|
|
|
|
|
|
CopyMem (&KeyData.Key, Key, sizeof (EFI_INPUT_KEY));
|
|
|
|
KeyData.KeyState.KeyShiftState = 0;
|
|
|
|
KeyData.KeyState.KeyToggleState = 0;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Invoke notification functions if exist
|
|
|
|
//
|
|
|
|
NotifyList = &TerminalDevice->NotifyList;
|
|
|
|
for (Link = GetFirstNode (NotifyList); !IsNull (NotifyList,Link); Link = GetNextNode (NotifyList,Link)) {
|
|
|
|
CurrentNotify = CR (
|
|
|
|
Link,
|
|
|
|
TERMINAL_CONSOLE_IN_EX_NOTIFY,
|
|
|
|
NotifyEntry,
|
|
|
|
TERMINAL_CONSOLE_IN_EX_NOTIFY_SIGNATURE
|
|
|
|
);
|
|
|
|
if (IsKeyRegistered (&CurrentNotify->KeyData, &KeyData)) {
|
|
|
|
CurrentNotify->KeyNotificationFn (&KeyData);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (IsEfiKeyFiFoFull (TerminalDevice)) {
|
|
|
|
//
|
|
|
|
// Efi Key FIFO is full
|
|
|
|
//
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
CopyMem (&TerminalDevice->EfiKeyFiFo->Data[Tail], Key, sizeof (EFI_INPUT_KEY));
|
|
|
|
|
|
|
|
TerminalDevice->EfiKeyFiFo->Tail = (UINT8) ((Tail + 1) % (FIFO_MAX_NUMBER + 1));
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Remove one pre-fetched key out of the FIFO buffer.
|
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure.
|
|
|
|
@param Output The key will be removed.
|
|
|
|
|
|
|
|
@retval TRUE If insert successfully.
|
2014-02-12 02:21:51 +01:00
|
|
|
@retval FALSE If FIFO buffer is empty before remove operation.
|
2011-10-08 04:44:59 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
EfiKeyFiFoRemoveOneKey (
|
|
|
|
TERMINAL_DEV *TerminalDevice,
|
|
|
|
EFI_INPUT_KEY *Output
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT8 Head;
|
|
|
|
|
|
|
|
Head = TerminalDevice->EfiKeyFiFo->Head;
|
|
|
|
ASSERT (Head < FIFO_MAX_NUMBER + 1);
|
|
|
|
|
|
|
|
if (IsEfiKeyFiFoEmpty (TerminalDevice)) {
|
|
|
|
//
|
|
|
|
// FIFO is empty
|
|
|
|
//
|
|
|
|
Output->ScanCode = SCAN_NULL;
|
|
|
|
Output->UnicodeChar = 0;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2014-12-04 01:32:24 +01:00
|
|
|
CopyMem (Output, &TerminalDevice->EfiKeyFiFo->Data[Head], sizeof (EFI_INPUT_KEY));
|
2011-10-08 04:44:59 +02:00
|
|
|
|
|
|
|
TerminalDevice->EfiKeyFiFo->Head = (UINT8) ((Head + 1) % (FIFO_MAX_NUMBER + 1));
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Clarify whether FIFO buffer is empty.
|
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure
|
|
|
|
|
|
|
|
@retval TRUE If FIFO buffer is empty.
|
2014-02-12 02:21:51 +01:00
|
|
|
@retval FALSE If FIFO buffer is not empty.
|
2011-10-08 04:44:59 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
IsEfiKeyFiFoEmpty (
|
|
|
|
TERMINAL_DEV *TerminalDevice
|
|
|
|
)
|
|
|
|
{
|
|
|
|
if (TerminalDevice->EfiKeyFiFo->Head == TerminalDevice->EfiKeyFiFo->Tail) {
|
|
|
|
return TRUE;
|
|
|
|
} else {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Clarify whether FIFO buffer is full.
|
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure
|
|
|
|
|
|
|
|
@retval TRUE If FIFO buffer is full.
|
2014-02-12 02:21:51 +01:00
|
|
|
@retval FALSE If FIFO buffer is not full.
|
2011-10-08 04:44:59 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
IsEfiKeyFiFoFull (
|
|
|
|
TERMINAL_DEV *TerminalDevice
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT8 Tail;
|
|
|
|
UINT8 Head;
|
|
|
|
|
|
|
|
Tail = TerminalDevice->EfiKeyFiFo->Tail;
|
|
|
|
Head = TerminalDevice->EfiKeyFiFo->Head;
|
|
|
|
|
|
|
|
if (((Tail + 1) % (FIFO_MAX_NUMBER + 1)) == Head) {
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Insert one pre-fetched key into the Unicode FIFO buffer.
|
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure.
|
|
|
|
@param Input The key will be input.
|
|
|
|
|
|
|
|
@retval TRUE If insert successfully.
|
2014-02-12 02:21:51 +01:00
|
|
|
@retval FALSE If Unicode FIFO buffer is full before key insertion,
|
2011-10-08 04:44:59 +02:00
|
|
|
and the key is lost.
|
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
UnicodeFiFoInsertOneKey (
|
|
|
|
TERMINAL_DEV *TerminalDevice,
|
|
|
|
UINT16 Input
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT8 Tail;
|
|
|
|
|
|
|
|
Tail = TerminalDevice->UnicodeFiFo->Tail;
|
|
|
|
ASSERT (Tail < FIFO_MAX_NUMBER + 1);
|
|
|
|
|
|
|
|
|
|
|
|
if (IsUnicodeFiFoFull (TerminalDevice)) {
|
|
|
|
//
|
|
|
|
// Unicode FIFO is full
|
|
|
|
//
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
TerminalDevice->UnicodeFiFo->Data[Tail] = Input;
|
|
|
|
|
|
|
|
TerminalDevice->UnicodeFiFo->Tail = (UINT8) ((Tail + 1) % (FIFO_MAX_NUMBER + 1));
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Remove one pre-fetched key out of the Unicode FIFO buffer.
|
2014-02-11 01:13:50 +01:00
|
|
|
The caller should guarantee that Unicode FIFO buffer is not empty
|
|
|
|
by IsUnicodeFiFoEmpty ().
|
2011-10-08 04:44:59 +02:00
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure.
|
|
|
|
@param Output The key will be removed.
|
|
|
|
|
|
|
|
**/
|
2014-02-11 01:13:50 +01:00
|
|
|
VOID
|
2011-10-08 04:44:59 +02:00
|
|
|
UnicodeFiFoRemoveOneKey (
|
|
|
|
TERMINAL_DEV *TerminalDevice,
|
|
|
|
UINT16 *Output
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT8 Head;
|
|
|
|
|
|
|
|
Head = TerminalDevice->UnicodeFiFo->Head;
|
|
|
|
ASSERT (Head < FIFO_MAX_NUMBER + 1);
|
|
|
|
|
|
|
|
*Output = TerminalDevice->UnicodeFiFo->Data[Head];
|
|
|
|
|
|
|
|
TerminalDevice->UnicodeFiFo->Head = (UINT8) ((Head + 1) % (FIFO_MAX_NUMBER + 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Clarify whether Unicode FIFO buffer is empty.
|
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure
|
|
|
|
|
|
|
|
@retval TRUE If Unicode FIFO buffer is empty.
|
2014-02-12 02:21:51 +01:00
|
|
|
@retval FALSE If Unicode FIFO buffer is not empty.
|
2011-10-08 04:44:59 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
IsUnicodeFiFoEmpty (
|
|
|
|
TERMINAL_DEV *TerminalDevice
|
|
|
|
)
|
|
|
|
{
|
|
|
|
if (TerminalDevice->UnicodeFiFo->Head == TerminalDevice->UnicodeFiFo->Tail) {
|
|
|
|
return TRUE;
|
|
|
|
} else {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Clarify whether Unicode FIFO buffer is full.
|
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure
|
|
|
|
|
|
|
|
@retval TRUE If Unicode FIFO buffer is full.
|
2014-02-12 02:21:51 +01:00
|
|
|
@retval FALSE If Unicode FIFO buffer is not full.
|
2011-10-08 04:44:59 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
IsUnicodeFiFoFull (
|
|
|
|
TERMINAL_DEV *TerminalDevice
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT8 Tail;
|
|
|
|
UINT8 Head;
|
|
|
|
|
|
|
|
Tail = TerminalDevice->UnicodeFiFo->Tail;
|
|
|
|
Head = TerminalDevice->UnicodeFiFo->Head;
|
|
|
|
|
|
|
|
if (((Tail + 1) % (FIFO_MAX_NUMBER + 1)) == Head) {
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Count Unicode FIFO buffer.
|
|
|
|
|
|
|
|
@param TerminalDevice Terminal driver private structure
|
|
|
|
|
|
|
|
@return The count in bytes of Unicode FIFO.
|
|
|
|
|
|
|
|
**/
|
|
|
|
UINT8
|
|
|
|
UnicodeFiFoGetKeyCount (
|
|
|
|
TERMINAL_DEV *TerminalDevice
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT8 Tail;
|
|
|
|
UINT8 Head;
|
|
|
|
|
|
|
|
Tail = TerminalDevice->UnicodeFiFo->Tail;
|
|
|
|
Head = TerminalDevice->UnicodeFiFo->Head;
|
|
|
|
|
|
|
|
if (Tail >= Head) {
|
|
|
|
return (UINT8) (Tail - Head);
|
|
|
|
} else {
|
|
|
|
return (UINT8) (Tail + FIFO_MAX_NUMBER + 1 - Head);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Update the Unicode characters from a terminal input device into EFI Keys FIFO.
|
|
|
|
|
|
|
|
@param TerminalDevice The terminal device to use to translate raw input into EFI Keys
|
|
|
|
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
UnicodeToEfiKeyFlushState (
|
|
|
|
IN TERMINAL_DEV *TerminalDevice
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_INPUT_KEY Key;
|
|
|
|
UINT32 InputState;
|
|
|
|
|
|
|
|
InputState = TerminalDevice->InputState;
|
|
|
|
|
|
|
|
if (IsEfiKeyFiFoFull (TerminalDevice)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((InputState & INPUT_STATE_ESC) != 0) {
|
|
|
|
Key.ScanCode = SCAN_ESC;
|
|
|
|
Key.UnicodeChar = 0;
|
|
|
|
EfiKeyFiFoInsertOneKey (TerminalDevice, &Key);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((InputState & INPUT_STATE_CSI) != 0) {
|
|
|
|
Key.ScanCode = SCAN_NULL;
|
|
|
|
Key.UnicodeChar = CSI;
|
|
|
|
EfiKeyFiFoInsertOneKey (TerminalDevice, &Key);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((InputState & INPUT_STATE_LEFTOPENBRACKET) != 0) {
|
|
|
|
Key.ScanCode = SCAN_NULL;
|
|
|
|
Key.UnicodeChar = LEFTOPENBRACKET;
|
|
|
|
EfiKeyFiFoInsertOneKey (TerminalDevice, &Key);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((InputState & INPUT_STATE_O) != 0) {
|
|
|
|
Key.ScanCode = SCAN_NULL;
|
|
|
|
Key.UnicodeChar = 'O';
|
|
|
|
EfiKeyFiFoInsertOneKey (TerminalDevice, &Key);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((InputState & INPUT_STATE_2) != 0) {
|
|
|
|
Key.ScanCode = SCAN_NULL;
|
|
|
|
Key.UnicodeChar = '2';
|
|
|
|
EfiKeyFiFoInsertOneKey (TerminalDevice, &Key);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Cancel the timer.
|
|
|
|
//
|
|
|
|
gBS->SetTimer (
|
|
|
|
TerminalDevice->TwoSecondTimeOut,
|
|
|
|
TimerCancel,
|
|
|
|
0
|
|
|
|
);
|
|
|
|
|
|
|
|
TerminalDevice->InputState = INPUT_STATE_DEFAULT;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Converts a stream of Unicode characters from a terminal input device into EFI Keys that
|
|
|
|
can be read through the Simple Input Protocol.
|
|
|
|
|
|
|
|
The table below shows the keyboard input mappings that this function supports.
|
|
|
|
If the ESC sequence listed in one of the columns is presented, then it is translated
|
|
|
|
into the corresponding EFI Scan Code. If a matching sequence is not found, then the raw
|
|
|
|
key strokes are converted into EFI Keys.
|
|
|
|
|
|
|
|
2 seconds are allowed for an ESC sequence to be completed. If the ESC sequence is not
|
|
|
|
completed in 2 seconds, then the raw key strokes of the partial ESC sequence are
|
|
|
|
converted into EFI Keys.
|
|
|
|
There is one special input sequence that will force the system to reset.
|
|
|
|
This is ESC R ESC r ESC R.
|
|
|
|
|
|
|
|
Note: current implementation support terminal types include: PC ANSI, VT100+/VTUTF8, VT100.
|
|
|
|
The table below is not same with UEFI Spec 2.3 Appendix B Table 201(not support ANSI X3.64 /
|
|
|
|
DEC VT200-500 and extra support PC ANSI, VT100)since UEFI Table 201 is just an example.
|
|
|
|
|
|
|
|
Symbols used in table below
|
|
|
|
===========================
|
|
|
|
ESC = 0x1B
|
|
|
|
CSI = 0x9B
|
|
|
|
DEL = 0x7f
|
|
|
|
^ = CTRL
|
|
|
|
|
|
|
|
+=========+======+===========+==========+==========+
|
|
|
|
| | EFI | UEFI 2.0 | | |
|
|
|
|
| | Scan | | VT100+ | |
|
|
|
|
| KEY | Code | PC ANSI | VTUTF8 | VT100 |
|
|
|
|
+=========+======+===========+==========+==========+
|
|
|
|
| NULL | 0x00 | | | |
|
|
|
|
| UP | 0x01 | ESC [ A | ESC [ A | ESC [ A |
|
|
|
|
| DOWN | 0x02 | ESC [ B | ESC [ B | ESC [ B |
|
|
|
|
| RIGHT | 0x03 | ESC [ C | ESC [ C | ESC [ C |
|
|
|
|
| LEFT | 0x04 | ESC [ D | ESC [ D | ESC [ D |
|
|
|
|
| HOME | 0x05 | ESC [ H | ESC h | ESC [ H |
|
|
|
|
| END | 0x06 | ESC [ F | ESC k | ESC [ K |
|
|
|
|
| INSERT | 0x07 | ESC [ @ | ESC + | ESC [ @ |
|
|
|
|
| | | ESC [ L | | ESC [ L |
|
|
|
|
| DELETE | 0x08 | ESC [ X | ESC - | ESC [ P |
|
|
|
|
| PG UP | 0x09 | ESC [ I | ESC ? | ESC [ V |
|
|
|
|
| | | | | ESC [ ? |
|
|
|
|
| PG DOWN | 0x0A | ESC [ G | ESC / | ESC [ U |
|
|
|
|
| | | | | ESC [ / |
|
|
|
|
| F1 | 0x0B | ESC [ M | ESC 1 | ESC O P |
|
|
|
|
| F2 | 0x0C | ESC [ N | ESC 2 | ESC O Q |
|
|
|
|
| F3 | 0x0D | ESC [ O | ESC 3 | ESC O w |
|
|
|
|
| F4 | 0x0E | ESC [ P | ESC 4 | ESC O x |
|
|
|
|
| F5 | 0x0F | ESC [ Q | ESC 5 | ESC O t |
|
|
|
|
| F6 | 0x10 | ESC [ R | ESC 6 | ESC O u |
|
|
|
|
| F7 | 0x11 | ESC [ S | ESC 7 | ESC O q |
|
|
|
|
| F8 | 0x12 | ESC [ T | ESC 8 | ESC O r |
|
|
|
|
| F9 | 0x13 | ESC [ U | ESC 9 | ESC O p |
|
|
|
|
| F10 | 0x14 | ESC [ V | ESC 0 | ESC O M |
|
|
|
|
| Escape | 0x17 | ESC | ESC | ESC |
|
|
|
|
| F11 | 0x15 | | ESC ! | |
|
|
|
|
| F12 | 0x16 | | ESC @ | |
|
|
|
|
+=========+======+===========+==========+==========+
|
|
|
|
|
|
|
|
Special Mappings
|
|
|
|
================
|
|
|
|
ESC R ESC r ESC R = Reset System
|
|
|
|
|
|
|
|
@param TerminalDevice The terminal device to use to translate raw input into EFI Keys
|
|
|
|
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
UnicodeToEfiKey (
|
|
|
|
IN TERMINAL_DEV *TerminalDevice
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_STATUS TimerStatus;
|
|
|
|
UINT16 UnicodeChar;
|
|
|
|
EFI_INPUT_KEY Key;
|
|
|
|
BOOLEAN SetDefaultResetState;
|
|
|
|
|
|
|
|
TimerStatus = gBS->CheckEvent (TerminalDevice->TwoSecondTimeOut);
|
|
|
|
|
|
|
|
if (!EFI_ERROR (TimerStatus)) {
|
|
|
|
UnicodeToEfiKeyFlushState (TerminalDevice);
|
|
|
|
TerminalDevice->ResetState = RESET_STATE_DEFAULT;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (!IsUnicodeFiFoEmpty (TerminalDevice) && !IsEfiKeyFiFoFull (TerminalDevice)) {
|
|
|
|
|
|
|
|
if (TerminalDevice->InputState != INPUT_STATE_DEFAULT) {
|
|
|
|
//
|
|
|
|
// Check to see if the 2 seconds timer has expired
|
|
|
|
//
|
|
|
|
TimerStatus = gBS->CheckEvent (TerminalDevice->TwoSecondTimeOut);
|
|
|
|
if (!EFI_ERROR (TimerStatus)) {
|
|
|
|
UnicodeToEfiKeyFlushState (TerminalDevice);
|
|
|
|
TerminalDevice->ResetState = RESET_STATE_DEFAULT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Fetch one Unicode character from the Unicode FIFO
|
|
|
|
//
|
|
|
|
UnicodeFiFoRemoveOneKey (TerminalDevice, &UnicodeChar);
|
|
|
|
|
|
|
|
SetDefaultResetState = TRUE;
|
|
|
|
|
|
|
|
switch (TerminalDevice->InputState) {
|
|
|
|
case INPUT_STATE_DEFAULT:
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case INPUT_STATE_ESC:
|
|
|
|
|
|
|
|
if (UnicodeChar == LEFTOPENBRACKET) {
|
|
|
|
TerminalDevice->InputState |= INPUT_STATE_LEFTOPENBRACKET;
|
|
|
|
TerminalDevice->ResetState = RESET_STATE_DEFAULT;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
Accept VT220 DEL and function keys for TTY terminal type
Accept the VT220 escape code [3~ as backspace for TtyTerm terminals. This is
sent by many Linux terminals by default. Also accept VT220 function keys
F1-F12, and VT100 F1-F4 keys as these are commonly sent by Linux terminals.
The VT220 escape codes are longer, and variable length so a new state is added
to the state machine along with a variable to construct the multibyte escape
sequence.
There are currently no ambiguous escape sequence prefixes accepted, so the TTY
terminal accepts escape sequences for a variety of terminals. The goal is to
'just work' with as many terminals as possible, rather than properly emulating
any specific terminal. Backspace, Del, and F10 have been tested on xterm,
rxvt, tmux, and screen.
Note: The existing vt100 function key handling does not match the vt100
documentation that I found, so I added the TTY terminal handling
of VT100 F1-F4 (really PF1-PF4 on vt100) separately. The vt100
has no F5-F10 keys, so I don't know what the current vt100 code
is based on.
Contributed-under: TianoCore Contribution Agreement 1.0
Signed-off-by: Roy Franz <roy.franz@linaro.org>
Reviewed-by: Feng Tian <feng.tian@intel.com>
git-svn-id: https://svn.code.sf.net/p/edk2/code/trunk/edk2@17897 6f19259b-4bc3-4df7-8a09-765794883524
2015-07-09 08:24:20 +02:00
|
|
|
if (UnicodeChar == 'O' && (TerminalDevice->TerminalType == VT100TYPE ||
|
|
|
|
TerminalDevice->TerminalType == TTYTERMTYPE)) {
|
2011-10-08 04:44:59 +02:00
|
|
|
TerminalDevice->InputState |= INPUT_STATE_O;
|
|
|
|
TerminalDevice->ResetState = RESET_STATE_DEFAULT;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
Key.ScanCode = SCAN_NULL;
|
|
|
|
|
|
|
|
if (TerminalDevice->TerminalType == VT100PLUSTYPE ||
|
|
|
|
TerminalDevice->TerminalType == VTUTF8TYPE) {
|
|
|
|
switch (UnicodeChar) {
|
|
|
|
case '1':
|
|
|
|
Key.ScanCode = SCAN_F1;
|
|
|
|
break;
|
|
|
|
case '2':
|
|
|
|
Key.ScanCode = SCAN_F2;
|
|
|
|
break;
|
|
|
|
case '3':
|
|
|
|
Key.ScanCode = SCAN_F3;
|
|
|
|
break;
|
|
|
|
case '4':
|
|
|
|
Key.ScanCode = SCAN_F4;
|
|
|
|
break;
|
|
|
|
case '5':
|
|
|
|
Key.ScanCode = SCAN_F5;
|
|
|
|
break;
|
|
|
|
case '6':
|
|
|
|
Key.ScanCode = SCAN_F6;
|
|
|
|
break;
|
|
|
|
case '7':
|
|
|
|
Key.ScanCode = SCAN_F7;
|
|
|
|
break;
|
|
|
|
case '8':
|
|
|
|
Key.ScanCode = SCAN_F8;
|
|
|
|
break;
|
|
|
|
case '9':
|
|
|
|
Key.ScanCode = SCAN_F9;
|
|
|
|
break;
|
|
|
|
case '0':
|
|
|
|
Key.ScanCode = SCAN_F10;
|
|
|
|
break;
|
|
|
|
case '!':
|
|
|
|
Key.ScanCode = SCAN_F11;
|
|
|
|
break;
|
|
|
|
case '@':
|
|
|
|
Key.ScanCode = SCAN_F12;
|
|
|
|
break;
|
|
|
|
case 'h':
|
|
|
|
Key.ScanCode = SCAN_HOME;
|
|
|
|
break;
|
|
|
|
case 'k':
|
|
|
|
Key.ScanCode = SCAN_END;
|
|
|
|
break;
|
|
|
|
case '+':
|
|
|
|
Key.ScanCode = SCAN_INSERT;
|
|
|
|
break;
|
|
|
|
case '-':
|
|
|
|
Key.ScanCode = SCAN_DELETE;
|
|
|
|
break;
|
|
|
|
case '/':
|
|
|
|
Key.ScanCode = SCAN_PAGE_DOWN;
|
|
|
|
break;
|
|
|
|
case '?':
|
|
|
|
Key.ScanCode = SCAN_PAGE_UP;
|
|
|
|
break;
|
|
|
|
default :
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (UnicodeChar) {
|
|
|
|
case 'R':
|
|
|
|
if (TerminalDevice->ResetState == RESET_STATE_DEFAULT) {
|
|
|
|
TerminalDevice->ResetState = RESET_STATE_ESC_R;
|
|
|
|
SetDefaultResetState = FALSE;
|
|
|
|
} else if (TerminalDevice->ResetState == RESET_STATE_ESC_R_ESC_R) {
|
|
|
|
gRT->ResetSystem (EfiResetWarm, EFI_SUCCESS, 0, NULL);
|
|
|
|
}
|
|
|
|
Key.ScanCode = SCAN_NULL;
|
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
if (TerminalDevice->ResetState == RESET_STATE_ESC_R) {
|
|
|
|
TerminalDevice->ResetState = RESET_STATE_ESC_R_ESC_R;
|
|
|
|
SetDefaultResetState = FALSE;
|
|
|
|
}
|
|
|
|
Key.ScanCode = SCAN_NULL;
|
|
|
|
break;
|
|
|
|
default :
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (SetDefaultResetState) {
|
|
|
|
TerminalDevice->ResetState = RESET_STATE_DEFAULT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Key.ScanCode != SCAN_NULL) {
|
|
|
|
Key.UnicodeChar = 0;
|
|
|
|
EfiKeyFiFoInsertOneKey (TerminalDevice, &Key);
|
|
|
|
TerminalDevice->InputState = INPUT_STATE_DEFAULT;
|
|
|
|
UnicodeToEfiKeyFlushState (TerminalDevice);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
UnicodeToEfiKeyFlushState (TerminalDevice);
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case INPUT_STATE_ESC | INPUT_STATE_O:
|
|
|
|
|
|
|
|
TerminalDevice->ResetState = RESET_STATE_DEFAULT;
|
|
|
|
|
|
|
|
Key.ScanCode = SCAN_NULL;
|
|
|
|
|
|
|
|
if (TerminalDevice->TerminalType == VT100TYPE) {
|
|
|
|
switch (UnicodeChar) {
|
|
|
|
case 'P':
|
|
|
|
Key.ScanCode = SCAN_F1;
|
|
|
|
break;
|
|
|
|
case 'Q':
|
|
|
|
Key.ScanCode = SCAN_F2;
|
|
|
|
break;
|
|
|
|
case 'w':
|
|
|
|
Key.ScanCode = SCAN_F3;
|
|
|
|
break;
|
|
|
|
case 'x':
|
|
|
|
Key.ScanCode = SCAN_F4;
|
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
Key.ScanCode = SCAN_F5;
|
|
|
|
break;
|
|
|
|
case 'u':
|
|
|
|
Key.ScanCode = SCAN_F6;
|
|
|
|
break;
|
|
|
|
case 'q':
|
|
|
|
Key.ScanCode = SCAN_F7;
|
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
Key.ScanCode = SCAN_F8;
|
|
|
|
break;
|
|
|
|
case 'p':
|
|
|
|
Key.ScanCode = SCAN_F9;
|
|
|
|
break;
|
|
|
|
case 'M':
|
|
|
|
Key.ScanCode = SCAN_F10;
|
|
|
|
break;
|
|
|
|
default :
|
|
|
|
break;
|
|
|
|
}
|
Accept VT220 DEL and function keys for TTY terminal type
Accept the VT220 escape code [3~ as backspace for TtyTerm terminals. This is
sent by many Linux terminals by default. Also accept VT220 function keys
F1-F12, and VT100 F1-F4 keys as these are commonly sent by Linux terminals.
The VT220 escape codes are longer, and variable length so a new state is added
to the state machine along with a variable to construct the multibyte escape
sequence.
There are currently no ambiguous escape sequence prefixes accepted, so the TTY
terminal accepts escape sequences for a variety of terminals. The goal is to
'just work' with as many terminals as possible, rather than properly emulating
any specific terminal. Backspace, Del, and F10 have been tested on xterm,
rxvt, tmux, and screen.
Note: The existing vt100 function key handling does not match the vt100
documentation that I found, so I added the TTY terminal handling
of VT100 F1-F4 (really PF1-PF4 on vt100) separately. The vt100
has no F5-F10 keys, so I don't know what the current vt100 code
is based on.
Contributed-under: TianoCore Contribution Agreement 1.0
Signed-off-by: Roy Franz <roy.franz@linaro.org>
Reviewed-by: Feng Tian <feng.tian@intel.com>
git-svn-id: https://svn.code.sf.net/p/edk2/code/trunk/edk2@17897 6f19259b-4bc3-4df7-8a09-765794883524
2015-07-09 08:24:20 +02:00
|
|
|
} else if (TerminalDevice->TerminalType == TTYTERMTYPE) {
|
|
|
|
/* Also accept VT100 escape codes for F1-F4 for TTY term */
|
|
|
|
switch (UnicodeChar) {
|
|
|
|
case 'P':
|
|
|
|
Key.ScanCode = SCAN_F1;
|
|
|
|
break;
|
|
|
|
case 'Q':
|
|
|
|
Key.ScanCode = SCAN_F2;
|
|
|
|
break;
|
|
|
|
case 'R':
|
|
|
|
Key.ScanCode = SCAN_F3;
|
|
|
|
break;
|
|
|
|
case 'S':
|
|
|
|
Key.ScanCode = SCAN_F4;
|
|
|
|
break;
|
|
|
|
}
|
2011-10-08 04:44:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (Key.ScanCode != SCAN_NULL) {
|
|
|
|
Key.UnicodeChar = 0;
|
|
|
|
EfiKeyFiFoInsertOneKey (TerminalDevice, &Key);
|
|
|
|
TerminalDevice->InputState = INPUT_STATE_DEFAULT;
|
|
|
|
UnicodeToEfiKeyFlushState (TerminalDevice);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
UnicodeToEfiKeyFlushState (TerminalDevice);
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case INPUT_STATE_ESC | INPUT_STATE_LEFTOPENBRACKET:
|
|
|
|
|
|
|
|
TerminalDevice->ResetState = RESET_STATE_DEFAULT;
|
|
|
|
|
|
|
|
Key.ScanCode = SCAN_NULL;
|
|
|
|
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE ||
|
|
|
|
TerminalDevice->TerminalType == VT100TYPE ||
|
|
|
|
TerminalDevice->TerminalType == VT100PLUSTYPE ||
|
2015-07-09 08:24:11 +02:00
|
|
|
TerminalDevice->TerminalType == VTUTF8TYPE ||
|
|
|
|
TerminalDevice->TerminalType == TTYTERMTYPE) {
|
2011-10-08 04:44:59 +02:00
|
|
|
switch (UnicodeChar) {
|
|
|
|
case 'A':
|
|
|
|
Key.ScanCode = SCAN_UP;
|
|
|
|
break;
|
|
|
|
case 'B':
|
|
|
|
Key.ScanCode = SCAN_DOWN;
|
|
|
|
break;
|
|
|
|
case 'C':
|
|
|
|
Key.ScanCode = SCAN_RIGHT;
|
|
|
|
break;
|
|
|
|
case 'D':
|
|
|
|
Key.ScanCode = SCAN_LEFT;
|
|
|
|
break;
|
|
|
|
case 'H':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE ||
|
|
|
|
TerminalDevice->TerminalType == VT100TYPE) {
|
|
|
|
Key.ScanCode = SCAN_HOME;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'F':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE) {
|
|
|
|
Key.ScanCode = SCAN_END;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'K':
|
|
|
|
if (TerminalDevice->TerminalType == VT100TYPE) {
|
|
|
|
Key.ScanCode = SCAN_END;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'L':
|
|
|
|
case '@':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE ||
|
|
|
|
TerminalDevice->TerminalType == VT100TYPE) {
|
|
|
|
Key.ScanCode = SCAN_INSERT;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'X':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE) {
|
|
|
|
Key.ScanCode = SCAN_DELETE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'P':
|
|
|
|
if (TerminalDevice->TerminalType == VT100TYPE) {
|
|
|
|
Key.ScanCode = SCAN_DELETE;
|
|
|
|
} else if (TerminalDevice->TerminalType == PCANSITYPE) {
|
|
|
|
Key.ScanCode = SCAN_F4;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'I':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE) {
|
|
|
|
Key.ScanCode = SCAN_PAGE_UP;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'V':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE) {
|
|
|
|
Key.ScanCode = SCAN_F10;
|
|
|
|
}
|
2012-08-17 06:01:48 +02:00
|
|
|
break;
|
2011-10-08 04:44:59 +02:00
|
|
|
case '?':
|
|
|
|
if (TerminalDevice->TerminalType == VT100TYPE) {
|
|
|
|
Key.ScanCode = SCAN_PAGE_UP;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'G':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE) {
|
|
|
|
Key.ScanCode = SCAN_PAGE_DOWN;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'U':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE) {
|
|
|
|
Key.ScanCode = SCAN_F9;
|
|
|
|
}
|
2012-08-17 06:01:48 +02:00
|
|
|
break;
|
2011-10-08 04:44:59 +02:00
|
|
|
case '/':
|
|
|
|
if (TerminalDevice->TerminalType == VT100TYPE) {
|
|
|
|
Key.ScanCode = SCAN_PAGE_DOWN;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'M':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE) {
|
|
|
|
Key.ScanCode = SCAN_F1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'N':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE) {
|
|
|
|
Key.ScanCode = SCAN_F2;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'O':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE) {
|
|
|
|
Key.ScanCode = SCAN_F3;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'Q':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE) {
|
|
|
|
Key.ScanCode = SCAN_F5;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'R':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE) {
|
|
|
|
Key.ScanCode = SCAN_F6;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'S':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE) {
|
|
|
|
Key.ScanCode = SCAN_F7;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'T':
|
|
|
|
if (TerminalDevice->TerminalType == PCANSITYPE) {
|
|
|
|
Key.ScanCode = SCAN_F8;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default :
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Accept VT220 DEL and function keys for TTY terminal type
Accept the VT220 escape code [3~ as backspace for TtyTerm terminals. This is
sent by many Linux terminals by default. Also accept VT220 function keys
F1-F12, and VT100 F1-F4 keys as these are commonly sent by Linux terminals.
The VT220 escape codes are longer, and variable length so a new state is added
to the state machine along with a variable to construct the multibyte escape
sequence.
There are currently no ambiguous escape sequence prefixes accepted, so the TTY
terminal accepts escape sequences for a variety of terminals. The goal is to
'just work' with as many terminals as possible, rather than properly emulating
any specific terminal. Backspace, Del, and F10 have been tested on xterm,
rxvt, tmux, and screen.
Note: The existing vt100 function key handling does not match the vt100
documentation that I found, so I added the TTY terminal handling
of VT100 F1-F4 (really PF1-PF4 on vt100) separately. The vt100
has no F5-F10 keys, so I don't know what the current vt100 code
is based on.
Contributed-under: TianoCore Contribution Agreement 1.0
Signed-off-by: Roy Franz <roy.franz@linaro.org>
Reviewed-by: Feng Tian <feng.tian@intel.com>
git-svn-id: https://svn.code.sf.net/p/edk2/code/trunk/edk2@17897 6f19259b-4bc3-4df7-8a09-765794883524
2015-07-09 08:24:20 +02:00
|
|
|
/*
|
|
|
|
* The VT220 escape codes that the TTY terminal accepts all have
|
|
|
|
* numeric codes, and there are no ambiguous prefixes shared with
|
|
|
|
* other terminal types.
|
|
|
|
*/
|
|
|
|
if (TerminalDevice->TerminalType == TTYTERMTYPE &&
|
|
|
|
Key.ScanCode == SCAN_NULL &&
|
|
|
|
UnicodeChar >= '0' &&
|
|
|
|
UnicodeChar <= '9') {
|
|
|
|
TerminalDevice->TtyEscapeStr[0] = UnicodeChar;
|
|
|
|
TerminalDevice->TtyEscapeIndex = 1;
|
|
|
|
TerminalDevice->InputState |= INPUT_STATE_LEFTOPENBRACKET_2;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2011-10-08 04:44:59 +02:00
|
|
|
if (Key.ScanCode != SCAN_NULL) {
|
|
|
|
Key.UnicodeChar = 0;
|
|
|
|
EfiKeyFiFoInsertOneKey (TerminalDevice, &Key);
|
|
|
|
TerminalDevice->InputState = INPUT_STATE_DEFAULT;
|
|
|
|
UnicodeToEfiKeyFlushState (TerminalDevice);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
UnicodeToEfiKeyFlushState (TerminalDevice);
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
Accept VT220 DEL and function keys for TTY terminal type
Accept the VT220 escape code [3~ as backspace for TtyTerm terminals. This is
sent by many Linux terminals by default. Also accept VT220 function keys
F1-F12, and VT100 F1-F4 keys as these are commonly sent by Linux terminals.
The VT220 escape codes are longer, and variable length so a new state is added
to the state machine along with a variable to construct the multibyte escape
sequence.
There are currently no ambiguous escape sequence prefixes accepted, so the TTY
terminal accepts escape sequences for a variety of terminals. The goal is to
'just work' with as many terminals as possible, rather than properly emulating
any specific terminal. Backspace, Del, and F10 have been tested on xterm,
rxvt, tmux, and screen.
Note: The existing vt100 function key handling does not match the vt100
documentation that I found, so I added the TTY terminal handling
of VT100 F1-F4 (really PF1-PF4 on vt100) separately. The vt100
has no F5-F10 keys, so I don't know what the current vt100 code
is based on.
Contributed-under: TianoCore Contribution Agreement 1.0
Signed-off-by: Roy Franz <roy.franz@linaro.org>
Reviewed-by: Feng Tian <feng.tian@intel.com>
git-svn-id: https://svn.code.sf.net/p/edk2/code/trunk/edk2@17897 6f19259b-4bc3-4df7-8a09-765794883524
2015-07-09 08:24:20 +02:00
|
|
|
case INPUT_STATE_ESC | INPUT_STATE_LEFTOPENBRACKET | INPUT_STATE_LEFTOPENBRACKET_2:
|
|
|
|
/*
|
|
|
|
* Here we handle the VT220 escape codes that we accept. This
|
|
|
|
* state is only used by the TTY terminal type.
|
|
|
|
*/
|
|
|
|
Key.ScanCode = SCAN_NULL;
|
|
|
|
if (TerminalDevice->TerminalType == TTYTERMTYPE) {
|
|
|
|
|
|
|
|
if (UnicodeChar == '~' && TerminalDevice->TtyEscapeIndex <= 2) {
|
2015-07-10 05:21:09 +02:00
|
|
|
UINT16 EscCode;
|
Accept VT220 DEL and function keys for TTY terminal type
Accept the VT220 escape code [3~ as backspace for TtyTerm terminals. This is
sent by many Linux terminals by default. Also accept VT220 function keys
F1-F12, and VT100 F1-F4 keys as these are commonly sent by Linux terminals.
The VT220 escape codes are longer, and variable length so a new state is added
to the state machine along with a variable to construct the multibyte escape
sequence.
There are currently no ambiguous escape sequence prefixes accepted, so the TTY
terminal accepts escape sequences for a variety of terminals. The goal is to
'just work' with as many terminals as possible, rather than properly emulating
any specific terminal. Backspace, Del, and F10 have been tested on xterm,
rxvt, tmux, and screen.
Note: The existing vt100 function key handling does not match the vt100
documentation that I found, so I added the TTY terminal handling
of VT100 F1-F4 (really PF1-PF4 on vt100) separately. The vt100
has no F5-F10 keys, so I don't know what the current vt100 code
is based on.
Contributed-under: TianoCore Contribution Agreement 1.0
Signed-off-by: Roy Franz <roy.franz@linaro.org>
Reviewed-by: Feng Tian <feng.tian@intel.com>
git-svn-id: https://svn.code.sf.net/p/edk2/code/trunk/edk2@17897 6f19259b-4bc3-4df7-8a09-765794883524
2015-07-09 08:24:20 +02:00
|
|
|
TerminalDevice->TtyEscapeStr[TerminalDevice->TtyEscapeIndex] = 0; /* Terminate string */
|
2015-07-10 05:21:09 +02:00
|
|
|
EscCode = (UINT16) StrDecimalToUintn(TerminalDevice->TtyEscapeStr);
|
Accept VT220 DEL and function keys for TTY terminal type
Accept the VT220 escape code [3~ as backspace for TtyTerm terminals. This is
sent by many Linux terminals by default. Also accept VT220 function keys
F1-F12, and VT100 F1-F4 keys as these are commonly sent by Linux terminals.
The VT220 escape codes are longer, and variable length so a new state is added
to the state machine along with a variable to construct the multibyte escape
sequence.
There are currently no ambiguous escape sequence prefixes accepted, so the TTY
terminal accepts escape sequences for a variety of terminals. The goal is to
'just work' with as many terminals as possible, rather than properly emulating
any specific terminal. Backspace, Del, and F10 have been tested on xterm,
rxvt, tmux, and screen.
Note: The existing vt100 function key handling does not match the vt100
documentation that I found, so I added the TTY terminal handling
of VT100 F1-F4 (really PF1-PF4 on vt100) separately. The vt100
has no F5-F10 keys, so I don't know what the current vt100 code
is based on.
Contributed-under: TianoCore Contribution Agreement 1.0
Signed-off-by: Roy Franz <roy.franz@linaro.org>
Reviewed-by: Feng Tian <feng.tian@intel.com>
git-svn-id: https://svn.code.sf.net/p/edk2/code/trunk/edk2@17897 6f19259b-4bc3-4df7-8a09-765794883524
2015-07-09 08:24:20 +02:00
|
|
|
switch (EscCode) {
|
|
|
|
case 3:
|
|
|
|
Key.ScanCode = SCAN_DELETE;
|
|
|
|
break;
|
|
|
|
case 11:
|
|
|
|
case 12:
|
|
|
|
case 13:
|
|
|
|
case 14:
|
|
|
|
case 15:
|
|
|
|
Key.ScanCode = SCAN_F1 + EscCode - 11;
|
|
|
|
break;
|
|
|
|
case 17:
|
|
|
|
case 18:
|
|
|
|
case 19:
|
|
|
|
case 20:
|
|
|
|
case 21:
|
|
|
|
Key.ScanCode = SCAN_F6 + EscCode - 17;
|
|
|
|
break;
|
|
|
|
case 23:
|
|
|
|
case 24:
|
|
|
|
Key.ScanCode = SCAN_F11 + EscCode - 23;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if (TerminalDevice->TtyEscapeIndex == 1){
|
|
|
|
/* 2 character escape code */
|
|
|
|
TerminalDevice->TtyEscapeStr[TerminalDevice->TtyEscapeIndex++] = UnicodeChar;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
DEBUG ((EFI_D_ERROR, "Unexpected state in escape2\n"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
TerminalDevice->ResetState = RESET_STATE_DEFAULT;
|
|
|
|
|
|
|
|
if (Key.ScanCode != SCAN_NULL) {
|
|
|
|
Key.UnicodeChar = 0;
|
|
|
|
EfiKeyFiFoInsertOneKey (TerminalDevice, &Key);
|
|
|
|
TerminalDevice->InputState = INPUT_STATE_DEFAULT;
|
|
|
|
UnicodeToEfiKeyFlushState (TerminalDevice);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
UnicodeToEfiKeyFlushState (TerminalDevice);
|
|
|
|
break;
|
|
|
|
|
2011-10-08 04:44:59 +02:00
|
|
|
default:
|
|
|
|
//
|
|
|
|
// Invalid state. This should never happen.
|
|
|
|
//
|
|
|
|
ASSERT (FALSE);
|
|
|
|
|
|
|
|
UnicodeToEfiKeyFlushState (TerminalDevice);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (UnicodeChar == ESC) {
|
|
|
|
TerminalDevice->InputState = INPUT_STATE_ESC;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (UnicodeChar == CSI) {
|
|
|
|
TerminalDevice->InputState = INPUT_STATE_CSI;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (TerminalDevice->InputState != INPUT_STATE_DEFAULT) {
|
|
|
|
Status = gBS->SetTimer(
|
|
|
|
TerminalDevice->TwoSecondTimeOut,
|
|
|
|
TimerRelative,
|
|
|
|
(UINT64)20000000
|
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (SetDefaultResetState) {
|
|
|
|
TerminalDevice->ResetState = RESET_STATE_DEFAULT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (UnicodeChar == DEL) {
|
2015-07-09 08:24:15 +02:00
|
|
|
if (TerminalDevice->TerminalType == TTYTERMTYPE) {
|
|
|
|
Key.ScanCode = SCAN_NULL;
|
|
|
|
Key.UnicodeChar = CHAR_BACKSPACE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
Key.ScanCode = SCAN_DELETE;
|
|
|
|
Key.UnicodeChar = 0;
|
|
|
|
}
|
2011-10-08 04:44:59 +02:00
|
|
|
} else {
|
|
|
|
Key.ScanCode = SCAN_NULL;
|
|
|
|
Key.UnicodeChar = UnicodeChar;
|
|
|
|
}
|
|
|
|
|
|
|
|
EfiKeyFiFoInsertOneKey (TerminalDevice, &Key);
|
|
|
|
}
|
|
|
|
}
|