process startup: Always call Init[Auxiliary]Process() before BaseInit().

For EXEC_BACKEND InitProcess()/InitAuxiliaryProcess() needs to have been
called well before we call BaseInit(), as SubPostmasterMain() needs LWLocks to
work. Having the order of initialization differ between platforms makes it
unnecessarily hard to understand the system and to add initialization points
for new subsystems without a lot of duplication.

To be able to change the order, BaseInit() cannot trigger
CreateSharedMemoryAndSemaphores() anymore - obviously that needs to have
happened before we can call InitProcess(). It seems cleaner to create shared
memory explicitly in single user/bootstrap mode anyway.

After this change the separation of bufmgr initialization into
InitBufferPoolAccess() / InitBufferPoolBackend() is not meaningful anymore so
the latter is removed.

Author: Andres Freund <andres@anarazel.de>
Reviewed-By: Kyotaro Horiguchi <horikyota.ntt@gmail.com>
Discussion: https://postgr.es/m/20210802164124.ufo5buo4apl6yuvs@alap3.anarazel.de
pull/68/head
Andres Freund 4 years ago
parent 0de13bbc47
commit b406478b87
  1. 18
      src/backend/bootstrap/bootstrap.c
  2. 12
      src/backend/postmaster/autovacuum.c
  3. 7
      src/backend/postmaster/auxprocess.c
  4. 16
      src/backend/postmaster/bgworker.c
  5. 21
      src/backend/storage/buffer/bufmgr.c
  6. 8
      src/backend/tcop/postgres.c
  7. 37
      src/backend/utils/init/postinit.c
  8. 1
      src/include/storage/bufmgr.h

@ -178,8 +178,8 @@ static IndexList *ILHead = NULL;
/* /*
* In shared memory checker mode, all we really want to do is create shared * In shared memory checker mode, all we really want to do is create shared
* memory and semaphores (just to prove we can do it with the current GUC * memory and semaphores (just to prove we can do it with the current GUC
* settings). Since, in fact, that was already done by BaseInit(), * settings). Since, in fact, that was already done by
* we have nothing more to do here. * CreateSharedMemoryAndSemaphores(), we have nothing more to do here.
*/ */
static void static void
CheckerModeMain(void) CheckerModeMain(void)
@ -324,7 +324,7 @@ BootstrapModeMain(int argc, char *argv[], bool check_only)
InitializeMaxBackends(); InitializeMaxBackends();
BaseInit(); CreateSharedMemoryAndSemaphores();
/* /*
* XXX: It might make sense to move this into its own function at some * XXX: It might make sense to move this into its own function at some
@ -338,6 +338,13 @@ BootstrapModeMain(int argc, char *argv[], bool check_only)
abort(); abort();
} }
/*
* Do backend-like initialization for bootstrap mode
*/
InitProcess();
BaseInit();
bootstrap_signals(); bootstrap_signals();
BootStrapXLOG(); BootStrapXLOG();
@ -348,11 +355,6 @@ BootstrapModeMain(int argc, char *argv[], bool check_only)
if (pg_link_canary_is_frontend()) if (pg_link_canary_is_frontend())
elog(ERROR, "backend is incorrectly linked to frontend functions"); elog(ERROR, "backend is incorrectly linked to frontend functions");
/*
* Do backend-like initialization for bootstrap mode
*/
InitProcess();
InitPostgres(NULL, InvalidOid, NULL, InvalidOid, NULL, false); InitPostgres(NULL, InvalidOid, NULL, InvalidOid, NULL, false);
/* Initialize stuff for bootstrap-file processing */ /* Initialize stuff for bootstrap-file processing */

