diff options
Diffstat (limited to 'kernel/sched/cpupri.c')
| -rw-r--r-- | kernel/sched/cpupri.c | 48 | 
1 files changed, 46 insertions, 2 deletions
| diff --git a/kernel/sched/cpupri.c b/kernel/sched/cpupri.c index 981fcd7dc394..14225d5d8617 100644 --- a/kernel/sched/cpupri.c +++ b/kernel/sched/cpupri.c @@ -27,6 +27,8 @@   *  of the License.   */ +#include "sched.h" +  #include <linux/gfp.h>  #include <linux/sched.h>  #include <linux/sched/rt.h> @@ -51,6 +53,27 @@ static int convert_prio(int prio)  }  /** + * drop_nopreempt_cpus - remove a cpu from the mask if it is likely + *			 non-preemptible + * @lowest_mask: mask with selected CPUs (non-NULL) + */ +static void +drop_nopreempt_cpus(struct cpumask *lowest_mask) +{ +	unsigned int cpu = cpumask_first(lowest_mask); + +	while (cpu < nr_cpu_ids) { +		/* unlocked access */ +		struct task_struct *task = READ_ONCE(cpu_rq(cpu)->curr); + +		if (task_may_not_preempt(task, cpu)) +			cpumask_clear_cpu(cpu, lowest_mask); + +		cpu = cpumask_next(cpu, lowest_mask); +	} +} + +/**   * cpupri_find - find the best (lowest-pri) CPU in the system   * @cp: The cpupri context   * @p: The task @@ -70,9 +93,11 @@ int cpupri_find(struct cpupri *cp, struct task_struct *p,  {  	int idx = 0;  	int task_pri = convert_prio(p->prio); +	bool drop_nopreempts = task_pri <= MAX_RT_PRIO;  	BUG_ON(task_pri >= CPUPRI_NR_PRIORITIES); +retry:  	for (idx = 0; idx < task_pri; idx++) {  		struct cpupri_vec *vec  = &cp->pri_to_cpu[idx];  		int skip = 0; @@ -108,7 +133,8 @@ int cpupri_find(struct cpupri *cp, struct task_struct *p,  		if (lowest_mask) {  			cpumask_and(lowest_mask, &p->cpus_allowed, vec->mask); - +			if (drop_nopreempts) +				drop_nopreempt_cpus(lowest_mask);  			/*  			 * We have to ensure that we have at least one bit  			 * still set in the array, since the map could have @@ -123,7 +149,14 @@ int cpupri_find(struct cpupri *cp, struct task_struct *p,  		return 1;  	} - +	/* +	 * If we can't find any non-preemptible cpu's, retry so we can +	 * find the lowest priority target and avoid priority inversion. +	 */ +	if (drop_nopreempts) { +		drop_nopreempts = false; +		goto retry; +	}  	return 0;  } @@ -246,3 +279,14 @@ void cpupri_cleanup(struct cpupri *cp)  	for (i = 0; i < CPUPRI_NR_PRIORITIES; i++)  		free_cpumask_var(cp->pri_to_cpu[i].mask);  } + +/* + * cpupri_check_rt - check if CPU has a RT task + * should be called from rcu-sched read section. + */ +bool cpupri_check_rt(void) +{ +	int cpu = raw_smp_processor_id(); + +	return cpu_rq(cpu)->rd->cpupri.cpu_to_pri[cpu] > CPUPRI_NORMAL; +} | 
