2016-12-22 12:49:06 +01:00
|
|
|
/* Copyright (c) 2002, 2016, Oracle and/or its affiliates.
|
|
|
|
Copyright (c) 2010, 2016, MariaDB
|
2002-05-12 22:46:42 +02:00
|
|
|
|
|
|
|
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
|
2006-12-23 20:17:15 +01:00
|
|
|
the Free Software Foundation; version 2 of the License.
|
2002-05-12 22:46:42 +02:00
|
|
|
|
|
|
|
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
|
2019-05-11 20:29:06 +02:00
|
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1335 USA */
|
2002-05-12 22:46:42 +02:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
/**
|
|
|
|
@file
|
|
|
|
|
|
|
|
@brief
|
|
|
|
subselect Item
|
2002-05-12 22:46:42 +02:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
@todo
|
|
|
|
- add function from mysql_select that use JOIN* as parameter to JOIN
|
|
|
|
methods (sql_select.h/sql_select.cc)
|
2002-05-12 22:46:42 +02:00
|
|
|
*/
|
|
|
|
|
2005-05-26 12:09:14 +02:00
|
|
|
#ifdef USE_PRAGMA_IMPLEMENTATION
|
2002-05-12 22:46:42 +02:00
|
|
|
#pragma implementation // gcc: Class implementation
|
|
|
|
#endif
|
|
|
|
|
2017-06-18 05:42:16 +02:00
|
|
|
#include "mariadb.h"
|
2010-03-31 16:05:33 +02:00
|
|
|
#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"
|
2002-05-12 22:46:42 +02:00
|
|
|
#include "sql_select.h"
|
2010-03-31 16:05:33 +02:00
|
|
|
#include "sql_parse.h" // check_stack_overrun
|
2017-04-20 22:09:31 +02:00
|
|
|
#include "sql_cte.h"
|
2011-05-05 08:13:22 +02:00
|
|
|
#include "sql_test.h"
|
2002-05-12 22:46:42 +02:00
|
|
|
|
2011-05-25 17:31:13 +02:00
|
|
|
double get_post_group_estimate(JOIN* join, double join_op_rows);
|
|
|
|
|
2017-04-23 18:39:57 +02:00
|
|
|
LEX_CSTRING exists_outer_expr_name= { STRING_WITH_LEN("<exists outer expr>") };
|
2013-02-26 00:20:17 +01:00
|
|
|
|
|
|
|
int check_and_do_in_subquery_rewrites(JOIN *join);
|
2002-12-28 00:01:05 +01:00
|
|
|
|
2015-08-11 09:18:38 +02:00
|
|
|
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),
|
2015-04-22 10:40:23 +02:00
|
|
|
have_to_be_excluded(0),
|
2011-03-22 11:09:55 +01:00
|
|
|
inside_first_fix_fields(0), done_first_fix_fields(FALSE),
|
2011-06-21 14:50:07 +02:00
|
|
|
expr_cache(0), forced_const(FALSE), substitution(0), engine(0), eliminated(FALSE),
|
2016-05-24 20:29:52 +02:00
|
|
|
changed(0), is_correlated(FALSE), with_recursive_reference(0)
|
2002-05-12 22:46:42 +02:00
|
|
|
{
|
2011-05-04 17:08:58 +02:00
|
|
|
DBUG_ENTER("Item_subselect::Item_subselect");
|
2017-09-19 19:45:17 +02:00
|
|
|
DBUG_PRINT("enter", ("this: %p", this));
|
2016-03-22 20:51:59 +01:00
|
|
|
sortbuffer.str= 0;
|
|
|
|
|
2011-06-21 14:50:07 +02:00
|
|
|
#ifndef DBUG_OFF
|
|
|
|
exec_counter= 0;
|
|
|
|
#endif
|
2002-12-06 20:55:53 +01:00
|
|
|
reset();
|
2002-10-08 22:49:59 +02:00
|
|
|
/*
|
2010-01-28 14:48:33 +01:00
|
|
|
Item value is NULL if select_result_interceptor didn't change this value
|
2002-10-08 22:49:59 +02:00
|
|
|
(i.e. some rows will be found returned)
|
|
|
|
*/
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= TRUE;
|
2011-05-04 17:08:58 +02:00
|
|
|
DBUG_VOID_RETURN;
|
2002-10-08 22:49:59 +02:00
|
|
|
}
|
|
|
|
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2003-10-02 21:19:41 +02:00
|
|
|
void Item_subselect::init(st_select_lex *select_lex,
|
2010-01-28 14:48:33 +01:00
|
|
|
select_result_interceptor *result)
|
2002-10-08 22:49:59 +02:00
|
|
|
{
|
Bug#21904 (parser problem when using IN with a double "(())")
Before this fix, a IN predicate of the form: "IN (( subselect ))", with two
parenthesis, would be evaluated as a single row subselect: if the subselect
returns more that 1 row, the statement would fail.
The SQL:2003 standard defines a special exception in the specification,
and mandates that this particular form of IN predicate shall be equivalent
to "IN ( subselect )", which involves a table subquery and works with more
than 1 row.
This fix implements "IN (( subselect ))", "IN ((( subselect )))" etc
as per the SQL:2003 requirement.
All the details related to the implementation of this change have been
commented in the code, and the relevant sections of the SQL:2003 spec
are given for reference, so they are not repeated here.
Having access to the spec is a requirement to review in depth this patch.
mysql-test/r/subselect.result:
Implement IN predicate special exceptions with subselects.
mysql-test/t/subselect.test:
Implement IN predicate special exceptions with subselects.
sql/item_subselect.cc:
Implement IN predicate special exceptions with subselects.
sql/item_subselect.h:
Implement IN predicate special exceptions with subselects.
sql/sql_yacc.yy:
Implement IN predicate special exceptions with subselects, cleanup.
2007-01-30 01:32:52 +01:00
|
|
|
/*
|
|
|
|
Please see Item_singlerow_subselect::invalidate_and_restore_select_lex(),
|
|
|
|
which depends on alterations to the parse tree implemented here.
|
|
|
|
*/
|
2002-10-08 22:49:59 +02:00
|
|
|
|
|
|
|
DBUG_ENTER("Item_subselect::init");
|
2017-09-19 19:45:17 +02:00
|
|
|
DBUG_PRINT("enter", ("select_lex: %p this: %p",
|
|
|
|
select_lex, this));
|
2004-02-08 19:14:13 +01:00
|
|
|
unit= select_lex->master_unit();
|
2002-09-03 08:50:36 +02:00
|
|
|
|
2004-05-07 22:06:11 +02:00
|
|
|
if (unit->item)
|
|
|
|
{
|
|
|
|
engine= unit->item->engine;
|
2004-08-13 09:01:30 +02:00
|
|
|
parsing_place= unit->item->parsing_place;
|
2013-02-26 00:20:17 +01:00
|
|
|
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
|
|
|
|
*/
|
2016-06-28 11:25:59 +02:00
|
|
|
unit->thd->change_item_tree((Item**)&unit->item, this);
|
2013-02-26 00:20:17 +01:00
|
|
|
engine->change_result(this, result, TRUE);
|
|
|
|
}
|
2004-05-07 22:06:11 +02:00
|
|
|
}
|
2002-09-03 08:50:36 +02:00
|
|
|
else
|
2004-05-07 22:06:11 +02:00
|
|
|
{
|
2004-08-23 21:31:01 +02:00
|
|
|
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);
|
2017-10-28 20:54:18 +02:00
|
|
|
if (unit->is_unit_op() && unit->first_select()->next_select())
|
2016-06-22 14:17:06 +02:00
|
|
|
engine= new subselect_union_engine(unit, result, this);
|
2004-05-07 22:06:11 +02:00
|
|
|
else
|
2016-06-22 14:17:06 +02:00
|
|
|
engine= new subselect_single_select_engine(select_lex, result, this);
|
2004-05-07 22:06:11 +02:00
|
|
|
}
|
2004-06-09 22:32:20 +02:00
|
|
|
{
|
|
|
|
SELECT_LEX *upper= unit->outer_select();
|
2004-08-13 09:01:30 +02:00
|
|
|
if (upper->parsing_place == IN_HAVING)
|
2004-06-09 22:32:20 +02:00
|
|
|
upper->subquery_in_having= 1;
|
2010-07-10 12:37:30 +02:00
|
|
|
/* The subquery is an expression cache candidate */
|
|
|
|
upper->expr_cache_may_be_used[upper->parsing_place]= TRUE;
|
2004-06-09 22:32:20 +02:00
|
|
|
}
|
2017-09-19 19:45:17 +02:00
|
|
|
DBUG_PRINT("info", ("engine: %p", engine));
|
2002-05-12 22:46:42 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
Bug#21904 (parser problem when using IN with a double "(())")
Before this fix, a IN predicate of the form: "IN (( subselect ))", with two
parenthesis, would be evaluated as a single row subselect: if the subselect
returns more that 1 row, the statement would fail.
The SQL:2003 standard defines a special exception in the specification,
and mandates that this particular form of IN predicate shall be equivalent
to "IN ( subselect )", which involves a table subquery and works with more
than 1 row.
This fix implements "IN (( subselect ))", "IN ((( subselect )))" etc
as per the SQL:2003 requirement.
All the details related to the implementation of this change have been
commented in the code, and the relevant sections of the SQL:2003 spec
are given for reference, so they are not repeated here.
Having access to the spec is a requirement to review in depth this patch.
mysql-test/r/subselect.result:
Implement IN predicate special exceptions with subselects.
mysql-test/t/subselect.test:
Implement IN predicate special exceptions with subselects.
sql/item_subselect.cc:
Implement IN predicate special exceptions with subselects.
sql/item_subselect.h:
Implement IN predicate special exceptions with subselects.
sql/sql_yacc.yy:
Implement IN predicate special exceptions with subselects, cleanup.
2007-01-30 01:32:52 +01:00
|
|
|
st_select_lex *
|
|
|
|
Item_subselect::get_select_lex()
|
|
|
|
{
|
|
|
|
return unit->first_select();
|
|
|
|
}
|
|
|
|
|
2003-12-30 11:08:19 +01:00
|
|
|
void Item_subselect::cleanup()
|
|
|
|
{
|
2004-02-08 19:14:13 +01:00
|
|
|
DBUG_ENTER("Item_subselect::cleanup");
|
2003-12-30 11:08:19 +01:00
|
|
|
Item_result_field::cleanup();
|
2004-02-08 19:14:13 +01:00
|
|
|
if (old_engine)
|
|
|
|
{
|
2004-05-07 22:06:11 +02:00
|
|
|
if (engine)
|
|
|
|
engine->cleanup();
|
2004-02-08 19:14:13 +01:00
|
|
|
engine= old_engine;
|
|
|
|
old_engine= 0;
|
|
|
|
}
|
2004-05-07 22:06:11 +02:00
|
|
|
if (engine)
|
|
|
|
engine->cleanup();
|
2004-02-08 19:14:13 +01:00
|
|
|
reset();
|
2016-03-22 20:51:59 +01:00
|
|
|
filesort_buffer.free_sort_buffer();
|
|
|
|
my_free(sortbuffer.str);
|
2018-05-17 07:42:53 +02:00
|
|
|
sortbuffer.str= 0;
|
2016-03-22 20:51:59 +01:00
|
|
|
|
2004-02-08 19:14:13 +01:00
|
|
|
value_assigned= 0;
|
2010-09-06 14:34:24 +02:00
|
|
|
expr_cache= 0;
|
2010-10-23 20:28:58 +02:00
|
|
|
forced_const= FALSE;
|
2011-06-21 14:50:07 +02:00
|
|
|
DBUG_PRINT("info", ("exec_counter: %d", exec_counter));
|
|
|
|
#ifndef DBUG_OFF
|
|
|
|
exec_counter= 0;
|
|
|
|
#endif
|
2004-02-08 19:14:13 +01:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
2010-04-06 09:26:59 +02:00
|
|
|
|
2004-02-08 19:14:13 +01:00
|
|
|
void Item_singlerow_subselect::cleanup()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_singlerow_subselect::cleanup");
|
|
|
|
value= 0; row= 0;
|
|
|
|
Item_subselect::cleanup();
|
|
|
|
DBUG_VOID_RETURN;
|
2003-12-30 11:08:19 +01:00
|
|
|
}
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2010-01-28 14:48:33 +01:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
******
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
2011-11-12 10:29:12 +01:00
|
|
|
/*
|
|
|
|
TODO: This breaks the commented assert in add_strategy().
|
|
|
|
in_strategy&= ~SUBS_STRATEGY_CHOSEN;
|
|
|
|
*/
|
2010-01-28 14:48:33 +01:00
|
|
|
first_execution= TRUE;
|
2010-09-05 17:43:47 +02:00
|
|
|
pushed_cond_guards= NULL;
|
2010-01-28 14:48:33 +01:00
|
|
|
Item_subselect::cleanup();
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
2011-07-18 22:45:38 +02:00
|
|
|
|
|
|
|
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())
|
2011-12-19 22:05:44 +01:00
|
|
|
if (test_set_strategy(SUBS_MAXMIN_INJECTED))
|
2011-07-18 22:45:38 +02:00
|
|
|
sl->with_sum_func= false;
|
|
|
|
Item_in_subselect::cleanup();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-09-03 08:50:36 +02:00
|
|
|
Item_subselect::~Item_subselect()
|
|
|
|
{
|
2011-05-04 17:08:58 +02:00
|
|
|
DBUG_ENTER("Item_subselect::~Item_subselect");
|
2017-09-19 19:45:17 +02:00
|
|
|
DBUG_PRINT("enter", ("this: %p", this));
|
2011-05-11 23:14:15 +02:00
|
|
|
if (own_engine)
|
2011-05-04 17:08:58 +02:00
|
|
|
delete engine;
|
|
|
|
else
|
|
|
|
engine->cleanup();
|
2010-07-16 12:52:02 +02:00
|
|
|
engine= NULL;
|
2011-05-04 17:08:58 +02:00
|
|
|
DBUG_VOID_RETURN;
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
bool
|
2004-07-04 07:46:28 +02:00
|
|
|
Item_subselect::select_transformer(JOIN *join)
|
2002-10-27 22:27:00 +01:00
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_subselect::select_transformer");
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(thd == join->thd);
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(false);
|
2002-10-27 22:27:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-07-01 06:05:42 +02:00
|
|
|
bool Item_subselect::fix_fields(THD *thd_param, Item **ref)
|
2002-05-12 22:46:42 +02:00
|
|
|
{
|
2004-09-06 14:14:10 +02:00
|
|
|
char const *save_where= thd_param->where;
|
Table definition cache, part 2
The table opening process now works the following way:
- Create common TABLE_SHARE object
- Read the .frm file and unpack it into the TABLE_SHARE object
- Create a TABLE object based on the information in the TABLE_SHARE
object and open a handler to the table object
Other noteworthy changes:
- In TABLE_SHARE the most common strings are now LEX_STRING's
- Better error message when table is not found
- Variable table_cache is now renamed 'table_open_cache'
- New variable 'table_definition_cache' that is the number of table defintions that will be cached
- strxnmov() calls are now fixed to avoid overflows
- strxnmov() will now always add one end \0 to result
- engine objects are now created with a TABLE_SHARE object instead of a TABLE object.
- After creating a field object one must call field->init(table) before using it
- For a busy system this change will give you:
- Less memory usage for table object
- Faster opening of tables (if it's has been in use or is in table definition cache)
- Allow you to cache many table definitions objects
- Faster drop of table
mysql-test/mysql-test-run.sh:
Fixed some problems with --gdb option
Test both with socket and tcp/ip port that all old servers are killed
mysql-test/r/flush_table.result:
More tests with lock table with 2 threads + flush table
mysql-test/r/information_schema.result:
Removed old (now wrong) result
mysql-test/r/innodb.result:
Better error messages (thanks to TDC patch)
mysql-test/r/merge.result:
Extra flush table test
mysql-test/r/ndb_bitfield.result:
Better error messages (thanks to TDC patch)
mysql-test/r/ndb_partition_error.result:
Better error messages (thanks to TDC patch)
mysql-test/r/query_cache.result:
Remove tables left from old tests
mysql-test/r/temp_table.result:
Test truncate with temporary tables
mysql-test/r/variables.result:
Table_cache -> Table_open_cache
mysql-test/t/flush_table.test:
More tests with lock table with 2 threads + flush table
mysql-test/t/merge.test:
Extra flush table test
mysql-test/t/multi_update.test:
Added 'sleep' to make test predictable
mysql-test/t/query_cache.test:
Remove tables left from old tests
mysql-test/t/temp_table.test:
Test truncate with temporary tables
mysql-test/t/variables.test:
Table_cache -> Table_open_cache
mysql-test/valgrind.supp:
Remove warning that may happens becasue threads dies in different order
mysys/hash.c:
Fixed wrong DBUG_PRINT
mysys/mf_dirname.c:
More DBUG
mysys/mf_pack.c:
Better comment
mysys/mf_tempdir.c:
More DBUG
Ensure that we call cleanup_dirname() on all temporary directory paths.
If we don't do this, we will get a failure when comparing temporary table
names as in some cases the temporary table name is run through convert_dirname())
mysys/my_alloc.c:
Indentation fix
sql/examples/ha_example.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/examples/ha_example.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/examples/ha_tina.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/examples/ha_tina.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/field.cc:
Update for table definition cache:
- Field creation now takes TABLE_SHARE instead of TABLE as argument
(This is becasue field definitions are now cached in TABLE_SHARE)
When a field is created, one now must call field->init(TABLE) before using it
- Use s->db instead of s->table_cache_key
- Added Field::clone() to create a field in TABLE from a field in TABLE_SHARE
- make_field() takes TABLE_SHARE as argument instead of TABLE
- move_field() -> move_field_offset()
sql/field.h:
Update for table definition cache:
- Field creation now takes TABLE_SHARE instead of TABLE as argument
(This is becasue field definitions are now cached in TABLE_SHARE)
When a field is created, one now must call field->init(TABLE) before using it
- Added Field::clone() to create a field in TABLE from a field in TABLE_SHARE
- make_field() takes TABLE_SHARE as argument instead of TABLE
- move_field() -> move_field_offset()
sql/ha_archive.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_archive.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_berkeley.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Changed name of argument create() to not hide internal 'table' variable.
table->s -> table_share
sql/ha_berkeley.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_blackhole.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_blackhole.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_federated.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Fixed comments
Remove index variable and replace with pointers (simple optimization)
move_field() -> move_field_offset()
Removed some strlen() calls
sql/ha_federated.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_heap.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Simplify delete_table() and create() as the given file names are now without extension
sql/ha_heap.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_innodb.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_innodb.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_myisam.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Remove not needed fn_format()
Fixed for new table->s structure
sql/ha_myisam.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_myisammrg.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Don't set 'is_view' for MERGE tables
Use new interface to find_temporary_table()
sql/ha_myisammrg.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Added flag HA_NO_COPY_ON_ALTER
sql/ha_ndbcluster.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Fixed wrong calls to strxnmov()
Give error HA_ERR_TABLE_DEF_CHANGED if table definition has changed
drop_table -> intern_drop_table()
table->s -> table_share
Move part_info to TABLE
Fixed comments & DBUG print's
New arguments to print_error()
sql/ha_ndbcluster.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_partition.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
We can't set up or use part_info when creating handler as there is not yet any table object
New ha_intialise() to work with TDC (Done by Mikael)
sql/ha_partition.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Got set_part_info() from Mikael
sql/handler.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
ha_delete_table() now also takes database as an argument
handler::ha_open() now takes TABLE as argument
ha_open() now calls ha_allocate_read_write_set()
Simplify ha_allocate_read_write_set()
Remove ha_deallocate_read_write_set()
Use table_share (Cached by table definition cache)
sql/handler.h:
New table flag: HA_NO_COPY_ON_ALTER (used by merge tables)
Remove ha_deallocate_read_write_set()
get_new_handler() now takes TABLE_SHARE as argument
ha_delete_table() now gets database as argument
sql/item.cc:
table_name and db are now LEX_STRING objects
When creating fields, we have now have to call field->init(table)
move_field -> move_field_offset()
sql/item.h:
tmp_table_field_from_field_type() now takes an extra paramenter 'fixed_length' to allow one to force usage of CHAR
instead of BLOB
sql/item_cmpfunc.cc:
Fixed call to tmp_table_field_from_field_type()
sql/item_create.cc:
Assert if new not handled cast type
sql/item_func.cc:
When creating fields, we have now have to call field->init(table)
dummy_table used by 'sp' now needs a TABLE_SHARE object
sql/item_subselect.cc:
Trivial code cleanups
sql/item_sum.cc:
When creating fields, we have now have to call field->init(table)
sql/item_timefunc.cc:
Item_func_str_to_date::tmp_table_field() now replaced by call to
tmp_table_field_from_field_type() (see item_timefunc.h)
sql/item_timefunc.h:
Simply tmp_table_field()
sql/item_uniq.cc:
When creating fields, we have now have to call field->init(table)
sql/key.cc:
Added 'KEY' argument to 'find_ref_key' to simplify code
sql/lock.cc:
More debugging
Use create_table_def_key() to create key for table cache
Allocate TABLE_SHARE properly when creating name lock
Fix that locked_table_name doesn't test same table twice
sql/mysql_priv.h:
New functions for table definition cache
New interfaces to a lot of functions.
New faster interface to find_temporary_table() and close_temporary_table()
sql/mysqld.cc:
Added support for table definition cache of size 'table_def_size'
Fixed som calls to strnmov()
Changed name of 'table_cache' to 'table_open_cache'
sql/opt_range.cc:
Use new interfaces
Fixed warnings from valgrind
sql/parse_file.cc:
Safer calls to strxnmov()
Fixed typo
sql/set_var.cc:
Added variable 'table_definition_cache'
Variable table_cache renamed to 'table_open_cache'
sql/slave.cc:
Use new interface
sql/sp.cc:
Proper use of TABLE_SHARE
sql/sp_head.cc:
Remove compiler warnings
We have now to call field->init(table)
sql/sp_head.h:
Pointers to parsed strings are now const
sql/sql_acl.cc:
table_name is now a LEX_STRING
sql/sql_base.cc:
Main implementation of table definition cache
(The #ifdef's are there for the future when table definition cache will replace open table cache)
Now table definitions are cached indepndent of open tables, which will speed up things when a table is in use at once from several places
Views are not yet cached; For the moment we only cache if a table is a view or not.
Faster implementation of find_temorary_table()
Replace 'wait_for_refresh()' with the more general function 'wait_for_condition()'
Drop table is slightly faster as we can use the table definition cache to know the type of the table
sql/sql_cache.cc:
table_cache_key and table_name are now LEX_STRING
'sDBUG print fixes
sql/sql_class.cc:
table_cache_key is now a LEX_STRING
safer strxnmov()
sql/sql_class.h:
Added number of open table shares (table definitions)
sql/sql_db.cc:
safer strxnmov()
sql/sql_delete.cc:
Use new interface to find_temporary_table()
sql/sql_derived.cc:
table_name is now a LEX_STRING
sql/sql_handler.cc:
TABLE_SHARE->db and TABLE_SHARE->table_name are now LEX_STRING's
sql/sql_insert.cc:
TABLE_SHARE->db and TABLE_SHARE->table_name are now LEX_STRING's
sql/sql_lex.cc:
Make parsed string a const (to quickly find out if anything is trying to change the query string)
sql/sql_lex.h:
Make parsed string a const (to quickly find out if anything is trying to change the query string)
sql/sql_load.cc:
Safer strxnmov()
sql/sql_parse.cc:
Better error if wrong DB name
sql/sql_partition.cc:
part_info moved to TABLE from TABLE_SHARE
Indentation changes
sql/sql_select.cc:
Indentation fixes
Call field->init(TABLE) for new created fields
Update create_tmp_table() to use TABLE_SHARE properly
sql/sql_select.h:
Call field->init(TABLE) for new created fields
sql/sql_show.cc:
table_name is now a LEX_STRING
part_info moved to TABLE
sql/sql_table.cc:
Use table definition cache to speed up delete of tables
Fixed calls to functions with new interfaces
Don't use 'share_not_to_be_used'
Instead of doing openfrm() when doing repair, we now have to call
get_table_share() followed by open_table_from_share().
Replace some fn_format() with faster unpack_filename().
Safer strxnmov()
part_info is now in TABLE
Added Mikaels patch for partition and ALTER TABLE
Instead of using 'TABLE_SHARE->is_view' use 'table_flags() & HA_NO_COPY_ON_ALTER
sql/sql_test.cc:
table_name and table_cache_key are now LEX_STRING's
sql/sql_trigger.cc:
TABLE_SHARE->db and TABLE_SHARE->table_name are now LEX_STRING's
safer strxnmov()
Removed compiler warnings
sql/sql_update.cc:
Call field->init(TABLE) after field is created
sql/sql_view.cc:
safer strxnmov()
Create common TABLE_SHARE object for views to allow us to cache if table is a view
sql/structs.h:
Added SHOW_TABLE_DEFINITIONS
sql/table.cc:
Creation and destruct of TABLE_SHARE objects that are common for many TABLE objects
The table opening process now works the following way:
- Create common TABLE_SHARE object
- Read the .frm file and unpack it into the TABLE_SHARE object
- Create a TABLE object based on the information in the TABLE_SHARE
object and open a handler to the table object
open_table_def() is written in such a way that it should be trival to add parsing of the .frm files in new formats
sql/table.h:
TABLE objects for the same database table now share a common TABLE_SHARE object
In TABLE_SHARE the most common strings are now LEX_STRING's
sql/unireg.cc:
Changed arguments to rea_create_table() to have same order as other functions
Call field->init(table) for new created fields
sql/unireg.h:
Added OPEN_VIEW
strings/strxnmov.c:
Change strxnmov() to always add end \0
This makes usage of strxnmov() safer as most of MySQL code assumes that strxnmov() will create a null terminated string
2005-11-23 21:45:02 +01:00
|
|
|
uint8 uncacheable;
|
2004-10-20 03:04:37 +02:00
|
|
|
bool res;
|
2004-09-06 14:14:10 +02:00
|
|
|
|
2016-06-22 14:17:06 +02:00
|
|
|
thd= thd_param;
|
|
|
|
|
|
|
|
DBUG_ASSERT(unit->thd == thd);
|
|
|
|
|
2012-10-04 22:52:11 +02:00
|
|
|
status_var_increment(thd_param->status_var.feature_subquery);
|
2012-09-09 00:22:06 +02:00
|
|
|
|
2004-03-17 13:26:26 +01:00
|
|
|
DBUG_ASSERT(fixed == 0);
|
2012-01-03 05:06:36 +01:00
|
|
|
engine->set_thd((thd= thd_param));
|
2010-02-12 00:59:58 +01:00
|
|
|
if (!done_first_fix_fields)
|
|
|
|
{
|
|
|
|
done_first_fix_fields= TRUE;
|
|
|
|
inside_first_fix_fields= TRUE;
|
2010-02-20 09:23:29 +01:00
|
|
|
upper_refs.empty();
|
|
|
|
/*
|
|
|
|
psergey-todo: remove _first_fix_fields calls, we need changes on every
|
|
|
|
execution
|
|
|
|
*/
|
2010-02-12 00:59:58 +01:00
|
|
|
}
|
|
|
|
|
2009-06-22 13:46:31 +02:00
|
|
|
eliminated= FALSE;
|
2010-02-12 00:59:58 +01:00
|
|
|
parent_select= thd_param->lex->current_select;
|
2003-05-28 15:52:56 +02:00
|
|
|
|
WL#3817: Simplify string / memory area types and make things more consistent (first part)
The following type conversions was done:
- Changed byte to uchar
- Changed gptr to uchar*
- Change my_string to char *
- Change my_size_t to size_t
- Change size_s to size_t
Removed declaration of byte, gptr, my_string, my_size_t and size_s.
Following function parameter changes was done:
- All string functions in mysys/strings was changed to use size_t
instead of uint for string lengths.
- All read()/write() functions changed to use size_t (including vio).
- All protocoll functions changed to use size_t instead of uint
- Functions that used a pointer to a string length was changed to use size_t*
- Changed malloc(), free() and related functions from using gptr to use void *
as this requires fewer casts in the code and is more in line with how the
standard functions work.
- Added extra length argument to dirname_part() to return the length of the
created string.
- Changed (at least) following functions to take uchar* as argument:
- db_dump()
- my_net_write()
- net_write_command()
- net_store_data()
- DBUG_DUMP()
- decimal2bin() & bin2decimal()
- Changed my_compress() and my_uncompress() to use size_t. Changed one
argument to my_uncompress() from a pointer to a value as we only return
one value (makes function easier to use).
- Changed type of 'pack_data' argument to packfrm() to avoid casts.
- Changed in readfrm() and writefrom(), ha_discover and handler::discover()
the type for argument 'frmdata' to uchar** to avoid casts.
- Changed most Field functions to use uchar* instead of char* (reduced a lot of
casts).
- Changed field->val_xxx(xxx, new_ptr) to take const pointers.
Other changes:
- Removed a lot of not needed casts
- Added a few new cast required by other changes
- Added some cast to my_multi_malloc() arguments for safety (as string lengths
needs to be uint, not size_t).
- Fixed all calls to hash-get-key functions to use size_t*. (Needed to be done
explicitely as this conflict was often hided by casting the function to
hash_get_key).
- Changed some buffers to memory regions to uchar* to avoid casts.
- Changed some string lengths from uint to size_t.
- Changed field->ptr to be uchar* instead of char*. This allowed us to
get rid of a lot of casts.
- Some changes from true -> TRUE, false -> FALSE, unsigned char -> uchar
- Include zlib.h in some files as we needed declaration of crc32()
- Changed MY_FILE_ERROR to be (size_t) -1.
- Changed many variables to hold the result of my_read() / my_write() to be
size_t. This was needed to properly detect errors (which are
returned as (size_t) -1).
- Removed some very old VMS code
- Changed packfrm()/unpackfrm() to not be depending on uint size
(portability fix)
- Removed windows specific code to restore cursor position as this
causes slowdown on windows and we should not mix read() and pread()
calls anyway as this is not thread safe. Updated function comment to
reflect this. Changed function that depended on original behavior of
my_pwrite() to itself restore the cursor position (one such case).
- Added some missing checking of return value of malloc().
- Changed definition of MOD_PAD_CHAR_TO_FULL_LENGTH to avoid 'long' overflow.
- Changed type of table_def::m_size from my_size_t to ulong to reflect that
m_size is the number of elements in the array, not a string/memory
length.
- Moved THD::max_row_length() to table.cc (as it's not depending on THD).
Inlined max_row_length_blob() into this function.
- More function comments
- Fixed some compiler warnings when compiled without partitions.
- Removed setting of LEX_STRING() arguments in declaration (portability fix).
- Some trivial indentation/variable name changes.
- Some trivial code simplifications:
- Replaced some calls to alloc_root + memcpy to use
strmake_root()/strdup_root().
- Changed some calls from memdup() to strmake() (Safety fix)
- Simpler loops in client-simple.c
BitKeeper/etc/ignore:
added libmysqld/ha_ndbcluster_cond.cc
---
added debian/defs.mk debian/control
client/completion_hash.cc:
Remove not needed casts
client/my_readline.h:
Remove some old types
client/mysql.cc:
Simplify types
client/mysql_upgrade.c:
Remove some old types
Update call to dirname_part
client/mysqladmin.cc:
Remove some old types
client/mysqlbinlog.cc:
Remove some old types
Change some buffers to be uchar to avoid casts
client/mysqlcheck.c:
Remove some old types
client/mysqldump.c:
Remove some old types
Remove some not needed casts
Change some string lengths to size_t
client/mysqlimport.c:
Remove some old types
client/mysqlshow.c:
Remove some old types
client/mysqlslap.c:
Remove some old types
Remove some not needed casts
client/mysqltest.c:
Removed some old types
Removed some not needed casts
Updated hash-get-key function arguments
Updated parameters to dirname_part()
client/readline.cc:
Removed some old types
Removed some not needed casts
Changed some string lengths to use size_t
client/sql_string.cc:
Removed some old types
dbug/dbug.c:
Removed some old types
Changed some string lengths to use size_t
Changed some prototypes to avoid casts
extra/comp_err.c:
Removed some old types
extra/innochecksum.c:
Removed some old types
extra/my_print_defaults.c:
Removed some old types
extra/mysql_waitpid.c:
Removed some old types
extra/perror.c:
Removed some old types
extra/replace.c:
Removed some old types
Updated parameters to dirname_part()
extra/resolve_stack_dump.c:
Removed some old types
extra/resolveip.c:
Removed some old types
include/config-win.h:
Removed some old types
include/decimal.h:
Changed binary strings to be uchar* instead of char*
include/ft_global.h:
Removed some old types
include/hash.h:
Removed some old types
include/heap.h:
Removed some old types
Changed records_under_level to be 'ulong' instead of 'uint' to clarify usage of variable
include/keycache.h:
Removed some old types
include/m_ctype.h:
Removed some old types
Changed some string lengths to use size_t
Changed character length functions to return uint
unsigned char -> uchar
include/m_string.h:
Removed some old types
Changed some string lengths to use size_t
include/my_alloc.h:
Changed some string lengths to use size_t
include/my_base.h:
Removed some old types
include/my_dbug.h:
Removed some old types
Changed some string lengths to use size_t
Changed db_dump() to take uchar * as argument for memory to reduce number of casts in usage
include/my_getopt.h:
Removed some old types
include/my_global.h:
Removed old types:
my_size_t -> size_t
byte -> uchar
gptr -> uchar *
include/my_list.h:
Removed some old types
include/my_nosys.h:
Removed some old types
include/my_pthread.h:
Removed some old types
include/my_sys.h:
Removed some old types
Changed MY_FILE_ERROR to be in line with new definitions of my_write()/my_read()
Changed some string lengths to use size_t
my_malloc() / my_free() now uses void *
Updated parameters to dirname_part() & my_uncompress()
include/my_tree.h:
Removed some old types
include/my_trie.h:
Removed some old types
include/my_user.h:
Changed some string lengths to use size_t
include/my_vle.h:
Removed some old types
include/my_xml.h:
Removed some old types
Changed some string lengths to use size_t
include/myisam.h:
Removed some old types
include/myisammrg.h:
Removed some old types
include/mysql.h:
Removed some old types
Changed byte streams to use uchar* instead of char*
include/mysql_com.h:
Removed some old types
Changed some string lengths to use size_t
Changed some buffers to be uchar* to avoid casts
include/queues.h:
Removed some old types
include/sql_common.h:
Removed some old types
include/sslopt-longopts.h:
Removed some old types
include/violite.h:
Removed some old types
Changed some string lengths to use size_t
libmysql/client_settings.h:
Removed some old types
libmysql/libmysql.c:
Removed some old types
libmysql/manager.c:
Removed some old types
libmysqld/emb_qcache.cc:
Removed some old types
libmysqld/emb_qcache.h:
Removed some old types
libmysqld/lib_sql.cc:
Removed some old types
Removed some not needed casts
Changed some buffers to be uchar* to avoid casts
true -> TRUE, false -> FALSE
mysys/array.c:
Removed some old types
mysys/charset.c:
Changed some string lengths to use size_t
mysys/checksum.c:
Include zlib to get definition for crc32
Removed some old types
mysys/default.c:
Removed some old types
Changed some string lengths to use size_t
mysys/default_modify.c:
Changed some string lengths to use size_t
Removed some not needed casts
mysys/hash.c:
Removed some old types
Changed some string lengths to use size_t
Note: Prototype of hash_key() has changed which may cause problems if client uses hash_init() with a cast for the hash-get-key function.
hash_element now takes 'ulong' as the index type (cleanup)
mysys/list.c:
Removed some old types
mysys/mf_cache.c:
Changed some string lengths to use size_t
mysys/mf_dirname.c:
Removed some old types
Changed some string lengths to use size_t
Added argument to dirname_part() to avoid calculation of length for 'to'
mysys/mf_fn_ext.c:
Removed some old types
Updated parameters to dirname_part()
mysys/mf_format.c:
Removed some old types
Changed some string lengths to use size_t
mysys/mf_getdate.c:
Removed some old types
mysys/mf_iocache.c:
Removed some old types
Changed some string lengths to use size_t
Changed calculation of 'max_length' to be done the same way in all functions
mysys/mf_iocache2.c:
Removed some old types
Changed some string lengths to use size_t
Clean up comments
Removed not needed indentation
mysys/mf_keycache.c:
Removed some old types
mysys/mf_keycaches.c:
Removed some old types
mysys/mf_loadpath.c:
Removed some old types
mysys/mf_pack.c:
Removed some old types
Changed some string lengths to use size_t
Removed some not needed casts
Removed very old VMS code
Updated parameters to dirname_part()
Use result of dirnam_part() to remove call to strcat()
mysys/mf_path.c:
Removed some old types
mysys/mf_radix.c:
Removed some old types
mysys/mf_same.c:
Removed some old types
mysys/mf_sort.c:
Removed some old types
mysys/mf_soundex.c:
Removed some old types
mysys/mf_strip.c:
Removed some old types
mysys/mf_tempdir.c:
Removed some old types
mysys/mf_unixpath.c:
Removed some old types
mysys/mf_wfile.c:
Removed some old types
mysys/mulalloc.c:
Removed some old types
mysys/my_alloc.c:
Removed some old types
Changed some string lengths to use size_t
Use void* as type for allocated memory area
Removed some not needed casts
Changed argument 'Size' to 'length' according coding guidelines
mysys/my_chsize.c:
Changed some buffers to be uchar* to avoid casts
mysys/my_compress.c:
More comments
Removed some old types
Changed string lengths to use size_t
Changed arguments to my_uncompress() to make them easier to understand
Changed packfrm()/unpackfrm() to not be depending on uint size (portability fix)
Changed type of 'pack_data' argument to packfrm() to avoid casts.
mysys/my_conio.c:
Changed some string lengths to use size_t
mysys/my_create.c:
Removed some old types
mysys/my_div.c:
Removed some old types
mysys/my_error.c:
Removed some old types
mysys/my_fopen.c:
Removed some old types
mysys/my_fstream.c:
Removed some old types
Changed some string lengths to use size_t
writen -> written
mysys/my_getopt.c:
Removed some old types
mysys/my_getwd.c:
Removed some old types
More comments
mysys/my_init.c:
Removed some old types
mysys/my_largepage.c:
Removed some old types
Changed some string lengths to use size_t
mysys/my_lib.c:
Removed some old types
mysys/my_lockmem.c:
Removed some old types
mysys/my_malloc.c:
Removed some old types
Changed malloc(), free() and related functions to use void *
Changed all functions to use size_t
mysys/my_memmem.c:
Indentation cleanup
mysys/my_once.c:
Removed some old types
Changed malloc(), free() and related functions to use void *
mysys/my_open.c:
Removed some old types
mysys/my_pread.c:
Removed some old types
Changed all functions to use size_t
Added comment for how my_pread() / my_pwrite() are supposed to work.
Removed windows specific code to restore cursor position as this causes slowdown on windows and we should not mix read() and pread() calls anyway as this is not thread safe.
(If we ever would really need this, it should be enabled only with a flag argument)
mysys/my_quick.c:
Removed some old types
Changed all functions to use size_t
mysys/my_read.c:
Removed some old types
Changed all functions to use size_t
mysys/my_realloc.c:
Removed some old types
Use void* as type for allocated memory area
Changed all functions to use size_t
mysys/my_static.c:
Removed some old types
mysys/my_static.h:
Removed some old types
mysys/my_vle.c:
Removed some old types
mysys/my_wincond.c:
Removed some old types
mysys/my_windac.c:
Removed some old types
mysys/my_write.c:
Removed some old types
Changed all functions to use size_t
mysys/ptr_cmp.c:
Removed some old types
Changed all functions to use size_t
mysys/queues.c:
Removed some old types
mysys/safemalloc.c:
Removed some old types
Changed malloc(), free() and related functions to use void *
Changed all functions to use size_t
mysys/string.c:
Removed some old types
Changed all functions to use size_t
mysys/testhash.c:
Removed some old types
mysys/thr_alarm.c:
Removed some old types
mysys/thr_lock.c:
Removed some old types
mysys/tree.c:
Removed some old types
mysys/trie.c:
Removed some old types
mysys/typelib.c:
Removed some old types
plugin/daemon_example/daemon_example.cc:
Removed some old types
regex/reginit.c:
Removed some old types
server-tools/instance-manager/buffer.cc:
Changed some string lengths to use size_t
Changed buffer to be of type uchar*
server-tools/instance-manager/buffer.h:
Changed some string lengths to use size_t
Changed buffer to be of type uchar*
server-tools/instance-manager/commands.cc:
Removed some old types
Changed some string lengths to use size_t
Changed buffer to be of type uchar*
server-tools/instance-manager/instance_map.cc:
Removed some old types
Changed some string lengths to use size_t
Changed buffer to be of type uchar*
server-tools/instance-manager/instance_options.cc:
Changed buffer to be of type uchar*
Replaced alloc_root + strcpy() with strdup_root()
server-tools/instance-manager/mysql_connection.cc:
Changed buffer to be of type uchar*
server-tools/instance-manager/options.cc:
Removed some old types
server-tools/instance-manager/parse.cc:
Changed some string lengths to use size_t
server-tools/instance-manager/parse.h:
Removed some old types
Changed some string lengths to use size_t
server-tools/instance-manager/protocol.cc:
Changed some buffers to be uchar* to avoid casts
Changed some string lengths to use size_t
server-tools/instance-manager/protocol.h:
Changed some string lengths to use size_t
server-tools/instance-manager/user_map.cc:
Removed some old types
Changed some string lengths to use size_t
sql/derror.cc:
Removed some old types
Changed some buffers to be uchar* to avoid casts
Changed some string lengths to use size_t
sql/discover.cc:
Changed in readfrm() and writefrom() the type for argument 'frmdata' to uchar** to avoid casts
Changed some string lengths to use size_t
Changed some buffers to be uchar* to avoid casts
sql/event_data_objects.cc:
Removed some old types
Added missing casts for alloc() and sprintf()
sql/event_db_repository.cc:
Changed some buffers to be uchar* to avoid casts
Added missing casts for sprintf()
sql/event_queue.cc:
Removed some old types
sql/field.cc:
Removed some old types
Changed memory buffers to be uchar*
Changed some string lengths to use size_t
Removed a lot of casts
Safety fix in Field_blob::val_decimal() to not access zero pointer
sql/field.h:
Removed some old types
Changed memory buffers to be uchar* (except of store() as this would have caused too many other changes).
Changed some string lengths to use size_t
Removed some not needed casts
Changed val_xxx(xxx, new_ptr) to take const pointers
sql/field_conv.cc:
Removed some old types
Added casts required because memory area pointers are now uchar*
sql/filesort.cc:
Initalize variable that was used unitialized in error conditions
sql/gen_lex_hash.cc:
Removed some old types
Changed memory buffers to be uchar*
Changed some string lengths to use size_t
Removed a lot of casts
Safety fix in Field_blob::val_decimal() to not access zero pointer
sql/gstream.h:
Added required cast
sql/ha_ndbcluster.cc:
Removed some old types
Updated hash-get-key function arguments
Changed some buffers to be uchar* to avoid casts
Added required casts
Removed some not needed casts
sql/ha_ndbcluster.h:
Removed some old types
sql/ha_ndbcluster_binlog.cc:
Removed some old types
Changed some buffers to be uchar* to avoid casts
Replaced sql_alloc() + memcpy() + set end 0 with sql_strmake()
Changed some string lengths to use size_t
Added missing casts for alloc() and sprintf()
sql/ha_ndbcluster_binlog.h:
Removed some old types
sql/ha_ndbcluster_cond.cc:
Removed some old types
Removed some not needed casts
sql/ha_ndbcluster_cond.h:
Removed some old types
sql/ha_partition.cc:
Removed some old types
Changed prototype for change_partition() to avoid casts
sql/ha_partition.h:
Removed some old types
sql/handler.cc:
Removed some old types
Changed some string lengths to use size_t
sql/handler.h:
Removed some old types
Changed some string lengths to use size_t
Changed type for 'frmblob' parameter for discover() and ha_discover() to get fewer casts
sql/hash_filo.h:
Removed some old types
Changed all functions to use size_t
sql/hostname.cc:
Removed some old types
sql/item.cc:
Removed some old types
Changed some string lengths to use size_t
Use strmake() instead of memdup() to create a null terminated string.
Updated calls to new Field()
sql/item.h:
Removed some old types
Changed malloc(), free() and related functions to use void *
Changed some buffers to be uchar* to avoid casts
sql/item_cmpfunc.cc:
Removed some old types
Changed some buffers to be uchar* to avoid casts
sql/item_cmpfunc.h:
Removed some old types
sql/item_create.cc:
Removed some old types
sql/item_func.cc:
Removed some old types
Changed some buffers to be uchar* to avoid casts
Removed some not needed casts
Added test for failing alloc() in init_result_field()
Remove old confusing comment
Fixed compiler warning
sql/item_func.h:
Removed some old types
sql/item_row.cc:
Removed some old types
sql/item_row.h:
Removed some old types
sql/item_strfunc.cc:
Include zlib (needed becasue we call crc32)
Removed some old types
sql/item_strfunc.h:
Removed some old types
Changed some types to match new function prototypes
sql/item_subselect.cc:
Removed some old types
sql/item_subselect.h:
Removed some old types
sql/item_sum.cc:
Removed some old types
Changed some buffers to be uchar* to avoid casts
Removed some not needed casts
sql/item_sum.h:
Removed some old types
sql/item_timefunc.cc:
Removed some old types
Changed some string lengths to use size_t
sql/item_timefunc.h:
Removed some old types
sql/item_xmlfunc.cc:
Changed some string lengths to use size_t
sql/item_xmlfunc.h:
Removed some old types
sql/key.cc:
Removed some old types
Removed some not needed casts
sql/lock.cc:
Removed some old types
Added some cast to my_multi_malloc() arguments for safety
sql/log.cc:
Removed some old types
Changed some string lengths to use size_t
Changed some buffers to be uchar* to avoid casts
Changed usage of pwrite() to not assume it holds the cursor position for the file
Made usage of my_read() safer
sql/log_event.cc:
Removed some old types
Added checking of return value of malloc() in pack_info()
Changed some buffers to be uchar* to avoid casts
Removed some 'const' to avoid casts
Added missing casts for alloc() and sprintf()
Added required casts
Removed some not needed casts
Added some cast to my_multi_malloc() arguments for safety
sql/log_event.h:
Removed some old types
Changed some buffers to be uchar* to avoid casts
sql/log_event_old.cc:
Changed some buffers to be uchar* to avoid casts
Removed some not needed casts
sql/log_event_old.h:
Changed some buffers to be uchar* to avoid casts
sql/mf_iocache.cc:
Removed some old types
sql/my_decimal.cc:
Changed memory area to use uchar*
sql/my_decimal.h:
Changed memory area to use uchar*
sql/mysql_priv.h:
Removed some old types
Changed malloc(), free() and related functions to use void *
Changed some string lengths to use size_t
Changed definition of MOD_PAD_CHAR_TO_FULL_LENGTH to avoid long overflow
Changed some buffers to be uchar* to avoid casts
sql/mysqld.cc:
Removed some old types
sql/net_serv.cc:
Removed some old types
Changed some string lengths to use size_t
Changed some buffers to be uchar* to avoid casts
Ensure that vio_read()/vio_write() return values are stored in a size_t variable
Removed some not needed casts
sql/opt_range.cc:
Removed some old types
Changed some buffers to be uchar* to avoid casts
Removed some not needed casts
sql/opt_range.h:
Removed some old types
Changed some buffers to be uchar* to avoid casts
sql/opt_sum.cc:
Removed some old types
Removed some not needed casts
sql/parse_file.cc:
Removed some old types
Changed some string lengths to use size_t
Changed alloc_root + memcpy + set end 0 -> strmake_root()
sql/parse_file.h:
Removed some old types
sql/partition_info.cc:
Removed some old types
Added missing casts for alloc()
Changed some buffers to be uchar* to avoid casts
sql/partition_info.h:
Changed some buffers to be uchar* to avoid casts
sql/protocol.cc:
Removed some old types
Changed some buffers to be uchar* to avoid casts
Removed some not needed casts
sql/protocol.h:
Removed some old types
Changed some buffers to be uchar* to avoid casts
Changed some string lengths to use size_t
sql/records.cc:
Removed some old types
sql/repl_failsafe.cc:
Removed some old types
Changed some string lengths to use size_t
Added required casts
sql/rpl_filter.cc:
Removed some old types
Updated hash-get-key function arguments
Changed some string lengths to use size_t
sql/rpl_filter.h:
Changed some string lengths to use size_t
sql/rpl_injector.h:
Removed some old types
sql/rpl_record.cc:
Removed some old types
Removed some not needed casts
Changed some buffers to be uchar* to avoid casts
sql/rpl_record.h:
Removed some old types
Changed some buffers to be uchar* to avoid casts
sql/rpl_record_old.cc:
Removed some old types
Changed some buffers to be uchar* to avoid casts
Removed some not needed casts
sql/rpl_record_old.h:
Removed some old types
Changed some buffers to be uchar* to avoid cast
sql/rpl_rli.cc:
Removed some old types
sql/rpl_tblmap.cc:
Removed some old types
sql/rpl_tblmap.h:
Removed some old types
sql/rpl_utility.cc:
Removed some old types
sql/rpl_utility.h:
Removed some old types
Changed type of m_size from my_size_t to ulong to reflect that m_size is the number of elements in the array, not a string/memory length
sql/set_var.cc:
Removed some old types
Updated parameters to dirname_part()
sql/set_var.h:
Removed some old types
sql/slave.cc:
Removed some old types
Changed some string lengths to use size_t
sql/slave.h:
Removed some old types
sql/sp.cc:
Removed some old types
Added missing casts for printf()
sql/sp.h:
Removed some old types
Updated hash-get-key function arguments
sql/sp_cache.cc:
Removed some old types
Added missing casts for printf()
Updated hash-get-key function arguments
sql/sp_head.cc:
Removed some old types
Added missing casts for alloc() and printf()
Added required casts
Updated hash-get-key function arguments
sql/sp_head.h:
Removed some old types
sql/sp_pcontext.cc:
Removed some old types
sql/sp_pcontext.h:
Removed some old types
sql/sql_acl.cc:
Removed some old types
Changed some string lengths to use size_t
Changed some buffers to be uchar* to avoid casts
Removed some not needed casts
Added required casts
sql/sql_analyse.cc:
Changed some buffers to be uchar* to avoid casts
sql/sql_analyse.h:
Changed some buffers to be uchar* to avoid casts
sql/sql_array.h:
Removed some old types
sql/sql_base.cc:
Removed some old types
Updated hash-get-key function arguments
sql/sql_binlog.cc:
Removed some old types
Added missing casts for printf()
sql/sql_cache.cc:
Removed some old types
Updated hash-get-key function arguments
Removed some not needed casts
Changed some string lengths to use size_t
sql/sql_cache.h:
Removed some old types
Removed reference to not existing function cache_key()
Updated hash-get-key function arguments
sql/sql_class.cc:
Removed some old types
Updated hash-get-key function arguments
Added missing casts for alloc()
Updated hash-get-key function arguments
Moved THD::max_row_length() to table.cc (as it's not depending on THD)
Removed some not needed casts
sql/sql_class.h:
Removed some old types
Changed malloc(), free() and related functions to use void *
Removed some not needed casts
Changed some string lengths to use size_t
Moved max_row_length and max_row_length_blob() to table.cc, as they are not depending on THD
sql/sql_connect.cc:
Removed some old types
Added required casts
sql/sql_db.cc:
Removed some old types
Removed some not needed casts
Added some cast to my_multi_malloc() arguments for safety
Added missing casts for alloc()
sql/sql_delete.cc:
Removed some old types
sql/sql_handler.cc:
Removed some old types
Updated hash-get-key function arguments
Added some cast to my_multi_malloc() arguments for safety
sql/sql_help.cc:
Removed some old types
Changed some buffers to be uchar* to avoid casts
Removed some not needed casts
sql/sql_insert.cc:
Removed some old types
Added missing casts for alloc() and printf()
sql/sql_lex.cc:
Removed some old types
sql/sql_lex.h:
Removed some old types
Removed some not needed casts
sql/sql_list.h:
Removed some old types
Removed some not needed casts
sql/sql_load.cc:
Removed some old types
Removed compiler warning
sql/sql_manager.cc:
Removed some old types
sql/sql_map.cc:
Removed some old types
sql/sql_map.h:
Removed some old types
sql/sql_olap.cc:
Removed some old types
sql/sql_parse.cc:
Removed some old types
Trivial move of code lines to make things more readable
Changed some string lengths to use size_t
Added missing casts for alloc()
sql/sql_partition.cc:
Removed some old types
Removed compiler warnings about not used functions
Changed some buffers to be uchar* to avoid casts
Removed some not needed casts
sql/sql_partition.h:
Removed some old types
Changed some buffers to be uchar* to avoid casts
sql/sql_plugin.cc:
Removed some old types
Added missing casts for alloc()
Updated hash-get-key function arguments
sql/sql_prepare.cc:
Removed some old types
Changed some buffers to be uchar* to avoid casts
Added missing casts for alloc() and printf()
sql-common/client.c:
Removed some old types
Changed some memory areas to use uchar*
sql-common/my_user.c:
Changed some string lengths to use size_t
sql-common/pack.c:
Changed some buffers to be uchar* to avoid casts
sql/sql_repl.cc:
Added required casts
Changed some buffers to be uchar* to avoid casts
Changed some string lengths to use size_t
sql/sql_select.cc:
Removed some old types
Changed some buffers to be uchar* to avoid casts
Removed some old types
sql/sql_select.h:
Removed some old types
Changed some buffers to be uchar* to avoid casts
sql/sql_servers.cc:
Removed some old types
Updated hash-get-key function arguments
sql/sql_show.cc:
Removed some old types
Added missing casts for alloc()
Removed some not needed casts
sql/sql_string.cc:
Removed some old types
Added required casts
sql/sql_table.cc:
Removed some old types
Removed compiler warning about not used variable
Changed some buffers to be uchar* to avoid casts
Removed some not needed casts
sql/sql_test.cc:
Removed some old types
sql/sql_trigger.cc:
Removed some old types
Added missing casts for alloc()
sql/sql_udf.cc:
Removed some old types
Updated hash-get-key function arguments
sql/sql_union.cc:
Removed some old types
sql/sql_update.cc:
Removed some old types
Removed some not needed casts
sql/sql_view.cc:
Removed some old types
sql/sql_yacc.yy:
Removed some old types
Changed some string lengths to use size_t
Added missing casts for alloc()
sql/stacktrace.c:
Removed some old types
sql/stacktrace.h:
Removed some old types
sql/structs.h:
Removed some old types
sql/table.cc:
Removed some old types
Updated hash-get-key function arguments
Changed some buffers to be uchar* to avoid casts
Removed setting of LEX_STRING() arguments in declaration
Added required casts
More function comments
Moved max_row_length() here from sql_class.cc/sql_class.h
sql/table.h:
Removed some old types
Changed some string lengths to use size_t
sql/thr_malloc.cc:
Use void* as type for allocated memory area
Changed all functions to use size_t
sql/tzfile.h:
Changed some buffers to be uchar* to avoid casts
sql/tztime.cc:
Changed some buffers to be uchar* to avoid casts
Updated hash-get-key function arguments
Added missing casts for alloc()
Removed some not needed casts
sql/uniques.cc:
Removed some old types
Removed some not needed casts
sql/unireg.cc:
Removed some old types
Changed some buffers to be uchar* to avoid casts
Removed some not needed casts
Added missing casts for alloc()
storage/archive/archive_reader.c:
Removed some old types
storage/archive/azio.c:
Removed some old types
Removed some not needed casts
storage/archive/ha_archive.cc:
Removed some old types
Changed type for 'frmblob' in archive_discover() to match handler
Updated hash-get-key function arguments
Removed some not needed casts
storage/archive/ha_archive.h:
Removed some old types
storage/blackhole/ha_blackhole.cc:
Removed some old types
storage/blackhole/ha_blackhole.h:
Removed some old types
storage/csv/ha_tina.cc:
Removed some old types
Updated hash-get-key function arguments
Changed some buffers to be uchar* to avoid casts
storage/csv/ha_tina.h:
Removed some old types
Removed some not needed casts
storage/csv/transparent_file.cc:
Removed some old types
Changed type of 'bytes_read' to be able to detect read errors
Fixed indentation
storage/csv/transparent_file.h:
Removed some old types
storage/example/ha_example.cc:
Removed some old types
Updated hash-get-key function arguments
storage/example/ha_example.h:
Removed some old types
storage/federated/ha_federated.cc:
Removed some old types
Updated hash-get-key function arguments
Removed some not needed casts
storage/federated/ha_federated.h:
Removed some old types
storage/heap/_check.c:
Changed some buffers to be uchar* to avoid casts
storage/heap/_rectest.c:
Removed some old types
storage/heap/ha_heap.cc:
Removed some old types
storage/heap/ha_heap.h:
Removed some old types
storage/heap/heapdef.h:
Removed some old types
storage/heap/hp_block.c:
Removed some old types
Changed some string lengths to use size_t
storage/heap/hp_clear.c:
Removed some old types
storage/heap/hp_close.c:
Removed some old types
storage/heap/hp_create.c:
Removed some old types
storage/heap/hp_delete.c:
Removed some old types
storage/heap/hp_hash.c:
Removed some old types
storage/heap/hp_info.c:
Removed some old types
storage/heap/hp_open.c:
Removed some old types
storage/heap/hp_rfirst.c:
Removed some old types
storage/heap/hp_rkey.c:
Removed some old types
storage/heap/hp_rlast.c:
Removed some old types
storage/heap/hp_rnext.c:
Removed some old types
storage/heap/hp_rprev.c:
Removed some old types
storage/heap/hp_rrnd.c:
Removed some old types
storage/heap/hp_rsame.c:
Removed some old types
storage/heap/hp_scan.c:
Removed some old types
storage/heap/hp_test1.c:
Removed some old types
storage/heap/hp_test2.c:
Removed some old types
storage/heap/hp_update.c:
Removed some old types
storage/heap/hp_write.c:
Removed some old types
Changed some string lengths to use size_t
storage/innobase/handler/ha_innodb.cc:
Removed some old types
Updated hash-get-key function arguments
Added missing casts for alloc() and printf()
Removed some not needed casts
storage/innobase/handler/ha_innodb.h:
Removed some old types
storage/myisam/ft_boolean_search.c:
Removed some old types
storage/myisam/ft_nlq_search.c:
Removed some old types
storage/myisam/ft_parser.c:
Removed some old types
Changed some buffers to be uchar* to avoid casts
storage/myisam/ft_static.c:
Removed some old types
storage/myisam/ft_stopwords.c:
Removed some old types
storage/myisam/ft_update.c:
Removed some old types
Changed some buffers to be uchar* to avoid casts
storage/myisam/ftdefs.h:
Removed some old types
Changed some buffers to be uchar* to avoid casts
storage/myisam/fulltext.h:
Removed some old types
storage/myisam/ha_myisam.cc:
Removed some old types
storage/myisam/ha_myisam.h:
Removed some old types
storage/myisam/mi_cache.c:
Removed some old types
Changed some buffers to be uchar* to avoid casts
storage/myisam/mi_check.c:
Removed some old types
storage/myisam/mi_checksum.c:
Removed some old types
storage/myisam/mi_close.c:
Removed some old types
storage/myisam/mi_create.c:
Removed some old types
storage/myisam/mi_delete.c:
Removed some old types
storage/myisam/mi_delete_all.c:
Removed some old types
storage/myisam/mi_dynrec.c:
Removed some old types
storage/myisam/mi_extra.c:
Removed some old types
storage/myisam/mi_key.c:
Removed some old types
storage/myisam/mi_locking.c:
Removed some old types
storage/myisam/mi_log.c:
Removed some old types
storage/myisam/mi_open.c:
Removed some old types
Removed some not needed casts
Check argument of my_write()/my_pwrite() in functions returning int
Added casting of string lengths to size_t
storage/myisam/mi_packrec.c:
Removed some old types
Changed some buffers to be uchar* to avoid casts
storage/myisam/mi_page.c:
Removed some old types
storage/myisam/mi_preload.c:
Removed some old types
storage/myisam/mi_range.c:
Removed some old types
storage/myisam/mi_rfirst.c:
Removed some old types
storage/myisam/mi_rkey.c:
Removed some old types
storage/myisam/mi_rlast.c:
Removed some old types
storage/myisam/mi_rnext.c:
Removed some old types
storage/myisam/mi_rnext_same.c:
Removed some old types
storage/myisam/mi_rprev.c:
Removed some old types
storage/myisam/mi_rrnd.c:
Removed some old types
storage/myisam/mi_rsame.c:
Removed some old types
storage/myisam/mi_rsamepos.c:
Removed some old types
storage/myisam/mi_scan.c:
Removed some old types
storage/myisam/mi_search.c:
Removed some old types
storage/myisam/mi_static.c:
Removed some old types
storage/myisam/mi_statrec.c:
Removed some old types
storage/myisam/mi_test1.c:
Removed some old types
storage/myisam/mi_test2.c:
Removed some old types
storage/myisam/mi_test3.c:
Removed some old types
storage/myisam/mi_unique.c:
Removed some old types
storage/myisam/mi_update.c:
Removed some old types
storage/myisam/mi_write.c:
Removed some old types
storage/myisam/myisam_ftdump.c:
Removed some old types
storage/myisam/myisamchk.c:
Removed some old types
storage/myisam/myisamdef.h:
Removed some old types
storage/myisam/myisamlog.c:
Removed some old types
Indentation fix
storage/myisam/myisampack.c:
Removed some old types
storage/myisam/rt_index.c:
Removed some old types
storage/myisam/rt_split.c:
Removed some old types
storage/myisam/sort.c:
Removed some old types
storage/myisam/sp_defs.h:
Removed some old types
storage/myisam/sp_key.c:
Removed some old types
storage/myisammrg/ha_myisammrg.cc:
Removed some old types
storage/myisammrg/ha_myisammrg.h:
Removed some old types
storage/myisammrg/myrg_close.c:
Removed some old types
storage/myisammrg/myrg_def.h:
Removed some old types
storage/myisammrg/myrg_delete.c:
Removed some old types
storage/myisammrg/myrg_open.c:
Removed some old types
Updated parameters to dirname_part()
storage/myisammrg/myrg_queue.c:
Removed some old types
storage/myisammrg/myrg_rfirst.c:
Removed some old types
storage/myisammrg/myrg_rkey.c:
Removed some old types
storage/myisammrg/myrg_rlast.c:
Removed some old types
storage/myisammrg/myrg_rnext.c:
Removed some old types
storage/myisammrg/myrg_rnext_same.c:
Removed some old types
storage/myisammrg/myrg_rprev.c:
Removed some old types
storage/myisammrg/myrg_rrnd.c:
Removed some old types
storage/myisammrg/myrg_rsame.c:
Removed some old types
storage/myisammrg/myrg_update.c:
Removed some old types
storage/myisammrg/myrg_write.c:
Removed some old types
storage/ndb/include/util/ndb_opts.h:
Removed some old types
storage/ndb/src/cw/cpcd/main.cpp:
Removed some old types
storage/ndb/src/kernel/vm/Configuration.cpp:
Removed some old types
storage/ndb/src/mgmclient/main.cpp:
Removed some old types
storage/ndb/src/mgmsrv/InitConfigFileParser.cpp:
Removed some old types
Removed old disabled code
storage/ndb/src/mgmsrv/main.cpp:
Removed some old types
storage/ndb/src/ndbapi/NdbBlob.cpp:
Removed some old types
storage/ndb/src/ndbapi/NdbOperationDefine.cpp:
Removed not used variable
storage/ndb/src/ndbapi/NdbOperationInt.cpp:
Added required casts
storage/ndb/src/ndbapi/NdbScanOperation.cpp:
Added required casts
storage/ndb/tools/delete_all.cpp:
Removed some old types
storage/ndb/tools/desc.cpp:
Removed some old types
storage/ndb/tools/drop_index.cpp:
Removed some old types
storage/ndb/tools/drop_tab.cpp:
Removed some old types
storage/ndb/tools/listTables.cpp:
Removed some old types
storage/ndb/tools/ndb_config.cpp:
Removed some old types
storage/ndb/tools/restore/consumer_restore.cpp:
Changed some buffers to be uchar* to avoid casts with new defintion of packfrm()
storage/ndb/tools/restore/restore_main.cpp:
Removed some old types
storage/ndb/tools/select_all.cpp:
Removed some old types
storage/ndb/tools/select_count.cpp:
Removed some old types
storage/ndb/tools/waiter.cpp:
Removed some old types
strings/bchange.c:
Changed function to use uchar * and size_t
strings/bcmp.c:
Changed function to use uchar * and size_t
strings/bmove512.c:
Changed function to use uchar * and size_t
strings/bmove_upp.c:
Changed function to use uchar * and size_t
strings/ctype-big5.c:
Changed functions to use size_t
Changed character length functions to return uint
strings/ctype-bin.c:
Changed functions to use size_t
strings/ctype-cp932.c:
Changed functions to use size_t
Changed character length functions to return uint
strings/ctype-czech.c:
Fixed indentation
Changed functions to use size_t
strings/ctype-euc_kr.c:
Changed functions to use size_t
Changed character length functions to return uint
strings/ctype-eucjpms.c:
Changed functions to use size_t
Changed character length functions to return uint
unsigned char -> uchar
strings/ctype-gb2312.c:
Changed functions to use size_t
Changed character length functions to return uint
strings/ctype-gbk.c:
Changed functions to use size_t
Changed character length functions to return uint
strings/ctype-latin1.c:
Changed functions to use size_t
Changed character length functions to return uint
unsigned char -> uchar
strings/ctype-mb.c:
Changed functions to use size_t
Changed character length functions to return uint
strings/ctype-simple.c:
Changed functions to use size_t
Simpler loops for caseup/casedown
unsigned int -> uint
unsigned char -> uchar
strings/ctype-sjis.c:
Changed functions to use size_t
Changed character length functions to return uint
strings/ctype-tis620.c:
Changed functions to use size_t
Changed character length functions to return uint
unsigned char -> uchar
strings/ctype-uca.c:
Changed functions to use size_t
unsigned char -> uchar
strings/ctype-ucs2.c:
Moved inclusion of stdarg.h to other includes
usigned char -> uchar
Changed functions to use size_t
Changed character length functions to return uint
strings/ctype-ujis.c:
Changed functions to use size_t
Changed character length functions to return uint
unsigned char -> uchar
strings/ctype-utf8.c:
Changed functions to use size_t
unsigned char -> uchar
Indentation fixes
strings/ctype-win1250ch.c:
Indentation fixes
Changed functions to use size_t
strings/ctype.c:
Changed functions to use size_t
strings/decimal.c:
Changed type for memory argument to uchar *
strings/do_ctype.c:
Indentation fixes
strings/my_strtoll10.c:
unsigned char -> uchar
strings/my_vsnprintf.c:
Changed functions to use size_t
strings/r_strinstr.c:
Removed some old types
Changed functions to use size_t
strings/str_test.c:
Removed some old types
strings/strappend.c:
Changed functions to use size_t
strings/strcont.c:
Removed some old types
strings/strfill.c:
Removed some old types
strings/strinstr.c:
Changed functions to use size_t
strings/strlen.c:
Changed functions to use size_t
strings/strmake.c:
Changed functions to use size_t
strings/strnlen.c:
Changed functions to use size_t
strings/strnmov.c:
Changed functions to use size_t
strings/strto.c:
unsigned char -> uchar
strings/strtod.c:
Changed functions to use size_t
strings/strxnmov.c:
Changed functions to use size_t
strings/xml.c:
Changed functions to use size_t
Indentation fixes
tests/mysql_client_test.c:
Removed some old types
tests/thread_test.c:
Removed some old types
vio/test-ssl.c:
Removed some old types
vio/test-sslclient.c:
Removed some old types
vio/test-sslserver.c:
Removed some old types
vio/vio.c:
Removed some old types
vio/vio_priv.h:
Removed some old types
Changed vio_read()/vio_write() to work with size_t
vio/viosocket.c:
Changed vio_read()/vio_write() to work with size_t
Indentation fixes
vio/viossl.c:
Changed vio_read()/vio_write() to work with size_t
Indentation fixes
vio/viosslfactories.c:
Removed some old types
vio/viotest-ssl.c:
Removed some old types
win/README:
More explanations
2007-05-10 11:59:39 +02:00
|
|
|
if (check_stack_overrun(thd, STACK_MIN_SIZE, (uchar*)&res))
|
2004-10-20 03:04:37 +02:00
|
|
|
return TRUE;
|
2009-06-22 13:46:31 +02:00
|
|
|
|
2017-09-04 22:29:58 +02:00
|
|
|
for (SELECT_LEX *sl= unit->first_select(); sl; sl= sl->next_select())
|
2017-08-29 16:58:32 +02:00
|
|
|
{
|
2017-09-04 22:29:58 +02:00
|
|
|
if (sl->tvc)
|
|
|
|
{
|
2019-05-08 09:08:09 +02:00
|
|
|
wrap_tvc_into_select(thd, sl);
|
2017-09-04 22:29:58 +02:00
|
|
|
}
|
2017-08-29 16:58:32 +02:00
|
|
|
}
|
2011-08-29 17:57:41 +02:00
|
|
|
|
2016-06-22 14:17:06 +02:00
|
|
|
if (!(res= engine->prepare(thd)))
|
2002-10-12 00:09:47 +02:00
|
|
|
{
|
2009-09-04 10:14:54 +02:00
|
|
|
// all transformation is done (used by prepared statements)
|
|
|
|
changed= 1;
|
2010-09-05 17:43:47 +02:00
|
|
|
inside_first_fix_fields= FALSE;
|
2010-01-28 14:48:33 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
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.
|
|
|
|
*/
|
2003-05-14 20:51:33 +02:00
|
|
|
if (substitution)
|
|
|
|
{
|
2010-09-05 17:43:47 +02:00
|
|
|
/*
|
|
|
|
If the top item of the WHERE/HAVING condition changed,
|
|
|
|
set correct WHERE/HAVING for PS.
|
|
|
|
*/
|
2004-02-08 19:14:13 +01:00
|
|
|
if (unit->outer_select()->where == (*ref))
|
2011-01-18 18:15:55 +01:00
|
|
|
unit->outer_select()->where= substitution;
|
2004-12-08 22:37:17 +01:00
|
|
|
else if (unit->outer_select()->having == (*ref))
|
2011-01-18 18:15:55 +01:00
|
|
|
unit->outer_select()->having= substitution;
|
2004-02-08 19:14:13 +01:00
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
(*ref)= substitution;
|
|
|
|
substitution->name= name;
|
|
|
|
if (have_to_be_excluded)
|
|
|
|
engine->exclude();
|
|
|
|
substitution= 0;
|
2004-07-04 07:46:28 +02:00
|
|
|
thd->where= "checking transformed subquery";
|
2018-06-05 08:25:39 +02:00
|
|
|
res= (*ref)->fix_fields_if_needed(thd, ref);
|
2009-11-16 21:49:51 +01:00
|
|
|
goto end;
|
2010-05-10 17:28:19 +02:00
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
2002-10-12 00:09:47 +02:00
|
|
|
// Is it one field subselect?
|
|
|
|
if (engine->cols() > max_columns)
|
2004-07-04 07:46:28 +02:00
|
|
|
{
|
2003-10-06 21:35:05 +02:00
|
|
|
my_error(ER_OPERAND_COLUMNS, MYF(0), 1);
|
2018-05-08 15:26:26 +02:00
|
|
|
res= TRUE;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
if (fix_length_and_dec())
|
|
|
|
{
|
|
|
|
res= TRUE;
|
2009-11-16 21:49:51 +01:00
|
|
|
goto end;
|
2002-10-12 00:09:47 +02:00
|
|
|
}
|
|
|
|
}
|
2005-01-18 15:26:05 +01:00
|
|
|
else
|
2009-11-16 21:49:51 +01:00
|
|
|
goto end;
|
Table definition cache, part 2
The table opening process now works the following way:
- Create common TABLE_SHARE object
- Read the .frm file and unpack it into the TABLE_SHARE object
- Create a TABLE object based on the information in the TABLE_SHARE
object and open a handler to the table object
Other noteworthy changes:
- In TABLE_SHARE the most common strings are now LEX_STRING's
- Better error message when table is not found
- Variable table_cache is now renamed 'table_open_cache'
- New variable 'table_definition_cache' that is the number of table defintions that will be cached
- strxnmov() calls are now fixed to avoid overflows
- strxnmov() will now always add one end \0 to result
- engine objects are now created with a TABLE_SHARE object instead of a TABLE object.
- After creating a field object one must call field->init(table) before using it
- For a busy system this change will give you:
- Less memory usage for table object
- Faster opening of tables (if it's has been in use or is in table definition cache)
- Allow you to cache many table definitions objects
- Faster drop of table
mysql-test/mysql-test-run.sh:
Fixed some problems with --gdb option
Test both with socket and tcp/ip port that all old servers are killed
mysql-test/r/flush_table.result:
More tests with lock table with 2 threads + flush table
mysql-test/r/information_schema.result:
Removed old (now wrong) result
mysql-test/r/innodb.result:
Better error messages (thanks to TDC patch)
mysql-test/r/merge.result:
Extra flush table test
mysql-test/r/ndb_bitfield.result:
Better error messages (thanks to TDC patch)
mysql-test/r/ndb_partition_error.result:
Better error messages (thanks to TDC patch)
mysql-test/r/query_cache.result:
Remove tables left from old tests
mysql-test/r/temp_table.result:
Test truncate with temporary tables
mysql-test/r/variables.result:
Table_cache -> Table_open_cache
mysql-test/t/flush_table.test:
More tests with lock table with 2 threads + flush table
mysql-test/t/merge.test:
Extra flush table test
mysql-test/t/multi_update.test:
Added 'sleep' to make test predictable
mysql-test/t/query_cache.test:
Remove tables left from old tests
mysql-test/t/temp_table.test:
Test truncate with temporary tables
mysql-test/t/variables.test:
Table_cache -> Table_open_cache
mysql-test/valgrind.supp:
Remove warning that may happens becasue threads dies in different order
mysys/hash.c:
Fixed wrong DBUG_PRINT
mysys/mf_dirname.c:
More DBUG
mysys/mf_pack.c:
Better comment
mysys/mf_tempdir.c:
More DBUG
Ensure that we call cleanup_dirname() on all temporary directory paths.
If we don't do this, we will get a failure when comparing temporary table
names as in some cases the temporary table name is run through convert_dirname())
mysys/my_alloc.c:
Indentation fix
sql/examples/ha_example.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/examples/ha_example.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/examples/ha_tina.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/examples/ha_tina.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/field.cc:
Update for table definition cache:
- Field creation now takes TABLE_SHARE instead of TABLE as argument
(This is becasue field definitions are now cached in TABLE_SHARE)
When a field is created, one now must call field->init(TABLE) before using it
- Use s->db instead of s->table_cache_key
- Added Field::clone() to create a field in TABLE from a field in TABLE_SHARE
- make_field() takes TABLE_SHARE as argument instead of TABLE
- move_field() -> move_field_offset()
sql/field.h:
Update for table definition cache:
- Field creation now takes TABLE_SHARE instead of TABLE as argument
(This is becasue field definitions are now cached in TABLE_SHARE)
When a field is created, one now must call field->init(TABLE) before using it
- Added Field::clone() to create a field in TABLE from a field in TABLE_SHARE
- make_field() takes TABLE_SHARE as argument instead of TABLE
- move_field() -> move_field_offset()
sql/ha_archive.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_archive.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_berkeley.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Changed name of argument create() to not hide internal 'table' variable.
table->s -> table_share
sql/ha_berkeley.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_blackhole.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_blackhole.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_federated.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Fixed comments
Remove index variable and replace with pointers (simple optimization)
move_field() -> move_field_offset()
Removed some strlen() calls
sql/ha_federated.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_heap.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Simplify delete_table() and create() as the given file names are now without extension
sql/ha_heap.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_innodb.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_innodb.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_myisam.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Remove not needed fn_format()
Fixed for new table->s structure
sql/ha_myisam.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_myisammrg.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Don't set 'is_view' for MERGE tables
Use new interface to find_temporary_table()
sql/ha_myisammrg.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Added flag HA_NO_COPY_ON_ALTER
sql/ha_ndbcluster.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Fixed wrong calls to strxnmov()
Give error HA_ERR_TABLE_DEF_CHANGED if table definition has changed
drop_table -> intern_drop_table()
table->s -> table_share
Move part_info to TABLE
Fixed comments & DBUG print's
New arguments to print_error()
sql/ha_ndbcluster.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_partition.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
We can't set up or use part_info when creating handler as there is not yet any table object
New ha_intialise() to work with TDC (Done by Mikael)
sql/ha_partition.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Got set_part_info() from Mikael
sql/handler.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
ha_delete_table() now also takes database as an argument
handler::ha_open() now takes TABLE as argument
ha_open() now calls ha_allocate_read_write_set()
Simplify ha_allocate_read_write_set()
Remove ha_deallocate_read_write_set()
Use table_share (Cached by table definition cache)
sql/handler.h:
New table flag: HA_NO_COPY_ON_ALTER (used by merge tables)
Remove ha_deallocate_read_write_set()
get_new_handler() now takes TABLE_SHARE as argument
ha_delete_table() now gets database as argument
sql/item.cc:
table_name and db are now LEX_STRING objects
When creating fields, we have now have to call field->init(table)
move_field -> move_field_offset()
sql/item.h:
tmp_table_field_from_field_type() now takes an extra paramenter 'fixed_length' to allow one to force usage of CHAR
instead of BLOB
sql/item_cmpfunc.cc:
Fixed call to tmp_table_field_from_field_type()
sql/item_create.cc:
Assert if new not handled cast type
sql/item_func.cc:
When creating fields, we have now have to call field->init(table)
dummy_table used by 'sp' now needs a TABLE_SHARE object
sql/item_subselect.cc:
Trivial code cleanups
sql/item_sum.cc:
When creating fields, we have now have to call field->init(table)
sql/item_timefunc.cc:
Item_func_str_to_date::tmp_table_field() now replaced by call to
tmp_table_field_from_field_type() (see item_timefunc.h)
sql/item_timefunc.h:
Simply tmp_table_field()
sql/item_uniq.cc:
When creating fields, we have now have to call field->init(table)
sql/key.cc:
Added 'KEY' argument to 'find_ref_key' to simplify code
sql/lock.cc:
More debugging
Use create_table_def_key() to create key for table cache
Allocate TABLE_SHARE properly when creating name lock
Fix that locked_table_name doesn't test same table twice
sql/mysql_priv.h:
New functions for table definition cache
New interfaces to a lot of functions.
New faster interface to find_temporary_table() and close_temporary_table()
sql/mysqld.cc:
Added support for table definition cache of size 'table_def_size'
Fixed som calls to strnmov()
Changed name of 'table_cache' to 'table_open_cache'
sql/opt_range.cc:
Use new interfaces
Fixed warnings from valgrind
sql/parse_file.cc:
Safer calls to strxnmov()
Fixed typo
sql/set_var.cc:
Added variable 'table_definition_cache'
Variable table_cache renamed to 'table_open_cache'
sql/slave.cc:
Use new interface
sql/sp.cc:
Proper use of TABLE_SHARE
sql/sp_head.cc:
Remove compiler warnings
We have now to call field->init(table)
sql/sp_head.h:
Pointers to parsed strings are now const
sql/sql_acl.cc:
table_name is now a LEX_STRING
sql/sql_base.cc:
Main implementation of table definition cache
(The #ifdef's are there for the future when table definition cache will replace open table cache)
Now table definitions are cached indepndent of open tables, which will speed up things when a table is in use at once from several places
Views are not yet cached; For the moment we only cache if a table is a view or not.
Faster implementation of find_temorary_table()
Replace 'wait_for_refresh()' with the more general function 'wait_for_condition()'
Drop table is slightly faster as we can use the table definition cache to know the type of the table
sql/sql_cache.cc:
table_cache_key and table_name are now LEX_STRING
'sDBUG print fixes
sql/sql_class.cc:
table_cache_key is now a LEX_STRING
safer strxnmov()
sql/sql_class.h:
Added number of open table shares (table definitions)
sql/sql_db.cc:
safer strxnmov()
sql/sql_delete.cc:
Use new interface to find_temporary_table()
sql/sql_derived.cc:
table_name is now a LEX_STRING
sql/sql_handler.cc:
TABLE_SHARE->db and TABLE_SHARE->table_name are now LEX_STRING's
sql/sql_insert.cc:
TABLE_SHARE->db and TABLE_SHARE->table_name are now LEX_STRING's
sql/sql_lex.cc:
Make parsed string a const (to quickly find out if anything is trying to change the query string)
sql/sql_lex.h:
Make parsed string a const (to quickly find out if anything is trying to change the query string)
sql/sql_load.cc:
Safer strxnmov()
sql/sql_parse.cc:
Better error if wrong DB name
sql/sql_partition.cc:
part_info moved to TABLE from TABLE_SHARE
Indentation changes
sql/sql_select.cc:
Indentation fixes
Call field->init(TABLE) for new created fields
Update create_tmp_table() to use TABLE_SHARE properly
sql/sql_select.h:
Call field->init(TABLE) for new created fields
sql/sql_show.cc:
table_name is now a LEX_STRING
part_info moved to TABLE
sql/sql_table.cc:
Use table definition cache to speed up delete of tables
Fixed calls to functions with new interfaces
Don't use 'share_not_to_be_used'
Instead of doing openfrm() when doing repair, we now have to call
get_table_share() followed by open_table_from_share().
Replace some fn_format() with faster unpack_filename().
Safer strxnmov()
part_info is now in TABLE
Added Mikaels patch for partition and ALTER TABLE
Instead of using 'TABLE_SHARE->is_view' use 'table_flags() & HA_NO_COPY_ON_ALTER
sql/sql_test.cc:
table_name and table_cache_key are now LEX_STRING's
sql/sql_trigger.cc:
TABLE_SHARE->db and TABLE_SHARE->table_name are now LEX_STRING's
safer strxnmov()
Removed compiler warnings
sql/sql_update.cc:
Call field->init(TABLE) after field is created
sql/sql_view.cc:
safer strxnmov()
Create common TABLE_SHARE object for views to allow us to cache if table is a view
sql/structs.h:
Added SHOW_TABLE_DEFINITIONS
sql/table.cc:
Creation and destruct of TABLE_SHARE objects that are common for many TABLE objects
The table opening process now works the following way:
- Create common TABLE_SHARE object
- Read the .frm file and unpack it into the TABLE_SHARE object
- Create a TABLE object based on the information in the TABLE_SHARE
object and open a handler to the table object
open_table_def() is written in such a way that it should be trival to add parsing of the .frm files in new formats
sql/table.h:
TABLE objects for the same database table now share a common TABLE_SHARE object
In TABLE_SHARE the most common strings are now LEX_STRING's
sql/unireg.cc:
Changed arguments to rea_create_table() to have same order as other functions
Call field->init(table) for new created fields
sql/unireg.h:
Added OPEN_VIEW
strings/strxnmov.c:
Change strxnmov() to always add end \0
This makes usage of strxnmov() safer as most of MySQL code assumes that strxnmov() will create a null terminated string
2005-11-23 21:45:02 +01:00
|
|
|
|
2017-04-20 22:09:31 +02:00
|
|
|
if ((uncacheable= engine->uncacheable() & ~UNCACHEABLE_EXPLAIN) ||
|
|
|
|
with_recursive_reference)
|
2003-10-23 19:50:53 +02:00
|
|
|
{
|
|
|
|
const_item_cache= 0;
|
2003-11-17 19:53:40 +01:00
|
|
|
if (uncacheable & UNCACHEABLE_RAND)
|
|
|
|
used_tables_cache|= RAND_TABLE_BIT;
|
2003-10-23 19:50:53 +02:00
|
|
|
}
|
2002-11-21 10:01:33 +01:00
|
|
|
fixed= 1;
|
Table definition cache, part 2
The table opening process now works the following way:
- Create common TABLE_SHARE object
- Read the .frm file and unpack it into the TABLE_SHARE object
- Create a TABLE object based on the information in the TABLE_SHARE
object and open a handler to the table object
Other noteworthy changes:
- In TABLE_SHARE the most common strings are now LEX_STRING's
- Better error message when table is not found
- Variable table_cache is now renamed 'table_open_cache'
- New variable 'table_definition_cache' that is the number of table defintions that will be cached
- strxnmov() calls are now fixed to avoid overflows
- strxnmov() will now always add one end \0 to result
- engine objects are now created with a TABLE_SHARE object instead of a TABLE object.
- After creating a field object one must call field->init(table) before using it
- For a busy system this change will give you:
- Less memory usage for table object
- Faster opening of tables (if it's has been in use or is in table definition cache)
- Allow you to cache many table definitions objects
- Faster drop of table
mysql-test/mysql-test-run.sh:
Fixed some problems with --gdb option
Test both with socket and tcp/ip port that all old servers are killed
mysql-test/r/flush_table.result:
More tests with lock table with 2 threads + flush table
mysql-test/r/information_schema.result:
Removed old (now wrong) result
mysql-test/r/innodb.result:
Better error messages (thanks to TDC patch)
mysql-test/r/merge.result:
Extra flush table test
mysql-test/r/ndb_bitfield.result:
Better error messages (thanks to TDC patch)
mysql-test/r/ndb_partition_error.result:
Better error messages (thanks to TDC patch)
mysql-test/r/query_cache.result:
Remove tables left from old tests
mysql-test/r/temp_table.result:
Test truncate with temporary tables
mysql-test/r/variables.result:
Table_cache -> Table_open_cache
mysql-test/t/flush_table.test:
More tests with lock table with 2 threads + flush table
mysql-test/t/merge.test:
Extra flush table test
mysql-test/t/multi_update.test:
Added 'sleep' to make test predictable
mysql-test/t/query_cache.test:
Remove tables left from old tests
mysql-test/t/temp_table.test:
Test truncate with temporary tables
mysql-test/t/variables.test:
Table_cache -> Table_open_cache
mysql-test/valgrind.supp:
Remove warning that may happens becasue threads dies in different order
mysys/hash.c:
Fixed wrong DBUG_PRINT
mysys/mf_dirname.c:
More DBUG
mysys/mf_pack.c:
Better comment
mysys/mf_tempdir.c:
More DBUG
Ensure that we call cleanup_dirname() on all temporary directory paths.
If we don't do this, we will get a failure when comparing temporary table
names as in some cases the temporary table name is run through convert_dirname())
mysys/my_alloc.c:
Indentation fix
sql/examples/ha_example.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/examples/ha_example.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/examples/ha_tina.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/examples/ha_tina.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/field.cc:
Update for table definition cache:
- Field creation now takes TABLE_SHARE instead of TABLE as argument
(This is becasue field definitions are now cached in TABLE_SHARE)
When a field is created, one now must call field->init(TABLE) before using it
- Use s->db instead of s->table_cache_key
- Added Field::clone() to create a field in TABLE from a field in TABLE_SHARE
- make_field() takes TABLE_SHARE as argument instead of TABLE
- move_field() -> move_field_offset()
sql/field.h:
Update for table definition cache:
- Field creation now takes TABLE_SHARE instead of TABLE as argument
(This is becasue field definitions are now cached in TABLE_SHARE)
When a field is created, one now must call field->init(TABLE) before using it
- Added Field::clone() to create a field in TABLE from a field in TABLE_SHARE
- make_field() takes TABLE_SHARE as argument instead of TABLE
- move_field() -> move_field_offset()
sql/ha_archive.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_archive.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_berkeley.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Changed name of argument create() to not hide internal 'table' variable.
table->s -> table_share
sql/ha_berkeley.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_blackhole.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_blackhole.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_federated.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Fixed comments
Remove index variable and replace with pointers (simple optimization)
move_field() -> move_field_offset()
Removed some strlen() calls
sql/ha_federated.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_heap.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Simplify delete_table() and create() as the given file names are now without extension
sql/ha_heap.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_innodb.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_innodb.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_myisam.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Remove not needed fn_format()
Fixed for new table->s structure
sql/ha_myisam.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_myisammrg.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Don't set 'is_view' for MERGE tables
Use new interface to find_temporary_table()
sql/ha_myisammrg.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Added flag HA_NO_COPY_ON_ALTER
sql/ha_ndbcluster.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Fixed wrong calls to strxnmov()
Give error HA_ERR_TABLE_DEF_CHANGED if table definition has changed
drop_table -> intern_drop_table()
table->s -> table_share
Move part_info to TABLE
Fixed comments & DBUG print's
New arguments to print_error()
sql/ha_ndbcluster.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_partition.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
We can't set up or use part_info when creating handler as there is not yet any table object
New ha_intialise() to work with TDC (Done by Mikael)
sql/ha_partition.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Got set_part_info() from Mikael
sql/handler.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
ha_delete_table() now also takes database as an argument
handler::ha_open() now takes TABLE as argument
ha_open() now calls ha_allocate_read_write_set()
Simplify ha_allocate_read_write_set()
Remove ha_deallocate_read_write_set()
Use table_share (Cached by table definition cache)
sql/handler.h:
New table flag: HA_NO_COPY_ON_ALTER (used by merge tables)
Remove ha_deallocate_read_write_set()
get_new_handler() now takes TABLE_SHARE as argument
ha_delete_table() now gets database as argument
sql/item.cc:
table_name and db are now LEX_STRING objects
When creating fields, we have now have to call field->init(table)
move_field -> move_field_offset()
sql/item.h:
tmp_table_field_from_field_type() now takes an extra paramenter 'fixed_length' to allow one to force usage of CHAR
instead of BLOB
sql/item_cmpfunc.cc:
Fixed call to tmp_table_field_from_field_type()
sql/item_create.cc:
Assert if new not handled cast type
sql/item_func.cc:
When creating fields, we have now have to call field->init(table)
dummy_table used by 'sp' now needs a TABLE_SHARE object
sql/item_subselect.cc:
Trivial code cleanups
sql/item_sum.cc:
When creating fields, we have now have to call field->init(table)
sql/item_timefunc.cc:
Item_func_str_to_date::tmp_table_field() now replaced by call to
tmp_table_field_from_field_type() (see item_timefunc.h)
sql/item_timefunc.h:
Simply tmp_table_field()
sql/item_uniq.cc:
When creating fields, we have now have to call field->init(table)
sql/key.cc:
Added 'KEY' argument to 'find_ref_key' to simplify code
sql/lock.cc:
More debugging
Use create_table_def_key() to create key for table cache
Allocate TABLE_SHARE properly when creating name lock
Fix that locked_table_name doesn't test same table twice
sql/mysql_priv.h:
New functions for table definition cache
New interfaces to a lot of functions.
New faster interface to find_temporary_table() and close_temporary_table()
sql/mysqld.cc:
Added support for table definition cache of size 'table_def_size'
Fixed som calls to strnmov()
Changed name of 'table_cache' to 'table_open_cache'
sql/opt_range.cc:
Use new interfaces
Fixed warnings from valgrind
sql/parse_file.cc:
Safer calls to strxnmov()
Fixed typo
sql/set_var.cc:
Added variable 'table_definition_cache'
Variable table_cache renamed to 'table_open_cache'
sql/slave.cc:
Use new interface
sql/sp.cc:
Proper use of TABLE_SHARE
sql/sp_head.cc:
Remove compiler warnings
We have now to call field->init(table)
sql/sp_head.h:
Pointers to parsed strings are now const
sql/sql_acl.cc:
table_name is now a LEX_STRING
sql/sql_base.cc:
Main implementation of table definition cache
(The #ifdef's are there for the future when table definition cache will replace open table cache)
Now table definitions are cached indepndent of open tables, which will speed up things when a table is in use at once from several places
Views are not yet cached; For the moment we only cache if a table is a view or not.
Faster implementation of find_temorary_table()
Replace 'wait_for_refresh()' with the more general function 'wait_for_condition()'
Drop table is slightly faster as we can use the table definition cache to know the type of the table
sql/sql_cache.cc:
table_cache_key and table_name are now LEX_STRING
'sDBUG print fixes
sql/sql_class.cc:
table_cache_key is now a LEX_STRING
safer strxnmov()
sql/sql_class.h:
Added number of open table shares (table definitions)
sql/sql_db.cc:
safer strxnmov()
sql/sql_delete.cc:
Use new interface to find_temporary_table()
sql/sql_derived.cc:
table_name is now a LEX_STRING
sql/sql_handler.cc:
TABLE_SHARE->db and TABLE_SHARE->table_name are now LEX_STRING's
sql/sql_insert.cc:
TABLE_SHARE->db and TABLE_SHARE->table_name are now LEX_STRING's
sql/sql_lex.cc:
Make parsed string a const (to quickly find out if anything is trying to change the query string)
sql/sql_lex.h:
Make parsed string a const (to quickly find out if anything is trying to change the query string)
sql/sql_load.cc:
Safer strxnmov()
sql/sql_parse.cc:
Better error if wrong DB name
sql/sql_partition.cc:
part_info moved to TABLE from TABLE_SHARE
Indentation changes
sql/sql_select.cc:
Indentation fixes
Call field->init(TABLE) for new created fields
Update create_tmp_table() to use TABLE_SHARE properly
sql/sql_select.h:
Call field->init(TABLE) for new created fields
sql/sql_show.cc:
table_name is now a LEX_STRING
part_info moved to TABLE
sql/sql_table.cc:
Use table definition cache to speed up delete of tables
Fixed calls to functions with new interfaces
Don't use 'share_not_to_be_used'
Instead of doing openfrm() when doing repair, we now have to call
get_table_share() followed by open_table_from_share().
Replace some fn_format() with faster unpack_filename().
Safer strxnmov()
part_info is now in TABLE
Added Mikaels patch for partition and ALTER TABLE
Instead of using 'TABLE_SHARE->is_view' use 'table_flags() & HA_NO_COPY_ON_ALTER
sql/sql_test.cc:
table_name and table_cache_key are now LEX_STRING's
sql/sql_trigger.cc:
TABLE_SHARE->db and TABLE_SHARE->table_name are now LEX_STRING's
safer strxnmov()
Removed compiler warnings
sql/sql_update.cc:
Call field->init(TABLE) after field is created
sql/sql_view.cc:
safer strxnmov()
Create common TABLE_SHARE object for views to allow us to cache if table is a view
sql/structs.h:
Added SHOW_TABLE_DEFINITIONS
sql/table.cc:
Creation and destruct of TABLE_SHARE objects that are common for many TABLE objects
The table opening process now works the following way:
- Create common TABLE_SHARE object
- Read the .frm file and unpack it into the TABLE_SHARE object
- Create a TABLE object based on the information in the TABLE_SHARE
object and open a handler to the table object
open_table_def() is written in such a way that it should be trival to add parsing of the .frm files in new formats
sql/table.h:
TABLE objects for the same database table now share a common TABLE_SHARE object
In TABLE_SHARE the most common strings are now LEX_STRING's
sql/unireg.cc:
Changed arguments to rea_create_table() to have same order as other functions
Call field->init(table) for new created fields
sql/unireg.h:
Added OPEN_VIEW
strings/strxnmov.c:
Change strxnmov() to always add end \0
This makes usage of strxnmov() safer as most of MySQL code assumes that strxnmov() will create a null terminated string
2005-11-23 21:45:02 +01:00
|
|
|
|
2009-11-16 21:49:51 +01:00
|
|
|
end:
|
2010-02-21 04:36:18 +01:00
|
|
|
done_first_fix_fields= FALSE;
|
2010-05-10 17:28:19 +02:00
|
|
|
inside_first_fix_fields= FALSE;
|
2002-11-24 20:10:52 +01:00
|
|
|
thd->where= save_where;
|
2002-09-28 17:34:56 +02:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
BitKeeper/etc/ignore:
added mysys/test_bitmap
include/base64.h:
Removed my_global.h, as this must be included first in any program
include/heap.h:
Added heap_reset() (Required by new handler interface)
include/my_base.h:
Removed HA_EXTRA_RESET. MySQL will now call ::reset() instead of ::extra(HA_EXTRA_RESET).
HA_EXTRA_RETRIVE_ALL_COLS and HA_EXTRA_RETRIVE_PRIMARY key are deleted as the column bitmaps makes these unnecessary
include/my_bitmap.h:
Remove my_pthread.h (should be included at upper level)
Introduced my_bitmap_map typedef to make it the bitmap handling more like a black box
Added bitmap_is_overlapping(), bitmap_test_and_clear(), bitmap_copy() and bitmap_cmp()
Made bitmap_set_bit(), bitmap_flip_bit(), bitmap_clear_bit() return void
include/myisam.h:
Added mi_reset() (Required by new handler interface)
include/myisammrg.h:
Added myrg_reset() (Required by new handler interface)
include/mysql_com.h:
Added flag FIELD_IN_ADD_INDEX to be able to remove Field->add_index
mysql-test/extra/binlog_tests/mix_innodb_myisam_binlog.test:
Added testing of CREATE ... SELECT in a mixed environment
(This found some bugs that Mats is going to fix shortly)
mysql-test/install_test_db.sh:
Simplify ldata usage
Added --tmpdir=. option to mysqld bootstrap (Removed some warnings when TMPDIR was wrongly set)
mysql-test/mysql-test-run.pl:
Added --tmpdir=. to bootstrap
mysql-test/mysql-test-run.sh:
Use copy instead of INSTALL_DB for master and slave databases.
(Speeds up startup time a lot!)
Remove snapshot directories at startup (removes some strange warnings)
mysql-test/r/binlog_row_mix_innodb_myisam.result:
Added testing of CREATE ... SELECT in a mixed environment
(This found some bugs that Mats is going to fix shortly)
mysql-test/r/binlog_stm_mix_innodb_myisam.result:
Added testing of CREATE ... SELECT in a mixed environment
mysql-test/r/create.result:
Some extra tests of warnings and number of tables opened by CREATE ... SELECT
mysql-test/r/federated.result:
Drop some left over tables
Added testing of multiple table update and multiple table delete (with and without keys)
mysql-test/r/func_gconcat.result:
Enable some disabled tests (converted them slightly to be predictable)
mysql-test/r/func_time.result:
Added drop of test function
mysql-test/r/innodb_mysql.result:
Added tests for CREATE ... SELECT
mysql-test/r/insert.result:
More tests
Added testing of duplicate columns in insert
mysql-test/r/loaddata.result:
Added testing LOAD DATA ... SET ...
mysql-test/r/multi_update.result:
Test multi updates and deletes using primary key and without
mysql-test/r/ndb_index_unique.result:
Better error message
mysql-test/r/ndb_replace.result:
New correct result after fixing REPLACE handling with NDB
mysql-test/r/rpl_ddl.result:
Now we don't get these (wrong) warnings anymore
mysql-test/r/view_grant.result:
Drop used views
mysql-test/t/create.test:
Some extra tests of warnings and number of tables opened by CREATE ... SELECT
mysql-test/t/federated.test:
Drop some left over tables
Added testing of multiple table update and multiple table delete (with and without keys)
mysql-test/t/func_gconcat.test:
Enable some disabled tests (converted them slightly to be predictable)
mysql-test/t/func_time.test:
Added drop of test function
mysql-test/t/innodb_mysql.test:
Added tests for CREATE ... SELECT
mysql-test/t/insert.test:
More tests
Added testing of duplicate columns in insert
mysql-test/t/loaddata.test:
Added testing LOAD DATA ... SET ...
mysql-test/t/multi_update.test:
Test multi updates and deletes using primary key and without
mysql-test/t/view_grant.test:
Drop used views
mysql-test/valgrind.supp:
Added supression of not needed warnings when printing stack trace
mysys/base64.c:
Include my_global.h first
mysys/my_bitmap.c:
Added bitmap_is_overlapping(), bitmap_test_and_clear() and bitmap_copy()
Changed logic of bitmap handling to be a bit more efficent (Did this together with Mikael Ronström)
Now the 'extra, not used bits' in the bitmap are assumed to have a 'random value' and the bitmap functions are free to change them whenever needed.
Changed how mutex is allocated to make 'bitmap_free()' function simpler.
mysys/thr_lock.c:
Added 0x before thread pointers (for easier comparison of DBUG traces)
sql/event.cc:
Ensure 'use_all_columns()' is used for event tables
Don't print warning that event table is damaged if it doesn't exists.
sql/field.cc:
Added ASSERT_COLUMN_MARKED_FOR_WRITE in all store() methods and ASSERT_COLUMN_MARKED_FOR_READ in all val() methods to catch wrong setting if table->read_set and table->write_set
(Rest of changes are only indentation cleanups)
sql/field.h:
Removed Field->query_id (replaced by table->read_set and table->write_set)
Removed Field->fieldnr (use Field->field_index instead)
Removed Field->add_index (Use Field->flags instead)
Add Field->part_of_key_not_clustered (for usage in opt_range.cc)
sql/filesort.cc:
Added paramater sort_postion to filesort() to force sorting by position instead of storing all fields in the result set.
This allowed me to remove checking of sql_command.
Create a temporary column bitmap for fields that are used by the sorting process.
Use column bitmaps instead of query_id
sql/ha_berkeley.cc:
Update to 'newer' table handler interface
sql/ha_berkeley.h:
Update to 'newer' table handler interface
sql/ha_federated.cc:
Update to 'newer' table handler interface
Only read columns that are needed from remote server.
In case of eq ranges, don't generate two conditions in the WHERE clause
(this can still be optimized, but would require a bigger code change)
Use 'simpler to use' XXXX_LEN' macros
A bit simpler logic in ::write_row() when creating statements.
In update, only include test of fields actually read.
(This greatly simplifies the queries sent by the federated engine)
Similar changes done for delete_row()
sql/ha_federated.h:
Update to 'newer' table handler interface
Changed XXX_LEN macros to use sizeof(...)-1, to simplify usage in ha_federated.cc
Added HA_PRIMARY_KEY_REQUIRED_FOR_DELETE to tell MySQL to read all primary key columns in case of DELETE
sql/ha_heap.cc:
Update to 'newer' table handler interface
sql/ha_heap.h:
Update to 'newer' table handler interface
sql/ha_innodb.cc:
Update to 'newer' table handler interface
- Update innobase_create_handler() to new interface
- Removed HA_NOT_EXACT_COUNT (not needed)
- Renamed HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- Prefixed base status variables with 'stats'
- Use table column bitmaps instead of ha_get_bit_in_read_set()
- Added ::reset(), with code from ::extra(HA_EXTRA_RESET)
- Removed HA_EXTRA_RETRIVE_ALL_COLS and HA_EXTRA_RETRIEVE_PRIMARY_KEY as
the table->read_set and table->write_set bitmaps now are accurate
sql/ha_innodb.h:
Update to 'newer' table handler interface
- table_flags are now ulonglong
- Added reset() method
- Removed not needed ha_retrieve_all_cols() and ha_retrieve_all_pk() columns.
- Made build_template() a class function to be able to easier access class variables
sql/ha_myisam.cc:
Update to 'newer' table handler interface
sql/ha_myisam.h:
Update to 'newer' table handler interface
sql/ha_myisammrg.cc:
Update to 'newer' table handler interface
sql/ha_myisammrg.h:
Update to 'newer' table handler interface
sql/ha_ndbcluster.cc:
Update to 'newer' table handler interface
Fixed use_blob_value() to be accurate
In ::complemented_read() we have to check both the read and write bitmap as the old code did mark all changed columns also in the read map
Correct dumping of field data with DBUG_DUMP
Prefix addresses in DBUG_PRINT with 0x
Fixed usage of not initialized memory
Update to use field->flags & FIELD_IN_ADD_INDEX instead of field->add_index.
sql/ha_ndbcluster.h:
Update to 'newer' table handler interface
sql/ha_ndbcluster_binlog.cc:
Mark usage of all columns in ndbcluster binlog tables
false -> FALSE, true -> TRUE
Use table->s->all_set instead of creating a temporary bitmap.
sql/ha_partition.cc:
Update to 'newer' table handler interface
Added memroot to initialise_partitions() and related functions to get faster memory allocation.
partition_create_handler() is now responsible for initialisation of the partition object
Some trivial optimizations and indentation fixes
Ensure that table_flags() are up to date
Removed documentation for removed HA_EXTRA flags
Fixed 'strange' usage of m_file[i] in new_handlers_from_part_info()that worked in current code 'by chance'
sql/ha_partition.h:
Update to 'newer' table handler interface
sql/handler.cc:
create_xxx handler now takes MEMROOT as an argument to simplify memory allocation.
Much simpler get_new_handler()
(Initialization of the object is now handled by the create method for the engine)
Moved all allocation of bitmap handling to the TABLE object (in table.cc)
Added column_bitmaps_signal() to signal column usage changes.
Changed binlog_log_row() to use the exiusting all_set bitmap in the table object.
Added ha_reset() function to test that the file object is ok at end of statement and call handler::reset()
Added use_hidden_primary_key() to signal handler that we we are going to read and update + delete the row and the handler should thus remember the position for the row
sql/handler.h:
Added HA_NO_TRANSACTIONS, HA_PARTIAL_COLUMN_READ, HA_REQUIRES_KEY_COLUMNS_FOR_DELETE,HA_PRIMARY_KEY_REQUIRED_FOR_DELETE and HA_HAS_RECORDS
Removed HA_NOT_EXACT_COUNT, HA_READ_RND_SAME
HA_DUPP_POS -> HA_DUPLICATE_POS
HA_NOT_EXACT_COUNT replaced by HA_STATS_RECORDS_IS_EXACT, HA_HAS_RECORDS and records()
HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION
Added future row type 'ROW_TYPE_PAGES'
Added MEM_ROOT to handlerton 'create' function
Added ha_statistics, a structure for all status variable in the base handler class.
Moved all status variables in the handler class into a stats structs to improve readability.
ha_table_flags() is now a cached (not virtual) version of table_flags()
reset() doesn't anymore call extra(HA_EXTRA_RESET) but is a function of it's own.
Renamed dupp_ref to dup_ref
Renamed not used handler::sortkey
Moved read_set and write_set to TABLE structure
handler::init() function added for cacheing of virtual constants from engine.
sql/item.cc:
Added register_field_in_read_map() for marking used columns in expression.
This is used by filesort() for creating an optimal column bitmap while retrieving columns for sorting.
Initalize value.cs_info.character_set_client to fix core dump bug with --debug
set_query_id -> mark_used_columns
Mark used columns in read_set OR write_set.
sql/item.h:
Removed reset_query_id_processor() as it's not needed anymore.
Added register_field_in_read_map()
Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
sql/item_cmpfunc.cc:
Temporary mark used columns to be read/writable
Update Item::walk to new interface
sql/item_cmpfunc.h:
Added extra argument to Item::walk() to indicate if we should also traverse sub queries.
sql/item_func.cc:
Update Item::walk() to new interface
table_flags() -> ha_table_flags()
sql/item_func.h:
Update Item::walk() to new interface
sql/item_row.cc:
Update Item::walk() to new interface
sql/item_row.h:
Update Item::walk() to new interface
sql/item_strfunc.h:
Update Item::walk() to new interface
sql/item_subselect.cc:
Added Item_subselect::walk()
(It was a bug it was missing before. Not sure what kind of bugs this could have caused)
sql/item_subselect.h:
Update Item::walk() to new interface
sql/item_sum.cc:
Update Item::walk() to new interface
Updates for new handler interace
sql/item_sum.h:
Update Item::walk() to new interface
sql/key.cc:
Updates for new handler interace
sql/log.cc:
Mark all columns used for log tables
Split options flag
Ensured that second argument to trans_register_ha is a bool
sql/log_event.cc:
Fixed comments to be withing 79 characters
Use OPTION_KEEP_LOG instead of OPTION_STATUS_NO_TRANS_UPDATE to remove wrong warnings
Updates for new handler interface
Use 0x%lx instead of %p (portability problem)
sql/mysql_priv.h:
Added OPTION_KEEP_LOG to indicate that we should replicate the binlog even on rollback
Removed not used 'conds' argument to setup_tables
sql/mysqld.cc:
Indentation fixes and removed old comment
sql/opt_range.cc:
Update to new handler and bitmap interface.
Fixed calls to cp_buffer_from_ref() and walk() (new argument).
Create new temporary bitmaps for ror scans.
(Needed because of handler changes and to get more accurate column bitmaps than before)
Remove not needed file->ha_reset() call before file->close().
Some trivial optimization and indentation fixes.
Use Field->part_of_key_not_clustered() to check if field is part of a key, instead of looping over all key parts.
Added flag 'in_ror_merged_scan' to allow ::get_next() to know that we need a special column bitmap to only fetch pointer to record.
This is needed because ror scan uses the same TABLE object but different file objects, which creates problem for the column bitmap handling.
(This is a temporary solution. A better one would be to allocate an own TABLE object for ROR scans)
Optimized bitmap handling in ror scans:
- Start bitmap at position 0, not 1
- Use same bitmap size as in TABLE
- Use table->read_set and table->write_set to create column bitmaps instead of looping over all fields in table
sql/opt_range.h:
Added 'in_ror_merged_scan' to indicate if we are doing a ROR scan
Added temporary column bitmaps used in ROR scans
sql/opt_sum.cc:
Added get_ext_record_count() which is used in COUNT() optimization if handler has HA_HAS_RECORDS
Note that we don't call this if handler has HA_STATS_RECORDS_IS_EXACT set.
sql/protocol.cc:
We need to mark columns as readable in ::store() as we sometimes return default value for fields to the user
sql/records.cc:
Updates for new handler interface
sql/set_var.cc:
Handle splitting OPTION_STATUS_NO_TRANS_UPDATE to two flags
sql/share/errmsg.txt:
Fixed wrong
sql/sp.cc:
Mark that we are using all columns for the proc table
Update call to setup_tables() to use new prototype
sql/sp_head.cc:
Removed QQ comment
sql/spatial.cc:
Removed wrong QQ comment
sql/sql_acl.cc:
Mark that we need all columns for acl tables
Supply memroot to some 'new' calls.
Indentation fixes
sql/sql_base.cc:
set_query_id removed
Ensure we call ha_reset() at end of each statement
Mark read columns in read_set and changed columns in write_set (Before all columns was marked in read set)
Fixed marking of some columns that was not proplerly marked before
Maintain in TABLE->merge_keys set of all keys that are used in some way
Removed not used 'conds' argument from setup_tables()
Remove not used setting of 'dupp_field' in insert_fields()
Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after crash)
sql/sql_bitmap.h:
Added is_overlapping()
sql/sql_class.cc:
Slow_logs was not properly initialized, which could maybe cause extra/lost entries in slow log.
set_query_id -> mark_used_columns
Simpler variable usage in pack_row() (cleanup)
Moved some variable declartion at start of function for better code readability
sql/sql_class.h:
Added enum_mark_columns
Updated comments
Renamed dupp_field -> dup_field
Added virtual function 'can_rollback_data()' to select_insert() to be used in CREATE ... SELECT to optimize use of OPTION_STATUS_NO_TRANS_UPDATE.
(This fixes a bug in CREATE ... SELECT where we did give wrong warnings when using non transacational tables)
sql/sql_delete.cc:
Updates to new handler interface
Call table->mark_columns_needed_for_delete() to allow us to put additional columns in column usage maps if handler so requires.
Call table->prepare_for_position() to tell handler that we are going to call ha_position().
Removed call to free_io_cache(). (io_cache is now removed in ha_reset()).
Fixed calls to setup_tables()
sql/sql_do.cc:
Update call to setup_fields()
sql/sql_handler.cc:
Tell handler tables to always read all columns.
Use temporary column map when storing value in field for later index usage
sql/sql_help.cc:
Makr all used fields to be read
Update call to setup_fields()
sql/sql_insert.cc:
Tell handler we are going to update the auto_increment column
dupp_field -> dup_field
Set column usage bits for timestamp field.
Call table->mark_columns_needed_for_insert() and table->mark_auto_increment_column()
Removed not used argument from mysql_prepare_insert_check_table().
If we get an duplicate row on insert, change column map to read and write all columns while retrying the operatation.
This is required by the definition of REPLACE and also ensures that fields that are only part of UPDATE are properly handled.
This fixed a bug in NDB and REPLACE where REPLACE wrongly copied some column values from the replaced row.
Setup new bitmaps for delayed insert rows
Remove reseting of next_number_fields as it will be reset on next call to handler_insert()
Fixed usage of thd->options and OPTION_STATUS_NO_TRANS_UPDATE.
The issue was that one should not to reset this flag as it may be set by a previous statement.
The way it was now used caused us to loose some warnings and get other wrong warnings when using non transactional tables mixed with transactional.
I fixed it by introducing 'select_insert::can_rollback_data' to inform send_error() that the given statement can be rolled back (which in case of CREATE TABLE can always be done)
Don't close tables created with CREATE ... SELECT but keep them in the table cache.
Moved out MY_HOOKS from inside function (better readability)
sql/sql_load.cc:
Update to use new handler and column marking interface
Update using setup_tables()
sql/sql_olap.cc:
Update calls to setup_tables
Use enums instead of constants to setup_fields()
sql/sql_parse.cc:
Handle OPTION_KEEP_LOG:
- Set it on CREATE TEMPORARY TABLE / DROP TABLE
- Reset it when OPTION_STATUS_NO_TRANS_UPDATE is reset
- Don't set it for CREATE ... SELECT (this is handled in select_create class)
Remove reseting of OPTION_STATUS_NO_TRANS_UPDATE in begin_trans() as this should already be reset.
If in autocommit mode, reset OPTION_KEEP_LOG and OPTION_STATUS_NO_TRANS_UPDATE to not give warnings in future commands
sql/sql_partition.cc:
Update walk() usage
Trivial indentation fixes
sql/sql_plugin.cc:
Mark all columns as used for plugins
sql/sql_prepare.cc:
Added assert to find out hidden bugs in character_set_client (got an error in debug binary when this not set correctly)
Updates for new handler interface
Update calls to setup_fields()
sql/sql_repl.cc:
Indentation fixes
sql/sql_select.cc:
Update call to setup_tables() and setup_fields()
Remove some old disabled code
Update to new hadler interface
Indentation cleanups
Added column bitmaps for temporary tables.
Remove updating of the removed slots in the Field class
Added TABLE argument to cp_buffer_from_ref() (To be able to install temporary column maps)
For internal temporary tables, use handler::write_row(), handler::delete_row() and handler::update_row() instead of handler::ha_xxxx() for faster execution.
sql/sql_select.h:
Indentaition fixes.
Install temporary column usage maps when needed
Added TABLE element to cp_buffer_from_ref()
sql/sql_show.cc:
Update to new handler interface
Mark all columns used for internal tables.
Style fixes.
Added support for 'future' ROW_TYPE_PAGES.
Don't allocate TMP_TABLE_PARAM with calloc. The 'init()' function will initialize the structure properly.
sql/sql_table.cc:
Update to new handler interface
Simple my_snprintf -> strmake()
Changed some constants to defines
Don't test for NULL in primary key (as we a couple of line above force the PRIMARY KEY to be NOT NULL)
Change field->add_index to use field->flags & FIELD_IN_ADD_INDEX
Mark all columns as used for ALTER TABLE
Style fixes
Update call to filesort()
sql/sql_trigger.h:
Added friend functions to be able to test if triggers exists for table we are going to insert/update or delete in.
sql/sql_udf.cc:
Mark all columns as used for udf system table.
sql/sql_union.cc:
Update call to walk()
Update to new handler interface
sql/sql_update.cc:
Remove query_id argument from compare_record()
Use column bitmaps instead of query_id.
We can't use compare_records() to skip updates for handlers that returns a partial column set and the read_set doesn't cover all columns in the write set, because compare_record() can't in this case know if a not read column changed value.
Update call to setup_fields()
Using separate column read and write sets allows for easier checking of timestamp field was set by statement.
Removed call to free_io_cache() as this is now done in ha_reset()
Call table->mark_columns_needed_for_update() and table->prepare_for_position()
Style fixes
sql/sql_view.cc:
Style fixes
sql/table.cc:
Remove implicitely include 'errno.h'
Remove code for building normalized path, as this is now identical to 'path'
Remove field->fieldnr
Added update of field->part_of_key_not_clustered()
Create column bitmaps in TABLE and TABLE_SHARE
Don't setup a temporary MEM_ROOT object as a thread specific variable for the handler. Instead we send the to-be-used MEMROOT to get_new_handler()
Update to new handler interface
Update call to walk()
Added new functions:
- st_table::clear_column_bitmaps()
- st_table::prepare_for_position()
- st_table::mark_columns_used_by_index()
- st_table::restore_column_maps_after_mark_index()
- st_table::mark_columns_used_by_index_no_reset()
- st_table::mark_auto_increment_column()
- st_table::mark_columns_needed_for_delete()
- st_table::mark_columns_needed_for_update()
- st_table::mark_columns_needed_for_insert()
sql/table.h:
Moved column usage bitmaps from handler to TABLE
Added to TABLE_SHARE all_set and column_bitmap_size
Added to TABLE merge_keys, bitmap_init_values, def_read_set, def_write_set, tmp_set, read_set and write_set.
Declared all new table column bitmap functions
Added TABLE functions column_bitmaps_set(), column_bitmaps_set_no_signal(), use_all_columns() and default_column_bitmaps()
Added functions: tmp_use_all_columns() and tmp_restore_column_map() to temporarly switch column bitmaps
Added functions: dbug_tmp_use_all_columns() and dbug_tmp_restore_column_map() to temporarly switch column bitmaps to avoid asserts in Field::store() and Field::val().
sql/tztime.cc:
Mark all columns as used for timezone tables
storage/archive/ha_archive.cc:
Update to new handler interface
storage/archive/ha_archive.h:
Update to new handler interface
storage/blackhole/ha_blackhole.cc:
Update to new handler interface
storage/blackhole/ha_blackhole.h:
Update to new handler interface
removed not needed flag HA_DUPP_POS
storage/csv/ha_tina.cc:
Update to new handler interface
storage/csv/ha_tina.h:
Update to new handler interface
storage/example/ha_example.cc:
Update to new handler interface
storage/example/ha_example.h:
Update to new handler interface
storage/heap/hp_extra.c:
Added heap_reset() (Required by new handler interface)
storage/heap/hp_test2.c:
Use heap_reset()
storage/myisam/ft_boolean_search.c:
Fixed compiler warning
storage/myisam/mi_extra.c:
Added mi_reset() (Required by new handler interface)
storage/myisam/mi_search.c:
Fixed DBUG_PRINT messages to use 0x%lx instead of %lx
storage/myisam/mi_test2.c:
Use mi_reset()
storage/myisam/myisampack.c:
Use mi_reset()
storage/myisammrg/myrg_extra.c:
Added myrg_reset() (Required by new handler interface)
unittest/mysys/base64.t.c:
Include my_global.h
Don't include implictely include file 'stdlib.h'
2006-06-04 17:52:22 +02:00
|
|
|
|
2016-06-26 22:42:48 +02:00
|
|
|
bool Item_subselect::enumerate_field_refs_processor(void *arg)
|
2009-06-22 13:46:31 +02:00
|
|
|
{
|
2010-02-12 00:59:58 +01:00
|
|
|
List_iterator<Ref_to_outside> it(upper_refs);
|
|
|
|
Ref_to_outside *upper;
|
|
|
|
|
|
|
|
while ((upper= it++))
|
2009-06-22 13:46:31 +02:00
|
|
|
{
|
2017-07-12 12:49:29 +02:00
|
|
|
if (upper->item &&
|
|
|
|
upper->item->walk(&Item::enumerate_field_refs_processor, FALSE, arg))
|
2009-06-22 13:46:31 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2016-06-26 22:42:48 +02:00
|
|
|
bool Item_subselect::mark_as_eliminated_processor(void *arg)
|
2009-06-22 13:46:31 +02:00
|
|
|
{
|
|
|
|
eliminated= TRUE;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2010-02-12 00:59:58 +01:00
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
/**
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2016-06-26 22:42:48 +02:00
|
|
|
bool Item_subselect::eliminate_subselect_processor(void *arg)
|
2010-09-05 17:43:47 +02:00
|
|
|
{
|
|
|
|
unit->item= NULL;
|
|
|
|
unit->exclude_from_tree();
|
|
|
|
eliminated= TRUE;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
/**
|
2010-09-05 17:43:47 +02:00
|
|
|
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.
|
2010-10-05 15:00:31 +02:00
|
|
|
|
|
|
|
@param arg pointer to the fake select
|
|
|
|
|
|
|
|
@return
|
|
|
|
FALSE to force the evaluation of the processor for the subsequent items.
|
2010-09-05 17:43:47 +02:00
|
|
|
*/
|
|
|
|
|
2016-06-26 22:42:48 +02:00
|
|
|
bool Item_subselect::set_fake_select_as_master_processor(void *arg)
|
2010-09-05 17:43:47 +02:00
|
|
|
{
|
|
|
|
SELECT_LEX *fake_select= (SELECT_LEX*) arg;
|
|
|
|
/*
|
2010-10-05 15:00:31 +02:00
|
|
|
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
|
2011-02-10 15:16:06 +01:00
|
|
|
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.
|
2010-09-05 17:43:47 +02:00
|
|
|
*/
|
|
|
|
if (unit->outer_select()->master_unit()->fake_select_lex == fake_select)
|
|
|
|
{
|
|
|
|
/*
|
2011-02-10 15:16:06 +01:00
|
|
|
Set the master of the subquery to be the fake select (i.e. the whole
|
|
|
|
UNION), instead of the last query in the UNION.
|
2010-09-05 17:43:47 +02:00
|
|
|
*/
|
2011-02-10 15:16:06 +01:00
|
|
|
fake_select->add_slave(unit);
|
|
|
|
DBUG_ASSERT(unit->outer_select() == fake_select);
|
2010-09-05 17:43:47 +02:00
|
|
|
/* 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);
|
|
|
|
/*
|
2010-10-05 15:00:31 +02:00
|
|
|
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.
|
2010-09-05 17:43:47 +02:00
|
|
|
*/
|
|
|
|
unit->item= this;
|
|
|
|
eliminated= FALSE;
|
|
|
|
}
|
2010-10-05 15:00:31 +02:00
|
|
|
return FALSE;
|
2010-09-05 17:43:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-02-12 00:59:58 +01:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2010-02-21 04:36:18 +01:00
|
|
|
|
2010-02-12 00:59:58 +01:00
|
|
|
/*
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2017-11-08 15:47:49 +01:00
|
|
|
void Item_subselect::fix_after_pullout(st_select_lex *new_parent,
|
|
|
|
Item **ref, bool merge)
|
2010-02-12 00:59:58 +01:00
|
|
|
{
|
|
|
|
recalc_used_tables(new_parent, TRUE);
|
|
|
|
parent_select= new_parent;
|
|
|
|
}
|
|
|
|
|
2010-02-21 04:36:18 +01:00
|
|
|
|
2010-02-12 00:59:58 +01:00
|
|
|
class Field_fixer: public Field_enumerator
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
table_map used_tables; /* Collect used_tables here */
|
|
|
|
st_select_lex *new_parent; /* Select we're in */
|
2010-02-21 07:32:23 +01:00
|
|
|
virtual void visit_field(Item_field *item)
|
2010-02-12 00:59:58 +01:00
|
|
|
{
|
|
|
|
//for (TABLE_LIST *tbl= new_parent->leaf_tables; tbl; tbl= tbl->next_local)
|
|
|
|
//{
|
|
|
|
// if (tbl->table == field->table)
|
|
|
|
// {
|
2010-02-21 07:32:23 +01:00
|
|
|
used_tables|= item->field->table->map;
|
2010-02-12 00:59:58 +01:00
|
|
|
// 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)
|
|
|
|
{
|
2013-02-26 00:20:17 +01:00
|
|
|
List_iterator_fast<Ref_to_outside> it(upper_refs);
|
2010-02-12 00:59:58 +01:00
|
|
|
Ref_to_outside *upper;
|
2015-06-25 22:34:54 +02:00
|
|
|
DBUG_ENTER("recalc_used_tables");
|
2010-02-12 00:59:58 +01:00
|
|
|
|
|
|
|
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;
|
2016-06-26 22:42:48 +02:00
|
|
|
upper->item->walk(&Item::enumerate_field_refs_processor, 0, &fixer);
|
2010-02-12 00:59:58 +01:00
|
|
|
used_tables_cache |= fixer.used_tables;
|
2011-10-22 09:14:27 +02:00
|
|
|
upper->item->walk(&Item::update_table_bitmaps_processor, FALSE, NULL);
|
2011-06-30 18:49:11 +02:00
|
|
|
/*
|
2010-02-12 00:59:58 +01:00
|
|
|
if (after_pullout)
|
|
|
|
upper->item->fix_after_pullout(new_parent, &(upper->item));
|
|
|
|
upper->item->update_used_tables();
|
2011-06-30 18:49:11 +02:00
|
|
|
*/
|
2010-02-12 00:59:58 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
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.
|
|
|
|
*/
|
2015-06-25 22:34:54 +02:00
|
|
|
DBUG_PRINT("exit", ("used_tables_cache: %llx", used_tables_cache));
|
|
|
|
DBUG_VOID_RETURN;
|
2010-02-12 00:59:58 +01:00
|
|
|
}
|
|
|
|
|
2012-05-17 12:46:05 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
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;
|
2016-06-22 11:17:44 +02:00
|
|
|
bool all_are_simple= true;
|
|
|
|
|
|
|
|
/* check extremely simple select */
|
|
|
|
if (!unit->first_select()->next_select()) // no union
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
such single selects works even without optimization because
|
|
|
|
can not makes loops
|
|
|
|
*/
|
|
|
|
SELECT_LEX *sl= unit->first_select();
|
|
|
|
JOIN *join = sl->join;
|
|
|
|
if (join && !join->tables_list && !sl->first_inner_unit())
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-05-17 12:46:05 +02:00
|
|
|
|
|
|
|
for (SELECT_LEX *sl= unit->first_select(); sl; sl= sl->next_select())
|
|
|
|
{
|
|
|
|
JOIN *cur_join= sl->join;
|
2016-02-21 22:00:58 +01:00
|
|
|
|
|
|
|
/* not optimized subquery */
|
2012-05-17 12:46:05 +02:00
|
|
|
if (!cur_join)
|
2016-02-21 22:00:58 +01:00
|
|
|
return true;
|
|
|
|
|
|
|
|
/*
|
|
|
|
If the subquery is not optimised or in the process of optimization
|
|
|
|
it supposed to be expensive
|
|
|
|
*/
|
2016-06-22 11:17:44 +02:00
|
|
|
if (cur_join->optimization_state != JOIN::OPTIMIZATION_DONE)
|
2016-02-21 22:00:58 +01:00
|
|
|
return true;
|
2012-05-17 12:46:05 +02:00
|
|
|
|
2016-06-22 11:17:44 +02:00
|
|
|
if (!cur_join->tables_list && !sl->first_inner_unit())
|
|
|
|
continue;
|
|
|
|
|
2013-03-29 16:53:21 +01:00
|
|
|
/*
|
|
|
|
Subqueries whose result is known after optimization are not expensive.
|
|
|
|
Such subqueries have all tables optimized away, thus have no join plan.
|
|
|
|
*/
|
2016-02-21 22:00:58 +01:00
|
|
|
if ((cur_join->zero_result_cause || !cur_join->tables_list))
|
2016-06-22 11:17:44 +02:00
|
|
|
continue;
|
|
|
|
|
|
|
|
/*
|
|
|
|
This is not simple SELECT in union so we can not go by simple condition
|
|
|
|
*/
|
|
|
|
all_are_simple= false;
|
2013-03-29 16:53:21 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
If a subquery is not optimized we cannot estimate its cost. A subquery is
|
|
|
|
considered optimized if it has a join plan.
|
|
|
|
*/
|
2016-02-21 22:00:58 +01:00
|
|
|
if (!cur_join->join_tab)
|
2012-05-17 12:46:05 +02:00
|
|
|
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();
|
|
|
|
}
|
|
|
|
|
2016-09-21 18:36:34 +02:00
|
|
|
// here we are sure that subquery is optimized so thd is set
|
2016-06-22 11:17:44 +02:00
|
|
|
return !all_are_simple &&
|
|
|
|
(examined_rows > thd->variables.expensive_subquery_limit);
|
2012-05-17 12:46:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
BitKeeper/etc/ignore:
added mysys/test_bitmap
include/base64.h:
Removed my_global.h, as this must be included first in any program
include/heap.h:
Added heap_reset() (Required by new handler interface)
include/my_base.h:
Removed HA_EXTRA_RESET. MySQL will now call ::reset() instead of ::extra(HA_EXTRA_RESET).
HA_EXTRA_RETRIVE_ALL_COLS and HA_EXTRA_RETRIVE_PRIMARY key are deleted as the column bitmaps makes these unnecessary
include/my_bitmap.h:
Remove my_pthread.h (should be included at upper level)
Introduced my_bitmap_map typedef to make it the bitmap handling more like a black box
Added bitmap_is_overlapping(), bitmap_test_and_clear(), bitmap_copy() and bitmap_cmp()
Made bitmap_set_bit(), bitmap_flip_bit(), bitmap_clear_bit() return void
include/myisam.h:
Added mi_reset() (Required by new handler interface)
include/myisammrg.h:
Added myrg_reset() (Required by new handler interface)
include/mysql_com.h:
Added flag FIELD_IN_ADD_INDEX to be able to remove Field->add_index
mysql-test/extra/binlog_tests/mix_innodb_myisam_binlog.test:
Added testing of CREATE ... SELECT in a mixed environment
(This found some bugs that Mats is going to fix shortly)
mysql-test/install_test_db.sh:
Simplify ldata usage
Added --tmpdir=. option to mysqld bootstrap (Removed some warnings when TMPDIR was wrongly set)
mysql-test/mysql-test-run.pl:
Added --tmpdir=. to bootstrap
mysql-test/mysql-test-run.sh:
Use copy instead of INSTALL_DB for master and slave databases.
(Speeds up startup time a lot!)
Remove snapshot directories at startup (removes some strange warnings)
mysql-test/r/binlog_row_mix_innodb_myisam.result:
Added testing of CREATE ... SELECT in a mixed environment
(This found some bugs that Mats is going to fix shortly)
mysql-test/r/binlog_stm_mix_innodb_myisam.result:
Added testing of CREATE ... SELECT in a mixed environment
mysql-test/r/create.result:
Some extra tests of warnings and number of tables opened by CREATE ... SELECT
mysql-test/r/federated.result:
Drop some left over tables
Added testing of multiple table update and multiple table delete (with and without keys)
mysql-test/r/func_gconcat.result:
Enable some disabled tests (converted them slightly to be predictable)
mysql-test/r/func_time.result:
Added drop of test function
mysql-test/r/innodb_mysql.result:
Added tests for CREATE ... SELECT
mysql-test/r/insert.result:
More tests
Added testing of duplicate columns in insert
mysql-test/r/loaddata.result:
Added testing LOAD DATA ... SET ...
mysql-test/r/multi_update.result:
Test multi updates and deletes using primary key and without
mysql-test/r/ndb_index_unique.result:
Better error message
mysql-test/r/ndb_replace.result:
New correct result after fixing REPLACE handling with NDB
mysql-test/r/rpl_ddl.result:
Now we don't get these (wrong) warnings anymore
mysql-test/r/view_grant.result:
Drop used views
mysql-test/t/create.test:
Some extra tests of warnings and number of tables opened by CREATE ... SELECT
mysql-test/t/federated.test:
Drop some left over tables
Added testing of multiple table update and multiple table delete (with and without keys)
mysql-test/t/func_gconcat.test:
Enable some disabled tests (converted them slightly to be predictable)
mysql-test/t/func_time.test:
Added drop of test function
mysql-test/t/innodb_mysql.test:
Added tests for CREATE ... SELECT
mysql-test/t/insert.test:
More tests
Added testing of duplicate columns in insert
mysql-test/t/loaddata.test:
Added testing LOAD DATA ... SET ...
mysql-test/t/multi_update.test:
Test multi updates and deletes using primary key and without
mysql-test/t/view_grant.test:
Drop used views
mysql-test/valgrind.supp:
Added supression of not needed warnings when printing stack trace
mysys/base64.c:
Include my_global.h first
mysys/my_bitmap.c:
Added bitmap_is_overlapping(), bitmap_test_and_clear() and bitmap_copy()
Changed logic of bitmap handling to be a bit more efficent (Did this together with Mikael Ronström)
Now the 'extra, not used bits' in the bitmap are assumed to have a 'random value' and the bitmap functions are free to change them whenever needed.
Changed how mutex is allocated to make 'bitmap_free()' function simpler.
mysys/thr_lock.c:
Added 0x before thread pointers (for easier comparison of DBUG traces)
sql/event.cc:
Ensure 'use_all_columns()' is used for event tables
Don't print warning that event table is damaged if it doesn't exists.
sql/field.cc:
Added ASSERT_COLUMN_MARKED_FOR_WRITE in all store() methods and ASSERT_COLUMN_MARKED_FOR_READ in all val() methods to catch wrong setting if table->read_set and table->write_set
(Rest of changes are only indentation cleanups)
sql/field.h:
Removed Field->query_id (replaced by table->read_set and table->write_set)
Removed Field->fieldnr (use Field->field_index instead)
Removed Field->add_index (Use Field->flags instead)
Add Field->part_of_key_not_clustered (for usage in opt_range.cc)
sql/filesort.cc:
Added paramater sort_postion to filesort() to force sorting by position instead of storing all fields in the result set.
This allowed me to remove checking of sql_command.
Create a temporary column bitmap for fields that are used by the sorting process.
Use column bitmaps instead of query_id
sql/ha_berkeley.cc:
Update to 'newer' table handler interface
sql/ha_berkeley.h:
Update to 'newer' table handler interface
sql/ha_federated.cc:
Update to 'newer' table handler interface
Only read columns that are needed from remote server.
In case of eq ranges, don't generate two conditions in the WHERE clause
(this can still be optimized, but would require a bigger code change)
Use 'simpler to use' XXXX_LEN' macros
A bit simpler logic in ::write_row() when creating statements.
In update, only include test of fields actually read.
(This greatly simplifies the queries sent by the federated engine)
Similar changes done for delete_row()
sql/ha_federated.h:
Update to 'newer' table handler interface
Changed XXX_LEN macros to use sizeof(...)-1, to simplify usage in ha_federated.cc
Added HA_PRIMARY_KEY_REQUIRED_FOR_DELETE to tell MySQL to read all primary key columns in case of DELETE
sql/ha_heap.cc:
Update to 'newer' table handler interface
sql/ha_heap.h:
Update to 'newer' table handler interface
sql/ha_innodb.cc:
Update to 'newer' table handler interface
- Update innobase_create_handler() to new interface
- Removed HA_NOT_EXACT_COUNT (not needed)
- Renamed HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- Prefixed base status variables with 'stats'
- Use table column bitmaps instead of ha_get_bit_in_read_set()
- Added ::reset(), with code from ::extra(HA_EXTRA_RESET)
- Removed HA_EXTRA_RETRIVE_ALL_COLS and HA_EXTRA_RETRIEVE_PRIMARY_KEY as
the table->read_set and table->write_set bitmaps now are accurate
sql/ha_innodb.h:
Update to 'newer' table handler interface
- table_flags are now ulonglong
- Added reset() method
- Removed not needed ha_retrieve_all_cols() and ha_retrieve_all_pk() columns.
- Made build_template() a class function to be able to easier access class variables
sql/ha_myisam.cc:
Update to 'newer' table handler interface
sql/ha_myisam.h:
Update to 'newer' table handler interface
sql/ha_myisammrg.cc:
Update to 'newer' table handler interface
sql/ha_myisammrg.h:
Update to 'newer' table handler interface
sql/ha_ndbcluster.cc:
Update to 'newer' table handler interface
Fixed use_blob_value() to be accurate
In ::complemented_read() we have to check both the read and write bitmap as the old code did mark all changed columns also in the read map
Correct dumping of field data with DBUG_DUMP
Prefix addresses in DBUG_PRINT with 0x
Fixed usage of not initialized memory
Update to use field->flags & FIELD_IN_ADD_INDEX instead of field->add_index.
sql/ha_ndbcluster.h:
Update to 'newer' table handler interface
sql/ha_ndbcluster_binlog.cc:
Mark usage of all columns in ndbcluster binlog tables
false -> FALSE, true -> TRUE
Use table->s->all_set instead of creating a temporary bitmap.
sql/ha_partition.cc:
Update to 'newer' table handler interface
Added memroot to initialise_partitions() and related functions to get faster memory allocation.
partition_create_handler() is now responsible for initialisation of the partition object
Some trivial optimizations and indentation fixes
Ensure that table_flags() are up to date
Removed documentation for removed HA_EXTRA flags
Fixed 'strange' usage of m_file[i] in new_handlers_from_part_info()that worked in current code 'by chance'
sql/ha_partition.h:
Update to 'newer' table handler interface
sql/handler.cc:
create_xxx handler now takes MEMROOT as an argument to simplify memory allocation.
Much simpler get_new_handler()
(Initialization of the object is now handled by the create method for the engine)
Moved all allocation of bitmap handling to the TABLE object (in table.cc)
Added column_bitmaps_signal() to signal column usage changes.
Changed binlog_log_row() to use the exiusting all_set bitmap in the table object.
Added ha_reset() function to test that the file object is ok at end of statement and call handler::reset()
Added use_hidden_primary_key() to signal handler that we we are going to read and update + delete the row and the handler should thus remember the position for the row
sql/handler.h:
Added HA_NO_TRANSACTIONS, HA_PARTIAL_COLUMN_READ, HA_REQUIRES_KEY_COLUMNS_FOR_DELETE,HA_PRIMARY_KEY_REQUIRED_FOR_DELETE and HA_HAS_RECORDS
Removed HA_NOT_EXACT_COUNT, HA_READ_RND_SAME
HA_DUPP_POS -> HA_DUPLICATE_POS
HA_NOT_EXACT_COUNT replaced by HA_STATS_RECORDS_IS_EXACT, HA_HAS_RECORDS and records()
HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION
Added future row type 'ROW_TYPE_PAGES'
Added MEM_ROOT to handlerton 'create' function
Added ha_statistics, a structure for all status variable in the base handler class.
Moved all status variables in the handler class into a stats structs to improve readability.
ha_table_flags() is now a cached (not virtual) version of table_flags()
reset() doesn't anymore call extra(HA_EXTRA_RESET) but is a function of it's own.
Renamed dupp_ref to dup_ref
Renamed not used handler::sortkey
Moved read_set and write_set to TABLE structure
handler::init() function added for cacheing of virtual constants from engine.
sql/item.cc:
Added register_field_in_read_map() for marking used columns in expression.
This is used by filesort() for creating an optimal column bitmap while retrieving columns for sorting.
Initalize value.cs_info.character_set_client to fix core dump bug with --debug
set_query_id -> mark_used_columns
Mark used columns in read_set OR write_set.
sql/item.h:
Removed reset_query_id_processor() as it's not needed anymore.
Added register_field_in_read_map()
Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
sql/item_cmpfunc.cc:
Temporary mark used columns to be read/writable
Update Item::walk to new interface
sql/item_cmpfunc.h:
Added extra argument to Item::walk() to indicate if we should also traverse sub queries.
sql/item_func.cc:
Update Item::walk() to new interface
table_flags() -> ha_table_flags()
sql/item_func.h:
Update Item::walk() to new interface
sql/item_row.cc:
Update Item::walk() to new interface
sql/item_row.h:
Update Item::walk() to new interface
sql/item_strfunc.h:
Update Item::walk() to new interface
sql/item_subselect.cc:
Added Item_subselect::walk()
(It was a bug it was missing before. Not sure what kind of bugs this could have caused)
sql/item_subselect.h:
Update Item::walk() to new interface
sql/item_sum.cc:
Update Item::walk() to new interface
Updates for new handler interace
sql/item_sum.h:
Update Item::walk() to new interface
sql/key.cc:
Updates for new handler interace
sql/log.cc:
Mark all columns used for log tables
Split options flag
Ensured that second argument to trans_register_ha is a bool
sql/log_event.cc:
Fixed comments to be withing 79 characters
Use OPTION_KEEP_LOG instead of OPTION_STATUS_NO_TRANS_UPDATE to remove wrong warnings
Updates for new handler interface
Use 0x%lx instead of %p (portability problem)
sql/mysql_priv.h:
Added OPTION_KEEP_LOG to indicate that we should replicate the binlog even on rollback
Removed not used 'conds' argument to setup_tables
sql/mysqld.cc:
Indentation fixes and removed old comment
sql/opt_range.cc:
Update to new handler and bitmap interface.
Fixed calls to cp_buffer_from_ref() and walk() (new argument).
Create new temporary bitmaps for ror scans.
(Needed because of handler changes and to get more accurate column bitmaps than before)
Remove not needed file->ha_reset() call before file->close().
Some trivial optimization and indentation fixes.
Use Field->part_of_key_not_clustered() to check if field is part of a key, instead of looping over all key parts.
Added flag 'in_ror_merged_scan' to allow ::get_next() to know that we need a special column bitmap to only fetch pointer to record.
This is needed because ror scan uses the same TABLE object but different file objects, which creates problem for the column bitmap handling.
(This is a temporary solution. A better one would be to allocate an own TABLE object for ROR scans)
Optimized bitmap handling in ror scans:
- Start bitmap at position 0, not 1
- Use same bitmap size as in TABLE
- Use table->read_set and table->write_set to create column bitmaps instead of looping over all fields in table
sql/opt_range.h:
Added 'in_ror_merged_scan' to indicate if we are doing a ROR scan
Added temporary column bitmaps used in ROR scans
sql/opt_sum.cc:
Added get_ext_record_count() which is used in COUNT() optimization if handler has HA_HAS_RECORDS
Note that we don't call this if handler has HA_STATS_RECORDS_IS_EXACT set.
sql/protocol.cc:
We need to mark columns as readable in ::store() as we sometimes return default value for fields to the user
sql/records.cc:
Updates for new handler interface
sql/set_var.cc:
Handle splitting OPTION_STATUS_NO_TRANS_UPDATE to two flags
sql/share/errmsg.txt:
Fixed wrong
sql/sp.cc:
Mark that we are using all columns for the proc table
Update call to setup_tables() to use new prototype
sql/sp_head.cc:
Removed QQ comment
sql/spatial.cc:
Removed wrong QQ comment
sql/sql_acl.cc:
Mark that we need all columns for acl tables
Supply memroot to some 'new' calls.
Indentation fixes
sql/sql_base.cc:
set_query_id removed
Ensure we call ha_reset() at end of each statement
Mark read columns in read_set and changed columns in write_set (Before all columns was marked in read set)
Fixed marking of some columns that was not proplerly marked before
Maintain in TABLE->merge_keys set of all keys that are used in some way
Removed not used 'conds' argument from setup_tables()
Remove not used setting of 'dupp_field' in insert_fields()
Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after crash)
sql/sql_bitmap.h:
Added is_overlapping()
sql/sql_class.cc:
Slow_logs was not properly initialized, which could maybe cause extra/lost entries in slow log.
set_query_id -> mark_used_columns
Simpler variable usage in pack_row() (cleanup)
Moved some variable declartion at start of function for better code readability
sql/sql_class.h:
Added enum_mark_columns
Updated comments
Renamed dupp_field -> dup_field
Added virtual function 'can_rollback_data()' to select_insert() to be used in CREATE ... SELECT to optimize use of OPTION_STATUS_NO_TRANS_UPDATE.
(This fixes a bug in CREATE ... SELECT where we did give wrong warnings when using non transacational tables)
sql/sql_delete.cc:
Updates to new handler interface
Call table->mark_columns_needed_for_delete() to allow us to put additional columns in column usage maps if handler so requires.
Call table->prepare_for_position() to tell handler that we are going to call ha_position().
Removed call to free_io_cache(). (io_cache is now removed in ha_reset()).
Fixed calls to setup_tables()
sql/sql_do.cc:
Update call to setup_fields()
sql/sql_handler.cc:
Tell handler tables to always read all columns.
Use temporary column map when storing value in field for later index usage
sql/sql_help.cc:
Makr all used fields to be read
Update call to setup_fields()
sql/sql_insert.cc:
Tell handler we are going to update the auto_increment column
dupp_field -> dup_field
Set column usage bits for timestamp field.
Call table->mark_columns_needed_for_insert() and table->mark_auto_increment_column()
Removed not used argument from mysql_prepare_insert_check_table().
If we get an duplicate row on insert, change column map to read and write all columns while retrying the operatation.
This is required by the definition of REPLACE and also ensures that fields that are only part of UPDATE are properly handled.
This fixed a bug in NDB and REPLACE where REPLACE wrongly copied some column values from the replaced row.
Setup new bitmaps for delayed insert rows
Remove reseting of next_number_fields as it will be reset on next call to handler_insert()
Fixed usage of thd->options and OPTION_STATUS_NO_TRANS_UPDATE.
The issue was that one should not to reset this flag as it may be set by a previous statement.
The way it was now used caused us to loose some warnings and get other wrong warnings when using non transactional tables mixed with transactional.
I fixed it by introducing 'select_insert::can_rollback_data' to inform send_error() that the given statement can be rolled back (which in case of CREATE TABLE can always be done)
Don't close tables created with CREATE ... SELECT but keep them in the table cache.
Moved out MY_HOOKS from inside function (better readability)
sql/sql_load.cc:
Update to use new handler and column marking interface
Update using setup_tables()
sql/sql_olap.cc:
Update calls to setup_tables
Use enums instead of constants to setup_fields()
sql/sql_parse.cc:
Handle OPTION_KEEP_LOG:
- Set it on CREATE TEMPORARY TABLE / DROP TABLE
- Reset it when OPTION_STATUS_NO_TRANS_UPDATE is reset
- Don't set it for CREATE ... SELECT (this is handled in select_create class)
Remove reseting of OPTION_STATUS_NO_TRANS_UPDATE in begin_trans() as this should already be reset.
If in autocommit mode, reset OPTION_KEEP_LOG and OPTION_STATUS_NO_TRANS_UPDATE to not give warnings in future commands
sql/sql_partition.cc:
Update walk() usage
Trivial indentation fixes
sql/sql_plugin.cc:
Mark all columns as used for plugins
sql/sql_prepare.cc:
Added assert to find out hidden bugs in character_set_client (got an error in debug binary when this not set correctly)
Updates for new handler interface
Update calls to setup_fields()
sql/sql_repl.cc:
Indentation fixes
sql/sql_select.cc:
Update call to setup_tables() and setup_fields()
Remove some old disabled code
Update to new hadler interface
Indentation cleanups
Added column bitmaps for temporary tables.
Remove updating of the removed slots in the Field class
Added TABLE argument to cp_buffer_from_ref() (To be able to install temporary column maps)
For internal temporary tables, use handler::write_row(), handler::delete_row() and handler::update_row() instead of handler::ha_xxxx() for faster execution.
sql/sql_select.h:
Indentaition fixes.
Install temporary column usage maps when needed
Added TABLE element to cp_buffer_from_ref()
sql/sql_show.cc:
Update to new handler interface
Mark all columns used for internal tables.
Style fixes.
Added support for 'future' ROW_TYPE_PAGES.
Don't allocate TMP_TABLE_PARAM with calloc. The 'init()' function will initialize the structure properly.
sql/sql_table.cc:
Update to new handler interface
Simple my_snprintf -> strmake()
Changed some constants to defines
Don't test for NULL in primary key (as we a couple of line above force the PRIMARY KEY to be NOT NULL)
Change field->add_index to use field->flags & FIELD_IN_ADD_INDEX
Mark all columns as used for ALTER TABLE
Style fixes
Update call to filesort()
sql/sql_trigger.h:
Added friend functions to be able to test if triggers exists for table we are going to insert/update or delete in.
sql/sql_udf.cc:
Mark all columns as used for udf system table.
sql/sql_union.cc:
Update call to walk()
Update to new handler interface
sql/sql_update.cc:
Remove query_id argument from compare_record()
Use column bitmaps instead of query_id.
We can't use compare_records() to skip updates for handlers that returns a partial column set and the read_set doesn't cover all columns in the write set, because compare_record() can't in this case know if a not read column changed value.
Update call to setup_fields()
Using separate column read and write sets allows for easier checking of timestamp field was set by statement.
Removed call to free_io_cache() as this is now done in ha_reset()
Call table->mark_columns_needed_for_update() and table->prepare_for_position()
Style fixes
sql/sql_view.cc:
Style fixes
sql/table.cc:
Remove implicitely include 'errno.h'
Remove code for building normalized path, as this is now identical to 'path'
Remove field->fieldnr
Added update of field->part_of_key_not_clustered()
Create column bitmaps in TABLE and TABLE_SHARE
Don't setup a temporary MEM_ROOT object as a thread specific variable for the handler. Instead we send the to-be-used MEMROOT to get_new_handler()
Update to new handler interface
Update call to walk()
Added new functions:
- st_table::clear_column_bitmaps()
- st_table::prepare_for_position()
- st_table::mark_columns_used_by_index()
- st_table::restore_column_maps_after_mark_index()
- st_table::mark_columns_used_by_index_no_reset()
- st_table::mark_auto_increment_column()
- st_table::mark_columns_needed_for_delete()
- st_table::mark_columns_needed_for_update()
- st_table::mark_columns_needed_for_insert()
sql/table.h:
Moved column usage bitmaps from handler to TABLE
Added to TABLE_SHARE all_set and column_bitmap_size
Added to TABLE merge_keys, bitmap_init_values, def_read_set, def_write_set, tmp_set, read_set and write_set.
Declared all new table column bitmap functions
Added TABLE functions column_bitmaps_set(), column_bitmaps_set_no_signal(), use_all_columns() and default_column_bitmaps()
Added functions: tmp_use_all_columns() and tmp_restore_column_map() to temporarly switch column bitmaps
Added functions: dbug_tmp_use_all_columns() and dbug_tmp_restore_column_map() to temporarly switch column bitmaps to avoid asserts in Field::store() and Field::val().
sql/tztime.cc:
Mark all columns as used for timezone tables
storage/archive/ha_archive.cc:
Update to new handler interface
storage/archive/ha_archive.h:
Update to new handler interface
storage/blackhole/ha_blackhole.cc:
Update to new handler interface
storage/blackhole/ha_blackhole.h:
Update to new handler interface
removed not needed flag HA_DUPP_POS
storage/csv/ha_tina.cc:
Update to new handler interface
storage/csv/ha_tina.h:
Update to new handler interface
storage/example/ha_example.cc:
Update to new handler interface
storage/example/ha_example.h:
Update to new handler interface
storage/heap/hp_extra.c:
Added heap_reset() (Required by new handler interface)
storage/heap/hp_test2.c:
Use heap_reset()
storage/myisam/ft_boolean_search.c:
Fixed compiler warning
storage/myisam/mi_extra.c:
Added mi_reset() (Required by new handler interface)
storage/myisam/mi_search.c:
Fixed DBUG_PRINT messages to use 0x%lx instead of %lx
storage/myisam/mi_test2.c:
Use mi_reset()
storage/myisam/myisampack.c:
Use mi_reset()
storage/myisammrg/myrg_extra.c:
Added myrg_reset() (Required by new handler interface)
unittest/mysys/base64.t.c:
Include my_global.h
Don't include implictely include file 'stdlib.h'
2006-06-04 17:52:22 +02:00
|
|
|
bool Item_subselect::walk(Item_processor processor, bool walk_subquery,
|
2016-06-26 22:42:48 +02:00
|
|
|
void *argument)
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
BitKeeper/etc/ignore:
added mysys/test_bitmap
include/base64.h:
Removed my_global.h, as this must be included first in any program
include/heap.h:
Added heap_reset() (Required by new handler interface)
include/my_base.h:
Removed HA_EXTRA_RESET. MySQL will now call ::reset() instead of ::extra(HA_EXTRA_RESET).
HA_EXTRA_RETRIVE_ALL_COLS and HA_EXTRA_RETRIVE_PRIMARY key are deleted as the column bitmaps makes these unnecessary
include/my_bitmap.h:
Remove my_pthread.h (should be included at upper level)
Introduced my_bitmap_map typedef to make it the bitmap handling more like a black box
Added bitmap_is_overlapping(), bitmap_test_and_clear(), bitmap_copy() and bitmap_cmp()
Made bitmap_set_bit(), bitmap_flip_bit(), bitmap_clear_bit() return void
include/myisam.h:
Added mi_reset() (Required by new handler interface)
include/myisammrg.h:
Added myrg_reset() (Required by new handler interface)
include/mysql_com.h:
Added flag FIELD_IN_ADD_INDEX to be able to remove Field->add_index
mysql-test/extra/binlog_tests/mix_innodb_myisam_binlog.test:
Added testing of CREATE ... SELECT in a mixed environment
(This found some bugs that Mats is going to fix shortly)
mysql-test/install_test_db.sh:
Simplify ldata usage
Added --tmpdir=. option to mysqld bootstrap (Removed some warnings when TMPDIR was wrongly set)
mysql-test/mysql-test-run.pl:
Added --tmpdir=. to bootstrap
mysql-test/mysql-test-run.sh:
Use copy instead of INSTALL_DB for master and slave databases.
(Speeds up startup time a lot!)
Remove snapshot directories at startup (removes some strange warnings)
mysql-test/r/binlog_row_mix_innodb_myisam.result:
Added testing of CREATE ... SELECT in a mixed environment
(This found some bugs that Mats is going to fix shortly)
mysql-test/r/binlog_stm_mix_innodb_myisam.result:
Added testing of CREATE ... SELECT in a mixed environment
mysql-test/r/create.result:
Some extra tests of warnings and number of tables opened by CREATE ... SELECT
mysql-test/r/federated.result:
Drop some left over tables
Added testing of multiple table update and multiple table delete (with and without keys)
mysql-test/r/func_gconcat.result:
Enable some disabled tests (converted them slightly to be predictable)
mysql-test/r/func_time.result:
Added drop of test function
mysql-test/r/innodb_mysql.result:
Added tests for CREATE ... SELECT
mysql-test/r/insert.result:
More tests
Added testing of duplicate columns in insert
mysql-test/r/loaddata.result:
Added testing LOAD DATA ... SET ...
mysql-test/r/multi_update.result:
Test multi updates and deletes using primary key and without
mysql-test/r/ndb_index_unique.result:
Better error message
mysql-test/r/ndb_replace.result:
New correct result after fixing REPLACE handling with NDB
mysql-test/r/rpl_ddl.result:
Now we don't get these (wrong) warnings anymore
mysql-test/r/view_grant.result:
Drop used views
mysql-test/t/create.test:
Some extra tests of warnings and number of tables opened by CREATE ... SELECT
mysql-test/t/federated.test:
Drop some left over tables
Added testing of multiple table update and multiple table delete (with and without keys)
mysql-test/t/func_gconcat.test:
Enable some disabled tests (converted them slightly to be predictable)
mysql-test/t/func_time.test:
Added drop of test function
mysql-test/t/innodb_mysql.test:
Added tests for CREATE ... SELECT
mysql-test/t/insert.test:
More tests
Added testing of duplicate columns in insert
mysql-test/t/loaddata.test:
Added testing LOAD DATA ... SET ...
mysql-test/t/multi_update.test:
Test multi updates and deletes using primary key and without
mysql-test/t/view_grant.test:
Drop used views
mysql-test/valgrind.supp:
Added supression of not needed warnings when printing stack trace
mysys/base64.c:
Include my_global.h first
mysys/my_bitmap.c:
Added bitmap_is_overlapping(), bitmap_test_and_clear() and bitmap_copy()
Changed logic of bitmap handling to be a bit more efficent (Did this together with Mikael Ronström)
Now the 'extra, not used bits' in the bitmap are assumed to have a 'random value' and the bitmap functions are free to change them whenever needed.
Changed how mutex is allocated to make 'bitmap_free()' function simpler.
mysys/thr_lock.c:
Added 0x before thread pointers (for easier comparison of DBUG traces)
sql/event.cc:
Ensure 'use_all_columns()' is used for event tables
Don't print warning that event table is damaged if it doesn't exists.
sql/field.cc:
Added ASSERT_COLUMN_MARKED_FOR_WRITE in all store() methods and ASSERT_COLUMN_MARKED_FOR_READ in all val() methods to catch wrong setting if table->read_set and table->write_set
(Rest of changes are only indentation cleanups)
sql/field.h:
Removed Field->query_id (replaced by table->read_set and table->write_set)
Removed Field->fieldnr (use Field->field_index instead)
Removed Field->add_index (Use Field->flags instead)
Add Field->part_of_key_not_clustered (for usage in opt_range.cc)
sql/filesort.cc:
Added paramater sort_postion to filesort() to force sorting by position instead of storing all fields in the result set.
This allowed me to remove checking of sql_command.
Create a temporary column bitmap for fields that are used by the sorting process.
Use column bitmaps instead of query_id
sql/ha_berkeley.cc:
Update to 'newer' table handler interface
sql/ha_berkeley.h:
Update to 'newer' table handler interface
sql/ha_federated.cc:
Update to 'newer' table handler interface
Only read columns that are needed from remote server.
In case of eq ranges, don't generate two conditions in the WHERE clause
(this can still be optimized, but would require a bigger code change)
Use 'simpler to use' XXXX_LEN' macros
A bit simpler logic in ::write_row() when creating statements.
In update, only include test of fields actually read.
(This greatly simplifies the queries sent by the federated engine)
Similar changes done for delete_row()
sql/ha_federated.h:
Update to 'newer' table handler interface
Changed XXX_LEN macros to use sizeof(...)-1, to simplify usage in ha_federated.cc
Added HA_PRIMARY_KEY_REQUIRED_FOR_DELETE to tell MySQL to read all primary key columns in case of DELETE
sql/ha_heap.cc:
Update to 'newer' table handler interface
sql/ha_heap.h:
Update to 'newer' table handler interface
sql/ha_innodb.cc:
Update to 'newer' table handler interface
- Update innobase_create_handler() to new interface
- Removed HA_NOT_EXACT_COUNT (not needed)
- Renamed HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- Prefixed base status variables with 'stats'
- Use table column bitmaps instead of ha_get_bit_in_read_set()
- Added ::reset(), with code from ::extra(HA_EXTRA_RESET)
- Removed HA_EXTRA_RETRIVE_ALL_COLS and HA_EXTRA_RETRIEVE_PRIMARY_KEY as
the table->read_set and table->write_set bitmaps now are accurate
sql/ha_innodb.h:
Update to 'newer' table handler interface
- table_flags are now ulonglong
- Added reset() method
- Removed not needed ha_retrieve_all_cols() and ha_retrieve_all_pk() columns.
- Made build_template() a class function to be able to easier access class variables
sql/ha_myisam.cc:
Update to 'newer' table handler interface
sql/ha_myisam.h:
Update to 'newer' table handler interface
sql/ha_myisammrg.cc:
Update to 'newer' table handler interface
sql/ha_myisammrg.h:
Update to 'newer' table handler interface
sql/ha_ndbcluster.cc:
Update to 'newer' table handler interface
Fixed use_blob_value() to be accurate
In ::complemented_read() we have to check both the read and write bitmap as the old code did mark all changed columns also in the read map
Correct dumping of field data with DBUG_DUMP
Prefix addresses in DBUG_PRINT with 0x
Fixed usage of not initialized memory
Update to use field->flags & FIELD_IN_ADD_INDEX instead of field->add_index.
sql/ha_ndbcluster.h:
Update to 'newer' table handler interface
sql/ha_ndbcluster_binlog.cc:
Mark usage of all columns in ndbcluster binlog tables
false -> FALSE, true -> TRUE
Use table->s->all_set instead of creating a temporary bitmap.
sql/ha_partition.cc:
Update to 'newer' table handler interface
Added memroot to initialise_partitions() and related functions to get faster memory allocation.
partition_create_handler() is now responsible for initialisation of the partition object
Some trivial optimizations and indentation fixes
Ensure that table_flags() are up to date
Removed documentation for removed HA_EXTRA flags
Fixed 'strange' usage of m_file[i] in new_handlers_from_part_info()that worked in current code 'by chance'
sql/ha_partition.h:
Update to 'newer' table handler interface
sql/handler.cc:
create_xxx handler now takes MEMROOT as an argument to simplify memory allocation.
Much simpler get_new_handler()
(Initialization of the object is now handled by the create method for the engine)
Moved all allocation of bitmap handling to the TABLE object (in table.cc)
Added column_bitmaps_signal() to signal column usage changes.
Changed binlog_log_row() to use the exiusting all_set bitmap in the table object.
Added ha_reset() function to test that the file object is ok at end of statement and call handler::reset()
Added use_hidden_primary_key() to signal handler that we we are going to read and update + delete the row and the handler should thus remember the position for the row
sql/handler.h:
Added HA_NO_TRANSACTIONS, HA_PARTIAL_COLUMN_READ, HA_REQUIRES_KEY_COLUMNS_FOR_DELETE,HA_PRIMARY_KEY_REQUIRED_FOR_DELETE and HA_HAS_RECORDS
Removed HA_NOT_EXACT_COUNT, HA_READ_RND_SAME
HA_DUPP_POS -> HA_DUPLICATE_POS
HA_NOT_EXACT_COUNT replaced by HA_STATS_RECORDS_IS_EXACT, HA_HAS_RECORDS and records()
HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION
Added future row type 'ROW_TYPE_PAGES'
Added MEM_ROOT to handlerton 'create' function
Added ha_statistics, a structure for all status variable in the base handler class.
Moved all status variables in the handler class into a stats structs to improve readability.
ha_table_flags() is now a cached (not virtual) version of table_flags()
reset() doesn't anymore call extra(HA_EXTRA_RESET) but is a function of it's own.
Renamed dupp_ref to dup_ref
Renamed not used handler::sortkey
Moved read_set and write_set to TABLE structure
handler::init() function added for cacheing of virtual constants from engine.
sql/item.cc:
Added register_field_in_read_map() for marking used columns in expression.
This is used by filesort() for creating an optimal column bitmap while retrieving columns for sorting.
Initalize value.cs_info.character_set_client to fix core dump bug with --debug
set_query_id -> mark_used_columns
Mark used columns in read_set OR write_set.
sql/item.h:
Removed reset_query_id_processor() as it's not needed anymore.
Added register_field_in_read_map()
Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
sql/item_cmpfunc.cc:
Temporary mark used columns to be read/writable
Update Item::walk to new interface
sql/item_cmpfunc.h:
Added extra argument to Item::walk() to indicate if we should also traverse sub queries.
sql/item_func.cc:
Update Item::walk() to new interface
table_flags() -> ha_table_flags()
sql/item_func.h:
Update Item::walk() to new interface
sql/item_row.cc:
Update Item::walk() to new interface
sql/item_row.h:
Update Item::walk() to new interface
sql/item_strfunc.h:
Update Item::walk() to new interface
sql/item_subselect.cc:
Added Item_subselect::walk()
(It was a bug it was missing before. Not sure what kind of bugs this could have caused)
sql/item_subselect.h:
Update Item::walk() to new interface
sql/item_sum.cc:
Update Item::walk() to new interface
Updates for new handler interace
sql/item_sum.h:
Update Item::walk() to new interface
sql/key.cc:
Updates for new handler interace
sql/log.cc:
Mark all columns used for log tables
Split options flag
Ensured that second argument to trans_register_ha is a bool
sql/log_event.cc:
Fixed comments to be withing 79 characters
Use OPTION_KEEP_LOG instead of OPTION_STATUS_NO_TRANS_UPDATE to remove wrong warnings
Updates for new handler interface
Use 0x%lx instead of %p (portability problem)
sql/mysql_priv.h:
Added OPTION_KEEP_LOG to indicate that we should replicate the binlog even on rollback
Removed not used 'conds' argument to setup_tables
sql/mysqld.cc:
Indentation fixes and removed old comment
sql/opt_range.cc:
Update to new handler and bitmap interface.
Fixed calls to cp_buffer_from_ref() and walk() (new argument).
Create new temporary bitmaps for ror scans.
(Needed because of handler changes and to get more accurate column bitmaps than before)
Remove not needed file->ha_reset() call before file->close().
Some trivial optimization and indentation fixes.
Use Field->part_of_key_not_clustered() to check if field is part of a key, instead of looping over all key parts.
Added flag 'in_ror_merged_scan' to allow ::get_next() to know that we need a special column bitmap to only fetch pointer to record.
This is needed because ror scan uses the same TABLE object but different file objects, which creates problem for the column bitmap handling.
(This is a temporary solution. A better one would be to allocate an own TABLE object for ROR scans)
Optimized bitmap handling in ror scans:
- Start bitmap at position 0, not 1
- Use same bitmap size as in TABLE
- Use table->read_set and table->write_set to create column bitmaps instead of looping over all fields in table
sql/opt_range.h:
Added 'in_ror_merged_scan' to indicate if we are doing a ROR scan
Added temporary column bitmaps used in ROR scans
sql/opt_sum.cc:
Added get_ext_record_count() which is used in COUNT() optimization if handler has HA_HAS_RECORDS
Note that we don't call this if handler has HA_STATS_RECORDS_IS_EXACT set.
sql/protocol.cc:
We need to mark columns as readable in ::store() as we sometimes return default value for fields to the user
sql/records.cc:
Updates for new handler interface
sql/set_var.cc:
Handle splitting OPTION_STATUS_NO_TRANS_UPDATE to two flags
sql/share/errmsg.txt:
Fixed wrong
sql/sp.cc:
Mark that we are using all columns for the proc table
Update call to setup_tables() to use new prototype
sql/sp_head.cc:
Removed QQ comment
sql/spatial.cc:
Removed wrong QQ comment
sql/sql_acl.cc:
Mark that we need all columns for acl tables
Supply memroot to some 'new' calls.
Indentation fixes
sql/sql_base.cc:
set_query_id removed
Ensure we call ha_reset() at end of each statement
Mark read columns in read_set and changed columns in write_set (Before all columns was marked in read set)
Fixed marking of some columns that was not proplerly marked before
Maintain in TABLE->merge_keys set of all keys that are used in some way
Removed not used 'conds' argument from setup_tables()
Remove not used setting of 'dupp_field' in insert_fields()
Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after crash)
sql/sql_bitmap.h:
Added is_overlapping()
sql/sql_class.cc:
Slow_logs was not properly initialized, which could maybe cause extra/lost entries in slow log.
set_query_id -> mark_used_columns
Simpler variable usage in pack_row() (cleanup)
Moved some variable declartion at start of function for better code readability
sql/sql_class.h:
Added enum_mark_columns
Updated comments
Renamed dupp_field -> dup_field
Added virtual function 'can_rollback_data()' to select_insert() to be used in CREATE ... SELECT to optimize use of OPTION_STATUS_NO_TRANS_UPDATE.
(This fixes a bug in CREATE ... SELECT where we did give wrong warnings when using non transacational tables)
sql/sql_delete.cc:
Updates to new handler interface
Call table->mark_columns_needed_for_delete() to allow us to put additional columns in column usage maps if handler so requires.
Call table->prepare_for_position() to tell handler that we are going to call ha_position().
Removed call to free_io_cache(). (io_cache is now removed in ha_reset()).
Fixed calls to setup_tables()
sql/sql_do.cc:
Update call to setup_fields()
sql/sql_handler.cc:
Tell handler tables to always read all columns.
Use temporary column map when storing value in field for later index usage
sql/sql_help.cc:
Makr all used fields to be read
Update call to setup_fields()
sql/sql_insert.cc:
Tell handler we are going to update the auto_increment column
dupp_field -> dup_field
Set column usage bits for timestamp field.
Call table->mark_columns_needed_for_insert() and table->mark_auto_increment_column()
Removed not used argument from mysql_prepare_insert_check_table().
If we get an duplicate row on insert, change column map to read and write all columns while retrying the operatation.
This is required by the definition of REPLACE and also ensures that fields that are only part of UPDATE are properly handled.
This fixed a bug in NDB and REPLACE where REPLACE wrongly copied some column values from the replaced row.
Setup new bitmaps for delayed insert rows
Remove reseting of next_number_fields as it will be reset on next call to handler_insert()
Fixed usage of thd->options and OPTION_STATUS_NO_TRANS_UPDATE.
The issue was that one should not to reset this flag as it may be set by a previous statement.
The way it was now used caused us to loose some warnings and get other wrong warnings when using non transactional tables mixed with transactional.
I fixed it by introducing 'select_insert::can_rollback_data' to inform send_error() that the given statement can be rolled back (which in case of CREATE TABLE can always be done)
Don't close tables created with CREATE ... SELECT but keep them in the table cache.
Moved out MY_HOOKS from inside function (better readability)
sql/sql_load.cc:
Update to use new handler and column marking interface
Update using setup_tables()
sql/sql_olap.cc:
Update calls to setup_tables
Use enums instead of constants to setup_fields()
sql/sql_parse.cc:
Handle OPTION_KEEP_LOG:
- Set it on CREATE TEMPORARY TABLE / DROP TABLE
- Reset it when OPTION_STATUS_NO_TRANS_UPDATE is reset
- Don't set it for CREATE ... SELECT (this is handled in select_create class)
Remove reseting of OPTION_STATUS_NO_TRANS_UPDATE in begin_trans() as this should already be reset.
If in autocommit mode, reset OPTION_KEEP_LOG and OPTION_STATUS_NO_TRANS_UPDATE to not give warnings in future commands
sql/sql_partition.cc:
Update walk() usage
Trivial indentation fixes
sql/sql_plugin.cc:
Mark all columns as used for plugins
sql/sql_prepare.cc:
Added assert to find out hidden bugs in character_set_client (got an error in debug binary when this not set correctly)
Updates for new handler interface
Update calls to setup_fields()
sql/sql_repl.cc:
Indentation fixes
sql/sql_select.cc:
Update call to setup_tables() and setup_fields()
Remove some old disabled code
Update to new hadler interface
Indentation cleanups
Added column bitmaps for temporary tables.
Remove updating of the removed slots in the Field class
Added TABLE argument to cp_buffer_from_ref() (To be able to install temporary column maps)
For internal temporary tables, use handler::write_row(), handler::delete_row() and handler::update_row() instead of handler::ha_xxxx() for faster execution.
sql/sql_select.h:
Indentaition fixes.
Install temporary column usage maps when needed
Added TABLE element to cp_buffer_from_ref()
sql/sql_show.cc:
Update to new handler interface
Mark all columns used for internal tables.
Style fixes.
Added support for 'future' ROW_TYPE_PAGES.
Don't allocate TMP_TABLE_PARAM with calloc. The 'init()' function will initialize the structure properly.
sql/sql_table.cc:
Update to new handler interface
Simple my_snprintf -> strmake()
Changed some constants to defines
Don't test for NULL in primary key (as we a couple of line above force the PRIMARY KEY to be NOT NULL)
Change field->add_index to use field->flags & FIELD_IN_ADD_INDEX
Mark all columns as used for ALTER TABLE
Style fixes
Update call to filesort()
sql/sql_trigger.h:
Added friend functions to be able to test if triggers exists for table we are going to insert/update or delete in.
sql/sql_udf.cc:
Mark all columns as used for udf system table.
sql/sql_union.cc:
Update call to walk()
Update to new handler interface
sql/sql_update.cc:
Remove query_id argument from compare_record()
Use column bitmaps instead of query_id.
We can't use compare_records() to skip updates for handlers that returns a partial column set and the read_set doesn't cover all columns in the write set, because compare_record() can't in this case know if a not read column changed value.
Update call to setup_fields()
Using separate column read and write sets allows for easier checking of timestamp field was set by statement.
Removed call to free_io_cache() as this is now done in ha_reset()
Call table->mark_columns_needed_for_update() and table->prepare_for_position()
Style fixes
sql/sql_view.cc:
Style fixes
sql/table.cc:
Remove implicitely include 'errno.h'
Remove code for building normalized path, as this is now identical to 'path'
Remove field->fieldnr
Added update of field->part_of_key_not_clustered()
Create column bitmaps in TABLE and TABLE_SHARE
Don't setup a temporary MEM_ROOT object as a thread specific variable for the handler. Instead we send the to-be-used MEMROOT to get_new_handler()
Update to new handler interface
Update call to walk()
Added new functions:
- st_table::clear_column_bitmaps()
- st_table::prepare_for_position()
- st_table::mark_columns_used_by_index()
- st_table::restore_column_maps_after_mark_index()
- st_table::mark_columns_used_by_index_no_reset()
- st_table::mark_auto_increment_column()
- st_table::mark_columns_needed_for_delete()
- st_table::mark_columns_needed_for_update()
- st_table::mark_columns_needed_for_insert()
sql/table.h:
Moved column usage bitmaps from handler to TABLE
Added to TABLE_SHARE all_set and column_bitmap_size
Added to TABLE merge_keys, bitmap_init_values, def_read_set, def_write_set, tmp_set, read_set and write_set.
Declared all new table column bitmap functions
Added TABLE functions column_bitmaps_set(), column_bitmaps_set_no_signal(), use_all_columns() and default_column_bitmaps()
Added functions: tmp_use_all_columns() and tmp_restore_column_map() to temporarly switch column bitmaps
Added functions: dbug_tmp_use_all_columns() and dbug_tmp_restore_column_map() to temporarly switch column bitmaps to avoid asserts in Field::store() and Field::val().
sql/tztime.cc:
Mark all columns as used for timezone tables
storage/archive/ha_archive.cc:
Update to new handler interface
storage/archive/ha_archive.h:
Update to new handler interface
storage/blackhole/ha_blackhole.cc:
Update to new handler interface
storage/blackhole/ha_blackhole.h:
Update to new handler interface
removed not needed flag HA_DUPP_POS
storage/csv/ha_tina.cc:
Update to new handler interface
storage/csv/ha_tina.h:
Update to new handler interface
storage/example/ha_example.cc:
Update to new handler interface
storage/example/ha_example.h:
Update to new handler interface
storage/heap/hp_extra.c:
Added heap_reset() (Required by new handler interface)
storage/heap/hp_test2.c:
Use heap_reset()
storage/myisam/ft_boolean_search.c:
Fixed compiler warning
storage/myisam/mi_extra.c:
Added mi_reset() (Required by new handler interface)
storage/myisam/mi_search.c:
Fixed DBUG_PRINT messages to use 0x%lx instead of %lx
storage/myisam/mi_test2.c:
Use mi_reset()
storage/myisam/myisampack.c:
Use mi_reset()
storage/myisammrg/myrg_extra.c:
Added myrg_reset() (Required by new handler interface)
unittest/mysys/base64.t.c:
Include my_global.h
Don't include implictely include file 'stdlib.h'
2006-06-04 17:52:22 +02:00
|
|
|
{
|
2011-10-05 23:11:08 +02:00
|
|
|
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).
|
|
|
|
*/
|
2011-11-13 11:02:13 +01:00
|
|
|
return (this->*processor)(argument);
|
2011-10-05 23:11:08 +02:00
|
|
|
}
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
BitKeeper/etc/ignore:
added mysys/test_bitmap
include/base64.h:
Removed my_global.h, as this must be included first in any program
include/heap.h:
Added heap_reset() (Required by new handler interface)
include/my_base.h:
Removed HA_EXTRA_RESET. MySQL will now call ::reset() instead of ::extra(HA_EXTRA_RESET).
HA_EXTRA_RETRIVE_ALL_COLS and HA_EXTRA_RETRIVE_PRIMARY key are deleted as the column bitmaps makes these unnecessary
include/my_bitmap.h:
Remove my_pthread.h (should be included at upper level)
Introduced my_bitmap_map typedef to make it the bitmap handling more like a black box
Added bitmap_is_overlapping(), bitmap_test_and_clear(), bitmap_copy() and bitmap_cmp()
Made bitmap_set_bit(), bitmap_flip_bit(), bitmap_clear_bit() return void
include/myisam.h:
Added mi_reset() (Required by new handler interface)
include/myisammrg.h:
Added myrg_reset() (Required by new handler interface)
include/mysql_com.h:
Added flag FIELD_IN_ADD_INDEX to be able to remove Field->add_index
mysql-test/extra/binlog_tests/mix_innodb_myisam_binlog.test:
Added testing of CREATE ... SELECT in a mixed environment
(This found some bugs that Mats is going to fix shortly)
mysql-test/install_test_db.sh:
Simplify ldata usage
Added --tmpdir=. option to mysqld bootstrap (Removed some warnings when TMPDIR was wrongly set)
mysql-test/mysql-test-run.pl:
Added --tmpdir=. to bootstrap
mysql-test/mysql-test-run.sh:
Use copy instead of INSTALL_DB for master and slave databases.
(Speeds up startup time a lot!)
Remove snapshot directories at startup (removes some strange warnings)
mysql-test/r/binlog_row_mix_innodb_myisam.result:
Added testing of CREATE ... SELECT in a mixed environment
(This found some bugs that Mats is going to fix shortly)
mysql-test/r/binlog_stm_mix_innodb_myisam.result:
Added testing of CREATE ... SELECT in a mixed environment
mysql-test/r/create.result:
Some extra tests of warnings and number of tables opened by CREATE ... SELECT
mysql-test/r/federated.result:
Drop some left over tables
Added testing of multiple table update and multiple table delete (with and without keys)
mysql-test/r/func_gconcat.result:
Enable some disabled tests (converted them slightly to be predictable)
mysql-test/r/func_time.result:
Added drop of test function
mysql-test/r/innodb_mysql.result:
Added tests for CREATE ... SELECT
mysql-test/r/insert.result:
More tests
Added testing of duplicate columns in insert
mysql-test/r/loaddata.result:
Added testing LOAD DATA ... SET ...
mysql-test/r/multi_update.result:
Test multi updates and deletes using primary key and without
mysql-test/r/ndb_index_unique.result:
Better error message
mysql-test/r/ndb_replace.result:
New correct result after fixing REPLACE handling with NDB
mysql-test/r/rpl_ddl.result:
Now we don't get these (wrong) warnings anymore
mysql-test/r/view_grant.result:
Drop used views
mysql-test/t/create.test:
Some extra tests of warnings and number of tables opened by CREATE ... SELECT
mysql-test/t/federated.test:
Drop some left over tables
Added testing of multiple table update and multiple table delete (with and without keys)
mysql-test/t/func_gconcat.test:
Enable some disabled tests (converted them slightly to be predictable)
mysql-test/t/func_time.test:
Added drop of test function
mysql-test/t/innodb_mysql.test:
Added tests for CREATE ... SELECT
mysql-test/t/insert.test:
More tests
Added testing of duplicate columns in insert
mysql-test/t/loaddata.test:
Added testing LOAD DATA ... SET ...
mysql-test/t/multi_update.test:
Test multi updates and deletes using primary key and without
mysql-test/t/view_grant.test:
Drop used views
mysql-test/valgrind.supp:
Added supression of not needed warnings when printing stack trace
mysys/base64.c:
Include my_global.h first
mysys/my_bitmap.c:
Added bitmap_is_overlapping(), bitmap_test_and_clear() and bitmap_copy()
Changed logic of bitmap handling to be a bit more efficent (Did this together with Mikael Ronström)
Now the 'extra, not used bits' in the bitmap are assumed to have a 'random value' and the bitmap functions are free to change them whenever needed.
Changed how mutex is allocated to make 'bitmap_free()' function simpler.
mysys/thr_lock.c:
Added 0x before thread pointers (for easier comparison of DBUG traces)
sql/event.cc:
Ensure 'use_all_columns()' is used for event tables
Don't print warning that event table is damaged if it doesn't exists.
sql/field.cc:
Added ASSERT_COLUMN_MARKED_FOR_WRITE in all store() methods and ASSERT_COLUMN_MARKED_FOR_READ in all val() methods to catch wrong setting if table->read_set and table->write_set
(Rest of changes are only indentation cleanups)
sql/field.h:
Removed Field->query_id (replaced by table->read_set and table->write_set)
Removed Field->fieldnr (use Field->field_index instead)
Removed Field->add_index (Use Field->flags instead)
Add Field->part_of_key_not_clustered (for usage in opt_range.cc)
sql/filesort.cc:
Added paramater sort_postion to filesort() to force sorting by position instead of storing all fields in the result set.
This allowed me to remove checking of sql_command.
Create a temporary column bitmap for fields that are used by the sorting process.
Use column bitmaps instead of query_id
sql/ha_berkeley.cc:
Update to 'newer' table handler interface
sql/ha_berkeley.h:
Update to 'newer' table handler interface
sql/ha_federated.cc:
Update to 'newer' table handler interface
Only read columns that are needed from remote server.
In case of eq ranges, don't generate two conditions in the WHERE clause
(this can still be optimized, but would require a bigger code change)
Use 'simpler to use' XXXX_LEN' macros
A bit simpler logic in ::write_row() when creating statements.
In update, only include test of fields actually read.
(This greatly simplifies the queries sent by the federated engine)
Similar changes done for delete_row()
sql/ha_federated.h:
Update to 'newer' table handler interface
Changed XXX_LEN macros to use sizeof(...)-1, to simplify usage in ha_federated.cc
Added HA_PRIMARY_KEY_REQUIRED_FOR_DELETE to tell MySQL to read all primary key columns in case of DELETE
sql/ha_heap.cc:
Update to 'newer' table handler interface
sql/ha_heap.h:
Update to 'newer' table handler interface
sql/ha_innodb.cc:
Update to 'newer' table handler interface
- Update innobase_create_handler() to new interface
- Removed HA_NOT_EXACT_COUNT (not needed)
- Renamed HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- Prefixed base status variables with 'stats'
- Use table column bitmaps instead of ha_get_bit_in_read_set()
- Added ::reset(), with code from ::extra(HA_EXTRA_RESET)
- Removed HA_EXTRA_RETRIVE_ALL_COLS and HA_EXTRA_RETRIEVE_PRIMARY_KEY as
the table->read_set and table->write_set bitmaps now are accurate
sql/ha_innodb.h:
Update to 'newer' table handler interface
- table_flags are now ulonglong
- Added reset() method
- Removed not needed ha_retrieve_all_cols() and ha_retrieve_all_pk() columns.
- Made build_template() a class function to be able to easier access class variables
sql/ha_myisam.cc:
Update to 'newer' table handler interface
sql/ha_myisam.h:
Update to 'newer' table handler interface
sql/ha_myisammrg.cc:
Update to 'newer' table handler interface
sql/ha_myisammrg.h:
Update to 'newer' table handler interface
sql/ha_ndbcluster.cc:
Update to 'newer' table handler interface
Fixed use_blob_value() to be accurate
In ::complemented_read() we have to check both the read and write bitmap as the old code did mark all changed columns also in the read map
Correct dumping of field data with DBUG_DUMP
Prefix addresses in DBUG_PRINT with 0x
Fixed usage of not initialized memory
Update to use field->flags & FIELD_IN_ADD_INDEX instead of field->add_index.
sql/ha_ndbcluster.h:
Update to 'newer' table handler interface
sql/ha_ndbcluster_binlog.cc:
Mark usage of all columns in ndbcluster binlog tables
false -> FALSE, true -> TRUE
Use table->s->all_set instead of creating a temporary bitmap.
sql/ha_partition.cc:
Update to 'newer' table handler interface
Added memroot to initialise_partitions() and related functions to get faster memory allocation.
partition_create_handler() is now responsible for initialisation of the partition object
Some trivial optimizations and indentation fixes
Ensure that table_flags() are up to date
Removed documentation for removed HA_EXTRA flags
Fixed 'strange' usage of m_file[i] in new_handlers_from_part_info()that worked in current code 'by chance'
sql/ha_partition.h:
Update to 'newer' table handler interface
sql/handler.cc:
create_xxx handler now takes MEMROOT as an argument to simplify memory allocation.
Much simpler get_new_handler()
(Initialization of the object is now handled by the create method for the engine)
Moved all allocation of bitmap handling to the TABLE object (in table.cc)
Added column_bitmaps_signal() to signal column usage changes.
Changed binlog_log_row() to use the exiusting all_set bitmap in the table object.
Added ha_reset() function to test that the file object is ok at end of statement and call handler::reset()
Added use_hidden_primary_key() to signal handler that we we are going to read and update + delete the row and the handler should thus remember the position for the row
sql/handler.h:
Added HA_NO_TRANSACTIONS, HA_PARTIAL_COLUMN_READ, HA_REQUIRES_KEY_COLUMNS_FOR_DELETE,HA_PRIMARY_KEY_REQUIRED_FOR_DELETE and HA_HAS_RECORDS
Removed HA_NOT_EXACT_COUNT, HA_READ_RND_SAME
HA_DUPP_POS -> HA_DUPLICATE_POS
HA_NOT_EXACT_COUNT replaced by HA_STATS_RECORDS_IS_EXACT, HA_HAS_RECORDS and records()
HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION
Added future row type 'ROW_TYPE_PAGES'
Added MEM_ROOT to handlerton 'create' function
Added ha_statistics, a structure for all status variable in the base handler class.
Moved all status variables in the handler class into a stats structs to improve readability.
ha_table_flags() is now a cached (not virtual) version of table_flags()
reset() doesn't anymore call extra(HA_EXTRA_RESET) but is a function of it's own.
Renamed dupp_ref to dup_ref
Renamed not used handler::sortkey
Moved read_set and write_set to TABLE structure
handler::init() function added for cacheing of virtual constants from engine.
sql/item.cc:
Added register_field_in_read_map() for marking used columns in expression.
This is used by filesort() for creating an optimal column bitmap while retrieving columns for sorting.
Initalize value.cs_info.character_set_client to fix core dump bug with --debug
set_query_id -> mark_used_columns
Mark used columns in read_set OR write_set.
sql/item.h:
Removed reset_query_id_processor() as it's not needed anymore.
Added register_field_in_read_map()
Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
sql/item_cmpfunc.cc:
Temporary mark used columns to be read/writable
Update Item::walk to new interface
sql/item_cmpfunc.h:
Added extra argument to Item::walk() to indicate if we should also traverse sub queries.
sql/item_func.cc:
Update Item::walk() to new interface
table_flags() -> ha_table_flags()
sql/item_func.h:
Update Item::walk() to new interface
sql/item_row.cc:
Update Item::walk() to new interface
sql/item_row.h:
Update Item::walk() to new interface
sql/item_strfunc.h:
Update Item::walk() to new interface
sql/item_subselect.cc:
Added Item_subselect::walk()
(It was a bug it was missing before. Not sure what kind of bugs this could have caused)
sql/item_subselect.h:
Update Item::walk() to new interface
sql/item_sum.cc:
Update Item::walk() to new interface
Updates for new handler interace
sql/item_sum.h:
Update Item::walk() to new interface
sql/key.cc:
Updates for new handler interace
sql/log.cc:
Mark all columns used for log tables
Split options flag
Ensured that second argument to trans_register_ha is a bool
sql/log_event.cc:
Fixed comments to be withing 79 characters
Use OPTION_KEEP_LOG instead of OPTION_STATUS_NO_TRANS_UPDATE to remove wrong warnings
Updates for new handler interface
Use 0x%lx instead of %p (portability problem)
sql/mysql_priv.h:
Added OPTION_KEEP_LOG to indicate that we should replicate the binlog even on rollback
Removed not used 'conds' argument to setup_tables
sql/mysqld.cc:
Indentation fixes and removed old comment
sql/opt_range.cc:
Update to new handler and bitmap interface.
Fixed calls to cp_buffer_from_ref() and walk() (new argument).
Create new temporary bitmaps for ror scans.
(Needed because of handler changes and to get more accurate column bitmaps than before)
Remove not needed file->ha_reset() call before file->close().
Some trivial optimization and indentation fixes.
Use Field->part_of_key_not_clustered() to check if field is part of a key, instead of looping over all key parts.
Added flag 'in_ror_merged_scan' to allow ::get_next() to know that we need a special column bitmap to only fetch pointer to record.
This is needed because ror scan uses the same TABLE object but different file objects, which creates problem for the column bitmap handling.
(This is a temporary solution. A better one would be to allocate an own TABLE object for ROR scans)
Optimized bitmap handling in ror scans:
- Start bitmap at position 0, not 1
- Use same bitmap size as in TABLE
- Use table->read_set and table->write_set to create column bitmaps instead of looping over all fields in table
sql/opt_range.h:
Added 'in_ror_merged_scan' to indicate if we are doing a ROR scan
Added temporary column bitmaps used in ROR scans
sql/opt_sum.cc:
Added get_ext_record_count() which is used in COUNT() optimization if handler has HA_HAS_RECORDS
Note that we don't call this if handler has HA_STATS_RECORDS_IS_EXACT set.
sql/protocol.cc:
We need to mark columns as readable in ::store() as we sometimes return default value for fields to the user
sql/records.cc:
Updates for new handler interface
sql/set_var.cc:
Handle splitting OPTION_STATUS_NO_TRANS_UPDATE to two flags
sql/share/errmsg.txt:
Fixed wrong
sql/sp.cc:
Mark that we are using all columns for the proc table
Update call to setup_tables() to use new prototype
sql/sp_head.cc:
Removed QQ comment
sql/spatial.cc:
Removed wrong QQ comment
sql/sql_acl.cc:
Mark that we need all columns for acl tables
Supply memroot to some 'new' calls.
Indentation fixes
sql/sql_base.cc:
set_query_id removed
Ensure we call ha_reset() at end of each statement
Mark read columns in read_set and changed columns in write_set (Before all columns was marked in read set)
Fixed marking of some columns that was not proplerly marked before
Maintain in TABLE->merge_keys set of all keys that are used in some way
Removed not used 'conds' argument from setup_tables()
Remove not used setting of 'dupp_field' in insert_fields()
Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after crash)
sql/sql_bitmap.h:
Added is_overlapping()
sql/sql_class.cc:
Slow_logs was not properly initialized, which could maybe cause extra/lost entries in slow log.
set_query_id -> mark_used_columns
Simpler variable usage in pack_row() (cleanup)
Moved some variable declartion at start of function for better code readability
sql/sql_class.h:
Added enum_mark_columns
Updated comments
Renamed dupp_field -> dup_field
Added virtual function 'can_rollback_data()' to select_insert() to be used in CREATE ... SELECT to optimize use of OPTION_STATUS_NO_TRANS_UPDATE.
(This fixes a bug in CREATE ... SELECT where we did give wrong warnings when using non transacational tables)
sql/sql_delete.cc:
Updates to new handler interface
Call table->mark_columns_needed_for_delete() to allow us to put additional columns in column usage maps if handler so requires.
Call table->prepare_for_position() to tell handler that we are going to call ha_position().
Removed call to free_io_cache(). (io_cache is now removed in ha_reset()).
Fixed calls to setup_tables()
sql/sql_do.cc:
Update call to setup_fields()
sql/sql_handler.cc:
Tell handler tables to always read all columns.
Use temporary column map when storing value in field for later index usage
sql/sql_help.cc:
Makr all used fields to be read
Update call to setup_fields()
sql/sql_insert.cc:
Tell handler we are going to update the auto_increment column
dupp_field -> dup_field
Set column usage bits for timestamp field.
Call table->mark_columns_needed_for_insert() and table->mark_auto_increment_column()
Removed not used argument from mysql_prepare_insert_check_table().
If we get an duplicate row on insert, change column map to read and write all columns while retrying the operatation.
This is required by the definition of REPLACE and also ensures that fields that are only part of UPDATE are properly handled.
This fixed a bug in NDB and REPLACE where REPLACE wrongly copied some column values from the replaced row.
Setup new bitmaps for delayed insert rows
Remove reseting of next_number_fields as it will be reset on next call to handler_insert()
Fixed usage of thd->options and OPTION_STATUS_NO_TRANS_UPDATE.
The issue was that one should not to reset this flag as it may be set by a previous statement.
The way it was now used caused us to loose some warnings and get other wrong warnings when using non transactional tables mixed with transactional.
I fixed it by introducing 'select_insert::can_rollback_data' to inform send_error() that the given statement can be rolled back (which in case of CREATE TABLE can always be done)
Don't close tables created with CREATE ... SELECT but keep them in the table cache.
Moved out MY_HOOKS from inside function (better readability)
sql/sql_load.cc:
Update to use new handler and column marking interface
Update using setup_tables()
sql/sql_olap.cc:
Update calls to setup_tables
Use enums instead of constants to setup_fields()
sql/sql_parse.cc:
Handle OPTION_KEEP_LOG:
- Set it on CREATE TEMPORARY TABLE / DROP TABLE
- Reset it when OPTION_STATUS_NO_TRANS_UPDATE is reset
- Don't set it for CREATE ... SELECT (this is handled in select_create class)
Remove reseting of OPTION_STATUS_NO_TRANS_UPDATE in begin_trans() as this should already be reset.
If in autocommit mode, reset OPTION_KEEP_LOG and OPTION_STATUS_NO_TRANS_UPDATE to not give warnings in future commands
sql/sql_partition.cc:
Update walk() usage
Trivial indentation fixes
sql/sql_plugin.cc:
Mark all columns as used for plugins
sql/sql_prepare.cc:
Added assert to find out hidden bugs in character_set_client (got an error in debug binary when this not set correctly)
Updates for new handler interface
Update calls to setup_fields()
sql/sql_repl.cc:
Indentation fixes
sql/sql_select.cc:
Update call to setup_tables() and setup_fields()
Remove some old disabled code
Update to new hadler interface
Indentation cleanups
Added column bitmaps for temporary tables.
Remove updating of the removed slots in the Field class
Added TABLE argument to cp_buffer_from_ref() (To be able to install temporary column maps)
For internal temporary tables, use handler::write_row(), handler::delete_row() and handler::update_row() instead of handler::ha_xxxx() for faster execution.
sql/sql_select.h:
Indentaition fixes.
Install temporary column usage maps when needed
Added TABLE element to cp_buffer_from_ref()
sql/sql_show.cc:
Update to new handler interface
Mark all columns used for internal tables.
Style fixes.
Added support for 'future' ROW_TYPE_PAGES.
Don't allocate TMP_TABLE_PARAM with calloc. The 'init()' function will initialize the structure properly.
sql/sql_table.cc:
Update to new handler interface
Simple my_snprintf -> strmake()
Changed some constants to defines
Don't test for NULL in primary key (as we a couple of line above force the PRIMARY KEY to be NOT NULL)
Change field->add_index to use field->flags & FIELD_IN_ADD_INDEX
Mark all columns as used for ALTER TABLE
Style fixes
Update call to filesort()
sql/sql_trigger.h:
Added friend functions to be able to test if triggers exists for table we are going to insert/update or delete in.
sql/sql_udf.cc:
Mark all columns as used for udf system table.
sql/sql_union.cc:
Update call to walk()
Update to new handler interface
sql/sql_update.cc:
Remove query_id argument from compare_record()
Use column bitmaps instead of query_id.
We can't use compare_records() to skip updates for handlers that returns a partial column set and the read_set doesn't cover all columns in the write set, because compare_record() can't in this case know if a not read column changed value.
Update call to setup_fields()
Using separate column read and write sets allows for easier checking of timestamp field was set by statement.
Removed call to free_io_cache() as this is now done in ha_reset()
Call table->mark_columns_needed_for_update() and table->prepare_for_position()
Style fixes
sql/sql_view.cc:
Style fixes
sql/table.cc:
Remove implicitely include 'errno.h'
Remove code for building normalized path, as this is now identical to 'path'
Remove field->fieldnr
Added update of field->part_of_key_not_clustered()
Create column bitmaps in TABLE and TABLE_SHARE
Don't setup a temporary MEM_ROOT object as a thread specific variable for the handler. Instead we send the to-be-used MEMROOT to get_new_handler()
Update to new handler interface
Update call to walk()
Added new functions:
- st_table::clear_column_bitmaps()
- st_table::prepare_for_position()
- st_table::mark_columns_used_by_index()
- st_table::restore_column_maps_after_mark_index()
- st_table::mark_columns_used_by_index_no_reset()
- st_table::mark_auto_increment_column()
- st_table::mark_columns_needed_for_delete()
- st_table::mark_columns_needed_for_update()
- st_table::mark_columns_needed_for_insert()
sql/table.h:
Moved column usage bitmaps from handler to TABLE
Added to TABLE_SHARE all_set and column_bitmap_size
Added to TABLE merge_keys, bitmap_init_values, def_read_set, def_write_set, tmp_set, read_set and write_set.
Declared all new table column bitmap functions
Added TABLE functions column_bitmaps_set(), column_bitmaps_set_no_signal(), use_all_columns() and default_column_bitmaps()
Added functions: tmp_use_all_columns() and tmp_restore_column_map() to temporarly switch column bitmaps
Added functions: dbug_tmp_use_all_columns() and dbug_tmp_restore_column_map() to temporarly switch column bitmaps to avoid asserts in Field::store() and Field::val().
sql/tztime.cc:
Mark all columns as used for timezone tables
storage/archive/ha_archive.cc:
Update to new handler interface
storage/archive/ha_archive.h:
Update to new handler interface
storage/blackhole/ha_blackhole.cc:
Update to new handler interface
storage/blackhole/ha_blackhole.h:
Update to new handler interface
removed not needed flag HA_DUPP_POS
storage/csv/ha_tina.cc:
Update to new handler interface
storage/csv/ha_tina.h:
Update to new handler interface
storage/example/ha_example.cc:
Update to new handler interface
storage/example/ha_example.h:
Update to new handler interface
storage/heap/hp_extra.c:
Added heap_reset() (Required by new handler interface)
storage/heap/hp_test2.c:
Use heap_reset()
storage/myisam/ft_boolean_search.c:
Fixed compiler warning
storage/myisam/mi_extra.c:
Added mi_reset() (Required by new handler interface)
storage/myisam/mi_search.c:
Fixed DBUG_PRINT messages to use 0x%lx instead of %lx
storage/myisam/mi_test2.c:
Use mi_reset()
storage/myisam/myisampack.c:
Use mi_reset()
storage/myisammrg/myrg_extra.c:
Added myrg_reset() (Required by new handler interface)
unittest/mysys/base64.t.c:
Include my_global.h
Don't include implictely include file 'stdlib.h'
2006-06-04 17:52:22 +02:00
|
|
|
|
|
|
|
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;
|
2009-06-25 12:05:53 +02:00
|
|
|
/* TODO: why does this walk WHERE/HAVING but not ON expressions of outer joins? */
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
BitKeeper/etc/ignore:
added mysys/test_bitmap
include/base64.h:
Removed my_global.h, as this must be included first in any program
include/heap.h:
Added heap_reset() (Required by new handler interface)
include/my_base.h:
Removed HA_EXTRA_RESET. MySQL will now call ::reset() instead of ::extra(HA_EXTRA_RESET).
HA_EXTRA_RETRIVE_ALL_COLS and HA_EXTRA_RETRIVE_PRIMARY key are deleted as the column bitmaps makes these unnecessary
include/my_bitmap.h:
Remove my_pthread.h (should be included at upper level)
Introduced my_bitmap_map typedef to make it the bitmap handling more like a black box
Added bitmap_is_overlapping(), bitmap_test_and_clear(), bitmap_copy() and bitmap_cmp()
Made bitmap_set_bit(), bitmap_flip_bit(), bitmap_clear_bit() return void
include/myisam.h:
Added mi_reset() (Required by new handler interface)
include/myisammrg.h:
Added myrg_reset() (Required by new handler interface)
include/mysql_com.h:
Added flag FIELD_IN_ADD_INDEX to be able to remove Field->add_index
mysql-test/extra/binlog_tests/mix_innodb_myisam_binlog.test:
Added testing of CREATE ... SELECT in a mixed environment
(This found some bugs that Mats is going to fix shortly)
mysql-test/install_test_db.sh:
Simplify ldata usage
Added --tmpdir=. option to mysqld bootstrap (Removed some warnings when TMPDIR was wrongly set)
mysql-test/mysql-test-run.pl:
Added --tmpdir=. to bootstrap
mysql-test/mysql-test-run.sh:
Use copy instead of INSTALL_DB for master and slave databases.
(Speeds up startup time a lot!)
Remove snapshot directories at startup (removes some strange warnings)
mysql-test/r/binlog_row_mix_innodb_myisam.result:
Added testing of CREATE ... SELECT in a mixed environment
(This found some bugs that Mats is going to fix shortly)
mysql-test/r/binlog_stm_mix_innodb_myisam.result:
Added testing of CREATE ... SELECT in a mixed environment
mysql-test/r/create.result:
Some extra tests of warnings and number of tables opened by CREATE ... SELECT
mysql-test/r/federated.result:
Drop some left over tables
Added testing of multiple table update and multiple table delete (with and without keys)
mysql-test/r/func_gconcat.result:
Enable some disabled tests (converted them slightly to be predictable)
mysql-test/r/func_time.result:
Added drop of test function
mysql-test/r/innodb_mysql.result:
Added tests for CREATE ... SELECT
mysql-test/r/insert.result:
More tests
Added testing of duplicate columns in insert
mysql-test/r/loaddata.result:
Added testing LOAD DATA ... SET ...
mysql-test/r/multi_update.result:
Test multi updates and deletes using primary key and without
mysql-test/r/ndb_index_unique.result:
Better error message
mysql-test/r/ndb_replace.result:
New correct result after fixing REPLACE handling with NDB
mysql-test/r/rpl_ddl.result:
Now we don't get these (wrong) warnings anymore
mysql-test/r/view_grant.result:
Drop used views
mysql-test/t/create.test:
Some extra tests of warnings and number of tables opened by CREATE ... SELECT
mysql-test/t/federated.test:
Drop some left over tables
Added testing of multiple table update and multiple table delete (with and without keys)
mysql-test/t/func_gconcat.test:
Enable some disabled tests (converted them slightly to be predictable)
mysql-test/t/func_time.test:
Added drop of test function
mysql-test/t/innodb_mysql.test:
Added tests for CREATE ... SELECT
mysql-test/t/insert.test:
More tests
Added testing of duplicate columns in insert
mysql-test/t/loaddata.test:
Added testing LOAD DATA ... SET ...
mysql-test/t/multi_update.test:
Test multi updates and deletes using primary key and without
mysql-test/t/view_grant.test:
Drop used views
mysql-test/valgrind.supp:
Added supression of not needed warnings when printing stack trace
mysys/base64.c:
Include my_global.h first
mysys/my_bitmap.c:
Added bitmap_is_overlapping(), bitmap_test_and_clear() and bitmap_copy()
Changed logic of bitmap handling to be a bit more efficent (Did this together with Mikael Ronström)
Now the 'extra, not used bits' in the bitmap are assumed to have a 'random value' and the bitmap functions are free to change them whenever needed.
Changed how mutex is allocated to make 'bitmap_free()' function simpler.
mysys/thr_lock.c:
Added 0x before thread pointers (for easier comparison of DBUG traces)
sql/event.cc:
Ensure 'use_all_columns()' is used for event tables
Don't print warning that event table is damaged if it doesn't exists.
sql/field.cc:
Added ASSERT_COLUMN_MARKED_FOR_WRITE in all store() methods and ASSERT_COLUMN_MARKED_FOR_READ in all val() methods to catch wrong setting if table->read_set and table->write_set
(Rest of changes are only indentation cleanups)
sql/field.h:
Removed Field->query_id (replaced by table->read_set and table->write_set)
Removed Field->fieldnr (use Field->field_index instead)
Removed Field->add_index (Use Field->flags instead)
Add Field->part_of_key_not_clustered (for usage in opt_range.cc)
sql/filesort.cc:
Added paramater sort_postion to filesort() to force sorting by position instead of storing all fields in the result set.
This allowed me to remove checking of sql_command.
Create a temporary column bitmap for fields that are used by the sorting process.
Use column bitmaps instead of query_id
sql/ha_berkeley.cc:
Update to 'newer' table handler interface
sql/ha_berkeley.h:
Update to 'newer' table handler interface
sql/ha_federated.cc:
Update to 'newer' table handler interface
Only read columns that are needed from remote server.
In case of eq ranges, don't generate two conditions in the WHERE clause
(this can still be optimized, but would require a bigger code change)
Use 'simpler to use' XXXX_LEN' macros
A bit simpler logic in ::write_row() when creating statements.
In update, only include test of fields actually read.
(This greatly simplifies the queries sent by the federated engine)
Similar changes done for delete_row()
sql/ha_federated.h:
Update to 'newer' table handler interface
Changed XXX_LEN macros to use sizeof(...)-1, to simplify usage in ha_federated.cc
Added HA_PRIMARY_KEY_REQUIRED_FOR_DELETE to tell MySQL to read all primary key columns in case of DELETE
sql/ha_heap.cc:
Update to 'newer' table handler interface
sql/ha_heap.h:
Update to 'newer' table handler interface
sql/ha_innodb.cc:
Update to 'newer' table handler interface
- Update innobase_create_handler() to new interface
- Removed HA_NOT_EXACT_COUNT (not needed)
- Renamed HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- Prefixed base status variables with 'stats'
- Use table column bitmaps instead of ha_get_bit_in_read_set()
- Added ::reset(), with code from ::extra(HA_EXTRA_RESET)
- Removed HA_EXTRA_RETRIVE_ALL_COLS and HA_EXTRA_RETRIEVE_PRIMARY_KEY as
the table->read_set and table->write_set bitmaps now are accurate
sql/ha_innodb.h:
Update to 'newer' table handler interface
- table_flags are now ulonglong
- Added reset() method
- Removed not needed ha_retrieve_all_cols() and ha_retrieve_all_pk() columns.
- Made build_template() a class function to be able to easier access class variables
sql/ha_myisam.cc:
Update to 'newer' table handler interface
sql/ha_myisam.h:
Update to 'newer' table handler interface
sql/ha_myisammrg.cc:
Update to 'newer' table handler interface
sql/ha_myisammrg.h:
Update to 'newer' table handler interface
sql/ha_ndbcluster.cc:
Update to 'newer' table handler interface
Fixed use_blob_value() to be accurate
In ::complemented_read() we have to check both the read and write bitmap as the old code did mark all changed columns also in the read map
Correct dumping of field data with DBUG_DUMP
Prefix addresses in DBUG_PRINT with 0x
Fixed usage of not initialized memory
Update to use field->flags & FIELD_IN_ADD_INDEX instead of field->add_index.
sql/ha_ndbcluster.h:
Update to 'newer' table handler interface
sql/ha_ndbcluster_binlog.cc:
Mark usage of all columns in ndbcluster binlog tables
false -> FALSE, true -> TRUE
Use table->s->all_set instead of creating a temporary bitmap.
sql/ha_partition.cc:
Update to 'newer' table handler interface
Added memroot to initialise_partitions() and related functions to get faster memory allocation.
partition_create_handler() is now responsible for initialisation of the partition object
Some trivial optimizations and indentation fixes
Ensure that table_flags() are up to date
Removed documentation for removed HA_EXTRA flags
Fixed 'strange' usage of m_file[i] in new_handlers_from_part_info()that worked in current code 'by chance'
sql/ha_partition.h:
Update to 'newer' table handler interface
sql/handler.cc:
create_xxx handler now takes MEMROOT as an argument to simplify memory allocation.
Much simpler get_new_handler()
(Initialization of the object is now handled by the create method for the engine)
Moved all allocation of bitmap handling to the TABLE object (in table.cc)
Added column_bitmaps_signal() to signal column usage changes.
Changed binlog_log_row() to use the exiusting all_set bitmap in the table object.
Added ha_reset() function to test that the file object is ok at end of statement and call handler::reset()
Added use_hidden_primary_key() to signal handler that we we are going to read and update + delete the row and the handler should thus remember the position for the row
sql/handler.h:
Added HA_NO_TRANSACTIONS, HA_PARTIAL_COLUMN_READ, HA_REQUIRES_KEY_COLUMNS_FOR_DELETE,HA_PRIMARY_KEY_REQUIRED_FOR_DELETE and HA_HAS_RECORDS
Removed HA_NOT_EXACT_COUNT, HA_READ_RND_SAME
HA_DUPP_POS -> HA_DUPLICATE_POS
HA_NOT_EXACT_COUNT replaced by HA_STATS_RECORDS_IS_EXACT, HA_HAS_RECORDS and records()
HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION
Added future row type 'ROW_TYPE_PAGES'
Added MEM_ROOT to handlerton 'create' function
Added ha_statistics, a structure for all status variable in the base handler class.
Moved all status variables in the handler class into a stats structs to improve readability.
ha_table_flags() is now a cached (not virtual) version of table_flags()
reset() doesn't anymore call extra(HA_EXTRA_RESET) but is a function of it's own.
Renamed dupp_ref to dup_ref
Renamed not used handler::sortkey
Moved read_set and write_set to TABLE structure
handler::init() function added for cacheing of virtual constants from engine.
sql/item.cc:
Added register_field_in_read_map() for marking used columns in expression.
This is used by filesort() for creating an optimal column bitmap while retrieving columns for sorting.
Initalize value.cs_info.character_set_client to fix core dump bug with --debug
set_query_id -> mark_used_columns
Mark used columns in read_set OR write_set.
sql/item.h:
Removed reset_query_id_processor() as it's not needed anymore.
Added register_field_in_read_map()
Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
sql/item_cmpfunc.cc:
Temporary mark used columns to be read/writable
Update Item::walk to new interface
sql/item_cmpfunc.h:
Added extra argument to Item::walk() to indicate if we should also traverse sub queries.
sql/item_func.cc:
Update Item::walk() to new interface
table_flags() -> ha_table_flags()
sql/item_func.h:
Update Item::walk() to new interface
sql/item_row.cc:
Update Item::walk() to new interface
sql/item_row.h:
Update Item::walk() to new interface
sql/item_strfunc.h:
Update Item::walk() to new interface
sql/item_subselect.cc:
Added Item_subselect::walk()
(It was a bug it was missing before. Not sure what kind of bugs this could have caused)
sql/item_subselect.h:
Update Item::walk() to new interface
sql/item_sum.cc:
Update Item::walk() to new interface
Updates for new handler interace
sql/item_sum.h:
Update Item::walk() to new interface
sql/key.cc:
Updates for new handler interace
sql/log.cc:
Mark all columns used for log tables
Split options flag
Ensured that second argument to trans_register_ha is a bool
sql/log_event.cc:
Fixed comments to be withing 79 characters
Use OPTION_KEEP_LOG instead of OPTION_STATUS_NO_TRANS_UPDATE to remove wrong warnings
Updates for new handler interface
Use 0x%lx instead of %p (portability problem)
sql/mysql_priv.h:
Added OPTION_KEEP_LOG to indicate that we should replicate the binlog even on rollback
Removed not used 'conds' argument to setup_tables
sql/mysqld.cc:
Indentation fixes and removed old comment
sql/opt_range.cc:
Update to new handler and bitmap interface.
Fixed calls to cp_buffer_from_ref() and walk() (new argument).
Create new temporary bitmaps for ror scans.
(Needed because of handler changes and to get more accurate column bitmaps than before)
Remove not needed file->ha_reset() call before file->close().
Some trivial optimization and indentation fixes.
Use Field->part_of_key_not_clustered() to check if field is part of a key, instead of looping over all key parts.
Added flag 'in_ror_merged_scan' to allow ::get_next() to know that we need a special column bitmap to only fetch pointer to record.
This is needed because ror scan uses the same TABLE object but different file objects, which creates problem for the column bitmap handling.
(This is a temporary solution. A better one would be to allocate an own TABLE object for ROR scans)
Optimized bitmap handling in ror scans:
- Start bitmap at position 0, not 1
- Use same bitmap size as in TABLE
- Use table->read_set and table->write_set to create column bitmaps instead of looping over all fields in table
sql/opt_range.h:
Added 'in_ror_merged_scan' to indicate if we are doing a ROR scan
Added temporary column bitmaps used in ROR scans
sql/opt_sum.cc:
Added get_ext_record_count() which is used in COUNT() optimization if handler has HA_HAS_RECORDS
Note that we don't call this if handler has HA_STATS_RECORDS_IS_EXACT set.
sql/protocol.cc:
We need to mark columns as readable in ::store() as we sometimes return default value for fields to the user
sql/records.cc:
Updates for new handler interface
sql/set_var.cc:
Handle splitting OPTION_STATUS_NO_TRANS_UPDATE to two flags
sql/share/errmsg.txt:
Fixed wrong
sql/sp.cc:
Mark that we are using all columns for the proc table
Update call to setup_tables() to use new prototype
sql/sp_head.cc:
Removed QQ comment
sql/spatial.cc:
Removed wrong QQ comment
sql/sql_acl.cc:
Mark that we need all columns for acl tables
Supply memroot to some 'new' calls.
Indentation fixes
sql/sql_base.cc:
set_query_id removed
Ensure we call ha_reset() at end of each statement
Mark read columns in read_set and changed columns in write_set (Before all columns was marked in read set)
Fixed marking of some columns that was not proplerly marked before
Maintain in TABLE->merge_keys set of all keys that are used in some way
Removed not used 'conds' argument from setup_tables()
Remove not used setting of 'dupp_field' in insert_fields()
Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after crash)
sql/sql_bitmap.h:
Added is_overlapping()
sql/sql_class.cc:
Slow_logs was not properly initialized, which could maybe cause extra/lost entries in slow log.
set_query_id -> mark_used_columns
Simpler variable usage in pack_row() (cleanup)
Moved some variable declartion at start of function for better code readability
sql/sql_class.h:
Added enum_mark_columns
Updated comments
Renamed dupp_field -> dup_field
Added virtual function 'can_rollback_data()' to select_insert() to be used in CREATE ... SELECT to optimize use of OPTION_STATUS_NO_TRANS_UPDATE.
(This fixes a bug in CREATE ... SELECT where we did give wrong warnings when using non transacational tables)
sql/sql_delete.cc:
Updates to new handler interface
Call table->mark_columns_needed_for_delete() to allow us to put additional columns in column usage maps if handler so requires.
Call table->prepare_for_position() to tell handler that we are going to call ha_position().
Removed call to free_io_cache(). (io_cache is now removed in ha_reset()).
Fixed calls to setup_tables()
sql/sql_do.cc:
Update call to setup_fields()
sql/sql_handler.cc:
Tell handler tables to always read all columns.
Use temporary column map when storing value in field for later index usage
sql/sql_help.cc:
Makr all used fields to be read
Update call to setup_fields()
sql/sql_insert.cc:
Tell handler we are going to update the auto_increment column
dupp_field -> dup_field
Set column usage bits for timestamp field.
Call table->mark_columns_needed_for_insert() and table->mark_auto_increment_column()
Removed not used argument from mysql_prepare_insert_check_table().
If we get an duplicate row on insert, change column map to read and write all columns while retrying the operatation.
This is required by the definition of REPLACE and also ensures that fields that are only part of UPDATE are properly handled.
This fixed a bug in NDB and REPLACE where REPLACE wrongly copied some column values from the replaced row.
Setup new bitmaps for delayed insert rows
Remove reseting of next_number_fields as it will be reset on next call to handler_insert()
Fixed usage of thd->options and OPTION_STATUS_NO_TRANS_UPDATE.
The issue was that one should not to reset this flag as it may be set by a previous statement.
The way it was now used caused us to loose some warnings and get other wrong warnings when using non transactional tables mixed with transactional.
I fixed it by introducing 'select_insert::can_rollback_data' to inform send_error() that the given statement can be rolled back (which in case of CREATE TABLE can always be done)
Don't close tables created with CREATE ... SELECT but keep them in the table cache.
Moved out MY_HOOKS from inside function (better readability)
sql/sql_load.cc:
Update to use new handler and column marking interface
Update using setup_tables()
sql/sql_olap.cc:
Update calls to setup_tables
Use enums instead of constants to setup_fields()
sql/sql_parse.cc:
Handle OPTION_KEEP_LOG:
- Set it on CREATE TEMPORARY TABLE / DROP TABLE
- Reset it when OPTION_STATUS_NO_TRANS_UPDATE is reset
- Don't set it for CREATE ... SELECT (this is handled in select_create class)
Remove reseting of OPTION_STATUS_NO_TRANS_UPDATE in begin_trans() as this should already be reset.
If in autocommit mode, reset OPTION_KEEP_LOG and OPTION_STATUS_NO_TRANS_UPDATE to not give warnings in future commands
sql/sql_partition.cc:
Update walk() usage
Trivial indentation fixes
sql/sql_plugin.cc:
Mark all columns as used for plugins
sql/sql_prepare.cc:
Added assert to find out hidden bugs in character_set_client (got an error in debug binary when this not set correctly)
Updates for new handler interface
Update calls to setup_fields()
sql/sql_repl.cc:
Indentation fixes
sql/sql_select.cc:
Update call to setup_tables() and setup_fields()
Remove some old disabled code
Update to new hadler interface
Indentation cleanups
Added column bitmaps for temporary tables.
Remove updating of the removed slots in the Field class
Added TABLE argument to cp_buffer_from_ref() (To be able to install temporary column maps)
For internal temporary tables, use handler::write_row(), handler::delete_row() and handler::update_row() instead of handler::ha_xxxx() for faster execution.
sql/sql_select.h:
Indentaition fixes.
Install temporary column usage maps when needed
Added TABLE element to cp_buffer_from_ref()
sql/sql_show.cc:
Update to new handler interface
Mark all columns used for internal tables.
Style fixes.
Added support for 'future' ROW_TYPE_PAGES.
Don't allocate TMP_TABLE_PARAM with calloc. The 'init()' function will initialize the structure properly.
sql/sql_table.cc:
Update to new handler interface
Simple my_snprintf -> strmake()
Changed some constants to defines
Don't test for NULL in primary key (as we a couple of line above force the PRIMARY KEY to be NOT NULL)
Change field->add_index to use field->flags & FIELD_IN_ADD_INDEX
Mark all columns as used for ALTER TABLE
Style fixes
Update call to filesort()
sql/sql_trigger.h:
Added friend functions to be able to test if triggers exists for table we are going to insert/update or delete in.
sql/sql_udf.cc:
Mark all columns as used for udf system table.
sql/sql_union.cc:
Update call to walk()
Update to new handler interface
sql/sql_update.cc:
Remove query_id argument from compare_record()
Use column bitmaps instead of query_id.
We can't use compare_records() to skip updates for handlers that returns a partial column set and the read_set doesn't cover all columns in the write set, because compare_record() can't in this case know if a not read column changed value.
Update call to setup_fields()
Using separate column read and write sets allows for easier checking of timestamp field was set by statement.
Removed call to free_io_cache() as this is now done in ha_reset()
Call table->mark_columns_needed_for_update() and table->prepare_for_position()
Style fixes
sql/sql_view.cc:
Style fixes
sql/table.cc:
Remove implicitely include 'errno.h'
Remove code for building normalized path, as this is now identical to 'path'
Remove field->fieldnr
Added update of field->part_of_key_not_clustered()
Create column bitmaps in TABLE and TABLE_SHARE
Don't setup a temporary MEM_ROOT object as a thread specific variable for the handler. Instead we send the to-be-used MEMROOT to get_new_handler()
Update to new handler interface
Update call to walk()
Added new functions:
- st_table::clear_column_bitmaps()
- st_table::prepare_for_position()
- st_table::mark_columns_used_by_index()
- st_table::restore_column_maps_after_mark_index()
- st_table::mark_columns_used_by_index_no_reset()
- st_table::mark_auto_increment_column()
- st_table::mark_columns_needed_for_delete()
- st_table::mark_columns_needed_for_update()
- st_table::mark_columns_needed_for_insert()
sql/table.h:
Moved column usage bitmaps from handler to TABLE
Added to TABLE_SHARE all_set and column_bitmap_size
Added to TABLE merge_keys, bitmap_init_values, def_read_set, def_write_set, tmp_set, read_set and write_set.
Declared all new table column bitmap functions
Added TABLE functions column_bitmaps_set(), column_bitmaps_set_no_signal(), use_all_columns() and default_column_bitmaps()
Added functions: tmp_use_all_columns() and tmp_restore_column_map() to temporarly switch column bitmaps
Added functions: dbug_tmp_use_all_columns() and dbug_tmp_restore_column_map() to temporarly switch column bitmaps to avoid asserts in Field::store() and Field::val().
sql/tztime.cc:
Mark all columns as used for timezone tables
storage/archive/ha_archive.cc:
Update to new handler interface
storage/archive/ha_archive.h:
Update to new handler interface
storage/blackhole/ha_blackhole.cc:
Update to new handler interface
storage/blackhole/ha_blackhole.h:
Update to new handler interface
removed not needed flag HA_DUPP_POS
storage/csv/ha_tina.cc:
Update to new handler interface
storage/csv/ha_tina.h:
Update to new handler interface
storage/example/ha_example.cc:
Update to new handler interface
storage/example/ha_example.h:
Update to new handler interface
storage/heap/hp_extra.c:
Added heap_reset() (Required by new handler interface)
storage/heap/hp_test2.c:
Use heap_reset()
storage/myisam/ft_boolean_search.c:
Fixed compiler warning
storage/myisam/mi_extra.c:
Added mi_reset() (Required by new handler interface)
storage/myisam/mi_search.c:
Fixed DBUG_PRINT messages to use 0x%lx instead of %lx
storage/myisam/mi_test2.c:
Use mi_reset()
storage/myisam/myisampack.c:
Use mi_reset()
storage/myisammrg/myrg_extra.c:
Added myrg_reset() (Required by new handler interface)
unittest/mysys/base64.t.c:
Include my_global.h
Don't include implictely include file 'stdlib.h'
2006-06-04 17:52:22 +02:00
|
|
|
|
|
|
|
while ((item=li++))
|
|
|
|
{
|
|
|
|
if (item->walk(processor, walk_subquery, argument))
|
|
|
|
return 1;
|
|
|
|
}
|
2010-06-10 22:45:22 +02:00
|
|
|
for (order= lex->order_list.first ; order; order= order->next)
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
BitKeeper/etc/ignore:
added mysys/test_bitmap
include/base64.h:
Removed my_global.h, as this must be included first in any program
include/heap.h:
Added heap_reset() (Required by new handler interface)
include/my_base.h:
Removed HA_EXTRA_RESET. MySQL will now call ::reset() instead of ::extra(HA_EXTRA_RESET).
HA_EXTRA_RETRIVE_ALL_COLS and HA_EXTRA_RETRIVE_PRIMARY key are deleted as the column bitmaps makes these unnecessary
include/my_bitmap.h:
Remove my_pthread.h (should be included at upper level)
Introduced my_bitmap_map typedef to make it the bitmap handling more like a black box
Added bitmap_is_overlapping(), bitmap_test_and_clear(), bitmap_copy() and bitmap_cmp()
Made bitmap_set_bit(), bitmap_flip_bit(), bitmap_clear_bit() return void
include/myisam.h:
Added mi_reset() (Required by new handler interface)
include/myisammrg.h:
Added myrg_reset() (Required by new handler interface)
include/mysql_com.h:
Added flag FIELD_IN_ADD_INDEX to be able to remove Field->add_index
mysql-test/extra/binlog_tests/mix_innodb_myisam_binlog.test:
Added testing of CREATE ... SELECT in a mixed environment
(This found some bugs that Mats is going to fix shortly)
mysql-test/install_test_db.sh:
Simplify ldata usage
Added --tmpdir=. option to mysqld bootstrap (Removed some warnings when TMPDIR was wrongly set)
mysql-test/mysql-test-run.pl:
Added --tmpdir=. to bootstrap
mysql-test/mysql-test-run.sh:
Use copy instead of INSTALL_DB for master and slave databases.
(Speeds up startup time a lot!)
Remove snapshot directories at startup (removes some strange warnings)
mysql-test/r/binlog_row_mix_innodb_myisam.result:
Added testing of CREATE ... SELECT in a mixed environment
(This found some bugs that Mats is going to fix shortly)
mysql-test/r/binlog_stm_mix_innodb_myisam.result:
Added testing of CREATE ... SELECT in a mixed environment
mysql-test/r/create.result:
Some extra tests of warnings and number of tables opened by CREATE ... SELECT
mysql-test/r/federated.result:
Drop some left over tables
Added testing of multiple table update and multiple table delete (with and without keys)
mysql-test/r/func_gconcat.result:
Enable some disabled tests (converted them slightly to be predictable)
mysql-test/r/func_time.result:
Added drop of test function
mysql-test/r/innodb_mysql.result:
Added tests for CREATE ... SELECT
mysql-test/r/insert.result:
More tests
Added testing of duplicate columns in insert
mysql-test/r/loaddata.result:
Added testing LOAD DATA ... SET ...
mysql-test/r/multi_update.result:
Test multi updates and deletes using primary key and without
mysql-test/r/ndb_index_unique.result:
Better error message
mysql-test/r/ndb_replace.result:
New correct result after fixing REPLACE handling with NDB
mysql-test/r/rpl_ddl.result:
Now we don't get these (wrong) warnings anymore
mysql-test/r/view_grant.result:
Drop used views
mysql-test/t/create.test:
Some extra tests of warnings and number of tables opened by CREATE ... SELECT
mysql-test/t/federated.test:
Drop some left over tables
Added testing of multiple table update and multiple table delete (with and without keys)
mysql-test/t/func_gconcat.test:
Enable some disabled tests (converted them slightly to be predictable)
mysql-test/t/func_time.test:
Added drop of test function
mysql-test/t/innodb_mysql.test:
Added tests for CREATE ... SELECT
mysql-test/t/insert.test:
More tests
Added testing of duplicate columns in insert
mysql-test/t/loaddata.test:
Added testing LOAD DATA ... SET ...
mysql-test/t/multi_update.test:
Test multi updates and deletes using primary key and without
mysql-test/t/view_grant.test:
Drop used views
mysql-test/valgrind.supp:
Added supression of not needed warnings when printing stack trace
mysys/base64.c:
Include my_global.h first
mysys/my_bitmap.c:
Added bitmap_is_overlapping(), bitmap_test_and_clear() and bitmap_copy()
Changed logic of bitmap handling to be a bit more efficent (Did this together with Mikael Ronström)
Now the 'extra, not used bits' in the bitmap are assumed to have a 'random value' and the bitmap functions are free to change them whenever needed.
Changed how mutex is allocated to make 'bitmap_free()' function simpler.
mysys/thr_lock.c:
Added 0x before thread pointers (for easier comparison of DBUG traces)
sql/event.cc:
Ensure 'use_all_columns()' is used for event tables
Don't print warning that event table is damaged if it doesn't exists.
sql/field.cc:
Added ASSERT_COLUMN_MARKED_FOR_WRITE in all store() methods and ASSERT_COLUMN_MARKED_FOR_READ in all val() methods to catch wrong setting if table->read_set and table->write_set
(Rest of changes are only indentation cleanups)
sql/field.h:
Removed Field->query_id (replaced by table->read_set and table->write_set)
Removed Field->fieldnr (use Field->field_index instead)
Removed Field->add_index (Use Field->flags instead)
Add Field->part_of_key_not_clustered (for usage in opt_range.cc)
sql/filesort.cc:
Added paramater sort_postion to filesort() to force sorting by position instead of storing all fields in the result set.
This allowed me to remove checking of sql_command.
Create a temporary column bitmap for fields that are used by the sorting process.
Use column bitmaps instead of query_id
sql/ha_berkeley.cc:
Update to 'newer' table handler interface
sql/ha_berkeley.h:
Update to 'newer' table handler interface
sql/ha_federated.cc:
Update to 'newer' table handler interface
Only read columns that are needed from remote server.
In case of eq ranges, don't generate two conditions in the WHERE clause
(this can still be optimized, but would require a bigger code change)
Use 'simpler to use' XXXX_LEN' macros
A bit simpler logic in ::write_row() when creating statements.
In update, only include test of fields actually read.
(This greatly simplifies the queries sent by the federated engine)
Similar changes done for delete_row()
sql/ha_federated.h:
Update to 'newer' table handler interface
Changed XXX_LEN macros to use sizeof(...)-1, to simplify usage in ha_federated.cc
Added HA_PRIMARY_KEY_REQUIRED_FOR_DELETE to tell MySQL to read all primary key columns in case of DELETE
sql/ha_heap.cc:
Update to 'newer' table handler interface
sql/ha_heap.h:
Update to 'newer' table handler interface
sql/ha_innodb.cc:
Update to 'newer' table handler interface
- Update innobase_create_handler() to new interface
- Removed HA_NOT_EXACT_COUNT (not needed)
- Renamed HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- Prefixed base status variables with 'stats'
- Use table column bitmaps instead of ha_get_bit_in_read_set()
- Added ::reset(), with code from ::extra(HA_EXTRA_RESET)
- Removed HA_EXTRA_RETRIVE_ALL_COLS and HA_EXTRA_RETRIEVE_PRIMARY_KEY as
the table->read_set and table->write_set bitmaps now are accurate
sql/ha_innodb.h:
Update to 'newer' table handler interface
- table_flags are now ulonglong
- Added reset() method
- Removed not needed ha_retrieve_all_cols() and ha_retrieve_all_pk() columns.
- Made build_template() a class function to be able to easier access class variables
sql/ha_myisam.cc:
Update to 'newer' table handler interface
sql/ha_myisam.h:
Update to 'newer' table handler interface
sql/ha_myisammrg.cc:
Update to 'newer' table handler interface
sql/ha_myisammrg.h:
Update to 'newer' table handler interface
sql/ha_ndbcluster.cc:
Update to 'newer' table handler interface
Fixed use_blob_value() to be accurate
In ::complemented_read() we have to check both the read and write bitmap as the old code did mark all changed columns also in the read map
Correct dumping of field data with DBUG_DUMP
Prefix addresses in DBUG_PRINT with 0x
Fixed usage of not initialized memory
Update to use field->flags & FIELD_IN_ADD_INDEX instead of field->add_index.
sql/ha_ndbcluster.h:
Update to 'newer' table handler interface
sql/ha_ndbcluster_binlog.cc:
Mark usage of all columns in ndbcluster binlog tables
false -> FALSE, true -> TRUE
Use table->s->all_set instead of creating a temporary bitmap.
sql/ha_partition.cc:
Update to 'newer' table handler interface
Added memroot to initialise_partitions() and related functions to get faster memory allocation.
partition_create_handler() is now responsible for initialisation of the partition object
Some trivial optimizations and indentation fixes
Ensure that table_flags() are up to date
Removed documentation for removed HA_EXTRA flags
Fixed 'strange' usage of m_file[i] in new_handlers_from_part_info()that worked in current code 'by chance'
sql/ha_partition.h:
Update to 'newer' table handler interface
sql/handler.cc:
create_xxx handler now takes MEMROOT as an argument to simplify memory allocation.
Much simpler get_new_handler()
(Initialization of the object is now handled by the create method for the engine)
Moved all allocation of bitmap handling to the TABLE object (in table.cc)
Added column_bitmaps_signal() to signal column usage changes.
Changed binlog_log_row() to use the exiusting all_set bitmap in the table object.
Added ha_reset() function to test that the file object is ok at end of statement and call handler::reset()
Added use_hidden_primary_key() to signal handler that we we are going to read and update + delete the row and the handler should thus remember the position for the row
sql/handler.h:
Added HA_NO_TRANSACTIONS, HA_PARTIAL_COLUMN_READ, HA_REQUIRES_KEY_COLUMNS_FOR_DELETE,HA_PRIMARY_KEY_REQUIRED_FOR_DELETE and HA_HAS_RECORDS
Removed HA_NOT_EXACT_COUNT, HA_READ_RND_SAME
HA_DUPP_POS -> HA_DUPLICATE_POS
HA_NOT_EXACT_COUNT replaced by HA_STATS_RECORDS_IS_EXACT, HA_HAS_RECORDS and records()
HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION
Added future row type 'ROW_TYPE_PAGES'
Added MEM_ROOT to handlerton 'create' function
Added ha_statistics, a structure for all status variable in the base handler class.
Moved all status variables in the handler class into a stats structs to improve readability.
ha_table_flags() is now a cached (not virtual) version of table_flags()
reset() doesn't anymore call extra(HA_EXTRA_RESET) but is a function of it's own.
Renamed dupp_ref to dup_ref
Renamed not used handler::sortkey
Moved read_set and write_set to TABLE structure
handler::init() function added for cacheing of virtual constants from engine.
sql/item.cc:
Added register_field_in_read_map() for marking used columns in expression.
This is used by filesort() for creating an optimal column bitmap while retrieving columns for sorting.
Initalize value.cs_info.character_set_client to fix core dump bug with --debug
set_query_id -> mark_used_columns
Mark used columns in read_set OR write_set.
sql/item.h:
Removed reset_query_id_processor() as it's not needed anymore.
Added register_field_in_read_map()
Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
sql/item_cmpfunc.cc:
Temporary mark used columns to be read/writable
Update Item::walk to new interface
sql/item_cmpfunc.h:
Added extra argument to Item::walk() to indicate if we should also traverse sub queries.
sql/item_func.cc:
Update Item::walk() to new interface
table_flags() -> ha_table_flags()
sql/item_func.h:
Update Item::walk() to new interface
sql/item_row.cc:
Update Item::walk() to new interface
sql/item_row.h:
Update Item::walk() to new interface
sql/item_strfunc.h:
Update Item::walk() to new interface
sql/item_subselect.cc:
Added Item_subselect::walk()
(It was a bug it was missing before. Not sure what kind of bugs this could have caused)
sql/item_subselect.h:
Update Item::walk() to new interface
sql/item_sum.cc:
Update Item::walk() to new interface
Updates for new handler interace
sql/item_sum.h:
Update Item::walk() to new interface
sql/key.cc:
Updates for new handler interace
sql/log.cc:
Mark all columns used for log tables
Split options flag
Ensured that second argument to trans_register_ha is a bool
sql/log_event.cc:
Fixed comments to be withing 79 characters
Use OPTION_KEEP_LOG instead of OPTION_STATUS_NO_TRANS_UPDATE to remove wrong warnings
Updates for new handler interface
Use 0x%lx instead of %p (portability problem)
sql/mysql_priv.h:
Added OPTION_KEEP_LOG to indicate that we should replicate the binlog even on rollback
Removed not used 'conds' argument to setup_tables
sql/mysqld.cc:
Indentation fixes and removed old comment
sql/opt_range.cc:
Update to new handler and bitmap interface.
Fixed calls to cp_buffer_from_ref() and walk() (new argument).
Create new temporary bitmaps for ror scans.
(Needed because of handler changes and to get more accurate column bitmaps than before)
Remove not needed file->ha_reset() call before file->close().
Some trivial optimization and indentation fixes.
Use Field->part_of_key_not_clustered() to check if field is part of a key, instead of looping over all key parts.
Added flag 'in_ror_merged_scan' to allow ::get_next() to know that we need a special column bitmap to only fetch pointer to record.
This is needed because ror scan uses the same TABLE object but different file objects, which creates problem for the column bitmap handling.
(This is a temporary solution. A better one would be to allocate an own TABLE object for ROR scans)
Optimized bitmap handling in ror scans:
- Start bitmap at position 0, not 1
- Use same bitmap size as in TABLE
- Use table->read_set and table->write_set to create column bitmaps instead of looping over all fields in table
sql/opt_range.h:
Added 'in_ror_merged_scan' to indicate if we are doing a ROR scan
Added temporary column bitmaps used in ROR scans
sql/opt_sum.cc:
Added get_ext_record_count() which is used in COUNT() optimization if handler has HA_HAS_RECORDS
Note that we don't call this if handler has HA_STATS_RECORDS_IS_EXACT set.
sql/protocol.cc:
We need to mark columns as readable in ::store() as we sometimes return default value for fields to the user
sql/records.cc:
Updates for new handler interface
sql/set_var.cc:
Handle splitting OPTION_STATUS_NO_TRANS_UPDATE to two flags
sql/share/errmsg.txt:
Fixed wrong
sql/sp.cc:
Mark that we are using all columns for the proc table
Update call to setup_tables() to use new prototype
sql/sp_head.cc:
Removed QQ comment
sql/spatial.cc:
Removed wrong QQ comment
sql/sql_acl.cc:
Mark that we need all columns for acl tables
Supply memroot to some 'new' calls.
Indentation fixes
sql/sql_base.cc:
set_query_id removed
Ensure we call ha_reset() at end of each statement
Mark read columns in read_set and changed columns in write_set (Before all columns was marked in read set)
Fixed marking of some columns that was not proplerly marked before
Maintain in TABLE->merge_keys set of all keys that are used in some way
Removed not used 'conds' argument from setup_tables()
Remove not used setting of 'dupp_field' in insert_fields()
Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after crash)
sql/sql_bitmap.h:
Added is_overlapping()
sql/sql_class.cc:
Slow_logs was not properly initialized, which could maybe cause extra/lost entries in slow log.
set_query_id -> mark_used_columns
Simpler variable usage in pack_row() (cleanup)
Moved some variable declartion at start of function for better code readability
sql/sql_class.h:
Added enum_mark_columns
Updated comments
Renamed dupp_field -> dup_field
Added virtual function 'can_rollback_data()' to select_insert() to be used in CREATE ... SELECT to optimize use of OPTION_STATUS_NO_TRANS_UPDATE.
(This fixes a bug in CREATE ... SELECT where we did give wrong warnings when using non transacational tables)
sql/sql_delete.cc:
Updates to new handler interface
Call table->mark_columns_needed_for_delete() to allow us to put additional columns in column usage maps if handler so requires.
Call table->prepare_for_position() to tell handler that we are going to call ha_position().
Removed call to free_io_cache(). (io_cache is now removed in ha_reset()).
Fixed calls to setup_tables()
sql/sql_do.cc:
Update call to setup_fields()
sql/sql_handler.cc:
Tell handler tables to always read all columns.
Use temporary column map when storing value in field for later index usage
sql/sql_help.cc:
Makr all used fields to be read
Update call to setup_fields()
sql/sql_insert.cc:
Tell handler we are going to update the auto_increment column
dupp_field -> dup_field
Set column usage bits for timestamp field.
Call table->mark_columns_needed_for_insert() and table->mark_auto_increment_column()
Removed not used argument from mysql_prepare_insert_check_table().
If we get an duplicate row on insert, change column map to read and write all columns while retrying the operatation.
This is required by the definition of REPLACE and also ensures that fields that are only part of UPDATE are properly handled.
This fixed a bug in NDB and REPLACE where REPLACE wrongly copied some column values from the replaced row.
Setup new bitmaps for delayed insert rows
Remove reseting of next_number_fields as it will be reset on next call to handler_insert()
Fixed usage of thd->options and OPTION_STATUS_NO_TRANS_UPDATE.
The issue was that one should not to reset this flag as it may be set by a previous statement.
The way it was now used caused us to loose some warnings and get other wrong warnings when using non transactional tables mixed with transactional.
I fixed it by introducing 'select_insert::can_rollback_data' to inform send_error() that the given statement can be rolled back (which in case of CREATE TABLE can always be done)
Don't close tables created with CREATE ... SELECT but keep them in the table cache.
Moved out MY_HOOKS from inside function (better readability)
sql/sql_load.cc:
Update to use new handler and column marking interface
Update using setup_tables()
sql/sql_olap.cc:
Update calls to setup_tables
Use enums instead of constants to setup_fields()
sql/sql_parse.cc:
Handle OPTION_KEEP_LOG:
- Set it on CREATE TEMPORARY TABLE / DROP TABLE
- Reset it when OPTION_STATUS_NO_TRANS_UPDATE is reset
- Don't set it for CREATE ... SELECT (this is handled in select_create class)
Remove reseting of OPTION_STATUS_NO_TRANS_UPDATE in begin_trans() as this should already be reset.
If in autocommit mode, reset OPTION_KEEP_LOG and OPTION_STATUS_NO_TRANS_UPDATE to not give warnings in future commands
sql/sql_partition.cc:
Update walk() usage
Trivial indentation fixes
sql/sql_plugin.cc:
Mark all columns as used for plugins
sql/sql_prepare.cc:
Added assert to find out hidden bugs in character_set_client (got an error in debug binary when this not set correctly)
Updates for new handler interface
Update calls to setup_fields()
sql/sql_repl.cc:
Indentation fixes
sql/sql_select.cc:
Update call to setup_tables() and setup_fields()
Remove some old disabled code
Update to new hadler interface
Indentation cleanups
Added column bitmaps for temporary tables.
Remove updating of the removed slots in the Field class
Added TABLE argument to cp_buffer_from_ref() (To be able to install temporary column maps)
For internal temporary tables, use handler::write_row(), handler::delete_row() and handler::update_row() instead of handler::ha_xxxx() for faster execution.
sql/sql_select.h:
Indentaition fixes.
Install temporary column usage maps when needed
Added TABLE element to cp_buffer_from_ref()
sql/sql_show.cc:
Update to new handler interface
Mark all columns used for internal tables.
Style fixes.
Added support for 'future' ROW_TYPE_PAGES.
Don't allocate TMP_TABLE_PARAM with calloc. The 'init()' function will initialize the structure properly.
sql/sql_table.cc:
Update to new handler interface
Simple my_snprintf -> strmake()
Changed some constants to defines
Don't test for NULL in primary key (as we a couple of line above force the PRIMARY KEY to be NOT NULL)
Change field->add_index to use field->flags & FIELD_IN_ADD_INDEX
Mark all columns as used for ALTER TABLE
Style fixes
Update call to filesort()
sql/sql_trigger.h:
Added friend functions to be able to test if triggers exists for table we are going to insert/update or delete in.
sql/sql_udf.cc:
Mark all columns as used for udf system table.
sql/sql_union.cc:
Update call to walk()
Update to new handler interface
sql/sql_update.cc:
Remove query_id argument from compare_record()
Use column bitmaps instead of query_id.
We can't use compare_records() to skip updates for handlers that returns a partial column set and the read_set doesn't cover all columns in the write set, because compare_record() can't in this case know if a not read column changed value.
Update call to setup_fields()
Using separate column read and write sets allows for easier checking of timestamp field was set by statement.
Removed call to free_io_cache() as this is now done in ha_reset()
Call table->mark_columns_needed_for_update() and table->prepare_for_position()
Style fixes
sql/sql_view.cc:
Style fixes
sql/table.cc:
Remove implicitely include 'errno.h'
Remove code for building normalized path, as this is now identical to 'path'
Remove field->fieldnr
Added update of field->part_of_key_not_clustered()
Create column bitmaps in TABLE and TABLE_SHARE
Don't setup a temporary MEM_ROOT object as a thread specific variable for the handler. Instead we send the to-be-used MEMROOT to get_new_handler()
Update to new handler interface
Update call to walk()
Added new functions:
- st_table::clear_column_bitmaps()
- st_table::prepare_for_position()
- st_table::mark_columns_used_by_index()
- st_table::restore_column_maps_after_mark_index()
- st_table::mark_columns_used_by_index_no_reset()
- st_table::mark_auto_increment_column()
- st_table::mark_columns_needed_for_delete()
- st_table::mark_columns_needed_for_update()
- st_table::mark_columns_needed_for_insert()
sql/table.h:
Moved column usage bitmaps from handler to TABLE
Added to TABLE_SHARE all_set and column_bitmap_size
Added to TABLE merge_keys, bitmap_init_values, def_read_set, def_write_set, tmp_set, read_set and write_set.
Declared all new table column bitmap functions
Added TABLE functions column_bitmaps_set(), column_bitmaps_set_no_signal(), use_all_columns() and default_column_bitmaps()
Added functions: tmp_use_all_columns() and tmp_restore_column_map() to temporarly switch column bitmaps
Added functions: dbug_tmp_use_all_columns() and dbug_tmp_restore_column_map() to temporarly switch column bitmaps to avoid asserts in Field::store() and Field::val().
sql/tztime.cc:
Mark all columns as used for timezone tables
storage/archive/ha_archive.cc:
Update to new handler interface
storage/archive/ha_archive.h:
Update to new handler interface
storage/blackhole/ha_blackhole.cc:
Update to new handler interface
storage/blackhole/ha_blackhole.h:
Update to new handler interface
removed not needed flag HA_DUPP_POS
storage/csv/ha_tina.cc:
Update to new handler interface
storage/csv/ha_tina.h:
Update to new handler interface
storage/example/ha_example.cc:
Update to new handler interface
storage/example/ha_example.h:
Update to new handler interface
storage/heap/hp_extra.c:
Added heap_reset() (Required by new handler interface)
storage/heap/hp_test2.c:
Use heap_reset()
storage/myisam/ft_boolean_search.c:
Fixed compiler warning
storage/myisam/mi_extra.c:
Added mi_reset() (Required by new handler interface)
storage/myisam/mi_search.c:
Fixed DBUG_PRINT messages to use 0x%lx instead of %lx
storage/myisam/mi_test2.c:
Use mi_reset()
storage/myisam/myisampack.c:
Use mi_reset()
storage/myisammrg/myrg_extra.c:
Added myrg_reset() (Required by new handler interface)
unittest/mysys/base64.t.c:
Include my_global.h
Don't include implictely include file 'stdlib.h'
2006-06-04 17:52:22 +02:00
|
|
|
{
|
|
|
|
if ((*order->item)->walk(processor, walk_subquery, argument))
|
|
|
|
return 1;
|
|
|
|
}
|
2010-06-10 22:45:22 +02:00
|
|
|
for (order= lex->group_list.first ; order; order= order->next)
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
BitKeeper/etc/ignore:
added mysys/test_bitmap
include/base64.h:
Removed my_global.h, as this must be included first in any program
include/heap.h:
Added heap_reset() (Required by new handler interface)
include/my_base.h:
Removed HA_EXTRA_RESET. MySQL will now call ::reset() instead of ::extra(HA_EXTRA_RESET).
HA_EXTRA_RETRIVE_ALL_COLS and HA_EXTRA_RETRIVE_PRIMARY key are deleted as the column bitmaps makes these unnecessary
include/my_bitmap.h:
Remove my_pthread.h (should be included at upper level)
Introduced my_bitmap_map typedef to make it the bitmap handling more like a black box
Added bitmap_is_overlapping(), bitmap_test_and_clear(), bitmap_copy() and bitmap_cmp()
Made bitmap_set_bit(), bitmap_flip_bit(), bitmap_clear_bit() return void
include/myisam.h:
Added mi_reset() (Required by new handler interface)
include/myisammrg.h:
Added myrg_reset() (Required by new handler interface)
include/mysql_com.h:
Added flag FIELD_IN_ADD_INDEX to be able to remove Field->add_index
mysql-test/extra/binlog_tests/mix_innodb_myisam_binlog.test:
Added testing of CREATE ... SELECT in a mixed environment
(This found some bugs that Mats is going to fix shortly)
mysql-test/install_test_db.sh:
Simplify ldata usage
Added --tmpdir=. option to mysqld bootstrap (Removed some warnings when TMPDIR was wrongly set)
mysql-test/mysql-test-run.pl:
Added --tmpdir=. to bootstrap
mysql-test/mysql-test-run.sh:
Use copy instead of INSTALL_DB for master and slave databases.
(Speeds up startup time a lot!)
Remove snapshot directories at startup (removes some strange warnings)
mysql-test/r/binlog_row_mix_innodb_myisam.result:
Added testing of CREATE ... SELECT in a mixed environment
(This found some bugs that Mats is going to fix shortly)
mysql-test/r/binlog_stm_mix_innodb_myisam.result:
Added testing of CREATE ... SELECT in a mixed environment
mysql-test/r/create.result:
Some extra tests of warnings and number of tables opened by CREATE ... SELECT
mysql-test/r/federated.result:
Drop some left over tables
Added testing of multiple table update and multiple table delete (with and without keys)
mysql-test/r/func_gconcat.result:
Enable some disabled tests (converted them slightly to be predictable)
mysql-test/r/func_time.result:
Added drop of test function
mysql-test/r/innodb_mysql.result:
Added tests for CREATE ... SELECT
mysql-test/r/insert.result:
More tests
Added testing of duplicate columns in insert
mysql-test/r/loaddata.result:
Added testing LOAD DATA ... SET ...
mysql-test/r/multi_update.result:
Test multi updates and deletes using primary key and without
mysql-test/r/ndb_index_unique.result:
Better error message
mysql-test/r/ndb_replace.result:
New correct result after fixing REPLACE handling with NDB
mysql-test/r/rpl_ddl.result:
Now we don't get these (wrong) warnings anymore
mysql-test/r/view_grant.result:
Drop used views
mysql-test/t/create.test:
Some extra tests of warnings and number of tables opened by CREATE ... SELECT
mysql-test/t/federated.test:
Drop some left over tables
Added testing of multiple table update and multiple table delete (with and without keys)
mysql-test/t/func_gconcat.test:
Enable some disabled tests (converted them slightly to be predictable)
mysql-test/t/func_time.test:
Added drop of test function
mysql-test/t/innodb_mysql.test:
Added tests for CREATE ... SELECT
mysql-test/t/insert.test:
More tests
Added testing of duplicate columns in insert
mysql-test/t/loaddata.test:
Added testing LOAD DATA ... SET ...
mysql-test/t/multi_update.test:
Test multi updates and deletes using primary key and without
mysql-test/t/view_grant.test:
Drop used views
mysql-test/valgrind.supp:
Added supression of not needed warnings when printing stack trace
mysys/base64.c:
Include my_global.h first
mysys/my_bitmap.c:
Added bitmap_is_overlapping(), bitmap_test_and_clear() and bitmap_copy()
Changed logic of bitmap handling to be a bit more efficent (Did this together with Mikael Ronström)
Now the 'extra, not used bits' in the bitmap are assumed to have a 'random value' and the bitmap functions are free to change them whenever needed.
Changed how mutex is allocated to make 'bitmap_free()' function simpler.
mysys/thr_lock.c:
Added 0x before thread pointers (for easier comparison of DBUG traces)
sql/event.cc:
Ensure 'use_all_columns()' is used for event tables
Don't print warning that event table is damaged if it doesn't exists.
sql/field.cc:
Added ASSERT_COLUMN_MARKED_FOR_WRITE in all store() methods and ASSERT_COLUMN_MARKED_FOR_READ in all val() methods to catch wrong setting if table->read_set and table->write_set
(Rest of changes are only indentation cleanups)
sql/field.h:
Removed Field->query_id (replaced by table->read_set and table->write_set)
Removed Field->fieldnr (use Field->field_index instead)
Removed Field->add_index (Use Field->flags instead)
Add Field->part_of_key_not_clustered (for usage in opt_range.cc)
sql/filesort.cc:
Added paramater sort_postion to filesort() to force sorting by position instead of storing all fields in the result set.
This allowed me to remove checking of sql_command.
Create a temporary column bitmap for fields that are used by the sorting process.
Use column bitmaps instead of query_id
sql/ha_berkeley.cc:
Update to 'newer' table handler interface
sql/ha_berkeley.h:
Update to 'newer' table handler interface
sql/ha_federated.cc:
Update to 'newer' table handler interface
Only read columns that are needed from remote server.
In case of eq ranges, don't generate two conditions in the WHERE clause
(this can still be optimized, but would require a bigger code change)
Use 'simpler to use' XXXX_LEN' macros
A bit simpler logic in ::write_row() when creating statements.
In update, only include test of fields actually read.
(This greatly simplifies the queries sent by the federated engine)
Similar changes done for delete_row()
sql/ha_federated.h:
Update to 'newer' table handler interface
Changed XXX_LEN macros to use sizeof(...)-1, to simplify usage in ha_federated.cc
Added HA_PRIMARY_KEY_REQUIRED_FOR_DELETE to tell MySQL to read all primary key columns in case of DELETE
sql/ha_heap.cc:
Update to 'newer' table handler interface
sql/ha_heap.h:
Update to 'newer' table handler interface
sql/ha_innodb.cc:
Update to 'newer' table handler interface
- Update innobase_create_handler() to new interface
- Removed HA_NOT_EXACT_COUNT (not needed)
- Renamed HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- Prefixed base status variables with 'stats'
- Use table column bitmaps instead of ha_get_bit_in_read_set()
- Added ::reset(), with code from ::extra(HA_EXTRA_RESET)
- Removed HA_EXTRA_RETRIVE_ALL_COLS and HA_EXTRA_RETRIEVE_PRIMARY_KEY as
the table->read_set and table->write_set bitmaps now are accurate
sql/ha_innodb.h:
Update to 'newer' table handler interface
- table_flags are now ulonglong
- Added reset() method
- Removed not needed ha_retrieve_all_cols() and ha_retrieve_all_pk() columns.
- Made build_template() a class function to be able to easier access class variables
sql/ha_myisam.cc:
Update to 'newer' table handler interface
sql/ha_myisam.h:
Update to 'newer' table handler interface
sql/ha_myisammrg.cc:
Update to 'newer' table handler interface
sql/ha_myisammrg.h:
Update to 'newer' table handler interface
sql/ha_ndbcluster.cc:
Update to 'newer' table handler interface
Fixed use_blob_value() to be accurate
In ::complemented_read() we have to check both the read and write bitmap as the old code did mark all changed columns also in the read map
Correct dumping of field data with DBUG_DUMP
Prefix addresses in DBUG_PRINT with 0x
Fixed usage of not initialized memory
Update to use field->flags & FIELD_IN_ADD_INDEX instead of field->add_index.
sql/ha_ndbcluster.h:
Update to 'newer' table handler interface
sql/ha_ndbcluster_binlog.cc:
Mark usage of all columns in ndbcluster binlog tables
false -> FALSE, true -> TRUE
Use table->s->all_set instead of creating a temporary bitmap.
sql/ha_partition.cc:
Update to 'newer' table handler interface
Added memroot to initialise_partitions() and related functions to get faster memory allocation.
partition_create_handler() is now responsible for initialisation of the partition object
Some trivial optimizations and indentation fixes
Ensure that table_flags() are up to date
Removed documentation for removed HA_EXTRA flags
Fixed 'strange' usage of m_file[i] in new_handlers_from_part_info()that worked in current code 'by chance'
sql/ha_partition.h:
Update to 'newer' table handler interface
sql/handler.cc:
create_xxx handler now takes MEMROOT as an argument to simplify memory allocation.
Much simpler get_new_handler()
(Initialization of the object is now handled by the create method for the engine)
Moved all allocation of bitmap handling to the TABLE object (in table.cc)
Added column_bitmaps_signal() to signal column usage changes.
Changed binlog_log_row() to use the exiusting all_set bitmap in the table object.
Added ha_reset() function to test that the file object is ok at end of statement and call handler::reset()
Added use_hidden_primary_key() to signal handler that we we are going to read and update + delete the row and the handler should thus remember the position for the row
sql/handler.h:
Added HA_NO_TRANSACTIONS, HA_PARTIAL_COLUMN_READ, HA_REQUIRES_KEY_COLUMNS_FOR_DELETE,HA_PRIMARY_KEY_REQUIRED_FOR_DELETE and HA_HAS_RECORDS
Removed HA_NOT_EXACT_COUNT, HA_READ_RND_SAME
HA_DUPP_POS -> HA_DUPLICATE_POS
HA_NOT_EXACT_COUNT replaced by HA_STATS_RECORDS_IS_EXACT, HA_HAS_RECORDS and records()
HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION
Added future row type 'ROW_TYPE_PAGES'
Added MEM_ROOT to handlerton 'create' function
Added ha_statistics, a structure for all status variable in the base handler class.
Moved all status variables in the handler class into a stats structs to improve readability.
ha_table_flags() is now a cached (not virtual) version of table_flags()
reset() doesn't anymore call extra(HA_EXTRA_RESET) but is a function of it's own.
Renamed dupp_ref to dup_ref
Renamed not used handler::sortkey
Moved read_set and write_set to TABLE structure
handler::init() function added for cacheing of virtual constants from engine.
sql/item.cc:
Added register_field_in_read_map() for marking used columns in expression.
This is used by filesort() for creating an optimal column bitmap while retrieving columns for sorting.
Initalize value.cs_info.character_set_client to fix core dump bug with --debug
set_query_id -> mark_used_columns
Mark used columns in read_set OR write_set.
sql/item.h:
Removed reset_query_id_processor() as it's not needed anymore.
Added register_field_in_read_map()
Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
sql/item_cmpfunc.cc:
Temporary mark used columns to be read/writable
Update Item::walk to new interface
sql/item_cmpfunc.h:
Added extra argument to Item::walk() to indicate if we should also traverse sub queries.
sql/item_func.cc:
Update Item::walk() to new interface
table_flags() -> ha_table_flags()
sql/item_func.h:
Update Item::walk() to new interface
sql/item_row.cc:
Update Item::walk() to new interface
sql/item_row.h:
Update Item::walk() to new interface
sql/item_strfunc.h:
Update Item::walk() to new interface
sql/item_subselect.cc:
Added Item_subselect::walk()
(It was a bug it was missing before. Not sure what kind of bugs this could have caused)
sql/item_subselect.h:
Update Item::walk() to new interface
sql/item_sum.cc:
Update Item::walk() to new interface
Updates for new handler interace
sql/item_sum.h:
Update Item::walk() to new interface
sql/key.cc:
Updates for new handler interace
sql/log.cc:
Mark all columns used for log tables
Split options flag
Ensured that second argument to trans_register_ha is a bool
sql/log_event.cc:
Fixed comments to be withing 79 characters
Use OPTION_KEEP_LOG instead of OPTION_STATUS_NO_TRANS_UPDATE to remove wrong warnings
Updates for new handler interface
Use 0x%lx instead of %p (portability problem)
sql/mysql_priv.h:
Added OPTION_KEEP_LOG to indicate that we should replicate the binlog even on rollback
Removed not used 'conds' argument to setup_tables
sql/mysqld.cc:
Indentation fixes and removed old comment
sql/opt_range.cc:
Update to new handler and bitmap interface.
Fixed calls to cp_buffer_from_ref() and walk() (new argument).
Create new temporary bitmaps for ror scans.
(Needed because of handler changes and to get more accurate column bitmaps than before)
Remove not needed file->ha_reset() call before file->close().
Some trivial optimization and indentation fixes.
Use Field->part_of_key_not_clustered() to check if field is part of a key, instead of looping over all key parts.
Added flag 'in_ror_merged_scan' to allow ::get_next() to know that we need a special column bitmap to only fetch pointer to record.
This is needed because ror scan uses the same TABLE object but different file objects, which creates problem for the column bitmap handling.
(This is a temporary solution. A better one would be to allocate an own TABLE object for ROR scans)
Optimized bitmap handling in ror scans:
- Start bitmap at position 0, not 1
- Use same bitmap size as in TABLE
- Use table->read_set and table->write_set to create column bitmaps instead of looping over all fields in table
sql/opt_range.h:
Added 'in_ror_merged_scan' to indicate if we are doing a ROR scan
Added temporary column bitmaps used in ROR scans
sql/opt_sum.cc:
Added get_ext_record_count() which is used in COUNT() optimization if handler has HA_HAS_RECORDS
Note that we don't call this if handler has HA_STATS_RECORDS_IS_EXACT set.
sql/protocol.cc:
We need to mark columns as readable in ::store() as we sometimes return default value for fields to the user
sql/records.cc:
Updates for new handler interface
sql/set_var.cc:
Handle splitting OPTION_STATUS_NO_TRANS_UPDATE to two flags
sql/share/errmsg.txt:
Fixed wrong
sql/sp.cc:
Mark that we are using all columns for the proc table
Update call to setup_tables() to use new prototype
sql/sp_head.cc:
Removed QQ comment
sql/spatial.cc:
Removed wrong QQ comment
sql/sql_acl.cc:
Mark that we need all columns for acl tables
Supply memroot to some 'new' calls.
Indentation fixes
sql/sql_base.cc:
set_query_id removed
Ensure we call ha_reset() at end of each statement
Mark read columns in read_set and changed columns in write_set (Before all columns was marked in read set)
Fixed marking of some columns that was not proplerly marked before
Maintain in TABLE->merge_keys set of all keys that are used in some way
Removed not used 'conds' argument from setup_tables()
Remove not used setting of 'dupp_field' in insert_fields()
Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after crash)
sql/sql_bitmap.h:
Added is_overlapping()
sql/sql_class.cc:
Slow_logs was not properly initialized, which could maybe cause extra/lost entries in slow log.
set_query_id -> mark_used_columns
Simpler variable usage in pack_row() (cleanup)
Moved some variable declartion at start of function for better code readability
sql/sql_class.h:
Added enum_mark_columns
Updated comments
Renamed dupp_field -> dup_field
Added virtual function 'can_rollback_data()' to select_insert() to be used in CREATE ... SELECT to optimize use of OPTION_STATUS_NO_TRANS_UPDATE.
(This fixes a bug in CREATE ... SELECT where we did give wrong warnings when using non transacational tables)
sql/sql_delete.cc:
Updates to new handler interface
Call table->mark_columns_needed_for_delete() to allow us to put additional columns in column usage maps if handler so requires.
Call table->prepare_for_position() to tell handler that we are going to call ha_position().
Removed call to free_io_cache(). (io_cache is now removed in ha_reset()).
Fixed calls to setup_tables()
sql/sql_do.cc:
Update call to setup_fields()
sql/sql_handler.cc:
Tell handler tables to always read all columns.
Use temporary column map when storing value in field for later index usage
sql/sql_help.cc:
Makr all used fields to be read
Update call to setup_fields()
sql/sql_insert.cc:
Tell handler we are going to update the auto_increment column
dupp_field -> dup_field
Set column usage bits for timestamp field.
Call table->mark_columns_needed_for_insert() and table->mark_auto_increment_column()
Removed not used argument from mysql_prepare_insert_check_table().
If we get an duplicate row on insert, change column map to read and write all columns while retrying the operatation.
This is required by the definition of REPLACE and also ensures that fields that are only part of UPDATE are properly handled.
This fixed a bug in NDB and REPLACE where REPLACE wrongly copied some column values from the replaced row.
Setup new bitmaps for delayed insert rows
Remove reseting of next_number_fields as it will be reset on next call to handler_insert()
Fixed usage of thd->options and OPTION_STATUS_NO_TRANS_UPDATE.
The issue was that one should not to reset this flag as it may be set by a previous statement.
The way it was now used caused us to loose some warnings and get other wrong warnings when using non transactional tables mixed with transactional.
I fixed it by introducing 'select_insert::can_rollback_data' to inform send_error() that the given statement can be rolled back (which in case of CREATE TABLE can always be done)
Don't close tables created with CREATE ... SELECT but keep them in the table cache.
Moved out MY_HOOKS from inside function (better readability)
sql/sql_load.cc:
Update to use new handler and column marking interface
Update using setup_tables()
sql/sql_olap.cc:
Update calls to setup_tables
Use enums instead of constants to setup_fields()
sql/sql_parse.cc:
Handle OPTION_KEEP_LOG:
- Set it on CREATE TEMPORARY TABLE / DROP TABLE
- Reset it when OPTION_STATUS_NO_TRANS_UPDATE is reset
- Don't set it for CREATE ... SELECT (this is handled in select_create class)
Remove reseting of OPTION_STATUS_NO_TRANS_UPDATE in begin_trans() as this should already be reset.
If in autocommit mode, reset OPTION_KEEP_LOG and OPTION_STATUS_NO_TRANS_UPDATE to not give warnings in future commands
sql/sql_partition.cc:
Update walk() usage
Trivial indentation fixes
sql/sql_plugin.cc:
Mark all columns as used for plugins
sql/sql_prepare.cc:
Added assert to find out hidden bugs in character_set_client (got an error in debug binary when this not set correctly)
Updates for new handler interface
Update calls to setup_fields()
sql/sql_repl.cc:
Indentation fixes
sql/sql_select.cc:
Update call to setup_tables() and setup_fields()
Remove some old disabled code
Update to new hadler interface
Indentation cleanups
Added column bitmaps for temporary tables.
Remove updating of the removed slots in the Field class
Added TABLE argument to cp_buffer_from_ref() (To be able to install temporary column maps)
For internal temporary tables, use handler::write_row(), handler::delete_row() and handler::update_row() instead of handler::ha_xxxx() for faster execution.
sql/sql_select.h:
Indentaition fixes.
Install temporary column usage maps when needed
Added TABLE element to cp_buffer_from_ref()
sql/sql_show.cc:
Update to new handler interface
Mark all columns used for internal tables.
Style fixes.
Added support for 'future' ROW_TYPE_PAGES.
Don't allocate TMP_TABLE_PARAM with calloc. The 'init()' function will initialize the structure properly.
sql/sql_table.cc:
Update to new handler interface
Simple my_snprintf -> strmake()
Changed some constants to defines
Don't test for NULL in primary key (as we a couple of line above force the PRIMARY KEY to be NOT NULL)
Change field->add_index to use field->flags & FIELD_IN_ADD_INDEX
Mark all columns as used for ALTER TABLE
Style fixes
Update call to filesort()
sql/sql_trigger.h:
Added friend functions to be able to test if triggers exists for table we are going to insert/update or delete in.
sql/sql_udf.cc:
Mark all columns as used for udf system table.
sql/sql_union.cc:
Update call to walk()
Update to new handler interface
sql/sql_update.cc:
Remove query_id argument from compare_record()
Use column bitmaps instead of query_id.
We can't use compare_records() to skip updates for handlers that returns a partial column set and the read_set doesn't cover all columns in the write set, because compare_record() can't in this case know if a not read column changed value.
Update call to setup_fields()
Using separate column read and write sets allows for easier checking of timestamp field was set by statement.
Removed call to free_io_cache() as this is now done in ha_reset()
Call table->mark_columns_needed_for_update() and table->prepare_for_position()
Style fixes
sql/sql_view.cc:
Style fixes
sql/table.cc:
Remove implicitely include 'errno.h'
Remove code for building normalized path, as this is now identical to 'path'
Remove field->fieldnr
Added update of field->part_of_key_not_clustered()
Create column bitmaps in TABLE and TABLE_SHARE
Don't setup a temporary MEM_ROOT object as a thread specific variable for the handler. Instead we send the to-be-used MEMROOT to get_new_handler()
Update to new handler interface
Update call to walk()
Added new functions:
- st_table::clear_column_bitmaps()
- st_table::prepare_for_position()
- st_table::mark_columns_used_by_index()
- st_table::restore_column_maps_after_mark_index()
- st_table::mark_columns_used_by_index_no_reset()
- st_table::mark_auto_increment_column()
- st_table::mark_columns_needed_for_delete()
- st_table::mark_columns_needed_for_update()
- st_table::mark_columns_needed_for_insert()
sql/table.h:
Moved column usage bitmaps from handler to TABLE
Added to TABLE_SHARE all_set and column_bitmap_size
Added to TABLE merge_keys, bitmap_init_values, def_read_set, def_write_set, tmp_set, read_set and write_set.
Declared all new table column bitmap functions
Added TABLE functions column_bitmaps_set(), column_bitmaps_set_no_signal(), use_all_columns() and default_column_bitmaps()
Added functions: tmp_use_all_columns() and tmp_restore_column_map() to temporarly switch column bitmaps
Added functions: dbug_tmp_use_all_columns() and dbug_tmp_restore_column_map() to temporarly switch column bitmaps to avoid asserts in Field::store() and Field::val().
sql/tztime.cc:
Mark all columns as used for timezone tables
storage/archive/ha_archive.cc:
Update to new handler interface
storage/archive/ha_archive.h:
Update to new handler interface
storage/blackhole/ha_blackhole.cc:
Update to new handler interface
storage/blackhole/ha_blackhole.h:
Update to new handler interface
removed not needed flag HA_DUPP_POS
storage/csv/ha_tina.cc:
Update to new handler interface
storage/csv/ha_tina.h:
Update to new handler interface
storage/example/ha_example.cc:
Update to new handler interface
storage/example/ha_example.h:
Update to new handler interface
storage/heap/hp_extra.c:
Added heap_reset() (Required by new handler interface)
storage/heap/hp_test2.c:
Use heap_reset()
storage/myisam/ft_boolean_search.c:
Fixed compiler warning
storage/myisam/mi_extra.c:
Added mi_reset() (Required by new handler interface)
storage/myisam/mi_search.c:
Fixed DBUG_PRINT messages to use 0x%lx instead of %lx
storage/myisam/mi_test2.c:
Use mi_reset()
storage/myisam/myisampack.c:
Use mi_reset()
storage/myisammrg/myrg_extra.c:
Added myrg_reset() (Required by new handler interface)
unittest/mysys/base64.t.c:
Include my_global.h
Don't include implictely include file 'stdlib.h'
2006-06-04 17:52:22 +02:00
|
|
|
{
|
|
|
|
if ((*order->item)->walk(processor, walk_subquery, argument))
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (this->*processor)(argument);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
bool Item_subselect::exec()
|
2003-05-28 15:52:56 +02:00
|
|
|
{
|
2012-10-25 14:50:10 +02:00
|
|
|
subselect_engine *org_engine= engine;
|
|
|
|
|
2011-02-17 13:41:25 +01:00
|
|
|
DBUG_ENTER("Item_subselect::exec");
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(fixed);
|
2004-11-08 00:13:54 +01:00
|
|
|
|
2019-03-21 10:43:17 +01:00
|
|
|
DBUG_EXECUTE_IF("Item_subselect",
|
2019-05-13 16:31:15 +02:00
|
|
|
Item::Print print(this,
|
|
|
|
enum_query_type(QT_TO_SYSTEM_CHARSET |
|
|
|
|
QT_WITHOUT_INTRODUCERS));
|
|
|
|
|
|
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_NOTE,
|
|
|
|
ER_UNKNOWN_ERROR, "DBUG: Item_subselect::exec %.*s",
|
|
|
|
print.length(),print.ptr());
|
|
|
|
);
|
2010-03-09 13:51:56 +01:00
|
|
|
/*
|
|
|
|
Do not execute subselect in case of a fatal error
|
|
|
|
or if the query has been killed.
|
|
|
|
*/
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely(thd->is_error() || thd->killed))
|
2011-02-17 13:41:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2010-03-09 13:51:56 +01:00
|
|
|
|
Fixed following problems:
--Bug#52157 various crashes and assertions with multi-table update, stored function
--Bug#54475 improper error handling causes cascading crashing failures in innodb/ndb
--Bug#57703 create view cause Assertion failed: 0, file .\item_subselect.cc, line 846
--Bug#57352 valgrind warnings when creating view
--Recently discovered problem when a nested materialized derived table is used
before being populated and it leads to incorrect result
We have several modes when we should disable subquery evaluation.
The reasons for disabling are different. It could be
uselessness of the evaluation as in case of 'CREATE VIEW'
or 'PREPARE stmt', or we should disable subquery evaluation
if tables are not locked yet as it happens in bug#54475, or
too early evaluation of subqueries can lead to wrong result
as it happened in Bug#19077.
Main problem is that if subquery items are treated as const
they are evaluated in ::fix_fields(), ::fix_length_and_dec()
of the parental items as a lot of these methods have
Item::val_...() calls inside.
We have to make subqueries non-const to prevent unnecessary
subquery evaluation. At the moment we have different methods
for this. Here is a list of these modes:
1. PREPARE stmt;
We use UNCACHEABLE_PREPARE flag.
It is set during parsing in sql_parse.cc, mysql_new_select() for
each SELECT_LEX object and cleared at the end of PREPARE in
sql_prepare.cc, init_stmt_after_parse(). If this flag is set
subquery becomes non-const and evaluation does not happen.
2. CREATE|ALTER VIEW, SHOW CREATE VIEW, I_S tables which
process FRM files
We use LEX::view_prepare_mode field. We set it before
view preparation and check this flag in
::fix_fields(), ::fix_length_and_dec().
Some bugs are fixed using this approach,
some are not(Bug#57352, Bug#57703). The problem here is
that we have a lot of ::fix_fields(), ::fix_length_and_dec()
where we use Item::val_...() calls for const items.
3. Derived tables with subquery = wrong result(Bug19077)
The reason of this bug is too early subquery evaluation.
It was fixed by adding Item::with_subselect field
The check of this field in appropriate places prevents
const item evaluation if the item have subquery.
The fix for Bug19077 fixes only the problem with
convert_constant_item() function and does not cover
other places(::fix_fields(), ::fix_length_and_dec() again)
where subqueries could be evaluated.
Example:
CREATE TABLE t1 (i INT, j BIGINT);
INSERT INTO t1 VALUES (1, 2), (2, 2), (3, 2);
SELECT * FROM (SELECT MIN(i) FROM t1
WHERE j = SUBSTRING('12', (SELECT * FROM (SELECT MIN(j) FROM t1) t2))) t3;
DROP TABLE t1;
4. Derived tables with subquery where subquery
is evaluated before table locking(Bug#54475, Bug#52157)
Suggested solution is following:
-Introduce new field LEX::context_analysis_only with the following
possible flags:
#define CONTEXT_ANALYSIS_ONLY_PREPARE 1
#define CONTEXT_ANALYSIS_ONLY_VIEW 2
#define CONTEXT_ANALYSIS_ONLY_DERIVED 4
-Set/clean these flags when we perform
context analysis operation
-Item_subselect::const_item() returns
result depending on LEX::context_analysis_only.
If context_analysis_only is set then we return
FALSE that means that subquery is non-const.
As all subquery types are wrapped by Item_subselect
it allow as to make subquery non-const when
it's necessary.
mysql-test/r/derived.result:
test case
mysql-test/r/multi_update.result:
test case
mysql-test/r/view.result:
test case
mysql-test/suite/innodb/r/innodb_multi_update.result:
test case
mysql-test/suite/innodb/t/innodb_multi_update.test:
test case
mysql-test/suite/innodb_plugin/r/innodb_multi_update.result:
test case
mysql-test/suite/innodb_plugin/t/innodb_multi_update.test:
test case
mysql-test/t/derived.test:
test case
mysql-test/t/multi_update.test:
test case
mysql-test/t/view.test:
test case
sql/item.cc:
--removed unnecessary code
sql/item_cmpfunc.cc:
--removed unnecessary checks
--THD::is_context_analysis_only() is replaced with LEX::is_ps_or_view_context_analysis()
sql/item_func.cc:
--refactored context analysis checks
sql/item_row.cc:
--removed unnecessary checks
sql/item_subselect.cc:
--removed unnecessary code
--added DBUG_ASSERT into Item_subselect::exec()
which asserts that subquery execution can not happen
if LEX::context_analysis_only is set, i.e. at context
analysis stage.
--Item_subselect::const_item()
Return FALSE if LEX::context_analysis_only is set.
It prevents subquery evaluation in ::fix_fields &
::fix_length_and_dec at context analysis stage.
sql/item_subselect.h:
--removed unnecessary code
sql/mysql_priv.h:
--Added new set of flags.
sql/sql_class.h:
--removed unnecessary code
sql/sql_derived.cc:
--added LEX::context_analysis_only analysis intialization/cleanup
sql/sql_lex.cc:
--init LEX::context_analysis_only field
sql/sql_lex.h:
--New LEX::context_analysis_only field
sql/sql_parse.cc:
--removed unnecessary code
sql/sql_prepare.cc:
--removed unnecessary code
--added LEX::context_analysis_only analysis intialization/cleanup
sql/sql_select.cc:
--refactored context analysis checks
sql/sql_show.cc:
--added LEX::context_analysis_only analysis intialization/cleanup
sql/sql_view.cc:
--added LEX::context_analysis_only analysis intialization/cleanup
2010-12-14 10:33:03 +01:00
|
|
|
DBUG_ASSERT(!thd->lex->context_analysis_only);
|
2008-07-04 16:02:17 +02:00
|
|
|
/*
|
|
|
|
Simulate a failure in sub-query execution. Used to test e.g.
|
|
|
|
out of memory or query being killed conditions.
|
|
|
|
*/
|
2011-02-17 13:41:25 +01:00
|
|
|
DBUG_EXECUTE_IF("subselect_exec_fail", DBUG_RETURN(true););
|
2004-11-08 00:13:54 +01:00
|
|
|
|
2011-02-17 13:41:25 +01:00
|
|
|
bool res= engine->exec();
|
2004-11-08 00:13:54 +01:00
|
|
|
|
2011-06-21 14:50:07 +02:00
|
|
|
#ifndef DBUG_OFF
|
|
|
|
++exec_counter;
|
|
|
|
#endif
|
2012-10-25 14:50:10 +02:00
|
|
|
if (engine != org_engine)
|
2003-07-07 17:40:19 +02:00
|
|
|
{
|
2012-10-25 14:50:10 +02:00
|
|
|
/*
|
|
|
|
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());
|
2003-07-07 17:40:19 +02:00
|
|
|
}
|
2011-02-17 13:41:25 +01:00
|
|
|
DBUG_RETURN(res);
|
2003-05-28 15:52:56 +02:00
|
|
|
}
|
|
|
|
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2011-07-19 22:19:10 +02:00
|
|
|
void Item_subselect::get_cache_parameters(List<Item> ¶meters)
|
|
|
|
{
|
2013-02-26 00:20:17 +01:00
|
|
|
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
|
|
|
|
};
|
2016-06-26 22:42:48 +02:00
|
|
|
walk(&Item::collect_outer_ref_processor, TRUE, &prm);
|
2011-07-19 22:19:10 +02:00
|
|
|
}
|
|
|
|
|
2011-05-25 17:31:13 +02:00
|
|
|
int Item_in_subselect::optimize(double *out_rows, double *cost)
|
|
|
|
{
|
|
|
|
int res;
|
|
|
|
DBUG_ENTER("Item_in_subselect::optimize");
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(fixed);
|
2011-05-25 17:31:13 +02:00
|
|
|
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 */
|
2016-05-08 22:04:41 +02:00
|
|
|
if (join->group_list_for_estimates)
|
2011-05-25 17:31:13 +02:00
|
|
|
{
|
|
|
|
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);
|
|
|
|
|
|
|
|
}
|
2011-05-30 09:19:40 +02:00
|
|
|
|
|
|
|
|
2010-07-10 12:37:30 +02:00
|
|
|
/**
|
|
|
|
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)
|
|
|
|
{
|
2011-07-19 22:19:10 +02:00
|
|
|
return ((engine->uncacheable() & UNCACHEABLE_DEPENDENT) &&
|
2010-07-10 12:37:30 +02:00
|
|
|
engine->cols() == 1 &&
|
|
|
|
optimizer_flag(thd, OPTIMIZER_SWITCH_SUBQUERY_CACHE) &&
|
|
|
|
!(engine->uncacheable() & (UNCACHEABLE_RAND |
|
2016-05-24 20:29:52 +02:00
|
|
|
UNCACHEABLE_SIDEEFFECT)) &&
|
|
|
|
!with_recursive_reference);
|
2010-07-10 12:37:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
/**
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-10 12:37:30 +02:00
|
|
|
/**
|
|
|
|
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
|
2014-11-14 13:46:21 +01:00
|
|
|
Item_in_optimizer and result of the IN subquery will be scalar in this
|
2010-07-10 12:37:30 +02:00
|
|
|
case.
|
|
|
|
|
|
|
|
@retval TRUE cache is needed
|
|
|
|
@retval FALSE otherwise
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool Item_in_subselect::expr_cache_is_needed(THD *thd)
|
|
|
|
{
|
2011-07-19 22:19:10 +02:00
|
|
|
return (optimizer_flag(thd, OPTIMIZER_SWITCH_SUBQUERY_CACHE) &&
|
2010-07-10 12:37:30 +02:00
|
|
|
!(engine->uncacheable() & (UNCACHEABLE_RAND |
|
2016-05-24 20:29:52 +02:00
|
|
|
UNCACHEABLE_SIDEEFFECT)) &&
|
|
|
|
!with_recursive_reference);
|
2010-07-10 12:37:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-28 14:48:33 +01:00
|
|
|
/*
|
|
|
|
Compute the IN predicate if the left operand's cache changed.
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool Item_in_subselect::exec()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_in_subselect::exec");
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(fixed);
|
2010-01-28 14:48:33 +01:00
|
|
|
/*
|
|
|
|
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.
|
|
|
|
*/
|
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
******
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
2011-11-12 10:29:12 +01:00
|
|
|
if (!left_expr_cache && (test_strategy(SUBS_MATERIALIZATION)))
|
2010-01-28 14:48:33 +01:00
|
|
|
init_left_expr_cache();
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
/*
|
|
|
|
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);
|
2010-01-28 14:48:33 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
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());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-07-04 07:46:28 +02:00
|
|
|
Item::Type Item_subselect::type() const
|
2002-12-19 06:38:33 +01:00
|
|
|
{
|
|
|
|
return SUBSELECT_ITEM;
|
|
|
|
}
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2018-05-08 15:26:26 +02:00
|
|
|
bool Item_subselect::fix_length_and_dec()
|
2002-09-28 17:34:56 +02:00
|
|
|
{
|
2018-05-08 15:26:26 +02:00
|
|
|
if (engine->fix_length_and_dec(0))
|
|
|
|
return TRUE;
|
|
|
|
return FALSE;
|
2002-05-12 22:46:42 +02:00
|
|
|
}
|
2002-06-19 16:52:44 +02:00
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2003-10-16 23:36:01 +02:00
|
|
|
table_map Item_subselect::used_tables() const
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
2012-04-19 02:50:32 +02:00
|
|
|
return (table_map) ((engine->uncacheable() & ~UNCACHEABLE_EXPLAIN)?
|
|
|
|
used_tables_cache : 0L);
|
2002-06-19 16:52:44 +02:00
|
|
|
}
|
|
|
|
|
2003-10-28 11:45:37 +01:00
|
|
|
|
2003-10-16 23:36:01 +02:00
|
|
|
bool Item_subselect::const_item() const
|
|
|
|
{
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(thd);
|
2017-04-20 22:09:31 +02:00
|
|
|
return (thd->lex->context_analysis_only || with_recursive_reference ?
|
2012-08-25 08:15:57 +02:00
|
|
|
FALSE :
|
|
|
|
forced_const || const_item_cache);
|
2003-10-16 23:36:01 +02:00
|
|
|
}
|
|
|
|
|
2006-12-14 23:51:37 +01:00
|
|
|
Item *Item_subselect::get_tmp_table_item(THD *thd_arg)
|
2003-12-06 20:37:24 +01:00
|
|
|
{
|
|
|
|
if (!with_sum_func && !const_item())
|
2015-10-06 17:03:10 +02:00
|
|
|
return new (thd->mem_root) Item_temptable_field(thd_arg, result_field);
|
2006-12-14 23:51:37 +01:00
|
|
|
return copy_or_same(thd_arg);
|
2003-12-06 20:37:24 +01:00
|
|
|
}
|
2003-10-28 11:45:37 +01:00
|
|
|
|
2003-10-16 23:36:01 +02:00
|
|
|
void Item_subselect::update_used_tables()
|
|
|
|
{
|
2010-10-23 20:28:58 +02:00
|
|
|
if (!forced_const)
|
2003-10-16 23:36:01 +02:00
|
|
|
{
|
2010-10-23 20:28:58 +02:00
|
|
|
recalc_used_tables(parent_select, FALSE);
|
2017-01-24 04:40:22 +01:00
|
|
|
if (!(engine->uncacheable() & ~UNCACHEABLE_EXPLAIN))
|
2010-10-23 20:28:58 +02:00
|
|
|
{
|
|
|
|
// did all used tables become static?
|
2017-04-20 22:09:31 +02:00
|
|
|
if (!(used_tables_cache & ~engine->upper_select_const_tables()) &&
|
|
|
|
! with_recursive_reference)
|
2010-10-23 20:28:58 +02:00
|
|
|
const_item_cache= 1;
|
|
|
|
}
|
2003-10-16 23:36:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2008-02-22 11:30:33 +01:00
|
|
|
void Item_subselect::print(String *str, enum_query_type query_type)
|
2003-10-16 14:54:47 +02:00
|
|
|
{
|
2015-09-10 12:04:52 +02:00
|
|
|
if (query_type & QT_ITEM_SUBSELECT_ID_ONLY)
|
2014-11-27 17:32:48 +01:00
|
|
|
{
|
2017-06-19 05:34:38 +02:00
|
|
|
str->append(STRING_WITH_LEN("(subquery#"));
|
2014-12-06 02:02:30 +01:00
|
|
|
if (unit && unit->first_select())
|
2014-11-27 17:32:48 +01:00
|
|
|
{
|
|
|
|
char buf[64];
|
2014-12-06 02:02:30 +01:00
|
|
|
ll2str(unit->first_select()->select_number, buf, 10, 0);
|
2014-11-27 17:32:48 +01:00
|
|
|
str->append(buf);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
str->append("NULL"); // TODO: what exactly does this mean?
|
|
|
|
|
|
|
|
str->append(")");
|
|
|
|
return;
|
|
|
|
}
|
2009-10-30 00:01:54 +01:00
|
|
|
if (engine)
|
|
|
|
{
|
|
|
|
str->append('(');
|
|
|
|
engine->print(str, query_type);
|
|
|
|
str->append(')');
|
|
|
|
}
|
|
|
|
else
|
|
|
|
str->append("(...)");
|
2003-10-16 14:54:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-08-11 09:18:38 +02:00
|
|
|
Item_singlerow_subselect::Item_singlerow_subselect(THD *thd, st_select_lex *select_lex):
|
|
|
|
Item_subselect(thd), value(0)
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
2002-12-19 20:15:09 +01:00
|
|
|
DBUG_ENTER("Item_singlerow_subselect::Item_singlerow_subselect");
|
2015-08-11 09:18:38 +02:00
|
|
|
init(select_lex, new (thd->mem_root) select_singlerow_subselect(thd, this));
|
2002-06-19 16:52:44 +02:00
|
|
|
maybe_null= 1;
|
2002-12-19 06:38:33 +01:00
|
|
|
max_columns= UINT_MAX;
|
2002-10-27 22:27:00 +01:00
|
|
|
DBUG_VOID_RETURN;
|
2002-06-19 16:52:44 +02:00
|
|
|
}
|
|
|
|
|
Bug#21904 (parser problem when using IN with a double "(())")
Before this fix, a IN predicate of the form: "IN (( subselect ))", with two
parenthesis, would be evaluated as a single row subselect: if the subselect
returns more that 1 row, the statement would fail.
The SQL:2003 standard defines a special exception in the specification,
and mandates that this particular form of IN predicate shall be equivalent
to "IN ( subselect )", which involves a table subquery and works with more
than 1 row.
This fix implements "IN (( subselect ))", "IN ((( subselect )))" etc
as per the SQL:2003 requirement.
All the details related to the implementation of this change have been
commented in the code, and the relevant sections of the SQL:2003 spec
are given for reference, so they are not repeated here.
Having access to the spec is a requirement to review in depth this patch.
mysql-test/r/subselect.result:
Implement IN predicate special exceptions with subselects.
mysql-test/t/subselect.test:
Implement IN predicate special exceptions with subselects.
sql/item_subselect.cc:
Implement IN predicate special exceptions with subselects.
sql/item_subselect.h:
Implement IN predicate special exceptions with subselects.
sql/sql_yacc.yy:
Implement IN predicate special exceptions with subselects, cleanup.
2007-01-30 01:32:52 +01:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2015-08-11 09:18:38 +02:00
|
|
|
Item_maxmin_subselect::Item_maxmin_subselect(THD *thd,
|
2004-09-10 12:09:27 +02:00
|
|
|
Item_subselect *parent,
|
2003-10-27 00:01:27 +01:00
|
|
|
st_select_lex *select_lex,
|
2015-08-11 09:18:38 +02:00
|
|
|
bool max_arg):
|
|
|
|
Item_singlerow_subselect(thd), was_values(TRUE)
|
2003-08-12 11:38:03 +02:00
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_maxmin_subselect::Item_maxmin_subselect");
|
2003-10-16 14:54:47 +02:00
|
|
|
max= max_arg;
|
2011-09-02 09:11:13 +02:00
|
|
|
init(select_lex,
|
2015-08-11 09:18:38 +02:00
|
|
|
new (thd->mem_root) select_max_min_finder_subselect(thd,
|
2015-04-22 11:29:56 +02:00
|
|
|
this, max_arg, parent->substype() == Item_subselect::ALL_SUBS));
|
2003-08-12 11:38:03 +02:00
|
|
|
max_columns= 1;
|
|
|
|
maybe_null= 1;
|
|
|
|
max_columns= 1;
|
2003-10-27 00:01:27 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
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();
|
2010-10-23 20:28:58 +02:00
|
|
|
const_item_cache= parent->const_item();
|
2004-07-04 07:46:28 +02:00
|
|
|
|
2003-08-12 11:38:03 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
2004-11-18 17:10:07 +01:00
|
|
|
void Item_maxmin_subselect::cleanup()
|
|
|
|
{
|
2004-12-07 20:18:15 +01:00
|
|
|
DBUG_ENTER("Item_maxmin_subselect::cleanup");
|
|
|
|
Item_singlerow_subselect::cleanup();
|
|
|
|
|
2004-11-18 17:10:07 +01:00
|
|
|
/*
|
2004-12-07 20:18:15 +01:00
|
|
|
By default it is TRUE to avoid TRUE reporting by
|
2004-11-18 17:10:07 +01:00
|
|
|
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.
|
2004-12-07 20:18:15 +01:00
|
|
|
select_max_min_finder_subselect::send_data() set it back to TRUE if some
|
|
|
|
value will be found.
|
2004-11-18 17:10:07 +01:00
|
|
|
*/
|
|
|
|
was_values= TRUE;
|
2004-12-07 20:18:15 +01:00
|
|
|
DBUG_VOID_RETURN;
|
2004-11-18 17:10:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-22 11:30:33 +01:00
|
|
|
void Item_maxmin_subselect::print(String *str, enum_query_type query_type)
|
2003-10-16 14:54:47 +02:00
|
|
|
{
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append(max?"<max>":"<min>", 5);
|
2008-02-22 11:30:33 +01:00
|
|
|
Item_singlerow_subselect::print(str, query_type);
|
2003-10-16 14:54:47 +02:00
|
|
|
}
|
|
|
|
|
2004-11-18 17:10:07 +01:00
|
|
|
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
void Item_maxmin_subselect::no_rows_in_result()
|
|
|
|
{
|
Fix bug lp:1008686
Analysis:
The fix for bug lp:985667 implements the method Item_subselect::no_rows_in_result()
for all main kinds of subqueries. The purpose of this method is to be called from
return_zero_rows() and set Items to some default value in the case when a query
returns no rows. Aggregates and subqueries require special treatment in this case.
Every implementation of Item_subselect::no_rows_in_result() called
Item_subselect::make_const() to set the subquery predicate to its default value
irrespective of where the predicate was located in the query. Once the predicate
was set to a constant it was never executed.
At the same time, the JOIN object of the fake select for UNIONs (the one used for
the final result of the UNION), was set after all subqueries in the union were
executed. Since we set the subquery as constant, it was never executed, and the
corresponding JOIN was never created.
In order to decide whether the result of NOT IN is NULL or FALSE, Item_in_optimizer
needs to check if the subquery result was empty or not. This is where we got the
crash, because subselect_union_engine::no_rows() checks for
unit->fake_select_lex->join->send_records, and the join object was NULL.
Solution:
If a subquery is in the HAVING clause it must be evaluated in order to know its
result, so that we can properly filter the result records. Once subqueries in the
HAVING clause are executed even in the case of no result rows, this specific
crash will be solved, because the UNION will be executed, and its JOIN will be
constructed. Therefore the fix for this crash is to narrow the fix for lp:985667,
and to apply Item_subselect::no_rows_in_result() only when the subquery predicate
is in the SELECT clause.
2012-06-15 10:33:24 +02:00
|
|
|
/*
|
|
|
|
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())
|
2012-06-14 16:03:09 +02:00
|
|
|
return;
|
2016-11-26 18:19:48 +01:00
|
|
|
value= (new (thd->mem_root) Item_null(thd))->get_cache(thd);
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
null_value= 0;
|
|
|
|
was_values= 0;
|
|
|
|
make_const();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-22 14:22:55 +02:00
|
|
|
void Item_singlerow_subselect::no_rows_in_result()
|
|
|
|
{
|
Fix bug lp:1008686
Analysis:
The fix for bug lp:985667 implements the method Item_subselect::no_rows_in_result()
for all main kinds of subqueries. The purpose of this method is to be called from
return_zero_rows() and set Items to some default value in the case when a query
returns no rows. Aggregates and subqueries require special treatment in this case.
Every implementation of Item_subselect::no_rows_in_result() called
Item_subselect::make_const() to set the subquery predicate to its default value
irrespective of where the predicate was located in the query. Once the predicate
was set to a constant it was never executed.
At the same time, the JOIN object of the fake select for UNIONs (the one used for
the final result of the UNION), was set after all subqueries in the union were
executed. Since we set the subquery as constant, it was never executed, and the
corresponding JOIN was never created.
In order to decide whether the result of NOT IN is NULL or FALSE, Item_in_optimizer
needs to check if the subquery result was empty or not. This is where we got the
crash, because subselect_union_engine::no_rows() checks for
unit->fake_select_lex->join->send_records, and the join object was NULL.
Solution:
If a subquery is in the HAVING clause it must be evaluated in order to know its
result, so that we can properly filter the result records. Once subqueries in the
HAVING clause are executed even in the case of no result rows, this specific
crash will be solved, because the UNION will be executed, and its JOIN will be
constructed. Therefore the fix for this crash is to narrow the fix for lp:985667,
and to apply Item_subselect::no_rows_in_result() only when the subquery predicate
is in the SELECT clause.
2012-06-15 10:33:24 +02:00
|
|
|
/*
|
|
|
|
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())
|
2012-06-14 16:03:09 +02:00
|
|
|
return;
|
2016-11-26 18:19:48 +01:00
|
|
|
value= (new (thd->mem_root) Item_null(thd))->get_cache(thd);
|
2012-05-22 14:22:55 +02:00
|
|
|
reset();
|
|
|
|
make_const();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-12-19 20:15:09 +01:00
|
|
|
void Item_singlerow_subselect::reset()
|
2002-09-28 17:34:56 +02:00
|
|
|
{
|
2010-10-23 20:28:58 +02:00
|
|
|
Item_subselect::reset();
|
2002-12-19 06:38:32 +01:00
|
|
|
if (value)
|
2011-11-28 11:42:14 +01:00
|
|
|
{
|
|
|
|
for(uint i= 0; i < engine->cols(); i++)
|
|
|
|
row[i]->set_null();
|
|
|
|
}
|
2002-09-28 17:34:56 +02:00
|
|
|
}
|
|
|
|
|
2004-11-18 17:10:07 +01:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
/**
|
|
|
|
@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.
|
2010-12-15 11:54:25 +01:00
|
|
|
|
|
|
|
@param join Join object of the subquery (i.e. 'child' join).
|
|
|
|
|
|
|
|
@retval false The subquery was transformed
|
2007-10-11 19:29:09 +02:00
|
|
|
*/
|
2010-12-15 11:54:25 +01:00
|
|
|
bool
|
2003-07-02 00:45:22 +02:00
|
|
|
Item_singlerow_subselect::select_transformer(JOIN *join)
|
2002-12-26 00:28:59 +01:00
|
|
|
{
|
2010-01-28 14:48:33 +01:00
|
|
|
DBUG_ENTER("Item_singlerow_subselect::select_transformer");
|
2004-02-08 19:14:13 +01:00
|
|
|
if (changed)
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(false);
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(join->thd == thd);
|
2004-07-04 07:46:28 +02:00
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
SELECT_LEX *select_lex= join->select_lex;
|
2005-09-02 15:21:19 +02:00
|
|
|
Query_arena *arena= thd->stmt_arena;
|
2016-09-21 18:36:34 +02:00
|
|
|
|
2017-03-14 11:52:00 +01:00
|
|
|
if (!select_lex->master_unit()->is_unit_op() &&
|
2003-05-14 20:51:33 +02:00
|
|
|
!select_lex->table_list.elements &&
|
2002-12-31 17:39:16 +01:00
|
|
|
select_lex->item_list.elements == 1 &&
|
2004-10-27 20:11:06 +02:00
|
|
|
!select_lex->item_list.head()->with_sum_func &&
|
2003-02-02 22:30:01 +01:00
|
|
|
/*
|
|
|
|
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
|
|
|
|
*/
|
2002-12-31 17:39:16 +01:00
|
|
|
!(select_lex->item_list.head()->type() == FIELD_ITEM ||
|
2004-10-27 20:11:06 +02:00
|
|
|
select_lex->item_list.head()->type() == REF_ITEM) &&
|
2006-12-12 03:57:23 +01:00
|
|
|
!join->conds && !join->having &&
|
2004-10-27 20:11:06 +02:00
|
|
|
/*
|
2005-02-08 23:50:45 +01:00
|
|
|
switch off this optimization for prepare statement,
|
2004-10-27 20:11:06 +02:00
|
|
|
because we do not rollback this changes
|
|
|
|
TODO: make rollback for it, or special name resolving mode in 5.0.
|
|
|
|
*/
|
2005-06-02 22:02:47 +02:00
|
|
|
!arena->is_stmt_prepare_or_first_sp_execute()
|
2002-12-31 17:39:16 +01:00
|
|
|
)
|
2002-12-26 00:28:59 +01:00
|
|
|
{
|
|
|
|
have_to_be_excluded= 1;
|
2004-09-10 01:22:44 +02:00
|
|
|
if (thd->lex->describe)
|
2002-12-26 00:28:59 +01:00
|
|
|
{
|
|
|
|
char warn_buff[MYSQL_ERRMSG_SIZE];
|
2015-07-06 19:24:14 +02:00
|
|
|
sprintf(warn_buff, ER_THD(thd, ER_SELECT_REDUCED),
|
|
|
|
select_lex->select_number);
|
2013-06-15 17:32:08 +02:00
|
|
|
push_warning(thd, Sql_condition::WARN_LEVEL_NOTE,
|
2002-12-26 00:28:59 +01:00
|
|
|
ER_SELECT_REDUCED, warn_buff);
|
|
|
|
}
|
|
|
|
substitution= select_lex->item_list.head();
|
2003-07-02 12:12:18 +02:00
|
|
|
/*
|
2012-09-20 11:48:59 +02:00
|
|
|
as far as we moved content to upper level we have to fix dependences & Co
|
2003-07-02 12:12:18 +02:00
|
|
|
*/
|
2017-11-08 15:47:49 +01:00
|
|
|
substitution->fix_after_pullout(select_lex->outer_select(),
|
|
|
|
&substitution, TRUE);
|
2002-12-26 00:28:59 +01:00
|
|
|
}
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(false);
|
2002-12-26 00:28:59 +01:00
|
|
|
}
|
|
|
|
|
2004-10-29 18:26:52 +02:00
|
|
|
|
2002-12-19 20:15:09 +01:00
|
|
|
void Item_singlerow_subselect::store(uint i, Item *item)
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
2002-12-19 06:38:33 +01:00
|
|
|
row[i]->store(item);
|
2009-11-06 20:34:25 +01:00
|
|
|
row[i]->cache_value();
|
2002-06-19 16:52:44 +02:00
|
|
|
}
|
|
|
|
|
2016-11-26 18:19:48 +01:00
|
|
|
const Type_handler *Item_singlerow_subselect::type_handler() const
|
|
|
|
{
|
|
|
|
return engine->type_handler();
|
2013-03-17 11:41:25 +01:00
|
|
|
}
|
|
|
|
|
2018-05-08 15:26:26 +02:00
|
|
|
bool Item_singlerow_subselect::fix_length_and_dec()
|
2002-09-28 17:34:56 +02:00
|
|
|
{
|
2002-12-19 06:38:33 +01:00
|
|
|
if ((max_columns= engine->cols()) == 1)
|
|
|
|
{
|
2018-05-08 15:26:26 +02:00
|
|
|
if (engine->fix_length_and_dec(row= &value))
|
|
|
|
return TRUE;
|
2002-12-19 06:38:33 +01:00
|
|
|
}
|
|
|
|
else
|
2002-12-19 06:38:32 +01:00
|
|
|
{
|
2015-11-19 13:50:09 +01:00
|
|
|
if (!(row= (Item_cache**) current_thd->alloc(sizeof(Item_cache*) *
|
2018-05-08 15:26:26 +02:00
|
|
|
max_columns)) ||
|
|
|
|
engine->fix_length_and_dec(row))
|
|
|
|
return TRUE;
|
2002-12-19 06:38:33 +01:00
|
|
|
value= *row;
|
2002-12-19 06:38:32 +01:00
|
|
|
}
|
2006-05-25 09:39:18 +02:00
|
|
|
unsigned_flag= value->unsigned_flag;
|
2004-10-27 20:11:06 +02:00
|
|
|
/*
|
|
|
|
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();
|
2011-11-28 11:42:14 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
for (uint i= 0; i < max_columns; i++)
|
|
|
|
row[i]->maybe_null= TRUE;
|
|
|
|
}
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2002-09-28 17:34:56 +02:00
|
|
|
}
|
|
|
|
|
2010-07-10 12:37:30 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
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
|
2014-11-14 13:46:21 +01:00
|
|
|
Item_cache_wrapper with an appropriate expression cache set up there.
|
2010-07-10 12:37:30 +02:00
|
|
|
|
|
|
|
@note
|
|
|
|
used from Item::transform()
|
|
|
|
|
|
|
|
@return
|
|
|
|
new wrapper item if an expression cache is needed,
|
|
|
|
this item - otherwise
|
|
|
|
*/
|
|
|
|
|
2015-08-11 09:18:38 +02:00
|
|
|
Item* Item_singlerow_subselect::expr_cache_insert_transformer(THD *tmp_thd,
|
|
|
|
uchar *unused)
|
2010-07-10 12:37:30 +02:00
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_singlerow_subselect::expr_cache_insert_transformer");
|
|
|
|
|
2015-07-06 19:24:14 +02:00
|
|
|
DBUG_ASSERT(thd == tmp_thd);
|
|
|
|
|
2010-09-06 14:34:24 +02:00
|
|
|
if (expr_cache)
|
|
|
|
DBUG_RETURN(expr_cache);
|
|
|
|
|
2015-07-06 19:24:14 +02:00
|
|
|
if (expr_cache_is_needed(tmp_thd) &&
|
|
|
|
(expr_cache= set_expr_cache(tmp_thd)))
|
2015-03-25 18:27:10 +01:00
|
|
|
{
|
2015-07-09 13:47:32 +02:00
|
|
|
init_expr_cache_tracker(tmp_thd);
|
2010-09-06 14:34:24 +02:00
|
|
|
DBUG_RETURN(expr_cache);
|
2015-03-25 18:27:10 +01:00
|
|
|
}
|
2010-07-10 12:37:30 +02:00
|
|
|
DBUG_RETURN(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-05-06 13:05:59 +02:00
|
|
|
uint Item_singlerow_subselect::cols() const
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
2002-12-19 06:38:33 +01:00
|
|
|
return engine->cols();
|
|
|
|
}
|
|
|
|
|
2002-12-19 20:15:09 +01:00
|
|
|
bool Item_singlerow_subselect::check_cols(uint c)
|
2002-12-19 06:38:33 +01:00
|
|
|
{
|
|
|
|
if (c != engine->cols())
|
|
|
|
{
|
2003-10-06 21:35:05 +02:00
|
|
|
my_error(ER_OPERAND_COLUMNS, MYF(0), c);
|
2002-12-19 06:38:33 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-12-19 20:15:09 +01:00
|
|
|
bool Item_singlerow_subselect::null_inside()
|
2002-12-19 06:38:33 +01:00
|
|
|
{
|
|
|
|
for (uint i= 0; i < max_columns ; i++)
|
|
|
|
{
|
|
|
|
if (row[i]->null_value)
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-12-19 20:15:09 +01:00
|
|
|
void Item_singlerow_subselect::bring_value()
|
2002-12-19 06:38:33 +01:00
|
|
|
{
|
2010-09-09 14:46:13 +02:00
|
|
|
if (!exec() && assigned())
|
|
|
|
null_value= 0;
|
|
|
|
else
|
|
|
|
reset();
|
2002-06-19 16:52:44 +02:00
|
|
|
}
|
|
|
|
|
2004-11-11 19:39:35 +01:00
|
|
|
double Item_singlerow_subselect::val_real()
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
if (forced_const)
|
|
|
|
return value->val_real();
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (!exec() && !value->null_value)
|
2002-12-19 06:38:32 +01:00
|
|
|
{
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= FALSE;
|
2004-11-11 19:39:35 +01:00
|
|
|
return value->val_real();
|
2002-12-19 06:38:32 +01:00
|
|
|
}
|
|
|
|
else
|
2002-10-07 21:21:17 +02:00
|
|
|
{
|
2002-12-06 20:55:53 +01:00
|
|
|
reset();
|
2002-06-19 16:52:44 +02:00
|
|
|
return 0;
|
2002-10-07 21:21:17 +02:00
|
|
|
}
|
2002-06-19 16:52:44 +02:00
|
|
|
}
|
|
|
|
|
2004-07-04 07:46:28 +02:00
|
|
|
longlong Item_singlerow_subselect::val_int()
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
if (forced_const)
|
|
|
|
return value->val_int();
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (!exec() && !value->null_value)
|
2002-12-19 06:38:32 +01:00
|
|
|
{
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= FALSE;
|
2002-12-19 06:38:32 +01:00
|
|
|
return value->val_int();
|
|
|
|
}
|
|
|
|
else
|
2002-10-07 21:21:17 +02:00
|
|
|
{
|
2002-12-06 20:55:53 +01:00
|
|
|
reset();
|
2002-06-19 16:52:44 +02:00
|
|
|
return 0;
|
2002-10-07 21:21:17 +02:00
|
|
|
}
|
2002-06-19 16:52:44 +02:00
|
|
|
}
|
|
|
|
|
2005-02-08 23:50:45 +01:00
|
|
|
String *Item_singlerow_subselect::val_str(String *str)
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
|
|
|
if (forced_const)
|
|
|
|
return value->val_str(str);
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (!exec() && !value->null_value)
|
2002-12-19 06:38:32 +01:00
|
|
|
{
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= FALSE;
|
2002-12-19 06:38:32 +01:00
|
|
|
return value->val_str(str);
|
|
|
|
}
|
|
|
|
else
|
2002-10-07 21:21:17 +02:00
|
|
|
{
|
2002-12-06 20:55:53 +01:00
|
|
|
reset();
|
2002-06-19 16:52:44 +02:00
|
|
|
return 0;
|
2002-10-07 21:21:17 +02:00
|
|
|
}
|
2002-06-19 16:52:44 +02:00
|
|
|
}
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2005-02-08 23:50:45 +01:00
|
|
|
my_decimal *Item_singlerow_subselect::val_decimal(my_decimal *decimal_value)
|
|
|
|
{
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
|
|
|
if (forced_const)
|
|
|
|
return value->val_decimal(decimal_value);
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (!exec() && !value->null_value)
|
2005-02-08 23:50:45 +01:00
|
|
|
{
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= FALSE;
|
2005-02-08 23:50:45 +01:00
|
|
|
return value->val_decimal(decimal_value);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
reset();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Item_singlerow_subselect::val_bool()
|
|
|
|
{
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
|
|
|
if (forced_const)
|
|
|
|
return value->val_bool();
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (!exec() && !value->null_value)
|
2005-02-08 23:50:45 +01:00
|
|
|
{
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= FALSE;
|
2005-02-08 23:50:45 +01:00
|
|
|
return value->val_bool();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
reset();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-04 07:16:38 +02:00
|
|
|
bool Item_singlerow_subselect::get_date(MYSQL_TIME *ltime,ulonglong fuzzydate)
|
2012-05-02 15:23:49 +02:00
|
|
|
{
|
|
|
|
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();
|
2012-08-22 18:40:27 +02:00
|
|
|
return 1;
|
2012-05-02 15:23:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-08-11 09:18:38 +02:00
|
|
|
Item_exists_subselect::Item_exists_subselect(THD *thd,
|
2015-04-22 11:29:56 +02:00
|
|
|
st_select_lex *select_lex):
|
2015-08-11 09:18:38 +02:00
|
|
|
Item_subselect(thd), upper_not(NULL), abort_on_null(0),
|
2013-02-26 00:20:17 +01:00
|
|
|
emb_on_expr_nest(NULL), optimizer(0), exists_transformed(0)
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
2002-10-27 22:27:00 +01:00
|
|
|
DBUG_ENTER("Item_exists_subselect::Item_exists_subselect");
|
2015-08-11 09:18:38 +02:00
|
|
|
init(select_lex, new (thd->mem_root) select_exists_subselect(thd, this));
|
2002-06-19 16:52:44 +02:00
|
|
|
max_columns= UINT_MAX;
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= FALSE; //can't be NULL
|
2002-06-19 16:52:44 +02:00
|
|
|
maybe_null= 0; //can't be NULL
|
|
|
|
value= 0;
|
2002-10-27 22:27:00 +01:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2008-02-22 11:30:33 +01:00
|
|
|
void Item_exists_subselect::print(String *str, enum_query_type query_type)
|
2003-10-16 14:54:47 +02:00
|
|
|
{
|
2005-11-20 19:47:07 +01:00
|
|
|
str->append(STRING_WITH_LEN("exists"));
|
2008-02-22 11:30:33 +01:00
|
|
|
Item_subselect::print(str, query_type);
|
2003-10-16 14:54:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-08-15 15:43:08 +02:00
|
|
|
bool Item_in_subselect::test_limit(st_select_lex_unit *unit_arg)
|
2003-05-14 20:51:33 +02:00
|
|
|
{
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely(unit_arg->fake_select_lex &&
|
|
|
|
unit_arg->fake_select_lex->test_limit()))
|
2003-05-14 20:51:33 +02:00
|
|
|
return(1);
|
2003-07-02 00:45:22 +02:00
|
|
|
|
2006-12-14 23:51:37 +01:00
|
|
|
SELECT_LEX *sl= unit_arg->first_select();
|
2003-05-14 20:51:33 +02:00
|
|
|
for (; sl; sl= sl->next_select())
|
|
|
|
{
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely(sl->test_limit()))
|
2003-05-14 20:51:33 +02:00
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
2015-08-11 09:18:38 +02:00
|
|
|
Item_in_subselect::Item_in_subselect(THD *thd, Item * left_exp,
|
2002-10-27 22:27:00 +01:00
|
|
|
st_select_lex *select_lex):
|
2015-08-11 09:18:38 +02:00
|
|
|
Item_exists_subselect(thd), left_expr_cache(0), first_execution(TRUE),
|
2013-02-26 00:20:17 +01:00
|
|
|
in_strategy(SUBS_NOT_TRANSFORMED),
|
2017-05-17 14:42:36 +02:00
|
|
|
pushed_cond_guards(NULL), do_not_convert_to_sj(FALSE), is_jtbm_merged(FALSE),
|
|
|
|
is_jtbm_const_tab(FALSE), is_flattenable_semijoin(FALSE),
|
|
|
|
is_registered_semijoin(FALSE),
|
2011-05-25 17:31:13 +02:00
|
|
|
upper_item(0)
|
2002-10-27 22:27:00 +01:00
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_in_subselect::Item_in_subselect");
|
2013-02-26 00:20:17 +01:00
|
|
|
DBUG_PRINT("info", ("in_strategy: %u", (uint)in_strategy));
|
2015-04-23 20:08:57 +02:00
|
|
|
left_expr_orig= left_expr= left_exp;
|
2017-02-18 17:47:31 +01:00
|
|
|
/* prepare to possible disassembling the item in convert_subq_to_sj() */
|
|
|
|
if (left_exp->type() == Item::ROW_ITEM)
|
2017-10-22 13:03:41 +02:00
|
|
|
left_expr_orig= new (thd->mem_root)
|
|
|
|
Item_row(thd, static_cast<Item_row*>(left_exp));
|
2010-09-05 17:43:47 +02:00
|
|
|
func= &eq_creator;
|
2015-08-11 09:18:38 +02:00
|
|
|
init(select_lex, new (thd->mem_root) select_exists_subselect(thd, this));
|
2002-11-07 22:45:19 +01:00
|
|
|
max_columns= UINT_MAX;
|
2002-12-06 20:55:53 +01:00
|
|
|
maybe_null= 1;
|
|
|
|
reset();
|
2003-07-02 00:45:22 +02:00
|
|
|
//if test_limit will fail then error will be reported to client
|
2003-05-14 20:51:33 +02:00
|
|
|
test_limit(select_lex->master_unit());
|
2002-11-07 22:45:19 +01:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
2010-03-29 16:04:35 +02:00
|
|
|
int Item_in_subselect::get_identifier()
|
|
|
|
{
|
|
|
|
return engine->get_identifier();
|
|
|
|
}
|
|
|
|
|
2015-08-11 09:18:38 +02:00
|
|
|
Item_allany_subselect::Item_allany_subselect(THD *thd, Item * left_exp,
|
2006-07-21 01:04:04 +02:00
|
|
|
chooser_compare_func_creator fc,
|
2003-10-16 14:54:47 +02:00
|
|
|
st_select_lex *select_lex,
|
2015-08-11 09:18:38 +02:00
|
|
|
bool all_arg):
|
|
|
|
Item_in_subselect(thd), func_creator(fc), all(all_arg)
|
2002-11-07 22:45:19 +01:00
|
|
|
{
|
2010-01-28 14:48:33 +01:00
|
|
|
DBUG_ENTER("Item_allany_subselect::Item_allany_subselect");
|
2015-04-23 20:08:57 +02:00
|
|
|
left_expr_orig= left_expr= left_exp;
|
2017-02-18 17:47:31 +01:00
|
|
|
/* prepare to possible disassembling the item in convert_subq_to_sj() */
|
|
|
|
if (left_exp->type() == Item::ROW_ITEM)
|
2017-10-22 13:03:41 +02:00
|
|
|
left_expr_orig= new (thd->mem_root)
|
|
|
|
Item_row(thd, static_cast<Item_row*>(left_exp));
|
2006-07-21 01:04:04 +02:00
|
|
|
func= func_creator(all_arg);
|
2015-08-11 09:18:38 +02:00
|
|
|
init(select_lex, new (thd->mem_root) select_exists_subselect(thd, this));
|
2002-12-19 20:15:09 +01:00
|
|
|
max_columns= 1;
|
2003-05-14 20:51:33 +02:00
|
|
|
abort_on_null= 0;
|
2002-12-06 20:55:53 +01:00
|
|
|
reset();
|
2003-07-02 00:45:22 +02:00
|
|
|
//if test_limit will fail then error will be reported to client
|
2003-05-14 20:51:33 +02:00
|
|
|
test_limit(select_lex->master_unit());
|
2002-10-27 22:27:00 +01:00
|
|
|
DBUG_VOID_RETURN;
|
2002-06-19 16:52:44 +02:00
|
|
|
}
|
|
|
|
|
2002-10-27 22:27:00 +01:00
|
|
|
|
2011-05-11 23:14:15 +02:00
|
|
|
/**
|
|
|
|
Initialize length and decimals for EXISTS and inherited (IN/ALL/ANY)
|
|
|
|
subqueries
|
|
|
|
*/
|
|
|
|
|
|
|
|
void Item_exists_subselect::init_length_and_dec()
|
2002-09-28 17:34:56 +02:00
|
|
|
{
|
2011-05-04 17:08:58 +02:00
|
|
|
decimals= 0;
|
|
|
|
max_length= 1;
|
|
|
|
max_columns= engine->cols();
|
2011-05-11 23:14:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-05-08 15:26:26 +02:00
|
|
|
bool Item_exists_subselect::fix_length_and_dec()
|
2002-09-28 17:34:56 +02:00
|
|
|
{
|
2011-05-11 23:14:15 +02:00
|
|
|
DBUG_ENTER("Item_exists_subselect::fix_length_and_dec");
|
|
|
|
init_length_and_dec();
|
2019-07-17 12:31:45 +02:00
|
|
|
// If limit is not set or it is constant more than 1
|
2019-07-25 13:27:11 +02:00
|
|
|
if (!unit->global_parameters()->select_limit ||
|
|
|
|
(unit->global_parameters()->select_limit->basic_const_item() &&
|
|
|
|
unit->global_parameters()->select_limit->val_int() > 1))
|
2019-07-17 12:31:45 +02:00
|
|
|
{
|
|
|
|
/*
|
2019-07-26 07:03:39 +02:00
|
|
|
We need only 1 row to determine existence (i.e. any EXISTS that is not
|
|
|
|
an IN always requires LIMIT 1)
|
|
|
|
*/
|
|
|
|
Item *item= new (thd->mem_root) Item_int(thd, (int32) 1);
|
|
|
|
if (!item)
|
|
|
|
DBUG_RETURN(TRUE);
|
2019-07-25 13:27:11 +02:00
|
|
|
thd->change_item_tree(&unit->global_parameters()->select_limit,
|
2019-07-26 07:03:39 +02:00
|
|
|
item);
|
2019-07-25 13:27:11 +02:00
|
|
|
unit->global_parameters()->explicit_limit= 1; // we set the limit
|
2019-07-17 12:31:45 +02:00
|
|
|
DBUG_PRINT("info", ("Set limit to 1"));
|
|
|
|
}
|
2018-05-08 15:26:26 +02:00
|
|
|
DBUG_RETURN(FALSE);
|
2011-05-04 17:08:58 +02:00
|
|
|
}
|
|
|
|
|
2011-05-11 23:14:15 +02:00
|
|
|
|
2018-05-08 15:26:26 +02:00
|
|
|
bool Item_in_subselect::fix_length_and_dec()
|
2011-05-04 17:08:58 +02:00
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_in_subselect::fix_length_and_dec");
|
2011-05-11 23:14:15 +02:00
|
|
|
init_length_and_dec();
|
|
|
|
/*
|
|
|
|
Unlike Item_exists_subselect, LIMIT 1 is set later for
|
|
|
|
Item_in_subselect, depending on the chosen strategy.
|
|
|
|
*/
|
2018-05-08 15:26:26 +02:00
|
|
|
DBUG_RETURN(FALSE);
|
2002-09-28 17:34:56 +02:00
|
|
|
}
|
|
|
|
|
2010-07-10 12:37:30 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
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
|
2014-11-14 13:46:21 +01:00
|
|
|
Item_cache_wrapper with an appropriate expression cache set up there.
|
2010-07-10 12:37:30 +02:00
|
|
|
|
|
|
|
@note
|
|
|
|
used from Item::transform()
|
|
|
|
|
|
|
|
@return
|
|
|
|
new wrapper item if an expression cache is needed,
|
|
|
|
this item - otherwise
|
|
|
|
*/
|
|
|
|
|
2015-08-11 09:18:38 +02:00
|
|
|
Item* Item_exists_subselect::expr_cache_insert_transformer(THD *tmp_thd,
|
|
|
|
uchar *unused)
|
2010-07-10 12:37:30 +02:00
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_exists_subselect::expr_cache_insert_transformer");
|
2015-07-06 19:24:14 +02:00
|
|
|
DBUG_ASSERT(thd == tmp_thd);
|
2010-07-10 12:37:30 +02:00
|
|
|
|
2010-09-06 14:34:24 +02:00
|
|
|
if (expr_cache)
|
|
|
|
DBUG_RETURN(expr_cache);
|
|
|
|
|
2015-07-06 19:24:14 +02:00
|
|
|
if (substype() == EXISTS_SUBS && expr_cache_is_needed(tmp_thd) &&
|
|
|
|
(expr_cache= set_expr_cache(tmp_thd)))
|
2015-03-25 18:27:10 +01:00
|
|
|
{
|
2015-07-09 13:47:32 +02:00
|
|
|
init_expr_cache_tracker(tmp_thd);
|
2010-09-06 14:34:24 +02:00
|
|
|
DBUG_RETURN(expr_cache);
|
2015-03-25 18:27:10 +01:00
|
|
|
}
|
2010-07-10 12:37:30 +02:00
|
|
|
DBUG_RETURN(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
void Item_exists_subselect::no_rows_in_result()
|
|
|
|
{
|
Fix bug lp:1008686
Analysis:
The fix for bug lp:985667 implements the method Item_subselect::no_rows_in_result()
for all main kinds of subqueries. The purpose of this method is to be called from
return_zero_rows() and set Items to some default value in the case when a query
returns no rows. Aggregates and subqueries require special treatment in this case.
Every implementation of Item_subselect::no_rows_in_result() called
Item_subselect::make_const() to set the subquery predicate to its default value
irrespective of where the predicate was located in the query. Once the predicate
was set to a constant it was never executed.
At the same time, the JOIN object of the fake select for UNIONs (the one used for
the final result of the UNION), was set after all subqueries in the union were
executed. Since we set the subquery as constant, it was never executed, and the
corresponding JOIN was never created.
In order to decide whether the result of NOT IN is NULL or FALSE, Item_in_optimizer
needs to check if the subquery result was empty or not. This is where we got the
crash, because subselect_union_engine::no_rows() checks for
unit->fake_select_lex->join->send_records, and the join object was NULL.
Solution:
If a subquery is in the HAVING clause it must be evaluated in order to know its
result, so that we can properly filter the result records. Once subqueries in the
HAVING clause are executed even in the case of no result rows, this specific
crash will be solved, because the UNION will be executed, and its JOIN will be
constructed. Therefore the fix for this crash is to narrow the fix for lp:985667,
and to apply Item_subselect::no_rows_in_result() only when the subquery predicate
is in the SELECT clause.
2012-06-15 10:33:24 +02:00
|
|
|
/*
|
|
|
|
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())
|
2012-06-14 16:03:09 +02:00
|
|
|
return;
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
value= 0;
|
|
|
|
null_value= 0;
|
|
|
|
make_const();
|
|
|
|
}
|
|
|
|
|
2004-11-11 19:39:35 +01:00
|
|
|
double Item_exists_subselect::val_real()
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
if (!forced_const && exec())
|
2002-10-07 21:21:17 +02:00
|
|
|
{
|
2002-12-06 20:55:53 +01:00
|
|
|
reset();
|
2002-06-19 16:52:44 +02:00
|
|
|
return 0;
|
2002-10-07 21:21:17 +02:00
|
|
|
}
|
2002-06-19 16:52:44 +02:00
|
|
|
return (double) value;
|
|
|
|
}
|
|
|
|
|
2004-07-04 07:46:28 +02:00
|
|
|
longlong Item_exists_subselect::val_int()
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
if (!forced_const && exec())
|
2002-10-07 21:21:17 +02:00
|
|
|
{
|
2002-12-06 20:55:53 +01:00
|
|
|
reset();
|
2002-06-19 16:52:44 +02:00
|
|
|
return 0;
|
2002-10-07 21:21:17 +02:00
|
|
|
}
|
2002-06-19 16:52:44 +02:00
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2008-07-04 16:02:17 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2002-06-19 16:52:44 +02:00
|
|
|
String *Item_exists_subselect::val_str(String *str)
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
if (!forced_const && exec())
|
2002-12-06 20:55:53 +01:00
|
|
|
reset();
|
2005-04-01 01:14:30 +02:00
|
|
|
str->set((ulonglong)value,&my_charset_bin);
|
2002-12-06 20:55:53 +01:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2005-02-08 23:50:45 +01:00
|
|
|
|
2008-07-04 16:02:17 +02:00
|
|
|
/**
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2005-02-08 23:50:45 +01:00
|
|
|
my_decimal *Item_exists_subselect::val_decimal(my_decimal *decimal_value)
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
if (!forced_const && exec())
|
2005-02-08 23:50:45 +01:00
|
|
|
reset();
|
|
|
|
int2my_decimal(E_DEC_FATAL_ERROR, value, 0, decimal_value);
|
|
|
|
return decimal_value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Item_exists_subselect::val_bool()
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
if (!forced_const && exec())
|
2005-02-08 23:50:45 +01:00
|
|
|
{
|
|
|
|
reset();
|
|
|
|
return 0;
|
|
|
|
}
|
2005-03-19 01:12:25 +01:00
|
|
|
return value != 0;
|
2005-02-08 23:50:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-11-11 19:39:35 +01:00
|
|
|
double Item_in_subselect::val_real()
|
2002-12-06 20:55:53 +01:00
|
|
|
{
|
2005-03-30 09:07:08 +02:00
|
|
|
/*
|
|
|
|
As far as Item_in_subselect called only from Item_in_optimizer this
|
|
|
|
method should not be used
|
|
|
|
*/
|
|
|
|
DBUG_ASSERT(0);
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2012-08-25 08:15:57 +02:00
|
|
|
if (forced_const)
|
|
|
|
return value;
|
|
|
|
DBUG_ASSERT((engine->uncacheable() & ~UNCACHEABLE_EXPLAIN) ||
|
|
|
|
! engine->is_executed());
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= was_null= FALSE;
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (exec())
|
2002-12-06 20:55:53 +01:00
|
|
|
{
|
|
|
|
reset();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (was_null && !value)
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= TRUE;
|
2002-12-06 20:55:53 +01:00
|
|
|
return (double) value;
|
|
|
|
}
|
|
|
|
|
2005-03-30 09:07:08 +02:00
|
|
|
|
2004-07-04 07:46:28 +02:00
|
|
|
longlong Item_in_subselect::val_int()
|
2002-12-06 20:55:53 +01:00
|
|
|
{
|
2005-03-31 09:39:48 +02:00
|
|
|
/*
|
|
|
|
As far as Item_in_subselect called only from Item_in_optimizer this
|
|
|
|
method should not be used
|
|
|
|
*/
|
|
|
|
DBUG_ASSERT(0);
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2012-08-25 08:15:57 +02:00
|
|
|
if (forced_const)
|
|
|
|
return value;
|
|
|
|
DBUG_ASSERT((engine->uncacheable() & ~UNCACHEABLE_EXPLAIN) ||
|
|
|
|
! engine->is_executed());
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= was_null= FALSE;
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (exec())
|
2002-12-06 20:55:53 +01:00
|
|
|
{
|
|
|
|
reset();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (was_null && !value)
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= TRUE;
|
2002-12-06 20:55:53 +01:00
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2005-03-30 09:07:08 +02:00
|
|
|
|
2002-12-06 20:55:53 +01:00
|
|
|
String *Item_in_subselect::val_str(String *str)
|
|
|
|
{
|
2005-03-30 09:07:08 +02:00
|
|
|
/*
|
|
|
|
As far as Item_in_subselect called only from Item_in_optimizer this
|
|
|
|
method should not be used
|
|
|
|
*/
|
|
|
|
DBUG_ASSERT(0);
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2012-08-25 08:15:57 +02:00
|
|
|
if (forced_const)
|
|
|
|
goto value_is_ready;
|
|
|
|
DBUG_ASSERT((engine->uncacheable() & ~UNCACHEABLE_EXPLAIN) ||
|
|
|
|
! engine->is_executed());
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= was_null= FALSE;
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (exec())
|
2002-12-06 20:55:53 +01:00
|
|
|
{
|
|
|
|
reset();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (was_null && !value)
|
|
|
|
{
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= TRUE;
|
2002-06-19 16:52:44 +02:00
|
|
|
return 0;
|
2002-10-07 21:21:17 +02:00
|
|
|
}
|
2012-08-25 08:15:57 +02:00
|
|
|
value_is_ready:
|
2005-04-01 01:14:30 +02:00
|
|
|
str->set((ulonglong)value, &my_charset_bin);
|
2002-06-19 16:52:44 +02:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2002-11-07 22:45:19 +01:00
|
|
|
|
2005-04-01 01:14:30 +02:00
|
|
|
bool Item_in_subselect::val_bool()
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(fixed == 1);
|
2010-10-23 20:28:58 +02:00
|
|
|
if (forced_const)
|
2010-02-19 22:55:57 +01:00
|
|
|
return value;
|
2012-08-25 08:15:57 +02:00
|
|
|
DBUG_ASSERT((engine->uncacheable() & ~UNCACHEABLE_EXPLAIN) ||
|
2017-04-20 22:09:31 +02:00
|
|
|
! engine->is_executed() || with_recursive_reference);
|
2011-07-14 23:23:57 +02:00
|
|
|
null_value= was_null= FALSE;
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (exec())
|
2005-04-01 01:14:30 +02:00
|
|
|
{
|
|
|
|
reset();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (was_null && !value)
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= TRUE;
|
2005-04-01 01:14:30 +02:00
|
|
|
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);
|
2012-08-25 08:15:57 +02:00
|
|
|
if (forced_const)
|
|
|
|
goto value_is_ready;
|
|
|
|
DBUG_ASSERT((engine->uncacheable() & ~UNCACHEABLE_EXPLAIN) ||
|
|
|
|
! engine->is_executed());
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= was_null= FALSE;
|
2005-04-01 01:14:30 +02:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (exec())
|
2005-04-01 01:14:30 +02:00
|
|
|
{
|
|
|
|
reset();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (was_null && !value)
|
2010-09-07 11:21:09 +02:00
|
|
|
null_value= TRUE;
|
2012-08-25 08:15:57 +02:00
|
|
|
value_is_ready:
|
2005-04-01 01:14:30 +02:00
|
|
|
int2my_decimal(E_DEC_FATAL_ERROR, value, 0, decimal_value);
|
|
|
|
return decimal_value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
/**
|
2011-05-11 23:14:15 +02:00
|
|
|
Prepare a single-column IN/ALL/ANY subselect for rewriting.
|
2006-10-31 18:51:09 +01:00
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
@param join Join object of the subquery (i.e. 'child' join).
|
2006-10-31 18:51:09 +01:00
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
@details
|
2006-10-31 18:51:09 +01:00
|
|
|
|
2011-05-11 23:14:15 +02:00
|
|
|
Prepare a single-column subquery to be rewritten. Given the subquery.
|
2006-10-31 18:51:09 +01:00
|
|
|
|
2011-05-11 23:14:15 +02:00
|
|
|
If the subquery has no tables it will be turned to an expression between
|
|
|
|
left part and SELECT list.
|
2006-10-31 22:27:51 +01:00
|
|
|
|
2011-05-11 23:14:15 +02:00
|
|
|
In other cases the subquery will be wrapped with Item_in_optimizer which
|
|
|
|
allow later to turn it to EXISTS or MAX/MIN.
|
2006-10-31 18:51:09 +01:00
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
@retval false The subquery was transformed
|
|
|
|
@retval true Error
|
2006-10-31 18:51:09 +01:00
|
|
|
*/
|
2004-10-04 16:58:06 +02:00
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
bool
|
2010-09-05 17:43:47 +02:00
|
|
|
Item_in_subselect::single_value_transformer(JOIN *join)
|
2002-11-07 22:45:19 +01:00
|
|
|
{
|
2003-05-14 20:51:33 +02:00
|
|
|
SELECT_LEX *select_lex= join->select_lex;
|
2005-05-30 19:48:40 +02:00
|
|
|
DBUG_ENTER("Item_in_subselect::single_value_transformer");
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(thd == join->thd);
|
2004-09-06 14:14:10 +02:00
|
|
|
|
2004-10-04 16:58:06 +02:00
|
|
|
/*
|
|
|
|
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 * ...)
|
|
|
|
*/
|
2010-01-28 14:48:33 +01:00
|
|
|
// psergey: duplicated_subselect_card_check
|
2003-10-23 22:54:21 +02:00
|
|
|
if (select_lex->item_list.elements > 1)
|
|
|
|
{
|
|
|
|
my_error(ER_OPERAND_COLUMNS, MYF(0), 1);
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2003-10-23 22:54:21 +02:00
|
|
|
}
|
|
|
|
|
2010-09-05 17:43:47 +02:00
|
|
|
Item* join_having= join->having ? join->having : join->tmp_having;
|
|
|
|
if (!(join_having || select_lex->with_sum_func ||
|
|
|
|
select_lex->group_list.elements) &&
|
2018-03-20 21:04:45 +01:00
|
|
|
select_lex->table_list.elements == 0 && !join->conds &&
|
2017-03-14 11:52:00 +01:00
|
|
|
!select_lex->master_unit()->is_unit_op())
|
2003-07-24 14:26:21 +02:00
|
|
|
{
|
2010-09-05 17:43:47 +02:00
|
|
|
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,
|
2016-06-26 22:42:48 +02:00
|
|
|
select_lex->outer_select());
|
2013-05-07 11:10:58 +02:00
|
|
|
/*
|
2013-09-06 22:31:30 +02:00
|
|
|
fix_field of substitution item will be done in time of
|
|
|
|
substituting.
|
2013-05-07 11:10:58 +02:00
|
|
|
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.
|
|
|
|
*/
|
2015-08-11 09:18:38 +02:00
|
|
|
substitution= func->create(thd, left_expr, where_item);
|
2010-09-05 17:43:47 +02:00
|
|
|
have_to_be_excluded= 1;
|
|
|
|
if (thd->lex->describe)
|
2003-07-24 14:26:21 +02:00
|
|
|
{
|
2010-09-05 17:43:47 +02:00
|
|
|
char warn_buff[MYSQL_ERRMSG_SIZE];
|
2015-07-06 19:24:14 +02:00
|
|
|
sprintf(warn_buff, ER_THD(thd, ER_SELECT_REDUCED),
|
|
|
|
select_lex->select_number);
|
2013-06-15 17:32:08 +02:00
|
|
|
push_warning(thd, Sql_condition::WARN_LEVEL_NOTE,
|
2010-09-05 17:43:47 +02:00
|
|
|
ER_SELECT_REDUCED, warn_buff);
|
2003-07-24 14:26:21 +02:00
|
|
|
}
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(false);
|
2003-07-24 14:26:21 +02:00
|
|
|
}
|
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
/*
|
|
|
|
Wrap the current IN predicate in an Item_in_optimizer. The actual
|
|
|
|
substitution in the Item tree takes place in Item_subselect::fix_fields.
|
|
|
|
*/
|
2003-05-14 20:51:33 +02:00
|
|
|
if (!substitution)
|
2002-10-27 22:27:00 +01:00
|
|
|
{
|
2007-01-12 20:11:40 +01:00
|
|
|
/* We're invoked for the 1st (or the only) SELECT in the subquery UNION */
|
2005-03-10 13:01:22 +01:00
|
|
|
substitution= optimizer;
|
2003-05-14 20:51:33 +02:00
|
|
|
|
2008-02-13 20:27:12 +01:00
|
|
|
SELECT_LEX *current= thd->lex->current_select;
|
2003-07-03 01:30:52 +02:00
|
|
|
|
2008-02-13 20:27:12 +01:00
|
|
|
thd->lex->current_select= current->return_after_parsing();
|
2013-02-26 00:20:17 +01:00
|
|
|
if (!optimizer || optimizer->fix_left(thd))
|
2002-12-23 17:25:25 +01:00
|
|
|
{
|
2004-02-08 19:14:13 +01:00
|
|
|
thd->lex->current_select= current;
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2002-12-23 17:25:25 +01:00
|
|
|
}
|
2004-02-08 19:14:13 +01:00
|
|
|
thd->lex->current_select= current;
|
2002-12-23 17:25:25 +01:00
|
|
|
|
2010-01-17 15:51:10 +01:00
|
|
|
/* We will refer to upper level cache array => we have to save it for SP */
|
|
|
|
optimizer->keep_top_level_cache();
|
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
/*
|
2014-11-14 13:46:21 +01:00
|
|
|
As far as Item_in_optimizer does not substitute itself on fix_fields
|
2003-05-14 20:51:33 +02:00
|
|
|
we can use same item for all selects.
|
|
|
|
*/
|
2015-08-20 14:24:13 +02:00
|
|
|
expr= new (thd->mem_root) Item_direct_ref(thd, &select_lex->context,
|
2005-07-01 06:05:42 +02:00
|
|
|
(Item**)optimizer->get_cache(),
|
2017-04-23 18:39:57 +02:00
|
|
|
"<no matter>",
|
|
|
|
&in_left_expr_name);
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2011-05-04 17:08:58 +02:00
|
|
|
DBUG_RETURN(false);
|
|
|
|
}
|
|
|
|
|
2011-05-11 23:14:15 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
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)
|
2011-05-04 17:08:58 +02:00
|
|
|
{
|
2011-05-11 23:14:15 +02:00
|
|
|
DBUG_ENTER("Item_allany_subselect::transform_into_max_min");
|
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
******
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
2011-11-12 10:29:12 +01:00
|
|
|
if (!test_strategy(SUBS_MAXMIN_INJECTED | SUBS_MAXMIN_ENGINE))
|
2011-05-11 23:14:15 +02:00
|
|
|
DBUG_RETURN(false);
|
2011-05-04 17:08:58 +02:00
|
|
|
Item **place= optimizer->arguments() + 1;
|
|
|
|
SELECT_LEX *select_lex= join->select_lex;
|
|
|
|
Item *subs;
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(thd == join->thd);
|
2011-05-04 17:08:58 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
*/
|
|
|
|
DBUG_ASSERT(!substitution);
|
|
|
|
|
2011-09-02 14:10:10 +02:00
|
|
|
/*
|
|
|
|
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*/
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
{
|
2019-08-07 12:13:44 +02:00
|
|
|
Item_sum_min_max *item;
|
2011-05-04 17:08:58 +02:00
|
|
|
nesting_map save_allow_sum_func;
|
|
|
|
if (func->l_op())
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
(ALL && (> || =>)) || (ANY && (< || =<))
|
|
|
|
for ALL condition is inverted
|
|
|
|
*/
|
2016-05-08 22:04:41 +02:00
|
|
|
item= new (thd->mem_root) Item_sum_max(thd,
|
|
|
|
select_lex->ref_pointer_array[0]);
|
2011-05-04 17:08:58 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
(ALL && (< || =<)) || (ANY && (> || =>))
|
|
|
|
for ALL condition is inverted
|
|
|
|
*/
|
2016-05-08 22:04:41 +02:00
|
|
|
item= new (thd->mem_root) Item_sum_min(thd,
|
|
|
|
select_lex->ref_pointer_array[0]);
|
2011-05-04 17:08:58 +02:00
|
|
|
}
|
|
|
|
if (upper_item)
|
|
|
|
upper_item->set_sum_test(item);
|
2016-05-08 22:04:41 +02:00
|
|
|
thd->change_item_tree(&select_lex->ref_pointer_array[0], item);
|
2011-05-04 17:08:58 +02:00
|
|
|
{
|
|
|
|
List_iterator<Item> it(select_lex->item_list);
|
|
|
|
it++;
|
|
|
|
thd->change_item_tree(it.ref(), item);
|
|
|
|
}
|
|
|
|
|
2011-12-11 10:34:44 +01:00
|
|
|
DBUG_EXECUTE("where",
|
|
|
|
print_where(item, "rewrite with MIN/MAX", QT_ORDINARY););
|
|
|
|
|
2011-05-04 17:08:58 +02:00
|
|
|
save_allow_sum_func= thd->lex->allow_sum_func;
|
2018-07-05 17:49:44 +02:00
|
|
|
thd->lex->allow_sum_func.set_bit(thd->lex->current_select->nest_level);
|
2011-03-03 22:48:31 +01:00
|
|
|
/*
|
2011-05-04 17:08:58 +02:00
|
|
|
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
|
2011-03-03 22:48:31 +01:00
|
|
|
*/
|
2011-05-04 17:08:58 +02:00
|
|
|
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);
|
2015-04-22 11:29:56 +02:00
|
|
|
subs= new (thd->mem_root) Item_singlerow_subselect(thd, select_lex);
|
2002-10-31 01:11:59 +01:00
|
|
|
|
2011-07-18 22:45:38 +02:00
|
|
|
/*
|
|
|
|
Remove other strategies if any (we already changed the query and
|
|
|
|
can't apply other strategy).
|
|
|
|
*/
|
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
******
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
2011-11-12 10:29:12 +01:00
|
|
|
set_strategy(SUBS_MAXMIN_INJECTED);
|
2011-05-04 17:08:58 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Item_maxmin_subselect *item;
|
2015-08-20 14:24:13 +02:00
|
|
|
subs= item= new (thd->mem_root) Item_maxmin_subselect(thd, this, select_lex, func->l_op());
|
2011-05-04 17:08:58 +02:00
|
|
|
if (upper_item)
|
|
|
|
upper_item->set_sub_test(item);
|
2011-07-18 22:45:38 +02:00
|
|
|
/*
|
|
|
|
Remove other strategies if any (we already changed the query and
|
|
|
|
can't apply other strategy).
|
|
|
|
*/
|
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
******
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
2011-11-12 10:29:12 +01:00
|
|
|
set_strategy(SUBS_MAXMIN_ENGINE);
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
2010-01-28 14:48:33 +01:00
|
|
|
/*
|
Fix of LP BUG#780386.
ALL subquery should return TRUE if subquery rowa set is empty independently
of left part. The problem was that Item_func_(eq,ne,gt,ge,lt,le) do not
call execution of second argument if first is NULL no in this case subquery
will not be executed and when Item_func_not_all calls any_value() of the
subquery or aggregation function which report that there was rows. So for
NULL < ALL (SELECT...) result was FALSE instead of TRUE.
Fix is just swapping of arguments of Item_func_(eq,ne,gt,ge,lt,le) (with
changing the operation if it is needed) so that result will be the same
(for examole a < b is equal to b > a). This fix exploit the fact that
first argument will be executed in any case.
mysql-test/r/subselect.result:
The test suite added.
mysql-test/r/subselect_no_mat.result:
The test suite added.
mysql-test/r/subselect_no_opts.result:
The test suite added.
mysql-test/r/subselect_no_semijoin.result:
The test suite added.
mysql-test/r/subselect_scache.result:
The test suite added.
mysql-test/t/subselect.test:
The test suite added.
sql/item_cmpfunc.cc:
Swap arguments creation methods added.
sql/item_cmpfunc.h:
Swap arguments creation methods added.
sql/item_subselect.cc:
Swap arguments of the comparison.
2011-07-20 20:48:41 +02:00
|
|
|
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.
|
2010-01-28 14:48:33 +01:00
|
|
|
*/
|
2015-08-11 09:18:38 +02:00
|
|
|
subs= func->create_swap(thd, *(optimizer->get_cache()), subs);
|
2011-05-04 17:08:58 +02:00
|
|
|
thd->change_item_tree(place, subs);
|
|
|
|
if (subs->fix_fields(thd, &subs))
|
2011-05-11 23:14:15 +02:00
|
|
|
DBUG_RETURN(true);
|
2011-05-04 17:08:58 +02:00
|
|
|
DBUG_ASSERT(subs == (*place)); // There was no substitutions
|
|
|
|
|
|
|
|
select_lex->master_unit()->uncacheable&= ~UNCACHEABLE_DEPENDENT_INJECTED;
|
|
|
|
select_lex->uncacheable&= ~UNCACHEABLE_DEPENDENT_INJECTED;
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(false);
|
2010-01-28 14:48:33 +01:00
|
|
|
}
|
2002-10-31 01:11:59 +01:00
|
|
|
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-09-30 17:32:44 +02:00
|
|
|
bool Item_in_subselect::fix_having(Item *having, SELECT_LEX *select_lex)
|
|
|
|
{
|
|
|
|
bool fix_res= 0;
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(thd);
|
2010-09-30 17:32:44 +02:00
|
|
|
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;
|
2010-01-28 14:48:33 +01:00
|
|
|
}
|
|
|
|
|
2011-05-04 17:08:58 +02:00
|
|
|
bool Item_allany_subselect::is_maxmin_applicable(JOIN *join)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
Check if max/min optimization applicable: It is top item of
|
|
|
|
WHERE condition.
|
|
|
|
*/
|
2011-08-17 13:10:32 +02:00
|
|
|
return (abort_on_null || (upper_item && upper_item->is_top_level_item())) &&
|
2012-06-19 11:53:16 +02:00
|
|
|
!(join->select_lex->master_unit()->uncacheable & ~UNCACHEABLE_EXPLAIN) && !func->eqne_op();
|
2010-01-28 14:48:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2010-10-05 15:00:31 +02:00
|
|
|
Create the predicates needed to transform a single-column IN/ALL/ANY
|
|
|
|
subselect into a correlated EXISTS via predicate injection.
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
@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
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
@details
|
|
|
|
The correlated predicates are created as follows:
|
2010-01-28 14:48:33 +01:00
|
|
|
|
|
|
|
- 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:
|
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
SELECT ie FROM ... WHERE subq_where AND (oe $cmp$ ie)
|
2010-01-28 14:48:33 +01:00
|
|
|
|
|
|
|
= If we need to distinguish between those:
|
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
SELECT ie FROM ...
|
2010-01-28 14:48:33 +01:00
|
|
|
WHERE subq_where AND trigcond((oe $cmp$ ie) OR (ie IS NULL))
|
|
|
|
HAVING trigcond(<is_not_null_test>(ie))
|
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
@retval false If the new conditions were created successfully
|
|
|
|
@retval true Error
|
2010-01-28 14:48:33 +01:00
|
|
|
*/
|
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
bool
|
2015-06-25 22:34:54 +02:00
|
|
|
Item_in_subselect::create_single_in_to_exists_cond(JOIN *join,
|
2010-09-05 17:43:47 +02:00
|
|
|
Item **where_item,
|
|
|
|
Item **having_item)
|
2010-01-28 14:48:33 +01:00
|
|
|
{
|
|
|
|
SELECT_LEX *select_lex= join->select_lex;
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(thd == join->thd);
|
2011-03-03 22:48:31 +01:00
|
|
|
/*
|
2010-10-05 15:00:31 +02:00
|
|
|
The non-transformed HAVING clause of 'join' may be stored in two ways
|
|
|
|
during JOIN::optimize: this->tmp_having= this->having; this->having= 0;
|
2011-03-03 22:48:31 +01:00
|
|
|
*/
|
2010-09-05 17:43:47 +02:00
|
|
|
Item* join_having= join->having ? join->having : join->tmp_having;
|
|
|
|
DBUG_ENTER("Item_in_subselect::create_single_in_to_exists_cond");
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-09-30 17:32:44 +02:00
|
|
|
*where_item= NULL;
|
|
|
|
*having_item= NULL;
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-09-05 17:43:47 +02:00
|
|
|
if (join_having || select_lex->with_sum_func ||
|
2003-05-14 20:51:33 +02:00
|
|
|
select_lex->group_list.elements)
|
|
|
|
{
|
2017-04-23 18:39:57 +02:00
|
|
|
const char *tmp= this->full_name();
|
|
|
|
LEX_CSTRING field_name= {tmp, safe_strlen(tmp)};
|
2015-08-11 09:18:38 +02:00
|
|
|
Item *item= func->create(thd, expr,
|
2015-04-17 12:30:15 +02:00
|
|
|
new (thd->mem_root) Item_ref_null_helper(
|
2015-08-11 09:18:38 +02:00
|
|
|
thd,
|
2015-04-17 12:30:15 +02:00
|
|
|
&select_lex->context,
|
2005-07-01 06:05:42 +02:00
|
|
|
this,
|
2016-05-08 22:04:41 +02:00
|
|
|
&select_lex->
|
|
|
|
ref_pointer_array[0],
|
2004-09-06 14:14:10 +02:00
|
|
|
(char *)"<ref>",
|
2017-04-23 18:39:57 +02:00
|
|
|
&field_name));
|
2007-01-12 20:11:40 +01:00
|
|
|
if (!abort_on_null && left_expr->maybe_null)
|
2006-10-31 18:51:09 +01:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
We can encounter "NULL IN (SELECT ...)". Wrap the added condition
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
within a trig_cond.
|
2006-10-31 18:51:09 +01:00
|
|
|
*/
|
2017-03-05 06:28:05 +01:00
|
|
|
disable_cond_guard_for_const_null_left_expr(0);
|
2015-08-20 14:24:13 +02:00
|
|
|
item= new (thd->mem_root) Item_func_trig_cond(thd, item, get_cond_guard(0));
|
2006-10-31 18:51:09 +01:00
|
|
|
}
|
2010-07-18 13:46:08 +02:00
|
|
|
|
2010-09-30 17:32:44 +02:00
|
|
|
if (!join_having)
|
2017-04-23 18:39:57 +02:00
|
|
|
item->name= in_having_cond;
|
2010-09-30 17:32:44 +02:00
|
|
|
if (fix_having(item, select_lex))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2010-07-18 14:59:24 +02:00
|
|
|
*having_item= item;
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-01-29 09:44:17 +01:00
|
|
|
Item *item= (Item*) select_lex->item_list.head();
|
|
|
|
if (item->type() != REF_ITEM ||
|
|
|
|
((Item_ref*)item)->ref_type() != Item_ref::VIEW_REF)
|
|
|
|
item= item->real_item();
|
2004-09-06 14:14:10 +02:00
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
if (select_lex->table_list.elements)
|
2002-10-31 01:11:59 +01:00
|
|
|
{
|
2010-07-18 13:46:08 +02:00
|
|
|
Item *having= item;
|
|
|
|
Item *orig_item= item;
|
2006-10-31 18:51:09 +01:00
|
|
|
|
2015-08-11 09:18:38 +02:00
|
|
|
item= func->create(thd, expr, item);
|
2004-02-02 01:23:53 +01:00
|
|
|
if (!abort_on_null && orig_item->maybe_null)
|
2002-11-28 18:29:26 +01:00
|
|
|
{
|
2015-08-11 09:18:38 +02:00
|
|
|
having= new (thd->mem_root) Item_is_not_null_test(thd, this, having);
|
2007-01-12 20:11:40 +01:00
|
|
|
if (left_expr->maybe_null)
|
|
|
|
{
|
2017-03-05 06:28:05 +01:00
|
|
|
disable_cond_guard_for_const_null_left_expr(0);
|
2015-08-11 09:18:38 +02:00
|
|
|
if (!(having= new (thd->mem_root) Item_func_trig_cond(thd, having,
|
2015-04-17 12:30:15 +02:00
|
|
|
get_cond_guard(0))))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2007-01-12 20:11:40 +01:00
|
|
|
}
|
2017-04-23 18:39:57 +02:00
|
|
|
having->name= in_having_cond;
|
2010-09-30 17:32:44 +02:00
|
|
|
if (fix_having(having, select_lex))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2010-07-18 14:59:24 +02:00
|
|
|
*having_item= having;
|
2010-07-18 13:46:08 +02:00
|
|
|
|
2015-08-11 09:18:38 +02:00
|
|
|
item= new (thd->mem_root) Item_cond_or(thd, item,
|
|
|
|
new (thd->mem_root) Item_func_isnull(thd, orig_item));
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
2007-01-12 20:11:40 +01:00
|
|
|
/*
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
If we may encounter NULL IN (SELECT ...) and care whether subquery
|
|
|
|
result is NULL or FALSE, wrap condition in a trig_cond.
|
2007-01-12 20:11:40 +01:00
|
|
|
*/
|
|
|
|
if (!abort_on_null && left_expr->maybe_null)
|
|
|
|
{
|
2017-03-05 06:28:05 +01:00
|
|
|
disable_cond_guard_for_const_null_left_expr(0);
|
2015-08-11 09:18:38 +02:00
|
|
|
if (!(item= new (thd->mem_root) Item_func_trig_cond(thd, item,
|
2015-04-17 12:30:15 +02:00
|
|
|
get_cond_guard(0))))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2007-01-12 20:11:40 +01:00
|
|
|
}
|
2010-07-18 13:46:08 +02:00
|
|
|
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
/*
|
|
|
|
TODO: figure out why the following is done here in
|
|
|
|
single_value_transformer but there is no corresponding action in
|
|
|
|
row_value_transformer?
|
|
|
|
*/
|
2017-04-23 18:39:57 +02:00
|
|
|
item->name= in_additional_cond;
|
2018-06-05 08:25:39 +02:00
|
|
|
if (item->fix_fields_if_needed(thd, 0))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2010-07-18 14:59:24 +02:00
|
|
|
*where_item= item;
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-03-14 11:52:00 +01:00
|
|
|
if (select_lex->master_unit()->is_unit_op())
|
2003-05-14 20:51:33 +02:00
|
|
|
{
|
2017-04-23 18:39:57 +02:00
|
|
|
LEX_CSTRING field_name= {STRING_WITH_LEN("<result>") };
|
2006-10-31 18:51:09 +01:00
|
|
|
Item *new_having=
|
2015-08-11 09:18:38 +02:00
|
|
|
func->create(thd, expr,
|
|
|
|
new (thd->mem_root) Item_ref_null_helper(thd,
|
2015-04-17 12:30:15 +02:00
|
|
|
&select_lex->context,
|
|
|
|
this,
|
2016-05-08 22:04:41 +02:00
|
|
|
&select_lex->ref_pointer_array[0],
|
2015-04-17 12:30:15 +02:00
|
|
|
(char *)"<no matter>",
|
2017-04-23 18:39:57 +02:00
|
|
|
&field_name));
|
2007-01-12 20:11:40 +01:00
|
|
|
if (!abort_on_null && left_expr->maybe_null)
|
|
|
|
{
|
2017-03-05 06:28:05 +01:00
|
|
|
disable_cond_guard_for_const_null_left_expr(0);
|
2015-08-11 09:18:38 +02:00
|
|
|
if (!(new_having= new (thd->mem_root) Item_func_trig_cond(thd, new_having,
|
2015-04-17 12:30:15 +02:00
|
|
|
get_cond_guard(0))))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2007-01-12 20:11:40 +01:00
|
|
|
}
|
2010-07-18 13:46:08 +02:00
|
|
|
|
2017-04-23 18:39:57 +02:00
|
|
|
new_having->name= in_having_cond;
|
2010-09-30 17:32:44 +02:00
|
|
|
if (fix_having(new_having, select_lex))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2010-07-18 14:59:24 +02:00
|
|
|
*having_item= new_having;
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
|
|
|
else
|
2011-05-18 00:23:22 +02:00
|
|
|
DBUG_ASSERT(false);
|
2002-10-31 01:11:59 +01:00
|
|
|
}
|
2002-10-27 22:27:00 +01:00
|
|
|
}
|
2004-02-08 19:14:13 +01:00
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(false);
|
2002-10-27 22:27:00 +01:00
|
|
|
}
|
2002-09-03 08:50:36 +02:00
|
|
|
|
2003-11-28 11:18:13 +01:00
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
/**
|
|
|
|
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.
|
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
@retval false The subquery was transformed
|
|
|
|
@retval true Error
|
2010-10-05 15:00:31 +02:00
|
|
|
*/
|
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
bool
|
2003-11-02 16:27:35 +01:00
|
|
|
Item_in_subselect::row_value_transformer(JOIN *join)
|
2002-12-19 20:15:09 +01:00
|
|
|
{
|
2004-09-06 14:14:10 +02:00
|
|
|
SELECT_LEX *select_lex= join->select_lex;
|
2005-08-13 06:45:14 +02:00
|
|
|
uint cols_num= left_expr->cols();
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2002-12-19 20:15:09 +01:00
|
|
|
DBUG_ENTER("Item_in_subselect::row_value_transformer");
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(thd == join->thd);
|
2002-12-31 17:39:16 +01:00
|
|
|
|
2010-01-28 14:48:33 +01:00
|
|
|
// psergey: duplicated_subselect_card_check
|
2010-02-19 22:55:57 +01:00
|
|
|
if (select_lex->item_list.elements != cols_num)
|
2003-10-23 22:54:21 +02:00
|
|
|
{
|
2010-02-19 22:55:57 +01:00
|
|
|
my_error(ER_OPERAND_COLUMNS, MYF(0), cols_num);
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2003-10-23 22:54:21 +02:00
|
|
|
}
|
|
|
|
|
2010-01-28 14:48:33 +01:00
|
|
|
/*
|
|
|
|
Wrap the current IN predicate in an Item_in_optimizer. The actual
|
|
|
|
substitution in the Item tree takes place in Item_subselect::fix_fields.
|
|
|
|
*/
|
2003-05-14 20:51:33 +02:00
|
|
|
if (!substitution)
|
2002-12-19 20:15:09 +01:00
|
|
|
{
|
2003-05-14 20:51:33 +02:00
|
|
|
//first call for this unit
|
2006-12-14 23:51:37 +01:00
|
|
|
SELECT_LEX_UNIT *master_unit= select_lex->master_unit();
|
2005-03-10 13:01:22 +01:00
|
|
|
substitution= optimizer;
|
2003-05-14 20:51:33 +02:00
|
|
|
|
2008-02-13 20:27:12 +01:00
|
|
|
SELECT_LEX *current= thd->lex->current_select;
|
|
|
|
thd->lex->current_select= current->return_after_parsing();
|
2013-02-26 00:20:17 +01:00
|
|
|
if (!optimizer || optimizer->fix_left(thd))
|
2002-12-25 11:03:08 +01:00
|
|
|
{
|
2004-02-08 19:14:13 +01:00
|
|
|
thd->lex->current_select= current;
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2002-12-25 11:03:08 +01:00
|
|
|
}
|
2004-02-12 02:10:26 +01:00
|
|
|
|
2005-02-08 23:50:45 +01:00
|
|
|
// we will refer to upper level cache array => we have to save it in PS
|
2004-02-12 02:10:26 +01:00
|
|
|
optimizer->keep_top_level_cache();
|
|
|
|
|
2004-02-08 19:14:13 +01:00
|
|
|
thd->lex->current_select= current;
|
2011-03-03 22:48:31 +01:00
|
|
|
/*
|
|
|
|
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
|
2016-06-08 13:14:42 +02:00
|
|
|
plans with a temp table where the original JOIN was overridden by
|
2011-03-03 22:48:31 +01:00
|
|
|
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;
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(false);
|
2010-01-28 14:48:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2010-10-05 15:00:31 +02:00
|
|
|
Create the predicates needed to transform a multi-column IN/ALL/ANY
|
|
|
|
subselect into a correlated EXISTS via predicate injection.
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
@details
|
2010-12-15 11:54:25 +01:00
|
|
|
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)
|
2010-10-05 15:00:31 +02:00
|
|
|
|
|
|
|
@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
|
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
@retval false If the new conditions were created successfully
|
|
|
|
@retval true Error
|
2010-01-28 14:48:33 +01:00
|
|
|
*/
|
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
bool
|
2010-09-05 17:43:47 +02:00
|
|
|
Item_in_subselect::create_row_in_to_exists_cond(JOIN * join,
|
|
|
|
Item **where_item,
|
|
|
|
Item **having_item)
|
2010-01-28 14:48:33 +01:00
|
|
|
{
|
|
|
|
SELECT_LEX *select_lex= join->select_lex;
|
|
|
|
uint cols_num= left_expr->cols();
|
2010-09-05 17:43:47 +02:00
|
|
|
/*
|
2010-10-05 15:00:31 +02:00
|
|
|
The non-transformed HAVING clause of 'join' may be stored in two ways
|
|
|
|
during JOIN::optimize: this->tmp_having= this->having; this->having= 0;
|
2010-09-05 17:43:47 +02:00
|
|
|
*/
|
|
|
|
Item* join_having= join->having ? join->having : join->tmp_having;
|
|
|
|
bool is_having_used= (join_having || select_lex->with_sum_func ||
|
2010-01-28 14:48:33 +01:00
|
|
|
select_lex->group_list.first ||
|
|
|
|
!select_lex->table_list.elements);
|
2017-04-23 18:39:57 +02:00
|
|
|
LEX_CSTRING list_ref= { STRING_WITH_LEN("<list ref>")};
|
2010-09-05 17:43:47 +02:00
|
|
|
DBUG_ENTER("Item_in_subselect::create_row_in_to_exists_cond");
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(thd == join->thd);
|
2010-07-18 14:59:24 +02:00
|
|
|
|
|
|
|
*where_item= NULL;
|
|
|
|
*having_item= NULL;
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2005-08-13 06:45:14 +02:00
|
|
|
if (is_having_used)
|
2004-02-08 19:14:13 +01:00
|
|
|
{
|
2010-12-15 11:54:25 +01:00
|
|
|
/* TODO: say here explicitly if the order of AND parts matters or not. */
|
2005-08-13 06:45:14 +02:00
|
|
|
Item *item_having_part2= 0;
|
|
|
|
for (uint i= 0; i < cols_num; i++)
|
2004-02-08 19:14:13 +01:00
|
|
|
{
|
2009-06-09 14:55:30 +02:00
|
|
|
DBUG_ASSERT((left_expr->fixed &&
|
2010-07-18 14:59:24 +02:00
|
|
|
|
2009-06-09 14:55:30 +02:00
|
|
|
select_lex->ref_pointer_array[i]->fixed) ||
|
2008-03-14 20:11:59 +01:00
|
|
|
(select_lex->ref_pointer_array[i]->type() == REF_ITEM &&
|
|
|
|
((Item_ref*)(select_lex->ref_pointer_array[i]))->ref_type() ==
|
|
|
|
Item_ref::OUTER_REF));
|
2005-01-24 14:56:57 +01:00
|
|
|
if (select_lex->ref_pointer_array[i]->
|
2006-12-14 23:51:37 +01:00
|
|
|
check_cols(left_expr->element_index(i)->cols()))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2017-04-23 18:39:57 +02:00
|
|
|
|
2005-08-13 06:45:14 +02:00
|
|
|
Item *item_eq=
|
2015-08-20 14:24:13 +02:00
|
|
|
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>",
|
2017-04-23 18:39:57 +02:00
|
|
|
&in_left_expr_name),
|
2015-08-20 14:24:13 +02:00
|
|
|
new (thd->mem_root)
|
|
|
|
Item_ref(thd, &select_lex->context,
|
2016-05-08 22:04:41 +02:00
|
|
|
&select_lex->ref_pointer_array[i],
|
2015-08-20 14:24:13 +02:00
|
|
|
(char *)"<no matter>",
|
2017-04-23 18:39:57 +02:00
|
|
|
&list_ref));
|
2015-08-20 14:24:13 +02:00
|
|
|
Item *item_isnull=
|
|
|
|
new (thd->mem_root)
|
|
|
|
Item_func_isnull(thd,
|
|
|
|
new (thd->mem_root)
|
2015-08-11 09:18:38 +02:00
|
|
|
Item_ref(thd, &select_lex->context,
|
2016-05-08 22:04:41 +02:00
|
|
|
&select_lex->ref_pointer_array[i],
|
2006-08-24 18:56:28 +02:00
|
|
|
(char *)"<no matter>",
|
2017-04-23 18:39:57 +02:00
|
|
|
&list_ref));
|
2015-08-20 14:24:13 +02:00
|
|
|
Item *col_item= new (thd->mem_root)
|
|
|
|
Item_cond_or(thd, item_eq, item_isnull);
|
Fixed the bug mdev-12564.
Here's what started happening after the patch that fixed
the bug mdev-10454 with query reported for the bug
SELECT * FROM t t1 right JOIN t t2 ON (t2.pk = t1.pk)
WHERE (t2.i, t2.pk) NOT IN ( SELECT t3.i, t3.i FROM t t3, t t4 )
AND t1.c = 'foo';
The patch added an implementation of propagate_equal_fields() for
the class Item_row and thus opened the possibility of equal fields
substitutions.
At the prepare stage after setup_conds() called for WHERE condition
had completed the flag of maybe_null of the Item_row object created
for (t2.i, t2.pk) was set to false, because the maybe_null flags of
both elements were set to false. However the flag of maybe_null for
t1.pk from the ON condition were set to true, because t1 was an inner
table of an outer join.
At the optimization stage the outer join was converted to inner join,
but the maybe_null flags were not corrected and remained the same.
So after the substitution t2.pk/t1.pk. the maybe_null flag for the
row remained false while the maybe_flag for the second element of
the row was true. As a result, when the in-to_exists transformation
was performed for the NOT IN predicate the guards variables were
not created for the elements of the row, but a guard object for
the second element was created. The object were not valid because
it referred to NULL as a guard variable. This ultimately caused
a crash when the expression with the guard was evaluated at the
execution stage.
The patch made sure that the guard objects are not created without
guard variables.
Yet it does not resolve the problem of inconsistent maybe_null flags.
and it might be that the problem will pop op in other pieces of code.
The resolution of this problem is not easy, but the problem should
be resolved in future versions.
2017-04-24 23:56:53 +02:00
|
|
|
if (!abort_on_null && left_expr->element_index(i)->maybe_null &&
|
|
|
|
get_cond_guard(i))
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
{
|
2017-03-05 06:28:05 +01:00
|
|
|
disable_cond_guard_for_const_null_left_expr(i);
|
2015-08-20 14:24:13 +02:00
|
|
|
if (!(col_item= new (thd->mem_root)
|
|
|
|
Item_func_trig_cond(thd, col_item, get_cond_guard(i))))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
}
|
2015-08-11 09:18:38 +02:00
|
|
|
*having_item= and_items(thd, *having_item, col_item);
|
2010-07-18 14:59:24 +02:00
|
|
|
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
Item *item_nnull_test=
|
2015-08-20 14:24:13 +02:00
|
|
|
new (thd->mem_root)
|
|
|
|
Item_is_not_null_test(thd, this,
|
|
|
|
new (thd->mem_root)
|
|
|
|
Item_ref(thd, &select_lex->context,
|
2016-05-08 22:04:41 +02:00
|
|
|
&select_lex->
|
|
|
|
ref_pointer_array[i],
|
2015-08-20 14:24:13 +02:00
|
|
|
(char *)"<no matter>",
|
2017-04-23 18:39:57 +02:00
|
|
|
&list_ref));
|
Fixed the bug mdev-12564.
Here's what started happening after the patch that fixed
the bug mdev-10454 with query reported for the bug
SELECT * FROM t t1 right JOIN t t2 ON (t2.pk = t1.pk)
WHERE (t2.i, t2.pk) NOT IN ( SELECT t3.i, t3.i FROM t t3, t t4 )
AND t1.c = 'foo';
The patch added an implementation of propagate_equal_fields() for
the class Item_row and thus opened the possibility of equal fields
substitutions.
At the prepare stage after setup_conds() called for WHERE condition
had completed the flag of maybe_null of the Item_row object created
for (t2.i, t2.pk) was set to false, because the maybe_null flags of
both elements were set to false. However the flag of maybe_null for
t1.pk from the ON condition were set to true, because t1 was an inner
table of an outer join.
At the optimization stage the outer join was converted to inner join,
but the maybe_null flags were not corrected and remained the same.
So after the substitution t2.pk/t1.pk. the maybe_null flag for the
row remained false while the maybe_flag for the second element of
the row was true. As a result, when the in-to_exists transformation
was performed for the NOT IN predicate the guards variables were
not created for the elements of the row, but a guard object for
the second element was created. The object were not valid because
it referred to NULL as a guard variable. This ultimately caused
a crash when the expression with the guard was evaluated at the
execution stage.
The patch made sure that the guard objects are not created without
guard variables.
Yet it does not resolve the problem of inconsistent maybe_null flags.
and it might be that the problem will pop op in other pieces of code.
The resolution of this problem is not easy, but the problem should
be resolved in future versions.
2017-04-24 23:56:53 +02:00
|
|
|
if (!abort_on_null && left_expr->element_index(i)->maybe_null &&
|
|
|
|
get_cond_guard(i) )
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
{
|
2017-03-05 06:28:05 +01:00
|
|
|
disable_cond_guard_for_const_null_left_expr(i);
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (!(item_nnull_test=
|
2015-08-20 14:24:13 +02:00
|
|
|
new (thd->mem_root)
|
|
|
|
Item_func_trig_cond(thd, item_nnull_test, get_cond_guard(i))))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
}
|
2015-08-11 09:18:38 +02:00
|
|
|
item_having_part2= and_items(thd, item_having_part2, item_nnull_test);
|
2005-08-13 06:45:14 +02:00
|
|
|
item_having_part2->top_level_item();
|
2004-02-08 19:14:13 +01:00
|
|
|
}
|
2015-08-11 09:18:38 +02:00
|
|
|
*having_item= and_items(thd, *having_item, item_having_part2);
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
2005-08-13 06:45:14 +02:00
|
|
|
else
|
2003-05-14 20:51:33 +02:00
|
|
|
{
|
2005-08-13 06:45:14 +02:00
|
|
|
for (uint i= 0; i < cols_num; i++)
|
|
|
|
{
|
|
|
|
Item *item, *item_isnull;
|
2009-06-09 14:55:30 +02:00
|
|
|
DBUG_ASSERT((left_expr->fixed &&
|
|
|
|
select_lex->ref_pointer_array[i]->fixed) ||
|
2008-03-14 20:11:59 +01:00
|
|
|
(select_lex->ref_pointer_array[i]->type() == REF_ITEM &&
|
|
|
|
((Item_ref*)(select_lex->ref_pointer_array[i]))->ref_type() ==
|
|
|
|
Item_ref::OUTER_REF));
|
2005-08-13 06:45:14 +02:00
|
|
|
if (select_lex->ref_pointer_array[i]->
|
2006-12-14 23:51:37 +01:00
|
|
|
check_cols(left_expr->element_index(i)->cols()))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2015-08-20 14:24:13 +02:00
|
|
|
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>",
|
2017-04-23 18:39:57 +02:00
|
|
|
&in_left_expr_name),
|
2015-08-20 14:24:13 +02:00
|
|
|
new (thd->mem_root)
|
|
|
|
Item_direct_ref(thd, &select_lex->context,
|
2016-05-08 22:04:41 +02:00
|
|
|
&select_lex->
|
|
|
|
ref_pointer_array[i],
|
2015-08-20 14:24:13 +02:00
|
|
|
(char *)"<no matter>",
|
2017-04-23 18:39:57 +02:00
|
|
|
&list_ref));
|
2013-02-07 14:33:24 +01:00
|
|
|
if (!abort_on_null && select_lex->ref_pointer_array[i]->maybe_null)
|
2005-08-13 06:45:14 +02:00
|
|
|
{
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
Item *having_col_item=
|
2015-08-20 14:24:13 +02:00
|
|
|
new (thd->mem_root)
|
|
|
|
Item_is_not_null_test(thd, this,
|
|
|
|
new (thd->mem_root)
|
|
|
|
Item_ref(thd, &select_lex->context,
|
2016-05-08 22:04:41 +02:00
|
|
|
&select_lex->ref_pointer_array[i],
|
2015-08-20 14:24:13 +02:00
|
|
|
(char *)"<no matter>",
|
2017-04-23 18:39:57 +02:00
|
|
|
&list_ref));
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
|
2015-08-20 14:24:13 +02:00
|
|
|
item_isnull= new (thd->mem_root)
|
|
|
|
Item_func_isnull(thd,
|
|
|
|
new (thd->mem_root)
|
2015-08-11 09:18:38 +02:00
|
|
|
Item_direct_ref(thd, &select_lex->context,
|
2016-05-08 22:04:41 +02:00
|
|
|
&select_lex->
|
|
|
|
ref_pointer_array[i],
|
2005-08-13 10:15:17 +02:00
|
|
|
(char *)"<no matter>",
|
2017-04-23 18:39:57 +02:00
|
|
|
&list_ref));
|
2015-08-20 14:24:13 +02:00
|
|
|
item= new (thd->mem_root) Item_cond_or(thd, item, item_isnull);
|
Fixed the bug mdev-12564.
Here's what started happening after the patch that fixed
the bug mdev-10454 with query reported for the bug
SELECT * FROM t t1 right JOIN t t2 ON (t2.pk = t1.pk)
WHERE (t2.i, t2.pk) NOT IN ( SELECT t3.i, t3.i FROM t t3, t t4 )
AND t1.c = 'foo';
The patch added an implementation of propagate_equal_fields() for
the class Item_row and thus opened the possibility of equal fields
substitutions.
At the prepare stage after setup_conds() called for WHERE condition
had completed the flag of maybe_null of the Item_row object created
for (t2.i, t2.pk) was set to false, because the maybe_null flags of
both elements were set to false. However the flag of maybe_null for
t1.pk from the ON condition were set to true, because t1 was an inner
table of an outer join.
At the optimization stage the outer join was converted to inner join,
but the maybe_null flags were not corrected and remained the same.
So after the substitution t2.pk/t1.pk. the maybe_null flag for the
row remained false while the maybe_flag for the second element of
the row was true. As a result, when the in-to_exists transformation
was performed for the NOT IN predicate the guards variables were
not created for the elements of the row, but a guard object for
the second element was created. The object were not valid because
it referred to NULL as a guard variable. This ultimately caused
a crash when the expression with the guard was evaluated at the
execution stage.
The patch made sure that the guard objects are not created without
guard variables.
Yet it does not resolve the problem of inconsistent maybe_null flags.
and it might be that the problem will pop op in other pieces of code.
The resolution of this problem is not easy, but the problem should
be resolved in future versions.
2017-04-24 23:56:53 +02:00
|
|
|
if (left_expr->element_index(i)->maybe_null && get_cond_guard(i))
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
{
|
2017-03-05 06:28:05 +01:00
|
|
|
disable_cond_guard_for_const_null_left_expr(i);
|
2015-08-20 14:24:13 +02:00
|
|
|
if (!(item= new (thd->mem_root)
|
|
|
|
Item_func_trig_cond(thd, item, get_cond_guard(i))))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2015-08-20 14:24:13 +02:00
|
|
|
if (!(having_col_item= new (thd->mem_root)
|
|
|
|
Item_func_trig_cond(thd, having_col_item, get_cond_guard(i))))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
}
|
2015-08-11 09:18:38 +02:00
|
|
|
*having_item= and_items(thd, *having_item, having_col_item);
|
2005-08-13 06:45:14 +02:00
|
|
|
}
|
Fixed the bug mdev-12564.
Here's what started happening after the patch that fixed
the bug mdev-10454 with query reported for the bug
SELECT * FROM t t1 right JOIN t t2 ON (t2.pk = t1.pk)
WHERE (t2.i, t2.pk) NOT IN ( SELECT t3.i, t3.i FROM t t3, t t4 )
AND t1.c = 'foo';
The patch added an implementation of propagate_equal_fields() for
the class Item_row and thus opened the possibility of equal fields
substitutions.
At the prepare stage after setup_conds() called for WHERE condition
had completed the flag of maybe_null of the Item_row object created
for (t2.i, t2.pk) was set to false, because the maybe_null flags of
both elements were set to false. However the flag of maybe_null for
t1.pk from the ON condition were set to true, because t1 was an inner
table of an outer join.
At the optimization stage the outer join was converted to inner join,
but the maybe_null flags were not corrected and remained the same.
So after the substitution t2.pk/t1.pk. the maybe_null flag for the
row remained false while the maybe_flag for the second element of
the row was true. As a result, when the in-to_exists transformation
was performed for the NOT IN predicate the guards variables were
not created for the elements of the row, but a guard object for
the second element was created. The object were not valid because
it referred to NULL as a guard variable. This ultimately caused
a crash when the expression with the guard was evaluated at the
execution stage.
The patch made sure that the guard objects are not created without
guard variables.
Yet it does not resolve the problem of inconsistent maybe_null flags.
and it might be that the problem will pop op in other pieces of code.
The resolution of this problem is not easy, but the problem should
be resolved in future versions.
2017-04-24 23:56:53 +02:00
|
|
|
if (!abort_on_null && left_expr->element_index(i)->maybe_null &&
|
|
|
|
get_cond_guard(i))
|
2013-02-07 14:33:24 +01:00
|
|
|
{
|
2015-08-20 14:24:13 +02:00
|
|
|
if (!(item= new (thd->mem_root)
|
|
|
|
Item_func_trig_cond(thd, item, get_cond_guard(i))))
|
2013-02-07 14:33:24 +01:00
|
|
|
DBUG_RETURN(true);
|
|
|
|
}
|
2015-08-11 09:18:38 +02:00
|
|
|
*where_item= and_items(thd, *where_item, item);
|
2005-08-13 06:45:14 +02:00
|
|
|
}
|
2002-12-19 20:15:09 +01:00
|
|
|
}
|
2010-07-18 14:59:24 +02:00
|
|
|
|
2010-09-30 17:32:44 +02:00
|
|
|
if (*where_item)
|
2003-05-14 20:51:33 +02:00
|
|
|
{
|
2018-06-05 08:25:39 +02:00
|
|
|
if ((*where_item)->fix_fields_if_needed(thd, 0))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2010-09-30 17:32:44 +02:00
|
|
|
(*where_item)->top_level_item();
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
2004-02-08 19:14:13 +01:00
|
|
|
|
2010-09-30 17:32:44 +02:00
|
|
|
if (*having_item)
|
2003-05-14 20:51:33 +02:00
|
|
|
{
|
2010-09-30 17:32:44 +02:00
|
|
|
if (!join_having)
|
2017-04-23 18:39:57 +02:00
|
|
|
(*having_item)->name= in_having_cond;
|
2010-09-30 17:32:44 +02:00
|
|
|
if (fix_having(*having_item, select_lex))
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2010-09-30 17:32:44 +02:00
|
|
|
(*having_item)->top_level_item();
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
2004-02-08 19:14:13 +01:00
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(false);
|
2002-12-19 20:15:09 +01:00
|
|
|
}
|
|
|
|
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
bool
|
2003-07-02 00:45:22 +02:00
|
|
|
Item_in_subselect::select_transformer(JOIN *join)
|
2002-11-07 22:45:19 +01:00
|
|
|
{
|
2010-09-05 17:43:47 +02:00
|
|
|
return select_in_like_transformer(join);
|
2005-03-10 13:01:22 +01:00
|
|
|
}
|
|
|
|
|
2013-02-26 00:20:17 +01:00
|
|
|
bool
|
|
|
|
Item_exists_subselect::select_transformer(JOIN *join)
|
|
|
|
{
|
|
|
|
return select_prepare_to_be_in();
|
|
|
|
}
|
|
|
|
|
2005-03-10 13:01:22 +01:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
/**
|
2010-10-05 15:00:31 +02:00
|
|
|
Create the predicates needed to transform an IN/ALL/ANY subselect into a
|
|
|
|
correlated EXISTS via predicate injection.
|
2005-03-10 13:01:22 +01:00
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
@param join_arg Join object of the subquery.
|
2005-03-10 13:01:22 +01:00
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
@retval FALSE ok
|
|
|
|
@retval TRUE error
|
|
|
|
*/
|
|
|
|
|
2010-09-05 17:43:47 +02:00
|
|
|
bool Item_in_subselect::create_in_to_exists_cond(JOIN *join_arg)
|
|
|
|
{
|
2010-12-15 11:54:25 +01:00
|
|
|
bool res;
|
2010-09-05 17:43:47 +02:00
|
|
|
|
|
|
|
DBUG_ASSERT(engine->engine_type() == subselect_engine::SINGLE_SELECT_ENGINE ||
|
|
|
|
engine->engine_type() == subselect_engine::UNION_ENGINE);
|
|
|
|
/*
|
2010-10-05 15:00:31 +02:00
|
|
|
TODO: the call to init_cond_guards allocates and initializes an
|
2010-09-05 17:43:47 +02:00
|
|
|
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.
|
2010-10-05 15:00:31 +02:00
|
|
|
If the dependency is removed, the call can be moved to a later phase.
|
2010-09-05 17:43:47 +02:00
|
|
|
*/
|
|
|
|
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));
|
2011-05-04 17:08:58 +02:00
|
|
|
|
2011-05-11 23:14:15 +02:00
|
|
|
/*
|
|
|
|
The IN=>EXISTS transformation makes non-correlated subqueries correlated.
|
|
|
|
*/
|
2012-02-02 00:48:02 +01:00
|
|
|
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;
|
|
|
|
}
|
2011-05-04 17:08:58 +02:00
|
|
|
/*
|
|
|
|
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
|
2016-06-08 13:14:42 +02:00
|
|
|
plans with a temp table where the original JOIN was overridden by
|
2011-05-04 17:08:58 +02:00
|
|
|
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;
|
2010-12-15 11:54:25 +01:00
|
|
|
return (res);
|
2010-09-05 17:43:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
/**
|
|
|
|
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
|
|
|
|
*/
|
|
|
|
|
2010-09-30 17:32:44 +02:00
|
|
|
bool Item_in_subselect::inject_in_to_exists_cond(JOIN *join_arg)
|
2010-09-05 17:43:47 +02:00
|
|
|
{
|
2010-09-30 17:32:44 +02:00
|
|
|
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;
|
2010-09-05 17:43:47 +02:00
|
|
|
|
2010-09-30 17:32:44 +02:00
|
|
|
DBUG_ENTER("Item_in_subselect::inject_in_to_exists_cond");
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(thd == join_arg->thd);
|
2010-09-30 17:32:44 +02:00
|
|
|
|
2017-05-15 18:51:01 +02:00
|
|
|
if (select_lex->min_max_opt_list.elements)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
MIN/MAX optimizations have been applied to Item_sum objects
|
|
|
|
of the subquery this subquery predicate in opt_sum_query().
|
|
|
|
Injection of new condition invalidates this optimizations.
|
|
|
|
Thus those optimizations must be rolled back.
|
|
|
|
*/
|
|
|
|
List_iterator_fast<Item_sum> it(select_lex->min_max_opt_list);
|
|
|
|
Item_sum *item;
|
|
|
|
while ((item= it++))
|
|
|
|
{
|
|
|
|
item->clear();
|
|
|
|
item->reset_forced_const();
|
|
|
|
}
|
|
|
|
if (where_item)
|
|
|
|
where_item->update_used_tables();
|
|
|
|
if (having_item)
|
|
|
|
having_item->update_used_tables();
|
|
|
|
}
|
|
|
|
|
2010-09-30 17:32:44 +02:00
|
|
|
if (where_item)
|
|
|
|
{
|
Fix LP BUG#718593
Analysis:
Build_equal_items_for_cond() rewrites the WHERE clause in such a way,
that it may merge the list join->cond_equal->current_level with the
list of child Items in an AND condition of the WHERE clause.
The place where this is done is:
static COND *build_equal_items_for_cond(THD *thd, COND *cond,
COND_EQUAL *inherited)
{
...
if (and_level)
{
args->concat(&eq_list);
args->concat((List<Item> *)&cond_equal.current_level);
}
...
}
As a result, later transformations on the WHERE clause may change the
structure of the list join->cond_equal->current_level without knowing this.
Specifically in this bug, Item_in_subselect::inject_in_to_exists_cond
creates a new AND of the old WHERE clause and the IN->EXISTS conditions.
It then calls fix_fields() for the new AND. Among other things, fix_fields
flattens all nested ANDs into one by merging the AND argument lists.
When there is a cond_equal for the JOIN, its list of Item_equal objects
is attached to the end of the original AND. When a lower-level AND is
merged into the top-level one, the argument list of the lower-level AND
is concatenated to the list of multiple equalities in the upper-level AND.
As a result, when substitute_for_best_equal_field processes the
multiple equalities, it turns out that the multiple equality list contains
the Items from the lower-level AND which were concatenated to the end of
the join->cond_equal->current_level list. This results in a crash because
this list must not contain any other Items except for the previously found
Item_equal ones.
Solution:
When performing IN->EXIST predicate injection, and the where clause is an
AND, detach the list of Item_equal objects before calling fix_fields on
the injected where clause.
After fix_fields is done, reattach back the multiple equalities list to
the end of the argument list of the new AND.
2011-04-28 16:15:05 +02:00
|
|
|
List<Item> *and_args= NULL;
|
2005-02-08 23:50:45 +01:00
|
|
|
/*
|
Fix LP BUG#718593
Analysis:
Build_equal_items_for_cond() rewrites the WHERE clause in such a way,
that it may merge the list join->cond_equal->current_level with the
list of child Items in an AND condition of the WHERE clause.
The place where this is done is:
static COND *build_equal_items_for_cond(THD *thd, COND *cond,
COND_EQUAL *inherited)
{
...
if (and_level)
{
args->concat(&eq_list);
args->concat((List<Item> *)&cond_equal.current_level);
}
...
}
As a result, later transformations on the WHERE clause may change the
structure of the list join->cond_equal->current_level without knowing this.
Specifically in this bug, Item_in_subselect::inject_in_to_exists_cond
creates a new AND of the old WHERE clause and the IN->EXISTS conditions.
It then calls fix_fields() for the new AND. Among other things, fix_fields
flattens all nested ANDs into one by merging the AND argument lists.
When there is a cond_equal for the JOIN, its list of Item_equal objects
is attached to the end of the original AND. When a lower-level AND is
merged into the top-level one, the argument list of the lower-level AND
is concatenated to the list of multiple equalities in the upper-level AND.
As a result, when substitute_for_best_equal_field processes the
multiple equalities, it turns out that the multiple equality list contains
the Items from the lower-level AND which were concatenated to the end of
the join->cond_equal->current_level list. This results in a crash because
this list must not contain any other Items except for the previously found
Item_equal ones.
Solution:
When performing IN->EXIST predicate injection, and the where clause is an
AND, detach the list of Item_equal objects before calling fix_fields on
the injected where clause.
After fix_fields is done, reattach back the multiple equalities list to
the end of the argument list of the new AND.
2011-04-28 16:15:05 +02:00
|
|
|
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.
|
2005-02-08 23:50:45 +01:00
|
|
|
*/
|
Fix LP BUG#718593
Analysis:
Build_equal_items_for_cond() rewrites the WHERE clause in such a way,
that it may merge the list join->cond_equal->current_level with the
list of child Items in an AND condition of the WHERE clause.
The place where this is done is:
static COND *build_equal_items_for_cond(THD *thd, COND *cond,
COND_EQUAL *inherited)
{
...
if (and_level)
{
args->concat(&eq_list);
args->concat((List<Item> *)&cond_equal.current_level);
}
...
}
As a result, later transformations on the WHERE clause may change the
structure of the list join->cond_equal->current_level without knowing this.
Specifically in this bug, Item_in_subselect::inject_in_to_exists_cond
creates a new AND of the old WHERE clause and the IN->EXISTS conditions.
It then calls fix_fields() for the new AND. Among other things, fix_fields
flattens all nested ANDs into one by merging the AND argument lists.
When there is a cond_equal for the JOIN, its list of Item_equal objects
is attached to the end of the original AND. When a lower-level AND is
merged into the top-level one, the argument list of the lower-level AND
is concatenated to the list of multiple equalities in the upper-level AND.
As a result, when substitute_for_best_equal_field processes the
multiple equalities, it turns out that the multiple equality list contains
the Items from the lower-level AND which were concatenated to the end of
the join->cond_equal->current_level list. This results in a crash because
this list must not contain any other Items except for the previously found
Item_equal ones.
Solution:
When performing IN->EXIST predicate injection, and the where clause is an
AND, detach the list of Item_equal objects before calling fix_fields on
the injected where clause.
After fix_fields is done, reattach back the multiple equalities list to
the end of the argument list of the new AND.
2011-04-28 16:15:05 +02:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2015-08-11 09:18:38 +02:00
|
|
|
where_item= and_items(thd, join_arg->conds, where_item);
|
2018-06-05 08:25:39 +02:00
|
|
|
if (where_item->fix_fields_if_needed(thd, 0))
|
2010-09-30 17:32:44 +02:00
|
|
|
DBUG_RETURN(true);
|
2010-10-05 15:00:31 +02:00
|
|
|
// TIMOUR TODO: call optimize_cond() for the new where clause
|
2010-09-30 17:32:44 +02:00
|
|
|
thd->change_item_tree(&select_lex->where, where_item);
|
2005-08-13 06:45:14 +02:00
|
|
|
select_lex->where->top_level_item();
|
2010-09-30 17:32:44 +02:00
|
|
|
join_arg->conds= select_lex->where;
|
Fix LP BUG#718593
Analysis:
Build_equal_items_for_cond() rewrites the WHERE clause in such a way,
that it may merge the list join->cond_equal->current_level with the
list of child Items in an AND condition of the WHERE clause.
The place where this is done is:
static COND *build_equal_items_for_cond(THD *thd, COND *cond,
COND_EQUAL *inherited)
{
...
if (and_level)
{
args->concat(&eq_list);
args->concat((List<Item> *)&cond_equal.current_level);
}
...
}
As a result, later transformations on the WHERE clause may change the
structure of the list join->cond_equal->current_level without knowing this.
Specifically in this bug, Item_in_subselect::inject_in_to_exists_cond
creates a new AND of the old WHERE clause and the IN->EXISTS conditions.
It then calls fix_fields() for the new AND. Among other things, fix_fields
flattens all nested ANDs into one by merging the AND argument lists.
When there is a cond_equal for the JOIN, its list of Item_equal objects
is attached to the end of the original AND. When a lower-level AND is
merged into the top-level one, the argument list of the lower-level AND
is concatenated to the list of multiple equalities in the upper-level AND.
As a result, when substitute_for_best_equal_field processes the
multiple equalities, it turns out that the multiple equality list contains
the Items from the lower-level AND which were concatenated to the end of
the join->cond_equal->current_level list. This results in a crash because
this list must not contain any other Items except for the previously found
Item_equal ones.
Solution:
When performing IN->EXIST predicate injection, and the where clause is an
AND, detach the list of Item_equal objects before calling fix_fields on
the injected where clause.
After fix_fields is done, reattach back the multiple equalities list to
the end of the argument list of the new AND.
2011-04-28 16:15:05 +02:00
|
|
|
|
|
|
|
/* 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();
|
2011-09-04 14:35:37 +02:00
|
|
|
List_iterator<Item_equal> li(join_arg->cond_equal->current_level);
|
|
|
|
Item_equal *elem;
|
|
|
|
while ((elem= li++))
|
|
|
|
{
|
2015-08-24 13:42:07 +02:00
|
|
|
and_args->push_back(elem, thd->mem_root);
|
2011-09-04 14:35:37 +02:00
|
|
|
}
|
Fix LP BUG#718593
Analysis:
Build_equal_items_for_cond() rewrites the WHERE clause in such a way,
that it may merge the list join->cond_equal->current_level with the
list of child Items in an AND condition of the WHERE clause.
The place where this is done is:
static COND *build_equal_items_for_cond(THD *thd, COND *cond,
COND_EQUAL *inherited)
{
...
if (and_level)
{
args->concat(&eq_list);
args->concat((List<Item> *)&cond_equal.current_level);
}
...
}
As a result, later transformations on the WHERE clause may change the
structure of the list join->cond_equal->current_level without knowing this.
Specifically in this bug, Item_in_subselect::inject_in_to_exists_cond
creates a new AND of the old WHERE clause and the IN->EXISTS conditions.
It then calls fix_fields() for the new AND. Among other things, fix_fields
flattens all nested ANDs into one by merging the AND argument lists.
When there is a cond_equal for the JOIN, its list of Item_equal objects
is attached to the end of the original AND. When a lower-level AND is
merged into the top-level one, the argument list of the lower-level AND
is concatenated to the list of multiple equalities in the upper-level AND.
As a result, when substitute_for_best_equal_field processes the
multiple equalities, it turns out that the multiple equality list contains
the Items from the lower-level AND which were concatenated to the end of
the join->cond_equal->current_level list. This results in a crash because
this list must not contain any other Items except for the previously found
Item_equal ones.
Solution:
When performing IN->EXIST predicate injection, and the where clause is an
AND, detach the list of Item_equal objects before calling fix_fields on
the injected where clause.
After fix_fields is done, reattach back the multiple equalities list to
the end of the argument list of the new AND.
2011-04-28 16:15:05 +02:00
|
|
|
}
|
2002-12-19 20:15:09 +01:00
|
|
|
}
|
2005-03-10 13:01:22 +01:00
|
|
|
|
2005-08-13 06:45:14 +02:00
|
|
|
if (having_item)
|
2003-05-14 20:51:33 +02:00
|
|
|
{
|
2010-09-30 17:32:44 +02:00
|
|
|
Item* join_having= join_arg->having ? join_arg->having:join_arg->tmp_having;
|
2015-08-11 09:18:38 +02:00
|
|
|
having_item= and_items(thd, join_having, having_item);
|
2010-09-30 17:32:44 +02:00
|
|
|
if (fix_having(having_item, select_lex))
|
|
|
|
DBUG_RETURN(true);
|
2010-10-05 15:00:31 +02:00
|
|
|
// TIMOUR TODO: call optimize_cond() for the new having clause
|
2010-09-30 17:32:44 +02:00
|
|
|
thd->change_item_tree(&select_lex->having, having_item);
|
2005-08-13 06:45:14 +02:00
|
|
|
select_lex->having->top_level_item();
|
2010-09-30 17:32:44 +02:00
|
|
|
join_arg->having= select_lex->having;
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
2014-10-14 18:36:50 +02:00
|
|
|
join_arg->thd->change_item_tree(&unit->global_parameters()->select_limit,
|
2015-08-24 13:42:07 +02:00
|
|
|
new (thd->mem_root)
|
|
|
|
Item_int(thd, (int32) 1));
|
2011-05-04 17:08:58 +02:00
|
|
|
unit->select_limit_cnt= 1;
|
2004-02-08 19:14:13 +01:00
|
|
|
|
2010-09-30 17:32:44 +02:00
|
|
|
DBUG_RETURN(false);
|
2005-03-10 13:01:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-26 00:20:17 +01:00
|
|
|
/*
|
|
|
|
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);
|
2015-08-20 14:24:13 +02:00
|
|
|
result= (!(optimizer= new (thd->mem_root) Item_in_optimizer(thd, new (thd->mem_root) Item_int(thd, 1), this)));
|
2013-02-26 00:20:17 +01:00
|
|
|
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;
|
2015-09-21 10:40:07 +02:00
|
|
|
DBUG_ASSERT(func->argument_count() == 2);
|
2013-02-26 00:20:17 +01:00
|
|
|
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) */
|
2016-10-27 19:07:55 +02:00
|
|
|
DBUG_ASSERT(args[1]->type() == Item::FIELD_ITEM ||
|
|
|
|
args[1]->type() == Item::REF_ITEM);
|
2013-02-26 00:20:17 +01:00
|
|
|
*local_field= (Item_ident *)args[1];
|
|
|
|
*outer_exp= args[0];
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct st_eq_field_outer
|
|
|
|
{
|
2014-04-10 11:14:18 +02:00
|
|
|
Item **eq_ref;
|
2013-02-26 00:20:17 +01:00
|
|
|
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;
|
2014-04-10 11:14:18 +02:00
|
|
|
element.eq_ref= li.ref();
|
2013-02-26 00:20:17 +01:00
|
|
|
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;
|
2014-04-10 11:14:18 +02:00
|
|
|
element.eq_ref= conds;
|
2013-02-26 00:20:17 +01:00
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2016-06-26 22:42:48 +02:00
|
|
|
bool Item_exists_subselect::exists2in_processor(void *opt_arg)
|
2013-02-26 00:20:17 +01:00
|
|
|
{
|
|
|
|
THD *thd= (THD *)opt_arg;
|
|
|
|
SELECT_LEX *first_select=unit->first_select(), *save_select;
|
|
|
|
JOIN *join= first_select->join;
|
2014-04-10 11:14:18 +02:00
|
|
|
Item **eq_ref= NULL;
|
2013-02-26 00:20:17 +01:00
|
|
|
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||
|
2017-04-20 22:09:31 +02:00
|
|
|
!join->conds ||
|
|
|
|
with_recursive_reference)
|
2013-02-26 00:20:17 +01:00
|
|
|
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
|
|
|
|
};
|
2016-06-26 22:42:48 +02:00
|
|
|
walk(&Item::collect_outer_ref_processor, TRUE, &prm);
|
2013-02-26 00:20:17 +01:00
|
|
|
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
|
|
|
|
{
|
2015-08-24 13:42:07 +02:00
|
|
|
first_select->item_list.push_back(local_field, thd->mem_root);
|
2013-02-26 00:20:17 +01:00
|
|
|
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)
|
2015-08-20 14:24:13 +02:00
|
|
|
*eq_ref= new (thd->mem_root) Item_int(thd, 1);
|
2013-02-26 00:20:17 +01:00
|
|
|
else
|
|
|
|
{
|
2015-08-20 14:24:13 +02:00
|
|
|
*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));
|
2013-02-26 00:20:17 +01:00
|
|
|
if((*eq_ref)->fix_fields(thd, (Item **)eq_ref))
|
|
|
|
{
|
|
|
|
res= TRUE;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
2017-11-08 15:47:49 +01:00
|
|
|
outer_exp->fix_after_pullout(unit->outer_select(), &outer_exp, FALSE);
|
2013-02-26 00:20:17 +01:00
|
|
|
outer_exp->update_used_tables();
|
2015-08-24 13:42:07 +02:00
|
|
|
outer.push_back(outer_exp, thd->mem_root);
|
2013-02-26 00:20:17 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
join->conds->update_used_tables();
|
|
|
|
|
|
|
|
/* make IN SUBQUERY and put outer_exp as left part */
|
|
|
|
if (eqs.elements() == 1)
|
|
|
|
left_exp= outer_exp;
|
|
|
|
else
|
|
|
|
{
|
2015-08-20 14:24:13 +02:00
|
|
|
if (!(left_exp= new (thd->mem_root) Item_row(thd, outer)))
|
2013-02-26 00:20:17 +01:00
|
|
|
{
|
|
|
|
res= TRUE;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* make EXISTS->IN permanet (see Item_subselect::init()) */
|
|
|
|
set_exists_transformed();
|
|
|
|
|
|
|
|
first_select->select_limit= NULL;
|
2015-04-22 11:29:56 +02:00
|
|
|
if (!(in_subs= new (thd->mem_root) Item_in_subselect(thd, left_exp,
|
2015-08-11 09:18:38 +02:00
|
|
|
first_select)))
|
2013-02-26 00:20:17 +01:00
|
|
|
{
|
|
|
|
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.
|
|
|
|
*/
|
2015-08-20 14:24:13 +02:00
|
|
|
in_subs->expr= new (thd->mem_root)
|
|
|
|
Item_direct_ref(thd, &first_select->context,
|
|
|
|
(Item**)optimizer->get_cache(),
|
|
|
|
(char *)"<no matter>",
|
2017-04-23 18:39:57 +02:00
|
|
|
&in_left_expr_name);
|
2013-02-26 00:20:17 +01:00
|
|
|
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->
|
2016-06-26 22:42:48 +02:00
|
|
|
walk(&Item::find_item_processor, TRUE, upper->item))
|
2013-02-26 00:20:17 +01:00
|
|
|
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 ?
|
2015-08-20 14:24:13 +02:00
|
|
|
(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>",
|
2017-04-23 18:39:57 +02:00
|
|
|
&exists_outer_expr_name)),
|
2015-08-20 14:24:13 +02:00
|
|
|
optimizer) :
|
2013-02-26 00:20:17 +01:00
|
|
|
(Item *)optimizer);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
List<Item> *and_list= new List<Item>;
|
|
|
|
if (!and_list)
|
|
|
|
{
|
|
|
|
res= TRUE;
|
|
|
|
goto out;
|
|
|
|
}
|
2013-04-09 15:49:00 +02:00
|
|
|
for (size_t i= 0; i < eqs.elements(); i++)
|
2013-02-26 00:20:17 +01:00
|
|
|
{
|
|
|
|
if (optimizer->arguments()[0]->maybe_null)
|
|
|
|
{
|
|
|
|
and_list->
|
2015-08-20 14:24:13 +02:00
|
|
|
push_front(new (thd->mem_root)
|
|
|
|
Item_func_isnotnull(thd,
|
|
|
|
new (thd->mem_root)
|
|
|
|
Item_direct_ref(thd,
|
|
|
|
&unit->outer_select()->context,
|
2018-02-06 13:55:58 +01:00
|
|
|
optimizer->arguments()[0]->addr((int)i),
|
2015-08-20 14:24:13 +02:00
|
|
|
(char *)"<no matter>",
|
2017-04-23 18:39:57 +02:00
|
|
|
&exists_outer_expr_name)),
|
2015-08-24 13:42:07 +02:00
|
|
|
thd->mem_root);
|
2013-02-26 00:20:17 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (and_list->elements > 0)
|
|
|
|
{
|
2015-08-24 13:42:07 +02:00
|
|
|
and_list->push_front(optimizer, thd->mem_root);
|
2015-08-20 14:24:13 +02:00
|
|
|
exp= new (thd->mem_root) Item_cond_and(thd, *and_list);
|
2013-02-26 00:20:17 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
exp= optimizer;
|
|
|
|
}
|
|
|
|
upper_not->arguments()[0]= exp;
|
2018-06-05 08:25:39 +02:00
|
|
|
if (exp->fix_fields_if_needed(thd, upper_not->arguments()))
|
2013-02-26 00:20:17 +01:00
|
|
|
{
|
|
|
|
res= TRUE;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
thd->lex->current_select= save_select;
|
|
|
|
if (arena)
|
|
|
|
thd->restore_active_arena(arena, &backup);
|
|
|
|
DBUG_RETURN(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
/**
|
2010-12-15 11:54:25 +01:00
|
|
|
Prepare IN/ALL/ANY/SOME subquery transformation and call the appropriate
|
2007-10-11 19:29:09 +02:00
|
|
|
transformation function.
|
2005-03-10 13:01:22 +01:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
@param join JOIN object of transforming subquery
|
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
@notes
|
|
|
|
To decide which transformation procedure (scalar or row) applicable here
|
2010-12-15 11:54:25 +01:00
|
|
|
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
|
2010-10-05 15:00:31 +02:00
|
|
|
underlying transformation methods.
|
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
@retval false OK
|
|
|
|
@retval true Error
|
2005-03-10 13:01:22 +01:00
|
|
|
*/
|
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
bool
|
2010-09-05 17:43:47 +02:00
|
|
|
Item_in_subselect::select_in_like_transformer(JOIN *join)
|
2005-03-10 13:01:22 +01:00
|
|
|
{
|
2013-11-11 16:28:14 +01:00
|
|
|
Query_arena *arena= 0, backup;
|
2008-02-13 20:27:12 +01:00
|
|
|
SELECT_LEX *current= thd->lex->current_select;
|
2005-03-10 13:01:22 +01:00
|
|
|
const char *save_where= thd->where;
|
2010-12-15 11:54:25 +01:00
|
|
|
bool trans_res= true;
|
2005-03-10 13:01:22 +01:00
|
|
|
bool result;
|
|
|
|
|
|
|
|
DBUG_ENTER("Item_in_subselect::select_in_like_transformer");
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(thd == join->thd);
|
2005-03-30 09:07:08 +02:00
|
|
|
|
Fix LP BUG#715069
Analysis:
The wrong result is a consquence of sorting the subquery
result and then selecting only the first row due to the
artificial LIMIT 1 introduced by the fix_fields phase.
Normally, if there is an ORDER BY in a subquery, the ORDER
is removed (Item_in_subselect::select_in_like_transformer),
however if a GROUP BY is transformed into ORDER, this happens
later, after the removal of the ORDER clause of subqueries, so
we end up with a subquery with an ORDER clause, and an artificially
added LIMIT 1.
The reason why the same works in the main 5.3 without MWL#89, is
that the 5.3 performs all subquery transformations, including
IN->EXISTS before JOIN::optimize(). The beginning of JOIN::optimize
does:
if (having || (select_options & OPTION_FOUND_ROWS))
select_limit= HA_POS_ERROR;
which sets the limit back to infinity, thus 5.3 sorts the whole
subquery result, and IN performs the lookup into all subquery result
rows.
Solution:
Sorting of subqueries without LIMIT is meaningless. Since LIMIT in
subqueries is not supported, the patch removes sorting by setting
join->skip_sort_order= true
for each subquery JOIN object. This improves a number of execution
plans to not perform unnecessary sorting at all.
2011-04-20 17:36:55 +02:00
|
|
|
/*
|
|
|
|
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())
|
2008-03-27 17:49:32 +01:00
|
|
|
{
|
Fix LP BUG#715069
Analysis:
The wrong result is a consquence of sorting the subquery
result and then selecting only the first row due to the
artificial LIMIT 1 introduced by the fix_fields phase.
Normally, if there is an ORDER BY in a subquery, the ORDER
is removed (Item_in_subselect::select_in_like_transformer),
however if a GROUP BY is transformed into ORDER, this happens
later, after the removal of the ORDER clause of subqueries, so
we end up with a subquery with an ORDER clause, and an artificially
added LIMIT 1.
The reason why the same works in the main 5.3 without MWL#89, is
that the 5.3 performs all subquery transformations, including
IN->EXISTS before JOIN::optimize(). The beginning of JOIN::optimize
does:
if (having || (select_options & OPTION_FOUND_ROWS))
select_limit= HA_POS_ERROR;
which sets the limit back to infinity, thus 5.3 sorts the whole
subquery result, and IN performs the lookup into all subquery result
rows.
Solution:
Sorting of subqueries without LIMIT is meaningless. Since LIMIT in
subqueries is not supported, the patch removes sorting by setting
join->skip_sort_order= true
for each subquery JOIN object. This improves a number of execution
plans to not perform unnecessary sorting at all.
2011-04-20 17:36:55 +02:00
|
|
|
if (sl->join)
|
2008-03-27 17:49:32 +01:00
|
|
|
{
|
Fix LP BUG#715069
Analysis:
The wrong result is a consquence of sorting the subquery
result and then selecting only the first row due to the
artificial LIMIT 1 introduced by the fix_fields phase.
Normally, if there is an ORDER BY in a subquery, the ORDER
is removed (Item_in_subselect::select_in_like_transformer),
however if a GROUP BY is transformed into ORDER, this happens
later, after the removal of the ORDER clause of subqueries, so
we end up with a subquery with an ORDER clause, and an artificially
added LIMIT 1.
The reason why the same works in the main 5.3 without MWL#89, is
that the 5.3 performs all subquery transformations, including
IN->EXISTS before JOIN::optimize(). The beginning of JOIN::optimize
does:
if (having || (select_options & OPTION_FOUND_ROWS))
select_limit= HA_POS_ERROR;
which sets the limit back to infinity, thus 5.3 sorts the whole
subquery result, and IN performs the lookup into all subquery result
rows.
Solution:
Sorting of subqueries without LIMIT is meaningless. Since LIMIT in
subqueries is not supported, the patch removes sorting by setting
join->skip_sort_order= true
for each subquery JOIN object. This improves a number of execution
plans to not perform unnecessary sorting at all.
2011-04-20 17:36:55 +02:00
|
|
|
sl->join->order= 0;
|
|
|
|
sl->join->skip_sort_order= 1;
|
2008-03-27 17:49:32 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-03-10 13:01:22 +01:00
|
|
|
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.
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2012-03-12 15:20:30 +01:00
|
|
|
note: we won't need Item_in_optimizer when handling degenerate cases
|
|
|
|
like "... IN (SELECT 1)"
|
2005-03-10 13:01:22 +01:00
|
|
|
*/
|
2013-11-11 16:28:14 +01:00
|
|
|
arena= thd->activate_stmt_arena_if_needed(&backup);
|
2005-03-10 13:01:22 +01:00
|
|
|
if (!optimizer)
|
|
|
|
{
|
2015-10-12 00:37:58 +02:00
|
|
|
optimizer= new (thd->mem_root) Item_in_optimizer(thd, left_expr_orig, this);
|
|
|
|
if ((result= !optimizer))
|
2013-11-11 16:28:14 +01:00
|
|
|
goto out;
|
2005-03-10 13:01:22 +01:00
|
|
|
}
|
|
|
|
|
2008-02-13 20:27:12 +01:00
|
|
|
thd->lex->current_select= current->return_after_parsing();
|
2013-11-11 21:47:04 +01:00
|
|
|
result= optimizer->fix_left(thd);
|
2005-03-10 13:01:22 +01:00
|
|
|
thd->lex->current_select= current;
|
2013-11-11 16:28:14 +01:00
|
|
|
|
|
|
|
if (changed)
|
|
|
|
{
|
|
|
|
trans_res= false;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-03-10 13:01:22 +01:00
|
|
|
if (result)
|
2013-11-11 16:28:14 +01:00
|
|
|
goto out;
|
2005-03-10 13:01:22 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
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)
|
|
|
|
*/
|
2002-12-19 20:15:09 +01:00
|
|
|
if (left_expr->cols() == 1)
|
2010-12-15 11:54:25 +01:00
|
|
|
trans_res= single_value_transformer(join);
|
2005-03-10 13:01:22 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* we do not support row operation for ALL/ANY/SOME */
|
|
|
|
if (func != &eq_creator)
|
|
|
|
{
|
2005-03-30 09:07:08 +02:00
|
|
|
if (arena)
|
2005-09-02 15:21:19 +02:00
|
|
|
thd->restore_active_arena(arena, &backup);
|
2005-03-10 13:01:22 +01:00
|
|
|
my_error(ER_OPERAND_COLUMNS, MYF(0), 1);
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(true);
|
2005-03-10 13:01:22 +01:00
|
|
|
}
|
2010-12-15 11:54:25 +01:00
|
|
|
trans_res= row_value_transformer(join);
|
2005-03-10 13:01:22 +01:00
|
|
|
}
|
2013-11-11 16:28:14 +01:00
|
|
|
out:
|
2005-03-10 13:01:22 +01:00
|
|
|
if (arena)
|
2005-09-02 15:21:19 +02:00
|
|
|
thd->restore_active_arena(arena, &backup);
|
2005-03-10 13:01:22 +01:00
|
|
|
thd->where= save_where;
|
2010-12-15 11:54:25 +01:00
|
|
|
DBUG_RETURN(trans_res);
|
2002-11-07 22:45:19 +01:00
|
|
|
}
|
|
|
|
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2008-02-22 11:30:33 +01:00
|
|
|
void Item_in_subselect::print(String *str, enum_query_type query_type)
|
2003-10-16 14:54:47 +02:00
|
|
|
{
|
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
******
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
2011-11-12 10:29:12 +01:00
|
|
|
if (test_strategy(SUBS_IN_TO_EXISTS))
|
2005-11-20 19:47:07 +01:00
|
|
|
str->append(STRING_WITH_LEN("<exists>"));
|
2003-10-16 14:54:47 +02:00
|
|
|
else
|
|
|
|
{
|
2008-02-22 11:30:33 +01:00
|
|
|
left_expr->print(str, query_type);
|
2005-11-20 19:47:07 +01:00
|
|
|
str->append(STRING_WITH_LEN(" in "));
|
2003-10-16 14:54:47 +02:00
|
|
|
}
|
2008-02-22 11:30:33 +01:00
|
|
|
Item_subselect::print(str, query_type);
|
2003-10-16 14:54:47 +02:00
|
|
|
}
|
|
|
|
|
2015-08-11 09:18:38 +02:00
|
|
|
bool Item_exists_subselect::fix_fields(THD *thd, Item **ref)
|
2013-02-26 00:20:17 +01:00
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_exists_subselect::fix_fields");
|
|
|
|
if (exists_transformed)
|
2015-08-20 14:24:13 +02:00
|
|
|
DBUG_RETURN( !( (*ref)= new (thd->mem_root) Item_int(thd, 1)));
|
2015-08-11 09:18:38 +02:00
|
|
|
DBUG_RETURN(Item_subselect::fix_fields(thd, ref));
|
2013-02-26 00:20:17 +01:00
|
|
|
}
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2006-12-14 23:51:37 +01:00
|
|
|
bool Item_in_subselect::fix_fields(THD *thd_arg, Item **ref)
|
2006-05-11 14:30:54 +02:00
|
|
|
{
|
2010-02-19 22:55:57 +01:00
|
|
|
uint outer_cols_num;
|
|
|
|
List<Item> *inner_cols;
|
2016-06-22 14:17:06 +02:00
|
|
|
char const *save_where= thd_arg->where;
|
2013-02-26 00:20:17 +01:00
|
|
|
DBUG_ENTER("Item_in_subselect::fix_fields");
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2016-06-22 14:17:06 +02:00
|
|
|
thd= thd_arg;
|
|
|
|
DBUG_ASSERT(unit->thd == thd);
|
2010-01-28 14:48:33 +01:00
|
|
|
|
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
******
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
2011-11-12 10:29:12 +01:00
|
|
|
if (test_strategy(SUBS_SEMI_JOIN))
|
2015-08-20 14:24:13 +02:00
|
|
|
DBUG_RETURN( !( (*ref)= new (thd->mem_root) Item_int(thd, 1)) );
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2015-04-23 19:16:57 +02:00
|
|
|
thd->where= "IN/ALL/ANY subquery";
|
2010-02-19 22:55:57 +01:00
|
|
|
/*
|
|
|
|
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();
|
|
|
|
|
2017-03-14 11:52:00 +01:00
|
|
|
if (unit->is_unit_op())
|
2010-02-19 22:55:57 +01:00
|
|
|
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);
|
2015-04-23 19:16:57 +02:00
|
|
|
goto err;
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
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()))
|
2015-04-23 19:16:57 +02:00
|
|
|
goto err;
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-05 08:25:39 +02:00
|
|
|
if (left_expr && left_expr->fix_fields_if_needed(thd_arg, &left_expr))
|
2015-04-23 19:16:57 +02:00
|
|
|
goto err;
|
2012-02-02 00:48:02 +01:00
|
|
|
else
|
2010-02-19 22:55:57 +01:00
|
|
|
if (Item_subselect::fix_fields(thd_arg, ref))
|
2015-10-09 17:12:26 +02:00
|
|
|
goto err;
|
2010-02-19 22:55:57 +01:00
|
|
|
fixed= TRUE;
|
2015-04-23 19:16:57 +02:00
|
|
|
thd->where= save_where;
|
2013-02-26 00:20:17 +01:00
|
|
|
DBUG_RETURN(FALSE);
|
2015-04-23 19:16:57 +02:00
|
|
|
|
|
|
|
err:
|
|
|
|
thd->where= save_where;
|
2015-10-09 17:12:26 +02:00
|
|
|
DBUG_RETURN(TRUE);
|
2006-05-11 14:30:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-11-08 15:47:49 +01:00
|
|
|
void Item_in_subselect::fix_after_pullout(st_select_lex *new_parent,
|
|
|
|
Item **ref, bool merge)
|
2002-11-07 22:45:19 +01:00
|
|
|
{
|
2017-11-08 15:47:49 +01:00
|
|
|
left_expr->fix_after_pullout(new_parent, &left_expr, merge);
|
|
|
|
Item_subselect::fix_after_pullout(new_parent, ref, merge);
|
2011-09-22 23:25:08 +02:00
|
|
|
used_tables_cache |= left_expr->used_tables();
|
2002-11-07 22:45:19 +01:00
|
|
|
}
|
|
|
|
|
2010-02-12 00:59:58 +01:00
|
|
|
void Item_in_subselect::update_used_tables()
|
2003-10-16 14:54:47 +02:00
|
|
|
{
|
2010-02-12 00:59:58 +01:00
|
|
|
Item_subselect::update_used_tables();
|
|
|
|
left_expr->update_used_tables();
|
2012-04-02 19:41:54 +02:00
|
|
|
//used_tables_cache |= left_expr->used_tables();
|
|
|
|
used_tables_cache= Item_subselect::used_tables() | left_expr->used_tables();
|
2003-10-16 14:54:47 +02:00
|
|
|
}
|
|
|
|
|
2010-10-05 15:00:31 +02:00
|
|
|
|
2010-01-28 14:48:33 +01:00
|
|
|
/**
|
2010-10-05 15:00:31 +02:00
|
|
|
Try to create and initialize an engine to compute a subselect via
|
|
|
|
materialization.
|
2010-01-28 14:48:33 +01:00
|
|
|
|
|
|
|
@details
|
2010-10-05 15:00:31 +02:00
|
|
|
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,
|
2010-01-28 14:48:33 +01:00
|
|
|
|
|
|
|
@returns
|
|
|
|
@retval TRUE memory allocation error occurred
|
|
|
|
@retval FALSE an execution method was chosen successfully
|
|
|
|
*/
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2010-09-05 17:43:47 +02:00
|
|
|
bool Item_in_subselect::setup_mat_engine()
|
2010-01-28 14:48:33 +01:00
|
|
|
{
|
2010-07-16 12:52:02 +02:00
|
|
|
subselect_hash_sj_engine *mat_engine= NULL;
|
|
|
|
subselect_single_select_engine *select_engine;
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-09-05 17:43:47 +02:00
|
|
|
DBUG_ENTER("Item_in_subselect::setup_mat_engine");
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(thd);
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-07-16 12:52:02 +02:00
|
|
|
/*
|
|
|
|
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;
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-07-16 12:52:02 +02:00
|
|
|
/* Create/initialize execution objects. */
|
|
|
|
if (!(mat_engine= new subselect_hash_sj_engine(thd, this, select_engine)))
|
|
|
|
DBUG_RETURN(TRUE);
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2016-06-22 14:17:06 +02:00
|
|
|
if (mat_engine->prepare(thd) ||
|
|
|
|
mat_engine->init(&select_engine->join->fields_list,
|
2011-05-25 17:31:13 +02:00
|
|
|
engine->get_identifier()))
|
2010-09-05 17:43:47 +02:00
|
|
|
DBUG_RETURN(TRUE);
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-07-16 12:52:02 +02:00
|
|
|
engine= mat_engine;
|
|
|
|
DBUG_RETURN(FALSE);
|
2010-01-28 14:48:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
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;
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(thd);
|
2010-01-28 14:48:33 +01:00
|
|
|
|
|
|
|
outer_join= unit->outer_select()->join;
|
|
|
|
/*
|
|
|
|
An IN predicate might be evaluated in a query for which all tables have
|
|
|
|
been optimzied away.
|
|
|
|
*/
|
2011-03-27 00:45:16 +01:00
|
|
|
if (!outer_join || !outer_join->table_count || !outer_join->tables_list)
|
2010-01-28 14:48:33 +01:00
|
|
|
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),
|
2010-07-06 17:16:24 +02:00
|
|
|
FALSE);
|
2015-08-24 13:42:07 +02:00
|
|
|
if (!cur_item_cache || left_expr_cache->push_front(cur_item_cache,
|
|
|
|
thd->mem_root))
|
2010-01-28 14:48:33 +01:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-05 17:43:47 +02:00
|
|
|
bool Item_in_subselect::init_cond_guards()
|
2010-01-28 14:48:33 +01:00
|
|
|
{
|
2016-09-21 18:36:34 +02:00
|
|
|
DBUG_ASSERT(thd);
|
2010-09-05 17:43:47 +02:00
|
|
|
uint cols_num= left_expr->cols();
|
2017-07-12 08:05:42 +02:00
|
|
|
if (!abort_on_null && !pushed_cond_guards &&
|
|
|
|
(left_expr->maybe_null || cols_num > 1))
|
2010-09-05 17:43:47 +02:00
|
|
|
{
|
|
|
|
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;
|
2010-01-28 14:48:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-15 11:54:25 +01:00
|
|
|
bool
|
2003-07-02 00:45:22 +02:00
|
|
|
Item_allany_subselect::select_transformer(JOIN *join)
|
2002-11-07 22:45:19 +01:00
|
|
|
{
|
2010-01-28 14:48:33 +01:00
|
|
|
DBUG_ENTER("Item_allany_subselect::select_transformer");
|
2011-07-18 22:45:38 +02:00
|
|
|
DBUG_ASSERT((in_strategy & ~(SUBS_MAXMIN_INJECTED | SUBS_MAXMIN_ENGINE |
|
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
******
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
2011-11-12 10:29:12 +01:00
|
|
|
SUBS_IN_TO_EXISTS | SUBS_STRATEGY_CHOSEN)) == 0);
|
2004-11-18 17:10:07 +01:00
|
|
|
if (upper_item)
|
|
|
|
upper_item->show= 1;
|
2010-09-05 17:43:47 +02:00
|
|
|
DBUG_RETURN(select_in_like_transformer(join));
|
2002-11-07 22:45:19 +01:00
|
|
|
}
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2008-02-22 11:30:33 +01:00
|
|
|
void Item_allany_subselect::print(String *str, enum_query_type query_type)
|
2003-10-16 14:54:47 +02:00
|
|
|
{
|
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
******
Fix MySQL BUG#12329653
In MariaDB, when running in ONLY_FULL_GROUP_BY mode,
the server produced in incorrect error message that there
is an aggregate function without GROUP BY, for artificially
created MIN/MAX functions during subquery MIN/MAX optimization.
The fix introduces a way to distinguish between artifially
created MIN/MAX functions as a result of a rewrite, and normal
ones present in the query. The test for ONLY_FULL_GROUP_BY violation
now tests in addition if a MIN/MAX function was part of a MIN/MAX
subquery rewrite.
In order to be able to distinguish these MIN/MAX functions, the
patch introduces an additional flag in Item_in_subselect::in_strategy -
SUBS_STRATEGY_CHOSEN. This flag is set when the optimizer makes its
final choice of a subuqery strategy. In order to make the choice
consistent, access to Item_in_subselect::in_strategy is provided
via new class methods.
2011-11-12 10:29:12 +01:00
|
|
|
if (test_strategy(SUBS_IN_TO_EXISTS))
|
2005-11-20 19:47:07 +01:00
|
|
|
str->append(STRING_WITH_LEN("<exists>"));
|
2003-10-16 14:54:47 +02:00
|
|
|
else
|
|
|
|
{
|
2008-02-22 11:30:33 +01:00
|
|
|
left_expr->print(str, query_type);
|
2003-10-16 14:54:47 +02:00
|
|
|
str->append(' ');
|
2003-11-03 11:28:36 +01:00
|
|
|
str->append(func->symbol(all));
|
|
|
|
str->append(all ? " all " : " any ", 5);
|
2003-10-16 14:54:47 +02:00
|
|
|
}
|
2008-02-22 11:30:33 +01:00
|
|
|
Item_subselect::print(str, query_type);
|
2003-10-16 14:54:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
void Item_allany_subselect::no_rows_in_result()
|
|
|
|
{
|
Fix bug lp:1008686
Analysis:
The fix for bug lp:985667 implements the method Item_subselect::no_rows_in_result()
for all main kinds of subqueries. The purpose of this method is to be called from
return_zero_rows() and set Items to some default value in the case when a query
returns no rows. Aggregates and subqueries require special treatment in this case.
Every implementation of Item_subselect::no_rows_in_result() called
Item_subselect::make_const() to set the subquery predicate to its default value
irrespective of where the predicate was located in the query. Once the predicate
was set to a constant it was never executed.
At the same time, the JOIN object of the fake select for UNIONs (the one used for
the final result of the UNION), was set after all subqueries in the union were
executed. Since we set the subquery as constant, it was never executed, and the
corresponding JOIN was never created.
In order to decide whether the result of NOT IN is NULL or FALSE, Item_in_optimizer
needs to check if the subquery result was empty or not. This is where we got the
crash, because subselect_union_engine::no_rows() checks for
unit->fake_select_lex->join->send_records, and the join object was NULL.
Solution:
If a subquery is in the HAVING clause it must be evaluated in order to know its
result, so that we can properly filter the result records. Once subqueries in the
HAVING clause are executed even in the case of no result rows, this specific
crash will be solved, because the UNION will be executed, and its JOIN will be
constructed. Therefore the fix for this crash is to narrow the fix for lp:985667,
and to apply Item_subselect::no_rows_in_result() only when the subquery predicate
is in the SELECT clause.
2012-06-15 10:33:24 +02:00
|
|
|
/*
|
|
|
|
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())
|
2012-06-14 16:03:09 +02:00
|
|
|
return;
|
Fix bug lp:985667, MDEV-229
Analysis:
The reason for the wrong result is the interaction between constant
optimization (in this case 1-row table) and subquery optimization.
- First the outer query is optimized, and 'make_join_statistics' finds that
table t2 has one row, reads that row, and marks the whole table as constant.
This also means that all fields of t2 are constant.
- Next, we optimize the subquery in the end of the outer 'make_join_statistics'.
The field 'f2' is considered constant, with value '3'. The subquery predicate
is rewritten as the constant TRUE.
- The outer query execution detects early that the whole query result is empty
and calls 'return_zero_rows'. Since the query is with implicit grouping, we
have to produce one row with special values for the aggregates (depending on
each aggregate function), and NULL values for all non-aggregate fields. This
function calls 'no_rows_in_result' to set each aggregate function to the
default value when it aggregates over an empty result, and then calls
'send_data', which in turn evaluates each Item in the SELECT list.
- When evaluation reaches the subquery predicate, it executes the subquery
with field 'f2' having a constant value '3', and the subquery produces the
incorrect result '7'.
Solution:
Implement Item::no_rows_in_result for all subquery predicates. In order to
make this work, it is also needed to make all val_* methods of all subquery
predicates respect the Item_subselect::forced_const flag. Otherwise subqueries
are executed anyways, and override the default value set by no_rows_in_result
with whatever result is produced from the subquery evaluation.
2012-04-27 11:59:17 +02:00
|
|
|
value= 0;
|
|
|
|
null_value= 0;
|
|
|
|
was_null= 0;
|
|
|
|
make_const();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-08-09 22:23:56 +02:00
|
|
|
void subselect_engine::set_thd(THD *thd_arg)
|
|
|
|
{
|
|
|
|
thd= thd_arg;
|
|
|
|
if (result)
|
|
|
|
result->set_thd(thd_arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-24 14:26:21 +02:00
|
|
|
subselect_single_select_engine::
|
2016-06-22 14:17:06 +02:00
|
|
|
subselect_single_select_engine(st_select_lex *select,
|
2010-01-28 14:48:33 +01:00
|
|
|
select_result_interceptor *result_arg,
|
2006-12-14 23:51:37 +01:00
|
|
|
Item_subselect *item_arg)
|
2016-06-22 14:17:06 +02:00
|
|
|
:subselect_engine(item_arg, result_arg),
|
2014-02-13 20:18:07 +01:00
|
|
|
prepared(0), executed(0),
|
2005-05-30 18:54:37 +02:00
|
|
|
select_lex(select), join(0)
|
2002-09-03 08:50:36 +02:00
|
|
|
{
|
2006-12-14 23:51:37 +01:00
|
|
|
select_lex->master_unit()->item= item_arg;
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
|
|
|
|
2010-03-29 16:04:35 +02:00
|
|
|
int subselect_single_select_engine::get_identifier()
|
|
|
|
{
|
|
|
|
return select_lex->select_number;
|
|
|
|
}
|
2004-02-08 19:14:13 +01:00
|
|
|
|
2017-04-20 22:09:31 +02:00
|
|
|
void subselect_single_select_engine::force_reexecution()
|
|
|
|
{
|
|
|
|
executed= false;
|
|
|
|
}
|
|
|
|
|
2003-12-30 11:08:19 +01:00
|
|
|
void subselect_single_select_engine::cleanup()
|
|
|
|
{
|
2004-02-08 19:14:13 +01:00
|
|
|
DBUG_ENTER("subselect_single_select_engine::cleanup");
|
2014-02-13 20:18:07 +01:00
|
|
|
prepared= executed= 0;
|
2004-02-12 02:10:26 +01:00
|
|
|
join= 0;
|
2005-01-26 14:27:45 +01:00
|
|
|
result->cleanup();
|
2011-05-04 17:08:58 +02:00
|
|
|
select_lex->uncacheable&= ~UNCACHEABLE_DEPENDENT_INJECTED;
|
2004-02-08 19:14:13 +01:00
|
|
|
DBUG_VOID_RETURN;
|
2003-12-30 11:08:19 +01:00
|
|
|
}
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2004-02-08 19:14:13 +01:00
|
|
|
|
|
|
|
void subselect_union_engine::cleanup()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("subselect_union_engine::cleanup");
|
|
|
|
unit->reinit_exec_mechanism();
|
2005-01-26 14:27:45 +01:00
|
|
|
result->cleanup();
|
2011-05-04 17:08:58 +02:00
|
|
|
unit->uncacheable&= ~UNCACHEABLE_DEPENDENT_INJECTED;
|
|
|
|
for (SELECT_LEX *sl= unit->first_select(); sl; sl= sl->next_select())
|
|
|
|
sl->uncacheable&= ~UNCACHEABLE_DEPENDENT_INJECTED;
|
2004-02-08 19:14:13 +01:00
|
|
|
DBUG_VOID_RETURN;
|
2003-12-30 11:08:19 +01:00
|
|
|
}
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2004-02-08 19:14:13 +01:00
|
|
|
|
2005-10-13 09:53:00 +02:00
|
|
|
bool subselect_union_engine::is_executed() const
|
|
|
|
{
|
|
|
|
return unit->executed;
|
|
|
|
}
|
|
|
|
|
2017-04-20 22:09:31 +02:00
|
|
|
void subselect_union_engine::force_reexecution()
|
|
|
|
{
|
|
|
|
unit->executed= false;
|
|
|
|
}
|
|
|
|
|
2005-10-13 09:53:00 +02:00
|
|
|
|
2006-10-31 18:51:09 +01:00
|
|
|
/*
|
|
|
|
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: */
|
2017-01-16 02:18:14 +01:00
|
|
|
if (unit->fake_select_lex)
|
|
|
|
{
|
|
|
|
JOIN *join= unit->fake_select_lex->join;
|
|
|
|
if (join)
|
|
|
|
return MY_TEST(!join->send_records);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return MY_TEST(!(((select_union_direct *)(unit->get_union_result()))
|
2014-10-19 18:50:50 +02:00
|
|
|
->send_records));
|
2006-10-31 18:51:09 +01:00
|
|
|
}
|
|
|
|
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2004-02-08 19:14:13 +01:00
|
|
|
void subselect_uniquesubquery_engine::cleanup()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("subselect_uniquesubquery_engine::cleanup");
|
2011-10-24 13:22:17 +02:00
|
|
|
/*
|
|
|
|
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()
|
|
|
|
*/
|
2004-02-08 19:14:13 +01:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-06-22 14:17:06 +02:00
|
|
|
subselect_union_engine::subselect_union_engine(st_select_lex_unit *u,
|
2010-01-28 14:48:33 +01:00
|
|
|
select_result_interceptor *result_arg,
|
2003-11-28 11:18:13 +01:00
|
|
|
Item_subselect *item_arg)
|
2016-06-22 14:17:06 +02:00
|
|
|
:subselect_engine(item_arg, result_arg)
|
2002-09-03 08:50:36 +02:00
|
|
|
{
|
|
|
|
unit= u;
|
2003-11-28 11:18:13 +01:00
|
|
|
unit->item= item_arg;
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
|
|
|
|
2003-11-28 11:18:13 +01:00
|
|
|
|
2010-01-28 14:48:33 +01:00
|
|
|
/**
|
|
|
|
Create and prepare the JOIN object that represents the query execution
|
|
|
|
plan for the subquery.
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
@details
|
2010-01-28 14:48:33 +01:00
|
|
|
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
|
|
|
|
*/
|
|
|
|
|
2016-06-22 14:17:06 +02:00
|
|
|
int subselect_single_select_engine::prepare(THD *thd)
|
2002-09-03 08:50:36 +02:00
|
|
|
{
|
2002-10-13 13:25:16 +02:00
|
|
|
if (prepared)
|
|
|
|
return 0;
|
2016-06-22 14:17:06 +02:00
|
|
|
set_thd(thd);
|
2010-03-09 20:29:05 +01:00
|
|
|
if (select_lex->join)
|
|
|
|
{
|
|
|
|
select_lex->cleanup();
|
|
|
|
}
|
2003-12-10 21:46:14 +01:00
|
|
|
join= new JOIN(thd, select_lex->item_list,
|
|
|
|
select_lex->options | SELECT_NO_UNLOCK, result);
|
2003-10-02 21:19:41 +02:00
|
|
|
if (!join || !result)
|
2009-11-10 21:31:28 +01:00
|
|
|
return 1; /* Fatal error is set already. */
|
2002-10-13 13:25:16 +02:00
|
|
|
prepared= 1;
|
2003-12-19 18:52:13 +01:00
|
|
|
SELECT_LEX *save_select= thd->lex->current_select;
|
|
|
|
thd->lex->current_select= select_lex;
|
2016-05-08 22:04:41 +02:00
|
|
|
if (join->prepare(select_lex->table_list.first,
|
2003-01-25 12:19:46 +01:00
|
|
|
select_lex->with_wild,
|
2002-12-28 00:01:05 +01:00
|
|
|
select_lex->where,
|
2003-01-25 12:19:46 +01:00
|
|
|
select_lex->order_list.elements +
|
|
|
|
select_lex->group_list.elements,
|
2010-06-10 22:45:22 +02:00
|
|
|
select_lex->order_list.first,
|
2013-10-29 11:39:03 +01:00
|
|
|
false,
|
2010-06-10 22:45:22 +02:00
|
|
|
select_lex->group_list.first,
|
2002-12-28 00:01:05 +01:00
|
|
|
select_lex->having,
|
2010-06-10 22:45:22 +02:00
|
|
|
NULL, select_lex,
|
2003-11-23 01:01:15 +01:00
|
|
|
select_lex->master_unit()))
|
2002-09-03 08:50:36 +02:00
|
|
|
return 1;
|
2003-12-19 18:52:13 +01:00
|
|
|
thd->lex->current_select= save_select;
|
2002-09-03 08:50:36 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-06-22 14:17:06 +02:00
|
|
|
int subselect_union_engine::prepare(THD *thd_arg)
|
2002-09-03 08:50:36 +02:00
|
|
|
{
|
2016-06-22 14:17:06 +02:00
|
|
|
set_thd(thd_arg);
|
2018-04-18 14:29:48 +02:00
|
|
|
return unit->prepare(unit->derived, result, SELECT_NO_UNLOCK);
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
|
|
|
|
2016-06-22 14:17:06 +02:00
|
|
|
int subselect_uniquesubquery_engine::prepare(THD *)
|
2003-07-07 17:40:19 +02:00
|
|
|
{
|
2010-01-28 14:48:33 +01:00
|
|
|
/* Should never be called. */
|
|
|
|
DBUG_ASSERT(FALSE);
|
2003-07-07 17:40:19 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2006-10-31 18:51:09 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
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();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-05-08 15:26:26 +02:00
|
|
|
/**
|
|
|
|
Makes storage for the output values for the subquery and calcuates
|
2006-11-07 17:16:17 +01:00
|
|
|
their data and column types and their nullability.
|
2018-05-08 15:26:26 +02:00
|
|
|
*/
|
|
|
|
bool subselect_engine::set_row(List<Item> &item_list, Item_cache **row)
|
2002-09-28 17:34:56 +02:00
|
|
|
{
|
2002-12-19 06:38:33 +01:00
|
|
|
Item *sel_item;
|
2003-11-23 01:01:15 +01:00
|
|
|
List_iterator_fast<Item> li(item_list);
|
2016-11-26 18:19:48 +01:00
|
|
|
set_handler(&type_handler_varchar);
|
2002-12-19 06:38:33 +01:00
|
|
|
for (uint i= 0; (sel_item= li++); i++)
|
|
|
|
{
|
|
|
|
item->max_length= sel_item->max_length;
|
2016-11-26 18:19:48 +01:00
|
|
|
set_handler(sel_item->type_handler());
|
2002-12-19 06:38:33 +01:00
|
|
|
item->decimals= sel_item->decimals;
|
2006-06-09 19:35:54 +02:00
|
|
|
item->unsigned_flag= sel_item->unsigned_flag;
|
2006-11-18 18:49:59 +01:00
|
|
|
maybe_null= sel_item->maybe_null;
|
2016-11-26 18:19:48 +01:00
|
|
|
if (!(row[i]= sel_item->get_cache(thd)))
|
2018-05-08 15:26:26 +02:00
|
|
|
return TRUE;
|
2015-08-11 09:18:38 +02:00
|
|
|
row[i]->setup(thd, sel_item);
|
2010-01-28 14:48:33 +01:00
|
|
|
//psergey-backport-timours: row[i]->store(sel_item);
|
2002-12-19 06:38:33 +01:00
|
|
|
}
|
2003-11-23 01:01:15 +01:00
|
|
|
if (item_list.elements > 1)
|
2016-11-26 18:19:48 +01:00
|
|
|
set_handler(&type_handler_row);
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2002-09-28 17:34:56 +02:00
|
|
|
}
|
|
|
|
|
2018-05-08 15:26:26 +02:00
|
|
|
bool subselect_single_select_engine::fix_length_and_dec(Item_cache **row)
|
2002-09-28 17:34:56 +02:00
|
|
|
{
|
2002-12-19 06:38:33 +01:00
|
|
|
DBUG_ASSERT(row || select_lex->item_list.elements==1);
|
2018-05-08 15:26:26 +02:00
|
|
|
if (set_row(select_lex->item_list, row))
|
|
|
|
return TRUE;
|
2003-07-30 11:15:25 +02:00
|
|
|
item->collation.set(row[0]->collation);
|
2002-12-27 20:19:25 +01:00
|
|
|
if (cols() != 1)
|
|
|
|
maybe_null= 0;
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2002-12-19 06:38:33 +01:00
|
|
|
}
|
|
|
|
|
2018-05-08 15:26:26 +02:00
|
|
|
bool subselect_union_engine::fix_length_and_dec(Item_cache **row)
|
2002-12-19 06:38:33 +01:00
|
|
|
{
|
|
|
|
DBUG_ASSERT(row || unit->first_select()->item_list.elements==1);
|
|
|
|
|
|
|
|
if (unit->first_select()->item_list.elements == 1)
|
2003-12-25 15:50:22 +01:00
|
|
|
{
|
2018-05-08 15:26:26 +02:00
|
|
|
if (set_row(unit->types, row))
|
|
|
|
return TRUE;
|
2003-12-25 15:50:22 +01:00
|
|
|
item->collation.set(row[0]->collation);
|
|
|
|
}
|
2002-12-19 06:38:33 +01:00
|
|
|
else
|
|
|
|
{
|
2006-11-07 17:16:17 +01:00
|
|
|
bool maybe_null_saved= maybe_null;
|
2018-05-08 15:26:26 +02:00
|
|
|
if (set_row(unit->types, row))
|
|
|
|
return TRUE;
|
2006-11-07 17:16:17 +01:00
|
|
|
maybe_null= maybe_null_saved;
|
2002-09-28 17:34:56 +02:00
|
|
|
}
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2002-09-28 17:34:56 +02:00
|
|
|
}
|
2002-09-03 08:50:36 +02:00
|
|
|
|
2018-05-08 15:26:26 +02:00
|
|
|
bool subselect_uniquesubquery_engine::fix_length_and_dec(Item_cache **row)
|
2003-07-07 17:40:19 +02:00
|
|
|
{
|
|
|
|
//this never should be called
|
|
|
|
DBUG_ASSERT(0);
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2003-07-07 17:40:19 +02:00
|
|
|
}
|
|
|
|
|
2009-10-16 17:58:26 +02:00
|
|
|
int read_first_record_seq(JOIN_TAB *tab);
|
|
|
|
int rr_sequential(READ_RECORD *info);
|
2006-10-31 18:51:09 +01:00
|
|
|
int join_read_always_key_or_null(JOIN_TAB *tab);
|
|
|
|
int join_read_next_same_or_null(READ_RECORD *info);
|
|
|
|
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
int subselect_single_select_engine::exec()
|
2002-09-03 08:50:36 +02:00
|
|
|
{
|
|
|
|
DBUG_ENTER("subselect_single_select_engine::exec");
|
2011-10-05 11:28:20 +02:00
|
|
|
|
2004-09-10 01:22:44 +02:00
|
|
|
char const *save_where= thd->where;
|
|
|
|
SELECT_LEX *save_select= thd->lex->current_select;
|
|
|
|
thd->lex->current_select= select_lex;
|
2011-12-11 10:34:44 +01:00
|
|
|
|
2016-06-22 11:17:44 +02:00
|
|
|
if (join->optimization_state == JOIN::NOT_OPTIMIZED)
|
2002-09-03 08:50:36 +02:00
|
|
|
{
|
2005-05-30 18:54:37 +02:00
|
|
|
SELECT_LEX_UNIT *unit= select_lex->master_unit();
|
|
|
|
|
2014-10-14 18:36:50 +02:00
|
|
|
unit->set_limit(unit->global_parameters());
|
2002-09-03 08:50:36 +02:00
|
|
|
if (join->optimize())
|
|
|
|
{
|
2004-09-10 01:22:44 +02:00
|
|
|
thd->where= save_where;
|
2014-02-13 20:18:07 +01:00
|
|
|
executed= 1;
|
2004-09-10 01:22:44 +02:00
|
|
|
thd->lex->current_select= save_select;
|
2004-06-23 12:29:05 +02:00
|
|
|
DBUG_RETURN(join->error ? join->error : 1);
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
2007-05-04 09:48:51 +02:00
|
|
|
if (!select_lex->uncacheable && thd->lex->describe &&
|
2010-10-18 14:12:27 +02:00
|
|
|
!(join->select_options & SELECT_DESCRIBE))
|
2007-05-04 09:48:51 +02:00
|
|
|
{
|
2010-10-01 12:08:38 +02:00
|
|
|
item->update_used_tables();
|
|
|
|
if (item->const_item())
|
|
|
|
{
|
2010-10-18 14:12:27 +02:00
|
|
|
/*
|
|
|
|
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;
|
2010-10-01 12:08:38 +02:00
|
|
|
}
|
2007-05-04 09:48:51 +02:00
|
|
|
}
|
2012-10-25 14:50:10 +02:00
|
|
|
if (item->engine_changed(this))
|
2017-05-19 08:54:43 +02:00
|
|
|
{
|
|
|
|
thd->lex->current_select= save_select;
|
2003-07-07 17:40:19 +02:00
|
|
|
DBUG_RETURN(1);
|
2017-05-19 08:54:43 +02:00
|
|
|
}
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
2007-11-19 21:00:25 +01:00
|
|
|
if (select_lex->uncacheable &&
|
|
|
|
select_lex->uncacheable != UNCACHEABLE_EXPLAIN
|
|
|
|
&& executed)
|
2002-09-03 08:50:36 +02:00
|
|
|
{
|
|
|
|
if (join->reinit())
|
2002-11-24 20:10:52 +01:00
|
|
|
{
|
2004-09-10 01:22:44 +02:00
|
|
|
thd->where= save_where;
|
|
|
|
thd->lex->current_select= save_select;
|
2002-09-03 08:50:36 +02:00
|
|
|
DBUG_RETURN(1);
|
2002-11-24 20:10:52 +01:00
|
|
|
}
|
2002-12-06 20:55:53 +01:00
|
|
|
item->reset();
|
2002-09-03 08:50:36 +02:00
|
|
|
item->assigned((executed= 0));
|
|
|
|
}
|
|
|
|
if (!executed)
|
|
|
|
{
|
2004-11-18 17:10:07 +01:00
|
|
|
item->reset_value_registration();
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
JOIN_TAB *changed_tabs[MAX_TABLES];
|
|
|
|
JOIN_TAB **last_changed_tab= changed_tabs;
|
|
|
|
if (item->have_guarded_conds())
|
2006-10-31 18:51:09 +01:00
|
|
|
{
|
|
|
|
/*
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
For at least one of the pushed predicates the following is true:
|
2006-10-31 18:51:09 +01:00
|
|
|
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.
|
|
|
|
*/
|
2014-07-23 17:53:29 +02:00
|
|
|
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))
|
2006-10-31 18:51:09 +01:00
|
|
|
{
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (tab && tab->keyuse)
|
2006-10-31 18:51:09 +01:00
|
|
|
{
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
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 */
|
2007-05-12 04:37:32 +02:00
|
|
|
tab->save_read_first_record= tab->read_first_record;
|
2017-08-25 13:36:13 +02:00
|
|
|
tab->save_read_record= tab->read_record.read_record_func;
|
|
|
|
tab->read_record.read_record_func= rr_sequential;
|
2009-10-16 17:58:26 +02:00
|
|
|
tab->read_first_record= read_first_record_seq;
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
tab->read_record.thd= join->thd;
|
|
|
|
tab->read_record.ref_length= tab->table->file->ref_length;
|
A fix and a test case for
Bug#41756 "Strange error messages about locks from InnoDB".
In JT_EQ_REF (join_read_key()) access method,
don't try to unlock rows in the handler, unless certain that
a) they were locked
b) they are not used.
Unlocking of rows is done by the logic of the nested join loop,
and is unaware of the possible caching that the access method may
have. This could lead to double unlocking, when a row
was unlocked first after reading into the cache, and then
when taken from cache, as well as to unlocking of rows which
were actually used (but taken from cache).
Delegate part of the unlocking logic to the access method,
and in JT_EQ_REF count how many times a record was actually
used in the join. Unlock it only if it's usage count is 0.
Implemented review comments.
mysql-test/r/bug41756.result:
Add result file (Bug#41756)
mysql-test/t/bug41756-master.opt:
Use --innodb-locks-unsafe-for-binlog, as in 5.0 just
using read_committed isolation is not sufficient to
reproduce the bug.
mysql-test/t/bug41756.test:
Add a test file (Bug#41756)
sql/item_subselect.cc:
Complete struct READ_RECORD initialization with a new
member to unlock records.
sql/records.cc:
Extend READ_RECORD API with a method to unlock read records.
sql/sql_select.cc:
In JT_EQ_REF (join_read_key()) access method,
don't try to unlock rows in the handler, unless certain that
a) they were locked
b) they are not used.
sql/sql_select.h:
Add members to TABLE_REF to count TABLE_REF buffer usage count.
sql/structs.h:
Update declarations.
2009-11-03 17:58:54 +01:00
|
|
|
tab->read_record.unlock_row= rr_unlock_row;
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
*(last_changed_tab++)= tab;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2006-10-31 18:51:09 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-09-03 08:50:36 +02:00
|
|
|
join->exec();
|
2006-10-31 18:51:09 +01:00
|
|
|
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
/* Enable the optimizations back */
|
|
|
|
for (JOIN_TAB **ptab= changed_tabs; ptab != last_changed_tab; ptab++)
|
2006-10-31 18:51:09 +01:00
|
|
|
{
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
JOIN_TAB *tab= *ptab;
|
|
|
|
tab->read_record.ref_length= 0;
|
2017-08-25 13:36:13 +02:00
|
|
|
tab->read_first_record= tab->save_read_first_record;
|
|
|
|
tab->read_record.read_record_func= tab->save_read_record;
|
2006-10-31 18:51:09 +01:00
|
|
|
}
|
2002-09-03 08:50:36 +02:00
|
|
|
executed= 1;
|
2017-04-20 22:09:31 +02:00
|
|
|
if (!(uncacheable() & ~UNCACHEABLE_EXPLAIN) &&
|
|
|
|
!item->with_recursive_reference)
|
2012-08-25 08:15:57 +02:00
|
|
|
item->make_const();
|
2004-09-10 01:22:44 +02:00
|
|
|
thd->where= save_where;
|
|
|
|
thd->lex->current_select= save_select;
|
2011-09-03 10:50:56 +02:00
|
|
|
DBUG_RETURN(join->error || thd->is_fatal_error || thd->is_error());
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
2004-09-10 01:22:44 +02:00
|
|
|
thd->where= save_where;
|
|
|
|
thd->lex->current_select= save_select;
|
2002-09-03 08:50:36 +02:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
int subselect_union_engine::exec()
|
2002-09-03 08:50:36 +02:00
|
|
|
{
|
2004-09-10 01:22:44 +02:00
|
|
|
char const *save_where= thd->where;
|
2002-11-24 20:10:52 +01:00
|
|
|
int res= unit->exec();
|
2004-09-10 01:22:44 +02:00
|
|
|
thd->where= save_where;
|
2002-11-24 20:10:52 +01:00
|
|
|
return res;
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
|
|
|
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2006-10-31 18:51:09 +01:00
|
|
|
/*
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
Search for at least one row satisfying select condition
|
2006-10-31 18:51:09 +01:00
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
subselect_uniquesubquery_engine::scan_table()
|
|
|
|
|
|
|
|
DESCRIPTION
|
|
|
|
Scan the table using sequential access until we find at least one row
|
|
|
|
satisfying select condition.
|
|
|
|
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
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.
|
2006-10-31 18:51:09 +01:00
|
|
|
|
|
|
|
RETURN
|
|
|
|
FALSE - OK
|
|
|
|
TRUE - Error
|
|
|
|
*/
|
|
|
|
|
|
|
|
int subselect_uniquesubquery_engine::scan_table()
|
2003-07-07 17:40:19 +02:00
|
|
|
{
|
|
|
|
int error;
|
|
|
|
TABLE *table= tab->table;
|
2006-10-31 18:51:09 +01:00
|
|
|
DBUG_ENTER("subselect_uniquesubquery_engine::scan_table");
|
|
|
|
|
2012-10-08 16:10:30 +02:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2006-10-31 18:51:09 +01:00
|
|
|
table->file->extra_opt(HA_EXTRA_CACHE,
|
2016-07-29 18:21:08 +02:00
|
|
|
get_thd()->variables.read_buff_size);
|
2006-10-31 18:51:09 +01:00
|
|
|
table->null_row= 0;
|
|
|
|
for (;;)
|
2003-07-07 17:40:19 +02:00
|
|
|
{
|
This is based on the userstatv2 patch from Percona and OurDelta.
The original code comes, as far as I know, from Google (Mark Callaghan's team) with additional work from Percona, Ourdelta and Weldon Whipple.
This code provides the same functionallity, but with a lot of changes to make it faster and better fit the MariaDB infrastucture.
Added new status variables:
- Com_show_client_statistics, Com_show_index_statistics, Com_show_table_statistics, Com_show_user_statistics
- Access_denied_errors, Busy_time (clock time), Binlog_bytes_written, Cpu_time, Empty_queries, Rows_sent, Rows_read
Added new variable / startup option 'userstat' to control if user statistics should be enabled or not
Added my_getcputime(); Returns cpu time used by this thread.
New FLUSH commands:
- FLUSH SLOW QUERY LOG
- FLUSH TABLE_STATISTICS
- FLUSH INDEX_STATISTICS
- FLUSH USER_STATISTICS
- FLUSH CLIENT_STATISTICS
New SHOW commands:
- SHOW CLIENT_STATISTICS
- SHOW USER_STATISTICS
- SHOW TABLE_STATISTICS
- SHOW INDEX_STATISTICS
New Information schemas:
- CLIENT_STATISTICS
- USER_STATISTICS
- INDEX_STATISTICS
- TABLE_STATISTICS
Added support for all new flush commands to mysqladmin
Added handler::ha_... wrappers for all handler read calls to do statistics counting
- Changed all code to use new ha_... calls
- Count number of read rows, changed rows and rows read trough an index
Added counting of number of bytes sent to binary log (status variable Binlog_bytes_written)
Added counting of access denied errors (status variable Access_denied_erors)
Bugs fixed:
- Fixed bug in add_to_status() and add_diff_to_status() where longlong variables where threated as long
- CLOCK_GETTIME was not propely working on Linuxm
client/mysqladmin.cc:
Added support for all new flush commmands and some common combinations:
flush-slow-log
flush-table-statistics
flush-index-statistics
flush-user-statistics
flush-client-statistics
flush-all-status
flush-all-statistics
configure.in:
Added checking if clock_gettime needs the librt.
(Fixes Bug #37639 clock_gettime is never used/enabled in Linux/Unix)
include/my_sys.h:
Added my_getcputime()
include/mysql_com.h:
Added LIST_PROCESS_HOST_LEN & new REFRESH target defines
mysql-test/r/information_schema.result:
New information schema tables added
mysql-test/r/information_schema_all_engines.result:
New information schema tables added
mysql-test/r/information_schema_db.result:
New information schema tables added
mysql-test/r/log_slow.result:
Added testing that flosh slow query logs is accepted
mysql-test/r/status_user.result:
Basic testing of user, client, table and index statistics
mysql-test/t/log_slow.test:
Added testing that flosh slow query logs is accepted
mysql-test/t/status_user-master.opt:
Ensure that we get a fresh restart before running status_user.test
mysql-test/t/status_user.test:
Basic testing of user, client, table and index statistics
mysys/my_getsystime.c:
Added my_getcputime()
Returns cpu time used by this thread.
sql/authors.h:
Updated authors to have core and original MySQL developers first.
sql/event_data_objects.cc:
Updated call to mysql_reset_thd_for_next_command()
sql/event_db_repository.cc:
Changed to use new ha_... calls
sql/filesort.cc:
Changed to use new ha_... calls
sql/ha_partition.cc:
Changed to use new ha_... calls
Fixed comment syntax
sql/handler.cc:
Changed to use new ha_... calls
Reset table statistics
Added code to update global table and index status
Added counting of rows changed
sql/handler.h:
Added table and index statistics variables
Added function reset_statistics()
Added handler::ha_... wrappers for all handler read calls to do statistics counting
Protected all normal read calls to ensure we use the new calls in the server.
Made ha_partition a friend class so that partition code can call the old read functions
sql/item_subselect.cc:
Changed to use new ha_... calls
sql/lex.h:
Added keywords for new information schema tables and flush commands
sql/log.cc:
Added flush_slow_log()
Added counting of number of bytes sent to binary log
Removed not needed test of thd (It's used before, so it's safe to use)
Added THD object to MYSQL_BIN_LOG::write_cache() to simplify statistics counting
sql/log.h:
Added new parameter to write_cache()
Added flush_slow_log() functions.
sql/log_event.cc:
Updated call to mysql_reset_thd_for_next_command()
Changed to use new ha_... calls
sql/log_event_old.cc:
Updated call to mysql_reset_thd_for_next_command()
Changed to use new ha_... calls
sql/mysql_priv.h:
Updated call to mysql_reset_thd_for_next_command()
Added new statistics functions and variables needed by these.
sql/mysqld.cc:
Added new statistics variables and structures to handle these
Added new status variables:
- Com_show_client_statistics, Com_show_index_statistics, Com_show_table_statistics, Com_show_user_statistics
- Access_denied_errors, Busy_time (clock time), Binlog_bytes_written, Cpu_time, Empty_queries, Rows_set, Rows_read
Added new option 'userstat' to control if user statistics should be enabled or not
sql/opt_range.cc:
Changed to use new ha_... calls
sql/opt_range.h:
Changed to use new ha_... calls
sql/opt_sum.cc:
Changed to use new ha_... calls
sql/records.cc:
Changed to use new ha_... calls
sql/set_var.cc:
Added variable 'userstat'
sql/sp.cc:
Changed to use new ha_... calls
sql/sql_acl.cc:
Changed to use new ha_... calls
Added counting of access_denied_errors
sql/sql_base.cc:
Added call to statistics functions
sql/sql_class.cc:
Added usage of org_status_var, to store status variables at start of command
Added functions THD::update_stats(), THD::update_all_stats()
Fixed bug in add_to_status() and add_diff_to_status() where longlong variables where threated as long
sql/sql_class.h:
Added new status variables to status_var
Moved variables that was not ulong in status_var last.
Added variables to THD for storing temporary values during statistics counting
sql/sql_connect.cc:
Variables and functions to calculate user and client statistics
Added counting of access_denied_errors and lost_connections
sql/sql_cursor.cc:
Changed to use new ha_... calls
sql/sql_handler.cc:
Changed to use new ha_... calls
sql/sql_help.cc:
Changed to use new ha_... calls
sql/sql_insert.cc:
Changed to use new ha_... calls
sql/sql_lex.h:
Added SQLCOM_SHOW_USER_STATS, SQLCOM_SHOW_TABLE_STATS, SQLCOM_SHOW_INDEX_STATS, SQLCOM_SHOW_CLIENT_STATS
sql/sql_parse.cc:
Added handling of:
- SHOW CLIENT_STATISTICS
- SHOW USER_STATISTICS
- SHOW TABLE_STATISTICS
- SHOW INDEX_STATISTICS
Added handling of new FLUSH commands:
- FLUSH SLOW QUERY LOGS
- FLUSH TABLE_STATISTICS
- FLUSH INDEX_STATISTICS
- FLUSH USER_STATISTICS
- FLUSH CLIENT_STATISTICS
Added THD parameter to mysql_reset_thd_for_next_command()
Added initialization and calls to user statistics functions
Added increment of statistics variables empty_queries, rows_sent and access_denied_errors.
Added counting of cpu time per query
sql/sql_plugin.cc:
Changed to use new ha_... calls
sql/sql_prepare.cc:
Updated call to mysql_reset_thd_for_next_command()
sql/sql_select.cc:
Changed to use new ha_... calls
Indentation changes
sql/sql_servers.cc:
Changed to use new ha_... calls
sql/sql_show.cc:
Added counting of access denied errors
Added function for new information schema tables:
- CLIENT_STATISTICS
- USER_STATISTICS
- INDEX_STATISTICS
- TABLE_STATISTICS
Changed to use new ha_... calls
sql/sql_table.cc:
Changed to use new ha_... calls
sql/sql_udf.cc:
Changed to use new ha_... calls
sql/sql_update.cc:
Changed to use new ha_... calls
sql/sql_yacc.yy:
Add new show and flush commands
sql/structs.h:
Add name_length to KEY to avoid some strlen
Added cache_name to KEY for fast storage of keyvalue in cache
Added structs USER_STATS, TABLE_STATS, INDEX_STATS
Added function prototypes for statistics functions
sql/table.cc:
Store db+table+index name into keyinfo->cache_name
sql/table.h:
Added new information schema tables
sql/tztime.cc:
Changed to use new ha_... calls
2009-10-19 19:14:48 +02:00
|
|
|
error=table->file->ha_rnd_next(table->record[0]);
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely(error))
|
|
|
|
{
|
2010-02-19 22:55:57 +01:00
|
|
|
if (error == HA_ERR_END_OF_FILE)
|
|
|
|
{
|
|
|
|
error= 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
error= report_error(table, error);
|
|
|
|
break;
|
|
|
|
}
|
2006-10-31 18:51:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!cond || cond->val_int())
|
|
|
|
{
|
|
|
|
empty_result_set= FALSE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
table->file->ha_rnd_end();
|
|
|
|
DBUG_RETURN(error != 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
/**
|
|
|
|
Copy ref key for index access into the only subquery table.
|
2007-05-17 18:38:34 +02:00
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
@details
|
2012-09-14 10:26:01 +02:00
|
|
|
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.
|
2006-10-31 18:51:09 +01:00
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
@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.
|
2006-10-31 18:51:09 +01:00
|
|
|
*/
|
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
bool subselect_uniquesubquery_engine::copy_ref_key(bool skip_constants)
|
2006-10-31 18:51:09 +01:00
|
|
|
{
|
|
|
|
DBUG_ENTER("subselect_uniquesubquery_engine::copy_ref_key");
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
for (store_key **copy= tab->ref.key_copy ; *copy ; copy++)
|
|
|
|
{
|
|
|
|
enum store_key::store_key_result store_res;
|
2012-09-17 10:13:46 +02:00
|
|
|
if (skip_constants && (*copy)->store_key_is_const())
|
2012-01-21 08:54:43 +01:00
|
|
|
continue;
|
2010-02-19 22:55:57 +01:00
|
|
|
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.
|
|
|
|
*/
|
2012-09-14 10:26:01 +02:00
|
|
|
DBUG_RETURN(true);
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
}
|
2012-09-14 10:26:01 +02:00
|
|
|
DBUG_RETURN(false);
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
/**
|
|
|
|
Execute subselect via unique index lookup
|
2004-08-12 16:31:23 +02:00
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
@details
|
2006-10-31 18:51:09 +01:00
|
|
|
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)
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
the caller doesn't distinguish between NULL and FALSE result and we just
|
2006-10-31 18:51:09 +01:00
|
|
|
return FALSE.
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
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.
|
2006-10-31 18:51:09 +01:00
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
@returns
|
|
|
|
@retval 0 OK
|
|
|
|
@retval 1 notify caller to call Item_subselect::reset(),
|
|
|
|
in most cases reset() sets the result to NULL
|
2006-10-31 18:51:09 +01:00
|
|
|
*/
|
2004-08-12 16:31:23 +02:00
|
|
|
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
int subselect_uniquesubquery_engine::exec()
|
2006-10-31 18:51:09 +01:00
|
|
|
{
|
|
|
|
DBUG_ENTER("subselect_uniquesubquery_engine::exec");
|
|
|
|
int error;
|
|
|
|
TABLE *table= tab->table;
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
empty_result_set= TRUE;
|
2007-05-17 18:38:34 +02:00
|
|
|
table->status= 0;
|
2012-09-14 10:26:01 +02:00
|
|
|
Item_in_subselect *in_subs= (Item_in_subselect *) item;
|
2012-09-17 10:13:46 +02:00
|
|
|
|
|
|
|
if (!tab->preread_init_done && tab->preread_init())
|
|
|
|
DBUG_RETURN(1);
|
2006-10-31 18:51:09 +01:00
|
|
|
|
2012-09-14 10:26:01 +02:00
|
|
|
if (in_subs->left_expr_has_null())
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
|
|
|
/*
|
2012-09-14 10:26:01 +02:00
|
|
|
The case when all values in left_expr are NULL is handled by
|
|
|
|
Item_in_optimizer::val_int().
|
2010-02-19 22:55:57 +01:00
|
|
|
*/
|
2012-09-14 10:26:01 +02:00
|
|
|
if (in_subs->is_top_level_item())
|
|
|
|
DBUG_RETURN(1); /* notify caller to call reset() and set NULL value. */
|
|
|
|
else
|
|
|
|
DBUG_RETURN(scan_table());
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
2012-09-14 10:26:01 +02:00
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
if (copy_ref_key(true))
|
2007-05-17 18:38:34 +02:00
|
|
|
{
|
2012-09-14 10:26:01 +02:00
|
|
|
/* We know that there will be no rows even if we scan. */
|
|
|
|
in_subs->value= 0;
|
2007-05-17 18:38:34 +02:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
2006-10-31 18:51:09 +01:00
|
|
|
|
2012-10-08 16:10:30 +02:00
|
|
|
if (!table->file->inited &&
|
|
|
|
(error= table->file->ha_index_init(tab->ref.key, 0)))
|
|
|
|
{
|
|
|
|
(void) report_error(table, error);
|
|
|
|
DBUG_RETURN(true);
|
|
|
|
}
|
|
|
|
|
This is based on the userstatv2 patch from Percona and OurDelta.
The original code comes, as far as I know, from Google (Mark Callaghan's team) with additional work from Percona, Ourdelta and Weldon Whipple.
This code provides the same functionallity, but with a lot of changes to make it faster and better fit the MariaDB infrastucture.
Added new status variables:
- Com_show_client_statistics, Com_show_index_statistics, Com_show_table_statistics, Com_show_user_statistics
- Access_denied_errors, Busy_time (clock time), Binlog_bytes_written, Cpu_time, Empty_queries, Rows_sent, Rows_read
Added new variable / startup option 'userstat' to control if user statistics should be enabled or not
Added my_getcputime(); Returns cpu time used by this thread.
New FLUSH commands:
- FLUSH SLOW QUERY LOG
- FLUSH TABLE_STATISTICS
- FLUSH INDEX_STATISTICS
- FLUSH USER_STATISTICS
- FLUSH CLIENT_STATISTICS
New SHOW commands:
- SHOW CLIENT_STATISTICS
- SHOW USER_STATISTICS
- SHOW TABLE_STATISTICS
- SHOW INDEX_STATISTICS
New Information schemas:
- CLIENT_STATISTICS
- USER_STATISTICS
- INDEX_STATISTICS
- TABLE_STATISTICS
Added support for all new flush commands to mysqladmin
Added handler::ha_... wrappers for all handler read calls to do statistics counting
- Changed all code to use new ha_... calls
- Count number of read rows, changed rows and rows read trough an index
Added counting of number of bytes sent to binary log (status variable Binlog_bytes_written)
Added counting of access denied errors (status variable Access_denied_erors)
Bugs fixed:
- Fixed bug in add_to_status() and add_diff_to_status() where longlong variables where threated as long
- CLOCK_GETTIME was not propely working on Linuxm
client/mysqladmin.cc:
Added support for all new flush commmands and some common combinations:
flush-slow-log
flush-table-statistics
flush-index-statistics
flush-user-statistics
flush-client-statistics
flush-all-status
flush-all-statistics
configure.in:
Added checking if clock_gettime needs the librt.
(Fixes Bug #37639 clock_gettime is never used/enabled in Linux/Unix)
include/my_sys.h:
Added my_getcputime()
include/mysql_com.h:
Added LIST_PROCESS_HOST_LEN & new REFRESH target defines
mysql-test/r/information_schema.result:
New information schema tables added
mysql-test/r/information_schema_all_engines.result:
New information schema tables added
mysql-test/r/information_schema_db.result:
New information schema tables added
mysql-test/r/log_slow.result:
Added testing that flosh slow query logs is accepted
mysql-test/r/status_user.result:
Basic testing of user, client, table and index statistics
mysql-test/t/log_slow.test:
Added testing that flosh slow query logs is accepted
mysql-test/t/status_user-master.opt:
Ensure that we get a fresh restart before running status_user.test
mysql-test/t/status_user.test:
Basic testing of user, client, table and index statistics
mysys/my_getsystime.c:
Added my_getcputime()
Returns cpu time used by this thread.
sql/authors.h:
Updated authors to have core and original MySQL developers first.
sql/event_data_objects.cc:
Updated call to mysql_reset_thd_for_next_command()
sql/event_db_repository.cc:
Changed to use new ha_... calls
sql/filesort.cc:
Changed to use new ha_... calls
sql/ha_partition.cc:
Changed to use new ha_... calls
Fixed comment syntax
sql/handler.cc:
Changed to use new ha_... calls
Reset table statistics
Added code to update global table and index status
Added counting of rows changed
sql/handler.h:
Added table and index statistics variables
Added function reset_statistics()
Added handler::ha_... wrappers for all handler read calls to do statistics counting
Protected all normal read calls to ensure we use the new calls in the server.
Made ha_partition a friend class so that partition code can call the old read functions
sql/item_subselect.cc:
Changed to use new ha_... calls
sql/lex.h:
Added keywords for new information schema tables and flush commands
sql/log.cc:
Added flush_slow_log()
Added counting of number of bytes sent to binary log
Removed not needed test of thd (It's used before, so it's safe to use)
Added THD object to MYSQL_BIN_LOG::write_cache() to simplify statistics counting
sql/log.h:
Added new parameter to write_cache()
Added flush_slow_log() functions.
sql/log_event.cc:
Updated call to mysql_reset_thd_for_next_command()
Changed to use new ha_... calls
sql/log_event_old.cc:
Updated call to mysql_reset_thd_for_next_command()
Changed to use new ha_... calls
sql/mysql_priv.h:
Updated call to mysql_reset_thd_for_next_command()
Added new statistics functions and variables needed by these.
sql/mysqld.cc:
Added new statistics variables and structures to handle these
Added new status variables:
- Com_show_client_statistics, Com_show_index_statistics, Com_show_table_statistics, Com_show_user_statistics
- Access_denied_errors, Busy_time (clock time), Binlog_bytes_written, Cpu_time, Empty_queries, Rows_set, Rows_read
Added new option 'userstat' to control if user statistics should be enabled or not
sql/opt_range.cc:
Changed to use new ha_... calls
sql/opt_range.h:
Changed to use new ha_... calls
sql/opt_sum.cc:
Changed to use new ha_... calls
sql/records.cc:
Changed to use new ha_... calls
sql/set_var.cc:
Added variable 'userstat'
sql/sp.cc:
Changed to use new ha_... calls
sql/sql_acl.cc:
Changed to use new ha_... calls
Added counting of access_denied_errors
sql/sql_base.cc:
Added call to statistics functions
sql/sql_class.cc:
Added usage of org_status_var, to store status variables at start of command
Added functions THD::update_stats(), THD::update_all_stats()
Fixed bug in add_to_status() and add_diff_to_status() where longlong variables where threated as long
sql/sql_class.h:
Added new status variables to status_var
Moved variables that was not ulong in status_var last.
Added variables to THD for storing temporary values during statistics counting
sql/sql_connect.cc:
Variables and functions to calculate user and client statistics
Added counting of access_denied_errors and lost_connections
sql/sql_cursor.cc:
Changed to use new ha_... calls
sql/sql_handler.cc:
Changed to use new ha_... calls
sql/sql_help.cc:
Changed to use new ha_... calls
sql/sql_insert.cc:
Changed to use new ha_... calls
sql/sql_lex.h:
Added SQLCOM_SHOW_USER_STATS, SQLCOM_SHOW_TABLE_STATS, SQLCOM_SHOW_INDEX_STATS, SQLCOM_SHOW_CLIENT_STATS
sql/sql_parse.cc:
Added handling of:
- SHOW CLIENT_STATISTICS
- SHOW USER_STATISTICS
- SHOW TABLE_STATISTICS
- SHOW INDEX_STATISTICS
Added handling of new FLUSH commands:
- FLUSH SLOW QUERY LOGS
- FLUSH TABLE_STATISTICS
- FLUSH INDEX_STATISTICS
- FLUSH USER_STATISTICS
- FLUSH CLIENT_STATISTICS
Added THD parameter to mysql_reset_thd_for_next_command()
Added initialization and calls to user statistics functions
Added increment of statistics variables empty_queries, rows_sent and access_denied_errors.
Added counting of cpu time per query
sql/sql_plugin.cc:
Changed to use new ha_... calls
sql/sql_prepare.cc:
Updated call to mysql_reset_thd_for_next_command()
sql/sql_select.cc:
Changed to use new ha_... calls
Indentation changes
sql/sql_servers.cc:
Changed to use new ha_... calls
sql/sql_show.cc:
Added counting of access denied errors
Added function for new information schema tables:
- CLIENT_STATISTICS
- USER_STATISTICS
- INDEX_STATISTICS
- TABLE_STATISTICS
Changed to use new ha_... calls
sql/sql_table.cc:
Changed to use new ha_... calls
sql/sql_udf.cc:
Changed to use new ha_... calls
sql/sql_update.cc:
Changed to use new ha_... calls
sql/sql_yacc.yy:
Add new show and flush commands
sql/structs.h:
Add name_length to KEY to avoid some strlen
Added cache_name to KEY for fast storage of keyvalue in cache
Added structs USER_STATS, TABLE_STATS, INDEX_STATS
Added function prototypes for statistics functions
sql/table.cc:
Store db+table+index name into keyinfo->cache_name
sql/table.h:
Added new information schema tables
sql/tztime.cc:
Changed to use new ha_... calls
2009-10-19 19:14:48 +02:00
|
|
|
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);
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely(error &&
|
|
|
|
error != HA_ERR_KEY_NOT_FOUND && error != HA_ERR_END_OF_FILE))
|
2004-08-12 16:31:23 +02:00
|
|
|
error= report_error(table, error);
|
2003-07-07 17:40:19 +02:00
|
|
|
else
|
|
|
|
{
|
2004-08-12 16:31:23 +02:00
|
|
|
error= 0;
|
|
|
|
table->null_row= 0;
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (!table->status && (!cond || cond->val_int()))
|
|
|
|
{
|
|
|
|
((Item_in_subselect *) item)->value= 1;
|
|
|
|
empty_result_set= FALSE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
((Item_in_subselect *) item)->value= 0;
|
2003-07-07 17:40:19 +02:00
|
|
|
}
|
2004-08-12 16:31:23 +02:00
|
|
|
|
2003-09-29 11:39:38 +02:00
|
|
|
DBUG_RETURN(error != 0);
|
2003-07-07 23:08:00 +02:00
|
|
|
}
|
|
|
|
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
/*
|
2010-03-09 11:14:06 +01:00
|
|
|
TIMOUR: write comment
|
2010-02-19 22:55:57 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
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));
|
2010-03-09 11:14:06 +01:00
|
|
|
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely(error && error != HA_ERR_KEY_NOT_FOUND &&
|
|
|
|
error != HA_ERR_END_OF_FILE))
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
2010-03-09 11:14:06 +01:00
|
|
|
/*
|
|
|
|
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);
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
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);
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2003-09-29 11:39:38 +02:00
|
|
|
subselect_uniquesubquery_engine::~subselect_uniquesubquery_engine()
|
2003-07-07 23:08:00 +02:00
|
|
|
{
|
2003-09-29 11:39:38 +02:00
|
|
|
/* Tell handler we don't need the index anymore */
|
2010-01-28 14:48:33 +01:00
|
|
|
//psergey-merge-todo: the following was gone in 6.0:
|
|
|
|
//psergey-merge: don't need this after all: tab->table->file->ha_index_end();
|
2003-07-07 23:08:00 +02:00
|
|
|
}
|
|
|
|
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
/**
|
|
|
|
Execute subselect via unique index lookup
|
2006-10-31 18:51:09 +01:00
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
@details
|
2006-10-31 19:30:40 +01:00
|
|
|
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.
|
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
@todo
|
2006-10-31 19:30:40 +01:00
|
|
|
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.
|
2006-10-31 18:51:09 +01:00
|
|
|
|
2006-10-31 19:30:40 +01:00
|
|
|
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.
|
2006-10-31 18:51:09 +01:00
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
@returns
|
|
|
|
@retval 0 OK
|
|
|
|
@retval 1 notify caller to call Item_subselect::reset(),
|
|
|
|
in most cases reset() sets the result to NULL
|
2006-10-31 18:51:09 +01:00
|
|
|
*/
|
|
|
|
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
int subselect_indexsubquery_engine::exec()
|
2003-07-07 23:08:00 +02:00
|
|
|
{
|
2010-05-26 22:18:18 +02:00
|
|
|
DBUG_ENTER("subselect_indexsubquery_engine");
|
2003-07-07 23:08:00 +02:00
|
|
|
int error;
|
2003-09-08 20:58:09 +02:00
|
|
|
bool null_finding= 0;
|
2003-07-07 23:08:00 +02:00
|
|
|
TABLE *table= tab->table;
|
2012-09-14 10:26:01 +02:00
|
|
|
Item_in_subselect *in_subs= (Item_in_subselect *) item;
|
2003-07-17 18:39:31 +02:00
|
|
|
|
2003-07-07 23:08:00 +02:00
|
|
|
((Item_in_subselect *) item)->value= 0;
|
2006-10-31 18:51:09 +01:00
|
|
|
empty_result_set= TRUE;
|
2007-11-10 20:44:48 +01:00
|
|
|
table->status= 0;
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2003-07-17 18:39:31 +02:00
|
|
|
if (check_null)
|
|
|
|
{
|
2003-09-29 11:39:38 +02:00
|
|
|
/* We need to check for NULL if there wasn't a matching value */
|
2004-01-31 07:04:16 +01:00
|
|
|
*tab->ref.null_ref_key= 0; // Search first for not null
|
2003-07-17 18:39:31 +02:00
|
|
|
((Item_in_subselect *) item)->was_null= 0;
|
|
|
|
}
|
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
if (!tab->preread_init_done && tab->preread_init())
|
2006-10-31 18:51:09 +01:00
|
|
|
DBUG_RETURN(1);
|
|
|
|
|
2012-09-14 10:26:01 +02:00
|
|
|
if (in_subs->left_expr_has_null())
|
2007-11-10 20:44:48 +01:00
|
|
|
{
|
2012-09-14 10:26:01 +02:00
|
|
|
/*
|
|
|
|
The case when all values in left_expr are NULL is handled by
|
|
|
|
Item_in_optimizer::val_int().
|
2007-11-10 20:44:48 +01:00
|
|
|
*/
|
2012-09-14 10:26:01 +02:00
|
|
|
if (in_subs->is_top_level_item())
|
|
|
|
DBUG_RETURN(1); /* notify caller to call reset() and set NULL value. */
|
|
|
|
else
|
|
|
|
DBUG_RETURN(scan_table());
|
2007-11-10 20:44:48 +01:00
|
|
|
}
|
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
if (copy_ref_key(true))
|
2012-09-14 10:26:01 +02:00
|
|
|
{
|
|
|
|
/* We know that there will be no rows even if we scan. */
|
|
|
|
in_subs->value= 0;
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
2004-08-12 16:31:23 +02:00
|
|
|
|
2012-10-08 16:10:30 +02:00
|
|
|
if (!table->file->inited &&
|
|
|
|
(error= table->file->ha_index_init(tab->ref.key, 1)))
|
|
|
|
{
|
|
|
|
(void) report_error(table, error);
|
|
|
|
DBUG_RETURN(true);
|
|
|
|
}
|
2013-01-15 19:13:32 +01:00
|
|
|
|
This is based on the userstatv2 patch from Percona and OurDelta.
The original code comes, as far as I know, from Google (Mark Callaghan's team) with additional work from Percona, Ourdelta and Weldon Whipple.
This code provides the same functionallity, but with a lot of changes to make it faster and better fit the MariaDB infrastucture.
Added new status variables:
- Com_show_client_statistics, Com_show_index_statistics, Com_show_table_statistics, Com_show_user_statistics
- Access_denied_errors, Busy_time (clock time), Binlog_bytes_written, Cpu_time, Empty_queries, Rows_sent, Rows_read
Added new variable / startup option 'userstat' to control if user statistics should be enabled or not
Added my_getcputime(); Returns cpu time used by this thread.
New FLUSH commands:
- FLUSH SLOW QUERY LOG
- FLUSH TABLE_STATISTICS
- FLUSH INDEX_STATISTICS
- FLUSH USER_STATISTICS
- FLUSH CLIENT_STATISTICS
New SHOW commands:
- SHOW CLIENT_STATISTICS
- SHOW USER_STATISTICS
- SHOW TABLE_STATISTICS
- SHOW INDEX_STATISTICS
New Information schemas:
- CLIENT_STATISTICS
- USER_STATISTICS
- INDEX_STATISTICS
- TABLE_STATISTICS
Added support for all new flush commands to mysqladmin
Added handler::ha_... wrappers for all handler read calls to do statistics counting
- Changed all code to use new ha_... calls
- Count number of read rows, changed rows and rows read trough an index
Added counting of number of bytes sent to binary log (status variable Binlog_bytes_written)
Added counting of access denied errors (status variable Access_denied_erors)
Bugs fixed:
- Fixed bug in add_to_status() and add_diff_to_status() where longlong variables where threated as long
- CLOCK_GETTIME was not propely working on Linuxm
client/mysqladmin.cc:
Added support for all new flush commmands and some common combinations:
flush-slow-log
flush-table-statistics
flush-index-statistics
flush-user-statistics
flush-client-statistics
flush-all-status
flush-all-statistics
configure.in:
Added checking if clock_gettime needs the librt.
(Fixes Bug #37639 clock_gettime is never used/enabled in Linux/Unix)
include/my_sys.h:
Added my_getcputime()
include/mysql_com.h:
Added LIST_PROCESS_HOST_LEN & new REFRESH target defines
mysql-test/r/information_schema.result:
New information schema tables added
mysql-test/r/information_schema_all_engines.result:
New information schema tables added
mysql-test/r/information_schema_db.result:
New information schema tables added
mysql-test/r/log_slow.result:
Added testing that flosh slow query logs is accepted
mysql-test/r/status_user.result:
Basic testing of user, client, table and index statistics
mysql-test/t/log_slow.test:
Added testing that flosh slow query logs is accepted
mysql-test/t/status_user-master.opt:
Ensure that we get a fresh restart before running status_user.test
mysql-test/t/status_user.test:
Basic testing of user, client, table and index statistics
mysys/my_getsystime.c:
Added my_getcputime()
Returns cpu time used by this thread.
sql/authors.h:
Updated authors to have core and original MySQL developers first.
sql/event_data_objects.cc:
Updated call to mysql_reset_thd_for_next_command()
sql/event_db_repository.cc:
Changed to use new ha_... calls
sql/filesort.cc:
Changed to use new ha_... calls
sql/ha_partition.cc:
Changed to use new ha_... calls
Fixed comment syntax
sql/handler.cc:
Changed to use new ha_... calls
Reset table statistics
Added code to update global table and index status
Added counting of rows changed
sql/handler.h:
Added table and index statistics variables
Added function reset_statistics()
Added handler::ha_... wrappers for all handler read calls to do statistics counting
Protected all normal read calls to ensure we use the new calls in the server.
Made ha_partition a friend class so that partition code can call the old read functions
sql/item_subselect.cc:
Changed to use new ha_... calls
sql/lex.h:
Added keywords for new information schema tables and flush commands
sql/log.cc:
Added flush_slow_log()
Added counting of number of bytes sent to binary log
Removed not needed test of thd (It's used before, so it's safe to use)
Added THD object to MYSQL_BIN_LOG::write_cache() to simplify statistics counting
sql/log.h:
Added new parameter to write_cache()
Added flush_slow_log() functions.
sql/log_event.cc:
Updated call to mysql_reset_thd_for_next_command()
Changed to use new ha_... calls
sql/log_event_old.cc:
Updated call to mysql_reset_thd_for_next_command()
Changed to use new ha_... calls
sql/mysql_priv.h:
Updated call to mysql_reset_thd_for_next_command()
Added new statistics functions and variables needed by these.
sql/mysqld.cc:
Added new statistics variables and structures to handle these
Added new status variables:
- Com_show_client_statistics, Com_show_index_statistics, Com_show_table_statistics, Com_show_user_statistics
- Access_denied_errors, Busy_time (clock time), Binlog_bytes_written, Cpu_time, Empty_queries, Rows_set, Rows_read
Added new option 'userstat' to control if user statistics should be enabled or not
sql/opt_range.cc:
Changed to use new ha_... calls
sql/opt_range.h:
Changed to use new ha_... calls
sql/opt_sum.cc:
Changed to use new ha_... calls
sql/records.cc:
Changed to use new ha_... calls
sql/set_var.cc:
Added variable 'userstat'
sql/sp.cc:
Changed to use new ha_... calls
sql/sql_acl.cc:
Changed to use new ha_... calls
Added counting of access_denied_errors
sql/sql_base.cc:
Added call to statistics functions
sql/sql_class.cc:
Added usage of org_status_var, to store status variables at start of command
Added functions THD::update_stats(), THD::update_all_stats()
Fixed bug in add_to_status() and add_diff_to_status() where longlong variables where threated as long
sql/sql_class.h:
Added new status variables to status_var
Moved variables that was not ulong in status_var last.
Added variables to THD for storing temporary values during statistics counting
sql/sql_connect.cc:
Variables and functions to calculate user and client statistics
Added counting of access_denied_errors and lost_connections
sql/sql_cursor.cc:
Changed to use new ha_... calls
sql/sql_handler.cc:
Changed to use new ha_... calls
sql/sql_help.cc:
Changed to use new ha_... calls
sql/sql_insert.cc:
Changed to use new ha_... calls
sql/sql_lex.h:
Added SQLCOM_SHOW_USER_STATS, SQLCOM_SHOW_TABLE_STATS, SQLCOM_SHOW_INDEX_STATS, SQLCOM_SHOW_CLIENT_STATS
sql/sql_parse.cc:
Added handling of:
- SHOW CLIENT_STATISTICS
- SHOW USER_STATISTICS
- SHOW TABLE_STATISTICS
- SHOW INDEX_STATISTICS
Added handling of new FLUSH commands:
- FLUSH SLOW QUERY LOGS
- FLUSH TABLE_STATISTICS
- FLUSH INDEX_STATISTICS
- FLUSH USER_STATISTICS
- FLUSH CLIENT_STATISTICS
Added THD parameter to mysql_reset_thd_for_next_command()
Added initialization and calls to user statistics functions
Added increment of statistics variables empty_queries, rows_sent and access_denied_errors.
Added counting of cpu time per query
sql/sql_plugin.cc:
Changed to use new ha_... calls
sql/sql_prepare.cc:
Updated call to mysql_reset_thd_for_next_command()
sql/sql_select.cc:
Changed to use new ha_... calls
Indentation changes
sql/sql_servers.cc:
Changed to use new ha_... calls
sql/sql_show.cc:
Added counting of access denied errors
Added function for new information schema tables:
- CLIENT_STATISTICS
- USER_STATISTICS
- INDEX_STATISTICS
- TABLE_STATISTICS
Changed to use new ha_... calls
sql/sql_table.cc:
Changed to use new ha_... calls
sql/sql_udf.cc:
Changed to use new ha_... calls
sql/sql_update.cc:
Changed to use new ha_... calls
sql/sql_yacc.yy:
Add new show and flush commands
sql/structs.h:
Add name_length to KEY to avoid some strlen
Added cache_name to KEY for fast storage of keyvalue in cache
Added structs USER_STATS, TABLE_STATS, INDEX_STATS
Added function prototypes for statistics functions
sql/table.cc:
Store db+table+index name into keyinfo->cache_name
sql/table.h:
Added new information schema tables
sql/tztime.cc:
Changed to use new ha_... calls
2009-10-19 19:14:48 +02:00
|
|
|
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);
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely(error &&
|
|
|
|
error != HA_ERR_KEY_NOT_FOUND && error != HA_ERR_END_OF_FILE))
|
2004-08-12 16:31:23 +02:00
|
|
|
error= report_error(table, error);
|
2003-07-07 23:08:00 +02:00
|
|
|
else
|
|
|
|
{
|
2004-08-12 16:31:23 +02:00
|
|
|
for (;;)
|
2003-07-07 17:40:19 +02:00
|
|
|
{
|
2004-08-12 16:31:23 +02:00
|
|
|
error= 0;
|
|
|
|
table->null_row= 0;
|
|
|
|
if (!table->status)
|
2003-07-07 23:08:00 +02:00
|
|
|
{
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if ((!cond || cond->val_int()) && (!having || having->val_int()))
|
2004-08-12 16:31:23 +02:00
|
|
|
{
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
empty_result_set= FALSE;
|
2004-08-12 16:31:23 +02:00
|
|
|
if (null_finding)
|
|
|
|
((Item_in_subselect *) item)->was_null= 1;
|
|
|
|
else
|
|
|
|
((Item_in_subselect *) item)->value= 1;
|
|
|
|
break;
|
|
|
|
}
|
This is based on the userstatv2 patch from Percona and OurDelta.
The original code comes, as far as I know, from Google (Mark Callaghan's team) with additional work from Percona, Ourdelta and Weldon Whipple.
This code provides the same functionallity, but with a lot of changes to make it faster and better fit the MariaDB infrastucture.
Added new status variables:
- Com_show_client_statistics, Com_show_index_statistics, Com_show_table_statistics, Com_show_user_statistics
- Access_denied_errors, Busy_time (clock time), Binlog_bytes_written, Cpu_time, Empty_queries, Rows_sent, Rows_read
Added new variable / startup option 'userstat' to control if user statistics should be enabled or not
Added my_getcputime(); Returns cpu time used by this thread.
New FLUSH commands:
- FLUSH SLOW QUERY LOG
- FLUSH TABLE_STATISTICS
- FLUSH INDEX_STATISTICS
- FLUSH USER_STATISTICS
- FLUSH CLIENT_STATISTICS
New SHOW commands:
- SHOW CLIENT_STATISTICS
- SHOW USER_STATISTICS
- SHOW TABLE_STATISTICS
- SHOW INDEX_STATISTICS
New Information schemas:
- CLIENT_STATISTICS
- USER_STATISTICS
- INDEX_STATISTICS
- TABLE_STATISTICS
Added support for all new flush commands to mysqladmin
Added handler::ha_... wrappers for all handler read calls to do statistics counting
- Changed all code to use new ha_... calls
- Count number of read rows, changed rows and rows read trough an index
Added counting of number of bytes sent to binary log (status variable Binlog_bytes_written)
Added counting of access denied errors (status variable Access_denied_erors)
Bugs fixed:
- Fixed bug in add_to_status() and add_diff_to_status() where longlong variables where threated as long
- CLOCK_GETTIME was not propely working on Linuxm
client/mysqladmin.cc:
Added support for all new flush commmands and some common combinations:
flush-slow-log
flush-table-statistics
flush-index-statistics
flush-user-statistics
flush-client-statistics
flush-all-status
flush-all-statistics
configure.in:
Added checking if clock_gettime needs the librt.
(Fixes Bug #37639 clock_gettime is never used/enabled in Linux/Unix)
include/my_sys.h:
Added my_getcputime()
include/mysql_com.h:
Added LIST_PROCESS_HOST_LEN & new REFRESH target defines
mysql-test/r/information_schema.result:
New information schema tables added
mysql-test/r/information_schema_all_engines.result:
New information schema tables added
mysql-test/r/information_schema_db.result:
New information schema tables added
mysql-test/r/log_slow.result:
Added testing that flosh slow query logs is accepted
mysql-test/r/status_user.result:
Basic testing of user, client, table and index statistics
mysql-test/t/log_slow.test:
Added testing that flosh slow query logs is accepted
mysql-test/t/status_user-master.opt:
Ensure that we get a fresh restart before running status_user.test
mysql-test/t/status_user.test:
Basic testing of user, client, table and index statistics
mysys/my_getsystime.c:
Added my_getcputime()
Returns cpu time used by this thread.
sql/authors.h:
Updated authors to have core and original MySQL developers first.
sql/event_data_objects.cc:
Updated call to mysql_reset_thd_for_next_command()
sql/event_db_repository.cc:
Changed to use new ha_... calls
sql/filesort.cc:
Changed to use new ha_... calls
sql/ha_partition.cc:
Changed to use new ha_... calls
Fixed comment syntax
sql/handler.cc:
Changed to use new ha_... calls
Reset table statistics
Added code to update global table and index status
Added counting of rows changed
sql/handler.h:
Added table and index statistics variables
Added function reset_statistics()
Added handler::ha_... wrappers for all handler read calls to do statistics counting
Protected all normal read calls to ensure we use the new calls in the server.
Made ha_partition a friend class so that partition code can call the old read functions
sql/item_subselect.cc:
Changed to use new ha_... calls
sql/lex.h:
Added keywords for new information schema tables and flush commands
sql/log.cc:
Added flush_slow_log()
Added counting of number of bytes sent to binary log
Removed not needed test of thd (It's used before, so it's safe to use)
Added THD object to MYSQL_BIN_LOG::write_cache() to simplify statistics counting
sql/log.h:
Added new parameter to write_cache()
Added flush_slow_log() functions.
sql/log_event.cc:
Updated call to mysql_reset_thd_for_next_command()
Changed to use new ha_... calls
sql/log_event_old.cc:
Updated call to mysql_reset_thd_for_next_command()
Changed to use new ha_... calls
sql/mysql_priv.h:
Updated call to mysql_reset_thd_for_next_command()
Added new statistics functions and variables needed by these.
sql/mysqld.cc:
Added new statistics variables and structures to handle these
Added new status variables:
- Com_show_client_statistics, Com_show_index_statistics, Com_show_table_statistics, Com_show_user_statistics
- Access_denied_errors, Busy_time (clock time), Binlog_bytes_written, Cpu_time, Empty_queries, Rows_set, Rows_read
Added new option 'userstat' to control if user statistics should be enabled or not
sql/opt_range.cc:
Changed to use new ha_... calls
sql/opt_range.h:
Changed to use new ha_... calls
sql/opt_sum.cc:
Changed to use new ha_... calls
sql/records.cc:
Changed to use new ha_... calls
sql/set_var.cc:
Added variable 'userstat'
sql/sp.cc:
Changed to use new ha_... calls
sql/sql_acl.cc:
Changed to use new ha_... calls
Added counting of access_denied_errors
sql/sql_base.cc:
Added call to statistics functions
sql/sql_class.cc:
Added usage of org_status_var, to store status variables at start of command
Added functions THD::update_stats(), THD::update_all_stats()
Fixed bug in add_to_status() and add_diff_to_status() where longlong variables where threated as long
sql/sql_class.h:
Added new status variables to status_var
Moved variables that was not ulong in status_var last.
Added variables to THD for storing temporary values during statistics counting
sql/sql_connect.cc:
Variables and functions to calculate user and client statistics
Added counting of access_denied_errors and lost_connections
sql/sql_cursor.cc:
Changed to use new ha_... calls
sql/sql_handler.cc:
Changed to use new ha_... calls
sql/sql_help.cc:
Changed to use new ha_... calls
sql/sql_insert.cc:
Changed to use new ha_... calls
sql/sql_lex.h:
Added SQLCOM_SHOW_USER_STATS, SQLCOM_SHOW_TABLE_STATS, SQLCOM_SHOW_INDEX_STATS, SQLCOM_SHOW_CLIENT_STATS
sql/sql_parse.cc:
Added handling of:
- SHOW CLIENT_STATISTICS
- SHOW USER_STATISTICS
- SHOW TABLE_STATISTICS
- SHOW INDEX_STATISTICS
Added handling of new FLUSH commands:
- FLUSH SLOW QUERY LOGS
- FLUSH TABLE_STATISTICS
- FLUSH INDEX_STATISTICS
- FLUSH USER_STATISTICS
- FLUSH CLIENT_STATISTICS
Added THD parameter to mysql_reset_thd_for_next_command()
Added initialization and calls to user statistics functions
Added increment of statistics variables empty_queries, rows_sent and access_denied_errors.
Added counting of cpu time per query
sql/sql_plugin.cc:
Changed to use new ha_... calls
sql/sql_prepare.cc:
Updated call to mysql_reset_thd_for_next_command()
sql/sql_select.cc:
Changed to use new ha_... calls
Indentation changes
sql/sql_servers.cc:
Changed to use new ha_... calls
sql/sql_show.cc:
Added counting of access denied errors
Added function for new information schema tables:
- CLIENT_STATISTICS
- USER_STATISTICS
- INDEX_STATISTICS
- TABLE_STATISTICS
Changed to use new ha_... calls
sql/sql_table.cc:
Changed to use new ha_... calls
sql/sql_udf.cc:
Changed to use new ha_... calls
sql/sql_update.cc:
Changed to use new ha_... calls
sql/sql_yacc.yy:
Add new show and flush commands
sql/structs.h:
Add name_length to KEY to avoid some strlen
Added cache_name to KEY for fast storage of keyvalue in cache
Added structs USER_STATS, TABLE_STATS, INDEX_STATS
Added function prototypes for statistics functions
sql/table.cc:
Store db+table+index name into keyinfo->cache_name
sql/table.h:
Added new information schema tables
sql/tztime.cc:
Changed to use new ha_... calls
2009-10-19 19:14:48 +02:00
|
|
|
error= table->file->ha_index_next_same(table->record[0],
|
|
|
|
tab->ref.key_buff,
|
|
|
|
tab->ref.key_length);
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely(error && error != HA_ERR_END_OF_FILE))
|
2004-08-12 16:31:23 +02:00
|
|
|
{
|
|
|
|
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 */
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely((error= (safe_index_read(tab) == 1))))
|
2004-08-12 16:31:23 +02:00
|
|
|
break;
|
2003-07-07 23:08:00 +02:00
|
|
|
}
|
2003-07-07 17:40:19 +02:00
|
|
|
}
|
|
|
|
}
|
2003-09-29 11:39:38 +02:00
|
|
|
DBUG_RETURN(error != 0);
|
2003-07-07 17:40:19 +02:00
|
|
|
}
|
|
|
|
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2017-05-06 13:05:59 +02:00
|
|
|
uint subselect_single_select_engine::cols() const
|
2002-09-03 08:50:36 +02:00
|
|
|
{
|
2010-01-17 15:51:10 +01:00
|
|
|
//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;
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2017-05-06 13:05:59 +02:00
|
|
|
uint subselect_union_engine::cols() const
|
2002-09-03 08:50:36 +02:00
|
|
|
{
|
2005-01-24 13:25:44 +01:00
|
|
|
DBUG_ASSERT(unit->is_prepared()); // should be called after fix_fields()
|
|
|
|
return unit->types.elements;
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2003-11-17 19:53:40 +01:00
|
|
|
uint8 subselect_single_select_engine::uncacheable()
|
2003-01-28 13:48:12 +01:00
|
|
|
{
|
|
|
|
return select_lex->uncacheable;
|
|
|
|
}
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2003-11-17 19:53:40 +01:00
|
|
|
uint8 subselect_union_engine::uncacheable()
|
2003-01-28 13:48:12 +01:00
|
|
|
{
|
|
|
|
return unit->uncacheable;
|
|
|
|
}
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2002-11-28 18:29:26 +01:00
|
|
|
void subselect_single_select_engine::exclude()
|
|
|
|
{
|
|
|
|
select_lex->master_unit()->exclude_level();
|
|
|
|
}
|
|
|
|
|
|
|
|
void subselect_union_engine::exclude()
|
|
|
|
{
|
|
|
|
unit->exclude_level();
|
|
|
|
}
|
2003-07-07 17:40:19 +02:00
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2003-09-14 08:40:57 +02:00
|
|
|
void subselect_uniquesubquery_engine::exclude()
|
2003-07-07 17:40:19 +02:00
|
|
|
{
|
|
|
|
//this never should be called
|
|
|
|
DBUG_ASSERT(0);
|
|
|
|
}
|
2003-10-16 23:36:01 +02:00
|
|
|
|
|
|
|
|
2010-05-26 22:18:18 +02:00
|
|
|
table_map subselect_engine::calc_const_tables(List<TABLE_LIST> &list)
|
2003-10-16 23:36:01 +02:00
|
|
|
{
|
|
|
|
table_map map= 0;
|
2010-05-26 22:18:18 +02:00
|
|
|
List_iterator<TABLE_LIST> ti(list);
|
|
|
|
TABLE_LIST *table;
|
|
|
|
//for (; table; table= table->next_leaf)
|
|
|
|
while ((table= ti++))
|
2003-10-16 23:36:01 +02:00
|
|
|
{
|
|
|
|
TABLE *tbl= table->table;
|
|
|
|
if (tbl && tbl->const_table)
|
|
|
|
map|= tbl->map;
|
|
|
|
}
|
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
table_map subselect_single_select_engine::upper_select_const_tables()
|
|
|
|
{
|
2010-06-10 22:45:22 +02:00
|
|
|
return calc_const_tables(select_lex->outer_select()->leaf_tables);
|
2003-10-16 23:36:01 +02:00
|
|
|
}
|
|
|
|
|
2003-10-28 11:45:37 +01:00
|
|
|
|
2003-10-16 23:36:01 +02:00
|
|
|
table_map subselect_union_engine::upper_select_const_tables()
|
|
|
|
{
|
2010-06-10 22:45:22 +02:00
|
|
|
return calc_const_tables(unit->outer_select()->leaf_tables);
|
2003-10-16 23:36:01 +02:00
|
|
|
}
|
2003-10-28 11:45:37 +01:00
|
|
|
|
|
|
|
|
2008-02-22 11:30:33 +01:00
|
|
|
void subselect_single_select_engine::print(String *str,
|
|
|
|
enum_query_type query_type)
|
2003-10-16 14:54:47 +02:00
|
|
|
{
|
2018-01-19 00:56:28 +01:00
|
|
|
With_clause* with_clause= select_lex->get_with_clause();
|
|
|
|
if (with_clause)
|
|
|
|
with_clause->print(str, query_type);
|
2016-07-29 18:21:08 +02:00
|
|
|
select_lex->print(get_thd(), str, query_type);
|
2003-10-16 14:54:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-22 11:30:33 +01:00
|
|
|
void subselect_union_engine::print(String *str, enum_query_type query_type)
|
2003-10-16 14:54:47 +02:00
|
|
|
{
|
2008-02-22 11:30:33 +01:00
|
|
|
unit->print(str, query_type);
|
2003-10-16 14:54:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-22 11:30:33 +01:00
|
|
|
void subselect_uniquesubquery_engine::print(String *str,
|
|
|
|
enum_query_type query_type)
|
2003-10-16 14:54:47 +02:00
|
|
|
{
|
2005-11-20 19:47:07 +01:00
|
|
|
str->append(STRING_WITH_LEN("<primary_index_lookup>("));
|
2008-02-22 11:30:33 +01:00
|
|
|
tab->ref.items[0]->print(str, query_type);
|
2005-11-20 19:47:07 +01:00
|
|
|
str->append(STRING_WITH_LEN(" in "));
|
2010-01-28 14:48:33 +01:00
|
|
|
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
|
2017-06-19 05:34:38 +02:00
|
|
|
str->append(&tab->table->s->table_name);
|
2003-10-19 13:22:17 +02:00
|
|
|
KEY *key_info= tab->table->key_info+ tab->ref.key;
|
2005-11-20 19:47:07 +01:00
|
|
|
str->append(STRING_WITH_LEN(" on "));
|
2017-06-18 11:28:40 +02:00
|
|
|
str->append(&key_info->name);
|
2003-10-16 14:54:47 +02:00
|
|
|
if (cond)
|
|
|
|
{
|
2005-11-20 19:47:07 +01:00
|
|
|
str->append(STRING_WITH_LEN(" where "));
|
2008-02-22 11:30:33 +01:00
|
|
|
cond->print(str, query_type);
|
2003-10-16 14:54:47 +02:00
|
|
|
}
|
|
|
|
str->append(')');
|
|
|
|
}
|
|
|
|
|
2010-01-28 14:48:33 +01:00
|
|
|
/*
|
|
|
|
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>("));
|
2013-05-21 21:00:08 +02:00
|
|
|
for (uint i= 0; i < key_info->user_defined_key_parts; i++)
|
2010-01-28 14:48:33 +01:00
|
|
|
tab->ref.items[i]->print(str);
|
|
|
|
str->append(STRING_WITH_LEN(" in "));
|
2017-06-19 05:34:38 +02:00
|
|
|
str->append(&tab->table->s->table_name);
|
2010-01-28 14:48:33 +01:00
|
|
|
str->append(STRING_WITH_LEN(" on "));
|
2017-06-18 11:28:40 +02:00
|
|
|
str->append(&key_info->name);
|
2010-01-28 14:48:33 +01:00
|
|
|
if (cond)
|
|
|
|
{
|
|
|
|
str->append(STRING_WITH_LEN(" where "));
|
|
|
|
cond->print(str);
|
|
|
|
}
|
|
|
|
str->append(')');
|
|
|
|
}
|
|
|
|
*/
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2008-02-22 11:30:33 +01:00
|
|
|
void subselect_indexsubquery_engine::print(String *str,
|
|
|
|
enum_query_type query_type)
|
2003-10-16 14:54:47 +02:00
|
|
|
{
|
2005-11-20 19:47:07 +01:00
|
|
|
str->append(STRING_WITH_LEN("<index_lookup>("));
|
2008-02-22 11:30:33 +01:00
|
|
|
tab->ref.items[0]->print(str, query_type);
|
2005-11-20 19:47:07 +01:00
|
|
|
str->append(STRING_WITH_LEN(" in "));
|
Table definition cache, part 2
The table opening process now works the following way:
- Create common TABLE_SHARE object
- Read the .frm file and unpack it into the TABLE_SHARE object
- Create a TABLE object based on the information in the TABLE_SHARE
object and open a handler to the table object
Other noteworthy changes:
- In TABLE_SHARE the most common strings are now LEX_STRING's
- Better error message when table is not found
- Variable table_cache is now renamed 'table_open_cache'
- New variable 'table_definition_cache' that is the number of table defintions that will be cached
- strxnmov() calls are now fixed to avoid overflows
- strxnmov() will now always add one end \0 to result
- engine objects are now created with a TABLE_SHARE object instead of a TABLE object.
- After creating a field object one must call field->init(table) before using it
- For a busy system this change will give you:
- Less memory usage for table object
- Faster opening of tables (if it's has been in use or is in table definition cache)
- Allow you to cache many table definitions objects
- Faster drop of table
mysql-test/mysql-test-run.sh:
Fixed some problems with --gdb option
Test both with socket and tcp/ip port that all old servers are killed
mysql-test/r/flush_table.result:
More tests with lock table with 2 threads + flush table
mysql-test/r/information_schema.result:
Removed old (now wrong) result
mysql-test/r/innodb.result:
Better error messages (thanks to TDC patch)
mysql-test/r/merge.result:
Extra flush table test
mysql-test/r/ndb_bitfield.result:
Better error messages (thanks to TDC patch)
mysql-test/r/ndb_partition_error.result:
Better error messages (thanks to TDC patch)
mysql-test/r/query_cache.result:
Remove tables left from old tests
mysql-test/r/temp_table.result:
Test truncate with temporary tables
mysql-test/r/variables.result:
Table_cache -> Table_open_cache
mysql-test/t/flush_table.test:
More tests with lock table with 2 threads + flush table
mysql-test/t/merge.test:
Extra flush table test
mysql-test/t/multi_update.test:
Added 'sleep' to make test predictable
mysql-test/t/query_cache.test:
Remove tables left from old tests
mysql-test/t/temp_table.test:
Test truncate with temporary tables
mysql-test/t/variables.test:
Table_cache -> Table_open_cache
mysql-test/valgrind.supp:
Remove warning that may happens becasue threads dies in different order
mysys/hash.c:
Fixed wrong DBUG_PRINT
mysys/mf_dirname.c:
More DBUG
mysys/mf_pack.c:
Better comment
mysys/mf_tempdir.c:
More DBUG
Ensure that we call cleanup_dirname() on all temporary directory paths.
If we don't do this, we will get a failure when comparing temporary table
names as in some cases the temporary table name is run through convert_dirname())
mysys/my_alloc.c:
Indentation fix
sql/examples/ha_example.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/examples/ha_example.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/examples/ha_tina.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/examples/ha_tina.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/field.cc:
Update for table definition cache:
- Field creation now takes TABLE_SHARE instead of TABLE as argument
(This is becasue field definitions are now cached in TABLE_SHARE)
When a field is created, one now must call field->init(TABLE) before using it
- Use s->db instead of s->table_cache_key
- Added Field::clone() to create a field in TABLE from a field in TABLE_SHARE
- make_field() takes TABLE_SHARE as argument instead of TABLE
- move_field() -> move_field_offset()
sql/field.h:
Update for table definition cache:
- Field creation now takes TABLE_SHARE instead of TABLE as argument
(This is becasue field definitions are now cached in TABLE_SHARE)
When a field is created, one now must call field->init(TABLE) before using it
- Added Field::clone() to create a field in TABLE from a field in TABLE_SHARE
- make_field() takes TABLE_SHARE as argument instead of TABLE
- move_field() -> move_field_offset()
sql/ha_archive.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_archive.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_berkeley.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Changed name of argument create() to not hide internal 'table' variable.
table->s -> table_share
sql/ha_berkeley.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_blackhole.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_blackhole.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_federated.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Fixed comments
Remove index variable and replace with pointers (simple optimization)
move_field() -> move_field_offset()
Removed some strlen() calls
sql/ha_federated.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_heap.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Simplify delete_table() and create() as the given file names are now without extension
sql/ha_heap.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_innodb.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_innodb.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_myisam.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Remove not needed fn_format()
Fixed for new table->s structure
sql/ha_myisam.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_myisammrg.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Don't set 'is_view' for MERGE tables
Use new interface to find_temporary_table()
sql/ha_myisammrg.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Added flag HA_NO_COPY_ON_ALTER
sql/ha_ndbcluster.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Fixed wrong calls to strxnmov()
Give error HA_ERR_TABLE_DEF_CHANGED if table definition has changed
drop_table -> intern_drop_table()
table->s -> table_share
Move part_info to TABLE
Fixed comments & DBUG print's
New arguments to print_error()
sql/ha_ndbcluster.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
sql/ha_partition.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
We can't set up or use part_info when creating handler as there is not yet any table object
New ha_intialise() to work with TDC (Done by Mikael)
sql/ha_partition.h:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
Got set_part_info() from Mikael
sql/handler.cc:
We new use TABLE_SHARE instead of TABLE when creating engine handlers
ha_delete_table() now also takes database as an argument
handler::ha_open() now takes TABLE as argument
ha_open() now calls ha_allocate_read_write_set()
Simplify ha_allocate_read_write_set()
Remove ha_deallocate_read_write_set()
Use table_share (Cached by table definition cache)
sql/handler.h:
New table flag: HA_NO_COPY_ON_ALTER (used by merge tables)
Remove ha_deallocate_read_write_set()
get_new_handler() now takes TABLE_SHARE as argument
ha_delete_table() now gets database as argument
sql/item.cc:
table_name and db are now LEX_STRING objects
When creating fields, we have now have to call field->init(table)
move_field -> move_field_offset()
sql/item.h:
tmp_table_field_from_field_type() now takes an extra paramenter 'fixed_length' to allow one to force usage of CHAR
instead of BLOB
sql/item_cmpfunc.cc:
Fixed call to tmp_table_field_from_field_type()
sql/item_create.cc:
Assert if new not handled cast type
sql/item_func.cc:
When creating fields, we have now have to call field->init(table)
dummy_table used by 'sp' now needs a TABLE_SHARE object
sql/item_subselect.cc:
Trivial code cleanups
sql/item_sum.cc:
When creating fields, we have now have to call field->init(table)
sql/item_timefunc.cc:
Item_func_str_to_date::tmp_table_field() now replaced by call to
tmp_table_field_from_field_type() (see item_timefunc.h)
sql/item_timefunc.h:
Simply tmp_table_field()
sql/item_uniq.cc:
When creating fields, we have now have to call field->init(table)
sql/key.cc:
Added 'KEY' argument to 'find_ref_key' to simplify code
sql/lock.cc:
More debugging
Use create_table_def_key() to create key for table cache
Allocate TABLE_SHARE properly when creating name lock
Fix that locked_table_name doesn't test same table twice
sql/mysql_priv.h:
New functions for table definition cache
New interfaces to a lot of functions.
New faster interface to find_temporary_table() and close_temporary_table()
sql/mysqld.cc:
Added support for table definition cache of size 'table_def_size'
Fixed som calls to strnmov()
Changed name of 'table_cache' to 'table_open_cache'
sql/opt_range.cc:
Use new interfaces
Fixed warnings from valgrind
sql/parse_file.cc:
Safer calls to strxnmov()
Fixed typo
sql/set_var.cc:
Added variable 'table_definition_cache'
Variable table_cache renamed to 'table_open_cache'
sql/slave.cc:
Use new interface
sql/sp.cc:
Proper use of TABLE_SHARE
sql/sp_head.cc:
Remove compiler warnings
We have now to call field->init(table)
sql/sp_head.h:
Pointers to parsed strings are now const
sql/sql_acl.cc:
table_name is now a LEX_STRING
sql/sql_base.cc:
Main implementation of table definition cache
(The #ifdef's are there for the future when table definition cache will replace open table cache)
Now table definitions are cached indepndent of open tables, which will speed up things when a table is in use at once from several places
Views are not yet cached; For the moment we only cache if a table is a view or not.
Faster implementation of find_temorary_table()
Replace 'wait_for_refresh()' with the more general function 'wait_for_condition()'
Drop table is slightly faster as we can use the table definition cache to know the type of the table
sql/sql_cache.cc:
table_cache_key and table_name are now LEX_STRING
'sDBUG print fixes
sql/sql_class.cc:
table_cache_key is now a LEX_STRING
safer strxnmov()
sql/sql_class.h:
Added number of open table shares (table definitions)
sql/sql_db.cc:
safer strxnmov()
sql/sql_delete.cc:
Use new interface to find_temporary_table()
sql/sql_derived.cc:
table_name is now a LEX_STRING
sql/sql_handler.cc:
TABLE_SHARE->db and TABLE_SHARE->table_name are now LEX_STRING's
sql/sql_insert.cc:
TABLE_SHARE->db and TABLE_SHARE->table_name are now LEX_STRING's
sql/sql_lex.cc:
Make parsed string a const (to quickly find out if anything is trying to change the query string)
sql/sql_lex.h:
Make parsed string a const (to quickly find out if anything is trying to change the query string)
sql/sql_load.cc:
Safer strxnmov()
sql/sql_parse.cc:
Better error if wrong DB name
sql/sql_partition.cc:
part_info moved to TABLE from TABLE_SHARE
Indentation changes
sql/sql_select.cc:
Indentation fixes
Call field->init(TABLE) for new created fields
Update create_tmp_table() to use TABLE_SHARE properly
sql/sql_select.h:
Call field->init(TABLE) for new created fields
sql/sql_show.cc:
table_name is now a LEX_STRING
part_info moved to TABLE
sql/sql_table.cc:
Use table definition cache to speed up delete of tables
Fixed calls to functions with new interfaces
Don't use 'share_not_to_be_used'
Instead of doing openfrm() when doing repair, we now have to call
get_table_share() followed by open_table_from_share().
Replace some fn_format() with faster unpack_filename().
Safer strxnmov()
part_info is now in TABLE
Added Mikaels patch for partition and ALTER TABLE
Instead of using 'TABLE_SHARE->is_view' use 'table_flags() & HA_NO_COPY_ON_ALTER
sql/sql_test.cc:
table_name and table_cache_key are now LEX_STRING's
sql/sql_trigger.cc:
TABLE_SHARE->db and TABLE_SHARE->table_name are now LEX_STRING's
safer strxnmov()
Removed compiler warnings
sql/sql_update.cc:
Call field->init(TABLE) after field is created
sql/sql_view.cc:
safer strxnmov()
Create common TABLE_SHARE object for views to allow us to cache if table is a view
sql/structs.h:
Added SHOW_TABLE_DEFINITIONS
sql/table.cc:
Creation and destruct of TABLE_SHARE objects that are common for many TABLE objects
The table opening process now works the following way:
- Create common TABLE_SHARE object
- Read the .frm file and unpack it into the TABLE_SHARE object
- Create a TABLE object based on the information in the TABLE_SHARE
object and open a handler to the table object
open_table_def() is written in such a way that it should be trival to add parsing of the .frm files in new formats
sql/table.h:
TABLE objects for the same database table now share a common TABLE_SHARE object
In TABLE_SHARE the most common strings are now LEX_STRING's
sql/unireg.cc:
Changed arguments to rea_create_table() to have same order as other functions
Call field->init(table) for new created fields
sql/unireg.h:
Added OPEN_VIEW
strings/strxnmov.c:
Change strxnmov() to always add end \0
This makes usage of strxnmov() safer as most of MySQL code assumes that strxnmov() will create a null terminated string
2005-11-23 21:45:02 +01:00
|
|
|
str->append(tab->table->s->table_name.str, tab->table->s->table_name.length);
|
2003-10-19 13:22:17 +02:00
|
|
|
KEY *key_info= tab->table->key_info+ tab->ref.key;
|
2005-11-20 19:47:07 +01:00
|
|
|
str->append(STRING_WITH_LEN(" on "));
|
2017-06-18 11:28:40 +02:00
|
|
|
str->append(&key_info->name);
|
2003-10-16 14:54:47 +02:00
|
|
|
if (check_null)
|
2005-11-20 19:47:07 +01:00
|
|
|
str->append(STRING_WITH_LEN(" checking NULL"));
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (cond)
|
2003-10-16 14:54:47 +02:00
|
|
|
{
|
2005-11-20 19:47:07 +01:00
|
|
|
str->append(STRING_WITH_LEN(" where "));
|
2008-02-22 11:30:33 +01:00
|
|
|
cond->print(str, query_type);
|
2003-10-16 14:54:47 +02:00
|
|
|
}
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
if (having)
|
|
|
|
{
|
|
|
|
str->append(STRING_WITH_LEN(" having "));
|
2008-02-22 11:30:33 +01:00
|
|
|
having->print(str, query_type);
|
BUG#24127: (a,b) IN (SELECT c,d ...) can produce wrong results if a and/or b are NULLs:
- Make the code produce correct result: use an array of triggers to turn on/off equalities for each
compared column. Also turn on/off optimizations based on those equalities.
- Make EXPLAIN output show "Full scan on NULL key" for tables for which we switch between
ref/unique_subquery/index_subquery and ALL access.
- index_subquery engine now has HAVING clause when it is needed, and it is
displayed in EXPLAIN EXTENDED
- Fix incorrect presense of "Using index" for index/unique-based subqueries (BUG#22930)
// bk trigger note: this commit refers to BUG#24127
mysql-test/r/ndb_subquery.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect2.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Updated test results (checked)
mysql-test/r/subselect3.result:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
mysql-test/t/subselect3.test:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Testcases
sql/item_cmpfunc.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- For row-based IN subqueries, use one flag per each column. Set the flags appropriately before
running the subquery.
sql/item_cmpfunc.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added Item_func_trig_cond::get_triv_var()
sql/item_subselect.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/item_subselect.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Item_subselect::exec() and subselect_*_engine::exec() don't have parameter
anymore - now Item_subselect owns the pushed down predicates guard flags.
- A correct set of conditional predicates is now pushed into row-based IN
subquery.
- select_indexsubquery_engine now has "HAVING clause" (needed for correct query
results), and it is shown in EXPLAIN EXTENDED
sql/mysql_priv.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/mysqld.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Added "in_having_cond" special Item name
sql/sql_lex.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
sql/sql_select.cc:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix subquery optimization code to match the changes in what kinds of
conditions are pushed down into subqueries
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
sql/sql_select.h:
BUG#24127: wrong result for (null,not-null) IN (SELECT a,b ...)
- Make "ref" analyzer be able to work with conditional equalities
- Fix wrong EXPLAIN output in some queries with subquery (BUG#22390)
2007-01-12 21:22:41 +01:00
|
|
|
}
|
2003-10-16 14:54:47 +02:00
|
|
|
str->append(')');
|
|
|
|
}
|
2004-05-07 22:06:11 +02:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
/**
|
|
|
|
change select_result object of engine.
|
2004-05-07 22:06:11 +02:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
@param si new subselect Item
|
|
|
|
@param res new select_result object
|
2011-05-11 23:14:15 +02:00
|
|
|
@param temp temporary assignment
|
2004-05-07 22:06:11 +02:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
@retval
|
2004-10-20 03:04:37 +02:00
|
|
|
FALSE OK
|
2007-10-11 19:29:09 +02:00
|
|
|
@retval
|
2004-10-20 03:04:37 +02:00
|
|
|
TRUE error
|
2004-05-07 22:06:11 +02:00
|
|
|
*/
|
|
|
|
|
2011-05-04 17:08:58 +02:00
|
|
|
bool
|
|
|
|
subselect_single_select_engine::change_result(Item_subselect *si,
|
|
|
|
select_result_interceptor *res,
|
|
|
|
bool temp)
|
2004-05-07 22:06:11 +02:00
|
|
|
{
|
2013-02-26 00:20:17 +01:00
|
|
|
DBUG_ENTER("subselect_single_select_engine::change_result");
|
2004-05-07 22:06:11 +02:00
|
|
|
item= si;
|
2011-05-04 17:08:58 +02:00
|
|
|
if (temp)
|
2011-05-11 23:14:15 +02:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
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.
|
|
|
|
*/
|
2011-05-04 17:08:58 +02:00
|
|
|
thd->change_item_tree((Item**) &result, (Item*)res);
|
2011-05-11 23:14:15 +02:00
|
|
|
}
|
2011-05-04 17:08:58 +02:00
|
|
|
else
|
|
|
|
result= res;
|
2011-05-17 23:47:56 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
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.
|
|
|
|
*/
|
2014-10-14 18:36:50 +02:00
|
|
|
DBUG_RETURN(select_lex->join->change_result(res, NULL));
|
2004-05-07 22:06:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
/**
|
|
|
|
change select_result object of engine.
|
2004-05-07 22:06:11 +02:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
@param si new subselect Item
|
|
|
|
@param res new select_result object
|
2004-05-07 22:06:11 +02:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
@retval
|
2004-10-20 03:04:37 +02:00
|
|
|
FALSE OK
|
2007-10-11 19:29:09 +02:00
|
|
|
@retval
|
2004-10-20 03:04:37 +02:00
|
|
|
TRUE error
|
2004-05-07 22:06:11 +02:00
|
|
|
*/
|
|
|
|
|
2004-10-20 03:04:37 +02:00
|
|
|
bool subselect_union_engine::change_result(Item_subselect *si,
|
2011-05-04 17:08:58 +02:00
|
|
|
select_result_interceptor *res,
|
|
|
|
bool temp)
|
2004-05-07 22:06:11 +02:00
|
|
|
{
|
|
|
|
item= si;
|
|
|
|
int rc= unit->change_result(res, result);
|
2011-05-04 17:08:58 +02:00
|
|
|
if (temp)
|
|
|
|
thd->change_item_tree((Item**) &result, (Item*)res);
|
|
|
|
else
|
|
|
|
result= res;
|
2004-05-07 22:06:11 +02:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
/**
|
|
|
|
change select_result emulation, never should be called.
|
2004-05-07 22:06:11 +02:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
@param si new subselect Item
|
|
|
|
@param res new select_result object
|
2004-05-07 22:06:11 +02:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
@retval
|
2004-10-20 03:04:37 +02:00
|
|
|
FALSE OK
|
2007-10-11 19:29:09 +02:00
|
|
|
@retval
|
2004-10-20 03:04:37 +02:00
|
|
|
TRUE error
|
2004-05-07 22:06:11 +02:00
|
|
|
*/
|
|
|
|
|
2011-05-04 17:08:58 +02:00
|
|
|
bool
|
|
|
|
subselect_uniquesubquery_engine::change_result(Item_subselect *si,
|
|
|
|
select_result_interceptor *res,
|
|
|
|
bool temp
|
|
|
|
__attribute__((unused)))
|
2004-05-07 22:06:11 +02:00
|
|
|
{
|
|
|
|
DBUG_ASSERT(0);
|
2004-10-20 03:04:37 +02:00
|
|
|
return TRUE;
|
2004-05-07 22:06:11 +02:00
|
|
|
}
|
2004-10-27 20:11:06 +02:00
|
|
|
|
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
/**
|
|
|
|
Report about presence of tables in subquery.
|
2004-10-27 20:11:06 +02:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
@retval
|
2004-10-27 20:11:06 +02:00
|
|
|
TRUE there are not tables used in subquery
|
2007-10-11 19:29:09 +02:00
|
|
|
@retval
|
2004-10-27 20:11:06 +02:00
|
|
|
FALSE there are some tables in subquery
|
|
|
|
*/
|
|
|
|
bool subselect_single_select_engine::no_tables()
|
|
|
|
{
|
|
|
|
return(select_lex->table_list.elements == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-12-12 03:57:23 +01:00
|
|
|
/*
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
/**
|
|
|
|
Report about presence of tables in subquery.
|
2004-10-27 20:11:06 +02:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
@retval
|
2004-10-27 20:11:06 +02:00
|
|
|
TRUE there are not tables used in subquery
|
2007-10-11 19:29:09 +02:00
|
|
|
@retval
|
2004-10-27 20:11:06 +02:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
/**
|
|
|
|
Report about presence of tables in subquery.
|
2004-10-27 20:11:06 +02:00
|
|
|
|
2007-10-11 19:29:09 +02:00
|
|
|
@retval
|
2004-10-27 20:11:06 +02:00
|
|
|
TRUE there are not tables used in subquery
|
2007-10-11 19:29:09 +02:00
|
|
|
@retval
|
2004-10-27 20:11:06 +02:00
|
|
|
FALSE there are some tables in subquery
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool subselect_uniquesubquery_engine::no_tables()
|
|
|
|
{
|
|
|
|
/* returning value is correct, but this method should never be called */
|
2010-02-19 22:55:57 +01:00
|
|
|
DBUG_ASSERT(FALSE);
|
2004-10-27 20:11:06 +02:00
|
|
|
return 0;
|
|
|
|
}
|
2010-01-28 14:48:33 +01:00
|
|
|
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
WL#1110 - Implementation of class subselect_hash_sj_engine
|
|
|
|
******************************************************************************/
|
|
|
|
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
/**
|
|
|
|
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.
|
2010-03-09 11:14:06 +01:00
|
|
|
|
|
|
|
@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
|
2010-02-19 22:55:57 +01:00
|
|
|
*/
|
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
subselect_hash_sj_engine::exec_strategy
|
|
|
|
subselect_hash_sj_engine::get_strategy_using_schema()
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
|
|
|
Item_in_subselect *item_in= (Item_in_subselect *) item;
|
|
|
|
|
|
|
|
if (item_in->is_top_level_item())
|
2010-03-09 11:14:06 +01:00
|
|
|
return COMPLETE_MATCH;
|
2010-02-19 22:55:57 +01:00
|
|
|
else
|
|
|
|
{
|
2016-11-21 19:33:06 +01:00
|
|
|
List_iterator<Item> inner_col_it(*item_in->unit->get_column_types(false));
|
2010-02-19 22:55:57 +01:00
|
|
|
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)
|
2010-03-09 11:14:06 +01:00
|
|
|
return PARTIAL_MATCH;
|
|
|
|
return COMPLETE_MATCH;
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
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.
|
2010-03-09 11:14:06 +01:00
|
|
|
|
|
|
|
@retval PARTIAL_MATCH if a partial match should be used
|
|
|
|
@retval COMPLETE_MATCH if a complete match (index lookup) should be used
|
2010-02-19 22:55:57 +01:00
|
|
|
*/
|
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
subselect_hash_sj_engine::exec_strategy
|
|
|
|
subselect_hash_sj_engine::get_strategy_using_data()
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
|
|
|
Item_in_subselect *item_in= (Item_in_subselect *) item;
|
|
|
|
select_materialize_with_stats *result_sink=
|
|
|
|
(select_materialize_with_stats *) result;
|
|
|
|
Item *outer_col;
|
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
/*
|
|
|
|
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;
|
2010-02-19 22:55:57 +01:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2011-07-13 16:09:09 +02:00
|
|
|
if (result_sink->get_null_count_of_col(i) == tmp_table->file->stats.records)
|
2010-02-19 22:55:57 +01:00
|
|
|
++count_null_only_columns;
|
2011-08-22 23:00:13 +02:00
|
|
|
if (result_sink->get_null_count_of_col(i))
|
|
|
|
++count_columns_with_nulls;
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* If no column contains NULLs use regular hash index lookups. */
|
|
|
|
if (!count_partial_match_columns)
|
2010-03-09 11:14:06 +01:00
|
|
|
return COMPLETE_MATCH;
|
|
|
|
return PARTIAL_MATCH;
|
|
|
|
}
|
2010-02-19 22:55:57 +01:00
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
|
|
|
|
void
|
|
|
|
subselect_hash_sj_engine::choose_partial_match_strategy(
|
|
|
|
bool has_non_null_key, bool has_covering_null_row,
|
2015-07-06 19:24:14 +02:00
|
|
|
MY_BITMAP *partial_match_key_parts_arg)
|
2010-03-09 11:14:06 +01:00
|
|
|
{
|
2010-11-30 00:27:14 +01:00
|
|
|
ulonglong pm_buff_size;
|
2010-03-09 11:14:06 +01:00
|
|
|
|
|
|
|
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
|
2016-06-08 13:14:42 +02:00
|
|
|
when this will be overridden is when the total size of all buffers for the
|
2010-03-09 11:14:06 +01:00
|
|
|
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,
|
2015-07-06 19:24:14 +02:00
|
|
|
partial_match_key_parts_arg);
|
2010-03-09 11:14:06 +01:00
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2010-11-30 00:27:14 +01:00
|
|
|
ulonglong subselect_hash_sj_engine::rowid_merge_buff_size(
|
2010-03-09 11:14:06 +01:00
|
|
|
bool has_non_null_key, bool has_covering_null_row,
|
|
|
|
MY_BITMAP *partial_match_key_parts)
|
|
|
|
{
|
2010-11-30 00:27:14 +01:00
|
|
|
/* Total size of all buffers used by partial matching. */
|
|
|
|
ulonglong buff_size;
|
2010-03-09 11:14:06 +01:00
|
|
|
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;
|
2011-11-24 22:15:40 +01:00
|
|
|
ha_rows max_null_row;
|
2010-03-09 11:14:06 +01:00
|
|
|
|
|
|
|
/* 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 */
|
2011-11-24 22:15:40 +01:00
|
|
|
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);
|
2010-03-09 11:14:06 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return buff_size;
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Initialize a MY_BITMAP with a buffer allocated on the current
|
|
|
|
memory root.
|
|
|
|
TIMOUR: move to bitmap C file?
|
|
|
|
*/
|
|
|
|
|
|
|
|
static my_bool
|
2014-01-02 10:19:19 +01:00
|
|
|
my_bitmap_init_memroot(MY_BITMAP *map, uint n_bits, MEM_ROOT *mem_root)
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
|
|
|
my_bitmap_map *bitmap_buf;
|
|
|
|
|
|
|
|
if (!(bitmap_buf= (my_bitmap_map*) alloc_root(mem_root,
|
|
|
|
bitmap_buffer_size(n_bits))) ||
|
2014-01-02 10:19:19 +01:00
|
|
|
my_bitmap_init(map, bitmap_buf, n_bits, FALSE))
|
2010-02-19 22:55:57 +01:00
|
|
|
return TRUE;
|
|
|
|
bitmap_clear_all(map);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-28 14:48:33 +01:00
|
|
|
/**
|
|
|
|
Create all structures needed for IN execution that can live between PS
|
|
|
|
reexecution.
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
@param tmp_columns the items that produce the data for the temp table
|
2011-02-18 12:17:28 +01:00
|
|
|
@param subquery_id subquery's identifier (to make "<subquery%d>" name for
|
|
|
|
EXPLAIN)
|
2010-02-19 22:55:57 +01:00
|
|
|
|
|
|
|
@details
|
2010-01-28 14:48:33 +01:00
|
|
|
- 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
|
|
|
|
*/
|
|
|
|
|
2011-05-25 17:31:13 +02:00
|
|
|
bool subselect_hash_sj_engine::init(List<Item> *tmp_columns, uint subquery_id)
|
2010-01-28 14:48:33 +01:00
|
|
|
{
|
2016-07-29 18:21:08 +02:00
|
|
|
THD *thd= get_thd();
|
2017-03-14 11:52:00 +01:00
|
|
|
select_unit *result_sink;
|
2010-02-19 22:55:57 +01:00
|
|
|
/* Options to create_tmp_table. */
|
2011-04-25 17:22:25 +02:00
|
|
|
ulonglong tmp_create_options= thd->variables.option_bits | TMP_TABLE_ALL_COLUMNS;
|
2010-02-19 22:55:57 +01:00
|
|
|
/* | TMP_TABLE_FORCE_MYISAM; TIMOUR: force MYISAM */
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-07-16 12:52:02 +02:00
|
|
|
DBUG_ENTER("subselect_hash_sj_engine::init");
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2014-01-02 10:19:19 +01:00
|
|
|
if (my_bitmap_init_memroot(&non_null_key_parts, tmp_columns->elements,
|
2010-02-19 22:55:57 +01:00
|
|
|
thd->mem_root) ||
|
2014-01-02 10:19:19 +01:00
|
|
|
my_bitmap_init_memroot(&partial_match_key_parts, tmp_columns->elements,
|
2010-02-19 22:55:57 +01:00
|
|
|
thd->mem_root))
|
|
|
|
DBUG_RETURN(TRUE);
|
2010-01-28 14:48:33 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
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.
|
|
|
|
*/
|
2010-02-19 22:55:57 +01:00
|
|
|
/*
|
|
|
|
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))
|
2010-01-28 14:48:33 +01:00
|
|
|
DBUG_RETURN(TRUE);
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
*/
|
2015-04-22 11:29:56 +02:00
|
|
|
if (!(result_sink= new (thd->mem_root) select_materialize_with_stats(thd)))
|
2010-02-19 22:55:57 +01:00
|
|
|
DBUG_RETURN(TRUE);
|
2011-05-25 17:31:13 +02:00
|
|
|
|
2010-04-06 23:29:09 +02:00
|
|
|
char buf[32];
|
2018-01-07 17:03:44 +01:00
|
|
|
LEX_CSTRING name;
|
|
|
|
name.length= my_snprintf(buf, sizeof(buf), "<subquery%u>", subquery_id);
|
|
|
|
if (!(name.str= (char*) thd->memdup(buf, name.length + 1)))
|
2010-04-06 23:29:09 +02:00
|
|
|
DBUG_RETURN(TRUE);
|
2010-02-19 22:55:57 +01:00
|
|
|
|
2010-07-16 12:52:02 +02:00
|
|
|
result_sink->get_tmp_table_param()->materialized_subquery= true;
|
2012-02-25 01:50:22 +01:00
|
|
|
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;
|
|
|
|
}
|
2010-07-16 12:52:02 +02:00
|
|
|
if (result_sink->create_result_table(thd, tmp_columns, TRUE,
|
|
|
|
tmp_create_options,
|
2018-01-07 17:03:44 +01:00
|
|
|
&name, TRUE, TRUE, FALSE, 0))
|
2010-01-28 14:48:33 +01:00
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
|
2010-07-16 12:52:02 +02:00
|
|
|
tmp_table= result_sink->table;
|
|
|
|
result= result_sink;
|
2010-01-28 14:48:33 +01:00
|
|
|
|
|
|
|
/*
|
2016-12-16 13:44:08 +01:00
|
|
|
If the subquery has blobs, or the total key length is bigger than
|
2010-02-19 22:55:57 +01:00
|
|
|
some length, or the total number of key parts is more than the
|
2016-12-16 13:44:08 +01:00
|
|
|
allowed maximum (currently MAX_REF_PARTS == 32), then the created
|
2010-02-19 22:55:57 +01:00
|
|
|
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.
|
2010-01-28 14:48:33 +01:00
|
|
|
*/
|
|
|
|
if (tmp_table->s->keys == 0)
|
|
|
|
{
|
2011-12-07 16:21:51 +01:00
|
|
|
//fprintf(stderr, "Q: %s\n", current_thd->query());
|
|
|
|
DBUG_ASSERT(0);
|
2010-01-28 14:48:33 +01:00
|
|
|
DBUG_ASSERT(
|
|
|
|
tmp_table->s->uniques ||
|
|
|
|
tmp_table->key_info->key_length >= tmp_table->file->max_key_length() ||
|
2013-05-21 21:00:08 +02:00
|
|
|
tmp_table->key_info->user_defined_key_parts >
|
|
|
|
tmp_table->file->max_key_parts());
|
2010-01-28 14:48:33 +01:00
|
|
|
free_tmp_table(thd, tmp_table);
|
2010-02-19 22:55:57 +01:00
|
|
|
tmp_table= NULL;
|
2010-01-28 14:48:33 +01:00
|
|
|
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.
|
|
|
|
*/
|
2010-02-19 22:55:57 +01:00
|
|
|
DBUG_ASSERT(tmp_table->s->keys == 1 &&
|
|
|
|
((Item_in_subselect *) item)->left_expr->cols() ==
|
2013-05-21 21:00:08 +02:00
|
|
|
tmp_table->key_info->user_defined_key_parts);
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
if (make_semi_join_conds() ||
|
|
|
|
/* A unique_engine is used both for complete and partial matching. */
|
|
|
|
!(lookup_engine= make_unique_engine()))
|
2010-02-19 22:55:57 +01:00
|
|
|
DBUG_RETURN(TRUE);
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-07-16 12:52:02 +02:00
|
|
|
/*
|
|
|
|
Repeat name resolution for 'cond' since cond is not part of any
|
|
|
|
clause of the query, and it is not 'fixed' during JOIN::prepare.
|
|
|
|
*/
|
2018-06-05 08:25:39 +02:00
|
|
|
if (semi_join_conds &&
|
|
|
|
semi_join_conds->fix_fields_if_needed(thd, (Item**)&semi_join_conds))
|
2010-07-16 12:52:02 +02:00
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
/* Let our engine reuse this query plan for materialization. */
|
|
|
|
materialize_join= materialize_engine->join;
|
2014-10-14 18:36:50 +02:00
|
|
|
materialize_join->change_result(result, NULL);
|
2010-07-16 12:52:02 +02:00
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
2010-01-28 14:48:33 +01:00
|
|
|
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
/*
|
|
|
|
Create an artificial condition to post-filter those rows matched by index
|
|
|
|
lookups that cannot be distinguished by the index lookup procedure.
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
@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()
|
|
|
|
{
|
2010-01-28 14:48:33 +01:00
|
|
|
/*
|
|
|
|
Table reference for tmp_table that is used to resolve column references
|
|
|
|
(Item_fields) to columns in tmp_table.
|
|
|
|
*/
|
|
|
|
TABLE_LIST *tmp_table_ref;
|
2010-02-19 22:55:57 +01:00
|
|
|
/* Name resolution context for all tmp_table columns created below. */
|
|
|
|
Name_resolution_context *context;
|
|
|
|
Item_in_subselect *item_in= (Item_in_subselect *) item;
|
2018-01-07 17:03:44 +01:00
|
|
|
LEX_CSTRING table_name;
|
2010-02-19 22:55:57 +01:00
|
|
|
DBUG_ENTER("subselect_hash_sj_engine::make_semi_join_conds");
|
|
|
|
DBUG_ASSERT(semi_join_conds == NULL);
|
|
|
|
|
2015-08-20 14:24:13 +02:00
|
|
|
if (!(semi_join_conds= new (thd->mem_root) Item_cond_and(thd)))
|
2010-02-19 22:55:57 +01:00
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
|
2010-01-28 14:48:33 +01:00
|
|
|
if (!(tmp_table_ref= (TABLE_LIST*) thd->alloc(sizeof(TABLE_LIST))))
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
|
2018-01-07 17:03:44 +01:00
|
|
|
table_name.str= tmp_table->alias.c_ptr();
|
|
|
|
table_name.length= tmp_table->alias.length(),
|
|
|
|
tmp_table_ref->init_one_table(&empty_clex_str, &table_name, NULL, TL_READ);
|
2010-01-28 14:48:33 +01:00
|
|
|
tmp_table_ref->table= tmp_table;
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
context= new Name_resolution_context;
|
2010-01-28 14:48:33 +01:00
|
|
|
context->init();
|
|
|
|
context->first_name_resolution_table=
|
|
|
|
context->last_name_resolution_table= tmp_table_ref;
|
2011-05-28 18:34:04 +02:00
|
|
|
semi_join_conds_context= context;
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
for (uint i= 0; i < item_in->left_expr->cols(); i++)
|
2010-01-28 14:48:33 +01:00
|
|
|
{
|
2015-08-24 13:42:07 +02:00
|
|
|
/* New equi-join condition for the current column. */
|
|
|
|
Item_func_eq *eq_cond;
|
2010-01-28 14:48:33 +01:00
|
|
|
/* Item for the corresponding field from the materialized temp table. */
|
|
|
|
Item_field *right_col_item;
|
|
|
|
|
2015-08-24 13:42:07 +02:00
|
|
|
if (!(right_col_item= new (thd->mem_root)
|
2015-10-06 17:03:10 +02:00
|
|
|
Item_temptable_field(thd, context, tmp_table->field[i])) ||
|
2015-08-24 13:42:07 +02:00
|
|
|
!(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)))
|
2010-01-28 14:48:33 +01:00
|
|
|
{
|
2010-02-19 22:55:57 +01:00
|
|
|
delete semi_join_conds;
|
|
|
|
semi_join_conds= NULL;
|
2010-01-28 14:48:33 +01:00
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
}
|
2010-03-09 11:14:06 +01:00
|
|
|
if (semi_join_conds->fix_fields(thd, (Item**)&semi_join_conds))
|
2010-01-28 14:48:33 +01:00
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
|
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2010-02-19 22:55:57 +01:00
|
|
|
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;
|
2010-07-10 12:37:30 +02:00
|
|
|
Item_iterator_row it(item_in->left_expr);
|
2010-02-19 22:55:57 +01:00
|
|
|
/* 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);
|
|
|
|
|
2010-07-10 12:37:30 +02:00
|
|
|
tab->table= tmp_table;
|
2011-05-23 00:47:46 +02:00
|
|
|
tab->preread_init_done= FALSE;
|
2010-07-10 12:37:30 +02:00
|
|
|
tab->ref.tmp_table_index_lookup_init(thd, tmp_key, it, FALSE);
|
2010-02-19 22:55:57 +01:00
|
|
|
|
|
|
|
DBUG_RETURN(new subselect_uniquesubquery_engine(thd, tab, item,
|
|
|
|
semi_join_conds));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-16 12:52:02 +02:00
|
|
|
subselect_hash_sj_engine::~subselect_hash_sj_engine()
|
|
|
|
{
|
|
|
|
delete lookup_engine;
|
|
|
|
delete result;
|
|
|
|
if (tmp_table)
|
|
|
|
free_tmp_table(thd, tmp_table);
|
|
|
|
}
|
2010-01-28 14:48:33 +01:00
|
|
|
|
|
|
|
|
2016-06-22 14:17:06 +02:00
|
|
|
int subselect_hash_sj_engine::prepare(THD *thd_arg)
|
2010-01-28 14:48:33 +01:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
Create and optimize the JOIN that will be used to materialize
|
|
|
|
the subquery if not yet created.
|
|
|
|
*/
|
2016-06-22 14:17:06 +02:00
|
|
|
set_thd(thd_arg);
|
|
|
|
return materialize_engine->prepare(thd);
|
2010-01-28 14:48:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Cleanup performed after each PS execution.
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
@details
|
2010-01-28 14:48:33 +01:00
|
|
|
Called in the end of JOIN::prepare for PS from Item_subselect::cleanup.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void subselect_hash_sj_engine::cleanup()
|
|
|
|
{
|
2010-03-09 11:14:06 +01:00
|
|
|
enum_engine_type lookup_engine_type= lookup_engine->engine_type();
|
2010-01-28 14:48:33 +01:00
|
|
|
is_materialized= FALSE;
|
2010-03-09 11:14:06 +01:00
|
|
|
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;
|
2010-01-28 14:48:33 +01:00
|
|
|
materialize_engine->cleanup();
|
2010-07-16 12:52:02 +02:00
|
|
|
/*
|
|
|
|
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;
|
2010-03-09 11:14:06 +01:00
|
|
|
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. */
|
2010-07-16 12:52:02 +02:00
|
|
|
DBUG_ASSERT(tmp_table);
|
|
|
|
free_tmp_table(thd, tmp_table);
|
|
|
|
tmp_table= NULL;
|
2010-01-28 14:48:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-09 11:35:55 +02:00
|
|
|
/*
|
|
|
|
Get fanout produced by tables specified in the table_map
|
|
|
|
*/
|
|
|
|
|
|
|
|
double get_fanout_with_deps(JOIN *join, table_map tset)
|
|
|
|
{
|
2011-05-10 12:10:21 +02:00
|
|
|
/* Handle the case of "Impossible WHERE" */
|
|
|
|
if (join->table_count == 0)
|
|
|
|
return 0.0;
|
|
|
|
|
2011-05-09 11:35:55 +02:00
|
|
|
/* 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))
|
|
|
|
{
|
2011-09-28 10:58:01 +02:00
|
|
|
/*
|
|
|
|
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 &&
|
2011-05-09 21:11:36 +02:00
|
|
|
tab->records_read != 0)
|
|
|
|
{
|
2012-11-15 11:54:50 +01:00
|
|
|
fanout *= tab->records_read;
|
2011-05-09 21:11:36 +02:00
|
|
|
}
|
|
|
|
}
|
2011-05-09 11:35:55 +02:00
|
|
|
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
|
|
|
|
|
|
|
|
|
2011-05-10 12:31:02 +02:00
|
|
|
/*
|
|
|
|
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
|
|
|
|
*/
|
|
|
|
|
2011-05-09 21:11:36 +02:00
|
|
|
double get_post_group_estimate(JOIN* join, double join_op_rows)
|
2011-05-09 11:35:55 +02:00
|
|
|
{
|
|
|
|
table_map tables_in_group_list= table_map(0);
|
|
|
|
|
|
|
|
/* Find out which tables are used in GROUP BY list */
|
2016-05-08 22:04:41 +02:00
|
|
|
for (ORDER *order= join->group_list_for_estimates; order; order= order->next)
|
2011-05-09 11:35:55 +02:00
|
|
|
{
|
|
|
|
Item *item= order->item[0];
|
2015-05-13 16:17:22 +02:00
|
|
|
table_map item_used_tables= item->used_tables();
|
|
|
|
if (item_used_tables & RAND_TABLE_BIT)
|
2011-05-09 21:11:36 +02:00
|
|
|
{
|
|
|
|
/* Each join output record will be in its own group */
|
|
|
|
return join_op_rows;
|
|
|
|
}
|
2015-05-13 16:17:22 +02:00
|
|
|
tables_in_group_list|= item_used_tables;
|
2011-05-09 11:35:55 +02:00
|
|
|
}
|
|
|
|
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);
|
2011-05-10 12:31:02 +02:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* The following will be needed when making use of index stats: */
|
2011-05-09 11:35:55 +02:00
|
|
|
/*
|
|
|
|
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);
|
2011-05-10 12:31:02 +02:00
|
|
|
#endif
|
|
|
|
|
2011-05-09 11:35:55 +02:00
|
|
|
return out_rows;
|
2010-01-28 14:48:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Execute a subquery IN predicate via materialization.
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
@details
|
2010-01-28 14:48:33 +01:00
|
|
|
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;
|
2010-02-19 22:55:57 +01:00
|
|
|
SELECT_LEX *save_select= thd->lex->current_select;
|
2010-03-09 11:14:06 +01:00
|
|
|
subselect_partial_match_engine *pm_engine= NULL;
|
2010-02-19 22:55:57 +01:00
|
|
|
int res= 0;
|
2010-01-28 14:48:33 +01:00
|
|
|
|
|
|
|
DBUG_ENTER("subselect_hash_sj_engine::exec");
|
|
|
|
|
|
|
|
/*
|
|
|
|
Optimize and materialize the subquery during the first execution of
|
|
|
|
the subquery predicate.
|
|
|
|
*/
|
2010-02-19 22:55:57 +01:00
|
|
|
thd->lex->current_select= materialize_engine->select_lex;
|
2010-07-16 12:52:02 +02:00
|
|
|
/* The subquery should be optimized, and materialized only once. */
|
2016-06-22 11:17:44 +02:00
|
|
|
DBUG_ASSERT(materialize_join->optimization_state == JOIN::OPTIMIZATION_DONE &&
|
|
|
|
!is_materialized);
|
2010-02-19 22:55:57 +01:00
|
|
|
materialize_join->exec();
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely((res= MY_TEST(materialize_join->error || thd->is_fatal_error ||
|
|
|
|
thd->is_error()))))
|
2010-02-19 22:55:57 +01:00
|
|
|
goto err;
|
2010-01-28 14:48:33 +01:00
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
/*
|
|
|
|
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. */
|
2010-10-23 20:28:58 +02:00
|
|
|
item_in->reset();
|
|
|
|
item_in->make_const();
|
2010-02-19 22:55:57 +01:00
|
|
|
item_in->set_first_execution();
|
2016-02-18 17:20:48 +01:00
|
|
|
thd->lex->current_select= save_select;
|
2010-02-19 22:55:57 +01:00
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
/*
|
|
|
|
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();
|
2010-02-19 22:55:57 +01:00
|
|
|
if (strategy == PARTIAL_MATCH)
|
|
|
|
{
|
2010-03-09 11:14:06 +01:00
|
|
|
uint count_pm_keys; /* Total number of keys needed for partial matching. */
|
2011-07-13 23:15:07 +02:00
|
|
|
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;
|
2010-02-19 22:55:57 +01:00
|
|
|
select_materialize_with_stats *result_sink=
|
|
|
|
(select_materialize_with_stats *) result;
|
2011-07-13 23:15:07 +02:00
|
|
|
uint field_count= tmp_table->s->fields;
|
2010-02-19 22:55:57 +01:00
|
|
|
|
2011-07-13 23:15:07 +02:00
|
|
|
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);
|
2010-02-19 22:55:57 +01:00
|
|
|
|
2011-07-14 23:23:57 +02:00
|
|
|
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();
|
2016-02-18 17:20:48 +01:00
|
|
|
thd->lex->current_select= save_select;
|
2011-07-14 23:23:57 +02:00
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
2010-02-19 22:55:57 +01:00
|
|
|
|
2011-07-13 23:15:07 +02:00
|
|
|
if (has_covering_null_row)
|
|
|
|
{
|
2018-01-15 01:23:30 +01:00
|
|
|
DBUG_ASSERT(count_partial_match_columns == field_count);
|
2011-07-13 23:15:07 +02:00
|
|
|
count_pm_keys= 0;
|
|
|
|
}
|
|
|
|
else if (has_covering_null_columns)
|
|
|
|
count_pm_keys= 1;
|
2010-02-19 22:55:57 +01:00
|
|
|
else
|
|
|
|
count_pm_keys= count_partial_match_columns - count_null_only_columns +
|
2011-07-13 23:15:07 +02:00
|
|
|
(nn_key_parts ? 1 : 0);
|
2010-02-19 22:55:57 +01:00
|
|
|
|
2014-02-19 11:05:15 +01:00
|
|
|
choose_partial_match_strategy(MY_TEST(nn_key_parts),
|
2011-07-13 23:15:07 +02:00
|
|
|
has_covering_null_row,
|
2010-03-09 11:14:06 +01:00
|
|
|
&partial_match_key_parts);
|
|
|
|
DBUG_ASSERT(strategy == PARTIAL_MATCH_MERGE ||
|
|
|
|
strategy == PARTIAL_MATCH_SCAN);
|
|
|
|
if (strategy == PARTIAL_MATCH_MERGE)
|
2010-01-28 14:48:33 +01:00
|
|
|
{
|
2010-03-09 11:14:06 +01:00
|
|
|
pm_engine=
|
2016-06-22 14:17:06 +02:00
|
|
|
new subselect_rowid_merge_engine((subselect_uniquesubquery_engine*)
|
2010-03-09 11:14:06 +01:00
|
|
|
lookup_engine, tmp_table,
|
|
|
|
count_pm_keys,
|
2011-07-13 23:15:07 +02:00
|
|
|
has_covering_null_row,
|
|
|
|
has_covering_null_columns,
|
2011-08-22 23:00:13 +02:00
|
|
|
count_columns_with_nulls,
|
2010-03-09 11:14:06 +01:00
|
|
|
item, result,
|
|
|
|
semi_join_conds->argument_list());
|
|
|
|
if (!pm_engine ||
|
2016-06-22 14:17:06 +02:00
|
|
|
pm_engine->prepare(thd) ||
|
2010-03-09 11:14:06 +01:00
|
|
|
((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;
|
|
|
|
}
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
2010-03-09 11:14:06 +01:00
|
|
|
|
|
|
|
if (strategy == PARTIAL_MATCH_SCAN)
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
2010-03-09 11:14:06 +01:00
|
|
|
if (!(pm_engine=
|
2016-06-22 14:17:06 +02:00
|
|
|
new subselect_table_scan_engine((subselect_uniquesubquery_engine*)
|
2010-03-09 11:14:06 +01:00
|
|
|
lookup_engine, tmp_table,
|
|
|
|
item, result,
|
|
|
|
semi_join_conds->argument_list(),
|
2011-07-13 23:15:07 +02:00
|
|
|
has_covering_null_row,
|
2011-08-22 23:00:13 +02:00
|
|
|
has_covering_null_columns,
|
2016-06-22 14:17:06 +02:00
|
|
|
count_columns_with_nulls)) ||
|
|
|
|
pm_engine->prepare(thd))
|
2010-03-09 11:14:06 +01:00
|
|
|
{
|
|
|
|
/* This is an irrecoverable error. */
|
|
|
|
res= 1;
|
|
|
|
goto err;
|
|
|
|
}
|
2010-01-28 14:48:33 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
if (pm_engine)
|
|
|
|
lookup_engine= pm_engine;
|
2010-02-19 22:55:57 +01:00
|
|
|
item_in->change_engine(lookup_engine);
|
|
|
|
|
|
|
|
err:
|
|
|
|
thd->lex->current_select= save_select;
|
|
|
|
DBUG_RETURN(res);
|
2010-01-28 14:48:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
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(" ), "));
|
2010-02-19 22:55:57 +01:00
|
|
|
|
|
|
|
if (lookup_engine)
|
|
|
|
lookup_engine->print(str, query_type);
|
2010-01-28 14:48:33 +01:00
|
|
|
else
|
|
|
|
str->append(STRING_WITH_LEN(
|
2010-02-22 16:16:55 +01:00
|
|
|
"<engine selected at execution time>"
|
2010-01-28 14:48:33 +01:00
|
|
|
));
|
|
|
|
}
|
2010-02-19 22:55:57 +01:00
|
|
|
|
2018-05-08 15:26:26 +02:00
|
|
|
bool subselect_hash_sj_engine::fix_length_and_dec(Item_cache** row)
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
|
|
|
DBUG_ASSERT(FALSE);
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
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,
|
2011-05-04 17:08:58 +02:00
|
|
|
select_result_interceptor *res,
|
|
|
|
bool temp __attribute__((unused)))
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
|
|
|
DBUG_ASSERT(FALSE);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-02-22 16:16:55 +01:00
|
|
|
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),
|
2010-02-19 22:55:57 +01:00
|
|
|
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()
|
|
|
|
{
|
2011-04-25 17:22:25 +02:00
|
|
|
my_free(key_buff);
|
2014-01-02 10:19:19 +01:00
|
|
|
my_bitmap_free(&null_key);
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
/*
|
|
|
|
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*));
|
|
|
|
|
2011-11-17 00:25:10 +01:00
|
|
|
if (!key_columns || !compare_pred)
|
|
|
|
return TRUE; /* Revert to table scan partial match. */
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
for (uint i= 0; i < columns_to_index->n_bits; i++)
|
|
|
|
{
|
|
|
|
if (!bitmap_is_set(columns_to_index, i))
|
|
|
|
continue;
|
2015-08-20 14:24:13 +02:00
|
|
|
cur_tmp_field= new (thd->mem_root) Item_field(thd, tbl->field[i]);
|
2010-02-19 22:55:57 +01:00
|
|
|
/* Create the predicate (tmp_column[i] < outer_ref[i]). */
|
2015-08-20 14:24:13 +02:00
|
|
|
fn_less_than= new (thd->mem_root) Item_func_lt(thd, cur_tmp_field,
|
2010-02-19 22:55:57 +01:00
|
|
|
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*));
|
|
|
|
|
2015-08-20 14:24:13 +02:00
|
|
|
key_columns[0]= new (thd->mem_root) Item_field(thd, tbl->field[col_idx]);
|
2010-02-19 22:55:57 +01:00
|
|
|
/* Create the predicate (tmp_column[i] < outer_ref[i]). */
|
2015-08-20 14:24:13 +02:00
|
|
|
compare_pred[0]= new (thd->mem_root) Item_func_lt(thd, key_columns[0],
|
2010-02-19 22:55:57 +01:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
/*
|
|
|
|
Allocate the buffers for both the row number, and the NULL-bitmap indexes.
|
|
|
|
*/
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
bool Ordered_key::alloc_keys_buffers()
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(key_buff_elements > 0);
|
|
|
|
|
2011-02-18 23:31:01 +01:00
|
|
|
if (!(key_buff= (rownum_t*) my_malloc((size_t)(key_buff_elements *
|
MDEV-4011 Added per thread memory counting and usage
Base code and idea from a patch from by plinux at Taobao.
The idea is that we mark all memory that are thread specific with MY_THREAD_SPECIFIC.
Memory counting is done per thread in the my_malloc_size_cb_func callback function from my_malloc().
There are plenty of new asserts to ensure that for a debug server the counting is correct.
Information_schema.processlist gets two new columns: MEMORY_USED and EXAMINED_ROWS.
- The later is there mainly to show how query is progressing.
The following changes in interfaces was needed to get this to work:
- init_alloc_root() amd init_sql_alloc() has extra option so that one can mark memory with MY_THREAD_SPECIFIC
- One now have to use alloc_root_set_min_malloc() to set min memory to be allocated by alloc_root()
- my_init_dynamic_array() has extra option so that one can mark memory with MY_THREAD_SPECIFIC
- my_net_init() has extra option so that one can mark memory with MY_THREAD_SPECIFIC
- Added flag for hash_init() so that one can mark hash table to be thread specific.
- Added flags to init_tree() so that one can mark tree to be thread specific.
- Removed with_delete option to init_tree(). Now one should instead use MY_TREE_WITH_DELETE_FLAG.
- Added flag to Warning_info::Warning_info() if the structure should be fully initialized.
- String elements can now be marked as thread specific.
- Internal HEAP tables are now marking it's memory as MY_THREAD_SPECIFIC.
- Changed type of myf from int to ulong, as this is always a set of bit flags.
Other things:
- Removed calls to net_end() and thd->cleanup() as these are now done in ~THD()
- We now also show EXAMINED_ROWS in SHOW PROCESSLIST
- Added new variable 'memory_used'
- Fixed bug where kill_threads_for_user() was using the wrong mem_root to allocate memory.
- Removed calls to the obsoleted function init_dynamic_array()
- Use set_current_thd() instead of my_pthread_setspecific_ptr(THR_THD,...)
client/completion_hash.cc:
Updated call to init_alloc_root()
client/mysql.cc:
Updated call to init_alloc_root()
client/mysqlbinlog.cc:
init_dynamic_array() -> my_init_dynamic_array()
Updated call to init_alloc_root()
client/mysqlcheck.c:
Updated call to my_init_dynamic_array()
client/mysqldump.c:
Updated call to init_alloc_root()
client/mysqltest.cc:
Updated call to init_alloc_root()
Updated call to my_init_dynamic_array()
Fixed compiler warnings
extra/comp_err.c:
Updated call to my_init_dynamic_array()
extra/resolve_stack_dump.c:
Updated call to my_init_dynamic_array()
include/hash.h:
Added HASH_THREAD_SPECIFIC
include/heap.h:
Added flag is internal temporary table.
include/my_dir.h:
Safety fix: Ensure that MY_DONT_SORT and MY_WANT_STAT don't interfer with other mysys flags
include/my_global.h:
Changed type of myf from int to ulong, as this is always a set of bit flags.
include/my_sys.h:
Added MY_THREAD_SPECIFIC and MY_THREAD_MOVE
Added malloc_flags to DYNAMIC_ARRAY
Added extra mysys flag argument to my_init_dynamic_array()
Removed deprecated functions init_dynamic_array() and my_init_dynamic_array.._ci
Updated paramaters for init_alloc_root()
include/my_tree.h:
Added my_flags to allow one to use MY_THREAD_SPECIFIC with hash tables.
Removed with_delete. One should now instead use MY_TREE_WITH_DELETE_FLAG
Updated parameters to init_tree()
include/myisamchk.h:
Added malloc_flags to allow one to use MY_THREAD_SPECIFIC for checks.
include/mysql.h:
Added MYSQL_THREAD_SPECIFIC_MALLOC
Used 'unused1' to mark memory as thread specific.
include/mysql.h.pp:
Updated file
include/mysql_com.h:
Used 'unused1' to mark memory as thread specific.
Updated parameters for my_net_init()
libmysql/libmysql.c:
Updated call to init_alloc_root() to mark memory thread specific.
libmysqld/emb_qcache.cc:
Updated call to init_alloc_root()
libmysqld/lib_sql.cc:
Updated call to init_alloc_root()
mysql-test/r/create.result:
Updated results
mysql-test/r/user_var.result:
Updated results
mysql-test/suite/funcs_1/datadict/processlist_priv.inc:
Update to handle new format of SHOW PROCESSLIST
mysql-test/suite/funcs_1/datadict/processlist_val.inc:
Update to handle new format of SHOW PROCESSLIST
mysql-test/suite/funcs_1/r/is_columns_is.result:
Update to handle new format of SHOW PROCESSLIST
mysql-test/suite/funcs_1/r/processlist_priv_no_prot.result:
Updated results
mysql-test/suite/funcs_1/r/processlist_val_no_prot.result:
Updated results
mysql-test/t/show_explain.test:
Fixed usage of debug variable so that one can run test with --debug
mysql-test/t/user_var.test:
Added test of memory_usage variable.
mysys/array.c:
Added extra my_flags option to init_dynamic_array() and init_dynamic_array2() so that one can mark memory with MY_THREAD_SPECIFIC
All allocated memory is marked with the given my_flags.
Removed obsolete function init_dynamic_array()
mysys/default.c:
Updated call to init_alloc_root()
Updated call to my_init_dynamic_array()
mysys/hash.c:
Updated call to my_init_dynamic_array_ci().
Allocated memory is marked with MY_THREAD_SPECIFIC if HASH_THREAD_SPECIFIC is used.
mysys/ma_dyncol.c:
init_dynamic_array() -> my_init_dynamic_array()
Added #if to get rid of compiler warnings
mysys/mf_tempdir.c:
Updated call to my_init_dynamic_array()
mysys/my_alloc.c:
Added extra parameter to init_alloc_root() so that one can mark memory with MY_THREAD_SPECIFIC
Extend MEM_ROOT with a flag if memory is thread specific.
This is stored in block_size, to keep the size of the MEM_ROOT object identical as before.
Allocated memory is marked with MY_THREAD_SPECIFIC if used with init_alloc_root()
mysys/my_chmod.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_chsize.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_copy.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_create.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_delete.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_error.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_fopen.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_fstream.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_getwd.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_lib.c:
Updated call to init_alloc_root()
Updated call to my_init_dynamic_array()
Updated DBUG_PRINT because of change of myf type
mysys/my_lock.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_malloc.c:
Store at start of each allocated memory block the size of the block and if the block is thread specific.
Call malloc_size_cb_func, if set, with the memory allocated/freed.
Updated DBUG_PRINT because of change of myf type
mysys/my_open.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_pread.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_read.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_redel.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_rename.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_seek.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_sync.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_thr_init.c:
Ensure that one can call my_thread_dbug_id() even if thread is not properly initialized.
mysys/my_write.c:
Updated DBUG_PRINT because of change of myf type
mysys/mysys_priv.h:
Updated parameters to sf_malloc and sf_realloc()
mysys/safemalloc.c:
Added checking that for memory marked with MY_THREAD_SPECIFIC that it's the same thread that is allocation and freeing the memory.
Added sf_malloc_dbug_id() to allow MariaDB to specify which THD is handling the memory.
Added my_flags arguments to sf_malloc() and sf_realloc() to be able to mark memory with MY_THREAD_SPECIFIC.
Added sf_report_leaked_memory() to get list of memory not freed by a thread.
mysys/tree.c:
Added flags to init_tree() so that one can mark tree to be thread specific.
Removed with_delete option to init_tree(). Now one should instead use MY_TREE_WITH_DELETE_FLAG.
Updated call to init_alloc_root()
All allocated memory is marked with the given malloc flags
mysys/waiting_threads.c:
Updated call to my_init_dynamic_array()
sql-common/client.c:
Updated call to init_alloc_root() and my_net_init() to mark memory thread specific.
Updated call to my_init_dynamic_array().
Added MYSQL_THREAD_SPECIFIC_MALLOC so that client can mark memory as MY_THREAD_SPECIFIC.
sql-common/client_plugin.c:
Updated call to init_alloc_root()
sql/debug_sync.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/event_scheduler.cc:
Removed calls to net_end() as this is now done in ~THD()
Call set_current_thd() to ensure that memory is assigned to right thread.
sql/events.cc:
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/filesort.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/filesort_utils.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/ha_ndbcluster.cc:
Updated call to init_alloc_root()
Updated call to my_net_init()
Removed calls to net_end() and thd->cleanup() as these are now done in ~THD()
sql/ha_ndbcluster_binlog.cc:
Updated call to my_net_init()
Updated call to init_sql_alloc()
Removed calls to net_end() and thd->cleanup() as these are now done in ~THD()
sql/ha_partition.cc:
Updated call to init_alloc_root()
sql/handler.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
Added missing call to my_dir_end()
sql/item_func.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/item_subselect.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/item_sum.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/log.cc:
More DBUG
Updated call to init_alloc_root()
sql/mdl.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/mysqld.cc:
Added total_memory_used
Updated call to init_alloc_root()
Move mysql_cond_broadcast() before my_thread_end()
Added mariadb_dbug_id() to count memory per THD instead of per thread.
Added my_malloc_size_cb_func() callback function for my_malloc() to count memory.
Move initialization of mysqld_server_started and mysqld_server_initialized earlier.
Updated call to my_init_dynamic_array().
Updated call to my_net_init().
Call my_pthread_setspecific_ptr(THR_THD,...) to ensure that memory is assigned to right thread.
Added status variable 'memory_used'.
Updated call to init_alloc_root()
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/mysqld.h:
Added set_current_thd()
sql/net_serv.cc:
Added new parameter to my_net_init() so that one can mark memory with MY_THREAD_SPECIFIC.
Store in net->thread_specific_malloc if memory is thread specific.
Mark memory to be thread specific if requested.
sql/opt_range.cc:
Updated call to my_init_dynamic_array()
Updated call to init_sql_alloc()
Added MY_THREAD_SPECIFIC to allocated memory.
sql/opt_subselect.cc:
Updated call to init_sql_alloc() to mark memory thread specific.
sql/protocol.cc:
Fixed compiler warning
sql/records.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/rpl_filter.cc:
Updated call to my_init_dynamic_array()
sql/rpl_handler.cc:
Updated call to my_init_dynamic_array2()
sql/rpl_handler.h:
Updated call to init_sql_alloc()
sql/rpl_mi.cc:
Updated call to my_init_dynamic_array()
sql/rpl_tblmap.cc:
Updated call to init_alloc_root()
sql/rpl_utility.cc:
Updated call to my_init_dynamic_array()
sql/slave.cc:
Initialize things properly before calling functions that allocate memory.
Removed calls to net_end() as this is now done in ~THD()
sql/sp_head.cc:
Updated call to init_sql_alloc()
Updated call to my_init_dynamic_array()
Added parameter to warning_info() that it should be fully initialized.
sql/sp_pcontext.cc:
Updated call to my_init_dynamic_array()
sql/sql_acl.cc:
Updated call to init_sql_alloc()
Updated call to my_init_dynamic_array()
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/sql_admin.cc:
Added parameter to warning_info() that it should be fully initialized.
sql/sql_analyse.h:
Updated call to init_tree() to mark memory thread specific.
sql/sql_array.h:
Updated call to my_init_dynamic_array() to mark memory thread specific.
sql/sql_audit.cc:
Updated call to my_init_dynamic_array()
sql/sql_base.cc:
Updated call to init_sql_alloc()
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/sql_cache.cc:
Updated comment
sql/sql_class.cc:
Added parameter to warning_info() that not initialize it until THD is fully created.
Updated call to init_sql_alloc()
Mark THD::user_vars has to be thread specific.
Updated call to my_init_dynamic_array()
Ensure that memory allocated by THD is assigned to the THD.
More DBUG
Always acll net_end() in ~THD()
Assert that all memory signed to this THD is really deleted at ~THD.
Fixed set_status_var_init() to not reset memory_used.
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/sql_class.h:
Added MY_THREAD_SPECIFIC to allocated memory.
Added malloc_size to THD to record allocated memory per THD.
sql/sql_delete.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/sql_error.cc:
Added 'initialize' parameter to Warning_info() to say if should allocate memory for it's structures.
This is used by THD::THD() to not allocate memory until THD is ready.
Added Warning_info::free_memory()
sql/sql_error.h:
Updated Warning_info() class.
sql/sql_handler.cc:
Updated call to init_alloc_root() to mark memory thread specific.
sql/sql_insert.cc:
More DBUG
sql/sql_join_cache.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/sql_lex.cc:
Updated call to my_init_dynamic_array()
sql/sql_lex.h:
Updated call to my_init_dynamic_array()
sql/sql_load.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/sql_parse.cc:
Removed calls to net_end() and thd->cleanup() as these are now done in ~THD()
Ensure that examined_row_count() is reset before query.
Fixed bug where kill_threads_for_user() was using the wrong mem_root to allocate memory.
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
Don't restore thd->status_var.memory_used when restoring thd->status_var
sql/sql_plugin.cc:
Updated call to init_alloc_root()
Updated call to my_init_dynamic_array()
Don't allocate THD on the stack, as this causes problems with valgrind when doing thd memory counting.
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/sql_prepare.cc:
Added parameter to warning_info() that it should be fully initialized.
Updated call to init_sql_alloc() to mark memory thread specific.
sql/sql_reload.cc:
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/sql_select.cc:
Updated call to my_init_dynamic_array() and init_sql_alloc() to mark memory thread specific.
Added MY_THREAD_SPECIFIC to allocated memory.
More DBUG
sql/sql_servers.cc:
Updated call to init_sql_alloc() to mark memory some memory thread specific.
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/sql_show.cc:
Updated call to my_init_dynamic_array()
Mark my_dir() memory thread specific.
Use my_pthread_setspecific_ptr(THR_THD,...) to mark that allocated memory should be allocated to calling thread.
More DBUG.
Added malloc_size and examined_row_count to SHOW PROCESSLIST.
Added MY_THREAD_SPECIFIC to allocated memory.
Updated call to init_sql_alloc()
Added parameter to warning_info() that it should be fully initialized.
sql/sql_statistics.cc:
Fixed compiler warning
sql/sql_string.cc:
String elements can now be marked as thread specific.
sql/sql_string.h:
String elements can now be marked as thread specific.
sql/sql_table.cc:
Updated call to init_sql_alloc() and my_malloc() to mark memory thread specific
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
Fixed compiler warning
sql/sql_test.cc:
Updated call to my_init_dynamic_array() to mark memory thread specific.
sql/sql_trigger.cc:
Updated call to init_sql_alloc()
sql/sql_udf.cc:
Updated call to init_sql_alloc()
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/sql_update.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/table.cc:
Updated call to init_sql_alloc().
Mark memory used by temporary tables, that are not for slave threads, as MY_THREAD_SPECIFIC
Updated call to init_sql_alloc()
sql/thr_malloc.cc:
Added my_flags argument to init_sql_alloc() to be able to mark memory as MY_THREAD_SPECIFIC.
sql/thr_malloc.h:
Updated prototype for init_sql_alloc()
sql/tztime.cc:
Updated call to init_sql_alloc()
Updated call to init_alloc_root() to mark memory thread specific.
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/uniques.cc:
Updated calls to init_tree(), my_init_dynamic_array() and my_malloc() to mark memory thread specific.
sql/unireg.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
storage/csv/ha_tina.cc:
Updated call to init_alloc_root()
storage/federated/ha_federated.cc:
Updated call to init_alloc_root()
Updated call to my_init_dynamic_array()
Ensure that memory allocated by fedarated is registered for the system, not for the thread.
storage/federatedx/federatedx_io_mysql.cc:
Updated call to my_init_dynamic_array()
storage/federatedx/ha_federatedx.cc:
Updated call to init_alloc_root()
Updated call to my_init_dynamic_array()
storage/heap/ha_heap.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
storage/heap/heapdef.h:
Added parameter to hp_get_new_block() to be able to do thread specific memory tagging.
storage/heap/hp_block.c:
Added parameter to hp_get_new_block() to be able to do thread specific memory tagging.
storage/heap/hp_create.c:
- Internal HEAP tables are now marking it's memory as MY_THREAD_SPECIFIC.
- Use MY_TREE_WITH_DELETE instead of removed option 'with_delete'.
storage/heap/hp_open.c:
Internal HEAP tables are now marking it's memory as MY_THREAD_SPECIFIC.
storage/heap/hp_write.c:
Added new parameter to hp_get_new_block()
storage/maria/ma_bitmap.c:
Updated call to my_init_dynamic_array()
storage/maria/ma_blockrec.c:
Updated call to my_init_dynamic_array()
storage/maria/ma_check.c:
Updated call to init_alloc_root()
storage/maria/ma_ft_boolean_search.c:
Updated calls to init_tree() and init_alloc_root()
storage/maria/ma_ft_nlq_search.c:
Updated call to init_tree()
storage/maria/ma_ft_parser.c:
Updated call to init_tree()
Updated call to init_alloc_root()
storage/maria/ma_loghandler.c:
Updated call to my_init_dynamic_array()
storage/maria/ma_open.c:
Updated call to my_init_dynamic_array()
storage/maria/ma_sort.c:
Updated call to my_init_dynamic_array()
storage/maria/ma_write.c:
Updated calls to my_init_dynamic_array() and init_tree()
storage/maria/maria_pack.c:
Updated call to init_tree()
storage/maria/unittest/sequence_storage.c:
Updated call to my_init_dynamic_array()
storage/myisam/ft_boolean_search.c:
Updated call to init_tree()
Updated call to init_alloc_root()
storage/myisam/ft_nlq_search.c:
Updated call to init_tree()
storage/myisam/ft_parser.c:
Updated call to init_tree()
Updated call to init_alloc_root()
storage/myisam/ft_stopwords.c:
Updated call to init_tree()
storage/myisam/mi_check.c:
Updated call to init_alloc_root()
storage/myisam/mi_write.c:
Updated call to my_init_dynamic_array()
Updated call to init_tree()
storage/myisam/myisamlog.c:
Updated call to init_tree()
storage/myisam/myisampack.c:
Updated call to init_tree()
storage/myisam/sort.c:
Updated call to my_init_dynamic_array()
storage/myisammrg/ha_myisammrg.cc:
Updated call to init_sql_alloc()
storage/perfschema/pfs_check.cc:
Rest current_thd
storage/perfschema/pfs_instr.cc:
Removed DBUG_ENTER/DBUG_VOID_RETURN as at this point my_thread_var is not allocated anymore, which can cause problems.
support-files/compiler_warnings.supp:
Disable compiler warning from offsetof macro.
2013-01-23 16:16:14 +01:00
|
|
|
sizeof(rownum_t)), MYF(MY_WME | MY_THREAD_SPECIFIC))))
|
2010-02-19 22:55:57 +01:00
|
|
|
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.
|
|
|
|
*/
|
2010-03-09 11:14:06 +01:00
|
|
|
/* Notice that max_null_row is max array index, we need count, so +1. */
|
2014-01-02 10:19:19 +01:00
|
|
|
if (my_bitmap_init(&null_key, NULL, (uint)(max_null_row + 1), FALSE))
|
2010-02-19 22:55:57 +01:00
|
|
|
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;
|
2012-06-05 13:09:18 +02:00
|
|
|
int __attribute__((unused)) error;
|
|
|
|
int cmp_res;
|
2010-02-19 22:55:57 +01:00
|
|
|
/* 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. */
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely((error= tbl->file->ha_rnd_pos(tbl->record[0], rowid_a))))
|
2012-08-14 18:59:28 +02:00
|
|
|
{
|
2012-08-15 08:34:18 +02:00
|
|
|
/* purecov: begin inspected */
|
2012-08-14 18:59:28 +02:00
|
|
|
tbl->file->print_error(error, MYF(ME_FATALERROR)); // Sets fatal_error
|
|
|
|
return 0;
|
2012-08-15 08:34:18 +02:00
|
|
|
/* purecov: end */
|
2012-08-14 18:59:28 +02:00
|
|
|
}
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely((error= tbl->file->ha_rnd_pos(tbl->record[1], rowid_b))))
|
2012-08-14 18:59:28 +02:00
|
|
|
{
|
2012-08-15 08:34:18 +02:00
|
|
|
/* purecov: begin inspected */
|
2012-08-14 18:59:28 +02:00
|
|
|
tbl->file->print_error(error, MYF(ME_FATALERROR)); // Sets fatal_error
|
|
|
|
return 0;
|
2012-08-15 08:34:18 +02:00
|
|
|
/* purecov: end */
|
2012-08-14 18:59:28 +02:00
|
|
|
}
|
2010-02-19 22:55:57 +01:00
|
|
|
/*
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-12-20 09:52:34 +01:00
|
|
|
bool Ordered_key::sort_keys()
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
2018-12-20 09:52:34 +01:00
|
|
|
if (tbl->file->ha_rnd_init_with_error(0))
|
|
|
|
return TRUE;
|
2011-02-18 23:31:01 +01:00
|
|
|
my_qsort2(key_buff, (size_t) key_buff_elements, sizeof(rownum_t),
|
2010-02-19 22:55:57 +01:00
|
|
|
(qsort2_cmp) &cmp_keys_by_row_data_and_rownum, (void*) this);
|
|
|
|
/* Invalidate the current row position. */
|
|
|
|
cur_key_idx= HA_POS_ERROR;
|
2018-12-20 09:52:34 +01:00
|
|
|
tbl->file->ha_rnd_end();
|
|
|
|
return FALSE;
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-02-22 16:16:55 +01:00
|
|
|
/*
|
2010-03-09 11:14:06 +01:00
|
|
|
The fraction of rows that do not contain NULL in the columns indexed by
|
|
|
|
this key.
|
|
|
|
|
2010-02-22 16:16:55 +01:00
|
|
|
@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);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
/*
|
|
|
|
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;
|
2012-06-05 13:09:18 +02:00
|
|
|
int __attribute__((unused)) error;
|
|
|
|
int cmp_res;
|
2010-02-19 22:55:57 +01:00
|
|
|
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely((error= tbl->file->ha_rnd_pos(tbl->record[0], cur_rowid))))
|
2012-08-14 18:59:28 +02:00
|
|
|
{
|
2012-08-15 08:34:18 +02:00
|
|
|
/* purecov: begin inspected */
|
2012-08-14 18:59:28 +02:00
|
|
|
tbl->file->print_error(error, MYF(ME_FATALERROR)); // Sets fatal_error
|
|
|
|
return 0;
|
2012-08-15 08:34:18 +02:00
|
|
|
/* purecov: end */
|
2012-08-14 18:59:28 +02:00
|
|
|
}
|
2010-02-19 22:55:57 +01:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-02-22 16:16:55 +01:00
|
|
|
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++)
|
|
|
|
{
|
2017-04-23 18:39:57 +02:00
|
|
|
str->append(&key_columns[i]->field->field_name);
|
2010-02-22 16:16:55 +01:00
|
|
|
str->append(", ");
|
|
|
|
}
|
2017-04-23 18:39:57 +02:00
|
|
|
str->append(&key_columns[i]->field->field_name);
|
2010-02-22 16:16:55 +01:00
|
|
|
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('}');
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
subselect_partial_match_engine::subselect_partial_match_engine(
|
2016-06-22 14:17:06 +02:00
|
|
|
subselect_uniquesubquery_engine *engine_arg,
|
2010-03-09 11:14:06 +01:00
|
|
|
TABLE *tmp_table_arg, Item_subselect *item_arg,
|
|
|
|
select_result_interceptor *result_arg,
|
|
|
|
List<Item> *equi_join_conds_arg,
|
2011-07-13 23:15:07 +02:00
|
|
|
bool has_covering_null_row_arg,
|
2011-08-22 23:00:13 +02:00
|
|
|
bool has_covering_null_columns_arg,
|
|
|
|
uint count_columns_with_nulls_arg)
|
2016-06-22 14:17:06 +02:00
|
|
|
:subselect_engine(item_arg, result_arg),
|
2010-03-09 11:14:06 +01:00
|
|
|
tmp_table(tmp_table_arg), lookup_engine(engine_arg),
|
|
|
|
equi_join_conds(equi_join_conds_arg),
|
2011-07-13 23:15:07 +02:00
|
|
|
has_covering_null_row(has_covering_null_row_arg),
|
2011-08-22 23:00:13 +02:00
|
|
|
has_covering_null_columns(has_covering_null_columns_arg),
|
|
|
|
count_columns_with_nulls(count_columns_with_nulls_arg)
|
2010-03-09 11:14:06 +01:00
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
|
|
int subselect_partial_match_engine::exec()
|
|
|
|
{
|
|
|
|
Item_in_subselect *item_in= (Item_in_subselect *) item;
|
2012-09-17 10:13:46 +02:00
|
|
|
int lookup_res;
|
2010-03-09 11:14:06 +01:00
|
|
|
|
2012-09-17 10:13:46 +02:00
|
|
|
DBUG_ASSERT(!(item_in->left_expr_has_null() &&
|
|
|
|
item_in->is_top_level_item()));
|
|
|
|
|
|
|
|
if (!item_in->left_expr_has_null())
|
2010-03-09 11:14:06 +01:00
|
|
|
{
|
2012-09-17 10:13:46 +02:00
|
|
|
/* Try to find a matching row by index lookup. */
|
|
|
|
if (lookup_engine->copy_ref_key(false))
|
2010-03-09 11:14:06 +01:00
|
|
|
{
|
2012-09-17 10:13:46 +02:00
|
|
|
/* The result is FALSE based on the outer reference. */
|
2010-03-09 11:14:06 +01:00
|
|
|
item_in->value= 0;
|
|
|
|
item_in->null_value= 0;
|
2012-09-17 10:13:46 +02:00
|
|
|
return 0;
|
2010-03-09 11:14:06 +01:00
|
|
|
}
|
2012-09-17 10:13:46 +02:00
|
|
|
else
|
2010-03-09 11:14:06 +01:00
|
|
|
{
|
2012-09-17 10:13:46 +02:00
|
|
|
/* Search for a complete match. */
|
|
|
|
if ((lookup_res= lookup_engine->index_lookup()))
|
|
|
|
{
|
2016-03-04 01:09:37 +01:00
|
|
|
/* An error occurred during lookup(). */
|
2012-09-17 10:13:46 +02:00
|
|
|
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;
|
|
|
|
}
|
2010-03-09 11:14:06 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-13 23:15:07 +02:00
|
|
|
if (has_covering_null_row)
|
2010-03-09 11:14:06 +01:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
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.
|
2010-10-26 13:55:42 +02:00
|
|
|
DBUG_ASSERT(FALSE);
|
2010-03-09 11:14:06 +01:00
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
/*
|
|
|
|
@param non_null_key_parts
|
|
|
|
@param partial_match_key_parts A union of all single-column NULL key parts.
|
2010-03-09 11:14:06 +01:00
|
|
|
|
|
|
|
@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
|
2010-02-19 22:55:57 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
bool
|
|
|
|
subselect_rowid_merge_engine::init(MY_BITMAP *non_null_key_parts,
|
|
|
|
MY_BITMAP *partial_match_key_parts)
|
|
|
|
{
|
2016-07-29 18:21:08 +02:00
|
|
|
THD *thd= get_thd();
|
2010-02-19 22:55:57 +01:00
|
|
|
/* 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;
|
2010-02-22 16:16:55 +01:00
|
|
|
uint cur_keyid= 0;
|
2010-02-19 22:55:57 +01:00
|
|
|
Item_in_subselect *item_in= (Item_in_subselect*) item;
|
|
|
|
int error;
|
|
|
|
|
2011-07-13 16:09:09 +02:00
|
|
|
if (merge_keys_count == 0)
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
2011-07-13 23:15:07 +02:00
|
|
|
DBUG_ASSERT(bitmap_bits_set(partial_match_key_parts) == 0 ||
|
|
|
|
has_covering_null_row);
|
2010-02-19 22:55:57 +01:00
|
|
|
/* There is nothing to initialize, we will only do regular lookups. */
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2011-07-13 23:15:07 +02:00
|
|
|
/*
|
|
|
|
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));
|
2010-03-09 11:14:06 +01:00
|
|
|
/*
|
|
|
|
Allocate buffers to hold the merged keys and the mapping between rowids and
|
2011-11-17 00:25:10 +01:00
|
|
|
row numbers. All small buffers are allocated in the runtime memroot. Big
|
|
|
|
buffers are allocated from the OS via malloc.
|
2010-03-09 11:14:06 +01:00
|
|
|
*/
|
2011-07-13 16:09:09 +02:00
|
|
|
if (!(merge_keys= (Ordered_key**) thd->alloc(merge_keys_count *
|
2010-02-19 22:55:57 +01:00
|
|
|
sizeof(Ordered_key*))) ||
|
2011-11-17 00:25:10 +01:00
|
|
|
!(null_bitmaps= (MY_BITMAP**) thd->alloc(merge_keys_count *
|
|
|
|
sizeof(MY_BITMAP*))) ||
|
2011-02-18 23:31:01 +01:00
|
|
|
!(row_num_to_rowid= (uchar*) my_malloc((size_t)(row_count * rowid_length),
|
MDEV-4011 Added per thread memory counting and usage
Base code and idea from a patch from by plinux at Taobao.
The idea is that we mark all memory that are thread specific with MY_THREAD_SPECIFIC.
Memory counting is done per thread in the my_malloc_size_cb_func callback function from my_malloc().
There are plenty of new asserts to ensure that for a debug server the counting is correct.
Information_schema.processlist gets two new columns: MEMORY_USED and EXAMINED_ROWS.
- The later is there mainly to show how query is progressing.
The following changes in interfaces was needed to get this to work:
- init_alloc_root() amd init_sql_alloc() has extra option so that one can mark memory with MY_THREAD_SPECIFIC
- One now have to use alloc_root_set_min_malloc() to set min memory to be allocated by alloc_root()
- my_init_dynamic_array() has extra option so that one can mark memory with MY_THREAD_SPECIFIC
- my_net_init() has extra option so that one can mark memory with MY_THREAD_SPECIFIC
- Added flag for hash_init() so that one can mark hash table to be thread specific.
- Added flags to init_tree() so that one can mark tree to be thread specific.
- Removed with_delete option to init_tree(). Now one should instead use MY_TREE_WITH_DELETE_FLAG.
- Added flag to Warning_info::Warning_info() if the structure should be fully initialized.
- String elements can now be marked as thread specific.
- Internal HEAP tables are now marking it's memory as MY_THREAD_SPECIFIC.
- Changed type of myf from int to ulong, as this is always a set of bit flags.
Other things:
- Removed calls to net_end() and thd->cleanup() as these are now done in ~THD()
- We now also show EXAMINED_ROWS in SHOW PROCESSLIST
- Added new variable 'memory_used'
- Fixed bug where kill_threads_for_user() was using the wrong mem_root to allocate memory.
- Removed calls to the obsoleted function init_dynamic_array()
- Use set_current_thd() instead of my_pthread_setspecific_ptr(THR_THD,...)
client/completion_hash.cc:
Updated call to init_alloc_root()
client/mysql.cc:
Updated call to init_alloc_root()
client/mysqlbinlog.cc:
init_dynamic_array() -> my_init_dynamic_array()
Updated call to init_alloc_root()
client/mysqlcheck.c:
Updated call to my_init_dynamic_array()
client/mysqldump.c:
Updated call to init_alloc_root()
client/mysqltest.cc:
Updated call to init_alloc_root()
Updated call to my_init_dynamic_array()
Fixed compiler warnings
extra/comp_err.c:
Updated call to my_init_dynamic_array()
extra/resolve_stack_dump.c:
Updated call to my_init_dynamic_array()
include/hash.h:
Added HASH_THREAD_SPECIFIC
include/heap.h:
Added flag is internal temporary table.
include/my_dir.h:
Safety fix: Ensure that MY_DONT_SORT and MY_WANT_STAT don't interfer with other mysys flags
include/my_global.h:
Changed type of myf from int to ulong, as this is always a set of bit flags.
include/my_sys.h:
Added MY_THREAD_SPECIFIC and MY_THREAD_MOVE
Added malloc_flags to DYNAMIC_ARRAY
Added extra mysys flag argument to my_init_dynamic_array()
Removed deprecated functions init_dynamic_array() and my_init_dynamic_array.._ci
Updated paramaters for init_alloc_root()
include/my_tree.h:
Added my_flags to allow one to use MY_THREAD_SPECIFIC with hash tables.
Removed with_delete. One should now instead use MY_TREE_WITH_DELETE_FLAG
Updated parameters to init_tree()
include/myisamchk.h:
Added malloc_flags to allow one to use MY_THREAD_SPECIFIC for checks.
include/mysql.h:
Added MYSQL_THREAD_SPECIFIC_MALLOC
Used 'unused1' to mark memory as thread specific.
include/mysql.h.pp:
Updated file
include/mysql_com.h:
Used 'unused1' to mark memory as thread specific.
Updated parameters for my_net_init()
libmysql/libmysql.c:
Updated call to init_alloc_root() to mark memory thread specific.
libmysqld/emb_qcache.cc:
Updated call to init_alloc_root()
libmysqld/lib_sql.cc:
Updated call to init_alloc_root()
mysql-test/r/create.result:
Updated results
mysql-test/r/user_var.result:
Updated results
mysql-test/suite/funcs_1/datadict/processlist_priv.inc:
Update to handle new format of SHOW PROCESSLIST
mysql-test/suite/funcs_1/datadict/processlist_val.inc:
Update to handle new format of SHOW PROCESSLIST
mysql-test/suite/funcs_1/r/is_columns_is.result:
Update to handle new format of SHOW PROCESSLIST
mysql-test/suite/funcs_1/r/processlist_priv_no_prot.result:
Updated results
mysql-test/suite/funcs_1/r/processlist_val_no_prot.result:
Updated results
mysql-test/t/show_explain.test:
Fixed usage of debug variable so that one can run test with --debug
mysql-test/t/user_var.test:
Added test of memory_usage variable.
mysys/array.c:
Added extra my_flags option to init_dynamic_array() and init_dynamic_array2() so that one can mark memory with MY_THREAD_SPECIFIC
All allocated memory is marked with the given my_flags.
Removed obsolete function init_dynamic_array()
mysys/default.c:
Updated call to init_alloc_root()
Updated call to my_init_dynamic_array()
mysys/hash.c:
Updated call to my_init_dynamic_array_ci().
Allocated memory is marked with MY_THREAD_SPECIFIC if HASH_THREAD_SPECIFIC is used.
mysys/ma_dyncol.c:
init_dynamic_array() -> my_init_dynamic_array()
Added #if to get rid of compiler warnings
mysys/mf_tempdir.c:
Updated call to my_init_dynamic_array()
mysys/my_alloc.c:
Added extra parameter to init_alloc_root() so that one can mark memory with MY_THREAD_SPECIFIC
Extend MEM_ROOT with a flag if memory is thread specific.
This is stored in block_size, to keep the size of the MEM_ROOT object identical as before.
Allocated memory is marked with MY_THREAD_SPECIFIC if used with init_alloc_root()
mysys/my_chmod.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_chsize.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_copy.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_create.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_delete.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_error.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_fopen.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_fstream.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_getwd.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_lib.c:
Updated call to init_alloc_root()
Updated call to my_init_dynamic_array()
Updated DBUG_PRINT because of change of myf type
mysys/my_lock.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_malloc.c:
Store at start of each allocated memory block the size of the block and if the block is thread specific.
Call malloc_size_cb_func, if set, with the memory allocated/freed.
Updated DBUG_PRINT because of change of myf type
mysys/my_open.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_pread.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_read.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_redel.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_rename.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_seek.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_sync.c:
Updated DBUG_PRINT because of change of myf type
mysys/my_thr_init.c:
Ensure that one can call my_thread_dbug_id() even if thread is not properly initialized.
mysys/my_write.c:
Updated DBUG_PRINT because of change of myf type
mysys/mysys_priv.h:
Updated parameters to sf_malloc and sf_realloc()
mysys/safemalloc.c:
Added checking that for memory marked with MY_THREAD_SPECIFIC that it's the same thread that is allocation and freeing the memory.
Added sf_malloc_dbug_id() to allow MariaDB to specify which THD is handling the memory.
Added my_flags arguments to sf_malloc() and sf_realloc() to be able to mark memory with MY_THREAD_SPECIFIC.
Added sf_report_leaked_memory() to get list of memory not freed by a thread.
mysys/tree.c:
Added flags to init_tree() so that one can mark tree to be thread specific.
Removed with_delete option to init_tree(). Now one should instead use MY_TREE_WITH_DELETE_FLAG.
Updated call to init_alloc_root()
All allocated memory is marked with the given malloc flags
mysys/waiting_threads.c:
Updated call to my_init_dynamic_array()
sql-common/client.c:
Updated call to init_alloc_root() and my_net_init() to mark memory thread specific.
Updated call to my_init_dynamic_array().
Added MYSQL_THREAD_SPECIFIC_MALLOC so that client can mark memory as MY_THREAD_SPECIFIC.
sql-common/client_plugin.c:
Updated call to init_alloc_root()
sql/debug_sync.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/event_scheduler.cc:
Removed calls to net_end() as this is now done in ~THD()
Call set_current_thd() to ensure that memory is assigned to right thread.
sql/events.cc:
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/filesort.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/filesort_utils.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/ha_ndbcluster.cc:
Updated call to init_alloc_root()
Updated call to my_net_init()
Removed calls to net_end() and thd->cleanup() as these are now done in ~THD()
sql/ha_ndbcluster_binlog.cc:
Updated call to my_net_init()
Updated call to init_sql_alloc()
Removed calls to net_end() and thd->cleanup() as these are now done in ~THD()
sql/ha_partition.cc:
Updated call to init_alloc_root()
sql/handler.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
Added missing call to my_dir_end()
sql/item_func.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/item_subselect.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/item_sum.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/log.cc:
More DBUG
Updated call to init_alloc_root()
sql/mdl.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/mysqld.cc:
Added total_memory_used
Updated call to init_alloc_root()
Move mysql_cond_broadcast() before my_thread_end()
Added mariadb_dbug_id() to count memory per THD instead of per thread.
Added my_malloc_size_cb_func() callback function for my_malloc() to count memory.
Move initialization of mysqld_server_started and mysqld_server_initialized earlier.
Updated call to my_init_dynamic_array().
Updated call to my_net_init().
Call my_pthread_setspecific_ptr(THR_THD,...) to ensure that memory is assigned to right thread.
Added status variable 'memory_used'.
Updated call to init_alloc_root()
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/mysqld.h:
Added set_current_thd()
sql/net_serv.cc:
Added new parameter to my_net_init() so that one can mark memory with MY_THREAD_SPECIFIC.
Store in net->thread_specific_malloc if memory is thread specific.
Mark memory to be thread specific if requested.
sql/opt_range.cc:
Updated call to my_init_dynamic_array()
Updated call to init_sql_alloc()
Added MY_THREAD_SPECIFIC to allocated memory.
sql/opt_subselect.cc:
Updated call to init_sql_alloc() to mark memory thread specific.
sql/protocol.cc:
Fixed compiler warning
sql/records.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/rpl_filter.cc:
Updated call to my_init_dynamic_array()
sql/rpl_handler.cc:
Updated call to my_init_dynamic_array2()
sql/rpl_handler.h:
Updated call to init_sql_alloc()
sql/rpl_mi.cc:
Updated call to my_init_dynamic_array()
sql/rpl_tblmap.cc:
Updated call to init_alloc_root()
sql/rpl_utility.cc:
Updated call to my_init_dynamic_array()
sql/slave.cc:
Initialize things properly before calling functions that allocate memory.
Removed calls to net_end() as this is now done in ~THD()
sql/sp_head.cc:
Updated call to init_sql_alloc()
Updated call to my_init_dynamic_array()
Added parameter to warning_info() that it should be fully initialized.
sql/sp_pcontext.cc:
Updated call to my_init_dynamic_array()
sql/sql_acl.cc:
Updated call to init_sql_alloc()
Updated call to my_init_dynamic_array()
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/sql_admin.cc:
Added parameter to warning_info() that it should be fully initialized.
sql/sql_analyse.h:
Updated call to init_tree() to mark memory thread specific.
sql/sql_array.h:
Updated call to my_init_dynamic_array() to mark memory thread specific.
sql/sql_audit.cc:
Updated call to my_init_dynamic_array()
sql/sql_base.cc:
Updated call to init_sql_alloc()
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/sql_cache.cc:
Updated comment
sql/sql_class.cc:
Added parameter to warning_info() that not initialize it until THD is fully created.
Updated call to init_sql_alloc()
Mark THD::user_vars has to be thread specific.
Updated call to my_init_dynamic_array()
Ensure that memory allocated by THD is assigned to the THD.
More DBUG
Always acll net_end() in ~THD()
Assert that all memory signed to this THD is really deleted at ~THD.
Fixed set_status_var_init() to not reset memory_used.
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/sql_class.h:
Added MY_THREAD_SPECIFIC to allocated memory.
Added malloc_size to THD to record allocated memory per THD.
sql/sql_delete.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/sql_error.cc:
Added 'initialize' parameter to Warning_info() to say if should allocate memory for it's structures.
This is used by THD::THD() to not allocate memory until THD is ready.
Added Warning_info::free_memory()
sql/sql_error.h:
Updated Warning_info() class.
sql/sql_handler.cc:
Updated call to init_alloc_root() to mark memory thread specific.
sql/sql_insert.cc:
More DBUG
sql/sql_join_cache.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/sql_lex.cc:
Updated call to my_init_dynamic_array()
sql/sql_lex.h:
Updated call to my_init_dynamic_array()
sql/sql_load.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/sql_parse.cc:
Removed calls to net_end() and thd->cleanup() as these are now done in ~THD()
Ensure that examined_row_count() is reset before query.
Fixed bug where kill_threads_for_user() was using the wrong mem_root to allocate memory.
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
Don't restore thd->status_var.memory_used when restoring thd->status_var
sql/sql_plugin.cc:
Updated call to init_alloc_root()
Updated call to my_init_dynamic_array()
Don't allocate THD on the stack, as this causes problems with valgrind when doing thd memory counting.
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/sql_prepare.cc:
Added parameter to warning_info() that it should be fully initialized.
Updated call to init_sql_alloc() to mark memory thread specific.
sql/sql_reload.cc:
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/sql_select.cc:
Updated call to my_init_dynamic_array() and init_sql_alloc() to mark memory thread specific.
Added MY_THREAD_SPECIFIC to allocated memory.
More DBUG
sql/sql_servers.cc:
Updated call to init_sql_alloc() to mark memory some memory thread specific.
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/sql_show.cc:
Updated call to my_init_dynamic_array()
Mark my_dir() memory thread specific.
Use my_pthread_setspecific_ptr(THR_THD,...) to mark that allocated memory should be allocated to calling thread.
More DBUG.
Added malloc_size and examined_row_count to SHOW PROCESSLIST.
Added MY_THREAD_SPECIFIC to allocated memory.
Updated call to init_sql_alloc()
Added parameter to warning_info() that it should be fully initialized.
sql/sql_statistics.cc:
Fixed compiler warning
sql/sql_string.cc:
String elements can now be marked as thread specific.
sql/sql_string.h:
String elements can now be marked as thread specific.
sql/sql_table.cc:
Updated call to init_sql_alloc() and my_malloc() to mark memory thread specific
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
Fixed compiler warning
sql/sql_test.cc:
Updated call to my_init_dynamic_array() to mark memory thread specific.
sql/sql_trigger.cc:
Updated call to init_sql_alloc()
sql/sql_udf.cc:
Updated call to init_sql_alloc()
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/sql_update.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
sql/table.cc:
Updated call to init_sql_alloc().
Mark memory used by temporary tables, that are not for slave threads, as MY_THREAD_SPECIFIC
Updated call to init_sql_alloc()
sql/thr_malloc.cc:
Added my_flags argument to init_sql_alloc() to be able to mark memory as MY_THREAD_SPECIFIC.
sql/thr_malloc.h:
Updated prototype for init_sql_alloc()
sql/tztime.cc:
Updated call to init_sql_alloc()
Updated call to init_alloc_root() to mark memory thread specific.
my_pthread_setspecific_ptr(THR_THD,...) -> set_current_thd()
sql/uniques.cc:
Updated calls to init_tree(), my_init_dynamic_array() and my_malloc() to mark memory thread specific.
sql/unireg.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
storage/csv/ha_tina.cc:
Updated call to init_alloc_root()
storage/federated/ha_federated.cc:
Updated call to init_alloc_root()
Updated call to my_init_dynamic_array()
Ensure that memory allocated by fedarated is registered for the system, not for the thread.
storage/federatedx/federatedx_io_mysql.cc:
Updated call to my_init_dynamic_array()
storage/federatedx/ha_federatedx.cc:
Updated call to init_alloc_root()
Updated call to my_init_dynamic_array()
storage/heap/ha_heap.cc:
Added MY_THREAD_SPECIFIC to allocated memory.
storage/heap/heapdef.h:
Added parameter to hp_get_new_block() to be able to do thread specific memory tagging.
storage/heap/hp_block.c:
Added parameter to hp_get_new_block() to be able to do thread specific memory tagging.
storage/heap/hp_create.c:
- Internal HEAP tables are now marking it's memory as MY_THREAD_SPECIFIC.
- Use MY_TREE_WITH_DELETE instead of removed option 'with_delete'.
storage/heap/hp_open.c:
Internal HEAP tables are now marking it's memory as MY_THREAD_SPECIFIC.
storage/heap/hp_write.c:
Added new parameter to hp_get_new_block()
storage/maria/ma_bitmap.c:
Updated call to my_init_dynamic_array()
storage/maria/ma_blockrec.c:
Updated call to my_init_dynamic_array()
storage/maria/ma_check.c:
Updated call to init_alloc_root()
storage/maria/ma_ft_boolean_search.c:
Updated calls to init_tree() and init_alloc_root()
storage/maria/ma_ft_nlq_search.c:
Updated call to init_tree()
storage/maria/ma_ft_parser.c:
Updated call to init_tree()
Updated call to init_alloc_root()
storage/maria/ma_loghandler.c:
Updated call to my_init_dynamic_array()
storage/maria/ma_open.c:
Updated call to my_init_dynamic_array()
storage/maria/ma_sort.c:
Updated call to my_init_dynamic_array()
storage/maria/ma_write.c:
Updated calls to my_init_dynamic_array() and init_tree()
storage/maria/maria_pack.c:
Updated call to init_tree()
storage/maria/unittest/sequence_storage.c:
Updated call to my_init_dynamic_array()
storage/myisam/ft_boolean_search.c:
Updated call to init_tree()
Updated call to init_alloc_root()
storage/myisam/ft_nlq_search.c:
Updated call to init_tree()
storage/myisam/ft_parser.c:
Updated call to init_tree()
Updated call to init_alloc_root()
storage/myisam/ft_stopwords.c:
Updated call to init_tree()
storage/myisam/mi_check.c:
Updated call to init_alloc_root()
storage/myisam/mi_write.c:
Updated call to my_init_dynamic_array()
Updated call to init_tree()
storage/myisam/myisamlog.c:
Updated call to init_tree()
storage/myisam/myisampack.c:
Updated call to init_tree()
storage/myisam/sort.c:
Updated call to my_init_dynamic_array()
storage/myisammrg/ha_myisammrg.cc:
Updated call to init_sql_alloc()
storage/perfschema/pfs_check.cc:
Rest current_thd
storage/perfschema/pfs_instr.cc:
Removed DBUG_ENTER/DBUG_VOID_RETURN as at this point my_thread_var is not allocated anymore, which can cause problems.
support-files/compiler_warnings.supp:
Disable compiler warning from offsetof macro.
2013-01-23 16:16:14 +01:00
|
|
|
MYF(MY_WME | MY_THREAD_SPECIFIC))))
|
2010-02-19 22:55:57 +01:00
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
/* Create the only non-NULL key if there is any. */
|
|
|
|
if (non_null_key_parts)
|
|
|
|
{
|
2010-02-22 16:16:55 +01:00
|
|
|
non_null_key= new Ordered_key(cur_keyid, tmp_table, item_in->left_expr,
|
2010-02-19 22:55:57 +01:00
|
|
|
0, 0, 0, row_num_to_rowid);
|
|
|
|
if (non_null_key->init(non_null_key_parts))
|
|
|
|
return TRUE;
|
2010-02-22 16:16:55 +01:00
|
|
|
merge_keys[cur_keyid]= non_null_key;
|
|
|
|
merge_keys[cur_keyid]->first();
|
|
|
|
++cur_keyid;
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2011-07-13 23:15:07 +02:00
|
|
|
If all nullable columns contain NULLs, the only key that is needed is the
|
|
|
|
only non-NULL key that is already created above.
|
2010-02-19 22:55:57 +01:00
|
|
|
*/
|
2011-07-13 23:15:07 +02:00
|
|
|
if (!has_covering_null_columns)
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
2014-01-02 10:19:19 +01:00
|
|
|
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))
|
2010-02-19 22:55:57 +01:00
|
|
|
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++)
|
|
|
|
{
|
2011-07-13 16:09:09 +02:00
|
|
|
/* 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)
|
2010-02-19 22:55:57 +01:00
|
|
|
continue;
|
|
|
|
|
2011-07-13 16:09:09 +02:00
|
|
|
merge_keys[cur_keyid]= new Ordered_key(
|
2010-02-22 16:16:55 +01:00
|
|
|
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);
|
2011-07-13 16:09:09 +02:00
|
|
|
if (merge_keys[cur_keyid]->init(i))
|
|
|
|
return TRUE;
|
|
|
|
merge_keys[cur_keyid]->first();
|
2010-02-22 16:16:55 +01:00
|
|
|
++cur_keyid;
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
}
|
2011-07-13 16:09:09 +02:00
|
|
|
DBUG_ASSERT(cur_keyid == merge_keys_count);
|
2010-02-19 22:55:57 +01:00
|
|
|
|
|
|
|
/* Populate the indexes with data from the temporary table. */
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely(tmp_table->file->ha_rnd_init_with_error(1)))
|
2010-11-02 10:03:33 +01:00
|
|
|
return TRUE;
|
2010-02-19 22:55:57 +01:00
|
|
|
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]);
|
|
|
|
/*
|
|
|
|
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);
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely(error == HA_ERR_END_OF_FILE))
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2011-07-13 16:09:09 +02:00
|
|
|
for (uint i= (non_null_key ? 1 : 0); i < merge_keys_count; i++)
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
Check if the first and only indexed column contains NULL in the curent
|
|
|
|
row, and add the row number to the corresponding key.
|
|
|
|
*/
|
2016-01-13 17:43:54 +01:00
|
|
|
if (merge_keys[i]->get_field(0)->is_null())
|
2010-02-19 22:55:57 +01:00
|
|
|
merge_keys[i]->set_null(cur_rownum);
|
|
|
|
else
|
|
|
|
merge_keys[i]->add_key(cur_rownum);
|
|
|
|
}
|
|
|
|
++cur_rownum;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp_table->file->ha_rnd_end();
|
|
|
|
|
2010-02-22 16:16:55 +01:00
|
|
|
/* Sort all the keys by their NULL selectivity. */
|
2011-07-13 16:09:09 +02:00
|
|
|
my_qsort(merge_keys, merge_keys_count, sizeof(Ordered_key*),
|
2010-02-22 16:16:55 +01:00
|
|
|
(qsort_cmp) cmp_keys_by_null_selectivity);
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
/* Sort the keys in each of the indexes. */
|
2011-07-13 16:09:09 +02:00
|
|
|
for (uint i= 0; i < merge_keys_count; i++)
|
2018-12-20 09:52:34 +01:00
|
|
|
if (merge_keys[i]->sort_keys())
|
|
|
|
return TRUE;
|
2010-02-19 22:55:57 +01:00
|
|
|
|
2011-07-13 16:09:09 +02:00
|
|
|
if (init_queue(&pq, merge_keys_count, 0, FALSE,
|
2010-07-16 09:33:01 +02:00
|
|
|
subselect_rowid_merge_engine::cmp_keys_by_cur_rownum, NULL,
|
|
|
|
0, 0))
|
2010-02-19 22:55:57 +01:00
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
subselect_rowid_merge_engine::~subselect_rowid_merge_engine()
|
|
|
|
{
|
2010-03-09 11:14:06 +01:00
|
|
|
/* None of the resources below is allocated if there are no ordered keys. */
|
2011-07-13 16:09:09 +02:00
|
|
|
if (merge_keys_count)
|
2010-03-09 11:14:06 +01:00
|
|
|
{
|
2011-04-25 17:22:25 +02:00
|
|
|
my_free(row_num_to_rowid);
|
2011-07-13 16:09:09 +02:00
|
|
|
for (uint i= 0; i < merge_keys_count; i++)
|
2010-03-09 11:14:06 +01:00
|
|
|
delete merge_keys[i];
|
|
|
|
delete_queue(&pq);
|
|
|
|
if (tmp_table->file->inited == handler::RND)
|
|
|
|
tmp_table->file->ha_rnd_end();
|
|
|
|
}
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void subselect_rowid_merge_engine::cleanup()
|
|
|
|
{
|
2010-02-22 16:16:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
/*
|
2010-02-22 16:16:55 +01:00
|
|
|
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
|
2010-02-19 22:55:57 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
int
|
2010-02-22 16:16:55 +01:00
|
|
|
subselect_rowid_merge_engine::cmp_keys_by_null_selectivity(Ordered_key **k1,
|
|
|
|
Ordered_key **k2)
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
2010-02-22 16:16:55 +01:00
|
|
|
double k1_sel= (*k1)->null_selectivity();
|
|
|
|
double k2_sel= (*k2)->null_selectivity();
|
|
|
|
if (k1_sel < k2_sel)
|
2010-02-19 22:55:57 +01:00
|
|
|
return 1;
|
2010-02-22 16:16:55 +01:00
|
|
|
if (k1_sel > k2_sel)
|
|
|
|
return -1;
|
|
|
|
return 0;
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*/
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
2011-07-13 16:09:09 +02:00
|
|
|
@note
|
|
|
|
There is no need to check the columns that contain only NULLs, because
|
|
|
|
those are guaranteed to match.
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
@retval TRUE if a NULL row exists
|
|
|
|
@retval FALSE otherwise
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool subselect_rowid_merge_engine::test_null_row(rownum_t row_num)
|
|
|
|
{
|
2010-02-22 16:16:55 +01:00
|
|
|
Ordered_key *cur_key;
|
2011-07-13 16:09:09 +02:00
|
|
|
for (uint i = 0; i < merge_keys_count; i++)
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
2010-02-22 16:16:55 +01:00
|
|
|
cur_key= merge_keys[i];
|
2011-07-13 16:09:09 +02:00
|
|
|
if (bitmap_is_set(&matching_keys, cur_key->get_keyid()))
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
|
|
|
/*
|
2011-07-13 16:09:09 +02:00
|
|
|
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.
|
2010-02-19 22:55:57 +01:00
|
|
|
*/
|
|
|
|
continue;
|
|
|
|
}
|
2010-02-22 16:16:55 +01:00
|
|
|
if (!cur_key->is_null(row_num))
|
2010-02-19 22:55:57 +01:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-17 00:25:10 +01:00
|
|
|
/**
|
|
|
|
Test if a subset of NULL-able columns contains a row of NULLs.
|
2011-11-23 22:13:51 +01:00
|
|
|
@retval TRUE if such a row exists
|
|
|
|
@retval FALSE no complementing null row
|
2011-11-17 00:25:10 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
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;
|
2011-11-23 22:13:51 +01:00
|
|
|
uint count_null_keys, i;
|
2011-11-17 00:25:10 +01:00
|
|
|
Ordered_key *cur_key;
|
|
|
|
|
2011-11-23 22:13:51 +01:00
|
|
|
if (!count_columns_with_nulls)
|
2011-11-17 00:25:10 +01:00
|
|
|
{
|
|
|
|
/*
|
2011-11-23 22:13:51 +01:00
|
|
|
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.
|
2011-11-17 00:25:10 +01:00
|
|
|
*/
|
2011-11-23 22:13:51 +01:00
|
|
|
return FALSE;
|
2011-11-17 00:25:10 +01:00
|
|
|
}
|
|
|
|
|
2011-11-23 22:13:51 +01:00
|
|
|
for (i= (non_null_key ? 1 : 0), count_null_keys= 0; i < merge_keys_count; i++)
|
2011-11-17 00:25:10 +01:00
|
|
|
{
|
|
|
|
cur_key= merge_keys[i];
|
|
|
|
if (bitmap_is_set(keys_to_complement, cur_key->get_keyid()))
|
|
|
|
continue;
|
2011-11-23 22:13:51 +01:00
|
|
|
if (!cur_key->get_null_count())
|
|
|
|
{
|
|
|
|
/* If there is column without NULLs, there cannot be a partial match. */
|
|
|
|
return FALSE;
|
|
|
|
}
|
2011-11-17 00:25:10 +01:00
|
|
|
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();
|
2011-11-23 22:13:51 +01:00
|
|
|
null_bitmaps[count_null_keys++]= cur_key->get_null_key();
|
2011-11-17 00:25:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
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,
|
2011-11-24 22:15:40 +01:00
|
|
|
(uint)highest_min_row, (uint)lowest_max_row);
|
2011-11-17 00:25:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
/*
|
|
|
|
@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;
|
2011-11-17 00:25:10 +01:00
|
|
|
uint max_null_in_any_row=
|
2011-10-03 21:48:15 +02:00
|
|
|
((select_materialize_with_stats *) result)->get_max_nulls_in_row();
|
2010-03-09 11:14:06 +01:00
|
|
|
bool res= FALSE;
|
2010-02-19 22:55:57 +01:00
|
|
|
|
|
|
|
/* 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));
|
2010-08-30 10:07:16 +02:00
|
|
|
/* The prioryty queue for keys must be empty. */
|
|
|
|
DBUG_ASSERT(!pq.elements);
|
2010-03-09 11:14:06 +01:00
|
|
|
|
|
|
|
/* All data accesses during execution are via handler::ha_rnd_pos() */
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely(tmp_table->file->ha_rnd_init_with_error(0)))
|
2010-11-02 10:03:33 +01:00
|
|
|
{
|
|
|
|
res= FALSE;
|
|
|
|
goto end;
|
|
|
|
}
|
2010-03-09 11:14:06 +01:00
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
/* Check if there is a match for the columns of the only non-NULL key. */
|
|
|
|
if (non_null_key && !non_null_key->lookup())
|
2010-03-09 11:14:06 +01:00
|
|
|
{
|
|
|
|
res= FALSE;
|
|
|
|
goto end;
|
|
|
|
}
|
2010-02-19 22:55:57 +01:00
|
|
|
|
|
|
|
/*
|
2011-07-13 23:15:07 +02:00
|
|
|
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)
|
2010-03-09 11:14:06 +01:00
|
|
|
{
|
|
|
|
res= TRUE;
|
|
|
|
goto end;
|
|
|
|
}
|
2010-02-19 22:55:57 +01:00
|
|
|
|
|
|
|
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);
|
2014-02-19 11:05:15 +01:00
|
|
|
for (uint i= MY_TEST(non_null_key); i < merge_keys_count; i++)
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
|
|
|
DBUG_ASSERT(merge_keys[i]->get_column_count() == 1);
|
2011-03-08 22:23:44 +01:00
|
|
|
if (merge_keys[i]->get_search_key(0)->null_value)
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
|
|
|
++count_nulls_in_search_key;
|
2010-02-22 16:16:55 +01:00
|
|
|
bitmap_set_bit(&matching_outer_cols, merge_keys[i]->get_keyid());
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
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
|
2011-11-17 00:25:10 +01:00
|
|
|
nullable columns (above we guarantee there is a match for the non-null
|
|
|
|
coumns), the result is UNKNOWN.
|
2010-02-19 22:55:57 +01:00
|
|
|
*/
|
2014-02-19 11:05:15 +01:00
|
|
|
if (count_nulls_in_search_key == merge_keys_count - MY_TEST(non_null_key))
|
2010-03-09 11:14:06 +01:00
|
|
|
{
|
|
|
|
res= TRUE;
|
|
|
|
goto end;
|
|
|
|
}
|
2010-02-19 22:55:57 +01:00
|
|
|
|
2011-11-17 00:25:10 +01:00
|
|
|
/*
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2010-02-19 22:55:57 +01:00
|
|
|
/*
|
|
|
|
If there is no NULL (sub)row that covers all NULL columns, and there is no
|
2011-11-17 00:25:10 +01:00
|
|
|
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.
|
2010-02-19 22:55:57 +01:00
|
|
|
*/
|
2011-11-17 00:25:10 +01:00
|
|
|
if (!count_nulls_in_search_key &&
|
|
|
|
(!pq.elements ||
|
|
|
|
(pq.elements == 1 && non_null_key &&
|
|
|
|
max_null_in_any_row < merge_keys_count-1)))
|
2010-03-09 11:14:06 +01:00
|
|
|
{
|
2011-11-17 00:25:10 +01:00
|
|
|
if (!pq.elements)
|
2011-10-03 21:48:15 +02:00
|
|
|
{
|
2011-11-17 00:25:10 +01:00
|
|
|
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);
|
2011-10-03 21:48:15 +02:00
|
|
|
}
|
2010-03-09 11:14:06 +01:00
|
|
|
res= FALSE;
|
|
|
|
goto end;
|
|
|
|
}
|
2010-02-19 22:55:57 +01:00
|
|
|
|
|
|
|
DBUG_ASSERT(pq.elements);
|
|
|
|
|
2010-08-30 10:07:16 +02:00
|
|
|
min_key= (Ordered_key*) queue_remove_top(&pq);
|
2010-02-19 22:55:57 +01:00
|
|
|
min_row_num= min_key->current();
|
2010-02-22 16:16:55 +01:00
|
|
|
bitmap_set_bit(&matching_keys, min_key->get_keyid());
|
2010-02-19 22:55:57 +01:00
|
|
|
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.
|
|
|
|
*/
|
2010-03-09 11:14:06 +01:00
|
|
|
res= test_null_row(min_row_num);
|
|
|
|
goto end;
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
while (TRUE)
|
|
|
|
{
|
2010-08-30 10:07:16 +02:00
|
|
|
cur_key= (Ordered_key*) queue_remove_top(&pq);
|
2010-02-19 22:55:57 +01:00
|
|
|
cur_row_num= cur_key->current();
|
|
|
|
|
|
|
|
if (cur_row_num == min_row_num)
|
2010-02-22 16:16:55 +01:00
|
|
|
bitmap_set_bit(&matching_keys, cur_key->get_keyid());
|
2010-02-19 22:55:57 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Follows from the correct use of priority queue. */
|
|
|
|
DBUG_ASSERT(cur_row_num > min_row_num);
|
|
|
|
if (test_null_row(min_row_num))
|
2010-03-09 11:14:06 +01:00
|
|
|
{
|
|
|
|
res= TRUE;
|
|
|
|
goto end;
|
|
|
|
}
|
2010-02-19 22:55:57 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
min_key= cur_key;
|
|
|
|
min_row_num= cur_row_num;
|
2011-08-22 23:00:13 +02:00
|
|
|
bitmap_clear_all(&matching_keys);
|
2010-02-22 16:16:55 +01:00
|
|
|
bitmap_set_bit(&matching_keys, min_key->get_keyid());
|
2010-02-19 22:55:57 +01:00
|
|
|
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. */
|
2010-03-09 11:14:06 +01:00
|
|
|
res= test_null_row(min_row_num);
|
|
|
|
goto end;
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
/* We should never get here - all branches must be handled explicitly above. */
|
2010-02-19 22:55:57 +01:00
|
|
|
DBUG_ASSERT(FALSE);
|
2010-03-09 11:14:06 +01:00
|
|
|
|
|
|
|
end:
|
2011-10-04 22:57:46 +02:00
|
|
|
if (!has_covering_null_columns)
|
|
|
|
bitmap_clear_all(&matching_keys);
|
2010-08-30 10:07:16 +02:00
|
|
|
queue_remove_all(&pq);
|
2010-03-09 11:14:06 +01:00
|
|
|
tmp_table->file->ha_rnd_end();
|
|
|
|
return res;
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
subselect_table_scan_engine::subselect_table_scan_engine(
|
2016-06-22 14:17:06 +02:00
|
|
|
subselect_uniquesubquery_engine *engine_arg,
|
2010-03-09 11:14:06 +01:00
|
|
|
TABLE *tmp_table_arg,
|
|
|
|
Item_subselect *item_arg,
|
|
|
|
select_result_interceptor *result_arg,
|
|
|
|
List<Item> *equi_join_conds_arg,
|
2011-07-13 23:15:07 +02:00
|
|
|
bool has_covering_null_row_arg,
|
2011-08-22 23:00:13 +02:00
|
|
|
bool has_covering_null_columns_arg,
|
|
|
|
uint count_columns_with_nulls_arg)
|
2016-06-22 14:17:06 +02:00
|
|
|
:subselect_partial_match_engine(engine_arg, tmp_table_arg, item_arg,
|
2010-03-09 11:14:06 +01:00
|
|
|
result_arg, equi_join_conds_arg,
|
2011-07-13 23:15:07 +02:00
|
|
|
has_covering_null_row_arg,
|
2011-08-22 23:00:13 +02:00
|
|
|
has_covering_null_columns_arg,
|
|
|
|
count_columns_with_nulls_arg)
|
2010-03-09 11:14:06 +01:00
|
|
|
{}
|
2010-02-19 22:55:57 +01:00
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
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
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
2010-03-09 11:14:06 +01:00
|
|
|
count_matches= 0;
|
|
|
|
for each row element row_i[j]
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
2010-03-09 11:14:06 +01:00
|
|
|
if (outer_ref[j] is NULL || row_i[j] is NULL || outer_ref[j] == row_i[j])
|
|
|
|
++count_matches;
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
2010-03-09 11:14:06 +01:00
|
|
|
if (count_matches == outer_ref.elements)
|
|
|
|
return TRUE
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
2010-03-09 11:14:06 +01:00
|
|
|
return FALSE
|
|
|
|
*/
|
2010-02-19 22:55:57 +01:00
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
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;
|
2010-02-19 22:55:57 +01:00
|
|
|
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely(tmp_table->file->ha_rnd_init_with_error(1)))
|
2010-11-02 10:03:33 +01:00
|
|
|
{
|
|
|
|
res= FALSE;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
tmp_table->file->extra_opt(HA_EXTRA_CACHE,
|
2016-07-29 18:21:08 +02:00
|
|
|
get_thd()->variables.read_buff_size);
|
2010-03-09 11:14:06 +01:00
|
|
|
for (;;)
|
2010-02-19 22:55:57 +01:00
|
|
|
{
|
2010-03-09 11:14:06 +01:00
|
|
|
error= tmp_table->file->ha_rnd_next(tmp_table->record[0]);
|
2018-04-04 11:16:12 +02:00
|
|
|
if (unlikely(error))
|
|
|
|
{
|
2010-03-09 11:14:06 +01:00
|
|
|
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;
|
|
|
|
}
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
2010-03-09 11:14:06 +01:00
|
|
|
|
|
|
|
res= FALSE;
|
|
|
|
end:
|
2010-02-19 22:55:57 +01:00
|
|
|
tmp_table->file->ha_rnd_end();
|
2010-03-09 11:14:06 +01:00
|
|
|
return res;
|
|
|
|
}
|
2010-02-19 22:55:57 +01:00
|
|
|
|
2010-03-09 11:14:06 +01:00
|
|
|
|
|
|
|
void subselect_table_scan_engine::cleanup()
|
|
|
|
{
|
2010-02-19 22:55:57 +01:00
|
|
|
}
|
2011-07-19 22:19:10 +02:00
|
|
|
|
2015-07-01 19:03:29 +02:00
|
|
|
|
2017-04-20 22:09:31 +02:00
|
|
|
void Item_subselect::register_as_with_rec_ref(With_element *with_elem)
|
|
|
|
{
|
|
|
|
with_elem->sq_with_rec_ref.link_in_list(this, &this->next_with_rec_ref);
|
|
|
|
with_recursive_reference= true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-07-01 19:03:29 +02:00
|
|
|
/*
|
|
|
|
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)
|
2015-03-25 18:27:10 +01:00
|
|
|
{
|
|
|
|
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);
|
2015-07-01 19:03:29 +02:00
|
|
|
node->cache_tracker= ((Item_cache_wrapper *)expr_cache)->init_tracker(qw->mem_root);
|
2015-03-25 18:27:10 +01:00
|
|
|
}
|