staging/rdma/hfi1: Put QPs into error state after SL->SC table changes
authorKaike Wan <kaike.wan@intel.com>
Sun, 14 Feb 2016 20:10:20 +0000 (12:10 -0800)
committerDoug Ledford <dledford@redhat.com>
Fri, 11 Mar 2016 01:38:08 +0000 (20:38 -0500)
If an SL->SC mapping table change occurs after an RC/UC QP is created,
there is no mechanism to change the SC nor the VL for that QP. The fix
is to place the QP into error state so that ULP can recreate the QP with
the new SL->SC mapping.

Reviewed-by: Ira Weiny <ira.weiny@intel.com>
Reviewed-by: Mike Marciniszyn <mike.marciniszyn@intel.com>
Signed-off-by: Kaike Wan <kaike.wan@intel.com>
Signed-off-by: Doug Ledford <dledford@redhat.com>
drivers/staging/rdma/hfi1/mad.c
drivers/staging/rdma/hfi1/qp.c
drivers/staging/rdma/hfi1/qp.h

index 2fcc9f3290d7a712e028222ccbbc50eaf301d28c..d9efe223328ba884561782269c1f178ae2874356 100644 (file)
@@ -55,6 +55,7 @@
 #include "hfi.h"
 #include "mad.h"
 #include "trace.h"
+#include "qp.h"
 
 /* the reset value from the FM is supposed to be 0xffff, handle both */
 #define OPA_LINK_WIDTH_RESET_OLD 0x0fff
@@ -1517,14 +1518,22 @@ static int __subn_set_opa_sl_to_sc(struct opa_smp *smp, u32 am, u8 *data,
        struct hfi1_ibport *ibp = to_iport(ibdev, port);
        u8 *p = data;
        int i;
+       u8 sc;
 
        if (am) {
                smp->status |= IB_SMP_INVALID_FIELD;
                return reply((struct ib_mad_hdr *)smp);
        }
 
-       for (i = 0; i <  ARRAY_SIZE(ibp->sl_to_sc); i++)
-               ibp->sl_to_sc[i] = *p++;
+       for (i = 0; i <  ARRAY_SIZE(ibp->sl_to_sc); i++) {
+               sc = *p++;
+               if (ibp->sl_to_sc[i] != sc) {
+                       ibp->sl_to_sc[i] = sc;
+
+                       /* Put all stale qps into error state */
+                       hfi1_error_port_qps(ibp, i);
+               }
+       }
 
        return __subn_get_opa_sl_to_sc(smp, am, data, ibdev, port, resp_len);
 }
index 00866c07fddcf888279833b3fb28cbde3725ed8d..9e0531434effd6e0fa727e034476b4bac3e4d669 100644 (file)
@@ -840,3 +840,55 @@ void notify_error_qp(struct rvt_qp *qp)
        }
 }
 
+/**
+ * hfi1_error_port_qps - put a port's RC/UC qps into error state
+ * @ibp: the ibport.
+ * @sl: the service level.
+ *
+ * This function places all RC/UC qps with a given service level into error
+ * state. It is generally called to force upper lay apps to abandon stale qps
+ * after an sl->sc mapping change.
+ */
+void hfi1_error_port_qps(struct hfi1_ibport *ibp, u8 sl)
+{
+       struct rvt_qp *qp = NULL;
+       struct hfi1_pportdata *ppd = ppd_from_ibp(ibp);
+       struct hfi1_ibdev *dev = &ppd->dd->verbs_dev;
+       int n;
+       int lastwqe;
+       struct ib_event ev;
+
+       rcu_read_lock();
+
+       /* Deal only with RC/UC qps that use the given SL. */
+       for (n = 0; n < dev->rdi.qp_dev->qp_table_size; n++) {
+               for (qp = rcu_dereference(dev->rdi.qp_dev->qp_table[n]); qp;
+                       qp = rcu_dereference(qp->next)) {
+                       if (qp->port_num == ppd->port &&
+                           (qp->ibqp.qp_type == IB_QPT_UC ||
+                            qp->ibqp.qp_type == IB_QPT_RC) &&
+                           qp->remote_ah_attr.sl == sl &&
+                           (ib_rvt_state_ops[qp->state] &
+                            RVT_POST_SEND_OK)) {
+                               spin_lock_irq(&qp->r_lock);
+                               spin_lock(&qp->s_hlock);
+                               spin_lock(&qp->s_lock);
+                               lastwqe = rvt_error_qp(qp,
+                                                      IB_WC_WR_FLUSH_ERR);
+                               spin_unlock(&qp->s_lock);
+                               spin_unlock(&qp->s_hlock);
+                               spin_unlock_irq(&qp->r_lock);
+                               if (lastwqe) {
+                                       ev.device = qp->ibqp.device;
+                                       ev.element.qp = &qp->ibqp;
+                                       ev.event =
+                                               IB_EVENT_QP_LAST_WQE_REACHED;
+                                       qp->ibqp.event_handler(&ev,
+                                               qp->ibqp.qp_context);
+                               }
+                       }
+               }
+       }
+
+       rcu_read_unlock();
+}
index 98827b5dd2a1e2bd181ae9d0027d9dff3609fa4a..afc2b4d242b7e8127998353a53ba4f4a8f97da0f 100644 (file)
@@ -158,4 +158,5 @@ void stop_send_queue(struct rvt_qp *qp);
 void quiesce_qp(struct rvt_qp *qp);
 u32 mtu_from_qp(struct rvt_dev_info *rdi, struct rvt_qp *qp, u32 pmtu);
 int mtu_to_path_mtu(u32 mtu);
+void hfi1_error_port_qps(struct hfi1_ibport *ibp, u8 sl);
 #endif /* _QP_H */