static DEFINE_XARRAY_ALLOC(multicast_table);
 
 static const struct file_operations ucma_fops;
+static int __destroy_id(struct ucma_context *ctx);
 
 static inline struct ucma_context *_ucma_find_context(int id,
                                                      struct ucma_file *file)
        ctx = xa_load(&ctx_table, id);
        if (!ctx)
                ctx = ERR_PTR(-ENOENT);
-       else if (ctx->file != file || !ctx->cm_id)
+       else if (ctx->file != file)
                ctx = ERR_PTR(-EINVAL);
        return ctx;
 }
        refcount_set(&ctx->ref, 1);
        init_completion(&ctx->comp);
        INIT_LIST_HEAD(&ctx->mc_list);
+       /* So list_del() will work if we don't do ucma_finish_ctx() */
+       INIT_LIST_HEAD(&ctx->list);
        ctx->file = file;
        mutex_init(&ctx->mutex);
 
-       if (xa_alloc(&ctx_table, &ctx->id, ctx, xa_limit_32b, GFP_KERNEL))
-               goto error;
-
-       list_add_tail(&ctx->list, &file->ctx_list);
+       if (xa_alloc(&ctx_table, &ctx->id, NULL, xa_limit_32b, GFP_KERNEL)) {
+               kfree(ctx);
+               return NULL;
+       }
        return ctx;
+}
 
-error:
-       kfree(ctx);
-       return NULL;
+static void ucma_finish_ctx(struct ucma_context *ctx)
+{
+       lockdep_assert_held(&ctx->file->mut);
+       list_add_tail(&ctx->list, &ctx->file->ctx_list);
+       xa_store(&ctx_table, ctx->id, ctx, GFP_KERNEL);
 }
 
 static struct ucma_multicast* ucma_alloc_multicast(struct ucma_context *ctx)
 static ssize_t ucma_get_event(struct ucma_file *file, const char __user *inbuf,
                              int in_len, int out_len)
 {
-       struct ucma_context *ctx;
+       struct ucma_context *ctx = NULL;
        struct rdma_ucm_get_event cmd;
        struct ucma_event *uevent;
        int ret = 0;
                mutex_lock(&file->mut);
        }
 
-       uevent = list_entry(file->event_list.next, struct ucma_event, list);
+       uevent = list_first_entry(&file->event_list, struct ucma_event, list);
 
        if (uevent->resp.event == RDMA_CM_EVENT_CONNECT_REQUEST) {
                ctx = ucma_alloc_ctx(file);
                if (!ctx) {
                        ret = -ENOMEM;
-                       goto done;
+                       goto err_unlock;
                }
-               uevent->ctx->backlog++;
-               ctx->cm_id = uevent->cm_id;
-               ctx->cm_id->context = ctx;
                uevent->resp.id = ctx->id;
+               ctx->cm_id = uevent->cm_id;
        }
 
        if (copy_to_user(u64_to_user_ptr(cmd.response),
                         &uevent->resp,
                         min_t(size_t, out_len, sizeof(uevent->resp)))) {
                ret = -EFAULT;
-               goto done;
+               goto err_ctx;
+       }
+
+       if (ctx) {
+               uevent->ctx->backlog++;
+               uevent->cm_id->context = ctx;
+               ucma_finish_ctx(ctx);
        }
 
        list_del(&uevent->list);
        uevent->ctx->events_reported++;
        if (uevent->mc)
                uevent->mc->events_reported++;
+       mutex_unlock(&file->mut);
+
        kfree(uevent);
-done:
+       return 0;
+
+err_ctx:
+       if (ctx) {
+               xa_erase(&ctx_table, ctx->id);
+               kfree(ctx);
+       }
+err_unlock:
        mutex_unlock(&file->mut);
        return ret;
 }
        if (ret)
                return ret;
 
-       mutex_lock(&file->mut);
        ctx = ucma_alloc_ctx(file);
-       mutex_unlock(&file->mut);
        if (!ctx)
                return -ENOMEM;
 
                ret = PTR_ERR(cm_id);
                goto err1;
        }
+       ctx->cm_id = cm_id;
 
        resp.id = ctx->id;
        if (copy_to_user(u64_to_user_ptr(cmd.response),
                         &resp, sizeof(resp))) {
-               ret = -EFAULT;
-               goto err2;
+               xa_erase(&ctx_table, ctx->id);
+               __destroy_id(ctx);
+               return -EFAULT;
        }
 
-       ctx->cm_id = cm_id;
+       mutex_lock(&file->mut);
+       ucma_finish_ctx(ctx);
+       mutex_unlock(&file->mut);
        return 0;
 
-err2:
-       rdma_destroy_id(cm_id);
 err1:
        xa_erase(&ctx_table, ctx->id);
-       mutex_lock(&file->mut);
-       list_del(&ctx->list);
-       mutex_unlock(&file->mut);
        kfree(ctx);
        return ret;
 }