Commit b4a607c9 authored by Tom Lane's avatar Tom Lane

Modify RelationFlushRelation so that if the relcache entry

has positive refcount, it is rebuilt from pg_class data.  This ensures
that relcache entries will track changes made by other backends.  Formerly,
a shared inval report would just be ignored if it happened to arrive while
the relcache entry was in use.  Also, fix relcache to reset ref counts
to zero during transaction abort.  Finally, change LockRelation() so that
it checks for shared inval reports after obtaining the lock.  In this way,
once any kind of lock has been obtained on a rel, we can trust the relcache
entry to be up-to-date.
parent 8add6d71
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/access/transam/xact.c,v 1.47 1999/08/08 20:12:52 tgl Exp $ * $Header: /cvsroot/pgsql/src/backend/access/transam/xact.c,v 1.48 1999/09/04 18:42:15 tgl Exp $
* *
* NOTES * NOTES
* Transaction aborts can now occur two ways: * Transaction aborts can now occur two ways:
...@@ -755,6 +755,7 @@ static void ...@@ -755,6 +755,7 @@ static void
AtAbort_Cache() AtAbort_Cache()
{ {
RegisterInvalid(false); RegisterInvalid(false);
RelationCacheAbort();
} }
/* -------------------------------- /* --------------------------------
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/storage/lmgr/lmgr.c,v 1.33 1999/07/17 20:17:46 momjian Exp $ * $Header: /cvsroot/pgsql/src/backend/storage/lmgr/lmgr.c,v 1.34 1999/09/04 18:42:14 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -22,6 +22,7 @@ ...@@ -22,6 +22,7 @@
#include "postgres.h" #include "postgres.h"
#include "access/transam.h" #include "access/transam.h"
#include "catalog/catalog.h" #include "catalog/catalog.h"
#include "utils/inval.h"
extern Oid MyDatabaseId; extern Oid MyDatabaseId;
...@@ -161,7 +162,16 @@ LockRelation(Relation relation, LOCKMODE lockmode) ...@@ -161,7 +162,16 @@ LockRelation(Relation relation, LOCKMODE lockmode)
tag.objId.blkno = InvalidBlockNumber; tag.objId.blkno = InvalidBlockNumber;
LockAcquire(LockTableId, &tag, lockmode); LockAcquire(LockTableId, &tag, lockmode);
return;
/*
* Check to see if the relcache entry has been invalidated
* while we were waiting to lock it. If so, rebuild it,
* or elog() trying. Increment the refcount to ensure that
* RelationFlushRelation will rebuild it and not just delete it.
*/
RelationIncrementReferenceCount(relation);
DiscardInvalid();
RelationDecrementReferenceCount(relation);
} }
/* /*
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $Header: /cvsroot/pgsql/src/backend/utils/cache/relcache.c,v 1.68 1999/09/02 02:57:50 tgl Exp $ * $Header: /cvsroot/pgsql/src/backend/utils/cache/relcache.c,v 1.69 1999/09/04 18:42:13 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -61,6 +61,8 @@ ...@@ -61,6 +61,8 @@
static void RelationFlushRelation(Relation *relationPtr, static void RelationFlushRelation(Relation *relationPtr,
bool onlyFlushReferenceCountZero); bool onlyFlushReferenceCountZero);
static Relation RelationNameCacheGetRelation(char *relationName); static Relation RelationNameCacheGetRelation(char *relationName);
static void RelationCacheAbortWalker(Relation *relationPtr,
int dummy);
static void init_irels(void); static void init_irels(void);
static void write_irels(void); static void write_irels(void);
...@@ -213,14 +215,16 @@ static void RelationFlushIndexes(Relation *r, Oid accessMethodId); ...@@ -213,14 +215,16 @@ static void RelationFlushIndexes(Relation *r, Oid accessMethodId);
static HeapTuple ScanPgRelation(RelationBuildDescInfo buildinfo); static HeapTuple ScanPgRelation(RelationBuildDescInfo buildinfo);
static HeapTuple scan_pg_rel_seq(RelationBuildDescInfo buildinfo); static HeapTuple scan_pg_rel_seq(RelationBuildDescInfo buildinfo);
static HeapTuple scan_pg_rel_ind(RelationBuildDescInfo buildinfo); static HeapTuple scan_pg_rel_ind(RelationBuildDescInfo buildinfo);
static Relation AllocateRelationDesc(u_int natts, Form_pg_class relp); static Relation AllocateRelationDesc(Relation relation, u_int natts,
Form_pg_class relp);
static void RelationBuildTupleDesc(RelationBuildDescInfo buildinfo, static void RelationBuildTupleDesc(RelationBuildDescInfo buildinfo,
Relation relation, u_int natts); Relation relation, u_int natts);
static void build_tupdesc_seq(RelationBuildDescInfo buildinfo, static void build_tupdesc_seq(RelationBuildDescInfo buildinfo,
Relation relation, u_int natts); Relation relation, u_int natts);
static void build_tupdesc_ind(RelationBuildDescInfo buildinfo, static void build_tupdesc_ind(RelationBuildDescInfo buildinfo,
Relation relation, u_int natts); Relation relation, u_int natts);
static Relation RelationBuildDesc(RelationBuildDescInfo buildinfo); static Relation RelationBuildDesc(RelationBuildDescInfo buildinfo,
Relation oldrelation);
static void IndexedAccessMethodInitialize(Relation relation); static void IndexedAccessMethodInitialize(Relation relation);
static void AttrDefaultFetch(Relation relation); static void AttrDefaultFetch(Relation relation);
static void RelCheckFetch(Relation relation); static void RelCheckFetch(Relation relation);
...@@ -405,12 +409,16 @@ scan_pg_rel_ind(RelationBuildDescInfo buildinfo) ...@@ -405,12 +409,16 @@ scan_pg_rel_ind(RelationBuildDescInfo buildinfo)
* *
* This is used to allocate memory for a new relation descriptor * This is used to allocate memory for a new relation descriptor
* and initialize the rd_rel field. * and initialize the rd_rel field.
*
* If 'relation' is NULL, allocate a new RelationData object.
* If not, reuse the given object (that path is taken only when
* we have to rebuild a relcache entry during RelationFlushRelation).
* ---------------- * ----------------
*/ */
static Relation static Relation
AllocateRelationDesc(u_int natts, Form_pg_class relp) AllocateRelationDesc(Relation relation, u_int natts,
Form_pg_class relp)
{ {
Relation relation;
Size len; Size len;
Form_pg_class relationForm; Form_pg_class relationForm;
...@@ -424,10 +432,11 @@ AllocateRelationDesc(u_int natts, Form_pg_class relp) ...@@ -424,10 +432,11 @@ AllocateRelationDesc(u_int natts, Form_pg_class relp)
memmove((char *) relationForm, (char *) relp, CLASS_TUPLE_SIZE); memmove((char *) relationForm, (char *) relp, CLASS_TUPLE_SIZE);
/* ---------------- /* ----------------
* allocate space for new relation descriptor * allocate space for new relation descriptor, if needed
*/ */
len = sizeof(RelationData) + 10; /* + 10 is voodoo XXX mao */ len = sizeof(RelationData);
if (relation == NULL)
relation = (Relation) palloc(len); relation = (Relation) palloc(len);
/* ---------------- /* ----------------
...@@ -436,6 +445,9 @@ AllocateRelationDesc(u_int natts, Form_pg_class relp) ...@@ -436,6 +445,9 @@ AllocateRelationDesc(u_int natts, Form_pg_class relp)
*/ */
MemSet((char *) relation, 0, len); MemSet((char *) relation, 0, len);
/* make sure relation is marked as having no open file yet */
relation->rd_fd = -1;
/* initialize attribute tuple form */ /* initialize attribute tuple form */
relation->rd_att = CreateTemplateTupleDesc(natts); relation->rd_att = CreateTemplateTupleDesc(natts);
...@@ -737,6 +749,11 @@ RelationBuildRuleLock(Relation relation) ...@@ -737,6 +749,11 @@ RelationBuildRuleLock(Relation relation)
/* -------------------------------- /* --------------------------------
* RelationBuildDesc * RelationBuildDesc
* *
* Build a relation descriptor --- either a new one, or by
* recycling the given old relation object. The latter case
* supports rebuilding a relcache entry without invalidating
* pointers to it.
*
* To build a relation descriptor, we have to allocate space, * To build a relation descriptor, we have to allocate space,
* open the underlying unix file and initialize the following * open the underlying unix file and initialize the following
* fields: * fields:
...@@ -758,7 +775,8 @@ RelationBuildRuleLock(Relation relation) ...@@ -758,7 +775,8 @@ RelationBuildRuleLock(Relation relation)
* -------------------------------- * --------------------------------
*/ */
static Relation static Relation
RelationBuildDesc(RelationBuildDescInfo buildinfo) RelationBuildDesc(RelationBuildDescInfo buildinfo,
Relation oldrelation)
{ {
File fd; File fd;
Relation relation; Relation relation;
...@@ -803,7 +821,7 @@ RelationBuildDesc(RelationBuildDescInfo buildinfo) ...@@ -803,7 +821,7 @@ RelationBuildDesc(RelationBuildDescInfo buildinfo)
* initialize relation->rd_rel and get the access method id. * initialize relation->rd_rel and get the access method id.
* ---------------- * ----------------
*/ */
relation = AllocateRelationDesc(natts, relp); relation = AllocateRelationDesc(oldrelation, natts, relp);
relam = relation->rd_rel->relam; relam = relation->rd_rel->relam;
/* ---------------- /* ----------------
...@@ -833,9 +851,6 @@ RelationBuildDesc(RelationBuildDescInfo buildinfo) ...@@ -833,9 +851,6 @@ RelationBuildDesc(RelationBuildDescInfo buildinfo)
/* ---------------- /* ----------------
* initialize the tuple descriptor (relation->rd_att). * initialize the tuple descriptor (relation->rd_att).
* remember, rd_att is an array of attribute pointers that lives
* off the end of the relation descriptor structure so space was
* already allocated for it by AllocateRelationDesc.
* ---------------- * ----------------
*/ */
RelationBuildTupleDesc(buildinfo, relation, natts); RelationBuildTupleDesc(buildinfo, relation, natts);
...@@ -1153,7 +1168,7 @@ RelationIdGetRelation(Oid relationId) ...@@ -1153,7 +1168,7 @@ RelationIdGetRelation(Oid relationId)
buildinfo.infotype = INFO_RELID; buildinfo.infotype = INFO_RELID;
buildinfo.i.info_id = relationId; buildinfo.i.info_id = relationId;
rd = RelationBuildDesc(buildinfo); rd = RelationBuildDesc(buildinfo, NULL);
return rd; return rd;
} }
...@@ -1193,7 +1208,7 @@ RelationNameGetRelation(char *relationName) ...@@ -1193,7 +1208,7 @@ RelationNameGetRelation(char *relationName)
buildinfo.infotype = INFO_RELNAME; buildinfo.infotype = INFO_RELNAME;
buildinfo.i.info_name = relationName; buildinfo.i.info_name = relationName;
rd = RelationBuildDesc(buildinfo); rd = RelationBuildDesc(buildinfo, NULL);
return rd; return rd;
} }
...@@ -1236,7 +1251,7 @@ RelationClose(Relation relation) ...@@ -1236,7 +1251,7 @@ RelationClose(Relation relation)
/* -------------------------------- /* --------------------------------
* RelationFlushRelation * RelationFlushRelation
* *
* Actually blows away a relation... RelationFree doesn't do * Actually blows away a relation cache entry... RelationFree doesn't do
* anything anymore. * anything anymore.
* -------------------------------- * --------------------------------
*/ */
...@@ -1244,50 +1259,74 @@ static void ...@@ -1244,50 +1259,74 @@ static void
RelationFlushRelation(Relation *relationPtr, RelationFlushRelation(Relation *relationPtr,
bool onlyFlushReferenceCountZero) bool onlyFlushReferenceCountZero)
{ {
MemoryContext oldcxt;
Relation relation = *relationPtr; Relation relation = *relationPtr;
MemoryContext oldcxt;
/*
* Make sure smgr and lower levels close the relation's files,
* if they weren't closed already. We do this unconditionally;
* if the relation is not deleted, the next smgr access should
* reopen the files automatically. This ensures that the low-level
* file access state is updated after, say, a vacuum truncation.
* NOTE: this call is a no-op if the relation's smgr file is already
* closed or unlinked.
*/
smgrclose(DEFAULT_SMGR, relation);
if (relation->rd_isnailed) if (relation->rd_isnailed)
{ {
/* this is a nailed special relation for bootstraping */ /* this is a nailed special relation for bootstrapping */
return; return;
} }
if (!onlyFlushReferenceCountZero ||
RelationHasReferenceCountZero(relation))
{
oldcxt = MemoryContextSwitchTo((MemoryContext) CacheCxt); oldcxt = MemoryContextSwitchTo((MemoryContext) CacheCxt);
/* make sure smgr and lower levels close the relation's files, /* Remove relation from hash tables
* if they weren't closed already *
* Note: we might be reinserting it momentarily, but we must not have it
* visible in the hash tables until it's valid again, so don't try to
* optimize this away...
*/ */
smgrclose(DEFAULT_SMGR, relation);
RelationCacheDelete(relation); RelationCacheDelete(relation);
FreeTupleDesc(relation->rd_att); /* Clear out catcache's entries for this relation */
SystemCacheRelationFlushed(RelationGetRelid(relation)); SystemCacheRelationFlushed(RelationGetRelid(relation));
/* Free all the subsidiary data structures of the relcache entry */
FreeTupleDesc(relation->rd_att);
FreeTriggerDesc(relation); FreeTriggerDesc(relation);
pfree(RelationGetLockInfo(relation));
pfree(RelationGetForm(relation));
#ifdef NOT_USED /* If we're really done with the relcache entry, blow it away.
if (relation->rd_rules) * But if someone is still using it, reconstruct the whole deal
* without moving the physical RelationData record (so that the
* someone's pointer is still valid). Preserve ref count, too.
*/
if (!onlyFlushReferenceCountZero ||
RelationHasReferenceCountZero(relation))
{ {
int j; pfree(relation);
for (j = 0; j < relation->rd_rules->numLocks; j++)
pfree(relation->rd_rules->rules[j]);
pfree(relation->rd_rules->rules);
pfree(relation->rd_rules);
} }
#endif else
{
uint16 old_refcnt = relation->rd_refcnt;
RelationBuildDescInfo buildinfo;
pfree(RelationGetLockInfo(relation)); buildinfo.infotype = INFO_RELID;
pfree(RelationGetForm(relation)); buildinfo.i.info_id = RelationGetRelid(relation);
if (RelationBuildDesc(buildinfo, relation) != relation)
{
/* Should only get here if relation was deleted */
pfree(relation); pfree(relation);
elog(ERROR, "RelationFlushRelation: relation %u deleted while still in use",
buildinfo.i.info_id);
}
RelationSetReferenceCount(relation, old_refcnt);
}
MemoryContextSwitchTo(oldcxt); MemoryContextSwitchTo(oldcxt);
}
} }
/* -------------------------------- /* --------------------------------
...@@ -1442,6 +1481,36 @@ RelationCacheInvalidate(bool onlyFlushReferenceCountZero) ...@@ -1442,6 +1481,36 @@ RelationCacheInvalidate(bool onlyFlushReferenceCountZero)
} }
} }
/*
* RelationCacheAbort
*
* Clean up the relcache at transaction abort.
*
* What we need to do here is reset relcache entry ref counts to
* their normal not-in-a-transaction state. A ref count may be
* too high because some routine was exited by elog() between
* incrementing and decrementing the count.
*
* XXX Maybe we should do this at transaction commit, too, in case
* someone forgets to decrement a refcount in a non-error path?
*/
void
RelationCacheAbort(void)
{
HashTableWalk(RelationNameCache, (HashtFunc) RelationCacheAbortWalker,
0);
}
static void
RelationCacheAbortWalker(Relation *relationPtr, int dummy)
{
Relation relation = *relationPtr;
if (relation->rd_isnailed)
RelationSetReferenceCount(relation, 1);
else
RelationSetReferenceCount(relation, 0);
}
/* -------------------------------- /* --------------------------------
* RelationRegisterRelation - * RelationRegisterRelation -
...@@ -1523,7 +1592,7 @@ RelationPurgeLocalRelation(bool xactCommitted) ...@@ -1523,7 +1592,7 @@ RelationPurgeLocalRelation(bool xactCommitted)
reln->rd_myxactonly = FALSE; reln->rd_myxactonly = FALSE;
if (!IsBootstrapProcessingMode()) if (!IsBootstrapProcessingMode())
RelationFlushRelation(&reln, FALSE); RelationFlushRelation(&reln, false);
newlyCreatedRelns = lnext(newlyCreatedRelns); newlyCreatedRelns = lnext(newlyCreatedRelns);
pfree(l); pfree(l);
...@@ -2010,15 +2079,15 @@ write_irels(void) ...@@ -2010,15 +2079,15 @@ write_irels(void)
bi.infotype = INFO_RELNAME; bi.infotype = INFO_RELNAME;
bi.i.info_name = AttributeNumIndex; bi.i.info_name = AttributeNumIndex;
irel[0] = RelationBuildDesc(bi); irel[0] = RelationBuildDesc(bi, NULL);
irel[0]->rd_isnailed = true; irel[0]->rd_isnailed = true;
bi.i.info_name = ClassNameIndex; bi.i.info_name = ClassNameIndex;
irel[1] = RelationBuildDesc(bi); irel[1] = RelationBuildDesc(bi, NULL);
irel[1]->rd_isnailed = true; irel[1]->rd_isnailed = true;
bi.i.info_name = ClassOidIndex; bi.i.info_name = ClassOidIndex;
irel[2] = RelationBuildDesc(bi); irel[2] = RelationBuildDesc(bi, NULL);
irel[2]->rd_isnailed = true; irel[2]->rd_isnailed = true;
SetProcessingMode(oldmode); SetProcessingMode(oldmode);
......
...@@ -6,7 +6,7 @@ ...@@ -6,7 +6,7 @@
* *
* Copyright (c) 1994, Regents of the University of California * Copyright (c) 1994, Regents of the University of California
* *
* $Id: relcache.h,v 1.13 1999/07/15 23:04:23 momjian Exp $ * $Id: relcache.h,v 1.14 1999/09/04 18:42:11 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -34,6 +34,8 @@ extern void RelationRegisterRelation(Relation relation); ...@@ -34,6 +34,8 @@ extern void RelationRegisterRelation(Relation relation);
extern void RelationPurgeLocalRelation(bool xactComitted); extern void RelationPurgeLocalRelation(bool xactComitted);
extern void RelationInitialize(void); extern void RelationInitialize(void);
extern void RelationCacheAbort(void);
/* /*
* both vacuum.c and relcache.c need to know the name of the relcache init file * both vacuum.c and relcache.c need to know the name of the relcache init file
*/ */
......
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