LCOV - code coverage report
Current view: top level - src/backend/access/gist - gistscan.c (source / functions) Coverage Total Hit
Test: PostgreSQL 19devel Lines: 95.5 % 112 107
Test Date: 2026-03-27 22:16:19 Functions: 100.0 % 4 4
Legend: Lines:     hit not hit

            Line data    Source code
       1              : /*-------------------------------------------------------------------------
       2              :  *
       3              :  * gistscan.c
       4              :  *    routines to manage scans on GiST index relations
       5              :  *
       6              :  *
       7              :  * Portions Copyright (c) 1996-2026, PostgreSQL Global Development Group
       8              :  * Portions Copyright (c) 1994, Regents of the University of California
       9              :  *
      10              :  * IDENTIFICATION
      11              :  *    src/backend/access/gist/gistscan.c
      12              :  *
      13              :  *-------------------------------------------------------------------------
      14              :  */
      15              : #include "postgres.h"
      16              : 
      17              : #include "access/gist_private.h"
      18              : #include "access/gistscan.h"
      19              : #include "access/relscan.h"
      20              : #include "utils/float.h"
      21              : #include "utils/lsyscache.h"
      22              : #include "utils/memutils.h"
      23              : #include "utils/rel.h"
      24              : 
      25              : 
      26              : /*
      27              :  * Pairing heap comparison function for the GISTSearchItem queue
      28              :  */
      29              : static int
      30       197654 : pairingheap_GISTSearchItem_cmp(const pairingheap_node *a, const pairingheap_node *b, void *arg)
      31              : {
      32       197654 :     const GISTSearchItem *sa = (const GISTSearchItem *) a;
      33       197654 :     const GISTSearchItem *sb = (const GISTSearchItem *) b;
      34       197654 :     IndexScanDesc scan = (IndexScanDesc) arg;
      35              :     int         i;
      36              : 
      37              :     /* Order according to distance comparison */
      38       199789 :     for (i = 0; i < scan->numberOfOrderBys; i++)
      39              :     {
      40        37991 :         if (sa->distances[i].isnull)
      41              :         {
      42           68 :             if (!sb->distances[i].isnull)
      43           68 :                 return -1;
      44              :         }
      45        37923 :         else if (sb->distances[i].isnull)
      46              :         {
      47           16 :             return 1;
      48              :         }
      49              :         else
      50              :         {
      51        75814 :             int         cmp = -float8_cmp_internal(sa->distances[i].value,
      52        37907 :                                                    sb->distances[i].value);
      53              : 
      54        37907 :             if (cmp != 0)
      55        35772 :                 return cmp;
      56              :         }
      57              :     }
      58              : 
      59              :     /* Heap items go before inner pages, to ensure a depth-first search */
      60       161798 :     if (GISTSearchItemIsHeap(*sa) && !GISTSearchItemIsHeap(*sb))
      61            0 :         return 1;
      62       161798 :     if (!GISTSearchItemIsHeap(*sa) && GISTSearchItemIsHeap(*sb))
      63            2 :         return -1;
      64              : 
      65       161796 :     return 0;
      66              : }
      67              : 
      68              : 
      69              : /*
      70              :  * Index AM API functions for scanning GiST indexes
      71              :  */
      72              : 
      73              : IndexScanDesc
      74         3748 : gistbeginscan(Relation r, int nkeys, int norderbys)
      75              : {
      76              :     IndexScanDesc scan;
      77              :     GISTSTATE  *giststate;
      78              :     GISTScanOpaque so;
      79              :     MemoryContext oldCxt;
      80              : 
      81         3748 :     scan = RelationGetIndexScan(r, nkeys, norderbys);
      82              : 
      83              :     /* First, set up a GISTSTATE with a scan-lifespan memory context */
      84         3748 :     giststate = initGISTstate(scan->indexRelation);
      85              : 
      86              :     /*
      87              :      * Everything made below is in the scanCxt, or is a child of the scanCxt,
      88              :      * so it'll all go away automatically in gistendscan.
      89              :      */
      90         3748 :     oldCxt = MemoryContextSwitchTo(giststate->scanCxt);
      91              : 
      92              :     /* initialize opaque data */
      93         3748 :     so = palloc0_object(GISTScanOpaqueData);
      94         3748 :     so->giststate = giststate;
      95         3748 :     giststate->tempCxt = createTempGistContext();
      96         3748 :     so->queue = NULL;
      97         3748 :     so->queueCxt = giststate->scanCxt;    /* see gistrescan */
      98              : 
      99              :     /* workspaces with size dependent on numberOfOrderBys: */
     100         3748 :     so->distances = palloc(sizeof(so->distances[0]) * scan->numberOfOrderBys);
     101         3748 :     so->qual_ok = true;          /* in case there are zero keys */
     102         3748 :     if (scan->numberOfOrderBys > 0)
     103              :     {
     104           75 :         scan->xs_orderbyvals = palloc0_array(Datum, scan->numberOfOrderBys);
     105           75 :         scan->xs_orderbynulls = palloc_array(bool, scan->numberOfOrderBys);
     106           75 :         memset(scan->xs_orderbynulls, true, sizeof(bool) * scan->numberOfOrderBys);
     107              :     }
     108              : 
     109         3748 :     so->killedItems = NULL;      /* until needed */
     110         3748 :     so->numKilled = 0;
     111         3748 :     so->curBlkno = InvalidBlockNumber;
     112         3748 :     so->curPageLSN = InvalidXLogRecPtr;
     113              : 
     114         3748 :     scan->opaque = so;
     115              : 
     116              :     /*
     117              :      * All fields required for index-only scans are initialized in gistrescan,
     118              :      * as we don't know yet if we're doing an index-only scan or not.
     119              :      */
     120              : 
     121         3748 :     MemoryContextSwitchTo(oldCxt);
     122              : 
     123         3748 :     return scan;
     124              : }
     125              : 
     126              : void
     127         3808 : gistrescan(IndexScanDesc scan, ScanKey key, int nkeys,
     128              :            ScanKey orderbys, int norderbys)
     129              : {
     130              :     /* nkeys and norderbys arguments are ignored */
     131         3808 :     GISTScanOpaque so = (GISTScanOpaque) scan->opaque;
     132              :     bool        first_time;
     133              :     int         i;
     134              :     MemoryContext oldCxt;
     135              : 
     136              :     /* rescan an existing indexscan --- reset state */
     137              : 
     138              :     /*
     139              :      * The first time through, we create the search queue in the scanCxt.
     140              :      * Subsequent times through, we create the queue in a separate queueCxt,
     141              :      * which is created on the second call and reset on later calls.  Thus, in
     142              :      * the common case where a scan is only rescan'd once, we just put the
     143              :      * queue in scanCxt and don't pay the overhead of making a second memory
     144              :      * context.  If we do rescan more than once, the first queue is just left
     145              :      * for dead until end of scan; this small wastage seems worth the savings
     146              :      * in the common case.
     147              :      */
     148         3808 :     if (so->queue == NULL)
     149              :     {
     150              :         /* first time through */
     151              :         Assert(so->queueCxt == so->giststate->scanCxt);
     152         3748 :         first_time = true;
     153              :     }
     154           60 :     else if (so->queueCxt == so->giststate->scanCxt)
     155              :     {
     156              :         /* second time through */
     157           20 :         so->queueCxt = AllocSetContextCreate(so->giststate->scanCxt,
     158              :                                              "GiST queue context",
     159              :                                              ALLOCSET_DEFAULT_SIZES);
     160           20 :         first_time = false;
     161              :     }
     162              :     else
     163              :     {
     164              :         /* third or later time through */
     165           40 :         MemoryContextReset(so->queueCxt);
     166           40 :         first_time = false;
     167              :     }
     168              : 
     169              :     /*
     170              :      * If we're doing an index-only scan, on the first call, also initialize a
     171              :      * tuple descriptor to represent the returned index tuples and create a
     172              :      * memory context to hold them during the scan.
     173              :      */
     174         3808 :     if (scan->xs_want_itup && !scan->xs_hitupdesc)
     175              :     {
     176              :         int         natts;
     177              :         int         nkeyatts;
     178              :         int         attno;
     179              : 
     180              :         /*
     181              :          * The storage type of the index can be different from the original
     182              :          * datatype being indexed, so we cannot just grab the index's tuple
     183              :          * descriptor. Instead, construct a descriptor with the original data
     184              :          * types.
     185              :          */
     186          406 :         natts = RelationGetNumberOfAttributes(scan->indexRelation);
     187          406 :         nkeyatts = IndexRelationGetNumberOfKeyAttributes(scan->indexRelation);
     188          406 :         so->giststate->fetchTupdesc = CreateTemplateTupleDesc(natts);
     189          829 :         for (attno = 1; attno <= nkeyatts; attno++)
     190              :         {
     191          423 :             TupleDescInitEntry(so->giststate->fetchTupdesc, attno, NULL,
     192          423 :                                scan->indexRelation->rd_opcintype[attno - 1],
     193              :                                -1, 0);
     194              :         }
     195              : 
     196          406 :         for (; attno <= natts; attno++)
     197              :         {
     198              :             /* taking opcintype from giststate->tupdesc */
     199            0 :             TupleDescInitEntry(so->giststate->fetchTupdesc, attno, NULL,
     200            0 :                                TupleDescAttr(so->giststate->leafTupdesc,
     201              :                                              attno - 1)->atttypid,
     202              :                                -1, 0);
     203              :         }
     204          406 :         TupleDescFinalize(so->giststate->fetchTupdesc);
     205          406 :         scan->xs_hitupdesc = so->giststate->fetchTupdesc;
     206              : 
     207              :         /* Also create a memory context that will hold the returned tuples */
     208          406 :         so->pageDataCxt = AllocSetContextCreate(so->giststate->scanCxt,
     209              :                                                 "GiST page data context",
     210              :                                                 ALLOCSET_DEFAULT_SIZES);
     211              :     }
     212              : 
     213              :     /* create new, empty pairing heap for search queue */
     214         3808 :     oldCxt = MemoryContextSwitchTo(so->queueCxt);
     215         3808 :     so->queue = pairingheap_allocate(pairingheap_GISTSearchItem_cmp, scan);
     216         3808 :     MemoryContextSwitchTo(oldCxt);
     217              : 
     218         3808 :     so->firstCall = true;
     219              : 
     220              :     /* Update scan key, if a new one is given */
     221         3808 :     if (key && scan->numberOfKeys > 0)
     222              :     {
     223         3717 :         void      **fn_extras = NULL;
     224              : 
     225              :         /*
     226              :          * If this isn't the first time through, preserve the fn_extra
     227              :          * pointers, so that if the consistentFns are using them to cache
     228              :          * data, that data is not leaked across a rescan.
     229              :          */
     230         3717 :         if (!first_time)
     231              :         {
     232           20 :             fn_extras = (void **) palloc(scan->numberOfKeys * sizeof(void *));
     233           40 :             for (i = 0; i < scan->numberOfKeys; i++)
     234           20 :                 fn_extras[i] = scan->keyData[i].sk_func.fn_extra;
     235              :         }
     236              : 
     237         3717 :         memcpy(scan->keyData, key, scan->numberOfKeys * sizeof(ScanKeyData));
     238              : 
     239              :         /*
     240              :          * Modify the scan key so that the Consistent method is called for all
     241              :          * comparisons. The original operator is passed to the Consistent
     242              :          * function in the form of its strategy number, which is available
     243              :          * from the sk_strategy field, and its subtype from the sk_subtype
     244              :          * field.
     245              :          *
     246              :          * Next, if any of keys is a NULL and that key is not marked with
     247              :          * SK_SEARCHNULL/SK_SEARCHNOTNULL then nothing can be found (ie, we
     248              :          * assume all indexable operators are strict).
     249              :          */
     250         3717 :         so->qual_ok = true;
     251              : 
     252         9340 :         for (i = 0; i < scan->numberOfKeys; i++)
     253              :         {
     254         5623 :             ScanKey     skey = scan->keyData + i;
     255              : 
     256              :             /*
     257              :              * Copy consistent support function to ScanKey structure instead
     258              :              * of function implementing filtering operator.
     259              :              */
     260         5623 :             fmgr_info_copy(&(skey->sk_func),
     261         5623 :                            &(so->giststate->consistentFn[skey->sk_attno - 1]),
     262         5623 :                            so->giststate->scanCxt);
     263              : 
     264              :             /* Restore prior fn_extra pointers, if not first time */
     265         5623 :             if (!first_time)
     266           20 :                 skey->sk_func.fn_extra = fn_extras[i];
     267              : 
     268         5623 :             if (skey->sk_flags & SK_ISNULL)
     269              :             {
     270           17 :                 if (!(skey->sk_flags & (SK_SEARCHNULL | SK_SEARCHNOTNULL)))
     271            0 :                     so->qual_ok = false;
     272              :             }
     273              :         }
     274              : 
     275         3717 :         if (!first_time)
     276           20 :             pfree(fn_extras);
     277              :     }
     278              : 
     279              :     /* Update order-by key, if a new one is given */
     280         3808 :     if (orderbys && scan->numberOfOrderBys > 0)
     281              :     {
     282          123 :         void      **fn_extras = NULL;
     283              : 
     284              :         /* As above, preserve fn_extra if not first time through */
     285          123 :         if (!first_time)
     286              :         {
     287           48 :             fn_extras = (void **) palloc(scan->numberOfOrderBys * sizeof(void *));
     288           96 :             for (i = 0; i < scan->numberOfOrderBys; i++)
     289           48 :                 fn_extras[i] = scan->orderByData[i].sk_func.fn_extra;
     290              :         }
     291              : 
     292          123 :         memcpy(scan->orderByData, orderbys, scan->numberOfOrderBys * sizeof(ScanKeyData));
     293              : 
     294          123 :         so->orderByTypes = (Oid *) palloc(scan->numberOfOrderBys * sizeof(Oid));
     295              : 
     296              :         /*
     297              :          * Modify the order-by key so that the Distance method is called for
     298              :          * all comparisons. The original operator is passed to the Distance
     299              :          * function in the form of its strategy number, which is available
     300              :          * from the sk_strategy field, and its subtype from the sk_subtype
     301              :          * field.
     302              :          */
     303          246 :         for (i = 0; i < scan->numberOfOrderBys; i++)
     304              :         {
     305          123 :             ScanKey     skey = scan->orderByData + i;
     306          123 :             FmgrInfo   *finfo = &(so->giststate->distanceFn[skey->sk_attno - 1]);
     307              : 
     308              :             /* Check we actually have a distance function ... */
     309          123 :             if (!OidIsValid(finfo->fn_oid))
     310            0 :                 elog(ERROR, "missing support function %d for attribute %d of index \"%s\"",
     311              :                      GIST_DISTANCE_PROC, skey->sk_attno,
     312              :                      RelationGetRelationName(scan->indexRelation));
     313              : 
     314              :             /*
     315              :              * Look up the datatype returned by the original ordering
     316              :              * operator. GiST always uses a float8 for the distance function,
     317              :              * but the ordering operator could be anything else.
     318              :              *
     319              :              * XXX: The distance function is only allowed to be lossy if the
     320              :              * ordering operator's result type is float4 or float8.  Otherwise
     321              :              * we don't know how to return the distance to the executor.  But
     322              :              * we cannot check that here, as we won't know if the distance
     323              :              * function is lossy until it returns *recheck = true for the
     324              :              * first time.
     325              :              */
     326          123 :             so->orderByTypes[i] = get_func_rettype(skey->sk_func.fn_oid);
     327              : 
     328              :             /*
     329              :              * Copy distance support function to ScanKey structure instead of
     330              :              * function implementing ordering operator.
     331              :              */
     332          123 :             fmgr_info_copy(&(skey->sk_func), finfo, so->giststate->scanCxt);
     333              : 
     334              :             /* Restore prior fn_extra pointers, if not first time */
     335          123 :             if (!first_time)
     336           48 :                 skey->sk_func.fn_extra = fn_extras[i];
     337              :         }
     338              : 
     339          123 :         if (!first_time)
     340           48 :             pfree(fn_extras);
     341              :     }
     342              : 
     343              :     /* any previous xs_hitup will have been pfree'd in context resets above */
     344         3808 :     scan->xs_hitup = NULL;
     345         3808 : }
     346              : 
     347              : void
     348         3561 : gistendscan(IndexScanDesc scan)
     349              : {
     350         3561 :     GISTScanOpaque so = (GISTScanOpaque) scan->opaque;
     351              : 
     352              :     /*
     353              :      * freeGISTstate is enough to clean up everything made by gistbeginscan,
     354              :      * as well as the queueCxt if there is a separate context for it.
     355              :      */
     356         3561 :     freeGISTstate(so->giststate);
     357         3561 : }
        

Generated by: LCOV version 2.0-1