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
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
mariadb
Commits
6348e63f
Commit
6348e63f
authored
Jul 07, 2003
by
bell@sanja.is.com.ua
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Optimisation if simple IN subselect with primary index
(SCRUM) (part of WL#818)
parent
7ad53317
Changes
7
Show whitespace changes
Inline
Side-by-side
Showing
7 changed files
with
177 additions
and
10 deletions
+177
-10
mysql-test/r/subselect.result
mysql-test/r/subselect.result
+9
-1
mysql-test/t/subselect.test
mysql-test/t/subselect.test
+2
-0
sql/item_subselect.cc
sql/item_subselect.cc
+78
-4
sql/item_subselect.h
sql/item_subselect.h
+43
-0
sql/sql_lex.h
sql/sql_lex.h
+1
-1
sql/sql_select.cc
sql/sql_select.cc
+41
-2
sql/sql_select.h
sql/sql_select.h
+3
-2
No files found.
mysql-test/r/subselect.result
View file @
6348e63f
...
@@ -1226,7 +1226,15 @@ a
...
@@ -1226,7 +1226,15 @@ a
explain select * from t2 where t2.a in (select a from t1);
explain select * from t2 where t2.a in (select a from t1);
id select_type table type possible_keys key key_len ref rows Extra
id select_type table type possible_keys key key_len ref rows Extra
1 PRIMARY t2 index NULL PRIMARY 4 NULL 4 Using where; Using index
1 PRIMARY t2 index NULL PRIMARY 4 NULL 4 Using where; Using index
2 DEPENDENT SUBQUERY t1 eq_ref PRIMARY PRIMARY 4 func 1 Using where; Using index
2 DEPENDENT SUBQUERY t1 simple_in PRIMARY PRIMARY 4 func 1 Using index
select * from t2 where t2.a in (select a from t1 where t1.b <> 30);
a
2
4
explain select * from t2 where t2.a in (select a from t1 where t1.b <> 30);
id select_type table type possible_keys key key_len ref rows Extra
1 PRIMARY t2 index NULL PRIMARY 4 NULL 4 Using where; Using index
2 DEPENDENT SUBQUERY t1 simple_in PRIMARY PRIMARY 4 func 1 Using index; Using where
select * from t2 where t2.a in (select t1.a from t1,t3 where t1.b=t3.a);
select * from t2 where t2.a in (select t1.a from t1,t3 where t1.b=t3.a);
a
a
2
2
...
...
mysql-test/t/subselect.test
View file @
6348e63f
...
@@ -819,6 +819,8 @@ insert into t2 values (2), (3), (4), (5);
...
@@ -819,6 +819,8 @@ insert into t2 values (2), (3), (4), (5);
insert
into
t3
values
(
10
,
3
),
(
20
,
4
),
(
30
,
5
);
insert
into
t3
values
(
10
,
3
),
(
20
,
4
),
(
30
,
5
);
select
*
from
t2
where
t2
.
a
in
(
select
a
from
t1
);
select
*
from
t2
where
t2
.
a
in
(
select
a
from
t1
);
explain
select
*
from
t2
where
t2
.
a
in
(
select
a
from
t1
);
explain
select
*
from
t2
where
t2
.
a
in
(
select
a
from
t1
);
select
*
from
t2
where
t2
.
a
in
(
select
a
from
t1
where
t1
.
b
<>
30
);
explain
select
*
from
t2
where
t2
.
a
in
(
select
a
from
t1
where
t1
.
b
<>
30
);
select
*
from
t2
where
t2
.
a
in
(
select
t1
.
a
from
t1
,
t3
where
t1
.
b
=
t3
.
a
);
select
*
from
t2
where
t2
.
a
in
(
select
t1
.
a
from
t1
,
t3
where
t1
.
b
=
t3
.
a
);
explain
select
*
from
t2
where
t2
.
a
in
(
select
t1
.
a
from
t1
,
t3
where
t1
.
b
=
t3
.
a
);
explain
select
*
from
t2
where
t2
.
a
in
(
select
t1
.
a
from
t1
,
t3
where
t1
.
b
=
t3
.
a
);
drop
table
t1
,
t2
,
t3
;
drop
table
t1
,
t2
,
t3
;
...
...
sql/item_subselect.cc
View file @
6348e63f
...
@@ -36,7 +36,7 @@ inline Item * and_items(Item* cond, Item *item)
...
@@ -36,7 +36,7 @@ inline Item * and_items(Item* cond, Item *item)
Item_subselect
::
Item_subselect
()
:
Item_subselect
::
Item_subselect
()
:
Item_result_field
(),
engine_owner
(
1
),
value_assigned
(
0
),
substitution
(
0
),
Item_result_field
(),
engine_owner
(
1
),
value_assigned
(
0
),
substitution
(
0
),
have_to_be_excluded
(
0
)
have_to_be_excluded
(
0
)
,
engine_changed
(
0
)
{
{
reset
();
reset
();
/*
/*
...
@@ -117,16 +117,22 @@ bool Item_subselect::fix_fields(THD *thd_param, TABLE_LIST *tables, Item **ref)
...
@@ -117,16 +117,22 @@ bool Item_subselect::fix_fields(THD *thd_param, TABLE_LIST *tables, Item **ref)
bool
Item_subselect
::
exec
()
bool
Item_subselect
::
exec
()
{
{
int
res
;
MEM_ROOT
*
old_root
=
my_pthread_getspecific_ptr
(
MEM_ROOT
*
,
THR_MALLOC
);
MEM_ROOT
*
old_root
=
my_pthread_getspecific_ptr
(
MEM_ROOT
*
,
THR_MALLOC
);
if
(
&
thd
->
mem_root
!=
old_root
)
if
(
&
thd
->
mem_root
!=
old_root
)
{
{
my_pthread_setspecific_ptr
(
THR_MALLOC
,
&
thd
->
mem_root
);
my_pthread_setspecific_ptr
(
THR_MALLOC
,
&
thd
->
mem_root
);
int
res
=
engine
->
exec
();
res
=
engine
->
exec
();
my_pthread_setspecific_ptr
(
THR_MALLOC
,
old_root
);
my_pthread_setspecific_ptr
(
THR_MALLOC
,
old_root
);
return
(
res
);
}
}
else
else
return
engine
->
exec
();
res
=
engine
->
exec
();
if
(
engine_changed
)
{
engine_changed
=
0
;
return
exec
();
}
return
(
res
);
}
}
Item
::
Type
Item_subselect
::
type
()
const
Item
::
Type
Item_subselect
::
type
()
const
...
@@ -795,6 +801,13 @@ int subselect_union_engine::prepare()
...
@@ -795,6 +801,13 @@ int subselect_union_engine::prepare()
return
unit
->
prepare
(
thd
,
result
,
0
);
return
unit
->
prepare
(
thd
,
result
,
0
);
}
}
int
subselect_simplein_engine
::
prepare
()
{
//this never should be called
DBUG_ASSERT
(
0
);
return
1
;
}
static
Item_result
set_row
(
SELECT_LEX
*
select_lex
,
Item
*
item
,
static
Item_result
set_row
(
SELECT_LEX
*
select_lex
,
Item
*
item
,
Item_cache
**
row
,
bool
*
maybe_null
)
Item_cache
**
row
,
bool
*
maybe_null
)
{
{
...
@@ -873,6 +886,12 @@ void subselect_union_engine::fix_length_and_dec(Item_cache **row)
...
@@ -873,6 +886,12 @@ void subselect_union_engine::fix_length_and_dec(Item_cache **row)
}
}
}
}
void
subselect_simplein_engine
::
fix_length_and_dec
(
Item_cache
**
row
)
{
//this never should be called
DBUG_ASSERT
(
0
);
}
int
subselect_single_select_engine
::
exec
()
int
subselect_single_select_engine
::
exec
()
{
{
DBUG_ENTER
(
"subselect_single_select_engine::exec"
);
DBUG_ENTER
(
"subselect_single_select_engine::exec"
);
...
@@ -889,6 +908,10 @@ int subselect_single_select_engine::exec()
...
@@ -889,6 +908,10 @@ int subselect_single_select_engine::exec()
join
->
thd
->
lex
.
current_select
=
save_select
;
join
->
thd
->
lex
.
current_select
=
save_select
;
DBUG_RETURN
(
join
->
error
?
join
->
error
:
1
);
DBUG_RETURN
(
join
->
error
?
join
->
error
:
1
);
}
}
if
(
item
->
engine_changed
)
{
DBUG_RETURN
(
1
);
}
}
}
if
((
select_lex
->
dependent
||
select_lex
->
uncacheable
)
&&
executed
)
if
((
select_lex
->
dependent
||
select_lex
->
uncacheable
)
&&
executed
)
{
{
...
@@ -922,6 +945,51 @@ int subselect_union_engine::exec()
...
@@ -922,6 +945,51 @@ int subselect_union_engine::exec()
return
res
;
return
res
;
}
}
int
subselect_simplein_engine
::
exec
()
{
DBUG_ENTER
(
"subselect_simplein_engine::exec"
);
int
error
;
TABLE
*
table
=
tab
->
table
;
if
((
tab
->
ref
.
key_err
=
(
*
tab
->
ref
.
key_copy
)
->
copy
()))
{
table
->
status
=
STATUS_NOT_FOUND
;
error
=
-
1
;
}
else
{
error
=
table
->
file
->
index_read
(
table
->
record
[
0
],
tab
->
ref
.
key_buff
,
tab
->
ref
.
key_length
,
HA_READ_KEY_EXACT
);
if
(
error
&&
error
!=
HA_ERR_KEY_NOT_FOUND
)
error
=
report_error
(
table
,
error
);
else
{
error
=
0
;
table
->
null_row
=
0
;
if
(
table
->
status
)
((
Item_in_subselect
*
)
item
)
->
value
=
0
;
else
((
Item_in_subselect
*
)
item
)
->
value
=
(
!
cond
||
cond
->
val_int
()
?
1
:
0
);
}
}
{
int
tmp
=
0
;
if
((
tmp
=
table
->
file
->
extra
(
HA_EXTRA_NO_CACHE
)))
{
DBUG_PRINT
(
"error"
,
(
"extra(HA_EXTRA_NO_CACHE) failed"
));
error
=
1
;
}
if
((
tmp
=
table
->
file
->
index_end
()))
{
DBUG_PRINT
(
"error"
,
(
"index_end() failed"
));
error
=
1
;
}
if
(
error
==
1
)
table
->
file
->
print_error
(
tmp
,
MYF
(
0
));
}
DBUG_RETURN
(
error
!=
0
)
}
uint
subselect_single_select_engine
::
cols
()
uint
subselect_single_select_engine
::
cols
()
{
{
return
select_lex
->
item_list
.
elements
;
return
select_lex
->
item_list
.
elements
;
...
@@ -961,3 +1029,9 @@ void subselect_union_engine::exclude()
...
@@ -961,3 +1029,9 @@ void subselect_union_engine::exclude()
{
{
unit
->
exclude_level
();
unit
->
exclude_level
();
}
}
void
subselect_simplein_engine
::
exclude
()
{
//this never should be called
DBUG_ASSERT
(
0
);
}
sql/item_subselect.h
View file @
6348e63f
...
@@ -48,7 +48,12 @@ protected:
...
@@ -48,7 +48,12 @@ protected:
bool
have_to_be_excluded
;
bool
have_to_be_excluded
;
public:
public:
/* changed engine indicator */
bool
engine_changed
;
enum
trans_res
{
OK
,
REDUCE
,
ERROR
};
enum
trans_res
{
OK
,
REDUCE
,
ERROR
};
enum
subs_type
{
UNKNOWN_SUBS
,
SINGLEROW_SUBS
,
EXISTS_SUBS
,
IN_SUBS
,
ALLANY_SUBS
};
Item_subselect
();
Item_subselect
();
Item_subselect
(
Item_subselect
*
item
)
Item_subselect
(
Item_subselect
*
item
)
...
@@ -59,9 +64,12 @@ public:
...
@@ -59,9 +64,12 @@ public:
max_columns
=
item
->
max_columns
;
max_columns
=
item
->
max_columns
;
engine
=
item
->
engine
;
engine
=
item
->
engine
;
engine_owner
=
0
;
engine_owner
=
0
;
engine_changed
=
item
->
engine_changed
;
name
=
item
->
name
;
name
=
item
->
name
;
}
}
virtual
subs_type
substype
()
{
return
UNKNOWN_SUBS
;
}
/*
/*
We need this method, because some compilers do not allow 'this'
We need this method, because some compilers do not allow 'this'
pointer in constructor initialization list, but we need pass pointer
pointer in constructor initialization list, but we need pass pointer
...
@@ -95,6 +103,12 @@ public:
...
@@ -95,6 +103,12 @@ public:
else
else
str
->
append
(
"-subselect-"
);
str
->
append
(
"-subselect-"
);
}
}
bool
change_engine
(
subselect_engine
*
eng
)
{
engine
=
eng
;
engine_changed
=
1
;
return
eng
==
0
;
}
friend
class
select_subselect
;
friend
class
select_subselect
;
friend
class
Item_in_optimizer
;
friend
class
Item_in_optimizer
;
...
@@ -116,6 +130,9 @@ public:
...
@@ -116,6 +130,9 @@ public:
max_length
=
item
->
max_length
;
max_length
=
item
->
max_length
;
decimals
=
item
->
decimals
;
decimals
=
item
->
decimals
;
}
}
subs_type
substype
()
{
return
SINGLEROW_SUBS
;
}
void
reset
();
void
reset
();
trans_res
select_transformer
(
JOIN
*
join
);
trans_res
select_transformer
(
JOIN
*
join
);
void
store
(
uint
i
,
Item
*
item
);
void
store
(
uint
i
,
Item
*
item
);
...
@@ -152,6 +169,7 @@ public:
...
@@ -152,6 +169,7 @@ public:
}
}
Item_exists_subselect
()
:
Item_subselect
()
{}
Item_exists_subselect
()
:
Item_subselect
()
{}
subs_type
substype
()
{
return
EXISTS_SUBS
;
}
void
reset
()
void
reset
()
{
{
value
=
0
;
value
=
0
;
...
@@ -165,6 +183,7 @@ public:
...
@@ -165,6 +183,7 @@ public:
void
fix_length_and_dec
();
void
fix_length_and_dec
();
friend
class
select_exists_subselect
;
friend
class
select_exists_subselect
;
friend
class
subselect_simplein_engine
;
};
};
/* IN subselect */
/* IN subselect */
...
@@ -185,6 +204,8 @@ public:
...
@@ -185,6 +204,8 @@ public:
Item_in_subselect
(
THD
*
thd
,
Item
*
left_expr
,
st_select_lex
*
select_lex
);
Item_in_subselect
(
THD
*
thd
,
Item
*
left_expr
,
st_select_lex
*
select_lex
);
Item_in_subselect
(
Item_in_subselect
*
item
);
Item_in_subselect
(
Item_in_subselect
*
item
);
Item_in_subselect
()
:
Item_exists_subselect
(),
abort_on_null
(
0
)
{}
Item_in_subselect
()
:
Item_exists_subselect
(),
abort_on_null
(
0
)
{}
subs_type
substype
()
{
return
IN_SUBS
;
}
void
reset
()
void
reset
()
{
{
value
=
0
;
value
=
0
;
...
@@ -218,6 +239,7 @@ public:
...
@@ -218,6 +239,7 @@ public:
Item_allany_subselect
(
THD
*
thd
,
Item
*
left_expr
,
compare_func_creator
f
,
Item_allany_subselect
(
THD
*
thd
,
Item
*
left_expr
,
compare_func_creator
f
,
st_select_lex
*
select_lex
);
st_select_lex
*
select_lex
);
Item_allany_subselect
(
Item_allany_subselect
*
item
);
Item_allany_subselect
(
Item_allany_subselect
*
item
);
subs_type
substype
()
{
return
ALLANY_SUBS
;
}
trans_res
select_transformer
(
JOIN
*
join
);
trans_res
select_transformer
(
JOIN
*
join
);
};
};
...
@@ -288,3 +310,24 @@ public:
...
@@ -288,3 +310,24 @@ public:
bool
uncacheable
();
bool
uncacheable
();
void
exclude
();
void
exclude
();
};
};
struct
st_join_table
;
class
subselect_simplein_engine
:
public
subselect_engine
{
st_join_table
*
tab
;
Item
*
cond
;
public:
subselect_simplein_engine
(
THD
*
thd
,
st_join_table
*
tab_arg
,
Item_subselect
*
subs
,
Item
*
where
)
:
subselect_engine
(
thd
,
subs
,
0
),
tab
(
tab_arg
),
cond
(
where
)
{}
int
prepare
();
void
fix_length_and_dec
(
Item_cache
**
row
);
int
exec
();
uint
cols
()
{
return
1
;
}
bool
dependent
()
{
return
1
;
}
bool
uncacheable
()
{
return
1
;
}
void
exclude
();
};
sql/sql_lex.h
View file @
6348e63f
...
@@ -283,7 +283,7 @@ public:
...
@@ -283,7 +283,7 @@ public:
st_select_lex
*
return_to
;
st_select_lex
*
return_to
;
/* LIMIT clause runtime counters */
/* LIMIT clause runtime counters */
ha_rows
select_limit_cnt
,
offset_limit_cnt
;
ha_rows
select_limit_cnt
,
offset_limit_cnt
;
/* not NULL if uni
on
used in subselect, point to subselect item */
/* not NULL if uni
t
used in subselect, point to subselect item */
Item_subselect
*
item
;
Item_subselect
*
item
;
/* thread handler */
/* thread handler */
THD
*
thd
;
THD
*
thd
;
...
...
sql/sql_select.cc
View file @
6348e63f
...
@@ -32,7 +32,7 @@
...
@@ -32,7 +32,7 @@
const
char
*
join_type_str
[]
=
{
"UNKNOWN"
,
"system"
,
"const"
,
"eq_ref"
,
"ref"
,
const
char
*
join_type_str
[]
=
{
"UNKNOWN"
,
"system"
,
"const"
,
"eq_ref"
,
"ref"
,
"MAYBE_REF"
,
"ALL"
,
"range"
,
"index"
,
"fulltext"
,
"MAYBE_REF"
,
"ALL"
,
"range"
,
"index"
,
"fulltext"
,
"ref_or_null"
"ref_or_null"
,
"simple_in"
};
};
static
void
optimize_keyuse
(
JOIN
*
join
,
DYNAMIC_ARRAY
*
keyuse_array
);
static
void
optimize_keyuse
(
JOIN
*
join
,
DYNAMIC_ARRAY
*
keyuse_array
);
...
@@ -723,6 +723,45 @@ JOIN::optimize()
...
@@ -723,6 +723,45 @@ JOIN::optimize()
(
select_lex
->
ftfunc_list
->
elements
?
(
select_lex
->
ftfunc_list
->
elements
?
SELECT_NO_JOIN_CACHE
:
0
));
SELECT_NO_JOIN_CACHE
:
0
));
/*
is this simple IN subquery?
*/
if
(
!
group_list
&&
!
order
&&
!
having
&&
unit
->
item
&&
unit
->
item
->
substype
()
==
Item_subselect
::
IN_SUBS
&&
tables
==
1
&&
join_tab
[
0
].
type
==
JT_EQ_REF
&&
conds
&&
!
unit
->
first_select
()
->
next_select
())
{
Item
*
where
=
0
;
bool
ok
=
0
;
if
(
conds
->
type
()
==
Item
::
FUNC_ITEM
&&
((
class
Item_func
*
)
this
->
conds
)
->
functype
()
==
Item_func
::
EQ_FUNC
&&
((
Item_func
*
)
conds
)
->
arguments
()[
0
]
->
type
()
==
Item
::
REF_ITEM
&&
((
Item_func
*
)
conds
)
->
arguments
()[
1
]
->
type
()
==
Item
::
FIELD_ITEM
)
{
ok
=
1
;
join_tab
->
info
=
"Using index"
;
}
else
if
(
conds
->
type
()
==
Item
::
COND_ITEM
&&
((
class
Item_func
*
)
this
->
conds
)
->
functype
()
==
Item_func
::
COND_AND_FUNC
)
{
ok
=
1
;
where
=
conds
;
join_tab
->
info
=
"Using index; Using where"
;
}
if
(
ok
)
{
join_tab
[
0
].
type
=
JT_SIMPLE_IN
;
error
=
0
;
DBUG_RETURN
(
unit
->
item
->
change_engine
(
new
subselect_simplein_engine
(
thd
,
join_tab
,
unit
->
item
,
where
)));
}
}
/*
/*
Need to tell Innobase that to play it safe, it should fetch all
Need to tell Innobase that to play it safe, it should fetch all
columns of the tables: this is because MySQL may build row
columns of the tables: this is because MySQL may build row
...
@@ -5337,7 +5376,7 @@ flush_cached_records(JOIN *join,JOIN_TAB *join_tab,bool skipp_last)
...
@@ -5337,7 +5376,7 @@ flush_cached_records(JOIN *join,JOIN_TAB *join_tab,bool skipp_last)
/* Help function when we get some an error from the table handler */
/* Help function when we get some an error from the table handler */
static
int
report_error
(
TABLE
*
table
,
int
error
)
int
report_error
(
TABLE
*
table
,
int
error
)
{
{
if
(
error
==
HA_ERR_END_OF_FILE
||
error
==
HA_ERR_KEY_NOT_FOUND
)
if
(
error
==
HA_ERR_END_OF_FILE
||
error
==
HA_ERR_KEY_NOT_FOUND
)
{
{
...
...
sql/sql_select.h
View file @
6348e63f
...
@@ -75,7 +75,8 @@ typedef struct st_join_cache {
...
@@ -75,7 +75,8 @@ typedef struct st_join_cache {
*/
*/
enum
join_type
{
JT_UNKNOWN
,
JT_SYSTEM
,
JT_CONST
,
JT_EQ_REF
,
JT_REF
,
JT_MAYBE_REF
,
enum
join_type
{
JT_UNKNOWN
,
JT_SYSTEM
,
JT_CONST
,
JT_EQ_REF
,
JT_REF
,
JT_MAYBE_REF
,
JT_ALL
,
JT_RANGE
,
JT_NEXT
,
JT_FT
,
JT_REF_OR_NULL
};
JT_ALL
,
JT_RANGE
,
JT_NEXT
,
JT_FT
,
JT_REF_OR_NULL
,
JT_SIMPLE_IN
};
class
JOIN
;
class
JOIN
;
...
@@ -305,7 +306,6 @@ bool create_myisam_from_heap(THD *thd, TABLE *table, TMP_TABLE_PARAM *param,
...
@@ -305,7 +306,6 @@ bool create_myisam_from_heap(THD *thd, TABLE *table, TMP_TABLE_PARAM *param,
/* functions from opt_sum.cc */
/* functions from opt_sum.cc */
int
opt_sum_query
(
TABLE_LIST
*
tables
,
List
<
Item
>
&
all_fields
,
COND
*
conds
);
int
opt_sum_query
(
TABLE_LIST
*
tables
,
List
<
Item
>
&
all_fields
,
COND
*
conds
);
/* class to copying an field/item to a key struct */
/* class to copying an field/item to a key struct */
class
store_key
:
public
Sql_alloc
class
store_key
:
public
Sql_alloc
...
@@ -407,3 +407,4 @@ public:
...
@@ -407,3 +407,4 @@ public:
bool
cp_buffer_from_ref
(
TABLE_REF
*
ref
);
bool
cp_buffer_from_ref
(
TABLE_REF
*
ref
);
bool
error_if_full_join
(
JOIN
*
join
);
bool
error_if_full_join
(
JOIN
*
join
);
void
relink_tables
(
SELECT_LEX
*
select_lex
);
void
relink_tables
(
SELECT_LEX
*
select_lex
);
int
report_error
(
TABLE
*
table
,
int
error
);
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