]> granicus.if.org Git - postgresql/commitdiff
Don't generate parallel paths for rels with parallel-restricted outputs.
authorRobert Haas <rhaas@postgresql.org>
Thu, 9 Jun 2016 16:40:23 +0000 (12:40 -0400)
committerRobert Haas <rhaas@postgresql.org>
Thu, 9 Jun 2016 16:43:36 +0000 (12:43 -0400)
Such paths are unsafe.  To make it cheaper to detect when this case
applies, track whether a relation's default PathTarget contains any
non-Vars.  In most cases, the answer will be no, which enables us to
determine cheaply that the target list for a proposed path is
parallel-safe.  However, subquery pull-up can create cases that
require us to inspect the target list more carefully.

Amit Kapila, reviewed by me.

src/backend/nodes/outfuncs.c
src/backend/optimizer/path/allpaths.c
src/backend/optimizer/util/placeholder.c
src/backend/optimizer/util/relnode.c
src/include/nodes/relation.h

index c7b4153c0308becb4364c896b6c648d91e567722..2796e5353aea1add2da2eb238a4c8ae8a30830cb 100644 (file)
@@ -2083,6 +2083,7 @@ _outRelOptInfo(StringInfo str, const RelOptInfo *node)
        WRITE_BOOL_FIELD(consider_param_startup);
        WRITE_BOOL_FIELD(consider_parallel);
        WRITE_NODE_FIELD(reltarget);
+       WRITE_BOOL_FIELD(reltarget_has_non_vars);
        WRITE_NODE_FIELD(pathlist);
        WRITE_NODE_FIELD(ppilist);
        WRITE_NODE_FIELD(partial_pathlist);
index cf6a9a9f82ace946ed50b2fe791bb037f7a90a32..6deb2cf0c9c689b03cb2cee431720bc989b6d59d 100644 (file)
@@ -608,6 +608,15 @@ set_rel_consider_parallel(PlannerInfo *root, RelOptInfo *rel,
        if (has_parallel_hazard((Node *) rel->baserestrictinfo, false))
                return;
 
+       /*
+        * If the relation's outputs are not parallel-safe, we must give up.
+        * In the common case where the relation only outputs Vars, this check is
+        * very cheap; otherwise, we have to do more work.
+        */
+       if (rel->reltarget_has_non_vars &&
+               has_parallel_hazard((Node *) rel->reltarget->exprs, false))
+               return;
+
        /* We have a winner. */
        rel->consider_parallel = true;
 }
@@ -971,6 +980,7 @@ set_append_rel_size(PlannerInfo *root, RelOptInfo *rel,
                        adjust_appendrel_attrs(root,
                                                                   (Node *) rel->reltarget->exprs,
                                                                   appinfo);
+               childrel->reltarget_has_non_vars = rel->reltarget_has_non_vars;
 
                /*
                 * We have to make child entries in the EquivalenceClass data
index b210914b85391c1397762cb0f795a7c1f055c216..5b85a4ddadcd14991b2550697fe8837bb6c83412 100644 (file)
@@ -393,6 +393,7 @@ add_placeholders_to_base_rels(PlannerInfo *root)
 
                        rel->reltarget->exprs = lappend(rel->reltarget->exprs,
                                                                                        copyObject(phinfo->ph_var));
+                       rel->reltarget_has_non_vars = true;
                        /* reltarget's cost and width fields will be updated later */
                }
        }
@@ -427,6 +428,7 @@ add_placeholders_to_joinrel(PlannerInfo *root, RelOptInfo *joinrel,
                                /* Yup, add it to the output */
                                joinrel->reltarget->exprs = lappend(joinrel->reltarget->exprs,
                                                                                                        phinfo->ph_var);
+                               joinrel->reltarget_has_non_vars = true;
                                joinrel->reltarget->width += phinfo->ph_width;
 
                                /*
index e2ebf4792962770d9408a6414552d1757687deab..2def06dd922303362522a75bedb14546c87d5131 100644 (file)
@@ -109,6 +109,7 @@ build_simple_rel(PlannerInfo *root, int relid, RelOptKind reloptkind)
        rel->consider_parallel = false;         /* might get changed later */
        rel->rel_parallel_workers = -1; /* set up in GetRelationInfo */
        rel->reltarget = create_empty_pathtarget();
+       rel->reltarget_has_non_vars = false;
        rel->pathlist = NIL;
        rel->ppilist = NIL;
        rel->partial_pathlist = NIL;
@@ -396,6 +397,7 @@ build_join_rel(PlannerInfo *root,
        joinrel->consider_param_startup = false;
        joinrel->consider_parallel = false;
        joinrel->reltarget = create_empty_pathtarget();
+       joinrel->reltarget_has_non_vars = false;
        joinrel->pathlist = NIL;
        joinrel->ppilist = NIL;
        joinrel->partial_pathlist = NIL;
@@ -506,8 +508,8 @@ build_join_rel(PlannerInfo *root,
         * Set the consider_parallel flag if this joinrel could potentially be
         * scanned within a parallel worker.  If this flag is false for either
         * inner_rel or outer_rel, then it must be false for the joinrel also.
-        * Even if both are true, there might be parallel-restricted quals at our
-        * level.
+        * Even if both are true, there might be parallel-restricted expressions
+        * in the targetlist or quals.
         *
         * Note that if there are more than two rels in this relation, they could
         * be divided between inner_rel and outer_rel in any arbitrary way.  We
@@ -517,7 +519,9 @@ build_join_rel(PlannerInfo *root,
         * here.
         */
        if (inner_rel->consider_parallel && outer_rel->consider_parallel &&
-               !has_parallel_hazard((Node *) restrictlist, false))
+               !has_parallel_hazard((Node *) restrictlist, false) &&
+               !(joinrel->reltarget_has_non_vars &&
+                 has_parallel_hazard((Node *) joinrel->reltarget->exprs, false)))
                joinrel->consider_parallel = true;
 
        /*
index 88669f38765d9474a05dcad3a9a9fa77dc6969f1..b2b204693f7b9396946b234cd9c5bdf6935c8245 100644 (file)
@@ -490,6 +490,8 @@ typedef struct RelOptInfo
 
        /* default result targetlist for Paths scanning this relation */
        struct PathTarget *reltarget;           /* list of Vars/Exprs, cost, width */
+       bool            reltarget_has_non_vars; /* true if any expression in
+                                                                                * PathTarget is a non-Var */
 
        /* materialization information */
        List       *pathlist;           /* Path structures */