LCOV - code coverage report
Current view: top level - contrib/intarray - _int_tool.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 176 185 95.1 %
Date: 2025-01-18 04:15:08 Functions: 18 19 94.7 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*
       2             :  * contrib/intarray/_int_tool.c
       3             :  */
       4             : #include "postgres.h"
       5             : 
       6             : #include <limits.h>
       7             : 
       8             : #include "_int.h"
       9             : #include "catalog/pg_type.h"
      10             : #include "common/int.h"
      11             : #include "lib/qunique.h"
      12             : 
      13             : /* arguments are assumed sorted & unique-ified */
      14             : bool
      15      185976 : inner_int_contains(ArrayType *a, ArrayType *b)
      16             : {
      17             :     int         na,
      18             :                 nb;
      19             :     int         i,
      20             :                 j,
      21             :                 n;
      22             :     int        *da,
      23             :                *db;
      24             : 
      25      185976 :     na = ARRNELEMS(a);
      26      185976 :     nb = ARRNELEMS(b);
      27      185976 :     da = ARRPTR(a);
      28      185976 :     db = ARRPTR(b);
      29             : 
      30      185976 :     i = j = n = 0;
      31      450556 :     while (i < na && j < nb)
      32             :     {
      33      437346 :         if (da[i] < db[j])
      34      252026 :             i++;
      35      185320 :         else if (da[i] == db[j])
      36             :         {
      37       12554 :             n++;
      38       12554 :             i++;
      39       12554 :             j++;
      40             :         }
      41             :         else
      42      172766 :             break;              /* db[j] is not in da */
      43             :     }
      44             : 
      45      185976 :     return (n == nb);
      46             : }
      47             : 
      48             : /* arguments are assumed sorted */
      49             : bool
      50       45686 : inner_int_overlap(ArrayType *a, ArrayType *b)
      51             : {
      52             :     int         na,
      53             :                 nb;
      54             :     int         i,
      55             :                 j;
      56             :     int        *da,
      57             :                *db;
      58             : 
      59       45686 :     na = ARRNELEMS(a);
      60       45686 :     nb = ARRNELEMS(b);
      61       45686 :     da = ARRPTR(a);
      62       45686 :     db = ARRPTR(b);
      63             : 
      64       45686 :     i = j = 0;
      65      214286 :     while (i < na && j < nb)
      66             :     {
      67      174370 :         if (da[i] < db[j])
      68       88584 :             i++;
      69       85786 :         else if (da[i] == db[j])
      70        5770 :             return true;
      71             :         else
      72       80016 :             j++;
      73             :     }
      74             : 
      75       39916 :     return false;
      76             : }
      77             : 
      78             : ArrayType *
      79     4445676 : inner_int_union(ArrayType *a, ArrayType *b)
      80             : {
      81     4445676 :     ArrayType  *r = NULL;
      82             : 
      83     4445676 :     CHECKARRVALID(a);
      84     4445676 :     CHECKARRVALID(b);
      85             : 
      86     4445676 :     if (ARRISEMPTY(a) && ARRISEMPTY(b))
      87         302 :         return new_intArrayType(0);
      88     4445374 :     if (ARRISEMPTY(a))
      89       27870 :         r = copy_intArrayType(b);
      90     4445374 :     if (ARRISEMPTY(b))
      91        8424 :         r = copy_intArrayType(a);
      92             : 
      93     4445374 :     if (!r)
      94             :     {
      95     4409080 :         int         na = ARRNELEMS(a),
      96     4409080 :                     nb = ARRNELEMS(b);
      97     4409080 :         int        *da = ARRPTR(a),
      98     4409080 :                    *db = ARRPTR(b);
      99             :         int         i,
     100             :                     j,
     101             :                    *dr;
     102             : 
     103     4409080 :         r = new_intArrayType(na + nb);
     104     4409080 :         dr = ARRPTR(r);
     105             : 
     106             :         /* union */
     107     4409080 :         i = j = 0;
     108   163036942 :         while (i < na && j < nb)
     109             :         {
     110   158627862 :             if (da[i] == db[j])
     111             :             {
     112    10184782 :                 *dr++ = da[i++];
     113    10184782 :                 j++;
     114             :             }
     115   148443080 :             else if (da[i] < db[j])
     116   125022244 :                 *dr++ = da[i++];
     117             :             else
     118    23420836 :                 *dr++ = db[j++];
     119             :         }
     120             : 
     121    15415704 :         while (i < na)
     122    11006624 :             *dr++ = da[i++];
     123     8491784 :         while (j < nb)
     124     4082704 :             *dr++ = db[j++];
     125             : 
     126     4409080 :         r = resize_intArrayType(r, dr - ARRPTR(r));
     127             :     }
     128             : 
     129     4445374 :     if (ARRNELEMS(r) > 1)
     130     4445366 :         r = _int_unique(r);
     131             : 
     132     4445374 :     return r;
     133             : }
     134             : 
     135             : ArrayType *
     136     3571102 : inner_int_inter(ArrayType *a, ArrayType *b)
     137             : {
     138             :     ArrayType  *r;
     139             :     int         na,
     140             :                 nb;
     141             :     int        *da,
     142             :                *db,
     143             :                *dr;
     144             :     int         i,
     145             :                 j,
     146             :                 k;
     147             : 
     148     3571102 :     if (ARRISEMPTY(a) || ARRISEMPTY(b))
     149       35380 :         return new_intArrayType(0);
     150             : 
     151     3535722 :     na = ARRNELEMS(a);
     152     3535722 :     nb = ARRNELEMS(b);
     153     3535722 :     da = ARRPTR(a);
     154     3535722 :     db = ARRPTR(b);
     155     3535722 :     r = new_intArrayType(Min(na, nb));
     156     3535722 :     dr = ARRPTR(r);
     157             : 
     158     3535722 :     i = j = k = 0;
     159    44142080 :     while (i < na && j < nb)
     160             :     {
     161    40606358 :         if (da[i] < db[j])
     162    18736046 :             i++;
     163    21870312 :         else if (da[i] == db[j])
     164             :         {
     165     3547202 :             if (k == 0 || dr[k - 1] != db[j])
     166     3547202 :                 dr[k++] = db[j];
     167     3547202 :             i++;
     168     3547202 :             j++;
     169             :         }
     170             :         else
     171    18323110 :             j++;
     172             :     }
     173             : 
     174     3535722 :     if (k == 0)
     175             :     {
     176     2799936 :         pfree(r);
     177     2799936 :         return new_intArrayType(0);
     178             :     }
     179             :     else
     180      735786 :         return resize_intArrayType(r, k);
     181             : }
     182             : 
     183             : void
     184     8630452 : rt__int_size(ArrayType *a, float *size)
     185             : {
     186     8630452 :     *size = (float) ARRNELEMS(a);
     187     8630452 : }
     188             : 
     189             : /* qsort_arg comparison function for isort() */
     190             : static int
     191   206573792 : isort_cmp(const void *a, const void *b, void *arg)
     192             : {
     193   206573792 :     int32       aval = *((const int32 *) a);
     194   206573792 :     int32       bval = *((const int32 *) b);
     195             : 
     196   206573792 :     if (aval < bval)
     197     1240202 :         return -1;
     198   205333590 :     if (aval > bval)
     199   204303830 :         return 1;
     200             : 
     201             :     /*
     202             :      * Report if we have any duplicates.  If there are equal keys, qsort must
     203             :      * compare them at some point, else it wouldn't know whether one should go
     204             :      * before or after the other.
     205             :      */
     206     1029760 :     *((bool *) arg) = true;
     207     1029760 :     return 0;
     208             : }
     209             : 
     210             : /* Sort the given data (len >= 2).  Return true if any duplicates found */
     211             : bool
     212      566400 : isort(int32 *a, int len)
     213             : {
     214      566400 :     bool        r = false;
     215             : 
     216      566400 :     qsort_arg(a, len, sizeof(int32), isort_cmp, &r);
     217      566400 :     return r;
     218             : }
     219             : 
     220             : /* Create a new int array with room for "num" elements */
     221             : ArrayType *
     222    10985698 : new_intArrayType(int num)
     223             : {
     224             :     ArrayType  *r;
     225             :     int         nbytes;
     226             : 
     227             :     /* if no elements, return a zero-dimensional array */
     228    10985698 :     if (num <= 0)
     229             :     {
     230             :         Assert(num == 0);
     231     2835618 :         r = construct_empty_array(INT4OID);
     232     2835618 :         return r;
     233             :     }
     234             : 
     235     8150080 :     nbytes = ARR_OVERHEAD_NONULLS(1) + sizeof(int) * num;
     236             : 
     237     8150080 :     r = (ArrayType *) palloc0(nbytes);
     238             : 
     239     8150080 :     SET_VARSIZE(r, nbytes);
     240     8150080 :     ARR_NDIM(r) = 1;
     241     8150080 :     r->dataoffset = 0;           /* marker for no null bitmap */
     242     8150080 :     ARR_ELEMTYPE(r) = INT4OID;
     243     8150080 :     ARR_DIMS(r)[0] = num;
     244     8150080 :     ARR_LBOUND(r)[0] = 1;
     245             : 
     246     8150080 :     return r;
     247             : }
     248             : 
     249             : ArrayType *
     250     9726158 : resize_intArrayType(ArrayType *a, int num)
     251             : {
     252             :     int         nbytes;
     253             :     int         i;
     254             : 
     255             :     /* if no elements, return a zero-dimensional array */
     256     9726158 :     if (num <= 0)
     257             :     {
     258             :         Assert(num == 0);
     259           0 :         a = construct_empty_array(INT4OID);
     260           0 :         return a;
     261             :     }
     262             : 
     263     9726158 :     if (num == ARRNELEMS(a))
     264     7424128 :         return a;
     265             : 
     266     2302030 :     nbytes = ARR_DATA_OFFSET(a) + sizeof(int) * num;
     267             : 
     268     2302030 :     a = (ArrayType *) repalloc(a, nbytes);
     269             : 
     270     2302030 :     SET_VARSIZE(a, nbytes);
     271             :     /* usually the array should be 1-D already, but just in case ... */
     272     4604060 :     for (i = 0; i < ARR_NDIM(a); i++)
     273             :     {
     274     2302030 :         ARR_DIMS(a)[i] = num;
     275     2302030 :         num = 1;
     276             :     }
     277     2302030 :     return a;
     278             : }
     279             : 
     280             : ArrayType *
     281       38750 : copy_intArrayType(ArrayType *a)
     282             : {
     283             :     ArrayType  *r;
     284       38750 :     int         n = ARRNELEMS(a);
     285             : 
     286       38750 :     r = new_intArrayType(n);
     287       38750 :     memcpy(ARRPTR(r), ARRPTR(a), n * sizeof(int32));
     288       38750 :     return r;
     289             : }
     290             : 
     291             : /* num for compressed key */
     292             : int
     293       56340 : internal_size(int *a, int len)
     294             : {
     295             :     int         i;
     296       56340 :     int64       size = 0;
     297             : 
     298    13435652 :     for (i = 0; i < len; i += 2)
     299             :     {
     300    13379312 :         if (!i || a[i] != a[i - 1]) /* do not count repeated range */
     301    13379312 :             size += (int64) (a[i + 1]) - (int64) (a[i]) + 1;
     302             :     }
     303             : 
     304       56340 :     if (size > (int64) INT_MAX || size < (int64) INT_MIN)
     305           0 :         return -1;              /* overflow */
     306       56340 :     return (int) size;
     307             : }
     308             : 
     309             : /* unique-ify elements of r in-place ... r must be sorted already */
     310             : ArrayType *
     311     4575612 : _int_unique(ArrayType *r)
     312             : {
     313     4575612 :     int         num = ARRNELEMS(r);
     314             :     bool        duplicates_found;   /* not used */
     315             : 
     316     4575612 :     num = qunique_arg(ARRPTR(r), num, sizeof(int), isort_cmp,
     317             :                       &duplicates_found);
     318             : 
     319     4575612 :     return resize_intArrayType(r, num);
     320             : }
     321             : 
     322             : void
     323           0 : gensign(BITVECP sign, int *a, int len, int siglen)
     324             : {
     325             :     int         i;
     326             : 
     327             :     /* we assume that the sign vector is previously zeroed */
     328           0 :     for (i = 0; i < len; i++)
     329             :     {
     330           0 :         HASH(sign, *a, siglen);
     331           0 :         a++;
     332             :     }
     333           0 : }
     334             : 
     335             : int32
     336           2 : intarray_match_first(ArrayType *a, int32 elem)
     337             : {
     338             :     int32      *aa,
     339             :                 c,
     340             :                 i;
     341             : 
     342           2 :     CHECKARRVALID(a);
     343           2 :     c = ARRNELEMS(a);
     344           2 :     aa = ARRPTR(a);
     345           4 :     for (i = 0; i < c; i++)
     346           4 :         if (aa[i] == elem)
     347           2 :             return (i + 1);
     348           0 :     return 0;
     349             : }
     350             : 
     351             : ArrayType *
     352           8 : intarray_add_elem(ArrayType *a, int32 elem)
     353             : {
     354             :     ArrayType  *result;
     355             :     int32      *r;
     356             :     int32       c;
     357             : 
     358           8 :     CHECKARRVALID(a);
     359           8 :     c = ARRNELEMS(a);
     360           8 :     result = new_intArrayType(c + 1);
     361           8 :     r = ARRPTR(result);
     362           8 :     if (c > 0)
     363           8 :         memcpy(r, ARRPTR(a), c * sizeof(int32));
     364           8 :     r[c] = elem;
     365           8 :     return result;
     366             : }
     367             : 
     368             : ArrayType *
     369           2 : intarray_concat_arrays(ArrayType *a, ArrayType *b)
     370             : {
     371             :     ArrayType  *result;
     372           2 :     int32       ac = ARRNELEMS(a);
     373           2 :     int32       bc = ARRNELEMS(b);
     374             : 
     375           2 :     CHECKARRVALID(a);
     376           2 :     CHECKARRVALID(b);
     377           2 :     result = new_intArrayType(ac + bc);
     378           2 :     if (ac)
     379           2 :         memcpy(ARRPTR(result), ARRPTR(a), ac * sizeof(int32));
     380           2 :     if (bc)
     381           2 :         memcpy(ARRPTR(result) + ac, ARRPTR(b), bc * sizeof(int32));
     382           2 :     return result;
     383             : }
     384             : 
     385             : ArrayType *
     386           2 : int_to_intset(int32 elem)
     387             : {
     388             :     ArrayType  *result;
     389             :     int32      *aa;
     390             : 
     391           2 :     result = new_intArrayType(1);
     392           2 :     aa = ARRPTR(result);
     393           2 :     aa[0] = elem;
     394           2 :     return result;
     395             : }
     396             : 
     397             : int
     398   468616818 : compASC(const void *a, const void *b)
     399             : {
     400   468616818 :     return pg_cmp_s32(*(const int32 *) a, *(const int32 *) b);
     401             : }
     402             : 
     403             : int
     404          12 : compDESC(const void *a, const void *b)
     405             : {
     406          12 :     return pg_cmp_s32(*(const int32 *) b, *(const int32 *) a);
     407             : }

Generated by: LCOV version 1.14