LCOV - code coverage report
Current view: top level - src/test/modules/spgist_name_ops - spgist_name_ops.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 197 233 84.5 %
Date: 2025-01-18 05:15:39 Functions: 14 14 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*--------------------------------------------------------------------------
       2             :  *
       3             :  * spgist_name_ops.c
       4             :  *      Test opclass for SP-GiST
       5             :  *
       6             :  * This indexes input values of type "name", but the index storage is "text",
       7             :  * with the same choices as made in the core SP-GiST text_ops opclass.
       8             :  * Much of the code is identical to src/backend/access/spgist/spgtextproc.c,
       9             :  * which see for a more detailed header comment.
      10             :  *
      11             :  * Unlike spgtextproc.c, we don't bother with collation-aware logic.
      12             :  *
      13             :  *
      14             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
      15             :  * Portions Copyright (c) 1994, Regents of the University of California
      16             :  *
      17             :  * IDENTIFICATION
      18             :  *      src/test/modules/spgist_name_ops/spgist_name_ops.c
      19             :  *
      20             :  * -------------------------------------------------------------------------
      21             :  */
      22             : #include "postgres.h"
      23             : 
      24             : #include "access/spgist.h"
      25             : #include "catalog/pg_type.h"
      26             : #include "utils/datum.h"
      27             : #include "varatt.h"
      28             : 
      29           2 : PG_MODULE_MAGIC;
      30             : 
      31             : 
      32           4 : PG_FUNCTION_INFO_V1(spgist_name_config);
      33             : Datum
      34          14 : spgist_name_config(PG_FUNCTION_ARGS)
      35             : {
      36             :     /* spgConfigIn *cfgin = (spgConfigIn *) PG_GETARG_POINTER(0); */
      37          14 :     spgConfigOut *cfg = (spgConfigOut *) PG_GETARG_POINTER(1);
      38             : 
      39          14 :     cfg->prefixType = TEXTOID;
      40          14 :     cfg->labelType = INT2OID;
      41          14 :     cfg->leafType = TEXTOID;
      42          14 :     cfg->canReturnData = true;
      43          14 :     cfg->longValuesOK = true;    /* suffixing will shorten long values */
      44          14 :     PG_RETURN_VOID();
      45             : }
      46             : 
      47             : /*
      48             :  * Form a text datum from the given not-necessarily-null-terminated string,
      49             :  * using short varlena header format if possible
      50             :  */
      51             : static Datum
      52       35978 : formTextDatum(const char *data, int datalen)
      53             : {
      54             :     char       *p;
      55             : 
      56       35978 :     p = (char *) palloc(datalen + VARHDRSZ);
      57             : 
      58       35978 :     if (datalen + VARHDRSZ_SHORT <= VARATT_SHORT_MAX)
      59             :     {
      60       35978 :         SET_VARSIZE_SHORT(p, datalen + VARHDRSZ_SHORT);
      61       35978 :         if (datalen)
      62       35888 :             memcpy(p + VARHDRSZ_SHORT, data, datalen);
      63             :     }
      64             :     else
      65             :     {
      66           0 :         SET_VARSIZE(p, datalen + VARHDRSZ);
      67           0 :         memcpy(p + VARHDRSZ, data, datalen);
      68             :     }
      69             : 
      70       35978 :     return PointerGetDatum(p);
      71             : }
      72             : 
      73             : /*
      74             :  * Find the length of the common prefix of a and b
      75             :  */
      76             : static int
      77        2038 : commonPrefix(const char *a, const char *b, int lena, int lenb)
      78             : {
      79        2038 :     int         i = 0;
      80             : 
      81        4668 :     while (i < lena && i < lenb && *a == *b)
      82             :     {
      83        2630 :         a++;
      84        2630 :         b++;
      85        2630 :         i++;
      86             :     }
      87             : 
      88        2038 :     return i;
      89             : }
      90             : 
      91             : /*
      92             :  * Binary search an array of int16 datums for a match to c
      93             :  *
      94             :  * On success, *i gets the match location; on failure, it gets where to insert
      95             :  */
      96             : static bool
      97       22660 : searchChar(Datum *nodeLabels, int nNodes, int16 c, int *i)
      98             : {
      99       22660 :     int         StopLow = 0,
     100       22660 :                 StopHigh = nNodes;
     101             : 
     102       72826 :     while (StopLow < StopHigh)
     103             :     {
     104       72588 :         int         StopMiddle = (StopLow + StopHigh) >> 1;
     105       72588 :         int16       middle = DatumGetInt16(nodeLabels[StopMiddle]);
     106             : 
     107       72588 :         if (c < middle)
     108       29000 :             StopHigh = StopMiddle;
     109       43588 :         else if (c > middle)
     110       21166 :             StopLow = StopMiddle + 1;
     111             :         else
     112             :         {
     113       22422 :             *i = StopMiddle;
     114       22422 :             return true;
     115             :         }
     116             :     }
     117             : 
     118         238 :     *i = StopHigh;
     119         238 :     return false;
     120             : }
     121             : 
     122           4 : PG_FUNCTION_INFO_V1(spgist_name_choose);
     123             : Datum
     124       22672 : spgist_name_choose(PG_FUNCTION_ARGS)
     125             : {
     126       22672 :     spgChooseIn *in = (spgChooseIn *) PG_GETARG_POINTER(0);
     127       22672 :     spgChooseOut *out = (spgChooseOut *) PG_GETARG_POINTER(1);
     128       22672 :     Name        inName = DatumGetName(in->datum);
     129       22672 :     char       *inStr = NameStr(*inName);
     130       22672 :     int         inSize = strlen(inStr);
     131       22672 :     char       *prefixStr = NULL;
     132       22672 :     int         prefixSize = 0;
     133       22672 :     int         commonLen = 0;
     134       22672 :     int16       nodeChar = 0;
     135       22672 :     int         i = 0;
     136             : 
     137             :     /* Check for prefix match, set nodeChar to first byte after prefix */
     138       22672 :     if (in->hasPrefix)
     139             :     {
     140        2038 :         text       *prefixText = DatumGetTextPP(in->prefixDatum);
     141             : 
     142        2038 :         prefixStr = VARDATA_ANY(prefixText);
     143        2038 :         prefixSize = VARSIZE_ANY_EXHDR(prefixText);
     144             : 
     145        2038 :         commonLen = commonPrefix(inStr + in->level,
     146             :                                  prefixStr,
     147        2038 :                                  inSize - in->level,
     148             :                                  prefixSize);
     149             : 
     150        2038 :         if (commonLen == prefixSize)
     151             :         {
     152        2026 :             if (inSize - in->level > commonLen)
     153        2022 :                 nodeChar = *(unsigned char *) (inStr + in->level + commonLen);
     154             :             else
     155           4 :                 nodeChar = -1;
     156             :         }
     157             :         else
     158             :         {
     159             :             /* Must split tuple because incoming value doesn't match prefix */
     160          12 :             out->resultType = spgSplitTuple;
     161             : 
     162          12 :             if (commonLen == 0)
     163             :             {
     164           4 :                 out->result.splitTuple.prefixHasPrefix = false;
     165             :             }
     166             :             else
     167             :             {
     168           8 :                 out->result.splitTuple.prefixHasPrefix = true;
     169           8 :                 out->result.splitTuple.prefixPrefixDatum =
     170           8 :                     formTextDatum(prefixStr, commonLen);
     171             :             }
     172          12 :             out->result.splitTuple.prefixNNodes = 1;
     173          12 :             out->result.splitTuple.prefixNodeLabels =
     174          12 :                 (Datum *) palloc(sizeof(Datum));
     175          24 :             out->result.splitTuple.prefixNodeLabels[0] =
     176          12 :                 Int16GetDatum(*(unsigned char *) (prefixStr + commonLen));
     177             : 
     178          12 :             out->result.splitTuple.childNodeN = 0;
     179             : 
     180          12 :             if (prefixSize - commonLen == 1)
     181             :             {
     182           8 :                 out->result.splitTuple.postfixHasPrefix = false;
     183             :             }
     184             :             else
     185             :             {
     186           4 :                 out->result.splitTuple.postfixHasPrefix = true;
     187           4 :                 out->result.splitTuple.postfixPrefixDatum =
     188           4 :                     formTextDatum(prefixStr + commonLen + 1,
     189           4 :                                   prefixSize - commonLen - 1);
     190             :             }
     191             : 
     192          12 :             PG_RETURN_VOID();
     193             :         }
     194             :     }
     195       20634 :     else if (inSize > in->level)
     196             :     {
     197       20594 :         nodeChar = *(unsigned char *) (inStr + in->level);
     198             :     }
     199             :     else
     200             :     {
     201          40 :         nodeChar = -1;
     202             :     }
     203             : 
     204             :     /* Look up nodeChar in the node label array */
     205       22660 :     if (searchChar(in->nodeLabels, in->nNodes, nodeChar, &i))
     206             :     {
     207             :         /*
     208             :          * Descend to existing node.  (If in->allTheSame, the core code will
     209             :          * ignore our nodeN specification here, but that's OK.  We still have
     210             :          * to provide the correct levelAdd and restDatum values, and those are
     211             :          * the same regardless of which node gets chosen by core.)
     212             :          */
     213             :         int         levelAdd;
     214             : 
     215       22422 :         out->resultType = spgMatchNode;
     216       22422 :         out->result.matchNode.nodeN = i;
     217       22422 :         levelAdd = commonLen;
     218       22422 :         if (nodeChar >= 0)
     219       22378 :             levelAdd++;
     220       22422 :         out->result.matchNode.levelAdd = levelAdd;
     221       22422 :         if (inSize - in->level - levelAdd > 0)
     222       22332 :             out->result.matchNode.restDatum =
     223       22332 :                 formTextDatum(inStr + in->level + levelAdd,
     224       22332 :                               inSize - in->level - levelAdd);
     225             :         else
     226          90 :             out->result.matchNode.restDatum =
     227          90 :                 formTextDatum(NULL, 0);
     228             :     }
     229         238 :     else if (in->allTheSame)
     230             :     {
     231             :         /*
     232             :          * Can't use AddNode action, so split the tuple.  The upper tuple has
     233             :          * the same prefix as before and uses a dummy node label -2 for the
     234             :          * lower tuple.  The lower tuple has no prefix and the same node
     235             :          * labels as the original tuple.
     236             :          *
     237             :          * Note: it might seem tempting to shorten the upper tuple's prefix,
     238             :          * if it has one, then use its last byte as label for the lower tuple.
     239             :          * But that doesn't win since we know the incoming value matches the
     240             :          * whole prefix: we'd just end up splitting the lower tuple again.
     241             :          */
     242           0 :         out->resultType = spgSplitTuple;
     243           0 :         out->result.splitTuple.prefixHasPrefix = in->hasPrefix;
     244           0 :         out->result.splitTuple.prefixPrefixDatum = in->prefixDatum;
     245           0 :         out->result.splitTuple.prefixNNodes = 1;
     246           0 :         out->result.splitTuple.prefixNodeLabels = (Datum *) palloc(sizeof(Datum));
     247           0 :         out->result.splitTuple.prefixNodeLabels[0] = Int16GetDatum(-2);
     248           0 :         out->result.splitTuple.childNodeN = 0;
     249           0 :         out->result.splitTuple.postfixHasPrefix = false;
     250             :     }
     251             :     else
     252             :     {
     253             :         /* Add a node for the not-previously-seen nodeChar value */
     254         238 :         out->resultType = spgAddNode;
     255         238 :         out->result.addNode.nodeLabel = Int16GetDatum(nodeChar);
     256         238 :         out->result.addNode.nodeN = i;
     257             :     }
     258             : 
     259       22660 :     PG_RETURN_VOID();
     260             : }
     261             : 
     262             : /* The picksplit function is identical to the core opclass, so just use that */
     263             : 
     264           4 : PG_FUNCTION_INFO_V1(spgist_name_inner_consistent);
     265             : Datum
     266           8 : spgist_name_inner_consistent(PG_FUNCTION_ARGS)
     267             : {
     268           8 :     spgInnerConsistentIn *in = (spgInnerConsistentIn *) PG_GETARG_POINTER(0);
     269           8 :     spgInnerConsistentOut *out = (spgInnerConsistentOut *) PG_GETARG_POINTER(1);
     270             :     text       *reconstructedValue;
     271             :     text       *reconstrText;
     272             :     int         maxReconstrLen;
     273           8 :     text       *prefixText = NULL;
     274           8 :     int         prefixSize = 0;
     275             :     int         i;
     276             : 
     277             :     /*
     278             :      * Reconstruct values represented at this tuple, including parent data,
     279             :      * prefix of this tuple if any, and the node label if it's non-dummy.
     280             :      * in->level should be the length of the previously reconstructed value,
     281             :      * and the number of bytes added here is prefixSize or prefixSize + 1.
     282             :      *
     283             :      * Recall that reconstructedValues are assumed to be the same type as leaf
     284             :      * datums, so we must use "text" not "name" for them.
     285             :      *
     286             :      * Note: we assume that in->reconstructedValue isn't toasted and doesn't
     287             :      * have a short varlena header.  This is okay because it must have been
     288             :      * created by a previous invocation of this routine, and we always emit
     289             :      * long-format reconstructed values.
     290             :      */
     291           8 :     reconstructedValue = (text *) DatumGetPointer(in->reconstructedValue);
     292             :     Assert(reconstructedValue == NULL ? in->level == 0 :
     293             :            VARSIZE_ANY_EXHDR(reconstructedValue) == in->level);
     294             : 
     295           8 :     maxReconstrLen = in->level + 1;
     296           8 :     if (in->hasPrefix)
     297             :     {
     298           0 :         prefixText = DatumGetTextPP(in->prefixDatum);
     299           0 :         prefixSize = VARSIZE_ANY_EXHDR(prefixText);
     300           0 :         maxReconstrLen += prefixSize;
     301             :     }
     302             : 
     303           8 :     reconstrText = palloc(VARHDRSZ + maxReconstrLen);
     304           8 :     SET_VARSIZE(reconstrText, VARHDRSZ + maxReconstrLen);
     305             : 
     306           8 :     if (in->level)
     307           4 :         memcpy(VARDATA(reconstrText),
     308           4 :                VARDATA(reconstructedValue),
     309           4 :                in->level);
     310           8 :     if (prefixSize)
     311           0 :         memcpy(((char *) VARDATA(reconstrText)) + in->level,
     312           0 :                VARDATA_ANY(prefixText),
     313             :                prefixSize);
     314             :     /* last byte of reconstrText will be filled in below */
     315             : 
     316             :     /*
     317             :      * Scan the child nodes.  For each one, complete the reconstructed value
     318             :      * and see if it's consistent with the query.  If so, emit an entry into
     319             :      * the output arrays.
     320             :      */
     321           8 :     out->nodeNumbers = (int *) palloc(sizeof(int) * in->nNodes);
     322           8 :     out->levelAdds = (int *) palloc(sizeof(int) * in->nNodes);
     323           8 :     out->reconstructedValues = (Datum *) palloc(sizeof(Datum) * in->nNodes);
     324           8 :     out->nNodes = 0;
     325             : 
     326         140 :     for (i = 0; i < in->nNodes; i++)
     327             :     {
     328         132 :         int16       nodeChar = DatumGetInt16(in->nodeLabels[i]);
     329             :         int         thisLen;
     330         132 :         bool        res = true;
     331             :         int         j;
     332             : 
     333             :         /* If nodeChar is a dummy value, don't include it in data */
     334         132 :         if (nodeChar <= 0)
     335           0 :             thisLen = maxReconstrLen - 1;
     336             :         else
     337             :         {
     338         132 :             ((unsigned char *) VARDATA(reconstrText))[maxReconstrLen - 1] = nodeChar;
     339         132 :             thisLen = maxReconstrLen;
     340             :         }
     341             : 
     342         256 :         for (j = 0; j < in->nkeys; j++)
     343             :         {
     344         248 :             StrategyNumber strategy = in->scankeys[j].sk_strategy;
     345             :             Name        inName;
     346             :             char       *inStr;
     347             :             int         inSize;
     348             :             int         r;
     349             : 
     350         248 :             inName = DatumGetName(in->scankeys[j].sk_argument);
     351         248 :             inStr = NameStr(*inName);
     352         248 :             inSize = strlen(inStr);
     353             : 
     354         248 :             r = memcmp(VARDATA(reconstrText), inStr,
     355         248 :                        Min(inSize, thisLen));
     356             : 
     357         248 :             switch (strategy)
     358             :             {
     359         116 :                 case BTLessStrategyNumber:
     360             :                 case BTLessEqualStrategyNumber:
     361         116 :                     if (r > 0)
     362         108 :                         res = false;
     363         116 :                     break;
     364           0 :                 case BTEqualStrategyNumber:
     365           0 :                     if (r != 0 || inSize < thisLen)
     366           0 :                         res = false;
     367           0 :                     break;
     368         132 :                 case BTGreaterEqualStrategyNumber:
     369             :                 case BTGreaterStrategyNumber:
     370         132 :                     if (r < 0)
     371          16 :                         res = false;
     372         132 :                     break;
     373           0 :                 default:
     374           0 :                     elog(ERROR, "unrecognized strategy number: %d",
     375             :                          in->scankeys[j].sk_strategy);
     376             :                     break;
     377             :             }
     378             : 
     379         248 :             if (!res)
     380         124 :                 break;          /* no need to consider remaining conditions */
     381             :         }
     382             : 
     383         132 :         if (res)
     384             :         {
     385           8 :             out->nodeNumbers[out->nNodes] = i;
     386           8 :             out->levelAdds[out->nNodes] = thisLen - in->level;
     387           8 :             SET_VARSIZE(reconstrText, VARHDRSZ + thisLen);
     388          16 :             out->reconstructedValues[out->nNodes] =
     389           8 :                 datumCopy(PointerGetDatum(reconstrText), false, -1);
     390           8 :             out->nNodes++;
     391             :         }
     392             :     }
     393             : 
     394           8 :     PG_RETURN_VOID();
     395             : }
     396             : 
     397           4 : PG_FUNCTION_INFO_V1(spgist_name_leaf_consistent);
     398             : Datum
     399         248 : spgist_name_leaf_consistent(PG_FUNCTION_ARGS)
     400             : {
     401         248 :     spgLeafConsistentIn *in = (spgLeafConsistentIn *) PG_GETARG_POINTER(0);
     402         248 :     spgLeafConsistentOut *out = (spgLeafConsistentOut *) PG_GETARG_POINTER(1);
     403         248 :     int         level = in->level;
     404             :     text       *leafValue,
     405         248 :                *reconstrValue = NULL;
     406             :     char       *fullValue;
     407             :     int         fullLen;
     408             :     bool        res;
     409             :     int         j;
     410             : 
     411             :     /* all tests are exact */
     412         248 :     out->recheck = false;
     413             : 
     414         248 :     leafValue = DatumGetTextPP(in->leafDatum);
     415             : 
     416             :     /* As above, in->reconstructedValue isn't toasted or short. */
     417         248 :     if (DatumGetPointer(in->reconstructedValue))
     418         248 :         reconstrValue = (text *) DatumGetPointer(in->reconstructedValue);
     419             : 
     420             :     Assert(reconstrValue == NULL ? level == 0 :
     421             :            VARSIZE_ANY_EXHDR(reconstrValue) == level);
     422             : 
     423             :     /* Reconstruct the Name represented by this leaf tuple */
     424         248 :     fullValue = palloc0(NAMEDATALEN);
     425         248 :     fullLen = level + VARSIZE_ANY_EXHDR(leafValue);
     426             :     Assert(fullLen < NAMEDATALEN);
     427         248 :     if (VARSIZE_ANY_EXHDR(leafValue) == 0 && level > 0)
     428             :     {
     429           0 :         memcpy(fullValue, VARDATA(reconstrValue),
     430           0 :                VARSIZE_ANY_EXHDR(reconstrValue));
     431             :     }
     432             :     else
     433             :     {
     434         248 :         if (level)
     435         248 :             memcpy(fullValue, VARDATA(reconstrValue), level);
     436         248 :         if (VARSIZE_ANY_EXHDR(leafValue) > 0)
     437         248 :             memcpy(fullValue + level, VARDATA_ANY(leafValue),
     438         248 :                    VARSIZE_ANY_EXHDR(leafValue));
     439             :     }
     440         248 :     out->leafValue = PointerGetDatum(fullValue);
     441             : 
     442             :     /* Perform the required comparison(s) */
     443         248 :     res = true;
     444         516 :     for (j = 0; j < in->nkeys; j++)
     445             :     {
     446         464 :         StrategyNumber strategy = in->scankeys[j].sk_strategy;
     447         464 :         Name        queryName = DatumGetName(in->scankeys[j].sk_argument);
     448         464 :         char       *queryStr = NameStr(*queryName);
     449         464 :         int         queryLen = strlen(queryStr);
     450             :         int         r;
     451             : 
     452             :         /* Non-collation-aware comparison */
     453         464 :         r = memcmp(fullValue, queryStr, Min(queryLen, fullLen));
     454             : 
     455         464 :         if (r == 0)
     456             :         {
     457          52 :             if (queryLen > fullLen)
     458           0 :                 r = -1;
     459          52 :             else if (queryLen < fullLen)
     460          52 :                 r = 1;
     461             :         }
     462             : 
     463         464 :         switch (strategy)
     464             :         {
     465         216 :             case BTLessStrategyNumber:
     466         216 :                 res = (r < 0);
     467         216 :                 break;
     468           0 :             case BTLessEqualStrategyNumber:
     469           0 :                 res = (r <= 0);
     470           0 :                 break;
     471           0 :             case BTEqualStrategyNumber:
     472           0 :                 res = (r == 0);
     473           0 :                 break;
     474           0 :             case BTGreaterEqualStrategyNumber:
     475           0 :                 res = (r >= 0);
     476           0 :                 break;
     477         248 :             case BTGreaterStrategyNumber:
     478         248 :                 res = (r > 0);
     479         248 :                 break;
     480           0 :             default:
     481           0 :                 elog(ERROR, "unrecognized strategy number: %d",
     482             :                      in->scankeys[j].sk_strategy);
     483             :                 res = false;
     484             :                 break;
     485             :         }
     486             : 
     487         464 :         if (!res)
     488         196 :             break;              /* no need to consider remaining conditions */
     489             :     }
     490             : 
     491         248 :     PG_RETURN_BOOL(res);
     492             : }
     493             : 
     494           4 : PG_FUNCTION_INFO_V1(spgist_name_compress);
     495             : Datum
     496       13544 : spgist_name_compress(PG_FUNCTION_ARGS)
     497             : {
     498       13544 :     Name        inName = PG_GETARG_NAME(0);
     499       13544 :     char       *inStr = NameStr(*inName);
     500             : 
     501       13544 :     PG_RETURN_DATUM(formTextDatum(inStr, strlen(inStr)));
     502             : }

Generated by: LCOV version 1.14