[PATCH RT] fix migrating softirq [cause of network hang]
From: Steven Rostedt
Date: Wed Jun 13 2007 - 08:48:09 EST
Softirqs are bound to a single CPU. That is to say, that once a softirq
function starts to run, it will stay on the CPU that it is running on
while it's running.
In RT, softirqs are threads, and we have a softirq thread per cpu. Each
softirq thread is bound to a single CPU that it represents.
In order to speed things up and lower context switches in RT, if a
softirq thread is of the same priority as an interrupt thread, then when
the interrupt thread is about to exit, it tests to see if any softirq
threads need to be run on that cpu. Instead of running the softirq
thread, it simply performs the functions for the softirq within the
interrupt thread.
The problem is, nothing prevents the interrupt thread from migrating.
So while the interrupt thread is running the softirq function, it may
migrate to another CPU in the middle of that function. This means that
any CPU data that the softirq is touching can be corrupted.
I was experiencing a network hang that sometimes would come back, and
sometimes not. Using my logdev debugger, I started to debug this
problem. I came across this at the moment of the hang:
[ 389.131279] cpu:0 (IRQ-11:427) tcp_rcv_established:4056 rcv_nxt=-1665585797
[ 389.131615] cpu:1 192.168.23.72:22 <== 192.168.23.60:41352 ack:2629381499 seq:1773074099 (----A-) len:0 win:790 end_seq:1773074099
[ 389.131626] cpu:1 (IRQ-11:427) ip_finish_output2:187 dst->hh=ffff81003b213080
[ 389.131635] cpu:1 (IRQ-11:427) ip_finish_output2:189 hh_output=ffffffff80429009
Here we see IRQ-11 in the process of finishing up the softirq-net-tx
function. In the middle of it, we receive a packet, and that must have
pushed the interrupt thread over to CPU 1, and it finished up the
softirq there.
This patch temporarily binds the hardirq thread on the CPU that it runs
the softirqs on. With this patch I have not seen my network hang. I ran
it over night, doing compiles and such, and it seems fine. I would be
able to cause the hang with various loads within a minute, now I can't
cause it after several minutes.
I'm assuming that this fix may fix other bugs too.
Signed-off-by: Steven Rostedt <rostedt@xxxxxxxxxxx>
Index: linux-2.6.21.4-rt12/kernel/softirq.c
===================================================================
--- linux-2.6.21.4-rt12.orig/kernel/softirq.c
+++ linux-2.6.21.4-rt12/kernel/softirq.c
@@ -426,8 +426,25 @@ void do_softirq_from_hardirq(void)
current->flags &= ~PF_HARDIRQ;
current->flags |= PF_SOFTIRQ;
+#ifdef CONFIG_PREEMPT_HARDIRQS
+ {
+ cpumask_t save_cpus_allowed, cpus_allowed;
+
+ /* Don't let ourselves migrate */
+ save_cpus_allowed = current->cpus_allowed;
+ cpu_set(smp_processor_id(), cpus_allowed);
+ set_cpus_allowed(current, cpus_allowed);
+#endif
+
___do_softirq(1);
+#ifdef CONFIG_PREEMPT_HARDIRQS
+ /* Put back our original mask. */
+ WARN_ON(!cpus_equal(current->cpus_allowed, cpus_allowed));
+ set_cpus_allowed(current, save_cpus_allowed);
+ }
+#endif
+
#ifndef CONFIG_PREEMPT_SOFTIRQS
trace_softirq_exit();
#endif
-
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/