mariadb/mysql-test/main/mdev-34413-icp-reverse-order.result
Dave Gosselin e059b131ba MDEV-34413 Index Condition Pushdown for reverse ordered scans
Allows index condition pushdown for reverse ordered scans, a previously
disabled feature due to poor performance.  This patch adds a new
API to the handler class called set_end_range which allows callers to
tell the handler what the end of the index range will be when scanning.
Combined with a pushed index condition, the handler can scan the index
efficiently and not read beyond the end of the given range.  When
checking if the pushed index condition matches, the handler will also
check if scanning has reached the end of the provided range and stop if
so.

If we instead only enabled ICP for reverse ordered scans without
also calling this new API, then the handler would perform unnecessary
index condition checks.  In fact this would continue until the end of
the index is reached.

These changes are agnostic of storage engine.  That is, any storage
engine that supports index condition pushdown will inhereit this new
behavior as it is implemented in the SQL and storage engine
API layers.

The partitioned tables storage meta-engine (ha_partition) adds an
override of set_end_range which recursively calls set_end_range on its
child storage engine (handler) implementations.

This commit updates the test made in an earlier commit to show that
ICP matches happen for the reverse ordered case.

This patch is based on changes written by Olav Sandstaa in
MySQL commit da1d92fd46071cd86de61058b6ea39fd9affcd87
2025-03-18 18:35:06 +01:00

278 lines
9 KiB
Text