@ -469,9 +469,6 @@ AutoVacLauncherMain(int argc, char *argv[])
pqsignal(SIGFPE, FloatExceptionHandler); pqsignal(SIGFPE, FloatExceptionHandler);
pqsignal(SIGCHLD, SIG_DFL); pqsignal(SIGCHLD, SIG_DFL);
/* Early initialization */
BaseInit();
/* /*
* Create a per-backend PGPROC struct in shared memory, except in the * Create a per-backend PGPROC struct in shared memory, except in the
* EXEC_BACKEND case where this was done in SubPostmasterMain. We must do * EXEC_BACKEND case where this was done in SubPostmasterMain. We must do
@ -482,6 +479,9 @@ AutoVacLauncherMain(int argc, char *argv[])
InitProcess(); InitProcess();
#endif #endif
/* Early initialization */
BaseInit();
InitPostgres(NULL, InvalidOid, NULL, InvalidOid, NULL, false); InitPostgres(NULL, InvalidOid, NULL, InvalidOid, NULL, false);
SetProcessingMode(NormalProcessing); SetProcessingMode(NormalProcessing);
@ -1547,9 +1547,6 @@ AutoVacWorkerMain(int argc, char *argv[])
pqsignal(SIGFPE, FloatExceptionHandler); pqsignal(SIGFPE, FloatExceptionHandler);
pqsignal(SIGCHLD, SIG_DFL); pqsignal(SIGCHLD, SIG_DFL);
/* Early initialization */
BaseInit();
/* /*
* Create a per-backend PGPROC struct in shared memory, except in the * Create a per-backend PGPROC struct in shared memory, except in the
* EXEC_BACKEND case where this was done in SubPostmasterMain. We must do * EXEC_BACKEND case where this was done in SubPostmasterMain. We must do
@ -1560,6 +1557,9 @@ AutoVacWorkerMain(int argc, char *argv[])
InitProcess(); InitProcess();
#endif #endif
/* Early initialization */
BaseInit();
/* /*
* If an exception is encountered, processing resumes here. * If an exception is encountered, processing resumes here.
* *

@ -90,8 +90,6 @@ AuxiliaryProcessMain(AuxProcType auxtype)
SetProcessingMode(BootstrapProcessing); SetProcessingMode(BootstrapProcessing);
IgnoreSystemIndexes = true; IgnoreSystemIndexes = true;
BaseInit();
/* /*
* As an auxiliary process, we aren't going to do the full InitPostgres * As an auxiliary process, we aren't going to do the full InitPostgres
* pushups, but there are a couple of things that need to get lit up even * pushups, but there are a couple of things that need to get lit up even
@ -106,6 +104,8 @@ AuxiliaryProcessMain(AuxProcType auxtype)
InitAuxiliaryProcess(); InitAuxiliaryProcess();
#endif #endif
BaseInit();
/* /*
* Assign the ProcSignalSlot for an auxiliary process. Since it doesn't * Assign the ProcSignalSlot for an auxiliary process. Since it doesn't
* have a BackendId, the slot is statically allocated based on the * have a BackendId, the slot is statically allocated based on the
@ -118,9 +118,6 @@ AuxiliaryProcessMain(AuxProcType auxtype)
*/ */
ProcSignalInit(MaxBackends + MyAuxProcType + 1); ProcSignalInit(MaxBackends + MyAuxProcType + 1);
/* finish setting up bufmgr.c */
InitBufferPoolBackend();
/* /*
* Auxiliary processes don't run transactions, but they may need a * Auxiliary processes don't run transactions, but they may need a
* resource owner anyway to manage buffer pins acquired outside * resource owner anyway to manage buffer pins acquired outside

@ -837,14 +837,6 @@ StartBackgroundWorker(void)
*/ */
if (worker->bgw_flags & BGWORKER_SHMEM_ACCESS) if (worker->bgw_flags & BGWORKER_SHMEM_ACCESS)
{ {
/*
* Early initialization. Some of this could be useful even for
* background workers that aren't using shared memory, but they can
* call the individual startup routines for those subsystems if
* needed.
*/
BaseInit();
/* /*
* Create a per-backend PGPROC struct in shared memory, except in the * Create a per-backend PGPROC struct in shared memory, except in the
* EXEC_BACKEND case where this was done in SubPostmasterMain. We must * EXEC_BACKEND case where this was done in SubPostmasterMain. We must
@ -854,6 +846,14 @@ StartBackgroundWorker(void)
#ifndef EXEC_BACKEND #ifndef EXEC_BACKEND
InitProcess(); InitProcess();
#endif #endif
/*
* Early initialization. Some of this could be useful even for
* background workers that aren't using shared memory, but they can
* call the individual startup routines for those subsystems if
* needed.
*/
BaseInit();
} }
/* /*

@ -2582,11 +2582,6 @@ AtEOXact_Buffers(bool isCommit)
* This is called during backend startup (whether standalone or under the * This is called during backend startup (whether standalone or under the
* postmaster). It sets up for this backend's access to the already-existing * postmaster). It sets up for this backend's access to the already-existing
* buffer pool. * buffer pool.
*
* NB: this is called before InitProcess(), so we do not have a PGPROC and
* cannot do LWLockAcquire; hence we can't actually access stuff in
* shared memory yet. We are only initializing local data here.
* (See also InitBufferPoolBackend)
*/ */
void void
InitBufferPoolAccess(void) InitBufferPoolAccess(void)
@ -2600,20 +2595,12 @@ InitBufferPoolAccess(void)
PrivateRefCountHash = hash_create("PrivateRefCount", 100, &hash_ctl, PrivateRefCountHash = hash_create("PrivateRefCount", 100, &hash_ctl,
HASH_ELEM | HASH_BLOBS); HASH_ELEM | HASH_BLOBS);
}
/* /*
* InitBufferPoolBackend --- second-stage initialization of a new backend * AtProcExit_Buffers needs LWLock access, and thereby has to be called at
* * the corresponding phase of backend shutdown.
* This is called after we have acquired a PGPROC and so can safely get
* LWLocks. We don't currently need to do anything at this stage ...
* except register a shmem-exit callback. AtProcExit_Buffers needs LWLock
* access, and thereby has to be called at the corresponding phase of
* backend shutdown.
*/ */
void Assert(MyProc != NULL);
InitBufferPoolBackend(void)
{
on_shmem_exit(AtProcExit_Buffers, 0); on_shmem_exit(AtProcExit_Buffers, 0);
} }

