IPC: consolidate sem_exit_ns(), msg_exit_ns() and shm_exit_ns()
[linux-2.6.git] / ipc / namespace.c
index cef1139..1b96765 100644 (file)
 
 static struct ipc_namespace *clone_ipc_ns(struct ipc_namespace *old_ns)
 {
-       int err;
        struct ipc_namespace *ns;
 
-       err = -ENOMEM;
        ns = kmalloc(sizeof(struct ipc_namespace), GFP_KERNEL);
        if (ns == NULL)
-               goto err_mem;
-
-       err = sem_init_ns(ns);
-       if (err)
-               goto err_sem;
-       err = msg_init_ns(ns);
-       if (err)
-               goto err_msg;
-       err = shm_init_ns(ns);
-       if (err)
-               goto err_shm;
+               return ERR_PTR(-ENOMEM);
+
+       sem_init_ns(ns);
+       msg_init_ns(ns);
+       shm_init_ns(ns);
 
        kref_init(&ns->kref);
        return ns;
-
-err_shm:
-       msg_exit_ns(ns);
-err_msg:
-       sem_exit_ns(ns);
-err_sem:
-       kfree(ns);
-err_mem:
-       return ERR_PTR(err);
 }
 
 struct ipc_namespace *copy_ipcs(unsigned long flags, struct ipc_namespace *ns)
@@ -61,6 +44,36 @@ struct ipc_namespace *copy_ipcs(unsigned long flags, struct ipc_namespace *ns)
        return new_ns;
 }
 
+/*
+ * free_ipcs - free all ipcs of one type
+ * @ns:   the namespace to remove the ipcs from
+ * @ids:  the table of ipcs to free
+ * @free: the function called to free each individual ipc
+ *
+ * Called for each kind of ipc when an ipc_namespace exits.
+ */
+void free_ipcs(struct ipc_namespace *ns, struct ipc_ids *ids,
+              void (*free)(struct ipc_namespace *, struct kern_ipc_perm *))
+{
+       struct kern_ipc_perm *perm;
+       int next_id;
+       int total, in_use;
+
+       down_write(&ids->rw_mutex);
+
+       in_use = ids->in_use;
+
+       for (total = 0, next_id = 0; total < in_use; next_id++) {
+               perm = idr_find(&ids->ipcs_idr, next_id);
+               if (perm == NULL)
+                       continue;
+               ipc_lock_by_ptr(perm);
+               free(ns, perm);
+               total++;
+       }
+       up_write(&ids->rw_mutex);
+}
+
 void free_ipc_ns(struct kref *kref)
 {
        struct ipc_namespace *ns;