2002-05-12 22:46:42 +02:00
|
|
|
/* Copyright (C) 2000 MySQL AB
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
|
|
|
|
|
2004-07-04 07:46:28 +02:00
|
|
|
/*
|
2002-05-12 22:46:42 +02:00
|
|
|
subselect Item
|
|
|
|
|
|
|
|
SUBSELECT TODO:
|
|
|
|
- add function from mysql_select that use JOIN* as parameter to JOIN methods
|
|
|
|
(sql_select.h/sql_select.cc)
|
|
|
|
*/
|
|
|
|
|
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
|
2005-06-05 19:38:52 +02:00
|
|
|
|
|
|
|
#include "mysql_priv.h"
|
2002-05-12 22:46:42 +02:00
|
|
|
#include "sql_select.h"
|
|
|
|
|
2002-12-28 00:01:05 +01:00
|
|
|
inline Item * and_items(Item* cond, Item *item)
|
|
|
|
{
|
|
|
|
return (cond? (new Item_cond_and(cond, item)) : item);
|
|
|
|
}
|
|
|
|
|
2002-10-08 22:49:59 +02:00
|
|
|
Item_subselect::Item_subselect():
|
2004-02-08 19:14:13 +01:00
|
|
|
Item_result_field(), value_assigned(0), thd(0), substitution(0),
|
|
|
|
engine(0), old_engine(0), used_tables_cache(0), have_to_be_excluded(0),
|
|
|
|
const_item_cache(1), engine_changed(0), changed(0)
|
2002-05-12 22:46:42 +02:00
|
|
|
{
|
2002-12-06 20:55:53 +01:00
|
|
|
reset();
|
2002-10-08 22:49:59 +02:00
|
|
|
/*
|
2004-07-04 07:46:28 +02:00
|
|
|
item value is NULL if select_subselect not changed this value
|
2002-10-08 22:49:59 +02:00
|
|
|
(i.e. some rows will be found returned)
|
|
|
|
*/
|
|
|
|
null_value= 1;
|
|
|
|
}
|
|
|
|
|
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,
|
2002-11-07 22:45:19 +01:00
|
|
|
select_subselect *result)
|
2002-10-08 22:49:59 +02:00
|
|
|
{
|
|
|
|
|
|
|
|
DBUG_ENTER("Item_subselect::init");
|
2002-10-27 22:27:00 +01:00
|
|
|
DBUG_PRINT("subs", ("select_lex 0x%xl", (ulong) select_lex));
|
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)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
Item can be changed in JOIN::prepare while engine in JOIN::optimize
|
|
|
|
=> we do not copy old_engine here
|
|
|
|
*/
|
|
|
|
engine= unit->item->engine;
|
2004-08-13 09:01:30 +02:00
|
|
|
parsing_place= unit->item->parsing_place;
|
2004-05-07 22:06:11 +02:00
|
|
|
unit->item->engine= 0;
|
|
|
|
unit->item= this;
|
|
|
|
engine->change_item(this, result);
|
|
|
|
}
|
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);
|
2004-05-07 22:06:11 +02:00
|
|
|
if (select_lex->next_select())
|
|
|
|
engine= new subselect_union_engine(unit, result, this);
|
|
|
|
else
|
|
|
|
engine= new subselect_single_select_engine(select_lex, result, this);
|
|
|
|
}
|
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;
|
|
|
|
}
|
2002-05-12 22:46:42 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
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();
|
|
|
|
value_assigned= 0;
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
2002-09-03 08:50:36 +02:00
|
|
|
Item_subselect::~Item_subselect()
|
|
|
|
{
|
2003-10-27 00:01:27 +01:00
|
|
|
delete engine;
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
Item_subselect::trans_res
|
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");
|
2003-08-28 12:21:30 +02:00
|
|
|
DBUG_RETURN(RES_OK);
|
2002-10-27 22:27:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-05-28 15:52:56 +02:00
|
|
|
bool Item_subselect::fix_fields(THD *thd_param, TABLE_LIST *tables, Item **ref)
|
2002-05-12 22:46:42 +02:00
|
|
|
{
|
2004-03-17 13:26:26 +01:00
|
|
|
DBUG_ASSERT(fixed == 0);
|
2003-10-02 21:19:41 +02:00
|
|
|
engine->set_thd((thd= thd_param));
|
2003-05-28 15:52:56 +02:00
|
|
|
|
2002-11-24 20:10:52 +01:00
|
|
|
char const *save_where= thd->where;
|
2004-07-14 22:57:14 +02:00
|
|
|
int res;
|
|
|
|
|
|
|
|
if (check_stack_overrun(thd, (gptr)&res))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
res= engine->prepare();
|
2004-02-08 19:14:13 +01:00
|
|
|
|
|
|
|
// all transformetion is done (used by prepared statements)
|
|
|
|
changed= 1;
|
|
|
|
|
2002-10-08 13:50:12 +02:00
|
|
|
if (!res)
|
2002-10-12 00:09:47 +02:00
|
|
|
{
|
2003-05-14 20:51:33 +02:00
|
|
|
if (substitution)
|
|
|
|
{
|
2004-03-20 12:36:26 +01:00
|
|
|
int ret= 0;
|
|
|
|
|
2004-02-08 19:14:13 +01:00
|
|
|
// did we changed top item of WHERE condition
|
|
|
|
if (unit->outer_select()->where == (*ref))
|
|
|
|
unit->outer_select()->where= substitution; // correct WHERE for PS
|
2004-12-08 22:37:17 +01:00
|
|
|
else if (unit->outer_select()->having == (*ref))
|
|
|
|
unit->outer_select()->having= substitution; // correct HAVING for PS
|
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";
|
2004-03-17 13:26:26 +01:00
|
|
|
if (!(*ref)->fixed)
|
|
|
|
ret= (*ref)->fix_fields(thd, tables, ref);
|
2004-10-27 20:11:06 +02:00
|
|
|
thd->where= save_where;
|
2003-05-14 20:51:33 +02:00
|
|
|
return ret;
|
|
|
|
}
|
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);
|
2002-10-12 00:09:47 +02:00
|
|
|
return 1;
|
|
|
|
}
|
2002-10-08 13:50:12 +02:00
|
|
|
fix_length_and_dec();
|
2002-10-12 00:09:47 +02:00
|
|
|
}
|
2005-01-18 15:26:05 +01:00
|
|
|
else
|
|
|
|
return 1;
|
2003-11-17 19:53:40 +01:00
|
|
|
uint8 uncacheable= engine->uncacheable();
|
|
|
|
if (uncacheable)
|
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;
|
2002-11-24 20:10:52 +01:00
|
|
|
thd->where= save_where;
|
2002-09-28 17:34:56 +02:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2003-05-28 15:52:56 +02:00
|
|
|
bool Item_subselect::exec()
|
|
|
|
{
|
2003-07-07 17:40:19 +02:00
|
|
|
int res;
|
2004-11-08 00:13:54 +01:00
|
|
|
MEM_ROOT *old_root= thd->mem_root;
|
|
|
|
|
|
|
|
/*
|
|
|
|
As this is execution, all objects should be allocated through the main
|
|
|
|
mem root
|
|
|
|
*/
|
|
|
|
thd->mem_root= &thd->main_mem_root;
|
|
|
|
res= engine->exec();
|
|
|
|
thd->mem_root= old_root;
|
|
|
|
|
2003-07-07 17:40:19 +02:00
|
|
|
if (engine_changed)
|
|
|
|
{
|
|
|
|
engine_changed= 0;
|
|
|
|
return exec();
|
|
|
|
}
|
|
|
|
return (res);
|
2003-05-28 15:52:56 +02:00
|
|
|
}
|
|
|
|
|
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
|
|
|
|
2002-09-28 17:34:56 +02:00
|
|
|
void Item_subselect::fix_length_and_dec()
|
|
|
|
{
|
2002-12-19 06:38:33 +01:00
|
|
|
engine->fix_length_and_dec(0);
|
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
|
|
|
{
|
2003-11-17 19:53:40 +01:00
|
|
|
return (table_map) (engine->uncacheable() ? 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
|
|
|
|
{
|
2003-10-23 19:50:53 +02:00
|
|
|
return const_item_cache;
|
2003-10-16 23:36:01 +02:00
|
|
|
}
|
|
|
|
|
2003-12-06 20:37:24 +01:00
|
|
|
Item *Item_subselect::get_tmp_table_item(THD *thd)
|
|
|
|
{
|
|
|
|
if (!with_sum_func && !const_item())
|
2004-03-20 12:36:26 +01:00
|
|
|
return new Item_field(result_field);
|
2003-12-06 20:37:24 +01:00
|
|
|
return copy_or_same(thd);
|
|
|
|
}
|
2003-10-28 11:45:37 +01:00
|
|
|
|
2003-10-16 23:36:01 +02:00
|
|
|
void Item_subselect::update_used_tables()
|
|
|
|
{
|
|
|
|
if (!engine->uncacheable())
|
|
|
|
{
|
|
|
|
// did all used tables become ststic?
|
2003-11-01 18:02:43 +01:00
|
|
|
if (!(used_tables_cache & ~engine->upper_select_const_tables()))
|
2003-10-16 23:36:01 +02:00
|
|
|
const_item_cache= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
|
|
|
void Item_subselect::print(String *str)
|
|
|
|
{
|
|
|
|
str->append('(');
|
|
|
|
engine->print(str);
|
|
|
|
str->append(')');
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-07 07:51:09 +02:00
|
|
|
Item_singlerow_subselect::Item_singlerow_subselect(st_select_lex *select_lex)
|
2003-08-12 11:38:03 +02:00
|
|
|
:Item_subselect(), 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");
|
2003-10-02 21:19:41 +02:00
|
|
|
init(select_lex, new select_singlerow_subselect(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
|
|
|
}
|
|
|
|
|
2003-10-27 00:01:27 +01:00
|
|
|
Item_maxmin_subselect::Item_maxmin_subselect(Item_subselect *parent,
|
|
|
|
st_select_lex *select_lex,
|
2003-10-16 14:54:47 +02:00
|
|
|
bool max_arg)
|
2004-11-18 17:10:07 +01:00
|
|
|
:Item_singlerow_subselect(), 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;
|
|
|
|
init(select_lex, new select_max_min_finder_subselect(this, max_arg));
|
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();
|
|
|
|
const_item_cache= parent->get_const_item_cache();
|
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
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
void Item_maxmin_subselect::print(String *str)
|
|
|
|
{
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append(max?"<max>":"<min>", 5);
|
2003-10-16 14:54:47 +02:00
|
|
|
Item_singlerow_subselect::print(str);
|
|
|
|
}
|
|
|
|
|
2004-11-18 17:10:07 +01:00
|
|
|
|
2002-12-19 20:15:09 +01:00
|
|
|
void Item_singlerow_subselect::reset()
|
2002-09-28 17:34:56 +02:00
|
|
|
{
|
2002-12-19 06:38:32 +01:00
|
|
|
null_value= 1;
|
|
|
|
if (value)
|
|
|
|
value->null_value= 1;
|
2002-09-28 17:34:56 +02:00
|
|
|
}
|
|
|
|
|
2004-11-18 17:10:07 +01:00
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
Item_subselect::trans_res
|
2003-07-02 00:45:22 +02:00
|
|
|
Item_singlerow_subselect::select_transformer(JOIN *join)
|
2002-12-26 00:28:59 +01:00
|
|
|
{
|
2004-02-08 19:14:13 +01:00
|
|
|
if (changed)
|
|
|
|
return RES_OK;
|
2004-07-04 07:46:28 +02:00
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
SELECT_LEX *select_lex= join->select_lex;
|
Fix for bug#4912 "mysqld crashs in case a statement is executed
a second time". The bug was caused by incompatibility of
negations elimination algorithm and PS: during first statement
execute a subtree with negation was replaced with equivalent
subtree without NOTs.
The problem was that although this transformation was permanent,
items of the new subtree were created in execute-local memory.
The patch adds means to check if it is the first execute of a
prepared statement, and if this is the case, to allocate items
in memory of the prepared statement.
The implementation:
- backports Item_arena from 5.0
- adds Item_arena::is_stmt_prepare(),
Item_arena::is_first_stmt_execute().
- deletes THD::allocate_temporary_pool_for_ps_preparing(),
THD::free_temporary_pool_for_ps_preparing(); they
were redundant.
and adds a few invariants:
- thd->free_list never contains junk (= freed items)
- thd->current_arena is never null. If there is no
prepared statement, it points at the thd.
The rest of the patch contains mainly mechanical changes and
cleanups.
mysql-test/r/ps.result:
Test results updated (test case for Bug#4912)
mysql-test/t/ps.test:
A test case for Bug#4912 "mysqld crashs in case a statement is
executed a second time"
sql/item_cmpfunc.cc:
current_statement -> current_arena
sql/item_subselect.cc:
Statement -> Item_arena, current_statement -> current_arena
sql/item_subselect.h:
Item_subselect does not need to save thd->current_statement.
sql/item_sum.cc:
Statement -> Item_arena
sql/item_sum.h:
Statement -> Item_arena
sql/mysql_priv.h:
Statement -> Item_arena
sql/sql_base.cc:
current_statement -> current_arena
sql/sql_class.cc:
- Item_arena
- convenient set_n_backup_statement, restore_backup_statement
(nice idea, Sanja)
sql/sql_class.h:
- Item_arena: backport from 5.0
- allocate_temporary_pool_for_ps_preparing,
free_temporary_pool_for_ps_preparing removed.
sql/sql_derived.cc:
current_statement -> current_arena
sql/sql_lex.cc:
current_statement -> current_arena
sql/sql_parse.cc:
Deploy invariant that thd->free_list never contains junk items
(backport from 5.0).
sql/sql_prepare.cc:
- backporting Item_arena
- no need to allocate_temporary_pool_for_ps_preparing().
sql/sql_select.cc:
Fix for bug#4912 "mysqld crashs in case a statement is
executed a second time": if this is the first execute of
a prepared statement, negation elimination is
done in memory of the prepared statement.
sql/sql_union.cc:
Backporting Item_arena from 5.0.
2004-08-21 00:02:46 +02:00
|
|
|
|
|
|
|
/* Juggle with current arena only if we're in prepared statement prepare */
|
|
|
|
Item_arena *arena= join->thd->current_arena;
|
2004-02-08 19:14:13 +01:00
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
if (!select_lex->master_unit()->first_select()->next_select() &&
|
|
|
|
!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) &&
|
|
|
|
/*
|
|
|
|
switch off this optimisation for prepare statement,
|
|
|
|
because we do not rollback this changes
|
|
|
|
TODO: make rollback for it, or special name resolving mode in 5.0.
|
|
|
|
*/
|
|
|
|
!arena->is_stmt_prepare()
|
2002-12-31 17:39:16 +01:00
|
|
|
)
|
2002-12-26 00:28:59 +01:00
|
|
|
{
|
2004-07-04 07:46:28 +02:00
|
|
|
|
2002-12-26 00:28:59 +01:00
|
|
|
have_to_be_excluded= 1;
|
2003-12-19 18:52:13 +01:00
|
|
|
if (join->thd->lex->describe)
|
2002-12-26 00:28:59 +01:00
|
|
|
{
|
|
|
|
char warn_buff[MYSQL_ERRMSG_SIZE];
|
|
|
|
sprintf(warn_buff, ER(ER_SELECT_REDUCED), select_lex->select_number);
|
2003-07-02 00:45:22 +02:00
|
|
|
push_warning(join->thd, MYSQL_ERROR::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
|
|
|
/*
|
2005-03-30 09:07:08 +02:00
|
|
|
as far as we moved content to upper level, field which depend of
|
2003-07-02 12:12:18 +02:00
|
|
|
'upper' select is not really dependent => we remove this dependence
|
|
|
|
*/
|
|
|
|
substitution->walk(&Item::remove_dependence_processor,
|
|
|
|
(byte *) select_lex->outer_select());
|
2005-03-30 09:07:08 +02:00
|
|
|
/* SELECT without FROM clause can't have WHERE or HAVING clause */
|
|
|
|
DBUG_ASSERT(join->conds == 0 && join->having == 0);
|
2003-08-28 12:21:30 +02:00
|
|
|
return RES_REDUCE;
|
2002-12-26 00:28:59 +01:00
|
|
|
}
|
2003-08-28 12:21:30 +02:00
|
|
|
return RES_OK;
|
2002-12-26 00:28:59 +01: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);
|
2002-06-19 16:52:44 +02:00
|
|
|
}
|
|
|
|
|
2002-12-19 20:15:09 +01:00
|
|
|
enum Item_result Item_singlerow_subselect::result_type() const
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
2002-12-19 06:38:32 +01:00
|
|
|
return engine->type();
|
|
|
|
}
|
|
|
|
|
2002-12-19 20:15:09 +01:00
|
|
|
void 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)
|
|
|
|
{
|
|
|
|
engine->fix_length_and_dec(row= &value);
|
|
|
|
}
|
|
|
|
else
|
2002-12-19 06:38:32 +01:00
|
|
|
{
|
2003-11-28 11:18:13 +01:00
|
|
|
if (!(row= (Item_cache**) sql_alloc(sizeof(Item_cache*)*max_columns)))
|
2002-12-19 06:38:33 +01:00
|
|
|
return;
|
|
|
|
engine->fix_length_and_dec(row);
|
|
|
|
value= *row;
|
2002-12-19 06:38:32 +01:00
|
|
|
}
|
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();
|
2002-09-28 17:34:56 +02:00
|
|
|
}
|
|
|
|
|
2002-12-19 20:15:09 +01:00
|
|
|
uint Item_singlerow_subselect::cols()
|
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
|
|
|
{
|
2003-05-28 15:52:56 +02:00
|
|
|
exec();
|
2002-06-19 16:52:44 +02:00
|
|
|
}
|
|
|
|
|
2004-07-04 07:46:28 +02:00
|
|
|
double Item_singlerow_subselect::val()
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-28 15:52:56 +02:00
|
|
|
if (!exec() && !value->null_value)
|
2002-12-19 06:38:32 +01:00
|
|
|
{
|
|
|
|
null_value= 0;
|
|
|
|
return value->val();
|
|
|
|
}
|
|
|
|
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);
|
2003-05-28 15:52:56 +02:00
|
|
|
if (!exec() && !value->null_value)
|
2002-12-19 06:38:32 +01:00
|
|
|
{
|
|
|
|
null_value= 0;
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2004-07-04 07:46:28 +02:00
|
|
|
String *Item_singlerow_subselect::val_str (String *str)
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
2003-05-28 15:52:56 +02:00
|
|
|
if (!exec() && !value->null_value)
|
2002-12-19 06:38:32 +01:00
|
|
|
{
|
|
|
|
null_value= 0;
|
|
|
|
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
|
|
|
|
2003-10-02 21:19:41 +02:00
|
|
|
Item_exists_subselect::Item_exists_subselect(st_select_lex *select_lex):
|
2002-10-08 22:49:59 +02:00
|
|
|
Item_subselect()
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
2002-10-27 22:27:00 +01:00
|
|
|
DBUG_ENTER("Item_exists_subselect::Item_exists_subselect");
|
2003-10-02 21:19:41 +02:00
|
|
|
init(select_lex, new select_exists_subselect(this));
|
2002-06-19 16:52:44 +02:00
|
|
|
max_columns= UINT_MAX;
|
|
|
|
null_value= 0; //can't be NULL
|
|
|
|
maybe_null= 0; //can't be NULL
|
|
|
|
value= 0;
|
2002-10-27 22:27:00 +01:00
|
|
|
// We need only 1 row to determinate existence
|
|
|
|
select_lex->master_unit()->global_parameters->select_limit= 1;
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
|
|
|
void Item_exists_subselect::print(String *str)
|
|
|
|
{
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append("exists", 6);
|
2003-10-16 14:54:47 +02:00
|
|
|
Item_subselect::print(str);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
bool Item_in_subselect::test_limit(SELECT_LEX_UNIT *unit)
|
|
|
|
{
|
2003-07-03 01:30:52 +02:00
|
|
|
if (unit->fake_select_lex &&
|
|
|
|
unit->fake_select_lex->test_limit())
|
2003-05-14 20:51:33 +02:00
|
|
|
return(1);
|
2003-07-02 00:45:22 +02:00
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
SELECT_LEX *sl= unit->first_select();
|
|
|
|
for (; sl; sl= sl->next_select())
|
|
|
|
{
|
2003-07-02 00:45:22 +02:00
|
|
|
if (sl->test_limit())
|
2003-05-14 20:51:33 +02:00
|
|
|
return(1);
|
|
|
|
}
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
2003-10-02 21:19:41 +02:00
|
|
|
Item_in_subselect::Item_in_subselect(Item * left_exp,
|
2002-10-27 22:27:00 +01:00
|
|
|
st_select_lex *select_lex):
|
2005-03-10 13:01:22 +01:00
|
|
|
Item_exists_subselect(), optimizer(0), transformed(0), upper_item(0)
|
2002-10-27 22:27:00 +01:00
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_in_subselect::Item_in_subselect");
|
2002-11-07 22:45:19 +01:00
|
|
|
left_expr= left_exp;
|
2003-10-02 21:19:41 +02:00
|
|
|
init(select_lex, new select_exists_subselect(this));
|
2002-11-07 22:45:19 +01:00
|
|
|
max_columns= UINT_MAX;
|
2002-12-06 20:55:53 +01:00
|
|
|
maybe_null= 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-11-07 22:45:19 +01:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
2003-10-02 21:19:41 +02:00
|
|
|
Item_allany_subselect::Item_allany_subselect(Item * left_exp,
|
2003-11-03 11:28:36 +01:00
|
|
|
Comp_creator *fn,
|
2003-10-16 14:54:47 +02:00
|
|
|
st_select_lex *select_lex,
|
|
|
|
bool all_arg)
|
|
|
|
:Item_in_subselect(), all(all_arg)
|
2002-11-07 22:45:19 +01:00
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_in_subselect::Item_in_subselect");
|
|
|
|
left_expr= left_exp;
|
2003-05-14 20:51:33 +02:00
|
|
|
func= fn;
|
2003-10-02 21:19:41 +02:00
|
|
|
init(select_lex, new select_exists_subselect(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
|
|
|
|
2002-09-28 17:34:56 +02:00
|
|
|
void Item_exists_subselect::fix_length_and_dec()
|
|
|
|
{
|
2002-12-19 06:38:33 +01:00
|
|
|
decimals= 0;
|
2002-12-06 20:55:53 +01:00
|
|
|
max_length= 1;
|
2002-12-19 20:15:09 +01:00
|
|
|
max_columns= engine->cols();
|
2002-09-28 17:34:56 +02:00
|
|
|
}
|
|
|
|
|
2003-12-19 15:25:50 +01:00
|
|
|
double Item_exists_subselect::val()
|
2002-06-19 16:52:44 +02:00
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-28 15:52:56 +02:00
|
|
|
if (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);
|
2003-05-28 15:52:56 +02:00
|
|
|
if (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;
|
|
|
|
}
|
|
|
|
|
|
|
|
String *Item_exists_subselect::val_str(String *str)
|
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-28 15:52:56 +02:00
|
|
|
if (exec())
|
2002-10-07 21:21:17 +02:00
|
|
|
{
|
2002-12-06 20:55:53 +01:00
|
|
|
reset();
|
|
|
|
return 0;
|
|
|
|
}
|
2003-12-19 15:25:50 +01:00
|
|
|
str->set(value,&my_charset_bin);
|
2002-12-06 20:55:53 +01:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2005-03-30 09:07:08 +02:00
|
|
|
|
2003-12-19 15:25:50 +01:00
|
|
|
double Item_in_subselect::val()
|
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);
|
2003-05-28 15:52:56 +02:00
|
|
|
if (exec())
|
2002-12-06 20:55:53 +01:00
|
|
|
{
|
|
|
|
reset();
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (was_null && !value)
|
|
|
|
null_value= 1;
|
|
|
|
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
|
|
|
{
|
2004-03-18 14:14:36 +01:00
|
|
|
DBUG_ASSERT(fixed == 1);
|
2003-05-28 15:52:56 +02:00
|
|
|
if (exec())
|
2002-12-06 20:55:53 +01:00
|
|
|
{
|
|
|
|
reset();
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (was_null && !value)
|
|
|
|
null_value= 1;
|
|
|
|
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);
|
2003-05-28 15:52:56 +02:00
|
|
|
if (exec())
|
2002-12-06 20:55:53 +01:00
|
|
|
{
|
|
|
|
reset();
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (was_null && !value)
|
|
|
|
{
|
|
|
|
null_value= 1;
|
2002-06-19 16:52:44 +02:00
|
|
|
return 0;
|
2002-10-07 21:21:17 +02:00
|
|
|
}
|
2003-12-19 15:25:50 +01:00
|
|
|
str->set(value, &my_charset_bin);
|
2002-06-19 16:52:44 +02:00
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2002-11-07 22:45:19 +01:00
|
|
|
|
2004-10-04 16:58:06 +02:00
|
|
|
/* Rewrite a single-column IN/ALL/ANY subselect. */
|
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
Item_subselect::trans_res
|
2003-07-02 00:45:22 +02:00
|
|
|
Item_in_subselect::single_value_transformer(JOIN *join,
|
2003-11-03 11:28:36 +01:00
|
|
|
Comp_creator *func)
|
2002-11-07 22:45:19 +01:00
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_in_subselect::single_value_transformer");
|
2003-01-25 01:25:52 +01:00
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
SELECT_LEX *select_lex= join->select_lex;
|
2002-12-28 00:01:05 +01: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 * ...)
|
|
|
|
*/
|
2003-10-23 22:54:21 +02:00
|
|
|
if (select_lex->item_list.elements > 1)
|
|
|
|
{
|
|
|
|
my_error(ER_OPERAND_COLUMNS, MYF(0), 1);
|
2005-03-10 13:01:22 +01:00
|
|
|
DBUG_RETURN(RES_ERROR);
|
2003-10-23 22:54:21 +02:00
|
|
|
}
|
|
|
|
|
2004-10-04 16:58:06 +02:00
|
|
|
/*
|
|
|
|
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.
|
|
|
|
*/
|
2004-11-18 17:10:07 +01:00
|
|
|
if ((abort_on_null || (upper_item && upper_item->top_level())) &&
|
2003-11-17 19:53:40 +01:00
|
|
|
!select_lex->master_unit()->uncacheable && !func->eqne_op())
|
2003-07-24 14:26:21 +02:00
|
|
|
{
|
2003-10-27 00:01:27 +01:00
|
|
|
if (substitution)
|
|
|
|
{
|
|
|
|
// It is second (third, ...) SELECT of UNION => All is done
|
2005-03-10 13:01:22 +01:00
|
|
|
DBUG_RETURN(RES_OK);
|
2003-10-27 00:01:27 +01:00
|
|
|
}
|
|
|
|
|
2003-08-12 11:38:03 +02:00
|
|
|
Item *subs;
|
|
|
|
if (!select_lex->group_list.elements &&
|
2005-03-30 09:07:08 +02:00
|
|
|
!select_lex->having &&
|
2003-10-27 00:01:27 +01:00
|
|
|
!select_lex->with_sum_func &&
|
|
|
|
!(select_lex->next_select()))
|
2003-07-24 14:26:21 +02:00
|
|
|
{
|
2004-11-18 17:10:07 +01:00
|
|
|
Item_sum_hybrid *item;
|
2003-11-03 11:28:36 +01:00
|
|
|
if (func->l_op())
|
2003-08-12 11:38:03 +02:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
(ALL && (> || =>)) || (ANY && (< || =<))
|
|
|
|
for ALL condition is inverted
|
|
|
|
*/
|
|
|
|
item= new Item_sum_max(*select_lex->ref_pointer_array);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
(ALL && (< || =<)) || (ANY && (> || =>))
|
|
|
|
for ALL condition is inverted
|
|
|
|
*/
|
|
|
|
item= new Item_sum_min(*select_lex->ref_pointer_array);
|
|
|
|
}
|
2004-11-18 17:10:07 +01:00
|
|
|
if (upper_item)
|
|
|
|
upper_item->set_sum_test(item);
|
2003-08-12 11:38:03 +02:00
|
|
|
*select_lex->ref_pointer_array= item;
|
2004-05-07 22:06:11 +02:00
|
|
|
{
|
|
|
|
List_iterator<Item> it(select_lex->item_list);
|
|
|
|
it++;
|
|
|
|
it.replace(item);
|
|
|
|
}
|
2004-03-09 11:52:25 +01:00
|
|
|
|
2004-05-07 22:06:11 +02:00
|
|
|
/*
|
|
|
|
Item_sum_(max|min) can't substitute other item => we can use 0 as
|
|
|
|
reference
|
|
|
|
*/
|
|
|
|
if (item->fix_fields(thd, join->tables_list, 0))
|
2005-03-10 13:01:22 +01:00
|
|
|
DBUG_RETURN(RES_ERROR);
|
2004-05-07 22:06:11 +02:00
|
|
|
/* we added aggregate function => we have to change statistic */
|
|
|
|
count_field_types(&join->tmp_table_param, join->all_fields, 0);
|
2004-02-08 19:14:13 +01:00
|
|
|
|
2003-10-07 12:31:44 +02:00
|
|
|
subs= new Item_singlerow_subselect(select_lex);
|
2003-07-24 14:26:21 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2004-11-18 17:10:07 +01:00
|
|
|
Item_maxmin_subselect *item;
|
2003-08-12 11:38:03 +02:00
|
|
|
// remove LIMIT placed by ALL/ANY subquery
|
|
|
|
select_lex->master_unit()->global_parameters->select_limit=
|
|
|
|
HA_POS_ERROR;
|
2004-11-18 17:10:07 +01:00
|
|
|
subs= item= new Item_maxmin_subselect(this, select_lex, func->l_op());
|
|
|
|
if (upper_item)
|
|
|
|
upper_item->set_sub_test(item);
|
2003-07-24 14:26:21 +02:00
|
|
|
}
|
2005-03-10 13:01:22 +01:00
|
|
|
/* fix fields is already called for left expression */
|
2003-11-03 11:28:36 +01:00
|
|
|
substitution= func->create(left_expr, subs);
|
2005-03-10 13:01:22 +01:00
|
|
|
DBUG_RETURN(RES_OK);
|
2003-07-24 14:26:21 +02:00
|
|
|
}
|
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
if (!substitution)
|
2002-10-27 22:27:00 +01:00
|
|
|
{
|
2003-05-14 20:51:33 +02:00
|
|
|
//first call for this unit
|
|
|
|
SELECT_LEX_UNIT *unit= select_lex->master_unit();
|
2005-03-10 13:01:22 +01:00
|
|
|
substitution= optimizer;
|
2003-05-14 20:51:33 +02:00
|
|
|
|
2004-02-08 19:14:13 +01:00
|
|
|
SELECT_LEX *current= thd->lex->current_select, *up;
|
2003-07-03 01:30:52 +02:00
|
|
|
|
2004-02-08 19:14:13 +01:00
|
|
|
thd->lex->current_select= up= current->return_after_parsing();
|
2003-05-14 20:51:33 +02:00
|
|
|
//optimizer never use Item **ref => we can pass 0 as parameter
|
2004-02-08 19:14:13 +01:00
|
|
|
if (!optimizer || optimizer->fix_left(thd, up->get_table_list(), 0))
|
2002-12-23 17:25:25 +01:00
|
|
|
{
|
2004-02-08 19:14:13 +01:00
|
|
|
thd->lex->current_select= current;
|
2005-03-10 13:01:22 +01:00
|
|
|
DBUG_RETURN(RES_ERROR);
|
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
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
/*
|
|
|
|
As far as Item_ref_in_optimizer do not substitude itself on fix_fields
|
|
|
|
we can use same item for all selects.
|
|
|
|
*/
|
2004-12-11 16:13:19 +01:00
|
|
|
expr= new Item_direct_ref((Item**)optimizer->get_cache(),
|
|
|
|
(char *)"<no matter>",
|
|
|
|
(char *)in_left_expr_name);
|
2003-05-14 20:51:33 +02:00
|
|
|
|
2003-11-17 19:53:40 +01:00
|
|
|
unit->uncacheable|= UNCACHEABLE_DEPENDENT;
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
2002-10-31 01:11:59 +01:00
|
|
|
|
2003-11-17 19:53:40 +01:00
|
|
|
select_lex->uncacheable|= UNCACHEABLE_DEPENDENT;
|
2003-05-14 20:51:33 +02:00
|
|
|
Item *item;
|
2003-07-02 00:45:22 +02:00
|
|
|
|
|
|
|
item= (Item*) select_lex->item_list.head();
|
2004-10-04 16:58:06 +02:00
|
|
|
/*
|
|
|
|
Add the left part of a subselect to a WHERE or HAVING clause of
|
|
|
|
the right part, e.g. SELECT 1 IN (SELECT a FROM t1) =>
|
|
|
|
SELECT Item_in_optimizer(1, SELECT a FROM t1 WHERE a=1)
|
|
|
|
HAVING is used only if the right part contains a SUM function, a GROUP
|
|
|
|
BY or a HAVING clause.
|
|
|
|
*/
|
2003-05-14 20:51:33 +02:00
|
|
|
if (join->having || select_lex->with_sum_func ||
|
|
|
|
select_lex->group_list.elements)
|
|
|
|
{
|
2003-11-03 11:28:36 +01:00
|
|
|
item= func->create(expr,
|
|
|
|
new Item_ref_null_helper(this,
|
|
|
|
select_lex->ref_pointer_array,
|
|
|
|
(char *)"<ref>",
|
|
|
|
this->full_name()));
|
2004-02-18 21:14:41 +01:00
|
|
|
/*
|
|
|
|
AND and comparison functions can't be changed during fix_fields()
|
|
|
|
we can assign select_lex->having here, and pass 0 as last
|
|
|
|
argument (reference) to fix_fields()
|
|
|
|
*/
|
2004-02-08 19:14:13 +01:00
|
|
|
select_lex->having= join->having= and_items(join->having, item);
|
2003-05-14 20:51:33 +02:00
|
|
|
select_lex->having_fix_field= 1;
|
2004-02-18 21:14:41 +01:00
|
|
|
if (join->having->fix_fields(thd, join->tables_list, 0))
|
2002-10-31 01:11:59 +01:00
|
|
|
{
|
2003-05-14 20:51:33 +02:00
|
|
|
select_lex->having_fix_field= 0;
|
2005-03-10 13:01:22 +01:00
|
|
|
DBUG_RETURN(RES_ERROR);
|
2002-10-31 01:11:59 +01:00
|
|
|
}
|
2003-05-14 20:51:33 +02:00
|
|
|
select_lex->having_fix_field= 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
select_lex->item_list.empty();
|
|
|
|
select_lex->item_list.push_back(new Item_int("Not_used",
|
|
|
|
(longlong) 1, 21));
|
|
|
|
select_lex->ref_pointer_array[0]= select_lex->item_list.head();
|
|
|
|
if (select_lex->table_list.elements)
|
2002-10-31 01:11:59 +01:00
|
|
|
{
|
2004-02-02 01:23:53 +01:00
|
|
|
Item *having= item, *orig_item= item;
|
2003-11-03 11:28:36 +01:00
|
|
|
item= func->create(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
|
|
|
{
|
2003-05-14 20:51:33 +02:00
|
|
|
having= new Item_is_not_null_test(this, having);
|
2004-02-18 21:14:41 +01:00
|
|
|
/*
|
|
|
|
Item_is_not_null_test can't be changed during fix_fields()
|
|
|
|
we can assign select_lex->having here, and pass 0 as last
|
|
|
|
argument (reference) to fix_fields()
|
|
|
|
*/
|
2004-02-08 19:14:13 +01:00
|
|
|
select_lex->having=
|
|
|
|
join->having= (join->having ?
|
|
|
|
new Item_cond_and(having, join->having) :
|
|
|
|
having);
|
2003-05-14 20:51:33 +02:00
|
|
|
select_lex->having_fix_field= 1;
|
2004-02-18 21:14:41 +01:00
|
|
|
if (join->having->fix_fields(thd, join->tables_list, 0))
|
2002-11-28 18:29:26 +01:00
|
|
|
{
|
2003-05-14 20:51:33 +02:00
|
|
|
select_lex->having_fix_field= 0;
|
2005-03-10 13:01:22 +01:00
|
|
|
DBUG_RETURN(RES_ERROR);
|
2002-11-28 18:29:26 +01:00
|
|
|
}
|
2003-05-14 20:51:33 +02:00
|
|
|
select_lex->having_fix_field= 0;
|
|
|
|
item= new Item_cond_or(item,
|
2004-02-02 01:23:53 +01:00
|
|
|
new Item_func_isnull(orig_item));
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
2003-09-08 20:58:09 +02:00
|
|
|
item->name= (char *)in_additional_cond;
|
2004-02-18 21:14:41 +01:00
|
|
|
/*
|
|
|
|
AND can't be changed during fix_fields()
|
|
|
|
we can assign select_lex->having here, and pass 0 as last
|
|
|
|
argument (reference) to fix_fields()
|
|
|
|
*/
|
2004-02-08 19:14:13 +01:00
|
|
|
select_lex->where= join->conds= and_items(join->conds, item);
|
2005-08-13 06:45:14 +02:00
|
|
|
select_lex->where->top_level_item();
|
2004-02-18 21:14:41 +01:00
|
|
|
if (join->conds->fix_fields(thd, join->tables_list, 0))
|
2005-03-10 13:01:22 +01:00
|
|
|
DBUG_RETURN(RES_ERROR);
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (select_lex->master_unit()->first_select()->next_select())
|
|
|
|
{
|
2004-02-18 21:14:41 +01:00
|
|
|
/*
|
|
|
|
comparison functions can't be changed during fix_fields()
|
|
|
|
we can assign select_lex->having here, and pass 0 as last
|
|
|
|
argument (reference) to fix_fields()
|
|
|
|
*/
|
2005-02-09 20:08:08 +01:00
|
|
|
select_lex->having=
|
|
|
|
join->having=
|
|
|
|
func->create(expr,
|
|
|
|
new Item_null_helper(this, item,
|
2004-02-08 19:14:13 +01:00
|
|
|
(char *)"<no matter>",
|
|
|
|
(char *)"<result>"));
|
2003-05-14 20:51:33 +02:00
|
|
|
select_lex->having_fix_field= 1;
|
2004-02-08 19:14:13 +01:00
|
|
|
if (join->having->fix_fields(thd, join->tables_list,
|
2004-02-18 21:14:41 +01:00
|
|
|
0))
|
2003-05-14 20:51:33 +02:00
|
|
|
{
|
|
|
|
select_lex->having_fix_field= 0;
|
2005-03-10 13:01:22 +01:00
|
|
|
DBUG_RETURN(RES_ERROR);
|
2002-11-28 18:29:26 +01:00
|
|
|
}
|
2003-05-14 20:51:33 +02:00
|
|
|
select_lex->having_fix_field= 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// it is single select without tables => possible optimization
|
2003-11-03 11:28:36 +01:00
|
|
|
item= func->create(left_expr, item);
|
2004-07-04 07:46:28 +02:00
|
|
|
// fix_field of item will be done in time of substituting
|
2003-05-14 20:51:33 +02:00
|
|
|
substitution= item;
|
|
|
|
have_to_be_excluded= 1;
|
2004-02-08 19:14:13 +01:00
|
|
|
if (thd->lex->describe)
|
2002-11-28 18:29:26 +01:00
|
|
|
{
|
2003-05-14 20:51:33 +02:00
|
|
|
char warn_buff[MYSQL_ERRMSG_SIZE];
|
|
|
|
sprintf(warn_buff, ER(ER_SELECT_REDUCED), select_lex->select_number);
|
2004-02-08 19:14:13 +01:00
|
|
|
push_warning(thd, MYSQL_ERROR::WARN_LEVEL_NOTE,
|
2003-05-14 20:51:33 +02:00
|
|
|
ER_SELECT_REDUCED, warn_buff);
|
2002-11-28 18:29:26 +01:00
|
|
|
}
|
2003-08-28 12:21:30 +02:00
|
|
|
DBUG_RETURN(RES_REDUCE);
|
2002-11-28 18:29:26 +01:00
|
|
|
}
|
2002-10-31 01:11:59 +01:00
|
|
|
}
|
2002-10-27 22:27:00 +01:00
|
|
|
}
|
2004-02-08 19:14:13 +01:00
|
|
|
|
2003-08-28 12:21:30 +02:00
|
|
|
DBUG_RETURN(RES_OK);
|
2002-10-27 22:27:00 +01:00
|
|
|
}
|
2002-09-03 08:50:36 +02:00
|
|
|
|
2003-11-28 11:18:13 +01:00
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
Item_subselect::trans_res
|
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-11-08 00:13:54 +01:00
|
|
|
SELECT_LEX *select_lex= join->select_lex;
|
2005-08-13 06:45:14 +02:00
|
|
|
Item *having_item= 0;
|
|
|
|
uint cols_num= left_expr->cols();
|
|
|
|
bool is_having_used= (join->having || select_lex->with_sum_func ||
|
|
|
|
select_lex->group_list.first ||
|
|
|
|
!select_lex->table_list.elements);
|
|
|
|
DBUG_ENTER("Item_in_subselect::row_value_transformer");
|
2004-02-08 19:14:13 +01:00
|
|
|
|
2003-10-23 22:54:21 +02:00
|
|
|
if (select_lex->item_list.elements != left_expr->cols())
|
|
|
|
{
|
|
|
|
my_error(ER_OPERAND_COLUMNS, MYF(0), left_expr->cols());
|
2005-03-10 13:01:22 +01:00
|
|
|
DBUG_RETURN(RES_ERROR);
|
2003-10-23 22:54:21 +02:00
|
|
|
}
|
|
|
|
|
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
|
|
|
|
SELECT_LEX_UNIT *unit= select_lex->master_unit();
|
2005-03-10 13:01:22 +01:00
|
|
|
substitution= optimizer;
|
2003-05-14 20:51:33 +02:00
|
|
|
|
2004-02-08 19:14:13 +01:00
|
|
|
SELECT_LEX *current= thd->lex->current_select, *up;
|
|
|
|
thd->lex->current_select= up= current->return_after_parsing();
|
2003-05-14 20:51:33 +02:00
|
|
|
//optimizer never use Item **ref => we can pass 0 as parameter
|
2004-02-08 19:14:13 +01:00
|
|
|
if (!optimizer || optimizer->fix_left(thd, up->get_table_list(), 0))
|
2002-12-25 11:03:08 +01:00
|
|
|
{
|
2004-02-08 19:14:13 +01:00
|
|
|
thd->lex->current_select= current;
|
2005-03-10 13:01:22 +01:00
|
|
|
DBUG_RETURN(RES_ERROR);
|
2002-12-25 11:03:08 +01:00
|
|
|
}
|
2004-02-12 02:10:26 +01:00
|
|
|
|
|
|
|
// we will refer to apper level cache array => we have to save it in PS
|
|
|
|
optimizer->keep_top_level_cache();
|
|
|
|
|
2004-02-08 19:14:13 +01:00
|
|
|
thd->lex->current_select= current;
|
2003-11-17 19:53:40 +01:00
|
|
|
unit->uncacheable|= UNCACHEABLE_DEPENDENT;
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
|
|
|
|
2003-11-17 19:53:40 +01:00
|
|
|
select_lex->uncacheable|= UNCACHEABLE_DEPENDENT;
|
2005-08-13 06:45:14 +02:00
|
|
|
if (is_having_used)
|
2004-02-08 19:14:13 +01:00
|
|
|
{
|
2005-08-13 06:45:14 +02:00
|
|
|
/*
|
|
|
|
(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
|
|
|
|
*/
|
|
|
|
Item *item_having_part2= 0;
|
|
|
|
for (uint i= 0; i < cols_num; i++)
|
2004-02-08 19:14:13 +01:00
|
|
|
{
|
2005-01-24 14:56:57 +01:00
|
|
|
DBUG_ASSERT(left_expr->fixed && select_lex->ref_pointer_array[i]->fixed);
|
|
|
|
if (select_lex->ref_pointer_array[i]->
|
|
|
|
check_cols(left_expr->el(i)->cols()))
|
2005-03-10 13:01:22 +01:00
|
|
|
DBUG_RETURN(RES_ERROR);
|
2005-08-13 06:45:14 +02:00
|
|
|
Item *item_eq=
|
|
|
|
new Item_func_eq(new
|
|
|
|
Item_direct_ref((*optimizer->get_cache())->
|
|
|
|
addr(i),
|
|
|
|
(char *)"<no matter>",
|
|
|
|
(char *)in_left_expr_name),
|
|
|
|
new
|
|
|
|
Item_direct_ref(select_lex->ref_pointer_array + i,
|
|
|
|
(char *)"<no matter>",
|
|
|
|
(char *)"<list ref>")
|
|
|
|
);
|
|
|
|
Item *item_isnull=
|
|
|
|
new Item_func_isnull(new
|
|
|
|
Item_direct_ref( select_lex->
|
|
|
|
ref_pointer_array+i,
|
|
|
|
(char *)"<no matter>",
|
|
|
|
(char *)"<list ref>")
|
|
|
|
);
|
|
|
|
having_item=
|
|
|
|
and_items(having_item,
|
|
|
|
new Item_cond_or(item_eq, item_isnull));
|
|
|
|
item_having_part2=
|
|
|
|
and_items(item_having_part2,
|
|
|
|
new
|
|
|
|
Item_is_not_null_test(this,
|
|
|
|
new
|
|
|
|
Item_direct_ref(select_lex->
|
|
|
|
ref_pointer_array + i,
|
|
|
|
(char *)"<no matter>",
|
|
|
|
(char *)"<list ref>")
|
|
|
|
)
|
|
|
|
);
|
|
|
|
item_having_part2->top_level_item();
|
2004-02-08 19:14:13 +01:00
|
|
|
}
|
2005-08-13 06:45:14 +02:00
|
|
|
having_item= and_items(having_item, item_having_part2);
|
|
|
|
having_item->top_level_item();
|
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
|
|
|
/*
|
|
|
|
(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 register 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)
|
|
|
|
*/
|
|
|
|
Item *where_item= 0;
|
|
|
|
for (uint i= 0; i < cols_num; i++)
|
|
|
|
{
|
|
|
|
Item *item, *item_isnull;
|
|
|
|
DBUG_ASSERT(left_expr->fixed && select_lex->ref_pointer_array[i]->fixed);
|
|
|
|
if (select_lex->ref_pointer_array[i]->
|
|
|
|
check_cols(left_expr->el(i)->cols()))
|
|
|
|
DBUG_RETURN(RES_ERROR);
|
|
|
|
item=
|
|
|
|
new Item_func_eq(new
|
|
|
|
Item_direct_ref((*optimizer->get_cache())->
|
|
|
|
addr(i),
|
|
|
|
(char *)"<no matter>",
|
|
|
|
(char *)in_left_expr_name),
|
|
|
|
new
|
|
|
|
Item_direct_ref( select_lex->
|
|
|
|
ref_pointer_array+i,
|
|
|
|
(char *)"<no matter>",
|
|
|
|
(char *)"<list ref>")
|
|
|
|
);
|
|
|
|
if (!abort_on_null)
|
|
|
|
{
|
|
|
|
having_item=
|
|
|
|
and_items(having_item,
|
|
|
|
new
|
|
|
|
Item_is_not_null_test(this,
|
|
|
|
new
|
|
|
|
Item_direct_ref(select_lex->
|
|
|
|
ref_pointer_array + i,
|
|
|
|
(char *)"<no matter>",
|
|
|
|
(char *)"<list ref>")
|
|
|
|
)
|
|
|
|
);
|
|
|
|
item_isnull= new
|
|
|
|
Item_func_isnull(new
|
|
|
|
Item_direct_ref( select_lex->
|
|
|
|
ref_pointer_array+i,
|
|
|
|
(char *)"<no matter>",
|
|
|
|
(char *)"<list ref>")
|
|
|
|
);
|
|
|
|
|
|
|
|
item= new Item_cond_or(item, item_isnull);
|
|
|
|
}
|
|
|
|
|
|
|
|
where_item= and_items(where_item, item);
|
|
|
|
}
|
2004-02-18 21:14:41 +01:00
|
|
|
/*
|
|
|
|
AND can't be changed during fix_fields()
|
2005-08-13 06:45:14 +02:00
|
|
|
we can assign select_lex->where here, and pass 0 as last
|
2004-02-18 21:14:41 +01:00
|
|
|
argument (reference) to fix_fields()
|
|
|
|
*/
|
2005-08-13 06:45:14 +02:00
|
|
|
select_lex->where= join->conds= and_items(join->conds, where_item);
|
|
|
|
select_lex->where->top_level_item();
|
|
|
|
if (join->conds->fix_fields(thd, join->tables_list, 0))
|
2005-03-10 13:01:22 +01:00
|
|
|
DBUG_RETURN(RES_ERROR);
|
2002-12-19 20:15:09 +01:00
|
|
|
}
|
2005-08-13 06:45:14 +02:00
|
|
|
if (having_item)
|
2003-05-14 20:51:33 +02:00
|
|
|
{
|
2005-08-13 06:45:14 +02:00
|
|
|
bool res;
|
|
|
|
select_lex->having= join->having= and_items(join->having, having_item);
|
|
|
|
select_lex->having->top_level_item();
|
2004-02-18 21:14:41 +01:00
|
|
|
/*
|
|
|
|
AND can't be changed during fix_fields()
|
|
|
|
we can assign select_lex->having here, and pass 0 as last
|
|
|
|
argument (reference) to fix_fields()
|
|
|
|
*/
|
2005-08-13 06:45:14 +02:00
|
|
|
select_lex->having_fix_field= 1;
|
|
|
|
res= join->having->fix_fields(thd, join->tables_list, 0);
|
|
|
|
select_lex->having_fix_field= 0;
|
|
|
|
if (res)
|
|
|
|
{
|
2005-03-10 13:01:22 +01:00
|
|
|
DBUG_RETURN(RES_ERROR);
|
2005-08-13 06:45:14 +02:00
|
|
|
}
|
2003-05-14 20:51:33 +02:00
|
|
|
}
|
2003-08-28 12:21:30 +02:00
|
|
|
DBUG_RETURN(RES_OK);
|
2002-12-19 20:15:09 +01:00
|
|
|
}
|
|
|
|
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
Item_subselect::trans_res
|
2003-07-02 00:45:22 +02:00
|
|
|
Item_in_subselect::select_transformer(JOIN *join)
|
2002-11-07 22:45:19 +01:00
|
|
|
{
|
2005-03-10 13:01:22 +01:00
|
|
|
return select_in_like_transformer(join, &eq_creator);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Prepare IN/ALL/ANY/SOME subquery transformation and call appropriate
|
|
|
|
transformation function
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
Item_in_subselect::select_in_like_transformer()
|
|
|
|
join JOIN object of transforming subquery
|
|
|
|
func creator of condition function of subquery
|
|
|
|
|
|
|
|
DESCRIPTION
|
|
|
|
To decide which transformation procedure (scalar or row) applicable here
|
|
|
|
we have to call fix_fields() for left expression to be able to call
|
|
|
|
cols() method on it. Also this method make arena management for
|
|
|
|
underlying transformation methods.
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
RES_OK OK
|
|
|
|
RES_REDUCE OK, and current subquery was reduced during transformation
|
|
|
|
RES_ERROR Error
|
|
|
|
*/
|
|
|
|
|
|
|
|
Item_subselect::trans_res
|
|
|
|
Item_in_subselect::select_in_like_transformer(JOIN *join, Comp_creator *func)
|
|
|
|
{
|
|
|
|
Item_arena *arena, backup;
|
|
|
|
SELECT_LEX *current= thd->lex->current_select, *up;
|
|
|
|
const char *save_where= thd->where;
|
|
|
|
Item_subselect::trans_res res= RES_ERROR;
|
|
|
|
bool result;
|
|
|
|
|
|
|
|
DBUG_ENTER("Item_in_subselect::select_in_like_transformer");
|
2005-03-30 09:07:08 +02:00
|
|
|
|
2005-03-10 13:01:22 +01:00
|
|
|
if (changed)
|
|
|
|
{
|
|
|
|
DBUG_RETURN(RES_OK);
|
|
|
|
}
|
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
if (!optimizer)
|
|
|
|
{
|
|
|
|
arena= thd->change_arena_if_needed(&backup);
|
|
|
|
result= (!(optimizer= new Item_in_optimizer(left_expr, this)));
|
|
|
|
if (arena)
|
|
|
|
thd->restore_backup_item_arena(arena, &backup);
|
|
|
|
if (result)
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
thd->lex->current_select= up= current->return_after_parsing();
|
|
|
|
result= (!left_expr->fixed &&
|
|
|
|
left_expr->fix_fields(thd, up->get_table_list(),
|
|
|
|
optimizer->arguments()));
|
|
|
|
/* fix_fields can change reference to left_expr, we need reassign it */
|
|
|
|
left_expr= optimizer->arguments()[0];
|
|
|
|
|
|
|
|
thd->lex->current_select= current;
|
|
|
|
if (result)
|
|
|
|
goto err;
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
transformed= 1;
|
2005-03-10 13:01:22 +01:00
|
|
|
arena= thd->change_arena_if_needed(&backup);
|
|
|
|
/*
|
|
|
|
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)
|
2005-03-10 13:01:22 +01:00
|
|
|
res= single_value_transformer(join, func);
|
|
|
|
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)
|
|
|
|
thd->restore_backup_item_arena(arena, &backup);
|
2005-03-10 13:01:22 +01:00
|
|
|
my_error(ER_OPERAND_COLUMNS, MYF(0), 1);
|
|
|
|
DBUG_RETURN(RES_ERROR);
|
|
|
|
}
|
|
|
|
res= row_value_transformer(join);
|
|
|
|
}
|
|
|
|
if (arena)
|
|
|
|
thd->restore_backup_item_arena(arena, &backup);
|
|
|
|
err:
|
|
|
|
thd->where= save_where;
|
|
|
|
DBUG_RETURN(res);
|
2002-11-07 22:45:19 +01:00
|
|
|
}
|
|
|
|
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
void Item_in_subselect::print(String *str)
|
|
|
|
{
|
|
|
|
if (transformed)
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append("<exists>", 8);
|
2003-10-16 14:54:47 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
left_expr->print(str);
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append(" in ", 4);
|
2003-10-16 14:54:47 +02:00
|
|
|
}
|
|
|
|
Item_subselect::print(str);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-05-14 20:51:33 +02:00
|
|
|
Item_subselect::trans_res
|
2003-07-02 00:45:22 +02:00
|
|
|
Item_allany_subselect::select_transformer(JOIN *join)
|
2002-11-07 22:45:19 +01:00
|
|
|
{
|
2003-10-16 14:54:47 +02:00
|
|
|
transformed= 1;
|
2004-11-18 17:10:07 +01:00
|
|
|
if (upper_item)
|
|
|
|
upper_item->show= 1;
|
2005-03-10 13:01:22 +01:00
|
|
|
return select_in_like_transformer(join, func);
|
2002-11-07 22:45:19 +01:00
|
|
|
}
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
|
|
|
void Item_allany_subselect::print(String *str)
|
|
|
|
{
|
|
|
|
if (transformed)
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append("<exists>", 8);
|
2003-10-16 14:54:47 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
left_expr->print(str);
|
|
|
|
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
|
|
|
}
|
|
|
|
Item_subselect::print(str);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-24 14:26:21 +02:00
|
|
|
subselect_single_select_engine::
|
2003-10-02 21:19:41 +02:00
|
|
|
subselect_single_select_engine(st_select_lex *select,
|
|
|
|
select_subselect *result,
|
|
|
|
Item_subselect *item)
|
|
|
|
:subselect_engine(item, result),
|
|
|
|
prepared(0), optimized(0), executed(0), join(0)
|
2002-09-03 08:50:36 +02:00
|
|
|
{
|
|
|
|
select_lex= select;
|
|
|
|
SELECT_LEX_UNIT *unit= select_lex->master_unit();
|
|
|
|
unit->offset_limit_cnt= unit->global_parameters->offset_limit;
|
|
|
|
unit->select_limit_cnt= unit->global_parameters->select_limit+
|
|
|
|
unit->global_parameters ->offset_limit;
|
|
|
|
if (unit->select_limit_cnt < unit->global_parameters->select_limit)
|
2003-11-28 11:18:13 +01:00
|
|
|
unit->select_limit_cnt= HA_POS_ERROR; // no limit
|
2002-09-03 08:50:36 +02:00
|
|
|
if (unit->select_limit_cnt == HA_POS_ERROR)
|
|
|
|
select_lex->options&= ~OPTION_FOUND_ROWS;
|
2002-11-13 23:26:18 +01:00
|
|
|
unit->item= item;
|
2002-09-03 08:50:36 +02:00
|
|
|
this->select_lex= select_lex;
|
|
|
|
}
|
|
|
|
|
2004-02-08 19:14:13 +01:00
|
|
|
|
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");
|
|
|
|
prepared= optimized= executed= 0;
|
2004-02-12 02:10:26 +01:00
|
|
|
join= 0;
|
2005-01-26 14:27:45 +01:00
|
|
|
result->cleanup();
|
2004-02-08 19:14:13 +01:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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();
|
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_uniquesubquery_engine::cleanup()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("subselect_uniquesubquery_engine::cleanup");
|
2005-01-26 14:27:45 +01:00
|
|
|
/*
|
|
|
|
subselect_uniquesubquery_engine have not 'result' assigbed, so we do not
|
|
|
|
cleanup() it
|
|
|
|
*/
|
2004-02-08 19:14:13 +01:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-02 21:19:41 +02:00
|
|
|
subselect_union_engine::subselect_union_engine(st_select_lex_unit *u,
|
2003-11-28 11:18:13 +01:00
|
|
|
select_subselect *result_arg,
|
|
|
|
Item_subselect *item_arg)
|
|
|
|
:subselect_engine(item_arg, result_arg)
|
2002-09-03 08:50:36 +02:00
|
|
|
{
|
|
|
|
unit= u;
|
2003-11-28 11:18:13 +01:00
|
|
|
if (!result_arg) //out of memory
|
2003-10-02 21:19:41 +02:00
|
|
|
current_thd->fatal_error();
|
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
|
|
|
|
2002-09-03 08:50:36 +02:00
|
|
|
int subselect_single_select_engine::prepare()
|
|
|
|
{
|
2002-10-13 13:25:16 +02:00
|
|
|
if (prepared)
|
|
|
|
return 0;
|
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)
|
|
|
|
{
|
2003-11-28 11:18:13 +01:00
|
|
|
thd->fatal_error(); //out of memory
|
2003-10-02 21:19:41 +02:00
|
|
|
return 1;
|
|
|
|
}
|
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;
|
2003-01-25 12:19:46 +01:00
|
|
|
if (join->prepare(&select_lex->ref_pointer_array,
|
|
|
|
(TABLE_LIST*) select_lex->table_list.first,
|
|
|
|
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,
|
2002-12-28 00:01:05 +01:00
|
|
|
(ORDER*) select_lex->order_list.first,
|
|
|
|
(ORDER*) select_lex->group_list.first,
|
|
|
|
select_lex->having,
|
2004-07-04 07:46:28 +02:00
|
|
|
(ORDER*) 0, 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;
|
|
|
|
}
|
|
|
|
|
|
|
|
int subselect_union_engine::prepare()
|
|
|
|
{
|
2005-03-23 07:36:48 +01:00
|
|
|
return unit->prepare(thd, result, SELECT_NO_UNLOCK, "");
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
|
|
|
|
2003-09-14 08:40:57 +02:00
|
|
|
int subselect_uniquesubquery_engine::prepare()
|
2003-07-07 17:40:19 +02:00
|
|
|
{
|
|
|
|
//this never should be called
|
|
|
|
DBUG_ASSERT(0);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2003-11-23 01:01:15 +01:00
|
|
|
static Item_result set_row(List<Item> &item_list, Item *item,
|
2002-12-27 20:19:25 +01:00
|
|
|
Item_cache **row, bool *maybe_null)
|
2002-09-28 17:34:56 +02:00
|
|
|
{
|
2002-12-19 06:38:33 +01:00
|
|
|
Item_result res_type= STRING_RESULT;
|
|
|
|
Item *sel_item;
|
2003-11-23 01:01:15 +01:00
|
|
|
List_iterator_fast<Item> li(item_list);
|
2002-12-19 06:38:33 +01:00
|
|
|
for (uint i= 0; (sel_item= li++); i++)
|
|
|
|
{
|
|
|
|
item->max_length= sel_item->max_length;
|
|
|
|
res_type= sel_item->result_type();
|
|
|
|
item->decimals= sel_item->decimals;
|
2002-12-27 20:19:25 +01:00
|
|
|
*maybe_null= sel_item->maybe_null;
|
2003-12-25 15:50:22 +01:00
|
|
|
if (!(row[i]= Item_cache::get_cache(res_type)))
|
|
|
|
return STRING_RESULT; // we should return something
|
|
|
|
row[i]->setup(sel_item);
|
2002-12-19 06:38:33 +01:00
|
|
|
}
|
2003-11-23 01:01:15 +01:00
|
|
|
if (item_list.elements > 1)
|
2002-12-19 06:38:33 +01:00
|
|
|
res_type= ROW_RESULT;
|
|
|
|
return res_type;
|
2002-09-28 17:34:56 +02:00
|
|
|
}
|
|
|
|
|
2002-12-19 06:38:33 +01:00
|
|
|
void 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);
|
2003-11-23 01:01:15 +01:00
|
|
|
res_type= set_row(select_lex->item_list, item, row, &maybe_null);
|
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;
|
2002-12-19 06:38:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void subselect_union_engine::fix_length_and_dec(Item_cache **row)
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(row || unit->first_select()->item_list.elements==1);
|
|
|
|
|
|
|
|
if (unit->first_select()->item_list.elements == 1)
|
2003-12-25 15:50:22 +01:00
|
|
|
{
|
2003-11-23 01:01:15 +01:00
|
|
|
res_type= set_row(unit->types, item, row, &maybe_null);
|
2003-12-25 15:50:22 +01:00
|
|
|
item->collation.set(row[0]->collation);
|
|
|
|
}
|
2002-12-19 06:38:33 +01:00
|
|
|
else
|
|
|
|
{
|
2002-12-27 20:19:25 +01:00
|
|
|
bool fake= 0;
|
2003-11-23 01:01:15 +01:00
|
|
|
res_type= set_row(unit->types, item, row, &fake);
|
2002-09-28 17:34:56 +02:00
|
|
|
}
|
|
|
|
}
|
2002-09-03 08:50:36 +02:00
|
|
|
|
2003-09-14 08:40:57 +02:00
|
|
|
void 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);
|
|
|
|
}
|
|
|
|
|
2002-09-03 08:50:36 +02:00
|
|
|
int subselect_single_select_engine::exec()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("subselect_single_select_engine::exec");
|
2002-11-24 20:10:52 +01:00
|
|
|
char const *save_where= join->thd->where;
|
2003-12-19 18:52:13 +01:00
|
|
|
SELECT_LEX *save_select= join->thd->lex->current_select;
|
|
|
|
join->thd->lex->current_select= select_lex;
|
2002-09-03 08:50:36 +02:00
|
|
|
if (!optimized)
|
|
|
|
{
|
|
|
|
optimized=1;
|
|
|
|
if (join->optimize())
|
|
|
|
{
|
2002-11-24 20:10:52 +01:00
|
|
|
join->thd->where= save_where;
|
2002-09-03 08:50:36 +02:00
|
|
|
executed= 1;
|
2003-12-19 18:52:13 +01:00
|
|
|
join->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
|
|
|
}
|
2003-07-07 17:40:19 +02:00
|
|
|
if (item->engine_changed)
|
|
|
|
{
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
2003-11-17 19:53:40 +01:00
|
|
|
if (select_lex->uncacheable && executed)
|
2002-09-03 08:50:36 +02:00
|
|
|
{
|
|
|
|
if (join->reinit())
|
2002-11-24 20:10:52 +01:00
|
|
|
{
|
|
|
|
join->thd->where= save_where;
|
2003-12-19 18:52:13 +01:00
|
|
|
join->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();
|
2002-09-03 08:50:36 +02:00
|
|
|
join->exec();
|
|
|
|
executed= 1;
|
2002-11-24 20:10:52 +01:00
|
|
|
join->thd->where= save_where;
|
2003-12-19 18:52:13 +01:00
|
|
|
join->thd->lex->current_select= save_select;
|
2003-01-30 21:15:44 +01:00
|
|
|
DBUG_RETURN(join->error||thd->is_fatal_error);
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
2002-11-24 20:10:52 +01:00
|
|
|
join->thd->where= save_where;
|
2003-12-19 18:52:13 +01:00
|
|
|
join->thd->lex->current_select= save_select;
|
2002-09-03 08:50:36 +02:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int subselect_union_engine::exec()
|
|
|
|
{
|
2002-11-24 20:10:52 +01:00
|
|
|
char const *save_where= unit->thd->where;
|
|
|
|
int res= unit->exec();
|
|
|
|
unit->thd->where= save_where;
|
|
|
|
return res;
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
|
|
|
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2003-09-14 08:40:57 +02:00
|
|
|
int subselect_uniquesubquery_engine::exec()
|
2003-07-07 17:40:19 +02:00
|
|
|
{
|
2003-09-14 08:40:57 +02:00
|
|
|
DBUG_ENTER("subselect_uniquesubquery_engine::exec");
|
2003-07-07 17:40:19 +02:00
|
|
|
int error;
|
|
|
|
TABLE *table= tab->table;
|
2004-08-12 16:31:23 +02:00
|
|
|
for (store_key **copy=tab->ref.key_copy ; *copy ; copy++)
|
2003-07-07 17:40:19 +02:00
|
|
|
{
|
func_str.result, func_str.test:
Added a test case for bug #10124.
sql_select.h, item_subselect.cc, sql_select.cc:
Fixed bug #10124.
The copy method of the store_key classes can return
STORE_KEY_OK=0, STORE_KEY_FATAL=1, STORE_KEY_CONV=2 now.
field.cc:
Fixed bug #10124.
When ussuing a warning the store methods return 2 instead of 1 now.
sql/field.cc:
Fixed bug #10124.
When ussuing a warning the store methods return 2 instead of 1 now.
sql/sql_select.cc:
Fixed bug #10124.
The copy method of the store_key classes can return
STORE_KEY_OK=0, STORE_KEY_FATAL=1, STORE_KEY_CONV=2 now.
sql/item_subselect.cc:
Fixed bug #10124.
The copy method of the store_key classes can return
STORE_KEY_OK=0, STORE_KEY_FATAL=1, STORE_KEY_CONV=2 now.
sql/sql_select.h:
Fixed bug #10124.
The copy method of the store_key classes can return
STORE_KEY_OK=0, STORE_KEY_FATAL=1, STORE_KEY_CONV=2 now.
mysql-test/t/func_str.test:
Added a test case for bug #10124.
mysql-test/r/func_str.result:
Added a test case for bug #10124.
2005-06-23 15:15:50 +02:00
|
|
|
if ((tab->ref.key_err= (*copy)->copy()) & 1)
|
2004-08-12 16:31:23 +02:00
|
|
|
{
|
|
|
|
table->status= STATUS_NOT_FOUND;
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
2003-07-07 17:40:19 +02:00
|
|
|
}
|
2004-08-12 16:31:23 +02:00
|
|
|
|
|
|
|
if (!table->file->inited)
|
|
|
|
table->file->ha_index_init(tab->ref.key);
|
|
|
|
error= table->file->index_read(table->record[0],
|
|
|
|
tab->ref.key_buff,
|
|
|
|
tab->ref.key_length,HA_READ_KEY_EXACT);
|
2004-10-07 13:13:42 +02:00
|
|
|
if (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;
|
|
|
|
((Item_in_subselect *) item)->value= (!table->status &&
|
|
|
|
(!cond || cond->val_int()) ? 1 :
|
|
|
|
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
|
|
|
|
|
|
|
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 */
|
2004-06-23 12:29:05 +02:00
|
|
|
tab->table->file->ha_index_end();
|
2003-07-07 23:08:00 +02:00
|
|
|
}
|
|
|
|
|
2003-09-29 11:39:38 +02:00
|
|
|
|
2003-09-14 08:40:57 +02:00
|
|
|
int subselect_indexsubquery_engine::exec()
|
2003-07-07 23:08:00 +02:00
|
|
|
{
|
2003-09-14 08:40:57 +02:00
|
|
|
DBUG_ENTER("subselect_indexsubselect_engine::exec");
|
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;
|
2003-07-17 18:39:31 +02:00
|
|
|
|
2003-07-07 23:08:00 +02:00
|
|
|
((Item_in_subselect *) item)->value= 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;
|
|
|
|
}
|
|
|
|
|
2004-08-12 16:31:23 +02:00
|
|
|
for (store_key **copy=tab->ref.key_copy ; *copy ; copy++)
|
2003-07-07 23:08:00 +02:00
|
|
|
{
|
func_str.result, func_str.test:
Added a test case for bug #10124.
sql_select.h, item_subselect.cc, sql_select.cc:
Fixed bug #10124.
The copy method of the store_key classes can return
STORE_KEY_OK=0, STORE_KEY_FATAL=1, STORE_KEY_CONV=2 now.
field.cc:
Fixed bug #10124.
When ussuing a warning the store methods return 2 instead of 1 now.
sql/field.cc:
Fixed bug #10124.
When ussuing a warning the store methods return 2 instead of 1 now.
sql/sql_select.cc:
Fixed bug #10124.
The copy method of the store_key classes can return
STORE_KEY_OK=0, STORE_KEY_FATAL=1, STORE_KEY_CONV=2 now.
sql/item_subselect.cc:
Fixed bug #10124.
The copy method of the store_key classes can return
STORE_KEY_OK=0, STORE_KEY_FATAL=1, STORE_KEY_CONV=2 now.
sql/sql_select.h:
Fixed bug #10124.
The copy method of the store_key classes can return
STORE_KEY_OK=0, STORE_KEY_FATAL=1, STORE_KEY_CONV=2 now.
mysql-test/t/func_str.test:
Added a test case for bug #10124.
mysql-test/r/func_str.result:
Added a test case for bug #10124.
2005-06-23 15:15:50 +02:00
|
|
|
if ((tab->ref.key_err= (*copy)->copy()) & 1)
|
2004-08-12 16:31:23 +02:00
|
|
|
{
|
|
|
|
table->status= STATUS_NOT_FOUND;
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
2003-07-07 23:08:00 +02:00
|
|
|
}
|
2004-08-12 16:31:23 +02:00
|
|
|
|
|
|
|
if (!table->file->inited)
|
|
|
|
table->file->ha_index_init(tab->ref.key);
|
|
|
|
error= table->file->index_read(table->record[0],
|
|
|
|
tab->ref.key_buff,
|
|
|
|
tab->ref.key_length,HA_READ_KEY_EXACT);
|
2004-10-07 13:13:42 +02:00
|
|
|
if (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
|
|
|
{
|
2004-08-12 16:31:23 +02:00
|
|
|
if (!cond || cond->val_int())
|
|
|
|
{
|
|
|
|
if (null_finding)
|
|
|
|
((Item_in_subselect *) item)->was_null= 1;
|
|
|
|
else
|
|
|
|
((Item_in_subselect *) item)->value= 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
error= table->file->index_next_same(table->record[0],
|
|
|
|
tab->ref.key_buff,
|
|
|
|
tab->ref.key_length);
|
|
|
|
if (error && error != HA_ERR_END_OF_FILE)
|
|
|
|
{
|
|
|
|
error= report_error(table, error);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!check_null || null_finding)
|
|
|
|
break; /* We don't need to check nulls */
|
|
|
|
*tab->ref.null_ref_key= 1;
|
|
|
|
null_finding= 1;
|
|
|
|
/* Check if there exists a row with a null value in the index */
|
|
|
|
if ((error= (safe_index_read(tab) == 1)))
|
|
|
|
break;
|
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
|
|
|
|
2002-09-03 08:50:36 +02:00
|
|
|
uint subselect_single_select_engine::cols()
|
|
|
|
{
|
2005-01-24 13:25:44 +01:00
|
|
|
DBUG_ASSERT(select_lex->join); // should be called after fix_fields()
|
|
|
|
return select_lex->join->fields_list.elements;
|
2002-09-03 08:50:36 +02:00
|
|
|
}
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
|
2002-09-03 08:50:36 +02:00
|
|
|
uint subselect_union_engine::cols()
|
|
|
|
{
|
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
|
|
|
|
|
|
|
|
|
|
|
table_map subselect_engine::calc_const_tables(TABLE_LIST *table)
|
|
|
|
{
|
|
|
|
table_map map= 0;
|
2004-04-15 09:17:55 +02:00
|
|
|
for (; table; table= table->next)
|
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()
|
|
|
|
{
|
|
|
|
return calc_const_tables((TABLE_LIST *) select_lex->outer_select()->
|
|
|
|
table_list.first);
|
|
|
|
}
|
|
|
|
|
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()
|
|
|
|
{
|
|
|
|
return calc_const_tables((TABLE_LIST *) unit->outer_select()->
|
|
|
|
table_list.first);
|
|
|
|
}
|
2003-10-28 11:45:37 +01:00
|
|
|
|
|
|
|
|
2003-10-16 14:54:47 +02:00
|
|
|
void subselect_single_select_engine::print(String *str)
|
|
|
|
{
|
|
|
|
select_lex->print(thd, str);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void subselect_union_engine::print(String *str)
|
|
|
|
{
|
|
|
|
unit->print(str);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void subselect_uniquesubquery_engine::print(String *str)
|
|
|
|
{
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append("<primary_index_lookup>(", 23);
|
2003-10-16 14:54:47 +02:00
|
|
|
tab->ref.items[0]->print(str);
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append(" in ", 4);
|
2003-10-16 14:54:47 +02:00
|
|
|
str->append(tab->table->real_name);
|
2003-10-19 13:22:17 +02:00
|
|
|
KEY *key_info= tab->table->key_info+ tab->ref.key;
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append(" on ", 4);
|
2003-10-19 13:22:17 +02:00
|
|
|
str->append(key_info->name);
|
2003-10-16 14:54:47 +02:00
|
|
|
if (cond)
|
|
|
|
{
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append(" where ", 7);
|
2003-10-16 14:54:47 +02:00
|
|
|
cond->print(str);
|
|
|
|
}
|
|
|
|
str->append(')');
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void subselect_indexsubquery_engine::print(String *str)
|
|
|
|
{
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append("<index_lookup>(", 15);
|
2003-10-16 14:54:47 +02:00
|
|
|
tab->ref.items[0]->print(str);
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append(" in ", 4);
|
2003-10-16 14:54:47 +02:00
|
|
|
str->append(tab->table->real_name);
|
2003-10-19 13:22:17 +02:00
|
|
|
KEY *key_info= tab->table->key_info+ tab->ref.key;
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append(" on ", 4);
|
2003-10-19 13:22:17 +02:00
|
|
|
str->append(key_info->name);
|
2003-10-16 14:54:47 +02:00
|
|
|
if (check_null)
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append(" chicking NULL", 14);
|
2003-10-16 14:54:47 +02:00
|
|
|
if (cond)
|
|
|
|
{
|
2003-10-30 11:57:26 +01:00
|
|
|
str->append(" where ", 7);
|
2003-10-16 14:54:47 +02:00
|
|
|
cond->print(str);
|
|
|
|
}
|
|
|
|
str->append(')');
|
|
|
|
}
|
2004-05-07 22:06:11 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
change select_result object of engine
|
|
|
|
|
|
|
|
SINOPSYS
|
|
|
|
subselect_single_select_engine::change_result()
|
|
|
|
si new subselect Item
|
|
|
|
res new select_result object
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
0 OK
|
|
|
|
-1 error
|
|
|
|
*/
|
|
|
|
|
|
|
|
int subselect_single_select_engine::change_item(Item_subselect *si,
|
|
|
|
select_subselect *res)
|
|
|
|
{
|
|
|
|
item= si;
|
|
|
|
result= res;
|
|
|
|
return select_lex->join->change_result(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
change select_result object of engine
|
|
|
|
|
|
|
|
SINOPSYS
|
|
|
|
subselect_single_select_engine::change_result()
|
|
|
|
si new subselect Item
|
|
|
|
res new select_result object
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
0 OK
|
|
|
|
-1 error
|
|
|
|
*/
|
|
|
|
|
|
|
|
int subselect_union_engine::change_item(Item_subselect *si,
|
|
|
|
select_subselect *res)
|
|
|
|
{
|
|
|
|
item= si;
|
|
|
|
int rc= unit->change_result(res, result);
|
|
|
|
result= res;
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
change select_result emulation, never should be called
|
|
|
|
|
|
|
|
SINOPSYS
|
|
|
|
subselect_single_select_engine::change_result()
|
|
|
|
si new subselect Item
|
|
|
|
res new select_result object
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
-1 error
|
|
|
|
*/
|
|
|
|
|
|
|
|
int subselect_uniquesubquery_engine::change_item(Item_subselect *si,
|
|
|
|
select_subselect *res)
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(0);
|
|
|
|
return -1;
|
|
|
|
}
|
2004-10-27 20:11:06 +02:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Report about presence of tables in subquery
|
|
|
|
|
|
|
|
SINOPSYS
|
|
|
|
subselect_single_select_engine::no_tables()
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
TRUE there are not tables used in subquery
|
|
|
|
FALSE there are some tables in subquery
|
|
|
|
*/
|
|
|
|
bool subselect_single_select_engine::no_tables()
|
|
|
|
{
|
|
|
|
return(select_lex->table_list.elements == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Report about presence of tables in subquery
|
|
|
|
|
|
|
|
SINOPSYS
|
|
|
|
subselect_union_engine::no_tables()
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
TRUE there are not tables used in subquery
|
|
|
|
FALSE there are some tables in subquery
|
|
|
|
*/
|
|
|
|
bool subselect_union_engine::no_tables()
|
|
|
|
{
|
|
|
|
for (SELECT_LEX *sl= unit->first_select(); sl; sl= sl->next_select())
|
|
|
|
{
|
|
|
|
if (sl->table_list.elements)
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Report about presence of tables in subquery
|
|
|
|
|
|
|
|
SINOPSYS
|
|
|
|
subselect_uniquesubquery_engine::no_tables()
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
TRUE there are not tables used in subquery
|
|
|
|
FALSE there are some tables in subquery
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool subselect_uniquesubquery_engine::no_tables()
|
|
|
|
{
|
|
|
|
/* returning value is correct, but this method should never be called */
|
|
|
|
return 0;
|
|
|
|
}
|