Commit 8b9bc234 authored by Tom Lane's avatar Tom Lane

Remove the limit on the number of entries allowed in catcaches, and

remove the infrastructure needed to enforce the limit, ie, the global
LRU list of cache entries.  On small-to-middling databases this wins
because maintaining the LRU list is a waste of time.  On large databases
this wins because it's better to keep more cache entries (we assume
such users can afford to use some more per-backend memory than was
contemplated in the Berkeley-era catcache design).  This provides a
noticeable improvement in the speed of psql \d on a 10000-table
database, though it doesn't make it instantaneous.

While at it, use per-catcache settings for the number of hash buckets
per catcache, rather than the former one-size-fits-all value.  It's a
bit silly to be using the same number of hash buckets for, eg, pg_am
and pg_attribute.  The specific values I used might need some tuning,
but they seem to be in the right ballpark based on CATCACHE_STATS
results from the standard regression tests.
parent e1e133f2
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/utils/cache/catcache.c,v 1.128 2006/03/05 15:58:45 momjian Exp $ * $PostgreSQL: pgsql/src/backend/utils/cache/catcache.c,v 1.129 2006/06/15 02:08:09 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -37,20 +37,6 @@ ...@@ -37,20 +37,6 @@
/* #define CACHEDEBUG */ /* turns DEBUG elogs on */ /* #define CACHEDEBUG */ /* turns DEBUG elogs on */
/*
* Constants related to size of the catcache.
*
* NCCBUCKETS must be a power of two and must be less than 64K (because
* SharedInvalCatcacheMsg crams hash indexes into a uint16 field). In
* practice it should be a lot less, anyway, to avoid chewing up too much
* space on hash bucket headers.
*
* MAXCCTUPLES could be as small as a few hundred, if per-backend memory
* consumption is at a premium.
*/
#define NCCBUCKETS 256 /* Hash buckets per CatCache */
#define MAXCCTUPLES 5000 /* Maximum # of tuples in all caches */
/* /*
* Given a hash value and the size of the hash table, find the bucket * Given a hash value and the size of the hash table, find the bucket
* in which the hash value belongs. Since the hash table must contain * in which the hash value belongs. Since the hash table must contain
...@@ -89,7 +75,7 @@ static uint32 CatalogCacheComputeTupleHashValue(CatCache *cache, ...@@ -89,7 +75,7 @@ static uint32 CatalogCacheComputeTupleHashValue(CatCache *cache,
HeapTuple tuple); HeapTuple tuple);
#ifdef CATCACHE_STATS #ifdef CATCACHE_STATS
static void CatCachePrintStats(void); static void CatCachePrintStats(int code, Datum arg);
#endif #endif
static void CatCacheRemoveCTup(CatCache *cache, CatCTup *ct); static void CatCacheRemoveCTup(CatCache *cache, CatCTup *ct);
static void CatCacheRemoveCList(CatCache *cache, CatCList *cl); static void CatCacheRemoveCList(CatCache *cache, CatCList *cl);
...@@ -97,7 +83,6 @@ static void CatalogCacheInitializeCache(CatCache *cache); ...@@ -97,7 +83,6 @@ static void CatalogCacheInitializeCache(CatCache *cache);
static CatCTup *CatalogCacheCreateEntry(CatCache *cache, HeapTuple ntp, static CatCTup *CatalogCacheCreateEntry(CatCache *cache, HeapTuple ntp,
uint32 hashValue, Index hashIndex, uint32 hashValue, Index hashIndex,
bool negative); bool negative);
static void CatalogCacheCleanup(CatCTup *savect);
static HeapTuple build_dummy_tuple(CatCache *cache, int nkeys, ScanKey skeys); static HeapTuple build_dummy_tuple(CatCache *cache, int nkeys, ScanKey skeys);
...@@ -281,7 +266,7 @@ CatalogCacheComputeTupleHashValue(CatCache *cache, HeapTuple tuple) ...@@ -281,7 +266,7 @@ CatalogCacheComputeTupleHashValue(CatCache *cache, HeapTuple tuple)
#ifdef CATCACHE_STATS #ifdef CATCACHE_STATS
static void static void
CatCachePrintStats(void) CatCachePrintStats(int code, Datum arg)
{ {
CatCache *cache; CatCache *cache;
long cc_searches = 0; long cc_searches = 0;
...@@ -289,18 +274,14 @@ CatCachePrintStats(void) ...@@ -289,18 +274,14 @@ CatCachePrintStats(void)
long cc_neg_hits = 0; long cc_neg_hits = 0;
long cc_newloads = 0; long cc_newloads = 0;
long cc_invals = 0; long cc_invals = 0;
long cc_discards = 0;
long cc_lsearches = 0; long cc_lsearches = 0;
long cc_lhits = 0; long cc_lhits = 0;
elog(DEBUG2, "catcache stats dump: %d/%d tuples in catcaches",
CacheHdr->ch_ntup, CacheHdr->ch_maxtup);
for (cache = CacheHdr->ch_caches; cache; cache = cache->cc_next) for (cache = CacheHdr->ch_caches; cache; cache = cache->cc_next)
{ {
if (cache->cc_ntup == 0 && cache->cc_searches == 0) if (cache->cc_ntup == 0 && cache->cc_searches == 0)
continue; /* don't print unused caches */ continue; /* don't print unused caches */
elog(DEBUG2, "catcache %s/%u: %d tup, %ld srch, %ld+%ld=%ld hits, %ld+%ld=%ld loads, %ld invals, %ld discards, %ld lsrch, %ld lhits", elog(DEBUG2, "catcache %s/%u: %d tup, %ld srch, %ld+%ld=%ld hits, %ld+%ld=%ld loads, %ld invals, %ld lsrch, %ld lhits",
cache->cc_relname, cache->cc_relname,
cache->cc_indexoid, cache->cc_indexoid,
cache->cc_ntup, cache->cc_ntup,
...@@ -312,7 +293,6 @@ CatCachePrintStats(void) ...@@ -312,7 +293,6 @@ CatCachePrintStats(void)
cache->cc_searches - cache->cc_hits - cache->cc_neg_hits - cache->cc_newloads, cache->cc_searches - cache->cc_hits - cache->cc_neg_hits - cache->cc_newloads,
cache->cc_searches - cache->cc_hits - cache->cc_neg_hits, cache->cc_searches - cache->cc_hits - cache->cc_neg_hits,
cache->cc_invals, cache->cc_invals,
cache->cc_discards,
cache->cc_lsearches, cache->cc_lsearches,
cache->cc_lhits); cache->cc_lhits);
cc_searches += cache->cc_searches; cc_searches += cache->cc_searches;
...@@ -320,11 +300,10 @@ CatCachePrintStats(void) ...@@ -320,11 +300,10 @@ CatCachePrintStats(void)
cc_neg_hits += cache->cc_neg_hits; cc_neg_hits += cache->cc_neg_hits;
cc_newloads += cache->cc_newloads; cc_newloads += cache->cc_newloads;
cc_invals += cache->cc_invals; cc_invals += cache->cc_invals;
cc_discards += cache->cc_discards;
cc_lsearches += cache->cc_lsearches; cc_lsearches += cache->cc_lsearches;
cc_lhits += cache->cc_lhits; cc_lhits += cache->cc_lhits;
} }
elog(DEBUG2, "catcache totals: %d tup, %ld srch, %ld+%ld=%ld hits, %ld+%ld=%ld loads, %ld invals, %ld discards, %ld lsrch, %ld lhits", elog(DEBUG2, "catcache totals: %d tup, %ld srch, %ld+%ld=%ld hits, %ld+%ld=%ld loads, %ld invals, %ld lsrch, %ld lhits",
CacheHdr->ch_ntup, CacheHdr->ch_ntup,
cc_searches, cc_searches,
cc_hits, cc_hits,
...@@ -334,7 +313,6 @@ CatCachePrintStats(void) ...@@ -334,7 +313,6 @@ CatCachePrintStats(void)
cc_searches - cc_hits - cc_neg_hits - cc_newloads, cc_searches - cc_hits - cc_neg_hits - cc_newloads,
cc_searches - cc_hits - cc_neg_hits, cc_searches - cc_hits - cc_neg_hits,
cc_invals, cc_invals,
cc_discards,
cc_lsearches, cc_lsearches,
cc_lhits); cc_lhits);
} }
...@@ -367,8 +345,7 @@ CatCacheRemoveCTup(CatCache *cache, CatCTup *ct) ...@@ -367,8 +345,7 @@ CatCacheRemoveCTup(CatCache *cache, CatCTup *ct)
return; /* nothing left to do */ return; /* nothing left to do */
} }
/* delink from linked lists */ /* delink from linked list */
DLRemove(&ct->lrulist_elem);
DLRemove(&ct->cache_elem); DLRemove(&ct->cache_elem);
/* free associated tuple data */ /* free associated tuple data */
...@@ -568,11 +545,13 @@ AtEOXact_CatCache(bool isCommit) ...@@ -568,11 +545,13 @@ AtEOXact_CatCache(bool isCommit)
if (assert_enabled) if (assert_enabled)
{ {
CatCache *ccp; CatCache *ccp;
Dlelem *elt;
/* Check CatCLists */
for (ccp = CacheHdr->ch_caches; ccp; ccp = ccp->cc_next) for (ccp = CacheHdr->ch_caches; ccp; ccp = ccp->cc_next)
{ {
Dlelem *elt;
int i;
/* Check CatCLists */
for (elt = DLGetHead(&ccp->cc_lists); elt; elt = DLGetSucc(elt)) for (elt = DLGetHead(&ccp->cc_lists); elt; elt = DLGetSucc(elt))
{ {
CatCList *cl = (CatCList *) DLE_VAL(elt); CatCList *cl = (CatCList *) DLE_VAL(elt);
...@@ -581,10 +560,13 @@ AtEOXact_CatCache(bool isCommit) ...@@ -581,10 +560,13 @@ AtEOXact_CatCache(bool isCommit)
Assert(cl->refcount == 0); Assert(cl->refcount == 0);
Assert(!cl->dead); Assert(!cl->dead);
} }
}
/* Check individual tuples */ /* Check individual tuples */
for (elt = DLGetHead(&CacheHdr->ch_lrulist); elt; elt = DLGetSucc(elt)) for (i = 0; i < ccp->cc_nbuckets; i++)
{
for (elt = DLGetHead(&ccp->cc_bucket[i]);
elt;
elt = DLGetSucc(elt))
{ {
CatCTup *ct = (CatCTup *) DLE_VAL(elt); CatCTup *ct = (CatCTup *) DLE_VAL(elt);
...@@ -593,6 +575,8 @@ AtEOXact_CatCache(bool isCommit) ...@@ -593,6 +575,8 @@ AtEOXact_CatCache(bool isCommit)
Assert(!ct->dead); Assert(!ct->dead);
} }
} }
}
}
#endif #endif
} }
...@@ -796,12 +780,27 @@ InitCatCache(int id, ...@@ -796,12 +780,27 @@ InitCatCache(int id,
Oid indexoid, Oid indexoid,
int reloidattr, int reloidattr,
int nkeys, int nkeys,
const int *key) const int *key,
int nbuckets)
{ {
CatCache *cp; CatCache *cp;
MemoryContext oldcxt; MemoryContext oldcxt;
int i; int i;
/*
* nbuckets is the number of hash buckets to use in this catcache.
* Currently we just use a hard-wired estimate of an appropriate size
* for each cache; maybe later make them dynamically resizable?
*
* nbuckets must be a power of two. We check this via Assert rather than
* a full runtime check because the values will be coming from constant
* tables.
*
* If you're confused by the power-of-two check, see comments in
* bitmapset.c for an explanation.
*/
Assert(nbuckets > 0 && (nbuckets & -nbuckets) == nbuckets);
/* /*
* first switch to the cache context so our allocations do not vanish at * first switch to the cache context so our allocations do not vanish at
* the end of a transaction * the end of a transaction
...@@ -812,17 +811,15 @@ InitCatCache(int id, ...@@ -812,17 +811,15 @@ InitCatCache(int id,
oldcxt = MemoryContextSwitchTo(CacheMemoryContext); oldcxt = MemoryContextSwitchTo(CacheMemoryContext);
/* /*
* if first time through, initialize the cache group header, including * if first time through, initialize the cache group header
* global LRU list header
*/ */
if (CacheHdr == NULL) if (CacheHdr == NULL)
{ {
CacheHdr = (CatCacheHeader *) palloc(sizeof(CatCacheHeader)); CacheHdr = (CatCacheHeader *) palloc(sizeof(CatCacheHeader));
CacheHdr->ch_caches = NULL; CacheHdr->ch_caches = NULL;
CacheHdr->ch_ntup = 0; CacheHdr->ch_ntup = 0;
CacheHdr->ch_maxtup = MAXCCTUPLES;
DLInitList(&CacheHdr->ch_lrulist);
#ifdef CATCACHE_STATS #ifdef CATCACHE_STATS
/* set up to dump stats at backend exit */
on_proc_exit(CatCachePrintStats, 0); on_proc_exit(CatCachePrintStats, 0);
#endif #endif
} }
...@@ -832,7 +829,7 @@ InitCatCache(int id, ...@@ -832,7 +829,7 @@ InitCatCache(int id,
* *
* Note: we assume zeroing initializes the Dllist headers correctly * Note: we assume zeroing initializes the Dllist headers correctly
*/ */
cp = (CatCache *) palloc0(sizeof(CatCache) + NCCBUCKETS * sizeof(Dllist)); cp = (CatCache *) palloc0(sizeof(CatCache) + nbuckets * sizeof(Dllist));
/* /*
* initialize the cache's relation information for the relation * initialize the cache's relation information for the relation
...@@ -847,7 +844,7 @@ InitCatCache(int id, ...@@ -847,7 +844,7 @@ InitCatCache(int id,
cp->cc_tupdesc = (TupleDesc) NULL; cp->cc_tupdesc = (TupleDesc) NULL;
cp->cc_reloidattr = reloidattr; cp->cc_reloidattr = reloidattr;
cp->cc_ntup = 0; cp->cc_ntup = 0;
cp->cc_nbuckets = NCCBUCKETS; cp->cc_nbuckets = nbuckets;
cp->cc_nkeys = nkeys; cp->cc_nkeys = nkeys;
for (i = 0; i < nkeys; ++i) for (i = 0; i < nkeys; ++i)
cp->cc_key[i] = key[i]; cp->cc_key[i] = key[i];
...@@ -1162,13 +1159,11 @@ SearchCatCache(CatCache *cache, ...@@ -1162,13 +1159,11 @@ SearchCatCache(CatCache *cache,
continue; continue;
/* /*
* we found a match in the cache: move it to the front of the global * We found a match in the cache. Move it to the front of the list
* LRU list. We also move it to the front of the list for its * for its hashbucket, in order to speed subsequent searches. (The
* hashbucket, in order to speed subsequent searches. (The most * most frequently accessed elements in any hashbucket will tend to be
* frequently accessed elements in any hashbucket will tend to be near * near the front of the hashbucket's list.)
* the front of the hashbucket's list.)
*/ */
DLMoveToFront(&ct->lrulist_elem);
DLMoveToFront(&ct->cache_elem); DLMoveToFront(&ct->cache_elem);
/* /*
...@@ -1414,14 +1409,12 @@ SearchCatCacheList(CatCache *cache, ...@@ -1414,14 +1409,12 @@ SearchCatCacheList(CatCache *cache,
continue; continue;
/* /*
* We found a matching list: mark it as touched since the last * We found a matching list. Move the list to the front of the
* CatalogCacheCleanup() sweep. Also move the list to the front of * cache's list-of-lists, to speed subsequent searches. (We do not
* the cache's list-of-lists, to speed subsequent searches. (We do not
* move the members to the fronts of their hashbucket lists, however, * move the members to the fronts of their hashbucket lists, however,
* since there's no point in that unless they are searched for * since there's no point in that unless they are searched for
* individually.) * individually.)
*/ */
cl->touched = true;
DLMoveToFront(&cl->cache_elem); DLMoveToFront(&cl->cache_elem);
/* Bump the list's refcount and return it */ /* Bump the list's refcount and return it */
...@@ -1504,10 +1497,7 @@ SearchCatCacheList(CatCache *cache, ...@@ -1504,10 +1497,7 @@ SearchCatCacheList(CatCache *cache,
if (ct->c_list) if (ct->c_list)
continue; continue;
/* Found a match, so move it to front */ break; /* A-OK */
DLMoveToFront(&ct->lrulist_elem);
break;
} }
if (elt == NULL) if (elt == NULL)
...@@ -1577,7 +1567,6 @@ SearchCatCacheList(CatCache *cache, ...@@ -1577,7 +1567,6 @@ SearchCatCacheList(CatCache *cache,
cl->refcount = 0; /* for the moment */ cl->refcount = 0; /* for the moment */
cl->dead = false; cl->dead = false;
cl->ordered = ordered; cl->ordered = ordered;
cl->touched = false; /* we already moved members to front */
cl->nkeys = nkeys; cl->nkeys = nkeys;
cl->hash_value = lHashValue; cl->hash_value = lHashValue;
cl->n_members = nmembers; cl->n_members = nmembers;
...@@ -1654,11 +1643,10 @@ CatalogCacheCreateEntry(CatCache *cache, HeapTuple ntp, ...@@ -1654,11 +1643,10 @@ CatalogCacheCreateEntry(CatCache *cache, HeapTuple ntp,
/* /*
* Finish initializing the CatCTup header, and add it to the cache's * Finish initializing the CatCTup header, and add it to the cache's
* linked lists and counts. * linked list and counts.
*/ */
ct->ct_magic = CT_MAGIC; ct->ct_magic = CT_MAGIC;
ct->my_cache = cache; ct->my_cache = cache;
DLInitElem(&ct->lrulist_elem, (void *) ct);
DLInitElem(&ct->cache_elem, (void *) ct); DLInitElem(&ct->cache_elem, (void *) ct);
ct->c_list = NULL; ct->c_list = NULL;
ct->refcount = 0; /* for the moment */ ct->refcount = 0; /* for the moment */
...@@ -1666,97 +1654,14 @@ CatalogCacheCreateEntry(CatCache *cache, HeapTuple ntp, ...@@ -1666,97 +1654,14 @@ CatalogCacheCreateEntry(CatCache *cache, HeapTuple ntp,
ct->negative = negative; ct->negative = negative;
ct->hash_value = hashValue; ct->hash_value = hashValue;
DLAddHead(&CacheHdr->ch_lrulist, &ct->lrulist_elem);
DLAddHead(&cache->cc_bucket[hashIndex], &ct->cache_elem); DLAddHead(&cache->cc_bucket[hashIndex], &ct->cache_elem);
cache->cc_ntup++; cache->cc_ntup++;
CacheHdr->ch_ntup++; CacheHdr->ch_ntup++;
/*
* If we've exceeded the desired size of the caches, try to throw away the
* least recently used entry(s). NB: be careful not to throw away the
* newly-built entry...
*/
if (CacheHdr->ch_ntup > CacheHdr->ch_maxtup)
CatalogCacheCleanup(ct);
return ct; return ct;
} }
/*
* CatalogCacheCleanup
* Try to reduce the size of the catcaches when they get too big
*
* savect can be NULL, or a specific CatCTup not to remove even if it
* has zero refcount.
*/
static void
CatalogCacheCleanup(CatCTup *savect)
{
int tup_target;
CatCache *ccp;
Dlelem *elt,
*prevelt;
/*
* Each time we have to do this, try to cut the cache size down to about
* 90% of the maximum.
*/
tup_target = (CacheHdr->ch_maxtup * 9) / 10;
/*
* Our strategy for managing CatCLists is that, each time we have to throw
* away some cache entries, we first move-to-front all the members of
* CatCLists that have been touched since the last cleanup sweep. Then we
* do strict LRU elimination by individual tuples, zapping a list if any
* of its members gets zapped. Before PostgreSQL 8.1, we moved members to
* front each time their owning list was touched, which was arguably more
* fair in balancing list members against standalone tuples --- but the
* overhead for large lists was horrendous. This scheme is more heavily
* biased towards preserving lists, but that is not necessarily bad
* either.
*/
for (ccp = CacheHdr->ch_caches; ccp; ccp = ccp->cc_next)
{
for (elt = DLGetHead(&ccp->cc_lists); elt; elt = DLGetSucc(elt))
{
CatCList *cl = (CatCList *) DLE_VAL(elt);
Assert(cl->cl_magic == CL_MAGIC);
if (cl->touched && !cl->dead)
{
int i;
for (i = 0; i < cl->n_members; i++)
DLMoveToFront(&cl->members[i]->lrulist_elem);
}
cl->touched = false;
}
}
/* Now get rid of unreferenced tuples in reverse global LRU order */
for (elt = DLGetTail(&CacheHdr->ch_lrulist); elt; elt = prevelt)
{
CatCTup *ct = (CatCTup *) DLE_VAL(elt);
prevelt = DLGetPred(elt);
if (ct->refcount == 0 &&
(ct->c_list == NULL || ct->c_list->refcount == 0) &&
ct != savect)
{
#ifdef CATCACHE_STATS
ct->my_cache->cc_discards++;
#endif
CatCacheRemoveCTup(ct->my_cache, ct);
/* Quit when we've removed enough tuples */
if (CacheHdr->ch_ntup <= tup_target)
break;
}
}
}
/* /*
* build_dummy_tuple * build_dummy_tuple
* Generate a palloc'd HeapTuple that contains the specified key * Generate a palloc'd HeapTuple that contains the specified key
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/utils/cache/syscache.c,v 1.103 2006/05/03 22:45:26 tgl Exp $ * $PostgreSQL: pgsql/src/backend/utils/cache/syscache.c,v 1.104 2006/06/15 02:08:09 tgl Exp $
* *
* NOTES * NOTES
* These routines allow the parser/planner/executor to perform * These routines allow the parser/planner/executor to perform
...@@ -56,13 +56,17 @@ ...@@ -56,13 +56,17 @@
accordingly. accordingly.
Add your entry to the cacheinfo[] array below. All cache lists are Add your entry to the cacheinfo[] array below. All cache lists are
alphabetical, so add it in the proper place. Specify the relation alphabetical, so add it in the proper place. Specify the relation OID,
OID, index OID, number of keys, and key attribute numbers. If the index OID, number of keys, key attribute numbers, and number of hash
relation contains tuples that are associated with a particular relation buckets. If the relation contains tuples that are associated with a
(for example, its attributes, rules, triggers, etc) then specify the particular relation (for example, its attributes, rules, triggers, etc)
attribute number that contains the OID of the associated relation. then specify the attribute number that contains the OID of the associated
This is used by CatalogCacheFlushRelation() to remove the correct relation. This is used by CatalogCacheFlushRelation() to remove the
tuples during a table drop or relcache invalidation event. correct tuples during a table drop or relcache invalidation event.
The number of hash buckets must be a power of 2. It's reasonable to
set this to the number of entries that might be in the particular cache
in a medium-size database.
There must be a unique index underlying each syscache (ie, an index There must be a unique index underlying each syscache (ie, an index
whose key is the same as that of the cache). If there is not one whose key is the same as that of the cache). If there is not one
...@@ -90,6 +94,7 @@ struct cachedesc ...@@ -90,6 +94,7 @@ struct cachedesc
int reloidattr; /* attr number of rel OID reference, or 0 */ int reloidattr; /* attr number of rel OID reference, or 0 */
int nkeys; /* # of keys needed for cache lookup */ int nkeys; /* # of keys needed for cache lookup */
int key[4]; /* attribute numbers of key attrs */ int key[4]; /* attribute numbers of key attrs */
int nbuckets; /* number of hash buckets for this cache */
}; };
static const struct cachedesc cacheinfo[] = { static const struct cachedesc cacheinfo[] = {
...@@ -102,7 +107,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -102,7 +107,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
32
},
{AccessMethodRelationId, /* AMNAME */ {AccessMethodRelationId, /* AMNAME */
AmNameIndexId, AmNameIndexId,
0, 0,
...@@ -112,7 +119,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -112,7 +119,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
4
},
{AccessMethodRelationId, /* AMOID */ {AccessMethodRelationId, /* AMOID */
AmOidIndexId, AmOidIndexId,
0, 0,
...@@ -122,7 +131,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -122,7 +131,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
4
},
{AccessMethodOperatorRelationId, /* AMOPOPID */ {AccessMethodOperatorRelationId, /* AMOPOPID */
AccessMethodOperatorIndexId, AccessMethodOperatorIndexId,
0, 0,
...@@ -132,7 +143,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -132,7 +143,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_amop_amopclaid, Anum_pg_amop_amopclaid,
0, 0,
0 0
}}, },
64
},
{AccessMethodOperatorRelationId, /* AMOPSTRATEGY */ {AccessMethodOperatorRelationId, /* AMOPSTRATEGY */
AccessMethodStrategyIndexId, AccessMethodStrategyIndexId,
0, 0,
...@@ -142,7 +155,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -142,7 +155,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_amop_amopsubtype, Anum_pg_amop_amopsubtype,
Anum_pg_amop_amopstrategy, Anum_pg_amop_amopstrategy,
0 0
}}, },
64
},
{AccessMethodProcedureRelationId, /* AMPROCNUM */ {AccessMethodProcedureRelationId, /* AMPROCNUM */
AccessMethodProcedureIndexId, AccessMethodProcedureIndexId,
0, 0,
...@@ -152,7 +167,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -152,7 +167,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_amproc_amprocsubtype, Anum_pg_amproc_amprocsubtype,
Anum_pg_amproc_amprocnum, Anum_pg_amproc_amprocnum,
0 0
}}, },
64
},
{AttributeRelationId, /* ATTNAME */ {AttributeRelationId, /* ATTNAME */
AttributeRelidNameIndexId, AttributeRelidNameIndexId,
Anum_pg_attribute_attrelid, Anum_pg_attribute_attrelid,
...@@ -162,7 +179,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -162,7 +179,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_attribute_attname, Anum_pg_attribute_attname,
0, 0,
0 0
}}, },
2048
},
{AttributeRelationId, /* ATTNUM */ {AttributeRelationId, /* ATTNUM */
AttributeRelidNumIndexId, AttributeRelidNumIndexId,
Anum_pg_attribute_attrelid, Anum_pg_attribute_attrelid,
...@@ -172,7 +191,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -172,7 +191,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_attribute_attnum, Anum_pg_attribute_attnum,
0, 0,
0 0
}}, },
2048
},
{AuthMemRelationId, /* AUTHMEMMEMROLE */ {AuthMemRelationId, /* AUTHMEMMEMROLE */
AuthMemMemRoleIndexId, AuthMemMemRoleIndexId,
0, 0,
...@@ -182,7 +203,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -182,7 +203,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_auth_members_roleid, Anum_pg_auth_members_roleid,
0, 0,
0 0
}}, },
128
},
{AuthMemRelationId, /* AUTHMEMROLEMEM */ {AuthMemRelationId, /* AUTHMEMROLEMEM */
AuthMemRoleMemIndexId, AuthMemRoleMemIndexId,
0, 0,
...@@ -192,7 +215,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -192,7 +215,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_auth_members_member, Anum_pg_auth_members_member,
0, 0,
0 0
}}, },
128
},
{AuthIdRelationId, /* AUTHNAME */ {AuthIdRelationId, /* AUTHNAME */
AuthIdRolnameIndexId, AuthIdRolnameIndexId,
0, 0,
...@@ -202,7 +227,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -202,7 +227,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
128
},
{AuthIdRelationId, /* AUTHOID */ {AuthIdRelationId, /* AUTHOID */
AuthIdOidIndexId, AuthIdOidIndexId,
0, 0,
...@@ -212,7 +239,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -212,7 +239,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
128
},
{ {
CastRelationId, /* CASTSOURCETARGET */ CastRelationId, /* CASTSOURCETARGET */
CastSourceTargetIndexId, CastSourceTargetIndexId,
...@@ -223,7 +252,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -223,7 +252,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_cast_casttarget, Anum_pg_cast_casttarget,
0, 0,
0 0
}}, },
256
},
{OperatorClassRelationId, /* CLAAMNAMENSP */ {OperatorClassRelationId, /* CLAAMNAMENSP */
OpclassAmNameNspIndexId, OpclassAmNameNspIndexId,
0, 0,
...@@ -233,7 +264,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -233,7 +264,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_opclass_opcname, Anum_pg_opclass_opcname,
Anum_pg_opclass_opcnamespace, Anum_pg_opclass_opcnamespace,
0 0
}}, },
64
},
{OperatorClassRelationId, /* CLAOID */ {OperatorClassRelationId, /* CLAOID */
OpclassOidIndexId, OpclassOidIndexId,
0, 0,
...@@ -243,7 +276,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -243,7 +276,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
64
},
{ConversionRelationId, /* CONDEFAULT */ {ConversionRelationId, /* CONDEFAULT */
ConversionDefaultIndexId, ConversionDefaultIndexId,
0, 0,
...@@ -253,7 +288,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -253,7 +288,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_conversion_conforencoding, Anum_pg_conversion_conforencoding,
Anum_pg_conversion_contoencoding, Anum_pg_conversion_contoencoding,
ObjectIdAttributeNumber, ObjectIdAttributeNumber,
}}, },
128
},
{ConversionRelationId, /* CONNAMENSP */ {ConversionRelationId, /* CONNAMENSP */
ConversionNameNspIndexId, ConversionNameNspIndexId,
0, 0,
...@@ -263,7 +300,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -263,7 +300,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_conversion_connamespace, Anum_pg_conversion_connamespace,
0, 0,
0 0
}}, },
128
},
{ConversionRelationId, /* CONOID */ {ConversionRelationId, /* CONOID */
ConversionOidIndexId, ConversionOidIndexId,
0, 0,
...@@ -273,7 +312,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -273,7 +312,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
128
},
{DatabaseRelationId, /* DATABASEOID */ {DatabaseRelationId, /* DATABASEOID */
DatabaseOidIndexId, DatabaseOidIndexId,
0, 0,
...@@ -283,7 +324,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -283,7 +324,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
4
},
{IndexRelationId, /* INDEXRELID */ {IndexRelationId, /* INDEXRELID */
IndexRelidIndexId, IndexRelidIndexId,
Anum_pg_index_indrelid, Anum_pg_index_indrelid,
...@@ -293,7 +336,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -293,7 +336,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
1024
},
{InheritsRelationId, /* INHRELID */ {InheritsRelationId, /* INHRELID */
InheritsRelidSeqnoIndexId, InheritsRelidSeqnoIndexId,
Anum_pg_inherits_inhrelid, Anum_pg_inherits_inhrelid,
...@@ -303,7 +348,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -303,7 +348,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_inherits_inhseqno, Anum_pg_inherits_inhseqno,
0, 0,
0 0
}}, },
256
},
{LanguageRelationId, /* LANGNAME */ {LanguageRelationId, /* LANGNAME */
LanguageNameIndexId, LanguageNameIndexId,
0, 0,
...@@ -313,7 +360,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -313,7 +360,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
4
},
{LanguageRelationId, /* LANGOID */ {LanguageRelationId, /* LANGOID */
LanguageOidIndexId, LanguageOidIndexId,
0, 0,
...@@ -323,7 +372,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -323,7 +372,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
4
},
{NamespaceRelationId, /* NAMESPACENAME */ {NamespaceRelationId, /* NAMESPACENAME */
NamespaceNameIndexId, NamespaceNameIndexId,
0, 0,
...@@ -333,7 +384,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -333,7 +384,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
256
},
{NamespaceRelationId, /* NAMESPACEOID */ {NamespaceRelationId, /* NAMESPACEOID */
NamespaceOidIndexId, NamespaceOidIndexId,
0, 0,
...@@ -343,7 +396,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -343,7 +396,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
256
},
{OperatorRelationId, /* OPERNAMENSP */ {OperatorRelationId, /* OPERNAMENSP */
OperatorNameNspIndexId, OperatorNameNspIndexId,
0, 0,
...@@ -353,7 +408,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -353,7 +408,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_operator_oprleft, Anum_pg_operator_oprleft,
Anum_pg_operator_oprright, Anum_pg_operator_oprright,
Anum_pg_operator_oprnamespace Anum_pg_operator_oprnamespace
}}, },
1024
},
{OperatorRelationId, /* OPEROID */ {OperatorRelationId, /* OPEROID */
OperatorOidIndexId, OperatorOidIndexId,
0, 0,
...@@ -363,7 +420,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -363,7 +420,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
1024
},
{ProcedureRelationId, /* PROCNAMEARGSNSP */ {ProcedureRelationId, /* PROCNAMEARGSNSP */
ProcedureNameArgsNspIndexId, ProcedureNameArgsNspIndexId,
0, 0,
...@@ -373,7 +432,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -373,7 +432,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_proc_proargtypes, Anum_pg_proc_proargtypes,
Anum_pg_proc_pronamespace, Anum_pg_proc_pronamespace,
0 0
}}, },
2048
},
{ProcedureRelationId, /* PROCOID */ {ProcedureRelationId, /* PROCOID */
ProcedureOidIndexId, ProcedureOidIndexId,
0, 0,
...@@ -383,7 +444,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -383,7 +444,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
2048
},
{RelationRelationId, /* RELNAMENSP */ {RelationRelationId, /* RELNAMENSP */
ClassNameNspIndexId, ClassNameNspIndexId,
ObjectIdAttributeNumber, ObjectIdAttributeNumber,
...@@ -393,7 +456,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -393,7 +456,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_class_relnamespace, Anum_pg_class_relnamespace,
0, 0,
0 0
}}, },
1024
},
{RelationRelationId, /* RELOID */ {RelationRelationId, /* RELOID */
ClassOidIndexId, ClassOidIndexId,
ObjectIdAttributeNumber, ObjectIdAttributeNumber,
...@@ -403,7 +468,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -403,7 +468,9 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}}, },
1024
},
{RewriteRelationId, /* RULERELNAME */ {RewriteRelationId, /* RULERELNAME */
RewriteRelRulenameIndexId, RewriteRelRulenameIndexId,
Anum_pg_rewrite_ev_class, Anum_pg_rewrite_ev_class,
...@@ -413,7 +480,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -413,7 +480,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_rewrite_rulename, Anum_pg_rewrite_rulename,
0, 0,
0 0
}}, },
1024
},
{StatisticRelationId, /* STATRELATT */ {StatisticRelationId, /* STATRELATT */
StatisticRelidAttnumIndexId, StatisticRelidAttnumIndexId,
Anum_pg_statistic_starelid, Anum_pg_statistic_starelid,
...@@ -423,7 +492,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -423,7 +492,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_statistic_staattnum, Anum_pg_statistic_staattnum,
0, 0,
0 0
}}, },
1024
},
{TypeRelationId, /* TYPENAMENSP */ {TypeRelationId, /* TYPENAMENSP */
TypeNameNspIndexId, TypeNameNspIndexId,
Anum_pg_type_typrelid, Anum_pg_type_typrelid,
...@@ -433,7 +504,9 @@ static const struct cachedesc cacheinfo[] = { ...@@ -433,7 +504,9 @@ static const struct cachedesc cacheinfo[] = {
Anum_pg_type_typnamespace, Anum_pg_type_typnamespace,
0, 0,
0 0
}}, },
1024
},
{TypeRelationId, /* TYPEOID */ {TypeRelationId, /* TYPEOID */
TypeOidIndexId, TypeOidIndexId,
Anum_pg_type_typrelid, Anum_pg_type_typrelid,
...@@ -443,11 +516,12 @@ static const struct cachedesc cacheinfo[] = { ...@@ -443,11 +516,12 @@ static const struct cachedesc cacheinfo[] = {
0, 0,
0, 0,
0 0
}} },
1024
}
}; };
static CatCache *SysCache[ static CatCache *SysCache[lengthof(cacheinfo)];
lengthof(cacheinfo)];
static int SysCacheSize = lengthof(cacheinfo); static int SysCacheSize = lengthof(cacheinfo);
static bool CacheInitialized = false; static bool CacheInitialized = false;
...@@ -476,7 +550,8 @@ InitCatalogCache(void) ...@@ -476,7 +550,8 @@ InitCatalogCache(void)
cacheinfo[cacheId].indoid, cacheinfo[cacheId].indoid,
cacheinfo[cacheId].reloidattr, cacheinfo[cacheId].reloidattr,
cacheinfo[cacheId].nkeys, cacheinfo[cacheId].nkeys,
cacheinfo[cacheId].key); cacheinfo[cacheId].key,
cacheinfo[cacheId].nbuckets);
if (!PointerIsValid(SysCache[cacheId])) if (!PointerIsValid(SysCache[cacheId]))
elog(ERROR, "could not initialize cache %u (%d)", elog(ERROR, "could not initialize cache %u (%d)",
cacheinfo[cacheId].reloid, cacheId); cacheinfo[cacheId].reloid, cacheId);
......
...@@ -13,7 +13,7 @@ ...@@ -13,7 +13,7 @@
* Portions Copyright (c) 1996-2006, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2006, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/utils/catcache.h,v 1.58 2006/03/05 15:59:07 momjian Exp $ * $PostgreSQL: pgsql/src/include/utils/catcache.h,v 1.59 2006/06/15 02:08:09 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -60,7 +60,6 @@ typedef struct catcache ...@@ -60,7 +60,6 @@ typedef struct catcache
* searches, each of which will result in loading a negative entry * searches, each of which will result in loading a negative entry
*/ */
long cc_invals; /* # of entries invalidated from cache */ long cc_invals; /* # of entries invalidated from cache */
long cc_discards; /* # of entries discarded due to overflow */
long cc_lsearches; /* total # list-searches */ long cc_lsearches; /* total # list-searches */
long cc_lhits; /* # of matches against existing lists */ long cc_lhits; /* # of matches against existing lists */
#endif #endif
...@@ -75,11 +74,10 @@ typedef struct catctup ...@@ -75,11 +74,10 @@ typedef struct catctup
CatCache *my_cache; /* link to owning catcache */ CatCache *my_cache; /* link to owning catcache */
/* /*
* Each tuple in a cache is a member of two Dllists: one lists all the * Each tuple in a cache is a member of a Dllist that stores the elements
* elements in all the caches in LRU order, and the other lists just the * of its hash bucket. We keep each Dllist in LRU order to speed repeated
* elements in one hashbucket of one cache, also in LRU order. * lookups.
*/ */
Dlelem lrulist_elem; /* list member of global LRU list */
Dlelem cache_elem; /* list member of per-bucket list */ Dlelem cache_elem; /* list member of per-bucket list */
/* /*
...@@ -125,9 +123,8 @@ typedef struct catclist ...@@ -125,9 +123,8 @@ typedef struct catclist
* table rows satisfying the partial key. (Note: none of these will be * table rows satisfying the partial key. (Note: none of these will be
* negative cache entries.) * negative cache entries.)
* *
* A CatCList is only a member of a per-cache list; we do not do separate * A CatCList is only a member of a per-cache list; we do not currently
* LRU management for CatCLists. See CatalogCacheCleanup() for the * divide them into hash buckets.
* details of the management algorithm.
* *
* A list marked "dead" must not be returned by subsequent searches. * A list marked "dead" must not be returned by subsequent searches.
* However, it won't be physically deleted from the cache until its * However, it won't be physically deleted from the cache until its
...@@ -143,7 +140,6 @@ typedef struct catclist ...@@ -143,7 +140,6 @@ typedef struct catclist
int refcount; /* number of active references */ int refcount; /* number of active references */
bool dead; /* dead but not yet removed? */ bool dead; /* dead but not yet removed? */
bool ordered; /* members listed in index order? */ bool ordered; /* members listed in index order? */
bool touched; /* used since last CatalogCacheCleanup? */
short nkeys; /* number of lookup keys specified */ short nkeys; /* number of lookup keys specified */
uint32 hash_value; /* hash value for lookup keys */ uint32 hash_value; /* hash value for lookup keys */
HeapTupleData tuple; /* header for tuple holding keys */ HeapTupleData tuple; /* header for tuple holding keys */
...@@ -156,8 +152,6 @@ typedef struct catcacheheader ...@@ -156,8 +152,6 @@ typedef struct catcacheheader
{ {
CatCache *ch_caches; /* head of list of CatCache structs */ CatCache *ch_caches; /* head of list of CatCache structs */
int ch_ntup; /* # of tuples in all caches */ int ch_ntup; /* # of tuples in all caches */
int ch_maxtup; /* max # of tuples allowed (LRU) */
Dllist ch_lrulist; /* overall LRU list, most recent first */
} CatCacheHeader; } CatCacheHeader;
...@@ -169,7 +163,8 @@ extern void AtEOXact_CatCache(bool isCommit); ...@@ -169,7 +163,8 @@ extern void AtEOXact_CatCache(bool isCommit);
extern CatCache *InitCatCache(int id, Oid reloid, Oid indexoid, extern CatCache *InitCatCache(int id, Oid reloid, Oid indexoid,
int reloidattr, int reloidattr,
int nkeys, const int *key); int nkeys, const int *key,
int nbuckets);
extern void InitCatCachePhase2(CatCache *cache); extern void InitCatCachePhase2(CatCache *cache);
extern HeapTuple SearchCatCache(CatCache *cache, extern HeapTuple SearchCatCache(CatCache *cache,
......
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