From: Rik van Riel Date: Tue, 26 Mar 2013 23:26:02 +0000 (+1100) Subject: ipc,sem: open code and rename sem_lock X-Git-Tag: next-20130404~3^2~103 X-Git-Url: https://git.karo-electronics.de/?a=commitdiff_plain;h=e0e798bb0e887a4d3737d0c6df505c5bad7607b8;p=karo-tx-linux.git ipc,sem: open code and rename sem_lock Rename sem_lock() to sem_obtain_lock(), so we can introduce a sem_lock() later that only locks the sem_array and does nothing else. Open code the locking from ipc_lock() in sem_obtain_lock() so we can introduce finer grained locking for the sem_array in the next patch. Signed-off-by: Rik van Riel Acked-by: Davidlohr Bueso Cc: Chegu Vinod Cc: Emmanuel Benisty Cc: Jason Low Cc: Linus Torvalds Cc: Michel Lespinasse Cc: Peter Hurley Cc: Stanislav Kinsbursky Signed-off-by: Andrew Morton --- diff --git a/ipc/sem.c b/ipc/sem.c index 2c57e2143c0e..26c39ec01ab7 100644 --- a/ipc/sem.c +++ b/ipc/sem.c @@ -194,14 +194,29 @@ void __init sem_init (void) * sem_lock_(check_) routines are called in the paths where the rw_mutex * is not held. */ -static inline struct sem_array *sem_lock(struct ipc_namespace *ns, int id) +static inline struct sem_array *sem_obtain_lock(struct ipc_namespace *ns, int id) { - struct kern_ipc_perm *ipcp = ipc_lock(&sem_ids(ns), id); + struct kern_ipc_perm *ipcp; + rcu_read_lock(); + ipcp = ipc_obtain_object(&sem_ids(ns), id); if (IS_ERR(ipcp)) - return (struct sem_array *)ipcp; + goto err1; + + spin_lock(&ipcp->lock); + + /* ipc_rmid() may have already freed the ID while sem_lock + * was spinning: verify that the structure is still valid + */ + if (ipcp->deleted) + goto err0; return container_of(ipcp, struct sem_array, sem_perm); +err0: + spin_unlock(&ipcp->lock); +err1: + rcu_read_unlock(); + return ERR_PTR(-EINVAL); } static inline struct sem_array *sem_obtain_object(struct ipc_namespace *ns, int id) @@ -1591,7 +1606,7 @@ sleep_again: goto out_free; } - sma = sem_lock(ns, semid); + sma = sem_obtain_lock(ns, semid); /* * Wait until it's guaranteed that no wakeup_sem_queue_do() is ongoing.