mirror of
https://github.com/postgres/postgres.git
synced 2025-07-28 23:42:10 +03:00
Reduce idle power consumption of walwriter and checkpointer processes.
This patch modifies the walwriter process so that, when it has not found anything useful to do for many consecutive wakeup cycles, it extends its sleep time to reduce the server's idle power consumption. It reverts to normal as soon as it's done any successful flushes. It's still true that during any async commit, backends check for completed, unflushed pages of WAL and signal the walwriter if there are any; so that in practice the walwriter can get awakened and returned to normal operation sooner than the sleep time might suggest. Also, improve the checkpointer so that it uses a latch and a computed delay time to not wake up at all except when it has something to do, replacing a previous hardcoded 0.5 sec wakeup cycle. This also is primarily useful for reducing the server's power consumption when idle. In passing, get rid of the dedicated latch for signaling the walwriter in favor of using its procLatch, since that comports better with possible generic signal handlers using that latch. Also, fix a pre-existing bug with failure to save/restore errno in walwriter's signal handlers. Peter Geoghegan, somewhat simplified by Tom
This commit is contained in:
@ -433,11 +433,6 @@ typedef struct XLogCtlData
|
||||
*/
|
||||
Latch recoveryWakeupLatch;
|
||||
|
||||
/*
|
||||
* WALWriterLatch is used to wake up the WALWriter to write some WAL.
|
||||
*/
|
||||
Latch WALWriterLatch;
|
||||
|
||||
/*
|
||||
* During recovery, we keep a copy of the latest checkpoint record here.
|
||||
* Used by the background writer when it wants to create a restartpoint.
|
||||
@ -1935,7 +1930,8 @@ XLogSetAsyncXactLSN(XLogRecPtr asyncXactLSN)
|
||||
/*
|
||||
* Nudge the WALWriter if we have a full page of WAL to write.
|
||||
*/
|
||||
SetLatch(&XLogCtl->WALWriterLatch);
|
||||
if (ProcGlobal->walwriterLatch)
|
||||
SetLatch(ProcGlobal->walwriterLatch);
|
||||
}
|
||||
|
||||
/*
|
||||
@ -2167,22 +2163,25 @@ XLogFlush(XLogRecPtr record)
|
||||
* block, and flush through the latest one of those. Thus, if async commits
|
||||
* are not being used, we will flush complete blocks only. We can guarantee
|
||||
* that async commits reach disk after at most three cycles; normally only
|
||||
* one or two. (We allow XLogWrite to write "flexibly", meaning it can stop
|
||||
* at the end of the buffer ring; this makes a difference only with very high
|
||||
* load or long wal_writer_delay, but imposes one extra cycle for the worst
|
||||
* case for async commits.)
|
||||
* one or two. (When flushing complete blocks, we allow XLogWrite to write
|
||||
* "flexibly", meaning it can stop at the end of the buffer ring; this makes a
|
||||
* difference only with very high load or long wal_writer_delay, but imposes
|
||||
* one extra cycle for the worst case for async commits.)
|
||||
*
|
||||
* This routine is invoked periodically by the background walwriter process.
|
||||
*
|
||||
* Returns TRUE if we flushed anything.
|
||||
*/
|
||||
void
|
||||
bool
|
||||
XLogBackgroundFlush(void)
|
||||
{
|
||||
XLogRecPtr WriteRqstPtr;
|
||||
bool flexible = true;
|
||||
bool wrote_something = false;
|
||||
|
||||
/* XLOG doesn't need flushing during recovery */
|
||||
if (RecoveryInProgress())
|
||||
return;
|
||||
return false;
|
||||
|
||||
/* read LogwrtResult and update local state */
|
||||
{
|
||||
@ -2224,7 +2223,7 @@ XLogBackgroundFlush(void)
|
||||
XLogFileClose();
|
||||
}
|
||||
}
|
||||
return;
|
||||
return false;
|
||||
}
|
||||
|
||||
#ifdef WAL_DEBUG
|
||||
@ -2247,10 +2246,13 @@ XLogBackgroundFlush(void)
|
||||
WriteRqst.Write = WriteRqstPtr;
|
||||
WriteRqst.Flush = WriteRqstPtr;
|
||||
XLogWrite(WriteRqst, flexible, false);
|
||||
wrote_something = true;
|
||||
}
|
||||
LWLockRelease(WALWriteLock);
|
||||
|
||||
END_CRIT_SECTION();
|
||||
|
||||
return wrote_something;
|
||||
}
|
||||
|
||||
/*
|
||||
@ -5101,7 +5103,6 @@ XLOGShmemInit(void)
|
||||
XLogCtl->Insert.currpage = (XLogPageHeader) (XLogCtl->pages);
|
||||
SpinLockInit(&XLogCtl->info_lck);
|
||||
InitSharedLatch(&XLogCtl->recoveryWakeupLatch);
|
||||
InitSharedLatch(&XLogCtl->WALWriterLatch);
|
||||
|
||||
/*
|
||||
* If we are not in bootstrap mode, pg_control should already exist. Read
|
||||
@ -10478,12 +10479,3 @@ WakeupRecovery(void)
|
||||
{
|
||||
SetLatch(&XLogCtl->recoveryWakeupLatch);
|
||||
}
|
||||
|
||||
/*
|
||||
* Manage the WALWriterLatch
|
||||
*/
|
||||
Latch *
|
||||
WALWriterLatch(void)
|
||||
{
|
||||
return &XLogCtl->WALWriterLatch;
|
||||
}
|
||||
|
Reference in New Issue
Block a user