Commit 1b19e2c0 authored by Tom Lane's avatar Tom Lane

Fix up handling of simple-form CASE with constant test expression.

eval_const_expressions() can replace CaseTestExprs with constants when
the surrounding CASE's test expression is a constant.  This confuses
ruleutils.c's heuristic for deparsing simple-form CASEs, leading to
Assert failures or "unexpected CASE WHEN clause" errors.  I had put in
a hack solution for that years ago (see commit
514ce7a3 of 2006-10-01), but bug #5794
from Peter Speck shows that that solution failed to cover all cases.

Fortunately, there's a much better way, which came to me upon reflecting
that Peter's "CASE TRUE WHEN" seemed pretty redundant: we can "simplify"
the simple-form CASE to the general form of CASE, by simply omitting the
constant test expression from the rebuilt CASE construct.  This is
intuitively valid because there is no need for the executor to evaluate
the test expression at runtime; it will never be referenced, because any
CaseTestExprs that would have referenced it are now replaced by constants.
This won't save a whole lot of cycles, since evaluating a Const is pretty
cheap, but a cycle saved is a cycle earned.  In any case it beats kluging
ruleutils.c still further.  So this patch improves const-simplification
and reverts the previous change in ruleutils.c.

Back-patch to all supported branches.  The bug exists in 8.1 too, but it's
out of warranty.
parent abc10262
...@@ -2583,7 +2583,18 @@ eval_const_expressions_mutator(Node *node, ...@@ -2583,7 +2583,18 @@ eval_const_expressions_mutator(Node *node,
* placeholder nodes, so that we have the opportunity to reduce * placeholder nodes, so that we have the opportunity to reduce
* constant test conditions. For example this allows * constant test conditions. For example this allows
* CASE 0 WHEN 0 THEN 1 ELSE 1/0 END * CASE 0 WHEN 0 THEN 1 ELSE 1/0 END
* to reduce to 1 rather than drawing a divide-by-0 error. * to reduce to 1 rather than drawing a divide-by-0 error. Note
* that when the test expression is constant, we don't have to
* include it in the resulting CASE; for example
* CASE 0 WHEN x THEN y ELSE z END
* is transformed by the parser to
* CASE 0 WHEN CaseTestExpr = x THEN y ELSE z END
* which we can simplify to
* CASE WHEN 0 = x THEN y ELSE z END
* It is not necessary for the executor to evaluate the "arg"
* expression when executing the CASE, since any contained
* CaseTestExprs that might have referred to it will have been
* replaced by the constant.
*---------- *----------
*/ */
CaseExpr *caseexpr = (CaseExpr *) node; CaseExpr *caseexpr = (CaseExpr *) node;
...@@ -2602,7 +2613,10 @@ eval_const_expressions_mutator(Node *node, ...@@ -2602,7 +2613,10 @@ eval_const_expressions_mutator(Node *node,
/* Set up for contained CaseTestExpr nodes */ /* Set up for contained CaseTestExpr nodes */
save_case_val = context->case_val; save_case_val = context->case_val;
if (newarg && IsA(newarg, Const)) if (newarg && IsA(newarg, Const))
{
context->case_val = newarg; context->case_val = newarg;
newarg = NULL; /* not needed anymore, see comment above */
}
else else
context->case_val = NULL; context->case_val = NULL;
......
...@@ -5146,23 +5146,19 @@ get_rule_expr(Node *node, deparse_context *context, ...@@ -5146,23 +5146,19 @@ get_rule_expr(Node *node, deparse_context *context,
* boolexpr WHEN TRUE THEN ...", then the optimizer's * boolexpr WHEN TRUE THEN ...", then the optimizer's
* simplify_boolean_equality() may have reduced this * simplify_boolean_equality() may have reduced this
* to just "CaseTestExpr" or "NOT CaseTestExpr", for * to just "CaseTestExpr" or "NOT CaseTestExpr", for
* which we have to show "TRUE" or "FALSE". Also, * which we have to show "TRUE" or "FALSE". We have
* depending on context the original CaseTestExpr * also to consider the possibility that an implicit
* might have been reduced to a Const (but we won't * coercion was inserted between the CaseTestExpr and
* see "WHEN Const"). We have also to consider the * the operator.
* possibility that an implicit coercion was inserted
* between the CaseTestExpr and the operator.
*/ */
if (IsA(w, OpExpr)) if (IsA(w, OpExpr))
{ {
List *args = ((OpExpr *) w)->args; List *args = ((OpExpr *) w)->args;
Node *lhs;
Node *rhs; Node *rhs;
Assert(list_length(args) == 2); Assert(list_length(args) == 2);
lhs = strip_implicit_coercions(linitial(args)); Assert(IsA(strip_implicit_coercions(linitial(args)),
Assert(IsA(lhs, CaseTestExpr) || CaseTestExpr));
IsA(lhs, Const));
rhs = (Node *) lsecond(args); rhs = (Node *) lsecond(args);
get_rule_expr(rhs, context, false); get_rule_expr(rhs, context, false);
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment