LCOV - differential code coverage report
Current view: top level - src/backend/utils/adt - tsginidx.c (source / functions) Coverage Total Hit UBC GNC CBC EUB ECB DCB
Current: bed3ffbf9d952be6c7d739d068cdce44c046dfb7 vs 574581b50ac9c63dd9e4abebb731a3b67e5b50f6 Lines: 81.6 % 125 102 23 5 97 5
Current Date: 2026-05-05 10:23:31 +0900 Functions: 58.3 % 12 7 5 4 3
Baseline: lcov-20260505-025707-baseline Branches: 58.9 % 56 33 23 33 32 16
Baseline Date: 2026-05-05 10:27:06 +0900 Line coverage date bins:
Legend: Lines:     hit not hit
Branches: + taken - not taken # not executed
(30,360] days: 100.0 % 5 5 5
(360..) days: 80.8 % 120 97 23 97
Function coverage date bins:
(360..) days: 58.3 % 12 7 5 4 3
Branch coverage date bins:
(360..) days: 31.7 % 104 33 23 33 32 16

 Age         Owner                    Branch data    TLA  Line data    Source code
                                  1                 :                : /*-------------------------------------------------------------------------
                                  2                 :                :  *
                                  3                 :                :  * tsginidx.c
                                  4                 :                :  *   GIN support functions for tsvector_ops
                                  5                 :                :  *
                                  6                 :                :  * Portions Copyright (c) 1996-2026, PostgreSQL Global Development Group
                                  7                 :                :  *
                                  8                 :                :  *
                                  9                 :                :  * IDENTIFICATION
                                 10                 :                :  *    src/backend/utils/adt/tsginidx.c
                                 11                 :                :  *
                                 12                 :                :  *-------------------------------------------------------------------------
                                 13                 :                :  */
                                 14                 :                : #include "postgres.h"
                                 15                 :                : 
                                 16                 :                : #include "access/gin.h"
                                 17                 :                : #include "tsearch/ts_type.h"
                                 18                 :                : #include "tsearch/ts_utils.h"
                                 19                 :                : #include "utils/builtins.h"
                                 20                 :                : #include "varatt.h"
                                 21                 :                : 
                                 22                 :                : 
                                 23                 :                : Datum
 6563 tgl@sss.pgh.pa.us          24                 :CBC     1258175 : gin_cmp_tslexeme(PG_FUNCTION_ARGS)
                                 25                 :                : {
 6172 bruce@momjian.us           26                 :        1258175 :     text       *a = PG_GETARG_TEXT_PP(0);
                                 27                 :        1258175 :     text       *b = PG_GETARG_TEXT_PP(1);
                                 28                 :                :     int         cmp;
                                 29                 :                : 
 5595 tgl@sss.pgh.pa.us          30   [ -  +  -  -  :        1258175 :     cmp = tsCompareString(VARDATA_ANY(a), VARSIZE_ANY_EXHDR(a),
                                     -  -  -  -  +  
                                           +  +  + ]
 6172 bruce@momjian.us           31   [ -  +  -  -  :        1258175 :                           VARDATA_ANY(b), VARSIZE_ANY_EXHDR(b),
                                     -  -  -  -  +  
                                           +  +  + ]
                                 32                 :                :                           false);
                                 33                 :                : 
                                 34         [ -  + ]:        1258175 :     PG_FREE_IF_COPY(a, 0);
                                 35         [ -  + ]:        1258175 :     PG_FREE_IF_COPY(b, 1);
                                 36                 :        1258175 :     PG_RETURN_INT32(cmp);
                                 37                 :                : }
                                 38                 :                : 
                                 39                 :                : Datum
 6563 tgl@sss.pgh.pa.us          40                 :            296 : gin_cmp_prefix(PG_FUNCTION_ARGS)
                                 41                 :                : {
 6172 bruce@momjian.us           42                 :            296 :     text       *a = PG_GETARG_TEXT_PP(0);
                                 43                 :            296 :     text       *b = PG_GETARG_TEXT_PP(1);
                                 44                 :                : 
                                 45                 :                : #ifdef NOT_USED
                                 46                 :                :     StrategyNumber strategy = PG_GETARG_UINT16(2);
                                 47                 :                :     Pointer     extra_data = PG_GETARG_POINTER(3);
                                 48                 :                : #endif
                                 49                 :                :     int         cmp;
                                 50                 :                : 
 5595 tgl@sss.pgh.pa.us          51   [ -  +  -  -  :            296 :     cmp = tsCompareString(VARDATA_ANY(a), VARSIZE_ANY_EXHDR(a),
                                     -  -  -  -  -  
                                           +  -  + ]
 6172 bruce@momjian.us           52   [ -  +  -  -  :            296 :                           VARDATA_ANY(b), VARSIZE_ANY_EXHDR(b),
                                     -  -  -  -  +  
                                           -  +  - ]
                                 53                 :                :                           true);
                                 54                 :                : 
                                 55         [ +  + ]:            296 :     if (cmp < 0)
                                 56                 :              8 :         cmp = 1;                /* prevent continue scan */
                                 57                 :                : 
                                 58         [ -  + ]:            296 :     PG_FREE_IF_COPY(a, 0);
                                 59         [ -  + ]:            296 :     PG_FREE_IF_COPY(b, 1);
                                 60                 :            296 :     PG_RETURN_INT32(cmp);
                                 61                 :                : }
                                 62                 :                : 
                                 63                 :                : Datum
 6832 tgl@sss.pgh.pa.us          64                 :           2064 : gin_extract_tsvector(PG_FUNCTION_ARGS)
                                 65                 :                : {
                                 66                 :           2064 :     TSVector    vector = PG_GETARG_TSVECTOR(0);
 6815 teodor@sigaev.ru           67                 :           2064 :     int32      *nentries = (int32 *) PG_GETARG_POINTER(1);
 6832 tgl@sss.pgh.pa.us          68                 :           2064 :     Datum      *entries = NULL;
                                 69                 :                : 
 6811 teodor@sigaev.ru           70                 :           2064 :     *nentries = vector->size;
 6832 tgl@sss.pgh.pa.us          71         [ +  + ]:           2064 :     if (vector->size > 0)
                                 72                 :                :     {
                                 73                 :                :         int         i;
                                 74                 :           2028 :         WordEntry  *we = ARRPTR(vector);
                                 75                 :                : 
  146 michael@paquier.xyz        76                 :GNC        2028 :         entries = palloc_array(Datum, vector->size);
                                 77                 :                : 
 6832 tgl@sss.pgh.pa.us          78         [ +  + ]:CBC      117316 :         for (i = 0; i < vector->size; i++)
                                 79                 :                :         {
                                 80                 :                :             text       *txt;
                                 81                 :                : 
 6615                            82                 :         115288 :             txt = cstring_to_text_with_len(STRPTR(vector) + we->pos, we->len);
 6832                            83                 :         115288 :             entries[i] = PointerGetDatum(txt);
                                 84                 :                : 
                                 85                 :         115288 :             we++;
                                 86                 :                :         }
                                 87                 :                :     }
                                 88                 :                : 
                                 89         [ +  + ]:           2064 :     PG_FREE_IF_COPY(vector, 0);
                                 90                 :           2064 :     PG_RETURN_POINTER(entries);
                                 91                 :                : }
                                 92                 :                : 
                                 93                 :                : Datum
 6733                            94                 :            342 : gin_extract_tsquery(PG_FUNCTION_ARGS)
                                 95                 :                : {
 6832                            96                 :            342 :     TSQuery     query = PG_GETARG_TSQUERY(0);
 6815 teodor@sigaev.ru           97                 :            342 :     int32      *nentries = (int32 *) PG_GETARG_POINTER(1);
                                 98                 :                : #ifdef NOT_USED
                                 99                 :                :     StrategyNumber strategy = PG_GETARG_UINT16(2);
                                100                 :                : #endif
 6172 bruce@momjian.us          101                 :            342 :     bool      **ptr_partialmatch = (bool **) PG_GETARG_POINTER(3);
                                102                 :            342 :     Pointer   **extra_data = (Pointer **) PG_GETARG_POINTER(4);
                                103                 :                : #ifdef NOT_USED
                                104                 :                :     bool      **nullFlags = (bool **) PG_GETARG_POINTER(5);
                                105                 :                : #endif
 5595 tgl@sss.pgh.pa.us         106                 :            342 :     int32      *searchMode = (int32 *) PG_GETARG_POINTER(6);
 6832                           107                 :            342 :     Datum      *entries = NULL;
                                108                 :                : 
                                109                 :            342 :     *nentries = 0;
                                110                 :                : 
                                111         [ +  - ]:            342 :     if (query->size > 0)
                                112                 :                :     {
 5595                           113                 :            342 :         QueryItem  *item = GETQUERY(query);
                                114                 :                :         int32       i,
                                115                 :                :                     j;
                                116                 :                :         bool       *partialmatch;
                                117                 :                :         int        *map_item_operand;
                                118                 :                : 
                                119                 :                :         /*
                                120                 :                :          * If the query doesn't have any required positive matches (for
                                121                 :                :          * instance, it's something like '! foo'), we have to do a full index
                                122                 :                :          * scan.
                                123                 :                :          */
                                124         [ +  + ]:            342 :         if (tsquery_requires_match(item))
                                125                 :            251 :             *searchMode = GIN_SEARCH_MODE_DEFAULT;
                                126                 :                :         else
                                127                 :             91 :             *searchMode = GIN_SEARCH_MODE_ALL;
                                128                 :                : 
                                129                 :                :         /* count number of VAL items */
                                130                 :            342 :         j = 0;
 6832                           131         [ +  + ]:           1289 :         for (i = 0; i < query->size; i++)
                                132                 :                :         {
 6815 teodor@sigaev.ru          133         [ +  + ]:            947 :             if (item[i].type == QI_VAL)
 5595 tgl@sss.pgh.pa.us         134                 :            581 :                 j++;
                                135                 :                :         }
                                136                 :            342 :         *nentries = j;
                                137                 :                : 
  146 michael@paquier.xyz       138                 :GNC         342 :         entries = palloc_array(Datum, j);
                                139                 :            342 :         partialmatch = *ptr_partialmatch = palloc_array(bool, j);
                                140                 :                : 
                                141                 :                :         /*
                                142                 :                :          * Make map to convert item's number to corresponding operand's (the
                                143                 :                :          * same, entry's) number. Entry's number is used in check array in
                                144                 :                :          * consistent method. We use the same map for each entry.
                                145                 :                :          */
                                146                 :            342 :         *extra_data = palloc_array(Pointer, j);
                                147                 :            342 :         map_item_operand = palloc0_array(int, query->size);
                                148                 :                : 
                                149                 :                :         /* Now rescan the VAL items and fill in the arrays */
 5595 tgl@sss.pgh.pa.us         150                 :CBC         342 :         j = 0;
 6832                           151         [ +  + ]:           1289 :         for (i = 0; i < query->size; i++)
                                152                 :                :         {
 6815 teodor@sigaev.ru          153         [ +  + ]:            947 :             if (item[i].type == QI_VAL)
                                154                 :                :             {
 6137 peter_e@gmx.net           155                 :            581 :                 QueryOperand *val = &item[i].qoperand;
                                156                 :                :                 text       *txt;
                                157                 :                : 
 6615 tgl@sss.pgh.pa.us         158                 :            581 :                 txt = cstring_to_text_with_len(GETOPERAND(query) + val->distance,
                                159                 :            581 :                                                val->length);
 5595                           160                 :            581 :                 entries[j] = PointerGetDatum(txt);
                                161                 :            581 :                 partialmatch[j] = val->prefix;
 6172 bruce@momjian.us          162                 :            581 :                 (*extra_data)[j] = (Pointer) map_item_operand;
 6250 tgl@sss.pgh.pa.us         163                 :            581 :                 map_item_operand[i] = j;
 5595                           164                 :            581 :                 j++;
                                165                 :                :             }
                                166                 :                :         }
                                167                 :                :     }
                                168                 :                : 
 6832                           169         [ -  + ]:            342 :     PG_FREE_IF_COPY(query, 0);
                                170                 :                : 
                                171                 :            342 :     PG_RETURN_POINTER(entries);
                                172                 :                : }
                                173                 :                : 
                                174                 :                : typedef struct
                                175                 :                : {
                                176                 :                :     QueryItem  *first_item;
                                177                 :                :     GinTernaryValue *check;
                                178                 :                :     int        *map_item_operand;
                                179                 :                : } GinChkVal;
                                180                 :                : 
                                181                 :                : /*
                                182                 :                :  * TS_execute callback for matching a tsquery operand to GIN index data
                                183                 :                :  */
                                184                 :                : static TSTernaryValue
 2111                           185                 :          32244 : checkcondition_gin(void *checkval, QueryOperand *val, ExecPhraseData *data)
                                186                 :                : {
                                187                 :          32244 :     GinChkVal  *gcv = (GinChkVal *) checkval;
                                188                 :                :     int         j;
                                189                 :                :     GinTernaryValue result;
                                190                 :                : 
                                191                 :                :     /* convert item's number to corresponding entry's (operand's) number */
 6172 bruce@momjian.us          192                 :          32244 :     j = gcv->map_item_operand[((QueryItem *) val) - gcv->first_item];
                                193                 :                : 
                                194                 :                :     /* determine presence of current entry in indexed value */
 1904 tgl@sss.pgh.pa.us         195                 :          32244 :     result = gcv->check[j];
                                196                 :                : 
                                197                 :                :     /*
                                198                 :                :      * If any val requiring a weight is used or caller needs position
                                199                 :                :      * information then we must recheck, so replace TRUE with MAYBE.
                                200                 :                :      */
                                201         [ +  + ]:          32244 :     if (result == GIN_TRUE)
                                202                 :                :     {
 2111                           203   [ +  +  +  + ]:          10436 :         if (val->weight != 0 || data != NULL)
 1904                           204                 :           4332 :             result = GIN_MAYBE;
                                205                 :                :     }
                                206                 :                : 
                                207                 :                :     /*
                                208                 :                :      * We rely on GinTernaryValue and TSTernaryValue using equivalent value
                                209                 :                :      * assignments.  We could use a switch statement to map the values if that
                                210                 :                :      * ever stops being true, but it seems unlikely to happen.
                                211                 :                :      */
                                212                 :          32244 :     return (TSTernaryValue) result;
                                213                 :                : }
                                214                 :                : 
                                215                 :                : Datum
 6733                           216                 :             16 : gin_tsquery_consistent(PG_FUNCTION_ARGS)
                                217                 :                : {
 6832                           218                 :             16 :     bool       *check = (bool *) PG_GETARG_POINTER(0);
                                219                 :                : #ifdef NOT_USED
                                220                 :                :     StrategyNumber strategy = PG_GETARG_UINT16(1);
                                221                 :                : #endif
                                222                 :             16 :     TSQuery     query = PG_GETARG_TSQUERY(2);
                                223                 :                : #ifdef NOT_USED
                                224                 :                :     int32       nkeys = PG_GETARG_INT32(3);
                                225                 :                : #endif
 6172 bruce@momjian.us          226                 :             16 :     Pointer    *extra_data = (Pointer *) PG_GETARG_POINTER(4);
 6250 tgl@sss.pgh.pa.us         227                 :             16 :     bool       *recheck = (bool *) PG_GETARG_POINTER(5);
 3184 peter_e@gmx.net           228                 :             16 :     bool        res = false;
                                229                 :                : 
                                230                 :                :     /* Initially assume query doesn't require recheck */
 6595 tgl@sss.pgh.pa.us         231                 :             16 :     *recheck = false;
                                232                 :                : 
 6832                           233         [ +  - ]:             16 :     if (query->size > 0)
                                234                 :                :     {
                                235                 :                :         GinChkVal   gcv;
                                236                 :                : 
                                237                 :                :         /*
                                238                 :                :          * check-parameter array has one entry for each value (operand) in the
                                239                 :                :          * query.
                                240                 :                :          */
 3427                           241                 :             16 :         gcv.first_item = GETQUERY(query);
 2966 peter_e@gmx.net           242                 :             16 :         gcv.check = (GinTernaryValue *) check;
 6172 bruce@momjian.us          243                 :             16 :         gcv.map_item_operand = (int *) (extra_data[0]);
                                244                 :                : 
 1904 tgl@sss.pgh.pa.us         245   [ -  +  -  - ]:             16 :         switch (TS_execute_ternary(GETQUERY(query),
                                246                 :                :                                    &gcv,
                                247                 :                :                                    TS_EXEC_PHRASE_NO_POS,
                                248                 :                :                                    checkcondition_gin))
                                249                 :                :         {
 1904 tgl@sss.pgh.pa.us         250                 :UBC           0 :             case TS_NO:
                                251                 :              0 :                 res = false;
                                252                 :              0 :                 break;
 1904 tgl@sss.pgh.pa.us         253                 :CBC          16 :             case TS_YES:
                                254                 :             16 :                 res = true;
                                255                 :             16 :                 break;
 1904 tgl@sss.pgh.pa.us         256                 :UBC           0 :             case TS_MAYBE:
                                257                 :              0 :                 res = true;
                                258                 :              0 :                 *recheck = true;
                                259                 :              0 :                 break;
                                260                 :                :         }
                                261                 :                :     }
                                262                 :                : 
 6832 tgl@sss.pgh.pa.us         263                 :CBC          16 :     PG_RETURN_BOOL(res);
                                264                 :                : }
                                265                 :                : 
                                266                 :                : Datum
 4437 heikki.linnakangas@i      267                 :          24612 : gin_tsquery_triconsistent(PG_FUNCTION_ARGS)
                                268                 :                : {
 4418                           269                 :          24612 :     GinTernaryValue *check = (GinTernaryValue *) PG_GETARG_POINTER(0);
                                270                 :                : #ifdef NOT_USED
                                271                 :                :     StrategyNumber strategy = PG_GETARG_UINT16(1);
                                272                 :                : #endif
 4437                           273                 :          24612 :     TSQuery     query = PG_GETARG_TSQUERY(2);
                                274                 :                : #ifdef NOT_USED
                                275                 :                :     int32       nkeys = PG_GETARG_INT32(3);
                                276                 :                : #endif
                                277                 :          24612 :     Pointer    *extra_data = (Pointer *) PG_GETARG_POINTER(4);
 4418                           278                 :          24612 :     GinTernaryValue res = GIN_FALSE;
                                279                 :                : 
 4437                           280         [ +  - ]:          24612 :     if (query->size > 0)
                                281                 :                :     {
                                282                 :                :         GinChkVal   gcv;
                                283                 :                : 
                                284                 :                :         /*
                                285                 :                :          * check-parameter array has one entry for each value (operand) in the
                                286                 :                :          * query.
                                287                 :                :          */
 3427 tgl@sss.pgh.pa.us         288                 :          24612 :         gcv.first_item = GETQUERY(query);
 4437 heikki.linnakangas@i      289                 :          24612 :         gcv.check = check;
                                290                 :          24612 :         gcv.map_item_operand = (int *) (extra_data[0]);
                                291                 :                : 
 1904 tgl@sss.pgh.pa.us         292                 :          24612 :         res = TS_execute_ternary(GETQUERY(query),
                                293                 :                :                                  &gcv,
                                294                 :                :                                  TS_EXEC_PHRASE_NO_POS,
                                295                 :                :                                  checkcondition_gin);
                                296                 :                :     }
                                297                 :                : 
 4418 heikki.linnakangas@i      298                 :          24612 :     PG_RETURN_GIN_TERNARY_VALUE(res);
                                299                 :                : }
                                300                 :                : 
                                301                 :                : /*
                                302                 :                :  * Formerly, gin_extract_tsvector had only two arguments.  Now it has three,
                                303                 :                :  * but we still need a pg_proc entry with two args to support reloading
                                304                 :                :  * pre-9.1 contrib/tsearch2 opclass declarations.  This compatibility
                                305                 :                :  * function should go away eventually.  (Note: you might say "hey, but the
                                306                 :                :  * code above is only *using* two args, so let's just declare it that way".
                                307                 :                :  * If you try that you'll find the opr_sanity regression test complains.)
                                308                 :                :  */
                                309                 :                : Datum
 5557 tgl@sss.pgh.pa.us         310                 :UBC           0 : gin_extract_tsvector_2args(PG_FUNCTION_ARGS)
                                311                 :                : {
                                312         [ #  # ]:              0 :     if (PG_NARGS() < 3)          /* should not happen */
                                313         [ #  # ]:              0 :         elog(ERROR, "gin_extract_tsvector requires three arguments");
                                314                 :              0 :     return gin_extract_tsvector(fcinfo);
                                315                 :                : }
                                316                 :                : 
                                317                 :                : /*
                                318                 :                :  * Likewise, we need a stub version of gin_extract_tsquery declared with
                                319                 :                :  * only five arguments.
                                320                 :                :  */
                                321                 :                : Datum
                                322                 :              0 : gin_extract_tsquery_5args(PG_FUNCTION_ARGS)
                                323                 :                : {
                                324         [ #  # ]:              0 :     if (PG_NARGS() < 7)          /* should not happen */
                                325         [ #  # ]:              0 :         elog(ERROR, "gin_extract_tsquery requires seven arguments");
                                326                 :              0 :     return gin_extract_tsquery(fcinfo);
                                327                 :                : }
                                328                 :                : 
                                329                 :                : /*
                                330                 :                :  * Likewise, we need a stub version of gin_tsquery_consistent declared with
                                331                 :                :  * only six arguments.
                                332                 :                :  */
                                333                 :                : Datum
                                334                 :              0 : gin_tsquery_consistent_6args(PG_FUNCTION_ARGS)
                                335                 :                : {
                                336         [ #  # ]:              0 :     if (PG_NARGS() < 8)          /* should not happen */
                                337         [ #  # ]:              0 :         elog(ERROR, "gin_tsquery_consistent requires eight arguments");
                                338                 :              0 :     return gin_tsquery_consistent(fcinfo);
                                339                 :                : }
                                340                 :                : 
                                341                 :                : /*
                                342                 :                :  * Likewise, a stub version of gin_extract_tsquery declared with argument
                                343                 :                :  * types that are no longer considered appropriate.
                                344                 :                :  */
                                345                 :                : Datum
 3716                           346                 :              0 : gin_extract_tsquery_oldsig(PG_FUNCTION_ARGS)
                                347                 :                : {
                                348                 :              0 :     return gin_extract_tsquery(fcinfo);
                                349                 :                : }
                                350                 :                : 
                                351                 :                : /*
                                352                 :                :  * Likewise, a stub version of gin_tsquery_consistent declared with argument
                                353                 :                :  * types that are no longer considered appropriate.
                                354                 :                :  */
                                355                 :                : Datum
                                356                 :              0 : gin_tsquery_consistent_oldsig(PG_FUNCTION_ARGS)
                                357                 :                : {
                                358                 :              0 :     return gin_tsquery_consistent(fcinfo);
                                359                 :                : }
        

Generated by: LCOV version 2.5.0-beta