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

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * attoptcache.c
       4             :  *    Attribute options cache management.
       5             :  *
       6             :  * Attribute options are cached separately from the fixed-size portion of
       7             :  * pg_attribute entries, which are handled by the relcache.
       8             :  *
       9             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
      10             :  * Portions Copyright (c) 1994, Regents of the University of California
      11             :  *
      12             :  * IDENTIFICATION
      13             :  *    src/backend/utils/cache/attoptcache.c
      14             :  *
      15             :  *-------------------------------------------------------------------------
      16             :  */
      17             : #include "postgres.h"
      18             : 
      19             : #include "access/reloptions.h"
      20             : #include "utils/attoptcache.h"
      21             : #include "utils/catcache.h"
      22             : #include "utils/hsearch.h"
      23             : #include "utils/inval.h"
      24             : #include "utils/syscache.h"
      25             : #include "varatt.h"
      26             : 
      27             : 
      28             : /* Hash table for information about each attribute's options */
      29             : static HTAB *AttoptCacheHash = NULL;
      30             : 
      31             : /* attrelid and attnum form the lookup key, and must appear first */
      32             : typedef struct
      33             : {
      34             :     Oid         attrelid;
      35             :     int         attnum;
      36             : } AttoptCacheKey;
      37             : 
      38             : typedef struct
      39             : {
      40             :     AttoptCacheKey key;         /* lookup key - must be first */
      41             :     AttributeOpts *opts;        /* options, or NULL if none */
      42             : } AttoptCacheEntry;
      43             : 
      44             : 
      45             : /*
      46             :  * InvalidateAttoptCacheCallback
      47             :  *      Flush cache entry (or entries) when pg_attribute is updated.
      48             :  *
      49             :  * When pg_attribute is updated, we must flush the cache entry at least
      50             :  * for that attribute.
      51             :  */
      52             : static void
      53      779470 : InvalidateAttoptCacheCallback(Datum arg, int cacheid, uint32 hashvalue)
      54             : {
      55             :     HASH_SEQ_STATUS status;
      56             :     AttoptCacheEntry *attopt;
      57             : 
      58             :     /*
      59             :      * By convention, zero hash value is passed to the callback as a sign that
      60             :      * it's time to invalidate the whole cache. See sinval.c, inval.c and
      61             :      * InvalidateSystemCachesExtended().
      62             :      */
      63      779470 :     if (hashvalue == 0)
      64          44 :         hash_seq_init(&status, AttoptCacheHash);
      65             :     else
      66      779426 :         hash_seq_init_with_hash_value(&status, AttoptCacheHash, hashvalue);
      67             : 
      68      784268 :     while ((attopt = (AttoptCacheEntry *) hash_seq_search(&status)) != NULL)
      69             :     {
      70        4798 :         if (attopt->opts)
      71           6 :             pfree(attopt->opts);
      72        4798 :         if (hash_search(AttoptCacheHash,
      73        4798 :                         &attopt->key,
      74             :                         HASH_REMOVE,
      75             :                         NULL) == NULL)
      76           0 :             elog(ERROR, "hash table corrupted");
      77             :     }
      78      779470 : }
      79             : 
      80             : /*
      81             :  * Hash function compatible with two-arg system cache hash function.
      82             :  */
      83             : static uint32
      84      142920 : relatt_cache_syshash(const void *key, Size keysize)
      85             : {
      86      142920 :     const AttoptCacheKey *ckey = key;
      87             : 
      88             :     Assert(keysize == sizeof(*ckey));
      89      142920 :     return GetSysCacheHashValue2(ATTNUM, ckey->attrelid, ckey->attnum);
      90             : }
      91             : 
      92             : /*
      93             :  * InitializeAttoptCache
      94             :  *      Initialize the attribute options cache.
      95             :  */
      96             : static void
      97         522 : InitializeAttoptCache(void)
      98             : {
      99             :     HASHCTL     ctl;
     100             : 
     101             :     /* Initialize the hash table. */
     102         522 :     ctl.keysize = sizeof(AttoptCacheKey);
     103         522 :     ctl.entrysize = sizeof(AttoptCacheEntry);
     104             : 
     105             :     /*
     106             :      * AttoptCacheEntry takes hash value from the system cache. For
     107             :      * AttoptCacheHash we use the same hash in order to speedup search by hash
     108             :      * value. This is used by hash_seq_init_with_hash_value().
     109             :      */
     110         522 :     ctl.hash = relatt_cache_syshash;
     111             : 
     112         522 :     AttoptCacheHash =
     113         522 :         hash_create("Attopt cache", 256, &ctl,
     114             :                     HASH_ELEM | HASH_FUNCTION);
     115             : 
     116             :     /* Make sure we've initialized CacheMemoryContext. */
     117         522 :     if (!CacheMemoryContext)
     118           0 :         CreateCacheMemoryContext();
     119             : 
     120             :     /* Watch for invalidation events. */
     121         522 :     CacheRegisterSyscacheCallback(ATTNUM,
     122             :                                   InvalidateAttoptCacheCallback,
     123             :                                   (Datum) 0);
     124         522 : }
     125             : 
     126             : /*
     127             :  * get_attribute_options
     128             :  *      Fetch attribute options for a specified table OID.
     129             :  */
     130             : AttributeOpts *
     131       71316 : get_attribute_options(Oid attrelid, int attnum)
     132             : {
     133             :     AttoptCacheKey key;
     134             :     AttoptCacheEntry *attopt;
     135             :     AttributeOpts *result;
     136             :     HeapTuple   tp;
     137             : 
     138             :     /* Find existing cache entry, if any. */
     139       71316 :     if (!AttoptCacheHash)
     140         522 :         InitializeAttoptCache();
     141       71316 :     memset(&key, 0, sizeof(key));   /* make sure any padding bits are unset */
     142       71316 :     key.attrelid = attrelid;
     143       71316 :     key.attnum = attnum;
     144             :     attopt =
     145       71316 :         (AttoptCacheEntry *) hash_search(AttoptCacheHash,
     146             :                                          &key,
     147             :                                          HASH_FIND,
     148             :                                          NULL);
     149             : 
     150             :     /* Not found in Attopt cache.  Construct new cache entry. */
     151       71316 :     if (!attopt)
     152             :     {
     153             :         AttributeOpts *opts;
     154             : 
     155       66806 :         tp = SearchSysCache2(ATTNUM,
     156             :                              ObjectIdGetDatum(attrelid),
     157             :                              Int16GetDatum(attnum));
     158             : 
     159             :         /*
     160             :          * If we don't find a valid HeapTuple, it must mean someone has
     161             :          * managed to request attribute details for a non-existent attribute.
     162             :          * We treat that case as if no options were specified.
     163             :          */
     164       66806 :         if (!HeapTupleIsValid(tp))
     165          54 :             opts = NULL;
     166             :         else
     167             :         {
     168             :             Datum       datum;
     169             :             bool        isNull;
     170             : 
     171       66752 :             datum = SysCacheGetAttr(ATTNUM,
     172             :                                     tp,
     173             :                                     Anum_pg_attribute_attoptions,
     174             :                                     &isNull);
     175       66752 :             if (isNull)
     176       66746 :                 opts = NULL;
     177             :             else
     178             :             {
     179           6 :                 bytea      *bytea_opts = attribute_reloptions(datum, false);
     180             : 
     181           6 :                 opts = MemoryContextAlloc(CacheMemoryContext,
     182           6 :                                           VARSIZE(bytea_opts));
     183           6 :                 memcpy(opts, bytea_opts, VARSIZE(bytea_opts));
     184             :             }
     185       66752 :             ReleaseSysCache(tp);
     186             :         }
     187             : 
     188             :         /*
     189             :          * It's important to create the actual cache entry only after reading
     190             :          * pg_attribute, since the read could cause a cache flush.
     191             :          */
     192       66806 :         attopt = (AttoptCacheEntry *) hash_search(AttoptCacheHash,
     193             :                                                   &key,
     194             :                                                   HASH_ENTER,
     195             :                                                   NULL);
     196       66806 :         attopt->opts = opts;
     197             :     }
     198             : 
     199             :     /* Return results in caller's memory context. */
     200       71316 :     if (attopt->opts == NULL)
     201       71310 :         return NULL;
     202           6 :     result = palloc(VARSIZE(attopt->opts));
     203           6 :     memcpy(result, attopt->opts, VARSIZE(attopt->opts));
     204           6 :     return result;
     205             : }

Generated by: LCOV version 1.14