LCOV - code coverage report
Current view: top level - contrib/hstore - hstore_op.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 522 571 91.4 %
Date: 2025-01-18 05:15:39 Functions: 57 86 66.3 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*
       2             :  * contrib/hstore/hstore_op.c
       3             :  */
       4             : #include "postgres.h"
       5             : 
       6             : #include "access/htup_details.h"
       7             : #include "catalog/pg_type.h"
       8             : #include "common/hashfn.h"
       9             : #include "funcapi.h"
      10             : #include "hstore.h"
      11             : #include "utils/builtins.h"
      12             : #include "utils/memutils.h"
      13             : 
      14             : /* old names for C functions */
      15           0 : HSTORE_POLLUTE(hstore_fetchval, fetchval);
      16           0 : HSTORE_POLLUTE(hstore_exists, exists);
      17           0 : HSTORE_POLLUTE(hstore_defined, defined);
      18           0 : HSTORE_POLLUTE(hstore_delete, delete);
      19           0 : HSTORE_POLLUTE(hstore_concat, hs_concat);
      20           0 : HSTORE_POLLUTE(hstore_contains, hs_contains);
      21           0 : HSTORE_POLLUTE(hstore_contained, hs_contained);
      22           0 : HSTORE_POLLUTE(hstore_akeys, akeys);
      23           0 : HSTORE_POLLUTE(hstore_avals, avals);
      24           0 : HSTORE_POLLUTE(hstore_skeys, skeys);
      25           0 : HSTORE_POLLUTE(hstore_svals, svals);
      26           0 : HSTORE_POLLUTE(hstore_each, each);
      27             : 
      28             : 
      29             : /*
      30             :  * We're often finding a sequence of keys in ascending order. The
      31             :  * "lowbound" parameter is used to cache lower bounds of searches
      32             :  * between calls, based on this assumption. Pass NULL for it for
      33             :  * one-off or unordered searches.
      34             :  */
      35             : int
      36       18912 : hstoreFindKey(HStore *hs, int *lowbound, char *key, int keylen)
      37             : {
      38       18912 :     HEntry     *entries = ARRPTR(hs);
      39       18912 :     int         stopLow = lowbound ? *lowbound : 0;
      40       18912 :     int         stopHigh = HS_COUNT(hs);
      41             :     int         stopMiddle;
      42       18912 :     char       *base = STRPTR(hs);
      43             : 
      44       49698 :     while (stopLow < stopHigh)
      45             :     {
      46             :         int         difference;
      47             : 
      48       37112 :         stopMiddle = stopLow + (stopHigh - stopLow) / 2;
      49             : 
      50       37112 :         if (HSTORE_KEYLEN(entries, stopMiddle) == keylen)
      51       14826 :             difference = memcmp(HSTORE_KEY(entries, base, stopMiddle), key, keylen);
      52             :         else
      53       22286 :             difference = (HSTORE_KEYLEN(entries, stopMiddle) > keylen) ? 1 : -1;
      54             : 
      55       37112 :         if (difference == 0)
      56             :         {
      57        6326 :             if (lowbound)
      58        5038 :                 *lowbound = stopMiddle + 1;
      59        6326 :             return stopMiddle;
      60             :         }
      61       30786 :         else if (difference < 0)
      62       16456 :             stopLow = stopMiddle + 1;
      63             :         else
      64       14330 :             stopHigh = stopMiddle;
      65             :     }
      66             : 
      67       12586 :     if (lowbound)
      68       10658 :         *lowbound = stopLow;
      69       12586 :     return -1;
      70             : }
      71             : 
      72             : Pairs *
      73        5684 : hstoreArrayToPairs(ArrayType *a, int *npairs)
      74             : {
      75             :     Datum      *key_datums;
      76             :     bool       *key_nulls;
      77             :     int         key_count;
      78             :     Pairs      *key_pairs;
      79             :     int         bufsiz;
      80             :     int         i,
      81             :                 j;
      82             : 
      83        5684 :     deconstruct_array_builtin(a, TEXTOID, &key_datums, &key_nulls, &key_count);
      84             : 
      85        5684 :     if (key_count == 0)
      86             :     {
      87          10 :         *npairs = 0;
      88          10 :         return NULL;
      89             :     }
      90             : 
      91             :     /*
      92             :      * A text array uses at least eight bytes per element, so any overflow in
      93             :      * "key_count * sizeof(Pairs)" is small enough for palloc() to catch.
      94             :      * However, credible improvements to the array format could invalidate
      95             :      * that assumption.  Therefore, use an explicit check rather than relying
      96             :      * on palloc() to complain.
      97             :      */
      98        5674 :     if (key_count > MaxAllocSize / sizeof(Pairs))
      99           0 :         ereport(ERROR,
     100             :                 (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
     101             :                  errmsg("number of pairs (%d) exceeds the maximum allowed (%d)",
     102             :                         key_count, (int) (MaxAllocSize / sizeof(Pairs)))));
     103             : 
     104        5674 :     key_pairs = palloc(sizeof(Pairs) * key_count);
     105             : 
     106       17032 :     for (i = 0, j = 0; i < key_count; i++)
     107             :     {
     108       11358 :         if (!key_nulls[i])
     109             :         {
     110       11358 :             key_pairs[j].key = VARDATA(key_datums[i]);
     111       11358 :             key_pairs[j].keylen = VARSIZE(key_datums[i]) - VARHDRSZ;
     112       11358 :             key_pairs[j].val = NULL;
     113       11358 :             key_pairs[j].vallen = 0;
     114       11358 :             key_pairs[j].needfree = 0;
     115       11358 :             key_pairs[j].isnull = 1;
     116       11358 :             j++;
     117             :         }
     118             :     }
     119             : 
     120        5674 :     *npairs = hstoreUniquePairs(key_pairs, j, &bufsiz);
     121             : 
     122        5674 :     return key_pairs;
     123             : }
     124             : 
     125             : 
     126          16 : PG_FUNCTION_INFO_V1(hstore_fetchval);
     127             : Datum
     128          12 : hstore_fetchval(PG_FUNCTION_ARGS)
     129             : {
     130          12 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
     131          12 :     text       *key = PG_GETARG_TEXT_PP(1);
     132          12 :     HEntry     *entries = ARRPTR(hs);
     133             :     text       *out;
     134          24 :     int         idx = hstoreFindKey(hs, NULL,
     135          24 :                                     VARDATA_ANY(key), VARSIZE_ANY_EXHDR(key));
     136             : 
     137          12 :     if (idx < 0 || HSTORE_VALISNULL(entries, idx))
     138           4 :         PG_RETURN_NULL();
     139             : 
     140           8 :     out = cstring_to_text_with_len(HSTORE_VAL(entries, STRPTR(hs), idx),
     141           8 :                                    HSTORE_VALLEN(entries, idx));
     142             : 
     143           8 :     PG_RETURN_TEXT_P(out);
     144             : }
     145             : 
     146             : 
     147          32 : PG_FUNCTION_INFO_V1(hstore_exists);
     148             : Datum
     149        2880 : hstore_exists(PG_FUNCTION_ARGS)
     150             : {
     151        2880 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
     152        2880 :     text       *key = PG_GETARG_TEXT_PP(1);
     153        5760 :     int         idx = hstoreFindKey(hs, NULL,
     154        5760 :                                     VARDATA_ANY(key), VARSIZE_ANY_EXHDR(key));
     155             : 
     156        2880 :     PG_RETURN_BOOL(idx >= 0);
     157             : }
     158             : 
     159             : 
     160          16 : PG_FUNCTION_INFO_V1(hstore_exists_any);
     161             : Datum
     162        3454 : hstore_exists_any(PG_FUNCTION_ARGS)
     163             : {
     164        3454 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
     165        3454 :     ArrayType  *keys = PG_GETARG_ARRAYTYPE_P(1);
     166             :     int         nkeys;
     167        3454 :     Pairs      *key_pairs = hstoreArrayToPairs(keys, &nkeys);
     168             :     int         i;
     169        3454 :     int         lowbound = 0;
     170        3454 :     bool        res = false;
     171             : 
     172             :     /*
     173             :      * we exploit the fact that the pairs list is already sorted into strictly
     174             :      * increasing order to narrow the hstoreFindKey search; each search can
     175             :      * start one entry past the previous "found" entry, or at the lower bound
     176             :      * of the last search.
     177             :      */
     178        7160 :     for (i = 0; i < nkeys; i++)
     179             :     {
     180        5734 :         int         idx = hstoreFindKey(hs, &lowbound,
     181        5734 :                                         key_pairs[i].key, key_pairs[i].keylen);
     182             : 
     183        5734 :         if (idx >= 0)
     184             :         {
     185        2028 :             res = true;
     186        2028 :             break;
     187             :         }
     188             :     }
     189             : 
     190        3454 :     PG_RETURN_BOOL(res);
     191             : }
     192             : 
     193             : 
     194          16 : PG_FUNCTION_INFO_V1(hstore_exists_all);
     195             : Datum
     196        2194 : hstore_exists_all(PG_FUNCTION_ARGS)
     197             : {
     198        2194 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
     199        2194 :     ArrayType  *keys = PG_GETARG_ARRAYTYPE_P(1);
     200             :     int         nkeys;
     201        2194 :     Pairs      *key_pairs = hstoreArrayToPairs(keys, &nkeys);
     202             :     int         i;
     203        2194 :     int         lowbound = 0;
     204        2194 :     bool        res = true;
     205             : 
     206             :     /*
     207             :      * we exploit the fact that the pairs list is already sorted into strictly
     208             :      * increasing order to narrow the hstoreFindKey search; each search can
     209             :      * start one entry past the previous "found" entry, or at the lower bound
     210             :      * of the last search.
     211             :      */
     212        3014 :     for (i = 0; i < nkeys; i++)
     213             :     {
     214        2756 :         int         idx = hstoreFindKey(hs, &lowbound,
     215        2756 :                                         key_pairs[i].key, key_pairs[i].keylen);
     216             : 
     217        2756 :         if (idx < 0)
     218             :         {
     219        1936 :             res = false;
     220        1936 :             break;
     221             :         }
     222             :     }
     223             : 
     224        2194 :     PG_RETURN_BOOL(res);
     225             : }
     226             : 
     227             : 
     228          30 : PG_FUNCTION_INFO_V1(hstore_defined);
     229             : Datum
     230           8 : hstore_defined(PG_FUNCTION_ARGS)
     231             : {
     232           8 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
     233           8 :     text       *key = PG_GETARG_TEXT_PP(1);
     234           8 :     HEntry     *entries = ARRPTR(hs);
     235          16 :     int         idx = hstoreFindKey(hs, NULL,
     236          16 :                                     VARDATA_ANY(key), VARSIZE_ANY_EXHDR(key));
     237           8 :     bool        res = (idx >= 0 && !HSTORE_VALISNULL(entries, idx));
     238             : 
     239           8 :     PG_RETURN_BOOL(res);
     240             : }
     241             : 
     242             : 
     243          16 : PG_FUNCTION_INFO_V1(hstore_delete);
     244             : Datum
     245          22 : hstore_delete(PG_FUNCTION_ARGS)
     246             : {
     247          22 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
     248          22 :     text       *key = PG_GETARG_TEXT_PP(1);
     249          22 :     char       *keyptr = VARDATA_ANY(key);
     250          22 :     int         keylen = VARSIZE_ANY_EXHDR(key);
     251          22 :     HStore     *out = palloc(VARSIZE(hs));
     252             :     char       *bufs,
     253             :                *bufd,
     254             :                *ptrd;
     255             :     HEntry     *es,
     256             :                *ed;
     257             :     int         i;
     258          22 :     int         count = HS_COUNT(hs);
     259          22 :     int         outcount = 0;
     260             : 
     261          22 :     SET_VARSIZE(out, VARSIZE(hs));
     262          22 :     HS_SETCOUNT(out, count);    /* temporary! */
     263             : 
     264          22 :     bufs = STRPTR(hs);
     265          22 :     es = ARRPTR(hs);
     266          22 :     bufd = ptrd = STRPTR(out);
     267          22 :     ed = ARRPTR(out);
     268             : 
     269          88 :     for (i = 0; i < count; ++i)
     270             :     {
     271          66 :         int         len = HSTORE_KEYLEN(es, i);
     272          66 :         char       *ptrs = HSTORE_KEY(es, bufs, i);
     273             : 
     274          66 :         if (!(len == keylen && memcmp(ptrs, keyptr, keylen) == 0))
     275             :         {
     276          48 :             int         vallen = HSTORE_VALLEN(es, i);
     277             : 
     278          48 :             HS_COPYITEM(ed, bufd, ptrd, ptrs, len, vallen,
     279             :                         HSTORE_VALISNULL(es, i));
     280          48 :             ++outcount;
     281             :         }
     282             :     }
     283             : 
     284          22 :     HS_FINALIZE(out, outcount, bufd, ptrd);
     285             : 
     286          22 :     PG_RETURN_POINTER(out);
     287             : }
     288             : 
     289             : 
     290          16 : PG_FUNCTION_INFO_V1(hstore_delete_array);
     291             : Datum
     292          24 : hstore_delete_array(PG_FUNCTION_ARGS)
     293             : {
     294          24 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
     295          24 :     HStore     *out = palloc(VARSIZE(hs));
     296          24 :     int         hs_count = HS_COUNT(hs);
     297             :     char       *ps,
     298             :                *bufd,
     299             :                *pd;
     300             :     HEntry     *es,
     301             :                *ed;
     302             :     int         i,
     303             :                 j;
     304          24 :     int         outcount = 0;
     305          24 :     ArrayType  *key_array = PG_GETARG_ARRAYTYPE_P(1);
     306             :     int         nkeys;
     307          24 :     Pairs      *key_pairs = hstoreArrayToPairs(key_array, &nkeys);
     308             : 
     309          24 :     SET_VARSIZE(out, VARSIZE(hs));
     310          24 :     HS_SETCOUNT(out, hs_count); /* temporary! */
     311             : 
     312          24 :     ps = STRPTR(hs);
     313          24 :     es = ARRPTR(hs);
     314          24 :     bufd = pd = STRPTR(out);
     315          24 :     ed = ARRPTR(out);
     316             : 
     317          24 :     if (nkeys == 0)
     318             :     {
     319             :         /* return a copy of the input, unchanged */
     320           6 :         memcpy(out, hs, VARSIZE(hs));
     321           6 :         HS_FIXSIZE(out, hs_count);
     322           6 :         HS_SETCOUNT(out, hs_count);
     323           6 :         PG_RETURN_POINTER(out);
     324             :     }
     325             : 
     326             :     /*
     327             :      * this is in effect a merge between hs and key_pairs, both of which are
     328             :      * already sorted by (keylen,key); we take keys from hs only
     329             :      */
     330             : 
     331          72 :     for (i = j = 0; i < hs_count;)
     332             :     {
     333             :         int         difference;
     334             : 
     335          54 :         if (j >= nkeys)
     336           0 :             difference = -1;
     337             :         else
     338             :         {
     339          54 :             int         skeylen = HSTORE_KEYLEN(es, i);
     340             : 
     341          54 :             if (skeylen == key_pairs[j].keylen)
     342          54 :                 difference = memcmp(HSTORE_KEY(es, ps, i),
     343          54 :                                     key_pairs[j].key,
     344          54 :                                     key_pairs[j].keylen);
     345             :             else
     346           0 :                 difference = (skeylen > key_pairs[j].keylen) ? 1 : -1;
     347             :         }
     348             : 
     349          54 :         if (difference > 0)
     350           0 :             ++j;
     351          54 :         else if (difference == 0)
     352          28 :             ++i, ++j;
     353             :         else
     354             :         {
     355          26 :             HS_COPYITEM(ed, bufd, pd,
     356             :                         HSTORE_KEY(es, ps, i), HSTORE_KEYLEN(es, i),
     357             :                         HSTORE_VALLEN(es, i), HSTORE_VALISNULL(es, i));
     358          26 :             ++outcount;
     359          26 :             ++i;
     360             :         }
     361             :     }
     362             : 
     363          18 :     HS_FINALIZE(out, outcount, bufd, pd);
     364             : 
     365          18 :     PG_RETURN_POINTER(out);
     366             : }
     367             : 
     368             : 
     369          16 : PG_FUNCTION_INFO_V1(hstore_delete_hstore);
     370             : Datum
     371          24 : hstore_delete_hstore(PG_FUNCTION_ARGS)
     372             : {
     373          24 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
     374          24 :     HStore     *hs2 = PG_GETARG_HSTORE_P(1);
     375          24 :     HStore     *out = palloc(VARSIZE(hs));
     376          24 :     int         hs_count = HS_COUNT(hs);
     377          24 :     int         hs2_count = HS_COUNT(hs2);
     378             :     char       *ps,
     379             :                *ps2,
     380             :                *bufd,
     381             :                *pd;
     382             :     HEntry     *es,
     383             :                *es2,
     384             :                *ed;
     385             :     int         i,
     386             :                 j;
     387          24 :     int         outcount = 0;
     388             : 
     389          24 :     SET_VARSIZE(out, VARSIZE(hs));
     390          24 :     HS_SETCOUNT(out, hs_count); /* temporary! */
     391             : 
     392          24 :     ps = STRPTR(hs);
     393          24 :     es = ARRPTR(hs);
     394          24 :     ps2 = STRPTR(hs2);
     395          24 :     es2 = ARRPTR(hs2);
     396          24 :     bufd = pd = STRPTR(out);
     397          24 :     ed = ARRPTR(out);
     398             : 
     399          24 :     if (hs2_count == 0)
     400             :     {
     401             :         /* return a copy of the input, unchanged */
     402           6 :         memcpy(out, hs, VARSIZE(hs));
     403           6 :         HS_FIXSIZE(out, hs_count);
     404           6 :         HS_SETCOUNT(out, hs_count);
     405           6 :         PG_RETURN_POINTER(out);
     406             :     }
     407             : 
     408             :     /*
     409             :      * this is in effect a merge between hs and hs2, both of which are already
     410             :      * sorted by (keylen,key); we take keys from hs only; for equal keys, we
     411             :      * take the value from hs unless the values are equal
     412             :      */
     413             : 
     414          72 :     for (i = j = 0; i < hs_count;)
     415             :     {
     416             :         int         difference;
     417             : 
     418          54 :         if (j >= hs2_count)
     419          10 :             difference = -1;
     420             :         else
     421             :         {
     422          44 :             int         skeylen = HSTORE_KEYLEN(es, i);
     423          44 :             int         s2keylen = HSTORE_KEYLEN(es2, j);
     424             : 
     425          44 :             if (skeylen == s2keylen)
     426          40 :                 difference = memcmp(HSTORE_KEY(es, ps, i),
     427          40 :                                     HSTORE_KEY(es2, ps2, j),
     428             :                                     skeylen);
     429             :             else
     430           4 :                 difference = (skeylen > s2keylen) ? 1 : -1;
     431             :         }
     432             : 
     433          54 :         if (difference > 0)
     434           0 :             ++j;
     435          54 :         else if (difference == 0)
     436             :         {
     437          34 :             int         svallen = HSTORE_VALLEN(es, i);
     438          34 :             int         snullval = HSTORE_VALISNULL(es, i);
     439             : 
     440          34 :             if (snullval != HSTORE_VALISNULL(es2, j) ||
     441          30 :                 (!snullval && (svallen != HSTORE_VALLEN(es2, j) ||
     442          30 :                                memcmp(HSTORE_VAL(es, ps, i),
     443          30 :                                       HSTORE_VAL(es2, ps2, j),
     444             :                                       svallen) != 0)))
     445             :             {
     446           8 :                 HS_COPYITEM(ed, bufd, pd,
     447             :                             HSTORE_KEY(es, ps, i), HSTORE_KEYLEN(es, i),
     448             :                             svallen, snullval);
     449           8 :                 ++outcount;
     450             :             }
     451          34 :             ++i, ++j;
     452             :         }
     453             :         else
     454             :         {
     455          20 :             HS_COPYITEM(ed, bufd, pd,
     456             :                         HSTORE_KEY(es, ps, i), HSTORE_KEYLEN(es, i),
     457             :                         HSTORE_VALLEN(es, i), HSTORE_VALISNULL(es, i));
     458          20 :             ++outcount;
     459          20 :             ++i;
     460             :         }
     461             :     }
     462             : 
     463          18 :     HS_FINALIZE(out, outcount, bufd, pd);
     464             : 
     465          18 :     PG_RETURN_POINTER(out);
     466             : }
     467             : 
     468             : 
     469          16 : PG_FUNCTION_INFO_V1(hstore_concat);
     470             : Datum
     471          60 : hstore_concat(PG_FUNCTION_ARGS)
     472             : {
     473          60 :     HStore     *s1 = PG_GETARG_HSTORE_P(0);
     474          60 :     HStore     *s2 = PG_GETARG_HSTORE_P(1);
     475          60 :     HStore     *out = palloc(VARSIZE(s1) + VARSIZE(s2));
     476             :     char       *ps1,
     477             :                *ps2,
     478             :                *bufd,
     479             :                *pd;
     480             :     HEntry     *es1,
     481             :                *es2,
     482             :                *ed;
     483             :     int         s1idx;
     484             :     int         s2idx;
     485          60 :     int         s1count = HS_COUNT(s1);
     486          60 :     int         s2count = HS_COUNT(s2);
     487          60 :     int         outcount = 0;
     488             : 
     489          60 :     SET_VARSIZE(out, VARSIZE(s1) + VARSIZE(s2) - HSHRDSIZE);
     490          60 :     HS_SETCOUNT(out, s1count + s2count);
     491             : 
     492          60 :     if (s1count == 0)
     493             :     {
     494             :         /* return a copy of the input, unchanged */
     495           8 :         memcpy(out, s2, VARSIZE(s2));
     496           8 :         HS_FIXSIZE(out, s2count);
     497           8 :         HS_SETCOUNT(out, s2count);
     498           8 :         PG_RETURN_POINTER(out);
     499             :     }
     500             : 
     501          52 :     if (s2count == 0)
     502             :     {
     503             :         /* return a copy of the input, unchanged */
     504           4 :         memcpy(out, s1, VARSIZE(s1));
     505           4 :         HS_FIXSIZE(out, s1count);
     506           4 :         HS_SETCOUNT(out, s1count);
     507           4 :         PG_RETURN_POINTER(out);
     508             :     }
     509             : 
     510          48 :     ps1 = STRPTR(s1);
     511          48 :     ps2 = STRPTR(s2);
     512          48 :     bufd = pd = STRPTR(out);
     513          48 :     es1 = ARRPTR(s1);
     514          48 :     es2 = ARRPTR(s2);
     515          48 :     ed = ARRPTR(out);
     516             : 
     517             :     /*
     518             :      * this is in effect a merge between s1 and s2, both of which are already
     519             :      * sorted by (keylen,key); we take s2 for equal keys
     520             :      */
     521             : 
     522         180 :     for (s1idx = s2idx = 0; s1idx < s1count || s2idx < s2count; ++outcount)
     523             :     {
     524             :         int         difference;
     525             : 
     526         132 :         if (s1idx >= s1count)
     527          26 :             difference = 1;
     528         106 :         else if (s2idx >= s2count)
     529          14 :             difference = -1;
     530             :         else
     531             :         {
     532          92 :             int         s1keylen = HSTORE_KEYLEN(es1, s1idx);
     533          92 :             int         s2keylen = HSTORE_KEYLEN(es2, s2idx);
     534             : 
     535          92 :             if (s1keylen == s2keylen)
     536          82 :                 difference = memcmp(HSTORE_KEY(es1, ps1, s1idx),
     537          82 :                                     HSTORE_KEY(es2, ps2, s2idx),
     538             :                                     s1keylen);
     539             :             else
     540          10 :                 difference = (s1keylen > s2keylen) ? 1 : -1;
     541             :         }
     542             : 
     543         132 :         if (difference >= 0)
     544             :         {
     545          76 :             HS_COPYITEM(ed, bufd, pd,
     546             :                         HSTORE_KEY(es2, ps2, s2idx), HSTORE_KEYLEN(es2, s2idx),
     547             :                         HSTORE_VALLEN(es2, s2idx), HSTORE_VALISNULL(es2, s2idx));
     548          76 :             ++s2idx;
     549          76 :             if (difference == 0)
     550          38 :                 ++s1idx;
     551             :         }
     552             :         else
     553             :         {
     554          56 :             HS_COPYITEM(ed, bufd, pd,
     555             :                         HSTORE_KEY(es1, ps1, s1idx), HSTORE_KEYLEN(es1, s1idx),
     556             :                         HSTORE_VALLEN(es1, s1idx), HSTORE_VALISNULL(es1, s1idx));
     557          56 :             ++s1idx;
     558             :         }
     559             :     }
     560             : 
     561          48 :     HS_FINALIZE(out, outcount, bufd, pd);
     562             : 
     563          48 :     PG_RETURN_POINTER(out);
     564             : }
     565             : 
     566             : 
     567          16 : PG_FUNCTION_INFO_V1(hstore_slice_to_array);
     568             : Datum
     569           8 : hstore_slice_to_array(PG_FUNCTION_ARGS)
     570             : {
     571           8 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
     572           8 :     HEntry     *entries = ARRPTR(hs);
     573           8 :     char       *ptr = STRPTR(hs);
     574           8 :     ArrayType  *key_array = PG_GETARG_ARRAYTYPE_P(1);
     575             :     ArrayType  *aout;
     576             :     Datum      *key_datums;
     577             :     bool       *key_nulls;
     578             :     Datum      *out_datums;
     579             :     bool       *out_nulls;
     580             :     int         key_count;
     581             :     int         i;
     582             : 
     583           8 :     deconstruct_array_builtin(key_array, TEXTOID, &key_datums, &key_nulls, &key_count);
     584             : 
     585           8 :     if (key_count == 0)
     586             :     {
     587           0 :         aout = construct_empty_array(TEXTOID);
     588           0 :         PG_RETURN_POINTER(aout);
     589             :     }
     590             : 
     591           8 :     out_datums = palloc(sizeof(Datum) * key_count);
     592           8 :     out_nulls = palloc(sizeof(bool) * key_count);
     593             : 
     594          30 :     for (i = 0; i < key_count; ++i)
     595             :     {
     596          22 :         text       *key = (text *) DatumGetPointer(key_datums[i]);
     597             :         int         idx;
     598             : 
     599          22 :         if (key_nulls[i])
     600           2 :             idx = -1;
     601             :         else
     602          20 :             idx = hstoreFindKey(hs, NULL, VARDATA(key), VARSIZE(key) - VARHDRSZ);
     603             : 
     604          22 :         if (idx < 0 || HSTORE_VALISNULL(entries, idx))
     605             :         {
     606           4 :             out_nulls[i] = true;
     607           4 :             out_datums[i] = (Datum) 0;
     608             :         }
     609             :         else
     610             :         {
     611          36 :             out_datums[i] =
     612          18 :                 PointerGetDatum(cstring_to_text_with_len(HSTORE_VAL(entries, ptr, idx),
     613          18 :                                                          HSTORE_VALLEN(entries, idx)));
     614          18 :             out_nulls[i] = false;
     615             :         }
     616             :     }
     617             : 
     618           8 :     aout = construct_md_array(out_datums, out_nulls,
     619             :                               ARR_NDIM(key_array),
     620             :                               ARR_DIMS(key_array),
     621           8 :                               ARR_LBOUND(key_array),
     622             :                               TEXTOID, -1, false, TYPALIGN_INT);
     623             : 
     624           8 :     PG_RETURN_POINTER(aout);
     625             : }
     626             : 
     627             : 
     628          16 : PG_FUNCTION_INFO_V1(hstore_slice_to_hstore);
     629             : Datum
     630          12 : hstore_slice_to_hstore(PG_FUNCTION_ARGS)
     631             : {
     632          12 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
     633          12 :     HEntry     *entries = ARRPTR(hs);
     634          12 :     char       *ptr = STRPTR(hs);
     635          12 :     ArrayType  *key_array = PG_GETARG_ARRAYTYPE_P(1);
     636             :     HStore     *out;
     637             :     int         nkeys;
     638          12 :     Pairs      *key_pairs = hstoreArrayToPairs(key_array, &nkeys);
     639             :     Pairs      *out_pairs;
     640             :     int         bufsiz;
     641          12 :     int         lastidx = 0;
     642             :     int         i;
     643          12 :     int         out_count = 0;
     644             : 
     645          12 :     if (nkeys == 0)
     646             :     {
     647           0 :         out = hstorePairs(NULL, 0, 0);
     648           0 :         PG_RETURN_POINTER(out);
     649             :     }
     650             : 
     651             :     /* hstoreArrayToPairs() checked overflow */
     652          12 :     out_pairs = palloc(sizeof(Pairs) * nkeys);
     653          12 :     bufsiz = 0;
     654             : 
     655             :     /*
     656             :      * we exploit the fact that the pairs list is already sorted into strictly
     657             :      * increasing order to narrow the hstoreFindKey search; each search can
     658             :      * start one entry past the previous "found" entry, or at the lower bound
     659             :      * of the last search.
     660             :      */
     661             : 
     662          42 :     for (i = 0; i < nkeys; ++i)
     663             :     {
     664          30 :         int         idx = hstoreFindKey(hs, &lastidx,
     665          30 :                                         key_pairs[i].key, key_pairs[i].keylen);
     666             : 
     667          30 :         if (idx >= 0)
     668             :         {
     669          24 :             out_pairs[out_count].key = key_pairs[i].key;
     670          24 :             bufsiz += (out_pairs[out_count].keylen = key_pairs[i].keylen);
     671          24 :             out_pairs[out_count].val = HSTORE_VAL(entries, ptr, idx);
     672          24 :             bufsiz += (out_pairs[out_count].vallen = HSTORE_VALLEN(entries, idx));
     673          24 :             out_pairs[out_count].isnull = HSTORE_VALISNULL(entries, idx);
     674          24 :             out_pairs[out_count].needfree = false;
     675          24 :             ++out_count;
     676             :         }
     677             :     }
     678             : 
     679             :     /*
     680             :      * we don't use hstoreUniquePairs here because we know that the pairs list
     681             :      * is already sorted and uniq'ed.
     682             :      */
     683             : 
     684          12 :     out = hstorePairs(out_pairs, out_count, bufsiz);
     685             : 
     686          12 :     PG_RETURN_POINTER(out);
     687             : }
     688             : 
     689             : 
     690          16 : PG_FUNCTION_INFO_V1(hstore_akeys);
     691             : Datum
     692           6 : hstore_akeys(PG_FUNCTION_ARGS)
     693             : {
     694           6 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
     695             :     Datum      *d;
     696             :     ArrayType  *a;
     697           6 :     HEntry     *entries = ARRPTR(hs);
     698           6 :     char       *base = STRPTR(hs);
     699           6 :     int         count = HS_COUNT(hs);
     700             :     int         i;
     701             : 
     702           6 :     if (count == 0)
     703             :     {
     704           2 :         a = construct_empty_array(TEXTOID);
     705           2 :         PG_RETURN_POINTER(a);
     706             :     }
     707             : 
     708           4 :     d = (Datum *) palloc(sizeof(Datum) * count);
     709             : 
     710          14 :     for (i = 0; i < count; ++i)
     711             :     {
     712          10 :         text       *t = cstring_to_text_with_len(HSTORE_KEY(entries, base, i),
     713          10 :                                                  HSTORE_KEYLEN(entries, i));
     714             : 
     715          10 :         d[i] = PointerGetDatum(t);
     716             :     }
     717             : 
     718           4 :     a = construct_array_builtin(d, count, TEXTOID);
     719             : 
     720           4 :     PG_RETURN_POINTER(a);
     721             : }
     722             : 
     723             : 
     724          16 : PG_FUNCTION_INFO_V1(hstore_avals);
     725             : Datum
     726           8 : hstore_avals(PG_FUNCTION_ARGS)
     727             : {
     728           8 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
     729             :     Datum      *d;
     730             :     bool       *nulls;
     731             :     ArrayType  *a;
     732           8 :     HEntry     *entries = ARRPTR(hs);
     733           8 :     char       *base = STRPTR(hs);
     734           8 :     int         count = HS_COUNT(hs);
     735           8 :     int         lb = 1;
     736             :     int         i;
     737             : 
     738           8 :     if (count == 0)
     739             :     {
     740           2 :         a = construct_empty_array(TEXTOID);
     741           2 :         PG_RETURN_POINTER(a);
     742             :     }
     743             : 
     744           6 :     d = (Datum *) palloc(sizeof(Datum) * count);
     745           6 :     nulls = (bool *) palloc(sizeof(bool) * count);
     746             : 
     747          24 :     for (i = 0; i < count; ++i)
     748             :     {
     749          18 :         if (HSTORE_VALISNULL(entries, i))
     750             :         {
     751           2 :             d[i] = (Datum) 0;
     752           2 :             nulls[i] = true;
     753             :         }
     754             :         else
     755             :         {
     756          16 :             text       *item = cstring_to_text_with_len(HSTORE_VAL(entries, base, i),
     757          16 :                                                         HSTORE_VALLEN(entries, i));
     758             : 
     759          16 :             d[i] = PointerGetDatum(item);
     760          16 :             nulls[i] = false;
     761             :         }
     762             :     }
     763             : 
     764           6 :     a = construct_md_array(d, nulls, 1, &count, &lb,
     765             :                            TEXTOID, -1, false, TYPALIGN_INT);
     766             : 
     767           6 :     PG_RETURN_POINTER(a);
     768             : }
     769             : 
     770             : 
     771             : static ArrayType *
     772           8 : hstore_to_array_internal(HStore *hs, int ndims)
     773             : {
     774           8 :     HEntry     *entries = ARRPTR(hs);
     775           8 :     char       *base = STRPTR(hs);
     776           8 :     int         count = HS_COUNT(hs);
     777           8 :     int         out_size[2] = {0, 2};
     778           8 :     int         lb[2] = {1, 1};
     779             :     Datum      *out_datums;
     780             :     bool       *out_nulls;
     781             :     int         i;
     782             : 
     783             :     Assert(ndims < 3);
     784             : 
     785           8 :     if (count == 0 || ndims == 0)
     786           0 :         return construct_empty_array(TEXTOID);
     787             : 
     788           8 :     out_size[0] = count * 2 / ndims;
     789           8 :     out_datums = palloc(sizeof(Datum) * count * 2);
     790           8 :     out_nulls = palloc(sizeof(bool) * count * 2);
     791             : 
     792          40 :     for (i = 0; i < count; ++i)
     793             :     {
     794          32 :         text       *key = cstring_to_text_with_len(HSTORE_KEY(entries, base, i),
     795          32 :                                                    HSTORE_KEYLEN(entries, i));
     796             : 
     797          32 :         out_datums[i * 2] = PointerGetDatum(key);
     798          32 :         out_nulls[i * 2] = false;
     799             : 
     800          32 :         if (HSTORE_VALISNULL(entries, i))
     801             :         {
     802           8 :             out_datums[i * 2 + 1] = (Datum) 0;
     803           8 :             out_nulls[i * 2 + 1] = true;
     804             :         }
     805             :         else
     806             :         {
     807          24 :             text       *item = cstring_to_text_with_len(HSTORE_VAL(entries, base, i),
     808          24 :                                                         HSTORE_VALLEN(entries, i));
     809             : 
     810          24 :             out_datums[i * 2 + 1] = PointerGetDatum(item);
     811          24 :             out_nulls[i * 2 + 1] = false;
     812             :         }
     813             :     }
     814             : 
     815           8 :     return construct_md_array(out_datums, out_nulls,
     816             :                               ndims, out_size, lb,
     817             :                               TEXTOID, -1, false, TYPALIGN_INT);
     818             : }
     819             : 
     820          16 : PG_FUNCTION_INFO_V1(hstore_to_array);
     821             : Datum
     822           4 : hstore_to_array(PG_FUNCTION_ARGS)
     823             : {
     824           4 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
     825           4 :     ArrayType  *out = hstore_to_array_internal(hs, 1);
     826             : 
     827           4 :     PG_RETURN_POINTER(out);
     828             : }
     829             : 
     830          16 : PG_FUNCTION_INFO_V1(hstore_to_matrix);
     831             : Datum
     832           4 : hstore_to_matrix(PG_FUNCTION_ARGS)
     833             : {
     834           4 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
     835           4 :     ArrayType  *out = hstore_to_array_internal(hs, 2);
     836             : 
     837           4 :     PG_RETURN_POINTER(out);
     838             : }
     839             : 
     840             : /*
     841             :  * Common initialization function for the various set-returning
     842             :  * funcs. fcinfo is only passed if the function is to return a
     843             :  * composite; it will be used to look up the return tupledesc.
     844             :  * we stash a copy of the hstore in the multi-call context in
     845             :  * case it was originally toasted. (At least I assume that's why;
     846             :  * there was no explanatory comment in the original code. --AG)
     847             :  */
     848             : 
     849             : static void
     850        4020 : setup_firstcall(FuncCallContext *funcctx, HStore *hs,
     851             :                 FunctionCallInfo fcinfo)
     852             : {
     853             :     MemoryContext oldcontext;
     854             :     HStore     *st;
     855             : 
     856        4020 :     oldcontext = MemoryContextSwitchTo(funcctx->multi_call_memory_ctx);
     857             : 
     858        4020 :     st = (HStore *) palloc(VARSIZE(hs));
     859        4020 :     memcpy(st, hs, VARSIZE(hs));
     860             : 
     861        4020 :     funcctx->user_fctx = st;
     862             : 
     863        4020 :     if (fcinfo)
     864             :     {
     865             :         TupleDesc   tupdesc;
     866             : 
     867             :         /* Build a tuple descriptor for our result type */
     868        4006 :         if (get_call_result_type(fcinfo, NULL, &tupdesc) != TYPEFUNC_COMPOSITE)
     869           0 :             elog(ERROR, "return type must be a row type");
     870             : 
     871        4006 :         funcctx->tuple_desc = BlessTupleDesc(tupdesc);
     872             :     }
     873             : 
     874        4020 :     MemoryContextSwitchTo(oldcontext);
     875        4020 : }
     876             : 
     877             : 
     878          16 : PG_FUNCTION_INFO_V1(hstore_skeys);
     879             : Datum
     880          16 : hstore_skeys(PG_FUNCTION_ARGS)
     881             : {
     882             :     FuncCallContext *funcctx;
     883             :     HStore     *hs;
     884             :     int         i;
     885             : 
     886          16 :     if (SRF_IS_FIRSTCALL())
     887             :     {
     888           6 :         hs = PG_GETARG_HSTORE_P(0);
     889           6 :         funcctx = SRF_FIRSTCALL_INIT();
     890           6 :         setup_firstcall(funcctx, hs, NULL);
     891             :     }
     892             : 
     893          16 :     funcctx = SRF_PERCALL_SETUP();
     894          16 :     hs = (HStore *) funcctx->user_fctx;
     895          16 :     i = funcctx->call_cntr;
     896             : 
     897          16 :     if (i < HS_COUNT(hs))
     898             :     {
     899          10 :         HEntry     *entries = ARRPTR(hs);
     900             :         text       *item;
     901             : 
     902          10 :         item = cstring_to_text_with_len(HSTORE_KEY(entries, STRPTR(hs), i),
     903          10 :                                         HSTORE_KEYLEN(entries, i));
     904             : 
     905          10 :         SRF_RETURN_NEXT(funcctx, PointerGetDatum(item));
     906             :     }
     907             : 
     908           6 :     SRF_RETURN_DONE(funcctx);
     909             : }
     910             : 
     911             : 
     912          16 : PG_FUNCTION_INFO_V1(hstore_svals);
     913             : Datum
     914          26 : hstore_svals(PG_FUNCTION_ARGS)
     915             : {
     916             :     FuncCallContext *funcctx;
     917             :     HStore     *hs;
     918             :     int         i;
     919             : 
     920          26 :     if (SRF_IS_FIRSTCALL())
     921             :     {
     922           8 :         hs = PG_GETARG_HSTORE_P(0);
     923           8 :         funcctx = SRF_FIRSTCALL_INIT();
     924           8 :         setup_firstcall(funcctx, hs, NULL);
     925             :     }
     926             : 
     927          26 :     funcctx = SRF_PERCALL_SETUP();
     928          26 :     hs = (HStore *) funcctx->user_fctx;
     929          26 :     i = funcctx->call_cntr;
     930             : 
     931          26 :     if (i < HS_COUNT(hs))
     932             :     {
     933          18 :         HEntry     *entries = ARRPTR(hs);
     934             : 
     935          18 :         if (HSTORE_VALISNULL(entries, i))
     936             :         {
     937             :             ReturnSetInfo *rsi;
     938             : 
     939             :             /* ugly ugly ugly. why no macro for this? */
     940           2 :             (funcctx)->call_cntr++;
     941           2 :             rsi = (ReturnSetInfo *) fcinfo->resultinfo;
     942           2 :             rsi->isDone = ExprMultipleResult;
     943           2 :             PG_RETURN_NULL();
     944             :         }
     945             :         else
     946             :         {
     947             :             text       *item;
     948             : 
     949          16 :             item = cstring_to_text_with_len(HSTORE_VAL(entries, STRPTR(hs), i),
     950          16 :                                             HSTORE_VALLEN(entries, i));
     951             : 
     952          16 :             SRF_RETURN_NEXT(funcctx, PointerGetDatum(item));
     953             :         }
     954             :     }
     955             : 
     956           8 :     SRF_RETURN_DONE(funcctx);
     957             : }
     958             : 
     959             : 
     960          16 : PG_FUNCTION_INFO_V1(hstore_contains);
     961             : Datum
     962        7120 : hstore_contains(PG_FUNCTION_ARGS)
     963             : {
     964        7120 :     HStore     *val = PG_GETARG_HSTORE_P(0);
     965        7120 :     HStore     *tmpl = PG_GETARG_HSTORE_P(1);
     966        7120 :     bool        res = true;
     967        7120 :     HEntry     *te = ARRPTR(tmpl);
     968        7120 :     char       *tstr = STRPTR(tmpl);
     969        7120 :     HEntry     *ve = ARRPTR(val);
     970        7120 :     char       *vstr = STRPTR(val);
     971        7120 :     int         tcount = HS_COUNT(tmpl);
     972        7120 :     int         lastidx = 0;
     973             :     int         i;
     974             : 
     975             :     /*
     976             :      * we exploit the fact that keys in "tmpl" are in strictly increasing
     977             :      * order to narrow the hstoreFindKey search; each search can start one
     978             :      * entry past the previous "found" entry, or at the lower bound of the
     979             :      * search
     980             :      */
     981             : 
     982       14296 :     for (i = 0; res && i < tcount; ++i)
     983             :     {
     984        7176 :         int         idx = hstoreFindKey(val, &lastidx,
     985        7176 :                                         HSTORE_KEY(te, tstr, i),
     986        7176 :                                         HSTORE_KEYLEN(te, i));
     987             : 
     988        7176 :         if (idx >= 0)
     989             :         {
     990        2166 :             bool        nullval = HSTORE_VALISNULL(te, i);
     991        2166 :             int         vallen = HSTORE_VALLEN(te, i);
     992             : 
     993        2166 :             if (nullval != HSTORE_VALISNULL(ve, idx) ||
     994        1026 :                 (!nullval && (vallen != HSTORE_VALLEN(ve, idx) ||
     995         610 :                               memcmp(HSTORE_VAL(te, tstr, i),
     996         610 :                                      HSTORE_VAL(ve, vstr, idx),
     997             :                                      vallen) != 0)))
     998        1960 :                 res = false;
     999             :         }
    1000             :         else
    1001        5010 :             res = false;
    1002             :     }
    1003             : 
    1004        7120 :     PG_RETURN_BOOL(res);
    1005             : }
    1006             : 
    1007             : 
    1008          14 : PG_FUNCTION_INFO_V1(hstore_contained);
    1009             : Datum
    1010           0 : hstore_contained(PG_FUNCTION_ARGS)
    1011             : {
    1012           0 :     PG_RETURN_DATUM(DirectFunctionCall2(hstore_contains,
    1013             :                                         PG_GETARG_DATUM(1),
    1014             :                                         PG_GETARG_DATUM(0)
    1015             :                                         ));
    1016             : }
    1017             : 
    1018             : 
    1019          16 : PG_FUNCTION_INFO_V1(hstore_each);
    1020             : Datum
    1021       23136 : hstore_each(PG_FUNCTION_ARGS)
    1022             : {
    1023             :     FuncCallContext *funcctx;
    1024             :     HStore     *hs;
    1025             :     int         i;
    1026             : 
    1027       23136 :     if (SRF_IS_FIRSTCALL())
    1028             :     {
    1029        4006 :         hs = PG_GETARG_HSTORE_P(0);
    1030        4006 :         funcctx = SRF_FIRSTCALL_INIT();
    1031        4006 :         setup_firstcall(funcctx, hs, fcinfo);
    1032             :     }
    1033             : 
    1034       23136 :     funcctx = SRF_PERCALL_SETUP();
    1035       23136 :     hs = (HStore *) funcctx->user_fctx;
    1036       23136 :     i = funcctx->call_cntr;
    1037             : 
    1038       23136 :     if (i < HS_COUNT(hs))
    1039             :     {
    1040       19130 :         HEntry     *entries = ARRPTR(hs);
    1041       19130 :         char       *ptr = STRPTR(hs);
    1042             :         Datum       res,
    1043             :                     dvalues[2];
    1044       19130 :         bool        nulls[2] = {false, false};
    1045             :         text       *item;
    1046             :         HeapTuple   tuple;
    1047             : 
    1048       19130 :         item = cstring_to_text_with_len(HSTORE_KEY(entries, ptr, i),
    1049       19130 :                                         HSTORE_KEYLEN(entries, i));
    1050       19130 :         dvalues[0] = PointerGetDatum(item);
    1051             : 
    1052       19130 :         if (HSTORE_VALISNULL(entries, i))
    1053             :         {
    1054           6 :             dvalues[1] = (Datum) 0;
    1055           6 :             nulls[1] = true;
    1056             :         }
    1057             :         else
    1058             :         {
    1059       19124 :             item = cstring_to_text_with_len(HSTORE_VAL(entries, ptr, i),
    1060       19124 :                                             HSTORE_VALLEN(entries, i));
    1061       19124 :             dvalues[1] = PointerGetDatum(item);
    1062             :         }
    1063             : 
    1064       19130 :         tuple = heap_form_tuple(funcctx->tuple_desc, dvalues, nulls);
    1065       19130 :         res = HeapTupleGetDatum(tuple);
    1066             : 
    1067       19130 :         SRF_RETURN_NEXT(funcctx, res);
    1068             :     }
    1069             : 
    1070        4006 :     SRF_RETURN_DONE(funcctx);
    1071             : }
    1072             : 
    1073             : 
    1074             : /*
    1075             :  * btree sort order for hstores isn't intended to be useful; we really only
    1076             :  * care about equality versus non-equality.  we compare the entire string
    1077             :  * buffer first, then the entry pos array.
    1078             :  */
    1079             : 
    1080          16 : PG_FUNCTION_INFO_V1(hstore_cmp);
    1081             : Datum
    1082       86278 : hstore_cmp(PG_FUNCTION_ARGS)
    1083             : {
    1084       86278 :     HStore     *hs1 = PG_GETARG_HSTORE_P(0);
    1085       86278 :     HStore     *hs2 = PG_GETARG_HSTORE_P(1);
    1086       86278 :     int         hcount1 = HS_COUNT(hs1);
    1087       86278 :     int         hcount2 = HS_COUNT(hs2);
    1088       86278 :     int         res = 0;
    1089             : 
    1090       86278 :     if (hcount1 == 0 || hcount2 == 0)
    1091             :     {
    1092             :         /*
    1093             :          * if either operand is empty, and the other is nonempty, the nonempty
    1094             :          * one is larger. If both are empty they are equal.
    1095             :          */
    1096        7352 :         if (hcount1 > 0)
    1097         506 :             res = 1;
    1098        6846 :         else if (hcount2 > 0)
    1099        3270 :             res = -1;
    1100             :     }
    1101             :     else
    1102             :     {
    1103             :         /* here we know both operands are nonempty */
    1104       78926 :         char       *str1 = STRPTR(hs1);
    1105       78926 :         char       *str2 = STRPTR(hs2);
    1106       78926 :         HEntry     *ent1 = ARRPTR(hs1);
    1107       78926 :         HEntry     *ent2 = ARRPTR(hs2);
    1108       78926 :         size_t      len1 = HSE_ENDPOS(ent1[2 * hcount1 - 1]);
    1109       78926 :         size_t      len2 = HSE_ENDPOS(ent2[2 * hcount2 - 1]);
    1110             : 
    1111       78926 :         res = memcmp(str1, str2, Min(len1, len2));
    1112             : 
    1113       78926 :         if (res == 0)
    1114             :         {
    1115        5540 :             if (len1 > len2)
    1116           0 :                 res = 1;
    1117        5540 :             else if (len1 < len2)
    1118           0 :                 res = -1;
    1119        5540 :             else if (hcount1 > hcount2)
    1120           0 :                 res = 1;
    1121        5540 :             else if (hcount2 > hcount1)
    1122           0 :                 res = -1;
    1123             :             else
    1124             :             {
    1125        5540 :                 int         count = hcount1 * 2;
    1126             :                 int         i;
    1127             : 
    1128       65540 :                 for (i = 0; i < count; ++i)
    1129       60000 :                     if (HSE_ENDPOS(ent1[i]) != HSE_ENDPOS(ent2[i]) ||
    1130       60000 :                         HSE_ISNULL(ent1[i]) != HSE_ISNULL(ent2[i]))
    1131             :                         break;
    1132        5540 :                 if (i < count)
    1133             :                 {
    1134           0 :                     if (HSE_ENDPOS(ent1[i]) < HSE_ENDPOS(ent2[i]))
    1135           0 :                         res = -1;
    1136           0 :                     else if (HSE_ENDPOS(ent1[i]) > HSE_ENDPOS(ent2[i]))
    1137           0 :                         res = 1;
    1138           0 :                     else if (HSE_ISNULL(ent1[i]))
    1139           0 :                         res = 1;
    1140           0 :                     else if (HSE_ISNULL(ent2[i]))
    1141           0 :                         res = -1;
    1142             :                 }
    1143             :             }
    1144             :         }
    1145             :         else
    1146             :         {
    1147       73386 :             res = (res > 0) ? 1 : -1;
    1148             :         }
    1149             :     }
    1150             : 
    1151             :     /*
    1152             :      * this is a btree support function; this is one of the few places where
    1153             :      * memory needs to be explicitly freed.
    1154             :      */
    1155       86278 :     PG_FREE_IF_COPY(hs1, 0);
    1156       86278 :     PG_FREE_IF_COPY(hs2, 1);
    1157       86278 :     PG_RETURN_INT32(res);
    1158             : }
    1159             : 
    1160             : 
    1161          16 : PG_FUNCTION_INFO_V1(hstore_eq);
    1162             : Datum
    1163        8242 : hstore_eq(PG_FUNCTION_ARGS)
    1164             : {
    1165        8242 :     int         res = DatumGetInt32(DirectFunctionCall2(hstore_cmp,
    1166             :                                                         PG_GETARG_DATUM(0),
    1167             :                                                         PG_GETARG_DATUM(1)));
    1168             : 
    1169        8242 :     PG_RETURN_BOOL(res == 0);
    1170             : }
    1171             : 
    1172          14 : PG_FUNCTION_INFO_V1(hstore_ne);
    1173             : Datum
    1174           0 : hstore_ne(PG_FUNCTION_ARGS)
    1175             : {
    1176           0 :     int         res = DatumGetInt32(DirectFunctionCall2(hstore_cmp,
    1177             :                                                         PG_GETARG_DATUM(0),
    1178             :                                                         PG_GETARG_DATUM(1)));
    1179             : 
    1180           0 :     PG_RETURN_BOOL(res != 0);
    1181             : }
    1182             : 
    1183          16 : PG_FUNCTION_INFO_V1(hstore_gt);
    1184             : Datum
    1185          10 : hstore_gt(PG_FUNCTION_ARGS)
    1186             : {
    1187          10 :     int         res = DatumGetInt32(DirectFunctionCall2(hstore_cmp,
    1188             :                                                         PG_GETARG_DATUM(0),
    1189             :                                                         PG_GETARG_DATUM(1)));
    1190             : 
    1191          10 :     PG_RETURN_BOOL(res > 0);
    1192             : }
    1193             : 
    1194          14 : PG_FUNCTION_INFO_V1(hstore_ge);
    1195             : Datum
    1196           0 : hstore_ge(PG_FUNCTION_ARGS)
    1197             : {
    1198           0 :     int         res = DatumGetInt32(DirectFunctionCall2(hstore_cmp,
    1199             :                                                         PG_GETARG_DATUM(0),
    1200             :                                                         PG_GETARG_DATUM(1)));
    1201             : 
    1202           0 :     PG_RETURN_BOOL(res >= 0);
    1203             : }
    1204             : 
    1205          14 : PG_FUNCTION_INFO_V1(hstore_lt);
    1206             : Datum
    1207           0 : hstore_lt(PG_FUNCTION_ARGS)
    1208             : {
    1209           0 :     int         res = DatumGetInt32(DirectFunctionCall2(hstore_cmp,
    1210             :                                                         PG_GETARG_DATUM(0),
    1211             :                                                         PG_GETARG_DATUM(1)));
    1212             : 
    1213           0 :     PG_RETURN_BOOL(res < 0);
    1214             : }
    1215             : 
    1216          14 : PG_FUNCTION_INFO_V1(hstore_le);
    1217             : Datum
    1218           0 : hstore_le(PG_FUNCTION_ARGS)
    1219             : {
    1220           0 :     int         res = DatumGetInt32(DirectFunctionCall2(hstore_cmp,
    1221             :                                                         PG_GETARG_DATUM(0),
    1222             :                                                         PG_GETARG_DATUM(1)));
    1223             : 
    1224           0 :     PG_RETURN_BOOL(res <= 0);
    1225             : }
    1226             : 
    1227             : 
    1228          16 : PG_FUNCTION_INFO_V1(hstore_hash);
    1229             : Datum
    1230        4028 : hstore_hash(PG_FUNCTION_ARGS)
    1231             : {
    1232        4028 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
    1233        4028 :     Datum       hval = hash_any((unsigned char *) VARDATA(hs),
    1234        4028 :                                 VARSIZE(hs) - VARHDRSZ);
    1235             : 
    1236             :     /*
    1237             :      * This (along with hstore_hash_extended) is the only place in the code
    1238             :      * that cares whether the overall varlena size exactly matches the true
    1239             :      * data size; this assertion should be maintained by all the other code,
    1240             :      * but we make it explicit here.
    1241             :      */
    1242             :     Assert(VARSIZE(hs) ==
    1243             :            (HS_COUNT(hs) != 0 ?
    1244             :             CALCDATASIZE(HS_COUNT(hs),
    1245             :                          HSE_ENDPOS(ARRPTR(hs)[2 * HS_COUNT(hs) - 1])) :
    1246             :             HSHRDSIZE));
    1247             : 
    1248        4028 :     PG_FREE_IF_COPY(hs, 0);
    1249        4028 :     PG_RETURN_DATUM(hval);
    1250             : }
    1251             : 
    1252          16 : PG_FUNCTION_INFO_V1(hstore_hash_extended);
    1253             : Datum
    1254          20 : hstore_hash_extended(PG_FUNCTION_ARGS)
    1255             : {
    1256          20 :     HStore     *hs = PG_GETARG_HSTORE_P(0);
    1257          20 :     uint64      seed = PG_GETARG_INT64(1);
    1258             :     Datum       hval;
    1259             : 
    1260          20 :     hval = hash_any_extended((unsigned char *) VARDATA(hs),
    1261          20 :                              VARSIZE(hs) - VARHDRSZ,
    1262             :                              seed);
    1263             : 
    1264             :     /* See comment in hstore_hash */
    1265             :     Assert(VARSIZE(hs) ==
    1266             :            (HS_COUNT(hs) != 0 ?
    1267             :             CALCDATASIZE(HS_COUNT(hs),
    1268             :                          HSE_ENDPOS(ARRPTR(hs)[2 * HS_COUNT(hs) - 1])) :
    1269             :             HSHRDSIZE));
    1270             : 
    1271          20 :     PG_FREE_IF_COPY(hs, 0);
    1272          20 :     PG_RETURN_DATUM(hval);
    1273             : }

Generated by: LCOV version 1.14