LCOV - code coverage report
Current view: top level - src/backend/replication - slot.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 748 872 85.8 %
Date: 2025-01-18 04:15:08 Functions: 41 42 97.6 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * slot.c
       4             :  *     Replication slot management.
       5             :  *
       6             :  *
       7             :  * Copyright (c) 2012-2025, PostgreSQL Global Development Group
       8             :  *
       9             :  *
      10             :  * IDENTIFICATION
      11             :  *    src/backend/replication/slot.c
      12             :  *
      13             :  * NOTES
      14             :  *
      15             :  * Replication slots are used to keep state about replication streams
      16             :  * originating from this cluster.  Their primary purpose is to prevent the
      17             :  * premature removal of WAL or of old tuple versions in a manner that would
      18             :  * interfere with replication; they are also useful for monitoring purposes.
      19             :  * Slots need to be permanent (to allow restarts), crash-safe, and allocatable
      20             :  * on standbys (to support cascading setups).  The requirement that slots be
      21             :  * usable on standbys precludes storing them in the system catalogs.
      22             :  *
      23             :  * Each replication slot gets its own directory inside the directory
      24             :  * $PGDATA / PG_REPLSLOT_DIR.  Inside that directory the state file will
      25             :  * contain the slot's own data.  Additional data can be stored alongside that
      26             :  * file if required.  While the server is running, the state data is also
      27             :  * cached in memory for efficiency.
      28             :  *
      29             :  * ReplicationSlotAllocationLock must be taken in exclusive mode to allocate
      30             :  * or free a slot. ReplicationSlotControlLock must be taken in shared mode
      31             :  * to iterate over the slots, and in exclusive mode to change the in_use flag
      32             :  * of a slot.  The remaining data in each slot is protected by its mutex.
      33             :  *
      34             :  *-------------------------------------------------------------------------
      35             :  */
      36             : 
      37             : #include "postgres.h"
      38             : 
      39             : #include <unistd.h>
      40             : #include <sys/stat.h>
      41             : 
      42             : #include "access/transam.h"
      43             : #include "access/xlog_internal.h"
      44             : #include "access/xlogrecovery.h"
      45             : #include "common/file_utils.h"
      46             : #include "common/string.h"
      47             : #include "miscadmin.h"
      48             : #include "pgstat.h"
      49             : #include "postmaster/interrupt.h"
      50             : #include "replication/slotsync.h"
      51             : #include "replication/slot.h"
      52             : #include "replication/walsender_private.h"
      53             : #include "storage/fd.h"
      54             : #include "storage/ipc.h"
      55             : #include "storage/proc.h"
      56             : #include "storage/procarray.h"
      57             : #include "utils/builtins.h"
      58             : #include "utils/guc_hooks.h"
      59             : #include "utils/varlena.h"
      60             : 
      61             : /*
      62             :  * Replication slot on-disk data structure.
      63             :  */
      64             : typedef struct ReplicationSlotOnDisk
      65             : {
      66             :     /* first part of this struct needs to be version independent */
      67             : 
      68             :     /* data not covered by checksum */
      69             :     uint32      magic;
      70             :     pg_crc32c   checksum;
      71             : 
      72             :     /* data covered by checksum */
      73             :     uint32      version;
      74             :     uint32      length;
      75             : 
      76             :     /*
      77             :      * The actual data in the slot that follows can differ based on the above
      78             :      * 'version'.
      79             :      */
      80             : 
      81             :     ReplicationSlotPersistentData slotdata;
      82             : } ReplicationSlotOnDisk;
      83             : 
      84             : /*
      85             :  * Struct for the configuration of synchronized_standby_slots.
      86             :  *
      87             :  * Note: this must be a flat representation that can be held in a single chunk
      88             :  * of guc_malloc'd memory, so that it can be stored as the "extra" data for the
      89             :  * synchronized_standby_slots GUC.
      90             :  */
      91             : typedef struct
      92             : {
      93             :     /* Number of slot names in the slot_names[] */
      94             :     int         nslotnames;
      95             : 
      96             :     /*
      97             :      * slot_names contains 'nslotnames' consecutive null-terminated C strings.
      98             :      */
      99             :     char        slot_names[FLEXIBLE_ARRAY_MEMBER];
     100             : } SyncStandbySlotsConfigData;
     101             : 
     102             : /*
     103             :  * Lookup table for slot invalidation causes.
     104             :  */
     105             : const char *const SlotInvalidationCauses[] = {
     106             :     [RS_INVAL_NONE] = "none",
     107             :     [RS_INVAL_WAL_REMOVED] = "wal_removed",
     108             :     [RS_INVAL_HORIZON] = "rows_removed",
     109             :     [RS_INVAL_WAL_LEVEL] = "wal_level_insufficient",
     110             : };
     111             : 
     112             : /* Maximum number of invalidation causes */
     113             : #define RS_INVAL_MAX_CAUSES RS_INVAL_WAL_LEVEL
     114             : 
     115             : StaticAssertDecl(lengthof(SlotInvalidationCauses) == (RS_INVAL_MAX_CAUSES + 1),
     116             :                  "array length mismatch");
     117             : 
     118             : /* size of version independent data */
     119             : #define ReplicationSlotOnDiskConstantSize \
     120             :     offsetof(ReplicationSlotOnDisk, slotdata)
     121             : /* size of the part of the slot not covered by the checksum */
     122             : #define ReplicationSlotOnDiskNotChecksummedSize  \
     123             :     offsetof(ReplicationSlotOnDisk, version)
     124             : /* size of the part covered by the checksum */
     125             : #define ReplicationSlotOnDiskChecksummedSize \
     126             :     sizeof(ReplicationSlotOnDisk) - ReplicationSlotOnDiskNotChecksummedSize
     127             : /* size of the slot data that is version dependent */
     128             : #define ReplicationSlotOnDiskV2Size \
     129             :     sizeof(ReplicationSlotOnDisk) - ReplicationSlotOnDiskConstantSize
     130             : 
     131             : #define SLOT_MAGIC      0x1051CA1   /* format identifier */
     132             : #define SLOT_VERSION    5       /* version for new files */
     133             : 
     134             : /* Control array for replication slot management */
     135             : ReplicationSlotCtlData *ReplicationSlotCtl = NULL;
     136             : 
     137             : /* My backend's replication slot in the shared memory array */
     138             : ReplicationSlot *MyReplicationSlot = NULL;
     139             : 
     140             : /* GUC variables */
     141             : int         max_replication_slots = 10; /* the maximum number of replication
     142             :                                          * slots */
     143             : 
     144             : /*
     145             :  * This GUC lists streaming replication standby server slot names that
     146             :  * logical WAL sender processes will wait for.
     147             :  */
     148             : char       *synchronized_standby_slots;
     149             : 
     150             : /* This is the parsed and cached configuration for synchronized_standby_slots */
     151             : static SyncStandbySlotsConfigData *synchronized_standby_slots_config;
     152             : 
     153             : /*
     154             :  * Oldest LSN that has been confirmed to be flushed to the standbys
     155             :  * corresponding to the physical slots specified in the synchronized_standby_slots GUC.
     156             :  */
     157             : static XLogRecPtr ss_oldest_flush_lsn = InvalidXLogRecPtr;
     158             : 
     159             : static void ReplicationSlotShmemExit(int code, Datum arg);
     160             : static void ReplicationSlotDropPtr(ReplicationSlot *slot);
     161             : 
     162             : /* internal persistency functions */
     163             : static void RestoreSlotFromDisk(const char *name);
     164             : static void CreateSlotOnDisk(ReplicationSlot *slot);
     165             : static void SaveSlotToPath(ReplicationSlot *slot, const char *dir, int elevel);
     166             : 
     167             : /*
     168             :  * Report shared-memory space needed by ReplicationSlotsShmemInit.
     169             :  */
     170             : Size
     171        7398 : ReplicationSlotsShmemSize(void)
     172             : {
     173        7398 :     Size        size = 0;
     174             : 
     175        7398 :     if (max_replication_slots == 0)
     176           4 :         return size;
     177             : 
     178        7394 :     size = offsetof(ReplicationSlotCtlData, replication_slots);
     179        7394 :     size = add_size(size,
     180             :                     mul_size(max_replication_slots, sizeof(ReplicationSlot)));
     181             : 
     182        7394 :     return size;
     183             : }
     184             : 
     185             : /*
     186             :  * Allocate and initialize shared memory for replication slots.
     187             :  */
     188             : void
     189        1918 : ReplicationSlotsShmemInit(void)
     190             : {
     191             :     bool        found;
     192             : 
     193        1918 :     if (max_replication_slots == 0)
     194           2 :         return;
     195             : 
     196        1916 :     ReplicationSlotCtl = (ReplicationSlotCtlData *)
     197        1916 :         ShmemInitStruct("ReplicationSlot Ctl", ReplicationSlotsShmemSize(),
     198             :                         &found);
     199             : 
     200        1916 :     if (!found)
     201             :     {
     202             :         int         i;
     203             : 
     204             :         /* First time through, so initialize */
     205        3736 :         MemSet(ReplicationSlotCtl, 0, ReplicationSlotsShmemSize());
     206             : 
     207       20684 :         for (i = 0; i < max_replication_slots; i++)
     208             :         {
     209       18768 :             ReplicationSlot *slot = &ReplicationSlotCtl->replication_slots[i];
     210             : 
     211             :             /* everything else is zeroed by the memset above */
     212       18768 :             SpinLockInit(&slot->mutex);
     213       18768 :             LWLockInitialize(&slot->io_in_progress_lock,
     214             :                              LWTRANCHE_REPLICATION_SLOT_IO);
     215       18768 :             ConditionVariableInit(&slot->active_cv);
     216             :         }
     217             :     }
     218             : }
     219             : 
     220             : /*
     221             :  * Register the callback for replication slot cleanup and releasing.
     222             :  */
     223             : void
     224       34686 : ReplicationSlotInitialize(void)
     225             : {
     226       34686 :     before_shmem_exit(ReplicationSlotShmemExit, 0);
     227       34686 : }
     228             : 
     229             : /*
     230             :  * Release and cleanup replication slots.
     231             :  */
     232             : static void
     233       34686 : ReplicationSlotShmemExit(int code, Datum arg)
     234             : {
     235             :     /* Make sure active replication slots are released */
     236       34686 :     if (MyReplicationSlot != NULL)
     237         422 :         ReplicationSlotRelease();
     238             : 
     239             :     /* Also cleanup all the temporary slots. */
     240       34686 :     ReplicationSlotCleanup(false);
     241       34686 : }
     242             : 
     243             : /*
     244             :  * Check whether the passed slot name is valid and report errors at elevel.
     245             :  *
     246             :  * Slot names may consist out of [a-z0-9_]{1,NAMEDATALEN-1} which should allow
     247             :  * the name to be used as a directory name on every supported OS.
     248             :  *
     249             :  * Returns whether the directory name is valid or not if elevel < ERROR.
     250             :  */
     251             : bool
     252        1670 : ReplicationSlotValidateName(const char *name, int elevel)
     253             : {
     254             :     const char *cp;
     255             : 
     256        1670 :     if (strlen(name) == 0)
     257             :     {
     258           6 :         ereport(elevel,
     259             :                 (errcode(ERRCODE_INVALID_NAME),
     260             :                  errmsg("replication slot name \"%s\" is too short",
     261             :                         name)));
     262           0 :         return false;
     263             :     }
     264             : 
     265        1664 :     if (strlen(name) >= NAMEDATALEN)
     266             :     {
     267           0 :         ereport(elevel,
     268             :                 (errcode(ERRCODE_NAME_TOO_LONG),
     269             :                  errmsg("replication slot name \"%s\" is too long",
     270             :                         name)));
     271           0 :         return false;
     272             :     }
     273             : 
     274       33654 :     for (cp = name; *cp; cp++)
     275             :     {
     276       31992 :         if (!((*cp >= 'a' && *cp <= 'z')
     277       16020 :               || (*cp >= '0' && *cp <= '9')
     278        3092 :               || (*cp == '_')))
     279             :         {
     280           2 :             ereport(elevel,
     281             :                     (errcode(ERRCODE_INVALID_NAME),
     282             :                      errmsg("replication slot name \"%s\" contains invalid character",
     283             :                             name),
     284             :                      errhint("Replication slot names may only contain lower case letters, numbers, and the underscore character.")));
     285           0 :             return false;
     286             :         }
     287             :     }
     288        1662 :     return true;
     289             : }
     290             : 
     291             : /*
     292             :  * Create a new replication slot and mark it as used by this backend.
     293             :  *
     294             :  * name: Name of the slot
     295             :  * db_specific: logical decoding is db specific; if the slot is going to
     296             :  *     be used for that pass true, otherwise false.
     297             :  * two_phase: Allows decoding of prepared transactions. We allow this option
     298             :  *     to be enabled only at the slot creation time. If we allow this option
     299             :  *     to be changed during decoding then it is quite possible that we skip
     300             :  *     prepare first time because this option was not enabled. Now next time
     301             :  *     during getting changes, if the two_phase option is enabled it can skip
     302             :  *     prepare because by that time start decoding point has been moved. So the
     303             :  *     user will only get commit prepared.
     304             :  * failover: If enabled, allows the slot to be synced to standbys so
     305             :  *     that logical replication can be resumed after failover.
     306             :  * synced: True if the slot is synchronized from the primary server.
     307             :  */
     308             : void
     309        1216 : ReplicationSlotCreate(const char *name, bool db_specific,
     310             :                       ReplicationSlotPersistency persistency,
     311             :                       bool two_phase, bool failover, bool synced)
     312             : {
     313        1216 :     ReplicationSlot *slot = NULL;
     314             :     int         i;
     315             : 
     316             :     Assert(MyReplicationSlot == NULL);
     317             : 
     318        1216 :     ReplicationSlotValidateName(name, ERROR);
     319             : 
     320        1214 :     if (failover)
     321             :     {
     322             :         /*
     323             :          * Do not allow users to create the failover enabled slots on the
     324             :          * standby as we do not support sync to the cascading standby.
     325             :          *
     326             :          * However, failover enabled slots can be created during slot
     327             :          * synchronization because we need to retain the same values as the
     328             :          * remote slot.
     329             :          */
     330          42 :         if (RecoveryInProgress() && !IsSyncingReplicationSlots())
     331           0 :             ereport(ERROR,
     332             :                     errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
     333             :                     errmsg("cannot enable failover for a replication slot created on the standby"));
     334             : 
     335             :         /*
     336             :          * Do not allow users to create failover enabled temporary slots,
     337             :          * because temporary slots will not be synced to the standby.
     338             :          *
     339             :          * However, failover enabled temporary slots can be created during
     340             :          * slot synchronization. See the comments atop slotsync.c for details.
     341             :          */
     342          42 :         if (persistency == RS_TEMPORARY && !IsSyncingReplicationSlots())
     343           2 :             ereport(ERROR,
     344             :                     errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
     345             :                     errmsg("cannot enable failover for a temporary replication slot"));
     346             :     }
     347             : 
     348             :     /*
     349             :      * If some other backend ran this code concurrently with us, we'd likely
     350             :      * both allocate the same slot, and that would be bad.  We'd also be at
     351             :      * risk of missing a name collision.  Also, we don't want to try to create
     352             :      * a new slot while somebody's busy cleaning up an old one, because we
     353             :      * might both be monkeying with the same directory.
     354             :      */
     355        1212 :     LWLockAcquire(ReplicationSlotAllocationLock, LW_EXCLUSIVE);
     356             : 
     357             :     /*
     358             :      * Check for name collision, and identify an allocatable slot.  We need to
     359             :      * hold ReplicationSlotControlLock in shared mode for this, so that nobody
     360             :      * else can change the in_use flags while we're looking at them.
     361             :      */
     362        1212 :     LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
     363       11596 :     for (i = 0; i < max_replication_slots; i++)
     364             :     {
     365       10390 :         ReplicationSlot *s = &ReplicationSlotCtl->replication_slots[i];
     366             : 
     367       10390 :         if (s->in_use && strcmp(name, NameStr(s->data.name)) == 0)
     368           6 :             ereport(ERROR,
     369             :                     (errcode(ERRCODE_DUPLICATE_OBJECT),
     370             :                      errmsg("replication slot \"%s\" already exists", name)));
     371       10384 :         if (!s->in_use && slot == NULL)
     372        1204 :             slot = s;
     373             :     }
     374        1206 :     LWLockRelease(ReplicationSlotControlLock);
     375             : 
     376             :     /* If all slots are in use, we're out of luck. */
     377        1206 :     if (slot == NULL)
     378           2 :         ereport(ERROR,
     379             :                 (errcode(ERRCODE_CONFIGURATION_LIMIT_EXCEEDED),
     380             :                  errmsg("all replication slots are in use"),
     381             :                  errhint("Free one or increase \"max_replication_slots\".")));
     382             : 
     383             :     /*
     384             :      * Since this slot is not in use, nobody should be looking at any part of
     385             :      * it other than the in_use field unless they're trying to allocate it.
     386             :      * And since we hold ReplicationSlotAllocationLock, nobody except us can
     387             :      * be doing that.  So it's safe to initialize the slot.
     388             :      */
     389             :     Assert(!slot->in_use);
     390             :     Assert(slot->active_pid == 0);
     391             : 
     392             :     /* first initialize persistent data */
     393        1204 :     memset(&slot->data, 0, sizeof(ReplicationSlotPersistentData));
     394        1204 :     namestrcpy(&slot->data.name, name);
     395        1204 :     slot->data.database = db_specific ? MyDatabaseId : InvalidOid;
     396        1204 :     slot->data.persistency = persistency;
     397        1204 :     slot->data.two_phase = two_phase;
     398        1204 :     slot->data.two_phase_at = InvalidXLogRecPtr;
     399        1204 :     slot->data.failover = failover;
     400        1204 :     slot->data.synced = synced;
     401             : 
     402             :     /* and then data only present in shared memory */
     403        1204 :     slot->just_dirtied = false;
     404        1204 :     slot->dirty = false;
     405        1204 :     slot->effective_xmin = InvalidTransactionId;
     406        1204 :     slot->effective_catalog_xmin = InvalidTransactionId;
     407        1204 :     slot->candidate_catalog_xmin = InvalidTransactionId;
     408        1204 :     slot->candidate_xmin_lsn = InvalidXLogRecPtr;
     409        1204 :     slot->candidate_restart_valid = InvalidXLogRecPtr;
     410        1204 :     slot->candidate_restart_lsn = InvalidXLogRecPtr;
     411        1204 :     slot->last_saved_confirmed_flush = InvalidXLogRecPtr;
     412        1204 :     slot->inactive_since = 0;
     413             : 
     414             :     /*
     415             :      * Create the slot on disk.  We haven't actually marked the slot allocated
     416             :      * yet, so no special cleanup is required if this errors out.
     417             :      */
     418        1204 :     CreateSlotOnDisk(slot);
     419             : 
     420             :     /*
     421             :      * We need to briefly prevent any other backend from iterating over the
     422             :      * slots while we flip the in_use flag. We also need to set the active
     423             :      * flag while holding the ControlLock as otherwise a concurrent
     424             :      * ReplicationSlotAcquire() could acquire the slot as well.
     425             :      */
     426        1204 :     LWLockAcquire(ReplicationSlotControlLock, LW_EXCLUSIVE);
     427             : 
     428        1204 :     slot->in_use = true;
     429             : 
     430             :     /* We can now mark the slot active, and that makes it our slot. */
     431        1204 :     SpinLockAcquire(&slot->mutex);
     432             :     Assert(slot->active_pid == 0);
     433        1204 :     slot->active_pid = MyProcPid;
     434        1204 :     SpinLockRelease(&slot->mutex);
     435        1204 :     MyReplicationSlot = slot;
     436             : 
     437        1204 :     LWLockRelease(ReplicationSlotControlLock);
     438             : 
     439             :     /*
     440             :      * Create statistics entry for the new logical slot. We don't collect any
     441             :      * stats for physical slots, so no need to create an entry for the same.
     442             :      * See ReplicationSlotDropPtr for why we need to do this before releasing
     443             :      * ReplicationSlotAllocationLock.
     444             :      */
     445        1204 :     if (SlotIsLogical(slot))
     446         866 :         pgstat_create_replslot(slot);
     447             : 
     448             :     /*
     449             :      * Now that the slot has been marked as in_use and active, it's safe to
     450             :      * let somebody else try to allocate a slot.
     451             :      */
     452        1204 :     LWLockRelease(ReplicationSlotAllocationLock);
     453             : 
     454             :     /* Let everybody know we've modified this slot */
     455        1204 :     ConditionVariableBroadcast(&slot->active_cv);
     456        1204 : }
     457             : 
     458             : /*
     459             :  * Search for the named replication slot.
     460             :  *
     461             :  * Return the replication slot if found, otherwise NULL.
     462             :  */
     463             : ReplicationSlot *
     464        2630 : SearchNamedReplicationSlot(const char *name, bool need_lock)
     465             : {
     466             :     int         i;
     467        2630 :     ReplicationSlot *slot = NULL;
     468             : 
     469        2630 :     if (need_lock)
     470         156 :         LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
     471             : 
     472        4496 :     for (i = 0; i < max_replication_slots; i++)
     473             :     {
     474        4456 :         ReplicationSlot *s = &ReplicationSlotCtl->replication_slots[i];
     475             : 
     476        4456 :         if (s->in_use && strcmp(name, NameStr(s->data.name)) == 0)
     477             :         {
     478        2590 :             slot = s;
     479        2590 :             break;
     480             :         }
     481             :     }
     482             : 
     483        2630 :     if (need_lock)
     484         156 :         LWLockRelease(ReplicationSlotControlLock);
     485             : 
     486        2630 :     return slot;
     487             : }
     488             : 
     489             : /*
     490             :  * Return the index of the replication slot in
     491             :  * ReplicationSlotCtl->replication_slots.
     492             :  *
     493             :  * This is mainly useful to have an efficient key for storing replication slot
     494             :  * stats.
     495             :  */
     496             : int
     497       15296 : ReplicationSlotIndex(ReplicationSlot *slot)
     498             : {
     499             :     Assert(slot >= ReplicationSlotCtl->replication_slots &&
     500             :            slot < ReplicationSlotCtl->replication_slots + max_replication_slots);
     501             : 
     502       15296 :     return slot - ReplicationSlotCtl->replication_slots;
     503             : }
     504             : 
     505             : /*
     506             :  * If the slot at 'index' is unused, return false. Otherwise 'name' is set to
     507             :  * the slot's name and true is returned.
     508             :  *
     509             :  * This likely is only useful for pgstat_replslot.c during shutdown, in other
     510             :  * cases there are obvious TOCTOU issues.
     511             :  */
     512             : bool
     513         150 : ReplicationSlotName(int index, Name name)
     514             : {
     515             :     ReplicationSlot *slot;
     516             :     bool        found;
     517             : 
     518         150 :     slot = &ReplicationSlotCtl->replication_slots[index];
     519             : 
     520             :     /*
     521             :      * Ensure that the slot cannot be dropped while we copy the name. Don't
     522             :      * need the spinlock as the name of an existing slot cannot change.
     523             :      */
     524         150 :     LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
     525         150 :     found = slot->in_use;
     526         150 :     if (slot->in_use)
     527         150 :         namestrcpy(name, NameStr(slot->data.name));
     528         150 :     LWLockRelease(ReplicationSlotControlLock);
     529             : 
     530         150 :     return found;
     531             : }
     532             : 
     533             : /*
     534             :  * Find a previously created slot and mark it as used by this process.
     535             :  *
     536             :  * An error is raised if nowait is true and the slot is currently in use. If
     537             :  * nowait is false, we sleep until the slot is released by the owning process.
     538             :  */
     539             : void
     540        2338 : ReplicationSlotAcquire(const char *name, bool nowait)
     541             : {
     542             :     ReplicationSlot *s;
     543             :     int         active_pid;
     544             : 
     545             :     Assert(name != NULL);
     546             : 
     547        2338 : retry:
     548             :     Assert(MyReplicationSlot == NULL);
     549             : 
     550        2338 :     LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
     551             : 
     552             :     /* Check if the slot exits with the given name. */
     553        2338 :     s = SearchNamedReplicationSlot(name, false);
     554        2338 :     if (s == NULL || !s->in_use)
     555             :     {
     556          16 :         LWLockRelease(ReplicationSlotControlLock);
     557             : 
     558          16 :         ereport(ERROR,
     559             :                 (errcode(ERRCODE_UNDEFINED_OBJECT),
     560             :                  errmsg("replication slot \"%s\" does not exist",
     561             :                         name)));
     562             :     }
     563             : 
     564             :     /*
     565             :      * This is the slot we want; check if it's active under some other
     566             :      * process.  In single user mode, we don't need this check.
     567             :      */
     568        2322 :     if (IsUnderPostmaster)
     569             :     {
     570             :         /*
     571             :          * Get ready to sleep on the slot in case it is active.  (We may end
     572             :          * up not sleeping, but we don't want to do this while holding the
     573             :          * spinlock.)
     574             :          */
     575        2322 :         if (!nowait)
     576         500 :             ConditionVariablePrepareToSleep(&s->active_cv);
     577             : 
     578        2322 :         SpinLockAcquire(&s->mutex);
     579        2322 :         if (s->active_pid == 0)
     580        2050 :             s->active_pid = MyProcPid;
     581        2322 :         active_pid = s->active_pid;
     582        2322 :         SpinLockRelease(&s->mutex);
     583             :     }
     584             :     else
     585           0 :         active_pid = MyProcPid;
     586        2322 :     LWLockRelease(ReplicationSlotControlLock);
     587             : 
     588             :     /*
     589             :      * If we found the slot but it's already active in another process, we
     590             :      * wait until the owning process signals us that it's been released, or
     591             :      * error out.
     592             :      */
     593        2322 :     if (active_pid != MyProcPid)
     594             :     {
     595           0 :         if (!nowait)
     596             :         {
     597             :             /* Wait here until we get signaled, and then restart */
     598           0 :             ConditionVariableSleep(&s->active_cv,
     599             :                                    WAIT_EVENT_REPLICATION_SLOT_DROP);
     600           0 :             ConditionVariableCancelSleep();
     601           0 :             goto retry;
     602             :         }
     603             : 
     604           0 :         ereport(ERROR,
     605             :                 (errcode(ERRCODE_OBJECT_IN_USE),
     606             :                  errmsg("replication slot \"%s\" is active for PID %d",
     607             :                         NameStr(s->data.name), active_pid)));
     608             :     }
     609        2322 :     else if (!nowait)
     610         500 :         ConditionVariableCancelSleep(); /* no sleep needed after all */
     611             : 
     612             :     /* Let everybody know we've modified this slot */
     613        2322 :     ConditionVariableBroadcast(&s->active_cv);
     614             : 
     615             :     /* We made this slot active, so it's ours now. */
     616        2322 :     MyReplicationSlot = s;
     617             : 
     618             :     /*
     619             :      * The call to pgstat_acquire_replslot() protects against stats for a
     620             :      * different slot, from before a restart or such, being present during
     621             :      * pgstat_report_replslot().
     622             :      */
     623        2322 :     if (SlotIsLogical(s))
     624        1934 :         pgstat_acquire_replslot(s);
     625             : 
     626             :     /*
     627             :      * Reset the time since the slot has become inactive as the slot is active
     628             :      * now.
     629             :      */
     630        2322 :     SpinLockAcquire(&s->mutex);
     631        2322 :     s->inactive_since = 0;
     632        2322 :     SpinLockRelease(&s->mutex);
     633             : 
     634        2322 :     if (am_walsender)
     635             :     {
     636        1582 :         ereport(log_replication_commands ? LOG : DEBUG1,
     637             :                 SlotIsLogical(s)
     638             :                 ? errmsg("acquired logical replication slot \"%s\"",
     639             :                          NameStr(s->data.name))
     640             :                 : errmsg("acquired physical replication slot \"%s\"",
     641             :                          NameStr(s->data.name)));
     642             :     }
     643        2322 : }
     644             : 
     645             : /*
     646             :  * Release the replication slot that this backend considers to own.
     647             :  *
     648             :  * This or another backend can re-acquire the slot later.
     649             :  * Resources this slot requires will be preserved.
     650             :  */
     651             : void
     652        2816 : ReplicationSlotRelease(void)
     653             : {
     654        2816 :     ReplicationSlot *slot = MyReplicationSlot;
     655        2816 :     char       *slotname = NULL;    /* keep compiler quiet */
     656        2816 :     bool        is_logical = false; /* keep compiler quiet */
     657        2816 :     TimestampTz now = 0;
     658             : 
     659             :     Assert(slot != NULL && slot->active_pid != 0);
     660             : 
     661        2816 :     if (am_walsender)
     662             :     {
     663        1974 :         slotname = pstrdup(NameStr(slot->data.name));
     664        1974 :         is_logical = SlotIsLogical(slot);
     665             :     }
     666             : 
     667        2816 :     if (slot->data.persistency == RS_EPHEMERAL)
     668             :     {
     669             :         /*
     670             :          * Delete the slot. There is no !PANIC case where this is allowed to
     671             :          * fail, all that may happen is an incomplete cleanup of the on-disk
     672             :          * data.
     673             :          */
     674          10 :         ReplicationSlotDropAcquired();
     675             :     }
     676             : 
     677             :     /*
     678             :      * If slot needed to temporarily restrain both data and catalog xmin to
     679             :      * create the catalog snapshot, remove that temporary constraint.
     680             :      * Snapshots can only be exported while the initial snapshot is still
     681             :      * acquired.
     682             :      */
     683        2816 :     if (!TransactionIdIsValid(slot->data.xmin) &&
     684        2768 :         TransactionIdIsValid(slot->effective_xmin))
     685             :     {
     686         368 :         SpinLockAcquire(&slot->mutex);
     687         368 :         slot->effective_xmin = InvalidTransactionId;
     688         368 :         SpinLockRelease(&slot->mutex);
     689         368 :         ReplicationSlotsComputeRequiredXmin(false);
     690             :     }
     691             : 
     692             :     /*
     693             :      * Set the time since the slot has become inactive. We get the current
     694             :      * time beforehand to avoid system call while holding the spinlock.
     695             :      */
     696        2816 :     now = GetCurrentTimestamp();
     697             : 
     698        2816 :     if (slot->data.persistency == RS_PERSISTENT)
     699             :     {
     700             :         /*
     701             :          * Mark persistent slot inactive.  We're not freeing it, just
     702             :          * disconnecting, but wake up others that may be waiting for it.
     703             :          */
     704        2258 :         SpinLockAcquire(&slot->mutex);
     705        2258 :         slot->active_pid = 0;
     706        2258 :         slot->inactive_since = now;
     707        2258 :         SpinLockRelease(&slot->mutex);
     708        2258 :         ConditionVariableBroadcast(&slot->active_cv);
     709             :     }
     710             :     else
     711             :     {
     712         558 :         SpinLockAcquire(&slot->mutex);
     713         558 :         slot->inactive_since = now;
     714         558 :         SpinLockRelease(&slot->mutex);
     715             :     }
     716             : 
     717        2816 :     MyReplicationSlot = NULL;
     718             : 
     719             :     /* might not have been set when we've been a plain slot */
     720        2816 :     LWLockAcquire(ProcArrayLock, LW_EXCLUSIVE);
     721        2816 :     MyProc->statusFlags &= ~PROC_IN_LOGICAL_DECODING;
     722        2816 :     ProcGlobal->statusFlags[MyProc->pgxactoff] = MyProc->statusFlags;
     723        2816 :     LWLockRelease(ProcArrayLock);
     724             : 
     725        2816 :     if (am_walsender)
     726             :     {
     727        1974 :         ereport(log_replication_commands ? LOG : DEBUG1,
     728             :                 is_logical
     729             :                 ? errmsg("released logical replication slot \"%s\"",
     730             :                          slotname)
     731             :                 : errmsg("released physical replication slot \"%s\"",
     732             :                          slotname));
     733             : 
     734        1974 :         pfree(slotname);
     735             :     }
     736        2816 : }
     737             : 
     738             : /*
     739             :  * Cleanup temporary slots created in current session.
     740             :  *
     741             :  * Cleanup only synced temporary slots if 'synced_only' is true, else
     742             :  * cleanup all temporary slots.
     743             :  */
     744             : void
     745       76670 : ReplicationSlotCleanup(bool synced_only)
     746             : {
     747             :     int         i;
     748             : 
     749             :     Assert(MyReplicationSlot == NULL);
     750             : 
     751       76670 : restart:
     752       76670 :     LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
     753      830524 :     for (i = 0; i < max_replication_slots; i++)
     754             :     {
     755      754130 :         ReplicationSlot *s = &ReplicationSlotCtl->replication_slots[i];
     756             : 
     757      754130 :         if (!s->in_use)
     758      729564 :             continue;
     759             : 
     760       24566 :         SpinLockAcquire(&s->mutex);
     761       24566 :         if ((s->active_pid == MyProcPid &&
     762         276 :              (!synced_only || s->data.synced)))
     763             :         {
     764             :             Assert(s->data.persistency == RS_TEMPORARY);
     765         276 :             SpinLockRelease(&s->mutex);
     766         276 :             LWLockRelease(ReplicationSlotControlLock);  /* avoid deadlock */
     767             : 
     768         276 :             ReplicationSlotDropPtr(s);
     769             : 
     770         276 :             ConditionVariableBroadcast(&s->active_cv);
     771         276 :             goto restart;
     772             :         }
     773             :         else
     774       24290 :             SpinLockRelease(&s->mutex);
     775             :     }
     776             : 
     777       76394 :     LWLockRelease(ReplicationSlotControlLock);
     778       76394 : }
     779             : 
     780             : /*
     781             :  * Permanently drop replication slot identified by the passed in name.
     782             :  */
     783             : void
     784         746 : ReplicationSlotDrop(const char *name, bool nowait)
     785             : {
     786             :     Assert(MyReplicationSlot == NULL);
     787             : 
     788         746 :     ReplicationSlotAcquire(name, nowait);
     789             : 
     790             :     /*
     791             :      * Do not allow users to drop the slots which are currently being synced
     792             :      * from the primary to the standby.
     793             :      */
     794         736 :     if (RecoveryInProgress() && MyReplicationSlot->data.synced)
     795           2 :         ereport(ERROR,
     796             :                 errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
     797             :                 errmsg("cannot drop replication slot \"%s\"", name),
     798             :                 errdetail("This replication slot is being synchronized from the primary server."));
     799             : 
     800         734 :     ReplicationSlotDropAcquired();
     801         734 : }
     802             : 
     803             : /*
     804             :  * Change the definition of the slot identified by the specified name.
     805             :  */
     806             : void
     807          12 : ReplicationSlotAlter(const char *name, const bool *failover,
     808             :                      const bool *two_phase)
     809             : {
     810          12 :     bool        update_slot = false;
     811             : 
     812             :     Assert(MyReplicationSlot == NULL);
     813             :     Assert(failover || two_phase);
     814             : 
     815          12 :     ReplicationSlotAcquire(name, false);
     816             : 
     817          12 :     if (SlotIsPhysical(MyReplicationSlot))
     818           0 :         ereport(ERROR,
     819             :                 errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
     820             :                 errmsg("cannot use %s with a physical replication slot",
     821             :                        "ALTER_REPLICATION_SLOT"));
     822             : 
     823          12 :     if (MyReplicationSlot->data.invalidated != RS_INVAL_NONE)
     824           2 :         ereport(ERROR,
     825             :                 errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
     826             :                 errmsg("cannot alter invalid replication slot \"%s\"", name),
     827             :                 errdetail("This replication slot has been invalidated due to \"%s\".",
     828             :                           SlotInvalidationCauses[MyReplicationSlot->data.invalidated]));
     829             : 
     830          10 :     if (RecoveryInProgress())
     831             :     {
     832             :         /*
     833             :          * Do not allow users to alter the slots which are currently being
     834             :          * synced from the primary to the standby.
     835             :          */
     836           2 :         if (MyReplicationSlot->data.synced)
     837           2 :             ereport(ERROR,
     838             :                     errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
     839             :                     errmsg("cannot alter replication slot \"%s\"", name),
     840             :                     errdetail("This replication slot is being synchronized from the primary server."));
     841             : 
     842             :         /*
     843             :          * Do not allow users to enable failover on the standby as we do not
     844             :          * support sync to the cascading standby.
     845             :          */
     846           0 :         if (failover && *failover)
     847           0 :             ereport(ERROR,
     848             :                     errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
     849             :                     errmsg("cannot enable failover for a replication slot"
     850             :                            " on the standby"));
     851             :     }
     852             : 
     853           8 :     if (failover)
     854             :     {
     855             :         /*
     856             :          * Do not allow users to enable failover for temporary slots as we do
     857             :          * not support syncing temporary slots to the standby.
     858             :          */
     859           6 :         if (*failover && MyReplicationSlot->data.persistency == RS_TEMPORARY)
     860           0 :             ereport(ERROR,
     861             :                     errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
     862             :                     errmsg("cannot enable failover for a temporary replication slot"));
     863             : 
     864           6 :         if (MyReplicationSlot->data.failover != *failover)
     865             :         {
     866           6 :             SpinLockAcquire(&MyReplicationSlot->mutex);
     867           6 :             MyReplicationSlot->data.failover = *failover;
     868           6 :             SpinLockRelease(&MyReplicationSlot->mutex);
     869             : 
     870           6 :             update_slot = true;
     871             :         }
     872             :     }
     873             : 
     874           8 :     if (two_phase && MyReplicationSlot->data.two_phase != *two_phase)
     875             :     {
     876           2 :         SpinLockAcquire(&MyReplicationSlot->mutex);
     877           2 :         MyReplicationSlot->data.two_phase = *two_phase;
     878           2 :         SpinLockRelease(&MyReplicationSlot->mutex);
     879             : 
     880           2 :         update_slot = true;
     881             :     }
     882             : 
     883           8 :     if (update_slot)
     884             :     {
     885           8 :         ReplicationSlotMarkDirty();
     886           8 :         ReplicationSlotSave();
     887             :     }
     888             : 
     889           8 :     ReplicationSlotRelease();
     890           8 : }
     891             : 
     892             : /*
     893             :  * Permanently drop the currently acquired replication slot.
     894             :  */
     895             : void
     896         758 : ReplicationSlotDropAcquired(void)
     897             : {
     898         758 :     ReplicationSlot *slot = MyReplicationSlot;
     899             : 
     900             :     Assert(MyReplicationSlot != NULL);
     901             : 
     902             :     /* slot isn't acquired anymore */
     903         758 :     MyReplicationSlot = NULL;
     904             : 
     905         758 :     ReplicationSlotDropPtr(slot);
     906         758 : }
     907             : 
     908             : /*
     909             :  * Permanently drop the replication slot which will be released by the point
     910             :  * this function returns.
     911             :  */
     912             : static void
     913        1034 : ReplicationSlotDropPtr(ReplicationSlot *slot)
     914             : {
     915             :     char        path[MAXPGPATH];
     916             :     char        tmppath[MAXPGPATH];
     917             : 
     918             :     /*
     919             :      * If some other backend ran this code concurrently with us, we might try
     920             :      * to delete a slot with a certain name while someone else was trying to
     921             :      * create a slot with the same name.
     922             :      */
     923        1034 :     LWLockAcquire(ReplicationSlotAllocationLock, LW_EXCLUSIVE);
     924             : 
     925             :     /* Generate pathnames. */
     926        1034 :     sprintf(path, "%s/%s", PG_REPLSLOT_DIR, NameStr(slot->data.name));
     927        1034 :     sprintf(tmppath, "%s/%s.tmp", PG_REPLSLOT_DIR, NameStr(slot->data.name));
     928             : 
     929             :     /*
     930             :      * Rename the slot directory on disk, so that we'll no longer recognize
     931             :      * this as a valid slot.  Note that if this fails, we've got to mark the
     932             :      * slot inactive before bailing out.  If we're dropping an ephemeral or a
     933             :      * temporary slot, we better never fail hard as the caller won't expect
     934             :      * the slot to survive and this might get called during error handling.
     935             :      */
     936        1034 :     if (rename(path, tmppath) == 0)
     937             :     {
     938             :         /*
     939             :          * We need to fsync() the directory we just renamed and its parent to
     940             :          * make sure that our changes are on disk in a crash-safe fashion.  If
     941             :          * fsync() fails, we can't be sure whether the changes are on disk or
     942             :          * not.  For now, we handle that by panicking;
     943             :          * StartupReplicationSlots() will try to straighten it out after
     944             :          * restart.
     945             :          */
     946        1034 :         START_CRIT_SECTION();
     947        1034 :         fsync_fname(tmppath, true);
     948        1034 :         fsync_fname(PG_REPLSLOT_DIR, true);
     949        1034 :         END_CRIT_SECTION();
     950             :     }
     951             :     else
     952             :     {
     953           0 :         bool        fail_softly = slot->data.persistency != RS_PERSISTENT;
     954             : 
     955           0 :         SpinLockAcquire(&slot->mutex);
     956           0 :         slot->active_pid = 0;
     957           0 :         SpinLockRelease(&slot->mutex);
     958             : 
     959             :         /* wake up anyone waiting on this slot */
     960           0 :         ConditionVariableBroadcast(&slot->active_cv);
     961             : 
     962           0 :         ereport(fail_softly ? WARNING : ERROR,
     963             :                 (errcode_for_file_access(),
     964             :                  errmsg("could not rename file \"%s\" to \"%s\": %m",
     965             :                         path, tmppath)));
     966             :     }
     967             : 
     968             :     /*
     969             :      * The slot is definitely gone.  Lock out concurrent scans of the array
     970             :      * long enough to kill it.  It's OK to clear the active PID here without
     971             :      * grabbing the mutex because nobody else can be scanning the array here,
     972             :      * and nobody can be attached to this slot and thus access it without
     973             :      * scanning the array.
     974             :      *
     975             :      * Also wake up processes waiting for it.
     976             :      */
     977        1034 :     LWLockAcquire(ReplicationSlotControlLock, LW_EXCLUSIVE);
     978        1034 :     slot->active_pid = 0;
     979        1034 :     slot->in_use = false;
     980        1034 :     LWLockRelease(ReplicationSlotControlLock);
     981        1034 :     ConditionVariableBroadcast(&slot->active_cv);
     982             : 
     983             :     /*
     984             :      * Slot is dead and doesn't prevent resource removal anymore, recompute
     985             :      * limits.
     986             :      */
     987        1034 :     ReplicationSlotsComputeRequiredXmin(false);
     988        1034 :     ReplicationSlotsComputeRequiredLSN();
     989             : 
     990             :     /*
     991             :      * If removing the directory fails, the worst thing that will happen is
     992             :      * that the user won't be able to create a new slot with the same name
     993             :      * until the next server restart.  We warn about it, but that's all.
     994             :      */
     995        1034 :     if (!rmtree(tmppath, true))
     996           0 :         ereport(WARNING,
     997             :                 (errmsg("could not remove directory \"%s\"", tmppath)));
     998             : 
     999             :     /*
    1000             :      * Drop the statistics entry for the replication slot.  Do this while
    1001             :      * holding ReplicationSlotAllocationLock so that we don't drop a
    1002             :      * statistics entry for another slot with the same name just created in
    1003             :      * another session.
    1004             :      */
    1005        1034 :     if (SlotIsLogical(slot))
    1006         742 :         pgstat_drop_replslot(slot);
    1007             : 
    1008             :     /*
    1009             :      * We release this at the very end, so that nobody starts trying to create
    1010             :      * a slot while we're still cleaning up the detritus of the old one.
    1011             :      */
    1012        1034 :     LWLockRelease(ReplicationSlotAllocationLock);
    1013        1034 : }
    1014             : 
    1015             : /*
    1016             :  * Serialize the currently acquired slot's state from memory to disk, thereby
    1017             :  * guaranteeing the current state will survive a crash.
    1018             :  */
    1019             : void
    1020        2422 : ReplicationSlotSave(void)
    1021             : {
    1022             :     char        path[MAXPGPATH];
    1023             : 
    1024             :     Assert(MyReplicationSlot != NULL);
    1025             : 
    1026        2422 :     sprintf(path, "%s/%s", PG_REPLSLOT_DIR, NameStr(MyReplicationSlot->data.name));
    1027        2422 :     SaveSlotToPath(MyReplicationSlot, path, ERROR);
    1028        2422 : }
    1029             : 
    1030             : /*
    1031             :  * Signal that it would be useful if the currently acquired slot would be
    1032             :  * flushed out to disk.
    1033             :  *
    1034             :  * Note that the actual flush to disk can be delayed for a long time, if
    1035             :  * required for correctness explicitly do a ReplicationSlotSave().
    1036             :  */
    1037             : void
    1038       11478 : ReplicationSlotMarkDirty(void)
    1039             : {
    1040       11478 :     ReplicationSlot *slot = MyReplicationSlot;
    1041             : 
    1042             :     Assert(MyReplicationSlot != NULL);
    1043             : 
    1044       11478 :     SpinLockAcquire(&slot->mutex);
    1045       11478 :     MyReplicationSlot->just_dirtied = true;
    1046       11478 :     MyReplicationSlot->dirty = true;
    1047       11478 :     SpinLockRelease(&slot->mutex);
    1048       11478 : }
    1049             : 
    1050             : /*
    1051             :  * Convert a slot that's marked as RS_EPHEMERAL or RS_TEMPORARY to a
    1052             :  * RS_PERSISTENT slot, guaranteeing it will be there after an eventual crash.
    1053             :  */
    1054             : void
    1055         840 : ReplicationSlotPersist(void)
    1056             : {
    1057         840 :     ReplicationSlot *slot = MyReplicationSlot;
    1058             : 
    1059             :     Assert(slot != NULL);
    1060             :     Assert(slot->data.persistency != RS_PERSISTENT);
    1061             : 
    1062         840 :     SpinLockAcquire(&slot->mutex);
    1063         840 :     slot->data.persistency = RS_PERSISTENT;
    1064         840 :     SpinLockRelease(&slot->mutex);
    1065             : 
    1066         840 :     ReplicationSlotMarkDirty();
    1067         840 :     ReplicationSlotSave();
    1068         840 : }
    1069             : 
    1070             : /*
    1071             :  * Compute the oldest xmin across all slots and store it in the ProcArray.
    1072             :  *
    1073             :  * If already_locked is true, ProcArrayLock has already been acquired
    1074             :  * exclusively.
    1075             :  */
    1076             : void
    1077        4192 : ReplicationSlotsComputeRequiredXmin(bool already_locked)
    1078             : {
    1079             :     int         i;
    1080        4192 :     TransactionId agg_xmin = InvalidTransactionId;
    1081        4192 :     TransactionId agg_catalog_xmin = InvalidTransactionId;
    1082             : 
    1083             :     Assert(ReplicationSlotCtl != NULL);
    1084             : 
    1085        4192 :     LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
    1086             : 
    1087       42162 :     for (i = 0; i < max_replication_slots; i++)
    1088             :     {
    1089       37970 :         ReplicationSlot *s = &ReplicationSlotCtl->replication_slots[i];
    1090             :         TransactionId effective_xmin;
    1091             :         TransactionId effective_catalog_xmin;
    1092             :         bool        invalidated;
    1093             : 
    1094       37970 :         if (!s->in_use)
    1095       34146 :             continue;
    1096             : 
    1097        3824 :         SpinLockAcquire(&s->mutex);
    1098        3824 :         effective_xmin = s->effective_xmin;
    1099        3824 :         effective_catalog_xmin = s->effective_catalog_xmin;
    1100        3824 :         invalidated = s->data.invalidated != RS_INVAL_NONE;
    1101        3824 :         SpinLockRelease(&s->mutex);
    1102             : 
    1103             :         /* invalidated slots need not apply */
    1104        3824 :         if (invalidated)
    1105          44 :             continue;
    1106             : 
    1107             :         /* check the data xmin */
    1108        3780 :         if (TransactionIdIsValid(effective_xmin) &&
    1109           6 :             (!TransactionIdIsValid(agg_xmin) ||
    1110           6 :              TransactionIdPrecedes(effective_xmin, agg_xmin)))
    1111         552 :             agg_xmin = effective_xmin;
    1112             : 
    1113             :         /* check the catalog xmin */
    1114        3780 :         if (TransactionIdIsValid(effective_catalog_xmin) &&
    1115        1554 :             (!TransactionIdIsValid(agg_catalog_xmin) ||
    1116        1554 :              TransactionIdPrecedes(effective_catalog_xmin, agg_catalog_xmin)))
    1117        2086 :             agg_catalog_xmin = effective_catalog_xmin;
    1118             :     }
    1119             : 
    1120        4192 :     LWLockRelease(ReplicationSlotControlLock);
    1121             : 
    1122        4192 :     ProcArraySetReplicationSlotXmin(agg_xmin, agg_catalog_xmin, already_locked);
    1123        4192 : }
    1124             : 
    1125             : /*
    1126             :  * Compute the oldest restart LSN across all slots and inform xlog module.
    1127             :  *
    1128             :  * Note: while max_slot_wal_keep_size is theoretically relevant for this
    1129             :  * purpose, we don't try to account for that, because this module doesn't
    1130             :  * know what to compare against.
    1131             :  */
    1132             : void
    1133       12308 : ReplicationSlotsComputeRequiredLSN(void)
    1134             : {
    1135             :     int         i;
    1136       12308 :     XLogRecPtr  min_required = InvalidXLogRecPtr;
    1137             : 
    1138             :     Assert(ReplicationSlotCtl != NULL);
    1139             : 
    1140       12308 :     LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
    1141      129064 :     for (i = 0; i < max_replication_slots; i++)
    1142             :     {
    1143      116756 :         ReplicationSlot *s = &ReplicationSlotCtl->replication_slots[i];
    1144             :         XLogRecPtr  restart_lsn;
    1145             :         bool        invalidated;
    1146             : 
    1147      116756 :         if (!s->in_use)
    1148      104932 :             continue;
    1149             : 
    1150       11824 :         SpinLockAcquire(&s->mutex);
    1151       11824 :         restart_lsn = s->data.restart_lsn;
    1152       11824 :         invalidated = s->data.invalidated != RS_INVAL_NONE;
    1153       11824 :         SpinLockRelease(&s->mutex);
    1154             : 
    1155             :         /* invalidated slots need not apply */
    1156       11824 :         if (invalidated)
    1157          46 :             continue;
    1158             : 
    1159       11778 :         if (restart_lsn != InvalidXLogRecPtr &&
    1160        1486 :             (min_required == InvalidXLogRecPtr ||
    1161             :              restart_lsn < min_required))
    1162       10352 :             min_required = restart_lsn;
    1163             :     }
    1164       12308 :     LWLockRelease(ReplicationSlotControlLock);
    1165             : 
    1166       12308 :     XLogSetReplicationSlotMinimumLSN(min_required);
    1167       12308 : }
    1168             : 
    1169             : /*
    1170             :  * Compute the oldest WAL LSN required by *logical* decoding slots..
    1171             :  *
    1172             :  * Returns InvalidXLogRecPtr if logical decoding is disabled or no logical
    1173             :  * slots exist.
    1174             :  *
    1175             :  * NB: this returns a value >= ReplicationSlotsComputeRequiredLSN(), since it
    1176             :  * ignores physical replication slots.
    1177             :  *
    1178             :  * The results aren't required frequently, so we don't maintain a precomputed
    1179             :  * value like we do for ComputeRequiredLSN() and ComputeRequiredXmin().
    1180             :  */
    1181             : XLogRecPtr
    1182        4952 : ReplicationSlotsComputeLogicalRestartLSN(void)
    1183             : {
    1184        4952 :     XLogRecPtr  result = InvalidXLogRecPtr;
    1185             :     int         i;
    1186             : 
    1187        4952 :     if (max_replication_slots <= 0)
    1188           4 :         return InvalidXLogRecPtr;
    1189             : 
    1190        4948 :     LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
    1191             : 
    1192       53360 :     for (i = 0; i < max_replication_slots; i++)
    1193             :     {
    1194             :         ReplicationSlot *s;
    1195             :         XLogRecPtr  restart_lsn;
    1196             :         bool        invalidated;
    1197             : 
    1198       48412 :         s = &ReplicationSlotCtl->replication_slots[i];
    1199             : 
    1200             :         /* cannot change while ReplicationSlotCtlLock is held */
    1201       48412 :         if (!s->in_use)
    1202       47116 :             continue;
    1203             : 
    1204             :         /* we're only interested in logical slots */
    1205        1296 :         if (!SlotIsLogical(s))
    1206         948 :             continue;
    1207             : 
    1208             :         /* read once, it's ok if it increases while we're checking */
    1209         348 :         SpinLockAcquire(&s->mutex);
    1210         348 :         restart_lsn = s->data.restart_lsn;
    1211         348 :         invalidated = s->data.invalidated != RS_INVAL_NONE;
    1212         348 :         SpinLockRelease(&s->mutex);
    1213             : 
    1214             :         /* invalidated slots need not apply */
    1215         348 :         if (invalidated)
    1216           8 :             continue;
    1217             : 
    1218         340 :         if (restart_lsn == InvalidXLogRecPtr)
    1219           0 :             continue;
    1220             : 
    1221         340 :         if (result == InvalidXLogRecPtr ||
    1222             :             restart_lsn < result)
    1223         280 :             result = restart_lsn;
    1224             :     }
    1225             : 
    1226        4948 :     LWLockRelease(ReplicationSlotControlLock);
    1227             : 
    1228        4948 :     return result;
    1229             : }
    1230             : 
    1231             : /*
    1232             :  * ReplicationSlotsCountDBSlots -- count the number of slots that refer to the
    1233             :  * passed database oid.
    1234             :  *
    1235             :  * Returns true if there are any slots referencing the database. *nslots will
    1236             :  * be set to the absolute number of slots in the database, *nactive to ones
    1237             :  * currently active.
    1238             :  */
    1239             : bool
    1240          72 : ReplicationSlotsCountDBSlots(Oid dboid, int *nslots, int *nactive)
    1241             : {
    1242             :     int         i;
    1243             : 
    1244          72 :     *nslots = *nactive = 0;
    1245             : 
    1246          72 :     if (max_replication_slots <= 0)
    1247           0 :         return false;
    1248             : 
    1249          72 :     LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
    1250         734 :     for (i = 0; i < max_replication_slots; i++)
    1251             :     {
    1252             :         ReplicationSlot *s;
    1253             : 
    1254         662 :         s = &ReplicationSlotCtl->replication_slots[i];
    1255             : 
    1256             :         /* cannot change while ReplicationSlotCtlLock is held */
    1257         662 :         if (!s->in_use)
    1258         626 :             continue;
    1259             : 
    1260             :         /* only logical slots are database specific, skip */
    1261          36 :         if (!SlotIsLogical(s))
    1262          20 :             continue;
    1263             : 
    1264             :         /* not our database, skip */
    1265          16 :         if (s->data.database != dboid)
    1266          10 :             continue;
    1267             : 
    1268             :         /* NB: intentionally counting invalidated slots */
    1269             : 
    1270             :         /* count slots with spinlock held */
    1271           6 :         SpinLockAcquire(&s->mutex);
    1272           6 :         (*nslots)++;
    1273           6 :         if (s->active_pid != 0)
    1274           2 :             (*nactive)++;
    1275           6 :         SpinLockRelease(&s->mutex);
    1276             :     }
    1277          72 :     LWLockRelease(ReplicationSlotControlLock);
    1278             : 
    1279          72 :     if (*nslots > 0)
    1280           6 :         return true;
    1281          66 :     return false;
    1282             : }
    1283             : 
    1284             : /*
    1285             :  * ReplicationSlotsDropDBSlots -- Drop all db-specific slots relating to the
    1286             :  * passed database oid. The caller should hold an exclusive lock on the
    1287             :  * pg_database oid for the database to prevent creation of new slots on the db
    1288             :  * or replay from existing slots.
    1289             :  *
    1290             :  * Another session that concurrently acquires an existing slot on the target DB
    1291             :  * (most likely to drop it) may cause this function to ERROR. If that happens
    1292             :  * it may have dropped some but not all slots.
    1293             :  *
    1294             :  * This routine isn't as efficient as it could be - but we don't drop
    1295             :  * databases often, especially databases with lots of slots.
    1296             :  */
    1297             : void
    1298          96 : ReplicationSlotsDropDBSlots(Oid dboid)
    1299             : {
    1300             :     int         i;
    1301             : 
    1302          96 :     if (max_replication_slots <= 0)
    1303           0 :         return;
    1304             : 
    1305          96 : restart:
    1306         106 :     LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
    1307         992 :     for (i = 0; i < max_replication_slots; i++)
    1308             :     {
    1309             :         ReplicationSlot *s;
    1310             :         char       *slotname;
    1311             :         int         active_pid;
    1312             : 
    1313         896 :         s = &ReplicationSlotCtl->replication_slots[i];
    1314             : 
    1315             :         /* cannot change while ReplicationSlotCtlLock is held */
    1316         896 :         if (!s->in_use)
    1317         842 :             continue;
    1318             : 
    1319             :         /* only logical slots are database specific, skip */
    1320          54 :         if (!SlotIsLogical(s))
    1321          22 :             continue;
    1322             : 
    1323             :         /* not our database, skip */
    1324          32 :         if (s->data.database != dboid)
    1325          22 :             continue;
    1326             : 
    1327             :         /* NB: intentionally including invalidated slots */
    1328             : 
    1329             :         /* acquire slot, so ReplicationSlotDropAcquired can be reused  */
    1330          10 :         SpinLockAcquire(&s->mutex);
    1331             :         /* can't change while ReplicationSlotControlLock is held */
    1332          10 :         slotname = NameStr(s->data.name);
    1333          10 :         active_pid = s->active_pid;
    1334          10 :         if (active_pid == 0)
    1335             :         {
    1336          10 :             MyReplicationSlot = s;
    1337          10 :             s->active_pid = MyProcPid;
    1338             :         }
    1339          10 :         SpinLockRelease(&s->mutex);
    1340             : 
    1341             :         /*
    1342             :          * Even though we hold an exclusive lock on the database object a
    1343             :          * logical slot for that DB can still be active, e.g. if it's
    1344             :          * concurrently being dropped by a backend connected to another DB.
    1345             :          *
    1346             :          * That's fairly unlikely in practice, so we'll just bail out.
    1347             :          *
    1348             :          * The slot sync worker holds a shared lock on the database before
    1349             :          * operating on synced logical slots to avoid conflict with the drop
    1350             :          * happening here. The persistent synced slots are thus safe but there
    1351             :          * is a possibility that the slot sync worker has created a temporary
    1352             :          * slot (which stays active even on release) and we are trying to drop
    1353             :          * that here. In practice, the chances of hitting this scenario are
    1354             :          * less as during slot synchronization, the temporary slot is
    1355             :          * immediately converted to persistent and thus is safe due to the
    1356             :          * shared lock taken on the database. So, we'll just bail out in such
    1357             :          * a case.
    1358             :          *
    1359             :          * XXX: We can consider shutting down the slot sync worker before
    1360             :          * trying to drop synced temporary slots here.
    1361             :          */
    1362          10 :         if (active_pid)
    1363           0 :             ereport(ERROR,
    1364             :                     (errcode(ERRCODE_OBJECT_IN_USE),
    1365             :                      errmsg("replication slot \"%s\" is active for PID %d",
    1366             :                             slotname, active_pid)));
    1367             : 
    1368             :         /*
    1369             :          * To avoid duplicating ReplicationSlotDropAcquired() and to avoid
    1370             :          * holding ReplicationSlotControlLock over filesystem operations,
    1371             :          * release ReplicationSlotControlLock and use
    1372             :          * ReplicationSlotDropAcquired.
    1373             :          *
    1374             :          * As that means the set of slots could change, restart scan from the
    1375             :          * beginning each time we release the lock.
    1376             :          */
    1377          10 :         LWLockRelease(ReplicationSlotControlLock);
    1378          10 :         ReplicationSlotDropAcquired();
    1379          10 :         goto restart;
    1380             :     }
    1381          96 :     LWLockRelease(ReplicationSlotControlLock);
    1382             : }
    1383             : 
    1384             : 
    1385             : /*
    1386             :  * Check whether the server's configuration supports using replication
    1387             :  * slots.
    1388             :  */
    1389             : void
    1390        3204 : CheckSlotRequirements(void)
    1391             : {
    1392             :     /*
    1393             :      * NB: Adding a new requirement likely means that RestoreSlotFromDisk()
    1394             :      * needs the same check.
    1395             :      */
    1396             : 
    1397        3204 :     if (max_replication_slots == 0)
    1398           0 :         ereport(ERROR,
    1399             :                 (errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
    1400             :                  errmsg("replication slots can only be used if \"max_replication_slots\" > 0")));
    1401             : 
    1402        3204 :     if (wal_level < WAL_LEVEL_REPLICA)
    1403           0 :         ereport(ERROR,
    1404             :                 (errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
    1405             :                  errmsg("replication slots can only be used if \"wal_level\" >= \"replica\"")));
    1406        3204 : }
    1407             : 
    1408             : /*
    1409             :  * Check whether the user has privilege to use replication slots.
    1410             :  */
    1411             : void
    1412        1044 : CheckSlotPermissions(void)
    1413             : {
    1414        1044 :     if (!has_rolreplication(GetUserId()))
    1415          10 :         ereport(ERROR,
    1416             :                 (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
    1417             :                  errmsg("permission denied to use replication slots"),
    1418             :                  errdetail("Only roles with the %s attribute may use replication slots.",
    1419             :                            "REPLICATION")));
    1420        1034 : }
    1421             : 
    1422             : /*
    1423             :  * Reserve WAL for the currently active slot.
    1424             :  *
    1425             :  * Compute and set restart_lsn in a manner that's appropriate for the type of
    1426             :  * the slot and concurrency safe.
    1427             :  */
    1428             : void
    1429        1128 : ReplicationSlotReserveWal(void)
    1430             : {
    1431        1128 :     ReplicationSlot *slot = MyReplicationSlot;
    1432             : 
    1433             :     Assert(slot != NULL);
    1434             :     Assert(slot->data.restart_lsn == InvalidXLogRecPtr);
    1435             : 
    1436             :     /*
    1437             :      * The replication slot mechanism is used to prevent removal of required
    1438             :      * WAL. As there is no interlock between this routine and checkpoints, WAL
    1439             :      * segments could concurrently be removed when a now stale return value of
    1440             :      * ReplicationSlotsComputeRequiredLSN() is used. In the unlikely case that
    1441             :      * this happens we'll just retry.
    1442             :      */
    1443             :     while (true)
    1444           0 :     {
    1445             :         XLogSegNo   segno;
    1446             :         XLogRecPtr  restart_lsn;
    1447             : 
    1448             :         /*
    1449             :          * For logical slots log a standby snapshot and start logical decoding
    1450             :          * at exactly that position. That allows the slot to start up more
    1451             :          * quickly. But on a standby we cannot do WAL writes, so just use the
    1452             :          * replay pointer; effectively, an attempt to create a logical slot on
    1453             :          * standby will cause it to wait for an xl_running_xact record to be
    1454             :          * logged independently on the primary, so that a snapshot can be
    1455             :          * built using the record.
    1456             :          *
    1457             :          * None of this is needed (or indeed helpful) for physical slots as
    1458             :          * they'll start replay at the last logged checkpoint anyway. Instead
    1459             :          * return the location of the last redo LSN. While that slightly
    1460             :          * increases the chance that we have to retry, it's where a base
    1461             :          * backup has to start replay at.
    1462             :          */
    1463        1128 :         if (SlotIsPhysical(slot))
    1464         288 :             restart_lsn = GetRedoRecPtr();
    1465         840 :         else if (RecoveryInProgress())
    1466          44 :             restart_lsn = GetXLogReplayRecPtr(NULL);
    1467             :         else
    1468         796 :             restart_lsn = GetXLogInsertRecPtr();
    1469             : 
    1470        1128 :         SpinLockAcquire(&slot->mutex);
    1471        1128 :         slot->data.restart_lsn = restart_lsn;
    1472        1128 :         SpinLockRelease(&slot->mutex);
    1473             : 
    1474             :         /* prevent WAL removal as fast as possible */
    1475        1128 :         ReplicationSlotsComputeRequiredLSN();
    1476             : 
    1477             :         /*
    1478             :          * If all required WAL is still there, great, otherwise retry. The
    1479             :          * slot should prevent further removal of WAL, unless there's a
    1480             :          * concurrent ReplicationSlotsComputeRequiredLSN() after we've written
    1481             :          * the new restart_lsn above, so normally we should never need to loop
    1482             :          * more than twice.
    1483             :          */
    1484        1128 :         XLByteToSeg(slot->data.restart_lsn, segno, wal_segment_size);
    1485        1128 :         if (XLogGetLastRemovedSegno() < segno)
    1486        1128 :             break;
    1487             :     }
    1488             : 
    1489        1128 :     if (!RecoveryInProgress() && SlotIsLogical(slot))
    1490             :     {
    1491             :         XLogRecPtr  flushptr;
    1492             : 
    1493             :         /* make sure we have enough information to start */
    1494         796 :         flushptr = LogStandbySnapshot();
    1495             : 
    1496             :         /* and make sure it's fsynced to disk */
    1497         796 :         XLogFlush(flushptr);
    1498             :     }
    1499        1128 : }
    1500             : 
    1501             : /*
    1502             :  * Report that replication slot needs to be invalidated
    1503             :  */
    1504             : static void
    1505          42 : ReportSlotInvalidation(ReplicationSlotInvalidationCause cause,
    1506             :                        bool terminating,
    1507             :                        int pid,
    1508             :                        NameData slotname,
    1509             :                        XLogRecPtr restart_lsn,
    1510             :                        XLogRecPtr oldestLSN,
    1511             :                        TransactionId snapshotConflictHorizon)
    1512             : {
    1513             :     StringInfoData err_detail;
    1514          42 :     bool        hint = false;
    1515             : 
    1516          42 :     initStringInfo(&err_detail);
    1517             : 
    1518          42 :     switch (cause)
    1519             :     {
    1520          12 :         case RS_INVAL_WAL_REMOVED:
    1521             :             {
    1522          12 :                 unsigned long long ex = oldestLSN - restart_lsn;
    1523             : 
    1524          12 :                 hint = true;
    1525          12 :                 appendStringInfo(&err_detail,
    1526          12 :                                  ngettext("The slot's restart_lsn %X/%X exceeds the limit by %llu byte.",
    1527             :                                           "The slot's restart_lsn %X/%X exceeds the limit by %llu bytes.",
    1528             :                                           ex),
    1529          12 :                                  LSN_FORMAT_ARGS(restart_lsn),
    1530             :                                  ex);
    1531          12 :                 break;
    1532             :             }
    1533          24 :         case RS_INVAL_HORIZON:
    1534          24 :             appendStringInfo(&err_detail, _("The slot conflicted with xid horizon %u."),
    1535             :                              snapshotConflictHorizon);
    1536          24 :             break;
    1537             : 
    1538           6 :         case RS_INVAL_WAL_LEVEL:
    1539           6 :             appendStringInfoString(&err_detail, _("Logical decoding on standby requires \"wal_level\" >= \"logical\" on the primary server."));
    1540           6 :             break;
    1541             :         case RS_INVAL_NONE:
    1542             :             pg_unreachable();
    1543             :     }
    1544             : 
    1545          42 :     ereport(LOG,
    1546             :             terminating ?
    1547             :             errmsg("terminating process %d to release replication slot \"%s\"",
    1548             :                    pid, NameStr(slotname)) :
    1549             :             errmsg("invalidating obsolete replication slot \"%s\"",
    1550             :                    NameStr(slotname)),
    1551             :             errdetail_internal("%s", err_detail.data),
    1552             :             hint ? errhint("You might need to increase \"%s\".", "max_slot_wal_keep_size") : 0);
    1553             : 
    1554          42 :     pfree(err_detail.data);
    1555          42 : }
    1556             : 
    1557             : /*
    1558             :  * Helper for InvalidateObsoleteReplicationSlots
    1559             :  *
    1560             :  * Acquires the given slot and mark it invalid, if necessary and possible.
    1561             :  *
    1562             :  * Returns whether ReplicationSlotControlLock was released in the interim (and
    1563             :  * in that case we're not holding the lock at return, otherwise we are).
    1564             :  *
    1565             :  * Sets *invalidated true if the slot was invalidated. (Untouched otherwise.)
    1566             :  *
    1567             :  * This is inherently racy, because we release the LWLock
    1568             :  * for syscalls, so caller must restart if we return true.
    1569             :  */
    1570             : static bool
    1571         742 : InvalidatePossiblyObsoleteSlot(ReplicationSlotInvalidationCause cause,
    1572             :                                ReplicationSlot *s,
    1573             :                                XLogRecPtr oldestLSN,
    1574             :                                Oid dboid, TransactionId snapshotConflictHorizon,
    1575             :                                bool *invalidated)
    1576             : {
    1577         742 :     int         last_signaled_pid = 0;
    1578         742 :     bool        released_lock = false;
    1579         742 :     bool        terminated = false;
    1580         742 :     TransactionId initial_effective_xmin = InvalidTransactionId;
    1581         742 :     TransactionId initial_catalog_effective_xmin = InvalidTransactionId;
    1582         742 :     XLogRecPtr  initial_restart_lsn = InvalidXLogRecPtr;
    1583         742 :     ReplicationSlotInvalidationCause invalidation_cause_prev PG_USED_FOR_ASSERTS_ONLY = RS_INVAL_NONE;
    1584             : 
    1585             :     for (;;)
    1586          14 :     {
    1587             :         XLogRecPtr  restart_lsn;
    1588             :         NameData    slotname;
    1589         756 :         int         active_pid = 0;
    1590         756 :         ReplicationSlotInvalidationCause invalidation_cause = RS_INVAL_NONE;
    1591             : 
    1592             :         Assert(LWLockHeldByMeInMode(ReplicationSlotControlLock, LW_SHARED));
    1593             : 
    1594         756 :         if (!s->in_use)
    1595             :         {
    1596           0 :             if (released_lock)
    1597           0 :                 LWLockRelease(ReplicationSlotControlLock);
    1598           0 :             break;
    1599             :         }
    1600             : 
    1601             :         /*
    1602             :          * Check if the slot needs to be invalidated. If it needs to be
    1603             :          * invalidated, and is not currently acquired, acquire it and mark it
    1604             :          * as having been invalidated.  We do this with the spinlock held to
    1605             :          * avoid race conditions -- for example the restart_lsn could move
    1606             :          * forward, or the slot could be dropped.
    1607             :          */
    1608         756 :         SpinLockAcquire(&s->mutex);
    1609             : 
    1610         756 :         restart_lsn = s->data.restart_lsn;
    1611             : 
    1612             :         /* we do nothing if the slot is already invalid */
    1613         756 :         if (s->data.invalidated == RS_INVAL_NONE)
    1614             :         {
    1615             :             /*
    1616             :              * The slot's mutex will be released soon, and it is possible that
    1617             :              * those values change since the process holding the slot has been
    1618             :              * terminated (if any), so record them here to ensure that we
    1619             :              * would report the correct invalidation cause.
    1620             :              */
    1621         686 :             if (!terminated)
    1622             :             {
    1623         672 :                 initial_restart_lsn = s->data.restart_lsn;
    1624         672 :                 initial_effective_xmin = s->effective_xmin;
    1625         672 :                 initial_catalog_effective_xmin = s->effective_catalog_xmin;
    1626             :             }
    1627             : 
    1628         686 :             switch (cause)
    1629             :             {
    1630         636 :                 case RS_INVAL_WAL_REMOVED:
    1631         636 :                     if (initial_restart_lsn != InvalidXLogRecPtr &&
    1632             :                         initial_restart_lsn < oldestLSN)
    1633          12 :                         invalidation_cause = cause;
    1634         636 :                     break;
    1635          44 :                 case RS_INVAL_HORIZON:
    1636          44 :                     if (!SlotIsLogical(s))
    1637           0 :                         break;
    1638             :                     /* invalid DB oid signals a shared relation */
    1639          44 :                     if (dboid != InvalidOid && dboid != s->data.database)
    1640           0 :                         break;
    1641          44 :                     if (TransactionIdIsValid(initial_effective_xmin) &&
    1642           0 :                         TransactionIdPrecedesOrEquals(initial_effective_xmin,
    1643             :                                                       snapshotConflictHorizon))
    1644           0 :                         invalidation_cause = cause;
    1645          88 :                     else if (TransactionIdIsValid(initial_catalog_effective_xmin) &&
    1646          44 :                              TransactionIdPrecedesOrEquals(initial_catalog_effective_xmin,
    1647             :                                                            snapshotConflictHorizon))
    1648          24 :                         invalidation_cause = cause;
    1649          44 :                     break;
    1650           6 :                 case RS_INVAL_WAL_LEVEL:
    1651           6 :                     if (SlotIsLogical(s))
    1652           6 :                         invalidation_cause = cause;
    1653           6 :                     break;
    1654             :                 case RS_INVAL_NONE:
    1655             :                     pg_unreachable();
    1656             :             }
    1657          70 :         }
    1658             : 
    1659             :         /*
    1660             :          * The invalidation cause recorded previously should not change while
    1661             :          * the process owning the slot (if any) has been terminated.
    1662             :          */
    1663             :         Assert(!(invalidation_cause_prev != RS_INVAL_NONE && terminated &&
    1664             :                  invalidation_cause_prev != invalidation_cause));
    1665             : 
    1666             :         /* if there's no invalidation, we're done */
    1667         756 :         if (invalidation_cause == RS_INVAL_NONE)
    1668             :         {
    1669         714 :             SpinLockRelease(&s->mutex);
    1670         714 :             if (released_lock)
    1671           0 :                 LWLockRelease(ReplicationSlotControlLock);
    1672         714 :             break;
    1673             :         }
    1674             : 
    1675          42 :         slotname = s->data.name;
    1676          42 :         active_pid = s->active_pid;
    1677             : 
    1678             :         /*
    1679             :          * If the slot can be acquired, do so and mark it invalidated
    1680             :          * immediately.  Otherwise we'll signal the owning process, below, and
    1681             :          * retry.
    1682             :          */
    1683          42 :         if (active_pid == 0)
    1684             :         {
    1685          28 :             MyReplicationSlot = s;
    1686          28 :             s->active_pid = MyProcPid;
    1687          28 :             s->data.invalidated = invalidation_cause;
    1688             : 
    1689             :             /*
    1690             :              * XXX: We should consider not overwriting restart_lsn and instead
    1691             :              * just rely on .invalidated.
    1692             :              */
    1693          28 :             if (invalidation_cause == RS_INVAL_WAL_REMOVED)
    1694           8 :                 s->data.restart_lsn = InvalidXLogRecPtr;
    1695             : 
    1696             :             /* Let caller know */
    1697          28 :             *invalidated = true;
    1698             :         }
    1699             : 
    1700          42 :         SpinLockRelease(&s->mutex);
    1701             : 
    1702             :         /*
    1703             :          * The logical replication slots shouldn't be invalidated as GUC
    1704             :          * max_slot_wal_keep_size is set to -1 during the binary upgrade. See
    1705             :          * check_old_cluster_for_valid_slots() where we ensure that no
    1706             :          * invalidated before the upgrade.
    1707             :          */
    1708             :         Assert(!(*invalidated && SlotIsLogical(s) && IsBinaryUpgrade));
    1709             : 
    1710          42 :         if (active_pid != 0)
    1711             :         {
    1712             :             /*
    1713             :              * Prepare the sleep on the slot's condition variable before
    1714             :              * releasing the lock, to close a possible race condition if the
    1715             :              * slot is released before the sleep below.
    1716             :              */
    1717          14 :             ConditionVariablePrepareToSleep(&s->active_cv);
    1718             : 
    1719          14 :             LWLockRelease(ReplicationSlotControlLock);
    1720          14 :             released_lock = true;
    1721             : 
    1722             :             /*
    1723             :              * Signal to terminate the process that owns the slot, if we
    1724             :              * haven't already signalled it.  (Avoidance of repeated
    1725             :              * signalling is the only reason for there to be a loop in this
    1726             :              * routine; otherwise we could rely on caller's restart loop.)
    1727             :              *
    1728             :              * There is the race condition that other process may own the slot
    1729             :              * after its current owner process is terminated and before this
    1730             :              * process owns it. To handle that, we signal only if the PID of
    1731             :              * the owning process has changed from the previous time. (This
    1732             :              * logic assumes that the same PID is not reused very quickly.)
    1733             :              */
    1734          14 :             if (last_signaled_pid != active_pid)
    1735             :             {
    1736          14 :                 ReportSlotInvalidation(invalidation_cause, true, active_pid,
    1737             :                                        slotname, restart_lsn,
    1738             :                                        oldestLSN, snapshotConflictHorizon);
    1739             : 
    1740          14 :                 if (MyBackendType == B_STARTUP)
    1741          10 :                     (void) SendProcSignal(active_pid,
    1742             :                                           PROCSIG_RECOVERY_CONFLICT_LOGICALSLOT,
    1743             :                                           INVALID_PROC_NUMBER);
    1744             :                 else
    1745           4 :                     (void) kill(active_pid, SIGTERM);
    1746             : 
    1747          14 :                 last_signaled_pid = active_pid;
    1748          14 :                 terminated = true;
    1749          14 :                 invalidation_cause_prev = invalidation_cause;
    1750             :             }
    1751             : 
    1752             :             /* Wait until the slot is released. */
    1753          14 :             ConditionVariableSleep(&s->active_cv,
    1754             :                                    WAIT_EVENT_REPLICATION_SLOT_DROP);
    1755             : 
    1756             :             /*
    1757             :              * Re-acquire lock and start over; we expect to invalidate the
    1758             :              * slot next time (unless another process acquires the slot in the
    1759             :              * meantime).
    1760             :              */
    1761          14 :             LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
    1762          14 :             continue;
    1763             :         }
    1764             :         else
    1765             :         {
    1766             :             /*
    1767             :              * We hold the slot now and have already invalidated it; flush it
    1768             :              * to ensure that state persists.
    1769             :              *
    1770             :              * Don't want to hold ReplicationSlotControlLock across file
    1771             :              * system operations, so release it now but be sure to tell caller
    1772             :              * to restart from scratch.
    1773             :              */
    1774          28 :             LWLockRelease(ReplicationSlotControlLock);
    1775          28 :             released_lock = true;
    1776             : 
    1777             :             /* Make sure the invalidated state persists across server restart */
    1778          28 :             ReplicationSlotMarkDirty();
    1779          28 :             ReplicationSlotSave();
    1780          28 :             ReplicationSlotRelease();
    1781             : 
    1782          28 :             ReportSlotInvalidation(invalidation_cause, false, active_pid,
    1783             :                                    slotname, restart_lsn,
    1784             :                                    oldestLSN, snapshotConflictHorizon);
    1785             : 
    1786             :             /* done with this slot for now */
    1787          28 :             break;
    1788             :         }
    1789             :     }
    1790             : 
    1791             :     Assert(released_lock == !LWLockHeldByMe(ReplicationSlotControlLock));
    1792             : 
    1793         742 :     return released_lock;
    1794             : }
    1795             : 
    1796             : /*
    1797             :  * Invalidate slots that require resources about to be removed.
    1798             :  *
    1799             :  * Returns true when any slot have got invalidated.
    1800             :  *
    1801             :  * Whether a slot needs to be invalidated depends on the cause. A slot is
    1802             :  * removed if it:
    1803             :  * - RS_INVAL_WAL_REMOVED: requires a LSN older than the given segment
    1804             :  * - RS_INVAL_HORIZON: requires a snapshot <= the given horizon in the given
    1805             :  *   db; dboid may be InvalidOid for shared relations
    1806             :  * - RS_INVAL_WAL_LEVEL: is logical
    1807             :  *
    1808             :  * NB - this runs as part of checkpoint, so avoid raising errors if possible.
    1809             :  */
    1810             : bool
    1811        2512 : InvalidateObsoleteReplicationSlots(ReplicationSlotInvalidationCause cause,
    1812             :                                    XLogSegNo oldestSegno, Oid dboid,
    1813             :                                    TransactionId snapshotConflictHorizon)
    1814             : {
    1815             :     XLogRecPtr  oldestLSN;
    1816        2512 :     bool        invalidated = false;
    1817             : 
    1818             :     Assert(cause != RS_INVAL_HORIZON || TransactionIdIsValid(snapshotConflictHorizon));
    1819             :     Assert(cause != RS_INVAL_WAL_REMOVED || oldestSegno > 0);
    1820             :     Assert(cause != RS_INVAL_NONE);
    1821             : 
    1822        2512 :     if (max_replication_slots == 0)
    1823           2 :         return invalidated;
    1824             : 
    1825        2510 :     XLogSegNoOffsetToRecPtr(oldestSegno, 0, wal_segment_size, oldestLSN);
    1826             : 
    1827        2538 : restart:
    1828        2538 :     LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
    1829       26946 :     for (int i = 0; i < max_replication_slots; i++)
    1830             :     {
    1831       24436 :         ReplicationSlot *s = &ReplicationSlotCtl->replication_slots[i];
    1832             : 
    1833       24436 :         if (!s->in_use)
    1834       23694 :             continue;
    1835             : 
    1836         742 :         if (InvalidatePossiblyObsoleteSlot(cause, s, oldestLSN, dboid,
    1837             :                                            snapshotConflictHorizon,
    1838             :                                            &invalidated))
    1839             :         {
    1840             :             /* if the lock was released, start from scratch */
    1841          28 :             goto restart;
    1842             :         }
    1843             :     }
    1844        2510 :     LWLockRelease(ReplicationSlotControlLock);
    1845             : 
    1846             :     /*
    1847             :      * If any slots have been invalidated, recalculate the resource limits.
    1848             :      */
    1849        2510 :     if (invalidated)
    1850             :     {
    1851          18 :         ReplicationSlotsComputeRequiredXmin(false);
    1852          18 :         ReplicationSlotsComputeRequiredLSN();
    1853             :     }
    1854             : 
    1855        2510 :     return invalidated;
    1856             : }
    1857             : 
    1858             : /*
    1859             :  * Flush all replication slots to disk.
    1860             :  *
    1861             :  * It is convenient to flush dirty replication slots at the time of checkpoint.
    1862             :  * Additionally, in case of a shutdown checkpoint, we also identify the slots
    1863             :  * for which the confirmed_flush LSN has been updated since the last time it
    1864             :  * was saved and flush them.
    1865             :  */
    1866             : void
    1867        2476 : CheckPointReplicationSlots(bool is_shutdown)
    1868             : {
    1869             :     int         i;
    1870             : 
    1871        2476 :     elog(DEBUG1, "performing replication slot checkpoint");
    1872             : 
    1873             :     /*
    1874             :      * Prevent any slot from being created/dropped while we're active. As we
    1875             :      * explicitly do *not* want to block iterating over replication_slots or
    1876             :      * acquiring a slot we cannot take the control lock - but that's OK,
    1877             :      * because holding ReplicationSlotAllocationLock is strictly stronger, and
    1878             :      * enough to guarantee that nobody can change the in_use bits on us.
    1879             :      */
    1880        2476 :     LWLockAcquire(ReplicationSlotAllocationLock, LW_SHARED);
    1881             : 
    1882       26682 :     for (i = 0; i < max_replication_slots; i++)
    1883             :     {
    1884       24206 :         ReplicationSlot *s = &ReplicationSlotCtl->replication_slots[i];
    1885             :         char        path[MAXPGPATH];
    1886             : 
    1887       24206 :         if (!s->in_use)
    1888       23558 :             continue;
    1889             : 
    1890             :         /* save the slot to disk, locking is handled in SaveSlotToPath() */
    1891         648 :         sprintf(path, "%s/%s", PG_REPLSLOT_DIR, NameStr(s->data.name));
    1892             : 
    1893             :         /*
    1894             :          * Slot's data is not flushed each time the confirmed_flush LSN is
    1895             :          * updated as that could lead to frequent writes.  However, we decide
    1896             :          * to force a flush of all logical slot's data at the time of shutdown
    1897             :          * if the confirmed_flush LSN is changed since we last flushed it to
    1898             :          * disk.  This helps in avoiding an unnecessary retreat of the
    1899             :          * confirmed_flush LSN after restart.
    1900             :          */
    1901         648 :         if (is_shutdown && SlotIsLogical(s))
    1902             :         {
    1903         122 :             SpinLockAcquire(&s->mutex);
    1904             : 
    1905         122 :             if (s->data.invalidated == RS_INVAL_NONE &&
    1906         122 :                 s->data.confirmed_flush > s->last_saved_confirmed_flush)
    1907             :             {
    1908          70 :                 s->just_dirtied = true;
    1909          70 :                 s->dirty = true;
    1910             :             }
    1911         122 :             SpinLockRelease(&s->mutex);
    1912             :         }
    1913             : 
    1914         648 :         SaveSlotToPath(s, path, LOG);
    1915             :     }
    1916        2476 :     LWLockRelease(ReplicationSlotAllocationLock);
    1917        2476 : }
    1918             : 
    1919             : /*
    1920             :  * Load all replication slots from disk into memory at server startup. This
    1921             :  * needs to be run before we start crash recovery.
    1922             :  */
    1923             : void
    1924        1650 : StartupReplicationSlots(void)
    1925             : {
    1926             :     DIR        *replication_dir;
    1927             :     struct dirent *replication_de;
    1928             : 
    1929        1650 :     elog(DEBUG1, "starting up replication slots");
    1930             : 
    1931             :     /* restore all slots by iterating over all on-disk entries */
    1932        1650 :     replication_dir = AllocateDir(PG_REPLSLOT_DIR);
    1933        5086 :     while ((replication_de = ReadDir(replication_dir, PG_REPLSLOT_DIR)) != NULL)
    1934             :     {
    1935             :         char        path[MAXPGPATH + sizeof(PG_REPLSLOT_DIR)];
    1936             :         PGFileType  de_type;
    1937             : 
    1938        3436 :         if (strcmp(replication_de->d_name, ".") == 0 ||
    1939        1786 :             strcmp(replication_de->d_name, "..") == 0)
    1940        3300 :             continue;
    1941             : 
    1942         136 :         snprintf(path, sizeof(path), "%s/%s", PG_REPLSLOT_DIR, replication_de->d_name);
    1943         136 :         de_type = get_dirent_type(path, replication_de, false, DEBUG1);
    1944             : 
    1945             :         /* we're only creating directories here, skip if it's not our's */
    1946         136 :         if (de_type != PGFILETYPE_ERROR && de_type != PGFILETYPE_DIR)
    1947           0 :             continue;
    1948             : 
    1949             :         /* we crashed while a slot was being setup or deleted, clean up */
    1950         136 :         if (pg_str_endswith(replication_de->d_name, ".tmp"))
    1951             :         {
    1952           0 :             if (!rmtree(path, true))
    1953             :             {
    1954           0 :                 ereport(WARNING,
    1955             :                         (errmsg("could not remove directory \"%s\"",
    1956             :                                 path)));
    1957           0 :                 continue;
    1958             :             }
    1959           0 :             fsync_fname(PG_REPLSLOT_DIR, true);
    1960           0 :             continue;
    1961             :         }
    1962             : 
    1963             :         /* looks like a slot in a normal state, restore */
    1964         136 :         RestoreSlotFromDisk(replication_de->d_name);
    1965             :     }
    1966        1650 :     FreeDir(replication_dir);
    1967             : 
    1968             :     /* currently no slots exist, we're done. */
    1969        1650 :     if (max_replication_slots <= 0)
    1970           2 :         return;
    1971             : 
    1972             :     /* Now that we have recovered all the data, compute replication xmin */
    1973        1648 :     ReplicationSlotsComputeRequiredXmin(false);
    1974        1648 :     ReplicationSlotsComputeRequiredLSN();
    1975             : }
    1976             : 
    1977             : /* ----
    1978             :  * Manipulation of on-disk state of replication slots
    1979             :  *
    1980             :  * NB: none of the routines below should take any notice whether a slot is the
    1981             :  * current one or not, that's all handled a layer above.
    1982             :  * ----
    1983             :  */
    1984             : static void
    1985        1204 : CreateSlotOnDisk(ReplicationSlot *slot)
    1986             : {
    1987             :     char        tmppath[MAXPGPATH];
    1988             :     char        path[MAXPGPATH];
    1989             :     struct stat st;
    1990             : 
    1991             :     /*
    1992             :      * No need to take out the io_in_progress_lock, nobody else can see this
    1993             :      * slot yet, so nobody else will write. We're reusing SaveSlotToPath which
    1994             :      * takes out the lock, if we'd take the lock here, we'd deadlock.
    1995             :      */
    1996             : 
    1997        1204 :     sprintf(path, "%s/%s", PG_REPLSLOT_DIR, NameStr(slot->data.name));
    1998        1204 :     sprintf(tmppath, "%s/%s.tmp", PG_REPLSLOT_DIR, NameStr(slot->data.name));
    1999             : 
    2000             :     /*
    2001             :      * It's just barely possible that some previous effort to create or drop a
    2002             :      * slot with this name left a temp directory lying around. If that seems
    2003             :      * to be the case, try to remove it.  If the rmtree() fails, we'll error
    2004             :      * out at the MakePGDirectory() below, so we don't bother checking
    2005             :      * success.
    2006             :      */
    2007        1204 :     if (stat(tmppath, &st) == 0 && S_ISDIR(st.st_mode))
    2008           0 :         rmtree(tmppath, true);
    2009             : 
    2010             :     /* Create and fsync the temporary slot directory. */
    2011        1204 :     if (MakePGDirectory(tmppath) < 0)
    2012           0 :         ereport(ERROR,
    2013             :                 (errcode_for_file_access(),
    2014             :                  errmsg("could not create directory \"%s\": %m",
    2015             :                         tmppath)));
    2016        1204 :     fsync_fname(tmppath, true);
    2017             : 
    2018             :     /* Write the actual state file. */
    2019        1204 :     slot->dirty = true;          /* signal that we really need to write */
    2020        1204 :     SaveSlotToPath(slot, tmppath, ERROR);
    2021             : 
    2022             :     /* Rename the directory into place. */
    2023        1204 :     if (rename(tmppath, path) != 0)
    2024           0 :         ereport(ERROR,
    2025             :                 (errcode_for_file_access(),
    2026             :                  errmsg("could not rename file \"%s\" to \"%s\": %m",
    2027             :                         tmppath, path)));
    2028             : 
    2029             :     /*
    2030             :      * If we'd now fail - really unlikely - we wouldn't know whether this slot
    2031             :      * would persist after an OS crash or not - so, force a restart. The
    2032             :      * restart would try to fsync this again till it works.
    2033             :      */
    2034        1204 :     START_CRIT_SECTION();
    2035             : 
    2036        1204 :     fsync_fname(path, true);
    2037        1204 :     fsync_fname(PG_REPLSLOT_DIR, true);
    2038             : 
    2039        1204 :     END_CRIT_SECTION();
    2040        1204 : }
    2041             : 
    2042             : /*
    2043             :  * Shared functionality between saving and creating a replication slot.
    2044             :  */
    2045             : static void
    2046        4274 : SaveSlotToPath(ReplicationSlot *slot, const char *dir, int elevel)
    2047             : {
    2048             :     char        tmppath[MAXPGPATH];
    2049             :     char        path[MAXPGPATH];
    2050             :     int         fd;
    2051             :     ReplicationSlotOnDisk cp;
    2052             :     bool        was_dirty;
    2053             : 
    2054             :     /* first check whether there's something to write out */
    2055        4274 :     SpinLockAcquire(&slot->mutex);
    2056        4274 :     was_dirty = slot->dirty;
    2057        4274 :     slot->just_dirtied = false;
    2058        4274 :     SpinLockRelease(&slot->mutex);
    2059             : 
    2060             :     /* and don't do anything if there's nothing to write */
    2061        4274 :     if (!was_dirty)
    2062         194 :         return;
    2063             : 
    2064        4080 :     LWLockAcquire(&slot->io_in_progress_lock, LW_EXCLUSIVE);
    2065             : 
    2066             :     /* silence valgrind :( */
    2067        4080 :     memset(&cp, 0, sizeof(ReplicationSlotOnDisk));
    2068             : 
    2069        4080 :     sprintf(tmppath, "%s/state.tmp", dir);
    2070        4080 :     sprintf(path, "%s/state", dir);
    2071             : 
    2072        4080 :     fd = OpenTransientFile(tmppath, O_CREAT | O_EXCL | O_WRONLY | PG_BINARY);
    2073        4080 :     if (fd < 0)
    2074             :     {
    2075             :         /*
    2076             :          * If not an ERROR, then release the lock before returning.  In case
    2077             :          * of an ERROR, the error recovery path automatically releases the
    2078             :          * lock, but no harm in explicitly releasing even in that case.  Note
    2079             :          * that LWLockRelease() could affect errno.
    2080             :          */
    2081           0 :         int         save_errno = errno;
    2082             : 
    2083           0 :         LWLockRelease(&slot->io_in_progress_lock);
    2084           0 :         errno = save_errno;
    2085           0 :         ereport(elevel,
    2086             :                 (errcode_for_file_access(),
    2087             :                  errmsg("could not create file \"%s\": %m",
    2088             :                         tmppath)));
    2089           0 :         return;
    2090             :     }
    2091             : 
    2092        4080 :     cp.magic = SLOT_MAGIC;
    2093        4080 :     INIT_CRC32C(cp.checksum);
    2094        4080 :     cp.version = SLOT_VERSION;
    2095        4080 :     cp.length = ReplicationSlotOnDiskV2Size;
    2096             : 
    2097        4080 :     SpinLockAcquire(&slot->mutex);
    2098             : 
    2099        4080 :     memcpy(&cp.slotdata, &slot->data, sizeof(ReplicationSlotPersistentData));
    2100             : 
    2101        4080 :     SpinLockRelease(&slot->mutex);
    2102             : 
    2103        4080 :     COMP_CRC32C(cp.checksum,
    2104             :                 (char *) (&cp) + ReplicationSlotOnDiskNotChecksummedSize,
    2105             :                 ReplicationSlotOnDiskChecksummedSize);
    2106        4080 :     FIN_CRC32C(cp.checksum);
    2107             : 
    2108        4080 :     errno = 0;
    2109        4080 :     pgstat_report_wait_start(WAIT_EVENT_REPLICATION_SLOT_WRITE);
    2110        4080 :     if ((write(fd, &cp, sizeof(cp))) != sizeof(cp))
    2111             :     {
    2112           0 :         int         save_errno = errno;
    2113             : 
    2114           0 :         pgstat_report_wait_end();
    2115           0 :         CloseTransientFile(fd);
    2116           0 :         LWLockRelease(&slot->io_in_progress_lock);
    2117             : 
    2118             :         /* if write didn't set errno, assume problem is no disk space */
    2119           0 :         errno = save_errno ? save_errno : ENOSPC;
    2120           0 :         ereport(elevel,
    2121             :                 (errcode_for_file_access(),
    2122             :                  errmsg("could not write to file \"%s\": %m",
    2123             :                         tmppath)));
    2124           0 :         return;
    2125             :     }
    2126        4080 :     pgstat_report_wait_end();
    2127             : 
    2128             :     /* fsync the temporary file */
    2129        4080 :     pgstat_report_wait_start(WAIT_EVENT_REPLICATION_SLOT_SYNC);
    2130        4080 :     if (pg_fsync(fd) != 0)
    2131             :     {
    2132           0 :         int         save_errno = errno;
    2133             : 
    2134           0 :         pgstat_report_wait_end();
    2135           0 :         CloseTransientFile(fd);
    2136           0 :         LWLockRelease(&slot->io_in_progress_lock);
    2137           0 :         errno = save_errno;
    2138           0 :         ereport(elevel,
    2139             :                 (errcode_for_file_access(),
    2140             :                  errmsg("could not fsync file \"%s\": %m",
    2141             :                         tmppath)));
    2142           0 :         return;
    2143             :     }
    2144        4080 :     pgstat_report_wait_end();
    2145             : 
    2146        4080 :     if (CloseTransientFile(fd) != 0)
    2147             :     {
    2148           0 :         int         save_errno = errno;
    2149             : 
    2150           0 :         LWLockRelease(&slot->io_in_progress_lock);
    2151           0 :         errno = save_errno;
    2152           0 :         ereport(elevel,
    2153             :                 (errcode_for_file_access(),
    2154             :                  errmsg("could not close file \"%s\": %m",
    2155             :                         tmppath)));
    2156           0 :         return;
    2157             :     }
    2158             : 
    2159             :     /* rename to permanent file, fsync file and directory */
    2160        4080 :     if (rename(tmppath, path) != 0)
    2161             :     {
    2162           0 :         int         save_errno = errno;
    2163             : 
    2164           0 :         LWLockRelease(&slot->io_in_progress_lock);
    2165           0 :         errno = save_errno;
    2166           0 :         ereport(elevel,
    2167             :                 (errcode_for_file_access(),
    2168             :                  errmsg("could not rename file \"%s\" to \"%s\": %m",
    2169             :                         tmppath, path)));
    2170           0 :         return;
    2171             :     }
    2172             : 
    2173             :     /*
    2174             :      * Check CreateSlotOnDisk() for the reasoning of using a critical section.
    2175             :      */
    2176        4080 :     START_CRIT_SECTION();
    2177             : 
    2178        4080 :     fsync_fname(path, false);
    2179        4080 :     fsync_fname(dir, true);
    2180        4080 :     fsync_fname(PG_REPLSLOT_DIR, true);
    2181             : 
    2182        4080 :     END_CRIT_SECTION();
    2183             : 
    2184             :     /*
    2185             :      * Successfully wrote, unset dirty bit, unless somebody dirtied again
    2186             :      * already and remember the confirmed_flush LSN value.
    2187             :      */
    2188        4080 :     SpinLockAcquire(&slot->mutex);
    2189        4080 :     if (!slot->just_dirtied)
    2190        4064 :         slot->dirty = false;
    2191        4080 :     slot->last_saved_confirmed_flush = cp.slotdata.confirmed_flush;
    2192        4080 :     SpinLockRelease(&slot->mutex);
    2193             : 
    2194        4080 :     LWLockRelease(&slot->io_in_progress_lock);
    2195             : }
    2196             : 
    2197             : /*
    2198             :  * Load a single slot from disk into memory.
    2199             :  */
    2200             : static void
    2201         136 : RestoreSlotFromDisk(const char *name)
    2202             : {
    2203             :     ReplicationSlotOnDisk cp;
    2204             :     int         i;
    2205             :     char        slotdir[MAXPGPATH + sizeof(PG_REPLSLOT_DIR)];
    2206             :     char        path[MAXPGPATH + sizeof(PG_REPLSLOT_DIR) + 10];
    2207             :     int         fd;
    2208         136 :     bool        restored = false;
    2209             :     int         readBytes;
    2210             :     pg_crc32c   checksum;
    2211             : 
    2212             :     /* no need to lock here, no concurrent access allowed yet */
    2213             : 
    2214             :     /* delete temp file if it exists */
    2215         136 :     sprintf(slotdir, "%s/%s", PG_REPLSLOT_DIR, name);
    2216         136 :     sprintf(path, "%s/state.tmp", slotdir);
    2217         136 :     if (unlink(path) < 0 && errno != ENOENT)
    2218           0 :         ereport(PANIC,
    2219             :                 (errcode_for_file_access(),
    2220             :                  errmsg("could not remove file \"%s\": %m", path)));
    2221             : 
    2222         136 :     sprintf(path, "%s/state", slotdir);
    2223             : 
    2224         136 :     elog(DEBUG1, "restoring replication slot from \"%s\"", path);
    2225             : 
    2226             :     /* on some operating systems fsyncing a file requires O_RDWR */
    2227         136 :     fd = OpenTransientFile(path, O_RDWR | PG_BINARY);
    2228             : 
    2229             :     /*
    2230             :      * We do not need to handle this as we are rename()ing the directory into
    2231             :      * place only after we fsync()ed the state file.
    2232             :      */
    2233         136 :     if (fd < 0)
    2234           0 :         ereport(PANIC,
    2235             :                 (errcode_for_file_access(),
    2236             :                  errmsg("could not open file \"%s\": %m", path)));
    2237             : 
    2238             :     /*
    2239             :      * Sync state file before we're reading from it. We might have crashed
    2240             :      * while it wasn't synced yet and we shouldn't continue on that basis.
    2241             :      */
    2242         136 :     pgstat_report_wait_start(WAIT_EVENT_REPLICATION_SLOT_RESTORE_SYNC);
    2243         136 :     if (pg_fsync(fd) != 0)
    2244           0 :         ereport(PANIC,
    2245             :                 (errcode_for_file_access(),
    2246             :                  errmsg("could not fsync file \"%s\": %m",
    2247             :                         path)));
    2248         136 :     pgstat_report_wait_end();
    2249             : 
    2250             :     /* Also sync the parent directory */
    2251         136 :     START_CRIT_SECTION();
    2252         136 :     fsync_fname(slotdir, true);
    2253         136 :     END_CRIT_SECTION();
    2254             : 
    2255             :     /* read part of statefile that's guaranteed to be version independent */
    2256         136 :     pgstat_report_wait_start(WAIT_EVENT_REPLICATION_SLOT_READ);
    2257         136 :     readBytes = read(fd, &cp, ReplicationSlotOnDiskConstantSize);
    2258         136 :     pgstat_report_wait_end();
    2259         136 :     if (readBytes != ReplicationSlotOnDiskConstantSize)
    2260             :     {
    2261           0 :         if (readBytes < 0)
    2262           0 :             ereport(PANIC,
    2263             :                     (errcode_for_file_access(),
    2264             :                      errmsg("could not read file \"%s\": %m", path)));
    2265             :         else
    2266           0 :             ereport(PANIC,
    2267             :                     (errcode(ERRCODE_DATA_CORRUPTED),
    2268             :                      errmsg("could not read file \"%s\": read %d of %zu",
    2269             :                             path, readBytes,
    2270             :                             (Size) ReplicationSlotOnDiskConstantSize)));
    2271             :     }
    2272             : 
    2273             :     /* verify magic */
    2274         136 :     if (cp.magic != SLOT_MAGIC)
    2275           0 :         ereport(PANIC,
    2276             :                 (errcode(ERRCODE_DATA_CORRUPTED),
    2277             :                  errmsg("replication slot file \"%s\" has wrong magic number: %u instead of %u",
    2278             :                         path, cp.magic, SLOT_MAGIC)));
    2279             : 
    2280             :     /* verify version */
    2281         136 :     if (cp.version != SLOT_VERSION)
    2282           0 :         ereport(PANIC,
    2283             :                 (errcode(ERRCODE_DATA_CORRUPTED),
    2284             :                  errmsg("replication slot file \"%s\" has unsupported version %u",
    2285             :                         path, cp.version)));
    2286             : 
    2287             :     /* boundary check on length */
    2288         136 :     if (cp.length != ReplicationSlotOnDiskV2Size)
    2289           0 :         ereport(PANIC,
    2290             :                 (errcode(ERRCODE_DATA_CORRUPTED),
    2291             :                  errmsg("replication slot file \"%s\" has corrupted length %u",
    2292             :                         path, cp.length)));
    2293             : 
    2294             :     /* Now that we know the size, read the entire file */
    2295         136 :     pgstat_report_wait_start(WAIT_EVENT_REPLICATION_SLOT_READ);
    2296         272 :     readBytes = read(fd,
    2297             :                      (char *) &cp + ReplicationSlotOnDiskConstantSize,
    2298         136 :                      cp.length);
    2299         136 :     pgstat_report_wait_end();
    2300         136 :     if (readBytes != cp.length)
    2301             :     {
    2302           0 :         if (readBytes < 0)
    2303           0 :             ereport(PANIC,
    2304             :                     (errcode_for_file_access(),
    2305             :                      errmsg("could not read file \"%s\": %m", path)));
    2306             :         else
    2307           0 :             ereport(PANIC,
    2308             :                     (errcode(ERRCODE_DATA_CORRUPTED),
    2309             :                      errmsg("could not read file \"%s\": read %d of %zu",
    2310             :                             path, readBytes, (Size) cp.length)));
    2311             :     }
    2312             : 
    2313         136 :     if (CloseTransientFile(fd) != 0)
    2314           0 :         ereport(PANIC,
    2315             :                 (errcode_for_file_access(),
    2316             :                  errmsg("could not close file \"%s\": %m", path)));
    2317             : 
    2318             :     /* now verify the CRC */
    2319         136 :     INIT_CRC32C(checksum);
    2320         136 :     COMP_CRC32C(checksum,
    2321             :                 (char *) &cp + ReplicationSlotOnDiskNotChecksummedSize,
    2322             :                 ReplicationSlotOnDiskChecksummedSize);
    2323         136 :     FIN_CRC32C(checksum);
    2324             : 
    2325         136 :     if (!EQ_CRC32C(checksum, cp.checksum))
    2326           0 :         ereport(PANIC,
    2327             :                 (errmsg("checksum mismatch for replication slot file \"%s\": is %u, should be %u",
    2328             :                         path, checksum, cp.checksum)));
    2329             : 
    2330             :     /*
    2331             :      * If we crashed with an ephemeral slot active, don't restore but delete
    2332             :      * it.
    2333             :      */
    2334         136 :     if (cp.slotdata.persistency != RS_PERSISTENT)
    2335             :     {
    2336           0 :         if (!rmtree(slotdir, true))
    2337             :         {
    2338           0 :             ereport(WARNING,
    2339             :                     (errmsg("could not remove directory \"%s\"",
    2340             :                             slotdir)));
    2341             :         }
    2342           0 :         fsync_fname(PG_REPLSLOT_DIR, true);
    2343           0 :         return;
    2344             :     }
    2345             : 
    2346             :     /*
    2347             :      * Verify that requirements for the specific slot type are met. That's
    2348             :      * important because if these aren't met we're not guaranteed to retain
    2349             :      * all the necessary resources for the slot.
    2350             :      *
    2351             :      * NB: We have to do so *after* the above checks for ephemeral slots,
    2352             :      * because otherwise a slot that shouldn't exist anymore could prevent
    2353             :      * restarts.
    2354             :      *
    2355             :      * NB: Changing the requirements here also requires adapting
    2356             :      * CheckSlotRequirements() and CheckLogicalDecodingRequirements().
    2357             :      */
    2358         136 :     if (cp.slotdata.database != InvalidOid && wal_level < WAL_LEVEL_LOGICAL)
    2359           0 :         ereport(FATAL,
    2360             :                 (errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
    2361             :                  errmsg("logical replication slot \"%s\" exists, but \"wal_level\" < \"logical\"",
    2362             :                         NameStr(cp.slotdata.name)),
    2363             :                  errhint("Change \"wal_level\" to be \"logical\" or higher.")));
    2364         136 :     else if (wal_level < WAL_LEVEL_REPLICA)
    2365           0 :         ereport(FATAL,
    2366             :                 (errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
    2367             :                  errmsg("physical replication slot \"%s\" exists, but \"wal_level\" < \"replica\"",
    2368             :                         NameStr(cp.slotdata.name)),
    2369             :                  errhint("Change \"wal_level\" to be \"replica\" or higher.")));
    2370             : 
    2371             :     /* nothing can be active yet, don't lock anything */
    2372         188 :     for (i = 0; i < max_replication_slots; i++)
    2373             :     {
    2374             :         ReplicationSlot *slot;
    2375             : 
    2376         188 :         slot = &ReplicationSlotCtl->replication_slots[i];
    2377             : 
    2378         188 :         if (slot->in_use)
    2379          52 :             continue;
    2380             : 
    2381             :         /* restore the entire set of persistent data */
    2382         136 :         memcpy(&slot->data, &cp.slotdata,
    2383             :                sizeof(ReplicationSlotPersistentData));
    2384             : 
    2385             :         /* initialize in memory state */
    2386         136 :         slot->effective_xmin = cp.slotdata.xmin;
    2387         136 :         slot->effective_catalog_xmin = cp.slotdata.catalog_xmin;
    2388         136 :         slot->last_saved_confirmed_flush = cp.slotdata.confirmed_flush;
    2389             : 
    2390         136 :         slot->candidate_catalog_xmin = InvalidTransactionId;
    2391         136 :         slot->candidate_xmin_lsn = InvalidXLogRecPtr;
    2392         136 :         slot->candidate_restart_lsn = InvalidXLogRecPtr;
    2393         136 :         slot->candidate_restart_valid = InvalidXLogRecPtr;
    2394             : 
    2395         136 :         slot->in_use = true;
    2396         136 :         slot->active_pid = 0;
    2397             : 
    2398             :         /*
    2399             :          * Set the time since the slot has become inactive after loading the
    2400             :          * slot from the disk into memory. Whoever acquires the slot i.e.
    2401             :          * makes the slot active will reset it.
    2402             :          */
    2403         136 :         slot->inactive_since = GetCurrentTimestamp();
    2404             : 
    2405         136 :         restored = true;
    2406         136 :         break;
    2407             :     }
    2408             : 
    2409         136 :     if (!restored)
    2410           0 :         ereport(FATAL,
    2411             :                 (errmsg("too many replication slots active before shutdown"),
    2412             :                  errhint("Increase \"max_replication_slots\" and try again.")));
    2413             : }
    2414             : 
    2415             : /*
    2416             :  * Maps an invalidation reason for a replication slot to
    2417             :  * ReplicationSlotInvalidationCause.
    2418             :  */
    2419             : ReplicationSlotInvalidationCause
    2420           0 : GetSlotInvalidationCause(const char *invalidation_reason)
    2421             : {
    2422             :     ReplicationSlotInvalidationCause cause;
    2423           0 :     ReplicationSlotInvalidationCause result = RS_INVAL_NONE;
    2424           0 :     bool        found PG_USED_FOR_ASSERTS_ONLY = false;
    2425             : 
    2426             :     Assert(invalidation_reason);
    2427             : 
    2428           0 :     for (cause = RS_INVAL_NONE; cause <= RS_INVAL_MAX_CAUSES; cause++)
    2429             :     {
    2430           0 :         if (strcmp(SlotInvalidationCauses[cause], invalidation_reason) == 0)
    2431             :         {
    2432           0 :             found = true;
    2433           0 :             result = cause;
    2434           0 :             break;
    2435             :         }
    2436             :     }
    2437             : 
    2438             :     Assert(found);
    2439           0 :     return result;
    2440             : }
    2441             : 
    2442             : /*
    2443             :  * A helper function to validate slots specified in GUC synchronized_standby_slots.
    2444             :  *
    2445             :  * The rawname will be parsed, and the result will be saved into *elemlist.
    2446             :  */
    2447             : static bool
    2448          12 : validate_sync_standby_slots(char *rawname, List **elemlist)
    2449             : {
    2450             :     bool        ok;
    2451             : 
    2452             :     /* Verify syntax and parse string into a list of identifiers */
    2453          12 :     ok = SplitIdentifierString(rawname, ',', elemlist);
    2454             : 
    2455          12 :     if (!ok)
    2456             :     {
    2457           0 :         GUC_check_errdetail("List syntax is invalid.");
    2458             :     }
    2459          12 :     else if (MyProc)
    2460             :     {
    2461             :         /*
    2462             :          * Check that each specified slot exist and is physical.
    2463             :          *
    2464             :          * Because we need an LWLock, we cannot do this on processes without a
    2465             :          * PGPROC, so we skip it there; but see comments in
    2466             :          * StandbySlotsHaveCaughtup() as to why that's not a problem.
    2467             :          */
    2468           6 :         LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
    2469             : 
    2470          18 :         foreach_ptr(char, name, *elemlist)
    2471             :         {
    2472             :             ReplicationSlot *slot;
    2473             : 
    2474           6 :             slot = SearchNamedReplicationSlot(name, false);
    2475             : 
    2476           6 :             if (!slot)
    2477             :             {
    2478           0 :                 GUC_check_errdetail("Replication slot \"%s\" does not exist.",
    2479             :                                     name);
    2480           0 :                 ok = false;
    2481           0 :                 break;
    2482             :             }
    2483             : 
    2484           6 :             if (!SlotIsPhysical(slot))
    2485             :             {
    2486           0 :                 GUC_check_errdetail("\"%s\" is not a physical replication slot.",
    2487             :                                     name);
    2488           0 :                 ok = false;
    2489           0 :                 break;
    2490             :             }
    2491             :         }
    2492             : 
    2493           6 :         LWLockRelease(ReplicationSlotControlLock);
    2494             :     }
    2495             : 
    2496          12 :     return ok;
    2497             : }
    2498             : 
    2499             : /*
    2500             :  * GUC check_hook for synchronized_standby_slots
    2501             :  */
    2502             : bool
    2503        2002 : check_synchronized_standby_slots(char **newval, void **extra, GucSource source)
    2504             : {
    2505             :     char       *rawname;
    2506             :     char       *ptr;
    2507             :     List       *elemlist;
    2508             :     int         size;
    2509             :     bool        ok;
    2510             :     SyncStandbySlotsConfigData *config;
    2511             : 
    2512        2002 :     if ((*newval)[0] == '\0')
    2513        1990 :         return true;
    2514             : 
    2515             :     /* Need a modifiable copy of the GUC string */
    2516          12 :     rawname = pstrdup(*newval);
    2517             : 
    2518             :     /* Now verify if the specified slots exist and have correct type */
    2519          12 :     ok = validate_sync_standby_slots(rawname, &elemlist);
    2520             : 
    2521          12 :     if (!ok || elemlist == NIL)
    2522             :     {
    2523           0 :         pfree(rawname);
    2524           0 :         list_free(elemlist);
    2525           0 :         return ok;
    2526             :     }
    2527             : 
    2528             :     /* Compute the size required for the SyncStandbySlotsConfigData struct */
    2529          12 :     size = offsetof(SyncStandbySlotsConfigData, slot_names);
    2530          36 :     foreach_ptr(char, slot_name, elemlist)
    2531          12 :         size += strlen(slot_name) + 1;
    2532             : 
    2533             :     /* GUC extra value must be guc_malloc'd, not palloc'd */
    2534          12 :     config = (SyncStandbySlotsConfigData *) guc_malloc(LOG, size);
    2535             : 
    2536             :     /* Transform the data into SyncStandbySlotsConfigData */
    2537          12 :     config->nslotnames = list_length(elemlist);
    2538             : 
    2539          12 :     ptr = config->slot_names;
    2540          36 :     foreach_ptr(char, slot_name, elemlist)
    2541             :     {
    2542          12 :         strcpy(ptr, slot_name);
    2543          12 :         ptr += strlen(slot_name) + 1;
    2544             :     }
    2545             : 
    2546          12 :     *extra = config;
    2547             : 
    2548          12 :     pfree(rawname);
    2549          12 :     list_free(elemlist);
    2550          12 :     return true;
    2551             : }
    2552             : 
    2553             : /*
    2554             :  * GUC assign_hook for synchronized_standby_slots
    2555             :  */
    2556             : void
    2557        2002 : assign_synchronized_standby_slots(const char *newval, void *extra)
    2558             : {
    2559             :     /*
    2560             :      * The standby slots may have changed, so we must recompute the oldest
    2561             :      * LSN.
    2562             :      */
    2563        2002 :     ss_oldest_flush_lsn = InvalidXLogRecPtr;
    2564             : 
    2565        2002 :     synchronized_standby_slots_config = (SyncStandbySlotsConfigData *) extra;
    2566        2002 : }
    2567             : 
    2568             : /*
    2569             :  * Check if the passed slot_name is specified in the synchronized_standby_slots GUC.
    2570             :  */
    2571             : bool
    2572        8204 : SlotExistsInSyncStandbySlots(const char *slot_name)
    2573             : {
    2574             :     const char *standby_slot_name;
    2575             : 
    2576             :     /* Return false if there is no value in synchronized_standby_slots */
    2577        8204 :     if (synchronized_standby_slots_config == NULL)
    2578        8194 :         return false;
    2579             : 
    2580             :     /*
    2581             :      * XXX: We are not expecting this list to be long so a linear search
    2582             :      * shouldn't hurt but if that turns out not to be true then we can cache
    2583             :      * this information for each WalSender as well.
    2584             :      */
    2585          10 :     standby_slot_name = synchronized_standby_slots_config->slot_names;
    2586          10 :     for (int i = 0; i < synchronized_standby_slots_config->nslotnames; i++)
    2587             :     {
    2588          10 :         if (strcmp(standby_slot_name, slot_name) == 0)
    2589          10 :             return true;
    2590             : 
    2591           0 :         standby_slot_name += strlen(standby_slot_name) + 1;
    2592             :     }
    2593             : 
    2594           0 :     return false;
    2595             : }
    2596             : 
    2597             : /*
    2598             :  * Return true if the slots specified in synchronized_standby_slots have caught up to
    2599             :  * the given WAL location, false otherwise.
    2600             :  *
    2601             :  * The elevel parameter specifies the error level used for logging messages
    2602             :  * related to slots that do not exist, are invalidated, or are inactive.
    2603             :  */
    2604             : bool
    2605        1218 : StandbySlotsHaveCaughtup(XLogRecPtr wait_for_lsn, int elevel)
    2606             : {
    2607             :     const char *name;
    2608        1218 :     int         caught_up_slot_num = 0;
    2609        1218 :     XLogRecPtr  min_restart_lsn = InvalidXLogRecPtr;
    2610             : 
    2611             :     /*
    2612             :      * Don't need to wait for the standbys to catch up if there is no value in
    2613             :      * synchronized_standby_slots.
    2614             :      */
    2615        1218 :     if (synchronized_standby_slots_config == NULL)
    2616        1192 :         return true;
    2617             : 
    2618             :     /*
    2619             :      * Don't need to wait for the standbys to catch up if we are on a standby
    2620             :      * server, since we do not support syncing slots to cascading standbys.
    2621             :      */
    2622          26 :     if (RecoveryInProgress())
    2623           0 :         return true;
    2624             : 
    2625             :     /*
    2626             :      * Don't need to wait for the standbys to catch up if they are already
    2627             :      * beyond the specified WAL location.
    2628             :      */
    2629          26 :     if (!XLogRecPtrIsInvalid(ss_oldest_flush_lsn) &&
    2630          18 :         ss_oldest_flush_lsn >= wait_for_lsn)
    2631          10 :         return true;
    2632             : 
    2633             :     /*
    2634             :      * To prevent concurrent slot dropping and creation while filtering the
    2635             :      * slots, take the ReplicationSlotControlLock outside of the loop.
    2636             :      */
    2637          16 :     LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
    2638             : 
    2639          16 :     name = synchronized_standby_slots_config->slot_names;
    2640          22 :     for (int i = 0; i < synchronized_standby_slots_config->nslotnames; i++)
    2641             :     {
    2642             :         XLogRecPtr  restart_lsn;
    2643             :         bool        invalidated;
    2644             :         bool        inactive;
    2645             :         ReplicationSlot *slot;
    2646             : 
    2647          16 :         slot = SearchNamedReplicationSlot(name, false);
    2648             : 
    2649             :         /*
    2650             :          * If a slot name provided in synchronized_standby_slots does not
    2651             :          * exist, report a message and exit the loop.
    2652             :          *
    2653             :          * Though validate_sync_standby_slots (the GUC check_hook) tries to
    2654             :          * avoid this, it can nonetheless happen because the user can specify
    2655             :          * a nonexistent slot name before server startup. That function cannot
    2656             :          * validate such a slot during startup, as ReplicationSlotCtl is not
    2657             :          * initialized by then.  Also, the user might have dropped one slot.
    2658             :          */
    2659          16 :         if (!slot)
    2660             :         {
    2661           0 :             ereport(elevel,
    2662             :                     errcode(ERRCODE_INVALID_PARAMETER_VALUE),
    2663             :                     errmsg("replication slot \"%s\" specified in parameter \"%s\" does not exist",
    2664             :                            name, "synchronized_standby_slots"),
    2665             :                     errdetail("Logical replication is waiting on the standby associated with replication slot \"%s\".",
    2666             :                               name),
    2667             :                     errhint("Create the replication slot \"%s\" or amend parameter \"%s\".",
    2668             :                             name, "synchronized_standby_slots"));
    2669           0 :             break;
    2670             :         }
    2671             : 
    2672             :         /* Same as above: if a slot is not physical, exit the loop. */
    2673          16 :         if (SlotIsLogical(slot))
    2674             :         {
    2675           0 :             ereport(elevel,
    2676             :                     errcode(ERRCODE_INVALID_PARAMETER_VALUE),
    2677             :                     errmsg("cannot specify logical replication slot \"%s\" in parameter \"%s\"",
    2678             :                            name, "synchronized_standby_slots"),
    2679             :                     errdetail("Logical replication is waiting for correction on replication slot \"%s\".",
    2680             :                               name),
    2681             :                     errhint("Remove the logical replication slot \"%s\" from parameter \"%s\".",
    2682             :                             name, "synchronized_standby_slots"));
    2683           0 :             break;
    2684             :         }
    2685             : 
    2686          16 :         SpinLockAcquire(&slot->mutex);
    2687          16 :         restart_lsn = slot->data.restart_lsn;
    2688          16 :         invalidated = slot->data.invalidated != RS_INVAL_NONE;
    2689          16 :         inactive = slot->active_pid == 0;
    2690          16 :         SpinLockRelease(&slot->mutex);
    2691             : 
    2692          16 :         if (invalidated)
    2693             :         {
    2694             :             /* Specified physical slot has been invalidated */
    2695           0 :             ereport(elevel,
    2696             :                     errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
    2697             :                     errmsg("physical replication slot \"%s\" specified in parameter \"%s\" has been invalidated",
    2698             :                            name, "synchronized_standby_slots"),
    2699             :                     errdetail("Logical replication is waiting on the standby associated with replication slot \"%s\".",
    2700             :                               name),
    2701             :                     errhint("Drop and recreate the replication slot \"%s\", or amend parameter \"%s\".",
    2702             :                             name, "synchronized_standby_slots"));
    2703           0 :             break;
    2704             :         }
    2705             : 
    2706          16 :         if (XLogRecPtrIsInvalid(restart_lsn) || restart_lsn < wait_for_lsn)
    2707             :         {
    2708             :             /* Log a message if no active_pid for this physical slot */
    2709          10 :             if (inactive)
    2710           8 :                 ereport(elevel,
    2711             :                         errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
    2712             :                         errmsg("replication slot \"%s\" specified in parameter \"%s\" does not have active_pid",
    2713             :                                name, "synchronized_standby_slots"),
    2714             :                         errdetail("Logical replication is waiting on the standby associated with replication slot \"%s\".",
    2715             :                                   name),
    2716             :                         errhint("Start the standby associated with the replication slot \"%s\", or amend parameter \"%s\".",
    2717             :                                 name, "synchronized_standby_slots"));
    2718             : 
    2719             :             /* Continue if the current slot hasn't caught up. */
    2720          10 :             break;
    2721             :         }
    2722             : 
    2723             :         Assert(restart_lsn >= wait_for_lsn);
    2724             : 
    2725           6 :         if (XLogRecPtrIsInvalid(min_restart_lsn) ||
    2726             :             min_restart_lsn > restart_lsn)
    2727           6 :             min_restart_lsn = restart_lsn;
    2728             : 
    2729           6 :         caught_up_slot_num++;
    2730             : 
    2731           6 :         name += strlen(name) + 1;
    2732             :     }
    2733             : 
    2734          16 :     LWLockRelease(ReplicationSlotControlLock);
    2735             : 
    2736             :     /*
    2737             :      * Return false if not all the standbys have caught up to the specified
    2738             :      * WAL location.
    2739             :      */
    2740          16 :     if (caught_up_slot_num != synchronized_standby_slots_config->nslotnames)
    2741          10 :         return false;
    2742             : 
    2743             :     /* The ss_oldest_flush_lsn must not retreat. */
    2744             :     Assert(XLogRecPtrIsInvalid(ss_oldest_flush_lsn) ||
    2745             :            min_restart_lsn >= ss_oldest_flush_lsn);
    2746             : 
    2747           6 :     ss_oldest_flush_lsn = min_restart_lsn;
    2748             : 
    2749           6 :     return true;
    2750             : }
    2751             : 
    2752             : /*
    2753             :  * Wait for physical standbys to confirm receiving the given lsn.
    2754             :  *
    2755             :  * Used by logical decoding SQL functions. It waits for physical standbys
    2756             :  * corresponding to the physical slots specified in the synchronized_standby_slots GUC.
    2757             :  */
    2758             : void
    2759         430 : WaitForStandbyConfirmation(XLogRecPtr wait_for_lsn)
    2760             : {
    2761             :     /*
    2762             :      * Don't need to wait for the standby to catch up if the current acquired
    2763             :      * slot is not a logical failover slot, or there is no value in
    2764             :      * synchronized_standby_slots.
    2765             :      */
    2766         430 :     if (!MyReplicationSlot->data.failover || !synchronized_standby_slots_config)
    2767         428 :         return;
    2768             : 
    2769           2 :     ConditionVariablePrepareToSleep(&WalSndCtl->wal_confirm_rcv_cv);
    2770             : 
    2771             :     for (;;)
    2772             :     {
    2773           4 :         CHECK_FOR_INTERRUPTS();
    2774             : 
    2775           4 :         if (ConfigReloadPending)
    2776             :         {
    2777           2 :             ConfigReloadPending = false;
    2778           2 :             ProcessConfigFile(PGC_SIGHUP);
    2779             :         }
    2780             : 
    2781             :         /* Exit if done waiting for every slot. */
    2782           4 :         if (StandbySlotsHaveCaughtup(wait_for_lsn, WARNING))
    2783           2 :             break;
    2784             : 
    2785             :         /*
    2786             :          * Wait for the slots in the synchronized_standby_slots to catch up,
    2787             :          * but use a timeout (1s) so we can also check if the
    2788             :          * synchronized_standby_slots has been changed.
    2789             :          */
    2790           2 :         ConditionVariableTimedSleep(&WalSndCtl->wal_confirm_rcv_cv, 1000,
    2791             :                                     WAIT_EVENT_WAIT_FOR_STANDBY_CONFIRMATION);
    2792             :     }
    2793             : 
    2794           2 :     ConditionVariableCancelSleep();
    2795             : }

Generated by: LCOV version 1.14