* [PATCH] Make CTRL and ALT keys work as expected on EFI systems (version 5).
@ 2014-02-25 22:12 Peter Jones
  2014-02-26  2:58 ` Mroczek, Joseph T
                   ` (2 more replies)
  0 siblings, 3 replies; 8+ messages in thread
From: Peter Jones @ 2014-02-25 22:12 UTC (permalink / raw)
  To: grub-devel
This is version 4.
Changes from version 1:
- handles SHIFT as a modifier
- handles F11 and F12 keys
- uses the handle provided by the system table to find our _EX protocol.
Changes from version 2:
- eliminate duplicate keycode translation.
Changes from version 3:
- Do not add the shift modifier for any ascii character between space
  (0x20) and DEL (0x7f); the combination of the modifier and many of the
  keys causes it not to be recognized at all.  Specifically, if we
  include the modifier on any querty punctuation character, i.e.
  anything the string "~!@#$%^&*()_+{}|:\"<>?" represents in C, it stops
  being recognized whatsoever.
Changes from version 4:
- Always initialize term->data from locate protocol (i.e. make it
  unconditional.)
Signed-off-by: Peter Jones <pjones@redhat.com>
---
 grub-core/term/efi/console.c | 118 +++++++++++++++++++++++++++++++++++--------
 include/grub/efi/api.h       |  65 +++++++++++++++++++++++-
 2 files changed, 161 insertions(+), 22 deletions(-)
diff --git a/grub-core/term/efi/console.c b/grub-core/term/efi/console.c
index a37eb84..677eab5 100644
--- a/grub-core/term/efi/console.c
+++ b/grub-core/term/efi/console.c
@@ -104,26 +104,12 @@ const unsigned efi_codes[] =
     GRUB_TERM_KEY_DC, GRUB_TERM_KEY_PPAGE, GRUB_TERM_KEY_NPAGE, GRUB_TERM_KEY_F1,
     GRUB_TERM_KEY_F2, GRUB_TERM_KEY_F3, GRUB_TERM_KEY_F4, GRUB_TERM_KEY_F5,
     GRUB_TERM_KEY_F6, GRUB_TERM_KEY_F7, GRUB_TERM_KEY_F8, GRUB_TERM_KEY_F9,
-    GRUB_TERM_KEY_F10, 0, 0, '\e'
+    GRUB_TERM_KEY_F10, GRUB_TERM_KEY_F10, GRUB_TERM_KEY_F11, '\e'
   };
 
-
 static int
