Whamcloud - gitweb
LU-13569 lnet: Age peer NI out of recovery
[fs/lustre-release.git] / lnet / lnet / lib-msg.c
index 0c469d4..94f2a78 100644 (file)
@@ -48,7 +48,7 @@ lnet_build_unlink_event(struct lnet_libmd *md, struct lnet_event *ev)
        ev->status   = 0;
        ev->unlinked = 1;
        ev->type     = LNET_EVENT_UNLINK;
-       lnet_md_deconstruct(md, &ev->md);
+       lnet_md_deconstruct(md, ev);
        lnet_md2handle(&ev->md_handle, md);
        EXIT;
 }
@@ -360,7 +360,7 @@ lnet_msg_attach_md(struct lnet_msg *msg, struct lnet_libmd *md,
 
        /* build umd in event */
        lnet_md2handle(&msg->msg_ev.md_handle, md);
-       lnet_md_deconstruct(md, &msg->msg_ev.md);
+       lnet_md_deconstruct(md, &msg->msg_ev);
 }
 
 static int
@@ -484,6 +484,7 @@ lnet_handle_local_failure(struct lnet_ni *local_ni)
        lnet_net_unlock(0);
 }
 
+/* must hold net_lock/0 */
 void
 lnet_handle_remote_failure_locked(struct lnet_peer_ni *lpni)
 {
@@ -491,14 +492,6 @@ lnet_handle_remote_failure_locked(struct lnet_peer_ni *lpni)
        __u32 lp_sensitivity;
 
        /*
-        * NO-OP if:
-        * 1. lpni could be NULL if we're in the LOLND case
-        * 2. this is a recovery message
-        */
-       if (!lpni)
-               return;
-
-       /*
         * If there is a health sensitivity in the peer then use that
         * instead of the globally set one.
         */
@@ -518,7 +511,9 @@ lnet_handle_remote_failure_locked(struct lnet_peer_ni *lpni)
         * value will not be reduced. In this case, there is no reason to
         * invoke recovery
         */
-       lnet_peer_ni_add_to_recoveryq_locked(lpni);
+       lnet_peer_ni_add_to_recoveryq_locked(lpni,
+                                            &the_lnet.ln_mt_peerNIRecovq,
+                                            ktime_get_seconds());
 }
 
 static void
@@ -539,10 +534,9 @@ lnet_handle_remote_failure(struct lnet_peer_ni *lpni)
 }
 
 static void
