diff options
Diffstat (limited to 'kernel/hung_task.c')
| -rw-r--r-- | kernel/hung_task.c | 30 | 
1 files changed, 24 insertions, 6 deletions
diff --git a/kernel/hung_task.c b/kernel/hung_task.c index 3e97fb126e6..06db12434d7 100644 --- a/kernel/hung_task.c +++ b/kernel/hung_task.c @@ -16,11 +16,12 @@  #include <linux/export.h>  #include <linux/sysctl.h>  #include <linux/utsname.h> +#include <trace/events/sched.h>  /*   * The number of tasks checked:   */ -unsigned long __read_mostly sysctl_hung_task_check_count = PID_MAX_LIMIT; +int __read_mostly sysctl_hung_task_check_count = PID_MAX_LIMIT;  /*   * Limit number of tasks checked in a batch. @@ -36,7 +37,7 @@ unsigned long __read_mostly sysctl_hung_task_check_count = PID_MAX_LIMIT;   */  unsigned long __read_mostly sysctl_hung_task_timeout_secs = CONFIG_DEFAULT_HUNG_TASK_TIMEOUT; -unsigned long __read_mostly sysctl_hung_task_warnings = 10; +int __read_mostly sysctl_hung_task_warnings = 10;  static int __read_mostly did_panic; @@ -51,8 +52,10 @@ unsigned int __read_mostly sysctl_hung_task_panic =  static int __init hung_task_panic_setup(char *str)  { -	sysctl_hung_task_panic = simple_strtoul(str, NULL, 0); +	int rc = kstrtouint(str, 0, &sysctl_hung_task_panic); +	if (rc) +		return rc;  	return 1;  }  __setup("hung_task_panic=", hung_task_panic_setup); @@ -92,9 +95,14 @@ static void check_hung_task(struct task_struct *t, unsigned long timeout)  		t->last_switch_count = switch_count;  		return;  	} + +	trace_sched_process_hang(t); +  	if (!sysctl_hung_task_warnings)  		return; -	sysctl_hung_task_warnings--; + +	if (sysctl_hung_task_warnings > 0) +		sysctl_hung_task_warnings--;  	/*  	 * Ok, the task did not get scheduled for more than 2 minutes, @@ -203,6 +211,14 @@ int proc_dohung_task_timeout_secs(struct ctl_table *table, int write,  	return ret;  } +static atomic_t reset_hung_task = ATOMIC_INIT(0); + +void reset_hung_task_detector(void) +{ +	atomic_set(&reset_hung_task, 1); +} +EXPORT_SYMBOL_GPL(reset_hung_task_detector); +  /*   * kthread which checks for tasks stuck in D state   */ @@ -216,6 +232,9 @@ static int watchdog(void *dummy)  		while (schedule_timeout_interruptible(timeout_jiffies(timeout)))  			timeout = sysctl_hung_task_timeout_secs; +		if (atomic_xchg(&reset_hung_task, 0)) +			continue; +  		check_hung_uninterruptible_tasks(timeout);  	} @@ -229,5 +248,4 @@ static int __init hung_task_init(void)  	return 0;  } - -module_init(hung_task_init); +subsys_initcall(hung_task_init);  | 
