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.
This commit is contained in:
parent
e1e133f264
commit
8b9bc234ad
197
src/backend/utils/cache/catcache.c
vendored
197
src/backend/utils/cache/catcache.c
vendored
@ -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 @@
|
|||||||
|
|
||||||
/* #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,
|
|||||||
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);
|
|||||||
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)
|
|||||||
#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)
|
|||||||
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)
|
|||||||
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)
|
|||||||
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)
|
|||||||
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)
|
|||||||
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)
|
|||||||
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,16 +560,21 @@ 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++)
|
||||||
{
|
{
|
||||||
CatCTup *ct = (CatCTup *) DLE_VAL(elt);
|
for (elt = DLGetHead(&ccp->cc_bucket[i]);
|
||||||
|
elt;
|
||||||
|
elt = DLGetSucc(elt))
|
||||||
|
{
|
||||||
|
CatCTup *ct = (CatCTup *) DLE_VAL(elt);
|
||||||
|
|
||||||
Assert(ct->ct_magic == CT_MAGIC);
|
Assert(ct->ct_magic == CT_MAGIC);
|
||||||
Assert(ct->refcount == 0);
|
Assert(ct->refcount == 0);
|
||||||
Assert(!ct->dead);
|
Assert(!ct->dead);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@ -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,
|
|||||||
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,
|
|||||||
*
|
*
|
||||||
* 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,
|
|||||||
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,
|
|||||||
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,
|
|||||||
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,
|
|||||||
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,
|
|||||||
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,
|
|||||||
|
|
||||||
/*
|
/*
|
||||||
* 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,
|
|||||||
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
|
||||||
|
169
src/backend/utils/cache/syscache.c
vendored
169
src/backend/utils/cache/syscache.c
vendored
@ -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
|
||||||
@ -55,14 +55,18 @@
|
|||||||
the list sorted alphabetically and adjust the cache numbers
|
the list sorted alphabetically and adjust the cache numbers
|
||||||
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
|
|||||||
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[] = {
|
|||||||
0,
|
0,
|
||||||
0,
|
0,
|
||||||
0
|
0
|
||||||
}},
|
},
|
||||||
|
32
|
||||||
|
},
|
||||||
{AccessMethodRelationId, /* AMNAME */
|
{AccessMethodRelationId, /* AMNAME */
|
||||||
AmNameIndexId,
|
AmNameIndexId,
|
||||||
0,
|
0,
|
||||||
@ -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[] = {
|
|||||||
0,
|
0,
|
||||||
0,
|
0,
|
||||||
0
|
0
|
||||||
}},
|
},
|
||||||
|
4
|
||||||
|
},
|
||||||
{AccessMethodOperatorRelationId, /* AMOPOPID */
|
{AccessMethodOperatorRelationId, /* AMOPOPID */
|
||||||
AccessMethodOperatorIndexId,
|
AccessMethodOperatorIndexId,
|
||||||
0,
|
0,
|
||||||
@ -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[] = {
|
|||||||
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[] = {
|
|||||||
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[] = {
|
|||||||
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[] = {
|
|||||||
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[] = {
|
|||||||
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[] = {
|
|||||||
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[] = {
|
|||||||
0,
|
0,
|
||||||
0,
|
0,
|
||||||
0
|
0
|
||||||
}},
|
},
|
||||||
|
128
|
||||||
|
},
|
||||||
{AuthIdRelationId, /* AUTHOID */
|
{AuthIdRelationId, /* AUTHOID */
|
||||||
AuthIdOidIndexId,
|
AuthIdOidIndexId,
|
||||||
0,
|
0,
|
||||||
@ -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[] = {
|
|||||||
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[] = {
|
|||||||
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[] = {
|
|||||||
0,
|
0,
|
||||||
0,
|
0,
|
||||||
0
|
0
|
||||||
}},
|
},
|
||||||
|
64
|
||||||
|
},
|
||||||
{ConversionRelationId, /* CONDEFAULT */
|
{ConversionRelationId, /* CONDEFAULT */
|
||||||
ConversionDefaultIndexId,
|
ConversionDefaultIndexId,
|
||||||
0,
|
0,
|
||||||
@ -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[] = {
|
|||||||
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[] = {
|
|||||||
0,
|
0,
|
||||||
0,
|
0,
|
||||||
0
|
0
|
||||||
}},
|
},
|
||||||
|
128
|
||||||
|
},
|
||||||
{DatabaseRelationId, /* DATABASEOID */
|
{DatabaseRelationId, /* DATABASEOID */
|
||||||
DatabaseOidIndexId,
|
DatabaseOidIndexId,
|
||||||
0,
|
0,
|
||||||
@ -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[] = {
|
|||||||
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[] = {
|
|||||||
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[] = {
|
|||||||
0,
|
0,
|
||||||
0,
|
0,
|
||||||
0
|
0
|
||||||
}},
|
},
|
||||||
|
4
|
||||||
|
},
|
||||||
{LanguageRelationId, /* LANGOID */
|
{LanguageRelationId, /* LANGOID */
|
||||||
LanguageOidIndexId,
|
LanguageOidIndexId,
|
||||||
0,
|
0,
|
||||||
@ -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[] = {
|
|||||||
0,
|
0,
|
||||||
0,
|
0,
|
||||||
0
|
0
|
||||||
}},
|
},
|
||||||
|
256
|
||||||
|
},
|
||||||
{NamespaceRelationId, /* NAMESPACEOID */
|
{NamespaceRelationId, /* NAMESPACEOID */
|
||||||
NamespaceOidIndexId,
|
NamespaceOidIndexId,
|
||||||
0,
|
0,
|
||||||
@ -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[] = {
|
|||||||
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[] = {
|
|||||||
0,
|
0,
|
||||||
0,
|
0,
|
||||||
0
|
0
|
||||||
}},
|
},
|
||||||
|
1024
|
||||||
|
},
|
||||||
{ProcedureRelationId, /* PROCNAMEARGSNSP */
|
{ProcedureRelationId, /* PROCNAMEARGSNSP */
|
||||||
ProcedureNameArgsNspIndexId,
|
ProcedureNameArgsNspIndexId,
|
||||||
0,
|
0,
|
||||||
@ -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[] = {
|
|||||||
0,
|
0,
|
||||||
0,
|
0,
|
||||||
0
|
0
|
||||||
}},
|
},
|
||||||
|
2048
|
||||||
|
},
|
||||||
{RelationRelationId, /* RELNAMENSP */
|
{RelationRelationId, /* RELNAMENSP */
|
||||||
ClassNameNspIndexId,
|
ClassNameNspIndexId,
|
||||||
ObjectIdAttributeNumber,
|
ObjectIdAttributeNumber,
|
||||||
@ -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[] = {
|
|||||||
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[] = {
|
|||||||
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[] = {
|
|||||||
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[] = {
|
|||||||
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[] = {
|
|||||||
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)
|
|||||||
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 @@
|
|||||||
* 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
|
|||||||
* 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
|
|||||||
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
|
|||||||
* 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
|
|||||||
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
|
|||||||
{
|
{
|
||||||
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);
|
|||||||
|
|
||||||
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,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user