Page 1 of 1
How do 0xE0 keyboard scancodes (PS/2) work?
Posted: Mon Jan 01, 2018 3:18 pm
by thumble
I'm working on PS/2 keyboard support on my kernel. There are scancodes with "E0, xx", and I'm not too sure how these are supposed to work.
I
think this means that I read the scancode, 0xE0, and if it's 0xE0 I read another byte, which is the scancode. This doesn't really work.
Here is my keyboard IRQ handler right now, stripped down for debugging purposes:
Code: Select all
void kb_irq_handler(struct regs *r) {
byte scancode;
scancode = inb(0x60);
}
When pressing any keys with an E0 scancode it immediately prints the second part (i.e. 1D for RCTRL) which is correct, but I'm not getting an E0. Why might this be? Am I misunderstanding something?
edit: It works fine for keys without an E0.
Re: How do 0xE0 keyboard scancodes (PS/2) work?
Posted: Mon Jan 01, 2018 5:37 pm
by CorruptedByCPU
Just remember of 0xE0 appearance and leave interrupt.
In next interrupt check if 0xE0 occurred. If yes, then it was special key pressed.
For example, if i get scancode of 0x2A and 0xE0 occured before, then this is "Left Shift" key, after that just forget about 0xE0.
Re: How do 0xE0 keyboard scancodes (PS/2) work?
Posted: Mon Jan 01, 2018 9:25 pm
by Brendan
Hi,
thumble wrote:I'm working on PS/2 keyboard support on my kernel. There are scancodes with "E0, xx", and I'm not too sure how these are supposed to work.
I think this means that I read the scancode, 0xE0, and if it's 0xE0 I read another byte, which is the scancode. This doesn't really work.
Typically you have a state machine, where you have a default state (nothing received yet) and an "0xE0 received" state (and a bunch of other states for other sequences); and when you receive a byte you look at both the state and the byte received to determine what to do and which state to move into next.
The fastest and most flexible way to handle this might be with a table of function pointers; like:
Code: Select all
int state = 0;
int newByte = 0;
void kb_irq_handler(void) {
newByte = inb(0x60);
state = (*functionTable[state << 8 | newByte])();
}
In this case; during initialisation you'd configure the table of function pointers; possibly starting with "pre-configuration" where you add table entries for special bytes (0x00 = keyboard error, 0xAA = self test passed, 0xEE = echo, ...) and set everything else to a generic "invalid sequence" function, and then you'd parse some kind of keyboard layout file to determine how the rest of the table of function pointers should be setup; so that the driver is able to handle radically different keyboards just by providing different keyboard layout files (potentially including being able to support different scan code sets).
thumble wrote:When pressing any keys with an E0 scancode it immediately prints the second part (i.e. 1D for RCTRL) which is correct, but I'm not getting an E0. Why might this be? Am I misunderstanding something?
edit: It works fine for keys without an E0.
How do you know that you're not getting an 0xE0? For example, if you're doing some kind of "printChar(ASCII);" then maybe the problem is that there's no way to convert the 0xE0 byte into valid ASCII, if you're doing some kind of "printHex(byte)" then maybe your "printHex()" function is buggy, ...
Also note that some emulators struggle to emulate keyboards 100% accurately. For example, the host OS might tell the emulator that a control key was pressed (but not which one), so the emulator might tell the guest OS that left control was pressed (0x1D) even though the user actually pressed the right control key (0xE0, 0x1D).
Cheers,
Brendan
Re: How do 0xE0 keyboard scancodes (PS/2) work?
Posted: Wed Jan 03, 2018 6:12 pm
by Sik
Yeah, as mentioned the next byte will come in the next interrupt, not immediately. You need to remember the previous bytes somehow.
This seems to be one of the most common mistakes when dealing with interrupt-based hardware, I think that a lot of tutorials and documentation just says you get two bytes without being explicit that they mean it comes in separate interrupts (when teaching something where a similar situation hadn't been brought up before). It gets even worse when sending multi-byte commands where you're expected to get an ACK after every byte, the ACK will come in interrupts (likely from an entirely unrelated context than the code sending the command) so you actually need to wait for them to come in which can take a long while (i.e. busylooping is not a good option!).
Re: How do 0xE0 keyboard scancodes (PS/2) work?
Posted: Wed Jan 03, 2018 11:35 pm
by thumble
Ah! That makes sense now. Thank you so much.
edit: It works!
This may not be the best solution, but I store the last scancode and check if it is 0xE0 in my conversion function.
Re: How do 0xE0 keyboard scancodes (PS/2) work?
Posted: Thu Jan 04, 2018 5:33 am
by Octacone
This is actually a very complex topic. It took me a while to realize how to do a state machine.
The main problem is that people don't know that one interrupt equals one byte of a scancode sequence.
0xE0 is not the only thing that you need to implement. There are also multi-byte (up to 6 if I remember) scancodes per key press that you need to handle.
State machine would be the best option.
Re: How do 0xE0 keyboard scancodes (PS/2) work?
Posted: Thu Jan 04, 2018 2:28 pm
by Sik
Octacone wrote:There are also multi-byte (up to 6 if I remember) scancodes per key press that you need to handle.
Yeah, Pause key does it (sends the equivalent of Shift+* key down and key up, if I recall correctly). But that one is easier since it's the only scancode that starts with 0xE1 (presumably signaling that it's two extended bytes instead of one - again, remember it sends two state changes in a row), if you get it you know the next five scancodes are for that key. I imagine it was intended to trigger a shortcut used by very early PCs.
Every other key is limited to just one or two scancodes if you're using set 1 (which seems to be the case here, judging from 0xE0 being an extension prefix). A bigger issue is that keyboards may have trouble using set 1 (instead of set 2), so consider adding support for set 2 when feasible.