]> granicus.if.org Git - postgresql/commitdiff
Make pg_promote() detect postmaster death while waiting for promotion to end.
authorFujii Masao <fujii@postgresql.org>
Fri, 6 Sep 2019 05:27:25 +0000 (14:27 +0900)
committerFujii Masao <fujii@postgresql.org>
Fri, 6 Sep 2019 05:27:25 +0000 (14:27 +0900)
Previously even if postmaster died and WaitLatch() woke up with that event
while pg_promote() was waiting for the standby promotion to finish,
pg_promote() did nothing special and kept waiting until timeout occurred.
This could cause a busy loop.

This patch make pg_promote() return false immediately when postmaster
dies, to avoid such a busy loop.

Back-patch to v12 where pg_promote() was added.

Author: Fujii Masao
Reviewed-by: Michael Paquier
Discussion: https://postgr.es/m/CAHGQGwEs9ROgSp+QF+YdDU+xP8W=CY1k-_Ov-d_Z3JY+to3eXA@mail.gmail.com

src/backend/access/transam/xlogfuncs.c

index 4795c6fa947a8a379516e8fe8a4cc8aecd975dd3..8a70503228dd8a3808f0fe275ee508ac13219109 100644 (file)
@@ -759,6 +759,8 @@ pg_promote(PG_FUNCTION_ARGS)
 #define WAITS_PER_SECOND 10
        for (i = 0; i < WAITS_PER_SECOND * wait_seconds; i++)
        {
+               int                     rc;
+
                ResetLatch(MyLatch);
 
                if (!RecoveryInProgress())
@@ -766,10 +768,17 @@ pg_promote(PG_FUNCTION_ARGS)
 
                CHECK_FOR_INTERRUPTS();
 
-               (void) WaitLatch(MyLatch,
-                                                WL_LATCH_SET | WL_TIMEOUT | WL_POSTMASTER_DEATH,
-                                                1000L / WAITS_PER_SECOND,
-                                                WAIT_EVENT_PROMOTE);
+               rc = WaitLatch(MyLatch,
+                                          WL_LATCH_SET | WL_TIMEOUT | WL_POSTMASTER_DEATH,
+                                          1000L / WAITS_PER_SECOND,
+                                          WAIT_EVENT_PROMOTE);
+
+               /*
+                * Emergency bailout if postmaster has died.  This is to avoid the
+                * necessity for manual cleanup of all postmaster children.
+                */
+               if (rc & WL_POSTMASTER_DEATH)
+                       PG_RETURN_BOOL(false);
        }
 
        ereport(WARNING,