1c1311ad4SAlexander Graf /*
2c1311ad4SAlexander Graf * EFI application console interface
3c1311ad4SAlexander Graf *
4c1311ad4SAlexander Graf * Copyright (c) 2016 Alexander Graf
5c1311ad4SAlexander Graf *
6c1311ad4SAlexander Graf * SPDX-License-Identifier: GPL-2.0+
7c1311ad4SAlexander Graf */
8c1311ad4SAlexander Graf
9c1311ad4SAlexander Graf #include <common.h>
10*407d40f4SRob Clark #include <charset.h>
11c1311ad4SAlexander Graf #include <efi_loader.h>
12c1311ad4SAlexander Graf
13c1311ad4SAlexander Graf static bool console_size_queried;
14c1311ad4SAlexander Graf
155be8b0a3SEmmanuel Vadot #define EFI_COUT_MODE_2 2
165be8b0a3SEmmanuel Vadot #define EFI_MAX_COUT_MODE 3
175be8b0a3SEmmanuel Vadot
185be8b0a3SEmmanuel Vadot struct cout_mode {
195be8b0a3SEmmanuel Vadot unsigned long columns;
205be8b0a3SEmmanuel Vadot unsigned long rows;
215be8b0a3SEmmanuel Vadot int present;
225be8b0a3SEmmanuel Vadot };
235be8b0a3SEmmanuel Vadot
245be8b0a3SEmmanuel Vadot static struct cout_mode efi_cout_modes[] = {
255be8b0a3SEmmanuel Vadot /* EFI Mode 0 is 80x25 and always present */
265be8b0a3SEmmanuel Vadot {
275be8b0a3SEmmanuel Vadot .columns = 80,
285be8b0a3SEmmanuel Vadot .rows = 25,
295be8b0a3SEmmanuel Vadot .present = 1,
305be8b0a3SEmmanuel Vadot },
315be8b0a3SEmmanuel Vadot /* EFI Mode 1 is always 80x50 */
325be8b0a3SEmmanuel Vadot {
335be8b0a3SEmmanuel Vadot .columns = 80,
345be8b0a3SEmmanuel Vadot .rows = 50,
355be8b0a3SEmmanuel Vadot .present = 0,
365be8b0a3SEmmanuel Vadot },
375be8b0a3SEmmanuel Vadot /* Value are unknown until we query the console */
385be8b0a3SEmmanuel Vadot {
395be8b0a3SEmmanuel Vadot .columns = 0,
405be8b0a3SEmmanuel Vadot .rows = 0,
415be8b0a3SEmmanuel Vadot .present = 0,
425be8b0a3SEmmanuel Vadot },
435be8b0a3SEmmanuel Vadot };
445be8b0a3SEmmanuel Vadot
45c1311ad4SAlexander Graf const efi_guid_t efi_guid_console_control = CONSOLE_CONTROL_GUID;
46c1311ad4SAlexander Graf
47c1311ad4SAlexander Graf #define cESC '\x1b'
48c1311ad4SAlexander Graf #define ESC "\x1b"
49c1311ad4SAlexander Graf
efi_cin_get_mode(struct efi_console_control_protocol * this,int * mode,char * uga_exists,char * std_in_locked)50c1311ad4SAlexander Graf static efi_status_t EFIAPI efi_cin_get_mode(
51c1311ad4SAlexander Graf struct efi_console_control_protocol *this,
52c1311ad4SAlexander Graf int *mode, char *uga_exists, char *std_in_locked)
53c1311ad4SAlexander Graf {
54c1311ad4SAlexander Graf EFI_ENTRY("%p, %p, %p, %p", this, mode, uga_exists, std_in_locked);
55c1311ad4SAlexander Graf
56c1311ad4SAlexander Graf if (mode)
57c1311ad4SAlexander Graf *mode = EFI_CONSOLE_MODE_TEXT;
58c1311ad4SAlexander Graf if (uga_exists)
59c1311ad4SAlexander Graf *uga_exists = 0;
60c1311ad4SAlexander Graf if (std_in_locked)
61c1311ad4SAlexander Graf *std_in_locked = 0;
62c1311ad4SAlexander Graf
63c1311ad4SAlexander Graf return EFI_EXIT(EFI_SUCCESS);
64c1311ad4SAlexander Graf }
65c1311ad4SAlexander Graf
efi_cin_set_mode(struct efi_console_control_protocol * this,int mode)66c1311ad4SAlexander Graf static efi_status_t EFIAPI efi_cin_set_mode(
67c1311ad4SAlexander Graf struct efi_console_control_protocol *this, int mode)
68c1311ad4SAlexander Graf {
69c1311ad4SAlexander Graf EFI_ENTRY("%p, %d", this, mode);
70c1311ad4SAlexander Graf return EFI_EXIT(EFI_UNSUPPORTED);
71c1311ad4SAlexander Graf }
72c1311ad4SAlexander Graf
efi_cin_lock_std_in(struct efi_console_control_protocol * this,uint16_t * password)73c1311ad4SAlexander Graf static efi_status_t EFIAPI efi_cin_lock_std_in(
74c1311ad4SAlexander Graf struct efi_console_control_protocol *this,
75c1311ad4SAlexander Graf uint16_t *password)
76c1311ad4SAlexander Graf {
77c1311ad4SAlexander Graf EFI_ENTRY("%p, %p", this, password);
78c1311ad4SAlexander Graf return EFI_EXIT(EFI_UNSUPPORTED);
79c1311ad4SAlexander Graf }
80c1311ad4SAlexander Graf
81c1311ad4SAlexander Graf const struct efi_console_control_protocol efi_console_control = {
82c1311ad4SAlexander Graf .get_mode = efi_cin_get_mode,
83c1311ad4SAlexander Graf .set_mode = efi_cin_set_mode,
84c1311ad4SAlexander Graf .lock_std_in = efi_cin_lock_std_in,
85c1311ad4SAlexander Graf };
86c1311ad4SAlexander Graf
875be8b0a3SEmmanuel Vadot /* Default to mode 0 */
88c1311ad4SAlexander Graf static struct simple_text_output_mode efi_con_mode = {
895be8b0a3SEmmanuel Vadot .max_mode = 1,
90c1311ad4SAlexander Graf .mode = 0,
91c1311ad4SAlexander Graf .attribute = 0,
92c1311ad4SAlexander Graf .cursor_column = 0,
93c1311ad4SAlexander Graf .cursor_row = 0,
94c1311ad4SAlexander Graf .cursor_visible = 1,
95c1311ad4SAlexander Graf };
96c1311ad4SAlexander Graf
term_read_reply(int * n,int maxnum,char end_char)97c1311ad4SAlexander Graf static int term_read_reply(int *n, int maxnum, char end_char)
98c1311ad4SAlexander Graf {
99c1311ad4SAlexander Graf char c;
100c1311ad4SAlexander Graf int i = 0;
101c1311ad4SAlexander Graf
102c1311ad4SAlexander Graf c = getc();
103c1311ad4SAlexander Graf if (c != cESC)
104c1311ad4SAlexander Graf return -1;
105c1311ad4SAlexander Graf c = getc();
106c1311ad4SAlexander Graf if (c != '[')
107c1311ad4SAlexander Graf return -1;
108c1311ad4SAlexander Graf
109c1311ad4SAlexander Graf n[0] = 0;
110c1311ad4SAlexander Graf while (1) {
111c1311ad4SAlexander Graf c = getc();
112c1311ad4SAlexander Graf if (c == ';') {
113c1311ad4SAlexander Graf i++;
114c1311ad4SAlexander Graf if (i >= maxnum)
115c1311ad4SAlexander Graf return -1;
116c1311ad4SAlexander Graf n[i] = 0;
117c1311ad4SAlexander Graf continue;
118c1311ad4SAlexander Graf } else if (c == end_char) {
119c1311ad4SAlexander Graf break;
120c1311ad4SAlexander Graf } else if (c > '9' || c < '0') {
121c1311ad4SAlexander Graf return -1;
122c1311ad4SAlexander Graf }
123c1311ad4SAlexander Graf
124c1311ad4SAlexander Graf /* Read one more decimal position */
125c1311ad4SAlexander Graf n[i] *= 10;
126c1311ad4SAlexander Graf n[i] += c - '0';
127c1311ad4SAlexander Graf }
128c1311ad4SAlexander Graf
129c1311ad4SAlexander Graf return 0;
130c1311ad4SAlexander Graf }
131c1311ad4SAlexander Graf
efi_cout_reset(struct efi_simple_text_output_protocol * this,char extended_verification)132c1311ad4SAlexander Graf static efi_status_t EFIAPI efi_cout_reset(
133c1311ad4SAlexander Graf struct efi_simple_text_output_protocol *this,
134c1311ad4SAlexander Graf char extended_verification)
135c1311ad4SAlexander Graf {
136c1311ad4SAlexander Graf EFI_ENTRY("%p, %d", this, extended_verification);
137c1311ad4SAlexander Graf return EFI_EXIT(EFI_UNSUPPORTED);
138c1311ad4SAlexander Graf }
139c1311ad4SAlexander Graf
print_unicode_in_utf8(u16 c)140c1311ad4SAlexander Graf static void print_unicode_in_utf8(u16 c)
141c1311ad4SAlexander Graf {
142*407d40f4SRob Clark char utf8[MAX_UTF8_PER_UTF16] = { 0 };
143*407d40f4SRob Clark utf16_to_utf8((u8 *)utf8, &c, 1);
144c1311ad4SAlexander Graf puts(utf8);
145c1311ad4SAlexander Graf }
146c1311ad4SAlexander Graf
efi_cout_output_string(struct efi_simple_text_output_protocol * this,const unsigned short * string)147c1311ad4SAlexander Graf static efi_status_t EFIAPI efi_cout_output_string(
148c1311ad4SAlexander Graf struct efi_simple_text_output_protocol *this,
149c1311ad4SAlexander Graf const unsigned short *string)
150c1311ad4SAlexander Graf {
1515be8b0a3SEmmanuel Vadot struct cout_mode *mode;
152c1311ad4SAlexander Graf u16 ch;
153c1311ad4SAlexander Graf
1545be8b0a3SEmmanuel Vadot mode = &efi_cout_modes[efi_con_mode.mode];
155c1311ad4SAlexander Graf EFI_ENTRY("%p, %p", this, string);
156c1311ad4SAlexander Graf for (;(ch = *string); string++) {
157c1311ad4SAlexander Graf print_unicode_in_utf8(ch);
158c1311ad4SAlexander Graf efi_con_mode.cursor_column++;
159c1311ad4SAlexander Graf if (ch == '\n') {
160c1311ad4SAlexander Graf efi_con_mode.cursor_column = 1;
161c1311ad4SAlexander Graf efi_con_mode.cursor_row++;
1625be8b0a3SEmmanuel Vadot } else if (efi_con_mode.cursor_column > mode->columns) {
163c1311ad4SAlexander Graf efi_con_mode.cursor_column = 1;
164c1311ad4SAlexander Graf efi_con_mode.cursor_row++;
165c1311ad4SAlexander Graf }
1665be8b0a3SEmmanuel Vadot if (efi_con_mode.cursor_row > mode->rows)
1675be8b0a3SEmmanuel Vadot efi_con_mode.cursor_row = mode->rows;
168c1311ad4SAlexander Graf }
169c1311ad4SAlexander Graf
170c1311ad4SAlexander Graf return EFI_EXIT(EFI_SUCCESS);
171c1311ad4SAlexander Graf }
172c1311ad4SAlexander Graf
efi_cout_test_string(struct efi_simple_text_output_protocol * this,const unsigned short * string)173c1311ad4SAlexander Graf static efi_status_t EFIAPI efi_cout_test_string(
174c1311ad4SAlexander Graf struct efi_simple_text_output_protocol *this,
175c1311ad4SAlexander Graf const unsigned short *string)
176c1311ad4SAlexander Graf {
177c1311ad4SAlexander Graf EFI_ENTRY("%p, %p", this, string);
178c1311ad4SAlexander Graf return EFI_EXIT(EFI_SUCCESS);
179c1311ad4SAlexander Graf }
180c1311ad4SAlexander Graf
cout_mode_matches(struct cout_mode * mode,int rows,int cols)1815be8b0a3SEmmanuel Vadot static bool cout_mode_matches(struct cout_mode *mode, int rows, int cols)
1825be8b0a3SEmmanuel Vadot {
1835be8b0a3SEmmanuel Vadot if (!mode->present)
1845be8b0a3SEmmanuel Vadot return false;
1855be8b0a3SEmmanuel Vadot
1865be8b0a3SEmmanuel Vadot return (mode->rows == rows) && (mode->columns == cols);
1875be8b0a3SEmmanuel Vadot }
1885be8b0a3SEmmanuel Vadot
efi_cout_query_mode(struct efi_simple_text_output_protocol * this,unsigned long mode_number,unsigned long * columns,unsigned long * rows)189c1311ad4SAlexander Graf static efi_status_t EFIAPI efi_cout_query_mode(
190c1311ad4SAlexander Graf struct efi_simple_text_output_protocol *this,
191c1311ad4SAlexander Graf unsigned long mode_number, unsigned long *columns,
192c1311ad4SAlexander Graf unsigned long *rows)
193c1311ad4SAlexander Graf {
194c1311ad4SAlexander Graf EFI_ENTRY("%p, %ld, %p, %p", this, mode_number, columns, rows);
195c1311ad4SAlexander Graf
196c1311ad4SAlexander Graf if (!console_size_queried) {
197c1311ad4SAlexander Graf /* Ask the terminal about its size */
198c1311ad4SAlexander Graf int n[3];
1995be8b0a3SEmmanuel Vadot int cols;
2005be8b0a3SEmmanuel Vadot int rows;
201c1311ad4SAlexander Graf u64 timeout;
202c1311ad4SAlexander Graf
203c1311ad4SAlexander Graf console_size_queried = true;
204c1311ad4SAlexander Graf
205c1311ad4SAlexander Graf /* Empty input buffer */
206c1311ad4SAlexander Graf while (tstc())
207c1311ad4SAlexander Graf getc();
208c1311ad4SAlexander Graf
209c1311ad4SAlexander Graf printf(ESC"[18t");
210c1311ad4SAlexander Graf
211c1311ad4SAlexander Graf /* Check if we have a terminal that understands */
212c1311ad4SAlexander Graf timeout = timer_get_us() + 1000000;
213c1311ad4SAlexander Graf while (!tstc())
214c1311ad4SAlexander Graf if (timer_get_us() > timeout)
215c1311ad4SAlexander Graf goto out;
216c1311ad4SAlexander Graf
217c1311ad4SAlexander Graf /* Read {depth,rows,cols} */
218c1311ad4SAlexander Graf if (term_read_reply(n, 3, 't')) {
219c1311ad4SAlexander Graf goto out;
220c1311ad4SAlexander Graf }
221c1311ad4SAlexander Graf
2225be8b0a3SEmmanuel Vadot cols = n[2];
2235be8b0a3SEmmanuel Vadot rows = n[1];
2245be8b0a3SEmmanuel Vadot
2255be8b0a3SEmmanuel Vadot /* Test if we can have Mode 1 */
2265be8b0a3SEmmanuel Vadot if (cols >= 80 && rows >= 50) {
2275be8b0a3SEmmanuel Vadot efi_cout_modes[1].present = 1;
2285be8b0a3SEmmanuel Vadot efi_con_mode.max_mode = 2;
229c1311ad4SAlexander Graf }
230c1311ad4SAlexander Graf
2315be8b0a3SEmmanuel Vadot /*
2325be8b0a3SEmmanuel Vadot * Install our mode as mode 2 if it is different
2335be8b0a3SEmmanuel Vadot * than mode 0 or 1 and set it as the currently selected mode
2345be8b0a3SEmmanuel Vadot */
2355be8b0a3SEmmanuel Vadot if (!cout_mode_matches(&efi_cout_modes[0], rows, cols) &&
2365be8b0a3SEmmanuel Vadot !cout_mode_matches(&efi_cout_modes[1], rows, cols)) {
2375be8b0a3SEmmanuel Vadot efi_cout_modes[EFI_COUT_MODE_2].columns = cols;
2385be8b0a3SEmmanuel Vadot efi_cout_modes[EFI_COUT_MODE_2].rows = rows;
2395be8b0a3SEmmanuel Vadot efi_cout_modes[EFI_COUT_MODE_2].present = 1;
2405be8b0a3SEmmanuel Vadot efi_con_mode.max_mode = EFI_MAX_COUT_MODE;
2415be8b0a3SEmmanuel Vadot efi_con_mode.mode = EFI_COUT_MODE_2;
2425be8b0a3SEmmanuel Vadot }
2435be8b0a3SEmmanuel Vadot }
2445be8b0a3SEmmanuel Vadot
2455be8b0a3SEmmanuel Vadot if (mode_number >= efi_con_mode.max_mode)
2465be8b0a3SEmmanuel Vadot return EFI_EXIT(EFI_UNSUPPORTED);
2475be8b0a3SEmmanuel Vadot
2485be8b0a3SEmmanuel Vadot if (efi_cout_modes[mode_number].present != 1)
2495be8b0a3SEmmanuel Vadot return EFI_EXIT(EFI_UNSUPPORTED);
2505be8b0a3SEmmanuel Vadot
251c1311ad4SAlexander Graf out:
252c1311ad4SAlexander Graf if (columns)
2535be8b0a3SEmmanuel Vadot *columns = efi_cout_modes[mode_number].columns;
254c1311ad4SAlexander Graf if (rows)
2555be8b0a3SEmmanuel Vadot *rows = efi_cout_modes[mode_number].rows;
256c1311ad4SAlexander Graf
257c1311ad4SAlexander Graf return EFI_EXIT(EFI_SUCCESS);
258c1311ad4SAlexander Graf }
259c1311ad4SAlexander Graf
efi_cout_set_mode(struct efi_simple_text_output_protocol * this,unsigned long mode_number)260c1311ad4SAlexander Graf static efi_status_t EFIAPI efi_cout_set_mode(
261c1311ad4SAlexander Graf struct efi_simple_text_output_protocol *this,
262c1311ad4SAlexander Graf unsigned long mode_number)
263c1311ad4SAlexander Graf {
264c1311ad4SAlexander Graf EFI_ENTRY("%p, %ld", this, mode_number);
265c1311ad4SAlexander Graf
266c1311ad4SAlexander Graf
2675be8b0a3SEmmanuel Vadot if (mode_number > efi_con_mode.max_mode)
268c1311ad4SAlexander Graf return EFI_EXIT(EFI_UNSUPPORTED);
2695be8b0a3SEmmanuel Vadot
2705be8b0a3SEmmanuel Vadot efi_con_mode.mode = mode_number;
2715be8b0a3SEmmanuel Vadot efi_con_mode.cursor_column = 0;
2725be8b0a3SEmmanuel Vadot efi_con_mode.cursor_row = 0;
2735be8b0a3SEmmanuel Vadot
2745be8b0a3SEmmanuel Vadot return EFI_EXIT(EFI_SUCCESS);
275c1311ad4SAlexander Graf }
276c1311ad4SAlexander Graf
efi_cout_set_attribute(struct efi_simple_text_output_protocol * this,unsigned long attribute)277c1311ad4SAlexander Graf static efi_status_t EFIAPI efi_cout_set_attribute(
278c1311ad4SAlexander Graf struct efi_simple_text_output_protocol *this,
279c1311ad4SAlexander Graf unsigned long attribute)
280c1311ad4SAlexander Graf {
281c1311ad4SAlexander Graf EFI_ENTRY("%p, %lx", this, attribute);
282c1311ad4SAlexander Graf
283c1311ad4SAlexander Graf /* Just ignore attributes (colors) for now */
284c1311ad4SAlexander Graf return EFI_EXIT(EFI_UNSUPPORTED);
285c1311ad4SAlexander Graf }
286c1311ad4SAlexander Graf
efi_cout_clear_screen(struct efi_simple_text_output_protocol * this)287c1311ad4SAlexander Graf static efi_status_t EFIAPI efi_cout_clear_screen(
288c1311ad4SAlexander Graf struct efi_simple_text_output_protocol *this)
289c1311ad4SAlexander Graf {
290c1311ad4SAlexander Graf EFI_ENTRY("%p", this);
291c1311ad4SAlexander Graf
292c1311ad4SAlexander Graf printf(ESC"[2J");
293c1311ad4SAlexander Graf
294c1311ad4SAlexander Graf return EFI_EXIT(EFI_SUCCESS);
295c1311ad4SAlexander Graf }
296c1311ad4SAlexander Graf
efi_cout_set_cursor_position(struct efi_simple_text_output_protocol * this,unsigned long column,unsigned long row)297c1311ad4SAlexander Graf static efi_status_t EFIAPI efi_cout_set_cursor_position(
298c1311ad4SAlexander Graf struct efi_simple_text_output_protocol *this,
299c1311ad4SAlexander Graf unsigned long column, unsigned long row)
300c1311ad4SAlexander Graf {
301c1311ad4SAlexander Graf EFI_ENTRY("%p, %ld, %ld", this, column, row);
302c1311ad4SAlexander Graf
303c1311ad4SAlexander Graf printf(ESC"[%d;%df", (int)row, (int)column);
304c1311ad4SAlexander Graf efi_con_mode.cursor_column = column;
305c1311ad4SAlexander Graf efi_con_mode.cursor_row = row;
306c1311ad4SAlexander Graf
307c1311ad4SAlexander Graf return EFI_EXIT(EFI_SUCCESS);
308c1311ad4SAlexander Graf }
309c1311ad4SAlexander Graf
efi_cout_enable_cursor(struct efi_simple_text_output_protocol * this,bool enable)310c1311ad4SAlexander Graf static efi_status_t EFIAPI efi_cout_enable_cursor(
311c1311ad4SAlexander Graf struct efi_simple_text_output_protocol *this,
312c1311ad4SAlexander Graf bool enable)
313c1311ad4SAlexander Graf {
314c1311ad4SAlexander Graf EFI_ENTRY("%p, %d", this, enable);
315c1311ad4SAlexander Graf
316c1311ad4SAlexander Graf printf(ESC"[?25%c", enable ? 'h' : 'l');
317c1311ad4SAlexander Graf
318c1311ad4SAlexander Graf return EFI_EXIT(EFI_SUCCESS);
319c1311ad4SAlexander Graf }
320c1311ad4SAlexander Graf
321c1311ad4SAlexander Graf const struct efi_simple_text_output_protocol efi_con_out = {
322c1311ad4SAlexander Graf .reset = efi_cout_reset,
323c1311ad4SAlexander Graf .output_string = efi_cout_output_string,
324c1311ad4SAlexander Graf .test_string = efi_cout_test_string,
325c1311ad4SAlexander Graf .query_mode = efi_cout_query_mode,
326c1311ad4SAlexander Graf .set_mode = efi_cout_set_mode,
327c1311ad4SAlexander Graf .set_attribute = efi_cout_set_attribute,
328c1311ad4SAlexander Graf .clear_screen = efi_cout_clear_screen,
329c1311ad4SAlexander Graf .set_cursor_position = efi_cout_set_cursor_position,
330c1311ad4SAlexander Graf .enable_cursor = efi_cout_enable_cursor,
331c1311ad4SAlexander Graf .mode = (void*)&efi_con_mode,
332c1311ad4SAlexander Graf };
333c1311ad4SAlexander Graf
efi_cin_reset(struct efi_simple_input_interface * this,bool extended_verification)334c1311ad4SAlexander Graf static efi_status_t EFIAPI efi_cin_reset(
335c1311ad4SAlexander Graf struct efi_simple_input_interface *this,
336c1311ad4SAlexander Graf bool extended_verification)
337c1311ad4SAlexander Graf {
338c1311ad4SAlexander Graf EFI_ENTRY("%p, %d", this, extended_verification);
339c1311ad4SAlexander Graf return EFI_EXIT(EFI_UNSUPPORTED);
340c1311ad4SAlexander Graf }
341c1311ad4SAlexander Graf
efi_cin_read_key_stroke(struct efi_simple_input_interface * this,struct efi_input_key * key)342c1311ad4SAlexander Graf static efi_status_t EFIAPI efi_cin_read_key_stroke(
343c1311ad4SAlexander Graf struct efi_simple_input_interface *this,
344c1311ad4SAlexander Graf struct efi_input_key *key)
345c1311ad4SAlexander Graf {
346c1311ad4SAlexander Graf struct efi_input_key pressed_key = {
347c1311ad4SAlexander Graf .scan_code = 0,
348c1311ad4SAlexander Graf .unicode_char = 0,
349c1311ad4SAlexander Graf };
350c1311ad4SAlexander Graf char ch;
351c1311ad4SAlexander Graf
352c1311ad4SAlexander Graf EFI_ENTRY("%p, %p", this, key);
353c1311ad4SAlexander Graf
354c1311ad4SAlexander Graf /* We don't do interrupts, so check for timers cooperatively */
355c1311ad4SAlexander Graf efi_timer_check();
356c1311ad4SAlexander Graf
357c1311ad4SAlexander Graf if (!tstc()) {
358c1311ad4SAlexander Graf /* No key pressed */
359c1311ad4SAlexander Graf return EFI_EXIT(EFI_NOT_READY);
360c1311ad4SAlexander Graf }
361c1311ad4SAlexander Graf
362c1311ad4SAlexander Graf ch = getc();
363c1311ad4SAlexander Graf if (ch == cESC) {
364c1311ad4SAlexander Graf /* Escape Sequence */
365c1311ad4SAlexander Graf ch = getc();
366c1311ad4SAlexander Graf switch (ch) {
367c1311ad4SAlexander Graf case cESC: /* ESC */
368c1311ad4SAlexander Graf pressed_key.scan_code = 23;
369c1311ad4SAlexander Graf break;
370c1311ad4SAlexander Graf case 'O': /* F1 - F4 */
371c1311ad4SAlexander Graf pressed_key.scan_code = getc() - 'P' + 11;
372c1311ad4SAlexander Graf break;
373c1311ad4SAlexander Graf case 'a'...'z':
374c1311ad4SAlexander Graf ch = ch - 'a';
375c1311ad4SAlexander Graf break;
376c1311ad4SAlexander Graf case '[':
377c1311ad4SAlexander Graf ch = getc();
378c1311ad4SAlexander Graf switch (ch) {
379c1311ad4SAlexander Graf case 'A'...'D': /* up, down right, left */
380c1311ad4SAlexander Graf pressed_key.scan_code = ch - 'A' + 1;
381c1311ad4SAlexander Graf break;
382c1311ad4SAlexander Graf case 'F': /* End */
383c1311ad4SAlexander Graf pressed_key.scan_code = 6;
384c1311ad4SAlexander Graf break;
385c1311ad4SAlexander Graf case 'H': /* Home */
386c1311ad4SAlexander Graf pressed_key.scan_code = 5;
387c1311ad4SAlexander Graf break;
388c1311ad4SAlexander Graf case '1': /* F5 - F8 */
389c1311ad4SAlexander Graf pressed_key.scan_code = getc() - '0' + 11;
390c1311ad4SAlexander Graf getc();
391c1311ad4SAlexander Graf break;
392c1311ad4SAlexander Graf case '2': /* F9 - F12 */
393c1311ad4SAlexander Graf pressed_key.scan_code = getc() - '0' + 19;
394c1311ad4SAlexander Graf getc();
395c1311ad4SAlexander Graf break;
396c1311ad4SAlexander Graf case '3': /* DEL */
397c1311ad4SAlexander Graf pressed_key.scan_code = 8;
398c1311ad4SAlexander Graf getc();
399c1311ad4SAlexander Graf break;
400c1311ad4SAlexander Graf }
401c1311ad4SAlexander Graf break;
402c1311ad4SAlexander Graf }
403c1311ad4SAlexander Graf } else if (ch == 0x7f) {
404c1311ad4SAlexander Graf /* Backspace */
405c1311ad4SAlexander Graf ch = 0x08;
406c1311ad4SAlexander Graf }
407c1311ad4SAlexander Graf pressed_key.unicode_char = ch;
408c1311ad4SAlexander Graf *key = pressed_key;
409c1311ad4SAlexander Graf
410c1311ad4SAlexander Graf return EFI_EXIT(EFI_SUCCESS);
411c1311ad4SAlexander Graf }
412c1311ad4SAlexander Graf
41391be9a77Sxypron.glpk@gmx.de struct efi_simple_input_interface efi_con_in = {
414c1311ad4SAlexander Graf .reset = efi_cin_reset,
415c1311ad4SAlexander Graf .read_key_stroke = efi_cin_read_key_stroke,
416c1311ad4SAlexander Graf .wait_for_key = NULL,
417c1311ad4SAlexander Graf };
41891be9a77Sxypron.glpk@gmx.de
41991be9a77Sxypron.glpk@gmx.de static struct efi_event *console_timer_event;
42091be9a77Sxypron.glpk@gmx.de
efi_key_notify(struct efi_event * event,void * context)421ff925938Sxypron.glpk@gmx.de static void EFIAPI efi_key_notify(struct efi_event *event, void *context)
42291be9a77Sxypron.glpk@gmx.de {
42391be9a77Sxypron.glpk@gmx.de }
42491be9a77Sxypron.glpk@gmx.de
efi_console_timer_notify(struct efi_event * event,void * context)425ff925938Sxypron.glpk@gmx.de static void EFIAPI efi_console_timer_notify(struct efi_event *event,
426ff925938Sxypron.glpk@gmx.de void *context)
42791be9a77Sxypron.glpk@gmx.de {
42891be9a77Sxypron.glpk@gmx.de EFI_ENTRY("%p, %p", event, context);
42991be9a77Sxypron.glpk@gmx.de if (tstc())
43091be9a77Sxypron.glpk@gmx.de efi_signal_event(efi_con_in.wait_for_key);
43191be9a77Sxypron.glpk@gmx.de EFI_EXIT(EFI_SUCCESS);
43291be9a77Sxypron.glpk@gmx.de }
43391be9a77Sxypron.glpk@gmx.de
434a17e62ccSRob Clark
435a17e62ccSRob Clark static struct efi_object efi_console_control_obj =
436a17e62ccSRob Clark EFI_PROTOCOL_OBJECT(efi_guid_console_control, &efi_console_control);
437a17e62ccSRob Clark static struct efi_object efi_console_output_obj =
438a17e62ccSRob Clark EFI_PROTOCOL_OBJECT(EFI_SIMPLE_TEXT_OUTPUT_PROTOCOL_GUID, &efi_con_out);
439a17e62ccSRob Clark static struct efi_object efi_console_input_obj =
440a17e62ccSRob Clark EFI_PROTOCOL_OBJECT(EFI_SIMPLE_TEXT_INPUT_PROTOCOL_GUID, &efi_con_in);
441a17e62ccSRob Clark
44291be9a77Sxypron.glpk@gmx.de /* This gets called from do_bootefi_exec(). */
efi_console_register(void)44391be9a77Sxypron.glpk@gmx.de int efi_console_register(void)
44491be9a77Sxypron.glpk@gmx.de {
44591be9a77Sxypron.glpk@gmx.de efi_status_t r;
446a17e62ccSRob Clark
447a17e62ccSRob Clark /* Hook up to the device list */
448a17e62ccSRob Clark list_add_tail(&efi_console_control_obj.link, &efi_obj_list);
449a17e62ccSRob Clark list_add_tail(&efi_console_output_obj.link, &efi_obj_list);
450a17e62ccSRob Clark list_add_tail(&efi_console_input_obj.link, &efi_obj_list);
451a17e62ccSRob Clark
45291be9a77Sxypron.glpk@gmx.de r = efi_create_event(EVT_NOTIFY_WAIT, TPL_CALLBACK,
45391be9a77Sxypron.glpk@gmx.de efi_key_notify, NULL, &efi_con_in.wait_for_key);
45491be9a77Sxypron.glpk@gmx.de if (r != EFI_SUCCESS) {
45591be9a77Sxypron.glpk@gmx.de printf("ERROR: Failed to register WaitForKey event\n");
45691be9a77Sxypron.glpk@gmx.de return r;
45791be9a77Sxypron.glpk@gmx.de }
45891be9a77Sxypron.glpk@gmx.de r = efi_create_event(EVT_TIMER | EVT_NOTIFY_SIGNAL, TPL_CALLBACK,
45991be9a77Sxypron.glpk@gmx.de efi_console_timer_notify, NULL,
46091be9a77Sxypron.glpk@gmx.de &console_timer_event);
46191be9a77Sxypron.glpk@gmx.de if (r != EFI_SUCCESS) {
46291be9a77Sxypron.glpk@gmx.de printf("ERROR: Failed to register console event\n");
46391be9a77Sxypron.glpk@gmx.de return r;
46491be9a77Sxypron.glpk@gmx.de }
46591be9a77Sxypron.glpk@gmx.de /* 5000 ns cycle is sufficient for 2 MBaud */
46691be9a77Sxypron.glpk@gmx.de r = efi_set_timer(console_timer_event, EFI_TIMER_PERIODIC, 50);
46791be9a77Sxypron.glpk@gmx.de if (r != EFI_SUCCESS)
46891be9a77Sxypron.glpk@gmx.de printf("ERROR: Failed to set console timer\n");
46991be9a77Sxypron.glpk@gmx.de return r;
47091be9a77Sxypron.glpk@gmx.de }
471