up the comments later.
*
*
* IDENTIFICATION
- * $Header: /cvsroot/pgsql/src/backend/utils/cache/catcache.c,v 1.75 2001/01/24 19:43:14 momjian Exp $
+ * $Header: /cvsroot/pgsql/src/backend/utils/cache/catcache.c,v 1.76 2001/02/22 18:39:19 momjian Exp $
*
*-------------------------------------------------------------------------
*/
static void CatalogCacheInitializeCache(CatCache *cache);
static Datum cc_hashname(PG_FUNCTION_ARGS);
-/* ----------------
+/*
* variables, macros and other stuff
- * ----------------
+ *
*/
#ifdef CACHEDEBUG
static CatCache *Caches = NULL; /* head of list of caches */
-/* ----------------
+/*
* EQPROC is used in CatalogCacheInitializeCache to find the equality
* functions for system types that are used as cache key fields.
* See also GetCCHashFunc, which should support the same set of types.
*
* XXX this should be replaced by catalog lookups,
* but that seems to pose considerable risk of circularity...
- * ----------------
+ *
*/
static const Oid eqproc[] = {
F_BOOLEQ, InvalidOid, F_CHAREQ, F_NAMEEQ, InvalidOid,
#define EQPROC(SYSTEMTYPEOID) eqproc[(SYSTEMTYPEOID)-BOOLOID]
-/* ----------------------------------------------------------------
+/*
* internal support functions
- * ----------------------------------------------------------------
+ *
*/
static PGFunction
}
-/* --------------------------------
+/*
* CatalogCacheInitializeCache
*
* This function does final initialization of a catcache: obtain the tuple
* descriptor and set up the hash and equality function links. We assume
* that the relcache entry can be opened at this point!
- * --------------------------------
+ *
*/
#ifdef CACHEDEBUG
#define CatalogCacheInitializeCache_DEBUG1 \
relation = heap_openr(cache->cc_relname, NoLock);
Assert(RelationIsValid(relation));
- /* ----------------
+ /*
* switch to the cache context so our allocations
* do not vanish at the end of a transaction
- * ----------------
+ *
*/
if (!CacheMemoryContext)
CreateCacheMemoryContext();
oldcxt = MemoryContextSwitchTo(CacheMemoryContext);
- /* ----------------
+ /*
* copy the relcache's tuple descriptor to permanent cache storage
- * ----------------
+ *
*/
tupdesc = CreateTupleDescCopyConstr(RelationGetDescr(relation));
- /* ----------------
+ /*
* return to the caller's memory context and close the rel
- * ----------------
+ *
*/
MemoryContextSwitchTo(oldcxt);
CACHE3_elog(DEBUG, "CatalogCacheInitializeCache: %s, %d keys",
cache->cc_relname, cache->cc_nkeys);
- /* ----------------
+ /*
* initialize cache's key information
- * ----------------
+ *
*/
for (i = 0; i < cache->cc_nkeys; ++i)
{
cache);
}
- /* ----------------
+ /*
* mark this cache fully initialized
- * ----------------
+ *
*/
cache->cc_tupdesc = tupdesc;
}
-/* --------------------------------
+/*
* CatalogCacheComputeHashIndex
- * --------------------------------
+ *
*/
static Index
CatalogCacheComputeHashIndex(CatCache *cache, ScanKey cur_skey)
return (Index) hashIndex;
}
-/* --------------------------------
+/*
* CatalogCacheComputeTupleHashIndex
- * --------------------------------
+ *
*/
static Index
CatalogCacheComputeTupleHashIndex(CatCache *cache,
return CatalogCacheComputeHashIndex(cache, cur_skey);
}
-/* --------------------------------
+/*
* CatCacheRemoveCTup
- * --------------------------------
+ *
*/
static void
CatCacheRemoveCTup(CatCache *cache, CatCTup *ct)
--cache->cc_ntup;
}
-/* --------------------------------
+/*
* CatalogCacheIdInvalidate()
*
* Invalidate a tuple given a cache id. In this case the id should always
* be found (whether the cache has opened its relation or not). Of course,
* if the cache has yet to open its relation, there will be no tuples so
* no problem.
- * --------------------------------
+ *
*/
void
CatalogCacheIdInvalidate(int cacheId,
{
CatCache *ccp;
- /* ----------------
+ /*
* sanity checks
- * ----------------
+ *
*/
Assert(hashIndex < NCCBUCK);
Assert(ItemPointerIsValid(pointer));
CACHE1_elog(DEBUG, "CatalogCacheIdInvalidate: called");
- /* ----------------
+ /*
* inspect caches to find the proper cache
- * ----------------
+ *
*/
for (ccp = Caches; ccp; ccp = ccp->cc_next)
{
if (cacheId != ccp->id)
continue;
- /* ----------------
+ /*
* inspect the hash bucket until we find a match or exhaust
- * ----------------
+ *
*/
for (elt = DLGetHead(&ccp->cc_cache[hashIndex]); elt; elt = nextelt)
{
*/
-/* --------------------------------
+/*
* AtEOXact_CatCache
*
* Clean up catcaches at end of transaction (either commit or abort)
* necessary in the abort case, since elog() may have interrupted routines.
* In the commit case, any nonzero counts indicate failure to call
* ReleaseSysCache, so we put out a notice for debugging purposes.
- * --------------------------------
+ *
*/
void
AtEOXact_CatCache(bool isCommit)
}
}
-/* --------------------------------
+/*
* ResetSystemCache
*
* Reset caches when a shared cache inval event forces it
- * --------------------------------
+ *
*/
void
ResetSystemCache(void)
CACHE1_elog(DEBUG, "end of ResetSystemCache call");
}
-/* --------------------------------
+/*
* SystemCacheRelationFlushed
*
* This is called by RelationFlushRelation() to clear out cached information
* We now avoid the need to do it by copying cc_tupdesc out of the relcache,
* rather than relying on the relcache to keep a tupdesc for us. Of course
* this assumes the tupdesc of a cachable system table will not change...)
- * --------------------------------
+ *
*/
void
SystemCacheRelationFlushed(Oid relId)
ResetSystemCache();
}
-/* --------------------------------
+/*
* InitCatCache
*
* This allocates and initializes a cache for a system catalog relation.
* Actually, the cache is only partially initialized to avoid opening the
* relation. The relation will be opened and the rest of the cache
* structure initialized on the first access.
- * --------------------------------
+ *
*/
#ifdef CACHEDEBUG
#define InitCatCache_DEBUG1 \
MemoryContext oldcxt;
int i;
- /* ----------------
+ /*
* first switch to the cache context so our allocations
* do not vanish at the end of a transaction
- * ----------------
+ *
*/
if (!CacheMemoryContext)
CreateCacheMemoryContext();
oldcxt = MemoryContextSwitchTo(CacheMemoryContext);
- /* ----------------
+ /*
* allocate a new cache structure
- * ----------------
+ *
*/
cp = (CatCache *) palloc(sizeof(CatCache));
MemSet((char *) cp, 0, sizeof(CatCache));
- /* ----------------
+ /*
* initialize the cache buckets (each bucket is a list header)
* and the LRU tuple list
- * ----------------
+ *
*/
DLInitList(&cp->cc_lrulist);
for (i = 0; i < NCCBUCK; ++i)
DLInitList(&cp->cc_cache[i]);
- /* ----------------
+ /*
* Caches is the pointer to the head of the list of all the
* system caches. here we add the new cache to the top of the list.
- * ----------------
+ *
*/
cp->cc_next = Caches; /* list of caches (single link) */
Caches = cp;
- /* ----------------
+ /*
* initialize the cache's relation information for the relation
* corresponding to this cache, and initialize some of the new
* cache's other internal fields. But don't open the relation yet.
- * ----------------
+ *
*/
cp->cc_relname = relname;
cp->cc_indname = indname;
for (i = 0; i < nkeys; ++i)
cp->cc_key[i] = key[i];
- /* ----------------
+ /*
* all done. new cache is initialized. print some debugging
* information, if appropriate.
- * ----------------
+ *
*/
InitCatCache_DEBUG1;
- /* ----------------
+ /*
* back to the old context before we return...
- * ----------------
+ *
*/
MemoryContextSwitchTo(oldcxt);
}
-/* --------------------------------
+/*
* IndexScanOK
*
* This function checks for tuples that will be fetched by
* certain system indexes that support critical syscaches.
* We can't use an indexscan to fetch these, else we'll get into
* infinite recursion. A plain heap scan will work, however.
- * --------------------------------
+ *
*/
static bool
IndexScanOK(CatCache *cache, ScanKey cur_skey)
return true;
}
-/* --------------------------------
+/*
* SearchCatCache
*
* This call searches a system cache for a tuple, opening the relation
* if necessary (the first access to a particular cache).
- * --------------------------------
+ *
*/
HeapTuple
SearchCatCache(CatCache *cache,
Relation relation;
MemoryContext oldcxt;
- /* ----------------
+ /*
* one-time startup overhead
- * ----------------
+ *
*/
if (cache->cc_tupdesc == NULL)
CatalogCacheInitializeCache(cache);
- /* ----------------
+ /*
* initialize the search key information
- * ----------------
+ *
*/
memcpy(cur_skey, cache->cc_skey, sizeof(cur_skey));
cur_skey[0].sk_argument = v1;
cur_skey[2].sk_argument = v3;
cur_skey[3].sk_argument = v4;
- /* ----------------
+ /*
* find the hash bucket in which to look for the tuple
- * ----------------
+ *
*/
hash = CatalogCacheComputeHashIndex(cache, cur_skey);
- /* ----------------
+ /*
* scan the hash bucket until we find a match or exhaust our tuples
- * ----------------
+ *
*/
for (elt = DLGetHead(&cache->cc_cache[hash]);
elt;
if (ct->dead)
continue; /* ignore dead entries */
- /* ----------------
+ /*
* see if the cached tuple matches our key.
* (should we be worried about time ranges? -cim 10/2/90)
- * ----------------
+ *
*/
HeapKeyTest(&ct->tuple,
cache->cc_tupdesc,
if (! res)
continue;
- /* ----------------
+ /*
* we found a tuple in the cache: bump its refcount, move it to
* the front of the LRU list, and return it. We also move it
* to the front of the list for its hashbucket, in order to speed
* subsequent searches. (The most frequently accessed elements
* in any hashbucket will tend to be near the front of the
* hashbucket's list.)
- * ----------------
+ *
*/
ct->refcount++;
return &ct->tuple;
}
- /* ----------------
+ /*
* Tuple was not found in cache, so we have to try and
* retrieve it directly from the relation. If it's found,
* we add it to the cache.
* first copy will never be referenced again, and will eventually
* age out of the cache, so there's no functional problem. This case
* is rare enough that it's not worth expending extra cycles to detect.
- * ----------------
+ *
*/
- /* ----------------
+ /*
* open the relation associated with the cache
- * ----------------
+ *
*/
relation = heap_openr(cache->cc_relname, AccessShareLock);
- /* ----------------
+ /*
* Scan the relation to find the tuple. If there's an index, and
* if it's safe to do so, use the index. Else do a heap scan.
- * ----------------
+ *
*/
ct = NULL;
heap_endscan(sd);
}
- /* ----------------
+ /*
* close the relation
- * ----------------
+ *
*/
heap_close(relation, AccessShareLock);
- /* ----------------
+ /*
* scan is complete. if tup was found, we can add it to the cache.
- * ----------------
+ *
*/
if (ct == NULL)
return NULL;
- /* ----------------
+ /*
* Finish initializing the CatCTup header, and add it to the
* linked lists.
- * ----------------
+ *
*/
CACHE1_elog(DEBUG, "SearchCatCache: found tuple");
DLAddHead(&cache->cc_lrulist, &ct->lrulist_elem);
DLAddHead(&cache->cc_cache[hash], &ct->cache_elem);
- /* ----------------
+ /*
* If we've exceeded the desired size of this cache,
* try to throw away the least recently used entry.
- * ----------------
+ *
*/
if (++cache->cc_ntup > cache->cc_maxtup)
{
return &ct->tuple;
}
-/* --------------------------------
+/*
* ReleaseCatCache()
*
* Decrement the reference count of a catcache entry (releasing the
* will be freed as soon as their refcount goes to zero. In combination
* with aset.c's CLOBBER_FREED_MEMORY option, this provides a good test
* to catch references to already-released catcache entries.
- * --------------------------------
+ *
*/
void
ReleaseCatCache(HeapTuple tuple)
}
}
-/* --------------------------------
+/*
* PrepareToInvalidateCacheTuple()
*
* This is part of a rather subtle chain of events, so pay attention:
* specified relation. inval.c doesn't know exactly which rels have
* catcaches --- it will call this routine for any tuple that's in a
* system relation.
- * --------------------------------
+ *
*/
void
PrepareToInvalidateCacheTuple(Relation relation,
{
CatCache *ccp;
- /* ----------------
+ /*
* sanity checks
- * ----------------
+ *
*/
Assert(RelationIsValid(relation));
Assert(HeapTupleIsValid(tuple));
* Portions Copyright (c) 1994, Regents of the University of California
*
* IDENTIFICATION
- * $Header: /cvsroot/pgsql/src/backend/utils/cache/inval.c,v 1.40 2001/01/24 19:43:14 momjian Exp $
+ * $Header: /cvsroot/pgsql/src/backend/utils/cache/inval.c,v 1.41 2001/02/22 18:39:19 momjian Exp $
*
* Note - this code is real crufty... badly needs a rewrite to improve
* readability and portability. (Shouldn't assume Oid == Index, for example)
#include "utils/relcache.h"
-/* ----------------
+/*
* private invalidation structures
- * ----------------
+ *
*/
typedef struct InvalidationUserData
typedef InvalidationMessageData *InvalidationMessage;
-/* ----------------
+/*
* variables and macros
- * ----------------
+ *
*/
/*
* ----------------------------------------------------------------
*/
-/* --------------------------------
+/*
* InvalidationEntryAllocate
* Allocates an invalidation entry.
- * --------------------------------
+ *
*/
static InvalidationEntry
InvalidationEntryAllocate(uint16 size)
return (Pointer) &entryDataP->userData;
}
-/* --------------------------------
+/*
* LocalInvalidRegister
* Link an invalidation entry into a chain of them. Really ugly
* coding here.
- * --------------------------------
+ *
*/
static LocalInvalid
LocalInvalidRegister(LocalInvalid invalid,
return entry;
}
-/* --------------------------------
+/*
* LocalInvalidInvalidate
* Processes, then frees all entries in a local cache
* invalidation list unless freemember parameter is false.
- * --------------------------------
+ *
*/
static void
LocalInvalidInvalidate(LocalInvalid invalid,
#define CacheIdRegisterLocalRollback_DEBUG1
#endif /* INVALIDDEBUG */
-/* --------------------------------
+/*
* CacheIdRegisterSpecifiedLocalInvalid
- * --------------------------------
+ *
*/
static LocalInvalid
CacheIdRegisterSpecifiedLocalInvalid(LocalInvalid invalid,
{
InvalidationMessage message;
- /* ----------------
+ /*
* debugging stuff
- * ----------------
+ *
*/
CacheIdRegisterSpecifiedLocalInvalid_DEBUG1;
- /* ----------------
+ /*
* create a message describing the system catalog tuple
* we wish to invalidate.
- * ----------------
+ *
*/
message = (InvalidationMessage)
InvalidationEntryAllocate(sizeof(InvalidationMessageData));
ItemPointerCopy(pointer, &message->any.catalog.pointerData);
- /* ----------------
+ /*
* Add message to linked list of unprocessed messages.
- * ----------------
+ *
*/
invalid = LocalInvalidRegister(invalid, (InvalidationEntry) message);
return invalid;
}
-/* --------------------------------
+/*
* CacheIdRegisterLocalInvalid
- * --------------------------------
+ *
*/
static void
CacheIdRegisterLocalInvalid(int cacheId,
Index hashIndex,
ItemPointer pointer)
{
- /* ----------------
+ /*
* debugging stuff
- * ----------------
+ *
*/
CacheIdRegisterLocalInvalid_DEBUG1;
- /* ----------------
+ /*
* Add message to InvalidForall linked list.
- * ----------------
+ *
*/
InvalidForall = CacheIdRegisterSpecifiedLocalInvalid(InvalidForall,
cacheId, hashIndex, pointer);
- /* ----------------
+ /*
* Add message to InvalidLocal linked list.
- * ----------------
+ *
*/
InvalidLocal = CacheIdRegisterSpecifiedLocalInvalid(InvalidLocal,
cacheId, hashIndex, pointer);
}
-/* --------------------------------
+/*
* CacheIdRegisterLocalRollback
- * --------------------------------
+ *
*/
static void
CacheIdRegisterLocalRollback(int cacheId,
ItemPointer pointer)
{
- /* ----------------
+ /*
* debugging stuff
- * ----------------
+ *
*/
CacheIdRegisterLocalRollback_DEBUG1;
- /* ----------------
+ /*
* Add message to RollbackStack linked list.
- * ----------------
+ *
*/
RollbackStack = CacheIdRegisterSpecifiedLocalInvalid(
RollbackStack, cacheId, hashIndex, pointer);
}
-/* --------------------------------
+/*
* RelationIdRegisterSpecifiedLocalInvalid
- * --------------------------------
+ *
*/
static LocalInvalid
RelationIdRegisterSpecifiedLocalInvalid(LocalInvalid invalid,
{
InvalidationMessage message;
- /* ----------------
+ /*
* debugging stuff
- * ----------------
+ *
*/
#ifdef INVALIDDEBUG
elog(DEBUG, "RelationRegisterSpecifiedLocalInvalid(%u, %u)", relationId,
objectId);
#endif /* defined(INVALIDDEBUG) */
- /* ----------------
+ /*
* create a message describing the relation descriptor
* we wish to invalidate.
- * ----------------
+ *
*/
message = (InvalidationMessage)
InvalidationEntryAllocate(sizeof(InvalidationMessageData));
message->any.relation.relationId = relationId;
message->any.relation.objectId = objectId;
- /* ----------------
+ /*
* Add message to linked list of unprocessed messages.
- * ----------------
+ *
*/
invalid = LocalInvalidRegister(invalid, (InvalidationEntry) message);
return invalid;
}
-/* --------------------------------
+/*
* RelationIdRegisterLocalInvalid
- * --------------------------------
+ *
*/
static void
RelationIdRegisterLocalInvalid(Oid relationId, Oid objectId)
{
- /* ----------------
+ /*
* debugging stuff
- * ----------------
+ *
*/
#ifdef INVALIDDEBUG
elog(DEBUG, "RelationRegisterLocalInvalid(%u, %u)", relationId,
objectId);
#endif /* defined(INVALIDDEBUG) */
- /* ----------------
+ /*
* Add message to InvalidForall linked list.
- * ----------------
+ *
*/
InvalidForall = RelationIdRegisterSpecifiedLocalInvalid(InvalidForall,
relationId, objectId);
- /* ----------------
+ /*
* Add message to InvalidLocal linked list.
- * ----------------
+ *
*/
InvalidLocal = RelationIdRegisterSpecifiedLocalInvalid(InvalidLocal,
relationId, objectId);
}
-/* --------------------------------
+/*
* RelationIdRegisterLocalRollback
- * --------------------------------
+ *
*/
static void
RelationIdRegisterLocalRollback(Oid relationId, Oid objectId)
{
- /* ----------------
+ /*
* debugging stuff
- * ----------------
+ *
*/
#ifdef INVALIDDEBUG
elog(DEBUG, "RelationRegisterLocalRollback(%u, %u)", relationId,
objectId);
#endif /* defined(INVALIDDEBUG) */
- /* ----------------
+ /*
* Add message to RollbackStack linked list.
- * ----------------
+ *
*/
RollbackStack = RelationIdRegisterSpecifiedLocalInvalid(
RollbackStack, relationId, objectId);
}
-/* --------------------------------
+/*
* CacheIdInvalidate
*
* This routine can invalidate a tuple in a system catalog cache
* or a cached relation descriptor. You pay your money and you
* take your chances...
- * --------------------------------
+ *
*/
#ifdef INVALIDDEBUG
#define CacheIdInvalidate_DEBUG1 \
Index hashIndex,
ItemPointer pointer)
{
- /* ----------------
+ /*
* assume that if the item pointer is valid, then we are
* invalidating an item in the specified system catalog cache.
- * ----------------
+ *
*/
if (ItemPointerIsValid(pointer))
{
CacheIdInvalidate_DEBUG1;
- /* ----------------
+ /*
* if the cacheId is the oid of any of the following system relations,
* then assume we are invalidating a relation descriptor
- * ----------------
+ *
*/
if (cacheId == RelOid_pg_class)
{
return;
}
- /* ----------------
+ /*
* Yow! the caller asked us to invalidate something else.
- * ----------------
+ *
*/
elog(FATAL, "CacheIdInvalidate: cacheId=%d relation id?", cacheId);
}
-/* --------------------------------
+/*
* ResetSystemCaches
*
* This blows away all tuples in the system catalog caches and
* all the cached relation descriptors (and closes their files too).
* Relation descriptors that have positive refcounts are then rebuilt.
- * --------------------------------
+ *
*/
static void
ResetSystemCaches(void)
RelationCacheInvalidate();
}
-/* --------------------------------
+/*
* InvalidationMessageRegisterSharedInvalid
- * --------------------------------
+ *
*/
#ifdef INVALIDDEBUG
#define InvalidationMessageRegisterSharedInvalid_DEBUG1 \
}
}
-/* --------------------------------
+/*
* InvalidationMessageCacheInvalidate
- * --------------------------------
+ *
*/
#ifdef INVALIDDEBUG
#define InvalidationMessageCacheInvalidate_DEBUG1 \
}
}
-/* --------------------------------
+/*
* PrepareToInvalidateRelationCache
- * --------------------------------
+ *
*/
static void
PrepareToInvalidateRelationCache(Relation relation,
Oid relationId;
Oid objectId;
- /* ----------------
+ /*
* get the relation object id
- * ----------------
+ *
*/
relationId = RelationGetRelid(relation);
- /* ----------------
+ /*
* is it one of the ones we need to send an SI message for?
- * ----------------
+ *
*/
if (relationId == RelOid_pg_class)
objectId = tuple->t_data->t_oid;
else
return;
- /* ----------------
+ /*
* register the relcache-invalidation action in the appropriate list
- * ----------------
+ *
*/
Assert(PointerIsValid(function));
void
DiscardInvalid(void)
{
- /* ----------------
+ /*
* debugging stuff
- * ----------------
+ *
*/
#ifdef INVALIDDEBUG
elog(DEBUG, "DiscardInvalid called");
{
LocalInvalid invalid;
- /* ----------------
+ /*
* debugging stuff
- * ----------------
+ *
*/
#ifdef INVALIDDEBUG
elog(DEBUG, "RegisterInvalid(%d) called", send);
#endif /* defined(INVALIDDEBUG) */
- /* ----------------
+ /*
* Process and free the current list of inval messages.
- * ----------------
+ *
*/
DiscardInvalidStack(&InvalidLocal);
{
LocalInvalid invalid;
- /* ----------------
+ /*
* debugging stuff
- * ----------------
+ *
*/
#ifdef INVALIDDEBUG
elog(DEBUG, "ImmediateLocalInvalidation(%d) called", send);
#endif /* defined(INVALIDDEBUG) */
- /* ----------------
+ /*
* Process and free the local list of inval messages.
- * ----------------
+ *
*/
if (send)
void (*RelationIdRegisterFunc) (Oid, Oid),
const char *funcname)
{
- /* ----------------
+ /*
* sanity checks
- * ----------------
+ *
*/
Assert(RelationIsValid(relation));
Assert(HeapTupleIsValid(tuple));
if (IsBootstrapProcessingMode())
return;
- /* ----------------
+ /*
* We only need to worry about invalidation for tuples that are in
* system relations; user-relation tuples are never in catcaches
* and can't affect the relcache either.
- * ----------------
+ *
*/
if (!IsSystemRelationName(NameStr(RelationGetForm(relation)->relname)))
return;
- /* ----------------
+ /*
* debugging stuff
- * ----------------
+ *
*/
PrepareForTupleInvalidation_DEBUG1;
*
*
* IDENTIFICATION
- * $Header: /cvsroot/pgsql/src/backend/utils/cache/relcache.c,v 1.127 2001/01/24 19:43:15 momjian Exp $
+ * $Header: /cvsroot/pgsql/src/backend/utils/cache/relcache.c,v 1.128 2001/02/22 18:39:19 momjian Exp $
*
*-------------------------------------------------------------------------
*/
#include "utils/temprel.h"
-/* ----------------
+/*
* hardcoded tuple descriptors. see lib/backend/catalog/pg_attribute.h
- * ----------------
+ *
*/
static FormData_pg_attribute Desc_pg_class[Natts_pg_class] = {Schema_pg_class};
static FormData_pg_attribute Desc_pg_attribute[Natts_pg_attribute] = {Schema_pg_attribute};
static FormData_pg_attribute Desc_pg_variable[Natts_pg_variable] = {Schema_pg_variable};
static FormData_pg_attribute Desc_pg_log[Natts_pg_log] = {Schema_pg_log};
-/* ----------------
+/*
* Hash tables that index the relation cache
*
* Relations are looked up two ways, by name and by id,
* thus there are two hash tables for referencing them.
- * ----------------
+ *
*/
static HTAB *RelationNameCache;
static HTAB *RelationIdCache;
static bool criticalRelcachesBuilt = false;
-/* ----------------
+/*
* RelationBuildDescInfo exists so code can be shared
* between RelationIdGetRelation() and RelationNameGetRelation()
- * ----------------
+ *
*/
typedef struct RelationBuildDescInfo
{
Relation reldesc;
} RelNodeCacheEnt;
-/* -----------------
+/*
* macros to manipulate name cache and id cache
- * -----------------
+ *
*/
#define RelationCacheInsert(RELATION) \
do { \
static List *insert_ordered_oid(List *list, Oid datum);
-/* ----------------------------------------------------------------
+/*
* RelationIdGetRelation() and RelationNameGetRelation()
* support functions
- * ----------------------------------------------------------------
+ *
*/
-/* --------------------------------
+/*
* ScanPgRelation
*
* this is used by RelationBuildDesc to find a pg_class
*
* NB: the returned tuple has been copied into palloc'd storage
* and must eventually be freed with heap_freetuple.
- * --------------------------------
+ *
*/
static HeapTuple
ScanPgRelation(RelationBuildDescInfo buildinfo)
HeapScanDesc pg_class_scan;
ScanKeyData key;
- /* ----------------
+ /*
* form a scan key
- * ----------------
+ *
*/
switch (buildinfo.infotype)
{
return NULL;
}
- /* ----------------
+ /*
* open pg_class and fetch a tuple
- * ----------------
+ *
*/
pg_class_desc = heap_openr(RelationRelationName, AccessShareLock);
pg_class_scan = heap_beginscan(pg_class_desc, 0, SnapshotNow, 1, &key);
pg_class_tuple = heap_getnext(pg_class_scan, 0);
- /* ----------------
+ /*
* get set to return tuple
- * ----------------
+ *
*/
if (!HeapTupleIsValid(pg_class_tuple))
return_tuple = pg_class_tuple;
else
{
- /* ------------------
+ /*
* a satanic bug used to live here: pg_class_tuple used to be
* returned here without having the corresponding buffer pinned.
* so when the buffer gets replaced, all hell breaks loose.
* this bug is discovered and killed by wei on 9/27/91.
- * -------------------
+ *
*/
return_tuple = heap_copytuple(pg_class_tuple);
}
return return_tuple;
}
-/* ----------------
+/*
* AllocateRelationDesc
*
* This is used to allocate memory for a new relation descriptor
* 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 RelationClearRelation).
- * ----------------
+ *
*/
static Relation
AllocateRelationDesc(Relation relation, Form_pg_class relp)
/* Relcache entries must live in CacheMemoryContext */
oldcxt = MemoryContextSwitchTo(CacheMemoryContext);
- /* ----------------
+ /*
* allocate space for new relation descriptor, if needed
- * ----------------
+ *
*/
if (relation == NULL)
relation = (Relation) palloc(sizeof(RelationData));
- /* ----------------
+ /*
* clear all fields of reldesc
- * ----------------
+ *
*/
MemSet((char *) relation, 0, sizeof(RelationData));
/* make sure relation is marked as having no open file yet */
relation->rd_fd = -1;
- /* ----------------
+ /*
* Copy the relation tuple form
*
* We only allocate space for the fixed fields, ie, CLASS_TUPLE_SIZE.
* wouldn't know if the value is valid ... bottom line is that relacl
* *cannot* be retrieved from the relcache. Get it from the syscache
* if you need it.
- * ----------------
+ *
*/
relationForm = (Form_pg_class) palloc(CLASS_TUPLE_SIZE);
return relation;
}
-/* --------------------------------
+/*
* RelationBuildTupleDesc
*
* Form the relation's tuple descriptor from information in
* the pg_attribute, pg_attrdef & pg_relcheck system cataloges.
- * --------------------------------
+ *
*/
static void
RelationBuildTupleDesc(RelationBuildDescInfo buildinfo,
sizeof(TupleConstr));
constr->has_not_null = false;
- /* ----------------
+ /*
* form a scan key
- * ----------------
+ *
*/
ScanKeyEntryInitialize(&key, 0,
Anum_pg_attribute_attrelid,
F_OIDEQ,
ObjectIdGetDatum(RelationGetRelid(relation)));
- /* ----------------
+ /*
* open pg_attribute and begin a scan
- * ----------------
+ *
*/
pg_attribute_desc = heap_openr(AttributeRelationName, AccessShareLock);
pg_attribute_scan = heap_beginscan(pg_attribute_desc, 0, SnapshotNow, 1, &key);
- /* ----------------
+ /*
* add attribute data to relation->rd_att
- * ----------------
+ *
*/
need = relation->rd_rel->relnatts;
elog(ERROR, "catalog is missing %d attribute%s for relid %u",
need, (need == 1 ? "" : "s"), RelationGetRelid(relation));
- /* ----------------
+ /*
* end the scan and close the attribute relation
- * ----------------
+ *
*/
heap_endscan(pg_attribute_scan);
heap_close(pg_attribute_desc, AccessShareLock);
- /* ----------------
+ /*
* The attcacheoff values we read from pg_attribute should all be -1
* ("unknown"). Verify this if assert checking is on. They will be
* computed when and if needed during tuple access.
- * ----------------
+ *
*/
#ifdef USE_ASSERT_CHECKING
{
}
#endif
- /* ----------------
+ /*
* However, we can easily set the attcacheoff value for the first
* attribute: it must be zero. This eliminates the need for special
* cases for attnum=1 that used to exist in fastgetattr() and
* index_getattr().
- * ----------------
+ *
*/
relation->rd_att->attrs[0]->attcacheoff = 0;
heap_close(attrel, AccessShareLock);
- /* ----------------
+ /*
* The attcacheoff values we read from pg_attribute should all be -1
* ("unknown"). Verify this if assert checking is on. They will be
* computed when and if needed during tuple access.
- * ----------------
+ *
*/
#ifdef USE_ASSERT_CHECKING
for (i = 0; i < relation->rd_rel->relnatts; i++)
}
#endif
- /* ----------------
+ /*
* However, we can easily set the attcacheoff value for the first
* attribute: it must be zero. This eliminates the need for special
* cases for attnum=1 that used to exist in fastgetattr() and
* index_getattr().
- * ----------------
+ *
*/
relation->rd_att->attrs[0]->attcacheoff = 0;
SetConstrOfRelation(relation, constr, ndef, attrdef);
}
-/* --------------------------------
+/*
* RelationBuildRuleLock
*
* Form the relation's rewrite rules from information in
* entry, because that keeps the update logic in RelationClearRelation()
* manageable. The other subsidiary data structures are simple enough
* to be easy to free explicitly, anyway.
- * --------------------------------
+ *
*/
static void
RelationBuildRuleLock(Relation relation)
1024); /* maxsize */
relation->rd_rulescxt = rulescxt;
- /* ----------------
+ /*
* form an array to hold the rewrite rules (the array is extended if
* necessary)
- * ----------------
+ *
*/
maxlocks = 4;
rules = (RewriteRule **)
MemoryContextAlloc(rulescxt, sizeof(RewriteRule *) * maxlocks);
numlocks = 0;
- /* ----------------
+ /*
* form a scan key
- * ----------------
+ *
*/
ScanKeyEntryInitialize(&key, 0,
Anum_pg_rewrite_ev_class,
F_OIDEQ,
ObjectIdGetDatum(RelationGetRelid(relation)));
- /* ----------------
+ /*
* open pg_rewrite and begin a scan
- * ----------------
+ *
*/
pg_rewrite_desc = heap_openr(RewriteRelationName, AccessShareLock);
pg_rewrite_scan = heap_beginscan(pg_rewrite_desc, 0, SnapshotNow, 1, &key);
rules[numlocks++] = rule;
}
- /* ----------------
+ /*
* end the scan and close the attribute relation
- * ----------------
+ *
*/
heap_endscan(pg_rewrite_scan);
heap_close(pg_rewrite_desc, AccessShareLock);
- /* ----------------
+ /*
* form a RuleLock and insert into relation
- * ----------------
+ *
*/
rulelock = (RuleLock *) MemoryContextAlloc(rulescxt, sizeof(RuleLock));
rulelock->numLocks = numlocks;
relation->rd_rules = rulelock;
}
-/* --------------------------------
+/*
* equalRuleLocks
*
* Determine whether two RuleLocks are equivalent
*
* Probably this should be in the rules code someplace...
- * --------------------------------
+ *
*/
static bool
equalRuleLocks(RuleLock *rlock1, RuleLock *rlock2)
}
-/* --------------------------------
+/* ----------------------------------
* RelationBuildDesc
*
* Build a relation descriptor --- either a new one, or by
Form_pg_class relp;
MemoryContext oldcxt;
- /* ----------------
+ /*
* find the tuple in pg_class corresponding to the given relation id
- * ----------------
+ *
*/
pg_class_tuple = ScanPgRelation(buildinfo);
- /* ----------------
+ /*
* if no such tuple exists, return NULL
- * ----------------
+ *
*/
if (!HeapTupleIsValid(pg_class_tuple))
return NULL;
- /* ----------------
+ /*
* get information from the pg_class_tuple
- * ----------------
+ *
*/
relid = pg_class_tuple->t_data->t_oid;
relp = (Form_pg_class) GETSTRUCT(pg_class_tuple);
- /* ----------------
+ /*
* allocate storage for the relation descriptor,
* and copy pg_class_tuple to relation->rd_rel.
- * ----------------
+ *
*/
relation = AllocateRelationDesc(oldrelation, relp);
- /* -------------------
+ /*
* now we can free the memory allocated for pg_class_tuple
- * -------------------
+ *
*/
heap_freetuple(pg_class_tuple);
- /* ----------------
+ /*
* initialize the relation's relation id (relation->rd_id)
- * ----------------
+ *
*/
RelationGetRelid(relation) = relid;
- /* ----------------
+ /*
* initialize relation->rd_refcnt
- * ----------------
+ *
*/
RelationSetReferenceCount(relation, 1);
- /* ----------------
+ /*
* normal relations are not nailed into the cache
- * ----------------
+ *
*/
relation->rd_isnailed = false;
- /* ----------------
+ /*
* initialize the access method information (relation->rd_am)
- * ----------------
+ *
*/
relam = relation->rd_rel->relam;
if (OidIsValid(relam))
relation->rd_am = AccessMethodObjectIdGetForm(relam,
CacheMemoryContext);
- /* ----------------
+ /*
* initialize the tuple descriptor (relation->rd_att).
- * ----------------
+ *
*/
RelationBuildTupleDesc(buildinfo, relation);
- /* ----------------
+ /*
* Fetch rules and triggers that affect this relation
- * ----------------
+ *
*/
if (relation->rd_rel->relhasrules)
RelationBuildRuleLock(relation);
else
relation->trigdesc = NULL;
- /* ----------------
+ /*
* initialize index strategy and support information for this relation
- * ----------------
+ *
*/
if (OidIsValid(relam))
IndexedAccessMethodInitialize(relation);
- /* ----------------
+ /*
* initialize the relation lock manager information
- * ----------------
+ *
*/
RelationInitLockInfo(relation); /* see lmgr.c */
relation->rd_node.tblNode = MyDatabaseId;
relation->rd_node.relNode = relation->rd_rel->relfilenode;
- /* ----------------
+ /*
* open the relation and assign the file descriptor returned
* by the storage manager code to rd_fd.
- * ----------------
+ *
*/
if (relation->rd_rel->relkind != RELKIND_VIEW)
relation->rd_fd = smgropen(DEFAULT_SMGR, relation, false);
else
relation->rd_fd = -1;
- /* ----------------
+ /*
* insert newly created relation into proper relcaches,
* restore memory context and return the new reldesc.
- * ----------------
+ *
*/
oldcxt = MemoryContextSwitchTo(CacheMemoryContext);
RelationCacheInsert(relation);
RelationSetIndexSupport(relation, strategy, support);
}
-/* --------------------------------
+/*
* formrdesc
*
* This is a special cut-down version of RelationBuildDesc()
* catalogs...
*
* NOTE: we assume we are already switched into CacheMemoryContext.
- * --------------------------------
+ *
*/
static void
formrdesc(char *relationName,
Relation relation;
int i;
- /* ----------------
+ /*
* allocate new relation desc
- * ----------------
+ *
*/
relation = (Relation) palloc(sizeof(RelationData));
MemSet((char *) relation, 0, sizeof(RelationData));
- /* ----------------
+ /*
* don't open the unix file yet..
- * ----------------
+ *
*/
relation->rd_fd = -1;
- /* ----------------
+ /*
* initialize reference count
- * ----------------
+ *
*/
RelationSetReferenceCount(relation, 1);
- /* ----------------
+ /*
* all entries built with this routine are nailed-in-cache
- * ----------------
+ *
*/
relation->rd_isnailed = true;
- /* ----------------
+ /*
* initialize relation tuple form
*
* The data we insert here is pretty incomplete/bogus, but it'll
* serve to get us launched. RelationCacheInitializePhase2() will
* read the real data from pg_class and replace what we've done here.
- * ----------------
+ *
*/
relation->rd_rel = (Form_pg_class) palloc(CLASS_TUPLE_SIZE);
MemSet(relation->rd_rel, 0, CLASS_TUPLE_SIZE);
relation->rd_rel->relkind = RELKIND_RELATION;
relation->rd_rel->relnatts = (int16) natts;
- /* ----------------
+ /*
* initialize attribute tuple form
- * ----------------
+ *
*/
relation->rd_att = CreateTemplateTupleDesc(natts);
- /* ----------------
+ /*
* initialize tuple desc info
- * ----------------
+ *
*/
for (i = 0; i < natts; i++)
{
ATTRIBUTE_TUPLE_SIZE);
}
- /* ----------------
+ /*
* initialize relation id
- * ----------------
+ *
*/
RelationGetRelid(relation) = relation->rd_att->attrs[0]->attrelid;
- /* ----------------
+ /*
* initialize the relation's lock manager and RelFileNode information
- * ----------------
+ *
*/
RelationInitLockInfo(relation); /* see lmgr.c */
relation->rd_node.relNode =
relation->rd_rel->relfilenode = RelationGetRelid(relation);
- /* ----------------
+ /*
* initialize the rel-has-index flag, using hardwired knowledge
- * ----------------
+ *
*/
relation->rd_rel->relhasindex = false;
}
}
- /* ----------------
+ /*
* add new reldesc to relcache
- * ----------------
+ *
*/
RelationCacheInsert(relation);
}
-/* --------------------------------
+/*
* fixrdesc
*
* Update the phony data inserted by formrdesc() with real info
* from pg_class.
- * --------------------------------
+ *
*/
static void
fixrdesc(char *relationName)
Form_pg_class relp;
Relation relation;
- /* ----------------
+ /*
* find the tuple in pg_class corresponding to the given relation name
- * ----------------
+ *
*/
buildinfo.infotype = INFO_RELNAME;
buildinfo.i.info_name = relationName;
relationName);
relp = (Form_pg_class) GETSTRUCT(pg_class_tuple);
- /* ----------------
+ /*
* find the pre-made relcache entry (better be there!)
- * ----------------
+ *
*/
relation = RelationNameCacheGetRelation(relationName);
if (!RelationIsValid(relation))
elog(FATAL, "fixrdesc: no existing relcache entry for %s",
relationName);
- /* ----------------
+ /*
* and copy pg_class_tuple to relation->rd_rel.
* (See notes in AllocateRelationDesc())
- * ----------------
+ *
*/
Assert(relation->rd_rel != NULL);
memcpy((char *) relation->rd_rel, (char *) relp, CLASS_TUPLE_SIZE);
* ----------------------------------------------------------------
*/
-/* --------------------------------
+/*
* RelationIdCacheGetRelation
*
* Lookup an existing reldesc by OID.
* NB: relation ref count is incremented if successful.
* Caller should eventually decrement count. (Usually,
* that happens by calling RelationClose().)
- * --------------------------------
+ *
*/
Relation
RelationIdCacheGetRelation(Oid relationId)
return rd;
}
-/* --------------------------------
+/*
* RelationNameCacheGetRelation
*
* As above, but lookup by name.
- * --------------------------------
+ *
*/
static Relation
RelationNameCacheGetRelation(const char *relationName)
return rd;
}
-/* --------------------------------
+/*
* RelationIdGetRelation
*
* Lookup a reldesc by OID; make one if not already in cache.
* NB: relation ref count is incremented, or set to 1 if new entry.
* Caller should eventually decrement count. (Usually,
* that happens by calling RelationClose().)
- * --------------------------------
+ *
*/
Relation
RelationIdGetRelation(Oid relationId)
Relation rd;
RelationBuildDescInfo buildinfo;
- /* ----------------
+ /*
* increment access statistics
- * ----------------
+ *
*/
IncrHeapAccessStat(local_RelationIdGetRelation);
IncrHeapAccessStat(global_RelationIdGetRelation);
- /* ----------------
+ /*
* first try and get a reldesc from the cache
- * ----------------
+ *
*/
rd = RelationIdCacheGetRelation(relationId);
if (RelationIsValid(rd))
return rd;
- /* ----------------
+ /*
* no reldesc in the cache, so have RelationBuildDesc()
* build one and add it.
- * ----------------
+ *
*/
buildinfo.infotype = INFO_RELID;
buildinfo.i.info_id = relationId;
return rd;
}
-/* --------------------------------
+/*
* RelationNameGetRelation
*
* As above, but lookup by name.
- * --------------------------------
+ *
*/
Relation
RelationNameGetRelation(const char *relationName)
Relation rd;
RelationBuildDescInfo buildinfo;
- /* ----------------
+ /*
* increment access statistics
- * ----------------
+ *
*/
IncrHeapAccessStat(local_RelationNameGetRelation);
IncrHeapAccessStat(global_RelationNameGetRelation);
- /* ----------------
+ /*
* if caller is looking for a temp relation, substitute its real name;
* we only index temp rels by their real names.
- * ----------------
+ *
*/
temprelname = get_temp_rel_by_username(relationName);
if (temprelname != NULL)
relationName = temprelname;
- /* ----------------
+ /*
* first try and get a reldesc from the cache
- * ----------------
+ *
*/
rd = RelationNameCacheGetRelation(relationName);
if (RelationIsValid(rd))
return rd;
- /* ----------------
+ /*
* no reldesc in the cache, so have RelationBuildDesc()
* build one and add it.
- * ----------------
+ *
*/
buildinfo.infotype = INFO_RELNAME;
buildinfo.i.info_name = (char *) relationName;
* ----------------------------------------------------------------
*/
-/* --------------------------------
+/*
* RelationClose - close an open relation
*
* Actually, we just decrement the refcount.
* with aset.c's CLOBBER_FREED_MEMORY option, this provides a good test
* to catch references to already-released relcache entries. It slows
* things down quite a bit, however.
- * --------------------------------
+ *
*/
void
RelationClose(Relation relation)
}
#ifdef ENABLE_REINDEX_NAILED_RELATIONS
-/* --------------------------------
+/*
* RelationReloadClassinfo
*
* This function is especially for nailed relations.
* relhasindex/relfilenode could be changed even for
* nailed relations.
- * --------------------------------
+ *
*/
static void
RelationReloadClassinfo(Relation relation)
}
#endif /* ENABLE_REINDEX_NAILED_RELATIONS */
-/* --------------------------------
+/*
* RelationClearRelation
*
* Physically blow away a relation cache entry, or reset it and rebuild
* usually used when we are notified of a change to an open relation
* (one with refcount > 0). However, this routine just does whichever
* it's told to do; callers must determine which they want.
- * --------------------------------
+ *
*/
static void
RelationClearRelation(Relation relation, bool rebuildIt)
}
}
-/* --------------------------------
+/*
* RelationFlushRelation
*
* Rebuild the relation if it is open (refcount > 0), else blow it away.
- * --------------------------------
+ *
*/
static void
RelationFlushRelation(Relation relation)
RelationClearRelation(relation, rebuildIt);
}
-/* --------------------------------
+/*
* RelationForgetRelation -
*
* RelationClearRelation + if the relation is myxactonly then
* remove the relation descriptor from the newly created
* relation list.
- * --------------------------------
+ *
*/
void
RelationForgetRelation(Oid rid)
}
}
-/* --------------------------------
+/*
* RelationIdInvalidateRelationCacheByRelationId
*
* This routine is invoked for SI cache flush messages.
* safer to process them, so that our *own* SI update messages will
* have the same effects during CommandCounterIncrement for both
* local and nonlocal relations.
- * --------------------------------
+ *
*/
void
RelationIdInvalidateRelationCacheByRelationId(Oid relationId)
RelationSetReferenceCount(relation, 0);
}
-/* --------------------------------
+/*
* RelationRegisterRelation -
* register the Relation descriptor of a newly created relation
* with the relation descriptor Cache.
- * --------------------------------
+ *
*/
void
RelationRegisterRelation(Relation relation)
}
}
-/* --------------------------------
+/*
* RelationCacheInitialize
*
* This initializes the relation descriptor cache.
- * --------------------------------
+ *
*/
#define INITRELCACHESIZE 400
MemoryContext oldcxt;
HASHCTL ctl;
- /* ----------------
+ /*
* switch to cache memory context
- * ----------------
+ *
*/
if (!CacheMemoryContext)
CreateCacheMemoryContext();
oldcxt = MemoryContextSwitchTo(CacheMemoryContext);
- /* ----------------
+ /*
* create global caches
- * ----------------
+ *
*/
MemSet(&ctl, 0, (int) sizeof(ctl));
ctl.keysize = sizeof(NameData);
RelationNodeCache = hash_create(INITRELCACHESIZE, &ctl,
HASH_ELEM | HASH_FUNCTION);
- /* ----------------
+ /*
* initialize the cache with pre-made relation descriptors
* for some of the more important system relations. These
* relations should always be in the cache.
*
* NB: see also the list in RelationCacheInitializePhase2().
- * ----------------
+ *
*/
formrdesc(RelationRelationName, Natts_pg_class, Desc_pg_class);
formrdesc(AttributeRelationName, Natts_pg_attribute, Desc_pg_attribute);
MemoryContextSwitchTo(oldcxt);
}
-/* --------------------------------
+/*
* RelationCacheInitializePhase2
*
* This completes initialization of the relcache after catcache
* is functional and we are able to actually load data from pg_class.
- * --------------------------------
+ *
*/
void
RelationCacheInitializePhase2(void)
*
*
* IDENTIFICATION
- * $Header: /cvsroot/pgsql/src/backend/utils/cache/syscache.c,v 1.58 2001/01/24 19:43:15 momjian Exp $
+ * $Header: /cvsroot/pgsql/src/backend/utils/cache/syscache.c,v 1.59 2001/02/22 18:39:20 momjian Exp $
*
* NOTES
* These routines allow the parser/planner/executor to perform
---------------------------------------------------------------------------
*/
-/* ----------------
+/*
* struct cachedesc: information defining a single syscache
- * ----------------
+ *
*/
struct cachedesc
{
* Portions Copyright (c) 1996-2001, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California
*
- * $Id: shmem.h,v 1.26 2001/01/24 19:43:28 momjian Exp $
+ * $Id: shmem.h,v 1.27 2001/02/22 18:39:20 momjian Exp $
*
*-------------------------------------------------------------------------
*/
*/
typedef unsigned long SHMEM_OFFSET;
-#define INVALID_OFFSET (-1)
-#define BAD_LOCATION (-1)
+#define INVALID_OFFSET (-1)
+#define BAD_LOCATION (-1)
/*
* Start of the primary shared memory region, in this process' address space.