2008-04-10 10:49:28 +02:00
|
|
|
/** @file
|
2008-12-17 10:34:06 +01:00
|
|
|
Produces Simple Text Input Protocol, Simple Text Input Extended Protocol and
|
2008-04-10 10:49:28 +02:00
|
|
|
Simple Text Output Protocol upon Serial IO Protocol.
|
2007-07-05 09:05:28 +02:00
|
|
|
|
2019-09-06 09:38:00 +02:00
|
|
|
Copyright (c) 2006 - 2019, Intel Corporation. All rights reserved.<BR>
|
2019-04-04 01:05:13 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2007-07-05 09:05:28 +02:00
|
|
|
|
2008-04-10 10:49:28 +02:00
|
|
|
**/
|
2007-07-05 09:05:28 +02:00
|
|
|
|
|
|
|
|
|
|
|
#include "Terminal.h"
|
|
|
|
|
|
|
|
//
|
|
|
|
// Globals
|
|
|
|
//
|
|
|
|
EFI_DRIVER_BINDING_PROTOCOL gTerminalDriverBinding = {
|
|
|
|
TerminalDriverBindingSupported,
|
|
|
|
TerminalDriverBindingStart,
|
|
|
|
TerminalDriverBindingStop,
|
|
|
|
0xa,
|
|
|
|
NULL,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2017-01-10 04:33:51 +01:00
|
|
|
EFI_GUID *mTerminalType[] = {
|
2007-08-17 00:15:08 +02:00
|
|
|
&gEfiPcAnsiGuid,
|
|
|
|
&gEfiVT100Guid,
|
|
|
|
&gEfiVT100PlusGuid,
|
2015-07-09 08:24:11 +02:00
|
|
|
&gEfiVTUTF8Guid,
|
2019-09-06 09:38:00 +02:00
|
|
|
&gEfiTtyTermGuid,
|
|
|
|
&gEdkiiLinuxTermGuid,
|
|
|
|
&gEdkiiXtermR6Guid,
|
|
|
|
&gEdkiiVT400Guid,
|
|
|
|
&gEdkiiSCOTermGuid
|
2007-08-17 00:15:08 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2017-01-10 06:05:46 +01:00
|
|
|
CHAR16 *mSerialConsoleNames[] = {
|
|
|
|
L"PC-ANSI Serial Console",
|
|
|
|
L"VT-100 Serial Console",
|
|
|
|
L"VT-100+ Serial Console",
|
|
|
|
L"VT-UTF8 Serial Console",
|
2019-09-06 09:38:00 +02:00
|
|
|
L"Tty Terminal Serial Console",
|
|
|
|
L"Linux Terminal Serial Console",
|
|
|
|
L"Xterm R6 Serial Console",
|
|
|
|
L"VT-400 Serial Console",
|
|
|
|
L"SCO Terminal Serial Console"
|
2017-01-10 06:05:46 +01:00
|
|
|
};
|
|
|
|
|
2008-11-21 09:32:38 +01:00
|
|
|
TERMINAL_DEV mTerminalDevTemplate = {
|
2007-08-17 00:15:08 +02:00
|
|
|
TERMINAL_DEV_SIGNATURE,
|
|
|
|
NULL,
|
|
|
|
0,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
{ // SimpleTextInput
|
|
|
|
TerminalConInReset,
|
|
|
|
TerminalConInReadKeyStroke,
|
|
|
|
NULL
|
|
|
|
},
|
|
|
|
{ // SimpleTextOutput
|
|
|
|
TerminalConOutReset,
|
|
|
|
TerminalConOutOutputString,
|
|
|
|
TerminalConOutTestString,
|
|
|
|
TerminalConOutQueryMode,
|
|
|
|
TerminalConOutSetMode,
|
|
|
|
TerminalConOutSetAttribute,
|
|
|
|
TerminalConOutClearScreen,
|
|
|
|
TerminalConOutSetCursorPosition,
|
|
|
|
TerminalConOutEnableCursor,
|
|
|
|
NULL
|
|
|
|
},
|
|
|
|
{ // SimpleTextOutputMode
|
|
|
|
1, // MaxMode
|
2008-12-17 10:34:06 +01:00
|
|
|
0, // Mode
|
2007-08-17 00:15:08 +02:00
|
|
|
EFI_TEXT_ATTR (EFI_LIGHTGRAY, EFI_BLACK), // Attribute
|
|
|
|
0, // CursorColumn
|
|
|
|
0, // CursorRow
|
|
|
|
TRUE // CursorVisible
|
|
|
|
},
|
2011-11-25 09:35:00 +01:00
|
|
|
NULL, // TerminalConsoleModeData
|
2008-12-17 10:34:06 +01:00
|
|
|
0, // SerialInTimeOut
|
2009-02-03 08:25:00 +01:00
|
|
|
|
|
|
|
NULL, // RawFifo
|
|
|
|
NULL, // UnicodeFiFo
|
|
|
|
NULL, // EfiKeyFiFo
|
2016-12-19 15:21:11 +01:00
|
|
|
NULL, // EfiKeyFiFoForNotify
|
2009-02-03 08:25:00 +01:00
|
|
|
|
2007-08-17 00:15:08 +02:00
|
|
|
NULL, // ControllerNameTable
|
2011-04-19 08:51:44 +02:00
|
|
|
NULL, // TimerEvent
|
2008-12-17 10:34:06 +01:00
|
|
|
NULL, // TwoSecondTimeOut
|
2007-08-17 00:15:08 +02:00
|
|
|
INPUT_STATE_DEFAULT,
|
|
|
|
RESET_STATE_DEFAULT,
|
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
|
|
|
{
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0
|
|
|
|
},
|
|
|
|
0,
|
2007-10-19 04:35:29 +02:00
|
|
|
FALSE,
|
|
|
|
{ // SimpleTextInputEx
|
|
|
|
TerminalConInResetEx,
|
|
|
|
TerminalConInReadKeyStrokeEx,
|
|
|
|
NULL,
|
|
|
|
TerminalConInSetState,
|
|
|
|
TerminalConInRegisterKeyNotify,
|
|
|
|
TerminalConInUnregisterKeyNotify,
|
|
|
|
},
|
2008-12-17 10:34:06 +01:00
|
|
|
{ // NotifyList
|
2007-10-19 04:35:29 +02:00
|
|
|
NULL,
|
|
|
|
NULL,
|
2016-12-19 15:21:11 +01:00
|
|
|
},
|
|
|
|
NULL // KeyNotifyProcessEvent
|
2007-08-17 00:15:08 +02:00
|
|
|
};
|
|
|
|
|
2011-11-25 09:35:00 +01:00
|
|
|
TERMINAL_CONSOLE_MODE_DATA mTerminalConsoleModeData[] = {
|
2017-01-10 06:11:25 +01:00
|
|
|
{80, 25},
|
|
|
|
{80, 50},
|
2011-11-25 09:35:00 +01:00
|
|
|
{100, 31},
|
|
|
|
//
|
|
|
|
// New modes can be added here.
|
|
|
|
//
|
|
|
|
};
|
|
|
|
|
2017-01-10 04:33:51 +01:00
|
|
|
/**
|
|
|
|
Convert the GUID representation of terminal type to enum type.
|
|
|
|
|
|
|
|
@param Guid The GUID representation of terminal type.
|
|
|
|
|
|
|
|
@return The terminal type in enum type.
|
|
|
|
**/
|
|
|
|
TERMINAL_TYPE
|
|
|
|
TerminalTypeFromGuid (
|
|
|
|
IN EFI_GUID *Guid
|
|
|
|
)
|
|
|
|
{
|
|
|
|
TERMINAL_TYPE Type;
|
|
|
|
|
|
|
|
for (Type = 0; Type < ARRAY_SIZE (mTerminalType); Type++) {
|
|
|
|
if (CompareGuid (Guid, mTerminalType[Type])) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Type;
|
|
|
|
}
|
|
|
|
|
2008-07-03 07:17:49 +02:00
|
|
|
/**
|
2009-01-08 09:46:31 +01:00
|
|
|
Test to see if this driver supports Controller.
|
2008-07-03 07:17:49 +02:00
|
|
|
|
|
|
|
@param This Protocol instance pointer.
|
2008-07-18 10:20:08 +02:00
|
|
|
@param Controller Handle of device to test
|
2008-07-03 07:17:49 +02:00
|
|
|
@param RemainingDevicePath Optional parameter use to pick a specific child
|
|
|
|
device to start.
|
|
|
|
|
2008-07-18 10:20:08 +02:00
|
|
|
@retval EFI_SUCCESS This driver supports this device.
|
|
|
|
@retval EFI_ALREADY_STARTED This driver is already running on this device.
|
|
|
|
@retval other This driver does not support this device.
|
2007-08-17 00:15:08 +02:00
|
|
|
|
2008-07-03 07:17:49 +02:00
|
|
|
**/
|
2007-07-05 09:05:28 +02:00
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
TerminalDriverBindingSupported (
|
|
|
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
|
|
|
IN EFI_HANDLE Controller,
|
|
|
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *ParentDevicePath;
|
|
|
|
EFI_SERIAL_IO_PROTOCOL *SerialIo;
|
|
|
|
VENDOR_DEVICE_PATH *Node;
|
|
|
|
|
|
|
|
//
|
|
|
|
// If remaining device path is not NULL, then make sure it is a
|
|
|
|
// device path that describes a terminal communications protocol.
|
|
|
|
//
|
|
|
|
if (RemainingDevicePath != NULL) {
|
|
|
|
//
|
2010-01-28 13:29:25 +01:00
|
|
|
// Check if RemainingDevicePath is the End of Device Path Node,
|
2009-09-14 10:55:03 +02:00
|
|
|
// if yes, go on checking other conditions
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
2009-09-14 10:55:03 +02:00
|
|
|
if (!IsDevicePathEnd (RemainingDevicePath)) {
|
|
|
|
//
|
|
|
|
// If RemainingDevicePath isn't the End of Device Path Node,
|
|
|
|
// check its validation
|
|
|
|
//
|
|
|
|
Node = (VENDOR_DEVICE_PATH *) RemainingDevicePath;
|
2010-01-28 13:29:25 +01:00
|
|
|
|
2009-09-14 10:55:03 +02:00
|
|
|
if (Node->Header.Type != MESSAGING_DEVICE_PATH ||
|
|
|
|
Node->Header.SubType != MSG_VENDOR_DP ||
|
|
|
|
DevicePathNodeLength(&Node->Header) != sizeof(VENDOR_DEVICE_PATH)) {
|
2010-01-28 13:29:25 +01:00
|
|
|
|
2009-09-14 10:55:03 +02:00
|
|
|
return EFI_UNSUPPORTED;
|
2010-01-28 13:29:25 +01:00
|
|
|
|
2009-09-14 10:55:03 +02:00
|
|
|
}
|
|
|
|
//
|
2019-09-06 09:38:00 +02:00
|
|
|
// only supports PC ANSI, VT100, VT100+, VT-UTF8, TtyTerm
|
|
|
|
// Linux, XtermR6, VT400 and SCO terminal types
|
2009-09-14 10:55:03 +02:00
|
|
|
//
|
2017-01-10 04:33:51 +01:00
|
|
|
if (TerminalTypeFromGuid (&Node->Guid) == ARRAY_SIZE (mTerminalType)) {
|
2009-09-14 10:55:03 +02:00
|
|
|
return EFI_UNSUPPORTED;
|
|
|
|
}
|
2007-07-05 09:05:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
//
|
|
|
|
// Open the IO Abstraction(s) needed to perform the supported test
|
2009-09-14 10:55:03 +02:00
|
|
|
// The Controller must support the Serial I/O Protocol.
|
|
|
|
// This driver is a bus driver with at most 1 child device, so it is
|
|
|
|
// ok for it to be already started.
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
|
|
|
Status = gBS->OpenProtocol (
|
|
|
|
Controller,
|
2009-09-14 10:55:03 +02:00
|
|
|
&gEfiSerialIoProtocolGuid,
|
|
|
|
(VOID **) &SerialIo,
|
2007-07-05 09:05:28 +02:00
|
|
|
This->DriverBindingHandle,
|
|
|
|
Controller,
|
|
|
|
EFI_OPEN_PROTOCOL_BY_DRIVER
|
|
|
|
);
|
|
|
|
if (Status == EFI_ALREADY_STARTED) {
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
2009-09-14 10:55:03 +02:00
|
|
|
//
|
|
|
|
// Close the I/O Abstraction(s) used to perform the supported test
|
|
|
|
//
|
2007-07-05 09:05:28 +02:00
|
|
|
gBS->CloseProtocol (
|
|
|
|
Controller,
|
2009-09-14 10:55:03 +02:00
|
|
|
&gEfiSerialIoProtocolGuid,
|
2007-07-05 09:05:28 +02:00
|
|
|
This->DriverBindingHandle,
|
|
|
|
Controller
|
|
|
|
);
|
|
|
|
|
|
|
|
//
|
2009-09-14 10:55:03 +02:00
|
|
|
// Open the EFI Device Path protocol needed to perform the supported test
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
|
|
|
Status = gBS->OpenProtocol (
|
|
|
|
Controller,
|
2009-09-14 10:55:03 +02:00
|
|
|
&gEfiDevicePathProtocolGuid,
|
|
|
|
(VOID **) &ParentDevicePath,
|
2007-07-05 09:05:28 +02:00
|
|
|
This->DriverBindingHandle,
|
|
|
|
Controller,
|
|
|
|
EFI_OPEN_PROTOCOL_BY_DRIVER
|
|
|
|
);
|
|
|
|
if (Status == EFI_ALREADY_STARTED) {
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
2009-09-14 10:55:03 +02:00
|
|
|
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
2009-09-14 10:55:03 +02:00
|
|
|
// Close protocol, don't use device path protocol in the Support() function
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
|
|
|
gBS->CloseProtocol (
|
|
|
|
Controller,
|
2009-09-14 10:55:03 +02:00
|
|
|
&gEfiDevicePathProtocolGuid,
|
2007-07-05 09:05:28 +02:00
|
|
|
This->DriverBindingHandle,
|
|
|
|
Controller
|
|
|
|
);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
2010-01-28 13:29:25 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
Free notify functions list.
|
|
|
|
|
|
|
|
@param ListHead The list head
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS Free the notify list successfully.
|
|
|
|
@retval EFI_INVALID_PARAMETER ListHead is NULL.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
TerminalFreeNotifyList (
|
|
|
|
IN OUT LIST_ENTRY *ListHead
|
|
|
|
)
|
|
|
|
{
|
|
|
|
TERMINAL_CONSOLE_IN_EX_NOTIFY *NotifyNode;
|
|
|
|
|
|
|
|
if (ListHead == NULL) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
while (!IsListEmpty (ListHead)) {
|
|
|
|
NotifyNode = CR (
|
|
|
|
ListHead->ForwardLink,
|
|
|
|
TERMINAL_CONSOLE_IN_EX_NOTIFY,
|
|
|
|
NotifyEntry,
|
|
|
|
TERMINAL_CONSOLE_IN_EX_NOTIFY_SIGNATURE
|
|
|
|
);
|
|
|
|
RemoveEntryList (ListHead->ForwardLink);
|
|
|
|
FreePool (NotifyNode);
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2011-11-25 09:35:00 +01:00
|
|
|
/**
|
|
|
|
Initialize all the text modes which the terminal console supports.
|
|
|
|
|
|
|
|
It returns information for available text modes that the terminal can support.
|
|
|
|
|
|
|
|
@param[out] TextModeCount The total number of text modes that terminal console supports.
|
|
|
|
|
2017-01-10 06:11:25 +01:00
|
|
|
@return The buffer to the text modes column and row information.
|
|
|
|
Caller is responsible to free it when it's non-NULL.
|
2011-11-25 09:35:00 +01:00
|
|
|
|
|
|
|
**/
|
2017-01-10 06:11:25 +01:00
|
|
|
TERMINAL_CONSOLE_MODE_DATA *
|
2011-11-25 09:35:00 +01:00
|
|
|
InitializeTerminalConsoleTextMode (
|
2017-01-10 06:11:25 +01:00
|
|
|
OUT INT32 *TextModeCount
|
|
|
|
)
|
2011-11-25 09:35:00 +01:00
|
|
|
{
|
2017-01-10 06:11:25 +01:00
|
|
|
TERMINAL_CONSOLE_MODE_DATA *TextModeData;
|
|
|
|
|
|
|
|
ASSERT (TextModeCount != NULL);
|
|
|
|
|
|
|
|
TextModeData = AllocateCopyPool (sizeof (mTerminalConsoleModeData), mTerminalConsoleModeData);
|
|
|
|
if (TextModeData == NULL) {
|
|
|
|
return NULL;
|
2011-11-25 09:35:00 +01:00
|
|
|
}
|
2017-01-10 06:11:25 +01:00
|
|
|
*TextModeCount = ARRAY_SIZE (mTerminalConsoleModeData);
|
|
|
|
|
2011-11-25 09:35:00 +01:00
|
|
|
DEBUG_CODE (
|
2017-01-10 06:11:25 +01:00
|
|
|
INT32 Index;
|
|
|
|
for (Index = 0; Index < *TextModeCount; Index++) {
|
|
|
|
DEBUG ((DEBUG_INFO, "Terminal - Mode %d, Column = %d, Row = %d\n",
|
|
|
|
Index, TextModeData[Index].Columns, TextModeData[Index].Rows));
|
2011-11-25 09:35:00 +01:00
|
|
|
}
|
|
|
|
);
|
2017-01-10 06:11:25 +01:00
|
|
|
return TextModeData;
|
2011-11-25 09:35:00 +01:00
|
|
|
}
|
2010-01-28 13:29:25 +01:00
|
|
|
|
2017-01-10 06:25:27 +01:00
|
|
|
/**
|
|
|
|
Stop the terminal state machine.
|
|
|
|
|
|
|
|
@param TerminalDevice The terminal device.
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
StopTerminalStateMachine (
|
|
|
|
TERMINAL_DEV *TerminalDevice
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_TPL OriginalTpl;
|
|
|
|
|
|
|
|
OriginalTpl = gBS->RaiseTPL (TPL_NOTIFY);
|
|
|
|
|
|
|
|
gBS->CloseEvent (TerminalDevice->TimerEvent);
|
|
|
|
gBS->CloseEvent (TerminalDevice->TwoSecondTimeOut);
|
|
|
|
|
|
|
|
gBS->RestoreTPL (OriginalTpl);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Start the terminal state machine.
|
|
|
|
|
|
|
|
@param TerminalDevice The terminal device.
|
|
|
|
**/
|
|
|
|
VOID
|
|
|
|
StartTerminalStateMachine (
|
|
|
|
TERMINAL_DEV *TerminalDevice
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
Status = gBS->CreateEvent (
|
|
|
|
EVT_TIMER | EVT_NOTIFY_SIGNAL,
|
|
|
|
TPL_NOTIFY,
|
|
|
|
TerminalConInTimerHandler,
|
|
|
|
TerminalDevice,
|
|
|
|
&TerminalDevice->TimerEvent
|
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
|
|
|
Status = gBS->SetTimer (
|
|
|
|
TerminalDevice->TimerEvent,
|
|
|
|
TimerPeriodic,
|
|
|
|
KEYBOARD_TIMER_INTERVAL
|
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
|
|
|
Status = gBS->CreateEvent (
|
|
|
|
EVT_TIMER,
|
|
|
|
TPL_CALLBACK,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
&TerminalDevice->TwoSecondTimeOut
|
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
}
|
|
|
|
|
2017-01-10 06:05:46 +01:00
|
|
|
/**
|
|
|
|
Initialize the controller name table.
|
|
|
|
|
|
|
|
@param TerminalType The terminal type.
|
|
|
|
@param ControllerNameTable The controller name table.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The controller name table is initialized successfully.
|
|
|
|
@retval others Return status of AddUnicodeString2 ().
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
InitializeControllerNameTable (
|
|
|
|
TERMINAL_TYPE TerminalType,
|
|
|
|
EFI_UNICODE_STRING_TABLE **ControllerNameTable
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_UNICODE_STRING_TABLE *Table;
|
|
|
|
|
|
|
|
ASSERT (TerminalType < ARRAY_SIZE (mTerminalType));
|
|
|
|
Table = NULL;
|
|
|
|
Status = AddUnicodeString2 (
|
|
|
|
"eng",
|
|
|
|
gTerminalComponentName.SupportedLanguages,
|
|
|
|
&Table,
|
|
|
|
mSerialConsoleNames[TerminalType],
|
|
|
|
TRUE
|
|
|
|
);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
Status = AddUnicodeString2 (
|
|
|
|
"en",
|
|
|
|
gTerminalComponentName2.SupportedLanguages,
|
|
|
|
&Table,
|
|
|
|
mSerialConsoleNames[TerminalType],
|
|
|
|
FALSE
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
FreeUnicodeStringTable (Table);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
*ControllerNameTable = Table;
|
|
|
|
}
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
2008-07-01 08:34:04 +02:00
|
|
|
/**
|
2008-07-18 10:20:08 +02:00
|
|
|
Start this driver on Controller by opening a Serial IO protocol,
|
2008-07-03 07:17:49 +02:00
|
|
|
reading Device Path, and creating a child handle with a Simple Text In,
|
|
|
|
Simple Text In Ex and Simple Text Out protocol, and device path protocol.
|
|
|
|
And store Console Device Environment Variables.
|
2008-07-01 08:34:04 +02:00
|
|
|
|
2008-07-03 07:17:49 +02:00
|
|
|
@param This Protocol instance pointer.
|
2008-07-18 10:20:08 +02:00
|
|
|
@param Controller Handle of device to bind driver to
|
2008-07-03 07:17:49 +02:00
|
|
|
@param RemainingDevicePath Optional parameter use to pick a specific child
|
|
|
|
device to start.
|
2008-07-01 08:34:04 +02:00
|
|
|
|
2008-07-18 10:20:08 +02:00
|
|
|
@retval EFI_SUCCESS This driver is added to Controller.
|
|
|
|
@retval EFI_ALREADY_STARTED This driver is already running on Controller.
|
|
|
|
@retval other This driver does not support this device.
|
2008-07-01 08:34:04 +02:00
|
|
|
|
|
|
|
**/
|
2007-07-05 09:05:28 +02:00
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
TerminalDriverBindingStart (
|
|
|
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
|
|
|
IN EFI_HANDLE Controller,
|
|
|
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_SERIAL_IO_PROTOCOL *SerialIo;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *ParentDevicePath;
|
2017-01-10 07:04:00 +01:00
|
|
|
EFI_DEVICE_PATH_PROTOCOL *Vendor;
|
|
|
|
EFI_HANDLE SerialIoHandle;
|
2007-07-05 09:05:28 +02:00
|
|
|
EFI_SERIAL_IO_MODE *Mode;
|
|
|
|
UINTN SerialInTimeOut;
|
|
|
|
TERMINAL_DEV *TerminalDevice;
|
2017-01-10 07:04:00 +01:00
|
|
|
UINT8 TerminalType;
|
2007-07-05 09:05:28 +02:00
|
|
|
EFI_OPEN_PROTOCOL_INFORMATION_ENTRY *OpenInfoBuffer;
|
|
|
|
UINTN EntryCount;
|
|
|
|
UINTN Index;
|
2008-12-17 13:07:02 +01:00
|
|
|
EFI_SIMPLE_TEXT_OUTPUT_PROTOCOL *SimpleTextOutput;
|
2010-01-28 13:29:25 +01:00
|
|
|
EFI_SIMPLE_TEXT_INPUT_PROTOCOL *SimpleTextInput;
|
2017-01-10 07:04:00 +01:00
|
|
|
EFI_UNICODE_STRING_TABLE *ControllerNameTable;
|
|
|
|
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
|
|
|
// Get the Device Path Protocol to build the device path of the child device
|
|
|
|
//
|
|
|
|
Status = gBS->OpenProtocol (
|
|
|
|
Controller,
|
|
|
|
&gEfiDevicePathProtocolGuid,
|
|
|
|
(VOID **) &ParentDevicePath,
|
|
|
|
This->DriverBindingHandle,
|
|
|
|
Controller,
|
|
|
|
EFI_OPEN_PROTOCOL_BY_DRIVER
|
|
|
|
);
|
2017-01-10 07:04:00 +01:00
|
|
|
ASSERT ((Status == EFI_SUCCESS) || (Status == EFI_ALREADY_STARTED));
|
2018-04-13 06:13:52 +02:00
|
|
|
if (EFI_ERROR (Status) && (Status != EFI_ALREADY_STARTED)) {
|
|
|
|
return Status;
|
|
|
|
}
|
2007-07-05 09:05:28 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Open the Serial I/O Protocol BY_DRIVER. It might already be started.
|
|
|
|
//
|
|
|
|
Status = gBS->OpenProtocol (
|
|
|
|
Controller,
|
|
|
|
&gEfiSerialIoProtocolGuid,
|
|
|
|
(VOID **) &SerialIo,
|
|
|
|
This->DriverBindingHandle,
|
|
|
|
Controller,
|
|
|
|
EFI_OPEN_PROTOCOL_BY_DRIVER
|
|
|
|
);
|
2017-01-10 07:04:00 +01:00
|
|
|
ASSERT ((Status == EFI_SUCCESS) || (Status == EFI_ALREADY_STARTED));
|
2018-04-13 06:13:52 +02:00
|
|
|
if (EFI_ERROR (Status) && (Status != EFI_ALREADY_STARTED)) {
|
|
|
|
return Status;
|
|
|
|
}
|
2007-07-05 09:05:28 +02:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
if (!IsHotPlugDevice (ParentDevicePath)) {
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
// if the serial device is a hot plug device, do not update the
|
|
|
|
// ConInDev, ConOutDev, and StdErrDev variables.
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
TerminalUpdateConsoleDevVariable (EFI_CON_IN_DEV_VARIABLE_NAME, ParentDevicePath);
|
|
|
|
TerminalUpdateConsoleDevVariable (EFI_CON_OUT_DEV_VARIABLE_NAME, ParentDevicePath);
|
|
|
|
TerminalUpdateConsoleDevVariable (EFI_ERR_OUT_DEV_VARIABLE_NAME, ParentDevicePath);
|
2007-07-05 09:05:28 +02:00
|
|
|
}
|
2010-01-28 13:29:25 +01:00
|
|
|
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
// Do not create any child for END remaining device path.
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
if ((RemainingDevicePath != NULL) && IsDevicePathEnd (RemainingDevicePath)) {
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
2007-07-05 09:05:28 +02:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
if (Status == EFI_ALREADY_STARTED) {
|
|
|
|
|
|
|
|
if (RemainingDevicePath == NULL) {
|
2010-01-28 13:29:25 +01:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
// If RemainingDevicePath is NULL or is the End of Device Path Node
|
2010-01-28 13:29:25 +01:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
return EFI_SUCCESS;
|
2007-07-05 09:05:28 +02:00
|
|
|
}
|
2010-01-28 13:29:25 +01:00
|
|
|
|
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
// This driver can only produce one child per serial port.
|
|
|
|
// Change its terminal type as remaining device path requests.
|
2010-01-28 13:29:25 +01:00
|
|
|
//
|
|
|
|
Status = gBS->OpenProtocolInformation (
|
|
|
|
Controller,
|
|
|
|
&gEfiSerialIoProtocolGuid,
|
|
|
|
&OpenInfoBuffer,
|
|
|
|
&EntryCount
|
|
|
|
);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
2017-01-10 07:04:00 +01:00
|
|
|
Status = EFI_NOT_FOUND;
|
2010-01-28 13:29:25 +01:00
|
|
|
for (Index = 0; Index < EntryCount; Index++) {
|
|
|
|
if ((OpenInfoBuffer[Index].Attributes & EFI_OPEN_PROTOCOL_BY_CHILD_CONTROLLER) != 0) {
|
2017-01-10 07:04:00 +01:00
|
|
|
Status = gBS->OpenProtocol (
|
|
|
|
OpenInfoBuffer[Index].ControllerHandle,
|
|
|
|
&gEfiSimpleTextInProtocolGuid,
|
|
|
|
(VOID **) &SimpleTextInput,
|
|
|
|
This->DriverBindingHandle,
|
|
|
|
Controller,
|
|
|
|
EFI_OPEN_PROTOCOL_GET_PROTOCOL
|
|
|
|
);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
TerminalDevice = TERMINAL_CON_IN_DEV_FROM_THIS (SimpleTextInput);
|
|
|
|
TerminalType = TerminalTypeFromGuid (&((VENDOR_DEVICE_PATH *) RemainingDevicePath)->Guid);
|
|
|
|
ASSERT (TerminalType < ARRAY_SIZE (mTerminalType));
|
|
|
|
if (TerminalDevice->TerminalType != TerminalType) {
|
|
|
|
Status = InitializeControllerNameTable (TerminalType, &ControllerNameTable);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
StopTerminalStateMachine (TerminalDevice);
|
|
|
|
//
|
|
|
|
// Update the device path
|
|
|
|
//
|
|
|
|
Vendor = TerminalDevice->DevicePath;
|
|
|
|
Status = gBS->LocateDevicePath (&gEfiSerialIoProtocolGuid, &Vendor, &SerialIoHandle);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
CopyGuid (&((VENDOR_DEVICE_PATH *) Vendor)->Guid, mTerminalType[TerminalType]);
|
|
|
|
Status = gBS->ReinstallProtocolInterface (
|
|
|
|
TerminalDevice->Handle,
|
|
|
|
&gEfiDevicePathProtocolGuid,
|
|
|
|
TerminalDevice->DevicePath,
|
|
|
|
TerminalDevice->DevicePath
|
|
|
|
);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
TerminalDevice->TerminalType = TerminalType;
|
|
|
|
StartTerminalStateMachine (TerminalDevice);
|
|
|
|
FreeUnicodeStringTable (TerminalDevice->ControllerNameTable);
|
|
|
|
TerminalDevice->ControllerNameTable = ControllerNameTable;
|
|
|
|
} else {
|
|
|
|
//
|
|
|
|
// Restore the device path on failure
|
|
|
|
//
|
|
|
|
CopyGuid (&((VENDOR_DEVICE_PATH *) Vendor)->Guid, mTerminalType[TerminalDevice->TerminalType]);
|
|
|
|
FreeUnicodeStringTable (ControllerNameTable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2010-01-28 13:29:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
FreePool (OpenInfoBuffer);
|
|
|
|
}
|
2017-01-10 07:04:00 +01:00
|
|
|
return Status;
|
|
|
|
}
|
2010-01-28 13:29:25 +01:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
//
|
|
|
|
// Initialize the Terminal Dev
|
|
|
|
//
|
|
|
|
TerminalDevice = AllocateCopyPool (sizeof (TERMINAL_DEV), &mTerminalDevTemplate);
|
|
|
|
if (TerminalDevice == NULL) {
|
|
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
|
|
goto CloseProtocols;
|
|
|
|
}
|
2017-01-10 06:11:25 +01:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
if (RemainingDevicePath == NULL) {
|
2010-01-28 13:29:25 +01:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
// If RemainingDevicePath is NULL, use default terminal type
|
2010-01-28 13:29:25 +01:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
TerminalDevice->TerminalType = PcdGet8 (PcdDefaultTerminalType);
|
|
|
|
} else {
|
2010-01-28 13:29:25 +01:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
// End of Device Path Node is handled in above.
|
2010-01-28 13:29:25 +01:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
ASSERT (!IsDevicePathEnd (RemainingDevicePath));
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
// If RemainingDevicePath isn't the End of Device Path Node,
|
|
|
|
// Use the RemainingDevicePath to determine the terminal type
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
TerminalDevice->TerminalType = TerminalTypeFromGuid (&((VENDOR_DEVICE_PATH *) RemainingDevicePath)->Guid);
|
|
|
|
}
|
|
|
|
ASSERT (TerminalDevice->TerminalType < ARRAY_SIZE (mTerminalType));
|
|
|
|
TerminalDevice->SerialIo = SerialIo;
|
2010-12-30 23:29:54 +01:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
//
|
|
|
|
// Build the component name for the child device
|
|
|
|
//
|
|
|
|
Status = InitializeControllerNameTable (TerminalDevice->TerminalType, &TerminalDevice->ControllerNameTable);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
goto FreeResources;
|
|
|
|
}
|
2010-12-30 23:29:54 +01:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
//
|
|
|
|
// Build the device path for the child device
|
|
|
|
//
|
|
|
|
Status = SetTerminalDevicePath (TerminalDevice->TerminalType, ParentDevicePath, &TerminalDevice->DevicePath);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
goto FreeResources;
|
|
|
|
}
|
2010-12-30 23:29:54 +01:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
InitializeListHead (&TerminalDevice->NotifyList);
|
|
|
|
Status = gBS->CreateEvent (
|
|
|
|
EVT_NOTIFY_WAIT,
|
|
|
|
TPL_NOTIFY,
|
|
|
|
TerminalConInWaitForKeyEx,
|
|
|
|
TerminalDevice,
|
|
|
|
&TerminalDevice->SimpleInputEx.WaitForKeyEx
|
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
2016-12-19 15:21:11 +01:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
Status = gBS->CreateEvent (
|
|
|
|
EVT_NOTIFY_WAIT,
|
|
|
|
TPL_NOTIFY,
|
|
|
|
TerminalConInWaitForKey,
|
|
|
|
TerminalDevice,
|
|
|
|
&TerminalDevice->SimpleInput.WaitForKey
|
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
Status = gBS->CreateEvent (
|
|
|
|
EVT_NOTIFY_SIGNAL,
|
|
|
|
TPL_CALLBACK,
|
|
|
|
KeyNotifyProcessHandler,
|
|
|
|
TerminalDevice,
|
|
|
|
&TerminalDevice->KeyNotifyProcessEvent
|
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
2007-07-05 09:05:28 +02:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
//
|
|
|
|
// Allocates and initializes the FIFO buffer to be zero, used for accommodating
|
|
|
|
// the pre-read pending characters.
|
|
|
|
//
|
|
|
|
TerminalDevice->RawFiFo = AllocateZeroPool (sizeof (RAW_DATA_FIFO));
|
|
|
|
if (TerminalDevice->RawFiFo == NULL) {
|
|
|
|
goto FreeResources;
|
|
|
|
}
|
|
|
|
TerminalDevice->UnicodeFiFo = AllocateZeroPool (sizeof (UNICODE_FIFO));
|
|
|
|
if (TerminalDevice->UnicodeFiFo == NULL) {
|
|
|
|
goto FreeResources;
|
|
|
|
}
|
|
|
|
TerminalDevice->EfiKeyFiFo = AllocateZeroPool (sizeof (EFI_KEY_FIFO));
|
|
|
|
if (TerminalDevice->EfiKeyFiFo == NULL) {
|
|
|
|
goto FreeResources;
|
|
|
|
}
|
|
|
|
TerminalDevice->EfiKeyFiFoForNotify = AllocateZeroPool (sizeof (EFI_KEY_FIFO));
|
|
|
|
if (TerminalDevice->EfiKeyFiFoForNotify == NULL) {
|
|
|
|
goto FreeResources;
|
2007-07-05 09:05:28 +02:00
|
|
|
}
|
|
|
|
|
2008-01-17 06:56:45 +01:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
// Set the timeout value of serial buffer for keystroke response performance issue
|
2008-01-17 06:56:45 +01:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
Mode = TerminalDevice->SerialIo->Mode;
|
2008-01-17 06:56:45 +01:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
SerialInTimeOut = 0;
|
|
|
|
if (Mode->BaudRate != 0) {
|
|
|
|
SerialInTimeOut = (1 + Mode->DataBits + Mode->StopBits) * 2 * 1000000 / (UINTN) Mode->BaudRate;
|
|
|
|
}
|
2007-07-05 09:05:28 +02:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
Status = TerminalDevice->SerialIo->SetAttributes (
|
|
|
|
TerminalDevice->SerialIo,
|
|
|
|
Mode->BaudRate,
|
|
|
|
Mode->ReceiveFifoDepth,
|
|
|
|
(UINT32) SerialInTimeOut,
|
|
|
|
(EFI_PARITY_TYPE) (Mode->Parity),
|
|
|
|
(UINT8) Mode->DataBits,
|
|
|
|
(EFI_STOP_BITS_TYPE) (Mode->StopBits)
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
//
|
|
|
|
// if set attributes operation fails, invalidate
|
|
|
|
// the value of SerialInTimeOut,thus make it
|
|
|
|
// inconsistent with the default timeout value
|
|
|
|
// of serial buffer. This will invoke the recalculation
|
|
|
|
// in the readkeystroke routine.
|
|
|
|
//
|
|
|
|
TerminalDevice->SerialInTimeOut = 0;
|
2010-01-28 13:29:25 +01:00
|
|
|
} else {
|
2017-01-10 07:04:00 +01:00
|
|
|
TerminalDevice->SerialInTimeOut = SerialInTimeOut;
|
2007-07-05 09:05:28 +02:00
|
|
|
}
|
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
SimpleTextOutput = &TerminalDevice->SimpleTextOutput;
|
|
|
|
SimpleTextInput = &TerminalDevice->SimpleInput;
|
|
|
|
|
2010-01-28 13:29:25 +01:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
// Initialize SimpleTextOut instance
|
2010-01-28 13:29:25 +01:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
SimpleTextOutput->Mode = &TerminalDevice->SimpleTextOutputMode;
|
|
|
|
TerminalDevice->TerminalConsoleModeData = InitializeTerminalConsoleTextMode (
|
|
|
|
&SimpleTextOutput->Mode->MaxMode
|
|
|
|
);
|
|
|
|
if (TerminalDevice->TerminalConsoleModeData == NULL) {
|
|
|
|
goto FreeResources;
|
2007-07-05 09:05:28 +02:00
|
|
|
}
|
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
// For terminal devices, cursor is always visible
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
SimpleTextOutput->Mode->CursorVisible = TRUE;
|
|
|
|
Status = SimpleTextOutput->SetAttribute (SimpleTextOutput, EFI_TEXT_ATTR (EFI_LIGHTGRAY, EFI_BLACK));
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
Status = SimpleTextOutput->Reset (SimpleTextOutput, FALSE);
|
|
|
|
}
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
goto ReportError;
|
2007-07-05 09:05:28 +02:00
|
|
|
}
|
2009-02-25 08:54:57 +01:00
|
|
|
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
// Initialize SimpleTextInput instance
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
2017-01-10 07:04:00 +01:00
|
|
|
Status = SimpleTextInput->Reset (SimpleTextInput, FALSE);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
goto ReportError;
|
2007-07-05 09:05:28 +02:00
|
|
|
}
|
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
Status = gBS->InstallMultipleProtocolInterfaces (
|
|
|
|
&TerminalDevice->Handle,
|
|
|
|
&gEfiSimpleTextInProtocolGuid, &TerminalDevice->SimpleInput,
|
|
|
|
&gEfiSimpleTextInputExProtocolGuid, &TerminalDevice->SimpleInputEx,
|
|
|
|
&gEfiSimpleTextOutProtocolGuid, &TerminalDevice->SimpleTextOutput,
|
|
|
|
&gEfiDevicePathProtocolGuid, TerminalDevice->DevicePath,
|
|
|
|
NULL
|
|
|
|
);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
Status = gBS->OpenProtocol (
|
|
|
|
Controller,
|
|
|
|
&gEfiSerialIoProtocolGuid,
|
|
|
|
(VOID **) &TerminalDevice->SerialIo,
|
|
|
|
This->DriverBindingHandle,
|
|
|
|
TerminalDevice->Handle,
|
|
|
|
EFI_OPEN_PROTOCOL_BY_CHILD_CONTROLLER
|
2010-01-28 13:29:25 +01:00
|
|
|
);
|
2017-01-10 07:04:00 +01:00
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
StartTerminalStateMachine (TerminalDevice);
|
|
|
|
return Status;
|
2010-01-28 13:29:25 +01:00
|
|
|
}
|
2007-07-05 09:05:28 +02:00
|
|
|
|
|
|
|
ReportError:
|
|
|
|
REPORT_STATUS_CODE_WITH_DEVICE_PATH (
|
|
|
|
EFI_ERROR_CODE | EFI_ERROR_MINOR,
|
2010-01-27 05:00:58 +01:00
|
|
|
(EFI_PERIPHERAL_REMOTE_CONSOLE | EFI_P_EC_CONTROLLER_ERROR),
|
2017-01-10 07:04:00 +01:00
|
|
|
ParentDevicePath
|
2007-07-05 09:05:28 +02:00
|
|
|
);
|
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
FreeResources:
|
|
|
|
ASSERT (TerminalDevice != NULL);
|
2007-07-05 09:05:28 +02:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
if (TerminalDevice->SimpleInput.WaitForKey != NULL) {
|
|
|
|
gBS->CloseEvent (TerminalDevice->SimpleInput.WaitForKey);
|
|
|
|
}
|
|
|
|
if (TerminalDevice->SimpleInputEx.WaitForKeyEx != NULL) {
|
|
|
|
gBS->CloseEvent (TerminalDevice->SimpleInputEx.WaitForKeyEx);
|
|
|
|
}
|
|
|
|
if (TerminalDevice->KeyNotifyProcessEvent != NULL) {
|
|
|
|
gBS->CloseEvent (TerminalDevice->KeyNotifyProcessEvent);
|
|
|
|
}
|
2011-04-19 08:51:44 +02:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
if (TerminalDevice->RawFiFo != NULL) {
|
|
|
|
FreePool (TerminalDevice->RawFiFo);
|
|
|
|
}
|
|
|
|
if (TerminalDevice->UnicodeFiFo != NULL) {
|
|
|
|
FreePool (TerminalDevice->UnicodeFiFo);
|
|
|
|
}
|
|
|
|
if (TerminalDevice->EfiKeyFiFo != NULL) {
|
|
|
|
FreePool (TerminalDevice->EfiKeyFiFo);
|
|
|
|
}
|
|
|
|
if (TerminalDevice->EfiKeyFiFoForNotify != NULL) {
|
|
|
|
FreePool (TerminalDevice->EfiKeyFiFoForNotify);
|
|
|
|
}
|
2007-07-05 09:05:28 +02:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
if (TerminalDevice->ControllerNameTable != NULL) {
|
|
|
|
FreeUnicodeStringTable (TerminalDevice->ControllerNameTable);
|
|
|
|
}
|
2007-10-19 04:35:29 +02:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
if (TerminalDevice->DevicePath != NULL) {
|
|
|
|
FreePool (TerminalDevice->DevicePath);
|
|
|
|
}
|
2007-10-19 04:35:29 +02:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
if (TerminalDevice->TerminalConsoleModeData != NULL) {
|
|
|
|
FreePool (TerminalDevice->TerminalConsoleModeData);
|
|
|
|
}
|
2007-07-05 09:05:28 +02:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
FreePool (TerminalDevice);
|
2007-07-05 09:05:28 +02:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
CloseProtocols:
|
2011-11-25 09:35:00 +01:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
//
|
|
|
|
// Remove Parent Device Path from
|
|
|
|
// the Console Device Environment Variables
|
|
|
|
//
|
|
|
|
TerminalRemoveConsoleDevVariable (EFI_CON_IN_DEV_VARIABLE_NAME, ParentDevicePath);
|
|
|
|
TerminalRemoveConsoleDevVariable (EFI_CON_OUT_DEV_VARIABLE_NAME, ParentDevicePath);
|
|
|
|
TerminalRemoveConsoleDevVariable (EFI_ERR_OUT_DEV_VARIABLE_NAME, ParentDevicePath);
|
2007-07-05 09:05:28 +02:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
Status = gBS->CloseProtocol (
|
|
|
|
Controller,
|
|
|
|
&gEfiSerialIoProtocolGuid,
|
|
|
|
This->DriverBindingHandle,
|
|
|
|
Controller
|
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
2007-07-05 09:05:28 +02:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
Status = gBS->CloseProtocol (
|
|
|
|
Controller,
|
|
|
|
&gEfiDevicePathProtocolGuid,
|
|
|
|
This->DriverBindingHandle,
|
|
|
|
Controller
|
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
2007-07-05 09:05:28 +02:00
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
2008-07-01 08:34:04 +02:00
|
|
|
/**
|
2008-07-18 10:20:08 +02:00
|
|
|
Stop this driver on Controller by closing Simple Text In, Simple Text
|
2008-07-03 07:17:49 +02:00
|
|
|
In Ex, Simple Text Out protocol, and removing parent device path from
|
2009-01-08 09:46:31 +01:00
|
|
|
Console Device Environment Variables.
|
2008-07-01 08:34:04 +02:00
|
|
|
|
2008-07-03 07:17:49 +02:00
|
|
|
@param This Protocol instance pointer.
|
2008-07-18 10:20:08 +02:00
|
|
|
@param Controller Handle of device to stop driver on
|
2008-07-03 07:17:49 +02:00
|
|
|
@param NumberOfChildren Number of Handles in ChildHandleBuffer. If number of
|
|
|
|
children is zero stop the entire bus driver.
|
|
|
|
@param ChildHandleBuffer List of Child Handles to Stop.
|
2008-07-01 08:34:04 +02:00
|
|
|
|
2008-07-18 10:20:08 +02:00
|
|
|
@retval EFI_SUCCESS This driver is removed Controller.
|
2008-07-03 07:17:49 +02:00
|
|
|
@retval other This driver could not be removed from this device.
|
2008-07-01 08:34:04 +02:00
|
|
|
|
|
|
|
**/
|
2007-07-05 09:05:28 +02:00
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
TerminalDriverBindingStop (
|
|
|
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
|
|
|
IN EFI_HANDLE Controller,
|
|
|
|
IN UINTN NumberOfChildren,
|
|
|
|
IN EFI_HANDLE *ChildHandleBuffer
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN Index;
|
|
|
|
BOOLEAN AllChildrenStopped;
|
|
|
|
EFI_SIMPLE_TEXT_OUTPUT_PROTOCOL *SimpleTextOutput;
|
|
|
|
TERMINAL_DEV *TerminalDevice;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *ParentDevicePath;
|
|
|
|
EFI_SERIAL_IO_PROTOCOL *SerialIo;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Complete all outstanding transactions to Controller.
|
|
|
|
// Don't allow any new transaction to Controller to be started.
|
|
|
|
//
|
|
|
|
if (NumberOfChildren == 0) {
|
|
|
|
//
|
|
|
|
// Close the bus driver
|
|
|
|
//
|
|
|
|
Status = gBS->OpenProtocol (
|
|
|
|
Controller,
|
2017-01-10 07:04:00 +01:00
|
|
|
&gEfiDevicePathProtocolGuid,
|
2007-07-05 09:05:28 +02:00
|
|
|
(VOID **) &ParentDevicePath,
|
|
|
|
This->DriverBindingHandle,
|
|
|
|
Controller,
|
|
|
|
EFI_OPEN_PROTOCOL_GET_PROTOCOL
|
|
|
|
);
|
2017-01-10 07:04:00 +01:00
|
|
|
ASSERT_EFI_ERROR (Status);
|
2007-07-05 09:05:28 +02:00
|
|
|
|
2017-01-10 07:04:00 +01:00
|
|
|
//
|
|
|
|
// Remove Parent Device Path from
|
|
|
|
// the Console Device Environment Variables
|
|
|
|
//
|
|
|
|
TerminalRemoveConsoleDevVariable (EFI_CON_IN_DEV_VARIABLE_NAME, ParentDevicePath);
|
|
|
|
TerminalRemoveConsoleDevVariable (EFI_CON_OUT_DEV_VARIABLE_NAME, ParentDevicePath);
|
|
|
|
TerminalRemoveConsoleDevVariable (EFI_ERR_OUT_DEV_VARIABLE_NAME, ParentDevicePath);
|
2007-07-05 09:05:28 +02:00
|
|
|
|
|
|
|
gBS->CloseProtocol (
|
|
|
|
Controller,
|
|
|
|
&gEfiSerialIoProtocolGuid,
|
|
|
|
This->DriverBindingHandle,
|
|
|
|
Controller
|
|
|
|
);
|
|
|
|
|
|
|
|
gBS->CloseProtocol (
|
|
|
|
Controller,
|
|
|
|
&gEfiDevicePathProtocolGuid,
|
|
|
|
This->DriverBindingHandle,
|
|
|
|
Controller
|
|
|
|
);
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
AllChildrenStopped = TRUE;
|
|
|
|
|
|
|
|
for (Index = 0; Index < NumberOfChildren; Index++) {
|
|
|
|
|
|
|
|
Status = gBS->OpenProtocol (
|
|
|
|
ChildHandleBuffer[Index],
|
|
|
|
&gEfiSimpleTextOutProtocolGuid,
|
|
|
|
(VOID **) &SimpleTextOutput,
|
|
|
|
This->DriverBindingHandle,
|
|
|
|
ChildHandleBuffer[Index],
|
|
|
|
EFI_OPEN_PROTOCOL_GET_PROTOCOL
|
|
|
|
);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
|
|
|
|
TerminalDevice = TERMINAL_CON_OUT_DEV_FROM_THIS (SimpleTextOutput);
|
|
|
|
|
|
|
|
gBS->CloseProtocol (
|
|
|
|
Controller,
|
|
|
|
&gEfiSerialIoProtocolGuid,
|
|
|
|
This->DriverBindingHandle,
|
|
|
|
ChildHandleBuffer[Index]
|
|
|
|
);
|
|
|
|
|
|
|
|
Status = gBS->UninstallMultipleProtocolInterfaces (
|
|
|
|
ChildHandleBuffer[Index],
|
|
|
|
&gEfiSimpleTextInProtocolGuid,
|
|
|
|
&TerminalDevice->SimpleInput,
|
2007-10-19 04:35:29 +02:00
|
|
|
&gEfiSimpleTextInputExProtocolGuid,
|
|
|
|
&TerminalDevice->SimpleInputEx,
|
2007-07-05 09:05:28 +02:00
|
|
|
&gEfiSimpleTextOutProtocolGuid,
|
|
|
|
&TerminalDevice->SimpleTextOutput,
|
|
|
|
&gEfiDevicePathProtocolGuid,
|
|
|
|
TerminalDevice->DevicePath,
|
|
|
|
NULL
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
gBS->OpenProtocol (
|
|
|
|
Controller,
|
|
|
|
&gEfiSerialIoProtocolGuid,
|
|
|
|
(VOID **) &SerialIo,
|
|
|
|
This->DriverBindingHandle,
|
|
|
|
ChildHandleBuffer[Index],
|
|
|
|
EFI_OPEN_PROTOCOL_BY_CHILD_CONTROLLER
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
|
2017-01-10 06:55:22 +01:00
|
|
|
FreeUnicodeStringTable (TerminalDevice->ControllerNameTable);
|
2017-01-10 06:25:27 +01:00
|
|
|
StopTerminalStateMachine (TerminalDevice);
|
2007-07-05 09:05:28 +02:00
|
|
|
gBS->CloseEvent (TerminalDevice->SimpleInput.WaitForKey);
|
2007-10-19 04:35:29 +02:00
|
|
|
gBS->CloseEvent (TerminalDevice->SimpleInputEx.WaitForKeyEx);
|
2016-12-19 15:21:11 +01:00
|
|
|
gBS->CloseEvent (TerminalDevice->KeyNotifyProcessEvent);
|
2007-10-19 04:35:29 +02:00
|
|
|
TerminalFreeNotifyList (&TerminalDevice->NotifyList);
|
2007-07-05 09:05:28 +02:00
|
|
|
FreePool (TerminalDevice->DevicePath);
|
2017-01-10 06:55:22 +01:00
|
|
|
FreePool (TerminalDevice->TerminalConsoleModeData);
|
2007-07-05 09:05:28 +02:00
|
|
|
FreePool (TerminalDevice);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
AllChildrenStopped = FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!AllChildrenStopped) {
|
|
|
|
return EFI_DEVICE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2017-04-19 04:42:01 +02:00
|
|
|
/**
|
|
|
|
Compare a device path data structure to that of all the nodes of a
|
|
|
|
second device path instance.
|
|
|
|
|
|
|
|
@param Multi A pointer to a multi-instance device path data structure.
|
|
|
|
@param Single A pointer to a single-instance device path data structure.
|
|
|
|
|
|
|
|
@retval TRUE If the Single is contained within Multi.
|
|
|
|
@retval FALSE The Single is not match within Multi.
|
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
MatchDevicePaths (
|
|
|
|
IN EFI_DEVICE_PATH_PROTOCOL *Multi,
|
|
|
|
IN EFI_DEVICE_PATH_PROTOCOL *Single
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *DevicePath;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *DevicePathInst;
|
|
|
|
UINTN Size;
|
|
|
|
|
|
|
|
DevicePath = Multi;
|
|
|
|
DevicePathInst = GetNextDevicePathInstance (&DevicePath, &Size);
|
|
|
|
//
|
|
|
|
// Search for the match of 'Single' in 'Multi'
|
|
|
|
//
|
|
|
|
while (DevicePathInst != NULL) {
|
|
|
|
//
|
|
|
|
// If the single device path is found in multiple device paths,
|
|
|
|
// return success
|
|
|
|
//
|
|
|
|
if (CompareMem (Single, DevicePathInst, Size) == 0) {
|
|
|
|
FreePool (DevicePathInst);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
FreePool (DevicePathInst);
|
|
|
|
DevicePathInst = GetNextDevicePathInstance (&DevicePath, &Size);
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2008-07-03 07:17:49 +02:00
|
|
|
/**
|
|
|
|
Update terminal device path in Console Device Environment Variables.
|
|
|
|
|
|
|
|
@param VariableName The Console Device Environment Variable.
|
2008-12-17 10:34:06 +01:00
|
|
|
@param ParentDevicePath The terminal device path to be updated.
|
2008-07-03 07:17:49 +02:00
|
|
|
|
|
|
|
**/
|
2007-07-05 09:05:28 +02:00
|
|
|
VOID
|
|
|
|
TerminalUpdateConsoleDevVariable (
|
|
|
|
IN CHAR16 *VariableName,
|
|
|
|
IN EFI_DEVICE_PATH_PROTOCOL *ParentDevicePath
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
2014-03-25 04:54:34 +01:00
|
|
|
UINTN NameSize;
|
2007-07-05 09:05:28 +02:00
|
|
|
UINTN VariableSize;
|
2017-01-10 04:23:17 +01:00
|
|
|
TERMINAL_TYPE TerminalType;
|
2007-07-05 09:05:28 +02:00
|
|
|
EFI_DEVICE_PATH_PROTOCOL *Variable;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *NewVariable;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *TempDevicePath;
|
2014-03-25 04:54:34 +01:00
|
|
|
EDKII_SET_VARIABLE_STATUS *SetVariableStatus;
|
2007-07-05 09:05:28 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Get global variable and its size according to the name given.
|
|
|
|
//
|
2017-04-19 04:42:01 +02:00
|
|
|
Status = GetEfiGlobalVariable2 (VariableName, (VOID**)&Variable, NULL);
|
|
|
|
if (Status == EFI_NOT_FOUND) {
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
Variable = NULL;
|
|
|
|
}
|
|
|
|
if (EFI_ERROR (Status)) {
|
2010-01-28 13:29:25 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
|
|
|
// Append terminal device path onto the variable.
|
|
|
|
//
|
2017-01-10 04:33:51 +01:00
|
|
|
for (TerminalType = 0; TerminalType < ARRAY_SIZE (mTerminalType); TerminalType++) {
|
2007-07-05 09:05:28 +02:00
|
|
|
SetTerminalDevicePath (TerminalType, ParentDevicePath, &TempDevicePath);
|
|
|
|
|
|
|
|
if (TempDevicePath != NULL) {
|
2017-04-19 04:42:01 +02:00
|
|
|
if (!MatchDevicePaths (Variable, TempDevicePath)) {
|
|
|
|
NewVariable = AppendDevicePathInstance (Variable, TempDevicePath);
|
|
|
|
if (NewVariable != NULL) {
|
|
|
|
if (Variable != NULL) {
|
|
|
|
FreePool (Variable);
|
|
|
|
}
|
|
|
|
Variable = NewVariable;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-05 09:05:28 +02:00
|
|
|
FreePool (TempDevicePath);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
VariableSize = GetDevicePathSize (Variable);
|
|
|
|
|
|
|
|
Status = gRT->SetVariable (
|
|
|
|
VariableName,
|
|
|
|
&gEfiGlobalVariableGuid,
|
|
|
|
EFI_VARIABLE_BOOTSERVICE_ACCESS | EFI_VARIABLE_RUNTIME_ACCESS,
|
|
|
|
VariableSize,
|
|
|
|
Variable
|
|
|
|
);
|
2014-03-25 04:54:34 +01:00
|
|
|
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
NameSize = StrSize (VariableName);
|
|
|
|
SetVariableStatus = AllocatePool (sizeof (EDKII_SET_VARIABLE_STATUS) + NameSize + VariableSize);
|
|
|
|
if (SetVariableStatus != NULL) {
|
|
|
|
CopyGuid (&SetVariableStatus->Guid, &gEfiGlobalVariableGuid);
|
|
|
|
SetVariableStatus->NameSize = NameSize;
|
|
|
|
SetVariableStatus->DataSize = VariableSize;
|
|
|
|
SetVariableStatus->SetStatus = Status;
|
|
|
|
SetVariableStatus->Attributes = EFI_VARIABLE_BOOTSERVICE_ACCESS | EFI_VARIABLE_RUNTIME_ACCESS;
|
|
|
|
CopyMem (SetVariableStatus + 1, VariableName, NameSize);
|
|
|
|
CopyMem (((UINT8 *) (SetVariableStatus + 1)) + NameSize, Variable, VariableSize);
|
|
|
|
|
|
|
|
REPORT_STATUS_CODE_EX (
|
|
|
|
EFI_ERROR_CODE,
|
|
|
|
PcdGet32 (PcdErrorCodeSetVariable),
|
|
|
|
0,
|
|
|
|
NULL,
|
|
|
|
&gEdkiiStatusCodeDataTypeVariableGuid,
|
|
|
|
SetVariableStatus,
|
|
|
|
sizeof (EDKII_SET_VARIABLE_STATUS) + NameSize + VariableSize
|
|
|
|
);
|
|
|
|
|
|
|
|
FreePool (SetVariableStatus);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-05 09:05:28 +02:00
|
|
|
FreePool (Variable);
|
|
|
|
|
|
|
|
return ;
|
|
|
|
}
|
|
|
|
|
2008-07-01 08:34:04 +02:00
|
|
|
|
|
|
|
/**
|
2008-07-03 07:17:49 +02:00
|
|
|
Remove terminal device path from Console Device Environment Variables.
|
2008-07-01 08:34:04 +02:00
|
|
|
|
2008-07-03 07:17:49 +02:00
|
|
|
@param VariableName Console Device Environment Variables.
|
2008-12-17 10:34:06 +01:00
|
|
|
@param ParentDevicePath The terminal device path to be updated.
|
2008-07-01 08:34:04 +02:00
|
|
|
|
|
|
|
**/
|
2007-07-05 09:05:28 +02:00
|
|
|
VOID
|
|
|
|
TerminalRemoveConsoleDevVariable (
|
|
|
|
IN CHAR16 *VariableName,
|
|
|
|
IN EFI_DEVICE_PATH_PROTOCOL *ParentDevicePath
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
BOOLEAN FoundOne;
|
|
|
|
BOOLEAN Match;
|
|
|
|
UINTN VariableSize;
|
|
|
|
UINTN InstanceSize;
|
2017-01-10 04:23:17 +01:00
|
|
|
TERMINAL_TYPE TerminalType;
|
2007-07-05 09:05:28 +02:00
|
|
|
EFI_DEVICE_PATH_PROTOCOL *Instance;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *Variable;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *OriginalVariable;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *NewVariable;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *SavedNewVariable;
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *TempDevicePath;
|
|
|
|
|
|
|
|
Instance = NULL;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Get global variable and its size according to the name given.
|
|
|
|
//
|
2012-05-31 10:35:48 +02:00
|
|
|
GetEfiGlobalVariable2 (VariableName, (VOID**)&Variable, NULL);
|
2007-07-05 09:05:28 +02:00
|
|
|
if (Variable == NULL) {
|
|
|
|
return ;
|
|
|
|
}
|
|
|
|
|
|
|
|
FoundOne = FALSE;
|
|
|
|
OriginalVariable = Variable;
|
|
|
|
NewVariable = NULL;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Get first device path instance from Variable
|
|
|
|
//
|
|
|
|
Instance = GetNextDevicePathInstance (&Variable, &InstanceSize);
|
|
|
|
if (Instance == NULL) {
|
|
|
|
FreePool (OriginalVariable);
|
|
|
|
return ;
|
|
|
|
}
|
|
|
|
//
|
|
|
|
// Loop through all the device path instances of Variable
|
|
|
|
//
|
|
|
|
do {
|
|
|
|
//
|
|
|
|
// Loop through all the terminal types that this driver supports
|
|
|
|
//
|
|
|
|
Match = FALSE;
|
2017-01-10 04:33:51 +01:00
|
|
|
for (TerminalType = 0; TerminalType < ARRAY_SIZE (mTerminalType); TerminalType++) {
|
2007-07-05 09:05:28 +02:00
|
|
|
|
|
|
|
SetTerminalDevicePath (TerminalType, ParentDevicePath, &TempDevicePath);
|
|
|
|
|
|
|
|
//
|
2008-12-17 10:34:06 +01:00
|
|
|
// Compare the generated device path to the current device path instance
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
|
|
|
if (TempDevicePath != NULL) {
|
|
|
|
if (CompareMem (Instance, TempDevicePath, InstanceSize) == 0) {
|
|
|
|
Match = TRUE;
|
|
|
|
FoundOne = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
FreePool (TempDevicePath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//
|
|
|
|
// If a match was not found, then keep the current device path instance
|
|
|
|
//
|
|
|
|
if (!Match) {
|
|
|
|
SavedNewVariable = NewVariable;
|
|
|
|
NewVariable = AppendDevicePathInstance (NewVariable, Instance);
|
|
|
|
if (SavedNewVariable != NULL) {
|
|
|
|
FreePool (SavedNewVariable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//
|
|
|
|
// Get next device path instance from Variable
|
|
|
|
//
|
|
|
|
FreePool (Instance);
|
|
|
|
Instance = GetNextDevicePathInstance (&Variable, &InstanceSize);
|
|
|
|
} while (Instance != NULL);
|
|
|
|
|
|
|
|
FreePool (OriginalVariable);
|
|
|
|
|
|
|
|
if (FoundOne) {
|
|
|
|
VariableSize = GetDevicePathSize (NewVariable);
|
|
|
|
|
|
|
|
Status = gRT->SetVariable (
|
|
|
|
VariableName,
|
|
|
|
&gEfiGlobalVariableGuid,
|
|
|
|
EFI_VARIABLE_BOOTSERVICE_ACCESS | EFI_VARIABLE_RUNTIME_ACCESS,
|
|
|
|
VariableSize,
|
|
|
|
NewVariable
|
|
|
|
);
|
2014-03-25 04:54:34 +01:00
|
|
|
//
|
|
|
|
// Shrinking variable with existing variable driver implementation shouldn't fail.
|
|
|
|
//
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
2007-07-05 09:05:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (NewVariable != NULL) {
|
|
|
|
FreePool (NewVariable);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ;
|
|
|
|
}
|
|
|
|
|
2008-07-03 07:17:49 +02:00
|
|
|
/**
|
2008-12-17 10:34:06 +01:00
|
|
|
Build terminal device path according to terminal type.
|
2008-07-03 07:17:49 +02:00
|
|
|
|
|
|
|
@param TerminalType The terminal type is PC ANSI, VT100, VT100+ or VT-UTF8.
|
2008-12-17 10:34:06 +01:00
|
|
|
@param ParentDevicePath Parent device path.
|
2008-07-03 07:17:49 +02:00
|
|
|
@param TerminalDevicePath Returned terminal device path, if building successfully.
|
|
|
|
|
|
|
|
@retval EFI_UNSUPPORTED Terminal does not belong to the supported type.
|
|
|
|
@retval EFI_OUT_OF_RESOURCES Generate terminal device path failed.
|
|
|
|
@retval EFI_SUCCESS Build terminal device path successfully.
|
|
|
|
|
|
|
|
**/
|
2007-07-05 09:05:28 +02:00
|
|
|
EFI_STATUS
|
|
|
|
SetTerminalDevicePath (
|
2017-01-10 04:23:17 +01:00
|
|
|
IN TERMINAL_TYPE TerminalType,
|
2007-07-05 09:05:28 +02:00
|
|
|
IN EFI_DEVICE_PATH_PROTOCOL *ParentDevicePath,
|
|
|
|
OUT EFI_DEVICE_PATH_PROTOCOL **TerminalDevicePath
|
|
|
|
)
|
|
|
|
{
|
|
|
|
VENDOR_DEVICE_PATH Node;
|
|
|
|
|
2017-01-10 06:17:35 +01:00
|
|
|
ASSERT (TerminalType < ARRAY_SIZE (mTerminalType));
|
2007-07-05 09:05:28 +02:00
|
|
|
Node.Header.Type = MESSAGING_DEVICE_PATH;
|
|
|
|
Node.Header.SubType = MSG_VENDOR_DP;
|
2017-01-12 02:45:31 +01:00
|
|
|
SetDevicePathNodeLength (&Node.Header, sizeof (VENDOR_DEVICE_PATH));
|
2017-01-10 06:17:35 +01:00
|
|
|
CopyGuid (&Node.Guid, mTerminalType[TerminalType]);
|
2008-07-18 10:20:08 +02:00
|
|
|
|
2007-07-05 09:05:28 +02:00
|
|
|
//
|
2008-07-18 10:20:08 +02:00
|
|
|
// Append the terminal node onto parent device path
|
2007-07-05 09:05:28 +02:00
|
|
|
// to generate a complete terminal device path.
|
|
|
|
//
|
|
|
|
*TerminalDevicePath = AppendDevicePathNode (
|
|
|
|
ParentDevicePath,
|
|
|
|
(EFI_DEVICE_PATH_PROTOCOL *) &Node
|
|
|
|
);
|
|
|
|
if (*TerminalDevicePath == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2007-07-10 01:39:54 +02:00
|
|
|
/**
|
|
|
|
The user Entry Point for module Terminal. The user code starts with this function.
|
|
|
|
|
2008-07-18 10:20:08 +02:00
|
|
|
@param ImageHandle The firmware allocated handle for the EFI image.
|
|
|
|
@param SystemTable A pointer to the EFI System Table.
|
2008-04-10 10:49:28 +02:00
|
|
|
|
2007-07-10 01:39:54 +02:00
|
|
|
@retval EFI_SUCCESS The entry point is executed successfully.
|
|
|
|
@retval other Some error occurs when executing this entry point.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
InitializeTerminal(
|
|
|
|
IN EFI_HANDLE ImageHandle,
|
|
|
|
IN EFI_SYSTEM_TABLE *SystemTable
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Install driver model protocol(s).
|
|
|
|
//
|
2007-09-30 05:05:10 +02:00
|
|
|
Status = EfiLibInstallDriverBindingComponentName2 (
|
2007-07-10 01:39:54 +02:00
|
|
|
ImageHandle,
|
|
|
|
SystemTable,
|
|
|
|
&gTerminalDriverBinding,
|
|
|
|
ImageHandle,
|
|
|
|
&gTerminalComponentName,
|
2007-09-30 05:05:10 +02:00
|
|
|
&gTerminalComponentName2
|
2007-07-10 01:39:54 +02:00
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (Status);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
2009-02-25 08:54:57 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
Check if the device supports hot-plug through its device path.
|
|
|
|
|
|
|
|
This function could be updated to check more types of Hot Plug devices.
|
|
|
|
Currently, it checks USB and PCCard device.
|
|
|
|
|
|
|
|
@param DevicePath Pointer to device's device path.
|
|
|
|
|
|
|
|
@retval TRUE The devcie is a hot-plug device
|
|
|
|
@retval FALSE The devcie is not a hot-plug device.
|
|
|
|
|
|
|
|
**/
|
|
|
|
BOOLEAN
|
|
|
|
IsHotPlugDevice (
|
|
|
|
IN EFI_DEVICE_PATH_PROTOCOL *DevicePath
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_DEVICE_PATH_PROTOCOL *CheckDevicePath;
|
|
|
|
|
|
|
|
CheckDevicePath = DevicePath;
|
|
|
|
while (!IsDevicePathEnd (CheckDevicePath)) {
|
|
|
|
//
|
|
|
|
// Check device whether is hot plug device or not throught Device Path
|
2010-01-28 13:29:25 +01:00
|
|
|
//
|
2009-02-25 08:54:57 +01:00
|
|
|
if ((DevicePathType (CheckDevicePath) == MESSAGING_DEVICE_PATH) &&
|
|
|
|
(DevicePathSubType (CheckDevicePath) == MSG_USB_DP ||
|
|
|
|
DevicePathSubType (CheckDevicePath) == MSG_USB_CLASS_DP ||
|
|
|
|
DevicePathSubType (CheckDevicePath) == MSG_USB_WWID_DP)) {
|
|
|
|
//
|
|
|
|
// If Device is USB device
|
|
|
|
//
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
if ((DevicePathType (CheckDevicePath) == HARDWARE_DEVICE_PATH) &&
|
|
|
|
(DevicePathSubType (CheckDevicePath) == HW_PCCARD_DP)) {
|
|
|
|
//
|
|
|
|
// If Device is PCCard
|
|
|
|
//
|
|
|
|
return TRUE;
|
|
|
|
}
|
2010-01-28 13:29:25 +01:00
|
|
|
|
2009-02-25 08:54:57 +01:00
|
|
|
CheckDevicePath = NextDevicePathNode (CheckDevicePath);
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|