LCOV - differential code coverage report
Current view: top level - src/backend/optimizer/util - joininfo.c (source / functions) Coverage Total Hit GIC GNC CBC ECB DCB
Current: Differential Code Coverage HEAD vs 15 Lines: 100.0 % 32 32 5 6 21 9 2
Current Date: 2023-04-08 15:15:32 Functions: 100.0 % 3 3 1 1 1 1
Baseline: 15
Baseline Date: 2023-04-08 15:09:40
Legend: Lines: hit not hit

           TLA  Line data    Source code
       1                 : /*-------------------------------------------------------------------------
       2                 :  *
       3                 :  * joininfo.c
       4                 :  *    joininfo list manipulation routines
       5                 :  *
       6                 :  * Portions Copyright (c) 1996-2023, PostgreSQL Global Development Group
       7                 :  * Portions Copyright (c) 1994, Regents of the University of California
       8                 :  *
       9                 :  *
      10                 :  * IDENTIFICATION
      11                 :  *    src/backend/optimizer/util/joininfo.c
      12                 :  *
      13                 :  *-------------------------------------------------------------------------
      14                 :  */
      15                 : #include "postgres.h"
      16                 : 
      17                 : #include "optimizer/joininfo.h"
      18                 : #include "optimizer/pathnode.h"
      19                 : #include "optimizer/paths.h"
      20                 : 
      21                 : 
      22                 : /*
      23                 :  * have_relevant_joinclause
      24                 :  *      Detect whether there is a joinclause that involves
      25                 :  *      the two given relations.
      26                 :  *
      27                 :  * Note: the joinclause does not have to be evaluable with only these two
      28                 :  * relations.  This is intentional.  For example consider
      29                 :  *      SELECT * FROM a, b, c WHERE a.x = (b.y + c.z)
      30                 :  * If a is much larger than the other tables, it may be worthwhile to
      31                 :  * cross-join b and c and then use an inner indexscan on a.x.  Therefore
      32                 :  * we should consider this joinclause as reason to join b to c, even though
      33                 :  * it can't be applied at that join step.
      34                 :  */
      35                 : bool
      36 CBC      141471 : have_relevant_joinclause(PlannerInfo *root,
      37                 :                          RelOptInfo *rel1, RelOptInfo *rel2)
      38                 : {
      39          141471 :     bool        result = false;
      40                 :     List       *joininfo;
      41                 :     Relids      other_relids;
      42                 :     ListCell   *l;
      43                 : 
      44                 :     /*
      45                 :      * We could scan either relation's joininfo list; may as well use the
      46                 :      * shorter one.
      47                 :      */
      48          141471 :     if (list_length(rel1->joininfo) <= list_length(rel2->joininfo))
      49                 :     {
      50          106996 :         joininfo = rel1->joininfo;
      51          106996 :         other_relids = rel2->relids;
      52                 :     }
      53                 :     else
      54                 :     {
      55           34475 :         joininfo = rel2->joininfo;
      56           34475 :         other_relids = rel1->relids;
      57                 :     }
      58                 : 
      59          156764 :     foreach(l, joininfo)
      60                 :     {
      61           75709 :         RestrictInfo *rinfo = (RestrictInfo *) lfirst(l);
      62                 : 
      63           75709 :         if (bms_overlap(other_relids, rinfo->required_relids))
      64                 :         {
      65           60416 :             result = true;
      66           60416 :             break;
      67                 :         }
      68                 :     }
      69                 : 
      70                 :     /*
      71                 :      * We also need to check the EquivalenceClass data structure, which might
      72                 :      * contain relationships not emitted into the joininfo lists.
      73                 :      */
      74          141471 :     if (!result && rel1->has_eclass_joins && rel2->has_eclass_joins)
      75           60654 :         result = have_relevant_eclass_joinclause(root, rel1, rel2);
      76                 : 
      77          141471 :     return result;
      78                 : }
      79                 : 
      80                 : 
      81                 : /*
      82                 :  * add_join_clause_to_rels
      83                 :  *    Add 'restrictinfo' to the joininfo list of each relation it requires.
      84                 :  *
      85                 :  * Note that the same copy of the restrictinfo node is linked to by all the
      86                 :  * lists it is in.  This allows us to exploit caching of information about
      87                 :  * the restriction clause (but we must be careful that the information does
      88                 :  * not depend on context).
      89                 :  *
      90                 :  * 'restrictinfo' describes the join clause
      91                 :  * 'join_relids' is the set of relations participating in the join clause
      92                 :  *               (some of these could be outer joins)
      93                 :  */
      94                 : void
      95           27014 : add_join_clause_to_rels(PlannerInfo *root,
      96                 :                         RestrictInfo *restrictinfo,
      97                 :                         Relids join_relids)
      98                 : {
      99                 :     int         cur_relid;
     100                 : 
     101           27014 :     cur_relid = -1;
     102           87039 :     while ((cur_relid = bms_next_member(join_relids, cur_relid)) >= 0)
     103                 :     {
     104 GNC       60025 :         RelOptInfo *rel = find_base_rel_ignore_join(root, cur_relid);
     105                 : 
     106                 :         /* We only need to add the clause to baserels */
     107           60025 :         if (rel == NULL)
     108            3094 :             continue;
     109 GIC       56931 :         rel->joininfo = lappend(rel->joininfo, restrictinfo);
     110 ECB             :     }
     111 CBC       27014 : }
     112 ECB             : 
     113                 : /*
     114                 :  * remove_join_clause_from_rels
     115                 :  *    Delete 'restrictinfo' from all the joininfo lists it is in
     116                 :  *
     117                 :  * This reverses the effect of add_join_clause_to_rels.  It's used when we
     118                 :  * discover that a relation need not be joined at all.
     119                 :  *
     120                 :  * 'restrictinfo' describes the join clause
     121                 :  * 'join_relids' is the set of relations participating in the join clause
     122                 :  *               (some of these could be outer joins)
     123                 :  */
     124                 : void
     125 GIC        4096 : remove_join_clause_from_rels(PlannerInfo *root,
     126                 :                              RestrictInfo *restrictinfo,
     127                 :                              Relids join_relids)
     128 ECB             : {
     129                 :     int         cur_relid;
     130                 : 
     131 GIC        4096 :     cur_relid = -1;
     132           12385 :     while ((cur_relid = bms_next_member(join_relids, cur_relid)) >= 0)
     133                 :     {
     134 GNC        8289 :         RelOptInfo *rel = find_base_rel_ignore_join(root, cur_relid);
     135                 : 
     136                 :         /* We would only have added the clause to baserels */
     137            8289 :         if (rel == NULL)
     138              79 :             continue;
     139 ECB             : 
     140                 :         /*
     141                 :          * Remove the restrictinfo from the list.  Pointer comparison is
     142                 :          * sufficient.
     143                 :          */
     144 CBC        8210 :         Assert(list_member_ptr(rel->joininfo, restrictinfo));
     145            8210 :         rel->joininfo = list_delete_ptr(rel->joininfo, restrictinfo);
     146                 :     }
     147 GIC        4096 : }
        

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