mirror of
https://git.postgresql.org/git/postgresql.git
synced 2025-01-12 18:34:36 +08:00
VariableCache (next XID generator) is placed in shmem.
This commit is contained in:
parent
224a62c5b7
commit
5afe171443
@ -7,7 +7,7 @@
|
|||||||
*
|
*
|
||||||
*
|
*
|
||||||
* IDENTIFICATION
|
* IDENTIFICATION
|
||||||
* $Header: /cvsroot/pgsql/src/backend/access/transam/varsup.c,v 1.15 1998/01/07 21:02:21 momjian Exp $
|
* $Header: /cvsroot/pgsql/src/backend/access/transam/varsup.c,v 1.16 1998/07/21 06:17:13 vadim Exp $
|
||||||
*
|
*
|
||||||
*-------------------------------------------------------------------------
|
*-------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
@ -31,6 +31,8 @@ static void VariableRelationPutNextOid(Oid *oidP);
|
|||||||
*/
|
*/
|
||||||
int OidGenLockId;
|
int OidGenLockId;
|
||||||
|
|
||||||
|
VariableCache ShmemVariableCache = NULL;
|
||||||
|
|
||||||
/* ----------------------------------------------------------------
|
/* ----------------------------------------------------------------
|
||||||
* variable relation query/update routines
|
* variable relation query/update routines
|
||||||
* ----------------------------------------------------------------
|
* ----------------------------------------------------------------
|
||||||
@ -258,16 +260,7 @@ VariableRelationPutNextOid(Oid *oidP)
|
|||||||
* In the version 2 transaction system, transaction id's are
|
* In the version 2 transaction system, transaction id's are
|
||||||
* restricted in several ways.
|
* restricted in several ways.
|
||||||
*
|
*
|
||||||
* First, all transaction id's are even numbers (4, 88, 121342, etc).
|
* -- Old comments removed --
|
||||||
* This means the binary representation of the number will never
|
|
||||||
* have the least significent bit set. This bit is reserved to
|
|
||||||
* indicate that the transaction id does not in fact hold an XID,
|
|
||||||
* but rather a commit time. This makes it possible for the
|
|
||||||
* vaccuum daemon to disgard information from the log and time
|
|
||||||
* relations for committed tuples. This is important when archiving
|
|
||||||
* tuples to an optical disk because tuples with commit times
|
|
||||||
* stored in their xid fields will not need to consult the log
|
|
||||||
* and time relations.
|
|
||||||
*
|
*
|
||||||
* Second, since we may someday preform compression of the data
|
* Second, since we may someday preform compression of the data
|
||||||
* in the log and time relations, we cause the numbering of the
|
* in the log and time relations, we cause the numbering of the
|
||||||
@ -276,32 +269,16 @@ VariableRelationPutNextOid(Oid *oidP)
|
|||||||
* transaction id's 0 - 510 will never be used. This space is
|
* transaction id's 0 - 510 will never be used. This space is
|
||||||
* in fact used to store the version number of the postgres
|
* in fact used to store the version number of the postgres
|
||||||
* transaction log and will someday store compression information
|
* transaction log and will someday store compression information
|
||||||
* about the log.
|
* about the log. -- this is also old comments...
|
||||||
*
|
|
||||||
* Lastly, rather then access the variable relation each time
|
|
||||||
* a backend requests a new transction id, we "prefetch" 32
|
|
||||||
* transaction id's by incrementing the nextXid stored in the
|
|
||||||
* var relation by 64 (remember only even xid's are legal) and then
|
|
||||||
* returning these id's one at a time until they are exhausted.
|
|
||||||
* This means we reduce the number of accesses to the variable
|
|
||||||
* relation by 32 for each backend.
|
|
||||||
*
|
|
||||||
* Note: 32 has no special significance. We don't want the
|
|
||||||
* number to be too large because if when the backend
|
|
||||||
* terminates, we lose the xid's we cached.
|
|
||||||
*
|
*
|
||||||
* ----------------
|
* ----------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#define VAR_XID_PREFETCH 32
|
#define VAR_XID_PREFETCH 1024
|
||||||
|
|
||||||
static int prefetched_xid_count = 0;
|
|
||||||
static TransactionId next_prefetched_xid;
|
|
||||||
|
|
||||||
void
|
void
|
||||||
GetNewTransactionId(TransactionId *xid)
|
GetNewTransactionId(TransactionId *xid)
|
||||||
{
|
{
|
||||||
TransactionId nextid;
|
|
||||||
|
|
||||||
/* ----------------
|
/* ----------------
|
||||||
* during bootstrap initialization, we return the special
|
* during bootstrap initialization, we return the special
|
||||||
@ -314,51 +291,24 @@ GetNewTransactionId(TransactionId *xid)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ----------------
|
SpinAcquire(OidGenLockId); /* not good for concurrency... */
|
||||||
* if we run out of prefetched xids, then we get some
|
|
||||||
* more before handing them out to the caller.
|
|
||||||
* ----------------
|
|
||||||
*/
|
|
||||||
|
|
||||||
if (prefetched_xid_count == 0)
|
if (ShmemVariableCache->xid_count == 0)
|
||||||
{
|
{
|
||||||
/* ----------------
|
TransactionId nextid;
|
||||||
* obtain exclusive access to the variable relation page
|
|
||||||
*
|
|
||||||
* get the "next" xid from the variable relation
|
|
||||||
* and save it in the prefetched id.
|
|
||||||
* ----------------
|
|
||||||
*/
|
|
||||||
SpinAcquire(OidGenLockId);
|
|
||||||
VariableRelationGetNextXid(&nextid);
|
|
||||||
TransactionIdStore(nextid, &next_prefetched_xid);
|
|
||||||
|
|
||||||
/* ----------------
|
VariableRelationGetNextXid(&nextid);
|
||||||
* now increment the variable relation's next xid
|
TransactionIdStore(nextid, &(ShmemVariableCache->nextXid));
|
||||||
* and reset the prefetched_xid_count. We multiply
|
ShmemVariableCache->xid_count = VAR_XID_PREFETCH;
|
||||||
* the id by two because our xid's are always even.
|
TransactionIdAdd(&nextid, VAR_XID_PREFETCH);
|
||||||
* ----------------
|
|
||||||
*/
|
|
||||||
prefetched_xid_count = VAR_XID_PREFETCH;
|
|
||||||
TransactionIdAdd(&nextid, prefetched_xid_count);
|
|
||||||
VariableRelationPutNextXid(nextid);
|
VariableRelationPutNextXid(nextid);
|
||||||
SpinRelease(OidGenLockId);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ----------------
|
TransactionIdStore(ShmemVariableCache->nextXid, xid);
|
||||||
* return the next prefetched xid in the pointer passed by
|
TransactionIdAdd(&(ShmemVariableCache->nextXid), 1);
|
||||||
* the user and decrement the prefetch count. We add two
|
(ShmemVariableCache->xid_count)--;
|
||||||
* to id we return the next time this is called because our
|
|
||||||
* transaction ids are always even.
|
SpinRelease(OidGenLockId);
|
||||||
*
|
|
||||||
* XXX Transaction Ids used to be even as the low order bit was
|
|
||||||
* used to determine commit status. This is no long true so
|
|
||||||
* we now use even and odd transaction ids. -mer 5/26/92
|
|
||||||
* ----------------
|
|
||||||
*/
|
|
||||||
TransactionIdStore(next_prefetched_xid, xid);
|
|
||||||
TransactionIdAdd(&next_prefetched_xid, 1);
|
|
||||||
prefetched_xid_count--;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ----------------------------------------------------------------
|
/* ----------------------------------------------------------------
|
||||||
|
@ -7,7 +7,7 @@
|
|||||||
*
|
*
|
||||||
*
|
*
|
||||||
* IDENTIFICATION
|
* IDENTIFICATION
|
||||||
* $Header: /cvsroot/pgsql/src/backend/storage/ipc/shmem.c,v 1.27 1998/06/30 19:09:57 momjian Exp $
|
* $Header: /cvsroot/pgsql/src/backend/storage/ipc/shmem.c,v 1.28 1998/07/21 06:17:35 vadim Exp $
|
||||||
*
|
*
|
||||||
*-------------------------------------------------------------------------
|
*-------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
@ -67,6 +67,8 @@
|
|||||||
#include "storage/proc.h"
|
#include "storage/proc.h"
|
||||||
#include "utils/dynahash.h"
|
#include "utils/dynahash.h"
|
||||||
#include "utils/hsearch.h"
|
#include "utils/hsearch.h"
|
||||||
|
#include "utils/memutils.h"
|
||||||
|
#include "access/transam.h"
|
||||||
|
|
||||||
/* shared memory global variables */
|
/* shared memory global variables */
|
||||||
|
|
||||||
@ -74,6 +76,8 @@ unsigned long ShmemBase = 0; /* start and end address of shared memory */
|
|||||||
static unsigned long ShmemEnd = 0;
|
static unsigned long ShmemEnd = 0;
|
||||||
static unsigned long ShmemSize = 0; /* current size (and default) */
|
static unsigned long ShmemSize = 0; /* current size (and default) */
|
||||||
|
|
||||||
|
extern VariableCache ShmemVariableCache; /* varsup.c */
|
||||||
|
|
||||||
SPINLOCK ShmemLock; /* lock for shared memory allocation */
|
SPINLOCK ShmemLock; /* lock for shared memory allocation */
|
||||||
|
|
||||||
SPINLOCK ShmemIndexLock; /* lock for shmem index access */
|
SPINLOCK ShmemIndexLock; /* lock for shmem index access */
|
||||||
@ -151,7 +155,6 @@ InitShmem(unsigned int key, unsigned int size)
|
|||||||
item;
|
item;
|
||||||
bool found;
|
bool found;
|
||||||
IpcMemoryId shmid;
|
IpcMemoryId shmid;
|
||||||
|
|
||||||
/* if zero key, use default memory size */
|
/* if zero key, use default memory size */
|
||||||
if (size)
|
if (size)
|
||||||
ShmemSize = size;
|
ShmemSize = size;
|
||||||
@ -180,9 +183,12 @@ InitShmem(unsigned int key, unsigned int size)
|
|||||||
ShmemFreeStart = (unsigned long *) ShmemBase;
|
ShmemFreeStart = (unsigned long *) ShmemBase;
|
||||||
/* next is a shmem pointer to the shmem index */
|
/* next is a shmem pointer to the shmem index */
|
||||||
ShmemIndexOffset = ShmemFreeStart + 1;
|
ShmemIndexOffset = ShmemFreeStart + 1;
|
||||||
|
/* next is ShmemVariableCache */
|
||||||
|
ShmemVariableCache = (VariableCache) (ShmemIndexOffset + 1);
|
||||||
|
|
||||||
currFreeSpace +=
|
currFreeSpace +=
|
||||||
sizeof(ShmemFreeStart) + sizeof(ShmemIndexOffset);
|
sizeof(ShmemFreeStart) + sizeof(ShmemIndexOffset) +
|
||||||
|
LONGALIGN(sizeof(VariableCacheData));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* bootstrap initialize spin locks so we can start to use the
|
* bootstrap initialize spin locks so we can start to use the
|
||||||
@ -196,7 +202,10 @@ InitShmem(unsigned int key, unsigned int size)
|
|||||||
* setup the global free space count
|
* setup the global free space count
|
||||||
*/
|
*/
|
||||||
if (ShmemBootstrap)
|
if (ShmemBootstrap)
|
||||||
|
{
|
||||||
*ShmemFreeStart = currFreeSpace;
|
*ShmemFreeStart = currFreeSpace;
|
||||||
|
memset (ShmemVariableCache, 0, sizeof(*ShmemVariableCache));
|
||||||
|
}
|
||||||
|
|
||||||
/* if ShmemFreeStart is NULL, then the allocator won't work */
|
/* if ShmemFreeStart is NULL, then the allocator won't work */
|
||||||
Assert(*ShmemFreeStart);
|
Assert(*ShmemFreeStart);
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
*
|
*
|
||||||
* Copyright (c) 1994, Regents of the University of California
|
* Copyright (c) 1994, Regents of the University of California
|
||||||
*
|
*
|
||||||
* $Id: transam.h,v 1.14 1998/02/26 04:40:30 momjian Exp $
|
* $Id: transam.h,v 1.15 1998/07/21 06:17:39 vadim Exp $
|
||||||
*
|
*
|
||||||
* NOTES
|
* NOTES
|
||||||
* Transaction System Version 101 now support proper oid
|
* Transaction System Version 101 now support proper oid
|
||||||
@ -114,6 +114,24 @@ typedef struct VariableRelationContentsData
|
|||||||
|
|
||||||
typedef VariableRelationContentsData *VariableRelationContents;
|
typedef VariableRelationContentsData *VariableRelationContents;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* VariableCache is placed in shmem and used by backends to
|
||||||
|
* get next available XID & OID without access to
|
||||||
|
* variable relation. Actually, I would like to have two
|
||||||
|
* different on-disk storages for next XID and OID...
|
||||||
|
* But hoping that someday we will use per database OID
|
||||||
|
* generator I leaved this as is. - vadim 07/21/98
|
||||||
|
*/
|
||||||
|
typedef struct VariableCacheData
|
||||||
|
{
|
||||||
|
uint32 xid_count;
|
||||||
|
TransactionId nextXid;
|
||||||
|
uint32 oid_count; /* not implemented, yet */
|
||||||
|
Oid nextOid;
|
||||||
|
} VariableCacheData;
|
||||||
|
|
||||||
|
typedef VariableCacheData *VariableCache;
|
||||||
|
|
||||||
/* ----------------
|
/* ----------------
|
||||||
* extern declarations
|
* extern declarations
|
||||||
* ----------------
|
* ----------------
|
||||||
|
Loading…
Reference in New Issue
Block a user