LCOV - code coverage report
Current view: top level - src/backend/access/gin - ginscan.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 175 186 94.1 %
Date: 2025-01-18 05:15:39 Functions: 8 8 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * ginscan.c
       4             :  *    routines to manage scans of inverted index relations
       5             :  *
       6             :  *
       7             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       8             :  * Portions Copyright (c) 1994, Regents of the University of California
       9             :  *
      10             :  * IDENTIFICATION
      11             :  *          src/backend/access/gin/ginscan.c
      12             :  *-------------------------------------------------------------------------
      13             :  */
      14             : 
      15             : #include "postgres.h"
      16             : 
      17             : #include "access/gin_private.h"
      18             : #include "access/relscan.h"
      19             : #include "pgstat.h"
      20             : #include "utils/memutils.h"
      21             : #include "utils/rel.h"
      22             : 
      23             : 
      24             : IndexScanDesc
      25        1592 : ginbeginscan(Relation rel, int nkeys, int norderbys)
      26             : {
      27             :     IndexScanDesc scan;
      28             :     GinScanOpaque so;
      29             : 
      30             :     /* no order by operators allowed */
      31             :     Assert(norderbys == 0);
      32             : 
      33        1592 :     scan = RelationGetIndexScan(rel, nkeys, norderbys);
      34             : 
      35             :     /* allocate private workspace */
      36        1592 :     so = (GinScanOpaque) palloc(sizeof(GinScanOpaqueData));
      37        1592 :     so->keys = NULL;
      38        1592 :     so->nkeys = 0;
      39        1592 :     so->tempCtx = AllocSetContextCreate(CurrentMemoryContext,
      40             :                                         "Gin scan temporary context",
      41             :                                         ALLOCSET_DEFAULT_SIZES);
      42        1592 :     so->keyCtx = AllocSetContextCreate(CurrentMemoryContext,
      43             :                                        "Gin scan key context",
      44             :                                        ALLOCSET_DEFAULT_SIZES);
      45        1592 :     initGinState(&so->ginstate, scan->indexRelation);
      46             : 
      47        1592 :     scan->opaque = so;
      48             : 
      49        1592 :     return scan;
      50             : }
      51             : 
      52             : /*
      53             :  * Create a new GinScanEntry, unless an equivalent one already exists,
      54             :  * in which case just return it
      55             :  */
      56             : static GinScanEntry
      57        6692 : ginFillScanEntry(GinScanOpaque so, OffsetNumber attnum,
      58             :                  StrategyNumber strategy, int32 searchMode,
      59             :                  Datum queryKey, GinNullCategory queryCategory,
      60             :                  bool isPartialMatch, Pointer extra_data)
      61             : {
      62        6692 :     GinState   *ginstate = &so->ginstate;
      63             :     GinScanEntry scanEntry;
      64             :     uint32      i;
      65             : 
      66             :     /*
      67             :      * Look for an existing equivalent entry.
      68             :      *
      69             :      * Entries with non-null extra_data are never considered identical, since
      70             :      * we can't know exactly what the opclass might be doing with that.
      71             :      */
      72        6692 :     if (extra_data == NULL)
      73             :     {
      74      457010 :         for (i = 0; i < so->totalentries; i++)
      75             :         {
      76      452000 :             GinScanEntry prevEntry = so->entries[i];
      77             : 
      78      452000 :             if (prevEntry->extra_data == NULL &&
      79      451700 :                 prevEntry->isPartialMatch == isPartialMatch &&
      80      451700 :                 prevEntry->strategy == strategy &&
      81      451560 :                 prevEntry->searchMode == searchMode &&
      82      903096 :                 prevEntry->attnum == attnum &&
      83      451536 :                 ginCompareEntries(ginstate, attnum,
      84             :                                   prevEntry->queryKey,
      85      451536 :                                   prevEntry->queryCategory,
      86             :                                   queryKey,
      87             :                                   queryCategory) == 0)
      88             :             {
      89             :                 /* Successful match */
      90           0 :                 return prevEntry;
      91             :             }
      92             :         }
      93             :     }
      94             : 
      95             :     /* Nope, create a new entry */
      96        6692 :     scanEntry = (GinScanEntry) palloc(sizeof(GinScanEntryData));
      97        6692 :     scanEntry->queryKey = queryKey;
      98        6692 :     scanEntry->queryCategory = queryCategory;
      99        6692 :     scanEntry->isPartialMatch = isPartialMatch;
     100        6692 :     scanEntry->extra_data = extra_data;
     101        6692 :     scanEntry->strategy = strategy;
     102        6692 :     scanEntry->searchMode = searchMode;
     103        6692 :     scanEntry->attnum = attnum;
     104             : 
     105        6692 :     scanEntry->buffer = InvalidBuffer;
     106        6692 :     ItemPointerSetMin(&scanEntry->curItem);
     107        6692 :     scanEntry->matchBitmap = NULL;
     108        6692 :     scanEntry->matchIterator = NULL;
     109        6692 :     scanEntry->matchResult = NULL;
     110        6692 :     scanEntry->list = NULL;
     111        6692 :     scanEntry->nlist = 0;
     112        6692 :     scanEntry->offset = InvalidOffsetNumber;
     113        6692 :     scanEntry->isFinished = false;
     114        6692 :     scanEntry->reduceResult = false;
     115             : 
     116             :     /* Add it to so's array */
     117        6692 :     if (so->totalentries >= so->allocentries)
     118             :     {
     119          46 :         so->allocentries *= 2;
     120          46 :         so->entries = (GinScanEntry *)
     121          46 :             repalloc(so->entries, so->allocentries * sizeof(GinScanEntry));
     122             :     }
     123        6692 :     so->entries[so->totalentries++] = scanEntry;
     124             : 
     125        6692 :     return scanEntry;
     126             : }
     127             : 
     128             : /*
     129             :  * Append hidden scan entry of given category to the scan key.
     130             :  *
     131             :  * NB: this had better be called at most once per scan key, since
     132             :  * ginFillScanKey leaves room for only one hidden entry.  Currently,
     133             :  * it seems sufficiently clear that this is true that we don't bother
     134             :  * with any cross-check logic.
     135             :  */
     136             : static void
     137         326 : ginScanKeyAddHiddenEntry(GinScanOpaque so, GinScanKey key,
     138             :                          GinNullCategory queryCategory)
     139             : {
     140         326 :     int         i = key->nentries++;
     141             : 
     142             :     /* strategy is of no interest because this is not a partial-match item */
     143         326 :     key->scanEntry[i] = ginFillScanEntry(so, key->attnum,
     144             :                                          InvalidStrategy, key->searchMode,
     145             :                                          (Datum) 0, queryCategory,
     146             :                                          false, NULL);
     147         326 : }
     148             : 
     149             : /*
     150             :  * Initialize the next GinScanKey using the output from the extractQueryFn
     151             :  */
     152             : static void
     153        1714 : ginFillScanKey(GinScanOpaque so, OffsetNumber attnum,
     154             :                StrategyNumber strategy, int32 searchMode,
     155             :                Datum query, uint32 nQueryValues,
     156             :                Datum *queryValues, GinNullCategory *queryCategories,
     157             :                bool *partial_matches, Pointer *extra_data)
     158             : {
     159        1714 :     GinScanKey  key = &(so->keys[so->nkeys++]);
     160        1714 :     GinState   *ginstate = &so->ginstate;
     161             :     uint32      i;
     162             : 
     163        1714 :     key->nentries = nQueryValues;
     164        1714 :     key->nuserentries = nQueryValues;
     165             : 
     166             :     /* Allocate one extra array slot for possible "hidden" entry */
     167        3428 :     key->scanEntry = (GinScanEntry *) palloc(sizeof(GinScanEntry) *
     168        1714 :                                              (nQueryValues + 1));
     169        3428 :     key->entryRes = (GinTernaryValue *) palloc0(sizeof(GinTernaryValue) *
     170        1714 :                                                 (nQueryValues + 1));
     171             : 
     172        1714 :     key->query = query;
     173        1714 :     key->queryValues = queryValues;
     174        1714 :     key->queryCategories = queryCategories;
     175        1714 :     key->extra_data = extra_data;
     176        1714 :     key->strategy = strategy;
     177        1714 :     key->searchMode = searchMode;
     178        1714 :     key->attnum = attnum;
     179             : 
     180             :     /*
     181             :      * Initially, scan keys of GIN_SEARCH_MODE_ALL mode are marked
     182             :      * excludeOnly.  This might get changed later.
     183             :      */
     184        1714 :     key->excludeOnly = (searchMode == GIN_SEARCH_MODE_ALL);
     185             : 
     186        1714 :     ItemPointerSetMin(&key->curItem);
     187        1714 :     key->curItemMatches = false;
     188        1714 :     key->recheckCurItem = false;
     189        1714 :     key->isFinished = false;
     190        1714 :     key->nrequired = 0;
     191        1714 :     key->nadditional = 0;
     192        1714 :     key->requiredEntries = NULL;
     193        1714 :     key->additionalEntries = NULL;
     194             : 
     195        1714 :     ginInitConsistentFunction(ginstate, key);
     196             : 
     197             :     /* Set up normal scan entries using extractQueryFn's outputs */
     198        8080 :     for (i = 0; i < nQueryValues; i++)
     199             :     {
     200             :         Datum       queryKey;
     201             :         GinNullCategory queryCategory;
     202             :         bool        isPartialMatch;
     203             :         Pointer     this_extra;
     204             : 
     205        6366 :         queryKey = queryValues[i];
     206        6366 :         queryCategory = queryCategories[i];
     207        6366 :         isPartialMatch =
     208        6366 :             (ginstate->canPartialMatch[attnum - 1] && partial_matches)
     209        6366 :             ? partial_matches[i] : false;
     210        6366 :         this_extra = (extra_data) ? extra_data[i] : NULL;
     211             : 
     212        6366 :         key->scanEntry[i] = ginFillScanEntry(so, attnum,
     213             :                                              strategy, searchMode,
     214             :                                              queryKey, queryCategory,
     215             :                                              isPartialMatch, this_extra);
     216             :     }
     217             : 
     218             :     /*
     219             :      * For GIN_SEARCH_MODE_INCLUDE_EMPTY and GIN_SEARCH_MODE_EVERYTHING search
     220             :      * modes, we add the "hidden" entry immediately.  GIN_SEARCH_MODE_ALL is
     221             :      * handled later, since we might be able to omit the hidden entry for it.
     222             :      */
     223        1714 :     if (searchMode == GIN_SEARCH_MODE_INCLUDE_EMPTY)
     224          44 :         ginScanKeyAddHiddenEntry(so, key, GIN_CAT_EMPTY_ITEM);
     225        1670 :     else if (searchMode == GIN_SEARCH_MODE_EVERYTHING)
     226           0 :         ginScanKeyAddHiddenEntry(so, key, GIN_CAT_EMPTY_QUERY);
     227        1714 : }
     228             : 
     229             : /*
     230             :  * Release current scan keys, if any.
     231             :  */
     232             : void
     233        4788 : ginFreeScanKeys(GinScanOpaque so)
     234             : {
     235             :     uint32      i;
     236             : 
     237        4788 :     if (so->keys == NULL)
     238        3190 :         return;
     239             : 
     240        8290 :     for (i = 0; i < so->totalentries; i++)
     241             :     {
     242        6692 :         GinScanEntry entry = so->entries[i];
     243             : 
     244        6692 :         if (entry->buffer != InvalidBuffer)
     245           0 :             ReleaseBuffer(entry->buffer);
     246        6692 :         if (entry->list)
     247        4436 :             pfree(entry->list);
     248        6692 :         if (entry->matchIterator)
     249           0 :             tbm_end_private_iterate(entry->matchIterator);
     250        6692 :         if (entry->matchBitmap)
     251         534 :             tbm_free(entry->matchBitmap);
     252             :     }
     253             : 
     254        1598 :     MemoryContextReset(so->keyCtx);
     255             : 
     256        1598 :     so->keys = NULL;
     257        1598 :     so->nkeys = 0;
     258        1598 :     so->entries = NULL;
     259        1598 :     so->totalentries = 0;
     260             : }
     261             : 
     262             : void
     263        1598 : ginNewScanKey(IndexScanDesc scan)
     264             : {
     265        1598 :     ScanKey     scankey = scan->keyData;
     266        1598 :     GinScanOpaque so = (GinScanOpaque) scan->opaque;
     267             :     int         i;
     268        1598 :     bool        hasNullQuery = false;
     269        1598 :     bool        attrHasNormalScan[INDEX_MAX_KEYS] = {false};
     270             :     MemoryContext oldCtx;
     271             : 
     272             :     /*
     273             :      * Allocate all the scan key information in the key context. (If
     274             :      * extractQuery leaks anything there, it won't be reset until the end of
     275             :      * scan or rescan, but that's OK.)
     276             :      */
     277        1598 :     oldCtx = MemoryContextSwitchTo(so->keyCtx);
     278             : 
     279             :     /* if no scan keys provided, allocate extra EVERYTHING GinScanKey */
     280        1598 :     so->keys = (GinScanKey)
     281        1598 :         palloc(Max(scan->numberOfKeys, 1) * sizeof(GinScanKeyData));
     282        1598 :     so->nkeys = 0;
     283             : 
     284             :     /* initialize expansible array of GinScanEntry pointers */
     285        1598 :     so->totalentries = 0;
     286        1598 :     so->allocentries = 32;
     287        1598 :     so->entries = (GinScanEntry *)
     288        1598 :         palloc(so->allocentries * sizeof(GinScanEntry));
     289             : 
     290        1598 :     so->isVoidRes = false;
     291             : 
     292        3312 :     for (i = 0; i < scan->numberOfKeys; i++)
     293             :     {
     294        1726 :         ScanKey     skey = &scankey[i];
     295             :         Datum      *queryValues;
     296        1726 :         int32       nQueryValues = 0;
     297        1726 :         bool       *partial_matches = NULL;
     298        1726 :         Pointer    *extra_data = NULL;
     299        1726 :         bool       *nullFlags = NULL;
     300             :         GinNullCategory *categories;
     301        1726 :         int32       searchMode = GIN_SEARCH_MODE_DEFAULT;
     302             : 
     303             :         /*
     304             :          * We assume that GIN-indexable operators are strict, so a null query
     305             :          * argument means an unsatisfiable query.
     306             :          */
     307        1726 :         if (skey->sk_flags & SK_ISNULL)
     308             :         {
     309           0 :             so->isVoidRes = true;
     310          12 :             break;
     311             :         }
     312             : 
     313             :         /* OK to call the extractQueryFn */
     314             :         queryValues = (Datum *)
     315        5178 :             DatumGetPointer(FunctionCall7Coll(&so->ginstate.extractQueryFn[skey->sk_attno - 1],
     316        1726 :                                               so->ginstate.supportCollation[skey->sk_attno - 1],
     317             :                                               skey->sk_argument,
     318             :                                               PointerGetDatum(&nQueryValues),
     319        1726 :                                               UInt16GetDatum(skey->sk_strategy),
     320             :                                               PointerGetDatum(&partial_matches),
     321             :                                               PointerGetDatum(&extra_data),
     322             :                                               PointerGetDatum(&nullFlags),
     323             :                                               PointerGetDatum(&searchMode)));
     324             : 
     325             :         /*
     326             :          * If bogus searchMode is returned, treat as GIN_SEARCH_MODE_ALL; note
     327             :          * in particular we don't allow extractQueryFn to select
     328             :          * GIN_SEARCH_MODE_EVERYTHING.
     329             :          */
     330        1726 :         if (searchMode < GIN_SEARCH_MODE_DEFAULT ||
     331        1726 :             searchMode > GIN_SEARCH_MODE_ALL)
     332           0 :             searchMode = GIN_SEARCH_MODE_ALL;
     333             : 
     334             :         /* Non-default modes require the index to have placeholders */
     335        1726 :         if (searchMode != GIN_SEARCH_MODE_DEFAULT)
     336         364 :             hasNullQuery = true;
     337             : 
     338             :         /*
     339             :          * In default mode, no keys means an unsatisfiable query.
     340             :          */
     341        1726 :         if (queryValues == NULL || nQueryValues <= 0)
     342             :         {
     343         304 :             if (searchMode == GIN_SEARCH_MODE_DEFAULT)
     344             :             {
     345          12 :                 so->isVoidRes = true;
     346          12 :                 break;
     347             :             }
     348         292 :             nQueryValues = 0;   /* ensure sane value */
     349             :         }
     350             : 
     351             :         /*
     352             :          * Create GinNullCategory representation.  If the extractQueryFn
     353             :          * didn't create a nullFlags array, we assume everything is non-null.
     354             :          * While at it, detect whether any null keys are present.
     355             :          */
     356        1714 :         categories = (GinNullCategory *) palloc0(nQueryValues * sizeof(GinNullCategory));
     357        1714 :         if (nullFlags)
     358             :         {
     359             :             int32       j;
     360             : 
     361        4034 :             for (j = 0; j < nQueryValues; j++)
     362             :             {
     363        3464 :                 if (nullFlags[j])
     364             :                 {
     365           0 :                     categories[j] = GIN_CAT_NULL_KEY;
     366           0 :                     hasNullQuery = true;
     367             :                 }
     368             :             }
     369             :         }
     370             : 
     371        1714 :         ginFillScanKey(so, skey->sk_attno,
     372        1714 :                        skey->sk_strategy, searchMode,
     373             :                        skey->sk_argument, nQueryValues,
     374             :                        queryValues, categories,
     375             :                        partial_matches, extra_data);
     376             : 
     377             :         /* Remember if we had any non-excludeOnly keys */
     378        1714 :         if (searchMode != GIN_SEARCH_MODE_ALL)
     379        1394 :             attrHasNormalScan[skey->sk_attno - 1] = true;
     380             :     }
     381             : 
     382             :     /*
     383             :      * Processing GIN_SEARCH_MODE_ALL scan keys requires us to make a second
     384             :      * pass over the scan keys.  Above we marked each such scan key as
     385             :      * excludeOnly.  If the involved column has any normal (not excludeOnly)
     386             :      * scan key as well, then we can leave it like that.  Otherwise, one
     387             :      * excludeOnly scan key must receive a GIN_CAT_EMPTY_QUERY hidden entry
     388             :      * and be set to normal (excludeOnly = false).
     389             :      */
     390        3312 :     for (i = 0; i < so->nkeys; i++)
     391             :     {
     392        1714 :         GinScanKey  key = &so->keys[i];
     393             : 
     394        1714 :         if (key->searchMode != GIN_SEARCH_MODE_ALL)
     395        1394 :             continue;
     396             : 
     397         320 :         if (!attrHasNormalScan[key->attnum - 1])
     398             :         {
     399         282 :             key->excludeOnly = false;
     400         282 :             ginScanKeyAddHiddenEntry(so, key, GIN_CAT_EMPTY_QUERY);
     401         282 :             attrHasNormalScan[key->attnum - 1] = true;
     402             :         }
     403             :     }
     404             : 
     405             :     /*
     406             :      * If there are no regular scan keys, generate an EVERYTHING scankey to
     407             :      * drive a full-index scan.
     408             :      */
     409        1598 :     if (so->nkeys == 0 && !so->isVoidRes)
     410             :     {
     411           0 :         hasNullQuery = true;
     412           0 :         ginFillScanKey(so, FirstOffsetNumber,
     413             :                        InvalidStrategy, GIN_SEARCH_MODE_EVERYTHING,
     414             :                        (Datum) 0, 0,
     415             :                        NULL, NULL, NULL, NULL);
     416             :     }
     417             : 
     418             :     /*
     419             :      * If the index is version 0, it may be missing null and placeholder
     420             :      * entries, which would render searches for nulls and full-index scans
     421             :      * unreliable.  Throw an error if so.
     422             :      */
     423        1598 :     if (hasNullQuery && !so->isVoidRes)
     424             :     {
     425             :         GinStatsData ginStats;
     426             : 
     427         324 :         ginGetStats(scan->indexRelation, &ginStats);
     428         324 :         if (ginStats.ginVersion < 1)
     429           0 :             ereport(ERROR,
     430             :                     (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
     431             :                      errmsg("old GIN indexes do not support whole-index scans nor searches for nulls"),
     432             :                      errhint("To fix this, do REINDEX INDEX \"%s\".",
     433             :                              RelationGetRelationName(scan->indexRelation))));
     434             :     }
     435             : 
     436        1598 :     MemoryContextSwitchTo(oldCtx);
     437             : 
     438        1598 :     pgstat_count_index_scan(scan->indexRelation);
     439        1598 : }
     440             : 
     441             : void
     442        1598 : ginrescan(IndexScanDesc scan, ScanKey scankey, int nscankeys,
     443             :           ScanKey orderbys, int norderbys)
     444             : {
     445        1598 :     GinScanOpaque so = (GinScanOpaque) scan->opaque;
     446             : 
     447        1598 :     ginFreeScanKeys(so);
     448             : 
     449        1598 :     if (scankey && scan->numberOfKeys > 0)
     450        1598 :         memcpy(scan->keyData, scankey, scan->numberOfKeys * sizeof(ScanKeyData));
     451        1598 : }
     452             : 
     453             : 
     454             : void
     455        1592 : ginendscan(IndexScanDesc scan)
     456             : {
     457        1592 :     GinScanOpaque so = (GinScanOpaque) scan->opaque;
     458             : 
     459        1592 :     ginFreeScanKeys(so);
     460             : 
     461        1592 :     MemoryContextDelete(so->tempCtx);
     462        1592 :     MemoryContextDelete(so->keyCtx);
     463             : 
     464        1592 :     pfree(so);
     465        1592 : }

Generated by: LCOV version 1.14