static XLogRecPtr LastRec;
 
-/* Local copy of WalRcv->receivedUpto */
-static XLogRecPtr receivedUpto = 0;
+/* Local copy of WalRcv->flushedUpto */
+static XLogRecPtr flushedUpto = 0;
 static TimeLineID receiveTLI = 0;
 
 /*
     * Retreat _logSegNo using the current end of xlog replayed or received,
     * whichever is later.
     */
-   receivePtr = GetWalRcvWriteRecPtr(NULL, NULL);
+   receivePtr = GetWalRcvFlushRecPtr(NULL, NULL);
    replayPtr = GetXLogReplayRecPtr(&replayTLI);
    endptr = (receivePtr < replayPtr) ? replayPtr : receivePtr;
    KeepLogSeg(endptr, &_logSegNo);
    /* See if we need to retrieve more data */
    if (readFile < 0 ||
        (readSource == XLOG_FROM_STREAM &&
-        receivedUpto < targetPagePtr + reqLen))
+        flushedUpto < targetPagePtr + reqLen))
    {
        if (!WaitForWALToBecomeAvailable(targetPagePtr + reqLen,
                                         private->randAccess,
     */
    if (readSource == XLOG_FROM_STREAM)
    {
-       if (((targetPagePtr) / XLOG_BLCKSZ) != (receivedUpto / XLOG_BLCKSZ))
+       if (((targetPagePtr) / XLOG_BLCKSZ) != (flushedUpto / XLOG_BLCKSZ))
            readLen = XLOG_BLCKSZ;
        else
-           readLen = XLogSegmentOffset(receivedUpto, wal_segment_size) -
+           readLen = XLogSegmentOffset(flushedUpto, wal_segment_size) -
                targetPageOff;
    }
    else
                        RequestXLogStreaming(tli, ptr, PrimaryConnInfo,
                                             PrimarySlotName,
                                             wal_receiver_create_temp_slot);
-                       receivedUpto = 0;
+                       flushedUpto = 0;
                    }
 
                    /*
                     * XLogReceiptTime will not advance, so the grace time
                     * allotted to conflicting queries will decrease.
                     */
-                   if (RecPtr < receivedUpto)
+                   if (RecPtr < flushedUpto)
                        havedata = true;
                    else
                    {
                        XLogRecPtr  latestChunkStart;
 
-                       receivedUpto = GetWalRcvWriteRecPtr(&latestChunkStart, &receiveTLI);
-                       if (RecPtr < receivedUpto && receiveTLI == curFileTLI)
+                       flushedUpto = GetWalRcvFlushRecPtr(&latestChunkStart, &receiveTLI);
+                       if (RecPtr < flushedUpto && receiveTLI == curFileTLI)
                        {
                            havedata = true;
                            if (latestChunkStart <= RecPtr)
 
 {
    XLogRecPtr  recptr;
 
-   recptr = GetWalRcvWriteRecPtr(NULL, NULL);
+   recptr = GetWalRcvFlushRecPtr(NULL, NULL);
 
    if (recptr == 0)
        PG_RETURN_NULL();
 
 WalRcvData->receiveStart.
 
 As walreceiver receives WAL from the master server, and writes and flushes
-it to disk (in pg_wal), it updates WalRcvData->receivedUpto and signals
+it to disk (in pg_wal), it updates WalRcvData->flushedUpto and signals
 the startup process to know how far WAL replay can advance.
 
 Walreceiver sends information about replication progress to the master server
 
  * in the primary server), and then keeps receiving XLOG records and
  * writing them to the disk as long as the connection is alive. As XLOG
  * records are received and flushed to disk, it updates the
- * WalRcv->receivedUpto variable in shared memory, to inform the startup
+ * WalRcv->flushedUpto variable in shared memory, to inform the startup
  * process of how far it can proceed with XLOG replay.
  *
  * A WAL receiver cannot directly load GUC parameters used when establishing
 
    SpinLockRelease(&walrcv->mutex);
 
+   pg_atomic_init_u64(&WalRcv->writtenUpto, 0);
+
    /* Arrange to clean up at walreceiver exit */
    on_shmem_exit(WalRcvDie, 0);
 
 
        LogstreamResult.Write = recptr;
    }
+
+   /* Update shared-memory status */
+   pg_atomic_write_u64(&WalRcv->writtenUpto, LogstreamResult.Write);
 }
 
 /*
 
        /* Update shared-memory status */
        SpinLockAcquire(&walrcv->mutex);
-       if (walrcv->receivedUpto < LogstreamResult.Flush)
+       if (walrcv->flushedUpto < LogstreamResult.Flush)
        {
-           walrcv->latestChunkStart = walrcv->receivedUpto;
-           walrcv->receivedUpto = LogstreamResult.Flush;
+           walrcv->latestChunkStart = walrcv->flushedUpto;
+           walrcv->flushedUpto = LogstreamResult.Flush;
            walrcv->receivedTLI = ThisTimeLineID;
        }
        SpinLockRelease(&walrcv->mutex);
    state = WalRcv->walRcvState;
    receive_start_lsn = WalRcv->receiveStart;
    receive_start_tli = WalRcv->receiveStartTLI;
-   received_lsn = WalRcv->receivedUpto;
+   received_lsn = WalRcv->flushedUpto;
    received_tli = WalRcv->receivedTLI;
    last_send_time = WalRcv->lastMsgSendTime;
    last_receipt_time = WalRcv->lastMsgReceiptTime;
 
 
    /*
     * If this is the first startup of walreceiver (on this timeline),
-    * initialize receivedUpto and latestChunkStart to the starting point.
+    * initialize flushedUpto and latestChunkStart to the starting point.
     */
    if (walrcv->receiveStart == 0 || walrcv->receivedTLI != tli)
    {
-       walrcv->receivedUpto = recptr;
+       walrcv->flushedUpto = recptr;
        walrcv->receivedTLI = tli;
        walrcv->latestChunkStart = recptr;
    }
 }
 
 /*
- * Returns the last+1 byte position that walreceiver has written.
+ * Returns the last+1 byte position that walreceiver has flushed.
  *
  * Optionally, returns the previous chunk start, that is the first byte
  * written in the most recent walreceiver flush cycle.  Callers not
  * receiveTLI.
  */
 XLogRecPtr
-GetWalRcvWriteRecPtr(XLogRecPtr *latestChunkStart, TimeLineID *receiveTLI)
+GetWalRcvFlushRecPtr(XLogRecPtr *latestChunkStart, TimeLineID *receiveTLI)
 {
    WalRcvData *walrcv = WalRcv;
    XLogRecPtr  recptr;
 
    SpinLockAcquire(&walrcv->mutex);
-   recptr = walrcv->receivedUpto;
+   recptr = walrcv->flushedUpto;
    if (latestChunkStart)
        *latestChunkStart = walrcv->latestChunkStart;
    if (receiveTLI)
    return recptr;
 }
 
+/*
+ * Returns the last+1 byte position that walreceiver has written.
+ * This returns a recently written value without taking a lock.
+ */
+XLogRecPtr
+GetWalRcvWriteRecPtr(void)
+{
+   WalRcvData *walrcv = WalRcv;
+
+   return pg_atomic_read_u64(&walrcv->writtenUpto);
+}
+
 /*
  * Returns the replication apply delay in ms or -1
  * if the apply delay info is not available
    TimestampTz chunkReplayStartTime;
 
    SpinLockAcquire(&walrcv->mutex);
-   receivePtr = walrcv->receivedUpto;
+   receivePtr = walrcv->flushedUpto;
    SpinLockRelease(&walrcv->mutex);
 
    replayPtr = GetXLogReplayRecPtr(NULL);
 
     * has streamed, but hasn't been replayed yet.
     */
 
-   receivePtr = GetWalRcvWriteRecPtr(NULL, &receiveTLI);
+   receivePtr = GetWalRcvFlushRecPtr(NULL, &receiveTLI);
    replayPtr = GetXLogReplayRecPtr(&replayTLI);
 
    ThisTimeLineID = replayTLI;
 
 #include "access/xlogdefs.h"
 #include "getaddrinfo.h"       /* for NI_MAXHOST */
 #include "pgtime.h"
+#include "port/atomics.h"
 #include "replication/logicalproto.h"
 #include "replication/walsender.h"
 #include "storage/latch.h"
    TimeLineID  receiveStartTLI;
 
    /*
-    * receivedUpto-1 is the last byte position that has already been
+    * flushedUpto-1 is the last byte position that has already been
     * received, and receivedTLI is the timeline it came from.  At the first
     * startup of walreceiver, these are set to receiveStart and
     * receiveStartTLI. After that, walreceiver updates these whenever it
     * flushes the received WAL to disk.
     */
-   XLogRecPtr  receivedUpto;
+   XLogRecPtr  flushedUpto;
    TimeLineID  receivedTLI;
 
    /*
     * latestChunkStart is the starting byte position of the current "batch"
     * of received WAL.  It's actually the same as the previous value of
-    * receivedUpto before the last flush to disk.  Startup process can use
+    * flushedUpto before the last flush to disk.  Startup process can use
     * this to detect whether it's keeping up or not.
     */
    XLogRecPtr  latestChunkStart;
 
    slock_t     mutex;          /* locks shared variables shown above */
 
+   /*
+    * Like flushedUpto, but advanced after writing and before flushing,
+    * without the need to acquire the spin lock.  Data can be read by another
+    * process up to this point, but shouldn't be used for data integrity
+    * purposes.
+    */
+   pg_atomic_uint64 writtenUpto;
+
    /*
     * force walreceiver reply?  This doesn't need to be locked; memory
     * barriers for ordering are sufficient.  But we do need atomic fetch and
 extern void RequestXLogStreaming(TimeLineID tli, XLogRecPtr recptr,
                                 const char *conninfo, const char *slotname,
                                 bool create_temp_slot);
-extern XLogRecPtr GetWalRcvWriteRecPtr(XLogRecPtr *latestChunkStart, TimeLineID *receiveTLI);
+extern XLogRecPtr GetWalRcvFlushRecPtr(XLogRecPtr *latestChunkStart, TimeLineID *receiveTLI);
+extern XLogRecPtr GetWalRcvWriteRecPtr(void);
 extern int GetReplicationApplyDelay(void);
 extern int GetReplicationTransferLatency(void);
 extern void WalRcvForceReply(void);