drbd: skip spurious timeout (ping-timeo) when failing promote
authorLars Ellenberg <lars.ellenberg@linbit.com>
Thu, 20 Dec 2018 16:23:41 +0000 (17:23 +0100)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Tue, 12 Feb 2019 18:47:14 +0000 (19:47 +0100)
[ Upstream commit 9848b6ddd8c92305252f94592c5e278574e7a6ac ]

If you try to promote a Secondary while connected to a Primary
and allow-two-primaries is NOT set, we will wait for "ping-timeout"
to give this node a chance to detect a dead primary,
in case the cluster manager noticed faster than we did.

But if we then are *still* connected to a Primary,
we fail (after an additional timeout of ping-timout).

This change skips the spurious second timeout.

Most people won't notice really,
since "ping-timeout" by default is half a second.

But in some installations, ping-timeout may be 10 or 20 seconds or more,
and spuriously delaying the error return becomes annoying.

Signed-off-by: Lars Ellenberg <lars.ellenberg@linbit.com>
Signed-off-by: Jens Axboe <axboe@kernel.dk>
Signed-off-by: Sasha Levin <sashal@kernel.org>
drivers/block/drbd/drbd_nl.c

index b4f02768ba475c13620582e44332b32ac9a65b48..319fabdd63a3f4b453904e729a70d49634afe0d5 100644 (file)
@@ -668,14 +668,15 @@ drbd_set_role(struct drbd_device *const device, enum drbd_role new_role, int for
                if (rv == SS_TWO_PRIMARIES) {
                        /* Maybe the peer is detected as dead very soon...
                           retry at most once more in this case. */
-                       int timeo;
-                       rcu_read_lock();
-                       nc = rcu_dereference(connection->net_conf);
-                       timeo = nc ? (nc->ping_timeo + 1) * HZ / 10 : 1;
-                       rcu_read_unlock();
-                       schedule_timeout_interruptible(timeo);
-                       if (try < max_tries)
+                       if (try < max_tries) {
+                               int timeo;
                                try = max_tries - 1;
+                               rcu_read_lock();
+                               nc = rcu_dereference(connection->net_conf);
+                               timeo = nc ? (nc->ping_timeo + 1) * HZ / 10 : 1;
+                               rcu_read_unlock();
+                               schedule_timeout_interruptible(timeo);
+                       }
                        continue;
                }
                if (rv < SS_SUCCESS) {