/* Track exiting tasks in order to allow them to be waited for. */
 DEFINE_STATIC_SRCU(tasks_rcu_exit_srcu);
 
+#ifdef CONFIG_TASKS_RCU
+/* Report delay in synchronize_srcu() completion in rcu_tasks_postscan(). */
+static void tasks_rcu_exit_srcu_stall(struct timer_list *unused);
+static DEFINE_TIMER(tasks_rcu_exit_srcu_stall_timer, tasks_rcu_exit_srcu_stall);
+#endif
+
 /* Avoid IPIing CPUs early in the grace period. */
 #define RCU_TASK_IPI_DELAY (IS_ENABLED(CONFIG_TASKS_TRACE_RCU_READ_MB) ? HZ / 2 : 0)
 static int rcu_task_ipi_delay __read_mostly = RCU_TASK_IPI_DELAY;
 /* Processing between scanning taskslist and draining the holdout list. */
 static void rcu_tasks_postscan(struct list_head *hop)
 {
+       int rtsi = READ_ONCE(rcu_task_stall_info);
+
+       if (!IS_ENABLED(CONFIG_TINY_RCU)) {
+               tasks_rcu_exit_srcu_stall_timer.expires = jiffies + rtsi;
+               add_timer(&tasks_rcu_exit_srcu_stall_timer);
+       }
+
        /*
         * Exiting tasks may escape the tasklist scan. Those are vulnerable
         * until their final schedule() with TASK_DEAD state. To cope with
         * call to synchronize_rcu().
         */
        synchronize_srcu(&tasks_rcu_exit_srcu);
+
+       if (!IS_ENABLED(CONFIG_TINY_RCU))
+               del_timer_sync(&tasks_rcu_exit_srcu_stall_timer);
 }
 
 /* See if tasks are still holding out, complain if so. */
 void call_rcu_tasks(struct rcu_head *rhp, rcu_callback_t func);
 DEFINE_RCU_TASKS(rcu_tasks, rcu_tasks_wait_gp, call_rcu_tasks, "RCU Tasks");
 
+static void tasks_rcu_exit_srcu_stall(struct timer_list *unused)
+{
+#ifndef CONFIG_TINY_RCU
+       int rtsi;
+
+       rtsi = READ_ONCE(rcu_task_stall_info);
+       pr_info("%s: %s grace period number %lu (since boot) gp_state: %s is %lu jiffies old.\n",
+               __func__, rcu_tasks.kname, rcu_tasks.tasks_gp_seq,
+               tasks_gp_state_getname(&rcu_tasks), jiffies - rcu_tasks.gp_jiffies);
+       pr_info("Please check any exiting tasks stuck between calls to exit_tasks_rcu_start() and exit_tasks_rcu_finish()\n");
+       tasks_rcu_exit_srcu_stall_timer.expires = jiffies + rtsi;
+       add_timer(&tasks_rcu_exit_srcu_stall_timer);
+#endif // #ifndef CONFIG_TINY_RCU
+}
+
 /**
  * call_rcu_tasks() - Queue an RCU for invocation task-based grace period
  * @rhp: structure to be used for queueing the RCU updates.