-grub_console_getkey (struct grub_term_input *term __attribute__ ((unused)))
+grub_efi_translate_key (grub_efi_input_key_t key)
 {
-  grub_efi_simple_input_interface_t *i;
-  grub_efi_input_key_t key;
-  grub_efi_status_t status;
-
-  if (grub_efi_is_finished)
-    return 0;
-
-  i = grub_efi_system_table->con_in;
-  status = efi_call_2 (i->read_key_stroke, i, &key);
-
-  if (status != GRUB_EFI_SUCCESS)
-    return GRUB_TERM_NO_KEY;
-
   if (key.scan_code == 0)
     {
       /* Some firmware implementations use VT100-style codes against the spec.
@@ -139,9 +125,98 @@ grub_console_getkey (struct grub_term_input *term __attribute__ ((unused)))
   else if (key.scan_code < ARRAY_SIZE (efi_codes))
     return efi_codes[key.scan_code];
 
+  if (key.unicode_char >= 0x20 && key.unicode_char <= 0x7f)
+    return key.unicode_char;
+
   return GRUB_TERM_NO_KEY;
 }
 
+static int
+grub_console_getkey_con (struct grub_term_input *term __attribute__ ((unused)))
+{
+  grub_efi_simple_input_interface_t *i;
+  grub_efi_input_key_t key;
+  grub_efi_status_t status;
+
+  i = grub_efi_system_table->con_in;
+  status = efi_call_2 (i->read_key_stroke, i, &key);
+
+  if (status != GRUB_EFI_SUCCESS)
+    return GRUB_TERM_NO_KEY;
+
+  return grub_efi_translate_key(key);
+}
+
+static int
+grub_console_getkey_ex(struct grub_term_input *term)
+{
+  grub_efi_key_data_t key_data;
+  grub_efi_status_t status;
+  grub_efi_uint32_t kss;
+  int key = -1;
+
+  grub_efi_simple_text_input_ex_interface_t *text_input = term->data;
+
+  status = efi_call_2 (text_input->read_key_stroke, text_input, &key_data);
+
+  if (status != GRUB_EFI_SUCCESS)
+    return GRUB_TERM_NO_KEY;
+
+  kss = key_data.key_state.key_shift_state;
+  key = grub_efi_translate_key(key_data.key);
+
+  if (key == GRUB_TERM_NO_KEY)
+    return GRUB_TERM_NO_KEY;
+
+  if (kss & GRUB_EFI_SHIFT_STATE_VALID)
+    {
+      if ((kss & GRUB_EFI_LEFT_SHIFT_PRESSED
+	   || kss & GRUB_EFI_RIGHT_SHIFT_PRESSED)
+	  && !(key >= 0x20 && key <= 0x7f))
+	key |= GRUB_TERM_SHIFT;
+      if (kss & GRUB_EFI_LEFT_ALT_PRESSED || kss & GRUB_EFI_RIGHT_ALT_PRESSED)
+	key |= GRUB_TERM_ALT;
+      if (kss & GRUB_EFI_LEFT_CONTROL_PRESSED
+	  || kss & GRUB_EFI_RIGHT_CONTROL_PRESSED)
+	key |= GRUB_TERM_CTRL;
+    }
+
+  return key;
+}
+
+static grub_err_t
+grub_efi_console_input_init (struct grub_term_input *term)
+{
+  grub_efi_guid_t text_input_ex_guid =
+    GRUB_EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL_GUID;
+
+  if (grub_efi_is_finished)
+    return 0;
+
+  grub_efi_simple_text_input_ex_interface_t *text_input = term->data;
+  if (text_input)
+    return 0;
+
+  text_input = grub_efi_open_protocol(grub_efi_system_table->console_in_handler,
+				      &text_input_ex_guid,
+				      GRUB_EFI_OPEN_PROTOCOL_GET_PROTOCOL);
+  term->data = (void *)text_input;
+
+  return 0;
+}
+
+static int
+grub_console_getkey (struct grub_term_input *term)
+{
+  if (grub_efi_is_finished)
+    return 0;
+
+  if (term->data)
+    return grub_console_getkey_ex(term);
+  else
+    return grub_console_getkey_con(term);
+}
+
 static struct grub_term_coordinate
 grub_console_getwh (struct grub_term_output *term __attribute__ ((unused)))
 {
@@ -243,7 +318,7 @@ grub_console_setcursor (struct grub_term_output *term __attribute__ ((unused)),
 }
 
 static grub_err_t
-grub_efi_console_init (struct grub_term_output *term)
+grub_efi_console_output_init (struct grub_term_output *term)
 {
   grub_efi_set_text_mode (1);
   grub_console_setcursor (term, 1);
@@ -251,7 +326,7 @@ grub_efi_console_init (struct grub_term_output *term)
 }
 
 static grub_err_t
-grub_efi_console_fini (struct grub_term_output *term)
+grub_efi_console_output_fini (struct grub_term_output *term)
 {
   grub_console_setcursor (term, 0);
   grub_efi_set_text_mode (0);
@@ -262,13 +337,14 @@ static struct grub_term_input grub_console_term_input =
   {
     .name = "console",
     .getkey = grub_console_getkey,
+    .init = grub_efi_console_input_init,
   };
 
 static struct grub_term_output grub_console_term_output =
   {
     .name = "console",
-    .init = grub_efi_console_init,
-    .fini = grub_efi_console_fini,
+    .init = grub_efi_console_output_init,
+    .fini = grub_efi_console_output_fini,
     .putchar = grub_console_putchar,
     .getwh = grub_console_getwh,
     .getxy = grub_console_getxy,
@@ -291,8 +367,8 @@ grub_console_init (void)
       return;
     }
 
-  grub_term_register_input ("console", &grub_console_term_input);
   grub_term_register_output ("console", &grub_console_term_output);
+  grub_term_register_input ("console", &grub_console_term_input);
 }
 
 void
diff --git a/include/grub/efi/api.h b/include/grub/efi/api.h
index e5dd543..1423403 100644
--- a/include/grub/efi/api.h
+++ b/include/grub/efi/api.h
@@ -111,7 +111,7 @@
     { 0x8e, 0x39, 0x00, 0xa0, 0xc9, 0x69, 0x72, 0x3b } \
   }
 
-#define EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL_GUID \
+#define GRUB_EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL_GUID \
   { 0xdd9e7534, 0x7762, 0x4698, \
     { 0x8c, 0x14, 0xf5, 0x85, 0x17, 0xa6, 0x25, 0xaa } \
   }
@@ -952,6 +952,32 @@ struct grub_efi_input_key
 };
 typedef struct grub_efi_input_key grub_efi_input_key_t;
 
+typedef grub_efi_uint8_t grub_efi_key_toggle_state_t;
+struct grub_efi_key_state
+{
+	grub_efi_uint32_t key_shift_state;
+	grub_efi_key_toggle_state_t key_toggle_state;
+};
+typedef struct grub_efi_key_state grub_efi_key_state_t;
+
+#define GRUB_EFI_SHIFT_STATE_VALID     0x80000000
+#define GRUB_EFI_RIGHT_SHIFT_PRESSED   0x00000001
+#define GRUB_EFI_LEFT_SHIFT_PRESSED    0x00000002
+#define GRUB_EFI_RIGHT_CONTROL_PRESSED 0x00000004
+#define GRUB_EFI_LEFT_CONTROL_PRESSED  0x00000008
+#define GRUB_EFI_RIGHT_ALT_PRESSED     0x00000010
+#define GRUB_EFI_LEFT_ALT_PRESSED      0x00000020
+#define GRUB_EFI_RIGHT_LOGO_PRESSED    0x00000040
+#define GRUB_EFI_LEFT_LOGO_PRESSED     0x00000080
+#define GRUB_EFI_MENU_KEY_PRESSED      0x00000100
+#define GRUB_EFI_SYS_REQ_PRESSED       0x00000200
+
+#define GRUB_EFI_TOGGLE_STATE_VALID 0x80
+#define GRUB_EFI_KEY_STATE_EXPOSED  0x40
+#define GRUB_EFI_SCROLL_LOCK_ACTIVE 0x01
+#define GRUB_EFI_NUM_LOCK_ACTIVE    0x02
+#define GRUB_EFI_CAPS_LOCK_ACTIVE   0x04
+
 struct grub_efi_simple_text_output_mode
 {
   grub_efi_int32_t max_mode;
@@ -1294,6 +1320,43 @@ struct grub_efi_simple_input_interface
 };
 typedef struct grub_efi_simple_input_interface grub_efi_simple_input_interface_t;
 
+struct grub_efi_key_data {
+	grub_efi_input_key_t key;
+	grub_efi_key_state_t key_state;
+};
+typedef struct grub_efi_key_data grub_efi_key_data_t;
+
+typedef grub_efi_status_t (*grub_efi_key_notify_function_t) (
+	grub_efi_key_data_t *key_data
+	);
+
+struct grub_efi_simple_text_input_ex_interface
+{
+	grub_efi_status_t
+	(*reset) (struct grub_efi_simple_text_input_ex_interface *this,
+		  grub_efi_boolean_t extended_verification);
+
+	grub_efi_status_t
+	(*read_key_stroke) (struct grub_efi_simple_text_input_ex_interface *this,
+			    grub_efi_key_data_t *key_data);
+
+	grub_efi_event_t wait_for_key;
+
+	grub_efi_status_t
+	(*set_state) (struct grub_efi_simple_text_input_ex_interface *this,
+		      grub_efi_key_toggle_state_t *key_toggle_state);
+
+	grub_efi_status_t
+	(*register_key_notify) (struct grub_efi_simple_text_input_ex_interface *this,
+				grub_efi_key_data_t *key_data,
+				grub_efi_key_notify_function_t key_notification_function);
+
+	grub_efi_status_t
+	(*unregister_key_notify) (struct grub_efi_simple_text_input_ex_interface *this,
+				  void *notification_handle);
+};
+typedef struct grub_efi_simple_text_input_ex_interface grub_efi_simple_text_input_ex_interface_t;
+
 struct grub_efi_simple_text_output_interface
 {
   grub_efi_status_t
-- 
1.8.5.3
^ permalink raw reply related	[flat|nested] 8+ messages in thread
* RE: [PATCH] Make CTRL and ALT keys work as expected on EFI systems (version 5).
  2014-02-25 22:12 [PATCH] Make CTRL and ALT keys work as expected on EFI systems (version 5) Peter Jones
@ 2014-02-26  2:58 ` Mroczek, Joseph T
  2014-02-26 18:51   ` Vladimir 'φ-coder/phcoder' Serbinenko
  2015-10-10 18:48 ` Andrei Borzenkov
  2015-10-25 15:41 ` Vladimir 'φ-coder/phcoder' Serbinenko
  2 siblings, 1 reply; 8+ messages in thread
From: Mroczek, Joseph T @ 2014-02-26  2:58 UTC (permalink / raw)
  To: The development of GNU GRUB
> From: Peter Jones
> Sent: Tuesday, February 25, 2014 2:12 PM
> 
> This is version 4.
> 
> Changes from version 1:
> - handles SHIFT as a modifier
> - handles F11 and F12 keys
> - uses the handle provided by the system table to find our _EX protocol.
> 
> Changes from version 2:
> - eliminate duplicate keycode translation.
> 
> Changes from version 3:
> - Do not add the shift modifier for any ascii character between space
>   (0x20) and DEL (0x7f); the combination of the modifier and many of the
>   keys causes it not to be recognized at all.  Specifically, if we
>   include the modifier on any querty punctuation character, i.e.
>   anything the string "~!@#$%^&*()_+{}|:\"<>?" represents in C, it stops
>   being recognized whatsoever.
> 
> Changes from version 4:
> - Always initialize term->data from locate protocol (i.e. make it
>   unconditional.)
> 
May I ask you to add one further extension. Add caps lock asserted to the conditions to bring up the menu. If this is not relevant to this patch or you wont add it, let me know and I will see if I can add on after the patch is accepted.
The reason for this is that there are some system that take as long as 15 minutes to post. Trying to hit a specific key in a small window (even 30s) is difficult. Allowing caps lock to perform the same action as holding shift allows the user to essentially queue the request for the menu. This behavior is consistent with how syslinux/pxelinux works. 
Thank you for any consideration you can pay this request.
~joe
^ permalink raw reply	[flat|nested] 8+ messages in thread
* Re: [PATCH] Make CTRL and ALT keys work as expected on EFI systems (version 5).
  2014-02-26  2:58 ` Mroczek, Joseph T
@ 2014-02-26 18:51   ` Vladimir 'φ-coder/phcoder' Serbinenko
  0 siblings, 0 replies; 8+ messages in thread
From: Vladimir 'φ-coder/phcoder' Serbinenko @ 2014-02-26 18:51 UTC (permalink / raw)
  To: grub-devel
[-- Attachment #1: Type: text/plain, Size: 1849 bytes --]
On 26.02.2014 03:58, Mroczek, Joseph T wrote:
>> From: Peter Jones
>> Sent: Tuesday, February 25, 2014 2:12 PM
>>
>> This is version 4.
>>
>> Changes from version 1:
>> - handles SHIFT as a modifier
>> - handles F11 and F12 keys
>> - uses the handle provided by the system table to find our _EX protocol.
>>
>> Changes from version 2:
>> - eliminate duplicate keycode translation.
>>
>> Changes from version 3:
>> - Do not add the shift modifier for any ascii character between space
>>   (0x20) and DEL (0x7f); the combination of the modifier and many of the
>>   keys causes it not to be recognized at all.  Specifically, if we
>>   include the modifier on any querty punctuation character, i.e.
>>   anything the string "~!@#$%^&*()_+{}|:\"<>?" represents in C, it stops
>>   being recognized whatsoever.
>>
>> Changes from version 4:
>> - Always initialize term->data from locate protocol (i.e. make it
>>   unconditional.)
>>
> 
> May I ask you to add one further extension. Add caps lock asserted to the conditions to bring up the menu. If this is not relevant to this patch or you wont add it, let me know and I will see if I can add on after the patch is accepted.
> 
Has absolutely nothing to do with this patch.
> The reason for this is that there are some system that take as long as 15 minutes to post. Trying to hit a specific key in a small window (even 30s) is difficult. Allowing caps lock to perform the same action as holding shift allows the user to essentially queue the request for the menu. This behavior is consistent with how syslinux/pxelinux works. 
> 
> Thank you for any consideration you can pay this request.
> 
> ~joe
> 
> _______________________________________________
> Grub-devel mailing list
> Grub-devel@gnu.org
> https://lists.gnu.org/mailman/listinfo/grub-devel
> 
[-- Attachment #2: OpenPGP digital signature --]
[-- Type: application/pgp-signature, Size: 274 bytes --]
^ permalink raw reply	[flat|nested] 8+ messages in thread
* Re: [PATCH] Make CTRL and ALT keys work as expected on EFI systems (version 5).
  2014-02-25 22:12 [PATCH] Make CTRL and ALT keys work as expected on EFI systems (version 5) Peter Jones
  2014-02-26  2:58 ` Mroczek, Joseph T
@ 2015-10-10 18:48 ` Andrei Borzenkov
  2015-10-12 14:19   ` Peter Jones
  2015-10-12 14:47   ` Peter Jones
  2015-10-25 15:41 ` Vladimir 'φ-coder/phcoder' Serbinenko
  2 siblings, 2 replies; 8+ messages in thread
From: Andrei Borzenkov @ 2015-10-10 18:48 UTC (permalink / raw)
  To: The development of GNU GRUB
26.02.2014 02:12, Peter Jones пишет:
> This is version 4.
>
> Changes from version 1:
> - handles SHIFT as a modifier
> - handles F11 and F12 keys
> - uses the handle provided by the system table to find our _EX protocol.
>
> Changes from version 2:
> - eliminate duplicate keycode translation.
>
> Changes from version 3:
> - Do not add the shift modifier for any ascii character between space
>    (0x20) and DEL (0x7f); the combination of the modifier and many of the
>    keys causes it not to be recognized at all.  Specifically, if we
>    include the modifier on any querty punctuation character, i.e.
>    anything the string "~!@#$%^&*()_+{}|:\"<>?" represents in C, it stops
>    being recognized whatsoever.
>
> Changes from version 4:
> - Always initialize term->data from locate protocol (i.e. make it
>    unconditional.)
>
Are there open issues with this patch? Is it used by Fedora? The part 
about SHIFT state bothers me, what happens for non-ASCII printable 
characters? UEFI spec is extremely vague here.
As currently there is no way to actually input Ctrl-X or similar this is 
needed. It may also allow us to actually implement keystatus on EFI.
> Signed-off-by: Peter Jones <pjones@redhat.com>
> ---
>   grub-core/term/efi/console.c | 118 +++++++++++++++++++++++++++++++++++--------
>   include/grub/efi/api.h       |  65 +++++++++++++++++++++++-
>   2 files changed, 161 insertions(+), 22 deletions(-)
>
> diff --git a/grub-core/term/efi/console.c b/grub-core/term/efi/console.c
> index a37eb84..677eab5 100644
> --- a/grub-core/term/efi/console.c
> +++ b/grub-core/term/efi/console.c
> @@ -104,26 +104,12 @@ const unsigned efi_codes[] =
>       GRUB_TERM_KEY_DC, GRUB_TERM_KEY_PPAGE, GRUB_TERM_KEY_NPAGE, GRUB_TERM_KEY_F1,
>       GRUB_TERM_KEY_F2, GRUB_TERM_KEY_F3, GRUB_TERM_KEY_F4, GRUB_TERM_KEY_F5,
>       GRUB_TERM_KEY_F6, GRUB_TERM_KEY_F7, GRUB_TERM_KEY_F8, GRUB_TERM_KEY_F9,
> -    GRUB_TERM_KEY_F10, 0, 0, '\e'
> +    GRUB_TERM_KEY_F10, GRUB_TERM_KEY_F10, GRUB_TERM_KEY_F11, '\e'
>     };
>
> -
>   static int
> -grub_console_getkey (struct grub_term_input *term __attribute__ ((unused)))
> +grub_efi_translate_key (grub_efi_input_key_t key)
>   {
> -  grub_efi_simple_input_interface_t *i;
> -  grub_efi_input_key_t key;
> -  grub_efi_status_t status;
> -
> -  if (grub_efi_is_finished)
> -    return 0;
> -
> -  i = grub_efi_system_table->con_in;
> -  status = efi_call_2 (i->read_key_stroke, i, &key);
> -
> -  if (status != GRUB_EFI_SUCCESS)
> -    return GRUB_TERM_NO_KEY;
> -
>     if (key.scan_code == 0)
>       {
>         /* Some firmware implementations use VT100-style codes against the spec.
> @@ -139,9 +125,98 @@ grub_console_getkey (struct grub_term_input *term __attribute__ ((unused)))
>     else if (key.scan_code < ARRAY_SIZE (efi_codes))
>       return efi_codes[key.scan_code];
>
> +  if (key.unicode_char >= 0x20 && key.unicode_char <= 0x7f)
> +    return key.unicode_char;
> +
>     return GRUB_TERM_NO_KEY;
>   }
>
> +static int
> +grub_console_getkey_con (struct grub_term_input *term __attribute__ ((unused)))
> +{
> +  grub_efi_simple_input_interface_t *i;
> +  grub_efi_input_key_t key;
> +  grub_efi_status_t status;
> +
> +  i = grub_efi_system_table->con_in;
> +  status = efi_call_2 (i->read_key_stroke, i, &key);
> +
> +  if (status != GRUB_EFI_SUCCESS)
> +    return GRUB_TERM_NO_KEY;
> +
> +  return grub_efi_translate_key(key);
> +}
> +
> +static int
> +grub_console_getkey_ex(struct grub_term_input *term)
> +{
> +  grub_efi_key_data_t key_data;
> +  grub_efi_status_t status;
> +  grub_efi_uint32_t kss;
> +  int key = -1;
> +
> +  grub_efi_simple_text_input_ex_interface_t *text_input = term->data;
> +
> +  status = efi_call_2 (text_input->read_key_stroke, text_input, &key_data);
> +
> +  if (status != GRUB_EFI_SUCCESS)
> +    return GRUB_TERM_NO_KEY;
> +
> +  kss = key_data.key_state.key_shift_state;
> +  key = grub_efi_translate_key(key_data.key);
> +
> +  if (key == GRUB_TERM_NO_KEY)
> +    return GRUB_TERM_NO_KEY;
> +
> +  if (kss & GRUB_EFI_SHIFT_STATE_VALID)
> +    {
> +      if ((kss & GRUB_EFI_LEFT_SHIFT_PRESSED
> +	   || kss & GRUB_EFI_RIGHT_SHIFT_PRESSED)
> +	  && !(key >= 0x20 && key <= 0x7f))
> +	key |= GRUB_TERM_SHIFT;
> +      if (kss & GRUB_EFI_LEFT_ALT_PRESSED || kss & GRUB_EFI_RIGHT_ALT_PRESSED)
> +	key |= GRUB_TERM_ALT;
> +      if (kss & GRUB_EFI_LEFT_CONTROL_PRESSED
> +	  || kss & GRUB_EFI_RIGHT_CONTROL_PRESSED)
> +	key |= GRUB_TERM_CTRL;
> +    }
> +
> +  return key;
> +}
> +
> +static grub_err_t
> +grub_efi_console_input_init (struct grub_term_input *term)
> +{
> +  grub_efi_guid_t text_input_ex_guid =
> +    GRUB_EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL_GUID;
> +
> +  if (grub_efi_is_finished)
> +    return 0;
> +
> +  grub_efi_simple_text_input_ex_interface_t *text_input = term->data;
> +  if (text_input)
> +    return 0;
> +
> +  text_input = grub_efi_open_protocol(grub_efi_system_table->console_in_handler,
> +				      &text_input_ex_guid,
> +				      GRUB_EFI_OPEN_PROTOCOL_GET_PROTOCOL);
> +  term->data = (void *)text_input;
> +
> +  return 0;
> +}
> +
> +static int
> +grub_console_getkey (struct grub_term_input *term)
> +{
> +  if (grub_efi_is_finished)
> +    return 0;
> +
> +  if (term->data)
> +    return grub_console_getkey_ex(term);
> +  else
> +    return grub_console_getkey_con(term);
> +}
> +
>   static struct grub_term_coordinate
>   grub_console_getwh (struct grub_term_output *term __attribute__ ((unused)))
>   {
> @@ -243,7 +318,7 @@ grub_console_setcursor (struct grub_term_output *term __attribute__ ((unused)),
>   }
>
>   static grub_err_t
> -grub_efi_console_init (struct grub_term_output *term)
> +grub_efi_console_output_init (struct grub_term_output *term)
>   {
>     grub_efi_set_text_mode (1);
>     grub_console_setcursor (term, 1);
> @@ -251,7 +326,7 @@ grub_efi_console_init (struct grub_term_output *term)
>   }
>
>   static grub_err_t
> -grub_efi_console_fini (struct grub_term_output *term)
> +grub_efi_console_output_fini (struct grub_term_output *term)
>   {
>     grub_console_setcursor (term, 0);
>     grub_efi_set_text_mode (0);
> @@ -262,13 +337,14 @@ static struct grub_term_input grub_console_term_input =
>     {
>       .name = "console",
>       .getkey = grub_console_getkey,
> +    .init = grub_efi_console_input_init,
>     };
>
>   static struct grub_term_output grub_console_term_output =
>     {
>       .name = "console",
> -    .init = grub_efi_console_init,
> -    .fini = grub_efi_console_fini,
> +    .init = grub_efi_console_output_init,
> +    .fini = grub_efi_console_output_fini,
>       .putchar = grub_console_putchar,
>       .getwh = grub_console_getwh,
>       .getxy = grub_console_getxy,
> @@ -291,8 +367,8 @@ grub_console_init (void)
>         return;
>       }
>
> -  grub_term_register_input ("console", &grub_console_term_input);
>     grub_term_register_output ("console", &grub_console_term_output);
> +  grub_term_register_input ("console", &grub_console_term_input);
>   }
>
>   void
> diff --git a/include/grub/efi/api.h b/include/grub/efi/api.h
> index e5dd543..1423403 100644
> --- a/include/grub/efi/api.h
> +++ b/include/grub/efi/api.h
> @@ -111,7 +111,7 @@
>       { 0x8e, 0x39, 0x00, 0xa0, 0xc9, 0x69, 0x72, 0x3b } \
>     }
>
> -#define EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL_GUID \
> +#define GRUB_EFI_SIMPLE_TEXT_INPUT_EX_PROTOCOL_GUID \
>     { 0xdd9e7534, 0x7762, 0x4698, \
>       { 0x8c, 0x14, 0xf5, 0x85, 0x17, 0xa6, 0x25, 0xaa } \
>     }
> @@ -952,6 +952,32 @@ struct grub_efi_input_key
>   };
>   typedef struct grub_efi_input_key grub_efi_input_key_t;
>
> +typedef grub_efi_uint8_t grub_efi_key_toggle_state_t;
> +struct grub_efi_key_state
> +{
> +	grub_efi_uint32_t key_shift_state;
> +	grub_efi_key_toggle_state_t key_toggle_state;
> +};
> +typedef struct grub_efi_key_state grub_efi_key_state_t;
> +
> +#define GRUB_EFI_SHIFT_STATE_VALID     0x80000000
> +#define GRUB_EFI_RIGHT_SHIFT_PRESSED   0x00000001
> +#define GRUB_EFI_LEFT_SHIFT_PRESSED    0x00000002
> +#define GRUB_EFI_RIGHT_CONTROL_PRESSED 0x00000004
> +#define GRUB_EFI_LEFT_CONTROL_PRESSED  0x00000008
> +#define GRUB_EFI_RIGHT_ALT_PRESSED     0x00000010
> +#define GRUB_EFI_LEFT_ALT_PRESSED      0x00000020
> +#define GRUB_EFI_RIGHT_LOGO_PRESSED    0x00000040
> +#define GRUB_EFI_LEFT_LOGO_PRESSED     0x00000080
> +#define GRUB_EFI_MENU_KEY_PRESSED      0x00000100
> +#define GRUB_EFI_SYS_REQ_PRESSED       0x00000200
> +
> +#define GRUB_EFI_TOGGLE_STATE_VALID 0x80
> +#define GRUB_EFI_KEY_STATE_EXPOSED  0x40
> +#define GRUB_EFI_SCROLL_LOCK_ACTIVE 0x01
> +#define GRUB_EFI_NUM_LOCK_ACTIVE    0x02
> +#define GRUB_EFI_CAPS_LOCK_ACTIVE   0x04
> +
>   struct grub_efi_simple_text_output_mode
>   {
>     grub_efi_int32_t max_mode;
> @@ -1294,6 +1320,43 @@ struct grub_efi_simple_input_interface
>   };
>   typedef struct grub_efi_simple_input_interface grub_efi_simple_input_interface_t;
>
> +struct grub_efi_key_data {
> +	grub_efi_input_key_t key;
> +	grub_efi_key_state_t key_state;
> +};
> +typedef struct grub_efi_key_data grub_efi_key_data_t;
> +
> +typedef grub_efi_status_t (*grub_efi_key_notify_function_t) (
> +	grub_efi_key_data_t *key_data
> +	);
> +
> +struct grub_efi_simple_text_input_ex_interface
> +{
> +	grub_efi_status_t
> +	(*reset) (struct grub_efi_simple_text_input_ex_interface *this,
> +		  grub_efi_boolean_t extended_verification);
> +
> +	grub_efi_status_t
> +	(*read_key_stroke) (struct grub_efi_simple_text_input_ex_interface *this,
> +			    grub_efi_key_data_t *key_data);
> +
> +	grub_efi_event_t wait_for_key;
> +
> +	grub_efi_status_t
> +	(*set_state) (struct grub_efi_simple_text_input_ex_interface *this,
> +		      grub_efi_key_toggle_state_t *key_toggle_state);
> +
> +	grub_efi_status_t
> +	(*register_key_notify) (struct grub_efi_simple_text_input_ex_interface *this,
> +				grub_efi_key_data_t *key_data,
> +				grub_efi_key_notify_function_t key_notification_function);
> +
> +	grub_efi_status_t
> +	(*unregister_key_notify) (struct grub_efi_simple_text_input_ex_interface *this,
> +				  void *notification_handle);
> +};
> +typedef struct grub_efi_simple_text_input_ex_interface grub_efi_simple_text_input_ex_interface_t;
> +
>   struct grub_efi_simple_text_output_interface
>   {
>     grub_efi_status_t
>
^ permalink raw reply	[flat|nested] 8+ messages in thread
* Re: [PATCH] Make CTRL and ALT keys work as expected on EFI systems (version 5).
  2015-10-10 18:48 ` Andrei Borzenkov
@ 2015-10-12 14:19   ` Peter Jones
  2015-10-12 14:47   ` Peter Jones
  1 sibling, 0 replies; 8+ messages in thread
From: Peter Jones @ 2015-10-12 14:19 UTC (permalink / raw)
  To: The development of GNU GRUB
On Sat, Oct 10, 2015 at 09:48:58PM +0300, Andrei Borzenkov wrote:
> 26.02.2014 02:12, Peter Jones пишет:
> >This is version 4.
> >
> >Changes from version 1:
> >- handles SHIFT as a modifier
> >- handles F11 and F12 keys
> >- uses the handle provided by the system table to find our _EX protocol.
> >
> >Changes from version 2:
> >- eliminate duplicate keycode translation.
> >
> >Changes from version 3:
> >- Do not add the shift modifier for any ascii character between space
> >   (0x20) and DEL (0x7f); the combination of the modifier and many of the
> >   keys causes it not to be recognized at all.  Specifically, if we
> >   include the modifier on any querty punctuation character, i.e.
> >   anything the string "~!@#$%^&*()_+{}|:\"<>?" represents in C, it stops
> >   being recognized whatsoever.
> >
> >Changes from version 4:
> >- Always initialize term->data from locate protocol (i.e. make it
> >   unconditional.)
> >
> 
> Are there open issues with this patch? Is it used by Fedora? The part about
> SHIFT state bothers me, what happens for non-ASCII printable characters?
> UEFI spec is extremely vague here.
> 
> As currently there is no way to actually input Ctrl-X or similar this is
> needed. It may also allow us to actually implement keystatus on EFI.
Fedora is using it, and has been for 18 months or so.  As far as I know,
there aren't any known issues.
-- 
        Peter
^ permalink raw reply	[flat|nested] 8+ messages in thread
* Re: [PATCH] Make CTRL and ALT keys work as expected on EFI systems (version 5).
  2015-10-10 18:48 ` Andrei Borzenkov
  2015-10-12 14:19   ` Peter Jones
@ 2015-10-12 14:47   ` Peter Jones
  2015-10-13  3:50     ` Andrei Borzenkov
  1 sibling, 1 reply; 8+ messages in thread
From: Peter Jones @ 2015-10-12 14:47 UTC (permalink / raw)
  To: The development of GNU GRUB
On Sat, Oct 10, 2015 at 09:48:58PM +0300, Andrei Borzenkov wrote:
Sorry - realized I should have addressed your question and said one more
thing.
> Are there open issues with this patch? Is it used by Fedora? The part about
> SHIFT state bothers me, what happens for non-ASCII printable characters?
> UEFI spec is extremely vague here.
As for non-ASCII printables, I have no idea.  I literally couldn't find
a keyboard that generated any around my office.  Even my European and
Japanese coworkers don't seem to have them.  That said, given the
comment here:
 https://github.com/vathpela/edk2/blob/master/MdeModulePkg/Bus/Usb/UsbKbDxe/KeyBoard.c#L481
and the two tables here:
 https://github.com/vathpela/edk2/blob/master/MdeModulePkg/Bus/Usb/UsbKbDxe/KeyBoard.c#L36
 https://github.com/vathpela/edk2/blob/master/MdeModulePkg/Bus/Usb/UsbKbDxe/KeyBoard.c#L149
my guess is that on the vast majority of systems they just don't produce
any keypress.  We could do things like make right-alt act like AltGr and
synthesize them ourselves, of course, if there's a significant need.
(Awesomely, AltGr is defined by the EFI modifier list but not mapped
from the USB bits.  Which probably doesn't matter, since nobody uses
keyboards with AltGr keys.)
> As currently there is no way to actually input Ctrl-X or similar this is
> needed. It may also allow us to actually implement keystatus on EFI.
One issue related to implementing keystatus is the "Windows Fast Boot"
feature, enabled by default in nearly all "client" machines now.  (It's
a Windows logo requirement.)  This feature basically says not to probe
USB and the like by default.  So on any machines where your input
device is USB, what winds up happening is that it probes for HII devices
when you call ->read_key_stroke() or ->read_key_stroke_ex() the first
time.  Depending on the hardware configuration, how many devices are
plugged in, how many hubs away the keyboard is, etc., this can take a
surprisingly long period of time.  (FWIW, I *think*
register_key_notify() and set_key_state() will also trigger the driver
loading and probing on most implementations.)
For practical purposes right now, this is pretty much okay with how most
people are using GRUB.  But if somebody wanted to implement "don't show
menus unless the user requested it before the reboot, or the last boot
failed" in their grub config file, then the user gets a much quicker
boot experience if we don't scan the keyboard.
-- 
        Peter
^ permalink raw reply	[flat|nested] 8+ messages in thread
* Re: [PATCH] Make CTRL and ALT keys work as expected on EFI systems (version 5).
  2015-10-12 14:47   ` Peter Jones
@ 2015-10-13  3:50     ` Andrei Borzenkov
  0 siblings, 0 replies; 8+ messages in thread
From: Andrei Borzenkov @ 2015-10-13  3:50 UTC (permalink / raw)
  To: grub-devel; +Cc: edk2-devel-01
12.10.2015 17:47, Peter Jones пишет:
> On Sat, Oct 10, 2015 at 09:48:58PM +0300, Andrei Borzenkov wrote:
>
> Sorry - realized I should have addressed your question and said one more
> thing.
>
>> Are there open issues with this patch? Is it used by Fedora? The part about
>> SHIFT state bothers me, what happens for non-ASCII printable characters?
>> UEFI spec is extremely vague here.
>
> As for non-ASCII printables, I have no idea.  I literally couldn't find
> a keyboard that generated any around my office.  Even my European and
> Japanese coworkers don't seem to have them.  That said, given the
May be we should ignore SHIFT bit for any key event that has valid 
UNICODE character. At least that seems to be what UEFI spec intends to 
say. Code in EDK2 does it only for letters (characters affected by Caps 
Lock) but I do not see how program can usefully interpret SHIFT modifier 
on something like `5' or `/'. Adding EDK2 to Cc.
> comment here:
>   https://github.com/vathpela/edk2/blob/master/MdeModulePkg/Bus/Usb/UsbKbDxe/KeyBoard.c#L481
> and the two tables here:
>   https://github.com/vathpela/edk2/blob/master/MdeModulePkg/Bus/Usb/UsbKbDxe/KeyBoard.c#L36
>   https://github.com/vathpela/edk2/blob/master/MdeModulePkg/Bus/Usb/UsbKbDxe/KeyBoard.c#L149
> my guess is that on the vast majority of systems they just don't produce
> any keypress.  We could do things like make right-alt act like AltGr and
> synthesize them ourselves, of course, if there's a significant need.
> (Awesomely, AltGr is defined by the EFI modifier list but not mapped
> from the USB bits.  Which probably doesn't matter, since nobody uses
> keyboards with AltGr keys.)
>
>> As currently there is no way to actually input Ctrl-X or similar this is
>> needed. It may also allow us to actually implement keystatus on EFI.
>
> One issue related to implementing keystatus is the "Windows Fast Boot"
> feature, enabled by default in nearly all "client" machines now.  (It's
> a Windows logo requirement.)  This feature basically says not to probe
> USB and the like by default.  So on any machines where your input
> device is USB, what winds up happening is that it probes for HII devices
> when you call ->read_key_stroke() or ->read_key_stroke_ex() the first
> time.  Depending on the hardware configuration, how many devices are
> plugged in, how many hubs away the keyboard is, etc., this can take a
> surprisingly long period of time.  (FWIW, I *think*
> register_key_notify() and set_key_state() will also trigger the driver
> loading and probing on most implementations.)
>
But that should be OK. keystatus would set EXPOSE and call 
read_key_stroke_ex which would kick off USB scan. Or so I understand.
> For practical purposes right now, this is pretty much okay with how most
> people are using GRUB.  But if somebody wanted to implement "don't show
> menus unless the user requested it before the reboot, or the last boot
> failed" in their grub config file, then the user gets a much quicker
> boot experience if we don't scan the keyboard.
>
^ permalink raw reply	[flat|nested] 8+ messages in thread
* Re: [PATCH] Make CTRL and ALT keys work as expected on EFI systems (version 5).
  2014-02-25 22:12 [PATCH] Make CTRL and ALT keys work as expected on EFI systems (version 5) Peter Jones
  2014-02-26  2:58 ` Mroczek, Joseph T
  2015-10-10 18:48 ` Andrei Borzenkov
@ 2015-10-25 15:41 ` Vladimir 'φ-coder/phcoder' Serbinenko
  2 siblings, 0 replies; 8+ messages in thread
From: Vladimir 'φ-coder/phcoder' Serbinenko @ 2015-10-25 15:41 UTC (permalink / raw)
  To: The development of GNU GRUB
[-- Attachment #1: Type: text/plain, Size: 2753 bytes --]
On 25.02.2014 23:12, Peter Jones wrote:
> This is version 4.
> 
> Changes from version 1:
> - handles SHIFT as a modifier
> - handles F11 and F12 keys
> - uses the handle provided by the system table to find our _EX protocol.
> 
> Changes from version 2:
> - eliminate duplicate keycode translation.
> 
> Changes from version 3:
> - Do not add the shift modifier for any ascii character between space
>   (0x20) and DEL (0x7f); the combination of the modifier and many of the
>   keys causes it not to be recognized at all.  Specifically, if we
>   include the modifier on any querty punctuation character, i.e.
>   anything the string "~!@#$%^&*()_+{}|:\"<>?" represents in C, it stops
>   being recognized whatsoever.
> 
> Changes from version 4:
> - Always initialize term->data from locate protocol (i.e. make it
>   unconditional.)
> 
> Signed-off-by: Peter Jones <pjones@redhat.com>
> ---
>  grub-core/term/efi/console.c | 118 +++++++++++++++++++++++++++++++++++--------
>  include/grub/efi/api.h       |  65 +++++++++++++++++++++++-
>  2 files changed, 161 insertions(+), 22 deletions(-)
> 
> diff --git a/grub-core/term/efi/console.c b/grub-core/term/efi/console.c
> index a37eb84..677eab5 100644
> --- a/grub-core/term/efi/console.c
> +++ b/grub-core/term/efi/console.c
> @@ -104,26 +104,12 @@ const unsigned efi_codes[] =
>      GRUB_TERM_KEY_DC, GRUB_TERM_KEY_PPAGE, GRUB_TERM_KEY_NPAGE, GRUB_TERM_KEY_F1,
>      GRUB_TERM_KEY_F2, GRUB_TERM_KEY_F3, GRUB_TERM_KEY_F4, GRUB_TERM_KEY_F5,
>      GRUB_TERM_KEY_F6, GRUB_TERM_KEY_F7, GRUB_TERM_KEY_F8, GRUB_TERM_KEY_F9,
> -    GRUB_TERM_KEY_F10, 0, 0, '\e'
> +    GRUB_TERM_KEY_F10, GRUB_TERM_KEY_F10, GRUB_TERM_KEY_F11, '\e'
>    };
>  
Yikes
> -
>  static int
> -grub_console_getkey (struct grub_term_input *term __attribute__ ((unused)))
> +grub_efi_translate_key (grub_efi_input_key_t key)
>  {
> -  grub_efi_simple_input_interface_t *i;
> -  grub_efi_input_key_t key;
> -  grub_efi_status_t status;
> -
> -  if (grub_efi_is_finished)
> -    return 0;
> -
> -  i = grub_efi_system_table->con_in;
> -  status = efi_call_2 (i->read_key_stroke, i, &key);
> -
> -  if (status != GRUB_EFI_SUCCESS)
> -    return GRUB_TERM_NO_KEY;
> -
>    if (key.scan_code == 0)
>      {
>        /* Some firmware implementations use VT100-style codes against the spec.
> @@ -139,9 +125,98 @@ grub_console_getkey (struct grub_term_input *term __attribute__ ((unused)))
>    else if (key.scan_code < ARRAY_SIZE (efi_codes))
>      return efi_codes[key.scan_code];
>  
> +  if (key.unicode_char >= 0x20 && key.unicode_char <= 0x7f)
> +    return key.unicode_char;
> +
This ignores enter, tab and so on.
I fixed 2 above issues and committed.
[-- Attachment #2: OpenPGP digital signature --]
[-- Type: application/pgp-signature, Size: 213 bytes --]
^ permalink raw reply	[flat|nested] 8+ messages in thread
end of thread, other threads:[~2015-10-25 15:41 UTC | newest]
Thread overview: 8+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2014-02-25 22:12 [PATCH] Make CTRL and ALT keys work as expected on EFI systems (version 5) Peter Jones
2014-02-26  2:58 ` Mroczek, Joseph T
2014-02-26 18:51   ` Vladimir 'φ-coder/phcoder' Serbinenko
2015-10-10 18:48 ` Andrei Borzenkov
2015-10-12 14:19   ` Peter Jones
2015-10-12 14:47   ` Peter Jones
2015-10-13  3:50     ` Andrei Borzenkov
2015-10-25 15:41 ` Vladimir 'φ-coder/phcoder' Serbinenko
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).