LCOV - code coverage report
Current view: top level - src/backend/utils/adt - tsquery_gist.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 26 116 22.4 %
Date: 2025-01-18 04:15:08 Functions: 2 10 20.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * tsquery_gist.c
       4             :  *    GiST index support for tsquery
       5             :  *
       6             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       7             :  *
       8             :  *
       9             :  * IDENTIFICATION
      10             :  *    src/backend/utils/adt/tsquery_gist.c
      11             :  *
      12             :  *-------------------------------------------------------------------------
      13             :  */
      14             : 
      15             : #include "postgres.h"
      16             : 
      17             : #include "access/gist.h"
      18             : #include "access/stratnum.h"
      19             : #include "common/int.h"
      20             : #include "tsearch/ts_utils.h"
      21             : #include "utils/fmgrprotos.h"
      22             : 
      23             : #define GETENTRY(vec,pos) DatumGetTSQuerySign((vec)->vector[pos].key)
      24             : 
      25             : 
      26             : Datum
      27          36 : gtsquery_compress(PG_FUNCTION_ARGS)
      28             : {
      29          36 :     GISTENTRY  *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
      30          36 :     GISTENTRY  *retval = entry;
      31             : 
      32          36 :     if (entry->leafkey)
      33             :     {
      34             :         TSQuerySign sign;
      35             : 
      36          36 :         retval = (GISTENTRY *) palloc(sizeof(GISTENTRY));
      37          36 :         sign = makeTSQuerySign(DatumGetTSQuery(entry->key));
      38             : 
      39          36 :         gistentryinit(*retval, TSQuerySignGetDatum(sign),
      40             :                       entry->rel, entry->page,
      41             :                       entry->offset, false);
      42             :     }
      43             : 
      44          36 :     PG_RETURN_POINTER(retval);
      45             : }
      46             : 
      47             : /*
      48             :  * We do not need a decompress function, because the other gtsquery
      49             :  * support functions work with the compressed representation.
      50             :  */
      51             : 
      52             : Datum
      53          96 : gtsquery_consistent(PG_FUNCTION_ARGS)
      54             : {
      55          96 :     GISTENTRY  *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
      56          96 :     TSQuery     query = PG_GETARG_TSQUERY(1);
      57          96 :     StrategyNumber strategy = (StrategyNumber) PG_GETARG_UINT16(2);
      58             : 
      59             :     /* Oid      subtype = PG_GETARG_OID(3); */
      60          96 :     bool       *recheck = (bool *) PG_GETARG_POINTER(4);
      61          96 :     TSQuerySign key = DatumGetTSQuerySign(entry->key);
      62          96 :     TSQuerySign sq = makeTSQuerySign(query);
      63             :     bool        retval;
      64             : 
      65             :     /* All cases served by this function are inexact */
      66          96 :     *recheck = true;
      67             : 
      68          96 :     switch (strategy)
      69             :     {
      70          48 :         case RTContainsStrategyNumber:
      71          48 :             if (GIST_LEAF(entry))
      72          48 :                 retval = (key & sq) == sq;
      73             :             else
      74           0 :                 retval = (key & sq) != 0;
      75          48 :             break;
      76          48 :         case RTContainedByStrategyNumber:
      77          48 :             if (GIST_LEAF(entry))
      78          48 :                 retval = (key & sq) == key;
      79             :             else
      80           0 :                 retval = (key & sq) != 0;
      81          48 :             break;
      82           0 :         default:
      83           0 :             retval = false;
      84             :     }
      85          96 :     PG_RETURN_BOOL(retval);
      86             : }
      87             : 
      88             : Datum
      89           0 : gtsquery_union(PG_FUNCTION_ARGS)
      90             : {
      91           0 :     GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
      92           0 :     int        *size = (int *) PG_GETARG_POINTER(1);
      93             :     TSQuerySign sign;
      94             :     int         i;
      95             : 
      96           0 :     sign = 0;
      97             : 
      98           0 :     for (i = 0; i < entryvec->n; i++)
      99           0 :         sign |= GETENTRY(entryvec, i);
     100             : 
     101           0 :     *size = sizeof(TSQuerySign);
     102             : 
     103           0 :     PG_RETURN_TSQUERYSIGN(sign);
     104             : }
     105             : 
     106             : Datum
     107           0 : gtsquery_same(PG_FUNCTION_ARGS)
     108             : {
     109           0 :     TSQuerySign a = PG_GETARG_TSQUERYSIGN(0);
     110           0 :     TSQuerySign b = PG_GETARG_TSQUERYSIGN(1);
     111           0 :     bool       *result = (bool *) PG_GETARG_POINTER(2);
     112             : 
     113           0 :     *result = (a == b);
     114             : 
     115           0 :     PG_RETURN_POINTER(result);
     116             : }
     117             : 
     118             : static int
     119           0 : sizebitvec(TSQuerySign sign)
     120             : {
     121           0 :     int         size = 0,
     122             :                 i;
     123             : 
     124           0 :     for (i = 0; i < TSQS_SIGLEN; i++)
     125           0 :         size += 0x01 & (sign >> i);
     126             : 
     127           0 :     return size;
     128             : }
     129             : 
     130             : static int
     131           0 : hemdist(TSQuerySign a, TSQuerySign b)
     132             : {
     133           0 :     TSQuerySign res = a ^ b;
     134             : 
     135           0 :     return sizebitvec(res);
     136             : }
     137             : 
     138             : Datum
     139           0 : gtsquery_penalty(PG_FUNCTION_ARGS)
     140             : {
     141           0 :     TSQuerySign origval = DatumGetTSQuerySign(((GISTENTRY *) PG_GETARG_POINTER(0))->key);
     142           0 :     TSQuerySign newval = DatumGetTSQuerySign(((GISTENTRY *) PG_GETARG_POINTER(1))->key);
     143           0 :     float      *penalty = (float *) PG_GETARG_POINTER(2);
     144             : 
     145           0 :     *penalty = hemdist(origval, newval);
     146             : 
     147           0 :     PG_RETURN_POINTER(penalty);
     148             : }
     149             : 
     150             : 
     151             : typedef struct
     152             : {
     153             :     OffsetNumber pos;
     154             :     int32       cost;
     155             : } SPLITCOST;
     156             : 
     157             : static int
     158           0 : comparecost(const void *a, const void *b)
     159             : {
     160           0 :     return pg_cmp_s32(((const SPLITCOST *) a)->cost,
     161             :                       ((const SPLITCOST *) b)->cost);
     162             : }
     163             : 
     164             : #define WISH_F(a,b,c) (double)( -(double)(((a)-(b))*((a)-(b))*((a)-(b)))*(c) )
     165             : 
     166             : Datum
     167           0 : gtsquery_picksplit(PG_FUNCTION_ARGS)
     168             : {
     169           0 :     GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
     170           0 :     GIST_SPLITVEC *v = (GIST_SPLITVEC *) PG_GETARG_POINTER(1);
     171           0 :     OffsetNumber maxoff = entryvec->n - 2;
     172             :     OffsetNumber k,
     173             :                 j;
     174             :     TSQuerySign datum_l,
     175             :                 datum_r;
     176             :     int32       size_alpha,
     177             :                 size_beta;
     178             :     int32       size_waste,
     179           0 :                 waste = -1;
     180             :     int32       nbytes;
     181           0 :     OffsetNumber seed_1 = 0,
     182           0 :                 seed_2 = 0;
     183             :     OffsetNumber *left,
     184             :                *right;
     185             : 
     186             :     SPLITCOST  *costvector;
     187             : 
     188           0 :     nbytes = (maxoff + 2) * sizeof(OffsetNumber);
     189           0 :     left = v->spl_left = (OffsetNumber *) palloc(nbytes);
     190           0 :     right = v->spl_right = (OffsetNumber *) palloc(nbytes);
     191           0 :     v->spl_nleft = v->spl_nright = 0;
     192             : 
     193           0 :     for (k = FirstOffsetNumber; k < maxoff; k = OffsetNumberNext(k))
     194           0 :         for (j = OffsetNumberNext(k); j <= maxoff; j = OffsetNumberNext(j))
     195             :         {
     196           0 :             size_waste = hemdist(GETENTRY(entryvec, j), GETENTRY(entryvec, k));
     197           0 :             if (size_waste > waste)
     198             :             {
     199           0 :                 waste = size_waste;
     200           0 :                 seed_1 = k;
     201           0 :                 seed_2 = j;
     202             :             }
     203             :         }
     204             : 
     205             : 
     206           0 :     if (seed_1 == 0 || seed_2 == 0)
     207             :     {
     208           0 :         seed_1 = 1;
     209           0 :         seed_2 = 2;
     210             :     }
     211             : 
     212           0 :     datum_l = GETENTRY(entryvec, seed_1);
     213           0 :     datum_r = GETENTRY(entryvec, seed_2);
     214             : 
     215           0 :     maxoff = OffsetNumberNext(maxoff);
     216           0 :     costvector = (SPLITCOST *) palloc(sizeof(SPLITCOST) * maxoff);
     217           0 :     for (j = FirstOffsetNumber; j <= maxoff; j = OffsetNumberNext(j))
     218             :     {
     219           0 :         costvector[j - 1].pos = j;
     220           0 :         size_alpha = hemdist(GETENTRY(entryvec, seed_1), GETENTRY(entryvec, j));
     221           0 :         size_beta = hemdist(GETENTRY(entryvec, seed_2), GETENTRY(entryvec, j));
     222           0 :         costvector[j - 1].cost = abs(size_alpha - size_beta);
     223             :     }
     224           0 :     qsort(costvector, maxoff, sizeof(SPLITCOST), comparecost);
     225             : 
     226           0 :     for (k = 0; k < maxoff; k++)
     227             :     {
     228           0 :         j = costvector[k].pos;
     229           0 :         if (j == seed_1)
     230             :         {
     231           0 :             *left++ = j;
     232           0 :             v->spl_nleft++;
     233           0 :             continue;
     234             :         }
     235           0 :         else if (j == seed_2)
     236             :         {
     237           0 :             *right++ = j;
     238           0 :             v->spl_nright++;
     239           0 :             continue;
     240             :         }
     241           0 :         size_alpha = hemdist(datum_l, GETENTRY(entryvec, j));
     242           0 :         size_beta = hemdist(datum_r, GETENTRY(entryvec, j));
     243             : 
     244           0 :         if (size_alpha < size_beta + WISH_F(v->spl_nleft, v->spl_nright, 0.05))
     245             :         {
     246           0 :             datum_l |= GETENTRY(entryvec, j);
     247           0 :             *left++ = j;
     248           0 :             v->spl_nleft++;
     249             :         }
     250             :         else
     251             :         {
     252           0 :             datum_r |= GETENTRY(entryvec, j);
     253           0 :             *right++ = j;
     254           0 :             v->spl_nright++;
     255             :         }
     256             :     }
     257             : 
     258           0 :     *right = *left = FirstOffsetNumber;
     259           0 :     v->spl_ldatum = TSQuerySignGetDatum(datum_l);
     260           0 :     v->spl_rdatum = TSQuerySignGetDatum(datum_r);
     261             : 
     262           0 :     PG_RETURN_POINTER(v);
     263             : }
     264             : 
     265             : /*
     266             :  * Formerly, gtsquery_consistent was declared in pg_proc.h with arguments
     267             :  * that did not match the documented conventions for GiST support functions.
     268             :  * We fixed that, but we still need a pg_proc entry with the old signature
     269             :  * to support reloading pre-9.6 contrib/tsearch2 opclass declarations.
     270             :  * This compatibility function should go away eventually.
     271             :  */
     272             : Datum
     273           0 : gtsquery_consistent_oldsig(PG_FUNCTION_ARGS)
     274             : {
     275           0 :     return gtsquery_consistent(fcinfo);
     276             : }

Generated by: LCOV version 1.14