Commit 57cb8063 authored by Thomas Munro's avatar Thomas Munro

Fix locking bugs that could corrupt pg_control.

The redo routines for XLOG_CHECKPOINT_{ONLINE,SHUTDOWN} must acquire
ControlFileLock before modifying ControlFile->checkPointCopy, or the
checkpointer could write out a control file with a bad checksum.

Likewise, XLogReportParameters() must acquire ControlFileLock before
modifying ControlFile and calling UpdateControlFile().

Back-patch to all supported releases.

Author: Nathan Bossart <bossartn@amazon.com>
Author: Fujii Masao <masao.fujii@oss.nttdata.com>
Reviewed-by: default avatarFujii Masao <masao.fujii@oss.nttdata.com>
Reviewed-by: default avatarMichael Paquier <michael@paquier.xyz>
Reviewed-by: default avatarThomas Munro <thomas.munro@gmail.com>
Discussion: https://postgr.es/m/70BF24D6-DC51-443F-B55A-95735803842A%40amazon.com
parent d094bf93
...@@ -9743,6 +9743,8 @@ XLogReportParameters(void) ...@@ -9743,6 +9743,8 @@ XLogReportParameters(void)
XLogFlush(recptr); XLogFlush(recptr);
} }
LWLockAcquire(ControlFileLock, LW_EXCLUSIVE);
ControlFile->MaxConnections = MaxConnections; ControlFile->MaxConnections = MaxConnections;
ControlFile->max_worker_processes = max_worker_processes; ControlFile->max_worker_processes = max_worker_processes;
ControlFile->max_wal_senders = max_wal_senders; ControlFile->max_wal_senders = max_wal_senders;
...@@ -9752,6 +9754,8 @@ XLogReportParameters(void) ...@@ -9752,6 +9754,8 @@ XLogReportParameters(void)
ControlFile->wal_log_hints = wal_log_hints; ControlFile->wal_log_hints = wal_log_hints;
ControlFile->track_commit_timestamp = track_commit_timestamp; ControlFile->track_commit_timestamp = track_commit_timestamp;
UpdateControlFile(); UpdateControlFile();
LWLockRelease(ControlFileLock);
} }
} }
...@@ -9976,7 +9980,9 @@ xlog_redo(XLogReaderState *record) ...@@ -9976,7 +9980,9 @@ xlog_redo(XLogReaderState *record)
} }
/* ControlFile->checkPointCopy always tracks the latest ckpt XID */ /* ControlFile->checkPointCopy always tracks the latest ckpt XID */
LWLockAcquire(ControlFileLock, LW_EXCLUSIVE);
ControlFile->checkPointCopy.nextFullXid = checkPoint.nextFullXid; ControlFile->checkPointCopy.nextFullXid = checkPoint.nextFullXid;
LWLockRelease(ControlFileLock);
/* Update shared-memory copy of checkpoint XID/epoch */ /* Update shared-memory copy of checkpoint XID/epoch */
SpinLockAcquire(&XLogCtl->info_lck); SpinLockAcquire(&XLogCtl->info_lck);
...@@ -10033,7 +10039,9 @@ xlog_redo(XLogReaderState *record) ...@@ -10033,7 +10039,9 @@ xlog_redo(XLogReaderState *record)
SetTransactionIdLimit(checkPoint.oldestXid, SetTransactionIdLimit(checkPoint.oldestXid,
checkPoint.oldestXidDB); checkPoint.oldestXidDB);
/* ControlFile->checkPointCopy always tracks the latest ckpt XID */ /* ControlFile->checkPointCopy always tracks the latest ckpt XID */
LWLockAcquire(ControlFileLock, LW_EXCLUSIVE);
ControlFile->checkPointCopy.nextFullXid = checkPoint.nextFullXid; ControlFile->checkPointCopy.nextFullXid = checkPoint.nextFullXid;
LWLockRelease(ControlFileLock);
/* Update shared-memory copy of checkpoint XID/epoch */ /* Update shared-memory copy of checkpoint XID/epoch */
SpinLockAcquire(&XLogCtl->info_lck); SpinLockAcquire(&XLogCtl->info_lck);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment