Commit b5df077e authored by Monty's avatar Monty Committed by Sergei Petrunia

Fixed wrong selectivity calculation in table_after_join_selectivity()

The old code counted selectivity double in case of queries like:
WHERE key_part1=1 and key_part2 < 100
if the optimizer would decide to use a REF access on key_part1.

The new code in best_access_path() that changes REF access to RANGE
if the RANGE key is longer makes this issue less likely to happen.

I was not able to create a test case for 11.0, however if one ports this
patch to a MariaDB version without the change of REF to RANGE, the
selectivity will be counted double.
parent ed0a7235
...@@ -10440,14 +10440,15 @@ double table_after_join_selectivity(JOIN *join, uint idx, JOIN_TAB *s, ...@@ -10440,14 +10440,15 @@ double table_after_join_selectivity(JOIN *join, uint idx, JOIN_TAB *s,
{ {
key_part_map quick_key_map= (key_part_map(1) << key_part_map quick_key_map= (key_part_map(1) <<
table->opt_range[key].key_parts) - 1; table->opt_range[key].key_parts) - 1;
if ((table->opt_range[key].rows && if (s->type == JT_RANGE ||
!(quick_key_map & ~table->const_key_parts[key])) || (table->opt_range[key].rows && (table->const_key_parts[key] & 1)))
s->type == JT_RANGE)
{ {
/* /*
Ok, there is an equality for each of the key parts used by the We are either using a range or we are using a REF which the
quick select. This means, quick select's estimate can be reused to same key as an active range and the first key part is a constant.
discount the selectivity of a prefix of a ref access.
In both cases we have to discount the selectivity for the range
as otherwise we are using the selectivity twice.
*/ */
for (; quick_key_map & 1 ; quick_key_map>>= 1) for (; quick_key_map & 1 ; quick_key_map>>= 1)
{ {
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