blob: f571d564ff299ab8618e381d935ea2cef67b2081 [file] [log] [blame]
Jordan Crousef6145c32008-03-19 23:56:58 +00001/*
2 * This file is part of the libpayload project.
3 *
4 * Copyright (C) 2007 Uwe Hermann <uwe@hermann-uwe.de>
5 * Copyright (C) 2008 Advanced Micro Devices, Inc.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * 3. The name of the author may not be used to endorse or promote products
16 * derived from this software without specific prior written permission.
17 *
18 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
19 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
22 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28 * SUCH DAMAGE.
29 */
30
31/*
32 * This file handles reading keystrokes from serial and the console
33 * and "cooking" them so that they are correct for curses.
34 * Also, implement key related functions (mainly wgetch)
35 *
36 * TODO:
37 * Actually cook the serial (handle special keys)
38 */
39
Jordan Crousedb8c0ab2008-11-24 17:54:46 +000040#include <libpayload-config.h>
Patrick Georgi4727c072008-10-16 19:20:51 +000041#include <usb/usb.h>
Jordan Crousef6145c32008-03-19 23:56:58 +000042#include "local.h"
43
Jordan Crouse672d0ae2008-04-08 23:21:33 +000044static int _halfdelay = 0;
45
Jordan Crousef6145c32008-03-19 23:56:58 +000046/* ============== Serial ==================== */
47
Stefan Reinauer1b4d3942015-06-29 15:47:34 -070048#if IS_ENABLED(CONFIG_LP_SERIAL_CONSOLE)
Jordan Crouse3b470652008-06-20 00:01:42 +000049/* We treat serial like a vt100 terminal. For now we
50 do the cooking in here, but we should probably eventually
51 pass it to dedicated vt100 code */
52
Patrick Georgic4f294f2008-10-23 12:22:24 +000053static int getkeyseq(char *buffer, int len, int max)
Jordan Crouse3b470652008-06-20 00:01:42 +000054{
55 int i;
56
Patrick Georgic4f294f2008-10-23 12:22:24 +000057 while (1) {
58 for(i = 0; i < 75; i++) {
59 if (serial_havechar())
60 break;
61 mdelay(1);
62 }
63
64 if (i == 75)
65 return len;
66
67 buffer[len++] = serial_getchar();
68 if (len == max)
69 return len;
Jordan Crouse3b470652008-06-20 00:01:42 +000070 }
Jordan Crouse3b470652008-06-20 00:01:42 +000071}
72
73static struct {
Patrick Georgi7f965832011-04-21 18:57:16 +020074 const char *seq;
Jordan Crouse3b470652008-06-20 00:01:42 +000075 int key;
76} escape_codes[] = {
77 { "[A", KEY_UP },
78 { "[B", KEY_DOWN },
79 { "[C", KEY_RIGHT },
80 { "[D", KEY_LEFT },
Ulf Jordanf044f282008-09-05 21:23:02 +000081 { "[F", KEY_END },
82 { "[H", KEY_HOME },
83 { "[2~", KEY_IC },
84 { "[3~", KEY_DC },
85 { "[5~", KEY_PPAGE },
86 { "[6~", KEY_NPAGE },
Jordan Crouse3b470652008-06-20 00:01:42 +000087 { "OP", KEY_F(1) },
88 { "OQ", KEY_F(2) },
89 { "OR", KEY_F(3) },
90 { "OS", KEY_F(4) },
91 { "[15~", KEY_F(5) },
92 { "[17~", KEY_F(6) },
93 { "[18~", KEY_F(7) },
94 { "[19~", KEY_F(8) },
95 { "[20~", KEY_F(9) },
96 { "[21~", KEY_F(10) },
Ulf Jordanf044f282008-09-05 21:23:02 +000097 { "[23~", KEY_F(11) },
Jordan Crouse3b470652008-06-20 00:01:42 +000098 { "[24~", KEY_F(12) },
99 { NULL },
100};
101
102static int handle_escape(void)
103{
104 char buffer[5];
Patrick Georgic4f294f2008-10-23 12:22:24 +0000105 int len = getkeyseq(buffer, 0, sizeof(buffer));
Jordan Crouse3b470652008-06-20 00:01:42 +0000106 int i, t;
107
108 if (len == 0)
109 return 27;
110
111 for(i = 0; escape_codes[i].seq != NULL; i++) {
Patrick Georgi7f965832011-04-21 18:57:16 +0200112 const char *p = escape_codes[i].seq;
Jordan Crouse3b470652008-06-20 00:01:42 +0000113
114 for(t = 0; t < len; t++) {
115 if (!*p || *p != buffer[t])
116 break;
117 p++;
118 }
119
120 if (t == len)
121 return escape_codes[i].key;
122 }
123
124 return 0;
125}
Jordan Crousef6145c32008-03-19 23:56:58 +0000126
127static int cook_serial(unsigned char ch)
128{
Jordan Crouse3b470652008-06-20 00:01:42 +0000129 switch(ch) {
130 case 8:
131 return KEY_BACKSPACE;
132
133 case 13:
134 return KEY_ENTER;
135
136 case 27:
137 return handle_escape();
138
139 default:
140 return ch;
141 }
Jordan Crousef6145c32008-03-19 23:56:58 +0000142}
Stefan Reinauere75c3d82008-09-26 18:36:26 +0000143#endif
Jordan Crousef6145c32008-03-19 23:56:58 +0000144
145/* ================ Keyboard ================ */
146
Patrick Georgi7f965832011-04-21 18:57:16 +0200147static int curses_getchar(int _delay)
Uwe Hermann6a441bf2008-03-20 19:54:59 +0000148{
Stefan Reinauer1b4d3942015-06-29 15:47:34 -0700149#if IS_ENABLED(CONFIG_LP_USB_HID) || IS_ENABLED(CONFIG_LP_PC_KEYBOARD) || \
150 IS_ENABLED(CONFIG_LP_SERIAL_CONSOLE)
Patrick Georgie6408a32008-09-17 18:12:46 +0000151 unsigned short c;
Mart Raudseppc5a2ec62009-02-22 23:13:33 +0000152#endif
Jordan Crousef6145c32008-03-19 23:56:58 +0000153
154 do {
Stefan Reinauer1b4d3942015-06-29 15:47:34 -0700155#if IS_ENABLED(CONFIG_LP_USB_HID)
Patrick Georgi4727c072008-10-16 19:20:51 +0000156 usb_poll();
157 if ((curses_flags & F_ENABLE_CONSOLE) &&
158 usbhid_havechar()) {
159 c = usbhid_getchar();
160 if (c != 0) return c;
161 }
162#endif
Stefan Reinauer1b4d3942015-06-29 15:47:34 -0700163#if IS_ENABLED(CONFIG_LP_PC_KEYBOARD)
Patrick Georgie6408a32008-09-17 18:12:46 +0000164 if ((curses_flags & F_ENABLE_CONSOLE) &&
165 keyboard_havechar()) {
166 c = keyboard_getchar();
167 if (c != 0) return c;
168 }
169#endif
Jordan Crousef6145c32008-03-19 23:56:58 +0000170
Stefan Reinauer1b4d3942015-06-29 15:47:34 -0700171#if IS_ENABLED(CONFIG_LP_SERIAL_CONSOLE)
Patrick Georgie6408a32008-09-17 18:12:46 +0000172 if ((curses_flags & F_ENABLE_SERIAL) &&
173 serial_havechar()) {
174 c = serial_getchar();
175 return cook_serial(c);
176 }
177#endif
Jordan Crousef6145c32008-03-19 23:56:58 +0000178
Nico Huber6a008362013-06-25 15:19:48 +0200179 if (_delay == 0) {
Patrick Georgie6408a32008-09-17 18:12:46 +0000180 break;
Nico Huber6a008362013-06-25 15:19:48 +0200181 } else if (_delay >= 10) {
182 mdelay(10);
183 _delay -= 10;
184 } else if (_delay > 0) {
185 mdelay(_delay);
186 _delay = 0;
Jordan Crousef6145c32008-03-19 23:56:58 +0000187 }
Uwe Hermann6a441bf2008-03-20 19:54:59 +0000188 } while (1);
Jordan Crousef6145c32008-03-19 23:56:58 +0000189
190 return ERR;
191}
192
193/* === Public functions === */
194
195int wgetch(WINDOW *win)
196{
Patrick Georgi7f965832011-04-21 18:57:16 +0200197 int _delay = -1;
Jordan Crouse672d0ae2008-04-08 23:21:33 +0000198
Stefan Reinauer131c0072009-07-31 11:38:59 +0000199 if (_halfdelay)
Patrick Georgi7f965832011-04-21 18:57:16 +0200200 _delay = _halfdelay;
Stefan Reinauer14e22772010-04-27 06:56:47 +0000201 else
Patrick Georgi7f965832011-04-21 18:57:16 +0200202 _delay = win->_delay;
Jordan Crouse672d0ae2008-04-08 23:21:33 +0000203
Patrick Georgi7f965832011-04-21 18:57:16 +0200204 return curses_getchar(_delay);
Jordan Crousef6145c32008-03-19 23:56:58 +0000205}
206
207int nodelay(WINDOW *win, NCURSES_BOOL flag)
208{
Stefan Reinauer131c0072009-07-31 11:38:59 +0000209 win->_delay = flag ? 0 : -1;
Jordan Crouse672d0ae2008-04-08 23:21:33 +0000210 return 0;
211}
212
213int halfdelay(int tenths)
214{
215 if (tenths > 255)
216 return ERR;
217
218 _halfdelay = tenths;
219 return 0;
220}
221
222int nocbreak(void)
223{
224 /* Remove half delay timeout. */
225 _halfdelay = 0;
Uwe Hermann14a3feb2008-03-20 20:46:44 +0000226 return 0;
Jordan Crousef6145c32008-03-19 23:56:58 +0000227}
228
Stefan Reinauer1b4d3942015-06-29 15:47:34 -0700229#if IS_ENABLED(CONFIG_LP_VGA_VIDEO_CONSOLE)
Jordan Crousef6145c32008-03-19 23:56:58 +0000230void curses_enable_vga(int state)
231{
232 if (state)
233 curses_flags |= F_ENABLE_CONSOLE;
234 else
235 curses_flags &= ~F_ENABLE_CONSOLE;
236}
Stefan Reinauere75c3d82008-09-26 18:36:26 +0000237
238int curses_vga_enabled(void)
239{
240 return (curses_flags & F_ENABLE_CONSOLE) != 0;
241}
Jordan Crousef6145c32008-03-19 23:56:58 +0000242#else
243void curses_enable_vga(int state) { }
Stefan Reinauere75c3d82008-09-26 18:36:26 +0000244int curses_vga_enabled(void) { return 0; }
Jordan Crousef6145c32008-03-19 23:56:58 +0000245#endif
246
Stefan Reinauer1b4d3942015-06-29 15:47:34 -0700247#if IS_ENABLED(CONFIG_LP_SERIAL_CONSOLE)
Jordan Crousef6145c32008-03-19 23:56:58 +0000248void curses_enable_serial(int state)
249{
250 if (state)
251 curses_flags |= F_ENABLE_SERIAL;
252 else
253 curses_flags &= ~F_ENABLE_SERIAL;
254}
Stefan Reinauere75c3d82008-09-26 18:36:26 +0000255
256int curses_serial_enabled(void)
257{
258 return (curses_flags & F_ENABLE_SERIAL) != 0;
259}
260
Jordan Crousef6145c32008-03-19 23:56:58 +0000261#else
262void curses_enable_serial(int state) { }
Stefan Reinauere75c3d82008-09-26 18:36:26 +0000263int curses_serial_enabled(void) { return 0; }
Jordan Crousef6145c32008-03-19 23:56:58 +0000264#endif