]> git.ipfire.org Git - thirdparty/postgresql.git/commitdiff
Fix buggy interaction between array subscripts and subplan params
authorAndres Freund <andres@anarazel.de>
Wed, 7 Jan 2026 00:51:10 +0000 (19:51 -0500)
committerAndres Freund <andres@anarazel.de>
Wed, 7 Jan 2026 00:51:10 +0000 (19:51 -0500)
In a7f107df2 I changed subplan param evaluation to happen within the
containing expression. As part of that, ExecInitSubPlanExpr() was changed to
evaluate parameters via a new EEOP_PARAM_SET expression step. These parameters
were temporarily stored into ExprState->resvalue/resnull, with some reasoning
why that would be fine. Unfortunately, that analysis was wrong -
ExecInitSubscriptionRef() evaluates the input array into "resv"/"resnull",
which will often point to ExprState->resvalue/resnull. This means that the
EEOP_PARAM_SET, if inside an array subscript, would overwrite the input array
to array subscript.

The fix is fairly simple - instead of evaluating into
ExprState->resvalue/resnull, store the temporary result of the subplan in the
subplan's return value.

Bug: #19370
Reported-by: Zepeng Zhang <redraiment@gmail.com>
Diagnosed-by: Tom Lane <tgl@sss.pgh.pa.us>
Diagnosed-by: Andres Freund <andres@anarazel.de>
Discussion: https://postgr.es/m/19370-7fb7a5854b7618f1@postgresql.org
Backpatch-through: 18

src/backend/executor/execExpr.c
src/backend/executor/execExprInterp.c
src/test/regress/expected/subselect.out
src/test/regress/sql/subselect.sql

index e0a1fb76aa85ab176cc34f853deebb1f7cf10fa3..088eca24021dd0e70c8676fdecbcbe6701adc407 100644 (file)
@@ -2826,12 +2826,13 @@ ExecInitSubPlanExpr(SubPlan *subplan,
        /*
         * Generate steps to evaluate input arguments for the subplan.
         *
-        * We evaluate the argument expressions into ExprState's resvalue/resnull,
-        * and then use PARAM_SET to update the parameter. We do that, instead of
-        * evaluating directly into the param, to avoid depending on the pointer
-        * value remaining stable / being included in the generated expression. No
-        * danger of conflicts with other uses of resvalue/resnull as storing and
-        * using the value always is in subsequent steps.
+        * We evaluate the argument expressions into resv/resnull, and then use
+        * PARAM_SET to update the parameter. We do that, instead of evaluating
+        * directly into the param, to avoid depending on the pointer value
+        * remaining stable / being included in the generated expression. It's ok
+        * to use resv/resnull for multiple params, as each parameter evaluation
+        * is immediately followed by an EEOP_PARAM_SET (and thus are saved before
+        * they could be overwritten again).
         *
         * Any calculation we have to do can be done in the parent econtext, since
         * the Param values don't need to have per-query lifetime.
@@ -2842,10 +2843,11 @@ ExecInitSubPlanExpr(SubPlan *subplan,
                int                     paramid = lfirst_int(l);
                Expr       *arg = (Expr *) lfirst(pvar);
 
-               ExecInitExprRec(arg, state,
-                                               &state->resvalue, &state->resnull);
+               ExecInitExprRec(arg, state, resv, resnull);
 
                scratch.opcode = EEOP_PARAM_SET;
+               scratch.resvalue = resv;
+               scratch.resnull = resnull;
                scratch.d.param.paramid = paramid;
                /* paramtype's not actually used, but we might as well fill it */
                scratch.d.param.paramtype = exprType((Node *) arg);
index 86ab3704b66e0c1e672623b34884c034d8de604a..a7a5ac1e83b0e878c61c4576bde42a816aa315c7 100644 (file)
@@ -3107,7 +3107,7 @@ ExecEvalParamExtern(ExprState *state, ExprEvalStep *op, ExprContext *econtext)
 
 /*
  * Set value of a param (currently always PARAM_EXEC) from
- * state->res{value,null}.
+ * op->res{value,null}.
  */
 void
 ExecEvalParamSet(ExprState *state, ExprEvalStep *op, ExprContext *econtext)
@@ -3119,8 +3119,8 @@ ExecEvalParamSet(ExprState *state, ExprEvalStep *op, ExprContext *econtext)
        /* Shouldn't have a pending evaluation anymore */
        Assert(prm->execPlan == NULL);
 
-       prm->value = state->resvalue;
-       prm->isnull = state->resnull;
+       prm->value = *op->resvalue;
+       prm->isnull = *op->resnull;
 }
 
 /*
index 774c22b6e3a795b0af50698cc581033dd44f295b..37ed59e73bf37d69221c11772a4abfa8254135db 100644 (file)
@@ -676,6 +676,23 @@ select * from (
    0
 (1 row)
 
+--
+-- Test cases for interactions between PARAM_EXEC, subplans and array
+-- subscripts
+--
+-- check that array subscription doesn't conflict with PARAM_EXEC (see #19370)
+SELECT (array[1,2])[(SELECT g.i)] FROM generate_series(1, 1) g(i);
+ array 
+-------
+     1
+(1 row)
+
+SELECT (array[1,2])[(SELECT g.i):(SELECT g.i + 1)] FROM generate_series(1, 1) g(i);
+ array 
+-------
+ {1,2}
+(1 row)
+
 --
 -- Test that an IN implemented using a UniquePath does unique-ification
 -- with the right semantics, as per bug #4113.  (Unfortunately we have
index 7b4ebff46d8ce93d2dd52591e265b15f827c3006..192a6f96b936ccd85fa24807675d4d00159ee57b 100644 (file)
@@ -340,6 +340,17 @@ select * from (
   where not exists (select 1 from tenk1 as b where b.unique2 = 10000)
 ) ss;
 
+
+--
+-- Test cases for interactions between PARAM_EXEC, subplans and array
+-- subscripts
+--
+
+-- check that array subscription doesn't conflict with PARAM_EXEC (see #19370)
+SELECT (array[1,2])[(SELECT g.i)] FROM generate_series(1, 1) g(i);
+SELECT (array[1,2])[(SELECT g.i):(SELECT g.i + 1)] FROM generate_series(1, 1) g(i);
+
+
 --
 -- Test that an IN implemented using a UniquePath does unique-ification
 -- with the right semantics, as per bug #4113.  (Unfortunately we have