create table ten(a int);
insert into ten values (0),(1),(2),(3),(4),(5),(6),(7),(8),(9);
create table one_k(a int);
insert into one_k select A.a + B.a* 10 + C.a * 100 from ten A, ten B, ten C;
create table t10 (a int, b int, c int, key(a,b));
insert into t10 select a,a,a from one_k;
select * from t10 force index(a) where a between 10 and 20 and b+1 <3333 order by a desc, b desc;
a b c
20 20 20
19 19 19
18 18 18
17 17 17
16 16 16
15 15 15
14 14 14
13 13 13
12 12 12
11 11 11
10 10 10
explain select * from t10 force index(a) where a between 10 and 20 and b+1 <3333 order by a desc, b desc;
id select_type table type possible_keys key key_len ref rows Extra
1 SIMPLE t10 range a a 5 NULL 11 Using index condition
flush status;
select * from t10 force index(a) where a between 10 and 20 and b+1 <3333 order by a desc, b desc;
a b c
20 20 20
19 19 19
18 18 18
17 17 17
16 16 16
15 15 15
14 14 14
13 13 13
12 12 12
11 11 11
10 10 10
SELECT * FROM information_schema.SESSION_STATUS WHERE VARIABLE_NAME LIKE '%icp%';
VARIABLE_NAME VARIABLE_VALUE
HANDLER_ICP_ATTEMPTS 11
HANDLER_ICP_MATCH 11
select * from t10 force index(a) where a between 10 and 20 and b+1 <3333 order by a asc, b asc;
a b c
10 10 10
11 11 11
12 12 12
13 13 13
14 14 14
15 15 15
16 16 16
17 17 17
18 18 18
19 19 19
20 20 20
explain select * from t10 force index(a) where a between 10 and 20 and b+1 <3333 order by a asc, b asc;
id select_type table type possible_keys key key_len ref rows Extra
1 SIMPLE t10 range a a 5 NULL 11 Using index condition
flush status;
select * from t10 force index(a) where a between 10 and 20 and b+1 <3333 order by a asc, b asc;
a b c
10 10 10
11 11 11
12 12 12
13 13 13
14 14 14
15 15 15
16 16 16
17 17 17
18 18 18
19 19 19
20 20 20
SELECT * FROM information_schema.SESSION_STATUS WHERE VARIABLE_NAME LIKE '%icp%';
VARIABLE_NAME VARIABLE_VALUE
HANDLER_ICP_ATTEMPTS 11
HANDLER_ICP_MATCH 11
select * from t10 force index(a) where a=10 and b+1 <3333 order by a desc, b desc;
a b c
10 10 10
explain select * from t10 force index(a) where a=10 and b+1 <3333 order by a desc, b desc;
id select_type table type possible_keys key key_len ref rows Extra
1 SIMPLE t10 ref a a 5 const 1 Using index condition; Using where
flush status;
select * from t10 force index(a) where a=10 and b+1 <3333 order by a desc, b desc;
a b c
10 10 10
SELECT * FROM information_schema.SESSION_STATUS WHERE VARIABLE_NAME LIKE '%icp%';
VARIABLE_NAME VARIABLE_VALUE
HANDLER_ICP_ATTEMPTS 1
HANDLER_ICP_MATCH 1
select * from t10 force index(a) where a=10 and b+1 <3333 order by a asc, b asc;
a b c
10 10 10
explain select * from t10 force index(a) where a=10 and b+1 <3333 order by a asc, b asc;
id select_type table type possible_keys key key_len ref rows Extra
1 SIMPLE t10 ref a a 5 const 1 Using index condition; Using where
flush status;
select * from t10 force index(a) where a=10 and b+1 <3333 order by a asc, b asc;
a b c
10 10 10
SELECT * FROM information_schema.SESSION_STATUS WHERE VARIABLE_NAME LIKE '%icp%';
VARIABLE_NAME VARIABLE_VALUE
HANDLER_ICP_ATTEMPTS 1
HANDLER_ICP_MATCH 1
select * from t10 force index(a) where a=10 and b+1 <3333 order by a asc, b desc;
a b c
10 10 10
explain select * from t10 force index(a) where a=10 and b+1 <3333 order by a asc, b desc;
id select_type table type possible_keys key key_len ref rows Extra
1 SIMPLE t10 ref a a 5 const 1 Using index condition; Using where
flush status;
select * from t10 force index(a) where a=10 and b+1 <3333 order by a asc, b desc;
a b c
10 10 10
SELECT * FROM information_schema.SESSION_STATUS WHERE VARIABLE_NAME LIKE '%icp%';
VARIABLE_NAME VARIABLE_VALUE
HANDLER_ICP_ATTEMPTS 1
HANDLER_ICP_MATCH 1
select * from t10 force index(a) where a=10 and b+1 <3333 order by a desc, b asc;
a b c
10 10 10
explain select * from t10 force index(a) where a=10 and b+1 <3333 order by a desc, b asc;
id select_type table type possible_keys key key_len ref rows Extra
1 SIMPLE t10 ref a a 5 const 1 Using index condition; Using where
flush status;
select * from t10 force index(a) where a=10 and b+1 <3333 order by a desc, b asc;
a b c
10 10 10
SELECT * FROM information_schema.SESSION_STATUS WHERE VARIABLE_NAME LIKE '%icp%';
VARIABLE_NAME VARIABLE_VALUE
HANDLER_ICP_ATTEMPTS 1
HANDLER_ICP_MATCH 1
create table t1 (a int, b int, c int, key(a,b));
insert into t1 (a, b, c) values (1,10,100),(2,20,200),(3,30,300),(4,40,400),(5,50,500),(6,60,600),(7,70,700),(8,80,800),(9,90,900),(10,100,1000);
select * from t1 where a >= 3 and a <= 3 order by a desc, b desc;
a b c
3 30 300
explain select * from t1 where a >= 3 and a <= 3 order by a desc, b desc;
id select_type table type possible_keys key key_len ref rows Extra
1 SIMPLE t1 range a a 5 NULL 1 Using index condition
flush status;
select * from t1 where a >= 3 and a <= 3 order by a desc, b desc;
a b c
3 30 300
SELECT * FROM information_schema.SESSION_STATUS WHERE VARIABLE_NAME LIKE '%icp%';
VARIABLE_NAME VARIABLE_VALUE
HANDLER_ICP_ATTEMPTS 1
HANDLER_ICP_MATCH 1
select * from t1 where a >= 3 and a <= 3 order by a asc, b asc;
a b c
3 30 300
explain select * from t1 where a >= 3 and a <= 3 order by a asc, b asc;
id select_type table type possible_keys key key_len ref rows Extra
1 SIMPLE t1 range a a 5 NULL 1 Using index condition
flush status;
select * from t1 where a >= 3 and a <= 3 order by a asc, b asc;
a b c
3 30 300
SELECT * FROM information_schema.SESSION_STATUS WHERE VARIABLE_NAME LIKE '%icp%';
VARIABLE_NAME VARIABLE_VALUE
HANDLER_ICP_ATTEMPTS 1
HANDLER_ICP_MATCH 1
drop table t1;
create table t1 (a int, b int, c int, key(a,b));
insert into t1 (a, b, c) values (1,10,100),(2,20,200),(3,30,300);
select * from t1 where a >= 2 and a <= 2 order by a desc, b desc;
a b c
2 20 200
explain select * from t1 where a >= 2 and a <= 2 order by a desc, b desc;
id select_type table type possible_keys key key_len ref rows Extra
1 SIMPLE t1 range a a 5 NULL 1 Using index condition
flush status;
select * from t1 where a >= 2 and a <= 2 order by a desc, b desc;
a b c
2 20 200
SELECT * FROM information_schema.SESSION_STATUS WHERE VARIABLE_NAME LIKE '%icp%';
VARIABLE_NAME VARIABLE_VALUE
HANDLER_ICP_ATTEMPTS 1
HANDLER_ICP_MATCH 1
select * from t1 where a >= 2 and a <= 2 order by a asc, b asc;
a b c
2 20 200
explain select * from t1 where a >= 2 and a <= 2 order by a asc, b asc;
id select_type table type possible_keys key key_len ref rows Extra
1 SIMPLE t1 range a a 5 NULL 1 Using index condition
flush status;
select * from t1 where a >= 2 and a <= 2 order by a asc, b asc;
a b c
2 20 200
SELECT * FROM information_schema.SESSION_STATUS WHERE VARIABLE_NAME LIKE '%icp%';
VARIABLE_NAME VARIABLE_VALUE
HANDLER_ICP_ATTEMPTS 1
HANDLER_ICP_MATCH 1
drop table ten, one_k, t10, t1;
create table t1 (
a int not null,
b int not null,
c int not null,
key (a,b)
) partition by range ((a)) (
partition p0 values less than (5),
partition p1 values less than (10),
partition p2 values less than (15),
partition p3 values less than (20)
);
insert into t1 (a,b,c) values (1,1,1),(2,2,2),(3,3,3),
(4,4,4),(5,5,5),(6,6,6),(7,7,7),(8,8,8),(9,9,9),(10,10,10),
(11,11,11),(12,12,12),(13,13,13),(14,14,14),(15,15,15),
(16,16,16),(17,17,17),(18,18,18),(19,19,19);
select * from t1 where a >= 3 and a <= 7 order by a desc;
a b c
7 7 7
6 6 6
5 5 5
4 4 4
3 3 3
explain select * from t1 where a >= 3 and a <= 7 order by a desc;
id select_type table type possible_keys key key_len ref rows Extra
1 SIMPLE t1 range a a 4 NULL 5 Using index condition
flush status;
select * from t1 where a >= 3 and a <= 7 order by a desc;
a b c
7 7 7
6 6 6
5 5 5
4 4 4
3 3 3
SELECT * FROM information_schema.SESSION_STATUS WHERE VARIABLE_NAME LIKE '%icp%';
VARIABLE_NAME VARIABLE_VALUE
HANDLER_ICP_ATTEMPTS 6
HANDLER_ICP_MATCH 6
select * from t1 where a >= 3 and a <= 7 order by a desc, b desc;
a b c
7 7 7
6 6 6
5 5 5
4 4 4
3 3 3
explain select * from t1 where a >= 3 and a <= 7 order by a desc, b desc;
id select_type table type possible_keys key key_len ref rows Extra
1 SIMPLE t1 range a a 4 NULL 5 Using index condition
flush status;
select * from t1 where a >= 3 and a <= 7 order by a desc, b desc;
a b c
7 7 7
6 6 6
5 5 5
4 4 4
3 3 3
SELECT * FROM information_schema.SESSION_STATUS WHERE VARIABLE_NAME LIKE '%icp%';
VARIABLE_NAME VARIABLE_VALUE
HANDLER_ICP_ATTEMPTS 6
HANDLER_ICP_MATCH 6
drop table t1;
create table t1 (
pk int primary key,
kp1 int, kp2 int,
col1 int,
index (kp1,kp2)
) partition by hash (pk) partitions 10;
insert into t1 select seq, seq, seq, seq from seq_1_to_1000;
select * from t1 where kp1 between 950 and 960 and kp2+1 >33333 order by kp1 asc, kp2 asc;
pk kp1 kp2 col1
flush status;
select * from t1 where kp1 between 950 and 960 and kp2+1 >33333 order by kp1 asc, kp2 asc;
pk kp1 kp2 col1
SELECT * FROM information_schema.SESSION_STATUS WHERE VARIABLE_NAME LIKE '%icp%';
VARIABLE_NAME VARIABLE_VALUE
HANDLER_ICP_ATTEMPTS 11
HANDLER_ICP_MATCH 0
select * from t1 where kp1 between 950 and 960 and kp2+1 >33333 order by kp1 desc, kp2 desc;
pk kp1 kp2 col1
flush status;
select * from t1 where kp1 between 950 and 960 and kp2+1 >33333 order by kp1 desc, kp2 desc;
pk kp1 kp2 col1
SELECT * FROM information_schema.SESSION_STATUS WHERE VARIABLE_NAME LIKE '%icp%';
VARIABLE_NAME VARIABLE_VALUE
HANDLER_ICP_ATTEMPTS 11
HANDLER_ICP_MATCH 0
drop table t1;