mirror of
https://github.com/MariaDB/server.git
synced 2025-01-18 04:53:01 +01:00
2cfc450bf7
"Re-factor the code for post-join operations". The patch mainly contains the code ported from mysql-5.6 and created for two essential architectural changes: 1. WL#5558: Resolve ORDER BY execution method at the optimization stage 2. WL#6071: Inline tmp tables into the nested loops algorithm The first task was implemented for mysql-5.6 by Ole John Aske. It allows to make all decisions on ORDER BY operation at the optimization stage. The second task implemented for mysql-5.6 by Evgeny Potemkin adds JOIN_TAB nodes for post-join operations that require temporary tables. It allows to execute these operations within the nested loops algorithm that used to be used before this task only for join queries. Besides these task moves all planning on the execution of these operations from the execution phase to the optimization phase. Some other re-factoring changes of mysql-5.6 were pulled in, mainly because it was easier to pull them in than roll them back. In particular all changes concerning Ref_ptr_array were incorporated. The port required some changes in the MariaDB code that concerned the functionality of EXPLAIN and ANALYZE. This was done mainly by Sergey Petrunia.
6610 lines
190 KiB
C++
6610 lines
190 KiB
C++
/* Copyright (c) 2002, 2015, Oracle and/or its affiliates.
|
|
Copyright (c) 2010, 2015, MariaDB
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; version 2 of the License.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
|
|
|
/**
|
|
@file
|
|
|
|
@brief
|
|
subselect Item
|
|
|
|
@todo
|
|
- add function from mysql_select that use JOIN* as parameter to JOIN
|
|
methods (sql_select.h/sql_select.cc)
|
|
*/
|
|
|
|
#ifdef USE_PRAGMA_IMPLEMENTATION
|
|
#pragma implementation // gcc: Class implementation
|
|
#endif
|
|
|
|
#include <my_global.h>
|
|
#include "sql_priv.h"
|
|
/*
|
|
It is necessary to include set_var.h instead of item.h because there
|
|
are dependencies on include order for set_var.h and item.h. This
|
|
will be resolved later.
|
|
*/
|
|
#include "sql_class.h" // set_var.h: THD
|
|
#include "set_var.h"
|
|
#include "sql_select.h"
|
|
#include "sql_parse.h" // check_stack_overrun
|
|
#include "sql_test.h"
|
|
|
|
double get_post_group_estimate(JOIN* join, double join_op_rows);
|
|
|
|
const char *exists_outer_expr_name= "<exists outer expr>";
|
|
|
|
int check_and_do_in_subquery_rewrites(JOIN *join);
|
|
|
|
Item_subselect::Item_subselect(THD *thd_arg):
|
|
Item_result_field(thd_arg), Used_tables_and_const_cache(),
|
|
value_assigned(0), own_engine(0), thd(0), old_engine(0),
|
|
have_to_be_excluded(0),
|
|
inside_first_fix_fields(0), done_first_fix_fields(FALSE),
|
|
expr_cache(0), forced_const(FALSE), substitution(0), engine(0), eliminated(FALSE),
|
|
changed(0), is_correlated(FALSE)
|
|
{
|
|
DBUG_ENTER("Item_subselect::Item_subselect");
|
|
DBUG_PRINT("enter", ("this: 0x%lx", (ulong) this));
|
|
#ifndef DBUG_OFF
|
|
exec_counter= 0;
|
|
#endif
|
|
with_subselect= 1;
|
|
reset();
|
|
/*
|
|
Item value is NULL if select_result_interceptor didn't change this value
|
|
(i.e. some rows will be found returned)
|
|
*/
|
|
null_value= TRUE;
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
void Item_subselect::init(st_select_lex *select_lex,
|
|
select_result_interceptor *result)
|
|
{
|
|
/*
|
|
Please see Item_singlerow_subselect::invalidate_and_restore_select_lex(),
|
|
which depends on alterations to the parse tree implemented here.
|
|
*/
|
|
|
|
DBUG_ENTER("Item_subselect::init");
|
|
DBUG_PRINT("enter", ("select_lex: 0x%lx this: 0x%lx",
|
|
(ulong) select_lex, (ulong) this));
|
|
unit= select_lex->master_unit();
|
|
thd= unit->thd;
|
|
|
|
if (unit->item)
|
|
{
|
|
engine= unit->item->engine;
|
|
parsing_place= unit->item->parsing_place;
|
|
if (unit->item->substype() == EXISTS_SUBS &&
|
|
((Item_exists_subselect *)unit->item)->exists_transformed)
|
|
{
|
|
/* it is permanent transformation of EXISTS to IN */
|
|
unit->item= this;
|
|
engine->change_result(this, result, FALSE);
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
Item can be changed in JOIN::prepare while engine in JOIN::optimize
|
|
=> we do not copy old_engine here
|
|
*/
|
|
thd->change_item_tree((Item**)&unit->item, this);
|
|
engine->change_result(this, result, TRUE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SELECT_LEX *outer_select= unit->outer_select();
|
|
/*
|
|
do not take into account expression inside aggregate functions because
|
|
they can access original table fields
|
|
*/
|
|
parsing_place= (outer_select->in_sum_expr ?
|
|
NO_MATTER :
|
|
outer_select->parsing_place);
|
|
if (unit->is_union())
|
|
engine= new subselect_union_engine(thd, unit, result, this);
|
|
else
|
|
engine= new subselect_single_select_engine(thd, select_lex, result, this);
|
|
}
|
|
{
|
|
SELECT_LEX *upper= unit->outer_select();
|
|
if (upper->parsing_place == IN_HAVING)
|
|
upper->subquery_in_having= 1;
|
|
/* The subquery is an expression cache candidate */
|
|
upper->expr_cache_may_be_used[upper->parsing_place]= TRUE;
|
|
}
|
|
DBUG_PRINT("info", ("engine: 0x%lx", (ulong)engine));
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
st_select_lex *
|
|
Item_subselect::get_select_lex()
|
|
{
|
|
return unit->first_select();
|
|
}
|
|
|
|
void Item_subselect::cleanup()
|
|
{
|
|
DBUG_ENTER("Item_subselect::cleanup");
|
|
Item_result_field::cleanup();
|
|
if (old_engine)
|
|
{
|
|
if (engine)
|
|
engine->cleanup();
|
|
engine= old_engine;
|
|
old_engine= 0;
|
|
}
|
|
if (engine)
|
|
engine->cleanup();
|
|
reset();
|
|
value_assigned= 0;
|
|
expr_cache= 0;
|
|
forced_const= FALSE;
|
|
DBUG_PRINT("info", ("exec_counter: %d", exec_counter));
|
|
#ifndef DBUG_OFF
|
|
exec_counter= 0;
|
|
#endif
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
void Item_singlerow_subselect::cleanup()
|
|
{
|
|
DBUG_ENTER("Item_singlerow_subselect::cleanup");
|
|
value= 0; row= 0;
|
|
Item_subselect::cleanup();
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
void Item_in_subselect::cleanup()
|
|
{
|
|
DBUG_ENTER("Item_in_subselect::cleanup");
|
|
if (left_expr_cache)
|
|
{
|
|
left_expr_cache->delete_elements();
|
|
delete left_expr_cache;
|
|
left_expr_cache= NULL;
|
|
}
|
|
/*
|
|
TODO: This breaks the commented assert in add_strategy().
|
|
in_strategy&= ~SUBS_STRATEGY_CHOSEN;
|
|
*/
|
|
first_execution= TRUE;
|
|
pushed_cond_guards= NULL;
|
|
Item_subselect::cleanup();
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
void Item_allany_subselect::cleanup()
|
|
{
|
|
/*
|
|
The MAX/MIN transformation through injection is reverted through the
|
|
change_item_tree() mechanism. Revert the select_lex object of the
|
|
query to its initial state.
|
|
*/
|
|
for (SELECT_LEX *sl= unit->first_select();
|
|
sl; sl= sl->next_select())
|
|
if (test_set_strategy(SUBS_MAXMIN_INJECTED))
|
|
sl->with_sum_func= false;
|
|
Item_in_subselect::cleanup();
|
|
}
|
|
|
|
|
|
Item_subselect::~Item_subselect()
|
|
{
|
|
DBUG_ENTER("Item_subselect::~Item_subselect");
|
|
DBUG_PRINT("enter", ("this: 0x%lx", (ulong) this));
|
|
if (own_engine)
|
|
delete engine;
|
|
else
|
|
engine->cleanup();
|
|
engine= NULL;
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
bool
|
|
Item_subselect::select_transformer(JOIN *join)
|
|
{
|
|
DBUG_ENTER("Item_subselect::select_transformer");
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
|
|
bool Item_subselect::fix_fields(THD *thd_param, Item **ref)
|
|
{
|
|
char const *save_where= thd_param->where;
|
|
uint8 uncacheable;
|
|
bool res;
|
|
|
|
status_var_increment(thd_param->status_var.feature_subquery);
|
|
|
|
DBUG_ASSERT(fixed == 0);
|
|
engine->set_thd((thd= thd_param));
|
|
if (!done_first_fix_fields)
|
|
{
|
|
done_first_fix_fields= TRUE;
|
|
inside_first_fix_fields= TRUE;
|
|
upper_refs.empty();
|
|
/*
|
|
psergey-todo: remove _first_fix_fields calls, we need changes on every
|
|
execution
|
|
*/
|
|
}
|
|
|
|
eliminated= FALSE;
|
|
parent_select= thd_param->lex->current_select;
|
|
|
|
if (check_stack_overrun(thd, STACK_MIN_SIZE, (uchar*)&res))
|
|
return TRUE;
|
|
|
|
|
|
if (!(res= engine->prepare()))
|
|
{
|
|
// all transformation is done (used by prepared statements)
|
|
changed= 1;
|
|
inside_first_fix_fields= FALSE;
|
|
|
|
/*
|
|
Substitute the current item with an Item_in_optimizer that was
|
|
created by Item_in_subselect::select_in_like_transformer and
|
|
call fix_fields for the substituted item which in turn calls
|
|
engine->prepare for the subquery predicate.
|
|
*/
|
|
if (substitution)
|
|
{
|
|
/*
|
|
If the top item of the WHERE/HAVING condition changed,
|
|
set correct WHERE/HAVING for PS.
|
|
*/
|
|
if (unit->outer_select()->where == (*ref))
|
|
unit->outer_select()->where= substitution;
|
|
else if (unit->outer_select()->having == (*ref))
|
|
unit->outer_select()->having= substitution;
|
|
|
|
(*ref)= substitution;
|
|
substitution->name= name;
|
|
substitution->name_length= name_length;
|
|
if (have_to_be_excluded)
|
|
engine->exclude();
|
|
substitution= 0;
|
|
thd->where= "checking transformed subquery";
|
|
if (!(*ref)->fixed)
|
|
res= (*ref)->fix_fields(thd, ref);
|
|
goto end;
|
|
|
|
}
|
|
// Is it one field subselect?
|
|
if (engine->cols() > max_columns)
|
|
{
|
|
my_error(ER_OPERAND_COLUMNS, MYF(0), 1);
|
|
|
|
goto end;
|
|
}
|
|
fix_length_and_dec();
|
|
}
|
|
else
|
|
goto end;
|
|
|
|
if ((uncacheable= engine->uncacheable() & ~UNCACHEABLE_EXPLAIN))
|
|
{
|
|
const_item_cache= 0;
|
|
if (uncacheable & UNCACHEABLE_RAND)
|
|
used_tables_cache|= RAND_TABLE_BIT;
|
|
}
|
|
fixed= 1;
|
|
|
|
end:
|
|
done_first_fix_fields= FALSE;
|
|
inside_first_fix_fields= FALSE;
|
|
thd->where= save_where;
|
|
return res;
|
|
}
|
|
|
|
|
|
bool Item_subselect::enumerate_field_refs_processor(uchar *arg)
|
|
{
|
|
List_iterator<Ref_to_outside> it(upper_refs);
|
|
Ref_to_outside *upper;
|
|
|
|
while ((upper= it++))
|
|
{
|
|
if (upper->item->walk(&Item::enumerate_field_refs_processor, FALSE, arg))
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
bool Item_subselect::mark_as_eliminated_processor(uchar *arg)
|
|
{
|
|
eliminated= TRUE;
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/**
|
|
Remove a subselect item from its unit so that the unit no longer
|
|
represents a subquery.
|
|
|
|
@param arg unused parameter
|
|
|
|
@return
|
|
FALSE to force the evaluation of the processor for the subsequent items.
|
|
*/
|
|
|
|
bool Item_subselect::eliminate_subselect_processor(uchar *arg)
|
|
{
|
|
unit->item= NULL;
|
|
unit->exclude_from_tree();
|
|
eliminated= TRUE;
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/**
|
|
Adjust the master select of the subquery to be the fake_select which
|
|
represents the whole UNION right above the subquery, instead of the
|
|
last query of the UNION.
|
|
|
|
@param arg pointer to the fake select
|
|
|
|
@return
|
|
FALSE to force the evaluation of the processor for the subsequent items.
|
|
*/
|
|
|
|
bool Item_subselect::set_fake_select_as_master_processor(uchar *arg)
|
|
{
|
|
SELECT_LEX *fake_select= (SELECT_LEX*) arg;
|
|
/*
|
|
Move the st_select_lex_unit of a subquery from a global ORDER BY clause to
|
|
become a direct child of the fake_select of a UNION. In this way the
|
|
ORDER BY that is applied to the temporary table that contains the result of
|
|
the whole UNION, and all columns in the subquery are resolved against this
|
|
table. The transformation is applied only for immediate child subqueries of
|
|
a UNION query.
|
|
*/
|
|
if (unit->outer_select()->master_unit()->fake_select_lex == fake_select)
|
|
{
|
|
/*
|
|
Set the master of the subquery to be the fake select (i.e. the whole
|
|
UNION), instead of the last query in the UNION.
|
|
*/
|
|
fake_select->add_slave(unit);
|
|
DBUG_ASSERT(unit->outer_select() == fake_select);
|
|
/* Adjust the name resolution context hierarchy accordingly. */
|
|
for (SELECT_LEX *sl= unit->first_select(); sl; sl= sl->next_select())
|
|
sl->context.outer_context= &(fake_select->context);
|
|
/*
|
|
Undo Item_subselect::eliminate_subselect_processor because at that phase
|
|
we don't know yet that the ORDER clause will be moved to the fake select.
|
|
*/
|
|
unit->item= this;
|
|
eliminated= FALSE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
bool Item_subselect::mark_as_dependent(THD *thd, st_select_lex *select,
|
|
Item *item)
|
|
{
|
|
if (inside_first_fix_fields)
|
|
{
|
|
is_correlated= TRUE;
|
|
Ref_to_outside *upper;
|
|
if (!(upper= new (thd->stmt_arena->mem_root) Ref_to_outside()))
|
|
return TRUE;
|
|
upper->select= select;
|
|
upper->item= item;
|
|
if (upper_refs.push_back(upper, thd->stmt_arena->mem_root))
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*
|
|
Adjust attributes after our parent select has been merged into grandparent
|
|
|
|
DESCRIPTION
|
|
Subquery is a composite object which may be correlated, that is, it may
|
|
have
|
|
1. references to tables of the parent select (i.e. one that has the clause
|
|
with the subquery predicate)
|
|
2. references to tables of the grandparent select
|
|
3. references to tables of further ancestors.
|
|
|
|
Before the pullout, this item indicates:
|
|
- #1 with table bits in used_tables()
|
|
- #2 and #3 with OUTER_REF_TABLE_BIT.
|
|
|
|
After parent has been merged with grandparent:
|
|
- references to parent and grandparent tables should be indicated with
|
|
table bits.
|
|
- references to greatgrandparent and further ancestors - with
|
|
OUTER_REF_TABLE_BIT.
|
|
*/
|
|
|
|
void Item_subselect::fix_after_pullout(st_select_lex *new_parent, Item **ref)
|
|
{
|
|
recalc_used_tables(new_parent, TRUE);
|
|
parent_select= new_parent;
|
|
}
|
|
|
|
|
|
class Field_fixer: public Field_enumerator
|
|
{
|
|
public:
|
|
table_map used_tables; /* Collect used_tables here */
|
|
st_select_lex *new_parent; /* Select we're in */
|
|
virtual void visit_field(Item_field *item)
|
|
{
|
|
//for (TABLE_LIST *tbl= new_parent->leaf_tables; tbl; tbl= tbl->next_local)
|
|
//{
|
|
// if (tbl->table == field->table)
|
|
// {
|
|
used_tables|= item->field->table->map;
|
|
// return;
|
|
// }
|
|
//}
|
|
//used_tables |= OUTER_REF_TABLE_BIT;
|
|
}
|
|
};
|
|
|
|
|
|
/*
|
|
Recalculate used_tables_cache
|
|
*/
|
|
|
|
void Item_subselect::recalc_used_tables(st_select_lex *new_parent,
|
|
bool after_pullout)
|
|
{
|
|
List_iterator_fast<Ref_to_outside> it(upper_refs);
|
|
Ref_to_outside *upper;
|
|
DBUG_ENTER("recalc_used_tables");
|
|
|
|
used_tables_cache= 0;
|
|
while ((upper= it++))
|
|
{
|
|
bool found= FALSE;
|
|
/*
|
|
Check if
|
|
1. the upper reference refers to the new immediate parent select, or
|
|
2. one of the further ancestors.
|
|
|
|
We rely on the fact that the tree of selects is modified by some kind of
|
|
'flattening', i.e. a process where child selects are merged into their
|
|
parents.
|
|
The merged selects are removed from the select tree but keep pointers to
|
|
their parents.
|
|
*/
|
|
for (st_select_lex *sel= upper->select; sel; sel= sel->outer_select())
|
|
{
|
|
/*
|
|
If we've reached the new parent select by walking upwards from
|
|
reference's original select, this means that the reference is now
|
|
referring to the direct parent:
|
|
*/
|
|
if (sel == new_parent)
|
|
{
|
|
found= TRUE;
|
|
/*
|
|
upper->item may be NULL when we've referred to a grouping function,
|
|
in which case we don't care about what it's table_map really is,
|
|
because item->with_sum_func==1 will ensure correct placement of the
|
|
item.
|
|
*/
|
|
if (upper->item)
|
|
{
|
|
// Now, iterate over fields and collect used_tables() attribute:
|
|
Field_fixer fixer;
|
|
fixer.used_tables= 0;
|
|
fixer.new_parent= new_parent;
|
|
upper->item->walk(&Item::enumerate_field_refs_processor, FALSE,
|
|
(uchar*)&fixer);
|
|
used_tables_cache |= fixer.used_tables;
|
|
upper->item->walk(&Item::update_table_bitmaps_processor, FALSE, NULL);
|
|
/*
|
|
if (after_pullout)
|
|
upper->item->fix_after_pullout(new_parent, &(upper->item));
|
|
upper->item->update_used_tables();
|
|
*/
|
|
}
|
|
}
|
|
}
|
|
if (!found)
|
|
used_tables_cache|= OUTER_REF_TABLE_BIT;
|
|
}
|
|
/*
|
|
Don't update const_tables_cache yet as we don't yet know which of the
|
|
parent's tables are constant. Parent will call update_used_tables() after
|
|
he has done const table detection, and that will be our chance to update
|
|
const_tables_cache.
|
|
*/
|
|
DBUG_PRINT("exit", ("used_tables_cache: %llx", used_tables_cache));
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
/**
|
|
Determine if a subquery is expensive to execute during query optimization.
|
|
|
|
@details The cost of execution of a subquery is estimated based on an
|
|
estimate of the number of rows the subquery will access during execution.
|
|
This measure is used instead of JOIN::read_time, because it is considered
|
|
to be much more reliable than the cost estimate.
|
|
|
|
@return true if the subquery is expensive
|
|
@return false otherwise
|
|
*/
|
|
bool Item_subselect::is_expensive()
|
|
{
|
|
double examined_rows= 0;
|
|
|
|
for (SELECT_LEX *sl= unit->first_select(); sl; sl= sl->next_select())
|
|
{
|
|
JOIN *cur_join= sl->join;
|
|
if (!cur_join)
|
|
continue;
|
|
|
|
/*
|
|
Subqueries whose result is known after optimization are not expensive.
|
|
Such subqueries have all tables optimized away, thus have no join plan.
|
|
*/
|
|
if (cur_join->optimized &&
|
|
(cur_join->zero_result_cause || !cur_join->tables_list))
|
|
return false;
|
|
|
|
/*
|
|
If a subquery is not optimized we cannot estimate its cost. A subquery is
|
|
considered optimized if it has a join plan.
|
|
*/
|
|
if (!(cur_join->optimized && cur_join->join_tab))
|
|
return true;
|
|
|
|
if (sl->first_inner_unit())
|
|
{
|
|
/*
|
|
Subqueries that contain subqueries are considered expensive.
|
|
@todo: accumulate the cost of subqueries.
|
|
*/
|
|
return true;
|
|
}
|
|
|
|
examined_rows+= cur_join->get_examined_rows();
|
|
}
|
|
|
|
return (examined_rows > thd->variables.expensive_subquery_limit);
|
|
}
|
|
|
|
|
|
bool Item_subselect::walk(Item_processor processor, bool walk_subquery,
|
|
uchar *argument)
|
|
{
|
|
if (!(unit->uncacheable & ~UNCACHEABLE_DEPENDENT) && engine->is_executed() &&
|
|
!unit->describe)
|
|
{
|
|
/*
|
|
The subquery has already been executed (for real, it wasn't EXPLAIN's
|
|
fake execution) so it should not matter what it has inside.
|
|
|
|
The actual reason for not walking inside is that parts of the subquery
|
|
(e.g. JTBM join nests and their IN-equality conditions may have been
|
|
invalidated by irreversible cleanups (those happen after an uncorrelated
|
|
subquery has been executed).
|
|
*/
|
|
return (this->*processor)(argument);
|
|
}
|
|
|
|
if (walk_subquery)
|
|
{
|
|
for (SELECT_LEX *lex= unit->first_select(); lex; lex= lex->next_select())
|
|
{
|
|
List_iterator<Item> li(lex->item_list);
|
|
Item *item;
|
|
ORDER *order;
|
|
|
|
if (lex->where && (lex->where)->walk(processor, walk_subquery, argument))
|
|
return 1;
|
|
if (lex->having && (lex->having)->walk(processor, walk_subquery,
|
|
argument))
|
|
return 1;
|
|
/* TODO: why does this walk WHERE/HAVING but not ON expressions of outer joins? */
|
|
|
|
while ((item=li++))
|
|
{
|
|
if (item->walk(processor, walk_subquery, argument))
|
|
return 1;
|
|
}
|
|
for (order= lex->order_list.first ; order; order= order->next)
|
|
{
|
|
if ((*order->item)->walk(processor, walk_subquery, argument))
|
|
return 1;
|
|
}
|
|
for (order= lex->group_list.first ; order; order= order->next)
|
|
{
|
|
if ((*order->item)->walk(processor, walk_subquery, argument))
|
|
return 1;
|
|
}
|
|
}
|
|
}
|
|
return (this->*processor)(argument);
|
|
}
|
|
|
|
|
|
bool Item_subselect::exec()
|
|
{
|
|
subselect_engine *org_engine= engine;
|
|
|
|
DBUG_ENTER("Item_subselect::exec");
|
|
|
|
/*
|
|
Do not execute subselect in case of a fatal error
|
|
or if the query has been killed.
|
|
*/
|
|
if (thd->is_error() || thd->killed)
|
|
DBUG_RETURN(true);
|
|
|
|
DBUG_ASSERT(!thd->lex->context_analysis_only);
|
|
/*
|
|
Simulate a failure in sub-query execution. Used to test e.g.
|
|
out of memory or query being killed conditions.
|
|
*/
|
|
DBUG_EXECUTE_IF("subselect_exec_fail", DBUG_RETURN(true););
|
|
|
|
bool res= engine->exec();
|
|
|
|
#ifndef DBUG_OFF
|
|
++exec_counter;
|
|
#endif
|
|
if (engine != org_engine)
|
|
{
|
|
/*
|
|
If the subquery engine changed during execution due to lazy subquery
|
|
optimization, or because the original engine found a more efficient other
|
|
engine, re-execute the subquery with the new engine.
|
|
*/
|
|
DBUG_RETURN(exec());
|
|
}
|
|
DBUG_RETURN(res);
|
|
}
|
|
|
|
|
|
void Item_subselect::get_cache_parameters(List<Item> ¶meters)
|
|
{
|
|
Collect_deps_prm prm= {¶meters, // parameters
|
|
unit->first_select()->nest_level_base, // nest_level_base
|
|
0, // count
|
|
unit->first_select()->nest_level, // nest_level
|
|
TRUE // collect
|
|
};
|
|
walk(&Item::collect_outer_ref_processor, TRUE, (uchar*)&prm);
|
|
}
|
|
|
|
int Item_in_subselect::optimize(double *out_rows, double *cost)
|
|
{
|
|
int res;
|
|
DBUG_ENTER("Item_in_subselect::optimize");
|
|
SELECT_LEX *save_select= thd->lex->current_select;
|
|
JOIN *join= unit->first_select()->join;
|
|
|
|
thd->lex->current_select= join->select_lex;
|
|
if ((res= join->optimize()))
|
|
DBUG_RETURN(res);
|
|
|
|
/* Calculate #rows and cost of join execution */
|
|
join->get_partial_cost_and_fanout(join->table_count - join->const_tables,
|
|
table_map(-1),
|
|
cost, out_rows);
|
|
|
|
/*
|
|
Adjust join output cardinality. There can be these cases:
|
|
- Have no GROUP BY and no aggregate funcs: we won't get into this
|
|
function because such join will be processed as a merged semi-join
|
|
(TODO: does it really mean we don't need to handle such cases here at
|
|
all? put ASSERT)
|
|
- Have no GROUP BY but have aggregate funcs: output is 1 record.
|
|
- Have GROUP BY and have (or not) aggregate funcs: need to adjust output
|
|
cardinality.
|
|
*/
|
|
thd->lex->current_select= save_select;
|
|
if (!join->group_list && !join->group_optimized_away &&
|
|
join->tmp_table_param.sum_func_count)
|
|
{
|
|
DBUG_PRINT("info",("Materialized join will have only 1 row (it has "
|
|
"aggregates but no GROUP BY"));
|
|
*out_rows= 1;
|
|
}
|
|
|
|
/* Now with grouping */
|
|
if (join->group_list_for_estimates)
|
|
{
|
|
DBUG_PRINT("info",("Materialized join has grouping, trying to estimate it"));
|
|
double output_rows= get_post_group_estimate(join, *out_rows);
|
|
DBUG_PRINT("info",("Got value of %g", output_rows));
|
|
*out_rows= output_rows;
|
|
}
|
|
|
|
DBUG_RETURN(res);
|
|
|
|
}
|
|
|
|
|
|
/**
|
|
Check if an expression cache is needed for this subquery
|
|
|
|
@param thd Thread handle
|
|
|
|
@details
|
|
The function checks whether a cache is needed for a subquery and whether
|
|
the result of the subquery can be put in cache.
|
|
|
|
@retval TRUE cache is needed
|
|
@retval FALSE otherwise
|
|
*/
|
|
|
|
bool Item_subselect::expr_cache_is_needed(THD *thd)
|
|
{
|
|
return ((engine->uncacheable() & UNCACHEABLE_DEPENDENT) &&
|
|
engine->cols() == 1 &&
|
|
optimizer_flag(thd, OPTIMIZER_SWITCH_SUBQUERY_CACHE) &&
|
|
!(engine->uncacheable() & (UNCACHEABLE_RAND |
|
|
UNCACHEABLE_SIDEEFFECT)));
|
|
}
|
|
|
|
|
|
/**
|
|
Check if the left IN argument contains NULL values.
|
|
|
|
@retval TRUE there are NULLs
|
|
@retval FALSE otherwise
|
|
*/
|
|
|
|
inline bool Item_in_subselect::left_expr_has_null()
|
|
{
|
|
return (*(optimizer->get_cache()))->null_value;
|
|
}
|
|
|
|
|
|
/**
|
|
Check if an expression cache is needed for this subquery
|
|
|
|
@param thd Thread handle
|
|
|
|
@details
|
|
The function checks whether a cache is needed for a subquery and whether
|
|
the result of the subquery can be put in cache.
|
|
|
|
@note
|
|
This method allows many columns in the subquery because it is supported by
|
|
Item_in_optimizer and result of the IN subquery will be scalar in this
|
|
case.
|
|
|
|
@retval TRUE cache is needed
|
|
@retval FALSE otherwise
|
|
*/
|
|
|
|
bool Item_in_subselect::expr_cache_is_needed(THD *thd)
|
|
{
|
|
return (optimizer_flag(thd, OPTIMIZER_SWITCH_SUBQUERY_CACHE) &&
|
|
!(engine->uncacheable() & (UNCACHEABLE_RAND |
|
|
UNCACHEABLE_SIDEEFFECT)));
|
|
}
|
|
|
|
|
|
/*
|
|
Compute the IN predicate if the left operand's cache changed.
|
|
*/
|
|
|
|
bool Item_in_subselect::exec()
|
|
{
|
|
DBUG_ENTER("Item_in_subselect::exec");
|
|
/*
|
|
Initialize the cache of the left predicate operand. This has to be done as
|
|
late as now, because Cached_item directly contains a resolved field (not
|
|
an item, and in some cases (when temp tables are created), these fields
|
|
end up pointing to the wrong field. One solution is to change Cached_item
|
|
to not resolve its field upon creation, but to resolve it dynamically
|
|
from a given Item_ref object.
|
|
TODO: the cache should be applied conditionally based on:
|
|
- rules - e.g. only if the left operand is known to be ordered, and/or
|
|
- on a cost-based basis, that takes into account the cost of a cache
|
|
lookup, the cache hit rate, and the savings per cache hit.
|
|
*/
|
|
if (!left_expr_cache && (test_strategy(SUBS_MATERIALIZATION)))
|
|
init_left_expr_cache();
|
|
|
|
/*
|
|
If the new left operand is already in the cache, reuse the old result.
|
|
Use the cached result only if this is not the first execution of IN
|
|
because the cache is not valid for the first execution.
|
|
*/
|
|
if (!first_execution && left_expr_cache &&
|
|
test_if_item_cache_changed(*left_expr_cache) < 0)
|
|
DBUG_RETURN(FALSE);
|
|
|
|
/*
|
|
The exec() method below updates item::value, and item::null_value, thus if
|
|
we don't call it, the next call to item::val_int() will return whatever
|
|
result was computed by its previous call.
|
|
*/
|
|
DBUG_RETURN(Item_subselect::exec());
|
|
}
|
|
|
|
|
|
Item::Type Item_subselect::type() const
|
|
{
|
|
return SUBSELECT_ITEM;
|
|
}
|
|
|
|
|
|
void Item_subselect::fix_length_and_dec()
|
|
{
|
|
engine->fix_length_and_dec(0);
|
|
}
|
|
|
|
|
|
table_map Item_subselect::used_tables() const
|
|
{
|
|
return (table_map) ((engine->uncacheable() & ~UNCACHEABLE_EXPLAIN)?
|
|
used_tables_cache : 0L);
|
|
}
|
|
|
|
|
|
bool Item_subselect::const_item() const
|
|
{
|
|
return (thd->lex->context_analysis_only ?
|
|
FALSE :
|
|
forced_const || const_item_cache);
|
|
}
|
|
|
|
Item *Item_subselect::get_tmp_table_item(THD *thd_arg)
|
|
{
|
|
if (!with_sum_func && !const_item())
|
|
return new (thd->mem_root) Item_temptable_field(thd_arg, result_field);
|
|
return copy_or_same(thd_arg);
|
|
}
|
|
|
|
void Item_subselect::update_used_tables()
|
|
{
|
|
if (!forced_const)
|
|
{
|
|
recalc_used_tables(parent_select, FALSE);
|
|
if (!engine->uncacheable())
|
|
{
|
|
// did all used tables become static?
|
|
if (!(used_tables_cache & ~engine->upper_select_const_tables()))
|
|
const_item_cache= 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void Item_subselect::print(String *str, enum_query_type query_type)
|
|
{
|
|
if (query_type & QT_ITEM_SUBSELECT_ID_ONLY)
|
|
{
|
|
str->append("(subquery#");
|
|
if (unit && unit->first_select())
|
|
{
|
|
char buf[64];
|
|
ll2str(unit->first_select()->select_number, buf, 10, 0);
|
|
str->append(buf);
|
|
}
|
|
else
|
|
str->append("NULL"); // TODO: what exactly does this mean?
|
|
|
|
str->append(")");
|
|
return;
|
|
}
|
|
if (engine)
|
|
{
|
|
str->append('(');
|
|
engine->print(str, query_type);
|
|
str->append(')');
|
|
}
|
|
else
|
|
str->append("(...)");
|
|
}
|
|
|
|
|
|
Item_singlerow_subselect::Item_singlerow_subselect(THD *thd, st_select_lex *select_lex):
|
|
Item_subselect(thd), value(0)
|
|
{
|
|
DBUG_ENTER("Item_singlerow_subselect::Item_singlerow_subselect");
|
|
init(select_lex, new (thd->mem_root) select_singlerow_subselect(thd, this));
|
|
maybe_null= 1;
|
|
max_columns= UINT_MAX;
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
st_select_lex *
|
|
Item_singlerow_subselect::invalidate_and_restore_select_lex()
|
|
{
|
|
DBUG_ENTER("Item_singlerow_subselect::invalidate_and_restore_select_lex");
|
|
st_select_lex *result= get_select_lex();
|
|
|
|
DBUG_ASSERT(result);
|
|
|
|
/*
|
|
This code restore the parse tree in it's state before the execution of
|
|
Item_singlerow_subselect::Item_singlerow_subselect(),
|
|
and in particular decouples this object from the SELECT_LEX,
|
|
so that the SELECT_LEX can be used with a different flavor
|
|
or Item_subselect instead, as part of query rewriting.
|
|
*/
|
|
unit->item= NULL;
|
|
|
|
DBUG_RETURN(result);
|
|
}
|
|
|
|
Item_maxmin_subselect::Item_maxmin_subselect(THD *thd,
|
|
Item_subselect *parent,
|
|
st_select_lex *select_lex,
|
|
bool max_arg):
|
|
Item_singlerow_subselect(thd), was_values(TRUE)
|
|
{
|
|
DBUG_ENTER("Item_maxmin_subselect::Item_maxmin_subselect");
|
|
max= max_arg;
|
|
init(select_lex,
|
|
new (thd->mem_root) select_max_min_finder_subselect(thd,
|
|
this, max_arg, parent->substype() == Item_subselect::ALL_SUBS));
|
|
max_columns= 1;
|
|
maybe_null= 1;
|
|
max_columns= 1;
|
|
|
|
/*
|
|
Following information was collected during performing fix_fields()
|
|
of Items belonged to subquery, which will be not repeated
|
|
*/
|
|
used_tables_cache= parent->get_used_tables_cache();
|
|
const_item_cache= parent->const_item();
|
|
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
void Item_maxmin_subselect::cleanup()
|
|
{
|
|
DBUG_ENTER("Item_maxmin_subselect::cleanup");
|
|
Item_singlerow_subselect::cleanup();
|
|
|
|
/*
|
|
By default it is TRUE to avoid TRUE reporting by
|
|
Item_func_not_all/Item_func_nop_all if this item was never called.
|
|
|
|
Engine exec() set it to FALSE by reset_value_registration() call.
|
|
select_max_min_finder_subselect::send_data() set it back to TRUE if some
|
|
value will be found.
|
|
*/
|
|
was_values= TRUE;
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
void Item_maxmin_subselect::print(String *str, enum_query_type query_type)
|
|
{
|
|
str->append(max?"<max>":"<min>", 5);
|
|
Item_singlerow_subselect::print(str, query_type);
|
|
}
|
|
|
|
|
|
void Item_maxmin_subselect::no_rows_in_result()
|
|
{
|
|
/*
|
|
Subquery predicates outside of the SELECT list must be evaluated in order
|
|
to possibly filter the special result row generated for implicit grouping
|
|
if the subquery is in the HAVING clause.
|
|
If the predicate is constant, we need its actual value in the only result
|
|
row for queries with implicit grouping.
|
|
*/
|
|
if (parsing_place != SELECT_LIST || const_item())
|
|
return;
|
|
value= Item_cache::get_cache(thd, new (thd->mem_root) Item_null(thd));
|
|
null_value= 0;
|
|
was_values= 0;
|
|
make_const();
|
|
}
|
|
|
|
|
|
void Item_singlerow_subselect::no_rows_in_result()
|
|
{
|
|
/*
|
|
Subquery predicates outside of the SELECT list must be evaluated in order
|
|
to possibly filter the special result row generated for implicit grouping
|
|
if the subquery is in the HAVING clause.
|
|
If the predicate is constant, we need its actual value in the only result
|
|
row for queries with implicit grouping.
|
|
*/
|
|
if (parsing_place != SELECT_LIST || const_item())
|
|
return;
|
|
value= Item_cache::get_cache(thd, new (thd->mem_root) Item_null(thd));
|
|
reset();
|
|
make_const();
|
|
}
|
|
|
|
|
|
void Item_singlerow_subselect::reset()
|
|
{
|
|
Item_subselect::reset();
|
|
if (value)
|
|
{
|
|
for(uint i= 0; i < engine->cols(); i++)
|
|
row[i]->set_null();
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
@todo
|
|
- We cant change name of Item_field or Item_ref, because it will
|
|
prevent it's correct resolving, but we should save name of
|
|
removed item => we do not make optimization if top item of
|
|
list is field or reference.
|
|
- switch off this optimization for prepare statement,
|
|
because we do not rollback this changes.
|
|
Make rollback for it, or special name resolving mode in 5.0.
|
|
|
|
@param join Join object of the subquery (i.e. 'child' join).
|
|
|
|
@retval false The subquery was transformed
|
|
*/
|
|
bool
|
|
Item_singlerow_subselect::select_transformer(JOIN *join)
|
|
{
|
|
DBUG_ENTER("Item_singlerow_subselect::select_transformer");
|
|
if (changed)
|
|
DBUG_RETURN(false);
|
|
|
|
SELECT_LEX *select_lex= join->select_lex;
|
|
Query_arena *arena= thd->stmt_arena;
|
|
|
|
if (!select_lex->master_unit()->is_union() &&
|
|
!select_lex->table_list.elements &&
|
|
select_lex->item_list.elements == 1 &&
|
|
!select_lex->item_list.head()->with_sum_func &&
|
|
/*
|
|
We cant change name of Item_field or Item_ref, because it will
|
|
prevent it's correct resolving, but we should save name of
|
|
removed item => we do not make optimization if top item of
|
|
list is field or reference.
|
|
TODO: solve above problem
|
|
*/
|
|
!(select_lex->item_list.head()->type() == FIELD_ITEM ||
|
|
select_lex->item_list.head()->type() == REF_ITEM) &&
|
|
!join->conds && !join->having &&
|
|
/*
|
|
switch off this optimization for prepare statement,
|
|
because we do not rollback this changes
|
|
TODO: make rollback for it, or special name resolving mode in 5.0.
|
|
*/
|
|
!arena->is_stmt_prepare_or_first_sp_execute()
|
|
)
|
|
{
|
|
have_to_be_excluded= 1;
|
|
if (thd->lex->describe)
|
|
{
|
|
char warn_buff[MYSQL_ERRMSG_SIZE];
|
|
sprintf(warn_buff, ER_THD(thd, ER_SELECT_REDUCED),
|
|
select_lex->select_number);
|
|
push_warning(thd, Sql_condition::WARN_LEVEL_NOTE,
|
|
ER_SELECT_REDUCED, warn_buff);
|
|
}
|
|
substitution= select_lex->item_list.head();
|
|
/*
|
|
as far as we moved content to upper level we have to fix dependences & Co
|
|
*/
|
|
substitution->fix_after_pullout(select_lex->outer_select(), &substitution);
|
|
}
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
|
|
void Item_singlerow_subselect::store(uint i, Item *item)
|
|
{
|
|
row[i]->store(item);
|
|
row[i]->cache_value();
|
|
}
|
|
|
|
enum Item_result Item_singlerow_subselect::result_type() const
|
|
{
|
|
return engine->type();
|
|
}
|
|
|
|
enum Item_result Item_singlerow_subselect::cmp_type() const
|
|
{
|
|
return engine->cmptype();
|
|
}
|
|
|
|
/*
|
|
Don't rely on the result type to calculate field type.
|
|
Ask the engine instead.
|
|
*/
|
|
enum_field_types Item_singlerow_subselect::field_type() const
|
|
{
|
|
return engine->field_type();
|
|
}
|
|
|
|
void Item_singlerow_subselect::fix_length_and_dec()
|
|
{
|
|
if ((max_columns= engine->cols()) == 1)
|
|
{
|
|
engine->fix_length_and_dec(row= &value);
|
|
}
|
|
else
|
|
{
|
|
if (!(row= (Item_cache**) current_thd->alloc(sizeof(Item_cache*) *
|
|
max_columns)))
|
|
return;
|
|
engine->fix_length_and_dec(row);
|
|
value= *row;
|
|
}
|
|
unsigned_flag= value->unsigned_flag;
|
|
/*
|
|
If there are not tables in subquery then ability to have NULL value
|
|
depends on SELECT list (if single row subquery have tables then it
|
|
always can be NULL if there are not records fetched).
|
|
*/
|
|
if (engine->no_tables())
|
|
maybe_null= engine->may_be_null();
|
|
else
|
|
{
|
|
for (uint i= 0; i < max_columns; i++)
|
|
row[i]->maybe_null= TRUE;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
Add an expression cache for this subquery if it is needed
|
|
|
|
@param thd_arg Thread handle
|
|
|
|
@details
|
|
The function checks whether an expression cache is needed for this item
|
|
and if if so wraps the item into an item of the class
|
|
Item_cache_wrapper with an appropriate expression cache set up there.
|
|
|
|
@note
|
|
used from Item::transform()
|
|
|
|
@return
|
|
new wrapper item if an expression cache is needed,
|
|
this item - otherwise
|
|
*/
|
|
|
|
Item* Item_singlerow_subselect::expr_cache_insert_transformer(THD *tmp_thd,
|
|
uchar *unused)
|
|
{
|
|
DBUG_ENTER("Item_singlerow_subselect::expr_cache_insert_transformer");
|
|
|
|
DBUG_ASSERT(thd == tmp_thd);
|
|
|
|
if (expr_cache)
|
|
DBUG_RETURN(expr_cache);
|
|
|
|
if (expr_cache_is_needed(tmp_thd) &&
|
|
(expr_cache= set_expr_cache(tmp_thd)))
|
|
{
|
|
init_expr_cache_tracker(tmp_thd);
|
|
DBUG_RETURN(expr_cache);
|
|
}
|
|
DBUG_RETURN(this);
|
|
}
|
|
|
|
|
|
uint Item_singlerow_subselect::cols()
|
|
{
|
|
return engine->cols();
|
|
}
|
|
|
|
bool Item_singlerow_subselect::check_cols(uint c)
|
|
{
|
|
if (c != engine->cols())
|
|
{
|
|
my_error(ER_OPERAND_COLUMNS, MYF(0), c);
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
bool Item_singlerow_subselect::null_inside()
|
|
{
|
|
for (uint i= 0; i < max_columns ; i++)
|
|
{
|
|
if (row[i]->null_value)
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void Item_singlerow_subselect::bring_value()
|
|
{
|
|
if (!exec() && assigned())
|
|
null_value= 0;
|
|
else
|
|
reset();
|
|
}
|
|
|
|
double Item_singlerow_subselect::val_real()
|
|
{
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (forced_const)
|
|
return value->val_real();
|
|
if (!exec() && !value->null_value)
|
|
{
|
|
null_value= FALSE;
|
|
return value->val_real();
|
|
}
|
|
else
|
|
{
|
|
reset();
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
longlong Item_singlerow_subselect::val_int()
|
|
{
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (forced_const)
|
|
return value->val_int();
|
|
if (!exec() && !value->null_value)
|
|
{
|
|
null_value= FALSE;
|
|
return value->val_int();
|
|
}
|
|
else
|
|
{
|
|
reset();
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
String *Item_singlerow_subselect::val_str(String *str)
|
|
{
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (forced_const)
|
|
return value->val_str(str);
|
|
if (!exec() && !value->null_value)
|
|
{
|
|
null_value= FALSE;
|
|
return value->val_str(str);
|
|
}
|
|
else
|
|
{
|
|
reset();
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
my_decimal *Item_singlerow_subselect::val_decimal(my_decimal *decimal_value)
|
|
{
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (forced_const)
|
|
return value->val_decimal(decimal_value);
|
|
if (!exec() && !value->null_value)
|
|
{
|
|
null_value= FALSE;
|
|
return value->val_decimal(decimal_value);
|
|
}
|
|
else
|
|
{
|
|
reset();
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
bool Item_singlerow_subselect::val_bool()
|
|
{
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (forced_const)
|
|
return value->val_bool();
|
|
if (!exec() && !value->null_value)
|
|
{
|
|
null_value= FALSE;
|
|
return value->val_bool();
|
|
}
|
|
else
|
|
{
|
|
reset();
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
bool Item_singlerow_subselect::get_date(MYSQL_TIME *ltime,ulonglong fuzzydate)
|
|
{
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (forced_const)
|
|
return value->get_date(ltime, fuzzydate);
|
|
if (!exec() && !value->null_value)
|
|
{
|
|
null_value= FALSE;
|
|
return value->get_date(ltime, fuzzydate);
|
|
}
|
|
else
|
|
{
|
|
reset();
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
|
|
Item_exists_subselect::Item_exists_subselect(THD *thd,
|
|
st_select_lex *select_lex):
|
|
Item_subselect(thd), upper_not(NULL), abort_on_null(0),
|
|
emb_on_expr_nest(NULL), optimizer(0), exists_transformed(0)
|
|
{
|
|
DBUG_ENTER("Item_exists_subselect::Item_exists_subselect");
|
|
init(select_lex, new (thd->mem_root) select_exists_subselect(thd, this));
|
|
max_columns= UINT_MAX;
|
|
null_value= FALSE; //can't be NULL
|
|
maybe_null= 0; //can't be NULL
|
|
value= 0;
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
void Item_exists_subselect::print(String *str, enum_query_type query_type)
|
|
{
|
|
str->append(STRING_WITH_LEN("exists"));
|
|
Item_subselect::print(str, query_type);
|
|
}
|
|
|
|
|
|
bool Item_in_subselect::test_limit(st_select_lex_unit *unit_arg)
|
|
{
|
|
if (unit_arg->fake_select_lex &&
|
|
unit_arg->fake_select_lex->test_limit())
|
|
return(1);
|
|
|
|
SELECT_LEX *sl= unit_arg->first_select();
|
|
for (; sl; sl= sl->next_select())
|
|
{
|
|
if (sl->test_limit())
|
|
return(1);
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
Item_in_subselect::Item_in_subselect(THD *thd, Item * left_exp,
|
|
st_select_lex *select_lex):
|
|
Item_exists_subselect(thd), left_expr_cache(0), first_execution(TRUE),
|
|
in_strategy(SUBS_NOT_TRANSFORMED),
|
|
pushed_cond_guards(NULL), is_jtbm_merged(FALSE), is_jtbm_const_tab(FALSE),
|
|
is_flattenable_semijoin(FALSE), is_registered_semijoin(FALSE),
|
|
upper_item(0)
|
|
{
|
|
DBUG_ENTER("Item_in_subselect::Item_in_subselect");
|
|
DBUG_PRINT("info", ("in_strategy: %u", (uint)in_strategy));
|
|
left_expr_orig= left_expr= left_exp;
|
|
func= &eq_creator;
|
|
init(select_lex, new (thd->mem_root) select_exists_subselect(thd, this));
|
|
max_columns= UINT_MAX;
|
|
maybe_null= 1;
|
|
reset();
|
|
//if test_limit will fail then error will be reported to client
|
|
test_limit(select_lex->master_unit());
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
int Item_in_subselect::get_identifier()
|
|
{
|
|
return engine->get_identifier();
|
|
}
|
|
|
|
Item_allany_subselect::Item_allany_subselect(THD *thd, Item * left_exp,
|
|
chooser_compare_func_creator fc,
|
|
st_select_lex *select_lex,
|
|
bool all_arg):
|
|
Item_in_subselect(thd), func_creator(fc), all(all_arg)
|
|
{
|
|
DBUG_ENTER("Item_allany_subselect::Item_allany_subselect");
|
|
left_expr_orig= left_expr= left_exp;
|
|
func= func_creator(all_arg);
|
|
init(select_lex, new (thd->mem_root) select_exists_subselect(thd, this));
|
|
max_columns= 1;
|
|
abort_on_null= 0;
|
|
reset();
|
|
//if test_limit will fail then error will be reported to client
|
|
test_limit(select_lex->master_unit());
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
/**
|
|
Initialize length and decimals for EXISTS and inherited (IN/ALL/ANY)
|
|
subqueries
|
|
*/
|
|
|
|
void Item_exists_subselect::init_length_and_dec()
|
|
{
|
|
decimals= 0;
|
|
max_length= 1;
|
|
max_columns= engine->cols();
|
|
}
|
|
|
|
|
|
void Item_exists_subselect::fix_length_and_dec()
|
|
{
|
|
DBUG_ENTER("Item_exists_subselect::fix_length_and_dec");
|
|
init_length_and_dec();
|
|
/*
|
|
We need only 1 row to determine existence (i.e. any EXISTS that is not
|
|
an IN always requires LIMIT 1)
|
|
*/
|
|
thd->change_item_tree(&unit->global_parameters()->select_limit,
|
|
new (thd->mem_root) Item_int(thd, (int32) 1));
|
|
DBUG_PRINT("info", ("Set limit to 1"));
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
void Item_in_subselect::fix_length_and_dec()
|
|
{
|
|
DBUG_ENTER("Item_in_subselect::fix_length_and_dec");
|
|
init_length_and_dec();
|
|
/*
|
|
Unlike Item_exists_subselect, LIMIT 1 is set later for
|
|
Item_in_subselect, depending on the chosen strategy.
|
|
*/
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
/**
|
|
Add an expression cache for this subquery if it is needed
|
|
|
|
@param thd_arg Thread handle
|
|
|
|
@details
|
|
The function checks whether an expression cache is needed for this item
|
|
and if if so wraps the item into an item of the class
|
|
Item_cache_wrapper with an appropriate expression cache set up there.
|
|
|
|
@note
|
|
used from Item::transform()
|
|
|
|
@return
|
|
new wrapper item if an expression cache is needed,
|
|
this item - otherwise
|
|
*/
|
|
|
|
Item* Item_exists_subselect::expr_cache_insert_transformer(THD *tmp_thd,
|
|
uchar *unused)
|
|
{
|
|
DBUG_ENTER("Item_exists_subselect::expr_cache_insert_transformer");
|
|
DBUG_ASSERT(thd == tmp_thd);
|
|
|
|
if (expr_cache)
|
|
DBUG_RETURN(expr_cache);
|
|
|
|
if (substype() == EXISTS_SUBS && expr_cache_is_needed(tmp_thd) &&
|
|
(expr_cache= set_expr_cache(tmp_thd)))
|
|
{
|
|
init_expr_cache_tracker(tmp_thd);
|
|
DBUG_RETURN(expr_cache);
|
|
}
|
|
DBUG_RETURN(this);
|
|
}
|
|
|
|
|
|
void Item_exists_subselect::no_rows_in_result()
|
|
{
|
|
/*
|
|
Subquery predicates outside of the SELECT list must be evaluated in order
|
|
to possibly filter the special result row generated for implicit grouping
|
|
if the subquery is in the HAVING clause.
|
|
If the predicate is constant, we need its actual value in the only result
|
|
row for queries with implicit grouping.
|
|
*/
|
|
if (parsing_place != SELECT_LIST || const_item())
|
|
return;
|
|
value= 0;
|
|
null_value= 0;
|
|
make_const();
|
|
}
|
|
|
|
double Item_exists_subselect::val_real()
|
|
{
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (!forced_const && exec())
|
|
{
|
|
reset();
|
|
return 0;
|
|
}
|
|
return (double) value;
|
|
}
|
|
|
|
longlong Item_exists_subselect::val_int()
|
|
{
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (!forced_const && exec())
|
|
{
|
|
reset();
|
|
return 0;
|
|
}
|
|
return value;
|
|
}
|
|
|
|
|
|
/**
|
|
Return the result of EXISTS as a string value
|
|
|
|
Converts the true/false result into a string value.
|
|
Note that currently this cannot be NULL, so if the query exection fails
|
|
it will return 0.
|
|
|
|
@param decimal_value[out] buffer to hold the resulting string value
|
|
@retval Pointer to the converted string.
|
|
Can't be a NULL pointer, as currently
|
|
EXISTS cannot return NULL.
|
|
*/
|
|
|
|
String *Item_exists_subselect::val_str(String *str)
|
|
{
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (!forced_const && exec())
|
|
reset();
|
|
str->set((ulonglong)value,&my_charset_bin);
|
|
return str;
|
|
}
|
|
|
|
|
|
/**
|
|
Return the result of EXISTS as a decimal value
|
|
|
|
Converts the true/false result into a decimal value.
|
|
Note that currently this cannot be NULL, so if the query exection fails
|
|
it will return 0.
|
|
|
|
@param decimal_value[out] Buffer to hold the resulting decimal value
|
|
@retval Pointer to the converted decimal.
|
|
Can't be a NULL pointer, as currently
|
|
EXISTS cannot return NULL.
|
|
*/
|
|
|
|
my_decimal *Item_exists_subselect::val_decimal(my_decimal *decimal_value)
|
|
{
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (!forced_const && exec())
|
|
reset();
|
|
int2my_decimal(E_DEC_FATAL_ERROR, value, 0, decimal_value);
|
|
return decimal_value;
|
|
}
|
|
|
|
|
|
bool Item_exists_subselect::val_bool()
|
|
{
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (!forced_const && exec())
|
|
{
|
|
reset();
|
|
return 0;
|
|
}
|
|
return value != 0;
|
|
}
|
|
|
|
|
|
double Item_in_subselect::val_real()
|
|
{
|
|
/*
|
|
As far as Item_in_subselect called only from Item_in_optimizer this
|
|
method should not be used
|
|
*/
|
|
DBUG_ASSERT(0);
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (forced_const)
|
|
return value;
|
|
DBUG_ASSERT((engine->uncacheable() & ~UNCACHEABLE_EXPLAIN) ||
|
|
! engine->is_executed());
|
|
null_value= was_null= FALSE;
|
|
if (exec())
|
|
{
|
|
reset();
|
|
return 0;
|
|
}
|
|
if (was_null && !value)
|
|
null_value= TRUE;
|
|
return (double) value;
|
|
}
|
|
|
|
|
|
longlong Item_in_subselect::val_int()
|
|
{
|
|
/*
|
|
As far as Item_in_subselect called only from Item_in_optimizer this
|
|
method should not be used
|
|
*/
|
|
DBUG_ASSERT(0);
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (forced_const)
|
|
return value;
|
|
DBUG_ASSERT((engine->uncacheable() & ~UNCACHEABLE_EXPLAIN) ||
|
|
! engine->is_executed());
|
|
null_value= was_null= FALSE;
|
|
if (exec())
|
|
{
|
|
reset();
|
|
return 0;
|
|
}
|
|
if (was_null && !value)
|
|
null_value= TRUE;
|
|
return value;
|
|
}
|
|
|
|
|
|
String *Item_in_subselect::val_str(String *str)
|
|
{
|
|
/*
|
|
As far as Item_in_subselect called only from Item_in_optimizer this
|
|
method should not be used
|
|
*/
|
|
DBUG_ASSERT(0);
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (forced_const)
|
|
goto value_is_ready;
|
|
DBUG_ASSERT((engine->uncacheable() & ~UNCACHEABLE_EXPLAIN) ||
|
|
! engine->is_executed());
|
|
null_value= was_null= FALSE;
|
|
if (exec())
|
|
{
|
|
reset();
|
|
return 0;
|
|
}
|
|
if (was_null && !value)
|
|
{
|
|
null_value= TRUE;
|
|
return 0;
|
|
}
|
|
value_is_ready:
|
|
str->set((ulonglong)value, &my_charset_bin);
|
|
return str;
|
|
}
|
|
|
|
|
|
bool Item_in_subselect::val_bool()
|
|
{
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (forced_const)
|
|
return value;
|
|
DBUG_ASSERT((engine->uncacheable() & ~UNCACHEABLE_EXPLAIN) ||
|
|
! engine->is_executed());
|
|
null_value= was_null= FALSE;
|
|
if (exec())
|
|
{
|
|
reset();
|
|
return 0;
|
|
}
|
|
if (was_null && !value)
|
|
null_value= TRUE;
|
|
return value;
|
|
}
|
|
|
|
my_decimal *Item_in_subselect::val_decimal(my_decimal *decimal_value)
|
|
{
|
|
/*
|
|
As far as Item_in_subselect called only from Item_in_optimizer this
|
|
method should not be used
|
|
*/
|
|
DBUG_ASSERT(0);
|
|
if (forced_const)
|
|
goto value_is_ready;
|
|
DBUG_ASSERT((engine->uncacheable() & ~UNCACHEABLE_EXPLAIN) ||
|
|
! engine->is_executed());
|
|
null_value= was_null= FALSE;
|
|
DBUG_ASSERT(fixed == 1);
|
|
if (exec())
|
|
{
|
|
reset();
|
|
return 0;
|
|
}
|
|
if (was_null && !value)
|
|
null_value= TRUE;
|
|
value_is_ready:
|
|
int2my_decimal(E_DEC_FATAL_ERROR, value, 0, decimal_value);
|
|
return decimal_value;
|
|
}
|
|
|
|
|
|
/**
|
|
Prepare a single-column IN/ALL/ANY subselect for rewriting.
|
|
|
|
@param join Join object of the subquery (i.e. 'child' join).
|
|
|
|
@details
|
|
|
|
Prepare a single-column subquery to be rewritten. Given the subquery.
|
|
|
|
If the subquery has no tables it will be turned to an expression between
|
|
left part and SELECT list.
|
|
|
|
In other cases the subquery will be wrapped with Item_in_optimizer which
|
|
allow later to turn it to EXISTS or MAX/MIN.
|
|
|
|
@retval false The subquery was transformed
|
|
@retval true Error
|
|
*/
|
|
|
|
bool
|
|
Item_in_subselect::single_value_transformer(JOIN *join)
|
|
{
|
|
SELECT_LEX *select_lex= join->select_lex;
|
|
DBUG_ENTER("Item_in_subselect::single_value_transformer");
|
|
|
|
/*
|
|
Check that the right part of the subselect contains no more than one
|
|
column. E.g. in SELECT 1 IN (SELECT * ..) the right part is (SELECT * ...)
|
|
*/
|
|
// psergey: duplicated_subselect_card_check
|
|
if (select_lex->item_list.elements > 1)
|
|
{
|
|
my_error(ER_OPERAND_COLUMNS, MYF(0), 1);
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
Item* join_having= join->having ? join->having : join->tmp_having;
|
|
if (!(join_having || select_lex->with_sum_func ||
|
|
select_lex->group_list.elements) &&
|
|
select_lex->table_list.elements == 0 &&
|
|
!select_lex->master_unit()->is_union())
|
|
{
|
|
Item *where_item= (Item*) select_lex->item_list.head();
|
|
/*
|
|
it is single select without tables => possible optimization
|
|
remove the dependence mark since the item is moved to upper
|
|
select and is not outer anymore.
|
|
*/
|
|
where_item->walk(&Item::remove_dependence_processor, 0,
|
|
(uchar *) select_lex->outer_select());
|
|
/*
|
|
fix_field of substitution item will be done in time of
|
|
substituting.
|
|
Note that real_item() should be used instead of
|
|
original left expression because left_expr can be
|
|
runtime created Ref item which is deleted at the end
|
|
of the statement. Thus one of 'substitution' arguments
|
|
can be broken in case of PS.
|
|
*/
|
|
substitution= func->create(thd, left_expr, where_item);
|
|
have_to_be_excluded= 1;
|
|
if (thd->lex->describe)
|
|
{
|
|
char warn_buff[MYSQL_ERRMSG_SIZE];
|
|
sprintf(warn_buff, ER_THD(thd, ER_SELECT_REDUCED),
|
|
select_lex->select_number);
|
|
push_warning(thd, Sql_condition::WARN_LEVEL_NOTE,
|
|
ER_SELECT_REDUCED, warn_buff);
|
|
}
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
/*
|
|
Wrap the current IN predicate in an Item_in_optimizer. The actual
|
|
substitution in the Item tree takes place in Item_subselect::fix_fields.
|
|
*/
|
|
if (!substitution)
|
|
{
|
|
/* We're invoked for the 1st (or the only) SELECT in the subquery UNION */
|
|
substitution= optimizer;
|
|
|
|
SELECT_LEX *current= thd->lex->current_select;
|
|
|
|
thd->lex->current_select= current->return_after_parsing();
|
|
if (!optimizer || optimizer->fix_left(thd))
|
|
{
|
|
thd->lex->current_select= current;
|
|
DBUG_RETURN(true);
|
|
}
|
|
thd->lex->current_select= current;
|
|
|
|
/* We will refer to upper level cache array => we have to save it for SP */
|
|
optimizer->keep_top_level_cache();
|
|
|
|
/*
|
|
As far as Item_in_optimizer does not substitute itself on fix_fields
|
|
we can use same item for all selects.
|
|
*/
|
|
expr= new (thd->mem_root) Item_direct_ref(thd, &select_lex->context,
|
|
(Item**)optimizer->get_cache(),
|
|
(char *)"<no matter>",
|
|
(char *)in_left_expr_name);
|
|
}
|
|
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
|
|
/**
|
|
Apply transformation max/min transwormation to ALL/ANY subquery if it is
|
|
possible.
|
|
|
|
@param join Join object of the subquery (i.e. 'child' join).
|
|
|
|
@details
|
|
|
|
If this is an ALL/ANY single-value subselect, try to rewrite it with
|
|
a MIN/MAX subselect. We can do that if a possible NULL result of the
|
|
subselect can be ignored.
|
|
E.g. SELECT * FROM t1 WHERE b > ANY (SELECT a FROM t2) can be rewritten
|
|
with SELECT * FROM t1 WHERE b > (SELECT MAX(a) FROM t2).
|
|
We can't check that this optimization is safe if it's not a top-level
|
|
item of the WHERE clause (e.g. because the WHERE clause can contain IS
|
|
NULL/IS NOT NULL functions). If so, we rewrite ALL/ANY with NOT EXISTS
|
|
later in this method.
|
|
|
|
@retval false The subquery was transformed
|
|
@retval true Error
|
|
*/
|
|
|
|
bool Item_allany_subselect::transform_into_max_min(JOIN *join)
|
|
{
|
|
DBUG_ENTER("Item_allany_subselect::transform_into_max_min");
|
|
if (!test_strategy(SUBS_MAXMIN_INJECTED | SUBS_MAXMIN_ENGINE))
|
|
DBUG_RETURN(false);
|
|
Item **place= optimizer->arguments() + 1;
|
|
SELECT_LEX *select_lex= join->select_lex;
|
|
Item *subs;
|
|
|
|
/*
|
|
*/
|
|
DBUG_ASSERT(!substitution);
|
|
|
|
/*
|
|
Check if optimization with aggregate min/max possible
|
|
1 There is no aggregate in the subquery
|
|
2 It is not UNION
|
|
3 There is tables
|
|
4 It is not ALL subquery with possible NULLs in the SELECT list
|
|
*/
|
|
if (!select_lex->group_list.elements && /*1*/
|
|
!select_lex->having && /*1*/
|
|
!select_lex->with_sum_func && /*1*/
|
|
!(select_lex->next_select()) && /*2*/
|
|
select_lex->table_list.elements && /*3*/
|
|
(!select_lex->ref_pointer_array[0]->maybe_null || /*4*/
|
|
substype() != Item_subselect::ALL_SUBS)) /*4*/
|
|
{
|
|
Item_sum_hybrid *item;
|
|
nesting_map save_allow_sum_func;
|
|
if (func->l_op())
|
|
{
|
|
/*
|
|
(ALL && (> || =>)) || (ANY && (< || =<))
|
|
for ALL condition is inverted
|
|
*/
|
|
item= new (thd->mem_root) Item_sum_max(thd,
|
|
select_lex->ref_pointer_array[0]);
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
(ALL && (< || =<)) || (ANY && (> || =>))
|
|
for ALL condition is inverted
|
|
*/
|
|
item= new (thd->mem_root) Item_sum_min(thd,
|
|
select_lex->ref_pointer_array[0]);
|
|
}
|
|
if (upper_item)
|
|
upper_item->set_sum_test(item);
|
|
thd->change_item_tree(&select_lex->ref_pointer_array[0], item);
|
|
{
|
|
List_iterator<Item> it(select_lex->item_list);
|
|
it++;
|
|
thd->change_item_tree(it.ref(), item);
|
|
}
|
|
|
|
DBUG_EXECUTE("where",
|
|
print_where(item, "rewrite with MIN/MAX", QT_ORDINARY););
|
|
|
|
save_allow_sum_func= thd->lex->allow_sum_func;
|
|
thd->lex->allow_sum_func|=
|
|
(nesting_map)1 << thd->lex->current_select->nest_level;
|
|
/*
|
|
Item_sum_(max|min) can't substitute other item => we can use 0 as
|
|
reference, also Item_sum_(max|min) can't be fixed after creation, so
|
|
we do not check item->fixed
|
|
*/
|
|
if (item->fix_fields(thd, 0))
|
|
DBUG_RETURN(true);
|
|
thd->lex->allow_sum_func= save_allow_sum_func;
|
|
/* we added aggregate function => we have to change statistic */
|
|
count_field_types(select_lex, &join->tmp_table_param, join->all_fields,
|
|
0);
|
|
if (join->prepare_stage2())
|
|
DBUG_RETURN(true);
|
|
subs= new (thd->mem_root) Item_singlerow_subselect(thd, select_lex);
|
|
|
|
/*
|
|
Remove other strategies if any (we already changed the query and
|
|
can't apply other strategy).
|
|
*/
|
|
set_strategy(SUBS_MAXMIN_INJECTED);
|
|
}
|
|
else
|
|
{
|
|
Item_maxmin_subselect *item;
|
|
subs= item= new (thd->mem_root) Item_maxmin_subselect(thd, this, select_lex, func->l_op());
|
|
if (upper_item)
|
|
upper_item->set_sub_test(item);
|
|
/*
|
|
Remove other strategies if any (we already changed the query and
|
|
can't apply other strategy).
|
|
*/
|
|
set_strategy(SUBS_MAXMIN_ENGINE);
|
|
}
|
|
/*
|
|
The swap is needed for expressions of type 'f1 < ALL ( SELECT ....)'
|
|
where we want to evaluate the sub query even if f1 would be null.
|
|
*/
|
|
subs= func->create_swap(thd, *(optimizer->get_cache()), subs);
|
|
thd->change_item_tree(place, subs);
|
|
if (subs->fix_fields(thd, &subs))
|
|
DBUG_RETURN(true);
|
|
DBUG_ASSERT(subs == (*place)); // There was no substitutions
|
|
|
|
select_lex->master_unit()->uncacheable&= ~UNCACHEABLE_DEPENDENT_INJECTED;
|
|
select_lex->uncacheable&= ~UNCACHEABLE_DEPENDENT_INJECTED;
|
|
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
|
|
bool Item_in_subselect::fix_having(Item *having, SELECT_LEX *select_lex)
|
|
{
|
|
bool fix_res= 0;
|
|
if (!having->fixed)
|
|
{
|
|
select_lex->having_fix_field= 1;
|
|
fix_res= having->fix_fields(thd, 0);
|
|
select_lex->having_fix_field= 0;
|
|
}
|
|
return fix_res;
|
|
}
|
|
|
|
bool Item_allany_subselect::is_maxmin_applicable(JOIN *join)
|
|
{
|
|
/*
|
|
Check if max/min optimization applicable: It is top item of
|
|
WHERE condition.
|
|
*/
|
|
return (abort_on_null || (upper_item && upper_item->is_top_level_item())) &&
|
|
!(join->select_lex->master_unit()->uncacheable & ~UNCACHEABLE_EXPLAIN) && !func->eqne_op();
|
|
}
|
|
|
|
|
|
/**
|
|
Create the predicates needed to transform a single-column IN/ALL/ANY
|
|
subselect into a correlated EXISTS via predicate injection.
|
|
|
|
@param join[in] Join object of the subquery (i.e. 'child' join).
|
|
@param where_item[out] the in-to-exists addition to the where clause
|
|
@param having_item[out] the in-to-exists addition to the having clause
|
|
|
|
@details
|
|
The correlated predicates are created as follows:
|
|
|
|
- If the subquery has aggregates, GROUP BY, or HAVING, convert to
|
|
|
|
SELECT ie FROM ... HAVING subq_having AND
|
|
trigcond(oe $cmp$ ref_or_null_helper<ie>)
|
|
|
|
the addition is wrapped into trigger only when we want to distinguish
|
|
between NULL and FALSE results.
|
|
|
|
- Otherwise (no aggregates/GROUP BY/HAVING) convert it to one of the
|
|
following:
|
|
|
|
= If we don't need to distinguish between NULL and FALSE subquery:
|
|
|
|
SELECT ie FROM ... WHERE subq_where AND (oe $cmp$ ie)
|
|
|
|
= If we need to distinguish between those:
|
|
|
|
SELECT ie FROM ...
|
|
WHERE subq_where AND trigcond((oe $cmp$ ie) OR (ie IS NULL))
|
|
HAVING trigcond(<is_not_null_test>(ie))
|
|
|
|
@retval false If the new conditions were created successfully
|
|
@retval true Error
|
|
*/
|
|
|
|
bool
|
|
Item_in_subselect::create_single_in_to_exists_cond(JOIN *join,
|
|
Item **where_item,
|
|
Item **having_item)
|
|
{
|
|
SELECT_LEX *select_lex= join->select_lex;
|
|
/*
|
|
The non-transformed HAVING clause of 'join' may be stored in two ways
|
|
during JOIN::optimize: this->tmp_having= this->having; this->having= 0;
|
|
*/
|
|
Item* join_having= join->having ? join->having : join->tmp_having;
|
|
DBUG_ENTER("Item_in_subselect::create_single_in_to_exists_cond");
|
|
|
|
*where_item= NULL;
|
|
*having_item= NULL;
|
|
|
|
if (join_having || select_lex->with_sum_func ||
|
|
select_lex->group_list.elements)
|
|
{
|
|
Item *item= func->create(thd, expr,
|
|
new (thd->mem_root) Item_ref_null_helper(
|
|
thd,
|
|
&select_lex->context,
|
|
this,
|
|
&select_lex->
|
|
ref_pointer_array[0],
|
|
(char *)"<ref>",
|
|
this->full_name()));
|
|
if (!abort_on_null && left_expr->maybe_null)
|
|
{
|
|
/*
|
|
We can encounter "NULL IN (SELECT ...)". Wrap the added condition
|
|
within a trig_cond.
|
|
*/
|
|
item= new (thd->mem_root) Item_func_trig_cond(thd, item, get_cond_guard(0));
|
|
}
|
|
|
|
if (!join_having)
|
|
item->name= (char*) in_having_cond;
|
|
if (fix_having(item, select_lex))
|
|
DBUG_RETURN(true);
|
|
*having_item= item;
|
|
}
|
|
else
|
|
{
|
|
Item *item= (Item*) select_lex->item_list.head()->real_item();
|
|
|
|
if (select_lex->table_list.elements)
|
|
{
|
|
Item *having= item;
|
|
Item *orig_item= item;
|
|
|
|
item= func->create(thd, expr, item);
|
|
if (!abort_on_null && orig_item->maybe_null)
|
|
{
|
|
having= new (thd->mem_root) Item_is_not_null_test(thd, this, having);
|
|
if (left_expr->maybe_null)
|
|
{
|
|
if (!(having= new (thd->mem_root) Item_func_trig_cond(thd, having,
|
|
get_cond_guard(0))))
|
|
DBUG_RETURN(true);
|
|
}
|
|
having->name= (char*) in_having_cond;
|
|
if (fix_having(having, select_lex))
|
|
DBUG_RETURN(true);
|
|
*having_item= having;
|
|
|
|
item= new (thd->mem_root) Item_cond_or(thd, item,
|
|
new (thd->mem_root) Item_func_isnull(thd, orig_item));
|
|
}
|
|
/*
|
|
If we may encounter NULL IN (SELECT ...) and care whether subquery
|
|
result is NULL or FALSE, wrap condition in a trig_cond.
|
|
*/
|
|
if (!abort_on_null && left_expr->maybe_null)
|
|
{
|
|
if (!(item= new (thd->mem_root) Item_func_trig_cond(thd, item,
|
|
get_cond_guard(0))))
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
/*
|
|
TODO: figure out why the following is done here in
|
|
single_value_transformer but there is no corresponding action in
|
|
row_value_transformer?
|
|
*/
|
|
item->name= (char *) in_additional_cond;
|
|
if (!item->fixed && item->fix_fields(thd, 0))
|
|
DBUG_RETURN(true);
|
|
*where_item= item;
|
|
}
|
|
else
|
|
{
|
|
if (select_lex->master_unit()->is_union())
|
|
{
|
|
Item *new_having=
|
|
func->create(thd, expr,
|
|
new (thd->mem_root) Item_ref_null_helper(thd,
|
|
&select_lex->context,
|
|
this,
|
|
&select_lex->ref_pointer_array[0],
|
|
(char *)"<no matter>",
|
|
(char *)"<result>"));
|
|
if (!abort_on_null && left_expr->maybe_null)
|
|
{
|
|
if (!(new_having= new (thd->mem_root) Item_func_trig_cond(thd, new_having,
|
|
get_cond_guard(0))))
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
new_having->name= (char*) in_having_cond;
|
|
if (fix_having(new_having, select_lex))
|
|
DBUG_RETURN(true);
|
|
*having_item= new_having;
|
|
}
|
|
else
|
|
DBUG_ASSERT(false);
|
|
}
|
|
}
|
|
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
|
|
/**
|
|
Wrap a multi-column IN/ALL/ANY subselect into an Item_in_optimizer.
|
|
|
|
@param join Join object of the subquery (i.e. 'child' join).
|
|
|
|
@details
|
|
The subquery predicate is wrapped into an Item_in_optimizer. Later the query
|
|
optimization phase chooses whether the subquery under the Item_in_optimizer
|
|
will be further transformed into an equivalent correlated EXISTS by injecting
|
|
additional predicates, or will be executed via subquery materialization in its
|
|
unmodified form.
|
|
|
|
@retval false The subquery was transformed
|
|
@retval true Error
|
|
*/
|
|
|
|
bool
|
|
Item_in_subselect::row_value_transformer(JOIN *join)
|
|
{
|
|
SELECT_LEX *select_lex= join->select_lex;
|
|
uint cols_num= left_expr->cols();
|
|
|
|
DBUG_ENTER("Item_in_subselect::row_value_transformer");
|
|
|
|
// psergey: duplicated_subselect_card_check
|
|
if (select_lex->item_list.elements != cols_num)
|
|
{
|
|
my_error(ER_OPERAND_COLUMNS, MYF(0), cols_num);
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
/*
|
|
Wrap the current IN predicate in an Item_in_optimizer. The actual
|
|
substitution in the Item tree takes place in Item_subselect::fix_fields.
|
|
*/
|
|
if (!substitution)
|
|
{
|
|
//first call for this unit
|
|
SELECT_LEX_UNIT *master_unit= select_lex->master_unit();
|
|
substitution= optimizer;
|
|
|
|
SELECT_LEX *current= thd->lex->current_select;
|
|
thd->lex->current_select= current->return_after_parsing();
|
|
if (!optimizer || optimizer->fix_left(thd))
|
|
{
|
|
thd->lex->current_select= current;
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
// we will refer to upper level cache array => we have to save it in PS
|
|
optimizer->keep_top_level_cache();
|
|
|
|
thd->lex->current_select= current;
|
|
/*
|
|
The uncacheable property controls a number of actions, e.g. whether to
|
|
save/restore (via init_save_join_tab/restore_tmp) the original JOIN for
|
|
plans with a temp table where the original JOIN was overriden by
|
|
make_simple_join. The UNCACHEABLE_EXPLAIN is ignored by EXPLAIN, thus
|
|
non-correlated subqueries will not appear as such to EXPLAIN.
|
|
*/
|
|
master_unit->uncacheable|= UNCACHEABLE_EXPLAIN;
|
|
select_lex->uncacheable|= UNCACHEABLE_EXPLAIN;
|
|
}
|
|
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
|
|
/**
|
|
Create the predicates needed to transform a multi-column IN/ALL/ANY
|
|
subselect into a correlated EXISTS via predicate injection.
|
|
|
|
@details
|
|
The correlated predicates are created as follows:
|
|
|
|
- If the subquery has aggregates, GROUP BY, or HAVING, convert to
|
|
|
|
(l1, l2, l3) IN (SELECT v1, v2, v3 ... HAVING having)
|
|
=>
|
|
EXISTS (SELECT ... HAVING having and
|
|
(l1 = v1 or is null v1) and
|
|
(l2 = v2 or is null v2) and
|
|
(l3 = v3 or is null v3) and
|
|
is_not_null_test(v1) and
|
|
is_not_null_test(v2) and
|
|
is_not_null_test(v3))
|
|
|
|
where is_not_null_test used to register nulls in case if we have
|
|
not found matching to return correct NULL value.
|
|
|
|
- Otherwise (no aggregates/GROUP BY/HAVING) convert the subquery as follows:
|
|
|
|
(l1, l2, l3) IN (SELECT v1, v2, v3 ... WHERE where)
|
|
=>
|
|
EXISTS (SELECT ... WHERE where and
|
|
(l1 = v1 or is null v1) and
|
|
(l2 = v2 or is null v2) and
|
|
(l3 = v3 or is null v3)
|
|
HAVING is_not_null_test(v1) and
|
|
is_not_null_test(v2) and
|
|
is_not_null_test(v3))
|
|
where is_not_null_test registers NULLs values but reject rows.
|
|
|
|
in case when we do not need correct NULL, we have simplier construction:
|
|
EXISTS (SELECT ... WHERE where and
|
|
(l1 = v1) and
|
|
(l2 = v2) and
|
|
(l3 = v3)
|
|
|
|
@param join[in] Join object of the subquery (i.e. 'child' join).
|
|
@param where_item[out] the in-to-exists addition to the where clause
|
|
@param having_item[out] the in-to-exists addition to the having clause
|
|
|
|
@retval false If the new conditions were created successfully
|
|
@retval true Error
|
|
*/
|
|
|
|
bool
|
|
Item_in_subselect::create_row_in_to_exists_cond(JOIN * join,
|
|
Item **where_item,
|
|
Item **having_item)
|
|
{
|
|
SELECT_LEX *select_lex= join->select_lex;
|
|
uint cols_num= left_expr->cols();
|
|
/*
|
|
The non-transformed HAVING clause of 'join' may be stored in two ways
|
|
during JOIN::optimize: this->tmp_having= this->having; this->having= 0;
|
|
*/
|
|
Item* join_having= join->having ? join->having : join->tmp_having;
|
|
bool is_having_used= (join_having || select_lex->with_sum_func ||
|
|
select_lex->group_list.first ||
|
|
!select_lex->table_list.elements);
|
|
|
|
DBUG_ENTER("Item_in_subselect::create_row_in_to_exists_cond");
|
|
|
|
*where_item= NULL;
|
|
*having_item= NULL;
|
|
|
|
if (is_having_used)
|
|
{
|
|
/* TODO: say here explicitly if the order of AND parts matters or not. */
|
|
Item *item_having_part2= 0;
|
|
for (uint i= 0; i < cols_num; i++)
|
|
{
|
|
DBUG_ASSERT((left_expr->fixed &&
|
|
|
|
select_lex->ref_pointer_array[i]->fixed) ||
|
|
(select_lex->ref_pointer_array[i]->type() == REF_ITEM &&
|
|
((Item_ref*)(select_lex->ref_pointer_array[i]))->ref_type() ==
|
|
Item_ref::OUTER_REF));
|
|
if (select_lex->ref_pointer_array[i]->
|
|
check_cols(left_expr->element_index(i)->cols()))
|
|
DBUG_RETURN(true);
|
|
Item *item_eq=
|
|
new (thd->mem_root)
|
|
Item_func_eq(thd, new (thd->mem_root)
|
|
Item_direct_ref(thd, &select_lex->context,
|
|
(*optimizer->get_cache())->
|
|
addr(i),
|
|
(char *)"<no matter>",
|
|
(char *)in_left_expr_name),
|
|
new (thd->mem_root)
|
|
Item_ref(thd, &select_lex->context,
|
|
&select_lex->ref_pointer_array[i],
|
|
(char *)"<no matter>",
|
|
(char *)"<list ref>"));
|
|
Item *item_isnull=
|
|
new (thd->mem_root)
|
|
Item_func_isnull(thd,
|
|
new (thd->mem_root)
|
|
Item_ref(thd, &select_lex->context,
|
|
&select_lex->ref_pointer_array[i],
|
|
(char *)"<no matter>",
|
|
(char *)"<list ref>"));
|
|
Item *col_item= new (thd->mem_root)
|
|
Item_cond_or(thd, item_eq, item_isnull);
|
|
if (!abort_on_null && left_expr->element_index(i)->maybe_null)
|
|
{
|
|
if (!(col_item= new (thd->mem_root)
|
|
Item_func_trig_cond(thd, col_item, get_cond_guard(i))))
|
|
DBUG_RETURN(true);
|
|
}
|
|
*having_item= and_items(thd, *having_item, col_item);
|
|
|
|
Item *item_nnull_test=
|
|
new (thd->mem_root)
|
|
Item_is_not_null_test(thd, this,
|
|
new (thd->mem_root)
|
|
Item_ref(thd, &select_lex->context,
|
|
&select_lex->
|
|
ref_pointer_array[i],
|
|
(char *)"<no matter>",
|
|
(char *)"<list ref>"));
|
|
if (!abort_on_null && left_expr->element_index(i)->maybe_null)
|
|
{
|
|
if (!(item_nnull_test=
|
|
new (thd->mem_root)
|
|
Item_func_trig_cond(thd, item_nnull_test, get_cond_guard(i))))
|
|
DBUG_RETURN(true);
|
|
}
|
|
item_having_part2= and_items(thd, item_having_part2, item_nnull_test);
|
|
item_having_part2->top_level_item();
|
|
}
|
|
*having_item= and_items(thd, *having_item, item_having_part2);
|
|
}
|
|
else
|
|
{
|
|
for (uint i= 0; i < cols_num; i++)
|
|
{
|
|
Item *item, *item_isnull;
|
|
DBUG_ASSERT((left_expr->fixed &&
|
|
select_lex->ref_pointer_array[i]->fixed) ||
|
|
(select_lex->ref_pointer_array[i]->type() == REF_ITEM &&
|
|
((Item_ref*)(select_lex->ref_pointer_array[i]))->ref_type() ==
|
|
Item_ref::OUTER_REF));
|
|
if (select_lex->ref_pointer_array[i]->
|
|
check_cols(left_expr->element_index(i)->cols()))
|
|
DBUG_RETURN(true);
|
|
item= new (thd->mem_root)
|
|
Item_func_eq(thd,
|
|
new (thd->mem_root)
|
|
Item_direct_ref(thd, &select_lex->context,
|
|
(*optimizer->get_cache())->
|
|
addr(i),
|
|
(char *)"<no matter>",
|
|
(char *)in_left_expr_name),
|
|
new (thd->mem_root)
|
|
Item_direct_ref(thd, &select_lex->context,
|
|
&select_lex->
|
|
ref_pointer_array[i],
|
|
(char *)"<no matter>",
|
|
(char *)"<list ref>"));
|
|
if (!abort_on_null && select_lex->ref_pointer_array[i]->maybe_null)
|
|
{
|
|
Item *having_col_item=
|
|
new (thd->mem_root)
|
|
Item_is_not_null_test(thd, this,
|
|
new (thd->mem_root)
|
|
Item_ref(thd, &select_lex->context,
|
|
&select_lex->ref_pointer_array[i],
|
|
(char *)"<no matter>",
|
|
(char *)"<list ref>"));
|
|
|
|
|
|
item_isnull= new (thd->mem_root)
|
|
Item_func_isnull(thd,
|
|
new (thd->mem_root)
|
|
Item_direct_ref(thd, &select_lex->context,
|
|
&select_lex->
|
|
ref_pointer_array[i],
|
|
(char *)"<no matter>",
|
|
(char *)"<list ref>"));
|
|
item= new (thd->mem_root) Item_cond_or(thd, item, item_isnull);
|
|
if (left_expr->element_index(i)->maybe_null)
|
|
{
|
|
if (!(item= new (thd->mem_root)
|
|
Item_func_trig_cond(thd, item, get_cond_guard(i))))
|
|
DBUG_RETURN(true);
|
|
if (!(having_col_item= new (thd->mem_root)
|
|
Item_func_trig_cond(thd, having_col_item, get_cond_guard(i))))
|
|
DBUG_RETURN(true);
|
|
}
|
|
*having_item= and_items(thd, *having_item, having_col_item);
|
|
}
|
|
if (!abort_on_null && left_expr->element_index(i)->maybe_null)
|
|
{
|
|
if (!(item= new (thd->mem_root)
|
|
Item_func_trig_cond(thd, item, get_cond_guard(i))))
|
|
DBUG_RETURN(true);
|
|
}
|
|
*where_item= and_items(thd, *where_item, item);
|
|
}
|
|
}
|
|
|
|
if (*where_item)
|
|
{
|
|
if (!(*where_item)->fixed && (*where_item)->fix_fields(thd, 0))
|
|
DBUG_RETURN(true);
|
|
(*where_item)->top_level_item();
|
|
}
|
|
|
|
if (*having_item)
|
|
{
|
|
if (!join_having)
|
|
(*having_item)->name= (char*) in_having_cond;
|
|
if (fix_having(*having_item, select_lex))
|
|
DBUG_RETURN(true);
|
|
(*having_item)->top_level_item();
|
|
}
|
|
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
|
|
bool
|
|
Item_in_subselect::select_transformer(JOIN *join)
|
|
{
|
|
return select_in_like_transformer(join);
|
|
}
|
|
|
|
bool
|
|
Item_exists_subselect::select_transformer(JOIN *join)
|
|
{
|
|
return select_prepare_to_be_in();
|
|
}
|
|
|
|
|
|
/**
|
|
Create the predicates needed to transform an IN/ALL/ANY subselect into a
|
|
correlated EXISTS via predicate injection.
|
|
|
|
@param join_arg Join object of the subquery.
|
|
|
|
@retval FALSE ok
|
|
@retval TRUE error
|
|
*/
|
|
|
|
bool Item_in_subselect::create_in_to_exists_cond(JOIN *join_arg)
|
|
{
|
|
bool res;
|
|
|
|
DBUG_ASSERT(engine->engine_type() == subselect_engine::SINGLE_SELECT_ENGINE ||
|
|
engine->engine_type() == subselect_engine::UNION_ENGINE);
|
|
/*
|
|
TODO: the call to init_cond_guards allocates and initializes an
|
|
array of booleans that may not be used later because we may choose
|
|
materialization.
|
|
The two calls below to create_XYZ_cond depend on this boolean array.
|
|
If the dependency is removed, the call can be moved to a later phase.
|
|
*/
|
|
init_cond_guards();
|
|
if (left_expr->cols() == 1)
|
|
res= create_single_in_to_exists_cond(join_arg,
|
|
&(join_arg->in_to_exists_where),
|
|
&(join_arg->in_to_exists_having));
|
|
else
|
|
res= create_row_in_to_exists_cond(join_arg,
|
|
&(join_arg->in_to_exists_where),
|
|
&(join_arg->in_to_exists_having));
|
|
|
|
/*
|
|
The IN=>EXISTS transformation makes non-correlated subqueries correlated.
|
|
*/
|
|
if (!left_expr->const_item() || left_expr->is_expensive())
|
|
{
|
|
join_arg->select_lex->uncacheable|= UNCACHEABLE_DEPENDENT_INJECTED;
|
|
join_arg->select_lex->master_unit()->uncacheable|=
|
|
UNCACHEABLE_DEPENDENT_INJECTED;
|
|
}
|
|
/*
|
|
The uncacheable property controls a number of actions, e.g. whether to
|
|
save/restore (via init_save_join_tab/restore_tmp) the original JOIN for
|
|
plans with a temp table where the original JOIN was overriden by
|
|
make_simple_join. The UNCACHEABLE_EXPLAIN is ignored by EXPLAIN, thus
|
|
non-correlated subqueries will not appear as such to EXPLAIN.
|
|
*/
|
|
join_arg->select_lex->master_unit()->uncacheable|= UNCACHEABLE_EXPLAIN;
|
|
join_arg->select_lex->uncacheable|= UNCACHEABLE_EXPLAIN;
|
|
return (res);
|
|
}
|
|
|
|
|
|
/**
|
|
Transform an IN/ALL/ANY subselect into a correlated EXISTS via injecting
|
|
correlated in-to-exists predicates.
|
|
|
|
@param join_arg Join object of the subquery.
|
|
|
|
@retval FALSE ok
|
|
@retval TRUE error
|
|
*/
|
|
|
|
bool Item_in_subselect::inject_in_to_exists_cond(JOIN *join_arg)
|
|
{
|
|
SELECT_LEX *select_lex= join_arg->select_lex;
|
|
Item *where_item= join_arg->in_to_exists_where;
|
|
Item *having_item= join_arg->in_to_exists_having;
|
|
|
|
DBUG_ENTER("Item_in_subselect::inject_in_to_exists_cond");
|
|
|
|
if (where_item)
|
|
{
|
|
List<Item> *and_args= NULL;
|
|
/*
|
|
If the top-level Item of the WHERE clause is an AND, detach the multiple
|
|
equality list that was attached to the end of the AND argument list by
|
|
build_equal_items_for_cond(). The multiple equalities must be detached
|
|
because fix_fields merges lower level AND arguments into the upper AND.
|
|
As a result, the arguments from lower-level ANDs are concatenated after
|
|
the multiple equalities. When the multiple equality list is treated as
|
|
such, it turns out that it contains non-Item_equal object which is wrong.
|
|
*/
|
|
if (join_arg->conds && join_arg->conds->type() == Item::COND_ITEM &&
|
|
((Item_cond*) join_arg->conds)->functype() == Item_func::COND_AND_FUNC)
|
|
{
|
|
and_args= ((Item_cond*) join_arg->conds)->argument_list();
|
|
if (join_arg->cond_equal)
|
|
and_args->disjoin((List<Item> *) &join_arg->cond_equal->current_level);
|
|
}
|
|
|
|
where_item= and_items(thd, join_arg->conds, where_item);
|
|
if (!where_item->fixed && where_item->fix_fields(thd, 0))
|
|
DBUG_RETURN(true);
|
|
// TIMOUR TODO: call optimize_cond() for the new where clause
|
|
thd->change_item_tree(&select_lex->where, where_item);
|
|
select_lex->where->top_level_item();
|
|
join_arg->conds= select_lex->where;
|
|
|
|
/* Attach back the list of multiple equalities to the new top-level AND. */
|
|
if (and_args && join_arg->cond_equal)
|
|
{
|
|
/* The argument list of the top-level AND may change after fix fields. */
|
|
and_args= ((Item_cond*) join_arg->conds)->argument_list();
|
|
List_iterator<Item_equal> li(join_arg->cond_equal->current_level);
|
|
Item_equal *elem;
|
|
while ((elem= li++))
|
|
{
|
|
and_args->push_back(elem, thd->mem_root);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (having_item)
|
|
{
|
|
Item* join_having= join_arg->having ? join_arg->having:join_arg->tmp_having;
|
|
having_item= and_items(thd, join_having, having_item);
|
|
if (fix_having(having_item, select_lex))
|
|
DBUG_RETURN(true);
|
|
// TIMOUR TODO: call optimize_cond() for the new having clause
|
|
thd->change_item_tree(&select_lex->having, having_item);
|
|
select_lex->having->top_level_item();
|
|
join_arg->having= select_lex->having;
|
|
}
|
|
join_arg->thd->change_item_tree(&unit->global_parameters()->select_limit,
|
|
new (thd->mem_root)
|
|
Item_int(thd, (int32) 1));
|
|
unit->select_limit_cnt= 1;
|
|
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
|
|
/*
|
|
If this select can potentially be converted by EXISTS->IN conversion, wrap it
|
|
in an Item_in_optimizer object. Final decision whether to do the conversion
|
|
is done at a later phase.
|
|
*/
|
|
|
|
bool Item_exists_subselect::select_prepare_to_be_in()
|
|
{
|
|
bool trans_res= FALSE;
|
|
DBUG_ENTER("Item_exists_subselect::select_prepare_to_be_in");
|
|
if (!optimizer &&
|
|
thd->lex->sql_command == SQLCOM_SELECT &&
|
|
!unit->first_select()->is_part_of_union() &&
|
|
optimizer_flag(thd, OPTIMIZER_SWITCH_EXISTS_TO_IN) &&
|
|
(is_top_level_item() ||
|
|
(upper_not && upper_not->is_top_level_item())))
|
|
{
|
|
Query_arena *arena, backup;
|
|
bool result;
|
|
arena= thd->activate_stmt_arena_if_needed(&backup);
|
|
result= (!(optimizer= new (thd->mem_root) Item_in_optimizer(thd, new (thd->mem_root) Item_int(thd, 1), this)));
|
|
if (arena)
|
|
thd->restore_active_arena(arena, &backup);
|
|
if (result)
|
|
trans_res= TRUE;
|
|
else
|
|
substitution= optimizer;
|
|
}
|
|
DBUG_RETURN(trans_res);
|
|
}
|
|
|
|
/**
|
|
Check if 'func' is an equality in form "inner_table.column = outer_expr"
|
|
|
|
@param func Expression to check
|
|
@param local_field OUT Return "inner_table.column" here
|
|
@param outer_expr OUT Return outer_expr here
|
|
|
|
@return true - 'func' is an Equality.
|
|
*/
|
|
|
|
static bool check_equality_for_exist2in(Item_func *func,
|
|
Item_ident **local_field,
|
|
Item **outer_exp)
|
|
{
|
|
Item **args;
|
|
if (func->functype() != Item_func::EQ_FUNC)
|
|
return FALSE;
|
|
DBUG_ASSERT(func->argument_count() == 2);
|
|
args= func->arguments();
|
|
if (args[0]->real_type() == Item::FIELD_ITEM &&
|
|
args[0]->all_used_tables() != OUTER_REF_TABLE_BIT &&
|
|
args[1]->all_used_tables() == OUTER_REF_TABLE_BIT)
|
|
{
|
|
/* It is Item_field or Item_direct_view_ref) */
|
|
DBUG_ASSERT(args[0]->type() == Item::FIELD_ITEM ||
|
|
args[0]->type() == Item::REF_ITEM);
|
|
*local_field= (Item_ident *)args[0];
|
|
*outer_exp= args[1];
|
|
return TRUE;
|
|
}
|
|
else if (args[1]->real_type() == Item::FIELD_ITEM &&
|
|
args[1]->all_used_tables() != OUTER_REF_TABLE_BIT &&
|
|
args[0]->all_used_tables() == OUTER_REF_TABLE_BIT)
|
|
{
|
|
/* It is Item_field or Item_direct_view_ref) */
|
|
DBUG_ASSERT(args[0]->type() == Item::FIELD_ITEM ||
|
|
args[0]->type() == Item::REF_ITEM);
|
|
*local_field= (Item_ident *)args[1];
|
|
*outer_exp= args[0];
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
typedef struct st_eq_field_outer
|
|
{
|
|
Item **eq_ref;
|
|
Item_ident *local_field;
|
|
Item *outer_exp;
|
|
} EQ_FIELD_OUTER;
|
|
|
|
|
|
/**
|
|
Check if 'conds' is a set of AND-ed outer_expr=inner_table.col equalities
|
|
|
|
@detail
|
|
Check if 'conds' has form
|
|
|
|
outer1=inner_tbl1.col1 AND ... AND outer2=inner_tbl1.col2 AND remainder_cond
|
|
|
|
@param conds Condition to be checked
|
|
@parm result Array to collect EQ_FIELD_OUTER elements describing
|
|
inner-vs-outer equalities the function has found.
|
|
@return
|
|
false - some inner-vs-outer equalities were found
|
|
true - otherwise.
|
|
*/
|
|
|
|
static bool find_inner_outer_equalities(Item **conds,
|
|
Dynamic_array<EQ_FIELD_OUTER> &result)
|
|
{
|
|
bool found= FALSE;
|
|
EQ_FIELD_OUTER element;
|
|
if (is_cond_and(*conds))
|
|
{
|
|
List_iterator<Item> li(*((Item_cond*)*conds)->argument_list());
|
|
Item *item;
|
|
while ((item= li++))
|
|
{
|
|
if (item->type() == Item::FUNC_ITEM &&
|
|
check_equality_for_exist2in((Item_func *)item,
|
|
&element.local_field,
|
|
&element.outer_exp))
|
|
{
|
|
found= TRUE;
|
|
element.eq_ref= li.ref();
|
|
if (result.append(element))
|
|
goto alloc_err;
|
|
}
|
|
}
|
|
}
|
|
else if ((*conds)->type() == Item::FUNC_ITEM &&
|
|
check_equality_for_exist2in((Item_func *)*conds,
|
|
&element.local_field,
|
|
&element.outer_exp))
|
|
{
|
|
found= TRUE;
|
|
element.eq_ref= conds;
|
|
if (result.append(element))
|
|
goto alloc_err;
|
|
}
|
|
|
|
return !found;
|
|
alloc_err:
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
Converts EXISTS subquery to IN subquery if it is possible and has sense
|
|
|
|
@param opt_arg Pointer on THD
|
|
|
|
@return TRUE in case of error and FALSE otherwise.
|
|
*/
|
|
|
|
bool Item_exists_subselect::exists2in_processor(uchar *opt_arg)
|
|
{
|
|
THD *thd= (THD *)opt_arg;
|
|
SELECT_LEX *first_select=unit->first_select(), *save_select;
|
|
JOIN *join= first_select->join;
|
|
Item **eq_ref= NULL;
|
|
Item_ident *local_field= NULL;
|
|
Item *outer_exp= NULL;
|
|
Item *left_exp= NULL; Item_in_subselect *in_subs;
|
|
Query_arena *arena= NULL, backup;
|
|
int res= FALSE;
|
|
List<Item> outer;
|
|
Dynamic_array<EQ_FIELD_OUTER> eqs(5, 5);
|
|
bool will_be_correlated;
|
|
DBUG_ENTER("Item_exists_subselect::exists2in_processor");
|
|
|
|
if (!optimizer ||
|
|
!optimizer_flag(thd, OPTIMIZER_SWITCH_EXISTS_TO_IN) ||
|
|
(!is_top_level_item() && (!upper_not ||
|
|
!upper_not->is_top_level_item())) ||
|
|
first_select->is_part_of_union() ||
|
|
first_select->group_list.elements ||
|
|
first_select->order_list.elements ||
|
|
join->having ||
|
|
first_select->with_sum_func ||
|
|
!first_select->leaf_tables.elements||
|
|
!join->conds)
|
|
DBUG_RETURN(FALSE);
|
|
|
|
DBUG_ASSERT(first_select->order_list.elements == 0 &&
|
|
first_select->group_list.elements == 0 &&
|
|
first_select->having == NULL);
|
|
|
|
if (find_inner_outer_equalities(&join->conds, eqs))
|
|
DBUG_RETURN(FALSE);
|
|
|
|
DBUG_ASSERT(eqs.elements() != 0);
|
|
|
|
save_select= thd->lex->current_select;
|
|
thd->lex->current_select= first_select;
|
|
|
|
/* check that the subquery has only dependencies we are going pull out */
|
|
{
|
|
List<Item> unused;
|
|
Collect_deps_prm prm= {&unused, // parameters
|
|
unit->first_select()->nest_level_base, // nest_level_base
|
|
0, // count
|
|
unit->first_select()->nest_level, // nest_level
|
|
FALSE // collect
|
|
};
|
|
walk(&Item::collect_outer_ref_processor, TRUE, (uchar*)&prm);
|
|
DBUG_ASSERT(prm.count > 0);
|
|
DBUG_ASSERT(prm.count >= (uint)eqs.elements());
|
|
will_be_correlated= prm.count > (uint)eqs.elements();
|
|
if (upper_not && will_be_correlated)
|
|
goto out;
|
|
}
|
|
|
|
if ((uint)eqs.elements() > (first_select->item_list.elements +
|
|
first_select->select_n_reserved))
|
|
goto out;
|
|
/* It is simple query */
|
|
DBUG_ASSERT(first_select->join->all_fields.elements ==
|
|
first_select->item_list.elements);
|
|
|
|
arena= thd->activate_stmt_arena_if_needed(&backup);
|
|
|
|
while (first_select->item_list.elements > (uint)eqs.elements())
|
|
{
|
|
first_select->item_list.pop();
|
|
first_select->join->all_fields.elements--;
|
|
}
|
|
{
|
|
List_iterator<Item> it(first_select->item_list);
|
|
|
|
for (uint i= 0; i < (uint)eqs.elements(); i++)
|
|
{
|
|
Item *item= it++;
|
|
eq_ref= eqs.at(i).eq_ref;
|
|
local_field= eqs.at(i).local_field;
|
|
outer_exp= eqs.at(i).outer_exp;
|
|
/* Add the field to the SELECT_LIST */
|
|
if (item)
|
|
it.replace(local_field);
|
|
else
|
|
{
|
|
first_select->item_list.push_back(local_field, thd->mem_root);
|
|
first_select->join->all_fields.elements++;
|
|
}
|
|
first_select->ref_pointer_array[i]= (Item *)local_field;
|
|
|
|
/* remove the parts from condition */
|
|
if (!upper_not || !local_field->maybe_null)
|
|
*eq_ref= new (thd->mem_root) Item_int(thd, 1);
|
|
else
|
|
{
|
|
*eq_ref= new (thd->mem_root)
|
|
Item_func_isnotnull(thd,
|
|
new (thd->mem_root)
|
|
Item_field(thd,
|
|
((Item_field*)(local_field->
|
|
real_item()))->context,
|
|
((Item_field*)(local_field->
|
|
real_item()))->field));
|
|
if((*eq_ref)->fix_fields(thd, (Item **)eq_ref))
|
|
{
|
|
res= TRUE;
|
|
goto out;
|
|
}
|
|
}
|
|
outer_exp->fix_after_pullout(unit->outer_select(), &outer_exp);
|
|
outer_exp->update_used_tables();
|
|
outer.push_back(outer_exp, thd->mem_root);
|
|
}
|
|
}
|
|
|
|
join->conds->update_used_tables();
|
|
|
|
/* make IN SUBQUERY and put outer_exp as left part */
|
|
if (eqs.elements() == 1)
|
|
left_exp= outer_exp;
|
|
else
|
|
{
|
|
if (!(left_exp= new (thd->mem_root) Item_row(thd, outer)))
|
|
{
|
|
res= TRUE;
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
/* make EXISTS->IN permanet (see Item_subselect::init()) */
|
|
set_exists_transformed();
|
|
|
|
first_select->select_limit= NULL;
|
|
if (!(in_subs= new (thd->mem_root) Item_in_subselect(thd, left_exp,
|
|
first_select)))
|
|
{
|
|
res= TRUE;
|
|
goto out;
|
|
}
|
|
in_subs->set_exists_transformed();
|
|
optimizer->arguments()[0]= left_exp;
|
|
optimizer->arguments()[1]= in_subs;
|
|
in_subs->optimizer= optimizer;
|
|
DBUG_ASSERT(is_top_level_item() ||
|
|
(upper_not && upper_not->is_top_level_item()));
|
|
in_subs->top_level_item();
|
|
{
|
|
SELECT_LEX *current= thd->lex->current_select;
|
|
optimizer->reset_cache(); // renew cache, and we will not keep it
|
|
thd->lex->current_select= unit->outer_select();
|
|
DBUG_ASSERT(optimizer);
|
|
if (optimizer->fix_left(thd))
|
|
{
|
|
res= TRUE;
|
|
/*
|
|
We should not restore thd->lex->current_select because it will be
|
|
reset on exit from this procedure
|
|
*/
|
|
goto out;
|
|
}
|
|
/*
|
|
As far as Item_ref_in_optimizer do not substitute itself on fix_fields
|
|
we can use same item for all selects.
|
|
*/
|
|
in_subs->expr= new (thd->mem_root)
|
|
Item_direct_ref(thd, &first_select->context,
|
|
(Item**)optimizer->get_cache(),
|
|
(char *)"<no matter>",
|
|
(char *)in_left_expr_name);
|
|
if (in_subs->fix_fields(thd, optimizer->arguments() + 1))
|
|
{
|
|
res= TRUE;
|
|
/*
|
|
We should not restore thd->lex->current_select because it will be
|
|
reset on exit from this procedure
|
|
*/
|
|
goto out;
|
|
}
|
|
{
|
|
/* Move dependence list */
|
|
List_iterator_fast<Ref_to_outside> it(upper_refs);
|
|
Ref_to_outside *upper;
|
|
while ((upper= it++))
|
|
{
|
|
uint i;
|
|
for (i= 0; i < (uint)eqs.elements(); i++)
|
|
if (eqs.at(i).outer_exp->
|
|
walk(&Item::find_item_processor, TRUE, (uchar*)upper->item))
|
|
break;
|
|
if (i == (uint)eqs.elements() &&
|
|
(in_subs->upper_refs.push_back(upper, thd->stmt_arena->mem_root)))
|
|
goto out;
|
|
}
|
|
}
|
|
in_subs->update_used_tables();
|
|
/*
|
|
The engine of the subquery is fixed so above fix_fields() is not
|
|
complete and should be fixed
|
|
*/
|
|
in_subs->upper_refs= upper_refs;
|
|
upper_refs.empty();
|
|
thd->lex->current_select= current;
|
|
}
|
|
|
|
DBUG_ASSERT(unit->item == in_subs);
|
|
DBUG_ASSERT(join == first_select->join);
|
|
/*
|
|
Fix dependency info
|
|
*/
|
|
in_subs->is_correlated= will_be_correlated;
|
|
if (!will_be_correlated)
|
|
{
|
|
first_select->uncacheable&= ~UNCACHEABLE_DEPENDENT_GENERATED;
|
|
unit->uncacheable&= ~UNCACHEABLE_DEPENDENT_GENERATED;
|
|
}
|
|
/*
|
|
set possible optimization strategies
|
|
*/
|
|
in_subs->emb_on_expr_nest= emb_on_expr_nest;
|
|
res= check_and_do_in_subquery_rewrites(join);
|
|
first_select->join->prepare_stage2();
|
|
|
|
first_select->fix_prepare_information(thd, &join->conds, &join->having);
|
|
|
|
if (upper_not)
|
|
{
|
|
Item *exp;
|
|
if (eqs.elements() == 1)
|
|
{
|
|
exp= (optimizer->arguments()[0]->maybe_null ?
|
|
(Item*) new (thd->mem_root)
|
|
Item_cond_and(thd,
|
|
new (thd->mem_root)
|
|
Item_func_isnotnull(thd,
|
|
new (thd->mem_root)
|
|
Item_direct_ref(thd,
|
|
&unit->outer_select()->context,
|
|
optimizer->arguments(),
|
|
(char *)"<no matter>",
|
|
(char *)exists_outer_expr_name)),
|
|
optimizer) :
|
|
(Item *)optimizer);
|
|
}
|
|
else
|
|
{
|
|
List<Item> *and_list= new List<Item>;
|
|
if (!and_list)
|
|
{
|
|
res= TRUE;
|
|
goto out;
|
|
}
|
|
for (size_t i= 0; i < eqs.elements(); i++)
|
|
{
|
|
if (optimizer->arguments()[0]->maybe_null)
|
|
{
|
|
and_list->
|
|
push_front(new (thd->mem_root)
|
|
Item_func_isnotnull(thd,
|
|
new (thd->mem_root)
|
|
Item_direct_ref(thd,
|
|
&unit->outer_select()->context,
|
|
optimizer->arguments()[0]->addr(i),
|
|
(char *)"<no matter>",
|
|
(char *)exists_outer_expr_name)),
|
|
thd->mem_root);
|
|
}
|
|
}
|
|
if (and_list->elements > 0)
|
|
{
|
|
and_list->push_front(optimizer, thd->mem_root);
|
|
exp= new (thd->mem_root) Item_cond_and(thd, *and_list);
|
|
}
|
|
else
|
|
exp= optimizer;
|
|
}
|
|
upper_not->arguments()[0]= exp;
|
|
if (!exp->fixed && exp->fix_fields(thd, upper_not->arguments()))
|
|
{
|
|
res= TRUE;
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
out:
|
|
thd->lex->current_select= save_select;
|
|
if (arena)
|
|
thd->restore_active_arena(arena, &backup);
|
|
DBUG_RETURN(res);
|
|
}
|
|
|
|
|
|
/**
|
|
Prepare IN/ALL/ANY/SOME subquery transformation and call the appropriate
|
|
transformation function.
|
|
|
|
@param join JOIN object of transforming subquery
|
|
|
|
@notes
|
|
To decide which transformation procedure (scalar or row) applicable here
|
|
we have to call fix_fields() for the left expression to be able to call
|
|
cols() method on it. Also this method makes arena management for
|
|
underlying transformation methods.
|
|
|
|
@retval false OK
|
|
@retval true Error
|
|
*/
|
|
|
|
bool
|
|
Item_in_subselect::select_in_like_transformer(JOIN *join)
|
|
{
|
|
Query_arena *arena= 0, backup;
|
|
SELECT_LEX *current= thd->lex->current_select;
|
|
const char *save_where= thd->where;
|
|
bool trans_res= true;
|
|
bool result;
|
|
|
|
DBUG_ENTER("Item_in_subselect::select_in_like_transformer");
|
|
|
|
/*
|
|
IN/SOME/ALL/ANY subqueries aren't support LIMIT clause. Without it
|
|
ORDER BY clause becomes meaningless thus we drop it here.
|
|
*/
|
|
for (SELECT_LEX *sl= current->master_unit()->first_select();
|
|
sl; sl= sl->next_select())
|
|
{
|
|
if (sl->join)
|
|
{
|
|
sl->join->order= 0;
|
|
sl->join->skip_sort_order= 1;
|
|
}
|
|
}
|
|
|
|
thd->where= "IN/ALL/ANY subquery";
|
|
|
|
/*
|
|
In some optimisation cases we will not need this Item_in_optimizer
|
|
object, but we can't know it here, but here we need address correct
|
|
reference on left expresion.
|
|
|
|
note: we won't need Item_in_optimizer when handling degenerate cases
|
|
like "... IN (SELECT 1)"
|
|
*/
|
|
arena= thd->activate_stmt_arena_if_needed(&backup);
|
|
if (!optimizer)
|
|
{
|
|
optimizer= new (thd->mem_root) Item_in_optimizer(thd, left_expr_orig, this);
|
|
if ((result= !optimizer))
|
|
goto out;
|
|
}
|
|
|
|
thd->lex->current_select= current->return_after_parsing();
|
|
result= optimizer->fix_left(thd);
|
|
thd->lex->current_select= current;
|
|
|
|
if (changed)
|
|
{
|
|
trans_res= false;
|
|
goto out;
|
|
}
|
|
|
|
|
|
if (result)
|
|
goto out;
|
|
|
|
/*
|
|
Both transformers call fix_fields() only for Items created inside them,
|
|
and all that items do not make permanent changes in current item arena
|
|
which allow to us call them with changed arena (if we do not know nature
|
|
of Item, we have to call fix_fields() for it only with original arena to
|
|
avoid memory leack)
|
|
*/
|
|
if (left_expr->cols() == 1)
|
|
trans_res= single_value_transformer(join);
|
|
else
|
|
{
|
|
/* we do not support row operation for ALL/ANY/SOME */
|
|
if (func != &eq_creator)
|
|
{
|
|
if (arena)
|
|
thd->restore_active_arena(arena, &backup);
|
|
my_error(ER_OPERAND_COLUMNS, MYF(0), 1);
|
|
DBUG_RETURN(true);
|
|
}
|
|
trans_res= row_value_transformer(join);
|
|
}
|
|
out:
|
|
if (arena)
|
|
thd->restore_active_arena(arena, &backup);
|
|
thd->where= save_where;
|
|
DBUG_RETURN(trans_res);
|
|
}
|
|
|
|
|
|
void Item_in_subselect::print(String *str, enum_query_type query_type)
|
|
{
|
|
if (test_strategy(SUBS_IN_TO_EXISTS))
|
|
str->append(STRING_WITH_LEN("<exists>"));
|
|
else
|
|
{
|
|
left_expr->print(str, query_type);
|
|
str->append(STRING_WITH_LEN(" in "));
|
|
}
|
|
Item_subselect::print(str, query_type);
|
|
}
|
|
|
|
bool Item_exists_subselect::fix_fields(THD *thd, Item **ref)
|
|
{
|
|
DBUG_ENTER("Item_exists_subselect::fix_fields");
|
|
if (exists_transformed)
|
|
DBUG_RETURN( !( (*ref)= new (thd->mem_root) Item_int(thd, 1)));
|
|
DBUG_RETURN(Item_subselect::fix_fields(thd, ref));
|
|
}
|
|
|
|
|
|
bool Item_in_subselect::fix_fields(THD *thd_arg, Item **ref)
|
|
{
|
|
uint outer_cols_num;
|
|
List<Item> *inner_cols;
|
|
char const *save_where= thd->where;
|
|
DBUG_ENTER("Item_in_subselect::fix_fields");
|
|
|
|
if (test_strategy(SUBS_SEMI_JOIN))
|
|
DBUG_RETURN( !( (*ref)= new (thd->mem_root) Item_int(thd, 1)) );
|
|
|
|
thd->where= "IN/ALL/ANY subquery";
|
|
/*
|
|
Check if the outer and inner IN operands match in those cases when we
|
|
will not perform IN=>EXISTS transformation. Currently this is when we
|
|
use subquery materialization.
|
|
|
|
The condition below is true when this method was called recursively from
|
|
inside JOIN::prepare for the JOIN object created by the call chain
|
|
Item_subselect::fix_fields -> subselect_single_select_engine::prepare,
|
|
which creates a JOIN object for the subquery and calls JOIN::prepare for
|
|
the JOIN of the subquery.
|
|
Notice that in some cases, this doesn't happen, and the check_cols()
|
|
test for each Item happens later in
|
|
Item_in_subselect::row_value_in_to_exists_transformer.
|
|
The reason for this mess is that our JOIN::prepare phase works top-down
|
|
instead of bottom-up, so we first do name resoluton and semantic checks
|
|
for the outer selects, then for the inner.
|
|
*/
|
|
if (engine &&
|
|
engine->engine_type() == subselect_engine::SINGLE_SELECT_ENGINE &&
|
|
((subselect_single_select_engine*)engine)->join)
|
|
{
|
|
outer_cols_num= left_expr->cols();
|
|
|
|
if (unit->is_union())
|
|
inner_cols= &(unit->types);
|
|
else
|
|
inner_cols= &(unit->first_select()->item_list);
|
|
if (outer_cols_num != inner_cols->elements)
|
|
{
|
|
my_error(ER_OPERAND_COLUMNS, MYF(0), outer_cols_num);
|
|
goto err;
|
|
}
|
|
if (outer_cols_num > 1)
|
|
{
|
|
List_iterator<Item> inner_col_it(*inner_cols);
|
|
Item *inner_col;
|
|
for (uint i= 0; i < outer_cols_num; i++)
|
|
{
|
|
inner_col= inner_col_it++;
|
|
if (inner_col->check_cols(left_expr->element_index(i)->cols()))
|
|
goto err;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (left_expr && !left_expr->fixed &&
|
|
left_expr->fix_fields(thd_arg, &left_expr))
|
|
goto err;
|
|
else
|
|
if (Item_subselect::fix_fields(thd_arg, ref))
|
|
goto err;
|
|
fixed= TRUE;
|
|
thd->where= save_where;
|
|
DBUG_RETURN(FALSE);
|
|
|
|
err:
|
|
thd->where= save_where;
|
|
DBUG_RETURN(TRUE);
|
|
}
|
|
|
|
|
|
void Item_in_subselect::fix_after_pullout(st_select_lex *new_parent, Item **ref)
|
|
{
|
|
left_expr->fix_after_pullout(new_parent, &left_expr);
|
|
Item_subselect::fix_after_pullout(new_parent, ref);
|
|
used_tables_cache |= left_expr->used_tables();
|
|
}
|
|
|
|
void Item_in_subselect::update_used_tables()
|
|
{
|
|
Item_subselect::update_used_tables();
|
|
left_expr->update_used_tables();
|
|
//used_tables_cache |= left_expr->used_tables();
|
|
used_tables_cache= Item_subselect::used_tables() | left_expr->used_tables();
|
|
}
|
|
|
|
|
|
/**
|
|
Try to create and initialize an engine to compute a subselect via
|
|
materialization.
|
|
|
|
@details
|
|
The method creates a new engine for materialized execution, and initializes
|
|
the engine. The initialization may fail
|
|
- either because it wasn't possible to create the needed temporary table
|
|
and its index,
|
|
- or because of a memory allocation error,
|
|
|
|
@returns
|
|
@retval TRUE memory allocation error occurred
|
|
@retval FALSE an execution method was chosen successfully
|
|
*/
|
|
|
|
bool Item_in_subselect::setup_mat_engine()
|
|
{
|
|
subselect_hash_sj_engine *mat_engine= NULL;
|
|
subselect_single_select_engine *select_engine;
|
|
|
|
DBUG_ENTER("Item_in_subselect::setup_mat_engine");
|
|
|
|
/*
|
|
The select_engine (that executes transformed IN=>EXISTS subselects) is
|
|
pre-created at parse time, and is stored in statment memory (preserved
|
|
across PS executions).
|
|
*/
|
|
DBUG_ASSERT(engine->engine_type() == subselect_engine::SINGLE_SELECT_ENGINE);
|
|
select_engine= (subselect_single_select_engine*) engine;
|
|
|
|
/* Create/initialize execution objects. */
|
|
if (!(mat_engine= new subselect_hash_sj_engine(thd, this, select_engine)))
|
|
DBUG_RETURN(TRUE);
|
|
|
|
if (mat_engine->init(&select_engine->join->fields_list,
|
|
engine->get_identifier()))
|
|
DBUG_RETURN(TRUE);
|
|
|
|
engine= mat_engine;
|
|
DBUG_RETURN(FALSE);
|
|
}
|
|
|
|
|
|
/**
|
|
Initialize the cache of the left operand of the IN predicate.
|
|
|
|
@note This method has the same purpose as alloc_group_fields(),
|
|
but it takes a different kind of collection of items, and the
|
|
list we push to is dynamically allocated.
|
|
|
|
@retval TRUE if a memory allocation error occurred or the cache is
|
|
not applicable to the current query
|
|
@retval FALSE if success
|
|
*/
|
|
|
|
bool Item_in_subselect::init_left_expr_cache()
|
|
{
|
|
JOIN *outer_join;
|
|
|
|
outer_join= unit->outer_select()->join;
|
|
/*
|
|
An IN predicate might be evaluated in a query for which all tables have
|
|
been optimzied away.
|
|
*/
|
|
if (!outer_join || !outer_join->table_count || !outer_join->tables_list)
|
|
return TRUE;
|
|
|
|
if (!(left_expr_cache= new List<Cached_item>))
|
|
return TRUE;
|
|
|
|
for (uint i= 0; i < left_expr->cols(); i++)
|
|
{
|
|
Cached_item *cur_item_cache= new_Cached_item(thd,
|
|
left_expr->element_index(i),
|
|
FALSE);
|
|
if (!cur_item_cache || left_expr_cache->push_front(cur_item_cache,
|
|
thd->mem_root))
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
bool Item_in_subselect::init_cond_guards()
|
|
{
|
|
uint cols_num= left_expr->cols();
|
|
if (!abort_on_null && left_expr->maybe_null && !pushed_cond_guards)
|
|
{
|
|
if (!(pushed_cond_guards= (bool*)thd->alloc(sizeof(bool) * cols_num)))
|
|
return TRUE;
|
|
for (uint i= 0; i < cols_num; i++)
|
|
pushed_cond_guards[i]= TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
bool
|
|
Item_allany_subselect::select_transformer(JOIN *join)
|
|
{
|
|
DBUG_ENTER("Item_allany_subselect::select_transformer");
|
|
DBUG_ASSERT((in_strategy & ~(SUBS_MAXMIN_INJECTED | SUBS_MAXMIN_ENGINE |
|
|
SUBS_IN_TO_EXISTS | SUBS_STRATEGY_CHOSEN)) == 0);
|
|
if (upper_item)
|
|
upper_item->show= 1;
|
|
DBUG_RETURN(select_in_like_transformer(join));
|
|
}
|
|
|
|
|
|
void Item_allany_subselect::print(String *str, enum_query_type query_type)
|
|
{
|
|
if (test_strategy(SUBS_IN_TO_EXISTS))
|
|
str->append(STRING_WITH_LEN("<exists>"));
|
|
else
|
|
{
|
|
left_expr->print(str, query_type);
|
|
str->append(' ');
|
|
str->append(func->symbol(all));
|
|
str->append(all ? " all " : " any ", 5);
|
|
}
|
|
Item_subselect::print(str, query_type);
|
|
}
|
|
|
|
|
|
void Item_allany_subselect::no_rows_in_result()
|
|
{
|
|
/*
|
|
Subquery predicates outside of the SELECT list must be evaluated in order
|
|
to possibly filter the special result row generated for implicit grouping
|
|
if the subquery is in the HAVING clause.
|
|
If the predicate is constant, we need its actual value in the only result
|
|
row for queries with implicit grouping.
|
|
*/
|
|
if (parsing_place != SELECT_LIST || const_item())
|
|
return;
|
|
value= 0;
|
|
null_value= 0;
|
|
was_null= 0;
|
|
make_const();
|
|
}
|
|
|
|
|
|
void subselect_engine::set_thd(THD *thd_arg)
|
|
{
|
|
thd= thd_arg;
|
|
if (result)
|
|
result->set_thd(thd_arg);
|
|
}
|
|
|
|
|
|
subselect_single_select_engine::
|
|
subselect_single_select_engine(THD *thd_arg, st_select_lex *select,
|
|
select_result_interceptor *result_arg,
|
|
Item_subselect *item_arg)
|
|
:subselect_engine(thd_arg, item_arg, result_arg),
|
|
prepared(0), executed(0),
|
|
select_lex(select), join(0)
|
|
{
|
|
select_lex->master_unit()->item= item_arg;
|
|
}
|
|
|
|
int subselect_single_select_engine::get_identifier()
|
|
{
|
|
return select_lex->select_number;
|
|
}
|
|
|
|
void subselect_single_select_engine::cleanup()
|
|
{
|
|
DBUG_ENTER("subselect_single_select_engine::cleanup");
|
|
prepared= executed= 0;
|
|
join= 0;
|
|
result->cleanup();
|
|
select_lex->uncacheable&= ~UNCACHEABLE_DEPENDENT_INJECTED;
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
void subselect_union_engine::cleanup()
|
|
{
|
|
DBUG_ENTER("subselect_union_engine::cleanup");
|
|
unit->reinit_exec_mechanism();
|
|
result->cleanup();
|
|
unit->uncacheable&= ~UNCACHEABLE_DEPENDENT_INJECTED;
|
|
for (SELECT_LEX *sl= unit->first_select(); sl; sl= sl->next_select())
|
|
sl->uncacheable&= ~UNCACHEABLE_DEPENDENT_INJECTED;
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
bool subselect_union_engine::is_executed() const
|
|
{
|
|
return unit->executed;
|
|
}
|
|
|
|
|
|
/*
|
|
Check if last execution of the subquery engine produced any rows
|
|
|
|
SYNOPSIS
|
|
subselect_union_engine::no_rows()
|
|
|
|
DESCRIPTION
|
|
Check if last execution of the subquery engine produced any rows. The
|
|
return value is undefined if last execution ended in an error.
|
|
|
|
RETURN
|
|
TRUE - Last subselect execution has produced no rows
|
|
FALSE - Otherwise
|
|
*/
|
|
|
|
bool subselect_union_engine::no_rows()
|
|
{
|
|
/* Check if we got any rows when reading UNION result from temp. table: */
|
|
return MY_TEST(!(unit->fake_select_lex ?
|
|
unit->fake_select_lex->join->send_records :
|
|
((select_union_direct *)(unit->get_union_result()))
|
|
->send_records));
|
|
}
|
|
|
|
|
|
void subselect_uniquesubquery_engine::cleanup()
|
|
{
|
|
DBUG_ENTER("subselect_uniquesubquery_engine::cleanup");
|
|
/*
|
|
Note for mergers: we don't have to, and actually must not de-initialize
|
|
tab->table->file here.
|
|
- We don't have to, because free_tmp_table() will call ha_index_or_rnd_end
|
|
- We must not do it, because tab->table may be a derived table which
|
|
has been already dropped by close_thread_tables(), while we here are
|
|
called from cleanup_items()
|
|
*/
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
subselect_union_engine::subselect_union_engine(THD *thd_arg, st_select_lex_unit *u,
|
|
select_result_interceptor *result_arg,
|
|
Item_subselect *item_arg)
|
|
:subselect_engine(thd_arg, item_arg, result_arg)
|
|
{
|
|
unit= u;
|
|
unit->item= item_arg;
|
|
}
|
|
|
|
|
|
/**
|
|
Create and prepare the JOIN object that represents the query execution
|
|
plan for the subquery.
|
|
|
|
@details
|
|
This method is called from Item_subselect::fix_fields. For prepared
|
|
statements it is called both during the PREPARE and EXECUTE phases in the
|
|
following ways:
|
|
- During PREPARE the optimizer needs some properties
|
|
(join->fields_list.elements) of the JOIN to proceed with preparation of
|
|
the remaining query (namely to complete ::fix_fields for the subselect
|
|
related classes. In the end of PREPARE the JOIN is deleted.
|
|
- When we EXECUTE the query, Item_subselect::fix_fields is called again, and
|
|
the JOIN object is re-created again, prepared and executed. In the end of
|
|
execution it is deleted.
|
|
In all cases the JOIN is created in runtime memory (not in the permanent
|
|
memory root).
|
|
|
|
@todo
|
|
Re-check what properties of 'join' are needed during prepare, and see if
|
|
we can avoid creating a JOIN during JOIN::prepare of the outer join.
|
|
|
|
@retval 0 if success
|
|
@retval 1 if error
|
|
*/
|
|
|
|
int subselect_single_select_engine::prepare()
|
|
{
|
|
if (prepared)
|
|
return 0;
|
|
if (select_lex->join)
|
|
{
|
|
select_lex->cleanup();
|
|
}
|
|
join= new JOIN(thd, select_lex->item_list,
|
|
select_lex->options | SELECT_NO_UNLOCK, result);
|
|
if (!join || !result)
|
|
return 1; /* Fatal error is set already. */
|
|
prepared= 1;
|
|
SELECT_LEX *save_select= thd->lex->current_select;
|
|
thd->lex->current_select= select_lex;
|
|
if (join->prepare(select_lex->table_list.first,
|
|
select_lex->with_wild,
|
|
select_lex->where,
|
|
select_lex->order_list.elements +
|
|
select_lex->group_list.elements,
|
|
select_lex->order_list.first,
|
|
false,
|
|
select_lex->group_list.first,
|
|
select_lex->having,
|
|
NULL, select_lex,
|
|
select_lex->master_unit()))
|
|
return 1;
|
|
thd->lex->current_select= save_select;
|
|
return 0;
|
|
}
|
|
|
|
int subselect_union_engine::prepare()
|
|
{
|
|
return unit->prepare(thd, result, SELECT_NO_UNLOCK);
|
|
}
|
|
|
|
int subselect_uniquesubquery_engine::prepare()
|
|
{
|
|
/* Should never be called. */
|
|
DBUG_ASSERT(FALSE);
|
|
return 1;
|
|
}
|
|
|
|
|
|
/*
|
|
Check if last execution of the subquery engine produced any rows
|
|
|
|
SYNOPSIS
|
|
subselect_single_select_engine::no_rows()
|
|
|
|
DESCRIPTION
|
|
Check if last execution of the subquery engine produced any rows. The
|
|
return value is undefined if last execution ended in an error.
|
|
|
|
RETURN
|
|
TRUE - Last subselect execution has produced no rows
|
|
FALSE - Otherwise
|
|
*/
|
|
|
|
bool subselect_single_select_engine::no_rows()
|
|
{
|
|
return !item->assigned();
|
|
}
|
|
|
|
|
|
/*
|
|
makes storage for the output values for the subquery and calcuates
|
|
their data and column types and their nullability.
|
|
*/
|
|
void subselect_engine::set_row(List<Item> &item_list, Item_cache **row)
|
|
{
|
|
Item *sel_item;
|
|
List_iterator_fast<Item> li(item_list);
|
|
cmp_type= res_type= STRING_RESULT;
|
|
res_field_type= MYSQL_TYPE_VAR_STRING;
|
|
for (uint i= 0; (sel_item= li++); i++)
|
|
{
|
|
item->max_length= sel_item->max_length;
|
|
res_type= sel_item->result_type();
|
|
cmp_type= sel_item->cmp_type();
|
|
res_field_type= sel_item->field_type();
|
|
item->decimals= sel_item->decimals;
|
|
item->unsigned_flag= sel_item->unsigned_flag;
|
|
maybe_null= sel_item->maybe_null;
|
|
if (!(row[i]= Item_cache::get_cache(thd, sel_item, sel_item->cmp_type())))
|
|
return;
|
|
row[i]->setup(thd, sel_item);
|
|
//psergey-backport-timours: row[i]->store(sel_item);
|
|
}
|
|
if (item_list.elements > 1)
|
|
cmp_type= res_type= ROW_RESULT;
|
|
}
|
|
|
|
void subselect_single_select_engine::fix_length_and_dec(Item_cache **row)
|
|
{
|
|
DBUG_ASSERT(row || select_lex->item_list.elements==1);
|
|
set_row(select_lex->item_list, row);
|
|
item->collation.set(row[0]->collation);
|
|
if (cols() != 1)
|
|
maybe_null= 0;
|
|
}
|
|
|
|
void subselect_union_engine::fix_length_and_dec(Item_cache **row)
|
|
{
|
|
DBUG_ASSERT(row || unit->first_select()->item_list.elements==1);
|
|
|
|
if (unit->first_select()->item_list.elements == 1)
|
|
{
|
|
set_row(unit->types, row);
|
|
item->collation.set(row[0]->collation);
|
|
}
|
|
else
|
|
{
|
|
bool maybe_null_saved= maybe_null;
|
|
set_row(unit->types, row);
|
|
maybe_null= maybe_null_saved;
|
|
}
|
|
}
|
|
|
|
void subselect_uniquesubquery_engine::fix_length_and_dec(Item_cache **row)
|
|
{
|
|
//this never should be called
|
|
DBUG_ASSERT(0);
|
|
}
|
|
|
|
int read_first_record_seq(JOIN_TAB *tab);
|
|
int rr_sequential(READ_RECORD *info);
|
|
int join_read_always_key_or_null(JOIN_TAB *tab);
|
|
int join_read_next_same_or_null(READ_RECORD *info);
|
|
|
|
int subselect_single_select_engine::exec()
|
|
{
|
|
DBUG_ENTER("subselect_single_select_engine::exec");
|
|
|
|
char const *save_where= thd->where;
|
|
SELECT_LEX *save_select= thd->lex->current_select;
|
|
thd->lex->current_select= select_lex;
|
|
|
|
if (!join->optimized)
|
|
{
|
|
SELECT_LEX_UNIT *unit= select_lex->master_unit();
|
|
|
|
unit->set_limit(unit->global_parameters());
|
|
if (join->optimize())
|
|
{
|
|
thd->where= save_where;
|
|
executed= 1;
|
|
thd->lex->current_select= save_select;
|
|
DBUG_RETURN(join->error ? join->error : 1);
|
|
}
|
|
if (!select_lex->uncacheable && thd->lex->describe &&
|
|
!(join->select_options & SELECT_DESCRIBE))
|
|
{
|
|
item->update_used_tables();
|
|
if (item->const_item())
|
|
{
|
|
/*
|
|
It's necessary to keep original JOIN table because
|
|
create_sort_index() function may overwrite original
|
|
JOIN_TAB::type and wrong optimization method can be
|
|
selected on re-execution.
|
|
*/
|
|
select_lex->uncacheable|= UNCACHEABLE_EXPLAIN;
|
|
select_lex->master_unit()->uncacheable|= UNCACHEABLE_EXPLAIN;
|
|
}
|
|
}
|
|
if (item->engine_changed(this))
|
|
DBUG_RETURN(1);
|
|
}
|
|
if (select_lex->uncacheable &&
|
|
select_lex->uncacheable != UNCACHEABLE_EXPLAIN
|
|
&& executed)
|
|
{
|
|
if (join->reinit())
|
|
{
|
|
thd->where= save_where;
|
|
thd->lex->current_select= save_select;
|
|
DBUG_RETURN(1);
|
|
}
|
|
item->reset();
|
|
item->assigned((executed= 0));
|
|
}
|
|
if (!executed)
|
|
{
|
|
item->reset_value_registration();
|
|
JOIN_TAB *changed_tabs[MAX_TABLES];
|
|
JOIN_TAB **last_changed_tab= changed_tabs;
|
|
if (item->have_guarded_conds())
|
|
{
|
|
/*
|
|
For at least one of the pushed predicates the following is true:
|
|
We should not apply optimizations based on the condition that was
|
|
pushed down into the subquery. Those optimizations are ref[_or_null]
|
|
acceses. Change them to be full table scans.
|
|
*/
|
|
JOIN_TAB *tab;
|
|
for (tab= first_linear_tab(join, WITH_BUSH_ROOTS, WITHOUT_CONST_TABLES);
|
|
tab; tab= next_linear_tab(join, tab, WITH_BUSH_ROOTS))
|
|
{
|
|
if (tab && tab->keyuse)
|
|
{
|
|
for (uint i= 0; i < tab->ref.key_parts; i++)
|
|
{
|
|
bool *cond_guard= tab->ref.cond_guards[i];
|
|
if (cond_guard && !*cond_guard)
|
|
{
|
|
/* Change the access method to full table scan */
|
|
tab->save_read_first_record= tab->read_first_record;
|
|
tab->save_read_record= tab->read_record.read_record;
|
|
tab->read_record.read_record= rr_sequential;
|
|
tab->read_first_record= read_first_record_seq;
|
|
tab->read_record.record= tab->table->record[0];
|
|
tab->read_record.thd= join->thd;
|
|
tab->read_record.ref_length= tab->table->file->ref_length;
|
|
tab->read_record.unlock_row= rr_unlock_row;
|
|
*(last_changed_tab++)= tab;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
join->exec();
|
|
|
|
/* Enable the optimizations back */
|
|
for (JOIN_TAB **ptab= changed_tabs; ptab != last_changed_tab; ptab++)
|
|
{
|
|
JOIN_TAB *tab= *ptab;
|
|
tab->read_record.record= 0;
|
|
tab->read_record.ref_length= 0;
|
|
tab->read_first_record= tab->save_read_first_record;
|
|
tab->read_record.read_record= tab->save_read_record;
|
|
}
|
|
executed= 1;
|
|
if (!(uncacheable() & ~UNCACHEABLE_EXPLAIN))
|
|
item->make_const();
|
|
thd->where= save_where;
|
|
thd->lex->current_select= save_select;
|
|
DBUG_RETURN(join->error || thd->is_fatal_error || thd->is_error());
|
|
}
|
|
thd->where= save_where;
|
|
thd->lex->current_select= save_select;
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
int subselect_union_engine::exec()
|
|
{
|
|
char const *save_where= thd->where;
|
|
int res= unit->exec();
|
|
thd->where= save_where;
|
|
return res;
|
|
}
|
|
|
|
|
|
/*
|
|
Search for at least one row satisfying select condition
|
|
|
|
SYNOPSIS
|
|
subselect_uniquesubquery_engine::scan_table()
|
|
|
|
DESCRIPTION
|
|
Scan the table using sequential access until we find at least one row
|
|
satisfying select condition.
|
|
|
|
The caller must set this->empty_result_set=FALSE before calling this
|
|
function. This function will set it to TRUE if it finds a matching row.
|
|
|
|
RETURN
|
|
FALSE - OK
|
|
TRUE - Error
|
|
*/
|
|
|
|
int subselect_uniquesubquery_engine::scan_table()
|
|
{
|
|
int error;
|
|
TABLE *table= tab->table;
|
|
DBUG_ENTER("subselect_uniquesubquery_engine::scan_table");
|
|
|
|
if ((table->file->inited &&
|
|
(error= table->file->ha_index_end())) ||
|
|
(error= table->file->ha_rnd_init(1)))
|
|
{
|
|
(void) report_error(table, error);
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
table->file->extra_opt(HA_EXTRA_CACHE,
|
|
current_thd->variables.read_buff_size);
|
|
table->null_row= 0;
|
|
for (;;)
|
|
{
|
|
error=table->file->ha_rnd_next(table->record[0]);
|
|
if (error) {
|
|
if (error == HA_ERR_RECORD_DELETED)
|
|
{
|
|
error= 0;
|
|
continue;
|
|
}
|
|
if (error == HA_ERR_END_OF_FILE)
|
|
{
|
|
error= 0;
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
error= report_error(table, error);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!cond || cond->val_int())
|
|
{
|
|
empty_result_set= FALSE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
table->file->ha_rnd_end();
|
|
DBUG_RETURN(error != 0);
|
|
}
|
|
|
|
|
|
/**
|
|
Copy ref key for index access into the only subquery table.
|
|
|
|
@details
|
|
Copy ref key and check for conversion problems.
|
|
If there is an error converting the left IN operand to the column type of
|
|
the right IN operand count it as no match. In this case IN has the value of
|
|
FALSE. We mark the subquery table cursor as having no more rows (to ensure
|
|
that the processing that follows will not find a match) and return FALSE,
|
|
so IN is not treated as returning NULL.
|
|
|
|
@returns
|
|
@retval FALSE The outer ref was copied into an index lookup key.
|
|
@retval TRUE The outer ref cannot possibly match any row, IN is FALSE.
|
|
*/
|
|
|
|
bool subselect_uniquesubquery_engine::copy_ref_key(bool skip_constants)
|
|
{
|
|
DBUG_ENTER("subselect_uniquesubquery_engine::copy_ref_key");
|
|
|
|
for (store_key **copy= tab->ref.key_copy ; *copy ; copy++)
|
|
{
|
|
enum store_key::store_key_result store_res;
|
|
if (skip_constants && (*copy)->store_key_is_const())
|
|
continue;
|
|
store_res= (*copy)->copy();
|
|
tab->ref.key_err= store_res;
|
|
|
|
if (store_res == store_key::STORE_KEY_FATAL)
|
|
{
|
|
/*
|
|
Error converting the left IN operand to the column type of the right
|
|
IN operand.
|
|
*/
|
|
DBUG_RETURN(true);
|
|
}
|
|
}
|
|
DBUG_RETURN(false);
|
|
}
|
|
|
|
|
|
/**
|
|
Execute subselect via unique index lookup
|
|
|
|
@details
|
|
Find rows corresponding to the ref key using index access.
|
|
If some part of the lookup key is NULL, then we're evaluating
|
|
NULL IN (SELECT ... )
|
|
This is a special case, we don't need to search for NULL in the table,
|
|
instead, the result value is
|
|
- NULL if select produces empty row set
|
|
- FALSE otherwise.
|
|
|
|
In some cases (IN subselect is a top level item, i.e. abort_on_null==TRUE)
|
|
the caller doesn't distinguish between NULL and FALSE result and we just
|
|
return FALSE.
|
|
Otherwise we make a full table scan to see if there is at least one
|
|
matching row.
|
|
|
|
The result of this function (info about whether a row was found) is
|
|
stored in this->empty_result_set.
|
|
|
|
@returns
|
|
@retval 0 OK
|
|
@retval 1 notify caller to call Item_subselect::reset(),
|
|
in most cases reset() sets the result to NULL
|
|
*/
|
|
|
|
int subselect_uniquesubquery_engine::exec()
|
|
{
|
|
DBUG_ENTER("subselect_uniquesubquery_engine::exec");
|
|
int error;
|
|
TABLE *table= tab->table;
|
|
empty_result_set= TRUE;
|
|
table->status= 0;
|
|
Item_in_subselect *in_subs= (Item_in_subselect *) item;
|
|
|
|
if (!tab->preread_init_done && tab->preread_init())
|
|
DBUG_RETURN(1);
|
|
|
|
if (in_subs->left_expr_has_null())
|
|
{
|
|
/*
|
|
The case when all values in left_expr are NULL is handled by
|
|
Item_in_optimizer::val_int().
|
|
*/
|
|
if (in_subs->is_top_level_item())
|
|
DBUG_RETURN(1); /* notify caller to call reset() and set NULL value. */
|
|
else
|
|
DBUG_RETURN(scan_table());
|
|
}
|
|
|
|
if (copy_ref_key(true))
|
|
{
|
|
/* We know that there will be no rows even if we scan. */
|
|
in_subs->value= 0;
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
if (!table->file->inited &&
|
|
(error= table->file->ha_index_init(tab->ref.key, 0)))
|
|
{
|
|
(void) report_error(table, error);
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
error= table->file->ha_index_read_map(table->record[0],
|
|
tab->ref.key_buff,
|
|
make_prev_keypart_map(tab->
|
|
ref.key_parts),
|
|
HA_READ_KEY_EXACT);
|
|
if (error &&
|
|
error != HA_ERR_KEY_NOT_FOUND && error != HA_ERR_END_OF_FILE)
|
|
error= report_error(table, error);
|
|
else
|
|
{
|
|
error= 0;
|
|
table->null_row= 0;
|
|
if (!table->status && (!cond || cond->val_int()))
|
|
{
|
|
((Item_in_subselect *) item)->value= 1;
|
|
empty_result_set= FALSE;
|
|
}
|
|
else
|
|
((Item_in_subselect *) item)->value= 0;
|
|
}
|
|
|
|
DBUG_RETURN(error != 0);
|
|
}
|
|
|
|
|
|
/*
|
|
TIMOUR: write comment
|
|
*/
|
|
|
|
int subselect_uniquesubquery_engine::index_lookup()
|
|
{
|
|
DBUG_ENTER("subselect_uniquesubquery_engine::index_lookup");
|
|
int error;
|
|
TABLE *table= tab->table;
|
|
|
|
if (!table->file->inited)
|
|
table->file->ha_index_init(tab->ref.key, 0);
|
|
error= table->file->ha_index_read_map(table->record[0],
|
|
tab->ref.key_buff,
|
|
make_prev_keypart_map(tab->
|
|
ref.key_parts),
|
|
HA_READ_KEY_EXACT);
|
|
DBUG_PRINT("info", ("lookup result: %i", error));
|
|
|
|
if (error && error != HA_ERR_KEY_NOT_FOUND && error != HA_ERR_END_OF_FILE)
|
|
{
|
|
/*
|
|
TIMOUR: I don't understand at all when do we need to call report_error.
|
|
In most places where we access an index, we don't do this. Why here?
|
|
*/
|
|
error= report_error(table, error);
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
table->null_row= 0;
|
|
if (!error && (!cond || cond->val_int()))
|
|
((Item_in_subselect *) item)->value= 1;
|
|
else
|
|
((Item_in_subselect *) item)->value= 0;
|
|
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
|
|
|
|
subselect_uniquesubquery_engine::~subselect_uniquesubquery_engine()
|
|
{
|
|
/* Tell handler we don't need the index anymore */
|
|
//psergey-merge-todo: the following was gone in 6.0:
|
|
//psergey-merge: don't need this after all: tab->table->file->ha_index_end();
|
|
}
|
|
|
|
|
|
/**
|
|
Execute subselect via unique index lookup
|
|
|
|
@details
|
|
The engine is used to resolve subqueries in form
|
|
|
|
oe IN (SELECT key FROM tbl WHERE subq_where)
|
|
|
|
The value of the predicate is calculated as follows:
|
|
1. If oe IS NULL, this is a special case, do a full table scan on
|
|
table tbl and search for row that satisfies subq_where. If such
|
|
row is found, return NULL, otherwise return FALSE.
|
|
2. Make an index lookup via key=oe, search for a row that satisfies
|
|
subq_where. If found, return TRUE.
|
|
3. If check_null==TRUE, make another lookup via key=NULL, search for a
|
|
row that satisfies subq_where. If found, return NULL, otherwise
|
|
return FALSE.
|
|
|
|
@todo
|
|
The step #1 can be optimized further when the index has several key
|
|
parts. Consider a subquery:
|
|
|
|
(oe1, oe2) IN (SELECT keypart1, keypart2 FROM tbl WHERE subq_where)
|
|
|
|
and suppose we need to evaluate it for {oe1, oe2}=={const1, NULL}.
|
|
Current code will do a full table scan and obtain correct result. There
|
|
is a better option: instead of evaluating
|
|
|
|
SELECT keypart1, keypart2 FROM tbl WHERE subq_where (1)
|
|
|
|
and checking if it has produced any matching rows, evaluate
|
|
|
|
SELECT keypart2 FROM tbl WHERE subq_where AND keypart1=const1 (2)
|
|
|
|
If this query produces a row, the result is NULL (as we're evaluating
|
|
"(const1, NULL) IN { (const1, X), ... }", which has a value of UNKNOWN,
|
|
i.e. NULL). If the query produces no rows, the result is FALSE.
|
|
|
|
We currently evaluate (1) by doing a full table scan. (2) can be
|
|
evaluated by doing a "ref" scan on "keypart1=const1", which can be much
|
|
cheaper. We can use index statistics to quickly check whether "ref" scan
|
|
will be cheaper than full table scan.
|
|
|
|
@returns
|
|
@retval 0 OK
|
|
@retval 1 notify caller to call Item_subselect::reset(),
|
|
in most cases reset() sets the result to NULL
|
|
*/
|
|
|
|
int subselect_indexsubquery_engine::exec()
|
|
{
|
|
DBUG_ENTER("subselect_indexsubquery_engine");
|
|
int error;
|
|
bool null_finding= 0;
|
|
TABLE *table= tab->table;
|
|
Item_in_subselect *in_subs= (Item_in_subselect *) item;
|
|
|
|
((Item_in_subselect *) item)->value= 0;
|
|
empty_result_set= TRUE;
|
|
table->status= 0;
|
|
|
|
if (check_null)
|
|
{
|
|
/* We need to check for NULL if there wasn't a matching value */
|
|
*tab->ref.null_ref_key= 0; // Search first for not null
|
|
((Item_in_subselect *) item)->was_null= 0;
|
|
}
|
|
|
|
if (!tab->preread_init_done && tab->preread_init())
|
|
DBUG_RETURN(1);
|
|
|
|
if (in_subs->left_expr_has_null())
|
|
{
|
|
/*
|
|
The case when all values in left_expr are NULL is handled by
|
|
Item_in_optimizer::val_int().
|
|
*/
|
|
if (in_subs->is_top_level_item())
|
|
DBUG_RETURN(1); /* notify caller to call reset() and set NULL value. */
|
|
else
|
|
DBUG_RETURN(scan_table());
|
|
}
|
|
|
|
if (copy_ref_key(true))
|
|
{
|
|
/* We know that there will be no rows even if we scan. */
|
|
in_subs->value= 0;
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
if (!table->file->inited &&
|
|
(error= table->file->ha_index_init(tab->ref.key, 1)))
|
|
{
|
|
(void) report_error(table, error);
|
|
DBUG_RETURN(true);
|
|
}
|
|
|
|
error= table->file->ha_index_read_map(table->record[0],
|
|
tab->ref.key_buff,
|
|
make_prev_keypart_map(tab->
|
|
ref.key_parts),
|
|
HA_READ_KEY_EXACT);
|
|
if (error &&
|
|
error != HA_ERR_KEY_NOT_FOUND && error != HA_ERR_END_OF_FILE)
|
|
error= report_error(table, error);
|
|
else
|
|
{
|
|
for (;;)
|
|
{
|
|
error= 0;
|
|
table->null_row= 0;
|
|
if (!table->status)
|
|
{
|
|
if ((!cond || cond->val_int()) && (!having || having->val_int()))
|
|
{
|
|
empty_result_set= FALSE;
|
|
if (null_finding)
|
|
((Item_in_subselect *) item)->was_null= 1;
|
|
else
|
|
((Item_in_subselect *) item)->value= 1;
|
|
break;
|
|
}
|
|
error= table->file->ha_index_next_same(table->record[0],
|
|
tab->ref.key_buff,
|
|
tab->ref.key_length);
|
|
if (error && error != HA_ERR_END_OF_FILE)
|
|
{
|
|
error= report_error(table, error);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (!check_null || null_finding)
|
|
break; /* We don't need to check nulls */
|
|
*tab->ref.null_ref_key= 1;
|
|
null_finding= 1;
|
|
/* Check if there exists a row with a null value in the index */
|
|
if ((error= (safe_index_read(tab) == 1)))
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
DBUG_RETURN(error != 0);
|
|
}
|
|
|
|
|
|
uint subselect_single_select_engine::cols()
|
|
{
|
|
//psergey-sj-backport: the following assert was gone in 6.0:
|
|
//DBUG_ASSERT(select_lex->join != 0); // should be called after fix_fields()
|
|
//return select_lex->join->fields_list.elements;
|
|
return select_lex->item_list.elements;
|
|
}
|
|
|
|
|
|
uint subselect_union_engine::cols()
|
|
{
|
|
DBUG_ASSERT(unit->is_prepared()); // should be called after fix_fields()
|
|
return unit->types.elements;
|
|
}
|
|
|
|
|
|
uint8 subselect_single_select_engine::uncacheable()
|
|
{
|
|
return select_lex->uncacheable;
|
|
}
|
|
|
|
|
|
uint8 subselect_union_engine::uncacheable()
|
|
{
|
|
return unit->uncacheable;
|
|
}
|
|
|
|
|
|
void subselect_single_select_engine::exclude()
|
|
{
|
|
select_lex->master_unit()->exclude_level();
|
|
}
|
|
|
|
void subselect_union_engine::exclude()
|
|
{
|
|
unit->exclude_level();
|
|
}
|
|
|
|
|
|
void subselect_uniquesubquery_engine::exclude()
|
|
{
|
|
//this never should be called
|
|
DBUG_ASSERT(0);
|
|
}
|
|
|
|
|
|
table_map subselect_engine::calc_const_tables(List<TABLE_LIST> &list)
|
|
{
|
|
table_map map= 0;
|
|
List_iterator<TABLE_LIST> ti(list);
|
|
TABLE_LIST *table;
|
|
//for (; table; table= table->next_leaf)
|
|
while ((table= ti++))
|
|
{
|
|
TABLE *tbl= table->table;
|
|
if (tbl && tbl->const_table)
|
|
map|= tbl->map;
|
|
}
|
|
return map;
|
|
}
|
|
|
|
|
|
table_map subselect_single_select_engine::upper_select_const_tables()
|
|
{
|
|
return calc_const_tables(select_lex->outer_select()->leaf_tables);
|
|
}
|
|
|
|
|
|
table_map subselect_union_engine::upper_select_const_tables()
|
|
{
|
|
return calc_const_tables(unit->outer_select()->leaf_tables);
|
|
}
|
|
|
|
|
|
void subselect_single_select_engine::print(String *str,
|
|
enum_query_type query_type)
|
|
{
|
|
select_lex->print(thd, str, query_type);
|
|
}
|
|
|
|
|
|
void subselect_union_engine::print(String *str, enum_query_type query_type)
|
|
{
|
|
unit->print(str, query_type);
|
|
}
|
|
|
|
|
|
void subselect_uniquesubquery_engine::print(String *str,
|
|
enum_query_type query_type)
|
|
{
|
|
char *table_name= tab->table->s->table_name.str;
|
|
str->append(STRING_WITH_LEN("<primary_index_lookup>("));
|
|
tab->ref.items[0]->print(str, query_type);
|
|
str->append(STRING_WITH_LEN(" in "));
|
|
if (tab->table->s->table_category == TABLE_CATEGORY_TEMPORARY)
|
|
{
|
|
/*
|
|
Temporary tables' names change across runs, so they can't be used for
|
|
EXPLAIN EXTENDED.
|
|
*/
|
|
str->append(STRING_WITH_LEN("<temporary table>"));
|
|
}
|
|
else
|
|
str->append(table_name, tab->table->s->table_name.length);
|
|
KEY *key_info= tab->table->key_info+ tab->ref.key;
|
|
str->append(STRING_WITH_LEN(" on "));
|
|
str->append(key_info->name);
|
|
if (cond)
|
|
{
|
|
str->append(STRING_WITH_LEN(" where "));
|
|
cond->print(str, query_type);
|
|
}
|
|
str->append(')');
|
|
}
|
|
|
|
/*
|
|
TODO:
|
|
The above ::print method should be changed as below. Do it after
|
|
all other tests pass.
|
|
|
|
void subselect_uniquesubquery_engine::print(String *str)
|
|
{
|
|
KEY *key_info= tab->table->key_info + tab->ref.key;
|
|
str->append(STRING_WITH_LEN("<primary_index_lookup>("));
|
|
for (uint i= 0; i < key_info->user_defined_key_parts; i++)
|
|
tab->ref.items[i]->print(str);
|
|
str->append(STRING_WITH_LEN(" in "));
|
|
str->append(tab->table->s->table_name.str, tab->table->s->table_name.length);
|
|
str->append(STRING_WITH_LEN(" on "));
|
|
str->append(key_info->name);
|
|
if (cond)
|
|
{
|
|
str->append(STRING_WITH_LEN(" where "));
|
|
cond->print(str);
|
|
}
|
|
str->append(')');
|
|
}
|
|
*/
|
|
|
|
void subselect_indexsubquery_engine::print(String *str,
|
|
enum_query_type query_type)
|
|
{
|
|
str->append(STRING_WITH_LEN("<index_lookup>("));
|
|
tab->ref.items[0]->print(str, query_type);
|
|
str->append(STRING_WITH_LEN(" in "));
|
|
str->append(tab->table->s->table_name.str, tab->table->s->table_name.length);
|
|
KEY *key_info= tab->table->key_info+ tab->ref.key;
|
|
str->append(STRING_WITH_LEN(" on "));
|
|
str->append(key_info->name);
|
|
if (check_null)
|
|
str->append(STRING_WITH_LEN(" checking NULL"));
|
|
if (cond)
|
|
{
|
|
str->append(STRING_WITH_LEN(" where "));
|
|
cond->print(str, query_type);
|
|
}
|
|
if (having)
|
|
{
|
|
str->append(STRING_WITH_LEN(" having "));
|
|
having->print(str, query_type);
|
|
}
|
|
str->append(')');
|
|
}
|
|
|
|
/**
|
|
change select_result object of engine.
|
|
|
|
@param si new subselect Item
|
|
@param res new select_result object
|
|
@param temp temporary assignment
|
|
|
|
@retval
|
|
FALSE OK
|
|
@retval
|
|
TRUE error
|
|
*/
|
|
|
|
bool
|
|
subselect_single_select_engine::change_result(Item_subselect *si,
|
|
select_result_interceptor *res,
|
|
bool temp)
|
|
{
|
|
DBUG_ENTER("subselect_single_select_engine::change_result");
|
|
item= si;
|
|
if (temp)
|
|
{
|
|
/*
|
|
Here we reuse change_item_tree to roll back assignment. It has
|
|
nothing special about Item* pointer so it is safe conversion. We do
|
|
not change the interface to be compatible with MySQL.
|
|
*/
|
|
thd->change_item_tree((Item**) &result, (Item*)res);
|
|
}
|
|
else
|
|
result= res;
|
|
|
|
/*
|
|
We can't use 'result' below as gcc 4.2.4's alias optimization
|
|
assumes that result was not changed by thd->change_item_tree().
|
|
I tried to find a solution to make gcc happy, but could not find anything
|
|
that would not require a lot of extra code that would be harder to manage
|
|
than the current code.
|
|
*/
|
|
DBUG_RETURN(select_lex->join->change_result(res, NULL));
|
|
}
|
|
|
|
|
|
/**
|
|
change select_result object of engine.
|
|
|
|
@param si new subselect Item
|
|
@param res new select_result object
|
|
|
|
@retval
|
|
FALSE OK
|
|
@retval
|
|
TRUE error
|
|
*/
|
|
|
|
bool subselect_union_engine::change_result(Item_subselect *si,
|
|
select_result_interceptor *res,
|
|
bool temp)
|
|
{
|
|
item= si;
|
|
int rc= unit->change_result(res, result);
|
|
if (temp)
|
|
thd->change_item_tree((Item**) &result, (Item*)res);
|
|
else
|
|
result= res;
|
|
return rc;
|
|
}
|
|
|
|
|
|
/**
|
|
change select_result emulation, never should be called.
|
|
|
|
@param si new subselect Item
|
|
@param res new select_result object
|
|
|
|
@retval
|
|
FALSE OK
|
|
@retval
|
|
TRUE error
|
|
*/
|
|
|
|
bool
|
|
subselect_uniquesubquery_engine::change_result(Item_subselect *si,
|
|
select_result_interceptor *res,
|
|
bool temp
|
|
__attribute__((unused)))
|
|
{
|
|
DBUG_ASSERT(0);
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/**
|
|
Report about presence of tables in subquery.
|
|
|
|
@retval
|
|
TRUE there are not tables used in subquery
|
|
@retval
|
|
FALSE there are some tables in subquery
|
|
*/
|
|
bool subselect_single_select_engine::no_tables()
|
|
{
|
|
return(select_lex->table_list.elements == 0);
|
|
}
|
|
|
|
|
|
/*
|
|
Check statically whether the subquery can return NULL
|
|
|
|
SINOPSYS
|
|
subselect_single_select_engine::may_be_null()
|
|
|
|
RETURN
|
|
FALSE can guarantee that the subquery never return NULL
|
|
TRUE otherwise
|
|
*/
|
|
bool subselect_single_select_engine::may_be_null()
|
|
{
|
|
return ((no_tables() && !join->conds && !join->having) ? maybe_null : 1);
|
|
}
|
|
|
|
|
|
/**
|
|
Report about presence of tables in subquery.
|
|
|
|
@retval
|
|
TRUE there are not tables used in subquery
|
|
@retval
|
|
FALSE there are some tables in subquery
|
|
*/
|
|
bool subselect_union_engine::no_tables()
|
|
{
|
|
for (SELECT_LEX *sl= unit->first_select(); sl; sl= sl->next_select())
|
|
{
|
|
if (sl->table_list.elements)
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/**
|
|
Report about presence of tables in subquery.
|
|
|
|
@retval
|
|
TRUE there are not tables used in subquery
|
|
@retval
|
|
FALSE there are some tables in subquery
|
|
*/
|
|
|
|
bool subselect_uniquesubquery_engine::no_tables()
|
|
{
|
|
/* returning value is correct, but this method should never be called */
|
|
DBUG_ASSERT(FALSE);
|
|
return 0;
|
|
}
|
|
|
|
|
|
/******************************************************************************
|
|
WL#1110 - Implementation of class subselect_hash_sj_engine
|
|
******************************************************************************/
|
|
|
|
|
|
/**
|
|
Check if an IN predicate should be executed via partial matching using
|
|
only schema information.
|
|
|
|
@details
|
|
This test essentially has three results:
|
|
- partial matching is applicable, but cannot be executed due to a
|
|
limitation in the total number of indexes, as a result we can't
|
|
use subquery materialization at all.
|
|
- partial matching is either applicable or not, and this can be
|
|
determined by looking at 'this->max_keys'.
|
|
If max_keys > 1, then we need partial matching because there are
|
|
more indexes than just the one we use during materialization to
|
|
remove duplicates.
|
|
|
|
@note
|
|
TIMOUR: The schema-based analysis for partial matching can be done once for
|
|
prepared statement and remembered. It is done here to remove the need to
|
|
save/restore all related variables between each re-execution, thus making
|
|
the code simpler.
|
|
|
|
@retval PARTIAL_MATCH if a partial match should be used
|
|
@retval COMPLETE_MATCH if a complete match (index lookup) should be used
|
|
*/
|
|
|
|
subselect_hash_sj_engine::exec_strategy
|
|
subselect_hash_sj_engine::get_strategy_using_schema()
|
|
{
|
|
Item_in_subselect *item_in= (Item_in_subselect *) item;
|
|
|
|
if (item_in->is_top_level_item())
|
|
return COMPLETE_MATCH;
|
|
else
|
|
{
|
|
List_iterator<Item> inner_col_it(*item_in->unit->get_unit_column_types());
|
|
Item *outer_col, *inner_col;
|
|
|
|
for (uint i= 0; i < item_in->left_expr->cols(); i++)
|
|
{
|
|
outer_col= item_in->left_expr->element_index(i);
|
|
inner_col= inner_col_it++;
|
|
|
|
if (!inner_col->maybe_null && !outer_col->maybe_null)
|
|
bitmap_set_bit(&non_null_key_parts, i);
|
|
else
|
|
{
|
|
bitmap_set_bit(&partial_match_key_parts, i);
|
|
++count_partial_match_columns;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* If no column contains NULLs use regular hash index lookups. */
|
|
if (count_partial_match_columns)
|
|
return PARTIAL_MATCH;
|
|
return COMPLETE_MATCH;
|
|
}
|
|
|
|
|
|
/**
|
|
Test whether an IN predicate must be computed via partial matching
|
|
based on the NULL statistics for each column of a materialized subquery.
|
|
|
|
@details The procedure analyzes column NULL statistics, updates the
|
|
matching type of columns that cannot be NULL or that contain only NULLs.
|
|
Based on this, the procedure determines the final execution strategy for
|
|
the [NOT] IN predicate.
|
|
|
|
@retval PARTIAL_MATCH if a partial match should be used
|
|
@retval COMPLETE_MATCH if a complete match (index lookup) should be used
|
|
*/
|
|
|
|
subselect_hash_sj_engine::exec_strategy
|
|
subselect_hash_sj_engine::get_strategy_using_data()
|
|
{
|
|
Item_in_subselect *item_in= (Item_in_subselect *) item;
|
|
select_materialize_with_stats *result_sink=
|
|
(select_materialize_with_stats *) result;
|
|
Item *outer_col;
|
|
|
|
/*
|
|
If we already determined that a complete match is enough based on schema
|
|
information, nothing can be better.
|
|
*/
|
|
if (strategy == COMPLETE_MATCH)
|
|
return COMPLETE_MATCH;
|
|
|
|
for (uint i= 0; i < item_in->left_expr->cols(); i++)
|
|
{
|
|
if (!bitmap_is_set(&partial_match_key_parts, i))
|
|
continue;
|
|
outer_col= item_in->left_expr->element_index(i);
|
|
/*
|
|
If column 'i' doesn't contain NULLs, and the corresponding outer reference
|
|
cannot have a NULL value, then 'i' is a non-nullable column.
|
|
*/
|
|
if (result_sink->get_null_count_of_col(i) == 0 && !outer_col->maybe_null)
|
|
{
|
|
bitmap_clear_bit(&partial_match_key_parts, i);
|
|
bitmap_set_bit(&non_null_key_parts, i);
|
|
--count_partial_match_columns;
|
|
}
|
|
if (result_sink->get_null_count_of_col(i) == tmp_table->file->stats.records)
|
|
++count_null_only_columns;
|
|
if (result_sink->get_null_count_of_col(i))
|
|
++count_columns_with_nulls;
|
|
}
|
|
|
|
/* If no column contains NULLs use regular hash index lookups. */
|
|
if (!count_partial_match_columns)
|
|
return COMPLETE_MATCH;
|
|
return PARTIAL_MATCH;
|
|
}
|
|
|
|
|
|
void
|
|
subselect_hash_sj_engine::choose_partial_match_strategy(
|
|
bool has_non_null_key, bool has_covering_null_row,
|
|
MY_BITMAP *partial_match_key_parts_arg)
|
|
{
|
|
ulonglong pm_buff_size;
|
|
|
|
DBUG_ASSERT(strategy == PARTIAL_MATCH);
|
|
/*
|
|
Choose according to global optimizer switch. If only one of the switches is
|
|
'ON', then the remaining strategy is the only possible one. The only cases
|
|
when this will be overriden is when the total size of all buffers for the
|
|
merge strategy is bigger than the 'rowid_merge_buff_size' system variable,
|
|
or if there isn't enough physical memory to allocate the buffers.
|
|
*/
|
|
if (!optimizer_flag(thd, OPTIMIZER_SWITCH_PARTIAL_MATCH_ROWID_MERGE) &&
|
|
optimizer_flag(thd, OPTIMIZER_SWITCH_PARTIAL_MATCH_TABLE_SCAN))
|
|
strategy= PARTIAL_MATCH_SCAN;
|
|
else if
|
|
( optimizer_flag(thd, OPTIMIZER_SWITCH_PARTIAL_MATCH_ROWID_MERGE) &&
|
|
!optimizer_flag(thd, OPTIMIZER_SWITCH_PARTIAL_MATCH_TABLE_SCAN))
|
|
strategy= PARTIAL_MATCH_MERGE;
|
|
|
|
/*
|
|
If both switches are ON, or both are OFF, we interpret that as "let the
|
|
optimizer decide". Perform a cost based choice between the two partial
|
|
matching strategies.
|
|
*/
|
|
/*
|
|
TIMOUR: the above interpretation of the switch values could be changed to:
|
|
- if both are ON - let the optimizer decide,
|
|
- if both are OFF - do not use partial matching, therefore do not use
|
|
materialization in non-top-level predicates.
|
|
The problem with this is that we know for sure if we need partial matching
|
|
only after the subquery is materialized, and this is too late to revert to
|
|
the IN=>EXISTS strategy.
|
|
*/
|
|
if (strategy == PARTIAL_MATCH)
|
|
{
|
|
/*
|
|
TIMOUR: Currently we use a super simplistic measure. This will be
|
|
addressed in a separate task.
|
|
*/
|
|
if (tmp_table->file->stats.records < 100)
|
|
strategy= PARTIAL_MATCH_SCAN;
|
|
else
|
|
strategy= PARTIAL_MATCH_MERGE;
|
|
}
|
|
|
|
/* Check if there is enough memory for the rowid merge strategy. */
|
|
if (strategy == PARTIAL_MATCH_MERGE)
|
|
{
|
|
pm_buff_size= rowid_merge_buff_size(has_non_null_key,
|
|
has_covering_null_row,
|
|
partial_match_key_parts_arg);
|
|
if (pm_buff_size > thd->variables.rowid_merge_buff_size)
|
|
strategy= PARTIAL_MATCH_SCAN;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
Compute the memory size of all buffers proportional to the number of rows
|
|
in tmp_table.
|
|
|
|
@details
|
|
If the result is bigger than thd->variables.rowid_merge_buff_size, partial
|
|
matching via merging is not applicable.
|
|
*/
|
|
|
|
ulonglong subselect_hash_sj_engine::rowid_merge_buff_size(
|
|
bool has_non_null_key, bool has_covering_null_row,
|
|
MY_BITMAP *partial_match_key_parts)
|
|
{
|
|
/* Total size of all buffers used by partial matching. */
|
|
ulonglong buff_size;
|
|
ha_rows row_count= tmp_table->file->stats.records;
|
|
uint rowid_length= tmp_table->file->ref_length;
|
|
select_materialize_with_stats *result_sink=
|
|
(select_materialize_with_stats *) result;
|
|
ha_rows max_null_row;
|
|
|
|
/* Size of the subselect_rowid_merge_engine::row_num_to_rowid buffer. */
|
|
buff_size= row_count * rowid_length * sizeof(uchar);
|
|
|
|
if (has_non_null_key)
|
|
{
|
|
/* Add the size of Ordered_key::key_buff of the only non-NULL key. */
|
|
buff_size+= row_count * sizeof(rownum_t);
|
|
}
|
|
|
|
if (!has_covering_null_row)
|
|
{
|
|
for (uint i= 0; i < partial_match_key_parts->n_bits; i++)
|
|
{
|
|
if (!bitmap_is_set(partial_match_key_parts, i) ||
|
|
result_sink->get_null_count_of_col(i) == row_count)
|
|
continue; /* In these cases we wouldn't construct Ordered keys. */
|
|
|
|
/* Add the size of Ordered_key::key_buff */
|
|
buff_size+= (row_count - result_sink->get_null_count_of_col(i)) *
|
|
sizeof(rownum_t);
|
|
/* Add the size of Ordered_key::null_key */
|
|
max_null_row= result_sink->get_max_null_of_col(i);
|
|
if (max_null_row >= UINT_MAX)
|
|
{
|
|
/*
|
|
There can be at most UINT_MAX bits in a MY_BITMAP that is used to
|
|
store NULLs in an Ordered_key. Return a number of bytes bigger than
|
|
the maximum allowed memory buffer for partial matching to disable
|
|
the rowid merge strategy.
|
|
*/
|
|
return ULONGLONG_MAX;
|
|
}
|
|
buff_size+= bitmap_buffer_size(max_null_row);
|
|
}
|
|
}
|
|
|
|
return buff_size;
|
|
}
|
|
|
|
|
|
/*
|
|
Initialize a MY_BITMAP with a buffer allocated on the current
|
|
memory root.
|
|
TIMOUR: move to bitmap C file?
|
|
*/
|
|
|
|
static my_bool
|
|
my_bitmap_init_memroot(MY_BITMAP *map, uint n_bits, MEM_ROOT *mem_root)
|
|
{
|
|
my_bitmap_map *bitmap_buf;
|
|
|
|
if (!(bitmap_buf= (my_bitmap_map*) alloc_root(mem_root,
|
|
bitmap_buffer_size(n_bits))) ||
|
|
my_bitmap_init(map, bitmap_buf, n_bits, FALSE))
|
|
return TRUE;
|
|
bitmap_clear_all(map);
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/**
|
|
Create all structures needed for IN execution that can live between PS
|
|
reexecution.
|
|
|
|
@param tmp_columns the items that produce the data for the temp table
|
|
@param subquery_id subquery's identifier (to make "<subquery%d>" name for
|
|
EXPLAIN)
|
|
|
|
@details
|
|
- Create a temporary table to store the result of the IN subquery. The
|
|
temporary table has one hash index on all its columns.
|
|
- Create a new result sink that sends the result stream of the subquery to
|
|
the temporary table,
|
|
|
|
@notice:
|
|
Currently Item_subselect::init() already chooses and creates at parse
|
|
time an engine with a corresponding JOIN to execute the subquery.
|
|
|
|
@retval TRUE if error
|
|
@retval FALSE otherwise
|
|
*/
|
|
|
|
bool subselect_hash_sj_engine::init(List<Item> *tmp_columns, uint subquery_id)
|
|
{
|
|
select_union *result_sink;
|
|
/* Options to create_tmp_table. */
|
|
ulonglong tmp_create_options= thd->variables.option_bits | TMP_TABLE_ALL_COLUMNS;
|
|
/* | TMP_TABLE_FORCE_MYISAM; TIMOUR: force MYISAM */
|
|
|
|
DBUG_ENTER("subselect_hash_sj_engine::init");
|
|
|
|
if (my_bitmap_init_memroot(&non_null_key_parts, tmp_columns->elements,
|
|
thd->mem_root) ||
|
|
my_bitmap_init_memroot(&partial_match_key_parts, tmp_columns->elements,
|
|
thd->mem_root))
|
|
DBUG_RETURN(TRUE);
|
|
|
|
/*
|
|
Create and initialize a select result interceptor that stores the
|
|
result stream in a temporary table. The temporary table itself is
|
|
managed (created/filled/etc) internally by the interceptor.
|
|
*/
|
|
/*
|
|
TIMOUR:
|
|
Select a more efficient result sink when we know there is no need to collect
|
|
data statistics.
|
|
|
|
if (strategy == COMPLETE_MATCH)
|
|
{
|
|
if (!(result= new select_union))
|
|
DBUG_RETURN(TRUE);
|
|
}
|
|
else if (strategy == PARTIAL_MATCH)
|
|
{
|
|
if (!(result= new select_materialize_with_stats))
|
|
DBUG_RETURN(TRUE);
|
|
}
|
|
*/
|
|
if (!(result_sink= new (thd->mem_root) select_materialize_with_stats(thd)))
|
|
DBUG_RETURN(TRUE);
|
|
|
|
char buf[32];
|
|
uint len= my_snprintf(buf, sizeof(buf), "<subquery%d>", subquery_id);
|
|
char *name;
|
|
if (!(name= (char*)thd->alloc(len + 1)))
|
|
DBUG_RETURN(TRUE);
|
|
memcpy(name, buf, len+1);
|
|
|
|
result_sink->get_tmp_table_param()->materialized_subquery= true;
|
|
if (item->substype() == Item_subselect::IN_SUBS &&
|
|
((Item_in_subselect*)item)->is_jtbm_merged)
|
|
{
|
|
result_sink->get_tmp_table_param()->force_not_null_cols= true;
|
|
}
|
|
if (result_sink->create_result_table(thd, tmp_columns, TRUE,
|
|
tmp_create_options,
|
|
name, TRUE, TRUE))
|
|
DBUG_RETURN(TRUE);
|
|
|
|
tmp_table= result_sink->table;
|
|
result= result_sink;
|
|
|
|
/*
|
|
If the subquery has blobs, or the total key lenght is bigger than
|
|
some length, or the total number of key parts is more than the
|
|
allowed maximum (currently MAX_REF_PARTS == 16), then the created
|
|
index cannot be used for lookups and we can't use hash semi
|
|
join. If this is the case, delete the temporary table since it
|
|
will not be used, and tell the caller we failed to initialize the
|
|
engine.
|
|
*/
|
|
if (tmp_table->s->keys == 0)
|
|
{
|
|
//fprintf(stderr, "Q: %s\n", current_thd->query());
|
|
DBUG_ASSERT(0);
|
|
DBUG_ASSERT(
|
|
tmp_table->s->uniques ||
|
|
tmp_table->key_info->key_length >= tmp_table->file->max_key_length() ||
|
|
tmp_table->key_info->user_defined_key_parts >
|
|
tmp_table->file->max_key_parts());
|
|
free_tmp_table(thd, tmp_table);
|
|
tmp_table= NULL;
|
|
delete result;
|
|
result= NULL;
|
|
DBUG_RETURN(TRUE);
|
|
}
|
|
|
|
/*
|
|
Make sure there is only one index on the temp table, and it doesn't have
|
|
the extra key part created when s->uniques > 0.
|
|
*/
|
|
DBUG_ASSERT(tmp_table->s->keys == 1 &&
|
|
((Item_in_subselect *) item)->left_expr->cols() ==
|
|
tmp_table->key_info->user_defined_key_parts);
|
|
|
|
if (make_semi_join_conds() ||
|
|
/* A unique_engine is used both for complete and partial matching. */
|
|
!(lookup_engine= make_unique_engine()))
|
|
DBUG_RETURN(TRUE);
|
|
|
|
/*
|
|
Repeat name resolution for 'cond' since cond is not part of any
|
|
clause of the query, and it is not 'fixed' during JOIN::prepare.
|
|
*/
|
|
if (semi_join_conds && !semi_join_conds->fixed &&
|
|
semi_join_conds->fix_fields(thd, (Item**)&semi_join_conds))
|
|
DBUG_RETURN(TRUE);
|
|
/* Let our engine reuse this query plan for materialization. */
|
|
materialize_join= materialize_engine->join;
|
|
materialize_join->change_result(result, NULL);
|
|
|
|
DBUG_RETURN(FALSE);
|
|
}
|
|
|
|
|
|
/*
|
|
Create an artificial condition to post-filter those rows matched by index
|
|
lookups that cannot be distinguished by the index lookup procedure.
|
|
|
|
@notes
|
|
The need for post-filtering may occur e.g. because of
|
|
truncation. Prepared statements execution requires that fix_fields is
|
|
called for every execution. In order to call fix_fields we need to
|
|
create a Name_resolution_context and a corresponding TABLE_LIST for
|
|
the temporary table for the subquery, so that all column references
|
|
to the materialized subquery table can be resolved correctly.
|
|
|
|
@returns
|
|
@retval TRUE memory allocation error occurred
|
|
@retval FALSE the conditions were created and resolved (fixed)
|
|
*/
|
|
|
|
bool subselect_hash_sj_engine::make_semi_join_conds()
|
|
{
|
|
/*
|
|
Table reference for tmp_table that is used to resolve column references
|
|
(Item_fields) to columns in tmp_table.
|
|
*/
|
|
TABLE_LIST *tmp_table_ref;
|
|
/* Name resolution context for all tmp_table columns created below. */
|
|
Name_resolution_context *context;
|
|
Item_in_subselect *item_in= (Item_in_subselect *) item;
|
|
|
|
DBUG_ENTER("subselect_hash_sj_engine::make_semi_join_conds");
|
|
DBUG_ASSERT(semi_join_conds == NULL);
|
|
|
|
if (!(semi_join_conds= new (thd->mem_root) Item_cond_and(thd)))
|
|
DBUG_RETURN(TRUE);
|
|
|
|
if (!(tmp_table_ref= (TABLE_LIST*) thd->alloc(sizeof(TABLE_LIST))))
|
|
DBUG_RETURN(TRUE);
|
|
|
|
tmp_table_ref->init_one_table(STRING_WITH_LEN(""),
|
|
tmp_table->alias.c_ptr(),
|
|
tmp_table->alias.length(),
|
|
NULL, TL_READ);
|
|
tmp_table_ref->table= tmp_table;
|
|
|
|
context= new Name_resolution_context;
|
|
context->init();
|
|
context->first_name_resolution_table=
|
|
context->last_name_resolution_table= tmp_table_ref;
|
|
semi_join_conds_context= context;
|
|
|
|
for (uint i= 0; i < item_in->left_expr->cols(); i++)
|
|
{
|
|
/* New equi-join condition for the current column. */
|
|
Item_func_eq *eq_cond;
|
|
/* Item for the corresponding field from the materialized temp table. */
|
|
Item_field *right_col_item;
|
|
|
|
if (!(right_col_item= new (thd->mem_root)
|
|
Item_temptable_field(thd, context, tmp_table->field[i])) ||
|
|
!(eq_cond= new (thd->mem_root)
|
|
Item_func_eq(thd, item_in->left_expr->element_index(i),
|
|
right_col_item)) ||
|
|
(((Item_cond_and*)semi_join_conds)->add(eq_cond, thd->mem_root)))
|
|
{
|
|
delete semi_join_conds;
|
|
semi_join_conds= NULL;
|
|
DBUG_RETURN(TRUE);
|
|
}
|
|
}
|
|
if (semi_join_conds->fix_fields(thd, (Item**)&semi_join_conds))
|
|
DBUG_RETURN(TRUE);
|
|
|
|
DBUG_RETURN(FALSE);
|
|
}
|
|
|
|
|
|
/**
|
|
Create a new uniquesubquery engine for the execution of an IN predicate.
|
|
|
|
@details
|
|
Create and initialize a new JOIN_TAB, and Table_ref objects to perform
|
|
lookups into the indexed temporary table.
|
|
|
|
@retval A new subselect_hash_sj_engine object
|
|
@retval NULL if a memory allocation error occurs
|
|
*/
|
|
|
|
subselect_uniquesubquery_engine*
|
|
subselect_hash_sj_engine::make_unique_engine()
|
|
{
|
|
Item_in_subselect *item_in= (Item_in_subselect *) item;
|
|
Item_iterator_row it(item_in->left_expr);
|
|
/* The only index on the temporary table. */
|
|
KEY *tmp_key= tmp_table->key_info;
|
|
JOIN_TAB *tab;
|
|
|
|
DBUG_ENTER("subselect_hash_sj_engine::make_unique_engine");
|
|
|
|
/*
|
|
Create and initialize the JOIN_TAB that represents an index lookup
|
|
plan operator into the materialized subquery result. Notice that:
|
|
- this JOIN_TAB has no corresponding JOIN (and doesn't need one), and
|
|
- here we initialize only those members that are used by
|
|
subselect_uniquesubquery_engine, so these objects are incomplete.
|
|
*/
|
|
if (!(tab= (JOIN_TAB*) thd->alloc(sizeof(JOIN_TAB))))
|
|
DBUG_RETURN(NULL);
|
|
|
|
tab->table= tmp_table;
|
|
tab->preread_init_done= FALSE;
|
|
tab->ref.tmp_table_index_lookup_init(thd, tmp_key, it, FALSE);
|
|
|
|
DBUG_RETURN(new subselect_uniquesubquery_engine(thd, tab, item,
|
|
semi_join_conds));
|
|
}
|
|
|
|
|
|
subselect_hash_sj_engine::~subselect_hash_sj_engine()
|
|
{
|
|
delete lookup_engine;
|
|
delete result;
|
|
if (tmp_table)
|
|
free_tmp_table(thd, tmp_table);
|
|
}
|
|
|
|
|
|
int subselect_hash_sj_engine::prepare()
|
|
{
|
|
/*
|
|
Create and optimize the JOIN that will be used to materialize
|
|
the subquery if not yet created.
|
|
*/
|
|
return materialize_engine->prepare();
|
|
}
|
|
|
|
|
|
/**
|
|
Cleanup performed after each PS execution.
|
|
|
|
@details
|
|
Called in the end of JOIN::prepare for PS from Item_subselect::cleanup.
|
|
*/
|
|
|
|
void subselect_hash_sj_engine::cleanup()
|
|
{
|
|
enum_engine_type lookup_engine_type= lookup_engine->engine_type();
|
|
is_materialized= FALSE;
|
|
bitmap_clear_all(&non_null_key_parts);
|
|
bitmap_clear_all(&partial_match_key_parts);
|
|
count_partial_match_columns= 0;
|
|
count_null_only_columns= 0;
|
|
strategy= UNDEFINED;
|
|
materialize_engine->cleanup();
|
|
/*
|
|
Restore the original Item_in_subselect engine. This engine is created once
|
|
at parse time and stored across executions, while all other materialization
|
|
related engines are created and chosen for each execution.
|
|
*/
|
|
((Item_in_subselect *) item)->engine= materialize_engine;
|
|
if (lookup_engine_type == TABLE_SCAN_ENGINE ||
|
|
lookup_engine_type == ROWID_MERGE_ENGINE)
|
|
{
|
|
subselect_engine *inner_lookup_engine;
|
|
inner_lookup_engine=
|
|
((subselect_partial_match_engine*) lookup_engine)->lookup_engine;
|
|
/*
|
|
Partial match engines are recreated for each PS execution inside
|
|
subselect_hash_sj_engine::exec().
|
|
*/
|
|
delete lookup_engine;
|
|
lookup_engine= inner_lookup_engine;
|
|
}
|
|
DBUG_ASSERT(lookup_engine->engine_type() == UNIQUESUBQUERY_ENGINE);
|
|
lookup_engine->cleanup();
|
|
result->cleanup(); /* Resets the temp table as well. */
|
|
DBUG_ASSERT(tmp_table);
|
|
free_tmp_table(thd, tmp_table);
|
|
tmp_table= NULL;
|
|
}
|
|
|
|
|
|
/*
|
|
Get fanout produced by tables specified in the table_map
|
|
*/
|
|
|
|
double get_fanout_with_deps(JOIN *join, table_map tset)
|
|
{
|
|
/* Handle the case of "Impossible WHERE" */
|
|
if (join->table_count == 0)
|
|
return 0.0;
|
|
|
|
/* First, recursively get all tables we depend on */
|
|
table_map deps_to_check= tset;
|
|
table_map checked_deps= 0;
|
|
table_map further_deps;
|
|
do
|
|
{
|
|
further_deps= 0;
|
|
Table_map_iterator tm_it(deps_to_check);
|
|
int tableno;
|
|
while ((tableno = tm_it.next_bit()) != Table_map_iterator::BITMAP_END)
|
|
{
|
|
/* get tableno's dependency tables that are not in needed_set */
|
|
further_deps |= join->map2table[tableno]->ref.depend_map & ~checked_deps;
|
|
}
|
|
|
|
checked_deps |= deps_to_check;
|
|
deps_to_check= further_deps;
|
|
} while (further_deps != 0);
|
|
|
|
|
|
/* Now, walk the join order and calculate the fanout */
|
|
double fanout= 1;
|
|
for (JOIN_TAB *tab= first_top_level_tab(join, WITHOUT_CONST_TABLES); tab;
|
|
tab= next_top_level_tab(join, tab))
|
|
{
|
|
/*
|
|
Ignore SJM nests. They have tab->table==NULL. There is no point to walk
|
|
inside them, because GROUP BY clause cannot refer to tables from within
|
|
subquery.
|
|
*/
|
|
if (!tab->is_sjm_nest() && (tab->table->map & checked_deps) &&
|
|
!tab->emb_sj_nest &&
|
|
tab->records_read != 0)
|
|
{
|
|
fanout *= tab->records_read;
|
|
}
|
|
}
|
|
return fanout;
|
|
}
|
|
|
|
|
|
#if 0
|
|
void check_out_index_stats(JOIN *join)
|
|
{
|
|
ORDER *order;
|
|
uint n_order_items;
|
|
|
|
/*
|
|
First, collect the keys that we can use in each table.
|
|
We can use a key if
|
|
- all tables refer to it.
|
|
*/
|
|
key_map key_start_use[MAX_TABLES];
|
|
key_map key_infix_use[MAX_TABLES];
|
|
table_map key_used=0;
|
|
table_map non_key_used= 0;
|
|
|
|
bzero(&key_start_use, sizeof(key_start_use)); //psergey-todo: safe initialization!
|
|
bzero(&key_infix_use, sizeof(key_infix_use));
|
|
|
|
for (order= join->group_list; order; order= order->next)
|
|
{
|
|
Item *item= order->item[0];
|
|
|
|
if (item->real_type() == Item::FIELD_ITEM)
|
|
{
|
|
if (item->used_tables() & OUTER_REF_TABLE_BIT)
|
|
continue; /* outside references are like constants for us */
|
|
|
|
Field *field= ((Item_field*)item->real_item())->field;
|
|
uint table_no= field->table->tablenr;
|
|
if (!(non_key_used && table_map(1) << table_no) &&
|
|
!field->part_of_key.is_clear_all())
|
|
{
|
|
key_map infix_map= field->part_of_key;
|
|
infix_map.subtract(field->key_start);
|
|
key_start_use[table_no].merge(field->key_start);
|
|
key_infix_use[table_no].merge(infix_map);
|
|
key_used |= table_no;
|
|
}
|
|
continue;
|
|
}
|
|
/*
|
|
Note: the below will cause clauses like GROUP BY YEAR(date) not to be
|
|
handled.
|
|
*/
|
|
non_key_used |= item->used_tables();
|
|
}
|
|
|
|
Table_map_iterator tm_it(key_used & ~non_key_used);
|
|
int tableno;
|
|
while ((tableno = tm_it.next_bit()) != Table_map_iterator::BITMAP_END)
|
|
{
|
|
key_map::iterator key_it(key_start_use);
|
|
int keyno;
|
|
while ((keyno = tm_it.next_bit()) != key_map::iterator::BITMAP_END)
|
|
{
|
|
for (order= join->group_list; order; order= order->next)
|
|
{
|
|
Item *item= order->item[0];
|
|
if (item->used_tables() & (table_map(1) << tableno))
|
|
{
|
|
DBUG_ASSERT(item->real_type() == Item::FIELD_ITEM);
|
|
}
|
|
}
|
|
/*
|
|
if (continuation)
|
|
{
|
|
walk through list and find which key parts are occupied;
|
|
// note that the above can't be made any faster.
|
|
}
|
|
else
|
|
use rec_per_key[0];
|
|
|
|
find out the cardinality.
|
|
check if cardinality decreases if we use it;
|
|
*/
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
/*
|
|
Get an estimate of how many records will be produced after the GROUP BY
|
|
operation.
|
|
|
|
@param join Join we're operating on
|
|
@param join_op_rows How many records will be produced by the join
|
|
operations (this is what join optimizer produces)
|
|
|
|
@seealso
|
|
See also optimize_semijoin_nests(), grep for "Adjust output cardinality
|
|
estimates". Very similar code there that is not joined with this one
|
|
because we operate on different data structs and too much effort is
|
|
needed to abstract them out.
|
|
|
|
@return
|
|
Number of records we expect to get after the GROUP BY operation
|
|
*/
|
|
|
|
double get_post_group_estimate(JOIN* join, double join_op_rows)
|
|
{
|
|
table_map tables_in_group_list= table_map(0);
|
|
|
|
/* Find out which tables are used in GROUP BY list */
|
|
for (ORDER *order= join->group_list_for_estimates; order; order= order->next)
|
|
{
|
|
Item *item= order->item[0];
|
|
table_map item_used_tables= item->used_tables();
|
|
if (item_used_tables & RAND_TABLE_BIT)
|
|
{
|
|
/* Each join output record will be in its own group */
|
|
return join_op_rows;
|
|
}
|
|
tables_in_group_list|= item_used_tables;
|
|
}
|
|
tables_in_group_list &= ~PSEUDO_TABLE_BITS;
|
|
|
|
/*
|
|
Use join fanouts to calculate the max. number of records in the group-list
|
|
*/
|
|
double fanout_rows[MAX_KEY];
|
|
bzero(&fanout_rows, sizeof(fanout_rows));
|
|
double out_rows;
|
|
|
|
out_rows= get_fanout_with_deps(join, tables_in_group_list);
|
|
|
|
#if 0
|
|
/* The following will be needed when making use of index stats: */
|
|
/*
|
|
Also generate max. number of records for each of the tables mentioned
|
|
in the group-list. We'll use that a baseline number that we'll try to
|
|
reduce by using
|
|
- #table-records
|
|
- index statistics.
|
|
*/
|
|
Table_map_iterator tm_it(tables_in_group_list);
|
|
int tableno;
|
|
while ((tableno = tm_it.next_bit()) != Table_map_iterator::BITMAP_END)
|
|
{
|
|
fanout_rows[tableno]= get_fanout_with_deps(join, table_map(1) << tableno);
|
|
}
|
|
|
|
/*
|
|
Try to bring down estimates using index statistics.
|
|
*/
|
|
//check_out_index_stats(join);
|
|
#endif
|
|
|
|
return out_rows;
|
|
}
|
|
|
|
|
|
/**
|
|
Execute a subquery IN predicate via materialization.
|
|
|
|
@details
|
|
If needed materialize the subquery into a temporary table, then
|
|
copmpute the predicate via a lookup into this table.
|
|
|
|
@retval TRUE if error
|
|
@retval FALSE otherwise
|
|
*/
|
|
|
|
int subselect_hash_sj_engine::exec()
|
|
{
|
|
Item_in_subselect *item_in= (Item_in_subselect *) item;
|
|
SELECT_LEX *save_select= thd->lex->current_select;
|
|
subselect_partial_match_engine *pm_engine= NULL;
|
|
int res= 0;
|
|
|
|
DBUG_ENTER("subselect_hash_sj_engine::exec");
|
|
|
|
/*
|
|
Optimize and materialize the subquery during the first execution of
|
|
the subquery predicate.
|
|
*/
|
|
thd->lex->current_select= materialize_engine->select_lex;
|
|
/* The subquery should be optimized, and materialized only once. */
|
|
DBUG_ASSERT(materialize_join->optimized && !is_materialized);
|
|
materialize_join->exec();
|
|
if ((res= MY_TEST(materialize_join->error || thd->is_fatal_error ||
|
|
thd->is_error())))
|
|
goto err;
|
|
|
|
/*
|
|
TODO:
|
|
- Unlock all subquery tables as we don't need them. To implement this
|
|
we need to add new functionality to JOIN::join_free that can unlock
|
|
all tables in a subquery (and all its subqueries).
|
|
- The temp table used for grouping in the subquery can be freed
|
|
immediately after materialization (yet it's done together with
|
|
unlocking).
|
|
*/
|
|
is_materialized= TRUE;
|
|
/*
|
|
If the subquery returned no rows, the temporary table is empty, so we know
|
|
directly that the result of IN is FALSE. We first update the table
|
|
statistics, then we test if the temporary table for the query result is
|
|
empty.
|
|
*/
|
|
tmp_table->file->info(HA_STATUS_VARIABLE);
|
|
if (!tmp_table->file->stats.records)
|
|
{
|
|
/* The value of IN will not change during this execution. */
|
|
item_in->reset();
|
|
item_in->make_const();
|
|
item_in->set_first_execution();
|
|
DBUG_RETURN(FALSE);
|
|
}
|
|
|
|
/*
|
|
TIMOUR: The schema-based analysis for partial matching can be done once for
|
|
prepared statement and remembered. It is done here to remove the need to
|
|
save/restore all related variables between each re-execution, thus making
|
|
the code simpler.
|
|
*/
|
|
strategy= get_strategy_using_schema();
|
|
/* This call may discover that we don't need partial matching at all. */
|
|
strategy= get_strategy_using_data();
|
|
if (strategy == PARTIAL_MATCH)
|
|
{
|
|
uint count_pm_keys; /* Total number of keys needed for partial matching. */
|
|
MY_BITMAP *nn_key_parts= NULL; /* Key parts of the only non-NULL index. */
|
|
uint count_non_null_columns= 0; /* Number of columns in nn_key_parts. */
|
|
bool has_covering_null_row;
|
|
bool has_covering_null_columns;
|
|
select_materialize_with_stats *result_sink=
|
|
(select_materialize_with_stats *) result;
|
|
uint field_count= tmp_table->s->fields;
|
|
|
|
if (count_partial_match_columns < field_count)
|
|
{
|
|
nn_key_parts= &non_null_key_parts;
|
|
count_non_null_columns= bitmap_bits_set(nn_key_parts);
|
|
}
|
|
has_covering_null_row= (result_sink->get_max_nulls_in_row() == field_count);
|
|
has_covering_null_columns= (count_non_null_columns +
|
|
count_null_only_columns == field_count);
|
|
|
|
if (has_covering_null_row && has_covering_null_columns)
|
|
{
|
|
/*
|
|
The whole table consist of only NULL values. The result of IN is
|
|
a constant UNKNOWN.
|
|
*/
|
|
DBUG_ASSERT(tmp_table->file->stats.records == 1);
|
|
item_in->value= 0;
|
|
item_in->null_value= 1;
|
|
item_in->make_const();
|
|
item_in->set_first_execution();
|
|
DBUG_RETURN(FALSE);
|
|
}
|
|
|
|
if (has_covering_null_row)
|
|
{
|
|
DBUG_ASSERT(count_partial_match_columns = field_count);
|
|
count_pm_keys= 0;
|
|
}
|
|
else if (has_covering_null_columns)
|
|
count_pm_keys= 1;
|
|
else
|
|
count_pm_keys= count_partial_match_columns - count_null_only_columns +
|
|
(nn_key_parts ? 1 : 0);
|
|
|
|
choose_partial_match_strategy(MY_TEST(nn_key_parts),
|
|
has_covering_null_row,
|
|
&partial_match_key_parts);
|
|
DBUG_ASSERT(strategy == PARTIAL_MATCH_MERGE ||
|
|
strategy == PARTIAL_MATCH_SCAN);
|
|
if (strategy == PARTIAL_MATCH_MERGE)
|
|
{
|
|
pm_engine=
|
|
new subselect_rowid_merge_engine(thd, (subselect_uniquesubquery_engine*)
|
|
lookup_engine, tmp_table,
|
|
count_pm_keys,
|
|
has_covering_null_row,
|
|
has_covering_null_columns,
|
|
count_columns_with_nulls,
|
|
item, result,
|
|
semi_join_conds->argument_list());
|
|
if (!pm_engine ||
|
|
((subselect_rowid_merge_engine*) pm_engine)->
|
|
init(nn_key_parts, &partial_match_key_parts))
|
|
{
|
|
/*
|
|
The call to init() would fail if there was not enough memory to allocate
|
|
all buffers for the rowid merge strategy. In this case revert to table
|
|
scanning which doesn't need any big buffers.
|
|
*/
|
|
delete pm_engine;
|
|
pm_engine= NULL;
|
|
strategy= PARTIAL_MATCH_SCAN;
|
|
}
|
|
}
|
|
|
|
if (strategy == PARTIAL_MATCH_SCAN)
|
|
{
|
|
if (!(pm_engine=
|
|
new subselect_table_scan_engine(thd, (subselect_uniquesubquery_engine*)
|
|
lookup_engine, tmp_table,
|
|
item, result,
|
|
semi_join_conds->argument_list(),
|
|
has_covering_null_row,
|
|
has_covering_null_columns,
|
|
count_columns_with_nulls)))
|
|
{
|
|
/* This is an irrecoverable error. */
|
|
res= 1;
|
|
goto err;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (pm_engine)
|
|
lookup_engine= pm_engine;
|
|
item_in->change_engine(lookup_engine);
|
|
|
|
err:
|
|
thd->lex->current_select= save_select;
|
|
DBUG_RETURN(res);
|
|
}
|
|
|
|
|
|
/**
|
|
Print the state of this engine into a string for debugging and views.
|
|
*/
|
|
|
|
void subselect_hash_sj_engine::print(String *str, enum_query_type query_type)
|
|
{
|
|
str->append(STRING_WITH_LEN(" <materialize> ("));
|
|
materialize_engine->print(str, query_type);
|
|
str->append(STRING_WITH_LEN(" ), "));
|
|
|
|
if (lookup_engine)
|
|
lookup_engine->print(str, query_type);
|
|
else
|
|
str->append(STRING_WITH_LEN(
|
|
"<engine selected at execution time>"
|
|
));
|
|
}
|
|
|
|
void subselect_hash_sj_engine::fix_length_and_dec(Item_cache** row)
|
|
{
|
|
DBUG_ASSERT(FALSE);
|
|
}
|
|
|
|
void subselect_hash_sj_engine::exclude()
|
|
{
|
|
DBUG_ASSERT(FALSE);
|
|
}
|
|
|
|
bool subselect_hash_sj_engine::no_tables()
|
|
{
|
|
DBUG_ASSERT(FALSE);
|
|
return FALSE;
|
|
}
|
|
|
|
bool subselect_hash_sj_engine::change_result(Item_subselect *si,
|
|
select_result_interceptor *res,
|
|
bool temp __attribute__((unused)))
|
|
{
|
|
DBUG_ASSERT(FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
Ordered_key::Ordered_key(uint keyid_arg, TABLE *tbl_arg, Item *search_key_arg,
|
|
ha_rows null_count_arg, ha_rows min_null_row_arg,
|
|
ha_rows max_null_row_arg, uchar *row_num_to_rowid_arg)
|
|
: keyid(keyid_arg), tbl(tbl_arg), search_key(search_key_arg),
|
|
row_num_to_rowid(row_num_to_rowid_arg), null_count(null_count_arg)
|
|
{
|
|
DBUG_ASSERT(tbl->file->stats.records > null_count);
|
|
key_buff_elements= tbl->file->stats.records - null_count;
|
|
cur_key_idx= HA_POS_ERROR;
|
|
|
|
DBUG_ASSERT((null_count && min_null_row_arg && max_null_row_arg) ||
|
|
(!null_count && !min_null_row_arg && !max_null_row_arg));
|
|
if (null_count)
|
|
{
|
|
/* The counters are 1-based, for key access we need 0-based indexes. */
|
|
min_null_row= min_null_row_arg - 1;
|
|
max_null_row= max_null_row_arg - 1;
|
|
}
|
|
else
|
|
min_null_row= max_null_row= 0;
|
|
}
|
|
|
|
|
|
Ordered_key::~Ordered_key()
|
|
{
|
|
my_free(key_buff);
|
|
my_bitmap_free(&null_key);
|
|
}
|
|
|
|
|
|
/*
|
|
Cleanup that needs to be done for each PS (re)execution.
|
|
*/
|
|
|
|
void Ordered_key::cleanup()
|
|
{
|
|
/*
|
|
Currently these keys are recreated for each PS re-execution, thus
|
|
there is nothing to cleanup, the whole object goes away after execution
|
|
is over. All handler related initialization/deinitialization is done by
|
|
the parent subselect_rowid_merge_engine object.
|
|
*/
|
|
}
|
|
|
|
|
|
/*
|
|
Initialize a multi-column index.
|
|
*/
|
|
|
|
bool Ordered_key::init(MY_BITMAP *columns_to_index)
|
|
{
|
|
THD *thd= tbl->in_use;
|
|
uint cur_key_col= 0;
|
|
Item_field *cur_tmp_field;
|
|
Item_func_lt *fn_less_than;
|
|
|
|
key_column_count= bitmap_bits_set(columns_to_index);
|
|
key_columns= (Item_field**) thd->alloc(key_column_count *
|
|
sizeof(Item_field*));
|
|
compare_pred= (Item_func_lt**) thd->alloc(key_column_count *
|
|
sizeof(Item_func_lt*));
|
|
|
|
if (!key_columns || !compare_pred)
|
|
return TRUE; /* Revert to table scan partial match. */
|
|
|
|
for (uint i= 0; i < columns_to_index->n_bits; i++)
|
|
{
|
|
if (!bitmap_is_set(columns_to_index, i))
|
|
continue;
|
|
cur_tmp_field= new (thd->mem_root) Item_field(thd, tbl->field[i]);
|
|
/* Create the predicate (tmp_column[i] < outer_ref[i]). */
|
|
fn_less_than= new (thd->mem_root) Item_func_lt(thd, cur_tmp_field,
|
|
search_key->element_index(i));
|
|
fn_less_than->fix_fields(thd, (Item**) &fn_less_than);
|
|
key_columns[cur_key_col]= cur_tmp_field;
|
|
compare_pred[cur_key_col]= fn_less_than;
|
|
++cur_key_col;
|
|
}
|
|
|
|
if (alloc_keys_buffers())
|
|
{
|
|
/* TIMOUR revert to partial match via table scan. */
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*
|
|
Initialize a single-column index.
|
|
*/
|
|
|
|
bool Ordered_key::init(int col_idx)
|
|
{
|
|
THD *thd= tbl->in_use;
|
|
|
|
key_column_count= 1;
|
|
|
|
// TIMOUR: check for mem allocation err, revert to scan
|
|
|
|
key_columns= (Item_field**) thd->alloc(sizeof(Item_field*));
|
|
compare_pred= (Item_func_lt**) thd->alloc(sizeof(Item_func_lt*));
|
|
|
|
key_columns[0]= new (thd->mem_root) Item_field(thd, tbl->field[col_idx]);
|
|
/* Create the predicate (tmp_column[i] < outer_ref[i]). */
|
|
compare_pred[0]= new (thd->mem_root) Item_func_lt(thd, key_columns[0],
|
|
search_key->element_index(col_idx));
|
|
compare_pred[0]->fix_fields(thd, (Item**)&compare_pred[0]);
|
|
|
|
if (alloc_keys_buffers())
|
|
{
|
|
/* TIMOUR revert to partial match via table scan. */
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*
|
|
Allocate the buffers for both the row number, and the NULL-bitmap indexes.
|
|
*/
|
|
|
|
bool Ordered_key::alloc_keys_buffers()
|
|
{
|
|
DBUG_ASSERT(key_buff_elements > 0);
|
|
|
|
if (!(key_buff= (rownum_t*) my_malloc((size_t)(key_buff_elements *
|
|
sizeof(rownum_t)), MYF(MY_WME | MY_THREAD_SPECIFIC))))
|
|
return TRUE;
|
|
|
|
/*
|
|
TIMOUR: it is enough to create bitmaps with size
|
|
(max_null_row - min_null_row), and then use min_null_row as
|
|
lookup offset.
|
|
*/
|
|
/* Notice that max_null_row is max array index, we need count, so +1. */
|
|
if (my_bitmap_init(&null_key, NULL, (uint)(max_null_row + 1), FALSE))
|
|
return TRUE;
|
|
|
|
cur_key_idx= HA_POS_ERROR;
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*
|
|
Quick sort comparison function that compares two rows of the same table
|
|
indentfied with their row numbers.
|
|
|
|
@retval -1
|
|
@retval 0
|
|
@retval +1
|
|
*/
|
|
|
|
int
|
|
Ordered_key::cmp_keys_by_row_data(ha_rows a, ha_rows b)
|
|
{
|
|
uchar *rowid_a, *rowid_b;
|
|
int __attribute__((unused)) error;
|
|
int cmp_res;
|
|
/* The length in bytes of the rowids (positions) of tmp_table. */
|
|
uint rowid_length= tbl->file->ref_length;
|
|
|
|
if (a == b)
|
|
return 0;
|
|
/* Get the corresponding rowids. */
|
|
rowid_a= row_num_to_rowid + a * rowid_length;
|
|
rowid_b= row_num_to_rowid + b * rowid_length;
|
|
/* Fetch the rows for comparison. */
|
|
if ((error= tbl->file->ha_rnd_pos(tbl->record[0], rowid_a)))
|
|
{
|
|
/* purecov: begin inspected */
|
|
tbl->file->print_error(error, MYF(ME_FATALERROR)); // Sets fatal_error
|
|
return 0;
|
|
/* purecov: end */
|
|
}
|
|
if ((error= tbl->file->ha_rnd_pos(tbl->record[1], rowid_b)))
|
|
{
|
|
/* purecov: begin inspected */
|
|
tbl->file->print_error(error, MYF(ME_FATALERROR)); // Sets fatal_error
|
|
return 0;
|
|
/* purecov: end */
|
|
}
|
|
/*
|
|
Compare the two rows by the corresponding values of the indexed
|
|
columns.
|
|
*/
|
|
for (uint i= 0; i < key_column_count; i++)
|
|
{
|
|
Field *cur_field= key_columns[i]->field;
|
|
if ((cmp_res= cur_field->cmp_offset(tbl->s->rec_buff_length)))
|
|
return (cmp_res > 0 ? 1 : -1);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
int
|
|
Ordered_key::cmp_keys_by_row_data_and_rownum(Ordered_key *key,
|
|
rownum_t* a, rownum_t* b)
|
|
{
|
|
/* The result of comparing the two keys according to their row data. */
|
|
int cmp_row_res= key->cmp_keys_by_row_data(*a, *b);
|
|
if (cmp_row_res)
|
|
return cmp_row_res;
|
|
return (*a < *b) ? -1 : (*a > *b) ? 1 : 0;
|
|
}
|
|
|
|
|
|
void Ordered_key::sort_keys()
|
|
{
|
|
my_qsort2(key_buff, (size_t) key_buff_elements, sizeof(rownum_t),
|
|
(qsort2_cmp) &cmp_keys_by_row_data_and_rownum, (void*) this);
|
|
/* Invalidate the current row position. */
|
|
cur_key_idx= HA_POS_ERROR;
|
|
}
|
|
|
|
|
|
/*
|
|
The fraction of rows that do not contain NULL in the columns indexed by
|
|
this key.
|
|
|
|
@retval 1 if there are no NULLs
|
|
@retval 0 if only NULLs
|
|
*/
|
|
|
|
double Ordered_key::null_selectivity()
|
|
{
|
|
/* We should not be processing empty tables. */
|
|
DBUG_ASSERT(tbl->file->stats.records);
|
|
return (1 - (double) null_count / (double) tbl->file->stats.records);
|
|
}
|
|
|
|
|
|
/*
|
|
Compare the value(s) of the current key in 'search_key' with the
|
|
data of the current table record.
|
|
|
|
@notes The comparison result follows from the way compare_pred
|
|
is created in Ordered_key::init. Currently compare_pred compares
|
|
a field in of the current row with the corresponding Item that
|
|
contains the search key.
|
|
|
|
@param row_num Number of the row (not index in the key_buff array)
|
|
|
|
@retval -1 if (current row < search_key)
|
|
@retval 0 if (current row == search_key)
|
|
@retval +1 if (current row > search_key)
|
|
*/
|
|
|
|
int Ordered_key::cmp_key_with_search_key(rownum_t row_num)
|
|
{
|
|
/* The length in bytes of the rowids (positions) of tmp_table. */
|
|
uint rowid_length= tbl->file->ref_length;
|
|
uchar *cur_rowid= row_num_to_rowid + row_num * rowid_length;
|
|
int __attribute__((unused)) error;
|
|
int cmp_res;
|
|
|
|
if ((error= tbl->file->ha_rnd_pos(tbl->record[0], cur_rowid)))
|
|
{
|
|
/* purecov: begin inspected */
|
|
tbl->file->print_error(error, MYF(ME_FATALERROR)); // Sets fatal_error
|
|
return 0;
|
|
/* purecov: end */
|
|
}
|
|
|
|
for (uint i= 0; i < key_column_count; i++)
|
|
{
|
|
cmp_res= compare_pred[i]->get_comparator()->compare();
|
|
/* Unlike Arg_comparator::compare_row() here there should be no NULLs. */
|
|
DBUG_ASSERT(!compare_pred[i]->null_value);
|
|
if (cmp_res)
|
|
return (cmp_res > 0 ? 1 : -1);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
Find a key in a sorted array of keys via binary search.
|
|
|
|
see create_subq_in_equalities()
|
|
*/
|
|
|
|
bool Ordered_key::lookup()
|
|
{
|
|
DBUG_ASSERT(key_buff_elements);
|
|
|
|
ha_rows lo= 0;
|
|
ha_rows hi= key_buff_elements - 1;
|
|
ha_rows mid;
|
|
int cmp_res;
|
|
|
|
while (lo <= hi)
|
|
{
|
|
mid= lo + (hi - lo) / 2;
|
|
cmp_res= cmp_key_with_search_key(key_buff[mid]);
|
|
/*
|
|
In order to find the minimum match, check if the pevious element is
|
|
equal or smaller than the found one. If equal, we need to search further
|
|
to the left.
|
|
*/
|
|
if (!cmp_res && mid > 0)
|
|
cmp_res= !cmp_key_with_search_key(key_buff[mid - 1]) ? 1 : 0;
|
|
|
|
if (cmp_res == -1)
|
|
{
|
|
/* row[mid] < search_key */
|
|
lo= mid + 1;
|
|
}
|
|
else if (cmp_res == 1)
|
|
{
|
|
/* row[mid] > search_key */
|
|
if (!mid)
|
|
goto not_found;
|
|
hi= mid - 1;
|
|
}
|
|
else
|
|
{
|
|
/* row[mid] == search_key */
|
|
cur_key_idx= mid;
|
|
return TRUE;
|
|
}
|
|
}
|
|
not_found:
|
|
cur_key_idx= HA_POS_ERROR;
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*
|
|
Move the current index pointer to the next key with the same column
|
|
values as the current key. Since the index is sorted, all such keys
|
|
are contiguous.
|
|
*/
|
|
|
|
bool Ordered_key::next_same()
|
|
{
|
|
DBUG_ASSERT(key_buff_elements);
|
|
|
|
if (cur_key_idx < key_buff_elements - 1)
|
|
{
|
|
/*
|
|
TIMOUR:
|
|
The below is quite inefficient, since as a result we will fetch every
|
|
row (except the last one) twice. There must be a more efficient way,
|
|
e.g. swapping record[0] and record[1], and reading only the new record.
|
|
*/
|
|
if (!cmp_keys_by_row_data(key_buff[cur_key_idx], key_buff[cur_key_idx + 1]))
|
|
{
|
|
++cur_key_idx;
|
|
return TRUE;
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
void Ordered_key::print(String *str)
|
|
{
|
|
uint i;
|
|
str->append("{idx=");
|
|
str->qs_append(keyid);
|
|
str->append(", (");
|
|
for (i= 0; i < key_column_count - 1; i++)
|
|
{
|
|
str->append(key_columns[i]->field->field_name);
|
|
str->append(", ");
|
|
}
|
|
str->append(key_columns[i]->field->field_name);
|
|
str->append("), ");
|
|
|
|
str->append("null_bitmap: (bits=");
|
|
str->qs_append(null_key.n_bits);
|
|
str->append(", nulls= ");
|
|
str->qs_append((double)null_count);
|
|
str->append(", min_null= ");
|
|
str->qs_append((double)min_null_row);
|
|
str->append(", max_null= ");
|
|
str->qs_append((double)max_null_row);
|
|
str->append("), ");
|
|
|
|
str->append('}');
|
|
}
|
|
|
|
|
|
subselect_partial_match_engine::subselect_partial_match_engine(
|
|
THD *thd_arg, subselect_uniquesubquery_engine *engine_arg,
|
|
TABLE *tmp_table_arg, Item_subselect *item_arg,
|
|
select_result_interceptor *result_arg,
|
|
List<Item> *equi_join_conds_arg,
|
|
bool has_covering_null_row_arg,
|
|
bool has_covering_null_columns_arg,
|
|
uint count_columns_with_nulls_arg)
|
|
:subselect_engine(thd_arg, item_arg, result_arg),
|
|
tmp_table(tmp_table_arg), lookup_engine(engine_arg),
|
|
equi_join_conds(equi_join_conds_arg),
|
|
has_covering_null_row(has_covering_null_row_arg),
|
|
has_covering_null_columns(has_covering_null_columns_arg),
|
|
count_columns_with_nulls(count_columns_with_nulls_arg)
|
|
{}
|
|
|
|
|
|
int subselect_partial_match_engine::exec()
|
|
{
|
|
Item_in_subselect *item_in= (Item_in_subselect *) item;
|
|
int lookup_res;
|
|
|
|
DBUG_ASSERT(!(item_in->left_expr_has_null() &&
|
|
item_in->is_top_level_item()));
|
|
|
|
if (!item_in->left_expr_has_null())
|
|
{
|
|
/* Try to find a matching row by index lookup. */
|
|
if (lookup_engine->copy_ref_key(false))
|
|
{
|
|
/* The result is FALSE based on the outer reference. */
|
|
item_in->value= 0;
|
|
item_in->null_value= 0;
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
/* Search for a complete match. */
|
|
if ((lookup_res= lookup_engine->index_lookup()))
|
|
{
|
|
/* An error occured during lookup(). */
|
|
item_in->value= 0;
|
|
item_in->null_value= 0;
|
|
return lookup_res;
|
|
}
|
|
else if (item_in->value || !count_columns_with_nulls)
|
|
{
|
|
/*
|
|
A complete match was found, the result of IN is TRUE.
|
|
If no match was found, and there are no NULLs in the materialized
|
|
subquery, then the result is guaranteed to be false because this
|
|
branch is executed when the outer reference has no NULLs as well.
|
|
Notice: (this->item == lookup_engine->item)
|
|
*/
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (has_covering_null_row)
|
|
{
|
|
/*
|
|
If there is a NULL-only row that coveres all columns the result of IN
|
|
is UNKNOWN.
|
|
*/
|
|
item_in->value= 0;
|
|
/*
|
|
TIMOUR: which one is the right way to propagate an UNKNOWN result?
|
|
Should we also set empty_result_set= FALSE; ???
|
|
*/
|
|
//item_in->was_null= 1;
|
|
item_in->null_value= 1;
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
There is no complete match. Look for a partial match (UNKNOWN result), or
|
|
no match (FALSE).
|
|
*/
|
|
if (tmp_table->file->inited)
|
|
tmp_table->file->ha_index_end();
|
|
|
|
if (partial_match())
|
|
{
|
|
/* The result of IN is UNKNOWN. */
|
|
item_in->value= 0;
|
|
/*
|
|
TIMOUR: which one is the right way to propagate an UNKNOWN result?
|
|
Should we also set empty_result_set= FALSE; ???
|
|
*/
|
|
//item_in->was_null= 1;
|
|
item_in->null_value= 1;
|
|
}
|
|
else
|
|
{
|
|
/* The result of IN is FALSE. */
|
|
item_in->value= 0;
|
|
/*
|
|
TIMOUR: which one is the right way to propagate an UNKNOWN result?
|
|
Should we also set empty_result_set= FALSE; ???
|
|
*/
|
|
//item_in->was_null= 0;
|
|
item_in->null_value= 0;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
void subselect_partial_match_engine::print(String *str,
|
|
enum_query_type query_type)
|
|
{
|
|
/*
|
|
Should never be called as the actual engine cannot be known at query
|
|
optimization time.
|
|
DBUG_ASSERT(FALSE);
|
|
*/
|
|
}
|
|
|
|
|
|
/*
|
|
@param non_null_key_parts
|
|
@param partial_match_key_parts A union of all single-column NULL key parts.
|
|
|
|
@retval FALSE the engine was initialized successfully
|
|
@retval TRUE there was some (memory allocation) error during initialization,
|
|
such errors should be interpreted as revert to other strategy
|
|
*/
|
|
|
|
bool
|
|
subselect_rowid_merge_engine::init(MY_BITMAP *non_null_key_parts,
|
|
MY_BITMAP *partial_match_key_parts)
|
|
{
|
|
/* The length in bytes of the rowids (positions) of tmp_table. */
|
|
uint rowid_length= tmp_table->file->ref_length;
|
|
ha_rows row_count= tmp_table->file->stats.records;
|
|
rownum_t cur_rownum= 0;
|
|
select_materialize_with_stats *result_sink=
|
|
(select_materialize_with_stats *) result;
|
|
uint cur_keyid= 0;
|
|
Item_in_subselect *item_in= (Item_in_subselect*) item;
|
|
int error;
|
|
|
|
if (merge_keys_count == 0)
|
|
{
|
|
DBUG_ASSERT(bitmap_bits_set(partial_match_key_parts) == 0 ||
|
|
has_covering_null_row);
|
|
/* There is nothing to initialize, we will only do regular lookups. */
|
|
return FALSE;
|
|
}
|
|
|
|
/*
|
|
If all nullable columns contain only NULLs, there must be one index
|
|
over all non-null columns.
|
|
*/
|
|
DBUG_ASSERT(!has_covering_null_columns ||
|
|
(has_covering_null_columns &&
|
|
merge_keys_count == 1 && non_null_key_parts));
|
|
/*
|
|
Allocate buffers to hold the merged keys and the mapping between rowids and
|
|
row numbers. All small buffers are allocated in the runtime memroot. Big
|
|
buffers are allocated from the OS via malloc.
|
|
*/
|
|
if (!(merge_keys= (Ordered_key**) thd->alloc(merge_keys_count *
|
|
sizeof(Ordered_key*))) ||
|
|
!(null_bitmaps= (MY_BITMAP**) thd->alloc(merge_keys_count *
|
|
sizeof(MY_BITMAP*))) ||
|
|
!(row_num_to_rowid= (uchar*) my_malloc((size_t)(row_count * rowid_length),
|
|
MYF(MY_WME | MY_THREAD_SPECIFIC))))
|
|
return TRUE;
|
|
|
|
/* Create the only non-NULL key if there is any. */
|
|
if (non_null_key_parts)
|
|
{
|
|
non_null_key= new Ordered_key(cur_keyid, tmp_table, item_in->left_expr,
|
|
0, 0, 0, row_num_to_rowid);
|
|
if (non_null_key->init(non_null_key_parts))
|
|
return TRUE;
|
|
merge_keys[cur_keyid]= non_null_key;
|
|
merge_keys[cur_keyid]->first();
|
|
++cur_keyid;
|
|
}
|
|
|
|
/*
|
|
If all nullable columns contain NULLs, the only key that is needed is the
|
|
only non-NULL key that is already created above.
|
|
*/
|
|
if (!has_covering_null_columns)
|
|
{
|
|
if (my_bitmap_init_memroot(&matching_keys, merge_keys_count, thd->mem_root) ||
|
|
my_bitmap_init_memroot(&matching_outer_cols, merge_keys_count, thd->mem_root))
|
|
return TRUE;
|
|
|
|
/*
|
|
Create one single-column NULL-key for each column in
|
|
partial_match_key_parts.
|
|
*/
|
|
for (uint i= 0; i < partial_match_key_parts->n_bits; i++)
|
|
{
|
|
/* Skip columns that have no NULLs, or contain only NULLs. */
|
|
if (!bitmap_is_set(partial_match_key_parts, i) ||
|
|
result_sink->get_null_count_of_col(i) == row_count)
|
|
continue;
|
|
|
|
merge_keys[cur_keyid]= new Ordered_key(
|
|
cur_keyid, tmp_table,
|
|
item_in->left_expr->element_index(i),
|
|
result_sink->get_null_count_of_col(i),
|
|
result_sink->get_min_null_of_col(i),
|
|
result_sink->get_max_null_of_col(i),
|
|
row_num_to_rowid);
|
|
if (merge_keys[cur_keyid]->init(i))
|
|
return TRUE;
|
|
merge_keys[cur_keyid]->first();
|
|
++cur_keyid;
|
|
}
|
|
}
|
|
DBUG_ASSERT(cur_keyid == merge_keys_count);
|
|
|
|
/* Populate the indexes with data from the temporary table. */
|
|
if (tmp_table->file->ha_rnd_init_with_error(1))
|
|
return TRUE;
|
|
tmp_table->file->extra_opt(HA_EXTRA_CACHE,
|
|
current_thd->variables.read_buff_size);
|
|
tmp_table->null_row= 0;
|
|
while (TRUE)
|
|
{
|
|
error= tmp_table->file->ha_rnd_next(tmp_table->record[0]);
|
|
if (error == HA_ERR_RECORD_DELETED)
|
|
{
|
|
/* We get this for duplicate records that should not be in tmp_table. */
|
|
continue;
|
|
}
|
|
/*
|
|
This is a temp table that we fully own, there should be no other
|
|
cause to stop the iteration than EOF.
|
|
*/
|
|
DBUG_ASSERT(!error || error == HA_ERR_END_OF_FILE);
|
|
if (error == HA_ERR_END_OF_FILE)
|
|
{
|
|
DBUG_ASSERT(cur_rownum == tmp_table->file->stats.records);
|
|
break;
|
|
}
|
|
|
|
/*
|
|
Save the position of this record in the row_num -> rowid mapping.
|
|
*/
|
|
tmp_table->file->position(tmp_table->record[0]);
|
|
memcpy(row_num_to_rowid + cur_rownum * rowid_length,
|
|
tmp_table->file->ref, rowid_length);
|
|
|
|
/* Add the current row number to the corresponding keys. */
|
|
if (non_null_key)
|
|
{
|
|
/* By definition there are no NULLs in the non-NULL key. */
|
|
non_null_key->add_key(cur_rownum);
|
|
}
|
|
|
|
for (uint i= (non_null_key ? 1 : 0); i < merge_keys_count; i++)
|
|
{
|
|
/*
|
|
Check if the first and only indexed column contains NULL in the curent
|
|
row, and add the row number to the corresponding key.
|
|
*/
|
|
if (tmp_table->field[merge_keys[i]->get_field_idx(0)]->is_null())
|
|
merge_keys[i]->set_null(cur_rownum);
|
|
else
|
|
merge_keys[i]->add_key(cur_rownum);
|
|
}
|
|
++cur_rownum;
|
|
}
|
|
|
|
tmp_table->file->ha_rnd_end();
|
|
|
|
/* Sort all the keys by their NULL selectivity. */
|
|
my_qsort(merge_keys, merge_keys_count, sizeof(Ordered_key*),
|
|
(qsort_cmp) cmp_keys_by_null_selectivity);
|
|
|
|
/* Sort the keys in each of the indexes. */
|
|
for (uint i= 0; i < merge_keys_count; i++)
|
|
merge_keys[i]->sort_keys();
|
|
|
|
if (init_queue(&pq, merge_keys_count, 0, FALSE,
|
|
subselect_rowid_merge_engine::cmp_keys_by_cur_rownum, NULL,
|
|
0, 0))
|
|
return TRUE;
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
subselect_rowid_merge_engine::~subselect_rowid_merge_engine()
|
|
{
|
|
/* None of the resources below is allocated if there are no ordered keys. */
|
|
if (merge_keys_count)
|
|
{
|
|
my_free(row_num_to_rowid);
|
|
for (uint i= 0; i < merge_keys_count; i++)
|
|
delete merge_keys[i];
|
|
delete_queue(&pq);
|
|
if (tmp_table->file->inited == handler::RND)
|
|
tmp_table->file->ha_rnd_end();
|
|
}
|
|
}
|
|
|
|
|
|
void subselect_rowid_merge_engine::cleanup()
|
|
{
|
|
}
|
|
|
|
|
|
/*
|
|
Quick sort comparison function to compare keys in order of decreasing bitmap
|
|
selectivity, so that the most selective keys come first.
|
|
|
|
@param k1 first key to compare
|
|
@param k2 second key to compare
|
|
|
|
@retval 1 if k1 is less selective than k2
|
|
@retval 0 if k1 is equally selective as k2
|
|
@retval -1 if k1 is more selective than k2
|
|
*/
|
|
|
|
int
|
|
subselect_rowid_merge_engine::cmp_keys_by_null_selectivity(Ordered_key **k1,
|
|
Ordered_key **k2)
|
|
{
|
|
double k1_sel= (*k1)->null_selectivity();
|
|
double k2_sel= (*k2)->null_selectivity();
|
|
if (k1_sel < k2_sel)
|
|
return 1;
|
|
if (k1_sel > k2_sel)
|
|
return -1;
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
*/
|
|
|
|
int
|
|
subselect_rowid_merge_engine::cmp_keys_by_cur_rownum(void *arg,
|
|
uchar *k1, uchar *k2)
|
|
{
|
|
rownum_t r1= ((Ordered_key*) k1)->current();
|
|
rownum_t r2= ((Ordered_key*) k2)->current();
|
|
|
|
return (r1 < r2) ? -1 : (r1 > r2) ? 1 : 0;
|
|
}
|
|
|
|
|
|
/*
|
|
Check if certain table row contains a NULL in all columns for which there is
|
|
no match in the corresponding value index.
|
|
|
|
@note
|
|
There is no need to check the columns that contain only NULLs, because
|
|
those are guaranteed to match.
|
|
|
|
@retval TRUE if a NULL row exists
|
|
@retval FALSE otherwise
|
|
*/
|
|
|
|
bool subselect_rowid_merge_engine::test_null_row(rownum_t row_num)
|
|
{
|
|
Ordered_key *cur_key;
|
|
for (uint i = 0; i < merge_keys_count; i++)
|
|
{
|
|
cur_key= merge_keys[i];
|
|
if (bitmap_is_set(&matching_keys, cur_key->get_keyid()))
|
|
{
|
|
/*
|
|
The key 'i' (with id 'cur_keyid') already matches a value in row
|
|
'row_num', thus we skip it as it can't possibly match a NULL.
|
|
*/
|
|
continue;
|
|
}
|
|
if (!cur_key->is_null(row_num))
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/**
|
|
Test if a subset of NULL-able columns contains a row of NULLs.
|
|
@retval TRUE if such a row exists
|
|
@retval FALSE no complementing null row
|
|
*/
|
|
|
|
bool subselect_rowid_merge_engine::
|
|
exists_complementing_null_row(MY_BITMAP *keys_to_complement)
|
|
{
|
|
rownum_t highest_min_row= 0;
|
|
rownum_t lowest_max_row= UINT_MAX;
|
|
uint count_null_keys, i;
|
|
Ordered_key *cur_key;
|
|
|
|
if (!count_columns_with_nulls)
|
|
{
|
|
/*
|
|
If there are both NULLs and non-NUll values in the outer reference, and
|
|
the subquery contains no NULLs, a complementing NULL row cannot exist.
|
|
*/
|
|
return FALSE;
|
|
}
|
|
|
|
for (i= (non_null_key ? 1 : 0), count_null_keys= 0; i < merge_keys_count; i++)
|
|
{
|
|
cur_key= merge_keys[i];
|
|
if (bitmap_is_set(keys_to_complement, cur_key->get_keyid()))
|
|
continue;
|
|
if (!cur_key->get_null_count())
|
|
{
|
|
/* If there is column without NULLs, there cannot be a partial match. */
|
|
return FALSE;
|
|
}
|
|
if (cur_key->get_min_null_row() > highest_min_row)
|
|
highest_min_row= cur_key->get_min_null_row();
|
|
if (cur_key->get_max_null_row() < lowest_max_row)
|
|
lowest_max_row= cur_key->get_max_null_row();
|
|
null_bitmaps[count_null_keys++]= cur_key->get_null_key();
|
|
}
|
|
|
|
if (lowest_max_row < highest_min_row)
|
|
{
|
|
/* The intersection of NULL rows is empty. */
|
|
return FALSE;
|
|
}
|
|
|
|
return bitmap_exists_intersection((const MY_BITMAP**) null_bitmaps,
|
|
count_null_keys,
|
|
(uint)highest_min_row, (uint)lowest_max_row);
|
|
}
|
|
|
|
|
|
/*
|
|
@retval TRUE there is a partial match (UNKNOWN)
|
|
@retval FALSE there is no match at all (FALSE)
|
|
*/
|
|
|
|
bool subselect_rowid_merge_engine::partial_match()
|
|
{
|
|
Ordered_key *min_key; /* Key that contains the current minimum position. */
|
|
rownum_t min_row_num; /* Current row number of min_key. */
|
|
Ordered_key *cur_key;
|
|
rownum_t cur_row_num;
|
|
uint count_nulls_in_search_key= 0;
|
|
uint max_null_in_any_row=
|
|
((select_materialize_with_stats *) result)->get_max_nulls_in_row();
|
|
bool res= FALSE;
|
|
|
|
/* If there is a non-NULL key, it must be the first key in the keys array. */
|
|
DBUG_ASSERT(!non_null_key || (non_null_key && merge_keys[0] == non_null_key));
|
|
/* The prioryty queue for keys must be empty. */
|
|
DBUG_ASSERT(!pq.elements);
|
|
|
|
/* All data accesses during execution are via handler::ha_rnd_pos() */
|
|
if (tmp_table->file->ha_rnd_init_with_error(0))
|
|
{
|
|
res= FALSE;
|
|
goto end;
|
|
}
|
|
|
|
/* Check if there is a match for the columns of the only non-NULL key. */
|
|
if (non_null_key && !non_null_key->lookup())
|
|
{
|
|
res= FALSE;
|
|
goto end;
|
|
}
|
|
|
|
/*
|
|
If all nullable columns contain only NULLs, then there is a guranteed
|
|
partial match, and we don't need to search for a matching row.
|
|
*/
|
|
if (has_covering_null_columns)
|
|
{
|
|
res= TRUE;
|
|
goto end;
|
|
}
|
|
|
|
if (non_null_key)
|
|
queue_insert(&pq, (uchar *) non_null_key);
|
|
/*
|
|
Do not add the non_null_key, since it was already processed above.
|
|
*/
|
|
bitmap_clear_all(&matching_outer_cols);
|
|
for (uint i= MY_TEST(non_null_key); i < merge_keys_count; i++)
|
|
{
|
|
DBUG_ASSERT(merge_keys[i]->get_column_count() == 1);
|
|
if (merge_keys[i]->get_search_key(0)->null_value)
|
|
{
|
|
++count_nulls_in_search_key;
|
|
bitmap_set_bit(&matching_outer_cols, merge_keys[i]->get_keyid());
|
|
}
|
|
else if (merge_keys[i]->lookup())
|
|
queue_insert(&pq, (uchar *) merge_keys[i]);
|
|
}
|
|
|
|
/*
|
|
If the outer reference consists of only NULLs, or if it has NULLs in all
|
|
nullable columns (above we guarantee there is a match for the non-null
|
|
coumns), the result is UNKNOWN.
|
|
*/
|
|
if (count_nulls_in_search_key == merge_keys_count - MY_TEST(non_null_key))
|
|
{
|
|
res= TRUE;
|
|
goto end;
|
|
}
|
|
|
|
/*
|
|
If the outer row has NULLs in some columns, and
|
|
there is no match for any of the remaining columns, and
|
|
there is a subquery row with NULLs in all unmatched columns,
|
|
then there is a partial match, otherwise the result is FALSE.
|
|
*/
|
|
if (count_nulls_in_search_key && !pq.elements)
|
|
{
|
|
DBUG_ASSERT(!non_null_key);
|
|
/*
|
|
Check if the intersection of all NULL bitmaps of all keys that
|
|
are not in matching_outer_cols is non-empty.
|
|
*/
|
|
res= exists_complementing_null_row(&matching_outer_cols);
|
|
goto end;
|
|
}
|
|
|
|
/*
|
|
If there is no NULL (sub)row that covers all NULL columns, and there is no
|
|
match for any of the NULL columns, the result is FALSE. Notice that if there
|
|
is a non-null key, and there is only one matching key, the non-null key is
|
|
the matching key. This is so, because this method returns FALSE if the
|
|
non-null key doesn't have a match.
|
|
*/
|
|
if (!count_nulls_in_search_key &&
|
|
(!pq.elements ||
|
|
(pq.elements == 1 && non_null_key &&
|
|
max_null_in_any_row < merge_keys_count-1)))
|
|
{
|
|
if (!pq.elements)
|
|
{
|
|
DBUG_ASSERT(!non_null_key);
|
|
/*
|
|
The case of a covering null row is handled by
|
|
subselect_partial_match_engine::exec()
|
|
*/
|
|
DBUG_ASSERT(max_null_in_any_row != tmp_table->s->fields);
|
|
}
|
|
res= FALSE;
|
|
goto end;
|
|
}
|
|
|
|
DBUG_ASSERT(pq.elements);
|
|
|
|
min_key= (Ordered_key*) queue_remove_top(&pq);
|
|
min_row_num= min_key->current();
|
|
bitmap_set_bit(&matching_keys, min_key->get_keyid());
|
|
bitmap_union(&matching_keys, &matching_outer_cols);
|
|
if (min_key->next_same())
|
|
queue_insert(&pq, (uchar *) min_key);
|
|
|
|
if (pq.elements == 0)
|
|
{
|
|
/*
|
|
Check the only matching row of the only key min_key for NULL matches
|
|
in the other columns.
|
|
*/
|
|
res= test_null_row(min_row_num);
|
|
goto end;
|
|
}
|
|
|
|
while (TRUE)
|
|
{
|
|
cur_key= (Ordered_key*) queue_remove_top(&pq);
|
|
cur_row_num= cur_key->current();
|
|
|
|
if (cur_row_num == min_row_num)
|
|
bitmap_set_bit(&matching_keys, cur_key->get_keyid());
|
|
else
|
|
{
|
|
/* Follows from the correct use of priority queue. */
|
|
DBUG_ASSERT(cur_row_num > min_row_num);
|
|
if (test_null_row(min_row_num))
|
|
{
|
|
res= TRUE;
|
|
goto end;
|
|
}
|
|
else
|
|
{
|
|
min_key= cur_key;
|
|
min_row_num= cur_row_num;
|
|
bitmap_clear_all(&matching_keys);
|
|
bitmap_set_bit(&matching_keys, min_key->get_keyid());
|
|
bitmap_union(&matching_keys, &matching_outer_cols);
|
|
}
|
|
}
|
|
|
|
if (cur_key->next_same())
|
|
queue_insert(&pq, (uchar *) cur_key);
|
|
|
|
if (pq.elements == 0)
|
|
{
|
|
/* Check the last row of the last column in PQ for NULL matches. */
|
|
res= test_null_row(min_row_num);
|
|
goto end;
|
|
}
|
|
}
|
|
|
|
/* We should never get here - all branches must be handled explicitly above. */
|
|
DBUG_ASSERT(FALSE);
|
|
|
|
end:
|
|
if (!has_covering_null_columns)
|
|
bitmap_clear_all(&matching_keys);
|
|
queue_remove_all(&pq);
|
|
tmp_table->file->ha_rnd_end();
|
|
return res;
|
|
}
|
|
|
|
|
|
subselect_table_scan_engine::subselect_table_scan_engine(
|
|
THD *thd_arg, subselect_uniquesubquery_engine *engine_arg,
|
|
TABLE *tmp_table_arg,
|
|
Item_subselect *item_arg,
|
|
select_result_interceptor *result_arg,
|
|
List<Item> *equi_join_conds_arg,
|
|
bool has_covering_null_row_arg,
|
|
bool has_covering_null_columns_arg,
|
|
uint count_columns_with_nulls_arg)
|
|
:subselect_partial_match_engine(thd_arg, engine_arg, tmp_table_arg, item_arg,
|
|
result_arg, equi_join_conds_arg,
|
|
has_covering_null_row_arg,
|
|
has_covering_null_columns_arg,
|
|
count_columns_with_nulls_arg)
|
|
{}
|
|
|
|
|
|
/*
|
|
TIMOUR:
|
|
This method is based on subselect_uniquesubquery_engine::scan_table().
|
|
Consider refactoring somehow, 80% of the code is the same.
|
|
|
|
for each row_i in tmp_table
|
|
{
|
|
count_matches= 0;
|
|
for each row element row_i[j]
|
|
{
|
|
if (outer_ref[j] is NULL || row_i[j] is NULL || outer_ref[j] == row_i[j])
|
|
++count_matches;
|
|
}
|
|
if (count_matches == outer_ref.elements)
|
|
return TRUE
|
|
}
|
|
return FALSE
|
|
*/
|
|
|
|
bool subselect_table_scan_engine::partial_match()
|
|
{
|
|
List_iterator_fast<Item> equality_it(*equi_join_conds);
|
|
Item *cur_eq;
|
|
uint count_matches;
|
|
int error;
|
|
bool res;
|
|
|
|
if (tmp_table->file->ha_rnd_init_with_error(1))
|
|
{
|
|
res= FALSE;
|
|
goto end;
|
|
}
|
|
|
|
tmp_table->file->extra_opt(HA_EXTRA_CACHE,
|
|
current_thd->variables.read_buff_size);
|
|
for (;;)
|
|
{
|
|
error= tmp_table->file->ha_rnd_next(tmp_table->record[0]);
|
|
if (error) {
|
|
if (error == HA_ERR_RECORD_DELETED)
|
|
{
|
|
error= 0;
|
|
continue;
|
|
}
|
|
if (error == HA_ERR_END_OF_FILE)
|
|
{
|
|
error= 0;
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
error= report_error(tmp_table, error);
|
|
break;
|
|
}
|
|
}
|
|
|
|
equality_it.rewind();
|
|
count_matches= 0;
|
|
while ((cur_eq= equality_it++))
|
|
{
|
|
DBUG_ASSERT(cur_eq->type() == Item::FUNC_ITEM &&
|
|
((Item_func*)cur_eq)->functype() == Item_func::EQ_FUNC);
|
|
if (!cur_eq->val_int() && !cur_eq->null_value)
|
|
break;
|
|
++count_matches;
|
|
}
|
|
if (count_matches == tmp_table->s->fields)
|
|
{
|
|
res= TRUE; /* Found a matching row. */
|
|
goto end;
|
|
}
|
|
}
|
|
|
|
res= FALSE;
|
|
end:
|
|
tmp_table->file->ha_rnd_end();
|
|
return res;
|
|
}
|
|
|
|
|
|
void subselect_table_scan_engine::cleanup()
|
|
{
|
|
}
|
|
|
|
|
|
/*
|
|
Create an execution tracker for the expression cache we're using for this
|
|
subselect; add the tracker to the query plan.
|
|
*/
|
|
|
|
void Item_subselect::init_expr_cache_tracker(THD *thd)
|
|
{
|
|
if(!expr_cache)
|
|
return;
|
|
|
|
Explain_query *qw= thd->lex->explain;
|
|
DBUG_ASSERT(qw);
|
|
Explain_node *node= qw->get_node(unit->first_select()->select_number);
|
|
if (!node)
|
|
return;
|
|
DBUG_ASSERT(expr_cache->type() == Item::EXPR_CACHE_ITEM);
|
|
node->cache_tracker= ((Item_cache_wrapper *)expr_cache)->init_tracker(qw->mem_root);
|
|
}
|
|
|