LCOV - differential code coverage report
Current view: top level - src/backend/nodes - multibitmapset.c (source / functions) Coverage Total Hit UNC GNC
Current: Differential Code Coverage HEAD vs 15 Lines: 81.0 % 42 34 8 34
Current Date: 2023-04-08 15:15:32 Functions: 80.0 % 5 4 1 4
Baseline: 15
Baseline Date: 2023-04-08 15:09:40
Legend: Lines: hit not hit

           TLA  Line data    Source code
       1                 : /*-------------------------------------------------------------------------
       2                 :  *
       3                 :  * multibitmapset.c
       4                 :  *    Lists of Bitmapsets
       5                 :  *
       6                 :  * A multibitmapset is useful in situations where members of a set can
       7                 :  * be identified by two small integers; for example, varno and varattno
       8                 :  * of a group of Vars within a query.  The implementation is a List of
       9                 :  * Bitmapsets, so that the empty set can be represented by NIL.  (But,
      10                 :  * as with Bitmapsets, that's not the only allowed representation.)
      11                 :  * The zero-based index of a List element is the first identifying value,
      12                 :  * and the (also zero-based) index of a bit within that Bitmapset is
      13                 :  * the second identifying value.  There is no expectation that the
      14                 :  * Bitmapsets should all be the same size.
      15                 :  *
      16                 :  * The available operations on multibitmapsets are intended to parallel
      17                 :  * those on bitmapsets, for example union and intersection.  So far only
      18                 :  * a small fraction of that has been built out; we'll add more as needed.
      19                 :  *
      20                 :  *
      21                 :  * Copyright (c) 2022-2023, PostgreSQL Global Development Group
      22                 :  *
      23                 :  * IDENTIFICATION
      24                 :  *    src/backend/nodes/multibitmapset.c
      25                 :  *
      26                 :  *-------------------------------------------------------------------------
      27                 :  */
      28                 : #include "postgres.h"
      29                 : 
      30                 : #include "nodes/multibitmapset.h"
      31                 : 
      32                 : 
      33                 : /*
      34                 :  * mbms_add_member
      35                 :  *      Add a new member to a multibitmapset.
      36                 :  *
      37                 :  * The new member is identified by "listidx", the zero-based index of the
      38                 :  * List element it should go into, and "bitidx", which specifies the bit
      39                 :  * number to be set therein.
      40                 :  *
      41                 :  * This is like bms_add_member, but for multibitmapsets.
      42                 :  */
      43                 : List *
      44 GNC       43102 : mbms_add_member(List *a, int listidx, int bitidx)
      45                 : {
      46                 :     Bitmapset  *bms;
      47                 :     ListCell   *lc;
      48                 : 
      49           43102 :     if (listidx < 0 || bitidx < 0)
      50 UNC           0 :         elog(ERROR, "negative multibitmapset member index not allowed");
      51                 :     /* Add empty elements as needed */
      52 GNC      204592 :     while (list_length(a) <= listidx)
      53          161490 :         a = lappend(a, NULL);
      54                 :     /* Update the target element */
      55           43102 :     lc = list_nth_cell(a, listidx);
      56           43102 :     bms = lfirst_node(Bitmapset, lc);
      57           43102 :     bms = bms_add_member(bms, bitidx);
      58           43102 :     lfirst(lc) = bms;
      59           43102 :     return a;
      60                 : }
      61                 : 
      62                 : /*
      63                 :  * mbms_add_members
      64                 :  *      Add all members of set b to set a.
      65                 :  *
      66                 :  * This is a UNION operation, but the left input is modified in-place.
      67                 :  *
      68                 :  * This is like bms_add_members, but for multibitmapsets.
      69                 :  */
      70                 : List *
      71          109138 : mbms_add_members(List *a, const List *b)
      72                 : {
      73                 :     ListCell   *lca,
      74                 :                *lcb;
      75                 : 
      76                 :     /* Add empty elements to a, as needed */
      77          301274 :     while (list_length(a) < list_length(b))
      78          192136 :         a = lappend(a, NULL);
      79                 :     /* forboth will stop at the end of the shorter list, which is fine */
      80          379060 :     forboth(lca, a, lcb, b)
      81                 :     {
      82          269922 :         Bitmapset  *bmsa = lfirst_node(Bitmapset, lca);
      83          269922 :         const Bitmapset *bmsb = lfirst_node(Bitmapset, lcb);
      84                 : 
      85          269922 :         bmsa = bms_add_members(bmsa, bmsb);
      86          269922 :         lfirst(lca) = bmsa;
      87                 :     }
      88          109138 :     return a;
      89                 : }
      90                 : 
      91                 : /*
      92                 :  * mbms_int_members
      93                 :  *      Reduce set a to its intersection with set b.
      94                 :  *
      95                 :  * This is an INTERSECT operation, but the left input is modified in-place.
      96                 :  *
      97                 :  * This is like bms_int_members, but for multibitmapsets.
      98                 :  */
      99                 : List *
     100             125 : mbms_int_members(List *a, const List *b)
     101                 : {
     102                 :     ListCell   *lca,
     103                 :                *lcb;
     104                 : 
     105                 :     /* Remove any elements of a that are no longer of use */
     106             125 :     a = list_truncate(a, list_length(b));
     107                 :     /* forboth will stop at the end of the shorter list, which is fine */
     108            1179 :     forboth(lca, a, lcb, b)
     109                 :     {
     110            1054 :         Bitmapset  *bmsa = lfirst_node(Bitmapset, lca);
     111            1054 :         const Bitmapset *bmsb = lfirst_node(Bitmapset, lcb);
     112                 : 
     113            1054 :         bmsa = bms_int_members(bmsa, bmsb);
     114            1054 :         lfirst(lca) = bmsa;
     115                 :     }
     116             125 :     return a;
     117                 : }
     118                 : 
     119                 : /*
     120                 :  * mbms_is_member
     121                 :  *      Is listidx/bitidx a member of A?
     122                 :  *
     123                 :  * This is like bms_is_member, but for multibitmapsets.
     124                 :  */
     125                 : bool
     126 UNC           0 : mbms_is_member(int listidx, int bitidx, const List *a)
     127                 : {
     128                 :     const Bitmapset *bms;
     129                 : 
     130                 :     /* XXX better to just return false for negative indexes? */
     131               0 :     if (listidx < 0 || bitidx < 0)
     132               0 :         elog(ERROR, "negative multibitmapset member index not allowed");
     133               0 :     if (listidx >= list_length(a))
     134               0 :         return false;
     135               0 :     bms = list_nth_node(Bitmapset, a, listidx);
     136               0 :     return bms_is_member(bitidx, bms);
     137                 : }
     138                 : 
     139                 : /*
     140                 :  * mbms_overlap_sets
     141                 :  *      Identify the bitmapsets having common members in a and b.
     142                 :  *
     143                 :  * The result is a bitmapset of the list indexes of bitmapsets that overlap.
     144                 :  */
     145                 : Bitmapset *
     146 GNC       17447 : mbms_overlap_sets(const List *a, const List *b)
     147                 : {
     148           17447 :     Bitmapset  *result = NULL;
     149                 :     ListCell   *lca,
     150                 :                *lcb;
     151                 : 
     152                 :     /* forboth will stop at the end of the shorter list, which is fine */
     153           19265 :     forboth(lca, a, lcb, b)
     154                 :     {
     155            1818 :         const Bitmapset *bmsa = lfirst_node(Bitmapset, lca);
     156            1818 :         const Bitmapset *bmsb = lfirst_node(Bitmapset, lcb);
     157                 : 
     158            1818 :         if (bms_overlap(bmsa, bmsb))
     159             493 :             result = bms_add_member(result, foreach_current_index(lca));
     160                 :     }
     161           17447 :     return result;
     162                 : }
        

Generated by: LCOV version v1.16-55-g56c0a2a