/*
* 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.
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);
/*
* 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)
/* 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;
}
/*
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
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