Re: Parallel safety for extern params - Mailing list pgsql-hackers

From Tom Lane
Subject Re: Parallel safety for extern params
Date
Msg-id 1565347.1742572349@sss.pgh.pa.us
Whole thread Raw
In response to Re: [HACKERS] Parallel safety for extern params  (Robert Haas <robertmhaas@gmail.com>)
Responses Re: Parallel safety for extern params
List pgsql-hackers
Robert Haas <robertmhaas@gmail.com> writes:
> On Sat, Oct 28, 2017 at 8:02 PM, Amit Kapila <amit.kapila16@gmail.com> wrote:
>> I think we need to make changes in exec_simple_recheck_plan to make
>> the behavior similar to HEAD.  With the attached patch, all tests
>> passed with force_parallel_mode.

> Committed to REL_10_STABLE.

Sorry for resurrecting such an old thread, but I just noticed what
commit 682ce911f did in exec_save_simple_expr:

@@ -6588,8 +6588,8 @@ exec_save_simple_expr(PLpgSQL_expr *expr, CachedPlan *cplan)
      * force_parallel_mode is on, the planner might've stuck a Gather node
      * atop that.  The simplest way to deal with this is to look through the
      * Gather node.  The Gather node's tlist would normally contain a Var
-     * referencing the child node's output ... but setrefs.c might also have
-     * copied a Const as-is.
+     * referencing the child node's output, but it could also be a Param, or
+     * it could be a Const that setrefs.c copied as-is.
      */
     plan = stmt->planTree;
     for (;;)
@@ -6616,9 +6616,9 @@ exec_save_simple_expr(PLpgSQL_expr *expr, CachedPlan *cplan)
             /* If setrefs.c copied up a Const, no need to look further */
             if (IsA(tle_expr, Const))
                 break;
-            /* Otherwise, it better be an outer Var */
-            Assert(IsA(tle_expr, Var));
-            Assert(((Var *) tle_expr)->varno == OUTER_VAR);
+            /* Otherwise, it had better be a Param or an outer Var */
+            Assert(IsA(tle_expr, Param) || (IsA(tle_expr, Var) &&
+                    ((Var *) tle_expr)->varno == OUTER_VAR));
             /* Descend to the child node */
             plan = plan->lefttree;
         }

I think this is completely wrong and should be reverted.  There
cannot be a Param there, and if there were it would not represent
a reference to the Gather's child.

The argument for this change seems to be what Amit said upthread:

>>> I am not sure if we can write the comment like that (.. copied a Const
>>> or Param as-is.) because fix_upper_expr_mutator in setrefs.c has a
>>> special handling for Var and Param where constants are copied as-is
>>> via expression_tree_mutator.

but AFAICS that is based on a misreading of what
fix_upper_expr_mutator does:

    /* Special cases (apply only AFTER failing to match to lower tlist) */
    if (IsA(node, Param))
        return fix_param_node(context->root, (Param *) node);

Note the comment.  A Param that matches something in the original
Result node would have been replaced by a Var reference to the
lower Result.  If we find a Param still surviving in the Gather's
tlist, then it is not such a reference, and descending as though
it were is wrong and dangerous.  AFAICS, the only case where we'd
actually find such a Param in a Gather is if it's a reference to
the value of an initplan --- but exec_save_simple_expr has already
asserted that there's no initPlans attached to the Gather.

I tried reverting this code change, and check-world still passes,
with or without debug_parallel_query = regress.  So if there is
a case I'm missing, the regression tests don't expose it.

            regards, tom lane



pgsql-hackers by date:

Previous
From: "David G. Johnston"
Date:
Subject: Re: Disabling vacuum truncate for autovacuum
Next
From: Matheus Alcantara
Date:
Subject: Re: RFC: Additional Directory for Extensions