mirror of
https://git.postgresql.org/git/postgresql.git
synced 2025-01-06 15:24:56 +08:00
Remove bogus Assert and dead code in remove_useless_results_recurse().
The JOIN_SEMI case Assert'ed that there are no PlaceHolderVars that need to be evaluated at the semijoin's RHS, which is wrong because there could be some in the semijoin's qual condition. However, there could not be any references further up than that, and within the qual there is not any way that such a PHV could have gone to null yet, so we don't really need the PHV and there is no need to avoid making the RHS-removal optimization. The upshot is that there's no actual bug in production code, and we ought to just remove this misguided Assert. While we're here, also drop the JOIN_RIGHT case, which is dead code because reduce_outer_joins() already got rid of JOIN_RIGHT. Per bug #17700 from Xin Wen. Uselessness of the JOIN_RIGHT case pointed out by Richard Guo. Back-patch to v12 where this code was added. Discussion: https://postgr.es/m/17700-2b5c10d917c30687@postgresql.org
This commit is contained in:
parent
ad86d159b6
commit
51dfaa0b01
@ -3251,16 +3251,6 @@ remove_useless_results_recurse(PlannerInfo *root, Node *jtnode)
|
|||||||
jtnode = j->larg;
|
jtnode = j->larg;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case JOIN_RIGHT:
|
|
||||||
/* Mirror-image of the JOIN_LEFT case */
|
|
||||||
if ((varno = get_result_relid(root, j->larg)) != 0 &&
|
|
||||||
(j->quals == NULL ||
|
|
||||||
!find_dependent_phvs(root, varno)))
|
|
||||||
{
|
|
||||||
remove_result_refs(root, varno, j->rarg);
|
|
||||||
jtnode = j->rarg;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case JOIN_SEMI:
|
case JOIN_SEMI:
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -3269,14 +3259,17 @@ remove_useless_results_recurse(PlannerInfo *root, Node *jtnode)
|
|||||||
* LHS, since we should either return the LHS row or not. For
|
* LHS, since we should either return the LHS row or not. For
|
||||||
* simplicity we inject the filter qual into a new FromExpr.
|
* simplicity we inject the filter qual into a new FromExpr.
|
||||||
*
|
*
|
||||||
* Unlike the LEFT/RIGHT cases, we just Assert that there are
|
* There is a fine point about PHVs that are supposed to be
|
||||||
* no PHVs that need to be evaluated at the semijoin's RHS,
|
* evaluated at the RHS. Such PHVs could only appear in the
|
||||||
* since the rest of the query couldn't reference any outputs
|
* semijoin's qual, since the rest of the query cannot
|
||||||
* of the semijoin's RHS.
|
* reference any outputs of the semijoin's RHS. Therefore,
|
||||||
|
* they can't actually go to null before being examined, and
|
||||||
|
* it'd be OK to just remove the PHV wrapping. We don't have
|
||||||
|
* infrastructure for that, but remove_result_refs() will
|
||||||
|
* relabel them as to be evaluated at the LHS, which is fine.
|
||||||
*/
|
*/
|
||||||
if ((varno = get_result_relid(root, j->rarg)) != 0)
|
if ((varno = get_result_relid(root, j->rarg)) != 0)
|
||||||
{
|
{
|
||||||
Assert(!find_dependent_phvs(root, varno));
|
|
||||||
remove_result_refs(root, varno, j->larg);
|
remove_result_refs(root, varno, j->larg);
|
||||||
if (j->quals)
|
if (j->quals)
|
||||||
jtnode = (Node *)
|
jtnode = (Node *)
|
||||||
@ -3290,6 +3283,7 @@ remove_useless_results_recurse(PlannerInfo *root, Node *jtnode)
|
|||||||
/* We have no special smarts for these cases */
|
/* We have no special smarts for these cases */
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
/* Note: JOIN_RIGHT should be gone at this point */
|
||||||
elog(ERROR, "unrecognized join type: %d",
|
elog(ERROR, "unrecognized join type: %d",
|
||||||
(int) j->jointype);
|
(int) j->jointype);
|
||||||
break;
|
break;
|
||||||
|
@ -3550,6 +3550,26 @@ where b;
|
|||||||
0 | t | t
|
0 | t | t
|
||||||
(2 rows)
|
(2 rows)
|
||||||
|
|
||||||
|
-- Test PHV in a semijoin qual, which confused useless-RTE removal (bug #17700)
|
||||||
|
explain (verbose, costs off)
|
||||||
|
with ctetable as not materialized ( select 1 as f1 )
|
||||||
|
select * from ctetable c1
|
||||||
|
where f1 in ( select c3.f1 from ctetable c2 full join ctetable c3 on true );
|
||||||
|
QUERY PLAN
|
||||||
|
----------------------------
|
||||||
|
Result
|
||||||
|
Output: 1
|
||||||
|
One-Time Filter: (1 = 1)
|
||||||
|
(3 rows)
|
||||||
|
|
||||||
|
with ctetable as not materialized ( select 1 as f1 )
|
||||||
|
select * from ctetable c1
|
||||||
|
where f1 in ( select c3.f1 from ctetable c2 full join ctetable c3 on true );
|
||||||
|
f1
|
||||||
|
----
|
||||||
|
1
|
||||||
|
(1 row)
|
||||||
|
|
||||||
--
|
--
|
||||||
-- test inlining of immutable functions
|
-- test inlining of immutable functions
|
||||||
--
|
--
|
||||||
|
@ -1156,6 +1156,16 @@ select * from
|
|||||||
select a as b) as t3
|
select a as b) as t3
|
||||||
where b;
|
where b;
|
||||||
|
|
||||||
|
-- Test PHV in a semijoin qual, which confused useless-RTE removal (bug #17700)
|
||||||
|
explain (verbose, costs off)
|
||||||
|
with ctetable as not materialized ( select 1 as f1 )
|
||||||
|
select * from ctetable c1
|
||||||
|
where f1 in ( select c3.f1 from ctetable c2 full join ctetable c3 on true );
|
||||||
|
|
||||||
|
with ctetable as not materialized ( select 1 as f1 )
|
||||||
|
select * from ctetable c1
|
||||||
|
where f1 in ( select c3.f1 from ctetable c2 full join ctetable c3 on true );
|
||||||
|
|
||||||
--
|
--
|
||||||
-- test inlining of immutable functions
|
-- test inlining of immutable functions
|
||||||
--
|
--
|
||||||
|
Loading…
Reference in New Issue
Block a user