}
 }
 
+/*
+ * Reset background worker crash state.
+ *
+ * We assume that, after a crash-and-restart cycle, background workers should
+ * be restarted immediately, instead of waiting for bgw_restart_time to
+ * elapse.
+ */
+void
+ResetBackgroundWorkerCrashTimes(void)
+{
+   slist_mutable_iter  iter;
+
+   slist_foreach_modify(iter, &BackgroundWorkerList)
+   {
+       RegisteredBgWorker *rw;
+
+       rw = slist_container(RegisteredBgWorker, rw_lnode, iter.cur);
+       rw->rw_crashed_at = 0;
+   }
+}
+
 #ifdef EXEC_BACKEND
 /*
  * In EXEC_BACKEND mode, workers use this to retrieve their details from
    on_exit_reset();
 
    /*
-    * Note we do exit(0) here, not exit(2) like quickdie.  The reason is that
-    * we don't want to be seen this worker as independently crashed, because
-    * then postmaster would delay restarting it again afterwards.  If some
-    * idiot DBA manually sends SIGQUIT to a random bgworker, the "dead man
-    * switch" will ensure that postmaster sees this as a crash.
+    * Note we do exit(2) not exit(0).  This is to force the postmaster into a
+    * system reset cycle if some idiot DBA sends a manual SIGQUIT to a random
+    * backend.  This is necessary precisely because we don't clean up our
+    * shared memory state.  (The "dead man switch" mechanism in pmsignal.c
+    * should ensure the postmaster sees this as a crash, too, but no harm in
+    * being doubly sure.)
     */
-   exit(0);
+   exit(2);
 }
 
 /*
 
            if (PgStatPID == 0)
                PgStatPID = pgstat_start();
 
-           /* some workers may be scheduled to start now */
+           /* workers may be scheduled to start now */
            maybe_start_bgworker();
 
            /* at this point we are really open for business */
        {
            if (!EXIT_STATUS_0(exitstatus) && !EXIT_STATUS_1(exitstatus))
            {
-               rw->rw_crashed_at = GetCurrentTimestamp();
                HandleChildCrash(pid, exitstatus, namebuf);
                return true;
            }
                 * Uh-oh, the child failed to clean itself up.  Treat as a
                 * crash after all.
                 */
-               rw->rw_crashed_at = GetCurrentTimestamp();
                HandleChildCrash(pid, exitstatus, namebuf);
                return true;
            }
        ereport(LOG,
                (errmsg("all server processes terminated; reinitializing")));
 
+       /* allow background workers to immediately restart */
+       ResetBackgroundWorkerCrashTimes();
+
        shmem_exit(1);
        reset_shared(PostPortNumber);
 
 
 extern void ForgetBackgroundWorker(slist_mutable_iter *cur);
 extern void ReportBackgroundWorkerPID(RegisteredBgWorker *);
 extern void BackgroundWorkerStopNotifications(pid_t pid);
+extern void ResetBackgroundWorkerCrashTimes(void);
 
 /* Function to start a background worker, called from postmaster.c */
 extern void StartBackgroundWorker(void);