Commit f61a89ca authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'sched_urgent_for_v6.5_rc2' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip

Pull scheduler fixes from Borislav Petkov:

 - Remove a cgroup from under a polling process properly

 - Fix the idle sibling selection

* tag 'sched_urgent_for_v6.5_rc2' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip:
  sched/psi: use kernfs polling functions for PSI trigger polling
  sched/fair: Use recent_used_cpu to test p->cpus_ptr
parents ede950b0 aff03707
...@@ -23,8 +23,9 @@ void psi_memstall_enter(unsigned long *flags); ...@@ -23,8 +23,9 @@ void psi_memstall_enter(unsigned long *flags);
void psi_memstall_leave(unsigned long *flags); void psi_memstall_leave(unsigned long *flags);
int psi_show(struct seq_file *s, struct psi_group *group, enum psi_res res); int psi_show(struct seq_file *s, struct psi_group *group, enum psi_res res);
struct psi_trigger *psi_trigger_create(struct psi_group *group, struct psi_trigger *psi_trigger_create(struct psi_group *group, char *buf,
char *buf, enum psi_res res, struct file *file); enum psi_res res, struct file *file,
struct kernfs_open_file *of);
void psi_trigger_destroy(struct psi_trigger *t); void psi_trigger_destroy(struct psi_trigger *t);
__poll_t psi_trigger_poll(void **trigger_ptr, struct file *file, __poll_t psi_trigger_poll(void **trigger_ptr, struct file *file,
......
...@@ -137,6 +137,9 @@ struct psi_trigger { ...@@ -137,6 +137,9 @@ struct psi_trigger {
/* Wait queue for polling */ /* Wait queue for polling */
wait_queue_head_t event_wait; wait_queue_head_t event_wait;
/* Kernfs file for cgroup triggers */
struct kernfs_open_file *of;
/* Pending event flag */ /* Pending event flag */
int event; int event;
......
...@@ -3730,7 +3730,7 @@ static ssize_t pressure_write(struct kernfs_open_file *of, char *buf, ...@@ -3730,7 +3730,7 @@ static ssize_t pressure_write(struct kernfs_open_file *of, char *buf,
} }
psi = cgroup_psi(cgrp); psi = cgroup_psi(cgrp);
new = psi_trigger_create(psi, buf, res, of->file); new = psi_trigger_create(psi, buf, res, of->file, of);
if (IS_ERR(new)) { if (IS_ERR(new)) {
cgroup_put(cgrp); cgroup_put(cgrp);
return PTR_ERR(new); return PTR_ERR(new);
......
...@@ -7174,7 +7174,7 @@ static int select_idle_sibling(struct task_struct *p, int prev, int target) ...@@ -7174,7 +7174,7 @@ static int select_idle_sibling(struct task_struct *p, int prev, int target)
recent_used_cpu != target && recent_used_cpu != target &&
cpus_share_cache(recent_used_cpu, target) && cpus_share_cache(recent_used_cpu, target) &&
(available_idle_cpu(recent_used_cpu) || sched_idle_cpu(recent_used_cpu)) && (available_idle_cpu(recent_used_cpu) || sched_idle_cpu(recent_used_cpu)) &&
cpumask_test_cpu(p->recent_used_cpu, p->cpus_ptr) && cpumask_test_cpu(recent_used_cpu, p->cpus_ptr) &&
asym_fits_cpu(task_util, util_min, util_max, recent_used_cpu)) { asym_fits_cpu(task_util, util_min, util_max, recent_used_cpu)) {
return recent_used_cpu; return recent_used_cpu;
} }
......
...@@ -493,8 +493,12 @@ static u64 update_triggers(struct psi_group *group, u64 now, bool *update_total, ...@@ -493,8 +493,12 @@ static u64 update_triggers(struct psi_group *group, u64 now, bool *update_total,
continue; continue;
/* Generate an event */ /* Generate an event */
if (cmpxchg(&t->event, 0, 1) == 0) if (cmpxchg(&t->event, 0, 1) == 0) {
if (t->of)
kernfs_notify(t->of->kn);
else
wake_up_interruptible(&t->event_wait); wake_up_interruptible(&t->event_wait);
}
t->last_event_time = now; t->last_event_time = now;
/* Reset threshold breach flag once event got generated */ /* Reset threshold breach flag once event got generated */
t->pending_event = false; t->pending_event = false;
...@@ -1271,8 +1275,9 @@ int psi_show(struct seq_file *m, struct psi_group *group, enum psi_res res) ...@@ -1271,8 +1275,9 @@ int psi_show(struct seq_file *m, struct psi_group *group, enum psi_res res)
return 0; return 0;
} }
struct psi_trigger *psi_trigger_create(struct psi_group *group, struct psi_trigger *psi_trigger_create(struct psi_group *group, char *buf,
char *buf, enum psi_res res, struct file *file) enum psi_res res, struct file *file,
struct kernfs_open_file *of)
{ {
struct psi_trigger *t; struct psi_trigger *t;
enum psi_states state; enum psi_states state;
...@@ -1331,6 +1336,8 @@ struct psi_trigger *psi_trigger_create(struct psi_group *group, ...@@ -1331,6 +1336,8 @@ struct psi_trigger *psi_trigger_create(struct psi_group *group,
t->event = 0; t->event = 0;
t->last_event_time = 0; t->last_event_time = 0;
t->of = of;
if (!of)
init_waitqueue_head(&t->event_wait); init_waitqueue_head(&t->event_wait);
t->pending_event = false; t->pending_event = false;
t->aggregator = privileged ? PSI_POLL : PSI_AVGS; t->aggregator = privileged ? PSI_POLL : PSI_AVGS;
...@@ -1388,7 +1395,10 @@ void psi_trigger_destroy(struct psi_trigger *t) ...@@ -1388,7 +1395,10 @@ void psi_trigger_destroy(struct psi_trigger *t)
* being accessed later. Can happen if cgroup is deleted from under a * being accessed later. Can happen if cgroup is deleted from under a
* polling process. * polling process.
*/ */
wake_up_pollfree(&t->event_wait); if (t->of)
kernfs_notify(t->of->kn);
else
wake_up_interruptible(&t->event_wait);
if (t->aggregator == PSI_AVGS) { if (t->aggregator == PSI_AVGS) {
mutex_lock(&group->avgs_lock); mutex_lock(&group->avgs_lock);
...@@ -1465,6 +1475,9 @@ __poll_t psi_trigger_poll(void **trigger_ptr, ...@@ -1465,6 +1475,9 @@ __poll_t psi_trigger_poll(void **trigger_ptr,
if (!t) if (!t)
return DEFAULT_POLLMASK | EPOLLERR | EPOLLPRI; return DEFAULT_POLLMASK | EPOLLERR | EPOLLPRI;
if (t->of)
kernfs_generic_poll(t->of, wait);
else
poll_wait(file, &t->event_wait, wait); poll_wait(file, &t->event_wait, wait);
if (cmpxchg(&t->event, 1, 0) == 1) if (cmpxchg(&t->event, 1, 0) == 1)
...@@ -1535,7 +1548,7 @@ static ssize_t psi_write(struct file *file, const char __user *user_buf, ...@@ -1535,7 +1548,7 @@ static ssize_t psi_write(struct file *file, const char __user *user_buf,
return -EBUSY; return -EBUSY;
} }
new = psi_trigger_create(&psi_system, buf, res, file); new = psi_trigger_create(&psi_system, buf, res, file, NULL);
if (IS_ERR(new)) { if (IS_ERR(new)) {
mutex_unlock(&seq->lock); mutex_unlock(&seq->lock);
return PTR_ERR(new); return PTR_ERR(new);
......
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