/* rxe_qp.c */
 int rxe_qp_chk_init(struct rxe_dev *rxe, struct ib_qp_init_attr *init);
-
 int rxe_qp_from_init(struct rxe_dev *rxe, struct rxe_qp *qp, struct rxe_pd *pd,
                     struct ib_qp_init_attr *init,
                     struct rxe_create_qp_resp __user *uresp,
                     struct ib_pd *ibpd, struct ib_udata *udata);
-
 int rxe_qp_to_init(struct rxe_qp *qp, struct ib_qp_init_attr *init);
-
 int rxe_qp_chk_attr(struct rxe_dev *rxe, struct rxe_qp *qp,
                    struct ib_qp_attr *attr, int mask);
-
 int rxe_qp_from_attr(struct rxe_qp *qp, struct ib_qp_attr *attr,
                     int mask, struct ib_udata *udata);
-
 int rxe_qp_to_attr(struct rxe_qp *qp, struct ib_qp_attr *attr, int mask);
-
 void rxe_qp_error(struct rxe_qp *qp);
-
+int rxe_qp_chk_destroy(struct rxe_qp *qp);
 void rxe_qp_destroy(struct rxe_qp *qp);
-
 void rxe_qp_cleanup(struct rxe_pool_elem *elem);
 
 static inline int qp_num(struct rxe_qp *qp)
 
        grp->num_qp++;
        elem->qp = qp;
        elem->grp = grp;
+       atomic_inc(&qp->mcg_num);
 
        list_add(&elem->qp_list, &grp->qp_list);
        list_add(&elem->grp_list, &qp->grp_list);
                        list_del(&elem->qp_list);
                        list_del(&elem->grp_list);
                        grp->num_qp--;
+                       atomic_dec(&qp->mcg_num);
 
                        spin_unlock_bh(&grp->mcg_lock);
                        spin_unlock_bh(&qp->grp_lock);
 
        return 0;
 }
 
+int rxe_qp_chk_destroy(struct rxe_qp *qp)
+{
+       /* See IBA o10-2.2.3
+        * An attempt to destroy a QP while attached to a mcast group
+        * will fail immediately.
+        */
+       if (atomic_read(&qp->mcg_num)) {
+               pr_debug("Attempt to destroy QP while attached to multicast group\n");
+               return -EBUSY;
+       }
+
+       return 0;
+}
+
 /* called by the destroy qp verb */
 void rxe_qp_destroy(struct rxe_qp *qp)
 {
 
        /* list of mcast groups qp has joined (for cleanup) */
        struct list_head        grp_list;
        spinlock_t              grp_lock; /* guard grp_list */
+       atomic_t                mcg_num;
 
        struct sk_buff_head     req_pkts;
        struct sk_buff_head     resp_pkts;