* [PATCH] sound: fix hang in mpu401_uart.c
@ 2006-04-16 3:12 Jon Masters
2006-04-18 13:18 ` Takashi Iwai
0 siblings, 1 reply; 4+ messages in thread
From: Jon Masters @ 2006-04-16 3:12 UTC (permalink / raw)
To: akpm; +Cc: linux-kernel
From: Jon Masters <jcm@jonmasters.org>
This fixes a hang in mpu401_uart.c that can occur when the mpu401
interface is non-existent or otherwise doesn't respond to commands but
we issue IO anyway. snd_mpu401_uart_cmd now returns an error code that is
passed up the stack so that an open() will fail immediately in such cases.
Eventually discovered after wine/cxoffice would constantly cause hard
lockups on my desktop immediately after loading (emulating Windows too
well). Turned out that I'd recently moved my sound cards around and
using /dev/sequencer now talks to a sound card with a broken MPU.
Signed-off-by: Jon Masters <jcm@jonmasters.org>
diff -urN linux-2.6.16.2_orig/sound/drivers/mpu401/mpu401_uart.c linux-2.6.16.2_dev/sound/drivers/mpu401/mpu401_uart.c
--- linux-2.6.16.2_orig/sound/drivers/mpu401/mpu401_uart.c 2006-04-07 17:56:47.000000000 +0100
+++ linux-2.6.16.2_dev/sound/drivers/mpu401/mpu401_uart.c 2006-04-16 03:33:15.000000000 +0100
@@ -183,7 +183,7 @@
*/
-static void snd_mpu401_uart_cmd(struct snd_mpu401 * mpu, unsigned char cmd, int ack)
+static int snd_mpu401_uart_cmd(struct snd_mpu401 * mpu, unsigned char cmd, int ack)
{
unsigned long flags;
int timeout, ok;
@@ -218,9 +218,12 @@
ok = 1;
}
spin_unlock_irqrestore(&mpu->input_lock, flags);
- if (! ok)
+ if (ok)
+ return 0;
+ else {
snd_printk("cmd: 0x%x failed at 0x%lx (status = 0x%x, data = 0x%x)\n", cmd, mpu->port, mpu->read(mpu, MPU401C(mpu)), mpu->read(mpu, MPU401D(mpu)));
- // snd_printk("cmd: 0x%x at 0x%lx (status = 0x%x, data = 0x%x)\n", cmd, mpu->port, mpu->read(mpu, MPU401C(mpu)), mpu->read(mpu, MPU401D(mpu)));
+ return 1;
+ }
}
/*
@@ -235,8 +238,12 @@
if (mpu->open_input && (err = mpu->open_input(mpu)) < 0)
return err;
if (! test_bit(MPU401_MODE_BIT_OUTPUT, &mpu->mode)) {
- snd_mpu401_uart_cmd(mpu, MPU401_RESET, 1);
- snd_mpu401_uart_cmd(mpu, MPU401_ENTER_UART, 1);
+ if ((err = snd_mpu401_uart_cmd(mpu, MPU401_RESET, 1))) {
+ return -EFAULT;
+ }
+ if ((err = snd_mpu401_uart_cmd(mpu, MPU401_ENTER_UART, 1))) {
+ return -EFAULT;
+ }
}
mpu->substream_input = substream;
set_bit(MPU401_MODE_BIT_INPUT, &mpu->mode);
@@ -252,8 +259,10 @@
if (mpu->open_output && (err = mpu->open_output(mpu)) < 0)
return err;
if (! test_bit(MPU401_MODE_BIT_INPUT, &mpu->mode)) {
- snd_mpu401_uart_cmd(mpu, MPU401_RESET, 1);
- snd_mpu401_uart_cmd(mpu, MPU401_ENTER_UART, 1);
+ if ((err = snd_mpu401_uart_cmd(mpu, MPU401_RESET, 1)))
+ return -EFAULT;
+ if ((err = snd_mpu401_uart_cmd(mpu, MPU401_ENTER_UART, 1)))
+ return -EFAULT;
}
mpu->substream_output = substream;
set_bit(MPU401_MODE_BIT_OUTPUT, &mpu->mode);
@@ -263,28 +272,34 @@
static int snd_mpu401_uart_input_close(struct snd_rawmidi_substream *substream)
{
struct snd_mpu401 *mpu;
-
+ int err = 0;
+
mpu = substream->rmidi->private_data;
clear_bit(MPU401_MODE_BIT_INPUT, &mpu->mode);
mpu->substream_input = NULL;
if (! test_bit(MPU401_MODE_BIT_OUTPUT, &mpu->mode))
- snd_mpu401_uart_cmd(mpu, MPU401_RESET, 0);
+ err = snd_mpu401_uart_cmd(mpu, MPU401_RESET, 0);
if (mpu->close_input)
mpu->close_input(mpu);
+ if (err)
+ return -EFAULT;
return 0;
}
static int snd_mpu401_uart_output_close(struct snd_rawmidi_substream *substream)
{
struct snd_mpu401 *mpu;
+ int err = 0;
mpu = substream->rmidi->private_data;
clear_bit(MPU401_MODE_BIT_OUTPUT, &mpu->mode);
mpu->substream_output = NULL;
if (! test_bit(MPU401_MODE_BIT_INPUT, &mpu->mode))
- snd_mpu401_uart_cmd(mpu, MPU401_RESET, 0);
+ err = snd_mpu401_uart_cmd(mpu, MPU401_RESET, 0);
if (mpu->close_output)
mpu->close_output(mpu);
+ if (err)
+ return -EFAULT;
return 0;
}
@@ -316,6 +331,7 @@
snd_mpu401_uart_remove_timer(mpu, 1);
clear_bit(MPU401_MODE_BIT_INPUT_TRIGGER, &mpu->mode);
}
+
}
/*
^ permalink raw reply [flat|nested] 4+ messages in thread
* Re: [PATCH] sound: fix hang in mpu401_uart.c
2006-04-16 3:12 [PATCH] sound: fix hang in mpu401_uart.c Jon Masters
@ 2006-04-18 13:18 ` Takashi Iwai
2006-04-18 23:45 ` Jon Masters
0 siblings, 1 reply; 4+ messages in thread
From: Takashi Iwai @ 2006-04-18 13:18 UTC (permalink / raw)
To: Jon Masters; +Cc: akpm, linux-kernel
At Sun, 16 Apr 2006 04:12:36 +0100,
Jon Masters wrote:
>
> From: Jon Masters <jcm@jonmasters.org>
>
> This fixes a hang in mpu401_uart.c that can occur when the mpu401
> interface is non-existent or otherwise doesn't respond to commands but
> we issue IO anyway. snd_mpu401_uart_cmd now returns an error code that is
> passed up the stack so that an open() will fail immediately in such cases.
>
> Eventually discovered after wine/cxoffice would constantly cause hard
> lockups on my desktop immediately after loading (emulating Windows too
> well). Turned out that I'd recently moved my sound cards around and
> using /dev/sequencer now talks to a sound card with a broken MPU.
>
> Signed-off-by: Jon Masters <jcm@jonmasters.org>
(snip)
> @@ -235,8 +238,12 @@
> if (mpu->open_input && (err = mpu->open_input(mpu)) < 0)
> return err;
> if (! test_bit(MPU401_MODE_BIT_OUTPUT, &mpu->mode)) {
> - snd_mpu401_uart_cmd(mpu, MPU401_RESET, 1);
> - snd_mpu401_uart_cmd(mpu, MPU401_ENTER_UART, 1);
> + if ((err = snd_mpu401_uart_cmd(mpu, MPU401_RESET, 1))) {
> + return -EFAULT;
IMO, -EFAULT isn't a good choice for this kind of error.
> @@ -252,8 +259,10 @@
> if (mpu->open_output && (err = mpu->open_output(mpu)) < 0)
> return err;
> if (! test_bit(MPU401_MODE_BIT_INPUT, &mpu->mode)) {
> - snd_mpu401_uart_cmd(mpu, MPU401_RESET, 1);
> - snd_mpu401_uart_cmd(mpu, MPU401_ENTER_UART, 1);
> + if ((err = snd_mpu401_uart_cmd(mpu, MPU401_RESET, 1)))
> + return -EFAULT;
> + if ((err = snd_mpu401_uart_cmd(mpu, MPU401_ENTER_UART, 1)))
> + return -EFAULT;
Missing close in the error path?
Takashi
^ permalink raw reply [flat|nested] 4+ messages in thread
* Re: [PATCH] sound: fix hang in mpu401_uart.c
2006-04-18 13:18 ` Takashi Iwai
@ 2006-04-18 23:45 ` Jon Masters
2006-04-19 9:33 ` Takashi Iwai
0 siblings, 1 reply; 4+ messages in thread
From: Jon Masters @ 2006-04-18 23:45 UTC (permalink / raw)
To: Takashi Iwai; +Cc: akpm, linux-kernel
On 4/18/06, Takashi Iwai <tiwai@suse.de> wrote:
> > + if ((err = snd_mpu401_uart_cmd(mpu, MPU401_RESET, 1))) {
> > + return -EFAULT;
>
> IMO, -EFAULT isn't a good choice for this kind of error.
What would you suggest?
> > if (mpu->open_output && (err = mpu->open_output(mpu)) < 0)
> > return err;
> > if (! test_bit(MPU401_MODE_BIT_INPUT, &mpu->mode)) {
> > - snd_mpu401_uart_cmd(mpu, MPU401_RESET, 1);
> > - snd_mpu401_uart_cmd(mpu, MPU401_ENTER_UART, 1);
> > + if ((err = snd_mpu401_uart_cmd(mpu, MPU401_RESET, 1)))
> > + return -EFAULT;
> > + if ((err = snd_mpu401_uart_cmd(mpu, MPU401_ENTER_UART, 1)))
> > + return -EFAULT;
> Missing close in the error path?
I'll respin the patch later, but first, what should I return on open
when the underlying hardware is not there?
Jon.
^ permalink raw reply [flat|nested] 4+ messages in thread
* Re: [PATCH] sound: fix hang in mpu401_uart.c
2006-04-18 23:45 ` Jon Masters
@ 2006-04-19 9:33 ` Takashi Iwai
0 siblings, 0 replies; 4+ messages in thread
From: Takashi Iwai @ 2006-04-19 9:33 UTC (permalink / raw)
To: Jon Masters; +Cc: akpm, linux-kernel
At Wed, 19 Apr 2006 00:45:14 +0100,
Jon Masters wrote:
>
> On 4/18/06, Takashi Iwai <tiwai@suse.de> wrote:
>
> > > + if ((err = snd_mpu401_uart_cmd(mpu, MPU401_RESET, 1))) {
> > > + return -EFAULT;
> >
> > IMO, -EFAULT isn't a good choice for this kind of error.
>
> What would you suggest?
I would use -EIO.
Takashi
^ permalink raw reply [flat|nested] 4+ messages in thread
end of thread, other threads:[~2006-04-19 9:34 UTC | newest]
Thread overview: 4+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2006-04-16 3:12 [PATCH] sound: fix hang in mpu401_uart.c Jon Masters
2006-04-18 13:18 ` Takashi Iwai
2006-04-18 23:45 ` Jon Masters
2006-04-19 9:33 ` Takashi Iwai
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox