item_subselect.cc 8.69 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
/* 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 */

/* 
   subselect Item

SUBSELECT TODO:
   - add function from mysql_select that use JOIN* as parameter to JOIN methods
     (sql_select.h/sql_select.cc)
   - remove double 'having' & 'having_list' from JOIN
     (sql_select.h/sql_select.cc)

*/

#ifdef __GNUC__
#pragma implementation				// gcc: Class implementation
#endif

#include "mysql_priv.h"
#include "sql_select.h"

35
Item_subselect::Item_subselect():
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
36
  Item_result_field(), engine_owner(1), value_assigned(0)
37
{
38 39 40 41 42 43 44 45 46 47 48 49 50
  assign_null();
  /*
    item value is NULL if select_subselect not changed this value 
    (i.e. some rows will be found returned)
  */
  null_value= 1;
}

void Item_subselect::init(THD *thd, st_select_lex *select_lex,
		     select_subselect *result)
{

  DBUG_ENTER("Item_subselect::init");
51
  DBUG_PRINT("subs", ("select_lex 0x%xl", (long) select_lex));
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
52 53 54 55 56 57 58

  if (select_lex->next_select())
    engine= new subselect_union_engine(thd, select_lex->master_unit(), result,
				       this);
  else
    engine= new subselect_single_select_engine(thd, select_lex, result,
					       this);
59 60 61
  DBUG_VOID_RETURN;
}

bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
62 63 64 65 66 67
Item_subselect::~Item_subselect()
{
  if (engine_owner)
    delete engine;
}

68 69 70 71 72 73 74 75 76 77 78 79 80 81
void Item_subselect::make_field (Send_field *tmp_field)
{
  if (null_value)
  {
    init_make_field(tmp_field,FIELD_TYPE_NULL);
    tmp_field->length=4;
  } else {
    init_make_field(tmp_field, ((result_type() == STRING_RESULT) ?
				FIELD_TYPE_VAR_STRING :
				(result_type() == INT_RESULT) ?
				FIELD_TYPE_LONGLONG : FIELD_TYPE_DOUBLE));
  }
}

82
bool Item_subselect::fix_fields(THD *thd, TABLE_LIST *tables, Item **ref)
83
{
84
  int res= engine->prepare();
85
  if (!res)
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
86 87 88 89 90 91 92
  {
    // Is it one field subselect?
    if (engine->cols() > max_columns)
    {  
      my_message(ER_SUBSELECT_NO_1_COL, ER(ER_SUBSELECT_NO_1_COL), MYF(0));
      return 1;
    }
93
    fix_length_and_dec();
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
94
  }
95
  fixed= 1;
96 97 98
  return res;
}

99 100 101 102 103 104 105 106 107
bool Item_subselect::check_loop(uint id)
{
  DBUG_ENTER("Item_subselect::check_loop");
  if (Item_result_field::check_loop(id))
    DBUG_RETURN(1);
  
  DBUG_RETURN(engine->check_loop(id));
}

108 109 110
void Item_subselect::fix_length_and_dec()
{
  engine->fix_length_and_dec();
111
}
112 113 114

inline table_map Item_subselect::used_tables() const
{
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
115
  return (table_map) engine->depended() ? 1L : 0L; 
116 117 118 119
}

Item_singleval_subselect::Item_singleval_subselect(THD *thd,
						   st_select_lex *select_lex):
120
  Item_subselect()
121
{
122
  init(thd, select_lex, new select_singleval_subselect(this));
123 124 125 126
  max_columns= 1;
  maybe_null= 1;
}

127 128 129 130 131 132
void Item_singleval_subselect::fix_length_and_dec()
{
  engine->fix_length_and_dec();
  res_type= engine->type();
}

133 134 135 136 137 138 139
Item::Type Item_subselect::type() const 
{
  return SUBSELECT_ITEM;
}

double Item_singleval_subselect::val () 
{
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
140
  if (engine->exec())
141 142
  {
    assign_null();
143
    return 0;
144
  }
145 146 147 148 149
  return real_value;
}

longlong Item_singleval_subselect::val_int () 
{
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
150
  if (engine->exec())
151 152
  {
    assign_null();
153
    return 0;
154
  }
155 156 157 158 159
  return int_value;
}

String *Item_singleval_subselect::val_str (String *str) 
{
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
160
  if (engine->exec() || null_value)
161 162
  {
    assign_null();
163
    return 0;
164
  }
165
  return &string_value;
166 167 168 169
}

Item_exists_subselect::Item_exists_subselect(THD *thd,
					     st_select_lex *select_lex):
170
  Item_subselect()
171
{
172
  init(thd, select_lex, new select_exists_subselect(this));
173 174 175 176 177 178 179
  max_columns= UINT_MAX;
  null_value= 0; //can't be NULL
  maybe_null= 0; //can't be NULL
  value= 0;
  select_lex->select_limit= 1; // we need only 1 row to determinate existence
}

180 181 182 183 184 185
void Item_exists_subselect::fix_length_and_dec()
{
  max_length= 1;
  
}

186 187
double Item_exists_subselect::val () 
{
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
188
  if (engine->exec())
189 190
  {
    assign_null();
191
    return 0;
192
  }
193 194 195 196 197
  return (double) value;
}

longlong Item_exists_subselect::val_int () 
{
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
198
  if (engine->exec())
199 200
  {
    assign_null();
201
    return 0;
202
  }
203 204 205 206 207
  return value;
}

String *Item_exists_subselect::val_str(String *str)
{
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
208
  if (engine->exec())
209 210
  {
    assign_null();
211
    return 0;
212
  }
213
  str->set(value,thd_charset());
214 215 216
  return str;
}

bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
217 218 219 220 221 222

subselect_single_select_engine::subselect_single_select_engine(THD *thd, 
							       st_select_lex *select,
							       select_subselect *result,
							       Item_subselect *item):
  subselect_engine(thd, item, result),
223
    prepared(0), optimized(0), executed(0)
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240
{
  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)
    unit->select_limit_cnt= HA_POS_ERROR;		// no limit
  if (unit->select_limit_cnt == HA_POS_ERROR)
    select_lex->options&= ~OPTION_FOUND_ROWS;
  join= new JOIN(thd, select_lex->item_list, select_lex->options, result);
  if (!join || !result)
  {
    //out of memory
    thd->fatal_error= 1;
    my_message(ER_OUT_OF_RESOURCES, ER(ER_OUT_OF_RESOURCES), MYF(0));
  } 
241
  unit->item= item;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262
  this->select_lex= select_lex;
}

subselect_union_engine::subselect_union_engine(THD *thd,
					       st_select_lex_unit *u,
					       select_subselect *result,
					       Item_subselect *item):
  subselect_engine(thd, item, result)
{
  unit= u;
  if( !result)
  {
    //out of memory
    thd->fatal_error= 1;
    my_message(ER_OUT_OF_RESOURCES, ER(ER_OUT_OF_RESOURCES), MYF(0));
  }
  unit->item= item;
}

int subselect_single_select_engine::prepare()
{
263 264 265
  if (prepared)
    return 0;
  prepared= 1;
266 267
  SELECT_LEX_NODE *save_select= thd->lex.current_select;
  thd->lex.current_select= select_lex;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
268 269 270 271 272 273 274 275
  if(join->prepare((TABLE_LIST*) select_lex->table_list.first,
		   select_lex->where,
		   (ORDER*) select_lex->order_list.first,
		   (ORDER*) select_lex->group_list.first,
		   select_lex->having,
		   (ORDER*) 0, select_lex, 
		   select_lex->master_unit(), 0))
    return 1;
276
  thd->lex.current_select= save_select;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
277 278 279 280 281 282 283 284
  return 0;
}

int subselect_union_engine::prepare()
{
  return unit->prepare(thd, result);
}

285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310
void subselect_single_select_engine::fix_length_and_dec()
{
  List_iterator_fast<Item> li(select_lex->item_list);
  Item *sel_item= li++;
  item->max_length= sel_item->max_length;
  res_type= sel_item->result_type();
  item->decimals= sel_item->decimals;
}

void subselect_union_engine::fix_length_and_dec()
{
  uint32 mlen= 0, len;
  Item *sel_item= 0;
  for(SELECT_LEX *sl= unit->first_select(); sl; sl= sl->next_select())
  {
    List_iterator_fast<Item> li(sl->item_list);
    Item *s_item= li++;
    if ((len= s_item->max_length))
      mlen= len;
    if (!sel_item)
      sel_item= s_item;
  }
  item->max_length= mlen;
  res_type= sel_item->result_type();
  item->decimals= sel_item->decimals;
}
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
311 312 313 314 315 316 317 318 319 320 321 322 323

int subselect_single_select_engine::exec()
{
  DBUG_ENTER("subselect_single_select_engine::exec");
  if (!optimized)
  {
    optimized=1;
    if (join->optimize())
    {
      executed= 1;
      DBUG_RETURN(join->error?join->error:1);
    }
  }
324
  if (select_lex->dependent && executed)
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
325 326 327 328 329 330 331 332
  {
    if (join->reinit())
      DBUG_RETURN(1);
    item->assign_null();
    item->assigned((executed= 0));
  }
  if (!executed)
  {
333 334
    SELECT_LEX_NODE *save_select= join->thd->lex.current_select;
    join->thd->lex.current_select= select_lex;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
335
    join->exec();
336
    join->thd->lex.current_select= save_select;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359
    executed= 1;
    DBUG_RETURN(join->error||thd->fatal_error);
  }
  DBUG_RETURN(0);
}

int subselect_union_engine::exec()
{
  return unit->exec();
}

uint subselect_single_select_engine::cols()
{
  return select_lex->item_list.elements;
}

uint subselect_union_engine::cols()
{
  return unit->first_select()->item_list.elements;
}

bool subselect_single_select_engine::depended()
{
360
  return select_lex->dependent;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
361 362 363 364
}

bool subselect_union_engine::depended()
{
365
  return unit->dependent;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
366
}
367 368 369 370 371 372 373 374 375 376 377 378 379 380 381

bool subselect_single_select_engine::check_loop(uint id)
{
  DBUG_ENTER("subselect_single_select_engine::check_loop");
  DBUG_RETURN(join->check_loop(id));
}

bool subselect_union_engine::check_loop(uint id)
{
  DBUG_ENTER("subselect_union_engine::check_loop");
  for (SELECT_LEX *sl= unit->first_select(); sl; sl= sl->next_select())
    if (sl->join && sl->join->check_loop(id))
      DBUG_RETURN(1);
  DBUG_RETURN(0);
}