-
Notifications
You must be signed in to change notification settings - Fork 1.7k
Improve CommonSubexprEliminate
rule with surely and conditionally evaluated stats
#11357
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Improve CommonSubexprEliminate
rule with surely and conditionally evaluated stats
#11357
Conversation
90a2227
to
6ffee44
Compare
cc @alamb, @haohuaijin. This PR fixes #11197 (comment) / #11265 (comment). |
75e15d6
to
54eb229
Compare
Thank you @peter-toth the CI clippy error was fixed in #11368 so if you merge up from main the tests should now pass I will review this PR tomorrow |
54eb229
to
dad557c
Compare
Thank you @alamb, I've rebased the PR on the latest |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks @peter-toth -- this PR is (like always) a joy to read and review.
I left some documentation suggestions, but the only thing I think that is needed prior to merge is some additional negative testing (I left suggestions)
I also reviewed the plan changes carefully and they looked great
15)----------------RepartitionExec: partitioning=RoundRobinBatch(4), input_partitions=1 | ||
16)------------------CsvExec: file_groups={1 group: [[WORKSPACE_ROOT/datafusion/sqllogictest/test_files/tpch/data/part.tbl]]}, projection=[p_partkey, p_type], has_header=false | ||
04)------AggregateExec: mode=Partial, gby=[], aggr=[sum(CASE WHEN part.p_type LIKE Utf8("PROMO%") THEN lineitem.l_extendedprice * Int64(1) - lineitem.l_discount ELSE Int64(0) END), sum(lineitem.l_extendedprice * Int64(1) - lineitem.l_discount)] | ||
05)--------ProjectionExec: expr=[l_extendedprice@0 * (Some(1),20,0 - l_discount@1) as __common_expr_1, p_type@2 as p_type] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
its interesting here that this plan shows the evaluation done below the aggregate but the aggregate doesn't seem to reflect that fact (e.g. the aggr expres don't refer to __common_expr_1
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Wow, this is a good catch. I have no idea why because the logical plan looks good.
I will try to look into this after this PR, might be some kind of logical->physical plan conversion bug?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actually, I'm not sure that the logical plan looks good as lineitem.l_extendedprice
and lineitem.l_discount
disappeared from the optimized plan Projection
.
Let me look into this before merging this PR.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
No sorry, I was wrong. Those 2 appear only in aliases in Aggregate
so the Projection
below the Aggregate
in the optimized logical plan seems correct.
01)Projection: t.y > Int32(0) AND Int64(1) / CAST(t.y AS Int64) < Int64(1) AS t.y > Int64(0) AND Int64(1) / t.y < Int64(1), t.x > Int32(0) AND t.y > Int32(0) AND Int64(1) / CAST(t.y AS Int64) < Int64(1) / CAST(t.x AS Int64) AS t.x > Int64(0) AND t.y > Int64(0) AND Int64(1) / t.y < Int64(1) / t.x | ||
02)--TableScan: t projection=[x, y] | ||
01)Projection: __common_expr_1 AND Int64(1) / CAST(t.y AS Int64) < Int64(1) AS t.y > Int64(0) AND Int64(1) / t.y < Int64(1), t.x > Int32(0) AND __common_expr_1 AND Int64(1) / CAST(t.y AS Int64) < Int64(1) / CAST(t.x AS Int64) AS t.x > Int64(0) AND t.y > Int64(0) AND Int64(1) / t.y < Int64(1) / t.x | ||
02)--Projection: t.y > Int32(0) AS __common_expr_1, t.x, t.y |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
👍 I verified that the common expressions do not include the 1 / y
term which can potentially generate a runtime error
FROM t1 | ||
---- | ||
logical_plan | ||
01)Projection: (__common_expr_1 OR random() = Float64(0)) AND __common_expr_1 AS c1, __common_expr_2 AND random() = Float64(0) OR __common_expr_2 AS c2, CASE WHEN __common_expr_3 = Float64(0) THEN __common_expr_3 ELSE Float64(0) END AS c3, CASE WHEN __common_expr_4 = Float64(0) THEN Int64(0) WHEN CAST(__common_expr_4 AS Boolean) THEN Int64(0) ELSE Int64(0) END AS c4, CASE WHEN __common_expr_5 = Float64(0) THEN Float64(0) WHEN random() = Float64(0) THEN __common_expr_5 ELSE Float64(0) END AS c5, CASE WHEN __common_expr_6 = Float64(0) THEN Float64(0) ELSE __common_expr_6 END AS c6 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
✅
.expect("traversal is infallable"); | ||
} | ||
|
||
/// Return all references to columns and their occurrence counts in the expression. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this new API makes sense to me as a parallel set of APIs for column_refs
/ add_column_refs
datafusion/expr/src/expr.rs
Outdated
/// Adds references to all columns and their occurrence counts in the expression to | ||
/// the map. | ||
/// | ||
/// See [`Self::column_refs`] for details |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
/// See [`Self::column_refs`] for details | |
/// See [`Self::column_refs_counts`] for details |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks, fixed in 17f33f7.
random_state: &'a RandomState, | ||
// a flag to indicate that common expression found | ||
found_common: bool, | ||
// if we are in a conditional branch |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think it would help to document more what is meant by 'conditional' means -- maybe like this
// if we are in a conditional branch | |
// if we are in a conditional branch. A conditional | |
// branch means that the expression **might** not be executed depending | |
// on the runtime values of other expressions, and thus can not be extracted | |
// as a common expression . |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed in b79a9a6.
Ok(match expr { | ||
// If we are already in a conditionally evaluated subtree then continue | ||
// traversal. | ||
_ if self.conditional => TreeNodeRecursion::Continue, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That is a fascinating construct that makes the condition handling uniform 👍
right, | ||
}) => { | ||
left.visit(self)?; | ||
self.conditionally(|visitor| right.visit(visitor).map(|_| ()))?; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
the use of conditionally
makes reading this logic quite elegant. Nice work
} else { | ||
*count += 1; | ||
} | ||
if *count > 1 || *count == 1 && *conditional_count > 0 { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I personally prefer explict parenthesis to avoid confusion
In this case, I think this is the same:
if *count > 1 || *count == 1 && *conditional_count > 0 { | |
if *count > 1 || (*count == 1 && *conditional_count > 0) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sure, fixed in b79a9a6.
physical_plan | ||
01)ProjectionExec: expr=[a@0 = random() AND b@1 = 0 as c1, a@0 = random() AND b@1 = 1 as c2, a@0 = 2 + random() OR b@1 = 4 as c3, a@0 = 2 + random() OR b@1 = 5 as c4, CASE WHEN a@0 = 4 + random() THEN 0 ELSE 1 END as c5, CASE WHEN a@0 = 4 + random() THEN 0 ELSE 2 END as c6] | ||
02)--MemoryExec: partitions=1, partition_sizes=[0] | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could we maybe add some negative tests if they aren't already handled
For example, I think these should not be CSE'd:
(random() = 0 OR a = 1) AND a = 1
(random() = 0 AND a = 1) OR a = 1
CASE
WHEN a + 10 = 0 THEN 0
WHEN random() > 0.5 THEN a+10
ELSE 0
END
CASE
WHEN random() > 0.5 THEN 0
WHEN a + 10 = 0 THEN 0
ELSE a + 10
END
CASE
WHEN a + 10 = 0 THEN 0
WHEN random() > 0.5
WHEN random() > 0.5 THEN a+10
ELSE 0
END
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is a good idea! I forgot about negative tests. But from the first and 3rd case when examples above we can extract a + 10
as it appears in the first when, so it is surely executed, and it also appear in the conditional subtrees.
I've added new tests in 6e3300f.
cc also @haohuaijin |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you @peter-toth . ❤️ -- really nice
Thanks for the review @alamb! |
…valuated stats (apache#11357) * Improve `CommonSubexprEliminate` rule with surely and conditionally evaluated stats * remove expression tree hashing as no longer needed * address review comments * add negative tests
…valuated stats (apache#11357) * Improve `CommonSubexprEliminate` rule with surely and conditionally evaluated stats * remove expression tree hashing as no longer needed * address review comments * add negative tests
…valuated stats (apache#11357) * Improve `CommonSubexprEliminate` rule with surely and conditionally evaluated stats * remove expression tree hashing as no longer needed * address review comments * add negative tests
…valuated stats (apache#11357) * Improve `CommonSubexprEliminate` rule with surely and conditionally evaluated stats * remove expression tree hashing as no longer needed * address review comments * add negative tests
Which issue does this PR close?
Part of #11194.
Rationale for this change
Currently
CommonSubexprEliminate
doesn't recurse into short-circuit expression and misses extracing common expressions from surely evaluated legs. I.e. from the plan:the expression
a + 1
is not extracted despite the fact that it is evaluated 2 times.Also, it would make sense to extract such expressions that are surely evalueted only once but there is a chance that they are evaluated conditionally as well. I.e. from the plan:
it would make sense to extract
a + 1
.What changes are included in this PR?
This PR:
ExprStats
with conditional evaluation counts.ExprIdentifierVisitor
to recurse into children of short-circuit expressions and maintain the state of beeing in a conditional expression branch with a newconditional
flag in the visitor.OptimizeProjections
rule as it currently merges consecutive projections when there are multiple references to a certain column but they occur in 1 project expression.I.e. it currently merges projections:
t1.a = Float64(1)
is used 2 times.Without this bugfix in
OptimizeProjections
the effect ofCommonSubexprEliminate
would be reverted in the optimizer.Expr:column_refs_counts()
andExpr::add_column_ref_counts()
APIs.Are these changes tested?
Yes, added new UTs.
Are there any user-facing changes?
No.