LCOV - differential code coverage report
Current view: top level - contrib/ltree - _ltree_gist.c (source / functions) Coverage Total Hit UNC UBC GNC CBC DUB DCB
Current: 380a8b2ea024c33a35e7abc8628e7c4f52f9f9f9 vs db5ed03217b9c238703df8b4b286115d6e940488 Lines: 88.6 % 271 240 1 30 3 237 1 3
Current Date: 2026-05-29 21:51:00 -0400 Functions: 96.0 % 25 24 1 3 21
Baseline: lcov-20260530-034037-baseline Branches: 63.9 % 147 94 1 52 1 93 1 1
Baseline Date: 2026-05-29 14:39:03 -0700 Line coverage date bins:
Legend: Lines:     hit not hit
Branches: + taken - not taken # not executed
(30,360] days: 75.0 % 4 3 1 3
(360..) days: 88.8 % 267 237 30 237
Function coverage date bins:
(360..) days: 96.0 % 25 24 1 3 21
Branch coverage date bins:
(30,360] days: 50.0 % 2 1 1 1
(360..) days: 64.1 % 145 93 52 93

 Age         Owner                    Branch data    TLA  Line data    Source code
                                  1                 :                : /*
                                  2                 :                :  * contrib/ltree/_ltree_gist.c
                                  3                 :                :  *
                                  4                 :                :  *
                                  5                 :                :  * GiST support for ltree[]
                                  6                 :                :  * Teodor Sigaev <teodor@stack.net>
                                  7                 :                :  */
                                  8                 :                : #include "postgres.h"
                                  9                 :                : 
                                 10                 :                : #include "access/gist.h"
                                 11                 :                : #include "access/reloptions.h"
                                 12                 :                : #include "access/stratnum.h"
                                 13                 :                : #include "crc32.h"
                                 14                 :                : #include "ltree.h"
                                 15                 :                : #include "port/pg_bitutils.h"
                                 16                 :                : #include "utils/array.h"
                                 17                 :                : 
 8669 bruce@momjian.us           18                 :CBC           3 : PG_FUNCTION_INFO_V1(_ltree_compress);
                                 19                 :              3 : PG_FUNCTION_INFO_V1(_ltree_same);
                                 20                 :              3 : PG_FUNCTION_INFO_V1(_ltree_union);
                                 21                 :              3 : PG_FUNCTION_INFO_V1(_ltree_penalty);
                                 22                 :              3 : PG_FUNCTION_INFO_V1(_ltree_picksplit);
                                 23                 :              3 : PG_FUNCTION_INFO_V1(_ltree_consistent);
 2252 akorotkov@postgresql       24                 :              3 : PG_FUNCTION_INFO_V1(_ltree_gist_options);
                                 25                 :                : 
                                 26                 :                : #define GETENTRY(vec,pos) ((ltree_gist *) DatumGetPointer((vec)->vector[(pos)].key))
                                 27                 :                : #define NEXTVAL(x) ( (ltree*)( (char*)(x) + INTALIGN( VARSIZE(x) ) ) )
                                 28                 :                : 
                                 29                 :                : #define WISH_F(a,b,c) (double)( -(double)(((a)-(b))*((a)-(b))*((a)-(b)))*(c) )
                                 30                 :                : 
                                 31                 :                : 
                                 32                 :                : static void
                                 33                 :           7058 : hashing(BITVECP sign, ltree *t, int siglen)
                                 34                 :                : {
 8669 bruce@momjian.us           35                 :           7058 :     int         tlen = t->numlevel;
 8705                            36                 :           7058 :     ltree_level *cur = LTREE_FIRST(t);
                                 37                 :                :     int         hash;
                                 38                 :                : 
 8669                            39         [ +  + ]:          53466 :     while (tlen > 0)
                                 40                 :                :     {
                                 41                 :          46408 :         hash = ltree_crc32_sz(cur->name, cur->len);
 2252 akorotkov@postgresql       42                 :          46408 :         AHASH(sign, hash, siglen);
 8705 bruce@momjian.us           43                 :          46408 :         cur = LEVEL_NEXT(cur);
                                 44                 :          46408 :         tlen--;
                                 45                 :                :     }
                                 46                 :           7058 : }
                                 47                 :                : 
                                 48                 :                : Datum
 8669                            49                 :           5888 : _ltree_compress(PG_FUNCTION_ARGS)
                                 50                 :                : {
                                 51                 :           5888 :     GISTENTRY  *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
 8705                            52                 :           5888 :     GISTENTRY  *retval = entry;
 2252 akorotkov@postgresql       53         [ +  - ]:           5888 :     int         siglen = LTREE_GET_ASIGLEN();
                                 54                 :                : 
 8669 bruce@momjian.us           55         [ +  + ]:           5888 :     if (entry->leafkey)
                                 56                 :                :     {                           /* ltree */
                                 57                 :                :         ltree_gist *key;
                                 58                 :           2000 :         ArrayType  *val = DatumGetArrayTypeP(entry->key);
                                 59                 :           2000 :         int         num = ArrayGetNItems(ARR_NDIM(val), ARR_DIMS(val));
                                 60         [ -  + ]:           2000 :         ltree      *item = (ltree *) ARR_DATA_PTR(val);
                                 61                 :                : 
 5939 tgl@sss.pgh.pa.us          62         [ -  + ]:           2000 :         if (ARR_NDIM(val) > 1)
 8346 tgl@sss.pgh.pa.us          63         [ #  # ]:UBC           0 :             ereport(ERROR,
                                 64                 :                :                     (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
                                 65                 :                :                      errmsg("array must be one-dimensional")));
 5620 tgl@sss.pgh.pa.us          66         [ -  + ]:CBC        2000 :         if (array_contains_nulls(val))
 7497 tgl@sss.pgh.pa.us          67         [ #  # ]:UBC           0 :             ereport(ERROR,
                                 68                 :                :                     (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
                                 69                 :                :                      errmsg("array must not contain nulls")));
                                 70                 :                : 
 2252 akorotkov@postgresql       71                 :CBC        2000 :         key = ltree_gist_alloc(false, NULL, siglen, NULL, NULL);
                                 72                 :                : 
 8669 bruce@momjian.us           73         [ +  + ]:           9058 :         while (num > 0)
                                 74                 :                :         {
 2252 akorotkov@postgresql       75                 :           7058 :             hashing(LTG_SIGN(key), item, siglen);
 8705 bruce@momjian.us           76                 :           7058 :             num--;
                                 77                 :           7058 :             item = NEXTVAL(item);
                                 78                 :                :         }
                                 79                 :                : 
  176 michael@paquier.xyz        80                 :GNC        2000 :         retval = palloc_object(GISTENTRY);
 8705 bruce@momjian.us           81                 :CBC        2000 :         gistentryinit(*retval, PointerGetDatum(key),
                                 82                 :                :                       entry->rel, entry->page,
                                 83                 :                :                       entry->offset, false);
                                 84                 :                :     }
  295 tgl@sss.pgh.pa.us          85         [ +  - ]:GNC        3888 :     else if (!LTG_ISALLTRUE(DatumGetPointer(entry->key)))
                                 86                 :                :     {
                                 87                 :                :         int32       i;
                                 88                 :                :         ltree_gist *key;
 8669 bruce@momjian.us           89                 :CBC        3888 :         BITVECP     sign = LTG_SIGN(DatumGetPointer(entry->key));
                                 90                 :                : 
 2252 akorotkov@postgresql       91         [ +  - ]:           3888 :         ALOOPBYTE(siglen)
                                 92                 :                :         {
 6770 bruce@momjian.us           93         [ +  - ]:           3888 :             if ((sign[i] & 0xff) != 0xff)
                                 94                 :           3888 :                 PG_RETURN_POINTER(retval);
                                 95                 :                :         }
                                 96                 :                : 
 2252 akorotkov@postgresql       97                 :UBC           0 :         key = ltree_gist_alloc(true, sign, siglen, NULL, NULL);
  176 michael@paquier.xyz        98                 :UNC           0 :         retval = palloc_object(GISTENTRY);
 8705 bruce@momjian.us           99                 :UBC           0 :         gistentryinit(*retval, PointerGetDatum(key),
                                100                 :                :                       entry->rel, entry->page,
                                101                 :                :                       entry->offset, false);
                                102                 :                :     }
 8705 bruce@momjian.us          103                 :CBC        2000 :     PG_RETURN_POINTER(retval);
                                104                 :                : }
                                105                 :                : 
                                106                 :                : Datum
 8669                           107                 :           8795 : _ltree_same(PG_FUNCTION_ARGS)
                                108                 :                : {
                                109                 :           8795 :     ltree_gist *a = (ltree_gist *) PG_GETARG_POINTER(0);
                                110                 :           8795 :     ltree_gist *b = (ltree_gist *) PG_GETARG_POINTER(1);
                                111                 :           8795 :     bool       *result = (bool *) PG_GETARG_POINTER(2);
 2252 akorotkov@postgresql      112         [ +  - ]:           8795 :     int         siglen = LTREE_GET_ASIGLEN();
                                113                 :                : 
 8669 bruce@momjian.us          114   [ -  +  -  - ]:           8795 :     if (LTG_ISALLTRUE(a) && LTG_ISALLTRUE(b))
 8705 bruce@momjian.us          115                 :UBC           0 :         *result = true;
 8669 bruce@momjian.us          116         [ -  + ]:CBC        8795 :     else if (LTG_ISALLTRUE(a))
 8705 bruce@momjian.us          117                 :UBC           0 :         *result = false;
 8669 bruce@momjian.us          118         [ -  + ]:CBC        8795 :     else if (LTG_ISALLTRUE(b))
 8705 bruce@momjian.us          119                 :UBC           0 :         *result = false;
                                120                 :                :     else
                                121                 :                :     {
                                122                 :                :         int32       i;
 8669 bruce@momjian.us          123                 :CBC        8795 :         BITVECP     sa = LTG_SIGN(a),
                                124                 :           8795 :                     sb = LTG_SIGN(b);
                                125                 :                : 
 8705                           126                 :           8795 :         *result = true;
 2252 akorotkov@postgresql      127         [ +  + ]:       13186612 :         ALOOPBYTE(siglen)
                                128                 :                :         {
 6770 bruce@momjian.us          129         [ +  + ]:       13179921 :             if (sa[i] != sb[i])
                                130                 :                :             {
                                131                 :           2104 :                 *result = false;
                                132                 :           2104 :                 break;
                                133                 :                :             }
                                134                 :                :         }
                                135                 :                :     }
 8669                           136                 :           8795 :     PG_RETURN_POINTER(result);
                                137                 :                : }
                                138                 :                : 
                                139                 :                : static int32
 2252 akorotkov@postgresql      140                 :          17590 : unionkey(BITVECP sbase, ltree_gist *add, int siglen)
                                141                 :                : {
                                142                 :                :     int32       i;
 8669 bruce@momjian.us          143                 :          17590 :     BITVECP     sadd = LTG_SIGN(add);
                                144                 :                : 
                                145         [ -  + ]:          17590 :     if (LTG_ISALLTRUE(add))
 8705 bruce@momjian.us          146                 :UBC           0 :         return 1;
                                147                 :                : 
 2252 akorotkov@postgresql      148         [ +  + ]:CBC    32190622 :     ALOOPBYTE(siglen)
 6770 bruce@momjian.us          149                 :       32173032 :         sbase[i] |= sadd[i];
 8705                           150                 :          17590 :     return 0;
                                151                 :                : }
                                152                 :                : 
                                153                 :                : Datum
 8669                           154                 :           8795 : _ltree_union(PG_FUNCTION_ARGS)
                                155                 :                : {
 7944                           156                 :           8795 :     GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
 8669                           157                 :           8795 :     int        *size = (int *) PG_GETARG_POINTER(1);
 2252 akorotkov@postgresql      158         [ +  - ]:           8795 :     int         siglen = LTREE_GET_ASIGLEN();
                                159                 :                :     int32       i;
                                160                 :           8795 :     ltree_gist *result = ltree_gist_alloc(false, NULL, siglen, NULL, NULL);
                                161                 :           8795 :     BITVECP     base = LTG_SIGN(result);
                                162                 :                : 
 8096 teodor@sigaev.ru          163         [ +  + ]:          26385 :     for (i = 0; i < entryvec->n; i++)
                                164                 :                :     {
 2252 akorotkov@postgresql      165         [ -  + ]:          17590 :         if (unionkey(base, GETENTRY(entryvec, i), siglen))
                                166                 :                :         {
 2252 akorotkov@postgresql      167                 :UBC           0 :             result->flag |= LTG_ALLTRUE;
                                168                 :              0 :             SET_VARSIZE(result, LTG_HDRSIZE);
 8705 bruce@momjian.us          169                 :              0 :             break;
                                170                 :                :         }
                                171                 :                :     }
                                172                 :                : 
 2252 akorotkov@postgresql      173                 :CBC        8795 :     *size = VARSIZE(result);
                                174                 :                : 
 8669 bruce@momjian.us          175                 :           8795 :     PG_RETURN_POINTER(result);
                                176                 :                : }
                                177                 :                : 
                                178                 :                : static int32
 2252 akorotkov@postgresql      179                 :UBC           0 : sizebitvec(BITVECP sign, int siglen)
                                180                 :                : {
                                181                 :              0 :     return pg_popcount((const char *) sign, siglen);
                                182                 :                : }
                                183                 :                : 
                                184                 :                : static int
 2252 akorotkov@postgresql      185                 :CBC      105420 : hemdistsign(BITVECP a, BITVECP b, int siglen)
                                186                 :                : {
                                187                 :                :     int         i,
                                188                 :                :                 diff,
 7944 bruce@momjian.us          189                 :         105420 :                 dist = 0;
                                190                 :                : 
 2252 akorotkov@postgresql      191         [ +  + ]:       60186692 :     ALOOPBYTE(siglen)
                                192                 :                :     {
 6770 bruce@momjian.us          193                 :       60081272 :         diff = (unsigned char) (a[i] ^ b[i]);
                                194                 :                :         /* Using the popcount functions here isn't likely to win */
 2661 tgl@sss.pgh.pa.us         195                 :       60081272 :         dist += pg_number_of_ones[diff];
                                196                 :                :     }
 8389 bruce@momjian.us          197                 :         105420 :     return dist;
                                198                 :                : }
                                199                 :                : 
                                200                 :                : static int
 2252 akorotkov@postgresql      201                 :         105420 : hemdist(ltree_gist *a, ltree_gist *b, int siglen)
                                202                 :                : {
 7944 bruce@momjian.us          203         [ -  + ]:         105420 :     if (LTG_ISALLTRUE(a))
                                204                 :                :     {
 7944 bruce@momjian.us          205         [ #  # ]:UBC           0 :         if (LTG_ISALLTRUE(b))
                                206                 :              0 :             return 0;
                                207                 :                :         else
 2252 akorotkov@postgresql      208                 :              0 :             return ASIGLENBIT(siglen) - sizebitvec(LTG_SIGN(b), siglen);
                                209                 :                :     }
 7944 bruce@momjian.us          210         [ -  + ]:CBC      105420 :     else if (LTG_ISALLTRUE(b))
 2252 akorotkov@postgresql      211                 :UBC           0 :         return ASIGLENBIT(siglen) - sizebitvec(LTG_SIGN(a), siglen);
                                212                 :                : 
 2252 akorotkov@postgresql      213                 :CBC      105420 :     return hemdistsign(LTG_SIGN(a), LTG_SIGN(b), siglen);
                                214                 :                : }
                                215                 :                : 
                                216                 :                : 
                                217                 :                : Datum
 8669 bruce@momjian.us          218                 :          20115 : _ltree_penalty(PG_FUNCTION_ARGS)
                                219                 :                : {
                                220                 :          20115 :     ltree_gist *origval = (ltree_gist *) DatumGetPointer(((GISTENTRY *) PG_GETARG_POINTER(0))->key);
                                221                 :          20115 :     ltree_gist *newval = (ltree_gist *) DatumGetPointer(((GISTENTRY *) PG_GETARG_POINTER(1))->key);
                                222                 :          20115 :     float      *penalty = (float *) PG_GETARG_POINTER(2);
 2252 akorotkov@postgresql      223         [ +  - ]:          20115 :     int         siglen = LTREE_GET_ASIGLEN();
                                224                 :                : 
                                225                 :          20115 :     *penalty = hemdist(origval, newval, siglen);
 8669 bruce@momjian.us          226                 :          20115 :     PG_RETURN_POINTER(penalty);
                                227                 :                : }
                                228                 :                : 
                                229                 :                : typedef struct
                                230                 :                : {
                                231                 :                :     OffsetNumber pos;
                                232                 :                :     int32       cost;
                                233                 :                : } SPLITCOST;
                                234                 :                : 
                                235                 :                : static int
                                236                 :           7253 : comparecost(const void *a, const void *b)
                                237                 :                : {
 5375 peter_e@gmx.net           238                 :           7253 :     return ((const SPLITCOST *) a)->cost - ((const SPLITCOST *) b)->cost;
                                239                 :                : }
                                240                 :                : 
                                241                 :                : Datum
 8669 bruce@momjian.us          242                 :            892 : _ltree_picksplit(PG_FUNCTION_ARGS)
                                243                 :                : {
 7944                           244                 :            892 :     GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
 8669                           245                 :            892 :     GIST_SPLITVEC *v = (GIST_SPLITVEC *) PG_GETARG_POINTER(1);
 2252 akorotkov@postgresql      246         [ +  - ]:            892 :     int         siglen = LTREE_GET_ASIGLEN();
                                247                 :                :     OffsetNumber k,
                                248                 :                :                 j;
                                249                 :                :     ltree_gist *datum_l,
                                250                 :                :                *datum_r;
                                251                 :                :     BITVECP     union_l,
                                252                 :                :                 union_r;
                                253                 :                :     int32       size_alpha,
                                254                 :                :                 size_beta;
                                255                 :                :     int32       size_waste,
 8389 bruce@momjian.us          256                 :            892 :                 waste = -1;
                                257                 :                :     int32       nbytes;
 8669                           258                 :            892 :     OffsetNumber seed_1 = 0,
                                259                 :            892 :                 seed_2 = 0;
                                260                 :                :     OffsetNumber *left,
                                261                 :                :                *right;
                                262                 :                :     OffsetNumber maxoff;
                                263                 :                :     BITVECP     ptr;
                                264                 :                :     int         i;
                                265                 :                :     SPLITCOST  *costvector;
                                266                 :                :     ltree_gist *_k,
                                267                 :                :                *_j;
                                268                 :                : 
 8096 teodor@sigaev.ru          269                 :            892 :     maxoff = entryvec->n - 2;
 8705 bruce@momjian.us          270                 :            892 :     nbytes = (maxoff + 2) * sizeof(OffsetNumber);
                                271                 :            892 :     v->spl_left = (OffsetNumber *) palloc(nbytes);
                                272                 :            892 :     v->spl_right = (OffsetNumber *) palloc(nbytes);
                                273                 :                : 
 7944                           274         [ +  + ]:           3635 :     for (k = FirstOffsetNumber; k < maxoff; k = OffsetNumberNext(k))
                                275                 :                :     {
 8669                           276                 :           2743 :         _k = GETENTRY(entryvec, k);
 7944                           277         [ +  + ]:          73508 :         for (j = OffsetNumberNext(k); j <= maxoff; j = OffsetNumberNext(j))
                                278                 :                :         {
 2252 akorotkov@postgresql      279                 :          70765 :             size_waste = hemdist(_k, GETENTRY(entryvec, j), siglen);
 7944 bruce@momjian.us          280         [ +  + ]:          70765 :             if (size_waste > waste)
                                281                 :                :             {
 8705                           282                 :           1650 :                 waste = size_waste;
                                283                 :           1650 :                 seed_1 = k;
                                284                 :           1650 :                 seed_2 = j;
                                285                 :                :             }
                                286                 :                :         }
                                287                 :                :     }
                                288                 :                : 
                                289                 :            892 :     left = v->spl_left;
                                290                 :            892 :     v->spl_nleft = 0;
                                291                 :            892 :     right = v->spl_right;
                                292                 :            892 :     v->spl_nright = 0;
                                293                 :                : 
 8669                           294   [ +  -  -  + ]:            892 :     if (seed_1 == 0 || seed_2 == 0)
                                295                 :                :     {
 8705 bruce@momjian.us          296                 :UBC           0 :         seed_1 = 1;
                                297                 :              0 :         seed_2 = 2;
                                298                 :                :     }
                                299                 :                : 
                                300                 :                :     /* form initial .. */
 2252 akorotkov@postgresql      301                 :CBC         892 :     datum_l = ltree_gist_alloc(LTG_ISALLTRUE(GETENTRY(entryvec, seed_1)),
                                302                 :            892 :                                LTG_SIGN(GETENTRY(entryvec, seed_1)),
                                303                 :                :                                siglen, NULL, NULL);
                                304                 :                : 
                                305                 :            892 :     datum_r = ltree_gist_alloc(LTG_ISALLTRUE(GETENTRY(entryvec, seed_2)),
                                306                 :            892 :                                LTG_SIGN(GETENTRY(entryvec, seed_2)),
                                307                 :                :                                siglen, NULL, NULL);
                                308                 :                : 
 8705 bruce@momjian.us          309                 :            892 :     maxoff = OffsetNumberNext(maxoff);
                                310                 :                :     /* sort before ... */
  176 michael@paquier.xyz       311                 :GNC         892 :     costvector = palloc_array(SPLITCOST, maxoff);
 8669 bruce@momjian.us          312         [ +  + ]:CBC        5419 :     for (j = FirstOffsetNumber; j <= maxoff; j = OffsetNumberNext(j))
                                313                 :                :     {
                                314                 :           4527 :         costvector[j - 1].pos = j;
                                315                 :           4527 :         _j = GETENTRY(entryvec, j);
 2252 akorotkov@postgresql      316                 :           4527 :         size_alpha = hemdist(datum_l, _j, siglen);
                                317                 :           4527 :         size_beta = hemdist(datum_r, _j, siglen);
 1331 peter@eisentraut.org      318                 :           4527 :         costvector[j - 1].cost = abs(size_alpha - size_beta);
                                319                 :                :     }
 1208                           320                 :            892 :     qsort(costvector, maxoff, sizeof(SPLITCOST), comparecost);
                                321                 :                : 
 7944 bruce@momjian.us          322                 :            892 :     union_l = LTG_SIGN(datum_l);
                                323                 :            892 :     union_r = LTG_SIGN(datum_r);
                                324                 :                : 
 8669                           325         [ +  + ]:           5419 :     for (k = 0; k < maxoff; k++)
                                326                 :                :     {
 8705                           327                 :           4527 :         j = costvector[k].pos;
 8669                           328         [ +  + ]:           4527 :         if (j == seed_1)
                                329                 :                :         {
 8705                           330                 :            892 :             *left++ = j;
                                331                 :            892 :             v->spl_nleft++;
                                332                 :            892 :             continue;
                                333                 :                :         }
 8669                           334         [ +  + ]:           3635 :         else if (j == seed_2)
                                335                 :                :         {
 8705                           336                 :            892 :             *right++ = j;
                                337                 :            892 :             v->spl_nright++;
                                338                 :            892 :             continue;
                                339                 :                :         }
 8389                           340                 :           2743 :         _j = GETENTRY(entryvec, j);
 2252 akorotkov@postgresql      341                 :           2743 :         size_alpha = hemdist(datum_l, _j, siglen);
                                342                 :           2743 :         size_beta = hemdist(datum_r, _j, siglen);
                                343                 :                : 
 7944 bruce@momjian.us          344         [ +  + ]:           2743 :         if (size_alpha < size_beta + WISH_F(v->spl_nleft, v->spl_nright, 0.00001))
                                345                 :                :         {
                                346   [ +  -  -  + ]:           1323 :             if (LTG_ISALLTRUE(datum_l) || LTG_ISALLTRUE(_j))
                                347                 :                :             {
 8389 bruce@momjian.us          348         [ #  # ]:UBC           0 :                 if (!LTG_ISALLTRUE(datum_l))
 1208 peter@eisentraut.org      349                 :              0 :                     memset(union_l, 0xff, siglen);
                                350                 :                :             }
                                351                 :                :             else
                                352                 :                :             {
 7944 bruce@momjian.us          353                 :CBC        1323 :                 ptr = LTG_SIGN(_j);
 2252 akorotkov@postgresql      354         [ +  + ]:        1693051 :                 ALOOPBYTE(siglen)
 6770 bruce@momjian.us          355                 :        1691728 :                     union_l[i] |= ptr[i];
                                356                 :                :             }
 8705                           357                 :           1323 :             *left++ = j;
                                358                 :           1323 :             v->spl_nleft++;
                                359                 :                :         }
                                360                 :                :         else
                                361                 :                :         {
 7944                           362   [ +  -  -  + ]:           1420 :             if (LTG_ISALLTRUE(datum_r) || LTG_ISALLTRUE(_j))
                                363                 :                :             {
 8389 bruce@momjian.us          364         [ #  # ]:UBC           0 :                 if (!LTG_ISALLTRUE(datum_r))
 1208 peter@eisentraut.org      365                 :              0 :                     memset(union_r, 0xff, siglen);
                                366                 :                :             }
                                367                 :                :             else
                                368                 :                :             {
 7944 bruce@momjian.us          369                 :CBC        1420 :                 ptr = LTG_SIGN(_j);
 2252 akorotkov@postgresql      370         [ +  + ]:        1919416 :                 ALOOPBYTE(siglen)
 6770 bruce@momjian.us          371                 :        1917996 :                     union_r[i] |= ptr[i];
                                372                 :                :             }
 8705                           373                 :           1420 :             *right++ = j;
                                374                 :           1420 :             v->spl_nright++;
                                375                 :                :         }
                                376                 :                :     }
                                377                 :                : 
                                378                 :            892 :     *right = *left = FirstOffsetNumber;
                                379                 :                : 
                                380                 :            892 :     v->spl_ldatum = PointerGetDatum(datum_l);
                                381                 :            892 :     v->spl_rdatum = PointerGetDatum(datum_r);
                                382                 :                : 
 8669                           383                 :            892 :     PG_RETURN_POINTER(v);
                                384                 :                : }
                                385                 :                : 
                                386                 :                : static bool
 2252 akorotkov@postgresql      387                 :           2548 : gist_te(ltree_gist *key, ltree *query, int siglen)
                                388                 :                : {
 8669 bruce@momjian.us          389                 :           2548 :     ltree_level *curq = LTREE_FIRST(query);
                                390                 :           2548 :     BITVECP     sign = LTG_SIGN(key);
                                391                 :           2548 :     int         qlen = query->numlevel;
                                392                 :                :     unsigned int hv;
                                393                 :                : 
                                394         [ -  + ]:           2548 :     if (LTG_ISALLTRUE(key))
 8705 bruce@momjian.us          395                 :UBC           0 :         return true;
                                396                 :                : 
 8669 bruce@momjian.us          397         [ +  + ]:CBC        8614 :     while (qlen > 0)
                                398                 :                :     {
                                399                 :           6592 :         hv = ltree_crc32_sz(curq->name, curq->len);
 2252 akorotkov@postgresql      400         [ +  + ]:           6592 :         if (!GETBIT(sign, AHASHVAL(hv, siglen)))
 8669 bruce@momjian.us          401                 :            526 :             return false;
 8705                           402                 :           6066 :         curq = LEVEL_NEXT(curq);
                                403                 :           6066 :         qlen--;
                                404                 :                :     }
                                405                 :                : 
                                406                 :           2022 :     return true;
                                407                 :                : }
                                408                 :                : 
                                409                 :                : typedef struct LtreeSignature
                                410                 :                : {
                                411                 :                :     BITVECP     sign;
                                412                 :                :     int         siglen;
                                413                 :                : } LtreeSignature;
                                414                 :                : 
                                415                 :                : static bool
 2252 akorotkov@postgresql      416                 :           3844 : checkcondition_bit(void *cxt, ITEM *val)
                                417                 :                : {
                                418                 :           3844 :     LtreeSignature *sig = cxt;
                                419                 :                : 
                                420   [ +  -  +  + ]:           3844 :     return (FLG_CANLOOKSIGN(val->flag)) ? GETBIT(sig->sign, AHASHVAL(val->val, sig->siglen)) : true;
                                421                 :                : }
                                422                 :                : 
                                423                 :                : static bool
                                424                 :           2296 : gist_qtxt(ltree_gist *key, ltxtquery *query, int siglen)
                                425                 :                : {
                                426                 :                :     LtreeSignature sig;
                                427                 :                : 
 8669 bruce@momjian.us          428         [ -  + ]:           2296 :     if (LTG_ISALLTRUE(key))
 8705 bruce@momjian.us          429                 :UBC           0 :         return true;
                                430                 :                : 
 2252 akorotkov@postgresql      431                 :CBC        2296 :     sig.sign = LTG_SIGN(key);
                                432                 :           2296 :     sig.siglen = siglen;
                                433                 :                : 
 2312 alvherre@alvh.no-ip.      434                 :           2296 :     return ltree_execute(GETQUERY(query),
                                435                 :                :                          &sig, false,
                                436                 :                :                          checkcondition_bit);
                                437                 :                : }
                                438                 :                : 
                                439                 :                : static bool
 2252 akorotkov@postgresql      440                 :          15676 : gist_qe(ltree_gist *key, lquery *query, int siglen)
                                441                 :                : {
 8669 bruce@momjian.us          442                 :          15676 :     lquery_level *curq = LQUERY_FIRST(query);
                                443                 :          15676 :     BITVECP     sign = LTG_SIGN(key);
                                444                 :          15676 :     int         qlen = query->numlevel;
                                445                 :                : 
                                446         [ -  + ]:          15676 :     if (LTG_ISALLTRUE(key))
 8705 bruce@momjian.us          447                 :UBC           0 :         return true;
                                448                 :                : 
 8669 bruce@momjian.us          449         [ +  + ]:CBC       47074 :     while (qlen > 0)
                                450                 :                :     {
                                451   [ +  +  +  - ]:          37770 :         if (curq->numvar && LQL_CANLOOKSIGN(curq))
                                452                 :                :         {
                                453                 :          26460 :             bool        isexist = false;
                                454                 :          26460 :             int         vlen = curq->numvar;
 8705                           455                 :          26460 :             lquery_variant *curv = LQL_FIRST(curq);
                                456                 :                : 
 8669                           457         [ +  + ]:          32832 :             while (vlen > 0)
                                458                 :                :             {
 2252 akorotkov@postgresql      459         [ +  + ]:          26460 :                 if (GETBIT(sign, AHASHVAL(curv->val, siglen)))
                                460                 :                :                 {
 8669 bruce@momjian.us          461                 :          20088 :                     isexist = true;
 8705                           462                 :          20088 :                     break;
                                463                 :                :                 }
                                464                 :           6372 :                 curv = LVAR_NEXT(curv);
                                465                 :           6372 :                 vlen--;
                                466                 :                :             }
 8669                           467         [ +  + ]:          26460 :             if (!isexist)
 8705                           468                 :           6372 :                 return false;
                                469                 :                :         }
                                470                 :                : 
                                471                 :          31398 :         curq = LQL_NEXT(curq);
                                472                 :          31398 :         qlen--;
                                473                 :                :     }
                                474                 :                : 
                                475                 :           9304 :     return true;
                                476                 :                : }
                                477                 :                : 
                                478                 :                : static bool
 2252 akorotkov@postgresql      479                 :           2488 : _arrq_cons(ltree_gist *key, ArrayType *_query, int siglen)
                                480                 :                : {
 7944 bruce@momjian.us          481         [ -  + ]:           2488 :     lquery     *query = (lquery *) ARR_DATA_PTR(_query);
                                482                 :           2488 :     int         num = ArrayGetNItems(ARR_NDIM(_query), ARR_DIMS(_query));
                                483                 :                : 
 5939 tgl@sss.pgh.pa.us         484         [ -  + ]:           2488 :     if (ARR_NDIM(_query) > 1)
 7944 bruce@momjian.us          485         [ #  # ]:UBC           0 :         ereport(ERROR,
                                486                 :                :                 (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
                                487                 :                :                  errmsg("array must be one-dimensional")));
 5620 tgl@sss.pgh.pa.us         488         [ -  + ]:CBC        2488 :     if (array_contains_nulls(_query))
 7497 tgl@sss.pgh.pa.us         489         [ #  # ]:UBC           0 :         ereport(ERROR,
                                490                 :                :                 (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
                                491                 :                :                  errmsg("array must not contain nulls")));
                                492                 :                : 
 7944 bruce@momjian.us          493         [ +  + ]:CBC        4616 :     while (num > 0)
                                494                 :                :     {
 2252 akorotkov@postgresql      495         [ +  + ]:           3674 :         if (gist_qe(key, query, siglen))
 7944 bruce@momjian.us          496                 :           1546 :             return true;
                                497                 :           2128 :         num--;
                                498                 :           2128 :         query = (lquery *) NEXTVAL(query);
                                499                 :                :     }
                                500                 :            942 :     return false;
                                501                 :                : }
                                502                 :                : 
                                503                 :                : Datum
 8669                           504                 :          19334 : _ltree_consistent(PG_FUNCTION_ARGS)
                                505                 :                : {
                                506                 :          19334 :     GISTENTRY  *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
  548 peter@eisentraut.org      507                 :          19334 :     void       *query = PG_DETOAST_DATUM(PG_GETARG_DATUM(1));
 8705 bruce@momjian.us          508                 :          19334 :     StrategyNumber strategy = (StrategyNumber) PG_GETARG_UINT16(2);
                                509                 :                : #ifdef NOT_USED
                                510                 :                :     Oid         subtype = PG_GETARG_OID(3);
                                511                 :                : #endif
 6620 tgl@sss.pgh.pa.us         512                 :          19334 :     bool       *recheck = (bool *) PG_GETARG_POINTER(4);
 2252 akorotkov@postgresql      513         [ +  - ]:          19334 :     int         siglen = LTREE_GET_ASIGLEN();
 6620 tgl@sss.pgh.pa.us         514                 :          19334 :     ltree_gist *key = (ltree_gist *) DatumGetPointer(entry->key);
 8669 bruce@momjian.us          515                 :          19334 :     bool        res = false;
                                516                 :                : 
                                517                 :                :     /* All cases served by this function are inexact */
 6620 tgl@sss.pgh.pa.us         518                 :          19334 :     *recheck = true;
                                519                 :                : 
 8669 bruce@momjian.us          520   [ +  +  +  +  :          19334 :     switch (strategy)
                                                 - ]
                                521                 :                :     {
 8705                           522                 :           2548 :         case 10:
                                523                 :                :         case 11:
 2252 akorotkov@postgresql      524                 :           2548 :             res = gist_te(key, (ltree *) query, siglen);
 8705 bruce@momjian.us          525                 :           2548 :             break;
                                526                 :          12002 :         case 12:
                                527                 :                :         case 13:
 2252 akorotkov@postgresql      528                 :          12002 :             res = gist_qe(key, (lquery *) query, siglen);
 8669 bruce@momjian.us          529                 :          12002 :             break;
 8705                           530                 :           2296 :         case 14:
                                531                 :                :         case 15:
 2252 akorotkov@postgresql      532                 :           2296 :             res = gist_qtxt(key, (ltxtquery *) query, siglen);
 8669 bruce@momjian.us          533                 :           2296 :             break;
 8501                           534                 :           2488 :         case 16:
                                535                 :                :         case 17:
 2252 akorotkov@postgresql      536                 :           2488 :             res = _arrq_cons(key, (ArrayType *) query, siglen);
 8501 bruce@momjian.us          537                 :           2488 :             break;
 8705 bruce@momjian.us          538                 :UBC           0 :         default:
                                539                 :                :             /* internal error */
 8346 tgl@sss.pgh.pa.us         540         [ #  # ]:              0 :             elog(ERROR, "unrecognized StrategyNumber: %d", strategy);
                                541                 :                :     }
 7178 bruce@momjian.us          542         [ -  + ]:CBC       19334 :     PG_FREE_IF_COPY(query, 1);
 8705                           543                 :          19334 :     PG_RETURN_BOOL(res);
                                544                 :                : }
                                545                 :                : 
                                546                 :                : Datum
 2252 akorotkov@postgresql      547                 :             10 : _ltree_gist_options(PG_FUNCTION_ARGS)
                                548                 :                : {
                                549                 :             10 :     local_relopts *relopts = (local_relopts *) PG_GETARG_POINTER(0);
                                550                 :                : 
                                551                 :             10 :     init_local_reloptions(relopts, sizeof(LtreeGistOptions));
                                552                 :             10 :     add_local_int_reloption(relopts, "siglen", "signature length",
                                553                 :                :                             LTREE_ASIGLEN_DEFAULT, 1, LTREE_ASIGLEN_MAX,
                                554                 :                :                             offsetof(LtreeGistOptions, siglen));
                                555                 :                : 
                                556                 :             10 :     PG_RETURN_VOID();
                                557                 :                : }
        

Generated by: LCOV version 2.5.0-beta