xref: /reactos/boot/armllb/hw/keyboard.c (revision 84ccccab)
1 /*
2  * PROJECT:         ReactOS Boot Loader
3  * LICENSE:         BSD - See COPYING.ARM in the top level directory
4  * FILE:            boot/armllb/hw/keyboard.c
5  * PURPOSE:         LLB Keyboard Routines
6  * PROGRAMMERS:     ReactOS Portable Systems Group
7  */
8 
9 #include "precomp.h"
10 
11 #define E0_KPENTER      96
12 #define E0_RCTRL        97
13 #define E0_KPSLASH      98
14 #define E0_PRSCR        99
15 #define E0_RALT         100
16 #define E0_BREAK        101
17 #define E0_HOME         102
18 #define E0_UP           103
19 #define E0_PGUP         104
20 #define E0_LEFT         105
21 #define E0_RIGHT        106
22 #define E0_END          107
23 #define E0_DOWN         108
24 #define E0_PGDN         109
25 #define E0_INS          110
26 #define E0_DEL          111
27 #define E1_PAUSE        119
28 #define E0_MACRO        112
29 #define E0_F13          113
30 #define E0_F14          114
31 #define E0_HELP         115
32 #define E0_DO           116
33 #define E0_F17          117
34 #define E0_KPMINPLUS    118
35 #define E0_OK           124
36 #define E0_MSLW         125
37 #define E0_MSRW         126
38 #define E0_MSTM         127
39 
40 /* US 101 KEYBOARD LAYOUT *****************************************************/
41 
42 CHAR LlbHwScanCodeToAsciiTable[58][2] =
43 {
44     {   0,0   } ,
45     {  27, 27 } ,
46     { '1','!' } ,
47     { '2','@' } ,
48     { '3','#' } ,
49     { '4','$' } ,
50     { '5','%' } ,
51     { '6','^' } ,
52     { '7','&' } ,
53     { '8','*' } ,
54     { '9','(' } ,
55     { '0',')' } ,
56     { '-','_' } ,
57     { '=','+' } ,
58     {   8,8   } ,
59     {   9,9   } ,
60     { 'q','Q' } ,
61     { 'w','W' } ,
62     { 'e','E' } ,
63     { 'r','R' } ,
64     { 't','T' } ,
65     { 'y','Y' } ,
66     { 'u','U' } ,
67     { 'i','I' } ,
68     { 'o','O' } ,
69     { 'p','P' } ,
70     { '[','{' } ,
71     { ']','}' } ,
72     {  13,13  } ,
73     {   0,0   } ,
74     { 'a','A' } ,
75     { 's','S' } ,
76     { 'd','D' } ,
77     { 'f','F' } ,
78     { 'g','G' } ,
79     { 'h','H' } ,
80     { 'j','J' } ,
81     { 'k','K' } ,
82     { 'l','L' } ,
83     { ';',':' } ,
84     {  39,34  } ,
85     { '`','~' } ,
86     {   0,0   } ,
87     { '\\','|'} ,
88     { 'z','Z' } ,
89     { 'x','X' } ,
90     { 'c','C' } ,
91     { 'v','V' } ,
92     { 'b','B' } ,
93     { 'n','N' } ,
94     { 'm','M' } ,
95     { ',','<' } ,
96     { '.','>' } ,
97     { '/','?' } ,
98     {   0,0   } ,
99     {   0,0   } ,
100     {   0,0   } ,
101     { ' ',' ' } ,
102 };
103 
104 /* EXTENDED KEY TABLE *********************************************************/
105 
106 UCHAR LlbHwExtendedScanCodeTable[128] =
107 {
108 	0,		0,		0,		0,
109     0,		0,		0,		0,
110  	0,		0,		0,		0,
111  	0,		0,		0,		0,
112  	0,		0,		0,		0,
113  	0,		0,		0,		0,
114  	0,		0,		0,		0,
115  	E0_KPENTER,	E0_RCTRL,	0,		0,
116  	0,		0,		0,		0,
117  	0,		0,		0,		0,
118  	0,		0,		0,		0,
119  	0,		0,		0,		0,
120  	0,		0,		0,		0,
121  	0,		E0_KPSLASH,	0,		E0_PRSCR,
122  	E0_RALT,	0,		0,		0,
123  	0,		E0_F13,		E0_F14,		E0_HELP,
124  	E0_DO,		E0_F17,		0,		0,
125  	0,		0,		E0_BREAK,	E0_HOME,
126  	E0_UP,		E0_PGUP,	0,		E0_LEFT,
127  	E0_OK,		E0_RIGHT,	E0_KPMINPLUS,	E0_END,
128  	E0_DOWN,	E0_PGDN,	E0_INS,		E0_DEL,
129  	0,		0,		0,		0,
130  	0,		0,		0,		E0_MSLW,
131  	E0_MSRW,	E0_MSTM,	0,		0,
132  	0,		0,		0,		0,
133  	0,		0,		0,		0,
134  	0,		0,		0,		0,
135  	0,		0,		0,		E0_MACRO,
136  	0,		0,		0,		0,
137  	0,		0,		0,		0,
138  	0,		0,		0,		0,
139  	0,		0,		0,		0
140  };
141 
142 
143 /* FUNCTIONS ******************************************************************/
144 
145 USHORT LlbKbdLastScanCode;
146 
147 UCHAR
148 NTAPI
149 LlbKbdTranslateScanCode(IN USHORT ScanCode,
150                         IN PUCHAR KeyCode)
151 {
152     ULONG LastScanCode;
153 
154     /* Check for extended scan codes */
155     if ((ScanCode == 0xE0) || (ScanCode == 0xE1))
156  	{
157  	    /* We'll get these on the next scan */
158  		LlbKbdLastScanCode = ScanCode;
159  		return 0;
160  	}
161 
162     /* Check for bogus scan codes */
163  	if ((ScanCode == 0x00) || (ScanCode == 0xFF))
164  	{
165  	    /* Reset */
166  		LlbKbdLastScanCode = 0;
167  		return 0;
168  	}
169 
170  	/* Only act on the break, not the make */
171     if (ScanCode > 0x80) return 0;
172 
173     /* Keep only simple scan codes */
174  	ScanCode &= 0x7F;
175 
176     /* Check if this was part of an extended sequence */
177  	if (LlbKbdLastScanCode)
178  	{
179  	    /* Save the last scan code and clear it, since we've consumed it now */
180         LastScanCode = LlbKbdLastScanCode;
181         LlbKbdLastScanCode = 0;
182  		switch (LastScanCode)
183  		{
184  		    /* E0 extended codes */
185  		    case 0xE0:
186 
187  		        /* Skip bogus codes */
188  			    if ((ScanCode == 0x2A) || (ScanCode == 0x36)) return 0;
189 
190  			    /* Lookup the code for it */
191                 if (!LlbHwExtendedScanCodeTable[ScanCode]) return 0;
192  				*KeyCode = LlbHwExtendedScanCodeTable[ScanCode];
193  			    break;
194 
195             /* E1 extended codes */
196  		    case 0xE1:
197 
198  		        /* Only recognize one (the SYSREQ/PAUSE sequence) */
199                 if (ScanCode != 0x1D) return 0;
200  			    LlbKbdLastScanCode = 0x100;
201  			    break;
202 
203             /* PAUSE sequence */
204  		    case 0x100:
205 
206  		        /* Make sure it's the one */
207                 if (ScanCode != 0x45) return 0;
208  				*KeyCode = E1_PAUSE;
209  			    break;
210  		    }
211  	}
212  	else
213  	{
214  	    /* Otherwise, the scancode is the key code */
215         LlbKbdLastScanCode = 0;
216  		*KeyCode = ScanCode;
217 	}
218 
219 	/* Translation success */
220  	return 1;
221  }
222 
223 CHAR
224 NTAPI
225 LlbKeyboardGetChar(VOID)
226 {
227     UCHAR ScanCode, KeyCode;
228 
229     do
230     {
231         /* Read the scan code and convert it to a virtual key code */
232         ScanCode = LlbHwKbdRead();
233     } while (!LlbKbdTranslateScanCode(ScanCode, &KeyCode));
234 
235     /* Is this ASCII? */
236     if (KeyCode > 96) return ScanCode;
237 
238     /* Return the ASCII character */
239     return LlbHwScanCodeToAsciiTable[KeyCode][0];
240 }
241 
242 /* EOF */
243