LCOV - code coverage report
Current view: top level - contrib/intarray - _int_gist.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 251 274 91.6 %
Date: 2025-01-18 04:15:08 Functions: 17 17 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*
       2             :  * contrib/intarray/_int_gist.c
       3             :  */
       4             : #include "postgres.h"
       5             : 
       6             : #include <limits.h>
       7             : #include <math.h>
       8             : 
       9             : #include "_int.h"
      10             : #include "access/gist.h"
      11             : #include "access/reloptions.h"
      12             : #include "access/stratnum.h"
      13             : 
      14             : #define GETENTRY(vec,pos) ((ArrayType *) DatumGetPointer((vec)->vector[(pos)].key))
      15             : 
      16             : /*
      17             :  * Control the maximum sparseness of compressed keys.
      18             :  *
      19             :  * The upper safe bound for this limit is half the maximum allocatable array
      20             :  * size. A lower bound would give more guarantees that pathological data
      21             :  * wouldn't eat excessive CPU and memory, but at the expense of breaking
      22             :  * possibly working (after a fashion) indexes.
      23             :  */
      24             : #define MAXNUMELTS (Min((MaxAllocSize / sizeof(Datum)),((MaxAllocSize - ARR_OVERHEAD_NONULLS(1)) / sizeof(int)))/2)
      25             : /* or: #define MAXNUMELTS 1000000 */
      26             : 
      27             : /*
      28             : ** GiST support methods
      29             : */
      30           4 : PG_FUNCTION_INFO_V1(g_int_consistent);
      31           4 : PG_FUNCTION_INFO_V1(g_int_compress);
      32           4 : PG_FUNCTION_INFO_V1(g_int_decompress);
      33           4 : PG_FUNCTION_INFO_V1(g_int_penalty);
      34           4 : PG_FUNCTION_INFO_V1(g_int_picksplit);
      35           4 : PG_FUNCTION_INFO_V1(g_int_union);
      36           4 : PG_FUNCTION_INFO_V1(g_int_same);
      37           4 : PG_FUNCTION_INFO_V1(g_int_options);
      38             : 
      39             : 
      40             : /*
      41             : ** The GiST Consistent method for _intments
      42             : ** Should return false if for all data items x below entry,
      43             : ** the predicate x op query == false, where op is the oper
      44             : ** corresponding to strategy in the pg_amop table.
      45             : */
      46             : Datum
      47      243380 : g_int_consistent(PG_FUNCTION_ARGS)
      48             : {
      49      243380 :     GISTENTRY  *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
      50      243380 :     ArrayType  *query = PG_GETARG_ARRAYTYPE_P_COPY(1);
      51      243380 :     StrategyNumber strategy = (StrategyNumber) PG_GETARG_UINT16(2);
      52             : 
      53             :     /* Oid      subtype = PG_GETARG_OID(3); */
      54      243380 :     bool       *recheck = (bool *) PG_GETARG_POINTER(4);
      55      243380 :     bool        retval = false; /* silence compiler warning */
      56             : 
      57             :     /* this is exact except for RTSameStrategyNumber */
      58      243380 :     *recheck = (strategy == RTSameStrategyNumber);
      59             : 
      60      243380 :     if (strategy == BooleanSearchStrategy)
      61             :     {
      62      157376 :         retval = execconsistent((QUERYTYPE *) query,
      63      157376 :                                 (ArrayType *) DatumGetPointer(entry->key),
      64      157376 :                                 GIST_LEAF(entry));
      65             : 
      66      157376 :         pfree(query);
      67      157376 :         PG_RETURN_BOOL(retval);
      68             :     }
      69             : 
      70             :     /* sort query for fast search, key is already sorted */
      71       86004 :     CHECKARRVALID(query);
      72       86004 :     PREPAREARR(query);
      73             : 
      74       86004 :     switch (strategy)
      75             :     {
      76       30578 :         case RTOverlapStrategyNumber:
      77       30578 :             retval = inner_int_overlap((ArrayType *) DatumGetPointer(entry->key),
      78             :                                        query);
      79       30578 :             break;
      80        6566 :         case RTSameStrategyNumber:
      81        6566 :             if (GIST_LEAF(entry))
      82        6018 :                 DirectFunctionCall3(g_int_same,
      83             :                                     entry->key,
      84             :                                     PointerGetDatum(query),
      85             :                                     PointerGetDatum(&retval));
      86             :             else
      87         548 :                 retval = inner_int_contains((ArrayType *) DatumGetPointer(entry->key),
      88             :                                             query);
      89        6566 :             break;
      90       48860 :         case RTContainsStrategyNumber:
      91             :         case RTOldContainsStrategyNumber:
      92       48860 :             retval = inner_int_contains((ArrayType *) DatumGetPointer(entry->key),
      93             :                                         query);
      94       48860 :             break;
      95           0 :         case RTContainedByStrategyNumber:
      96             :         case RTOldContainedByStrategyNumber:
      97             : 
      98             :             /*
      99             :              * This code is unreachable as of intarray 1.4, because the <@
     100             :              * operator has been removed from the opclass.  We keep it for now
     101             :              * to support older versions of the SQL definitions.
     102             :              */
     103           0 :             if (GIST_LEAF(entry))
     104           0 :                 retval = inner_int_contains(query,
     105           0 :                                             (ArrayType *) DatumGetPointer(entry->key));
     106             :             else
     107             :             {
     108             :                 /*
     109             :                  * Unfortunately, because empty arrays could be anywhere in
     110             :                  * the index, we must search the whole tree.
     111             :                  */
     112           0 :                 retval = true;
     113             :             }
     114           0 :             break;
     115           0 :         default:
     116           0 :             retval = false;
     117             :     }
     118       86004 :     pfree(query);
     119       86004 :     PG_RETURN_BOOL(retval);
     120             : }
     121             : 
     122             : Datum
     123      113006 : g_int_union(PG_FUNCTION_ARGS)
     124             : {
     125      113006 :     GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
     126      113006 :     int        *size = (int *) PG_GETARG_POINTER(1);
     127             :     int32       i,
     128             :                *ptr;
     129             :     ArrayType  *res;
     130      113006 :     int         totlen = 0;
     131             : 
     132      340660 :     for (i = 0; i < entryvec->n; i++)
     133             :     {
     134      227654 :         ArrayType  *ent = GETENTRY(entryvec, i);
     135             : 
     136      227654 :         CHECKARRVALID(ent);
     137      227654 :         totlen += ARRNELEMS(ent);
     138             :     }
     139             : 
     140      113006 :     res = new_intArrayType(totlen);
     141      113006 :     ptr = ARRPTR(res);
     142             : 
     143      340660 :     for (i = 0; i < entryvec->n; i++)
     144             :     {
     145      227654 :         ArrayType  *ent = GETENTRY(entryvec, i);
     146             :         int         nel;
     147             : 
     148      227654 :         nel = ARRNELEMS(ent);
     149      227654 :         memcpy(ptr, ARRPTR(ent), nel * sizeof(int32));
     150      227654 :         ptr += nel;
     151             :     }
     152             : 
     153      113006 :     QSORT(res, 1);
     154      113006 :     res = _int_unique(res);
     155      113006 :     *size = VARSIZE(res);
     156      113006 :     PG_RETURN_POINTER(res);
     157             : }
     158             : 
     159             : /*
     160             : ** GiST Compress and Decompress methods
     161             : */
     162             : Datum
     163       60444 : g_int_compress(PG_FUNCTION_ARGS)
     164             : {
     165       60444 :     GISTENTRY  *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
     166             :     GISTENTRY  *retval;
     167             :     ArrayType  *r;
     168       60444 :     int         num_ranges = G_INT_GET_NUMRANGES();
     169             :     int         len,
     170             :                 lenr;
     171             :     int        *dr;
     172             :     int         i,
     173             :                 j,
     174             :                 cand;
     175             :     int64       min;
     176             : 
     177       60444 :     if (entry->leafkey)
     178             :     {
     179       40544 :         r = DatumGetArrayTypePCopy(entry->key);
     180       40544 :         CHECKARRVALID(r);
     181       40544 :         PREPAREARR(r);
     182             : 
     183       40544 :         if (ARRNELEMS(r) >= 2 * num_ranges)
     184           2 :             ereport(ERROR,
     185             :                     (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
     186             :                      errmsg("input array is too big (%d maximum allowed, %d current), use gist__intbig_ops opclass instead",
     187             :                             2 * num_ranges - 1, ARRNELEMS(r))));
     188             : 
     189       40542 :         retval = palloc(sizeof(GISTENTRY));
     190       40542 :         gistentryinit(*retval, PointerGetDatum(r),
     191             :                       entry->rel, entry->page, entry->offset, false);
     192             : 
     193       40542 :         PG_RETURN_POINTER(retval);
     194             :     }
     195             : 
     196             :     /*
     197             :      * leaf entries never compress one more time, only when entry->leafkey
     198             :      * ==true, so now we work only with internal keys
     199             :      */
     200             : 
     201       19900 :     r = DatumGetArrayTypeP(entry->key);
     202       19900 :     CHECKARRVALID(r);
     203       19900 :     if (ARRISEMPTY(r))
     204             :     {
     205           0 :         if (r != (ArrayType *) DatumGetPointer(entry->key))
     206           0 :             pfree(r);
     207           0 :         PG_RETURN_POINTER(entry);
     208             :     }
     209             : 
     210       19900 :     if ((len = ARRNELEMS(r)) >= 2 * num_ranges)
     211             :     {                           /* compress */
     212        2838 :         if (r == (ArrayType *) DatumGetPointer(entry->key))
     213        2838 :             r = DatumGetArrayTypePCopy(entry->key);
     214        2838 :         r = resize_intArrayType(r, 2 * (len));
     215             : 
     216        2838 :         dr = ARRPTR(r);
     217             : 
     218             :         /*
     219             :          * "len" at this point is the number of ranges we will construct.
     220             :          * "lenr" is the number of ranges we must eventually remove by
     221             :          * merging, we must be careful to remove no more than this number.
     222             :          */
     223        2838 :         lenr = len - num_ranges;
     224             : 
     225             :         /*
     226             :          * Initially assume we can merge consecutive ints into a range. but we
     227             :          * must count every value removed and stop when lenr runs out
     228             :          */
     229      122956 :         for (j = i = len - 1; i > 0 && lenr > 0; i--, j--)
     230             :         {
     231      120118 :             int         r_end = dr[i];
     232      120118 :             int         r_start = r_end;
     233             : 
     234     1385374 :             while (i > 0 && lenr > 0 && dr[i - 1] == r_start - 1)
     235     1265256 :                 --r_start, --i, --lenr;
     236      120118 :             dr[2 * j] = r_start;
     237      120118 :             dr[2 * j + 1] = r_end;
     238             :         }
     239             :         /* just copy the rest, if any, as trivial ranges */
     240      547432 :         for (; i >= 0; i--, j--)
     241      544594 :             dr[2 * j] = dr[2 * j + 1] = dr[i];
     242             : 
     243        2838 :         if (++j)
     244             :         {
     245             :             /*
     246             :              * shunt everything down to start at the right place
     247             :              */
     248        2838 :             memmove(&dr[0], &dr[2 * j], 2 * (len - j) * sizeof(int32));
     249             :         }
     250             : 
     251             :         /*
     252             :          * make "len" be number of array elements, not ranges
     253             :          */
     254        2838 :         len = 2 * (len - j);
     255        2838 :         cand = 1;
     256        2838 :         while (len > num_ranges * 2)
     257             :         {
     258           0 :             min = PG_INT64_MAX;
     259           0 :             for (i = 2; i < len; i += 2)
     260           0 :                 if (min > ((int64) dr[i] - (int64) dr[i - 1]))
     261             :                 {
     262           0 :                     min = ((int64) dr[i] - (int64) dr[i - 1]);
     263           0 :                     cand = i;
     264             :                 }
     265           0 :             memmove(&dr[cand - 1], &dr[cand + 1], (len - cand - 1) * sizeof(int32));
     266           0 :             len -= 2;
     267             :         }
     268             : 
     269             :         /*
     270             :          * check sparseness of result
     271             :          */
     272        2838 :         lenr = internal_size(dr, len);
     273        2838 :         if (lenr < 0 || lenr > MAXNUMELTS)
     274           0 :             ereport(ERROR,
     275             :                     (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
     276             :                      errmsg("data is too sparse, recreate index using gist__intbig_ops opclass instead")));
     277             : 
     278        2838 :         r = resize_intArrayType(r, len);
     279        2838 :         retval = palloc(sizeof(GISTENTRY));
     280        2838 :         gistentryinit(*retval, PointerGetDatum(r),
     281             :                       entry->rel, entry->page, entry->offset, false);
     282        2838 :         PG_RETURN_POINTER(retval);
     283             :     }
     284             :     else
     285       17062 :         PG_RETURN_POINTER(entry);
     286             : }
     287             : 
     288             : Datum
     289     1267770 : g_int_decompress(PG_FUNCTION_ARGS)
     290             : {
     291     1267770 :     GISTENTRY  *entry = (GISTENTRY *) PG_GETARG_POINTER(0);
     292             :     GISTENTRY  *retval;
     293             :     ArrayType  *r;
     294     1267770 :     int         num_ranges = G_INT_GET_NUMRANGES();
     295             :     int        *dr,
     296             :                 lenr;
     297             :     ArrayType  *in;
     298             :     int         lenin;
     299             :     int        *din;
     300             :     int         i;
     301             : 
     302     1267770 :     in = DatumGetArrayTypeP(entry->key);
     303             : 
     304     1267770 :     CHECKARRVALID(in);
     305     1267770 :     if (ARRISEMPTY(in))
     306             :     {
     307         690 :         if (in != (ArrayType *) DatumGetPointer(entry->key))
     308             :         {
     309         690 :             retval = palloc(sizeof(GISTENTRY));
     310         690 :             gistentryinit(*retval, PointerGetDatum(in),
     311             :                           entry->rel, entry->page, entry->offset, false);
     312         690 :             PG_RETURN_POINTER(retval);
     313             :         }
     314             : 
     315           0 :         PG_RETURN_POINTER(entry);
     316             :     }
     317             : 
     318     1267080 :     lenin = ARRNELEMS(in);
     319             : 
     320     1267080 :     if (lenin < 2 * num_ranges)
     321             :     {                           /* not compressed value */
     322     1213578 :         if (in != (ArrayType *) DatumGetPointer(entry->key))
     323             :         {
     324      522180 :             retval = palloc(sizeof(GISTENTRY));
     325      522180 :             gistentryinit(*retval, PointerGetDatum(in),
     326             :                           entry->rel, entry->page, entry->offset, false);
     327             : 
     328      522180 :             PG_RETURN_POINTER(retval);
     329             :         }
     330      691398 :         PG_RETURN_POINTER(entry);
     331             :     }
     332             : 
     333       53502 :     din = ARRPTR(in);
     334       53502 :     lenr = internal_size(din, lenin);
     335       53502 :     if (lenr < 0 || lenr > MAXNUMELTS)
     336           0 :         ereport(ERROR,
     337             :                 (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
     338             :                  errmsg("compressed array is too big, recreate index using gist__intbig_ops opclass instead")));
     339             : 
     340       53502 :     r = new_intArrayType(lenr);
     341       53502 :     dr = ARRPTR(r);
     342             : 
     343    12768102 :     for (i = 0; i < lenin; i += 2)
     344             :     {
     345             :         /* use int64 for j in case din[i + 1] is INT_MAX */
     346    51240628 :         for (int64 j = din[i]; j <= din[i + 1]; j++)
     347    38526028 :             if ((!i) || *(dr - 1) != j)
     348    38526028 :                 *dr++ = (int) j;
     349             :     }
     350             : 
     351       53502 :     if (in != (ArrayType *) DatumGetPointer(entry->key))
     352       53502 :         pfree(in);
     353       53502 :     retval = palloc(sizeof(GISTENTRY));
     354       53502 :     gistentryinit(*retval, PointerGetDatum(r),
     355             :                   entry->rel, entry->page, entry->offset, false);
     356             : 
     357       53502 :     PG_RETURN_POINTER(retval);
     358             : }
     359             : 
     360             : /*
     361             : ** The GiST Penalty method for _intments
     362             : */
     363             : Datum
     364      611232 : g_int_penalty(PG_FUNCTION_ARGS)
     365             : {
     366      611232 :     GISTENTRY  *origentry = (GISTENTRY *) PG_GETARG_POINTER(0);
     367      611232 :     GISTENTRY  *newentry = (GISTENTRY *) PG_GETARG_POINTER(1);
     368      611232 :     float      *result = (float *) PG_GETARG_POINTER(2);
     369             :     ArrayType  *ud;
     370             :     float       tmp1,
     371             :                 tmp2;
     372             : 
     373      611232 :     ud = inner_int_union((ArrayType *) DatumGetPointer(origentry->key),
     374      611232 :                          (ArrayType *) DatumGetPointer(newentry->key));
     375      611232 :     rt__int_size(ud, &tmp1);
     376      611232 :     rt__int_size((ArrayType *) DatumGetPointer(origentry->key), &tmp2);
     377      611232 :     *result = tmp1 - tmp2;
     378      611232 :     pfree(ud);
     379             : 
     380      611232 :     PG_RETURN_POINTER(result);
     381             : }
     382             : 
     383             : 
     384             : 
     385             : Datum
     386      118960 : g_int_same(PG_FUNCTION_ARGS)
     387             : {
     388      118960 :     ArrayType  *a = PG_GETARG_ARRAYTYPE_P(0);
     389      118960 :     ArrayType  *b = PG_GETARG_ARRAYTYPE_P(1);
     390      118960 :     bool       *result = (bool *) PG_GETARG_POINTER(2);
     391      118960 :     int32       n = ARRNELEMS(a);
     392             :     int32      *da,
     393             :                *db;
     394             : 
     395      118960 :     CHECKARRVALID(a);
     396      118960 :     CHECKARRVALID(b);
     397             : 
     398      118960 :     if (n != ARRNELEMS(b))
     399             :     {
     400       22438 :         *result = false;
     401       22438 :         PG_RETURN_POINTER(result);
     402             :     }
     403       96522 :     *result = true;
     404       96522 :     da = ARRPTR(a);
     405       96522 :     db = ARRPTR(b);
     406    29276520 :     while (n--)
     407             :     {
     408    29180986 :         if (*da++ != *db++)
     409             :         {
     410         988 :             *result = false;
     411         988 :             break;
     412             :         }
     413             :     }
     414             : 
     415       96522 :     PG_RETURN_POINTER(result);
     416             : }
     417             : 
     418             : /*****************************************************************
     419             : ** Common GiST Method
     420             : *****************************************************************/
     421             : 
     422             : typedef struct
     423             : {
     424             :     OffsetNumber pos;
     425             :     float       cost;
     426             : } SPLITCOST;
     427             : 
     428             : static int
     429      125858 : comparecost(const void *a, const void *b)
     430             : {
     431      125858 :     if (((const SPLITCOST *) a)->cost == ((const SPLITCOST *) b)->cost)
     432       67110 :         return 0;
     433             :     else
     434       58748 :         return (((const SPLITCOST *) a)->cost > ((const SPLITCOST *) b)->cost) ? 1 : -1;
     435             : }
     436             : 
     437             : /*
     438             : ** The GiST PickSplit method for _intments
     439             : ** We use Guttman's poly time split algorithm
     440             : */
     441             : Datum
     442        1228 : g_int_picksplit(PG_FUNCTION_ARGS)
     443             : {
     444        1228 :     GistEntryVector *entryvec = (GistEntryVector *) PG_GETARG_POINTER(0);
     445        1228 :     GIST_SPLITVEC *v = (GIST_SPLITVEC *) PG_GETARG_POINTER(1);
     446             :     OffsetNumber i,
     447             :                 j;
     448             :     ArrayType  *datum_alpha,
     449             :                *datum_beta;
     450             :     ArrayType  *datum_l,
     451             :                *datum_r;
     452             :     ArrayType  *union_d,
     453             :                *union_dl,
     454             :                *union_dr;
     455             :     ArrayType  *inter_d;
     456             :     bool        firsttime;
     457             :     float       size_alpha,
     458             :                 size_beta,
     459             :                 size_union,
     460             :                 size_inter;
     461             :     float       size_waste,
     462             :                 waste;
     463             :     float       size_l,
     464             :                 size_r;
     465             :     int         nbytes;
     466        1228 :     OffsetNumber seed_1 = 0,
     467        1228 :                 seed_2 = 0;
     468             :     OffsetNumber *left,
     469             :                *right;
     470             :     OffsetNumber maxoff;
     471             :     SPLITCOST  *costvector;
     472             : 
     473             : #ifdef GIST_DEBUG
     474             :     elog(DEBUG3, "--------picksplit %d", entryvec->n);
     475             : #endif
     476             : 
     477        1228 :     maxoff = entryvec->n - 2;
     478        1228 :     nbytes = (maxoff + 2) * sizeof(OffsetNumber);
     479        1228 :     v->spl_left = (OffsetNumber *) palloc(nbytes);
     480        1228 :     v->spl_right = (OffsetNumber *) palloc(nbytes);
     481             : 
     482        1228 :     firsttime = true;
     483        1228 :     waste = 0.0;
     484       65838 :     for (i = FirstOffsetNumber; i < maxoff; i = OffsetNumberNext(i))
     485             :     {
     486       64610 :         datum_alpha = GETENTRY(entryvec, i);
     487     3635700 :         for (j = OffsetNumberNext(i); j <= maxoff; j = OffsetNumberNext(j))
     488             :         {
     489     3571090 :             datum_beta = GETENTRY(entryvec, j);
     490             : 
     491             :             /* compute the wasted space by unioning these guys */
     492             :             /* size_waste = size_union - size_inter; */
     493     3571090 :             union_d = inner_int_union(datum_alpha, datum_beta);
     494     3571090 :             rt__int_size(union_d, &size_union);
     495     3571090 :             inter_d = inner_int_inter(datum_alpha, datum_beta);
     496     3571090 :             rt__int_size(inter_d, &size_inter);
     497     3571090 :             size_waste = size_union - size_inter;
     498             : 
     499     3571090 :             pfree(union_d);
     500     3571090 :             pfree(inter_d);
     501             : 
     502             :             /*
     503             :              * are these a more promising split that what we've already seen?
     504             :              */
     505             : 
     506     3571090 :             if (size_waste > waste || firsttime)
     507             :             {
     508        5940 :                 waste = size_waste;
     509        5940 :                 seed_1 = i;
     510        5940 :                 seed_2 = j;
     511        5940 :                 firsttime = false;
     512             :             }
     513             :         }
     514             :     }
     515             : 
     516        1228 :     left = v->spl_left;
     517        1228 :     v->spl_nleft = 0;
     518        1228 :     right = v->spl_right;
     519        1228 :     v->spl_nright = 0;
     520        1228 :     if (seed_1 == 0 || seed_2 == 0)
     521             :     {
     522           0 :         seed_1 = 1;
     523           0 :         seed_2 = 2;
     524             :     }
     525             : 
     526        1228 :     datum_alpha = GETENTRY(entryvec, seed_1);
     527        1228 :     datum_l = copy_intArrayType(datum_alpha);
     528        1228 :     rt__int_size(datum_l, &size_l);
     529        1228 :     datum_beta = GETENTRY(entryvec, seed_2);
     530        1228 :     datum_r = copy_intArrayType(datum_beta);
     531        1228 :     rt__int_size(datum_r, &size_r);
     532             : 
     533        1228 :     maxoff = OffsetNumberNext(maxoff);
     534             : 
     535             :     /*
     536             :      * sort entries
     537             :      */
     538        1228 :     costvector = (SPLITCOST *) palloc(sizeof(SPLITCOST) * maxoff);
     539       68294 :     for (i = FirstOffsetNumber; i <= maxoff; i = OffsetNumberNext(i))
     540             :     {
     541       67066 :         costvector[i - 1].pos = i;
     542       67066 :         datum_alpha = GETENTRY(entryvec, i);
     543       67066 :         union_d = inner_int_union(datum_l, datum_alpha);
     544       67066 :         rt__int_size(union_d, &size_alpha);
     545       67066 :         pfree(union_d);
     546       67066 :         union_d = inner_int_union(datum_r, datum_alpha);
     547       67066 :         rt__int_size(union_d, &size_beta);
     548       67066 :         pfree(union_d);
     549       67066 :         costvector[i - 1].cost = fabsf((size_alpha - size_l) - (size_beta - size_r));
     550             :     }
     551        1228 :     qsort(costvector, maxoff, sizeof(SPLITCOST), comparecost);
     552             : 
     553             :     /*
     554             :      * Now split up the regions between the two seeds.  An important property
     555             :      * of this split algorithm is that the split vector v has the indices of
     556             :      * items to be split in order in its left and right vectors.  We exploit
     557             :      * this property by doing a merge in the code that actually splits the
     558             :      * page.
     559             :      *
     560             :      * For efficiency, we also place the new index tuple in this loop. This is
     561             :      * handled at the very end, when we have placed all the existing tuples
     562             :      * and i == maxoff + 1.
     563             :      */
     564             : 
     565             : 
     566       68294 :     for (j = 0; j < maxoff; j++)
     567             :     {
     568       67066 :         i = costvector[j].pos;
     569             : 
     570             :         /*
     571             :          * If we've already decided where to place this item, just put it on
     572             :          * the right list.  Otherwise, we need to figure out which page needs
     573             :          * the least enlargement in order to store the item.
     574             :          */
     575             : 
     576       67066 :         if (i == seed_1)
     577             :         {
     578        1228 :             *left++ = i;
     579        1228 :             v->spl_nleft++;
     580        1228 :             continue;
     581             :         }
     582       65838 :         else if (i == seed_2)
     583             :         {
     584        1228 :             *right++ = i;
     585        1228 :             v->spl_nright++;
     586        1228 :             continue;
     587             :         }
     588             : 
     589             :         /* okay, which page needs least enlargement? */
     590       64610 :         datum_alpha = GETENTRY(entryvec, i);
     591       64610 :         union_dl = inner_int_union(datum_l, datum_alpha);
     592       64610 :         union_dr = inner_int_union(datum_r, datum_alpha);
     593       64610 :         rt__int_size(union_dl, &size_alpha);
     594       64610 :         rt__int_size(union_dr, &size_beta);
     595             : 
     596             :         /* pick which page to add it to */
     597       64610 :         if (size_alpha - size_l < size_beta - size_r + WISH_F(v->spl_nleft, v->spl_nright, 0.01))
     598             :         {
     599       31354 :             pfree(datum_l);
     600       31354 :             pfree(union_dr);
     601       31354 :             datum_l = union_dl;
     602       31354 :             size_l = size_alpha;
     603       31354 :             *left++ = i;
     604       31354 :             v->spl_nleft++;
     605             :         }
     606             :         else
     607             :         {
     608       33256 :             pfree(datum_r);
     609       33256 :             pfree(union_dl);
     610       33256 :             datum_r = union_dr;
     611       33256 :             size_r = size_beta;
     612       33256 :             *right++ = i;
     613       33256 :             v->spl_nright++;
     614             :         }
     615             :     }
     616        1228 :     pfree(costvector);
     617        1228 :     *right = *left = FirstOffsetNumber;
     618             : 
     619        1228 :     v->spl_ldatum = PointerGetDatum(datum_l);
     620        1228 :     v->spl_rdatum = PointerGetDatum(datum_r);
     621             : 
     622        1228 :     PG_RETURN_POINTER(v);
     623             : }
     624             : 
     625             : Datum
     626          26 : g_int_options(PG_FUNCTION_ARGS)
     627             : {
     628          26 :     local_relopts *relopts = (local_relopts *) PG_GETARG_POINTER(0);
     629             : 
     630          26 :     init_local_reloptions(relopts, sizeof(GISTIntArrayOptions));
     631          26 :     add_local_int_reloption(relopts, "numranges",
     632             :                             "number of ranges for compression",
     633             :                             G_INT_NUMRANGES_DEFAULT, 1, G_INT_NUMRANGES_MAX,
     634             :                             offsetof(GISTIntArrayOptions, num_ranges));
     635             : 
     636          26 :     PG_RETURN_VOID();
     637             : }

Generated by: LCOV version 1.14