Commit a2066b24 authored by Sergei Petrunia's avatar Sergei Petrunia

MDEV-30651: Assertion `sel->quick' in make_range_rowid_filters

The optimizer deals with Rowid Filters this way:

1. First, range optimizer is invoked. It saves information
   about all potential range accesses.
2. A query plan is chosen. Suppose, it uses a Rowid Filter on
   index $IDX.
3. JOIN::make_range_rowid_filters() calls the range optimizer
again to create a quick select on index $IDX which will be used
to populate the rowid filter.

The problem: KILL command catches the query in step #3. Quick
Select is not created which causes a crash.

Fixed by checking if query was killed. Note: the problem also
affects 10.6, even if error handling for
SQL_SELECT::test_quick_select is different there.
parent b47bd3f8
...@@ -55,5 +55,33 @@ disconnect con1; ...@@ -55,5 +55,33 @@ disconnect con1;
reap; reap;
set debug_sync='RESET'; set debug_sync='RESET';
--echo #
--echo # MDEV-30651: SIGSEGV in st_join_table::save_explain_data and
--echo # Assertion `sel->quick' failed in make_range_rowid_filters
--echo #
--echo # Reusing table t2 and t3 from previous test
let $target_id= `select connection_id()`;
set debug_sync='in_forced_range_optimize SIGNAL ready1 WAIT_FOR go1';
send
explain
select * from t2, t3
where
t3.key1=t2.a and t3.key2 in (2,3);
connect (con1, localhost, root,,);
set debug_sync='now WAIT_FOR ready1';
evalp kill query $target_id;
set debug_sync='now SIGNAL go1';
connection default;
disconnect con1;
--error ER_QUERY_INTERRUPTED
reap;
set debug_sync='RESET';
drop table t2,t3; drop table t2,t3;
--source include/wait_until_count_sessions.inc --source include/wait_until_count_sessions.inc
...@@ -46,5 +46,23 @@ connection default; ...@@ -46,5 +46,23 @@ connection default;
disconnect con1; disconnect con1;
ERROR 70100: Query execution was interrupted ERROR 70100: Query execution was interrupted
set debug_sync='RESET'; set debug_sync='RESET';
#
# MDEV-30651: SIGSEGV in st_join_table::save_explain_data and
# Assertion `sel->quick' failed in make_range_rowid_filters
#
# Reusing table t2 and t3 from previous test
set debug_sync='in_forced_range_optimize SIGNAL ready1 WAIT_FOR go1';
explain
select * from t2, t3
where
t3.key1=t2.a and t3.key2 in (2,3);
connect con1, localhost, root,,;
set debug_sync='now WAIT_FOR ready1';
kill query $target_id;
set debug_sync='now SIGNAL go1';
connection default;
disconnect con1;
ERROR 70100: Query execution was interrupted
set debug_sync='RESET';
drop table t2,t3; drop table t2,t3;
set default_storage_engine=default; set default_storage_engine=default;
...@@ -45,4 +45,22 @@ connection default; ...@@ -45,4 +45,22 @@ connection default;
disconnect con1; disconnect con1;
ERROR 70100: Query execution was interrupted ERROR 70100: Query execution was interrupted
set debug_sync='RESET'; set debug_sync='RESET';
#
# MDEV-30651: SIGSEGV in st_join_table::save_explain_data and
# Assertion `sel->quick' failed in make_range_rowid_filters
#
# Reusing table t2 and t3 from previous test
set debug_sync='in_forced_range_optimize SIGNAL ready1 WAIT_FOR go1';
explain
select * from t2, t3
where
t3.key1=t2.a and t3.key2 in (2,3);
connect con1, localhost, root,,;
set debug_sync='now WAIT_FOR ready1';
kill query $target_id;
set debug_sync='now SIGNAL go1';
connection default;
disconnect con1;
ERROR 70100: Query execution was interrupted
set debug_sync='RESET';
drop table t2,t3; drop table t2,t3;
...@@ -2708,7 +2708,10 @@ int SQL_SELECT::test_quick_select(THD *thd, key_map keys_to_use, ...@@ -2708,7 +2708,10 @@ int SQL_SELECT::test_quick_select(THD *thd, key_map keys_to_use,
only_single_index_range_scan= 1; only_single_index_range_scan= 1;
if (head->force_index || force_quick_range) if (head->force_index || force_quick_range)
{
DEBUG_SYNC(thd, "in_forced_range_optimize");
scan_time= read_time= DBL_MAX; scan_time= read_time= DBL_MAX;
}
else else
{ {
scan_time= rows2double(records) / TIME_FOR_COMPARE; scan_time= rows2double(records) / TIME_FOR_COMPARE;
......
...@@ -1783,7 +1783,9 @@ int JOIN::optimize() ...@@ -1783,7 +1783,9 @@ int JOIN::optimize()
object a pointer to which is set in the field JOIN_TAB::rowid_filter of object a pointer to which is set in the field JOIN_TAB::rowid_filter of
the joined table. the joined table.
@retval false always @retval
false OK
true Error, query should abort
*/ */
bool JOIN::make_range_rowid_filters() bool JOIN::make_range_rowid_filters()
...@@ -1830,8 +1832,11 @@ bool JOIN::make_range_rowid_filters() ...@@ -1830,8 +1832,11 @@ bool JOIN::make_range_rowid_filters()
(ha_rows) HA_POS_ERROR, (ha_rows) HA_POS_ERROR,
true, false, true, true); true, false, true, true);
tab->table->force_index= force_index_save; tab->table->force_index= force_index_save;
if (thd->is_error()) if (thd->is_error() || thd->check_killed())
goto no_filter; {
delete sel;
DBUG_RETURN(true);
}
/* /*
If SUBS_IN_TO_EXISTS strtrategy is chosen for the subquery then If SUBS_IN_TO_EXISTS strtrategy is chosen for the subquery then
additional conditions are injected into WHERE/ON/HAVING and it may additional conditions are injected into WHERE/ON/HAVING and it may
...@@ -1855,8 +1860,6 @@ bool JOIN::make_range_rowid_filters() ...@@ -1855,8 +1860,6 @@ bool JOIN::make_range_rowid_filters()
continue; continue;
} }
no_filter: no_filter:
if (sel->quick)
delete sel->quick;
delete sel; delete sel;
} }
......
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