Skip to content

Commit

Permalink
ipc,msg: shorten critical region in msgsnd
Browse files Browse the repository at this point in the history
do_msgsnd() is another function that does too many things with the ipc
object lock acquired.  Take it only when needed when actually updating
msq.

Signed-off-by: Davidlohr Bueso <[email protected]>
Cc: Andi Kleen <[email protected]>
Cc: Rik van Riel <[email protected]>
Signed-off-by: Andrew Morton <[email protected]>
Signed-off-by: Linus Torvalds <[email protected]>
  • Loading branch information
Davidlohr Bueso authored and torvalds committed Jul 9, 2013
1 parent ac0ba20 commit 3dd1f78
Showing 1 changed file with 24 additions and 13 deletions.
37 changes: 24 additions & 13 deletions ipc/msg.c
Original file line number Diff line number Diff line change
Expand Up @@ -698,22 +698,23 @@ long do_msgsnd(int msqid, long mtype, void __user *mtext,
msg->m_type = mtype;
msg->m_ts = msgsz;

msq = msg_lock_check(ns, msqid);
rcu_read_lock();
msq = msq_obtain_object_check(ns, msqid);
if (IS_ERR(msq)) {
err = PTR_ERR(msq);
goto out_free;
goto out_unlock1;
}

for (;;) {
struct msg_sender s;

err = -EACCES;
if (ipcperms(ns, &msq->q_perm, S_IWUGO))
goto out_unlock_free;
goto out_unlock1;

err = security_msg_queue_msgsnd(msq, msg, msgflg);
if (err)
goto out_unlock_free;
goto out_unlock1;

if (msgsz + msq->q_cbytes <= msq->q_qbytes &&
1 + msq->q_qnum <= msq->q_qbytes) {
Expand All @@ -723,32 +724,41 @@ long do_msgsnd(int msqid, long mtype, void __user *mtext,
/* queue full, wait: */
if (msgflg & IPC_NOWAIT) {
err = -EAGAIN;
goto out_unlock_free;
goto out_unlock1;
}

ipc_lock_object(&msq->q_perm);
ss_add(msq, &s);

if (!ipc_rcu_getref(msq)) {
err = -EIDRM;
goto out_unlock_free;
goto out_unlock0;
}

msg_unlock(msq);
ipc_unlock_object(&msq->q_perm);
rcu_read_unlock();
schedule();

ipc_lock_by_ptr(&msq->q_perm);
rcu_read_lock();
ipc_lock_object(&msq->q_perm);

ipc_rcu_putref(msq);
if (msq->q_perm.deleted) {
err = -EIDRM;
goto out_unlock_free;
goto out_unlock0;
}

ss_del(&s);

if (signal_pending(current)) {
err = -ERESTARTNOHAND;
goto out_unlock_free;
goto out_unlock0;
}

ipc_unlock_object(&msq->q_perm);
}

ipc_lock_object(&msq->q_perm);
msq->q_lspid = task_tgid_vnr(current);
msq->q_stime = get_seconds();

Expand All @@ -764,9 +774,10 @@ long do_msgsnd(int msqid, long mtype, void __user *mtext,
err = 0;
msg = NULL;

out_unlock_free:
msg_unlock(msq);
out_free:
out_unlock0:
ipc_unlock_object(&msq->q_perm);
out_unlock1:
rcu_read_unlock();
if (msg != NULL)
free_msg(msg);
return err;
Expand Down

0 comments on commit 3dd1f78

Please sign in to comment.