Commit 9a7e0b18 authored by Peter Zijlstra's avatar Peter Zijlstra Committed by Ingo Molnar

sched: rt-bandwidth fixes

The last patch allows sysctl_sched_rt_runtime to disable bandwidth accounting
for the group scheduler - however it doesn't deal with sched_setscheduler(),
which will keep tasks out of groups that have no assigned runtime.

If we relax this, we get into the situation where RT tasks can get into a group
when we disable bandwidth control, and then starve them by enabling it again.

Rework the schedulability code to check for this condition and fail to turn
on bandwidth control with -EBUSY when this situation is found.
Signed-off-by: default avatarPeter Zijlstra <a.p.zijlstra@chello.nl>
Signed-off-by: default avatarIngo Molnar <mingo@elte.hu>
parent eb755805
...@@ -300,9 +300,9 @@ static DEFINE_PER_CPU(struct cfs_rq, init_cfs_rq) ____cacheline_aligned_in_smp; ...@@ -300,9 +300,9 @@ static DEFINE_PER_CPU(struct cfs_rq, init_cfs_rq) ____cacheline_aligned_in_smp;
static DEFINE_PER_CPU(struct sched_rt_entity, init_sched_rt_entity); static DEFINE_PER_CPU(struct sched_rt_entity, init_sched_rt_entity);
static DEFINE_PER_CPU(struct rt_rq, init_rt_rq) ____cacheline_aligned_in_smp; static DEFINE_PER_CPU(struct rt_rq, init_rt_rq) ____cacheline_aligned_in_smp;
#endif /* CONFIG_RT_GROUP_SCHED */ #endif /* CONFIG_RT_GROUP_SCHED */
#else /* !CONFIG_FAIR_GROUP_SCHED */ #else /* !CONFIG_USER_SCHED */
#define root_task_group init_task_group #define root_task_group init_task_group
#endif /* CONFIG_FAIR_GROUP_SCHED */ #endif /* CONFIG_USER_SCHED */
/* task_group_lock serializes add/remove of task groups and also changes to /* task_group_lock serializes add/remove of task groups and also changes to
* a task group's cpu shares. * a task group's cpu shares.
...@@ -1387,7 +1387,7 @@ static inline void dec_cpu_load(struct rq *rq, unsigned long load) ...@@ -1387,7 +1387,7 @@ static inline void dec_cpu_load(struct rq *rq, unsigned long load)
update_load_sub(&rq->load, load); update_load_sub(&rq->load, load);
} }
#if (defined(CONFIG_SMP) && defined(CONFIG_FAIR_GROUP_SCHED)) #if (defined(CONFIG_SMP) && defined(CONFIG_FAIR_GROUP_SCHED)) || defined(SCHED_RT_GROUP_SCHED)
typedef int (*tg_visitor)(struct task_group *, void *); typedef int (*tg_visitor)(struct task_group *, void *);
/* /*
...@@ -5082,7 +5082,8 @@ static int __sched_setscheduler(struct task_struct *p, int policy, ...@@ -5082,7 +5082,8 @@ static int __sched_setscheduler(struct task_struct *p, int policy,
* Do not allow realtime tasks into groups that have no runtime * Do not allow realtime tasks into groups that have no runtime
* assigned. * assigned.
*/ */
if (rt_policy(policy) && task_group(p)->rt_bandwidth.rt_runtime == 0) if (rt_bandwidth_enabled() && rt_policy(policy) &&
task_group(p)->rt_bandwidth.rt_runtime == 0)
return -EPERM; return -EPERM;
#endif #endif
...@@ -8707,73 +8708,77 @@ static DEFINE_MUTEX(rt_constraints_mutex); ...@@ -8707,73 +8708,77 @@ static DEFINE_MUTEX(rt_constraints_mutex);
static unsigned long to_ratio(u64 period, u64 runtime) static unsigned long to_ratio(u64 period, u64 runtime)
{ {
if (runtime == RUNTIME_INF) if (runtime == RUNTIME_INF)
return 1ULL << 16; return 1ULL << 20;
return div64_u64(runtime << 16, period); return div64_u64(runtime << 20, period);
} }
#ifdef CONFIG_CGROUP_SCHED /* Must be called with tasklist_lock held */
static int __rt_schedulable(struct task_group *tg, u64 period, u64 runtime) static inline int tg_has_rt_tasks(struct task_group *tg)
{ {
struct task_group *tgi, *parent = tg->parent; struct task_struct *g, *p;
unsigned long total = 0;
do_each_thread(g, p) {
if (rt_task(p) && rt_rq_of_se(&p->rt)->tg == tg)
return 1;
} while_each_thread(g, p);
if (!parent) {
if (global_rt_period() < period)
return 0; return 0;
}
return to_ratio(period, runtime) < struct rt_schedulable_data {
to_ratio(global_rt_period(), global_rt_runtime()); struct task_group *tg;
} u64 rt_period;
u64 rt_runtime;
};
if (ktime_to_ns(parent->rt_bandwidth.rt_period) < period) static int tg_schedulable(struct task_group *tg, void *data)
return 0; {
struct rt_schedulable_data *d = data;
struct task_group *child;
unsigned long total, sum = 0;
u64 period, runtime;
rcu_read_lock(); period = ktime_to_ns(tg->rt_bandwidth.rt_period);
list_for_each_entry_rcu(tgi, &parent->children, siblings) { runtime = tg->rt_bandwidth.rt_runtime;
if (tgi == tg)
continue;
total += to_ratio(ktime_to_ns(tgi->rt_bandwidth.rt_period), if (tg == d->tg) {
tgi->rt_bandwidth.rt_runtime); period = d->rt_period;
runtime = d->rt_runtime;
} }
rcu_read_unlock();
return total + to_ratio(period, runtime) <= if (rt_bandwidth_enabled() && !runtime && tg_has_rt_tasks(tg))
to_ratio(ktime_to_ns(parent->rt_bandwidth.rt_period), return -EBUSY;
parent->rt_bandwidth.rt_runtime);
}
#elif defined CONFIG_USER_SCHED
static int __rt_schedulable(struct task_group *tg, u64 period, u64 runtime)
{
struct task_group *tgi;
unsigned long total = 0;
unsigned long global_ratio =
to_ratio(global_rt_period(), global_rt_runtime());
rcu_read_lock(); total = to_ratio(period, runtime);
list_for_each_entry_rcu(tgi, &task_groups, list) {
if (tgi == tg) list_for_each_entry_rcu(child, &tg->children, siblings) {
continue; period = ktime_to_ns(child->rt_bandwidth.rt_period);
runtime = child->rt_bandwidth.rt_runtime;
total += to_ratio(ktime_to_ns(tgi->rt_bandwidth.rt_period), if (child == d->tg) {
tgi->rt_bandwidth.rt_runtime); period = d->rt_period;
runtime = d->rt_runtime;
} }
rcu_read_unlock();
return total + to_ratio(period, runtime) < global_ratio; sum += to_ratio(period, runtime);
}
if (sum > total)
return -EINVAL;
return 0;
} }
#endif
/* Must be called with tasklist_lock held */ static int __rt_schedulable(struct task_group *tg, u64 period, u64 runtime)
static inline int tg_has_rt_tasks(struct task_group *tg)
{ {
struct task_struct *g, *p; struct rt_schedulable_data data = {
do_each_thread(g, p) { .tg = tg,
if (rt_task(p) && rt_rq_of_se(&p->rt)->tg == tg) .rt_period = period,
return 1; .rt_runtime = runtime,
} while_each_thread(g, p); };
return 0;
return walk_tg_tree(tg_schedulable, tg_nop, &data);
} }
static int tg_set_bandwidth(struct task_group *tg, static int tg_set_bandwidth(struct task_group *tg,
...@@ -8783,14 +8788,9 @@ static int tg_set_bandwidth(struct task_group *tg, ...@@ -8783,14 +8788,9 @@ static int tg_set_bandwidth(struct task_group *tg,
mutex_lock(&rt_constraints_mutex); mutex_lock(&rt_constraints_mutex);
read_lock(&tasklist_lock); read_lock(&tasklist_lock);
if (rt_runtime == 0 && tg_has_rt_tasks(tg)) { err = __rt_schedulable(tg, rt_period, rt_runtime);
err = -EBUSY; if (err)
goto unlock; goto unlock;
}
if (!__rt_schedulable(tg, rt_period, rt_runtime)) {
err = -EINVAL;
goto unlock;
}
spin_lock_irq(&tg->rt_bandwidth.rt_runtime_lock); spin_lock_irq(&tg->rt_bandwidth.rt_runtime_lock);
tg->rt_bandwidth.rt_period = ns_to_ktime(rt_period); tg->rt_bandwidth.rt_period = ns_to_ktime(rt_period);
...@@ -8867,8 +8867,9 @@ static int sched_rt_global_constraints(void) ...@@ -8867,8 +8867,9 @@ static int sched_rt_global_constraints(void)
rt_runtime = tg->rt_bandwidth.rt_runtime; rt_runtime = tg->rt_bandwidth.rt_runtime;
mutex_lock(&rt_constraints_mutex); mutex_lock(&rt_constraints_mutex);
if (!__rt_schedulable(tg, rt_period, rt_runtime)) read_lock(&tasklist_lock);
ret = -EINVAL; ret = __rt_schedulable(tg, rt_period, rt_runtime);
read_unlock(&tasklist_lock);
mutex_unlock(&rt_constraints_mutex); mutex_unlock(&rt_constraints_mutex);
return ret; return ret;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment