LCOV - code coverage report
Current view: top level - src/backend/optimizer/path - clausesel.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 229 240 95.4 %
Date: 2025-01-18 04:15:08 Functions: 8 8 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * clausesel.c
       4             :  *    Routines to compute clause selectivities
       5             :  *
       6             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       7             :  * Portions Copyright (c) 1994, Regents of the University of California
       8             :  *
       9             :  *
      10             :  * IDENTIFICATION
      11             :  *    src/backend/optimizer/path/clausesel.c
      12             :  *
      13             :  *-------------------------------------------------------------------------
      14             :  */
      15             : #include "postgres.h"
      16             : 
      17             : #include "nodes/nodeFuncs.h"
      18             : #include "optimizer/clauses.h"
      19             : #include "optimizer/optimizer.h"
      20             : #include "optimizer/pathnode.h"
      21             : #include "optimizer/plancat.h"
      22             : #include "statistics/statistics.h"
      23             : #include "utils/fmgroids.h"
      24             : #include "utils/lsyscache.h"
      25             : #include "utils/selfuncs.h"
      26             : 
      27             : /*
      28             :  * Data structure for accumulating info about possible range-query
      29             :  * clause pairs in clauselist_selectivity.
      30             :  */
      31             : typedef struct RangeQueryClause
      32             : {
      33             :     struct RangeQueryClause *next;  /* next in linked list */
      34             :     Node       *var;            /* The common variable of the clauses */
      35             :     bool        have_lobound;   /* found a low-bound clause yet? */
      36             :     bool        have_hibound;   /* found a high-bound clause yet? */
      37             :     Selectivity lobound;        /* Selectivity of a var > something clause */
      38             :     Selectivity hibound;        /* Selectivity of a var < something clause */
      39             : } RangeQueryClause;
      40             : 
      41             : static void addRangeClause(RangeQueryClause **rqlist, Node *clause,
      42             :                            bool varonleft, bool isLTsel, Selectivity s2);
      43             : static RelOptInfo *find_single_rel_for_clauses(PlannerInfo *root,
      44             :                                                List *clauses);
      45             : static Selectivity clauselist_selectivity_or(PlannerInfo *root,
      46             :                                              List *clauses,
      47             :                                              int varRelid,
      48             :                                              JoinType jointype,
      49             :                                              SpecialJoinInfo *sjinfo,
      50             :                                              bool use_extended_stats);
      51             : 
      52             : /****************************************************************************
      53             :  *      ROUTINES TO COMPUTE SELECTIVITIES
      54             :  ****************************************************************************/
      55             : 
      56             : /*
      57             :  * clauselist_selectivity -
      58             :  *    Compute the selectivity of an implicitly-ANDed list of boolean
      59             :  *    expression clauses.  The list can be empty, in which case 1.0
      60             :  *    must be returned.  List elements may be either RestrictInfos
      61             :  *    or bare expression clauses --- the former is preferred since
      62             :  *    it allows caching of results.
      63             :  *
      64             :  * See clause_selectivity() for the meaning of the additional parameters.
      65             :  *
      66             :  * The basic approach is to apply extended statistics first, on as many
      67             :  * clauses as possible, in order to capture cross-column dependencies etc.
      68             :  * The remaining clauses are then estimated by taking the product of their
      69             :  * selectivities, but that's only right if they have independent
      70             :  * probabilities, and in reality they are often NOT independent even if they
      71             :  * only refer to a single column.  So, we want to be smarter where we can.
      72             :  *
      73             :  * We also recognize "range queries", such as "x > 34 AND x < 42".  Clauses
      74             :  * are recognized as possible range query components if they are restriction
      75             :  * opclauses whose operators have scalarltsel or a related function as their
      76             :  * restriction selectivity estimator.  We pair up clauses of this form that
      77             :  * refer to the same variable.  An unpairable clause of this kind is simply
      78             :  * multiplied into the selectivity product in the normal way.  But when we
      79             :  * find a pair, we know that the selectivities represent the relative
      80             :  * positions of the low and high bounds within the column's range, so instead
      81             :  * of figuring the selectivity as hisel * losel, we can figure it as hisel +
      82             :  * losel - 1.  (To visualize this, see that hisel is the fraction of the range
      83             :  * below the high bound, while losel is the fraction above the low bound; so
      84             :  * hisel can be interpreted directly as a 0..1 value but we need to convert
      85             :  * losel to 1-losel before interpreting it as a value.  Then the available
      86             :  * range is 1-losel to hisel.  However, this calculation double-excludes
      87             :  * nulls, so really we need hisel + losel + null_frac - 1.)
      88             :  *
      89             :  * If either selectivity is exactly DEFAULT_INEQ_SEL, we forget this equation
      90             :  * and instead use DEFAULT_RANGE_INEQ_SEL.  The same applies if the equation
      91             :  * yields an impossible (negative) result.
      92             :  *
      93             :  * A free side-effect is that we can recognize redundant inequalities such
      94             :  * as "x < 4 AND x < 5"; only the tighter constraint will be counted.
      95             :  *
      96             :  * Of course this is all very dependent on the behavior of the inequality
      97             :  * selectivity functions; perhaps some day we can generalize the approach.
      98             :  */
      99             : Selectivity
     100     2361776 : clauselist_selectivity(PlannerInfo *root,
     101             :                        List *clauses,
     102             :                        int varRelid,
     103             :                        JoinType jointype,
     104             :                        SpecialJoinInfo *sjinfo)
     105             : {
     106     2361776 :     return clauselist_selectivity_ext(root, clauses, varRelid,
     107             :                                       jointype, sjinfo, true);
     108             : }
     109             : 
     110             : /*
     111             :  * clauselist_selectivity_ext -
     112             :  *    Extended version of clauselist_selectivity().  If "use_extended_stats"
     113             :  *    is false, all extended statistics will be ignored, and only per-column
     114             :  *    statistics will be used.
     115             :  */
     116             : Selectivity
     117     2369068 : clauselist_selectivity_ext(PlannerInfo *root,
     118             :                            List *clauses,
     119             :                            int varRelid,
     120             :                            JoinType jointype,
     121             :                            SpecialJoinInfo *sjinfo,
     122             :                            bool use_extended_stats)
     123             : {
     124     2369068 :     Selectivity s1 = 1.0;
     125             :     RelOptInfo *rel;
     126     2369068 :     Bitmapset  *estimatedclauses = NULL;
     127     2369068 :     RangeQueryClause *rqlist = NULL;
     128             :     ListCell   *l;
     129             :     int         listidx;
     130             : 
     131             :     /*
     132             :      * If there's exactly one clause, just go directly to
     133             :      * clause_selectivity_ext(). None of what we might do below is relevant.
     134             :      */
     135     2369068 :     if (list_length(clauses) == 1)
     136     1189744 :         return clause_selectivity_ext(root, (Node *) linitial(clauses),
     137             :                                       varRelid, jointype, sjinfo,
     138             :                                       use_extended_stats);
     139             : 
     140             :     /*
     141             :      * Determine if these clauses reference a single relation.  If so, and if
     142             :      * it has extended statistics, try to apply those.
     143             :      */
     144     1179324 :     rel = find_single_rel_for_clauses(root, clauses);
     145     1179324 :     if (use_extended_stats && rel && rel->rtekind == RTE_RELATION && rel->statlist != NIL)
     146             :     {
     147             :         /*
     148             :          * Estimate as many clauses as possible using extended statistics.
     149             :          *
     150             :          * 'estimatedclauses' is populated with the 0-based list position
     151             :          * index of clauses estimated here, and that should be ignored below.
     152             :          */
     153        2364 :         s1 = statext_clauselist_selectivity(root, clauses, varRelid,
     154             :                                             jointype, sjinfo, rel,
     155             :                                             &estimatedclauses, false);
     156             :     }
     157             : 
     158             :     /*
     159             :      * Apply normal selectivity estimates for remaining clauses. We'll be
     160             :      * careful to skip any clauses which were already estimated above.
     161             :      *
     162             :      * Anything that doesn't look like a potential rangequery clause gets
     163             :      * multiplied into s1 and forgotten. Anything that does gets inserted into
     164             :      * an rqlist entry.
     165             :      */
     166     1179324 :     listidx = -1;
     167     1933352 :     foreach(l, clauses)
     168             :     {
     169      754028 :         Node       *clause = (Node *) lfirst(l);
     170             :         RestrictInfo *rinfo;
     171             :         Selectivity s2;
     172             : 
     173      754028 :         listidx++;
     174             : 
     175             :         /*
     176             :          * Skip this clause if it's already been estimated by some other
     177             :          * statistics above.
     178             :          */
     179      754028 :         if (bms_is_member(listidx, estimatedclauses))
     180        4662 :             continue;
     181             : 
     182             :         /* Compute the selectivity of this clause in isolation */
     183      749366 :         s2 = clause_selectivity_ext(root, clause, varRelid, jointype, sjinfo,
     184             :                                     use_extended_stats);
     185             : 
     186             :         /*
     187             :          * Check for being passed a RestrictInfo.
     188             :          *
     189             :          * If it's a pseudoconstant RestrictInfo, then s2 is either 1.0 or
     190             :          * 0.0; just use that rather than looking for range pairs.
     191             :          */
     192      749366 :         if (IsA(clause, RestrictInfo))
     193             :         {
     194      748042 :             rinfo = (RestrictInfo *) clause;
     195      748042 :             if (rinfo->pseudoconstant)
     196             :             {
     197       25724 :                 s1 = s1 * s2;
     198       25724 :                 continue;
     199             :             }
     200      722318 :             clause = (Node *) rinfo->clause;
     201             :         }
     202             :         else
     203        1324 :             rinfo = NULL;
     204             : 
     205             :         /*
     206             :          * See if it looks like a restriction clause with a pseudoconstant on
     207             :          * one side.  (Anything more complicated than that might not behave in
     208             :          * the simple way we are expecting.)  Most of the tests here can be
     209             :          * done more efficiently with rinfo than without.
     210             :          */
     211      723642 :         if (is_opclause(clause) && list_length(((OpExpr *) clause)->args) == 2)
     212             :         {
     213      620720 :             OpExpr     *expr = (OpExpr *) clause;
     214      620720 :             bool        varonleft = true;
     215             :             bool        ok;
     216             : 
     217      620720 :             if (rinfo)
     218             :             {
     219     1025198 :                 ok = (rinfo->num_base_rels == 1) &&
     220      399318 :                     (is_pseudo_constant_clause_relids(lsecond(expr->args),
     221        6468 :                                                       rinfo->right_relids) ||
     222        6468 :                      (varonleft = false,
     223        6468 :                       is_pseudo_constant_clause_relids(linitial(expr->args),
     224             :                                                        rinfo->left_relids)));
     225             :             }
     226             :             else
     227             :             {
     228        2544 :                 ok = (NumRelids(root, clause) == 1) &&
     229        1236 :                     (is_pseudo_constant_clause(lsecond(expr->args)) ||
     230           0 :                      (varonleft = false,
     231           0 :                       is_pseudo_constant_clause(linitial(expr->args))));
     232             :             }
     233             : 
     234      620720 :             if (ok)
     235             :             {
     236             :                 /*
     237             :                  * If it's not a "<"/"<="/">"/">=" operator, just merge the
     238             :                  * selectivity in generically.  But if it's the right oprrest,
     239             :                  * add the clause to rqlist for later processing.
     240             :                  */
     241      399962 :                 switch (get_oprrest(expr->opno))
     242             :                 {
     243       13188 :                     case F_SCALARLTSEL:
     244             :                     case F_SCALARLESEL:
     245       13188 :                         addRangeClause(&rqlist, clause,
     246             :                                        varonleft, true, s2);
     247       13188 :                         break;
     248       35544 :                     case F_SCALARGTSEL:
     249             :                     case F_SCALARGESEL:
     250       35544 :                         addRangeClause(&rqlist, clause,
     251             :                                        varonleft, false, s2);
     252       35544 :                         break;
     253      351230 :                     default:
     254             :                         /* Just merge the selectivity in generically */
     255      351230 :                         s1 = s1 * s2;
     256      351230 :                         break;
     257             :                 }
     258      399962 :                 continue;       /* drop to loop bottom */
     259             :             }
     260             :         }
     261             : 
     262             :         /* Not the right form, so treat it generically. */
     263      323680 :         s1 = s1 * s2;
     264             :     }
     265             : 
     266             :     /*
     267             :      * Now scan the rangequery pair list.
     268             :      */
     269     1219046 :     while (rqlist != NULL)
     270             :     {
     271             :         RangeQueryClause *rqnext;
     272             : 
     273       39722 :         if (rqlist->have_lobound && rqlist->have_hibound)
     274        8512 :         {
     275             :             /* Successfully matched a pair of range clauses */
     276             :             Selectivity s2;
     277             : 
     278             :             /*
     279             :              * Exact equality to the default value probably means the
     280             :              * selectivity function punted.  This is not airtight but should
     281             :              * be good enough.
     282             :              */
     283        8512 :             if (rqlist->hibound == DEFAULT_INEQ_SEL ||
     284        6390 :                 rqlist->lobound == DEFAULT_INEQ_SEL)
     285             :             {
     286        2122 :                 s2 = DEFAULT_RANGE_INEQ_SEL;
     287             :             }
     288             :             else
     289             :             {
     290        6390 :                 s2 = rqlist->hibound + rqlist->lobound - 1.0;
     291             : 
     292             :                 /* Adjust for double-exclusion of NULLs */
     293        6390 :                 s2 += nulltestsel(root, IS_NULL, rqlist->var,
     294             :                                   varRelid, jointype, sjinfo);
     295             : 
     296             :                 /*
     297             :                  * A zero or slightly negative s2 should be converted into a
     298             :                  * small positive value; we probably are dealing with a very
     299             :                  * tight range and got a bogus result due to roundoff errors.
     300             :                  * However, if s2 is very negative, then we probably have
     301             :                  * default selectivity estimates on one or both sides of the
     302             :                  * range that we failed to recognize above for some reason.
     303             :                  */
     304        6390 :                 if (s2 <= 0.0)
     305             :                 {
     306         950 :                     if (s2 < -0.01)
     307             :                     {
     308             :                         /*
     309             :                          * No data available --- use a default estimate that
     310             :                          * is small, but not real small.
     311             :                          */
     312          12 :                         s2 = DEFAULT_RANGE_INEQ_SEL;
     313             :                     }
     314             :                     else
     315             :                     {
     316             :                         /*
     317             :                          * It's just roundoff error; use a small positive
     318             :                          * value
     319             :                          */
     320         938 :                         s2 = 1.0e-10;
     321             :                     }
     322             :                 }
     323             :             }
     324             :             /* Merge in the selectivity of the pair of clauses */
     325        8512 :             s1 *= s2;
     326             :         }
     327             :         else
     328             :         {
     329             :             /* Only found one of a pair, merge it in generically */
     330       31210 :             if (rqlist->have_lobound)
     331       26630 :                 s1 *= rqlist->lobound;
     332             :             else
     333        4580 :                 s1 *= rqlist->hibound;
     334             :         }
     335             :         /* release storage and advance */
     336       39722 :         rqnext = rqlist->next;
     337       39722 :         pfree(rqlist);
     338       39722 :         rqlist = rqnext;
     339             :     }
     340             : 
     341     1179324 :     return s1;
     342             : }
     343             : 
     344             : /*
     345             :  * clauselist_selectivity_or -
     346             :  *    Compute the selectivity of an implicitly-ORed list of boolean
     347             :  *    expression clauses.  The list can be empty, in which case 0.0
     348             :  *    must be returned.  List elements may be either RestrictInfos
     349             :  *    or bare expression clauses --- the former is preferred since
     350             :  *    it allows caching of results.
     351             :  *
     352             :  * See clause_selectivity() for the meaning of the additional parameters.
     353             :  *
     354             :  * The basic approach is to apply extended statistics first, on as many
     355             :  * clauses as possible, in order to capture cross-column dependencies etc.
     356             :  * The remaining clauses are then estimated as if they were independent.
     357             :  */
     358             : static Selectivity
     359       12564 : clauselist_selectivity_or(PlannerInfo *root,
     360             :                           List *clauses,
     361             :                           int varRelid,
     362             :                           JoinType jointype,
     363             :                           SpecialJoinInfo *sjinfo,
     364             :                           bool use_extended_stats)
     365             : {
     366       12564 :     Selectivity s1 = 0.0;
     367             :     RelOptInfo *rel;
     368       12564 :     Bitmapset  *estimatedclauses = NULL;
     369             :     ListCell   *lc;
     370             :     int         listidx;
     371             : 
     372             :     /*
     373             :      * Determine if these clauses reference a single relation.  If so, and if
     374             :      * it has extended statistics, try to apply those.
     375             :      */
     376       12564 :     rel = find_single_rel_for_clauses(root, clauses);
     377       12564 :     if (use_extended_stats && rel && rel->rtekind == RTE_RELATION && rel->statlist != NIL)
     378             :     {
     379             :         /*
     380             :          * Estimate as many clauses as possible using extended statistics.
     381             :          *
     382             :          * 'estimatedclauses' is populated with the 0-based list position
     383             :          * index of clauses estimated here, and that should be ignored below.
     384             :          */
     385         114 :         s1 = statext_clauselist_selectivity(root, clauses, varRelid,
     386             :                                             jointype, sjinfo, rel,
     387             :                                             &estimatedclauses, true);
     388             :     }
     389             : 
     390             :     /*
     391             :      * Estimate the remaining clauses as if they were independent.
     392             :      *
     393             :      * Selectivities for an OR clause are computed as s1+s2 - s1*s2 to account
     394             :      * for the probable overlap of selected tuple sets.
     395             :      *
     396             :      * XXX is this too conservative?
     397             :      */
     398       12564 :     listidx = -1;
     399       41384 :     foreach(lc, clauses)
     400             :     {
     401             :         Selectivity s2;
     402             : 
     403       28820 :         listidx++;
     404             : 
     405             :         /*
     406             :          * Skip this clause if it's already been estimated by some other
     407             :          * statistics above.
     408             :          */
     409       28820 :         if (bms_is_member(listidx, estimatedclauses))
     410         240 :             continue;
     411             : 
     412       28580 :         s2 = clause_selectivity_ext(root, (Node *) lfirst(lc), varRelid,
     413             :                                     jointype, sjinfo, use_extended_stats);
     414             : 
     415       28580 :         s1 = s1 + s2 - s1 * s2;
     416             :     }
     417             : 
     418       12564 :     return s1;
     419             : }
     420             : 
     421             : /*
     422             :  * addRangeClause --- add a new range clause for clauselist_selectivity
     423             :  *
     424             :  * Here is where we try to match up pairs of range-query clauses
     425             :  */
     426             : static void
     427       48732 : addRangeClause(RangeQueryClause **rqlist, Node *clause,
     428             :                bool varonleft, bool isLTsel, Selectivity s2)
     429             : {
     430             :     RangeQueryClause *rqelem;
     431             :     Node       *var;
     432             :     bool        is_lobound;
     433             : 
     434       48732 :     if (varonleft)
     435             :     {
     436       48510 :         var = get_leftop((Expr *) clause);
     437       48510 :         is_lobound = !isLTsel;  /* x < something is high bound */
     438             :     }
     439             :     else
     440             :     {
     441         222 :         var = get_rightop((Expr *) clause);
     442         222 :         is_lobound = isLTsel;   /* something < x is low bound */
     443             :     }
     444             : 
     445       49966 :     for (rqelem = *rqlist; rqelem; rqelem = rqelem->next)
     446             :     {
     447             :         /*
     448             :          * We use full equal() here because the "var" might be a function of
     449             :          * one or more attributes of the same relation...
     450             :          */
     451       10244 :         if (!equal(var, rqelem->var))
     452        1234 :             continue;
     453             :         /* Found the right group to put this clause in */
     454        9010 :         if (is_lobound)
     455             :         {
     456         420 :             if (!rqelem->have_lobound)
     457             :             {
     458         180 :                 rqelem->have_lobound = true;
     459         180 :                 rqelem->lobound = s2;
     460             :             }
     461             :             else
     462             :             {
     463             : 
     464             :                 /*------
     465             :                  * We have found two similar clauses, such as
     466             :                  * x < y AND x <= z.
     467             :                  * Keep only the more restrictive one.
     468             :                  *------
     469             :                  */
     470         240 :                 if (rqelem->lobound > s2)
     471          60 :                     rqelem->lobound = s2;
     472             :             }
     473             :         }
     474             :         else
     475             :         {
     476        8590 :             if (!rqelem->have_hibound)
     477             :             {
     478        8332 :                 rqelem->have_hibound = true;
     479        8332 :                 rqelem->hibound = s2;
     480             :             }
     481             :             else
     482             :             {
     483             : 
     484             :                 /*------
     485             :                  * We have found two similar clauses, such as
     486             :                  * x > y AND x >= z.
     487             :                  * Keep only the more restrictive one.
     488             :                  *------
     489             :                  */
     490         258 :                 if (rqelem->hibound > s2)
     491          96 :                     rqelem->hibound = s2;
     492             :             }
     493             :         }
     494        9010 :         return;
     495             :     }
     496             : 
     497             :     /* No matching var found, so make a new clause-pair data structure */
     498       39722 :     rqelem = (RangeQueryClause *) palloc(sizeof(RangeQueryClause));
     499       39722 :     rqelem->var = var;
     500       39722 :     if (is_lobound)
     501             :     {
     502       34962 :         rqelem->have_lobound = true;
     503       34962 :         rqelem->have_hibound = false;
     504       34962 :         rqelem->lobound = s2;
     505             :     }
     506             :     else
     507             :     {
     508        4760 :         rqelem->have_lobound = false;
     509        4760 :         rqelem->have_hibound = true;
     510        4760 :         rqelem->hibound = s2;
     511             :     }
     512       39722 :     rqelem->next = *rqlist;
     513       39722 :     *rqlist = rqelem;
     514             : }
     515             : 
     516             : /*
     517             :  * find_single_rel_for_clauses
     518             :  *      Examine each clause in 'clauses' and determine if all clauses
     519             :  *      reference only a single relation.  If so return that relation,
     520             :  *      otherwise return NULL.
     521             :  */
     522             : static RelOptInfo *
     523     1194432 : find_single_rel_for_clauses(PlannerInfo *root, List *clauses)
     524             : {
     525     1194432 :     int         lastrelid = 0;
     526             :     ListCell   *l;
     527             : 
     528     1642058 :     foreach(l, clauses)
     529             :     {
     530      633452 :         RestrictInfo *rinfo = (RestrictInfo *) lfirst(l);
     531             :         int         relid;
     532             : 
     533             :         /*
     534             :          * If we have a list of bare clauses rather than RestrictInfos, we
     535             :          * could pull out their relids the hard way with pull_varnos().
     536             :          * However, currently the extended-stats machinery won't do anything
     537             :          * with non-RestrictInfo clauses anyway, so there's no point in
     538             :          * spending extra cycles; just fail if that's what we have.
     539             :          *
     540             :          * An exception to that rule is if we have a bare BoolExpr AND clause.
     541             :          * We treat this as a special case because the restrictinfo machinery
     542             :          * doesn't build RestrictInfos on top of AND clauses.
     543             :          */
     544      633452 :         if (is_andclause(rinfo))
     545             :         {
     546             :             RelOptInfo *rel;
     547             : 
     548        2544 :             rel = find_single_rel_for_clauses(root,
     549             :                                               ((BoolExpr *) rinfo)->args);
     550             : 
     551        2544 :             if (rel == NULL)
     552      185826 :                 return NULL;
     553        2024 :             if (lastrelid == 0)
     554        1012 :                 lastrelid = rel->relid;
     555        1012 :             else if (rel->relid != lastrelid)
     556           0 :                 return NULL;
     557             : 
     558       36148 :             continue;
     559             :         }
     560             : 
     561      630908 :         if (!IsA(rinfo, RestrictInfo))
     562        1130 :             return NULL;
     563             : 
     564      629778 :         if (bms_is_empty(rinfo->clause_relids))
     565       34124 :             continue;           /* we can ignore variable-free clauses */
     566      595654 :         if (!bms_get_singleton_member(rinfo->clause_relids, &relid))
     567      180910 :             return NULL;        /* multiple relations in this clause */
     568      414744 :         if (lastrelid == 0)
     569      205602 :             lastrelid = relid;  /* first clause referencing a relation */
     570      209142 :         else if (relid != lastrelid)
     571        3266 :             return NULL;        /* relation not same as last one */
     572             :     }
     573             : 
     574     1008606 :     if (lastrelid != 0)
     575      163616 :         return find_base_rel(root, lastrelid);
     576             : 
     577      844990 :     return NULL;                /* no clauses */
     578             : }
     579             : 
     580             : /*
     581             :  * treat_as_join_clause -
     582             :  *    Decide whether an operator clause is to be handled by the
     583             :  *    restriction or join estimator.  Subroutine for clause_selectivity().
     584             :  */
     585             : static inline bool
     586      841916 : treat_as_join_clause(PlannerInfo *root, Node *clause, RestrictInfo *rinfo,
     587             :                      int varRelid, SpecialJoinInfo *sjinfo)
     588             : {
     589      841916 :     if (varRelid != 0)
     590             :     {
     591             :         /*
     592             :          * Caller is forcing restriction mode (eg, because we are examining an
     593             :          * inner indexscan qual).
     594             :          */
     595      306784 :         return false;
     596             :     }
     597      535132 :     else if (sjinfo == NULL)
     598             :     {
     599             :         /*
     600             :          * It must be a restriction clause, since it's being evaluated at a
     601             :          * scan node.
     602             :          */
     603      326164 :         return false;
     604             :     }
     605             :     else
     606             :     {
     607             :         /*
     608             :          * Otherwise, it's a join if there's more than one base relation used.
     609             :          * We can optimize this calculation if an rinfo was passed.
     610             :          *
     611             :          * XXX  Since we know the clause is being evaluated at a join, the
     612             :          * only way it could be single-relation is if it was delayed by outer
     613             :          * joins.  We intentionally count only baserels here, not OJs that
     614             :          * might be present in rinfo->clause_relids, so that we direct such
     615             :          * cases to the restriction qual estimators not join estimators.
     616             :          * Eventually some notice should be taken of the possibility of
     617             :          * injected nulls, but we'll likely want to do that in the restriction
     618             :          * estimators rather than starting to treat such cases as join quals.
     619             :          */
     620      208968 :         if (rinfo)
     621      208506 :             return (rinfo->num_base_rels > 1);
     622             :         else
     623         462 :             return (NumRelids(root, clause) > 1);
     624             :     }
     625             : }
     626             : 
     627             : 
     628             : /*
     629             :  * clause_selectivity -
     630             :  *    Compute the selectivity of a general boolean expression clause.
     631             :  *
     632             :  * The clause can be either a RestrictInfo or a plain expression.  If it's
     633             :  * a RestrictInfo, we try to cache the selectivity for possible re-use,
     634             :  * so passing RestrictInfos is preferred.
     635             :  *
     636             :  * varRelid is either 0 or a rangetable index.
     637             :  *
     638             :  * When varRelid is not 0, only variables belonging to that relation are
     639             :  * considered in computing selectivity; other vars are treated as constants
     640             :  * of unknown values.  This is appropriate for estimating the selectivity of
     641             :  * a join clause that is being used as a restriction clause in a scan of a
     642             :  * nestloop join's inner relation --- varRelid should then be the ID of the
     643             :  * inner relation.
     644             :  *
     645             :  * When varRelid is 0, all variables are treated as variables.  This
     646             :  * is appropriate for ordinary join clauses and restriction clauses.
     647             :  *
     648             :  * jointype is the join type, if the clause is a join clause.  Pass JOIN_INNER
     649             :  * if the clause isn't a join clause.
     650             :  *
     651             :  * sjinfo is NULL for a non-join clause, otherwise it provides additional
     652             :  * context information about the join being performed.  There are some
     653             :  * special cases:
     654             :  *  1. For a special (not INNER) join, sjinfo is always a member of
     655             :  *     root->join_info_list.
     656             :  *  2. For an INNER join, sjinfo is just a transient struct, and only the
     657             :  *     relids and jointype fields in it can be trusted.
     658             :  * It is possible for jointype to be different from sjinfo->jointype.
     659             :  * This indicates we are considering a variant join: either with
     660             :  * the LHS and RHS switched, or with one input unique-ified.
     661             :  *
     662             :  * Note: when passing nonzero varRelid, it's normally appropriate to set
     663             :  * jointype == JOIN_INNER, sjinfo == NULL, even if the clause is really a
     664             :  * join clause; because we aren't treating it as a join clause.
     665             :  */
     666             : Selectivity
     667      463050 : clause_selectivity(PlannerInfo *root,
     668             :                    Node *clause,
     669             :                    int varRelid,
     670             :                    JoinType jointype,
     671             :                    SpecialJoinInfo *sjinfo)
     672             : {
     673      463050 :     return clause_selectivity_ext(root, clause, varRelid,
     674             :                                   jointype, sjinfo, true);
     675             : }
     676             : 
     677             : /*
     678             :  * clause_selectivity_ext -
     679             :  *    Extended version of clause_selectivity().  If "use_extended_stats" is
     680             :  *    false, all extended statistics will be ignored, and only per-column
     681             :  *    statistics will be used.
     682             :  */
     683             : Selectivity
     684     2439176 : clause_selectivity_ext(PlannerInfo *root,
     685             :                        Node *clause,
     686             :                        int varRelid,
     687             :                        JoinType jointype,
     688             :                        SpecialJoinInfo *sjinfo,
     689             :                        bool use_extended_stats)
     690             : {
     691     2439176 :     Selectivity s1 = 0.5;       /* default for any unhandled clause type */
     692     2439176 :     RestrictInfo *rinfo = NULL;
     693     2439176 :     bool        cacheable = false;
     694             : 
     695     2439176 :     if (clause == NULL)         /* can this still happen? */
     696           0 :         return s1;
     697             : 
     698     2439176 :     if (IsA(clause, RestrictInfo))
     699             :     {
     700     2420376 :         rinfo = (RestrictInfo *) clause;
     701             : 
     702             :         /*
     703             :          * If the clause is marked pseudoconstant, then it will be used as a
     704             :          * gating qual and should not affect selectivity estimates; hence
     705             :          * return 1.0.  The only exception is that a constant FALSE may be
     706             :          * taken as having selectivity 0.0, since it will surely mean no rows
     707             :          * out of the plan.  This case is simple enough that we need not
     708             :          * bother caching the result.
     709             :          */
     710     2420376 :         if (rinfo->pseudoconstant)
     711             :         {
     712       26078 :             if (!IsA(rinfo->clause, Const))
     713       25872 :                 return (Selectivity) 1.0;
     714             :         }
     715             : 
     716             :         /*
     717             :          * If possible, cache the result of the selectivity calculation for
     718             :          * the clause.  We can cache if varRelid is zero or the clause
     719             :          * contains only vars of that relid --- otherwise varRelid will affect
     720             :          * the result, so mustn't cache.  Outer join quals might be examined
     721             :          * with either their join's actual jointype or JOIN_INNER, so we need
     722             :          * two cache variables to remember both cases.  Note: we assume the
     723             :          * result won't change if we are switching the input relations or
     724             :          * considering a unique-ified case, so we only need one cache variable
     725             :          * for all non-JOIN_INNER cases.
     726             :          */
     727     2394504 :         if (varRelid == 0 ||
     728      973046 :             rinfo->num_base_rels == 0 ||
     729     1646212 :             (rinfo->num_base_rels == 1 &&
     730      673166 :              bms_is_member(varRelid, rinfo->clause_relids)))
     731             :         {
     732             :             /* Cacheable --- do we already have the result? */
     733     2092356 :             if (jointype == JOIN_INNER)
     734             :             {
     735     1796616 :                 if (rinfo->norm_selec >= 0)
     736     1304630 :                     return rinfo->norm_selec;
     737             :             }
     738             :             else
     739             :             {
     740      295740 :                 if (rinfo->outer_selec >= 0)
     741      186482 :                     return rinfo->outer_selec;
     742             :             }
     743      601244 :             cacheable = true;
     744             :         }
     745             : 
     746             :         /*
     747             :          * Proceed with examination of contained clause.  If the clause is an
     748             :          * OR-clause, we want to look at the variant with sub-RestrictInfos,
     749             :          * so that per-subclause selectivities can be cached.
     750             :          */
     751      903392 :         if (rinfo->orclause)
     752       12524 :             clause = (Node *) rinfo->orclause;
     753             :         else
     754      890868 :             clause = (Node *) rinfo->clause;
     755             :     }
     756             : 
     757      922192 :     if (IsA(clause, Var))
     758             :     {
     759       39538 :         Var        *var = (Var *) clause;
     760             : 
     761             :         /*
     762             :          * We probably shouldn't ever see an uplevel Var here, but if we do,
     763             :          * return the default selectivity...
     764             :          */
     765       39538 :         if (var->varlevelsup == 0 &&
     766         672 :             (varRelid == 0 || varRelid == (int) var->varno))
     767             :         {
     768             :             /* Use the restriction selectivity function for a bool Var */
     769       39226 :             s1 = boolvarsel(root, (Node *) var, varRelid);
     770             :         }
     771             :     }
     772      882654 :     else if (IsA(clause, Const))
     773             :     {
     774             :         /* bool constant is pretty easy... */
     775        3624 :         Const      *con = (Const *) clause;
     776             : 
     777        7132 :         s1 = con->constisnull ? 0.0 :
     778        3508 :             DatumGetBool(con->constvalue) ? 1.0 : 0.0;
     779             :     }
     780      879030 :     else if (IsA(clause, Param))
     781             :     {
     782             :         /* see if we can replace the Param */
     783          24 :         Node       *subst = estimate_expression_value(root, clause);
     784             : 
     785          24 :         if (IsA(subst, Const))
     786             :         {
     787             :             /* bool constant is pretty easy... */
     788           0 :             Const      *con = (Const *) subst;
     789             : 
     790           0 :             s1 = con->constisnull ? 0.0 :
     791           0 :                 DatumGetBool(con->constvalue) ? 1.0 : 0.0;
     792             :         }
     793             :         else
     794             :         {
     795             :             /* XXX any way to do better than default? */
     796             :         }
     797             :     }
     798      879006 :     else if (is_notclause(clause))
     799             :     {
     800             :         /* inverse of the selectivity of the underlying clause */
     801        8196 :         s1 = 1.0 - clause_selectivity_ext(root,
     802        8196 :                                           (Node *) get_notclausearg((Expr *) clause),
     803             :                                           varRelid,
     804             :                                           jointype,
     805             :                                           sjinfo,
     806             :                                           use_extended_stats);
     807             :     }
     808      870810 :     else if (is_andclause(clause))
     809             :     {
     810             :         /* share code with clauselist_selectivity() */
     811        3284 :         s1 = clauselist_selectivity_ext(root,
     812             :                                         ((BoolExpr *) clause)->args,
     813             :                                         varRelid,
     814             :                                         jointype,
     815             :                                         sjinfo,
     816             :                                         use_extended_stats);
     817             :     }
     818      867526 :     else if (is_orclause(clause))
     819             :     {
     820             :         /*
     821             :          * Almost the same thing as clauselist_selectivity, but with the
     822             :          * clauses connected by OR.
     823             :          */
     824       12564 :         s1 = clauselist_selectivity_or(root,
     825             :                                        ((BoolExpr *) clause)->args,
     826             :                                        varRelid,
     827             :                                        jointype,
     828             :                                        sjinfo,
     829             :                                        use_extended_stats);
     830             :     }
     831      854962 :     else if (is_opclause(clause) || IsA(clause, DistinctExpr))
     832      812736 :     {
     833      812760 :         OpExpr     *opclause = (OpExpr *) clause;
     834      812760 :         Oid         opno = opclause->opno;
     835             : 
     836      812760 :         if (treat_as_join_clause(root, clause, rinfo, varRelid, sjinfo))
     837             :         {
     838             :             /* Estimate selectivity for a join clause. */
     839      200124 :             s1 = join_selectivity(root, opno,
     840             :                                   opclause->args,
     841             :                                   opclause->inputcollid,
     842             :                                   jointype,
     843             :                                   sjinfo);
     844             :         }
     845             :         else
     846             :         {
     847             :             /* Estimate selectivity for a restriction clause. */
     848      612636 :             s1 = restriction_selectivity(root, opno,
     849             :                                          opclause->args,
     850             :                                          opclause->inputcollid,
     851             :                                          varRelid);
     852             :         }
     853             : 
     854             :         /*
     855             :          * DistinctExpr has the same representation as OpExpr, but the
     856             :          * contained operator is "=" not "<>", so we must negate the result.
     857             :          * This estimation method doesn't give the right behavior for nulls,
     858             :          * but it's better than doing nothing.
     859             :          */
     860      812736 :         if (IsA(clause, DistinctExpr))
     861         788 :             s1 = 1.0 - s1;
     862             :     }
     863       42202 :     else if (is_funcclause(clause))
     864             :     {
     865       10916 :         FuncExpr   *funcclause = (FuncExpr *) clause;
     866             : 
     867             :         /* Try to get an estimate from the support function, if any */
     868       10916 :         s1 = function_selectivity(root,
     869             :                                   funcclause->funcid,
     870             :                                   funcclause->args,
     871             :                                   funcclause->inputcollid,
     872       10916 :                                   treat_as_join_clause(root, clause, rinfo,
     873             :                                                        varRelid, sjinfo),
     874             :                                   varRelid,
     875             :                                   jointype,
     876             :                                   sjinfo);
     877             :     }
     878       31286 :     else if (IsA(clause, ScalarArrayOpExpr))
     879             :     {
     880             :         /* Use node specific selectivity calculation function */
     881       18240 :         s1 = scalararraysel(root,
     882             :                             (ScalarArrayOpExpr *) clause,
     883       18240 :                             treat_as_join_clause(root, clause, rinfo,
     884             :                                                  varRelid, sjinfo),
     885             :                             varRelid,
     886             :                             jointype,
     887             :                             sjinfo);
     888             :     }
     889       13046 :     else if (IsA(clause, RowCompareExpr))
     890             :     {
     891             :         /* Use node specific selectivity calculation function */
     892         216 :         s1 = rowcomparesel(root,
     893             :                            (RowCompareExpr *) clause,
     894             :                            varRelid,
     895             :                            jointype,
     896             :                            sjinfo);
     897             :     }
     898       12830 :     else if (IsA(clause, NullTest))
     899             :     {
     900             :         /* Use node specific selectivity calculation function */
     901        9348 :         s1 = nulltestsel(root,
     902             :                          ((NullTest *) clause)->nulltesttype,
     903        9348 :                          (Node *) ((NullTest *) clause)->arg,
     904             :                          varRelid,
     905             :                          jointype,
     906             :                          sjinfo);
     907             :     }
     908        3482 :     else if (IsA(clause, BooleanTest))
     909             :     {
     910             :         /* Use node specific selectivity calculation function */
     911         838 :         s1 = booltestsel(root,
     912             :                          ((BooleanTest *) clause)->booltesttype,
     913         838 :                          (Node *) ((BooleanTest *) clause)->arg,
     914             :                          varRelid,
     915             :                          jointype,
     916             :                          sjinfo);
     917             :     }
     918        2644 :     else if (IsA(clause, CurrentOfExpr))
     919             :     {
     920             :         /* CURRENT OF selects at most one row of its table */
     921         406 :         CurrentOfExpr *cexpr = (CurrentOfExpr *) clause;
     922         406 :         RelOptInfo *crel = find_base_rel(root, cexpr->cvarno);
     923             : 
     924         406 :         if (crel->tuples > 0)
     925         392 :             s1 = 1.0 / crel->tuples;
     926             :     }
     927        2238 :     else if (IsA(clause, RelabelType))
     928             :     {
     929             :         /* Not sure this case is needed, but it can't hurt */
     930           0 :         s1 = clause_selectivity_ext(root,
     931           0 :                                     (Node *) ((RelabelType *) clause)->arg,
     932             :                                     varRelid,
     933             :                                     jointype,
     934             :                                     sjinfo,
     935             :                                     use_extended_stats);
     936             :     }
     937        2238 :     else if (IsA(clause, CoerceToDomain))
     938             :     {
     939             :         /* Not sure this case is needed, but it can't hurt */
     940           0 :         s1 = clause_selectivity_ext(root,
     941           0 :                                     (Node *) ((CoerceToDomain *) clause)->arg,
     942             :                                     varRelid,
     943             :                                     jointype,
     944             :                                     sjinfo,
     945             :                                     use_extended_stats);
     946             :     }
     947             :     else
     948             :     {
     949             :         /*
     950             :          * For anything else, see if we can consider it as a boolean variable.
     951             :          * This only works if it's an immutable expression in Vars of a single
     952             :          * relation; but there's no point in us checking that here because
     953             :          * boolvarsel() will do it internally, and return a suitable default
     954             :          * selectivity if not.
     955             :          */
     956        2238 :         s1 = boolvarsel(root, clause, varRelid);
     957             :     }
     958             : 
     959             :     /* Cache the result if possible */
     960      922168 :     if (cacheable)
     961             :     {
     962      601220 :         if (jointype == JOIN_INNER)
     963      491962 :             rinfo->norm_selec = s1;
     964             :         else
     965      109258 :             rinfo->outer_selec = s1;
     966             :     }
     967             : 
     968             : #ifdef SELECTIVITY_DEBUG
     969             :     elog(DEBUG4, "clause_selectivity: s1 %f", s1);
     970             : #endif                          /* SELECTIVITY_DEBUG */
     971             : 
     972      922168 :     return s1;
     973             : }

Generated by: LCOV version 1.14