LCOV - differential code coverage report
Current view: top level - contrib/hstore - hstore_gist.c (source / functions) Coverage Total Hit UNC LBC UBC GNC CBC EUB ECB DUB DCB
Current: 0e5ff9b9b45a657aea12440478dc002e9b01f138 vs 0123ce131fca454009439dfa3b2266d1d40737d7 Lines: 87.1 % 264 230 1 33 4 226 1 4
Current Date: 2026-03-14 14:10:32 -0400 Functions: 88.9 % 27 24 3 3 21
Baseline: lcov-20260315-024220-baseline Branches: 69.1 % 162 112 1 49 112 9 3
Baseline Date: 2026-03-14 15:27:56 +0100 Line coverage date bins:
Legend: Lines:     hit not hit
Branches: + taken - not taken # not executed
(30,360] days: 80.0 % 5 4 1 4
(360..) days: 87.3 % 259 226 33 226
Function coverage date bins:
(360..) days: 88.9 % 27 24 3 3 21
Branch coverage date bins:
(360..) days: 64.4 % 174 112 1 49 112 9 3

 Age         Owner                    Branch data    TLA  Line data    Source code
                                  1                 :                : /*
                                  2                 :                :  * contrib/hstore/hstore_gist.c
                                  3                 :                :  */
                                  4                 :                : #include "postgres.h"
                                  5                 :                : 
                                  6                 :                : #include "access/gist.h"
                                  7                 :                : #include "access/reloptions.h"
                                  8                 :                : #include "access/stratnum.h"
                                  9                 :                : #include "catalog/pg_type.h"
                                 10                 :                : #include "common/int.h"
                                 11                 :                : #include "hstore.h"
                                 12                 :                : #include "utils/pg_crc.h"
                                 13                 :                : 
                                 14                 :                : /* gist_hstore_ops opclass options */
                                 15                 :                : typedef struct
                                 16                 :                : {
                                 17                 :                :     int32       vl_len_;        /* varlena header (do not touch directly!) */
                                 18                 :                :     int         siglen;         /* signature length in bytes */
                                 19                 :                : } GistHstoreOptions;
                                 20                 :                : 
                                 21                 :                : /* bigint defines */
                                 22                 :                : #define BITBYTE 8
                                 23                 :                : #define SIGLEN_DEFAULT  (sizeof(int32) * 4)
                                 24                 :                : #define SIGLEN_MAX      GISTMaxIndexKeySize
                                 25                 :                : #define SIGLENBIT(siglen) ((siglen) * BITBYTE)
                                 26                 :                : #define GET_SIGLEN()    (PG_HAS_OPCLASS_OPTIONS() ? \
                                 27                 :                :                          ((GistHstoreOptions *) PG_GET_OPCLASS_OPTIONS())->siglen : \
                                 28                 :                :                          SIGLEN_DEFAULT)
                                 29                 :                : 
                                 30                 :                : 
                                 31                 :                : typedef char *BITVECP;
                                 32                 :                : 
                                 33                 :                : #define LOOPBYTE(siglen) \
                                 34                 :                :             for (i = 0; i < (siglen); i++)
                                 35                 :                : 
                                 36                 :                : #define LOOPBIT(siglen) \
                                 37                 :                :             for (i = 0; i < SIGLENBIT(siglen); i++)
                                 38                 :                : 
                                 39                 :                : /* beware of multiple evaluation of arguments to these macros! */
                                 40                 :                : #define GETBYTE(x,i) ( *( (BITVECP)(x) + (int)( (i) / BITBYTE ) ) )
                                 41                 :                : #define GETBITBYTE(x,i) ( (*((char*)(x)) >> (i)) & 0x01 )
                                 42                 :                : #define CLRBIT(x,i)   GETBYTE(x,i) &= ~( 0x01 << ( (i) % BITBYTE ) )
                                 43                 :                : #define SETBIT(x,i)   GETBYTE(x,i) |=  ( 0x01 << ( (i) % BITBYTE ) )
                                 44                 :                : #define GETBIT(x,i) ( (GETBYTE(x,i) >> ( (i) % BITBYTE )) & 0x01 )
                                 45                 :                : #define HASHVAL(val, siglen) (((unsigned int)(val)) % SIGLENBIT(siglen))
                                 46                 :                : #define HASH(sign, val, siglen) SETBIT((sign), HASHVAL(val, siglen))
                                 47                 :                : 
                                 48                 :                : typedef struct
                                 49                 :                : {
                                 50                 :                :     int32       vl_len_;        /* varlena header (do not touch directly!) */
                                 51                 :                :     int32       flag;
                                 52                 :                :     char        data[FLEXIBLE_ARRAY_MEMBER];
                                 53                 :                : } GISTTYPE;
                                 54                 :                : 
                                 55                 :                : #define ALLISTRUE       0x04
                                 56                 :                : 
                                 57                 :                : #define ISALLTRUE(x)    ( ((GISTTYPE*)x)->flag & ALLISTRUE )
                                 58                 :                : 
                                 59                 :                : #define GTHDRSIZE       (VARHDRSZ + sizeof(int32))
                                 60                 :                : #define CALCGTSIZE(flag, siglen) ( GTHDRSIZE+(((flag) & ALLISTRUE) ? 0 : (siglen)) )
                                 61                 :                : 
                                 62                 :                : #define GETSIGN(x)      ( (BITVECP)( (char*)x+GTHDRSIZE ) )
                                 63                 :                : 
                                 64                 :                : #define SUMBIT(val) (       \
                                 65                 :                :     GETBITBYTE((val),0) + \
                                 66                 :                :     GETBITBYTE((val),1) + \
                                 67                 :                :     GETBITBYTE((val),2) + \
                                 68                 :                :     GETBITBYTE((val),3) + \
                                 69                 :                :     GETBITBYTE((val),4) + \
                                 70                 :                :     GETBITBYTE((val),5) + \
                                 71                 :                :     GETBITBYTE((val),6) + \
                                 72                 :                :     GETBITBYTE((val),7)   \
                                 73                 :                : )
                                 74                 :                : 
                                 75                 :                : #define GETENTRY(vec,pos) ((GISTTYPE *) DatumGetPointer((vec)->vector[(pos)].key))
                                 76                 :                : 
                                 77                 :                : #define WISH_F(a,b,c) (double)( -(double)(((a)-(b))*((a)-(b))*((a)-(b)))*(c) )
                                 78                 :                : 
                                 79                 :                : /* shorthand for calculating CRC-32 of a single chunk of data. */
                                 80                 :                : static pg_crc32
  892 peter@eisentraut.org       81                 :CBC       33069 : crc32_sz(const char *buf, int size)
                                 82                 :                : {
                                 83                 :                :     pg_crc32    crc;
                                 84                 :                : 
 4149 heikki.linnakangas@i       85                 :          33069 :     INIT_TRADITIONAL_CRC32(crc);
                                 86         [ +  + ]:         176340 :     COMP_TRADITIONAL_CRC32(crc, buf, size);
                                 87                 :          33069 :     FIN_TRADITIONAL_CRC32(crc);
                                 88                 :                : 
                                 89                 :          33069 :     return crc;
                                 90                 :                : }
                                 91                 :                : 
                                 92                 :                : 
 7131 teodor@sigaev.ru           93                 :              7 : PG_FUNCTION_INFO_V1(ghstore_in);
                                 94                 :              7 : PG_FUNCTION_INFO_V1(ghstore_out);
                                 95                 :                : 
                                 96                 :                : 
                                 97                 :                : Datum
 7102 bruce@momjian.us           98                 :UBC           0 : ghstore_in(PG_FUNCTION_ARGS)
                                 99                 :                : {
 1191 tgl@sss.pgh.pa.us         100         [ #  # ]:              0 :     ereport(ERROR,
                                101                 :                :             (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
                                102                 :                :              errmsg("cannot accept a value of type %s", "ghstore")));
                                103                 :                : 
                                104                 :                :     PG_RETURN_VOID();           /* keep compiler quiet */
                                105                 :                : }
                                106                 :                : 
                                107                 :                : Datum
 7102 bruce@momjian.us          108                 :              0 : ghstore_out(PG_FUNCTION_ARGS)
                                109                 :                : {
 1191 tgl@sss.pgh.pa.us         110         [ #  # ]:              0 :     ereport(ERROR,
                                111                 :                :             (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
                                112                 :                :              errmsg("cannot display a value of type %s", "ghstore")));
                                113                 :                : 
                                114                 :                :     PG_RETURN_VOID();           /* keep compiler quiet */
                                115                 :                : }
                                116                 :                : 
                                117                 :                : static GISTTYPE *
 2176 akorotkov@postgresql      118                 :CBC        9772 : ghstore_alloc(bool allistrue, int siglen, BITVECP sign)
                                119                 :                : {
                                120         [ -  + ]:           9772 :     int         flag = allistrue ? ALLISTRUE : 0;
                                121         [ +  - ]:           9772 :     int         size = CALCGTSIZE(flag, siglen);
                                122                 :           9772 :     GISTTYPE   *res = palloc(size);
                                123                 :                : 
                                124                 :           9772 :     SET_VARSIZE(res, size);
                                125                 :           9772 :     res->flag = flag;
                                126                 :                : 
                                127         [ +  - ]:           9772 :     if (!allistrue)
                                128                 :                :     {
                                129         [ +  + ]:           9772 :         if (sign)
                                130                 :           1440 :             memcpy(GETSIGN(res), sign, siglen);
                                131                 :                :         else
                                132                 :           8332 :             memset(GETSIGN(res), 0, siglen);
                                133                 :                :     }
                                134                 :                : 
                                135                 :           9772 :     return res;
                                136                 :                : }
                                137                 :                : 
 7131 teodor@sigaev.ru          138                 :              8 : PG_FUNCTION_INFO_V1(ghstore_consistent);
                                139                 :              8 : PG_FUNCTION_INFO_V1(ghstore_compress);
                                140                 :              8 : PG_FUNCTION_INFO_V1(ghstore_decompress);
                                141                 :              8 : PG_FUNCTION_INFO_V1(ghstore_penalty);
                                142                 :              8 : PG_FUNCTION_INFO_V1(ghstore_picksplit);
                                143                 :              8 : PG_FUNCTION_INFO_V1(ghstore_union);
                                144                 :              8 : PG_FUNCTION_INFO_V1(ghstore_same);
 2176 akorotkov@postgresql      145                 :              8 : PG_FUNCTION_INFO_V1(ghstore_options);
                                146                 :                : 
                                147                 :                : Datum
 7102 bruce@momjian.us          148                 :           8414 : ghstore_compress(PG_FUNCTION_ARGS)
                                149                 :                : {
 7131 teodor@sigaev.ru          150                 :           8414 :     GISTENTRY  *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
 2176 akorotkov@postgresql      151         [ +  - ]:           8414 :     int         siglen = GET_SIGLEN();
 7131 teodor@sigaev.ru          152                 :           8414 :     GISTENTRY  *retval = entry;
                                153                 :                : 
 7102 bruce@momjian.us          154         [ +  + ]:           8414 :     if (entry->leafkey)
                                155                 :                :     {
 2176 akorotkov@postgresql      156                 :           2002 :         GISTTYPE   *res = ghstore_alloc(false, siglen, NULL);
 6010 tgl@sss.pgh.pa.us         157                 :           2002 :         HStore     *val = DatumGetHStoreP(entry->key);
                                158                 :           2002 :         HEntry     *hsent = ARRPTR(val);
                                159                 :           2002 :         char       *ptr = STRPTR(val);
 5861 bruce@momjian.us          160                 :           2002 :         int         count = HS_COUNT(val);
                                161                 :                :         int         i;
                                162                 :                : 
 6010 tgl@sss.pgh.pa.us         163         [ +  + ]:          11564 :         for (i = 0; i < count; ++i)
                                164                 :                :         {
                                165                 :                :             int         h;
                                166                 :                : 
 3769                           167         [ +  + ]:           9562 :             h = crc32_sz((char *) HSTORE_KEY(hsent, ptr, i),
                                168         [ +  + ]:           9562 :                          HSTORE_KEYLEN(hsent, i));
 2176 akorotkov@postgresql      169                 :           9562 :             HASH(GETSIGN(res), h, siglen);
 3769 tgl@sss.pgh.pa.us         170         [ +  + ]:           9562 :             if (!HSTORE_VALISNULL(hsent, i))
                                171                 :                :             {
                                172         [ +  - ]:           9560 :                 h = crc32_sz((char *) HSTORE_VAL(hsent, ptr, i),
                                173         [ -  + ]:           9560 :                              HSTORE_VALLEN(hsent, i));
 2176 akorotkov@postgresql      174                 :           9560 :                 HASH(GETSIGN(res), h, siglen);
                                175                 :                :             }
                                176                 :                :         }
                                177                 :                : 
  100 michael@paquier.xyz       178                 :GNC        2002 :         retval = palloc_object(GISTENTRY);
 7131 teodor@sigaev.ru          179                 :CBC        2002 :         gistentryinit(*retval, PointerGetDatum(res),
                                180                 :                :                       entry->rel, entry->page,
                                181                 :                :                       entry->offset,
                                182                 :                :                       false);
                                183                 :                :     }
 7102 bruce@momjian.us          184         [ +  - ]:           6412 :     else if (!ISALLTRUE(DatumGetPointer(entry->key)))
                                185                 :                :     {
                                186                 :                :         int32       i;
                                187                 :                :         GISTTYPE   *res;
                                188                 :           6412 :         BITVECP     sign = GETSIGN(DatumGetPointer(entry->key));
                                189                 :                : 
 2176 akorotkov@postgresql      190         [ +  - ]:           6412 :         LOOPBYTE(siglen)
                                191                 :                :         {
 6694 bruce@momjian.us          192         [ +  - ]:           6412 :             if ((sign[i] & 0xff) != 0xff)
                                193                 :           6412 :                 PG_RETURN_POINTER(retval);
                                194                 :                :         }
                                195                 :                : 
 2176 akorotkov@postgresql      196                 :UBC           0 :         res = ghstore_alloc(true, siglen, NULL);
                                197                 :                : 
  100 michael@paquier.xyz       198                 :UNC           0 :         retval = palloc_object(GISTENTRY);
 7131 teodor@sigaev.ru          199                 :UBC           0 :         gistentryinit(*retval, PointerGetDatum(res),
                                200                 :                :                       entry->rel, entry->page,
                                201                 :                :                       entry->offset,
                                202                 :                :                       false);
                                203                 :                :     }
                                204                 :                : 
 7131 teodor@sigaev.ru          205                 :CBC        2002 :     PG_RETURN_POINTER(retval);
                                206                 :                : }
                                207                 :                : 
                                208                 :                : /*
                                209                 :                :  * Since type ghstore isn't toastable (and doesn't need to be),
                                210                 :                :  * this function can be a no-op.
                                211                 :                :  */
                                212                 :                : Datum
 7102 bruce@momjian.us          213                 :          49358 : ghstore_decompress(PG_FUNCTION_ARGS)
                                214                 :                : {
 5659 tgl@sss.pgh.pa.us         215                 :          49358 :     PG_RETURN_POINTER(PG_GETARG_POINTER(0));
                                216                 :                : }
                                217                 :                : 
                                218                 :                : Datum
 7102 bruce@momjian.us          219                 :           6330 : ghstore_same(PG_FUNCTION_ARGS)
                                220                 :                : {
 7131 teodor@sigaev.ru          221                 :           6330 :     GISTTYPE   *a = (GISTTYPE *) PG_GETARG_POINTER(0);
                                222                 :           6330 :     GISTTYPE   *b = (GISTTYPE *) PG_GETARG_POINTER(1);
 7102 bruce@momjian.us          223                 :           6330 :     bool       *result = (bool *) PG_GETARG_POINTER(2);
 2176 akorotkov@postgresql      224         [ +  - ]:           6330 :     int         siglen = GET_SIGLEN();
                                225                 :                : 
                                226                 :                : 
 7131 teodor@sigaev.ru          227   [ -  +  -  - ]:           6330 :     if (ISALLTRUE(a) && ISALLTRUE(b))
 7131 teodor@sigaev.ru          228                 :UBC           0 :         *result = true;
 7131 teodor@sigaev.ru          229         [ -  + ]:CBC        6330 :     else if (ISALLTRUE(a))
 7131 teodor@sigaev.ru          230                 :UBC           0 :         *result = false;
 7131 teodor@sigaev.ru          231         [ -  + ]:CBC        6330 :     else if (ISALLTRUE(b))
 7131 teodor@sigaev.ru          232                 :UBC           0 :         *result = false;
                                233                 :                :     else
                                234                 :                :     {
                                235                 :                :         int32       i;
 7102 bruce@momjian.us          236                 :CBC        6330 :         BITVECP     sa = GETSIGN(a),
                                237                 :           6330 :                     sb = GETSIGN(b);
                                238                 :                : 
 7131 teodor@sigaev.ru          239                 :           6330 :         *result = true;
 2176 akorotkov@postgresql      240         [ +  + ]:        4732759 :         LOOPBYTE(siglen)
                                241                 :                :         {
 6694 bruce@momjian.us          242         [ +  + ]:        4731401 :             if (sa[i] != sb[i])
                                243                 :                :             {
                                244                 :           4972 :                 *result = false;
                                245                 :           4972 :                 break;
                                246                 :                :             }
                                247                 :                :         }
                                248                 :                :     }
 7131 teodor@sigaev.ru          249                 :           6330 :     PG_RETURN_POINTER(result);
                                250                 :                : }
                                251                 :                : 
                                252                 :                : static int32
 2176 akorotkov@postgresql      253                 :UBC           0 : sizebitvec(BITVECP sign, int siglen)
                                254                 :                : {
 5011 peter_e@gmx.net           255                 :              0 :     int32       size = 0,
                                256                 :                :                 i;
                                257                 :                : 
 2176 akorotkov@postgresql      258         [ #  # ]:              0 :     LOOPBYTE(siglen)
                                259                 :                :     {
 6694 bruce@momjian.us          260                 :              0 :         size += SUMBIT(sign);
                                261                 :              0 :         sign = (BITVECP) (((char *) sign) + 1);
                                262                 :                :     }
 7131 teodor@sigaev.ru          263                 :              0 :     return size;
                                264                 :                : }
                                265                 :                : 
                                266                 :                : static int
 2176 akorotkov@postgresql      267                 :CBC      133266 : hemdistsign(BITVECP a, BITVECP b, int siglen)
                                268                 :                : {
                                269                 :                :     int         i,
 7102 bruce@momjian.us          270                 :         133266 :                 dist = 0;
                                271                 :                : 
 2176 akorotkov@postgresql      272         [ +  + ]:      400189202 :     LOOPBIT(siglen)
                                273                 :                :     {
 6694 bruce@momjian.us          274         [ +  + ]:      400055936 :         if (GETBIT(a, i) != GETBIT(b, i))
 7131 teodor@sigaev.ru          275                 :        3435598 :             dist++;
                                276                 :                :     }
                                277                 :         133266 :     return dist;
                                278                 :                : }
                                279                 :                : 
                                280                 :                : static int
 2176 akorotkov@postgresql      281                 :         133266 : hemdist(GISTTYPE *a, GISTTYPE *b, int siglen)
                                282                 :                : {
 7102 bruce@momjian.us          283         [ -  + ]:         133266 :     if (ISALLTRUE(a))
                                284                 :                :     {
 7131 teodor@sigaev.ru          285         [ #  # ]:UBC           0 :         if (ISALLTRUE(b))
                                286                 :              0 :             return 0;
                                287                 :                :         else
 2176 akorotkov@postgresql      288                 :              0 :             return SIGLENBIT(siglen) - sizebitvec(GETSIGN(b), siglen);
                                289                 :                :     }
 7102 bruce@momjian.us          290         [ -  + ]:CBC      133266 :     else if (ISALLTRUE(b))
 2176 akorotkov@postgresql      291                 :UBC           0 :         return SIGLENBIT(siglen) - sizebitvec(GETSIGN(a), siglen);
                                292                 :                : 
 2176 akorotkov@postgresql      293                 :CBC      133266 :     return hemdistsign(GETSIGN(a), GETSIGN(b), siglen);
                                294                 :                : }
                                295                 :                : 
                                296                 :                : static int32
                                297                 :          12660 : unionkey(BITVECP sbase, GISTTYPE *add, int siglen)
                                298                 :                : {
                                299                 :                :     int32       i;
 7102 bruce@momjian.us          300                 :          12660 :     BITVECP     sadd = GETSIGN(add);
                                301                 :                : 
 7131 teodor@sigaev.ru          302         [ -  + ]:          12660 :     if (ISALLTRUE(add))
 7131 teodor@sigaev.ru          303                 :UBC           0 :         return 1;
 2176 akorotkov@postgresql      304         [ +  + ]:CBC    22435748 :     LOOPBYTE(siglen)
 6694 bruce@momjian.us          305                 :       22423088 :         sbase[i] |= sadd[i];
 7131 teodor@sigaev.ru          306                 :          12660 :     return 0;
                                307                 :                : }
                                308                 :                : 
                                309                 :                : Datum
 7102 bruce@momjian.us          310                 :           6330 : ghstore_union(PG_FUNCTION_ARGS)
                                311                 :                : {
                                312                 :           6330 :     GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
 5011 peter_e@gmx.net           313                 :           6330 :     int32       len = entryvec->n;
                                314                 :                : 
 7102 bruce@momjian.us          315                 :           6330 :     int        *size = (int *) PG_GETARG_POINTER(1);
 2176 akorotkov@postgresql      316         [ +  - ]:           6330 :     int         siglen = GET_SIGLEN();
                                317                 :                :     int32       i;
                                318                 :           6330 :     GISTTYPE   *result = ghstore_alloc(false, siglen, NULL);
                                319                 :           6330 :     BITVECP     base = GETSIGN(result);
                                320                 :                : 
 7102 bruce@momjian.us          321         [ +  + ]:          18990 :     for (i = 0; i < len; i++)
                                322                 :                :     {
 2176 akorotkov@postgresql      323         [ -  + ]:          12660 :         if (unionkey(base, GETENTRY(entryvec, i), siglen))
                                324                 :                :         {
 2176 akorotkov@postgresql      325                 :UBC           0 :             result->flag |= ALLISTRUE;
                                326                 :              0 :             SET_VARSIZE(result, CALCGTSIZE(ALLISTRUE, siglen));
 7131 teodor@sigaev.ru          327                 :              0 :             break;
                                328                 :                :         }
                                329                 :                :     }
                                330                 :                : 
 2176 akorotkov@postgresql      331                 :CBC        6330 :     *size = VARSIZE(result);
                                332                 :                : 
 7131 teodor@sigaev.ru          333                 :           6330 :     PG_RETURN_POINTER(result);
                                334                 :                : }
                                335                 :                : 
                                336                 :                : Datum
 7102 bruce@momjian.us          337                 :          15966 : ghstore_penalty(PG_FUNCTION_ARGS)
                                338                 :                : {
 7131 teodor@sigaev.ru          339                 :          15966 :     GISTENTRY  *origentry = (GISTENTRY *) PG_GETARG_POINTER(0); /* always ISSIGNKEY */
                                340                 :          15966 :     GISTENTRY  *newentry = (GISTENTRY *) PG_GETARG_POINTER(1);
 7102 bruce@momjian.us          341                 :          15966 :     float      *penalty = (float *) PG_GETARG_POINTER(2);
 2176 akorotkov@postgresql      342         [ +  - ]:          15966 :     int         siglen = GET_SIGLEN();
 7131 teodor@sigaev.ru          343                 :          15966 :     GISTTYPE   *origval = (GISTTYPE *) DatumGetPointer(origentry->key);
                                344                 :          15966 :     GISTTYPE   *newval = (GISTTYPE *) DatumGetPointer(newentry->key);
                                345                 :                : 
 2176 akorotkov@postgresql      346                 :          15966 :     *penalty = hemdist(origval, newval, siglen);
 7131 teodor@sigaev.ru          347                 :          15966 :     PG_RETURN_POINTER(penalty);
                                348                 :                : }
                                349                 :                : 
                                350                 :                : 
                                351                 :                : typedef struct
                                352                 :                : {
                                353                 :                :     OffsetNumber pos;
                                354                 :                :     int32       cost;
                                355                 :                : } SPLITCOST;
                                356                 :                : 
                                357                 :                : static int
 7102 bruce@momjian.us          358                 :           6478 : comparecost(const void *a, const void *b)
                                359                 :                : {
  758 nathan@postgresql.or      360                 :          12956 :     return pg_cmp_s32(((const SPLITCOST *) a)->cost,
                                361                 :           6478 :                       ((const SPLITCOST *) b)->cost);
                                362                 :                : }
                                363                 :                : 
                                364                 :                : 
                                365                 :                : Datum
 7102 bruce@momjian.us          366                 :            720 : ghstore_picksplit(PG_FUNCTION_ARGS)
                                367                 :                : {
                                368                 :            720 :     GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
 7131 teodor@sigaev.ru          369                 :            720 :     OffsetNumber maxoff = entryvec->n - 2;
                                370                 :                : 
                                371                 :            720 :     GIST_SPLITVEC *v = (GIST_SPLITVEC *) PG_GETARG_POINTER(1);
 2176 akorotkov@postgresql      372         [ +  - ]:            720 :     int         siglen = GET_SIGLEN();
                                373                 :                :     OffsetNumber k,
                                374                 :                :                 j;
                                375                 :                :     GISTTYPE   *datum_l,
                                376                 :                :                *datum_r;
                                377                 :                :     BITVECP     union_l,
                                378                 :                :                 union_r;
                                379                 :                :     int32       size_alpha,
                                380                 :                :                 size_beta;
                                381                 :                :     int32       size_waste,
 7131 teodor@sigaev.ru          382                 :            720 :                 waste = -1;
                                383                 :                :     int32       nbytes;
                                384                 :            720 :     OffsetNumber seed_1 = 0,
                                385                 :            720 :                 seed_2 = 0;
                                386                 :                :     OffsetNumber *left,
                                387                 :                :                *right;
                                388                 :                :     BITVECP     ptr;
                                389                 :                :     int         i;
                                390                 :                :     SPLITCOST  *costvector;
                                391                 :                :     GISTTYPE   *_k,
                                392                 :                :                *_j;
                                393                 :                : 
                                394                 :            720 :     nbytes = (maxoff + 2) * sizeof(OffsetNumber);
                                395                 :            720 :     v->spl_left = (OffsetNumber *) palloc(nbytes);
                                396                 :            720 :     v->spl_right = (OffsetNumber *) palloc(nbytes);
                                397                 :                : 
 7102 bruce@momjian.us          398         [ +  + ]:           3160 :     for (k = FirstOffsetNumber; k < maxoff; k = OffsetNumberNext(k))
                                399                 :                :     {
 7131 teodor@sigaev.ru          400                 :           2440 :         _k = GETENTRY(entryvec, k);
 7102 bruce@momjian.us          401         [ +  + ]:         107100 :         for (j = OffsetNumberNext(k); j <= maxoff; j = OffsetNumberNext(j))
                                402                 :                :         {
 2176 akorotkov@postgresql      403                 :         104660 :             size_waste = hemdist(_k, GETENTRY(entryvec, j), siglen);
 7102 bruce@momjian.us          404         [ +  + ]:         104660 :             if (size_waste > waste)
                                405                 :                :             {
 7131 teodor@sigaev.ru          406                 :           1158 :                 waste = size_waste;
                                407                 :           1158 :                 seed_1 = k;
                                408                 :           1158 :                 seed_2 = j;
                                409                 :                :             }
                                410                 :                :         }
                                411                 :                :     }
                                412                 :                : 
                                413                 :            720 :     left = v->spl_left;
                                414                 :            720 :     v->spl_nleft = 0;
                                415                 :            720 :     right = v->spl_right;
                                416                 :            720 :     v->spl_nright = 0;
                                417                 :                : 
                                418   [ +  -  -  + ]:            720 :     if (seed_1 == 0 || seed_2 == 0)
                                419                 :                :     {
 7131 teodor@sigaev.ru          420                 :UBC           0 :         seed_1 = 1;
                                421                 :              0 :         seed_2 = 2;
                                422                 :                :     }
                                423                 :                : 
                                424                 :                :     /* form initial .. */
 2176 akorotkov@postgresql      425                 :CBC         720 :     datum_l = ghstore_alloc(ISALLTRUE(GETENTRY(entryvec, seed_1)), siglen,
                                426                 :            720 :                             GETSIGN(GETENTRY(entryvec, seed_1)));
                                427                 :            720 :     datum_r = ghstore_alloc(ISALLTRUE(GETENTRY(entryvec, seed_2)), siglen,
                                428                 :            720 :                             GETSIGN(GETENTRY(entryvec, seed_2)));
                                429                 :                : 
 7131 teodor@sigaev.ru          430                 :            720 :     maxoff = OffsetNumberNext(maxoff);
                                431                 :                :     /* sort before ... */
  100 michael@paquier.xyz       432                 :GNC         720 :     costvector = palloc_array(SPLITCOST, maxoff);
 7131 teodor@sigaev.ru          433         [ +  + ]:CBC        4600 :     for (j = FirstOffsetNumber; j <= maxoff; j = OffsetNumberNext(j))
                                434                 :                :     {
                                435                 :           3880 :         costvector[j - 1].pos = j;
                                436                 :           3880 :         _j = GETENTRY(entryvec, j);
 2176 akorotkov@postgresql      437                 :           3880 :         size_alpha = hemdist(datum_l, _j, siglen);
                                438                 :           3880 :         size_beta = hemdist(datum_r, _j, siglen);
 7131 teodor@sigaev.ru          439                 :           3880 :         costvector[j - 1].cost = abs(size_alpha - size_beta);
                                440                 :                :     }
 1132 peter@eisentraut.org      441                 :            720 :     qsort(costvector, maxoff, sizeof(SPLITCOST), comparecost);
                                442                 :                : 
 7102 bruce@momjian.us          443                 :            720 :     union_l = GETSIGN(datum_l);
                                444                 :            720 :     union_r = GETSIGN(datum_r);
                                445                 :                : 
                                446         [ +  + ]:           4600 :     for (k = 0; k < maxoff; k++)
                                447                 :                :     {
 7131 teodor@sigaev.ru          448                 :           3880 :         j = costvector[k].pos;
 7102 bruce@momjian.us          449         [ +  + ]:           3880 :         if (j == seed_1)
                                450                 :                :         {
 7131 teodor@sigaev.ru          451                 :            720 :             *left++ = j;
                                452                 :            720 :             v->spl_nleft++;
                                453                 :            720 :             continue;
                                454                 :                :         }
 7102 bruce@momjian.us          455         [ +  + ]:           3160 :         else if (j == seed_2)
                                456                 :                :         {
 7131 teodor@sigaev.ru          457                 :            720 :             *right++ = j;
                                458                 :            720 :             v->spl_nright++;
                                459                 :            720 :             continue;
                                460                 :                :         }
                                461                 :           2440 :         _j = GETENTRY(entryvec, j);
 2176 akorotkov@postgresql      462                 :           2440 :         size_alpha = hemdist(datum_l, _j, siglen);
                                463                 :           2440 :         size_beta = hemdist(datum_r, _j, siglen);
                                464                 :                : 
 7102 bruce@momjian.us          465         [ +  + ]:           2440 :         if (size_alpha < size_beta + WISH_F(v->spl_nleft, v->spl_nright, 0.0001))
                                466                 :                :         {
                                467   [ +  -  -  + ]:           1125 :             if (ISALLTRUE(datum_l) || ISALLTRUE(_j))
                                468                 :                :             {
 7131 teodor@sigaev.ru          469         [ #  # ]:UBC           0 :                 if (!ISALLTRUE(datum_l))
 1132 peter@eisentraut.org      470                 :              0 :                     memset(union_l, 0xff, siglen);
                                471                 :                :             }
                                472                 :                :             else
                                473                 :                :             {
 7102 bruce@momjian.us          474                 :CBC        1125 :                 ptr = GETSIGN(_j);
 2176 akorotkov@postgresql      475         [ +  + ]:        1320309 :                 LOOPBYTE(siglen)
 6694 bruce@momjian.us          476                 :        1319184 :                     union_l[i] |= ptr[i];
                                477                 :                :             }
 7131 teodor@sigaev.ru          478                 :           1125 :             *left++ = j;
                                479                 :           1125 :             v->spl_nleft++;
                                480                 :                :         }
                                481                 :                :         else
                                482                 :                :         {
 7102 bruce@momjian.us          483   [ +  -  -  + ]:           1315 :             if (ISALLTRUE(datum_r) || ISALLTRUE(_j))
                                484                 :                :             {
 7131 teodor@sigaev.ru          485         [ #  # ]:UBC           0 :                 if (!ISALLTRUE(datum_r))
 1132 peter@eisentraut.org      486                 :              0 :                     memset(union_r, 0xff, siglen);
                                487                 :                :             }
                                488                 :                :             else
                                489                 :                :             {
 7102 bruce@momjian.us          490                 :CBC        1315 :                 ptr = GETSIGN(_j);
 2176 akorotkov@postgresql      491         [ +  + ]:        1592611 :                 LOOPBYTE(siglen)
 6694 bruce@momjian.us          492                 :        1591296 :                     union_r[i] |= ptr[i];
                                493                 :                :             }
 7131 teodor@sigaev.ru          494                 :           1315 :             *right++ = j;
                                495                 :           1315 :             v->spl_nright++;
                                496                 :                :         }
                                497                 :                :     }
                                498                 :                : 
                                499                 :            720 :     *right = *left = FirstOffsetNumber;
                                500                 :                : 
                                501                 :            720 :     v->spl_ldatum = PointerGetDatum(datum_l);
                                502                 :            720 :     v->spl_rdatum = PointerGetDatum(datum_r);
                                503                 :                : 
                                504                 :            720 :     PG_RETURN_POINTER(v);
                                505                 :                : }
                                506                 :                : 
                                507                 :                : 
                                508                 :                : Datum
 7102 bruce@momjian.us          509                 :          10522 : ghstore_consistent(PG_FUNCTION_ARGS)
                                510                 :                : {
                                511                 :          10522 :     GISTTYPE   *entry = (GISTTYPE *) DatumGetPointer(((GISTENTRY *) PG_GETARG_POINTER(0))->key);
 6941 teodor@sigaev.ru          512                 :          10522 :     StrategyNumber strategy = (StrategyNumber) PG_GETARG_UINT16(2);
                                513                 :                : #ifdef NOT_USED
                                514                 :                :     Oid         subtype = PG_GETARG_OID(3);
                                515                 :                : #endif
 6544 tgl@sss.pgh.pa.us         516                 :          10522 :     bool       *recheck = (bool *) PG_GETARG_POINTER(4);
 2176 akorotkov@postgresql      517         [ +  - ]:          10522 :     int         siglen = GET_SIGLEN();
 7102 bruce@momjian.us          518                 :          10522 :     bool        res = true;
                                519                 :                :     BITVECP     sign;
                                520                 :                : 
                                521                 :                :     /* All cases served by this function are inexact */
 6544 tgl@sss.pgh.pa.us         522                 :          10522 :     *recheck = true;
                                523                 :                : 
 7102 bruce@momjian.us          524         [ -  + ]:          10522 :     if (ISALLTRUE(entry))
 7131 teodor@sigaev.ru          525                 :UBC           0 :         PG_RETURN_BOOL(true);
                                526                 :                : 
 7102 bruce@momjian.us          527                 :CBC       10522 :     sign = GETSIGN(entry);
                                528                 :                : 
 6010 tgl@sss.pgh.pa.us         529   [ +  +  -  + ]:          10522 :     if (strategy == HStoreContainsStrategyNumber ||
                                530                 :                :         strategy == HStoreOldContainsStrategyNumber)
 7102 bruce@momjian.us          531                 :           4723 :     {
 3100 tgl@sss.pgh.pa.us         532                 :           4723 :         HStore     *query = PG_GETARG_HSTORE_P(1);
 6941 teodor@sigaev.ru          533                 :           4723 :         HEntry     *qe = ARRPTR(query);
                                534                 :           4723 :         char       *qv = STRPTR(query);
 5861 bruce@momjian.us          535                 :           4723 :         int         count = HS_COUNT(query);
                                536                 :                :         int         i;
                                537                 :                : 
 6010 tgl@sss.pgh.pa.us         538   [ +  +  +  + ]:           9657 :         for (i = 0; res && i < count; ++i)
                                539                 :                :         {
 3769                           540         [ +  + ]:           4934 :             int         crc = crc32_sz((char *) HSTORE_KEY(qe, qv, i),
                                541         [ +  + ]:           4934 :                                        HSTORE_KEYLEN(qe, i));
                                542                 :                : 
 2176 akorotkov@postgresql      543         [ +  + ]:           4934 :             if (GETBIT(sign, HASHVAL(crc, siglen)))
                                544                 :                :             {
 3769 tgl@sss.pgh.pa.us         545         [ +  + ]:           1969 :                 if (!HSTORE_VALISNULL(qe, i))
                                546                 :                :                 {
                                547         [ +  - ]:           1154 :                     crc = crc32_sz((char *) HSTORE_VAL(qe, qv, i),
                                548         [ -  + ]:           1154 :                                    HSTORE_VALLEN(qe, i));
 2176 akorotkov@postgresql      549         [ +  + ]:           1154 :                     if (!GETBIT(sign, HASHVAL(crc, siglen)))
 6941 teodor@sigaev.ru          550                 :            557 :                         res = false;
                                551                 :                :                 }
                                552                 :                :             }
                                553                 :                :             else
                                554                 :           2965 :                 res = false;
                                555                 :                :         }
                                556                 :                :     }
 6694 bruce@momjian.us          557         [ +  + ]:           5799 :     else if (strategy == HStoreExistsStrategyNumber)
                                558                 :                :     {
 6010 tgl@sss.pgh.pa.us         559                 :           1961 :         text       *query = PG_GETARG_TEXT_PP(1);
                                560   [ -  +  -  -  :           1961 :         int         crc = crc32_sz(VARDATA_ANY(query), VARSIZE_ANY_EXHDR(query));
                                     -  -  -  -  -  
                                           +  -  + ]
                                561                 :                : 
 2176 akorotkov@postgresql      562                 :           1961 :         res = (GETBIT(sign, HASHVAL(crc, siglen))) ? true : false;
                                563                 :                :     }
 6010 tgl@sss.pgh.pa.us         564         [ +  + ]:           3838 :     else if (strategy == HStoreExistsAllStrategyNumber)
                                565                 :                :     {
 5861 bruce@momjian.us          566                 :           1580 :         ArrayType  *query = PG_GETARG_ARRAYTYPE_P(1);
                                567                 :                :         Datum      *key_datums;
                                568                 :                :         bool       *key_nulls;
                                569                 :                :         int         key_count;
                                570                 :                :         int         i;
                                571                 :                : 
 1353 peter@eisentraut.org      572                 :           1580 :         deconstruct_array_builtin(query, TEXTOID, &key_datums, &key_nulls, &key_count);
                                573                 :                : 
 6010 tgl@sss.pgh.pa.us         574   [ +  +  +  + ]:           3780 :         for (i = 0; res && i < key_count; ++i)
                                575                 :                :         {
                                576                 :                :             int         crc;
                                577                 :                : 
                                578         [ -  + ]:           2200 :             if (key_nulls[i])
 6010 tgl@sss.pgh.pa.us         579                 :UBC           0 :                 continue;
  222 peter@eisentraut.org      580                 :GNC        2200 :             crc = crc32_sz(VARDATA(DatumGetPointer(key_datums[i])), VARSIZE(DatumGetPointer(key_datums[i])) - VARHDRSZ);
 2176 akorotkov@postgresql      581         [ +  + ]:CBC        2200 :             if (!(GETBIT(sign, HASHVAL(crc, siglen))))
 3133 peter_e@gmx.net           582                 :           1260 :                 res = false;
                                583                 :                :         }
                                584                 :                :     }
 6010 tgl@sss.pgh.pa.us         585         [ +  - ]:           2258 :     else if (strategy == HStoreExistsAnyStrategyNumber)
                                586                 :                :     {
 5861 bruce@momjian.us          587                 :           2258 :         ArrayType  *query = PG_GETARG_ARRAYTYPE_P(1);
                                588                 :                :         Datum      *key_datums;
                                589                 :                :         bool       *key_nulls;
                                590                 :                :         int         key_count;
                                591                 :                :         int         i;
                                592                 :                : 
 1353 peter@eisentraut.org      593                 :           2258 :         deconstruct_array_builtin(query, TEXTOID, &key_datums, &key_nulls, &key_count);
                                594                 :                : 
 3133 peter_e@gmx.net           595                 :           2258 :         res = false;
                                596                 :                : 
 6010 tgl@sss.pgh.pa.us         597   [ +  +  +  + ]:           5956 :         for (i = 0; !res && i < key_count; ++i)
                                598                 :                :         {
                                599                 :                :             int         crc;
                                600                 :                : 
                                601         [ -  + ]:           3698 :             if (key_nulls[i])
 6010 tgl@sss.pgh.pa.us         602                 :UBC           0 :                 continue;
  222 peter@eisentraut.org      603                 :GNC        3698 :             crc = crc32_sz(VARDATA(DatumGetPointer(key_datums[i])), VARSIZE(DatumGetPointer(key_datums[i])) - VARHDRSZ);
 2176 akorotkov@postgresql      604         [ +  + ]:CBC        3698 :             if (GETBIT(sign, HASHVAL(crc, siglen)))
 3133 peter_e@gmx.net           605                 :           1232 :                 res = true;
                                606                 :                :         }
                                607                 :                :     }
                                608                 :                :     else
 6941 teodor@sigaev.ru          609         [ #  # ]:UBC           0 :         elog(ERROR, "Unsupported strategy number: %d", strategy);
                                610                 :                : 
 7131 teodor@sigaev.ru          611                 :CBC       10522 :     PG_RETURN_BOOL(res);
                                612                 :                : }
                                613                 :                : 
                                614                 :                : Datum
 2176 akorotkov@postgresql      615                 :             10 : ghstore_options(PG_FUNCTION_ARGS)
                                616                 :                : {
                                617                 :             10 :     local_relopts *relopts = (local_relopts *) PG_GETARG_POINTER(0);
                                618                 :                : 
                                619                 :             10 :     init_local_reloptions(relopts, sizeof(GistHstoreOptions));
                                620                 :             10 :     add_local_int_reloption(relopts, "siglen",
                                621                 :                :                             "signature length in bytes",
                                622                 :                :                             SIGLEN_DEFAULT, 1, SIGLEN_MAX,
                                623                 :                :                             offsetof(GistHstoreOptions, siglen));
                                624                 :                : 
                                625                 :             10 :     PG_RETURN_VOID();
                                626                 :                : }
        

Generated by: LCOV version 2.4-beta