-lnet_incr_hstats(struct lnet_msg *msg, enum lnet_msg_hstatus hstatus)
+lnet_incr_hstats(struct lnet_ni *ni, struct lnet_peer_ni *lpni,
+                enum lnet_msg_hstatus hstatus)
 {
-       struct lnet_ni *ni = msg->msg_txni;
-       struct lnet_peer_ni *lpni = msg->msg_txpeer;
        struct lnet_counters_health *health;
 
        health = &the_lnet.ln_counters[0]->lct_health;
@@ -781,6 +775,10 @@ lnet_health_check(struct lnet_msg *msg)
        struct lnet_peer_ni *lpni;
        struct lnet_ni *ni;
        bool lo = false;
+       bool attempt_local_resend;
+       bool attempt_remote_resend;
+       bool handle_local_health;
+       bool handle_remote_health;
 
        /* if we're shutting down no point in handling health. */
        if (the_lnet.ln_mt_state != LNET_MT_STATE_RUNNING)
@@ -792,11 +790,10 @@ lnet_health_check(struct lnet_msg *msg)
         * if we're sending to the LOLND then the msg_txpeer will not be
         * set. So no need to sanity check it.
         */
-       if (msg->msg_tx_committed &&
-           LNET_NETTYP(LNET_NIDNET(msg->msg_txni->ni_nid)) != LOLND)
+       if (msg->msg_tx_committed && msg->msg_txni->ni_nid != LNET_NID_LO_0)
                LASSERT(msg->msg_txpeer);
        else if (msg->msg_tx_committed &&
-                LNET_NETTYP(LNET_NIDNET(msg->msg_txni->ni_nid)) == LOLND)
+                msg->msg_txni->ni_nid == LNET_NID_LO_0)
                lo = true;
 
        if (hstatus != LNET_MSG_STATUS_OK &&
@@ -804,25 +801,46 @@ lnet_health_check(struct lnet_msg *msg)
                return -1;
 
        /*
-        * stats are only incremented for errors so avoid wasting time
-        * incrementing statistics if there is no error.
-        */
-       if (hstatus != LNET_MSG_STATUS_OK) {
-               lnet_net_lock(0);
-               lnet_incr_hstats(msg, hstatus);
-               lnet_net_unlock(0);
-       }
-
-       /*
         * always prefer txni/txpeer if they message is committed for both
         * directions.
         */
        if (msg->msg_tx_committed) {
                ni = msg->msg_txni;
                lpni = msg->msg_txpeer;
+               attempt_local_resend = attempt_remote_resend = true;
        } else {
                ni = msg->msg_rxni;
                lpni = msg->msg_rxpeer;
+               attempt_local_resend = attempt_remote_resend = false;
+       }
+
+       /* Don't further decrement the health value if a recovery message
+        * failed.
+        */
+       if (msg->msg_recovery)
+               handle_local_health = handle_remote_health = false;
+       else
+               handle_local_health = handle_remote_health = true;
+
+       /* For local failures, health/recovery/resends are not needed if I only
+        * have a single (non-lolnd) interface. NB: pb_nnis includes the lolnd
+        * interface, so a single-rail node would have pb_nnis == 2.
+        */
+       if (the_lnet.ln_ping_target->pb_nnis <= 2) {
+               handle_local_health = false;
+               attempt_local_resend = false;
+       }
+
+       /* For remote failures, health/recovery/resends are not needed if the
+        * peer only has a single interface. Special case for routers where we
+        * rely on health feature to manage route aliveness. NB: unlike pb_nnis
+        * above, lp_nnis does _not_ include the lolnd, so a single-rail node
+        * would have lp_nnis == 1.
+        */
+       if (lpni && lpni->lpni_peer_net->lpn_peer->lp_nnis <= 1) {
+               attempt_remote_resend = false;
+               if (!lnet_isrouter(lpni))
+                       handle_remote_health = false;
        }
 
        if (!lo)
@@ -836,6 +854,16 @@ lnet_health_check(struct lnet_msg *msg)
               lnet_msgtyp2str(msg->msg_type),
               lnet_health_error2str(hstatus));
 
+       /*
+        * stats are only incremented for errors so avoid wasting time
+        * incrementing statistics if there is no error.
+        */
+       if (hstatus != LNET_MSG_STATUS_OK) {
+               lnet_net_lock(0);
+               lnet_incr_hstats(ni, lpni, hstatus);
+               lnet_net_unlock(0);
+       }
+
        switch (hstatus) {
        case LNET_MSG_STATUS_OK:
                /*
@@ -868,6 +896,15 @@ lnet_health_check(struct lnet_msg *msg)
                                lnet_inc_lpni_healthv_locked(lpni,
                                        (sensitivity) ? sensitivity :
                                        lnet_health_sensitivity);
+                               /* This peer NI may have previously aged out
+                                * of recovery. Now that we've received a
+                                * message from it, we can continue recovery
+                                * if its health value is still below the
+                                * maximum.
+                                */
+                               lnet_peer_ni_add_to_recoveryq_locked(lpni,
+                                               &the_lnet.ln_mt_peerNIRecovq,
+                                               ktime_get_seconds());
                        }
                        lnet_net_unlock(0);
                }
@@ -879,46 +916,31 @@ lnet_health_check(struct lnet_msg *msg)
        case LNET_MSG_STATUS_LOCAL_ABORTED:
        case LNET_MSG_STATUS_LOCAL_NO_ROUTE:
        case LNET_MSG_STATUS_LOCAL_TIMEOUT:
-               /*
-                * don't further decrement the health value if the
-                * recovery message failed.
-                */
-               if (!msg->msg_recovery)
+               if (handle_local_health)
                        lnet_handle_local_failure(ni);
-               if (msg->msg_tx_committed)
-                       /* add to the re-send queue */
+               if (attempt_local_resend)
                        return lnet_attempt_msg_resend(msg);
                break;
