提交 a216a4b6 编写于 作者: D Dmitry Torokhov

Input: i8042 - fix shutdown issue on some boxes with active MUXes

MUX error handling has a workaround for KBCs that get confused which
port data came from and signal MUXERR while data is actually good.
Unfortunately this workaround hurts with KBCs that signal timeouts
as 0xfc (spec says that only 0xfd, 0xfe and 0xff are alowed with
MUXERR) since it causes endless attempts to rescan i8042 serio
ports. The solution is to treat 0xfc as timeout (0xfe).
Signed-off-by: NDmitry Torokhov <dtor@mail.ru>
上级 6c595fb1
...@@ -322,23 +322,27 @@ static irqreturn_t i8042_interrupt(int irq, void *dev_id) ...@@ -322,23 +322,27 @@ static irqreturn_t i8042_interrupt(int irq, void *dev_id)
dfl = 0; dfl = 0;
if (str & I8042_STR_MUXERR) { if (str & I8042_STR_MUXERR) {
dbg("MUX error, status is %02x, data is %02x", str, data); dbg("MUX error, status is %02x, data is %02x", str, data);
switch (data) {
default:
/* /*
* When MUXERR condition is signalled the data register can only contain * When MUXERR condition is signalled the data register can only contain
* 0xfd, 0xfe or 0xff if implementation follows the spec. Unfortunately * 0xfd, 0xfe or 0xff if implementation follows the spec. Unfortunately
* it is not always the case. Some KBC just get confused which port the * it is not always the case. Some KBCs also report 0xfc when there is
* data came from and signal error leaving the data intact. They _do not_ * nothing connected to the port while others sometimes get confused which
* revert to legacy mode (actually I've never seen KBC reverting to legacy * port the data came from and signal error leaving the data intact. They
* mode yet, when we see one we'll add proper handling). * _do not_ revert to legacy mode (actually I've never seen KBC reverting
* Anyway, we will assume that the data came from the same serio last byte * to legacy mode yet, when we see one we'll add proper handling).
* Anyway, we process 0xfc, 0xfd, 0xfe and 0xff as timeouts, and for the
* rest assume that the data came from the same serio last byte
* was transmitted (if transmission happened not too long ago). * was transmitted (if transmission happened not too long ago).
*/ */
switch (data) {
default:
if (time_before(jiffies, last_transmit + HZ/10)) { if (time_before(jiffies, last_transmit + HZ/10)) {
str = last_str; str = last_str;
break; break;
} }
/* fall through - report timeout */ /* fall through - report timeout */
case 0xfc:
case 0xfd: case 0xfd:
case 0xfe: dfl = SERIO_TIMEOUT; data = 0xfe; break; case 0xfe: dfl = SERIO_TIMEOUT; data = 0xfe; break;
case 0xff: dfl = SERIO_PARITY; data = 0xfe; break; case 0xff: dfl = SERIO_PARITY; data = 0xfe; break;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册