aboutsummaryrefslogtreecommitdiff
path: root/src/backend/executor
diff options
context:
space:
mode:
authorTom Lane <tgl@sss.pgh.pa.us>2014-01-08 20:18:06 -0500
committerTom Lane <tgl@sss.pgh.pa.us>2014-01-08 20:18:58 -0500
commit080b7db72ebbec22580237631d6b07d0e1147b01 (patch)
tree756ffb242ae077c408e04889810d1b0b0adbcdfe /src/backend/executor
parentdaa7527afc2274432094ebe7ceb03aa41f916607 (diff)
downloadpostgresql-080b7db72ebbec22580237631d6b07d0e1147b01.tar.gz
postgresql-080b7db72ebbec22580237631d6b07d0e1147b01.zip
Fix "cannot accept a set" error when only some arms of a CASE return a set.
In commit c1352052ef1d4eeb2eb1d822a207ddc2d106cb13, I implemented an optimization that assumed that a function's argument expressions would either always return a set (ie multiple rows), or always not. This is wrong however: we allow CASE expressions in which some arms return a set of some type and others just return a scalar of that type. There may be other examples as well. To fix, replace the run-time test of whether an argument returned a set with a static precheck (expression_returns_set). This adds a little bit of query startup overhead, but it seems barely measurable. Per bug #8228 from David Johnston. This has been broken since 8.0, so patch all supported branches.
Diffstat (limited to 'src/backend/executor')
-rw-r--r--src/backend/executor/execQual.c55
1 files changed, 38 insertions, 17 deletions
diff --git a/src/backend/executor/execQual.c b/src/backend/executor/execQual.c
index 5c14b4aa2d3..0eba025ef9a 100644
--- a/src/backend/executor/execQual.c
+++ b/src/backend/executor/execQual.c
@@ -1634,9 +1634,7 @@ tupledesc_match(TupleDesc dst_tupdesc, TupleDesc src_tupdesc)
* init_fcache is presumed already run on the FuncExprState.
*
* This function handles the most general case, wherein the function or
- * one of its arguments might (or might not) return a set. If we find
- * no sets involved, we will change the FuncExprState's function pointer
- * to use a simpler method on subsequent calls.
+ * one of its arguments can return a set.
*/
static Datum
ExecMakeFunctionResult(FuncExprState *fcache,
@@ -1906,13 +1904,12 @@ restart:
/*
* Non-set case: much easier.
*
- * We change the ExprState function pointer to use the simpler
- * ExecMakeFunctionResultNoSets on subsequent calls. This amounts to
- * assuming that no argument can return a set if it didn't do so the
- * first time.
+ * In common cases, this code path is unreachable because we'd have
+ * selected ExecMakeFunctionResultNoSets instead. However, it's
+ * possible to get here if an argument sometimes produces set results
+ * and sometimes scalar results. For example, a CASE expression might
+ * call a set-returning function in only some of its arms.
*/
- fcache->xprstate.evalfunc = (ExprStateEvalFunc) ExecMakeFunctionResultNoSets;
-
if (isDone)
*isDone = ExprSingleResult;
@@ -2371,10 +2368,22 @@ ExecEvalFunc(FuncExprState *fcache,
init_fcache(func->funcid, func->inputcollid, fcache,
econtext->ecxt_per_query_memory, true);
- /* Go directly to ExecMakeFunctionResult on subsequent uses */
- fcache->xprstate.evalfunc = (ExprStateEvalFunc) ExecMakeFunctionResult;
-
- return ExecMakeFunctionResult(fcache, econtext, isNull, isDone);
+ /*
+ * We need to invoke ExecMakeFunctionResult if either the function itself
+ * or any of its input expressions can return a set. Otherwise, invoke
+ * ExecMakeFunctionResultNoSets. In either case, change the evalfunc
+ * pointer to go directly there on subsequent uses.
+ */
+ if (fcache->func.fn_retset || expression_returns_set((Node *) func->args))
+ {
+ fcache->xprstate.evalfunc = (ExprStateEvalFunc) ExecMakeFunctionResult;
+ return ExecMakeFunctionResult(fcache, econtext, isNull, isDone);
+ }
+ else
+ {
+ fcache->xprstate.evalfunc = (ExprStateEvalFunc) ExecMakeFunctionResultNoSets;
+ return ExecMakeFunctionResultNoSets(fcache, econtext, isNull, isDone);
+ }
}
/* ----------------------------------------------------------------
@@ -2394,10 +2403,22 @@ ExecEvalOper(FuncExprState *fcache,
init_fcache(op->opfuncid, op->inputcollid, fcache,
econtext->ecxt_per_query_memory, true);
- /* Go directly to ExecMakeFunctionResult on subsequent uses */
- fcache->xprstate.evalfunc = (ExprStateEvalFunc) ExecMakeFunctionResult;
-
- return ExecMakeFunctionResult(fcache, econtext, isNull, isDone);
+ /*
+ * We need to invoke ExecMakeFunctionResult if either the function itself
+ * or any of its input expressions can return a set. Otherwise, invoke
+ * ExecMakeFunctionResultNoSets. In either case, change the evalfunc
+ * pointer to go directly there on subsequent uses.
+ */
+ if (fcache->func.fn_retset || expression_returns_set((Node *) op->args))
+ {
+ fcache->xprstate.evalfunc = (ExprStateEvalFunc) ExecMakeFunctionResult;
+ return ExecMakeFunctionResult(fcache, econtext, isNull, isDone);
+ }
+ else
+ {
+ fcache->xprstate.evalfunc = (ExprStateEvalFunc) ExecMakeFunctionResultNoSets;
+ return ExecMakeFunctionResultNoSets(fcache, econtext, isNull, isDone);
+ }
}
/* ----------------------------------------------------------------