Commit 6af9ee4c authored by Tom Lane's avatar Tom Lane

Make real sure we don't reassociate joins into or out of SEMI/ANTI joins.

Per the discussion in optimizer/README, it's unsafe to reassociate anything
into or out of the RHS of a SEMI or ANTI join.  An example from Piotr
Stefaniak showed that join_is_legal() wasn't sufficiently enforcing this
rule, so lock it down a little harder.

I couldn't find a reasonably simple example of the optimizer trying to
do this, so no new regression test.  (Piotr's example involved the random
search in GEQO accidentally trying an invalid case and triggering a sanity
check way downstream in clause selectivity estimation, which did not seem
like a sequence of events that would be useful to memorialize in a
regression test as-is.)

Back-patch to all active branches.
parent 18382ae7
...@@ -470,10 +470,14 @@ join_is_legal(PlannerInfo *root, RelOptInfo *rel1, RelOptInfo *rel2, ...@@ -470,10 +470,14 @@ join_is_legal(PlannerInfo *root, RelOptInfo *rel1, RelOptInfo *rel2,
/* /*
* Otherwise, the proposed join overlaps the RHS but isn't a valid * Otherwise, the proposed join overlaps the RHS but isn't a valid
* implementation of this SJ. It might still be a legal join, * implementation of this SJ. It might still be a legal join,
* however, if it does not overlap the LHS. * however, if it does not overlap the LHS. But we never allow
* violations of the RHS of SEMI or ANTI joins. (In practice,
* therefore, only LEFT joins ever allow RHS violation.)
*/ */
if (bms_overlap(joinrelids, sjinfo->min_lefthand)) if (sjinfo->jointype == JOIN_SEMI ||
return false; sjinfo->jointype == JOIN_ANTI ||
bms_overlap(joinrelids, sjinfo->min_lefthand))
return false; /* invalid join path */
/*---------- /*----------
* If both inputs overlap the RHS, assume that it's OK. Since the * If both inputs overlap the RHS, assume that it's OK. Since the
...@@ -498,15 +502,14 @@ join_is_legal(PlannerInfo *root, RelOptInfo *rel1, RelOptInfo *rel2, ...@@ -498,15 +502,14 @@ join_is_legal(PlannerInfo *root, RelOptInfo *rel1, RelOptInfo *rel2,
* Set flag here to check at bottom of loop. * Set flag here to check at bottom of loop.
*---------- *----------
*/ */
if (sjinfo->jointype != JOIN_SEMI && if (bms_overlap(rel1->relids, sjinfo->min_righthand) &&
bms_overlap(rel1->relids, sjinfo->min_righthand) &&
bms_overlap(rel2->relids, sjinfo->min_righthand)) bms_overlap(rel2->relids, sjinfo->min_righthand))
{ {
/* both overlap; assume OK */ /* both overlap; assume OK */
} }
else else
{ {
/* one overlaps, the other doesn't (or it's a semijoin) */ /* one overlaps, the other doesn't */
is_valid_inner = false; is_valid_inner = 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