From: Thomas Gleixner <t...@linutronix.de>

We can't rely on ksoftirqd anymore and we need to check the tasks
which run a particular softirq and if such a task is pi blocked ignore
the other pending bits of that task as well.

Signed-off-by: Thomas Gleixner <t...@linutronix.de>
Signed-off-by: Sebastian Andrzej Siewior <bige...@linutronix.de>
---
 kernel/softirq.c |   68 +++++++++++++++++++++++++++++++++++++++++-------------
 1 file changed, 52 insertions(+), 16 deletions(-)

diff --git a/kernel/softirq.c b/kernel/softirq.c
index 3f7b3fb..66999ad 100644
--- a/kernel/softirq.c
+++ b/kernel/softirq.c
@@ -65,45 +65,75 @@ char *softirq_to_name[NR_SOFTIRQS] = {
 
 #ifdef CONFIG_NO_HZ
 # ifdef CONFIG_PREEMPT_RT_FULL
+
+struct softirq_runner {
+       struct task_struct *runner[NR_SOFTIRQS];
+};
+
+static DEFINE_PER_CPU(struct softirq_runner, softirq_runners);
+
+static inline void softirq_set_runner(unsigned int sirq)
+{
+       struct softirq_runner *sr = &__get_cpu_var(softirq_runners);
+
+       sr->runner[sirq] = current;
+}
+
+static inline void softirq_clr_runner(unsigned int sirq)
+{
+       struct softirq_runner *sr = &__get_cpu_var(softirq_runners);
+
+       sr->runner[sirq] = NULL;
+}
+
 /*
- * On preempt-rt a softirq might be blocked on a lock. There might be
- * no other runnable task on this CPU because the lock owner runs on
- * some other CPU. So we have to go into idle with the pending bit
- * set. Therefor we need to check this otherwise we warn about false
- * positives which confuses users and defeats the whole purpose of
- * this test.
+ * On preempt-rt a softirq running context might be blocked on a
+ * lock. There might be no other runnable task on this CPU because the
+ * lock owner runs on some other CPU. So we have to go into idle with
+ * the pending bit set. Therefor we need to check this otherwise we
+ * warn about false positives which confuses users and defeats the
+ * whole purpose of this test.
  *
  * This code is called with interrupts disabled.
  */
 void softirq_check_pending_idle(void)
 {
        static int rate_limit;
-       u32 warnpending = 0, pending = local_softirq_pending();
+       struct softirq_runner *sr = &__get_cpu_var(softirq_runners);
+       u32 warnpending, pending = local_softirq_pending();
 
        if (rate_limit >= 10)
                return;
 
-       if (pending) {
+       warnpending = pending;
+
+       while (pending) {
                struct task_struct *tsk;
+               int i = __ffs(pending);
 
-               tsk = __get_cpu_var(ksoftirqd);
+               pending &= ~(1 << i);
+
+               tsk = sr->runner[i];
                /*
                 * The wakeup code in rtmutex.c wakes up the task
                 * _before_ it sets pi_blocked_on to NULL under
                 * tsk->pi_lock. So we need to check for both: state
                 * and pi_blocked_on.
                 */
-               raw_spin_lock(&tsk->pi_lock);
-
-               if (!tsk->pi_blocked_on && !(tsk->state == TASK_RUNNING))
-                       warnpending = 1;
-
-               raw_spin_unlock(&tsk->pi_lock);
+               if (tsk) {
+                       raw_spin_lock(&tsk->pi_lock);
+                       if (tsk->pi_blocked_on || tsk->state == TASK_RUNNING) {
+                               /* Clear all bits pending in that task */
+                               warnpending &= ~(tsk->softirqs_raised);
+                               warnpending &= ~(1 << i);
+                       }
+                       raw_spin_unlock(&tsk->pi_lock);
+               }
        }
 
        if (warnpending) {
                printk(KERN_ERR "NOHZ: local_softirq_pending %02x\n",
-                      pending);
+                      warnpending);
                rate_limit++;
        }
 }
@@ -122,6 +152,10 @@ void softirq_check_pending_idle(void)
        }
 }
 # endif
+
+#else /* !NO_HZ */
+static inline void softirq_set_runner(unsigned int sirq) { }
+static inline void softirq_clr_runner(unsigned int sirq) { }
 #endif
 
 /*
@@ -469,6 +503,7 @@ static void do_current_softirqs(int need_rcu_bh_qs)
                 */
                lock_softirq(i);
                local_irq_disable();
+               softirq_set_runner(i);
                /*
                 * Check with the local_softirq_pending() bits,
                 * whether we need to process this still or if someone
@@ -479,6 +514,7 @@ static void do_current_softirqs(int need_rcu_bh_qs)
                        set_softirq_pending(pending & ~mask);
                        do_single_softirq(i, need_rcu_bh_qs);
                }
+               softirq_clr_runner(i);
                unlock_softirq(i);
                WARN_ON(current->softirq_nestcnt != 1);
        }
-- 
1.7.10.4

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

Reply via email to