LCOV - code coverage report
Current view: top level - src/backend/utils/cache - syscache.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 170 184 92.4 %
Date: 2025-01-18 04:15:08 Functions: 27 27 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * syscache.c
       4             :  *    System cache management routines
       5             :  *
       6             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       7             :  * Portions Copyright (c) 1994, Regents of the University of California
       8             :  *
       9             :  *
      10             :  * IDENTIFICATION
      11             :  *    src/backend/utils/cache/syscache.c
      12             :  *
      13             :  * NOTES
      14             :  *    These routines allow the parser/planner/executor to perform
      15             :  *    rapid lookups on the contents of the system catalogs.
      16             :  *
      17             :  *    see utils/syscache.h for a list of the cache IDs
      18             :  *
      19             :  *-------------------------------------------------------------------------
      20             :  */
      21             : #include "postgres.h"
      22             : 
      23             : #include "access/htup_details.h"
      24             : #include "catalog/pg_db_role_setting_d.h"
      25             : #include "catalog/pg_depend_d.h"
      26             : #include "catalog/pg_description_d.h"
      27             : #include "catalog/pg_seclabel_d.h"
      28             : #include "catalog/pg_shdepend_d.h"
      29             : #include "catalog/pg_shdescription_d.h"
      30             : #include "catalog/pg_shseclabel_d.h"
      31             : #include "common/int.h"
      32             : #include "lib/qunique.h"
      33             : #include "miscadmin.h"
      34             : #include "storage/lmgr.h"
      35             : #include "utils/catcache.h"
      36             : #include "utils/inval.h"
      37             : #include "utils/lsyscache.h"
      38             : #include "utils/rel.h"
      39             : #include "utils/syscache.h"
      40             : 
      41             : /*---------------------------------------------------------------------------
      42             : 
      43             :     Adding system caches:
      44             : 
      45             :     There must be a unique index underlying each syscache (ie, an index
      46             :     whose key is the same as that of the cache).  If there is not one
      47             :     already, add the definition for it to include/catalog/pg_*.h using
      48             :     DECLARE_UNIQUE_INDEX.
      49             :     (Adding an index requires a catversion.h update, while simply
      50             :     adding/deleting caches only requires a recompile.)
      51             : 
      52             :     Add a MAKE_SYSCACHE call to the same pg_*.h file specifying the name of
      53             :     your cache, the underlying index, and the initial number of hash buckets.
      54             : 
      55             :     The number of hash buckets must be a power of 2.  It's reasonable to
      56             :     set this to the number of entries that might be in the particular cache
      57             :     in a medium-size database.
      58             : 
      59             :     Finally, any place your relation gets heap_insert() or
      60             :     heap_update() calls, use CatalogTupleInsert() or CatalogTupleUpdate()
      61             :     instead, which also update indexes.  The heap_* calls do not do that.
      62             : 
      63             : *---------------------------------------------------------------------------
      64             : */
      65             : 
      66             : /*
      67             :  *      struct cachedesc: information defining a single syscache
      68             :  */
      69             : struct cachedesc
      70             : {
      71             :     Oid         reloid;         /* OID of the relation being cached */
      72             :     Oid         indoid;         /* OID of index relation for this cache */
      73             :     int         nkeys;          /* # of keys needed for cache lookup */
      74             :     int         key[4];         /* attribute numbers of key attrs */
      75             :     int         nbuckets;       /* number of hash buckets for this cache */
      76             : };
      77             : 
      78             : /* Macro to provide nkeys and key array with convenient syntax. */
      79             : #define KEY(...) VA_ARGS_NARGS(__VA_ARGS__), { __VA_ARGS__ }
      80             : 
      81             : #include "catalog/syscache_info.h"
      82             : 
      83             : StaticAssertDecl(lengthof(cacheinfo) == SysCacheSize,
      84             :                  "SysCacheSize does not match syscache.c's array");
      85             : 
      86             : static CatCache *SysCache[SysCacheSize];
      87             : 
      88             : static bool CacheInitialized = false;
      89             : 
      90             : /* Sorted array of OIDs of tables that have caches on them */
      91             : static Oid  SysCacheRelationOid[SysCacheSize];
      92             : static int  SysCacheRelationOidSize;
      93             : 
      94             : /* Sorted array of OIDs of tables and indexes used by caches */
      95             : static Oid  SysCacheSupportingRelOid[SysCacheSize * 2];
      96             : static int  SysCacheSupportingRelOidSize;
      97             : 
      98             : static int  oid_compare(const void *a, const void *b);
      99             : 
     100             : 
     101             : /*
     102             :  * InitCatalogCache - initialize the caches
     103             :  *
     104             :  * Note that no database access is done here; we only allocate memory
     105             :  * and initialize the cache structure.  Interrogation of the database
     106             :  * to complete initialization of a cache happens upon first use
     107             :  * of that cache.
     108             :  */
     109             : void
     110       30406 : InitCatalogCache(void)
     111             : {
     112             :     int         cacheId;
     113             : 
     114             :     Assert(!CacheInitialized);
     115             : 
     116       30406 :     SysCacheRelationOidSize = SysCacheSupportingRelOidSize = 0;
     117             : 
     118     2614916 :     for (cacheId = 0; cacheId < SysCacheSize; cacheId++)
     119             :     {
     120             :         /*
     121             :          * Assert that every enumeration value defined in syscache.h has been
     122             :          * populated in the cacheinfo array.
     123             :          */
     124             :         Assert(OidIsValid(cacheinfo[cacheId].reloid));
     125             :         Assert(OidIsValid(cacheinfo[cacheId].indoid));
     126             :         /* .nbuckets and .key[] are checked by InitCatCache() */
     127             : 
     128     5169020 :         SysCache[cacheId] = InitCatCache(cacheId,
     129             :                                          cacheinfo[cacheId].reloid,
     130             :                                          cacheinfo[cacheId].indoid,
     131             :                                          cacheinfo[cacheId].nkeys,
     132     2584510 :                                          cacheinfo[cacheId].key,
     133             :                                          cacheinfo[cacheId].nbuckets);
     134     2584510 :         if (!PointerIsValid(SysCache[cacheId]))
     135           0 :             elog(ERROR, "could not initialize cache %u (%d)",
     136             :                  cacheinfo[cacheId].reloid, cacheId);
     137             :         /* Accumulate data for OID lists, too */
     138     2584510 :         SysCacheRelationOid[SysCacheRelationOidSize++] =
     139     2584510 :             cacheinfo[cacheId].reloid;
     140     2584510 :         SysCacheSupportingRelOid[SysCacheSupportingRelOidSize++] =
     141     2584510 :             cacheinfo[cacheId].reloid;
     142     2584510 :         SysCacheSupportingRelOid[SysCacheSupportingRelOidSize++] =
     143     2584510 :             cacheinfo[cacheId].indoid;
     144             :         /* see comments for RelationInvalidatesSnapshotsOnly */
     145             :         Assert(!RelationInvalidatesSnapshotsOnly(cacheinfo[cacheId].reloid));
     146             :     }
     147             : 
     148             :     Assert(SysCacheRelationOidSize <= lengthof(SysCacheRelationOid));
     149             :     Assert(SysCacheSupportingRelOidSize <= lengthof(SysCacheSupportingRelOid));
     150             : 
     151             :     /* Sort and de-dup OID arrays, so we can use binary search. */
     152       30406 :     qsort(SysCacheRelationOid, SysCacheRelationOidSize,
     153             :           sizeof(Oid), oid_compare);
     154       30406 :     SysCacheRelationOidSize =
     155       30406 :         qunique(SysCacheRelationOid, SysCacheRelationOidSize, sizeof(Oid),
     156             :                 oid_compare);
     157             : 
     158       30406 :     qsort(SysCacheSupportingRelOid, SysCacheSupportingRelOidSize,
     159             :           sizeof(Oid), oid_compare);
     160       30406 :     SysCacheSupportingRelOidSize =
     161       30406 :         qunique(SysCacheSupportingRelOid, SysCacheSupportingRelOidSize,
     162             :                 sizeof(Oid), oid_compare);
     163             : 
     164       30406 :     CacheInitialized = true;
     165       30406 : }
     166             : 
     167             : /*
     168             :  * InitCatalogCachePhase2 - finish initializing the caches
     169             :  *
     170             :  * Finish initializing all the caches, including necessary database
     171             :  * access.
     172             :  *
     173             :  * This is *not* essential; normally we allow syscaches to be initialized
     174             :  * on first use.  However, it is useful as a mechanism to preload the
     175             :  * relcache with entries for the most-commonly-used system catalogs.
     176             :  * Therefore, we invoke this routine when we need to write a new relcache
     177             :  * init file.
     178             :  */
     179             : void
     180        2950 : InitCatalogCachePhase2(void)
     181             : {
     182             :     int         cacheId;
     183             : 
     184             :     Assert(CacheInitialized);
     185             : 
     186      253624 :     for (cacheId = 0; cacheId < SysCacheSize; cacheId++)
     187      250678 :         InitCatCachePhase2(SysCache[cacheId], true);
     188        2946 : }
     189             : 
     190             : 
     191             : /*
     192             :  * SearchSysCache
     193             :  *
     194             :  *  A layer on top of SearchCatCache that does the initialization and
     195             :  *  key-setting for you.
     196             :  *
     197             :  *  Returns the cache copy of the tuple if one is found, NULL if not.
     198             :  *  The tuple is the 'cache' copy and must NOT be modified!
     199             :  *
     200             :  *  When the caller is done using the tuple, call ReleaseSysCache()
     201             :  *  to release the reference count grabbed by SearchSysCache().  If this
     202             :  *  is not done, the tuple will remain locked in cache until end of
     203             :  *  transaction, which is tolerable but not desirable.
     204             :  *
     205             :  *  CAUTION: The tuple that is returned must NOT be freed by the caller!
     206             :  */
     207             : HeapTuple
     208     5089390 : SearchSysCache(int cacheId,
     209             :                Datum key1,
     210             :                Datum key2,
     211             :                Datum key3,
     212             :                Datum key4)
     213             : {
     214             :     Assert(cacheId >= 0 && cacheId < SysCacheSize &&
     215             :            PointerIsValid(SysCache[cacheId]));
     216             : 
     217     5089390 :     return SearchCatCache(SysCache[cacheId], key1, key2, key3, key4);
     218             : }
     219             : 
     220             : HeapTuple
     221    62732392 : SearchSysCache1(int cacheId,
     222             :                 Datum key1)
     223             : {
     224             :     Assert(cacheId >= 0 && cacheId < SysCacheSize &&
     225             :            PointerIsValid(SysCache[cacheId]));
     226             :     Assert(SysCache[cacheId]->cc_nkeys == 1);
     227             : 
     228    62732392 :     return SearchCatCache1(SysCache[cacheId], key1);
     229             : }
     230             : 
     231             : HeapTuple
     232     5052396 : SearchSysCache2(int cacheId,
     233             :                 Datum key1, Datum key2)
     234             : {
     235             :     Assert(cacheId >= 0 && cacheId < SysCacheSize &&
     236             :            PointerIsValid(SysCache[cacheId]));
     237             :     Assert(SysCache[cacheId]->cc_nkeys == 2);
     238             : 
     239     5052396 :     return SearchCatCache2(SysCache[cacheId], key1, key2);
     240             : }
     241             : 
     242             : HeapTuple
     243     4750312 : SearchSysCache3(int cacheId,
     244             :                 Datum key1, Datum key2, Datum key3)
     245             : {
     246             :     Assert(cacheId >= 0 && cacheId < SysCacheSize &&
     247             :            PointerIsValid(SysCache[cacheId]));
     248             :     Assert(SysCache[cacheId]->cc_nkeys == 3);
     249             : 
     250     4750312 :     return SearchCatCache3(SysCache[cacheId], key1, key2, key3);
     251             : }
     252             : 
     253             : HeapTuple
     254     3953848 : SearchSysCache4(int cacheId,
     255             :                 Datum key1, Datum key2, Datum key3, Datum key4)
     256             : {
     257             :     Assert(cacheId >= 0 && cacheId < SysCacheSize &&
     258             :            PointerIsValid(SysCache[cacheId]));
     259             :     Assert(SysCache[cacheId]->cc_nkeys == 4);
     260             : 
     261     3953848 :     return SearchCatCache4(SysCache[cacheId], key1, key2, key3, key4);
     262             : }
     263             : 
     264             : /*
     265             :  * ReleaseSysCache
     266             :  *      Release previously grabbed reference count on a tuple
     267             :  */
     268             : void
     269    76243704 : ReleaseSysCache(HeapTuple tuple)
     270             : {
     271    76243704 :     ReleaseCatCache(tuple);
     272    76243704 : }
     273             : 
     274             : /*
     275             :  * SearchSysCacheLocked1
     276             :  *
     277             :  * Combine SearchSysCache1() with acquiring a LOCKTAG_TUPLE at mode
     278             :  * InplaceUpdateTupleLock.  This is a tool for complying with the
     279             :  * README.tuplock section "Locking to write inplace-updated tables".  After
     280             :  * the caller's heap_update(), it should UnlockTuple(InplaceUpdateTupleLock)
     281             :  * and ReleaseSysCache().
     282             :  *
     283             :  * The returned tuple may be the subject of an uncommitted update, so this
     284             :  * doesn't prevent the "tuple concurrently updated" error.
     285             :  */
     286             : HeapTuple
     287       44250 : SearchSysCacheLocked1(int cacheId,
     288             :                       Datum key1)
     289             : {
     290       44250 :     CatCache   *cache = SysCache[cacheId];
     291             :     ItemPointerData tid;
     292             :     LOCKTAG     tag;
     293             : 
     294             :     /*----------
     295             :      * Since inplace updates may happen just before our LockTuple(), we must
     296             :      * return content acquired after LockTuple() of the TID we return.  If we
     297             :      * just fetched twice instead of looping, the following sequence would
     298             :      * defeat our locking:
     299             :      *
     300             :      * GRANT:   SearchSysCache1() = TID (1,5)
     301             :      * GRANT:   LockTuple(pg_class, (1,5))
     302             :      * [no more inplace update of (1,5) until we release the lock]
     303             :      * CLUSTER: SearchSysCache1() = TID (1,5)
     304             :      * CLUSTER: heap_update() = TID (1,8)
     305             :      * CLUSTER: COMMIT
     306             :      * GRANT:   SearchSysCache1() = TID (1,8)
     307             :      * GRANT:   return (1,8) from SearchSysCacheLocked1()
     308             :      * VACUUM:  SearchSysCache1() = TID (1,8)
     309             :      * VACUUM:  LockTuple(pg_class, (1,8))  # two TIDs now locked for one rel
     310             :      * VACUUM:  inplace update
     311             :      * GRANT:   heap_update() = (1,9)  # lose inplace update
     312             :      *
     313             :      * In the happy case, this takes two fetches, one to determine the TID to
     314             :      * lock and another to get the content and confirm the TID didn't change.
     315             :      *
     316             :      * This is valid even if the row gets updated to a new TID, the old TID
     317             :      * becomes LP_UNUSED, and the row gets updated back to its old TID.  We'd
     318             :      * still hold the right LOCKTAG_TUPLE and a copy of the row captured after
     319             :      * the LOCKTAG_TUPLE.
     320             :      */
     321       44250 :     ItemPointerSetInvalid(&tid);
     322             :     for (;;)
     323       44252 :     {
     324             :         HeapTuple   tuple;
     325       88502 :         LOCKMODE    lockmode = InplaceUpdateTupleLock;
     326             : 
     327       88502 :         tuple = SearchSysCache1(cacheId, key1);
     328       88502 :         if (ItemPointerIsValid(&tid))
     329             :         {
     330       44252 :             if (!HeapTupleIsValid(tuple))
     331             :             {
     332           0 :                 LockRelease(&tag, lockmode, false);
     333           0 :                 return tuple;
     334             :             }
     335       44252 :             if (ItemPointerEquals(&tid, &tuple->t_self))
     336       44250 :                 return tuple;
     337           2 :             LockRelease(&tag, lockmode, false);
     338             :         }
     339       44250 :         else if (!HeapTupleIsValid(tuple))
     340           0 :             return tuple;
     341             : 
     342       44252 :         tid = tuple->t_self;
     343       44252 :         ReleaseSysCache(tuple);
     344             : 
     345             :         /*
     346             :          * Do like LockTuple(rel, &tid, lockmode).  While cc_relisshared won't
     347             :          * change from one iteration to another, it may have been a temporary
     348             :          * "false" until our first SearchSysCache1().
     349             :          */
     350       44252 :         SET_LOCKTAG_TUPLE(tag,
     351             :                           cache->cc_relisshared ? InvalidOid : MyDatabaseId,
     352             :                           cache->cc_reloid,
     353             :                           ItemPointerGetBlockNumber(&tid),
     354             :                           ItemPointerGetOffsetNumber(&tid));
     355       44252 :         (void) LockAcquire(&tag, lockmode, false, false);
     356             : 
     357             :         /*
     358             :          * If an inplace update just finished, ensure we process the syscache
     359             :          * inval.
     360             :          *
     361             :          * If a heap_update() call just released its LOCKTAG_TUPLE, we'll
     362             :          * probably find the old tuple and reach "tuple concurrently updated".
     363             :          * If that heap_update() aborts, our LOCKTAG_TUPLE blocks inplace
     364             :          * updates while our caller works.
     365             :          */
     366       44252 :         AcceptInvalidationMessages();
     367             :     }
     368             : }
     369             : 
     370             : /*
     371             :  * SearchSysCacheCopy
     372             :  *
     373             :  * A convenience routine that does SearchSysCache and (if successful)
     374             :  * returns a modifiable copy of the syscache entry.  The original
     375             :  * syscache entry is released before returning.  The caller should
     376             :  * heap_freetuple() the result when done with it.
     377             :  */
     378             : HeapTuple
     379      461932 : SearchSysCacheCopy(int cacheId,
     380             :                    Datum key1,
     381             :                    Datum key2,
     382             :                    Datum key3,
     383             :                    Datum key4)
     384             : {
     385             :     HeapTuple   tuple,
     386             :                 newtuple;
     387             : 
     388      461932 :     tuple = SearchSysCache(cacheId, key1, key2, key3, key4);
     389      461932 :     if (!HeapTupleIsValid(tuple))
     390      130256 :         return tuple;
     391      331676 :     newtuple = heap_copytuple(tuple);
     392      331676 :     ReleaseSysCache(tuple);
     393      331676 :     return newtuple;
     394             : }
     395             : 
     396             : /*
     397             :  * SearchSysCacheLockedCopy1
     398             :  *
     399             :  * Meld SearchSysCacheLockedCopy1 with SearchSysCacheCopy().  After the
     400             :  * caller's heap_update(), it should UnlockTuple(InplaceUpdateTupleLock) and
     401             :  * heap_freetuple().
     402             :  */
     403             : HeapTuple
     404       17048 : SearchSysCacheLockedCopy1(int cacheId,
     405             :                           Datum key1)
     406             : {
     407             :     HeapTuple   tuple,
     408             :                 newtuple;
     409             : 
     410       17048 :     tuple = SearchSysCacheLocked1(cacheId, key1);
     411       17048 :     if (!HeapTupleIsValid(tuple))
     412           0 :         return tuple;
     413       17048 :     newtuple = heap_copytuple(tuple);
     414       17048 :     ReleaseSysCache(tuple);
     415       17048 :     return newtuple;
     416             : }
     417             : 
     418             : /*
     419             :  * SearchSysCacheExists
     420             :  *
     421             :  * A convenience routine that just probes to see if a tuple can be found.
     422             :  * No lock is retained on the syscache entry.
     423             :  */
     424             : bool
     425     1248346 : SearchSysCacheExists(int cacheId,
     426             :                      Datum key1,
     427             :                      Datum key2,
     428             :                      Datum key3,
     429             :                      Datum key4)
     430             : {
     431             :     HeapTuple   tuple;
     432             : 
     433     1248346 :     tuple = SearchSysCache(cacheId, key1, key2, key3, key4);
     434     1248346 :     if (!HeapTupleIsValid(tuple))
     435      212182 :         return false;
     436     1036164 :     ReleaseSysCache(tuple);
     437     1036164 :     return true;
     438             : }
     439             : 
     440             : /*
     441             :  * GetSysCacheOid
     442             :  *
     443             :  * A convenience routine that does SearchSysCache and returns the OID in the
     444             :  * oidcol column of the found tuple, or InvalidOid if no tuple could be found.
     445             :  * No lock is retained on the syscache entry.
     446             :  */
     447             : Oid
     448     3379112 : GetSysCacheOid(int cacheId,
     449             :                AttrNumber oidcol,
     450             :                Datum key1,
     451             :                Datum key2,
     452             :                Datum key3,
     453             :                Datum key4)
     454             : {
     455             :     HeapTuple   tuple;
     456             :     bool        isNull;
     457             :     Oid         result;
     458             : 
     459     3379112 :     tuple = SearchSysCache(cacheId, key1, key2, key3, key4);
     460     3379112 :     if (!HeapTupleIsValid(tuple))
     461     1357874 :         return InvalidOid;
     462     4042476 :     result = heap_getattr(tuple, oidcol,
     463     2021238 :                           SysCache[cacheId]->cc_tupdesc,
     464             :                           &isNull);
     465             :     Assert(!isNull);            /* columns used as oids should never be NULL */
     466     2021238 :     ReleaseSysCache(tuple);
     467     2021238 :     return result;
     468             : }
     469             : 
     470             : 
     471             : /*
     472             :  * SearchSysCacheAttName
     473             :  *
     474             :  * This routine is equivalent to SearchSysCache on the ATTNAME cache,
     475             :  * except that it will return NULL if the found attribute is marked
     476             :  * attisdropped.  This is convenient for callers that want to act as
     477             :  * though dropped attributes don't exist.
     478             :  */
     479             : HeapTuple
     480      101660 : SearchSysCacheAttName(Oid relid, const char *attname)
     481             : {
     482             :     HeapTuple   tuple;
     483             : 
     484      101660 :     tuple = SearchSysCache2(ATTNAME,
     485             :                             ObjectIdGetDatum(relid),
     486             :                             CStringGetDatum(attname));
     487      101660 :     if (!HeapTupleIsValid(tuple))
     488         936 :         return NULL;
     489      100724 :     if (((Form_pg_attribute) GETSTRUCT(tuple))->attisdropped)
     490             :     {
     491          78 :         ReleaseSysCache(tuple);
     492          78 :         return NULL;
     493             :     }
     494      100646 :     return tuple;
     495             : }
     496             : 
     497             : /*
     498             :  * SearchSysCacheCopyAttName
     499             :  *
     500             :  * As above, an attisdropped-aware version of SearchSysCacheCopy.
     501             :  */
     502             : HeapTuple
     503        8600 : SearchSysCacheCopyAttName(Oid relid, const char *attname)
     504             : {
     505             :     HeapTuple   tuple,
     506             :                 newtuple;
     507             : 
     508        8600 :     tuple = SearchSysCacheAttName(relid, attname);
     509        8600 :     if (!HeapTupleIsValid(tuple))
     510         666 :         return tuple;
     511        7934 :     newtuple = heap_copytuple(tuple);
     512        7934 :     ReleaseSysCache(tuple);
     513        7934 :     return newtuple;
     514             : }
     515             : 
     516             : /*
     517             :  * SearchSysCacheExistsAttName
     518             :  *
     519             :  * As above, an attisdropped-aware version of SearchSysCacheExists.
     520             :  */
     521             : bool
     522        1014 : SearchSysCacheExistsAttName(Oid relid, const char *attname)
     523             : {
     524             :     HeapTuple   tuple;
     525             : 
     526        1014 :     tuple = SearchSysCacheAttName(relid, attname);
     527        1014 :     if (!HeapTupleIsValid(tuple))
     528          54 :         return false;
     529         960 :     ReleaseSysCache(tuple);
     530         960 :     return true;
     531             : }
     532             : 
     533             : 
     534             : /*
     535             :  * SearchSysCacheAttNum
     536             :  *
     537             :  * This routine is equivalent to SearchSysCache on the ATTNUM cache,
     538             :  * except that it will return NULL if the found attribute is marked
     539             :  * attisdropped.  This is convenient for callers that want to act as
     540             :  * though dropped attributes don't exist.
     541             :  */
     542             : HeapTuple
     543        1580 : SearchSysCacheAttNum(Oid relid, int16 attnum)
     544             : {
     545             :     HeapTuple   tuple;
     546             : 
     547        1580 :     tuple = SearchSysCache2(ATTNUM,
     548             :                             ObjectIdGetDatum(relid),
     549             :                             Int16GetDatum(attnum));
     550        1580 :     if (!HeapTupleIsValid(tuple))
     551          12 :         return NULL;
     552        1568 :     if (((Form_pg_attribute) GETSTRUCT(tuple))->attisdropped)
     553             :     {
     554           0 :         ReleaseSysCache(tuple);
     555           0 :         return NULL;
     556             :     }
     557        1568 :     return tuple;
     558             : }
     559             : 
     560             : /*
     561             :  * SearchSysCacheCopyAttNum
     562             :  *
     563             :  * As above, an attisdropped-aware version of SearchSysCacheCopy.
     564             :  */
     565             : HeapTuple
     566        1516 : SearchSysCacheCopyAttNum(Oid relid, int16 attnum)
     567             : {
     568             :     HeapTuple   tuple,
     569             :                 newtuple;
     570             : 
     571        1516 :     tuple = SearchSysCacheAttNum(relid, attnum);
     572        1516 :     if (!HeapTupleIsValid(tuple))
     573           0 :         return NULL;
     574        1516 :     newtuple = heap_copytuple(tuple);
     575        1516 :     ReleaseSysCache(tuple);
     576        1516 :     return newtuple;
     577             : }
     578             : 
     579             : 
     580             : /*
     581             :  * SysCacheGetAttr
     582             :  *
     583             :  *      Given a tuple previously fetched by SearchSysCache(),
     584             :  *      extract a specific attribute.
     585             :  *
     586             :  * This is equivalent to using heap_getattr() on a tuple fetched
     587             :  * from a non-cached relation.  Usually, this is only used for attributes
     588             :  * that could be NULL or variable length; the fixed-size attributes in
     589             :  * a system table are accessed just by mapping the tuple onto the C struct
     590             :  * declarations from include/catalog/.
     591             :  *
     592             :  * As with heap_getattr(), if the attribute is of a pass-by-reference type
     593             :  * then a pointer into the tuple data area is returned --- the caller must
     594             :  * not modify or pfree the datum!
     595             :  *
     596             :  * Note: it is legal to use SysCacheGetAttr() with a cacheId referencing
     597             :  * a different cache for the same catalog the tuple was fetched from.
     598             :  */
     599             : Datum
     600     5096628 : SysCacheGetAttr(int cacheId, HeapTuple tup,
     601             :                 AttrNumber attributeNumber,
     602             :                 bool *isNull)
     603             : {
     604             :     /*
     605             :      * We just need to get the TupleDesc out of the cache entry, and then we
     606             :      * can apply heap_getattr().  Normally the cache control data is already
     607             :      * valid (because the caller recently fetched the tuple via this same
     608             :      * cache), but there are cases where we have to initialize the cache here.
     609             :      */
     610     5096628 :     if (cacheId < 0 || cacheId >= SysCacheSize ||
     611     5096628 :         !PointerIsValid(SysCache[cacheId]))
     612           0 :         elog(ERROR, "invalid cache ID: %d", cacheId);
     613     5096628 :     if (!PointerIsValid(SysCache[cacheId]->cc_tupdesc))
     614             :     {
     615       28940 :         InitCatCachePhase2(SysCache[cacheId], false);
     616             :         Assert(PointerIsValid(SysCache[cacheId]->cc_tupdesc));
     617             :     }
     618             : 
     619    10193256 :     return heap_getattr(tup, attributeNumber,
     620     5096628 :                         SysCache[cacheId]->cc_tupdesc,
     621             :                         isNull);
     622             : }
     623             : 
     624             : /*
     625             :  * SysCacheGetAttrNotNull
     626             :  *
     627             :  * As above, a version of SysCacheGetAttr which knows that the attr cannot
     628             :  * be NULL.
     629             :  */
     630             : Datum
     631     2868834 : SysCacheGetAttrNotNull(int cacheId, HeapTuple tup,
     632             :                        AttrNumber attributeNumber)
     633             : {
     634             :     bool        isnull;
     635             :     Datum       attr;
     636             : 
     637     2868834 :     attr = SysCacheGetAttr(cacheId, tup, attributeNumber, &isnull);
     638             : 
     639     2868834 :     if (isnull)
     640             :     {
     641           0 :         elog(ERROR,
     642             :              "unexpected null value in cached tuple for catalog %s column %s",
     643             :              get_rel_name(cacheinfo[cacheId].reloid),
     644             :              NameStr(TupleDescAttr(SysCache[cacheId]->cc_tupdesc, attributeNumber - 1)->attname));
     645             :     }
     646             : 
     647     2868834 :     return attr;
     648             : }
     649             : 
     650             : /*
     651             :  * GetSysCacheHashValue
     652             :  *
     653             :  * Get the hash value that would be used for a tuple in the specified cache
     654             :  * with the given search keys.
     655             :  *
     656             :  * The reason for exposing this as part of the API is that the hash value is
     657             :  * exposed in cache invalidation operations, so there are places outside the
     658             :  * catcache code that need to be able to compute the hash values.
     659             :  */
     660             : uint32
     661     1002070 : GetSysCacheHashValue(int cacheId,
     662             :                      Datum key1,
     663             :                      Datum key2,
     664             :                      Datum key3,
     665             :                      Datum key4)
     666             : {
     667     1002070 :     if (cacheId < 0 || cacheId >= SysCacheSize ||
     668     1002070 :         !PointerIsValid(SysCache[cacheId]))
     669           0 :         elog(ERROR, "invalid cache ID: %d", cacheId);
     670             : 
     671     1002070 :     return GetCatCacheHashValue(SysCache[cacheId], key1, key2, key3, key4);
     672             : }
     673             : 
     674             : /*
     675             :  * List-search interface
     676             :  */
     677             : struct catclist *
     678     3354624 : SearchSysCacheList(int cacheId, int nkeys,
     679             :                    Datum key1, Datum key2, Datum key3)
     680             : {
     681     3354624 :     if (cacheId < 0 || cacheId >= SysCacheSize ||
     682     3354624 :         !PointerIsValid(SysCache[cacheId]))
     683           0 :         elog(ERROR, "invalid cache ID: %d", cacheId);
     684             : 
     685     3354624 :     return SearchCatCacheList(SysCache[cacheId], nkeys,
     686             :                               key1, key2, key3);
     687             : }
     688             : 
     689             : /*
     690             :  * SysCacheInvalidate
     691             :  *
     692             :  *  Invalidate entries in the specified cache, given a hash value.
     693             :  *  See CatCacheInvalidate() for more info.
     694             :  *
     695             :  *  This routine is only quasi-public: it should only be used by inval.c.
     696             :  */
     697             : void
     698    19548078 : SysCacheInvalidate(int cacheId, uint32 hashValue)
     699             : {
     700    19548078 :     if (cacheId < 0 || cacheId >= SysCacheSize)
     701           0 :         elog(ERROR, "invalid cache ID: %d", cacheId);
     702             : 
     703             :     /* if this cache isn't initialized yet, no need to do anything */
     704    19548078 :     if (!PointerIsValid(SysCache[cacheId]))
     705           0 :         return;
     706             : 
     707    19548078 :     CatCacheInvalidate(SysCache[cacheId], hashValue);
     708             : }
     709             : 
     710             : /*
     711             :  * Certain relations that do not have system caches send snapshot invalidation
     712             :  * messages in lieu of catcache messages.  This is for the benefit of
     713             :  * GetCatalogSnapshot(), which can then reuse its existing MVCC snapshot
     714             :  * for scanning one of those catalogs, rather than taking a new one, if no
     715             :  * invalidation has been received.
     716             :  *
     717             :  * Relations that have syscaches need not (and must not) be listed here.  The
     718             :  * catcache invalidation messages will also flush the snapshot.  If you add a
     719             :  * syscache for one of these relations, remove it from this list.
     720             :  */
     721             : bool
     722    14356970 : RelationInvalidatesSnapshotsOnly(Oid relid)
     723             : {
     724    14356970 :     switch (relid)
     725             :     {
     726     2420028 :         case DbRoleSettingRelationId:
     727             :         case DependRelationId:
     728             :         case SharedDependRelationId:
     729             :         case DescriptionRelationId:
     730             :         case SharedDescriptionRelationId:
     731             :         case SecLabelRelationId:
     732             :         case SharedSecLabelRelationId:
     733     2420028 :             return true;
     734    11936942 :         default:
     735    11936942 :             break;
     736             :     }
     737             : 
     738    11936942 :     return false;
     739             : }
     740             : 
     741             : /*
     742             :  * Test whether a relation has a system cache.
     743             :  */
     744             : bool
     745     8997686 : RelationHasSysCache(Oid relid)
     746             : {
     747     8997686 :     int         low = 0,
     748     8997686 :                 high = SysCacheRelationOidSize - 1;
     749             : 
     750    39740328 :     while (low <= high)
     751             :     {
     752    39295326 :         int         middle = low + (high - low) / 2;
     753             : 
     754    39295326 :         if (SysCacheRelationOid[middle] == relid)
     755     8552684 :             return true;
     756    30742642 :         if (SysCacheRelationOid[middle] < relid)
     757    10677798 :             low = middle + 1;
     758             :         else
     759    20064844 :             high = middle - 1;
     760             :     }
     761             : 
     762      445002 :     return false;
     763             : }
     764             : 
     765             : /*
     766             :  * Test whether a relation supports a system cache, ie it is either a
     767             :  * cached table or the index used for a cache.
     768             :  */
     769             : bool
     770     2286990 : RelationSupportsSysCache(Oid relid)
     771             : {
     772     2286990 :     int         low = 0,
     773     2286990 :                 high = SysCacheSupportingRelOidSize - 1;
     774             : 
     775    19098942 :     while (low <= high)
     776             :     {
     777    17292430 :         int         middle = low + (high - low) / 2;
     778             : 
     779    17292430 :         if (SysCacheSupportingRelOid[middle] == relid)
     780      480478 :             return true;
     781    16811952 :         if (SysCacheSupportingRelOid[middle] < relid)
     782    15230576 :             low = middle + 1;
     783             :         else
     784     1581376 :             high = middle - 1;
     785             :     }
     786             : 
     787     1806512 :     return false;
     788             : }
     789             : 
     790             : 
     791             : /*
     792             :  * OID comparator for qsort
     793             :  */
     794             : static int
     795    60112662 : oid_compare(const void *a, const void *b)
     796             : {
     797    60112662 :     Oid         oa = *((const Oid *) a);
     798    60112662 :     Oid         ob = *((const Oid *) b);
     799             : 
     800    60112662 :     return pg_cmp_u32(oa, ob);
     801             : }

Generated by: LCOV version 1.14