[PATCH 2.6 IrDA] IrCOMM might_sleep Oops

From: Jean Tourrilhes
Date: Wed Nov 05 2003 - 14:41:52 EST


ir2609_ircomm_might_sleep-2.diff :
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
<Original patch from Martin Diehl>
o [CRITICA] Don't do copy_from_user() under spinlock
o [CRITICA] Always access self->skb under spinlock


diff -u -p linux/net/irda/ircomm/ircomm_tty.d4.c linux/net/irda/ircomm/ircomm_tty.c
--- linux/net/irda/ircomm/ircomm_tty.d4.c Tue Nov 4 10:31:28 2003
+++ linux/net/irda/ircomm/ircomm_tty.c Tue Nov 4 10:50:14 2003
@@ -663,9 +663,10 @@ static void ircomm_tty_do_softint(void *
* accepted for writing. This routine is mandatory.
*/
static int ircomm_tty_write(struct tty_struct *tty, int from_user,
- const unsigned char *buf, int count)
+ const unsigned char *ubuf, int count)
{
struct ircomm_tty_cb *self = (struct ircomm_tty_cb *) tty->driver_data;
+ unsigned char *kbuf; /* Buffer in kernel space */
unsigned long flags;
struct sk_buff *skb;
int tailroom = 0;
@@ -702,6 +703,24 @@ static int ircomm_tty_write(struct tty_s
#endif
}

+ if (count < 1)
+ return 0;
+
+ /* Additional copy to avoid copy_from_user() under spinlock.
+ * We tradeoff this extra copy to allow to pack more the
+ * IrCOMM frames. This is advantageous because the IrDA link
+ * is the bottleneck. */
+ if (from_user) {
+ kbuf = kmalloc(count, GFP_KERNEL);
+ if (kbuf == NULL)
+ return -ENOMEM;
+ if (copy_from_user(kbuf, ubuf, count))
+ return -EFAULT;
+ } else
+ /* The buffer is already in kernel space */
+ kbuf = (unsigned char *) ubuf;
+
+ /* Protect our manipulation of self->tx_skb and related */
spin_lock_irqsave(&self->spinlock, flags);

/* Fetch current transmit buffer */
@@ -761,19 +780,19 @@ static int ircomm_tty_write(struct tty_s
* change later on - Jean II */
self->tx_data_size = self->max_data_size;
}
-
+
/* Copy data */
- if (from_user)
- copy_from_user(skb_put(skb,size), buf+len, size);
- else
- memcpy(skb_put(skb,size), buf+len, size);
-
+ memcpy(skb_put(skb,size), kbuf + len, size);
+
count -= size;
len += size;
}

spin_unlock_irqrestore(&self->spinlock, flags);

+ if (from_user)
+ kfree(kbuf);
+
/*
* Schedule a new thread which will transmit the frame as soon
* as possible, but at a safe point in time. We do this so the
@@ -837,6 +856,7 @@ static void ircomm_tty_wait_until_sent(s
{
struct ircomm_tty_cb *self = (struct ircomm_tty_cb *) tty->driver_data;
unsigned long orig_jiffies, poll_time;
+ unsigned long flags;

IRDA_DEBUG(2, "%s()\n", __FUNCTION__ );

@@ -848,14 +868,18 @@ static void ircomm_tty_wait_until_sent(s
/* Set poll time to 200 ms */
poll_time = IRDA_MIN(timeout, MSECS_TO_JIFFIES(200));

+ spin_lock_irqsave(&self->spinlock, flags);
while (self->tx_skb && self->tx_skb->len) {
+ spin_unlock_irqrestore(&self->spinlock, flags);
current->state = TASK_INTERRUPTIBLE;
schedule_timeout(poll_time);
+ spin_lock_irqsave(&self->spinlock, flags);
if (signal_pending(current))
break;
if (timeout && time_after(jiffies, orig_jiffies + timeout))
break;
}
+ spin_unlock_irqrestore(&self->spinlock, flags);
current->state = TASK_RUNNING;
}

-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/