LCOV - code coverage report
Current view: top level - src/backend/access/gist - gistscan.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 105 110 95.5 %
Date: 2025-01-18 04:15:08 Functions: 4 4 100.0 %
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-2025, 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      365734 : pairingheap_GISTSearchItem_cmp(const pairingheap_node *a, const pairingheap_node *b, void *arg)
      31             : {
      32      365734 :     const GISTSearchItem *sa = (const GISTSearchItem *) a;
      33      365734 :     const GISTSearchItem *sb = (const GISTSearchItem *) b;
      34      365734 :     IndexScanDesc scan = (IndexScanDesc) arg;
      35             :     int         i;
      36             : 
      37             :     /* Order according to distance comparison */
      38      369684 :     for (i = 0; i < scan->numberOfOrderBys; i++)
      39             :     {
      40       67662 :         if (sa->distances[i].isnull)
      41             :         {
      42         108 :             if (!sb->distances[i].isnull)
      43         108 :                 return -1;
      44             :         }
      45       67554 :         else if (sb->distances[i].isnull)
      46             :         {
      47          14 :             return 1;
      48             :         }
      49             :         else
      50             :         {
      51       67540 :             int         cmp = -float8_cmp_internal(sa->distances[i].value,
      52             :                                                    sb->distances[i].value);
      53             : 
      54       67540 :             if (cmp != 0)
      55       63590 :                 return cmp;
      56             :         }
      57             :     }
      58             : 
      59             :     /* Heap items go before inner pages, to ensure a depth-first search */
      60      302022 :     if (GISTSearchItemIsHeap(*sa) && !GISTSearchItemIsHeap(*sb))
      61           0 :         return 1;
      62      302022 :     if (!GISTSearchItemIsHeap(*sa) && GISTSearchItemIsHeap(*sb))
      63           4 :         return -1;
      64             : 
      65      302018 :     return 0;
      66             : }
      67             : 
      68             : 
      69             : /*
      70             :  * Index AM API functions for scanning GiST indexes
      71             :  */
      72             : 
      73             : IndexScanDesc
      74        6388 : gistbeginscan(Relation r, int nkeys, int norderbys)
      75             : {
      76             :     IndexScanDesc scan;
      77             :     GISTSTATE  *giststate;
      78             :     GISTScanOpaque so;
      79             :     MemoryContext oldCxt;
      80             : 
      81        6388 :     scan = RelationGetIndexScan(r, nkeys, norderbys);
      82             : 
      83             :     /* First, set up a GISTSTATE with a scan-lifespan memory context */
      84        6388 :     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        6388 :     oldCxt = MemoryContextSwitchTo(giststate->scanCxt);
      91             : 
      92             :     /* initialize opaque data */
      93        6388 :     so = (GISTScanOpaque) palloc0(sizeof(GISTScanOpaqueData));
      94        6388 :     so->giststate = giststate;
      95        6388 :     giststate->tempCxt = createTempGistContext();
      96        6388 :     so->queue = NULL;
      97        6388 :     so->queueCxt = giststate->scanCxt;    /* see gistrescan */
      98             : 
      99             :     /* workspaces with size dependent on numberOfOrderBys: */
     100        6388 :     so->distances = palloc(sizeof(so->distances[0]) * scan->numberOfOrderBys);
     101        6388 :     so->qual_ok = true;          /* in case there are zero keys */
     102        6388 :     if (scan->numberOfOrderBys > 0)
     103             :     {
     104         126 :         scan->xs_orderbyvals = palloc0(sizeof(Datum) * scan->numberOfOrderBys);
     105         126 :         scan->xs_orderbynulls = palloc(sizeof(bool) * scan->numberOfOrderBys);
     106         126 :         memset(scan->xs_orderbynulls, true, sizeof(bool) * scan->numberOfOrderBys);
     107             :     }
     108             : 
     109        6388 :     so->killedItems = NULL;      /* until needed */
     110        6388 :     so->numKilled = 0;
     111        6388 :     so->curBlkno = InvalidBlockNumber;
     112        6388 :     so->curPageLSN = InvalidXLogRecPtr;
     113             : 
     114        6388 :     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        6388 :     MemoryContextSwitchTo(oldCxt);
     122             : 
     123        6388 :     return scan;
     124             : }
     125             : 
     126             : void
     127        6478 : gistrescan(IndexScanDesc scan, ScanKey key, int nkeys,
     128             :            ScanKey orderbys, int norderbys)
     129             : {
     130             :     /* nkeys and norderbys arguments are ignored */
     131        6478 :     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        6478 :     if (so->queue == NULL)
     149             :     {
     150             :         /* first time through */
     151             :         Assert(so->queueCxt == so->giststate->scanCxt);
     152        6388 :         first_time = true;
     153             :     }
     154          90 :     else if (so->queueCxt == so->giststate->scanCxt)
     155             :     {
     156             :         /* second time through */
     157          30 :         so->queueCxt = AllocSetContextCreate(so->giststate->scanCxt,
     158             :                                              "GiST queue context",
     159             :                                              ALLOCSET_DEFAULT_SIZES);
     160          30 :         first_time = false;
     161             :     }
     162             :     else
     163             :     {
     164             :         /* third or later time through */
     165          60 :         MemoryContextReset(so->queueCxt);
     166          60 :         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        6478 :     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         668 :         natts = RelationGetNumberOfAttributes(scan->indexRelation);
     187         668 :         nkeyatts = IndexRelationGetNumberOfKeyAttributes(scan->indexRelation);
     188         668 :         so->giststate->fetchTupdesc = CreateTemplateTupleDesc(natts);
     189        1362 :         for (attno = 1; attno <= nkeyatts; attno++)
     190             :         {
     191         694 :             TupleDescInitEntry(so->giststate->fetchTupdesc, attno, NULL,
     192         694 :                                scan->indexRelation->rd_opcintype[attno - 1],
     193             :                                -1, 0);
     194             :         }
     195             : 
     196         668 :         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         668 :         scan->xs_hitupdesc = so->giststate->fetchTupdesc;
     205             : 
     206             :         /* Also create a memory context that will hold the returned tuples */
     207         668 :         so->pageDataCxt = AllocSetContextCreate(so->giststate->scanCxt,
     208             :                                                 "GiST page data context",
     209             :                                                 ALLOCSET_DEFAULT_SIZES);
     210             :     }
     211             : 
     212             :     /* create new, empty pairing heap for search queue */
     213        6478 :     oldCxt = MemoryContextSwitchTo(so->queueCxt);
     214        6478 :     so->queue = pairingheap_allocate(pairingheap_GISTSearchItem_cmp, scan);
     215        6478 :     MemoryContextSwitchTo(oldCxt);
     216             : 
     217        6478 :     so->firstCall = true;
     218             : 
     219             :     /* Update scan key, if a new one is given */
     220        6478 :     if (key && scan->numberOfKeys > 0)
     221             :     {
     222        6328 :         void      **fn_extras = NULL;
     223             : 
     224             :         /*
     225             :          * If this isn't the first time through, preserve the fn_extra
     226             :          * pointers, so that if the consistentFns are using them to cache
     227             :          * data, that data is not leaked across a rescan.
     228             :          */
     229        6328 :         if (!first_time)
     230             :         {
     231          30 :             fn_extras = (void **) palloc(scan->numberOfKeys * sizeof(void *));
     232          60 :             for (i = 0; i < scan->numberOfKeys; i++)
     233          30 :                 fn_extras[i] = scan->keyData[i].sk_func.fn_extra;
     234             :         }
     235             : 
     236        6328 :         memcpy(scan->keyData, key, scan->numberOfKeys * sizeof(ScanKeyData));
     237             : 
     238             :         /*
     239             :          * Modify the scan key so that the Consistent method is called for all
     240             :          * comparisons. The original operator is passed to the Consistent
     241             :          * function in the form of its strategy number, which is available
     242             :          * from the sk_strategy field, and its subtype from the sk_subtype
     243             :          * field.
     244             :          *
     245             :          * Next, if any of keys is a NULL and that key is not marked with
     246             :          * SK_SEARCHNULL/SK_SEARCHNOTNULL then nothing can be found (ie, we
     247             :          * assume all indexable operators are strict).
     248             :          */
     249        6328 :         so->qual_ok = true;
     250             : 
     251       15878 :         for (i = 0; i < scan->numberOfKeys; i++)
     252             :         {
     253        9550 :             ScanKey     skey = scan->keyData + i;
     254             : 
     255             :             /*
     256             :              * Copy consistent support function to ScanKey structure instead
     257             :              * of function implementing filtering operator.
     258             :              */
     259        9550 :             fmgr_info_copy(&(skey->sk_func),
     260        9550 :                            &(so->giststate->consistentFn[skey->sk_attno - 1]),
     261        9550 :                            so->giststate->scanCxt);
     262             : 
     263             :             /* Restore prior fn_extra pointers, if not first time */
     264        9550 :             if (!first_time)
     265          30 :                 skey->sk_func.fn_extra = fn_extras[i];
     266             : 
     267        9550 :             if (skey->sk_flags & SK_ISNULL)
     268             :             {
     269          26 :                 if (!(skey->sk_flags & (SK_SEARCHNULL | SK_SEARCHNOTNULL)))
     270           0 :                     so->qual_ok = false;
     271             :             }
     272             :         }
     273             : 
     274        6328 :         if (!first_time)
     275          30 :             pfree(fn_extras);
     276             :     }
     277             : 
     278             :     /* Update order-by key, if a new one is given */
     279        6478 :     if (orderbys && scan->numberOfOrderBys > 0)
     280             :     {
     281         198 :         void      **fn_extras = NULL;
     282             : 
     283             :         /* As above, preserve fn_extra if not first time through */
     284         198 :         if (!first_time)
     285             :         {
     286          72 :             fn_extras = (void **) palloc(scan->numberOfOrderBys * sizeof(void *));
     287         144 :             for (i = 0; i < scan->numberOfOrderBys; i++)
     288          72 :                 fn_extras[i] = scan->orderByData[i].sk_func.fn_extra;
     289             :         }
     290             : 
     291         198 :         memcpy(scan->orderByData, orderbys, scan->numberOfOrderBys * sizeof(ScanKeyData));
     292             : 
     293         198 :         so->orderByTypes = (Oid *) palloc(scan->numberOfOrderBys * sizeof(Oid));
     294             : 
     295             :         /*
     296             :          * Modify the order-by key so that the Distance method is called for
     297             :          * all comparisons. The original operator is passed to the Distance
     298             :          * function in the form of its strategy number, which is available
     299             :          * from the sk_strategy field, and its subtype from the sk_subtype
     300             :          * field.
     301             :          */
     302         396 :         for (i = 0; i < scan->numberOfOrderBys; i++)
     303             :         {
     304         198 :             ScanKey     skey = scan->orderByData + i;
     305         198 :             FmgrInfo   *finfo = &(so->giststate->distanceFn[skey->sk_attno - 1]);
     306             : 
     307             :             /* Check we actually have a distance function ... */
     308         198 :             if (!OidIsValid(finfo->fn_oid))
     309           0 :                 elog(ERROR, "missing support function %d for attribute %d of index \"%s\"",
     310             :                      GIST_DISTANCE_PROC, skey->sk_attno,
     311             :                      RelationGetRelationName(scan->indexRelation));
     312             : 
     313             :             /*
     314             :              * Look up the datatype returned by the original ordering
     315             :              * operator. GiST always uses a float8 for the distance function,
     316             :              * but the ordering operator could be anything else.
     317             :              *
     318             :              * XXX: The distance function is only allowed to be lossy if the
     319             :              * ordering operator's result type is float4 or float8.  Otherwise
     320             :              * we don't know how to return the distance to the executor.  But
     321             :              * we cannot check that here, as we won't know if the distance
     322             :              * function is lossy until it returns *recheck = true for the
     323             :              * first time.
     324             :              */
     325         198 :             so->orderByTypes[i] = get_func_rettype(skey->sk_func.fn_oid);
     326             : 
     327             :             /*
     328             :              * Copy distance support function to ScanKey structure instead of
     329             :              * function implementing ordering operator.
     330             :              */
     331         198 :             fmgr_info_copy(&(skey->sk_func), finfo, so->giststate->scanCxt);
     332             : 
     333             :             /* Restore prior fn_extra pointers, if not first time */
     334         198 :             if (!first_time)
     335          72 :                 skey->sk_func.fn_extra = fn_extras[i];
     336             :         }
     337             : 
     338         198 :         if (!first_time)
     339          72 :             pfree(fn_extras);
     340             :     }
     341             : 
     342             :     /* any previous xs_hitup will have been pfree'd in context resets above */
     343        6478 :     scan->xs_hitup = NULL;
     344        6478 : }
     345             : 
     346             : void
     347        6076 : gistendscan(IndexScanDesc scan)
     348             : {
     349        6076 :     GISTScanOpaque so = (GISTScanOpaque) scan->opaque;
     350             : 
     351             :     /*
     352             :      * freeGISTstate is enough to clean up everything made by gistbeginscan,
     353             :      * as well as the queueCxt if there is a separate context for it.
     354             :      */
     355        6076 :     freeGISTstate(so->giststate);
     356        6076 : }

Generated by: LCOV version 1.14