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 17:13:01 Functions: 80.0 % 5 4 1 4
Baseline: 15 Line coverage date bins:
Baseline Date: 2023-04-08 15:09:40 (120,180] days: 81.0 % 42 34 8 34
Legend: Lines: hit not hit Function coverage date bins:
(120,180] days: 80.0 % 5 4 1 4

 Age         Owner                  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 *
  144 tgl                        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)
  144 tgl                        50 UNC           0 :         elog(ERROR, "negative multibitmapset member index not allowed");
                                 51                 :     /* Add empty elements as needed */
  144 tgl                        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
  144 tgl                       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 *
  144 tgl                       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