-
-       /*
-        * These errors will not trigger a resend so simply
-        * finalize the message
-        */
        case LNET_MSG_STATUS_LOCAL_ERROR:
-               /*
-                * don't further decrement the health value if the
-                * recovery message failed.
-                */
-               if (!msg->msg_recovery)
+               if (handle_local_health)
                        lnet_handle_local_failure(ni);
                return -1;
-
-       /*
-        * TODO: since the remote dropped the message we can
-        * attempt a resend safely.
-        */
        case LNET_MSG_STATUS_REMOTE_DROPPED:
-               if (!msg->msg_recovery)
+               if (handle_remote_health)
                        lnet_handle_remote_failure(lpni);
-               if (msg->msg_tx_committed)
+               if (attempt_remote_resend)
                        return lnet_attempt_msg_resend(msg);
                break;
-
        case LNET_MSG_STATUS_REMOTE_ERROR:
        case LNET_MSG_STATUS_REMOTE_TIMEOUT:
+               if (handle_remote_health)
+                       lnet_handle_remote_failure(lpni);
+               return -1;
        case LNET_MSG_STATUS_NETWORK_TIMEOUT:
-               if (!msg->msg_recovery)
+               if (handle_remote_health)
                        lnet_handle_remote_failure(lpni);
+               if (handle_local_health)
+                       lnet_handle_local_failure(ni);
                return -1;
        default:
                LBUG();
@@ -929,11 +951,20 @@ lnet_health_check(struct lnet_msg *msg)
 }
 
 static void
-lnet_msg_detach_md(struct lnet_msg *msg, int cpt, int status)
+lnet_msg_detach_md(struct lnet_msg *msg, int status)
 {
        struct lnet_libmd *md = msg->msg_md;
+       lnet_handler_t handler = NULL;
+       int cpt = lnet_cpt_of_cookie(md->md_lh.lh_cookie);
        int unlink;
 
+       lnet_res_lock(cpt);
+       while (md->md_flags & LNET_MD_FLAG_HANDLING)
+               /* An event handler is running - wait for it to
+                * complete to avoid races.
+                */
+               lnet_md_wait_handling(md, cpt);
+
        /* Now it's safe to drop my caller's ref */
        md->md_refcount--;
        LASSERT(md->md_refcount >= 0);
@@ -947,17 +978,30 @@ lnet_msg_detach_md(struct lnet_msg *msg, int cpt, int status)
                        msg->msg_ev.status   = status;
                }
                msg->msg_ev.unlinked = unlink;
-               md->md_handler(&msg->msg_ev);
+               handler = md->md_handler;
+               if (!unlink)
+                       md->md_flags |= LNET_MD_FLAG_HANDLING;
        }
 
        if (unlink || (md->md_refcount == 0 &&
                       md->md_threshold == LNET_MD_THRESH_INF))
                lnet_detach_rsp_tracker(md, cpt);
 
+       msg->msg_md = NULL;
        if (unlink)
                lnet_md_unlink(md);
 
-       msg->msg_md = NULL;
+       lnet_res_unlock(cpt);
+
+       if (handler) {
+               handler(&msg->msg_ev);
+               if (!unlink) {
+                       lnet_res_lock(cpt);
+                       md->md_flags &= ~LNET_MD_FLAG_HANDLING;
+                       wake_up_var(md);
+                       lnet_res_unlock(cpt);
+               }
+       }
 }
 
 static bool
@@ -1093,12 +1137,8 @@ lnet_finalize(struct lnet_msg *msg, int status)
         * We're not going to resend this message so detach its MD and invoke
         * the appropriate callbacks
         */
-       if (msg->msg_md != NULL) {
-               cpt = lnet_cpt_of_cookie(msg->msg_md->md_lh.lh_cookie);
-               lnet_res_lock(cpt);
-               lnet_msg_detach_md(msg, cpt, status);
-               lnet_res_unlock(cpt);
-       }
+       if (msg->msg_md != NULL)
+               lnet_msg_detach_md(msg, status);
 
 again:
        if (!msg->msg_tx_committed && !msg->msg_rx_committed) {