@ -4050,10 +4050,9 @@ PostgresMain(int argc, char *argv[],
/* Initialize MaxBackends (if under postmaster, was done already) */ /* Initialize MaxBackends (if under postmaster, was done already) */
InitializeMaxBackends(); InitializeMaxBackends();
}
/* Early initialization */ CreateSharedMemoryAndSemaphores();
BaseInit(); }
/* /*
* Create a per-backend PGPROC struct in shared memory, except in the * Create a per-backend PGPROC struct in shared memory, except in the
@ -4068,6 +4067,9 @@ PostgresMain(int argc, char *argv[],
InitProcess(); InitProcess();
#endif #endif
/* Early initialization */
BaseInit();
/* We need to allow SIGINT, etc during the initial transaction */ /* We need to allow SIGINT, etc during the initial transaction */
PG_SETMASK(&UnBlockSig); PG_SETMASK(&UnBlockSig);

@ -67,7 +67,6 @@ static HeapTuple GetDatabaseTuple(const char *dbname);
static HeapTuple GetDatabaseTupleByOid(Oid dboid); static HeapTuple GetDatabaseTupleByOid(Oid dboid);
static void PerformAuthentication(Port *port); static void PerformAuthentication(Port *port);
static void CheckMyDatabase(const char *name, bool am_superuser, bool override_allow_connections); static void CheckMyDatabase(const char *name, bool am_superuser, bool override_allow_connections);
static void InitCommunication(void);
static void ShutdownPostgres(int code, Datum arg); static void ShutdownPostgres(int code, Datum arg);
static void StatementTimeoutHandler(void); static void StatementTimeoutHandler(void);
static void LockTimeoutHandler(void); static void LockTimeoutHandler(void);
@ -417,31 +416,6 @@ CheckMyDatabase(const char *name, bool am_superuser, bool override_allow_connect
} }
/* --------------------------------
* InitCommunication
*
* This routine initializes stuff needed for ipc, locking, etc.
* it should be called something more informative.
* --------------------------------
*/
static void
InitCommunication(void)
{
/*
* initialize shared memory and semaphores appropriately.
*/
if (!IsUnderPostmaster) /* postmaster already did this */
{
/*
* We're running a postgres bootstrap process or a standalone backend,
* so we need to set up shmem.
*/
CreateSharedMemoryAndSemaphores();
}
}
/* /*
* pg_split_opts -- split a string of options and append it to an argv array * pg_split_opts -- split a string of options and append it to an argv array
* *
@ -536,11 +510,11 @@ InitializeMaxBackends(void)
void void
BaseInit(void) BaseInit(void)
{ {
Assert(MyProc != NULL);
/* /*
* Attach to shared memory and semaphores, and initialize our * Initialize our input/output/debugging file descriptors.
* input/output/debugging file descriptors.
*/ */
InitCommunication();
DebugFileOpen(); DebugFileOpen();
/* Do local initialization of file, storage and buffer managers */ /* Do local initialization of file, storage and buffer managers */
@ -624,11 +598,6 @@ InitPostgres(const char *in_dbname, Oid dboid, const char *username,
RegisterTimeout(CLIENT_CONNECTION_CHECK_TIMEOUT, ClientCheckTimeoutHandler); RegisterTimeout(CLIENT_CONNECTION_CHECK_TIMEOUT, ClientCheckTimeoutHandler);
} }
/*
* bufmgr needs another initialization call too
*/
InitBufferPoolBackend();
/* /*
* Initialize local process's access to XLOG. * Initialize local process's access to XLOG.
*/ */

@ -194,7 +194,6 @@ extern Buffer ReleaseAndReadBuffer(Buffer buffer, Relation relation,
extern void InitBufferPool(void); extern void InitBufferPool(void);
extern void InitBufferPoolAccess(void); extern void InitBufferPoolAccess(void);
extern void InitBufferPoolBackend(void);
extern void AtEOXact_Buffers(bool isCommit); extern void AtEOXact_Buffers(bool isCommit);
extern void PrintBufferLeakWarning(Buffer buffer); extern void PrintBufferLeakWarning(Buffer buffer);
extern void CheckPointBuffers(int flags); extern void CheckPointBuffers(int flags);

Loading…
Cancel
Save