2010-04-03 03:17:17 +04:00
/*
2010-06-30 03:49:16 +04:00
* Read - Copy Update mechanism for mutual exclusion , the Bloatwatch edition
2010-04-03 03:17:17 +04:00
* Internal non - public definitions that provide either classic
2010-06-30 03:49:16 +04:00
* or preemptible semantics .
2010-04-03 03:17:17 +04:00
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
2013-12-03 22:02:52 +04:00
* along with this program ; if not , you can access it online at
* http : //www.gnu.org/licenses/gpl-2.0.html.
2010-04-03 03:17:17 +04:00
*
2010-06-30 03:49:16 +04:00
* Copyright ( c ) 2010 Linaro
2010-04-03 03:17:17 +04:00
*
* Author : Paul E . McKenney < paulmck @ linux . vnet . ibm . com >
*/
2010-09-10 00:40:39 +04:00
# include <linux/kthread.h>
2016-01-08 03:05:19 +03:00
# include <linux/init.h>
2010-10-01 08:26:52 +04:00
# include <linux/debugfs.h>
# include <linux/seq_file.h>
2010-09-28 04:25:23 +04:00
/* Global control variables for rcupdate callback mechanism. */
struct rcu_ctrlblk {
struct rcu_head * rcucblist ; /* List of pending callbacks (CBs). */
struct rcu_head * * donetail ; /* ->next pointer of last "done" CB. */
struct rcu_head * * curtail ; /* ->next pointer of last CB. */
2010-10-01 08:26:52 +04:00
RCU_TRACE ( long qlen ) ; /* Number of pending CBs. */
2012-10-19 23:49:17 +04:00
RCU_TRACE ( unsigned long gp_start ) ; /* Start time for stalls. */
RCU_TRACE ( unsigned long ticks_this_gp ) ; /* Statistic for stalls. */
RCU_TRACE ( unsigned long jiffies_stall ) ; /* Jiffies at next stall. */
2013-07-13 00:50:28 +04:00
RCU_TRACE ( const char * name ) ; /* Name of RCU type. */
2010-09-28 04:25:23 +04:00
} ;
/* Definition for rcupdate control block. */
static struct rcu_ctrlblk rcu_sched_ctrlblk = {
. donetail = & rcu_sched_ctrlblk . rcucblist ,
. curtail = & rcu_sched_ctrlblk . rcucblist ,
2011-06-21 11:13:44 +04:00
RCU_TRACE ( . name = " rcu_sched " )
2010-09-28 04:25:23 +04:00
} ;
static struct rcu_ctrlblk rcu_bh_ctrlblk = {
. donetail = & rcu_bh_ctrlblk . rcucblist ,
. curtail = & rcu_bh_ctrlblk . rcucblist ,
2011-06-21 11:13:44 +04:00
RCU_TRACE ( . name = " rcu_bh " )
2010-09-28 04:25:23 +04:00
} ;
# ifdef CONFIG_DEBUG_LOCK_ALLOC
2013-03-27 21:43:02 +04:00
# include <linux/kernel_stat.h>
2010-09-28 04:25:23 +04:00
int rcu_scheduler_active __read_mostly ;
EXPORT_SYMBOL_GPL ( rcu_scheduler_active ) ;
2010-04-03 03:17:17 +04:00
/*
* During boot , we forgive RCU lockdep issues . After this function is
* invoked , we start taking RCU lockdep issues seriously .
*/
2010-09-10 00:40:39 +04:00
void __init rcu_scheduler_starting ( void )
2010-04-03 03:17:17 +04:00
{
WARN_ON ( nr_context_switches ( ) > 0 ) ;
rcu_scheduler_active = 1 ;
}
# endif /* #ifdef CONFIG_DEBUG_LOCK_ALLOC */
2010-09-28 04:25:23 +04:00
2010-10-01 08:26:52 +04:00
# ifdef CONFIG_RCU_TRACE
static void rcu_trace_sub_qlen ( struct rcu_ctrlblk * rcp , int n )
{
unsigned long flags ;
2012-08-21 23:14:19 +04:00
local_irq_save ( flags ) ;
2010-10-01 08:26:52 +04:00
rcp - > qlen - = n ;
2012-08-21 23:14:19 +04:00
local_irq_restore ( flags ) ;
2010-10-01 08:26:52 +04:00
}
/*
* Dump statistics for TINY_RCU , such as they are .
*/
static int show_tiny_stats ( struct seq_file * m , void * unused )
{
seq_printf ( m , " rcu_sched: qlen: %ld \n " , rcu_sched_ctrlblk . qlen ) ;
seq_printf ( m , " rcu_bh: qlen: %ld \n " , rcu_bh_ctrlblk . qlen ) ;
return 0 ;
}
static int show_tiny_stats_open ( struct inode * inode , struct file * file )
{
return single_open ( file , show_tiny_stats , NULL ) ;
}
static const struct file_operations show_tiny_stats_fops = {
. owner = THIS_MODULE ,
. open = show_tiny_stats_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
static struct dentry * rcudir ;
static int __init rcutiny_trace_init ( void )
{
struct dentry * retval ;
rcudir = debugfs_create_dir ( " rcu " , NULL ) ;
if ( ! rcudir )
goto free_out ;
retval = debugfs_create_file ( " rcudata " , 0444 , rcudir ,
NULL , & show_tiny_stats_fops ) ;
if ( ! retval )
goto free_out ;
return 0 ;
free_out :
debugfs_remove_recursive ( rcudir ) ;
return 1 ;
}
2016-01-08 03:05:19 +03:00
device_initcall ( rcutiny_trace_init ) ;
2010-10-01 08:26:52 +04:00
2013-03-27 21:43:02 +04:00
static void check_cpu_stall ( struct rcu_ctrlblk * rcp )
{
unsigned long j ;
unsigned long js ;
if ( rcu_cpu_stall_suppress )
return ;
rcp - > ticks_this_gp + + ;
j = jiffies ;
2015-03-04 01:57:58 +03:00
js = READ_ONCE ( rcp - > jiffies_stall ) ;
2014-12-22 22:10:12 +03:00
if ( rcp - > rcucblist & & ULONG_CMP_GE ( j , js ) ) {
2013-03-27 21:43:02 +04:00
pr_err ( " INFO: %s stall on CPU (%lu ticks this GP) idle=%llx (t=%lu jiffies q=%ld) \n " ,
2014-12-09 12:53:34 +03:00
rcp - > name , rcp - > ticks_this_gp , DYNTICK_TASK_EXIT_IDLE ,
2013-03-27 21:43:02 +04:00
jiffies - rcp - > gp_start , rcp - > qlen ) ;
dump_stack ( ) ;
2015-03-04 01:57:58 +03:00
WRITE_ONCE ( rcp - > jiffies_stall ,
jiffies + 3 * rcu_jiffies_till_stall_check ( ) + 3 ) ;
2014-12-22 22:10:12 +03:00
} else if ( ULONG_CMP_GE ( j , js ) ) {
2015-03-04 01:57:58 +03:00
WRITE_ONCE ( rcp - > jiffies_stall ,
jiffies + rcu_jiffies_till_stall_check ( ) ) ;
2014-12-22 22:10:12 +03:00
}
2013-03-27 21:43:02 +04:00
}
static void reset_cpu_stall_ticks ( struct rcu_ctrlblk * rcp )
{
rcp - > ticks_this_gp = 0 ;
rcp - > gp_start = jiffies ;
2015-03-04 01:57:58 +03:00
WRITE_ONCE ( rcp - > jiffies_stall ,
jiffies + rcu_jiffies_till_stall_check ( ) ) ;
2013-03-27 21:43:02 +04:00
}
static void check_cpu_stalls ( void )
{
RCU_TRACE ( check_cpu_stall ( & rcu_bh_ctrlblk ) ) ;
RCU_TRACE ( check_cpu_stall ( & rcu_sched_ctrlblk ) ) ;
}
2013-04-16 18:49:22 +04:00
# endif /* #ifdef CONFIG_RCU_TRACE */