ipc/sem.c: move wake_up_process out of the spinlock section
The wake-up part of semtimedop() consists out of two steps: - the right tasks must be identified. - they must be woken up. Right now, both steps run while the array spinlock is held. This patch reorders the code and moves the actual wake_up_process() behind the point where the spinlock is dropped. The code also moves setting sem->sem_otime to one place: It does not make sense to set the last modify time multiple times. [akpm@linux-foundation.org: repair kerneldoc] [akpm@linux-foundation.org: fix uninitialised retval] Signed-off-by: Manfred Spraul <manfred@colorfullife.com> Cc: Chris Mason <chris.mason@oracle.com> Cc: Zach Brown <zach.brown@oracle.com> Cc: Jens Axboe <jens.axboe@oracle.com> Cc: Nick Piggin <npiggin@suse.de> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
fd5db42254
commit
0a2b9d4c79
1 changed files with 91 additions and 32 deletions
109
ipc/sem.c
109
ipc/sem.c
|
@ -381,7 +381,6 @@ static int try_atomic_semop (struct sem_array * sma, struct sembuf * sops,
|
||||||
sop--;
|
sop--;
|
||||||
}
|
}
|
||||||
|
|
||||||
sma->sem_otime = get_seconds();
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
out_of_range:
|
out_of_range:
|
||||||
|
@ -404,24 +403,50 @@ static int try_atomic_semop (struct sem_array * sma, struct sembuf * sops,
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/** wake_up_sem_queue_prepare(q, error): Prepare wake-up
|
||||||
* Wake up a process waiting on the sem queue with a given error.
|
* @q: queue entry that must be signaled
|
||||||
* The queue is invalid (may not be accessed) after the function returns.
|
* @error: Error value for the signal
|
||||||
|
*
|
||||||
|
* Prepare the wake-up of the queue entry q.
|
||||||
*/
|
*/
|
||||||
static void wake_up_sem_queue(struct sem_queue *q, int error)
|
static void wake_up_sem_queue_prepare(struct list_head *pt,
|
||||||
|
struct sem_queue *q, int error)
|
||||||
{
|
{
|
||||||
|
if (list_empty(pt)) {
|
||||||
/*
|
/*
|
||||||
* Hold preempt off so that we don't get preempted and have the
|
* Hold preempt off so that we don't get preempted and have the
|
||||||
* wakee busy-wait until we're scheduled back on. We're holding
|
* wakee busy-wait until we're scheduled back on.
|
||||||
* locks here so it may not strictly be needed, however if the
|
|
||||||
* locks become preemptible then this prevents such a problem.
|
|
||||||
*/
|
*/
|
||||||
preempt_disable();
|
preempt_disable();
|
||||||
|
}
|
||||||
q->status = IN_WAKEUP;
|
q->status = IN_WAKEUP;
|
||||||
|
q->pid = error;
|
||||||
|
|
||||||
|
list_add_tail(&q->simple_list, pt);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* wake_up_sem_queue_do(pt) - do the actual wake-up
|
||||||
|
* @pt: list of tasks to be woken up
|
||||||
|
*
|
||||||
|
* Do the actual wake-up.
|
||||||
|
* The function is called without any locks held, thus the semaphore array
|
||||||
|
* could be destroyed already and the tasks can disappear as soon as the
|
||||||
|
* status is set to the actual return code.
|
||||||
|
*/
|
||||||
|
static void wake_up_sem_queue_do(struct list_head *pt)
|
||||||
|
{
|
||||||
|
struct sem_queue *q, *t;
|
||||||
|
int did_something;
|
||||||
|
|
||||||
|
did_something = !list_empty(pt);
|
||||||
|
list_for_each_entry_safe(q, t, pt, simple_list) {
|
||||||
wake_up_process(q->sleeper);
|
wake_up_process(q->sleeper);
|
||||||
/* hands-off: q can disappear immediately after writing q->status. */
|
/* q can disappear immediately after writing q->status. */
|
||||||
smp_wmb();
|
smp_wmb();
|
||||||
q->status = error;
|
q->status = q->pid;
|
||||||
|
}
|
||||||
|
if (did_something)
|
||||||
preempt_enable();
|
preempt_enable();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -502,17 +527,22 @@ static int check_restart(struct sem_array *sma, struct sem_queue *q)
|
||||||
* update_queue(sma, semnum): Look for tasks that can be completed.
|
* update_queue(sma, semnum): Look for tasks that can be completed.
|
||||||
* @sma: semaphore array.
|
* @sma: semaphore array.
|
||||||
* @semnum: semaphore that was modified.
|
* @semnum: semaphore that was modified.
|
||||||
|
* @pt: list head for the tasks that must be woken up.
|
||||||
*
|
*
|
||||||
* update_queue must be called after a semaphore in a semaphore array
|
* update_queue must be called after a semaphore in a semaphore array
|
||||||
* was modified. If multiple semaphore were modified, then @semnum
|
* was modified. If multiple semaphore were modified, then @semnum
|
||||||
* must be set to -1.
|
* must be set to -1.
|
||||||
|
* The tasks that must be woken up are added to @pt. The return code
|
||||||
|
* is stored in q->pid.
|
||||||
|
* The function return 1 if at least one semop was completed successfully.
|
||||||
*/
|
*/
|
||||||
static void update_queue(struct sem_array *sma, int semnum)
|
static int update_queue(struct sem_array *sma, int semnum, struct list_head *pt)
|
||||||
{
|
{
|
||||||
struct sem_queue *q;
|
struct sem_queue *q;
|
||||||
struct list_head *walk;
|
struct list_head *walk;
|
||||||
struct list_head *pending_list;
|
struct list_head *pending_list;
|
||||||
int offset;
|
int offset;
|
||||||
|
int semop_completed = 0;
|
||||||
|
|
||||||
/* if there are complex operations around, then knowing the semaphore
|
/* if there are complex operations around, then knowing the semaphore
|
||||||
* that was modified doesn't help us. Assume that multiple semaphores
|
* that was modified doesn't help us. Assume that multiple semaphores
|
||||||
|
@ -557,40 +587,55 @@ static void update_queue(struct sem_array *sma, int semnum)
|
||||||
|
|
||||||
unlink_queue(sma, q);
|
unlink_queue(sma, q);
|
||||||
|
|
||||||
if (error)
|
if (error) {
|
||||||
restart = 0;
|
restart = 0;
|
||||||
else
|
} else {
|
||||||
|
semop_completed = 1;
|
||||||
restart = check_restart(sma, q);
|
restart = check_restart(sma, q);
|
||||||
|
}
|
||||||
|
|
||||||
wake_up_sem_queue(q, error);
|
wake_up_sem_queue_prepare(pt, q, error);
|
||||||
if (restart)
|
if (restart)
|
||||||
goto again;
|
goto again;
|
||||||
}
|
}
|
||||||
|
return semop_completed;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** do_smart_update(sma, sops, nsops): Optimized update_queue
|
/**
|
||||||
|
* do_smart_update(sma, sops, nsops, otime, pt) - optimized update_queue
|
||||||
* @sma: semaphore array
|
* @sma: semaphore array
|
||||||
* @sops: operations that were performed
|
* @sops: operations that were performed
|
||||||
* @nsops: number of operations
|
* @nsops: number of operations
|
||||||
|
* @otime: force setting otime
|
||||||
|
* @pt: list head of the tasks that must be woken up.
|
||||||
*
|
*
|
||||||
* do_smart_update() does the required called to update_queue, based on the
|
* do_smart_update() does the required called to update_queue, based on the
|
||||||
* actual changes that were performed on the semaphore array.
|
* actual changes that were performed on the semaphore array.
|
||||||
|
* Note that the function does not do the actual wake-up: the caller is
|
||||||
|
* responsible for calling wake_up_sem_queue_do(@pt).
|
||||||
|
* It is safe to perform this call after dropping all locks.
|
||||||
*/
|
*/
|
||||||
static void do_smart_update(struct sem_array *sma, struct sembuf *sops, int nsops)
|
static void do_smart_update(struct sem_array *sma, struct sembuf *sops, int nsops,
|
||||||
|
int otime, struct list_head *pt)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
if (sma->complex_count || sops == NULL) {
|
if (sma->complex_count || sops == NULL) {
|
||||||
update_queue(sma, -1);
|
if (update_queue(sma, -1, pt))
|
||||||
return;
|
otime = 1;
|
||||||
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0; i < nsops; i++) {
|
for (i = 0; i < nsops; i++) {
|
||||||
if (sops[i].sem_op > 0 ||
|
if (sops[i].sem_op > 0 ||
|
||||||
(sops[i].sem_op < 0 &&
|
(sops[i].sem_op < 0 &&
|
||||||
sma->sem_base[sops[i].sem_num].semval == 0))
|
sma->sem_base[sops[i].sem_num].semval == 0))
|
||||||
update_queue(sma, sops[i].sem_num);
|
if (update_queue(sma, sops[i].sem_num, pt))
|
||||||
|
otime = 1;
|
||||||
}
|
}
|
||||||
|
done:
|
||||||
|
if (otime)
|
||||||
|
sma->sem_otime = get_seconds();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -656,6 +701,7 @@ static void freeary(struct ipc_namespace *ns, struct kern_ipc_perm *ipcp)
|
||||||
struct sem_undo *un, *tu;
|
struct sem_undo *un, *tu;
|
||||||
struct sem_queue *q, *tq;
|
struct sem_queue *q, *tq;
|
||||||
struct sem_array *sma = container_of(ipcp, struct sem_array, sem_perm);
|
struct sem_array *sma = container_of(ipcp, struct sem_array, sem_perm);
|
||||||
|
struct list_head tasks;
|
||||||
|
|
||||||
/* Free the existing undo structures for this semaphore set. */
|
/* Free the existing undo structures for this semaphore set. */
|
||||||
assert_spin_locked(&sma->sem_perm.lock);
|
assert_spin_locked(&sma->sem_perm.lock);
|
||||||
|
@ -669,15 +715,17 @@ static void freeary(struct ipc_namespace *ns, struct kern_ipc_perm *ipcp)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Wake up all pending processes and let them fail with EIDRM. */
|
/* Wake up all pending processes and let them fail with EIDRM. */
|
||||||
|
INIT_LIST_HEAD(&tasks);
|
||||||
list_for_each_entry_safe(q, tq, &sma->sem_pending, list) {
|
list_for_each_entry_safe(q, tq, &sma->sem_pending, list) {
|
||||||
unlink_queue(sma, q);
|
unlink_queue(sma, q);
|
||||||
wake_up_sem_queue(q, -EIDRM);
|
wake_up_sem_queue_prepare(&tasks, q, -EIDRM);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Remove the semaphore set from the IDR */
|
/* Remove the semaphore set from the IDR */
|
||||||
sem_rmid(ns, sma);
|
sem_rmid(ns, sma);
|
||||||
sem_unlock(sma);
|
sem_unlock(sma);
|
||||||
|
|
||||||
|
wake_up_sem_queue_do(&tasks);
|
||||||
ns->used_sems -= sma->sem_nsems;
|
ns->used_sems -= sma->sem_nsems;
|
||||||
security_sem_free(sma);
|
security_sem_free(sma);
|
||||||
ipc_rcu_putref(sma);
|
ipc_rcu_putref(sma);
|
||||||
|
@ -799,11 +847,13 @@ static int semctl_main(struct ipc_namespace *ns, int semid, int semnum,
|
||||||
ushort fast_sem_io[SEMMSL_FAST];
|
ushort fast_sem_io[SEMMSL_FAST];
|
||||||
ushort* sem_io = fast_sem_io;
|
ushort* sem_io = fast_sem_io;
|
||||||
int nsems;
|
int nsems;
|
||||||
|
struct list_head tasks;
|
||||||
|
|
||||||
sma = sem_lock_check(ns, semid);
|
sma = sem_lock_check(ns, semid);
|
||||||
if (IS_ERR(sma))
|
if (IS_ERR(sma))
|
||||||
return PTR_ERR(sma);
|
return PTR_ERR(sma);
|
||||||
|
|
||||||
|
INIT_LIST_HEAD(&tasks);
|
||||||
nsems = sma->sem_nsems;
|
nsems = sma->sem_nsems;
|
||||||
|
|
||||||
err = -EACCES;
|
err = -EACCES;
|
||||||
|
@ -891,7 +941,7 @@ static int semctl_main(struct ipc_namespace *ns, int semid, int semnum,
|
||||||
}
|
}
|
||||||
sma->sem_ctime = get_seconds();
|
sma->sem_ctime = get_seconds();
|
||||||
/* maybe some queued-up processes were waiting for this */
|
/* maybe some queued-up processes were waiting for this */
|
||||||
update_queue(sma, -1);
|
do_smart_update(sma, NULL, 0, 0, &tasks);
|
||||||
err = 0;
|
err = 0;
|
||||||
goto out_unlock;
|
goto out_unlock;
|
||||||
}
|
}
|
||||||
|
@ -933,13 +983,15 @@ static int semctl_main(struct ipc_namespace *ns, int semid, int semnum,
|
||||||
curr->sempid = task_tgid_vnr(current);
|
curr->sempid = task_tgid_vnr(current);
|
||||||
sma->sem_ctime = get_seconds();
|
sma->sem_ctime = get_seconds();
|
||||||
/* maybe some queued-up processes were waiting for this */
|
/* maybe some queued-up processes were waiting for this */
|
||||||
update_queue(sma, semnum);
|
do_smart_update(sma, NULL, 0, 0, &tasks);
|
||||||
err = 0;
|
err = 0;
|
||||||
goto out_unlock;
|
goto out_unlock;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
out_unlock:
|
out_unlock:
|
||||||
sem_unlock(sma);
|
sem_unlock(sma);
|
||||||
|
wake_up_sem_queue_do(&tasks);
|
||||||
|
|
||||||
out_free:
|
out_free:
|
||||||
if(sem_io != fast_sem_io)
|
if(sem_io != fast_sem_io)
|
||||||
ipc_free(sem_io, sizeof(ushort)*nsems);
|
ipc_free(sem_io, sizeof(ushort)*nsems);
|
||||||
|
@ -1213,6 +1265,7 @@ SYSCALL_DEFINE4(semtimedop, int, semid, struct sembuf __user *, tsops,
|
||||||
struct sem_queue queue;
|
struct sem_queue queue;
|
||||||
unsigned long jiffies_left = 0;
|
unsigned long jiffies_left = 0;
|
||||||
struct ipc_namespace *ns;
|
struct ipc_namespace *ns;
|
||||||
|
struct list_head tasks;
|
||||||
|
|
||||||
ns = current->nsproxy->ipc_ns;
|
ns = current->nsproxy->ipc_ns;
|
||||||
|
|
||||||
|
@ -1261,6 +1314,8 @@ SYSCALL_DEFINE4(semtimedop, int, semid, struct sembuf __user *, tsops,
|
||||||
} else
|
} else
|
||||||
un = NULL;
|
un = NULL;
|
||||||
|
|
||||||
|
INIT_LIST_HEAD(&tasks);
|
||||||
|
|
||||||
sma = sem_lock_check(ns, semid);
|
sma = sem_lock_check(ns, semid);
|
||||||
if (IS_ERR(sma)) {
|
if (IS_ERR(sma)) {
|
||||||
if (un)
|
if (un)
|
||||||
|
@ -1309,7 +1364,7 @@ SYSCALL_DEFINE4(semtimedop, int, semid, struct sembuf __user *, tsops,
|
||||||
error = try_atomic_semop (sma, sops, nsops, un, task_tgid_vnr(current));
|
error = try_atomic_semop (sma, sops, nsops, un, task_tgid_vnr(current));
|
||||||
if (error <= 0) {
|
if (error <= 0) {
|
||||||
if (alter && error == 0)
|
if (alter && error == 0)
|
||||||
do_smart_update(sma, sops, nsops);
|
do_smart_update(sma, sops, nsops, 1, &tasks);
|
||||||
|
|
||||||
goto out_unlock_free;
|
goto out_unlock_free;
|
||||||
}
|
}
|
||||||
|
@ -1386,6 +1441,8 @@ SYSCALL_DEFINE4(semtimedop, int, semid, struct sembuf __user *, tsops,
|
||||||
|
|
||||||
out_unlock_free:
|
out_unlock_free:
|
||||||
sem_unlock(sma);
|
sem_unlock(sma);
|
||||||
|
|
||||||
|
wake_up_sem_queue_do(&tasks);
|
||||||
out_free:
|
out_free:
|
||||||
if(sops != fast_sops)
|
if(sops != fast_sops)
|
||||||
kfree(sops);
|
kfree(sops);
|
||||||
|
@ -1446,6 +1503,7 @@ void exit_sem(struct task_struct *tsk)
|
||||||
for (;;) {
|
for (;;) {
|
||||||
struct sem_array *sma;
|
struct sem_array *sma;
|
||||||
struct sem_undo *un;
|
struct sem_undo *un;
|
||||||
|
struct list_head tasks;
|
||||||
int semid;
|
int semid;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
|
@ -1509,10 +1567,11 @@ void exit_sem(struct task_struct *tsk)
|
||||||
semaphore->sempid = task_tgid_vnr(current);
|
semaphore->sempid = task_tgid_vnr(current);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
sma->sem_otime = get_seconds();
|
|
||||||
/* maybe some queued-up processes were waiting for this */
|
/* maybe some queued-up processes were waiting for this */
|
||||||
update_queue(sma, -1);
|
INIT_LIST_HEAD(&tasks);
|
||||||
|
do_smart_update(sma, NULL, 0, 1, &tasks);
|
||||||
sem_unlock(sma);
|
sem_unlock(sma);
|
||||||
|
wake_up_sem_queue_do(&tasks);
|
||||||
|
|
||||||
call_rcu(&un->rcu, free_un);
|
call_rcu(&un->rcu, free_un);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue