1 /** @file
2 USB Keyboard Driver that manages USB keyboard and produces Simple Text Input
3 Protocol and Simple Text Input Ex Protocol.
4
5 Copyright (c) 2004 - 2012, Intel Corporation. All rights reserved.<BR>
6 This program and the accompanying materials
7 are licensed and made available under the terms and conditions of the BSD License
8 which accompanies this distribution. The full text of the license may be found at
9 http://opensource.org/licenses/bsd-license.php
10
11 THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
12 WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
13
14 **/
15
16 #include "EfiKey.h"
17 #include "KeyBoard.h"
18
19 //
20 // USB Keyboard Driver Global Variables
21 //
22 EFI_DRIVER_BINDING_PROTOCOL gUsbKeyboardDriverBinding = {
23 USBKeyboardDriverBindingSupported,
24 USBKeyboardDriverBindingStart,
25 USBKeyboardDriverBindingStop,
26 0xa,
27 NULL,
28 NULL
29 };
30
31 /**
32 Entrypoint of USB Keyboard Driver.
33
34 This function is the entrypoint of USB Keyboard Driver. It installs Driver Binding
35 Protocols together with Component Name Protocols.
36
37 @param ImageHandle The firmware allocated handle for the EFI image.
38 @param SystemTable A pointer to the EFI System Table.
39
40 @retval EFI_SUCCESS The entry point is executed successfully.
41
42 **/
43 EFI_STATUS
44 EFIAPI
USBKeyboardDriverBindingEntryPoint(IN EFI_HANDLE ImageHandle,IN EFI_SYSTEM_TABLE * SystemTable)45 USBKeyboardDriverBindingEntryPoint (
46 IN EFI_HANDLE ImageHandle,
47 IN EFI_SYSTEM_TABLE *SystemTable
48 )
49 {
50 EFI_STATUS Status;
51
52 Status = EfiLibInstallDriverBindingComponentName2 (
53 ImageHandle,
54 SystemTable,
55 &gUsbKeyboardDriverBinding,
56 ImageHandle,
57 &gUsbKeyboardComponentName,
58 &gUsbKeyboardComponentName2
59 );
60 ASSERT_EFI_ERROR (Status);
61
62 return EFI_SUCCESS;
63 }
64
65 /**
66 Check whether USB keyboard driver supports this device.
67
68 @param This The USB keyboard driver binding protocol.
69 @param Controller The controller handle to check.
70 @param RemainingDevicePath The remaining device path.
71
72 @retval EFI_SUCCESS The driver supports this controller.
73 @retval other This device isn't supported.
74
75 **/
76 EFI_STATUS
77 EFIAPI
USBKeyboardDriverBindingSupported(IN EFI_DRIVER_BINDING_PROTOCOL * This,IN EFI_HANDLE Controller,IN EFI_DEVICE_PATH_PROTOCOL * RemainingDevicePath)78 USBKeyboardDriverBindingSupported (
79 IN EFI_DRIVER_BINDING_PROTOCOL *This,
80 IN EFI_HANDLE Controller,
81 IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath
82 )
83 {
84 EFI_STATUS Status;
85 EFI_USB_IO_PROTOCOL *UsbIo;
86
87 //
88 // Check if USB I/O Protocol is attached on the controller handle.
89 //
90 Status = gBS->OpenProtocol (
91 Controller,
92 &gEfiUsbIoProtocolGuid,
93 (VOID **) &UsbIo,
94 This->DriverBindingHandle,
95 Controller,
96 EFI_OPEN_PROTOCOL_BY_DRIVER
97 );
98 if (EFI_ERROR (Status)) {
99 return Status;
100 }
101
102 //
103 // Use the USB I/O Protocol interface to check whether Controller is
104 // a keyboard device that can be managed by this driver.
105 //
106 Status = EFI_SUCCESS;
107
108 if (!IsUSBKeyboard (UsbIo)) {
109 Status = EFI_UNSUPPORTED;
110 }
111
112 gBS->CloseProtocol (
113 Controller,
114 &gEfiUsbIoProtocolGuid,
115 This->DriverBindingHandle,
116 Controller
117 );
118
119 return Status;
120 }
121
122 /**
123 Starts the keyboard device with this driver.
124
125 This function produces Simple Text Input Protocol and Simple Text Input Ex Protocol,
126 initializes the keyboard device, and submit Asynchronous Interrupt Transfer to manage
127 this keyboard device.
128
129 @param This The USB keyboard driver binding instance.
130 @param Controller Handle of device to bind driver to.
131 @param RemainingDevicePath Optional parameter use to pick a specific child
132 device to start.
133
134 @retval EFI_SUCCESS The controller is controlled by the usb keyboard driver.
135 @retval EFI_UNSUPPORTED No interrupt endpoint can be found.
136 @retval Other This controller cannot be started.
137
138 **/
139 EFI_STATUS
140 EFIAPI
USBKeyboardDriverBindingStart(IN EFI_DRIVER_BINDING_PROTOCOL * This,IN EFI_HANDLE Controller,IN EFI_DEVICE_PATH_PROTOCOL * RemainingDevicePath)141 USBKeyboardDriverBindingStart (
142 IN EFI_DRIVER_BINDING_PROTOCOL *This,
143 IN EFI_HANDLE Controller,
144 IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath
145 )
146 {
147 EFI_STATUS Status;
148 EFI_USB_IO_PROTOCOL *UsbIo;
149 USB_KB_DEV *UsbKeyboardDevice;
150 UINT8 EndpointNumber;
151 EFI_USB_ENDPOINT_DESCRIPTOR EndpointDescriptor;
152 UINT8 Index;
153 UINT8 EndpointAddr;
154 UINT8 PollingInterval;
155 UINT8 PacketSize;
156 BOOLEAN Found;
157 EFI_TPL OldTpl;
158
159 OldTpl = gBS->RaiseTPL (TPL_CALLBACK);
160 //
161 // Open USB I/O Protocol
162 //
163 Status = gBS->OpenProtocol (
164 Controller,
165 &gEfiUsbIoProtocolGuid,
166 (VOID **) &UsbIo,
167 This->DriverBindingHandle,
168 Controller,
169 EFI_OPEN_PROTOCOL_BY_DRIVER
170 );
171 if (EFI_ERROR (Status)) {
172 goto ErrorExit1;
173 }
174
175 UsbKeyboardDevice = AllocateZeroPool (sizeof (USB_KB_DEV));
176 ASSERT (UsbKeyboardDevice != NULL);
177
178 //
179 // Get the Device Path Protocol on Controller's handle
180 //
181 Status = gBS->OpenProtocol (
182 Controller,
183 &gEfiDevicePathProtocolGuid,
184 (VOID **) &UsbKeyboardDevice->DevicePath,
185 This->DriverBindingHandle,
186 Controller,
187 EFI_OPEN_PROTOCOL_GET_PROTOCOL
188 );
189
190 if (EFI_ERROR (Status)) {
191 goto ErrorExit;
192 }
193 //
194 // Report that the USB keyboard is being enabled
195 //
196 REPORT_STATUS_CODE_WITH_DEVICE_PATH (
197 EFI_PROGRESS_CODE,
198 (EFI_PERIPHERAL_KEYBOARD | EFI_P_PC_ENABLE),
199 UsbKeyboardDevice->DevicePath
200 );
201
202 //
203 // This is pretty close to keyboard detection, so log progress
204 //
205 REPORT_STATUS_CODE_WITH_DEVICE_PATH (
206 EFI_PROGRESS_CODE,
207 (EFI_PERIPHERAL_KEYBOARD | EFI_P_PC_PRESENCE_DETECT),
208 UsbKeyboardDevice->DevicePath
209 );
210
211 UsbKeyboardDevice->UsbIo = UsbIo;
212
213 //
214 // Get interface & endpoint descriptor
215 //
216 UsbIo->UsbGetInterfaceDescriptor (
217 UsbIo,
218 &UsbKeyboardDevice->InterfaceDescriptor
219 );
220
221 EndpointNumber = UsbKeyboardDevice->InterfaceDescriptor.NumEndpoints;
222
223 //
224 // Traverse endpoints to find interrupt endpoint
225 //
226 Found = FALSE;
227 for (Index = 0; Index < EndpointNumber; Index++) {
228
229 UsbIo->UsbGetEndpointDescriptor (
230 UsbIo,
231 Index,
232 &EndpointDescriptor
233 );
234
235 if ((EndpointDescriptor.Attributes & (BIT0 | BIT1)) == USB_ENDPOINT_INTERRUPT) {
236 //
237 // We only care interrupt endpoint here
238 //
239 CopyMem(&UsbKeyboardDevice->IntEndpointDescriptor, &EndpointDescriptor, sizeof(EndpointDescriptor));
240 Found = TRUE;
241 break;
242 }
243 }
244
245 if (!Found) {
246 //
247 // Report Status Code to indicate that there is no USB keyboard
248 //
249 REPORT_STATUS_CODE (
250 EFI_ERROR_CODE | EFI_ERROR_MINOR,
251 (EFI_PERIPHERAL_KEYBOARD | EFI_P_EC_NOT_DETECTED)
252 );
253 //
254 // No interrupt endpoint found, then return unsupported.
255 //
256 Status = EFI_UNSUPPORTED;
257 goto ErrorExit;
258 }
259
260 REPORT_STATUS_CODE_WITH_DEVICE_PATH (
261 EFI_PROGRESS_CODE,
262 (EFI_PERIPHERAL_KEYBOARD | EFI_P_PC_DETECTED),
263 UsbKeyboardDevice->DevicePath
264 );
265
266 UsbKeyboardDevice->Signature = USB_KB_DEV_SIGNATURE;
267 UsbKeyboardDevice->SimpleInput.Reset = USBKeyboardReset;
268 UsbKeyboardDevice->SimpleInput.ReadKeyStroke = USBKeyboardReadKeyStroke;
269
270 UsbKeyboardDevice->SimpleInputEx.Reset = USBKeyboardResetEx;
271 UsbKeyboardDevice->SimpleInputEx.ReadKeyStrokeEx = USBKeyboardReadKeyStrokeEx;
272 UsbKeyboardDevice->SimpleInputEx.SetState = USBKeyboardSetState;
273 UsbKeyboardDevice->SimpleInputEx.RegisterKeyNotify = USBKeyboardRegisterKeyNotify;
274 UsbKeyboardDevice->SimpleInputEx.UnregisterKeyNotify = USBKeyboardUnregisterKeyNotify;
275
276 InitializeListHead (&UsbKeyboardDevice->NotifyList);
277
278 Status = gBS->CreateEvent (
279 EVT_TIMER | EVT_NOTIFY_SIGNAL,
280 TPL_NOTIFY,
281 USBKeyboardTimerHandler,
282 UsbKeyboardDevice,
283 &UsbKeyboardDevice->TimerEvent
284 );
285 if (!EFI_ERROR (Status)) {
286 Status = gBS->SetTimer (UsbKeyboardDevice->TimerEvent, TimerPeriodic, KEYBOARD_TIMER_INTERVAL);
287 }
288 if (EFI_ERROR (Status)) {
289 goto ErrorExit;
290 }
291
292 Status = gBS->CreateEvent (
293 EVT_NOTIFY_WAIT,
294 TPL_NOTIFY,
295 USBKeyboardWaitForKey,
296 UsbKeyboardDevice,
297 &(UsbKeyboardDevice->SimpleInputEx.WaitForKeyEx)
298 );
299
300 if (EFI_ERROR (Status)) {
301 goto ErrorExit;
302 }
303
304 Status = gBS->CreateEvent (
305 EVT_NOTIFY_WAIT,
306 TPL_NOTIFY,
307 USBKeyboardWaitForKey,
308 UsbKeyboardDevice,
309 &(UsbKeyboardDevice->SimpleInput.WaitForKey)
310 );
311 if (EFI_ERROR (Status)) {
312 goto ErrorExit;
313 }
314
315 //
316 // Install Simple Text Input Protocol and Simple Text Input Ex Protocol
317 // for the USB keyboard device.
318 // USB keyboard is a hot plug device, and expected to work immediately
319 // when plugging into system, other conventional console devices could
320 // distinguish it by its device path.
321 //
322 Status = gBS->InstallMultipleProtocolInterfaces (
323 &Controller,
324 &gEfiSimpleTextInProtocolGuid,
325 &UsbKeyboardDevice->SimpleInput,
326 &gEfiSimpleTextInputExProtocolGuid,
327 &UsbKeyboardDevice->SimpleInputEx,
328 NULL
329 );
330 if (EFI_ERROR (Status)) {
331 goto ErrorExit;
332 }
333
334 UsbKeyboardDevice->ControllerHandle = Controller;
335 Status = InitKeyboardLayout (UsbKeyboardDevice);
336 if (EFI_ERROR (Status)) {
337 gBS->UninstallMultipleProtocolInterfaces (
338 Controller,
339 &gEfiSimpleTextInProtocolGuid,
340 &UsbKeyboardDevice->SimpleInput,
341 &gEfiSimpleTextInputExProtocolGuid,
342 &UsbKeyboardDevice->SimpleInputEx,
343 NULL
344 );
345 goto ErrorExit;
346 }
347
348
349 //
350 // Reset USB Keyboard Device exhaustively.
351 //
352 Status = UsbKeyboardDevice->SimpleInputEx.Reset (
353 &UsbKeyboardDevice->SimpleInputEx,
354 TRUE
355 );
356 if (EFI_ERROR (Status)) {
357 gBS->UninstallMultipleProtocolInterfaces (
358 Controller,
359 &gEfiSimpleTextInProtocolGuid,
360 &UsbKeyboardDevice->SimpleInput,
361 &gEfiSimpleTextInputExProtocolGuid,
362 &UsbKeyboardDevice->SimpleInputEx,
363 NULL
364 );
365 goto ErrorExit;
366 }
367
368 //
369 // Submit Asynchronous Interrupt Transfer to manage this device.
370 //
371 EndpointAddr = UsbKeyboardDevice->IntEndpointDescriptor.EndpointAddress;
372 PollingInterval = UsbKeyboardDevice->IntEndpointDescriptor.Interval;
373 PacketSize = (UINT8) (UsbKeyboardDevice->IntEndpointDescriptor.MaxPacketSize);
374
375 Status = UsbIo->UsbAsyncInterruptTransfer (
376 UsbIo,
377 EndpointAddr,
378 TRUE,
379 PollingInterval,
380 PacketSize,
381 KeyboardHandler,
382 UsbKeyboardDevice
383 );
384
385 if (EFI_ERROR (Status)) {
386 gBS->UninstallMultipleProtocolInterfaces (
387 Controller,
388 &gEfiSimpleTextInProtocolGuid,
389 &UsbKeyboardDevice->SimpleInput,
390 &gEfiSimpleTextInputExProtocolGuid,
391 &UsbKeyboardDevice->SimpleInputEx,
392 NULL
393 );
394 goto ErrorExit;
395 }
396
397 UsbKeyboardDevice->ControllerNameTable = NULL;
398 AddUnicodeString2 (
399 "eng",
400 gUsbKeyboardComponentName.SupportedLanguages,
401 &UsbKeyboardDevice->ControllerNameTable,
402 L"Generic Usb Keyboard",
403 TRUE
404 );
405 AddUnicodeString2 (
406 "en",
407 gUsbKeyboardComponentName2.SupportedLanguages,
408 &UsbKeyboardDevice->ControllerNameTable,
409 L"Generic Usb Keyboard",
410 FALSE
411 );
412
413 gBS->RestoreTPL (OldTpl);
414 return EFI_SUCCESS;
415
416 //
417 // Error handler
418 //
419 ErrorExit:
420 if (UsbKeyboardDevice != NULL) {
421 if (UsbKeyboardDevice->TimerEvent != NULL) {
422 gBS->CloseEvent (UsbKeyboardDevice->TimerEvent);
423 }
424 if (UsbKeyboardDevice->SimpleInput.WaitForKey != NULL) {
425 gBS->CloseEvent (UsbKeyboardDevice->SimpleInput.WaitForKey);
426 }
427 if (UsbKeyboardDevice->SimpleInputEx.WaitForKeyEx != NULL) {
428 gBS->CloseEvent (UsbKeyboardDevice->SimpleInputEx.WaitForKeyEx);
429 }
430 if (UsbKeyboardDevice->KeyboardLayoutEvent != NULL) {
431 ReleaseKeyboardLayoutResources (UsbKeyboardDevice);
432 gBS->CloseEvent (UsbKeyboardDevice->KeyboardLayoutEvent);
433 }
434 FreePool (UsbKeyboardDevice);
435 UsbKeyboardDevice = NULL;
436 }
437 gBS->CloseProtocol (
438 Controller,
439 &gEfiUsbIoProtocolGuid,
440 This->DriverBindingHandle,
441 Controller
442 );
443
444 ErrorExit1:
445 gBS->RestoreTPL (OldTpl);
446
447 return Status;
448
449 }
450
451
452 /**
453 Stop the USB keyboard device handled by this driver.
454
455 @param This The USB keyboard driver binding protocol.
456 @param Controller The controller to release.
457 @param NumberOfChildren The number of handles in ChildHandleBuffer.
458 @param ChildHandleBuffer The array of child handle.
459
460 @retval EFI_SUCCESS The device was stopped.
461 @retval EFI_UNSUPPORTED Simple Text In Protocol or Simple Text In Ex Protocol
462 is not installed on Controller.
463 @retval EFI_DEVICE_ERROR The device could not be stopped due to a device error.
464 @retval Others Fail to uninstall protocols attached on the device.
465
466 **/
467 EFI_STATUS
468 EFIAPI
USBKeyboardDriverBindingStop(IN EFI_DRIVER_BINDING_PROTOCOL * This,IN EFI_HANDLE Controller,IN UINTN NumberOfChildren,IN EFI_HANDLE * ChildHandleBuffer)469 USBKeyboardDriverBindingStop (
470 IN EFI_DRIVER_BINDING_PROTOCOL *This,
471 IN EFI_HANDLE Controller,
472 IN UINTN NumberOfChildren,
473 IN EFI_HANDLE *ChildHandleBuffer
474 )
475 {
476 EFI_STATUS Status;
477 EFI_SIMPLE_TEXT_INPUT_PROTOCOL *SimpleInput;
478 USB_KB_DEV *UsbKeyboardDevice;
479
480 Status = gBS->OpenProtocol (
481 Controller,
482 &gEfiSimpleTextInProtocolGuid,
483 (VOID **) &SimpleInput,
484 This->DriverBindingHandle,
485 Controller,
486 EFI_OPEN_PROTOCOL_GET_PROTOCOL
487 );
488 if (EFI_ERROR (Status)) {
489 return EFI_UNSUPPORTED;
490 }
491
492 Status = gBS->OpenProtocol (
493 Controller,
494 &gEfiSimpleTextInputExProtocolGuid,
495 NULL,
496 This->DriverBindingHandle,
497 Controller,
498 EFI_OPEN_PROTOCOL_TEST_PROTOCOL
499 );
500 if (EFI_ERROR (Status)) {
501 return EFI_UNSUPPORTED;
502 }
503
504 UsbKeyboardDevice = USB_KB_DEV_FROM_THIS (SimpleInput);
505
506 //
507 // The key data input from this device will be disabled.
508 //
509 REPORT_STATUS_CODE_WITH_DEVICE_PATH (
510 EFI_PROGRESS_CODE,
511 (EFI_PERIPHERAL_KEYBOARD | EFI_P_PC_DISABLE),
512 UsbKeyboardDevice->DevicePath
513 );
514
515 //
516 // Delete the Asynchronous Interrupt Transfer from this device
517 //
518 UsbKeyboardDevice->UsbIo->UsbAsyncInterruptTransfer (
519 UsbKeyboardDevice->UsbIo,
520 UsbKeyboardDevice->IntEndpointDescriptor.EndpointAddress,
521 FALSE,
522 UsbKeyboardDevice->IntEndpointDescriptor.Interval,
523 0,
524 NULL,
525 NULL
526 );
527
528 gBS->CloseProtocol (
529 Controller,
530 &gEfiUsbIoProtocolGuid,
531 This->DriverBindingHandle,
532 Controller
533 );
534
535 Status = gBS->UninstallMultipleProtocolInterfaces (
536 Controller,
537 &gEfiSimpleTextInProtocolGuid,
538 &UsbKeyboardDevice->SimpleInput,
539 &gEfiSimpleTextInputExProtocolGuid,
540 &UsbKeyboardDevice->SimpleInputEx,
541 NULL
542 );
543 //
544 // Free all resources.
545 //
546 gBS->CloseEvent (UsbKeyboardDevice->TimerEvent);
547 gBS->CloseEvent (UsbKeyboardDevice->RepeatTimer);
548 gBS->CloseEvent (UsbKeyboardDevice->DelayedRecoveryEvent);
549 gBS->CloseEvent (UsbKeyboardDevice->SimpleInput.WaitForKey);
550 gBS->CloseEvent (UsbKeyboardDevice->SimpleInputEx.WaitForKeyEx);
551 KbdFreeNotifyList (&UsbKeyboardDevice->NotifyList);
552
553 ReleaseKeyboardLayoutResources (UsbKeyboardDevice);
554 gBS->CloseEvent (UsbKeyboardDevice->KeyboardLayoutEvent);
555
556 if (UsbKeyboardDevice->ControllerNameTable != NULL) {
557 FreeUnicodeStringTable (UsbKeyboardDevice->ControllerNameTable);
558 }
559
560 DestroyQueue (&UsbKeyboardDevice->UsbKeyQueue);
561 DestroyQueue (&UsbKeyboardDevice->EfiKeyQueue);
562
563 FreePool (UsbKeyboardDevice);
564
565 return Status;
566 }
567
568 /**
569 Internal function to read the next keystroke from the keyboard buffer.
570
571 @param UsbKeyboardDevice USB keyboard's private structure.
572 @param KeyData A pointer to buffer to hold the keystroke
573 data for the key that was pressed.
574
575 @retval EFI_SUCCESS The keystroke information was returned.
576 @retval EFI_NOT_READY There was no keystroke data availiable.
577 @retval EFI_DEVICE_ERROR The keystroke information was not returned due to
578 hardware errors.
579 @retval EFI_INVALID_PARAMETER KeyData is NULL.
580 @retval Others Fail to translate keycode into EFI_INPUT_KEY
581
582 **/
583 EFI_STATUS
USBKeyboardReadKeyStrokeWorker(IN OUT USB_KB_DEV * UsbKeyboardDevice,OUT EFI_KEY_DATA * KeyData)584 USBKeyboardReadKeyStrokeWorker (
585 IN OUT USB_KB_DEV *UsbKeyboardDevice,
586 OUT EFI_KEY_DATA *KeyData
587 )
588 {
589 if (KeyData == NULL) {
590 return EFI_INVALID_PARAMETER;
591 }
592
593 if (IsQueueEmpty (&UsbKeyboardDevice->EfiKeyQueue)) {
594 return EFI_NOT_READY;
595 }
596
597 Dequeue (&UsbKeyboardDevice->EfiKeyQueue, KeyData, sizeof (*KeyData));
598
599 return EFI_SUCCESS;
600 }
601
602 /**
603 Reset the input device and optionally run diagnostics
604
605 There are 2 types of reset for USB keyboard.
606 For non-exhaustive reset, only keyboard buffer is cleared.
607 For exhaustive reset, in addition to clearance of keyboard buffer, the hardware status
608 is also re-initialized.
609
610 @param This Protocol instance pointer.
611 @param ExtendedVerification Driver may perform diagnostics on reset.
612
613 @retval EFI_SUCCESS The device was reset.
614 @retval EFI_DEVICE_ERROR The device is not functioning properly and could not be reset.
615
616 **/
617 EFI_STATUS
618 EFIAPI
USBKeyboardReset(IN EFI_SIMPLE_TEXT_INPUT_PROTOCOL * This,IN BOOLEAN ExtendedVerification)619 USBKeyboardReset (
620 IN EFI_SIMPLE_TEXT_INPUT_PROTOCOL *This,
621 IN BOOLEAN ExtendedVerification
622 )
623 {
624 EFI_STATUS Status;
625 USB_KB_DEV *UsbKeyboardDevice;
626
627 UsbKeyboardDevice = USB_KB_DEV_FROM_THIS (This);
628
629 REPORT_STATUS_CODE_WITH_DEVICE_PATH (
630 EFI_PROGRESS_CODE,
631 (EFI_PERIPHERAL_KEYBOARD | EFI_P_PC_RESET),
632 UsbKeyboardDevice->DevicePath
633 );
634
635 //
636 // Non-exhaustive reset:
637 // only reset private data structures.
638 //
639 if (!ExtendedVerification) {
640 REPORT_STATUS_CODE_WITH_DEVICE_PATH (
641 EFI_PROGRESS_CODE,
642 (EFI_PERIPHERAL_KEYBOARD | EFI_P_KEYBOARD_PC_CLEAR_BUFFER),
643 UsbKeyboardDevice->DevicePath
644 );
645 //
646 // Clear the key buffer of this USB keyboard
647 //
648 InitQueue (&UsbKeyboardDevice->UsbKeyQueue, sizeof (USB_KEY));
649 InitQueue (&UsbKeyboardDevice->EfiKeyQueue, sizeof (EFI_KEY_DATA));
650
651 return EFI_SUCCESS;
652 }
653
654 //
655 // Exhaustive reset
656 //
657 Status = InitUSBKeyboard (UsbKeyboardDevice);
658 if (EFI_ERROR (Status)) {
659 return EFI_DEVICE_ERROR;
660 }
661
662 return EFI_SUCCESS;
663 }
664
665
666 /**
667 Reads the next keystroke from the input device.
668
669 @param This The EFI_SIMPLE_TEXT_INPUT_PROTOCOL instance.
670 @param Key A pointer to a buffer that is filled in with the keystroke
671 information for the key that was pressed.
672
673 @retval EFI_SUCCESS The keystroke information was returned.
674 @retval EFI_NOT_READY There was no keystroke data availiable.
675 @retval EFI_DEVICE_ERROR The keystroke information was not returned due to
676 hardware errors.
677
678 **/
679 EFI_STATUS
680 EFIAPI
USBKeyboardReadKeyStroke(IN EFI_SIMPLE_TEXT_INPUT_PROTOCOL * This,OUT EFI_INPUT_KEY * Key)681 USBKeyboardReadKeyStroke (
682 IN EFI_SIMPLE_TEXT_INPUT_PROTOCOL *This,
683 OUT EFI_INPUT_KEY *Key
684 )
685 {
686 USB_KB_DEV *UsbKeyboardDevice;
687 EFI_STATUS Status;
688 EFI_KEY_DATA KeyData;
689
690 UsbKeyboardDevice = USB_KB_DEV_FROM_THIS (This);
691
692 //
693 // Considering if the partial keystroke is enabled, there maybe a partial
694 // keystroke in the queue, so here skip the partial keystroke and get the
695 // next key from the queue
696 //
697 while (1) {
698 Status = USBKeyboardReadKeyStrokeWorker (UsbKeyboardDevice, &KeyData);
699 if (EFI_ERROR (Status)) {
700 return Status;
701 }
702 //
703 // SimpleTextIn Protocol doesn't support partial keystroke;
704 //
705 if (KeyData.Key.ScanCode == CHAR_NULL && KeyData.Key.UnicodeChar == SCAN_NULL) {
706 continue;
707 }
708 //
709 // Translate the CTRL-Alpha characters to their corresponding control value
710 // (ctrl-a = 0x0001 through ctrl-Z = 0x001A)
711 //
712 if ((KeyData.KeyState.KeyShiftState & (EFI_LEFT_CONTROL_PRESSED | EFI_RIGHT_CONTROL_PRESSED)) != 0) {
713 if (KeyData.Key.UnicodeChar >= L'a' && KeyData.Key.UnicodeChar <= L'z') {
714 KeyData.Key.UnicodeChar = (CHAR16) (KeyData.Key.UnicodeChar - L'a' + 1);
715 } else if (KeyData.Key.UnicodeChar >= L'A' && KeyData.Key.UnicodeChar <= L'Z') {
716 KeyData.Key.UnicodeChar = (CHAR16) (KeyData.Key.UnicodeChar - L'A' + 1);
717 }
718 }
719
720 CopyMem (Key, &KeyData.Key, sizeof (EFI_INPUT_KEY));
721 return EFI_SUCCESS;
722 }
723 }
724
725
726 /**
727 Event notification function registered for EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL.WaitForKeyEx
728 and EFI_SIMPLE_TEXT_INPUT_PROTOCOL.WaitForKey.
729
730 @param Event Event to be signaled when a key is pressed.
731 @param Context Points to USB_KB_DEV instance.
732
733 **/
734 VOID
735 EFIAPI
USBKeyboardWaitForKey(IN EFI_EVENT Event,IN VOID * Context)736 USBKeyboardWaitForKey (
737 IN EFI_EVENT Event,
738 IN VOID *Context
739 )
740 {
741 USB_KB_DEV *UsbKeyboardDevice;
742 EFI_KEY_DATA KeyData;
743 EFI_TPL OldTpl;
744
745 UsbKeyboardDevice = (USB_KB_DEV *) Context;
746
747 //
748 // Enter critical section
749 //
750 OldTpl = gBS->RaiseTPL (TPL_NOTIFY);
751
752 //
753 // WaitforKey doesn't suppor the partial key.
754 // Considering if the partial keystroke is enabled, there maybe a partial
755 // keystroke in the queue, so here skip the partial keystroke and get the
756 // next key from the queue
757 //
758 while (!IsQueueEmpty (&UsbKeyboardDevice->EfiKeyQueue)) {
759 //
760 // If there is pending key, signal the event.
761 //
762 CopyMem (
763 &KeyData,
764 UsbKeyboardDevice->EfiKeyQueue.Buffer[UsbKeyboardDevice->EfiKeyQueue.Head],
765 sizeof (EFI_KEY_DATA)
766 );
767 if (KeyData.Key.ScanCode == SCAN_NULL && KeyData.Key.UnicodeChar == CHAR_NULL) {
768 Dequeue (&UsbKeyboardDevice->EfiKeyQueue, &KeyData, sizeof (EFI_KEY_DATA));
769 continue;
770 }
771 gBS->SignalEvent (Event);
772 break;
773 }
774 //
775 // Leave critical section and return
776 //
777 gBS->RestoreTPL (OldTpl);
778 }
779
780 /**
781 Timer handler to convert the key from USB.
782
783 @param Event Indicates the event that invoke this function.
784 @param Context Indicates the calling context.
785 **/
786 VOID
787 EFIAPI
USBKeyboardTimerHandler(IN EFI_EVENT Event,IN VOID * Context)788 USBKeyboardTimerHandler (
789 IN EFI_EVENT Event,
790 IN VOID *Context
791 )
792 {
793 EFI_STATUS Status;
794 USB_KB_DEV *UsbKeyboardDevice;
795 UINT8 KeyCode;
796 EFI_KEY_DATA KeyData;
797
798 UsbKeyboardDevice = (USB_KB_DEV *) Context;
799
800 //
801 // Fetch raw data from the USB keyboard buffer,
802 // and translate it into USB keycode.
803 //
804 Status = USBParseKey (UsbKeyboardDevice, &KeyCode);
805 if (EFI_ERROR (Status)) {
806 return ;
807 }
808
809 //
810 // Translate saved USB keycode into EFI_INPUT_KEY
811 //
812 Status = UsbKeyCodeToEfiInputKey (UsbKeyboardDevice, KeyCode, &KeyData);
813 if (EFI_ERROR (Status)) {
814 return ;
815 }
816
817 //
818 // Insert to the EFI Key queue
819 //
820 Enqueue (&UsbKeyboardDevice->EfiKeyQueue, &KeyData, sizeof (KeyData));
821 }
822
823 /**
824 Free keyboard notify list.
825
826 @param NotifyList The keyboard notify list to free.
827
828 @retval EFI_SUCCESS Free the notify list successfully.
829 @retval EFI_INVALID_PARAMETER NotifyList is NULL.
830
831 **/
832 EFI_STATUS
KbdFreeNotifyList(IN OUT LIST_ENTRY * NotifyList)833 KbdFreeNotifyList (
834 IN OUT LIST_ENTRY *NotifyList
835 )
836 {
837 KEYBOARD_CONSOLE_IN_EX_NOTIFY *NotifyNode;
838 LIST_ENTRY *Link;
839
840 if (NotifyList == NULL) {
841 return EFI_INVALID_PARAMETER;
842 }
843 while (!IsListEmpty (NotifyList)) {
844 Link = GetFirstNode (NotifyList);
845 NotifyNode = CR (Link, KEYBOARD_CONSOLE_IN_EX_NOTIFY, NotifyEntry, USB_KB_CONSOLE_IN_EX_NOTIFY_SIGNATURE);
846 RemoveEntryList (Link);
847 FreePool (NotifyNode);
848 }
849
850 return EFI_SUCCESS;
851 }
852
853 /**
854 Check whether the pressed key matches a registered key or not.
855
856 @param RegsiteredData A pointer to keystroke data for the key that was registered.
857 @param InputData A pointer to keystroke data for the key that was pressed.
858
859 @retval TRUE Key pressed matches a registered key.
860 @retval FLASE Key pressed does not matches a registered key.
861
862 **/
863 BOOLEAN
IsKeyRegistered(IN EFI_KEY_DATA * RegsiteredData,IN EFI_KEY_DATA * InputData)864 IsKeyRegistered (
865 IN EFI_KEY_DATA *RegsiteredData,
866 IN EFI_KEY_DATA *InputData
867 )
868 {
869 ASSERT (RegsiteredData != NULL && InputData != NULL);
870
871 if ((RegsiteredData->Key.ScanCode != InputData->Key.ScanCode) ||
872 (RegsiteredData->Key.UnicodeChar != InputData->Key.UnicodeChar)) {
873 return FALSE;
874 }
875
876 //
877 // Assume KeyShiftState/KeyToggleState = 0 in Registered key data means these state could be ignored.
878 //
879 if (RegsiteredData->KeyState.KeyShiftState != 0 &&
880 RegsiteredData->KeyState.KeyShiftState != InputData->KeyState.KeyShiftState) {
881 return FALSE;
882 }
883 if (RegsiteredData->KeyState.KeyToggleState != 0 &&
884 RegsiteredData->KeyState.KeyToggleState != InputData->KeyState.KeyToggleState) {
885 return FALSE;
886 }
887
888 return TRUE;
889 }
890
891 //
892 // Simple Text Input Ex protocol functions
893 //
894 /**
895 Resets the input device hardware.
896
897 The Reset() function resets the input device hardware. As part
898 of initialization process, the firmware/device will make a quick
899 but reasonable attempt to verify that the device is functioning.
900 If the ExtendedVerification flag is TRUE the firmware may take
901 an extended amount of time to verify the device is operating on
902 reset. Otherwise the reset operation is to occur as quickly as
903 possible. The hardware verification process is not defined by
904 this specification and is left up to the platform firmware or
905 driver to implement.
906
907 @param This A pointer to the EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL instance.
908
909 @param ExtendedVerification Indicates that the driver may perform a more exhaustive
910 verification operation of the device during reset.
911
912 @retval EFI_SUCCESS The device was reset.
913 @retval EFI_DEVICE_ERROR The device is not functioning correctly and could not be reset.
914
915 **/
916 EFI_STATUS
917 EFIAPI
USBKeyboardResetEx(IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL * This,IN BOOLEAN ExtendedVerification)918 USBKeyboardResetEx (
919 IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL *This,
920 IN BOOLEAN ExtendedVerification
921 )
922 {
923 EFI_STATUS Status;
924 USB_KB_DEV *UsbKeyboardDevice;
925
926 UsbKeyboardDevice = TEXT_INPUT_EX_USB_KB_DEV_FROM_THIS (This);
927
928 Status = UsbKeyboardDevice->SimpleInput.Reset (&UsbKeyboardDevice->SimpleInput, ExtendedVerification);
929 if (EFI_ERROR (Status)) {
930 return EFI_DEVICE_ERROR;
931 }
932
933 UsbKeyboardDevice->KeyState.KeyShiftState = EFI_SHIFT_STATE_VALID;
934 UsbKeyboardDevice->KeyState.KeyToggleState = EFI_TOGGLE_STATE_VALID;
935
936 return EFI_SUCCESS;
937
938 }
939
940 /**
941 Reads the next keystroke from the input device.
942
943 @param This Protocol instance pointer.
944 @param KeyData A pointer to a buffer that is filled in with the keystroke
945 state data for the key that was pressed.
946
947 @retval EFI_SUCCESS The keystroke information was returned.
948 @retval EFI_NOT_READY There was no keystroke data available.
949 @retval EFI_DEVICE_ERROR The keystroke information was not returned due to
950 hardware errors.
951 @retval EFI_INVALID_PARAMETER KeyData is NULL.
952
953 **/
954 EFI_STATUS
955 EFIAPI
USBKeyboardReadKeyStrokeEx(IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL * This,OUT EFI_KEY_DATA * KeyData)956 USBKeyboardReadKeyStrokeEx (
957 IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL *This,
958 OUT EFI_KEY_DATA *KeyData
959 )
960 {
961 USB_KB_DEV *UsbKeyboardDevice;
962
963 if (KeyData == NULL) {
964 return EFI_INVALID_PARAMETER;
965 }
966
967 UsbKeyboardDevice = TEXT_INPUT_EX_USB_KB_DEV_FROM_THIS (This);
968
969 return USBKeyboardReadKeyStrokeWorker (UsbKeyboardDevice, KeyData);
970
971 }
972
973 /**
974 Set certain state for the input device.
975
976 @param This Protocol instance pointer.
977 @param KeyToggleState A pointer to the EFI_KEY_TOGGLE_STATE to set the
978 state for the input device.
979
980 @retval EFI_SUCCESS The device state was set appropriately.
981 @retval EFI_DEVICE_ERROR The device is not functioning correctly and could
982 not have the setting adjusted.
983 @retval EFI_UNSUPPORTED The device does not support the ability to have its state set.
984 @retval EFI_INVALID_PARAMETER KeyToggleState is NULL.
985
986 **/
987 EFI_STATUS
988 EFIAPI
USBKeyboardSetState(IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL * This,IN EFI_KEY_TOGGLE_STATE * KeyToggleState)989 USBKeyboardSetState (
990 IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL *This,
991 IN EFI_KEY_TOGGLE_STATE *KeyToggleState
992 )
993 {
994 USB_KB_DEV *UsbKeyboardDevice;
995
996 if (KeyToggleState == NULL) {
997 return EFI_INVALID_PARAMETER;
998 }
999
1000 UsbKeyboardDevice = TEXT_INPUT_EX_USB_KB_DEV_FROM_THIS (This);
1001
1002 if (((UsbKeyboardDevice->KeyState.KeyToggleState & EFI_TOGGLE_STATE_VALID) != EFI_TOGGLE_STATE_VALID) ||
1003 ((*KeyToggleState & EFI_TOGGLE_STATE_VALID) != EFI_TOGGLE_STATE_VALID)) {
1004 return EFI_UNSUPPORTED;
1005 }
1006
1007 //
1008 // Update the status light
1009 //
1010
1011 UsbKeyboardDevice->ScrollOn = FALSE;
1012 UsbKeyboardDevice->NumLockOn = FALSE;
1013 UsbKeyboardDevice->CapsOn = FALSE;
1014 UsbKeyboardDevice->IsSupportPartialKey = FALSE;
1015
1016 if ((*KeyToggleState & EFI_SCROLL_LOCK_ACTIVE) == EFI_SCROLL_LOCK_ACTIVE) {
1017 UsbKeyboardDevice->ScrollOn = TRUE;
1018 }
1019 if ((*KeyToggleState & EFI_NUM_LOCK_ACTIVE) == EFI_NUM_LOCK_ACTIVE) {
1020 UsbKeyboardDevice->NumLockOn = TRUE;
1021 }
1022 if ((*KeyToggleState & EFI_CAPS_LOCK_ACTIVE) == EFI_CAPS_LOCK_ACTIVE) {
1023 UsbKeyboardDevice->CapsOn = TRUE;
1024 }
1025 if ((*KeyToggleState & EFI_KEY_STATE_EXPOSED) == EFI_KEY_STATE_EXPOSED) {
1026 UsbKeyboardDevice->IsSupportPartialKey = TRUE;
1027 }
1028
1029 SetKeyLED (UsbKeyboardDevice);
1030
1031 UsbKeyboardDevice->KeyState.KeyToggleState = *KeyToggleState;
1032
1033 return EFI_SUCCESS;
1034
1035 }
1036
1037 /**
1038 Register a notification function for a particular keystroke for the input device.
1039
1040 @param This Protocol instance pointer.
1041 @param KeyData A pointer to a buffer that is filled in with the keystroke
1042 information data for the key that was pressed.
1043 @param KeyNotificationFunction Points to the function to be called when the key
1044 sequence is typed specified by KeyData.
1045 @param NotifyHandle Points to the unique handle assigned to the registered notification.
1046
1047 @retval EFI_SUCCESS The notification function was registered successfully.
1048 @retval EFI_OUT_OF_RESOURCES Unable to allocate resources for necessary data structures.
1049 @retval EFI_INVALID_PARAMETER KeyData or NotifyHandle or KeyNotificationFunction is NULL.
1050
1051 **/
1052 EFI_STATUS
1053 EFIAPI
USBKeyboardRegisterKeyNotify(IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL * This,IN EFI_KEY_DATA * KeyData,IN EFI_KEY_NOTIFY_FUNCTION KeyNotificationFunction,OUT VOID ** NotifyHandle)1054 USBKeyboardRegisterKeyNotify (
1055 IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL *This,
1056 IN EFI_KEY_DATA *KeyData,
1057 IN EFI_KEY_NOTIFY_FUNCTION KeyNotificationFunction,
1058 OUT VOID **NotifyHandle
1059 )
1060 {
1061 USB_KB_DEV *UsbKeyboardDevice;
1062 KEYBOARD_CONSOLE_IN_EX_NOTIFY *NewNotify;
1063 LIST_ENTRY *Link;
1064 LIST_ENTRY *NotifyList;
1065 KEYBOARD_CONSOLE_IN_EX_NOTIFY *CurrentNotify;
1066
1067 if (KeyData == NULL || NotifyHandle == NULL || KeyNotificationFunction == NULL) {
1068 return EFI_INVALID_PARAMETER;
1069 }
1070
1071 UsbKeyboardDevice = TEXT_INPUT_EX_USB_KB_DEV_FROM_THIS (This);
1072
1073 //
1074 // Return EFI_SUCCESS if the (KeyData, NotificationFunction) is already registered.
1075 //
1076 NotifyList = &UsbKeyboardDevice->NotifyList;
1077
1078 for (Link = GetFirstNode (NotifyList);
1079 !IsNull (NotifyList, Link);
1080 Link = GetNextNode (NotifyList, Link)) {
1081 CurrentNotify = CR (
1082 Link,
1083 KEYBOARD_CONSOLE_IN_EX_NOTIFY,
1084 NotifyEntry,
1085 USB_KB_CONSOLE_IN_EX_NOTIFY_SIGNATURE
1086 );
1087 if (IsKeyRegistered (&CurrentNotify->KeyData, KeyData)) {
1088 if (CurrentNotify->KeyNotificationFn == KeyNotificationFunction) {
1089 *NotifyHandle = CurrentNotify;
1090 return EFI_SUCCESS;
1091 }
1092 }
1093 }
1094
1095 //
1096 // Allocate resource to save the notification function
1097 //
1098 NewNotify = (KEYBOARD_CONSOLE_IN_EX_NOTIFY *) AllocateZeroPool (sizeof (KEYBOARD_CONSOLE_IN_EX_NOTIFY));
1099 if (NewNotify == NULL) {
1100 return EFI_OUT_OF_RESOURCES;
1101 }
1102
1103 NewNotify->Signature = USB_KB_CONSOLE_IN_EX_NOTIFY_SIGNATURE;
1104 NewNotify->KeyNotificationFn = KeyNotificationFunction;
1105 CopyMem (&NewNotify->KeyData, KeyData, sizeof (EFI_KEY_DATA));
1106 InsertTailList (&UsbKeyboardDevice->NotifyList, &NewNotify->NotifyEntry);
1107
1108
1109 *NotifyHandle = NewNotify;
1110
1111 return EFI_SUCCESS;
1112
1113 }
1114
1115 /**
1116 Remove a registered notification function from a particular keystroke.
1117
1118 @param This Protocol instance pointer.
1119 @param NotificationHandle The handle of the notification function being unregistered.
1120
1121 @retval EFI_SUCCESS The notification function was unregistered successfully.
1122 @retval EFI_INVALID_PARAMETER The NotificationHandle is invalid
1123
1124 **/
1125 EFI_STATUS
1126 EFIAPI
USBKeyboardUnregisterKeyNotify(IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL * This,IN VOID * NotificationHandle)1127 USBKeyboardUnregisterKeyNotify (
1128 IN EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL *This,
1129 IN VOID *NotificationHandle
1130 )
1131 {
1132 USB_KB_DEV *UsbKeyboardDevice;
1133 KEYBOARD_CONSOLE_IN_EX_NOTIFY *CurrentNotify;
1134 LIST_ENTRY *Link;
1135 LIST_ENTRY *NotifyList;
1136
1137 if (NotificationHandle == NULL) {
1138 return EFI_INVALID_PARAMETER;
1139 }
1140
1141 UsbKeyboardDevice = TEXT_INPUT_EX_USB_KB_DEV_FROM_THIS (This);
1142
1143 //
1144 // Traverse notify list of USB keyboard and remove the entry of NotificationHandle.
1145 //
1146 NotifyList = &UsbKeyboardDevice->NotifyList;
1147 for (Link = GetFirstNode (NotifyList);
1148 !IsNull (NotifyList, Link);
1149 Link = GetNextNode (NotifyList, Link)) {
1150 CurrentNotify = CR (
1151 Link,
1152 KEYBOARD_CONSOLE_IN_EX_NOTIFY,
1153 NotifyEntry,
1154 USB_KB_CONSOLE_IN_EX_NOTIFY_SIGNATURE
1155 );
1156 if (CurrentNotify == NotificationHandle) {
1157 //
1158 // Remove the notification function from NotifyList and free resources
1159 //
1160 RemoveEntryList (&CurrentNotify->NotifyEntry);
1161
1162 FreePool (CurrentNotify);
1163 return EFI_SUCCESS;
1164 }
1165 }
1166
1167 //
1168 // Cannot find the matching entry in database.
1169 //
1170 return EFI_INVALID_PARAMETER;
1171 }
1172
1173