Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
M
MariaDB
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
nexedi
MariaDB
Commits
f0dc1892
Commit
f0dc1892
authored
Jun 29, 2001
by
monty@hundin.mysql.fi
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fix ORDER BY ... DESC optimization
parent
545e596e
Changes
7
Expand all
Show whitespace changes
Inline
Side-by-side
Showing
7 changed files
with
355 additions
and
84 deletions
+355
-84
Docs/manual.texi
Docs/manual.texi
+107
-14
mysql-test/r/order_by.result
mysql-test/r/order_by.result
+84
-24
mysql-test/t/order_by.test
mysql-test/t/order_by.test
+35
-8
sql/opt_range.cc
sql/opt_range.cc
+90
-22
sql/opt_range.h
sql/opt_range.h
+6
-4
sql/sql_delete.cc
sql/sql_delete.cc
+1
-5
sql/sql_select.cc
sql/sql_select.cc
+32
-7
No files found.
Docs/manual.texi
View file @
f0dc1892
This diff is collapsed.
Click to expand it.
mysql-test/r/order_by.result
View file @
f0dc1892
...
@@ -112,58 +112,118 @@ member_id nickname voornaam
...
@@ -112,58 +112,118 @@ member_id nickname voornaam
1
1
2
2
table type possible_keys key key_len ref rows Extra
table type possible_keys key key_len ref rows Extra
t1
index NULL a 20 NULL 10
Using index
t1
range a a 20 NULL 2 where used;
Using index
a b c
a b c
1 NULL NULL
1 NULL b
1 NULL b
table type possible_keys key key_len ref rows Extra
t1 range a a 4 NULL 10 where used; Using index
a b c
2 3 c
2 2 b
2 2 a
2 1 b
2 1 a
1 3 b
1 1 b
1 1 b
1 1 NULL
1 NULL b
1 NULL NULL
table type possible_keys key key_len ref rows Extra
t1 ref a a 4 const 5 where used; Using index; Using filesort
a b c
1 3 b
1 1 NULL
1 1 NULL
1 1 b
1 1 b
1 1 b
1 1 b
2 0 a
1 NULL NULL
2 0 b
1 NULL b
table type possible_keys key key_len ref rows Extra
t1 ref a a 9 const,const 2 where used; Using index; Using filesort
a b c
1 NULL NULL
1 NULL b
table type possible_keys key key_len ref rows Extra
t1 range a a 9 NULL 8 where used; Using index; Using filesort
table type possible_keys key key_len ref rows Extra
t1 range a a 9 NULL 5 where used; Using index
table type possible_keys key key_len ref rows Extra
t1 ref a a 9 const,const 1 where used; Using index; Using filesort
table type possible_keys key key_len ref rows Extra
t1 range a a 9 NULL 6 where used; Using index
table type possible_keys key key_len ref rows Extra
t1 range a a 9 NULL 5 where used; Using index
table type possible_keys key key_len ref rows Extra
t1 range a a 9 NULL 2 where used; Using index; Using filesort
table type possible_keys key key_len ref rows Extra
t1 index NULL a 18 NULL 11 Using index
a b c
1 0
1 0 b
1 1
1 1 b
1 1 b
1 3 b
2 1 a
2 1 a
2 1 b
2 1 b
2 1 c
2 2 a
2 2 b
2 3 c
table type possible_keys key key_len ref rows Extra
table type possible_keys key key_len ref rows Extra
t1 index NULL a
20 NULL 10
Using index
t1 index NULL a
18 NULL 11
Using index
a b c
a b c
2 1 c
2 3 c
2 2 b
2 2 a
2 1 b
2 1 b
2 1 a
2 1 a
2 0 b
1 3 b
2 0 a
1 1 b
1 1 b
1 1 b
1 1 b
1 1
NULL
1 1
1
NULL
b
1
0
b
1
NULL NULL
1
0
table type possible_keys key key_len ref rows Extra
table type possible_keys key key_len ref rows Extra
t1 range a a
20 NULL 2
where used; Using index
t1 range a a
18 NULL 3
where used; Using index
a b c
a b c
1 1 b
1 1 b
1 1 b
1 1 b
table type possible_keys key key_len ref rows Extra
table type possible_keys key key_len ref rows Extra
t1 range a a 4 NULL
5
where used; Using index
t1 range a a 4 NULL
6
where used; Using index
a b c
a b c
1 1 b
1 1 b
1 1 b
1 1 b
1 1 NULL
1 1
1 0 b
1 0
count(*)
9
a b c
2 3 c
2 2 b
2 2 a
2 1 b
2 1 a
1 3 b
1 1 b
1 1 b
1 1
table type possible_keys key key_len ref rows Extra
table type possible_keys key key_len ref rows Extra
t1 range a a
9 NULL 7
where used; Using index
t1 range a a
8 NULL 10
where used; Using index
a b c
a b c
2 1 c
2 1 b
2 1 b
2 1 a
2 1 a
2 0 b
2 0 a
1 1 b
1 1 b
1 1 b
1 1 b
1 1 NULL
1 1
1 0 b
1 0
table type possible_keys key key_len ref rows Extra
table type possible_keys key key_len ref rows Extra
t1 range a a 4 NULL
4
where used; Using index
t1 range a a 4 NULL
5
where used; Using index
a b c
a b c
1 3 b
1 1 b
1 1 b
1 1 b
1 1 b
1 1
NULL
1 1
1
NULL
b
1
0
b
1
NULL NULL
1
0
mysql-test/t/order_by.test
View file @
f0dc1892
...
@@ -168,8 +168,8 @@ drop table t1,t2,t3;
...
@@ -168,8 +168,8 @@ drop table t1,t2,t3;
#bug reported by Wouter de Jong
#bug reported by Wouter de Jong
drop
table
if
exists
members
;
drop
table
if
exists
t1
;
CREATE
TABLE
members
(
CREATE
TABLE
t1
(
member_id
int
(
11
)
NOT
NULL
auto_increment
,
member_id
int
(
11
)
NOT
NULL
auto_increment
,
inschrijf_datum
varchar
(
20
)
NOT
NULL
default
''
,
inschrijf_datum
varchar
(
20
)
NOT
NULL
default
''
,
lastchange_datum
varchar
(
20
)
NOT
NULL
default
''
,
lastchange_datum
varchar
(
20
)
NOT
NULL
default
''
,
...
@@ -200,24 +200,51 @@ CREATE TABLE members (
...
@@ -200,24 +200,51 @@ CREATE TABLE members (
PRIMARY
KEY
(
member_id
)
PRIMARY
KEY
(
member_id
)
)
TYPE
=
MyISAM
PACK_KEYS
=
1
;
)
TYPE
=
MyISAM
PACK_KEYS
=
1
;
insert
into
members
(
member_id
)
values
(
1
),(
2
),(
3
);
insert
into
t1
(
member_id
)
values
(
1
),(
2
),(
3
);
select
member_id
,
nickname
,
voornaam
FROM
members
select
member_id
,
nickname
,
voornaam
FROM
t1
ORDER
by
lastchange_datum
DESC
LIMIT
2
;
ORDER
by
lastchange_datum
DESC
LIMIT
2
;
drop
table
members
;
drop
table
t1
;
#
# Test optimization of ORDER BY DESC
#
create
table
t1
(
a
int
not
null
,
b
int
,
c
varchar
(
10
),
key
(
a
,
b
,
c
));
create
table
t1
(
a
int
not
null
,
b
int
,
c
varchar
(
10
),
key
(
a
,
b
,
c
));
insert
into
t1
values
(
1
,
NULL
,
NULL
),
(
1
,
NULL
,
'b'
),
(
1
,
1
,
NULL
),
(
1
,
1
,
'b'
),
(
1
,
1
,
'b'
),
(
2
,
0
,
'a'
),
(
2
,
0
,
'b'
),
(
2
,
1
,
'a'
),
(
2
,
1
,
'b'
),
(
2
,
1
,
'c'
);
insert
into
t1
values
(
1
,
NULL
,
NULL
),
(
1
,
NULL
,
'b'
),
(
1
,
1
,
NULL
),
(
1
,
1
,
'b'
),
(
1
,
1
,
'b'
),
(
2
,
1
,
'a'
),
(
2
,
1
,
'b'
),
(
2
,
2
,
'a'
),
(
2
,
2
,
'b'
),
(
2
,
3
,
'c'
),(
1
,
3
,
'b'
);
explain
select
*
from
t1
where
(
a
=
1
and
b
is
null
and
c
=
'b'
)
or
(
a
>
2
)
order
by
a
desc
;
select
*
from
t1
where
(
a
=
1
and
b
is
null
and
c
=
'b'
)
or
(
a
>
2
)
order
by
a
desc
;
explain
select
*
from
t1
where
a
>=
1
and
a
<
3
order
by
a
desc
;
select
*
from
t1
where
a
>=
1
and
a
<
3
order
by
a
desc
;
explain
select
*
from
t1
where
a
=
1
order
by
a
desc
,
b
desc
;
select
*
from
t1
where
a
=
1
order
by
a
desc
,
b
desc
;
explain
select
*
from
t1
where
a
=
1
and
b
is
null
order
by
a
desc
,
b
desc
;
select
*
from
t1
where
a
=
1
and
b
is
null
order
by
a
desc
,
b
desc
;
explain
select
*
from
t1
where
a
>=
1
and
a
<
3
and
b
>
0
order
by
a
desc
,
b
desc
;
explain
select
*
from
t1
where
a
=
2
and
b
>
0
order
by
a
desc
,
b
desc
;
explain
select
*
from
t1
where
a
=
2
and
b
is
null
order
by
a
desc
,
b
desc
;
explain
select
*
from
t1
where
a
=
2
and
(
b
is
null
or
b
>
0
)
order
by
a
desc
,
b
desc
;
explain
select
*
from
t1
where
a
=
2
and
b
>
0
order
by
a
desc
,
b
desc
;
explain
select
*
from
t1
where
a
=
2
and
b
<
2
order
by
a
desc
,
b
desc
;
#
# Test things when we don't have NULL keys
#
alter
table
t1
modify
b
int
not
null
,
modify
c
varchar
(
10
)
not
null
;
explain
select
*
from
t1
order
by
a
,
b
,
c
;
explain
select
*
from
t1
order
by
a
,
b
,
c
;
select
*
from
t1
order
by
a
,
b
,
c
;
select
*
from
t1
order
by
a
,
b
,
c
;
explain
select
*
from
t1
order
by
a
desc
,
b
desc
,
c
desc
;
explain
select
*
from
t1
order
by
a
desc
,
b
desc
,
c
desc
;
select
*
from
t1
order
by
a
desc
,
b
desc
,
c
desc
;
select
*
from
t1
order
by
a
desc
,
b
desc
,
c
desc
;
# test multiple ranges, NO_MAX_RANGE and EQ_RANGE
# test multiple ranges, NO_MAX_RANGE and EQ_RANGE
explain
select
*
from
t1
where
(
a
=
1
and
b
is
null
and
c
=
'b'
)
or
(
a
>
2
)
order
by
a
desc
;
explain
select
*
from
t1
where
(
a
=
1
and
b
=
1
and
c
=
'b'
)
or
(
a
>
2
)
order
by
a
desc
;
select
*
from
t1
where
(
a
=
1
and
b
=
1
and
c
=
'b'
)
or
(
a
>
2
)
order
by
a
desc
;
select
*
from
t1
where
(
a
=
1
and
b
=
1
and
c
=
'b'
)
or
(
a
>
2
)
order
by
a
desc
;
# test NEAR_MAX, NO_MIN_RANGE
# test NEAR_MAX, NO_MIN_RANGE
explain
select
*
from
t1
where
a
<
2
and
b
<=
1
order
by
a
desc
,
b
desc
;
explain
select
*
from
t1
where
a
<
2
and
b
<=
1
order
by
a
desc
,
b
desc
;
select
*
from
t1
where
a
<
2
and
b
<=
1
order
by
a
desc
,
b
desc
;
select
*
from
t1
where
a
<
2
and
b
<=
1
order
by
a
desc
,
b
desc
;
select
count
(
*
)
from
t1
where
a
<
5
and
b
>
0
;
select
*
from
t1
where
a
<
5
and
b
>
0
order
by
a
desc
,
b
desc
;
# test HA_READ_AFTER_KEY (at the end of the file), NEAR_MIN
# test HA_READ_AFTER_KEY (at the end of the file), NEAR_MIN
explain
select
*
from
t1
where
a
between
1
and
3
and
b
<=
1
order
by
a
desc
,
b
desc
;
explain
select
*
from
t1
where
a
between
1
and
3
and
b
<=
1
order
by
a
desc
,
b
desc
;
select
*
from
t1
where
a
between
1
and
3
and
b
<=
1
order
by
a
desc
,
b
desc
;
select
*
from
t1
where
a
between
1
and
3
and
b
<=
1
order
by
a
desc
,
b
desc
;
...
@@ -226,4 +253,4 @@ explain select * from t1 where a between 0 and 1 order by a desc, b desc;
...
@@ -226,4 +253,4 @@ explain select * from t1 where a between 0 and 1 order by a desc, b desc;
select
*
from
t1
where
a
between
0
and
1
order
by
a
desc
,
b
desc
;
select
*
from
t1
where
a
between
0
and
1
order
by
a
desc
,
b
desc
;
drop
table
t1
;
drop
table
t1
;
/* vim:set ft=sql sw=2 noet: */
sql/opt_range.cc
View file @
f0dc1892
...
@@ -382,7 +382,7 @@ SQL_SELECT::~SQL_SELECT()
...
@@ -382,7 +382,7 @@ SQL_SELECT::~SQL_SELECT()
#undef index // Fix for Unixware 7
#undef index // Fix for Unixware 7
QUICK_SELECT
::
QUICK_SELECT
(
TABLE
*
table
,
uint
key_nr
,
bool
no_alloc
)
QUICK_SELECT
::
QUICK_SELECT
(
TABLE
*
table
,
uint
key_nr
,
bool
no_alloc
)
:
error
(
0
),
index
(
key_nr
),
max_used_key_length
(
0
),
head
(
table
),
:
dont_free
(
0
),
error
(
0
),
index
(
key_nr
),
max_used_key_length
(
0
),
head
(
table
),
it
(
ranges
),
range
(
0
)
it
(
ranges
),
range
(
0
)
{
{
if
(
!
no_alloc
)
if
(
!
no_alloc
)
...
@@ -399,8 +399,11 @@ QUICK_SELECT::QUICK_SELECT(TABLE *table,uint key_nr,bool no_alloc)
...
@@ -399,8 +399,11 @@ QUICK_SELECT::QUICK_SELECT(TABLE *table,uint key_nr,bool no_alloc)
QUICK_SELECT
::~
QUICK_SELECT
()
QUICK_SELECT
::~
QUICK_SELECT
()
{
{
if
(
!
dont_free
)
{
file
->
index_end
();
file
->
index_end
();
free_root
(
&
alloc
,
MYF
(
0
));
free_root
(
&
alloc
,
MYF
(
0
));
}
}
}
int
QUICK_SELECT
::
init
()
int
QUICK_SELECT
::
init
()
...
@@ -2516,6 +2519,7 @@ int QUICK_SELECT::cmp_next(QUICK_RANGE *range)
...
@@ -2516,6 +2519,7 @@ int QUICK_SELECT::cmp_next(QUICK_RANGE *range)
return
(
range
->
flag
&
NEAR_MAX
)
?
1
:
0
;
// Exact match
return
(
range
->
flag
&
NEAR_MAX
)
?
1
:
0
;
// Exact match
}
}
/*
/*
* This is a hack: we inherit from QUICK_SELECT so that we can use the
* This is a hack: we inherit from QUICK_SELECT so that we can use the
* get_next() interface, but we have to hold a pointer to the original
* get_next() interface, but we have to hold a pointer to the original
...
@@ -2525,29 +2529,44 @@ int QUICK_SELECT::cmp_next(QUICK_RANGE *range)
...
@@ -2525,29 +2529,44 @@ int QUICK_SELECT::cmp_next(QUICK_RANGE *range)
* which handle the ranges and implement the get_next() function. But
* which handle the ranges and implement the get_next() function. But
* for now, this seems to work right at least.
* for now, this seems to work right at least.
*/
*/
QUICK_SELECT_DESC
::
QUICK_SELECT_DESC
(
QUICK_SELECT
*
q
)
:
QUICK_SELECT
(
*
q
),
quick
(
q
),
rev_it
(
rev_ranges
)
QUICK_SELECT_DESC
::
QUICK_SELECT_DESC
(
QUICK_SELECT
*
q
,
uint
used_key_parts
)
:
QUICK_SELECT
(
*
q
),
rev_it
(
rev_ranges
)
{
{
bool
not_read_after_key
=
file
->
option_flag
()
&
HA_NOT_READ_AFTER_KEY
;
bool
not_read_after_key
=
file
->
option_flag
()
&
HA_NOT_READ_AFTER_KEY
;
for
(
QUICK_RANGE
*
r
=
it
++
;
r
;
r
=
it
++
)
for
(
QUICK_RANGE
*
r
=
it
++
;
r
;
r
=
it
++
)
{
{
rev_ranges
.
push_front
(
r
);
rev_ranges
.
push_front
(
r
);
if
(
not_read_after_key
&&
range_reads_after_key
(
r
))
if
(
not_read_after_key
&&
range_reads_after_key
(
r
)
||
test_if_null_range
(
r
,
used_key_parts
))
{
{
it
.
rewind
();
// Reset range
error
=
HA_ERR_UNSUPPORTED
;
error
=
HA_ERR_UNSUPPORTED
;
break
;
dont_free
=
1
;
// Don't free memory from 'q'
return
;
}
}
}
}
/* Remove EQ_RANGE flag for keys that are not using the full key */
for
(
QUICK_RANGE
*
r
=
rev_it
++
;
r
;
r
=
rev_it
++
)
{
if
((
r
->
flag
&
EQ_RANGE
)
&&
head
->
key_info
[
index
].
key_length
!=
r
->
max_length
)
r
->
flag
&=
~
EQ_RANGE
;
}
rev_it
.
rewind
();
q
->
dont_free
=
1
;
// Don't free shared mem
delete
q
;
}
}
int
QUICK_SELECT_DESC
::
get_next
()
int
QUICK_SELECT_DESC
::
get_next
()
{
{
DBUG_ENTER
(
"QUICK_SELECT_DESC::get_next"
);
DBUG_ENTER
(
"QUICK_SELECT_DESC::get_next"
);
/* The max key is handled as follows:
/* The max key is handled as follows:
* - if there is NO_MAX_RANGE, start at the end and move backwards
* - if there is NO_MAX_RANGE, start at the end and move backwards
* - if it is an EQ_RANGE
and max key covers the entire key, go directly
* - if it is an EQ_RANGE
, which means that max key covers the entire
* to the key and read through it (sorting backwards is
*
key, go directly
to the key and read through it (sorting backwards is
* same as sorting forwards)
* same as sorting forwards)
* - if it is NEAR_MAX, go to the key or next, step back once, and
* - if it is NEAR_MAX, go to the key or next, step back once, and
* move backwards
* move backwards
...
@@ -2560,8 +2579,8 @@ int QUICK_SELECT_DESC::get_next()
...
@@ -2560,8 +2579,8 @@ int QUICK_SELECT_DESC::get_next()
int
result
;
int
result
;
if
(
range
)
if
(
range
)
{
// Already read through key
{
// Already read through key
result
=
((
range
->
flag
&
EQ_RANGE
)
?
result
=
((
range
->
flag
&
EQ_RANGE
)
file
->
index_next_same
(
record
,
(
byte
*
)
range
->
min_key
,
?
file
->
index_next_same
(
record
,
(
byte
*
)
range
->
min_key
,
range
->
min_length
)
:
range
->
min_length
)
:
file
->
index_prev
(
record
));
file
->
index_prev
(
record
));
if
(
!
result
)
if
(
!
result
)
...
@@ -2587,8 +2606,7 @@ int QUICK_SELECT_DESC::get_next()
...
@@ -2587,8 +2606,7 @@ int QUICK_SELECT_DESC::get_next()
continue
;
continue
;
}
}
if
(
range
->
flag
&
EQ_RANGE
&&
if
(
range
->
flag
&
EQ_RANGE
)
head
->
key_info
[
index
].
key_length
==
range
->
max_length
)
{
{
result
=
file
->
index_read
(
record
,
(
byte
*
)
range
->
max_key
,
result
=
file
->
index_read
(
record
,
(
byte
*
)
range
->
max_key
,
range
->
max_length
,
HA_READ_KEY_EXACT
);
range
->
max_length
,
HA_READ_KEY_EXACT
);
...
@@ -2598,9 +2616,10 @@ int QUICK_SELECT_DESC::get_next()
...
@@ -2598,9 +2616,10 @@ int QUICK_SELECT_DESC::get_next()
dbug_assert
(
range
->
flag
&
NEAR_MAX
||
range_reads_after_key
(
range
));
dbug_assert
(
range
->
flag
&
NEAR_MAX
||
range_reads_after_key
(
range
));
/* Note: even if max_key is only a prefix, HA_READ_AFTER_KEY will
/* Note: even if max_key is only a prefix, HA_READ_AFTER_KEY will
* do the right thing - go past all keys which match the prefix */
* do the right thing - go past all keys which match the prefix */
file
->
index_read
(
record
,
(
byte
*
)
range
->
max_key
,
range
->
max_length
,
result
=
file
->
index_read
(
record
,
(
byte
*
)
range
->
max_key
,
range
->
max_length
,
((
range
->
flag
&
NEAR_MAX
)
?
((
range
->
flag
&
NEAR_MAX
)
?
HA_READ_KEY_OR_PREV
:
HA_READ_AFTER_KEY
));
HA_READ_KEY_EXACT
:
HA_READ_AFTER_KEY
));
result
=
file
->
index_prev
(
record
);
result
=
file
->
index_prev
(
record
);
}
}
if
(
result
)
if
(
result
)
...
@@ -2629,7 +2648,7 @@ int QUICK_SELECT_DESC::cmp_prev(QUICK_RANGE *range)
...
@@ -2629,7 +2648,7 @@ int QUICK_SELECT_DESC::cmp_prev(QUICK_RANGE *range)
if
(
range
->
flag
&
NO_MIN_RANGE
)
if
(
range
->
flag
&
NO_MIN_RANGE
)
return
(
0
);
/* key can't be to small */
return
(
0
);
/* key can't be to small */
KEY_PART
*
key_part
=
quick
->
key_parts
;
KEY_PART
*
key_part
=
key_parts
;
for
(
char
*
key
=
range
->
min_key
,
*
end
=
key
+
range
->
min_length
;
for
(
char
*
key
=
range
->
min_key
,
*
end
=
key
+
range
->
min_length
;
key
<
end
;
key
<
end
;
key
+=
key_part
++->
part_length
)
key
+=
key_part
++->
part_length
)
...
@@ -2659,15 +2678,64 @@ int QUICK_SELECT_DESC::cmp_prev(QUICK_RANGE *range)
...
@@ -2659,15 +2678,64 @@ int QUICK_SELECT_DESC::cmp_prev(QUICK_RANGE *range)
/*
/*
* True if this range will require using HA_READ_AFTER_KEY
* True if this range will require using HA_READ_AFTER_KEY
See comment in get_next() about this
*/
*/
bool
QUICK_SELECT_DESC
::
range_reads_after_key
(
QUICK_RANGE
*
range
)
bool
QUICK_SELECT_DESC
::
range_reads_after_key
(
QUICK_RANGE
*
range
)
{
{
// See comment in get_next()
return
((
range
->
flag
&
(
NO_MAX_RANGE
|
NEAR_MAX
))
||
return
range
->
flag
&
(
NO_MAX_RANGE
|
NEAR_MAX
)
?
1
:
!
(
range
->
flag
&
EQ_RANGE
)
||
(
range
->
flag
&
EQ_RANGE
&&
head
->
key_info
[
index
].
key_length
!=
range
->
max_length
)
?
1
:
0
;
head
->
key_info
[
index
].
key_length
==
range
->
max_length
)
?
0
:
1
;
}
/* True if we are reading over a key that may have a NULL value */
bool
QUICK_SELECT_DESC
::
test_if_null_range
(
QUICK_RANGE
*
range
,
uint
used_key_parts
)
{
uint
offset
,
end
;
KEY_PART
*
key_part
=
key_parts
,
*
key_part_end
=
key_part
+
used_key_parts
;
for
(
offset
=
0
,
end
=
min
(
range
->
min_length
,
range
->
max_length
)
;
offset
<
end
&&
key_part
!=
key_part_end
;
offset
+=
key_part
++->
part_length
)
{
uint
null_length
=
test
(
key_part
->
null_bit
);
if
(
!
memcmp
((
char
*
)
range
->
min_key
+
offset
,
(
char
*
)
range
->
max_key
+
offset
,
key_part
->
part_length
+
null_length
))
{
offset
+=
null_length
;
continue
;
}
if
(
null_length
&&
range
->
min_key
[
offset
])
return
1
;
// min_key is null and max_key isn't
// Range doesn't cover NULL. This is ok if there is no more null parts
break
;
}
/*
If the next min_range is > NULL, then we can use this, even if
it's a NULL key
Example: SELECT * FROM t1 WHERE a = 2 AND b >0 ORDER BY a DESC,b DESC;
*/
if
(
key_part
!=
key_part_end
&&
key_part
->
null_bit
)
{
if
(
offset
>=
range
->
min_length
||
range
->
min_key
[
offset
])
return
1
;
// Could be null
key_part
++
;
}
/*
If any of the key parts used in the ORDER BY could be NULL, we can't
use the key to sort the data.
*/
for
(;
key_part
!=
key_part_end
;
key_part
++
)
if
(
key_part
->
null_bit
)
return
1
;
// Covers null part
return
0
;
}
}
/*****************************************************************************
/*****************************************************************************
** Print a quick range for debugging
** Print a quick range for debugging
** TODO:
** TODO:
...
...
sql/opt_range.h
View file @
f0dc1892
...
@@ -54,9 +54,10 @@ class QUICK_RANGE :public Sql_alloc {
...
@@ -54,9 +54,10 @@ class QUICK_RANGE :public Sql_alloc {
{}
{}
};
};
class
QUICK_SELECT
{
class
QUICK_SELECT
{
public:
public:
bool
next
;
bool
next
,
dont_free
;
int
error
;
int
error
;
uint
index
,
max_used_key_length
;
uint
index
,
max_used_key_length
;
TABLE
*
head
;
TABLE
*
head
;
...
@@ -80,16 +81,17 @@ class QUICK_SELECT {
...
@@ -80,16 +81,17 @@ class QUICK_SELECT {
bool
unique_key_range
();
bool
unique_key_range
();
};
};
class
QUICK_SELECT_DESC
:
public
QUICK_SELECT
class
QUICK_SELECT_DESC
:
public
QUICK_SELECT
{
{
public:
public:
QUICK_SELECT_DESC
(
QUICK_SELECT
*
q
);
QUICK_SELECT_DESC
(
QUICK_SELECT
*
q
,
uint
used_key_parts
);
int
get_next
();
int
get_next
();
private:
private:
int
cmp_prev
(
QUICK_RANGE
*
range
);
int
cmp_prev
(
QUICK_RANGE
*
range
);
bool
range_reads_after_key
(
QUICK_RANGE
*
range
);
bool
range_reads_after_key
(
QUICK_RANGE
*
range
);
bool
test_if_null_range
(
QUICK_RANGE
*
range
,
uint
used_key_parts
);
QUICK_SELECT
*
quick
;
void
reset
(
void
)
{
next
=
0
;
rev_it
.
rewind
();
}
List
<
QUICK_RANGE
>
rev_ranges
;
List
<
QUICK_RANGE
>
rev_ranges
;
List_iterator
<
QUICK_RANGE
>
rev_it
;
List_iterator
<
QUICK_RANGE
>
rev_it
;
};
};
...
...
sql/sql_delete.cc
View file @
f0dc1892
...
@@ -290,11 +290,7 @@ int mysql_delete(THD *thd,
...
@@ -290,11 +290,7 @@ int mysql_delete(THD *thd,
** delete multiple tables from join
** delete multiple tables from join
***************************************************************************/
***************************************************************************/
#ifndef DBUG_OFF
#define MEM_STRIP_BUF_SIZE sortbuff_size
#define MEM_STRIP_BUF_SIZE 2048
#else
#define MEM_STRIP_BUF_SIZE sortbuffer_size
#endif
#ifndef SINISAS_STRIP
#ifndef SINISAS_STRIP
int
refposcmp2
(
void
*
arg
,
const
void
*
a
,
const
void
*
b
)
int
refposcmp2
(
void
*
arg
,
const
void
*
a
,
const
void
*
b
)
...
...
sql/sql_select.cc
View file @
f0dc1892
...
@@ -5153,9 +5153,11 @@ part_of_refkey(TABLE *table,Field *field)
...
@@ -5153,9 +5153,11 @@ part_of_refkey(TABLE *table,Field *field)
** Returns: 1 if key is ok.
** Returns: 1 if key is ok.
** 0 if key can't be used
** 0 if key can't be used
** -1 if reverse key can be used
** -1 if reverse key can be used
** used_key_parts is set to key parts used if length != 0
*****************************************************************************/
*****************************************************************************/
static
int
test_if_order_by_key
(
ORDER
*
order
,
TABLE
*
table
,
uint
idx
)
static
int
test_if_order_by_key
(
ORDER
*
order
,
TABLE
*
table
,
uint
idx
,
uint
*
used_key_parts
)
{
{
KEY_PART_INFO
*
key_part
,
*
key_part_end
;
KEY_PART_INFO
*
key_part
,
*
key_part_end
;
key_part
=
table
->
key_info
[
idx
].
key_part
;
key_part
=
table
->
key_info
[
idx
].
key_part
;
...
@@ -5187,6 +5189,7 @@ static int test_if_order_by_key(ORDER *order, TABLE *table, uint idx)
...
@@ -5187,6 +5189,7 @@ static int test_if_order_by_key(ORDER *order, TABLE *table, uint idx)
reverse
=
flag
;
// Remember if reverse
reverse
=
flag
;
// Remember if reverse
key_part
++
;
key_part
++
;
}
}
*
used_key_parts
=
(
uint
)
(
key_part
-
table
->
key_info
[
idx
].
key_part
);
return
reverse
;
return
reverse
;
}
}
...
@@ -5249,17 +5252,38 @@ test_if_skip_sort_order(JOIN_TAB *tab,ORDER *order,ha_rows select_limit,
...
@@ -5249,17 +5252,38 @@ test_if_skip_sort_order(JOIN_TAB *tab,ORDER *order,ha_rows select_limit,
if
(
ref_key
>=
0
)
if
(
ref_key
>=
0
)
{
{
int
order_direction
;
int
order_direction
;
uint
used_key_parts
;
/* Check if we get the rows in requested sorted order by using the key */
/* Check if we get the rows in requested sorted order by using the key */
if
((
usable_keys
&
((
key_map
)
1
<<
ref_key
))
&&
if
((
usable_keys
&
((
key_map
)
1
<<
ref_key
))
&&
(
order_direction
=
test_if_order_by_key
(
order
,
table
,
ref_key
)))
(
order_direction
=
test_if_order_by_key
(
order
,
table
,
ref_key
,
&
used_key_parts
)))
{
{
if
(
order_direction
==
-
1
&&
select
&&
select
->
quick
)
if
(
order_direction
==
-
1
)
{
if
(
select
&&
select
->
quick
)
{
{
// ORDER BY ref_key DESC
// ORDER BY ref_key DESC
select
->
quick
=
new
QUICK_SELECT_DESC
(
select
->
quick
);
QUICK_SELECT_DESC
*
tmp
=
new
QUICK_SELECT_DESC
(
select
->
quick
,
if
(
select
->
quick
->
error
)
used_key_parts
);
if
(
!
tmp
||
tmp
->
error
)
{
delete
tmp
;
DBUG_RETURN
(
0
);
// Reverse sort not supported
}
select
->
quick
=
tmp
;
DBUG_RETURN
(
1
);
}
if
(
tab
->
ref
.
key_parts
<
used_key_parts
)
{
/*
SELECT * FROM t1 WHERE a=1 ORDER BY a DESC,b DESC
TODO:
Add a new traversal function to read last matching row and
traverse backwards.
*/
DBUG_RETURN
(
0
);
DBUG_RETURN
(
0
);
}
}
}
DBUG_RETURN
(
1
);
/* No need to sort */
DBUG_RETURN
(
1
);
/* No need to sort */
}
}
}
}
...
@@ -5280,10 +5304,11 @@ test_if_skip_sort_order(JOIN_TAB *tab,ORDER *order,ha_rows select_limit,
...
@@ -5280,10 +5304,11 @@ test_if_skip_sort_order(JOIN_TAB *tab,ORDER *order,ha_rows select_limit,
for
(
nr
=
0
;
keys
;
keys
>>=
1
,
nr
++
)
for
(
nr
=
0
;
keys
;
keys
>>=
1
,
nr
++
)
{
{
uint
not_used
;
if
(
keys
&
1
)
if
(
keys
&
1
)
{
{
int
flag
;
int
flag
;
if
((
flag
=
test_if_order_by_key
(
order
,
table
,
nr
)))
if
((
flag
=
test_if_order_by_key
(
order
,
table
,
nr
,
&
not_used
)))
{
{
if
(
!
no_changes
)
if
(
!
no_changes
)
{
{
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment