mirror of
https://github.com/postgres/postgres.git
synced 2025-07-27 12:41:57 +03:00
When the checkpointer writes the shutdown checkpoint, it checks afterwards whether any WAL has been written since it started and throws a PANIC if so. At that point, only walsenders are still active, so one might think this could not happen, but walsenders can also generate WAL, for instance in BASE_BACKUP and logical decoding related commands (e.g. via hint bits). So they can trigger this panic if such a command is run while the shutdown checkpoint is being written. To fix this, divide the walsender shutdown into two phases. First, checkpointer, itself triggered by postmaster, sends a PROCSIG_WALSND_INIT_STOPPING signal to all walsenders. If the backend is idle or runs an SQL query this causes the backend to shutdown, if logical replication is in progress all existing WAL records are processed followed by a shutdown. Otherwise this causes the walsender to switch to the "stopping" state. In this state, the walsender will reject any further replication commands. The checkpointer begins the shutdown checkpoint once all walsenders are confirmed as stopping. When the shutdown checkpoint finishes, the postmaster sends us SIGUSR2. This instructs walsender to send any outstanding WAL, including the shutdown checkpoint record, wait for it to be replicated to the standby, and then exit. Author: Andres Freund, based on an earlier patch by Michael Paquier Reported-By: Fujii Masao, Andres Freund Reviewed-By: Michael Paquier Discussion: https://postgr.es/m/20170602002912.tqlwn4gymzlxpvs2@alap3.anarazel.de Backpatch: 9.4, where logical decoding was introduced
61 lines
1.9 KiB
C
61 lines
1.9 KiB
C
/*-------------------------------------------------------------------------
|
|
*
|
|
* procsignal.h
|
|
* Routines for interprocess signalling
|
|
*
|
|
*
|
|
* Portions Copyright (c) 1996-2014, PostgreSQL Global Development Group
|
|
* Portions Copyright (c) 1994, Regents of the University of California
|
|
*
|
|
* src/include/storage/procsignal.h
|
|
*
|
|
*-------------------------------------------------------------------------
|
|
*/
|
|
#ifndef PROCSIGNAL_H
|
|
#define PROCSIGNAL_H
|
|
|
|
#include "storage/backendid.h"
|
|
|
|
|
|
/*
|
|
* Reasons for signalling a Postgres child process (a backend or an auxiliary
|
|
* process, like checkpointer). We can cope with concurrent signals for different
|
|
* reasons. However, if the same reason is signaled multiple times in quick
|
|
* succession, the process is likely to observe only one notification of it.
|
|
* This is okay for the present uses.
|
|
*
|
|
* Also, because of race conditions, it's important that all the signals be
|
|
* defined so that no harm is done if a process mistakenly receives one.
|
|
*/
|
|
typedef enum
|
|
{
|
|
PROCSIG_CATCHUP_INTERRUPT, /* sinval catchup interrupt */
|
|
PROCSIG_NOTIFY_INTERRUPT, /* listen/notify interrupt */
|
|
PROCSIG_WALSND_INIT_STOPPING, /* ask walsenders to prepare for shutdown */
|
|
|
|
/* Recovery conflict reasons */
|
|
PROCSIG_RECOVERY_CONFLICT_DATABASE,
|
|
PROCSIG_RECOVERY_CONFLICT_TABLESPACE,
|
|
PROCSIG_RECOVERY_CONFLICT_LOCK,
|
|
PROCSIG_RECOVERY_CONFLICT_SNAPSHOT,
|
|
PROCSIG_RECOVERY_CONFLICT_BUFFERPIN,
|
|
PROCSIG_RECOVERY_CONFLICT_STARTUP_DEADLOCK,
|
|
|
|
NUM_PROCSIGNALS /* Must be last! */
|
|
} ProcSignalReason;
|
|
|
|
/*
|
|
* prototypes for functions in procsignal.c
|
|
*/
|
|
extern Size ProcSignalShmemSize(void);
|
|
extern void ProcSignalShmemInit(void);
|
|
|
|
extern void ProcSignalInit(int pss_idx);
|
|
extern int SendProcSignal(pid_t pid, ProcSignalReason reason,
|
|
BackendId backendId);
|
|
|
|
extern void procsignal_sigusr1_handler(SIGNAL_ARGS);
|
|
extern PGDLLIMPORT bool set_latch_on_sigusr1;
|
|
|
|
#endif /* PROCSIGNAL_H */
|